2017-07-12 21:06:57 -07:00
|
|
|
#![allow(dead_code)]
|
2017-07-13 14:18:26 -07:00
|
|
|
#![feature(unsize, coerce_unsized)]
|
|
|
|
|
2017-01-30 09:44:52 +01:00
|
|
|
#[repr(packed)]
|
|
|
|
struct S {
|
|
|
|
a: i32,
|
|
|
|
b: i64,
|
|
|
|
}
|
|
|
|
|
2017-07-12 21:06:57 -07:00
|
|
|
#[repr(packed)]
|
|
|
|
struct Test1<'a> {
|
|
|
|
x: u8,
|
|
|
|
other: &'a u32,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(packed)]
|
|
|
|
struct Test2<'a> {
|
|
|
|
x: u8,
|
|
|
|
other: &'a Test1<'a>,
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test(t: Test2) {
|
|
|
|
let x = *t.other.other;
|
|
|
|
assert_eq!(x, 42);
|
|
|
|
}
|
|
|
|
|
2017-07-13 14:18:26 -07:00
|
|
|
fn test_unsizing() {
|
|
|
|
#[repr(packed)]
|
|
|
|
struct UnalignedPtr<'a, T: ?Sized>
|
|
|
|
where T: 'a,
|
|
|
|
{
|
|
|
|
data: &'a T,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, T, U> std::ops::CoerceUnsized<UnalignedPtr<'a, U>> for UnalignedPtr<'a, T>
|
|
|
|
where
|
|
|
|
T: std::marker::Unsize<U> + ?Sized,
|
|
|
|
U: ?Sized,
|
|
|
|
{ }
|
|
|
|
|
|
|
|
let arr = [1, 2, 3];
|
|
|
|
let arr_unaligned: UnalignedPtr<[i32; 3]> = UnalignedPtr { data: &arr };
|
2017-07-26 23:43:13 -07:00
|
|
|
let arr_unaligned: UnalignedPtr<[i32]> = arr_unaligned;
|
|
|
|
let _unused = &arr_unaligned; // forcing an allocation, which could also yield "unaligned write"-errors
|
2017-07-13 14:18:26 -07:00
|
|
|
}
|
|
|
|
|
2018-08-14 20:09:07 +02:00
|
|
|
fn test_drop() {
|
|
|
|
struct Wrap(u32);
|
|
|
|
impl Drop for Wrap {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
// Do an (aligned) load
|
|
|
|
let _test = self.0;
|
|
|
|
// For the fun of it, test alignment
|
|
|
|
assert_eq!(&self.0 as *const _ as usize % std::mem::align_of::<u32>(), 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(packed,C)]
|
|
|
|
struct Packed<T> {
|
|
|
|
f1: u8, // this should move the second field to something not very aligned
|
|
|
|
f2: T,
|
|
|
|
}
|
|
|
|
|
|
|
|
let p = Packed { f1: 42, f2: Wrap(23) };
|
|
|
|
drop(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test_inner_packed() {
|
|
|
|
// Even if just the inner struct is packed, accesses to the outer field can get unaligned.
|
|
|
|
// Make sure that works.
|
|
|
|
#[repr(packed)]
|
|
|
|
#[derive(Clone,Copy)]
|
|
|
|
struct Inner(u32);
|
|
|
|
|
|
|
|
#[derive(Clone,Copy)]
|
|
|
|
struct Outer(u8, Inner);
|
|
|
|
|
|
|
|
let o = Outer(0, Inner(42));
|
|
|
|
let _x = o.1;
|
|
|
|
let _y = (o.1).0;
|
|
|
|
let _o2 = o.clone();
|
|
|
|
}
|
|
|
|
|
2017-01-30 09:44:52 +01:00
|
|
|
fn main() {
|
2017-07-12 17:46:56 -07:00
|
|
|
let mut x = S {
|
2017-01-30 09:44:52 +01:00
|
|
|
a: 42,
|
|
|
|
b: 99,
|
|
|
|
};
|
2017-01-31 10:51:52 +01:00
|
|
|
let a = x.a;
|
|
|
|
let b = x.b;
|
|
|
|
assert_eq!(a, 42);
|
|
|
|
assert_eq!(b, 99);
|
|
|
|
// can't do `assert_eq!(x.a, 42)`, because `assert_eq!` takes a reference
|
|
|
|
assert_eq!({x.a}, 42);
|
|
|
|
assert_eq!({x.b}, 99);
|
2017-07-12 17:46:56 -07:00
|
|
|
|
|
|
|
x.b = 77;
|
|
|
|
assert_eq!({x.b}, 77);
|
2017-07-12 21:06:57 -07:00
|
|
|
|
|
|
|
test(Test2 { x: 0, other: &Test1 { x: 0, other: &42 }});
|
2017-07-13 14:18:26 -07:00
|
|
|
|
|
|
|
test_unsizing();
|
2018-08-14 20:09:07 +02:00
|
|
|
test_drop();
|
|
|
|
test_inner_packed();
|
2017-01-30 09:44:52 +01:00
|
|
|
}
|