GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/glue_buffer.cc
Date: 2026-06-28 02:36:10
Exec Total Coverage
Lines: 201 273 73.6%
Branches: 75 180 41.7%

Line Branch Exec Source
1 /**
2 * This file is part of the CernVM File System.
3 */
4
5 #include "glue_buffer.h"
6
7 #include <errno.h>
8 #include <poll.h>
9 #include <unistd.h>
10
11 #include <cassert>
12 #include <cstdlib>
13 #include <cstring>
14 #include <string>
15
16 #include "util/exception.h"
17 #include "util/logging.h"
18 #include "util/mutex.h"
19 #include "util/posix.h"
20 #include "util/smalloc.h"
21
22 using namespace std; // NOLINT
23
24 namespace glue {
25
26 PathStore &PathStore::operator=(const PathStore &other) {
27 if (&other == this)
28 return *this;
29
30 delete string_heap_;
31 CopyFrom(other);
32 return *this;
33 }
34
35
36 PathStore::PathStore(const PathStore &other) { CopyFrom(other); }
37
38
39 void PathStore::CopyFrom(const PathStore &other) {
40 map_ = other.map_;
41
42 string_heap_ = new StringHeap(other.string_heap_->used());
43 const shash::Md5 empty_path = map_.empty_key();
44 for (unsigned i = 0; i < map_.capacity(); ++i) {
45 if (map_.keys()[i] != empty_path) {
46 (map_.values() + i)->name = string_heap_->AddString(
47 map_.values()[i].name.length(), map_.values()[i].name.data());
48 }
49 }
50 }
51
52
53 //------------------------------------------------------------------------------
54
55
56 618 void InodeTracker::InitLock() {
57 618 lock_ = reinterpret_cast<pthread_mutex_t *>(smalloc(sizeof(pthread_mutex_t)));
58 618 const int retval = pthread_mutex_init(lock_, NULL);
59
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 618 times.
618 assert(retval == 0);
60 618 }
61
62
63 void InodeTracker::CopyFrom(const InodeTracker &other) {
64 assert(other.version_ == kVersion);
65 version_ = kVersion;
66 path_map_ = other.path_map_;
67 inode_ex_map_ = other.inode_ex_map_;
68 inode_references_ = other.inode_references_;
69 statistics_ = other.statistics_;
70 }
71
72
73
2/4
✓ Branch 2 taken 618 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 618 times.
✗ Branch 6 not taken.
618 InodeTracker::InodeTracker() {
74 618 version_ = kVersion;
75 618 InitLock();
76 618 }
77
78
79 InodeTracker::InodeTracker(const InodeTracker &other) {
80 CopyFrom(other);
81 InitLock();
82 }
83
84
85 InodeTracker &InodeTracker::operator=(const InodeTracker &other) {
86 if (&other == this)
87 return *this;
88
89 CopyFrom(other);
90 return *this;
91 }
92
93
94 616 InodeTracker::~InodeTracker() {
95 616 pthread_mutex_destroy(lock_);
96 616 free(lock_);
97 616 }
98
99
100 //------------------------------------------------------------------------------
101
102 280 DentryTracker::DentryTracker() : version_(kVersion), is_active_(true) {
103 280 pipe_terminate_[0] = pipe_terminate_[1] = -1;
104 280 cleaning_interval_ms_ = -1;
105 280 InitLock();
106 280 }
107
108
109 304 DentryTracker::~DentryTracker() {
110
2/2
✓ Branch 0 taken 41 times.
✓ Branch 1 taken 263 times.
304 if (pipe_terminate_[1] >= 0) {
111 41 char t = 'T';
112 41 WritePipe(pipe_terminate_[1], &t, 1);
113 41 pthread_join(thread_cleaner_, NULL);
114 41 ClosePipe(pipe_terminate_);
115 }
116 304 pthread_mutex_destroy(lock_);
117 304 free(lock_);
118 304 }
119
120
121 65 DentryTracker::DentryTracker(const DentryTracker &other) {
122
1/2
✓ Branch 1 taken 65 times.
✗ Branch 2 not taken.
65 CopyFrom(other);
123 65 pipe_terminate_[0] = pipe_terminate_[1] = -1;
124 65 cleaning_interval_ms_ = -1;
125 65 InitLock();
126 65 }
127
128
129 DentryTracker &DentryTracker::operator=(const DentryTracker &other) {
130 if (&other == this)
131 return *this;
132
133 Lock();
134 CopyFrom(other);
135 Unlock();
136 return *this;
137 }
138
139
140 65 void DentryTracker::CopyFrom(const DentryTracker &other) {
141
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 65 times.
65 assert(other.version_ == kVersion);
142
143 65 version_ = kVersion;
144 65 statistics_ = other.statistics_;
145 65 is_active_ = other.is_active_;
146 65 entries_ = other.entries_;
147 65 }
148
149
150 65 DentryTracker *DentryTracker::Move() {
151 65 Lock();
152
1/2
✓ Branch 2 taken 65 times.
✗ Branch 3 not taken.
65 DentryTracker *new_tracker = new DentryTracker(*this);
153 65 statistics_.num_remove += entries_.size();
154 65 entries_.Clear();
155 65 Unlock();
156 65 return new_tracker;
157 }
158
159
160 41 void DentryTracker::SpawnCleaner(unsigned interval_s) {
161
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 41 times.
41 assert(pipe_terminate_[0] == -1);
162 41 cleaning_interval_ms_ = interval_s * 1000;
163
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 41 times.
41 if (cleaning_interval_ms_ == 0)
164 cleaning_interval_ms_ = -1;
165 41 MakePipe(pipe_terminate_);
166 41 const int retval = pthread_create(&thread_cleaner_, NULL, MainCleaner, this);
167
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 41 times.
41 assert(retval == 0);
168 41 }
169
170
171 41 void *DentryTracker::MainCleaner(void *data) {
172 41 DentryTracker *tracker = reinterpret_cast<DentryTracker *>(data);
173
1/2
✓ Branch 1 taken 41 times.
✗ Branch 2 not taken.
41 LogCvmfs(kLogCvmfs, kLogDebug, "starting negative entry cache cleaner");
174
175 struct pollfd watch_term;
176 41 watch_term.fd = tracker->pipe_terminate_[0];
177 41 watch_term.events = POLLIN | POLLPRI;
178 41 int timeout_ms = tracker->cleaning_interval_ms_;
179 ;
180 41 uint64_t deadline = platform_monotonic_time() + timeout_ms / 1000;
181 while (true) {
182 164 watch_term.revents = 0;
183
1/2
✓ Branch 1 taken 164 times.
✗ Branch 2 not taken.
164 const int retval = poll(&watch_term, 1, timeout_ms);
184
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 164 times.
164 if (retval < 0) {
185 if (errno == EINTR) {
186 if (timeout_ms >= 0) {
187 const uint64_t now = platform_monotonic_time();
188 timeout_ms = (now > deadline) ? 0 : (deadline - now) * 1000;
189 }
190 123 continue;
191 }
192 abort();
193 }
194 164 timeout_ms = tracker->cleaning_interval_ms_;
195 164 deadline = platform_monotonic_time() + timeout_ms / 1000;
196
197
2/2
✓ Branch 0 taken 123 times.
✓ Branch 1 taken 41 times.
164 if (retval == 0) {
198
1/2
✓ Branch 1 taken 123 times.
✗ Branch 2 not taken.
123 LogCvmfs(kLogCvmfs, kLogDebug, "negative entry cleaner: pruning");
199
1/2
✓ Branch 1 taken 123 times.
✗ Branch 2 not taken.
123 tracker->Prune();
200 123 continue;
201 }
202
203
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 41 times.
41 assert(watch_term.revents != 0);
204
205 41 char c = 0;
206
1/2
✓ Branch 1 taken 41 times.
✗ Branch 2 not taken.
41 ReadPipe(tracker->pipe_terminate_[0], &c, 1);
207
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 41 times.
41 assert(c == 'T');
208 41 break;
209 123 }
210
1/2
✓ Branch 1 taken 41 times.
✗ Branch 2 not taken.
41 LogCvmfs(kLogCvmfs, kLogDebug, "stopping negative entry cache cleaner");
211 41 return NULL;
212 }
213
214
215 345 void DentryTracker::InitLock() {
216 345 lock_ = reinterpret_cast<pthread_mutex_t *>(smalloc(sizeof(pthread_mutex_t)));
217 345 const int retval = pthread_mutex_init(lock_, NULL);
218
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 345 times.
345 assert(retval == 0);
219 345 }
220
221
222 188 void DentryTracker::Prune() {
223 188 Lock();
224 188 DoPrune(platform_monotonic_time());
225 188 Unlock();
226 188 }
227
228
229 319 DentryTracker::Cursor DentryTracker::BeginEnumerate() {
230 319 Entry *head = NULL;
231 319 Lock();
232
1/2
✓ Branch 1 taken 319 times.
✗ Branch 2 not taken.
319 entries_.Peek(&head);
233 319 return Cursor(head);
234 }
235
236
237 10678 bool DentryTracker::NextEntry(Cursor *cursor, uint64_t *inode_parent,
238 NameString *name) {
239
2/2
✓ Branch 0 taken 219 times.
✓ Branch 1 taken 10459 times.
10678 if (cursor->head == NULL)
240 219 return false;
241
2/2
✓ Branch 1 taken 92 times.
✓ Branch 2 taken 10367 times.
10459 if (cursor->pos >= entries_.size())
242 92 return false;
243 10367 Entry *e = cursor->head + cursor->pos;
244 10367 *inode_parent = e->inode_parent;
245 10367 *name = e->name;
246 10367 cursor->pos++;
247 10367 return true;
248 }
249
250
251 319 void DentryTracker::EndEnumerate(Cursor * /* cursor */) { Unlock(); }
252
253
254 //------------------------------------------------------------------------------
255
256
257
1/2
✓ Branch 3 taken 258 times.
✗ Branch 4 not taken.
258 PageCacheTracker::PageCacheTracker() : version_(kVersion), is_active_(true) {
258
1/2
✓ Branch 1 taken 258 times.
✗ Branch 2 not taken.
258 map_.Init(16, 0, hasher_inode);
259 258 InitLock();
260 258 }
261
262
263 256 PageCacheTracker::~PageCacheTracker() {
264 256 pthread_mutex_destroy(lock_);
265 256 free(lock_);
266 256 }
267
268
269 PageCacheTracker::PageCacheTracker(const PageCacheTracker &other) {
270 CopyFrom(other);
271 InitLock();
272 }
273
274
275 PageCacheTracker &PageCacheTracker::operator=(const PageCacheTracker &other) {
276 if (&other == this)
277 return *this;
278
279 const MutexLockGuard guard(lock_);
280 CopyFrom(other);
281 return *this;
282 }
283
284
285 void PageCacheTracker::CopyFrom(const PageCacheTracker &other) {
286 assert(other.version_ == kVersion);
287
288 version_ = kVersion;
289 is_active_ = other.is_active_;
290 statistics_ = other.statistics_;
291
292 map_.Init(16, 0, hasher_inode);
293 map_ = other.map_;
294 stat_store_ = other.stat_store_;
295 }
296
297
298 258 void PageCacheTracker::InitLock() {
299 258 lock_ = reinterpret_cast<pthread_mutex_t *>(smalloc(sizeof(pthread_mutex_t)));
300 258 const int retval = pthread_mutex_init(lock_, NULL);
301
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 258 times.
258 assert(retval == 0);
302 258 }
303
304 420 PageCacheTracker::OpenDirectives PageCacheTracker::Open(
305 uint64_t inode, const shash::Any &hash, const struct stat &info)
306 {
307 420 OpenDirectives open_directives;
308 // Old behavior: always flush page cache on open
309
2/2
✓ Branch 0 taken 41 times.
✓ Branch 1 taken 379 times.
420 if (!is_active_)
310 41 return open_directives;
311
312
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 379 times.
379 if (inode != info.st_ino) {
313 PANIC(kLogStderr | kLogDebug,
314 "invalid entry on open: %" PRIu64 " with st_ino=%" PRIu64,
315 " hash=%s size=%" PRIu64, inode, info.st_ino, hash.ToString().c_str(),
316 info.st_size);
317 }
318
319 379 const MutexLockGuard guard(lock_);
320
321
1/2
✓ Branch 1 taken 379 times.
✗ Branch 2 not taken.
379 Entry entry;
322
1/2
✓ Branch 1 taken 379 times.
✗ Branch 2 not taken.
379 const bool retval = map_.Lookup(inode, &entry);
323
2/2
✓ Branch 0 taken 127 times.
✓ Branch 1 taken 252 times.
379 if (!retval) {
324 127 open_directives.keep_cache = true;
325 127 open_directives.direct_io = false;
326 127 statistics_.n_insert++;
327 127 statistics_.n_open_cached++;
328
329 127 entry.nopen = 1;
330
1/2
✓ Branch 1 taken 127 times.
✗ Branch 2 not taken.
127 entry.idx_stat = stat_store_.Add(info);
331 127 entry.hash = hash;
332
1/2
✓ Branch 1 taken 127 times.
✗ Branch 2 not taken.
127 map_.Insert(inode, entry);
333 127 return open_directives;
334 }
335
336
2/2
✓ Branch 1 taken 168 times.
✓ Branch 2 taken 84 times.
252 if (entry.hash == hash) {
337 168 open_directives.direct_io = false;
338
2/2
✓ Branch 0 taken 41 times.
✓ Branch 1 taken 127 times.
168 if (entry.nopen < 0) {
339 // The page cache is still in the transition phase and may contain old
340 // content. So trigger a flush of the cache in any case.
341 41 open_directives.keep_cache = false;
342 41 statistics_.n_open_flush++;
343 41 entry.nopen--;
344
1/2
✓ Branch 1 taken 41 times.
✗ Branch 2 not taken.
41 map_.Insert(inode, entry);
345 41 return open_directives;
346 } else {
347 127 open_directives.keep_cache = true;
348 127 statistics_.n_open_cached++;
349
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 127 times.
127 if (entry.nopen++ == 0)
350 entry.idx_stat = stat_store_.Add(info);
351
1/2
✓ Branch 1 taken 127 times.
✗ Branch 2 not taken.
127 map_.Insert(inode, entry);
352 127 return open_directives;
353 }
354 }
355
356 // Page cache mismatch and old data has still open file attached to it,
357 // circumvent the page cache entirely and use direct I/O. In this case,
358 // cvmfs_close() will _not_ call Close().
359
2/2
✓ Branch 0 taken 43 times.
✓ Branch 1 taken 41 times.
84 if (entry.nopen != 0) {
360 43 open_directives.keep_cache = true;
361 43 open_directives.direct_io = true;
362 43 statistics_.n_open_direct++;
363 43 return open_directives;
364 }
365
366 // Stale data in the page cache, start the transition phase in which newly
367 // opened files flush the page cache and re-populate it with the new hash.
368 // The first file to reach Close() will finish the transition phase and
369 // mark the new hash as committed.
370 41 open_directives.direct_io = false;
371 41 open_directives.keep_cache = false;
372 41 statistics_.n_open_flush++;
373 41 entry.hash = hash;
374
1/2
✓ Branch 1 taken 41 times.
✗ Branch 2 not taken.
41 entry.idx_stat = stat_store_.Add(info);
375 41 entry.nopen = -1;
376
1/2
✓ Branch 1 taken 41 times.
✗ Branch 2 not taken.
41 map_.Insert(inode, entry);
377 41 return open_directives;
378 379 }
379
380 PageCacheTracker::OpenDirectives PageCacheTracker::OpenDirect() {
381 OpenDirectives open_directives(true, true);
382 // Old behavior: always flush page cache on open
383 if (!is_active_)
384 return open_directives;
385
386 const MutexLockGuard guard(lock_);
387 statistics_.n_open_direct++;
388 return open_directives;
389 }
390
391 334 void PageCacheTracker::Close(uint64_t inode) {
392
2/2
✓ Branch 0 taken 41 times.
✓ Branch 1 taken 293 times.
334 if (!is_active_)
393 41 return;
394
395 293 const MutexLockGuard guard(lock_);
396
1/2
✓ Branch 1 taken 293 times.
✗ Branch 2 not taken.
293 Entry entry;
397
1/2
✓ Branch 1 taken 293 times.
✗ Branch 2 not taken.
293 bool retval = map_.Lookup(inode, &entry);
398
399 293 if (!AssertOrLog(retval, kLogCvmfs, kLogSyslogWarn | kLogDebug,
400 "PageCacheTracker::Close Race condition? "
401 "Did not find inode %lu",
402 inode)
403
3/6
✓ Branch 0 taken 293 times.
✗ Branch 1 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 293 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 293 times.
293 || !AssertOrLog(entry.nopen != 0, kLogCvmfs, kLogSyslogWarn | kLogDebug,
404 "PageCacheTracker::Close Race condition? "
405 "Inode %lu has no open entries",
406 inode)) {
407 return;
408 }
409
410 293 const int32_t old_open = entry.nopen;
411
2/2
✓ Branch 0 taken 41 times.
✓ Branch 1 taken 252 times.
293 if (entry.nopen < 0) {
412 // At this point we know that any stale data has been flushed from the
413 // cache and only data related to the currently booked content hash
414 // can be present. So clear the transition bit (sign bit).
415 41 entry.nopen = -entry.nopen;
416 }
417 293 entry.nopen--;
418
2/2
✓ Branch 0 taken 126 times.
✓ Branch 1 taken 167 times.
293 if (entry.nopen == 0) {
419 // File closed, remove struct stat information
420
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 126 times.
126 if (entry.idx_stat < 0) {
421 PANIC(kLogSyslogErr | kLogDebug,
422 "page cache tracker: missing stat entry! Entry info: inode %" PRIu64
423 " - open counter %d - hash %s",
424 inode, old_open, entry.hash.ToString().c_str());
425 }
426
1/2
✓ Branch 1 taken 126 times.
✗ Branch 2 not taken.
126 const uint64_t inode_update = stat_store_.Erase(entry.idx_stat);
427
1/2
✓ Branch 1 taken 126 times.
✗ Branch 2 not taken.
126 Entry entry_update;
428
1/2
✓ Branch 1 taken 126 times.
✗ Branch 2 not taken.
126 retval = map_.Lookup(inode_update, &entry_update);
429
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 126 times.
126 if (!retval) {
430 PANIC(kLogSyslogErr | kLogDebug,
431 "invalid inode in page cache tracker: inode %" PRIu64
432 ", replacing %" PRIu64,
433 inode_update, inode);
434 }
435 126 entry_update.idx_stat = entry.idx_stat;
436
1/2
✓ Branch 1 taken 126 times.
✗ Branch 2 not taken.
126 map_.Insert(inode_update, entry_update);
437 126 entry.idx_stat = -1;
438 }
439
1/2
✓ Branch 1 taken 293 times.
✗ Branch 2 not taken.
293 map_.Insert(inode, entry);
440
1/2
✓ Branch 1 taken 293 times.
✗ Branch 2 not taken.
293 }
441
442 82 PageCacheTracker::EvictRaii::EvictRaii(PageCacheTracker *t) : tracker_(t) {
443 82 const int retval = pthread_mutex_lock(tracker_->lock_);
444
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 82 times.
82 assert(retval == 0);
445 82 }
446
447 82 PageCacheTracker::EvictRaii::~EvictRaii() {
448 82 const int retval = pthread_mutex_unlock(tracker_->lock_);
449
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 82 times.
82 assert(retval == 0);
450 82 }
451
452 82 void PageCacheTracker::EvictRaii::Evict(uint64_t inode) {
453
2/2
✓ Branch 0 taken 41 times.
✓ Branch 1 taken 41 times.
82 if (!tracker_->is_active_)
454 41 return;
455
456 41 const bool contained_inode = tracker_->map_.Erase(inode);
457
1/2
✓ Branch 0 taken 41 times.
✗ Branch 1 not taken.
41 if (contained_inode)
458 41 tracker_->statistics_.n_remove++;
459 }
460
461 } // namespace glue
462