#if !defined( INCLUDED_DEBUGGING_DEBUGGING_H ) #define INCLUDED_DEBUGGING_DEBUGGING_H /// \file /// \brief Debugging macros for fatal error/assert messages. #include "stream/textstream.h" #include "warnings.h" #include "generic/static.h" #if defined( _MSC_VER ) && defined( _M_IX86 ) #define DEBUGGER_BREAKPOINT() __asm { int 3 } #elif defined ( __i386__ ) && defined ( __GNUC__ ) && __GNUC__ >= 2 #define DEBUGGER_BREAKPOINT() __asm__ __volatile__ ( "int $03" ) #else #include #define DEBUGGER_BREAKPOINT() raise( SIGTRAP ); #endif #define FILE_LINE __FILE__ ":" << __LINE__ #if defined( _DEBUG ) || 1 #define DEBUG_ASSERTS #endif class DebugMessageHandler { public: virtual TextOutputStream& getOutputStream() = 0; virtual bool handleMessage() = 0; }; class NullDebugMessageHandler : public NullOutputStream, public DebugMessageHandler { public: virtual TextOutputStream& getOutputStream(){ return *this; } virtual bool handleMessage(){ return false; } }; class DefaultDebugMessageHandler : public DebugMessageHandler { public: virtual TextOutputStream& getOutputStream(){ return globalErrorStream(); } virtual bool handleMessage(){ #if defined( _DEBUG ) return false; // send debug-break #else return true; #endif } }; class DebugMessageHandlerRef : public DefaultDebugMessageHandler { DebugMessageHandler* m_handler; public: DebugMessageHandlerRef() : m_handler( this ){ } void setHandler( DebugMessageHandler& handler ){ m_handler = &handler; } DebugMessageHandler& getHandler(){ return *m_handler; } }; typedef Static GlobalDebugMessageHandler; inline DebugMessageHandler& globalDebugMessageHandler(){ return GlobalDebugMessageHandler::instance().getHandler(); } #if defined( DEBUG_ASSERTS ) /// \brief Sends a \p message to the current debug-message-handler text-output-stream if \p condition evaluates to false. #define ASSERT_MESSAGE( condition, message ) \ if ( !( condition ) ) \ { \ globalDebugMessageHandler().getOutputStream() << FILE_LINE << "\nassertion failure: " << message << "\n"; \ if ( !globalDebugMessageHandler().handleMessage() ) { DEBUGGER_BREAKPOINT(); } \ } else \ /// \brief Sends a \p message to the current debug-message-handler text-output-stream. #define ERROR_MESSAGE( message ) \ globalDebugMessageHandler().getOutputStream() << FILE_LINE << "\nruntime error: " << message << "\n"; \ if ( !globalDebugMessageHandler().handleMessage() ) { DEBUGGER_BREAKPOINT(); } else \ #define ASSERT_NOTNULL( ptr ) ASSERT_MESSAGE( ptr != 0, "pointer \"" # ptr "\" is null" ) #else #define ASSERT_MESSAGE( condition, message ) #define ASSERT_NOTNULL( ptr ) #endif #endif