/* 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 * Evaluator for types of constant expressions. */ #pragma once #include #include #include namespace solidity::langutil { class ErrorReporter; } namespace solidity::frontend { class TypeChecker; /** * Small drop-in replacement for TypeChecker to evaluate simple expressions of integer constants. */ class ConstantEvaluator: private ASTConstVisitor { public: struct TypedValue { TypePointer type; rational value; }; using EvaluationMap = std::map; ConstantEvaluator(langutil::ErrorReporter& _errorReporter, EvaluationMap& _evaluations): m_errorReporter{ _errorReporter }, m_evaluations{ _evaluations }, m_depth{ 0 } { } static std::optional evaluate( langutil::ErrorReporter& _errorReporter, Expression const& _expr ); std::optional evaluate(Expression const& _expr); private: void endVisit(BinaryOperation const& _operation) override; void endVisit(UnaryOperation const& _operation) override; void endVisit(Literal const& _literal) override; void endVisit(Identifier const& _identifier) override; void endVisit(TupleExpression const& _tuple) override; TypePointer type(ASTNode const& _node); std::optional value(ASTNode const& _node); /// @return typed evaluation result or std::nullopt if not evaluated yet. std::optional result(ASTNode const& _node); /// Conditionally sets the evaluation result for the given ASTNode @p _node. void setResult(ASTNode const& _node, std::optional _result); void setValue(ASTNode const& _node, rational const& _value) { setResult(_node, TypedValue{nullptr, _value}); } /// @returns boolean indicating whether or not given ASTNode @p _node has been evaluated already or not. bool evaluated(ASTNode const& _node) const noexcept; langutil::ErrorReporter& m_errorReporter; EvaluationMap& m_evaluations; /// Current recursion depth. size_t m_depth = 0; }; }