This change is a rewrite of the freezer code. When writing ancient chain data to the freezer, the previous version first encoded each individual item to a temporary buffer, then wrote the buffer. For small item sizes (for example, in the block hash freezer table), this strategy causes a lot of system calls for writing tiny chunks of data. It also allocated a lot of temporary []byte buffers. In the new version, we instead encode multiple items into a re-useable batch buffer, which is then written to the file all at once. This avoids performing a system call for every inserted item. To make the internal batching work, the ancient database API had to be changed. While integrating this new API in BlockChain.InsertReceiptChain, additional optimizations were also added there. Co-authored-by: Felix Lange <fjl@twurst.com>
		
			
				
	
	
		
			149 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			149 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2014 The go-ethereum Authors
 | |
| // This file is part of the go-ethereum library.
 | |
| //
 | |
| // The go-ethereum library is free software: you can redistribute it and/or modify
 | |
| // it under the terms of the GNU Lesser General Public License as published by
 | |
| // the Free Software Foundation, either version 3 of the License, or
 | |
| // (at your option) any later version.
 | |
| //
 | |
| // The go-ethereum library is distributed in the hope that it will be useful,
 | |
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | |
| // GNU Lesser General Public License for more details.
 | |
| //
 | |
| // You should have received a copy of the GNU Lesser General Public License
 | |
| // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
 | |
| 
 | |
| // Package ethdb defines the interfaces for an Ethereum data store.
 | |
| package ethdb
 | |
| 
 | |
| import "io"
 | |
| 
 | |
| // KeyValueReader wraps the Has and Get method of a backing data store.
 | |
| type KeyValueReader interface {
 | |
| 	// Has retrieves if a key is present in the key-value data store.
 | |
| 	Has(key []byte) (bool, error)
 | |
| 
 | |
| 	// Get retrieves the given key if it's present in the key-value data store.
 | |
| 	Get(key []byte) ([]byte, error)
 | |
| }
 | |
| 
 | |
| // KeyValueWriter wraps the Put method of a backing data store.
 | |
| type KeyValueWriter interface {
 | |
| 	// Put inserts the given value into the key-value data store.
 | |
| 	Put(key []byte, value []byte) error
 | |
| 
 | |
| 	// Delete removes the key from the key-value data store.
 | |
| 	Delete(key []byte) error
 | |
| }
 | |
| 
 | |
| // Stater wraps the Stat method of a backing data store.
 | |
| type Stater interface {
 | |
| 	// Stat returns a particular internal stat of the database.
 | |
| 	Stat(property string) (string, error)
 | |
| }
 | |
| 
 | |
| // Compacter wraps the Compact method of a backing data store.
 | |
| type Compacter interface {
 | |
| 	// Compact flattens the underlying data store for the given key range. In essence,
 | |
| 	// deleted and overwritten versions are discarded, and the data is rearranged to
 | |
| 	// reduce the cost of operations needed to access them.
 | |
| 	//
 | |
| 	// A nil start is treated as a key before all keys in the data store; a nil limit
 | |
| 	// is treated as a key after all keys in the data store. If both is nil then it
 | |
| 	// will compact entire data store.
 | |
| 	Compact(start []byte, limit []byte) error
 | |
| }
 | |
| 
 | |
| // KeyValueStore contains all the methods required to allow handling different
 | |
| // key-value data stores backing the high level database.
 | |
| type KeyValueStore interface {
 | |
| 	KeyValueReader
 | |
| 	KeyValueWriter
 | |
| 	Batcher
 | |
| 	Iteratee
 | |
| 	Stater
 | |
| 	Compacter
 | |
| 	io.Closer
 | |
| }
 | |
| 
 | |
| // AncientReader contains the methods required to read from immutable ancient data.
 | |
| type AncientReader interface {
 | |
| 	// HasAncient returns an indicator whether the specified data exists in the
 | |
| 	// ancient store.
 | |
| 	HasAncient(kind string, number uint64) (bool, error)
 | |
| 
 | |
| 	// Ancient retrieves an ancient binary blob from the append-only immutable files.
 | |
| 	Ancient(kind string, number uint64) ([]byte, error)
 | |
| 
 | |
| 	// ReadAncients retrieves multiple items in sequence, starting from the index 'start'.
 | |
| 	// It will return
 | |
| 	//  - at most 'count' items,
 | |
| 	//  - at least 1 item (even if exceeding the maxBytes), but will otherwise
 | |
| 	//   return as many items as fit into maxBytes.
 | |
| 	ReadAncients(kind string, start, count, maxBytes uint64) ([][]byte, error)
 | |
| 
 | |
| 	// Ancients returns the ancient item numbers in the ancient store.
 | |
| 	Ancients() (uint64, error)
 | |
| 
 | |
| 	// AncientSize returns the ancient size of the specified category.
 | |
| 	AncientSize(kind string) (uint64, error)
 | |
| }
 | |
| 
 | |
| // AncientWriter contains the methods required to write to immutable ancient data.
 | |
| type AncientWriter interface {
 | |
| 	// ModifyAncients runs a write operation on the ancient store.
 | |
| 	// If the function returns an error, any changes to the underlying store are reverted.
 | |
| 	// The integer return value is the total size of the written data.
 | |
| 	ModifyAncients(func(AncientWriteOp) error) (int64, error)
 | |
| 
 | |
| 	// TruncateAncients discards all but the first n ancient data from the ancient store.
 | |
| 	TruncateAncients(n uint64) error
 | |
| 
 | |
| 	// Sync flushes all in-memory ancient store data to disk.
 | |
| 	Sync() error
 | |
| }
 | |
| 
 | |
| // AncientWriteOp is given to the function argument of ModifyAncients.
 | |
| type AncientWriteOp interface {
 | |
| 	// Append adds an RLP-encoded item.
 | |
| 	Append(kind string, number uint64, item interface{}) error
 | |
| 
 | |
| 	// AppendRaw adds an item without RLP-encoding it.
 | |
| 	AppendRaw(kind string, number uint64, item []byte) error
 | |
| }
 | |
| 
 | |
| // Reader contains the methods required to read data from both key-value as well as
 | |
| // immutable ancient data.
 | |
| type Reader interface {
 | |
| 	KeyValueReader
 | |
| 	AncientReader
 | |
| }
 | |
| 
 | |
| // Writer contains the methods required to write data to both key-value as well as
 | |
| // immutable ancient data.
 | |
| type Writer interface {
 | |
| 	KeyValueWriter
 | |
| 	AncientWriter
 | |
| }
 | |
| 
 | |
| // AncientStore contains all the methods required to allow handling different
 | |
| // ancient data stores backing immutable chain data store.
 | |
| type AncientStore interface {
 | |
| 	AncientReader
 | |
| 	AncientWriter
 | |
| 	io.Closer
 | |
| }
 | |
| 
 | |
| // Database contains all the methods required by the high level database to not
 | |
| // only access the key-value data store but also the chain freezer.
 | |
| type Database interface {
 | |
| 	Reader
 | |
| 	Writer
 | |
| 	Batcher
 | |
| 	Iteratee
 | |
| 	Stater
 | |
| 	Compacter
 | |
| 	io.Closer
 | |
| }
 |