/* * Copyright 2023 WebAssembly Community Group participants * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include #include #include #include #include #include #include #include #include "support/name.h" #include "support/result.h" #include "support/string.h" #ifndef parser_lexer_h #define parser_lexer_h namespace wasm::WATParser { struct TextPos { size_t line; size_t col; bool operator==(const TextPos& other) const; bool operator!=(const TextPos& other) const { return !(*this == other); } friend std::ostream& operator<<(std::ostream& os, const TextPos& pos); }; // =========== // Annotations // =========== struct Annotation { Name kind; std::string_view contents; }; extern Name srcAnnotationKind; // ===== // Lexer // ===== struct Lexer { private: size_t pos = 0; std::vector annotations; public: std::string_view buffer; Lexer(std::string_view buffer) : buffer(buffer) { setPos(0); } size_t getPos() const { return pos; } void setPos(size_t i) { pos = i; advance(); } bool takeLParen(); bool peekLParen() { return Lexer(*this).takeLParen(); } bool takeRParen(); bool peekRParen() { return Lexer(*this).takeRParen(); } bool takeUntilParen() { while (true) { if (empty()) { return false; } if (peekLParen() || peekRParen()) { return true; } // Do not count the parentheses in strings. if (takeString()) { continue; } ++pos; advance(); } } std::optional takeID(); std::optional takeKeyword(); bool takeKeyword(std::string_view expected); std::optional peekKeyword() { return Lexer(*this).takeKeyword(); } std::optional takeOffset(); std::optional takeAlign(); std::optional takeU64() { return takeU(); } std::optional takeI64() { return takeI(); } std::optional takeU32() { return takeU(); } std::optional takeI32() { return takeI(); } std::optional takeI16() { return takeI(); } std::optional takeU8() { return takeU(); } std::optional takeI8() { return takeI(); } std::optional takeF64(); std::optional takeF32(); std::optional takeString(); std::optional takeName() { auto str = takeString(); if (!str || !String::isUTF8(*str)) { return std::nullopt; } return Name(*str); } bool takeSExprStart(std::string_view expected) { auto original = *this; if (takeLParen() && takeKeyword(expected)) { return true; } *this = original; return false; } bool peekSExprStart(std::string_view expected) { auto original = *this; if (!takeLParen()) { return false; } bool ret = takeKeyword(expected); *this = original; return ret; } std::string_view next() const { return buffer.substr(pos); } void advance() { annotations.clear(); skipSpace(); } bool empty() const { return pos == buffer.size(); } TextPos position(const char* c) const; TextPos position(size_t i) const { return position(buffer.data() + i); } TextPos position(std::string_view span) const { return position(span.data()); } TextPos position() const { return position(getPos()); } [[nodiscard]] Err err(size_t pos, std::string reason) { std::stringstream msg; msg << position(pos) << ": error: " << reason; return Err{msg.str()}; } [[nodiscard]] Err err(std::string reason) { return err(getPos(), reason); } const std::vector getAnnotations() { return annotations; } std::vector takeAnnotations() { return std::move(annotations); } void setAnnotations(std::vector&& annotations) { this->annotations = std::move(annotations); } private: template std::optional takeU(); template std::optional takeS(); template std::optional takeI(); void skipSpace(); }; } // namespace wasm::WATParser #endif // parser_lexer_h