1
+ use crate :: un_derefer:: UnDerefer ;
1
2
use rustc_index:: vec:: IndexVec ;
2
3
use rustc_middle:: mir:: tcx:: RvalueInitializationState ;
3
4
use rustc_middle:: mir:: * ;
@@ -19,6 +20,7 @@ struct MoveDataBuilder<'a, 'tcx> {
19
20
param_env : ty:: ParamEnv < ' tcx > ,
20
21
data : MoveData < ' tcx > ,
21
22
errors : Vec < ( Place < ' tcx > , MoveError < ' tcx > ) > ,
23
+ un_derefer : UnDerefer < ' tcx > ,
22
24
}
23
25
24
26
impl < ' a , ' tcx > MoveDataBuilder < ' a , ' tcx > {
@@ -32,6 +34,7 @@ impl<'a, 'tcx> MoveDataBuilder<'a, 'tcx> {
32
34
tcx,
33
35
param_env,
34
36
errors : Vec :: new ( ) ,
37
+ un_derefer : UnDerefer { tcx : tcx, derefer_sidetable : Default :: default ( ) } ,
35
38
data : MoveData {
36
39
moves : IndexVec :: new ( ) ,
37
40
loc_map : LocationMap :: new ( body) ,
@@ -94,6 +97,11 @@ impl<'b, 'a, 'tcx> Gatherer<'b, 'a, 'tcx> {
94
97
///
95
98
/// Maybe we should have separate "borrowck" and "moveck" modes.
96
99
fn move_path_for ( & mut self , place : Place < ' tcx > ) -> Result < MovePathIndex , MoveError < ' tcx > > {
100
+ if let Some ( new_place) = self . builder . un_derefer . derefer ( place. as_ref ( ) , self . builder . body )
101
+ {
102
+ return self . move_path_for ( new_place) ;
103
+ }
104
+
97
105
debug ! ( "lookup({:?})" , place) ;
98
106
let mut base = self . builder . data . rev_lookup . locals [ place. local ] ;
99
107
@@ -276,6 +284,12 @@ struct Gatherer<'b, 'a, 'tcx> {
276
284
impl < ' b , ' a , ' tcx > Gatherer < ' b , ' a , ' tcx > {
277
285
fn gather_statement ( & mut self , stmt : & Statement < ' tcx > ) {
278
286
match & stmt. kind {
287
+ StatementKind :: Assign ( box ( place, Rvalue :: CopyForDeref ( reffed) ) ) => {
288
+ assert ! ( place. projection. is_empty( ) ) ;
289
+ if self . builder . body . local_decls [ place. local ] . is_deref_temp ( ) {
290
+ self . builder . un_derefer . derefer_sidetable . insert ( place. local , * reffed) ;
291
+ }
292
+ }
279
293
StatementKind :: Assign ( box ( place, rval) ) => {
280
294
self . create_move_path ( * place) ;
281
295
if let RvalueInitializationState :: Shallow = rval. initialization_state ( ) {
@@ -294,7 +308,10 @@ impl<'b, 'a, 'tcx> Gatherer<'b, 'a, 'tcx> {
294
308
}
295
309
StatementKind :: StorageLive ( _) => { }
296
310
StatementKind :: StorageDead ( local) => {
297
- self . gather_move ( Place :: from ( * local) ) ;
311
+ // DerefTemp locals (results of CopyForDeref) don't actually move anything.
312
+ if !self . builder . un_derefer . derefer_sidetable . contains_key ( & local) {
313
+ self . gather_move ( Place :: from ( * local) ) ;
314
+ }
298
315
}
299
316
StatementKind :: SetDiscriminant { .. } | StatementKind :: Deinit ( ..) => {
300
317
span_bug ! (
@@ -328,6 +345,7 @@ impl<'b, 'a, 'tcx> Gatherer<'b, 'a, 'tcx> {
328
345
self . gather_operand ( operand) ;
329
346
}
330
347
}
348
+ Rvalue :: CopyForDeref ( ..) => unreachable ! ( ) ,
331
349
Rvalue :: Ref ( ..)
332
350
| Rvalue :: AddressOf ( ..)
333
351
| Rvalue :: Discriminant ( ..)
@@ -439,6 +457,11 @@ impl<'b, 'a, 'tcx> Gatherer<'b, 'a, 'tcx> {
439
457
440
458
fn gather_move ( & mut self , place : Place < ' tcx > ) {
441
459
debug ! ( "gather_move({:?}, {:?})" , self . loc, place) ;
460
+ if let Some ( new_place) = self . builder . un_derefer . derefer ( place. as_ref ( ) , self . builder . body )
461
+ {
462
+ self . gather_move ( new_place) ;
463
+ return ;
464
+ }
442
465
443
466
if let [ ref base @ .., ProjectionElem :: Subslice { from, to, from_end : false } ] =
444
467
* * place. projection
@@ -494,6 +517,11 @@ impl<'b, 'a, 'tcx> Gatherer<'b, 'a, 'tcx> {
494
517
fn gather_init ( & mut self , place : PlaceRef < ' tcx > , kind : InitKind ) {
495
518
debug ! ( "gather_init({:?}, {:?})" , self . loc, place) ;
496
519
520
+ if let Some ( new_place) = self . builder . un_derefer . derefer ( place, self . builder . body ) {
521
+ self . gather_init ( new_place. as_ref ( ) , kind) ;
522
+ return ;
523
+ }
524
+
497
525
let mut place = place;
498
526
499
527
// Check if we are assigning into a field of a union, if so, lookup the place
0 commit comments