Skip to content

Commit

Permalink
codegen: Properly track alignment of unions.
Browse files Browse the repository at this point in the history
This makes us not unnecessarily add repr(align) to unions.

Closes #1498.
  • Loading branch information
emilio committed Mar 4, 2019
1 parent ed6e1bb commit 9134679
Show file tree
Hide file tree
Showing 36 changed files with 201 additions and 73 deletions.
46 changes: 27 additions & 19 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,53 +9,56 @@
- [Removed](#removed)
- [Fixed](#fixed)
- [Security](#security)
- [0.47.3](#0473)
- [0.48.0](#0480)
- [Changed](#changed-1)
- [0.47.2](#0472)
- [Fixed](#fixed-1)
- [0.47.1](#0471)
- [0.47.3](#0473)
- [Changed](#changed-2)
- [0.47.2](#0472)
- [Fixed](#fixed-2)
- [0.47.0](#0470)
- [0.47.1](#0471)
- [Changed](#changed-3)
- [Fixed](#fixed-3)
- [0.47.0](#0470)
- [Changed](#changed-4)
- [Fixed](#fixed-4)
- [0.33.1 .. 0.46.0](#0331--0460)
- [Added](#added-1)
- [Removed](#removed-1)
- [Changed](#changed-4)
- [Fixed](#fixed-4)
- [0.33.1](#0331)
- [Changed](#changed-5)
- [Fixed](#fixed-5)
- [0.33.1](#0331)
- [Fixed](#fixed-6)
- [0.33.0](#0330)
- [Added](#added-2)
- [Changed](#changed-5)
- [Changed](#changed-6)
- [Deprecated](#deprecated-1)
- [Removed](#removed-2)
- [Fixed](#fixed-6)
- [Fixed](#fixed-7)
- [Security](#security-1)
- [0.32.2](#0322)
- [Fixed](#fixed-7)
- [0.32.1](#0321)
- [Fixed](#fixed-8)
- [0.32.1](#0321)
- [Fixed](#fixed-9)
- [0.32.0](#0320)
- [Added](#added-3)
- [Changed](#changed-6)
- [Fixed](#fixed-9)
- [Changed](#changed-7)
- [Fixed](#fixed-10)
- [0.31.0](#0310)
- [Added](#added-4)
- [Changed](#changed-7)
- [Changed](#changed-8)
- [Deprecated](#deprecated-2)
- [Removed](#removed-3)
- [Fixed](#fixed-10)
- [Fixed](#fixed-11)
- [0.30.0](#0300)
- [Added](#added-5)
- [Changed](#changed-8)
- [Changed](#changed-9)
- [Deprecated](#deprecated-3)
- [Fixed](#fixed-11)
- [Fixed](#fixed-12)
- [0.29.0](#0290)
- [Added](#added-6)
- [Changed](#changed-9)
- [Fixed](#fixed-12)
- [Changed](#changed-10)
- [Fixed](#fixed-13)

<!-- END doctoc generated TOC please keep comment here to allow auto update -->

Expand Down Expand Up @@ -101,7 +104,12 @@ Released 2019/03/04
* Default rust target was changed to 1.33, which means that bindgen can get much
more often the layout of structs right. [#1529][]

## Fixed

* Bindgen will output repr(align) just when needed for unions. [#1498][]

[#1529]: https://github.com/rust-lang-nursery/rust-bindgen/issues/1529
[#1498]: https://github.com/rust-lang-nursery/rust-bindgen/issues/1498

--------------------------------------------------------------------------------

Expand Down
3 changes: 1 addition & 2 deletions src/codegen/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1589,6 +1589,7 @@ impl CodeGenerator for CompInfo {
// TODO(emilio): It'd be nice to unify this with the struct path
// above somehow.
let layout = layout.expect("Unable to get layout information?");
struct_layout.saw_union(layout);

if struct_layout.requires_explicit_align(layout) {
explicit_align = Some(layout.align);
Expand All @@ -1600,8 +1601,6 @@ impl CodeGenerator for CompInfo {
_bindgen_union_align: #ty ,
}
} else {
struct_layout.saw_union(layout);

quote! {
pub bindgen_union_field: #ty ,
}
Expand Down
6 changes: 3 additions & 3 deletions src/codegen/struct_layout.rs
Original file line number Diff line number Diff line change
Expand Up @@ -85,9 +85,9 @@ impl<'a> StructLayoutTracker<'a> {
name: &'a str,
) -> Self {
StructLayoutTracker {
name: name,
ctx: ctx,
comp: comp,
name,
ctx,
comp,
is_packed: comp.is_packed(ctx, &ty.layout(ctx)),
latest_offset: 0,
padding_count: 0,
Expand Down
2 changes: 0 additions & 2 deletions tests/expectations/tests/16-byte-alignment.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,6 @@ pub struct rte_ipv4_tuple {
pub __bindgen_anon_1: rte_ipv4_tuple__bindgen_ty_1,
}
#[repr(C)]
#[repr(align(4))]
#[derive(Copy, Clone)]
pub union rte_ipv4_tuple__bindgen_ty_1 {
pub __bindgen_anon_1: rte_ipv4_tuple__bindgen_ty_1__bindgen_ty_1,
Expand Down Expand Up @@ -149,7 +148,6 @@ pub struct rte_ipv6_tuple {
pub __bindgen_anon_1: rte_ipv6_tuple__bindgen_ty_1,
}
#[repr(C)]
#[repr(align(4))]
#[derive(Copy, Clone)]
pub union rte_ipv6_tuple__bindgen_ty_1 {
pub __bindgen_anon_1: rte_ipv6_tuple__bindgen_ty_1__bindgen_ty_1,
Expand Down
1 change: 0 additions & 1 deletion tests/expectations/tests/anon_struct_in_union.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,6 @@ pub struct s {
pub u: s__bindgen_ty_1,
}
#[repr(C)]
#[repr(align(4))]
#[derive(Copy, Clone)]
pub union s__bindgen_ty_1 {
pub field: s__bindgen_ty_1_inner,
Expand Down
1 change: 0 additions & 1 deletion tests/expectations/tests/anon_union.rs
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,6 @@ pub struct TErrorResult_DOMExceptionInfo {
_unused: [u8; 0],
}
#[repr(C)]
#[repr(align(8))]
pub union TErrorResult__bindgen_ty_1 {
pub mMessage: *mut TErrorResult_Message,
pub mDOMExceptionInfo: *mut TErrorResult_DOMExceptionInfo,
Expand Down
1 change: 0 additions & 1 deletion tests/expectations/tests/class.rs
Original file line number Diff line number Diff line change
Expand Up @@ -335,7 +335,6 @@ impl Default for IncompleteArrayNonCopiable {
}
}
#[repr(C)]
#[repr(align(4))]
#[derive(Copy, Clone)]
pub union Union {
pub d: f32,
Expand Down
3 changes: 0 additions & 3 deletions tests/expectations/tests/class_with_inner_struct.rs
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,6 @@ fn bindgen_test_layout_A_Segment() {
);
}
#[repr(C)]
#[repr(align(4))]
#[derive(Copy, Clone)]
pub union A__bindgen_ty_1 {
pub f: ::std::os::raw::c_int,
Expand Down Expand Up @@ -89,7 +88,6 @@ impl Default for A__bindgen_ty_1 {
}
}
#[repr(C)]
#[repr(align(4))]
#[derive(Copy, Clone)]
pub union A__bindgen_ty_2 {
pub d: ::std::os::raw::c_int,
Expand Down Expand Up @@ -233,7 +231,6 @@ pub struct C {
pub __bindgen_anon_1: C__bindgen_ty_1,
}
#[repr(C)]
#[repr(align(4))]
#[derive(Copy, Clone)]
pub union C__bindgen_ty_1 {
pub mFunc: C__bindgen_ty_1__bindgen_ty_1,
Expand Down
1 change: 0 additions & 1 deletion tests/expectations/tests/derive-debug-mangle-name.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,6 @@ pub struct perf_event_attr {
pub __bindgen_anon_1: perf_event_attr__bindgen_ty_1,
}
#[repr(C)]
#[repr(align(4))]
#[derive(Copy, Clone)]
pub union perf_event_attr__bindgen_ty_1 {
pub b: ::std::os::raw::c_int,
Expand Down
1 change: 0 additions & 1 deletion tests/expectations/tests/derive-partialeq-anonfield.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,6 @@ pub struct rte_mbuf {
pub __bindgen_anon_1: rte_mbuf__bindgen_ty_1,
}
#[repr(C)]
#[repr(align(1))]
#[derive(Copy, Clone)]
pub union rte_mbuf__bindgen_ty_1 {
_bindgen_union_align: [u8; 0usize],
Expand Down
1 change: 0 additions & 1 deletion tests/expectations/tests/derive-partialeq-pointer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,6 @@ pub struct c {
pub __bindgen_anon_1: c__bindgen_ty_1,
}
#[repr(C)]
#[repr(align(1))]
#[derive(Copy, Clone)]
pub union c__bindgen_ty_1 {
_bindgen_union_align: u8,
Expand Down
1 change: 0 additions & 1 deletion tests/expectations/tests/derive-partialeq-union.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@

/// Deriving PartialEq for rust unions is not supported.
#[repr(C)]
#[repr(align(4))]
#[derive(Copy, Clone)]
pub union ShouldNotDerivePartialEq {
pub a: ::std::os::raw::c_char,
Expand Down
1 change: 0 additions & 1 deletion tests/expectations/tests/issue-1285.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,6 @@ pub struct foo {
pub bar: foo__bindgen_ty_1,
}
#[repr(C)]
#[repr(align(4))]
pub union foo__bindgen_ty_1 {
pub a: ::std::os::raw::c_uint,
pub b: ::std::os::raw::c_ushort,
Expand Down
153 changes: 153 additions & 0 deletions tests/expectations/tests/issue-1498.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,153 @@
/* automatically generated by rust-bindgen */

#![allow(
dead_code,
non_snake_case,
non_camel_case_types,
non_upper_case_globals
)]

#[repr(C, packed)]
#[derive(Copy, Clone)]
pub struct rte_memseg {
///< Start physical address.
pub phys_addr: u64,
pub __bindgen_anon_1: rte_memseg__bindgen_ty_1,
///< Length of the segment.
pub len: usize,
///< The pagesize of underlying memory
pub hugepage_sz: u64,
///< NUMA socket ID.
pub socket_id: i32,
///< Number of channels.
pub nchannel: u32,
///< Number of ranks.
pub nrank: u32,
}
#[repr(C)]
#[derive(Copy, Clone)]
pub union rte_memseg__bindgen_ty_1 {
///< Start virtual address.
pub addr: *mut ::std::os::raw::c_void,
///< Makes sure addr is always 64 bits
pub addr_64: u64,
_bindgen_union_align: u64,
}
#[test]
fn bindgen_test_layout_rte_memseg__bindgen_ty_1() {
assert_eq!(
::std::mem::size_of::<rte_memseg__bindgen_ty_1>(),
8usize,
concat!("Size of: ", stringify!(rte_memseg__bindgen_ty_1))
);
assert_eq!(
::std::mem::align_of::<rte_memseg__bindgen_ty_1>(),
8usize,
concat!("Alignment of ", stringify!(rte_memseg__bindgen_ty_1))
);
assert_eq!(
unsafe { &(*(::std::ptr::null::<rte_memseg__bindgen_ty_1>())).addr as *const _ as usize },
0usize,
concat!(
"Offset of field: ",
stringify!(rte_memseg__bindgen_ty_1),
"::",
stringify!(addr)
)
);
assert_eq!(
unsafe {
&(*(::std::ptr::null::<rte_memseg__bindgen_ty_1>())).addr_64 as *const _ as usize
},
0usize,
concat!(
"Offset of field: ",
stringify!(rte_memseg__bindgen_ty_1),
"::",
stringify!(addr_64)
)
);
}
impl Default for rte_memseg__bindgen_ty_1 {
fn default() -> Self {
unsafe { ::std::mem::zeroed() }
}
}
#[test]
fn bindgen_test_layout_rte_memseg() {
assert_eq!(
::std::mem::size_of::<rte_memseg>(),
44usize,
concat!("Size of: ", stringify!(rte_memseg))
);
assert_eq!(
::std::mem::align_of::<rte_memseg>(),
1usize,
concat!("Alignment of ", stringify!(rte_memseg))
);
assert_eq!(
unsafe { &(*(::std::ptr::null::<rte_memseg>())).phys_addr as *const _ as usize },
0usize,
concat!(
"Offset of field: ",
stringify!(rte_memseg),
"::",
stringify!(phys_addr)
)
);
assert_eq!(
unsafe { &(*(::std::ptr::null::<rte_memseg>())).len as *const _ as usize },
16usize,
concat!(
"Offset of field: ",
stringify!(rte_memseg),
"::",
stringify!(len)
)
);
assert_eq!(
unsafe { &(*(::std::ptr::null::<rte_memseg>())).hugepage_sz as *const _ as usize },
24usize,
concat!(
"Offset of field: ",
stringify!(rte_memseg),
"::",
stringify!(hugepage_sz)
)
);
assert_eq!(
unsafe { &(*(::std::ptr::null::<rte_memseg>())).socket_id as *const _ as usize },
32usize,
concat!(
"Offset of field: ",
stringify!(rte_memseg),
"::",
stringify!(socket_id)
)
);
assert_eq!(
unsafe { &(*(::std::ptr::null::<rte_memseg>())).nchannel as *const _ as usize },
36usize,
concat!(
"Offset of field: ",
stringify!(rte_memseg),
"::",
stringify!(nchannel)
)
);
assert_eq!(
unsafe { &(*(::std::ptr::null::<rte_memseg>())).nrank as *const _ as usize },
40usize,
concat!(
"Offset of field: ",
stringify!(rte_memseg),
"::",
stringify!(nrank)
)
);
}
impl Default for rte_memseg {
fn default() -> Self {
unsafe { ::std::mem::zeroed() }
}
}
1 change: 0 additions & 1 deletion tests/expectations/tests/issue-493.rs
Original file line number Diff line number Diff line change
Expand Up @@ -83,7 +83,6 @@ pub struct basic_string___short {
pub __data_: *mut basic_string_value_type,
}
#[repr(C)]
#[repr(align(1))]
pub union basic_string___short__bindgen_ty_1 {
pub __size_: ::std::os::raw::c_uchar,
pub __lx: basic_string_value_type,
Expand Down
Loading

0 comments on commit 9134679

Please sign in to comment.