libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
CosmosError.hxx
1#pragma once
2
3// C++
4#include <exception>
5#include <string>
6#include <string_view>
7
8// cosmos
9#include <cosmos/dso_export.h>
10#include <cosmos/error/macros.hxx>
11
12namespace cosmos {
13
14/*
15 * NOTE:
16 *
17 * It would be nice adding the possibility for a function call backtrace to
18 * the exception class. There are a bunch of obstacles to this, though:
19 *
20 * There exists a kind of base mechanism present in Glibc and GCC/Clang:
21 *
22 * - using backtrace() and backtrace_symbols() from <execinfo.h>
23 * - using abi::__cxa_demangle from <cxxabi.h>
24 *
25 * The limitations of this are:
26 *
27 * - no line number / source file information
28 * - symbolic names are only available if all objects file are compiled with
29 * `-rexport` and -fvisibility=default. Even then static function names
30 * cannot be resolved.
31 *
32 * There exists libunwind: is it is readily available and offers symbolic
33 * names even without changing the compilation and linking process. It does
34 * not offer symbol demangling though which needs to come from cxxabi.h
35 * instead. Also it comes with a file descriptor leak, because it creates a
36 * pipe pair that is never cleaned up. Also it does not offer line number of
37 * source file information.
38 *
39 * There are lesser known libraries like libbacktrace or backward-cpp, both of
40 * them are rather big and not readily available, also not very actively
41 * maintained (?).
42 *
43 * Further "hacks" to get additional debug information would be calling
44 * external tools like `addr2line` or `gdb`.
45 *
46 * Another problem is that adding only the option to have a backtrace can
47 * considerably increase the exception object size. A std::vector<std::string>
48 * would add 24 bytes to the object size. A single std::string is similarly
49 * sized. It could be heap-allocated which would reduce the overhead for
50 * non-use to 8 bytes but then we have to manage the pointer e.g. when
51 * copying, a shared_ptr would add 16 bytes of overhead.
52 *
53 * Overall to implement this feature maybe implementing parts of it on foot
54 * using existing code snippets from libbacktrace & friends could be
55 * investigated. Currently the problems seem not worth the gain.
56 */
57
59
74class COSMOS_API CosmosError :
75 public std::exception {
76public: // functions
77
78 explicit CosmosError(const std::string_view error_class, const std::string_view fixed_text = {}) :
79 m_error_class{error_class} {
80 m_msg = fixed_text;
81 }
82
84
89 CosmosError& setInfo(const char *file, const size_t line, const char *func) {
90 m_line = line;
91 m_file = file;
92 m_func = func;
93
94 return *this;
95 }
96
98
103 const char* what() const throw() override;
104
106 std::string shortWhat() const;
107
109 [[ noreturn ]] virtual void raise() = 0;
110
111protected: // functions
112
114
125 virtual void generateMsg() const {};
126
128 void setErrorClass(const std::string_view error_class) {
129 m_error_class = error_class;
130 }
131
132protected: // data
133
135 std::string_view m_error_class;
137 mutable std::string m_msg;
139 mutable bool m_msg_generated = false;
140 const char *m_file = nullptr;
141 const char *m_func = nullptr;
142 size_t m_line = 0;
143};
144
145} // end ns
Base class for libcosmos exceptions.
void setErrorClass(const std::string_view error_class)
Allows to override error class to allow simpler implementation of derived types.
std::string m_msg
Runtime generated error message.
CosmosError & setInfo(const char *file, const size_t line, const char *func)
Set exception context information.
std::string_view m_error_class
Descriptive, unique error class label.