use crate::ast::Ident;
use crate::errors::Handler;
use crate::ext::tt::macro_parser::{MatchedNonterminal, MatchedSeq, NamedMatch};
use crate::ext::tt::quoted;
use crate::parse::token::{self, NtTT, SubstNt, Token};
use crate::syntax_pos::{Span, DUMMY_SP};
use crate::tokenstream::{Delimited, TokenStream, TokenTree};
use crate::util::small_vector::SmallVector;
use std::collections::HashMap;
use std::mem;
use std::ops::Add;
use std::rc::Rc;
enum Frame {
Delimited {
forest: Rc<quoted::Delimited>,
idx: usize,
span: Span,
},
Sequence {
forest: Rc<quoted::SequenceRepetition>,
idx: usize,
sep: Option<Token>,
},
}
impl Frame {
fn new(tts: Vec<quoted::TokenTree>) -> Frame {
let forest = Rc::new(quoted::Delimited {
delim: token::NoDelim,
tts: tts,
});
Frame::Delimited {
forest: forest,
idx: 0,
span: DUMMY_SP,
}
}
}
impl Iterator for Frame {
type Item = quoted::TokenTree;
fn next(&mut self) -> Option<quoted::TokenTree> {
match *self {
Frame::Delimited {
ref forest,
ref mut idx,
..
} => {
*idx += 1;
forest.tts.get(*idx - 1).cloned()
}
Frame::Sequence {
ref forest,
ref mut idx,
..
} => {
*idx += 1;
forest.tts.get(*idx - 1).cloned()
}
}
}
}
pub fn transcribe(
sp_diag: &Handler,
interp: Option<HashMap<Ident, Rc<NamedMatch>>>,
src: Vec<quoted::TokenTree>,
) -> TokenStream {
let mut stack = SmallVector::one(Frame::new(src));
let interpolations = interp.unwrap_or_else(HashMap::new);
let mut repeats = Vec::new();
let mut result: Vec<TokenStream> = Vec::new();
let mut result_stack = Vec::new();
loop {
let tree = if let Some(tree) = stack.last_mut().unwrap().next() {
tree
} else {
if let Frame::Sequence {
ref mut idx,
ref sep,
..
} = *stack.last_mut().unwrap()
{
let (ref mut repeat_idx, repeat_len) = *repeats.last_mut().unwrap();
*repeat_idx += 1;
if *repeat_idx < repeat_len {
*idx = 0;
if let Some(sep) = sep.clone() {
let prev_span = match result.last() {
Some(stream) => stream.trees().next().unwrap().span(),
None => DUMMY_SP,
};
result.push(TokenTree::Token(prev_span, sep).into());
}
continue;
}
}
match stack.pop().unwrap() {
Frame::Sequence { .. } => {
repeats.pop();
}
Frame::Delimited { forest, span, .. } => {
if result_stack.is_empty() {
return TokenStream::concat(result);
}
let tree = TokenTree::Delimited(
span,
Delimited {
delim: forest.delim,
tts: TokenStream::concat(result).into(),
},
);
result = result_stack.pop().unwrap();
result.push(tree.into());
}
}
continue;
};
match tree {
quoted::TokenTree::Sequence(sp, seq) => {
match lockstep_iter_size(
"ed::TokenTree::Sequence(sp, seq.clone()),
&interpolations,
&repeats,
) {
LockstepIterSize::Unconstrained => {
panic!(sp_diag.span_fatal(
sp,
"attempted to repeat an expression \
containing no syntax \
variables matched as repeating at this depth"
));
}
LockstepIterSize::Contradiction(ref msg) => {
panic!(sp_diag.span_fatal(sp, &msg[..]));
}
LockstepIterSize::Constraint(len, _) => {
if len == 0 {
if seq.op == quoted::KleeneOp::OneOrMore {
panic!(sp_diag.span_fatal(sp, "this must repeat at least once"));
}
} else {
repeats.push((0, len));
stack.push(Frame::Sequence {
idx: 0,
sep: seq.separator.clone(),
forest: seq,
});
}
}
}
}
quoted::TokenTree::Token(sp, SubstNt(ident)) => {
match lookup_cur_matched(ident, &interpolations, &repeats) {
None => result.push(TokenTree::Token(sp, SubstNt(ident)).into()),
Some(cur_matched) => {
if let MatchedNonterminal(ref nt) = *cur_matched {
match **nt {
NtTT(ref tt) => result.push(tt.clone().into()),
_ => {
let token =
TokenTree::Token(sp, token::Interpolated(nt.clone()));
result.push(token.into());
}
}
} else {
panic!(sp_diag.span_fatal(
sp,
&format!("variable '{}' is still repeating at this depth", ident)
));
}
}
}
}
quoted::TokenTree::Delimited(span, delimited) => {
stack.push(Frame::Delimited {
forest: delimited,
idx: 0,
span: span,
});
result_stack.push(mem::replace(&mut result, Vec::new()));
}
quoted::TokenTree::Token(span, tok) => result.push(TokenTree::Token(span, tok).into()),
quoted::TokenTree::MetaVarDecl(..) => panic!("unexpected `TokenTree::MetaVarDecl"),
}
}
}
fn lookup_cur_matched(
ident: Ident,
interpolations: &HashMap<Ident, Rc<NamedMatch>>,
repeats: &[(usize, usize)],
) -> Option<Rc<NamedMatch>> {
interpolations.get(&ident).map(|matched| {
repeats.iter().fold(matched.clone(), |ad, &(idx, _)| {
match *ad {
MatchedNonterminal(_) => {
ad.clone()
}
MatchedSeq(ref ads, _) => ads[idx].clone(),
}
})
})
}
#[derive(Clone)]
enum LockstepIterSize {
Unconstrained,
Constraint(usize, Ident),
Contradiction(String),
}
impl Add for LockstepIterSize {
type Output = LockstepIterSize;
fn add(self, other: LockstepIterSize) -> LockstepIterSize {
match self {
LockstepIterSize::Unconstrained => other,
LockstepIterSize::Contradiction(_) => self,
LockstepIterSize::Constraint(l_len, ref l_id) => match other {
LockstepIterSize::Unconstrained => self.clone(),
LockstepIterSize::Contradiction(_) => other,
LockstepIterSize::Constraint(r_len, _) if l_len == r_len => self.clone(),
LockstepIterSize::Constraint(r_len, r_id) => {
let msg = format!(
"inconsistent lockstep iteration: \
'{}' has {} items, but '{}' has {}",
l_id, l_len, r_id, r_len
);
LockstepIterSize::Contradiction(msg)
}
},
}
}
}
fn lockstep_iter_size(
tree: "ed::TokenTree,
interpolations: &HashMap<Ident, Rc<NamedMatch>>,
repeats: &[(usize, usize)],
) -> LockstepIterSize {
use self::quoted::TokenTree;
match *tree {
TokenTree::Delimited(_, ref delimed) => delimed
.tts
.iter()
.fold(LockstepIterSize::Unconstrained, |size, tt| {
size + lockstep_iter_size(tt, interpolations, repeats)
}),
TokenTree::Sequence(_, ref seq) => seq
.tts
.iter()
.fold(LockstepIterSize::Unconstrained, |size, tt| {
size + lockstep_iter_size(tt, interpolations, repeats)
}),
TokenTree::Token(_, SubstNt(name)) | TokenTree::MetaVarDecl(_, name, _) => {
match lookup_cur_matched(name, interpolations, repeats) {
Some(matched) => match *matched {
MatchedNonterminal(_) => LockstepIterSize::Unconstrained,
MatchedSeq(ref ads, _) => LockstepIterSize::Constraint(ads.len(), name),
},
_ => LockstepIterSize::Unconstrained,
}
}
TokenTree::Token(..) => LockstepIterSize::Unconstrained,
}
}