@@ -1071,13 +1071,13 @@ enum RibKind<'a> {
1071
1071
/// The resolution keeps a separate stack of ribs as it traverses the AST for each namespace. When
1072
1072
/// resolving, the name is looked up from inside out.
1073
1073
#[ derive( Debug ) ]
1074
- struct Rib < ' a > {
1075
- bindings : FxHashMap < Ident , Res > ,
1074
+ struct Rib < ' a , R = Res > {
1075
+ bindings : FxHashMap < Ident , R > ,
1076
1076
kind : RibKind < ' a > ,
1077
1077
}
1078
1078
1079
- impl < ' a > Rib < ' a > {
1080
- fn new ( kind : RibKind < ' a > ) -> Rib < ' a > {
1079
+ impl < ' a , R > Rib < ' a , R > {
1080
+ fn new ( kind : RibKind < ' a > ) -> Rib < ' a , R > {
1081
1081
Rib {
1082
1082
bindings : Default :: default ( ) ,
1083
1083
kind,
@@ -1638,7 +1638,7 @@ pub struct Resolver<'a> {
1638
1638
ribs : PerNS < Vec < Rib < ' a > > > ,
1639
1639
1640
1640
/// The current set of local scopes, for labels.
1641
- label_ribs : Vec < Rib < ' a > > ,
1641
+ label_ribs : Vec < Rib < ' a , NodeId > > ,
1642
1642
1643
1643
/// The trait that the current context can refer to.
1644
1644
current_trait_ref : Option < ( Module < ' a > , TraitRef ) > ,
@@ -1663,6 +1663,8 @@ pub struct Resolver<'a> {
1663
1663
partial_res_map : NodeMap < PartialRes > ,
1664
1664
/// Resolutions for import nodes, which have multiple resolutions in different namespaces.
1665
1665
import_res_map : NodeMap < PerNS < Option < Res > > > ,
1666
+ /// Resolutions for labels (node IDs of their corresponding blocks or loops).
1667
+ label_res_map : NodeMap < NodeId > ,
1666
1668
1667
1669
pub freevars : FreevarMap ,
1668
1670
freevars_seen : NodeMap < NodeMap < usize > > ,
@@ -1841,6 +1843,10 @@ impl<'a> hir::lowering::Resolver for Resolver<'a> {
1841
1843
self . import_res_map . get ( & id) . cloned ( ) . unwrap_or_default ( )
1842
1844
}
1843
1845
1846
+ fn get_label_res ( & mut self , id : NodeId ) -> Option < NodeId > {
1847
+ self . label_res_map . get ( & id) . cloned ( )
1848
+ }
1849
+
1844
1850
fn definitions ( & mut self ) -> & mut Definitions {
1845
1851
& mut self . definitions
1846
1852
}
@@ -2024,6 +2030,7 @@ impl<'a> Resolver<'a> {
2024
2030
2025
2031
partial_res_map : Default :: default ( ) ,
2026
2032
import_res_map : Default :: default ( ) ,
2033
+ label_res_map : Default :: default ( ) ,
2027
2034
freevars : Default :: default ( ) ,
2028
2035
freevars_seen : Default :: default ( ) ,
2029
2036
export_map : FxHashMap :: default ( ) ,
@@ -2490,7 +2497,7 @@ impl<'a> Resolver<'a> {
2490
2497
///
2491
2498
/// Stops after meeting a closure.
2492
2499
fn search_label < P , R > ( & self , mut ident : Ident , pred : P ) -> Option < R >
2493
- where P : Fn ( & Rib < ' _ > , Ident ) -> Option < R >
2500
+ where P : Fn ( & Rib < ' _ , NodeId > , Ident ) -> Option < R >
2494
2501
{
2495
2502
for rib in self . label_ribs . iter ( ) . rev ( ) {
2496
2503
match rib. kind {
@@ -4332,10 +4339,9 @@ impl<'a> Resolver<'a> {
4332
4339
{
4333
4340
if let Some ( label) = label {
4334
4341
self . unused_labels . insert ( id, label. ident . span ) ;
4335
- let res = Res :: Label ( id) ;
4336
4342
self . with_label_rib ( |this| {
4337
4343
let ident = label. ident . modern_and_legacy ( ) ;
4338
- this. label_ribs . last_mut ( ) . unwrap ( ) . bindings . insert ( ident, res ) ;
4344
+ this. label_ribs . last_mut ( ) . unwrap ( ) . bindings . insert ( ident, id ) ;
4339
4345
f ( this) ;
4340
4346
} ) ;
4341
4347
} else {
@@ -4366,10 +4372,10 @@ impl<'a> Resolver<'a> {
4366
4372
}
4367
4373
4368
4374
ExprKind :: Break ( Some ( label) , _) | ExprKind :: Continue ( Some ( label) ) => {
4369
- let res = self . search_label ( label. ident , |rib, ident| {
4375
+ let node_id = self . search_label ( label. ident , |rib, ident| {
4370
4376
rib. bindings . get ( & ident. modern_and_legacy ( ) ) . cloned ( )
4371
4377
} ) ;
4372
- match res {
4378
+ match node_id {
4373
4379
None => {
4374
4380
// Search again for close matches...
4375
4381
// Picks the first label that is "close enough", which is not necessarily
@@ -4390,13 +4396,10 @@ impl<'a> Resolver<'a> {
4390
4396
ResolutionError :: UndeclaredLabel ( & label. ident . as_str ( ) ,
4391
4397
close_match) ) ;
4392
4398
}
4393
- Some ( Res :: Label ( id ) ) => {
4399
+ Some ( node_id ) => {
4394
4400
// Since this res is a label, it is never read.
4395
- self . record_partial_res ( expr. id , PartialRes :: new ( Res :: Label ( id) ) ) ;
4396
- self . unused_labels . remove ( & id) ;
4397
- }
4398
- Some ( _) => {
4399
- span_bug ! ( expr. span, "label wasn't mapped to a label res!" ) ;
4401
+ self . label_res_map . insert ( expr. id , node_id) ;
4402
+ self . unused_labels . remove ( & node_id) ;
4400
4403
}
4401
4404
}
4402
4405
0 commit comments