2015-07-07 03:54:22 +03:00
// Copyright 2015 The go-ethereum Authors
2015-07-22 19:48:40 +03:00
// This file is part of the go-ethereum library.
2015-07-07 03:54:22 +03:00
//
2015-07-23 19:35:11 +03:00
// The go-ethereum library is free software: you can redistribute it and/or modify
2015-07-07 03:54:22 +03:00
// 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.
//
2015-07-22 19:48:40 +03:00
// The go-ethereum library is distributed in the hope that it will be useful,
2015-07-07 03:54:22 +03:00
// but WITHOUT ANY WARRANTY; without even the implied warranty of
2015-07-22 19:48:40 +03:00
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
2015-07-07 03:54:22 +03:00
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
2015-07-22 19:48:40 +03:00
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
2015-07-07 03:54:22 +03:00
2015-02-04 15:52:59 +02:00
package miner
import (
2016-07-08 18:48:17 +03:00
"bytes"
2015-02-04 15:52:59 +02:00
"fmt"
"math/big"
2015-02-15 17:16:27 +02:00
"sync"
2015-03-26 18:45:03 +02:00
"sync/atomic"
2015-06-15 12:33:08 +03:00
"time"
2015-02-04 15:52:59 +02:00
2018-07-16 10:54:19 +03:00
mapset "github.com/deckarep/golang-set"
2015-03-18 14:00:01 +02:00
"github.com/ethereum/go-ethereum/common"
2017-04-05 01:16:29 +03:00
"github.com/ethereum/go-ethereum/consensus"
"github.com/ethereum/go-ethereum/consensus/misc"
2015-02-04 15:52:59 +02:00
"github.com/ethereum/go-ethereum/core"
2015-03-23 19:27:05 +02:00
"github.com/ethereum/go-ethereum/core/state"
2015-02-04 15:52:59 +02:00
"github.com/ethereum/go-ethereum/core/types"
2015-08-30 11:19:10 +03:00
"github.com/ethereum/go-ethereum/core/vm"
2015-02-04 15:52:59 +02:00
"github.com/ethereum/go-ethereum/event"
2017-02-22 15:10:07 +03:00
"github.com/ethereum/go-ethereum/log"
2016-07-08 13:00:37 +03:00
"github.com/ethereum/go-ethereum/params"
2015-02-04 15:52:59 +02:00
)
2015-07-11 21:45:59 +03:00
const (
2018-08-14 18:34:33 +03:00
// resultQueueSize is the size of channel listening to sealing result.
resultQueueSize = 10
2018-05-18 11:45:52 +03:00
// txChanSize is the size of channel listening to NewTxsEvent.
2017-08-18 13:58:36 +03: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
2018-08-14 18:34:33 +03:00
miningLogAtDepth = 5
2015-07-11 21:45:59 +03:00
)
2015-05-11 22:47:34 +03:00
2018-08-14 18:34:33 +03:00
// Env is the worker's current environment and holds all of the current state information.
2018-08-06 12:55:44 +03:00
type Env struct {
2016-11-02 15:44:13 +03:00
config * params . ChainConfig
signer types . Signer
2017-05-16 22:07:27 +03:00
state * state . StateDB // apply state changes here
2018-07-16 10:54:19 +03:00
ancestors mapset . Set // ancestor set (used for checking uncle parent validity)
family mapset . Set // family set (used for checking uncle invalidity)
uncles mapset . Set // uncle set
2017-05-16 22:07:27 +03:00
tcount int // tx count in cycle
2018-05-18 11:45:52 +03:00
gasPool * core . GasPool // available gas used to pack transactions
2015-02-04 15:52:59 +02:00
2015-06-16 13:41:50 +03:00
header * types . Header
txs [ ] * types . Transaction
receipts [ ] * types . Receipt
2018-08-14 18:34:33 +03:00
}
2015-07-11 21:45:59 +03:00
2018-08-14 18:34:33 +03:00
func ( env * Env ) commitTransaction ( tx * types . Transaction , bc * core . BlockChain , coinbase common . Address , gp * core . GasPool ) ( error , [ ] * types . Log ) {
snap := env . state . Snapshot ( )
receipt , _ , err := core . ApplyTransaction ( env . config , bc , & coinbase , gp , env . state , env . header , tx , & env . header . GasUsed , vm . Config { } )
if err != nil {
env . state . RevertToSnapshot ( snap )
return err , nil
}
env . txs = append ( env . txs , tx )
env . receipts = append ( env . receipts , receipt )
return nil , receipt . Logs
2015-07-11 21:45:59 +03:00
}
2018-08-14 18:34:33 +03:00
func ( env * Env ) commitTransactions ( mux * event . TypeMux , txs * types . TransactionsByPriceAndNonce , bc * core . BlockChain , coinbase common . Address ) {
if env . gasPool == nil {
env . gasPool = new ( core . GasPool ) . AddGas ( env . header . GasLimit )
}
var coalescedLogs [ ] * types . Log
for {
// If we don't have enough gas for any further transactions then we're done
if env . gasPool . Gas ( ) < params . TxGas {
log . Trace ( "Not enough gas for further transactions" , "have" , env . gasPool , "want" , params . TxGas )
break
}
// Retrieve the next transaction and abort if all done
tx := txs . Peek ( )
if tx == nil {
break
}
// Error may be ignored here. The error has already been checked
// during transaction acceptance is the transaction pool.
//
// 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 ) {
log . Trace ( "Ignoring reply protected transaction" , "hash" , tx . Hash ( ) , "eip155" , env . config . EIP155Block )
txs . Pop ( )
continue
}
// Start executing the transaction
env . state . Prepare ( tx . Hash ( ) , common . Hash { } , env . tcount )
err , logs := env . commitTransaction ( tx , bc , coinbase , env . gasPool )
switch err {
case core . ErrGasLimitReached :
// Pop the current out-of-gas transaction without shifting in the next from the account
log . Trace ( "Gas limit exceeded for current block" , "sender" , from )
txs . Pop ( )
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 ( )
case nil :
// Everything ok, collect the logs and shift in the next transaction from the same account
coalescedLogs = append ( coalescedLogs , logs ... )
env . tcount ++
txs . Shift ( )
default :
// 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 ( )
}
}
if len ( coalescedLogs ) > 0 || env . tcount > 0 {
// 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.
cpy := make ( [ ] * types . Log , len ( coalescedLogs ) )
for i , l := range coalescedLogs {
cpy [ i ] = new ( types . Log )
* cpy [ i ] = * l
}
go func ( logs [ ] * types . Log , tcount int ) {
if len ( logs ) > 0 {
mux . Post ( core . PendingLogsEvent { Logs : logs } )
}
if tcount > 0 {
mux . Post ( core . PendingStateEvent { } )
}
} ( cpy , env . tcount )
}
2015-02-04 15:52:59 +02:00
}
2018-08-14 18:34:33 +03:00
// task contains all information for consensus engine sealing and result submitting.
type task struct {
receipts [ ] * types . Receipt
state * state . StateDB
block * types . Block
createdAt time . Time
}
// worker is the main object which takes care of submitting new work to consensus engine
// and gathering the sealing result.
2015-02-04 15:52:59 +02:00
type worker struct {
2016-10-20 14:36:29 +03:00
config * params . ChainConfig
2017-04-05 01:16:29 +03:00
engine consensus . Engine
2018-08-14 18:34:33 +03:00
eth Backend
chain * core . BlockChain
2016-03-02 00:32:43 +02:00
2018-08-14 18:34:33 +03:00
// Subscriptions
2017-08-18 13:58:36 +03:00
mux * event . TypeMux
2018-05-18 11:45:52 +03:00
txsCh chan core . NewTxsEvent
2018-05-10 10:04:45 +03:00
txsSub event . Subscription
2017-08-18 13:58:36 +03:00
chainHeadCh chan core . ChainHeadEvent
chainHeadSub event . Subscription
chainSideCh chan core . ChainSideEvent
chainSideSub event . Subscription
2016-03-29 04:08:16 +03:00
2018-08-14 18:34:33 +03:00
// Channels
newWork chan struct { }
taskCh chan * task
resultCh chan * task
exitCh chan struct { }
2015-02-04 15:52:59 +02:00
2018-08-14 18:34:33 +03:00
current * Env // An environment for current running cycle.
possibleUncles map [ common . Hash ] * types . Block // A set of side blocks as the possible uncle blocks.
unconfirmed * unconfirmedBlocks // A set of locally mined blocks pending canonicalness confirmations.
2015-04-05 19:57:03 +03:00
2018-08-14 18:34:33 +03:00
mu sync . RWMutex // The lock used to protect the coinbase and extra fields
2015-03-18 14:00:01 +02:00
coinbase common . Address
2015-04-05 19:57:03 +03:00
extra [ ] byte
2015-02-04 15:52:59 +02:00
2018-08-14 18:34:33 +03:00
snapshotMu sync . RWMutex // The lock used to protect the block snapshot and state snapshot
2018-04-16 10:56:20 +03:00
snapshotBlock * types . Block
snapshotState * state . StateDB
2015-04-22 11:58:43 +03:00
// atomic status counters
2018-08-03 11:33:37 +03:00
running int32 // The indicator whether the consensus engine is running or not.
2018-08-14 18:34:33 +03:00
// Test hooks
newTaskHook func ( * task ) // Method to call upon receiving a new sealing task
fullTaskInterval func ( ) // Method to call before pushing the full sealing task
2015-02-04 15:52:59 +02:00
}
2018-08-03 11:33:37 +03:00
func newWorker ( config * params . ChainConfig , engine consensus . Engine , eth Backend , mux * event . TypeMux ) * worker {
2015-04-07 13:32:55 +03:00
worker := & worker {
2016-03-02 00:32:43 +02:00
config : config ,
2017-04-05 01:16:29 +03:00
engine : engine ,
2015-03-23 13:12:49 +02:00
eth : eth ,
2016-08-15 21:14:05 +03:00
mux : mux ,
2015-08-31 18:09:50 +03:00
chain : eth . BlockChain ( ) ,
2015-03-23 13:12:49 +02:00
possibleUncles : make ( map [ common . Hash ] * types . Block ) ,
2017-08-06 19:40:34 +03:00
unconfirmed : newUnconfirmedBlocks ( eth . BlockChain ( ) , miningLogAtDepth ) ,
2018-08-14 18:34:33 +03:00
txsCh : make ( chan core . NewTxsEvent , txChanSize ) ,
chainHeadCh : make ( chan core . ChainHeadEvent , chainHeadChanSize ) ,
chainSideCh : make ( chan core . ChainSideEvent , chainSideChanSize ) ,
newWork : make ( chan struct { } , 1 ) ,
taskCh : make ( chan * task ) ,
resultCh : make ( chan * task , resultQueueSize ) ,
exitCh : make ( chan struct { } ) ,
2015-02-09 17:20:34 +02:00
}
2018-05-18 11:45:52 +03:00
// Subscribe NewTxsEvent for tx pool
worker . txsSub = eth . TxPool ( ) . SubscribeNewTxsEvent ( worker . txsCh )
2017-08-18 13:58:36 +03:00
// Subscribe events for blockchain
worker . chainHeadSub = eth . BlockChain ( ) . SubscribeChainHeadEvent ( worker . chainHeadCh )
worker . chainSideSub = eth . BlockChain ( ) . SubscribeChainSideEvent ( worker . chainSideCh )
2015-04-07 13:32:55 +03:00
2018-08-14 18:34:33 +03:00
go worker . mainLoop ( )
go worker . resultLoop ( )
go worker . taskLoop ( )
2015-04-07 13:32:55 +03:00
2018-08-14 18:34:33 +03:00
// Submit first work to initialize pending state.
worker . newWork <- struct { } { }
2015-04-07 13:32:55 +03:00
return worker
2015-02-09 17:20:34 +02:00
}
2018-08-14 18:34:33 +03:00
// setEtherbase sets the etherbase used to initialize the block coinbase field.
func ( w * worker ) setEtherbase ( addr common . Address ) {
w . mu . Lock ( )
defer w . mu . Unlock ( )
w . coinbase = addr
2015-07-07 11:58:47 +03:00
}
2018-08-14 18:34:33 +03:00
// setExtra sets the content used to initialize the block extra field.
func ( w * worker ) setExtra ( extra [ ] byte ) {
w . mu . Lock ( )
defer w . mu . Unlock ( )
w . extra = extra
2016-12-13 15:03:18 +03:00
}
2018-08-14 18:34:33 +03:00
// pending returns the pending state and corresponding block.
func ( w * worker ) pending ( ) ( * types . Block , * state . StateDB ) {
2018-08-03 11:33:37 +03:00
// return a snapshot to avoid contention on currentMu mutex
2018-08-14 18:34:33 +03:00
w . snapshotMu . RLock ( )
defer w . snapshotMu . RUnlock ( )
if w . snapshotState == nil {
return nil , nil
}
return w . snapshotBlock , w . snapshotState . Copy ( )
2018-04-16 10:56:20 +03:00
}
2016-11-30 12:48:48 +03:00
2018-08-14 18:34:33 +03:00
// pendingBlock returns pending block.
func ( w * worker ) pendingBlock ( ) * types . Block {
2018-08-03 11:33:37 +03:00
// return a snapshot to avoid contention on currentMu mutex
2018-08-14 18:34:33 +03:00
w . snapshotMu . RLock ( )
defer w . snapshotMu . RUnlock ( )
return w . snapshotBlock
2016-11-30 12:48:48 +03:00
}
2018-08-14 18:34:33 +03:00
// start sets the running status as 1 and triggers new work submitting.
func ( w * worker ) start ( ) {
atomic . StoreInt32 ( & w . running , 1 )
w . newWork <- struct { } { }
2015-02-09 17:20:34 +02:00
}
2018-08-14 18:34:33 +03:00
// stop sets the running status as 0.
func ( w * worker ) stop ( ) {
atomic . StoreInt32 ( & w . running , 0 )
2015-02-09 17:20:34 +02:00
}
2018-08-14 18:34:33 +03:00
// isRunning returns an indicator whether worker is running or not.
func ( w * worker ) isRunning ( ) bool {
return atomic . LoadInt32 ( & w . running ) == 1
2018-08-03 11:33:37 +03:00
}
2018-08-14 18:34:33 +03:00
// close terminates all background threads maintained by the worker and cleans up buffered channels.
// Note the worker does not support being closed multiple times.
func ( w * worker ) close ( ) {
close ( w . exitCh )
// Clean up buffered channels
for empty := false ; ! empty ; {
select {
case <- w . resultCh :
default :
empty = true
}
2018-08-03 11:33:37 +03:00
}
2015-02-04 15:52:59 +02:00
}
2018-08-14 18:34:33 +03:00
// mainLoop is a standalone goroutine to regenerate the sealing task based on the received event.
func ( w * worker ) mainLoop ( ) {
defer w . txsSub . Unsubscribe ( )
defer w . chainHeadSub . Unsubscribe ( )
defer w . chainSideSub . Unsubscribe ( )
2017-08-18 13:58:36 +03:00
for {
select {
2018-08-14 18:34:33 +03:00
case <- w . newWork :
// Submit a work when the worker is created or started.
w . commitNewWork ( )
case <- w . chainHeadCh :
// Resubmit a work for new cycle once worker receives chain head event.
w . commitNewWork ( )
case ev := <- w . chainSideCh :
// Add side block to possible uncle block set.
w . possibleUncles [ ev . Block . Hash ( ) ] = ev . Block
case ev := <- w . txsCh :
2018-05-10 10:04:45 +03:00
// 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.
2018-08-14 18:34:33 +03:00
if ! w . isRunning ( ) && w . current != nil {
w . mu . Lock ( )
coinbase := w . coinbase
w . mu . Unlock ( )
2018-05-10 10:04:45 +03:00
txs := make ( map [ common . Address ] types . Transactions )
for _ , tx := range ev . Txs {
2018-08-14 18:34:33 +03:00
acc , _ := types . Sender ( w . current . signer , tx )
2018-05-10 10:04:45 +03:00
txs [ acc ] = append ( txs [ acc ] , tx )
}
2018-08-14 18:34:33 +03:00
txset := types . NewTransactionsByPriceAndNonce ( w . current . signer , txs )
w . current . commitTransactions ( w . mux , txset , w . chain , coinbase )
w . updateSnapshot ( )
2017-10-24 13:40:42 +03:00
} else {
// If we're mining, but nothing is being processed, wake on new transactions
2018-08-14 18:34:33 +03:00
if w . config . Clique != nil && w . config . Clique . Period == 0 {
w . commitNewWork ( )
2017-10-24 13:40:42 +03:00
}
2015-02-04 15:52:59 +02:00
}
2017-08-18 13:58:36 +03:00
// System stopped
2018-08-14 18:34:33 +03:00
case <- w . exitCh :
2017-08-18 13:58:36 +03:00
return
2018-08-14 18:34:33 +03:00
case <- w . txsSub . Err ( ) :
2017-08-18 13:58:36 +03:00
return
2018-08-14 18:34:33 +03:00
case <- w . chainHeadSub . Err ( ) :
return
case <- w . chainSideSub . Err ( ) :
2017-08-18 13:58:36 +03:00
return
2015-02-04 15:52:59 +02:00
}
}
}
2018-08-14 18:34:33 +03:00
// seal pushes a sealing task to consensus engine and submits the result.
func ( w * worker ) seal ( t * task , stop <- chan struct { } ) {
var (
err error
res * task
)
if t . block , err = w . engine . Seal ( w . chain , t . block , stop ) ; t . block != nil {
log . Info ( "Successfully sealed new block" , "number" , t . block . Number ( ) , "hash" , t . block . Hash ( ) ,
"elapsed" , common . PrettyDuration ( time . Since ( t . createdAt ) ) )
res = t
} else {
if err != nil {
log . Warn ( "Block sealing failed" , "err" , err )
}
res = nil
}
select {
case w . resultCh <- res :
case <- w . exitCh :
}
}
// taskLoop is a standalone goroutine to fetch sealing task from the generator and
// push them to consensus engine.
func ( w * worker ) taskLoop ( ) {
var stopCh chan struct { }
// interrupt aborts the in-flight sealing task.
interrupt := func ( ) {
if stopCh != nil {
close ( stopCh )
stopCh = nil
}
}
2015-02-09 17:20:34 +02:00
for {
2018-08-14 18:34:33 +03:00
select {
case task := <- w . taskCh :
if w . newTaskHook != nil {
w . newTaskHook ( task )
}
interrupt ( )
stopCh = make ( chan struct { } )
go w . seal ( task , stopCh )
case <- w . exitCh :
interrupt ( )
return
}
}
}
2015-03-26 18:45:03 +02:00
2018-08-14 18:34:33 +03:00
// resultLoop is a standalone goroutine to handle sealing result submitting
// and flush relative data to the database.
func ( w * worker ) resultLoop ( ) {
for {
select {
case result := <- w . resultCh :
2015-07-11 21:45:59 +03:00
if result == nil {
2015-03-26 18:45:03 +02:00
continue
}
2018-08-14 18:34:33 +03:00
block := result . block
2015-03-26 18:45:03 +02:00
2017-09-11 13:13:05 +03:00
// Update the block hash in all logs since it is now available and not when the
// receipt/log of individual transactions were created.
2018-08-14 18:34:33 +03:00
for _ , r := range result . receipts {
2017-09-11 13:13:05 +03:00
for _ , l := range r . Logs {
l . BlockHash = block . Hash ( )
2015-07-11 21:45:59 +03:00
}
2015-07-03 12:24:42 +03:00
}
2018-08-14 18:34:33 +03:00
for _ , log := range result . state . Logs ( ) {
2017-09-11 13:13:05 +03:00
log . BlockHash = block . Hash ( )
}
2018-08-14 18:34:33 +03:00
// Commit block and state to database.
stat , err := w . chain . WriteBlockWithState ( block , result . receipts , result . state )
2017-09-11 13:13:05 +03:00
if err != nil {
log . Error ( "Failed writing block to chain" , "err" , err )
continue
}
// Broadcast the block and announce chain insertion event
2018-08-14 18:34:33 +03:00
w . mux . Post ( core . NewMinedBlockEvent { Block : block } )
2017-09-11 13:13:05 +03:00
var (
events [ ] interface { }
2018-08-14 18:34:33 +03:00
logs = result . state . Logs ( )
2017-09-11 13:13:05 +03:00
)
2018-08-14 18:34:33 +03:00
switch stat {
case core . CanonStatTy :
events = append ( events , core . ChainEvent { Block : block , Hash : block . Hash ( ) , Logs : logs } )
2017-09-11 13:13:05 +03:00
events = append ( events , core . ChainHeadEvent { Block : block } )
2018-08-14 18:34:33 +03:00
case core . SideStatTy :
events = append ( events , core . ChainSideEvent { Block : block } )
2017-09-11 13:13:05 +03:00
}
2018-08-14 18:34:33 +03:00
w . chain . PostChainEvents ( events , logs )
2017-09-11 13:13:05 +03:00
2018-08-14 18:34:33 +03:00
// Insert the block into the set of pending ones to resultLoop for confirmations
w . unconfirmed . Insert ( block . NumberU64 ( ) , block . Hash ( ) )
2015-02-09 17:20:34 +02:00
2018-08-14 18:34:33 +03:00
case <- w . exitCh :
return
}
2015-02-04 15:52:59 +02:00
}
}
2015-06-16 13:41:50 +03:00
// makeCurrent creates a new environment for the current cycle.
2018-08-14 18:34:33 +03:00
func ( w * worker ) makeCurrent ( parent * types . Block , header * types . Header ) error {
state , err := w . chain . StateAt ( parent . Root ( ) )
2015-10-06 17:35:55 +03:00
if err != nil {
return err
}
2018-08-06 12:55:44 +03:00
env := & Env {
2018-08-14 18:34:33 +03:00
config : w . config ,
signer : types . NewEIP155Signer ( w . config . ChainID ) ,
2015-06-16 13:41:50 +03:00
state : state ,
2018-07-16 10:54:19 +03:00
ancestors : mapset . NewSet ( ) ,
family : mapset . NewSet ( ) ,
uncles : mapset . NewSet ( ) ,
2015-06-16 13:41:50 +03:00
header : header ,
2015-04-04 14:27:17 +03:00
}
2015-03-05 10:14:58 +02:00
2015-05-18 17:31:26 +03:00
// when 08 is processed ancestors contain 07 (quick block)
2018-08-14 18:34:33 +03:00
for _ , ancestor := range w . chain . GetBlocksFromHash ( parent . Hash ( ) , 7 ) {
2015-05-14 03:46:23 +03:00
for _ , uncle := range ancestor . Uncles ( ) {
2018-08-06 12:55:44 +03:00
env . family . Add ( uncle . Hash ( ) )
2015-05-14 03:46:23 +03:00
}
2018-08-06 12:55:44 +03:00
env . family . Add ( ancestor . Hash ( ) )
env . ancestors . Add ( ancestor . Hash ( ) )
2015-03-23 13:12:49 +02:00
}
2017-08-06 19:40:34 +03:00
2015-05-11 02:28:15 +03:00
// Keep track of transactions which return errors so they can be removed
2018-08-06 12:55:44 +03:00
env . tcount = 0
2018-08-14 18:34:33 +03:00
w . current = env
2015-10-06 17:35:55 +03:00
return nil
2015-04-07 13:32:55 +03:00
}
2018-08-14 18:34:33 +03:00
// commitUncle adds the given block to uncle block set, returns error if failed to add.
func ( w * worker ) commitUncle ( env * Env , uncle * types . Header ) error {
hash := uncle . Hash ( )
if env . uncles . Contains ( hash ) {
return fmt . Errorf ( "uncle not unique" )
}
if ! env . ancestors . Contains ( uncle . ParentHash ) {
return fmt . Errorf ( "uncle's parent unknown (%x)" , uncle . ParentHash [ 0 : 4 ] )
}
if env . family . Contains ( hash ) {
return fmt . Errorf ( "uncle already in family (%x)" , hash )
}
env . uncles . Add ( uncle . Hash ( ) )
return nil
}
// updateSnapshot updates pending snapshot block and state.
// Note this function assumes the current variable is thread safe.
func ( w * worker ) updateSnapshot ( ) {
w . snapshotMu . Lock ( )
defer w . snapshotMu . Unlock ( )
var uncles [ ] * types . Header
w . current . uncles . Each ( func ( item interface { } ) bool {
hash , ok := item . ( common . Hash )
if ! ok {
return false
}
uncle , exist := w . possibleUncles [ hash ]
if ! exist {
return false
}
uncles = append ( uncles , uncle . Header ( ) )
return true
} )
w . snapshotBlock = types . NewBlock (
w . current . header ,
w . current . txs ,
uncles ,
w . current . receipts ,
)
w . snapshotState = w . current . state . Copy ( )
}
// commitNewWork generates several new sealing tasks based on the parent block.
func ( w * worker ) commitNewWork ( ) {
w . mu . RLock ( )
defer w . mu . RUnlock ( )
2015-04-07 13:32:55 +03:00
2015-06-15 12:33:08 +03:00
tstart := time . Now ( )
2018-08-14 18:34:33 +03:00
parent := w . chain . CurrentBlock ( )
2016-11-28 16:59:06 +03:00
2015-06-16 13:41:50 +03:00
tstamp := tstart . Unix ( )
2015-09-03 00:24:17 +03:00
if parent . Time ( ) . Cmp ( new ( big . Int ) . SetInt64 ( tstamp ) ) >= 0 {
2015-08-24 03:52:53 +03:00
tstamp = parent . Time ( ) . Int64 ( ) + 1
2015-06-16 13:41:50 +03:00
}
2015-06-29 19:55:49 +03:00
// this will ensure we're not going off too far in the future
2017-04-05 01:16:29 +03:00
if now := time . Now ( ) . Unix ( ) ; tstamp > now + 1 {
2015-06-29 19:55:49 +03:00
wait := time . Duration ( tstamp - now ) * time . Second
2017-04-05 01:16:29 +03:00
log . Info ( "Mining too far in the future" , "wait" , common . PrettyDuration ( wait ) )
2015-06-29 19:55:49 +03:00
time . Sleep ( wait )
}
2015-06-16 13:41:50 +03:00
num := parent . Number ( )
header := & types . Header {
ParentHash : parent . Hash ( ) ,
Number : num . Add ( num , common . Big1 ) ,
GasLimit : core . CalcGasLimit ( parent ) ,
2018-08-14 18:34:33 +03:00
Extra : w . extra ,
2015-08-24 03:52:53 +03:00
Time : big . NewInt ( tstamp ) ,
2015-06-16 13:41:50 +03:00
}
2018-08-03 11:33:37 +03:00
// Only set the coinbase if our consensus engine is running (avoid spurious block rewards)
2018-08-14 18:34:33 +03:00
if w . isRunning ( ) {
if w . coinbase == ( common . Address { } ) {
2018-08-03 11:33:37 +03:00
log . Error ( "Refusing to mine without etherbase" )
return
}
2018-08-14 18:34:33 +03:00
header . Coinbase = w . coinbase
2017-04-05 01:16:29 +03:00
}
2018-08-14 18:34:33 +03:00
if err := w . engine . Prepare ( w . chain , header ) ; err != nil {
2017-04-05 01:16:29 +03:00
log . Error ( "Failed to prepare header for mining" , "err" , err )
return
}
2016-07-08 18:48:17 +03:00
// If we are care about TheDAO hard-fork check whether to override the extra-data or not
2018-08-14 18:34:33 +03:00
if daoBlock := w . config . DAOForkBlock ; daoBlock != nil {
2016-07-08 13:00:37 +03:00
// Check whether the block is among the fork extra-override range
limit := new ( big . Int ) . Add ( daoBlock , params . DAOForkExtraRange )
2016-07-14 11:22:58 +03:00
if header . Number . Cmp ( daoBlock ) >= 0 && header . Number . Cmp ( limit ) < 0 {
2016-07-08 18:48:17 +03:00
// Depending whether we support or oppose the fork, override differently
2018-08-14 18:34:33 +03:00
if w . config . DAOForkSupport {
2016-07-08 18:48:17 +03:00
header . Extra = common . CopyBytes ( params . DAOForkBlockExtra )
2017-01-06 18:44:20 +03:00
} else if bytes . Equal ( header . Extra , params . DAOForkBlockExtra ) {
2016-07-08 18:48:17 +03:00
header . Extra = [ ] byte { } // If miner opposes, don't let it use the reserved extra-data
}
2016-07-08 13:00:37 +03:00
}
}
2015-10-06 17:35:55 +03:00
// Could potentially happen if starting to mine in an odd state.
2018-08-14 18:34:33 +03:00
err := w . makeCurrent ( parent , header )
2015-10-06 17:35:55 +03:00
if err != nil {
2017-04-05 01:16:29 +03:00
log . Error ( "Failed to create mining context" , "err" , err )
2015-10-06 17:35:55 +03:00
return
}
2016-07-11 13:55:11 +03:00
// Create the current work task and check any fork transitions needed
2018-08-14 18:34:33 +03:00
env := w . current
if w . config . DAOForkSupport && w . config . DAOForkBlock != nil && w . config . DAOForkBlock . Cmp ( header . Number ) == 0 {
2018-08-06 12:55:44 +03:00
misc . ApplyDAOHardFork ( env . state )
2016-07-11 13:55:11 +03:00
}
2016-06-27 18:28:34 +03:00
2015-06-16 13:41:50 +03:00
// compute uncles for the new block.
2015-03-23 19:27:05 +02:00
var (
uncles [ ] * types . Header
badUncles [ ] common . Hash
)
2018-08-14 18:34:33 +03:00
for hash , uncle := range w . possibleUncles {
2015-03-23 17:14:33 +02:00
if len ( uncles ) == 2 {
2015-03-23 13:12:49 +02:00
break
}
2018-08-14 18:34:33 +03:00
if err := w . commitUncle ( env , uncle . Header ( ) ) ; err != nil {
2017-04-05 01:16:29 +03:00
log . Trace ( "Bad uncle found and will be removed" , "hash" , hash )
2017-02-22 15:10:07 +03:00
log . Trace ( fmt . Sprint ( uncle ) )
2015-03-23 19:27:05 +02:00
badUncles = append ( badUncles , hash )
2015-03-23 13:12:49 +02:00
} else {
2017-04-05 01:16:29 +03:00
log . Debug ( "Committing new uncle to block" , "hash" , hash )
2015-03-23 17:14:33 +02:00
uncles = append ( uncles , uncle . Header ( ) )
2015-03-23 13:12:49 +02:00
}
}
2015-03-23 19:27:05 +02:00
for _ , hash := range badUncles {
2018-08-14 18:34:33 +03:00
delete ( w . possibleUncles , hash )
2015-03-23 19:27:05 +02:00
}
2018-08-03 11:33:37 +03:00
2018-08-06 12:55:44 +03:00
var (
2018-08-14 18:34:33 +03:00
emptyBlock , fullBlock * types . Block
emptyState , fullState * state . StateDB
2018-08-06 12:55:44 +03:00
)
2018-08-03 11:33:37 +03:00
// Create an empty block based on temporary copied state for sealing in advance without waiting block
// execution finished.
2018-08-14 18:34:33 +03:00
emptyState = env . state . Copy ( )
if emptyBlock , err = w . engine . Finalize ( w . chain , header , emptyState , nil , uncles , nil ) ; err != nil {
2018-08-03 11:33:37 +03:00
log . Error ( "Failed to finalize block for temporary sealing" , "err" , err )
} else {
// Push empty work in advance without applying pending transaction.
// The reason is transactions execution can cost a lot and sealer need to
// take advantage of this part time.
2018-08-14 18:34:33 +03:00
if w . isRunning ( ) {
select {
case w . taskCh <- & task { receipts : nil , state : emptyState , block : emptyBlock , createdAt : time . Now ( ) } :
log . Info ( "Commit new empty mining work" , "number" , emptyBlock . Number ( ) , "uncles" , len ( uncles ) )
case <- w . exitCh :
log . Info ( "Worker has exited" )
return
}
2018-08-03 11:33:37 +03:00
}
}
// Fill the block with all available pending transactions.
2018-08-14 18:34:33 +03:00
pending , err := w . eth . TxPool ( ) . Pending ( )
2018-08-03 11:33:37 +03:00
if err != nil {
log . Error ( "Failed to fetch pending transactions" , "err" , err )
return
}
2018-08-14 18:34:33 +03:00
// Short circuit if there is no available pending transactions
if len ( pending ) == 0 {
w . updateSnapshot ( )
return
}
txs := types . NewTransactionsByPriceAndNonce ( w . current . signer , pending )
env . commitTransactions ( w . mux , txs , w . chain , w . coinbase )
2018-08-03 11:33:37 +03:00
// Create the full block to seal with the consensus engine
2018-08-14 18:34:33 +03:00
fullState = env . state . Copy ( )
if fullBlock , err = w . engine . Finalize ( w . chain , header , fullState , env . txs , uncles , env . receipts ) ; err != nil {
2017-04-05 01:16:29 +03:00
log . Error ( "Failed to finalize block for sealing" , "err" , err )
return
2015-06-29 16:17:05 +03:00
}
2018-08-14 18:34:33 +03:00
// Deep copy receipts here to avoid interaction between different tasks.
cpy := make ( [ ] * types . Receipt , len ( env . receipts ) )
for i , l := range env . receipts {
cpy [ i ] = new ( types . Receipt )
* cpy [ i ] = * l
2018-05-10 10:04:45 +03:00
}
2018-08-14 18:34:33 +03:00
// We only care about logging if we're actually mining.
if w . isRunning ( ) {
if w . fullTaskInterval != nil {
w . fullTaskInterval ( )
2015-05-11 02:28:15 +03:00
}
2016-11-28 16:59:06 +03:00
2018-08-14 18:34:33 +03:00
select {
case w . taskCh <- & task { receipts : cpy , state : fullState , block : fullBlock , createdAt : time . Now ( ) } :
w . unconfirmed . Shift ( fullBlock . NumberU64 ( ) - 1 )
log . Info ( "Commit new full mining work" , "number" , fullBlock . Number ( ) , "txs" , env . tcount , "uncles" , len ( uncles ) , "elapsed" , common . PrettyDuration ( time . Since ( tstart ) ) )
case <- w . exitCh :
log . Info ( "Worker has exited" )
2016-11-28 16:59:06 +03:00
}
2015-02-04 15:52:59 +02:00
}
2018-08-14 18:34:33 +03:00
w . updateSnapshot ( )
2015-02-04 15:52:59 +02:00
}