librustc_lexer: Simplify "lifetime_or_char" method
This commit is contained in:
parent
6e350bd999
commit
ecd26739d4
@ -498,34 +498,18 @@ fn number(&mut self, first_digit: char) -> LiteralKind {
|
|||||||
|
|
||||||
fn lifetime_or_char(&mut self) -> TokenKind {
|
fn lifetime_or_char(&mut self) -> TokenKind {
|
||||||
debug_assert!(self.prev() == '\'');
|
debug_assert!(self.prev() == '\'');
|
||||||
let mut starts_with_number = false;
|
|
||||||
|
|
||||||
// Check if the first symbol after '\'' is a valid identifier
|
let can_be_a_lifetime = if self.second() == '\'' {
|
||||||
// character or a number (not a digit followed by '\'').
|
// It's surely not a lifetime.
|
||||||
if (is_id_start(self.nth_char(0))
|
false
|
||||||
|| self.nth_char(0).is_digit(10) && {
|
|
||||||
starts_with_number = true;
|
|
||||||
true
|
|
||||||
})
|
|
||||||
&& self.nth_char(1) != '\''
|
|
||||||
{
|
|
||||||
self.bump();
|
|
||||||
|
|
||||||
// Skip the identifier.
|
|
||||||
while is_id_continue(self.nth_char(0)) {
|
|
||||||
self.bump();
|
|
||||||
}
|
|
||||||
|
|
||||||
return if self.nth_char(0) == '\'' {
|
|
||||||
self.bump();
|
|
||||||
let kind = Char { terminated: true };
|
|
||||||
Literal { kind, suffix_start: self.len_consumed() }
|
|
||||||
} else {
|
} else {
|
||||||
Lifetime { starts_with_number }
|
// If the first symbol is valid for identifier, it can be a lifetime.
|
||||||
|
// Also check if it's a number for a better error reporting (so '0 will
|
||||||
|
// be reported as invalid lifetime and not as unterminated char literal).
|
||||||
|
is_id_start(self.first()) || self.first().is_digit(10)
|
||||||
};
|
};
|
||||||
}
|
|
||||||
|
|
||||||
// This is not a lifetime (checked above), parse a char literal.
|
if !can_be_a_lifetime {
|
||||||
let terminated = self.single_quoted_string();
|
let terminated = self.single_quoted_string();
|
||||||
let suffix_start = self.len_consumed();
|
let suffix_start = self.len_consumed();
|
||||||
if terminated {
|
if terminated {
|
||||||
@ -535,6 +519,29 @@ fn lifetime_or_char(&mut self) -> TokenKind {
|
|||||||
return Literal { kind, suffix_start };
|
return Literal { kind, suffix_start };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Either a lifetime or a character literal with
|
||||||
|
// length greater than 1.
|
||||||
|
|
||||||
|
let starts_with_number = self.first().is_digit(10);
|
||||||
|
|
||||||
|
// Skip the literal contents.
|
||||||
|
// First symbol can be a number (which isn't a valid identifier start),
|
||||||
|
// so skip it without any checks.
|
||||||
|
self.bump();
|
||||||
|
self.eat_while(is_id_continue);
|
||||||
|
|
||||||
|
// Check if after skipping literal contents we've met a closing
|
||||||
|
// single quote (which means that user attempted to create a
|
||||||
|
// string with single quotes).
|
||||||
|
if self.first() == '\'' {
|
||||||
|
self.bump();
|
||||||
|
let kind = Char { terminated: true };
|
||||||
|
return Literal { kind, suffix_start: self.len_consumed() };
|
||||||
|
}
|
||||||
|
|
||||||
|
return Lifetime { starts_with_number };
|
||||||
|
}
|
||||||
|
|
||||||
fn single_quoted_string(&mut self) -> bool {
|
fn single_quoted_string(&mut self) -> bool {
|
||||||
debug_assert!(self.prev() == '\'');
|
debug_assert!(self.prev() == '\'');
|
||||||
// Check if it's a one-symbol literal.
|
// Check if it's a one-symbol literal.
|
||||||
|
Loading…
Reference in New Issue
Block a user