@@ -10,7 +10,7 @@ use rustc_middle::ty::layout::{LayoutOf, PrimitiveExt, TyAndLayout};
10
10
use rustc_middle:: ty:: print:: { FmtPrinter , PrettyPrinter , Printer } ;
11
11
use rustc_middle:: ty:: { ConstInt , DelaySpanBugEmitted , Ty } ;
12
12
use rustc_middle:: { mir, ty} ;
13
- use rustc_target:: abi:: { self , Abi , HasDataLayout , Size , TagEncoding } ;
13
+ use rustc_target:: abi:: { self , Abi , Align , HasDataLayout , Size , TagEncoding } ;
14
14
use rustc_target:: abi:: { VariantIdx , Variants } ;
15
15
16
16
use super :: {
@@ -177,10 +177,18 @@ pub enum Operand<Tag: Provenance = AllocId> {
177
177
pub struct OpTy < ' tcx , Tag : Provenance = AllocId > {
178
178
op : Operand < Tag > , // Keep this private; it helps enforce invariants.
179
179
pub layout : TyAndLayout < ' tcx > ,
180
+ /// rustc does not have a proper way to represent the type of a field of a `repr(packed)` struct:
181
+ /// it needs to have a different alignment than the field type would usually have.
182
+ /// So we represent this here with a separate field that "overwrites" `layout.align`.
183
+ /// This means `layout.align` should never be used for an `OpTy`!
184
+ /// `None` means "alignment does not matter since this is a by-value operand"
185
+ /// (`Operand::Immediate`); this field is only relevant for `Operand::Indirect`.
186
+ /// Also CTFE ignores alignment anyway, so this is for Miri only.
187
+ pub align : Option < Align > ,
180
188
}
181
189
182
190
#[ cfg( all( target_arch = "x86_64" , target_pointer_width = "64" ) ) ]
183
- rustc_data_structures:: static_assert_size!( OpTy <' _>, 80 ) ;
191
+ rustc_data_structures:: static_assert_size!( OpTy <' _>, 88 ) ;
184
192
185
193
impl < ' tcx , Tag : Provenance > std:: ops:: Deref for OpTy < ' tcx , Tag > {
186
194
type Target = Operand < Tag > ;
@@ -193,28 +201,28 @@ impl<'tcx, Tag: Provenance> std::ops::Deref for OpTy<'tcx, Tag> {
193
201
impl < ' tcx , Tag : Provenance > From < MPlaceTy < ' tcx , Tag > > for OpTy < ' tcx , Tag > {
194
202
#[ inline( always) ]
195
203
fn from ( mplace : MPlaceTy < ' tcx , Tag > ) -> Self {
196
- OpTy { op : Operand :: Indirect ( * mplace) , layout : mplace. layout }
204
+ OpTy { op : Operand :: Indirect ( * mplace) , layout : mplace. layout , align : Some ( mplace . align ) }
197
205
}
198
206
}
199
207
200
208
impl < ' tcx , Tag : Provenance > From < & ' _ MPlaceTy < ' tcx , Tag > > for OpTy < ' tcx , Tag > {
201
209
#[ inline( always) ]
202
210
fn from ( mplace : & MPlaceTy < ' tcx , Tag > ) -> Self {
203
- OpTy { op : Operand :: Indirect ( * * mplace) , layout : mplace. layout }
211
+ OpTy { op : Operand :: Indirect ( * * mplace) , layout : mplace. layout , align : Some ( mplace . align ) }
204
212
}
205
213
}
206
214
207
215
impl < ' tcx , Tag : Provenance > From < & ' _ mut MPlaceTy < ' tcx , Tag > > for OpTy < ' tcx , Tag > {
208
216
#[ inline( always) ]
209
217
fn from ( mplace : & mut MPlaceTy < ' tcx , Tag > ) -> Self {
210
- OpTy { op : Operand :: Indirect ( * * mplace) , layout : mplace. layout }
218
+ OpTy { op : Operand :: Indirect ( * * mplace) , layout : mplace. layout , align : Some ( mplace . align ) }
211
219
}
212
220
}
213
221
214
222
impl < ' tcx , Tag : Provenance > From < ImmTy < ' tcx , Tag > > for OpTy < ' tcx , Tag > {
215
223
#[ inline( always) ]
216
224
fn from ( val : ImmTy < ' tcx , Tag > ) -> Self {
217
- OpTy { op : Operand :: Immediate ( val. imm ) , layout : val. layout }
225
+ OpTy { op : Operand :: Immediate ( val. imm ) , layout : val. layout , align : None }
218
226
}
219
227
}
220
228
@@ -450,7 +458,7 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
450
458
) ,
451
459
} ;
452
460
453
- Ok ( OpTy { op : Operand :: Immediate ( field_val) , layout : field_layout } )
461
+ Ok ( OpTy { op : Operand :: Immediate ( field_val) , layout : field_layout, align : None } )
454
462
}
455
463
456
464
pub fn operand_index (
@@ -522,7 +530,7 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
522
530
///
523
531
/// This is public because it is used by [priroda](https://github.com/oli-obk/priroda) to get an
524
532
/// OpTy from a local
525
- pub fn access_local (
533
+ pub fn local_to_op (
526
534
& self ,
527
535
frame : & super :: Frame < ' mir , ' tcx , M :: PointerTag , M :: FrameExtra > ,
528
536
local : mir:: Local ,
@@ -535,7 +543,7 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
535
543
} else {
536
544
M :: access_local ( & self , frame, local) ?
537
545
} ;
538
- Ok ( OpTy { op, layout } )
546
+ Ok ( OpTy { op, layout, align : Some ( layout . align . abi ) } )
539
547
}
540
548
541
549
/// Every place can be read from, so we can turn them into an operand.
@@ -549,10 +557,10 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
549
557
let op = match * * place {
550
558
Place :: Ptr ( mplace) => Operand :: Indirect ( mplace) ,
551
559
Place :: Local { frame, local } => {
552
- * self . access_local ( & self . stack ( ) [ frame] , local, None ) ?
560
+ * self . local_to_op ( & self . stack ( ) [ frame] , local, None ) ?
553
561
}
554
562
} ;
555
- Ok ( OpTy { op, layout : place. layout } )
563
+ Ok ( OpTy { op, layout : place. layout , align : Some ( place . align ) } )
556
564
}
557
565
558
566
/// Evaluate a place with the goal of reading from it. This lets us sometimes
@@ -566,7 +574,7 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
566
574
// here is not the entire place.
567
575
let layout = if place. projection . is_empty ( ) { layout } else { None } ;
568
576
569
- let base_op = self . access_local ( self . frame ( ) , place. local , layout) ?;
577
+ let base_op = self . local_to_op ( self . frame ( ) , place. local , layout) ?;
570
578
571
579
let op = place
572
580
. projection
@@ -603,11 +611,11 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
603
611
Constant ( ref constant) => {
604
612
let val =
605
613
self . subst_from_current_frame_and_normalize_erasing_regions ( constant. literal ) ?;
614
+
606
615
// This can still fail:
607
616
// * During ConstProp, with `TooGeneric` or since the `required_consts` were not all
608
617
// checked yet.
609
618
// * During CTFE, since promoteds in `const`/`static` initializer bodies can fail.
610
-
611
619
self . mir_const_to_op ( & val, layout) ?
612
620
}
613
621
} ;
@@ -683,7 +691,7 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
683
691
// We rely on mutability being set correctly in that allocation to prevent writes
684
692
// where none should happen.
685
693
let ptr = self . global_base_pointer ( Pointer :: new ( id, offset) ) ?;
686
- Operand :: Indirect ( MemPlace :: from_ptr ( ptr. into ( ) , layout . align . abi ) )
694
+ Operand :: Indirect ( MemPlace :: from_ptr ( ptr. into ( ) ) )
687
695
}
688
696
ConstValue :: Scalar ( x) => Operand :: Immediate ( tag_scalar ( x) ?. into ( ) ) ,
689
697
ConstValue :: Slice { data, start, end } => {
@@ -700,7 +708,7 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
700
708
) )
701
709
}
702
710
} ;
703
- Ok ( OpTy { op, layout } )
711
+ Ok ( OpTy { op, layout, align : Some ( layout . align . abi ) } )
704
712
}
705
713
706
714
/// Read discriminant, return the runtime value as well as the variant index.
0 commit comments