/root/bitcoin/src/leveldb/db/write_batch.cc
Line | Count | Source (jump to first uncovered line) |
1 | | // Copyright (c) 2011 The LevelDB Authors. All rights reserved. |
2 | | // Use of this source code is governed by a BSD-style license that can be |
3 | | // found in the LICENSE file. See the AUTHORS file for names of contributors. |
4 | | // |
5 | | // WriteBatch::rep_ := |
6 | | // sequence: fixed64 |
7 | | // count: fixed32 |
8 | | // data: record[count] |
9 | | // record := |
10 | | // kTypeValue varstring varstring | |
11 | | // kTypeDeletion varstring |
12 | | // varstring := |
13 | | // len: varint32 |
14 | | // data: uint8[len] |
15 | | |
16 | | #include "leveldb/write_batch.h" |
17 | | |
18 | | #include "db/dbformat.h" |
19 | | #include "db/memtable.h" |
20 | | #include "db/write_batch_internal.h" |
21 | | #include "leveldb/db.h" |
22 | | #include "util/coding.h" |
23 | | |
24 | | namespace leveldb { |
25 | | |
26 | | // WriteBatch header has an 8-byte sequence number followed by a 4-byte count. |
27 | | static const size_t kHeader = 12; |
28 | | |
29 | 0 | WriteBatch::WriteBatch() { Clear(); } |
30 | | |
31 | 0 | WriteBatch::~WriteBatch() = default; |
32 | | |
33 | 0 | WriteBatch::Handler::~Handler() = default; |
34 | | |
35 | 0 | void WriteBatch::Clear() { |
36 | 0 | rep_.clear(); |
37 | 0 | rep_.resize(kHeader); |
38 | 0 | } |
39 | | |
40 | 0 | size_t WriteBatch::ApproximateSize() const { return rep_.size(); } |
41 | | |
42 | 0 | Status WriteBatch::Iterate(Handler* handler) const { |
43 | 0 | Slice input(rep_); |
44 | 0 | if (input.size() < kHeader) { |
45 | 0 | return Status::Corruption("malformed WriteBatch (too small)"); |
46 | 0 | } |
47 | | |
48 | 0 | input.remove_prefix(kHeader); |
49 | 0 | Slice key, value; |
50 | 0 | int found = 0; |
51 | 0 | while (!input.empty()) { |
52 | 0 | found++; |
53 | 0 | char tag = input[0]; |
54 | 0 | input.remove_prefix(1); |
55 | 0 | switch (tag) { |
56 | 0 | case kTypeValue: |
57 | 0 | if (GetLengthPrefixedSlice(&input, &key) && |
58 | 0 | GetLengthPrefixedSlice(&input, &value)) { |
59 | 0 | handler->Put(key, value); |
60 | 0 | } else { |
61 | 0 | return Status::Corruption("bad WriteBatch Put"); |
62 | 0 | } |
63 | 0 | break; |
64 | 0 | case kTypeDeletion: |
65 | 0 | if (GetLengthPrefixedSlice(&input, &key)) { |
66 | 0 | handler->Delete(key); |
67 | 0 | } else { |
68 | 0 | return Status::Corruption("bad WriteBatch Delete"); |
69 | 0 | } |
70 | 0 | break; |
71 | 0 | default: |
72 | 0 | return Status::Corruption("unknown WriteBatch tag"); |
73 | 0 | } |
74 | 0 | } |
75 | 0 | if (found != WriteBatchInternal::Count(this)) { |
76 | 0 | return Status::Corruption("WriteBatch has wrong count"); |
77 | 0 | } else { |
78 | 0 | return Status::OK(); |
79 | 0 | } |
80 | 0 | } |
81 | | |
82 | 0 | int WriteBatchInternal::Count(const WriteBatch* b) { |
83 | 0 | return DecodeFixed32(b->rep_.data() + 8); |
84 | 0 | } |
85 | | |
86 | 0 | void WriteBatchInternal::SetCount(WriteBatch* b, int n) { |
87 | 0 | EncodeFixed32(&b->rep_[8], n); |
88 | 0 | } |
89 | | |
90 | 0 | SequenceNumber WriteBatchInternal::Sequence(const WriteBatch* b) { |
91 | 0 | return SequenceNumber(DecodeFixed64(b->rep_.data())); |
92 | 0 | } |
93 | | |
94 | 0 | void WriteBatchInternal::SetSequence(WriteBatch* b, SequenceNumber seq) { |
95 | 0 | EncodeFixed64(&b->rep_[0], seq); |
96 | 0 | } |
97 | | |
98 | 0 | void WriteBatch::Put(const Slice& key, const Slice& value) { |
99 | 0 | WriteBatchInternal::SetCount(this, WriteBatchInternal::Count(this) + 1); |
100 | 0 | rep_.push_back(static_cast<char>(kTypeValue)); |
101 | 0 | PutLengthPrefixedSlice(&rep_, key); |
102 | 0 | PutLengthPrefixedSlice(&rep_, value); |
103 | 0 | } |
104 | | |
105 | 0 | void WriteBatch::Delete(const Slice& key) { |
106 | 0 | WriteBatchInternal::SetCount(this, WriteBatchInternal::Count(this) + 1); |
107 | 0 | rep_.push_back(static_cast<char>(kTypeDeletion)); |
108 | 0 | PutLengthPrefixedSlice(&rep_, key); |
109 | 0 | } |
110 | | |
111 | 0 | void WriteBatch::Append(const WriteBatch& source) { |
112 | 0 | WriteBatchInternal::Append(this, &source); |
113 | 0 | } |
114 | | |
115 | | namespace { |
116 | | class MemTableInserter : public WriteBatch::Handler { |
117 | | public: |
118 | | SequenceNumber sequence_; |
119 | | MemTable* mem_; |
120 | | |
121 | 0 | void Put(const Slice& key, const Slice& value) override { |
122 | 0 | mem_->Add(sequence_, kTypeValue, key, value); |
123 | 0 | sequence_++; |
124 | 0 | } |
125 | 0 | void Delete(const Slice& key) override { |
126 | 0 | mem_->Add(sequence_, kTypeDeletion, key, Slice()); |
127 | 0 | sequence_++; |
128 | 0 | } |
129 | | }; |
130 | | } // namespace |
131 | | |
132 | 0 | Status WriteBatchInternal::InsertInto(const WriteBatch* b, MemTable* memtable) { |
133 | 0 | MemTableInserter inserter; |
134 | 0 | inserter.sequence_ = WriteBatchInternal::Sequence(b); |
135 | 0 | inserter.mem_ = memtable; |
136 | 0 | return b->Iterate(&inserter); |
137 | 0 | } |
138 | | |
139 | 0 | void WriteBatchInternal::SetContents(WriteBatch* b, const Slice& contents) { |
140 | 0 | assert(contents.size() >= kHeader); |
141 | 0 | b->rep_.assign(contents.data(), contents.size()); |
142 | 0 | } |
143 | | |
144 | 0 | void WriteBatchInternal::Append(WriteBatch* dst, const WriteBatch* src) { |
145 | 0 | SetCount(dst, Count(dst) + Count(src)); |
146 | 0 | assert(src->rep_.size() >= kHeader); |
147 | 0 | dst->rep_.append(src->rep_.data() + kHeader, src->rep_.size() - kHeader); |
148 | 0 | } |
149 | | |
150 | | } // namespace leveldb |