| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2015 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-23 18:35:11 +02:00
										 |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02: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 18:48:40 +02:00
										 |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02: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 18:48:40 +02:00
										 |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | package miner | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2016-07-08 18:48:17 +03:00
										 |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"math/big" | 
					
						
							| 
									
										
										
										
											2015-02-15 16:16:27 +01:00
										 |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2015-03-26 17:45:03 +01:00
										 |  |  | 	"sync/atomic" | 
					
						
							| 
									
										
										
										
											2015-06-15 11:33:08 +02:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-16 00:54:19 -07:00
										 |  |  | 	mapset "github.com/deckarep/golang-set" | 
					
						
							| 
									
										
										
										
											2015-03-18 13:00:01 +01: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 05:52:59 -08:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core" | 
					
						
							| 
									
										
										
										
											2015-03-23 18:27:05 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/state" | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							| 
									
										
										
										
											2015-08-30 10:19:10 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/vm" | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | 	"github.com/ethereum/go-ethereum/event" | 
					
						
							| 
									
										
										
										
											2017-02-22 14:10:07 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/log" | 
					
						
							| 
									
										
										
										
											2016-07-08 13:00:37 +03:00
										 |  |  | 	"github.com/ethereum/go-ethereum/params" | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 20:45:59 +02:00
										 |  |  | const ( | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	// resultQueueSize is the size of channel listening to sealing result. | 
					
						
							|  |  |  | 	resultQueueSize = 10 | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-18 11:45:52 +03:00
										 |  |  | 	// txChanSize is the size of channel listening to NewTxsEvent. | 
					
						
							| 
									
										
										
										
											2017-08-18 18:58:36 +08:00
										 |  |  | 	// The number is referenced from the size of tx pool. | 
					
						
							|  |  |  | 	txChanSize = 4096 | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-18 18:58:36 +08:00
										 |  |  | 	// chainHeadChanSize is the size of channel listening to ChainHeadEvent. | 
					
						
							|  |  |  | 	chainHeadChanSize = 10 | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-18 18:58:36 +08:00
										 |  |  | 	// chainSideChanSize is the size of channel listening to ChainSideEvent. | 
					
						
							|  |  |  | 	chainSideChanSize = 10 | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// miningLogAtDepth is the number of confirmations before logging successful mining. | 
					
						
							|  |  |  | 	miningLogAtDepth = 5 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// blockRecommitInterval is the time interval to recreate the mining block with | 
					
						
							|  |  |  | 	// any newly arrived transactions. | 
					
						
							|  |  |  | 	blockRecommitInterval = 3 * time.Second | 
					
						
							| 
									
										
										
										
											2015-07-11 20:45:59 +02:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2015-05-11 21:47:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | // environment is the worker's current environment and holds all of the current state information. | 
					
						
							|  |  |  | type environment struct { | 
					
						
							| 
									
										
										
										
											2016-11-02 13:44:13 +01:00
										 |  |  | 	signer types.Signer | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-16 22:07:27 +03:00
										 |  |  | 	state     *state.StateDB // apply state changes here | 
					
						
							| 
									
										
										
										
											2018-07-16 00:54:19 -07: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 05:52:59 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 12:41:50 +02:00
										 |  |  | 	header   *types.Header | 
					
						
							|  |  |  | 	txs      []*types.Transaction | 
					
						
							|  |  |  | 	receipts []*types.Receipt | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-07-11 20:45:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | const ( | 
					
						
							|  |  |  | 	commitInterruptNone int32 = iota | 
					
						
							|  |  |  | 	commitInterruptNewHead | 
					
						
							|  |  |  | 	commitInterruptResubmit | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type newWorkReq struct { | 
					
						
							|  |  |  | 	interrupt *int32 | 
					
						
							|  |  |  | 	noempty   bool | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | // worker is the main object which takes care of submitting new work to consensus engine | 
					
						
							|  |  |  | // and gathering the sealing result. | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | type worker struct { | 
					
						
							| 
									
										
										
										
											2016-10-20 13:36:29 +02:00
										 |  |  | 	config *params.ChainConfig | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 	engine consensus.Engine | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	eth    Backend | 
					
						
							|  |  |  | 	chain  *core.BlockChain | 
					
						
							| 
									
										
										
										
											2016-03-01 23:32:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	// Subscriptions | 
					
						
							| 
									
										
										
										
											2017-08-18 18:58:36 +08:00
										 |  |  | 	mux          *event.TypeMux | 
					
						
							| 
									
										
										
										
											2018-05-18 11:45:52 +03:00
										 |  |  | 	txsCh        chan core.NewTxsEvent | 
					
						
							| 
									
										
										
										
											2018-05-10 15:04:45 +08:00
										 |  |  | 	txsSub       event.Subscription | 
					
						
							| 
									
										
										
										
											2017-08-18 18:58:36 +08:00
										 |  |  | 	chainHeadCh  chan core.ChainHeadEvent | 
					
						
							|  |  |  | 	chainHeadSub event.Subscription | 
					
						
							|  |  |  | 	chainSideCh  chan core.ChainSideEvent | 
					
						
							|  |  |  | 	chainSideSub event.Subscription | 
					
						
							| 
									
										
										
										
											2016-03-29 03:08:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	// Channels | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 	newWorkCh chan *newWorkReq | 
					
						
							|  |  |  | 	taskCh    chan *task | 
					
						
							|  |  |  | 	resultCh  chan *task | 
					
						
							|  |  |  | 	startCh   chan struct{} | 
					
						
							|  |  |  | 	exitCh    chan struct{} | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 	current        *environment                 // An environment for current running cycle. | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	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 18:57:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	mu       sync.RWMutex // The lock used to protect the coinbase and extra fields | 
					
						
							| 
									
										
										
										
											2015-03-18 13:00:01 +01:00
										 |  |  | 	coinbase common.Address | 
					
						
							| 
									
										
										
										
											2015-04-05 18:57:03 +02:00
										 |  |  | 	extra    []byte | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	snapshotMu    sync.RWMutex // The lock used to protect the block snapshot and state snapshot | 
					
						
							| 
									
										
										
										
											2018-04-16 00:56:20 -07:00
										 |  |  | 	snapshotBlock *types.Block | 
					
						
							|  |  |  | 	snapshotState *state.StateDB | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-22 10:58:43 +02:00
										 |  |  | 	// atomic status counters | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	running int32 // The indicator whether the consensus engine is running or not. | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Test hooks | 
					
						
							| 
									
										
										
										
											2018-08-15 19:09:17 +08:00
										 |  |  | 	newTaskHook  func(*task)      // Method to call upon receiving a new sealing task | 
					
						
							|  |  |  | 	skipSealHook func(*task) bool // Method to decide whether skipping the sealing. | 
					
						
							|  |  |  | 	fullTaskHook func()           // Method to call before pushing the full sealing task | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | func newWorker(config *params.ChainConfig, engine consensus.Engine, eth Backend, mux *event.TypeMux) *worker { | 
					
						
							| 
									
										
										
										
											2015-04-07 12:32:55 +02:00
										 |  |  | 	worker := &worker{ | 
					
						
							| 
									
										
										
										
											2016-03-01 23:32:43 +01:00
										 |  |  | 		config:         config, | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 		engine:         engine, | 
					
						
							| 
									
										
										
										
											2015-03-23 12:12:49 +01:00
										 |  |  | 		eth:            eth, | 
					
						
							| 
									
										
										
										
											2016-08-15 20:14:05 +02:00
										 |  |  | 		mux:            mux, | 
					
						
							| 
									
										
										
										
											2015-08-31 17:09:50 +02:00
										 |  |  | 		chain:          eth.BlockChain(), | 
					
						
							| 
									
										
										
										
											2015-03-23 12:12:49 +01:00
										 |  |  | 		possibleUncles: make(map[common.Hash]*types.Block), | 
					
						
							| 
									
										
										
										
											2017-08-06 19:40:34 +03:00
										 |  |  | 		unconfirmed:    newUnconfirmedBlocks(eth.BlockChain(), miningLogAtDepth), | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		txsCh:          make(chan core.NewTxsEvent, txChanSize), | 
					
						
							|  |  |  | 		chainHeadCh:    make(chan core.ChainHeadEvent, chainHeadChanSize), | 
					
						
							|  |  |  | 		chainSideCh:    make(chan core.ChainSideEvent, chainSideChanSize), | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 		newWorkCh:      make(chan *newWorkReq), | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		taskCh:         make(chan *task), | 
					
						
							|  |  |  | 		resultCh:       make(chan *task, resultQueueSize), | 
					
						
							|  |  |  | 		exitCh:         make(chan struct{}), | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 		startCh:        make(chan struct{}, 1), | 
					
						
							| 
									
										
										
										
											2015-02-09 16:20:34 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-05-18 11:45:52 +03:00
										 |  |  | 	// Subscribe NewTxsEvent for tx pool | 
					
						
							|  |  |  | 	worker.txsSub = eth.TxPool().SubscribeNewTxsEvent(worker.txsCh) | 
					
						
							| 
									
										
										
										
											2017-08-18 18:58:36 +08:00
										 |  |  | 	// Subscribe events for blockchain | 
					
						
							|  |  |  | 	worker.chainHeadSub = eth.BlockChain().SubscribeChainHeadEvent(worker.chainHeadCh) | 
					
						
							|  |  |  | 	worker.chainSideSub = eth.BlockChain().SubscribeChainSideEvent(worker.chainSideCh) | 
					
						
							| 
									
										
										
										
											2015-04-07 12:32:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	go worker.mainLoop() | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 	go worker.newWorkLoop() | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	go worker.resultLoop() | 
					
						
							|  |  |  | 	go worker.taskLoop() | 
					
						
							| 
									
										
										
										
											2015-04-07 12:32:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	// Submit first work to initialize pending state. | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 	worker.startCh <- struct{}{} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 12:32:55 +02:00
										 |  |  | 	return worker | 
					
						
							| 
									
										
										
										
											2015-02-09 16:20:34 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 10:58:47 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 14:03:18 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | // pending returns the pending state and corresponding block. | 
					
						
							|  |  |  | func (w *worker) pending() (*types.Block, *state.StateDB) { | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	// return a snapshot to avoid contention on currentMu mutex | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	w.snapshotMu.RLock() | 
					
						
							|  |  |  | 	defer w.snapshotMu.RUnlock() | 
					
						
							|  |  |  | 	if w.snapshotState == nil { | 
					
						
							|  |  |  | 		return nil, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return w.snapshotBlock, w.snapshotState.Copy() | 
					
						
							| 
									
										
										
										
											2018-04-16 00:56:20 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-11-30 10:48:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | // pendingBlock returns pending block. | 
					
						
							|  |  |  | func (w *worker) pendingBlock() *types.Block { | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	// return a snapshot to avoid contention on currentMu mutex | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	w.snapshotMu.RLock() | 
					
						
							|  |  |  | 	defer w.snapshotMu.RUnlock() | 
					
						
							|  |  |  | 	return w.snapshotBlock | 
					
						
							| 
									
										
										
										
											2016-11-30 10:48:48 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | // start sets the running status as 1 and triggers new work submitting. | 
					
						
							|  |  |  | func (w *worker) start() { | 
					
						
							|  |  |  | 	atomic.StoreInt32(&w.running, 1) | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 	w.startCh <- struct{}{} | 
					
						
							| 
									
										
										
										
											2015-02-09 16:20:34 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | // stop sets the running status as 0. | 
					
						
							|  |  |  | func (w *worker) stop() { | 
					
						
							|  |  |  | 	atomic.StoreInt32(&w.running, 0) | 
					
						
							| 
									
										
										
										
											2015-02-09 16:20:34 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 16:33:37 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 16:33:37 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | // newWorkLoop is a standalone goroutine to submit new mining work upon received events. | 
					
						
							|  |  |  | func (w *worker) newWorkLoop() { | 
					
						
							|  |  |  | 	var interrupt *int32 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	timer := time.NewTimer(0) | 
					
						
							|  |  |  | 	<-timer.C // discard the initial tick | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// recommit aborts in-flight transaction execution with given signal and resubmits a new one. | 
					
						
							|  |  |  | 	recommit := func(noempty bool, s int32) { | 
					
						
							|  |  |  | 		if interrupt != nil { | 
					
						
							|  |  |  | 			atomic.StoreInt32(interrupt, s) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		interrupt = new(int32) | 
					
						
							|  |  |  | 		w.newWorkCh <- &newWorkReq{interrupt: interrupt, noempty: noempty} | 
					
						
							|  |  |  | 		timer.Reset(blockRecommitInterval) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for { | 
					
						
							|  |  |  | 		select { | 
					
						
							|  |  |  | 		case <-w.startCh: | 
					
						
							|  |  |  | 			recommit(false, commitInterruptNewHead) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case <-w.chainHeadCh: | 
					
						
							|  |  |  | 			recommit(false, commitInterruptNewHead) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case <-timer.C: | 
					
						
							|  |  |  | 			// If mining is running resubmit a new work cycle periodically to pull in | 
					
						
							|  |  |  | 			// higher priced transactions. Disable this overhead for pending blocks. | 
					
						
							|  |  |  | 			if w.isRunning() && (w.config.Clique == nil || w.config.Clique.Period > 0) { | 
					
						
							|  |  |  | 				recommit(true, commitInterruptResubmit) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case <-w.exitCh: | 
					
						
							|  |  |  | 			return | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 18:58:36 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for { | 
					
						
							|  |  |  | 		select { | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 		case req := <-w.newWorkCh: | 
					
						
							|  |  |  | 			w.commitNewWork(req.interrupt, req.noempty) | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		case ev := <-w.chainSideCh: | 
					
						
							| 
									
										
										
										
											2018-08-15 19:09:17 +08:00
										 |  |  | 			if _, exist := w.possibleUncles[ev.Block.Hash()]; exist { | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 			// Add side block to possible uncle block set. | 
					
						
							|  |  |  | 			w.possibleUncles[ev.Block.Hash()] = ev.Block | 
					
						
							| 
									
										
										
										
											2018-08-15 19:09:17 +08:00
										 |  |  | 			// If our mining block contains less than 2 uncle blocks, | 
					
						
							|  |  |  | 			// add the new uncle block if valid and regenerate a mining block. | 
					
						
							|  |  |  | 			if w.isRunning() && w.current != nil && w.current.uncles.Cardinality() < 2 { | 
					
						
							|  |  |  | 				start := time.Now() | 
					
						
							|  |  |  | 				if err := w.commitUncle(w.current, ev.Block.Header()); err == nil { | 
					
						
							|  |  |  | 					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.commit(uncles, nil, true, start) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		case ev := <-w.txsCh: | 
					
						
							| 
									
										
										
										
											2018-05-10 15:04:45 +08: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 23:34:33 +08:00
										 |  |  | 			if !w.isRunning() && w.current != nil { | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 				w.mu.RLock() | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 				coinbase := w.coinbase | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 				w.mu.RUnlock() | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-10 15:04:45 +08:00
										 |  |  | 				txs := make(map[common.Address]types.Transactions) | 
					
						
							|  |  |  | 				for _, tx := range ev.Txs { | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 					acc, _ := types.Sender(w.current.signer, tx) | 
					
						
							| 
									
										
										
										
											2018-05-10 15:04:45 +08:00
										 |  |  | 					txs[acc] = append(txs[acc], tx) | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 				txset := types.NewTransactionsByPriceAndNonce(w.current.signer, txs) | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 				w.commitTransactions(txset, coinbase, nil) | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 				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 23:34:33 +08:00
										 |  |  | 				if w.config.Clique != nil && w.config.Clique.Period == 0 { | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 					w.commitNewWork(nil, false) | 
					
						
							| 
									
										
										
										
											2017-10-24 13:40:42 +03:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-08-18 18:58:36 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// System stopped | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		case <-w.exitCh: | 
					
						
							| 
									
										
										
										
											2017-08-18 18:58:36 +08:00
										 |  |  | 			return | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		case <-w.txsSub.Err(): | 
					
						
							| 
									
										
										
										
											2017-08-18 18:58:36 +08:00
										 |  |  | 			return | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		case <-w.chainHeadSub.Err(): | 
					
						
							|  |  |  | 			return | 
					
						
							|  |  |  | 		case <-w.chainSideSub.Err(): | 
					
						
							| 
									
										
										
										
											2017-08-18 18:58:36 +08:00
										 |  |  | 			return | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-15 19:09:17 +08:00
										 |  |  | 	if w.skipSealHook != nil && w.skipSealHook(t) { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	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 16:20:34 +01:00
										 |  |  | 	for { | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 17:45:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 20:45:59 +02:00
										 |  |  | 			if result == nil { | 
					
						
							| 
									
										
										
										
											2015-03-26 17:45:03 +01:00
										 |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 			block := result.block | 
					
						
							| 
									
										
										
										
											2015-03-26 17:45:03 +01: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 23:34:33 +08: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 20:45:59 +02:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2015-07-03 11:24:42 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 			for _, log := range result.state.Logs() { | 
					
						
							| 
									
										
										
										
											2017-09-11 13:13:05 +03:00
										 |  |  | 				log.BlockHash = block.Hash() | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 23:34:33 +08:00
										 |  |  | 			w.mux.Post(core.NewMinedBlockEvent{Block: block}) | 
					
						
							| 
									
										
										
										
											2017-09-11 13:13:05 +03:00
										 |  |  | 			var ( | 
					
						
							|  |  |  | 				events []interface{} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 				logs   = result.state.Logs() | 
					
						
							| 
									
										
										
										
											2017-09-11 13:13:05 +03:00
										 |  |  | 			) | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 23:34:33 +08:00
										 |  |  | 			case core.SideStatTy: | 
					
						
							|  |  |  | 				events = append(events, core.ChainSideEvent{Block: block}) | 
					
						
							| 
									
										
										
										
											2017-09-11 13:13:05 +03:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 			w.chain.PostChainEvents(events, logs) | 
					
						
							| 
									
										
										
										
											2017-09-11 13:13:05 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 			// Insert the block into the set of pending ones to resultLoop for confirmations | 
					
						
							|  |  |  | 			w.unconfirmed.Insert(block.NumberU64(), block.Hash()) | 
					
						
							| 
									
										
										
										
											2015-02-09 16:20:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		case <-w.exitCh: | 
					
						
							|  |  |  | 			return | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 12:41:50 +02:00
										 |  |  | // makeCurrent creates a new environment for the current cycle. | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | func (w *worker) makeCurrent(parent *types.Block, header *types.Header) error { | 
					
						
							|  |  |  | 	state, err := w.chain.StateAt(parent.Root()) | 
					
						
							| 
									
										
										
										
											2015-10-06 16:35:55 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 	env := &environment{ | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		signer:    types.NewEIP155Signer(w.config.ChainID), | 
					
						
							| 
									
										
										
										
											2015-06-16 12:41:50 +02:00
										 |  |  | 		state:     state, | 
					
						
							| 
									
										
										
										
											2018-07-16 00:54:19 -07:00
										 |  |  | 		ancestors: mapset.NewSet(), | 
					
						
							|  |  |  | 		family:    mapset.NewSet(), | 
					
						
							|  |  |  | 		uncles:    mapset.NewSet(), | 
					
						
							| 
									
										
										
										
											2015-06-16 12:41:50 +02:00
										 |  |  | 		header:    header, | 
					
						
							| 
									
										
										
										
											2015-04-04 13:27:17 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-05 09:14:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-18 15:31:26 +01:00
										 |  |  | 	// when 08 is processed ancestors contain 07 (quick block) | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	for _, ancestor := range w.chain.GetBlocksFromHash(parent.Hash(), 7) { | 
					
						
							| 
									
										
										
										
											2015-05-13 20:46:23 -04:00
										 |  |  | 		for _, uncle := range ancestor.Uncles() { | 
					
						
							| 
									
										
										
										
											2018-08-06 17:55:44 +08:00
										 |  |  | 			env.family.Add(uncle.Hash()) | 
					
						
							| 
									
										
										
										
											2015-05-13 20:46:23 -04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-06 17:55:44 +08:00
										 |  |  | 		env.family.Add(ancestor.Hash()) | 
					
						
							|  |  |  | 		env.ancestors.Add(ancestor.Hash()) | 
					
						
							| 
									
										
										
										
											2015-03-23 12:12:49 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-08-06 19:40:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 01:28:15 +02:00
										 |  |  | 	// Keep track of transactions which return errors so they can be removed | 
					
						
							| 
									
										
										
										
											2018-08-06 17:55:44 +08:00
										 |  |  | 	env.tcount = 0 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	w.current = env | 
					
						
							| 
									
										
										
										
											2015-10-06 16:35:55 +02:00
										 |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2015-04-07 12:32:55 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | // commitUncle adds the given block to uncle block set, returns error if failed to add. | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | func (w *worker) commitUncle(env *environment, uncle *types.Header) error { | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	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() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | func (w *worker) commitTransaction(tx *types.Transaction, coinbase common.Address) ([]*types.Log, error) { | 
					
						
							|  |  |  | 	snap := w.current.state.Snapshot() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	receipt, _, err := core.ApplyTransaction(w.config, w.chain, &coinbase, w.current.gasPool, w.current.state, w.current.header, tx, &w.current.header.GasUsed, vm.Config{}) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		w.current.state.RevertToSnapshot(snap) | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	w.current.txs = append(w.current.txs, tx) | 
					
						
							|  |  |  | 	w.current.receipts = append(w.current.receipts, receipt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return receipt.Logs, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (w *worker) commitTransactions(txs *types.TransactionsByPriceAndNonce, coinbase common.Address, interrupt *int32) bool { | 
					
						
							|  |  |  | 	// Short circuit if current is nil | 
					
						
							|  |  |  | 	if w.current == nil { | 
					
						
							|  |  |  | 		return true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if w.current.gasPool == nil { | 
					
						
							|  |  |  | 		w.current.gasPool = new(core.GasPool).AddGas(w.current.header.GasLimit) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var coalescedLogs []*types.Log | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for { | 
					
						
							|  |  |  | 		// In the following three cases, we will interrupt the execution of the transaction. | 
					
						
							|  |  |  | 		// (1) new head block event arrival, the interrupt signal is 1 | 
					
						
							|  |  |  | 		// (2) worker start or restart, the interrupt signal is 1 | 
					
						
							|  |  |  | 		// (3) worker recreate the mining block with any newly arrived transactions, the interrupt signal is 2. | 
					
						
							|  |  |  | 		// For the first two cases, the semi-finished work will be discarded. | 
					
						
							|  |  |  | 		// For the third case, the semi-finished work will be submitted to the consensus engine. | 
					
						
							|  |  |  | 		// TODO(rjl493456442) give feedback to newWorkLoop to adjust resubmit interval if it is too short. | 
					
						
							|  |  |  | 		if interrupt != nil && atomic.LoadInt32(interrupt) != commitInterruptNone { | 
					
						
							|  |  |  | 			return atomic.LoadInt32(interrupt) == commitInterruptNewHead | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// If we don't have enough gas for any further transactions then we're done | 
					
						
							|  |  |  | 		if w.current.gasPool.Gas() < params.TxGas { | 
					
						
							|  |  |  | 			log.Trace("Not enough gas for further transactions", "have", w.current.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(w.current.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() && !w.config.IsEIP155(w.current.header.Number) { | 
					
						
							|  |  |  | 			log.Trace("Ignoring reply protected transaction", "hash", tx.Hash(), "eip155", w.config.EIP155Block) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			txs.Pop() | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Start executing the transaction | 
					
						
							|  |  |  | 		w.current.state.Prepare(tx.Hash(), common.Hash{}, w.current.tcount) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		logs, err := w.commitTransaction(tx, coinbase) | 
					
						
							|  |  |  | 		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...) | 
					
						
							|  |  |  | 			w.current.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 !w.isRunning() && len(coalescedLogs) > 0 { | 
					
						
							|  |  |  | 		// We don't push the pendingLogsEvent while we are mining. The reason is that | 
					
						
							|  |  |  | 		// when we are mining, the worker will regenerate a mining block every 3 seconds. | 
					
						
							|  |  |  | 		// In order to avoid pushing the repeated pendingLog, we disable the pending log pushing. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// 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 w.mux.Post(core.PendingLogsEvent{Logs: cpy}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | // commitNewWork generates several new sealing tasks based on the parent block. | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | func (w *worker) commitNewWork(interrupt *int32, noempty bool) { | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	w.mu.RLock() | 
					
						
							|  |  |  | 	defer w.mu.RUnlock() | 
					
						
							| 
									
										
										
										
											2015-04-07 12:32:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-15 11:33:08 +02:00
										 |  |  | 	tstart := time.Now() | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	parent := w.chain.CurrentBlock() | 
					
						
							| 
									
										
										
										
											2016-11-28 14:59:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 12:41:50 +02:00
										 |  |  | 	tstamp := tstart.Unix() | 
					
						
							| 
									
										
										
										
											2015-09-02 23:24:17 +02:00
										 |  |  | 	if parent.Time().Cmp(new(big.Int).SetInt64(tstamp)) >= 0 { | 
					
						
							| 
									
										
										
										
											2015-08-24 02:52:53 +02:00
										 |  |  | 		tstamp = parent.Time().Int64() + 1 | 
					
						
							| 
									
										
										
										
											2015-06-16 12:41:50 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-06-29 18:55:49 +02: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 18:55:49 +02: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 18:55:49 +02:00
										 |  |  | 		time.Sleep(wait) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 12:41:50 +02:00
										 |  |  | 	num := parent.Number() | 
					
						
							|  |  |  | 	header := &types.Header{ | 
					
						
							|  |  |  | 		ParentHash: parent.Hash(), | 
					
						
							|  |  |  | 		Number:     num.Add(num, common.Big1), | 
					
						
							|  |  |  | 		GasLimit:   core.CalcGasLimit(parent), | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		Extra:      w.extra, | 
					
						
							| 
									
										
										
										
											2015-08-24 02:52:53 +02:00
										 |  |  | 		Time:       big.NewInt(tstamp), | 
					
						
							| 
									
										
										
										
											2015-06-16 12:41:50 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	// Only set the coinbase if our consensus engine is running (avoid spurious block rewards) | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	if w.isRunning() { | 
					
						
							|  |  |  | 		if w.coinbase == (common.Address{}) { | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 			log.Error("Refusing to mine without etherbase") | 
					
						
							|  |  |  | 			return | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		header.Coinbase = w.coinbase | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 23:34:33 +08: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 23:34:33 +08:00
										 |  |  | 			if w.config.DAOForkSupport { | 
					
						
							| 
									
										
										
										
											2016-07-08 18:48:17 +03:00
										 |  |  | 				header.Extra = common.CopyBytes(params.DAOForkBlockExtra) | 
					
						
							| 
									
										
										
										
											2017-01-06 16:44:20 +01: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 16:35:55 +02:00
										 |  |  | 	// Could potentially happen if starting to mine in an odd state. | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	err := w.makeCurrent(parent, header) | 
					
						
							| 
									
										
										
										
											2015-10-06 16:35:55 +02:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 		log.Error("Failed to create mining context", "err", err) | 
					
						
							| 
									
										
										
										
											2015-10-06 16:35:55 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-07-11 13:55:11 +03:00
										 |  |  | 	// Create the current work task and check any fork transitions needed | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	env := w.current | 
					
						
							|  |  |  | 	if w.config.DAOForkSupport && w.config.DAOForkBlock != nil && w.config.DAOForkBlock.Cmp(header.Number) == 0 { | 
					
						
							| 
									
										
										
										
											2018-08-06 17:55:44 +08:00
										 |  |  | 		misc.ApplyDAOHardFork(env.state) | 
					
						
							| 
									
										
										
										
											2016-07-11 13:55:11 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-06-27 18:28:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 12:41:50 +02:00
										 |  |  | 	// compute uncles for the new block. | 
					
						
							| 
									
										
										
										
											2015-03-23 18:27:05 +01:00
										 |  |  | 	var ( | 
					
						
							|  |  |  | 		uncles    []*types.Header | 
					
						
							|  |  |  | 		badUncles []common.Hash | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	for hash, uncle := range w.possibleUncles { | 
					
						
							| 
									
										
										
										
											2015-03-23 16:14:33 +01:00
										 |  |  | 		if len(uncles) == 2 { | 
					
						
							| 
									
										
										
										
											2015-03-23 12:12:49 +01:00
										 |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08: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 14:10:07 +02:00
										 |  |  | 			log.Trace(fmt.Sprint(uncle)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-23 18:27:05 +01:00
										 |  |  | 			badUncles = append(badUncles, hash) | 
					
						
							| 
									
										
										
										
											2015-03-23 12:12:49 +01:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2017-04-05 01:16:29 +03:00
										 |  |  | 			log.Debug("Committing new uncle to block", "hash", hash) | 
					
						
							| 
									
										
										
										
											2015-03-23 16:14:33 +01:00
										 |  |  | 			uncles = append(uncles, uncle.Header()) | 
					
						
							| 
									
										
										
										
											2015-03-23 12:12:49 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-23 18:27:05 +01:00
										 |  |  | 	for _, hash := range badUncles { | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		delete(w.possibleUncles, hash) | 
					
						
							| 
									
										
										
										
											2015-03-23 18:27:05 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 	if !noempty { | 
					
						
							|  |  |  | 		// Create an empty block based on temporary copied state for sealing in advance without waiting block | 
					
						
							|  |  |  | 		// execution finished. | 
					
						
							|  |  |  | 		w.commit(uncles, nil, false, tstart) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Fill the block with all available pending transactions. | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	pending, err := w.eth.TxPool().Pending() | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		log.Error("Failed to fetch pending transactions", "err", err) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	// Short circuit if there is no available pending transactions | 
					
						
							|  |  |  | 	if len(pending) == 0 { | 
					
						
							|  |  |  | 		w.updateSnapshot() | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	txs := types.NewTransactionsByPriceAndNonce(w.current.signer, pending) | 
					
						
							| 
									
										
										
										
											2018-08-16 19:14:33 +08:00
										 |  |  | 	if w.commitTransactions(txs, w.coinbase, interrupt) { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-03 16:33:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-15 19:09:17 +08:00
										 |  |  | 	w.commit(uncles, w.fullTaskHook, true, tstart) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // commit runs any post-transaction state modifications, assembles the final block | 
					
						
							|  |  |  | // and commits new work if consensus engine is running. | 
					
						
							|  |  |  | func (w *worker) commit(uncles []*types.Header, interval func(), update bool, start time.Time) error { | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	// Deep copy receipts here to avoid interaction between different tasks. | 
					
						
							| 
									
										
										
										
											2018-08-15 19:09:17 +08:00
										 |  |  | 	receipts := make([]*types.Receipt, len(w.current.receipts)) | 
					
						
							|  |  |  | 	for i, l := range w.current.receipts { | 
					
						
							|  |  |  | 		receipts[i] = new(types.Receipt) | 
					
						
							|  |  |  | 		*receipts[i] = *l | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	s := w.current.state.Copy() | 
					
						
							|  |  |  | 	block, err := w.engine.Finalize(w.chain, w.current.header, s, w.current.txs, uncles, w.current.receipts) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2018-05-10 15:04:45 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 	if w.isRunning() { | 
					
						
							| 
									
										
										
										
											2018-08-15 19:09:17 +08:00
										 |  |  | 		if interval != nil { | 
					
						
							|  |  |  | 			interval() | 
					
						
							| 
									
										
										
										
											2015-05-11 01:28:15 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		select { | 
					
						
							| 
									
										
										
										
											2018-08-15 19:09:17 +08:00
										 |  |  | 		case w.taskCh <- &task{receipts: receipts, state: s, block: block, createdAt: time.Now()}: | 
					
						
							|  |  |  | 			w.unconfirmed.Shift(block.NumberU64() - 1) | 
					
						
							| 
									
										
										
										
											2018-08-16 14:47:49 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			feesWei := new(big.Int) | 
					
						
							| 
									
										
										
										
											2018-08-17 15:47:14 +03:00
										 |  |  | 			for i, tx := range block.Transactions() { | 
					
						
							|  |  |  | 				feesWei.Add(feesWei, new(big.Int).Mul(new(big.Int).SetUint64(receipts[i].GasUsed), tx.GasPrice())) | 
					
						
							| 
									
										
										
										
											2018-08-16 14:47:49 +03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			feesEth := new(big.Float).Quo(new(big.Float).SetInt(feesWei), new(big.Float).SetInt(big.NewInt(params.Ether))) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			log.Info("Commit new mining work", "number", block.Number(), "uncles", len(uncles), "txs", w.current.tcount, | 
					
						
							|  |  |  | 				"gas", block.GasUsed(), "fees", feesEth, "elapsed", common.PrettyDuration(time.Since(start))) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 23:34:33 +08:00
										 |  |  | 		case <-w.exitCh: | 
					
						
							|  |  |  | 			log.Info("Worker has exited") | 
					
						
							| 
									
										
										
										
											2016-11-28 14:59:06 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-15 19:09:17 +08:00
										 |  |  | 	if update { | 
					
						
							|  |  |  | 		w.updateSnapshot() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2015-02-04 05:52:59 -08:00
										 |  |  | } |