/* * Copyright (C) 2014 Freie Universität Berlin * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup core_util * @{ * * @file * @brief Debug-header * * @details If *ENABLE_DEBUG* is defined inside an implementation file, all * calls to ::DEBUG will work the same as *printf* and output the * given information to stdout. If *ENABLE_DEBUG* is not defined, * all calls to ::DEBUG will be ignored. * * @author Kaspar Schleiser */ #ifndef DEBUG_H #define DEBUG_H #include #include "sched.h" #include "thread.h" #ifdef __cplusplus extern "C" { #endif /** * @def DEBUG_PRINT * * @brief Print debug information if the calling thread stack is large enough * * Use this macro the same as `printf`. When `DEVELHELP` is defined inside an * implementation file, all usages of ::DEBUG_PRINT will print the given * information to stdout after verifying the stack is big enough. If `DEVELHELP` * is not set, this check is not performed. (CPU exception may occur) */ #ifdef DEVELHELP #include "cpu_conf.h" #define DEBUG_PRINT(...) \ do { \ if ((thread_get_active() == NULL) || \ (thread_get_active()->stack_size >= \ THREAD_EXTRA_STACKSIZE_PRINTF)) { \ printf(__VA_ARGS__); \ } \ else { \ puts("Cannot debug, stack too small. Consider using DEBUG_PUTS()."); \ } \ } while (0) #else #define DEBUG_PRINT(...) printf(__VA_ARGS__) #endif /** * @def DEBUG_BREAKPOINT * * @brief Set a debug breakpoint * * When `DEVELHELP` is enabled, this traps the CPU and allows to debug the * program with e.g. `gdb`. * Without `DEVELHELP` this turns into a no-op. * * @warning If no Debugger is attached, the CPU might get stuck here * and consume a lot of power until reset. * * @param val Breakpoint context for debugger, usually ignored. */ #ifdef DEVELHELP #include "architecture.h" #define DEBUG_BREAKPOINT(val) ARCHITECTURE_BREAKPOINT(val) #else #define DEBUG_BREAKPOINT(val) (void)0 #endif /** * @name Debugging defines * @{ */ /** * @brief This macro can be defined as 0 or other on a file-based level. * @ref DEBUG() will generate output only if ENABLE_DEBUG is non-zero. */ #if !defined(ENABLE_DEBUG) || defined(DOXYGEN) #define ENABLE_DEBUG 0 #endif /** * @def DEBUG_FUNC * * @brief Contains the function name if given compiler supports it. * Otherwise it is an empty string. */ # if defined(__cplusplus) && defined(__GNUC__) # define DEBUG_FUNC __PRETTY_FUNCTION__ # elif __STDC_VERSION__ >= 199901L # define DEBUG_FUNC __func__ # elif __GNUC__ >= 2 # define DEBUG_FUNC __FUNCTION__ # else # define DEBUG_FUNC "" # endif /** * @def DEBUG * * @brief Print debug information to stdout * * @note Another name for ::DEBUG_PRINT */ #define DEBUG(...) do { if (ENABLE_DEBUG) { DEBUG_PRINT(__VA_ARGS__); } } while (0) /** * @def DEBUG_PUTS * * @brief Print debug information to stdout using puts(), so no stack size * restrictions do apply. */ #define DEBUG_PUTS(str) do { if (ENABLE_DEBUG) { puts(str); } } while (0) /** @} */ /** * @def DEBUG_EXTRA_STACKSIZE * * @brief Extra stacksize needed when ENABLE_DEBUG==1 */ #if ENABLE_DEBUG #define DEBUG_EXTRA_STACKSIZE THREAD_EXTRA_STACKSIZE_PRINTF #else #define DEBUG_EXTRA_STACKSIZE (0) #endif #ifdef __cplusplus } #endif #endif /* DEBUG_H */ /** @} */