2015-02-25 12:34:12 +00:00
package blockpool
import (
"fmt"
"math/big"
"sync"
"time"
2015-03-16 22:10:26 +00:00
"github.com/ethereum/go-ethereum/common"
2015-03-19 22:53:15 +00:00
"github.com/ethereum/go-ethereum/core"
2015-02-25 12:34:12 +00:00
"github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/errs"
2015-03-19 22:53:15 +00:00
"github.com/ethereum/go-ethereum/event"
2015-02-25 12:34:12 +00:00
ethlogger "github.com/ethereum/go-ethereum/logger"
"github.com/ethereum/go-ethereum/pow"
)
var plog = ethlogger . NewLogger ( "Blockpool" )
var (
// max number of block hashes sent in one request
2015-03-03 19:06:15 +00:00
blockHashesBatchSize = 256
2015-02-25 12:34:12 +00:00
// max number of blocks sent in one request
blockBatchSize = 64
// interval between two consecutive block checks (and requests)
blocksRequestInterval = 3 * time . Second
// level of redundancy in block requests sent
blocksRequestRepetition = 1
// interval between two consecutive block hash checks (and requests)
blockHashesRequestInterval = 3 * time . Second
// max number of idle iterations, ie., check through a section without new blocks coming in
2015-03-03 19:06:15 +00:00
blocksRequestMaxIdleRounds = 20
2015-02-25 12:34:12 +00:00
// timeout interval: max time allowed for peer without sending a block hash
blockHashesTimeout = 60 * time . Second
// timeout interval: max time allowed for peer without sending a block
2015-03-03 19:06:15 +00:00
blocksTimeout = 60 * time . Second
2015-03-19 22:53:15 +00:00
// timeout interval: max time allowed for best peer to remain idle (not send new block after sync complete)
idleBestPeerTimeout = 120 * time . Second
// duration of suspension after peer fatal error during which peer is not allowed to reconnect
2015-03-19 22:46:54 +00:00
peerSuspensionInterval = 300 * time . Second
2015-03-19 23:14:08 +00:00
// status is logged every statusUpdateInterval
statusUpdateInterval = 3 * time . Second
2015-02-25 12:34:12 +00:00
)
2015-03-19 23:14:08 +00:00
// blockpool config, values default to constants
2015-02-25 12:34:12 +00:00
type Config struct {
BlockHashesBatchSize int
BlockBatchSize int
BlocksRequestRepetition int
BlocksRequestMaxIdleRounds int
BlockHashesRequestInterval time . Duration
BlocksRequestInterval time . Duration
BlockHashesTimeout time . Duration
BlocksTimeout time . Duration
2015-03-03 19:06:15 +00:00
IdleBestPeerTimeout time . Duration
2015-03-19 22:46:54 +00:00
PeerSuspensionInterval time . Duration
2015-03-19 23:14:08 +00:00
StatusUpdateInterval time . Duration
2015-02-25 12:34:12 +00:00
}
// blockpool errors
const (
ErrInvalidBlock = iota
ErrInvalidPoW
ErrInsufficientChainInfo
2015-03-03 19:06:15 +00:00
ErrIdleTooLong
2015-03-19 23:00:19 +00:00
ErrIncorrectTD
2015-03-19 23:14:08 +00:00
ErrUnrequestedBlock
2015-02-25 12:34:12 +00:00
)
2015-03-19 23:14:08 +00:00
// error descriptions
2015-02-25 12:34:12 +00:00
var errorToString = map [ int ] string {
2015-03-19 23:14:08 +00:00
ErrInvalidBlock : "Invalid block" , // fatal
ErrInvalidPoW : "Invalid PoW" , // fatal
ErrInsufficientChainInfo : "Insufficient chain info" , // fatal
ErrIdleTooLong : "Idle too long" , // fatal
ErrIncorrectTD : "Incorrect Total Difficulty" , // fatal
2015-02-25 12:34:12 +00:00
ErrUnrequestedBlock : "Unrequested block" ,
}
2015-03-19 23:14:08 +00:00
// error severity
func severity ( code int ) ethlogger . LogLevel {
switch code {
case ErrUnrequestedBlock :
return ethlogger . WarnLevel
default :
return ethlogger . ErrorLevel
}
}
// init initialises the Config, zero values fall back to constants
2015-02-25 12:34:12 +00:00
func ( self * Config ) init ( ) {
if self . BlockHashesBatchSize == 0 {
self . BlockHashesBatchSize = blockHashesBatchSize
}
if self . BlockBatchSize == 0 {
self . BlockBatchSize = blockBatchSize
}
if self . BlocksRequestRepetition == 0 {
self . BlocksRequestRepetition = blocksRequestRepetition
}
if self . BlocksRequestMaxIdleRounds == 0 {
self . BlocksRequestMaxIdleRounds = blocksRequestMaxIdleRounds
}
if self . BlockHashesRequestInterval == 0 {
self . BlockHashesRequestInterval = blockHashesRequestInterval
}
if self . BlocksRequestInterval == 0 {
self . BlocksRequestInterval = blocksRequestInterval
}
if self . BlockHashesTimeout == 0 {
self . BlockHashesTimeout = blockHashesTimeout
}
if self . BlocksTimeout == 0 {
self . BlocksTimeout = blocksTimeout
}
2015-03-03 19:06:15 +00:00
if self . IdleBestPeerTimeout == 0 {
self . IdleBestPeerTimeout = idleBestPeerTimeout
}
2015-03-19 22:46:54 +00:00
if self . PeerSuspensionInterval == 0 {
self . PeerSuspensionInterval = peerSuspensionInterval
}
2015-03-19 23:14:08 +00:00
if self . StatusUpdateInterval == 0 {
self . StatusUpdateInterval = statusUpdateInterval
}
2015-02-25 12:34:12 +00:00
}
// node is the basic unit of the internal model of block chain/tree in the blockpool
type node struct {
lock sync . RWMutex
2015-03-16 22:10:26 +00:00
hash common . Hash
2015-02-25 12:34:12 +00:00
block * types . Block
hashBy string
blockBy string
td * big . Int
}
type index struct {
int
}
// entry is the struct kept and indexed in the pool
type entry struct {
node * node
section * section
index * index
}
type BlockPool struct {
Config * Config
2015-03-19 23:14:08 +00:00
// the minimal interface with blockchain manager
hasBlock func ( hash common . Hash ) bool // query if block is known
insertChain func ( types . Blocks ) error // add section to blockchain
verifyPoW func ( pow . Block ) bool // soft PoW verification
chainEvents * event . TypeMux // ethereum eventer for chainEvents
2015-03-19 22:53:15 +00:00
2015-03-19 23:14:08 +00:00
tdSub event . Subscription // subscription to core.ChainHeadEvent
td * big . Int // our own total difficulty
2015-02-25 12:34:12 +00:00
2015-03-19 23:14:08 +00:00
pool map [ string ] * entry // the actual blockpool
peers * peers // peers manager in peers.go
status * status // info about blockpool (UI interface) in status.go
2015-02-25 12:34:12 +00:00
lock sync . RWMutex
chainLock sync . RWMutex
// alloc-easy pool of hash slices
2015-03-16 22:10:26 +00:00
hashSlicePool chan [ ] common . Hash
2015-02-25 12:34:12 +00:00
2015-03-19 23:14:08 +00:00
// waitgroup is used in tests to wait for result-critical routines
// as well as in determining idle / syncing status
wg sync . WaitGroup //
quit chan bool // chan used for quitting parallel routines
running bool //
2015-02-25 12:34:12 +00:00
}
// public constructor
2015-03-19 23:14:08 +00:00
// after blockpool returned, config can be set
// BlockPool.Start will call Config.init to set missing values
2015-02-25 12:34:12 +00:00
func New (
2015-03-16 22:10:26 +00:00
hasBlock func ( hash common . Hash ) bool ,
2015-02-25 12:34:12 +00:00
insertChain func ( types . Blocks ) error ,
verifyPoW func ( pow . Block ) bool ,
2015-03-19 22:53:15 +00:00
chainEvents * event . TypeMux ,
td * big . Int ,
2015-02-25 12:34:12 +00:00
) * BlockPool {
return & BlockPool {
Config : & Config { } ,
hasBlock : hasBlock ,
insertChain : insertChain ,
verifyPoW : verifyPoW ,
2015-03-19 22:53:15 +00:00
chainEvents : chainEvents ,
td : td ,
2015-02-25 12:34:12 +00:00
}
}
// allows restart
func ( self * BlockPool ) Start ( ) {
self . lock . Lock ( )
defer self . lock . Unlock ( )
if self . running {
return
}
2015-03-19 23:14:08 +00:00
// set missing values
2015-02-25 12:34:12 +00:00
self . Config . init ( )
2015-03-19 23:14:08 +00:00
2015-03-16 22:10:26 +00:00
self . hashSlicePool = make ( chan [ ] common . Hash , 150 )
2015-02-25 12:34:12 +00:00
self . status = newStatus ( )
self . quit = make ( chan bool )
self . pool = make ( map [ string ] * entry )
self . running = true
self . peers = & peers {
errors : & errs . Errors {
Package : "Blockpool" ,
Errors : errorToString ,
2015-03-02 17:43:12 +00:00
Level : severity ,
2015-02-25 12:34:12 +00:00
} ,
2015-03-19 22:46:54 +00:00
peers : make ( map [ string ] * peer ) ,
blacklist : make ( map [ string ] time . Time ) ,
status : self . status ,
bp : self ,
2015-02-25 12:34:12 +00:00
}
2015-03-19 22:53:15 +00:00
2015-03-19 23:14:08 +00:00
// subscribe and listen to core.ChainHeadEvent{} for uptodate TD
2015-03-19 22:53:15 +00:00
self . tdSub = self . chainEvents . Subscribe ( core . ChainHeadEvent { } )
2015-03-19 23:14:08 +00:00
// status update interval
timer := time . NewTicker ( self . Config . StatusUpdateInterval )
2015-02-25 12:34:12 +00:00
go func ( ) {
for {
select {
case <- self . quit :
return
2015-03-19 22:53:15 +00:00
case event := <- self . tdSub . Chan ( ) :
if ev , ok := event . ( core . ChainHeadEvent ) ; ok {
td := ev . Block . Td
plog . DebugDetailf ( "td: %v" , td )
self . setTD ( td )
self . peers . lock . Lock ( )
if best := self . peers . best ; best != nil {
if td . Cmp ( best . td ) >= 0 {
self . peers . best = nil
self . switchPeer ( best , nil )
}
}
self . peers . lock . Unlock ( )
}
2015-02-25 12:34:12 +00:00
case <- timer . C :
2015-03-15 20:47:35 +00:00
plog . DebugDetailf ( "status:\n%v" , self . Status ( ) )
2015-02-25 12:34:12 +00:00
}
}
} ( )
plog . Infoln ( "Started" )
}
func ( self * BlockPool ) Stop ( ) {
self . lock . Lock ( )
if ! self . running {
self . lock . Unlock ( )
return
}
self . running = false
self . lock . Unlock ( )
plog . Infoln ( "Stopping..." )
2015-03-19 22:53:15 +00:00
self . tdSub . Unsubscribe ( )
2015-02-25 12:34:12 +00:00
close ( self . quit )
self . lock . Lock ( )
self . peers = nil
self . pool = nil
self . lock . Unlock ( )
plog . Infoln ( "Stopped" )
}
// Wait blocks until active processes finish
func ( self * BlockPool ) Wait ( t time . Duration ) {
self . lock . Lock ( )
if ! self . running {
self . lock . Unlock ( )
return
}
self . lock . Unlock ( )
plog . Infoln ( "Waiting for processes to complete..." )
w := make ( chan bool )
go func ( ) {
self . wg . Wait ( )
close ( w )
} ( )
select {
case <- w :
plog . Infoln ( "Processes complete" )
case <- time . After ( t ) :
plog . Warnf ( "Timeout" )
}
}
/ *
AddPeer is called by the eth protocol instance running on the peer after
the status message has been received with total difficulty and current block hash
2015-03-19 23:14:08 +00:00
Called a second time with the same peer id , it is used to update chain info for a peer .
This is used when a new ( mined ) block message is received .
2015-02-25 12:34:12 +00:00
RemovePeer needs to be called when the peer disconnects .
2015-03-19 23:14:08 +00:00
Peer info is currently not persisted across disconnects ( or sessions ) except for suspension
2015-02-25 12:34:12 +00:00
* /
func ( self * BlockPool ) AddPeer (
2015-03-16 22:10:26 +00:00
td * big . Int , currentBlockHash common . Hash ,
2015-02-25 12:34:12 +00:00
peerId string ,
2015-03-16 22:10:26 +00:00
requestBlockHashes func ( common . Hash ) error ,
requestBlocks func ( [ ] common . Hash ) error ,
2015-02-25 12:34:12 +00:00
peerError func ( * errs . Error ) ,
2015-03-19 22:46:54 +00:00
) ( best bool , suspended bool ) {
2015-02-25 12:34:12 +00:00
return self . peers . addPeer ( td , currentBlockHash , peerId , requestBlockHashes , requestBlocks , peerError )
}
// RemovePeer needs to be called when the peer disconnects
func ( self * BlockPool ) RemovePeer ( peerId string ) {
self . peers . removePeer ( peerId )
}
/ *
AddBlockHashes
Entry point for eth protocol to add block hashes received via BlockHashesMsg
2015-03-19 23:14:08 +00:00
Only hashes from the best peer are handled
2015-02-25 12:34:12 +00:00
2015-03-19 23:14:08 +00:00
Initiates further hash requests until a known parent is reached ( unless cancelled by a peerSwitch event , i . e . , when a better peer becomes best peer )
Launches all block request processes on each chain section
2015-02-25 12:34:12 +00:00
2015-03-19 23:14:08 +00:00
The first argument is an iterator function . Using this block hashes are decoded from the rlp message payload on demand . As a result , AddBlockHashes needs to run synchronously for one peer since the message is discarded if the caller thread returns .
2015-02-25 12:34:12 +00:00
* /
2015-03-16 22:10:26 +00:00
func ( self * BlockPool ) AddBlockHashes ( next func ( ) ( common . Hash , bool ) , peerId string ) {
2015-02-25 12:34:12 +00:00
bestpeer , best := self . peers . getPeer ( peerId )
if ! best {
return
}
// bestpeer is still the best peer
self . wg . Add ( 1 )
defer func ( ) { self . wg . Done ( ) } ( )
self . status . lock . Lock ( )
self . status . activePeers [ bestpeer . id ] ++
self . status . lock . Unlock ( )
var n int
2015-03-16 22:10:26 +00:00
var hash common . Hash
2015-02-25 12:34:12 +00:00
var ok , headSection , peerswitch bool
var sec , child , parent * section
var entry * entry
var nodes [ ] * node
hash , ok = next ( )
bestpeer . lock . Lock ( )
plog . Debugf ( "AddBlockHashes: peer <%s> starting from [%s] (peer head: %s)" , peerId , hex ( bestpeer . parentHash ) , hex ( bestpeer . currentBlockHash ) )
// first check if we are building the head section of a peer's chain
2015-03-16 22:10:26 +00:00
if bestpeer . parentHash == hash {
2015-02-25 12:34:12 +00:00
if self . hasBlock ( bestpeer . currentBlockHash ) {
return
}
/ *
2015-03-19 23:14:08 +00:00
When peer is promoted in switchPeer , a new header section process is launched .
Once the head section skeleton is actually created here , it is signaled to the process
so that it can quit .
In the special case that the node for parent of the head block is found in the blockpool
( with or without fetched block ) , a singleton section containing only the head block node is created .
2015-02-25 12:34:12 +00:00
* /
headSection = true
if entry := self . get ( bestpeer . currentBlockHash ) ; entry == nil {
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s) head section starting from [%s] " , peerId , hex ( bestpeer . currentBlockHash ) , hex ( bestpeer . parentHash ) )
// if head block is not yet in the pool, create entry and start node list for section
node := & node {
hash : bestpeer . currentBlockHash ,
block : bestpeer . currentBlock ,
hashBy : peerId ,
blockBy : peerId ,
2015-03-19 23:00:19 +00:00
td : bestpeer . td ,
2015-02-25 12:34:12 +00:00
}
// nodes is a list of nodes in one section ordered top-bottom (old to young)
nodes = append ( nodes , node )
n ++
} else {
// otherwise set child section iff found node is the root of a section
// this is a possible scenario when a singleton head section was created
2015-03-19 23:14:08 +00:00
// on an earlier occasion when this peer or another with the same block was best peer
2015-02-25 12:34:12 +00:00
if entry . node == entry . section . bottom {
child = entry . section
plog . DebugDetailf ( "AddBlockHashes: peer <%s>: connects to child section root %s" , peerId , hex ( bestpeer . currentBlockHash ) )
}
}
} else {
// otherwise : we are not building the head section of the peer
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s) section starting from [%s] " , peerId , hex ( bestpeer . currentBlockHash ) , hex ( hash ) )
}
// the switch channel signals peerswitch event
switchC := bestpeer . switchC
bestpeer . lock . Unlock ( )
// iterate over hashes coming from peer (first round we have hash set above)
LOOP :
for ; ok ; hash , ok = next ( ) {
select {
case <- self . quit :
// global quit for blockpool
return
case <- switchC :
// if the peer is demoted, no more hashes read
plog . DebugDetailf ( "AddBlockHashes: demoted peer <%s> (head: %s)" , peerId , hex ( bestpeer . currentBlockHash ) , hex ( hash ) )
peerswitch = true
break LOOP
default :
}
2015-03-19 23:14:08 +00:00
// if we reach the blockchain we stop reading further blockhashes
2015-02-25 12:34:12 +00:00
if self . hasBlock ( hash ) {
// check if known block connecting the downloaded chain to our blockchain
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s) found block %s in the blockchain" , peerId , hex ( bestpeer . currentBlockHash ) , hex ( hash ) )
if len ( nodes ) == 1 {
2015-03-02 17:43:12 +00:00
plog . DebugDetailf ( "AddBlockHashes: singleton section pushed to blockchain peer <%s> (head: %s) found block %s in the blockchain" , peerId , hex ( bestpeer . currentBlockHash ) , hex ( hash ) )
2015-02-25 12:34:12 +00:00
// create new section if needed and push it to the blockchain
sec = self . newSection ( nodes )
sec . addSectionToBlockChain ( bestpeer )
} else {
/ *
not added hash yet but according to peer child section built
earlier chain connects with blockchain
this maybe a potential vulnarability
the root block arrives ( or already there but its parenthash was not pointing to known block in the blockchain )
we start inserting - > error - > remove the entire chain
instead of punishing this peer
solution : when switching peers always make sure best peers own head block
and td together with blockBy are recorded on the node
* /
if len ( nodes ) == 0 && child != nil {
2015-03-02 17:43:12 +00:00
plog . DebugDetailf ( "AddBlockHashes: child section [%s] pushed to blockchain peer <%s> (head: %s) found block %s in the blockchain" , sectionhex ( child ) , peerId , hex ( bestpeer . currentBlockHash ) , hex ( hash ) )
2015-02-25 12:34:12 +00:00
child . addSectionToBlockChain ( bestpeer )
}
}
break LOOP
}
// look up node in the pool
entry = self . get ( hash )
if entry != nil {
// reached a known chain in the pool
if entry . node == entry . section . bottom && n == 1 {
/ *
2015-03-19 23:14:08 +00:00
The first block hash received is an orphan node in the pool
This also supports clients that ( despite the spec ) include < from > hash in their
2015-02-25 12:34:12 +00:00
response to hashes request . Note that by providing < from > we can link sections
2015-03-19 23:14:08 +00:00
without having to wait for the root block of the child section to arrive , so it allows for superior performance .
2015-02-25 12:34:12 +00:00
* /
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s) found head block [%s] as root of connecting child section [%s] skipping" , peerId , hex ( bestpeer . currentBlockHash ) , hex ( hash ) , sectionhex ( entry . section ) )
// record the entry's chain section as child section
child = entry . section
continue LOOP
}
// otherwise record entry's chain section as parent connecting it to the pool
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s) found block [%s] in section [%s]. Connected to pool." , peerId , hex ( bestpeer . currentBlockHash ) , hex ( hash ) , sectionhex ( entry . section ) )
parent = entry . section
break LOOP
}
// finally if node for block hash does not exist, create it and append node to section nodes
node := & node {
hash : hash ,
hashBy : peerId ,
}
nodes = append ( nodes , node )
} //for
/ *
we got here if
- run out of hashes ( parent = nil ) sent by our best peer
- our peer is demoted ( peerswitch = true )
- reached blockchain or blockpool
- quitting
* /
self . chainLock . Lock ( )
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s): %v nodes in new section" , peerId , hex ( bestpeer . currentBlockHash ) , len ( nodes ) )
/ *
2015-03-19 23:14:08 +00:00
Handle forks where connecting node is mid - section by splitting section at fork .
No splitting needed if connecting node is head of a section .
2015-02-25 12:34:12 +00:00
* /
if parent != nil && entry != nil && entry . node != parent . top && len ( nodes ) > 0 {
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s): fork after %s" , peerId , hex ( bestpeer . currentBlockHash ) , hex ( hash ) )
self . splitSection ( parent , entry )
self . status . lock . Lock ( )
self . status . values . Forks ++
self . status . lock . Unlock ( )
}
2015-03-19 23:14:08 +00:00
// If new section is created, link it to parent/child sections.
2015-02-25 12:34:12 +00:00
sec = self . linkSections ( nodes , parent , child )
2015-03-02 17:43:12 +00:00
if sec != nil {
self . status . lock . Lock ( )
self . status . values . BlockHashes += len ( nodes )
self . status . lock . Unlock ( )
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s): section [%s] created" , peerId , hex ( bestpeer . currentBlockHash ) , sectionhex ( sec ) )
}
2015-02-25 12:34:12 +00:00
self . chainLock . Unlock ( )
/ *
2015-03-19 23:14:08 +00:00
If a blockpool node is reached ( parent section is not nil ) ,
2015-02-25 12:34:12 +00:00
activate section ( unless our peer is demoted by now ) .
2015-03-19 23:14:08 +00:00
This can be the bottom half of a newly split section in case of a fork .
2015-02-25 12:34:12 +00:00
bestPeer is nil if we got here after our peer got demoted while processing .
2015-03-19 23:14:08 +00:00
In this case no activation should happen
2015-02-25 12:34:12 +00:00
* /
if parent != nil && ! peerswitch {
self . activateChain ( parent , bestpeer , nil )
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s): parent section [%s]" , peerId , hex ( bestpeer . currentBlockHash ) , sectionhex ( parent ) )
}
/ *
2015-03-19 23:14:08 +00:00
If a new section was created , register section iff head section or no child known
Activate it with this peer .
2015-02-25 12:34:12 +00:00
* /
if sec != nil {
// switch on section process (it is paused by switchC)
if ! peerswitch {
if headSection || child == nil {
bestpeer . lock . Lock ( )
bestpeer . sections = append ( bestpeer . sections , sec . top . hash )
bestpeer . lock . Unlock ( )
}
/ *
2015-03-19 23:14:08 +00:00
Request another batch of older block hashes for parent section here .
But only once , repeating only when the section ' s root block arrives .
Otherwise no way to check if it arrived .
2015-02-25 12:34:12 +00:00
* /
bestpeer . requestBlockHashes ( sec . bottom . hash )
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s): start requesting blocks for section [%s]" , peerId , hex ( bestpeer . currentBlockHash ) , sectionhex ( sec ) )
sec . activate ( bestpeer )
} else {
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s) no longer best: delay requesting blocks for section [%s]" , peerId , hex ( bestpeer . currentBlockHash ) , sectionhex ( sec ) )
sec . deactivate ( )
}
}
2015-03-19 23:14:08 +00:00
// If we are processing peer's head section, signal it to headSection process that it is created.
2015-02-25 12:34:12 +00:00
if headSection {
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s) head section registered on head section process" , peerId , hex ( bestpeer . currentBlockHash ) )
var headSec * section
switch {
case sec != nil :
headSec = sec
case child != nil :
headSec = child
default :
headSec = parent
}
if ! peerswitch {
plog . DebugDetailf ( "AddBlockHashes: peer <%s> (head: %s) head section [%s] created signalled to head section process" , peerId , hex ( bestpeer . currentBlockHash ) , sectionhex ( headSec ) )
bestpeer . headSectionC <- headSec
}
}
}
/ *
AddBlock is the entry point for the eth protocol to call when blockMsg is received .
2015-03-19 23:14:08 +00:00
It has a strict interpretation of the protocol in that if the block received has not been requested , it results in an error .
2015-02-25 12:34:12 +00:00
At the same time it is opportunistic in that if a requested block may be provided by any peer .
The received block is checked for PoW . Only the first PoW - valid block for a hash is considered legit .
2015-03-19 23:14:08 +00:00
If the block received is the head block of the current best peer , signal it to the head section process
2015-02-25 12:34:12 +00:00
* /
func ( self * BlockPool ) AddBlock ( block * types . Block , peerId string ) {
hash := block . Hash ( )
sender , _ := self . peers . getPeer ( peerId )
if sender == nil {
return
}
self . status . lock . Lock ( )
self . status . activePeers [ peerId ] ++
self . status . lock . Unlock ( )
entry := self . get ( hash )
// a peer's current head block is appearing the first time
2015-03-16 22:10:26 +00:00
if hash == sender . currentBlockHash {
2015-02-25 12:34:12 +00:00
if sender . currentBlock == nil {
plog . Debugf ( "AddBlock: add head block %s for peer <%s> (head: %s)" , hex ( hash ) , peerId , hex ( sender . currentBlockHash ) )
sender . setChainInfoFromBlock ( block )
self . status . lock . Lock ( )
2015-03-05 18:33:57 +00:00
self . status . values . BlockHashes ++
2015-02-25 12:34:12 +00:00
self . status . values . Blocks ++
self . status . values . BlocksInPool ++
self . status . lock . Unlock ( )
} else {
plog . DebugDetailf ( "AddBlock: head block %s for peer <%s> (head: %s) already known" , hex ( hash ) , peerId , hex ( sender . currentBlockHash ) )
2015-03-19 23:14:08 +00:00
// signal to head section process
2015-03-02 17:43:12 +00:00
sender . currentBlockC <- block
2015-02-25 12:34:12 +00:00
}
} else {
plog . DebugDetailf ( "AddBlock: block %s received from peer <%s> (head: %s)" , hex ( hash ) , peerId , hex ( sender . currentBlockHash ) )
sender . lock . Lock ( )
// update peer chain info if more recent than what we registered
if block . Td != nil && block . Td . Cmp ( sender . td ) > 0 {
sender . td = block . Td
sender . currentBlockHash = block . Hash ( )
sender . parentHash = block . ParentHash ( )
sender . currentBlock = block
sender . headSection = nil
}
sender . lock . Unlock ( )
if entry == nil {
plog . DebugDetailf ( "AddBlock: unrequested block %s received from peer <%s> (head: %s)" , hex ( hash ) , peerId , hex ( sender . currentBlockHash ) )
sender . addError ( ErrUnrequestedBlock , "%x" , hash )
self . status . lock . Lock ( )
self . status . badPeers [ peerId ] ++
self . status . lock . Unlock ( )
return
}
}
if entry == nil {
return
}
node := entry . node
node . lock . Lock ( )
defer node . lock . Unlock ( )
2015-03-19 23:14:08 +00:00
// check if block already received
2015-02-25 12:34:12 +00:00
if node . block != nil {
plog . DebugDetailf ( "AddBlock: block %s from peer <%s> (head: %s) already sent by <%s> " , hex ( hash ) , peerId , hex ( sender . currentBlockHash ) , node . blockBy )
return
}
// check if block is already inserted in the blockchain
if self . hasBlock ( hash ) {
plog . DebugDetailf ( "AddBlock: block %s from peer <%s> (head: %s) already in the blockchain" , hex ( hash ) , peerId , hex ( sender . currentBlockHash ) )
return
}
// validate block for PoW
if ! self . verifyPoW ( block ) {
plog . Warnf ( "AddBlock: invalid PoW on block %s from peer <%s> (head: %s)" , hex ( hash ) , peerId , hex ( sender . currentBlockHash ) )
sender . addError ( ErrInvalidPoW , "%x" , hash )
self . status . lock . Lock ( )
self . status . badPeers [ peerId ] ++
self . status . lock . Unlock ( )
return
}
node . block = block
node . blockBy = peerId
node . td = block . Td // optional field
self . status . lock . Lock ( )
self . status . values . Blocks ++
self . status . values . BlocksInPool ++
self . status . lock . Unlock ( )
}
/ *
2015-03-19 23:14:08 +00:00
activateChain iterates down a chain section by section .
It activates the section process on incomplete sections with peer .
It relinks orphaned sections with their parent if root block ( and its parent hash ) is known .
2015-02-25 12:34:12 +00:00
* /
func ( self * BlockPool ) activateChain ( sec * section , p * peer , connected map [ string ] * section ) {
p . lock . RLock ( )
switchC := p . switchC
p . lock . RUnlock ( )
var i int
LOOP :
for sec != nil {
parent := self . getParent ( sec )
plog . DebugDetailf ( "activateChain: section [%s] activated by peer <%s>" , sectionhex ( sec ) , p . id )
sec . activate ( p )
if i > 0 && connected != nil {
2015-03-16 22:10:26 +00:00
connected [ sec . top . hash . Str ( ) ] = sec
2015-02-25 12:34:12 +00:00
}
/ *
2015-03-19 23:14:08 +00:00
Need to relink both complete and incomplete sections
An incomplete section could have been blockHashesRequestsComplete before being delinked from its parent .
2015-02-25 12:34:12 +00:00
* /
2015-03-02 17:43:12 +00:00
if parent == nil {
if sec . bottom . block != nil {
if entry := self . get ( sec . bottom . block . ParentHash ( ) ) ; entry != nil {
parent = entry . section
plog . DebugDetailf ( "activateChain: [%s]-[%s] link" , sectionhex ( parent ) , sectionhex ( sec ) )
link ( parent , sec )
}
} else {
plog . DebugDetailf ( "activateChain: section [%s] activated by peer <%s> has missing root block" , sectionhex ( sec ) , p . id )
2015-02-25 12:34:12 +00:00
}
}
sec = parent
2015-03-19 23:14:08 +00:00
// stop if peer got demoted or global quit
2015-02-25 12:34:12 +00:00
select {
case <- switchC :
break LOOP
case <- self . quit :
break LOOP
default :
}
}
}
2015-03-19 23:14:08 +00:00
// check if block's actual TD (calculated after successful insertChain) is identical to TD advertised for peer's head block.
2015-03-19 23:00:19 +00:00
func ( self * BlockPool ) checkTD ( nodes ... * node ) {
for _ , n := range nodes {
if n . td != nil {
plog . DebugDetailf ( "peer td %v =?= block td %v" , n . td , n . block . Td )
if n . td . Cmp ( n . block . Td ) != 0 {
self . peers . peerError ( n . blockBy , ErrIncorrectTD , "on block %x" , n . hash )
}
}
}
}
2015-03-19 23:14:08 +00:00
// requestBlocks must run in separate go routine, otherwise
2015-02-25 12:34:12 +00:00
// switchpeer -> activateChain -> activate deadlocks on section process select and peers.lock
2015-03-16 22:10:26 +00:00
func ( self * BlockPool ) requestBlocks ( attempts int , hashes [ ] common . Hash ) {
2015-02-25 12:34:12 +00:00
self . wg . Add ( 1 )
go func ( ) {
self . peers . requestBlocks ( attempts , hashes )
self . wg . Done ( )
} ( )
}
// convenience methods to access adjacent sections
func ( self * BlockPool ) getParent ( sec * section ) * section {
self . chainLock . RLock ( )
defer self . chainLock . RUnlock ( )
return sec . parent
}
func ( self * BlockPool ) getChild ( sec * section ) * section {
self . chainLock . RLock ( )
defer self . chainLock . RUnlock ( )
return sec . child
}
// accessor and setter for entries in the pool
2015-03-16 22:10:26 +00:00
func ( self * BlockPool ) get ( hash common . Hash ) * entry {
2015-02-25 12:34:12 +00:00
self . lock . RLock ( )
defer self . lock . RUnlock ( )
2015-03-16 22:10:26 +00:00
return self . pool [ hash . Str ( ) ]
2015-02-25 12:34:12 +00:00
}
2015-03-16 22:10:26 +00:00
func ( self * BlockPool ) set ( hash common . Hash , e * entry ) {
2015-02-25 12:34:12 +00:00
self . lock . Lock ( )
defer self . lock . Unlock ( )
2015-03-16 22:10:26 +00:00
self . pool [ hash . Str ( ) ] = e
2015-02-25 12:34:12 +00:00
}
2015-03-19 22:53:15 +00:00
// accessor and setter for total difficulty
func ( self * BlockPool ) getTD ( ) * big . Int {
self . lock . RLock ( )
defer self . lock . RUnlock ( )
return self . td
}
func ( self * BlockPool ) setTD ( td * big . Int ) {
self . lock . Lock ( )
defer self . lock . Unlock ( )
self . td = td
}
2015-02-25 12:34:12 +00:00
func ( self * BlockPool ) remove ( sec * section ) {
// delete node entries from pool index under pool lock
self . lock . Lock ( )
defer self . lock . Unlock ( )
2015-03-02 17:43:12 +00:00
2015-02-25 12:34:12 +00:00
for _ , node := range sec . nodes {
2015-03-16 22:10:26 +00:00
delete ( self . pool , node . hash . Str ( ) )
2015-02-25 12:34:12 +00:00
}
2015-03-02 17:43:12 +00:00
if sec . initialised && sec . poolRootIndex != 0 {
self . status . lock . Lock ( )
self . status . values . BlocksInPool -= len ( sec . nodes ) - sec . missing
self . status . lock . Unlock ( )
}
2015-02-25 12:34:12 +00:00
}
2015-03-19 23:14:08 +00:00
// get/put for optimised allocation similar to sync.Pool
2015-03-16 22:10:26 +00:00
func ( self * BlockPool ) getHashSlice ( ) ( s [ ] common . Hash ) {
2015-02-25 12:34:12 +00:00
select {
case s = <- self . hashSlicePool :
default :
2015-03-16 22:10:26 +00:00
s = make ( [ ] common . Hash , self . Config . BlockBatchSize )
2015-02-25 12:34:12 +00:00
}
return
}
2015-03-16 22:10:26 +00:00
func ( self * BlockPool ) putHashSlice ( s [ ] common . Hash ) {
2015-02-25 12:34:12 +00:00
if len ( s ) == self . Config . BlockBatchSize {
select {
case self . hashSlicePool <- s :
default :
}
}
}
// pretty prints hash (byte array) with first 4 bytes in hex
2015-03-16 22:10:26 +00:00
func hex ( hash common . Hash ) ( name string ) {
if ( hash == common . Hash { } ) {
2015-02-25 12:34:12 +00:00
name = ""
} else {
name = fmt . Sprintf ( "%x" , hash [ : 4 ] )
}
return
}
// pretty prints a section using first 4 bytes in hex of bottom and top blockhash of the section
func sectionhex ( section * section ) ( name string ) {
if section == nil {
name = ""
} else {
name = fmt . Sprintf ( "%x-%x" , section . bottom . hash [ : 4 ] , section . top . hash [ : 4 ] )
}
return
}