/* 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 AssemblyDebuggerModel.cpp * @author Yann yann@ethdev.com * @date 2014 * used as a model to debug contract assembly code. */ #include #include #include #include #include #include #include "AssemblyDebuggerModel.h" #include "AppContext.h" #include "DebuggingStateWrapper.h" using namespace std; using namespace dev; using namespace dev::eth; namespace dev { namespace mix { AssemblyDebuggerModel::AssemblyDebuggerModel(): m_userAccount(KeyPair::create()) { resetState(10000000 * ether); } DebuggingContent AssemblyDebuggerModel::executeTransaction(bytesConstRef const& _rawTransaction) { QList machineStates; eth::Executive execution(m_executiveState, LastHashes(), 0); execution.setup(_rawTransaction); std::vector levels; bytes code; bytesConstRef data; bool firstIteration = true; auto onOp = [&](uint64_t steps, Instruction inst, dev::bigint newMemSize, dev::bigint gasCost, void* voidVM, void const* voidExt) { VM& vm = *(VM*)voidVM; ExtVM const& ext = *(ExtVM const*)voidExt; if (firstIteration) { code = ext.code; data = ext.data; firstIteration = false; } if (levels.size() < ext.depth) levels.push_back(&machineStates.back()); else levels.resize(ext.depth); machineStates.append(DebuggingState({steps, ext.myAddress, vm.curPC(), inst, newMemSize, vm.gas(), vm.stack(), vm.memory(), gasCost, ext.state().storage(ext.myAddress), levels})); }; execution.go(onOp); execution.finalize(); m_executiveState.completeMine(); DebuggingContent d; d.returnValue = execution.out().toVector(); d.machineStates = machineStates; d.executionCode = code; d.executionData = data; d.contentAvailable = true; d.message = "ok"; return d; } DebuggingContent AssemblyDebuggerModel::deployContract(bytes const& _code) { u256 gasPrice = 10000000000000; u256 gas = 1000000; u256 amount = 100; Transaction _tr(amount, gasPrice, min(gas, m_executiveState.gasLimitRemaining()), _code, m_executiveState.transactionsFrom(dev::toAddress(m_userAccount.secret())), m_userAccount.secret()); bytes b = _tr.rlp(); dev::bytesConstRef bytesRef = &b; DebuggingContent d = executeTransaction(bytesRef); h256 th = sha3(rlpList(_tr.sender(), _tr.nonce())); d.contractAddress = right160(th); return d; } DebuggingContent AssemblyDebuggerModel::callContract(Address const& _contract, bytes const& _data, TransactionSettings const& _tr) { Transaction tr = Transaction(_tr.value, _tr.gasPrice, min(_tr.gas, m_executiveState.gasLimitRemaining()), _contract, _data, m_executiveState.transactionsFrom(dev::toAddress(m_userAccount.secret())), m_userAccount.secret()); bytes b = tr.rlp(); dev::bytesConstRef bytesRef = &b; DebuggingContent d = executeTransaction(bytesRef); d.contractAddress = tr.receiveAddress(); return d; } void AssemblyDebuggerModel::resetState(u256 _balance) { m_executiveState = eth::State(Address(), m_overlayDB, BaseState::Empty); m_executiveState.addBalance(m_userAccount.address(), _balance); } } }