The tale of repr align.
Core team is focussed on broader things
Don't want to put a heap of time into something that's already underway
Implement [repr(align)]
#[repr(C)]struct S { field1: i8, // offset 0, 1 byte aligned // 3 bytes padding field2: i32, // offset 4, 4 byte aligned // 0 bytes padding field3: i16, // offset 8, 2 byte aligned // 2 bytes padding} // struct is 12 bytes and 4 byte aligned
This is how a struct is aligned and padded without custom alignment on x86.
So if you imagine an array of these, the data alignment would be correct
%S = type { i8, [3 x i8], i32, [0 x i8], i16, [2 x i8] }
struct S { field1: i8, // offset 6, 1 byte aligned field2: i32, // offset 0, 4 byte aligned field3: i16, // offset 4, 2 byte aligned // 1 bytes padding} // struct is 8 bytes and 4 byte aligned
[repr(C)]
disables reordering and means the struct can be used with FFI.%S = type { i32, [0 x i8], i16, [0 x i8], i8, [1 x i8] }
#![feature(attr_literals)]#![feature(repr_align)]#[repr(align(16))]struct Align16(i32);struct Nested { field1: i32, // offset 1, 4 byte aligned field2: Align16, // offset 0, 16 byte aligned // 12 bytes padding} // struct is 32 bytes and 16 byte aligned
Notice the offsets in Nested, this struct layout has been re-ordered by Rust.
Making a start
[repr(align)]
tell LLVM about it and write some tests.#[repr(align = "16")]struct Align16(i32);assert_eq!(mem::align_of::<Align16>(), 16);assert_eq!(mem::size_of::<Align16>(), 16);
repr
to work out how to parse.println
debugging rustc to understand what it's doing.CONTRIBUTING.md
src/bootstrap/README.md
for more about that.x.py build --stage 1
builds the stage 1 compiler,x.py test --stage 1 src/tests/run-pass --test-args align-struct.rs
builds stage 1 and runs matching tests.x.py test --stage 1
not that useful, some tests require a stage 2 compiler and will fail at stage 1.x.py test src/tools/tidy
performs rust lint checks.libsyntax
contains things purely concerned with syntax, e.g. parser, AST, macro expander, etc.librustc
contains the high-level analysis passes, such as the type checker, borrow checker, etc. It is the heart of the compiler.librustc_back
contains some very low-level details that are specific to different LLVM targets.librustc_trans
contains the code to convert from Rust IR into LLVM IR, and then from LLVM IR into machine code.librustc_driver
invokes the compiler from libsyntax
, then the analysis phases from librustc
, and finally the lowering and codegen passes from librustc_trans
.src/librustc/README.txt
contains more high level documentation of the compiler.Got the RFC example code working and announce on RFC task that I'm working on it and will have a PR soon!
mem::align_of
and mem::size_of
are reporting the right thing, it doesn't mean that everything is working.alloca
and manually pad fields and structs for custom alignment.Announce my realisation on tracking issue that this is harder than I thought :)
librustc
ty::layout::Layout
enum.[repr(align))]
is used.src/test/run-pass
- check what's supposed to work workssrc/test/compile-fail
- check compile errors are handled correctlysrc/test/ui
- print-type-sizes
Make a big push to get everything ready and open PR for repr(align)
support.
The first 90 percent of the code accounts for the first 90 percent of the development time. The remaining 10 percent of the code accounts for the other 90 percent of the development time. - Tom Cargill, Bell Labs
[repr(align = "n")]
to [repr(align(n))]
.librustc_trans
used hard coded indices for accessing fields, these were no longer correct.cp src/bootstrap/config.toml.example config.toml
export RUST_BACKTRACE=full
export RUST_LOG=rustc_trans=debug
sudo ~/.cargo/bin/rust-gdb -p <rustc pid>
break abort
- breaks on LLVM assertionbors submits the PR to master!
This:
#![feature(attr_literals)]#![feature(repr_align)]#[repr(align(64))]pub struct Align64(i32);pub struct Nested64 { a: Align64, b: i32,}#[no_mangle]pub fn nested64(a: Align64, b: i32) -> Nested64 { Nested64 { a, b }}
Note: [repr(align)]
is feature gated, I don't know what the processes or timeline for stabalisation is.
Becomes this:
%Nested64 = type { %Align64, [0 x i8], i32, [60 x i8] }%Align64 = type { i32, [60 x i8] }; Function Attrs: nounwind uwtabledefine void @nested64( %Nested64* noalias nocapture sret dereferenceable(128), %Align64* noalias nocapture readonly dereferenceable(64), i32) unnamed_addr #0 {start: %3 = bitcast %Align64* %1 to i8* %4 = bitcast %Nested64* %0 to i8* call void @llvm.memcpy.p0i8.p0i8.i64(i8* nonnull %4, i8* nonnull %3, i64 64, i32 64, i1 false) %5 = getelementptr inbounds %Nested64, %Nested64* %0, i64 0, i32 2 store i32 %2, i32* %5, align 4 ret void}
Nightly release, debug was too much code. Could look at playpen though
Keyboard shortcuts
↑, ←, Pg Up, k | Go to previous slide |
↓, →, Pg Dn, Space, j | Go to next slide |
Home | Go to first slide |
End | Go to last slide |
b / m / f | Toggle blackout / mirrored / fullscreen mode |
c | Clone slideshow |
p | Toggle presenter mode |
t | Restart the presentation timer |
?, h | Toggle this help |
Esc | Back to slideshow |