libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
cosmos.cxx
1// Linux
2#include <unistd.h>
3
4// C++
5#include <atomic>
6#include <iostream>
7#include <map>
8
9// Cosmos
10#include <cosmos/cosmos.hxx>
11#include <cosmos/private/Initable.hxx>
12#include <cosmos/private/cosmos.hxx>
13#include <cosmos/proc/process.hxx>
14
15namespace cosmos {
16
17// maintain a map with a priority value as key and the Initable as a value.
18// the priority value allows to give a defined initialization order.
19// lower priority value means earlier initialization.
20typedef std::map<InitPrio, Initable*> InitableMap;
21
22struct InitData {
23
24 void add(const InitPrio prio, Initable *init_if);
25 void init();
26 void finish();
27protected:
28 InitableMap initables;
29};
30
31namespace {
32 std::atomic<std::size_t> init_counter;
33 InitData *init_data = nullptr;
34
35 void freeInitData() {
36 delete init_data;
37 init_data = nullptr;
38 }
39
40 void createInitData() {
41 if (init_data)
42 return;
43
44 /*
45 * this gives us a problem with static initialization order.
46 * we need to keep the map on the heap to make sure the object
47 * is valid at this point in time (because this function could
48 * be invoked in static initialization context).
49 *
50 * This on the other hand gives us trouble with freeing the
51 * heap memory, since we want to allow re-initialization. An
52 * atexit() handler will at least silence memory leaks
53 * reported by valgrind.
54 */
55 init_data = new InitData;
56 atexit(freeInitData);
57 }
58} // end anon ns
59
60void InitData::add(const InitPrio prio, Initable *init_if) {
61
62 auto ret = initables.insert(std::make_pair(prio, init_if));
63
64 if (ret.second != true) {
65 std::cerr << "Conflicting priority of Initables!\n";
66 _exit(9);
67 }
68}
69
70void InitData::init() {
71 for (auto &[prio, initable]: initables) {
72 initable->libInit();
73 initable->m_lib_initialized = true;
74 }
75}
76
77void InitData::finish() {
78 // perform cleanup - in reverse order
79 for (auto it = initables.rbegin(); it != initables.rend(); it++) {
80 auto initable = it->second;
81 initable->libExit();
82 initable->m_lib_initialized = false;
83 }
84}
85
86void Initable::registerInitable(const InitPrio prio) {
87 createInitData();
88
89 init_data->add(prio, this);
90}
91
92void init() {
93 if (init_counter++ != 0)
94 return;
95
96 if (!init_data)
97 // no initables have been registered
98 return;
99
100 // okay we need to perform initialization
101 init_data->init();
102}
103
104void finish() {
105 if (--init_counter != 0)
106 return;
107
108 if (!init_data)
109 // no initables around
110 return;
111
112 init_data->finish();
113}
114
115RestartOnIntr auto_restart_syscalls{true};
116
117void set_restart_syscall_on_interrupt(const bool auto_restart) {
118 auto_restart_syscalls = RestartOnIntr{auto_restart};
119}
120
121void fatal_error(const std::string_view msg, const std::exception *ex) {
122 std::cerr << "[libcosmos] FATAL: " << msg << "\n";
123 if (ex) {
124 std::cerr << "Exception context:\n\n" << ex->what() << "\n";
125 }
126 std::cerr << "Aborting program.\n";
127 std::abort();
128}
129
130void noncritical_error(
131 const std::string_view msg,
132 const std::exception &ex) {
133 std::cerr << "[libcosmos] WARNING: " << msg << "\n";
134 std::cerr << "Exception context:\n\n" << ex.what() << "\n";
135}
136
137
138RunningOnValgrind running_on_valgrind{false};
139
140namespace {
141
143class RunningOnValgrindInit :
144 public Initable {
145public: // functions
146
147 RunningOnValgrindInit() : Initable(InitPrio::RUNNING_ON_VALGRIND) {
148 }
149
150protected: // functions
151
152 void libInit() override {
153 /* this is a heuristic to detect whether we are running in
154 * Valgrind context. If LD_PRELOAD has references to valgrind
155 * then it is most likely the case.
156 *
157 * For a proper check we'd need to include 'valgrind.h' which
158 * entails a lot of other stuff like a configure checks and so
159 * on.
160 */
161 if (auto ld_preload = proc::get_env_var("LD_PRELOAD"); ld_preload) {
162 auto view = ld_preload->view();
163 if (view.find("valgrind") != view.npos) {
164 running_on_valgrind = RunningOnValgrind{true};
165 }
166 }
167 }
168
169 void libExit() override {
170 // no-op
171 }
172};
173
174RunningOnValgrindInit g_running_on_valgrind_init;
175
176} // end anon ns
177
178} // end ns
Pure virtual base class for the library init system.
Definition Initable.hxx:28