/*
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 < http : //www.gnu.org/licenses/>.
*/
/** @file Executive.cpp
* @ author Gav Wood < i @ gavwood . com >
* @ date 2014
*/
# include "Executive.h"
# include "State.h"
# include "ExtVM.h"
# include "VM.h"
using namespace std ;
using namespace eth ;
Executive : : ~ Executive ( )
{
// TODO: Make safe.
delete m_ext ;
delete m_vm ;
}
u256 Executive : : gasUsed ( ) const
{
return m_t . gas - m_endGas ;
}
void Executive : : setup ( bytesConstRef _rlp )
{
// Entry point for a user-executed transaction.
m_t = Transaction ( _rlp ) ;
auto sender = m_t . sender ( ) ;
// Avoid invalid transactions.
auto nonceReq = m_s . transactionsFrom ( sender ) ;
if ( m_t . nonce ! = nonceReq )
{
clog ( StateChat ) < < " Invalid Nonce. " ;
throw InvalidNonce ( nonceReq , m_t . nonce ) ;
}
// Don't like transactions whose gas price is too low. NOTE: this won't stay here forever - it's just until we get a proper gas price discovery protocol going.
if ( m_t . gasPrice < m_s . m_currentBlock . minGasPrice )
{
clog ( StateChat ) < < " Offered gas-price is too low. " ;
throw GasPriceTooLow ( ) ;
}
// Check gas cost is enough.
u256 gasCost = m_t . data . size ( ) * c_txDataGas + c_txGas ;
if ( m_t . gas < gasCost )
{
clog ( StateChat ) < < " Not enough gas to pay for the transaction. " ;
throw OutOfGas ( ) ;
}
u256 cost = m_t . value + m_t . gas * m_t . gasPrice ;
// Avoid unaffordable transactions.
if ( m_s . balance ( sender ) < cost )
{
clog ( StateChat ) < < " Not enough cash. " ;
throw NotEnoughCash ( ) ;
}
// Increment associated nonce for sender.
m_s . noteSending ( sender ) ;
// Pay...
// cnote << "Paying" << formatBalance(cost) << "from sender (includes" << m_t.gas << "gas at" << formatBalance(m_t.gasPrice) << ")";
m_s . subBalance ( sender , cost ) ;
if ( m_t . isCreation ( ) )
create ( sender , m_t . value , m_t . gasPrice , m_t . gas - gasCost , & m_t . data , sender ) ;
else
call ( m_t . receiveAddress , sender , m_t . value , m_t . gasPrice , bytesConstRef ( & m_t . data ) , m_t . gas - gasCost , sender ) ;
}
void Executive : : call ( Address _receiveAddress , Address _senderAddress , u256 _value , u256 _gasPrice , bytesConstRef _data , u256 _gas , Address _originAddress )
{
// cnote << "Transferring" << formatBalance(_value) << "to receiver.";
m_s . addBalance ( _receiveAddress , _value ) ;
if ( m_s . addressHasCode ( _receiveAddress ) )
{
m_vm = new VM ( _gas ) ;
bytes const & c = m_s . code ( _receiveAddress ) ;
m_ext = new ExtVM ( m_s , _receiveAddress , _senderAddress , _originAddress , _value , _gasPrice , _data , & c ) ;
}
else
m_endGas = _gas ;
}
void Executive : : create ( Address _sender , u256 _endowment , u256 _gasPrice , u256 _gas , bytesConstRef _init , Address _origin )
{
m_newAddress = right160 ( sha3 ( rlpList ( _sender , m_s . transactionsFrom ( _sender ) - 1 ) ) ) ;
while ( m_s . addressInUse ( m_newAddress ) )
m_newAddress = ( u160 ) m_newAddress + 1 ;
// Set up new account...
m_s . m_cache [ m_newAddress ] = AddressState ( 0 , 0 , h256 ( ) , h256 ( ) ) ;
// Execute _init.
m_vm = new VM ( _gas ) ;
m_ext = new ExtVM ( m_s , m_newAddress , _sender , _origin , _endowment , _gasPrice , _init , _init ) ;
}
bool Executive : : go ( uint64_t _steps )
{
if ( m_vm )
{
bool revert = false ;
try
{
m_out = m_vm - > go ( * m_ext , _steps ) ;
m_endGas = m_vm - > gas ( ) ;
}
catch ( StepsDone const & )
{
return false ;
}
catch ( OutOfGas const & /*_e*/ )
{
clog ( StateChat ) < < " Out of Gas! Reverting. " ;
revert = true ;
}
catch ( VMException const & _e )
{
clog ( StateChat ) < < " VM Exception: " < < _e . description ( ) ;
}
catch ( Exception const & _e )
{
clog ( StateChat ) < < " Exception in VM: " < < _e . description ( ) ;
}
catch ( std : : exception const & _e )
{
clog ( StateChat ) < < " std::exception in VM: " < < _e . what ( ) ;
}
// Write state out only in the case of a non-excepted transaction.
if ( revert )
{
m_ext - > revert ( ) ;
if ( m_newAddress )
{
m_s . m_cache . erase ( m_newAddress ) ;
m_newAddress = Address ( ) ;
}
}
}
return true ;
}
u256 Executive : : gas ( ) const
{
return m_vm - > gas ( ) ;
}
void Executive : : finalize ( )
{
if ( m_t . isCreation ( ) & & m_newAddress & & m_out . size ( ) )
// non-reverted creation - put code in place.
m_s . m_cache [ m_newAddress ] . setCode ( m_out ) ;
// cnote << "Refunding" << formatBalance(m_endGas * m_ext->gasPrice) << "to origin (=" << m_endGas << "*" << formatBalance(m_ext->gasPrice) << ")";
m_s . addBalance ( m_ext - > origin , m_endGas * m_ext - > gasPrice ) ;
u256 gasSpentInEth = ( m_t . gas - m_endGas ) * m_ext - > gasPrice ;
/* unsigned c_feesKept = 8;
u256 feesEarned = gasSpentInEth - ( gasSpentInEth / c_feesKept ) ;
cnote < < " Transferring " < < ( 100.0 - 100.0 / c_feesKept ) < < " % of " < < formatBalance ( gasSpent ) < < " = " < < formatBalance ( feesEarned ) < < " to miner ( " < < formatBalance ( gasSpentInEth - feesEarned ) < < " is burnt). " ;
*/
u256 feesEarned = gasSpentInEth ;
// cnote << "Transferring" << formatBalance(gasSpent) << "to miner.";
m_s . addBalance ( m_s . m_currentBlock . coinbaseAddress , feesEarned ) ;
}