aboutsummaryrefslogtreecommitdiff
path: root/works/life/operating-system-experiment/Thread.cpp
blob: 0bc4c1804cbc652c1e40c0038d840eaec6b523db (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
#include "Thread.h"

#include <cassert>
#include <exception>
#include <utility>

namespace cru {
Thread::Thread(Thread &&other) noexcept
    : detached_(other.detached_), joined_(other.joined_),
#ifdef CRU_WINDOWS
      thread_handle_(other.thread_handle_)
#else
      thread_(std::move(other.thread_))
#endif
{
  other.joined_ = false;
#ifdef CRU_WINDOWS
  other.thread_handle_ = nullptr;
#endif
} // namespace cru

Thread &Thread::operator=(Thread &&other) noexcept {
  if (this != &other) {
    detached_ = other.detached_;
    joined_ = other.joined_;
#ifdef CRU_WINDOWS
    thread_handle_ = other.thread_handle_;
    other.thread_handle_ = nullptr;
#else
    thread_ = std::move(other.thread_);
#endif
    other.detached_ = false;
    other.joined_ = false;
  }

  return *this;
}

Thread::~Thread() { Destroy(); }

void Thread::Join() {
  joined_ = true;
#ifdef CRU_WINDOWS
  assert(thread_handle_);
  WaitForSingleObject(thread_handle_, INFINITE);
#else
  assert(thread_);
  auto c = pthread_join(*thread_, nullptr);
  assert(c == 0);
#endif
}

void Thread::Detach() {
#ifdef CRU_WINDOWS
  assert(thread_handle_);
#else
  assert(thread_);
#endif
  detached_ = true;
}

#ifdef CRU_WINDOWS
DWORD
#else
pthread_t
#endif
Thread::GetNativeID() {
#ifdef CRU_WINDOWS
  assert(thread_handle_);
  return thread_id_;
#else
  assert(thread_);
  return *thread_;
#endif
}

#ifdef CRU_WINDOWS
HANDLE
#else
pthread_t
#endif
Thread::GetNativeHandle() {
#ifdef CRU_WINDOWS
  assert(thread_handle_);
  return thread_handle_;
#else
  assert(thread_);
  return *thread_;
#endif
}

void Thread::swap(Thread &other) noexcept {
#ifdef CRU_WINDOWS
  Thread temp = std::move(*this);
  *this = std::move(other);
  other = std::move(temp);
#else
#endif
}

void Thread::Destroy() noexcept {
  if (!detached_ && !joined_ &&
#ifdef CRU_WINDOWS
      thread_handle_ != nullptr
#else
      thread_ != nullptr
#endif
  ) {
    std::terminate();
  } else {
    detached_ = false;
    joined_ = false;
#ifdef CRU_WINDOWS
    thread_id_ = 0;
    if (thread_handle_ != nullptr) {
      auto c = CloseHandle(thread_handle_);
      assert(c);
      thread_handle_ = nullptr;
    }
#else
    thread_ = nullptr;
#endif
  }
}

namespace {
#ifdef CRU_WINDOWS
DWORD WINAPI ThreadProc(_In_ LPVOID lpParameter) {
  auto p = static_cast<std::function<void()> *>(lpParameter);
  (*p)();
  delete p;
  return 0;
}
#else
void *ThreadProc(void *data) {
  auto p = static_cast<std::function<void()> *>(data);
  (*p)();
  delete p;
  return nullptr;
}

#endif
} // namespace

void Thread::CreateThread(std::function<void()> *proc) {
#ifdef CRU_WINDOWS
  thread_handle_ = ::CreateThread(nullptr, 0, ThreadProc,
                                  static_cast<void *>(proc), 0, &thread_id_);
  assert(thread_handle_);
#else
  thread_.reset(new pthread_t());
  auto c = pthread_create(thread_.get(), nullptr, ThreadProc,
                          static_cast<void *>(proc));
  assert(c == 0);
#endif
};

} // namespace cru