GM6000 Digital Heater Controller Branch: main
SDX-1330
Trace.h
Go to the documentation of this file.
1#ifndef Cpl_System_Trace_h_
2#define Cpl_System_Trace_h_
3/*-----------------------------------------------------------------------------
4* This file is part of the Colony.Core Project. The Colony.Core Project is an
5* open source project with a BSD type of licensing agreement. See the license
6* agreement (license.txt) in the top/ directory or on the Internet at
7* http://integerfox.com/colony.core/license.txt
8*
9* Copyright (c) 2014-2022 John T. Taylor
10*
11* Redistributions of the source code must retain the above copyright notice.
12*----------------------------------------------------------------------------*/
13/** @file */
14
15
16#include "colony_config.h"
17#include "Cpl/Text/String.h"
18#include "Cpl/System/Thread.h"
19#include "Cpl/Io/Output.h"
20
21
22///
23namespace Cpl {
24///
25namespace System {
26
27/** This concrete class provide a 'printf' tracing mechanism. The output of
28 tracing messages typically defaults (dependent on the target platform)
29 to stdout. However, the output cat be redirected to any
30 Cpl::Stream::Output instance.
31
32 The trace engine is thread safe, i.e. it guarantees that outputting a single
33 trace message (info + user msg) is an Atomic operation within a multi-thread
34 environment.
35
36 All of the `methods SHOULD be access via the preprocessor macros below to
37 allow the 'trace code' to be compiled out of a 'release' build if desired.
38 In addition to a compile time decision to use/exclude tracing, there are
39 run-time switches to selective enable and/or control what is displayed.
40
41 There are numerous application customizable (at compile time) options
42 for how the internals of how the trace engine works (see OPTION_xxx
43 macros below).
44
45 NOTE: The trace engine supports does NOT support tracing BEFORE the Colony
46 library is initialized. This means that constructors of objects that
47 are created statically can NOT contain trace logging calls.
48
49 <pre>
50
51 For trace message to appear on the logging media the following
52 must done:
53
54 1. Enable Trace at compile time by defining the USE_CPL_SYSTEM_TRACE
55 preprocessor symbol.
56 2. Enable the Trace engine at run time by calling CPL_SYSTEM_TRACE_ENABLE()
57 3. Enable at least one 'section' by calling CPL_SYSTEM_TRACE_ENABLE_SECTION()
58
59
60
61 How to use the CPL_SYSTEM_TRACE_xxx macros
62 ------------------------------------------
63 The trace functionality is provided by the CPL_SYSTEM_TRACE and
64 CPL_SYSTEM_TRACEMSG macros. The remaining macros are used to enable/disable
65 and control the actual trace output. If the symbol, USE_CPL_SYSTEM_TRACE,
66 is NOT defined at compile time -- all of the trace macros are compiled
67 out. If the symbol is defined, then the trace output can be enabled
68 and disabled at run time.
69
70 CPL_SYSTEM_TRACE() - This macro is used to trace the entry and exit of a
71 scope block (a scope block is a pair of '{}').
72 For example:
73 void foobar( int x )
74 {
75 // Traces the entry & exit of foobar(), when
76 // tracing is enabled for section "bob"
77 CPL_SYSTEM_TRACE("bob");
78 int y = 2*x;
79 ...
80 }
81
82 CPL_SYSTEM_TRACEMSG() - This macro is for the programmer/application to
83 generate formatted message that are outputted to the
84 "trace media." The macro has printf() semantics.
85 The problem is that ANSI C++03 does not directly
86 support var_args in a macro - so you HAVE TO USE AN
87 EXTRA SET OF '()'.
88
89 For example:
90 void foobar( int x )
91 {
92 // Traces the entry & exit of foobar(), when
93 // tracing is enabled for section "bob"
94 CPL_SYSTEM_TRACE("bob");
95 int y = 2*x;
96 ...
97 // Trace message outputs when tracing is
98 // enabled for section "bob"
99 CPL_SYSTEM_TRACEMSG( "bob, ("This is an example Trace Msg. Y=%d, X=d", y, x) );
100 ...
101 }
102
103</pre> */
104class Trace
105{
106public:
107 /// Constructor
108 Trace( const char* filename, int linenum, const char* funcname, const char* section, const char* scope_name );
109
110 /// Destructor
112
113public:
114 /** This enum is used to globally control the amount of 'Prologue' information
115 that is prepended to the Trace log messages. The specifics of what
116 information is added is platform specific.
117 */
119 {
120 eNONE=0, //!< Turns off ALL prologue info
121 eBRIEF, //!< Default setting
122 eINFO, //!< Additional prologue info
123 eVERBOSE, //!< more prologue info
124 eMAX //!< Maximum prologue info
125 };
126
127public:
128 /** This function is used to trace the 'location' for general purpose trace
129 messages.
130
131 NOTE: NEVER call this method directly
132 */
133 static void traceLocation_( const char* section, const char* filename, int linenum, const char* funcname );
134
135 /** This function is used to generate general purpose trace messages.
136
137 NOTE: NEVER call this method directly
138 */
139 static void traceUserMsg_( const char* format, ... );
140
141public:
142 /** This method enables the output/logging of trace message at run-time.
143 The default is start with tracing enabled.
144
145 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
146 macros.
147 */
148 static void enable_( void );
149
150 /** This method disables the output/logging of trace message at run-time.
151 The default is start with tracing enabled.
152
153 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
154 macros.
155 */
156 static void disable_( void );
157
158 /** Returns true if tracing is enabled, i.e. enable_() has been called.,
159
160 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
161 macros.
162 */
163 static bool isEnabled_( void );
164
165 /** This method sets the information level for the trace messages. The method
166 returns the previous level setting.
167
168 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
169 macros.
170 */
172
173 /** This method returns the current information level setting
174
175 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
176 macros.
177 */
178 static InfoLevel_T getInfoLevel_( void ) noexcept;
179
180 /** This method enables the output/logging of trace message at run-time for
181 the specified 'section'. A 'section' name can NOT contain any white
182 space. The method returns true if the section was enabled; else false
183 is return, i.e. exceeded max active enabled sections.
184
185 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
186 macros.
187 */
188 static bool enableSection_( const char* sectionToEnable );
189
190 /** This method disables the output/logging of trace message at run-time for
191 the specified 'section'. Setting 'sectionToDisable' to 0 will disable
192 all sections.
193
194 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
195 macros.
196 */
197 static void disableSection_( const char* sectionToDisable );
198
199 /** Returns true if tracing is enabled AND the specified 'section' has been
200 enabled.
201
202 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
203 macros.
204 */
205 static bool isSectionEnabled_( const char* section );
206
207 /** This method returns the number of enabled 'sections' and returns the
208 actual section name(s) via the String 'dst'. It is the caller
209 responsibility to ensure that 'dst' is large enough to hold all of the
210 enable sections. If 'dst' is to short, the results will be truncated.
211
212 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
213 macros.
214 */
215 static unsigned getSections_( Cpl::Text::String& dst );
216
217 /** This method applies a filter by thread name when generating output, i.e.
218 for a message to be output, trace must be enabled, the message's section
219 enabled, and must be running in the one of the threads specified
220 by the thread filter.
221
222 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
223 macros.
224 */
225 static void setThreadFilter_( const char* threadName1,
226 const char* threadName2 =0,
227 const char* threadName3 =0,
228 const char* threadName4 =0
229 );
230
231 /** This method returns the number of enabled 'thread filters' and returns
232 the actual thread name(s) via the String 'dst'. It is the caller
233 responsibility to ensure that 'dst' is large enough to hold all of the
234 enable thread filters. If 'dst' is to short, the results will be
235 truncated.
236
237 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
238 macros.
239 */
240 static unsigned getThreadFilters_( Cpl::Text::String& dst );
241
242 /** This method removes that thread filter (if there is one present). If
243 no filter has been applied when this method is called, then this
244 method does nothing.
245
246 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
247 macros.
248 */
249 static void clearThreadFilter_( void );
250
251 /** This method returns true if the thread filter has been set and the
252 current thread's name 'passes' the thread filter check, i.e. the specified
253 name match one of the names in the filter list.
254
255 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
256 macros.
257 */
258 static bool passedThreadFilter_();
259
260
261public:
262 /** This method is used to redirect the trace output to the provided Output
263 stream. There is no guaranty on what happens to trace message(s) during
264 the transition to the new output/media destination.
265
266 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
267 macros.
268 */
269 static void redirect_( Cpl::Io::Output& newMedia );
270
271 /** This method is used to revert the trace output to its default output
272 destination/stream/media. There is no guaranty on what happens to trace
273 message(s) during the transition to the default output/media
274 destination.
275
276 NOTE: NEVER call this method directly -->use the CPL_SYSTEM_TRACE_xxx()
277 macros.
278 */
279 static void revert_( void );
280
281public:
282 /** This COMPONENT Scoped method provides the Output stream to the Trace
283 Engine. The Output stream instance MUST be created statically. However,
284 the Output stream does not necessarily need to fully 'operational' when
285 statically constructed - it just need to be "operationally" by the time
286 Cpl::System::Api::initialize() is called.
287
288 NOTE: The Application is RESPONSIBLE for implementing this method!
289 */
290 static Cpl::Io::Output* getDefaultOutputStream_( void ) noexcept;
291
292
293protected:
294 /// Caches trace info for exit message
295 const char* m_filename;
296
297 /// Caches trace info for exit message
299
300 /// Caches trace info for exit message
301 const char* m_funcname;
302
303 /// Caches trace info for exit message
304 const char* m_section;
305
306 /// Caches trace info for exit message
307 const char* m_scope;
308};
309
310};
311}; // end namespaces
312
313//////////////////////////////////////////////////////////////////////////////
314#ifdef USE_CPL_SYSTEM_TRACE
315
316
317// PRETTY_FUNCTION macro is non-standard
318#if defined(__GNUC__)
319/// Take advantage of GCC's pretty function symbol
320#define CPL_SYSTEM_TRACE_PRETTY_FUNCNAME __PRETTY_FUNCTION__
321
322#elif defined(_MSC_VER)
323/// Take advantage of Microsoft's pretty function symbol
324#define CPL_SYSTEM_TRACE_PRETTY_FUNCNAME __FUNCSIG__
325
326#else
327/// Use C++11 function name
328
329#define CPL_SYSTEM_TRACE_PRETTY_FUNCNAME __func__
330#endif // end __PRETTY_FUNCTION__
331
332
333 /// Macro Wrapper
334#define CPL_SYSTEM_TRACE_FUNC(sect) Cpl::System::Trace cplSystemTraceInstance_ (__FILE__,__LINE__,CPL_SYSTEM_TRACE_PRETTY_FUNCNAME,sect,CPL_SYSTEM_TRACE_PRETTY_FUNCNAME)
335
336/// Macro Wrapper
337#define CPL_SYSTEM_TRACE_SCOPE(sect,label) Cpl::System::Trace cplSystemTraceInstance_ (__FILE__,__LINE__,CPL_SYSTEM_TRACE_PRETTY_FUNCNAME,sect,label)
338
339/// Macro Wrapper
340#define CPL_SYSTEM_TRACE_MSG(sect, var_args) do { if ( Cpl::System::Trace::isSectionEnabled_(sect) && Cpl::System::Trace::passedThreadFilter_() ) {Cpl::System::Trace::traceLocation_(sect,__FILE__,__LINE__,CPL_SYSTEM_TRACE_PRETTY_FUNCNAME);Cpl::System::Trace::traceUserMsg_ var_args;}} while(0)
341
342/// Macro Wrapper
343#define CPL_SYSTEM_TRACE_ENABLE() Cpl::System::Trace::enable_()
344
345/// Macro Wrapper
346#define CPL_SYSTEM_TRACE_DISABLE() Cpl::System::Trace::disable_()
347
348/// Macro Wrapper
349#define CPL_SYSTEM_TRACE_IS_ENABLED() Cpl::System::Trace::isEnabled_()
350
351/// Macro Wrapper
352#define CPL_SYSTEM_TRACE_IS_SECTION_ENABLED(sect) Cpl::System::Trace::isSectionEnabled(sect)
353
354/// Macro Wrapper
355#define CPL_SYSTEM_TRACE_SET_INFO_LEVEL(l) Cpl::System::Trace::setInfoLevel_(l)
356
357/// Macro Wrapper
358#define CPL_SYSTEM_TRACE_GET_INFO_LEVEL() Cpl::System::Trace::getInfoLevel_()
359
360/// Macro Wrapper
361#define CPL_SYSTEM_TRACE_ENABLE_SECTION(sect) Cpl::System::Trace::enableSection_(sect)
362
363/// Macro Wrapper
364#define CPL_SYSTEM_TRACE_DISABLE_SECTION(sect) Cpl::System::Trace::disableSection_(sect)
365
366/// Macro Wrapper
367#define CPL_SYSTEM_TRACE_GET_SECTIONS(dst) Cpl::System::Trace::getSections_(dst)
368
369/// Macro Wrapper
370#define CPL_SYSTEM_TRACE_SET_THREAD_FILTER(t1) Cpl::System::Trace::setThreadFilter_(t1)
371
372/// Macro Wrapper
373#define CPL_SYSTEM_TRACE_SET_THREAD_2FILTERS(t1,t2) Cpl::System::Trace::setThreadFilter_(t1,t2)
374
375/// Macro Wrapper
376#define CPL_SYSTEM_TRACE_SET_THREAD_3FILTERS(t1,t2,t3) Cpl::System::Trace::setThreadFilter_(t1,t2,t3)
377
378/// Macro Wrapper
379#define CPL_SYSTEM_TRACE_SET_THREAD_4FILTERS(t1,t2,t3,t4) Cpl::System::Trace::setThreadFilter_(t1,t2,t3,t4)
380
381/// Macro Wrapper
382#define CPL_SYSTEM_TRACE_GET_THREAD_FILTERS(dst) Cpl::System::Trace::getThreadFilters_(dst)
383
384/// Macro Wrapper
385#define CPL_SYSTEM_TRACE_CLEAR_THREAD_FILTER() Cpl::System::Trace::clearThreadFilter_()
386
387/// Macro Wrapper
388#define CPL_SYSTEM_TRACE_DISABLE_SECTION(sect) Cpl::System::Trace::disableSection_(sect)
389
390/// Macro Wrapper
391#define CPL_SYSTEM_TRACE_REDIRECT(newdst) Cpl::System::Trace::redirect_(newdst)
392
393/// Macro Wrapper
394#define CPL_SYSTEM_TRACE_REVERT() Cpl::System::Trace::revert_()
395
396/// Returns true if TRACE was activated/enabled at compile time
397#define CPL_SYSTEM_TRACE_IS_COMPILED() true
398
399/// Allocate a variable that is only 'used' when tracing is enabled
400#define CPL_SYSTEM_TRACE_ALLOCATE(type,varname,initval) type varname = initval
401
402
403#else
404 /// Macro Wrapper
405#define CPL_SYSTEM_TRACE_FUNC(sect)
406
407/// Macro Wrapper
408#define CPL_SYSTEM_TRACE_SCOPE(sect,label)
409
410/// Macro Wrapper
411#define CPL_SYSTEM_TRACE_MSG(sect, var_args)
412
413/// Macro Wrapper
414#define CPL_SYSTEM_TRACE_ENABLE()
415
416/// Macro Wrapper
417#define CPL_SYSTEM_TRACE_DISABLE()
418
419/// Macro Wrapper
420#define CPL_SYSTEM_TRACE_IS_ENABLED() false
421
422/// Macro Wrapper
423#define CPL_SYSTEM_TRACE_IS_SECTION_ENABLED(sect) false
424
425/// Macro Wrapper
426#define CPL_SYSTEM_TRACE_SET_INFO_LEVEL(l)
427
428/// Macro Wrapper
429#define CPL_SYSTEM_TRACE_GET_INFO_LEVEL() ((Cpl::System::Trace::InfoLevel_T)0)
430
431/// Macro Wrapper
432#define CPL_SYSTEM_TRACE_ENABLE_SECTION(sect)
433
434/// Macro Wrapper
435#define CPL_SYSTEM_TRACE_DISABLE_SECTION(sect)
436
437/// Macro Wrapper
438#define CPL_SYSTEM_TRACE_SET_THREAD_FILTER(t1)
439
440/// Macro Wrapper
441#define CPL_SYSTEM_TRACE_SET_THREAD_2FILTERS(t1,t2)
442
443/// Macro Wrapper
444#define CPL_SYSTEM_TRACE_SET_THREAD_3FILTERS(t1,t2,t3)
445
446/// Macro Wrapper
447#define CPL_SYSTEM_TRACE_SET_THREAD_4FILTERS(t1,t2,t3,t4)
448
449/// Macro Wrapper
450#define CPL_SYSTEM_TRACE_GET_THREAD_FILTERS(dst) 0
451
452/// Macro Wrapper
453#define CPL_SYSTEM_TRACE_CLEAR_THREAD_FILTER()
454
455/// Macro Wrapper
456#define CPL_SYSTEM_TRACE_GET_SECTIONS(dst) 0
457
458/// Macro Wrapper
459#define CPL_SYSTEM_TRACE_REDIRECT(newdst)
460
461/// Macro Wrapper
462#define CPL_SYSTEM_TRACE_REVERT()
463
464/// Returns false if TRACE was NOT activated/enabled at compile time
465#define CPL_SYSTEM_TRACE_IS_COMPILED() false
466
467/// Macro Wrapper
468#define CPL_SYSTEM_TRACE_ALLOCATE(type,varname,initval)
469
470#endif // USE_CPL_SYSTEM_TRACE
471
472
473//////////////////////////////////////////////////////////////////////////////
474/** The default maximum number of actively enabled 'sections' supported.
475 */
476#ifndef OPTION_CPL_SYSTEM_TRACE_MAX_SECTIONS
477#define OPTION_CPL_SYSTEM_TRACE_MAX_SECTIONS 8
478#endif
479
480 /** The default maximum size, in bytes, for a section name (not including
481 the null terminator).
482 */
483#ifndef OPTION_CPL_SYSTEM_TRACE_MAX_SECTION_NAME_LEN
484#define OPTION_CPL_SYSTEM_TRACE_MAX_SECTION_NAME_LEN 63
485#endif
486
487 /** The size, in bytes (without the null terminator), of trace engine's internal
488 String use to construct a complete (info + user msg) trace message.
489 */
490#ifndef OPTION_CPL_SYSTEM_TRACE_MAX_BUFFER
491#define OPTION_CPL_SYSTEM_TRACE_MAX_BUFFER 511
492#endif
493
494
495 /** String literal that is the prefix for all trace messages
496 */
497#ifndef OPTION_CPL_SYSTEM_TRACE_PREFIX_STRING
498#define OPTION_CPL_SYSTEM_TRACE_PREFIX_STRING ">> "
499#endif
500
501
502 /** String literal that is the final suffix for all trace messages
503 */
504#ifndef OPTION_CPL_SYSTEM_TRACE_SUFFIX_STRING
505#define OPTION_CPL_SYSTEM_TRACE_SUFFIX_STRING "\n"
506#endif
507
508 /** The compile time default setting for the info level
509 */
510#ifndef OPTION_CPL_SYSTEM_TRACE_DEFAULT_INFO_LEVEL
511#define OPTION_CPL_SYSTEM_TRACE_DEFAULT_INFO_LEVEL Cpl::System::Trace::eBRIEF
512#endif
513
514 /** The compile time default setting what enable/disable state the trace
515 starts in
516 */
517#ifndef OPTION_CPL_SYSTEM_TRACE_DEFAULT_ENABLE_STATE
518#define OPTION_CPL_SYSTEM_TRACE_DEFAULT_ENABLE_STATE true
519#endif
520
521
522#endif // end header latch
523
This partially abstract class defines a interface for operating on an output stream (example of a str...
Definition Output.h:34
This concrete class provide a 'printf' tracing mechanism.
Definition Trace.h:105
static InfoLevel_T getInfoLevel_(void) noexcept
This method returns the current information level setting.
static void enable_(void)
This method enables the output/logging of trace message at run-time.
static unsigned getSections_(Cpl::Text::String &dst)
This method returns the number of enabled 'sections' and returns the actual section name(s) via the S...
static InfoLevel_T setInfoLevel_(InfoLevel_T newLevel)
This method sets the information level for the trace messages.
const char * m_section
Caches trace info for exit message.
Definition Trace.h:304
static unsigned getThreadFilters_(Cpl::Text::String &dst)
This method returns the number of enabled 'thread filters' and returns the actual thread name(s) via ...
const char * m_filename
Caches trace info for exit message.
Definition Trace.h:295
const char * m_funcname
Caches trace info for exit message.
Definition Trace.h:301
int m_linenum
Caches trace info for exit message.
Definition Trace.h:298
static bool isSectionEnabled_(const char *section)
Returns true if tracing is enabled AND the specified 'section' has been enabled.
static void setThreadFilter_(const char *threadName1, const char *threadName2=0, const char *threadName3=0, const char *threadName4=0)
This method applies a filter by thread name when generating output, i.e.
~Trace()
Destructor.
static void revert_(void)
This method is used to revert the trace output to its default output destination/stream/media.
static void traceLocation_(const char *section, const char *filename, int linenum, const char *funcname)
This function is used to trace the 'location' for general purpose trace messages.
static bool enableSection_(const char *sectionToEnable)
This method enables the output/logging of trace message at run-time for the specified 'section'.
Trace(const char *filename, int linenum, const char *funcname, const char *section, const char *scope_name)
Constructor.
static void disableSection_(const char *sectionToDisable)
This method disables the output/logging of trace message at run-time for the specified 'section'.
InfoLevel_T
This enum is used to globally control the amount of 'Prologue' information that is prepended to the T...
Definition Trace.h:119
@ eNONE
Turns off ALL prologue info.
Definition Trace.h:120
@ eINFO
Additional prologue info.
Definition Trace.h:122
@ eMAX
Maximum prologue info.
Definition Trace.h:124
@ eVERBOSE
more prologue info
Definition Trace.h:123
@ eBRIEF
Default setting.
Definition Trace.h:121
static void redirect_(Cpl::Io::Output &newMedia)
This method is used to redirect the trace output to the provided Output stream.
const char * m_scope
Caches trace info for exit message.
Definition Trace.h:307
static bool isEnabled_(void)
Returns true if tracing is enabled, i.e.
static void disable_(void)
This method disables the output/logging of trace message at run-time.
static void traceUserMsg_(const char *format,...)
This function is used to generate general purpose trace messages.
static void clearThreadFilter_(void)
This method removes that thread filter (if there is one present).
static bool passedThreadFilter_()
This method returns true if the thread filter has been set and the current thread's name 'passes' the...
static Cpl::Io::Output * getDefaultOutputStream_(void) noexcept
This COMPONENT Scoped method provides the Output stream to the Trace Engine.
This abstract class defines the operations that can be before on a NULL terminated string.
Definition String.h:40
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20