forked from dgraph-io/badger
-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathstructs.go
111 lines (89 loc) · 2.35 KB
/
structs.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
package badger
import (
"bytes"
"encoding/binary"
"fmt"
"hash/crc32"
"github.com/pingcap/badger/y"
)
// header is used in value log as a header before Entry.
type header struct {
klen uint32
vlen uint32
ver uint64
meta byte
// umlen is the length of UserMeta
umlen byte
}
const (
headerBufSize = 18
metaNotEntryEncoded = 0
)
func (h header) Encode(out []byte) {
y.Assert(len(out) >= headerBufSize)
// Because meta can never be 0xff, so 0x00 in vlog file indicates there is not an entry.
out[0] = ^h.meta
binary.BigEndian.PutUint32(out[1:5], h.klen)
binary.BigEndian.PutUint32(out[5:9], h.vlen)
binary.BigEndian.PutUint64(out[9:17], h.ver)
out[17] = h.umlen
}
// Decodes h from buf.
func (h *header) Decode(buf []byte) {
h.meta = ^buf[0]
h.klen = binary.BigEndian.Uint32(buf[1:5])
h.vlen = binary.BigEndian.Uint32(buf[5:9])
h.ver = binary.BigEndian.Uint64(buf[9:17])
h.umlen = buf[17]
}
func isEncodedHeader(data []byte) bool {
if len(data) < 1 {
return false
}
return data[0] != metaNotEntryEncoded
}
// Entry provides Key, Value, UserMeta. This struct can be used by the user to set data.
type Entry struct {
Key y.Key
Value []byte
UserMeta []byte
meta byte
logOffset logOffset
// Fields maintained internally.
offset uint32
}
func (e *Entry) SetDelete() {
e.meta |= bitDelete
}
func (e *Entry) estimateSize() int {
return e.Key.Len() + len(e.Value) + len(e.UserMeta) + 2 // Meta, UserMeta
}
// Encodes e to buf. Returns number of bytes written.
func encodeEntry(e *Entry, buf *bytes.Buffer) (int, error) {
h := header{
klen: uint32(len(e.Key.UserKey)),
vlen: uint32(len(e.Value)),
ver: e.Key.Version,
meta: e.meta,
umlen: byte(len(e.UserMeta)),
}
var headerEnc [headerBufSize]byte
h.Encode(headerEnc[:])
hash := crc32.New(y.CastagnoliCrcTable)
buf.Write(headerEnc[:])
hash.Write(headerEnc[:])
buf.Write(e.UserMeta)
hash.Write(e.UserMeta)
buf.Write(e.Key.UserKey)
hash.Write(e.Key.UserKey)
buf.Write(e.Value)
hash.Write(e.Value)
var crcBuf [4]byte
binary.BigEndian.PutUint32(crcBuf[:], hash.Sum32())
buf.Write(crcBuf[:])
return len(headerEnc) + len(e.UserMeta) + len(e.Key.UserKey) + len(e.Value) + len(crcBuf), nil
}
func (e Entry) print(prefix string) {
fmt.Printf("%s Key: %s Meta: %d UserMeta: %v Offset: %d len(val)=%d",
prefix, e.Key, e.meta, e.UserMeta, e.offset, len(e.Value))
}