Rollup merge of #56097 - ogoffart:union-abi, r=eddyb
Fix invalid bitcast taking bool out of a union represented as a scalar As reported in https://github.com/rust-lang/rust/pull/54668#issuecomment-440186476
This commit is contained in:
commit
e0025df3fd
2 changed files with 20 additions and 3 deletions
|
|
@ -244,13 +244,24 @@ impl<'a, 'tcx: 'a, V: CodegenObject> OperandRef<'tcx, V> {
|
|||
};
|
||||
|
||||
// HACK(eddyb) have to bitcast pointers until LLVM removes pointee types.
|
||||
// Bools in union fields needs to be truncated.
|
||||
let to_immediate_or_cast = |bx: &mut Bx, val, ty| {
|
||||
if ty == bx.cx().type_i1() {
|
||||
bx.trunc(val, ty)
|
||||
} else {
|
||||
bx.bitcast(val, ty)
|
||||
}
|
||||
};
|
||||
|
||||
match val {
|
||||
OperandValue::Immediate(ref mut llval) => {
|
||||
*llval = bx.bitcast(*llval, bx.cx().immediate_backend_type(field));
|
||||
*llval = to_immediate_or_cast(bx, *llval, bx.cx().immediate_backend_type(field));
|
||||
}
|
||||
OperandValue::Pair(ref mut a, ref mut b) => {
|
||||
*a = bx.bitcast(*a, bx.cx().scalar_pair_element_backend_type(field, 0, true));
|
||||
*b = bx.bitcast(*b, bx.cx().scalar_pair_element_backend_type(field, 1, true));
|
||||
*a = to_immediate_or_cast(bx, *a, bx.cx()
|
||||
.scalar_pair_element_backend_type(field, 0, true));
|
||||
*b = to_immediate_or_cast(bx, *b, bx.cx()
|
||||
.scalar_pair_element_backend_type(field, 1, true));
|
||||
}
|
||||
OperandValue::Ref(..) => bug!()
|
||||
}
|
||||
|
|
|
|||
|
|
@ -78,3 +78,9 @@ pub union CUnionU128{a:u128}
|
|||
#[no_mangle]
|
||||
pub fn test_CUnionU128(_: CUnionU128) { loop {} }
|
||||
|
||||
pub union UnionBool { b:bool }
|
||||
// CHECK: define zeroext i1 @test_UnionBool(i8 %b)
|
||||
#[no_mangle]
|
||||
pub fn test_UnionBool(b: UnionBool) -> bool { unsafe { b.b } }
|
||||
// CHECK: %0 = trunc i8 %b to i1
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue