mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
260 lines
7.8 KiB
C++
260 lines
7.8 KiB
C++
/*
|
|
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/>.
|
|
*/
|
|
/**
|
|
* Optimiser component that performs function inlining for arbitrary functions.
|
|
*/
|
|
|
|
#include <libjulia/optimiser/FullInliner.h>
|
|
|
|
#include <libjulia/optimiser/ASTCopier.h>
|
|
#include <libjulia/optimiser/ASTWalker.h>
|
|
#include <libjulia/optimiser/NameCollector.h>
|
|
#include <libjulia/optimiser/Semantics.h>
|
|
#include <libjulia/Exceptions.h>
|
|
|
|
#include <libsolidity/inlineasm/AsmData.h>
|
|
|
|
#include <libdevcore/CommonData.h>
|
|
|
|
#include <boost/range/adaptor/reversed.hpp>
|
|
|
|
using namespace std;
|
|
using namespace dev;
|
|
using namespace dev::julia;
|
|
using namespace dev::solidity;
|
|
|
|
FullInliner::FullInliner(Block& _ast):
|
|
m_ast(_ast)
|
|
{
|
|
assertThrow(m_ast.statements.size() >= 1, OptimizerException, "");
|
|
assertThrow(m_ast.statements.front().type() == typeid(Block), OptimizerException, "");
|
|
m_nameDispenser.m_usedNames = NameCollector(m_ast).names();
|
|
|
|
for (size_t i = 1; i < m_ast.statements.size(); ++i)
|
|
{
|
|
assertThrow(m_ast.statements.at(i).type() == typeid(FunctionDefinition), OptimizerException, "");
|
|
FunctionDefinition& fun = boost::get<FunctionDefinition>(m_ast.statements.at(i));
|
|
m_functions[fun.name] = &fun;
|
|
m_functionsToVisit.insert(&fun);
|
|
}
|
|
}
|
|
|
|
void FullInliner::run()
|
|
{
|
|
assertThrow(m_ast.statements[0].type() == typeid(Block), OptimizerException, "");
|
|
InlineModifier(*this, m_nameDispenser, "").visit(m_ast.statements[0]);
|
|
while (!m_functionsToVisit.empty())
|
|
handleFunction(**m_functionsToVisit.begin());
|
|
}
|
|
|
|
void FullInliner::handleFunction(FunctionDefinition& _fun)
|
|
{
|
|
if (!m_functionsToVisit.count(&_fun))
|
|
return;
|
|
m_functionsToVisit.erase(&_fun);
|
|
(InlineModifier(*this, m_nameDispenser, _fun.name))(_fun.body);
|
|
}
|
|
|
|
void InlineModifier::operator()(FunctionalInstruction& _instruction)
|
|
{
|
|
visitArguments(_instruction.arguments);
|
|
}
|
|
|
|
void InlineModifier::operator()(FunctionCall&)
|
|
{
|
|
assertThrow(false, OptimizerException, "Should be handled in visit() instead.");
|
|
}
|
|
|
|
void InlineModifier::operator()(ForLoop& _loop)
|
|
{
|
|
(*this)(_loop.pre);
|
|
// Do not visit the condition because we cannot inline there.
|
|
(*this)(_loop.post);
|
|
(*this)(_loop.body);
|
|
}
|
|
|
|
void InlineModifier::operator()(Block& _block)
|
|
{
|
|
// This is only used if needed to minimize the number of move operations.
|
|
vector<Statement> modifiedStatements;
|
|
for (size_t i = 0; i < _block.statements.size(); ++i)
|
|
{
|
|
visit(_block.statements.at(i));
|
|
if (!m_statementsToPrefix.empty())
|
|
{
|
|
if (modifiedStatements.empty())
|
|
std::move(
|
|
_block.statements.begin(),
|
|
_block.statements.begin() + i,
|
|
back_inserter(modifiedStatements)
|
|
);
|
|
modifiedStatements += std::move(m_statementsToPrefix);
|
|
m_statementsToPrefix.clear();
|
|
}
|
|
if (!modifiedStatements.empty())
|
|
modifiedStatements.emplace_back(std::move(_block.statements[i]));
|
|
}
|
|
if (!modifiedStatements.empty())
|
|
_block.statements = std::move(modifiedStatements);
|
|
}
|
|
|
|
void InlineModifier::visit(Expression& _expression)
|
|
{
|
|
if (_expression.type() != typeid(FunctionCall))
|
|
return ASTModifier::visit(_expression);
|
|
|
|
FunctionCall& funCall = boost::get<FunctionCall>(_expression);
|
|
FunctionDefinition& fun = m_driver.function(funCall.functionName.name);
|
|
|
|
m_driver.handleFunction(fun);
|
|
|
|
// TODO: Insert good heuristic here. Perhaps implement that inside the driver.
|
|
bool doInline = funCall.functionName.name != m_currentFunction;
|
|
|
|
if (fun.returnVariables.size() > 1)
|
|
doInline = false;
|
|
|
|
{
|
|
vector<string> argNames;
|
|
vector<string> argTypes;
|
|
for (auto const& arg: fun.parameters)
|
|
{
|
|
argNames.push_back(fun.name + "_" + arg.name);
|
|
argTypes.push_back(arg.type);
|
|
}
|
|
visitArguments(funCall.arguments, argNames, argTypes, doInline);
|
|
}
|
|
|
|
if (!doInline)
|
|
return;
|
|
|
|
map<string, string> variableReplacements;
|
|
for (size_t i = 0; i < funCall.arguments.size(); ++i)
|
|
variableReplacements[fun.parameters[i].name] = boost::get<Identifier>(funCall.arguments[i]).name;
|
|
if (fun.returnVariables.empty())
|
|
_expression = noop(funCall.location);
|
|
else
|
|
{
|
|
string returnVariable = fun.returnVariables[0].name;
|
|
variableReplacements[returnVariable] = newName(fun.name + "_" + returnVariable);
|
|
|
|
m_statementsToPrefix.emplace_back(VariableDeclaration{
|
|
funCall.location,
|
|
{{funCall.location, variableReplacements[returnVariable], fun.returnVariables[0].type}},
|
|
{}
|
|
});
|
|
_expression = Identifier{funCall.location, variableReplacements[returnVariable]};
|
|
}
|
|
m_statementsToPrefix.emplace_back(BodyCopier(m_nameDispenser, fun.name + "_", variableReplacements)(fun.body));
|
|
}
|
|
|
|
void InlineModifier::visit(Statement& _statement)
|
|
{
|
|
ASTModifier::visit(_statement);
|
|
// Replace pop(0) expression statemets (and others) by empty blocks.
|
|
if (_statement.type() == typeid(ExpressionStatement))
|
|
{
|
|
ExpressionStatement& expSt = boost::get<ExpressionStatement>(_statement);
|
|
if (expSt.expression.type() == typeid(FunctionalInstruction))
|
|
{
|
|
FunctionalInstruction& funInstr = boost::get<FunctionalInstruction>(expSt.expression);
|
|
if (funInstr.instruction == solidity::Instruction::POP)
|
|
if (MovableChecker(funInstr.arguments.at(0)).movable())
|
|
_statement = Block{expSt.location, {}};
|
|
}
|
|
}
|
|
}
|
|
|
|
void InlineModifier::visitArguments(
|
|
vector<Expression>& _arguments,
|
|
vector<string> const& _nameHints,
|
|
vector<string> const& _types,
|
|
bool _moveToFront
|
|
)
|
|
{
|
|
// If one of the elements moves parts to the front, all other elements right of it
|
|
// also have to be moved to the front to keep the order of evaluation.
|
|
vector<Statement> prefix;
|
|
for (size_t i = 0; i < _arguments.size(); ++i)
|
|
{
|
|
auto& arg = _arguments[i];
|
|
// TODO optimize vector operations, check that it actually moves
|
|
auto internalPrefix = visitRecursively(arg);
|
|
if (!internalPrefix.empty())
|
|
{
|
|
_moveToFront = true;
|
|
// We go through the arguments left to right, so we have to invert
|
|
// the prefixes.
|
|
prefix = std::move(internalPrefix) + std::move(prefix);
|
|
}
|
|
else if (_moveToFront)
|
|
{
|
|
auto location = locationOf(arg);
|
|
string var = newName(i < _nameHints.size() ? _nameHints[i] : "");
|
|
prefix.emplace(prefix.begin(), VariableDeclaration{
|
|
location,
|
|
{{TypedName{location, var, i < _types.size() ? _types[i] : ""}}},
|
|
make_shared<Expression>(std::move(arg))
|
|
});
|
|
arg = Identifier{location, var};
|
|
}
|
|
}
|
|
m_statementsToPrefix += std::move(prefix);
|
|
}
|
|
|
|
vector<Statement> InlineModifier::visitRecursively(Expression& _expression)
|
|
{
|
|
vector<Statement> saved;
|
|
saved.swap(m_statementsToPrefix);
|
|
visit(_expression);
|
|
saved.swap(m_statementsToPrefix);
|
|
return saved;
|
|
}
|
|
|
|
string InlineModifier::newName(string const& _prefix)
|
|
{
|
|
return m_nameDispenser.newName(_prefix);
|
|
}
|
|
|
|
Expression InlineModifier::noop(SourceLocation const& _location)
|
|
{
|
|
return FunctionalInstruction{_location, solidity::Instruction::POP, {
|
|
Literal{_location, assembly::LiteralKind::Number, "0", ""}
|
|
}};
|
|
}
|
|
|
|
Statement BodyCopier::operator()(VariableDeclaration const& _varDecl)
|
|
{
|
|
for (auto const& var: _varDecl.variables)
|
|
m_variableReplacements[var.name] = m_nameDispenser.newName(m_varNamePrefix + var.name);
|
|
return ASTCopier::operator()(_varDecl);
|
|
}
|
|
|
|
Statement BodyCopier::operator()(FunctionDefinition const& _funDef)
|
|
{
|
|
assertThrow(false, OptimizerException, "Function hoisting has to be done before function inlining.");
|
|
return _funDef;
|
|
}
|
|
|
|
string BodyCopier::translateIdentifier(string const& _name)
|
|
{
|
|
if (m_variableReplacements.count(_name))
|
|
return m_variableReplacements.at(_name);
|
|
else
|
|
return _name;
|
|
}
|