@@ -8,15 +8,19 @@ use tracing::debug;
8
8
use crate :: {
9
9
Abi , AbiAndPrefAlign , Align , FieldsShape , IndexSlice , IndexVec , Integer , LayoutS , Niche ,
10
10
NonZeroUsize , Primitive , ReprOptions , Scalar , Size , StructKind , TagEncoding , TargetDataLayout ,
11
- VariantIdx , Variants , WrappingRange , FIRST_VARIANT ,
11
+ Variants , WrappingRange ,
12
12
} ;
13
13
pub trait LayoutCalculator {
14
14
type TargetDataLayoutRef : Borrow < TargetDataLayout > ;
15
15
16
16
fn delay_bug ( & self , txt : String ) ;
17
17
fn current_data_layout ( & self ) -> Self :: TargetDataLayoutRef ;
18
18
19
- fn scalar_pair < FieldIdx : Idx > ( & self , a : Scalar , b : Scalar ) -> LayoutS < FieldIdx > {
19
+ fn scalar_pair < FieldIdx : Idx , VariantIdx : Idx > (
20
+ & self ,
21
+ a : Scalar ,
22
+ b : Scalar ,
23
+ ) -> LayoutS < FieldIdx , VariantIdx > {
20
24
let dl = self . current_data_layout ( ) ;
21
25
let dl = dl. borrow ( ) ;
22
26
let b_align = b. align ( dl) ;
@@ -32,7 +36,7 @@ pub trait LayoutCalculator {
32
36
. max_by_key ( |niche| niche. available ( dl) ) ;
33
37
34
38
LayoutS {
35
- variants : Variants :: Single { index : FIRST_VARIANT } ,
39
+ variants : Variants :: Single { index : VariantIdx :: new ( 0 ) } ,
36
40
fields : FieldsShape :: Arbitrary {
37
41
offsets : [ Size :: ZERO , b_offset] . into ( ) ,
38
42
memory_index : [ 0 , 1 ] . into ( ) ,
@@ -46,13 +50,18 @@ pub trait LayoutCalculator {
46
50
}
47
51
}
48
52
49
- fn univariant < ' a , FieldIdx : Idx , F : Deref < Target = & ' a LayoutS < FieldIdx > > + fmt:: Debug > (
53
+ fn univariant <
54
+ ' a ,
55
+ FieldIdx : Idx ,
56
+ VariantIdx : Idx ,
57
+ F : Deref < Target = & ' a LayoutS < FieldIdx , VariantIdx > > + fmt:: Debug ,
58
+ > (
50
59
& self ,
51
60
dl : & TargetDataLayout ,
52
61
fields : & IndexSlice < FieldIdx , F > ,
53
62
repr : & ReprOptions ,
54
63
kind : StructKind ,
55
- ) -> Option < LayoutS < FieldIdx > > {
64
+ ) -> Option < LayoutS < FieldIdx , VariantIdx > > {
56
65
let layout = univariant ( self , dl, fields, repr, kind, NicheBias :: Start ) ;
57
66
// Enums prefer niches close to the beginning or the end of the variants so that other
58
67
// (smaller) data-carrying variants can be packed into the space after/before the niche.
@@ -115,11 +124,13 @@ pub trait LayoutCalculator {
115
124
layout
116
125
}
117
126
118
- fn layout_of_never_type < FieldIdx : Idx > ( & self ) -> LayoutS < FieldIdx > {
127
+ fn layout_of_never_type < FieldIdx : Idx , VariantIdx : Idx > (
128
+ & self ,
129
+ ) -> LayoutS < FieldIdx , VariantIdx > {
119
130
let dl = self . current_data_layout ( ) ;
120
131
let dl = dl. borrow ( ) ;
121
132
LayoutS {
122
- variants : Variants :: Single { index : FIRST_VARIANT } ,
133
+ variants : Variants :: Single { index : VariantIdx :: new ( 0 ) } ,
123
134
fields : FieldsShape :: Primitive ,
124
135
abi : Abi :: Uninhabited ,
125
136
largest_niche : None ,
@@ -133,7 +144,8 @@ pub trait LayoutCalculator {
133
144
fn layout_of_struct_or_enum <
134
145
' a ,
135
146
FieldIdx : Idx ,
136
- F : Deref < Target = & ' a LayoutS < FieldIdx > > + fmt:: Debug ,
147
+ VariantIdx : Idx ,
148
+ F : Deref < Target = & ' a LayoutS < FieldIdx , VariantIdx > > + fmt:: Debug ,
137
149
> (
138
150
& self ,
139
151
repr : & ReprOptions ,
@@ -145,7 +157,7 @@ pub trait LayoutCalculator {
145
157
discriminants : impl Iterator < Item = ( VariantIdx , i128 ) > ,
146
158
dont_niche_optimize_enum : bool ,
147
159
always_sized : bool ,
148
- ) -> Option < LayoutS < FieldIdx > > {
160
+ ) -> Option < LayoutS < FieldIdx , VariantIdx > > {
149
161
let dl = self . current_data_layout ( ) ;
150
162
let dl = dl. borrow ( ) ;
151
163
@@ -181,7 +193,7 @@ pub trait LayoutCalculator {
181
193
}
182
194
// If it's a struct, still compute a layout so that we can still compute the
183
195
// field offsets.
184
- None => FIRST_VARIANT ,
196
+ None => VariantIdx :: new ( 0 ) ,
185
197
} ;
186
198
187
199
let is_struct = !is_enum ||
@@ -284,12 +296,12 @@ pub trait LayoutCalculator {
284
296
// variant layouts, so we can't store them in the
285
297
// overall LayoutS. Store the overall LayoutS
286
298
// and the variant LayoutSs here until then.
287
- struct TmpLayout < FieldIdx : Idx > {
288
- layout : LayoutS < FieldIdx > ,
289
- variants : IndexVec < VariantIdx , LayoutS < FieldIdx > > ,
299
+ struct TmpLayout < FieldIdx : Idx , VariantIdx : Idx > {
300
+ layout : LayoutS < FieldIdx , VariantIdx > ,
301
+ variants : IndexVec < VariantIdx , LayoutS < FieldIdx , VariantIdx > > ,
290
302
}
291
303
292
- let calculate_niche_filling_layout = || -> Option < TmpLayout < FieldIdx > > {
304
+ let calculate_niche_filling_layout = || -> Option < TmpLayout < FieldIdx , VariantIdx > > {
293
305
if dont_niche_optimize_enum {
294
306
return None ;
295
307
}
@@ -327,7 +339,8 @@ pub trait LayoutCalculator {
327
339
let niche_variants = all_indices. clone ( ) . find ( |v| needs_disc ( * v) ) . unwrap ( )
328
340
..=all_indices. rev ( ) . find ( |v| needs_disc ( * v) ) . unwrap ( ) ;
329
341
330
- let count = niche_variants. size_hint ( ) . 1 . unwrap ( ) as u128 ;
342
+ let count =
343
+ niche_variants. end ( ) . index ( ) as u128 - niche_variants. start ( ) . index ( ) as u128 ;
331
344
332
345
// Find the field with the largest niche
333
346
let ( field_index, niche, ( niche_start, niche_scalar) ) = variants[ largest_variant_index]
@@ -660,7 +673,7 @@ pub trait LayoutCalculator {
660
673
// Common prim might be uninit.
661
674
Scalar :: Union { value : prim }
662
675
} ;
663
- let pair = self . scalar_pair ( tag, prim_scalar) ;
676
+ let pair = self . scalar_pair :: < FieldIdx , VariantIdx > ( tag, prim_scalar) ;
664
677
let pair_offsets = match pair. fields {
665
678
FieldsShape :: Arbitrary { ref offsets, ref memory_index } => {
666
679
assert_eq ! ( memory_index. raw, [ 0 , 1 ] ) ;
@@ -726,7 +739,7 @@ pub trait LayoutCalculator {
726
739
// pick the layout with the larger niche; otherwise,
727
740
// pick tagged as it has simpler codegen.
728
741
use cmp:: Ordering :: * ;
729
- let niche_size = |tmp_l : & TmpLayout < FieldIdx > | {
742
+ let niche_size = |tmp_l : & TmpLayout < FieldIdx , VariantIdx > | {
730
743
tmp_l. layout . largest_niche . map_or ( 0 , |n| n. available ( dl) )
731
744
} ;
732
745
match ( tl. layout . size . cmp ( & nl. layout . size ) , niche_size ( & tl) . cmp ( & niche_size ( & nl) ) ) {
@@ -748,11 +761,16 @@ pub trait LayoutCalculator {
748
761
Some ( best_layout. layout )
749
762
}
750
763
751
- fn layout_of_union < ' a , FieldIdx : Idx , F : Deref < Target = & ' a LayoutS < FieldIdx > > + fmt:: Debug > (
764
+ fn layout_of_union <
765
+ ' a ,
766
+ FieldIdx : Idx ,
767
+ VariantIdx : Idx ,
768
+ F : Deref < Target = & ' a LayoutS < FieldIdx , VariantIdx > > + fmt:: Debug ,
769
+ > (
752
770
& self ,
753
771
repr : & ReprOptions ,
754
772
variants : & IndexSlice < VariantIdx , IndexVec < FieldIdx , F > > ,
755
- ) -> Option < LayoutS < FieldIdx > > {
773
+ ) -> Option < LayoutS < FieldIdx , VariantIdx > > {
756
774
let dl = self . current_data_layout ( ) ;
757
775
let dl = dl. borrow ( ) ;
758
776
let mut align = if repr. pack . is_some ( ) { dl. i8_align } else { dl. aggregate_align } ;
@@ -769,7 +787,7 @@ pub trait LayoutCalculator {
769
787
} ;
770
788
771
789
let mut size = Size :: ZERO ;
772
- let only_variant = & variants[ FIRST_VARIANT ] ;
790
+ let only_variant = & variants[ VariantIdx :: new ( 0 ) ] ;
773
791
for field in only_variant {
774
792
if field. is_unsized ( ) {
775
793
self . delay_bug ( "unsized field in union" . to_string ( ) ) ;
@@ -836,7 +854,7 @@ pub trait LayoutCalculator {
836
854
} ;
837
855
838
856
Some ( LayoutS {
839
- variants : Variants :: Single { index : FIRST_VARIANT } ,
857
+ variants : Variants :: Single { index : VariantIdx :: new ( 0 ) } ,
840
858
fields : FieldsShape :: Union ( NonZeroUsize :: new ( only_variant. len ( ) ) ?) ,
841
859
abi,
842
860
largest_niche : None ,
@@ -854,14 +872,19 @@ enum NicheBias {
854
872
End ,
855
873
}
856
874
857
- fn univariant < ' a , FieldIdx : Idx , F : Deref < Target = & ' a LayoutS < FieldIdx > > + fmt:: Debug > (
875
+ fn univariant <
876
+ ' a ,
877
+ FieldIdx : Idx ,
878
+ VariantIdx : Idx ,
879
+ F : Deref < Target = & ' a LayoutS < FieldIdx , VariantIdx > > + fmt:: Debug ,
880
+ > (
858
881
this : & ( impl LayoutCalculator + ?Sized ) ,
859
882
dl : & TargetDataLayout ,
860
883
fields : & IndexSlice < FieldIdx , F > ,
861
884
repr : & ReprOptions ,
862
885
kind : StructKind ,
863
886
niche_bias : NicheBias ,
864
- ) -> Option < LayoutS < FieldIdx > > {
887
+ ) -> Option < LayoutS < FieldIdx , VariantIdx > > {
865
888
let pack = repr. pack ;
866
889
let mut align = if pack. is_some ( ) { dl. i8_align } else { dl. aggregate_align } ;
867
890
let mut max_repr_align = repr. align ;
@@ -1120,7 +1143,7 @@ fn univariant<'a, FieldIdx: Idx, F: Deref<Target = &'a LayoutS<FieldIdx>> + fmt:
1120
1143
} else {
1121
1144
( ( j, b) , ( i, a) )
1122
1145
} ;
1123
- let pair = this. scalar_pair ( a, b) ;
1146
+ let pair = this. scalar_pair :: < FieldIdx , VariantIdx > ( a, b) ;
1124
1147
let pair_offsets = match pair. fields {
1125
1148
FieldsShape :: Arbitrary { ref offsets, ref memory_index } => {
1126
1149
assert_eq ! ( memory_index. raw, [ 0 , 1 ] ) ;
@@ -1162,7 +1185,7 @@ fn univariant<'a, FieldIdx: Idx, F: Deref<Target = &'a LayoutS<FieldIdx>> + fmt:
1162
1185
} ;
1163
1186
1164
1187
Some ( LayoutS {
1165
- variants : Variants :: Single { index : FIRST_VARIANT } ,
1188
+ variants : Variants :: Single { index : VariantIdx :: new ( 0 ) } ,
1166
1189
fields : FieldsShape :: Arbitrary { offsets, memory_index } ,
1167
1190
abi,
1168
1191
largest_niche,
@@ -1173,8 +1196,13 @@ fn univariant<'a, FieldIdx: Idx, F: Deref<Target = &'a LayoutS<FieldIdx>> + fmt:
1173
1196
} )
1174
1197
}
1175
1198
1176
- fn format_field_niches < ' a , FieldIdx : Idx , F : Deref < Target = & ' a LayoutS < FieldIdx > > + fmt:: Debug > (
1177
- layout : & LayoutS < FieldIdx > ,
1199
+ fn format_field_niches <
1200
+ ' a ,
1201
+ FieldIdx : Idx ,
1202
+ VariantIdx : Idx ,
1203
+ F : Deref < Target = & ' a LayoutS < FieldIdx , VariantIdx > > + fmt:: Debug ,
1204
+ > (
1205
+ layout : & LayoutS < FieldIdx , VariantIdx > ,
1178
1206
fields : & IndexSlice < FieldIdx , F > ,
1179
1207
dl : & TargetDataLayout ,
1180
1208
) -> String {
0 commit comments