2019-09-30 03:58:53 -05:00
|
|
|
//! `tt` crate defines a `TokenTree` data structure: this is the interface (both
|
|
|
|
//! input and output) of macros. It closely mirrors `proc_macro` crate's
|
|
|
|
//! `TokenTree`.
|
2020-11-26 12:56:38 -06:00
|
|
|
use std::{fmt, panic::RefUnwindSafe};
|
2019-01-31 08:16:02 -06:00
|
|
|
|
2020-07-13 09:16:53 -05:00
|
|
|
use stdx::impl_from;
|
|
|
|
|
2020-03-26 11:41:44 -05:00
|
|
|
pub use smol_str::SmolStr;
|
2019-01-30 14:02:27 -06:00
|
|
|
|
2019-02-11 12:31:54 -06:00
|
|
|
/// Represents identity of the token.
|
|
|
|
///
|
|
|
|
/// For hygiene purposes, we need to track which expanded tokens originated from
|
|
|
|
/// which source tokens. We do it by assigning an distinct identity to each
|
|
|
|
/// source token and making sure that identities are preserved during macro
|
|
|
|
/// expansion.
|
2019-02-11 10:28:39 -06:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct TokenId(pub u32);
|
|
|
|
|
|
|
|
impl TokenId {
|
|
|
|
pub const fn unspecified() -> TokenId {
|
|
|
|
TokenId(!0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-27 23:41:58 -06:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
2019-01-31 04:40:05 -06:00
|
|
|
pub enum TokenTree {
|
2019-01-30 14:02:27 -06:00
|
|
|
Leaf(Leaf),
|
|
|
|
Subtree(Subtree),
|
|
|
|
}
|
2020-07-13 09:16:53 -05:00
|
|
|
impl_from!(Leaf, Subtree for TokenTree);
|
2019-01-30 14:02:27 -06:00
|
|
|
|
2020-03-13 07:03:31 -05:00
|
|
|
impl TokenTree {
|
|
|
|
pub fn empty() -> Self {
|
|
|
|
TokenTree::Subtree(Subtree::default())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-27 23:41:58 -06:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
2019-01-31 04:40:05 -06:00
|
|
|
pub enum Leaf {
|
2019-01-30 14:02:27 -06:00
|
|
|
Literal(Literal),
|
|
|
|
Punct(Punct),
|
|
|
|
Ident(Ident),
|
|
|
|
}
|
2020-07-13 09:16:53 -05:00
|
|
|
impl_from!(Literal, Punct, Ident for Leaf);
|
2019-01-30 14:02:27 -06:00
|
|
|
|
2020-04-09 10:48:08 -05:00
|
|
|
#[derive(Clone, PartialEq, Eq, Hash, Default)]
|
2019-01-31 04:40:05 -06:00
|
|
|
pub struct Subtree {
|
2019-12-13 07:53:34 -06:00
|
|
|
pub delimiter: Option<Delimiter>,
|
2019-01-31 04:40:05 -06:00
|
|
|
pub token_trees: Vec<TokenTree>,
|
2019-01-30 14:02:27 -06:00
|
|
|
}
|
|
|
|
|
2019-11-27 23:41:58 -06:00
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
2019-12-12 11:41:44 -06:00
|
|
|
pub struct Delimiter {
|
|
|
|
pub id: TokenId,
|
|
|
|
pub kind: DelimiterKind,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub enum DelimiterKind {
|
2019-01-30 14:02:27 -06:00
|
|
|
Parenthesis,
|
|
|
|
Brace,
|
|
|
|
Bracket,
|
|
|
|
}
|
|
|
|
|
2019-11-27 23:41:58 -06:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
2019-01-31 04:40:05 -06:00
|
|
|
pub struct Literal {
|
|
|
|
pub text: SmolStr,
|
2019-12-12 07:47:54 -06:00
|
|
|
pub id: TokenId,
|
2019-01-30 14:02:27 -06:00
|
|
|
}
|
|
|
|
|
2019-11-27 23:41:58 -06:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
2019-01-31 04:40:05 -06:00
|
|
|
pub struct Punct {
|
|
|
|
pub char: char,
|
2019-01-31 09:51:17 -06:00
|
|
|
pub spacing: Spacing,
|
2019-12-12 07:47:54 -06:00
|
|
|
pub id: TokenId,
|
2019-01-31 09:51:17 -06:00
|
|
|
}
|
|
|
|
|
2019-11-27 23:41:58 -06:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
2019-01-31 09:51:17 -06:00
|
|
|
pub enum Spacing {
|
|
|
|
Alone,
|
|
|
|
Joint,
|
2019-01-30 14:02:27 -06:00
|
|
|
}
|
|
|
|
|
2019-11-27 23:41:58 -06:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
2019-01-31 04:40:05 -06:00
|
|
|
pub struct Ident {
|
|
|
|
pub text: SmolStr,
|
2019-02-11 10:28:39 -06:00
|
|
|
pub id: TokenId,
|
2019-01-30 14:02:27 -06:00
|
|
|
}
|
2019-01-31 08:16:02 -06:00
|
|
|
|
2020-04-09 10:48:08 -05:00
|
|
|
fn print_debug_subtree(f: &mut fmt::Formatter<'_>, subtree: &Subtree, level: usize) -> fmt::Result {
|
|
|
|
let align = std::iter::repeat(" ").take(level).collect::<String>();
|
|
|
|
|
|
|
|
let aux = match subtree.delimiter.map(|it| (it.kind, it.id.0)) {
|
|
|
|
None => "$".to_string(),
|
|
|
|
Some((DelimiterKind::Parenthesis, id)) => format!("() {}", id),
|
|
|
|
Some((DelimiterKind::Brace, id)) => format!("{{}} {}", id),
|
|
|
|
Some((DelimiterKind::Bracket, id)) => format!("[] {}", id),
|
|
|
|
};
|
|
|
|
|
|
|
|
if subtree.token_trees.is_empty() {
|
|
|
|
write!(f, "{}SUBTREE {}", align, aux)?;
|
|
|
|
} else {
|
|
|
|
writeln!(f, "{}SUBTREE {}", align, aux)?;
|
|
|
|
for (idx, child) in subtree.token_trees.iter().enumerate() {
|
|
|
|
print_debug_token(f, child, level + 1)?;
|
|
|
|
if idx != subtree.token_trees.len() - 1 {
|
2020-08-10 07:05:01 -05:00
|
|
|
writeln!(f)?;
|
2020-04-09 10:48:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_debug_token(f: &mut fmt::Formatter<'_>, tkn: &TokenTree, level: usize) -> fmt::Result {
|
|
|
|
let align = std::iter::repeat(" ").take(level).collect::<String>();
|
|
|
|
|
|
|
|
match tkn {
|
|
|
|
TokenTree::Leaf(leaf) => match leaf {
|
|
|
|
Leaf::Literal(lit) => write!(f, "{}LITERAL {} {}", align, lit.text, lit.id.0)?,
|
|
|
|
Leaf::Punct(punct) => write!(
|
|
|
|
f,
|
|
|
|
"{}PUNCH {} [{}] {}",
|
|
|
|
align,
|
|
|
|
punct.char,
|
|
|
|
if punct.spacing == Spacing::Alone { "alone" } else { "joint" },
|
|
|
|
punct.id.0
|
|
|
|
)?,
|
|
|
|
Leaf::Ident(ident) => write!(f, "{}IDENT {} {}", align, ident.text, ident.id.0)?,
|
|
|
|
},
|
|
|
|
TokenTree::Subtree(subtree) => {
|
|
|
|
print_debug_subtree(f, subtree, level)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-11-26 12:56:38 -06:00
|
|
|
impl fmt::Debug for Subtree {
|
2020-04-09 10:48:08 -05:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
print_debug_subtree(f, self, 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-31 08:16:02 -06:00
|
|
|
impl fmt::Display for TokenTree {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match self {
|
|
|
|
TokenTree::Leaf(it) => fmt::Display::fmt(it, f),
|
|
|
|
TokenTree::Subtree(it) => fmt::Display::fmt(it, f),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Subtree {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2019-12-17 21:47:26 -06:00
|
|
|
let (l, r) = match self.delimiter_kind() {
|
2019-12-12 11:41:44 -06:00
|
|
|
Some(DelimiterKind::Parenthesis) => ("(", ")"),
|
|
|
|
Some(DelimiterKind::Brace) => ("{", "}"),
|
|
|
|
Some(DelimiterKind::Bracket) => ("[", "]"),
|
2019-12-13 07:53:34 -06:00
|
|
|
None => ("", ""),
|
2019-01-31 08:16:02 -06:00
|
|
|
};
|
2019-01-31 09:51:17 -06:00
|
|
|
f.write_str(l)?;
|
|
|
|
let mut needs_space = false;
|
|
|
|
for tt in self.token_trees.iter() {
|
|
|
|
if needs_space {
|
|
|
|
f.write_str(" ")?;
|
|
|
|
}
|
|
|
|
needs_space = true;
|
|
|
|
match tt {
|
|
|
|
TokenTree::Leaf(Leaf::Punct(p)) => {
|
|
|
|
needs_space = p.spacing == Spacing::Alone;
|
|
|
|
fmt::Display::fmt(p, f)?
|
|
|
|
}
|
|
|
|
tt => fmt::Display::fmt(tt, f)?,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
f.write_str(r)?;
|
|
|
|
Ok(())
|
2019-01-31 08:16:02 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Leaf {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match self {
|
|
|
|
Leaf::Ident(it) => fmt::Display::fmt(it, f),
|
|
|
|
Leaf::Literal(it) => fmt::Display::fmt(it, f),
|
|
|
|
Leaf::Punct(it) => fmt::Display::fmt(it, f),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Ident {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
fmt::Display::fmt(&self.text, f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Literal {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
fmt::Display::fmt(&self.text, f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Punct {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
fmt::Display::fmt(&self.char, f)
|
|
|
|
}
|
|
|
|
}
|
2019-04-22 02:33:55 -05:00
|
|
|
|
|
|
|
impl Subtree {
|
|
|
|
/// Count the number of tokens recursively
|
|
|
|
pub fn count(&self) -> usize {
|
2019-04-22 04:37:27 -05:00
|
|
|
let children_count = self
|
|
|
|
.token_trees
|
|
|
|
.iter()
|
|
|
|
.map(|c| match c {
|
2019-04-22 02:33:55 -05:00
|
|
|
TokenTree::Subtree(c) => c.count(),
|
|
|
|
_ => 0,
|
2019-04-22 04:37:27 -05:00
|
|
|
})
|
|
|
|
.sum::<usize>();
|
|
|
|
|
|
|
|
self.token_trees.len() + children_count
|
2019-04-22 02:33:55 -05:00
|
|
|
}
|
2019-12-17 21:47:26 -06:00
|
|
|
|
|
|
|
pub fn delimiter_kind(&self) -> Option<DelimiterKind> {
|
|
|
|
self.delimiter.map(|it| it.kind)
|
|
|
|
}
|
2019-04-22 02:33:55 -05:00
|
|
|
}
|
2019-05-21 23:30:36 -05:00
|
|
|
|
|
|
|
pub mod buffer;
|
2020-03-26 11:41:44 -05:00
|
|
|
|
2020-05-14 04:57:37 -05:00
|
|
|
#[derive(Debug, PartialEq, Eq, Clone)]
|
2020-03-26 15:26:34 -05:00
|
|
|
pub enum ExpansionError {
|
|
|
|
IOError(String),
|
|
|
|
JsonError(String),
|
|
|
|
Unknown(String),
|
|
|
|
ExpansionError(String),
|
|
|
|
}
|
2020-03-26 11:41:44 -05:00
|
|
|
|
2020-11-26 09:56:22 -06:00
|
|
|
impl fmt::Display for ExpansionError {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
match self {
|
|
|
|
ExpansionError::IOError(e) => write!(f, "I/O error: {}", e),
|
|
|
|
ExpansionError::JsonError(e) => write!(f, "JSON decoding error: {}", e),
|
2020-11-26 12:56:38 -06:00
|
|
|
ExpansionError::Unknown(e) => e.fmt(f),
|
2020-11-26 09:56:22 -06:00
|
|
|
ExpansionError::ExpansionError(e) => write!(f, "proc macro returned error: {}", e),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-26 12:56:38 -06:00
|
|
|
pub trait TokenExpander: fmt::Debug + Send + Sync + RefUnwindSafe {
|
2020-03-26 11:41:44 -05:00
|
|
|
fn expand(&self, subtree: &Subtree, attrs: Option<&Subtree>)
|
|
|
|
-> Result<Subtree, ExpansionError>;
|
|
|
|
}
|