mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
[SMTChecker] Move variable handling to EncodingContext
This commit is contained in:
parent
01dd9ba2ae
commit
ebbe03cad6
@ -21,42 +21,118 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::solidity;
|
|
||||||
using namespace dev::solidity::smt;
|
using namespace dev::solidity::smt;
|
||||||
|
|
||||||
EncodingContext::EncodingContext(SolverInterface& _solver):
|
EncodingContext::EncodingContext(SolverInterface& _solver):
|
||||||
m_solver(_solver),
|
m_solver(_solver),
|
||||||
m_thisAddress(make_unique<SymbolicAddressVariable>("this", m_solver))
|
m_thisAddress(make_unique<SymbolicAddressVariable>("this", m_solver))
|
||||||
{
|
{
|
||||||
auto sort = make_shared<smt::ArraySort>(
|
auto sort = make_shared<ArraySort>(
|
||||||
make_shared<smt::Sort>(smt::Kind::Int),
|
make_shared<Sort>(Kind::Int),
|
||||||
make_shared<smt::Sort>(smt::Kind::Int)
|
make_shared<Sort>(Kind::Int)
|
||||||
);
|
);
|
||||||
m_balances = make_unique<SymbolicVariable>(sort, "balances", m_solver);
|
m_balances = make_unique<SymbolicVariable>(sort, "balances", m_solver);
|
||||||
}
|
}
|
||||||
|
|
||||||
void EncodingContext::reset()
|
void EncodingContext::reset()
|
||||||
{
|
{
|
||||||
|
resetAllVariables();
|
||||||
m_thisAddress->increaseIndex();
|
m_thisAddress->increaseIndex();
|
||||||
m_balances->increaseIndex();
|
m_balances->increaseIndex();
|
||||||
}
|
}
|
||||||
|
|
||||||
smt::Expression EncodingContext::thisAddress()
|
/// Variables.
|
||||||
|
|
||||||
|
shared_ptr<SymbolicVariable> EncodingContext::variable(solidity::VariableDeclaration const& _varDecl)
|
||||||
|
{
|
||||||
|
solAssert(knownVariable(_varDecl), "");
|
||||||
|
return m_variables[&_varDecl];
|
||||||
|
}
|
||||||
|
|
||||||
|
bool EncodingContext::createVariable(solidity::VariableDeclaration const& _varDecl)
|
||||||
|
{
|
||||||
|
solAssert(!knownVariable(_varDecl), "");
|
||||||
|
auto const& type = _varDecl.type();
|
||||||
|
auto result = newSymbolicVariable(*type, _varDecl.name() + "_" + to_string(_varDecl.id()), m_solver);
|
||||||
|
m_variables.emplace(&_varDecl, result.second);
|
||||||
|
return result.first;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool EncodingContext::knownVariable(solidity::VariableDeclaration const& _varDecl)
|
||||||
|
{
|
||||||
|
return m_variables.count(&_varDecl);
|
||||||
|
}
|
||||||
|
|
||||||
|
void EncodingContext::resetVariable(solidity::VariableDeclaration const& _variable)
|
||||||
|
{
|
||||||
|
newValue(_variable);
|
||||||
|
setUnknownValue(_variable);
|
||||||
|
}
|
||||||
|
|
||||||
|
void EncodingContext::resetVariables(set<solidity::VariableDeclaration const*> const& _variables)
|
||||||
|
{
|
||||||
|
for (auto const* decl: _variables)
|
||||||
|
resetVariable(*decl);
|
||||||
|
}
|
||||||
|
|
||||||
|
void EncodingContext::resetVariables(function<bool(solidity::VariableDeclaration const&)> const& _filter)
|
||||||
|
{
|
||||||
|
for_each(begin(m_variables), end(m_variables), [&](auto _variable)
|
||||||
|
{
|
||||||
|
if (_filter(*_variable.first))
|
||||||
|
this->resetVariable(*_variable.first);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void EncodingContext::resetAllVariables()
|
||||||
|
{
|
||||||
|
resetVariables([&](solidity::VariableDeclaration const&) { return true; });
|
||||||
|
}
|
||||||
|
|
||||||
|
Expression EncodingContext::newValue(solidity::VariableDeclaration const& _decl)
|
||||||
|
{
|
||||||
|
solAssert(knownVariable(_decl), "");
|
||||||
|
return m_variables.at(&_decl)->increaseIndex();
|
||||||
|
}
|
||||||
|
|
||||||
|
void EncodingContext::setZeroValue(solidity::VariableDeclaration const& _decl)
|
||||||
|
{
|
||||||
|
solAssert(knownVariable(_decl), "");
|
||||||
|
setZeroValue(*m_variables.at(&_decl));
|
||||||
|
}
|
||||||
|
|
||||||
|
void EncodingContext::setZeroValue(SymbolicVariable& _variable)
|
||||||
|
{
|
||||||
|
setSymbolicZeroValue(_variable, m_solver);
|
||||||
|
}
|
||||||
|
|
||||||
|
void EncodingContext::setUnknownValue(solidity::VariableDeclaration const& _decl)
|
||||||
|
{
|
||||||
|
solAssert(knownVariable(_decl), "");
|
||||||
|
setUnknownValue(*m_variables.at(&_decl));
|
||||||
|
}
|
||||||
|
|
||||||
|
void EncodingContext::setUnknownValue(SymbolicVariable& _variable)
|
||||||
|
{
|
||||||
|
setSymbolicUnknownValue(_variable, m_solver);
|
||||||
|
}
|
||||||
|
|
||||||
|
Expression EncodingContext::thisAddress()
|
||||||
{
|
{
|
||||||
return m_thisAddress->currentValue();
|
return m_thisAddress->currentValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
smt::Expression EncodingContext::balance()
|
Expression EncodingContext::balance()
|
||||||
{
|
{
|
||||||
return balance(m_thisAddress->currentValue());
|
return balance(m_thisAddress->currentValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
smt::Expression EncodingContext::balance(smt::Expression _address)
|
Expression EncodingContext::balance(Expression _address)
|
||||||
{
|
{
|
||||||
return smt::Expression::select(m_balances->currentValue(), move(_address));
|
return Expression::select(m_balances->currentValue(), move(_address));
|
||||||
}
|
}
|
||||||
|
|
||||||
void EncodingContext::transfer(smt::Expression _from, smt::Expression _to, smt::Expression _value)
|
void EncodingContext::transfer(Expression _from, Expression _to, Expression _value)
|
||||||
{
|
{
|
||||||
unsigned indexBefore = m_balances->index();
|
unsigned indexBefore = m_balances->index();
|
||||||
addBalance(_from, 0 - _value);
|
addBalance(_from, 0 - _value);
|
||||||
@ -65,7 +141,7 @@ void EncodingContext::transfer(smt::Expression _from, smt::Expression _to, smt::
|
|||||||
solAssert(indexAfter > indexBefore, "");
|
solAssert(indexAfter > indexBefore, "");
|
||||||
m_balances->increaseIndex();
|
m_balances->increaseIndex();
|
||||||
/// Do not apply the transfer operation if _from == _to.
|
/// Do not apply the transfer operation if _from == _to.
|
||||||
auto newBalances = smt::Expression::ite(
|
auto newBalances = Expression::ite(
|
||||||
move(_from) == move(_to),
|
move(_from) == move(_to),
|
||||||
m_balances->valueAtIndex(indexBefore),
|
m_balances->valueAtIndex(indexBefore),
|
||||||
m_balances->valueAtIndex(indexAfter)
|
m_balances->valueAtIndex(indexAfter)
|
||||||
@ -73,9 +149,9 @@ void EncodingContext::transfer(smt::Expression _from, smt::Expression _to, smt::
|
|||||||
m_solver.addAssertion(m_balances->currentValue() == newBalances);
|
m_solver.addAssertion(m_balances->currentValue() == newBalances);
|
||||||
}
|
}
|
||||||
|
|
||||||
void EncodingContext::addBalance(smt::Expression _address, smt::Expression _value)
|
void EncodingContext::addBalance(Expression _address, Expression _value)
|
||||||
{
|
{
|
||||||
auto newBalances = smt::Expression::store(
|
auto newBalances = Expression::store(
|
||||||
m_balances->currentValue(),
|
m_balances->currentValue(),
|
||||||
_address,
|
_address,
|
||||||
balance(_address) + move(_value)
|
balance(_address) + move(_value)
|
||||||
|
@ -20,6 +20,9 @@
|
|||||||
#include <libsolidity/formal/SolverInterface.h>
|
#include <libsolidity/formal/SolverInterface.h>
|
||||||
#include <libsolidity/formal/SymbolicVariables.h>
|
#include <libsolidity/formal/SymbolicVariables.h>
|
||||||
|
|
||||||
|
#include <unordered_map>
|
||||||
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
namespace dev
|
||||||
{
|
{
|
||||||
namespace solidity
|
namespace solidity
|
||||||
@ -38,22 +41,60 @@ public:
|
|||||||
/// Resets the entire context.
|
/// Resets the entire context.
|
||||||
void reset();
|
void reset();
|
||||||
|
|
||||||
/// Value of `this` address.
|
/// Methods related to variables.
|
||||||
smt::Expression thisAddress();
|
//@{
|
||||||
|
/// @returns the symbolic representation of a program variable.
|
||||||
|
std::shared_ptr<SymbolicVariable> variable(solidity::VariableDeclaration const& _varDecl);
|
||||||
|
/// @returns all symbolic variables.
|
||||||
|
std::unordered_map<solidity::VariableDeclaration const*, std::shared_ptr<SymbolicVariable>> const& variables() const { return m_variables; }
|
||||||
|
|
||||||
|
/// Creates a symbolic variable and
|
||||||
|
/// @returns true if a variable's type is not supported and is therefore abstract.
|
||||||
|
bool createVariable(solidity::VariableDeclaration const& _varDecl);
|
||||||
|
/// @returns true if variable was created.
|
||||||
|
bool knownVariable(solidity::VariableDeclaration const& _varDecl);
|
||||||
|
|
||||||
|
/// Resets a specific variable.
|
||||||
|
void resetVariable(solidity::VariableDeclaration const& _variable);
|
||||||
|
/// Resets a set of variables.
|
||||||
|
void resetVariables(std::set<solidity::VariableDeclaration const*> const& _variables);
|
||||||
|
/// Resets variables according to a predicate.
|
||||||
|
void resetVariables(std::function<bool(solidity::VariableDeclaration const&)> const& _filter);
|
||||||
|
///Resets all variables.
|
||||||
|
void resetAllVariables();
|
||||||
|
|
||||||
|
/// Allocates a new index for the declaration, updates the current
|
||||||
|
/// index to this value and returns the expression.
|
||||||
|
Expression newValue(solidity::VariableDeclaration const& _decl);
|
||||||
|
/// Sets the value of the declaration to zero.
|
||||||
|
void setZeroValue(solidity::VariableDeclaration const& _decl);
|
||||||
|
void setZeroValue(SymbolicVariable& _variable);
|
||||||
|
/// Resets the variable to an unknown value (in its range).
|
||||||
|
void setUnknownValue(solidity::VariableDeclaration const& decl);
|
||||||
|
void setUnknownValue(SymbolicVariable& _variable);
|
||||||
|
//@}
|
||||||
|
|
||||||
|
/// Blockchain related methods.
|
||||||
|
//@{
|
||||||
|
/// Value of `this` address.
|
||||||
|
Expression thisAddress();
|
||||||
/// @returns the symbolic balance of address `this`.
|
/// @returns the symbolic balance of address `this`.
|
||||||
smt::Expression balance();
|
Expression balance();
|
||||||
/// @returns the symbolic balance of an address.
|
/// @returns the symbolic balance of an address.
|
||||||
smt::Expression balance(smt::Expression _address);
|
Expression balance(Expression _address);
|
||||||
/// Transfer _value from _from to _to.
|
/// Transfer _value from _from to _to.
|
||||||
void transfer(smt::Expression _from, smt::Expression _to, smt::Expression _value);
|
void transfer(Expression _from, Expression _to, Expression _value);
|
||||||
|
//@}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
/// Adds _value to _account's balance.
|
/// Adds _value to _account's balance.
|
||||||
void addBalance(smt::Expression _account, smt::Expression _value);
|
void addBalance(Expression _account, Expression _value);
|
||||||
|
|
||||||
SolverInterface& m_solver;
|
SolverInterface& m_solver;
|
||||||
|
|
||||||
|
/// Symbolic variables.
|
||||||
|
std::unordered_map<solidity::VariableDeclaration const*, std::shared_ptr<SymbolicVariable>> m_variables;
|
||||||
|
|
||||||
/// Symbolic `this` address.
|
/// Symbolic `this` address.
|
||||||
std::unique_ptr<SymbolicAddressVariable> m_thisAddress;
|
std::unique_ptr<SymbolicAddressVariable> m_thisAddress;
|
||||||
|
|
||||||
|
@ -87,7 +87,7 @@ bool SMTChecker::visit(ContractDefinition const& _contract)
|
|||||||
|
|
||||||
void SMTChecker::endVisit(ContractDefinition const&)
|
void SMTChecker::endVisit(ContractDefinition const&)
|
||||||
{
|
{
|
||||||
m_variables.clear();
|
m_context.resetAllVariables();
|
||||||
}
|
}
|
||||||
|
|
||||||
void SMTChecker::endVisit(VariableDeclaration const& _varDecl)
|
void SMTChecker::endVisit(VariableDeclaration const& _varDecl)
|
||||||
@ -247,7 +247,7 @@ bool SMTChecker::visit(WhileStatement const& _node)
|
|||||||
{
|
{
|
||||||
auto indicesBeforeLoop = copyVariableIndices();
|
auto indicesBeforeLoop = copyVariableIndices();
|
||||||
auto touchedVars = touchedVariables(_node);
|
auto touchedVars = touchedVariables(_node);
|
||||||
resetVariables(touchedVars);
|
m_context.resetVariables(touchedVars);
|
||||||
decltype(indicesBeforeLoop) indicesAfterLoop;
|
decltype(indicesBeforeLoop) indicesAfterLoop;
|
||||||
if (_node.isDoWhile())
|
if (_node.isDoWhile())
|
||||||
{
|
{
|
||||||
@ -295,7 +295,7 @@ bool SMTChecker::visit(ForStatement const& _node)
|
|||||||
if (_node.loopExpression())
|
if (_node.loopExpression())
|
||||||
touchedVars += touchedVariables(*_node.loopExpression());
|
touchedVars += touchedVariables(*_node.loopExpression());
|
||||||
|
|
||||||
resetVariables(touchedVars);
|
m_context.resetVariables(touchedVars);
|
||||||
|
|
||||||
if (_node.condition())
|
if (_node.condition())
|
||||||
{
|
{
|
||||||
@ -341,13 +341,13 @@ void SMTChecker::endVisit(VariableDeclarationStatement const& _varDecl)
|
|||||||
for (unsigned i = 0; i < declarations.size(); ++i)
|
for (unsigned i = 0; i < declarations.size(); ++i)
|
||||||
{
|
{
|
||||||
solAssert(components.at(i), "");
|
solAssert(components.at(i), "");
|
||||||
if (declarations.at(i) && knownVariable(*declarations.at(i)))
|
if (declarations.at(i) && m_context.knownVariable(*declarations.at(i)))
|
||||||
assignment(*declarations.at(i), components.at(i)->currentValue(), declarations.at(i)->location());
|
assignment(*declarations.at(i), components.at(i)->currentValue(), declarations.at(i)->location());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (knownVariable(*_varDecl.declarations().front()))
|
else if (m_context.knownVariable(*_varDecl.declarations().front()))
|
||||||
{
|
{
|
||||||
if (_varDecl.initialValue())
|
if (_varDecl.initialValue())
|
||||||
assignment(*_varDecl.declarations().front(), *_varDecl.initialValue(), _varDecl.location());
|
assignment(*_varDecl.declarations().front(), *_varDecl.initialValue(), _varDecl.location());
|
||||||
@ -383,7 +383,7 @@ void SMTChecker::endVisit(Assignment const& _assignment)
|
|||||||
);
|
);
|
||||||
// Give it a new index anyway to keep the SSA scheme sound.
|
// Give it a new index anyway to keep the SSA scheme sound.
|
||||||
if (auto varDecl = identifierToVariable(_assignment.leftHandSide()))
|
if (auto varDecl = identifierToVariable(_assignment.leftHandSide()))
|
||||||
newValue(*varDecl);
|
m_context.newValue(*varDecl);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -431,7 +431,7 @@ void SMTChecker::endVisit(TupleExpression const& _tuple)
|
|||||||
if (component)
|
if (component)
|
||||||
{
|
{
|
||||||
if (auto varDecl = identifierToVariable(*component))
|
if (auto varDecl = identifierToVariable(*component))
|
||||||
components.push_back(m_variables[varDecl]);
|
components.push_back(m_context.variable(*varDecl));
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
solAssert(knownExpr(*component), "");
|
solAssert(knownExpr(*component), "");
|
||||||
@ -570,15 +570,15 @@ void SMTChecker::endVisit(UnaryOperation const& _op)
|
|||||||
auto const& subExpr = _op.subExpression();
|
auto const& subExpr = _op.subExpression();
|
||||||
if (auto decl = identifierToVariable(subExpr))
|
if (auto decl = identifierToVariable(subExpr))
|
||||||
{
|
{
|
||||||
newValue(*decl);
|
m_context.newValue(*decl);
|
||||||
setZeroValue(*decl);
|
m_context.setZeroValue(*decl);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
solAssert(knownExpr(subExpr), "");
|
solAssert(knownExpr(subExpr), "");
|
||||||
auto const& symbVar = m_expressions[&subExpr];
|
auto const& symbVar = m_expressions[&subExpr];
|
||||||
symbVar->increaseIndex();
|
symbVar->increaseIndex();
|
||||||
setZeroValue(*symbVar);
|
m_context.setZeroValue(*symbVar);
|
||||||
if (dynamic_cast<IndexAccess const*>(&_op.subExpression()))
|
if (dynamic_cast<IndexAccess const*>(&_op.subExpression()))
|
||||||
arrayIndexAssignment(_op.subExpression(), symbVar->currentValue());
|
arrayIndexAssignment(_op.subExpression(), symbVar->currentValue());
|
||||||
else
|
else
|
||||||
@ -735,7 +735,7 @@ void SMTChecker::visitGasLeft(FunctionCall const& _funCall)
|
|||||||
auto const& symbolicVar = m_globalContext.at(gasLeft);
|
auto const& symbolicVar = m_globalContext.at(gasLeft);
|
||||||
unsigned index = symbolicVar->index();
|
unsigned index = symbolicVar->index();
|
||||||
// We set the current value to unknown anyway to add type constraints.
|
// We set the current value to unknown anyway to add type constraints.
|
||||||
setUnknownValue(*symbolicVar);
|
m_context.setUnknownValue(*symbolicVar);
|
||||||
if (index > 0)
|
if (index > 0)
|
||||||
m_interface->addAssertion(symbolicVar->currentValue() <= symbolicVar->valueAtIndex(index - 1));
|
m_interface->addAssertion(symbolicVar->currentValue() <= symbolicVar->valueAtIndex(index - 1));
|
||||||
}
|
}
|
||||||
@ -788,8 +788,8 @@ void SMTChecker::inlineFunctionCall(FunctionCall const& _funCall)
|
|||||||
vector<shared_ptr<smt::SymbolicVariable>> components;
|
vector<shared_ptr<smt::SymbolicVariable>> components;
|
||||||
for (auto param: returnParams)
|
for (auto param: returnParams)
|
||||||
{
|
{
|
||||||
solAssert(m_variables[param.get()], "");
|
solAssert(m_context.variable(*param), "");
|
||||||
components.push_back(m_variables[param.get()]);
|
components.push_back(m_context.variable(*param));
|
||||||
}
|
}
|
||||||
auto const& symbTuple = dynamic_pointer_cast<smt::SymbolicTupleVariable>(m_expressions[&_funCall]);
|
auto const& symbTuple = dynamic_pointer_cast<smt::SymbolicTupleVariable>(m_expressions[&_funCall]);
|
||||||
solAssert(symbTuple, "");
|
solAssert(symbTuple, "");
|
||||||
@ -869,7 +869,7 @@ void SMTChecker::visitTypeConversion(FunctionCall const& _funCall)
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
createExpr(_funCall);
|
createExpr(_funCall);
|
||||||
setUnknownValue(*m_expressions.at(&_funCall));
|
m_context.setUnknownValue(*m_expressions.at(&_funCall));
|
||||||
auto const& funCallCategory = _funCall.annotation().type->category();
|
auto const& funCallCategory = _funCall.annotation().type->category();
|
||||||
// TODO: truncating and bytesX needs a different approach because of right padding.
|
// TODO: truncating and bytesX needs a different approach because of right padding.
|
||||||
if (funCallCategory == Type::Category::Integer || funCallCategory == Type::Category::Address)
|
if (funCallCategory == Type::Category::Integer || funCallCategory == Type::Category::Address)
|
||||||
@ -944,10 +944,10 @@ void SMTChecker::endVisit(Return const& _return)
|
|||||||
solAssert(components.size() == returnParams.size(), "");
|
solAssert(components.size() == returnParams.size(), "");
|
||||||
for (unsigned i = 0; i < returnParams.size(); ++i)
|
for (unsigned i = 0; i < returnParams.size(); ++i)
|
||||||
if (components.at(i))
|
if (components.at(i))
|
||||||
m_interface->addAssertion(expr(*components.at(i)) == newValue(*returnParams.at(i)));
|
m_interface->addAssertion(expr(*components.at(i)) == m_context.newValue(*returnParams.at(i)));
|
||||||
}
|
}
|
||||||
else if (returnParams.size() == 1)
|
else if (returnParams.size() == 1)
|
||||||
m_interface->addAssertion(expr(*_return.expression()) == newValue(*returnParams.front()));
|
m_interface->addAssertion(expr(*_return.expression()) == m_context.newValue(*returnParams.front()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1011,7 +1011,7 @@ void SMTChecker::endVisit(IndexAccess const& _indexAccess)
|
|||||||
{
|
{
|
||||||
auto varDecl = identifierToVariable(*id);
|
auto varDecl = identifierToVariable(*id);
|
||||||
solAssert(varDecl, "");
|
solAssert(varDecl, "");
|
||||||
array = m_variables[varDecl];
|
array = m_context.variable(*varDecl);
|
||||||
}
|
}
|
||||||
else if (auto const& innerAccess = dynamic_cast<IndexAccess const*>(&_indexAccess.baseExpression()))
|
else if (auto const& innerAccess = dynamic_cast<IndexAccess const*>(&_indexAccess.baseExpression()))
|
||||||
{
|
{
|
||||||
@ -1055,7 +1055,7 @@ void SMTChecker::arrayIndexAssignment(Expression const& _expr, smt::Expression c
|
|||||||
solAssert(varDecl, "");
|
solAssert(varDecl, "");
|
||||||
|
|
||||||
if (varDecl->hasReferenceOrMappingType())
|
if (varDecl->hasReferenceOrMappingType())
|
||||||
resetVariables([&](VariableDeclaration const& _var) {
|
m_context.resetVariables([&](VariableDeclaration const& _var) {
|
||||||
if (_var == *varDecl)
|
if (_var == *varDecl)
|
||||||
return false;
|
return false;
|
||||||
TypePointer prefix = _var.type();
|
TypePointer prefix = _var.type();
|
||||||
@ -1084,15 +1084,15 @@ void SMTChecker::arrayIndexAssignment(Expression const& _expr, smt::Expression c
|
|||||||
});
|
});
|
||||||
|
|
||||||
smt::Expression store = smt::Expression::store(
|
smt::Expression store = smt::Expression::store(
|
||||||
m_variables[varDecl]->currentValue(),
|
m_context.variable(*varDecl)->currentValue(),
|
||||||
expr(*indexAccess.indexExpression()),
|
expr(*indexAccess.indexExpression()),
|
||||||
_rightHandSide
|
_rightHandSide
|
||||||
);
|
);
|
||||||
m_interface->addAssertion(newValue(*varDecl) == store);
|
m_interface->addAssertion(m_context.newValue(*varDecl) == store);
|
||||||
// Update the SMT select value after the assignment,
|
// Update the SMT select value after the assignment,
|
||||||
// necessary for sound models.
|
// necessary for sound models.
|
||||||
defineExpr(indexAccess, smt::Expression::select(
|
defineExpr(indexAccess, smt::Expression::select(
|
||||||
m_variables[varDecl]->currentValue(),
|
m_context.variable(*varDecl)->currentValue(),
|
||||||
expr(*indexAccess.indexExpression())
|
expr(*indexAccess.indexExpression())
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
@ -1102,7 +1102,7 @@ void SMTChecker::arrayIndexAssignment(Expression const& _expr, smt::Expression c
|
|||||||
if (identifier)
|
if (identifier)
|
||||||
{
|
{
|
||||||
auto varDecl = identifierToVariable(*identifier);
|
auto varDecl = identifierToVariable(*identifier);
|
||||||
newValue(*varDecl);
|
m_context.newValue(*varDecl);
|
||||||
}
|
}
|
||||||
|
|
||||||
m_errorReporter.warning(
|
m_errorReporter.warning(
|
||||||
@ -1123,7 +1123,7 @@ void SMTChecker::defineGlobalVariable(string const& _name, Expression const& _ex
|
|||||||
{
|
{
|
||||||
auto result = smt::newSymbolicVariable(*_expr.annotation().type, _name, *m_interface);
|
auto result = smt::newSymbolicVariable(*_expr.annotation().type, _name, *m_interface);
|
||||||
m_globalContext.emplace(_name, result.second);
|
m_globalContext.emplace(_name, result.second);
|
||||||
setUnknownValue(*result.second);
|
m_context.setUnknownValue(*result.second);
|
||||||
if (result.first)
|
if (result.first)
|
||||||
m_errorReporter.warning(
|
m_errorReporter.warning(
|
||||||
_expr.location(),
|
_expr.location(),
|
||||||
@ -1414,7 +1414,7 @@ void SMTChecker::assignment(VariableDeclaration const& _variable, smt::Expressio
|
|||||||
addOverflowTarget(OverflowTarget::Type::All, TypeProvider::uint(160), _value, _location);
|
addOverflowTarget(OverflowTarget::Type::All, TypeProvider::uint(160), _value, _location);
|
||||||
else if (type->category() == Type::Category::Mapping)
|
else if (type->category() == Type::Category::Mapping)
|
||||||
arrayAssignment();
|
arrayAssignment();
|
||||||
m_interface->addAssertion(newValue(_variable) == _value);
|
m_interface->addAssertion(m_context.newValue(_variable) == _value);
|
||||||
}
|
}
|
||||||
|
|
||||||
SMTChecker::VariableIndices SMTChecker::visitBranch(ASTNode const* _statement, smt::Expression _condition)
|
SMTChecker::VariableIndices SMTChecker::visitBranch(ASTNode const* _statement, smt::Expression _condition)
|
||||||
@ -1456,7 +1456,7 @@ void SMTChecker::checkCondition(
|
|||||||
expressionsToEvaluate.emplace_back(*_additionalValue);
|
expressionsToEvaluate.emplace_back(*_additionalValue);
|
||||||
expressionNames.push_back(_additionalValueName);
|
expressionNames.push_back(_additionalValueName);
|
||||||
}
|
}
|
||||||
for (auto const& var: m_variables)
|
for (auto const& var: m_context.variables())
|
||||||
{
|
{
|
||||||
if (var.first->type()->isValueType())
|
if (var.first->type()->isValueType())
|
||||||
{
|
{
|
||||||
@ -1653,7 +1653,7 @@ void SMTChecker::initializeFunctionCallParameters(CallableDeclaration const& _fu
|
|||||||
for (unsigned i = 0; i < funParams.size(); ++i)
|
for (unsigned i = 0; i < funParams.size(); ++i)
|
||||||
if (createVariable(*funParams[i]))
|
if (createVariable(*funParams[i]))
|
||||||
{
|
{
|
||||||
m_interface->addAssertion(_callArgs[i] == newValue(*funParams[i]));
|
m_interface->addAssertion(_callArgs[i] == m_context.newValue(*funParams[i]));
|
||||||
if (funParams[i]->annotation().type->category() == Type::Category::Mapping)
|
if (funParams[i]->annotation().type->category() == Type::Category::Mapping)
|
||||||
m_arrayAssignmentHappened = true;
|
m_arrayAssignmentHappened = true;
|
||||||
}
|
}
|
||||||
@ -1661,16 +1661,16 @@ void SMTChecker::initializeFunctionCallParameters(CallableDeclaration const& _fu
|
|||||||
for (auto const& variable: _function.localVariables())
|
for (auto const& variable: _function.localVariables())
|
||||||
if (createVariable(*variable))
|
if (createVariable(*variable))
|
||||||
{
|
{
|
||||||
newValue(*variable);
|
m_context.newValue(*variable);
|
||||||
setZeroValue(*variable);
|
m_context.setZeroValue(*variable);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_function.returnParameterList())
|
if (_function.returnParameterList())
|
||||||
for (auto const& retParam: _function.returnParameters())
|
for (auto const& retParam: _function.returnParameters())
|
||||||
if (createVariable(*retParam))
|
if (createVariable(*retParam))
|
||||||
{
|
{
|
||||||
newValue(*retParam);
|
m_context.newValue(*retParam);
|
||||||
setZeroValue(*retParam);
|
m_context.setZeroValue(*retParam);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1678,58 +1678,26 @@ void SMTChecker::initializeLocalVariables(FunctionDefinition const& _function)
|
|||||||
{
|
{
|
||||||
for (auto const& variable: _function.localVariables())
|
for (auto const& variable: _function.localVariables())
|
||||||
if (createVariable(*variable))
|
if (createVariable(*variable))
|
||||||
setZeroValue(*variable);
|
m_context.setZeroValue(*variable);
|
||||||
|
|
||||||
for (auto const& param: _function.parameters())
|
for (auto const& param: _function.parameters())
|
||||||
if (createVariable(*param))
|
if (createVariable(*param))
|
||||||
setUnknownValue(*param);
|
m_context.setUnknownValue(*param);
|
||||||
|
|
||||||
if (_function.returnParameterList())
|
if (_function.returnParameterList())
|
||||||
for (auto const& retParam: _function.returnParameters())
|
for (auto const& retParam: _function.returnParameters())
|
||||||
if (createVariable(*retParam))
|
if (createVariable(*retParam))
|
||||||
setZeroValue(*retParam);
|
m_context.setZeroValue(*retParam);
|
||||||
}
|
|
||||||
|
|
||||||
void SMTChecker::removeLocalVariables()
|
|
||||||
{
|
|
||||||
for (auto it = m_variables.begin(); it != m_variables.end(); )
|
|
||||||
{
|
|
||||||
if (it->first->isLocalVariable())
|
|
||||||
it = m_variables.erase(it);
|
|
||||||
else
|
|
||||||
++it;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void SMTChecker::resetVariable(VariableDeclaration const& _variable)
|
|
||||||
{
|
|
||||||
newValue(_variable);
|
|
||||||
setUnknownValue(_variable);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void SMTChecker::resetStateVariables()
|
void SMTChecker::resetStateVariables()
|
||||||
{
|
{
|
||||||
resetVariables([&](VariableDeclaration const& _variable) { return _variable.isStateVariable(); });
|
m_context.resetVariables([&](VariableDeclaration const& _variable) { return _variable.isStateVariable(); });
|
||||||
}
|
}
|
||||||
|
|
||||||
void SMTChecker::resetStorageReferences()
|
void SMTChecker::resetStorageReferences()
|
||||||
{
|
{
|
||||||
resetVariables([&](VariableDeclaration const& _variable) { return _variable.hasReferenceOrMappingType(); });
|
m_context.resetVariables([&](VariableDeclaration const& _variable) { return _variable.hasReferenceOrMappingType(); });
|
||||||
}
|
|
||||||
|
|
||||||
void SMTChecker::resetVariables(set<VariableDeclaration const*> const& _variables)
|
|
||||||
{
|
|
||||||
for (auto const* decl: _variables)
|
|
||||||
resetVariable(*decl);
|
|
||||||
}
|
|
||||||
|
|
||||||
void SMTChecker::resetVariables(function<bool(VariableDeclaration const&)> const& _filter)
|
|
||||||
{
|
|
||||||
for_each(begin(m_variables), end(m_variables), [&](auto _variable)
|
|
||||||
{
|
|
||||||
if (_filter(*_variable.first))
|
|
||||||
this->resetVariable(*_variable.first);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TypePointer SMTChecker::typeWithoutPointer(TypePointer const& _type)
|
TypePointer SMTChecker::typeWithoutPointer(TypePointer const& _type)
|
||||||
@ -1751,7 +1719,7 @@ void SMTChecker::mergeVariables(set<VariableDeclaration const*> const& _variable
|
|||||||
int trueIndex = _indicesEndTrue.at(decl);
|
int trueIndex = _indicesEndTrue.at(decl);
|
||||||
int falseIndex = _indicesEndFalse.at(decl);
|
int falseIndex = _indicesEndFalse.at(decl);
|
||||||
solAssert(trueIndex != falseIndex, "");
|
solAssert(trueIndex != falseIndex, "");
|
||||||
m_interface->addAssertion(newValue(*decl) == smt::Expression::ite(
|
m_interface->addAssertion(m_context.newValue(*decl) == smt::Expression::ite(
|
||||||
_condition,
|
_condition,
|
||||||
valueAtIndex(*decl, trueIndex),
|
valueAtIndex(*decl, trueIndex),
|
||||||
valueAtIndex(*decl, falseIndex))
|
valueAtIndex(*decl, falseIndex))
|
||||||
@ -1759,16 +1727,24 @@ void SMTChecker::mergeVariables(set<VariableDeclaration const*> const& _variable
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
smt::Expression SMTChecker::currentValue(VariableDeclaration const& _decl)
|
||||||
|
{
|
||||||
|
solAssert(m_context.knownVariable(_decl), "");
|
||||||
|
return m_context.variable(_decl)->currentValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
smt::Expression SMTChecker::valueAtIndex(VariableDeclaration const& _decl, int _index)
|
||||||
|
{
|
||||||
|
solAssert(m_context.knownVariable(_decl), "");
|
||||||
|
return m_context.variable(_decl)->valueAtIndex(_index);
|
||||||
|
}
|
||||||
|
|
||||||
bool SMTChecker::createVariable(VariableDeclaration const& _varDecl)
|
bool SMTChecker::createVariable(VariableDeclaration const& _varDecl)
|
||||||
{
|
{
|
||||||
// This might be the case for multiple calls to the same function.
|
if (m_context.knownVariable(_varDecl))
|
||||||
if (knownVariable(_varDecl))
|
|
||||||
return true;
|
return true;
|
||||||
auto const& type = _varDecl.type();
|
bool abstract = m_context.createVariable(_varDecl);
|
||||||
solAssert(m_variables.count(&_varDecl) == 0, "");
|
if (abstract)
|
||||||
auto result = smt::newSymbolicVariable(*type, _varDecl.name() + "_" + to_string(_varDecl.id()), *m_interface);
|
|
||||||
m_variables.emplace(&_varDecl, result.second);
|
|
||||||
if (result.first)
|
|
||||||
{
|
{
|
||||||
m_errorReporter.warning(
|
m_errorReporter.warning(
|
||||||
_varDecl.location(),
|
_varDecl.location(),
|
||||||
@ -1779,51 +1755,6 @@ bool SMTChecker::createVariable(VariableDeclaration const& _varDecl)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SMTChecker::knownVariable(VariableDeclaration const& _decl)
|
|
||||||
{
|
|
||||||
return m_variables.count(&_decl);
|
|
||||||
}
|
|
||||||
|
|
||||||
smt::Expression SMTChecker::currentValue(VariableDeclaration const& _decl)
|
|
||||||
{
|
|
||||||
solAssert(knownVariable(_decl), "");
|
|
||||||
return m_variables.at(&_decl)->currentValue();
|
|
||||||
}
|
|
||||||
|
|
||||||
smt::Expression SMTChecker::valueAtIndex(VariableDeclaration const& _decl, int _index)
|
|
||||||
{
|
|
||||||
solAssert(knownVariable(_decl), "");
|
|
||||||
return m_variables.at(&_decl)->valueAtIndex(_index);
|
|
||||||
}
|
|
||||||
|
|
||||||
smt::Expression SMTChecker::newValue(VariableDeclaration const& _decl)
|
|
||||||
{
|
|
||||||
solAssert(knownVariable(_decl), "");
|
|
||||||
return m_variables.at(&_decl)->increaseIndex();
|
|
||||||
}
|
|
||||||
|
|
||||||
void SMTChecker::setZeroValue(VariableDeclaration const& _decl)
|
|
||||||
{
|
|
||||||
solAssert(knownVariable(_decl), "");
|
|
||||||
setZeroValue(*m_variables.at(&_decl));
|
|
||||||
}
|
|
||||||
|
|
||||||
void SMTChecker::setZeroValue(smt::SymbolicVariable& _variable)
|
|
||||||
{
|
|
||||||
smt::setSymbolicZeroValue(_variable, *m_interface);
|
|
||||||
}
|
|
||||||
|
|
||||||
void SMTChecker::setUnknownValue(VariableDeclaration const& _decl)
|
|
||||||
{
|
|
||||||
solAssert(knownVariable(_decl), "");
|
|
||||||
setUnknownValue(*m_variables.at(&_decl));
|
|
||||||
}
|
|
||||||
|
|
||||||
void SMTChecker::setUnknownValue(smt::SymbolicVariable& _variable)
|
|
||||||
{
|
|
||||||
smt::setSymbolicUnknownValue(_variable, *m_interface);
|
|
||||||
}
|
|
||||||
|
|
||||||
smt::Expression SMTChecker::expr(Expression const& _e)
|
smt::Expression SMTChecker::expr(Expression const& _e)
|
||||||
{
|
{
|
||||||
if (!knownExpr(_e))
|
if (!knownExpr(_e))
|
||||||
@ -1938,7 +1869,7 @@ bool SMTChecker::visitedFunction(FunctionDefinition const* _funDef)
|
|||||||
SMTChecker::VariableIndices SMTChecker::copyVariableIndices()
|
SMTChecker::VariableIndices SMTChecker::copyVariableIndices()
|
||||||
{
|
{
|
||||||
VariableIndices indices;
|
VariableIndices indices;
|
||||||
for (auto const& var: m_variables)
|
for (auto const& var: m_context.variables())
|
||||||
indices.emplace(var.first, var.second->index());
|
indices.emplace(var.first, var.second->index());
|
||||||
return indices;
|
return indices;
|
||||||
}
|
}
|
||||||
@ -1946,7 +1877,7 @@ SMTChecker::VariableIndices SMTChecker::copyVariableIndices()
|
|||||||
void SMTChecker::resetVariableIndices(VariableIndices const& _indices)
|
void SMTChecker::resetVariableIndices(VariableIndices const& _indices)
|
||||||
{
|
{
|
||||||
for (auto const& var: _indices)
|
for (auto const& var: _indices)
|
||||||
m_variables.at(var.first)->index() = var.second;
|
m_context.variable(*var.first)->index() = var.second;
|
||||||
}
|
}
|
||||||
|
|
||||||
FunctionDefinition const* SMTChecker::inlinedFunctionCallToDefinition(FunctionCall const& _funCall)
|
FunctionDefinition const* SMTChecker::inlinedFunctionCallToDefinition(FunctionCall const& _funCall)
|
||||||
@ -2015,7 +1946,7 @@ VariableDeclaration const* SMTChecker::identifierToVariable(Expression const& _e
|
|||||||
{
|
{
|
||||||
if (auto decl = dynamic_cast<VariableDeclaration const*>(identifier->annotation().referencedDeclaration))
|
if (auto decl = dynamic_cast<VariableDeclaration const*>(identifier->annotation().referencedDeclaration))
|
||||||
{
|
{
|
||||||
solAssert(knownVariable(*decl), "");
|
solAssert(m_context.knownVariable(*decl), "");
|
||||||
return decl;
|
return decl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -216,11 +216,8 @@ private:
|
|||||||
|
|
||||||
void initializeLocalVariables(FunctionDefinition const& _function);
|
void initializeLocalVariables(FunctionDefinition const& _function);
|
||||||
void initializeFunctionCallParameters(CallableDeclaration const& _function, std::vector<smt::Expression> const& _callArgs);
|
void initializeFunctionCallParameters(CallableDeclaration const& _function, std::vector<smt::Expression> const& _callArgs);
|
||||||
void resetVariable(VariableDeclaration const& _variable);
|
|
||||||
void resetStateVariables();
|
void resetStateVariables();
|
||||||
void resetStorageReferences();
|
void resetStorageReferences();
|
||||||
void resetVariables(std::set<VariableDeclaration const*> const& _variables);
|
|
||||||
void resetVariables(std::function<bool(VariableDeclaration const&)> const& _filter);
|
|
||||||
/// @returns the type without storage pointer information if it has it.
|
/// @returns the type without storage pointer information if it has it.
|
||||||
TypePointer typeWithoutPointer(TypePointer const& _type);
|
TypePointer typeWithoutPointer(TypePointer const& _type);
|
||||||
|
|
||||||
@ -229,29 +226,14 @@ private:
|
|||||||
/// using the branch condition as guard.
|
/// using the branch condition as guard.
|
||||||
void mergeVariables(std::set<VariableDeclaration const*> const& _variables, smt::Expression const& _condition, VariableIndices const& _indicesEndTrue, VariableIndices const& _indicesEndFalse);
|
void mergeVariables(std::set<VariableDeclaration const*> const& _variables, smt::Expression const& _condition, VariableIndices const& _indicesEndTrue, VariableIndices const& _indicesEndFalse);
|
||||||
/// Tries to create an uninitialized variable and returns true on success.
|
/// Tries to create an uninitialized variable and returns true on success.
|
||||||
/// This fails if the type is not supported.
|
|
||||||
bool createVariable(VariableDeclaration const& _varDecl);
|
bool createVariable(VariableDeclaration const& _varDecl);
|
||||||
|
|
||||||
/// @returns true if _delc is a variable that is known at the current point, i.e.
|
|
||||||
/// has a valid index
|
|
||||||
bool knownVariable(VariableDeclaration const& _decl);
|
|
||||||
/// @returns an expression denoting the value of the variable declared in @a _decl
|
/// @returns an expression denoting the value of the variable declared in @a _decl
|
||||||
/// at the current point.
|
/// at the current point.
|
||||||
smt::Expression currentValue(VariableDeclaration const& _decl);
|
smt::Expression currentValue(VariableDeclaration const& _decl);
|
||||||
/// @returns an expression denoting the value of the variable declared in @a _decl
|
/// @returns an expression denoting the value of the variable declared in @a _decl
|
||||||
/// at the given index. Does not ensure that this index exists.
|
/// at the given index. Does not ensure that this index exists.
|
||||||
smt::Expression valueAtIndex(VariableDeclaration const& _decl, int _index);
|
smt::Expression valueAtIndex(VariableDeclaration const& _decl, int _index);
|
||||||
/// Allocates a new index for the declaration, updates the current
|
|
||||||
/// index to this value and returns the expression.
|
|
||||||
smt::Expression newValue(VariableDeclaration const& _decl);
|
|
||||||
|
|
||||||
/// Sets the value of the declaration to zero.
|
|
||||||
void setZeroValue(VariableDeclaration const& _decl);
|
|
||||||
void setZeroValue(smt::SymbolicVariable& _variable);
|
|
||||||
/// Resets the variable to an unknown value (in its range).
|
|
||||||
void setUnknownValue(VariableDeclaration const& decl);
|
|
||||||
void setUnknownValue(smt::SymbolicVariable& _variable);
|
|
||||||
|
|
||||||
/// Returns the expression corresponding to the AST node. Throws if the expression does not exist.
|
/// Returns the expression corresponding to the AST node. Throws if the expression does not exist.
|
||||||
smt::Expression expr(Expression const& _e);
|
smt::Expression expr(Expression const& _e);
|
||||||
/// Creates the expression (value can be arbitrary)
|
/// Creates the expression (value can be arbitrary)
|
||||||
@ -281,9 +263,6 @@ private:
|
|||||||
/// Add to the solver: the given expression implied by the current path conditions
|
/// Add to the solver: the given expression implied by the current path conditions
|
||||||
void addPathImpliedExpression(smt::Expression const& _e);
|
void addPathImpliedExpression(smt::Expression const& _e);
|
||||||
|
|
||||||
/// Removes local variables from the context.
|
|
||||||
void removeLocalVariables();
|
|
||||||
|
|
||||||
/// Copy the SSA indices of m_variables.
|
/// Copy the SSA indices of m_variables.
|
||||||
VariableIndices copyVariableIndices();
|
VariableIndices copyVariableIndices();
|
||||||
/// Resets the variable indices.
|
/// Resets the variable indices.
|
||||||
@ -305,7 +284,6 @@ private:
|
|||||||
/// An Expression may have multiple smt::Expression due to
|
/// An Expression may have multiple smt::Expression due to
|
||||||
/// repeated calls to the same function.
|
/// repeated calls to the same function.
|
||||||
std::unordered_map<Expression const*, std::shared_ptr<smt::SymbolicVariable>> m_expressions;
|
std::unordered_map<Expression const*, std::shared_ptr<smt::SymbolicVariable>> m_expressions;
|
||||||
std::unordered_map<VariableDeclaration const*, std::shared_ptr<smt::SymbolicVariable>> m_variables;
|
|
||||||
std::unordered_map<std::string, std::shared_ptr<smt::SymbolicVariable>> m_globalContext;
|
std::unordered_map<std::string, std::shared_ptr<smt::SymbolicVariable>> m_globalContext;
|
||||||
|
|
||||||
/// Stores the instances of an Uninterpreted Function applied to arguments.
|
/// Stores the instances of an Uninterpreted Function applied to arguments.
|
||||||
|
Loading…
Reference in New Issue
Block a user