diff --git a/CMakeLists.txt b/CMakeLists.txt index 9c5bf84..1b95379 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -115,6 +115,10 @@ if(BOOST_STACKTRACE_ENABLE_WINDBG) target_link_libraries(boost_stacktrace INTERFACE Boost::stacktrace_windbg) +elseif(BOOST_STACKTRACE_ENABLE_WINDBG_CACHED) + + target_link_libraries(boost_stacktrace INTERFACE Boost::stacktrace_windbg) + elseif(BOOST_STACKTRACE_ENABLE_BACKTRACE) target_link_libraries(boost_stacktrace INTERFACE Boost::stacktrace_backtrace) diff --git a/doc/stacktrace.qbk b/doc/stacktrace.qbk index 4f9ca6b..71ac73a 100644 --- a/doc/stacktrace.qbk +++ b/doc/stacktrace.qbk @@ -68,7 +68,7 @@ Pretty often assertions provide not enough information to locate the problem. Fo Aborted (core dumped) ``` -That's not enough to locate the problem without debugger. There may be thousand code lines in real world examples and hundred places where that assertion could happen. Let's try to improve the assertions, and make them more informative: +That's not enough to locate the problem without debugger. There may be thousand code lines in real world examples and hundred places where that assertion can happen. Let's try to improve the assertions, and make them more informative: [getting_started_assert_handlers] @@ -236,7 +236,7 @@ Code from above will output: [section Enabling and disabling stacktraces] -At some point arises a requirement to easily enable/disable stacktraces for a whole project. That could be easily achieved. +At some point arises a requirement to easily enable/disable stacktraces for a whole project. That can be easily achieved. Just define *BOOST_STACKTRACE_LINK* for a whole project. Now you can enable/disable stacktraces by just linking with different libraries: @@ -251,7 +251,7 @@ See [link stacktrace.configuration_and_build section "Configuration and Build"] [section Saving stacktraces by specified format] [classref boost::stacktrace::stacktrace] provides access to individual [classref boost::stacktrace::frame frames] of the stacktrace, -so that you could save stacktrace information in your own format. Consider the example, that saves only function addresses of each frame: +so that you can save stacktrace information in your own format. Consider the example, that saves only function addresses of each frame: [getting_started_trace_addresses] @@ -304,7 +304,63 @@ Terminate called: [section Configuration and Build] -By default Boost.Stacktrace is a header-only library, but you may change that and use the following macros to improve build times or to be able to tune library without recompiling your project: +[section Usage from CMake] + +CMake library `Boost::stacktrace` provides the best available implementation: + +``` +target_link_libraries(${PROJECT} + PUBLIC + Boost::stacktrace +) +``` + +There are also CMake libraries for fine grained selection of implementation: + +* `Boost::stacktrace_windbg` +* `Boost::stacktrace_windbg_cached` +* `Boost::stacktrace_backtrace` +* `Boost::stacktrace_addr2line` +* `Boost::stacktrace_basic` +* `Boost::stacktrace_noop` + +Note that `Boost::stacktrace_from_exception` is not used by default, so add it explicitly if its functionality is required: + +``` +target_link_libraries(${PROJECT} # your project + PUBLIC + Boost::stacktrace + Boost::stacktrace_from_exception +) +``` + +[endsect] + +[section CMake build notes] + +When building the Boost.Stacktrace libraries using `CMake` the `BOOST_STACKTRACE_ENABLE_*` options control the build. For example: + +* `cmake -DBOOST_STACKTRACE_ENABLE_NOOP=0 -DBOOST_STACKTRACE_ENABLE_BACKTRACE=1 -DBOOST_STACKTRACE_ENABLE_FROM_EXCEPTION=1` - do not build the `noop` implementation and force the builds of `backtrace` and `from_exception`. +* `cmake -DBOOST_STACKTRACE_ENABLE_NOOP=1 -DBOOST_STACKTRACE_ENABLE_WINDBG=1 -DBOOST_STACKTRACE_ENABLE_WINDBG_CACHED=0` - build the `noop` and `windbg` implementations and disable the build of `windbg_cached`. + +If options are not provided they are auto-detected and the result of detection is printed and implicitly used during build. + +[endsect] + +[section B2 build notes] + +When building the Boost.Stacktrace libraries using `b2` the `boost.stacktrace.*` options can be used to control the build. For example: + +* `b2 boost.stacktrace.noop=off boost.stacktrace.backtrace=on boost.stacktrace.from_exception=on` - do not build the `noop` implementation and force the builds of `backtrace` and `from_exception`. +* `b2 boost.stacktrace.noop=on boost.stacktrace.windbg=on boost.stacktrace.windbg_cached=off` - build the `noop` and `windbg` implementations and disable the build of `windbg_cached`. + +If options are not provided they are auto-detected and the result of detection is printed and implicitly used during build. + +[endsect] + +[section Header only options] + +If CMake is not used then the Boost.Stacktrace is a header-only library by default. To change that (to improve build times or to be able to tune library without recompiling your project) use the following macros: [table:linkmacro Link macros [[Macro name] [Effect]] [[*BOOST_STACKTRACE_LINK*] [Disable header-only build and require linking with shared or static library that contains the tracing implementation. If *BOOST_ALL_DYN_LINK* is defined, then link with shared library.]] @@ -312,13 +368,13 @@ By default Boost.Stacktrace is a header-only library, but you may change that an ] -In header only mode library could be tuned by macro. If one of the link macro from above is defined, you have to manually link with one of the libraries: +In header only mode library can be tuned by macro. If one of the link macro from above is defined, you have to manually link with one of the libraries: [table:libconfig Config [[Macro name or default] [Library] [Effect] [Platforms] [Uses debug information [footnote This will provide more readable backtraces with *source code locations* if the binary is built with debug information.]] [Uses dynamic exports information [footnote This will provide readable function names in backtrace for functions that are exported by the binary. Compiling with `-rdynamic` flag, without `-fvisibility=hidden` or marking functions as exported produce a better stacktraces.]] ] - [[['default for MSVC, Intel on Windows, MinGW-w64] / *BOOST_STACKTRACE_USE_WINDBG*] [*boost_stacktrace_windbg*] [ Uses `dbgeng.h` to show debug info, stores the implementation internals in a static varaible protected with mutex. May require linking with *ole32* and *dbgeng*. ] [MSVC, MinGW-w64, Intel on Windows] [yes] [no]] + [[['default for MSVC, Intel on Windows, MinGW-w64] / *BOOST_STACKTRACE_USE_WINDBG*] [*boost_stacktrace_windbg*] [ Uses `dbgeng.h` to show debug info, stores the implementation internals in a static variable protected with mutex. May require linking with *ole32* and *dbgeng*. ] [MSVC, MinGW-w64, Intel on Windows] [yes] [no]] [[['default for other platforms]] [*boost_stacktrace_basic*] [Uses compiler intrinsics to collect stacktrace and if possible `::dladdr` to show information about the symbol. Requires linking with *libdl* library on POSIX platforms.] [Any compiler on POSIX or MinGW] [no] [yes]] [[*BOOST_STACKTRACE_USE_WINDBG_CACHED*] [*boost_stacktrace_windbg_cached*] [ Uses `dbgeng.h` to show debug info and caches implementation internals in TLS for better performance. Useful only for cases when traces are gathered very often. May require linking with *ole32* and *dbgeng*. ] [MSVC, Intel on Windows] [yes] [no]] - [[*BOOST_STACKTRACE_USE_BACKTRACE*] [*boost_stacktrace_backtrace*] [Requires linking with *libdl* on POSIX and *libbacktrace* libraries[footnote Some *libbacktrace* packages SEGFAULT if there's a concurrent work with the same `backtrace_state` instance. To avoid that issue the Boost.Stacktrace library uses `thread_local` states, unfortunately this may consume a lot of memory if you often create and destroy execution threads in your application. Define *BOOST_STACKTRACE_BACKTRACE_FORCE_STATIC* to force single instance, but make sure that [@https://github.com/boostorg/stacktrace/blob/develop/test/thread_safety_checking.cpp thread_safety_checking.cpp] works well in your setup. ]. *libbacktrace* is probably already installed in your system[footnote If you are using Clang with libstdc++ you could get into troubles of including ``, because on some platforms Clang does not search for headers in the GCC's include paths and any attempt to add GCC's include path leads to linker errors. To explicitly specify a path to the `` header you could define the *BOOST_STACKTRACE_BACKTRACE_INCLUDE_FILE* to a full path to the header. For example on Ubuntu Xenial use the command line option *-DBOOST_STACKTRACE_BACKTRACE_INCLUDE_FILE=* while building with Clang. ], or built into your compiler. + [[*BOOST_STACKTRACE_USE_BACKTRACE*] [*boost_stacktrace_backtrace*] [Requires linking with *libdl* on POSIX and *libbacktrace* libraries[footnote Some *libbacktrace* packages SEGFAULT if there's a concurrent work with the same `backtrace_state` instance. To avoid that issue the Boost.Stacktrace library uses `thread_local` states, unfortunately this may consume a lot of memory if you often create and destroy execution threads in your application. Define *BOOST_STACKTRACE_BACKTRACE_FORCE_STATIC* to force single instance, but make sure that [@https://github.com/boostorg/stacktrace/blob/develop/test/thread_safety_checking.cpp thread_safety_checking.cpp] works well in your setup. ]. *libbacktrace* is probably already installed in your system[footnote If you are using Clang with libstdc++ you can get into troubles of including ``, because on some platforms Clang does not search for headers in the GCC's include paths and any attempt to add GCC's include path leads to linker errors. To explicitly specify a path to the `` header you can define the *BOOST_STACKTRACE_BACKTRACE_INCLUDE_FILE* to a full path to the header. For example on Ubuntu Xenial use the command line option *-DBOOST_STACKTRACE_BACKTRACE_INCLUDE_FILE=* while building with Clang. ], or built into your compiler. Otherwise (if you are a *MinGW*/*MinGW-w64* user for example) it can be downloaded [@https://github.com/ianlancetaylor/libbacktrace from here] or [@https://github.com/gcc-mirror/gcc/tree/master/libbacktrace from here]. ] [Any compiler on POSIX, or MinGW, or MinGW-w64] [yes] [yes]] [[*BOOST_STACKTRACE_USE_ADDR2LINE*] [*boost_stacktrace_addr2line*] [Use *addr2line* program to retrieve stacktrace. Requires linking with *libdl* library and `::fork` system call. Macro *BOOST_STACKTRACE_ADDR2LINE_LOCATION* must be defined to the absolute path to the addr2line executable if it is not located in /usr/bin/addr2line. ] [Any compiler on POSIX] [yes] [yes]] @@ -331,6 +387,8 @@ In header only mode library could be tuned by macro. If one of the link macro fr * if you wish to disable backtracing and *BOOST_STACKTRACE_LINK* is defined, you just need link with *-lboost_stacktrace_noop* * if you wish to disable backtracing and you use the library in header only mode, you just need to define *BOOST_STACKTRACE_USE_NOOP* for the whole project and recompile it +[endsect] + [section MinGW and MinGW-w64 specific notes] MinGW-w64 and MinGW (without -w64) users have to install libbacktrace for getting better stacktraces. Follow the instruction: @@ -370,11 +428,11 @@ In theory, walking the stack without decoding and demangling should be async sig In practice, it is not: * Looks like a page fault while dumping the trace on a containerized/virtualized - Windows system has a chance to deadlock. Page fault could happen easily + Windows system has a chance to deadlock. Page fault can happen easily as we have to write the dump either to memory or to a file. -* On POSIX systems a deadlock could happen if a signal is received when throwing +* On POSIX systems a deadlock can happen if a signal is received when throwing an exception [@https://github.com/boostorg/stacktrace/issues/131 #131]. - Theoretically this could be worked around by bypassing the mutex locking + Theoretically this can be worked around by bypassing the mutex locking in C++-runtime at exception throw ([@https://github.com/userver-framework/userver/blob/4246909c99506d3ab34bd130a5154b4acc8e87de/core/src/engine/task/exception_hacks.cpp#L241-L244 sample implementation] in the 🐙 userver framework), or by using a very modern runtime