2019-02-07 15:56:05 +00:00
|
|
|
use errors::DiagnosticBuilder;
|
2019-02-04 12:49:54 +00:00
|
|
|
|
2018-12-04 19:10:32 +00:00
|
|
|
use syntax::ast::{self, *};
|
2018-08-18 10:14:03 +00:00
|
|
|
use syntax::source_map::Spanned;
|
2018-03-07 07:13:15 +00:00
|
|
|
use syntax::ext::base::*;
|
|
|
|
use syntax::ext::build::AstBuilder;
|
|
|
|
use syntax::parse::token;
|
|
|
|
use syntax::print::pprust;
|
2018-12-04 19:10:32 +00:00
|
|
|
use syntax::ptr::P;
|
2018-03-19 00:54:56 +00:00
|
|
|
use syntax::symbol::Symbol;
|
2018-03-07 07:13:15 +00:00
|
|
|
use syntax::tokenstream::{TokenStream, TokenTree};
|
2018-03-14 09:11:42 +00:00
|
|
|
use syntax_pos::{Span, DUMMY_SP};
|
2018-03-07 07:13:15 +00:00
|
|
|
|
|
|
|
pub fn expand_assert<'cx>(
|
2019-02-04 12:49:54 +00:00
|
|
|
cx: &'cx mut ExtCtxt<'_>,
|
2018-03-07 07:13:15 +00:00
|
|
|
sp: Span,
|
|
|
|
tts: &[TokenTree],
|
2018-07-12 09:58:16 +00:00
|
|
|
) -> Box<dyn MacResult + 'cx> {
|
2018-12-04 19:10:32 +00:00
|
|
|
let Assert { cond_expr, custom_message } = match parse_assert(cx, sp, tts) {
|
|
|
|
Ok(assert) => assert,
|
|
|
|
Err(mut err) => {
|
|
|
|
err.emit();
|
|
|
|
return DummyResult::expr(sp);
|
2018-03-07 07:13:15 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-03-18 20:51:53 +00:00
|
|
|
let sp = sp.apply_mark(cx.current_expansion.mark);
|
2018-03-07 07:13:15 +00:00
|
|
|
let panic_call = Mac_ {
|
2018-03-19 00:54:56 +00:00
|
|
|
path: Path::from_ident(Ident::new(Symbol::intern("panic"), sp)),
|
2018-12-04 19:10:32 +00:00
|
|
|
tts: custom_message.unwrap_or_else(|| {
|
2018-03-14 09:11:42 +00:00
|
|
|
TokenStream::from(TokenTree::Token(
|
|
|
|
DUMMY_SP,
|
|
|
|
token::Literal(
|
2018-05-06 01:55:10 +00:00
|
|
|
token::Lit::Str_(Name::intern(&format!(
|
|
|
|
"assertion failed: {}",
|
|
|
|
pprust::expr_to_string(&cond_expr).escape_debug()
|
|
|
|
))),
|
2018-03-14 09:11:42 +00:00
|
|
|
None,
|
|
|
|
),
|
2018-12-04 19:10:32 +00:00
|
|
|
))
|
|
|
|
}).into(),
|
2018-05-22 15:01:21 +00:00
|
|
|
delim: MacDelimiter::Parenthesis,
|
2018-03-07 07:13:15 +00:00
|
|
|
};
|
|
|
|
let if_expr = cx.expr_if(
|
|
|
|
sp,
|
|
|
|
cx.expr(sp, ExprKind::Unary(UnOp::Not, cond_expr)),
|
|
|
|
cx.expr(
|
|
|
|
sp,
|
|
|
|
ExprKind::Mac(Spanned {
|
|
|
|
span: sp,
|
|
|
|
node: panic_call,
|
|
|
|
}),
|
|
|
|
),
|
|
|
|
None,
|
|
|
|
);
|
|
|
|
MacEager::expr(if_expr)
|
|
|
|
}
|
2018-12-04 19:10:32 +00:00
|
|
|
|
|
|
|
struct Assert {
|
|
|
|
cond_expr: P<ast::Expr>,
|
|
|
|
custom_message: Option<TokenStream>,
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_assert<'a>(
|
|
|
|
cx: &mut ExtCtxt<'a>,
|
|
|
|
sp: Span,
|
|
|
|
tts: &[TokenTree]
|
|
|
|
) -> Result<Assert, DiagnosticBuilder<'a>> {
|
|
|
|
let mut parser = cx.new_parser_from_tts(tts);
|
|
|
|
|
|
|
|
if parser.token == token::Eof {
|
|
|
|
let mut err = cx.struct_span_err(sp, "macro requires a boolean expression as an argument");
|
|
|
|
err.span_label(sp, "boolean expression required");
|
|
|
|
return Err(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(Assert {
|
|
|
|
cond_expr: parser.parse_expr()?,
|
|
|
|
custom_message: if parser.eat(&token::Comma) {
|
|
|
|
let ts = parser.parse_tokens();
|
|
|
|
if !ts.is_empty() {
|
|
|
|
Some(ts)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|