From c603b9043fd530c11d3001eb62b1315c5aa1afe0 Mon Sep 17 00:00:00 2001 From: Aleksey Kladov Date: Sat, 27 Nov 2021 20:50:07 +0300 Subject: [PATCH] feat: make hightlighting linear In https://youtu.be/qvIZZf5dmTE, we've noticed that AstIdMap does a linear lookup when going from SyntaxNode to Id. This leads to accidentally quadratic overall performance. Replace linear lookup with a O(1) hashmap lookup. Future work: don't duplicate `SyntaxNodePtr` in `AstIdMap` and switch to "call site dependency injection" style storage (eg, store a `HashSet`). See the explanation of the work here on YouTube https://youtu.be/wvEgymUm7cY :-) --- crates/hir_expand/src/ast_id_map.rs | 13 ++++++++----- 1 file changed, 8 insertions(+), 5 deletions(-) diff --git a/crates/hir_expand/src/ast_id_map.rs b/crates/hir_expand/src/ast_id_map.rs index 16cf299076..64387b8148 100644 --- a/crates/hir_expand/src/ast_id_map.rs +++ b/crates/hir_expand/src/ast_id_map.rs @@ -14,6 +14,7 @@ use std::{ use la_arena::{Arena, Idx}; use profile::Count; +use rustc_hash::FxHashMap; use syntax::{ast, match_ast, AstNode, AstPtr, SyntaxNode, SyntaxNodePtr}; /// `AstId` points to an AST node in a specific file. @@ -63,6 +64,7 @@ type ErasedFileAstId = Idx; #[derive(Debug, PartialEq, Eq, Default)] pub struct AstIdMap { arena: Arena, + map: FxHashMap, _c: Count, } @@ -89,6 +91,7 @@ impl AstIdMap { } } }); + res.map.extend(res.arena.iter().map(|(idx, ptr)| (ptr.clone(), idx))); res } @@ -96,16 +99,16 @@ impl AstIdMap { let raw = self.erased_ast_id(item.syntax()); FileAstId { raw, _ty: PhantomData } } + fn erased_ast_id(&self, item: &SyntaxNode) -> ErasedFileAstId { let ptr = SyntaxNodePtr::new(item); - match self.arena.iter().find(|(_id, i)| **i == ptr) { - Some((it, _)) => it, - None => panic!( + *self.map.get(&ptr).unwrap_or_else(|| { + panic!( "Can't find {:?} in AstIdMap:\n{:?}", item, self.arena.iter().map(|(_id, i)| i).collect::>(), - ), - } + ) + }) } pub fn get(&self, id: FileAstId) -> AstPtr {