/* 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 . */ /** * @author Federico Bond * @date 2016 * Static analyzer and checker. */ #pragma once #include #include #include #include #include namespace solidity::langutil { class ErrorReporter; } namespace solidity::frontend { class ConstructorUsesAssembly; /** * The module that performs static analysis on the AST. * In this context, static analysis is anything that can produce warnings which can help * programmers write cleaner code. For every warning generated here, it has to be possible to write * equivalent code that does not generate the warning. */ class StaticAnalyzer: private ASTConstVisitor { public: /// @param _errorReporter provides the error logging functionality. explicit StaticAnalyzer(langutil::ErrorReporter& _errorReporter); ~StaticAnalyzer() override; /// Performs static analysis on the given source unit and all of its sub-nodes. /// @returns true iff all checks passed. Note even if all checks passed, errors() can still contain warnings bool analyze(SourceUnit const& _sourceUnit); private: bool visit(ContractDefinition const& _contract) override; void endVisit(ContractDefinition const& _contract) override; bool visit(FunctionDefinition const& _function) override; void endVisit(FunctionDefinition const& _function) override; bool visit(ExpressionStatement const& _statement) override; bool visit(VariableDeclaration const& _variable) override; bool visit(Identifier const& _identifier) override; bool visit(Return const& _return) override; bool visit(MemberAccess const& _memberAccess) override; bool visit(InlineAssembly const& _inlineAssembly) override; bool visit(BinaryOperation const& _operation) override; bool visit(FunctionCall const& _functionCall) override; struct TypeComp { bool operator()(Type const* lhs, Type const* rhs) const { solAssert(lhs && rhs, ""); return lhs->richIdentifier() < rhs->richIdentifier(); } }; using TypeSet = std::set; /// @returns the size of this type in storage, including all sub-types. static bigint structureSizeEstimate( Type const& _type, std::set& _structsSeen, TypeSet& _oversizedSubTypes ); langutil::ErrorReporter& m_errorReporter; /// Flag that indicates whether the current contract definition is a library. bool m_library = false; /// Number of uses of each (named) local variable in a function, counter is initialized with zero. /// Pairs of AST ids and pointers are used as keys to ensure a deterministic order /// when traversing. std::map, int> m_localVarUseCount; /// Cache that holds information about whether a contract's constructor /// uses inline assembly. std::unique_ptr m_constructorUsesAssembly; FunctionDefinition const* m_currentFunction = nullptr; /// Flag that indicates a constructor. bool m_constructor = false; /// Current contract. ContractDefinition const* m_currentContract = nullptr; }; }