/*( 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 . */ /** * Optimisation stage that aggressively rematerializes certain variables ina a function to free * space on the stack until it is compilable. */ #include #include #include #include #include #include #include #include using namespace std; using namespace solidity; using namespace solidity::yul; namespace { /** * Class that discovers all variables that can be fully eliminated by rematerialization, * and the corresponding approximate costs. */ class RematCandidateSelector: public DataFlowAnalyzer { public: explicit RematCandidateSelector(Dialect const& _dialect): DataFlowAnalyzer(_dialect) {} /// @returns a set of tuples of rematerialisation costs, variable to rematerialise /// and variables that occur in its expression. /// Note that this set is sorted by cost. set>> candidates() { set>> cand; for (auto const& codeCost: m_expressionCodeCost) { size_t numRef = m_numReferences[codeCost.first]; cand.emplace(make_tuple(codeCost.second * numRef, codeCost.first, m_references[codeCost.first])); } return cand; } using DataFlowAnalyzer::operator(); void operator()(VariableDeclaration& _varDecl) override { DataFlowAnalyzer::operator()(_varDecl); if (_varDecl.variables.size() == 1) { YulString varName = _varDecl.variables.front().name; if (m_value.count(varName)) m_expressionCodeCost[varName] = CodeCost::codeCost(m_dialect, *m_value[varName].value); } } void operator()(Assignment& _assignment) override { for (auto const& var: _assignment.variableNames) rematImpossible(var.name); DataFlowAnalyzer::operator()(_assignment); } // We use visit(Expression) because operator()(Identifier) would also // get called on left-hand-sides of assignments. void visit(Expression& _e) override { if (holds_alternative(_e)) { YulString name = std::get(_e).name; if (m_expressionCodeCost.count(name)) { if (!m_value.count(name)) rematImpossible(name); else ++m_numReferences[name]; } } DataFlowAnalyzer::visit(_e); } /// Remove the variable from the candidate set. void rematImpossible(YulString _variable) { m_numReferences.erase(_variable); m_expressionCodeCost.erase(_variable); } /// Candidate variables and the code cost of their value. map m_expressionCodeCost; /// Number of references to each candidate variable. map m_numReferences; }; template void eliminateVariables( Dialect const& _dialect, ASTNode& _node, size_t _numVariables, bool _allowMSizeOptimization ) { RematCandidateSelector selector{_dialect}; selector(_node); // Select at most _numVariables set varsToEliminate; for (auto const& costs: selector.candidates()) { if (varsToEliminate.size() >= _numVariables) break; // If a variable we would like to eliminate references another one // we already selected for elimination, then stop selecting // candidates. If we would add that variable, then the cost calculation // for the previous variable would be off. Furthermore, we // do not skip the variable because it would be better to properly re-compute // the costs of all other variables instead. bool referencesVarToEliminate = false; for (YulString const& referencedVar: get<2>(costs)) if (varsToEliminate.count(referencedVar)) { referencesVarToEliminate = true; break; } if (referencesVarToEliminate) break; varsToEliminate.insert(get<1>(costs)); } Rematerialiser::run(_dialect, _node, std::move(varsToEliminate)); UnusedPruner::runUntilStabilised(_dialect, _node, _allowMSizeOptimization); } } bool StackCompressor::run( Dialect const& _dialect, Object& _object, bool _optimizeStackAllocation, size_t _maxIterations ) { yulAssert( _object.code && _object.code->statements.size() > 0 && holds_alternative(_object.code->statements.at(0)), "Need to run the function grouper before the stack compressor." ); bool allowMSizeOptimzation = !MSizeFinder::containsMSize(_dialect, *_object.code); for (size_t iterations = 0; iterations < _maxIterations; iterations++) { map stackSurplus = CompilabilityChecker(_dialect, _object, _optimizeStackAllocation).stackDeficit; if (stackSurplus.empty()) return true; if (stackSurplus.count(YulString{})) { yulAssert(stackSurplus.at({}) > 0, "Invalid surplus value."); eliminateVariables( _dialect, std::get(_object.code->statements.at(0)), static_cast(stackSurplus.at({})), allowMSizeOptimzation ); } for (size_t i = 1; i < _object.code->statements.size(); ++i) { auto& fun = std::get(_object.code->statements[i]); if (!stackSurplus.count(fun.name)) continue; yulAssert(stackSurplus.at(fun.name) > 0, "Invalid surplus value."); eliminateVariables( _dialect, fun, static_cast(stackSurplus.at(fun.name)), allowMSizeOptimzation ); } } return false; }