@@ -6,13 +6,79 @@ use crate::build::{BlockAnd, BlockAndExtension, Builder};
6
6
use crate :: hair:: * ;
7
7
use rustc:: mir:: interpret:: { PanicInfo :: BoundsCheck } ;
8
8
use rustc:: mir:: * ;
9
- use rustc:: ty:: { CanonicalUserTypeAnnotation , Variance } ;
9
+ use rustc:: ty:: { CanonicalUserTypeAnnotation , Ty , Variance } ;
10
10
11
11
use rustc_data_structures:: indexed_vec:: Idx ;
12
12
13
+ /// `PlaceBuilder` is used to create places during MIR construction. It allows you to "build up" a
14
+ /// place by pushing more and more projections onto the end, and then convert the final set into a
15
+ /// place using the `into_place` method.
16
+ ///
17
+ /// This is used internally when building a place for an expression like `a.b.c`. The fields `b`
18
+ /// and `c` can be progressively pushed onto the place builder that is created when converting `a`.
19
+ #[ derive( Clone ) ]
20
+ struct PlaceBuilder < ' tcx > {
21
+ base : PlaceBase < ' tcx > ,
22
+ projection : Vec < PlaceElem < ' tcx > > ,
23
+ }
24
+
25
+ impl PlaceBuilder < ' tcx > {
26
+ fn into_place ( self ) -> Place < ' tcx > {
27
+ Place {
28
+ base : self . base ,
29
+ projection : self . projection . into_boxed_slice ( ) ,
30
+ }
31
+ }
32
+
33
+ fn field ( self , f : Field , ty : Ty < ' tcx > ) -> Self {
34
+ self . project ( PlaceElem :: Field ( f, ty) )
35
+ }
36
+
37
+ fn deref ( self ) -> Self {
38
+ self . project ( PlaceElem :: Deref )
39
+ }
40
+
41
+ fn index ( self , index : Local ) -> Self {
42
+ self . project ( PlaceElem :: Index ( index) )
43
+ }
44
+
45
+ fn project ( mut self , elem : PlaceElem < ' tcx > ) -> Self {
46
+ self . projection . push ( elem) ;
47
+ self
48
+ }
49
+ }
50
+
51
+ impl From < Local > for PlaceBuilder < ' tcx > {
52
+ fn from ( local : Local ) -> Self {
53
+ Self {
54
+ base : local. into ( ) ,
55
+ projection : Vec :: new ( ) ,
56
+ }
57
+ }
58
+ }
59
+
60
+ impl From < PlaceBase < ' tcx > > for PlaceBuilder < ' tcx > {
61
+ fn from ( base : PlaceBase < ' tcx > ) -> Self {
62
+ Self {
63
+ base,
64
+ projection : Vec :: new ( ) ,
65
+ }
66
+ }
67
+ }
68
+
13
69
impl < ' a , ' tcx > Builder < ' a , ' tcx > {
14
70
/// Compile `expr`, yielding a place that we can move from etc.
15
- pub fn as_place < M > ( & mut self , block : BasicBlock , expr : M ) -> BlockAnd < Place < ' tcx > >
71
+ pub fn as_place < M > ( & mut self , mut block : BasicBlock , expr : M ) -> BlockAnd < Place < ' tcx > >
72
+ where
73
+ M : Mirror < ' tcx , Output = Expr < ' tcx > > ,
74
+ {
75
+ let place_builder = unpack ! ( block = self . as_place_builder( block, expr) ) ;
76
+ block. and ( place_builder. into_place ( ) )
77
+ }
78
+
79
+ /// This is used when constructing a compound `Place`, so that we can avoid creating
80
+ /// intermediate `Place` values until we know the full set of projections.
81
+ fn as_place_builder < M > ( & mut self , block : BasicBlock , expr : M ) -> BlockAnd < PlaceBuilder < ' tcx > >
16
82
where
17
83
M : Mirror < ' tcx , Output = Expr < ' tcx > > ,
18
84
{
@@ -25,7 +91,25 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
25
91
/// place. The place itself may or may not be mutable:
26
92
/// * If this expr is a place expr like a.b, then we will return that place.
27
93
/// * Otherwise, a temporary is created: in that event, it will be an immutable temporary.
28
- pub fn as_read_only_place < M > ( & mut self , block : BasicBlock , expr : M ) -> BlockAnd < Place < ' tcx > >
94
+ pub fn as_read_only_place < M > ( & mut self , mut block : BasicBlock , expr : M ) -> BlockAnd < Place < ' tcx > >
95
+ where
96
+ M : Mirror < ' tcx , Output = Expr < ' tcx > > ,
97
+ {
98
+ let place_builder = unpack ! ( block = self . as_read_only_place_builder( block, expr) ) ;
99
+ block. and ( place_builder. into_place ( ) )
100
+ }
101
+
102
+ /// This is used when constructing a compound `Place`, so that we can avoid creating
103
+ /// intermediate `Place` values until we know the full set of projections.
104
+ /// Mutability note: The caller of this method promises only to read from the resulting
105
+ /// place. The place itself may or may not be mutable:
106
+ /// * If this expr is a place expr like a.b, then we will return that place.
107
+ /// * Otherwise, a temporary is created: in that event, it will be an immutable temporary.
108
+ fn as_read_only_place_builder < M > (
109
+ & mut self ,
110
+ block : BasicBlock ,
111
+ expr : M ,
112
+ ) -> BlockAnd < PlaceBuilder < ' tcx > >
29
113
where
30
114
M : Mirror < ' tcx , Output = Expr < ' tcx > > ,
31
115
{
@@ -38,7 +122,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
38
122
mut block : BasicBlock ,
39
123
expr : Expr < ' tcx > ,
40
124
mutability : Mutability ,
41
- ) -> BlockAnd < Place < ' tcx > > {
125
+ ) -> BlockAnd < PlaceBuilder < ' tcx > > {
42
126
debug ! (
43
127
"expr_as_place(block={:?}, expr={:?}, mutability={:?})" ,
44
128
block, expr, mutability
@@ -54,25 +138,23 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
54
138
value,
55
139
} => this. in_scope ( ( region_scope, source_info) , lint_level, |this| {
56
140
if mutability == Mutability :: Not {
57
- this. as_read_only_place ( block, value)
141
+ this. as_read_only_place_builder ( block, value)
58
142
} else {
59
- this. as_place ( block, value)
143
+ this. as_place_builder ( block, value)
60
144
}
61
145
} ) ,
62
146
ExprKind :: Field { lhs, name } => {
63
- let place = unpack ! ( block = this. as_place( block, lhs) ) ;
64
- let place = place. field ( name, expr. ty ) ;
65
- block. and ( place)
147
+ let place_builder = unpack ! ( block = this. as_place_builder( block, lhs) ) ;
148
+ block. and ( place_builder. field ( name, expr. ty ) )
66
149
}
67
150
ExprKind :: Deref { arg } => {
68
- let place = unpack ! ( block = this. as_place( block, arg) ) ;
69
- let place = place. deref ( ) ;
70
- block. and ( place)
151
+ let place_builder = unpack ! ( block = this. as_place_builder( block, arg) ) ;
152
+ block. and ( place_builder. deref ( ) )
71
153
}
72
154
ExprKind :: Index { lhs, index } => {
73
155
let ( usize_ty, bool_ty) = ( this. hir . usize_ty ( ) , this. hir . bool_ty ( ) ) ;
74
156
75
- let slice = unpack ! ( block = this. as_place ( block, lhs) ) ;
157
+ let place_builder = unpack ! ( block = this. as_place_builder ( block, lhs) ) ;
76
158
// Making this a *fresh* temporary also means we do not have to worry about
77
159
// the index changing later: Nothing will ever change this temporary.
78
160
// The "retagging" transformation (for Stacked Borrows) relies on this.
@@ -83,6 +165,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
83
165
Mutability :: Not ,
84
166
) ) ;
85
167
168
+ let slice = place_builder. clone ( ) . into_place ( ) ;
86
169
// bounds check:
87
170
let ( len, lt) = (
88
171
this. temp ( usize_ty. clone ( ) , expr_span) ,
@@ -92,7 +175,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
92
175
block,
93
176
source_info, // len = len(slice)
94
177
& len,
95
- Rvalue :: Len ( slice. clone ( ) ) ,
178
+ Rvalue :: Len ( slice) ,
96
179
) ;
97
180
this. cfg . push_assign (
98
181
block,
@@ -110,30 +193,29 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
110
193
index : Operand :: Copy ( Place :: from ( idx) ) ,
111
194
} ;
112
195
let success = this. assert ( block, Operand :: Move ( lt) , true , msg, expr_span) ;
113
- success. and ( slice . index ( idx) )
196
+ success. and ( place_builder . index ( idx) )
114
197
}
115
- ExprKind :: SelfRef => block. and ( Place :: from ( Local :: new ( 1 ) ) ) ,
198
+ ExprKind :: SelfRef => block. and ( PlaceBuilder :: from ( Local :: new ( 1 ) ) ) ,
116
199
ExprKind :: VarRef { id } => {
117
- let place = if this. is_bound_var_in_guard ( id) {
200
+ let place_builder = if this. is_bound_var_in_guard ( id) {
118
201
let index = this. var_local_id ( id, RefWithinGuard ) ;
119
- Place :: from ( index) . deref ( )
202
+ PlaceBuilder :: from ( index) . deref ( )
120
203
} else {
121
204
let index = this. var_local_id ( id, OutsideGuard ) ;
122
- Place :: from ( index)
205
+ PlaceBuilder :: from ( index)
123
206
} ;
124
- block. and ( place )
207
+ block. and ( place_builder )
125
208
}
126
- ExprKind :: StaticRef { id } => block. and ( Place {
127
- base : PlaceBase :: Static ( Box :: new ( Static {
209
+ ExprKind :: StaticRef { id } => block. and ( PlaceBuilder :: from (
210
+ PlaceBase :: Static ( Box :: new ( Static {
128
211
ty : expr. ty ,
129
212
kind : StaticKind :: Static ,
130
213
def_id : id,
131
- } ) ) ,
132
- projection : box [ ] ,
133
- } ) ,
214
+ } ) )
215
+ ) ) ,
134
216
135
217
ExprKind :: PlaceTypeAscription { source, user_ty } => {
136
- let place = unpack ! ( block = this. as_place ( block, source) ) ;
218
+ let place_builder = unpack ! ( block = this. as_place_builder ( block, source) ) ;
137
219
if let Some ( user_ty) = user_ty {
138
220
let annotation_index = this. canonical_user_type_annotations . push (
139
221
CanonicalUserTypeAnnotation {
@@ -142,21 +224,23 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
142
224
inferred_ty : expr. ty ,
143
225
}
144
226
) ;
227
+
228
+ let place = place_builder. clone ( ) . into_place ( ) ;
145
229
this. cfg . push (
146
230
block,
147
231
Statement {
148
232
source_info,
149
233
kind : StatementKind :: AscribeUserType (
150
234
box (
151
- place. clone ( ) ,
235
+ place,
152
236
UserTypeProjection { base : annotation_index, projs : vec ! [ ] , }
153
237
) ,
154
238
Variance :: Invariant ,
155
239
) ,
156
240
} ,
157
241
) ;
158
242
}
159
- block. and ( place )
243
+ block. and ( place_builder )
160
244
}
161
245
ExprKind :: ValueTypeAscription { source, user_ty } => {
162
246
let source = this. hir . mirror ( source) ;
@@ -185,7 +269,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
185
269
} ,
186
270
) ;
187
271
}
188
- block. and ( Place :: from ( temp) )
272
+ block. and ( PlaceBuilder :: from ( temp) )
189
273
}
190
274
191
275
ExprKind :: Array { .. }
@@ -221,7 +305,7 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
221
305
} ) ;
222
306
let temp =
223
307
unpack ! ( block = this. as_temp( block, expr. temp_lifetime, expr, mutability) ) ;
224
- block. and ( Place :: from ( temp) )
308
+ block. and ( PlaceBuilder :: from ( temp) )
225
309
}
226
310
}
227
311
}
0 commit comments