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
2015-05-10 23:12:18 +02:00
"github.com/ethereum/go-ethereum/accounts"
2015-03-18 13:00:01 +01:00
"github.com/ethereum/go-ethereum/common"
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-09-14 09:35:57 +02:00
"github.com/ethereum/go-ethereum/ethdb"
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
"github.com/ethereum/go-ethereum/pow"
"gopkg.in/fatih/set.v0"
)
2015-07-11 20:45:59 +02:00
const (
resultQueueSize = 10
miningLogAtDepth = 5
)
2015-05-11 21:47:34 +02:00
// Agent can register themself with the worker
type Agent interface {
2015-07-11 20:45:59 +02:00
Work ( ) chan <- * Work
SetReturnCh ( chan <- * Result )
2015-05-11 21:47:34 +02:00
Stop ( )
Start ( )
GetHashRate ( ) int64
}
2016-08-15 20:14:05 +02:00
// Work is the workers current environment and holds
2015-05-11 21:47:34 +02:00
// all of the current state information
2015-07-11 20:45:59 +02:00
type Work struct {
2016-11-02 13:44:13 +01:00
config * params . ChainConfig
signer types . Signer
2016-12-13 13:23:12 +02:00
state * state . StateDB // apply state changes here
ancestors * set . Set // ancestor set (used for checking uncle parent validity)
family * set . Set // family set (used for checking uncle invalidity)
uncles * set . Set // uncle set
tcount int // tx count in cycle
ownedAccounts * set . Set
lowGasTxs types . Transactions
failedTxs types . Transactions
2015-02-04 05:52:59 -08:00
2015-07-11 20:45:59 +02:00
Block * types . Block // the new block
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
2015-07-11 20:45:59 +02:00
createdAt time . Time
}
type Result struct {
Work * Work
Block * types . Block
2015-02-04 05:52:59 -08:00
}
2015-05-11 21:47:34 +02:00
// worker is the main object which takes care of applying messages to the new state
2015-02-04 05:52:59 -08:00
type worker struct {
2016-10-20 13:36:29 +02:00
config * params . ChainConfig
2016-03-01 23:32:43 +01:00
2015-03-26 17:45:03 +01:00
mu sync . Mutex
2016-03-29 03:08:16 +02:00
// update loop
mux * event . TypeMux
2016-12-10 19:02:14 +01:00
events * event . TypeMuxSubscription
2016-03-29 03:08:16 +02:00
wg sync . WaitGroup
2015-10-29 13:28:00 +01:00
agents map [ Agent ] struct { }
2015-07-11 20:45:59 +02:00
recv chan * Result
2015-02-04 05:52:59 -08:00
pow pow . PoW
2016-08-15 20:14:05 +02:00
eth Backend
2015-08-31 17:09:50 +02:00
chain * core . BlockChain
2015-10-19 16:08:17 +02:00
proc core . Validator
2015-09-14 09:35:57 +02:00
chainDb ethdb . Database
2015-04-05 18:57:03 +02:00
2015-03-18 13:00:01 +01:00
coinbase common . Address
2015-05-09 12:04:00 +02:00
gasPrice * big . Int
2015-04-05 18:57:03 +02:00
extra [ ] byte
2015-02-04 05:52:59 -08:00
2015-04-07 12:32:55 +02:00
currentMu sync . Mutex
2015-07-11 20:45:59 +02:00
current * Work
2015-02-13 17:23:09 +01:00
2015-03-23 12:12:49 +01:00
uncleMu sync . Mutex
possibleUncles map [ common . Hash ] * types . Block
2015-04-07 12:32:55 +02:00
txQueueMu sync . Mutex
txQueue map [ common . Hash ] * types . Transaction
2016-12-13 15:10:52 +02:00
unconfirmed * unconfirmedBlocks // set of locally mined blocks pending canonicalness confirmations
2016-12-13 13:23:12 +02:00
2015-04-22 10:58:43 +02:00
// atomic status counters
mining int32
atWork int32
2015-07-11 20:45:59 +02:00
fullValidation bool
2015-02-04 05:52:59 -08:00
}
2016-10-20 13:36:29 +02:00
func newWorker ( config * params . ChainConfig , coinbase common . Address , 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 ,
2015-03-23 12:12:49 +01:00
eth : eth ,
2016-08-15 20:14:05 +02:00
mux : mux ,
2015-08-06 19:57:39 +02:00
chainDb : eth . ChainDb ( ) ,
2015-07-11 20:45:59 +02:00
recv : make ( chan * Result , resultQueueSize ) ,
2015-05-09 12:04:00 +02:00
gasPrice : new ( big . Int ) ,
2015-08-31 17:09:50 +02:00
chain : eth . BlockChain ( ) ,
2015-10-19 16:08:17 +02:00
proc : eth . BlockChain ( ) . Validator ( ) ,
2015-03-23 12:12:49 +01:00
possibleUncles : make ( map [ common . Hash ] * types . Block ) ,
coinbase : coinbase ,
2015-04-07 12:32:55 +02:00
txQueue : make ( map [ common . Hash ] * types . Transaction ) ,
2015-10-29 13:28:00 +01:00
agents : make ( map [ Agent ] struct { } ) ,
2016-12-13 15:10:52 +02:00
unconfirmed : newUnconfirmedBlocks ( eth . BlockChain ( ) , 5 ) ,
2015-07-11 20:45:59 +02:00
fullValidation : false ,
2015-02-09 16:20:34 +01:00
}
2016-03-29 03:08:16 +02:00
worker . events = worker . mux . Subscribe ( core . ChainHeadEvent { } , core . ChainSideEvent { } , core . TxPreEvent { } )
2015-04-07 12:32:55 +02:00
go worker . update ( )
2016-03-29 03:08:16 +02:00
go worker . wait ( )
2015-04-07 12:32:55 +02:00
worker . commitNewWork ( )
return worker
2015-02-09 16:20:34 +01:00
}
2015-07-07 10:58:47 +02:00
func ( self * worker ) setEtherbase ( addr common . Address ) {
self . mu . Lock ( )
defer self . mu . Unlock ( )
self . coinbase = addr
}
2016-12-13 14:03:18 +02:00
func ( self * worker ) setExtra ( extra [ ] byte ) {
self . mu . Lock ( )
defer self . mu . Unlock ( )
self . extra = extra
}
2016-03-16 11:20:02 +02:00
func ( self * worker ) pending ( ) ( * types . Block , * state . StateDB ) {
2015-04-07 12:32:55 +02:00
self . currentMu . Lock ( )
defer self . currentMu . Unlock ( )
2015-07-20 14:37:43 +03:00
2015-06-16 12:41:50 +02:00
if atomic . LoadInt32 ( & self . mining ) == 0 {
return types . NewBlock (
self . current . header ,
self . current . txs ,
nil ,
self . current . receipts ,
2016-10-18 21:34:37 +01:00
) , self . current . state . Copy ( )
2015-06-16 12:41:50 +02:00
}
2016-10-04 12:36:02 +02:00
return self . current . Block , self . current . state . Copy ( )
2015-04-07 12:32:55 +02:00
}
2016-11-30 10:48:48 +01:00
func ( self * worker ) pendingBlock ( ) * types . Block {
self . currentMu . Lock ( )
defer self . currentMu . Unlock ( )
if atomic . LoadInt32 ( & self . mining ) == 0 {
return types . NewBlock (
self . current . header ,
self . current . txs ,
nil ,
self . current . receipts ,
)
}
return self . current . Block
}
2015-04-07 12:32:55 +02:00
func ( self * worker ) start ( ) {
2015-05-09 12:32:36 +02:00
self . mu . Lock ( )
defer self . mu . Unlock ( )
2015-05-11 01:28:15 +02:00
atomic . StoreInt32 ( & self . mining , 1 )
2015-02-14 16:52:14 +01:00
// spin up agents
2015-10-29 13:28:00 +01:00
for agent := range self . agents {
2015-02-14 16:52:14 +01:00
agent . Start ( )
}
2015-02-09 16:20:34 +01:00
}
func ( self * worker ) stop ( ) {
2016-03-29 03:08:16 +02:00
self . wg . Wait ( )
2015-05-09 12:32:36 +02:00
self . mu . Lock ( )
defer self . mu . Unlock ( )
2015-04-22 10:58:43 +02:00
if atomic . LoadInt32 ( & self . mining ) == 1 {
2015-10-29 13:28:00 +01:00
// Stop all agents.
for agent := range self . agents {
2015-04-08 00:30:23 +02:00
agent . Stop ( )
2015-10-29 13:28:00 +01:00
// Remove CPU agents.
if _ , ok := agent . ( * CpuAgent ) ; ok {
delete ( self . agents , agent )
2015-05-11 15:43:14 +02:00
}
2015-04-08 00:30:23 +02:00
}
}
2015-02-13 17:23:09 +01:00
2015-04-22 10:58:43 +02:00
atomic . StoreInt32 ( & self . mining , 0 )
atomic . StoreInt32 ( & self . atWork , 0 )
2015-02-09 16:20:34 +01:00
}
func ( self * worker ) register ( agent Agent ) {
2015-05-09 12:32:36 +02:00
self . mu . Lock ( )
defer self . mu . Unlock ( )
2015-10-29 13:28:00 +01:00
self . agents [ agent ] = struct { } { }
2015-03-24 10:34:06 +01:00
agent . SetReturnCh ( self . recv )
2015-02-04 05:52:59 -08:00
}
2015-10-29 13:28:00 +01:00
func ( self * worker ) unregister ( agent Agent ) {
self . mu . Lock ( )
defer self . mu . Unlock ( )
delete ( self . agents , agent )
agent . Stop ( )
}
2015-02-04 05:52:59 -08:00
func ( self * worker ) update ( ) {
2016-03-29 03:08:16 +02:00
for event := range self . events . Chan ( ) {
// A real event arrived, process interesting content
switch ev := event . Data . ( type ) {
case core . ChainHeadEvent :
self . commitNewWork ( )
case core . ChainSideEvent :
self . uncleMu . Lock ( )
self . possibleUncles [ ev . Block . Hash ( ) ] = ev . Block
self . uncleMu . Unlock ( )
case core . TxPreEvent :
// Apply transaction to the pending state if we're not mining
if atomic . LoadInt32 ( & self . mining ) == 0 {
self . currentMu . Lock ( )
2016-08-09 14:54:36 +03:00
2016-11-02 13:44:13 +01:00
acc , _ := types . Sender ( self . current . signer , ev . Tx )
2017-01-06 15:52:03 +01:00
txs := map [ common . Address ] types . Transactions { acc : { ev . Tx } }
2016-08-09 14:54:36 +03:00
txset := types . NewTransactionsByPriceAndNonce ( txs )
self . current . commitTransactions ( self . mux , txset , self . gasPrice , self . chain )
2016-03-29 03:08:16 +02:00
self . currentMu . Unlock ( )
2015-02-04 05:52:59 -08:00
}
}
}
}
2015-02-09 16:20:34 +01:00
func ( self * worker ) wait ( ) {
for {
2016-11-28 14:59:06 +01:00
mustCommitNewWork := true
2015-07-11 20:45:59 +02:00
for result := range self . recv {
2015-04-22 10:58:43 +02:00
atomic . AddInt32 ( & self . atWork , - 1 )
2015-03-26 17:45:03 +01:00
2015-07-11 20:45:59 +02:00
if result == nil {
2015-03-26 17:45:03 +01:00
continue
}
2015-07-11 20:45:59 +02:00
block := result . Block
2015-08-04 20:38:57 +02:00
work := result . Work
2015-03-26 17:45:03 +01:00
2015-07-11 20:45:59 +02:00
if self . fullValidation {
if _ , err := self . chain . InsertChain ( types . Blocks { block } ) ; err != nil {
2017-02-22 14:10:07 +02:00
log . Error ( fmt . Sprint ( "mining err" , err ) )
2015-07-11 20:45:59 +02:00
continue
}
2016-04-15 11:06:57 +02:00
go self . mux . Post ( core . NewMinedBlockEvent { Block : block } )
2015-07-11 20:45:59 +02:00
} else {
2016-10-20 13:36:29 +02:00
work . state . Commit ( self . config . IsEIP158 ( block . Number ( ) ) )
2016-04-05 15:22:04 +02:00
parent := self . chain . GetBlock ( block . ParentHash ( ) , block . NumberU64 ( ) - 1 )
2015-07-11 20:45:59 +02:00
if parent == nil {
2017-02-22 14:10:07 +02:00
log . Error ( fmt . Sprint ( "Invalid block found during mining" ) )
2015-07-11 20:45:59 +02:00
continue
}
2015-10-19 16:08:17 +02:00
auxValidator := self . eth . BlockChain ( ) . AuxValidator ( )
2016-03-01 23:32:43 +01:00
if err := core . ValidateHeader ( self . config , auxValidator , block . Header ( ) , parent . Header ( ) , true , false ) ; err != nil && err != core . BlockFutureErr {
2017-02-22 14:10:07 +02:00
log . Error ( fmt . Sprint ( "Invalid header on mined block:" , err ) )
2015-07-11 20:45:59 +02:00
continue
}
2015-07-03 11:24:42 +02:00
2015-09-07 20:43:01 +03:00
stat , err := self . chain . WriteBlock ( block )
2015-07-11 20:45:59 +02:00
if err != nil {
2017-02-22 14:10:07 +02:00
log . Error ( fmt . Sprint ( "error writing block to chain" , err ) )
2015-07-11 20:45:59 +02:00
continue
}
2015-11-19 16:02:49 +01:00
// update block hash since it is now available and not when the receipt/log of individual transactions were created
for _ , r := range work . receipts {
for _ , l := range r . Logs {
l . BlockHash = block . Hash ( )
}
}
for _ , log := range work . state . Logs ( ) {
log . BlockHash = block . Hash ( )
}
2015-07-11 20:45:59 +02:00
// check if canon block and write transactions
if stat == core . CanonStatTy {
// This puts transactions in a extra db for rpc
2015-10-22 15:43:21 +03:00
core . WriteTransactions ( self . chainDb , block )
2015-07-11 20:45:59 +02:00
// store the receipts
2015-10-22 15:43:21 +03:00
core . WriteReceipts ( self . chainDb , work . receipts )
2015-10-12 17:58:51 +02:00
// Write map map bloom filters
core . WriteMipmapBloom ( self . chainDb , block . NumberU64 ( ) , work . receipts )
2016-11-28 14:59:06 +01:00
// implicit by posting ChainHeadEvent
mustCommitNewWork = false
2015-07-11 20:45:59 +02:00
}
// broadcast before waiting for validation
2017-01-05 14:03:50 +01:00
go func ( block * types . Block , logs [ ] * types . Log , receipts [ ] * types . Receipt ) {
2016-04-15 11:06:57 +02:00
self . mux . Post ( core . NewMinedBlockEvent { Block : block } )
self . mux . Post ( core . ChainEvent { Block : block , Hash : block . Hash ( ) , Logs : logs } )
2015-10-15 16:07:19 +02:00
2015-07-11 20:45:59 +02:00
if stat == core . CanonStatTy {
2016-04-15 11:06:57 +02:00
self . mux . Post ( core . ChainHeadEvent { Block : block } )
2015-07-11 20:45:59 +02:00
self . mux . Post ( logs )
}
2016-04-05 15:22:04 +02:00
if err := core . WriteBlockReceipts ( self . chainDb , block . Hash ( ) , block . NumberU64 ( ) , receipts ) ; err != nil {
2017-02-22 14:10:07 +02:00
log . Warn ( fmt . Sprint ( "error writing block receipts:" , err ) )
2015-08-18 21:16:33 +02:00
}
} ( block , work . state . Logs ( ) , work . receipts )
2015-07-03 11:24:42 +02:00
}
2016-12-13 13:23:12 +02:00
// Insert the block into the set of pending ones to wait for confirmations
2016-12-13 15:10:52 +02:00
self . unconfirmed . Insert ( block . NumberU64 ( ) , block . Hash ( ) )
2015-06-29 12:12:30 +02:00
2016-11-28 14:59:06 +01:00
if mustCommitNewWork {
self . commitNewWork ( )
}
2015-02-09 16:20:34 +01:00
}
}
}
2016-04-04 13:18:34 +03:00
// push sends a new work task to currently live miner agents.
2015-08-04 20:38:57 +02:00
func ( self * worker ) push ( work * Work ) {
2016-04-04 13:18:34 +03:00
if atomic . LoadInt32 ( & self . mining ) != 1 {
return
}
for agent := range self . agents {
atomic . AddInt32 ( & self . atWork , 1 )
if ch := agent . Work ( ) ; ch != nil {
ch <- work
2015-02-13 17:23:09 +01:00
}
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.
2015-10-06 16:35:55 +02:00
func ( self * worker ) makeCurrent ( parent * types . Block , header * types . Header ) error {
2016-09-27 13:13:13 +03:00
state , err := self . chain . StateAt ( parent . Root ( ) )
2015-10-06 16:35:55 +02:00
if err != nil {
return err
}
2015-08-04 20:38:57 +02:00
work := & Work {
2016-03-01 23:32:43 +01:00
config : self . config ,
2016-11-02 13:44:13 +01:00
signer : types . NewEIP155Signer ( self . config . ChainId ) ,
2015-06-16 12:41:50 +02:00
state : state ,
ancestors : set . New ( ) ,
family : set . New ( ) ,
uncles : set . New ( ) ,
header : header ,
2015-07-11 20:45:59 +02:00
createdAt : time . Now ( ) ,
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)
2015-06-16 12:41:50 +02:00
for _ , ancestor := range self . chain . GetBlocksFromHash ( parent . Hash ( ) , 7 ) {
2015-05-13 20:46:23 -04:00
for _ , uncle := range ancestor . Uncles ( ) {
2015-08-04 20:38:57 +02:00
work . family . Add ( uncle . Hash ( ) )
2015-05-13 20:46:23 -04:00
}
2015-08-04 20:38:57 +02:00
work . family . Add ( ancestor . Hash ( ) )
work . ancestors . Add ( ancestor . Hash ( ) )
2015-03-23 12:12:49 +01:00
}
2017-02-07 12:47:34 +02:00
wallets := self . eth . AccountManager ( ) . Wallets ( )
accounts := make ( [ ] accounts . Account , 0 , len ( wallets ) )
for _ , wallet := range wallets {
accounts = append ( accounts , wallet . Accounts ( ) ... )
}
2015-05-11 01:28:15 +02:00
// Keep track of transactions which return errors so they can be removed
2015-08-04 20:38:57 +02:00
work . tcount = 0
work . ownedAccounts = accountAddressesSet ( accounts )
self . current = work
2015-10-06 16:35:55 +02:00
return nil
2015-04-07 12:32:55 +02:00
}
2015-05-09 12:04:00 +02:00
func ( w * worker ) setGasPrice ( p * big . Int ) {
w . mu . Lock ( )
defer w . mu . Unlock ( )
2015-05-10 23:12:18 +02:00
// calculate the minimal gas price the miner accepts when sorting out transactions.
const pct = int64 ( 90 )
w . gasPrice = gasprice ( p , pct )
2016-04-15 11:06:57 +02:00
w . mux . Post ( core . GasPriceChanged { Price : w . gasPrice } )
2015-05-09 12:04:00 +02:00
}
2015-04-07 12:32:55 +02:00
func ( self * worker ) commitNewWork ( ) {
self . mu . Lock ( )
defer self . mu . Unlock ( )
self . uncleMu . Lock ( )
defer self . uncleMu . Unlock ( )
self . currentMu . Lock ( )
defer self . currentMu . Unlock ( )
2015-06-15 11:33:08 +02:00
tstart := time . Now ( )
2015-06-16 12:41:50 +02:00
parent := self . 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
if now := time . Now ( ) . Unix ( ) ; tstamp > now + 4 {
wait := time . Duration ( tstamp - now ) * time . Second
2017-02-22 14:10:07 +02:00
log . Info ( fmt . Sprint ( "We are too far in the future. Waiting for" , 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 ) ,
2016-03-01 23:32:43 +01:00
Difficulty : core . CalcDifficulty ( self . config , uint64 ( tstamp ) , parent . Time ( ) . Uint64 ( ) , parent . Number ( ) , parent . Difficulty ( ) ) ,
2015-06-16 12:41:50 +02:00
GasLimit : core . CalcGasLimit ( parent ) ,
GasUsed : new ( big . Int ) ,
Coinbase : self . coinbase ,
Extra : self . extra ,
2015-08-24 02:52:53 +02:00
Time : big . NewInt ( tstamp ) ,
2015-06-16 12:41:50 +02:00
}
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
2016-07-08 13:00:37 +03:00
if daoBlock := self . config . DAOForkBlock ; daoBlock != nil {
// Check whether the block is among the fork extra-override range
limit := new ( big . Int ) . Add ( daoBlock , params . DAOForkExtraRange )
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
if self . config . DAOForkSupport {
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.
err := self . makeCurrent ( parent , header )
if err != nil {
2017-02-22 14:10:07 +02:00
log . Info ( fmt . Sprint ( "Could not create new env for mining, retrying on next block." ) )
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
2015-08-04 20:38:57 +02:00
work := self . current
2016-07-11 13:55:11 +03:00
if self . config . DAOForkSupport && self . config . DAOForkBlock != nil && self . config . DAOForkBlock . Cmp ( header . Number ) == 0 {
core . ApplyDAOHardFork ( work . state )
}
2016-12-10 23:54:58 +01:00
pending , err := self . eth . TxPool ( ) . Pending ( )
if err != nil {
2017-02-22 14:10:07 +02:00
log . Error ( fmt . Sprintf ( "Could not fetch pending transactions: %v" , err ) )
2016-12-10 23:54:58 +01:00
return
}
txs := types . NewTransactionsByPriceAndNonce ( pending )
2016-08-09 14:54:36 +03:00
work . commitTransactions ( self . mux , txs , self . gasPrice , self . chain )
2016-06-27 18:28:34 +03:00
2016-07-01 18:59:55 +03:00
self . eth . TxPool ( ) . RemoveBatch ( work . lowGasTxs )
self . eth . TxPool ( ) . RemoveBatch ( work . failedTxs )
2015-02-04 05:52:59 -08: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
)
2015-03-23 12:12:49 +01:00
for hash , uncle := range self . possibleUncles {
2015-03-23 16:14:33 +01:00
if len ( uncles ) == 2 {
2015-03-23 12:12:49 +01:00
break
}
2015-08-04 20:38:57 +02:00
if err := self . commitUncle ( work , uncle . Header ( ) ) ; err != nil {
2017-02-22 14:10:07 +02:00
log . Trace ( fmt . Sprintf ( "Bad uncle found and will be removed (%x)\n" , hash [ : 4 ] ) )
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-02-22 14:10:07 +02:00
log . Debug ( fmt . Sprintf ( "committing %x as uncle\n" , hash [ : 4 ] ) )
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 {
delete ( self . possibleUncles , hash )
}
2015-03-23 16:35:44 +01:00
2015-06-29 15:17:05 +02:00
if atomic . LoadInt32 ( & self . mining ) == 1 {
// commit state root after all state transitions.
2015-08-04 20:38:57 +02:00
core . AccumulateRewards ( work . state , header , uncles )
2016-10-20 13:36:29 +02:00
header . Root = work . state . IntermediateRoot ( self . config . IsEIP158 ( header . Number ) )
2015-06-29 15:17:05 +02:00
}
2015-03-23 12:12:49 +01:00
2015-06-16 12:41:50 +02:00
// create the new block whose nonce will be mined.
2015-08-04 20:38:57 +02:00
work . Block = types . NewBlock ( header , work . txs , uncles , work . receipts )
2015-02-04 05:52:59 -08:00
2015-06-16 12:41:50 +02:00
// We only care about logging if we're actually mining.
if atomic . LoadInt32 ( & self . mining ) == 1 {
2017-02-22 14:10:07 +02:00
log . Info ( fmt . Sprintf ( "commit new work on block %v with %d txs & %d uncles. Took %v\n" , work . Block . Number ( ) , work . tcount , len ( uncles ) , time . Since ( tstart ) ) )
2016-12-13 15:10:52 +02:00
self . unconfirmed . Shift ( work . Block . NumberU64 ( ) - 1 )
2015-06-16 12:41:50 +02:00
}
2015-08-04 20:38:57 +02:00
self . push ( work )
2015-02-04 05:52:59 -08:00
}
2015-08-04 20:38:57 +02:00
func ( self * worker ) commitUncle ( work * Work , uncle * types . Header ) error {
2015-06-16 12:41:50 +02:00
hash := uncle . Hash ( )
2015-08-04 20:38:57 +02:00
if work . uncles . Has ( hash ) {
2015-02-04 05:52:59 -08:00
return core . UncleError ( "Uncle not unique" )
}
2015-08-04 20:38:57 +02:00
if ! work . ancestors . Has ( uncle . ParentHash ) {
2015-02-04 05:52:59 -08:00
return core . UncleError ( fmt . Sprintf ( "Uncle's parent unknown (%x)" , uncle . ParentHash [ 0 : 4 ] ) )
}
2015-08-04 20:38:57 +02:00
if work . family . Has ( hash ) {
2015-06-16 12:41:50 +02:00
return core . UncleError ( fmt . Sprintf ( "Uncle already in family (%x)" , hash ) )
2015-02-04 05:52:59 -08:00
}
2015-08-04 20:38:57 +02:00
work . uncles . Add ( uncle . Hash ( ) )
2015-02-04 05:52:59 -08:00
return nil
}
2016-08-09 14:54:36 +03:00
func ( env * Work ) commitTransactions ( mux * event . TypeMux , txs * types . TransactionsByPriceAndNonce , gasPrice * big . Int , bc * core . BlockChain ) {
2015-09-02 12:55:11 +02:00
gp := new ( core . GasPool ) . AddGas ( env . header . GasLimit )
2016-02-13 01:40:44 +01:00
2017-01-05 14:03:50 +01:00
var coalescedLogs [ ] * types . Log
2016-09-08 10:44:18 +02:00
2016-08-09 14:54:36 +03:00
for {
// Retrieve the next transaction and abort if all done
tx := txs . Peek ( )
if tx == nil {
break
}
2016-01-19 23:50:00 +01:00
// Error may be ignored here. The error has already been checked
// during transaction acceptance is the transaction pool.
2016-11-02 13:44:13 +01:00
//
// We use the eip155 signer regardless of the current hf.
from , _ := types . Sender ( env . signer , tx )
// Check whether the tx is replay protected. If we're not in the EIP155 hf
// phase, start ignoring the sender until we do.
if tx . Protected ( ) && ! env . config . IsEIP155 ( env . header . Number ) {
2017-02-22 14:10:07 +02:00
log . Trace ( fmt . Sprintf ( "Transaction (%x) is replay protected, but we haven't yet hardforked. Transaction will be ignored until we hardfork.\n" , tx . Hash ( ) ) )
2016-11-02 13:44:13 +01:00
txs . Pop ( )
continue
}
2015-05-11 01:28:15 +02:00
2016-08-09 14:54:36 +03:00
// Ignore any transactions (and accounts subsequently) with low gas limits
2015-06-16 12:41:50 +02:00
if tx . GasPrice ( ) . Cmp ( gasPrice ) < 0 && ! env . ownedAccounts . Has ( from ) {
2016-08-09 14:54:36 +03:00
// Pop the current low-priced transaction without shifting in the next from the account
common: move big integer math to common/math (#3699)
* common: remove CurrencyToString
Move denomination values to params instead.
* common: delete dead code
* common: move big integer operations to common/math
This commit consolidates all big integer operations into common/math and
adds tests and documentation.
There should be no change in semantics for BigPow, BigMin, BigMax, S256,
U256, Exp and their behaviour is now locked in by tests.
The BigD, BytesToBig and Bytes2Big functions don't provide additional
value, all uses are replaced by new(big.Int).SetBytes().
BigToBytes is now called PaddedBigBytes, its minimum output size
parameter is now specified as the number of bytes instead of bits. The
single use of this function is in the EVM's MSTORE instruction.
Big and String2Big are replaced by ParseBig, which is slightly stricter.
It previously accepted leading zeros for hexadecimal inputs but treated
decimal inputs as octal if a leading zero digit was present.
ParseUint64 is used in places where String2Big was used to decode a
uint64.
The new functions MustParseBig and MustParseUint64 are now used in many
places where parsing errors were previously ignored.
* common: delete unused big integer variables
* accounts/abi: replace uses of BytesToBig with use of encoding/binary
* common: remove BytesToBig
* common: remove Bytes2Big
* common: remove BigTrue
* cmd/utils: add BigFlag and use it for error-checked integer flags
While here, remove environment variable processing for DirectoryFlag
because we don't use it.
* core: add missing error checks in genesis block parser
* common: remove String2Big
* cmd/evm: use utils.BigFlag
* common/math: check for 256 bit overflow in ParseBig
This is supposed to prevent silent overflow/truncation of values in the
genesis block JSON. Without this check, a genesis block that set a
balance larger than 256 bits would lead to weird behaviour in the VM.
* cmd/utils: fixup import
2017-02-26 22:21:51 +01:00
log . Info ( fmt . Sprintf ( "Transaction (%x) below gas price (tx=%dwei ask=%dwei). All sequential txs from this address(%x) will be ignored\n" , tx . Hash ( ) . Bytes ( ) [ : 4 ] , tx . GasPrice ( ) , gasPrice , from [ : 4 ] ) )
2015-05-11 01:28:15 +02:00
2016-08-09 14:54:36 +03:00
env . lowGasTxs = append ( env . lowGasTxs , tx )
txs . Pop ( )
2015-05-11 01:28:15 +02:00
continue
}
2016-08-09 14:54:36 +03:00
// Start executing the transaction
2016-09-08 10:44:18 +02:00
env . state . StartRecord ( tx . Hash ( ) , common . Hash { } , env . tcount )
2015-05-11 01:28:15 +02:00
2016-02-13 01:40:44 +01:00
err , logs := env . commitTransaction ( tx , bc , gp )
2015-05-11 01:28:15 +02:00
switch {
2015-09-02 12:55:11 +02:00
case core . IsGasLimitErr ( err ) :
2016-08-09 14:54:36 +03:00
// Pop the current out-of-gas transaction without shifting in the next from the account
2017-02-22 14:10:07 +02:00
log . Trace ( fmt . Sprintf ( "Gas limit reached for (%x) in this block. Continue to try smaller txs\n" , from [ : 4 ] ) )
2016-08-09 14:54:36 +03:00
txs . Pop ( )
2015-07-06 11:54:11 +02:00
2016-06-27 18:28:34 +03:00
case err != nil :
2016-08-09 14:54:36 +03:00
// Pop the current failed transaction without shifting in the next from the account
2017-02-22 14:10:07 +02:00
log . Trace ( fmt . Sprintf ( "Transaction (%x) failed, will be removed: %v\n" , tx . Hash ( ) . Bytes ( ) [ : 4 ] , err ) )
2016-06-27 18:28:34 +03:00
env . failedTxs = append ( env . failedTxs , tx )
2016-08-09 14:54:36 +03:00
txs . Pop ( )
2015-05-11 01:28:15 +02:00
default :
2016-08-09 14:54:36 +03:00
// Everything ok, collect the logs and shift in the next transaction from the same account
2016-02-13 01:40:44 +01:00
coalescedLogs = append ( coalescedLogs , logs ... )
2016-08-09 14:54:36 +03:00
env . tcount ++
txs . Shift ( )
2015-05-11 01:28:15 +02:00
}
}
2016-11-28 14:59:06 +01:00
2016-02-26 18:48:39 +02:00
if len ( coalescedLogs ) > 0 || env . tcount > 0 {
2016-11-28 14:59:06 +01:00
// make a copy, the state caches the logs and these logs get "upgraded" from pending to mined
// logs by filling in the block hash when the block was mined by the local miner. This can
// cause a race condition if a log was "upgraded" before the PendingLogsEvent is processed.
2017-01-05 14:03:50 +01:00
cpy := make ( [ ] * types . Log , len ( coalescedLogs ) )
2016-11-28 14:59:06 +01:00
for i , l := range coalescedLogs {
2017-01-05 14:03:50 +01:00
cpy [ i ] = new ( types . Log )
2016-11-28 14:59:06 +01:00
* cpy [ i ] = * l
}
2017-01-05 14:03:50 +01:00
go func ( logs [ ] * types . Log , tcount int ) {
2016-02-26 18:48:39 +02:00
if len ( logs ) > 0 {
mux . Post ( core . PendingLogsEvent { Logs : logs } )
}
if tcount > 0 {
mux . Post ( core . PendingStateEvent { } )
}
2016-11-28 14:59:06 +01:00
} ( cpy , env . tcount )
2016-02-13 01:40:44 +01:00
}
2015-05-11 01:28:15 +02:00
}
2017-01-05 14:03:50 +01:00
func ( env * Work ) commitTransaction ( tx * types . Transaction , bc * core . BlockChain , gp * core . GasPool ) ( error , [ ] * types . Log ) {
2016-10-04 12:36:02 +02:00
snap := env . state . Snapshot ( )
2016-03-24 13:06:10 +01:00
2017-01-05 11:52:10 +01:00
receipt , _ , err := core . ApplyTransaction ( env . config , bc , gp , env . state , env . header , tx , env . header . GasUsed , vm . Config { } )
2015-07-06 11:54:11 +02:00
if err != nil {
2016-10-04 12:36:02 +02:00
env . state . RevertToSnapshot ( snap )
2016-02-13 01:40:44 +01:00
return err , nil
2015-02-04 05:52:59 -08:00
}
2015-06-16 12:41:50 +02:00
env . txs = append ( env . txs , tx )
env . receipts = append ( env . receipts , receipt )
2016-02-13 01:40:44 +01:00
2017-01-05 11:52:10 +01:00
return nil , receipt . Logs
2015-02-04 05:52:59 -08:00
}
2015-02-28 23:09:49 +01:00
2015-05-14 03:39:07 +02:00
// TODO: remove or use
2015-02-28 23:09:49 +01:00
func ( self * worker ) HashRate ( ) int64 {
2015-05-14 03:39:07 +02:00
return 0
2015-02-28 23:09:49 +01:00
}
2015-05-09 12:13:46 +02:00
// gasprice calculates a reduced gas price based on the pct
// XXX Use big.Rat?
func gasprice ( price * big . Int , pct int64 ) * big . Int {
p := new ( big . Int ) . Set ( price )
p . Div ( p , big . NewInt ( 100 ) )
p . Mul ( p , big . NewInt ( pct ) )
return p
}
2015-05-10 23:12:18 +02:00
func accountAddressesSet ( accounts [ ] accounts . Account ) * set . Set {
accountSet := set . New ( )
for _ , account := range accounts {
2015-05-10 20:30:02 +02:00
accountSet . Add ( account . Address )
2015-05-10 23:12:18 +02:00
}
return accountSet
}