GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/file_chunk.cc
Date: 2026-05-10 02:36:07
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 50 unsigned FileChunkReflist::FindChunkIdx(const uint64_t off) {
23
2/4
✓ Branch 0 taken 50 times.
✗ Branch 1 not taken.
✓ Branch 3 taken 50 times.
✗ Branch 4 not taken.
50 assert(list && (list->size() > 0));
24 50 unsigned idx_low = 0;
25 50 unsigned idx_high = list->size() - 1;
26 50 unsigned chunk_idx = idx_high / 2;
27
2/2
✓ Branch 0 taken 85 times.
✓ Branch 1 taken 40 times.
125 while (idx_low < idx_high) {
28
2/2
✓ Branch 2 taken 15 times.
✓ Branch 3 taken 70 times.
85 if (static_cast<uint64_t>(list->AtPtr(chunk_idx)->offset()) > off) {
29
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 15 times.
15 assert(idx_high > 0);
30 15 idx_high = chunk_idx - 1;
31 } else {
32 70 if ((chunk_idx == list->size() - 1)
33
5/6
✓ Branch 0 taken 70 times.
✗ Branch 1 not taken.
✓ Branch 4 taken 10 times.
✓ Branch 5 taken 60 times.
✓ Branch 6 taken 10 times.
✓ Branch 7 taken 60 times.
70 || (static_cast<uint64_t>(list->AtPtr(chunk_idx + 1)->offset())
34 > off)) {
35 10 break;
36 }
37 60 idx_low = chunk_idx + 1;
38 }
39 75 chunk_idx = idx_low + (idx_high - idx_low) / 2;
40 }
41 50 return chunk_idx;
42 }
43
44
45 /**
46 * Returns a consistent hash over hashes of the chunks. Used by libcvmfs.
47 */
48 4 shash::Any FileChunkReflist::HashChunkList() {
49 4 const shash::Algorithms algo = list->AtPtr(0)->content_hash().algorithm;
50
1/2
✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
4 shash::ContextPtr ctx(algo);
51 4 ctx.buffer = alloca(ctx.size);
52
1/2
✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
4 shash::Init(ctx);
53
2/2
✓ Branch 1 taken 8 times.
✓ Branch 2 taken 4 times.
12 for (unsigned i = 0; i < list->size(); ++i) {
54 8 shash::Update(
55
1/2
✓ Branch 3 taken 8 times.
✗ Branch 4 not taken.
8 list->AtPtr(i)->content_hash().digest, shash::kDigestSizes[algo], ctx);
56 }
57
1/2
✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
4 shash::Any result(algo);
58
1/2
✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
4 shash::Final(ctx, &result);
59 8 return result;
60 }
61
62
63 //------------------------------------------------------------------------------
64
65
66 199 void ChunkTables::InitLocks() {
67 199 lock = reinterpret_cast<pthread_mutex_t *>(smalloc(sizeof(pthread_mutex_t)));
68 199 const int retval = pthread_mutex_init(lock, NULL);
69
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 199 times.
199 assert(retval == 0);
70
71
2/2
✓ Branch 0 taken 25472 times.
✓ Branch 1 taken 199 times.
25671 for (unsigned i = 0; i < kNumHandleLocks; ++i) {
72 pthread_mutex_t *m = reinterpret_cast<pthread_mutex_t *>(
73 25472 smalloc(sizeof(pthread_mutex_t)));
74 25472 const int retval = pthread_mutex_init(m, NULL);
75
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 25472 times.
25472 assert(retval == 0);
76
1/2
✓ Branch 1 taken 25472 times.
✗ Branch 2 not taken.
25472 handle_locks.PushBack(m);
77 }
78 199 }
79
80
81 199 void ChunkTables::InitHashmaps() {
82 199 handle2uniqino.Init(16, 0, hasher_uint64t);
83 199 handle2fd.Init(16, 0, hasher_uint64t);
84 199 inode2chunks.Init(16, 0, hasher_uint64t);
85 199 inode2references.Init(16, 0, hasher_uint64t);
86 199 }
87
88
89
4/8
✓ Branch 2 taken 199 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 199 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 199 times.
✗ Branch 9 not taken.
✓ Branch 11 taken 199 times.
✗ Branch 12 not taken.
199 ChunkTables::ChunkTables() {
90 199 next_handle = 2;
91 199 version = kVersion;
92
1/2
✓ Branch 1 taken 199 times.
✗ Branch 2 not taken.
199 InitLocks();
93
1/2
✓ Branch 1 taken 199 times.
✗ Branch 2 not taken.
199 InitHashmaps();
94 199 }
95
96
97 199 ChunkTables::~ChunkTables() {
98 199 pthread_mutex_destroy(lock);
99 199 free(lock);
100
2/2
✓ Branch 0 taken 25472 times.
✓ Branch 1 taken 199 times.
25671 for (unsigned i = 0; i < kNumHandleLocks; ++i) {
101 25472 pthread_mutex_destroy(handle_locks.At(i));
102 25472 free(handle_locks.At(i));
103 }
104 199 }
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 61 SimpleChunkTables::SimpleChunkTables() {
151 61 lock_ = reinterpret_cast<pthread_mutex_t *>(smalloc(sizeof(pthread_mutex_t)));
152 61 const int retval = pthread_mutex_init(lock_, NULL);
153
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 61 times.
61 assert(retval == 0);
154 61 }
155
156
157 59 SimpleChunkTables::~SimpleChunkTables() {
158
2/2
✓ Branch 1 taken 4 times.
✓ Branch 2 taken 59 times.
63 for (unsigned i = 0; i < fd_table_.size(); ++i) {
159
1/2
✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
4 delete fd_table_[i].chunk_reflist.list;
160 }
161 59 pthread_mutex_destroy(lock_);
162 59 free(lock_);
163 59 }
164
165
166 32 int SimpleChunkTables::Add(FileChunkReflist chunks) {
167
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 32 times.
32 assert(chunks.list != NULL);
168
1/2
✓ Branch 1 taken 32 times.
✗ Branch 2 not taken.
32 OpenChunks new_entry;
169
1/2
✓ Branch 1 taken 32 times.
✗ Branch 2 not taken.
32 new_entry.chunk_reflist = chunks;
170
1/2
✓ Branch 1 taken 32 times.
✗ Branch 2 not taken.
32 new_entry.chunk_fd = new ChunkFd();
171 32 unsigned i = 0;
172 32 Lock();
173
2/2
✓ Branch 1 taken 60 times.
✓ Branch 2 taken 28 times.
88 for (; i < fd_table_.size(); ++i) {
174
2/2
✓ Branch 1 taken 4 times.
✓ Branch 2 taken 56 times.
60 if (fd_table_[i].chunk_reflist.list == NULL) {
175
1/2
✓ Branch 2 taken 4 times.
✗ Branch 3 not taken.
4 fd_table_[i] = new_entry;
176 4 Unlock();
177 4 return i;
178 }
179 }
180
1/2
✓ Branch 1 taken 28 times.
✗ Branch 2 not taken.
28 fd_table_.push_back(new_entry);
181 28 Unlock();
182 28 return i;
183 32 }
184
185
186 12 SimpleChunkTables::OpenChunks SimpleChunkTables::Get(int fd) {
187 12 OpenChunks result;
188
2/2
✓ Branch 0 taken 4 times.
✓ Branch 1 taken 8 times.
12 if (fd < 0)
189 4 return result;
190
191 8 const unsigned idx = static_cast<unsigned>(fd);
192 8 Lock();
193
2/2
✓ Branch 1 taken 4 times.
✓ Branch 2 taken 4 times.
8 if (idx < fd_table_.size())
194
1/2
✓ Branch 2 taken 4 times.
✗ Branch 3 not taken.
4 result = fd_table_[idx];
195 8 Unlock();
196 8 return result;
197 }
198
199
200 36 void SimpleChunkTables::Release(int fd) {
201
2/2
✓ Branch 0 taken 4 times.
✓ Branch 1 taken 32 times.
36 if (fd < 0)
202 4 return;
203
204 32 Lock();
205 32 const unsigned idx = static_cast<unsigned>(fd);
206
2/2
✓ Branch 1 taken 4 times.
✓ Branch 2 taken 28 times.
32 if (idx >= fd_table_.size()) {
207 4 Unlock();
208 4 return;
209 }
210
211
1/2
✓ Branch 1 taken 28 times.
✗ Branch 2 not taken.
28 delete fd_table_[idx].chunk_reflist.list;
212 28 fd_table_[idx].chunk_reflist.list = NULL;
213 28 fd_table_[idx].chunk_reflist.path.Assign("", 0);
214
1/2
✓ Branch 1 taken 28 times.
✗ Branch 2 not taken.
28 delete fd_table_[idx].chunk_fd;
215 28 fd_table_[idx].chunk_fd = NULL;
216
6/6
✓ Branch 1 taken 48 times.
✓ Branch 2 taken 4 times.
✓ Branch 4 taken 24 times.
✓ Branch 5 taken 24 times.
✓ Branch 6 taken 24 times.
✓ Branch 7 taken 28 times.
52 while (!fd_table_.empty() && (fd_table_.back().chunk_reflist.list == NULL)) {
217 24 fd_table_.pop_back();
218 }
219 28 Unlock();
220 }
221