2019-05-27 13:51:49 +00:00
|
|
|
package dbutils
|
|
|
|
|
|
|
|
import (
|
2019-12-06 09:02:43 +00:00
|
|
|
"bytes"
|
2019-05-27 13:51:49 +00:00
|
|
|
"encoding/binary"
|
|
|
|
|
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
|
|
|
)
|
|
|
|
|
|
|
|
// EncodeBlockNumber encodes a block number as big endian uint64
|
|
|
|
func EncodeBlockNumber(number uint64) []byte {
|
|
|
|
enc := make([]byte, 8)
|
|
|
|
binary.BigEndian.PutUint64(enc, number)
|
|
|
|
return enc
|
|
|
|
}
|
|
|
|
|
|
|
|
// headerKey = headerPrefix + num (uint64 big endian) + hash
|
|
|
|
func HeaderKey(number uint64, hash common.Hash) []byte {
|
|
|
|
return append(EncodeBlockNumber(number), hash.Bytes()...)
|
|
|
|
}
|
|
|
|
|
2019-12-06 09:02:43 +00:00
|
|
|
func IsHeaderKey(k []byte) bool {
|
|
|
|
l := common.BlockNumberLength + common.HashLength
|
|
|
|
if len(k) != l {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return !IsHeaderHashKey(k) && !IsHeaderTDKey(k)
|
|
|
|
}
|
|
|
|
|
2019-05-27 13:51:49 +00:00
|
|
|
// headerTDKey = headerPrefix + num (uint64 big endian) + hash + headerTDSuffix
|
|
|
|
func HeaderTDKey(number uint64, hash common.Hash) []byte {
|
|
|
|
return append(HeaderKey(number, hash), HeaderTDSuffix...)
|
|
|
|
}
|
|
|
|
|
2019-12-06 09:02:43 +00:00
|
|
|
func IsHeaderTDKey(k []byte) bool {
|
|
|
|
l := common.BlockNumberLength + common.HashLength + 1
|
2019-12-06 09:38:32 +00:00
|
|
|
return len(k) == l && bytes.Equal(k[l-1:], HeaderTDSuffix)
|
2019-12-06 09:02:43 +00:00
|
|
|
}
|
|
|
|
|
2019-05-27 13:51:49 +00:00
|
|
|
// headerHashKey = headerPrefix + num (uint64 big endian) + headerHashSuffix
|
|
|
|
func HeaderHashKey(number uint64) []byte {
|
|
|
|
return append(EncodeBlockNumber(number), HeaderHashSuffix...)
|
|
|
|
}
|
|
|
|
|
2020-06-10 20:07:14 +00:00
|
|
|
func CheckCanonicalKey(k []byte) bool {
|
|
|
|
return len(k) == 8+len(HeaderHashSuffix) && bytes.Equal(k[8:], HeaderHashSuffix)
|
|
|
|
}
|
|
|
|
|
2019-12-06 09:02:43 +00:00
|
|
|
func IsHeaderHashKey(k []byte) bool {
|
|
|
|
l := common.BlockNumberLength + 1
|
2019-12-06 09:38:32 +00:00
|
|
|
return len(k) == l && bytes.Equal(k[l-1:], HeaderHashSuffix)
|
2019-12-06 09:02:43 +00:00
|
|
|
}
|
|
|
|
|
2019-05-27 13:51:49 +00:00
|
|
|
// headerNumberKey = headerNumberPrefix + hash
|
|
|
|
func HeaderNumberKey(hash common.Hash) []byte {
|
|
|
|
return append(HeaderNumberPrefix, hash.Bytes()...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// blockBodyKey = blockBodyPrefix + num (uint64 big endian) + hash
|
|
|
|
func BlockBodyKey(number uint64, hash common.Hash) []byte {
|
|
|
|
return append(EncodeBlockNumber(number), hash.Bytes()...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// blockReceiptsKey = blockReceiptsPrefix + num (uint64 big endian) + hash
|
|
|
|
func BlockReceiptsKey(number uint64, hash common.Hash) []byte {
|
|
|
|
return append(EncodeBlockNumber(number), hash.Bytes()...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// txLookupKey = txLookupPrefix + hash
|
|
|
|
func TxLookupKey(hash common.Hash) []byte {
|
|
|
|
return append(TxLookupPrefix, hash.Bytes()...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// bloomBitsKey = bloomBitsPrefix + bit (uint16 big endian) + section (uint64 big endian) + hash
|
|
|
|
func BloomBitsKey(bit uint, section uint64, hash common.Hash) []byte {
|
|
|
|
key := append(make([]byte, 10), hash.Bytes()...)
|
|
|
|
|
|
|
|
binary.BigEndian.PutUint16(key[0:], uint16(bit))
|
|
|
|
binary.BigEndian.PutUint64(key[2:], section)
|
|
|
|
|
|
|
|
return key
|
|
|
|
}
|
|
|
|
|
|
|
|
// preimageKey = preimagePrefix + hash
|
|
|
|
func PreimageKey(hash common.Hash) []byte {
|
|
|
|
return append(PreimagePrefix, hash.Bytes()...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// configKey = configPrefix + hash
|
|
|
|
func ConfigKey(hash common.Hash) []byte {
|
|
|
|
return append(ConfigPrefix, hash.Bytes()...)
|
|
|
|
}
|
|
|
|
|
2019-11-07 15:51:25 +00:00
|
|
|
// AddrHash + KeyHash
|
|
|
|
// Only for trie
|
2019-05-27 13:51:49 +00:00
|
|
|
func GenerateCompositeTrieKey(addressHash common.Hash, seckey common.Hash) []byte {
|
|
|
|
compositeKey := make([]byte, 0, common.HashLength+common.HashLength)
|
|
|
|
compositeKey = append(compositeKey, addressHash[:]...)
|
|
|
|
compositeKey = append(compositeKey, seckey[:]...)
|
|
|
|
return compositeKey
|
|
|
|
}
|
2019-11-07 15:51:25 +00:00
|
|
|
|
|
|
|
// AddrHash + incarnation + KeyHash
|
|
|
|
// For contract storage
|
2019-05-27 13:51:49 +00:00
|
|
|
func GenerateCompositeStorageKey(addressHash common.Hash, incarnation uint64, seckey common.Hash) []byte {
|
|
|
|
compositeKey := make([]byte, 0, common.HashLength+8+common.HashLength)
|
2020-04-23 09:35:43 +00:00
|
|
|
compositeKey = append(compositeKey, GenerateStoragePrefix(addressHash[:], incarnation)...)
|
2019-05-27 13:51:49 +00:00
|
|
|
compositeKey = append(compositeKey, seckey[:]...)
|
|
|
|
return compositeKey
|
|
|
|
}
|
2019-12-20 12:25:40 +00:00
|
|
|
|
2020-05-15 07:52:45 +00:00
|
|
|
func ParseCompositeStorageKey(compositeKey []byte) (common.Hash, uint64, common.Hash) {
|
|
|
|
prefixLen := common.HashLength + common.IncarnationLength
|
|
|
|
addrHash, inc := ParseStoragePrefix(compositeKey[:prefixLen])
|
|
|
|
var key common.Hash
|
|
|
|
copy(key[:], compositeKey[prefixLen:prefixLen+common.HashLength])
|
|
|
|
return addrHash, inc, key
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddrHash + incarnation + KeyHash
|
|
|
|
// For contract storage (for plain state)
|
|
|
|
func PlainGenerateCompositeStorageKey(address common.Address, incarnation uint64, key common.Hash) []byte {
|
2020-06-04 18:30:28 +00:00
|
|
|
compositeKey := make([]byte, common.AddressLength+8+common.HashLength)
|
|
|
|
copy(compositeKey, address[:])
|
|
|
|
binary.BigEndian.PutUint64(compositeKey[common.AddressLength:], ^incarnation)
|
|
|
|
copy(compositeKey[common.AddressLength+8:], key[:])
|
2020-05-15 07:52:45 +00:00
|
|
|
return compositeKey
|
|
|
|
}
|
|
|
|
|
|
|
|
func PlainParseCompositeStorageKey(compositeKey []byte) (common.Address, uint64, common.Hash) {
|
|
|
|
prefixLen := common.AddressLength + common.IncarnationLength
|
|
|
|
addr, inc := PlainParseStoragePrefix(compositeKey[:prefixLen])
|
|
|
|
var key common.Hash
|
|
|
|
copy(key[:], compositeKey[prefixLen:prefixLen+common.HashLength])
|
|
|
|
return addr, inc, key
|
|
|
|
}
|
|
|
|
|
2020-04-16 07:42:25 +00:00
|
|
|
// AddrHash + incarnation + StorageHashPrefix
|
|
|
|
func GenerateCompositeStoragePrefix(addressHash []byte, incarnation uint64, storageHashPrefix []byte) []byte {
|
|
|
|
key := make([]byte, common.HashLength+8+len(storageHashPrefix))
|
|
|
|
copy(key, addressHash)
|
|
|
|
binary.BigEndian.PutUint64(key[common.HashLength:], ^incarnation)
|
|
|
|
copy(key[common.HashLength+8:], storageHashPrefix)
|
|
|
|
return key
|
|
|
|
}
|
|
|
|
|
2019-12-20 12:25:40 +00:00
|
|
|
// address hash + incarnation prefix
|
2020-04-23 09:35:43 +00:00
|
|
|
func GenerateStoragePrefix(addressHash []byte, incarnation uint64) []byte {
|
2020-03-20 11:30:14 +00:00
|
|
|
prefix := make([]byte, common.HashLength+8)
|
2020-04-23 09:35:43 +00:00
|
|
|
copy(prefix, addressHash)
|
2020-04-12 18:36:27 +00:00
|
|
|
binary.BigEndian.PutUint64(prefix[common.HashLength:], ^incarnation)
|
2019-05-27 13:51:49 +00:00
|
|
|
return prefix
|
|
|
|
}
|
|
|
|
|
2020-05-15 07:52:45 +00:00
|
|
|
// address hash + incarnation prefix (for plain state)
|
2020-06-09 05:52:38 +00:00
|
|
|
func PlainGenerateStoragePrefix(address []byte, incarnation uint64) []byte {
|
2020-05-15 07:52:45 +00:00
|
|
|
prefix := make([]byte, common.AddressLength+8)
|
2020-06-09 05:52:38 +00:00
|
|
|
copy(prefix, address)
|
2020-05-15 07:52:45 +00:00
|
|
|
binary.BigEndian.PutUint64(prefix[common.AddressLength:], ^incarnation)
|
|
|
|
return prefix
|
|
|
|
}
|
|
|
|
|
|
|
|
func PlainParseStoragePrefix(prefix []byte) (common.Address, uint64) {
|
|
|
|
var addr common.Address
|
|
|
|
copy(addr[:], prefix[:common.AddressLength])
|
|
|
|
inc := ^binary.BigEndian.Uint64(prefix[common.AddressLength : common.AddressLength+common.IncarnationLength])
|
|
|
|
return addr, inc
|
|
|
|
}
|
|
|
|
|
|
|
|
func ParseStoragePrefix(prefix []byte) (common.Hash, uint64) {
|
|
|
|
var addrHash common.Hash
|
|
|
|
copy(addrHash[:], prefix[:common.HashLength])
|
|
|
|
inc := ^binary.BigEndian.Uint64(prefix[common.HashLength : common.HashLength+common.IncarnationLength])
|
|
|
|
return addrHash, inc
|
|
|
|
}
|
|
|
|
|
2020-05-28 11:33:05 +00:00
|
|
|
func DecodeIncarnation(buf []byte) uint64 {
|
|
|
|
incarnation := binary.BigEndian.Uint64(buf)
|
|
|
|
return incarnation ^ ^uint64(0)
|
|
|
|
}
|
|
|
|
|
|
|
|
func EncodeIncarnation(incarnation uint64, buf []byte) {
|
|
|
|
binary.BigEndian.PutUint64(buf, ^incarnation)
|
|
|
|
}
|
|
|
|
|
|
|
|
func RemoveIncarnationFromKey(key []byte, out *[]byte) {
|
|
|
|
tmp := (*out)[:0]
|
|
|
|
if len(key) <= common.HashLength {
|
|
|
|
tmp = append(tmp, key...)
|
|
|
|
} else {
|
|
|
|
tmp = append(tmp, key[:common.HashLength]...)
|
|
|
|
tmp = append(tmp, key[common.HashLength+8:]...)
|
|
|
|
}
|
|
|
|
*out = tmp
|
|
|
|
}
|
2020-03-20 11:30:14 +00:00
|
|
|
|
2019-11-07 15:51:25 +00:00
|
|
|
// Key + blockNum
|
|
|
|
func CompositeKeySuffix(key []byte, timestamp uint64) (composite, encodedTS []byte) {
|
|
|
|
encodedTS = EncodeTimestamp(timestamp)
|
|
|
|
composite = make([]byte, len(key)+len(encodedTS))
|
2019-05-27 13:51:49 +00:00
|
|
|
copy(composite, key)
|
2019-11-07 15:51:25 +00:00
|
|
|
copy(composite[len(key):], encodedTS)
|
|
|
|
return composite, encodedTS
|
|
|
|
}
|