@@ -132,6 +132,9 @@ pub struct SemanticsImpl<'db> {
132
132
s2d_cache : RefCell < SourceToDefCache > ,
133
133
/// Rootnode to HirFileId cache
134
134
root_to_file_cache : RefCell < FxHashMap < SyntaxNode , HirFileId > > ,
135
+ /// HirFileId to Rootnode cache (this adds a layer over the database LRU cache to prevent
136
+ /// possibly frequent invalidation)
137
+ parse_cache : RefCell < FxHashMap < HirFileId , SyntaxNode > > ,
135
138
/// MacroCall to its expansion's MacroFileId cache
136
139
macro_call_cache : RefCell < FxHashMap < InFile < ast:: MacroCall > , MacroFileId > > ,
137
140
}
@@ -292,6 +295,7 @@ impl<'db> SemanticsImpl<'db> {
292
295
db,
293
296
s2d_cache : Default :: default ( ) ,
294
297
root_to_file_cache : Default :: default ( ) ,
298
+ parse_cache : Default :: default ( ) ,
295
299
macro_call_cache : Default :: default ( ) ,
296
300
}
297
301
}
@@ -303,6 +307,9 @@ impl<'db> SemanticsImpl<'db> {
303
307
}
304
308
305
309
pub fn parse_or_expand ( & self , file_id : HirFileId ) -> SyntaxNode {
310
+ if let Some ( root) = self . parse_cache . borrow ( ) . get ( & file_id) {
311
+ return root. clone ( ) ;
312
+ }
306
313
let node = self . db . parse_or_expand ( file_id) ;
307
314
self . cache ( node. clone ( ) , file_id) ;
308
315
node
@@ -977,6 +984,13 @@ impl<'db> SemanticsImpl<'db> {
977
984
let helpers =
978
985
def_map. derive_helpers_in_scope ( InFile :: new ( file_id, id) ) ?;
979
986
987
+ if !helpers. is_empty ( ) {
988
+ let text_range = attr. syntax ( ) . text_range ( ) ;
989
+ // remove any other token in this macro input, all their mappings are the
990
+ // same as this
991
+ tokens. retain ( |t| !text_range. contains_range ( t. text_range ( ) ) ) ;
992
+ }
993
+
980
994
let mut res = None ;
981
995
for ( .., derive) in
982
996
helpers. iter ( ) . filter ( |( helper, ..) | * helper == attr_name)
@@ -1407,6 +1421,7 @@ impl<'db> SemanticsImpl<'db> {
1407
1421
where
1408
1422
Def :: Ast : AstNode ,
1409
1423
{
1424
+ // FIXME: source call should go through the parse cache
1410
1425
let res = def. source ( self . db ) ?;
1411
1426
self . cache ( find_root ( res. value . syntax ( ) ) , res. file_id ) ;
1412
1427
Some ( res)
@@ -1464,8 +1479,9 @@ impl<'db> SemanticsImpl<'db> {
1464
1479
fn cache ( & self , root_node : SyntaxNode , file_id : HirFileId ) {
1465
1480
assert ! ( root_node. parent( ) . is_none( ) ) ;
1466
1481
let mut cache = self . root_to_file_cache . borrow_mut ( ) ;
1467
- let prev = cache. insert ( root_node, file_id) ;
1468
- assert ! ( prev. is_none( ) || prev == Some ( file_id) )
1482
+ let prev = cache. insert ( root_node. clone ( ) , file_id) ;
1483
+ assert ! ( prev. is_none( ) || prev == Some ( file_id) ) ;
1484
+ self . parse_cache . borrow_mut ( ) . insert ( file_id, root_node) ;
1469
1485
}
1470
1486
1471
1487
pub fn assert_contains_node ( & self , node : & SyntaxNode ) {
0 commit comments