|
@ -98,22 +98,30 @@ void MixClient::resetState(std::unordered_map<Address, Account> const& _accounts |
|
|
m_executions.clear(); |
|
|
m_executions.clear(); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
Transaction MixClient::replaceGas(Transaction const& _t, u256 const& _gas) |
|
|
Transaction MixClient::replaceGas(Transaction const& _t, u256 const& _gas, Secret const& _secret) |
|
|
{ |
|
|
{ |
|
|
Transaction ret; |
|
|
Transaction ret; |
|
|
if (_t.isCreation()) |
|
|
if (_secret) |
|
|
ret = Transaction(_t.value(), _t.gasPrice(), _gas, _t.data(), _t.nonce()); |
|
|
{ |
|
|
|
|
|
if (_t.isCreation()) |
|
|
|
|
|
ret = Transaction(_t.value(), _t.gasPrice(), _gas, _t.data(), _t.nonce(), _secret); |
|
|
|
|
|
else |
|
|
|
|
|
ret = Transaction(_t.value(), _t.gasPrice(), _gas, _t.receiveAddress(), _t.data(), _t.nonce(), _secret); |
|
|
|
|
|
} |
|
|
else |
|
|
else |
|
|
ret = Transaction(_t.value(), _t.gasPrice(), _gas, _t.receiveAddress(), _t.data(), _t.nonce()); |
|
|
{ |
|
|
ret.forceSender(_t.safeSender()); |
|
|
if (_t.isCreation()) |
|
|
|
|
|
ret = Transaction(_t.value(), _t.gasPrice(), _gas, _t.data(), _t.nonce()); |
|
|
|
|
|
else |
|
|
|
|
|
ret = Transaction(_t.value(), _t.gasPrice(), _gas, _t.receiveAddress(), _t.data(), _t.nonce()); |
|
|
|
|
|
ret.forceSender(_t.safeSender()); |
|
|
|
|
|
} |
|
|
return ret; |
|
|
return ret; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void MixClient::executeTransaction(Transaction const& _t, State& _state, bool _call, bool _gasAuto) |
|
|
void MixClient::executeTransaction(Transaction const& _t, State& _state, bool _call, bool _gasAuto, Secret const& _secret) |
|
|
{ |
|
|
{ |
|
|
Transaction t = _gasAuto ? replaceGas(_t, m_state.gasLimitRemaining()) : _t; |
|
|
Transaction t = _gasAuto ? replaceGas(_t, m_state.gasLimitRemaining()) : _t; |
|
|
bytes rlp = t.rlp(); |
|
|
|
|
|
|
|
|
|
|
|
// do debugging run first
|
|
|
// do debugging run first
|
|
|
LastHashes lastHashes(256); |
|
|
LastHashes lastHashes(256); |
|
|
lastHashes[0] = bc().numberHash(bc().number()); |
|
|
lastHashes[0] = bc().numberHash(bc().number()); |
|
@ -123,7 +131,7 @@ void MixClient::executeTransaction(Transaction const& _t, State& _state, bool _c |
|
|
State execState = _state; |
|
|
State execState = _state; |
|
|
execState.addBalance(t.sender(), t.gas() * t.gasPrice()); //give it enough balance for gas estimation
|
|
|
execState.addBalance(t.sender(), t.gas() * t.gasPrice()); //give it enough balance for gas estimation
|
|
|
Executive execution(execState, lastHashes, 0); |
|
|
Executive execution(execState, lastHashes, 0); |
|
|
execution.initialize(&rlp); |
|
|
execution.initialize(t); |
|
|
execution.execute(); |
|
|
execution.execute(); |
|
|
std::vector<MachineState> machineStates; |
|
|
std::vector<MachineState> machineStates; |
|
|
std::vector<unsigned> levels; |
|
|
std::vector<unsigned> levels; |
|
@ -222,7 +230,7 @@ void MixClient::executeTransaction(Transaction const& _t, State& _state, bool _c |
|
|
// execute on a state
|
|
|
// execute on a state
|
|
|
if (!_call) |
|
|
if (!_call) |
|
|
{ |
|
|
{ |
|
|
t = _gasAuto ? replaceGas(_t, d.gasUsed) : _t; |
|
|
t = _gasAuto ? replaceGas(_t, d.gasUsed, _secret) : _t; |
|
|
er =_state.execute(lastHashes, t); |
|
|
er =_state.execute(lastHashes, t); |
|
|
if (t.isCreation() && _state.code(d.contractAddress).empty()) |
|
|
if (t.isCreation() && _state.code(d.contractAddress).empty()) |
|
|
BOOST_THROW_EXCEPTION(OutOfGas() << errinfo_comment("Not enough gas for contract deployment")); |
|
|
BOOST_THROW_EXCEPTION(OutOfGas() << errinfo_comment("Not enough gas for contract deployment")); |
|
@ -285,7 +293,7 @@ void MixClient::submitTransaction(Secret _secret, u256 _value, Address _dest, by |
|
|
WriteGuard l(x_state); |
|
|
WriteGuard l(x_state); |
|
|
u256 n = m_state.transactionsFrom(toAddress(_secret)); |
|
|
u256 n = m_state.transactionsFrom(toAddress(_secret)); |
|
|
Transaction t(_value, _gasPrice, _gas, _dest, _data, n, _secret); |
|
|
Transaction t(_value, _gasPrice, _gas, _dest, _data, n, _secret); |
|
|
executeTransaction(t, m_state, false, _gasAuto); |
|
|
executeTransaction(t, m_state, false, _gasAuto, _secret); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
Address MixClient::submitTransaction(Secret _secret, u256 _endowment, bytes const& _init, u256 _gas, u256 _gasPrice, bool _gasAuto) |
|
|
Address MixClient::submitTransaction(Secret _secret, u256 _endowment, bytes const& _init, u256 _gas, u256 _gasPrice, bool _gasAuto) |
|
@ -293,7 +301,7 @@ Address MixClient::submitTransaction(Secret _secret, u256 _endowment, bytes cons |
|
|
WriteGuard l(x_state); |
|
|
WriteGuard l(x_state); |
|
|
u256 n = m_state.transactionsFrom(toAddress(_secret)); |
|
|
u256 n = m_state.transactionsFrom(toAddress(_secret)); |
|
|
eth::Transaction t(_endowment, _gasPrice, _gas, _init, n, _secret); |
|
|
eth::Transaction t(_endowment, _gasPrice, _gas, _init, n, _secret); |
|
|
executeTransaction(t, m_state, false, _gasAuto); |
|
|
executeTransaction(t, m_state, false, _gasAuto, _secret); |
|
|
Address address = right160(sha3(rlpList(t.sender(), t.nonce()))); |
|
|
Address address = right160(sha3(rlpList(t.sender(), t.nonce()))); |
|
|
return address; |
|
|
return address; |
|
|
} |
|
|
} |
|
@ -307,7 +315,6 @@ dev::eth::ExecutionResult MixClient::call(Address const& _from, u256 _value, Add |
|
|
t.forceSender(_from); |
|
|
t.forceSender(_from); |
|
|
if (_ff == FudgeFactor::Lenient) |
|
|
if (_ff == FudgeFactor::Lenient) |
|
|
temp.addBalance(_from, (u256)(t.gasRequired() * t.gasPrice() + t.value())); |
|
|
temp.addBalance(_from, (u256)(t.gasRequired() * t.gasPrice() + t.value())); |
|
|
bytes rlp = t.rlp(); |
|
|
|
|
|
WriteGuard lw(x_state); //TODO: lock is required only for last execution state
|
|
|
WriteGuard lw(x_state); //TODO: lock is required only for last execution state
|
|
|
executeTransaction(t, temp, true, _gasAuto); |
|
|
executeTransaction(t, temp, true, _gasAuto); |
|
|
return lastExecution().result; |
|
|
return lastExecution().result; |
|
|