Improve doc comment desugaring.
Sometimes the parser needs to desugar a doc comment into `#[doc = r"foo"]`. Currently it does this in a hacky way: by pushing a "fake" new frame (one without a delimiter) onto the `TokenCursor` stack. This commit changes things so that the token stream itself is modified in place. The nice thing about this is that it means `TokenCursorFrame::delim_sp` is now only `None` for the outermost frame.
This commit is contained in:
parent
97872b792c
commit
af1d16e82d
@ -614,6 +614,15 @@ impl Cursor {
|
||||
pub fn look_ahead(&self, n: usize) -> Option<&TokenTree> {
|
||||
self.stream.0.get(self.index + n)
|
||||
}
|
||||
|
||||
// Replace the previously obtained token tree with `tts`, and rewind to
|
||||
// just before them.
|
||||
pub fn replace_prev_and_rewind(&mut self, tts: Vec<TokenTree>) {
|
||||
assert!(self.index > 0);
|
||||
self.index -= 1;
|
||||
let stream = Lrc::make_mut(&mut self.stream.0);
|
||||
stream.splice(self.index..self.index + 1, tts);
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Copy, Clone, PartialEq, Encodable, Decodable, HashStable_Generic)]
|
||||
|
@ -224,7 +224,7 @@ impl<'a> Drop for Parser<'a> {
|
||||
#[derive(Clone)]
|
||||
struct TokenCursor {
|
||||
// The current (innermost) frame. `frame` and `stack` could be combined,
|
||||
// but it's faster to have them separately to access `frame` directly
|
||||
// but it's faster to keep them separate and access `frame` directly
|
||||
// rather than via something like `stack.last().unwrap()` or
|
||||
// `stack[stack.len() - 1]`.
|
||||
frame: TokenCursorFrame,
|
||||
@ -259,6 +259,7 @@ struct TokenCursor {
|
||||
|
||||
#[derive(Clone)]
|
||||
struct TokenCursorFrame {
|
||||
// This is `None` only for the outermost frame.
|
||||
delim_sp: Option<(Delimiter, DelimSpan)>,
|
||||
tree_cursor: tokenstream::Cursor,
|
||||
}
|
||||
@ -285,7 +286,9 @@ impl TokenCursor {
|
||||
match tree {
|
||||
&TokenTree::Token(ref token, spacing) => match (desugar_doc_comments, token) {
|
||||
(true, &Token { kind: token::DocComment(_, attr_style, data), span }) => {
|
||||
return self.desugar(attr_style, data, span);
|
||||
let desugared = self.desugar(attr_style, data, span);
|
||||
self.frame.tree_cursor.replace_prev_and_rewind(desugared);
|
||||
// Continue to get the first token of the desugared doc comment.
|
||||
}
|
||||
_ => return (token.clone(), spacing),
|
||||
},
|
||||
@ -300,19 +303,22 @@ impl TokenCursor {
|
||||
}
|
||||
};
|
||||
} else if let Some(frame) = self.stack.pop() {
|
||||
if let Some((delim, span)) = self.frame.delim_sp && delim != Delimiter::Invisible {
|
||||
self.frame = frame;
|
||||
// We have exhausted this frame. Move back to its parent frame.
|
||||
let (delim, span) = self.frame.delim_sp.unwrap();
|
||||
self.frame = frame;
|
||||
if delim != Delimiter::Invisible {
|
||||
return (Token::new(token::CloseDelim(delim), span.close), Spacing::Alone);
|
||||
}
|
||||
self.frame = frame;
|
||||
// No close delimiter to return; continue on to the next iteration.
|
||||
} else {
|
||||
// We have exhausted the outermost frame.
|
||||
return (Token::new(token::Eof, DUMMY_SP), Spacing::Alone);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn desugar(&mut self, attr_style: AttrStyle, data: Symbol, span: Span) -> (Token, Spacing) {
|
||||
// Desugar a doc comment into something like `#[doc = r"foo"]`.
|
||||
fn desugar(&mut self, attr_style: AttrStyle, data: Symbol, span: Span) -> Vec<TokenTree> {
|
||||
// Searches for the occurrences of `"#*` and returns the minimum number of `#`s
|
||||
// required to wrap the text. E.g.
|
||||
// - `abc d` is wrapped as `r"abc d"` (num_of_hashes = 0)
|
||||
@ -346,27 +352,15 @@ impl TokenCursor {
|
||||
.collect::<TokenStream>(),
|
||||
);
|
||||
|
||||
self.stack.push(mem::replace(
|
||||
&mut self.frame,
|
||||
TokenCursorFrame::new(
|
||||
None,
|
||||
if attr_style == AttrStyle::Inner {
|
||||
[
|
||||
TokenTree::token_alone(token::Pound, span),
|
||||
TokenTree::token_alone(token::Not, span),
|
||||
body,
|
||||
]
|
||||
.into_iter()
|
||||
.collect::<TokenStream>()
|
||||
} else {
|
||||
[TokenTree::token_alone(token::Pound, span), body]
|
||||
.into_iter()
|
||||
.collect::<TokenStream>()
|
||||
},
|
||||
),
|
||||
));
|
||||
|
||||
self.next(/* desugar_doc_comments */ false)
|
||||
if attr_style == AttrStyle::Inner {
|
||||
vec![
|
||||
TokenTree::token_alone(token::Pound, span),
|
||||
TokenTree::token_alone(token::Not, span),
|
||||
body,
|
||||
]
|
||||
} else {
|
||||
vec![TokenTree::token_alone(token::Pound, span), body]
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user