This repository has been archived by the owner on Feb 11, 2021. It is now read-only.
/
SDL_syscond.c
167 lines (142 loc) · 3.85 KB
1
2
/*
SDL - Simple DirectMedia Layer
3
Copyright (C) 1997-2010 Sam Lantinga
4
5
This library is free software; you can redistribute it and/or
6
modify it under the terms of the GNU Lesser General Public
7
License as published by the Free Software Foundation; either
8
version 2.1 of the License, or (at your option) any later version.
9
10
11
12
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13
Lesser General Public License for more details.
15
16
17
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18
19
20
21
Sam Lantinga
slouken@libsdl.org
*/
22
#include "SDL_config.h"
23
24
25
/* RISC OS implementations uses pthreads based on linux code */
26
#if SDL_THREADS_DISABLED
27
28
29
30
31
32
33
34
35
36
37
#include "../generic/SDL_syscond.c"
#else
#include <sys/time.h>
#include <unistd.h>
#include <pthread.h>
#include "SDL_thread.h"
#include "SDL_sysmutex_c.h"
struct SDL_cond
{
38
pthread_cond_t cond;
39
40
41
};
/* Create a condition variable */
42
43
SDL_cond *
SDL_CreateCond(void)
45
46
47
48
49
50
51
52
53
54
55
SDL_cond *cond;
cond = (SDL_cond *) SDL_malloc(sizeof(SDL_cond));
if (cond) {
if (pthread_cond_init(&cond->cond, NULL) < 0) {
SDL_SetError("pthread_cond_init() failed");
SDL_free(cond);
cond = NULL;
}
}
return (cond);
56
57
58
}
/* Destroy a condition variable */
59
60
void
SDL_DestroyCond(SDL_cond * cond)
62
63
64
65
if (cond) {
pthread_cond_destroy(&cond->cond);
SDL_free(cond);
}
66
67
68
}
/* Restart one of the threads that are waiting on the condition variable */
69
70
int
SDL_CondSignal(SDL_cond * cond)
72
73
74
75
76
77
78
79
80
81
82
83
84
int retval;
if (!cond) {
SDL_SetError("Passed a NULL condition variable");
return -1;
}
retval = 0;
if (pthread_cond_signal(&cond->cond) != 0) {
SDL_SetError("pthread_cond_signal() failed");
retval = -1;
}
return retval;
85
86
87
}
/* Restart all threads that are waiting on the condition variable */
88
89
int
SDL_CondBroadcast(SDL_cond * cond)
91
92
93
94
95
96
97
98
99
100
101
102
103
int retval;
if (!cond) {
SDL_SetError("Passed a NULL condition variable");
return -1;
}
retval = 0;
if (pthread_cond_broadcast(&cond->cond) != 0) {
SDL_SetError("pthread_cond_broadcast() failed");
retval = -1;
}
return retval;
106
107
int
SDL_CondWaitTimeout(SDL_cond * cond, SDL_mutex * mutex, Uint32 ms)
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
int retval;
struct timeval delta;
struct timespec abstime;
if (!cond) {
SDL_SetError("Passed a NULL condition variable");
return -1;
}
gettimeofday(&delta, NULL);
abstime.tv_sec = delta.tv_sec + (ms / 1000);
abstime.tv_nsec = (delta.tv_usec + (ms % 1000) * 1000) * 1000;
if (abstime.tv_nsec > 1000000000) {
abstime.tv_sec += 1;
abstime.tv_nsec -= 1000000000;
}
126
127
tryagain:
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
retval = pthread_cond_timedwait(&cond->cond, &mutex->id, &abstime);
switch (retval) {
case EINTR:
goto tryagain;
break;
case ETIMEDOUT:
retval = SDL_MUTEX_TIMEDOUT;
break;
case 0:
break;
default:
SDL_SetError("pthread_cond_timedwait() failed");
retval = -1;
break;
}
return retval;
144
145
146
147
148
}
/* Wait on the condition variable, unlocking the provided mutex.
The mutex must be locked before entering this function!
*/
149
150
int
SDL_CondWait(SDL_cond * cond, SDL_mutex * mutex)
152
153
154
155
156
157
158
159
160
161
162
163
164
int retval;
if (!cond) {
SDL_SetError("Passed a NULL condition variable");
return -1;
}
retval = 0;
if (pthread_cond_wait(&cond->cond, &mutex->id) != 0) {
SDL_SetError("pthread_cond_wait() failed");
retval = -1;
}
return retval;
165
166
}
#endif
167
/* vi: set ts=4 sw=4 expandtab: */