// Copyright 2012 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. use core::prelude::*; use ast; use ast::{token_tree, tt_delim, tt_tok, tt_seq, tt_nonterminal,ident}; use codemap::{span, dummy_sp}; use diagnostic::span_handler; use ext::tt::macro_parser::{named_match, matched_seq, matched_nonterminal}; use parse::token::{EOF, INTERPOLATED, IDENT, Token, nt_ident, ident_interner}; use parse::lexer::TokenAndSpan; use core::hashmap::linear::LinearMap; use core::option; use core::vec; /* FIXME #2811: figure out how to have a uniquely linked stack, and change to `~` */ ///an unzipping of `token_tree`s struct TtFrame { readme: @mut ~[ast::token_tree], idx: uint, dotdotdoted: bool, sep: Option, up: Option<@mut TtFrame>, } pub struct TtReader { sp_diag: @span_handler, interner: @ident_interner, // the unzipped tree: cur: @mut TtFrame, /* for MBE-style macro transcription */ interpolations: LinearMap, repeat_idx: ~[uint], repeat_len: ~[uint], /* cached: */ cur_tok: Token, cur_span: span } /** This can do Macro-By-Example transcription. On the other hand, if * `src` contains no `tt_seq`s and `tt_nonterminal`s, `interp` can (and * should) be none. */ pub fn new_tt_reader(sp_diag: @span_handler, itr: @ident_interner, interp: Option>, +src: ~[ast::token_tree]) -> @mut TtReader { let r = @mut TtReader { sp_diag: sp_diag, interner: itr, cur: @mut TtFrame { readme: @mut src, idx: 0u, dotdotdoted: false, sep: None, up: option::None }, interpolations: match interp { /* just a convienience */ None => LinearMap::new(), Some(x) => x }, repeat_idx: ~[], repeat_len: ~[], /* dummy values, never read: */ cur_tok: EOF, cur_span: dummy_sp() }; tt_next_token(r); /* get cur_tok and cur_span set up */ return r; } fn dup_tt_frame(f: @mut TtFrame) -> @mut TtFrame { @mut TtFrame { readme: @mut (copy *f.readme), idx: f.idx, dotdotdoted: f.dotdotdoted, sep: copy f.sep, up: match f.up { Some(up_frame) => Some(dup_tt_frame(up_frame)), None => None } } } pub fn dup_tt_reader(r: @mut TtReader) -> @mut TtReader { @mut TtReader { sp_diag: r.sp_diag, interner: r.interner, cur: dup_tt_frame(r.cur), interpolations: r.interpolations, repeat_idx: copy r.repeat_idx, repeat_len: copy r.repeat_len, cur_tok: copy r.cur_tok, cur_span: r.cur_span } } fn lookup_cur_matched_by_matched(r: &mut TtReader, start: @named_match) -> @named_match { fn red(+ad: @named_match, idx: &uint) -> @named_match { match *ad { matched_nonterminal(_) => { // end of the line; duplicate henceforth ad } matched_seq(ref ads, _) => ads[*idx] } } let r = &mut *r; let repeat_idx = &r.repeat_idx; vec::foldl(start, *repeat_idx, red) } fn lookup_cur_matched(r: &mut TtReader, name: ident) -> @named_match { // FIXME (#3850): this looks a bit silly with an extra scope. let start; { start = *r.interpolations.get(&name); } return lookup_cur_matched_by_matched(r, start); } enum lis { lis_unconstrained, lis_constraint(uint, ident), lis_contradiction(~str) } fn lockstep_iter_size(t: token_tree, r: &mut TtReader) -> lis { fn lis_merge(lhs: lis, rhs: lis, r: &mut TtReader) -> lis { match lhs { lis_unconstrained => copy rhs, lis_contradiction(_) => copy lhs, lis_constraint(l_len, l_id) => match rhs { lis_unconstrained => copy lhs, lis_contradiction(_) => copy rhs, lis_constraint(r_len, _) if l_len == r_len => copy lhs, lis_constraint(r_len, r_id) => { let l_n = copy *r.interner.get(l_id); let r_n = copy *r.interner.get(r_id); lis_contradiction(fmt!("Inconsistent lockstep iteration: \ '%s' has %u items, but '%s' has %u", l_n, l_len, r_n, r_len)) } } } } match t { tt_delim(ref tts) | tt_seq(_, ref tts, _, _) => { vec::foldl(lis_unconstrained, (*tts), |lis, tt| { let lis2 = lockstep_iter_size(*tt, r); lis_merge(lis, lis2, r) }) } tt_tok(*) => lis_unconstrained, tt_nonterminal(_, name) => match *lookup_cur_matched(r, name) { matched_nonterminal(_) => lis_unconstrained, matched_seq(ref ads, _) => lis_constraint(ads.len(), name) } } } pub fn tt_next_token(r: &mut TtReader) -> TokenAndSpan { let ret_val = TokenAndSpan { tok: copy r.cur_tok, sp: r.cur_span, }; loop { { let cur = &mut *r.cur; let readme = &mut *cur.readme; if cur.idx < readme.len() { break; } } /* done with this set; pop or repeat? */ if ! r.cur.dotdotdoted || { *r.repeat_idx.last() == *r.repeat_len.last() - 1 } { match r.cur.up { None => { r.cur_tok = EOF; return ret_val; } Some(tt_f) => { if r.cur.dotdotdoted { r.repeat_idx.pop(); r.repeat_len.pop(); } r.cur = tt_f; r.cur.idx += 1u; } } } else { /* repeat */ r.cur.idx = 0u; r.repeat_idx[r.repeat_idx.len() - 1u] += 1u; match r.cur.sep { Some(copy tk) => { r.cur_tok = tk; /* repeat same span, I guess */ return ret_val; } None => () } } } loop { /* because it's easiest, this handles `tt_delim` not starting with a `tt_tok`, even though it won't happen */ match r.cur.readme[r.cur.idx] { tt_delim(copy tts) => { r.cur = @mut TtFrame { readme: @mut tts, idx: 0u, dotdotdoted: false, sep: None, up: option::Some(r.cur) }; // if this could be 0-length, we'd need to potentially recur here } tt_tok(sp, copy tok) => { r.cur_span = sp; r.cur_tok = tok; r.cur.idx += 1u; return ret_val; } tt_seq(sp, copy tts, copy sep, zerok) => { let t = tt_seq(sp, copy tts, copy sep, zerok); match lockstep_iter_size(t, r) { lis_unconstrained => { r.sp_diag.span_fatal( sp, /* blame macro writer */ ~"attempted to repeat an expression \ containing no syntax \ variables matched as repeating at this depth"); } lis_contradiction(ref msg) => { /* FIXME #2887 blame macro invoker instead*/ r.sp_diag.span_fatal(sp, (*msg)); } lis_constraint(len, _) => { if len == 0 { if !zerok { r.sp_diag.span_fatal(sp, /* FIXME #2887 blame invoker */ ~"this must repeat at least \ once"); } r.cur.idx += 1u; return tt_next_token(r); } else { r.repeat_len.push(len); r.repeat_idx.push(0u); r.cur = @mut TtFrame { readme: @mut tts, idx: 0u, dotdotdoted: true, sep: sep, up: Some(r.cur) }; } } } } // FIXME #2887: think about span stuff here tt_nonterminal(sp, ident) => { match *lookup_cur_matched(r, ident) { /* sidestep the interpolation tricks for ident because (a) idents can be in lots of places, so it'd be a pain (b) we actually can, since it's a token. */ matched_nonterminal(nt_ident(sn,b)) => { r.cur_span = sp; r.cur_tok = IDENT(sn,b); r.cur.idx += 1u; return ret_val; } matched_nonterminal(ref other_whole_nt) => { r.cur_span = sp; r.cur_tok = INTERPOLATED(copy *other_whole_nt); r.cur.idx += 1u; return ret_val; } matched_seq(*) => { r.sp_diag.span_fatal( copy r.cur_span, /* blame the macro writer */ fmt!("variable '%s' is still repeating at this depth", *r.interner.get(ident))); } } } } } }