Skip to content
Header only logging utility
Branch: master
Clone or download
Fetching latest commit…
Cannot retrieve the latest commit at this time.
Type Name Latest commit message Commit time
Failed to load latest commit information.


Copyright (c) 2011 Seiya Tokui <>. All Rights Reserved.

eLog is a header only logging library for C++.

This library is distributed under MIT License, which is available as the file
of name 'LICENSE'.


- Header only library.
- Portable: it runs on both POSIX-like systems and Windows.
- Syntax like google-glog. eLog has a simpler syntax and limited methods.
- Multi-thread ready.
- Supports g++ 4.2.1+ (using GNU extension) and VC++ 2010.
- Typed and verbose logging with different verbosity for each type.
- Simple benchmark that counts and logs the time duration of processing.


If you have python, type `[sudo] ./waf install'. It will install all headers to
/usr/local/include/elog/. Just copying elog/*.h to /usr/local/include/elog/ is
also ok.


You can start using eLog by including <elog/elog.h> or, if you want to use
benchmark utilities, <elog/benchmark.h>.

simple logging

Basic logging syntax is similar to that of the famous logging library, glog:

  LOG(INFO) << "some message or value like " << 123;

operator<< of LOG(...) accepts any value that ostream accepts. There are four
levels of log: INFO, WARN, ERROR and FATAL. Especially, INFO can be omitted:

  LOG() << "INFO level message";

Logger emits any level message by default. You can restrict messages by
setting level. If the default logger is the current global one, then eigher of
below changes the level of logger.


After this line is executed, messages of level INFO and WARN are ignored.

NOTE: LOG becomes a namespace of eLog when used without trailing parentheses.
LOG(...) is a variadic macro.

LOG(FATAL) is special; it throws an exception of type LOG::FatalLogError after
outputing the message. eLog currently does not have stack-trace feature like

LOG(...) emits messages to std::clog by default. LOG::Logger is the interface
of log emission, and global logger can be exchanged by LOG::SetLogger.

  std::ostream& my_stream = ...;
  LOG::Logger* my_logger = new LOG::StreamLogger(my_stream);

SetLogger gives a reference to *my_logger to the global logger holder, so you
must keep my_logger alive while it is used as a global logger.

Typed and verbose logging

eLog has another syntax: typed and verbose logging.

  LOG(SomeType, 1) << "typed log with verbosity 1";

The first argument, called 'message type', can be any type. Typically user-
defined class is used.

  class KeyInput {};  // Message type cannot be incomplete.
  LOG(KeyInput, 1) << "verbose log about key input";

Another typical usage is set the message type to the class of the method in
which the LOG(...) statement is written.

  class Keyboard {
    void Method() {
      LOG(Keyboard, 0) << "verbose log from Keyboard";

The second argument is verbosity of the message. LOG(type, N) emits the message
if N is not greater than the verbosity of 'type' under the current global
logger. Default verbosity of each type is 0. You can modify verbosity of each
type. If default logger is the global one, then either of below changes the
verbosity of SomeType.

      LOG::TypeInfo(LOG::Type<SomeType>()), 2);

After this line is executed, LOG(SomeType, N) will emit messages only if N <= 2.


eLog also has assertion syntax:

  CHECK(!some_critical_condition) << "some_critical_condition occured";

It emits messages only if the operand is false, and after that it throws an
exception of type LOG::CheckError.


Benchmark utilities are useful to measure and output the time to execute code
fragment. For instance, unary BENCHMARK(...) is useful to quickly measure some

  BENCHMARK(bench_heavy_method) {

At the top of the clause, it outputs 'bench_heavy_method...' to the current
global logger, executes inside of braces, and outputs total execution time in

In case that you want to measure many code fragments, you can use
LOG::BenchmarkSuite to bundle all results. You can outputs results to the
current global logger by calling LogChart.

  LOG::BenchmarkSuite suite("my experiments");

  BENCHMARK(suite, first_exp) {
  BENCHMARK(suite, second_exp) {
  BENCHMARK(suite, nth_exp) {

  suite.LogChart();  // outputs results

It will emits chart of results.

BENCHMARK macro also has typed versions.

  struct Keyboard {};

  // Benchmark of Keyboard with verbosity 1
  BENCHMARK(Keyboard, 1, bench_kb) {

  LOG::BenchmarkSuite suite("keyboard bench");

  BENCHMARK(suite, Keyboard, 1, bench_kb) {

Debug versions

There are similar macro of LOG and CHECK, named DLOG and DCHECK. These are debug
version macro. DLOG and DCHECK do nothing if NDEBUG is defined. Otherwise these
are exactly same as LOG and CHECK, respectively.

Other note

NOTE(g++): LOG(...) and BENCHMARK(...) uses compiler extensions.
You can’t perform that action at this time.