diff --git a/CMakeLists.txt b/CMakeLists.txt index 69c18331746ed3afa08df83e028c1d42d38a5c92..a05ee6da91ba601d86d50aee769813f6421b4ab0 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -40,7 +40,7 @@ set(ARB_MODCC "" CACHE STRING "path to external modcc NMODL compiler") # Use libunwind to generate stack traces on errors? -option(ARB_UNWIND "Use libunwind for stack trace printing if available" OFF) +option(ARB_BACKTRACE "Enable stacktraces on assertion and exceptions (requires Boost)." OFF) # Specify GPU build type @@ -386,17 +386,15 @@ if(ARB_WITH_GPU) endif() endif() -# Use libunwind if requested for pretty printing stack traces -#------------------------------------------------------------- +# Use boost::stacktrace if requested for pretty printing stack traces +#-------------------------------------------------------------------- -if (ARB_UNWIND) - find_package(Unwind REQUIRED) - if(Unwind_FOUND) - target_link_libraries(arbor-private-deps INTERFACE Unwind::unwind) - target_compile_definitions(arbor-private-deps INTERFACE WITH_UNWIND) - - list(APPEND arbor_export_dependencies "Unwind") - endif() +if (ARB_BACKTRACE) + find_package(Boost REQUIRED + COMPONENTS stacktrace_basic + stacktrace_addr2line) + target_link_libraries(arbor-private-deps INTERFACE Boost::stacktrace_basic Boost::stacktrace_addr2line ${CMAKE_DL_LIBS}) + target_compile_definitions(arbor-private-deps INTERFACE WITH_BACKTRACE) endif() # Build and use modcc unless explicit path given @@ -547,7 +545,6 @@ install( FILES "${CMAKE_CURRENT_BINARY_DIR}/arbor-config.cmake" "${CMAKE_CURRENT_BINARY_DIR}/arbor-config-version.cmake" - cmake/FindUnwind.cmake DESTINATION "${cmake_config_dir}") add_subdirectory(lmorpho) diff --git a/arbor/arbexcept.cpp b/arbor/arbexcept.cpp index 00f0d8632a30509faaa701875097aaeaed537b24..74dca037f3b2e655bf65c2517c0ee4a2e81c65fd 100644 --- a/arbor/arbexcept.cpp +++ b/arbor/arbexcept.cpp @@ -4,12 +4,25 @@ #include <arbor/arbexcept.hpp> #include <arbor/common_types.hpp> +#include "util/unwind.hpp" #include "util/strprintf.hpp" namespace arb { using arb::util::pprintf; +arbor_exception::arbor_exception(const std::string& what): + std::runtime_error{what} { + // Backtrace w/o this c'tor and that of backtrace. + where = util::backtrace{}.pop(2).to_string(); +} + +arbor_internal_error::arbor_internal_error(const std::string& what): + std::logic_error(what) { + // Backtrace w/o this c'tor and that of backtrace. + where = util::backtrace{}.pop(2).to_string(); +} + domain_error::domain_error(const std::string& w): arbor_exception(w) {} bad_cell_probe::bad_cell_probe(cell_kind kind, cell_gid_type gid): diff --git a/arbor/assert.cpp b/arbor/assert.cpp index 4205f641132add2c4e5b65e142dbc26d28012323..ffcd325571caf5bf52e2ae901df1f4f36432b64f 100644 --- a/arbor/assert.cpp +++ b/arbor/assert.cpp @@ -12,7 +12,7 @@ void ARB_ARBOR_API abort_on_failed_assertion( int line, const char* func) { - // Emit stack trace If libunwind is being used. + // Emit stack trace if enabled. std::cerr << util::backtrace(); // Explicit flush, as we can't assume default buffering semantics on stderr/cerr, diff --git a/arbor/include/arbor/arbexcept.hpp b/arbor/include/arbor/arbexcept.hpp index 4acfb6a7bb2fead64499141517d61cc80d9f3be3..deb578e43cbb7b5db61b95877488acae4864cffb 100644 --- a/arbor/include/arbor/arbexcept.hpp +++ b/arbor/include/arbor/arbexcept.hpp @@ -15,18 +15,15 @@ namespace arb { // there is a bug in the library.) struct ARB_SYMBOL_VISIBLE arbor_internal_error: std::logic_error { - arbor_internal_error(const std::string& what_arg): - std::logic_error(what_arg) - {} + arbor_internal_error(const std::string&); + std::string where; }; - // Common base-class for arbor run-time errors. struct ARB_SYMBOL_VISIBLE arbor_exception: std::runtime_error { - arbor_exception(const std::string& what_arg): - std::runtime_error(what_arg) - {} + arbor_exception(const std::string&); + std::string where; }; // Logic errors diff --git a/arbor/util/unwind.cpp b/arbor/util/unwind.cpp index 28ba9afeaf29f54787c7ac3448eb9f4a8819ca2e..9594aa3bb5339ca03abd9b4ed585f467d7b0f642 100644 --- a/arbor/util/unwind.cpp +++ b/arbor/util/unwind.cpp @@ -1,124 +1,52 @@ #include <util/unwind.hpp> -#ifdef WITH_UNWIND - -#define UNW_LOCAL_ONLY -#include <libunwind.h> -#include <cxxabi.h> - -#include <memory/util.hpp> -#include <util/file.hpp> - -#include <cxxabi.h> -#include <cstdint> -#include <cstdio> +#include <sstream> #include <string> +#include <iomanip> #include <iostream> #include <vector> +#ifdef WITH_BACKTRACE +#define BOOST_STACKTRACE_GNU_SOURCE_NOT_REQUIRED +#define BOOST_STACKTRACE_USE_ADDR2LINE +#include <boost/stacktrace.hpp> +#endif + namespace arb { namespace util { -static_assert(sizeof(std::uintptr_t)>=sizeof(unw_word_t), - "assumption that libunwind unw_word_t can be stored in std::uintptr_t is not valid"); - -/// Builds a stack trace when constructed. -/// The trace can then be printed, or accessed via the stack() member function. backtrace::backtrace() { - unw_cursor_t cursor; - unw_context_t context; - - // initialize cursor to current frame for local unwinding. - unw_getcontext(&context); - unw_init_local(&cursor, &context); - - while (unw_step(&cursor) > 0) { - // find the stack position - unw_word_t offset, pc; - unw_get_reg(&cursor, UNW_REG_IP, &pc); - if (pc == 0) { - break; - } - - // get the name - char sym[512]; - if (unw_get_proc_name(&cursor, sym, sizeof(sym), &offset) == 0) { - frames_.push_back({std::string(sym), pc}); - } else { - frames_.push_back({std::string("???"), pc}); - } +#ifdef WITH_BACKTRACE + auto bt = boost::stacktrace::basic_stacktrace{}; + for (const auto& f: bt) { + frames_.push_back(source_location{f.name(), f.source_file(), f.source_line()}); } -} - -std::string demangle(std::string s) { - int status; - char* demangled = abi::__cxa_demangle(s.c_str(), nullptr, nullptr, &status); - - // __cxa_demangle only returns a non-empty string if it is passed a valid C++ - // mangled c++ symbol (i.e. it returns an empty string for normal c symbols) - if (status==0) { - s = demangled; - } - std::free(demangled); // don't leak the demangled string - - return s; +#endif } std::ostream& operator<<(std::ostream& out, const backtrace& trace) { - for (auto& f: trace.frames_) { - char loc_str[64]; - snprintf(loc_str, sizeof(loc_str), "0x%lx", f.position); - out << loc_str << " " << f.name << "\n"; - if (f.name=="main") { - break; - } +#ifdef WITH_BACKTRACE + out << "Backtrace:\n"; + int ix = 0; + for (const auto& f: trace.frames_) { + out << std::setw(8) << ix << " " << f.func << " (" << f.file << ":" << f.line << ")\n"; + ix++; } +#endif return out; } -#if 0 -// Temporarily deprecated: automatic writing to disk of strack traces -// needs to be run-time configurable. - -void backtrace::print(bool stop_at_main) const { - using namespace arb::memory::util; - - auto i = 0; - while (file_exists("backtrace_" + std::to_string(i))) { - ++i; - } - auto fname = "backtrace_" + std::to_string(i); - auto fid = std::ofstream(fname); - for (auto& f: frames_) { - char loc_str[64]; - snprintf(loc_str, sizeof(loc_str),"0x%lx", f.position); - fid << loc_str << " " << f.name << "\n"; - if (stop_at_main && f.name=="main") { - break; - } - } - std::cerr << "BACKTRACE: A backtrace was generated and stored in the file " << fname << ".\n"; - std::cerr << " View a brief summary of the backtrace by running \"scripts/print_backtrace " << fname << " -b\".\n"; - std::cerr << " Run \"scripts/print_backtrace -h\" for more options.\n"; +backtrace& backtrace::pop(std::size_t n) { + frames_.erase(frames_.begin(), + frames_.begin() + std::min(n, frames_.size())); + return *this; } -#endif -} // namespace util -} // namespace arb - -#else - -namespace arb { -namespace util { - -std::ostream& operator<<(std::ostream& out, const backtrace& trace) { - return out; +std::string backtrace::to_string() const { + std::stringstream ss; + ss << *this; + return ss.str(); } -//void arb::util::backtrace::print(bool) const {} - } // namespace util } // namespace arb - -#endif - diff --git a/arbor/util/unwind.hpp b/arbor/util/unwind.hpp index 293ecf50dcbe554ab18ab0c461fc9de525ce0d6d..6459eec20815c6493a081b869f274bd80436ccba 100644 --- a/arbor/util/unwind.hpp +++ b/arbor/util/unwind.hpp @@ -1,5 +1,6 @@ #pragma once +#include <iostream> #include <cstdint> #include <string> #include <vector> @@ -9,27 +10,27 @@ namespace util { /// Represents a source code location as a function name and address struct source_location { - std::string name; - std::uintptr_t position; // assume that unw_word_t is a unit64_t + std::string func; + std::string file; + std::size_t line; }; /// Builds a stack trace when constructed. -/// The trace can then be printed, or accessed via the stack() member function. -/// NOTE: if WITH_UNWIND is not defined, the methods are empty +/// NOTE: if WITH_BACKTRACE is not defined, the methods are empty class backtrace { public: /// the default constructor will build and store the strack trace. - backtrace() = default; + backtrace(); - /// Creates a new file named backtrace_# where # is a number chosen - /// The back trace is printed to the file, and a message printed to - /// std::cerr with the backtrace file name and instructions for how - /// to post-process it. - void print(bool stop_at_main=true) const; - const std::vector<source_location>& frames() const { return frames_; } + std::vector<source_location>& frames() { return frames_; } friend std::ostream& operator<<(std::ostream&, const backtrace&); + // remove the top N=1 frames + backtrace& pop(std::size_t n=1); + + std::string to_string() const; + private: std::vector<source_location> frames_; }; diff --git a/cmake/FindUnwind.cmake b/cmake/FindUnwind.cmake deleted file mode 100644 index 000f9dd60ca87ecf393cfff67c7b8d2f6b839555..0000000000000000000000000000000000000000 --- a/cmake/FindUnwind.cmake +++ /dev/null @@ -1,65 +0,0 @@ -# Find the libunwind library -# -# Unwind_FOUND - True if libunwind was found -# Unwind_LIBRARIES - The libraries needed to use libunwind -# Unwind_INCLUDE_DIR - Location of unwind.h and libunwind.h -# -# The environment and cmake variables Unwind_ROOT and Unwind_ROOT_DIR -# respectively can be used to help CMake finding the library if it -# is not installed in any of the usual locations. -# -# Registers "Unwind::unwind" as an import library. - -if(NOT Unwind_FOUND) - set(Unwind_SEARCH_DIR ${Unwind_ROOT_DIR} $ENV{Unwind_ROOT}) - - find_path(Unwind_INCLUDE_DIR libunwind.h - HINTS ${Unwind_SEARCH_DIR} - PATH_SUFFIXES include - ) - - # libunwind requires that we link agains both libunwind.so/a and a - # a target-specific library libunwind-target.so/a. - # This code sets the "target" string above in libunwind_arch. - if (CMAKE_SYSTEM_PROCESSOR MATCHES "^arm") - set(_libunwind_arch "arm") - elseif (CMAKE_SYSTEM_PROCESSOR STREQUAL "x86_64" OR CMAKE_SYSTEM_PROCESSOR STREQUAL "amd64") - set(_libunwind_arch "x86_64") - elseif (CMAKE_SYSTEM_PROCESSOR MATCHES "^i.86$") - set(_libunwind_arch "x86") - endif() - - find_library(_unwind_library_generic unwind - HINTS ${Unwind_SEARCH_DIR} - PATH_SUFFIXES lib64 lib - ) - - find_library(_unwind_library_target unwind-${_libunwind_arch} - HINTS ${Unwind_SEARCH_DIR} - PATH_SUFFIXES lib64 lib - ) - - set(Unwind_LIBRARIES ${_unwind_library_generic} ${_unwind_library_target}) - - include(FindPackageHandleStandardArgs) - find_package_handle_standard_args(Unwind DEFAULT_MSG Unwind_INCLUDE_DIR Unwind_LIBRARIES) - - mark_as_advanced(Unwind_LIBRARIES Unwind_INCLUDE_DIR) - - if(Unwind_FOUND) - set(Unwind_INCLUDE_DIRS ${Unwind_INCLUDE_DIR}) - if(NOT TARGET Unwind::unwind) - add_library(Unwind::unwind UNKNOWN IMPORTED) - set_target_properties(Unwind::unwind PROPERTIES - IMPORTED_LOCATION "${_unwind_library_generic}" - INTERFACE_LINK_LIBRARIES "${_unwind_library_target}" - INTERFACE_INCLUDE_DIRECTORIES "${Unwind_INCLUDE_DIR}" - ) - endif() - endif() - - unset(_unwind_search_dir) - unset(_unwind_library_generic) - unset(_unwind_library_target) - unset(_libunwind_arch) -endif() diff --git a/doc/contrib/index.rst b/doc/contrib/index.rst index 661e4de9a0efc11779be28bcf16e0e51607cb0a2..fc995c347a9afd637c1e9968e399e770ebbbb22f 100644 --- a/doc/contrib/index.rst +++ b/doc/contrib/index.rst @@ -61,12 +61,12 @@ share your Arbor simulations or publications! Filing an issue ~~~~~~~~~~~~~~~ -If you have found a bug or problem in Arbor, or want to request a -feature, you can use our `issue -tracker <https://github.com/arbor-sim/arbor/issues>`__. If you issue is -not yet filed in the issue tracker, please do so and describe the -problem, bug or feature as best you can. You can add supporting data, -code or documents to help make your point. +If you have found a bug or problem in Arbor, or want to request a feature, you +can use our `issue tracker <https://github.com/arbor-sim/arbor/issues>`__. If +you issue is not yet filed in the issue tracker, please do so and describe the +problem, bug or feature as best you can. You can add supporting data, code or +documents to help make your point. For bugs in particular, stacktraces (either +from inside a debugger or by enabling ``ARB_BACKTRACE``) are extremely useful. .. _contribindex-solveissue: diff --git a/doc/dev/debug.rst b/doc/dev/debug.rst new file mode 100644 index 0000000000000000000000000000000000000000..e3ae77e50059a85a6165b58ddd797dc8068afa8a --- /dev/null +++ b/doc/dev/debug.rst @@ -0,0 +1,45 @@ +.. _dev-debug: + +Debugging +========= + +Backtraces +---------- + +When building Arbor you can enable backtraces in the CMake configure step by +setting ``ARB_BACKTRACE=ON``. Beware aware that this requires the ``Boost`` +libraries to be installed on your system. This will cause the following +additions to Arbor's behaviour + +1. Failed assertions via ``asb_assert`` will print the corresponding stacktrace. +2. All exceptions deriving from ``arbor_exception`` and ``arbor_internal_error`` + will have stacktraces attached in the ``where`` field. +3. Python exceptions derived from these types will add that same stacktrace + information to their message. + +Alternatively, you can obtain the same information using a debugger like GDB or +LLDB. + +.. note:: + + Since Arbor often uses a buffer of instructions on how to construct a + particular object instead of perform the action right away, errors occur not + always at the location you might expect. + + Consider this (adapted from ``network_ring.py``) + + .. code-block:: python + + class Recipe(arb.recipe): + # [...] + def connections_on(self, gid): + return [arbor.connection((src, "detector"), "syn-NOT", w, d)] + + def cell_description(self, gid): + # [...] + decor = (arbor.decor() + .place('"synapse_site"', arbor.synapse("expsyn"), "syn")) + return arbor.cable_cell(tree, labels, decor) + + rec = Recipe() # "Ok" + sim = arb.simulation(rec) # ERROR here diff --git a/doc/dev/index.rst b/doc/dev/index.rst index b20ca5a64b44af4feeee76e7edf3c6d94e2b78c4..0174b15f05839b48ee98987f8523c78e16dcf015 100644 --- a/doc/dev/index.rst +++ b/doc/dev/index.rst @@ -16,6 +16,7 @@ Here we document internal components of Arbor. These pages can be useful if you' cable_cell cell_groups + debug matrix_solver simd_api shared_state diff --git a/doc/install/build_install.rst b/doc/install/build_install.rst index de3d12dee9ccbe43c37dfd748b8bc33fbc8404f9..e426fac28a1da3b6358052a2ee72ccbb5f8a0a98 100644 --- a/doc/install/build_install.rst +++ b/doc/install/build_install.rst @@ -150,12 +150,19 @@ an additional support library ``arborio``. This library requires with NeuroML support enabled. See :ref:`install-neuroml` for more information. +Boost +~~~~~ + +When ``ARB_BACKTRACE`` is set to ``ON`` during configure we use +``Boost::stacktrace`` to print stacktraces upon failed assertions and attach +them to the base exception types ``arbor_exception`` and +``arbor_internal_error`` as ``where``. Documentation ~~~~~~~~~~~~~~ To build a local copy of the html documentation that you are reading now, you will need to -install `Sphinx <http://www.sphinx-doc.org/en/master/>`_. +install ``Sphinx <http://www.sphinx-doc.org/en/master/>`_. .. _install-downloading: diff --git a/modcc/CMakeLists.txt b/modcc/CMakeLists.txt index 51857c36103312e8deae7204bc6d98bda391eca8..62fb49f5a66402cdeaaf6410f4ac0a8207b5b1d8 100644 --- a/modcc/CMakeLists.txt +++ b/modcc/CMakeLists.txt @@ -41,7 +41,6 @@ if (ARB_USE_BUNDLED_FMT) $<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}/../ext/fmt/include>) target_compile_definitions(libmodcc PRIVATE FMT_HEADER_ONLY) - else() target_include_directories(libmodcc PUBLIC diff --git a/python/pyarb.cpp b/python/pyarb.cpp index c168d8be4f48c9715bb92edf2c7f30217f2d7ce1..c71293e8a0db3613d04e764b7a7cf6e55473170c 100644 --- a/python/pyarb.cpp +++ b/python/pyarb.cpp @@ -1,6 +1,8 @@ #include <pybind11/pybind11.h> #include <pybind11/numpy.h> +#include <sstream> + #include <arbor/spike.hpp> #include <arbor/common_types.hpp> #include <arbor/arbexcept.hpp> @@ -44,10 +46,6 @@ PYBIND11_MODULE(_arbor, m) { m.doc() = "arbor: multicompartment neural network models."; m.attr("__version__") = ARB_VERSION; - // Translate Arbor errors -> Python exceptions. - pybind11::register_exception<arb::file_not_found_error>(m, "FileNotFoundError", PyExc_FileNotFoundError); - pybind11::register_exception<arb::zero_thread_requested_error>(m, "ValueError", PyExc_ValueError); - pyarb::register_cable_loader(m); pyarb::register_cable_probes(m, global_ptr); pyarb::register_cells(m); @@ -64,6 +62,33 @@ PYBIND11_MODULE(_arbor, m) { pyarb::register_simulation(m, global_ptr); pyarb::register_single_cell(m); + // This is the fallback. All specific translators take precedence by being + // registered *later*. + pybind11::register_exception_translator([](std::exception_ptr p) { + try { + if (p) std::rethrow_exception(p); + } + catch (const arb::arbor_exception& e) { + std::stringstream msg; + msg << e.what() + << "\n" + << e.where; + PyErr_SetString(PyExc_RuntimeError, msg.str().c_str()); + } + catch (const arb::arbor_internal_error& e) { + std::stringstream msg; + msg << e.what() + << "\n" + << e.where; + PyErr_SetString(PyExc_RuntimeError, msg.str().c_str()); + } + }); + + // Translate Arbor errors -> Python exceptions. + pybind11::register_exception<arb::file_not_found_error>(m, "FileNotFoundError", PyExc_FileNotFoundError); + pybind11::register_exception<arb::zero_thread_requested_error>(m, "ValueError", PyExc_ValueError); + + #ifdef ARB_MPI_ENABLED pyarb::register_mpi(m); #endif