2016-06-20 09:47:10 -05:00
// Copyright 2015 The go-ethereum Authors
// This file is part of the go-ethereum library.
//
// The go-ethereum library is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// The go-ethereum library 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 Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
package miner
import (
2016-07-26 05:54:35 +03:00
"bytes"
2016-06-20 09:47:10 -05:00
"fmt"
"math/big"
"sync"
"sync/atomic"
"time"
"github.com/ethereum/go-ethereum/common"
2017-05-01 14:09:48 +03:00
"github.com/ethereum/go-ethereum/consensus"
"github.com/ethereum/go-ethereum/consensus/misc"
2016-06-20 09:47:10 -05:00
"github.com/ethereum/go-ethereum/core"
"github.com/ethereum/go-ethereum/core/state"
"github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/core/vm"
"github.com/ethereum/go-ethereum/ethdb"
"github.com/ethereum/go-ethereum/event"
2017-05-01 14:09:48 +03:00
"github.com/ethereum/go-ethereum/log"
2016-07-26 05:54:35 +03:00
"github.com/ethereum/go-ethereum/params"
2016-06-20 09:47:10 -05:00
"gopkg.in/fatih/set.v0"
)
const (
resultQueueSize = 10
miningLogAtDepth = 5
2017-10-10 11:38:49 +02:00
2018-06-20 16:51:55 +02:00
// txChanSize is the size of channel listening to NewTxsEvent.
2017-10-10 11:38:49 +02:00
// The number is referenced from the size of tx pool.
txChanSize = 4096
// chainHeadChanSize is the size of channel listening to ChainHeadEvent.
chainHeadChanSize = 10
// chainSideChanSize is the size of channel listening to ChainSideEvent.
chainSideChanSize = 10
2016-06-20 09:47:10 -05:00
)
// Agent can register themself with the worker
type Agent interface {
Work ( ) chan <- * Work
SetReturnCh ( chan <- * Result )
Stop ( )
Start ( )
GetHashRate ( ) int64
}
2016-09-08 12:45:12 +03:00
// Work is the workers current environment and holds
2016-06-20 09:47:10 -05:00
// all of the current state information
type Work struct {
2016-11-25 08:50:30 +03:00
config * params . ChainConfig
signer types . Signer
2017-08-04 23:14:17 +07:00
state * state . StateDB // apply state changes here
ancestors * set . Set // ancestor set (used for checking uncle parent validity)
family * set . Set // family set (used for checking uncle invalidity)
uncles * set . Set // uncle set
tcount int // tx count in cycle
2018-06-20 16:51:55 +02:00
gasPool * core . GasPool // available gas used to pack transactions
2016-06-20 09:47:10 -05:00
Block * types . Block // the new block
header * types . Header
txs [ ] * types . Transaction
receipts [ ] * types . Receipt
createdAt time . Time
}
type Result struct {
Work * Work
Block * types . Block
}
// worker is the main object which takes care of applying messages to the new state
type worker struct {
2016-11-25 08:50:30 +03:00
config * params . ChainConfig
2017-05-01 14:09:48 +03:00
engine consensus . Engine
2016-06-20 09:47:10 -05:00
mu sync . Mutex
// update loop
2017-10-10 11:38:49 +02:00
mux * event . TypeMux
2018-06-20 16:51:55 +02:00
txsCh chan core . NewTxsEvent
txsSub event . Subscription
2017-10-10 11:38:49 +02:00
chainHeadCh chan core . ChainHeadEvent
chainHeadSub event . Subscription
chainSideCh chan core . ChainSideEvent
chainSideSub event . Subscription
wg sync . WaitGroup
2016-06-20 09:47:10 -05:00
agents map [ Agent ] struct { }
recv chan * Result
2016-09-08 12:45:12 +03:00
eth Backend
2016-06-20 09:47:10 -05:00
chain * core . BlockChain
proc core . Validator
chainDb ethdb . Database
coinbase common . Address
extra [ ] byte
currentMu sync . Mutex
current * Work
2018-04-24 18:50:26 +03:00
snapshotMu sync . RWMutex
snapshotBlock * types . Block
snapshotState * state . StateDB
2016-06-20 09:47:10 -05:00
uncleMu sync . Mutex
possibleUncles map [ common . Hash ] * types . Block
2017-02-23 03:22:43 +03:00
unconfirmed * unconfirmedBlocks // set of locally mined blocks pending canonicalness confirmations
2016-06-20 09:47:10 -05:00
// atomic status counters
mining int32
atWork int32
}
2017-05-01 14:09:48 +03:00
func newWorker ( config * params . ChainConfig , engine consensus . Engine , coinbase common . Address , eth Backend , mux * event . TypeMux ) * worker {
2016-06-20 09:47:10 -05:00
worker := & worker {
config : config ,
2017-05-01 14:09:48 +03:00
engine : engine ,
2016-06-20 09:47:10 -05:00
eth : eth ,
2016-09-08 12:45:12 +03:00
mux : mux ,
2018-06-20 16:51:55 +02:00
txsCh : make ( chan core . NewTxsEvent , txChanSize ) ,
2017-10-10 11:38:49 +02:00
chainHeadCh : make ( chan core . ChainHeadEvent , chainHeadChanSize ) ,
chainSideCh : make ( chan core . ChainSideEvent , chainSideChanSize ) ,
2016-06-20 09:47:10 -05:00
chainDb : eth . ChainDb ( ) ,
recv : make ( chan * Result , resultQueueSize ) ,
chain : eth . BlockChain ( ) ,
proc : eth . BlockChain ( ) . Validator ( ) ,
possibleUncles : make ( map [ common . Hash ] * types . Block ) ,
coinbase : coinbase ,
agents : make ( map [ Agent ] struct { } ) ,
2017-10-10 11:38:49 +02:00
unconfirmed : newUnconfirmedBlocks ( eth . BlockChain ( ) , miningLogAtDepth ) ,
2016-06-20 09:47:10 -05:00
}
2018-06-20 16:51:55 +02:00
// Subscribe NewTxsEvent for tx pool
worker . txsSub = eth . TxPool ( ) . SubscribeNewTxsEvent ( worker . txsCh )
2017-10-10 11:38:49 +02:00
// Subscribe events for blockchain
worker . chainHeadSub = eth . BlockChain ( ) . SubscribeChainHeadEvent ( worker . chainHeadCh )
worker . chainSideSub = eth . BlockChain ( ) . SubscribeChainSideEvent ( worker . chainSideCh )
2016-06-20 09:47:10 -05:00
go worker . update ( )
go worker . wait ( )
worker . commitNewWork ( )
return worker
}
func ( self * worker ) setEtherbase ( addr common . Address ) {
self . mu . Lock ( )
defer self . mu . Unlock ( )
self . coinbase = addr
}
2017-02-23 03:22:43 +03:00
func ( self * worker ) setExtra ( extra [ ] byte ) {
self . mu . Lock ( )
defer self . mu . Unlock ( )
self . extra = extra
}
2016-06-20 09:47:10 -05:00
func ( self * worker ) pending ( ) ( * types . Block , * state . StateDB ) {
if atomic . LoadInt32 ( & self . mining ) == 0 {
2018-04-24 18:50:26 +03:00
// return a snapshot to avoid contention on currentMu mutex
self . snapshotMu . RLock ( )
defer self . snapshotMu . RUnlock ( )
return self . snapshotBlock , self . snapshotState . Copy ( )
2016-06-20 09:47:10 -05:00
}
2016-12-07 23:54:59 +03:00
self . currentMu . Lock ( )
defer self . currentMu . Unlock ( )
2018-04-24 18:50:26 +03:00
return self . current . Block , self . current . state . Copy ( )
}
2016-12-07 23:54:59 +03:00
2018-04-24 18:50:26 +03:00
func ( self * worker ) pendingBlock ( ) * types . Block {
2016-12-07 23:54:59 +03:00
if atomic . LoadInt32 ( & self . mining ) == 0 {
2018-04-24 18:50:26 +03:00
// return a snapshot to avoid contention on currentMu mutex
self . snapshotMu . RLock ( )
defer self . snapshotMu . RUnlock ( )
return self . snapshotBlock
2016-12-07 23:54:59 +03:00
}
2018-04-24 18:50:26 +03:00
self . currentMu . Lock ( )
defer self . currentMu . Unlock ( )
2016-12-07 23:54:59 +03:00
return self . current . Block
}
2016-06-20 09:47:10 -05:00
func ( self * worker ) start ( ) {
self . mu . Lock ( )
defer self . mu . Unlock ( )
atomic . StoreInt32 ( & self . mining , 1 )
// spin up agents
for agent := range self . agents {
agent . Start ( )
}
}
func ( self * worker ) stop ( ) {
self . wg . Wait ( )
self . mu . Lock ( )
defer self . mu . Unlock ( )
if atomic . LoadInt32 ( & self . mining ) == 1 {
for agent := range self . agents {
agent . Stop ( )
}
}
atomic . StoreInt32 ( & self . mining , 0 )
atomic . StoreInt32 ( & self . atWork , 0 )
}
func ( self * worker ) register ( agent Agent ) {
self . mu . Lock ( )
defer self . mu . Unlock ( )
self . agents [ agent ] = struct { } { }
agent . SetReturnCh ( self . recv )
}
func ( self * worker ) unregister ( agent Agent ) {
self . mu . Lock ( )
defer self . mu . Unlock ( )
delete ( self . agents , agent )
agent . Stop ( )
}
func ( self * worker ) update ( ) {
2018-06-20 16:51:55 +02:00
defer self . txsSub . Unsubscribe ( )
2017-10-10 11:38:49 +02:00
defer self . chainHeadSub . Unsubscribe ( )
defer self . chainSideSub . Unsubscribe ( )
for {
2016-06-20 09:47:10 -05:00
// A real event arrived, process interesting content
2017-10-10 11:38:49 +02:00
select {
// Handle ChainHeadEvent
case <- self . chainHeadCh :
2016-06-20 09:47:10 -05:00
self . commitNewWork ( )
2017-10-10 11:38:49 +02:00
// Handle ChainSideEvent
case ev := <- self . chainSideCh :
2016-06-20 09:47:10 -05:00
self . uncleMu . Lock ( )
self . possibleUncles [ ev . Block . Hash ( ) ] = ev . Block
self . uncleMu . Unlock ( )
2017-10-10 11:38:49 +02:00
2018-06-20 16:51:55 +02:00
// Handle NewTxsEvent
case ev := <- self . txsCh :
// Apply transactions to the pending state if we're not mining.
//
// Note all transactions received may not be continuous with transactions
// already included in the current mining block. These transactions will
// be automatically eliminated.
2016-06-20 09:47:10 -05:00
if atomic . LoadInt32 ( & self . mining ) == 0 {
self . currentMu . Lock ( )
2018-06-20 16:51:55 +02:00
txs := make ( map [ common . Address ] types . Transactions )
for _ , tx := range ev . Txs {
acc , _ := types . Sender ( self . current . signer , tx )
txs [ acc ] = append ( txs [ acc ] , tx )
}
2017-10-10 11:38:49 +02:00
txset := types . NewTransactionsByPriceAndNonce ( self . current . signer , txs )
2017-08-04 23:14:17 +07:00
self . current . commitTransactions ( self . mux , txset , self . chain , self . coinbase )
2018-04-24 18:50:26 +03:00
self . updateSnapshot ( )
2016-06-20 09:47:10 -05:00
self . currentMu . Unlock ( )
2018-01-15 21:26:41 +01:00
} else {
// If we're mining, but nothing is being processed, wake on new transactions
if self . config . Clique != nil && self . config . Clique . Period == 0 {
self . commitNewWork ( )
}
2016-06-20 09:47:10 -05:00
}
2017-10-10 11:38:49 +02:00
// System stopped
2018-06-20 16:51:55 +02:00
case <- self . txsSub . Err ( ) :
2017-10-10 11:38:49 +02:00
return
case <- self . chainHeadSub . Err ( ) :
return
case <- self . chainSideSub . Err ( ) :
return
2016-06-20 09:47:10 -05:00
}
}
}
func ( self * worker ) wait ( ) {
for {
for result := range self . recv {
atomic . AddInt32 ( & self . atWork , - 1 )
if result == nil {
continue
}
block := result . Block
work := result . Work
2017-10-10 11:38:49 +02:00
// Update the block hash in all logs since it is now available and not when the
// receipt/log of individual transactions were created.
for _ , r := range work . receipts {
for _ , l := range r . Logs {
l . BlockHash = block . Hash ( )
2016-06-20 09:47:10 -05:00
}
}
2017-10-10 11:38:49 +02:00
for _ , log := range work . state . Logs ( ) {
log . BlockHash = block . Hash ( )
}
2018-02-27 11:39:30 +01:00
stat , err := self . chain . WriteBlockWithState ( block , work . receipts , work . state )
2017-10-10 11:38:49 +02:00
if err != nil {
log . Error ( "Failed writing block to chain" , "err" , err )
continue
}
// Broadcast the block and announce chain insertion event
self . mux . Post ( core . NewMinedBlockEvent { Block : block } )
var (
events [ ] interface { }
logs = work . state . Logs ( )
)
events = append ( events , core . ChainEvent { Block : block , Hash : block . Hash ( ) , Logs : logs } )
if stat == core . CanonStatTy {
events = append ( events , core . ChainHeadEvent { Block : block } )
}
self . chain . PostChainEvents ( events , logs )
2017-02-23 03:22:43 +03:00
// Insert the block into the set of pending ones to wait for confirmations
self . unconfirmed . Insert ( block . NumberU64 ( ) , block . Hash ( ) )
2016-06-20 09:47:10 -05:00
}
}
}
// push sends a new work task to currently live miner agents.
func ( self * worker ) push ( work * Work ) {
if atomic . LoadInt32 ( & self . mining ) != 1 {
return
}
for agent := range self . agents {
atomic . AddInt32 ( & self . atWork , 1 )
if ch := agent . Work ( ) ; ch != nil {
ch <- work
}
}
}
// makeCurrent creates a new environment for the current cycle.
func ( self * worker ) makeCurrent ( parent * types . Block , header * types . Header ) error {
2016-09-29 22:51:33 +03:00
state , err := self . chain . StateAt ( parent . Root ( ) )
2016-06-20 09:47:10 -05:00
if err != nil {
return err
}
work := & Work {
config : self . config ,
2018-06-20 16:51:55 +02:00
signer : types . NewEIP155Signer ( self . config . ChainID ) ,
2016-06-20 09:47:10 -05:00
state : state ,
ancestors : set . New ( ) ,
family : set . New ( ) ,
uncles : set . New ( ) ,
header : header ,
createdAt : time . Now ( ) ,
}
// when 08 is processed ancestors contain 07 (quick block)
for _ , ancestor := range self . chain . GetBlocksFromHash ( parent . Hash ( ) , 7 ) {
for _ , uncle := range ancestor . Uncles ( ) {
work . family . Add ( uncle . Hash ( ) )
}
work . family . Add ( ancestor . Hash ( ) )
work . ancestors . Add ( ancestor . Hash ( ) )
}
2017-10-10 11:38:49 +02:00
2016-06-20 09:47:10 -05:00
// Keep track of transactions which return errors so they can be removed
work . tcount = 0
self . current = work
return nil
}
func ( self * worker ) commitNewWork ( ) {
self . mu . Lock ( )
defer self . mu . Unlock ( )
self . uncleMu . Lock ( )
defer self . uncleMu . Unlock ( )
self . currentMu . Lock ( )
defer self . currentMu . Unlock ( )
tstart := time . Now ( )
parent := self . chain . CurrentBlock ( )
2016-12-07 23:54:59 +03:00
2016-06-20 09:47:10 -05:00
tstamp := tstart . Unix ( )
if parent . Time ( ) . Cmp ( new ( big . Int ) . SetInt64 ( tstamp ) ) >= 0 {
tstamp = parent . Time ( ) . Int64 ( ) + 1
}
// this will ensure we're not going off too far in the future
2017-05-01 14:09:48 +03:00
if now := time . Now ( ) . Unix ( ) ; tstamp > now + 1 {
2016-06-20 09:47:10 -05:00
wait := time . Duration ( tstamp - now ) * time . Second
2017-05-01 14:09:48 +03:00
log . Info ( "Mining too far in the future" , "wait" , common . PrettyDuration ( wait ) )
2016-06-20 09:47:10 -05:00
time . Sleep ( wait )
}
num := parent . Number ( )
header := & types . Header {
ParentHash : parent . Hash ( ) ,
Number : num . Add ( num , common . Big1 ) ,
GasLimit : core . CalcGasLimit ( parent ) ,
Extra : self . extra ,
Time : big . NewInt ( tstamp ) ,
}
2017-05-01 14:09:48 +03:00
// Only set the coinbase if we are mining (avoid spurious block rewards)
if atomic . LoadInt32 ( & self . mining ) == 1 {
header . Coinbase = self . coinbase
}
if err := self . engine . Prepare ( self . chain , header ) ; err != nil {
log . Error ( "Failed to prepare header for mining" , "err" , err )
return
}
2016-07-26 05:54:35 +03:00
// If we are care about TheDAO hard-fork check whether to override the extra-data or not
if daoBlock := self . config . DAOForkBlock ; daoBlock != nil {
// Check whether the block is among the fork extra-override range
limit := new ( big . Int ) . Add ( daoBlock , params . DAOForkExtraRange )
if header . Number . Cmp ( daoBlock ) >= 0 && header . Number . Cmp ( limit ) < 0 {
// Depending whether we support or oppose the fork, override differently
if self . config . DAOForkSupport {
header . Extra = common . CopyBytes ( params . DAOForkBlockExtra )
2017-02-23 03:22:43 +03:00
} else if bytes . Equal ( header . Extra , params . DAOForkBlockExtra ) {
2016-07-26 05:54:35 +03:00
header . Extra = [ ] byte { } // If miner opposes, don't let it use the reserved extra-data
}
}
}
2016-06-20 09:47:10 -05:00
// Could potentially happen if starting to mine in an odd state.
err := self . makeCurrent ( parent , header )
if err != nil {
2017-05-01 14:09:48 +03:00
log . Error ( "Failed to create mining context" , "err" , err )
2016-06-20 09:47:10 -05:00
return
}
2016-07-26 05:54:35 +03:00
// Create the current work task and check any fork transitions needed
2016-06-20 09:47:10 -05:00
work := self . current
2016-07-26 05:54:35 +03:00
if self . config . DAOForkSupport && self . config . DAOForkBlock != nil && self . config . DAOForkBlock . Cmp ( header . Number ) == 0 {
2017-05-01 14:09:48 +03:00
misc . ApplyDAOHardFork ( work . state )
2016-07-26 05:54:35 +03:00
}
2017-02-23 03:22:43 +03:00
pending , err := self . eth . TxPool ( ) . Pending ( )
if err != nil {
2017-05-01 14:09:48 +03:00
log . Error ( "Failed to fetch pending transactions" , "err" , err )
2017-02-23 03:22:43 +03:00
return
}
2017-10-10 11:38:49 +02:00
txs := types . NewTransactionsByPriceAndNonce ( self . current . signer , pending )
2017-08-04 23:14:17 +07:00
work . commitTransactions ( self . mux , txs , self . chain , self . coinbase )
2016-06-20 09:47:10 -05:00
// compute uncles for the new block.
var (
uncles [ ] * types . Header
badUncles [ ] common . Hash
)
for hash , uncle := range self . possibleUncles {
if len ( uncles ) == 2 {
break
}
if err := self . commitUncle ( work , uncle . Header ( ) ) ; err != nil {
2017-05-01 14:09:48 +03:00
log . Trace ( "Bad uncle found and will be removed" , "hash" , hash )
log . Trace ( fmt . Sprint ( uncle ) )
2016-06-20 09:47:10 -05:00
badUncles = append ( badUncles , hash )
} else {
2017-05-01 14:09:48 +03:00
log . Debug ( "Committing new uncle to block" , "hash" , hash )
2016-06-20 09:47:10 -05:00
uncles = append ( uncles , uncle . Header ( ) )
}
}
for _ , hash := range badUncles {
delete ( self . possibleUncles , hash )
}
2017-05-01 14:09:48 +03:00
// Create the new block to seal with the consensus engine
if work . Block , err = self . engine . Finalize ( self . chain , header , work . state , work . txs , uncles , work . receipts ) ; err != nil {
log . Error ( "Failed to finalize block for sealing" , "err" , err )
return
2016-06-20 09:47:10 -05:00
}
// We only care about logging if we're actually mining.
if atomic . LoadInt32 ( & self . mining ) == 1 {
2017-05-01 14:09:48 +03:00
log . Info ( "Commit new mining work" , "number" , work . Block . Number ( ) , "txs" , work . tcount , "uncles" , len ( uncles ) , "elapsed" , common . PrettyDuration ( time . Since ( tstart ) ) )
2017-02-23 03:22:43 +03:00
self . unconfirmed . Shift ( work . Block . NumberU64 ( ) - 1 )
2016-06-20 09:47:10 -05:00
}
self . push ( work )
2018-04-24 18:50:26 +03:00
self . updateSnapshot ( )
2016-06-20 09:47:10 -05:00
}
func ( self * worker ) commitUncle ( work * Work , uncle * types . Header ) error {
hash := uncle . Hash ( )
if work . uncles . Has ( hash ) {
2017-05-01 14:09:48 +03:00
return fmt . Errorf ( "uncle not unique" )
2016-06-20 09:47:10 -05:00
}
if ! work . ancestors . Has ( uncle . ParentHash ) {
2017-05-01 14:09:48 +03:00
return fmt . Errorf ( "uncle's parent unknown (%x)" , uncle . ParentHash [ 0 : 4 ] )
2016-06-20 09:47:10 -05:00
}
if work . family . Has ( hash ) {
2017-05-01 14:09:48 +03:00
return fmt . Errorf ( "uncle already in family (%x)" , hash )
2016-06-20 09:47:10 -05:00
}
work . uncles . Add ( uncle . Hash ( ) )
return nil
}
2018-04-24 18:50:26 +03:00
func ( self * worker ) updateSnapshot ( ) {
self . snapshotMu . Lock ( )
defer self . snapshotMu . Unlock ( )
self . snapshotBlock = types . NewBlock (
self . current . header ,
self . current . txs ,
nil ,
self . current . receipts ,
)
self . snapshotState = self . current . state . Copy ( )
}
2017-08-04 23:14:17 +07:00
func ( env * Work ) commitTransactions ( mux * event . TypeMux , txs * types . TransactionsByPriceAndNonce , bc * core . BlockChain , coinbase common . Address ) {
2018-06-20 16:51:55 +02:00
if env . gasPool == nil {
env . gasPool = new ( core . GasPool ) . AddGas ( env . header . GasLimit )
}
2016-06-20 09:47:10 -05:00
2017-02-23 03:22:43 +03:00
var coalescedLogs [ ] * types . Log
2016-09-29 22:51:33 +03:00
for {
2018-02-27 11:39:30 +01:00
// If we don't have enough gas for any further transactions then we're done
2018-06-20 16:51:55 +02:00
if env . gasPool . Gas ( ) < params . TxGas {
log . Trace ( "Not enough gas for further transactions" , "have" , env . gasPool , "want" , params . TxGas )
2018-02-27 11:39:30 +01:00
break
}
2016-09-29 22:51:33 +03:00
// Retrieve the next transaction and abort if all done
tx := txs . Peek ( )
if tx == nil {
break
}
2016-06-20 09:47:10 -05:00
// Error may be ignored here. The error has already been checked
// during transaction acceptance is the transaction pool.
2016-11-25 08:50:30 +03:00
//
// We use the eip155 signer regardless of the current hf.
from , _ := types . Sender ( env . signer , tx )
// Check whether the tx is replay protected. If we're not in the EIP155 hf
// phase, start ignoring the sender until we do.
if tx . Protected ( ) && ! env . config . IsEIP155 ( env . header . Number ) {
2017-05-01 14:09:48 +03:00
log . Trace ( "Ignoring reply protected transaction" , "hash" , tx . Hash ( ) , "eip155" , env . config . EIP155Block )
2016-11-25 08:50:30 +03:00
txs . Pop ( )
continue
}
2016-09-29 22:51:33 +03:00
// Start executing the transaction
2017-08-04 23:14:17 +07:00
env . state . Prepare ( tx . Hash ( ) , common . Hash { } , env . tcount )
2016-06-20 09:47:10 -05:00
2018-06-20 16:51:55 +02:00
err , logs := env . commitTransaction ( tx , bc , coinbase , env . gasPool )
2017-05-01 14:09:48 +03:00
switch err {
case core . ErrGasLimitReached :
2016-09-29 22:51:33 +03:00
// Pop the current out-of-gas transaction without shifting in the next from the account
2017-05-01 14:09:48 +03:00
log . Trace ( "Gas limit exceeded for current block" , "sender" , from )
2016-09-29 22:51:33 +03:00
txs . Pop ( )
2016-06-20 09:47:10 -05:00
2017-10-10 11:38:49 +02:00
case core . ErrNonceTooLow :
// New head notification data race between the transaction pool and miner, shift
log . Trace ( "Skipping transaction with low nonce" , "sender" , from , "nonce" , tx . Nonce ( ) )
txs . Shift ( )
case core . ErrNonceTooHigh :
// Reorg notification data race between the transaction pool and miner, skip account =
log . Trace ( "Skipping account with hight nonce" , "sender" , from , "nonce" , tx . Nonce ( ) )
txs . Pop ( )
2017-05-01 14:09:48 +03:00
case nil :
2016-09-29 22:51:33 +03:00
// Everything ok, collect the logs and shift in the next transaction from the same account
2016-06-20 09:47:10 -05:00
coalescedLogs = append ( coalescedLogs , logs ... )
2016-09-29 22:51:33 +03:00
env . tcount ++
txs . Shift ( )
2017-05-01 14:09:48 +03:00
default :
2017-10-10 11:38:49 +02:00
// Strange error, discard the transaction and get the next in line (note, the
// nonce-too-high clause will prevent us from executing in vain).
log . Debug ( "Transaction failed, account skipped" , "hash" , tx . Hash ( ) , "err" , err )
txs . Shift ( )
2016-06-20 09:47:10 -05:00
}
}
2016-12-07 23:54:59 +03:00
2016-06-20 09:47:10 -05:00
if len ( coalescedLogs ) > 0 || env . tcount > 0 {
2016-12-07 23:54:59 +03:00
// make a copy, the state caches the logs and these logs get "upgraded" from pending to mined
// logs by filling in the block hash when the block was mined by the local miner. This can
// cause a race condition if a log was "upgraded" before the PendingLogsEvent is processed.
2017-02-23 03:22:43 +03:00
cpy := make ( [ ] * types . Log , len ( coalescedLogs ) )
2016-12-07 23:54:59 +03:00
for i , l := range coalescedLogs {
2017-02-23 03:22:43 +03:00
cpy [ i ] = new ( types . Log )
2016-12-07 23:54:59 +03:00
* cpy [ i ] = * l
}
2017-02-23 03:22:43 +03:00
go func ( logs [ ] * types . Log , tcount int ) {
2016-06-20 09:47:10 -05:00
if len ( logs ) > 0 {
mux . Post ( core . PendingLogsEvent { Logs : logs } )
}
if tcount > 0 {
mux . Post ( core . PendingStateEvent { } )
}
2016-12-07 23:54:59 +03:00
} ( cpy , env . tcount )
2016-06-20 09:47:10 -05:00
}
}
2017-05-01 14:09:48 +03:00
func ( env * Work ) commitTransaction ( tx * types . Transaction , bc * core . BlockChain , coinbase common . Address , gp * core . GasPool ) ( error , [ ] * types . Log ) {
2016-10-12 01:56:36 +03:00
snap := env . state . Snapshot ( )
2016-06-20 09:47:10 -05:00
2018-02-27 11:39:30 +01:00
receipt , _ , err := core . ApplyTransaction ( env . config , bc , & coinbase , gp , env . state , env . header , tx , & env . header . GasUsed , vm . Config { } )
2016-06-20 09:47:10 -05:00
if err != nil {
2016-10-12 01:56:36 +03:00
env . state . RevertToSnapshot ( snap )
2016-06-20 09:47:10 -05:00
return err , nil
}
env . txs = append ( env . txs , tx )
env . receipts = append ( env . receipts , receipt )
2017-02-23 03:22:43 +03:00
return nil , receipt . Logs
2016-06-20 09:47:10 -05:00
}