GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/file_chunk.cc
Date: 2026-06-14 02:36:34
Exec Total Coverage
Lines: 108 136 79.4%
Branches: 63 110 57.3%

Line Branch Exec Source
1 /**
2 * This file is part of the CernVM File System.
3 */
4
5
6 #include "file_chunk.h"
7
8 #include <cassert>
9
10 #include "util/murmur.hxx"
11
12 using namespace std; // NOLINT
13
14 static inline uint32_t hasher_uint64t(const uint64_t &value) {
15 return MurmurHash2(&value, sizeof(value), 0x07387a4f);
16 }
17
18
19 //------------------------------------------------------------------------------
20
21
22 420 unsigned FileChunkReflist::FindChunkIdx(const uint64_t off) {
23
2/4
✓ Branch 0 taken 420 times.
✗ Branch 1 not taken.
✓ Branch 3 taken 420 times.
✗ Branch 4 not taken.
420 assert(list && (list->size() > 0));
24 420 unsigned idx_low = 0;
25 420 unsigned idx_high = list->size() - 1;
26 420 unsigned chunk_idx = idx_high / 2;
27
2/2
✓ Branch 0 taken 714 times.
✓ Branch 1 taken 336 times.
1050 while (idx_low < idx_high) {
28
2/2
✓ Branch 2 taken 126 times.
✓ Branch 3 taken 588 times.
714 if (static_cast<uint64_t>(list->AtPtr(chunk_idx)->offset()) > off) {
29
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 126 times.
126 assert(idx_high > 0);
30 126 idx_high = chunk_idx - 1;
31 } else {
32 588 if ((chunk_idx == list->size() - 1)
33
5/6
✓ Branch 0 taken 588 times.
✗ Branch 1 not taken.
✓ Branch 4 taken 84 times.
✓ Branch 5 taken 504 times.
✓ Branch 6 taken 84 times.
✓ Branch 7 taken 504 times.
588 || (static_cast<uint64_t>(list->AtPtr(chunk_idx + 1)->offset())
34 > off)) {
35 84 break;
36 }
37 504 idx_low = chunk_idx + 1;
38 }
39 630 chunk_idx = idx_low + (idx_high - idx_low) / 2;
40 }
41 420 return chunk_idx;
42 }
43
44
45 /**
46 * Returns a consistent hash over hashes of the chunks. Used by libcvmfs.
47 */
48 42 shash::Any FileChunkReflist::HashChunkList() {
49 42 const shash::Algorithms algo = list->AtPtr(0)->content_hash().algorithm;
50
1/2
✓ Branch 1 taken 42 times.
✗ Branch 2 not taken.
42 shash::ContextPtr ctx(algo);
51 42 ctx.buffer = alloca(ctx.size);
52
1/2
✓ Branch 1 taken 42 times.
✗ Branch 2 not taken.
42 shash::Init(ctx);
53
2/2
✓ Branch 1 taken 84 times.
✓ Branch 2 taken 42 times.
126 for (unsigned i = 0; i < list->size(); ++i) {
54 84 shash::Update(
55
1/2
✓ Branch 3 taken 84 times.
✗ Branch 4 not taken.
84 list->AtPtr(i)->content_hash().digest, shash::kDigestSizes[algo], ctx);
56 }
57
1/2
✓ Branch 1 taken 42 times.
✗ Branch 2 not taken.
42 shash::Any result(algo);
58
1/2
✓ Branch 1 taken 42 times.
✗ Branch 2 not taken.
42 shash::Final(ctx, &result);
59 84 return result;
60 }
61
62
63 //------------------------------------------------------------------------------
64
65
66 62 void ChunkTables::InitLocks() {
67 62 lock = reinterpret_cast<pthread_mutex_t *>(smalloc(sizeof(pthread_mutex_t)));
68 62 const int retval = pthread_mutex_init(lock, NULL);
69
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 62 times.
62 assert(retval == 0);
70
71
2/2
✓ Branch 0 taken 7936 times.
✓ Branch 1 taken 62 times.
7998 for (unsigned i = 0; i < kNumHandleLocks; ++i) {
72 pthread_mutex_t *m = reinterpret_cast<pthread_mutex_t *>(
73 7936 smalloc(sizeof(pthread_mutex_t)));
74 7936 const int retval = pthread_mutex_init(m, NULL);
75
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 7936 times.
7936 assert(retval == 0);
76
1/2
✓ Branch 1 taken 7936 times.
✗ Branch 2 not taken.
7936 handle_locks.PushBack(m);
77 }
78 62 }
79
80
81 62 void ChunkTables::InitHashmaps() {
82 62 handle2uniqino.Init(16, 0, hasher_uint64t);
83 62 handle2fd.Init(16, 0, hasher_uint64t);
84 62 inode2chunks.Init(16, 0, hasher_uint64t);
85 62 inode2references.Init(16, 0, hasher_uint64t);
86 62 }
87
88
89
4/8
✓ Branch 2 taken 62 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 62 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 62 times.
✗ Branch 9 not taken.
✓ Branch 11 taken 62 times.
✗ Branch 12 not taken.
62 ChunkTables::ChunkTables() {
90 62 next_handle = 2;
91 62 version = kVersion;
92
1/2
✓ Branch 1 taken 62 times.
✗ Branch 2 not taken.
62 InitLocks();
93
1/2
✓ Branch 1 taken 62 times.
✗ Branch 2 not taken.
62 InitHashmaps();
94 62 }
95
96
97 62 ChunkTables::~ChunkTables() {
98 62 pthread_mutex_destroy(lock);
99 62 free(lock);
100
2/2
✓ Branch 0 taken 7936 times.
✓ Branch 1 taken 62 times.
7998 for (unsigned i = 0; i < kNumHandleLocks; ++i) {
101 7936 pthread_mutex_destroy(handle_locks.At(i));
102 7936 free(handle_locks.At(i));
103 }
104 62 }
105
106
107 ChunkTables::ChunkTables(const ChunkTables &other) {
108 version = kVersion;
109 InitLocks();
110 InitHashmaps();
111 CopyFrom(other);
112 }
113
114
115 ChunkTables &ChunkTables::operator=(const ChunkTables &other) {
116 if (&other == this)
117 return *this;
118
119 handle2uniqino.Clear();
120 handle2fd.Clear();
121 inode2chunks.Clear();
122 inode2references.Clear();
123 CopyFrom(other);
124 return *this;
125 }
126
127
128 void ChunkTables::CopyFrom(const ChunkTables &other) {
129 assert(version == other.version);
130 next_handle = other.next_handle;
131 inode2references = other.inode2references;
132 inode2chunks = other.inode2chunks;
133 handle2fd = other.handle2fd;
134 handle2uniqino = other.handle2uniqino;
135 }
136
137
138 pthread_mutex_t *ChunkTables::Handle2Lock(const uint64_t handle) const {
139 const uint32_t hash = hasher_uint64t(handle);
140 const double bucket = static_cast<double>(hash)
141 * static_cast<double>(kNumHandleLocks)
142 / static_cast<double>((uint32_t)(-1));
143 return handle_locks.At((uint32_t)bucket % kNumHandleLocks);
144 }
145
146
147 //------------------------------------------------------------------------------
148
149
150 378 SimpleChunkTables::SimpleChunkTables() {
151 378 lock_ = reinterpret_cast<pthread_mutex_t *>(smalloc(sizeof(pthread_mutex_t)));
152 378 const int retval = pthread_mutex_init(lock_, NULL);
153
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 378 times.
378 assert(retval == 0);
154 378 }
155
156
157 376 SimpleChunkTables::~SimpleChunkTables() {
158
2/2
✓ Branch 1 taken 43 times.
✓ Branch 2 taken 376 times.
419 for (unsigned i = 0; i < fd_table_.size(); ++i) {
159
1/2
✓ Branch 1 taken 43 times.
✗ Branch 2 not taken.
43 delete fd_table_[i].chunk_reflist.list;
160 }
161 376 pthread_mutex_destroy(lock_);
162 376 free(lock_);
163 376 }
164
165
166 344 int SimpleChunkTables::Add(FileChunkReflist chunks) {
167
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 344 times.
344 assert(chunks.list != NULL);
168
1/2
✓ Branch 1 taken 344 times.
✗ Branch 2 not taken.
344 OpenChunks new_entry;
169
1/2
✓ Branch 1 taken 344 times.
✗ Branch 2 not taken.
344 new_entry.chunk_reflist = chunks;
170
1/2
✓ Branch 1 taken 344 times.
✗ Branch 2 not taken.
344 new_entry.chunk_fd = new ChunkFd();
171 344 unsigned i = 0;
172 344 Lock();
173
2/2
✓ Branch 1 taken 645 times.
✓ Branch 2 taken 301 times.
946 for (; i < fd_table_.size(); ++i) {
174
2/2
✓ Branch 1 taken 43 times.
✓ Branch 2 taken 602 times.
645 if (fd_table_[i].chunk_reflist.list == NULL) {
175
1/2
✓ Branch 2 taken 43 times.
✗ Branch 3 not taken.
43 fd_table_[i] = new_entry;
176 43 Unlock();
177 43 return i;
178 }
179 }
180
1/2
✓ Branch 1 taken 301 times.
✗ Branch 2 not taken.
301 fd_table_.push_back(new_entry);
181 301 Unlock();
182 301 return i;
183 344 }
184
185
186 129 SimpleChunkTables::OpenChunks SimpleChunkTables::Get(int fd) {
187 129 OpenChunks result;
188
2/2
✓ Branch 0 taken 43 times.
✓ Branch 1 taken 86 times.
129 if (fd < 0)
189 43 return result;
190
191 86 const unsigned idx = static_cast<unsigned>(fd);
192 86 Lock();
193
2/2
✓ Branch 1 taken 43 times.
✓ Branch 2 taken 43 times.
86 if (idx < fd_table_.size())
194
1/2
✓ Branch 2 taken 43 times.
✗ Branch 3 not taken.
43 result = fd_table_[idx];
195 86 Unlock();
196 86 return result;
197 }
198
199
200 387 void SimpleChunkTables::Release(int fd) {
201
2/2
✓ Branch 0 taken 43 times.
✓ Branch 1 taken 344 times.
387 if (fd < 0)
202 43 return;
203
204 344 Lock();
205 344 const unsigned idx = static_cast<unsigned>(fd);
206
2/2
✓ Branch 1 taken 43 times.
✓ Branch 2 taken 301 times.
344 if (idx >= fd_table_.size()) {
207 43 Unlock();
208 43 return;
209 }
210
211
1/2
✓ Branch 1 taken 301 times.
✗ Branch 2 not taken.
301 delete fd_table_[idx].chunk_reflist.list;
212 301 fd_table_[idx].chunk_reflist.list = NULL;
213 301 fd_table_[idx].chunk_reflist.path.Assign("", 0);
214
1/2
✓ Branch 1 taken 301 times.
✗ Branch 2 not taken.
301 delete fd_table_[idx].chunk_fd;
215 301 fd_table_[idx].chunk_fd = NULL;
216
6/6
✓ Branch 1 taken 516 times.
✓ Branch 2 taken 43 times.
✓ Branch 4 taken 258 times.
✓ Branch 5 taken 258 times.
✓ Branch 6 taken 258 times.
✓ Branch 7 taken 301 times.
559 while (!fd_table_.empty() && (fd_table_.back().chunk_reflist.list == NULL)) {
217 258 fd_table_.pop_back();
218 }
219 301 Unlock();
220 }
221