Merge pull request #6197 from ethereum/yul-opt-switch-rem-6189

YulOpt: Remove empty cases
This commit is contained in:
chriseth 2019-03-06 18:40:04 +01:00 committed by GitHub
commit 08146c76c9
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 94 additions and 5 deletions

View File

@ -21,6 +21,9 @@
#include <libdevcore/CommonData.h> #include <libdevcore/CommonData.h>
#include <libdevcore/Visitor.h> #include <libdevcore/Visitor.h>
#include <boost/range/algorithm_ext/erase.hpp>
#include <boost/range/algorithm/find_if.hpp>
using namespace std; using namespace std;
using namespace dev; using namespace dev;
using namespace yul; using namespace yul;
@ -83,9 +86,11 @@ void StructuralSimplifier::simplify(std::vector<yul::Statement>& _statements)
return {}; return {};
}, },
[&](Switch& _switchStmt) -> OptionalStatements { [&](Switch& _switchStmt) -> OptionalStatements {
if (_switchStmt.cases.size() == 1) auto& cases = _switchStmt.cases;
if (cases.size() == 1)
{ {
auto& switchCase = _switchStmt.cases.front(); auto& switchCase = cases.front();
auto loc = locationOf(*_switchStmt.expression); auto loc = locationOf(*_switchStmt.expression);
if (switchCase.value) if (switchCase.value)
{ {
@ -107,12 +112,14 @@ void StructuralSimplifier::simplify(std::vector<yul::Statement>& _statements)
return s; return s;
} }
} }
// Replace the whole switch with the resulting case body if arg. is
// a constant
else if (boost::optional<u256> const constExprVal = hasLiteralValue(*_switchStmt.expression)) else if (boost::optional<u256> const constExprVal = hasLiteralValue(*_switchStmt.expression))
{ {
Block* matchingCaseBlock = nullptr; Block* matchingCaseBlock = nullptr;
Case* defaultCase = nullptr; Case* defaultCase = nullptr;
for (auto& _case: _switchStmt.cases) for (auto& _case: cases)
{ {
if (_case.value && valueOfLiteral(*_case.value) == constExprVal) if (_case.value && valueOfLiteral(*_case.value) == constExprVal)
{ {
@ -133,7 +140,22 @@ void StructuralSimplifier::simplify(std::vector<yul::Statement>& _statements)
return s; return s;
} }
else
// Remove cases with empty body if no default case exists
auto const defaultCase = boost::find_if(
cases,
[](Case const& _case) { return !_case.value; });
if (
(defaultCase != cases.end() &&
defaultCase->body.statements.empty()) ||
defaultCase == cases.end()
)
boost::remove_erase_if(
cases,
[](Case const& _case) { return _case.body.statements.empty(); }
);
return {}; return {};
}, },
[&](ForLoop& _forLoop) -> OptionalStatements { [&](ForLoop& _forLoop) -> OptionalStatements {

View File

@ -0,0 +1,21 @@
{
let y := 200
switch y
case 0 { }
case 1 { y := 9 }
default { y := 100 }
}
// ----
// structuralSimplifier
// {
// let y := 200
// switch y
// case 0 {
// }
// case 1 {
// y := 9
// }
// default {
// y := 100
// }
// }

View File

@ -0,0 +1,15 @@
{
let y := 200
switch y
case 0 { }
case 1 { y := 9 }
}
// ----
// structuralSimplifier
// {
// let y := 200
// switch y
// case 1 {
// y := 9
// }
// }

View File

@ -0,0 +1,16 @@
{
let y := 200
switch y
case 0 { }
case 1 { y := 9 }
default { }
}
// ----
// structuralSimplifier
// {
// let y := 200
// switch y
// case 1 {
// y := 9
// }
// }

View File

@ -0,0 +1,15 @@
{
let y := 200
switch y
case 1 { y := 9 }
default { }
}
// ----
// structuralSimplifier
// {
// let y := 200
// switch y
// case 1 {
// y := 9
// }
// }