/* 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 /** * Optimiser component that turns complex expressions into multiple variable * declarations. */ #include #include #include #include #include #include #include #include using namespace std; using namespace solidity; using namespace solidity::yul; using namespace solidity::util; using namespace solidity::langutil; void ExpressionSplitter::run(OptimiserStepContext& _context, Block& _ast) { TypeInfo typeInfo(_context.dialect, _ast); ExpressionSplitter{_context.dialect, _context.dispenser, typeInfo}(_ast); } void ExpressionSplitter::operator()(FunctionCall& _funCall) { BuiltinFunction const* builtin = m_dialect.builtin(_funCall.functionName.name); for (size_t i = _funCall.arguments.size(); i > 0; i--) if (!builtin || !builtin->literalArgument(i - 1)) outlineExpression(_funCall.arguments[i - 1]); } void ExpressionSplitter::operator()(If& _if) { outlineExpression(*_if.condition); (*this)(_if.body); } void ExpressionSplitter::operator()(Switch& _switch) { outlineExpression(*_switch.expression); for (auto& _case: _switch.cases) // Do not visit the case expression, nothing to split there. (*this)(_case.body); } void ExpressionSplitter::operator()(ForLoop& _loop) { (*this)(_loop.pre); // Do not visit the condition because we cannot split expressions there. (*this)(_loop.post); (*this)(_loop.body); } void ExpressionSplitter::operator()(Block& _block) { vector saved; swap(saved, m_statementsToPrefix); function>(Statement&)> f = [&](Statement& _statement) -> std::optional> { m_statementsToPrefix.clear(); visit(_statement); if (m_statementsToPrefix.empty()) return {}; m_statementsToPrefix.emplace_back(std::move(_statement)); return std::move(m_statementsToPrefix); }; iterateReplacing(_block.statements, f); swap(saved, m_statementsToPrefix); } void ExpressionSplitter::outlineExpression(Expression& _expr) { if (holds_alternative(_expr)) return; // Do not split function calls with a zero (0) literal argument // in case the PUSH0 opcode is available if (auto&& evmDialect = dynamic_cast(&m_dialect)) if ( evmDialect->evmVersion().hasPush0() && holds_alternative(_expr) && valueOfLiteral(get(_expr)) == 0 ) return; visit(_expr); shared_ptr debugData = debugDataOf(_expr); YulString var = m_nameDispenser.newName({}); YulString type = m_typeInfo.typeOf(_expr); m_statementsToPrefix.emplace_back(VariableDeclaration{ debugData, {{TypedName{debugData, var, type}}}, make_unique(std::move(_expr)) }); _expr = Identifier{debugData, var}; m_typeInfo.setVariableType(var, type); }