erigon-pulse/rlp/encodel.go

104 lines
2.7 KiB
Go
Raw Normal View History

2021-07-27 03:13:50 +00:00
package rlp
import (
"encoding/binary"
"math/bits"
)
// General design:
2021-07-27 08:44:21 +00:00
// - no io.Writer, because it's incompatible with binary.BigEndian functions and Writer can't be used as temporary buffer
// - rlp has 2 data types: List and String (bytes array), and low-level funcs are operate with this types.
// - but for convenience and performance - provided higher-level functions (for example for EncodeHash - for []byte of len 32)
2021-07-27 09:44:47 +00:00
// - rlp package doesn't manage memory - and Caller must ensure buffers are big enough.
// - each Encode method does write to given buffer and return written len
// - each Parse method does write to given buffer and return written len
2021-07-27 03:13:50 +00:00
//
// General rules:
2021-07-27 08:44:21 +00:00
// - functions to calculate prefix len are fast (and pure). it's ok to call them multiple times during encoding of large object for readability.
2021-07-27 08:47:50 +00:00
// - functions to Parse (Decode) data - using data type as name (without any prefix): rlp.String(), rlp.List, rlp.U64(), rlp.U256()
2021-07-27 08:47:33 +00:00
//
2021-07-27 03:13:50 +00:00
func ListPrefixLen(dataLen int) int {
if dataLen >= 56 {
return 1 + (bits.Len64(uint64(dataLen))+7)/8
}
return 1
}
2021-07-27 09:44:47 +00:00
func EncodeListPrefix(dataLen int, to []byte) int {
2021-07-27 03:13:50 +00:00
if dataLen >= 56 {
_ = to[9]
beLen := (bits.Len64(uint64(dataLen)) + 7) / 8
binary.BigEndian.PutUint64(to[1:], uint64(dataLen))
to[8-beLen] = 247 + byte(beLen)
copy(to, to[8-beLen:9])
2021-07-27 09:44:47 +00:00
return 1 + beLen
2021-07-27 03:13:50 +00:00
}
to[0] = 192 + byte(dataLen)
2021-07-27 09:44:47 +00:00
return 1
2021-07-27 03:13:50 +00:00
}
func U64Len(i uint64) int {
if i > 128 {
return 1 + (bits.Len64(i)+7)/8
}
return 1
}
2021-07-27 09:44:47 +00:00
func EncodeU64(i uint64, to []byte) int {
2021-07-27 03:13:50 +00:00
if i > 128 {
2021-07-27 09:44:47 +00:00
beLen := (bits.Len64(i) + 7) / 8
to[0] = 128 + byte(beLen)
2021-07-27 03:13:50 +00:00
binary.BigEndian.PutUint64(to[1:], i)
2021-07-27 09:44:47 +00:00
copy(to[1:], to[1+8-beLen:1+8])
return 1 + beLen
2021-07-27 03:13:50 +00:00
}
if i == 0 {
to[0] = 128
2021-07-27 09:44:47 +00:00
return 1
2021-07-27 03:13:50 +00:00
}
to[0] = byte(i)
2021-07-27 09:44:47 +00:00
return 1
2021-07-27 03:13:50 +00:00
}
func EncodeString(s []byte, to []byte) {
switch {
case len(s) > 56:
beLen := (bits.Len(uint(len(s))) + 7) / 8
binary.BigEndian.PutUint64(to[1:], uint64(len(s)))
_ = to[beLen+len(s)]
to[8-beLen] = byte(beLen) + 183
copy(to, to[8-beLen:9])
copy(to[1+beLen:], s)
case len(s) == 0:
to[0] = 128
case len(s) == 1:
_ = to[1]
if s[0] >= 128 {
to[0] = 129
}
copy(to[1:], s)
default: // 1<s<56
_ = to[len(s)]
to[0] = byte(len(s)) + 128
copy(to[1:], s)
}
}
2021-07-27 05:18:40 +00:00
// EncodeHash assumes that `to` buffer is already 32bytes long
2021-07-27 09:44:47 +00:00
func EncodeHash(h, to []byte) int {
2021-07-27 03:13:50 +00:00
_ = to[32] // early bounds check to guarantee safety of writes below
to[0] = 128 + 32
2021-07-27 09:44:47 +00:00
copy(to[1:33], h[:32])
return 33
}
func EncodeHashes(hashes []byte, encodeBuf []byte) int {
pos := 0
hashesLen := len(hashes) / 32 * 33
pos += EncodeListPrefix(hashesLen, encodeBuf)
for i := 0; i < len(hashes); i += 32 {
pos += EncodeHash(hashes[i:], encodeBuf[pos:])
}
return pos
2021-07-27 03:13:50 +00:00
}