mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-02 01:00:07 +01:00
172 lines
5.1 KiB
Go
172 lines
5.1 KiB
Go
package mergeset
|
|
|
|
import (
|
|
"fmt"
|
|
"sort"
|
|
"unsafe"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/encoding"
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/logger"
|
|
)
|
|
|
|
type blockHeader struct {
|
|
// common prefix for all the items in the block.
|
|
commonPrefix []byte
|
|
|
|
// The first item.
|
|
firstItem []byte
|
|
|
|
// Marshal type used for block compression.
|
|
marshalType marshalType
|
|
|
|
// The number of items in the block, excluding the first item.
|
|
itemsCount uint32
|
|
|
|
// The offset of the items block.
|
|
itemsBlockOffset uint64
|
|
|
|
// The offset of the lens block.
|
|
lensBlockOffset uint64
|
|
|
|
// The size of the items block.
|
|
itemsBlockSize uint32
|
|
|
|
// The size of the lens block.
|
|
lensBlockSize uint32
|
|
}
|
|
|
|
func (bh *blockHeader) SizeBytes() int {
|
|
return int(unsafe.Sizeof(*bh)) + cap(bh.commonPrefix) + cap(bh.firstItem)
|
|
}
|
|
|
|
func (bh *blockHeader) Reset() {
|
|
bh.commonPrefix = bh.commonPrefix[:0]
|
|
bh.firstItem = bh.firstItem[:0]
|
|
bh.marshalType = marshalTypePlain
|
|
bh.itemsCount = 0
|
|
bh.itemsBlockOffset = 0
|
|
bh.lensBlockOffset = 0
|
|
bh.itemsBlockSize = 0
|
|
bh.lensBlockSize = 0
|
|
}
|
|
|
|
func (bh *blockHeader) Marshal(dst []byte) []byte {
|
|
dst = encoding.MarshalBytes(dst, bh.commonPrefix)
|
|
dst = encoding.MarshalBytes(dst, bh.firstItem)
|
|
dst = append(dst, byte(bh.marshalType))
|
|
dst = encoding.MarshalUint32(dst, bh.itemsCount)
|
|
dst = encoding.MarshalUint64(dst, bh.itemsBlockOffset)
|
|
dst = encoding.MarshalUint64(dst, bh.lensBlockOffset)
|
|
dst = encoding.MarshalUint32(dst, bh.itemsBlockSize)
|
|
dst = encoding.MarshalUint32(dst, bh.lensBlockSize)
|
|
return dst
|
|
}
|
|
|
|
func (bh *blockHeader) Unmarshal(src []byte) ([]byte, error) {
|
|
// Unmarshal commonPrefix
|
|
tail, cp, err := encoding.UnmarshalBytes(src)
|
|
if err != nil {
|
|
return tail, fmt.Errorf("cannot unmarshal commonPrefix: %w", err)
|
|
}
|
|
bh.commonPrefix = append(bh.commonPrefix[:0], cp...)
|
|
src = tail
|
|
|
|
// Unmarshal firstItem
|
|
tail, fi, err := encoding.UnmarshalBytes(src)
|
|
if err != nil {
|
|
return tail, fmt.Errorf("cannot unmarshal firstItem: %w", err)
|
|
}
|
|
bh.firstItem = append(bh.firstItem[:0], fi...)
|
|
src = tail
|
|
|
|
// Unmarshal marshalType
|
|
if len(src) == 0 {
|
|
return src, fmt.Errorf("cannot unmarshal marshalType from zero bytes")
|
|
}
|
|
bh.marshalType = marshalType(src[0])
|
|
src = src[1:]
|
|
if err := checkMarshalType(bh.marshalType); err != nil {
|
|
return src, fmt.Errorf("unexpected marshalType: %w", err)
|
|
}
|
|
|
|
// Unmarshal itemsCount
|
|
if len(src) < 4 {
|
|
return src, fmt.Errorf("cannot unmarshal itemsCount from %d bytes; need at least %d bytes", len(src), 4)
|
|
}
|
|
bh.itemsCount = encoding.UnmarshalUint32(src)
|
|
src = src[4:]
|
|
|
|
// Unmarshal itemsBlockOffset
|
|
if len(src) < 8 {
|
|
return src, fmt.Errorf("cannot unmarshal itemsBlockOffset from %d bytes; neet at least %d bytes", len(src), 8)
|
|
}
|
|
bh.itemsBlockOffset = encoding.UnmarshalUint64(src)
|
|
src = src[8:]
|
|
|
|
// Unmarshal lensBlockOffset
|
|
if len(src) < 8 {
|
|
return src, fmt.Errorf("cannot unmarshal lensBlockOffset from %d bytes; need at least %d bytes", len(src), 8)
|
|
}
|
|
bh.lensBlockOffset = encoding.UnmarshalUint64(src)
|
|
src = src[8:]
|
|
|
|
// Unmarshal itemsBlockSize
|
|
if len(src) < 4 {
|
|
return src, fmt.Errorf("cannot unmarshal itemsBlockSize from %d bytes; need at least %d bytes", len(src), 4)
|
|
}
|
|
bh.itemsBlockSize = encoding.UnmarshalUint32(src)
|
|
src = src[4:]
|
|
|
|
// Unmarshal lensBlockSize
|
|
if len(src) < 4 {
|
|
return src, fmt.Errorf("cannot unmarshal lensBlockSize from %d bytes; need at least %d bytes", len(src), 4)
|
|
}
|
|
bh.lensBlockSize = encoding.UnmarshalUint32(src)
|
|
src = src[4:]
|
|
|
|
if bh.itemsCount <= 0 {
|
|
return src, fmt.Errorf("itemsCount must be bigger than 0; got %d", bh.itemsCount)
|
|
}
|
|
if bh.itemsBlockSize > 2*maxInmemoryBlockSize {
|
|
return src, fmt.Errorf("too big itemsBlockSize; got %d; cannot exceed %d", bh.itemsBlockSize, 2*maxInmemoryBlockSize)
|
|
}
|
|
if bh.lensBlockSize > 2*8*maxInmemoryBlockSize {
|
|
return src, fmt.Errorf("too big lensBlockSize; got %d; cannot exceed %d", bh.lensBlockSize, 2*8*maxInmemoryBlockSize)
|
|
}
|
|
|
|
return src, nil
|
|
}
|
|
|
|
// unmarshalBlockHeaders unmarshals all the block headers from src,
|
|
// appends them to dst and returns the appended result.
|
|
//
|
|
// Block headers must be sorted by bh.firstItem.
|
|
func unmarshalBlockHeaders(dst []blockHeader, src []byte, blockHeadersCount int) ([]blockHeader, error) {
|
|
if blockHeadersCount <= 0 {
|
|
logger.Panicf("BUG: blockHeadersCount must be greater than 0; got %d", blockHeadersCount)
|
|
}
|
|
dstLen := len(dst)
|
|
if n := dstLen + blockHeadersCount - cap(dst); n > 0 {
|
|
dst = append(dst[:cap(dst)], make([]blockHeader, n)...)
|
|
}
|
|
dst = dst[:dstLen+blockHeadersCount]
|
|
for i := 0; i < blockHeadersCount; i++ {
|
|
tail, err := dst[dstLen+i].Unmarshal(src)
|
|
if err != nil {
|
|
return dst, fmt.Errorf("cannot unmarshal block header: %w", err)
|
|
}
|
|
src = tail
|
|
}
|
|
if len(src) > 0 {
|
|
return dst, fmt.Errorf("unexpected non-zero tail left after unmarshaling %d block headers; len(tail)=%d", blockHeadersCount, len(src))
|
|
}
|
|
newBHS := dst[dstLen:]
|
|
|
|
// Verify that block headers are sorted by firstItem.
|
|
if !sort.SliceIsSorted(newBHS, func(i, j int) bool { return string(newBHS[i].firstItem) < string(newBHS[j].firstItem) }) {
|
|
return dst, fmt.Errorf("block headers must be sorted by firstItem; unmarshaled unsorted block headers: %#v", newBHS)
|
|
}
|
|
|
|
return dst, nil
|
|
}
|