/*
* Copyright (c) 2016-2020 Belledonne Communications SARL.
*
* This file is part of bctoolbox.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
/**
* \file logging.h
* \brief Logging API.
*
**/
#ifndef BCTBX_LOGGING_H
#define BCTBX_LOGGING_H
#include
#include "bctoolbox/list.h"
/**
* The BCTBX_LOG_DOMAIN macro should be defined with a preprocessor directive (ex: -DBCTBX_LOG_DOMAIN="my-component") in every
* software entity (application, library, sub-parts of software etc) using the bctoolbox log facility, so that all invocations of the
* bctbx_message(), bctbx_warning(), bctbx_error(), bctbx_fatal() outputs their log within the domain title of the software part there are compiled in.
* It SHALL not be defined in any public header, otherwise it will conflict with upper layer using the logging facility for their own domain.
* As a special exception, bctoolbox defines the log domain to be "bctbx" if no preprocessor directive defines it.
* As a result, bctboolbox owns logs will be output into the "bctbx" domain.
**/
#ifndef BCTBX_LOG_DOMAIN
#define BCTBX_LOG_DOMAIN "bctbx"
#endif
#ifdef __cplusplus
extern "C"
{
#endif
typedef enum {
BCTBX_LOG_DEBUG=1,
BCTBX_LOG_TRACE=1<<1,
BCTBX_LOG_MESSAGE=1<<2,
BCTBX_LOG_WARNING=1<<3,
BCTBX_LOG_ERROR=1<<4,
BCTBX_LOG_FATAL=1<<5,
BCTBX_LOG_LOGLEV_END=1<<6
} BctbxLogLevel;
typedef struct _bctbx_log_handler_t bctbx_log_handler_t;
typedef void (*BctbxLogFunc)(const char *domain, BctbxLogLevel lev, const char *fmt, va_list args);
typedef void (*BctbxLogHandlerFunc)(void *info,const char *domain, BctbxLogLevel lev, const char *fmt, va_list args);
typedef void (*BctbxLogHandlerDestroyFunc)(bctbx_log_handler_t *handler);
/*
initialise logging functions, add default log handler for stdout output.
@param[in] bool_t create : No longer used, always created with a default logger to stdout.
*/
BCTBX_PUBLIC void bctbx_init_logger(bool_t create);
/*
free logging memory
*/
BCTBX_PUBLIC void bctbx_uninit_logger(void);
/*
Default functions to free log handlers
@param[in] bctbx_log_handler_t* handler : the handler to free
*/
BCTBX_PUBLIC void bctbx_logv_out_destroy(bctbx_log_handler_t *handler);
BCTBX_PUBLIC void bctbx_logv_file_destroy(bctbx_log_handler_t *handler);
/*
Function to create a log handler
@param[in] BctbxLogHandlerFunc func : the function to call to handle a new line of log
@param[in] BctbxLogHandlerDestroyFunc destroy : the function to call to free this handler particuler its user_info field
@param[in] void* user_info : complementary information to handle the logs if needed
@return a new bctbx_log_handler_t
*/
BCTBX_PUBLIC bctbx_log_handler_t* bctbx_create_log_handler(BctbxLogHandlerFunc func, BctbxLogHandlerDestroyFunc destroy, void* user_data);
/*
Function to create a file log handler
@param[in] uint64_t max_size : the maximum size of the log file before rotating to a new one (if 0 then no rotation)
@param[in] const char* path : the path where to put the log files
@param[in] const char* name : the name of the log files
@param[in] FILE* f : the file where to write the logs
@return a new bctbx_log_handler_t
*/
BCTBX_PUBLIC bctbx_log_handler_t* bctbx_create_file_log_handler(uint64_t max_size, const char* path, const char* name);
/**
* @brief Request reopening of the log file.
* @param[in] file_log_handler The log handler whose file will be reopened.
* @note This function is thread-safe and reopening is done asynchronously.
*/
BCTBX_PUBLIC void bctbx_file_log_handler_reopen(bctbx_log_handler_t *file_log_handler);
/* set domain the handler is limited to. NULL for ALL*/
BCTBX_PUBLIC void bctbx_log_handler_set_domain(bctbx_log_handler_t * log_handler,const char *domain);
BCTBX_PUBLIC void bctbx_log_handler_set_user_data(bctbx_log_handler_t*, void* user_data);
BCTBX_PUBLIC void *bctbx_log_handler_get_user_data(const bctbx_log_handler_t* log_handler);
BCTBX_PUBLIC void bctbx_add_log_handler(bctbx_log_handler_t* handler);
BCTBX_PUBLIC void bctbx_remove_log_handler(bctbx_log_handler_t* handler);
/*
* Set a callback function to render logs for the default handler.
*/
BCTBX_PUBLIC void bctbx_set_log_handler(BctbxLogFunc func);
/*
* Same as bctbx_set_log_handler but only for a domain. NULL for all.
* Be careful that if domain is specified, the default log handler will no longer output logs for other domains.
*/
BCTBX_PUBLIC void bctbx_set_log_handler_for_domain(BctbxLogFunc func, const char* domain);
/*Convenient function that creates a static log handler logging into supplied FILE argument.
Despite it is not recommended to use it in libraries, it can be useful for simple test programs.*/
BCTBX_PUBLIC void bctbx_set_log_file(FILE* f);
BCTBX_PUBLIC bctbx_list_t* bctbx_get_log_handlers(void);
BCTBX_PUBLIC void bctbx_logv_out(const char *domain, BctbxLogLevel level, const char *fmt, va_list args);
BCTBX_PUBLIC void bctbx_logv_file(void* user_info, const char *domain, BctbxLogLevel level, const char *fmt, va_list args);
/*
* Returns 1 if the log level 'level' is enabled for the calling thread, otherwise 0.
* This gives the condition to decide to output a log.
*/
BCTBX_PUBLIC int bctbx_log_level_enabled(const char *domain, BctbxLogLevel level);
BCTBX_PUBLIC void bctbx_logv(const char *domain, BctbxLogLevel level, const char *fmt, va_list args);
/**
* Flushes the log output queue.
* WARNING: Must be called from the thread that has been defined with bctbx_set_log_thread_id().
*/
BCTBX_PUBLIC void bctbx_logv_flush(void);
/**
* Activate all log level greater or equal than specified level argument.
**/
BCTBX_PUBLIC void bctbx_set_log_level(const char *domain, BctbxLogLevel level);
BCTBX_PUBLIC void bctbx_set_log_level_mask(const char *domain, int levelmask);
BCTBX_PUBLIC unsigned int bctbx_get_log_level_mask(const char *domain);
/**
* Set a specific log level for the calling thread for domain.
* When domain is NULL, the log level applies to all domains.
*/
BCTBX_PUBLIC void bctbx_set_thread_log_level(const char *domain, BctbxLogLevel level);
/**
* Clears the specific log level set for the calling thread by bctbx_set_thread_log_level().
* After calling this function, the global (not per thread) log level will apply to
* logs printed by this thread.
*/
BCTBX_PUBLIC void bctbx_clear_thread_log_level(const char *domain);
/**
* Tell oRTP the id of the thread used to output the logs.
* This is meant to output all the logs from the same thread to prevent deadlock problems at the application level.
* @param[in] thread_id The id of the thread that will output the logs (can be obtained using bctbx_thread_self()).
*/
BCTBX_PUBLIC void bctbx_set_log_thread_id(unsigned long thread_id);
#ifdef __GNUC__
#define CHECK_FORMAT_ARGS(m,n) __attribute__((format(printf,m,n)))
#else
#define CHECK_FORMAT_ARGS(m,n)
#endif
#ifdef __clang__
/*in case of compile with -g static inline can produce this type of warning*/
#pragma GCC diagnostic ignored "-Wunused-function"
#endif
#ifdef BCTBX_DEBUG_MODE
static BCTBX_INLINE void CHECK_FORMAT_ARGS(1,2) bctbx_debug(const char *fmt,...)
{
va_list args;
va_start (args, fmt);
bctbx_logv(BCTBX_LOG_DOMAIN, BCTBX_LOG_DEBUG, fmt, args);
va_end (args);
}
#else
#define bctbx_debug(...)
#endif
#ifdef BCTBX_NOMESSAGE_MODE
#define bctbx_log(...)
#define bctbx_message(...)
#define bctbx_warning(...)
#else
static BCTBX_INLINE void bctbx_log(const char* domain, BctbxLogLevel lev, const char *fmt,...) {
va_list args;
va_start (args, fmt);
bctbx_logv(domain, lev, fmt, args);
va_end (args);
}
static BCTBX_INLINE void CHECK_FORMAT_ARGS(1,2) bctbx_message(const char *fmt,...)
{
va_list args;
va_start (args, fmt);
bctbx_logv(BCTBX_LOG_DOMAIN, BCTBX_LOG_MESSAGE, fmt, args);
va_end (args);
}
static BCTBX_INLINE void CHECK_FORMAT_ARGS(1,2) bctbx_warning(const char *fmt,...)
{
va_list args;
va_start (args, fmt);
bctbx_logv(BCTBX_LOG_DOMAIN, BCTBX_LOG_WARNING, fmt, args);
va_end (args);
}
#endif
static BCTBX_INLINE void CHECK_FORMAT_ARGS(1,2) bctbx_error(const char *fmt,...)
{
va_list args;
va_start (args, fmt);
bctbx_logv(BCTBX_LOG_DOMAIN, BCTBX_LOG_ERROR, fmt, args);
va_end (args);
}
static BCTBX_INLINE void CHECK_FORMAT_ARGS(1,2) bctbx_fatal(const char *fmt,...)
{
va_list args;
va_start (args, fmt);
bctbx_logv(BCTBX_LOG_DOMAIN, BCTBX_LOG_FATAL, fmt, args);
va_end (args);
}
#ifdef __QNX__
void bctbx_qnx_log_handler(const char *domain, BctbxLogLevel lev, const char *fmt, va_list args);
#endif
#ifdef __cplusplus
}
#include
#include
#include
#if !defined(_WIN32) && !defined(__QNX__)
#include
#endif
namespace bctoolbox {
namespace log {
// Here we define our application severity levels.
enum level { normal, trace, debug, info, warning, error, fatal };
// The formatting logic for the severity level
template
inline std::basic_ostream &operator<<(std::basic_ostream &strm,
const bctoolbox::log::level &lvl) {
static const char *const str[] = {"normal", "trace", "debug", "info", "warning", "error", "fatal"};
if (static_cast(lvl) < (sizeof(str) / sizeof(*str)))
strm << str[lvl];
else
strm << static_cast(lvl);
return strm;
}
template
inline std::basic_istream &operator>>(std::basic_istream &strm,
bctoolbox::log::level &lvl) {
static const char *const str[] = {"normal", "trace", "debug", "info", "warning", "error", "fatal"};
std::string s;
strm >> s;
for (unsigned int n = 0; n < (sizeof(str) / sizeof(*str)); ++n) {
if (s == str[n]) {
lvl = static_cast(n);
return strm;
}
}
// Parse error
strm.setstate(std::ios_base::failbit);
return strm;
}
}
}
#include
class pumpstream : public std::ostringstream {
public:
/*contructor used to disable logging*/
pumpstream():mDomain(""),mLevel(BCTBX_LOG_DEBUG),mTraceEnabled(false){}
pumpstream(const char *domain, BctbxLogLevel level) : mDomain(domain ? domain : ""), mLevel(level),mTraceEnabled(true) {}
~pumpstream() {
const char *domain = mDomain.empty() ? NULL : mDomain.c_str();
if (mTraceEnabled && bctbx_log_level_enabled(domain, mLevel))
bctbx_log(domain, mLevel, "%s", str().c_str());
}
private:
const std::string mDomain;
const BctbxLogLevel mLevel;
const bool mTraceEnabled;
};
template inline pumpstream &operator<<(pumpstream &&__os, const _Tp &__x) {
(static_cast(__os)) << __x;
return __os;
}
#define BCTBX_SLOG(domain, thelevel) pumpstream(domain, thelevel)
#ifndef BCTBX_DEBUG_MODE
#define BCTBX_SLOGD BCTBX_SLOG(BCTBX_LOG_DOMAIN, BCTBX_LOG_DEBUG)
#else
#define BCTBX_SLOGD pumpstream()
#endif
#define BCTBX_SLOGI BCTBX_SLOG(BCTBX_LOG_DOMAIN, BCTBX_LOG_MESSAGE)
#define BCTBX_SLOGW BCTBX_SLOG(BCTBX_LOG_DOMAIN, BCTBX_LOG_WARNING)
#define BCTBX_SLOGE BCTBX_SLOG(BCTBX_LOG_DOMAIN, BCTBX_LOG_ERROR)
#endif
#endif