Add variable declaration initializer.

This commit is contained in:
chriseth 2018-12-13 15:26:01 +01:00
parent 633dd44576
commit 11209ec48a
10 changed files with 196 additions and 0 deletions

View File

@ -44,5 +44,6 @@ add_library(yul
optimiser/UnusedPruner.cpp
optimiser/Utilities.cpp
optimiser/VarDeclPropagator.cpp
optimiser/VarDeclInitializer.cpp
)
target_link_libraries(yul PUBLIC evmasm devcore langutil)

View File

@ -21,6 +21,7 @@
#include <libyul/optimiser/Suite.h>
#include <libyul/optimiser/Disambiguator.h>
#include <libyul/optimiser/VarDeclInitializer.h>
#include <libyul/optimiser/FunctionGrouper.h>
#include <libyul/optimiser/FunctionHoister.h>
#include <libyul/optimiser/ExpressionSplitter.h>
@ -56,6 +57,7 @@ void OptimiserSuite::run(
Block ast = boost::get<Block>(Disambiguator(_analysisInfo, reservedIdentifiers)(_ast));
(VarDeclInitializer{})(ast);
(FunctionHoister{})(ast);
(FunctionGrouper{})(ast);
(ForLoopInitRewriter{})(ast);

View File

@ -0,0 +1,56 @@
/*
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/VarDeclInitializer.h>
#include <libyul/AsmData.h>
#include <libdevcore/CommonData.h>
#include <libdevcore/Visitor.h>
using namespace std;
using namespace dev;
using namespace yul;
void VarDeclInitializer::operator()(Block& _block)
{
ASTModifier::operator()(_block);
static Expression const zero{Literal{{}, LiteralKind::Number, YulString{"0"}, {}}};
using OptionalStatements = boost::optional<vector<Statement>>;
GenericFallbackReturnsVisitor<OptionalStatements, VariableDeclaration> visitor{
[](VariableDeclaration& _varDecl) -> OptionalStatements
{
if (_varDecl.value)
return {};
else if (_varDecl.variables.size() == 1)
{
_varDecl.value = make_shared<Expression>(zero);
return {};
}
else
{
OptionalStatements ret{vector<Statement>{}};
langutil::SourceLocation loc{std::move(_varDecl.location)};
for (auto& var: _varDecl.variables)
ret->push_back(VariableDeclaration{loc, {std::move(var)}, make_shared<Expression>(zero)});
return ret;
}
}
};
iterateReplacing(_block.statements, boost::apply_visitor(visitor));
}

View File

@ -0,0 +1,38 @@
/*
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/>.
*/
#pragma once
#include <libyul/AsmDataForward.h>
#include <libyul/optimiser/ASTWalker.h>
namespace yul
{
/**
* Rewrites variable declarations so that all of them are initialized.
* Declarations like ``let x, y`` are split into multiple declaration
* statements.
* Only supports initializing with the zero literal for now.
*/
class VarDeclInitializer: public ASTModifier
{
public:
void operator()(Block& _block) override;
};
}

View File

@ -23,6 +23,7 @@
#include <libyul/optimiser/BlockFlattener.h>
#include <libyul/optimiser/VarDeclPropagator.h>
#include <libyul/optimiser/VarDeclInitializer.h>
#include <libyul/optimiser/Disambiguator.h>
#include <libyul/optimiser/CommonSubexpressionEliminator.h>
#include <libyul/optimiser/NameCollector.h>
@ -112,6 +113,8 @@ bool YulOptimizerTest::run(ostream& _stream, string const& _linePrefix, bool con
disambiguate();
VarDeclPropagator{}(*m_ast);
}
else if (m_optimizerStep == "varDeclInitializer")
VarDeclInitializer{}(*m_ast);
else if (m_optimizerStep == "forLoopInitRewriter")
{
disambiguate();

View File

@ -0,0 +1,29 @@
{
// This component does not need the disambiguator
function f() -> x, y {
let a, b
mstore(a, b)
let d
d := 2
}
let a
a := 4
let b := 2
let x, y := f()
}
// ----
// varDeclInitializer
// {
// function f() -> x, y
// {
// let a := 0
// let b := 0
// mstore(a, b)
// let d := 0
// d := 2
// }
// let a := 0
// a := 4
// let b := 2
// let x, y := f()
// }

View File

@ -0,0 +1,24 @@
{
function f() -> x, y {
let a, b
mstore(a, b)
let d
d := 2
}
let r
r := 4
}
// ----
// varDeclInitializer
// {
// function f() -> x, y
// {
// let a := 0
// let b := 0
// mstore(a, b)
// let d := 0
// d := 2
// }
// let r := 0
// r := 4
// }

View File

@ -0,0 +1,14 @@
{
let x, y, z
let a
let b
}
// ----
// varDeclInitializer
// {
// let x := 0
// let y := 0
// let z := 0
// let a := 0
// let b := 0
// }

View File

@ -0,0 +1,21 @@
{
function f() -> x, y {
let a, b := f()
let u
}
let r
let s := 3
let t
}
// ----
// varDeclInitializer
// {
// function f() -> x, y
// {
// let a, b := f()
// let u := 0
// }
// let r := 0
// let s := 3
// let t := 0
// }

View File

@ -0,0 +1,8 @@
{
let a
}
// ----
// varDeclInitializer
// {
// let a := 0
// }