Version: SMASH-3.2
einhard.hpp
Go to the documentation of this file.
1 
62 #ifndef EINHARD_HPP_
63 #define EINHARD_HPP_
64 
65 #include <assert.h>
66 #include <iostream>
67 #include <iomanip>
68 #include <ctime>
69 #include <cstring>
70 #include <sstream>
71 #include <bitset>
72 
73 #include "stacktrace.h"
74 
75 // This C header is sadly required to check whether writing to a terminal or a file
76 #include <cstdio>
77 
78 #include <unistd.h> // for isatty
79 
80 #ifdef __GNUC__
81 #define EINHARD_ALWAYS_INLINE_ __attribute__((always_inline))
82 #else
83 #define EINHARD_ALWAYS_INLINE_
84 #endif
85 
86 // Error on MacOS:
87 // “thread-local storage is unsupported for the current target”
88 // To enable a workaround the EINHARD_NO_THREAD_LOCAL macro must be defined.
89 #ifdef __APPLE__
90 #define EINHARD_NO_THREAD_LOCAL 1
91 #endif
92 
96 namespace einhard
97 {
101  extern char const VERSION[];
102 
108  enum LogLevel
109  {
110  ALL,
117  OFF
118  };
119 
125  template <LogLevel> const char *getLogLevelString() noexcept;
129  const char *getLogLevelString( LogLevel level );
130 
139  LogLevel getLogLevel( const std::string &level );
140 
141  template <LogLevel> const char *colorForLogLevel() noexcept;
142 
146  template <typename Parent> class Color
147  {
148  public:
150  EINHARD_ALWAYS_INLINE_ Color() noexcept : reset( true )
151  {
152  }
156  {
157  return {false};
158  }
159  EINHARD_ALWAYS_INLINE_ char const *ansiCode() const noexcept
160  {
161  return Parent::ANSI();
162  }
163  EINHARD_ALWAYS_INLINE_ bool resetColor() const noexcept
164  {
165  return reset;
166  }
167 
168  private:
169  EINHARD_ALWAYS_INLINE_ Color( bool r ) noexcept : reset( r )
170  {
171  }
172  bool reset;
173  };
174 #define _COLOR( name, code ) \
175  struct name##_t_ \
176  { \
177  static char const *ANSI() noexcept \
178  { \
179  return "\33[" code "m"; \
180  } \
181  }; \
182  typedef Color<name##_t_> name
183 
184  _COLOR(DGray, "01;30");
185  _COLOR(Black, "00;30");
186  _COLOR(Red, "01;31");
187  _COLOR(DRed, "00;31");
188  _COLOR(Green, "01;32");
189  _COLOR(DGreen, "00;32");
190  _COLOR(Yellow, "01;33");
191  _COLOR(Orange, "00;33");
192  _COLOR(Blue, "01;34");
193  _COLOR(DBlue, "00;34");
194  _COLOR(Magenta, "01;35");
195  _COLOR(DMagenta,"00;35");
196  _COLOR(Cyan, "01;36");
197  _COLOR(DCyan, "00;36");
198  _COLOR(White, "01;37");
199  _COLOR(Gray, "00;37");
200  _COLOR(NoColor, "0" );
201 #undef _COLOR
202 
208  {
209  template <typename T> EINHARD_ALWAYS_INLINE_ DummyOutputFormatter &operator<<( const T & ) noexcept
210  {
211  return *this;
212  }
214  std::ostream &( * )( std::ostream & ) ) noexcept
215  {
216  return *this;
217  }
218  };
219 
221  {
222  private:
223  // Pointer to the thread_local stringstream (if enabled)
224  std::ostringstream *out;
225 #ifdef EINHARD_NO_THREAD_LOCAL
226  // without thread_local we simply use a local stringstream object
227  std::ostringstream realOut;
228 #endif
229  // The number of chars required for aligning
230  unsigned char indent;
231  // Whether to colorize the output
232  const bool colorize;
233  // Whether the color needs to be reset with the next operator<<
234  bool resetColor = false;
235 
236  public:
237  template <LogLevel VERBOSITY>
238  EINHARD_ALWAYS_INLINE_ UnconditionalOutput( bool colorize_, const char *areaName,
239  std::integral_constant<LogLevel, VERBOSITY> )
240  : colorize( colorize_ )
241  {
242  doInit<VERBOSITY>( areaName );
243  }
244 
245  template <typename T> UnconditionalOutput &operator<<( const Color<T> &col )
246  {
247  if( colorize )
248  {
249  *out << col.ansiCode();
250  resetColor = col.resetColor();
251  }
252  return *this;
253  }
254 
255  EINHARD_ALWAYS_INLINE_ UnconditionalOutput &operator<<( std::ostream &( *manip )( std::ostream & ) )
256  {
257  *out << manip;
258  return *this;
259  }
260 
261  template <typename T> EINHARD_ALWAYS_INLINE_ UnconditionalOutput &operator<<( const T &msg )
262  {
263  *out << msg;
264  checkColorReset();
265  return *this;
266  }
267 
268  void doCleanup(std::FILE *outfile = stdout) noexcept;
269 
270  protected:
271  EINHARD_ALWAYS_INLINE_ UnconditionalOutput( bool colorize_ ) : colorize( colorize_ )
272  {
273  }
274  template <LogLevel VERBOSITY> void doInit( const char *areaName );
276  };
281  {
282  private:
283  // the output file (normally used to switch from stdout to stderr)
284  std::FILE *const outfile;
285  // Whether output is enabled
286  const bool enabled;
287 
288  public:
289  OutputFormatter( const OutputFormatter & ) = delete;
291 
292  template <LogLevel VERBOSITY>
293  EINHARD_ALWAYS_INLINE_ OutputFormatter( bool enabled_, bool const colorize_,
294  const char *areaName,
295  std::integral_constant<LogLevel, VERBOSITY>,
296  std::FILE *outfile_ = stdout )
297  : UnconditionalOutput( colorize_ ), outfile( outfile_ ), enabled( enabled_ )
298  {
299  if( enabled )
300  {
301  doInit<VERBOSITY>( areaName );
302  }
303  }
304 
305  template <typename T> OutputFormatter &operator<<( const Color<T> &col )
306  {
307  if( enabled )
308  {
310  }
311  return *this;
312  }
313 
314  EINHARD_ALWAYS_INLINE_ OutputFormatter &operator<<( std::ostream &( *manip )( std::ostream & ) )
315  {
316  if( enabled )
317  {
319  }
320  return *this;
321  }
322 
323  template <typename T> EINHARD_ALWAYS_INLINE_ OutputFormatter &operator<<( const T &msg )
324  {
325  if( enabled )
326  {
328  }
329  return *this;
330  }
331 
334  {
335  if( enabled )
336  {
337  doCleanup( outfile );
338  }
339  }
340  };
341 
351  template<LogLevel MAX = ALL> class Logger
352  {
353  private:
354  char areaName[32 - sizeof( LogLevel ) - sizeof( bool )] = {'\0'};
358 
359  public:
366  Logger( const LogLevel verbosity_in = WARN ) : verbosity( verbosity_in )
367  {
368  // use some, sadly not c++-ways to figure out whether we are writing ot a terminal
369  // only colorize when we are writing ot a terminal
370  colorize_stdout = isatty( fileno( stdout ) );
371  colorize_stderr = isatty( fileno( stderr ) );
372  };
379  Logger( const LogLevel verbosity_in, const bool colorize )
380  : verbosity( verbosity_in ), colorize_stdout( colorize ), colorize_stderr( colorize ) {};
381 
391  void setAreaName( const char *name )
392  {
393  assert( name );
394  std::strncpy( &areaName[0], name, sizeof( areaName ) - 1 );
395  areaName[sizeof( areaName ) - 1] = '\0';
396  }
398  void setAreaName( const std::string &name )
399  {
400  setAreaName(name.c_str());
401  }
402 
404 #ifdef NDEBUG
405  DummyOutputFormatter trace() const noexcept
406  {
407  return DummyOutputFormatter();
408  }
409 
410  template <typename... Ts> void trace( Ts &&... ) const noexcept
411  {
412  }
413 #else
415  {
416  return {isEnabled<TRACE>(), colorize_stdout, areaName,
417  std::integral_constant<LogLevel, TRACE>()};
418  }
419 
420  template <typename... Ts> void trace( Ts &&... args ) const noexcept
421  {
422  if( isEnabled<TRACE>() )
423  {
425  std::integral_constant<LogLevel, TRACE>()};
426  ((o << args), ...);
427  o.doCleanup();
428  }
429  }
430 #endif
432 #ifdef NDEBUG
433  DummyOutputFormatter debug() const noexcept
434  {
435  return DummyOutputFormatter();
436  }
437  template <typename... Ts> void debug( Ts &&... ) const noexcept
438  {
439  }
440 #else
442  {
443  return {isEnabled<DEBUG>(), colorize_stdout, areaName,
444  std::integral_constant<LogLevel, DEBUG>()};
445  }
446  template <typename... Ts> void debug( Ts &&... args ) const noexcept
447  {
448  if( isEnabled<DEBUG>() )
449  {
451  std::integral_constant<LogLevel, DEBUG>()};
452  ((o << args), ...);
453  o.doCleanup();
454  }
455  }
456 #endif
459  {
460  return {isEnabled<INFO>(), colorize_stdout, areaName,
461  std::integral_constant<LogLevel, INFO>()};
462  }
463  template <typename... Ts> void info( Ts &&... args ) const noexcept
464  {
465  if( isEnabled<INFO>() )
466  {
468  std::integral_constant<LogLevel, INFO>()};
469  ((o << args), ...);
470  o.doCleanup();
471  }
472  }
475  {
476  return {isEnabled<WARN>(), colorize_stderr, areaName,
477  std::integral_constant<LogLevel, WARN>(), stderr};
478  }
479  template <typename... Ts> void warn( Ts &&... args ) const noexcept
480  {
481  if( isEnabled<WARN>() )
482  {
484  std::integral_constant<LogLevel, WARN>()};
485  ((o << args), ...);
486  o.doCleanup(stderr);
487  }
488  }
491  {
492  return {isEnabled<ERROR>(), colorize_stderr, areaName,
493  std::integral_constant<LogLevel, ERROR>(), stderr};
494  }
495  template <typename... Ts> void error( Ts &&... args ) const noexcept
496  {
497  if( isEnabled<ERROR>() )
498  {
500  std::integral_constant<LogLevel, ERROR>()};
501  ((o << args), ...);
502  o.doCleanup(stderr);
503  }
504  }
507  {
508  return {isEnabled<FATAL>(), colorize_stderr, areaName,
509  std::integral_constant<LogLevel, FATAL>(), stderr};
510  }
512  template <typename... Ts> void fatal( Ts &&... args ) const noexcept
513  {
514  if( isEnabled<FATAL>() )
515  {
517  std::integral_constant<LogLevel, FATAL>()};
518  ((o << args), ...);
519  o.doCleanup(stderr);
520  print_stacktrace(stderr, 63, 2);
521  }
522  }
523 
524  template <LogLevel LEVEL> bool isEnabled() const noexcept
525  {
526 #ifdef NDEBUG
527  if( LEVEL == DEBUG || LEVEL == TRACE ) {
528  return false;
529  }
530 #endif
531  return ( MAX <= LEVEL && verbosity <= LEVEL );
532  }
533 
539  inline void setVerbosity( LogLevel verbosity_in ) noexcept
540  {
541  this->verbosity = verbosity_in;
542  }
549  inline LogLevel getVerbosity() const noexcept
550  {
551  return this->verbosity;
552  }
556  inline char const * getVerbosityString( ) const
557  {
558  return getLogLevelString(verbosity);
559  }
563  void setColorize( bool colorize ) noexcept
564  {
565  this->colorize_stdout = colorize;
566  this->colorize_stderr = colorize;
567  }
571  bool getColorize() const noexcept
572  {
573  return this->colorize_stdout || this->colorize_stderr;
574  }
575  };
576 }
577 
578 #endif
579 
580 // vim: ts=4 sw=4 tw=100 noet
A stream modifier that allows to colorize the log output.
Definition: einhard.hpp:147
EINHARD_ALWAYS_INLINE_ Color(bool r) noexcept
Definition: einhard.hpp:169
EINHARD_ALWAYS_INLINE_ bool resetColor() const noexcept
Definition: einhard.hpp:163
EINHARD_ALWAYS_INLINE_ Color() noexcept
The default color modifier only affects the next object in the stream.
Definition: einhard.hpp:150
EINHARD_ALWAYS_INLINE_ char const * ansiCode() const noexcept
Definition: einhard.hpp:159
EINHARD_ALWAYS_INLINE_ Color< Parent > operator~() const noexcept
With the ~ operator the color modifier affects the rest of the stream (or until another color object ...
Definition: einhard.hpp:155
A Logger object can be used to output messages to stdout.
Definition: einhard.hpp:352
EINHARD_ALWAYS_INLINE_ void setAreaName(const std::string &name)
Definition: einhard.hpp:398
void info(Ts &&... args) const noexcept
Definition: einhard.hpp:463
bool colorize_stderr
Definition: einhard.hpp:357
OutputFormatter error() const
Access to the error message stream.
Definition: einhard.hpp:490
LogLevel verbosity
Definition: einhard.hpp:355
void setVerbosity(LogLevel verbosity_in) noexcept
Modify the verbosity of the Logger.
Definition: einhard.hpp:539
void fatal(Ts &&... args) const noexcept
Will print stacktrace.
Definition: einhard.hpp:512
void debug(Ts &&... args) const noexcept
Definition: einhard.hpp:446
OutputFormatter fatal() const
Access to the fatal message stream.
Definition: einhard.hpp:506
char const * getVerbosityString() const
Retrieve a human readable representation of the current log level.
Definition: einhard.hpp:556
OutputFormatter info() const
Access to the info message stream.
Definition: einhard.hpp:458
void error(Ts &&... args) const noexcept
Definition: einhard.hpp:495
bool isEnabled() const noexcept
Definition: einhard.hpp:524
void warn(Ts &&... args) const noexcept
Definition: einhard.hpp:479
LogLevel getVerbosity() const noexcept
Retrieve the current log level.
Definition: einhard.hpp:549
OutputFormatter debug() const
Access to the debug message stream.
Definition: einhard.hpp:441
OutputFormatter trace() const
Access to the trace message stream.
Definition: einhard.hpp:414
void setAreaName(const char *name)
Set an area name.
Definition: einhard.hpp:391
char areaName[32 - sizeof(LogLevel) - sizeof(bool)]
Definition: einhard.hpp:354
Logger(const LogLevel verbosity_in, const bool colorize)
Create a new Logger object explicitly selecting whether to colorize the output or not.
Definition: einhard.hpp:379
void setColorize(bool colorize) noexcept
Select whether the output stream should be colorized.
Definition: einhard.hpp:563
OutputFormatter warn() const
Access to the warning message stream.
Definition: einhard.hpp:474
void trace(Ts &&... args) const noexcept
Definition: einhard.hpp:420
Logger(const LogLevel verbosity_in=WARN)
Create a new Logger object.
Definition: einhard.hpp:366
bool colorize_stdout
Definition: einhard.hpp:356
bool getColorize() const noexcept
Check whether the output stream is colorized.
Definition: einhard.hpp:571
A wrapper for the output stream taking care proper formatting and colorization of the output.
Definition: einhard.hpp:281
OutputFormatter(const OutputFormatter &)=delete
EINHARD_ALWAYS_INLINE_ OutputFormatter(bool enabled_, bool const colorize_, const char *areaName, std::integral_constant< LogLevel, VERBOSITY >, std::FILE *outfile_=stdout)
Definition: einhard.hpp:293
std::FILE *const outfile
Definition: einhard.hpp:284
OutputFormatter & operator<<(const Color< T > &col)
Definition: einhard.hpp:305
OutputFormatter(OutputFormatter &&)=default
EINHARD_ALWAYS_INLINE_ OutputFormatter & operator<<(const T &msg)
Definition: einhard.hpp:323
EINHARD_ALWAYS_INLINE_ ~OutputFormatter()
Definition: einhard.hpp:333
EINHARD_ALWAYS_INLINE_ OutputFormatter & operator<<(std::ostream &(*manip)(std::ostream &))
Definition: einhard.hpp:314
std::ostringstream * out
Definition: einhard.hpp:224
EINHARD_ALWAYS_INLINE_ UnconditionalOutput(bool colorize_, const char *areaName, std::integral_constant< LogLevel, VERBOSITY >)
Definition: einhard.hpp:238
EINHARD_ALWAYS_INLINE_ UnconditionalOutput(bool colorize_)
Definition: einhard.hpp:271
void doInit(const char *areaName)
EINHARD_ALWAYS_INLINE_ UnconditionalOutput & operator<<(const T &msg)
Definition: einhard.hpp:261
EINHARD_ALWAYS_INLINE_ UnconditionalOutput & operator<<(std::ostream &(*manip)(std::ostream &))
Definition: einhard.hpp:255
void doCleanup(std::FILE *outfile=stdout) noexcept
UnconditionalOutput & operator<<(const Color< T > &col)
Definition: einhard.hpp:245
#define EINHARD_ALWAYS_INLINE_
Definition: einhard.hpp:83
#define _COLOR(name, code)
Definition: einhard.hpp:174
This namespace contains all objects required for logging using Einhard.
Definition: einhard.hpp:97
LogLevel getLogLevel(const std::string &level)
Compares the string level against the strings for LogLevel and returns the one it matches.
const char * colorForLogLevel() noexcept
char const VERSION[]
Version string of the Einhard library.
LogLevel
Specification of the message severity.
Definition: einhard.hpp:109
@ TRACE
The lowes severity for messages describing the program flow.
Definition: einhard.hpp:111
@ OFF
If selected no messages will be output.
Definition: einhard.hpp:117
@ WARN
Warning messages.
Definition: einhard.hpp:114
@ ALL
Log all message.
Definition: einhard.hpp:110
@ ERROR
Non-fatal errors.
Definition: einhard.hpp:115
@ DEBUG
Debug messages.
Definition: einhard.hpp:112
@ FATAL
Messages that indicate terminal application failure.
Definition: einhard.hpp:116
@ INFO
Messages of informational nature, expected processing time e.g.
Definition: einhard.hpp:113
const char * getLogLevelString() noexcept
Retrieve a human readable representation of the given log level value.
A minimal class that implements the output stream operator to do nothing.
Definition: einhard.hpp:208
EINHARD_ALWAYS_INLINE_ DummyOutputFormatter & operator<<(std::ostream &(*)(std::ostream &)) noexcept
Definition: einhard.hpp:213
EINHARD_ALWAYS_INLINE_ DummyOutputFormatter & operator<<(const T &) noexcept
Definition: einhard.hpp:209