/* Copyright 2021 Erigon contributors Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ package rlp import ( "encoding/binary" "math/bits" ) // General design: // - rlp package doesn't manage memory - and Caller must ensure buffers are big enough. // - no io.Writer, because it's incompatible with binary.BigEndian functions and Writer can't be used as temporary buffer // // Composition: // - each Encode method does write to given buffer and return written len // - each Parse accept position in payload and return new position // // General rules: // - functions to calculate prefix len are fast (and pure). it's ok to call them multiple times during encoding of large object for readability. // - 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) // - functions to Parse (Decode) data - using data type as name (without any prefix): rlp.String(), rlp.List, rlp.U64(), rlp.U256() // func ListPrefixLen(dataLen int) int { if dataLen >= 56 { return 1 + (bits.Len64(uint64(dataLen))+7)/8 } return 1 } func EncodeListPrefix(dataLen int, to []byte) int { 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]) return 1 + beLen } to[0] = 192 + byte(dataLen) return 1 } func U64Len(i uint64) int { if i < 128 { return 1 } return 1 + (bits.Len64(i)+7)/8 } func EncodeU64(i uint64, to []byte) int { if i == 0 { to[0] = 128 return 1 } if i < 128 { to[0] = byte(i) // fits single byte return 1 } b := to[1:] l := 0 // writes i to b in big endian byte order, using the least number of bytes needed to represent i. switch { case i < (1 << 8): b[0] = byte(i) l = 1 case i < (1 << 16): b[0] = byte(i >> 8) b[1] = byte(i) l = 2 case i < (1 << 24): b[0] = byte(i >> 16) b[1] = byte(i >> 8) b[2] = byte(i) l = 3 case i < (1 << 32): b[0] = byte(i >> 24) b[1] = byte(i >> 16) b[2] = byte(i >> 8) b[3] = byte(i) l = 4 case i < (1 << 40): b[0] = byte(i >> 32) b[1] = byte(i >> 24) b[2] = byte(i >> 16) b[3] = byte(i >> 8) b[4] = byte(i) l = 5 case i < (1 << 48): b[0] = byte(i >> 40) b[1] = byte(i >> 32) b[2] = byte(i >> 24) b[3] = byte(i >> 16) b[4] = byte(i >> 8) b[5] = byte(i) l = 6 case i < (1 << 56): b[0] = byte(i >> 48) b[1] = byte(i >> 40) b[2] = byte(i >> 32) b[3] = byte(i >> 24) b[4] = byte(i >> 16) b[5] = byte(i >> 8) b[6] = byte(i) l = 7 default: b[0] = byte(i >> 56) b[1] = byte(i >> 48) b[2] = byte(i >> 40) b[3] = byte(i >> 32) b[4] = byte(i >> 24) b[5] = byte(i >> 16) b[6] = byte(i >> 8) b[7] = byte(i) l = 8 } to[0] = 128 + byte(l) return 1 + l } func StringLen(sLen int) int { switch { case sLen > 56: beLen := (bits.Len(uint(sLen)) + 7) / 8 return 1 + beLen + sLen case sLen == 0: return 1 case sLen == 1: return 1 + sLen default: // 1 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) return 1 + beLen + len(s) case len(s) == 0: to[0] = 128 return 1 case len(s) == 1: _ = to[1] if s[0] >= 128 { to[0] = 129 } copy(to[1:], s) return 1 + len(s) default: // 1