libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
PosixThread.cxx
1// C++
2#include <atomic>
3#include <cstdlib>
4#include <iostream>
5#include <utility>
6#include <variant>
7
8// cosmos
9#include <cosmos/error/ApiError.hxx>
10#include <cosmos/error/UsageError.hxx>
11#include <cosmos/formatting.hxx>
12#include <cosmos/private/cosmos.hxx>
13#include <cosmos/thread/PosixThread.hxx>
14
15using namespace cosmos::pthread;
16
17namespace cosmos {
18
19namespace {
20
21 std::atomic<size_t> num_threads = 0;
22
23 struct Context {
24 std::variant<PosixThread::PosixEntry, PosixThread::Entry> entry;
25 ThreadArg arg;
26 };
27
28 Context fetch_context(void *par) {
29 auto ctx = reinterpret_cast<Context*>(par);
30 auto ret = *ctx;
31 delete ctx;
32 return ret;
33 }
34
35 void* thread_entry(void *par) {
36
37 Context ctx = fetch_context(par);
38
39 if (std::holds_alternative<PosixThread::PosixEntry>(ctx.entry)) {
40 auto entry = std::get<PosixThread::PosixEntry>(ctx.entry);
41 auto ret = entry(ctx.arg);
42 return reinterpret_cast<void*>(ret);
43 } else {
44 auto entry = std::get<PosixThread::Entry>(ctx.entry);
45 entry();
46 return nullptr;
47 }
48 }
49
50 void create_thread(pthread_t &thread, Context *ctx) {
51
52 const auto res = ::pthread_create(
53 &thread,
54 nullptr /* keep default attributes */,
55 &thread_entry,
56 reinterpret_cast<void*>(ctx)
57 );
58
59 if (const auto error = Errno{res}; error != Errno::NO_ERROR) {
60 cosmos_throw (cosmos::ApiError("pthread_create()", error));
61 }
62
63 }
64} // end anon ns
65
66PosixThread::PosixThread(PosixEntry entry, pthread::ThreadArg arg, const std::string_view name) :
67 m_name{buildName(name, ++num_threads)} {
68
69 m_pthread = pthread_t{};
70 try {
71 create_thread(m_pthread.value(), new Context{entry, arg});
72 } catch(...) {
73 m_pthread.reset();
74 }
75}
76
77PosixThread::PosixThread(Entry entry, const std::string_view name) :
78 m_name{buildName(name, ++num_threads)} {
79 m_pthread = pthread_t{};
80 try {
81 create_thread(m_pthread.value(), new Context{entry, pthread::ThreadArg{0}});
82 } catch(...) {
83 m_pthread.reset();
84 }
85}
86
87PosixThread::PosixThread(PosixThread &&other) noexcept {
88 *this = std::move(other);
89}
90
91PosixThread::~PosixThread() {
92 if (joinable()) {
93 fatal_error(sprintf("Thread %s destroyed but not joined!", name().c_str()));
94 }
95}
96
97PosixThread& PosixThread::operator=(PosixThread &&other) noexcept {
98 if (joinable()) {
99 fatal_error("moving into not-yet-joined thread");
100 }
101 m_name = other.m_name;
102 m_pthread = *(other.m_pthread);
103
104 other.m_pthread.reset();
105 other.m_name.clear();
106 return *this;
107}
108
109void PosixThread::assertJoinConditions() {
110 if (!joinable()) {
111 cosmos_throw (UsageError("Attempted to join non-joinable thread (empty or detached)"));
112 } else if (isCallerThread()) {
113 cosmos_throw (UsageError("Attempted to join self"));
114 }
115}
116
117void PosixThread::reset() {
118 if (m_pthread) {
119 num_threads--;
120 m_pthread.reset();
121 m_name.clear();
122 }
123}
124
125pthread::ExitValue PosixThread::join() {
126 assertJoinConditions();
127
128 void *res = nullptr;
129 const auto join_res = ::pthread_join(*m_pthread, &res);
130
131 if (const auto error = Errno{join_res}; error != Errno::NO_ERROR) {
132 cosmos_throw (ApiError("pthread_join()", error));
133 }
134
135 reset();
136
137 return pthread::ExitValue{reinterpret_cast<intptr_t>(res)};
138}
139
140std::optional<pthread::ExitValue> PosixThread::tryJoin() {
141 assertJoinConditions();
142
143 void *res = nullptr;
144 const auto join_res = ::pthread_tryjoin_np(*m_pthread, &res);
145
146 if (const auto error = Errno{join_res}; error != Errno::NO_ERROR) {
147 if (error == Errno::BUSY) {
148 // cannot be joined yet
149 return {};
150 }
151
152 cosmos_throw (ApiError("pthread_tryjoin_np()", error));
153 }
154
155 reset();
156
157 return pthread::ExitValue{reinterpret_cast<intptr_t>(res)};
158}
159
160std::optional<pthread::ExitValue> PosixThread::joinTimed(const RealTime ts) {
161 assertJoinConditions();
162
163 void *res = nullptr;
164 const auto join_res = ::pthread_timedjoin_np(*m_pthread, &res, &ts);
165
166 if (const auto error = Errno{join_res}; error != Errno::NO_ERROR) {
167 if (error == Errno::TIMEDOUT) {
168 // couldn't join in time
169 return {};
170 }
171
172 cosmos_throw (ApiError("pthread_timedjoin_np()", error));
173 }
174
175 reset();
176
177 return pthread::ExitValue{reinterpret_cast<intptr_t>(res)};
178}
179
180void PosixThread::detach() {
181 // NOTE: in theory it is valid that a thread detaches itself, so let's
182 // not use assertJoinConditions() here ATM.
183 if (!joinable()) {
184 cosmos_throw (UsageError("Attempted to detach a non-joinable thread (empty or already detached)"));
185 }
186
187 const auto res = ::pthread_detach(*m_pthread);
188
189 if (const auto error = Errno{res}; error != Errno::NO_ERROR) {
190 cosmos_throw (ApiError("pthread_detach()", error));
191 }
192
193 reset();
194}
195
196std::string PosixThread::buildName(const std::string_view name, size_t nr) const {
197 if (!name.empty())
198 return std::string{name};
199 return std::string{"thread<" + std::to_string(nr) + ">"};
200}
201
202} // end ns
Specialized exception type used when system APIs fail.
Definition ApiError.hxx:18
A class representing a basic POSIX thread.
std::optional< pthread_t > m_pthread
POSIX thread handle.
PosixThread() noexcept
Creates an empty thread object.
std::function< pthread::ExitValue(pthread::ThreadArg)> PosixEntry
POSIX style entry function with a single input parameter and return value.
std::function< void(void)> Entry
Entry function without parameters for use with member functions or lambdas.
A C++ wrapper around the POSIX struct timespec coupled to a specific CLOCK type.
Definition types.hxx:57
Exception type for logical usage errors within the application.
Errno
Strong enum type representing errno error constants.
Definition errno.hxx:29
ExitValue
An integer or pointer return value from a pthread.
Definition pthread.hxx:46
ThreadArg
An integer or pointer value supplied to a pthread's entry function.
Definition pthread.hxx:49