diff options
| author | bors <bors@rust-lang.org> | 2015-12-09 04:22:05 +0000 |
|---|---|---|
| committer | bors <bors@rust-lang.org> | 2015-12-09 04:22:05 +0000 |
| commit | 4005d437653fc09b47cd3176bb65323c52cccd47 (patch) | |
| tree | 2708e9f2a98cacd29219d955d69e832eba87bc7a /src/test | |
| parent | 56a1f51ef0b26acb0d6eb2a7dbe9336f5e942061 (diff) | |
| parent | d6eb063fe8c60ff7cbd83c7a27ba4bc6a24bd174 (diff) | |
| download | rust-4005d437653fc09b47cd3176bb65323c52cccd47.tar.gz rust-4005d437653fc09b47cd3176bb65323c52cccd47.zip | |
Auto merge of #30245 - Aatch:dynamic-align-dst, r=pnkfelix
Fixes #26403 This adjusts the pointer, if needed, to the correct alignment by using the alignment information in the vtable. Handling zero might not be necessary, as it shouldn't actually occur. I've left it as it's own commit so it can be removed fairly easily if people don't think it's worth doing. The way it's handled though means that there shouldn't be much impact on performance.
Diffstat (limited to 'src/test')
| -rw-r--r-- | src/test/run-pass/dst-field-align.rs | 87 |
1 files changed, 87 insertions, 0 deletions
diff --git a/src/test/run-pass/dst-field-align.rs b/src/test/run-pass/dst-field-align.rs new file mode 100644 index 00000000000..cf2acfe986c --- /dev/null +++ b/src/test/run-pass/dst-field-align.rs @@ -0,0 +1,87 @@ +// Copyright 2015 The Rust Project Developers. See the COPYRIGHT +// file at the top-level directory of this distribution and at +// http://rust-lang.org/COPYRIGHT. +// +// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or +// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license +// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your +// option. This file may not be copied, modified, or distributed +// except according to those terms. + +struct Foo<T: ?Sized> { + a: u16, + b: T +} + +trait Bar { + fn get(&self) -> usize; +} + +impl Bar for usize { + fn get(&self) -> usize { *self } +} + +struct Baz<T: ?Sized> { + a: T +} + +#[repr(packed)] +struct Packed<T: ?Sized> { + a: u8, + b: T +} + +struct HasDrop<T: ?Sized> { + ptr: Box<usize>, + data: T +} + +fn main() { + // Test that zero-offset works properly + let b : Baz<usize> = Baz { a: 7 }; + assert_eq!(b.a.get(), 7); + let b : &Baz<Bar> = &b; + assert_eq!(b.a.get(), 7); + + // Test that the field is aligned properly + let f : Foo<usize> = Foo { a: 0, b: 11 }; + assert_eq!(f.b.get(), 11); + let ptr1 : *const u8 = &f.b as *const _ as *const u8; + + let f : &Foo<Bar> = &f; + let ptr2 : *const u8 = &f.b as *const _ as *const u8; + assert_eq!(f.b.get(), 11); + + // The pointers should be the same + assert_eq!(ptr1, ptr2); + + // Test that packed structs are handled correctly + let p : Packed<usize> = Packed { a: 0, b: 13 }; + assert_eq!(p.b.get(), 13); + let p : &Packed<Bar> = &p; + assert_eq!(p.b.get(), 13); + + // Test that nested DSTs work properly + let f : Foo<Foo<usize>> = Foo { a: 0, b: Foo { a: 1, b: 17 }}; + assert_eq!(f.b.b.get(), 17); + let f : &Foo<Foo<Bar>> = &f; + assert_eq!(f.b.b.get(), 17); + + // Test that get the pointer via destructuring works + + let f : Foo<usize> = Foo { a: 0, b: 11 }; + let f : &Foo<Bar> = &f; + let &Foo { a: _, b: ref bar } = f; + assert_eq!(bar.get(), 11); + + // Make sure that drop flags don't screw things up + + let d : HasDrop<Baz<[i32; 4]>> = HasDrop { + ptr: Box::new(0), + data: Baz { a: [1,2,3,4] } + }; + assert_eq!([1,2,3,4], d.data.a); + + let d : &HasDrop<Baz<[i32]>> = &d; + assert_eq!(&[1,2,3,4], &d.data.a); +} |
