//! Eager expansion related utils //! //! Here is a dump of a discussion from Vadim Petrochenkov about Eager Expansion and //! Its name resolution : //! //! > Eagerly expanded macros (and also macros eagerly expanded by eagerly expanded macros, //! > which actually happens in practice too!) are resolved at the location of the "root" macro //! > that performs the eager expansion on its arguments. //! > If some name cannot be resolved at the eager expansion time it's considered unresolved, //! > even if becomes available later (e.g. from a glob import or other macro). //! //! > Eagerly expanded macros don't add anything to the module structure of the crate and //! > don't build any speculative module structures, i.e. they are expanded in a "flat" //! > way even if tokens in them look like modules. //! //! > In other words, it kinda works for simple cases for which it was originally intended, //! > and we need to live with it because it's available on stable and widely relied upon. //! //! //! See the full discussion : use std::sync::Arc; use base_db::CrateId; use mbe::ExpandResult; use syntax::{ted, SyntaxNode}; use crate::{ ast::{self, AstNode}, db::AstDatabase, EagerCallInfo, ExpandTo, InFile, MacroCallId, MacroCallKind, MacroCallLoc, MacroDefId, MacroDefKind, }; #[derive(Debug)] pub struct ErrorEmitted { _private: (), } pub trait ErrorSink { fn emit(&mut self, err: mbe::ExpandError); fn option( &mut self, opt: Option, error: impl FnOnce() -> mbe::ExpandError, ) -> Result { match opt { Some(it) => Ok(it), None => { self.emit(error()); Err(ErrorEmitted { _private: () }) } } } fn option_with( &mut self, opt: impl FnOnce() -> Option, error: impl FnOnce() -> mbe::ExpandError, ) -> Result { self.option(opt(), error) } fn result(&mut self, res: Result) -> Result { match res { Ok(it) => Ok(it), Err(e) => { self.emit(e); Err(ErrorEmitted { _private: () }) } } } fn expand_result_option(&mut self, res: ExpandResult>) -> Result { match (res.value, res.err) { (None, Some(err)) => { self.emit(err); Err(ErrorEmitted { _private: () }) } (Some(value), opt_err) => { if let Some(err) = opt_err { self.emit(err); } Ok(value) } (None, None) => unreachable!("`ExpandResult` without value or error"), } } } impl ErrorSink for &'_ mut dyn FnMut(mbe::ExpandError) { fn emit(&mut self, err: mbe::ExpandError) { self(err); } } fn err(msg: impl Into) -> mbe::ExpandError { mbe::ExpandError::Other(msg.into()) } pub fn expand_eager_macro( db: &dyn AstDatabase, krate: CrateId, macro_call: InFile, def: MacroDefId, resolver: &dyn Fn(ast::Path) -> Option, mut diagnostic_sink: &mut dyn FnMut(mbe::ExpandError), ) -> Result { let parsed_args = diagnostic_sink.option_with( || Some(mbe::syntax_node_to_token_tree(macro_call.value.token_tree()?.syntax()).0), || err("malformed macro invocation"), )?; let ast_map = db.ast_id_map(macro_call.file_id); let call_id = InFile::new(macro_call.file_id, ast_map.ast_id(¯o_call.value)); let expand_to = ExpandTo::from_call_site(¯o_call.value); // Note: // When `lazy_expand` is called, its *parent* file must be already exists. // Here we store an eager macro id for the argument expanded subtree here // for that purpose. let arg_id = db.intern_macro(MacroCallLoc { def, krate, eager: Some(EagerCallInfo { arg_or_expansion: Arc::new(parsed_args.clone()), included_file: None, }), kind: MacroCallKind::FnLike { ast_id: call_id, expand_to: ExpandTo::Expr }, }); let arg_file_id = arg_id; let parsed_args = diagnostic_sink .result(mbe::token_tree_to_syntax_node(&parsed_args, mbe::ParserEntryPoint::Expr))? .0; let result = eager_macro_recur( db, InFile::new(arg_file_id.as_file(), parsed_args.syntax_node()), krate, resolver, diagnostic_sink, )?; let subtree = diagnostic_sink.option(to_subtree(&result), || err("failed to parse macro result"))?; if let MacroDefKind::BuiltInEager(eager, _) = def.kind { let res = eager.expand(db, arg_id, &subtree); let expanded = diagnostic_sink.expand_result_option(res)?; let loc = MacroCallLoc { def, krate, eager: Some(EagerCallInfo { arg_or_expansion: Arc::new(expanded.subtree), included_file: expanded.included_file, }), kind: MacroCallKind::FnLike { ast_id: call_id, expand_to }, }; Ok(db.intern_macro(loc)) } else { panic!("called `expand_eager_macro` on non-eager macro def {:?}", def); } } fn to_subtree(node: &SyntaxNode) -> Option { let mut subtree = mbe::syntax_node_to_token_tree(node).0; subtree.delimiter = None; Some(subtree) } fn lazy_expand( db: &dyn AstDatabase, def: &MacroDefId, macro_call: InFile, krate: CrateId, ) -> ExpandResult>> { let ast_id = db.ast_id_map(macro_call.file_id).ast_id(¯o_call.value); let expand_to = ExpandTo::from_call_site(¯o_call.value); let id = def.as_lazy_macro( db, krate, MacroCallKind::FnLike { ast_id: macro_call.with_value(ast_id), expand_to }, ); let err = db.macro_expand_error(id); let value = db.parse_or_expand(id.as_file()).map(|node| InFile::new(id.as_file(), node)); ExpandResult { value, err } } fn eager_macro_recur( db: &dyn AstDatabase, curr: InFile, krate: CrateId, macro_resolver: &dyn Fn(ast::Path) -> Option, mut diagnostic_sink: &mut dyn FnMut(mbe::ExpandError), ) -> Result { let original = curr.value.clone_for_update(); let children = original.descendants().filter_map(ast::MacroCall::cast); let mut replacements = Vec::new(); // Collect replacement for child in children { let def = diagnostic_sink.option_with( || macro_resolver(child.path()?), || { let path = child.path().map(|path| format!(" `{}!`", path)).unwrap_or_default(); err(format!("failed to resolve macro{}", path)) }, )?; let insert = match def.kind { MacroDefKind::BuiltInEager(..) => { let id = expand_eager_macro( db, krate, curr.with_value(child.clone()), def, macro_resolver, diagnostic_sink, )?; db.parse_or_expand(id.as_file()) .expect("successful macro expansion should be parseable") .clone_for_update() } MacroDefKind::Declarative(_) | MacroDefKind::BuiltIn(..) | MacroDefKind::BuiltInAttr(..) | MacroDefKind::BuiltInDerive(..) | MacroDefKind::ProcMacro(..) => { let res = lazy_expand(db, &def, curr.with_value(child.clone()), krate); let val = diagnostic_sink.expand_result_option(res)?; // replace macro inside eager_macro_recur(db, val, krate, macro_resolver, diagnostic_sink)? } }; // check if the whole original syntax is replaced if child.syntax() == &original { return Ok(insert); } replacements.push((child, insert)); } replacements.into_iter().rev().for_each(|(old, new)| ted::replace(old.syntax(), new)); Ok(original) }