| // Copyright 2020 The Tint Authors. |
| // |
| // 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 "src/tint/reader/wgsl/parser_impl.h" |
| |
| #include <limits> |
| |
| #include "src/tint/ast/array.h" |
| #include "src/tint/ast/assignment_statement.h" |
| #include "src/tint/ast/bitcast_expression.h" |
| #include "src/tint/ast/break_statement.h" |
| #include "src/tint/ast/call_statement.h" |
| #include "src/tint/ast/continue_statement.h" |
| #include "src/tint/ast/discard_statement.h" |
| #include "src/tint/ast/external_texture.h" |
| #include "src/tint/ast/fallthrough_statement.h" |
| #include "src/tint/ast/id_attribute.h" |
| #include "src/tint/ast/if_statement.h" |
| #include "src/tint/ast/increment_decrement_statement.h" |
| #include "src/tint/ast/invariant_attribute.h" |
| #include "src/tint/ast/loop_statement.h" |
| #include "src/tint/ast/return_statement.h" |
| #include "src/tint/ast/stage_attribute.h" |
| #include "src/tint/ast/switch_statement.h" |
| #include "src/tint/ast/type_name.h" |
| #include "src/tint/ast/unary_op_expression.h" |
| #include "src/tint/ast/variable_decl_statement.h" |
| #include "src/tint/ast/vector.h" |
| #include "src/tint/ast/workgroup_attribute.h" |
| #include "src/tint/reader/wgsl/lexer.h" |
| #include "src/tint/sem/depth_texture.h" |
| #include "src/tint/sem/external_texture.h" |
| #include "src/tint/sem/multisampled_texture.h" |
| #include "src/tint/sem/sampled_texture.h" |
| |
| namespace tint::reader::wgsl { |
| namespace { |
| |
| template <typename T> |
| using Expect = ParserImpl::Expect<T>; |
| |
| template <typename T> |
| using Maybe = ParserImpl::Maybe<T>; |
| |
| /// Controls the maximum number of times we'll call into the sync() and |
| /// unary_expression() functions from themselves. This is to guard against stack |
| /// overflow when there is an excessive number of blocks. |
| constexpr uint32_t kMaxParseDepth = 128; |
| |
| /// The maximum number of tokens to look ahead to try and sync the |
| /// parser on error. |
| constexpr size_t const kMaxResynchronizeLookahead = 32; |
| |
| const char kVertexStage[] = "vertex"; |
| const char kFragmentStage[] = "fragment"; |
| const char kComputeStage[] = "compute"; |
| |
| const char kReadAccess[] = "read"; |
| const char kWriteAccess[] = "write"; |
| const char kReadWriteAccess[] = "read_write"; |
| |
| const char kBindingAttribute[] = "binding"; |
| const char kBuiltinAttribute[] = "builtin"; |
| const char kGroupAttribute[] = "group"; |
| const char kIdAttribute[] = "id"; |
| const char kInterpolateAttribute[] = "interpolate"; |
| const char kInvariantAttribute[] = "invariant"; |
| const char kLocationAttribute[] = "location"; |
| const char kSizeAttribute[] = "size"; |
| const char kAlignAttribute[] = "align"; |
| const char kStageAttribute[] = "stage"; |
| const char kWorkgroupSizeAttribute[] = "workgroup_size"; |
| |
| // https://gpuweb.github.io/gpuweb/wgsl.html#reserved-keywords |
| bool is_reserved(const Token& t) { |
| return t == "asm" || t == "bf16" || t == "do" || t == "enum" || t == "f64" || t == "handle" || |
| t == "i8" || t == "i16" || t == "i64" || t == "mat" || t == "premerge" || |
| t == "regardless" || t == "typedef" || t == "u8" || t == "u16" || t == "u64" || |
| t == "unless" || t == "using" || t == "vec" || t == "void" || t == "while"; |
| } |
| |
| /// Enter-exit counters for block token types. |
| /// Used by sync_to() to skip over closing block tokens that were opened during |
| /// the forward scan. |
| struct BlockCounters { |
| int brace = 0; // { } |
| int bracket = 0; // [ ] |
| int paren = 0; // ( ) |
| |
| /// @return the current enter-exit depth for the given block token type. If |
| /// `t` is not a block token type, then 0 is always returned. |
| int consume(const Token& t) { |
| if (t.Is(Token::Type::kBraceLeft)) { |
| return brace++; |
| } |
| if (t.Is(Token::Type::kBraceRight)) { |
| return brace--; |
| } |
| if (t.Is(Token::Type::kBracketLeft)) { |
| return bracket++; |
| } |
| if (t.Is(Token::Type::kBracketRight)) { |
| return bracket--; |
| } |
| if (t.Is(Token::Type::kParenLeft)) { |
| return paren++; |
| } |
| if (t.Is(Token::Type::kParenRight)) { |
| return paren--; |
| } |
| return 0; |
| } |
| }; |
| } // namespace |
| |
| /// RAII helper that combines a Source on construction with the last token's |
| /// source when implicitly converted to `Source`. |
| class ParserImpl::MultiTokenSource { |
| public: |
| /// Constructor that starts with Source at the current peek position |
| /// @param parser the parser |
| explicit MultiTokenSource(ParserImpl* parser) |
| : MultiTokenSource(parser, parser->peek().source().Begin()) {} |
| |
| /// Constructor that starts with the input `start` Source |
| /// @param parser the parser |
| /// @param start the start source of the range |
| MultiTokenSource(ParserImpl* parser, const Source& start) : parser_(parser), start_(start) {} |
| |
| /// Implicit conversion to Source that returns the combined source from start |
| /// to the current last token's source. |
| operator Source() const { |
| Source end = parser_->last_source().End(); |
| if (end < start_) { |
| end = start_; |
| } |
| return Source::Combine(start_, end); |
| } |
| |
| private: |
| ParserImpl* parser_; |
| Source start_; |
| }; |
| |
| ParserImpl::TypedIdentifier::TypedIdentifier() = default; |
| |
| ParserImpl::TypedIdentifier::TypedIdentifier(const TypedIdentifier&) = default; |
| |
| ParserImpl::TypedIdentifier::TypedIdentifier(const ast::Type* type_in, |
| std::string name_in, |
| Source source_in) |
| : type(type_in), name(std::move(name_in)), source(std::move(source_in)) {} |
| |
| ParserImpl::TypedIdentifier::~TypedIdentifier() = default; |
| |
| ParserImpl::FunctionHeader::FunctionHeader() = default; |
| |
| ParserImpl::FunctionHeader::FunctionHeader(const FunctionHeader&) = default; |
| |
| ParserImpl::FunctionHeader::FunctionHeader(Source src, |
| std::string n, |
| utils::VectorRef<const ast::Parameter*> p, |
| const ast::Type* ret_ty, |
| utils::VectorRef<const ast::Attribute*> ret_attrs) |
| : source(src), |
| name(n), |
| params(std::move(p)), |
| return_type(ret_ty), |
| return_type_attributes(std::move(ret_attrs)) {} |
| |
| ParserImpl::FunctionHeader::~FunctionHeader() = default; |
| |
| ParserImpl::FunctionHeader& ParserImpl::FunctionHeader::operator=(const FunctionHeader& rhs) = |
| default; |
| |
| ParserImpl::VarDeclInfo::VarDeclInfo() = default; |
| |
| ParserImpl::VarDeclInfo::VarDeclInfo(const VarDeclInfo&) = default; |
| |
| ParserImpl::VarDeclInfo::VarDeclInfo(Source source_in, |
| std::string name_in, |
| ast::StorageClass storage_class_in, |
| ast::Access access_in, |
| const ast::Type* type_in) |
| : source(std::move(source_in)), |
| name(std::move(name_in)), |
| storage_class(storage_class_in), |
| access(access_in), |
| type(type_in) {} |
| |
| ParserImpl::VarDeclInfo::~VarDeclInfo() = default; |
| |
| ParserImpl::ParserImpl(Source::File const* file) : file_(file) {} |
| |
| ParserImpl::~ParserImpl() = default; |
| |
| ParserImpl::Failure::Errored ParserImpl::add_error(const Source& source, |
| std::string_view err, |
| std::string_view use) { |
| std::stringstream msg; |
| msg << err; |
| if (!use.empty()) { |
| msg << " for " << use; |
| } |
| add_error(source, msg.str()); |
| return Failure::kErrored; |
| } |
| |
| ParserImpl::Failure::Errored ParserImpl::add_error(const Token& t, const std::string& err) { |
| add_error(t.source(), err); |
| return Failure::kErrored; |
| } |
| |
| ParserImpl::Failure::Errored ParserImpl::add_error(const Source& source, const std::string& err) { |
| if (silence_errors_ == 0) { |
| builder_.Diagnostics().add_error(diag::System::Reader, err, source); |
| } |
| return Failure::kErrored; |
| } |
| |
| void ParserImpl::deprecated(const Source& source, const std::string& msg) { |
| builder_.Diagnostics().add_warning(diag::System::Reader, |
| "use of deprecated language feature: " + msg, source); |
| } |
| |
| const Token& ParserImpl::next() { |
| // If the next token is already an error or the end of file, stay there. |
| if (tokens_[next_token_idx_].IsEof() || tokens_[next_token_idx_].IsError()) { |
| return tokens_[next_token_idx_]; |
| } |
| |
| // Skip over any placeholder elements |
| while (true) { |
| if (!tokens_[next_token_idx_].IsPlaceholder()) { |
| break; |
| } |
| next_token_idx_++; |
| } |
| last_source_idx_ = next_token_idx_; |
| |
| if (!tokens_[next_token_idx_].IsEof() && !tokens_[next_token_idx_].IsError()) { |
| next_token_idx_++; |
| } |
| return tokens_[last_source_idx_]; |
| } |
| |
| const Token& ParserImpl::peek(size_t idx) { |
| if (next_token_idx_ + idx >= tokens_.size()) { |
| return tokens_[tokens_.size() - 1]; |
| } |
| |
| // Skip over any placeholder elements |
| while (true) { |
| if (!tokens_[next_token_idx_ + idx].IsPlaceholder()) { |
| break; |
| } |
| idx++; |
| } |
| if (next_token_idx_ + idx >= tokens_.size()) { |
| return tokens_[tokens_.size() - 1]; |
| } |
| |
| return tokens_[next_token_idx_ + idx]; |
| } |
| |
| bool ParserImpl::peek_is(Token::Type tok, size_t idx) { |
| return peek(idx).Is(tok); |
| } |
| |
| void ParserImpl::split_token(Token::Type lhs, Token::Type rhs) { |
| if (next_token_idx_ == 0) { |
| TINT_ICE(Reader, builder_.Diagnostics()) |
| << "attempt to update placeholder at beginning of tokens"; |
| } |
| if (next_token_idx_ >= tokens_.size()) { |
| TINT_ICE(Reader, builder_.Diagnostics()) |
| << "attempt to update placeholder past end of tokens"; |
| } |
| if (!tokens_[next_token_idx_].IsPlaceholder()) { |
| TINT_ICE(Reader, builder_.Diagnostics()) << "attempt to update non-placeholder token"; |
| } |
| tokens_[next_token_idx_ - 1].SetType(lhs); |
| tokens_[next_token_idx_].SetType(rhs); |
| } |
| |
| Source ParserImpl::last_source() const { |
| return tokens_[last_source_idx_].source(); |
| } |
| |
| void ParserImpl::InitializeLex() { |
| Lexer l{file_}; |
| tokens_ = l.Lex(); |
| } |
| |
| bool ParserImpl::Parse() { |
| InitializeLex(); |
| translation_unit(); |
| return !has_error(); |
| } |
| |
| // translation_unit |
| // : enable_directive* global_decl* EOF |
| void ParserImpl::translation_unit() { |
| bool after_global_decl = false; |
| while (continue_parsing()) { |
| auto& p = peek(); |
| if (p.IsEof()) { |
| break; |
| } |
| |
| auto ed = enable_directive(); |
| if (ed.matched) { |
| if (after_global_decl) { |
| add_error(p, "enable directives must come before all global declarations"); |
| } |
| } else if (ed.errored) { |
| // Found a invalid enable directive. |
| continue; |
| } else { |
| auto gd = global_decl(); |
| |
| if (gd.matched) { |
| after_global_decl = true; |
| } |
| |
| if (!gd.matched && !gd.errored) { |
| add_error(p, "unexpected token"); |
| } |
| } |
| |
| if (builder_.Diagnostics().error_count() >= max_errors_) { |
| add_error(Source{{}, p.source().file}, |
| "stopping after " + std::to_string(max_errors_) + " errors"); |
| break; |
| } |
| } |
| } |
| |
| // enable_directive |
| // : enable name SEMICLON |
| Maybe<bool> ParserImpl::enable_directive() { |
| auto decl = sync(Token::Type::kSemicolon, [&]() -> Maybe<bool> { |
| if (!match(Token::Type::kEnable)) { |
| return Failure::kNoMatch; |
| } |
| |
| // Match the extension name. |
| Expect<std::string> name = {""}; |
| auto& t = peek(); |
| if (t.IsIdentifier()) { |
| synchronized_ = true; |
| next(); |
| name = {t.to_str(), t.source()}; |
| } else if (t.Is(Token::Type::kF16)) { |
| // `f16` is a valid extension name and also a keyword |
| synchronized_ = true; |
| next(); |
| name = {"f16", t.source()}; |
| } else if (t.Is(Token::Type::kParenLeft)) { |
| // A common error case is writing `enable(foo);` instead of `enable foo;`. |
| synchronized_ = false; |
| return add_error(t.source(), "enable directives don't take parenthesis"); |
| } else if (handle_error(t)) { |
| // The token might itself be an error. |
| return Failure::kErrored; |
| } else { |
| // Failed to match an extension name. |
| synchronized_ = false; |
| return add_error(t.source(), "invalid extension name"); |
| } |
| |
| if (!expect("enable directive", Token::Type::kSemicolon)) { |
| return Failure::kErrored; |
| } |
| |
| auto extension = ast::ParseExtension(name.value); |
| if (extension == ast::Extension::kInvalid) { |
| return add_error(name.source, "unsupported extension: '" + name.value + "'"); |
| } |
| builder_.AST().AddEnable(create<ast::Enable>(name.source, extension)); |
| |
| return true; |
| }); |
| |
| if (decl.errored) { |
| return Failure::kErrored; |
| } |
| if (decl.matched) { |
| return true; |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // global_decl |
| // : SEMICOLON |
| // | global_variable_decl SEMICLON |
| // | global_constant_decl SEMICOLON |
| // | type_alias SEMICOLON |
| // | struct_decl |
| // | function_decl |
| Maybe<bool> ParserImpl::global_decl() { |
| if (match(Token::Type::kSemicolon) || match(Token::Type::kEOF)) { |
| return true; |
| } |
| |
| bool errored = false; |
| |
| auto attrs = attribute_list(); |
| if (attrs.errored) { |
| errored = true; |
| } |
| if (!continue_parsing()) { |
| return Failure::kErrored; |
| } |
| |
| auto decl = sync(Token::Type::kSemicolon, [&]() -> Maybe<bool> { |
| auto gv = global_variable_decl(attrs.value); |
| if (gv.errored) { |
| return Failure::kErrored; |
| } |
| if (gv.matched) { |
| if (!expect("variable declaration", Token::Type::kSemicolon)) { |
| return Failure::kErrored; |
| } |
| |
| builder_.AST().AddGlobalVariable(gv.value); |
| return true; |
| } |
| |
| auto gc = global_constant_decl(attrs.value); |
| if (gc.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (gc.matched) { |
| // Avoid the cost of the string allocation for the common no-error case |
| if (!peek().Is(Token::Type::kSemicolon)) { |
| std::string kind = gc->Kind(); |
| if (!expect("'" + kind + "' declaration", Token::Type::kSemicolon)) { |
| return Failure::kErrored; |
| } |
| } |
| |
| builder_.AST().AddGlobalVariable(gc.value); |
| return true; |
| } |
| |
| auto ta = type_alias(); |
| if (ta.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (ta.matched) { |
| if (!expect("type alias", Token::Type::kSemicolon)) { |
| return Failure::kErrored; |
| } |
| |
| builder_.AST().AddTypeDecl(ta.value); |
| return true; |
| } |
| |
| auto str = struct_decl(); |
| if (str.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (str.matched) { |
| builder_.AST().AddTypeDecl(str.value); |
| return true; |
| } |
| |
| return Failure::kNoMatch; |
| }); |
| |
| if (decl.errored) { |
| errored = true; |
| } |
| if (decl.matched) { |
| return expect_attributes_consumed(attrs.value); |
| } |
| |
| auto func = function_decl(attrs.value); |
| if (func.errored) { |
| errored = true; |
| } |
| if (func.matched) { |
| builder_.AST().AddFunction(func.value); |
| return true; |
| } |
| |
| if (errored) { |
| return Failure::kErrored; |
| } |
| |
| // Invalid syntax found - try and determine the best error message |
| |
| // We have attributes parsed, but nothing to consume them? |
| if (attrs.value.Length() > 0) { |
| return add_error(next(), "expected declaration after attributes"); |
| } |
| |
| // We have a statement outside of a function? |
| auto& t = peek(); |
| auto stat = without_error([&] { return statement(); }); |
| if (stat.matched) { |
| // Attempt to jump to the next '}' - the function might have just been |
| // missing an opening line. |
| sync_to(Token::Type::kBraceRight, true); |
| return add_error(t, "statement found outside of function body"); |
| } |
| if (!stat.errored) { |
| // No match, no error - the parser might not have progressed. |
| // Ensure we always make _some_ forward progress. |
| next(); |
| } |
| |
| // The token might itself be an error. |
| if (handle_error(t)) { |
| return Failure::kErrored; |
| } |
| |
| // Exhausted all attempts to make sense of where we're at. |
| // Return a no-match |
| |
| return Failure::kNoMatch; |
| } |
| |
| // global_variable_decl |
| // : variable_attribute_list* variable_decl |
| // | variable_attribute_list* variable_decl EQUAL const_expr |
| Maybe<const ast::Variable*> ParserImpl::global_variable_decl(AttributeList& attrs) { |
| auto decl = variable_decl(); |
| if (decl.errored) { |
| return Failure::kErrored; |
| } |
| if (!decl.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| const ast::Expression* initializer = nullptr; |
| if (match(Token::Type::kEqual)) { |
| auto expr = logical_or_expression(); |
| if (expr.errored) { |
| return Failure::kErrored; |
| } |
| if (!expr.matched) { |
| return add_error(peek(), "missing initializer for 'var' declaration"); |
| } |
| initializer = expr.value; |
| } |
| |
| TINT_DEFER(attrs.Clear()); |
| |
| return create<ast::Var>(decl->source, // source |
| builder_.Symbols().Register(decl->name), // symbol |
| decl->type, // type |
| decl->storage_class, // storage class |
| decl->access, // access control |
| initializer, // initializer |
| std::move(attrs)); // attributes |
| } |
| |
| // global_constant_decl : |
| // | LET (ident | variable_ident_decl) global_const_initializer |
| // | attribute* override (ident | variable_ident_decl) (equal expression)? |
| // global_const_initializer |
| // : EQUAL const_expr |
| Maybe<const ast::Variable*> ParserImpl::global_constant_decl(AttributeList& attrs) { |
| bool is_const = false; |
| bool is_overridable = false; |
| const char* use = nullptr; |
| Source source; |
| if (match(Token::Type::kConst)) { |
| use = "'const' declaration"; |
| } else if (match(Token::Type::kLet, &source)) { |
| use = "'let' declaration"; |
| deprecated(source, "module-scope 'let' has been replaced with 'const'"); |
| } else if (match(Token::Type::kOverride)) { |
| use = "'override' declaration"; |
| is_overridable = true; |
| } else { |
| return Failure::kNoMatch; |
| } |
| |
| auto decl = expect_variable_ident_decl(use, /* allow_inferred = */ true); |
| if (decl.errored) { |
| return Failure::kErrored; |
| } |
| |
| bool has_initializer = false; |
| if (is_overridable) { |
| has_initializer = match(Token::Type::kEqual); |
| } else { |
| if (!expect(use, Token::Type::kEqual)) { |
| return Failure::kErrored; |
| } |
| has_initializer = true; |
| } |
| |
| const ast::Expression* initializer = nullptr; |
| if (has_initializer) { |
| auto expr = logical_or_expression(); |
| if (expr.errored) { |
| return Failure::kErrored; |
| } |
| if (!expr.matched) { |
| return add_error(peek(), "missing initializer for " + std::string(use)); |
| } |
| initializer = std::move(expr.value); |
| } |
| |
| TINT_DEFER(attrs.Clear()); |
| |
| if (is_const) { |
| return create<ast::Const>(decl->source, // source |
| builder_.Symbols().Register(decl->name), // symbol |
| decl->type, // type |
| initializer, // initializer |
| std::move(attrs)); // attributes |
| } |
| if (is_overridable) { |
| return create<ast::Override>(decl->source, // source |
| builder_.Symbols().Register(decl->name), // symbol |
| decl->type, // type |
| initializer, // initializer |
| std::move(attrs)); // attributes |
| } |
| return create<ast::Const>(decl->source, // source |
| builder_.Symbols().Register(decl->name), // symbol |
| decl->type, // type |
| initializer, // initializer |
| std::move(attrs)); // attributes |
| } |
| |
| // variable_decl |
| // : VAR variable_qualifier? variable_ident_decl |
| Maybe<ParserImpl::VarDeclInfo> ParserImpl::variable_decl() { |
| Source source; |
| if (!match(Token::Type::kVar, &source)) { |
| return Failure::kNoMatch; |
| } |
| |
| VariableQualifier vq; |
| auto explicit_vq = variable_qualifier(); |
| if (explicit_vq.errored) { |
| return Failure::kErrored; |
| } |
| if (explicit_vq.matched) { |
| vq = explicit_vq.value; |
| } |
| |
| auto decl = expect_variable_ident_decl("variable declaration", |
| /*allow_inferred = */ true); |
| if (decl.errored) { |
| return Failure::kErrored; |
| } |
| |
| return VarDeclInfo{decl->source, decl->name, vq.storage_class, vq.access, decl->type}; |
| } |
| |
| // texture_samplers |
| // : sampler |
| // | depth_texture |
| // | sampled_texture LESS_THAN type_decl GREATER_THAN |
| // | multisampled_texture LESS_THAN type_decl GREATER_THAN |
| // | storage_texture LESS_THAN texel_format |
| // COMMA access GREATER_THAN |
| Maybe<const ast::Type*> ParserImpl::texture_samplers() { |
| auto type = sampler(); |
| if (type.matched) { |
| return type; |
| } |
| |
| type = depth_texture(); |
| if (type.matched) { |
| return type; |
| } |
| |
| type = external_texture(); |
| if (type.matched) { |
| return type.value; |
| } |
| |
| auto source_range = make_source_range(); |
| |
| auto dim = sampled_texture(); |
| if (dim.matched) { |
| const char* use = "sampled texture type"; |
| |
| auto subtype = expect_lt_gt_block(use, [&] { return expect_type(use); }); |
| if (subtype.errored) { |
| return Failure::kErrored; |
| } |
| |
| return builder_.ty.sampled_texture(source_range, dim.value, subtype.value); |
| } |
| |
| auto ms_dim = multisampled_texture(); |
| if (ms_dim.matched) { |
| const char* use = "multisampled texture type"; |
| |
| auto subtype = expect_lt_gt_block(use, [&] { return expect_type(use); }); |
| if (subtype.errored) { |
| return Failure::kErrored; |
| } |
| |
| return builder_.ty.multisampled_texture(source_range, ms_dim.value, subtype.value); |
| } |
| |
| auto storage = storage_texture(); |
| if (storage.matched) { |
| const char* use = "storage texture type"; |
| using StorageTextureInfo = std::pair<tint::ast::TexelFormat, tint::ast::Access>; |
| auto params = expect_lt_gt_block(use, [&]() -> Expect<StorageTextureInfo> { |
| auto format = expect_texel_format(use); |
| if (format.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (!expect("access control", Token::Type::kComma)) { |
| return Failure::kErrored; |
| } |
| |
| auto access = expect_access("access control"); |
| if (access.errored) { |
| return Failure::kErrored; |
| } |
| |
| return std::make_pair(format.value, access.value); |
| }); |
| |
| if (params.errored) { |
| return Failure::kErrored; |
| } |
| |
| return builder_.ty.storage_texture(source_range, storage.value, params->first, |
| params->second); |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // sampler |
| // : SAMPLER |
| // | SAMPLER_COMPARISON |
| Maybe<const ast::Type*> ParserImpl::sampler() { |
| Source source; |
| if (match(Token::Type::kSampler, &source)) { |
| return builder_.ty.sampler(source, ast::SamplerKind::kSampler); |
| } |
| |
| if (match(Token::Type::kComparisonSampler, &source)) { |
| return builder_.ty.sampler(source, ast::SamplerKind::kComparisonSampler); |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // sampled_texture |
| // : TEXTURE_SAMPLED_1D |
| // | TEXTURE_SAMPLED_2D |
| // | TEXTURE_SAMPLED_2D_ARRAY |
| // | TEXTURE_SAMPLED_3D |
| // | TEXTURE_SAMPLED_CUBE |
| // | TEXTURE_SAMPLED_CUBE_ARRAY |
| Maybe<const ast::TextureDimension> ParserImpl::sampled_texture() { |
| if (match(Token::Type::kTextureSampled1d)) { |
| return ast::TextureDimension::k1d; |
| } |
| |
| if (match(Token::Type::kTextureSampled2d)) { |
| return ast::TextureDimension::k2d; |
| } |
| |
| if (match(Token::Type::kTextureSampled2dArray)) { |
| return ast::TextureDimension::k2dArray; |
| } |
| |
| if (match(Token::Type::kTextureSampled3d)) { |
| return ast::TextureDimension::k3d; |
| } |
| |
| if (match(Token::Type::kTextureSampledCube)) { |
| return ast::TextureDimension::kCube; |
| } |
| |
| if (match(Token::Type::kTextureSampledCubeArray)) { |
| return ast::TextureDimension::kCubeArray; |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // external_texture |
| // : TEXTURE_EXTERNAL |
| Maybe<const ast::Type*> ParserImpl::external_texture() { |
| Source source; |
| if (match(Token::Type::kTextureExternal, &source)) { |
| return builder_.ty.external_texture(source); |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // multisampled_texture |
| // : TEXTURE_MULTISAMPLED_2D |
| Maybe<const ast::TextureDimension> ParserImpl::multisampled_texture() { |
| if (match(Token::Type::kTextureMultisampled2d)) { |
| return ast::TextureDimension::k2d; |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // storage_texture |
| // : TEXTURE_STORAGE_1D |
| // | TEXTURE_STORAGE_2D |
| // | TEXTURE_STORAGE_2D_ARRAY |
| // | TEXTURE_STORAGE_3D |
| Maybe<const ast::TextureDimension> ParserImpl::storage_texture() { |
| if (match(Token::Type::kTextureStorage1d)) { |
| return ast::TextureDimension::k1d; |
| } |
| if (match(Token::Type::kTextureStorage2d)) { |
| return ast::TextureDimension::k2d; |
| } |
| if (match(Token::Type::kTextureStorage2dArray)) { |
| return ast::TextureDimension::k2dArray; |
| } |
| if (match(Token::Type::kTextureStorage3d)) { |
| return ast::TextureDimension::k3d; |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // depth_texture |
| // : TEXTURE_DEPTH_2D |
| // | TEXTURE_DEPTH_2D_ARRAY |
| // | TEXTURE_DEPTH_CUBE |
| // | TEXTURE_DEPTH_CUBE_ARRAY |
| // | TEXTURE_DEPTH_MULTISAMPLED_2D |
| Maybe<const ast::Type*> ParserImpl::depth_texture() { |
| Source source; |
| if (match(Token::Type::kTextureDepth2d, &source)) { |
| return builder_.ty.depth_texture(source, ast::TextureDimension::k2d); |
| } |
| if (match(Token::Type::kTextureDepth2dArray, &source)) { |
| return builder_.ty.depth_texture(source, ast::TextureDimension::k2dArray); |
| } |
| if (match(Token::Type::kTextureDepthCube, &source)) { |
| return builder_.ty.depth_texture(source, ast::TextureDimension::kCube); |
| } |
| if (match(Token::Type::kTextureDepthCubeArray, &source)) { |
| return builder_.ty.depth_texture(source, ast::TextureDimension::kCubeArray); |
| } |
| if (match(Token::Type::kTextureDepthMultisampled2d, &source)) { |
| return builder_.ty.depth_multisampled_texture(source, ast::TextureDimension::k2d); |
| } |
| return Failure::kNoMatch; |
| } |
| |
| // texel_format |
| // : 'rgba8unorm' |
| // | 'rgba8snorm' |
| // | 'rgba8uint' |
| // | 'rgba8sint' |
| // | 'rgba16uint' |
| // | 'rgba16sint' |
| // | 'rgba16float' |
| // | 'r32uint' |
| // | 'r32sint' |
| // | 'r32float' |
| // | 'rg32uint' |
| // | 'rg32sint' |
| // | 'rg32float' |
| // | 'rgba32uint' |
| // | 'rgba32sint' |
| // | 'rgba32float' |
| Expect<ast::TexelFormat> ParserImpl::expect_texel_format(std::string_view use) { |
| auto& t = next(); |
| auto fmt = ast::ParseTexelFormat(t.to_str()); |
| if (fmt == ast::TexelFormat::kInvalid) { |
| return add_error(t.source(), "invalid format", use); |
| } |
| return fmt; |
| } |
| |
| // variable_ident_decl |
| // : IDENT COLON type_decl |
| Expect<ParserImpl::TypedIdentifier> ParserImpl::expect_variable_ident_decl(std::string_view use, |
| bool allow_inferred) { |
| auto ident = expect_ident(use); |
| if (ident.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (allow_inferred && !peek_is(Token::Type::kColon)) { |
| return TypedIdentifier{nullptr, ident.value, ident.source}; |
| } |
| |
| if (!expect(use, Token::Type::kColon)) { |
| return Failure::kErrored; |
| } |
| |
| auto& t = peek(); |
| auto type = type_decl(); |
| if (type.errored) { |
| return Failure::kErrored; |
| } |
| if (!type.matched) { |
| return add_error(t.source(), "invalid type", use); |
| } |
| |
| return TypedIdentifier{type.value, ident.value, ident.source}; |
| } |
| |
| Expect<ast::Access> ParserImpl::expect_access(std::string_view use) { |
| auto ident = expect_ident(use); |
| if (ident.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (ident.value == kReadAccess) { |
| return {ast::Access::kRead, ident.source}; |
| } |
| if (ident.value == kWriteAccess) { |
| return {ast::Access::kWrite, ident.source}; |
| } |
| if (ident.value == kReadWriteAccess) { |
| return {ast::Access::kReadWrite, ident.source}; |
| } |
| |
| return add_error(ident.source, "invalid value for access control"); |
| } |
| |
| // variable_qualifier |
| // : LESS_THAN storage_class (COMMA access_mode)? GREATER_THAN |
| Maybe<ParserImpl::VariableQualifier> ParserImpl::variable_qualifier() { |
| if (!peek_is(Token::Type::kLessThan)) { |
| return Failure::kNoMatch; |
| } |
| |
| auto* use = "variable declaration"; |
| auto vq = expect_lt_gt_block(use, [&]() -> Expect<VariableQualifier> { |
| auto source = make_source_range(); |
| auto sc = expect_storage_class(use); |
| if (sc.errored) { |
| return Failure::kErrored; |
| } |
| if (match(Token::Type::kComma)) { |
| auto ac = expect_access(use); |
| if (ac.errored) { |
| return Failure::kErrored; |
| } |
| return VariableQualifier{sc.value, ac.value}; |
| } |
| return Expect<VariableQualifier>{VariableQualifier{sc.value, ast::Access::kUndefined}, |
| source}; |
| }); |
| |
| if (vq.errored) { |
| return Failure::kErrored; |
| } |
| |
| return vq; |
| } |
| |
| // type_alias |
| // : TYPE IDENT EQUAL type_decl |
| Maybe<const ast::Alias*> ParserImpl::type_alias() { |
| if (!peek_is(Token::Type::kType)) { |
| return Failure::kNoMatch; |
| } |
| |
| auto& t = next(); |
| const char* use = "type alias"; |
| |
| auto name = expect_ident(use); |
| if (name.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (!expect(use, Token::Type::kEqual)) { |
| return Failure::kErrored; |
| } |
| |
| auto type = type_decl(); |
| if (type.errored) { |
| return Failure::kErrored; |
| } |
| if (!type.matched) { |
| return add_error(peek(), "invalid type alias"); |
| } |
| |
| return builder_.ty.alias(make_source_range_from(t.source()), name.value, type.value); |
| } |
| |
| // type_decl |
| // : IDENTIFIER |
| // | BOOL |
| // | FLOAT32 |
| // | INT32 |
| // | UINT32 |
| // | VEC2 LESS_THAN type_decl GREATER_THAN |
| // | VEC3 LESS_THAN type_decl GREATER_THAN |
| // | VEC4 LESS_THAN type_decl GREATER_THAN |
| // | PTR LESS_THAN storage_class, type_decl (COMMA access_mode)? GREATER_THAN |
| // | array_attribute_list* ARRAY LESS_THAN type_decl COMMA |
| // INT_LITERAL GREATER_THAN |
| // | array_attribute_list* ARRAY LESS_THAN type_decl |
| // GREATER_THAN |
| // | MAT2x2 LESS_THAN type_decl GREATER_THAN |
| // | MAT2x3 LESS_THAN type_decl GREATER_THAN |
| // | MAT2x4 LESS_THAN type_decl GREATER_THAN |
| // | MAT3x2 LESS_THAN type_decl GREATER_THAN |
| // | MAT3x3 LESS_THAN type_decl GREATER_THAN |
| // | MAT3x4 LESS_THAN type_decl GREATER_THAN |
| // | MAT4x2 LESS_THAN type_decl GREATER_THAN |
| // | MAT4x3 LESS_THAN type_decl GREATER_THAN |
| // | MAT4x4 LESS_THAN type_decl GREATER_THAN |
| // | texture_samplers |
| Maybe<const ast::Type*> ParserImpl::type_decl() { |
| auto& t = peek(); |
| Source source; |
| if (match(Token::Type::kIdentifier, &source)) { |
| return builder_.create<ast::TypeName>(source, builder_.Symbols().Register(t.to_str())); |
| } |
| |
| if (match(Token::Type::kBool, &source)) { |
| return builder_.ty.bool_(source); |
| } |
| |
| if (match(Token::Type::kF16, &source)) { |
| return builder_.ty.f16(source); |
| } |
| |
| if (match(Token::Type::kF32, &source)) { |
| return builder_.ty.f32(source); |
| } |
| |
| if (match(Token::Type::kI32, &source)) { |
| return builder_.ty.i32(source); |
| } |
| |
| if (match(Token::Type::kU32, &source)) { |
| return builder_.ty.u32(source); |
| } |
| |
| if (t.IsVector()) { |
| next(); // Consume the peek |
| return expect_type_decl_vector(t); |
| } |
| |
| if (match(Token::Type::kPtr)) { |
| return expect_type_decl_pointer(t); |
| } |
| |
| if (match(Token::Type::kAtomic)) { |
| return expect_type_decl_atomic(t); |
| } |
| |
| if (match(Token::Type::kArray, &source)) { |
| return expect_type_decl_array(t); |
| } |
| |
| if (t.IsMatrix()) { |
| next(); // Consume the peek |
| return expect_type_decl_matrix(t); |
| } |
| |
| auto texture_or_sampler = texture_samplers(); |
| if (texture_or_sampler.errored) { |
| return Failure::kErrored; |
| } |
| if (texture_or_sampler.matched) { |
| return texture_or_sampler; |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| Expect<const ast::Type*> ParserImpl::expect_type(std::string_view use) { |
| auto type = type_decl(); |
| if (type.errored) { |
| return Failure::kErrored; |
| } |
| if (!type.matched) { |
| return add_error(peek().source(), "invalid type", use); |
| } |
| return type.value; |
| } |
| |
| Expect<const ast::Type*> ParserImpl::expect_type_decl_pointer(const Token& t) { |
| const char* use = "ptr declaration"; |
| |
| auto storage_class = ast::StorageClass::kNone; |
| auto access = ast::Access::kUndefined; |
| |
| auto subtype = expect_lt_gt_block(use, [&]() -> Expect<const ast::Type*> { |
| auto sc = expect_storage_class(use); |
| if (sc.errored) { |
| return Failure::kErrored; |
| } |
| storage_class = sc.value; |
| |
| if (!expect(use, Token::Type::kComma)) { |
| return Failure::kErrored; |
| } |
| |
| auto type = expect_type(use); |
| if (type.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (match(Token::Type::kComma)) { |
| auto ac = expect_access("access control"); |
| if (ac.errored) { |
| return Failure::kErrored; |
| } |
| access = ac.value; |
| } |
| |
| return type.value; |
| }); |
| |
| if (subtype.errored) { |
| return Failure::kErrored; |
| } |
| |
| return builder_.ty.pointer(make_source_range_from(t.source()), subtype.value, storage_class, |
| access); |
| } |
| |
| Expect<const ast::Type*> ParserImpl::expect_type_decl_atomic(const Token& t) { |
| const char* use = "atomic declaration"; |
| |
| auto subtype = expect_lt_gt_block(use, [&] { return expect_type(use); }); |
| if (subtype.errored) { |
| return Failure::kErrored; |
| } |
| |
| return builder_.ty.atomic(make_source_range_from(t.source()), subtype.value); |
| } |
| |
| Expect<const ast::Type*> ParserImpl::expect_type_decl_vector(const Token& t) { |
| uint32_t count = 2; |
| if (t.Is(Token::Type::kVec3)) { |
| count = 3; |
| } else if (t.Is(Token::Type::kVec4)) { |
| count = 4; |
| } |
| |
| const ast::Type* subtype = nullptr; |
| if (peek_is(Token::Type::kLessThan)) { |
| const char* use = "vector"; |
| auto ty = expect_lt_gt_block(use, [&] { return expect_type(use); }); |
| if (ty.errored) { |
| return Failure::kErrored; |
| } |
| subtype = ty.value; |
| } |
| |
| return builder_.ty.vec(make_source_range_from(t.source()), subtype, count); |
| } |
| |
| Expect<const ast::Type*> ParserImpl::expect_type_decl_array(const Token& t) { |
| const char* use = "array declaration"; |
| |
| struct TypeAndSize { |
| const ast::Type* type = nullptr; |
| const ast::Expression* size = nullptr; |
| }; |
| |
| if (!peek_is(Token::Type::kLessThan)) { |
| return builder_.ty.array(make_source_range_from(t.source()), nullptr, nullptr); |
| } |
| |
| auto type_size = expect_lt_gt_block(use, [&]() -> Expect<TypeAndSize> { |
| auto type = expect_type(use); |
| if (type.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (!match(Token::Type::kComma)) { |
| return TypeAndSize{type.value, nullptr}; |
| } |
| |
| auto size = primary_expression(); |
| if (size.errored) { |
| return Failure::kErrored; |
| } else if (!size.matched) { |
| return add_error(peek(), "expected array size expression"); |
| } |
| |
| return TypeAndSize{type.value, size.value}; |
| }); |
| |
| if (type_size.errored) { |
| return Failure::kErrored; |
| } |
| |
| return builder_.ty.array(make_source_range_from(t.source()), type_size->type, type_size->size); |
| } |
| |
| Expect<const ast::Type*> ParserImpl::expect_type_decl_matrix(const Token& t) { |
| uint32_t rows = 2; |
| uint32_t columns = 2; |
| if (t.IsMat3xN()) { |
| columns = 3; |
| } else if (t.IsMat4xN()) { |
| columns = 4; |
| } |
| if (t.IsMatNx3()) { |
| rows = 3; |
| } else if (t.IsMatNx4()) { |
| rows = 4; |
| } |
| |
| const ast::Type* subtype = nullptr; |
| if (peek_is(Token::Type::kLessThan)) { |
| const char* use = "matrix"; |
| auto ty = expect_lt_gt_block(use, [&] { return expect_type(use); }); |
| if (ty.errored) { |
| return Failure::kErrored; |
| } |
| subtype = ty.value; |
| } |
| |
| return builder_.ty.mat(make_source_range_from(t.source()), subtype, columns, rows); |
| } |
| |
| Expect<ast::StorageClass> ParserImpl::expect_storage_class(std::string_view use) { |
| auto& t = peek(); |
| auto ident = expect_ident("storage class"); |
| if (ident.errored) { |
| return Failure::kErrored; |
| } |
| |
| auto storage_class = ast::ParseStorageClass(ident.value); |
| if (storage_class == ast::StorageClass::kInvalid) { |
| return add_error(t.source(), "invalid storage class", use); |
| } |
| |
| return {storage_class, t.source()}; |
| } |
| |
| // struct_decl |
| // : STRUCT IDENT struct_body_decl |
| Maybe<const ast::Struct*> ParserImpl::struct_decl() { |
| auto& t = peek(); |
| |
| if (!match(Token::Type::kStruct)) { |
| return Failure::kNoMatch; |
| } |
| |
| auto name = expect_ident("struct declaration"); |
| if (name.errored) { |
| return Failure::kErrored; |
| } |
| |
| auto body = expect_struct_body_decl(); |
| if (body.errored) { |
| return Failure::kErrored; |
| } |
| |
| auto sym = builder_.Symbols().Register(name.value); |
| return create<ast::Struct>(t.source(), sym, std::move(body.value), utils::Empty); |
| } |
| |
| // struct_body_decl |
| // : BRACE_LEFT (struct_member COMMA)* struct_member COMMA? BRACE_RIGHT |
| Expect<ParserImpl::StructMemberList> ParserImpl::expect_struct_body_decl() { |
| return expect_brace_block("struct declaration", [&]() -> Expect<StructMemberList> { |
| StructMemberList members; |
| bool errored = false; |
| while (continue_parsing()) { |
| // Check for the end of the list. |
| auto& t = peek(); |
| if (!t.IsIdentifier() && !t.Is(Token::Type::kAttr)) { |
| break; |
| } |
| |
| auto member = expect_struct_member(); |
| if (member.errored) { |
| errored = true; |
| if (!sync_to(Token::Type::kComma, /* consume: */ false)) { |
| return Failure::kErrored; |
| } |
| } else { |
| members.Push(member.value); |
| } |
| |
| if (!match(Token::Type::kComma)) { |
| break; |
| } |
| } |
| if (errored) { |
| return Failure::kErrored; |
| } |
| return members; |
| }); |
| } |
| |
| // struct_member |
| // : attribute* variable_ident_decl |
| Expect<ast::StructMember*> ParserImpl::expect_struct_member() { |
| auto attrs = attribute_list(); |
| if (attrs.errored) { |
| return Failure::kErrored; |
| } |
| |
| auto decl = expect_variable_ident_decl("struct member", |
| /*allow_inferred = */ false); |
| if (decl.errored) { |
| return Failure::kErrored; |
| } |
| |
| return create<ast::StructMember>(decl->source, builder_.Symbols().Register(decl->name), |
| decl->type, std::move(attrs.value)); |
| } |
| |
| // function_decl |
| // : function_header body_stmt |
| Maybe<const ast::Function*> ParserImpl::function_decl(AttributeList& attrs) { |
| auto header = function_header(); |
| if (header.errored) { |
| if (sync_to(Token::Type::kBraceLeft, /* consume: */ false)) { |
| // There were errors in the function header, but the parser has managed to |
| // resynchronize with the opening brace. As there's no outer |
| // synchronization token for function declarations, attempt to parse the |
| // function body. The AST isn't used as we've already errored, but this |
| // catches any errors inside the body, and can help keep the parser in |
| // sync. |
| expect_body_stmt(); |
| } |
| return Failure::kErrored; |
| } |
| if (!header.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| bool errored = false; |
| |
| auto body = expect_body_stmt(); |
| if (body.errored) { |
| errored = true; |
| } |
| |
| if (errored) { |
| return Failure::kErrored; |
| } |
| |
| TINT_DEFER(attrs.Clear()); |
| |
| return create<ast::Function>(header->source, builder_.Symbols().Register(header->name), |
| header->params, header->return_type, body.value, std::move(attrs), |
| header->return_type_attributes); |
| } |
| |
| // function_header |
| // : FN IDENT PAREN_LEFT param_list PAREN_RIGHT return_type_decl_optional |
| // return_type_decl_optional |
| // : |
| // | ARROW attribute_list* type_decl |
| Maybe<ParserImpl::FunctionHeader> ParserImpl::function_header() { |
| Source source; |
| if (!match(Token::Type::kFn, &source)) { |
| return Failure::kNoMatch; |
| } |
| |
| const char* use = "function declaration"; |
| bool errored = false; |
| |
| auto name = expect_ident(use); |
| if (name.errored) { |
| errored = true; |
| if (!sync_to(Token::Type::kParenLeft, /* consume: */ false)) { |
| return Failure::kErrored; |
| } |
| } |
| |
| auto params = expect_paren_block(use, [&] { return expect_param_list(); }); |
| if (params.errored) { |
| errored = true; |
| if (!synchronized_) { |
| return Failure::kErrored; |
| } |
| } |
| |
| const ast::Type* return_type = nullptr; |
| AttributeList return_attributes; |
| |
| if (match(Token::Type::kArrow)) { |
| auto attrs = attribute_list(); |
| if (attrs.errored) { |
| return Failure::kErrored; |
| } |
| return_attributes = attrs.value; |
| |
| auto type = type_decl(); |
| if (type.errored) { |
| errored = true; |
| } else if (!type.matched) { |
| return add_error(peek(), "unable to determine function return type"); |
| } else { |
| return_type = type.value; |
| } |
| } else { |
| return_type = builder_.ty.void_(); |
| } |
| |
| if (errored) { |
| return Failure::kErrored; |
| } |
| |
| return FunctionHeader{ |
| source, std::move(name.value), std::move(params.value), |
| return_type, std::move(return_attributes), |
| }; |
| } |
| |
| // param_list |
| // : |
| // | (param COMMA)* param COMMA? |
| Expect<ParserImpl::ParameterList> ParserImpl::expect_param_list() { |
| ParameterList ret; |
| while (continue_parsing()) { |
| // Check for the end of the list. |
| auto& t = peek(); |
| if (!t.IsIdentifier() && !t.Is(Token::Type::kAttr)) { |
| break; |
| } |
| |
| auto param = expect_param(); |
| if (param.errored) { |
| return Failure::kErrored; |
| } |
| ret.Push(param.value); |
| |
| if (!match(Token::Type::kComma)) { |
| break; |
| } |
| } |
| |
| return ret; |
| } |
| |
| // param |
| // : attribute_list* variable_ident_decl |
| Expect<ast::Parameter*> ParserImpl::expect_param() { |
| auto attrs = attribute_list(); |
| |
| auto decl = expect_variable_ident_decl("parameter", |
| /*allow_inferred = */ false); |
| if (decl.errored) { |
| return Failure::kErrored; |
| } |
| |
| return create<ast::Parameter>(decl->source, // source |
| builder_.Symbols().Register(decl->name), // symbol |
| decl->type, // type |
| std::move(attrs.value)); // attributes |
| } |
| |
| // pipeline_stage |
| // : VERTEX |
| // | FRAGMENT |
| // | COMPUTE |
| Expect<ast::PipelineStage> ParserImpl::expect_pipeline_stage() { |
| auto& t = peek(); |
| if (t == kVertexStage) { |
| next(); // Consume the peek |
| return {ast::PipelineStage::kVertex, t.source()}; |
| } |
| if (t == kFragmentStage) { |
| next(); // Consume the peek |
| return {ast::PipelineStage::kFragment, t.source()}; |
| } |
| if (t == kComputeStage) { |
| next(); // Consume the peek |
| return {ast::PipelineStage::kCompute, t.source()}; |
| } |
| return add_error(peek(), "invalid value for stage attribute"); |
| } |
| |
| Expect<ast::BuiltinValue> ParserImpl::expect_builtin() { |
| auto ident = expect_ident("builtin"); |
| if (ident.errored) { |
| return Failure::kErrored; |
| } |
| |
| ast::BuiltinValue builtin = ast::ParseBuiltinValue(ident.value); |
| if (builtin == ast::BuiltinValue::kInvalid) { |
| return add_error(ident.source, "invalid value for builtin attribute"); |
| } |
| |
| return {builtin, ident.source}; |
| } |
| |
| // body_stmt |
| // : BRACE_LEFT statements BRACE_RIGHT |
| Expect<ast::BlockStatement*> ParserImpl::expect_body_stmt() { |
| return expect_brace_block("", [&]() -> Expect<ast::BlockStatement*> { |
| auto stmts = expect_statements(); |
| if (stmts.errored) { |
| return Failure::kErrored; |
| } |
| return create<ast::BlockStatement>(Source{}, stmts.value); |
| }); |
| } |
| |
| // paren_expression |
| // : PAREN_LEFT logical_or_expression PAREN_RIGHT |
| Expect<const ast::Expression*> ParserImpl::expect_paren_expression() { |
| return expect_paren_block("", [&]() -> Expect<const ast::Expression*> { |
| auto expr = logical_or_expression(); |
| if (expr.errored) { |
| return Failure::kErrored; |
| } |
| if (!expr.matched) { |
| return add_error(peek(), "unable to parse expression"); |
| } |
| |
| return expr.value; |
| }); |
| } |
| |
| // statements |
| // : statement* |
| Expect<ParserImpl::StatementList> ParserImpl::expect_statements() { |
| bool errored = false; |
| StatementList stmts; |
| |
| while (continue_parsing()) { |
| auto stmt = statement(); |
| if (stmt.errored) { |
| errored = true; |
| } else if (stmt.matched) { |
| stmts.Push(stmt.value); |
| } else { |
| break; |
| } |
| } |
| |
| if (errored) { |
| return Failure::kErrored; |
| } |
| |
| return stmts; |
| } |
| |
| // statement |
| // : SEMICOLON |
| // | body_stmt? |
| // | if_stmt |
| // | switch_stmt |
| // | loop_stmt |
| // | for_stmt |
| // | while_stmt |
| // | non_block_statement |
| // : return_stmt SEMICOLON |
| // | func_call_stmt SEMICOLON |
| // | variable_stmt SEMICOLON |
| // | break_stmt SEMICOLON |
| // | continue_stmt SEMICOLON |
| // | DISCARD SEMICOLON |
| // | assignment_stmt SEMICOLON |
| // | increment_stmt SEMICOLON |
| // | decrement_stmt SEMICOLON |
| Maybe<const ast::Statement*> ParserImpl::statement() { |
| while (match(Token::Type::kSemicolon)) { |
| // Skip empty statements |
| } |
| |
| // Non-block statments that error can resynchronize on semicolon. |
| auto stmt = sync(Token::Type::kSemicolon, [&] { return non_block_statement(); }); |
| |
| if (stmt.errored) { |
| return Failure::kErrored; |
| } |
| if (stmt.matched) { |
| return stmt; |
| } |
| |
| auto stmt_if = if_stmt(); |
| if (stmt_if.errored) { |
| return Failure::kErrored; |
| } |
| if (stmt_if.matched) { |
| return stmt_if.value; |
| } |
| |
| auto sw = switch_stmt(); |
| if (sw.errored) { |
| return Failure::kErrored; |
| } |
| if (sw.matched) { |
| return sw.value; |
| } |
| |
| auto loop = loop_stmt(); |
| if (loop.errored) { |
| return Failure::kErrored; |
| } |
| if (loop.matched) { |
| return loop.value; |
| } |
| |
| auto stmt_for = for_stmt(); |
| if (stmt_for.errored) { |
| return Failure::kErrored; |
| } |
| if (stmt_for.matched) { |
| return stmt_for.value; |
| } |
| |
| auto stmt_while = while_stmt(); |
| if (stmt_while.errored) { |
| return Failure::kErrored; |
| } |
| if (stmt_while.matched) { |
| return stmt_while.value; |
| } |
| |
| if (peek_is(Token::Type::kBraceLeft)) { |
| auto body = expect_body_stmt(); |
| if (body.errored) { |
| return Failure::kErrored; |
| } |
| return body.value; |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // statement (continued) |
| // : return_stmt SEMICOLON |
| // | func_call_stmt SEMICOLON |
| // | variable_stmt SEMICOLON |
| // | break_stmt SEMICOLON |
| // | continue_stmt SEMICOLON |
| // | DISCARD SEMICOLON |
| // | assignment_stmt SEMICOLON |
| // | increment_stmt SEMICOLON |
| // | decrement_stmt SEMICOLON |
| Maybe<const ast::Statement*> ParserImpl::non_block_statement() { |
| auto stmt = [&]() -> Maybe<const ast::Statement*> { |
| auto ret_stmt = return_stmt(); |
| if (ret_stmt.errored) { |
| return Failure::kErrored; |
| } |
| if (ret_stmt.matched) { |
| return ret_stmt.value; |
| } |
| |
| auto func = func_call_stmt(); |
| if (func.errored) { |
| return Failure::kErrored; |
| } |
| if (func.matched) { |
| return func.value; |
| } |
| |
| auto var = variable_stmt(); |
| if (var.errored) { |
| return Failure::kErrored; |
| } |
| if (var.matched) { |
| return var.value; |
| } |
| |
| auto b = break_stmt(); |
| if (b.errored) { |
| return Failure::kErrored; |
| } |
| if (b.matched) { |
| return b.value; |
| } |
| |
| auto cont = continue_stmt(); |
| if (cont.errored) { |
| return Failure::kErrored; |
| } |
| if (cont.matched) { |
| return cont.value; |
| } |
| |
| auto assign = assignment_stmt(); |
| if (assign.errored) { |
| return Failure::kErrored; |
| } |
| if (assign.matched) { |
| return assign.value; |
| } |
| |
| Source source; |
| if (match(Token::Type::kDiscard, &source)) { |
| return create<ast::DiscardStatement>(source); |
| } |
| |
| return Failure::kNoMatch; |
| }(); |
| |
| if (stmt.matched && !expect(stmt->Name(), Token::Type::kSemicolon)) { |
| return Failure::kErrored; |
| } |
| |
| return stmt; |
| } |
| |
| // return_stmt |
| // : RETURN logical_or_expression? |
| Maybe<const ast::ReturnStatement*> ParserImpl::return_stmt() { |
| Source source; |
| if (!match(Token::Type::kReturn, &source)) { |
| return Failure::kNoMatch; |
| } |
| |
| if (peek_is(Token::Type::kSemicolon)) { |
| return create<ast::ReturnStatement>(source, nullptr); |
| } |
| |
| auto expr = logical_or_expression(); |
| if (expr.errored) { |
| return Failure::kErrored; |
| } |
| |
| // TODO(bclayton): Check matched? |
| return create<ast::ReturnStatement>(source, expr.value); |
| } |
| |
| // variable_stmt |
| // : variable_decl |
| // | variable_decl EQUAL logical_or_expression |
| // | CONST variable_ident_decl EQUAL logical_or_expression |
| Maybe<const ast::VariableDeclStatement*> ParserImpl::variable_stmt() { |
| if (match(Token::Type::kConst)) { |
| auto decl = expect_variable_ident_decl("'const' declaration", |
| /*allow_inferred = */ true); |
| if (decl.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (!expect("'const' declaration", Token::Type::kEqual)) { |
| return Failure::kErrored; |
| } |
| |
| auto initializer = logical_or_expression(); |
| if (initializer.errored) { |
| return Failure::kErrored; |
| } |
| if (!initializer.matched) { |
| return add_error(peek(), "missing initializer for 'const' declaration"); |
| } |
| |
| auto* const_ = create<ast::Const>(decl->source, // source |
| builder_.Symbols().Register(decl->name), // symbol |
| decl->type, // type |
| initializer.value, // initializer |
| utils::Empty); // attributes |
| |
| return create<ast::VariableDeclStatement>(decl->source, const_); |
| } |
| |
| if (match(Token::Type::kLet)) { |
| auto decl = expect_variable_ident_decl("'let' declaration", |
| /*allow_inferred = */ true); |
| if (decl.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (!expect("'let' declaration", Token::Type::kEqual)) { |
| return Failure::kErrored; |
| } |
| |
| auto initializer = logical_or_expression(); |
| if (initializer.errored) { |
| return Failure::kErrored; |
| } |
| if (!initializer.matched) { |
| return add_error(peek(), "missing initializer for 'let' declaration"); |
| } |
| |
| auto* let = create<ast::Let>(decl->source, // source |
| builder_.Symbols().Register(decl->name), // symbol |
| decl->type, // type |
| initializer.value, // initializer |
| utils::Empty); // attributes |
| |
| return create<ast::VariableDeclStatement>(decl->source, let); |
| } |
| |
| auto decl = variable_decl(); |
| if (decl.errored) { |
| return Failure::kErrored; |
| } |
| if (!decl.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| const ast::Expression* initializer = nullptr; |
| if (match(Token::Type::kEqual)) { |
| auto initializer_expr = logical_or_expression(); |
| if (initializer_expr.errored) { |
| return Failure::kErrored; |
| } |
| if (!initializer_expr.matched) { |
| return add_error(peek(), "missing initializer for 'var' declaration"); |
| } |
| |
| initializer = initializer_expr.value; |
| } |
| |
| auto* var = create<ast::Var>(decl->source, // source |
| builder_.Symbols().Register(decl->name), // symbol |
| decl->type, // type |
| decl->storage_class, // storage class |
| decl->access, // access control |
| initializer, // initializer |
| utils::Empty); // attributes |
| |
| return create<ast::VariableDeclStatement>(var->source, var); |
| } |
| |
| // if_stmt |
| // : IF expression compound_stmt ( ELSE else_stmt ) ? |
| // else_stmt |
| // : body_stmt |
| // | if_stmt |
| Maybe<const ast::IfStatement*> ParserImpl::if_stmt() { |
| // Parse if-else chains iteratively instead of recursively, to avoid |
| // stack-overflow for long chains of if-else statements. |
| |
| struct IfInfo { |
| Source source; |
| const ast::Expression* condition; |
| const ast::BlockStatement* body; |
| }; |
| |
| // Parse an if statement, capturing the source, condition, and body statement. |
| auto parse_if = [&]() -> Maybe<IfInfo> { |
| Source source; |
| if (!match(Token::Type::kIf, &source)) { |
| return Failure::kNoMatch; |
| } |
| |
| auto condition = logical_or_expression(); |
| if (condition.errored) { |
| return Failure::kErrored; |
| } |
| if (!condition.matched) { |
| return add_error(peek(), "unable to parse condition expression"); |
| } |
| |
| auto body = expect_body_stmt(); |
| if (body.errored) { |
| return Failure::kErrored; |
| } |
| |
| return IfInfo{source, condition.value, body.value}; |
| }; |
| |
| std::vector<IfInfo> statements; |
| |
| // Parse the first if statement. |
| auto first_if = parse_if(); |
| if (first_if.errored) { |
| return Failure::kErrored; |
| } else if (!first_if.matched) { |
| return Failure::kNoMatch; |
| } |
| statements.push_back(first_if.value); |
| |
| // Parse the components of every "else {if}" in the chain. |
| const ast::Statement* last_stmt = nullptr; |
| while (continue_parsing()) { |
| if (!match(Token::Type::kElse)) { |
| break; |
| } |
| |
| // Try to parse an "else if". |
| auto else_if = parse_if(); |
| if (else_if.errored) { |
| return Failure::kErrored; |
| } else if (else_if.matched) { |
| statements.push_back(else_if.value); |
| continue; |
| } |
| |
| // If it wasn't an "else if", it must just be an "else". |
| auto else_body = expect_body_stmt(); |
| if (else_body.errored) { |
| return Failure::kErrored; |
| } |
| last_stmt = else_body.value; |
| break; |
| } |
| |
| // Now walk back through the statements to create their AST nodes. |
| for (auto itr = statements.rbegin(); itr != statements.rend(); itr++) { |
| last_stmt = create<ast::IfStatement>(itr->source, itr->condition, itr->body, last_stmt); |
| } |
| |
| return last_stmt->As<ast::IfStatement>(); |
| } |
| |
| // switch_stmt |
| // : SWITCH paren_expression BRACKET_LEFT switch_body+ BRACKET_RIGHT |
| Maybe<const ast::SwitchStatement*> ParserImpl::switch_stmt() { |
| Source source; |
| if (!match(Token::Type::kSwitch, &source)) { |
| return Failure::kNoMatch; |
| } |
| |
| auto condition = logical_or_expression(); |
| if (condition.errored) { |
| return Failure::kErrored; |
| } |
| if (!condition.matched) { |
| return add_error(peek(), "unable to parse selector expression"); |
| } |
| |
| auto body = expect_brace_block("switch statement", [&]() -> Expect<CaseStatementList> { |
| bool errored = false; |
| CaseStatementList list; |
| while (continue_parsing()) { |
| auto stmt = switch_body(); |
| if (stmt.errored) { |
| errored = true; |
| continue; |
| } |
| if (!stmt.matched) { |
| break; |
| } |
| list.Push(stmt.value); |
| } |
| if (errored) { |
| return Failure::kErrored; |
| } |
| return list; |
| }); |
| |
| if (body.errored) { |
| return Failure::kErrored; |
| } |
| |
| return create<ast::SwitchStatement>(source, condition.value, body.value); |
| } |
| |
| // switch_body |
| // : CASE case_selectors COLON? BRACKET_LEFT case_body BRACKET_RIGHT |
| // | DEFAULT COLON? BRACKET_LEFT case_body BRACKET_RIGHT |
| Maybe<const ast::CaseStatement*> ParserImpl::switch_body() { |
| if (!peek_is(Token::Type::kCase) && !peek_is(Token::Type::kDefault)) { |
| return Failure::kNoMatch; |
| } |
| |
| auto& t = next(); |
| |
| CaseSelectorList selector_list; |
| if (t.Is(Token::Type::kCase)) { |
| auto selectors = expect_case_selectors(); |
| if (selectors.errored) { |
| return Failure::kErrored; |
| } |
| |
| selector_list = std::move(selectors.value); |
| } |
| |
| // Consume the optional colon if present. |
| match(Token::Type::kColon); |
| |
| const char* use = "case statement"; |
| auto body = expect_brace_block(use, [&] { return case_body(); }); |
| |
| if (body.errored) { |
| return Failure::kErrored; |
| } |
| if (!body.matched) { |
| return add_error(body.source, "expected case body"); |
| } |
| |
| return create<ast::CaseStatement>(t.source(), selector_list, body.value); |
| } |
| |
| // case_selectors |
| // : const_literal (COMMA const_literal)* COMMA? |
| Expect<ParserImpl::CaseSelectorList> ParserImpl::expect_case_selectors() { |
| CaseSelectorList selectors; |
| |
| while (continue_parsing()) { |
| auto cond = const_literal(); |
| if (cond.errored) { |
| return Failure::kErrored; |
| } else if (!cond.matched) { |
| break; |
| } else if (!cond->Is<ast::IntLiteralExpression>()) { |
| return add_error(cond.value->source, "invalid case selector must be an integer value"); |
| } |
| |
| selectors.Push(cond.value->As<ast::IntLiteralExpression>()); |
| |
| if (!match(Token::Type::kComma)) { |
| break; |
| } |
| } |
| |
| if (selectors.IsEmpty()) { |
| return add_error(peek(), "unable to parse case selectors"); |
| } |
| |
| return selectors; |
| } |
| |
| // case_body |
| // : |
| // | statement case_body |
| // | FALLTHROUGH SEMICOLON |
| Maybe<const ast::BlockStatement*> ParserImpl::case_body() { |
| StatementList stmts; |
| while (continue_parsing()) { |
| Source source; |
| if (match(Token::Type::kFallthrough, &source)) { |
| if (!expect("fallthrough statement", Token::Type::kSemicolon)) { |
| return Failure::kErrored; |
| } |
| |
| stmts.Push(create<ast::FallthroughStatement>(source)); |
| break; |
| } |
| |
| auto stmt = statement(); |
| if (stmt.errored) { |
| return Failure::kErrored; |
| } |
| if (!stmt.matched) { |
| break; |
| } |
| |
| stmts.Push(stmt.value); |
| } |
| |
| return create<ast::BlockStatement>(Source{}, stmts); |
| } |
| |
| // loop_stmt |
| // : LOOP BRACKET_LEFT statements continuing_stmt? BRACKET_RIGHT |
| Maybe<const ast::LoopStatement*> ParserImpl::loop_stmt() { |
| Source source; |
| if (!match(Token::Type::kLoop, &source)) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_brace_block("loop", [&]() -> Maybe<const ast::LoopStatement*> { |
| auto stmts = expect_statements(); |
| if (stmts.errored) { |
| return Failure::kErrored; |
| } |
| |
| auto continuing = continuing_stmt(); |
| if (continuing.errored) { |
| return Failure::kErrored; |
| } |
| |
| auto* body = create<ast::BlockStatement>(source, stmts.value); |
| return create<ast::LoopStatement>(source, body, continuing.value); |
| }); |
| } |
| |
| ForHeader::ForHeader(const ast::Statement* init, |
| const ast::Expression* cond, |
| const ast::Statement* cont) |
| : initializer(init), condition(cond), continuing(cont) {} |
| |
| ForHeader::~ForHeader() = default; |
| |
| // (variable_stmt | increment_stmt | decrement_stmt | assignment_stmt | |
| // func_call_stmt)? |
| Maybe<const ast::Statement*> ParserImpl::for_header_initializer() { |
| auto call = func_call_stmt(); |
| if (call.errored) { |
| return Failure::kErrored; |
| } |
| if (call.matched) { |
| return call.value; |
| } |
| |
| auto var = variable_stmt(); |
| if (var.errored) { |
| return Failure::kErrored; |
| } |
| if (var.matched) { |
| return var.value; |
| } |
| |
| auto assign = assignment_stmt(); |
| if (assign.errored) { |
| return Failure::kErrored; |
| } |
| if (assign.matched) { |
| return assign.value; |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // (increment_stmt | decrement_stmt | assignment_stmt | func_call_stmt)? |
| Maybe<const ast::Statement*> ParserImpl::for_header_continuing() { |
| auto call_stmt = func_call_stmt(); |
| if (call_stmt.errored) { |
| return Failure::kErrored; |
| } |
| if (call_stmt.matched) { |
| return call_stmt.value; |
| } |
| |
| auto assign = assignment_stmt(); |
| if (assign.errored) { |
| return Failure::kErrored; |
| } |
| if (assign.matched) { |
| return assign.value; |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // for_header |
| // : (variable_stmt | assignment_stmt | func_call_stmt)? |
| // SEMICOLON |
| // logical_or_expression? SEMICOLON |
| // (assignment_stmt | func_call_stmt)? |
| Expect<std::unique_ptr<ForHeader>> ParserImpl::expect_for_header() { |
| auto initializer = for_header_initializer(); |
| if (initializer.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (!expect("initializer in for loop", Token::Type::kSemicolon)) { |
| return Failure::kErrored; |
| } |
| |
| auto condition = logical_or_expression(); |
| if (condition.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (!expect("condition in for loop", Token::Type::kSemicolon)) { |
| return Failure::kErrored; |
| } |
| |
| auto continuing = for_header_continuing(); |
| if (continuing.errored) { |
| return Failure::kErrored; |
| } |
| |
| return std::make_unique<ForHeader>(initializer.value, condition.value, continuing.value); |
| } |
| |
| // for_statement |
| // : FOR PAREN_LEFT for_header PAREN_RIGHT BRACE_LEFT statements BRACE_RIGHT |
| Maybe<const ast::ForLoopStatement*> ParserImpl::for_stmt() { |
| Source source; |
| if (!match(Token::Type::kFor, &source)) { |
| return Failure::kNoMatch; |
| } |
| |
| auto header = expect_paren_block("for loop", [&] { return expect_for_header(); }); |
| if (header.errored) { |
| return Failure::kErrored; |
| } |
| |
| auto stmts = expect_brace_block("for loop", [&] { return expect_statements(); }); |
| if (stmts.errored) { |
| return Failure::kErrored; |
| } |
| |
| return create<ast::ForLoopStatement>(source, header->initializer, header->condition, |
| header->continuing, |
| create<ast::BlockStatement>(stmts.value)); |
| } |
| |
| // while_statement |
| // : WHILE expression compound_statement |
| Maybe<const ast::WhileStatement*> ParserImpl::while_stmt() { |
| Source source; |
| if (!match(Token::Type::kWhile, &source)) { |
| return Failure::kNoMatch; |
| } |
| |
| auto condition = logical_or_expression(); |
| if (condition.errored) { |
| return Failure::kErrored; |
| } |
| if (!condition.matched) { |
| return add_error(peek(), "unable to parse while condition expression"); |
| } |
| |
| auto body = expect_body_stmt(); |
| if (body.errored) { |
| return Failure::kErrored; |
| } |
| |
| return create<ast::WhileStatement>(source, condition.value, body.value); |
| } |
| |
| // func_call_stmt |
| // : IDENT argument_expression_list |
| Maybe<const ast::CallStatement*> ParserImpl::func_call_stmt() { |
| auto& t = peek(); |
| auto& t2 = peek(1); |
| if (!t.IsIdentifier() || !t2.Is(Token::Type::kParenLeft)) { |
| return Failure::kNoMatch; |
| } |
| |
| next(); // Consume the first peek |
| |
| auto params = expect_argument_expression_list("function call"); |
| if (params.errored) { |
| return Failure::kErrored; |
| } |
| |
| return create<ast::CallStatement>( |
| t.source(), |
| create<ast::CallExpression>( |
| t.source(), |
| create<ast::IdentifierExpression>(t.source(), builder_.Symbols().Register(t.to_str())), |
| std::move(params.value))); |
| } |
| |
| // break_stmt |
| // : BREAK |
| Maybe<const ast::BreakStatement*> ParserImpl::break_stmt() { |
| Source source; |
| if (!match(Token::Type::kBreak, &source)) { |
| return Failure::kNoMatch; |
| } |
| |
| return create<ast::BreakStatement>(source); |
| } |
| |
| // continue_stmt |
| // : CONTINUE |
| Maybe<const ast::ContinueStatement*> ParserImpl::continue_stmt() { |
| Source source; |
| if (!match(Token::Type::kContinue, &source)) { |
| return Failure::kNoMatch; |
| } |
| |
| return create<ast::ContinueStatement>(source); |
| } |
| |
| // continuing_stmt |
| // : CONTINUING body_stmt |
| Maybe<const ast::BlockStatement*> ParserImpl::continuing_stmt() { |
| if (!match(Token::Type::kContinuing)) { |
| return create<ast::BlockStatement>(Source{}, utils::Empty); |
| } |
| |
| return expect_body_stmt(); |
| } |
| |
| // primary_expression |
| // : IDENT argument_expression_list? |
| // | type_decl argument_expression_list |
| // | const_literal |
| // | paren_expression |
| // | BITCAST LESS_THAN type_decl GREATER_THAN paren_expression |
| Maybe<const ast::Expression*> ParserImpl::primary_expression() { |
| auto& t = peek(); |
| |
| auto lit = const_literal(); |
| if (lit.errored) { |
| return Failure::kErrored; |
| } |
| if (lit.matched) { |
| return lit.value; |
| } |
| |
| if (t.Is(Token::Type::kParenLeft)) { |
| auto paren = expect_paren_expression(); |
| if (paren.errored) { |
| return Failure::kErrored; |
| } |
| |
| return paren.value; |
| } |
| |
| if (match(Token::Type::kBitcast)) { |
| const char* use = "bitcast expression"; |
| |
| auto type = expect_lt_gt_block(use, [&] { return expect_type(use); }); |
| if (type.errored) { |
| return Failure::kErrored; |
| } |
| |
| auto params = expect_paren_expression(); |
| if (params.errored) { |
| return Failure::kErrored; |
| } |
| |
| return create<ast::BitcastExpression>(t.source(), type.value, params.value); |
| } |
| |
| if (t.IsIdentifier()) { |
| next(); |
| |
| auto* ident = |
| create<ast::IdentifierExpression>(t.source(), builder_.Symbols().Register(t.to_str())); |
| |
| if (peek_is(Token::Type::kParenLeft)) { |
| auto params = expect_argument_expression_list("function call"); |
| if (params.errored) { |
| return Failure::kErrored; |
| } |
| |
| return create<ast::CallExpression>(t.source(), ident, std::move(params.value)); |
| } |
| |
| return ident; |
| } |
| |
| auto type = type_decl(); |
| if (type.errored) { |
| return Failure::kErrored; |
| } |
| if (type.matched) { |
| auto params = expect_argument_expression_list("type constructor"); |
| if (params.errored) { |
| return Failure::kErrored; |
| } |
| |
| return builder_.Construct(t.source(), type.value, std::move(params.value)); |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| // postfix_expression |
| // : |
| // | BRACE_LEFT logical_or_expression BRACE_RIGHT postfix_expr |
| // | PERIOD IDENTIFIER postfix_expr |
| Maybe<const ast::Expression*> ParserImpl::postfix_expression(const ast::Expression* prefix) { |
| Source source; |
| |
| while (continue_parsing()) { |
| if (match(Token::Type::kBracketLeft, &source)) { |
| auto res = sync(Token::Type::kBracketRight, [&]() -> Maybe<const ast::Expression*> { |
| auto param = logical_or_expression(); |
| if (param.errored) { |
| return Failure::kErrored; |
| } |
| if (!param.matched) { |
| return add_error(peek(), "unable to parse expression inside []"); |
| } |
| |
| if (!expect("index accessor", Token::Type::kBracketRight)) { |
| return Failure::kErrored; |
| } |
| |
| return create<ast::IndexAccessorExpression>(source, prefix, param.value); |
| }); |
| |
| if (res.errored) { |
| return res; |
| } |
| prefix = res.value; |
| continue; |
| } |
| |
| if (match(Token::Type::kPeriod)) { |
| auto ident = expect_ident("member accessor"); |
| if (ident.errored) { |
| return Failure::kErrored; |
| } |
| |
| prefix = create<ast::MemberAccessorExpression>( |
| ident.source, prefix, |
| create<ast::IdentifierExpression>(ident.source, |
| builder_.Symbols().Register(ident.value))); |
| continue; |
| } |
| |
| return prefix; |
| } |
| |
| return Failure::kErrored; |
| } |
| |
| // singular_expression |
| // : primary_expression postfix_expr |
| Maybe<const ast::Expression*> ParserImpl::singular_expression() { |
| auto prefix = primary_expression(); |
| if (prefix.errored) { |
| return Failure::kErrored; |
| } |
| if (!prefix.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return postfix_expression(prefix.value); |
| } |
| |
| // argument_expression_list |
| // : PAREN_LEFT ((logical_or_expression COMMA)* logical_or_expression COMMA?)? |
| // PAREN_RIGHT |
| Expect<ParserImpl::ExpressionList> ParserImpl::expect_argument_expression_list( |
| std::string_view use) { |
| return expect_paren_block(use, [&]() -> Expect<ExpressionList> { |
| ExpressionList ret; |
| while (continue_parsing()) { |
| auto arg = logical_or_expression(); |
| if (arg.errored) { |
| return Failure::kErrored; |
| } else if (!arg.matched) { |
| break; |
| } |
| ret.Push(arg.value); |
| |
| if (!match(Token::Type::kComma)) { |
| break; |
| } |
| } |
| return ret; |
| }); |
| } |
| |
| // unary_expression |
| // : singular_expression |
| // | MINUS unary_expression |
| // | BANG unary_expression |
| // | TILDE unary_expression |
| // | STAR unary_expression |
| // | AND unary_expression |
| Maybe<const ast::Expression*> ParserImpl::unary_expression() { |
| auto& t = peek(); |
| |
| if (match(Token::Type::kPlusPlus) || match(Token::Type::kMinusMinus)) { |
| add_error(t.source(), |
| "prefix increment and decrement operators are reserved for a " |
| "future WGSL version"); |
| return Failure::kErrored; |
| } |
| |
| ast::UnaryOp op; |
| if (match(Token::Type::kMinus)) { |
| op = ast::UnaryOp::kNegation; |
| } else if (match(Token::Type::kBang)) { |
| op = ast::UnaryOp::kNot; |
| } else if (match(Token::Type::kTilde)) { |
| op = ast::UnaryOp::kComplement; |
| } else if (match(Token::Type::kStar)) { |
| op = ast::UnaryOp::kIndirection; |
| } else if (match(Token::Type::kAnd)) { |
| op = ast::UnaryOp::kAddressOf; |
| } else { |
| return singular_expression(); |
| } |
| |
| if (parse_depth_ >= kMaxParseDepth) { |
| // We've hit a maximum parser recursive depth. |
| // We can't call into unary_expression() as we might stack overflow. |
| // Instead, report an error |
| add_error(peek(), "maximum parser recursive depth reached"); |
| return Failure::kErrored; |
| } |
| |
| ++parse_depth_; |
| auto expr = unary_expression(); |
| --parse_depth_; |
| |
| if (expr.errored) { |
| return Failure::kErrored; |
| } |
| if (!expr.matched) { |
| return add_error( |
| peek(), "unable to parse right side of " + std::string(t.to_name()) + " expression"); |
| } |
| |
| return create<ast::UnaryOpExpression>(t.source(), op, expr.value); |
| } |
| |
| // multiplicative_expr |
| // : |
| // | STAR unary_expression multiplicative_expr |
| // | FORWARD_SLASH unary_expression multiplicative_expr |
| // | MODULO unary_expression multiplicative_expr |
| Expect<const ast::Expression*> ParserImpl::expect_multiplicative_expr(const ast::Expression* lhs) { |
| while (continue_parsing()) { |
| ast::BinaryOp op = ast::BinaryOp::kNone; |
| if (peek_is(Token::Type::kStar)) { |
| op = ast::BinaryOp::kMultiply; |
| } else if (peek_is(Token::Type::kForwardSlash)) { |
| op = ast::BinaryOp::kDivide; |
| } else if (peek_is(Token::Type::kMod)) { |
| op = ast::BinaryOp::kModulo; |
| } else { |
| return lhs; |
| } |
| |
| auto& t = next(); |
| |
| auto rhs = unary_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), "unable to parse right side of " + std::string(t.to_name()) + |
| " expression"); |
| } |
| |
| lhs = create<ast::BinaryExpression>(t.source(), op, lhs, rhs.value); |
| } |
| return Failure::kErrored; |
| } |
| |
| // multiplicative_expression |
| // : unary_expression multiplicative_expr |
| Maybe<const ast::Expression*> ParserImpl::multiplicative_expression() { |
| auto lhs = unary_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_multiplicative_expr(lhs.value); |
| } |
| |
| // additive_expr |
| // : |
| // | PLUS multiplicative_expression additive_expr |
| // | MINUS multiplicative_expression additive_expr |
| Expect<const ast::Expression*> ParserImpl::expect_additive_expr(const ast::Expression* lhs) { |
| while (continue_parsing()) { |
| ast::BinaryOp op = ast::BinaryOp::kNone; |
| if (peek_is(Token::Type::kPlus)) { |
| op = ast::BinaryOp::kAdd; |
| } else if (peek_is(Token::Type::kMinus)) { |
| op = ast::BinaryOp::kSubtract; |
| } else { |
| return lhs; |
| } |
| |
| auto& t = next(); |
| |
| auto rhs = multiplicative_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), "unable to parse right side of + expression"); |
| } |
| |
| lhs = create<ast::BinaryExpression>(t.source(), op, lhs, rhs.value); |
| } |
| return Failure::kErrored; |
| } |
| |
| // additive_expression |
| // : multiplicative_expression additive_expr |
| Maybe<const ast::Expression*> ParserImpl::additive_expression() { |
| auto lhs = multiplicative_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_additive_expr(lhs.value); |
| } |
| |
| // shift_expr |
| // : |
| // | SHIFT_LEFT additive_expression shift_expr |
| // | SHIFT_RIGHT additive_expression shift_expr |
| Expect<const ast::Expression*> ParserImpl::expect_shift_expr(const ast::Expression* lhs) { |
| while (continue_parsing()) { |
| auto* name = ""; |
| ast::BinaryOp op = ast::BinaryOp::kNone; |
| if (peek_is(Token::Type::kShiftLeft)) { |
| op = ast::BinaryOp::kShiftLeft; |
| name = "<<"; |
| } else if (peek_is(Token::Type::kShiftRight)) { |
| op = ast::BinaryOp::kShiftRight; |
| name = ">>"; |
| } else { |
| return lhs; |
| } |
| |
| auto& t = next(); |
| auto rhs = additive_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), |
| std::string("unable to parse right side of ") + name + " expression"); |
| } |
| |
| return lhs = create<ast::BinaryExpression>(t.source(), op, lhs, rhs.value); |
| } |
| return Failure::kErrored; |
| } |
| |
| // shift_expression |
| // : additive_expression shift_expr |
| Maybe<const ast::Expression*> ParserImpl::shift_expression() { |
| auto lhs = additive_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_shift_expr(lhs.value); |
| } |
| |
| // relational_expr |
| // : |
| // | LESS_THAN shift_expression relational_expr |
| // | GREATER_THAN shift_expression relational_expr |
| // | LESS_THAN_EQUAL shift_expression relational_expr |
| // | GREATER_THAN_EQUAL shift_expression relational_expr |
| Expect<const ast::Expression*> ParserImpl::expect_relational_expr(const ast::Expression* lhs) { |
| while (continue_parsing()) { |
| ast::BinaryOp op = ast::BinaryOp::kNone; |
| if (peek_is(Token::Type::kLessThan)) { |
| op = ast::BinaryOp::kLessThan; |
| } else if (peek_is(Token::Type::kGreaterThan)) { |
| op = ast::BinaryOp::kGreaterThan; |
| } else if (peek_is(Token::Type::kLessThanEqual)) { |
| op = ast::BinaryOp::kLessThanEqual; |
| } else if (peek_is(Token::Type::kGreaterThanEqual)) { |
| op = ast::BinaryOp::kGreaterThanEqual; |
| } else { |
| return lhs; |
| } |
| |
| auto& t = next(); |
| |
| auto rhs = shift_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), "unable to parse right side of " + std::string(t.to_name()) + |
| " expression"); |
| } |
| |
| lhs = create<ast::BinaryExpression>(t.source(), op, lhs, rhs.value); |
| } |
| return Failure::kErrored; |
| } |
| |
| // relational_expression |
| // : shift_expression relational_expr |
| Maybe<const ast::Expression*> ParserImpl::relational_expression() { |
| auto lhs = shift_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_relational_expr(lhs.value); |
| } |
| |
| // equality_expr |
| // : |
| // | EQUAL_EQUAL relational_expression equality_expr |
| // | NOT_EQUAL relational_expression equality_expr |
| Expect<const ast::Expression*> ParserImpl::expect_equality_expr(const ast::Expression* lhs) { |
| while (continue_parsing()) { |
| ast::BinaryOp op = ast::BinaryOp::kNone; |
| if (peek_is(Token::Type::kEqualEqual)) { |
| op = ast::BinaryOp::kEqual; |
| } else if (peek_is(Token::Type::kNotEqual)) { |
| op = ast::BinaryOp::kNotEqual; |
| } else { |
| return lhs; |
| } |
| |
| auto& t = next(); |
| |
| auto rhs = relational_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), "unable to parse right side of " + std::string(t.to_name()) + |
| " expression"); |
| } |
| |
| lhs = create<ast::BinaryExpression>(t.source(), op, lhs, rhs.value); |
| } |
| return Failure::kErrored; |
| } |
| |
| // equality_expression |
| // : relational_expression equality_expr |
| Maybe<const ast::Expression*> ParserImpl::equality_expression() { |
| auto lhs = relational_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_equality_expr(lhs.value); |
| } |
| |
| // and_expr |
| // : |
| // | AND equality_expression and_expr |
| Expect<const ast::Expression*> ParserImpl::expect_and_expr(const ast::Expression* lhs) { |
| while (continue_parsing()) { |
| if (!peek_is(Token::Type::kAnd)) { |
| return lhs; |
| } |
| |
| auto& t = next(); |
| |
| auto rhs = equality_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), "unable to parse right side of & expression"); |
| } |
| |
| lhs = create<ast::BinaryExpression>(t.source(), ast::BinaryOp::kAnd, lhs, rhs.value); |
| } |
| return Failure::kErrored; |
| } |
| |
| // and_expression |
| // : equality_expression and_expr |
| Maybe<const ast::Expression*> ParserImpl::and_expression() { |
| auto lhs = equality_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_and_expr(lhs.value); |
| } |
| |
| // exclusive_or_expr |
| // : |
| // | XOR and_expression exclusive_or_expr |
| Expect<const ast::Expression*> ParserImpl::expect_exclusive_or_expr(const ast::Expression* lhs) { |
| while (continue_parsing()) { |
| Source source; |
| if (!match(Token::Type::kXor, &source)) { |
| return lhs; |
| } |
| |
| auto rhs = and_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), "unable to parse right side of ^ expression"); |
| } |
| |
| lhs = create<ast::BinaryExpression>(source, ast::BinaryOp::kXor, lhs, rhs.value); |
| } |
| return Failure::kErrored; |
| } |
| |
| // exclusive_or_expression |
| // : and_expression exclusive_or_expr |
| Maybe<const ast::Expression*> ParserImpl::exclusive_or_expression() { |
| auto lhs = and_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_exclusive_or_expr(lhs.value); |
| } |
| |
| // inclusive_or_expr |
| // : |
| // | OR exclusive_or_expression inclusive_or_expr |
| Expect<const ast::Expression*> ParserImpl::expect_inclusive_or_expr(const ast::Expression* lhs) { |
| while (continue_parsing()) { |
| Source source; |
| if (!match(Token::Type::kOr, &source)) { |
| return lhs; |
| } |
| |
| auto rhs = exclusive_or_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), "unable to parse right side of | expression"); |
| } |
| |
| lhs = create<ast::BinaryExpression>(source, ast::BinaryOp::kOr, lhs, rhs.value); |
| } |
| return Failure::kErrored; |
| } |
| |
| // inclusive_or_expression |
| // : exclusive_or_expression inclusive_or_expr |
| Maybe<const ast::Expression*> ParserImpl::inclusive_or_expression() { |
| auto lhs = exclusive_or_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_inclusive_or_expr(lhs.value); |
| } |
| |
| // logical_and_expr |
| // : |
| // | AND_AND inclusive_or_expression logical_and_expr |
| Expect<const ast::Expression*> ParserImpl::expect_logical_and_expr(const ast::Expression* lhs) { |
| while (continue_parsing()) { |
| if (!peek_is(Token::Type::kAndAnd)) { |
| return lhs; |
| } |
| |
| auto& t = next(); |
| |
| auto rhs = inclusive_or_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), "unable to parse right side of && expression"); |
| } |
| |
| lhs = create<ast::BinaryExpression>(t.source(), ast::BinaryOp::kLogicalAnd, lhs, rhs.value); |
| } |
| return Failure::kErrored; |
| } |
| |
| // logical_and_expression |
| // : inclusive_or_expression logical_and_expr |
| Maybe<const ast::Expression*> ParserImpl::logical_and_expression() { |
| auto lhs = inclusive_or_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_logical_and_expr(lhs.value); |
| } |
| |
| // logical_or_expr |
| // : |
| // | OR_OR logical_and_expression logical_or_expr |
| Expect<const ast::Expression*> ParserImpl::expect_logical_or_expr(const ast::Expression* lhs) { |
| while (continue_parsing()) { |
| Source source; |
| if (!match(Token::Type::kOrOr, &source)) { |
| return lhs; |
| } |
| |
| auto rhs = logical_and_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), "unable to parse right side of || expression"); |
| } |
| |
| lhs = create<ast::BinaryExpression>(source, ast::BinaryOp::kLogicalOr, lhs, rhs.value); |
| } |
| return Failure::kErrored; |
| } |
| |
| // logical_or_expression |
| // : logical_and_expression logical_or_expr |
| Maybe<const ast::Expression*> ParserImpl::logical_or_expression() { |
| auto lhs = logical_and_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| return Failure::kNoMatch; |
| } |
| |
| return expect_logical_or_expr(lhs.value); |
| } |
| |
| // compound_assignment_operator: |
| // | plus_equal |
| // | minus_equal |
| // | times_equal |
| // | division_equal |
| // | modulo_equal |
| // | and_equal |
| // | or_equal |
| // | xor_equal |
| Maybe<ast::BinaryOp> ParserImpl::compound_assignment_operator() { |
| ast::BinaryOp compound_op = ast::BinaryOp::kNone; |
| if (peek_is(Token::Type::kPlusEqual)) { |
| compound_op = ast::BinaryOp::kAdd; |
| } else if (peek_is(Token::Type::kMinusEqual)) { |
| compound_op = ast::BinaryOp::kSubtract; |
| } else if (peek_is(Token::Type::kTimesEqual)) { |
| compound_op = ast::BinaryOp::kMultiply; |
| } else if (peek_is(Token::Type::kDivisionEqual)) { |
| compound_op = ast::BinaryOp::kDivide; |
| } else if (peek_is(Token::Type::kModuloEqual)) { |
| compound_op = ast::BinaryOp::kModulo; |
| } else if (peek_is(Token::Type::kAndEqual)) { |
| compound_op = ast::BinaryOp::kAnd; |
| } else if (peek_is(Token::Type::kOrEqual)) { |
| compound_op = ast::BinaryOp::kOr; |
| } else if (peek_is(Token::Type::kXorEqual)) { |
| compound_op = ast::BinaryOp::kXor; |
| } |
| if (compound_op != ast::BinaryOp::kNone) { |
| next(); |
| return compound_op; |
| } |
| return Failure::kNoMatch; |
| } |
| |
| // assignment_stmt |
| // | lhs_expression ( equal | compound_assignment_operator ) expression |
| // | underscore equal expression |
| // |
| // increment_stmt |
| // | lhs_expression PLUS_PLUS |
| // |
| // decrement_stmt |
| // | lhs_expression MINUS_MINUS |
| Maybe<const ast::Statement*> ParserImpl::assignment_stmt() { |
| auto& t = peek(); |
| |
| // tint:295 - Test for `ident COLON` - this is invalid grammar, and without |
| // special casing will error as "missing = for assignment", which is less |
| // helpful than this error message: |
| if (peek_is(Token::Type::kIdentifier) && peek_is(Token::Type::kColon, 1)) { |
| return add_error(peek(0).source(), "expected 'var' for variable declaration"); |
| } |
| |
| auto lhs = unary_expression(); |
| if (lhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!lhs.matched) { |
| Source source = t.source(); |
| if (!match(Token::Type::kUnderscore, &source)) { |
| return Failure::kNoMatch; |
| } |
| lhs = create<ast::PhonyExpression>(source); |
| } |
| |
| // Handle increment and decrement statements. |
| // We do this here because the parsing of the LHS expression overlaps with |
| // the assignment statement, and we cannot tell which we are parsing until we |
| // hit the ++/--/= token. |
| if (match(Token::Type::kPlusPlus)) { |
| return create<ast::IncrementDecrementStatement>(t.source(), lhs.value, true); |
| } else if (match(Token::Type::kMinusMinus)) { |
| return create<ast::IncrementDecrementStatement>(t.source(), lhs.value, false); |
| } |
| |
| auto compound_op = compound_assignment_operator(); |
| if (compound_op.errored) { |
| return Failure::kErrored; |
| } |
| if (!compound_op.matched) { |
| if (!expect("assignment", Token::Type::kEqual)) { |
| return Failure::kErrored; |
| } |
| } |
| |
| auto rhs = logical_or_expression(); |
| if (rhs.errored) { |
| return Failure::kErrored; |
| } |
| if (!rhs.matched) { |
| return add_error(peek(), "unable to parse right side of assignment"); |
| } |
| |
| if (compound_op.value != ast::BinaryOp::kNone) { |
| return create<ast::CompoundAssignmentStatement>(t.source(), lhs.value, rhs.value, |
| compound_op.value); |
| } else { |
| return create<ast::AssignmentStatement>(t.source(), lhs.value, rhs.value); |
| } |
| } |
| |
| // const_literal |
| // : INT_LITERAL |
| // | FLOAT_LITERAL |
| // | TRUE |
| // | FALSE |
| Maybe<const ast::LiteralExpression*> ParserImpl::const_literal() { |
| auto& t = peek(); |
| if (match(Token::Type::kIntLiteral)) { |
| return create<ast::IntLiteralExpression>(t.source(), t.to_i64(), |
| ast::IntLiteralExpression::Suffix::kNone); |
| } |
| if (match(Token::Type::kIntLiteral_I)) { |
| return create<ast::IntLiteralExpression>(t.source(), t.to_i64(), |
| ast::IntLiteralExpression::Suffix::kI); |
| } |
| if (match(Token::Type::kIntLiteral_U)) { |
| return create<ast::IntLiteralExpression>(t.source(), t.to_i64(), |
| ast::IntLiteralExpression::Suffix::kU); |
| } |
| if (match(Token::Type::kFloatLiteral)) { |
| return create<ast::FloatLiteralExpression>(t.source(), t.to_f64(), |
| ast::FloatLiteralExpression::Suffix::kNone); |
| } |
| if (match(Token::Type::kFloatLiteral_F)) { |
| return create<ast::FloatLiteralExpression>(t.source(), t.to_f64(), |
| ast::FloatLiteralExpression::Suffix::kF); |
| } |
| if (match(Token::Type::kFloatLiteral_H)) { |
| return create<ast::FloatLiteralExpression>(t.source(), t.to_f64(), |
| ast::FloatLiteralExpression::Suffix::kH); |
| } |
| if (match(Token::Type::kTrue)) { |
| return create<ast::BoolLiteralExpression>(t.source(), true); |
| } |
| if (match(Token::Type::kFalse)) { |
| return create<ast::BoolLiteralExpression>(t.source(), false); |
| } |
| if (handle_error(t)) { |
| return Failure::kErrored; |
| } |
| return Failure::kNoMatch; |
| } |
| |
| Maybe<ParserImpl::AttributeList> ParserImpl::attribute_list() { |
| bool errored = false; |
| AttributeList attrs; |
| |
| while (continue_parsing()) { |
| if (match(Token::Type::kAttr)) { |
| if (auto attr = expect_attribute(); attr.errored) { |
| errored = true; |
| } else { |
| attrs.Push(attr.value); |
| } |
| } else { |
| break; |
| } |
| } |
| |
| if (errored) { |
| return Failure::kErrored; |
| } |
| |
| if (attrs.IsEmpty()) { |
| return Failure::kNoMatch; |
| } |
| |
| return attrs; |
| } |
| |
| Expect<const ast::Attribute*> ParserImpl::expect_attribute() { |
| auto& t = peek(); |
| auto attr = attribute(); |
| if (attr.errored) { |
| return Failure::kErrored; |
| } |
| if (attr.matched) { |
| return attr.value; |
| } |
| return add_error(t, "expected attribute"); |
| } |
| |
| Maybe<const ast::Attribute*> ParserImpl::attribute() { |
| using Result = Maybe<const ast::Attribute*>; |
| auto& t = next(); |
| |
| if (!t.IsIdentifier()) { |
| return Failure::kNoMatch; |
| } |
| |
| if (t == kLocationAttribute) { |
| const char* use = "location attribute"; |
| return expect_paren_block(use, [&]() -> Result { |
| auto val = expect_positive_sint(use); |
| if (val.errored) { |
| return Failure::kErrored; |
| } |
| match(Token::Type::kComma); |
| |
| return create<ast::LocationAttribute>(t.source(), val.value); |
| }); |
| } |
| |
| if (t == kBindingAttribute) { |
| const char* use = "binding attribute"; |
| return expect_paren_block(use, [&]() -> Result { |
| auto val = expect_positive_sint(use); |
| if (val.errored) { |
| return Failure::kErrored; |
| } |
| match(Token::Type::kComma); |
| |
| return create<ast::BindingAttribute>(t.source(), val.value); |
| }); |
| } |
| |
| if (t == kGroupAttribute) { |
| const char* use = "group attribute"; |
| return expect_paren_block(use, [&]() -> Result { |
| auto val = expect_positive_sint(use); |
| if (val.errored) { |
| return Failure::kErrored; |
| } |
| match(Token::Type::kComma); |
| |
| return create<ast::GroupAttribute>(t.source(), val.value); |
| }); |
| } |
| |
| if (t == kInterpolateAttribute) { |
| return expect_paren_block("interpolate attribute", [&]() -> Result { |
| ast::InterpolationType type; |
| ast::InterpolationSampling sampling = ast::InterpolationSampling::kNone; |
| |
| auto& type_tok = next(); |
| if (type_tok == "perspective") { |
| type = ast::InterpolationType::kPerspective; |
| } else if (type_tok == "linear") { |
| type = ast::InterpolationType::kLinear; |
| } else if (type_tok == "flat") { |
| type = ast::InterpolationType::kFlat; |
| } else { |
| return add_error(type_tok, "invalid interpolation type"); |
| } |
| |
| if (match(Token::Type::kComma)) { |
| if (!peek_is(Token::Type::kParenRight)) { |
| auto& sampling_tok = next(); |
| if (sampling_tok == "center") { |
| sampling = ast::InterpolationSampling::kCenter; |
| } else if (sampling_tok == "centroid") { |
| sampling = ast::InterpolationSampling::kCentroid; |
| } else if (sampling_tok == "sample") { |
| sampling = ast::InterpolationSampling::kSample; |
| } else { |
| return add_error(sampling_tok, "invalid interpolation sampling"); |
| } |
| match(Token::Type::kComma); |
| } |
| } |
| |
| return create<ast::InterpolateAttribute>(t.source(), type, sampling); |
| }); |
| } |
| |
| if (t == kInvariantAttribute) { |
| return create<ast::InvariantAttribute>(t.source()); |
| } |
| |
| if (t == kBuiltinAttribute) { |
| return expect_paren_block("builtin attribute", [&]() -> Result { |
| auto builtin = expect_builtin(); |
| if (builtin.errored) { |
| return Failure::kErrored; |
| } |
| |
| match(Token::Type::kComma); |
| return create<ast::BuiltinAttribute>(t.source(), builtin.value); |
| }); |
| } |
| |
| if (t == kWorkgroupSizeAttribute) { |
| return expect_paren_block("workgroup_size attribute", [&]() -> Result { |
| const ast::Expression* x = nullptr; |
| const ast::Expression* y = nullptr; |
| const ast::Expression* z = nullptr; |
| |
| auto expr = primary_expression(); |
| if (expr.errored) { |
| return Failure::kErrored; |
| } else if (!expr.matched) { |
| return add_error(peek(), "expected workgroup_size x parameter"); |
| } |
| x = std::move(expr.value); |
| |
| if (match(Token::Type::kComma)) { |
| if (!peek_is(Token::Type::kParenRight)) { |
| expr = primary_expression(); |
| if (expr.errored) { |
| return Failure::kErrored; |
| } else if (!expr.matched) { |
| return add_error(peek(), "expected workgroup_size y parameter"); |
| } |
| y = std::move(expr.value); |
| |
| if (match(Token::Type::kComma)) { |
| if (!peek_is(Token::Type::kParenRight)) { |
| expr = primary_expression(); |
| if (expr.errored) { |
| return Failure::kErrored; |
| } else if (!expr.matched) { |
| return add_error(peek(), "expected workgroup_size z parameter"); |
| } |
| z = std::move(expr.value); |
| |
| match(Token::Type::kComma); |
| } |
| } |
| } |
| } |
| |
| return create<ast::WorkgroupAttribute>(t.source(), x, y, z); |
| }); |
| } |
| |
| // TODO(crbug.com/tint/1503): Remove when deprecation period is over. |
| if (t == kStageAttribute) { |
| return expect_paren_block("stage attribute", [&]() -> Result { |
| auto stage = expect_pipeline_stage(); |
| if (stage.errored) { |
| return Failure::kErrored; |
| } |
| |
| std::string warning = "remove stage and use @"; |
| switch (stage.value) { |
| case ast::PipelineStage::kVertex: |
| warning += "vertex"; |
| break; |
| case ast::PipelineStage::kFragment: |
| warning += "fragment"; |
| break; |
| case ast::PipelineStage::kCompute: |
| warning += "compute"; |
| break; |
| case ast::PipelineStage::kNone: |
| break; |
| } |
| deprecated(t.source(), warning); |
| |
| return create<ast::StageAttribute>(t.source(), stage.value); |
| }); |
| } |
| if (t == kComputeStage) { |
| return create<ast::StageAttribute>(t.source(), ast::PipelineStage::kCompute); |
| } |
| if (t == kVertexStage) { |
| return create<ast::StageAttribute>(t.source(), ast::PipelineStage::kVertex); |
| } |
| if (t == kFragmentStage) { |
| return create<ast::StageAttribute>(t.source(), ast::PipelineStage::kFragment); |
| } |
| |
| if (t == kSizeAttribute) { |
| const char* use = "size attribute"; |
| return expect_paren_block(use, [&]() -> Result { |
| auto val = expect_positive_sint(use); |
| if (val.errored) { |
| return Failure::kErrored; |
| } |
| match(Token::Type::kComma); |
| |
| return create<ast::StructMemberSizeAttribute>(t.source(), val.value); |
| }); |
| } |
| |
| if (t == kAlignAttribute) { |
| const char* use = "align attribute"; |
| return expect_paren_block(use, [&]() -> Result { |
| auto val = expect_positive_sint(use); |
| if (val.errored) { |
| return Failure::kErrored; |
| } |
| match(Token::Type::kComma); |
| |
| return create<ast::StructMemberAlignAttribute>(t.source(), val.value); |
| }); |
| } |
| |
| if (t == kIdAttribute) { |
| const char* use = "id attribute"; |
| return expect_paren_block(use, [&]() -> Result { |
| auto val = expect_positive_sint(use); |
| if (val.errored) { |
| return Failure::kErrored; |
| } |
| match(Token::Type::kComma); |
| |
| return create<ast::IdAttribute>(t.source(), val.value); |
| }); |
| } |
| |
| return Failure::kNoMatch; |
| } |
| |
| bool ParserImpl::expect_attributes_consumed(utils::VectorRef<const ast::Attribute*> in) { |
| if (in.IsEmpty()) { |
| return true; |
| } |
| add_error(in[0]->source, "unexpected attributes"); |
| return false; |
| } |
| |
| bool ParserImpl::match(Token::Type tok, Source* source /*= nullptr*/) { |
| auto& t = peek(); |
| |
| if (source != nullptr) { |
| *source = t.source(); |
| } |
| |
| if (t.Is(tok)) { |
| next(); |
| return true; |
| } |
| return false; |
| } |
| |
| bool ParserImpl::expect(std::string_view use, Token::Type tok) { |
| auto& t = peek(); |
| if (t.Is(tok)) { |
| next(); |
| synchronized_ = true; |
| return true; |
| } |
| |
| // Special case to split `>>` and `>=` tokens if we are looking for a `>`. |
| if (tok == Token::Type::kGreaterThan && |
| (t.Is(Token::Type::kShiftRight) || t.Is(Token::Type::kGreaterThanEqual))) { |
| next(); |
| |
| // Push the second character to the token queue. |
| if (t.Is(Token::Type::kShiftRight)) { |
| split_token(Token::Type::kGreaterThan, Token::Type::kGreaterThan); |
| } else if (t.Is(Token::Type::kGreaterThanEqual)) { |
| split_token(Token::Type::kGreaterThan, Token::Type::kEqual); |
| } |
| |
| synchronized_ = true; |
| return true; |
| } |
| |
| // Error cases |
| synchronized_ = false; |
| if (handle_error(t)) { |
| return false; |
| } |
| |
| std::stringstream err; |
| err << "expected '" << Token::TypeToName(tok) << "'"; |
| if (!use.empty()) { |
| err << " for " << use; |
| } |
| add_error(t, err.str()); |
| return false; |
| } |
| |
| Expect<int32_t> ParserImpl::expect_sint(std::string_view use) { |
| auto& t = peek(); |
| if (!t.Is(Token::Type::kIntLiteral) && !t.Is(Token::Type::kIntLiteral_I)) { |
| return add_error(t.source(), "expected signed integer literal", use); |
| } |
| |
| int64_t val = t.to_i64(); |
| if ((val > std::numeric_limits<int32_t>::max()) || |
| (val < std::numeric_limits<int32_t>::min())) { |
| // TODO(crbug.com/tint/1504): Test this when abstract int is implemented |
| return add_error(t.source(), "value overflows i32", use); |
| } |
| |
| next(); |
| return {static_cast<int32_t>(t.to_i64()), t.source()}; |
| } |
| |
| Expect<uint32_t> ParserImpl::expect_positive_sint(std::string_view use) { |
| auto sint = expect_sint(use); |
| if (sint.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (sint.value < 0) { |
| return add_error(sint.source, std::string(use) + " must be positive"); |
| } |
| |
| return {static_cast<uint32_t>(sint.value), sint.source}; |
| } |
| |
| Expect<uint32_t> ParserImpl::expect_nonzero_positive_sint(std::string_view use) { |
| auto sint = expect_sint(use); |
| if (sint.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (sint.value <= 0) { |
| return add_error(sint.source, std::string(use) + " must be greater than 0"); |
| } |
| |
| return {static_cast<uint32_t>(sint.value), sint.source}; |
| } |
| |
| Expect<std::string> ParserImpl::expect_ident(std::string_view use) { |
| auto& t = peek(); |
| if (t.IsIdentifier()) { |
| synchronized_ = true; |
| next(); |
| |
| if (is_reserved(t)) { |
| return add_error(t.source(), "'" + t.to_str() + "' is a reserved keyword"); |
| } |
| |
| return {t.to_str(), t.source()}; |
| } |
| if (handle_error(t)) { |
| return Failure::kErrored; |
| } |
| synchronized_ = false; |
| return add_error(t.source(), "expected identifier", use); |
| } |
| |
| template <typename F, typename T> |
| T ParserImpl::expect_block(Token::Type start, Token::Type end, std::string_view use, F&& body) { |
| if (!expect(use, start)) { |
| return Failure::kErrored; |
| } |
| |
| return sync(end, [&]() -> T { |
| auto res = body(); |
| |
| if (res.errored) { |
| return Failure::kErrored; |
| } |
| |
| if (!expect(use, end)) { |
| return Failure::kErrored; |
| } |
| |
| return res; |
| }); |
| } |
| |
| template <typename F, typename T> |
| T ParserImpl::expect_paren_block(std::string_view use, F&& body) { |
| return expect_block(Token::Type::kParenLeft, Token::Type::kParenRight, use, |
| std::forward<F>(body)); |
| } |
| |
| template <typename F, typename T> |
| T ParserImpl::expect_brace_block(std::string_view use, F&& body) { |
| return expect_block(Token::Type::kBraceLeft, Token::Type::kBraceRight, use, |
| std::forward<F>(body)); |
| } |
| |
| template <typename F, typename T> |
| T ParserImpl::expect_lt_gt_block(std::string_view use, F&& body) { |
| return expect_block(Token::Type::kLessThan, Token::Type::kGreaterThan, use, |
| std::forward<F>(body)); |
| } |
| |
| template <typename F, typename T> |
| T ParserImpl::sync(Token::Type tok, F&& body) { |
| if (parse_depth_ >= kMaxParseDepth) { |
| // We've hit a maximum parser recursive depth. |
| // We can't call into body() as we might stack overflow. |
| // Instead, report an error... |
| add_error(peek(), "maximum parser recursive depth reached"); |
| // ...and try to resynchronize. If we cannot resynchronize to `tok` then |
| // synchronized_ is set to false, and the parser knows that forward progress |
| // is not being made. |
| sync_to(tok, /* consume: */ true); |
| return Failure::kErrored; |
| } |
| |
| sync_tokens_.push_back(tok); |
| |
| ++parse_depth_; |
| auto result = body(); |
| --parse_depth_; |
| |
| if (sync_tokens_.back() != tok) { |
| TINT_ICE(Reader, builder_.Diagnostics()) << "sync_tokens is out of sync"; |
| } |
| sync_tokens_.pop_back(); |
| |
| if (result.errored) { |
| sync_to(tok, /* consume: */ true); |
| } |
| |
| return result; |
| } |
| |
| bool ParserImpl::sync_to(Token::Type tok, bool consume) { |
| // Clear the synchronized state - gets set to true again on success. |
| synchronized_ = false; |
| |
| BlockCounters counters; |
| |
| for (size_t i = 0; i < kMaxResynchronizeLookahead; i++) { |
| auto& t = peek(i); |
| if (counters.consume(t) > 0) { |
| continue; // Nested block |
| } |
| if (!t.Is(tok) && !is_sync_token(t)) { |
| continue; // Not a synchronization point |
| } |
| |
| // Synchronization point found. |
| |
| // Skip any tokens we don't understand, bringing us to just before the |
| // resync point. |
| while (i-- > 0) { |
| next(); |
| } |
| |
| // Is this synchronization token |tok|? |
| if (t.Is(tok)) { |
| if (consume) { |
| next(); |
| } |
| synchronized_ = true; |
| return true; |
| } |
| break; |
| } |
| |
| return false; |
| } |
| |
| bool ParserImpl::is_sync_token(const Token& t) const { |
| for (auto r : sync_tokens_) { |
| if (t.Is(r)) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| bool ParserImpl::handle_error(const Token& t) { |
| // The token might itself be an error. |
| if (t.IsError()) { |
| synchronized_ = false; |
| add_error(t.source(), t.to_str()); |
| return true; |
| } |
| return false; |
| } |
| |
| template <typename F, typename T> |
| T ParserImpl::without_error(F&& body) { |
| silence_errors_++; |
| auto result = body(); |
| silence_errors_--; |
| return result; |
| } |
| |
| ParserImpl::MultiTokenSource ParserImpl::make_source_range() { |
| return MultiTokenSource(this); |
| } |
| |
| ParserImpl::MultiTokenSource ParserImpl::make_source_range_from(const Source& start) { |
| return MultiTokenSource(this, start); |
| } |
| |
| } // namespace tint::reader::wgsl |