@@ -155,10 +155,6 @@ fn find_path_for_module(
155
155
module_id : ModuleId ,
156
156
max_len : usize ,
157
157
) -> Option < ( ModPath , Stability ) > {
158
- if max_len == 0 {
159
- return None ;
160
- }
161
-
162
158
if let Some ( crate_root) = module_id. as_crate_root ( ) {
163
159
if crate_root == ctx. from . derive_crate_root ( ) {
164
160
// - if the item is the crate root, return `crate`
@@ -314,6 +310,8 @@ fn calculate_best_path(
314
310
max_len : usize ,
315
311
) -> Option < ( ModPath , Stability ) > {
316
312
if max_len <= 1 {
313
+ // recursive base case, we can't find a path prefix of length 0, one segment is occupied by
314
+ // the item's name itself.
317
315
return None ;
318
316
}
319
317
@@ -341,18 +339,18 @@ fn calculate_best_path(
341
339
// Item was defined in the same crate that wants to import it. It cannot be found in any
342
340
// dependency in this case.
343
341
// FIXME: cache the `find_local_import_locations` output?
344
- for ( module_id , name ) in find_local_import_locations ( db, item, ctx. from , ctx. from_def_map ) {
342
+ find_local_import_locations ( db, item, ctx. from , ctx. from_def_map , |name , module_id| {
345
343
if !visited_modules. insert ( module_id) {
346
- continue ;
344
+ return ;
347
345
}
348
346
// we are looking for paths of length up to best_path_len, any longer will make it be
349
347
// less optimal. The -1 is due to us pushing name onto it afterwards.
350
348
if let Some ( path) =
351
349
find_path_for_module ( ctx, visited_modules, module_id, best_path_len - 1 )
352
350
{
353
- process ( path, name, & mut best_path_len) ;
351
+ process ( path, name. clone ( ) , & mut best_path_len) ;
354
352
}
355
- }
353
+ } )
356
354
} else {
357
355
// Item was defined in some upstream crate. This means that it must be exported from one,
358
356
// too (unless we can't name it at all). It could *also* be (re)exported by the same crate
@@ -454,14 +452,14 @@ fn select_best_path(
454
452
}
455
453
}
456
454
457
- // FIXME: Remove allocations
458
455
/// Finds locations in `from.krate` from which `item` can be imported by `from`.
459
456
fn find_local_import_locations (
460
457
db : & dyn DefDatabase ,
461
458
item : ItemInNs ,
462
459
from : ModuleId ,
463
460
def_map : & DefMap ,
464
- ) -> Vec < ( ModuleId , Name ) > {
461
+ mut cb : impl FnMut ( & Name , ModuleId ) ,
462
+ ) {
465
463
let _p = tracing:: span!( tracing:: Level :: INFO , "find_local_import_locations" ) . entered ( ) ;
466
464
467
465
// `from` can import anything below `from` with visibility of at least `from`, and anything
@@ -470,19 +468,17 @@ fn find_local_import_locations(
470
468
471
469
// Compute the initial worklist. We start with all direct child modules of `from` as well as all
472
470
// of its (recursive) parent modules.
473
- let data = & def_map[ from. local_id ] ;
474
- let mut worklist =
475
- data. children . values ( ) . map ( |child| def_map. module_id ( * child) ) . collect :: < Vec < _ > > ( ) ;
476
- // FIXME: do we need to traverse out of block expressions here?
477
- for ancestor in iter:: successors ( from. containing_module ( db) , |m| m. containing_module ( db) ) {
478
- worklist. push ( ancestor) ;
479
- }
471
+ let mut worklist = def_map[ from. local_id ]
472
+ . children
473
+ . values ( )
474
+ . map ( |child| def_map. module_id ( * child) )
475
+ // FIXME: do we need to traverse out of block expressions here?
476
+ . chain ( iter:: successors ( from. containing_module ( db) , |m| m. containing_module ( db) ) )
477
+ . collect :: < Vec < _ > > ( ) ;
478
+ let mut seen: FxHashSet < _ > = FxHashSet :: default ( ) ;
480
479
481
480
let def_map = def_map. crate_root ( ) . def_map ( db) ;
482
481
483
- let mut seen: FxHashSet < _ > = FxHashSet :: default ( ) ;
484
-
485
- let mut locations = Vec :: new ( ) ;
486
482
while let Some ( module) = worklist. pop ( ) {
487
483
if !seen. insert ( module) {
488
484
continue ; // already processed this module
@@ -523,7 +519,7 @@ fn find_local_import_locations(
523
519
// the item and we're a submodule of it, so can we.
524
520
// Also this keeps the cached data smaller.
525
521
if declared || is_pub_or_explicit {
526
- locations . push ( ( module , name . clone ( ) ) ) ;
522
+ cb ( name , module ) ;
527
523
}
528
524
}
529
525
}
@@ -535,8 +531,6 @@ fn find_local_import_locations(
535
531
}
536
532
}
537
533
}
538
-
539
- locations
540
534
}
541
535
542
536
#[ cfg( test) ]
0 commit comments