blob: cfc1d400f778b9bd2345476fa355a8027a46720d [file] [log] [blame] [edit]
// Protocol Buffers - Google's data interchange format
// Copyright 2023 Google LLC. All rights reserved.
//
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file or at
// https://developers.google.com/open-source/licenses/bsd
#include "upb/io/tokenizer.h"
#include "upb/io/string.h"
#include "upb/lex/strtod.h"
#include "upb/lex/unicode.h"
// Must be included last.
#include "upb/port/def.inc"
typedef enum {
// Started a line comment.
kUpb_CommentType_Line,
// Started a block comment.
kUpb_CommentType_Block,
// Consumed a slash, then realized it wasn't a comment. current_ has
// been filled in with a slash token. The caller should return it.
kUpb_CommentType_SlashNot,
// We do not appear to be starting a comment here.
kUpb_CommentType_None,
} upb_CommentType;
static bool upb_Tokenizer_IsUnprintable(char c) { return '\0' < c && c < ' '; }
// Since we count columns we need to interpret tabs somehow. We'll take
// the standard 8-character definition for lack of any way to do better.
static const int kUpb_Tokenizer_TabWidth = 8;
// Given a char, interpret it as a numeric digit and return its value.
// This supports any number base up to 36.
// Represents integer values of digits.
// Uses 36 to indicate an invalid character since we support
// bases up to 36.
static const int8_t kUpb_Tokenizer_AsciiToInt[256] = {
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // 00-0F
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // 10-1F
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // ' '-'/'
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, // '0'-'9'
36, 36, 36, 36, 36, 36, 36, // ':'-'@'
10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, // 'A'-'P'
26, 27, 28, 29, 30, 31, 32, 33, 34, 35, // 'Q'-'Z'
36, 36, 36, 36, 36, 36, // '['-'`'
10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, // 'a'-'p'
26, 27, 28, 29, 30, 31, 32, 33, 34, 35, // 'q'-'z'
36, 36, 36, 36, 36, // '{'-DEL
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // 80-8F
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // 90-9F
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // A0-AF
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // B0-BF
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // C0-CF
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // D0-DF
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // E0-EF
36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, 36, // F0-FF
};
static int DigitValue(char digit) {
return kUpb_Tokenizer_AsciiToInt[digit & 0xFF];
}
static bool upb_Tokenizer_IsLetter(char c) {
return ('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || (c == '_');
}
static bool upb_Tokenizer_IsDigit(char c) { return '0' <= c && c <= '9'; }
static bool upb_Tokenizer_IsOctalDigit(char c) { return '0' <= c && c <= '7'; }
static bool upb_Tokenizer_IsHexDigit(char c) {
return ('0' <= c && c <= '9') || ('a' <= c && c <= 'f') ||
('A' <= c && c <= 'F');
}
static bool upb_Tokenizer_IsAlphanumeric(char c) {
return upb_Tokenizer_IsLetter(c) || upb_Tokenizer_IsDigit(c);
}
static bool upb_Tokenizer_IsWhitespaceNoNewline(char c) {
return c == ' ' || c == '\t' || c == '\r' || c == '\v' || c == '\f';
}
static bool upb_Tokenizer_IsWhitespace(char c) {
return c == '\n' || upb_Tokenizer_IsWhitespaceNoNewline(c);
}
static bool upb_Tokenizer_IsEscape(char c) {
return c == 'a' || c == 'b' || c == 'f' || c == 'n' || c == 'r' || c == 't' ||
c == 'v' || c == '\\' || c == '?' || c == '\'' || c == '\"';
}
static char TranslateEscape(char c) {
switch (c) {
case 'a':
return '\a';
case 'b':
return '\b';
case 'f':
return '\f';
case 'n':
return '\n';
case 'r':
return '\r';
case 't':
return '\t';
case 'v':
return '\v';
case '\\':
return '\\';
case '?':
return '\?'; // Trigraphs = :(
case '\'':
return '\'';
case '"':
return '\"';
// We expect escape sequences to have been validated separately.
default:
return '?';
}
}
// ===================================================================
struct upb_Tokenizer {
upb_TokenType token_type; // The type of the current token.
// The exact text of the current token as it appeared in the input.
// e.g. tokens of TYPE_STRING will still be escaped and in quotes.
upb_String token_text;
// "line" and "column" specify the position of the first character of
// the token within the input stream. They are zero-based.
int token_line;
int token_column;
int token_end_column;
upb_ZeroCopyInputStream* input;
upb_Arena* arena;
upb_Status* status;
char current_char; // == buffer_[buffer_pos_], updated by NextChar().
const char* buffer; // Current buffer returned from input_.
size_t buffer_size; // Size of buffer_.
size_t buffer_pos; // Current position within the buffer.
bool read_error; // Did we previously encounter a read error?
// Line and column number of current_char_ within the whole input stream.
int line;
// By "column number", the proto compiler refers to a count of the number
// of bytes before a given byte, except that a tab character advances to
// the next multiple of 8 bytes. Note in particular that column numbers
// are zero-based, while many user interfaces use one-based column numbers.
int column;
// Cached values from before the most recent call to Next()
upb_TokenType previous_type;
int previous_line;
int previous_column;
int previous_end_column;
// String to which text should be appended as we advance through it.
// Call RecordTo(&str) to start recording and StopRecording() to stop.
// E.g. StartToken() calls RecordTo(&current_.text). record_start_ is the
// position within the current buffer where recording started.
upb_String* record_target;
int record_start;
int options;
jmp_buf err;
};
// Convenience methods to return an error at the current line and column.
UPB_NORETURN static void ReportError(upb_Tokenizer* t, const char* msg) {
upb_Status_SetErrorFormat(t->status, "%d:%d: %s", t->line, t->column, msg);
UPB_LONGJMP(t->err, 1);
}
UPB_NORETURN UPB_PRINTF(2, 3) static void ReportErrorFormat(upb_Tokenizer* t,
const char* fmt,
...) {
va_list args;
va_start(args, fmt);
char msg[128];
vsnprintf(msg, sizeof(msg), fmt, args);
ReportError(t, msg);
}
// Read a new buffer from the input.
static void Refresh(upb_Tokenizer* t) {
if (t->read_error) {
t->current_char = '\0';
return;
}
// If we're in a token, append the rest of the buffer to it.
if (t->record_target != NULL && t->record_start < t->buffer_size) {
upb_String_Append(t->record_target, t->buffer + t->record_start,
t->buffer_size - t->record_start);
t->record_start = 0;
}
t->buffer = NULL;
t->buffer_pos = 0;
upb_Status status;
const void* data =
upb_ZeroCopyInputStream_Next(t->input, &t->buffer_size, &status);
if (t->buffer_size > 0) {
t->buffer = data;
t->current_char = t->buffer[0];
} else {
// end of stream (or read error)
t->buffer_size = 0;
t->read_error = true;
t->current_char = '\0';
}
}
// Consume this character and advance to the next one.
static void NextChar(upb_Tokenizer* t) {
// Update our line and column counters based on the character being
// consumed.
if (t->current_char == '\n') {
t->line++;
t->column = 0;
} else if (t->current_char == '\t') {
t->column += kUpb_Tokenizer_TabWidth - t->column % kUpb_Tokenizer_TabWidth;
} else {
t->column++;
}
// Advance to the next character.
t->buffer_pos++;
if (t->buffer_pos < t->buffer_size) {
t->current_char = t->buffer[t->buffer_pos];
} else {
Refresh(t);
}
}
static void RecordTo(upb_Tokenizer* t, upb_String* target) {
t->record_target = target;
t->record_start = t->buffer_pos;
}
static void StopRecording(upb_Tokenizer* t) {
if (t->buffer_pos > t->record_start) {
upb_String_Append(t->record_target, t->buffer + t->record_start,
t->buffer_pos - t->record_start);
}
t->record_target = NULL;
t->record_start = -1;
}
// Called when the current character is the first character of a new
// token (not including whitespace or comments).
static void StartToken(upb_Tokenizer* t) {
t->token_type = kUpb_TokenType_Start;
upb_String_Clear(&t->token_text);
t->token_line = t->line;
t->token_column = t->column;
RecordTo(t, &t->token_text);
}
// Called when the current character is the first character after the
// end of the last token. After this returns, current_.text will
// contain all text consumed since StartToken() was called.
static void EndToken(upb_Tokenizer* t) {
StopRecording(t);
t->token_end_column = t->column;
}
// -----------------------------------------------------------------
// These helper methods make the parsing code more readable.
// The "character classes" referred to are defined at the top of the file.
// The method returns true if c is a member of this "class", like "Letter"
// or "Digit".
// Returns true if the current character is of the given character
// class, but does not consume anything.
static bool LookingAt(const upb_Tokenizer* t, bool (*f)(char)) {
return f(t->current_char);
}
// If the current character is in the given class, consume it and return true.
// Otherwise return false.
static bool TryConsumeOne(upb_Tokenizer* t, bool (*f)(char)) {
if (f(t->current_char)) {
NextChar(t);
return true;
} else {
return false;
}
}
// Like above, but try to consume the specific character indicated.
static bool TryConsume(upb_Tokenizer* t, char c) {
if (t->current_char == c) {
NextChar(t);
return true;
} else {
return false;
}
}
// Consume zero or more of the given character class.
static void ConsumeZeroOrMore(upb_Tokenizer* t, bool (*f)(char)) {
while (f(t->current_char)) {
NextChar(t);
}
}
// Consume one or more of the given character class or log the given
// error message.
static void ConsumeOneOrMore(upb_Tokenizer* t, bool (*f)(char),
const char* err_msg) {
if (!f(t->current_char)) {
ReportError(t, err_msg);
}
do {
NextChar(t);
} while (f(t->current_char));
}
// -----------------------------------------------------------------
// The following four methods are used to consume tokens of specific
// types. They are actually used to consume all characters *after*
// the first, since the calling function consumes the first character
// in order to decide what kind of token is being read.
// Read and consume a string, ending when the given delimiter is consumed.
static void ConsumeString(upb_Tokenizer* t, char delimiter) {
while (true) {
switch (t->current_char) {
case '\0':
ReportError(t, "Unexpected end of string.");
case '\n':
ReportError(t, "String literals cannot cross line boundaries.");
case '\\': {
// An escape sequence.
NextChar(t);
if (TryConsumeOne(t, upb_Tokenizer_IsEscape)) {
// Valid escape sequence.
} else if (TryConsumeOne(t, upb_Tokenizer_IsOctalDigit)) {
// Possibly followed by two more octal digits, but these will
// just be consumed by the main loop anyway so we don't need
// to do so explicitly here.
} else if (TryConsume(t, 'x')) {
if (!TryConsumeOne(t, upb_Tokenizer_IsHexDigit)) {
ReportError(t, "Expected hex digits for escape sequence.");
}
// Possibly followed by another hex digit, but again we don't care.
} else if (TryConsume(t, 'u')) {
if (!TryConsumeOne(t, upb_Tokenizer_IsHexDigit) ||
!TryConsumeOne(t, upb_Tokenizer_IsHexDigit) ||
!TryConsumeOne(t, upb_Tokenizer_IsHexDigit) ||
!TryConsumeOne(t, upb_Tokenizer_IsHexDigit)) {
ReportError(t, "Expected four hex digits for \\u escape sequence.");
}
} else if (TryConsume(t, 'U')) {
// We expect 8 hex digits; but only the range up to 0x10ffff is
// legal.
if (!TryConsume(t, '0') || !TryConsume(t, '0') ||
!(TryConsume(t, '0') || TryConsume(t, '1')) ||
!TryConsumeOne(t, upb_Tokenizer_IsHexDigit) ||
!TryConsumeOne(t, upb_Tokenizer_IsHexDigit) ||
!TryConsumeOne(t, upb_Tokenizer_IsHexDigit) ||
!TryConsumeOne(t, upb_Tokenizer_IsHexDigit) ||
!TryConsumeOne(t, upb_Tokenizer_IsHexDigit)) {
ReportError(t,
"Expected eight hex digits up to 10ffff for \\U escape "
"sequence");
}
} else {
ReportError(t, "Invalid escape sequence in string literal.");
}
break;
}
default: {
if (t->current_char == delimiter) {
NextChar(t);
return;
}
NextChar(t);
break;
}
}
}
}
// Read and consume a number, returning TYPE_FLOAT or TYPE_INTEGER depending
// on what was read. This needs to know if the first characer was a zero in
// order to correctly recognize hex and octal numbers. It also needs to know
// whether the first character was a '.' to parse floating point correctly.
static upb_TokenType ConsumeNumber(upb_Tokenizer* t, bool started_with_zero,
bool started_with_dot) {
bool is_float = false;
if (started_with_zero && (TryConsume(t, 'x') || TryConsume(t, 'X'))) {
// A hex number (started with "0x").
ConsumeOneOrMore(t, upb_Tokenizer_IsHexDigit,
"\"0x\" must be followed by hex digits.");
} else if (started_with_zero && LookingAt(t, upb_Tokenizer_IsDigit)) {
// An octal number (had a leading zero).
ConsumeZeroOrMore(t, upb_Tokenizer_IsOctalDigit);
if (LookingAt(t, upb_Tokenizer_IsDigit)) {
ReportError(t, "Numbers starting with leading zero must be in octal.");
}
} else {
// A decimal number.
if (started_with_dot) {
is_float = true;
ConsumeZeroOrMore(t, upb_Tokenizer_IsDigit);
} else {
ConsumeZeroOrMore(t, upb_Tokenizer_IsDigit);
if (TryConsume(t, '.')) {
is_float = true;
ConsumeZeroOrMore(t, upb_Tokenizer_IsDigit);
}
}
if (TryConsume(t, 'e') || TryConsume(t, 'E')) {
is_float = true;
if (!TryConsume(t, '-')) TryConsume(t, '+');
ConsumeOneOrMore(t, upb_Tokenizer_IsDigit,
"\"e\" must be followed by exponent.");
}
if (t->options & kUpb_TokenizerOption_AllowFAfterFloat) {
if (TryConsume(t, 'f') || TryConsume(t, 'F')) is_float = true;
}
}
if (LookingAt(t, upb_Tokenizer_IsLetter)) {
ReportError(t, "Need space between number and identifier.");
}
if (t->current_char == '.') {
if (is_float) {
ReportError(
t, "Already saw decimal point or exponent; can't have another one.");
} else {
ReportError(t, "Hex and octal numbers must be integers.");
}
}
return is_float ? kUpb_TokenType_Float : kUpb_TokenType_Integer;
}
// Consume the rest of a line.
static void ConsumeLineComment(upb_Tokenizer* t, upb_String* content) {
if (content != NULL) RecordTo(t, content);
while (t->current_char != '\0' && t->current_char != '\n') {
NextChar(t);
}
TryConsume(t, '\n');
if (content != NULL) StopRecording(t);
}
static void ConsumeBlockComment(upb_Tokenizer* t, upb_String* content) {
const int start_line = t->line;
const int start_column = t->column - 2;
if (content != NULL) RecordTo(t, content);
while (true) {
while (t->current_char != '\0' && t->current_char != '*' &&
t->current_char != '/' && t->current_char != '\n') {
NextChar(t);
}
if (TryConsume(t, '\n')) {
if (content != NULL) StopRecording(t);
// Consume leading whitespace and asterisk;
ConsumeZeroOrMore(t, upb_Tokenizer_IsWhitespaceNoNewline);
if (TryConsume(t, '*')) {
if (TryConsume(t, '/')) {
// End of comment.
break;
}
}
if (content != NULL) RecordTo(t, content);
} else if (TryConsume(t, '*') && TryConsume(t, '/')) {
// End of comment.
if (content != NULL) {
StopRecording(t);
// Strip trailing "*/".
upb_String_Erase(content, upb_String_Size(content) - 2, 2);
}
break;
} else if (TryConsume(t, '/') && t->current_char == '*') {
// Note: We didn't consume the '*' because if there is a '/' after it
// we want to interpret that as the end of the comment.
ReportError(
t, "\"/*\" inside block comment. Block comments cannot be nested.");
} else if (t->current_char == '\0') {
ReportErrorFormat(
t, "End-of-file inside block comment.\n%d:%d: Comment started here.",
start_line, start_column);
}
}
}
// If we're at the start of a new comment, consume it and return what kind
// of comment it is.
static upb_CommentType TryConsumeCommentStart(upb_Tokenizer* t) {
const bool style_sh = t->options & kUpb_TokenizerOption_CommentStyleShell;
const bool style_cpp = !style_sh;
if (style_cpp && TryConsume(t, '/')) {
if (TryConsume(t, '/')) {
return kUpb_CommentType_Line;
} else if (TryConsume(t, '*')) {
return kUpb_CommentType_Block;
} else {
// Oops, it was just a slash. Return it.
t->token_type = kUpb_TokenType_Symbol;
upb_String_Assign(&t->token_text, "/", 1);
t->token_line = t->line;
t->token_column = t->column - 1;
t->token_end_column = t->column;
return kUpb_CommentType_SlashNot;
}
} else if (style_sh && TryConsume(t, '#')) {
return kUpb_CommentType_Line;
} else {
return kUpb_CommentType_None;
}
}
// If we're looking at a TYPE_WHITESPACE token and `report_whitespace` is true,
// consume it and return true.
static bool TryConsumeWhitespace(upb_Tokenizer* t) {
if (t->options & kUpb_TokenizerOption_ReportNewlines) {
if (TryConsumeOne(t, upb_Tokenizer_IsWhitespaceNoNewline)) {
ConsumeZeroOrMore(t, upb_Tokenizer_IsWhitespaceNoNewline);
t->token_type = kUpb_TokenType_Whitespace;
return true;
}
return false;
}
if (TryConsumeOne(t, upb_Tokenizer_IsWhitespace)) {
ConsumeZeroOrMore(t, upb_Tokenizer_IsWhitespace);
t->token_type = kUpb_TokenType_Whitespace;
return (t->options & kUpb_TokenizerOption_ReportWhitespace) != 0;
}
return false;
}
// If we're looking at a TYPE_NEWLINE token and `report_newlines` is true,
// consume it and return true.
static bool TryConsumeNewline(upb_Tokenizer* t) {
if (t->options & kUpb_TokenizerOption_ReportNewlines) {
if (TryConsume(t, '\n')) {
t->token_type = kUpb_TokenType_Newline;
return true;
}
}
return false;
}
// -------------------------------------------------------------------
int upb_Tokenizer_Column(const upb_Tokenizer* t) { return t->token_column; }
int upb_Tokenizer_EndColumn(const upb_Tokenizer* t) {
return t->token_end_column;
}
int upb_Tokenizer_Line(const upb_Tokenizer* t) { return t->token_line; }
int upb_Tokenizer_TextSize(const upb_Tokenizer* t) {
return t->token_text.size_;
}
const char* upb_Tokenizer_TextData(const upb_Tokenizer* t) {
return t->token_text.data_;
}
upb_TokenType upb_Tokenizer_Type(const upb_Tokenizer* t) {
return t->token_type;
}
bool upb_Tokenizer_Next(upb_Tokenizer* t, upb_Status* status) {
t->status = status;
t->previous_type = t->token_type;
t->previous_line = t->token_line;
t->previous_column = t->token_column;
t->previous_end_column = t->token_end_column;
if (UPB_SETJMP(t->err)) return false;
while (!t->read_error) {
StartToken(t);
bool report_token = TryConsumeWhitespace(t) || TryConsumeNewline(t);
EndToken(t);
if (report_token) return true;
switch (TryConsumeCommentStart(t)) {
case kUpb_CommentType_Line:
ConsumeLineComment(t, NULL);
continue;
case kUpb_CommentType_Block:
ConsumeBlockComment(t, NULL);
continue;
case kUpb_CommentType_SlashNot:
return true;
case kUpb_CommentType_None:
break;
}
// Check for EOF before continuing.
if (t->read_error) break;
if (LookingAt(t, upb_Tokenizer_IsUnprintable) || t->current_char == '\0') {
ReportError(t, "Invalid control characters encountered in text.");
}
// Reading some sort of token.
StartToken(t);
if (TryConsumeOne(t, upb_Tokenizer_IsLetter)) {
ConsumeZeroOrMore(t, upb_Tokenizer_IsAlphanumeric);
t->token_type = kUpb_TokenType_Identifier;
} else if (TryConsume(t, '0')) {
t->token_type = ConsumeNumber(t, true, false);
} else if (TryConsume(t, '.')) {
// This could be the beginning of a floating-point number, or it could
// just be a '.' symbol.
if (TryConsumeOne(t, upb_Tokenizer_IsDigit)) {
// It's a floating-point number.
if (t->previous_type == kUpb_TokenType_Identifier &&
t->token_line == t->previous_line &&
t->token_column == t->previous_end_column) {
// We don't accept syntax like "blah.123".
t->column -= 2;
ReportError(t, "Need space between identifier and decimal point.");
}
t->token_type = ConsumeNumber(t, false, true);
} else {
t->token_type = kUpb_TokenType_Symbol;
}
} else if (TryConsumeOne(t, upb_Tokenizer_IsDigit)) {
t->token_type = ConsumeNumber(t, false, false);
} else if (TryConsume(t, '\"')) {
ConsumeString(t, '\"');
t->token_type = kUpb_TokenType_String;
} else if (TryConsume(t, '\'')) {
ConsumeString(t, '\'');
t->token_type = kUpb_TokenType_String;
} else {
// Check if the high order bit is set.
if (t->current_char & 0x80) {
ReportErrorFormat(t, "Interpreting non ascii codepoint %d.",
(uint8_t)t->current_char);
}
NextChar(t);
t->token_type = kUpb_TokenType_Symbol;
}
EndToken(t);
return true;
}
// EOF
t->token_type = kUpb_TokenType_End;
upb_String_Clear(&t->token_text);
t->token_line = t->line;
t->token_column = t->column;
t->token_end_column = t->column;
upb_Status_Clear(status);
return false;
}
// -------------------------------------------------------------------
// Token-parsing helpers. Remember that these don't need to report
// errors since any errors should already have been reported while
// tokenizing. Also, these can assume that whatever text they
// are given is text that the tokenizer actually parsed as a token
// of the given type.
bool upb_Parse_Integer(const char* text, uint64_t max_value, uint64_t* output) {
// We can't just use strtoull() because (a) it accepts negative numbers,
// (b) We want additional range checks, (c) it reports overflows via errno.
const char* ptr = text;
int base = 10;
uint64_t overflow_if_mul_base = (UINT64_MAX / 10) + 1;
if (ptr[0] == '0') {
if (ptr[1] == 'x' || ptr[1] == 'X') {
// This is hex.
base = 16;
overflow_if_mul_base = (UINT64_MAX / 16) + 1;
ptr += 2;
} else {
// This is octal.
base = 8;
overflow_if_mul_base = (UINT64_MAX / 8) + 1;
}
}
uint64_t result = 0;
// For all the leading '0's, and also the first non-zero character, we
// don't need to multiply.
while (*ptr != '\0') {
int digit = DigitValue(*ptr++);
if (digit >= base) {
// The token provided by Tokenizer is invalid. i.e., 099 is an invalid
// token, but Tokenizer still think it's integer.
return false;
}
if (digit != 0) {
result = digit;
break;
}
}
for (; *ptr != '\0'; ptr++) {
int digit = DigitValue(*ptr);
if (digit < 0 || digit >= base) {
// The token provided by Tokenizer is invalid. i.e., 099 is an invalid
// token, but Tokenizer still think it's integer.
return false;
}
if (result >= overflow_if_mul_base) {
// We know the multiply we're about to do will overflow, so exit now.
return false;
}
// We know that result * base won't overflow, but adding digit might...
result = result * base + digit;
// C++ guarantees defined "wrap" semantics when unsigned integer
// operations overflow, making this a fast way to check if adding
// digit made result overflow, and thus, wrap around.
if (result < (uint64_t)base) return false;
}
if (result > max_value) return false;
*output = result;
return true;
}
double upb_Parse_Float(const char* text) {
char* end;
double result = _upb_NoLocaleStrtod(text, &end);
// "1e" is not a valid float, but if the tokenizer reads it, it will
// report an error but still return it as a valid token. We need to
// accept anything the tokenizer could possibly return, error or not.
if (*end == 'e' || *end == 'E') {
++end;
if (*end == '-' || *end == '+') ++end;
}
// If the Tokenizer had allow_f_after_float_ enabled, the float may be
// suffixed with the letter 'f'.
if (*end == 'f' || *end == 'F') {
++end;
}
if ((end - text) != strlen(text) || *text == '-') {
fprintf(stderr,
"upb_Parse_Float() passed text that could not have"
" been tokenized as a float: %s\n",
text);
UPB_ASSERT(0);
}
return result;
}
// Append a Unicode code point to a string as UTF8.
static void AppendUTF8(uint32_t code_point, upb_String* output) {
char temp[24];
int len = upb_Unicode_ToUTF8(code_point, temp);
if (len == 0) {
// ConsumeString permits hex values up to 0x1FFFFF,
// and FetchUnicodePoint doesn't perform a range check.
// Unicode code points end at 0x10FFFF, so this is out-of-range.
len = snprintf(temp, sizeof temp, "\\U%08x", code_point);
}
upb_String_Append(output, temp, len);
}
// Try to read <len> hex digits from ptr, and stuff the numeric result into
// *result. Returns true if that many digits were successfully consumed.
static bool ReadHexDigits(const char* ptr, int len, uint32_t* result) {
*result = 0;
if (len == 0) return false;
for (const char* end = ptr + len; ptr < end; ++ptr) {
if (*ptr == '\0') return false;
*result = (*result << 4) + DigitValue(*ptr);
}
return true;
}
// Convert the escape sequence parameter to a number of expected hex digits.
static int UnicodeLength(char key) {
if (key == 'u') return 4;
if (key == 'U') return 8;
return 0;
}
// Given a pointer to the 'u' or 'U' starting a Unicode escape sequence, attempt
// to parse that sequence. On success, returns a pointer to the first char
// beyond that sequence, and fills in *code_point. On failure, returns ptr
// itself.
static const char* FetchUnicodePoint(const char* ptr, uint32_t* code_point) {
const char* p = ptr;
// Fetch the code point.
const int len = UnicodeLength(*p++);
if (!ReadHexDigits(p, len, code_point)) return ptr;
p += len;
// Check if the code point we read is a "head surrogate." If so, then we
// expect it to be immediately followed by another code point which is a valid
// "trail surrogate," and together they form a UTF-16 pair which decodes into
// a single Unicode point. Trail surrogates may only use \u, not \U.
if (upb_Unicode_IsHigh(*code_point) && *p == '\\' && *(p + 1) == 'u') {
uint32_t trail_surrogate;
if (ReadHexDigits(p + 2, 4, &trail_surrogate) &&
upb_Unicode_IsLow(trail_surrogate)) {
*code_point = upb_Unicode_FromPair(*code_point, trail_surrogate);
p += 6;
}
// If this failed, then we just emit the head surrogate as a code point.
// It's bogus, but so is the string.
}
return p;
}
// The text string must begin and end with single or double quote characters.
upb_StringView upb_Parse_String(const char* text, upb_Arena* arena) {
const size_t size = strlen(text);
upb_String output;
upb_String_Init(&output, arena);
// Reminder: text[0] is always a quote character.
// (If text is empty, it's invalid, so we'll just return).
if (size == 0) {
fprintf(stderr,
"Tokenizer::ParseStringAppend() passed text that could not"
" have been tokenized as a string: %s",
text);
UPB_ASSERT(0);
return upb_StringView_FromDataAndSize(NULL, 0);
}
// Reserve room for new string.
const size_t new_len = size + upb_String_Size(&output);
upb_String_Reserve(&output, new_len);
// Loop through the string copying characters to "output" and
// interpreting escape sequences. Note that any invalid escape
// sequences or other errors were already reported while tokenizing.
// In this case we do not need to produce valid results.
for (const char* ptr = text + 1; *ptr != '\0'; ptr++) {
if (*ptr == '\\' && ptr[1] != '\0') {
// An escape sequence.
++ptr;
if (upb_Tokenizer_IsOctalDigit(*ptr)) {
// An octal escape. May one, two, or three digits.
int code = DigitValue(*ptr);
if (upb_Tokenizer_IsOctalDigit(ptr[1])) {
++ptr;
code = code * 8 + DigitValue(*ptr);
}
if (upb_Tokenizer_IsOctalDigit(ptr[1])) {
++ptr;
code = code * 8 + DigitValue(*ptr);
}
upb_String_PushBack(&output, (char)code);
} else if (*ptr == 'x') {
// A hex escape. May zero, one, or two digits. (The zero case
// will have been caught as an error earlier.)
int code = 0;
if (upb_Tokenizer_IsHexDigit(ptr[1])) {
++ptr;
code = DigitValue(*ptr);
}
if (upb_Tokenizer_IsHexDigit(ptr[1])) {
++ptr;
code = code * 16 + DigitValue(*ptr);
}
upb_String_PushBack(&output, (char)code);
} else if (*ptr == 'u' || *ptr == 'U') {
uint32_t unicode;
const char* end = FetchUnicodePoint(ptr, &unicode);
if (end == ptr) {
// Failure: Just dump out what we saw, don't try to parse it.
upb_String_PushBack(&output, *ptr);
} else {
AppendUTF8(unicode, &output);
ptr = end - 1; // Because we're about to ++ptr.
}
} else {
// Some other escape code.
upb_String_PushBack(&output, TranslateEscape(*ptr));
}
} else if (*ptr == text[0] && ptr[1] == '\0') {
// Ignore final quote matching the starting quote.
} else {
upb_String_PushBack(&output, *ptr);
}
}
return upb_StringView_FromDataAndSize(upb_String_Data(&output),
upb_String_Size(&output));
}
static bool AllInClass(bool (*f)(char), const char* text, int size) {
for (int i = 0; i < size; i++) {
if (!f(text[i])) return false;
}
return true;
}
bool upb_Tokenizer_IsIdentifier(const char* data, int size) {
// Mirrors IDENTIFIER definition in Tokenizer::Next() above.
if (size == 0) return false;
if (!upb_Tokenizer_IsLetter(data[0])) return false;
if (!AllInClass(upb_Tokenizer_IsAlphanumeric, data + 1, size - 1))
return false;
return true;
}
upb_Tokenizer* upb_Tokenizer_New(const void* data, size_t size,
upb_ZeroCopyInputStream* input, int options,
upb_Arena* arena) {
upb_Tokenizer* t = upb_Arena_Malloc(arena, sizeof(upb_Tokenizer));
if (!t) return NULL;
t->input = input;
t->arena = arena;
t->buffer = data;
t->buffer_size = size;
t->buffer_pos = 0;
t->read_error = false;
t->line = 0;
t->column = 0;
t->record_target = NULL;
t->record_start = -1;
// ReportNewlines implies ReportWhitespace.
if (options & kUpb_TokenizerOption_ReportNewlines) {
options |= kUpb_TokenizerOption_ReportWhitespace;
}
t->options = options;
upb_String_Init(&t->token_text, arena);
t->token_type = kUpb_TokenType_Start;
t->token_line = 0;
t->token_column = 0;
t->token_end_column = 0;
t->previous_type = kUpb_TokenType_Start;
t->previous_line = 0;
t->previous_column = 0;
t->previous_end_column = 0;
if (size) {
t->current_char = t->buffer[0];
} else {
Refresh(t);
}
return t;
}
void upb_Tokenizer_Fini(upb_Tokenizer* t) {
// If we had any buffer left unread, return it to the underlying stream
// so that someone else can read it.
if (t->buffer_size > t->buffer_pos) {
upb_ZeroCopyInputStream_BackUp(t->input, t->buffer_size - t->buffer_pos);
}
}