/* 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 . */ #pragma once #include #include #include #include // because of instruction #include #include namespace solidity::smtutil { class SolverInterface; class Expression; struct Sort; } namespace solidity::yul { /** * Reasoning-based simplifier. * This optimizer uses SMT solvers to check whether `if` conditions are constant. * - If `constraints AND condition` is UNSAT, the condition is never true and the whole body can be removed. * - If `constraints AND NOT condition` is UNSAT, the condition is always true and can be replaced by `1`. * The simplifications above can only be applied if the condition is movable. * * It is only effective on the EVM dialect, but safe to use on other dialects. * * Prerequisite: Disambiguator, SSATransform. */ class ReasoningBasedSimplifier: public ASTModifier { public: static constexpr char const* name{"ReasoningBasedSimplifier"}; static void run(OptimiserStepContext& _context, Block& _ast); using ASTModifier::operator(); void operator()(VariableDeclaration& _varDecl) override; void operator()(If& _if) override; void operator()(ForLoop& _for) override; void operator()(FunctionDefinition& _function) override; private: explicit ReasoningBasedSimplifier( Dialect const& _dialect, std::set const& _ssaVariables ); smtutil::Expression encodeExpression( Expression const& _expression ); void handleDeclaration( YulString _varName, evmasm::Instruction _instruction, std::vector const& _arguments ); smtutil::Expression newRestrictedVariable(std::string const& _name = {}, bool _boolean = false); std::string uniqueName(); bool makesInfeasible(smtutil::Expression _constraint); bool feasible(); bool infeasible(); YulString localVariableFromExpression(std::string const& _expressionName); bool isBoolean(Expression const& _expression) const; std::shared_ptr defaultSort() const; smtutil::Expression literalValue(Literal const& _literal) const; Dialect const& m_dialect; std::set const& m_ssaVariables; std::unique_ptr m_solver; std::map m_variables; std::set m_booleanVariables; size_t m_varCounter = 0; }; }