2017-07-12 19:29:16 -07:00
|
|
|
#![allow(dead_code)]
|
|
|
|
|
|
|
|
// We use packed structs to get around alignment restrictions
|
|
|
|
#[repr(packed)]
|
|
|
|
struct Data {
|
|
|
|
pad: u8,
|
|
|
|
ptr: &'static i32,
|
|
|
|
}
|
|
|
|
|
|
|
|
// But we need to gurantee some alignment
|
|
|
|
struct Wrapper {
|
|
|
|
align: u64,
|
|
|
|
data: Data,
|
|
|
|
}
|
|
|
|
|
2022-06-20 23:40:39 -07:00
|
|
|
static G: i32 = 0;
|
2017-07-12 19:29:16 -07:00
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let mut w = Wrapper { align: 0, data: Data { pad: 0, ptr: &G } };
|
|
|
|
|
|
|
|
// Get a pointer to the beginning of the Data struct (one u8 byte, then the pointer bytes).
|
|
|
|
// Thanks to the wrapper, we know this is aligned-enough to perform a load at ptr size.
|
2019-02-16 01:43:56 +00:00
|
|
|
// We load at pointer type, so having a relocation is ok -- but here, the relocation
|
2017-07-12 19:29:16 -07:00
|
|
|
// starts 1 byte to the right, so using it would actually be wrong!
|
|
|
|
let d_alias = &mut w.data as *mut _ as *mut *const u8;
|
|
|
|
unsafe {
|
2022-08-27 15:59:27 -04:00
|
|
|
let x = *d_alias;
|
|
|
|
let _val = *x; //~ERROR: is a dangling pointer (it has no provenance)
|
2017-07-12 19:29:16 -07:00
|
|
|
}
|
|
|
|
}
|