Skip to content

Commit c995190

Browse files
committed
Auto merge of #17381 - roife:fix-issue-17378, r=Veykril
fix: ensure that the parent of a SourceRoot cannot be itself fix #17378. In `FileSetConfig.map`, different roots might be mapped to the same `root_id` due to deduplication in `ProjectFolders::new`: ```rust // Example from rustup /Users/roife/code/rustup/target/debug/build/rustup-863a063426b56c51/out /Users/roife/code/rustup ``` In `source_root_parent_map`, r-a might encounter paths where their SourceRootId (i.e. `root_id`) is identical, yet one the them is the parent of the another. This situation can cause the `root_id` to be its own parent, potentially leading to an infinite loop. This PR resolves such cases by adding a check.
2 parents 8fd1b50 + 5aa6137 commit c995190

File tree

1 file changed

+23
-3
lines changed
  • src/tools/rust-analyzer/crates/load-cargo/src

1 file changed

+23
-3
lines changed

src/tools/rust-analyzer/crates/load-cargo/src/lib.rs

+23-3
Original file line numberDiff line numberDiff line change
@@ -285,17 +285,21 @@ impl SourceRootConfig {
285285
/// If a `SourceRoot` doesn't have a parent and is local then it is not contained in this mapping but it can be asserted that it is a root `SourceRoot`.
286286
pub fn source_root_parent_map(&self) -> FxHashMap<SourceRootId, SourceRootId> {
287287
let roots = self.fsc.roots();
288-
let mut i = 0;
289288
roots
290289
.iter()
291290
.enumerate()
292291
.filter(|(_, (_, id))| self.local_filesets.contains(id))
293292
.filter_map(|(idx, (root, root_id))| {
294293
// We are interested in parents if they are also local source roots.
295294
// So instead of a non-local parent we may take a local ancestor as a parent to a node.
295+
//
296+
// Here paths in roots are sorted lexicographically, so if a root
297+
// is a parent of another root, it will be before it in the list.
296298
roots[..idx].iter().find_map(|(root2, root2_id)| {
297-
i += 1;
298-
if self.local_filesets.contains(root2_id) && root.starts_with(root2) {
299+
if self.local_filesets.contains(root2_id)
300+
&& root.starts_with(root2)
301+
&& root_id != root2_id
302+
{
299303
return Some((root_id, root2_id));
300304
}
301305
None
@@ -572,4 +576,20 @@ mod tests {
572576

573577
assert_eq!(vc, vec![(SourceRootId(3), SourceRootId(1)),])
574578
}
579+
580+
#[test]
581+
fn parents_with_identical_root_id() {
582+
let mut builder = FileSetConfigBuilder::default();
583+
builder.add_file_set(vec![
584+
VfsPath::new_virtual_path("/ROOT/def".to_owned()),
585+
VfsPath::new_virtual_path("/ROOT/def/abc/def".to_owned()),
586+
]);
587+
builder.add_file_set(vec![VfsPath::new_virtual_path("/ROOT/def/abc/def/ghi".to_owned())]);
588+
let fsc = builder.build();
589+
let src = SourceRootConfig { fsc, local_filesets: vec![0, 1] };
590+
let mut vc = src.source_root_parent_map().into_iter().collect::<Vec<_>>();
591+
vc.sort_by(|x, y| x.0 .0.cmp(&y.0 .0));
592+
593+
assert_eq!(vc, vec![(SourceRootId(1), SourceRootId(0)),])
594+
}
575595
}

0 commit comments

Comments
 (0)