2022-07-08 16:08:32 +00:00
|
|
|
//@compile-flags: -Zmiri-permissive-provenance
|
2022-07-24 15:23:39 -04:00
|
|
|
#![feature(ptr_sub_ptr)]
|
2020-04-16 09:06:21 +02:00
|
|
|
use std::{mem, ptr};
|
2018-12-26 16:23:04 +01:00
|
|
|
|
2017-06-04 18:18:37 -07:00
|
|
|
fn main() {
|
2022-07-24 15:23:39 -04:00
|
|
|
smoke();
|
2020-04-16 09:06:21 +02:00
|
|
|
test_offset_from();
|
|
|
|
test_vec_into_iter();
|
|
|
|
ptr_arith_offset();
|
|
|
|
ptr_arith_offset_overflow();
|
|
|
|
ptr_offset();
|
|
|
|
}
|
|
|
|
|
2022-07-24 15:23:39 -04:00
|
|
|
fn smoke() {
|
|
|
|
// Smoke-test various offsetting operations.
|
|
|
|
let ptr = &5;
|
|
|
|
let ptr = ptr as *const i32;
|
|
|
|
let _val = ptr.wrapping_offset(0);
|
|
|
|
let _val = unsafe { ptr.offset(0) };
|
|
|
|
let _val = ptr.wrapping_add(0);
|
|
|
|
let _val = unsafe { ptr.add(0) };
|
|
|
|
let _val = ptr.wrapping_sub(0);
|
|
|
|
let _val = unsafe { ptr.sub(0) };
|
|
|
|
let _val = unsafe { ptr.offset_from(ptr) };
|
|
|
|
let _val = unsafe { ptr.sub_ptr(ptr) };
|
|
|
|
}
|
|
|
|
|
2022-06-20 16:00:37 -07:00
|
|
|
fn test_offset_from() {
|
|
|
|
unsafe {
|
|
|
|
let buf = [0u32; 4];
|
2020-04-16 09:06:21 +02:00
|
|
|
|
2022-06-20 16:00:37 -07:00
|
|
|
let x = buf.as_ptr() as *const u8;
|
|
|
|
let y = x.offset(12);
|
2020-04-16 09:06:21 +02:00
|
|
|
|
2022-06-20 16:00:37 -07:00
|
|
|
assert_eq!(y.offset_from(x), 12);
|
2022-07-24 15:23:39 -04:00
|
|
|
assert_eq!(y.sub_ptr(x), 12);
|
2022-06-20 16:00:37 -07:00
|
|
|
assert_eq!(x.offset_from(y), -12);
|
|
|
|
assert_eq!((y as *const u32).offset_from(x as *const u32), 12 / 4);
|
|
|
|
assert_eq!((x as *const u32).offset_from(y as *const u32), -12 / 4);
|
2022-04-30 10:40:35 -07:00
|
|
|
|
2022-06-20 16:00:37 -07:00
|
|
|
let x = (((x as usize) * 2) / 2) as *const u8;
|
|
|
|
assert_eq!(y.offset_from(x), 12);
|
2022-07-24 15:23:39 -04:00
|
|
|
assert_eq!(y.sub_ptr(x), 12);
|
2022-06-20 16:00:37 -07:00
|
|
|
assert_eq!(x.offset_from(y), -12);
|
|
|
|
}
|
|
|
|
}
|
2020-04-16 09:06:21 +02:00
|
|
|
|
|
|
|
// This also internally uses offset_from.
|
|
|
|
fn test_vec_into_iter() {
|
|
|
|
let v = Vec::<i32>::new();
|
|
|
|
let i = v.into_iter();
|
|
|
|
i.size_hint();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn ptr_arith_offset() {
|
|
|
|
let v = [1i16, 2];
|
|
|
|
let x = &v as *const [i16] as *const i16;
|
|
|
|
let x = x.wrapping_offset(1);
|
|
|
|
assert_eq!(unsafe { *x }, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn ptr_arith_offset_overflow() {
|
|
|
|
let v = [1i16, 2];
|
|
|
|
let x = &mut ptr::null(); // going through memory as there are more sanity checks along that path
|
|
|
|
*x = v.as_ptr().wrapping_offset(1); // ptr to the 2nd element
|
|
|
|
// Adding 2*isize::max and then 1 is like substracting 1
|
|
|
|
*x = x.wrapping_offset(isize::MAX);
|
|
|
|
*x = x.wrapping_offset(isize::MAX);
|
|
|
|
*x = x.wrapping_offset(1);
|
|
|
|
assert_eq!(unsafe { **x }, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn ptr_offset() {
|
2022-06-20 16:00:37 -07:00
|
|
|
fn f() -> i32 {
|
|
|
|
42
|
|
|
|
}
|
2020-04-16 09:06:21 +02:00
|
|
|
|
2017-06-04 18:18:37 -07:00
|
|
|
let v = [1i16, 2];
|
2019-04-15 17:06:42 +02:00
|
|
|
let x = &v as *const [i16; 2] as *const i16;
|
2017-06-04 18:18:37 -07:00
|
|
|
let x = unsafe { x.offset(1) };
|
|
|
|
assert_eq!(unsafe { *x }, 2);
|
2018-12-26 16:23:04 +01:00
|
|
|
|
|
|
|
// fn ptr offset
|
|
|
|
unsafe {
|
|
|
|
let p = f as fn() -> i32 as usize;
|
|
|
|
let x = (p as *mut u32).offset(0) as usize;
|
2022-05-23 15:44:27 +02:00
|
|
|
// *cast* to ptr, then transmute to fn ptr.
|
|
|
|
// (transmuting int to [fn]ptr causes trouble.)
|
|
|
|
let f: fn() -> i32 = mem::transmute(x as *const ());
|
2018-12-26 16:23:04 +01:00
|
|
|
assert_eq!(f(), 42);
|
|
|
|
}
|
2017-06-04 18:18:37 -07:00
|
|
|
}
|