Skip to content

Real-time safe logger for C++ using lock free queues!

License

Notifications You must be signed in to change notification settings

cjappl/rtlog-cpp

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

57 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

rtlog-cpp 🔊

rtlog-cpp is a logging library designed specifically for logging messages from the real-time thread. This is particularly useful in the audio and embedded industries, where hard real-time requirements must be met, and logging in traditional ways from your real-time threads is unacceptable.

If you're looking for a general use logger, this probably isn't the library for you!

The design behind this logger was presented at ADCx 2023.

ADCx rtlog presentation on youtube

Slides: Slide Title Page

Features

  • Ability to log messages of any type and size from the real-time thread
  • Statically allocated memory at compile time, no allocations in the real-time thread
  • Support for printf-style format specifiers (using a version of the printf family that doesn't hit the localeconv lock) OR support for modern libfmt formatting.
  • Efficient thread-safe logging using a lock free queue.

Requirements

  • A C++17 compatible compiler
  • The C++17 standard library
  • farbot::fifo (will be downloaded via cmake if not provided)
  • stb's vsnprintf (will be downloaded via cmake if not provided) OR libfmt if cmake is run with the RTSAN_USE_FMTLIB option

Installation via CMake

In CMakeLists.txt

include(FetchContent)
FetchContent_Declare(rtlog-cpp
    GIT_REPOSITORY https://github.com/cjappl/rtlog-cpp
)
FetchContent_MakeAvailable(rtlog-cpp)

add_executable(audioapp ${SOURCES})
target_link_libraries(audioapp
     PRIVATE
         rtlog::rtlog
 )

To use formatlib, set the variable, either on the command line or in cmake:

cmake .. -DRTLOG_USE_FMTLIB=ON

Usage

For more fleshed out fully running examples check out examples/ and test/

After including via cmake:

  1. Include the rtlog/rtlog.h header file in your source code
  2. Create a rtlog::Logger object with the desired template parameters:
  3. Process the log messages on your own thread, or via the provided rtlog::LogProcessingThread
#include <rtlog/rtlog.h>

struct ExampleLogData
{
    ExampleLogLevel level;
    ExampleLogRegion region;
};

constexpr auto MAX_LOG_MESSAGE_LENGTH = 256;
constexpr auto MAX_NUM_LOG_MESSAGES = 100;

std::atomic<int> gSequenceNumber{0};

using RealtimeLogger = rtlog::Logger<ExampleLogData, MAX_NUM_LOG_MESSAGES, MAX_LOG_MESSAGE_LENGTH, gSequenceNumber>;

...

RealtimeLogger logger;

void SomeRealtimeCallback()
{
    logger.Log({ExampleLogLevel::Debug, ExampleLogRegion::Audio}, "Hello, world! %i", 42);

    // using RTSAN_USE_LIBFMT
    logger.Log({ExampleLogData::Debug, ExampleLogRegion::Audio, FMT_STRING("Hello, world! {}", 42);
}

...

To process the logs in another thread, call PrintAndClearLogQueue with a function to call on the output data.

static auto PrintMessage = [](const ExampleLogData& data, size_t sequenceNumber, const char* fstring, ...) __attribute__ ((format (printf, 4, 5)))
{
    std::array<char, MAX_LOG_MESSAGE_LENGTH> buffer;
    
    va_list args;
    va_start(args, fstring);
    vsnprintf(buffer.data(), buffer.size(), fstring, args);
    va_end(args);

    printf("{%lu} [%s] (%s): %s\n", 
        sequenceNumber, 
        rtlog::test::to_string(data.level), 
        rtlog::test::to_string(data.region), 
        buffer.data());
};

...

void LogProcessorThreadMain()
{
    while (running)
    {
        if (logger.PrintAndClearLogQueue(PrintMessage) == 0)
            std::this_thread::sleep_for(std::chrono::milliseconds(10);
    }
}

Or alternatively spin up a rtlog::LogProcessingThread

    rtlog::LogProcessingThread thread(logger, PrintMessage, std::chrono::milliseconds(10));

Customizing the queue type

rtlog provides two queue type variants: rtlog::SingleRealtimeWriterQueueType (SPSC - default) and rtlog::MultiRealtimeWriterQueueType (MPSC). It is always assummed that you have one log printing thread. These may be used by specifying them:

using SingleWriterRtLoggerType = rtlog::Logger<ExampleLogData, MAX_NUM_LOG_MESSAGES, MAX_LOG_MESSAGE_LENGTH, gSequenceNumber, SingleRealtimeWriterQueueType>;

SingleWriterRtLoggerType logger;
logger.Log({ExampleLogLevel::Debug, ExampleLogRegion::Audio}, "Hello, world! %i", 42);

...

using MultiWriterRtLoggerType = rtlog::Logger<ExampleLogData, MAX_NUM_LOG_MESSAGES, MAX_LOG_MESSAGE_LENGTH, gSequenceNumber, MultiRealtimeWriterQueueType>;

MultiWriterRtLoggerType logger;
logger.Log({ExampleLogLevel::Debug, ExampleLogRegion::Audio}, "Hello, world! %i", 42);

If you don't want to use either of these defaults, you may provide your own queue.

** IT IS UP TO YOU TO ENSURE THE QUEUE YOU PROVIDE IS LOCK-FREE AND REAL-TIME SAFE **

The queue must have the following:

template <typename T>
class MyQueue
{
public:
    using value_type = T;

    MyQueue(int capacity);
    bool try_dequeue(T& item); // MUST return false if the queue is empty

    bool try_enqueue(T&& item);
    // OR
    bool try_enqueue(const T& item);
};

Then, when creating the logger, provide the queue type as a template parameter:

using RealtimeLogger = rtlog::Logger<ExampleLogData, MAX_NUM_LOG_MESSAGES, MAX_LOG_MESSAGE_LENGTH, gSequenceNumber, MyQueue>;

You can see an example of wrapping a known rt-safe queue in examples/custom_queue_example.

About

Real-time safe logger for C++ using lock free queues!

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Contributors 3

  •  
  •  
  •