This document discusses how to build your project with libjbpf library. For full functionality, one also needs to add hook, maps or helper functions.
libjbpf is a light-weight library with a limited number of external dependencies. It creates up to three support threads:
- Maintenance thread (always created) is a lightweight thread that runs various management tasks, such as stats updates.
- Agent thread (optional) implements the life-cycle management functionality, if enabled.
- I/O thread (optional) implements the data collection and control functionality, if enabled.
To integrate libjbpf with a new project, one needs to build the project with the libjbpf library, and to initialize it correctly.
The library can be built as static (libjbpf.a
) or dynamic (libjbpf.so
).
To avoid installing dependencies locally, you can build the library using a Docker container.
- Build the container image:
OS=mariner # or ubuntu22.04, or ubuntu24.04
sudo -E docker build -t jbpf_lib -f deploy/${OS}.Dockerfile .
- Build the library using the container image:
DEST_PATH=the_abs_path_to_output_folder
sudo -E docker run -v $DEST_PATH:/jbpf_out_lib \
-e OPTION_NAME1={0,1} \
-e OPTION_NAME2={0,1} \
jbpf_lib
The $DEST_PATH
should be set to the absolute path of the directory where you want the output to be stored.
When building the library, you can pass various options using the -e OPTION_NAME={0,1} format. Here are some available options:
- JBPF_STATIC - Build jbpf as a static library (default: disabled)
- USE_NATIVE - Enable/disable
-march=native
compilation flag (default: enabled) - USE_JBPF_PERF_OPT - Performance optimizations that assume threads calling jbpf codelets are pinned to a certain core (default: enabled)
- USE_JBPF_PRINTF_HELPER - Disable the use of the helper function jbpf_printf_debug() (default: enabled)
- JBPF_THREADS_LARGE - Allow more threads to be registered by jbpf and the IO lib (default: disabled)
- JBPF_EXPERIMENTAL_FEATURES - Enable experimental features of jbpf (default: disabled)
- ENABLE_POISONING - Enable ASAN poisoning. Should not be used for IPC mode tests and must be used in conjunction with ASAN (default: disabled)
- CLANG_FORMAT_CHECK - Enable clang-format check (default: disabled)
- CPP_CHECK - Enable Cpp static code analyser (default: disabled)
- BUILD_TESTING - Build the tests (default: enabled)
Additionally, you can change the CMAKE_BUILD_TYPE
to:
- Release - Default Build Type.
- AddressSanitizer - Build with AddressSanitizer support to check for memory leaks.
- Debug - Build with jbpf debug symbols.
For a up-to-date list of available options, refer to the CMakelists.txt file.
The generated library will be stored in the $DEST_PATH/out
directory. Other useful files will be located in the out directory for inclusion in your build.
The library initialization requires two steps. The first step is to call jbpf_init()
function with the desired configuration parameters.
Below is the list of configuration parameters:
- Memory (
jbpf_config.mem_config
): The total size of memory (hugepages) allocated for maps, etc. - Life-cycle management model (
jbpf_config.lcm_ipc_config
): Specify whether to use C API or Unix sockets for life-cycle management (see example). - I/O model (
jbpf_config.io_config.io_type
): Specify whether to use an internal thread with call-backs, or an external data collector over IPC (see example). - Library thread config (
jbpf_config.io_config
andjbpf_config.thread_config
): Specify Linux thread priority, CPU affinity and scheduling policy (SCHED_OTHER
orSCHED_FIFO
) for various libjbpf threads.
To set the logging level, you can include the following in your source:
#include "jbpf_logging.h"
logging_level jbpf_logger_level;
jbpf_logger_level = JBPF_INFO; // JBPF_DEBUG, JBPF_INFO, JBPF_WARN, JBPF_ERROR, JBPF_CRITICAL
You can also provide a customized logger function, such as:
void jbpf_custom_logger(jbpf_logging_level level, const char* s, ...) {
if (level >= jbpf_logger_level) {
print_log_level(level);
va_list ap;
va_start(ap, s);
FILE *where = level >= ERROR ? stderr : stdout;
vfprintf(where, s, ap);
va_end(ap);
}
}
jbpf_va_logger_cb jbpf_va_logger_func = jbpf_custom_logger;
When environment variable JBPF_VERBOSE_LOGGING
is set, extra information will be shown per logging message: file, function and the line number.
libjbpf often runs concurrent operations, including memory management, access to running codelets, etc.
In order to implement them efficiently, we need to register each thread that invokes a hook, before the invocation takes place.
This is done with jbpf_register_thread()
call, typically at the start of the thread.
If this is not done, a hook invocation will cause a segfault. When a thread finishes, it is recommended to call jbpf_cleanup_thread()
.