11#ifndef O2_FRAMEWORK_SIGNPOST_H_
12#define O2_FRAMEWORK_SIGNPOST_H_
22 char const*
name =
nullptr;
32 std::array<char, N>
res = {};
35 for (
int i = 0;
i < N; ++
i) {
36 if (
src[
i] ==
'%' &&
src[
i + 1] ==
'{') {
38 while (
src[
i] !=
'}' &&
src[
i] != 0) {
58#ifdef O2_SIGNPOST_IMPLEMENTATION
63 static std::atomic<o2_log_handle_t*>
first =
nullptr;
78 bool cont = callback(current->
name, current->
log, context);
83 current = current->
next;
91#include <os/signpost.h>
93#define O2_LOG_DEBUG_MAC(log, format, ...) os_log_debug(private_o2_log_##log, format __VA_OPT__(, ) __VA_ARGS__)
94#define O2_SIGNPOST_EVENT_EMIT_MAC(log, id, name, format, ...) os_signpost_event_emit(private_o2_log_##log->os_log, (uint64_t)id.value, name, format __VA_OPT__(, ) __VA_ARGS__)
95#define O2_SIGNPOST_START_MAC(log, id, name, format, ...) os_signpost_interval_begin(private_o2_log_##log->os_log, (uint64_t)id.value, name, format __VA_OPT__(, ) __VA_ARGS__)
96#define O2_SIGNPOST_END_MAC(log, id, name, format, ...) os_signpost_interval_end(private_o2_log_##log->os_log, (uint64_t)id.value, name, format __VA_OPT__(, ) __VA_ARGS__)
97#define O2_SIGNPOST_ENABLED_MAC(log) os_signpost_enabled(private_o2_log_##log->os_log)
100#define O2_DECLARE_LOG_MAC(x, category)
101#define O2_LOG_DEBUG_MAC(log, ...)
102#define O2_SIGNPOST_EVENT_EMIT_MAC(log, id, name, format, ...)
103#define O2_SIGNPOST_START_MAC(log, id, name, format, ...)
104#define O2_SIGNPOST_END_MAC(log, id, name, format, ...)
105#define O2_SIGNPOST_ENABLED_MAC(log) false
109#if defined(__APPLE__) || defined(O2_FORCE_SIGNPOSTS) || !defined(O2_NSIGNPOSTS)
110#define O2_LOG_ENABLED(log) private_o2_log_##log->stacktrace
112#define O2_LOG_ENABLED(log) false
115#if !defined(O2_LOG_MACRO_RAW) && __has_include("Framework/Logger.h")
117#define O2_LOG_MACRO_RAW(level, ...) LOGF(level, __VA_ARGS__)
118#elif !defined(O2_LOG_MACRO_RAW)
121#define O2_LOG_MACRO_RAW(level, format, ...) \
123 printf(#level ":" #format, __VA_ARGS__); \
127#define O2_LOG_MACRO_RAW(...)
130#if !defined(O2_LOG_MACRO) && __has_include("Framework/Logger.h")
132#define O2_LOG_MACRO(...) LOGF(info, __VA_ARGS__)
133#elif !defined(O2_LOG_MACRO)
136#define O2_LOG_MACRO(...) \
138 printf(__VA_ARGS__); \
142#define O2_LOG_MACRO(...)
154 static constexpr size_t N = 1024;
155 std::atomic<size_t>
top = 0;
177 os_log_t os_log =
nullptr;
182 std::array<_o2_signpost_id_t, 256>
ids = {};
235 assert(((int64_t)
pointer & 1) != 1);
242#ifdef O2_SIGNPOST_IMPLEMENTATION
255 size_t currentTop =
stack.top.load(std::memory_order_relaxed);
261#if defined(__x86_64__) || defined(__i386__)
262 __asm__ __volatile__(
"pause" ::
264#elif defined(__aarch64__)
265 __asm__ __volatile__(
"yield" ::
271 if (
stack.top.compare_exchange_weak(currentTop, currentTop + 1,
272 std::memory_order_release,
273 std::memory_order_relaxed)) {
282 size_t currentTop =
stack.top.load(std::memory_order_relaxed);
284 if (currentTop == 0 && spin ==
false) {
286 }
else if (currentTop == 0) {
288#if defined(__x86_64__) || defined(__i386__)
289 __asm__ __volatile__(
"pause" ::
291#elif defined(__aarch64__)
292 __asm__ __volatile__(
"yield" ::
298 if (
stack.top.compare_exchange_weak(currentTop, currentTop - 1,
299 std::memory_order_acquire,
300 std::memory_order_relaxed)) {
312 char const*
name = (
char const*)context;
313 if (strcmp(
name, currentName) == 0) {
329 for (
int i = 0;
i <
n;
i++) {
333 log->defaultStacktrace = defaultStacktrace;
335 newHandle->log = log;
339 if (defaultStacktrace > 1) {
340 log->os_log = os_log_create(
name, OS_LOG_CATEGORY_DYNAMIC_STACK_TRACING);
342 log->os_log = os_log_create(
name, OS_LOG_CATEGORY_DYNAMIC_TRACING);
345 newHandle->name = strdup(
name);
350 while (!
o2_get_logs_tail().compare_exchange_weak(newHandle->next, newHandle,
351 std::memory_order_release,
352 std::memory_order_relaxed)) {
372 for (
i = 0;
i < log->
ids.size(); ++
i) {
373 if (log->
ids[
i].value ==
id.value) {
379 if (
i != log->
ids.size()) {
386 char prebuffer[4096];
387 int s = snprintf(prebuffer, 4096,
"id%.16" PRIx64
":%-16s*>%*c",
id.
value,
name, leading,
' ');
388 vsnprintf(prebuffer + s, 4096 - s,
format, args);
409 log->
ids[signpost_index].value =
id.value;
410 auto* activity = &log->
activities[signpost_index];
412 activity->name =
name;
413 int leading = activity->indentation * 2;
414 char prebuffer[4096];
415 int s = snprintf(prebuffer, 4096,
"id%.16" PRIx64
":%-16sS>%*c",
id.
value,
name, leading,
' ');
416 vsnprintf(prebuffer + s, 4096 - s,
format, args);
428 for (
i = 0;
i < log->
ids.size(); ++
i) {
429 if (log->
ids[
i].value ==
id.value) {
436 if (
i == log->
ids.size()) {
443 char prebuffer[4096];
444 int s = snprintf(prebuffer, 4096,
"id%.16" PRIx64
":%-16sE>%*c",
id.
value,
name, leading,
' ');
445 vsnprintf(prebuffer + s, 4096 - s,
format, args);
449 activity->
name =
nullptr;
450 log->
ids[
i].value = -1;
462 _o2_signpost_interval_end_v(log,
id,
name,
format, args);
473void o2_debug_log_set_stacktrace(
_o2_log_t* log,
int stacktrace)
480#if defined(__APPLE__) || defined(O2_FORCE_SIGNPOSTS) || !defined(O2_NSIGNPOSTS)
483#define O2_DECLARE_DYNAMIC_LOG(name) static _o2_log_t* private_o2_log_##name = (_o2_log_t*)_o2_log_create("ch.cern.aliceo2." #name, 1)
485#define O2_DECLARE_DYNAMIC_STACKTRACE_LOG(name) static _o2_log_t* private_o2_log_##name = (_o2_log_t*)_o2_log_create("ch.cern.aliceo2." #name, 64)
486#define O2_DECLARE_LOG(name, category) static _o2_log_t* private_o2_log_##name = (_o2_log_t*)_o2_log_create("ch.cern.aliceo2." #name, 1)
488#define O2_LOG_ENABLE(log) _o2_log_set_stacktrace(private_o2_log_##log, private_o2_log_##log->defaultStacktrace)
489#define O2_LOG_DISABLE(log) _o2_log_set_stacktrace(private_o2_log_##log, 0)
492#define O2_LOG_DEBUG(log, ...) __extension__({ \
493 if (O2_BUILTIN_UNLIKELY(O2_LOG_ENABLED(log))) { \
494 O2_LOG_MACRO(__VA_ARGS__); \
495 } else if (O2_BUILTIN_UNLIKELY(private_o2_log_##log->stacktrace)) { \
496 O2_LOG_MACRO(__VA_ARGS__); \
499#define O2_SIGNPOST_ID_FROM_POINTER(name, log, pointer) _o2_signpost_id_t name = _o2_signpost_id_make_with_pointer(private_o2_log_##log, pointer)
500#define O2_SIGNPOST_ID_GENERATE(name, log) _o2_signpost_id_t name = _o2_signpost_id_generate_local(private_o2_log_##log)
504#define O2_SIGNPOST_ACTION(log, callback) __extension__({ \
505 if (O2_BUILTIN_UNLIKELY(O2_SIGNPOST_ENABLED_MAC(log))) { \
506 _o2_signpost_action(private_o2_log_##log, callback); \
507 } else if (O2_BUILTIN_UNLIKELY(private_o2_log_##log->stacktrace)) { \
508 _o2_signpost_action(private_o2_log_##log, callback); \
516#define O2_SIGNPOST_EVENT_EMIT(log, id, name, format, ...) __extension__({ \
517 if (O2_BUILTIN_UNLIKELY(O2_SIGNPOST_ENABLED_MAC(log))) { \
518 O2_SIGNPOST_EVENT_EMIT_MAC(log, id, name, format, ##__VA_ARGS__); \
519 } else if (O2_BUILTIN_UNLIKELY(private_o2_log_##log->stacktrace)) { \
520 _o2_signpost_event_emit(private_o2_log_##log, id, name, remove_engineering_type(format).data(), ##__VA_ARGS__); \
525#define O2_SIGNPOST_EVENT_EMIT_INFO(log, id, name, format, ...) __extension__({ \
526 if (O2_BUILTIN_UNLIKELY(O2_SIGNPOST_ENABLED_MAC(log))) { \
527 O2_SIGNPOST_EVENT_EMIT_MAC(log, id, name, format, ##__VA_ARGS__); \
528 } else if (O2_BUILTIN_UNLIKELY(private_o2_log_##log->stacktrace)) { \
529 _o2_signpost_event_emit(private_o2_log_##log, id, name, remove_engineering_type(format).data(), ##__VA_ARGS__); \
531 O2_LOG_MACRO_RAW(info, remove_engineering_type(format).data(), ##__VA_ARGS__); \
536#define O2_SIGNPOST_EVENT_EMIT_ERROR(log, id, name, format, ...) __extension__({ \
537 if (O2_BUILTIN_UNLIKELY(O2_SIGNPOST_ENABLED_MAC(log))) { \
538 O2_SIGNPOST_EVENT_EMIT_MAC(log, id, name, format, ##__VA_ARGS__); \
539 } else if (O2_BUILTIN_UNLIKELY(private_o2_log_##log->stacktrace)) { \
540 _o2_signpost_event_emit(private_o2_log_##log, id, name, remove_engineering_type(format).data(), ##__VA_ARGS__); \
542 O2_LOG_MACRO_RAW(error, remove_engineering_type(format).data(), ##__VA_ARGS__); \
546#define O2_SIGNPOST_EVENT_EMIT_WARN(log, id, name, format, ...) __extension__({ \
547 if (O2_BUILTIN_UNLIKELY(O2_SIGNPOST_ENABLED_MAC(log))) { \
548 O2_SIGNPOST_EVENT_EMIT_MAC(log, id, name, format, ##__VA_ARGS__); \
549 } else if (O2_BUILTIN_UNLIKELY(private_o2_log_##log->stacktrace)) { \
550 _o2_signpost_event_emit(private_o2_log_##log, id, name, remove_engineering_type(format).data(), ##__VA_ARGS__); \
552 O2_LOG_MACRO_RAW(warn, remove_engineering_type(format).data(), ##__VA_ARGS__); \
556#define O2_SIGNPOST_EVENT_EMIT_CRITICAL(log, id, name, format, ...) __extension__({ \
557 if (O2_BUILTIN_UNLIKELY(O2_SIGNPOST_ENABLED_MAC(log))) { \
558 O2_SIGNPOST_EVENT_EMIT_MAC(log, id, name, format, ##__VA_ARGS__); \
559 } else if (O2_BUILTIN_UNLIKELY(private_o2_log_##log->stacktrace)) { \
560 _o2_signpost_event_emit(private_o2_log_##log, id, name, remove_engineering_type(format).data(), ##__VA_ARGS__); \
562 O2_LOG_MACRO_RAW(critical, remove_engineering_type(format).data(), ##__VA_ARGS__); \
565#define O2_SIGNPOST_START(log, id, name, format, ...) \
566 if (O2_BUILTIN_UNLIKELY(O2_SIGNPOST_ENABLED_MAC(log))) { \
567 O2_SIGNPOST_START_MAC(log, id, name, format, ##__VA_ARGS__); \
568 } else if (O2_BUILTIN_UNLIKELY(private_o2_log_##log->stacktrace)) { \
569 _o2_signpost_interval_begin(private_o2_log_##log, id, name, remove_engineering_type(format).data(), ##__VA_ARGS__); \
571#define O2_SIGNPOST_END(log, id, name, format, ...) \
572 if (O2_BUILTIN_UNLIKELY(O2_SIGNPOST_ENABLED_MAC(log))) { \
573 O2_SIGNPOST_END_MAC(log, id, name, format, ##__VA_ARGS__); \
574 } else if (O2_BUILTIN_UNLIKELY(private_o2_log_##log->stacktrace)) { \
575 _o2_signpost_interval_end(private_o2_log_##log, id, name, remove_engineering_type(format).data(), ##__VA_ARGS__); \
578#define O2_DECLARE_DYNAMIC_LOG(x)
579#define O2_DECLARE_DYNAMIC_STACKTRACE_LOG(x)
580#define O2_DECLARE_LOG(x, category)
581#define O2_LOG_ENABLE(log)
582#define O2_LOG_DISABLE(log)
583#define O2_LOG_DEBUG(log, ...)
584#define O2_SIGNPOST_ID_FROM_POINTER(name, log, pointer)
585#define O2_SIGNPOST_ID_GENERATE(name, log)
586#define O2_SIGNPOST_EVENT_EMIT(log, id, name, format, ...)
587#define O2_SIGNPOST_START(log, id, name, format, ...)
588#define O2_SIGNPOST_END(log, id, name, format, ...)
o2_log_handle_t * o2_walk_logs(bool(*callback)(char const *name, void *log, void *context), void *context=nullptr)
bool _o2_lock_free_stack_push(_o2_lock_free_stack &stack, const int &value, bool spin=false)
void _o2_signpost_interval_end(_o2_log_t *log, _o2_signpost_id_t id, char const *name, char const *const format,...)
void _o2_singpost_action(_o2_log_t *log, void(*callback)(void *))
void _o2_log_set_stacktrace(_o2_log_t *log, int stacktrace)
void _o2_signpost_event_emit(_o2_log_t *log, _o2_signpost_id_t id, char const *name, char const *const format,...)
bool _o2_lock_free_stack_pop(_o2_lock_free_stack &stack, int &value, bool spin=false)
_o2_signpost_id_t _o2_signpost_id_generate_local(_o2_log_t *log)
std::atomic< o2_log_handle_t * > & o2_get_logs_tail()
consteval auto remove_engineering_type(char const (&src)[N])
#define O2_LOG_MACRO(...)
void _o2_signpost_action(_o2_log_t *log, void(*callback)(void *))
_o2_signpost_id_t _o2_signpost_id_make_with_pointer(_o2_log_t *log, void const *pointer)
void _o2_signpost_interval_begin(_o2_log_t *log, _o2_signpost_id_t id, char const *name, char const *const format,...)
void * _o2_log_create(char const *name, int stacktrace)
GLdouble GLdouble GLdouble GLdouble top
GLuint const GLchar * name
GLsizei const GLfloat * value
GLint GLint GLsizei GLint GLenum format
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
unsigned char indentation
static constexpr size_t N
std::atomic< int64_t > current_indentation
_o2_lock_free_stack slots
std::array< _o2_signpost_id_t, 256 > ids
std::array< _o2_activity_t, 256 > activities
std::atomic< int64_t > unique_signpost
static void demangled_backtrace_symbols(void **backtrace, unsigned int total, int fd)
static constexpr unsigned int MAX_BACKTRACE_SIZE