Merge branch 'miner-broadcast' into core-optimisations-2
Conflicts: core/chain_manager.go miner/worker.go
This commit is contained in:
commit
a8ebf756c7
28
core/canary.go
Normal file
28
core/canary.go
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
package core
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math/big"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/common"
|
||||||
|
"github.com/ethereum/go-ethereum/core/state"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
jeff = common.HexToAddress("9d38997c624a71b21278389ea2fdc460d000e4b2")
|
||||||
|
vitalik = common.HexToAddress("b1e570be07eaa673e4fd0c8265b64ef739385709")
|
||||||
|
christoph = common.HexToAddress("529bc43a5d93789fa28de1961db6a07e752204ae")
|
||||||
|
gav = common.HexToAddress("e3e942b2aa524293c84ff6c7f87a6635790ad5e4")
|
||||||
|
)
|
||||||
|
|
||||||
|
// Canary will check the 0'd address of the 4 contracts above.
|
||||||
|
// If two or more are set to anything other than a 0 the canary
|
||||||
|
// dies a horrible death.
|
||||||
|
func Canary(statedb *state.StateDB) bool {
|
||||||
|
r := new(big.Int)
|
||||||
|
r.Add(r, statedb.GetState(jeff, common.Hash{}).Big())
|
||||||
|
r.Add(r, statedb.GetState(vitalik, common.Hash{}).Big())
|
||||||
|
r.Add(r, statedb.GetState(christoph, common.Hash{}).Big())
|
||||||
|
r.Add(r, statedb.GetState(gav, common.Hash{}).Big())
|
||||||
|
|
||||||
|
return r.Cmp(big.NewInt(1)) > 0
|
||||||
|
}
|
@ -541,6 +541,58 @@ func (self *ChainManager) flushQueuedBlocks() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type writeStatus byte
|
||||||
|
|
||||||
|
const (
|
||||||
|
nonStatTy writeStatus = iota
|
||||||
|
canonStatTy
|
||||||
|
splitStatTy
|
||||||
|
sideStatTy
|
||||||
|
)
|
||||||
|
|
||||||
|
func (self *ChainManager) WriteBlock(block *types.Block) (status writeStatus, err error) {
|
||||||
|
self.wg.Add(1)
|
||||||
|
defer self.wg.Done()
|
||||||
|
|
||||||
|
cblock := self.currentBlock
|
||||||
|
// Compare the TD of the last known block in the canonical chain to make sure it's greater.
|
||||||
|
// At this point it's possible that a different chain (fork) becomes the new canonical chain.
|
||||||
|
if block.Td.Cmp(self.Td()) > 0 {
|
||||||
|
// chain fork
|
||||||
|
if block.ParentHash() != cblock.Hash() {
|
||||||
|
// during split we merge two different chains and create the new canonical chain
|
||||||
|
err := self.merge(cblock, block)
|
||||||
|
if err != nil {
|
||||||
|
return nonStatTy, err
|
||||||
|
}
|
||||||
|
|
||||||
|
status = splitStatTy
|
||||||
|
}
|
||||||
|
|
||||||
|
self.mu.Lock()
|
||||||
|
self.setTotalDifficulty(block.Td)
|
||||||
|
self.insert(block)
|
||||||
|
self.mu.Unlock()
|
||||||
|
|
||||||
|
self.setTransState(state.New(block.Root(), self.stateDb))
|
||||||
|
self.txState.SetState(state.New(block.Root(), self.stateDb))
|
||||||
|
|
||||||
|
status = canonStatTy
|
||||||
|
} else {
|
||||||
|
status = sideStatTy
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write block to database. Eventually we'll have to improve on this and throw away blocks that are
|
||||||
|
// not in the canonical chain.
|
||||||
|
self.mu.Lock()
|
||||||
|
self.enqueueForWrite(block)
|
||||||
|
self.mu.Unlock()
|
||||||
|
// Delete from future blocks
|
||||||
|
self.futureBlocks.Delete(block.Hash())
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// InsertChain will attempt to insert the given chain in to the canonical chain or, otherwise, create a fork. It an error is returned
|
// InsertChain will attempt to insert the given chain in to the canonical chain or, otherwise, create a fork. It an error is returned
|
||||||
// it will return the index number of the failing block as well an error describing what went wrong (for possible errors see core/errors.go).
|
// it will return the index number of the failing block as well an error describing what went wrong (for possible errors see core/errors.go).
|
||||||
func (self *ChainManager) InsertChain(chain types.Blocks) (int, error) {
|
func (self *ChainManager) InsertChain(chain types.Blocks) (int, error) {
|
||||||
@ -635,57 +687,29 @@ func (self *ChainManager) InsertChain(chain types.Blocks) (int, error) {
|
|||||||
|
|
||||||
txcount += len(block.Transactions())
|
txcount += len(block.Transactions())
|
||||||
|
|
||||||
cblock := self.currentBlock
|
// write the block to the chain and get the status
|
||||||
// Compare the TD of the last known block in the canonical chain to make sure it's greater.
|
status, err := self.WriteBlock(block)
|
||||||
// At this point it's possible that a different chain (fork) becomes the new canonical chain.
|
if err != nil {
|
||||||
if block.Td.Cmp(self.Td()) > 0 {
|
return i, err
|
||||||
// chain fork
|
}
|
||||||
if block.ParentHash() != cblock.Hash() {
|
switch status {
|
||||||
// during split we merge two different chains and create the new canonical chain
|
case canonStatTy:
|
||||||
err := self.merge(cblock, block)
|
|
||||||
if err != nil {
|
|
||||||
return i, err
|
|
||||||
}
|
|
||||||
|
|
||||||
queue[i] = ChainSplitEvent{block, logs}
|
|
||||||
queueEvent.splitCount++
|
|
||||||
}
|
|
||||||
|
|
||||||
self.mu.Lock()
|
|
||||||
self.setTotalDifficulty(block.Td)
|
|
||||||
self.insert(block)
|
|
||||||
self.mu.Unlock()
|
|
||||||
|
|
||||||
jsonlogger.LogJson(&logger.EthChainNewHead{
|
|
||||||
BlockHash: block.Hash().Hex(),
|
|
||||||
BlockNumber: block.Number(),
|
|
||||||
ChainHeadHash: cblock.Hash().Hex(),
|
|
||||||
BlockPrevHash: block.ParentHash().Hex(),
|
|
||||||
})
|
|
||||||
|
|
||||||
self.setTransState(state.New(block.Root(), self.stateDb))
|
|
||||||
self.txState.SetState(state.New(block.Root(), self.stateDb))
|
|
||||||
|
|
||||||
queue[i] = ChainEvent{block, block.Hash(), logs}
|
|
||||||
queueEvent.canonicalCount++
|
|
||||||
|
|
||||||
if glog.V(logger.Debug) {
|
if glog.V(logger.Debug) {
|
||||||
glog.Infof("[%v] inserted block #%d (%d TXs %d UNCs) (%x...). Took %v\n", time.Now().UnixNano(), block.Number(), len(block.Transactions()), len(block.Uncles()), block.Hash().Bytes()[0:4], time.Since(bstart))
|
glog.Infof("[%v] inserted block #%d (%d TXs %d UNCs) (%x...). Took %v\n", time.Now().UnixNano(), block.Number(), len(block.Transactions()), len(block.Uncles()), block.Hash().Bytes()[0:4], time.Since(bstart))
|
||||||
}
|
}
|
||||||
} else {
|
queue[i] = ChainEvent{block, block.Hash(), logs}
|
||||||
|
queueEvent.canonicalCount++
|
||||||
|
case sideStatTy:
|
||||||
if glog.V(logger.Detail) {
|
if glog.V(logger.Detail) {
|
||||||
glog.Infof("inserted forked block #%d (TD=%v) (%d TXs %d UNCs) (%x...). Took %v\n", block.Number(), block.Difficulty(), len(block.Transactions()), len(block.Uncles()), block.Hash().Bytes()[0:4], time.Since(bstart))
|
glog.Infof("inserted forked block #%d (TD=%v) (%d TXs %d UNCs) (%x...). Took %v\n", block.Number(), block.Difficulty(), len(block.Transactions()), len(block.Uncles()), block.Hash().Bytes()[0:4], time.Since(bstart))
|
||||||
}
|
}
|
||||||
|
|
||||||
queue[i] = ChainSideEvent{block, logs}
|
queue[i] = ChainSideEvent{block, logs}
|
||||||
queueEvent.sideCount++
|
queueEvent.sideCount++
|
||||||
|
case splitStatTy:
|
||||||
|
queue[i] = ChainSplitEvent{block, logs}
|
||||||
|
queueEvent.splitCount++
|
||||||
}
|
}
|
||||||
self.enqueueForWrite(block)
|
|
||||||
// Delete from future blocks
|
|
||||||
self.futureBlocks.Delete(block.Hash())
|
|
||||||
|
|
||||||
stats.processed++
|
stats.processed++
|
||||||
blockInsertTimer.UpdateSince(bstart)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stats.queued > 0 || stats.processed > 0 || stats.ignored > 0) && bool(glog.V(logger.Info)) {
|
if (stats.queued > 0 || stats.processed > 0 || stats.ignored > 0) && bool(glog.V(logger.Info)) {
|
||||||
@ -744,9 +768,9 @@ func (self *ChainManager) diff(oldBlock, newBlock *types.Block) (types.Blocks, e
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if glog.V(logger.Info) {
|
if glog.V(logger.Debug) {
|
||||||
commonHash := commonBlock.Hash()
|
commonHash := commonBlock.Hash()
|
||||||
glog.Infof("Fork detected @ %x. Reorganising chain from #%v %x to %x", commonHash[:4], numSplit, oldStart.Hash().Bytes()[:4], newStart.Hash().Bytes()[:4])
|
glog.Infof("Chain split detected @ %x. Reorganising chain from #%v %x to %x", commonHash[:4], numSplit, oldStart.Hash().Bytes()[:4], newStart.Hash().Bytes()[:4])
|
||||||
}
|
}
|
||||||
|
|
||||||
return newChain, nil
|
return newChain, nil
|
||||||
|
@ -233,38 +233,40 @@ func (self *worker) wait() {
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, err := self.chain.InsertChain(types.Blocks{block}); err == nil {
|
_, err := self.chain.WriteBlock(block)
|
||||||
for _, uncle := range block.Uncles() {
|
if err != nil {
|
||||||
delete(self.possibleUncles, uncle.Hash())
|
glog.V(logger.Error).Infoln("error writing block to chain", err)
|
||||||
}
|
continue
|
||||||
self.mux.Post(core.NewMinedBlockEvent{block})
|
|
||||||
|
|
||||||
var stale, confirm string
|
|
||||||
canonBlock := self.chain.GetBlockByNumber(block.NumberU64())
|
|
||||||
if canonBlock != nil && canonBlock.Hash() != block.Hash() {
|
|
||||||
stale = "stale "
|
|
||||||
} else {
|
|
||||||
confirm = "Wait 5 blocks for confirmation"
|
|
||||||
self.current.localMinedBlocks = newLocalMinedBlock(block.Number().Uint64(), self.current.localMinedBlocks)
|
|
||||||
}
|
|
||||||
|
|
||||||
glog.V(logger.Info).Infof("🔨 Mined %sblock (#%v / %x). %s", stale, block.Number(), block.Hash().Bytes()[:4], confirm)
|
|
||||||
|
|
||||||
jsonlogger.LogJson(&logger.EthMinerNewBlock{
|
|
||||||
BlockHash: block.Hash().Hex(),
|
|
||||||
BlockNumber: block.Number(),
|
|
||||||
ChainHeadHash: block.ParentHash().Hex(),
|
|
||||||
BlockPrevHash: block.ParentHash().Hex(),
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
self.commitNewWork()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// check staleness and display confirmation
|
||||||
|
var stale, confirm string
|
||||||
|
canonBlock := self.chain.GetBlockByNumber(block.NumberU64())
|
||||||
|
if canonBlock != nil && canonBlock.Hash() != block.Hash() {
|
||||||
|
stale = "stale "
|
||||||
|
} else {
|
||||||
|
confirm = "Wait 5 blocks for confirmation"
|
||||||
|
self.current.localMinedBlocks = newLocalMinedBlock(block.Number().Uint64(), self.current.localMinedBlocks)
|
||||||
|
}
|
||||||
|
|
||||||
|
glog.V(logger.Info).Infof("🔨 Mined %sblock (#%v / %x). %s", stale, block.Number(), block.Hash().Bytes()[:4], confirm)
|
||||||
|
|
||||||
|
// broadcast before waiting for validation
|
||||||
|
go self.mux.Post(core.NewMinedBlockEvent{block})
|
||||||
|
|
||||||
|
self.commitNewWork()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *worker) push() {
|
func (self *worker) push() {
|
||||||
if atomic.LoadInt32(&self.mining) == 1 {
|
if atomic.LoadInt32(&self.mining) == 1 {
|
||||||
|
if core.Canary(self.current.state) {
|
||||||
|
glog.Infoln("Toxicity levels rising to deadly levels. Your canary has died. You can go back or continue down the mineshaft --more--")
|
||||||
|
glog.Infoln("You turn back and abort mining")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// push new work to agents
|
// push new work to agents
|
||||||
for _, agent := range self.agents {
|
for _, agent := range self.agents {
|
||||||
atomic.AddInt32(&self.atWork, 1)
|
atomic.AddInt32(&self.atWork, 1)
|
||||||
@ -369,6 +371,13 @@ func (self *worker) commitNewWork() {
|
|||||||
if tstamp <= parent.Time() {
|
if tstamp <= parent.Time() {
|
||||||
tstamp = parent.Time() + 1
|
tstamp = parent.Time() + 1
|
||||||
}
|
}
|
||||||
|
// 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
|
||||||
|
glog.V(logger.Info).Infoln("We are too far in the future. Waiting for", wait)
|
||||||
|
time.Sleep(wait)
|
||||||
|
}
|
||||||
|
|
||||||
num := parent.Number()
|
num := parent.Number()
|
||||||
header := &types.Header{
|
header := &types.Header{
|
||||||
ParentHash: parent.Hash(),
|
ParentHash: parent.Hash(),
|
||||||
@ -420,11 +429,13 @@ func (self *worker) commitNewWork() {
|
|||||||
// commit state root after all state transitions.
|
// commit state root after all state transitions.
|
||||||
core.AccumulateRewards(self.current.state, header, uncles)
|
core.AccumulateRewards(self.current.state, header, uncles)
|
||||||
current.state.Update()
|
current.state.Update()
|
||||||
|
self.current.state.Sync()
|
||||||
header.Root = current.state.Root()
|
header.Root = current.state.Root()
|
||||||
}
|
}
|
||||||
|
|
||||||
// create the new block whose nonce will be mined.
|
// create the new block whose nonce will be mined.
|
||||||
current.block = types.NewBlock(header, current.txs, uncles, current.receipts)
|
current.block = types.NewBlock(header, current.txs, uncles, current.receipts)
|
||||||
|
self.current.block.Td = new(big.Int).Set(core.CalcTD(self.current.block, self.chain.GetBlock(self.current.block.ParentHash())))
|
||||||
|
|
||||||
// We only care about logging if we're actually mining.
|
// We only care about logging if we're actually mining.
|
||||||
if atomic.LoadInt32(&self.mining) == 1 {
|
if atomic.LoadInt32(&self.mining) == 1 {
|
||||||
|
Loading…
Reference in New Issue
Block a user