From 2e8872cc3f2985dc2241d9226cdb9fef29ce317e Mon Sep 17 00:00:00 2001 From: Boshen <1430279+Boshen@users.noreply.github.com> Date: Fri, 20 Dec 2024 15:18:16 +0000 Subject: [PATCH] perf(semantic): allocate child scope in allocator (#8045) --- crates/oxc_semantic/src/scope.rs | 82 +++++++++++++--------- crates/oxc_traverse/src/context/scoping.rs | 3 +- 2 files changed, 49 insertions(+), 36 deletions(-) diff --git a/crates/oxc_semantic/src/scope.rs b/crates/oxc_semantic/src/scope.rs index 0b6144e41c3a8..86e47d9c36e04 100644 --- a/crates/oxc_semantic/src/scope.rs +++ b/crates/oxc_semantic/src/scope.rs @@ -2,8 +2,8 @@ use std::{fmt, mem}; use rustc_hash::{FxBuildHasher, FxHashMap}; -use oxc_allocator::Allocator; -use oxc_index::IndexVec; +use oxc_allocator::{Allocator, Vec as ArenaVec}; +use oxc_index::{Idx, IndexVec}; use oxc_span::CompactStr; use oxc_syntax::{ node::NodeId, @@ -29,9 +29,6 @@ pub struct ScopeTree { /// Maps a scope to the parent scope it belongs in. parent_ids: IndexVec>, - /// Maps a scope to direct children scopes. - child_ids: IndexVec>, - /// Runtime flag for constructing child_ids. pub(crate) build_child_ids: bool, @@ -55,13 +52,13 @@ impl Default for ScopeTree { fn default() -> Self { Self { parent_ids: IndexVec::new(), - child_ids: IndexVec::new(), build_child_ids: false, node_ids: IndexVec::new(), flags: IndexVec::new(), root_unresolved_references: UnresolvedReferences::default(), - cell: ScopeTreeCell::new(Allocator::default(), |_bump| ScopeTreeInner { + cell: ScopeTreeCell::new(Allocator::default(), |allocator| ScopeTreeInner { bindings: IndexVec::new(), + child_ids: ArenaVec::new_in(allocator), }), } } @@ -80,6 +77,9 @@ pub(crate) struct ScopeTreeInner<'cell> { /// /// A binding is a mapping from an identifier name to its [`SymbolId`] pub(crate) bindings: IndexVec>, + + /// Maps a scope to direct children scopes. + child_ids: ArenaVec<'cell, ArenaVec<'cell, ScopeId>>, } impl ScopeTree { @@ -189,7 +189,9 @@ impl ScopeTree { if self.build_child_ids { // Set this scope as child of parent scope if let Some(parent_id) = parent_id { - self.child_ids[parent_id].push(scope_id); + self.cell.with_dependent_mut(|_allocator, inner| { + inner.child_ids[parent_id.index()].push(scope_id); + }); } } } @@ -200,25 +202,29 @@ impl ScopeTree { pub fn change_parent_id(&mut self, scope_id: ScopeId, new_parent_id: Option) { let old_parent_id = mem::replace(&mut self.parent_ids[scope_id], new_parent_id); if self.build_child_ids { - // Remove this scope from old parent scope - if let Some(old_parent_id) = old_parent_id { - self.child_ids[old_parent_id].retain(|&child_id| child_id != scope_id); - } - // And add it to new parent scope - if let Some(parent_id) = new_parent_id { - self.child_ids[parent_id].push(scope_id); - } + self.cell.with_dependent_mut(|_allocator, inner| { + // Remove this scope from old parent scope + if let Some(old_parent_id) = old_parent_id { + inner.child_ids[old_parent_id.index()].retain(|&child_id| child_id != scope_id); + } + // And add it to new parent scope + if let Some(parent_id) = new_parent_id { + inner.child_ids[parent_id.index()].push(scope_id); + } + }); } } /// Delete a scope. pub fn delete_scope(&mut self, scope_id: ScopeId) { if self.build_child_ids { - self.child_ids[scope_id].clear(); - let parent_id = self.parent_ids[scope_id]; - if let Some(parent_id) = parent_id { - self.child_ids[parent_id].retain(|&child_id| child_id != scope_id); - } + self.cell.with_dependent_mut(|_allocator, inner| { + inner.child_ids[scope_id.index()].clear(); + let parent_id = self.parent_ids[scope_id]; + if let Some(parent_id) = parent_id { + inner.child_ids[parent_id.index()].retain(|&child_id| child_id != scope_id); + } + }); } } @@ -310,15 +316,18 @@ impl ScopeTree { /// Get the child scopes of a scope #[inline] pub fn get_child_ids(&self, scope_id: ScopeId) -> &[ScopeId] { - &self.child_ids[scope_id] + &self.cell.borrow_dependent().child_ids[scope_id.index()] } pub fn iter_all_child_ids(&self, scope_id: ScopeId) -> impl Iterator + '_ { - let mut stack = self.child_ids[scope_id].clone(); - let child_ids: &IndexVec> = &self.child_ids; + let mut stack = self.cell.borrow_dependent().child_ids[scope_id.index()] + .iter() + .copied() + .collect::>(); + let child_ids = &self.cell.borrow_dependent().child_ids; std::iter::from_fn(move || { if let Some(scope_id) = stack.pop() { - if let Some(children) = child_ids.get(scope_id) { + if let Some(children) = child_ids.get(scope_id.index()) { stack.extend(children.iter().copied()); } Some(scope_id) @@ -328,10 +337,11 @@ impl ScopeTree { }) } - /// Get a mutable reference to a scope's children - #[inline] - pub fn get_child_ids_mut(&mut self, scope_id: ScopeId) -> &mut Vec { - &mut self.child_ids[scope_id] + pub fn remove_child_scopes(&mut self, scope_id: ScopeId, child_scope_ids: &[ScopeId]) { + self.cell.with_dependent_mut(|_allocator, inner| { + inner.child_ids[scope_id.index()] + .retain(|scope_id| !child_scope_ids.contains(scope_id)); + }); } /// Create a scope. @@ -349,10 +359,12 @@ impl ScopeTree { }); self.node_ids.push(node_id); if self.build_child_ids { - self.child_ids.push(vec![]); - if let Some(parent_id) = parent_id { - self.child_ids[parent_id].push(scope_id); - } + self.cell.with_dependent_mut(|allocator, inner| { + inner.child_ids.push(ArenaVec::new_in(allocator)); + if let Some(parent_id) = parent_id { + inner.child_ids[parent_id.index()].push(scope_id); + } + }); } scope_id } @@ -418,7 +430,9 @@ impl ScopeTree { }); self.node_ids.reserve(additional); if self.build_child_ids { - self.child_ids.reserve(additional); + self.cell.with_dependent_mut(|_allocator, inner| { + inner.child_ids.reserve(additional); + }); } } } diff --git a/crates/oxc_traverse/src/context/scoping.rs b/crates/oxc_traverse/src/context/scoping.rs index 50bc2cf1e6868..407ed853545e2 100644 --- a/crates/oxc_traverse/src/context/scoping.rs +++ b/crates/oxc_traverse/src/context/scoping.rs @@ -133,8 +133,7 @@ impl TraverseScoping { fn insert_scope_below(&mut self, child_scope_ids: &[ScopeId], flags: ScopeFlags) -> ScopeId { // Remove these scopes from parent's children if self.scopes.has_child_ids() { - let current_child_scope_ids = self.scopes.get_child_ids_mut(self.current_scope_id); - current_child_scope_ids.retain(|scope_id| !child_scope_ids.contains(scope_id)); + self.scopes.remove_child_scopes(self.current_scope_id, child_scope_ids); } // Create new scope as child of parent