mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #5641 from ethereum/supportUnassigned
[Yul] Support unassigned variables in the SSA value tracker and the data flow analyzer.
This commit is contained in:
commit
642c69f935
@ -136,17 +136,22 @@ void DataFlowAnalyzer::operator()(Block& _block)
|
|||||||
|
|
||||||
void DataFlowAnalyzer::handleAssignment(set<YulString> const& _variables, Expression* _value)
|
void DataFlowAnalyzer::handleAssignment(set<YulString> const& _variables, Expression* _value)
|
||||||
{
|
{
|
||||||
|
static Expression const zero{Literal{{}, LiteralKind::Number, YulString{"0"}, {}}};
|
||||||
clearValues(_variables);
|
clearValues(_variables);
|
||||||
|
|
||||||
MovableChecker movableChecker;
|
MovableChecker movableChecker;
|
||||||
if (_value)
|
if (_value)
|
||||||
movableChecker.visit(*_value);
|
movableChecker.visit(*_value);
|
||||||
if (_variables.size() == 1)
|
else
|
||||||
|
for (auto const& var: _variables)
|
||||||
|
m_value[var] = &zero;
|
||||||
|
|
||||||
|
if (_value && _variables.size() == 1)
|
||||||
{
|
{
|
||||||
YulString name = *_variables.begin();
|
YulString name = *_variables.begin();
|
||||||
// Expression has to be movable and cannot contain a reference
|
// Expression has to be movable and cannot contain a reference
|
||||||
// to the variable that will be assigned to.
|
// to the variable that will be assigned to.
|
||||||
if (_value && movableChecker.movable() && !movableChecker.referencedVariables().count(name))
|
if (movableChecker.movable() && !movableChecker.referencedVariables().count(name))
|
||||||
m_value[name] = _value;
|
m_value[name] = _value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,6 +36,8 @@ namespace yul
|
|||||||
* Tracks assignments and is used as base class for both Rematerialiser and
|
* Tracks assignments and is used as base class for both Rematerialiser and
|
||||||
* Common Subexpression Eliminator.
|
* Common Subexpression Eliminator.
|
||||||
*
|
*
|
||||||
|
* A special zero constant expression is used for the default value of variables.
|
||||||
|
*
|
||||||
* Prerequisite: Disambiguator
|
* Prerequisite: Disambiguator
|
||||||
*/
|
*/
|
||||||
class DataFlowAnalyzer: public ASTModifier
|
class DataFlowAnalyzer: public ASTModifier
|
||||||
|
@ -35,11 +35,12 @@ void SSAValueTracker::operator()(Assignment const& _assignment)
|
|||||||
|
|
||||||
void SSAValueTracker::operator()(VariableDeclaration const& _varDecl)
|
void SSAValueTracker::operator()(VariableDeclaration const& _varDecl)
|
||||||
{
|
{
|
||||||
if (_varDecl.variables.size() == 1)
|
static Expression const zero{Literal{{}, LiteralKind::Number, YulString{"0"}, {}}};
|
||||||
setValue(_varDecl.variables.front().name, _varDecl.value.get());
|
if (!_varDecl.value)
|
||||||
else if (!_varDecl.value)
|
|
||||||
for (auto const& var: _varDecl.variables)
|
for (auto const& var: _varDecl.variables)
|
||||||
setValue(var.name, nullptr);
|
setValue(var.name, &zero);
|
||||||
|
else if (_varDecl.variables.size() == 1)
|
||||||
|
setValue(_varDecl.variables.front().name, _varDecl.value.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
void SSAValueTracker::setValue(YulString _name, Expression const* _value)
|
void SSAValueTracker::setValue(YulString _name, Expression const* _value)
|
||||||
|
@ -33,7 +33,7 @@ namespace yul
|
|||||||
* Class that walks the AST and stores the initial value of each variable
|
* Class that walks the AST and stores the initial value of each variable
|
||||||
* that is never assigned to.
|
* that is never assigned to.
|
||||||
*
|
*
|
||||||
* Default value is represented as nullptr.
|
* A special zero constant expression is used for the default value of variables.
|
||||||
*
|
*
|
||||||
* Prerequisite: Disambiguator
|
* Prerequisite: Disambiguator
|
||||||
*/
|
*/
|
||||||
|
@ -0,0 +1,14 @@
|
|||||||
|
{
|
||||||
|
// This does not replace b by a because there is no
|
||||||
|
// explicit assignment, even though both hold the same value.
|
||||||
|
let a
|
||||||
|
let b
|
||||||
|
mstore(sub(a, b), 7)
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// commonSubexpressionEliminator
|
||||||
|
// {
|
||||||
|
// let a
|
||||||
|
// let b
|
||||||
|
// mstore(sub(a, b), 7)
|
||||||
|
// }
|
@ -1,5 +1,4 @@
|
|||||||
// c & d can't be optimized as expression simplifier doesn't handle default
|
// Unassigned variables are assumed to be zero.
|
||||||
// values yet
|
|
||||||
{
|
{
|
||||||
let c, d
|
let c, d
|
||||||
let y := add(d, add(c, 7))
|
let y := add(d, add(c, 7))
|
||||||
@ -8,5 +7,5 @@
|
|||||||
// expressionSimplifier
|
// expressionSimplifier
|
||||||
// {
|
// {
|
||||||
// let c, d
|
// let c, d
|
||||||
// let y := add(add(d, c), 7)
|
// let y := 7
|
||||||
// }
|
// }
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
// c & d can't be optimized as expression simplifier doesn't handle default
|
// Unassigned variables are assumed to be zero.
|
||||||
// values yet
|
|
||||||
{
|
{
|
||||||
let c
|
let c
|
||||||
let d
|
let d
|
||||||
@ -10,5 +9,5 @@
|
|||||||
// {
|
// {
|
||||||
// let c
|
// let c
|
||||||
// let d
|
// let d
|
||||||
// let y := add(add(d, c), 7)
|
// let y := 7
|
||||||
// }
|
// }
|
||||||
|
@ -13,6 +13,9 @@
|
|||||||
let r := f(a)
|
let r := f(a)
|
||||||
// This should be inlined because it is a constant
|
// This should be inlined because it is a constant
|
||||||
let t := f(a2)
|
let t := f(a2)
|
||||||
|
let a3
|
||||||
|
// This should be inlined because it is a constant as well (zero)
|
||||||
|
let s := f(a3)
|
||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// fullInliner
|
// fullInliner
|
||||||
@ -30,6 +33,16 @@
|
|||||||
// let f_y := add(f_a, f_x)
|
// let f_y := add(f_a, f_x)
|
||||||
// sstore(f_y, 10)
|
// sstore(f_y, 10)
|
||||||
// let t := f_b
|
// let t := f_b
|
||||||
|
// let a3
|
||||||
|
// let f_a_5 := a3
|
||||||
|
// let f_b_6
|
||||||
|
// let f_x_7 := mload(f_a_5)
|
||||||
|
// f_b_6 := sload(f_x_7)
|
||||||
|
// let f_c_8 := 3
|
||||||
|
// mstore(mul(f_a_5, f_b_6), mload(f_x_7))
|
||||||
|
// let f_y_11 := add(f_a_5, f_x_7)
|
||||||
|
// sstore(f_y_11, 10)
|
||||||
|
// let s := f_b_6
|
||||||
// }
|
// }
|
||||||
// function f(a) -> b
|
// function f(a) -> b
|
||||||
// {
|
// {
|
||||||
|
@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
let x, y
|
||||||
|
if x { mstore(0, 0) }
|
||||||
|
if y { mstore(0, 0) }
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// structuralSimplifier
|
||||||
|
// {
|
||||||
|
// let x, y
|
||||||
|
// }
|
@ -0,0 +1,9 @@
|
|||||||
|
{
|
||||||
|
let x
|
||||||
|
if x { mstore(0, 0) }
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// structuralSimplifier
|
||||||
|
// {
|
||||||
|
// let x
|
||||||
|
// }
|
Loading…
Reference in New Issue
Block a user