/* This file is part of solidity. solidity is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. solidity is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with solidity. If not, see . */ // SPDX-License-Identifier: GPL-3.0 /** * @author Christian * @date 2015 * Object containing the type and other annotations for the AST nodes. */ #pragma once #include #include #include #include #include #include #include #include #include namespace solidity::yul { struct AsmAnalysisInfo; struct Identifier; struct Dialect; } namespace solidity::frontend { class Type; class ArrayType; struct CallGraph; struct ASTAnnotation { ASTAnnotation() = default; ASTAnnotation(ASTAnnotation const&) = delete; ASTAnnotation(ASTAnnotation&&) = delete; ASTAnnotation& operator=(ASTAnnotation const&) = delete; ASTAnnotation& operator=(ASTAnnotation&&) = delete; virtual ~ASTAnnotation() = default; }; struct DocTag { std::string content; ///< The text content of the tag. std::string paramName; ///< Only used for @param, stores the parameter name. }; struct StructurallyDocumentedAnnotation { StructurallyDocumentedAnnotation() = default; StructurallyDocumentedAnnotation(StructurallyDocumentedAnnotation const&) = delete; StructurallyDocumentedAnnotation(StructurallyDocumentedAnnotation&&) = delete; StructurallyDocumentedAnnotation& operator=(StructurallyDocumentedAnnotation const&) = delete; StructurallyDocumentedAnnotation& operator=(StructurallyDocumentedAnnotation&&) = delete; virtual ~StructurallyDocumentedAnnotation() = default; /// Mapping docstring tag name -> content. std::multimap docTags; /// contract that @inheritdoc references if it exists ContractDefinition const* inheritdocReference = nullptr; }; struct SourceUnitAnnotation: ASTAnnotation { /// The "absolute" (in the compiler sense) path of this source unit. util::SetOnce path; /// The exported symbols (all global symbols). util::SetOnce>> exportedSymbols; /// Experimental features. std::set experimentalFeatures; /// Using the new ABI coder. Set to `false` if using ABI coder v1. util::SetOnce useABICoderV2; }; struct ScopableAnnotation { ScopableAnnotation() = default; ScopableAnnotation(ScopableAnnotation const&) = delete; ScopableAnnotation(ScopableAnnotation&&) = delete; ScopableAnnotation& operator=(ScopableAnnotation const&) = delete; ScopableAnnotation& operator=(ScopableAnnotation&&) = delete; virtual ~ScopableAnnotation() = default; /// The scope this declaration resides in. Can be nullptr if it is the global scope. /// Filled by the Scoper. ASTNode const* scope = nullptr; /// Pointer to the contract this declaration resides in. Can be nullptr if the current scope /// is not part of a contract. Filled by the Scoper. ContractDefinition const* contract = nullptr; }; struct DeclarationAnnotation: ASTAnnotation, ScopableAnnotation { }; struct ImportAnnotation: DeclarationAnnotation { /// The absolute path of the source unit to import. util::SetOnce absolutePath; /// The actual source unit. SourceUnit const* sourceUnit = nullptr; }; struct TypeDeclarationAnnotation: DeclarationAnnotation { /// The name of this type, prefixed by proper namespaces if globally accessible. util::SetOnce canonicalName; }; struct StructDeclarationAnnotation: TypeDeclarationAnnotation { /// Whether the struct is recursive, i.e. if the struct (recursively) contains a member that involves a struct of the same /// type, either in a dynamic array, as member of another struct or inside a mapping. /// Only cases in which the recursive occurrence is within a dynamic array or a mapping are valid, while direct /// recursion immediately raises an error. /// Will be filled in by the DeclarationTypeChecker. std::optional recursive; /// Whether the struct contains a mapping type, either directly or, indirectly inside another /// struct or an array. std::optional containsNestedMapping; }; struct ContractDefinitionAnnotation: TypeDeclarationAnnotation, StructurallyDocumentedAnnotation { /// List of functions and modifiers without a body. Can also contain functions from base classes. std::optional> unimplementedDeclarations; /// List of all (direct and indirect) base contracts in order from derived to /// base, including the contract itself. std::vector linearizedBaseContracts; /// Mapping containing the nodes that define the arguments for base constructors. /// These can either be inheritance specifiers or modifier invocations. std::map baseConstructorArguments; /// A graph with edges representing calls between functions that may happen during contract construction. util::SetOnce> creationCallGraph; /// A graph with edges representing calls between functions that may happen in a deployed contract. util::SetOnce> deployedCallGraph; /// List of contracts whose bytecode is referenced by this contract, e.g. through "new". /// The Value represents the ast node that referenced the contract. std::map> contractDependencies; }; struct CallableDeclarationAnnotation: DeclarationAnnotation { /// The set of functions/modifiers/events this callable overrides. std::set baseFunctions; }; struct FunctionDefinitionAnnotation: CallableDeclarationAnnotation, StructurallyDocumentedAnnotation { }; struct EventDefinitionAnnotation: CallableDeclarationAnnotation, StructurallyDocumentedAnnotation { }; struct ErrorDefinitionAnnotation: CallableDeclarationAnnotation, StructurallyDocumentedAnnotation { }; struct ModifierDefinitionAnnotation: CallableDeclarationAnnotation, StructurallyDocumentedAnnotation { }; struct VariableDeclarationAnnotation: DeclarationAnnotation, StructurallyDocumentedAnnotation { /// Type of variable (type of identifier referencing this variable). Type const* type = nullptr; /// The set of functions this (public state) variable overrides. std::set baseFunctions; }; struct StatementAnnotation: ASTAnnotation { }; struct InlineAssemblyAnnotation: StatementAnnotation { struct ExternalIdentifierInfo { Declaration const* declaration = nullptr; /// Suffix used, one of "slot", "offset", "length", "address", "selector" or empty. std::string suffix; size_t valueSize = size_t(-1); }; /// Mapping containing resolved references to external identifiers and their value size std::map externalReferences; /// Information generated during analysis phase. std::shared_ptr analysisInfo; /// True, if the assembly block was annotated to be memory-safe. bool markedMemorySafe = false; /// True, if the assembly block involves any memory opcode or assigns to variables in memory. util::SetOnce hasMemoryEffects; }; struct BlockAnnotation: StatementAnnotation, ScopableAnnotation { }; struct TryCatchClauseAnnotation: ASTAnnotation, ScopableAnnotation { }; struct ForStatementAnnotation: StatementAnnotation, ScopableAnnotation { }; struct ReturnAnnotation: StatementAnnotation { /// Reference to the return parameters of the function. ParameterList const* functionReturnParameters = nullptr; }; struct TypeNameAnnotation: ASTAnnotation { /// Type declared by this type name, i.e. type of a variable where this type name is used. /// Set during reference resolution stage. Type const* type = nullptr; }; struct IdentifierPathAnnotation: ASTAnnotation { /// Referenced declaration, set during reference resolution stage. Declaration const* referencedDeclaration = nullptr; /// What kind of lookup needs to be done (static, virtual, super) find the declaration. util::SetOnce requiredLookup; /// Declaration of each path element. std::vector pathDeclarations; }; struct ExpressionAnnotation: ASTAnnotation { /// Inferred type of the expression. Type const* type = nullptr; /// Whether the expression is a constant variable util::SetOnce isConstant; /// Whether the expression is pure, i.e. compile-time constant. util::SetOnce isPure; /// Whether it is an LValue (i.e. something that can be assigned to). util::SetOnce isLValue; /// Whether the expression is used in a context where the LValue is actually required. bool willBeWrittenTo = false; /// Whether the expression is an lvalue that is only assigned. /// Would be false for --, ++, delete, +=, -=, .... /// Only relevant if isLvalue == true bool lValueOfOrdinaryAssignment = false; /// Types and - if given - names of arguments if the expr. is a function /// that is called, used for overload resolution std::optional arguments; /// True if the expression consists solely of the name of the function and the function is called immediately /// instead of being stored or processed. The name may be qualified with the name of a contract, library /// module, etc., that clarifies the scope. For example: `m.L.f()`, where `m` is a module, `L` is a library /// and `f` is a function is a direct call. This means that the function to be called is known at compilation /// time and it's not necessary to rely on any runtime dispatch mechanism to resolve it. /// Note that even the simplest expressions, like `(f)()`, result in an indirect call even if they consist of /// values known at compilation time. bool calledDirectly = false; }; struct IdentifierAnnotation: ExpressionAnnotation { /// Referenced declaration, set at latest during overload resolution stage. Declaration const* referencedDeclaration = nullptr; /// What kind of lookup needs to be done (static, virtual, super) find the declaration. util::SetOnce requiredLookup; /// List of possible declarations it could refer to (can contain duplicates). std::vector candidateDeclarations; /// List of possible declarations it could refer to. std::vector overloadedDeclarations; }; struct MemberAccessAnnotation: ExpressionAnnotation { /// Referenced declaration, set at latest during overload resolution stage. Declaration const* referencedDeclaration = nullptr; /// What kind of lookup needs to be done (static, virtual, super) find the declaration. util::SetOnce requiredLookup; }; struct OperationAnnotation: ExpressionAnnotation { util::SetOnce userDefinedFunction; }; struct BinaryOperationAnnotation: OperationAnnotation { /// The common type that is used for the operation, not necessarily the result type (which /// e.g. for comparisons is bool). Type const* commonType = nullptr; }; enum class FunctionCallKind { FunctionCall, TypeConversion, StructConstructorCall }; struct FunctionCallAnnotation: ExpressionAnnotation { util::SetOnce kind; /// If true, this is the external call of a try statement. bool tryCall = false; }; }