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) } }