GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/pack.cc
Date: 2026-04-26 02:35:59
Exec Total Coverage
Lines: 283 307 92.2%
Branches: 202 332 60.8%

Line Branch Exec Source
1 /**
2 * This file is part of the CernVM File System.
3 */
4
5 #include "pack.h"
6
7 #include <algorithm>
8 #include <cassert>
9 #include <cstring>
10 #include <map>
11
12 #include "util/exception.h"
13 #include "util/platform.h"
14 #include "util/smalloc.h"
15 #include "util/string.h"
16
17 using namespace std; // NOLINT
18
19 namespace { // some private utility functions used by ObjectPackProducer
20
21 2684 void InitializeHeader(const int version, const int num_objects,
22 const size_t pack_size, std::string *header) {
23
1/2
✓ Branch 0 taken 2684 times.
✗ Branch 1 not taken.
2684 if (header) {
24
2/4
✓ Branch 2 taken 2684 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 2684 times.
✗ Branch 6 not taken.
2684 *header = "V" + StringifyInt(version) + "\n";
25
3/6
✓ Branch 2 taken 2684 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 2684 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 2684 times.
✗ Branch 9 not taken.
2684 *header += "S" + StringifyInt(pack_size) + "\n";
26
3/6
✓ Branch 2 taken 2684 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 2684 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 2684 times.
✗ Branch 9 not taken.
2684 *header += "N" + StringifyInt(num_objects) + "\n";
27 2684 *header += "--\n";
28 }
29 2684 }
30
31 226948 void AppendItemToHeader(ObjectPack::BucketContentType object_type,
32 const std::string &hash_str, const size_t object_size,
33 const std::string &object_name, std::string *header) {
34 // If the item type is kName, the "item_name" parameter should not be empty
35
4/6
✓ Branch 0 taken 102 times.
✓ Branch 1 taken 226846 times.
✓ Branch 2 taken 102 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 102 times.
✗ Branch 6 not taken.
226948 assert((object_type == ObjectPack::kCas)
36 || ((object_type == ObjectPack::kNamed) && (!object_name.empty())));
37
1/2
✓ Branch 2 taken 226948 times.
✗ Branch 3 not taken.
226948 std::string line_prefix = "";
38
1/2
✓ Branch 2 taken 226948 times.
✗ Branch 3 not taken.
226948 std::string line_suffix = "";
39
2/3
✓ Branch 0 taken 102 times.
✓ Branch 1 taken 226846 times.
✗ Branch 2 not taken.
226948 switch (object_type) {
40 102 case ObjectPack::kNamed:
41
1/2
✓ Branch 1 taken 102 times.
✗ Branch 2 not taken.
102 line_prefix = "N ";
42
3/6
✓ Branch 1 taken 102 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 102 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 102 times.
✗ Branch 9 not taken.
102 line_suffix = std::string(" ") + Base64Url(object_name);
43 102 break;
44 226846 case ObjectPack::kCas:
45
1/2
✓ Branch 1 taken 226846 times.
✗ Branch 2 not taken.
226846 line_prefix = "C ";
46 226846 break;
47 default:
48 PANIC(kLogStderr, "Unknown object pack type to be added to header.");
49 }
50
1/2
✓ Branch 0 taken 226948 times.
✗ Branch 1 not taken.
226948 if (header) {
51
4/8
✓ Branch 1 taken 226948 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 226948 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 226948 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 226948 times.
✗ Branch 11 not taken.
453896 *header += line_prefix + hash_str + " " + StringifyInt(object_size)
52
3/6
✓ Branch 1 taken 226948 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 226948 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 226948 times.
✗ Branch 8 not taken.
226948 + line_suffix + "\n";
53 }
54 226948 }
55
56 } // namespace
57
58 3412187 ObjectPack::Bucket::Bucket()
59 3412187 : content(reinterpret_cast<unsigned char *>(smalloc(kInitialSize)))
60 3412187 , size(0)
61 3412187 , capacity(kInitialSize)
62 3412187 , content_type(kEmpty)
63 6824374 , name() { }
64
65 11813 void ObjectPack::Bucket::Add(const void *buf, const uint64_t buf_size) {
66
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 11813 times.
11813 if (buf_size == 0)
67 return;
68
69
2/2
✓ Branch 0 taken 135123 times.
✓ Branch 1 taken 11813 times.
146936 while (size + buf_size > capacity) {
70 135123 capacity *= 2;
71 135123 content = reinterpret_cast<unsigned char *>(srealloc(content, capacity));
72 }
73 11813 memcpy(content + size, buf, buf_size);
74 11813 size += buf_size;
75 }
76
77 3412187 ObjectPack::Bucket::~Bucket() { free(content); }
78
79 //------------------------------------------------------------------------------
80
81 2025 ObjectPack::ObjectPack(const uint64_t limit) : limit_(limit), size_(0) {
82 2025 InitLock();
83 2025 }
84
85 1998 ObjectPack::~ObjectPack() {
86 3996 for (std::set<BucketHandle>::const_iterator i = open_buckets_.begin(),
87 1998 iEnd = open_buckets_.end();
88
2/2
✓ Branch 1 taken 2244 times.
✓ Branch 2 taken 1998 times.
4242 i != iEnd;
89 2244 ++i) {
90
1/2
✓ Branch 1 taken 2244 times.
✗ Branch 2 not taken.
2244 delete *i;
91 }
92
93
2/2
✓ Branch 1 taken 3409807 times.
✓ Branch 2 taken 1998 times.
3411805 for (unsigned i = 0; i < buckets_.size(); ++i)
94
1/2
✓ Branch 1 taken 3409807 times.
✗ Branch 2 not taken.
3409807 delete buckets_[i];
95 1998 pthread_mutex_destroy(lock_);
96 1998 free(lock_);
97 1998 }
98
99 2769 void ObjectPack::AddToBucket(const void *buf, const uint64_t size,
100 const ObjectPack::BucketHandle handle) {
101 2769 handle->Add(buf, size);
102 2769 }
103
104 3412153 ObjectPack::BucketHandle ObjectPack::NewBucket() {
105
2/4
✓ Branch 1 taken 3412153 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 3412153 times.
✗ Branch 5 not taken.
3412153 BucketHandle handle = new Bucket();
106
107 3412153 const MutexLockGuard mutex_guard(lock_);
108
1/2
✓ Branch 1 taken 3412153 times.
✗ Branch 2 not taken.
3412153 open_buckets_.insert(handle);
109 3412153 return handle;
110 3412153 }
111
112 /**
113 * Can only fail due to insufficient remaining space in the ObjectPack.
114 */
115 3412517 bool ObjectPack::CommitBucket(const BucketContentType type,
116 const shash::Any &id,
117 const ObjectPack::BucketHandle handle,
118 const std::string &name) {
119 3412517 handle->id = id;
120
121 3412517 handle->content_type = type;
122
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 3412517 times.
3412517 if (type == kNamed) {
123 handle->name = name;
124 }
125
126 3412517 const MutexLockGuard mutex_guard(lock_);
127
2/2
✓ Branch 1 taken 34 times.
✓ Branch 2 taken 3412483 times.
3412517 if (buckets_.size() >= kMaxObjects)
128 34 return false;
129
2/2
✓ Branch 0 taken 2676 times.
✓ Branch 1 taken 3409807 times.
3412483 if (size_ + handle->size > limit_)
130 2676 return false;
131
1/2
✓ Branch 1 taken 3409807 times.
✗ Branch 2 not taken.
3409807 open_buckets_.erase(handle);
132
1/2
✓ Branch 1 taken 3409807 times.
✗ Branch 2 not taken.
3409807 buckets_.push_back(handle);
133 3409807 size_ += handle->size;
134 3409807 return true;
135 3412517 }
136
137 102 void ObjectPack::DiscardBucket(const BucketHandle handle) {
138 102 const MutexLockGuard mutex_guard(lock_);
139
1/2
✓ Branch 1 taken 102 times.
✗ Branch 2 not taken.
102 open_buckets_.erase(handle);
140
1/2
✓ Branch 0 taken 102 times.
✗ Branch 1 not taken.
102 delete handle;
141 102 }
142
143 2025 void ObjectPack::InitLock() {
144 2025 lock_ = reinterpret_cast<pthread_mutex_t *>(smalloc(sizeof(pthread_mutex_t)));
145 2025 const int retval = pthread_mutex_init(lock_, NULL);
146
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2025 times.
2025 assert(retval == 0);
147 2025 }
148
149 /**
150 * If a commit failed, an open Bucket can be transferred to another ObjectPack
151 * with more space.
152 */
153 628 void ObjectPack::TransferBucket(const ObjectPack::BucketHandle handle,
154 ObjectPack *other) {
155 628 const MutexLockGuard mutex_guard(lock_);
156
1/2
✓ Branch 1 taken 628 times.
✗ Branch 2 not taken.
628 open_buckets_.erase(handle);
157
1/2
✓ Branch 1 taken 628 times.
✗ Branch 2 not taken.
628 other->open_buckets_.insert(handle);
158 628 }
159
160 1283985 unsigned char *ObjectPack::BucketContent(size_t idx) const {
161
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 1283985 times.
1283985 assert(idx < buckets_.size());
162 1283985 return buckets_[idx]->content;
163 }
164
165 1510831 uint64_t ObjectPack::BucketSize(size_t idx) const {
166
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 1510831 times.
1510831 assert(idx < buckets_.size());
167 1510831 return buckets_[idx]->size;
168 }
169
170 226846 const shash::Any &ObjectPack::BucketId(size_t idx) const {
171
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 226846 times.
226846 assert(idx < buckets_.size());
172 226846 return buckets_[idx]->id;
173 }
174
175 //------------------------------------------------------------------------------
176
177 /**
178 * Hash over the header. The hash algorithm needs to be provided by hash.
179 */
180 2460 void ObjectPackProducer::GetDigest(shash::Any *hash) {
181
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2460 times.
2460 assert(hash);
182 2460 shash::HashString(header_, hash);
183 2460 }
184
185 2582 ObjectPackProducer::ObjectPackProducer(ObjectPack *pack)
186 2582 : pack_(pack), big_file_(NULL), pos_(0), idx_(0), pos_in_bucket_(0) {
187 2582 const unsigned N = pack->GetNoObjects();
188 // rough guess, most likely a little too much
189
1/2
✓ Branch 1 taken 2582 times.
✗ Branch 2 not taken.
2582 header_.reserve(30 + N * (2 * shash::kMaxDigestSize + 5));
190
191
1/2
✓ Branch 2 taken 2582 times.
✗ Branch 3 not taken.
2582 InitializeHeader(2, N, pack->size(), &header_);
192
193
2/2
✓ Branch 0 taken 226846 times.
✓ Branch 1 taken 2582 times.
229428 for (unsigned i = 0; i < N; ++i) {
194
3/6
✓ Branch 2 taken 226846 times.
✗ Branch 3 not taken.
✓ Branch 7 taken 226846 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 226846 times.
✗ Branch 11 not taken.
226846 AppendItemToHeader(ObjectPack::kCas, pack->BucketId(i).ToString(true),
195 pack->BucketSize(i), "", &header_);
196 }
197 2582 }
198
199 102 ObjectPackProducer::ObjectPackProducer(const shash::Any &id, FILE *big_file,
200 102 const std::string &file_name)
201 102 : pack_(NULL), big_file_(big_file), pos_(0), idx_(0), pos_in_bucket_(0) {
202 102 const int fd = fileno(big_file_);
203
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 102 times.
102 assert(fd >= 0);
204 platform_stat64 info;
205 102 const int retval = platform_fstat(fd, &info);
206
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 102 times.
102 assert(retval == 0);
207
208
1/2
✓ Branch 1 taken 102 times.
✗ Branch 2 not taken.
102 InitializeHeader(2, 1, info.st_size, &header_);
209
210
2/4
✓ Branch 1 taken 102 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 102 times.
✗ Branch 5 not taken.
102 AppendItemToHeader(ObjectPack::kNamed, id.ToString(true), info.st_size,
211 file_name, &header_);
212
213
1/2
✓ Branch 1 taken 102 times.
✗ Branch 2 not taken.
102 rewind(big_file);
214 102 }
215
216 /**
217 * Copies as many bytes as possible into buf. If the returned number of bytes
218 * is shorter than buf_size, everything has been produced.
219 */
220 1069164 unsigned ObjectPackProducer::ProduceNext(const unsigned buf_size,
221 unsigned char *buf) {
222 1069164 const unsigned remaining_in_header = (pos_ < header_.size())
223 11184 ? (header_.size() - pos_)
224
2/2
✓ Branch 0 taken 11184 times.
✓ Branch 1 taken 1057980 times.
1080348 : 0;
225 1069164 const unsigned nbytes_header = std::min(remaining_in_header, buf_size);
226
2/2
✓ Branch 0 taken 11184 times.
✓ Branch 1 taken 1057980 times.
1069164 if (nbytes_header) {
227 11184 memcpy(buf, header_.data() + pos_, nbytes_header);
228 11184 pos_ += nbytes_header;
229 }
230
231 1069164 unsigned remaining_in_buf = buf_size - nbytes_header;
232
2/2
✓ Branch 0 taken 8602 times.
✓ Branch 1 taken 1060562 times.
1069164 if (remaining_in_buf == 0)
233 8602 return nbytes_header;
234 1060562 unsigned nbytes_payload = 0;
235
236
2/2
✓ Branch 0 taken 646 times.
✓ Branch 1 taken 1059916 times.
1060562 if (big_file_) {
237
1/2
✓ Branch 1 taken 646 times.
✗ Branch 2 not taken.
646 const size_t nbytes = fread(buf + nbytes_header, 1, remaining_in_buf,
238 big_file_);
239 646 nbytes_payload = nbytes;
240 646 pos_ += nbytes_payload;
241
2/2
✓ Branch 1 taken 1059653 times.
✓ Branch 2 taken 263 times.
1059916 } else if (idx_ < pack_->GetNoObjects()) {
242 // Copy a few buckets more
243
6/6
✓ Branch 0 taken 1286431 times.
✓ Branch 1 taken 1057207 times.
✓ Branch 3 taken 1283985 times.
✓ Branch 4 taken 2446 times.
✓ Branch 5 taken 1283985 times.
✓ Branch 6 taken 1059653 times.
2343638 while ((remaining_in_buf) > 0 && (idx_ < pack_->GetNoObjects())) {
244 1283985 const unsigned remaining_in_bucket = pack_->BucketSize(idx_)
245 1283985 - pos_in_bucket_;
246 1283985 const unsigned nbytes = std::min(remaining_in_buf, remaining_in_bucket);
247 2567970 memcpy(buf + nbytes_header + nbytes_payload,
248 1283985 pack_->BucketContent(idx_) + pos_in_bucket_, nbytes);
249
250 1283985 pos_in_bucket_ += nbytes;
251 1283985 nbytes_payload += nbytes;
252 1283985 remaining_in_buf -= nbytes;
253
2/2
✓ Branch 0 taken 226846 times.
✓ Branch 1 taken 1057139 times.
1283985 if (nbytes == remaining_in_bucket) {
254 226846 pos_in_bucket_ = 0;
255 226846 idx_++;
256 }
257 }
258 }
259
260 1060562 return nbytes_header + nbytes_payload;
261 }
262
263 //------------------------------------------------------------------------------
264
265 2494 ObjectPackConsumer::ObjectPackConsumer(const shash::Any &expected_digest,
266 2494 const unsigned expected_header_size)
267 2494 : expected_digest_(expected_digest)
268 2494 , expected_header_size_(expected_header_size)
269 2494 , pos_(0)
270 2494 , idx_(0)
271 2494 , pos_in_object_(0)
272 2494 , pos_in_accu_(0)
273 2494 , state_(ObjectPackBuild::kStateContinue)
274 2494 , size_(0) {
275 // Upper limit of 100B per entry
276
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2494 times.
2494 if (expected_header_size > (100 * ObjectPack::kMaxObjects)) {
277 state_ = ObjectPackBuild::kStateHeaderTooBig;
278 return;
279 }
280
281
1/2
✓ Branch 1 taken 2494 times.
✗ Branch 2 not taken.
2494 raw_header_.reserve(expected_header_size);
282 }
283
284 /**
285 * At the end of the function, pos_ will have progressed by buf_size (unless
286 * the buffer contains trailing garbage bytes.
287 */
288 1067612 ObjectPackBuild::State ObjectPackConsumer::ConsumeNext(
289 const unsigned buf_size, const unsigned char *buf) {
290
2/2
✓ Branch 0 taken 2278 times.
✓ Branch 1 taken 1065334 times.
1067612 if (buf_size == 0)
291 2278 return state_;
292
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1065334 times.
1065334 if (state_ == ObjectPackBuild::kStateDone) {
293 state_ = ObjectPackBuild::kStateTrailingBytes;
294 return state_;
295 }
296
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1065334 times.
1065334 if (state_ != ObjectPackBuild::kStateContinue)
297 return state_;
298
299
2/2
✓ Branch 0 taken 8308 times.
✓ Branch 1 taken 1057026 times.
1065334 const unsigned remaining_in_header = (pos_ < expected_header_size_)
300 8308 ? (expected_header_size_ - pos_)
301 : 0;
302 1065334 const unsigned nbytes_header = std::min(remaining_in_header, buf_size);
303
2/2
✓ Branch 0 taken 8308 times.
✓ Branch 1 taken 1057026 times.
1065334 if (nbytes_header) {
304
2/4
✓ Branch 2 taken 8308 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 8308 times.
✗ Branch 6 not taken.
8308 raw_header_ += string(reinterpret_cast<const char *>(buf), nbytes_header);
305 8308 pos_ += nbytes_header;
306 }
307
308
2/2
✓ Branch 0 taken 5814 times.
✓ Branch 1 taken 1059520 times.
1065334 if (pos_ < expected_header_size_)
309 5814 return ObjectPackBuild::kStateContinue;
310
311 // This condition can only be true once through the lifetime of the
312 // Consumer.
313
3/4
✓ Branch 0 taken 2494 times.
✓ Branch 1 taken 1057026 times.
✓ Branch 2 taken 2494 times.
✗ Branch 3 not taken.
1059520 if (nbytes_header && (pos_ == expected_header_size_)) {
314
1/2
✓ Branch 1 taken 2494 times.
✗ Branch 2 not taken.
2494 shash::Any digest(expected_digest_.algorithm);
315
1/2
✓ Branch 1 taken 2494 times.
✗ Branch 2 not taken.
2494 shash::HashString(raw_header_, &digest);
316
2/4
✓ Branch 1 taken 2494 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 2494 times.
2494 if (digest != expected_digest_) {
317 state_ = ObjectPackBuild::kStateCorrupt;
318 68 return state_;
319 } else {
320
1/2
✓ Branch 1 taken 2494 times.
✗ Branch 2 not taken.
2494 const bool retval = ParseHeader();
321
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2494 times.
2494 if (!retval) {
322 state_ = ObjectPackBuild::kStateBadFormat;
323 return state_;
324 }
325 // We don't need the raw string anymore
326 2494 raw_header_.clear();
327 }
328
329 // Empty pack?
330
6/6
✓ Branch 0 taken 102 times.
✓ Branch 1 taken 2392 times.
✓ Branch 3 taken 68 times.
✓ Branch 4 taken 34 times.
✓ Branch 5 taken 68 times.
✓ Branch 6 taken 2426 times.
2494 if ((buf_size == nbytes_header) && (index_.size() == 0)) {
331 68 state_ = ObjectPackBuild::kStateDone;
332 68 return state_;
333 }
334 }
335
336 1059452 const unsigned remaining_in_buf = buf_size - nbytes_header;
337 1059452 const unsigned char *payload = buf + nbytes_header;
338
1/2
✓ Branch 1 taken 1059452 times.
✗ Branch 2 not taken.
1059452 return ConsumePayload(remaining_in_buf, payload);
339 }
340
341 /**
342 * Informs listeners for small complete objects. For large objects, buffers
343 * the
344 * input into reasonably sized chunks. buf can contain both a chunk of data
345 * that needs to be added to the consumer's accumulator and a bunch of
346 * complete small objects. We use the accumulator only if necessary to avoid
347 * unnecessary memory copies.
348 */
349 1059452 ObjectPackBuild::State ObjectPackConsumer::ConsumePayload(
350 const unsigned buf_size, const unsigned char *buf) {
351 1059452 uint64_t pos_in_buf = 0;
352 2500826 while ((idx_ < index_.size())
353
8/8
✓ Branch 0 taken 2498400 times.
✓ Branch 1 taken 2426 times.
✓ Branch 2 taken 1057060 times.
✓ Branch 3 taken 1441340 times.
✓ Branch 5 taken 34 times.
✓ Branch 6 taken 1057026 times.
✓ Branch 7 taken 1441374 times.
✓ Branch 8 taken 1059452 times.
2500826 && ((pos_in_buf < buf_size) || (index_[idx_].size == 0))) {
354 // Fill the accumulator or process next small object
355 uint64_t nbytes; // How many bytes are consumed in this iteration
356 1441374 const uint64_t remaining_in_buf = buf_size - pos_in_buf;
357 1441374 const uint64_t remaining_in_object = index_[idx_].size - pos_in_object_;
358 1441374 const bool is_small_rest = remaining_in_buf < kAccuSize;
359
360 // We use the accumulator if there is already something in or if we have a
361 // small piece of data of a larger object.
362 1441374 nbytes = std::min(remaining_in_object, remaining_in_buf);
363
2/2
✓ Branch 0 taken 1096546 times.
✓ Branch 1 taken 344828 times.
1441374 if ((pos_in_accu_ > 0)
364
4/4
✓ Branch 0 taken 873630 times.
✓ Branch 1 taken 222916 times.
✓ Branch 2 taken 161500 times.
✓ Branch 3 taken 712130 times.
1096546 || ((remaining_in_buf < remaining_in_object) && is_small_rest)) {
365 506328 const uint64_t remaining_in_accu = kAccuSize - pos_in_accu_;
366 506328 nbytes = std::min(remaining_in_accu, nbytes);
367 506328 memcpy(accumulator_ + pos_in_accu_, buf + pos_in_buf, nbytes);
368 506328 pos_in_accu_ += nbytes;
369
4/4
✓ Branch 0 taken 348602 times.
✓ Branch 1 taken 157726 times.
✓ Branch 2 taken 3774 times.
✓ Branch 3 taken 344828 times.
506328 if ((pos_in_accu_ == kAccuSize) || (nbytes == remaining_in_object)) {
370
1/2
✓ Branch 1 taken 161500 times.
✗ Branch 2 not taken.
161500 NotifyListeners(ObjectPackBuild::Event(
371
1/2
✓ Branch 3 taken 161500 times.
✗ Branch 4 not taken.
161500 index_[idx_].id, index_[idx_].size, pos_in_accu_, accumulator_,
372 161500 index_[idx_].entry_type, index_[idx_].entry_name));
373 161500 pos_in_accu_ = 0;
374 }
375 506328 } else { // directly trigger listeners using buf
376
1/2
✓ Branch 1 taken 935046 times.
✗ Branch 2 not taken.
935046 NotifyListeners(ObjectPackBuild::Event(
377
1/2
✓ Branch 3 taken 935046 times.
✗ Branch 4 not taken.
935046 index_[idx_].id, index_[idx_].size, nbytes, buf + pos_in_buf,
378 935046 index_[idx_].entry_type, index_[idx_].entry_name));
379 }
380
381 1441374 pos_in_buf += nbytes;
382 1441374 pos_in_object_ += nbytes;
383
2/2
✓ Branch 0 taken 226724 times.
✓ Branch 1 taken 1214650 times.
1441374 if (nbytes == remaining_in_object) {
384 226724 idx_++;
385 226724 pos_in_object_ = 0;
386 }
387 }
388
389 1059452 pos_ += buf_size;
390
391
2/2
✓ Branch 1 taken 2426 times.
✓ Branch 2 taken 1057026 times.
1059452 if (idx_ == index_.size())
392
1/2
✓ Branch 0 taken 2426 times.
✗ Branch 1 not taken.
2426 state_ = (pos_in_buf == buf_size) ? ObjectPackBuild::kStateDone
393 : ObjectPackBuild::kStateTrailingBytes;
394 else
395 1057026 state_ = ObjectPackBuild::kStateContinue;
396 1059452 return state_;
397 }
398
399 2494 bool ObjectPackConsumer::ParseHeader() {
400 2494 map<char, string> header;
401 const unsigned char *data = reinterpret_cast<const unsigned char *>(
402 2494 raw_header_.data());
403
1/2
✓ Branch 2 taken 2494 times.
✗ Branch 3 not taken.
2494 ParseKeyvalMem(data, raw_header_.size(), &header);
404
2/5
✓ Branch 2 taken 2494 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✗ Branch 5 not taken.
✓ Branch 6 taken 2494 times.
2494 if (header.find('V') == header.end())
405 return false;
406
3/7
✓ Branch 1 taken 2494 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 2494 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✓ Branch 7 taken 2494 times.
2494 if (header['V'] != "2")
407 return false;
408
2/4
✓ Branch 1 taken 2494 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 2494 times.
✗ Branch 5 not taken.
2494 size_ = String2Uint64(header['S']);
409
2/4
✓ Branch 1 taken 2494 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 2494 times.
✗ Branch 5 not taken.
2494 const unsigned nobjects = String2Uint64(header['N']);
410
411
2/2
✓ Branch 0 taken 68 times.
✓ Branch 1 taken 2426 times.
2494 if (nobjects == 0)
412 68 return true;
413
414 // Build the object index
415 2426 const size_t separator_idx = raw_header_.find("--\n");
416
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2426 times.
2426 if (separator_idx == string::npos)
417 return false;
418 2426 unsigned index_idx = separator_idx + 3;
419
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 2426 times.
2426 if (index_idx >= raw_header_.size())
420 return false;
421
422 2426 uint64_t sum_size = 0;
423 do {
424 226724 const unsigned remaining_in_header = raw_header_.size() - index_idx;
425 453448 const string line = GetLineMem(raw_header_.data() + index_idx,
426
1/2
✓ Branch 2 taken 226724 times.
✗ Branch 3 not taken.
226724 remaining_in_header);
427
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 226724 times.
226724 if (line == "")
428 break;
429
430
1/2
✓ Branch 1 taken 226724 times.
✗ Branch 2 not taken.
226724 IndexEntry entry;
431
2/4
✓ Branch 1 taken 226724 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 226724 times.
226724 if (!ParseItem(line, &entry, &sum_size)) {
432 break;
433 }
434
435
1/2
✓ Branch 1 taken 226724 times.
✗ Branch 2 not taken.
226724 index_.push_back(entry);
436 226724 index_idx += line.size() + 1;
437
4/6
✓ Branch 1 taken 226724 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 226724 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 224298 times.
✓ Branch 8 taken 2426 times.
453448 } while (index_idx < raw_header_.size());
438
439
2/4
✓ Branch 1 taken 2426 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 2426 times.
✗ Branch 4 not taken.
2426 return (nobjects == index_.size()) && (size_ == sum_size);
440 2494 }
441
442 226724 bool ObjectPackConsumer::ParseItem(const std::string &line,
443 ObjectPackConsumer::IndexEntry *entry,
444 uint64_t *sum_size) {
445
2/4
✓ Branch 0 taken 226724 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 226724 times.
226724 if (!entry || !sum_size) {
446 return false;
447 }
448
449
2/2
✓ Branch 1 taken 226690 times.
✓ Branch 2 taken 34 times.
226724 if (line[0] == 'C') { // CAS blob
450 226690 const ObjectPack::BucketContentType entry_type = ObjectPack::kCas;
451
452 // We could use SplitString but we can have many lines so we do something
453 // more efficient here
454 226690 const size_t separator = line.find(' ', 2);
455
3/6
✓ Branch 0 taken 226690 times.
✗ Branch 1 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 226690 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 226690 times.
226690 if ((separator == string::npos) || (separator == (line.size() - 1))) {
456 return false;
457 }
458
459
2/4
✓ Branch 1 taken 226690 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 226690 times.
✗ Branch 5 not taken.
226690 const uint64_t size = String2Uint64(line.substr(separator + 1));
460 226690 *sum_size += size;
461
462 // Warning do not construct a HexPtr with an rvalue!
463 // The constructor takes the address of its argument.
464
1/2
✓ Branch 1 taken 226690 times.
✗ Branch 2 not taken.
226690 const std::string hash_string = line.substr(2, separator - 2);
465 226690 const shash::HexPtr hex_ptr(hash_string);
466
467
1/2
✓ Branch 1 taken 226690 times.
✗ Branch 2 not taken.
226690 entry->id = shash::MkFromSuffixedHexPtr(hex_ptr);
468 226690 entry->size = size;
469 226690 entry->entry_type = entry_type;
470
1/2
✓ Branch 1 taken 226690 times.
✗ Branch 2 not taken.
226690 entry->entry_name = "";
471
1/2
✓ Branch 2 taken 34 times.
✗ Branch 3 not taken.
226724 } else if (line[0] == 'N') { // Named file
472 34 const ObjectPack::BucketContentType entry_type = ObjectPack::kNamed;
473
474 // First separator, before the size field
475 34 const size_t separator1 = line.find(' ', 2);
476
3/6
✓ Branch 0 taken 34 times.
✗ Branch 1 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 34 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 34 times.
34 if ((separator1 == string::npos) || (separator1 == (line.size() - 1))) {
477 return false;
478 }
479
480 // Second separator, before the name field
481 34 const size_t separator2 = line.find(' ', separator1 + 1);
482
1/2
✓ Branch 0 taken 34 times.
✗ Branch 1 not taken.
34 if ((separator1 == 0) || (separator1 == string::npos)
483
3/6
✓ Branch 0 taken 34 times.
✗ Branch 1 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 34 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 34 times.
68 || (separator1 == (line.size() - 1))) {
484 return false;
485 }
486
487
1/2
✓ Branch 1 taken 34 times.
✗ Branch 2 not taken.
34 const uint64_t size = String2Uint64(
488
1/2
✓ Branch 1 taken 34 times.
✗ Branch 2 not taken.
68 line.substr(separator1 + 1, separator2 - separator1 - 1));
489
490 34 std::string name;
491
3/7
✓ Branch 1 taken 34 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 34 times.
✗ Branch 5 not taken.
✗ Branch 7 not taken.
✓ Branch 8 taken 34 times.
34 if (!Debase64(line.substr(separator2 + 1), &name)) {
492 return false;
493 }
494
495 34 *sum_size += size;
496
497 // Warning do not construct a HexPtr with an rvalue!
498 // The constructor takes the address of its argument.
499
1/2
✓ Branch 1 taken 34 times.
✗ Branch 2 not taken.
34 const std::string hash_string = line.substr(2, separator1 - 2);
500 34 const shash::HexPtr hex_ptr(hash_string);
501
502
1/2
✓ Branch 1 taken 34 times.
✗ Branch 2 not taken.
34 entry->id = shash::MkFromSuffixedHexPtr(hex_ptr);
503 34 entry->size = size;
504 34 entry->entry_type = entry_type;
505
1/2
✓ Branch 1 taken 34 times.
✗ Branch 2 not taken.
34 entry->entry_name = name;
506
1/2
✓ Branch 2 taken 34 times.
✗ Branch 3 not taken.
34 } else { // Error
507 return false;
508 }
509
510 226724 return true;
511 }
512