Make MBE expansion more resilient (WIP)

This commit is contained in:
Florian Diebold 2020-03-13 13:03:31 +01:00 committed by Florian Diebold
parent d3773ec152
commit b973158aeb
10 changed files with 167 additions and 90 deletions

View file

@ -27,11 +27,12 @@ impl TokenExpander {
db: &dyn AstDatabase, db: &dyn AstDatabase,
id: LazyMacroId, id: LazyMacroId,
tt: &tt::Subtree, tt: &tt::Subtree,
) -> Result<tt::Subtree, mbe::ExpandError> { ) -> mbe::ExpandResult<tt::Subtree> {
match self { match self {
TokenExpander::MacroRules(it) => it.expand(tt), TokenExpander::MacroRules(it) => it.expand(tt),
TokenExpander::Builtin(it) => it.expand(db, id, tt), // FIXME switch these to ExpandResult as well
TokenExpander::BuiltinDerive(it) => it.expand(db, id, tt), TokenExpander::Builtin(it) => it.expand(db, id, tt).map_or_else(|e| (tt::Subtree::default(), Some(e)), |r| (r, None)),
TokenExpander::BuiltinDerive(it) => it.expand(db, id, tt).map_or_else(|e| (tt::Subtree::default(), Some(e)), |r| (r, None)),
} }
} }
@ -66,7 +67,7 @@ pub trait AstDatabase: SourceDatabase {
fn macro_def(&self, id: MacroDefId) -> Option<Arc<(TokenExpander, mbe::TokenMap)>>; fn macro_def(&self, id: MacroDefId) -> Option<Arc<(TokenExpander, mbe::TokenMap)>>;
fn parse_macro(&self, macro_file: MacroFile) fn parse_macro(&self, macro_file: MacroFile)
-> Option<(Parse<SyntaxNode>, Arc<mbe::TokenMap>)>; -> Option<(Parse<SyntaxNode>, Arc<mbe::TokenMap>)>;
fn macro_expand(&self, macro_call: MacroCallId) -> Result<Arc<tt::Subtree>, String>; fn macro_expand(&self, macro_call: MacroCallId) -> (Option<Arc<tt::Subtree>>, Option<String>);
#[salsa::interned] #[salsa::interned]
fn intern_eager_expansion(&self, eager: EagerCallLoc) -> EagerMacroId; fn intern_eager_expansion(&self, eager: EagerCallLoc) -> EagerMacroId;
@ -153,7 +154,7 @@ pub(crate) fn macro_arg(
pub(crate) fn macro_expand( pub(crate) fn macro_expand(
db: &dyn AstDatabase, db: &dyn AstDatabase,
id: MacroCallId, id: MacroCallId,
) -> Result<Arc<tt::Subtree>, String> { ) -> (Option<Arc<tt::Subtree>>, Option<String>) {
macro_expand_with_arg(db, id, None) macro_expand_with_arg(db, id, None)
} }
@ -174,31 +175,39 @@ fn macro_expand_with_arg(
db: &dyn AstDatabase, db: &dyn AstDatabase,
id: MacroCallId, id: MacroCallId,
arg: Option<Arc<(tt::Subtree, mbe::TokenMap)>>, arg: Option<Arc<(tt::Subtree, mbe::TokenMap)>>,
) -> Result<Arc<tt::Subtree>, String> { ) -> (Option<Arc<tt::Subtree>>, Option<String>) {
let lazy_id = match id { let lazy_id = match id {
MacroCallId::LazyMacro(id) => id, MacroCallId::LazyMacro(id) => id,
MacroCallId::EagerMacro(id) => { MacroCallId::EagerMacro(id) => {
if arg.is_some() { if arg.is_some() {
return Err( return (
"hypothetical macro expansion not implemented for eager macro".to_owned() None,
Some("hypothetical macro expansion not implemented for eager macro".to_owned())
); );
} else { } else {
return Ok(db.lookup_intern_eager_expansion(id).subtree); return (Some(db.lookup_intern_eager_expansion(id).subtree), None);
} }
} }
}; };
let loc = db.lookup_intern_macro(lazy_id); let loc = db.lookup_intern_macro(lazy_id);
let macro_arg = arg.or_else(|| db.macro_arg(id)).ok_or("Fail to args in to tt::TokenTree")?; let macro_arg = match arg.or_else(|| db.macro_arg(id)) {
Some(it) => it,
None => return (None, Some("Fail to args in to tt::TokenTree".into())),
};
let macro_rules = db.macro_def(loc.def).ok_or("Fail to find macro definition")?; let macro_rules = match db.macro_def(loc.def) {
let tt = macro_rules.0.expand(db, lazy_id, &macro_arg.0).map_err(|err| format!("{:?}", err))?; Some(it) => it,
None => return (None, Some("Fail to find macro definition".into())),
};
let (tt, err) = macro_rules.0.expand(db, lazy_id, &macro_arg.0);
// Set a hard limit for the expanded tt // Set a hard limit for the expanded tt
eprintln!("expansion size: {}", tt.count());
let count = tt.count(); let count = tt.count();
if count > 65536 { if count > 65536 {
return Err(format!("Total tokens count exceed limit : count = {}", count)); return (None, Some(format!("Total tokens count exceed limit : count = {}", count)));
} }
Ok(Arc::new(tt)) (Some(Arc::new(tt)), err.map(|e| format!("{:?}", e)))
} }
pub(crate) fn parse_or_expand(db: &dyn AstDatabase, file_id: HirFileId) -> Option<SyntaxNode> { pub(crate) fn parse_or_expand(db: &dyn AstDatabase, file_id: HirFileId) -> Option<SyntaxNode> {
@ -225,13 +234,12 @@ pub fn parse_macro_with_arg(
let _p = profile("parse_macro_query"); let _p = profile("parse_macro_query");
let macro_call_id = macro_file.macro_call_id; let macro_call_id = macro_file.macro_call_id;
let expansion = if let Some(arg) = arg { let (tt, err) = if let Some(arg) = arg {
macro_expand_with_arg(db, macro_call_id, Some(arg)) macro_expand_with_arg(db, macro_call_id, Some(arg))
} else { } else {
db.macro_expand(macro_call_id) db.macro_expand(macro_call_id)
}; };
let tt = expansion if let Some(err) = err {
.map_err(|err| {
// Note: // Note:
// The final goal we would like to make all parse_macro success, // The final goal we would like to make all parse_macro success,
// such that the following log will not call anyway. // such that the following log will not call anyway.
@ -259,8 +267,8 @@ pub fn parse_macro_with_arg(
log::warn!("fail on macro_parse: (reason: {})", err); log::warn!("fail on macro_parse: (reason: {})", err);
} }
} }
}) };
.ok()?; let tt = tt?;
let fragment_kind = to_fragment_kind(db, macro_call_id); let fragment_kind = to_fragment_kind(db, macro_call_id);

View file

@ -751,6 +751,43 @@ mod tests {
); );
} }
#[test]
fn macro_expansion_resilient() {
assert_debug_snapshot!(
do_ref_completion(
r"
macro_rules! dbg {
() => {};
($val:expr) => {
match $val { tmp => { tmp } }
};
// Trailing comma with single argument is ignored
($val:expr,) => { $crate::dbg!($val) };
($($val:expr),+ $(,)?) => {
($($crate::dbg!($val)),+,)
};
}
struct A { the_field: u32 }
fn foo(a: A) {
dbg!(a.<|>)
}
",
),
@r###"
[
CompletionItem {
label: "the_field",
source_range: [552; 553),
delete: [552; 553),
insert: "the_field",
kind: Field,
detail: "u32",
},
]
"###
);
}
#[test] #[test]
fn test_method_completion_3547() { fn test_method_completion_3547() {
assert_debug_snapshot!( assert_debug_snapshot!(

View file

@ -259,7 +259,7 @@ fn some_thing() -> u32 {
); );
assert_eq!(res.name, "foo"); assert_eq!(res.name, "foo");
assert_snapshot!(res.expansion, @r###"bar!()"###); assert_snapshot!(res.expansion, @r###""###);
} }
#[test] #[test]

View file

@ -30,6 +30,8 @@ pub enum ExpandError {
InvalidRepeat, InvalidRepeat,
} }
pub type ExpandResult<T> = (T, Option<ExpandError>);
pub use crate::syntax_bridge::{ pub use crate::syntax_bridge::{
ast_to_token_tree, parse_to_token_tree, syntax_node_to_token_tree, token_tree_to_syntax_node, ast_to_token_tree, parse_to_token_tree, syntax_node_to_token_tree, token_tree_to_syntax_node,
TokenMap, TokenMap,
@ -150,7 +152,7 @@ impl MacroRules {
Ok(MacroRules { rules, shift: Shift::new(tt) }) Ok(MacroRules { rules, shift: Shift::new(tt) })
} }
pub fn expand(&self, tt: &tt::Subtree) -> Result<tt::Subtree, ExpandError> { pub fn expand(&self, tt: &tt::Subtree) -> ExpandResult<tt::Subtree> {
// apply shift // apply shift
let mut tt = tt.clone(); let mut tt = tt.clone();
self.shift.shift_all(&mut tt); self.shift.shift_all(&mut tt);

View file

@ -8,19 +8,30 @@ mod transcriber;
use ra_syntax::SmolStr; use ra_syntax::SmolStr;
use rustc_hash::FxHashMap; use rustc_hash::FxHashMap;
use crate::ExpandError; use crate::{ExpandResult, ExpandError};
pub(crate) fn expand( pub(crate) fn expand(
rules: &crate::MacroRules, rules: &crate::MacroRules,
input: &tt::Subtree, input: &tt::Subtree,
) -> Result<tt::Subtree, ExpandError> { ) -> ExpandResult<tt::Subtree> {
rules.rules.iter().find_map(|it| expand_rule(it, input).ok()).ok_or(ExpandError::NoMatchingRule) let (mut result, mut err) = (tt::Subtree::default(), Some(ExpandError::NoMatchingRule));
for rule in &rules.rules {
let (res, e) = expand_rule(rule, input);
if e.is_none() {
// if we find a rule that applies without errors, we're done
return (res, None);
}
// TODO decide which result is better
result = res;
err = e;
}
(result, err)
} }
fn expand_rule(rule: &crate::Rule, input: &tt::Subtree) -> Result<tt::Subtree, ExpandError> { fn expand_rule(rule: &crate::Rule, input: &tt::Subtree) -> ExpandResult<tt::Subtree> {
let bindings = matcher::match_(&rule.lhs, input)?; let (bindings, bindings_err) = dbg!(matcher::match_(&rule.lhs, input));
let res = transcriber::transcribe(&rule.rhs, &bindings)?; let (res, transcribe_err) = dbg!(transcriber::transcribe(&rule.rhs, &bindings));
Ok(res) (res, bindings_err.or(transcribe_err))
} }
/// The actual algorithm for expansion is not too hard, but is pretty tricky. /// The actual algorithm for expansion is not too hard, but is pretty tricky.
@ -111,7 +122,7 @@ mod tests {
} }
fn assert_err(macro_body: &str, invocation: &str, err: ExpandError) { fn assert_err(macro_body: &str, invocation: &str, err: ExpandError) {
assert_eq!(expand_first(&create_rules(&format_macro(macro_body)), invocation), Err(err)); assert_eq!(expand_first(&create_rules(&format_macro(macro_body)), invocation).1, Some(err));
} }
fn format_macro(macro_body: &str) -> String { fn format_macro(macro_body: &str) -> String {
@ -138,7 +149,7 @@ mod tests {
fn expand_first( fn expand_first(
rules: &crate::MacroRules, rules: &crate::MacroRules,
invocation: &str, invocation: &str,
) -> Result<tt::Subtree, ExpandError> { ) -> ExpandResult<tt::Subtree> {
let source_file = ast::SourceFile::parse(invocation).ok().unwrap(); let source_file = ast::SourceFile::parse(invocation).ok().unwrap();
let macro_invocation = let macro_invocation =
source_file.syntax().descendants().find_map(ast::MacroCall::cast).unwrap(); source_file.syntax().descendants().find_map(ast::MacroCall::cast).unwrap();

View file

@ -11,6 +11,7 @@ use crate::{
use ra_parser::{FragmentKind::*, TreeSink}; use ra_parser::{FragmentKind::*, TreeSink};
use ra_syntax::{SmolStr, SyntaxKind}; use ra_syntax::{SmolStr, SyntaxKind};
use tt::buffer::{Cursor, TokenBuffer}; use tt::buffer::{Cursor, TokenBuffer};
use super::ExpandResult;
impl Bindings { impl Bindings {
fn push_optional(&mut self, name: &SmolStr) { fn push_optional(&mut self, name: &SmolStr) {
@ -64,19 +65,19 @@ macro_rules! bail {
}; };
} }
pub(super) fn match_(pattern: &tt::Subtree, src: &tt::Subtree) -> Result<Bindings, ExpandError> { pub(super) fn match_(pattern: &tt::Subtree, src: &tt::Subtree) -> ExpandResult<Bindings> {
assert!(pattern.delimiter == None); assert!(pattern.delimiter == None);
let mut res = Bindings::default(); let mut res = Bindings::default();
let mut src = TtIter::new(src); let mut src = TtIter::new(src);
match_subtree(&mut res, pattern, &mut src)?; let mut err = match_subtree(&mut res, pattern, &mut src).err();
if src.len() > 0 { if src.len() > 0 && err.is_none() {
bail!("leftover tokens"); err = Some(err!("leftover tokens"));
} }
Ok(res) (res, err)
} }
fn match_subtree( fn match_subtree(

View file

@ -3,6 +3,7 @@
use ra_syntax::SmolStr; use ra_syntax::SmolStr;
use super::ExpandResult;
use crate::{ use crate::{
mbe_expander::{Binding, Bindings, Fragment}, mbe_expander::{Binding, Bindings, Fragment},
parser::{parse_template, Op, RepeatKind, Separator}, parser::{parse_template, Op, RepeatKind, Separator},
@ -49,10 +50,7 @@ impl Bindings {
} }
} }
pub(super) fn transcribe( pub(super) fn transcribe(template: &tt::Subtree, bindings: &Bindings) -> ExpandResult<tt::Subtree> {
template: &tt::Subtree,
bindings: &Bindings,
) -> Result<tt::Subtree, ExpandError> {
assert!(template.delimiter == None); assert!(template.delimiter == None);
let mut ctx = ExpandCtx { bindings: &bindings, nesting: Vec::new() }; let mut ctx = ExpandCtx { bindings: &bindings, nesting: Vec::new() };
expand_subtree(&mut ctx, template) expand_subtree(&mut ctx, template)
@ -75,35 +73,46 @@ struct ExpandCtx<'a> {
nesting: Vec<NestingState>, nesting: Vec<NestingState>,
} }
fn expand_subtree(ctx: &mut ExpandCtx, template: &tt::Subtree) -> Result<tt::Subtree, ExpandError> { fn expand_subtree(ctx: &mut ExpandCtx, template: &tt::Subtree) -> ExpandResult<tt::Subtree> {
let mut buf: Vec<tt::TokenTree> = Vec::new(); let mut buf: Vec<tt::TokenTree> = Vec::new();
let mut err = None;
for op in parse_template(template) { for op in parse_template(template) {
match op? { let op = match op {
Ok(op) => op,
Err(e) => {
err = Some(e);
break;
}
};
match op {
Op::TokenTree(tt @ tt::TokenTree::Leaf(..)) => buf.push(tt.clone()), Op::TokenTree(tt @ tt::TokenTree::Leaf(..)) => buf.push(tt.clone()),
Op::TokenTree(tt::TokenTree::Subtree(tt)) => { Op::TokenTree(tt::TokenTree::Subtree(tt)) => {
let tt = expand_subtree(ctx, tt)?; let (tt, e) = expand_subtree(ctx, tt);
err = err.or(e);
buf.push(tt.into()); buf.push(tt.into());
} }
Op::Var { name, kind: _ } => { Op::Var { name, kind: _ } => {
let fragment = expand_var(ctx, name)?; let (fragment, e) = expand_var(ctx, name);
err = err.or(e);
push_fragment(&mut buf, fragment); push_fragment(&mut buf, fragment);
} }
Op::Repeat { subtree, kind, separator } => { Op::Repeat { subtree, kind, separator } => {
let fragment = expand_repeat(ctx, subtree, kind, separator)?; let (fragment, e) = expand_repeat(ctx, subtree, kind, separator);
err = err.or(e);
push_fragment(&mut buf, fragment) push_fragment(&mut buf, fragment)
} }
} }
} }
Ok(tt::Subtree { delimiter: template.delimiter, token_trees: buf }) (tt::Subtree { delimiter: template.delimiter, token_trees: buf }, err)
} }
fn expand_var(ctx: &mut ExpandCtx, v: &SmolStr) -> Result<Fragment, ExpandError> { fn expand_var(ctx: &mut ExpandCtx, v: &SmolStr) -> ExpandResult<Fragment> {
let res = if v == "crate" { if v == "crate" {
// We simply produce identifier `$crate` here. And it will be resolved when lowering ast to Path. // We simply produce identifier `$crate` here. And it will be resolved when lowering ast to Path.
let tt = let tt =
tt::Leaf::from(tt::Ident { text: "$crate".into(), id: tt::TokenId::unspecified() }) tt::Leaf::from(tt::Ident { text: "$crate".into(), id: tt::TokenId::unspecified() })
.into(); .into();
Fragment::Tokens(tt) (Fragment::Tokens(tt), None)
} else if !ctx.bindings.contains(v) { } else if !ctx.bindings.contains(v) {
// Note that it is possible to have a `$var` inside a macro which is not bound. // Note that it is possible to have a `$var` inside a macro which is not bound.
// For example: // For example:
@ -132,11 +141,13 @@ fn expand_var(ctx: &mut ExpandCtx, v: &SmolStr) -> Result<Fragment, ExpandError>
], ],
} }
.into(); .into();
Fragment::Tokens(tt) (Fragment::Tokens(tt), None)
} else { } else {
ctx.bindings.get(&v, &mut ctx.nesting)?.clone() ctx.bindings.get(&v, &mut ctx.nesting).map_or_else(
}; |e| (Fragment::Tokens(tt::TokenTree::empty()), Some(e)),
Ok(res) |b| (b.clone(), None),
)
}
} }
fn expand_repeat( fn expand_repeat(
@ -144,17 +155,17 @@ fn expand_repeat(
template: &tt::Subtree, template: &tt::Subtree,
kind: RepeatKind, kind: RepeatKind,
separator: Option<Separator>, separator: Option<Separator>,
) -> Result<Fragment, ExpandError> { ) -> ExpandResult<Fragment> {
let mut buf: Vec<tt::TokenTree> = Vec::new(); let mut buf: Vec<tt::TokenTree> = Vec::new();
ctx.nesting.push(NestingState { idx: 0, at_end: false, hit: false }); ctx.nesting.push(NestingState { idx: 0, at_end: false, hit: false });
// Dirty hack to make macro-expansion terminate. // Dirty hack to make macro-expansion terminate.
// This should be replaced by a propper macro-by-example implementation // This should be replaced by a proper macro-by-example implementation
let limit = 65536; let limit = 65536;
let mut has_seps = 0; let mut has_seps = 0;
let mut counter = 0; let mut counter = 0;
loop { loop {
let res = expand_subtree(ctx, template); let (mut t, e) = expand_subtree(ctx, template);
let nesting_state = ctx.nesting.last_mut().unwrap(); let nesting_state = ctx.nesting.last_mut().unwrap();
if nesting_state.at_end || !nesting_state.hit { if nesting_state.at_end || !nesting_state.hit {
break; break;
@ -172,10 +183,10 @@ fn expand_repeat(
break; break;
} }
let mut t = match res { if e.is_some() {
Ok(t) => t, continue;
Err(_) => continue, }
};
t.delimiter = None; t.delimiter = None;
push_subtree(&mut buf, t); push_subtree(&mut buf, t);
@ -209,14 +220,14 @@ fn expand_repeat(
buf.pop(); buf.pop();
} }
if RepeatKind::OneOrMore == kind && counter == 0 {
return Err(ExpandError::UnexpectedToken);
}
// Check if it is a single token subtree without any delimiter // Check if it is a single token subtree without any delimiter
// e.g {Delimiter:None> ['>'] /Delimiter:None>} // e.g {Delimiter:None> ['>'] /Delimiter:None>}
let tt = tt::Subtree { delimiter: None, token_trees: buf }.into(); let tt = tt::Subtree { delimiter: None, token_trees: buf }.into();
Ok(Fragment::Tokens(tt))
if RepeatKind::OneOrMore == kind && counter == 0 {
return (Fragment::Tokens(tt), Some(ExpandError::UnexpectedToken));
}
(Fragment::Tokens(tt), None)
} }
fn push_fragment(buf: &mut Vec<tt::TokenTree>, fragment: Fragment) { fn push_fragment(buf: &mut Vec<tt::TokenTree>, fragment: Fragment) {

View file

@ -1430,7 +1430,8 @@ impl MacroFixture {
let (invocation_tt, _) = let (invocation_tt, _) =
ast_to_token_tree(&macro_invocation.token_tree().unwrap()).unwrap(); ast_to_token_tree(&macro_invocation.token_tree().unwrap()).unwrap();
self.rules.expand(&invocation_tt) let (tt, err) = self.rules.expand(&invocation_tt);
err.map(Err).unwrap_or(Ok(tt))
} }
fn assert_expand_err(&self, invocation: &str, err: &ExpandError) { fn assert_expand_err(&self, invocation: &str, err: &ExpandError) {

View file

@ -40,6 +40,12 @@ pub enum TokenTree {
} }
impl_froms!(TokenTree: Leaf, Subtree); impl_froms!(TokenTree: Leaf, Subtree);
impl TokenTree {
pub fn empty() -> Self {
TokenTree::Subtree(Subtree::default())
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)] #[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum Leaf { pub enum Leaf {
Literal(Literal), Literal(Literal),