mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
[SMTChecker] Support tuple type declaration
This commit is contained in:
parent
80f3bd2413
commit
6c7527ac90
@ -5,6 +5,7 @@ Language Features:
|
|||||||
|
|
||||||
Compiler Features:
|
Compiler Features:
|
||||||
* SMTChecker: Support inherited state variables.
|
* SMTChecker: Support inherited state variables.
|
||||||
|
* SMTChecker: Support tuple declarations.
|
||||||
|
|
||||||
|
|
||||||
Bugfixes:
|
Bugfixes:
|
||||||
|
@ -383,17 +383,41 @@ void SMTChecker::endVisit(Assignment const& _assignment)
|
|||||||
|
|
||||||
void SMTChecker::endVisit(TupleExpression const& _tuple)
|
void SMTChecker::endVisit(TupleExpression const& _tuple)
|
||||||
{
|
{
|
||||||
if (
|
if (_tuple.isInlineArray())
|
||||||
_tuple.isInlineArray() ||
|
|
||||||
_tuple.components().size() != 1 ||
|
|
||||||
!isSupportedType(_tuple.components()[0]->annotation().type->category())
|
|
||||||
)
|
|
||||||
m_errorReporter.warning(
|
m_errorReporter.warning(
|
||||||
_tuple.location(),
|
_tuple.location(),
|
||||||
"Assertion checker does not yet implement tuples and inline arrays."
|
"Assertion checker does not yet implement inline arrays."
|
||||||
);
|
);
|
||||||
|
else if (_tuple.annotation().type->category() == Type::Category::Tuple)
|
||||||
|
{
|
||||||
|
createExpr(_tuple);
|
||||||
|
vector<shared_ptr<SymbolicVariable>> components;
|
||||||
|
for (auto const& component: _tuple.components())
|
||||||
|
if (component)
|
||||||
|
{
|
||||||
|
if (auto varDecl = identifierToVariable(*component))
|
||||||
|
components.push_back(m_variables[varDecl]);
|
||||||
else
|
else
|
||||||
defineExpr(_tuple, expr(*_tuple.components()[0]));
|
{
|
||||||
|
solAssert(knownExpr(*component), "");
|
||||||
|
components.push_back(m_expressions[component.get()]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
components.push_back(nullptr);
|
||||||
|
solAssert(components.size() == _tuple.components().size(), "");
|
||||||
|
auto const& symbTuple = dynamic_pointer_cast<SymbolicTupleVariable>(m_expressions[&_tuple]);
|
||||||
|
solAssert(symbTuple, "");
|
||||||
|
symbTuple->setComponents(move(components));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/// Parenthesized expressions are also TupleExpression regardless their type.
|
||||||
|
auto const& components = _tuple.components();
|
||||||
|
solAssert(components.size() == 1, "");
|
||||||
|
if (isSupportedType(components.front()->annotation().type->category()))
|
||||||
|
defineExpr(_tuple, expr(*components.front()));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void SMTChecker::addOverflowTarget(
|
void SMTChecker::addOverflowTarget(
|
||||||
|
@ -99,7 +99,8 @@ bool dev::solidity::isSupportedType(Type::Category _category)
|
|||||||
return isNumber(_category) ||
|
return isNumber(_category) ||
|
||||||
isBool(_category) ||
|
isBool(_category) ||
|
||||||
isMapping(_category) ||
|
isMapping(_category) ||
|
||||||
isArray(_category);
|
isArray(_category) ||
|
||||||
|
isTuple(_category);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool dev::solidity::isSupportedTypeDeclaration(Type::Category _category)
|
bool dev::solidity::isSupportedTypeDeclaration(Type::Category _category)
|
||||||
@ -151,6 +152,8 @@ pair<bool, shared_ptr<SymbolicVariable>> dev::solidity::newSymbolicVariable(
|
|||||||
var = make_shared<SymbolicMappingVariable>(type, _uniqueName, _solver);
|
var = make_shared<SymbolicMappingVariable>(type, _uniqueName, _solver);
|
||||||
else if (isArray(_type.category()))
|
else if (isArray(_type.category()))
|
||||||
var = make_shared<SymbolicArrayVariable>(type, _uniqueName, _solver);
|
var = make_shared<SymbolicArrayVariable>(type, _uniqueName, _solver);
|
||||||
|
else if (isTuple(_type.category()))
|
||||||
|
var = make_shared<SymbolicTupleVariable>(type, _uniqueName, _solver);
|
||||||
else
|
else
|
||||||
solAssert(false, "");
|
solAssert(false, "");
|
||||||
return make_pair(abstract, var);
|
return make_pair(abstract, var);
|
||||||
@ -226,6 +229,11 @@ bool dev::solidity::isArray(Type::Category _category)
|
|||||||
return _category == Type::Category::Array;
|
return _category == Type::Category::Array;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool dev::solidity::isTuple(Type::Category _category)
|
||||||
|
{
|
||||||
|
return _category == Type::Category::Tuple;
|
||||||
|
}
|
||||||
|
|
||||||
smt::Expression dev::solidity::minValue(IntegerType const& _type)
|
smt::Expression dev::solidity::minValue(IntegerType const& _type)
|
||||||
{
|
{
|
||||||
return smt::Expression(_type.minValue());
|
return smt::Expression(_type.minValue());
|
||||||
|
@ -51,6 +51,7 @@ bool isBool(Type::Category _category);
|
|||||||
bool isFunction(Type::Category _category);
|
bool isFunction(Type::Category _category);
|
||||||
bool isMapping(Type::Category _category);
|
bool isMapping(Type::Category _category);
|
||||||
bool isArray(Type::Category _category);
|
bool isArray(Type::Category _category);
|
||||||
|
bool isTuple(Type::Category _category);
|
||||||
|
|
||||||
/// Returns a new symbolic variable, according to _type.
|
/// Returns a new symbolic variable, according to _type.
|
||||||
/// Also returns whether the type is abstract or not,
|
/// Also returns whether the type is abstract or not,
|
||||||
|
@ -173,3 +173,21 @@ SymbolicEnumVariable::SymbolicEnumVariable(
|
|||||||
{
|
{
|
||||||
solAssert(isEnum(m_type->category()), "");
|
solAssert(isEnum(m_type->category()), "");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SymbolicTupleVariable::SymbolicTupleVariable(
|
||||||
|
TypePointer _type,
|
||||||
|
string _uniqueName,
|
||||||
|
smt::SolverInterface& _interface
|
||||||
|
):
|
||||||
|
SymbolicVariable(move(_type), move(_uniqueName), _interface)
|
||||||
|
{
|
||||||
|
solAssert(isTuple(m_type->category()), "");
|
||||||
|
}
|
||||||
|
|
||||||
|
void SymbolicTupleVariable::setComponents(vector<shared_ptr<SymbolicVariable>> _components)
|
||||||
|
{
|
||||||
|
solAssert(m_components.empty(), "");
|
||||||
|
auto const& tupleType = dynamic_cast<TupleType const*>(m_type);
|
||||||
|
solAssert(_components.size() == tupleType->components().size(), "");
|
||||||
|
m_components = move(_components);
|
||||||
|
}
|
||||||
|
@ -187,5 +187,28 @@ public:
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specialization of SymbolicVariable for Tuple
|
||||||
|
*/
|
||||||
|
class SymbolicTupleVariable: public SymbolicVariable
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
SymbolicTupleVariable(
|
||||||
|
TypePointer _type,
|
||||||
|
std::string _uniqueName,
|
||||||
|
smt::SolverInterface& _interface
|
||||||
|
);
|
||||||
|
|
||||||
|
std::vector<std::shared_ptr<SymbolicVariable>> const& components()
|
||||||
|
{
|
||||||
|
return m_components;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setComponents(std::vector<std::shared_ptr<SymbolicVariable>> _components);
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<std::shared_ptr<SymbolicVariable>> m_components;
|
||||||
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -12,4 +12,3 @@ contract C
|
|||||||
}
|
}
|
||||||
|
|
||||||
// ----
|
// ----
|
||||||
// Warning: (153-156): Assertion checker does not yet implement tuples and inline arrays.
|
|
||||||
|
@ -12,5 +12,4 @@ contract C
|
|||||||
}
|
}
|
||||||
|
|
||||||
// ----
|
// ----
|
||||||
// Warning: (153-156): Assertion checker does not yet implement tuples and inline arrays.
|
|
||||||
// Warning: (163-176): Assertion violation happens here
|
// Warning: (163-176): Assertion violation happens here
|
||||||
|
@ -12,5 +12,4 @@ contract C
|
|||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// Warning: (131-146): Insufficient funds happens here
|
// Warning: (131-146): Insufficient funds happens here
|
||||||
// Warning: (131-146): Assertion checker does not yet implement this type.
|
|
||||||
// Warning: (195-219): Assertion violation happens here
|
// Warning: (195-219): Assertion violation happens here
|
||||||
|
@ -15,7 +15,5 @@ contract C
|
|||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// Warning: (217-232): Insufficient funds happens here
|
// Warning: (217-232): Insufficient funds happens here
|
||||||
// Warning: (217-232): Assertion checker does not yet implement this type.
|
|
||||||
// Warning: (236-251): Insufficient funds happens here
|
// Warning: (236-251): Insufficient funds happens here
|
||||||
// Warning: (236-251): Assertion checker does not yet implement this type.
|
|
||||||
// Warning: (295-324): Assertion violation happens here
|
// Warning: (295-324): Assertion violation happens here
|
||||||
|
@ -12,7 +12,5 @@ contract C
|
|||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// Warning: (134-149): Insufficient funds happens here
|
// Warning: (134-149): Insufficient funds happens here
|
||||||
// Warning: (134-149): Assertion checker does not yet implement this type.
|
|
||||||
// Warning: (153-169): Insufficient funds happens here
|
// Warning: (153-169): Insufficient funds happens here
|
||||||
// Warning: (153-169): Assertion checker does not yet implement this type.
|
|
||||||
// Warning: (213-237): Assertion violation happens here
|
// Warning: (213-237): Assertion violation happens here
|
||||||
|
@ -8,5 +8,5 @@ contract C
|
|||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// Warning: (76-96): Unused local variable.
|
// Warning: (76-96): Unused local variable.
|
||||||
// Warning: (99-114): Assertion checker does not yet implement tuples and inline arrays.
|
// Warning: (99-114): Assertion checker does not yet implement inline arrays.
|
||||||
// Warning: (99-114): Internal error: Expression undefined for SMT solver.
|
// Warning: (99-114): Internal error: Expression undefined for SMT solver.
|
||||||
|
@ -0,0 +1,13 @@
|
|||||||
|
pragma experimental SMTChecker;
|
||||||
|
|
||||||
|
contract C
|
||||||
|
{
|
||||||
|
function f() public pure {
|
||||||
|
uint a = 1;
|
||||||
|
uint b = 3;
|
||||||
|
a += ((((b))));
|
||||||
|
assert(a == 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// Warning: (122-136): Assertion violation happens here
|
Loading…
Reference in New Issue
Block a user