proc_macro::Group::span_open and span_close
Before this addition, every delimited group like (...) [...] {...} has
only a single Span that covers the full source location from opening
delimiter to closing delimiter. This makes it impossible for a
procedural macro to trigger an error pointing to just the opening or
closing delimiter. The Rust compiler does not seem to have the same
limitation:
mod m {
type T =
}
error: expected type, found `}`
--> src/main.rs:3:1
|
3 | }
| ^
On that same input, a procedural macro would be forced to trigger the
error on the last token inside the block, on the entire block, or on the
next token after the block, none of which is really what you want for an
error like above.
This commit adds group.span_open() and group.span_close() which access
the Span associated with just the opening delimiter and just the closing
delimiter of the group. Relevant to Syn as we implement real error
messages for when parsing fails in a procedural macro.
This commit is contained in:
parent
3480ac2a80
commit
c5a561c0ab
1 changed files with 42 additions and 1 deletions
|
|
@ -64,7 +64,7 @@ use syntax::errors::DiagnosticBuilder;
|
|||
use syntax::parse::{self, token};
|
||||
use syntax::symbol::Symbol;
|
||||
use syntax::tokenstream;
|
||||
use syntax_pos::{Pos, FileName};
|
||||
use syntax_pos::{BytePos, Pos, FileName};
|
||||
|
||||
/// The main type provided by this crate, representing an abstract stream of
|
||||
/// tokens, or, more specifically, a sequence of token trees.
|
||||
|
|
@ -671,11 +671,52 @@ impl Group {
|
|||
|
||||
/// Returns the span for the delimiters of this token stream, spanning the
|
||||
/// entire `Group`.
|
||||
///
|
||||
/// ```text
|
||||
/// pub fn span(&self) -> Span {
|
||||
/// ^^^^^^^
|
||||
/// ```
|
||||
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
|
||||
pub fn span(&self) -> Span {
|
||||
self.span
|
||||
}
|
||||
|
||||
/// Returns the span pointing to the opening delimiter of this group, or the
|
||||
/// span of the entire group if this is a None-delimited group.
|
||||
///
|
||||
/// ```text
|
||||
/// pub fn span_open(&self) -> Span {
|
||||
/// ^
|
||||
/// ```
|
||||
#[unstable(feature = "proc_macro_span", issue = "38356")]
|
||||
pub fn span_open(&self) -> Span {
|
||||
if self.delimiter == Delimiter::None {
|
||||
self.span
|
||||
} else {
|
||||
let lo = self.span.0.lo();
|
||||
let new_hi = BytePos::from_usize(lo.to_usize() + 1);
|
||||
Span(self.span.0.with_hi(new_hi))
|
||||
}
|
||||
}
|
||||
|
||||
/// Returns the span pointing to the closing delimiter of this group, or the
|
||||
/// span of the entire group if this is a None-delimited group.
|
||||
///
|
||||
/// ```text
|
||||
/// pub fn span_close(&self) -> Span {
|
||||
/// ^
|
||||
/// ```
|
||||
#[unstable(feature = "proc_macro_span", issue = "38356")]
|
||||
pub fn span_close(&self) -> Span {
|
||||
let hi = self.span.0.hi();
|
||||
if self.delimiter == Delimiter::None || hi.to_usize() == 0 {
|
||||
self.span
|
||||
} else {
|
||||
let new_lo = BytePos::from_usize(hi.to_usize() - 1);
|
||||
Span(self.span.0.with_lo(new_lo))
|
||||
}
|
||||
}
|
||||
|
||||
/// Configures the span for this `Group`'s delimiters, but not its internal
|
||||
/// tokens.
|
||||
///
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue