1 /*
2 * libusb synchronization on Microsoft Windows
3 *
4 * Copyright © 2010 Michael Plante <michael.plante@gmail.com>
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19 */
20
21 #include <config.h>
22
23 #include <objbase.h>
24 #include <errno.h>
25
26 #include "libusbi.h"
27
28 struct usbi_cond_perthread {
29 struct list_head list;
30 DWORD tid;
31 HANDLE event;
32 };
33
usbi_mutex_static_lock(usbi_mutex_static_t * mutex)34 int usbi_mutex_static_lock(usbi_mutex_static_t *mutex)
35 {
36 if (!mutex)
37 return EINVAL;
38 while (InterlockedExchange(mutex, 1) == 1)
39 SleepEx(0, TRUE);
40 return 0;
41 }
42
usbi_mutex_static_unlock(usbi_mutex_static_t * mutex)43 int usbi_mutex_static_unlock(usbi_mutex_static_t *mutex)
44 {
45 if (!mutex)
46 return EINVAL;
47 InterlockedExchange(mutex, 0);
48 return 0;
49 }
50
usbi_mutex_init(usbi_mutex_t * mutex)51 int usbi_mutex_init(usbi_mutex_t *mutex)
52 {
53 if (!mutex)
54 return EINVAL;
55 *mutex = CreateMutex(NULL, FALSE, NULL);
56 if (!*mutex)
57 return ENOMEM;
58 return 0;
59 }
60
usbi_mutex_lock(usbi_mutex_t * mutex)61 int usbi_mutex_lock(usbi_mutex_t *mutex)
62 {
63 DWORD result;
64
65 if (!mutex)
66 return EINVAL;
67 result = WaitForSingleObject(*mutex, INFINITE);
68 if (result == WAIT_OBJECT_0 || result == WAIT_ABANDONED)
69 return 0; // acquired (ToDo: check that abandoned is ok)
70 else
71 return EINVAL; // don't know how this would happen
72 // so don't know proper errno
73 }
74
usbi_mutex_unlock(usbi_mutex_t * mutex)75 int usbi_mutex_unlock(usbi_mutex_t *mutex)
76 {
77 if (!mutex)
78 return EINVAL;
79 if (ReleaseMutex(*mutex))
80 return 0;
81 else
82 return EPERM;
83 }
84
usbi_mutex_trylock(usbi_mutex_t * mutex)85 int usbi_mutex_trylock(usbi_mutex_t *mutex)
86 {
87 DWORD result;
88
89 if (!mutex)
90 return EINVAL;
91 result = WaitForSingleObject(*mutex, 0);
92 if (result == WAIT_OBJECT_0 || result == WAIT_ABANDONED)
93 return 0; // acquired (ToDo: check that abandoned is ok)
94 else if (result == WAIT_TIMEOUT)
95 return EBUSY;
96 else
97 return EINVAL; // don't know how this would happen
98 // so don't know proper error
99 }
100
usbi_mutex_destroy(usbi_mutex_t * mutex)101 int usbi_mutex_destroy(usbi_mutex_t *mutex)
102 {
103 // It is not clear if CloseHandle failure is due to failure to unlock.
104 // If so, this should be errno=EBUSY.
105 if (!mutex || !CloseHandle(*mutex))
106 return EINVAL;
107 *mutex = NULL;
108 return 0;
109 }
110
usbi_cond_init(usbi_cond_t * cond)111 int usbi_cond_init(usbi_cond_t *cond)
112 {
113 if (!cond)
114 return EINVAL;
115 list_init(&cond->waiters);
116 list_init(&cond->not_waiting);
117 return 0;
118 }
119
usbi_cond_destroy(usbi_cond_t * cond)120 int usbi_cond_destroy(usbi_cond_t *cond)
121 {
122 // This assumes no one is using this anymore. The check MAY NOT BE safe.
123 struct usbi_cond_perthread *pos, *next_pos;
124
125 if(!cond)
126 return EINVAL;
127 if (!list_empty(&cond->waiters))
128 return EBUSY; // (!see above!)
129 list_for_each_entry_safe(pos, next_pos, &cond->not_waiting, list, struct usbi_cond_perthread) {
130 CloseHandle(pos->event);
131 list_del(&pos->list);
132 free(pos);
133 }
134 return 0;
135 }
136
usbi_cond_broadcast(usbi_cond_t * cond)137 int usbi_cond_broadcast(usbi_cond_t *cond)
138 {
139 // Assumes mutex is locked; this is not in keeping with POSIX spec, but
140 // libusb does this anyway, so we simplify by not adding more sync
141 // primitives to the CV definition!
142 int fail = 0;
143 struct usbi_cond_perthread *pos;
144
145 if (!cond)
146 return EINVAL;
147 list_for_each_entry(pos, &cond->waiters, list, struct usbi_cond_perthread) {
148 if (!SetEvent(pos->event))
149 fail = 1;
150 }
151 // The wait function will remove its respective item from the list.
152 return fail ? EINVAL : 0;
153 }
154
usbi_cond_intwait(usbi_cond_t * cond,usbi_mutex_t * mutex,DWORD timeout_ms)155 __inline static int usbi_cond_intwait(usbi_cond_t *cond,
156 usbi_mutex_t *mutex, DWORD timeout_ms)
157 {
158 struct usbi_cond_perthread *pos;
159 int r, found = 0;
160 DWORD r2, tid = GetCurrentThreadId();
161
162 if (!cond || !mutex)
163 return EINVAL;
164 list_for_each_entry(pos, &cond->not_waiting, list, struct usbi_cond_perthread) {
165 if(tid == pos->tid) {
166 found = 1;
167 break;
168 }
169 }
170
171 if (!found) {
172 pos = calloc(1, sizeof(struct usbi_cond_perthread));
173 if (!pos)
174 return ENOMEM; // This errno is not POSIX-allowed.
175 pos->tid = tid;
176 pos->event = CreateEvent(NULL, FALSE, FALSE, NULL); // auto-reset.
177 if (!pos->event) {
178 free(pos);
179 return ENOMEM;
180 }
181 list_add(&pos->list, &cond->not_waiting);
182 }
183
184 list_del(&pos->list); // remove from not_waiting list.
185 list_add(&pos->list, &cond->waiters);
186
187 r = usbi_mutex_unlock(mutex);
188 if (r)
189 return r;
190
191 r2 = WaitForSingleObject(pos->event, timeout_ms);
192 r = usbi_mutex_lock(mutex);
193 if (r)
194 return r;
195
196 list_del(&pos->list);
197 list_add(&pos->list, &cond->not_waiting);
198
199 if (r2 == WAIT_OBJECT_0)
200 return 0;
201 else if (r2 == WAIT_TIMEOUT)
202 return ETIMEDOUT;
203 else
204 return EINVAL;
205 }
206 // N.B.: usbi_cond_*wait() can also return ENOMEM, even though pthread_cond_*wait cannot!
usbi_cond_wait(usbi_cond_t * cond,usbi_mutex_t * mutex)207 int usbi_cond_wait(usbi_cond_t *cond, usbi_mutex_t *mutex)
208 {
209 return usbi_cond_intwait(cond, mutex, INFINITE);
210 }
211
usbi_cond_timedwait(usbi_cond_t * cond,usbi_mutex_t * mutex,const struct timeval * tv)212 int usbi_cond_timedwait(usbi_cond_t *cond,
213 usbi_mutex_t *mutex, const struct timeval *tv)
214 {
215 DWORD millis;
216
217 millis = (DWORD)(tv->tv_sec * 1000) + (tv->tv_usec / 1000);
218 /* round up to next millisecond */
219 if (tv->tv_usec % 1000)
220 millis++;
221 return usbi_cond_intwait(cond, mutex, millis);
222 }
223
usbi_tls_key_create(usbi_tls_key_t * key)224 int usbi_tls_key_create(usbi_tls_key_t *key)
225 {
226 if (!key)
227 return EINVAL;
228 *key = TlsAlloc();
229 if (*key == TLS_OUT_OF_INDEXES)
230 return ENOMEM;
231 else
232 return 0;
233 }
234
usbi_tls_key_get(usbi_tls_key_t key)235 void *usbi_tls_key_get(usbi_tls_key_t key)
236 {
237 return TlsGetValue(key);
238 }
239
usbi_tls_key_set(usbi_tls_key_t key,void * value)240 int usbi_tls_key_set(usbi_tls_key_t key, void *value)
241 {
242 if (TlsSetValue(key, value))
243 return 0;
244 else
245 return EINVAL;
246 }
247
usbi_tls_key_delete(usbi_tls_key_t key)248 int usbi_tls_key_delete(usbi_tls_key_t key)
249 {
250 if (TlsFree(key))
251 return 0;
252 else
253 return EINVAL;
254 }
255
usbi_get_tid(void)256 int usbi_get_tid(void)
257 {
258 return (int)GetCurrentThreadId();
259 }
260