GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/tracer.cc
Date: 2025-12-21 02:39:23
Exec Total Coverage
Lines: 159 165 96.4%
Branches: 103 162 63.6%

Line Branch Exec Source
1 /**
2 * This file is part of the CernVM File System.
3 */
4
5
6 #include "tracer.h"
7
8 #include <pthread.h>
9
10 #include <cassert>
11 #include <cerrno>
12 #include <cstdio>
13 #include <cstdlib>
14 #include <cstring>
15 #include <string>
16
17 #include "util/atomic.h"
18 #include "util/concurrency.h"
19 #include "util/posix.h"
20 #include "util/string.h"
21
22 using namespace std; // NOLINT
23
24
25 630 void Tracer::Activate(const int buffer_size,
26 const int flush_threshold,
27 const string &trace_file) {
28 630 trace_file_ = trace_file;
29 630 buffer_size_ = buffer_size;
30 630 flush_threshold_ = flush_threshold;
31
3/6
✓ Branch 0 taken 630 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 630 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 630 times.
✗ Branch 5 not taken.
630 assert(buffer_size_ > 1 && flush_threshold_ >= 0
32 && flush_threshold_ < buffer_size_);
33
34
4/8
✓ Branch 0 taken 630 times.
✗ Branch 1 not taken.
✓ Branch 4 taken 188730 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 188730 times.
✓ Branch 7 taken 630 times.
✗ Branch 9 not taken.
✗ Branch 10 not taken.
189360 ring_buffer_ = new BufferEntry[buffer_size_];
35
1/2
✓ Branch 0 taken 630 times.
✗ Branch 1 not taken.
630 commit_buffer_ = new atomic_int32[buffer_size_];
36
2/2
✓ Branch 0 taken 188730 times.
✓ Branch 1 taken 630 times.
189360 for (int i = 0; i < buffer_size_; i++)
37 188730 atomic_init32(&commit_buffer_[i]);
38
39 int retval;
40 630 retval = pthread_cond_init(&sig_continue_trace_, NULL);
41 630 retval |= pthread_mutex_init(&sig_continue_trace_mutex_, NULL);
42 630 retval |= pthread_cond_init(&sig_flush_, NULL);
43 630 retval |= pthread_mutex_init(&sig_flush_mutex_, NULL);
44
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 630 times.
630 assert(retval == 0);
45
46 630 active_ = true;
47 630 }
48
49
50 /**
51 * Trace a message. This is usually a lock-free procedure that just
52 * requires two fetch_and_add operations and a gettimeofday syscall.
53 * There are two exceptions:
54 * -# If the ring buffer is full, the function blocks until the flush
55 * thread made some space. Avoid that by carefully choosing size
56 * and threshold.
57 * -# If this message reaches the threshold, the flush thread gets
58 * signaled.
59 *
60 * \param[in] event Arbitrary code, for consistency applications should use one
61 * of the TraceEvents constants. Negative codes are reserved
62 * for internal use.
63 * \param[in] id Arbitrary id, for example file name or module name which is
64 * doing the trace.
65 * \return The sequence number which was used to trace the record
66 */
67 15093180 int32_t Tracer::DoTrace(const int event,
68 const PathString &path,
69 const string &msg) {
70 15093180 const int32_t my_seq_no = atomic_xadd32(&seq_no_, 1);
71 timeval now;
72 16001865 gettimeofday(&now, NULL);
73 15937605 const int pos = my_seq_no % buffer_size_;
74
75
2/2
✓ Branch 1 taken 5050800 times.
✓ Branch 2 taken 15950565 times.
21243150 while (my_seq_no - atomic_read32(&flushed_) >= buffer_size_) {
76 timespec timeout;
77 int retval;
78 5050800 GetTimespecRel(25, &timeout);
79 4977540 retval = pthread_mutex_lock(&sig_continue_trace_mutex_);
80
1/2
✓ Branch 1 taken 5453730 times.
✗ Branch 2 not taken.
5453730 retval |= pthread_cond_timedwait(&sig_continue_trace_,
81 &sig_continue_trace_mutex_, &timeout);
82 5453730 retval |= pthread_mutex_unlock(&sig_continue_trace_mutex_);
83
2/4
✓ Branch 0 taken 5309190 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 5309190 times.
5305545 assert(retval == ETIMEDOUT || retval == 0);
84 }
85
86 15950565 ring_buffer_[pos].time_stamp = now;
87 15950565 ring_buffer_[pos].code = event;
88
1/2
✓ Branch 1 taken 15420510 times.
✗ Branch 2 not taken.
15950565 ring_buffer_[pos].path = path;
89
1/2
✓ Branch 1 taken 15525585 times.
✗ Branch 2 not taken.
15420510 ring_buffer_[pos].msg = msg;
90 15525585 atomic_inc32(&commit_buffer_[pos]);
91
92
2/2
✓ Branch 1 taken 1746720 times.
✓ Branch 2 taken 14089455 times.
16072290 if (my_seq_no - atomic_read32(&flushed_) == flush_threshold_) {
93 1746720 const MutexLockGuard m(&sig_flush_mutex_);
94 const int err_code
95 1746765 __attribute__((unused)) = pthread_cond_signal(&sig_flush_);
96
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1746765 times.
1746765 assert(err_code == 0 && "Could not signal flush thread");
97 1746765 }
98
99 15831540 return my_seq_no;
100 }
101
102
103 64710 void Tracer::Flush() {
104
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 64710 times.
64710 if (!active_)
105 return;
106
107
3/6
✓ Branch 2 taken 65160 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 65835 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 66240 times.
✗ Branch 9 not taken.
64710 const int32_t save_seq_no = DoTrace(kEventFlush, PathString("Tracer", 6),
108 "flushed ring buffer");
109
2/2
✓ Branch 1 taken 58635 times.
✓ Branch 2 taken 66240 times.
125190 while (atomic_read32(&flushed_) <= save_seq_no) {
110 timespec timeout;
111 int retval;
112
113 58635 atomic_cas32(&flush_immediately_, 0, 1);
114 {
115 59400 const MutexLockGuard m(&sig_flush_mutex_);
116 59940 retval = pthread_cond_signal(&sig_flush_);
117
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 59940 times.
59940 assert(retval == 0);
118 59940 }
119
120 59760 GetTimespecRel(250, &timeout);
121 59715 retval = pthread_mutex_lock(&sig_continue_trace_mutex_);
122
1/2
✓ Branch 1 taken 59940 times.
✗ Branch 2 not taken.
59940 retval |= pthread_cond_timedwait(&sig_continue_trace_,
123 &sig_continue_trace_mutex_, &timeout);
124 59940 retval |= pthread_mutex_unlock(&sig_continue_trace_mutex_);
125
3/4
✓ Branch 0 taken 56115 times.
✓ Branch 1 taken 3105 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 56115 times.
59220 assert(retval == ETIMEDOUT || retval == 0);
126 }
127 }
128
129
130 6422085 void Tracer::GetTimespecRel(const int64_t ms, timespec *ts) {
131 timeval now;
132 6422085 gettimeofday(&now, NULL);
133 6400170 int64_t nsecs = now.tv_usec * 1000 + (ms % 1000) * 1000 * 1000;
134 6400170 int carry = 0;
135
2/2
✓ Branch 0 taken 149220 times.
✓ Branch 1 taken 6250950 times.
6400170 if (nsecs >= 1000 * 1000 * 1000) {
136 149220 carry = 1;
137 149220 nsecs -= 1000 * 1000 * 1000;
138 }
139 6400170 ts->tv_sec = now.tv_sec + ms / 1000 + carry;
140 6400170 ts->tv_nsec = nsecs;
141 6400170 }
142
143
144 630 void *Tracer::MainFlush(void *data) {
145 630 Tracer *tracer = reinterpret_cast<Tracer *>(data);
146 int retval;
147 630 const MutexLockGuard m(&tracer->sig_flush_mutex_);
148
1/2
✓ Branch 2 taken 630 times.
✗ Branch 3 not taken.
630 FILE *f = fopen(tracer->trace_file_.c_str(), "a");
149
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 630 times.
630 assert(f != NULL && "Could not open trace file");
150 struct timespec timeout;
151
152 do {
153 5318685 while (
154 6716340 (atomic_read32(&tracer->terminate_flush_thread_) == 0)
155
2/2
✓ Branch 1 taken 6695325 times.
✓ Branch 2 taken 20385 times.
6715710 && (atomic_read32(&tracer->flush_immediately_) == 0)
156
4/4
✓ Branch 0 taken 6715710 times.
✓ Branch 1 taken 630 times.
✓ Branch 4 taken 1397655 times.
✓ Branch 5 taken 5318685 times.
20127375 && (atomic_read32(&tracer->seq_no_) - atomic_read32(&tracer->flushed_)
157
2/2
✓ Branch 0 taken 1397655 times.
✓ Branch 1 taken 5297670 times.
6695325 <= tracer->flush_threshold_)) {
158 1397655 tracer->GetTimespecRel(2000, &timeout);
159
1/2
✓ Branch 1 taken 1397655 times.
✗ Branch 2 not taken.
1397655 retval = pthread_cond_timedwait(&tracer->sig_flush_,
160 &tracer->sig_flush_mutex_, &timeout);
161
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1397655 times.
1397655 assert(retval != EINVAL);
162 }
163
164 5318685 const int base = atomic_read32(&tracer->flushed_) % tracer->buffer_size_;
165 5318685 int pos, i = 0;
166 5318685 while ((i <= tracer->flush_threshold_)
167
6/6
✓ Branch 0 taken 19623960 times.
✓ Branch 1 taken 2463030 times.
✓ Branch 2 taken 16768305 times.
✓ Branch 3 taken 2855655 times.
✓ Branch 4 taken 16768305 times.
✓ Branch 5 taken 5318685 times.
41710950 && (atomic_read32(
168 19623960 &tracer->commit_buffer_[pos = ((base + i)
169 19623960 % tracer->buffer_size_)])
170 == 1)) {
171 16768305 string tmp;
172
1/2
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
16768305 tmp = StringifyTimeval(tracer->ring_buffer_[pos].time_stamp);
173
1/2
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
16768305 retval = tracer->WriteCsvFile(f, tmp);
174
1/2
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
16768305 retval |= fputc(',', f) - ',';
175
1/2
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
16768305 tmp = StringifyInt(tracer->ring_buffer_[pos].code);
176
1/2
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
16768305 retval = tracer->WriteCsvFile(f, tmp);
177
1/2
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
16768305 retval |= fputc(',', f) - ',';
178
1/2
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
16768305 retval |= tracer->WriteCsvFile(f,
179
1/2
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
33536610 tracer->ring_buffer_[pos].path.ToString());
180
1/2
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
16768305 retval |= fputc(',', f) - ',';
181
1/2
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
16768305 retval |= tracer->WriteCsvFile(f, tracer->ring_buffer_[pos].msg);
182
2/4
✓ Branch 1 taken 16768305 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 16768305 times.
✗ Branch 5 not taken.
16768305 retval |= (fputc(13, f) - 13) | (fputc(10, f) - 10);
183
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 16768305 times.
16768305 assert(retval == 0);
184
185 16768305 atomic_dec32(&tracer->commit_buffer_[pos]);
186 16768305 ++i;
187 16768305 }
188
1/2
✓ Branch 1 taken 5318685 times.
✗ Branch 2 not taken.
5318685 retval = fflush(f);
189
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 5318685 times.
5318685 assert(retval == 0);
190 5318685 atomic_xadd32(&tracer->flushed_, i);
191 5318685 atomic_cas32(&tracer->flush_immediately_, 1, 0);
192
193 {
194 5318685 const MutexLockGuard l(&tracer->sig_continue_trace_mutex_);
195 5318685 retval = pthread_cond_broadcast(&tracer->sig_continue_trace_);
196
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 5318685 times.
5318685 assert(retval == 0);
197 5318685 }
198 } while (
199 5318685 (atomic_read32(&tracer->terminate_flush_thread_) == 0)
200
6/6
✓ Branch 0 taken 765 times.
✓ Branch 1 taken 5317920 times.
✓ Branch 4 taken 135 times.
✓ Branch 5 taken 630 times.
✓ Branch 6 taken 5318055 times.
✓ Branch 7 taken 630 times.
5318685 || (atomic_read32(&tracer->flushed_) < atomic_read32(&tracer->seq_no_)));
201
202
1/2
✓ Branch 1 taken 630 times.
✗ Branch 2 not taken.
630 retval = fclose(f);
203
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 630 times.
630 assert(retval == 0);
204 630 return NULL;
205 630 }
206
207
208 675 void Tracer::Spawn() {
209
2/2
✓ Branch 0 taken 630 times.
✓ Branch 1 taken 45 times.
675 if (active_) {
210 630 const int retval = pthread_create(&thread_flush_, NULL, MainFlush, this);
211
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 630 times.
630 assert(retval == 0);
212
213 630 spawned_ = true;
214
3/6
✓ Branch 2 taken 630 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 630 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 630 times.
✗ Branch 9 not taken.
630 DoTrace(kEventStart, PathString("Tracer", 6), "Trace buffer created");
215 }
216 675 }
217
218
219 1299 Tracer::Tracer()
220 1299 : active_(false)
221 1299 , spawned_(false)
222 1299 , buffer_size_(0)
223 1299 , flush_threshold_(0)
224 1299 , ring_buffer_(NULL)
225 1299 , commit_buffer_(NULL) {
226 1299 memset(&thread_flush_, 0, sizeof(thread_flush_));
227 1299 atomic_init32(&seq_no_);
228 1299 atomic_init32(&flushed_);
229 1299 atomic_init32(&terminate_flush_thread_);
230 1299 atomic_init32(&flush_immediately_);
231 1299 }
232
233
234 1929 Tracer::~Tracer() {
235
2/2
✓ Branch 0 taken 669 times.
✓ Branch 1 taken 630 times.
1299 if (!active_)
236 669 return;
237 int retval;
238
239
1/2
✓ Branch 0 taken 630 times.
✗ Branch 1 not taken.
630 if (spawned_) {
240 630 DoTrace(kEventStop, PathString("Tracer", 6), "Destroying trace buffer...");
241
242 // Trigger flushing and wait for it
243 630 atomic_inc32(&terminate_flush_thread_);
244 {
245 630 const MutexLockGuard m(&sig_flush_mutex_);
246 630 retval = pthread_cond_signal(&sig_flush_);
247
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 630 times.
630 assert(retval == 0);
248 630 }
249 630 retval = pthread_join(thread_flush_, NULL);
250
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 630 times.
630 assert(retval == 0);
251 }
252
253 630 retval = pthread_cond_destroy(&sig_continue_trace_);
254 630 retval |= pthread_mutex_destroy(&sig_continue_trace_mutex_);
255 630 retval |= pthread_cond_destroy(&sig_flush_);
256 630 retval |= pthread_mutex_destroy(&sig_flush_mutex_);
257
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 630 times.
630 assert(retval == 0);
258
259
3/4
✓ Branch 0 taken 630 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 188730 times.
✓ Branch 3 taken 630 times.
189360 delete[] ring_buffer_;
260
1/2
✓ Branch 0 taken 630 times.
✗ Branch 1 not taken.
630 delete[] commit_buffer_;
261
2/2
✓ Branch 1 taken 630 times.
✓ Branch 2 taken 669 times.
1299 }
262
263
264 67073220 int Tracer::WriteCsvFile(FILE *fp, const string &field) {
265
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 67073220 times.
67073220 if (fp == NULL)
266 return 0;
267
268 int retval;
269
270
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 67073220 times.
67073220 if ((retval = fputc('"', fp)) != '"')
271 return retval;
272
273
2/2
✓ Branch 1 taken 967026960 times.
✓ Branch 2 taken 67073220 times.
1034100180 for (unsigned i = 0, l = field.length(); i < l; ++i) {
274
2/2
✓ Branch 1 taken 11745000 times.
✓ Branch 2 taken 955281960 times.
967026960 if (field[i] == '"') {
275
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 11745000 times.
11745000 if ((retval = fputc('"', fp)) != '"')
276 return retval;
277 }
278
1/2
✗ Branch 3 not taken.
✓ Branch 4 taken 967026960 times.
967026960 if ((retval = fputc(field[i], fp)) != field[i])
279 return retval;
280 }
281
282
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 67073220 times.
67073220 if ((retval = fputc('"', fp)) != '"')
283 return retval;
284
285 67073220 return 0;
286 }
287