mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Name displacer.
This commit is contained in:
parent
ee89a0353e
commit
d7b366ff46
@ -81,6 +81,22 @@ inline std::vector<T> operator+(std::vector<T>&& _a, std::vector<T>&& _b)
|
||||
ret += std::move(_b);
|
||||
return ret;
|
||||
}
|
||||
/// Concatenate something to a sets of elements.
|
||||
template <class T, class U>
|
||||
inline std::set<T> operator+(std::set<T> const& _a, U&& _b)
|
||||
{
|
||||
std::set<T> ret(_a);
|
||||
ret += std::forward<U>(_b);
|
||||
return ret;
|
||||
}
|
||||
/// Concatenate something to a sets of elements, move variant.
|
||||
template <class T, class U>
|
||||
inline std::set<T> operator+(std::set<T>&& _a, U&& _b)
|
||||
{
|
||||
std::set<T> ret(std::move(_a));
|
||||
ret += std::forward<U>(_b);
|
||||
return ret;
|
||||
}
|
||||
|
||||
namespace dev
|
||||
{
|
||||
|
@ -102,6 +102,8 @@ add_library(yul
|
||||
optimiser/NameCollector.h
|
||||
optimiser/NameDispenser.cpp
|
||||
optimiser/NameDispenser.h
|
||||
optimiser/NameDisplacer.cpp
|
||||
optimiser/NameDisplacer.h
|
||||
optimiser/OptimizerUtilities.cpp
|
||||
optimiser/OptimizerUtilities.h
|
||||
optimiser/RedundantAssignEliminator.cpp
|
||||
|
@ -35,6 +35,7 @@ void ASTWalker::operator()(FunctionalInstruction const& _instr)
|
||||
|
||||
void ASTWalker::operator()(FunctionCall const& _funCall)
|
||||
{
|
||||
// Does not visit _funCall.functionName on purpose
|
||||
walkVector(_funCall.arguments | boost::adaptors::reversed);
|
||||
}
|
||||
|
||||
@ -108,6 +109,7 @@ void ASTModifier::operator()(FunctionalInstruction& _instr)
|
||||
|
||||
void ASTModifier::operator()(FunctionCall& _funCall)
|
||||
{
|
||||
// Does not visit _funCall.functionName on purpose
|
||||
walkVector(_funCall.arguments | boost::adaptors::reversed);
|
||||
}
|
||||
|
||||
|
@ -32,8 +32,8 @@ using namespace std;
|
||||
using namespace dev;
|
||||
using namespace yul;
|
||||
|
||||
NameDispenser::NameDispenser(Dialect const& _dialect, Block const& _ast):
|
||||
NameDispenser(_dialect, NameCollector(_ast).names())
|
||||
NameDispenser::NameDispenser(Dialect const& _dialect, Block const& _ast, set<YulString> _reservedNames):
|
||||
NameDispenser(_dialect, NameCollector(_ast).names() + std::move(_reservedNames))
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -39,13 +39,17 @@ class NameDispenser
|
||||
{
|
||||
public:
|
||||
/// Initialize the name dispenser with all the names used in the given AST.
|
||||
explicit NameDispenser(Dialect const& _dialect, Block const& _ast);
|
||||
explicit NameDispenser(Dialect const& _dialect, Block const& _ast, std::set<YulString> _reservedNames = {});
|
||||
/// Initialize the name dispenser with the given used names.
|
||||
explicit NameDispenser(Dialect const& _dialect, std::set<YulString> _usedNames);
|
||||
|
||||
/// @returns a currently unused name that should be similar to _nameHint.
|
||||
YulString newName(YulString _nameHint);
|
||||
|
||||
/// Mark @a _name as used, i.e. the dispenser's newName function will not
|
||||
/// return it.
|
||||
void markUsed(YulString _name) { m_usedNames.insert(_name); }
|
||||
|
||||
private:
|
||||
bool illegalName(YulString _name);
|
||||
|
||||
|
86
libyul/optimiser/NameDisplacer.cpp
Normal file
86
libyul/optimiser/NameDisplacer.cpp
Normal file
@ -0,0 +1,86 @@
|
||||
/*
|
||||
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 renames identifiers to free up certain names.
|
||||
*/
|
||||
|
||||
#include <libyul/optimiser/NameDisplacer.h>
|
||||
|
||||
#include <libyul/AsmData.h>
|
||||
|
||||
|
||||
using namespace std;
|
||||
using namespace dev;
|
||||
using namespace yul;
|
||||
|
||||
void NameDisplacer::operator()(Identifier& _identifier)
|
||||
{
|
||||
checkAndReplace(_identifier.name);
|
||||
}
|
||||
|
||||
void NameDisplacer::operator()(VariableDeclaration& _varDecl)
|
||||
{
|
||||
for (TypedName& var: _varDecl.variables)
|
||||
checkAndReplaceNew(var.name);
|
||||
|
||||
ASTModifier::operator()(_varDecl);
|
||||
}
|
||||
|
||||
void NameDisplacer::operator()(FunctionDefinition& _function)
|
||||
{
|
||||
// Should have been done in the block already.
|
||||
yulAssert(!m_namesToFree.count(_function.name), "");
|
||||
|
||||
for (auto& param: _function.parameters)
|
||||
checkAndReplaceNew(param.name);
|
||||
for (auto& retVar: _function.returnVariables)
|
||||
checkAndReplaceNew(retVar.name);
|
||||
|
||||
ASTModifier::operator()(_function);
|
||||
}
|
||||
|
||||
void NameDisplacer::operator()(FunctionCall& _funCall)
|
||||
{
|
||||
checkAndReplace(_funCall.functionName.name);
|
||||
ASTModifier::operator()(_funCall);
|
||||
}
|
||||
|
||||
void NameDisplacer::operator()(Block& _block)
|
||||
{
|
||||
// First replace all the names of function definitions
|
||||
// because of scoping.
|
||||
for (auto& st: _block.statements)
|
||||
if (st.type() == typeid(FunctionDefinition))
|
||||
checkAndReplaceNew(boost::get<FunctionDefinition>(st).name);
|
||||
|
||||
ASTModifier::operator()(_block);
|
||||
}
|
||||
|
||||
void NameDisplacer::checkAndReplaceNew(YulString& _name)
|
||||
{
|
||||
yulAssert(!m_translations.count(_name), "");
|
||||
if (m_namesToFree.count(_name))
|
||||
_name = (m_translations[_name] = m_nameDispenser.newName(_name));
|
||||
}
|
||||
|
||||
void NameDisplacer::checkAndReplace(YulString& _name) const
|
||||
{
|
||||
if (m_translations.count(_name))
|
||||
_name = m_translations.at(_name);
|
||||
yulAssert(!m_namesToFree.count(_name), "");
|
||||
}
|
||||
|
70
libyul/optimiser/NameDisplacer.h
Normal file
70
libyul/optimiser/NameDisplacer.h
Normal file
@ -0,0 +1,70 @@
|
||||
/*
|
||||
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 renames identifiers to free up certain names.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <libyul/optimiser/ASTWalker.h>
|
||||
#include <libyul/optimiser/NameDispenser.h>
|
||||
|
||||
#include <set>
|
||||
#include <map>
|
||||
|
||||
namespace yul
|
||||
{
|
||||
struct Dialect;
|
||||
|
||||
/**
|
||||
* Optimiser component that renames identifiers to free up certain names.
|
||||
*
|
||||
* Prerequisites: Disambiguator
|
||||
*/
|
||||
class NameDisplacer: public ASTModifier
|
||||
{
|
||||
public:
|
||||
explicit NameDisplacer(
|
||||
NameDispenser& _dispenser,
|
||||
std::set<YulString> const& _namesToFree
|
||||
):
|
||||
m_nameDispenser(_dispenser),
|
||||
m_namesToFree(_namesToFree)
|
||||
{
|
||||
for (YulString n: _namesToFree)
|
||||
m_nameDispenser.markUsed(n);
|
||||
}
|
||||
|
||||
using ASTModifier::operator();
|
||||
void operator()(Identifier& _identifier) override;
|
||||
void operator()(VariableDeclaration& _varDecl) override;
|
||||
void operator()(FunctionDefinition& _function) override;
|
||||
void operator()(FunctionCall& _funCall) override;
|
||||
void operator()(Block& _block) override;
|
||||
|
||||
protected:
|
||||
/// Check if the newly introduced identifier @a _name has to be replaced.
|
||||
void checkAndReplaceNew(YulString& _name);
|
||||
/// Replace the identifier @a _name if it is in the translation map.
|
||||
void checkAndReplace(YulString& _name) const;
|
||||
|
||||
NameDispenser& m_nameDispenser;
|
||||
std::set<YulString> const& m_namesToFree;
|
||||
std::map<YulString, YulString> m_translations;
|
||||
};
|
||||
|
||||
}
|
@ -87,7 +87,7 @@ void OptimiserSuite::run(
|
||||
|
||||
// None of the above can make stack problems worse.
|
||||
|
||||
NameDispenser dispenser{_dialect, ast};
|
||||
NameDispenser dispenser{_dialect, ast, reservedIdentifiers};
|
||||
|
||||
size_t codeSize = 0;
|
||||
for (size_t rounds = 0; rounds < 12; ++rounds)
|
||||
|
@ -36,6 +36,7 @@
|
||||
#include <libyul/optimiser/ForLoopConditionIntoBody.h>
|
||||
#include <libyul/optimiser/ForLoopInitRewriter.h>
|
||||
#include <libyul/optimiser/MainFunction.h>
|
||||
#include <libyul/optimiser/NameDisplacer.h>
|
||||
#include <libyul/optimiser/Rematerialiser.h>
|
||||
#include <libyul/optimiser/ExpressionSimplifier.h>
|
||||
#include <libyul/optimiser/UnusedPruner.h>
|
||||
@ -110,6 +111,15 @@ TestCase::TestResult YulOptimizerTest::run(ostream& _stream, string const& _line
|
||||
soltestAssert(m_dialect, "Dialect not set.");
|
||||
if (m_optimizerStep == "disambiguator")
|
||||
disambiguate();
|
||||
else if (m_optimizerStep == "nameDisplacer")
|
||||
{
|
||||
disambiguate();
|
||||
NameDispenser nameDispenser{*m_dialect, *m_ast};
|
||||
NameDisplacer{
|
||||
nameDispenser,
|
||||
{"illegal1"_yulstring, "illegal2"_yulstring, "illegal3"_yulstring, "illegal4"_yulstring, "illegal5"_yulstring}
|
||||
}(*m_ast);
|
||||
}
|
||||
else if (m_optimizerStep == "blockFlattener")
|
||||
{
|
||||
disambiguate();
|
||||
|
25
test/libyul/yulOptimizerTests/nameDisplacer/funtion_call.yul
Normal file
25
test/libyul/yulOptimizerTests/nameDisplacer/funtion_call.yul
Normal file
@ -0,0 +1,25 @@
|
||||
{
|
||||
let x := illegal4(1, 2)
|
||||
function illegal4(illegal1, illegal2) -> illegal3 { illegal3 := add(illegal1, illegal2) }
|
||||
{
|
||||
let y := illegal5(3, 4)
|
||||
function illegal5(illegal1, illegal2) -> illegal3 { illegal3 := add(illegal1, illegal2) }
|
||||
}
|
||||
}
|
||||
// ====
|
||||
// step: nameDisplacer
|
||||
// ----
|
||||
// {
|
||||
// let x := illegal4_1(1, 2)
|
||||
// function illegal4_1(illegal1_2, illegal2_3) -> illegal3_4
|
||||
// {
|
||||
// illegal3_4 := add(illegal1_2, illegal2_3)
|
||||
// }
|
||||
// {
|
||||
// let y := illegal5_5(3, 4)
|
||||
// function illegal5_5(illegal1_1, illegal2_2) -> illegal3_3
|
||||
// {
|
||||
// illegal3_3 := add(illegal1_1, illegal2_2)
|
||||
// }
|
||||
// }
|
||||
// }
|
11
test/libyul/yulOptimizerTests/nameDisplacer/variables.yul
Normal file
11
test/libyul/yulOptimizerTests/nameDisplacer/variables.yul
Normal file
@ -0,0 +1,11 @@
|
||||
{ { let illegal1 := 1 } { let illegal2 := 2 let illegal3, illegal4 } }
|
||||
// ====
|
||||
// step: nameDisplacer
|
||||
// ----
|
||||
// {
|
||||
// { let illegal1_1 := 1 }
|
||||
// {
|
||||
// let illegal2_2 := 2
|
||||
// let illegal3_3, illegal4_4
|
||||
// }
|
||||
// }
|
@ -0,0 +1,16 @@
|
||||
{
|
||||
function f(illegal1, illegal2) -> illegal3 {
|
||||
let illegal4 := illegal1
|
||||
illegal3 := add(illegal1, illegal2)
|
||||
}
|
||||
}
|
||||
// ====
|
||||
// step: nameDisplacer
|
||||
// ----
|
||||
// {
|
||||
// function f(illegal1_1, illegal2_2) -> illegal3_3
|
||||
// {
|
||||
// let illegal4_4 := illegal1_1
|
||||
// illegal3_3 := add(illegal1_1, illegal2_2)
|
||||
// }
|
||||
// }
|
Loading…
Reference in New Issue
Block a user