From a6ad71bba939f8a3612667ad94935012ecb6fe9c Mon Sep 17 00:00:00 2001 From: buddh0 Date: Tue, 26 Mar 2024 15:52:02 +0800 Subject: [PATCH 01/15] 4844: bugfix and improve --- core/blockchain.go | 2 +- core/blockchain_reader.go | 2 +- core/data_availability.go | 2 +- core/rawdb/accessors_chain.go | 4 ++-- core/rawdb/accessors_chain_test.go | 6 +++--- core/rawdb/chain_freezer.go | 9 ++------ core/rawdb/freezer_table.go | 34 +++++++++++++++++++++++------- core/state/pruner/pruner.go | 2 +- eth/downloader/queue.go | 2 +- internal/ethapi/api_test.go | 2 +- 10 files changed, 39 insertions(+), 26 deletions(-) diff --git a/core/blockchain.go b/core/blockchain.go index 0045932e9d..a0023b460d 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -973,6 +973,7 @@ func (bc *BlockChain) setHeadBeyondRoot(head uint64, time uint64, root common.Ha // The header, total difficulty and canonical hash will be // removed in the hc.SetHead function. rawdb.DeleteBody(db, hash, num) + rawdb.DeleteBlobSidecars(db, hash, num) rawdb.DeleteReceipts(db, hash, num) } // Todo(rjl493456442) txlookup, bloombits, etc @@ -1405,7 +1406,6 @@ func (bc *BlockChain) InsertReceiptChain(blockChain types.Blocks, receiptChain [ // Write all chain data to ancients. td := bc.GetTd(first.Hash(), first.NumberU64()) writeSize, err := rawdb.WriteAncientBlocks(bc.db, blockChain, receiptChain, td) - if err != nil { log.Error("Error importing chain data to ancients", "err", err) return 0, err diff --git a/core/blockchain_reader.go b/core/blockchain_reader.go index 227da1eb69..2023bcf4d6 100644 --- a/core/blockchain_reader.go +++ b/core/blockchain_reader.go @@ -256,7 +256,7 @@ func (bc *BlockChain) GetSidecarsByHash(hash common.Hash) types.BlobSidecars { if number == nil { return nil } - sidecars := rawdb.ReadRawBlobSidecars(bc.db, hash, *number) + sidecars := rawdb.ReadBlobSidecars(bc.db, hash, *number) if sidecars == nil { return nil } diff --git a/core/data_availability.go b/core/data_availability.go index bd807e3541..73502065ff 100644 --- a/core/data_availability.go +++ b/core/data_availability.go @@ -65,7 +65,7 @@ func IsDataAvailable(chain consensus.ChainHeaderReader, block *types.Block) (err highest = current } defer func() { - log.Info("IsDataAvailable", "block", block.Number(), "hash", block.Hash(), "highest", highest.Number, "sidecars", len(block.Sidecars()), "err", err) + log.Debug("IsDataAvailable", "block", block.Number(), "hash", block.Hash(), "highest", highest.Number, "sidecars", len(block.Sidecars()), "err", err) }() if block.NumberU64()+params.MinBlocksForBlobRequests < highest.Number.Uint64() { // if we needn't check DA of this block, just clean it diff --git a/core/rawdb/accessors_chain.go b/core/rawdb/accessors_chain.go index 9cd37899dd..dd56ca1b64 100644 --- a/core/rawdb/accessors_chain.go +++ b/core/rawdb/accessors_chain.go @@ -853,8 +853,8 @@ func ReadBlobSidecarsRLP(db ethdb.Reader, hash common.Hash, number uint64) rlp.R return data } -// ReadRawBlobSidecars retrieves all the transaction blobs belonging to a block. -func ReadRawBlobSidecars(db ethdb.Reader, hash common.Hash, number uint64) types.BlobSidecars { +// ReadBlobSidecars retrieves all the transaction blobs belonging to a block. +func ReadBlobSidecars(db ethdb.Reader, hash common.Hash, number uint64) types.BlobSidecars { data := ReadBlobSidecarsRLP(db, hash, number) if len(data) == 0 { return nil diff --git a/core/rawdb/accessors_chain_test.go b/core/rawdb/accessors_chain_test.go index 77db41a137..1b3e45f3db 100644 --- a/core/rawdb/accessors_chain_test.go +++ b/core/rawdb/accessors_chain_test.go @@ -455,13 +455,13 @@ func TestBlockBlobSidecarsStorage(t *testing.T) { sidecars := types.BlobSidecars{types.NewBlobSidecarFromTx(tx1)} // Check that no sidecars entries are in a pristine database - if bs := ReadRawBlobSidecars(db, blkHash, 0); len(bs) != 0 { + if bs := ReadBlobSidecars(db, blkHash, 0); len(bs) != 0 { t.Fatalf("non existent sidecars returned: %v", bs) } WriteBody(db, blkHash, 0, body) WriteBlobSidecars(db, blkHash, 0, sidecars) - if bs := ReadRawBlobSidecars(db, blkHash, 0); len(bs) == 0 { + if bs := ReadBlobSidecars(db, blkHash, 0); len(bs) == 0 { t.Fatalf("no sidecars returned") } else { if err := checkBlobSidecarsRLP(bs, sidecars); err != nil { @@ -470,7 +470,7 @@ func TestBlockBlobSidecarsStorage(t *testing.T) { } DeleteBlobSidecars(db, blkHash, 0) - if bs := ReadRawBlobSidecars(db, blkHash, 0); len(bs) != 0 { + if bs := ReadBlobSidecars(db, blkHash, 0); len(bs) != 0 { t.Fatalf("deleted sidecars returned: %v", bs) } } diff --git a/core/rawdb/chain_freezer.go b/core/rawdb/chain_freezer.go index 2da59a3960..5d9443ad0f 100644 --- a/core/rawdb/chain_freezer.go +++ b/core/rawdb/chain_freezer.go @@ -312,9 +312,8 @@ func (f *chainFreezer) freezeRangeWithBlobs(nfdb *nofreezedb, number, limit uint var ( cancunNumber uint64 - found bool + preHashes []common.Hash ) - for i := number; i <= limit; i++ { hash := ReadCanonicalHash(nfdb, i) if hash == (common.Hash{}) { @@ -326,16 +325,12 @@ func (f *chainFreezer) freezeRangeWithBlobs(nfdb *nofreezedb, number, limit uint } if isCancun(env, h.Number, h.Time) { cancunNumber = i - found = true break } } - if !found { - return f.freezeRange(nfdb, number, limit) - } // freeze pre cancun - preHashes, err := f.freezeRange(nfdb, number, cancunNumber-1) + preHashes, err = f.freezeRange(nfdb, number, cancunNumber-1) if err != nil { return preHashes, err } diff --git a/core/rawdb/freezer_table.go b/core/rawdb/freezer_table.go index 8dd99d4f49..3fc313e749 100644 --- a/core/rawdb/freezer_table.go +++ b/core/rawdb/freezer_table.go @@ -1035,14 +1035,15 @@ func (t *freezerTable) resetItems(tail, head uint64) (*freezerTable, error) { if t.readonly { return nil, errors.New("resetItems in readonly mode") } + itemHidden := t.itemHidden.Load() items := t.items.Load() - if tail != head && (itemHidden > tail || items < head) { + if tail > head || (tail < head && (itemHidden > tail || items < head)) { return nil, errors.New("cannot reset to non-exist range") } var err error - if tail != head { + if tail < head { if err = t.truncateHead(head); err != nil { return nil, err } @@ -1052,15 +1053,27 @@ func (t *freezerTable) resetItems(tail, head uint64) (*freezerTable, error) { return t, nil } + t.lock.Lock() + defer t.lock.Unlock() + // if tail == head, it means table reset to 0 item - t.releaseFilesAfter(t.tailId-1, true) + // remove all data files t.head.Close() - os.Remove(t.head.Name()) - t.index.Close() - os.Remove(t.index.Name()) + t.releaseFilesAfter(0, true) + t.releaseFile(0) + + // reset meta data file + if err := writeMetadata(t.meta, newMetadata(items)); err != nil { + return nil, err + } + if err := t.meta.Sync(); err != nil { + return nil, err + } t.meta.Close() - os.Remove(t.meta.Name()) + // reset the index file + t.index.Close() + os.Remove(t.index.Name()) var idxName string if t.noCompression { idxName = fmt.Sprintf("%s.ridx", t.name) // raw index file @@ -1072,11 +1085,16 @@ func (t *freezerTable) resetItems(tail, head uint64) (*freezerTable, error) { return nil, err } tailIndex := indexEntry{ - offset: uint32(tail), + filenum: 0, + offset: uint32(tail), } if _, err = index.Write(tailIndex.append(nil)); err != nil { return nil, err } + if err := index.Sync(); err != nil { + return nil, err + } + index.Close() return newFreezerTable(t.path, t.name, t.noCompression, t.readonly) } diff --git a/core/state/pruner/pruner.go b/core/state/pruner/pruner.go index 26b827e3b9..89b50036c4 100644 --- a/core/state/pruner/pruner.go +++ b/core/state/pruner/pruner.go @@ -454,7 +454,7 @@ func (p *BlockPruner) backUpOldDb(name string, cache, handles int, namespace str return consensus.ErrUnknownAncestor } // if there has blobs, it needs to back up too. - blobs := rawdb.ReadRawBlobSidecars(chainDb, blockHash, blockNumber) + blobs := rawdb.ReadBlobSidecars(chainDb, blockHash, blockNumber) block = block.WithSidecars(blobs) // Write into new ancient_back db. if _, err := rawdb.WriteAncientBlocks(frdbBack, []*types.Block{block}, []types.Receipts{receipts}, td); err != nil { diff --git a/eth/downloader/queue.go b/eth/downloader/queue.go index 8a2de4104a..b97cfdec6d 100644 --- a/eth/downloader/queue.go +++ b/eth/downloader/queue.go @@ -827,7 +827,7 @@ func (q *queue) DeliverBodies(id string, txLists [][]*types.Transaction, txListH if want := *header.BlobGasUsed / params.BlobTxBlobGasPerBlob; uint64(blobs) != want { // div because the header is surely good vs the body might be bloated return errInvalidBody } - if blobs > params.MaxBlobGasPerBlock { + if blobs > params.MaxBlobGasPerBlock/params.BlobTxBlobGasPerBlob { return errInvalidBody } } else { diff --git a/internal/ethapi/api_test.go b/internal/ethapi/api_test.go index a496315636..5123f01d95 100644 --- a/internal/ethapi/api_test.go +++ b/internal/ethapi/api_test.go @@ -574,7 +574,7 @@ func (b testBackend) GetBlobSidecars(ctx context.Context, hash common.Hash) (typ if header == nil || err != nil { return nil, err } - blobSidecars := rawdb.ReadRawBlobSidecars(b.db, hash, header.Number.Uint64()) + blobSidecars := rawdb.ReadBlobSidecars(b.db, hash, header.Number.Uint64()) return blobSidecars, nil } func (b testBackend) GetTd(ctx context.Context, hash common.Hash) *big.Int { From 6a2cb4e632c0606a498499f3f36c6fc825e7c98c Mon Sep 17 00:00:00 2001 From: buddh0 Date: Tue, 26 Mar 2024 18:32:35 +0800 Subject: [PATCH 02/15] core: add debug log for CheckDataAvailableInBatch --- core/blockchain.go | 1 + core/data_availability.go | 4 ---- 2 files changed, 1 insertion(+), 4 deletions(-) diff --git a/core/blockchain.go b/core/blockchain.go index a0023b460d..e956af0fa9 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -1341,6 +1341,7 @@ func (bc *BlockChain) InsertReceiptChain(blockChain types.Blocks, receiptChain [ lastBlk := blockChain[len(blockChain)-1] if bc.chainConfig.Parlia != nil && bc.chainConfig.IsCancun(lastBlk.Number(), lastBlk.Time()) { if _, err := CheckDataAvailableInBatch(bc, blockChain); err != nil { + log.Debug("CheckDataAvailableInBatch", "err", err) return 0, err } } diff --git a/core/data_availability.go b/core/data_availability.go index 73502065ff..4b9097c0c4 100644 --- a/core/data_availability.go +++ b/core/data_availability.go @@ -11,7 +11,6 @@ import ( "github.com/ethereum/go-ethereum/consensus" "github.com/ethereum/go-ethereum/core/types" "github.com/ethereum/go-ethereum/crypto/kzg4844" - "github.com/ethereum/go-ethereum/log" "github.com/ethereum/go-ethereum/params" ) @@ -64,9 +63,6 @@ func IsDataAvailable(chain consensus.ChainHeaderReader, block *types.Block) (err if highest == nil || highest.Number.Cmp(current.Number) < 0 { highest = current } - defer func() { - log.Debug("IsDataAvailable", "block", block.Number(), "hash", block.Hash(), "highest", highest.Number, "sidecars", len(block.Sidecars()), "err", err) - }() if block.NumberU64()+params.MinBlocksForBlobRequests < highest.Number.Uint64() { // if we needn't check DA of this block, just clean it block.CleanSidecars() From 1443e89d72a321a55354d9e9611d502186ca1a36 Mon Sep 17 00:00:00 2001 From: buddh0 Date: Tue, 26 Mar 2024 19:33:21 +0800 Subject: [PATCH 03/15] narrow the semantics of func resetItems --- core/rawdb/freezer.go | 21 +++++++++++++++------ core/rawdb/freezer_table.go | 20 ++++---------------- 2 files changed, 19 insertions(+), 22 deletions(-) diff --git a/core/rawdb/freezer.go b/core/rawdb/freezer.go index 33603b973f..aa0fc3ad67 100644 --- a/core/rawdb/freezer.go +++ b/core/rawdb/freezer.go @@ -348,7 +348,7 @@ func (f *Freezer) TruncateHead(items uint64) (uint64, error) { if kind != ChainFreezerBlobSidecarTable { return 0, err } - nt, err := table.resetItems(items-f.offset, items-f.offset) + nt, err := table.resetItems(items - f.offset) if err != nil { return 0, err } @@ -489,7 +489,7 @@ func (f *Freezer) repair() error { if kind != ChainFreezerBlobSidecarTable { return err } - nt, err := table.resetItems(head, head) + nt, err := table.resetItems(head) if err != nil { return err } @@ -720,11 +720,20 @@ func (f *Freezer) ResetTable(kind string, tail, head uint64, onlyEmpty bool) err return nil } - nt, err := f.tables[kind].resetItems(tail-f.offset, head-f.offset) - if err != nil { - return err + if tail != head { + if err := f.tables[kind].truncateHead(head); err != nil { + return err + } + if err := f.tables[kind].truncateTail(tail); err != nil { + return err + } + } else { + nt, err := f.tables[kind].resetItems(head - f.offset) + if err != nil { + return err + } + f.tables[kind] = nt } - f.tables[kind] = nt if err := f.repair(); err != nil { for _, table := range f.tables { diff --git a/core/rawdb/freezer_table.go b/core/rawdb/freezer_table.go index 3fc313e749..68021b3bb1 100644 --- a/core/rawdb/freezer_table.go +++ b/core/rawdb/freezer_table.go @@ -1029,34 +1029,22 @@ func (t *freezerTable) ResetItemsOffset(virtualTail uint64) error { return nil } -// resetItems reset freezer table head & tail +// resetItems reset freezer table to 0 items with head `headAndTail` // only used for ChainFreezerBlobSidecarTable now -func (t *freezerTable) resetItems(tail, head uint64) (*freezerTable, error) { +func (t *freezerTable) resetItems(headAndTail uint64) (*freezerTable, error) { if t.readonly { return nil, errors.New("resetItems in readonly mode") } itemHidden := t.itemHidden.Load() items := t.items.Load() - if tail > head || (tail < head && (itemHidden > tail || items < head)) { + if itemHidden > headAndTail || items < headAndTail { return nil, errors.New("cannot reset to non-exist range") } - var err error - if tail < head { - if err = t.truncateHead(head); err != nil { - return nil, err - } - if err = t.truncateTail(tail); err != nil { - return nil, err - } - return t, nil - } - t.lock.Lock() defer t.lock.Unlock() - // if tail == head, it means table reset to 0 item // remove all data files t.head.Close() t.releaseFilesAfter(0, true) @@ -1086,7 +1074,7 @@ func (t *freezerTable) resetItems(tail, head uint64) (*freezerTable, error) { } tailIndex := indexEntry{ filenum: 0, - offset: uint32(tail), + offset: uint32(headAndTail), } if _, err = index.Write(tailIndex.append(nil)); err != nil { return nil, err From 48affc90bdca915b65aa19d1d00e81a07652abe6 Mon Sep 17 00:00:00 2001 From: GalaIO Date: Wed, 27 Mar 2024 17:43:42 +0800 Subject: [PATCH 04/15] freezer: refactor ResetTable & ResetItems; --- core/rawdb/chain_freezer.go | 9 +--- core/rawdb/database.go | 14 ++++-- core/rawdb/freezer.go | 63 +++++++++++++++++--------- core/rawdb/freezer_resettable.go | 23 +++++++--- core/rawdb/freezer_table.go | 23 ++++------ core/rawdb/freezer_table_test.go | 76 ++++++++++++-------------------- core/rawdb/freezer_test.go | 42 ++++++++++++++++-- core/rawdb/prunedfreezer.go | 8 +++- core/rawdb/table.go | 10 ++++- ethdb/database.go | 7 ++- ethdb/remotedb/remotedb.go | 14 ++++-- 11 files changed, 172 insertions(+), 117 deletions(-) diff --git a/core/rawdb/chain_freezer.go b/core/rawdb/chain_freezer.go index 5d9443ad0f..90df0ebb9f 100644 --- a/core/rawdb/chain_freezer.go +++ b/core/rawdb/chain_freezer.go @@ -273,13 +273,8 @@ func (f *chainFreezer) tryPruneBlobAncient(env *ethdb.FreezerEnv, num uint64) { return } expectTail := num - reserveThreshold - h, err := f.TableAncients(ChainFreezerBlobSidecarTable) - if err != nil { - log.Error("Cannot get blob ancient head when prune", "block", num) - return - } start := time.Now() - if err = f.ResetTable(ChainFreezerBlobSidecarTable, expectTail, h, false); err != nil { + if _, err := f.TruncateTableTail(ChainFreezerBlobSidecarTable, expectTail); err != nil { log.Error("Cannot prune blob ancient", "block", num, "expectTail", expectTail, "err", err) return } @@ -427,5 +422,5 @@ func isCancun(env *ethdb.FreezerEnv, num *big.Int, time uint64) bool { } func ResetEmptyBlobAncientTable(db ethdb.AncientWriter, next uint64) error { - return db.ResetTable(ChainFreezerBlobSidecarTable, next, next, true) + return db.ResetTable(ChainFreezerBlobSidecarTable, next, true) } diff --git a/core/rawdb/database.go b/core/rawdb/database.go index a0db10eb6e..0fe10da8b9 100644 --- a/core/rawdb/database.go +++ b/core/rawdb/database.go @@ -177,10 +177,6 @@ func (db *nofreezedb) ModifyAncients(func(ethdb.AncientWriteOp) error) (int64, e return 0, errNotSupported } -func (db *nofreezedb) ResetTable(kind string, tail uint64, head uint64, onlyEmpty bool) error { - return errNotSupported -} - // TruncateHead returns an error as we don't have a backing chain freezer. func (db *nofreezedb) TruncateHead(items uint64) (uint64, error) { return 0, errNotSupported @@ -191,6 +187,16 @@ func (db *nofreezedb) TruncateTail(items uint64) (uint64, error) { return 0, errNotSupported } +// TruncateTableTail will truncate certain table to new tail +func (db *nofreezedb) TruncateTableTail(kind string, tail uint64) (uint64, error) { + return 0, errNotSupported +} + +// ResetTable will reset certain table with new start point +func (db *nofreezedb) ResetTable(kind string, startAt uint64, onlyEmpty bool) error { + return errNotSupported +} + // Sync returns an error as we don't have a backing chain freezer. func (db *nofreezedb) Sync() error { return errNotSupported diff --git a/core/rawdb/freezer.go b/core/rawdb/freezer.go index aa0fc3ad67..f011224469 100644 --- a/core/rawdb/freezer.go +++ b/core/rawdb/freezer.go @@ -698,8 +698,39 @@ func (f *Freezer) MigrateTable(kind string, convert convertLegacyFn) error { return nil } +// TruncateTableTail will truncate certain table to new tail +func (f *Freezer) TruncateTableTail(kind string, tail uint64) (uint64, error) { + f.writeLock.Lock() + defer f.writeLock.Unlock() + + if f.readonly { + return 0, errReadOnly + } + + if !slices.Contains(additionTables, kind) { + return 0, errors.New("only new added table could be truncated independently") + } + if tail < f.offset { + return 0, errors.New("the input tail&head is less than offset") + } + t, exist := f.tables[kind] + if !exist { + return 0, errors.New("you reset a non-exist table") + } + + old := t.itemHidden.Load() + f.offset + if err := t.truncateTail(tail - f.offset); err != nil { + return 0, err + } + return old, nil +} + +// ResetTable will reset certain table with new start point // only used for ChainFreezerBlobSidecarTable now -func (f *Freezer) ResetTable(kind string, tail, head uint64, onlyEmpty bool) error { +func (f *Freezer) ResetTable(kind string, startAt uint64, onlyEmpty bool) error { + f.writeLock.Lock() + defer f.writeLock.Unlock() + if f.readonly { return errReadOnly } @@ -707,34 +738,22 @@ func (f *Freezer) ResetTable(kind string, tail, head uint64, onlyEmpty bool) err return err } - f.writeLock.Lock() - defer f.writeLock.Unlock() - if tail < f.offset || head < f.offset { - return errors.New("the input tail&head is less than offset") - } - if _, exist := f.tables[kind]; !exist { + t, exist := f.tables[kind] + if !exist { return errors.New("you reset a non-exist table") } + // if you reset a non empty table just skip - if onlyEmpty && !EmptyTable(f.tables[kind]) { + if onlyEmpty && !EmptyTable(t) { return nil } - - if tail != head { - if err := f.tables[kind].truncateHead(head); err != nil { - return err - } - if err := f.tables[kind].truncateTail(tail); err != nil { - return err - } - } else { - nt, err := f.tables[kind].resetItems(head - f.offset) - if err != nil { - return err - } - f.tables[kind] = nt + nt, err := t.resetItems(startAt - f.offset) + if err != nil { + return err } + f.tables[kind] = nt + // repair all tables with same tail & head if err := f.repair(); err != nil { for _, table := range f.tables { table.Close() diff --git a/core/rawdb/freezer_resettable.go b/core/rawdb/freezer_resettable.go index 1bbc39d7f5..dfefc6a45e 100644 --- a/core/rawdb/freezer_resettable.go +++ b/core/rawdb/freezer_resettable.go @@ -187,13 +187,6 @@ func (f *ResettableFreezer) ModifyAncients(fn func(ethdb.AncientWriteOp) error) return f.freezer.ModifyAncients(fn) } -func (f *ResettableFreezer) ResetTable(kind string, tail uint64, head uint64, onlyEmpty bool) error { - f.lock.RLock() - defer f.lock.RUnlock() - - return f.freezer.ResetTable(kind, tail, head, onlyEmpty) -} - // TruncateHead discards any recent data above the provided threshold number. // It returns the previous head number. func (f *ResettableFreezer) TruncateHead(items uint64) (uint64, error) { @@ -212,6 +205,22 @@ func (f *ResettableFreezer) TruncateTail(tail uint64) (uint64, error) { return f.freezer.TruncateTail(tail) } +// TruncateTableTail will truncate certain table to new tail +func (f *ResettableFreezer) TruncateTableTail(kind string, tail uint64) (uint64, error) { + f.lock.RLock() + defer f.lock.RUnlock() + + return f.freezer.TruncateTableTail(kind, tail) +} + +// ResetTable will reset certain table with new start point +func (f *ResettableFreezer) ResetTable(kind string, startAt uint64, onlyEmpty bool) error { + f.lock.RLock() + defer f.lock.RUnlock() + + return f.freezer.ResetTable(kind, startAt, onlyEmpty) +} + // Sync flushes all data tables to disk. func (f *ResettableFreezer) Sync() error { f.lock.RLock() diff --git a/core/rawdb/freezer_table.go b/core/rawdb/freezer_table.go index 68021b3bb1..2158bc4b0e 100644 --- a/core/rawdb/freezer_table.go +++ b/core/rawdb/freezer_table.go @@ -1029,29 +1029,22 @@ func (t *freezerTable) ResetItemsOffset(virtualTail uint64) error { return nil } -// resetItems reset freezer table to 0 items with head `headAndTail` +// resetItems reset freezer table to 0 items with new startAt // only used for ChainFreezerBlobSidecarTable now -func (t *freezerTable) resetItems(headAndTail uint64) (*freezerTable, error) { +func (t *freezerTable) resetItems(startAt uint64) (*freezerTable, error) { + t.lock.Lock() + defer t.lock.Unlock() if t.readonly { return nil, errors.New("resetItems in readonly mode") } - itemHidden := t.itemHidden.Load() - items := t.items.Load() - if itemHidden > headAndTail || items < headAndTail { - return nil, errors.New("cannot reset to non-exist range") - } - - t.lock.Lock() - defer t.lock.Unlock() - // remove all data files t.head.Close() t.releaseFilesAfter(0, true) t.releaseFile(0) - // reset meta data file - if err := writeMetadata(t.meta, newMetadata(items)); err != nil { + // overwrite metadata file + if err := writeMetadata(t.meta, newMetadata(startAt)); err != nil { return nil, err } if err := t.meta.Sync(); err != nil { @@ -1059,7 +1052,7 @@ func (t *freezerTable) resetItems(headAndTail uint64) (*freezerTable, error) { } t.meta.Close() - // reset the index file + // recreate the index file t.index.Close() os.Remove(t.index.Name()) var idxName string @@ -1074,7 +1067,7 @@ func (t *freezerTable) resetItems(headAndTail uint64) (*freezerTable, error) { } tailIndex := indexEntry{ filenum: 0, - offset: uint32(headAndTail), + offset: uint32(startAt), } if _, err = index.Write(tailIndex.append(nil)); err != nil { return nil, err diff --git a/core/rawdb/freezer_table_test.go b/core/rawdb/freezer_table_test.go index da8520fb3e..72dd60c67c 100644 --- a/core/rawdb/freezer_table_test.go +++ b/core/rawdb/freezer_table_test.go @@ -1384,59 +1384,40 @@ func TestResetItems(t *testing.T) { // Write 7 x 20 bytes, splitting out into four files batch := f.newBatch(0) - require.NoError(t, batch.AppendRaw(0, getChunk(20, 0xFF))) - require.NoError(t, batch.AppendRaw(1, getChunk(20, 0xEE))) - require.NoError(t, batch.AppendRaw(2, getChunk(20, 0xdd))) - require.NoError(t, batch.AppendRaw(3, getChunk(20, 0xcc))) - require.NoError(t, batch.AppendRaw(4, getChunk(20, 0xbb))) - require.NoError(t, batch.AppendRaw(5, getChunk(20, 0xaa))) - require.NoError(t, batch.AppendRaw(6, getChunk(20, 0x11))) + require.NoError(t, batch.AppendRaw(0, getChunk(20, 0x00))) + require.NoError(t, batch.AppendRaw(1, getChunk(20, 0x11))) + require.NoError(t, batch.AppendRaw(2, getChunk(20, 0x22))) + require.NoError(t, batch.AppendRaw(3, getChunk(20, 0x33))) + require.NoError(t, batch.AppendRaw(4, getChunk(20, 0x44))) + require.NoError(t, batch.AppendRaw(5, getChunk(20, 0x55))) + require.NoError(t, batch.AppendRaw(6, getChunk(20, 0x66))) require.NoError(t, batch.commit()) // nothing to do, all the items should still be there. - f, err = f.resetItems(0, 7) + f, err = f.resetItems(0) require.NoError(t, err) - fmt.Println(f.dumpIndexString(0, 1000)) - checkRetrieve(t, f, map[uint64][]byte{ - 0: getChunk(20, 0xFF), - 1: getChunk(20, 0xEE), - 2: getChunk(20, 0xdd), - 3: getChunk(20, 0xcc), - 4: getChunk(20, 0xbb), - 5: getChunk(20, 0xaa), - 6: getChunk(20, 0x11), - }) - - f, err = f.resetItems(1, 5) + f, err = f.resetItems(8) + require.NoError(t, err) + f, err = f.resetItems(7) require.NoError(t, err) - _, err = f.resetItems(0, 5) - require.Error(t, err) - _, err = f.resetItems(1, 6) - require.Error(t, err) - fmt.Println(f.dumpIndexString(0, 1000)) checkRetrieveError(t, f, map[uint64]error{ 0: errOutOfBounds, - }) - checkRetrieve(t, f, map[uint64][]byte{ - 1: getChunk(20, 0xEE), - 2: getChunk(20, 0xdd), - 3: getChunk(20, 0xcc), - 4: getChunk(20, 0xbb), - }) - - f, err = f.resetItems(4, 4) - require.NoError(t, err) - checkRetrieveError(t, f, map[uint64]error{ - 4: errOutOfBounds, + 6: errOutOfBounds, }) + // append batch = f.newBatch(0) require.Error(t, batch.AppendRaw(0, getChunk(20, 0xa0))) - require.NoError(t, batch.AppendRaw(4, getChunk(20, 0xa4))) - require.NoError(t, batch.AppendRaw(5, getChunk(20, 0xa5))) + require.NoError(t, batch.AppendRaw(7, getChunk(20, 0x77))) + require.NoError(t, batch.AppendRaw(8, getChunk(20, 0x88))) + require.NoError(t, batch.AppendRaw(9, getChunk(20, 0x99))) require.NoError(t, batch.commit()) fmt.Println(f.dumpIndexString(0, 1000)) + checkRetrieve(t, f, map[uint64][]byte{ + 7: getChunk(20, 0x77), + 9: getChunk(20, 0x99), + }) // Reopen the table, the deletion information should be persisted as well f.Close() @@ -1446,22 +1427,19 @@ func TestResetItems(t *testing.T) { } fmt.Println(f.dumpIndexString(0, 1000)) checkRetrieveError(t, f, map[uint64]error{ - 0: errOutOfBounds, + 0: errOutOfBounds, + 6: errOutOfBounds, + 10: errOutOfBounds, }) checkRetrieve(t, f, map[uint64][]byte{ - 4: getChunk(20, 0xa4), - 5: getChunk(20, 0xa5), + 7: getChunk(20, 0x77), + 9: getChunk(20, 0x99), }) // truncate all, the entire freezer should be deleted - f.truncateTail(6) + f.truncateTail(10) checkRetrieveError(t, f, map[uint64]error{ 0: errOutOfBounds, - 1: errOutOfBounds, - 2: errOutOfBounds, - 3: errOutOfBounds, - 4: errOutOfBounds, - 5: errOutOfBounds, - 6: errOutOfBounds, + 9: errOutOfBounds, }) } diff --git a/core/rawdb/freezer_test.go b/core/rawdb/freezer_test.go index 60103ab675..2920cfaa29 100644 --- a/core/rawdb/freezer_test.go +++ b/core/rawdb/freezer_test.go @@ -371,36 +371,70 @@ func TestFreezer_AdditionTables(t *testing.T) { f, err = NewFreezer(dir, "", false, 0, 2049, map[string]bool{"o1": true, "o2": true, "a1": true}) require.NoError(t, err) frozen, _ := f.Ancients() - f.ResetTable("a1", frozen, frozen, true) + require.NoError(t, f.ResetTable("a1", frozen, true)) _, err = f.ModifyAncients(func(op ethdb.AncientWriteOp) error { - if err := op.AppendRaw("o1", 2, item); err != nil { + if err := appendSameItem(op, []string{"o1", "o2", "a1"}, 2, item); err != nil { return err } - if err := op.AppendRaw("o2", 2, item); err != nil { + if err := appendSameItem(op, []string{"o1", "o2", "a1"}, 3, item); err != nil { return err } - if err := op.AppendRaw("a1", 2, item); err != nil { + if err := appendSameItem(op, []string{"o1", "o2", "a1"}, 4, item); err != nil { return err } return nil }) require.NoError(t, err) + + // check additional table boundary _, err = f.Ancient("a1", 1) require.Error(t, err) actual, err := f.Ancient("a1", 2) require.NoError(t, err) require.Equal(t, item, actual) + + // truncate additional table, and check boundary + _, err = f.TruncateTableTail("o1", 3) + require.Error(t, err) + _, err = f.TruncateTableTail("a1", 3) + require.NoError(t, err) + _, err = f.Ancient("a1", 2) + require.Error(t, err) + actual, err = f.Ancient("a1", 3) + require.NoError(t, err) + require.Equal(t, item, actual) + + // check additional table head + ancients, err := f.TableAncients("a1") + require.NoError(t, err) + require.Equal(t, uint64(5), ancients) require.NoError(t, f.Close()) // reopen and read f, err = NewFreezer(dir, "", true, 0, 2049, map[string]bool{"o1": true, "o2": true, "a1": true}) require.NoError(t, err) + + // recheck additional table boundary actual, err = f.Ancient("a1", 2) + require.Error(t, err) + actual, err = f.Ancient("a1", 3) require.NoError(t, err) require.Equal(t, item, actual) + ancients, err = f.TableAncients("a1") + require.NoError(t, err) + require.Equal(t, uint64(5), ancients) require.NoError(t, f.Close()) } +func appendSameItem(op ethdb.AncientWriteOp, tables []string, i uint64, item []byte) error { + for _, t := range tables { + if err := op.AppendRaw(t, i, item); err != nil { + return err + } + } + return nil +} + func newFreezerForTesting(t *testing.T, tables map[string]bool) (*Freezer, string) { t.Helper() diff --git a/core/rawdb/prunedfreezer.go b/core/rawdb/prunedfreezer.go index 992d7802ed..51c7729758 100644 --- a/core/rawdb/prunedfreezer.go +++ b/core/rawdb/prunedfreezer.go @@ -328,6 +328,12 @@ func (f *prunedfreezer) ModifyAncients(func(ethdb.AncientWriteOp) error) (int64, return 0, errNotSupported } -func (f *prunedfreezer) ResetTable(kind string, tail uint64, head uint64, onlyEmpty bool) error { +// TruncateTableTail will truncate certain table to new tail +func (f *prunedfreezer) TruncateTableTail(kind string, tail uint64) (uint64, error) { + return 0, errNotSupported +} + +// ResetTable will reset certain table with new start point +func (f *prunedfreezer) ResetTable(kind string, startAt uint64, onlyEmpty bool) error { return errNotSupported } diff --git a/core/rawdb/table.go b/core/rawdb/table.go index 6d594dfa14..15c3462560 100644 --- a/core/rawdb/table.go +++ b/core/rawdb/table.go @@ -101,8 +101,14 @@ func (t *table) ModifyAncients(fn func(ethdb.AncientWriteOp) error) (int64, erro return t.db.ModifyAncients(fn) } -func (t *table) ResetTable(kind string, tail uint64, head uint64, onlyEmpty bool) error { - return t.db.ResetTable(kind, tail, head, onlyEmpty) +// TruncateTableTail will truncate certain table to new tail +func (t *table) TruncateTableTail(kind string, tail uint64) (uint64, error) { + return t.db.TruncateTableTail(kind, tail) +} + +// ResetTable will reset certain table with new start point +func (t *table) ResetTable(kind string, startAt uint64, onlyEmpty bool) error { + return t.db.ResetTable(kind, startAt, onlyEmpty) } func (t *table) ReadAncients(fn func(reader ethdb.AncientReaderOp) error) (err error) { diff --git a/ethdb/database.go b/ethdb/database.go index f92ee77b84..463b5bb164 100644 --- a/ethdb/database.go +++ b/ethdb/database.go @@ -141,8 +141,11 @@ type AncientWriter interface { // in the newest format. MigrateTable(string, func([]byte) ([]byte, error)) error - // ResetTable will reset certain table to new boundary - ResetTable(kind string, tail uint64, head uint64, onlyEmpty bool) error + // TruncateTableTail will truncate certain table to new tail + TruncateTableTail(kind string, tail uint64) (uint64, error) + + // ResetTable will reset certain table with new start point + ResetTable(kind string, startAt uint64, onlyEmpty bool) error } type FreezerEnv struct { diff --git a/ethdb/remotedb/remotedb.go b/ethdb/remotedb/remotedb.go index 2d9768236e..8e2135c2b1 100644 --- a/ethdb/remotedb/remotedb.go +++ b/ethdb/remotedb/remotedb.go @@ -126,10 +126,6 @@ func (db *Database) ModifyAncients(f func(ethdb.AncientWriteOp) error) (int64, e panic("not supported") } -func (db *Database) ResetTable(kind string, tail uint64, head uint64, onlyEmpty bool) error { - panic("not supported") -} - func (db *Database) AncientReset(tail, head uint64) error { panic("not supported") } @@ -142,6 +138,16 @@ func (db *Database) TruncateTail(n uint64) (uint64, error) { panic("not supported") } +// TruncateTableTail will truncate certain table to new tail +func (db *Database) TruncateTableTail(kind string, tail uint64) (uint64, error) { + panic("not supported") +} + +// ResetTable will reset certain table with new start point +func (db *Database) ResetTable(kind string, startAt uint64, onlyEmpty bool) error { + panic("not supported") +} + func (db *Database) Sync() error { return nil } From 35be77ac966c17f3b70aedd7dc6263661c83332c Mon Sep 17 00:00:00 2001 From: GalaIO Date: Wed, 27 Mar 2024 17:59:55 +0800 Subject: [PATCH 05/15] freezer: refactor ResetTable & ResetItems; --- core/rawdb/freezer.go | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/core/rawdb/freezer.go b/core/rawdb/freezer.go index f011224469..4f94c5eb49 100644 --- a/core/rawdb/freezer.go +++ b/core/rawdb/freezer.go @@ -734,9 +734,6 @@ func (f *Freezer) ResetTable(kind string, startAt uint64, onlyEmpty bool) error if f.readonly { return errReadOnly } - if err := f.Sync(); err != nil { - return err - } t, exist := f.tables[kind] if !exist { @@ -747,6 +744,10 @@ func (f *Freezer) ResetTable(kind string, startAt uint64, onlyEmpty bool) error if onlyEmpty && !EmptyTable(t) { return nil } + + if err := f.Sync(); err != nil { + return err + } nt, err := t.resetItems(startAt - f.offset) if err != nil { return err From bce8a737be570ec9250da8bd05cfaaa448486d3e Mon Sep 17 00:00:00 2001 From: GalaIO Date: Wed, 27 Mar 2024 18:02:15 +0800 Subject: [PATCH 06/15] fix: fix some lint issues; --- core/rawdb/freezer_test.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/rawdb/freezer_test.go b/core/rawdb/freezer_test.go index 2920cfaa29..3dec88db9c 100644 --- a/core/rawdb/freezer_test.go +++ b/core/rawdb/freezer_test.go @@ -415,7 +415,7 @@ func TestFreezer_AdditionTables(t *testing.T) { require.NoError(t, err) // recheck additional table boundary - actual, err = f.Ancient("a1", 2) + _, err = f.Ancient("a1", 2) require.Error(t, err) actual, err = f.Ancient("a1", 3) require.NoError(t, err) From e3bcf46ae53a0205f625cd1e537b3aa78d0d2467 Mon Sep 17 00:00:00 2001 From: buddh0 Date: Wed, 27 Mar 2024 22:23:19 +0800 Subject: [PATCH 07/15] only newSnapshot for genesis block --- consensus/parlia/parlia.go | 14 +++++--------- 1 file changed, 5 insertions(+), 9 deletions(-) diff --git a/consensus/parlia/parlia.go b/consensus/parlia/parlia.go index e77e8c0b85..acf0f3b7be 100644 --- a/consensus/parlia/parlia.go +++ b/consensus/parlia/parlia.go @@ -701,10 +701,8 @@ func (p *Parlia) snapshot(chain consensus.ChainHeaderReader, number uint64, hash } } - // If we're at the genesis, snapshot the initial state. Alternatively if we have - // piled up more headers than allowed to be reorged (chain reinit from a freezer), - // consider the checkpoint trusted and snapshot it. - if number == 0 || (number%p.config.Epoch == 0 && (len(headers) > int(params.FullImmutabilityThreshold)/10)) { + // If we're at the genesis, snapshot the initial state. + if number == 0 { checkpoint := chain.GetHeaderByNumber(number) if checkpoint != nil { // get checkpoint data @@ -718,12 +716,10 @@ func (p *Parlia) snapshot(chain consensus.ChainHeaderReader, number uint64, hash // new snapshot snap = newSnapshot(p.config, p.signatures, number, hash, validators, voteAddrs, p.ethAPI) - if snap.Number%checkpointInterval == 0 { // snapshot will only be loaded when snap.Number%checkpointInterval == 0 - if err := snap.store(p.db); err != nil { - return nil, err - } - log.Info("Stored checkpoint snapshot to disk", "number", number, "hash", hash) + if err := snap.store(p.db); err != nil { + return nil, err } + log.Info("Stored checkpoint snapshot to disk", "number", number, "hash", hash) break } } From a76e9ed22f395a50fb5f38d0bb31637a05c94317 Mon Sep 17 00:00:00 2001 From: GalaIO Date: Thu, 28 Mar 2024 10:43:06 +0800 Subject: [PATCH 08/15] freezer: opt reset blob table logic; --- core/rawdb/accessors_chain.go | 42 ++++++++++++++++++++++++++++++----- core/rawdb/chain_freezer.go | 4 ++-- core/rawdb/freezer.go | 12 +++++----- 3 files changed, 44 insertions(+), 14 deletions(-) diff --git a/core/rawdb/accessors_chain.go b/core/rawdb/accessors_chain.go index dd56ca1b64..b5122c3da7 100644 --- a/core/rawdb/accessors_chain.go +++ b/core/rawdb/accessors_chain.go @@ -800,6 +800,7 @@ func WriteBlock(db ethdb.KeyValueWriter, block *types.Block) { // WriteAncientBlocks writes entire block data into ancient store and returns the total written size. func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts []types.Receipts, td *big.Int) (int64, error) { + // find cancun index, it's used for new added blob ancient table cancunIndex := -1 for i, block := range blocks { if block.Sidecars() != nil { @@ -808,17 +809,44 @@ func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts } } log.Info("WriteAncientBlocks", "startAt", blocks[0].Number(), "cancunIndex", cancunIndex, "len", len(blocks)) - if cancunIndex >= 0 { - if err := ResetEmptyBlobAncientTable(db, blocks[cancunIndex].NumberU64()); err != nil { - return 0, err - } - } var ( tdSum = new(big.Int).Set(td) stReceipts []*types.ReceiptForStorage + preSize int64 + err error ) - return db.ModifyAncients(func(op ethdb.AncientWriteOp) error { + + // handle pre-cancun blocks + if cancunIndex > 0 { + preSize, err = db.ModifyAncients(func(op ethdb.AncientWriteOp) error { + for i, block := range blocks[:cancunIndex] { + // Convert receipts to storage format and sum up total difficulty. + stReceipts = stReceipts[:0] + for _, receipt := range receipts[i] { + stReceipts = append(stReceipts, (*types.ReceiptForStorage)(receipt)) + } + header := block.Header() + if i > 0 { + tdSum.Add(tdSum, header.Difficulty) + } + if err := writeAncientBlock(op, block, header, stReceipts, tdSum); err != nil { + return err + } + } + return nil + }) + blocks = blocks[cancunIndex:] + } + + // It will reset blob ancient table at cancunIndex + if cancunIndex >= 0 { + if err = ResetEmptyBlobAncientTable(db, blocks[cancunIndex].NumberU64()); err != nil { + return 0, err + } + } + + postSize, err := db.ModifyAncients(func(op ethdb.AncientWriteOp) error { for i, block := range blocks { // Convert receipts to storage format and sum up total difficulty. stReceipts = stReceipts[:0] @@ -835,6 +863,8 @@ func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts } return nil }) + + return preSize + postSize, err } // ReadBlobSidecarsRLP retrieves all the transaction blobs belonging to a block in RLP encoding. diff --git a/core/rawdb/chain_freezer.go b/core/rawdb/chain_freezer.go index 90df0ebb9f..5f6cd15ad6 100644 --- a/core/rawdb/chain_freezer.go +++ b/core/rawdb/chain_freezer.go @@ -252,7 +252,7 @@ func (f *chainFreezer) freeze(db ethdb.KeyValueStore) { env, _ := f.freezeEnv.Load().(*ethdb.FreezerEnv) // try prune blob data after cancun fork if isCancun(env, head.Number, head.Time) { - f.tryPruneBlobAncient(env, *number) + f.tryPruneBlobAncientTable(env, *number) } // Avoid database thrashing with tiny writes @@ -262,7 +262,7 @@ func (f *chainFreezer) freeze(db ethdb.KeyValueStore) { } } -func (f *chainFreezer) tryPruneBlobAncient(env *ethdb.FreezerEnv, num uint64) { +func (f *chainFreezer) tryPruneBlobAncientTable(env *ethdb.FreezerEnv, num uint64) { extraReserve := getBlobExtraReserveFromEnv(env) // It means that there is no need for pruning if extraReserve == 0 { diff --git a/core/rawdb/freezer.go b/core/rawdb/freezer.go index 4f94c5eb49..ca237dd3aa 100644 --- a/core/rawdb/freezer.go +++ b/core/rawdb/freezer.go @@ -700,13 +700,13 @@ func (f *Freezer) MigrateTable(kind string, convert convertLegacyFn) error { // TruncateTableTail will truncate certain table to new tail func (f *Freezer) TruncateTableTail(kind string, tail uint64) (uint64, error) { - f.writeLock.Lock() - defer f.writeLock.Unlock() - if f.readonly { return 0, errReadOnly } + f.writeLock.Lock() + defer f.writeLock.Unlock() + if !slices.Contains(additionTables, kind) { return 0, errors.New("only new added table could be truncated independently") } @@ -728,13 +728,13 @@ func (f *Freezer) TruncateTableTail(kind string, tail uint64) (uint64, error) { // ResetTable will reset certain table with new start point // only used for ChainFreezerBlobSidecarTable now func (f *Freezer) ResetTable(kind string, startAt uint64, onlyEmpty bool) error { - f.writeLock.Lock() - defer f.writeLock.Unlock() - if f.readonly { return errReadOnly } + f.writeLock.Lock() + defer f.writeLock.Unlock() + t, exist := f.tables[kind] if !exist { return errors.New("you reset a non-exist table") From 9409dbfa2350ee4b86dee2954d98fe1ff6911798 Mon Sep 17 00:00:00 2001 From: GalaIO Date: Wed, 27 Mar 2024 21:20:14 +0800 Subject: [PATCH 09/15] fix: opt da check logic; --- core/data_availability.go | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/core/data_availability.go b/core/data_availability.go index 4b9097c0c4..cd8e58a147 100644 --- a/core/data_availability.go +++ b/core/data_availability.go @@ -48,13 +48,10 @@ func validateBlobSidecar(hashes []common.Hash, sidecar *types.BlobSidecar) error func IsDataAvailable(chain consensus.ChainHeaderReader, block *types.Block) (err error) { // refer logic in ValidateBody if !chain.Config().IsCancun(block.Number(), block.Time()) { - if block.Sidecars() == nil { - return nil - } else { + if block.Sidecars() != nil { return errors.New("sidecars present in block body before cancun") } - } else if block.Sidecars() == nil { - return errors.New("missing sidecars in block body after cancun") + return nil } // only required to check within MinBlocksForBlobRequests block's DA @@ -69,6 +66,10 @@ func IsDataAvailable(chain consensus.ChainHeaderReader, block *types.Block) (err return nil } + // if sidecar is nil, just clean it. And it will be used for saving in ancient. + if block.Sidecars() == nil { + block.CleanSidecars() + } sidecars := block.Sidecars() for _, s := range sidecars { if err := s.SanityCheck(block.Number(), block.Hash()); err != nil { From f3252365d887dbfb58c58af6ce095faf1eb90567 Mon Sep 17 00:00:00 2001 From: GalaIO Date: Thu, 28 Mar 2024 11:16:40 +0800 Subject: [PATCH 10/15] freezer: opt reset blob table logic; --- core/rawdb/accessors_chain.go | 3 +++ 1 file changed, 3 insertions(+) diff --git a/core/rawdb/accessors_chain.go b/core/rawdb/accessors_chain.go index b5122c3da7..c9cfa6364d 100644 --- a/core/rawdb/accessors_chain.go +++ b/core/rawdb/accessors_chain.go @@ -836,6 +836,9 @@ func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts } return nil }) + if err != nil { + return preSize, err + } blocks = blocks[cancunIndex:] } From 04c267072604699fa0a8db401dda5be5daf50c61 Mon Sep 17 00:00:00 2001 From: GalaIO Date: Thu, 28 Mar 2024 13:19:42 +0800 Subject: [PATCH 11/15] freezer: opt reset blob table logic; --- core/rawdb/accessors_chain.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/rawdb/accessors_chain.go b/core/rawdb/accessors_chain.go index c9cfa6364d..d06d60cc7c 100644 --- a/core/rawdb/accessors_chain.go +++ b/core/rawdb/accessors_chain.go @@ -839,7 +839,6 @@ func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts if err != nil { return preSize, err } - blocks = blocks[cancunIndex:] } // It will reset blob ancient table at cancunIndex @@ -847,6 +846,7 @@ func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts if err = ResetEmptyBlobAncientTable(db, blocks[cancunIndex].NumberU64()); err != nil { return 0, err } + blocks = blocks[cancunIndex:] } postSize, err := db.ModifyAncients(func(op ethdb.AncientWriteOp) error { From 5dfce5889b64889861da7b34de98f7ac51c82ed5 Mon Sep 17 00:00:00 2001 From: GalaIO Date: Thu, 28 Mar 2024 14:19:26 +0800 Subject: [PATCH 12/15] fix: fix failed UTs; --- core/data_availability_test.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/data_availability_test.go b/core/data_availability_test.go index 153f9b948b..e6963a2615 100644 --- a/core/data_availability_test.go +++ b/core/data_availability_test.go @@ -121,7 +121,7 @@ func TestIsDataAvailable(t *testing.T) { }, nil), chasingHead: params.MinBlocksForBlobRequests + 1, withSidecar: false, - err: true, + err: false, }, } From c6d219fce781d9fc321e44c55e06d912615bf040 Mon Sep 17 00:00:00 2001 From: buddh0 Date: Thu, 28 Mar 2024 15:43:47 +0800 Subject: [PATCH 13/15] core/types: fix EmptyBody --- core/types/block.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/types/block.go b/core/types/block.go index a01e09e0cf..d088a5e4c7 100644 --- a/core/types/block.go +++ b/core/types/block.go @@ -183,7 +183,7 @@ func (h *Header) SanityCheck() error { // that is: no transactions, no uncles and no withdrawals. func (h *Header) EmptyBody() bool { if h.WithdrawalsHash != nil { - return h.TxHash == EmptyTxsHash && *h.WithdrawalsHash == EmptyWithdrawalsHash + return h.TxHash == EmptyTxsHash && (*h.WithdrawalsHash == EmptyWithdrawalsHash || *h.WithdrawalsHash == common.Hash{}) } return h.TxHash == EmptyTxsHash && h.UncleHash == EmptyUncleHash } From cd1f42ae2d8055a77b5e49d56164e533d0ebe98e Mon Sep 17 00:00:00 2001 From: GalaIO Date: Thu, 28 Mar 2024 15:58:10 +0800 Subject: [PATCH 14/15] freezer: refactor write ancient blocks logic; --- core/blockchain.go | 2 +- core/rawdb/accessors_chain.go | 51 ++++++++++++++++------------------- core/state/pruner/pruner.go | 2 +- 3 files changed, 25 insertions(+), 30 deletions(-) diff --git a/core/blockchain.go b/core/blockchain.go index e956af0fa9..c0fd15e776 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -1406,7 +1406,7 @@ func (bc *BlockChain) InsertReceiptChain(blockChain types.Blocks, receiptChain [ // Write all chain data to ancients. td := bc.GetTd(first.Hash(), first.NumberU64()) - writeSize, err := rawdb.WriteAncientBlocks(bc.db, blockChain, receiptChain, td) + writeSize, err := rawdb.WriteAncientBlocksWithBlobs(bc.db, blockChain, receiptChain, td) if err != nil { log.Error("Error importing chain data to ancients", "err", err) return 0, err diff --git a/core/rawdb/accessors_chain.go b/core/rawdb/accessors_chain.go index d06d60cc7c..baa3df7ef6 100644 --- a/core/rawdb/accessors_chain.go +++ b/core/rawdb/accessors_chain.go @@ -798,8 +798,8 @@ func WriteBlock(db ethdb.KeyValueWriter, block *types.Block) { WriteHeader(db, block.Header()) } -// WriteAncientBlocks writes entire block data into ancient store and returns the total written size. -func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts []types.Receipts, td *big.Int) (int64, error) { +// WriteAncientBlocksWithBlobs writes entire block data with blobs into ancient store and returns the total written size. +func WriteAncientBlocksWithBlobs(db ethdb.AncientWriter, blocks []*types.Block, receipts []types.Receipts, td *big.Int) (int64, error) { // find cancun index, it's used for new added blob ancient table cancunIndex := -1 for i, block := range blocks { @@ -811,34 +811,21 @@ func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts log.Info("WriteAncientBlocks", "startAt", blocks[0].Number(), "cancunIndex", cancunIndex, "len", len(blocks)) var ( - tdSum = new(big.Int).Set(td) - stReceipts []*types.ReceiptForStorage - preSize int64 - err error + tdSum = new(big.Int).Set(td) + preSize int64 + err error ) - - // handle pre-cancun blocks if cancunIndex > 0 { - preSize, err = db.ModifyAncients(func(op ethdb.AncientWriteOp) error { - for i, block := range blocks[:cancunIndex] { - // Convert receipts to storage format and sum up total difficulty. - stReceipts = stReceipts[:0] - for _, receipt := range receipts[i] { - stReceipts = append(stReceipts, (*types.ReceiptForStorage)(receipt)) - } - header := block.Header() - if i > 0 { - tdSum.Add(tdSum, header.Difficulty) - } - if err := writeAncientBlock(op, block, header, stReceipts, tdSum); err != nil { - return err - } - } - return nil - }) + preSize, err = WriteAncientBlocks(db, blocks[:cancunIndex], receipts[:cancunIndex], td) if err != nil { return preSize, err } + for i, block := range blocks[:cancunIndex] { + if i > 0 { + tdSum.Add(tdSum, block.Difficulty()) + } + } + tdSum.Add(tdSum, blocks[cancunIndex].Difficulty()) } // It will reset blob ancient table at cancunIndex @@ -847,9 +834,19 @@ func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts return 0, err } blocks = blocks[cancunIndex:] + receipts = receipts[cancunIndex:] } + postSize, err := WriteAncientBlocks(db, blocks, receipts, tdSum) + return preSize + postSize, err +} - postSize, err := db.ModifyAncients(func(op ethdb.AncientWriteOp) error { +// WriteAncientBlocks writes entire block data into ancient store and returns the total written size. +func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts []types.Receipts, td *big.Int) (int64, error) { + var ( + tdSum = new(big.Int).Set(td) + stReceipts []*types.ReceiptForStorage + ) + return db.ModifyAncients(func(op ethdb.AncientWriteOp) error { for i, block := range blocks { // Convert receipts to storage format and sum up total difficulty. stReceipts = stReceipts[:0] @@ -866,8 +863,6 @@ func WriteAncientBlocks(db ethdb.AncientWriter, blocks []*types.Block, receipts } return nil }) - - return preSize + postSize, err } // ReadBlobSidecarsRLP retrieves all the transaction blobs belonging to a block in RLP encoding. diff --git a/core/state/pruner/pruner.go b/core/state/pruner/pruner.go index 89b50036c4..834cbcd16e 100644 --- a/core/state/pruner/pruner.go +++ b/core/state/pruner/pruner.go @@ -457,7 +457,7 @@ func (p *BlockPruner) backUpOldDb(name string, cache, handles int, namespace str blobs := rawdb.ReadBlobSidecars(chainDb, blockHash, blockNumber) block = block.WithSidecars(blobs) // Write into new ancient_back db. - if _, err := rawdb.WriteAncientBlocks(frdbBack, []*types.Block{block}, []types.Receipts{receipts}, td); err != nil { + if _, err := rawdb.WriteAncientBlocksWithBlobs(frdbBack, []*types.Block{block}, []types.Receipts{receipts}, td); err != nil { log.Error("failed to write new ancient", "error", err) return err } From fda22902dcf680feae47f092cbaaa236e97542f6 Mon Sep 17 00:00:00 2001 From: zzzckck <152148891+zzzckck@users.noreply.github.com> Date: Thu, 28 Mar 2024 16:30:30 +0800 Subject: [PATCH 15/15] code: update code owner file --- .github/CODEOWNERS | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS index c11f2348b2..b78b3ce988 100644 --- a/.github/CODEOWNERS +++ b/.github/CODEOWNERS @@ -1,4 +1,3 @@ # Lines starting with '#' are comments. # Each line is a file pattern followed by one or more owners. -* @zzzckck -* @zjubfd +* @zzzckck @zjubfd