/* This file is part of cpp-ethereum. cpp-ethereum 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. cpp-ethereum 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 cpp-ethereum. If not, see . */ /** @file Executive.h * @author Gav Wood * @date 2014 */ #pragma once #include #include #include #include #include #include "Transaction.h" namespace Json { class Value; } namespace dev { class OverlayDB; namespace eth { class State; class Block; class BlockChain; class ExtVM; struct Manifest; struct VMTraceChannel: public LogChannel { static const char* name(); static const int verbosity = 11; }; struct ExecutiveWarnChannel: public LogChannel { static const char* name(); static const int verbosity = 6; }; class StandardTrace { public: StandardTrace(); void operator()(uint64_t _steps, Instruction _inst, bigint _newMemSize, bigint _gasCost, bigint _gas, VM* _vm, ExtVMFace const* _extVM); void setShowMnemonics() { m_showMnemonics = true; } std::string json(bool _styled = false) const; OnOpFunc onOp() { return [=](uint64_t _steps, Instruction _inst, bigint _newMemSize, bigint _gasCost, bigint _gas, VM* _vm, ExtVMFace const* _extVM) { (*this)(_steps, _inst, _newMemSize, _gasCost, _gas, _vm, _extVM); }; } private: bool m_showMnemonics = false; std::vector m_lastInst; std::shared_ptr m_trace; }; /** * @brief Message-call/contract-creation executor; useful for executing transactions. * * Two ways of using this class - either as a transaction executive or a CALL/CREATE executive. * * In the first use, after construction, begin with initialize(), then execute() and end with finalize(). Call go() * after execute() only if it returns false. * * In the second use, after construction, begin with call() or create() and end with * accrueSubState(). Call go() after call()/create() only if it returns false. * * Example: * @code * Executive e(state, blockchain, 0); * e.initialize(transaction); * if (!e.execute()) * e.go(); * e.finalize(); * @endcode */ class Executive { public: /// Simple constructor; executive will operate on given state, with the given environment info. Executive(State& _s, EnvInfo const& _envInfo, unsigned _level = 0): m_s(_s), m_envInfo(_envInfo), m_depth(_level) {} /** Easiest constructor. * Creates executive to operate on the state of end of the given block, populating environment * info from given Block and the LastHashes portion from the BlockChain. */ Executive(Block& _s, BlockChain const& _bc, unsigned _level = 0); /** LastHashes-split constructor. * Creates executive to operate on the state of end of the given block, populating environment * info accordingly, with last hashes given explicitly. */ Executive(Block& _s, LastHashes const& _lh = LastHashes(), unsigned _level = 0); /** Partially-automated split constructor; executive will operate on given state, with the given * environment info, populating the last hashes from the given chain. */ Executive(State& _s, BlockChain const& _bc, EnvInfo const& _envInfo, unsigned _level = 0); /** * Automated split constructor; executive will operate on given state, with the environment info * populated from the given chain. * @note This will only work when the state to be operated on is already in the chain. */ Executive(State& _s, BlockChain const& _bc, unsigned _number, unsigned _level = 0); Executive(Executive const&) = delete; void operator=(Executive) = delete; /// Initializes the executive for evaluating a transaction. You must call finalize() at some point following this. void initialize(bytesConstRef _transaction) { initialize(Transaction(_transaction, CheckTransaction::None)); } void initialize(Transaction const& _transaction); /// Finalise a transaction previously set up with initialize(). /// @warning Only valid after initialize() and execute(), and possibly go(). void finalize(); /// Begins execution of a transaction. You must call finalize() following this. /// @returns true if the transaction is done, false if go() must be called. bool execute(); /// @returns the transaction from initialize(). /// @warning Only valid after initialize(). Transaction const& t() const { return m_t; } /// @returns the log entries created by this operation. /// @warning Only valid after finalise(). LogEntries const& logs() const { return m_logs; } /// @returns total gas used in the transaction/operation. /// @warning Only valid after finalise(). u256 gasUsed() const; /// @returns total gas used in the transaction/operation, excluding anything refunded. /// @warning Only valid after finalise(). u256 gasUsedNoRefunds() const; /// Set up the executive for evaluating a bare CREATE (contract-creation) operation. /// @returns false iff go() must be called (and thus a VM execution in required). bool create(Address _txSender, u256 _endowment, u256 _gasPrice, u256 _gas, bytesConstRef _code, Address _originAddress); /// Set up the executive for evaluating a bare CALL (message call) operation. /// @returns false iff go() must be called (and thus a VM execution in required). bool call(Address _receiveAddress, Address _txSender, u256 _txValue, u256 _gasPrice, bytesConstRef _txData, u256 _gas); bool call(CallParameters const& _cp, u256 const& _gasPrice, Address const& _origin); /// Finalise an operation through accruing the substate into the parent context. void accrueSubState(SubState& _parentContext); /// Executes (or continues execution of) the VM. /// @returns false iff go() must be called again to finish the transaction. bool go(OnOpFunc const& _onOp = OnOpFunc()); /// Operation function for providing a simple trace of the VM execution. static OnOpFunc simpleTrace(); /// Operation function for providing a simple trace of the VM execution. static OnOpFunc standardTrace(std::ostream& o_output); /// @returns gas remaining after the transaction/operation. Valid after the transaction has been executed. u256 gas() const { return m_gas; } /// @returns the new address for the created contract in the CREATE operation. h160 newAddress() const { return m_newAddress; } /// @returns true iff the operation ended with a VM exception. bool excepted() const { return m_excepted != TransactionException::None; } /// Collect execution results in the result storage provided. void setResultRecipient(ExecutionResult& _res) { m_res = &_res; } private: State& m_s; ///< The state to which this operation/transaction is applied. // TODO: consider changign to EnvInfo const& to avoid LastHashes copy at every CALL/CREATE EnvInfo m_envInfo; ///< Information on the runtime environment. std::shared_ptr m_ext; ///< The VM externality object for the VM execution or null if no VM is required. shared_ptr used only to allow ExtVM forward reference. This field does *NOT* survive this object. bytesRef m_outRef; ///< Reference to "expected output" buffer. ExecutionResult* m_res = nullptr; ///< Optional storage for execution results. Address m_newAddress; ///< The address of the created contract in the case of create() being called. unsigned m_depth = 0; ///< The context's call-depth. bool m_isCreation = false; ///< True if the transaction creates a contract, or if create() is called. TransactionException m_excepted = TransactionException::None; ///< Details if the VM's execution resulted in an exception. u256 m_gas = 0; ///< The gas for EVM code execution. Initial amount before go() execution, final amount after go() execution. u256 m_refunded = 0; ///< The amount of gas refunded. Transaction m_t; ///< The original transaction. Set by setup(). LogEntries m_logs; ///< The log entries created by this transaction. Set by finalize(). bigint m_gasCost; }; } }