3e4d3d2a29
This feature is aimed at giving proc macros access to powers similar to those used by builtin macros such as `format_args!` or `concat!`. These macros are able to accept macros in place of string literal parameters, such as the format string, as they perform recursive macro expansion while being expanded. This can be especially useful in many cases thanks to helper macros like `concat!`, `stringify!` and `include_str!` which are often used to construct string literals at compile-time in user code. For now, this method only allows expanding macros which produce literals, although more expresisons will be supported before the method is stabilized.
968 lines
36 KiB
Rust
968 lines
36 KiB
Rust
use crate::base::ExtCtxt;
|
|
|
|
use rustc_ast as ast;
|
|
use rustc_ast::token::{self, Nonterminal, NtIdent};
|
|
use rustc_ast::tokenstream::{self, CanSynthesizeMissingTokens};
|
|
use rustc_ast::tokenstream::{DelimSpan, Spacing::*, TokenStream, TreeAndSpacing};
|
|
use rustc_ast_pretty::pprust;
|
|
use rustc_data_structures::fx::FxHashMap;
|
|
use rustc_data_structures::sync::Lrc;
|
|
use rustc_errors::{Diagnostic, PResult};
|
|
use rustc_lint_defs::builtin::PROC_MACRO_BACK_COMPAT;
|
|
use rustc_lint_defs::BuiltinLintDiagnostics;
|
|
use rustc_parse::lexer::nfc_normalize;
|
|
use rustc_parse::{nt_to_tokenstream, parse_stream_from_source_str};
|
|
use rustc_session::parse::ParseSess;
|
|
use rustc_span::def_id::CrateNum;
|
|
use rustc_span::hygiene::ExpnKind;
|
|
use rustc_span::symbol::{self, kw, sym, Symbol};
|
|
use rustc_span::{BytePos, FileName, MultiSpan, Pos, RealFileName, SourceFile, Span};
|
|
|
|
use pm::bridge::{server, TokenTree};
|
|
use pm::{Delimiter, Level, LineColumn, Spacing};
|
|
use std::ops::Bound;
|
|
use std::{ascii, panic};
|
|
|
|
trait FromInternal<T> {
|
|
fn from_internal(x: T) -> Self;
|
|
}
|
|
|
|
trait ToInternal<T> {
|
|
fn to_internal(self) -> T;
|
|
}
|
|
|
|
impl FromInternal<token::DelimToken> for Delimiter {
|
|
fn from_internal(delim: token::DelimToken) -> Delimiter {
|
|
match delim {
|
|
token::Paren => Delimiter::Parenthesis,
|
|
token::Brace => Delimiter::Brace,
|
|
token::Bracket => Delimiter::Bracket,
|
|
token::NoDelim => Delimiter::None,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl ToInternal<token::DelimToken> for Delimiter {
|
|
fn to_internal(self) -> token::DelimToken {
|
|
match self {
|
|
Delimiter::Parenthesis => token::Paren,
|
|
Delimiter::Brace => token::Brace,
|
|
Delimiter::Bracket => token::Bracket,
|
|
Delimiter::None => token::NoDelim,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl FromInternal<(TreeAndSpacing, &'_ mut Vec<Self>, &mut Rustc<'_, '_>)>
|
|
for TokenTree<Group, Punct, Ident, Literal>
|
|
{
|
|
fn from_internal(
|
|
((tree, spacing), stack, rustc): (TreeAndSpacing, &mut Vec<Self>, &mut Rustc<'_, '_>),
|
|
) -> Self {
|
|
use rustc_ast::token::*;
|
|
|
|
let joint = spacing == Joint;
|
|
let Token { kind, span } = match tree {
|
|
tokenstream::TokenTree::Delimited(span, delim, tts) => {
|
|
let delimiter = Delimiter::from_internal(delim);
|
|
return TokenTree::Group(Group { delimiter, stream: tts, span, flatten: false });
|
|
}
|
|
tokenstream::TokenTree::Token(token) => token,
|
|
};
|
|
|
|
macro_rules! tt {
|
|
($ty:ident { $($field:ident $(: $value:expr)*),+ $(,)? }) => (
|
|
TokenTree::$ty(self::$ty {
|
|
$($field $(: $value)*,)+
|
|
span,
|
|
})
|
|
);
|
|
($ty:ident::$method:ident($($value:expr),*)) => (
|
|
TokenTree::$ty(self::$ty::$method($($value,)* span))
|
|
);
|
|
}
|
|
macro_rules! op {
|
|
($a:expr) => {
|
|
tt!(Punct::new($a, joint))
|
|
};
|
|
($a:expr, $b:expr) => {{
|
|
stack.push(tt!(Punct::new($b, joint)));
|
|
tt!(Punct::new($a, true))
|
|
}};
|
|
($a:expr, $b:expr, $c:expr) => {{
|
|
stack.push(tt!(Punct::new($c, joint)));
|
|
stack.push(tt!(Punct::new($b, true)));
|
|
tt!(Punct::new($a, true))
|
|
}};
|
|
}
|
|
|
|
match kind {
|
|
Eq => op!('='),
|
|
Lt => op!('<'),
|
|
Le => op!('<', '='),
|
|
EqEq => op!('=', '='),
|
|
Ne => op!('!', '='),
|
|
Ge => op!('>', '='),
|
|
Gt => op!('>'),
|
|
AndAnd => op!('&', '&'),
|
|
OrOr => op!('|', '|'),
|
|
Not => op!('!'),
|
|
Tilde => op!('~'),
|
|
BinOp(Plus) => op!('+'),
|
|
BinOp(Minus) => op!('-'),
|
|
BinOp(Star) => op!('*'),
|
|
BinOp(Slash) => op!('/'),
|
|
BinOp(Percent) => op!('%'),
|
|
BinOp(Caret) => op!('^'),
|
|
BinOp(And) => op!('&'),
|
|
BinOp(Or) => op!('|'),
|
|
BinOp(Shl) => op!('<', '<'),
|
|
BinOp(Shr) => op!('>', '>'),
|
|
BinOpEq(Plus) => op!('+', '='),
|
|
BinOpEq(Minus) => op!('-', '='),
|
|
BinOpEq(Star) => op!('*', '='),
|
|
BinOpEq(Slash) => op!('/', '='),
|
|
BinOpEq(Percent) => op!('%', '='),
|
|
BinOpEq(Caret) => op!('^', '='),
|
|
BinOpEq(And) => op!('&', '='),
|
|
BinOpEq(Or) => op!('|', '='),
|
|
BinOpEq(Shl) => op!('<', '<', '='),
|
|
BinOpEq(Shr) => op!('>', '>', '='),
|
|
At => op!('@'),
|
|
Dot => op!('.'),
|
|
DotDot => op!('.', '.'),
|
|
DotDotDot => op!('.', '.', '.'),
|
|
DotDotEq => op!('.', '.', '='),
|
|
Comma => op!(','),
|
|
Semi => op!(';'),
|
|
Colon => op!(':'),
|
|
ModSep => op!(':', ':'),
|
|
RArrow => op!('-', '>'),
|
|
LArrow => op!('<', '-'),
|
|
FatArrow => op!('=', '>'),
|
|
Pound => op!('#'),
|
|
Dollar => op!('$'),
|
|
Question => op!('?'),
|
|
SingleQuote => op!('\''),
|
|
|
|
Ident(name, false) if name == kw::DollarCrate => tt!(Ident::dollar_crate()),
|
|
Ident(name, is_raw) => tt!(Ident::new(rustc.sess(), name, is_raw)),
|
|
Lifetime(name) => {
|
|
let ident = symbol::Ident::new(name, span).without_first_quote();
|
|
stack.push(tt!(Ident::new(rustc.sess(), ident.name, false)));
|
|
tt!(Punct::new('\'', true))
|
|
}
|
|
Literal(lit) => tt!(Literal { lit }),
|
|
DocComment(_, attr_style, data) => {
|
|
let mut escaped = String::new();
|
|
for ch in data.as_str().chars() {
|
|
escaped.extend(ch.escape_debug());
|
|
}
|
|
let stream = vec![
|
|
Ident(sym::doc, false),
|
|
Eq,
|
|
TokenKind::lit(token::Str, Symbol::intern(&escaped), None),
|
|
]
|
|
.into_iter()
|
|
.map(|kind| tokenstream::TokenTree::token(kind, span))
|
|
.collect();
|
|
stack.push(TokenTree::Group(Group {
|
|
delimiter: Delimiter::Bracket,
|
|
stream,
|
|
span: DelimSpan::from_single(span),
|
|
flatten: false,
|
|
}));
|
|
if attr_style == ast::AttrStyle::Inner {
|
|
stack.push(tt!(Punct::new('!', false)));
|
|
}
|
|
tt!(Punct::new('#', false))
|
|
}
|
|
|
|
Interpolated(nt)
|
|
if let Some((name, is_raw)) = ident_name_compatibility_hack(&nt, span, rustc) =>
|
|
{
|
|
TokenTree::Ident(Ident::new(rustc.sess(), name.name, is_raw, name.span))
|
|
}
|
|
Interpolated(nt) => {
|
|
let stream = nt_to_tokenstream(&nt, rustc.sess(), CanSynthesizeMissingTokens::No);
|
|
TokenTree::Group(Group {
|
|
delimiter: Delimiter::None,
|
|
stream,
|
|
span: DelimSpan::from_single(span),
|
|
flatten: crate::base::pretty_printing_compatibility_hack(&nt, rustc.sess()),
|
|
})
|
|
}
|
|
|
|
OpenDelim(..) | CloseDelim(..) => unreachable!(),
|
|
Eof => unreachable!(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl ToInternal<TokenStream> for TokenTree<Group, Punct, Ident, Literal> {
|
|
fn to_internal(self) -> TokenStream {
|
|
use rustc_ast::token::*;
|
|
|
|
let (ch, joint, span) = match self {
|
|
TokenTree::Punct(Punct { ch, joint, span }) => (ch, joint, span),
|
|
TokenTree::Group(Group { delimiter, stream, span, .. }) => {
|
|
return tokenstream::TokenTree::Delimited(span, delimiter.to_internal(), stream)
|
|
.into();
|
|
}
|
|
TokenTree::Ident(self::Ident { sym, is_raw, span }) => {
|
|
return tokenstream::TokenTree::token(Ident(sym, is_raw), span).into();
|
|
}
|
|
TokenTree::Literal(self::Literal {
|
|
lit: token::Lit { kind: token::Integer, symbol, suffix },
|
|
span,
|
|
}) if symbol.as_str().starts_with('-') => {
|
|
let minus = BinOp(BinOpToken::Minus);
|
|
let symbol = Symbol::intern(&symbol.as_str()[1..]);
|
|
let integer = TokenKind::lit(token::Integer, symbol, suffix);
|
|
let a = tokenstream::TokenTree::token(minus, span);
|
|
let b = tokenstream::TokenTree::token(integer, span);
|
|
return vec![a, b].into_iter().collect();
|
|
}
|
|
TokenTree::Literal(self::Literal {
|
|
lit: token::Lit { kind: token::Float, symbol, suffix },
|
|
span,
|
|
}) if symbol.as_str().starts_with('-') => {
|
|
let minus = BinOp(BinOpToken::Minus);
|
|
let symbol = Symbol::intern(&symbol.as_str()[1..]);
|
|
let float = TokenKind::lit(token::Float, symbol, suffix);
|
|
let a = tokenstream::TokenTree::token(minus, span);
|
|
let b = tokenstream::TokenTree::token(float, span);
|
|
return vec![a, b].into_iter().collect();
|
|
}
|
|
TokenTree::Literal(self::Literal { lit, span }) => {
|
|
return tokenstream::TokenTree::token(Literal(lit), span).into();
|
|
}
|
|
};
|
|
|
|
let kind = match ch {
|
|
'=' => Eq,
|
|
'<' => Lt,
|
|
'>' => Gt,
|
|
'!' => Not,
|
|
'~' => Tilde,
|
|
'+' => BinOp(Plus),
|
|
'-' => BinOp(Minus),
|
|
'*' => BinOp(Star),
|
|
'/' => BinOp(Slash),
|
|
'%' => BinOp(Percent),
|
|
'^' => BinOp(Caret),
|
|
'&' => BinOp(And),
|
|
'|' => BinOp(Or),
|
|
'@' => At,
|
|
'.' => Dot,
|
|
',' => Comma,
|
|
';' => Semi,
|
|
':' => Colon,
|
|
'#' => Pound,
|
|
'$' => Dollar,
|
|
'?' => Question,
|
|
'\'' => SingleQuote,
|
|
_ => unreachable!(),
|
|
};
|
|
|
|
let tree = tokenstream::TokenTree::token(kind, span);
|
|
TokenStream::new(vec![(tree, if joint { Joint } else { Alone })])
|
|
}
|
|
}
|
|
|
|
impl ToInternal<rustc_errors::Level> for Level {
|
|
fn to_internal(self) -> rustc_errors::Level {
|
|
match self {
|
|
Level::Error => rustc_errors::Level::Error { lint: false },
|
|
Level::Warning => rustc_errors::Level::Warning,
|
|
Level::Note => rustc_errors::Level::Note,
|
|
Level::Help => rustc_errors::Level::Help,
|
|
_ => unreachable!("unknown proc_macro::Level variant: {:?}", self),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct FreeFunctions;
|
|
|
|
#[derive(Clone)]
|
|
pub struct TokenStreamIter {
|
|
cursor: tokenstream::Cursor,
|
|
stack: Vec<TokenTree<Group, Punct, Ident, Literal>>,
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
pub struct Group {
|
|
delimiter: Delimiter,
|
|
stream: TokenStream,
|
|
span: DelimSpan,
|
|
/// A hack used to pass AST fragments to attribute and derive macros
|
|
/// as a single nonterminal token instead of a token stream.
|
|
/// FIXME: It needs to be removed, but there are some compatibility issues (see #73345).
|
|
flatten: bool,
|
|
}
|
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash)]
|
|
pub struct Punct {
|
|
ch: char,
|
|
// NB. not using `Spacing` here because it doesn't implement `Hash`.
|
|
joint: bool,
|
|
span: Span,
|
|
}
|
|
|
|
impl Punct {
|
|
fn new(ch: char, joint: bool, span: Span) -> Punct {
|
|
const LEGAL_CHARS: &[char] = &[
|
|
'=', '<', '>', '!', '~', '+', '-', '*', '/', '%', '^', '&', '|', '@', '.', ',', ';',
|
|
':', '#', '$', '?', '\'',
|
|
];
|
|
if !LEGAL_CHARS.contains(&ch) {
|
|
panic!("unsupported character `{:?}`", ch)
|
|
}
|
|
Punct { ch, joint, span }
|
|
}
|
|
}
|
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash)]
|
|
pub struct Ident {
|
|
sym: Symbol,
|
|
is_raw: bool,
|
|
span: Span,
|
|
}
|
|
|
|
impl Ident {
|
|
fn new(sess: &ParseSess, sym: Symbol, is_raw: bool, span: Span) -> Ident {
|
|
let sym = nfc_normalize(&sym.as_str());
|
|
let string = sym.as_str();
|
|
if !rustc_lexer::is_ident(&string) {
|
|
panic!("`{:?}` is not a valid identifier", string)
|
|
}
|
|
if is_raw && !sym.can_be_raw() {
|
|
panic!("`{}` cannot be a raw identifier", string);
|
|
}
|
|
sess.symbol_gallery.insert(sym, span);
|
|
Ident { sym, is_raw, span }
|
|
}
|
|
fn dollar_crate(span: Span) -> Ident {
|
|
// `$crate` is accepted as an ident only if it comes from the compiler.
|
|
Ident { sym: kw::DollarCrate, is_raw: false, span }
|
|
}
|
|
}
|
|
|
|
// FIXME(eddyb) `Literal` should not expose internal `Debug` impls.
|
|
#[derive(Clone, Debug)]
|
|
pub struct Literal {
|
|
lit: token::Lit,
|
|
span: Span,
|
|
}
|
|
|
|
pub(crate) struct Rustc<'a, 'b> {
|
|
ecx: &'a mut ExtCtxt<'b>,
|
|
def_site: Span,
|
|
call_site: Span,
|
|
mixed_site: Span,
|
|
krate: CrateNum,
|
|
rebased_spans: FxHashMap<usize, Span>,
|
|
}
|
|
|
|
impl<'a, 'b> Rustc<'a, 'b> {
|
|
pub fn new(ecx: &'a mut ExtCtxt<'b>) -> Self {
|
|
let expn_data = ecx.current_expansion.id.expn_data();
|
|
Rustc {
|
|
def_site: ecx.with_def_site_ctxt(expn_data.def_site),
|
|
call_site: ecx.with_call_site_ctxt(expn_data.call_site),
|
|
mixed_site: ecx.with_mixed_site_ctxt(expn_data.call_site),
|
|
krate: expn_data.macro_def_id.unwrap().krate,
|
|
rebased_spans: FxHashMap::default(),
|
|
ecx,
|
|
}
|
|
}
|
|
|
|
fn sess(&self) -> &ParseSess {
|
|
self.ecx.parse_sess()
|
|
}
|
|
|
|
fn lit(&mut self, kind: token::LitKind, symbol: Symbol, suffix: Option<Symbol>) -> Literal {
|
|
Literal { lit: token::Lit::new(kind, symbol, suffix), span: server::Span::call_site(self) }
|
|
}
|
|
}
|
|
|
|
impl server::Types for Rustc<'_, '_> {
|
|
type FreeFunctions = FreeFunctions;
|
|
type TokenStream = TokenStream;
|
|
type TokenStreamBuilder = tokenstream::TokenStreamBuilder;
|
|
type TokenStreamIter = TokenStreamIter;
|
|
type Group = Group;
|
|
type Punct = Punct;
|
|
type Ident = Ident;
|
|
type Literal = Literal;
|
|
type SourceFile = Lrc<SourceFile>;
|
|
type MultiSpan = Vec<Span>;
|
|
type Diagnostic = Diagnostic;
|
|
type Span = Span;
|
|
}
|
|
|
|
impl server::FreeFunctions for Rustc<'_, '_> {
|
|
fn track_env_var(&mut self, var: &str, value: Option<&str>) {
|
|
self.sess()
|
|
.env_depinfo
|
|
.borrow_mut()
|
|
.insert((Symbol::intern(var), value.map(Symbol::intern)));
|
|
}
|
|
|
|
fn track_path(&mut self, path: &str) {
|
|
self.sess().file_depinfo.borrow_mut().insert(Symbol::intern(path));
|
|
}
|
|
}
|
|
|
|
impl server::TokenStream for Rustc<'_, '_> {
|
|
fn new(&mut self) -> Self::TokenStream {
|
|
TokenStream::default()
|
|
}
|
|
fn is_empty(&mut self, stream: &Self::TokenStream) -> bool {
|
|
stream.is_empty()
|
|
}
|
|
fn from_str(&mut self, src: &str) -> Self::TokenStream {
|
|
parse_stream_from_source_str(
|
|
FileName::proc_macro_source_code(src),
|
|
src.to_string(),
|
|
self.sess(),
|
|
Some(self.call_site),
|
|
)
|
|
}
|
|
fn to_string(&mut self, stream: &Self::TokenStream) -> String {
|
|
pprust::tts_to_string(stream)
|
|
}
|
|
fn expand_expr(&mut self, stream: &Self::TokenStream) -> Result<Self::TokenStream, ()> {
|
|
// Parse the expression from our tokenstream.
|
|
let expr: PResult<'_, _> = try {
|
|
let mut p = rustc_parse::stream_to_parser(
|
|
self.sess(),
|
|
stream.clone(),
|
|
Some("proc_macro expand expr"),
|
|
);
|
|
let expr = p.parse_expr()?;
|
|
if p.token != token::Eof {
|
|
p.unexpected()?;
|
|
}
|
|
expr
|
|
};
|
|
let expr = expr.map_err(|mut err| err.emit())?;
|
|
|
|
// Perform eager expansion on the expression.
|
|
let expr = self
|
|
.ecx
|
|
.expander()
|
|
.fully_expand_fragment(crate::expand::AstFragment::Expr(expr))
|
|
.make_expr();
|
|
|
|
// NOTE: For now, limit `expand_expr` to exclusively expand to literals.
|
|
// This may be relaxed in the future.
|
|
// We don't use `nt_to_tokenstream` as the tokenstream currently cannot
|
|
// be recovered in the general case.
|
|
match &expr.kind {
|
|
ast::ExprKind::Lit(l) => {
|
|
Ok(tokenstream::TokenTree::token(token::Literal(l.token), l.span).into())
|
|
}
|
|
ast::ExprKind::Unary(ast::UnOp::Neg, e) => match &e.kind {
|
|
ast::ExprKind::Lit(l) => match l.token {
|
|
token::Lit { kind: token::Integer | token::Float, .. } => {
|
|
Ok(std::array::IntoIter::new([
|
|
// FIXME: The span of the `-` token is lost when
|
|
// parsing, so we cannot faithfully recover it here.
|
|
tokenstream::TokenTree::token(token::BinOp(token::Minus), e.span),
|
|
tokenstream::TokenTree::token(token::Literal(l.token), l.span),
|
|
])
|
|
.collect())
|
|
}
|
|
_ => Err(()),
|
|
},
|
|
_ => Err(()),
|
|
},
|
|
_ => Err(()),
|
|
}
|
|
}
|
|
fn from_token_tree(
|
|
&mut self,
|
|
tree: TokenTree<Self::Group, Self::Punct, Self::Ident, Self::Literal>,
|
|
) -> Self::TokenStream {
|
|
tree.to_internal()
|
|
}
|
|
fn into_iter(&mut self, stream: Self::TokenStream) -> Self::TokenStreamIter {
|
|
TokenStreamIter { cursor: stream.trees(), stack: vec![] }
|
|
}
|
|
}
|
|
|
|
impl server::TokenStreamBuilder for Rustc<'_, '_> {
|
|
fn new(&mut self) -> Self::TokenStreamBuilder {
|
|
tokenstream::TokenStreamBuilder::new()
|
|
}
|
|
fn push(&mut self, builder: &mut Self::TokenStreamBuilder, stream: Self::TokenStream) {
|
|
builder.push(stream);
|
|
}
|
|
fn build(&mut self, builder: Self::TokenStreamBuilder) -> Self::TokenStream {
|
|
builder.build()
|
|
}
|
|
}
|
|
|
|
impl server::TokenStreamIter for Rustc<'_, '_> {
|
|
fn next(
|
|
&mut self,
|
|
iter: &mut Self::TokenStreamIter,
|
|
) -> Option<TokenTree<Self::Group, Self::Punct, Self::Ident, Self::Literal>> {
|
|
loop {
|
|
let tree = iter.stack.pop().or_else(|| {
|
|
let next = iter.cursor.next_with_spacing()?;
|
|
Some(TokenTree::from_internal((next, &mut iter.stack, self)))
|
|
})?;
|
|
// A hack used to pass AST fragments to attribute and derive macros
|
|
// as a single nonterminal token instead of a token stream.
|
|
// Such token needs to be "unwrapped" and not represented as a delimited group.
|
|
// FIXME: It needs to be removed, but there are some compatibility issues (see #73345).
|
|
if let TokenTree::Group(ref group) = tree {
|
|
if group.flatten {
|
|
iter.cursor.append(group.stream.clone());
|
|
continue;
|
|
}
|
|
}
|
|
return Some(tree);
|
|
}
|
|
}
|
|
}
|
|
|
|
impl server::Group for Rustc<'_, '_> {
|
|
fn new(&mut self, delimiter: Delimiter, stream: Self::TokenStream) -> Self::Group {
|
|
Group {
|
|
delimiter,
|
|
stream,
|
|
span: DelimSpan::from_single(server::Span::call_site(self)),
|
|
flatten: false,
|
|
}
|
|
}
|
|
fn delimiter(&mut self, group: &Self::Group) -> Delimiter {
|
|
group.delimiter
|
|
}
|
|
fn stream(&mut self, group: &Self::Group) -> Self::TokenStream {
|
|
group.stream.clone()
|
|
}
|
|
fn span(&mut self, group: &Self::Group) -> Self::Span {
|
|
group.span.entire()
|
|
}
|
|
fn span_open(&mut self, group: &Self::Group) -> Self::Span {
|
|
group.span.open
|
|
}
|
|
fn span_close(&mut self, group: &Self::Group) -> Self::Span {
|
|
group.span.close
|
|
}
|
|
fn set_span(&mut self, group: &mut Self::Group, span: Self::Span) {
|
|
group.span = DelimSpan::from_single(span);
|
|
}
|
|
}
|
|
|
|
impl server::Punct for Rustc<'_, '_> {
|
|
fn new(&mut self, ch: char, spacing: Spacing) -> Self::Punct {
|
|
Punct::new(ch, spacing == Spacing::Joint, server::Span::call_site(self))
|
|
}
|
|
fn as_char(&mut self, punct: Self::Punct) -> char {
|
|
punct.ch
|
|
}
|
|
fn spacing(&mut self, punct: Self::Punct) -> Spacing {
|
|
if punct.joint { Spacing::Joint } else { Spacing::Alone }
|
|
}
|
|
fn span(&mut self, punct: Self::Punct) -> Self::Span {
|
|
punct.span
|
|
}
|
|
fn with_span(&mut self, punct: Self::Punct, span: Self::Span) -> Self::Punct {
|
|
Punct { span, ..punct }
|
|
}
|
|
}
|
|
|
|
impl server::Ident for Rustc<'_, '_> {
|
|
fn new(&mut self, string: &str, span: Self::Span, is_raw: bool) -> Self::Ident {
|
|
Ident::new(self.sess(), Symbol::intern(string), is_raw, span)
|
|
}
|
|
fn span(&mut self, ident: Self::Ident) -> Self::Span {
|
|
ident.span
|
|
}
|
|
fn with_span(&mut self, ident: Self::Ident, span: Self::Span) -> Self::Ident {
|
|
Ident { span, ..ident }
|
|
}
|
|
}
|
|
|
|
impl server::Literal for Rustc<'_, '_> {
|
|
fn from_str(&mut self, s: &str) -> Result<Self::Literal, ()> {
|
|
let name = FileName::proc_macro_source_code(s);
|
|
let mut parser = rustc_parse::new_parser_from_source_str(self.sess(), name, s.to_owned());
|
|
|
|
let first_span = parser.token.span.data();
|
|
let minus_present = parser.eat(&token::BinOp(token::Minus));
|
|
|
|
let lit_span = parser.token.span.data();
|
|
let mut lit = match parser.token.kind {
|
|
token::Literal(lit) => lit,
|
|
_ => return Err(()),
|
|
};
|
|
|
|
// Check no comment or whitespace surrounding the (possibly negative)
|
|
// literal, or more tokens after it.
|
|
if (lit_span.hi.0 - first_span.lo.0) as usize != s.len() {
|
|
return Err(());
|
|
}
|
|
|
|
if minus_present {
|
|
// If minus is present, check no comment or whitespace in between it
|
|
// and the literal token.
|
|
if first_span.hi.0 != lit_span.lo.0 {
|
|
return Err(());
|
|
}
|
|
|
|
// Check literal is a kind we allow to be negated in a proc macro token.
|
|
match lit.kind {
|
|
token::LitKind::Bool
|
|
| token::LitKind::Byte
|
|
| token::LitKind::Char
|
|
| token::LitKind::Str
|
|
| token::LitKind::StrRaw(_)
|
|
| token::LitKind::ByteStr
|
|
| token::LitKind::ByteStrRaw(_)
|
|
| token::LitKind::Err => return Err(()),
|
|
token::LitKind::Integer | token::LitKind::Float => {}
|
|
}
|
|
|
|
// Synthesize a new symbol that includes the minus sign.
|
|
let symbol = Symbol::intern(&s[..1 + lit.symbol.as_str().len()]);
|
|
lit = token::Lit::new(lit.kind, symbol, lit.suffix);
|
|
}
|
|
|
|
Ok(Literal { lit, span: self.call_site })
|
|
}
|
|
fn to_string(&mut self, literal: &Self::Literal) -> String {
|
|
literal.lit.to_string()
|
|
}
|
|
fn debug_kind(&mut self, literal: &Self::Literal) -> String {
|
|
format!("{:?}", literal.lit.kind)
|
|
}
|
|
fn symbol(&mut self, literal: &Self::Literal) -> String {
|
|
literal.lit.symbol.to_string()
|
|
}
|
|
fn suffix(&mut self, literal: &Self::Literal) -> Option<String> {
|
|
literal.lit.suffix.as_ref().map(Symbol::to_string)
|
|
}
|
|
fn integer(&mut self, n: &str) -> Self::Literal {
|
|
self.lit(token::Integer, Symbol::intern(n), None)
|
|
}
|
|
fn typed_integer(&mut self, n: &str, kind: &str) -> Self::Literal {
|
|
self.lit(token::Integer, Symbol::intern(n), Some(Symbol::intern(kind)))
|
|
}
|
|
fn float(&mut self, n: &str) -> Self::Literal {
|
|
self.lit(token::Float, Symbol::intern(n), None)
|
|
}
|
|
fn f32(&mut self, n: &str) -> Self::Literal {
|
|
self.lit(token::Float, Symbol::intern(n), Some(sym::f32))
|
|
}
|
|
fn f64(&mut self, n: &str) -> Self::Literal {
|
|
self.lit(token::Float, Symbol::intern(n), Some(sym::f64))
|
|
}
|
|
fn string(&mut self, string: &str) -> Self::Literal {
|
|
let mut escaped = String::new();
|
|
for ch in string.chars() {
|
|
escaped.extend(ch.escape_debug());
|
|
}
|
|
self.lit(token::Str, Symbol::intern(&escaped), None)
|
|
}
|
|
fn character(&mut self, ch: char) -> Self::Literal {
|
|
let mut escaped = String::new();
|
|
escaped.extend(ch.escape_unicode());
|
|
self.lit(token::Char, Symbol::intern(&escaped), None)
|
|
}
|
|
fn byte_string(&mut self, bytes: &[u8]) -> Self::Literal {
|
|
let string = bytes
|
|
.iter()
|
|
.cloned()
|
|
.flat_map(ascii::escape_default)
|
|
.map(Into::<char>::into)
|
|
.collect::<String>();
|
|
self.lit(token::ByteStr, Symbol::intern(&string), None)
|
|
}
|
|
fn span(&mut self, literal: &Self::Literal) -> Self::Span {
|
|
literal.span
|
|
}
|
|
fn set_span(&mut self, literal: &mut Self::Literal, span: Self::Span) {
|
|
literal.span = span;
|
|
}
|
|
fn subspan(
|
|
&mut self,
|
|
literal: &Self::Literal,
|
|
start: Bound<usize>,
|
|
end: Bound<usize>,
|
|
) -> Option<Self::Span> {
|
|
let span = literal.span;
|
|
let length = span.hi().to_usize() - span.lo().to_usize();
|
|
|
|
let start = match start {
|
|
Bound::Included(lo) => lo,
|
|
Bound::Excluded(lo) => lo.checked_add(1)?,
|
|
Bound::Unbounded => 0,
|
|
};
|
|
|
|
let end = match end {
|
|
Bound::Included(hi) => hi.checked_add(1)?,
|
|
Bound::Excluded(hi) => hi,
|
|
Bound::Unbounded => length,
|
|
};
|
|
|
|
// Bounds check the values, preventing addition overflow and OOB spans.
|
|
if start > u32::MAX as usize
|
|
|| end > u32::MAX as usize
|
|
|| (u32::MAX - start as u32) < span.lo().to_u32()
|
|
|| (u32::MAX - end as u32) < span.lo().to_u32()
|
|
|| start >= end
|
|
|| end > length
|
|
{
|
|
return None;
|
|
}
|
|
|
|
let new_lo = span.lo() + BytePos::from_usize(start);
|
|
let new_hi = span.lo() + BytePos::from_usize(end);
|
|
Some(span.with_lo(new_lo).with_hi(new_hi))
|
|
}
|
|
}
|
|
|
|
impl server::SourceFile for Rustc<'_, '_> {
|
|
fn eq(&mut self, file1: &Self::SourceFile, file2: &Self::SourceFile) -> bool {
|
|
Lrc::ptr_eq(file1, file2)
|
|
}
|
|
fn path(&mut self, file: &Self::SourceFile) -> String {
|
|
match file.name {
|
|
FileName::Real(ref name) => name
|
|
.local_path()
|
|
.expect("attempting to get a file path in an imported file in `proc_macro::SourceFile::path`")
|
|
.to_str()
|
|
.expect("non-UTF8 file path in `proc_macro::SourceFile::path`")
|
|
.to_string(),
|
|
_ => file.name.prefer_local().to_string(),
|
|
}
|
|
}
|
|
fn is_real(&mut self, file: &Self::SourceFile) -> bool {
|
|
file.is_real_file()
|
|
}
|
|
}
|
|
|
|
impl server::MultiSpan for Rustc<'_, '_> {
|
|
fn new(&mut self) -> Self::MultiSpan {
|
|
vec![]
|
|
}
|
|
fn push(&mut self, spans: &mut Self::MultiSpan, span: Self::Span) {
|
|
spans.push(span)
|
|
}
|
|
}
|
|
|
|
impl server::Diagnostic for Rustc<'_, '_> {
|
|
fn new(&mut self, level: Level, msg: &str, spans: Self::MultiSpan) -> Self::Diagnostic {
|
|
let mut diag = Diagnostic::new(level.to_internal(), msg);
|
|
diag.set_span(MultiSpan::from_spans(spans));
|
|
diag
|
|
}
|
|
fn sub(
|
|
&mut self,
|
|
diag: &mut Self::Diagnostic,
|
|
level: Level,
|
|
msg: &str,
|
|
spans: Self::MultiSpan,
|
|
) {
|
|
diag.sub(level.to_internal(), msg, MultiSpan::from_spans(spans), None);
|
|
}
|
|
fn emit(&mut self, diag: Self::Diagnostic) {
|
|
self.sess().span_diagnostic.emit_diagnostic(&diag);
|
|
}
|
|
}
|
|
|
|
impl server::Span for Rustc<'_, '_> {
|
|
fn debug(&mut self, span: Self::Span) -> String {
|
|
if self.ecx.ecfg.span_debug {
|
|
format!("{:?}", span)
|
|
} else {
|
|
format!("{:?} bytes({}..{})", span.ctxt(), span.lo().0, span.hi().0)
|
|
}
|
|
}
|
|
fn def_site(&mut self) -> Self::Span {
|
|
self.def_site
|
|
}
|
|
fn call_site(&mut self) -> Self::Span {
|
|
self.call_site
|
|
}
|
|
fn mixed_site(&mut self) -> Self::Span {
|
|
self.mixed_site
|
|
}
|
|
fn source_file(&mut self, span: Self::Span) -> Self::SourceFile {
|
|
self.sess().source_map().lookup_char_pos(span.lo()).file
|
|
}
|
|
fn parent(&mut self, span: Self::Span) -> Option<Self::Span> {
|
|
span.parent_callsite()
|
|
}
|
|
fn source(&mut self, span: Self::Span) -> Self::Span {
|
|
span.source_callsite()
|
|
}
|
|
fn start(&mut self, span: Self::Span) -> LineColumn {
|
|
let loc = self.sess().source_map().lookup_char_pos(span.lo());
|
|
LineColumn { line: loc.line, column: loc.col.to_usize() }
|
|
}
|
|
fn end(&mut self, span: Self::Span) -> LineColumn {
|
|
let loc = self.sess().source_map().lookup_char_pos(span.hi());
|
|
LineColumn { line: loc.line, column: loc.col.to_usize() }
|
|
}
|
|
fn before(&mut self, span: Self::Span) -> Self::Span {
|
|
span.shrink_to_lo()
|
|
}
|
|
fn after(&mut self, span: Self::Span) -> Self::Span {
|
|
span.shrink_to_hi()
|
|
}
|
|
fn join(&mut self, first: Self::Span, second: Self::Span) -> Option<Self::Span> {
|
|
let self_loc = self.sess().source_map().lookup_char_pos(first.lo());
|
|
let other_loc = self.sess().source_map().lookup_char_pos(second.lo());
|
|
|
|
if self_loc.file.name != other_loc.file.name {
|
|
return None;
|
|
}
|
|
|
|
Some(first.to(second))
|
|
}
|
|
fn resolved_at(&mut self, span: Self::Span, at: Self::Span) -> Self::Span {
|
|
span.with_ctxt(at.ctxt())
|
|
}
|
|
fn source_text(&mut self, span: Self::Span) -> Option<String> {
|
|
self.sess().source_map().span_to_snippet(span).ok()
|
|
}
|
|
/// Saves the provided span into the metadata of
|
|
/// *the crate we are currently compiling*, which must
|
|
/// be a proc-macro crate. This id can be passed to
|
|
/// `recover_proc_macro_span` when our current crate
|
|
/// is *run* as a proc-macro.
|
|
///
|
|
/// Let's suppose that we have two crates - `my_client`
|
|
/// and `my_proc_macro`. The `my_proc_macro` crate
|
|
/// contains a procedural macro `my_macro`, which
|
|
/// is implemented as: `quote! { "hello" }`
|
|
///
|
|
/// When we *compile* `my_proc_macro`, we will execute
|
|
/// the `quote` proc-macro. This will save the span of
|
|
/// "hello" into the metadata of `my_proc_macro`. As a result,
|
|
/// the body of `my_proc_macro` (after expansion) will end
|
|
/// up containg a call that looks like this:
|
|
/// `proc_macro::Ident::new("hello", proc_macro::Span::recover_proc_macro_span(0))`
|
|
///
|
|
/// where `0` is the id returned by this function.
|
|
/// When `my_proc_macro` *executes* (during the compilation of `my_client`),
|
|
/// the call to `recover_proc_macro_span` will load the corresponding
|
|
/// span from the metadata of `my_proc_macro` (which we have access to,
|
|
/// since we've loaded `my_proc_macro` from disk in order to execute it).
|
|
/// In this way, we have obtained a span pointing into `my_proc_macro`
|
|
fn save_span(&mut self, span: Self::Span) -> usize {
|
|
self.sess().save_proc_macro_span(span)
|
|
}
|
|
fn recover_proc_macro_span(&mut self, id: usize) -> Self::Span {
|
|
let (resolver, krate, def_site) = (&*self.ecx.resolver, self.krate, self.def_site);
|
|
*self.rebased_spans.entry(id).or_insert_with(|| {
|
|
// FIXME: `SyntaxContext` for spans from proc macro crates is lost during encoding,
|
|
// replace it with a def-site context until we are encoding it properly.
|
|
resolver.get_proc_macro_quoted_span(krate, id).with_ctxt(def_site.ctxt())
|
|
})
|
|
}
|
|
}
|
|
|
|
// See issue #74616 for details
|
|
fn ident_name_compatibility_hack(
|
|
nt: &Nonterminal,
|
|
orig_span: Span,
|
|
rustc: &mut Rustc<'_, '_>,
|
|
) -> Option<(rustc_span::symbol::Ident, bool)> {
|
|
if let NtIdent(ident, is_raw) = nt {
|
|
if let ExpnKind::Macro(_, macro_name) = orig_span.ctxt().outer_expn_data().kind {
|
|
let source_map = rustc.sess().source_map();
|
|
let filename = source_map.span_to_filename(orig_span);
|
|
if let FileName::Real(RealFileName::LocalPath(path)) = filename {
|
|
let matches_prefix = |prefix, filename| {
|
|
// Check for a path that ends with 'prefix*/src/<filename>'
|
|
let mut iter = path.components().rev();
|
|
iter.next().and_then(|p| p.as_os_str().to_str()) == Some(filename)
|
|
&& iter.next().and_then(|p| p.as_os_str().to_str()) == Some("src")
|
|
&& iter
|
|
.next()
|
|
.and_then(|p| p.as_os_str().to_str())
|
|
.map_or(false, |p| p.starts_with(prefix))
|
|
};
|
|
|
|
let time_macros_impl =
|
|
macro_name == sym::impl_macros && matches_prefix("time-macros-impl", "lib.rs");
|
|
let js_sys = macro_name == sym::arrays && matches_prefix("js-sys", "lib.rs");
|
|
if time_macros_impl || js_sys {
|
|
let snippet = source_map.span_to_snippet(orig_span);
|
|
if snippet.as_deref() == Ok("$name") {
|
|
if time_macros_impl {
|
|
rustc.sess().buffer_lint_with_diagnostic(
|
|
&PROC_MACRO_BACK_COMPAT,
|
|
orig_span,
|
|
ast::CRATE_NODE_ID,
|
|
"using an old version of `time-macros-impl`",
|
|
BuiltinLintDiagnostics::ProcMacroBackCompat(
|
|
"the `time-macros-impl` crate will stop compiling in futures version of Rust. \
|
|
Please update to the latest version of the `time` crate to avoid breakage".to_string())
|
|
);
|
|
return Some((*ident, *is_raw));
|
|
}
|
|
if js_sys {
|
|
if let Some(c) = path
|
|
.components()
|
|
.flat_map(|c| c.as_os_str().to_str())
|
|
.find(|c| c.starts_with("js-sys"))
|
|
{
|
|
let mut version = c.trim_start_matches("js-sys-").split('.');
|
|
if version.next() == Some("0")
|
|
&& version.next() == Some("3")
|
|
&& version
|
|
.next()
|
|
.and_then(|c| c.parse::<u32>().ok())
|
|
.map_or(false, |v| v < 40)
|
|
{
|
|
rustc.sess().buffer_lint_with_diagnostic(
|
|
&PROC_MACRO_BACK_COMPAT,
|
|
orig_span,
|
|
ast::CRATE_NODE_ID,
|
|
"using an old version of `js-sys`",
|
|
BuiltinLintDiagnostics::ProcMacroBackCompat(
|
|
"older versions of the `js-sys` crate will stop compiling in future versions of Rust; \
|
|
please update to `js-sys` v0.3.40 or above".to_string())
|
|
);
|
|
return Some((*ident, *is_raw));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if macro_name == sym::tuple_from_req && matches_prefix("actix-web", "extract.rs") {
|
|
let snippet = source_map.span_to_snippet(orig_span);
|
|
if snippet.as_deref() == Ok("$T") {
|
|
if let FileName::Real(RealFileName::LocalPath(macro_path)) =
|
|
source_map.span_to_filename(rustc.def_site)
|
|
{
|
|
if macro_path.to_string_lossy().contains("pin-project-internal-0.") {
|
|
rustc.sess().buffer_lint_with_diagnostic(
|
|
&PROC_MACRO_BACK_COMPAT,
|
|
orig_span,
|
|
ast::CRATE_NODE_ID,
|
|
"using an old version of `actix-web`",
|
|
BuiltinLintDiagnostics::ProcMacroBackCompat(
|
|
"the version of `actix-web` you are using might stop compiling in future versions of Rust; \
|
|
please update to the latest version of the `actix-web` crate to avoid breakage".to_string())
|
|
);
|
|
return Some((*ident, *is_raw));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
None
|
|
}
|