a/u_logging: Docs for logging.

This commit is contained in:
Ryan Pavlik 2021-04-26 14:34:19 -05:00 committed by Jakob Bornecrantz
parent 8a936f2474
commit 8af1bfbf8c

View file

@ -31,6 +31,19 @@ struct xrt_device;
* @{ * @{
*/ */
/*!
* @brief Logging level enum
*/
enum u_logging_level
{
U_LOGGING_TRACE, //!< Trace messages, highly verbose.
U_LOGGING_DEBUG, //!< Debug messages, verbose.
U_LOGGING_INFO, //!< Info messages: not very verbose, not indicating a problem.
U_LOGGING_WARN, //!< Warning messages: indicating a potential problem
U_LOGGING_ERROR, //!< Error messages: indicating a problem
U_LOGGING_RAW, //!< Special level for raw printing, prints a new-line.
};
/*! /*!
* For places where you really just want printf, prints a new-line. * For places where you really just want printf, prints a new-line.
*/ */
@ -39,23 +52,63 @@ struct xrt_device;
u_log(__FILE__, __LINE__, __func__, U_LOGGING_RAW, __VA_ARGS__); \ u_log(__FILE__, __LINE__, __func__, U_LOGGING_RAW, __VA_ARGS__); \
} while (false) } while (false)
/*!
* @name Base Logging Utilities
* In most cases, you will want to use another macro from this file, or a module/driver-local macro, to do your logging.
* @{
*/
/*!
* @brief Log a message at @p level , with file, line, and function context (always logs) - typically wrapped
* in a helper macro.
*
* @param level A @ref u_logging_level value for this message.
* @param ... Format string and optional format arguments.
*/
#define U_LOG(level, ...) \ #define U_LOG(level, ...) \
do { \ do { \
u_log(__FILE__, __LINE__, __func__, level, __VA_ARGS__); \ u_log(__FILE__, __LINE__, __func__, level, __VA_ARGS__); \
} while (false) } while (false)
/*!
* @brief Log at @p level only if the level is at least @p cond_level - typically wrapped in a helper macro.
*
* Adds file, line, and function context. Like U_LOG() but conditional.
*
* @param level A @ref u_logging_level value for this message.
* @param cond_level The minimum @ref u_logging_level that will be actually output.
* @param ... Format string and optional format arguments.
*/
#define U_LOG_IFL(level, cond_level, ...) \ #define U_LOG_IFL(level, cond_level, ...) \
do { \ do { \
if (cond_level <= level) { \ if (cond_level <= level) { \
u_log(__FILE__, __LINE__, __func__, level, __VA_ARGS__); \ u_log(__FILE__, __LINE__, __func__, level, __VA_ARGS__); \
} \ } \
} while (false) } while (false)
/*!
* @brief Log at @p level for a given @ref xrt_device - typically wrapped in a helper macro.
*
* Adds file, line, and function context, and forwards device context from provided @p xdev .
*
* Like U_LOG() but calling u_log_xdev() (which takes a device) instead.
*
* @param level A @ref u_logging_level value for this message.
* @param xdev The @ref xrt_device pointer associated with this message.
* @param ... Format string and optional format arguments.
*/
#define U_LOG_XDEV(level, xdev, ...) \ #define U_LOG_XDEV(level, xdev, ...) \
do { \ do { \
u_log_xdev(__FILE__, __LINE__, __func__, level, xdev, __VA_ARGS__); \ u_log_xdev(__FILE__, __LINE__, __func__, level, xdev, __VA_ARGS__); \
} while (false) } while (false)
/*!
* @brief Log at @p level for a given @ref xrt_device, only if the level is at least @p cond_level - typically wrapped
* in a helper macro.
*
* Adds file, line, and function context, and forwards device context from provided @p xdev .
* @param level A @ref u_logging_level value for this message.
* @param cond_level The minimum @ref u_logging_level that will be actually output.
* @param xdev The @ref xrt_device pointer associated with this message.
* @param ... Format string and optional format arguments.
*/
#define U_LOG_XDEV_IFL(level, cond_level, xdev, ...) \ #define U_LOG_XDEV_IFL(level, cond_level, xdev, ...) \
do { \ do { \
if (cond_level <= level) { \ if (cond_level <= level) { \
@ -63,41 +116,7 @@ struct xrt_device;
} \ } \
} while (false) } while (false)
// clang-format off
#define U_LOG_T(...) U_LOG_IFL_T(u_log_get_global_level(), __VA_ARGS__)
#define U_LOG_D(...) U_LOG_IFL_D(u_log_get_global_level(), __VA_ARGS__)
#define U_LOG_I(...) U_LOG_IFL_I(u_log_get_global_level(), __VA_ARGS__)
#define U_LOG_W(...) U_LOG_IFL_W(u_log_get_global_level(), __VA_ARGS__)
#define U_LOG_E(...) U_LOG_IFL_E(u_log_get_global_level(), __VA_ARGS__)
#define U_LOG_IFL_T(cond_level, ...) U_LOG_IFL(U_LOGGING_TRACE, cond_level, __VA_ARGS__)
#define U_LOG_IFL_D(cond_level, ...) U_LOG_IFL(U_LOGGING_DEBUG, cond_level, __VA_ARGS__)
#define U_LOG_IFL_I(cond_level, ...) U_LOG_IFL(U_LOGGING_INFO, cond_level, __VA_ARGS__)
#define U_LOG_IFL_W(cond_level, ...) U_LOG_IFL(U_LOGGING_WARN, cond_level, __VA_ARGS__)
#define U_LOG_IFL_E(cond_level, ...) U_LOG_IFL(U_LOGGING_ERROR, cond_level, __VA_ARGS__)
#define U_LOG_XDEV_IFL_T(xdev, cond_level, ...) U_LOG_XDEV_IFL(U_LOGGING_TRACE, cond_level, xdev, __VA_ARGS__)
#define U_LOG_XDEV_IFL_D(xdev, cond_level, ...) U_LOG_XDEV_IFL(U_LOGGING_DEBUG, cond_level, xdev, __VA_ARGS__)
#define U_LOG_XDEV_IFL_I(xdev, cond_level, ...) U_LOG_XDEV_IFL(U_LOGGING_INFO, cond_level, xdev, __VA_ARGS__)
#define U_LOG_XDEV_IFL_W(xdev, cond_level, ...) U_LOG_XDEV_IFL(U_LOGGING_WARN, cond_level, xdev, __VA_ARGS__)
#define U_LOG_XDEV_IFL_E(xdev, cond_level, ...) U_LOG_XDEV_IFL(U_LOGGING_ERROR, cond_level, xdev, __VA_ARGS__)
#define U_LOG_XDEV_T(xdev, ...) U_LOG_XDEV(U_LOGGING_TRACE, xdev, __VA_ARGS__)
#define U_LOG_XDEV_D(xdev, ...) U_LOG_XDEV(U_LOGGING_DEBUG, xdev, __VA_ARGS__)
#define U_LOG_XDEV_I(xdev, ...) U_LOG_XDEV(U_LOGGING_INFO, xdev, __VA_ARGS__)
#define U_LOG_XDEV_W(xdev, ...) U_LOG_XDEV(U_LOGGING_WARN, xdev, __VA_ARGS__)
#define U_LOG_XDEV_E(xdev, ...) U_LOG_XDEV(U_LOGGING_ERROR, xdev, __VA_ARGS__)
// clang-format on
enum u_logging_level
{
U_LOGGING_TRACE,
U_LOGGING_DEBUG,
U_LOGGING_INFO,
U_LOGGING_WARN,
U_LOGGING_ERROR,
U_LOGGING_RAW, //!< Special level for raw printing, prints a new-line.
};
/*! /*!
* Returns the global logging level, subsystems own logging level take precedence. * Returns the global logging level, subsystems own logging level take precedence.
@ -106,16 +125,32 @@ enum u_logging_level
u_log_get_global_level(void); u_log_get_global_level(void);
/*! /*!
* This function always logs, level is used for printing or passed to native * @brief Main non-device-related log implementation function: do not call directly, use a macro that wraps it.
* logging functions. *
* This function always logs: level is used for printing or passed to native logging functions.
*
* @param file Source file name associated with a message
* @param line Source file line associated with a message
* @param func Function name associated with a message
* @param level Message level: used for formatting or forwarding to native log functions
* @param format Format string
* @param ... Format parameters
*/ */
void void
u_log(const char *file, int line, const char *func, enum u_logging_level level, const char *format, ...) u_log(const char *file, int line, const char *func, enum u_logging_level level, const char *format, ...)
XRT_PRINTF_FORMAT(5, 6); XRT_PRINTF_FORMAT(5, 6);
/*! /*!
* This function always logs, level is used for printing or passed to native * @brief Main device-related log implementation function: do not call directly, use a macro that wraps it.
* logging functions. *
* This function always logs: level is used for printing or passed to native logging functions.
* @param file Source file name associated with a message
* @param line Source file line associated with a message
* @param func Function name associated with a message
* @param level Message level: used for formatting or forwarding to native log functions
* @param xdev The associated @ref xrt_device
* @param format Format string
* @param ... Format parameters
*/ */
void void
u_log_xdev(const char *file, u_log_xdev(const char *file,
@ -126,6 +161,119 @@ u_log_xdev(const char *file,
const char *format, const char *format,
...) XRT_PRINTF_FORMAT(6, 7); ...) XRT_PRINTF_FORMAT(6, 7);
/*!
* @}
*/
/*!
* @name Logging macros conditional on global log level
*
* These each imply a log level, and will only log if the global log level is equal or lower.
* They are often used for one-off logging in a module with few other logging needs,
* where having a module-specific log level would be unnecessary.
*
* @see U_LOG_IFL, u_log_get_global_level()
* @param ... Format string and optional format arguments.
* @{
*/
//! Log a message at U_LOGGING_TRACE level, conditional on the global log level
#define U_LOG_T(...) U_LOG_IFL_T(u_log_get_global_level(), __VA_ARGS__)
//! Log a message at U_LOGGING_DEBUG level, conditional on the global log level
#define U_LOG_D(...) U_LOG_IFL_D(u_log_get_global_level(), __VA_ARGS__)
//! Log a message at U_LOGGING_INFO level, conditional on the global log level
#define U_LOG_I(...) U_LOG_IFL_I(u_log_get_global_level(), __VA_ARGS__)
//! Log a message at U_LOGGING_WARN level, conditional on the global log level
#define U_LOG_W(...) U_LOG_IFL_W(u_log_get_global_level(), __VA_ARGS__)
//! Log a message at U_LOGGING_ERROR level, conditional on the global log level
#define U_LOG_E(...) U_LOG_IFL_E(u_log_get_global_level(), __VA_ARGS__)
/*!
* @}
*/
/*!
* @name Logging macros conditional on provided log level
*
* These are often wrapped within a module, to automatically supply
* @p cond_level as appropriate for that module.
*
* @see U_LOG_IFL
* @param cond_level The minimum @ref u_logging_level that will be actually output.
* @param ... Format string and optional format arguments.
*
* @{
*/
//! Conditionally log a message at U_LOGGING_TRACE level.
#define U_LOG_IFL_T(cond_level, ...) U_LOG_IFL(U_LOGGING_TRACE, cond_level, __VA_ARGS__)
//! Conditionally log a message at U_LOGGING_DEBUG level.
#define U_LOG_IFL_D(cond_level, ...) U_LOG_IFL(U_LOGGING_DEBUG, cond_level, __VA_ARGS__)
//! Conditionally log a message at U_LOGGING_INFO level.
#define U_LOG_IFL_I(cond_level, ...) U_LOG_IFL(U_LOGGING_INFO, cond_level, __VA_ARGS__)
//! Conditionally log a message at U_LOGGING_WARN level.
#define U_LOG_IFL_W(cond_level, ...) U_LOG_IFL(U_LOGGING_WARN, cond_level, __VA_ARGS__)
//! Conditionally log a message at U_LOGGING_ERROR level.
#define U_LOG_IFL_E(cond_level, ...) U_LOG_IFL(U_LOGGING_ERROR, cond_level, __VA_ARGS__)
/*!
* @}
*/
/*!
* @name Device-related logging macros conditional on provided log level
*
* These are often wrapped within a driver, to automatically supply @p xdev and
* @p cond_level from their conventional names and log level member variable.
*
* @param level A @ref u_logging_level value for this message.
* @param cond_level The minimum @ref u_logging_level that will be actually output.
* @param xdev The @ref xrt_device pointer associated with this message.
* @param ... Format string and optional format arguments.
*
* @{
*/
//! Conditionally log a device-related message at U_LOGGING_TRACE level.
#define U_LOG_XDEV_IFL_T(xdev, cond_level, ...) U_LOG_XDEV_IFL(U_LOGGING_TRACE, cond_level, xdev, __VA_ARGS__)
//! Conditionally log a device-related message at U_LOGGING_DEBUG level.
#define U_LOG_XDEV_IFL_D(xdev, cond_level, ...) U_LOG_XDEV_IFL(U_LOGGING_DEBUG, cond_level, xdev, __VA_ARGS__)
//! Conditionally log a device-related message at U_LOGGING_INFO level.
#define U_LOG_XDEV_IFL_I(xdev, cond_level, ...) U_LOG_XDEV_IFL(U_LOGGING_INFO, cond_level, xdev, __VA_ARGS__)
//! Conditionally log a device-related message at U_LOGGING_WARN level.
#define U_LOG_XDEV_IFL_W(xdev, cond_level, ...) U_LOG_XDEV_IFL(U_LOGGING_WARN, cond_level, xdev, __VA_ARGS__)
//! Conditionally log a device-related message at U_LOGGING_ERROR level.
#define U_LOG_XDEV_IFL_E(xdev, cond_level, ...) U_LOG_XDEV_IFL(U_LOGGING_ERROR, cond_level, xdev, __VA_ARGS__)
/*!
* @}
*/
/*!
* @name Device-related logging macros that always log.
*
* These wrap U_LOG_XDEV() to supply the @p level - which is only used for formatting the output, these macros always
* log regardless of level.
*
* @param xdev The @ref xrt_device pointer associated with this message.
* @param ... Format string and optional format arguments.
* @{
*/
//! Log a device-related message at U_LOGGING_TRACE level (always logs).
#define U_LOG_XDEV_T(xdev, ...) U_LOG_XDEV(U_LOGGING_TRACE, xdev, __VA_ARGS__)
//! Log a device-related message at U_LOGGING_DEBUG level (always logs).
#define U_LOG_XDEV_D(xdev, ...) U_LOG_XDEV(U_LOGGING_DEBUG, xdev, __VA_ARGS__)
//! Log a device-related message at U_LOGGING_INFO level (always logs).
#define U_LOG_XDEV_I(xdev, ...) U_LOG_XDEV(U_LOGGING_INFO, xdev, __VA_ARGS__)
//! Log a device-related message at U_LOGGING_WARN level (always logs).
#define U_LOG_XDEV_W(xdev, ...) U_LOG_XDEV(U_LOGGING_WARN, xdev, __VA_ARGS__)
//! Log a device-related message at U_LOGGING_ERROR level (always logs).
#define U_LOG_XDEV_E(xdev, ...) U_LOG_XDEV(U_LOGGING_ERROR, xdev, __VA_ARGS__)
/*!
* @}
*/
/*! /*!
* @} * @}