-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmacrothreading_condition.c
142 lines (137 loc) · 3.8 KB
/
macrothreading_condition.c
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
#include "macrothreading_condition.h"
#include <stdio.h>
macrothread_condition_t macrothread_condition_init()
{
#if defined MACROTHREADING_ESP32
return xEventGroupCreate();
#elif defined MACROTHREADING_PTHREADS
macrothread_condition_t result;
result = (macrothread_condition_t)malloc(sizeof(macrothread_condition_struct_t));
if(result == NULL) return NULL;
result->mutex = (pthread_mutex_t*)malloc(sizeof(pthread_mutex_t));
if(result->mutex == NULL) {
free(result);
return NULL;
}
if(pthread_mutex_init(result->mutex, NULL) != 0) {
free(result->mutex);
free(result);
return NULL;
}
result->cond = (pthread_cond_t*)malloc(sizeof(pthread_cond_t));
if(result->cond == NULL) {
free(result->mutex);
free(result);
return NULL;
}
if(pthread_cond_init(result->cond, NULL) != 0) {
free(result->mutex);
free(result->cond);
free(result);
return NULL;
}
result->signaled = false;
return result;
#elif defined MACROTHREADING_WINDOWS
macrothread_condition_t result;
result = (macrothread_condition_t)malloc(sizeof(macrothread_condition_struct_t));
if(result == NULL) return NULL;
result->mutex = (CRITICAL_SECTION*)malloc(sizeof(CRITICAL_SECTION));
if(result->mutex == NULL) {
free(result);
return NULL;
}
InitializeCriticalSection(result->mutex);
result->cond = (CONDITION_VARIABLE*)malloc(sizeof(CONDITION_VARIABLE));
if(result->cond == NULL) {
free(result->mutex);
free(result);
return NULL;
}
InitializeConditionVariable(result->cond);
result->signaled = false;
return result;
#else
return false;
#endif
}
void macrothread_condition_wait(macrothread_condition_t cond)
{
#if defined MACROTHREADING_ESP32
xEventGroupWaitBits(
cond,
MACROTHREADING_EVENT_MASK,
pdTRUE,
pdTRUE,
portMAX_DELAY
);
#elif defined MACROTHREADING_PTHREADS
if(pthread_mutex_lock(cond->mutex) != 0) {
exit(1);
}
while(!cond->signaled) {
if(pthread_cond_wait(cond->cond, cond->mutex) != 0) {
exit(1);
}
}
cond->signaled = false;
if(pthread_mutex_unlock(cond->mutex) != 0) {
exit(1);
}
#elif defined MACROTHREADING_WINDOWS
EnterCriticalSection(cond->mutex);
while(!cond->signaled) {
SleepConditionVariableCS (cond->cond, cond->mutex, INFINITE);
}
cond->signaled = false;
LeaveCriticalSection (cond->mutex);
#else
// Do nothing
#endif
}
void macrothread_condition_signal(macrothread_condition_t cond)
{
#if defined MACROTHREADING_ESP32
xEventGroupSetBits(cond, MACROTHREADING_EVENT_MASK);
#elif defined MACROTHREADING_PTHREADS
if(pthread_mutex_lock(cond->mutex) != 0) {
exit(1);
}
cond->signaled = true;
if(pthread_cond_signal(cond->cond) != 0) {
exit(1);
}
if(pthread_mutex_unlock(cond->mutex) != 0) {
exit(1);
}
#elif defined MACROTHREADING_WINDOWS
EnterCriticalSection(cond->mutex);
cond->signaled = true;
LeaveCriticalSection (cond->mutex);
WakeConditionVariable(cond->cond);
#else
// Do nothing
#endif
}
void macrothread_condition_destroy(macrothread_condition_t cond)
{
#if defined MACROTHREADING_ESP32
vEventGroupDelete(cond);
#elif defined MACROTHREADING_PTHREADS
if(pthread_cond_destroy(cond->cond) != 0) {
perror("pthread_cond_destroy error");
}
if(pthread_mutex_destroy(cond->mutex) != 0) {
perror("pthread_mutex_destroy error");
}
free(cond->cond);
free(cond->mutex);
free(cond);
#elif defined MACROTHREADING_WINDOWS
free(cond->cond);
free(cond->mutex);
free(cond);
#else
// Do nothing
#endif
}