// 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. #ifndef SRC_RESOLVER_RESOLVER_H_ #define SRC_RESOLVER_RESOLVER_H_ #include #include #include #include #include "src/intrinsic_table.h" #include "src/program_builder.h" #include "src/scope_stack.h" #include "src/utils/unique_vector.h" namespace tint { // Forward declarations namespace ast { class ArrayAccessorExpression; class BinaryExpression; class BitcastExpression; class CallExpression; class CaseStatement; class ConstructorExpression; class Function; class IdentifierExpression; class MemberAccessorExpression; class UnaryOpExpression; class Variable; } // namespace ast namespace semantic { class Array; class Statement; } // namespace semantic namespace type { class Struct; } // namespace type namespace resolver { /// Resolves types for all items in the given tint program class Resolver { public: /// Constructor /// @param builder the program builder explicit Resolver(ProgramBuilder* builder); /// Destructor ~Resolver(); /// @returns error messages from the resolver std::string error() const { return diagnostics_.str(); } /// @returns true if the resolver was successful bool Resolve(); /// @param type the given type /// @returns true if the given type is storable. static bool IsStorable(type::Type* type); private: /// Structure holding semantic information about a variable. /// Used to build the semantic::Variable nodes at the end of resolving. struct VariableInfo { explicit VariableInfo(ast::Variable* decl); ~VariableInfo(); ast::Variable* const declaration; ast::StorageClass storage_class; std::vector users; }; /// Structure holding semantic information about a function. /// Used to build the semantic::Function nodes at the end of resolving. struct FunctionInfo { explicit FunctionInfo(ast::Function* decl); ~FunctionInfo(); ast::Function* const declaration; UniqueVector referenced_module_vars; UniqueVector local_referenced_module_vars; // List of transitive calls this function makes UniqueVector transitive_calls; }; /// Structure holding semantic information about an expression. /// Used to build the semantic::Expression nodes at the end of resolving. struct ExpressionInfo { type::Type* type; semantic::Statement* statement; }; /// Structure holding semantic information about a call expression to an /// ast::Function. /// Used to build the semantic::Call nodes at the end of resolving. struct FunctionCallInfo { FunctionInfo* function; semantic::Statement* statement; }; /// Structure holding semantic information about a block (i.e. scope), such as /// parent block and variables declared in the block. /// Used to validate variable scoping rules. struct BlockInfo { enum class Type { kGeneric, kLoop, kLoopContinuing, kSwitchCase }; BlockInfo(Type type, BlockInfo* parent); ~BlockInfo(); template BlockInfo* FindFirstParent(Pred&& pred) { BlockInfo* curr = this; while (curr && !pred(curr)) { curr = curr->parent; } return curr; } BlockInfo* FindFirstParent(BlockInfo::Type ty) { return FindFirstParent( [ty](auto* block_info) { return block_info->type == ty; }); } Type const type; BlockInfo* const parent; std::vector decls; // first_continue is set to the index of the first variable in decls // declared after the first continue statement in a loop block, if any. constexpr static size_t kNoContinue = size_t(~0); size_t first_continue = kNoContinue; }; /// Resolves the program, without creating final the semantic nodes. /// @returns true on success, false on error bool ResolveInternal(); /// Creates the nodes and adds them to the semantic::Info mappings of the /// ProgramBuilder. void CreateSemanticNodes() const; /// Retrieves information for the requested import. /// @param src the source of the import /// @param path the import path /// @param name the method name to get information on /// @param params the parameters to the method call /// @param id out parameter for the external call ID. Must not be a nullptr. /// @returns the return type of `name` in `path` or nullptr on error. type::Type* GetImportData(const Source& src, const std::string& path, const std::string& name, const ast::ExpressionList& params, uint32_t* id); void set_referenced_from_function_if_needed(VariableInfo* var, bool local); // AST and Type traversal methods // Each return true on success, false on failure. bool ArrayAccessor(ast::ArrayAccessorExpression*); bool Binary(ast::BinaryExpression*); bool Bitcast(ast::BitcastExpression*); bool BlockStatement(const ast::BlockStatement*); bool Call(ast::CallExpression*); bool CaseStatement(ast::CaseStatement*); bool Constructor(ast::ConstructorExpression*); bool Expression(ast::Expression*); bool Expressions(const ast::ExpressionList&); bool Function(ast::Function*); bool Functions(const ast::FunctionList&); bool Identifier(ast::IdentifierExpression*); bool IfStatement(ast::IfStatement*); bool IntrinsicCall(ast::CallExpression*, semantic::IntrinsicType); bool MemberAccessor(ast::MemberAccessorExpression*); bool Statement(ast::Statement*); bool Statements(const ast::StatementList&); bool UnaryOp(ast::UnaryOpExpression*); bool VariableDeclStatement(const ast::VariableDeclStatement*); bool VariableStorageClass(ast::Statement*); /// @returns the semantic information for the array `arr`, building it if it /// hasn't been constructed already. If an error is raised, nullptr is /// returned. const semantic::Array* Array(type::Array*); /// @returns the semantic information for the structure `str`, building it if /// it hasn't been constructed already. If an error is raised, nullptr is /// returned. const semantic::Struct* Structure(type::Struct* str); /// @param align the output default alignment in bytes for the type `ty` /// @param size the output default size in bytes for the type `ty` /// @returns true on success, false on error bool DefaultAlignAndSize(type::Type* ty, uint32_t& align, uint32_t& size); VariableInfo* CreateVariableInfo(ast::Variable*); /// @returns the resolved type of the ast::Expression `expr` /// @param expr the expression type::Type* TypeOf(ast::Expression* expr); /// Creates a semantic::Expression node with the resolved type `type`, and /// assigns this semantic node to the expression `expr`. /// @param expr the expression /// @param type the resolved type void SetType(ast::Expression* expr, type::Type* type); /// Constructs a new BlockInfo with the given type and with #current_block_ as /// its parent, assigns this to #current_block_, and then calls `callback`. /// The original #current_block_ is restored on exit. template bool BlockScope(BlockInfo::Type type, F&& callback); ProgramBuilder* const builder_; std::unique_ptr const intrinsic_table_; diag::List diagnostics_; BlockInfo* current_block_ = nullptr; ScopeStack variable_stack_; std::unordered_map symbol_to_function_; std::unordered_map function_to_info_; std::unordered_map variable_to_info_; std::unordered_map function_calls_; std::unordered_map expr_info_; FunctionInfo* current_function_ = nullptr; semantic::Statement* current_statement_ = nullptr; BlockAllocator variable_infos_; BlockAllocator function_infos_; }; } // namespace resolver } // namespace tint #endif // SRC_RESOLVER_RESOLVER_H_