2019-10-29 13:03:29 +00:00
|
|
|
//! Defines database & queries for macro expansion.
|
|
|
|
|
2019-10-29 11:55:39 +00:00
|
|
|
use std::sync::Arc;
|
|
|
|
|
2020-08-13 14:25:38 +00:00
|
|
|
use base_db::{salsa, SourceDatabase};
|
2022-01-07 17:51:10 +00:00
|
|
|
use either::Either;
|
2021-07-10 20:49:17 +00:00
|
|
|
use limit::Limit;
|
2022-02-21 18:14:06 +00:00
|
|
|
use mbe::syntax_node_to_token_tree;
|
2022-02-08 17:13:18 +00:00
|
|
|
use rustc_hash::FxHashSet;
|
2021-03-25 19:52:35 +00:00
|
|
|
use syntax::{
|
2022-01-07 17:51:10 +00:00
|
|
|
ast::{self, HasAttrs, HasDocComments},
|
2021-09-19 16:30:29 +00:00
|
|
|
AstNode, GreenNode, Parse, SyntaxNode, SyntaxToken, T,
|
2021-03-25 19:52:35 +00:00
|
|
|
};
|
2019-10-29 11:55:39 +00:00
|
|
|
|
|
|
|
use crate::{
|
2022-03-10 19:53:50 +00:00
|
|
|
ast_id_map::AstIdMap, builtin_attr_macro::pseudo_derive_attr_expansion, fixup,
|
|
|
|
hygiene::HygieneFrame, BuiltinAttrExpander, BuiltinDeriveExpander, BuiltinFnLikeExpander,
|
|
|
|
ExpandError, ExpandResult, ExpandTo, HirFileId, HirFileIdRepr, MacroCallId, MacroCallKind,
|
|
|
|
MacroCallLoc, MacroDefId, MacroDefKind, MacroFile, ProcMacroExpander,
|
2019-10-29 11:55:39 +00:00
|
|
|
};
|
|
|
|
|
2020-12-10 16:50:56 +00:00
|
|
|
/// Total limit on the number of tokens produced by any macro invocation.
|
|
|
|
///
|
|
|
|
/// If an invocation produces more tokens than this limit, it will not be stored in the database and
|
|
|
|
/// an error will be emitted.
|
2021-07-31 15:25:45 +00:00
|
|
|
///
|
|
|
|
/// Actual max for `analysis-stats .` at some point: 30672.
|
|
|
|
static TOKEN_LIMIT: Limit = Limit::new(524_288);
|
2020-12-10 16:50:56 +00:00
|
|
|
|
2019-11-10 03:03:24 +00:00
|
|
|
#[derive(Debug, Clone, Eq, PartialEq)]
|
|
|
|
pub enum TokenExpander {
|
2021-10-10 18:07:43 +00:00
|
|
|
/// Old-style `macro_rules` or the new macros 2.0
|
|
|
|
DeclarativeMacro { mac: mbe::DeclarativeMacro, def_site_token_map: mbe::TokenMap },
|
2021-05-04 15:20:10 +00:00
|
|
|
/// Stuff like `line!` and `file!`.
|
2019-11-23 13:54:39 +00:00
|
|
|
Builtin(BuiltinFnLikeExpander),
|
2021-06-09 16:02:31 +00:00
|
|
|
/// `global_allocator` and such.
|
|
|
|
BuiltinAttr(BuiltinAttrExpander),
|
2021-05-04 15:20:10 +00:00
|
|
|
/// `derive(Copy)` and such.
|
2019-12-05 14:10:33 +00:00
|
|
|
BuiltinDerive(BuiltinDeriveExpander),
|
2021-05-04 15:20:10 +00:00
|
|
|
/// The thing we love the most here in rust-analyzer -- procedural macros.
|
2020-03-18 09:47:59 +00:00
|
|
|
ProcMacro(ProcMacroExpander),
|
2019-11-10 03:03:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl TokenExpander {
|
2021-05-04 17:29:30 +00:00
|
|
|
fn expand(
|
2019-11-10 03:03:24 +00:00
|
|
|
&self,
|
|
|
|
db: &dyn AstDatabase,
|
2021-05-19 18:19:08 +00:00
|
|
|
id: MacroCallId,
|
2019-11-10 03:03:24 +00:00
|
|
|
tt: &tt::Subtree,
|
2022-02-21 18:14:06 +00:00
|
|
|
) -> ExpandResult<tt::Subtree> {
|
2019-11-10 03:03:24 +00:00
|
|
|
match self {
|
2022-02-21 18:14:06 +00:00
|
|
|
TokenExpander::DeclarativeMacro { mac, .. } => mac.expand(tt).map_err(Into::into),
|
|
|
|
TokenExpander::Builtin(it) => it.expand(db, id, tt).map_err(Into::into),
|
2021-08-21 10:55:05 +00:00
|
|
|
TokenExpander::BuiltinAttr(it) => it.expand(db, id, tt),
|
2021-08-20 12:28:36 +00:00
|
|
|
TokenExpander::BuiltinDerive(it) => it.expand(db, id, tt),
|
2020-05-14 09:57:37 +00:00
|
|
|
TokenExpander::ProcMacro(_) => {
|
2021-01-08 14:46:48 +00:00
|
|
|
// We store the result in salsa db to prevent non-deterministic behavior in
|
2020-05-14 09:57:37 +00:00
|
|
|
// some proc-macro implementation
|
|
|
|
// See #4315 for details
|
2021-08-20 12:34:46 +00:00
|
|
|
db.expand_proc_macro(id)
|
2020-05-14 09:57:37 +00:00
|
|
|
}
|
2019-11-10 03:03:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-04 17:29:30 +00:00
|
|
|
pub(crate) fn map_id_down(&self, id: tt::TokenId) -> tt::TokenId {
|
2019-11-10 03:03:24 +00:00
|
|
|
match self {
|
2021-10-10 18:07:43 +00:00
|
|
|
TokenExpander::DeclarativeMacro { mac, .. } => mac.map_id_down(id),
|
2021-05-04 19:03:16 +00:00
|
|
|
TokenExpander::Builtin(..)
|
2021-06-09 16:02:31 +00:00
|
|
|
| TokenExpander::BuiltinAttr(..)
|
2021-05-04 19:03:16 +00:00
|
|
|
| TokenExpander::BuiltinDerive(..)
|
|
|
|
| TokenExpander::ProcMacro(..) => id,
|
2019-11-17 16:11:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-04 17:29:30 +00:00
|
|
|
pub(crate) fn map_id_up(&self, id: tt::TokenId) -> (tt::TokenId, mbe::Origin) {
|
2019-11-17 16:11:43 +00:00
|
|
|
match self {
|
2021-10-10 18:07:43 +00:00
|
|
|
TokenExpander::DeclarativeMacro { mac, .. } => mac.map_id_up(id),
|
2021-05-04 19:03:16 +00:00
|
|
|
TokenExpander::Builtin(..)
|
2021-06-09 16:02:31 +00:00
|
|
|
| TokenExpander::BuiltinAttr(..)
|
2021-05-04 19:03:16 +00:00
|
|
|
| TokenExpander::BuiltinDerive(..)
|
|
|
|
| TokenExpander::ProcMacro(..) => (id, mbe::Origin::Call),
|
2019-11-10 03:03:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-29 12:20:08 +00:00
|
|
|
// FIXME: rename to ExpandDatabase
|
2019-10-29 11:55:39 +00:00
|
|
|
#[salsa::query_group(AstDatabaseStorage)]
|
|
|
|
pub trait AstDatabase: SourceDatabase {
|
|
|
|
fn ast_id_map(&self, file_id: HirFileId) -> Arc<AstIdMap>;
|
|
|
|
|
2021-06-08 19:51:28 +00:00
|
|
|
/// Main public API -- parses a hir file, not caring whether it's a real
|
2021-05-04 19:40:10 +00:00
|
|
|
/// file or a macro expansion.
|
2019-10-29 11:55:39 +00:00
|
|
|
#[salsa::transparent]
|
|
|
|
fn parse_or_expand(&self, file_id: HirFileId) -> Option<SyntaxNode>;
|
2021-05-04 19:40:10 +00:00
|
|
|
/// Implementation for the macro case.
|
2020-11-24 20:57:51 +00:00
|
|
|
fn parse_macro_expansion(
|
2020-11-24 18:00:23 +00:00
|
|
|
&self,
|
|
|
|
macro_file: MacroFile,
|
2020-11-26 15:48:17 +00:00
|
|
|
) -> ExpandResult<Option<(Parse<SyntaxNode>, Arc<mbe::TokenMap>)>>;
|
2020-11-26 19:09:54 +00:00
|
|
|
|
2021-05-04 19:40:10 +00:00
|
|
|
/// Macro ids. That's probably the tricksiest bit in rust-analyzer, and the
|
|
|
|
/// reason why we use salsa at all.
|
|
|
|
///
|
|
|
|
/// We encode macro definitions into ids of macro calls, this what allows us
|
|
|
|
/// to be incremental.
|
2021-05-04 18:49:00 +00:00
|
|
|
#[salsa::interned]
|
2021-11-14 15:25:40 +00:00
|
|
|
fn intern_macro_call(&self, macro_call: MacroCallLoc) -> MacroCallId;
|
2020-05-14 09:57:37 +00:00
|
|
|
|
2021-05-04 19:40:10 +00:00
|
|
|
/// Lowers syntactic macro call to a token tree representation.
|
2021-05-04 18:49:00 +00:00
|
|
|
#[salsa::transparent]
|
2022-02-09 10:58:52 +00:00
|
|
|
fn macro_arg(
|
|
|
|
&self,
|
|
|
|
id: MacroCallId,
|
2022-02-09 15:30:10 +00:00
|
|
|
) -> Option<Arc<(tt::Subtree, mbe::TokenMap, fixup::SyntaxFixupUndoInfo)>>;
|
2021-05-04 19:40:10 +00:00
|
|
|
/// Extracts syntax node, corresponding to a macro call. That's a firewall
|
|
|
|
/// query, only typing in the macro call itself changes the returned
|
|
|
|
/// subtree.
|
2021-05-04 18:49:00 +00:00
|
|
|
fn macro_arg_text(&self, id: MacroCallId) -> Option<GreenNode>;
|
2021-05-04 19:40:10 +00:00
|
|
|
/// Gets the expander for this macro. This compiles declarative macros, and
|
|
|
|
/// just fetches procedural ones.
|
2021-10-09 12:23:55 +00:00
|
|
|
fn macro_def(&self, id: MacroDefId) -> Result<Arc<TokenExpander>, mbe::ParseError>;
|
2021-05-04 18:49:00 +00:00
|
|
|
|
2021-05-04 19:40:10 +00:00
|
|
|
/// Expand macro call to a token tree. This query is LRUed (we keep 128 or so results in memory)
|
2021-05-04 18:49:00 +00:00
|
|
|
fn macro_expand(&self, macro_call: MacroCallId) -> ExpandResult<Option<Arc<tt::Subtree>>>;
|
2021-05-04 19:40:10 +00:00
|
|
|
/// Special case of the previous query for procedural macros. We can't LRU
|
|
|
|
/// proc macros, since they are not deterministic in general, and
|
|
|
|
/// non-determinism breaks salsa in a very, very, very bad way. @edwin0cheng
|
|
|
|
/// heroically debugged this once!
|
2021-08-20 12:34:46 +00:00
|
|
|
fn expand_proc_macro(&self, call: MacroCallId) -> ExpandResult<tt::Subtree>;
|
2021-05-04 18:49:00 +00:00
|
|
|
/// Firewall query that returns the error from the `macro_expand` query.
|
|
|
|
fn macro_expand_error(&self, macro_call: MacroCallId) -> Option<ExpandError>;
|
2021-01-04 02:53:31 +00:00
|
|
|
|
|
|
|
fn hygiene_frame(&self, file_id: HirFileId) -> Arc<HygieneFrame>;
|
2019-10-29 11:55:39 +00:00
|
|
|
}
|
|
|
|
|
2020-03-08 10:02:14 +00:00
|
|
|
/// This expands the given macro call, but with different arguments. This is
|
|
|
|
/// used for completion, where we want to see what 'would happen' if we insert a
|
|
|
|
/// token. The `token_to_map` mapped down into the expansion, with the mapped
|
|
|
|
/// token returned.
|
2021-05-24 19:21:25 +00:00
|
|
|
pub fn expand_speculative(
|
2020-03-13 15:05:46 +00:00
|
|
|
db: &dyn AstDatabase,
|
2020-03-08 10:02:14 +00:00
|
|
|
actual_macro_call: MacroCallId,
|
2021-09-02 16:54:09 +00:00
|
|
|
speculative_args: &SyntaxNode,
|
2021-05-04 17:36:48 +00:00
|
|
|
token_to_map: SyntaxToken,
|
|
|
|
) -> Option<(SyntaxNode, SyntaxToken)> {
|
2021-11-14 15:25:40 +00:00
|
|
|
let loc = db.lookup_intern_macro_call(actual_macro_call);
|
2021-10-09 12:23:55 +00:00
|
|
|
let macro_def = db.macro_def(loc.def).ok()?;
|
2021-09-13 22:04:04 +00:00
|
|
|
let token_range = token_to_map.text_range();
|
2021-09-02 16:54:09 +00:00
|
|
|
|
2021-09-13 22:04:04 +00:00
|
|
|
// Build the subtree and token mapping for the speculative args
|
2022-03-12 12:04:13 +00:00
|
|
|
let censor = censor_for_macro_input(&loc, speculative_args);
|
|
|
|
let mut fixups = fixup::fixup_syntax(speculative_args);
|
2022-07-16 10:38:33 +00:00
|
|
|
fixups.replace.extend(censor.into_iter().map(|node| (node.into(), Vec::new())));
|
2022-02-09 15:30:10 +00:00
|
|
|
let (mut tt, spec_args_tmap, _) = mbe::syntax_node_to_token_tree_with_modifications(
|
2022-03-12 12:04:13 +00:00
|
|
|
speculative_args,
|
2022-02-09 15:30:10 +00:00
|
|
|
fixups.token_map,
|
|
|
|
fixups.next_id,
|
2022-02-09 11:00:03 +00:00
|
|
|
fixups.replace,
|
|
|
|
fixups.append,
|
|
|
|
);
|
2021-09-13 22:04:04 +00:00
|
|
|
|
|
|
|
let (attr_arg, token_id) = match loc.kind {
|
2022-03-10 19:53:50 +00:00
|
|
|
MacroCallKind::Attr { invoc_attr_index, is_derive, .. } => {
|
|
|
|
let attr = if is_derive {
|
|
|
|
// for pseudo-derive expansion we actually pass the attribute itself only
|
|
|
|
ast::Attr::cast(speculative_args.clone())
|
|
|
|
} else {
|
|
|
|
// Attributes may have an input token tree, build the subtree and map for this as well
|
|
|
|
// then try finding a token id for our token if it is inside this input subtree.
|
|
|
|
let item = ast::Item::cast(speculative_args.clone())?;
|
|
|
|
item.doc_comments_and_attrs().nth(invoc_attr_index as usize).and_then(Either::left)
|
|
|
|
}?;
|
2021-09-13 22:04:04 +00:00
|
|
|
match attr.token_tree() {
|
|
|
|
Some(token_tree) => {
|
|
|
|
let (mut tree, map) = syntax_node_to_token_tree(attr.token_tree()?.syntax());
|
|
|
|
tree.delimiter = None;
|
|
|
|
|
|
|
|
let shift = mbe::Shift::new(&tt);
|
|
|
|
shift.shift_all(&mut tree);
|
|
|
|
|
|
|
|
let token_id = if token_tree.syntax().text_range().contains_range(token_range) {
|
|
|
|
let attr_input_start =
|
|
|
|
token_tree.left_delimiter_token()?.text_range().start();
|
|
|
|
let range = token_range.checked_sub(attr_input_start)?;
|
|
|
|
let token_id = shift.shift(map.token_by_range(range)?);
|
|
|
|
Some(token_id)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
|
|
|
(Some(tree), token_id)
|
|
|
|
}
|
|
|
|
_ => (None, None),
|
2021-09-02 16:54:09 +00:00
|
|
|
}
|
2021-09-13 22:04:04 +00:00
|
|
|
}
|
|
|
|
_ => (None, None),
|
|
|
|
};
|
|
|
|
let token_id = match token_id {
|
|
|
|
Some(token_id) => token_id,
|
|
|
|
// token wasn't inside an attribute input so it has to be in the general macro input
|
|
|
|
None => {
|
|
|
|
let range = token_range.checked_sub(speculative_args.text_range().start())?;
|
|
|
|
let token_id = spec_args_tmap.token_by_range(range)?;
|
|
|
|
macro_def.map_id_down(token_id)
|
|
|
|
}
|
|
|
|
};
|
2021-05-04 19:20:04 +00:00
|
|
|
|
2021-09-13 22:04:04 +00:00
|
|
|
// Do the actual expansion, we need to directly expand the proc macro due to the attribute args
|
|
|
|
// Otherwise the expand query will fetch the non speculative attribute args and pass those instead.
|
2022-03-10 19:53:50 +00:00
|
|
|
let mut speculative_expansion = match loc.def.kind {
|
|
|
|
MacroDefKind::ProcMacro(expander, ..) => {
|
|
|
|
tt.delimiter = None;
|
|
|
|
expander.expand(db, loc.krate, &tt, attr_arg.as_ref())
|
|
|
|
}
|
|
|
|
MacroDefKind::BuiltInAttr(BuiltinAttrExpander::Derive, _) => {
|
|
|
|
pseudo_derive_attr_expansion(&tt, attr_arg.as_ref()?)
|
|
|
|
}
|
|
|
|
_ => macro_def.expand(db, actual_macro_call, &tt),
|
2021-09-02 16:54:09 +00:00
|
|
|
};
|
2021-05-04 19:20:04 +00:00
|
|
|
|
2021-09-05 19:30:06 +00:00
|
|
|
let expand_to = macro_expand_to(db, actual_macro_call);
|
2022-02-09 15:30:10 +00:00
|
|
|
fixup::reverse_fixups(&mut speculative_expansion.value, &spec_args_tmap, &fixups.undo_info);
|
2022-01-24 16:27:39 +00:00
|
|
|
let (node, rev_tmap) = token_tree_to_syntax_node(&speculative_expansion.value, expand_to);
|
2021-05-04 18:49:00 +00:00
|
|
|
|
2022-10-10 20:47:52 +00:00
|
|
|
let syntax_node = node.syntax_node();
|
|
|
|
let token = rev_tmap
|
|
|
|
.ranges_by_token(token_id, token_to_map.kind())
|
|
|
|
.filter_map(|range| syntax_node.covering_element(range).into_token())
|
|
|
|
.min_by_key(|t| {
|
|
|
|
// prefer tokens of the same kind and text
|
|
|
|
// Note the inversion of the score here, as we want to prefer the first token in case
|
|
|
|
// of all tokens having the same score
|
|
|
|
(t.kind() != token_to_map.kind()) as u8 + (t.text() != token_to_map.text()) as u8
|
|
|
|
})?;
|
2020-03-08 10:02:14 +00:00
|
|
|
Some((node.syntax_node(), token))
|
|
|
|
}
|
|
|
|
|
2020-11-24 20:55:08 +00:00
|
|
|
fn ast_id_map(db: &dyn AstDatabase, file_id: HirFileId) -> Arc<AstIdMap> {
|
2021-05-04 18:49:00 +00:00
|
|
|
let map = db.parse_or_expand(file_id).map(|it| AstIdMap::from_source(&it)).unwrap_or_default();
|
2019-10-29 11:55:39 +00:00
|
|
|
Arc::new(map)
|
|
|
|
}
|
|
|
|
|
2021-05-04 18:49:00 +00:00
|
|
|
fn parse_or_expand(db: &dyn AstDatabase, file_id: HirFileId) -> Option<SyntaxNode> {
|
|
|
|
match file_id.0 {
|
|
|
|
HirFileIdRepr::FileId(file_id) => Some(db.parse(file_id).tree().syntax().clone()),
|
|
|
|
HirFileIdRepr::MacroFile(macro_file) => {
|
2022-01-02 16:03:38 +00:00
|
|
|
// FIXME: Note how we convert from `Parse` to `SyntaxNode` here,
|
|
|
|
// forgetting about parse errors.
|
2021-05-04 18:49:00 +00:00
|
|
|
db.parse_macro_expansion(macro_file).value.map(|(it, _)| it.syntax_node())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_macro_expansion(
|
|
|
|
db: &dyn AstDatabase,
|
|
|
|
macro_file: MacroFile,
|
|
|
|
) -> ExpandResult<Option<(Parse<SyntaxNode>, Arc<mbe::TokenMap>)>> {
|
|
|
|
let _p = profile::span("parse_macro_expansion");
|
2021-05-04 19:20:04 +00:00
|
|
|
let result = db.macro_expand(macro_file.macro_call_id);
|
2021-05-04 18:49:00 +00:00
|
|
|
|
|
|
|
if let Some(err) = &result.err {
|
|
|
|
// Note:
|
|
|
|
// The final goal we would like to make all parse_macro success,
|
|
|
|
// such that the following log will not call anyway.
|
2021-11-14 15:25:40 +00:00
|
|
|
let loc: MacroCallLoc = db.lookup_intern_macro_call(macro_file.macro_call_id);
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
let node = loc.kind.to_node(db);
|
2021-05-19 18:19:08 +00:00
|
|
|
|
|
|
|
// collect parent information for warning log
|
|
|
|
let parents =
|
|
|
|
std::iter::successors(loc.kind.file_id().call_node(db), |it| it.file_id.call_node(db))
|
2021-05-04 18:49:00 +00:00
|
|
|
.map(|n| format!("{:#}", n.value))
|
|
|
|
.collect::<Vec<_>>()
|
|
|
|
.join("\n");
|
|
|
|
|
2022-04-11 22:18:49 +00:00
|
|
|
tracing::debug!(
|
2021-05-19 18:19:08 +00:00
|
|
|
"fail on macro_parse: (reason: {:?} macro_call: {:#}) parents: {}",
|
|
|
|
err,
|
|
|
|
node.value,
|
|
|
|
parents
|
|
|
|
);
|
2021-05-04 18:49:00 +00:00
|
|
|
}
|
|
|
|
let tt = match result.value {
|
|
|
|
Some(tt) => tt,
|
|
|
|
None => return ExpandResult { value: None, err: result.err },
|
|
|
|
};
|
|
|
|
|
2021-09-05 19:30:06 +00:00
|
|
|
let expand_to = macro_expand_to(db, macro_file.macro_call_id);
|
2021-05-04 18:49:00 +00:00
|
|
|
|
2021-08-15 12:46:13 +00:00
|
|
|
tracing::debug!("expanded = {}", tt.as_debug_string());
|
2021-09-05 19:30:06 +00:00
|
|
|
tracing::debug!("kind = {:?}", expand_to);
|
2021-05-04 18:49:00 +00:00
|
|
|
|
2022-01-24 16:27:39 +00:00
|
|
|
let (parse, rev_token_map) = token_tree_to_syntax_node(&tt, expand_to);
|
2021-05-04 18:49:00 +00:00
|
|
|
|
2022-04-05 15:59:48 +00:00
|
|
|
ExpandResult { value: Some((parse, Arc::new(rev_token_map))), err: result.err }
|
2021-05-04 18:49:00 +00:00
|
|
|
}
|
|
|
|
|
2022-02-09 10:58:52 +00:00
|
|
|
fn macro_arg(
|
|
|
|
db: &dyn AstDatabase,
|
|
|
|
id: MacroCallId,
|
2022-02-09 15:30:10 +00:00
|
|
|
) -> Option<Arc<(tt::Subtree, mbe::TokenMap, fixup::SyntaxFixupUndoInfo)>> {
|
2021-05-04 18:49:00 +00:00
|
|
|
let arg = db.macro_arg_text(id)?;
|
2021-11-14 15:25:40 +00:00
|
|
|
let loc = db.lookup_intern_macro_call(id);
|
2021-08-25 16:57:24 +00:00
|
|
|
|
|
|
|
let node = SyntaxNode::new_root(arg);
|
2021-09-02 16:54:09 +00:00
|
|
|
let censor = censor_for_macro_input(&loc, &node);
|
2022-02-07 17:08:31 +00:00
|
|
|
let mut fixups = fixup::fixup_syntax(&node);
|
2022-07-16 10:38:33 +00:00
|
|
|
fixups.replace.extend(censor.into_iter().map(|node| (node.into(), Vec::new())));
|
2022-02-09 15:30:10 +00:00
|
|
|
let (mut tt, tmap, _) = mbe::syntax_node_to_token_tree_with_modifications(
|
|
|
|
&node,
|
|
|
|
fixups.token_map,
|
|
|
|
fixups.next_id,
|
|
|
|
fixups.replace,
|
|
|
|
fixups.append,
|
|
|
|
);
|
2022-02-07 17:08:31 +00:00
|
|
|
|
2021-09-02 16:54:09 +00:00
|
|
|
if loc.def.is_proc_macro() {
|
|
|
|
// proc macros expect their inputs without parentheses, MBEs expect it with them included
|
|
|
|
tt.delimiter = None;
|
|
|
|
}
|
|
|
|
|
2022-02-09 15:30:10 +00:00
|
|
|
Some(Arc::new((tt, tmap, fixups.undo_info)))
|
2021-09-02 16:54:09 +00:00
|
|
|
}
|
|
|
|
|
2021-09-19 16:30:29 +00:00
|
|
|
fn censor_for_macro_input(loc: &MacroCallLoc, node: &SyntaxNode) -> FxHashSet<SyntaxNode> {
|
|
|
|
(|| {
|
|
|
|
let censor = match loc.kind {
|
|
|
|
MacroCallKind::FnLike { .. } => return None,
|
2021-10-10 13:50:28 +00:00
|
|
|
MacroCallKind::Derive { derive_attr_index, .. } => {
|
|
|
|
cov_mark::hit!(derive_censoring);
|
|
|
|
ast::Item::cast(node.clone())?
|
|
|
|
.attrs()
|
|
|
|
.take(derive_attr_index as usize + 1)
|
2022-08-15 14:16:59 +00:00
|
|
|
// FIXME, this resolution should not be done syntactically
|
|
|
|
// derive is a proper macro now, no longer builtin
|
|
|
|
// But we do not have resolution at this stage, this means
|
|
|
|
// we need to know about all macro calls for the given ast item here
|
|
|
|
// so we require some kind of mapping...
|
2021-10-10 13:50:28 +00:00
|
|
|
.filter(|attr| attr.simple_name().as_deref() == Some("derive"))
|
|
|
|
.map(|it| it.syntax().clone())
|
|
|
|
.collect()
|
|
|
|
}
|
2022-02-21 01:42:58 +00:00
|
|
|
MacroCallKind::Attr { is_derive: true, .. } => return None,
|
2021-10-10 13:50:28 +00:00
|
|
|
MacroCallKind::Attr { invoc_attr_index, .. } => {
|
|
|
|
cov_mark::hit!(attribute_macro_attr_censoring);
|
|
|
|
ast::Item::cast(node.clone())?
|
2022-01-07 17:51:10 +00:00
|
|
|
.doc_comments_and_attrs()
|
2021-10-10 13:50:28 +00:00
|
|
|
.nth(invoc_attr_index as usize)
|
2022-01-30 21:18:32 +00:00
|
|
|
.and_then(Either::left)
|
2021-10-10 13:50:28 +00:00
|
|
|
.map(|attr| attr.syntax().clone())
|
|
|
|
.into_iter()
|
|
|
|
.collect()
|
|
|
|
}
|
2021-09-19 16:30:29 +00:00
|
|
|
};
|
|
|
|
Some(censor)
|
|
|
|
})()
|
|
|
|
.unwrap_or_default()
|
2021-05-04 18:49:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn macro_arg_text(db: &dyn AstDatabase, id: MacroCallId) -> Option<GreenNode> {
|
2021-11-14 15:25:40 +00:00
|
|
|
let loc = db.lookup_intern_macro_call(id);
|
2021-05-04 18:49:00 +00:00
|
|
|
let arg = loc.kind.arg(db)?;
|
fix: avoid pathological macro expansions
Today, rust-analyzer (and rustc, and bat, and IntelliJ) fail badly on
some kinds of maliciously constructed code, like a deep sequence of
nested parenthesis.
"Who writes 100k nested parenthesis" you'd ask?
Well, in a language with macros, a run-away macro expansion might do
that (see the added tests)! Such expansion can be broad, rather than
deep, so it bypasses recursion check at the macro-expansion layer, but
triggers deep recursion in parser.
In the ideal world, the parser would just handle deeply nested structs
gracefully. We'll get there some day, but at the moment, let's try to be
simple, and just avoid expanding macros with unbalanced parenthesis in
the first place.
closes #9358
2021-08-09 13:06:49 +00:00
|
|
|
if matches!(loc.kind, MacroCallKind::FnLike { .. }) {
|
|
|
|
let first = arg.first_child_or_token().map_or(T![.], |it| it.kind());
|
|
|
|
let last = arg.last_child_or_token().map_or(T![.], |it| it.kind());
|
|
|
|
let well_formed_tt =
|
|
|
|
matches!((first, last), (T!['('], T![')']) | (T!['['], T![']']) | (T!['{'], T!['}']));
|
|
|
|
if !well_formed_tt {
|
|
|
|
// Don't expand malformed (unbalanced) macro invocations. This is
|
|
|
|
// less than ideal, but trying to expand unbalanced macro calls
|
|
|
|
// sometimes produces pathological, deeply nested code which breaks
|
|
|
|
// all kinds of things.
|
|
|
|
//
|
|
|
|
// Some day, we'll have explicit recursion counters for all
|
|
|
|
// recursive things, at which point this code might be removed.
|
|
|
|
cov_mark::hit!(issue9358_bad_macro_stack_overflow);
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
}
|
2021-05-06 05:22:51 +00:00
|
|
|
Some(arg.green().into())
|
2021-05-04 18:49:00 +00:00
|
|
|
}
|
|
|
|
|
2021-10-09 12:23:55 +00:00
|
|
|
fn macro_def(db: &dyn AstDatabase, id: MacroDefId) -> Result<Arc<TokenExpander>, mbe::ParseError> {
|
2019-11-11 10:45:55 +00:00
|
|
|
match id.kind {
|
2021-10-10 18:07:43 +00:00
|
|
|
MacroDefKind::Declarative(ast_id) => {
|
|
|
|
let (mac, def_site_token_map) = match ast_id.to_node(db) {
|
|
|
|
ast::Macro::MacroRules(macro_rules) => {
|
|
|
|
let arg = macro_rules
|
|
|
|
.token_tree()
|
|
|
|
.ok_or_else(|| mbe::ParseError::Expected("expected a token tree".into()))?;
|
|
|
|
let (tt, def_site_token_map) = mbe::syntax_node_to_token_tree(arg.syntax());
|
|
|
|
let mac = mbe::DeclarativeMacro::parse_macro_rules(&tt)?;
|
|
|
|
(mac, def_site_token_map)
|
|
|
|
}
|
|
|
|
ast::Macro::MacroDef(macro_def) => {
|
|
|
|
let arg = macro_def
|
|
|
|
.body()
|
|
|
|
.ok_or_else(|| mbe::ParseError::Expected("expected a token tree".into()))?;
|
|
|
|
let (tt, def_site_token_map) = mbe::syntax_node_to_token_tree(arg.syntax());
|
|
|
|
let mac = mbe::DeclarativeMacro::parse_macro2(&tt)?;
|
|
|
|
(mac, def_site_token_map)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
Ok(Arc::new(TokenExpander::DeclarativeMacro { mac, def_site_token_map }))
|
|
|
|
}
|
2021-10-09 12:23:55 +00:00
|
|
|
MacroDefKind::BuiltIn(expander, _) => Ok(Arc::new(TokenExpander::Builtin(expander))),
|
2021-06-09 16:02:31 +00:00
|
|
|
MacroDefKind::BuiltInAttr(expander, _) => {
|
2021-10-09 12:23:55 +00:00
|
|
|
Ok(Arc::new(TokenExpander::BuiltinAttr(expander)))
|
2021-06-09 16:02:31 +00:00
|
|
|
}
|
2021-03-18 14:37:14 +00:00
|
|
|
MacroDefKind::BuiltInDerive(expander, _) => {
|
2021-10-09 12:23:55 +00:00
|
|
|
Ok(Arc::new(TokenExpander::BuiltinDerive(expander)))
|
2019-11-10 03:03:24 +00:00
|
|
|
}
|
2021-10-09 12:23:55 +00:00
|
|
|
MacroDefKind::BuiltInEager(..) => {
|
|
|
|
// FIXME: Return a random error here just to make the types align.
|
|
|
|
// This obviously should do something real instead.
|
2022-02-03 16:25:24 +00:00
|
|
|
Err(mbe::ParseError::UnexpectedToken("unexpected eager macro".into()))
|
2021-10-09 12:23:55 +00:00
|
|
|
}
|
|
|
|
MacroDefKind::ProcMacro(expander, ..) => Ok(Arc::new(TokenExpander::ProcMacro(expander))),
|
2019-11-10 03:03:24 +00:00
|
|
|
}
|
2019-10-29 12:11:42 +00:00
|
|
|
}
|
|
|
|
|
2020-11-26 15:48:17 +00:00
|
|
|
fn macro_expand(db: &dyn AstDatabase, id: MacroCallId) -> ExpandResult<Option<Arc<tt::Subtree>>> {
|
2020-11-27 17:07:16 +00:00
|
|
|
let _p = profile::span("macro_expand");
|
2021-11-14 15:25:40 +00:00
|
|
|
let loc: MacroCallLoc = db.lookup_intern_macro_call(id);
|
2021-05-19 18:19:08 +00:00
|
|
|
if let Some(eager) = &loc.eager {
|
2021-08-16 11:27:36 +00:00
|
|
|
return ExpandResult {
|
|
|
|
value: Some(eager.arg_or_expansion.clone()),
|
|
|
|
// FIXME: There could be errors here!
|
|
|
|
err: None,
|
|
|
|
};
|
2021-05-19 18:19:08 +00:00
|
|
|
}
|
2020-02-17 11:32:13 +00:00
|
|
|
|
2021-08-16 11:27:36 +00:00
|
|
|
let macro_arg = match db.macro_arg(id) {
|
2020-03-13 12:03:31 +00:00
|
|
|
Some(it) => it,
|
2022-02-21 18:14:06 +00:00
|
|
|
None => {
|
|
|
|
return ExpandResult::only_err(ExpandError::Other(
|
|
|
|
"Failed to lower macro args to token tree".into(),
|
|
|
|
))
|
|
|
|
}
|
2020-03-13 12:03:31 +00:00
|
|
|
};
|
2019-10-29 12:11:42 +00:00
|
|
|
|
2021-09-02 16:54:09 +00:00
|
|
|
let expander = match db.macro_def(loc.def) {
|
2021-10-09 12:23:55 +00:00
|
|
|
Ok(it) => it,
|
|
|
|
// FIXME: This is weird -- we effectively report macro *definition*
|
|
|
|
// errors lazily, when we try to expand the macro. Instead, they should
|
|
|
|
// be reported at the definition site (when we construct a def map).
|
2022-02-21 18:14:06 +00:00
|
|
|
Err(err) => {
|
|
|
|
return ExpandResult::only_err(ExpandError::Other(
|
|
|
|
format!("invalid macro definition: {}", err).into(),
|
|
|
|
))
|
|
|
|
}
|
2020-03-13 12:03:31 +00:00
|
|
|
};
|
2022-02-07 18:53:31 +00:00
|
|
|
let ExpandResult { value: mut tt, err } = expander.expand(db, id, ¯o_arg.0);
|
2019-10-29 12:11:42 +00:00
|
|
|
// Set a hard limit for the expanded tt
|
|
|
|
let count = tt.count();
|
2021-07-10 20:49:17 +00:00
|
|
|
if TOKEN_LIMIT.check(count).is_err() {
|
2022-02-21 18:14:06 +00:00
|
|
|
return ExpandResult::only_err(ExpandError::Other(
|
|
|
|
format!(
|
|
|
|
"macro invocation exceeds token limit: produced {} tokens, limit is {}",
|
|
|
|
count,
|
|
|
|
TOKEN_LIMIT.inner(),
|
|
|
|
)
|
|
|
|
.into(),
|
2020-11-26 15:48:17 +00:00
|
|
|
));
|
2019-10-29 12:11:42 +00:00
|
|
|
}
|
2020-11-24 18:00:23 +00:00
|
|
|
|
2022-02-09 10:58:52 +00:00
|
|
|
fixup::reverse_fixups(&mut tt, ¯o_arg.1, ¯o_arg.2);
|
2022-02-07 18:53:31 +00:00
|
|
|
|
2020-11-26 15:48:17 +00:00
|
|
|
ExpandResult { value: Some(Arc::new(tt)), err }
|
2019-10-29 12:11:42 +00:00
|
|
|
}
|
|
|
|
|
2021-08-16 11:27:36 +00:00
|
|
|
fn macro_expand_error(db: &dyn AstDatabase, macro_call: MacroCallId) -> Option<ExpandError> {
|
|
|
|
db.macro_expand(macro_call).err
|
|
|
|
}
|
|
|
|
|
2021-08-20 12:34:46 +00:00
|
|
|
fn expand_proc_macro(db: &dyn AstDatabase, id: MacroCallId) -> ExpandResult<tt::Subtree> {
|
2021-11-14 15:25:40 +00:00
|
|
|
let loc: MacroCallLoc = db.lookup_intern_macro_call(id);
|
2020-05-14 09:57:37 +00:00
|
|
|
let macro_arg = match db.macro_arg(id) {
|
|
|
|
Some(it) => it,
|
2022-02-21 18:14:06 +00:00
|
|
|
None => {
|
|
|
|
return ExpandResult::only_err(ExpandError::Other("No arguments for proc-macro".into()))
|
|
|
|
}
|
2020-05-14 09:57:37 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
let expander = match loc.def.kind {
|
2021-03-18 15:11:18 +00:00
|
|
|
MacroDefKind::ProcMacro(expander, ..) => expander,
|
2020-05-14 09:57:37 +00:00
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
2021-05-31 11:37:11 +00:00
|
|
|
let attr_arg = match &loc.kind {
|
2021-08-21 16:06:03 +00:00
|
|
|
MacroCallKind::Attr { attr_args, .. } => {
|
2021-08-21 16:19:18 +00:00
|
|
|
let mut attr_args = attr_args.0.clone();
|
2021-08-21 16:06:03 +00:00
|
|
|
mbe::Shift::new(¯o_arg.0).shift_all(&mut attr_args);
|
|
|
|
Some(attr_args)
|
|
|
|
}
|
2021-05-31 11:37:11 +00:00
|
|
|
_ => None,
|
|
|
|
};
|
|
|
|
|
2021-08-21 16:06:03 +00:00
|
|
|
expander.expand(db, loc.krate, ¯o_arg.0, attr_arg.as_ref())
|
2020-05-14 09:57:37 +00:00
|
|
|
}
|
|
|
|
|
2021-01-04 02:53:31 +00:00
|
|
|
fn hygiene_frame(db: &dyn AstDatabase, file_id: HirFileId) -> Arc<HygieneFrame> {
|
|
|
|
Arc::new(HygieneFrame::new(db, file_id))
|
|
|
|
}
|
|
|
|
|
2021-09-05 19:30:06 +00:00
|
|
|
fn macro_expand_to(db: &dyn AstDatabase, id: MacroCallId) -> ExpandTo {
|
2021-11-14 15:25:40 +00:00
|
|
|
let loc: MacroCallLoc = db.lookup_intern_macro_call(id);
|
2021-09-05 19:30:06 +00:00
|
|
|
loc.kind.expand_to()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn token_tree_to_syntax_node(
|
|
|
|
tt: &tt::Subtree,
|
|
|
|
expand_to: ExpandTo,
|
2022-01-24 16:27:39 +00:00
|
|
|
) -> (Parse<SyntaxNode>, mbe::TokenMap) {
|
2021-09-06 15:34:03 +00:00
|
|
|
let entry_point = match expand_to {
|
2021-12-27 14:54:51 +00:00
|
|
|
ExpandTo::Statements => mbe::TopEntryPoint::MacroStmts,
|
|
|
|
ExpandTo::Items => mbe::TopEntryPoint::MacroItems,
|
|
|
|
ExpandTo::Pattern => mbe::TopEntryPoint::Pattern,
|
|
|
|
ExpandTo::Type => mbe::TopEntryPoint::Type,
|
|
|
|
ExpandTo::Expr => mbe::TopEntryPoint::Expr,
|
2021-09-05 19:30:06 +00:00
|
|
|
};
|
2021-09-06 15:34:03 +00:00
|
|
|
mbe::token_tree_to_syntax_node(tt, entry_point)
|
2019-12-08 08:16:52 +00:00
|
|
|
}
|