Sunday, May 1, 2011

How to enable the TRACE macro in Release mode?

The TRACE macro can be used to output diagnostic messages to the debugger when the code is compiled in Debug mode. I need the same messages while in Release mode. Is there a way to achieve this?

(Please do not waste your time discussing why I should not be using TRACE in Release mode :-)

From stackoverflow
  • In MFC, TRACE is defined as ATLTRACE. And in release mode that is defined as:

    define ATLTRACE __noop

    So, using the out-the-box TRACE from MFC, you won't actually be able to read any TRACE text, because it won't even be written out. You could write your own TRACE function instead, then re-define the TRACE macro. You could do something like this:

    void MyTrace(const CString& text)
    {
      ::OutputDebugString(text); // Outputs to console, same as regular TRACE
      // TODO: Do whatever output you need here. Write to event log / write to text file / write to pipe etc.
    }
    

    Hope that helps.

    Mark

  • TRACE is just a macro for OutputDebugString. So you can easily just make your own TRACE macro (or call it something else) that will call OutputDebugString.

  • Actually, the TRACE macro is a lot more flexible than OutputDebugString. It takes a printf() style format string and parameter list whereas OutputDebugString just takes a single string. In order to implement the full TRACE functionality in release mode you need to do something like this:

    void trace(const char* format, ...)
    {
       char buffer[1000];
    
       va_list argptr;
       va_start(argptr, format);
       wvsprintf(buffer, format, argptr);
       va_end(argptr);
    
       OutputDebugString(buffer);
    }
    
    Aardvark : I know this is just an example, but there is a huge buffer overrun potential in this example take as is.
  • A few years back I needed similar functionality so I cobbled together the following code. Just save it into a file, e.g. rtrace.h, include it at the end of your stdafx.h, and add _RTRACE to the release mode Preprocessor defines.

    Maybe someone will find a use for it :-)

    John

    #pragma once
    
    //------------------------------------------------------------------------------------------------
    //
    // Author:   John Cullen
    // Date:     2006/04/12
    // Based On: MSDN examples for variable argument lists and ATL implementation of TRACE.
    //
    // Description: Allows the use of TRACE statements in RELEASE builds, by overriding the
    // TRACE macro definition and redefining in terms of the RTRACE class and overloaded
    // operator (). Trace output is generated by calling OutputDebugString() directly.
    //
    //
    // Usage:    Add to the end of stdafx.h and add _RTRACE to the preprocessor defines (typically
    //           for RELEASE builds, although the flag will be ignored for DEBUG builds.
    //
    //------------------------------------------------------------------------------------------------
    
    #ifdef _DEBUG
    
    // NL defined as a shortcut for writing FTRACE(_T("\n")); for example, instead write FTRACE(NL);
    #define NL _T("\n") 
    #define LTRACE TRACE(_T("%s(%d): "), __FILE__, __LINE__); TRACE
    #define FTRACE TRACE(_T("%s(%d): %s: "), __FILE__, __LINE__, __FUNCTION__); TRACE
    
    #else   // _DEBUG
    
    #ifdef _RTRACE
    #undef TRACE
    #define TRACE  RTRACE()
    #define LTRACE RTRACE(__FILE__, __LINE__)
    #define FTRACE RTRACE(__FILE__, __LINE__, __FUNCTION__)
    #define NL _T("\n") 
    
    class RTRACE
    {
    public:
        // default constructor, no params
        RTRACE(void) : m_pszFileName( NULL ), m_nLineNo( 0 ), m_pszFuncName( NULL ) {};
    
        // overloaded constructor, filename and lineno
        RTRACE(PCTSTR const pszFileName, int nLineNo) :
         m_pszFileName(pszFileName), m_nLineNo(nLineNo), m_pszFuncName(NULL) {};
    
        // overloaded constructor, filename, lineno, and function name
        RTRACE(PCTSTR const pszFileName, int nLineNo, PCTSTR const pszFuncName) :
         m_pszFileName(pszFileName), m_nLineNo(nLineNo), m_pszFuncName(pszFuncName) {};
    
        virtual ~RTRACE(void) {};
    
        // no arguments passed, e.g. RTRACE()()
        void operator()() const
        {
         // no arguments passed, just dump the file, line and function if requested
         OutputFileAndLine();
         OutputFunction();
        }
    
        // format string and parameters passed, e.g. RTRACE()(_T("%s\n"), someStringVar)
        void operator()(const PTCHAR pszFmt, ...) const
        {
         // dump the file, line and function if requested, followed by the TRACE arguments
         OutputFileAndLine();
         OutputFunction();
    
         // perform the standard TRACE output processing
         va_list ptr; va_start( ptr, pszFmt );
         INT len = _vsctprintf( pszFmt, ptr ) + 1;
         TCHAR* buffer = (PTCHAR) malloc( len * sizeof(TCHAR) );
         _vstprintf( buffer, pszFmt, ptr );
         OutputDebugString(buffer);
         free( buffer );
        }
    
    private:
        // output the current file and line
        inline void OutputFileAndLine() const
        {
         if (m_pszFileName && _tcslen(m_pszFileName) > 0)
         {
          INT len = _sctprintf( _T("%s(%d): "), m_pszFileName, m_nLineNo ) + 1;
          PTCHAR buffer = (PTCHAR) malloc( len * sizeof(TCHAR) );
          _stprintf( buffer, _T("%s(%d): "), m_pszFileName, m_nLineNo );
          OutputDebugString( buffer );
          free( buffer );
         }
        }
    
        // output the current function name
        inline void OutputFunction() const
        {
         if (m_pszFuncName && _tcslen(m_pszFuncName) > 0)
         {
          INT len = _sctprintf( _T("%s: "), m_pszFuncName ) + 1;
          PTCHAR buffer = (PTCHAR) malloc( len * sizeof(TCHAR) );
          _stprintf( buffer, _T("%s: "), m_pszFuncName );
          OutputDebugString( buffer );
          free( buffer );
         }
        }
    
    private:
        PCTSTR const m_pszFuncName;
        PCTSTR const m_pszFileName;
        const int m_nLineNo;
    };
    
    #endif // _RTRACE
    
    #endif // NDEBUG
    
    

0 comments:

Post a Comment