2 * Win32 process and thread synchronisation
4 * Copyright 1997 Alexandre Julliard
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.
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.
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
29 #include "file.h" /* for DOSFS_UnixTimeToFileTime */
32 #include "wine/server.h"
35 /***********************************************************************
38 inline static void get_timeout( struct timeval
*when
, int timeout
)
40 gettimeofday( when
, 0 );
43 long sec
= timeout
/ 1000;
44 if ((when
->tv_usec
+= (timeout
- 1000*sec
) * 1000) >= 1000000)
46 when
->tv_usec
-= 1000000;
53 static void CALLBACK
call_completion_routine(ULONG_PTR data
)
55 async_private
* ovp
= (async_private
*)data
;
57 ovp
->completion_func(ovp
->lpOverlapped
->Internal
,
58 ovp
->lpOverlapped
->InternalHigh
,
60 ovp
->completion_func
=NULL
;
61 HeapFree(GetProcessHeap(), 0, ovp
);
64 void finish_async(async_private
*ovp
, DWORD status
)
66 ovp
->lpOverlapped
->Internal
=status
;
68 /* call ReadFileEx/WriteFileEx's overlapped completion function */
69 if(ovp
->completion_func
)
71 QueueUserAPC(call_completion_routine
,GetCurrentThread(),(ULONG_PTR
)ovp
);
74 /* remove it from the active list */
76 ovp
->prev
->next
= ovp
->next
;
78 NtCurrentTeb()->pending_list
= ovp
->next
;
81 ovp
->next
->prev
= ovp
->prev
;
87 if(ovp
->event
!=INVALID_HANDLE_VALUE
)
88 NtSetEvent(ovp
->event
,NULL
);
89 if(!ovp
->completion_func
) HeapFree(GetProcessHeap(), 0, ovp
);
92 /***********************************************************************
95 * Process a status event from the server.
97 void WINAPI
check_async_list(LPOVERLAPPED overlapped
, DWORD status
)
101 /* fprintf(stderr,"overlapped %p status %x\n",overlapped,status); */
103 for(ovp
= NtCurrentTeb()->pending_list
; ovp
; ovp
= ovp
->next
)
104 if(ovp
->lpOverlapped
== overlapped
)
110 if(status
!= STATUS_ALERTED
)
111 ovp
->lpOverlapped
->Internal
= status
;
113 if(ovp
->lpOverlapped
->Internal
==STATUS_PENDING
)
116 FILE_StartAsync(ovp
->handle
, ovp
->lpOverlapped
, ovp
->type
, 0, ovp
->lpOverlapped
->Internal
);
119 if(ovp
->lpOverlapped
->Internal
!=STATUS_PENDING
)
120 finish_async(ovp
,ovp
->lpOverlapped
->Internal
);
124 /***********************************************************************
127 * Wait for a reply on the waiting pipe of the current thread.
129 static int wait_reply( void *cookie
)
132 struct wake_up_reply reply
;
136 ret
= read( NtCurrentTeb()->wait_fd
[0], &reply
, sizeof(reply
) );
137 if (ret
== sizeof(reply
))
139 if (!reply
.cookie
) break; /* thread got killed */
140 if (reply
.cookie
== cookie
) return reply
.signaled
;
141 /* we stole another reply, wait for the real one */
142 signaled
= wait_reply( cookie
);
143 /* and now put the wrong one back in the pipe */
146 ret
= write( NtCurrentTeb()->wait_fd
[1], &reply
, sizeof(reply
) );
147 if (ret
== sizeof(reply
)) break;
148 if (ret
>= 0) server_protocol_error( "partial wakeup write %d\n", ret
);
149 if (errno
== EINTR
) continue;
150 server_protocol_perror("wakeup write");
154 if (ret
>= 0) server_protocol_error( "partial wakeup read %d\n", ret
);
155 if (errno
== EINTR
) continue;
156 server_protocol_perror("wakeup read");
158 /* the server closed the connection; time to die... */
159 SYSDEPS_AbortThread(0);
163 /***********************************************************************
166 * Call outstanding APCs.
168 static void call_apcs( BOOL alertable
)
177 SERVER_START_REQ( get_apc
)
179 req
->alertable
= alertable
;
180 wine_server_set_reply( req
, args
, sizeof(args
) );
181 if (!wine_server_call( req
))
192 return; /* no more APCs */
194 proc( args
[0], args
[1]);
200 /* convert sec/usec to NT time */
201 DOSFS_UnixTimeToFileTime( (time_t)args
[0], &ft
, (DWORD
)args
[1] * 10 );
202 proc( args
[2], ft
.dwLowDateTime
, ft
.dwHighDateTime
);
205 server_protocol_error( "get_apc_request: bad type %d\n", type
);
211 /***********************************************************************
214 VOID WINAPI
Sleep( DWORD timeout
)
216 WaitForMultipleObjectsEx( 0, NULL
, FALSE
, timeout
, FALSE
);
219 /******************************************************************************
220 * SleepEx (KERNEL32.@)
222 DWORD WINAPI
SleepEx( DWORD timeout
, BOOL alertable
)
224 DWORD ret
= WaitForMultipleObjectsEx( 0, NULL
, FALSE
, timeout
, alertable
);
225 if (ret
!= WAIT_IO_COMPLETION
) ret
= 0;
230 /***********************************************************************
231 * WaitForSingleObject (KERNEL32.@)
233 DWORD WINAPI
WaitForSingleObject( HANDLE handle
, DWORD timeout
)
235 return WaitForMultipleObjectsEx( 1, &handle
, FALSE
, timeout
, FALSE
);
239 /***********************************************************************
240 * WaitForSingleObjectEx (KERNEL32.@)
242 DWORD WINAPI
WaitForSingleObjectEx( HANDLE handle
, DWORD timeout
,
245 return WaitForMultipleObjectsEx( 1, &handle
, FALSE
, timeout
, alertable
);
249 /***********************************************************************
250 * WaitForMultipleObjects (KERNEL32.@)
252 DWORD WINAPI
WaitForMultipleObjects( DWORD count
, const HANDLE
*handles
,
253 BOOL wait_all
, DWORD timeout
)
255 return WaitForMultipleObjectsEx( count
, handles
, wait_all
, timeout
, FALSE
);
259 /***********************************************************************
260 * WaitForMultipleObjectsEx (KERNEL32.@)
262 DWORD WINAPI
WaitForMultipleObjectsEx( DWORD count
, const HANDLE
*handles
,
263 BOOL wait_all
, DWORD timeout
,
269 if (count
> MAXIMUM_WAIT_OBJECTS
)
271 SetLastError( ERROR_INVALID_PARAMETER
);
275 if (timeout
== INFINITE
) tv
.tv_sec
= tv
.tv_usec
= 0;
276 else get_timeout( &tv
, timeout
);
280 SERVER_START_REQ( select
)
282 req
->flags
= SELECT_INTERRUPTIBLE
;
283 req
->cookie
= &cookie
;
284 req
->sec
= tv
.tv_sec
;
285 req
->usec
= tv
.tv_usec
;
286 wine_server_add_data( req
, handles
, count
* sizeof(HANDLE
) );
288 if (wait_all
) req
->flags
|= SELECT_ALL
;
289 if (alertable
) req
->flags
|= SELECT_ALERTABLE
;
290 if (timeout
!= INFINITE
) req
->flags
|= SELECT_TIMEOUT
;
292 ret
= wine_server_call( req
);
295 if (ret
== STATUS_PENDING
) ret
= wait_reply( &cookie
);
296 if (ret
!= STATUS_USER_APC
) break;
297 call_apcs( alertable
);
298 if (alertable
) break;
300 if (HIWORD(ret
)) /* is it an error code? */
302 SetLastError( RtlNtStatusToDosError(ret
) );
309 /***********************************************************************
310 * WaitForSingleObject (KERNEL.460)
312 DWORD WINAPI
WaitForSingleObject16( HANDLE handle
, DWORD timeout
)
314 DWORD retval
, mutex_count
;
316 ReleaseThunkLock( &mutex_count
);
317 retval
= WaitForSingleObject( handle
, timeout
);
318 RestoreThunkLock( mutex_count
);
322 /***********************************************************************
323 * WaitForMultipleObjects (KERNEL.461)
325 DWORD WINAPI
WaitForMultipleObjects16( DWORD count
, const HANDLE
*handles
,
326 BOOL wait_all
, DWORD timeout
)
328 DWORD retval
, mutex_count
;
330 ReleaseThunkLock( &mutex_count
);
331 retval
= WaitForMultipleObjectsEx( count
, handles
, wait_all
, timeout
, FALSE
);
332 RestoreThunkLock( mutex_count
);
336 /***********************************************************************
337 * WaitForMultipleObjectsEx (KERNEL.495)
339 DWORD WINAPI
WaitForMultipleObjectsEx16( DWORD count
, const HANDLE
*handles
,
340 BOOL wait_all
, DWORD timeout
, BOOL alertable
)
342 DWORD retval
, mutex_count
;
344 ReleaseThunkLock( &mutex_count
);
345 retval
= WaitForMultipleObjectsEx( count
, handles
, wait_all
, timeout
, alertable
);
346 RestoreThunkLock( mutex_count
);