@@ -1100,12 +1100,12 @@ impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
1100
1100
}
1101
1101
let decl_span = local_decl. source_info . span ;
1102
1102
1103
- let label = match * local_decl. local_info ( ) {
1103
+ let amp_mut_sugg = match * local_decl. local_info ( ) {
1104
1104
LocalInfo :: User ( mir:: BindingForm :: ImplicitSelf ( _) ) => {
1105
1105
let suggestion = suggest_ampmut_self ( self . infcx . tcx , decl_span) ;
1106
1106
let additional =
1107
1107
local_trait. map ( |span| ( span, suggest_ampmut_self ( self . infcx . tcx , span) ) ) ;
1108
- Some ( ( true , decl_span, suggestion, additional) )
1108
+ Some ( AmpMutSugg { has_sugg : true , span : decl_span, suggestion, additional } )
1109
1109
}
1110
1110
1111
1111
LocalInfo :: User ( mir:: BindingForm :: Var ( mir:: VarBindingForm {
@@ -1150,7 +1150,7 @@ impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
1150
1150
None
1151
1151
}
1152
1152
None => {
1153
- let ( has_sugg , decl_span , sugg ) = if name != kw:: SelfLower {
1153
+ if name != kw:: SelfLower {
1154
1154
suggest_ampmut (
1155
1155
self . infcx . tcx ,
1156
1156
local_decl. ty ,
@@ -1165,7 +1165,12 @@ impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
1165
1165
..
1166
1166
} ) ) => {
1167
1167
let sugg = suggest_ampmut_self ( self . infcx . tcx , decl_span) ;
1168
- ( true , decl_span, sugg)
1168
+ Some ( AmpMutSugg {
1169
+ has_sugg : true ,
1170
+ span : decl_span,
1171
+ suggestion : sugg,
1172
+ additional : None ,
1173
+ } )
1169
1174
}
1170
1175
// explicit self (eg `self: &'a Self`)
1171
1176
_ => suggest_ampmut (
@@ -1176,8 +1181,7 @@ impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
1176
1181
opt_ty_info,
1177
1182
) ,
1178
1183
}
1179
- } ;
1180
- Some ( ( has_sugg, decl_span, sugg, None ) )
1184
+ }
1181
1185
}
1182
1186
}
1183
1187
}
@@ -1187,15 +1191,24 @@ impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
1187
1191
..
1188
1192
} ) ) => {
1189
1193
let pattern_span: Span = local_decl. source_info . span ;
1190
- suggest_ref_mut ( self . infcx . tcx , pattern_span)
1191
- . map ( |span| ( true , span, "mut " . to_owned ( ) , None ) )
1194
+ suggest_ref_mut ( self . infcx . tcx , pattern_span) . map ( |span| AmpMutSugg {
1195
+ has_sugg : true ,
1196
+ span,
1197
+ suggestion : "mut " . to_owned ( ) ,
1198
+ additional : None ,
1199
+ } )
1192
1200
}
1193
1201
1194
1202
_ => unreachable ! ( ) ,
1195
1203
} ;
1196
1204
1197
- match label {
1198
- Some ( ( true , err_help_span, suggested_code, additional) ) => {
1205
+ match amp_mut_sugg {
1206
+ Some ( AmpMutSugg {
1207
+ has_sugg : true ,
1208
+ span : err_help_span,
1209
+ suggestion : suggested_code,
1210
+ additional,
1211
+ } ) => {
1199
1212
let mut sugg = vec ! [ ( err_help_span, suggested_code) ] ;
1200
1213
if let Some ( s) = additional {
1201
1214
sugg. push ( s) ;
@@ -1217,7 +1230,9 @@ impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
1217
1230
) ;
1218
1231
}
1219
1232
}
1220
- Some ( ( false , err_label_span, message, _) ) => {
1233
+ Some ( AmpMutSugg {
1234
+ has_sugg : false , span : err_label_span, suggestion : message, ..
1235
+ } ) => {
1221
1236
let def_id = self . body . source . def_id ( ) ;
1222
1237
let hir_id = if let Some ( local_def_id) = def_id. as_local ( )
1223
1238
&& let Some ( body) = self . infcx . tcx . hir ( ) . maybe_body_owned_by ( local_def_id)
@@ -1422,6 +1437,13 @@ fn suggest_ampmut_self<'tcx>(tcx: TyCtxt<'tcx>, span: Span) -> String {
1422
1437
}
1423
1438
}
1424
1439
1440
+ struct AmpMutSugg {
1441
+ has_sugg : bool ,
1442
+ span : Span ,
1443
+ suggestion : String ,
1444
+ additional : Option < ( Span , String ) > ,
1445
+ }
1446
+
1425
1447
// When we want to suggest a user change a local variable to be a `&mut`, there
1426
1448
// are three potential "obvious" things to highlight:
1427
1449
//
@@ -1443,7 +1465,7 @@ fn suggest_ampmut<'tcx>(
1443
1465
decl_span : Span ,
1444
1466
opt_assignment_rhs_span : Option < Span > ,
1445
1467
opt_ty_info : Option < Span > ,
1446
- ) -> ( bool , Span , String ) {
1468
+ ) -> Option < AmpMutSugg > {
1447
1469
// if there is a RHS and it starts with a `&` from it, then check if it is
1448
1470
// mutable, and if not, put suggest putting `mut ` to make it mutable.
1449
1471
// we don't have to worry about lifetime annotations here because they are
@@ -1456,6 +1478,11 @@ fn suggest_ampmut<'tcx>(
1456
1478
&& let Ok ( src) = tcx. sess . source_map ( ) . span_to_snippet ( assignment_rhs_span)
1457
1479
&& let Some ( stripped) = src. strip_prefix ( '&' )
1458
1480
{
1481
+ let is_raw_ref = stripped. trim_start ( ) . starts_with ( "raw " ) ;
1482
+ // We don't support raw refs yet
1483
+ if is_raw_ref {
1484
+ return None ;
1485
+ }
1459
1486
let is_mut = if let Some ( rest) = stripped. trim_start ( ) . strip_prefix ( "mut" ) {
1460
1487
match rest. chars ( ) . next ( ) {
1461
1488
// e.g. `&mut x`
@@ -1479,7 +1506,12 @@ fn suggest_ampmut<'tcx>(
1479
1506
1480
1507
// FIXME(Ezrashaw): returning is bad because we still might want to
1481
1508
// update the annotated type, see #106857.
1482
- return ( true , span, "mut " . to_owned ( ) ) ;
1509
+ return Some ( AmpMutSugg {
1510
+ has_sugg : true ,
1511
+ span,
1512
+ suggestion : "mut " . to_owned ( ) ,
1513
+ additional : None ,
1514
+ } ) ;
1483
1515
}
1484
1516
}
1485
1517
@@ -1504,18 +1536,23 @@ fn suggest_ampmut<'tcx>(
1504
1536
&& let Some ( ws_pos) = src. find ( char:: is_whitespace)
1505
1537
{
1506
1538
let span = span. with_lo ( span. lo ( ) + BytePos ( ws_pos as u32 ) ) . shrink_to_lo ( ) ;
1507
- ( true , span, " mut" . to_owned ( ) )
1539
+ Some ( AmpMutSugg { has_sugg : true , span, suggestion : " mut" . to_owned ( ) , additional : None } )
1508
1540
// if there is already a binding, we modify it to be `mut`
1509
1541
} else if binding_exists {
1510
1542
// shrink the span to just after the `&` in `&variable`
1511
1543
let span = span. with_lo ( span. lo ( ) + BytePos ( 1 ) ) . shrink_to_lo ( ) ;
1512
- ( true , span, "mut " . to_owned ( ) )
1544
+ Some ( AmpMutSugg { has_sugg : true , span, suggestion : "mut " . to_owned ( ) , additional : None } )
1513
1545
} else {
1514
1546
// otherwise, suggest that the user annotates the binding; we provide the
1515
1547
// type of the local.
1516
1548
let ty = decl_ty. builtin_deref ( true ) . unwrap ( ) ;
1517
1549
1518
- ( false , span, format ! ( "{}mut {}" , if decl_ty. is_ref( ) { "&" } else { "*" } , ty) )
1550
+ Some ( AmpMutSugg {
1551
+ has_sugg : false ,
1552
+ span,
1553
+ suggestion : format ! ( "{}mut {}" , if decl_ty. is_ref( ) { "&" } else { "*" } , ty) ,
1554
+ additional : None ,
1555
+ } )
1519
1556
}
1520
1557
}
1521
1558
0 commit comments