From 49f178ed94b5fad00d25dbd12adea0bf4732f803 Mon Sep 17 00:00:00 2001 From: gatecat Date: Fri, 8 Apr 2022 13:42:54 +0100 Subject: Split up common into kernel,place,route Signed-off-by: gatecat --- common/kernel/handle_error.cc | 61 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 61 insertions(+) create mode 100644 common/kernel/handle_error.cc (limited to 'common/kernel/handle_error.cc') diff --git a/common/kernel/handle_error.cc b/common/kernel/handle_error.cc new file mode 100644 index 00000000..d5542369 --- /dev/null +++ b/common/kernel/handle_error.cc @@ -0,0 +1,61 @@ +#ifndef NO_PYTHON + +#include +#include +#include "nextpnr.h" + +namespace py = pybind11; + +NEXTPNR_NAMESPACE_BEGIN + +// Parses the value of the active python exception +// NOTE SHOULD NOT BE CALLED IF NO EXCEPTION +std::string parse_python_exception() +{ + PyObject *type_ptr = NULL, *value_ptr = NULL, *traceback_ptr = NULL; + // Fetch the exception info from the Python C API + PyErr_Fetch(&type_ptr, &value_ptr, &traceback_ptr); + + // Fallback error + std::string ret("Unfetchable Python error"); + // If the fetch got a type pointer, parse the type into the exception string + if (type_ptr != NULL) { + py::object obj = py::reinterpret_borrow(type_ptr); + // If a valid string extraction is available, use it + // otherwise use fallback + if (py::isinstance(obj)) + ret = obj.cast(); + else + ret = "Unknown exception type"; + } + // Do the same for the exception value (the stringification of the + // exception) + if (value_ptr != NULL) { + py::object obj = py::reinterpret_borrow(value_ptr); + if (py::isinstance(obj)) + ret += ": " + obj.cast(); + else + ret += std::string(": Unparseable Python error: "); + } + // Parse lines from the traceback using the Python traceback module + if (traceback_ptr != NULL) { + py::handle h_tb(traceback_ptr); + // Load the traceback module and the format_tb function + py::object tb(py::module::import("traceback")); + py::object fmt_tb(tb.attr("format_tb")); + // Call format_tb to get a list of traceback strings + py::object tb_list(fmt_tb(h_tb)); + // Join the traceback strings into a single string + py::object tb_str(py::str("\n") + tb_list); + // Extract the string, check the extraction, and fallback in necessary + if (py::isinstance(tb_str)) + ret += ": " + tb_str.cast(); + else + ret += std::string(": Unparseable Python traceback"); + } + return ret; +} + +NEXTPNR_NAMESPACE_END + +#endif // NO_PYTHON \ No newline at end of file -- cgit v1.2.3