librustc_lexer: Simplify "lifetime_or_char" method

This commit is contained in:
Igor Aleksanov 2019-11-03 12:56:49 +03:00
parent 6e350bd999
commit ecd26739d4

View File

@ -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.