/*( 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 #include using namespace std; using namespace dev; using namespace 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.forward[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]); } } 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 (_e.type() == typeid(Identifier)) { YulString name = boost::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 && _object.code->statements.at(0).type() == typeid(Block), "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::run(_dialect, _object, _optimizeStackAllocation); if (stackSurplus.empty()) return true; if (stackSurplus.count(YulString{})) { yulAssert(stackSurplus.at({}) > 0, "Invalid surplus value."); eliminateVariables( _dialect, boost::get(_object.code->statements.at(0)), stackSurplus.at({}), allowMSizeOptimzation ); } for (size_t i = 1; i < _object.code->statements.size(); ++i) { FunctionDefinition& fun = boost::get(_object.code->statements[i]); if (!stackSurplus.count(fun.name)) continue; yulAssert(stackSurplus.at(fun.name) > 0, "Invalid surplus value."); eliminateVariables( _dialect, fun, stackSurplus.at(fun.name), allowMSizeOptimzation ); } } return false; }