Skip to content

Commit 153a2ba

Browse files
committed
Auto merge of #17457 - roife:remove-circle, r=Veykril
fix: ensure there are no cycles in the source_root_parent_map See #17409 We can view the connections between roots as a graph. The problem is that this graph may contain cycles, so when adding edges, it is necessary to check whether it will lead to a cycle. Since we ensure that each node has at most one outgoing edge (because each SourceRoot can have only one parent), we can use a disjoint-set to maintain the connectivity between nodes. If an edge’s two nodes belong to the same set, they are already connected. Additionally, this PR includes the following three changes: 1. Removed the workaround from #17409. 2. Added an optimization: If `map.contains_key(&SourceRootId(*root_id as u32))`, we can skip the current loop iteration since we have already found its parent. 3. Modified the inner loop to iterate in reverse order with `roots[..idx].iter().rev()` at line 319. This ensures that if we are looking for the parent of `a/b/c`, and both `a` and `a/b` meet the criteria, we will choose the longer match (`a/b`).
2 parents f18fe6c + 185971c commit 153a2ba

File tree

2 files changed

+63
-30
lines changed
  • src/tools/rust-analyzer/crates

2 files changed

+63
-30
lines changed

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

+63-21
Original file line numberDiff line numberDiff line change
@@ -285,28 +285,54 @@ 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-
roots
289-
.iter()
290-
.enumerate()
291-
.filter(|(_, (_, id))| self.local_filesets.contains(id))
292-
.filter_map(|(idx, (root, root_id))| {
293-
// We are interested in parents if they are also local source roots.
294-
// 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.
298-
roots[..idx].iter().find_map(|(root2, root2_id)| {
299-
if self.local_filesets.contains(root2_id)
300-
&& root.starts_with(root2)
301-
&& root_id != root2_id
302-
{
303-
return Some((root_id, root2_id));
288+
289+
let mut map = FxHashMap::default();
290+
291+
// See https://github.com/rust-lang/rust-analyzer/issues/17409
292+
//
293+
// We can view the connections between roots as a graph. The problem is
294+
// that this graph may contain cycles, so when adding edges, it is necessary
295+
// to check whether it will lead to a cycle.
296+
//
297+
// Since we ensure that each node has at most one outgoing edge (because
298+
// each SourceRoot can have only one parent), we can use a disjoint-set to
299+
// maintain the connectivity between nodes. If an edge’s two nodes belong
300+
// to the same set, they are already connected.
301+
let mut dsu = FxHashMap::default();
302+
fn find_parent(dsu: &mut FxHashMap<u64, u64>, id: u64) -> u64 {
303+
if let Some(&parent) = dsu.get(&id) {
304+
let parent = find_parent(dsu, parent);
305+
dsu.insert(id, parent);
306+
parent
307+
} else {
308+
id
309+
}
310+
}
311+
312+
for (idx, (root, root_id)) in roots.iter().enumerate() {
313+
if !self.local_filesets.contains(root_id)
314+
|| map.contains_key(&SourceRootId(*root_id as u32))
315+
{
316+
continue;
317+
}
318+
319+
for (root2, root2_id) in roots[..idx].iter().rev() {
320+
if self.local_filesets.contains(root2_id)
321+
&& root_id != root2_id
322+
&& root.starts_with(root2)
323+
{
324+
// check if the edge will create a cycle
325+
if find_parent(&mut dsu, *root_id) != find_parent(&mut dsu, *root2_id) {
326+
map.insert(SourceRootId(*root_id as u32), SourceRootId(*root2_id as u32));
327+
dsu.insert(*root_id, *root2_id);
304328
}
305-
None
306-
})
307-
})
308-
.map(|(&child, &parent)| (SourceRootId(child as u32), SourceRootId(parent as u32)))
309-
.collect()
329+
330+
break;
331+
}
332+
}
333+
}
334+
335+
map
310336
}
311337
}
312338

@@ -592,4 +618,20 @@ mod tests {
592618

593619
assert_eq!(vc, vec![(SourceRootId(1), SourceRootId(0)),])
594620
}
621+
622+
#[test]
623+
fn circular_reference() {
624+
let mut builder = FileSetConfigBuilder::default();
625+
builder.add_file_set(vec![
626+
VfsPath::new_virtual_path("/ROOT/def".to_owned()),
627+
VfsPath::new_virtual_path("/ROOT/def/abc/def".to_owned()),
628+
]);
629+
builder.add_file_set(vec![VfsPath::new_virtual_path("/ROOT/def/abc".to_owned())]);
630+
let fsc = builder.build();
631+
let src = SourceRootConfig { fsc, local_filesets: vec![0, 1] };
632+
let mut vc = src.source_root_parent_map().into_iter().collect::<Vec<_>>();
633+
vc.sort_by(|x, y| x.0 .0.cmp(&y.0 .0));
634+
635+
assert_eq!(vc, vec![(SourceRootId(1), SourceRootId(0)),])
636+
}
595637
}

src/tools/rust-analyzer/crates/rust-analyzer/src/config.rs

-9
Original file line numberDiff line numberDiff line change
@@ -2531,22 +2531,13 @@ macro_rules! _impl_for_config_data {
25312531
#[allow(non_snake_case)]
25322532
$vis fn $field(&self, source_root: Option<SourceRootId>) -> &$ty {
25332533
let mut par: Option<SourceRootId> = source_root;
2534-
let mut traversals = 0;
25352534
while let Some(source_root_id) = par {
25362535
par = self.source_root_parent_map.get(&source_root_id).copied();
25372536
if let Some((config, _)) = self.ratoml_files.get(&source_root_id) {
25382537
if let Some(value) = config.$field.as_ref() {
25392538
return value;
25402539
}
25412540
}
2542-
// Prevent infinite loops caused by cycles by giving up when it's
2543-
// clear that we must have either visited all source roots or
2544-
// encountered a cycle.
2545-
traversals += 1;
2546-
if traversals >= self.source_root_parent_map.len() {
2547-
// i.e. no source root contains the config we're looking for
2548-
break;
2549-
}
25502541
}
25512542

25522543
if let Some((root_path_ratoml, _)) = self.root_ratoml.as_ref() {

0 commit comments

Comments
 (0)