@@ -2,6 +2,7 @@ use super::place::PlaceRef;
2
2
use super :: { FunctionCx , LocalRef } ;
3
3
4
4
use crate :: base;
5
+ use crate :: common:: TypeKind ;
5
6
use crate :: glue;
6
7
use crate :: traits:: * ;
7
8
use crate :: MemFlags ;
@@ -236,19 +237,47 @@ impl<'a, 'tcx, V: CodegenObject> OperandRef<'tcx, V> {
236
237
} ;
237
238
238
239
match ( & mut val, field. abi ) {
239
- ( OperandValue :: Immediate ( llval) , _) => {
240
+ (
241
+ OperandValue :: Immediate ( llval) ,
242
+ Abi :: Scalar ( _) | Abi :: ScalarPair ( ..) | Abi :: Vector { .. } ,
243
+ ) => {
240
244
// Bools in union fields needs to be truncated.
241
245
* llval = bx. to_immediate ( * llval, field) ;
242
246
// HACK(eddyb) have to bitcast pointers until LLVM removes pointee types.
243
- * llval = bx. bitcast ( * llval, bx. cx ( ) . immediate_backend_type ( field) ) ;
247
+ let ty = bx. cx ( ) . immediate_backend_type ( field) ;
248
+ if bx. type_kind ( ty) == TypeKind :: Pointer {
249
+ * llval = bx. pointercast ( * llval, ty) ;
250
+ }
244
251
}
245
252
( OperandValue :: Pair ( a, b) , Abi :: ScalarPair ( a_abi, b_abi) ) => {
246
253
// Bools in union fields needs to be truncated.
247
254
* a = bx. to_immediate_scalar ( * a, a_abi) ;
248
255
* b = bx. to_immediate_scalar ( * b, b_abi) ;
249
256
// HACK(eddyb) have to bitcast pointers until LLVM removes pointee types.
250
- * a = bx. bitcast ( * a, bx. cx ( ) . scalar_pair_element_backend_type ( field, 0 , true ) ) ;
251
- * b = bx. bitcast ( * b, bx. cx ( ) . scalar_pair_element_backend_type ( field, 1 , true ) ) ;
257
+ let a_ty = bx. cx ( ) . scalar_pair_element_backend_type ( field, 0 , true ) ;
258
+ let b_ty = bx. cx ( ) . scalar_pair_element_backend_type ( field, 1 , true ) ;
259
+ if bx. type_kind ( a_ty) == TypeKind :: Pointer {
260
+ * a = bx. pointercast ( * a, a_ty) ;
261
+ }
262
+ if bx. type_kind ( b_ty) == TypeKind :: Pointer {
263
+ * b = bx. pointercast ( * b, b_ty) ;
264
+ }
265
+ }
266
+ // Newtype vector of array, e.g. #[repr(simd)] struct S([i32; 4]);
267
+ ( OperandValue :: Immediate ( llval) , Abi :: Aggregate { sized : true } ) => {
268
+ assert ! ( matches!( self . layout. abi, Abi :: Vector { .. } ) ) ;
269
+
270
+ let llty = bx. cx ( ) . backend_type ( self . layout ) ;
271
+ let llfield_ty = bx. cx ( ) . backend_type ( field) ;
272
+
273
+ // Can't bitcast an aggregate, so round trip through memory.
274
+ let lltemp = bx. alloca ( llfield_ty, field. align . abi ) ;
275
+ let llptr = bx. pointercast ( lltemp, bx. cx ( ) . type_ptr_to ( llty) ) ;
276
+ bx. store ( * llval, llptr, field. align . abi ) ;
277
+ * llval = bx. load ( llfield_ty, lltemp, field. align . abi ) ;
278
+ }
279
+ ( OperandValue :: Immediate ( _) , Abi :: Uninhabited | Abi :: Aggregate { sized : false } ) => {
280
+ bug ! ( )
252
281
}
253
282
( OperandValue :: Pair ( ..) , _) => bug ! ( ) ,
254
283
( OperandValue :: Ref ( ..) , _) => bug ! ( ) ,
0 commit comments