/* 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 . */ #pragma once #include #include #include #include namespace dev { namespace solidity { namespace smt { /** * Stores the context of the SMT encoding. */ class EncodingContext { public: EncodingContext(); /// Resets the entire context except for symbolic variables which stay /// alive because of state variables and inlined function calls. /// To be used in the beginning of a root function visit. void reset(); /// Clears the entire context, erasing everything. /// To be used before a model checking engine starts. void clear(); /// Sets the current solver used by the current engine for /// SMT variable declaration. void setSolver(std::shared_ptr _solver) { solAssert(_solver, ""); m_solver = _solver; } /// Forwards variable creation to the solver. Expression newVariable(std::string _name, SortPointer _sort) { solAssert(m_solver, ""); return m_solver->newVariable(move(_name), move(_sort)); } /// Variables. //@{ /// @returns the symbolic representation of a program variable. std::shared_ptr variable(solidity::VariableDeclaration const& _varDecl); /// @returns all symbolic variables. std::unordered_map> 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 const& _variables); /// Resets variables according to a predicate. void resetVariables(std::function 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); //@} /// Expressions. ////@{ /// @returns the symbolic representation of an AST node expression. std::shared_ptr expression(solidity::Expression const& _e); /// @returns all symbolic expressions. std::unordered_map> const& expressions() const { return m_expressions; } /// Creates the expression (value can be arbitrary). /// @returns true if type is not supported. bool createExpression(solidity::Expression const& _e, std::shared_ptr _symbExpr = nullptr); /// Checks if expression was created. bool knownExpression(solidity::Expression const& _e) const; //@} /// Global variables and functions. //@{ /// Global variables and functions. std::shared_ptr globalSymbol(std::string const& _name); /// @returns all symbolic globals. std::unordered_map> const& globalSymbols() const { return m_globalContext; } /// Defines a new global variable or function /// and @returns true if type was abstracted. bool createGlobalSymbol(std::string const& _name, solidity::Expression const& _expr); /// Checks if special variable or function was seen. bool knownGlobalSymbol(std::string const& _var) const; //@} /// Blockchain. //@{ /// Value of `this` address. Expression thisAddress(); /// @returns the symbolic balance of address `this`. Expression balance(); /// @returns the symbolic balance of an address. Expression balance(Expression _address); /// Transfer _value from _from to _to. void transfer(Expression _from, Expression _to, Expression _value); //@} /// Solver. //@{ /// @returns conjunction of all added assertions. Expression assertions(); void pushSolver(); void popSolver(); void addAssertion(Expression const& _e); std::shared_ptr solver() { solAssert(m_solver, ""); return m_solver; } //@} private: /// Adds _value to _account's balance. void addBalance(Expression _account, Expression _value); /// Symbolic expressions. //{@ /// Symbolic variables. std::unordered_map> m_variables; /// Symbolic expressions. std::unordered_map> m_expressions; /// Symbolic representation of global symbols including /// variables and functions. std::unordered_map> m_globalContext; /// Symbolic `this` address. std::unique_ptr m_thisAddress; /// Symbolic balances. std::unique_ptr m_balances; //@} /// Solver related. //@{ /// Solver can be SMT solver or Horn solver in the future. std::shared_ptr m_solver; /// Assertion stack. std::vector m_assertions; //@} }; } } }