mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Refactor RedundantAssignEliminator.
This commit is contained in:
parent
d7a802e4bf
commit
3622b30a1d
@ -295,6 +295,31 @@ decltype(auto) mapTuple(Callable&& _callable)
|
|||||||
return detail::MapTuple<Callable>{std::forward<Callable>(_callable)};
|
return detail::MapTuple<Callable>{std::forward<Callable>(_callable)};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Merges map @a _b into map @a _a. If the same key exists in both maps,
|
||||||
|
/// calls @a _conflictSolver to combine the two values.
|
||||||
|
template <class K, class V, class F>
|
||||||
|
void joinMap(std::map<K, V>& _a, std::map<K, V>&& _b, F _conflictSolver)
|
||||||
|
{
|
||||||
|
auto ita = _a.begin();
|
||||||
|
auto aend = _a.end();
|
||||||
|
auto itb = _b.begin();
|
||||||
|
auto bend = _b.end();
|
||||||
|
|
||||||
|
for (; itb != bend; ++ita)
|
||||||
|
{
|
||||||
|
if (ita == aend)
|
||||||
|
ita = _a.insert(ita, std::move(*itb++));
|
||||||
|
else if (ita->first < itb->first)
|
||||||
|
continue;
|
||||||
|
else if (itb->first < ita->first)
|
||||||
|
ita = _a.insert(ita, std::move(*itb++));
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_conflictSolver(ita->second, std::move(itb->second));
|
||||||
|
++itb;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// String conversion functions, mainly to/from hex/nibble/byte representations.
|
// String conversion functions, mainly to/from hex/nibble/byte representations.
|
||||||
|
|
||||||
|
@ -167,6 +167,8 @@ add_library(yul
|
|||||||
optimiser/ReasoningBasedSimplifier.h
|
optimiser/ReasoningBasedSimplifier.h
|
||||||
optimiser/RedundantAssignEliminator.cpp
|
optimiser/RedundantAssignEliminator.cpp
|
||||||
optimiser/RedundantAssignEliminator.h
|
optimiser/RedundantAssignEliminator.h
|
||||||
|
optimiser/RedundantStoreBase.cpp
|
||||||
|
optimiser/RedundantStoreBase.h
|
||||||
optimiser/Rematerialiser.cpp
|
optimiser/Rematerialiser.cpp
|
||||||
optimiser/Rematerialiser.h
|
optimiser/Rematerialiser.h
|
||||||
optimiser/SMTSolver.cpp
|
optimiser/SMTSolver.cpp
|
||||||
|
@ -38,7 +38,7 @@ void RedundantAssignEliminator::run(OptimiserStepContext& _context, Block& _ast)
|
|||||||
RedundantAssignEliminator rae{_context.dialect};
|
RedundantAssignEliminator rae{_context.dialect};
|
||||||
rae(_ast);
|
rae(_ast);
|
||||||
|
|
||||||
AssignmentRemover remover{rae.m_pendingRemovals};
|
StatementRemover remover{rae.m_pendingRemovals};
|
||||||
remover(_ast);
|
remover(_ast);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -49,7 +49,7 @@ void RedundantAssignEliminator::operator()(Identifier const& _identifier)
|
|||||||
|
|
||||||
void RedundantAssignEliminator::operator()(VariableDeclaration const& _variableDeclaration)
|
void RedundantAssignEliminator::operator()(VariableDeclaration const& _variableDeclaration)
|
||||||
{
|
{
|
||||||
ASTWalker::operator()(_variableDeclaration);
|
RedundantStoreBase::operator()(_variableDeclaration);
|
||||||
|
|
||||||
for (auto const& var: _variableDeclaration.variables)
|
for (auto const& var: _variableDeclaration.variables)
|
||||||
m_declaredVariables.emplace(var.name);
|
m_declaredVariables.emplace(var.name);
|
||||||
@ -60,151 +60,17 @@ void RedundantAssignEliminator::operator()(Assignment const& _assignment)
|
|||||||
visit(*_assignment.value);
|
visit(*_assignment.value);
|
||||||
for (auto const& var: _assignment.variableNames)
|
for (auto const& var: _assignment.variableNames)
|
||||||
changeUndecidedTo(var.name, State::Unused);
|
changeUndecidedTo(var.name, State::Unused);
|
||||||
|
|
||||||
if (_assignment.variableNames.size() == 1)
|
|
||||||
// Default-construct it in "Undecided" state if it does not yet exist.
|
|
||||||
m_assignments[_assignment.variableNames.front().name][&_assignment];
|
|
||||||
}
|
|
||||||
|
|
||||||
void RedundantAssignEliminator::operator()(If const& _if)
|
|
||||||
{
|
|
||||||
visit(*_if.condition);
|
|
||||||
|
|
||||||
TrackedAssignments skipBranch{m_assignments};
|
|
||||||
(*this)(_if.body);
|
|
||||||
|
|
||||||
merge(m_assignments, move(skipBranch));
|
|
||||||
}
|
|
||||||
|
|
||||||
void RedundantAssignEliminator::operator()(Switch const& _switch)
|
|
||||||
{
|
|
||||||
visit(*_switch.expression);
|
|
||||||
|
|
||||||
TrackedAssignments const preState{m_assignments};
|
|
||||||
|
|
||||||
bool hasDefault = false;
|
|
||||||
vector<TrackedAssignments> branches;
|
|
||||||
for (auto const& c: _switch.cases)
|
|
||||||
{
|
|
||||||
if (!c.value)
|
|
||||||
hasDefault = true;
|
|
||||||
(*this)(c.body);
|
|
||||||
branches.emplace_back(move(m_assignments));
|
|
||||||
m_assignments = preState;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (hasDefault)
|
|
||||||
{
|
|
||||||
m_assignments = move(branches.back());
|
|
||||||
branches.pop_back();
|
|
||||||
}
|
|
||||||
for (auto& branch: branches)
|
|
||||||
merge(m_assignments, move(branch));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void RedundantAssignEliminator::operator()(FunctionDefinition const& _functionDefinition)
|
void RedundantAssignEliminator::operator()(FunctionDefinition const& _functionDefinition)
|
||||||
{
|
{
|
||||||
std::set<YulString> outerDeclaredVariables;
|
ScopedSaveAndRestore outerDeclaredVariables(m_declaredVariables, {});
|
||||||
std::set<YulString> outerReturnVariables;
|
ScopedSaveAndRestore outerReturnVariables(m_returnVariables, {});
|
||||||
TrackedAssignments outerAssignments;
|
|
||||||
ForLoopInfo forLoopInfo;
|
|
||||||
swap(m_declaredVariables, outerDeclaredVariables);
|
|
||||||
swap(m_returnVariables, outerReturnVariables);
|
|
||||||
swap(m_assignments, outerAssignments);
|
|
||||||
swap(m_forLoopInfo, forLoopInfo);
|
|
||||||
|
|
||||||
for (auto const& retParam: _functionDefinition.returnVariables)
|
for (auto const& retParam: _functionDefinition.returnVariables)
|
||||||
m_returnVariables.insert(retParam.name);
|
m_returnVariables.insert(retParam.name);
|
||||||
|
|
||||||
(*this)(_functionDefinition.body);
|
RedundantStoreBase::operator()(_functionDefinition);
|
||||||
|
|
||||||
for (auto const& param: _functionDefinition.parameters)
|
|
||||||
finalize(param.name, State::Unused);
|
|
||||||
for (auto const& retParam: _functionDefinition.returnVariables)
|
|
||||||
finalize(retParam.name, State::Used);
|
|
||||||
|
|
||||||
swap(m_declaredVariables, outerDeclaredVariables);
|
|
||||||
swap(m_returnVariables, outerReturnVariables);
|
|
||||||
swap(m_assignments, outerAssignments);
|
|
||||||
swap(m_forLoopInfo, forLoopInfo);
|
|
||||||
}
|
|
||||||
|
|
||||||
void RedundantAssignEliminator::operator()(ForLoop const& _forLoop)
|
|
||||||
{
|
|
||||||
ForLoopInfo outerForLoopInfo;
|
|
||||||
swap(outerForLoopInfo, m_forLoopInfo);
|
|
||||||
++m_forLoopNestingDepth;
|
|
||||||
|
|
||||||
// If the pre block was not empty,
|
|
||||||
// we would have to deal with more complicated scoping rules.
|
|
||||||
assertThrow(_forLoop.pre.statements.empty(), OptimizerException, "");
|
|
||||||
|
|
||||||
// We just run the loop twice to account for the back edge.
|
|
||||||
// There need not be more runs because we only have three different states.
|
|
||||||
|
|
||||||
visit(*_forLoop.condition);
|
|
||||||
|
|
||||||
TrackedAssignments zeroRuns{m_assignments};
|
|
||||||
|
|
||||||
(*this)(_forLoop.body);
|
|
||||||
merge(m_assignments, move(m_forLoopInfo.pendingContinueStmts));
|
|
||||||
m_forLoopInfo.pendingContinueStmts = {};
|
|
||||||
(*this)(_forLoop.post);
|
|
||||||
|
|
||||||
visit(*_forLoop.condition);
|
|
||||||
|
|
||||||
if (m_forLoopNestingDepth < 6)
|
|
||||||
{
|
|
||||||
// Do the second run only for small nesting depths to avoid horrible runtime.
|
|
||||||
TrackedAssignments oneRun{m_assignments};
|
|
||||||
|
|
||||||
(*this)(_forLoop.body);
|
|
||||||
|
|
||||||
merge(m_assignments, move(m_forLoopInfo.pendingContinueStmts));
|
|
||||||
m_forLoopInfo.pendingContinueStmts.clear();
|
|
||||||
(*this)(_forLoop.post);
|
|
||||||
|
|
||||||
visit(*_forLoop.condition);
|
|
||||||
// Order of merging does not matter because "max" is commutative and associative.
|
|
||||||
merge(m_assignments, move(oneRun));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Shortcut to avoid horrible runtime:
|
|
||||||
// Change all assignments that were newly introduced in the for loop to "used".
|
|
||||||
// We do not have to do that with the "break" or "continue" paths, because
|
|
||||||
// they will be joined later anyway.
|
|
||||||
// TODO parallel traversal might be more efficient here.
|
|
||||||
for (auto& var: m_assignments)
|
|
||||||
for (auto& assignment: var.second)
|
|
||||||
{
|
|
||||||
auto zeroIt = zeroRuns.find(var.first);
|
|
||||||
if (zeroIt != zeroRuns.end() && zeroIt->second.count(assignment.first))
|
|
||||||
continue;
|
|
||||||
assignment.second = State::Value::Used;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Order of merging does not matter because "max" is commutative and associative.
|
|
||||||
merge(m_assignments, move(zeroRuns));
|
|
||||||
merge(m_assignments, move(m_forLoopInfo.pendingBreakStmts));
|
|
||||||
m_forLoopInfo.pendingBreakStmts.clear();
|
|
||||||
|
|
||||||
// Restore potential outer for-loop states.
|
|
||||||
swap(m_forLoopInfo, outerForLoopInfo);
|
|
||||||
--m_forLoopNestingDepth;
|
|
||||||
}
|
|
||||||
|
|
||||||
void RedundantAssignEliminator::operator()(Break const&)
|
|
||||||
{
|
|
||||||
m_forLoopInfo.pendingBreakStmts.emplace_back(move(m_assignments));
|
|
||||||
m_assignments.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
void RedundantAssignEliminator::operator()(Continue const&)
|
|
||||||
{
|
|
||||||
m_forLoopInfo.pendingContinueStmts.emplace_back(move(m_assignments));
|
|
||||||
m_assignments.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void RedundantAssignEliminator::operator()(Leave const&)
|
void RedundantAssignEliminator::operator()(Leave const&)
|
||||||
@ -215,101 +81,76 @@ void RedundantAssignEliminator::operator()(Leave const&)
|
|||||||
|
|
||||||
void RedundantAssignEliminator::operator()(Block const& _block)
|
void RedundantAssignEliminator::operator()(Block const& _block)
|
||||||
{
|
{
|
||||||
set<YulString> outerDeclaredVariables;
|
ScopedSaveAndRestore outerDeclaredVariables(m_declaredVariables, {});
|
||||||
swap(m_declaredVariables, outerDeclaredVariables);
|
|
||||||
|
|
||||||
ASTWalker::operator()(_block);
|
RedundantStoreBase::operator()(_block);
|
||||||
|
|
||||||
for (auto const& var: m_declaredVariables)
|
for (auto const& var: m_declaredVariables)
|
||||||
finalize(var, State::Unused);
|
finalize(var, State::Unused);
|
||||||
|
|
||||||
swap(m_declaredVariables, outerDeclaredVariables);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void RedundantAssignEliminator::visit(Statement const& _statement)
|
||||||
template <class K, class V, class F>
|
|
||||||
void joinMap(std::map<K, V>& _a, std::map<K, V>&& _b, F _conflictSolver)
|
|
||||||
{
|
{
|
||||||
// TODO Perhaps it is better to just create a sorted list
|
RedundantStoreBase::visit(_statement);
|
||||||
// and then use insert(begin, end)
|
|
||||||
|
|
||||||
auto ita = _a.begin();
|
if (auto const* assignment = get_if<Assignment>(&_statement))
|
||||||
auto aend = _a.end();
|
if (assignment->variableNames.size() == 1)
|
||||||
auto itb = _b.begin();
|
// Default-construct it in "Undecided" state if it does not yet exist.
|
||||||
auto bend = _b.end();
|
m_stores[assignment->variableNames.front().name][&_statement];
|
||||||
|
}
|
||||||
|
|
||||||
for (; itb != bend; ++ita)
|
void RedundantAssignEliminator::shortcutNestedLoop(TrackedStores const& _zeroRuns)
|
||||||
{
|
{
|
||||||
if (ita == aend)
|
// Shortcut to avoid horrible runtime:
|
||||||
ita = _a.insert(ita, std::move(*itb++));
|
// Change all assignments that were newly introduced in the for loop to "used".
|
||||||
else if (ita->first < itb->first)
|
// We do not have to do that with the "break" or "continue" paths, because
|
||||||
continue;
|
// they will be joined later anyway.
|
||||||
else if (itb->first < ita->first)
|
// TODO parallel traversal might be more efficient here.
|
||||||
ita = _a.insert(ita, std::move(*itb++));
|
for (auto& [variable, stores]: m_stores)
|
||||||
else
|
for (auto& assignment: stores)
|
||||||
{
|
{
|
||||||
_conflictSolver(ita->second, std::move(itb->second));
|
auto zeroIt = _zeroRuns.find(variable);
|
||||||
++itb;
|
if (zeroIt != _zeroRuns.end() && zeroIt->second.count(assignment.first))
|
||||||
|
continue;
|
||||||
|
assignment.second = State::Value::Used;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void RedundantAssignEliminator::merge(TrackedAssignments& _target, TrackedAssignments&& _other)
|
void RedundantAssignEliminator::finalizeFunctionDefinition(FunctionDefinition const& _functionDefinition)
|
||||||
{
|
{
|
||||||
joinMap(_target, move(_other), [](
|
for (auto const& param: _functionDefinition.parameters)
|
||||||
map<Assignment const*, State>& _assignmentHere,
|
finalize(param.name, State::Unused);
|
||||||
map<Assignment const*, State>&& _assignmentThere
|
for (auto const& retParam: _functionDefinition.returnVariables)
|
||||||
)
|
finalize(retParam.name, State::Used);
|
||||||
{
|
|
||||||
return joinMap(_assignmentHere, move(_assignmentThere), State::join);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
void RedundantAssignEliminator::merge(TrackedAssignments& _target, vector<TrackedAssignments>&& _source)
|
|
||||||
{
|
|
||||||
for (TrackedAssignments& ts: _source)
|
|
||||||
merge(_target, move(ts));
|
|
||||||
_source.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void RedundantAssignEliminator::changeUndecidedTo(YulString _variable, RedundantAssignEliminator::State _newState)
|
void RedundantAssignEliminator::changeUndecidedTo(YulString _variable, RedundantAssignEliminator::State _newState)
|
||||||
{
|
{
|
||||||
for (auto& assignment: m_assignments[_variable])
|
for (auto& assignment: m_stores[_variable])
|
||||||
if (assignment.second == State::Undecided)
|
if (assignment.second == State::Undecided)
|
||||||
assignment.second = _newState;
|
assignment.second = _newState;
|
||||||
}
|
}
|
||||||
|
|
||||||
void RedundantAssignEliminator::finalize(YulString _variable, RedundantAssignEliminator::State _finalState)
|
void RedundantAssignEliminator::finalize(YulString _variable, RedundantAssignEliminator::State _finalState)
|
||||||
{
|
{
|
||||||
std::map<Assignment const*, State> assignments;
|
std::map<Statement const*, State> stores = std::move(m_stores[_variable]);
|
||||||
joinMap(assignments, std::move(m_assignments[_variable]), State::join);
|
m_stores.erase(_variable);
|
||||||
m_assignments.erase(_variable);
|
|
||||||
|
|
||||||
for (auto& breakAssignments: m_forLoopInfo.pendingBreakStmts)
|
for (auto& breakAssignments: m_forLoopInfo.pendingBreakStmts)
|
||||||
{
|
{
|
||||||
joinMap(assignments, std::move(breakAssignments[_variable]), State::join);
|
util::joinMap(stores, std::move(breakAssignments[_variable]), State::join);
|
||||||
breakAssignments.erase(_variable);
|
breakAssignments.erase(_variable);
|
||||||
}
|
}
|
||||||
for (auto& continueAssignments: m_forLoopInfo.pendingContinueStmts)
|
for (auto& continueAssignments: m_forLoopInfo.pendingContinueStmts)
|
||||||
{
|
{
|
||||||
joinMap(assignments, std::move(continueAssignments[_variable]), State::join);
|
util::joinMap(stores, std::move(continueAssignments[_variable]), State::join);
|
||||||
continueAssignments.erase(_variable);
|
continueAssignments.erase(_variable);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto const& assignment: assignments)
|
for (auto&& [statement, state]: stores)
|
||||||
{
|
if (
|
||||||
State const state = assignment.second == State::Undecided ? _finalState : assignment.second;
|
(state == State::Unused || (state == State::Undecided && _finalState == State::Unused)) &&
|
||||||
|
SideEffectsCollector{m_dialect, *std::get<Assignment>(*statement).value}.movable()
|
||||||
if (state == State::Unused && SideEffectsCollector{*m_dialect, *assignment.first->value}.movable())
|
)
|
||||||
m_pendingRemovals.insert(assignment.first);
|
m_pendingRemovals.insert(statement);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void AssignmentRemover::operator()(Block& _block)
|
|
||||||
{
|
|
||||||
ranges::actions::remove_if(_block.statements, [&](Statement const& _statement) -> bool {
|
|
||||||
return holds_alternative<Assignment>(_statement) && m_toRemove.count(&std::get<Assignment>(_statement));
|
|
||||||
});
|
|
||||||
|
|
||||||
ASTModifier::operator()(_block);
|
|
||||||
}
|
}
|
||||||
|
@ -25,6 +25,7 @@
|
|||||||
#include <libyul/ASTForward.h>
|
#include <libyul/ASTForward.h>
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
#include <libyul/optimiser/OptimiserStep.h>
|
#include <libyul/optimiser/OptimiserStep.h>
|
||||||
|
#include <libyul/optimiser/RedundantStoreBase.h>
|
||||||
|
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
@ -106,91 +107,37 @@ struct Dialect;
|
|||||||
*
|
*
|
||||||
* Prerequisite: Disambiguator, ForLoopInitRewriter.
|
* Prerequisite: Disambiguator, ForLoopInitRewriter.
|
||||||
*/
|
*/
|
||||||
class RedundantAssignEliminator: public ASTWalker
|
class RedundantAssignEliminator: public RedundantStoreBase
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
static constexpr char const* name{"RedundantAssignEliminator"};
|
static constexpr char const* name{"RedundantAssignEliminator"};
|
||||||
static void run(OptimiserStepContext&, Block& _ast);
|
static void run(OptimiserStepContext&, Block& _ast);
|
||||||
|
|
||||||
explicit RedundantAssignEliminator(Dialect const& _dialect): m_dialect(&_dialect) {}
|
explicit RedundantAssignEliminator(Dialect const& _dialect): RedundantStoreBase(_dialect) {}
|
||||||
RedundantAssignEliminator() = delete;
|
|
||||||
RedundantAssignEliminator(RedundantAssignEliminator const&) = delete;
|
|
||||||
RedundantAssignEliminator& operator=(RedundantAssignEliminator const&) = delete;
|
|
||||||
RedundantAssignEliminator(RedundantAssignEliminator&&) = default;
|
|
||||||
RedundantAssignEliminator& operator=(RedundantAssignEliminator&&) = default;
|
|
||||||
|
|
||||||
void operator()(Identifier const& _identifier) override;
|
void operator()(Identifier const& _identifier) override;
|
||||||
void operator()(VariableDeclaration const& _variableDeclaration) override;
|
void operator()(VariableDeclaration const& _variableDeclaration) override;
|
||||||
void operator()(Assignment const& _assignment) override;
|
void operator()(Assignment const& _assignment) override;
|
||||||
void operator()(If const& _if) override;
|
|
||||||
void operator()(Switch const& _switch) override;
|
|
||||||
void operator()(FunctionDefinition const&) override;
|
void operator()(FunctionDefinition const&) override;
|
||||||
void operator()(ForLoop const&) override;
|
|
||||||
void operator()(Break const&) override;
|
|
||||||
void operator()(Continue const&) override;
|
|
||||||
void operator()(Leave const&) override;
|
void operator()(Leave const&) override;
|
||||||
void operator()(Block const& _block) override;
|
void operator()(Block const& _block) override;
|
||||||
|
|
||||||
|
using RedundantStoreBase::visit;
|
||||||
|
void visit(Statement const& _statement) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
class State
|
void shortcutNestedLoop(TrackedStores const& _beforeLoop) override;
|
||||||
{
|
void finalizeFunctionDefinition(FunctionDefinition const& _functionDefinition) override;
|
||||||
public:
|
|
||||||
enum Value { Unused, Undecided, Used };
|
|
||||||
State(Value _value = Undecided): m_value(_value) {}
|
|
||||||
inline bool operator==(State _other) const { return m_value == _other.m_value; }
|
|
||||||
inline bool operator!=(State _other) const { return !operator==(_other); }
|
|
||||||
static inline void join(State& _a, State const& _b)
|
|
||||||
{
|
|
||||||
// Using "max" works here because of the order of the values in the enum.
|
|
||||||
_a.m_value = Value(std::max(int(_a.m_value), int(_b.m_value)));
|
|
||||||
}
|
|
||||||
private:
|
|
||||||
Value m_value = Undecided;
|
|
||||||
};
|
|
||||||
|
|
||||||
// TODO check that this does not cause nondeterminism!
|
|
||||||
// This could also be a pseudo-map from state to assignment.
|
|
||||||
using TrackedAssignments = std::map<YulString, std::map<Assignment const*, State>>;
|
|
||||||
|
|
||||||
/// Joins the assignment mapping of @a _source into @a _target according to the rules laid out
|
|
||||||
/// above.
|
|
||||||
/// Will destroy @a _source.
|
|
||||||
static void merge(TrackedAssignments& _target, TrackedAssignments&& _source);
|
|
||||||
static void merge(TrackedAssignments& _target, std::vector<TrackedAssignments>&& _source);
|
|
||||||
void changeUndecidedTo(YulString _variable, State _newState);
|
void changeUndecidedTo(YulString _variable, State _newState);
|
||||||
/// Called when a variable goes out of scope. Sets the state of all still undecided
|
/// Called when a variable goes out of scope. Sets the state of all still undecided
|
||||||
/// assignments to the final state. In this case, this also applies to pending
|
/// assignments to the final state. In this case, this also applies to pending
|
||||||
/// break and continue TrackedAssignments.
|
/// break and continue TrackedStores.
|
||||||
void finalize(YulString _variable, State _finalState);
|
void finalize(YulString _variable, State _finalState);
|
||||||
|
|
||||||
Dialect const* m_dialect;
|
|
||||||
std::set<YulString> m_declaredVariables;
|
std::set<YulString> m_declaredVariables;
|
||||||
std::set<YulString> m_returnVariables;
|
std::set<YulString> m_returnVariables;
|
||||||
std::set<Assignment const*> m_pendingRemovals;
|
|
||||||
TrackedAssignments m_assignments;
|
|
||||||
|
|
||||||
/// Working data for traversing for-loops.
|
|
||||||
struct ForLoopInfo
|
|
||||||
{
|
|
||||||
/// Tracked assignment states for each break statement.
|
|
||||||
std::vector<TrackedAssignments> pendingBreakStmts;
|
|
||||||
/// Tracked assignment states for each continue statement.
|
|
||||||
std::vector<TrackedAssignments> pendingContinueStmts;
|
|
||||||
};
|
|
||||||
ForLoopInfo m_forLoopInfo;
|
|
||||||
size_t m_forLoopNestingDepth = 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
class AssignmentRemover: public ASTModifier
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
explicit AssignmentRemover(std::set<Assignment const*> const& _toRemove):
|
|
||||||
m_toRemove(_toRemove)
|
|
||||||
{}
|
|
||||||
void operator()(Block& _block) override;
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::set<Assignment const*> const& m_toRemove;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
166
libyul/optimiser/RedundantStoreBase.cpp
Normal file
166
libyul/optimiser/RedundantStoreBase.cpp
Normal file
@ -0,0 +1,166 @@
|
|||||||
|
/*
|
||||||
|
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 <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
// SPDX-License-Identifier: GPL-3.0
|
||||||
|
/**
|
||||||
|
* Base class for both RedundantAssignEliminator and RedundantStoreEliminator.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <libyul/optimiser/RedundantStoreBase.h>
|
||||||
|
|
||||||
|
#include <libyul/optimiser/Semantics.h>
|
||||||
|
#include <libyul/optimiser/OptimiserStep.h>
|
||||||
|
#include <libyul/AST.h>
|
||||||
|
|
||||||
|
#include <libsolutil/CommonData.h>
|
||||||
|
|
||||||
|
#include <range/v3/action/remove_if.hpp>
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
using namespace solidity;
|
||||||
|
using namespace solidity::yul;
|
||||||
|
|
||||||
|
void RedundantStoreBase::operator()(If const& _if)
|
||||||
|
{
|
||||||
|
visit(*_if.condition);
|
||||||
|
|
||||||
|
TrackedStores skipBranch{m_stores};
|
||||||
|
(*this)(_if.body);
|
||||||
|
|
||||||
|
merge(m_stores, move(skipBranch));
|
||||||
|
}
|
||||||
|
|
||||||
|
void RedundantStoreBase::operator()(Switch const& _switch)
|
||||||
|
{
|
||||||
|
visit(*_switch.expression);
|
||||||
|
|
||||||
|
TrackedStores const preState{m_stores};
|
||||||
|
|
||||||
|
bool hasDefault = false;
|
||||||
|
vector<TrackedStores> branches;
|
||||||
|
for (auto const& c: _switch.cases)
|
||||||
|
{
|
||||||
|
if (!c.value)
|
||||||
|
hasDefault = true;
|
||||||
|
(*this)(c.body);
|
||||||
|
branches.emplace_back(move(m_stores));
|
||||||
|
m_stores = preState;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hasDefault)
|
||||||
|
{
|
||||||
|
m_stores = move(branches.back());
|
||||||
|
branches.pop_back();
|
||||||
|
}
|
||||||
|
for (auto& branch: branches)
|
||||||
|
merge(m_stores, move(branch));
|
||||||
|
}
|
||||||
|
|
||||||
|
void RedundantStoreBase::operator()(FunctionDefinition const& _functionDefinition)
|
||||||
|
{
|
||||||
|
ScopedSaveAndRestore outerAssignments(m_stores, {});
|
||||||
|
ScopedSaveAndRestore forLoopInfo(m_forLoopInfo, {});
|
||||||
|
|
||||||
|
(*this)(_functionDefinition.body);
|
||||||
|
|
||||||
|
finalizeFunctionDefinition(_functionDefinition);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RedundantStoreBase::operator()(ForLoop const& _forLoop)
|
||||||
|
{
|
||||||
|
ScopedSaveAndRestore outerForLoopInfo(m_forLoopInfo, {});
|
||||||
|
ScopedSaveAndRestore forLoopNestingDepth(m_forLoopNestingDepth, m_forLoopNestingDepth + 1);
|
||||||
|
|
||||||
|
// If the pre block was not empty,
|
||||||
|
// we would have to deal with more complicated scoping rules.
|
||||||
|
assertThrow(_forLoop.pre.statements.empty(), OptimizerException, "");
|
||||||
|
|
||||||
|
// We just run the loop twice to account for the back edge.
|
||||||
|
// There need not be more runs because we only have three different states.
|
||||||
|
|
||||||
|
visit(*_forLoop.condition);
|
||||||
|
|
||||||
|
TrackedStores zeroRuns{m_stores};
|
||||||
|
|
||||||
|
(*this)(_forLoop.body);
|
||||||
|
merge(m_stores, move(m_forLoopInfo.pendingContinueStmts));
|
||||||
|
m_forLoopInfo.pendingContinueStmts = {};
|
||||||
|
(*this)(_forLoop.post);
|
||||||
|
|
||||||
|
visit(*_forLoop.condition);
|
||||||
|
|
||||||
|
if (m_forLoopNestingDepth < 6)
|
||||||
|
{
|
||||||
|
// Do the second run only for small nesting depths to avoid horrible runtime.
|
||||||
|
TrackedStores oneRun{m_stores};
|
||||||
|
|
||||||
|
(*this)(_forLoop.body);
|
||||||
|
|
||||||
|
merge(m_stores, move(m_forLoopInfo.pendingContinueStmts));
|
||||||
|
m_forLoopInfo.pendingContinueStmts.clear();
|
||||||
|
(*this)(_forLoop.post);
|
||||||
|
|
||||||
|
visit(*_forLoop.condition);
|
||||||
|
// Order of merging does not matter because "max" is commutative and associative.
|
||||||
|
merge(m_stores, move(oneRun));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
// Shortcut to avoid horrible runtime.
|
||||||
|
shortcutNestedLoop(zeroRuns);
|
||||||
|
|
||||||
|
// Order of merging does not matter because "max" is commutative and associative.
|
||||||
|
merge(m_stores, move(zeroRuns));
|
||||||
|
merge(m_stores, move(m_forLoopInfo.pendingBreakStmts));
|
||||||
|
m_forLoopInfo.pendingBreakStmts.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
void RedundantStoreBase::operator()(Break const&)
|
||||||
|
{
|
||||||
|
m_forLoopInfo.pendingBreakStmts.emplace_back(move(m_stores));
|
||||||
|
m_stores.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
void RedundantStoreBase::operator()(Continue const&)
|
||||||
|
{
|
||||||
|
m_forLoopInfo.pendingContinueStmts.emplace_back(move(m_stores));
|
||||||
|
m_stores.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
void RedundantStoreBase::merge(TrackedStores& _target, TrackedStores&& _other)
|
||||||
|
{
|
||||||
|
util::joinMap(_target, move(_other), [](
|
||||||
|
map<Statement const*, State>& _assignmentHere,
|
||||||
|
map<Statement const*, State>&& _assignmentThere
|
||||||
|
)
|
||||||
|
{
|
||||||
|
return util::joinMap(_assignmentHere, move(_assignmentThere), State::join);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void RedundantStoreBase::merge(TrackedStores& _target, vector<TrackedStores>&& _source)
|
||||||
|
{
|
||||||
|
for (TrackedStores& ts: _source)
|
||||||
|
merge(_target, move(ts));
|
||||||
|
_source.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
void StatementRemover::operator()(Block& _block)
|
||||||
|
{
|
||||||
|
ranges::actions::remove_if(_block.statements, [&](Statement const& _statement) -> bool {
|
||||||
|
return m_toRemove.count(&_statement);
|
||||||
|
});
|
||||||
|
ASTModifier::operator()(_block);
|
||||||
|
}
|
113
libyul/optimiser/RedundantStoreBase.h
Normal file
113
libyul/optimiser/RedundantStoreBase.h
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
/*
|
||||||
|
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 <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
// SPDX-License-Identifier: GPL-3.0
|
||||||
|
/**
|
||||||
|
* Base class for both RedundantAssignEliminator and RedundantStoreEliminator.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
#include <libyul/AST.h>
|
||||||
|
|
||||||
|
#include <range/v3/action/remove_if.hpp>
|
||||||
|
|
||||||
|
#include <variant>
|
||||||
|
|
||||||
|
|
||||||
|
namespace solidity::yul
|
||||||
|
{
|
||||||
|
struct Dialect;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Base class for both RedundantAssignEliminator and RedundantStoreEliminator.
|
||||||
|
*
|
||||||
|
* Prerequisite: Disambiguator, ForLoopInitRewriter.
|
||||||
|
*/
|
||||||
|
class RedundantStoreBase: public ASTWalker
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
explicit RedundantStoreBase(Dialect const& _dialect): m_dialect(_dialect) {}
|
||||||
|
|
||||||
|
using ASTWalker::operator();
|
||||||
|
void operator()(If const& _if) override;
|
||||||
|
void operator()(Switch const& _switch) override;
|
||||||
|
void operator()(FunctionDefinition const&) override;
|
||||||
|
void operator()(ForLoop const&) override;
|
||||||
|
void operator()(Break const&) override;
|
||||||
|
void operator()(Continue const&) override;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
class State
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum Value { Unused, Undecided, Used };
|
||||||
|
State(Value _value = Undecided): m_value(_value) {}
|
||||||
|
inline bool operator==(State _other) const { return m_value == _other.m_value; }
|
||||||
|
inline bool operator!=(State _other) const { return !operator==(_other); }
|
||||||
|
static inline void join(State& _a, State const& _b)
|
||||||
|
{
|
||||||
|
// Using "max" works here because of the order of the values in the enum.
|
||||||
|
_a.m_value = Value(std::max(int(_a.m_value), int(_b.m_value)));
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
Value m_value = Undecided;
|
||||||
|
};
|
||||||
|
|
||||||
|
using TrackedStores = std::map<YulString, std::map<Statement const*, State>>;
|
||||||
|
|
||||||
|
/// This function is called for a loop that is nested too deep to avoid
|
||||||
|
/// horrible runtime and should just resolve the situation in a pragmatic
|
||||||
|
/// and correct manner.
|
||||||
|
virtual void shortcutNestedLoop(TrackedStores const& _beforeLoop) = 0;
|
||||||
|
|
||||||
|
/// This function is called right before the scoped restore of the function definition.
|
||||||
|
virtual void finalizeFunctionDefinition(FunctionDefinition const& /*_functionDefinition*/) {}
|
||||||
|
|
||||||
|
/// Joins the assignment mapping of @a _source into @a _target according to the rules laid out
|
||||||
|
/// above.
|
||||||
|
/// Will destroy @a _source.
|
||||||
|
static void merge(TrackedStores& _target, TrackedStores&& _source);
|
||||||
|
static void merge(TrackedStores& _target, std::vector<TrackedStores>&& _source);
|
||||||
|
|
||||||
|
Dialect const& m_dialect;
|
||||||
|
std::set<Statement const*> m_pendingRemovals;
|
||||||
|
TrackedStores m_stores;
|
||||||
|
|
||||||
|
/// Working data for traversing for-loops.
|
||||||
|
struct ForLoopInfo
|
||||||
|
{
|
||||||
|
/// Tracked assignment states for each break statement.
|
||||||
|
std::vector<TrackedStores> pendingBreakStmts;
|
||||||
|
/// Tracked assignment states for each continue statement.
|
||||||
|
std::vector<TrackedStores> pendingContinueStmts;
|
||||||
|
};
|
||||||
|
ForLoopInfo m_forLoopInfo;
|
||||||
|
size_t m_forLoopNestingDepth = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class StatementRemover: public ASTModifier
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
explicit StatementRemover(std::set<Statement const*> const& _toRemove): m_toRemove(_toRemove) {}
|
||||||
|
|
||||||
|
void operator()(Block& _block) override;
|
||||||
|
private:
|
||||||
|
std::set<Statement const*> const& m_toRemove;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user