diff --git a/crates/red_knot_python_semantic/src/semantic_index/builder.rs b/crates/red_knot_python_semantic/src/semantic_index/builder.rs index 14e530c151c17..246c810216ae4 100644 --- a/crates/red_knot_python_semantic/src/semantic_index/builder.rs +++ b/crates/red_knot_python_semantic/src/semantic_index/builder.rs @@ -155,7 +155,7 @@ impl<'db> SemanticIndexBuilder<'db> { self.current_use_def_map_mut().restore(state); } - fn flow_merge(&mut self, state: &FlowSnapshot) { + fn flow_merge(&mut self, state: FlowSnapshot) { self.current_use_def_map_mut().merge(state); } @@ -497,7 +497,7 @@ where self.visit_elif_else_clause(clause); } for post_clause_state in post_clauses { - self.flow_merge(&post_clause_state); + self.flow_merge(post_clause_state); } let has_else = node .elif_else_clauses @@ -506,7 +506,7 @@ where if !has_else { // if there's no else clause, then it's possible we took none of the branches, // and the pre_if state can reach here - self.flow_merge(&pre_if); + self.flow_merge(pre_if); } } ast::Stmt::While(node) => { @@ -524,13 +524,13 @@ where // We may execute the `else` clause without ever executing the body, so merge in // the pre-loop state before visiting `else`. - self.flow_merge(&pre_loop); + self.flow_merge(pre_loop); self.visit_body(&node.orelse); // Breaking out of a while loop bypasses the `else` clause, so merge in the break // states after visiting `else`. for break_state in break_states { - self.flow_merge(&break_state); + self.flow_merge(break_state); } } ast::Stmt::Break(_) => { @@ -640,7 +640,7 @@ where let post_body = self.flow_snapshot(); self.flow_restore(pre_if); self.visit_expr(orelse); - self.flow_merge(&post_body); + self.flow_merge(post_body); } ast::Expr::ListComp( list_comprehension @ ast::ExprListComp { diff --git a/crates/red_knot_python_semantic/src/semantic_index/use_def.rs b/crates/red_knot_python_semantic/src/semantic_index/use_def.rs index 63b51b6c7734f..96fe0fd56d9af 100644 --- a/crates/red_knot_python_semantic/src/semantic_index/use_def.rs +++ b/crates/red_knot_python_semantic/src/semantic_index/use_def.rs @@ -334,7 +334,7 @@ impl<'db> UseDefMapBuilder<'db> { /// Merge the given snapshot into the current state, reflecting that we might have taken either /// path to get here. The new visible-definitions state for each symbol should include /// definitions from both the prior state and the snapshot. - pub(super) fn merge(&mut self, snapshot: &FlowSnapshot) { + pub(super) fn merge(&mut self, snapshot: FlowSnapshot) { // The tricky thing about merging two Ranges pointing into `all_definitions` is that if the // two Ranges aren't already adjacent in `all_definitions`, we will have to copy at least // one or the other of the ranges to the end of `all_definitions` so as to make them @@ -348,9 +348,10 @@ impl<'db> UseDefMapBuilder<'db> { // greater than the number of known symbols in a previously-taken snapshot. debug_assert!(self.definitions_by_symbol.len() >= snapshot.definitions_by_symbol.len()); - for (symbol_id, current) in self.definitions_by_symbol.iter_mut_enumerated() { - if let Some(snapshot) = snapshot.definitions_by_symbol.get(symbol_id) { - *current = SymbolState::merge(current, snapshot); + let mut snapshot_definitions_iter = snapshot.definitions_by_symbol.into_iter(); + for current in &mut self.definitions_by_symbol { + if let Some(snapshot) = snapshot_definitions_iter.next() { + current.merge(snapshot); } else { // Symbol not present in snapshot, so it's unbound from that path. current.add_unbound(); diff --git a/crates/red_knot_python_semantic/src/semantic_index/use_def/symbol_state.rs b/crates/red_knot_python_semantic/src/semantic_index/use_def/symbol_state.rs index fd62271ccda1f..c465bbe320b1f 100644 --- a/crates/red_knot_python_semantic/src/semantic_index/use_def/symbol_state.rs +++ b/crates/red_knot_python_semantic/src/semantic_index/use_def/symbol_state.rs @@ -66,9 +66,11 @@ const INLINE_CONSTRAINT_BLOCKS: usize = 2; const INLINE_VISIBLE_DEFINITIONS_PER_SYMBOL: usize = 4; /// One [`BitSet`] of applicable [`ScopedConstraintId`] per visible definition. -type Constraints = - SmallVec<[BitSet; INLINE_VISIBLE_DEFINITIONS_PER_SYMBOL]>; +type InlineConstraintArray = + [BitSet; INLINE_VISIBLE_DEFINITIONS_PER_SYMBOL]; +type Constraints = SmallVec; type ConstraintsIterator<'a> = std::slice::Iter<'a, BitSet>; +type ConstraintsIntoIterator = smallvec::IntoIter; /// Visible definitions and narrowing constraints for a single symbol at some point in control flow. #[derive(Clone, Debug, PartialEq, Eq)] @@ -126,17 +128,18 @@ impl SymbolState { } } - /// Merge two [`SymbolState`] and return the result. - pub(super) fn merge(a: &SymbolState, b: &SymbolState) -> SymbolState { - let mut merged = Self { + /// Merge another [`SymbolState`] into this one. + pub(super) fn merge(&mut self, b: SymbolState) { + let mut a = Self { visible_definitions: Definitions::default(), constraints: Constraints::default(), - may_be_unbound: a.may_be_unbound || b.may_be_unbound, + may_be_unbound: self.may_be_unbound || b.may_be_unbound, }; + std::mem::swap(&mut a, self); let mut a_defs_iter = a.visible_definitions.iter(); let mut b_defs_iter = b.visible_definitions.iter(); - let mut a_constraints_iter = a.constraints.iter(); - let mut b_constraints_iter = b.constraints.iter(); + let mut a_constraints_iter = a.constraints.into_iter(); + let mut b_constraints_iter = b.constraints.into_iter(); let mut opt_a_def: Option = a_defs_iter.next(); let mut opt_b_def: Option = b_defs_iter.next(); @@ -147,8 +150,8 @@ impl SymbolState { // of the constraints from the two paths; a constraint that applies from only one possible // path is irrelevant. - // Helper to push `def`, with constraints in `constraints_iter`, onto merged`. - let push = |def, constraints_iter: &mut ConstraintsIterator, merged: &mut Self| { + // Helper to push `def`, with constraints in `constraints_iter`, onto `self`. + let push = |def, constraints_iter: &mut ConstraintsIntoIterator, merged: &mut Self| { merged.visible_definitions.insert(def); // SAFETY: we only ever create SymbolState with either no definitions and no constraint // bitsets (`::unbound`) or one definition and one constraint bitset (`::with`), and @@ -158,59 +161,57 @@ impl SymbolState { let constraints = constraints_iter .next() .expect("definitions and constraints length mismatch"); - merged.constraints.push(constraints.clone()); + merged.constraints.push(constraints); }; loop { match (opt_a_def, opt_b_def) { (Some(a_def), Some(b_def)) => match a_def.cmp(&b_def) { std::cmp::Ordering::Less => { - // Next definition ID is only in `a`, push it to `merged` and advance `a`. - push(a_def, &mut a_constraints_iter, &mut merged); + // Next definition ID is only in `a`, push it to `self` and advance `a`. + push(a_def, &mut a_constraints_iter, self); opt_a_def = a_defs_iter.next(); } std::cmp::Ordering::Greater => { - // Next definition ID is only in `b`, push it to `merged` and advance `b`. - push(b_def, &mut b_constraints_iter, &mut merged); + // Next definition ID is only in `b`, push it to `self` and advance `b`. + push(b_def, &mut b_constraints_iter, self); opt_b_def = b_defs_iter.next(); } std::cmp::Ordering::Equal => { - // Next definition is in both; push to `merged` and intersect constraints. - push(a_def, &mut a_constraints_iter, &mut merged); + // Next definition is in both; push to `self` and intersect constraints. + push(a_def, &mut b_constraints_iter, self); // SAFETY: we only ever create SymbolState with either no definitions and // no constraint bitsets (`::unbound`) or one definition and one constraint // bitset (`::with`), and `::merge` always pushes one definition and one // constraint bitset together (just below), so the number of definitions // and the number of constraint bitsets can never get out of sync. - let b_constraints = b_constraints_iter + let a_constraints = a_constraints_iter .next() .expect("definitions and constraints length mismatch"); // If the same definition is visible through both paths, any constraint // that applies on only one path is irrelevant to the resulting type from // unioning the two paths, so we intersect the constraints. - merged - .constraints + self.constraints .last_mut() .unwrap() - .intersect(b_constraints); + .intersect(&a_constraints); opt_a_def = a_defs_iter.next(); opt_b_def = b_defs_iter.next(); } }, (Some(a_def), None) => { // We've exhausted `b`, just push the def from `a` and move on to the next. - push(a_def, &mut a_constraints_iter, &mut merged); + push(a_def, &mut a_constraints_iter, self); opt_a_def = a_defs_iter.next(); } (None, Some(b_def)) => { // We've exhausted `a`, just push the def from `b` and move on to the next. - push(b_def, &mut b_constraints_iter, &mut merged); + push(b_def, &mut b_constraints_iter, self); opt_b_def = b_defs_iter.next(); } (None, None) => break, } } - merged } /// Get iterator over visible definitions with constraints. @@ -340,7 +341,8 @@ mod tests { let mut cd0b = SymbolState::with(ScopedDefinitionId::from_u32(0)); cd0b.add_constraint(ScopedConstraintId::from_u32(0)); - let cd0 = SymbolState::merge(&cd0a, &cd0b); + cd0a.merge(cd0b); + let mut cd0 = cd0a; cd0.assert(false, &["0<0>"]); // merging the same definition with differing constraints drops all constraints @@ -350,7 +352,8 @@ mod tests { let mut cd1b = SymbolState::with(ScopedDefinitionId::from_u32(1)); cd1b.add_constraint(ScopedConstraintId::from_u32(2)); - let cd1 = SymbolState::merge(&cd1a, &cd1b); + cd1a.merge(cd1b); + let cd1 = cd1a; cd1.assert(false, &["1<>"]); // merging a constrained definition with unbound keeps both @@ -359,11 +362,13 @@ mod tests { let cd2b = SymbolState::unbound(); - let cd2 = SymbolState::merge(&cd2a, &cd2b); + cd2a.merge(cd2b); + let cd2 = cd2a; cd2.assert(true, &["2<3>"]); // merging different definitions keeps them each with their existing constraints - let cd = SymbolState::merge(&cd0, &cd2); + cd0.merge(cd2); + let cd = cd0; cd.assert(true, &["0<0>", "2<3>"]); } }