Objectively 1.0.0
Ultra-lightweight object oriented framework for GNU C.
Thread.c
Go to the documentation of this file.
1/*
2 * Objectively: Ultra-lightweight object oriented framework for GNU C.
3 * Copyright (C) 2014 Jay Dolan <jay@jaydolan.com>
4 *
5 * This software is provided 'as-is', without any express or implied
6 * warranty. In no event will the authors be held liable for any damages
7 * arising from the use of this software.
8 *
9 * Permission is granted to anyone to use this software for any purpose,
10 * including commercial applications, and to alter it and redistribute it
11 * freely, subject to the following restrictions:
12 *
13 * 1. The origin of this software must not be misrepresented; you must not
14 * claim that you wrote the original software. If you use this software
15 * in a product, an acknowledgment in the product documentation would be
16 * appreciated but is not required.
17 *
18 * 2. Altered source versions must be plainly marked as such, and must not be
19 * misrepresented as being the original software.
20 *
21 * 3. This notice may not be removed or altered from any source distribution.
22 */
23
24#include <assert.h>
25#include <errno.h>
26#include <signal.h>
27#include <stdlib.h>
28
29#include <pthread.h>
30
31#include "Thread.h"
32
33#define _Class _Thread
34
35#pragma mark - Object
36
40static Object *copy(const Object *self) {
41 return NULL;
42}
43
47static void dealloc(Object *self) {
48
49 Thread *this = (Thread *) self;
50
51 assert(this->isExecuting == false);
52
53 free(this->thread);
54
55 super(Object, self, dealloc);
56}
57
58#pragma mark - Thread
59
64static void cancel(Thread *self) {
65
66 assert(self->isCancelled == false);
67
68 // int err = pthread_cancel(*((pthread_t *) self->thread));
69 // assert(err == 0);
70
71 self->isCancelled = true;
72}
73
74static __thread Thread *_currentThread;
75
80static Thread *currentThread(void) {
81
82 return _currentThread;
83}
84
89static void detach(Thread *self) {
90
91 assert(self->isDetached == false);
92
93 int err = pthread_detach(*((pthread_t *) self->thread));
94 assert(err == 0);
95
96 self->isDetached = true;
97}
98
103static Thread *init(Thread *self) {
104
105 return $(self, initWithFunction, NULL, NULL);
106}
107
113
114 self = (Thread *) super(Object, self, init);
115 if (self) {
116 self->function = function;
117 self->data = data;
118
119 self->thread = calloc(1, sizeof(pthread_t));
120 assert(self->thread);
121 }
122
123 return self;
124}
125
130static void join(Thread *self, ident *status) {
131
132 int err = pthread_join(*((pthread_t *) self->thread), status);
133 assert(err == 0);
134}
135
140static void _kill(Thread *self, int signal) {
141
142 int err = pthread_kill(*((pthread_t *) self->thread), signal);
143 assert(err == 0);
144}
145
150
151 Thread *self = _currentThread = (Thread *) obj;
152
153 self->isExecuting = true;
154
155 ident ret = self->function(self);
156
157 self->isFinished = true;
158
159 self->isExecuting = false;
160
161 return ret;
162}
163
168static void start(Thread *self) {
169
170 assert(self->function);
171
172 assert(self->isCancelled == false);
173 assert(self->isDetached == false);
174 assert(self->isExecuting == false);
175 assert(self->isFinished == false);
176
177 int err = pthread_create(self->thread, NULL, run, self);
178 assert(err == 0);
179}
180
181#pragma mark - Class lifecycle
182
186static void initialize(Class *clazz) {
187
188 ((ObjectInterface *) clazz->interface)->copy = copy;
189 ((ObjectInterface *) clazz->interface)->dealloc = dealloc;
190
191 ((ThreadInterface *) clazz->interface)->cancel = cancel;
192 ((ThreadInterface *) clazz->interface)->currentThread = currentThread;
193 ((ThreadInterface *) clazz->interface)->detach = detach;
194 ((ThreadInterface *) clazz->interface)->init = init;
195 ((ThreadInterface *) clazz->interface)->initWithFunction = initWithFunction;
196 ((ThreadInterface *) clazz->interface)->join = join;
197 ((ThreadInterface *) clazz->interface)->kill = _kill;
198 ((ThreadInterface *) clazz->interface)->start = start;
199}
200
206 static Class *clazz;
207 static Once once;
208
209 do_once(&once, {
210 clazz = _initialize(&(const ClassDef) {
211 .name = "Thread",
212 .superclass = _Object(),
213 .instanceSize = sizeof(Thread),
214 .interfaceOffset = offsetof(Thread, interface),
215 .interfaceSize = sizeof(ThreadInterface),
217 });
218 });
219
220 return clazz;
221}
222
223#undef _Class
Class * _initialize(const ClassDef *def)
Initializes the given Class.
Definition: Class.c:91
#define obj
#define super(type, obj, method,...)
static void _kill(Thread *self, int signal)
Definition: Thread.c:140
static ident run(ident obj)
Wraps the user-specified ThreadFunction, providing cleanup.
Definition: Thread.c:149
static __thread Thread * _currentThread
Definition: Thread.c:74
static void initialize(Class *clazz)
Definition: Thread.c:186
POSIX Threads.
ident(* ThreadFunction)(Thread *thread)
The function type for Thread execution.
Definition: Thread.h:45
void * ident
The identity type, similar to Objective-C id.
Definition: Types.h:49
long Once
The Once type.
Definition: Once.h:37
#define do_once(once, block)
Executes the given block at most one time.
Definition: Once.h:43
ClassDefs are passed to _initialize via an archetype to initialize a Class.
Definition: Class.h:41
The runtime representation of a Class.
Definition: Class.h:95
ident interface
The interface of the Class.
Definition: Class.h:105
Condition * init(Condition *self)
Initializes this Condition.
Definition: Condition.c:67
void signal(Condition *self)
Signals a single Thread waiting on this Condition.
MutableData * data(void)
Returns a new MutableData.
Definition: MutableData.c:75
Object is the root Class of The Objectively Class hierarchy.
Definition: Object.h:46
Class * _Object(void)
The Object archetype.
Definition: Object.c:136
Object * copy(const Object *self)
Creates a shallow copy of this Object.
Definition: Array.c:40
void dealloc(Object *self)
Frees all resources held by this Object.
Definition: Array.c:50
void start(Operation *self)
Starts this Operation.
Definition: Operation.c:171
void cancel(Operation *self)
Cancels this Operation, allowing it to complete immediately.
Definition: Operation.c:74
Operation * initWithFunction(Operation *self, OperationFunction function, ident data)
Initializes a synchronous Operation with the given function.
Definition: Operation.c:119
POSIX Threads.
Definition: Thread.h:53
ident data
The user data.
Definition: Thread.h:69
void join(Thread *self, ident *status)
Wait for the specified Thread to terminate.
Definition: Thread.c:130
ThreadFunction function
The Thread function.
Definition: Thread.h:74
Class * _Thread(void)
The Thread archetype.
Definition: Thread.c:205
_Bool isExecuting
true when this Thread is executing, false otherwise.
Definition: Thread.h:89
Thread * currentThread(void)
Returns the currently executing Thread.
Definition: Thread.c:80
_Bool isFinished
true when this Thread is finished, false otherwise.
Definition: Thread.h:94
_Bool isDetached
true when this Thread has been detached, false otherwise.
Definition: Thread.h:84
void detach(Thread *self)
Daemonize this Thread.
Definition: Thread.c:89
_Bool isCancelled
true when this Thread has been cancelled, false otherwise.
Definition: Thread.h:79