mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #8267 from ghallak/prune-unnecessary-funcs
[Yul] Prune functions that call each other but are otherwise unreferenced
This commit is contained in:
commit
0e100e7e7e
@ -8,6 +8,7 @@ Compiler Features:
|
|||||||
* Code Generator: Use ``calldatacopy`` instead of ``codecopy`` to zero out memory past input.
|
* Code Generator: Use ``calldatacopy`` instead of ``codecopy`` to zero out memory past input.
|
||||||
* AST: Add a new node for doxygen-style, structured documentation that can be received by contract, function, event and modifier definitions.
|
* AST: Add a new node for doxygen-style, structured documentation that can be received by contract, function, event and modifier definitions.
|
||||||
* Debug: Provide reason strings for compiler-generated internal reverts when using the ``--revert-strings`` option or the ``settings.debug.revertStrings`` setting on ``debug`` mode.
|
* Debug: Provide reason strings for compiler-generated internal reverts when using the ``--revert-strings`` option or the ``settings.debug.revertStrings`` setting on ``debug`` mode.
|
||||||
|
* Yul Optimizer: Prune functions that call each other but are otherwise unreferenced.
|
||||||
|
|
||||||
|
|
||||||
Bugfixes:
|
Bugfixes:
|
||||||
|
@ -69,6 +69,8 @@ add_library(yul
|
|||||||
optimiser/BlockHasher.h
|
optimiser/BlockHasher.h
|
||||||
optimiser/CallGraphGenerator.cpp
|
optimiser/CallGraphGenerator.cpp
|
||||||
optimiser/CallGraphGenerator.h
|
optimiser/CallGraphGenerator.h
|
||||||
|
optimiser/CircularReferencesPruner.cpp
|
||||||
|
optimiser/CircularReferencesPruner.h
|
||||||
optimiser/CommonSubexpressionEliminator.cpp
|
optimiser/CommonSubexpressionEliminator.cpp
|
||||||
optimiser/CommonSubexpressionEliminator.h
|
optimiser/CommonSubexpressionEliminator.h
|
||||||
optimiser/ConditionalSimplifier.cpp
|
optimiser/ConditionalSimplifier.cpp
|
||||||
|
61
libyul/optimiser/CircularReferencesPruner.cpp
Normal file
61
libyul/optimiser/CircularReferencesPruner.cpp
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
/*
|
||||||
|
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/>.
|
||||||
|
*/
|
||||||
|
#include <libyul/optimiser/CircularReferencesPruner.h>
|
||||||
|
|
||||||
|
#include <libyul/optimiser/CallGraphGenerator.h>
|
||||||
|
#include <libyul/optimiser/OptimizerUtilities.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
|
#include <libsolutil/Algorithms.h>
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
using namespace solidity::yul;
|
||||||
|
|
||||||
|
void CircularReferencesPruner::run(OptimiserStepContext& _context, Block& _ast)
|
||||||
|
{
|
||||||
|
CircularReferencesPruner{_context.reservedIdentifiers}(_ast);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CircularReferencesPruner::operator()(Block& _block)
|
||||||
|
{
|
||||||
|
set<YulString> functionsToKeep =
|
||||||
|
functionsCalledFromOutermostContext(CallGraphGenerator::callGraph(_block));
|
||||||
|
|
||||||
|
for (auto&& statement: _block.statements)
|
||||||
|
if (holds_alternative<FunctionDefinition>(statement))
|
||||||
|
{
|
||||||
|
FunctionDefinition const& funDef = std::get<FunctionDefinition>(statement);
|
||||||
|
if (!functionsToKeep.count(funDef.name))
|
||||||
|
statement = Block{};
|
||||||
|
}
|
||||||
|
|
||||||
|
removeEmptyBlocks(_block);
|
||||||
|
}
|
||||||
|
|
||||||
|
set<YulString> CircularReferencesPruner::functionsCalledFromOutermostContext(CallGraph const& _callGraph)
|
||||||
|
{
|
||||||
|
set<YulString> verticesToTraverse = m_reservedIdentifiers;
|
||||||
|
verticesToTraverse.insert(YulString(""));
|
||||||
|
|
||||||
|
return util::BreadthFirstSearch<YulString>{{verticesToTraverse.begin(), verticesToTraverse.end()}}.run(
|
||||||
|
[&_callGraph](YulString _function, auto&& _addChild) {
|
||||||
|
if (_callGraph.functionCalls.count(_function))
|
||||||
|
for (auto const& callee: _callGraph.functionCalls.at(_function))
|
||||||
|
if (_callGraph.functionCalls.count(callee))
|
||||||
|
_addChild(callee);
|
||||||
|
}).visited;
|
||||||
|
}
|
58
libyul/optimiser/CircularReferencesPruner.h
Normal file
58
libyul/optimiser/CircularReferencesPruner.h
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
/*
|
||||||
|
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/>.
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* Optimization stage that removes functions that call each other but are
|
||||||
|
* otherwise unreferenced.
|
||||||
|
*
|
||||||
|
* Prerequisites: Disambiguator, FunctionHoister.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
#include <libyul/optimiser/CallGraphGenerator.h>
|
||||||
|
#include <libyul/optimiser/OptimiserStep.h>
|
||||||
|
|
||||||
|
namespace solidity::yul
|
||||||
|
{
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Optimization stage that removes functions that call each other but are
|
||||||
|
* neither externally referenced nor referenced from the outermost context.
|
||||||
|
*/
|
||||||
|
class CircularReferencesPruner: public ASTModifier
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
static constexpr char const* name{"CircularReferencesPruner"};
|
||||||
|
static void run(OptimiserStepContext& _context, Block& _ast);
|
||||||
|
|
||||||
|
using ASTModifier::operator();
|
||||||
|
void operator()(Block& _block) override;
|
||||||
|
private:
|
||||||
|
CircularReferencesPruner(std::set<YulString> const& _reservedIdentifiers):
|
||||||
|
m_reservedIdentifiers(_reservedIdentifiers)
|
||||||
|
{}
|
||||||
|
|
||||||
|
/// Run a breadth-first search starting from the outermost context and
|
||||||
|
/// externally referenced functions to find all the functions that are
|
||||||
|
/// called from there either directly or indirectly.
|
||||||
|
std::set<YulString> functionsCalledFromOutermostContext(CallGraph const& _callGraph);
|
||||||
|
|
||||||
|
std::set<YulString> const& m_reservedIdentifiers;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
@ -24,6 +24,7 @@
|
|||||||
#include <libyul/optimiser/VarDeclInitializer.h>
|
#include <libyul/optimiser/VarDeclInitializer.h>
|
||||||
#include <libyul/optimiser/BlockFlattener.h>
|
#include <libyul/optimiser/BlockFlattener.h>
|
||||||
#include <libyul/optimiser/CallGraphGenerator.h>
|
#include <libyul/optimiser/CallGraphGenerator.h>
|
||||||
|
#include <libyul/optimiser/CircularReferencesPruner.h>
|
||||||
#include <libyul/optimiser/ControlFlowSimplifier.h>
|
#include <libyul/optimiser/ControlFlowSimplifier.h>
|
||||||
#include <libyul/optimiser/ConditionalSimplifier.h>
|
#include <libyul/optimiser/ConditionalSimplifier.h>
|
||||||
#include <libyul/optimiser/ConditionalUnsimplifier.h>
|
#include <libyul/optimiser/ConditionalUnsimplifier.h>
|
||||||
@ -99,6 +100,7 @@ void OptimiserSuite::run(
|
|||||||
FunctionGrouper::name,
|
FunctionGrouper::name,
|
||||||
EquivalentFunctionCombiner::name,
|
EquivalentFunctionCombiner::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
BlockFlattener::name,
|
BlockFlattener::name,
|
||||||
ControlFlowSimplifier::name,
|
ControlFlowSimplifier::name,
|
||||||
LiteralRematerialiser::name,
|
LiteralRematerialiser::name,
|
||||||
@ -151,7 +153,8 @@ void OptimiserSuite::run(
|
|||||||
BlockFlattener::name,
|
BlockFlattener::name,
|
||||||
DeadCodeEliminator::name,
|
DeadCodeEliminator::name,
|
||||||
ForLoopConditionIntoBody::name,
|
ForLoopConditionIntoBody::name,
|
||||||
UnusedPruner::name
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name
|
||||||
}, ast);
|
}, ast);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -161,6 +164,7 @@ void OptimiserSuite::run(
|
|||||||
LoadResolver::name,
|
LoadResolver::name,
|
||||||
CommonSubexpressionEliminator::name,
|
CommonSubexpressionEliminator::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
}, ast);
|
}, ast);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -170,6 +174,7 @@ void OptimiserSuite::run(
|
|||||||
SSAReverser::name,
|
SSAReverser::name,
|
||||||
CommonSubexpressionEliminator::name,
|
CommonSubexpressionEliminator::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
|
|
||||||
ExpressionJoiner::name,
|
ExpressionJoiner::name,
|
||||||
ExpressionJoiner::name,
|
ExpressionJoiner::name,
|
||||||
@ -183,6 +188,7 @@ void OptimiserSuite::run(
|
|||||||
suite.runSequence({
|
suite.runSequence({
|
||||||
ExpressionInliner::name,
|
ExpressionInliner::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
}, ast);
|
}, ast);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -193,8 +199,10 @@ void OptimiserSuite::run(
|
|||||||
SSATransform::name,
|
SSATransform::name,
|
||||||
RedundantAssignEliminator::name,
|
RedundantAssignEliminator::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
RedundantAssignEliminator::name,
|
RedundantAssignEliminator::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
}, ast);
|
}, ast);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -244,6 +252,7 @@ void OptimiserSuite::run(
|
|||||||
RedundantAssignEliminator::name,
|
RedundantAssignEliminator::name,
|
||||||
ForLoopConditionIntoBody::name,
|
ForLoopConditionIntoBody::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
CommonSubexpressionEliminator::name,
|
CommonSubexpressionEliminator::name,
|
||||||
}, ast);
|
}, ast);
|
||||||
}
|
}
|
||||||
@ -255,10 +264,13 @@ void OptimiserSuite::run(
|
|||||||
ExpressionJoiner::name,
|
ExpressionJoiner::name,
|
||||||
Rematerialiser::name,
|
Rematerialiser::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
ExpressionJoiner::name,
|
ExpressionJoiner::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
ExpressionJoiner::name,
|
ExpressionJoiner::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
|
|
||||||
SSAReverser::name,
|
SSAReverser::name,
|
||||||
CommonSubexpressionEliminator::name,
|
CommonSubexpressionEliminator::name,
|
||||||
@ -266,10 +278,12 @@ void OptimiserSuite::run(
|
|||||||
ForLoopConditionOutOfBody::name,
|
ForLoopConditionOutOfBody::name,
|
||||||
CommonSubexpressionEliminator::name,
|
CommonSubexpressionEliminator::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
|
|
||||||
ExpressionJoiner::name,
|
ExpressionJoiner::name,
|
||||||
Rematerialiser::name,
|
Rematerialiser::name,
|
||||||
UnusedPruner::name,
|
UnusedPruner::name,
|
||||||
|
CircularReferencesPruner::name,
|
||||||
}, ast);
|
}, ast);
|
||||||
|
|
||||||
// This is a tuning parameter, but actually just prevents infinite loops.
|
// This is a tuning parameter, but actually just prevents infinite loops.
|
||||||
@ -339,6 +353,7 @@ map<string, unique_ptr<OptimiserStep>> const& OptimiserSuite::allSteps()
|
|||||||
if (instance.empty())
|
if (instance.empty())
|
||||||
instance = optimiserStepCollection<
|
instance = optimiserStepCollection<
|
||||||
BlockFlattener,
|
BlockFlattener,
|
||||||
|
CircularReferencesPruner,
|
||||||
CommonSubexpressionEliminator,
|
CommonSubexpressionEliminator,
|
||||||
ConditionalSimplifier,
|
ConditionalSimplifier,
|
||||||
ConditionalUnsimplifier,
|
ConditionalUnsimplifier,
|
||||||
@ -374,6 +389,7 @@ map<string, char> const& OptimiserSuite::stepNameToAbbreviationMap()
|
|||||||
{
|
{
|
||||||
static map<string, char> lookupTable{
|
static map<string, char> lookupTable{
|
||||||
{BlockFlattener::name, 'f'},
|
{BlockFlattener::name, 'f'},
|
||||||
|
{CircularReferencesPruner::name, 'l'},
|
||||||
{CommonSubexpressionEliminator::name, 'c'},
|
{CommonSubexpressionEliminator::name, 'c'},
|
||||||
{ConditionalSimplifier::name, 'C'},
|
{ConditionalSimplifier::name, 'C'},
|
||||||
{ConditionalUnsimplifier::name, 'U'},
|
{ConditionalUnsimplifier::name, 'U'},
|
||||||
|
@ -28,6 +28,7 @@
|
|||||||
#include <libyul/optimiser/DeadCodeEliminator.h>
|
#include <libyul/optimiser/DeadCodeEliminator.h>
|
||||||
#include <libyul/optimiser/Disambiguator.h>
|
#include <libyul/optimiser/Disambiguator.h>
|
||||||
#include <libyul/optimiser/CallGraphGenerator.h>
|
#include <libyul/optimiser/CallGraphGenerator.h>
|
||||||
|
#include <libyul/optimiser/CircularReferencesPruner.h>
|
||||||
#include <libyul/optimiser/ConditionalUnsimplifier.h>
|
#include <libyul/optimiser/ConditionalUnsimplifier.h>
|
||||||
#include <libyul/optimiser/ConditionalSimplifier.h>
|
#include <libyul/optimiser/ConditionalSimplifier.h>
|
||||||
#include <libyul/optimiser/CommonSubexpressionEliminator.h>
|
#include <libyul/optimiser/CommonSubexpressionEliminator.h>
|
||||||
@ -250,6 +251,7 @@ TestCase::TestResult YulOptimizerTest::run(ostream& _stream, string const& _line
|
|||||||
CommonSubexpressionEliminator::run(*m_context, *m_ast);
|
CommonSubexpressionEliminator::run(*m_context, *m_ast);
|
||||||
ExpressionSimplifier::run(*m_context, *m_ast);
|
ExpressionSimplifier::run(*m_context, *m_ast);
|
||||||
UnusedPruner::run(*m_context, *m_ast);
|
UnusedPruner::run(*m_context, *m_ast);
|
||||||
|
CircularReferencesPruner::run(*m_context, *m_ast);
|
||||||
DeadCodeEliminator::run(*m_context, *m_ast);
|
DeadCodeEliminator::run(*m_context, *m_ast);
|
||||||
ExpressionJoiner::run(*m_context, *m_ast);
|
ExpressionJoiner::run(*m_context, *m_ast);
|
||||||
ExpressionJoiner::run(*m_context, *m_ast);
|
ExpressionJoiner::run(*m_context, *m_ast);
|
||||||
@ -259,6 +261,12 @@ TestCase::TestResult YulOptimizerTest::run(ostream& _stream, string const& _line
|
|||||||
disambiguate();
|
disambiguate();
|
||||||
UnusedPruner::run(*m_context, *m_ast);
|
UnusedPruner::run(*m_context, *m_ast);
|
||||||
}
|
}
|
||||||
|
else if (m_optimizerStep == "circularReferencesPruner")
|
||||||
|
{
|
||||||
|
disambiguate();
|
||||||
|
FunctionHoister::run(*m_context, *m_ast);
|
||||||
|
CircularReferencesPruner::run(*m_context, *m_ast);
|
||||||
|
}
|
||||||
else if (m_optimizerStep == "deadCodeEliminator")
|
else if (m_optimizerStep == "deadCodeEliminator")
|
||||||
{
|
{
|
||||||
disambiguate();
|
disambiguate();
|
||||||
|
@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
let a
|
||||||
|
function f() -> x { x := g() }
|
||||||
|
function g() -> y { y := f() }
|
||||||
|
function h() -> z { z := g() }
|
||||||
|
a := h()
|
||||||
|
}
|
||||||
|
// ====
|
||||||
|
// step: circularReferencesPruner
|
||||||
|
// ----
|
||||||
|
// {
|
||||||
|
// let a
|
||||||
|
// a := h()
|
||||||
|
// function f() -> x
|
||||||
|
// { x := g() }
|
||||||
|
// function g() -> y
|
||||||
|
// { y := f() }
|
||||||
|
// function h() -> z
|
||||||
|
// { z := g() }
|
||||||
|
// }
|
@ -0,0 +1,14 @@
|
|||||||
|
{
|
||||||
|
{
|
||||||
|
function a() -> x { x := b() }
|
||||||
|
function b() -> y { y := a() }
|
||||||
|
}
|
||||||
|
{
|
||||||
|
function c() -> z { z := d() }
|
||||||
|
function d() -> w { w := c() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ====
|
||||||
|
// step: circularReferencesPruner
|
||||||
|
// ----
|
||||||
|
// { }
|
@ -0,0 +1,14 @@
|
|||||||
|
{
|
||||||
|
{
|
||||||
|
function z() -> x { x := y() }
|
||||||
|
function y() -> x { x := z() }
|
||||||
|
}
|
||||||
|
{
|
||||||
|
function z() -> x { x := y() }
|
||||||
|
function y() -> x { x := z() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ====
|
||||||
|
// step: circularReferencesPruner
|
||||||
|
// ----
|
||||||
|
// { }
|
@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
function f() -> x { x := g() }
|
||||||
|
function g() -> x { x := f() }
|
||||||
|
}
|
||||||
|
// ====
|
||||||
|
// step: circularReferencesPruner
|
||||||
|
// ----
|
||||||
|
// { }
|
@ -92,7 +92,7 @@ BOOST_AUTO_TEST_CASE(output_operator_should_create_concise_and_unambiguous_strin
|
|||||||
|
|
||||||
BOOST_TEST(chromosome.length() == allSteps.size());
|
BOOST_TEST(chromosome.length() == allSteps.size());
|
||||||
BOOST_TEST(chromosome.optimisationSteps() == allSteps);
|
BOOST_TEST(chromosome.optimisationSteps() == allSteps);
|
||||||
BOOST_TEST(toString(chromosome) == "fcCUnDvejsxIOoighTLMrmVatud");
|
BOOST_TEST(toString(chromosome) == "flcCUnDvejsxIOoighTLMrmVatud");
|
||||||
}
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_SUITE_END()
|
BOOST_AUTO_TEST_SUITE_END()
|
||||||
|
Loading…
Reference in New Issue
Block a user