1use std::fmt;
5
6use arrayvec::ArrayVec;
7use intern::sym;
8
9use crate::{Ident, Leaf, MAX_GLUED_PUNCT_LEN, Punct, Spacing, Subtree, TokenTree, TokenTreesView};
10
11#[derive(Clone)]
12pub struct TtIter<'a, S> {
13 inner: std::slice::Iter<'a, TokenTree<S>>,
14}
15
16impl<S: Copy + fmt::Debug> fmt::Debug for TtIter<'_, S> {
17 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
18 f.debug_struct("TtIter").field("remaining", &self.remaining()).finish()
19 }
20}
21
22#[derive(Clone, Copy)]
23pub struct TtIterSavepoint<'a, S>(&'a [TokenTree<S>]);
24
25impl<'a, S: Copy> TtIterSavepoint<'a, S> {
26 pub fn remaining(self) -> TokenTreesView<'a, S> {
27 TokenTreesView::new(self.0)
28 }
29}
30
31impl<'a, S: Copy> TtIter<'a, S> {
32 pub(crate) fn new(tt: &'a [TokenTree<S>]) -> TtIter<'a, S> {
33 TtIter { inner: tt.iter() }
34 }
35
36 pub fn expect_char(&mut self, char: char) -> Result<(), ()> {
37 match self.next() {
38 Some(TtElement::Leaf(&Leaf::Punct(Punct { char: c, .. }))) if c == char => Ok(()),
39 _ => Err(()),
40 }
41 }
42
43 pub fn expect_any_char(&mut self, chars: &[char]) -> Result<(), ()> {
44 match self.next() {
45 Some(TtElement::Leaf(Leaf::Punct(Punct { char: c, .. }))) if chars.contains(c) => {
46 Ok(())
47 }
48 _ => Err(()),
49 }
50 }
51
52 pub fn expect_subtree(&mut self) -> Result<(&'a Subtree<S>, TtIter<'a, S>), ()> {
53 match self.next() {
54 Some(TtElement::Subtree(subtree, iter)) => Ok((subtree, iter)),
55 _ => Err(()),
56 }
57 }
58
59 pub fn expect_leaf(&mut self) -> Result<&'a Leaf<S>, ()> {
60 match self.next() {
61 Some(TtElement::Leaf(it)) => Ok(it),
62 _ => Err(()),
63 }
64 }
65
66 pub fn expect_dollar(&mut self) -> Result<(), ()> {
67 match self.expect_leaf()? {
68 Leaf::Punct(Punct { char: '$', .. }) => Ok(()),
69 _ => Err(()),
70 }
71 }
72
73 pub fn expect_comma(&mut self) -> Result<(), ()> {
74 match self.expect_leaf()? {
75 Leaf::Punct(Punct { char: ',', .. }) => Ok(()),
76 _ => Err(()),
77 }
78 }
79
80 pub fn expect_ident(&mut self) -> Result<&'a Ident<S>, ()> {
81 match self.expect_leaf()? {
82 Leaf::Ident(it) if it.sym != sym::underscore => Ok(it),
83 _ => Err(()),
84 }
85 }
86
87 pub fn expect_ident_or_underscore(&mut self) -> Result<&'a Ident<S>, ()> {
88 match self.expect_leaf()? {
89 Leaf::Ident(it) => Ok(it),
90 _ => Err(()),
91 }
92 }
93
94 pub fn expect_literal(&mut self) -> Result<&'a Leaf<S>, ()> {
95 let it = self.expect_leaf()?;
96 match it {
97 Leaf::Literal(_) => Ok(it),
98 Leaf::Ident(ident) if ident.sym == sym::true_ || ident.sym == sym::false_ => Ok(it),
99 _ => Err(()),
100 }
101 }
102
103 pub fn expect_single_punct(&mut self) -> Result<&'a Punct<S>, ()> {
104 match self.expect_leaf()? {
105 Leaf::Punct(it) => Ok(it),
106 _ => Err(()),
107 }
108 }
109
110 pub fn expect_glued_punct(&mut self) -> Result<ArrayVec<Punct<S>, MAX_GLUED_PUNCT_LEN>, ()> {
115 let TtElement::Leaf(&Leaf::Punct(first)) = self.next().ok_or(())? else {
116 return Err(());
117 };
118
119 let mut res = ArrayVec::new();
120 if first.spacing == Spacing::Alone {
121 res.push(first);
122 return Ok(res);
123 }
124
125 let (second, third) = match (self.peek_n(0), self.peek_n(1)) {
126 (Some(TokenTree::Leaf(Leaf::Punct(p2))), Some(TokenTree::Leaf(Leaf::Punct(p3))))
127 if p2.spacing == Spacing::Joint =>
128 {
129 (p2, Some(p3))
130 }
131 (Some(TokenTree::Leaf(Leaf::Punct(p2))), _) => (p2, None),
132 _ => {
133 res.push(first);
134 return Ok(res);
135 }
136 };
137
138 match (first.char, second.char, third.map(|it| it.char)) {
139 ('.', '.', Some('.' | '=')) | ('<', '<', Some('=')) | ('>', '>', Some('=')) => {
140 let _ = self.next().unwrap();
141 let _ = self.next().unwrap();
142 res.push(first);
143 res.push(*second);
144 res.push(*third.unwrap());
145 }
146 ('-' | '!' | '*' | '/' | '&' | '%' | '^' | '+' | '<' | '=' | '>' | '|', '=', _)
147 | ('-' | '=' | '>', '>', _)
148 | ('<', '-', _)
149 | (':', ':', _)
150 | ('.', '.', _)
151 | ('&', '&', _)
152 | ('<', '<', _)
153 | ('|', '|', _) => {
154 let _ = self.next().unwrap();
155 res.push(first);
156 res.push(*second);
157 }
158 _ => res.push(first),
159 }
160 Ok(res)
161 }
162
163 fn peek_n(&self, n: usize) -> Option<&'a TokenTree<S>> {
165 self.inner.as_slice().get(n)
166 }
167
168 pub fn peek(&self) -> Option<TtElement<'a, S>> {
169 match self.inner.as_slice().first()? {
170 TokenTree::Leaf(leaf) => Some(TtElement::Leaf(leaf)),
171 TokenTree::Subtree(subtree) => {
172 let nested_iter =
173 TtIter { inner: self.inner.as_slice()[1..][..subtree.usize_len()].iter() };
174 Some(TtElement::Subtree(subtree, nested_iter))
175 }
176 }
177 }
178
179 pub fn is_empty(&self) -> bool {
181 self.inner.len() == 0
182 }
183
184 pub fn next_span(&self) -> Option<S> {
185 Some(self.inner.as_slice().first()?.first_span())
186 }
187
188 pub fn remaining(&self) -> TokenTreesView<'a, S> {
189 TokenTreesView::new(self.inner.as_slice())
190 }
191
192 pub fn flat_advance(&mut self, skip: usize) {
194 self.inner = self.inner.as_slice()[skip..].iter();
195 }
196
197 pub fn savepoint(&self) -> TtIterSavepoint<'a, S> {
198 TtIterSavepoint(self.inner.as_slice())
199 }
200
201 pub fn from_savepoint(&self, savepoint: TtIterSavepoint<'a, S>) -> TokenTreesView<'a, S> {
202 let len = (self.inner.as_slice().as_ptr() as usize - savepoint.0.as_ptr() as usize)
203 / size_of::<TokenTree<S>>();
204 TokenTreesView::new(&savepoint.0[..len])
205 }
206
207 pub fn next_as_view(&mut self) -> Option<TokenTreesView<'a, S>> {
208 let savepoint = self.savepoint();
209 self.next()?;
210 Some(self.from_savepoint(savepoint))
211 }
212}
213
214#[derive(Clone)]
215pub enum TtElement<'a, S> {
216 Leaf(&'a Leaf<S>),
217 Subtree(&'a Subtree<S>, TtIter<'a, S>),
218}
219
220impl<S: Copy + fmt::Debug> fmt::Debug for TtElement<'_, S> {
221 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
222 match self {
223 Self::Leaf(leaf) => f.debug_tuple("Leaf").field(leaf).finish(),
224 Self::Subtree(subtree, inner) => {
225 f.debug_tuple("Subtree").field(subtree).field(inner).finish()
226 }
227 }
228 }
229}
230
231impl<S: Copy> TtElement<'_, S> {
232 #[inline]
233 pub fn first_span(&self) -> S {
234 match self {
235 TtElement::Leaf(it) => *it.span(),
236 TtElement::Subtree(it, _) => it.delimiter.open,
237 }
238 }
239}
240
241impl<'a, S> Iterator for TtIter<'a, S> {
242 type Item = TtElement<'a, S>;
243 fn next(&mut self) -> Option<Self::Item> {
244 match self.inner.next()? {
245 TokenTree::Leaf(leaf) => Some(TtElement::Leaf(leaf)),
246 TokenTree::Subtree(subtree) => {
247 let nested_iter =
248 TtIter { inner: self.inner.as_slice()[..subtree.usize_len()].iter() };
249 self.inner = self.inner.as_slice()[subtree.usize_len()..].iter();
250 Some(TtElement::Subtree(subtree, nested_iter))
251 }
252 }
253 }
254}