Newer
Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
/*
Copyright 2018-2019, Barcelona Supercomputing Center (BSC), Spain
Copyright 2015-2019, Johannes Gutenberg Universitaet Mainz, Germany
This software was partially supported by the
EC H2020 funded project NEXTGenIO (Project ID: 671951, www.nextgenio.eu).
This software was partially supported by the
ADA-FS project under the SPPEXA project funded by the DFG.
SPDX-License-Identifier: MIT
*/
#ifndef LIBGKFS_LOGGING_HPP
#define LIBGKFS_LOGGING_HPP
#include <libsyscall_intercept_hook_point.h>
#include <type_traits>
#include <boost/optional.hpp>
#include <client/make_array.hpp>
#include <client/syscalls.hpp>
#include <fmt/fmt.h>
#include <fmt/ostr.h>
#include <date/tz.h>
namespace gkfs {
namespace log {
enum class log_level : short {
print_syscalls = 1 << 0,
print_syscalls_entry = 1 << 1,
print_info = 1 << 2,
print_critical = 1 << 3,
print_errors = 1 << 4,
print_warnings = 1 << 5,
print_mercury = 1 << 6,
print_debug = 1 << 7,
// for internal use
print_none = 0,
print_all = print_syscalls | print_syscalls_entry | print_info |
print_critical | print_errors | print_warnings |
print_mercury | print_debug,
print_most = print_all & ~print_syscalls_entry,
print_help = 1 << 10
};
inline constexpr log_level
operator&(log_level l1, log_level l2) {
return log_level(static_cast<short>(l1) &
static_cast<short>(l2));
}
inline constexpr log_level
operator|(log_level l1, log_level l2) {
return log_level(static_cast<short>(l1) |
static_cast<short>(l2));
}
inline constexpr log_level
operator^(log_level l1, log_level l2) {
return log_level(static_cast<short>(l1) ^
static_cast<short>(l2));
}
inline constexpr log_level
operator~(log_level l1) {
return log_level(~static_cast<short>(l1));
}
inline constexpr bool
operator!(log_level dm) {
return static_cast<short>(dm) == 0;
}
inline const log_level&
operator|=(log_level& l1, log_level l2) {
return l1 = l1 | l2;
}
inline const log_level&
operator&=(log_level& l1, log_level l2) {
return l1 = l1 & l2;
}
inline const log_level&
operator^=(log_level& l1, log_level l2) {
return l1 = l1 ^ l2;
}
static const auto constexpr syscall = log_level::print_syscalls;
static const auto constexpr syscall_at_entry = log_level::print_syscalls_entry;
static const auto constexpr info = log_level::print_info;
static const auto constexpr critical = log_level::print_critical;
static const auto constexpr error = log_level::print_errors;
static const auto constexpr warning = log_level::print_warnings;
static const auto constexpr mercury = log_level::print_mercury;
static const auto constexpr debug = log_level::print_debug;
static const auto constexpr none = log_level::print_none;
static const auto constexpr most = log_level::print_most;
static const auto constexpr all = log_level::print_all;
static const auto constexpr help = log_level::print_help;
static const auto constexpr level_names =
utils::make_array(
"syscall",
"syscall", // sycall_entry uses the same name as syscall
"info",
"critical",
"error",
"warning",
"mercury",
"debug"
);
inline constexpr auto
lookup_level_name(log_level l) {
assert(l != log::none && l != log::help);
// since all log levels are powers of 2, we can find a name
// very efficiently by counting the number of trailing 0-bits in l
const auto i = __builtin_ctz(static_cast<short>(l));
assert(i >= 0 && static_cast<std::size_t>(i) < level_names.size());
return level_names.at(i);
}
// forward declaration
struct logger;
namespace detail {
enum { inline_buffer_size = 0x1000 };
using safe_buffer = fmt::basic_memory_buffer<char, inline_buffer_size>;
template <typename Buffer>
static inline void
log_buffer(std::FILE* fp,
Buffer&& buffer) {
log_buffer(::fileno(fp), std::forward<Buffer>(buffer));
}
template <typename Buffer>
static inline void
log_buffer(int fd,
Buffer&& buffer) {
if(fd < 0) {
throw std::runtime_error("Invalid file descriptor");
}
::syscall_no_intercept(SYS_write, fd, buffer.data(), buffer.size());
}
static inline void
log_buffer(int fd,
const void* buffer,
std::size_t length) {
if(fd < 0) {
throw std::runtime_error("Invalid file descriptor");
}
::syscall_no_intercept(SYS_write, fd, buffer, length);
}
/**
* format_timestamp_to - safely format a timestamp for logging messages
*
* This function produes a timestamp that can be used to prefix logging
* messages. Since we are actively intercepting system calls, the formatting
* MUST NOT rely on internal system calls, otherwise we risk recursively
* calling ourselves for each syscall generated. Also, we cannot rely on
* the C formatting functions asctime, ctime, gmtime, localtime, mktime,
* asctime_r, ctime_r, gmtime_r, localtime_r, since they acquire a
* non-reentrant lock to determine the caller's timezone (yes, the assumedly
* reentrant *_r versions of the functions exhibit this problem as well,
* see https://sourceware.org/bugzilla/show_bug.cgi?id=16145). To solve this
* issue and still get readable timestamps, we determine and cache the
* timezone when the logger is created so that the lock is only held once, by
* one thread exactly, and we pass it as an argument whenever we need to
* format a timestamp. If no timezone is provided, we just format the epoch.
*
* NOTE: we use the date C++ library to query the timezone database and
* to format the timestamps.
*/
template <typename Buffer>
static inline void
format_timestamp_to(Buffer&& buffer,
const date::time_zone * const timezone = nullptr) {
struct ::timeval tv;
int rv = ::syscall_no_intercept(SYS_gettimeofday, &tv, NULL);
if(::syscall_error_code(rv) != 0) {
return;
}
date::sys_time<std::chrono::microseconds> now{
std::chrono::seconds{tv.tv_sec} +
std::chrono::microseconds{tv.tv_usec}};
if(!timezone) {
fmt::format_to(buffer, "[{}] ", now.time_since_epoch().count());
return;
}
fmt::format_to(buffer, "[{}] ",
date::zoned_time<std::chrono::microseconds>{timezone, now});
}
template <typename Buffer>
static inline void
format_syscall_info_to(Buffer&& buffer,
gkfs::syscall::info info) {
const auto ttid = syscall_no_intercept(SYS_gettid);
fmt::format_to(buffer, "[{}] [syscall] ", ttid);
char o;
char t;
switch(gkfs::syscall::origin(info)) {
case gkfs::syscall::from_internal_code:
o = 'i';
break;
case gkfs::syscall::from_external_code:
o = 'a';
break;
default:
o = '?';
break;
}
switch(gkfs::syscall::target(info)) {
case gkfs::syscall::to_hook:
t = 'h';
break;
case gkfs::syscall::to_kernel:
t = 'k';
break;
default:
t = '?';
break;
}
const std::array<char, 5> tmp = {'[', o, t, ']', ' '};
fmt::format_to(buffer, fmt::string_view(tmp.data(), tmp.size()));
}
} // namespace detail
struct logger {
logger(const std::string& opts,
const std::string& path,
bool trunc);
~logger();
template <typename... Args>
inline void
log(log_level level,
const char * const func,
const int lineno,
Args&&... args) {
if(!(level & log_mask_)) {
return;
}
detail::safe_buffer buffer;
detail::format_timestamp_to(buffer, timezone_);
fmt::format_to(buffer, "[{}] [{}] ",
::syscall_no_intercept(SYS_gettid),
lookup_level_name(level));
if(!!(level & log::debug)) {
fmt::format_to(buffer, "<{}():{}> ", func, lineno);
}
fmt::format_to(buffer, std::forward<Args>(args)...);
fmt::format_to(buffer, "\n");
detail::log_buffer(log_fd_, buffer);
}
inline int
log(log_level level,
const char *fmt,
va_list ap) {
if(!(level & log_mask_)) {
return 0;
}
// we use buffer views to compose the logging messages to
// avoid copying buffers as much as possible
struct buffer_view {
const void* addr;
std::size_t size;
};
// helper lambda to print an iterable of buffer_views
const auto log_buffer_views =
[this](const auto& buffers) {
std::size_t n = 0;
for(const auto& bv : buffers) {
if(bv.addr != nullptr) {
detail::log_buffer(log_fd_, bv.addr, bv.size);
n += bv.size;
}
}
return n;
};
char buffer[detail::inline_buffer_size];
detail::safe_buffer prefix;
detail::format_timestamp_to(prefix);
fmt::format_to(prefix, "[{}] [{}] ",
::syscall_no_intercept(SYS_gettid),
lookup_level_name(level));
const int n = vsnprintf(buffer, sizeof(buffer), fmt, ap);
std::array<buffer_view, 3> buffers{};
int i = 0;
int m = 0;
const char* addr = buffer;
const char* p = nullptr;
while((p = std::strstr(addr, "\n")) != nullptr) {
buffers[0] = buffer_view{prefix.data(), prefix.size()};
buffers[1] = buffer_view{addr, static_cast<std::size_t>(p - addr) + 1};
m += log_buffer_views(buffers);
addr = p + 1;
++i;
}
// original line might not end with (or include) '\n'
if(buffer[n-1] != '\n') {
buffers[0] = buffer_view{prefix.data(), prefix.size()};
buffers[1] = buffer_view{addr, static_cast<std::size_t>(&buffer[n] - addr)};
buffers[2] = buffer_view{"\n", 1};
m += log_buffer_views(buffers);
}
return m;
}
template <typename... Args>
static inline void
log_message(std::FILE* fp, Args&&... args) {
log_message(::fileno(fp), std::forward<Args>(args)...);
}
template <typename... Args>
static inline void
log_message(int fd, Args&&... args) {
if(fd < 0) {
throw std::runtime_error("Invalid file descriptor");
}
detail::safe_buffer buffer;
fmt::format_to(buffer, std::forward<Args>(args)...);
fmt::format_to(buffer, "\n");
detail::log_buffer(fd, buffer);
}
void
log_syscall(syscall::info info,
const long syscall_number,
const long args[6],
boost::optional<long> result = boost::none);
static std::shared_ptr<logger>& global_logger() {
static std::shared_ptr<logger> s_global_logger;
return s_global_logger;
}
int log_fd_;
log_level log_mask_;
const date::time_zone * const timezone_;
};
// the following static functions can be used to interact
// with a globally registered logger instance
template <typename... Args>
static inline void
create_global_logger(Args&&... args) {
auto foo = std::make_shared<logger>(std::forward<Args>(args)...);
logger::global_logger() = foo;
}
static inline void
register_global_logger(logger&& lg) {
logger::global_logger() = std::make_shared<logger>(std::move(lg));
}
static inline std::shared_ptr<logger>&
get_global_logger() {
return logger::global_logger();
}
static inline void
destroy_global_logger() {
logger::global_logger().reset();
}
} // namespace log
} // namespace gkfs
#define LOG(XXX, ...) LOG_##XXX(__VA_ARGS__)
#ifdef GKFS_DISABLE_LOGGING
#define LOG_INFO(...) do {} while(0);
#define LOG_WARNING(...) do {} while(0);
#define LOG_ERROR(...) do {} while(0);
#define LOG_CRITICAL(...) do {} while(0);
#define LOG_SYSCALL(...) do {} while(0);
#define LOG_DEBUG(...) do {} while(0);
#else // !GKFS_DISABLE_LOGGING
#define LOG_INFO(...) do { \
if(gkfs::log::get_global_logger()) { \
gkfs::log::get_global_logger()->log( \
gkfs::log::info, __func__, __LINE__, __VA_ARGS__); \
} \
} while(0);
#define LOG_WARNING(...) do { \
if(gkfs::log::get_global_logger()) { \
gkfs::log::get_global_logger()->log( \
gkfs::log::warning, __func__, __LINE__, __VA_ARGS__); \
} \
} while(0);
#define LOG_ERROR(...) do { \
if(gkfs::log::get_global_logger()) { \
gkfs::log::get_global_logger()->log( \
gkfs::log::error, __func__, __LINE__, __VA_ARGS__); \
} \
} while(0);
#define LOG_CRITICAL(...) do { \
if(gkfs::log::get_global_logger()) { \
gkfs::log::get_global_logger()->log( \
gkfs::log::critical, __func__, __LINE__, __VA_ARGS__); \
} \
} while(0);
#ifdef GKFS_DEBUG_BUILD
#define LOG_SYSCALL(...) do { \
if(gkfs::log::get_global_logger()) { \
gkfs::log::get_global_logger()->log_syscall(__VA_ARGS__); \
} \
} while(0);
#define LOG_DEBUG(...) do { \
if(gkfs::log::get_global_logger()) { \
gkfs::log::get_global_logger()->log( \
gkfs::log::debug, __func__, __LINE__, __VA_ARGS__); \
} \
} while(0);
#else // ! GKFS_DEBUG_BUILD
#define LOG_SYSCALL(...) do {} while(0);
#define LOG_DEBUG(...) do {} while(0);
#endif // ! GKFS_DEBUG_BUILD
#endif // !GKFS_DISABLE_LOGGING
#endif // LIBGKFS_LOGGING_HPP