mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-20 09:21:11 +00:00
2b74b43678
save state save state remove old history index fix lint compress index
114 lines
2.9 KiB
Go
114 lines
2.9 KiB
Go
package ethdb
|
|
|
|
import (
|
|
"bytes"
|
|
"github.com/ledgerwatch/turbo-geth/common/changeset"
|
|
"github.com/ledgerwatch/turbo-geth/common/dbutils"
|
|
)
|
|
|
|
// Maximum length (in bytes of encoded timestamp)
|
|
const MaxTimestampLength = 8
|
|
|
|
func encodingLen8to7(b []byte) int { //nolint
|
|
return (len(b)*8 + 6) / 7
|
|
}
|
|
|
|
// Transforms b into encoding where only
|
|
// 7 bits of each byte are used to encode the bits of b
|
|
// The most significant bit is left empty, for other purposes
|
|
func encode8to7(b []byte) []byte {
|
|
// Calculate number of bytes in the output
|
|
inbytes := len(b)
|
|
outbytes := (inbytes*8 + 6) / 7
|
|
out := make([]byte, outbytes)
|
|
for i := 0; i < outbytes; i++ {
|
|
out[i] = 0x80
|
|
}
|
|
outidx := 0
|
|
for inidx := 0; inidx < inbytes; inidx++ {
|
|
bb := b[inidx]
|
|
switch inidx % 7 {
|
|
case 0:
|
|
out[outidx] |= bb >> 1
|
|
out[outidx+1] |= (bb & 0x1) << 6
|
|
case 1:
|
|
out[outidx+1] |= (bb >> 2) & 0x3f
|
|
out[outidx+2] |= (bb & 0x3) << 5
|
|
case 2:
|
|
out[outidx+2] |= (bb >> 3) & 0x1f
|
|
out[outidx+3] |= (bb & 0x7) << 4
|
|
case 3:
|
|
out[outidx+3] |= (bb >> 4) & 0xf
|
|
out[outidx+4] |= (bb & 0xf) << 3
|
|
case 4:
|
|
out[outidx+4] |= (bb >> 5) & 0x7
|
|
out[outidx+5] |= (bb & 0x1f) << 2
|
|
case 5:
|
|
out[outidx+5] |= (bb >> 6) & 0x3
|
|
out[outidx+6] |= (bb & 0x3f) << 1
|
|
case 6:
|
|
out[outidx+6] |= bb >> 7
|
|
out[outidx+7] |= bb & 0x7f
|
|
outidx += 8
|
|
}
|
|
}
|
|
return out
|
|
}
|
|
|
|
func decode7to8(b []byte) []byte {
|
|
inbytes := len(b)
|
|
outbytes := inbytes * 7 / 8
|
|
out := make([]byte, outbytes)
|
|
inidx := 0
|
|
for outidx := 0; outidx < outbytes; outidx++ {
|
|
switch outidx % 7 {
|
|
case 0:
|
|
out[outidx] = ((b[inidx] & 0x7f) << 1) | ((b[inidx+1] >> 6) & 0x1)
|
|
case 1:
|
|
out[outidx] = ((b[inidx+1] & 0x3f) << 2) | ((b[inidx+2] >> 5) & 0x3)
|
|
case 2:
|
|
out[outidx] = ((b[inidx+2] & 0x1f) << 3) | ((b[inidx+3] >> 4) & 0x7)
|
|
case 3:
|
|
out[outidx] = ((b[inidx+3] & 0xf) << 4) | ((b[inidx+4] >> 3) & 0xf)
|
|
case 4:
|
|
out[outidx] = ((b[inidx+4] & 0x7) << 5) | ((b[inidx+5] >> 2) & 0x1f)
|
|
case 5:
|
|
out[outidx] = ((b[inidx+5] & 0x3) << 6) | ((b[inidx+6] >> 1) & 0x3f)
|
|
case 6:
|
|
out[outidx] = ((b[inidx+6] & 0x1) << 7) | (b[inidx+7] & 0x7f)
|
|
inidx += 8
|
|
}
|
|
}
|
|
return out
|
|
}
|
|
|
|
// addToChangeSet is not part the AccountChangeSet API, and it is only used in the test settings.
|
|
// In the production settings, ChangeSets encodings are never modified.
|
|
// In production settings (mutation.PutS) we always first populate AccountChangeSet object,
|
|
// then encode it once, and then only work with the encoding
|
|
func addToChangeSet(hb, b []byte, key []byte, value []byte) ([]byte, error) {
|
|
var (
|
|
cs *changeset.ChangeSet
|
|
err error
|
|
)
|
|
|
|
if bytes.Equal(hb, dbutils.AccountsHistoryBucket) {
|
|
cs, err = changeset.DecodeAccounts(b)
|
|
} else {
|
|
cs, err = changeset.DecodeStorage(b)
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
err = cs.Add(key, value)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if bytes.Equal(hb, dbutils.AccountsHistoryBucket) {
|
|
return changeset.EncodeAccounts(cs)
|
|
} else {
|
|
return changeset.EncodeStorage(cs)
|
|
}
|
|
}
|