1
1
//! Defines database & queries for macro expansion.
2
2
3
- use std:: sync:: OnceLock ;
4
-
5
3
use base_db:: {
6
4
salsa:: { self , debug:: DebugQueryTable } ,
7
- CrateId , Edition , FileId , SourceDatabase , VersionReq ,
5
+ CrateId , FileId , SourceDatabase ,
8
6
} ;
9
7
use either:: Either ;
10
8
use limit:: Limit ;
11
9
use mbe:: { syntax_node_to_token_tree, ValueResult } ;
12
10
use rustc_hash:: FxHashSet ;
13
- use span:: { Span , SyntaxContextId } ;
11
+ use span:: SyntaxContextId ;
14
12
use syntax:: {
15
13
ast:: { self , HasAttrs } ,
16
14
AstNode , Parse , SyntaxError , SyntaxNode , SyntaxToken , T ,
@@ -19,13 +17,14 @@ use triomphe::Arc;
19
17
20
18
use crate :: {
21
19
ast_id_map:: AstIdMap ,
22
- attrs:: { collect_attrs, RawAttrs } ,
20
+ attrs:: collect_attrs,
23
21
builtin_attr_macro:: pseudo_derive_attr_expansion,
24
22
builtin_fn_macro:: EagerExpander ,
23
+ declarative:: DeclarativeMacroExpander ,
25
24
fixup:: { self , reverse_fixups, SyntaxFixupUndoInfo } ,
26
25
hygiene:: {
27
- apply_mark , span_with_call_site_ctxt, span_with_def_site_ctxt, span_with_mixed_site_ctxt,
28
- SyntaxContextData , Transparency ,
26
+ span_with_call_site_ctxt, span_with_def_site_ctxt, span_with_mixed_site_ctxt,
27
+ SyntaxContextData ,
29
28
} ,
30
29
proc_macro:: ProcMacros ,
31
30
span_map:: { RealSpanMap , SpanMap , SpanMapRef } ,
@@ -43,82 +42,6 @@ use crate::{
43
42
/// Actual max for `analysis-stats .` at some point: 30672.
44
43
static TOKEN_LIMIT : Limit = Limit :: new ( 1_048_576 ) ;
45
44
46
- #[ derive( Debug , Clone , Eq , PartialEq ) ]
47
- /// Old-style `macro_rules` or the new macros 2.0
48
- pub struct DeclarativeMacroExpander {
49
- pub mac : mbe:: DeclarativeMacro < span:: Span > ,
50
- pub transparency : Transparency ,
51
- }
52
-
53
- // FIXME: Remove this once we drop support for 1.76
54
- static REQUIREMENT : OnceLock < VersionReq > = OnceLock :: new ( ) ;
55
-
56
- impl DeclarativeMacroExpander {
57
- pub fn expand (
58
- & self ,
59
- db : & dyn ExpandDatabase ,
60
- tt : tt:: Subtree ,
61
- call_id : MacroCallId ,
62
- ) -> ExpandResult < tt:: Subtree > {
63
- let loc = db. lookup_intern_macro_call ( call_id) ;
64
- let toolchain = & db. crate_graph ( ) [ loc. def . krate ] . toolchain ;
65
- let new_meta_vars = toolchain. as_ref ( ) . map_or ( false , |version| {
66
- REQUIREMENT . get_or_init ( || VersionReq :: parse ( ">=1.76" ) . unwrap ( ) ) . matches (
67
- & base_db:: Version {
68
- pre : base_db:: Prerelease :: EMPTY ,
69
- build : base_db:: BuildMetadata :: EMPTY ,
70
- major : version. major ,
71
- minor : version. minor ,
72
- patch : version. patch ,
73
- } ,
74
- )
75
- } ) ;
76
- match self . mac . err ( ) {
77
- Some ( e) => ExpandResult :: new (
78
- tt:: Subtree :: empty ( tt:: DelimSpan { open : loc. call_site , close : loc. call_site } ) ,
79
- ExpandError :: other ( format ! ( "invalid macro definition: {e}" ) ) ,
80
- ) ,
81
- None => self
82
- . mac
83
- . expand (
84
- & tt,
85
- |s| s. ctx = apply_mark ( db, s. ctx , call_id, self . transparency ) ,
86
- new_meta_vars,
87
- loc. call_site ,
88
- )
89
- . map_err ( Into :: into) ,
90
- }
91
- }
92
-
93
- pub fn expand_unhygienic (
94
- & self ,
95
- db : & dyn ExpandDatabase ,
96
- tt : tt:: Subtree ,
97
- krate : CrateId ,
98
- call_site : Span ,
99
- ) -> ExpandResult < tt:: Subtree > {
100
- let toolchain = & db. crate_graph ( ) [ krate] . toolchain ;
101
- let new_meta_vars = toolchain. as_ref ( ) . map_or ( false , |version| {
102
- REQUIREMENT . get_or_init ( || VersionReq :: parse ( ">=1.76" ) . unwrap ( ) ) . matches (
103
- & base_db:: Version {
104
- pre : base_db:: Prerelease :: EMPTY ,
105
- build : base_db:: BuildMetadata :: EMPTY ,
106
- major : version. major ,
107
- minor : version. minor ,
108
- patch : version. patch ,
109
- } ,
110
- )
111
- } ) ;
112
- match self . mac . err ( ) {
113
- Some ( e) => ExpandResult :: new (
114
- tt:: Subtree :: empty ( tt:: DelimSpan { open : call_site, close : call_site } ) ,
115
- ExpandError :: other ( format ! ( "invalid macro definition: {e}" ) ) ,
116
- ) ,
117
- None => self . mac . expand ( & tt, |_| ( ) , new_meta_vars, call_site) . map_err ( Into :: into) ,
118
- }
119
- }
120
- }
121
-
122
45
#[ derive( Debug , Clone , Eq , PartialEq ) ]
123
46
pub enum TokenExpander {
124
47
/// Old-style `macro_rules` or the new macros 2.0
@@ -141,6 +64,7 @@ pub trait ExpandDatabase: SourceDatabase {
141
64
#[ salsa:: input]
142
65
fn proc_macros ( & self ) -> Arc < ProcMacros > ;
143
66
67
+ #[ salsa:: invoke( AstIdMap :: ast_id_map) ]
144
68
fn ast_id_map ( & self , file_id : HirFileId ) -> Arc < AstIdMap > ;
145
69
146
70
/// Main public API -- parses a hir file, not caring whether it's a real
@@ -156,8 +80,10 @@ pub trait ExpandDatabase: SourceDatabase {
156
80
macro_file : MacroFileId ,
157
81
) -> ExpandResult < ( Parse < SyntaxNode > , Arc < ExpansionSpanMap > ) > ;
158
82
#[ salsa:: transparent]
83
+ #[ salsa:: invoke( SpanMap :: new) ]
159
84
fn span_map ( & self , file_id : HirFileId ) -> SpanMap ;
160
85
86
+ #[ salsa:: invoke( crate :: span_map:: real_span_map) ]
161
87
fn real_span_map ( & self , file_id : FileId ) -> Arc < RealSpanMap > ;
162
88
163
89
/// Macro ids. That's probably the tricksiest bit in rust-analyzer, and the
@@ -173,6 +99,7 @@ pub trait ExpandDatabase: SourceDatabase {
173
99
#[ salsa:: transparent]
174
100
fn setup_syntax_context_root ( & self ) -> ( ) ;
175
101
#[ salsa:: transparent]
102
+ #[ salsa:: invoke( crate :: hygiene:: dump_syntax_contexts) ]
176
103
fn dump_syntax_contexts ( & self ) -> String ;
177
104
178
105
/// Lowers syntactic macro call to a token tree representation. That's a firewall
@@ -184,8 +111,10 @@ pub trait ExpandDatabase: SourceDatabase {
184
111
) -> ValueResult < Option < ( Arc < tt:: Subtree > , SyntaxFixupUndoInfo ) > , Arc < Box < [ SyntaxError ] > > > ;
185
112
/// Fetches the expander for this macro.
186
113
#[ salsa:: transparent]
114
+ #[ salsa:: invoke( TokenExpander :: macro_expander) ]
187
115
fn macro_expander ( & self , id : MacroDefId ) -> TokenExpander ;
188
116
/// Fetches (and compiles) the expander of this decl macro.
117
+ #[ salsa:: invoke( DeclarativeMacroExpander :: expander) ]
189
118
fn decl_macro_expander (
190
119
& self ,
191
120
def_crate : CrateId ,
@@ -203,36 +132,6 @@ pub trait ExpandDatabase: SourceDatabase {
203
132
) -> ExpandResult < Box < [ SyntaxError ] > > ;
204
133
}
205
134
206
- #[ inline]
207
- pub fn span_map ( db : & dyn ExpandDatabase , file_id : HirFileId ) -> SpanMap {
208
- match file_id. repr ( ) {
209
- HirFileIdRepr :: FileId ( file_id) => SpanMap :: RealSpanMap ( db. real_span_map ( file_id) ) ,
210
- HirFileIdRepr :: MacroFile ( m) => {
211
- SpanMap :: ExpansionSpanMap ( db. parse_macro_expansion ( m) . value . 1 )
212
- }
213
- }
214
- }
215
-
216
- pub fn real_span_map ( db : & dyn ExpandDatabase , file_id : FileId ) -> Arc < RealSpanMap > {
217
- use syntax:: ast:: HasModuleItem ;
218
- let mut pairs = vec ! [ ( syntax:: TextSize :: new( 0 ) , span:: ROOT_ERASED_FILE_AST_ID ) ] ;
219
- let ast_id_map = db. ast_id_map ( file_id. into ( ) ) ;
220
- let tree = db. parse ( file_id) . tree ( ) ;
221
- // FIXME: Descend into modules and other item containing items that are not annotated with attributes
222
- // and allocate pairs for those as well. This gives us finer grained span anchors resulting in
223
- // better incrementality
224
- pairs. extend (
225
- tree. items ( )
226
- . map ( |item| ( item. syntax ( ) . text_range ( ) . start ( ) , ast_id_map. ast_id ( & item) . erase ( ) ) ) ,
227
- ) ;
228
-
229
- Arc :: new ( RealSpanMap :: from_file (
230
- file_id,
231
- pairs. into_boxed_slice ( ) ,
232
- tree. syntax ( ) . text_range ( ) . end ( ) ,
233
- ) )
234
- }
235
-
236
135
/// This expands the given macro call, but with different arguments. This is
237
136
/// used for completion, where we want to see what 'would happen' if we insert a
238
137
/// token. The `token_to_map` mapped down into the expansion, with the mapped
@@ -357,10 +256,6 @@ pub fn expand_speculative(
357
256
Some ( ( node. syntax_node ( ) , token) )
358
257
}
359
258
360
- fn ast_id_map ( db : & dyn ExpandDatabase , file_id : HirFileId ) -> Arc < AstIdMap > {
361
- Arc :: new ( AstIdMap :: from_source ( & db. parse_or_expand ( file_id) ) )
362
- }
363
-
364
259
fn parse_or_expand ( db : & dyn ExpandDatabase , file_id : HirFileId ) -> SyntaxNode {
365
260
match file_id. repr ( ) {
366
261
HirFileIdRepr :: FileId ( file_id) => db. parse ( file_id) . syntax_node ( ) ,
@@ -412,7 +307,10 @@ fn parse_macro_expansion_error(
412
307
. map ( |it| it. 0 . errors ( ) . to_vec ( ) . into_boxed_slice ( ) )
413
308
}
414
309
415
- fn parse_with_map ( db : & dyn ExpandDatabase , file_id : HirFileId ) -> ( Parse < SyntaxNode > , SpanMap ) {
310
+ pub ( crate ) fn parse_with_map (
311
+ db : & dyn ExpandDatabase ,
312
+ file_id : HirFileId ,
313
+ ) -> ( Parse < SyntaxNode > , SpanMap ) {
416
314
match file_id. repr ( ) {
417
315
HirFileIdRepr :: FileId ( file_id) => {
418
316
( db. parse ( file_id) . to_syntax ( ) , SpanMap :: RealSpanMap ( db. real_span_map ( file_id) ) )
@@ -581,100 +479,18 @@ fn censor_for_macro_input(loc: &MacroCallLoc, node: &SyntaxNode) -> FxHashSet<Sy
581
479
. unwrap_or_default ( )
582
480
}
583
481
584
- fn decl_macro_expander (
585
- db : & dyn ExpandDatabase ,
586
- def_crate : CrateId ,
587
- id : AstId < ast:: Macro > ,
588
- ) -> Arc < DeclarativeMacroExpander > {
589
- let crate_data = & db. crate_graph ( ) [ def_crate] ;
590
- let is_2021 = crate_data. edition >= Edition :: Edition2021 ;
591
- let ( root, map) = parse_with_map ( db, id. file_id ) ;
592
- let root = root. syntax_node ( ) ;
593
-
594
- let transparency = |node| {
595
- // ... would be nice to have the item tree here
596
- let attrs = RawAttrs :: new ( db, node, map. as_ref ( ) ) . filter ( db, def_crate) ;
597
- match & * attrs
598
- . iter ( )
599
- . find ( |it| {
600
- it. path . as_ident ( ) . and_then ( |it| it. as_str ( ) ) == Some ( "rustc_macro_transparency" )
601
- } ) ?
602
- . token_tree_value ( ) ?
603
- . token_trees
604
- {
605
- [ tt:: TokenTree :: Leaf ( tt:: Leaf :: Ident ( i) ) , ..] => match & * i. text {
606
- "transparent" => Some ( Transparency :: Transparent ) ,
607
- "semitransparent" => Some ( Transparency :: SemiTransparent ) ,
608
- "opaque" => Some ( Transparency :: Opaque ) ,
609
- _ => None ,
610
- } ,
611
- _ => None ,
612
- }
613
- } ;
614
- let toolchain = crate_data. toolchain . as_ref ( ) ;
615
- let new_meta_vars = toolchain. as_ref ( ) . map_or ( false , |version| {
616
- REQUIREMENT . get_or_init ( || VersionReq :: parse ( ">=1.76" ) . unwrap ( ) ) . matches (
617
- & base_db:: Version {
618
- pre : base_db:: Prerelease :: EMPTY ,
619
- build : base_db:: BuildMetadata :: EMPTY ,
620
- major : version. major ,
621
- minor : version. minor ,
622
- patch : version. patch ,
623
- } ,
624
- )
625
- } ) ;
626
-
627
- let ( mac, transparency) = match id. to_ptr ( db) . to_node ( & root) {
628
- ast:: Macro :: MacroRules ( macro_rules) => (
629
- match macro_rules. token_tree ( ) {
630
- Some ( arg) => {
631
- let tt = mbe:: syntax_node_to_token_tree (
632
- arg. syntax ( ) ,
633
- map. as_ref ( ) ,
634
- map. span_for_range ( macro_rules. macro_rules_token ( ) . unwrap ( ) . text_range ( ) ) ,
635
- ) ;
636
-
637
- mbe:: DeclarativeMacro :: parse_macro_rules ( & tt, is_2021, new_meta_vars)
638
- }
639
- None => mbe:: DeclarativeMacro :: from_err (
640
- mbe:: ParseError :: Expected ( "expected a token tree" . into ( ) ) ,
641
- is_2021,
642
- ) ,
643
- } ,
644
- transparency ( & macro_rules) . unwrap_or ( Transparency :: SemiTransparent ) ,
645
- ) ,
646
- ast:: Macro :: MacroDef ( macro_def) => (
647
- match macro_def. body ( ) {
648
- Some ( arg) => {
649
- let tt = mbe:: syntax_node_to_token_tree (
650
- arg. syntax ( ) ,
651
- map. as_ref ( ) ,
652
- map. span_for_range ( macro_def. macro_token ( ) . unwrap ( ) . text_range ( ) ) ,
653
- ) ;
654
-
655
- mbe:: DeclarativeMacro :: parse_macro2 ( & tt, is_2021, new_meta_vars)
656
- }
657
- None => mbe:: DeclarativeMacro :: from_err (
658
- mbe:: ParseError :: Expected ( "expected a token tree" . into ( ) ) ,
659
- is_2021,
660
- ) ,
661
- } ,
662
- transparency ( & macro_def) . unwrap_or ( Transparency :: Opaque ) ,
663
- ) ,
664
- } ;
665
- Arc :: new ( DeclarativeMacroExpander { mac, transparency } )
666
- }
667
-
668
- fn macro_expander ( db : & dyn ExpandDatabase , id : MacroDefId ) -> TokenExpander {
669
- match id. kind {
670
- MacroDefKind :: Declarative ( ast_id) => {
671
- TokenExpander :: DeclarativeMacro ( db. decl_macro_expander ( id. krate , ast_id) )
482
+ impl TokenExpander {
483
+ fn macro_expander ( db : & dyn ExpandDatabase , id : MacroDefId ) -> TokenExpander {
484
+ match id. kind {
485
+ MacroDefKind :: Declarative ( ast_id) => {
486
+ TokenExpander :: DeclarativeMacro ( db. decl_macro_expander ( id. krate , ast_id) )
487
+ }
488
+ MacroDefKind :: BuiltIn ( expander, _) => TokenExpander :: BuiltIn ( expander) ,
489
+ MacroDefKind :: BuiltInAttr ( expander, _) => TokenExpander :: BuiltInAttr ( expander) ,
490
+ MacroDefKind :: BuiltInDerive ( expander, _) => TokenExpander :: BuiltInDerive ( expander) ,
491
+ MacroDefKind :: BuiltInEager ( expander, ..) => TokenExpander :: BuiltInEager ( expander) ,
492
+ MacroDefKind :: ProcMacro ( expander, ..) => TokenExpander :: ProcMacro ( expander) ,
672
493
}
673
- MacroDefKind :: BuiltIn ( expander, _) => TokenExpander :: BuiltIn ( expander) ,
674
- MacroDefKind :: BuiltInAttr ( expander, _) => TokenExpander :: BuiltInAttr ( expander) ,
675
- MacroDefKind :: BuiltInDerive ( expander, _) => TokenExpander :: BuiltInDerive ( expander) ,
676
- MacroDefKind :: BuiltInEager ( expander, ..) => TokenExpander :: BuiltInEager ( expander) ,
677
- MacroDefKind :: ProcMacro ( expander, ..) => TokenExpander :: ProcMacro ( expander) ,
678
494
}
679
495
}
680
496
@@ -862,40 +678,3 @@ fn check_tt_count(tt: &tt::Subtree) -> Result<(), ExpandResult<()>> {
862
678
fn setup_syntax_context_root ( db : & dyn ExpandDatabase ) {
863
679
db. intern_syntax_context ( SyntaxContextData :: root ( ) ) ;
864
680
}
865
-
866
- fn dump_syntax_contexts ( db : & dyn ExpandDatabase ) -> String {
867
- let mut s = String :: from ( "Expansions:" ) ;
868
- let mut entries = InternMacroCallLookupQuery . in_db ( db) . entries :: < Vec < _ > > ( ) ;
869
- entries. sort_by_key ( |e| e. key ) ;
870
- for e in entries {
871
- let id = e. key ;
872
- let expn_data = e. value . as_ref ( ) . unwrap ( ) ;
873
- s. push_str ( & format ! (
874
- "\n {:?}: parent: {:?}, call_site_ctxt: {:?}, def_site_ctxt: {:?}, kind: {:?}" ,
875
- id,
876
- expn_data. kind. file_id( ) ,
877
- expn_data. call_site,
878
- SyntaxContextId :: ROOT , // FIXME expn_data.def_site,
879
- expn_data. kind. descr( ) ,
880
- ) ) ;
881
- }
882
-
883
- s. push_str ( "\n \n SyntaxContexts:\n " ) ;
884
- let mut entries = InternSyntaxContextLookupQuery . in_db ( db) . entries :: < Vec < _ > > ( ) ;
885
- entries. sort_by_key ( |e| e. key ) ;
886
- for e in entries {
887
- struct SyntaxContextDebug < ' a > (
888
- & ' a dyn ExpandDatabase ,
889
- SyntaxContextId ,
890
- & ' a SyntaxContextData ,
891
- ) ;
892
-
893
- impl < ' a > std:: fmt:: Debug for SyntaxContextDebug < ' a > {
894
- fn fmt ( & self , f : & mut std:: fmt:: Formatter < ' _ > ) -> std:: fmt:: Result {
895
- self . 2 . fancy_debug ( self . 1 , self . 0 , f)
896
- }
897
- }
898
- stdx:: format_to!( s, "{:?}\n " , SyntaxContextDebug ( db, e. key, & e. value. unwrap( ) ) ) ;
899
- }
900
- s
901
- }
0 commit comments