diff options
author | bojie <bojie@dexon.org> | 2019-01-07 15:40:56 +0800 |
---|---|---|
committer | Wei-Ning Huang <w@dexon.org> | 2019-01-14 15:26:26 +0800 |
commit | c90c2f0f036307824f1e64e8f0fba012fa1cc927 (patch) | |
tree | 06a84996647fae6039f293f0413bfad85ce35653 /core | |
parent | c0f0675284c5d2aace6719f3774f7df422b0c1d1 (diff) | |
download | dexon-c90c2f0f036307824f1e64e8f0fba012fa1cc927.tar dexon-c90c2f0f036307824f1e64e8f0fba012fa1cc927.tar.gz dexon-c90c2f0f036307824f1e64e8f0fba012fa1cc927.tar.bz2 dexon-c90c2f0f036307824f1e64e8f0fba012fa1cc927.tar.lz dexon-c90c2f0f036307824f1e64e8f0fba012fa1cc927.tar.xz dexon-c90c2f0f036307824f1e64e8f0fba012fa1cc927.tar.zst dexon-c90c2f0f036307824f1e64e8f0fba012fa1cc927.zip |
app: implement logic for prepare/verify correctly when chain number change (#118)
Diffstat (limited to 'core')
-rw-r--r-- | core/blockchain.go | 116 | ||||
-rw-r--r-- | core/blockchain_test.go | 6 | ||||
-rw-r--r-- | core/genesis.go | 2 | ||||
-rw-r--r-- | core/rawdb/accessors_chain.go | 15 | ||||
-rw-r--r-- | core/rawdb/schema.go | 2 |
5 files changed, 122 insertions, 19 deletions
diff --git a/core/blockchain.go b/core/blockchain.go index 0bb6b270b..3c00109ea 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -160,6 +160,7 @@ type BlockChain struct { pendingBlocks map[uint64]struct { block *types.Block receipts types.Receipts + proctime time.Duration } } @@ -199,6 +200,7 @@ func NewBlockChain(db ethdb.Database, cacheConfig *CacheConfig, chainConfig *par pendingBlocks: make(map[uint64]struct { block *types.Block receipts types.Receipts + proctime time.Duration }), confirmedBlocks: make(map[uint32]map[coreCommon.Hash]*blockInfo), addressNonce: make(map[uint32]map[common.Address]uint64), @@ -1073,6 +1075,13 @@ func (bc *BlockChain) InsertReceiptChain(blockChain types.Blocks, receiptChain [ bytes += batch.ValueSize() batch.Reset() } + + if i == len(blockChain)-1 { + err := bc.updateLastRoundNumber(block.Round()) + if err != nil { + return 0, err + } + } } if batch.ValueSize() > 0 { bytes += batch.ValueSize() @@ -1344,7 +1353,7 @@ func (bc *BlockChain) insertChain(chain types.Blocks, verifySeals bool) (int, [] case err == consensus.ErrPrunedAncestor: return bc.insertSidechain(it) - // First block is future, shove it (and all children) to the future queue (unknown ancestor) + // First block is future, shove it (and all children) to the future queue (unknown ancestor) case err == consensus.ErrFutureBlock || (err == consensus.ErrUnknownAncestor && bc.futureBlocks.Contains(it.first().ParentHash())): for block != nil && (it.index == 0 || err == consensus.ErrUnknownAncestor) { if err := bc.addFutureBlock(block); err != nil { @@ -1358,10 +1367,10 @@ func (bc *BlockChain) insertChain(chain types.Blocks, verifySeals bool) (int, [] // If there are any still remaining, mark as ignored return it.index, events, coalescedLogs, err - // First block (and state) is known - // 1. We did a roll-back, and should now do a re-import - // 2. The block is stored as a sidechain, and is lying about it's stateroot, and passes a stateroot - // from the canonical chain, which has not been verified. + // First block (and state) is known + // 1. We did a roll-back, and should now do a re-import + // 2. The block is stored as a sidechain, and is lying about it's stateroot, and passes a stateroot + // from the canonical chain, which has not been verified. case err == ErrKnownBlock: // Skip all known blocks that behind us current := bc.CurrentBlock().NumberU64() @@ -1372,7 +1381,7 @@ func (bc *BlockChain) insertChain(chain types.Blocks, verifySeals bool) (int, [] } // Falls through to the block import - // Some other error occurred, abort + // Some other error occurred, abort case err != nil: stats.ignored += len(it.chain) bc.reportBlock(block, nil, err) @@ -1780,6 +1789,13 @@ func (bc *BlockChain) insertDexonChain(chain types.Blocks) (int, []interface{}, cache, _ := bc.stateCache.TrieDB().Size() stats.report(chain, i, cache) + + if i == len(chain)-1 { + err = bc.updateLastRoundNumber(block.Round()) + if err != nil { + return 0, nil, nil, err + } + } } // Append a single chain head event if we've progressed the chain if lastCanon != nil && bc.CurrentBlock().Hash() == lastCanon.Hash() { @@ -1817,15 +1833,7 @@ func (bc *BlockChain) processPendingBlock( coalescedLogs []*types.Log ) - // Start a parallel signature recovery (signer will fluke on fork transition, minimal perf loss) - senderCacher.recoverFromBlocks(types.MakeSigner(bc.chainConfig, block.Number()), []*types.Block{block}) - - if atomic.LoadInt32(&bc.procInterrupt) == 1 { - log.Debug("Premature abort during blocks processing") - return nil, nil, nil, fmt.Errorf("interrupt") - } bstart := time.Now() - var witnessData types.WitnessData if err := rlp.Decode(bytes.NewReader(witness.Data), &witnessData); err != nil { log.Error("Witness rlp decode failed", "error", err) @@ -1892,13 +1900,18 @@ func (bc *BlockChain) processPendingBlock( } // Add into pending blocks. - bc.addPendingBlock(newPendingBlock, receipts) + bc.addPendingBlock(newPendingBlock, receipts, proctime) events = append(events, BlockConfirmedEvent{newPendingBlock}) log.Debug("Inserted pending block", "height", newPendingBlock.Number(), "hash", newPendingBlock.Hash()) // Start insert available pending blocks into db for pendingHeight := bc.CurrentBlock().NumberU64() + 1; pendingHeight <= witness.Height; pendingHeight++ { + if atomic.LoadInt32(&bc.procInterrupt) == 1 { + log.Debug("Premature abort during blocks processing") + return nil, nil, nil, fmt.Errorf("interrupt") + } + pendingIns, exist := bc.pendingBlocks[pendingHeight] if !exist { log.Error("Block has already inserted", "height", pendingHeight) @@ -1911,6 +1924,7 @@ func (bc *BlockChain) processPendingBlock( } // Write the block to the chain and get the status. + insertTime := time.Now() status, err := bc.WriteBlockWithState(pendingIns.block, pendingIns.receipts, s) if err != nil { return nil, events, coalescedLogs, fmt.Errorf("WriteBlockWithState error: %v", err) @@ -1927,12 +1941,12 @@ func (bc *BlockChain) processPendingBlock( allLogs = append(allLogs, r.Logs...) } coalescedLogs = append(coalescedLogs, allLogs...) - blockInsertTimer.UpdateSince(bstart) + blockInsertTimer.UpdateSince(insertTime) events = append(events, ChainEvent{pendingIns.block, pendingIns.block.Hash(), allLogs}) lastCanon = pendingIns.block // Only count canonical blocks for GC processing time - bc.gcproc += proctime + bc.gcproc += pendingIns.proctime case SideStatTy: return nil, nil, nil, fmt.Errorf("insert pending block and fork found") @@ -1944,6 +1958,13 @@ func (bc *BlockChain) processPendingBlock( cache, _ := bc.stateCache.TrieDB().Size() stats.report([]*types.Block{pendingIns.block}, 0, cache) + + if pendingHeight == witness.Height { + err = bc.updateLastRoundNumber(pendingIns.block.Round()) + if err != nil { + return nil, nil, nil, err + } + } } // Append a single chain head event if we've progressed the chain if lastCanon != nil && bc.CurrentBlock().Hash() == lastCanon.Hash() { @@ -1954,6 +1975,46 @@ func (bc *BlockChain) processPendingBlock( return &root, events, coalescedLogs, nil } +func (bc *BlockChain) ProcessEmptyBlock(block *types.Block) error { + bstart := time.Now() + var header = block.Header() + var parentBlock *types.Block + var pendingState *state.StateDB + var err error + parent, exist := bc.pendingBlocks[block.NumberU64()-1] + if !exist { + parentBlock = bc.GetBlockByNumber(block.NumberU64() - 1) + if parentBlock == nil { + return fmt.Errorf("parent block %d not exist", block.NumberU64()-1) + } + } else { + parentBlock = parent.block + } + + pendingState, err = state.New(parentBlock.Root(), bc.stateCache) + if err != nil { + return err + } + + header.ParentHash = parentBlock.Hash() + header.GasUsed = 0 + header.Root = pendingState.IntermediateRoot(true) + if header.Root != parentBlock.Root() { + return fmt.Errorf("empty block state root must same as parent") + } + + newPendingBlock := types.NewBlock(header, nil, nil, nil) + if _, ok := bc.GetRoundHeight(newPendingBlock.Round()); !ok { + bc.storeRoundHeight(newPendingBlock.Round(), newPendingBlock.NumberU64()) + } + + proctime := time.Since(bstart) + bc.addPendingBlock(newPendingBlock, nil, proctime) + bc.PostChainEvents([]interface{}{BlockConfirmedEvent{newPendingBlock}}, nil) + + return nil +} + func (bc *BlockChain) removePendingBlock(height uint64) { bc.pendingBlockMu.Lock() defer bc.pendingBlockMu.Unlock() @@ -1961,14 +2022,15 @@ func (bc *BlockChain) removePendingBlock(height uint64) { delete(bc.pendingBlocks, height) } -func (bc *BlockChain) addPendingBlock(block *types.Block, receipts types.Receipts) { +func (bc *BlockChain) addPendingBlock(block *types.Block, receipts types.Receipts, proctime time.Duration) { bc.pendingBlockMu.Lock() defer bc.pendingBlockMu.Unlock() bc.pendingBlocks[block.NumberU64()] = struct { block *types.Block receipts types.Receipts - }{block: block, receipts: receipts} + proctime time.Duration + }{block: block, receipts: receipts, proctime: proctime} bc.lastPendingHeight = block.NumberU64() } @@ -2412,3 +2474,19 @@ func (bc *BlockChain) GetRoundHeight(round uint64) (uint64, bool) { func (bc *BlockChain) storeRoundHeight(round uint64, height uint64) { bc.roundHeightMap.Store(round, height) } + +func (bc *BlockChain) updateLastRoundNumber(round uint64) error { + currentLastRound, err := rawdb.ReadLastRoundNumber(bc.db) + if err != nil { + return err + } + + if round > currentLastRound { + err = rawdb.WriteLastRoundNumber(bc.db, round) + if err != nil { + return err + } + } + + return nil +} diff --git a/core/blockchain_test.go b/core/blockchain_test.go index ae4747116..c57c8e2c1 100644 --- a/core/blockchain_test.go +++ b/core/blockchain_test.go @@ -654,6 +654,9 @@ func TestFastVsFullChains(t *testing.T) { // Fast import the chain as a non-archive node to test fastDb := ethdb.NewMemDatabase() gspec.MustCommit(fastDb) + if err := rawdb.WriteLastRoundNumber(fastDb, 0); err != nil { + t.Fatalf("failed to write last round: %v", err) + } fast, _ := NewBlockChain(fastDb, nil, gspec.Config, ethash.NewFaker(), vm.Config{}, nil) defer fast.Stop() @@ -745,6 +748,9 @@ func TestLightVsFastVsFullChainHeads(t *testing.T) { // Import the chain as a non-archive node and ensure all pointers are updated fastDb := ethdb.NewMemDatabase() gspec.MustCommit(fastDb) + if err := rawdb.WriteLastRoundNumber(fastDb, 0); err != nil { + t.Fatalf("failed to write last round: %v", err) + } fast, _ := NewBlockChain(fastDb, nil, gspec.Config, ethash.NewFaker(), vm.Config{}, nil) defer fast.Stop() diff --git a/core/genesis.go b/core/genesis.go index 8ad06bbe2..c1b5466fa 100644 --- a/core/genesis.go +++ b/core/genesis.go @@ -170,6 +170,8 @@ func SetupGenesisBlock(db ethdb.Database, genesis *Genesis) (*params.ChainConfig return SetupGenesisBlockWithOverride(db, genesis, nil) } func SetupGenesisBlockWithOverride(db ethdb.Database, genesis *Genesis, constantinopleOverride *big.Int) (*params.ChainConfig, common.Hash, error) { + rawdb.WriteLastRoundNumber(db, 0) + if genesis != nil && genesis.Config == nil { return params.AllEthashProtocolChanges, common.Hash{}, errGenesisNoConfig } diff --git a/core/rawdb/accessors_chain.go b/core/rawdb/accessors_chain.go index 801ad9362..5e23bef8f 100644 --- a/core/rawdb/accessors_chain.go +++ b/core/rawdb/accessors_chain.go @@ -348,6 +348,21 @@ func WriteBlock(db DatabaseWriter, block *types.Block) { WriteHeader(db, block.Header()) } +// ReadLastRoundNumber get current round number. +func ReadLastRoundNumber(db DatabaseReader) (uint64, error) { + number, err := db.Get(lastRoundNum) + if err != nil { + return 0, err + } + + return new(big.Int).SetBytes(number).Uint64(), nil +} + +// WriteLastRoundNumber write current round number. +func WriteLastRoundNumber(db DatabaseWriter, number uint64) error { + return db.Put(lastRoundNum, new(big.Int).SetUint64(number).Bytes()) +} + // DeleteBlock removes all block data associated with a hash. func DeleteBlock(db DatabaseDeleter, hash common.Hash, number uint64) { DeleteReceipts(db, hash, number) diff --git a/core/rawdb/schema.go b/core/rawdb/schema.go index 87a56c0ba..954ebfc1d 100644 --- a/core/rawdb/schema.go +++ b/core/rawdb/schema.go @@ -63,6 +63,8 @@ var ( // Chain index prefixes (use `i` + single byte to avoid mixing data types). BloomBitsIndexPrefix = []byte("iB") // BloomBitsIndexPrefix is the data table of a chain indexer to track its progress + lastRoundNum = []byte("LastRoundNum") + preimageCounter = metrics.NewRegisteredCounter("db/preimage/total", nil) preimageHitCounter = metrics.NewRegisteredCounter("db/preimage/hits", nil) ) |