2012-09-18 20:57:19 +04:00
|
|
|
/**
|
|
|
|
* WinPR: Windows Portable Runtime
|
|
|
|
* Synchronization Functions
|
|
|
|
*
|
|
|
|
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2012-09-19 02:36:13 +04:00
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
2013-08-13 16:05:38 +04:00
|
|
|
#include <assert.h>
|
2013-10-23 14:13:53 +04:00
|
|
|
#include <errno.h>
|
2013-08-13 16:05:38 +04:00
|
|
|
|
2012-09-19 02:36:13 +04:00
|
|
|
#include <winpr/crt.h>
|
2012-09-18 20:57:19 +04:00
|
|
|
#include <winpr/synch.h>
|
2013-10-25 18:48:37 +04:00
|
|
|
#include <winpr/platform.h>
|
2012-09-18 20:57:19 +04:00
|
|
|
|
|
|
|
#include "synch.h"
|
2013-01-23 21:54:09 +04:00
|
|
|
#include "../thread/thread.h"
|
|
|
|
#include <winpr/thread.h>
|
2012-09-18 20:57:19 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* WaitForSingleObject
|
|
|
|
* WaitForSingleObjectEx
|
|
|
|
* WaitForMultipleObjectsEx
|
2012-09-18 23:51:33 +04:00
|
|
|
* SignalObjectAndWait
|
2012-09-18 20:57:19 +04:00
|
|
|
*/
|
|
|
|
|
2012-09-19 01:33:52 +04:00
|
|
|
#ifndef _WIN32
|
|
|
|
|
2013-10-25 18:03:57 +04:00
|
|
|
#include <time.h>
|
|
|
|
#include <sys/time.h>
|
2013-09-23 01:23:00 +04:00
|
|
|
#include <sys/wait.h>
|
|
|
|
|
2013-05-17 01:32:58 +04:00
|
|
|
#include "../handle/handle.h"
|
|
|
|
|
2013-09-12 03:00:32 +04:00
|
|
|
#include "../pipe/pipe.h"
|
|
|
|
|
2013-10-25 18:03:57 +04:00
|
|
|
#ifdef __MACH__
|
|
|
|
|
|
|
|
#include <mach/mach_time.h>
|
|
|
|
|
|
|
|
#define CLOCK_REALTIME 0
|
|
|
|
#define CLOCK_MONOTONIC 0
|
|
|
|
|
|
|
|
int clock_gettime(int clk_id, struct timespec *t)
|
|
|
|
{
|
|
|
|
UINT64 time;
|
|
|
|
double seconds;
|
|
|
|
double nseconds;
|
|
|
|
mach_timebase_info_data_t timebase;
|
|
|
|
|
|
|
|
mach_timebase_info(&timebase);
|
|
|
|
time = mach_absolute_time();
|
|
|
|
|
|
|
|
nseconds = ((double) time * (double) timebase.numer) / ((double) timebase.denom);
|
|
|
|
seconds = ((double) time * (double) timebase.numer) / ((double) timebase.denom * 1e9);
|
|
|
|
|
|
|
|
t->tv_sec = seconds;
|
|
|
|
t->tv_nsec = nseconds;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2013-10-23 14:09:09 +04:00
|
|
|
/* Drop in replacement for the linux pthread_timedjoin_np and
|
|
|
|
* pthread_mutex_timedlock functions.
|
2013-10-23 14:13:53 +04:00
|
|
|
*/
|
2013-10-23 14:09:09 +04:00
|
|
|
#if !defined(HAVE_PTHREAD_GNU_EXT)
|
|
|
|
#include <pthread.h>
|
2013-11-14 16:48:20 +04:00
|
|
|
|
2013-11-18 20:59:53 +04:00
|
|
|
static long long ts_difftime(const struct timespec *o,
|
|
|
|
const struct timespec *n)
|
2013-11-14 16:48:20 +04:00
|
|
|
{
|
2013-11-18 20:59:53 +04:00
|
|
|
long long oldValue = o->tv_sec * 1000000000LL + o->tv_nsec;
|
|
|
|
long long newValue = n->tv_sec * 1000000000LL + n->tv_nsec;
|
2013-11-14 16:48:20 +04:00
|
|
|
|
2013-11-18 20:59:53 +04:00
|
|
|
return newValue - oldValue;
|
2013-11-14 16:48:20 +04:00
|
|
|
}
|
|
|
|
|
2013-10-23 14:09:09 +04:00
|
|
|
static int pthread_timedjoin_np(pthread_t td, void **res,
|
|
|
|
struct timespec *timeout)
|
|
|
|
{
|
2013-11-14 12:34:16 +04:00
|
|
|
struct timespec timenow;
|
2013-10-23 14:09:09 +04:00
|
|
|
struct timespec sleepytime;
|
|
|
|
/* This is just to avoid a completely busy wait */
|
|
|
|
sleepytime.tv_sec = 0;
|
|
|
|
sleepytime.tv_nsec = 10000000; /* 10ms */
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if (pthread_kill(td, 0))
|
|
|
|
return pthread_join(td, res);
|
|
|
|
|
|
|
|
nanosleep(&sleepytime, NULL);
|
2013-11-14 12:34:16 +04:00
|
|
|
|
|
|
|
clock_gettime(CLOCK_MONOTONIC, &timenow);
|
2013-10-25 18:03:57 +04:00
|
|
|
|
2013-11-14 16:48:20 +04:00
|
|
|
if (ts_difftime(timeout, &timenow) >= 0)
|
2013-10-25 18:03:57 +04:00
|
|
|
{
|
|
|
|
return ETIMEDOUT;
|
2013-10-23 14:09:09 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
while (TRUE);
|
|
|
|
|
|
|
|
return ETIMEDOUT;
|
|
|
|
}
|
|
|
|
|
2013-10-25 18:48:37 +04:00
|
|
|
static int pthread_mutex_timedlock(pthread_mutex_t *mutex, const struct timespec *timeout)
|
2013-10-23 14:09:09 +04:00
|
|
|
{
|
2013-11-14 12:34:16 +04:00
|
|
|
struct timespec timenow;
|
2013-10-25 18:48:37 +04:00
|
|
|
struct timespec sleepytime;
|
|
|
|
int retcode;
|
|
|
|
|
|
|
|
/* This is just to avoid a completely busy wait */
|
|
|
|
sleepytime.tv_sec = 0;
|
|
|
|
sleepytime.tv_nsec = 10000000; /* 10ms */
|
|
|
|
|
|
|
|
while ((retcode = pthread_mutex_trylock (mutex)) == EBUSY)
|
|
|
|
{
|
2013-11-14 12:34:16 +04:00
|
|
|
clock_gettime(CLOCK_MONOTONIC, &timenow);
|
2013-10-25 18:48:37 +04:00
|
|
|
|
2013-11-14 16:48:20 +04:00
|
|
|
if (ts_difftime(timeout, &timenow) >= 0)
|
2013-10-25 18:48:37 +04:00
|
|
|
{
|
|
|
|
return ETIMEDOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
nanosleep (&sleepytime, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
return retcode;
|
2013-10-23 14:09:09 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-08-14 14:15:05 +04:00
|
|
|
static void ts_add_ms(struct timespec *ts, DWORD dwMilliseconds)
|
|
|
|
{
|
|
|
|
ts->tv_sec += dwMilliseconds / 1000L;
|
|
|
|
ts->tv_nsec += (dwMilliseconds % 1000L) * 1000000L;
|
|
|
|
|
2014-02-04 20:35:16 +04:00
|
|
|
ts->tv_sec += ts->tv_nsec / 1000000000L;
|
|
|
|
ts->tv_nsec = ts->tv_nsec % 1000000000L;
|
2013-08-14 14:15:05 +04:00
|
|
|
}
|
|
|
|
|
2012-09-18 20:57:19 +04:00
|
|
|
DWORD WaitForSingleObject(HANDLE hHandle, DWORD dwMilliseconds)
|
|
|
|
{
|
2012-09-19 01:33:52 +04:00
|
|
|
ULONG Type;
|
|
|
|
PVOID Object;
|
|
|
|
|
|
|
|
if (!winpr_Handle_GetInfo(hHandle, &Type, &Object))
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForSingleObject failed: invalid hHandle.\n");
|
2012-09-19 01:33:52 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2012-09-19 01:33:52 +04:00
|
|
|
|
2012-09-19 03:24:03 +04:00
|
|
|
if (Type == HANDLE_TYPE_THREAD)
|
|
|
|
{
|
2013-08-14 14:15:05 +04:00
|
|
|
int status = 0;
|
2012-12-13 07:03:40 +04:00
|
|
|
WINPR_THREAD* thread;
|
2013-01-23 01:19:32 +04:00
|
|
|
void* thread_status = NULL;
|
2012-12-13 07:03:40 +04:00
|
|
|
|
|
|
|
thread = (WINPR_THREAD*) Object;
|
|
|
|
|
2013-07-29 00:31:38 +04:00
|
|
|
if (thread->started)
|
|
|
|
{
|
2013-08-13 16:05:38 +04:00
|
|
|
if (dwMilliseconds != INFINITE)
|
|
|
|
{
|
|
|
|
struct timespec timeout;
|
|
|
|
|
2013-10-23 14:09:09 +04:00
|
|
|
/* pthread_timedjoin_np returns ETIMEDOUT in case the timeout is 0,
|
|
|
|
* so set it to the smallest value to get a proper return value. */
|
|
|
|
if (dwMilliseconds == 0)
|
|
|
|
dwMilliseconds ++;
|
|
|
|
|
2013-11-14 16:48:20 +04:00
|
|
|
clock_gettime(CLOCK_MONOTONIC, &timeout);
|
2013-08-14 14:15:05 +04:00
|
|
|
ts_add_ms(&timeout, dwMilliseconds);
|
2013-08-13 16:05:38 +04:00
|
|
|
|
|
|
|
status = pthread_timedjoin_np(thread->thread, &thread_status, &timeout);
|
2013-10-25 18:48:37 +04:00
|
|
|
|
2013-10-23 14:09:09 +04:00
|
|
|
if (ETIMEDOUT == status)
|
|
|
|
return WAIT_TIMEOUT;
|
2013-08-13 16:05:38 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
status = pthread_join(thread->thread, &thread_status);
|
2013-01-23 01:19:32 +04:00
|
|
|
|
2013-07-29 00:31:38 +04:00
|
|
|
if (status != 0)
|
2013-10-23 14:09:09 +04:00
|
|
|
{
|
2013-08-19 19:44:52 +04:00
|
|
|
fprintf(stderr, "WaitForSingleObject: pthread_join failure: [%d] %s\n",
|
|
|
|
status, strerror(status));
|
2013-10-23 14:09:09 +04:00
|
|
|
}
|
2013-04-02 22:51:12 +04:00
|
|
|
|
2013-07-29 00:31:38 +04:00
|
|
|
if (thread_status)
|
|
|
|
thread->dwExitCode = ((DWORD) (size_t) thread_status);
|
|
|
|
}
|
2012-09-19 03:24:03 +04:00
|
|
|
}
|
2013-09-23 01:23:00 +04:00
|
|
|
else if (Type == HANDLE_TYPE_PROCESS)
|
|
|
|
{
|
|
|
|
WINPR_PROCESS* process;
|
|
|
|
|
|
|
|
process = (WINPR_PROCESS*) Object;
|
|
|
|
|
|
|
|
if (waitpid(process->pid, &(process->status), 0) != -1)
|
|
|
|
{
|
2013-12-18 03:20:33 +04:00
|
|
|
fprintf(stderr, "WaitForSingleObject: waitpid failure [%d] %s\n", errno, strerror(errno));
|
2013-09-23 01:23:00 +04:00
|
|
|
return WAIT_FAILED;
|
|
|
|
}
|
2013-09-23 22:44:59 +04:00
|
|
|
|
|
|
|
process->dwExitCode = (DWORD) process->status;
|
2013-09-23 01:23:00 +04:00
|
|
|
}
|
2013-02-12 05:38:19 +04:00
|
|
|
else if (Type == HANDLE_TYPE_MUTEX)
|
2012-09-19 01:33:52 +04:00
|
|
|
{
|
2013-05-17 01:32:58 +04:00
|
|
|
WINPR_MUTEX* mutex;
|
|
|
|
|
|
|
|
mutex = (WINPR_MUTEX*) Object;
|
|
|
|
|
2012-09-19 03:24:03 +04:00
|
|
|
if (dwMilliseconds != INFINITE)
|
2013-08-13 16:05:38 +04:00
|
|
|
{
|
2013-10-25 18:48:37 +04:00
|
|
|
int status;
|
2013-08-13 16:05:38 +04:00
|
|
|
struct timespec timeout;
|
|
|
|
|
2013-11-14 16:48:20 +04:00
|
|
|
clock_gettime(CLOCK_MONOTONIC, &timeout);
|
2013-08-14 14:15:05 +04:00
|
|
|
ts_add_ms(&timeout, dwMilliseconds);
|
2012-09-19 03:24:03 +04:00
|
|
|
|
2013-10-23 14:16:23 +04:00
|
|
|
status = pthread_mutex_timedlock(&mutex->mutex, &timeout);
|
2013-10-25 18:48:37 +04:00
|
|
|
|
2013-10-23 14:16:23 +04:00
|
|
|
if (ETIMEDOUT == status)
|
|
|
|
return WAIT_TIMEOUT;
|
2013-08-13 16:05:38 +04:00
|
|
|
}
|
|
|
|
else
|
2014-01-25 07:44:23 +04:00
|
|
|
{
|
2013-08-13 16:05:38 +04:00
|
|
|
pthread_mutex_lock(&mutex->mutex);
|
2014-01-25 07:44:23 +04:00
|
|
|
}
|
2012-09-19 01:33:52 +04:00
|
|
|
}
|
2012-09-19 02:36:13 +04:00
|
|
|
else if (Type == HANDLE_TYPE_EVENT)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
fd_set rfds;
|
|
|
|
WINPR_EVENT* event;
|
|
|
|
struct timeval timeout;
|
|
|
|
|
|
|
|
event = (WINPR_EVENT*) Object;
|
|
|
|
|
|
|
|
FD_ZERO(&rfds);
|
|
|
|
FD_SET(event->pipe_fd[0], &rfds);
|
|
|
|
ZeroMemory(&timeout, sizeof(timeout));
|
|
|
|
|
2012-09-23 21:54:14 +04:00
|
|
|
if ((dwMilliseconds != INFINITE) && (dwMilliseconds != 0))
|
2012-09-19 02:36:13 +04:00
|
|
|
{
|
2013-12-18 03:20:33 +04:00
|
|
|
timeout.tv_sec = dwMilliseconds / 1000;
|
|
|
|
timeout.tv_usec = (dwMilliseconds % 1000) * 1000;
|
2012-09-19 02:36:13 +04:00
|
|
|
}
|
|
|
|
|
2014-01-25 07:44:23 +04:00
|
|
|
do
|
|
|
|
{
|
|
|
|
status = select(event->pipe_fd[0] + 1, &rfds, NULL, NULL,
|
|
|
|
(dwMilliseconds == INFINITE) ? NULL : &timeout);
|
|
|
|
}
|
|
|
|
while (status < 0 && (errno == EINTR));
|
2012-09-19 02:36:13 +04:00
|
|
|
|
|
|
|
if (status < 0)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForSingleObject: event select() failure [%d] %s\n", errno, strerror(errno));
|
2012-09-19 02:36:13 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2012-09-19 02:36:13 +04:00
|
|
|
|
|
|
|
if (status != 1)
|
|
|
|
return WAIT_TIMEOUT;
|
|
|
|
}
|
2012-09-19 03:24:03 +04:00
|
|
|
else if (Type == HANDLE_TYPE_SEMAPHORE)
|
|
|
|
{
|
2012-11-28 21:47:04 +04:00
|
|
|
WINPR_SEMAPHORE* semaphore;
|
|
|
|
|
|
|
|
semaphore = (WINPR_SEMAPHORE*) Object;
|
|
|
|
|
|
|
|
#ifdef WINPR_PIPE_SEMAPHORE
|
|
|
|
if (semaphore->pipe_fd[0] != -1)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
int length;
|
|
|
|
fd_set rfds;
|
|
|
|
struct timeval timeout;
|
|
|
|
|
|
|
|
FD_ZERO(&rfds);
|
|
|
|
FD_SET(semaphore->pipe_fd[0], &rfds);
|
|
|
|
ZeroMemory(&timeout, sizeof(timeout));
|
|
|
|
|
|
|
|
if ((dwMilliseconds != INFINITE) && (dwMilliseconds != 0))
|
|
|
|
{
|
2013-12-18 03:20:33 +04:00
|
|
|
timeout.tv_sec = dwMilliseconds / 1000;
|
|
|
|
timeout.tv_usec = (dwMilliseconds % 1000) * 1000;
|
2012-11-28 21:47:04 +04:00
|
|
|
}
|
|
|
|
|
2014-01-25 07:44:23 +04:00
|
|
|
do
|
|
|
|
{
|
|
|
|
status = select(semaphore->pipe_fd[0] + 1, &rfds, 0, 0,
|
|
|
|
(dwMilliseconds == INFINITE) ? NULL : &timeout);
|
|
|
|
}
|
|
|
|
while (status < 0 && (errno == EINTR));
|
2012-11-28 21:47:04 +04:00
|
|
|
|
|
|
|
if (status < 0)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForSingleObject: semaphore select() failure [%d] %s\n", errno, strerror(errno));
|
2012-11-28 21:47:04 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2012-11-28 21:47:04 +04:00
|
|
|
|
|
|
|
if (status != 1)
|
|
|
|
return WAIT_TIMEOUT;
|
|
|
|
|
|
|
|
length = read(semaphore->pipe_fd[0], &length, 1);
|
|
|
|
|
|
|
|
if (length != 1)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForSingleObject: semaphore read failure [%d] %s\n", errno, strerror(errno));
|
2013-07-29 19:57:29 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2012-11-28 21:47:04 +04:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
|
2012-09-19 03:24:03 +04:00
|
|
|
#if defined __APPLE__
|
2012-11-28 21:47:04 +04:00
|
|
|
semaphore_wait(*((winpr_sem_t*) semaphore->sem));
|
2012-09-19 03:24:03 +04:00
|
|
|
#else
|
2012-11-28 21:47:04 +04:00
|
|
|
sem_wait((winpr_sem_t*) semaphore->sem);
|
|
|
|
#endif
|
|
|
|
|
2013-07-29 19:57:29 +04:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else if (Type == HANDLE_TYPE_TIMER)
|
|
|
|
{
|
|
|
|
WINPR_TIMER* timer;
|
|
|
|
|
|
|
|
timer = (WINPR_TIMER*) Object;
|
|
|
|
|
2013-08-06 04:50:23 +04:00
|
|
|
#ifdef HAVE_EVENTFD_H
|
2013-07-29 19:57:29 +04:00
|
|
|
if (timer->fd != -1)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
fd_set rfds;
|
2013-07-29 21:07:39 +04:00
|
|
|
UINT64 expirations;
|
2013-07-29 19:57:29 +04:00
|
|
|
struct timeval timeout;
|
|
|
|
|
|
|
|
FD_ZERO(&rfds);
|
|
|
|
FD_SET(timer->fd, &rfds);
|
|
|
|
ZeroMemory(&timeout, sizeof(timeout));
|
|
|
|
|
|
|
|
if ((dwMilliseconds != INFINITE) && (dwMilliseconds != 0))
|
|
|
|
{
|
2013-12-18 03:20:33 +04:00
|
|
|
timeout.tv_sec = dwMilliseconds / 1000;
|
|
|
|
timeout.tv_usec = (dwMilliseconds % 1000) * 1000;
|
2013-07-29 19:57:29 +04:00
|
|
|
}
|
|
|
|
|
2014-01-25 07:44:23 +04:00
|
|
|
do
|
|
|
|
{
|
|
|
|
status = select(timer->fd + 1, &rfds, 0, 0,
|
|
|
|
(dwMilliseconds == INFINITE) ? NULL : &timeout);
|
|
|
|
}
|
|
|
|
while (status < 0 && (errno == EINTR));
|
2013-07-29 19:57:29 +04:00
|
|
|
|
|
|
|
if (status < 0)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForSingleObject: timer select() failure [%d] %s\n", errno, strerror(errno));
|
2013-07-29 19:57:29 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2013-07-29 19:57:29 +04:00
|
|
|
|
|
|
|
if (status != 1)
|
|
|
|
return WAIT_TIMEOUT;
|
2013-07-29 21:07:39 +04:00
|
|
|
|
2013-12-19 07:02:59 +04:00
|
|
|
status = read(timer->fd, (void*) &expirations, sizeof(UINT64));
|
2013-07-29 21:07:39 +04:00
|
|
|
|
2013-12-19 07:02:59 +04:00
|
|
|
if (status != 8)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
2013-12-19 07:02:59 +04:00
|
|
|
if (status == -1)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
if (errno == ETIMEDOUT)
|
|
|
|
return WAIT_TIMEOUT;
|
|
|
|
|
|
|
|
fprintf(stderr, "WaitForSingleObject: timer read() failure [%d] %s\n", errno, strerror(errno));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForSingleObject: timer read() failure - incorrect number of bytes read");
|
|
|
|
}
|
|
|
|
|
|
|
|
return WAIT_FAILED;
|
|
|
|
}
|
2013-07-29 19:57:29 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-12-18 03:20:33 +04:00
|
|
|
fprintf(stderr, "WaitForSingleObject: invalid timer file descriptor\n");
|
2013-07-29 19:57:29 +04:00
|
|
|
return WAIT_FAILED;
|
|
|
|
}
|
2013-12-18 03:20:33 +04:00
|
|
|
|
2013-07-29 19:57:29 +04:00
|
|
|
#else
|
2013-12-18 03:20:33 +04:00
|
|
|
fprintf(stderr, "WaitForSingleObject: file descriptors not supported\n");
|
2013-07-29 19:57:29 +04:00
|
|
|
return WAIT_FAILED;
|
2012-09-19 03:24:03 +04:00
|
|
|
#endif
|
|
|
|
}
|
2013-09-12 03:00:32 +04:00
|
|
|
else if (Type == HANDLE_TYPE_NAMED_PIPE)
|
|
|
|
{
|
2013-09-26 03:13:39 +04:00
|
|
|
int fd;
|
2013-09-12 03:00:32 +04:00
|
|
|
int status;
|
|
|
|
fd_set rfds;
|
|
|
|
struct timeval timeout;
|
|
|
|
WINPR_NAMED_PIPE* pipe = (WINPR_NAMED_PIPE*) Object;
|
|
|
|
|
2013-09-26 03:13:39 +04:00
|
|
|
fd = (pipe->ServerMode) ? pipe->serverfd : pipe->clientfd;
|
|
|
|
|
|
|
|
if (fd == -1)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForSingleObject: invalid pipe file descriptor\n");
|
2013-09-26 03:13:39 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
|
|
|
|
2013-09-12 03:00:32 +04:00
|
|
|
FD_ZERO(&rfds);
|
2013-09-26 03:13:39 +04:00
|
|
|
FD_SET(fd, &rfds);
|
2013-09-12 03:00:32 +04:00
|
|
|
ZeroMemory(&timeout, sizeof(timeout));
|
|
|
|
|
|
|
|
if ((dwMilliseconds != INFINITE) && (dwMilliseconds != 0))
|
|
|
|
{
|
2013-12-18 03:20:33 +04:00
|
|
|
timeout.tv_sec = dwMilliseconds / 1000;
|
|
|
|
timeout.tv_usec = (dwMilliseconds % 1000) * 1000;
|
2013-09-12 03:00:32 +04:00
|
|
|
}
|
|
|
|
|
2014-01-07 19:36:41 +04:00
|
|
|
do
|
|
|
|
{
|
|
|
|
status = select(fd + 1, &rfds, NULL, NULL,
|
|
|
|
(dwMilliseconds == INFINITE) ? NULL : &timeout);
|
|
|
|
}
|
2014-01-25 07:44:23 +04:00
|
|
|
while (status < 0 && (errno == EINTR));
|
2013-09-12 03:00:32 +04:00
|
|
|
|
|
|
|
if (status < 0)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForSingleObject: named pipe select() failure [%d] %s\n", errno, strerror(errno));
|
2013-09-12 03:00:32 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2013-09-12 03:00:32 +04:00
|
|
|
|
|
|
|
if (status != 1)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
2013-09-12 03:00:32 +04:00
|
|
|
return WAIT_TIMEOUT;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2013-09-12 03:00:32 +04:00
|
|
|
}
|
2013-02-12 05:38:19 +04:00
|
|
|
else
|
|
|
|
{
|
2013-03-29 02:06:34 +04:00
|
|
|
fprintf(stderr, "WaitForSingleObject: unknown handle type %lu\n", Type);
|
2013-02-12 05:38:19 +04:00
|
|
|
}
|
2012-09-19 01:33:52 +04:00
|
|
|
|
|
|
|
return WAIT_OBJECT_0;
|
|
|
|
}
|
2012-09-18 20:57:19 +04:00
|
|
|
|
2012-09-19 01:33:52 +04:00
|
|
|
DWORD WaitForSingleObjectEx(HANDLE hHandle, DWORD dwMilliseconds, BOOL bAlertable)
|
|
|
|
{
|
2013-08-14 15:19:57 +04:00
|
|
|
fprintf(stderr, "[ERROR] %s: Function not implemented.\n", __func__);
|
2013-08-13 16:05:38 +04:00
|
|
|
assert(0);
|
2012-09-18 20:57:19 +04:00
|
|
|
return WAIT_OBJECT_0;
|
|
|
|
}
|
|
|
|
|
|
|
|
DWORD WaitForMultipleObjects(DWORD nCount, const HANDLE* lpHandles, BOOL bWaitAll, DWORD dwMilliseconds)
|
|
|
|
{
|
2012-12-17 08:03:00 +04:00
|
|
|
int fd = -1;
|
2012-11-27 03:02:41 +04:00
|
|
|
int maxfd;
|
|
|
|
int index;
|
|
|
|
int status;
|
|
|
|
fd_set fds;
|
|
|
|
ULONG Type;
|
|
|
|
PVOID Object;
|
|
|
|
struct timeval timeout;
|
|
|
|
|
2013-01-23 00:58:06 +04:00
|
|
|
if (!nCount)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForMultipleObjects: invalid handles count\n");
|
2013-01-23 00:58:06 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
|
|
|
|
2012-11-27 03:02:41 +04:00
|
|
|
maxfd = 0;
|
|
|
|
FD_ZERO(&fds);
|
|
|
|
ZeroMemory(&timeout, sizeof(timeout));
|
|
|
|
|
|
|
|
if (bWaitAll)
|
2013-08-14 14:15:05 +04:00
|
|
|
{
|
2013-03-29 02:06:34 +04:00
|
|
|
fprintf(stderr, "WaitForMultipleObjects: bWaitAll not yet implemented\n");
|
2013-08-14 14:15:05 +04:00
|
|
|
assert(0);
|
|
|
|
}
|
2012-11-27 03:02:41 +04:00
|
|
|
|
|
|
|
for (index = 0; index < nCount; index++)
|
|
|
|
{
|
|
|
|
if (!winpr_Handle_GetInfo(lpHandles[index], &Type, &Object))
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForMultipleObjects: invalid handle\n");
|
2014-02-17 05:41:19 +04:00
|
|
|
|
2012-11-27 03:02:41 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2012-11-27 03:02:41 +04:00
|
|
|
|
2012-11-28 22:38:01 +04:00
|
|
|
if (Type == HANDLE_TYPE_EVENT)
|
|
|
|
{
|
|
|
|
fd = ((WINPR_EVENT*) Object)->pipe_fd[0];
|
2013-12-18 03:20:33 +04:00
|
|
|
|
|
|
|
if (fd == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForMultipleObjects: invalid event file descriptor\n");
|
|
|
|
return WAIT_FAILED;
|
|
|
|
}
|
2012-11-28 22:38:01 +04:00
|
|
|
}
|
|
|
|
else if (Type == HANDLE_TYPE_SEMAPHORE)
|
|
|
|
{
|
|
|
|
#ifdef WINPR_PIPE_SEMAPHORE
|
|
|
|
fd = ((WINPR_SEMAPHORE*) Object)->pipe_fd[0];
|
|
|
|
#else
|
2013-12-18 03:20:33 +04:00
|
|
|
fprintf(stderr, "WaitForMultipleObjects: semaphore not supported\n");
|
2012-11-27 03:02:41 +04:00
|
|
|
return WAIT_FAILED;
|
2012-11-28 22:38:01 +04:00
|
|
|
#endif
|
|
|
|
}
|
2013-07-29 19:57:29 +04:00
|
|
|
else if (Type == HANDLE_TYPE_TIMER)
|
|
|
|
{
|
|
|
|
WINPR_TIMER* timer = (WINPR_TIMER*) Object;
|
|
|
|
fd = timer->fd;
|
2013-12-18 03:20:33 +04:00
|
|
|
|
|
|
|
if (fd == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForMultipleObjects: invalid timer file descriptor\n");
|
|
|
|
return WAIT_FAILED;
|
|
|
|
}
|
2013-07-29 19:57:29 +04:00
|
|
|
}
|
2013-09-12 03:00:32 +04:00
|
|
|
else if (Type == HANDLE_TYPE_NAMED_PIPE)
|
|
|
|
{
|
|
|
|
WINPR_NAMED_PIPE* pipe = (WINPR_NAMED_PIPE*) Object;
|
2013-09-26 03:13:39 +04:00
|
|
|
fd = (pipe->ServerMode) ? pipe->serverfd : pipe->clientfd;
|
|
|
|
|
|
|
|
if (fd == -1)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForMultipleObjects: invalid timer file descriptor\n");
|
2013-09-26 03:13:39 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2013-09-12 03:00:32 +04:00
|
|
|
}
|
2012-11-28 22:38:01 +04:00
|
|
|
else
|
|
|
|
{
|
2013-12-18 03:20:33 +04:00
|
|
|
fprintf(stderr, "WaitForMultipleObjects: unknown handle type %lu\n", Type);
|
2012-11-28 22:38:01 +04:00
|
|
|
return WAIT_FAILED;
|
|
|
|
}
|
2012-11-27 03:02:41 +04:00
|
|
|
|
2013-07-29 19:57:29 +04:00
|
|
|
if (fd == -1)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForMultipleObjects: invalid file descriptor\n");
|
2013-07-29 19:57:29 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2013-07-04 22:42:40 +04:00
|
|
|
|
2012-11-27 03:02:41 +04:00
|
|
|
FD_SET(fd, &fds);
|
|
|
|
|
|
|
|
if (fd > maxfd)
|
|
|
|
maxfd = fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((dwMilliseconds != INFINITE) && (dwMilliseconds != 0))
|
|
|
|
{
|
2013-12-18 03:21:12 +04:00
|
|
|
timeout.tv_sec = dwMilliseconds / 1000;
|
|
|
|
timeout.tv_usec = (dwMilliseconds % 1000) * 1000;
|
2012-11-27 03:02:41 +04:00
|
|
|
}
|
|
|
|
|
2014-01-07 19:36:41 +04:00
|
|
|
do
|
|
|
|
{
|
|
|
|
status = select(maxfd + 1, &fds, 0, 0,
|
|
|
|
(dwMilliseconds == INFINITE) ? NULL : &timeout);
|
|
|
|
}
|
|
|
|
while (status < 0 && errno == EINTR);
|
2012-11-27 03:02:41 +04:00
|
|
|
|
|
|
|
if (status < 0)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForMultipleObjects: select() failure [%d] %s\n", errno, strerror(errno));
|
2012-11-27 03:02:41 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2012-11-27 03:02:41 +04:00
|
|
|
|
|
|
|
if (status == 0)
|
|
|
|
return WAIT_TIMEOUT;
|
|
|
|
|
|
|
|
for (index = 0; index < nCount; index++)
|
|
|
|
{
|
|
|
|
winpr_Handle_GetInfo(lpHandles[index], &Type, &Object);
|
2012-11-28 22:38:01 +04:00
|
|
|
|
|
|
|
if (Type == HANDLE_TYPE_EVENT)
|
2013-07-29 19:57:29 +04:00
|
|
|
{
|
2012-11-28 22:38:01 +04:00
|
|
|
fd = ((WINPR_EVENT*) Object)->pipe_fd[0];
|
2013-07-29 19:57:29 +04:00
|
|
|
}
|
2012-11-28 22:38:01 +04:00
|
|
|
else if (Type == HANDLE_TYPE_SEMAPHORE)
|
2013-07-29 19:57:29 +04:00
|
|
|
{
|
2012-11-28 22:38:01 +04:00
|
|
|
fd = ((WINPR_SEMAPHORE*) Object)->pipe_fd[0];
|
2013-07-29 19:57:29 +04:00
|
|
|
}
|
|
|
|
else if (Type == HANDLE_TYPE_TIMER)
|
|
|
|
{
|
|
|
|
WINPR_TIMER* timer = (WINPR_TIMER*) Object;
|
|
|
|
fd = timer->fd;
|
|
|
|
}
|
2013-09-12 03:00:32 +04:00
|
|
|
else if (Type == HANDLE_TYPE_NAMED_PIPE)
|
|
|
|
{
|
|
|
|
WINPR_NAMED_PIPE* pipe = (WINPR_NAMED_PIPE*) Object;
|
2013-09-26 03:13:39 +04:00
|
|
|
fd = (pipe->ServerMode) ? pipe->serverfd : pipe->clientfd;
|
2013-09-12 03:00:32 +04:00
|
|
|
}
|
2012-11-27 03:02:41 +04:00
|
|
|
|
|
|
|
if (FD_ISSET(fd, &fds))
|
2012-11-28 22:38:01 +04:00
|
|
|
{
|
|
|
|
if (Type == HANDLE_TYPE_SEMAPHORE)
|
|
|
|
{
|
2013-07-29 21:07:39 +04:00
|
|
|
int length;
|
|
|
|
|
|
|
|
length = read(fd, &length, 1);
|
2012-11-28 22:38:01 +04:00
|
|
|
|
|
|
|
if (length != 1)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForMultipleObjects: semaphore read() failure [%d] %s\n", errno, strerror(errno));
|
2012-11-28 22:38:01 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2012-11-28 22:38:01 +04:00
|
|
|
}
|
2013-07-29 21:07:39 +04:00
|
|
|
else if (Type == HANDLE_TYPE_TIMER)
|
|
|
|
{
|
|
|
|
int length;
|
|
|
|
UINT64 expirations;
|
|
|
|
|
|
|
|
length = read(fd, (void*) &expirations, sizeof(UINT64));
|
|
|
|
|
|
|
|
if (length != 8)
|
2013-12-18 03:20:33 +04:00
|
|
|
{
|
|
|
|
if (length == -1)
|
|
|
|
{
|
|
|
|
if (errno == ETIMEDOUT)
|
|
|
|
return WAIT_TIMEOUT;
|
|
|
|
|
|
|
|
fprintf(stderr, "WaitForMultipleObjects: timer read() failure [%d] %s\n", errno, strerror(errno));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "WaitForMultipleObjects: timer read() failure - incorrect number of bytes read");
|
|
|
|
}
|
|
|
|
|
2013-07-29 21:07:39 +04:00
|
|
|
return WAIT_FAILED;
|
2013-12-18 03:20:33 +04:00
|
|
|
}
|
2013-07-29 21:07:39 +04:00
|
|
|
}
|
2012-11-28 22:38:01 +04:00
|
|
|
|
2012-11-27 03:02:41 +04:00
|
|
|
return (WAIT_OBJECT_0 + index);
|
2012-11-28 22:38:01 +04:00
|
|
|
}
|
2012-11-27 03:02:41 +04:00
|
|
|
}
|
|
|
|
|
2013-12-18 03:20:33 +04:00
|
|
|
fprintf(stderr, "WaitForMultipleObjects: failed (unknown error)\n");
|
2012-11-27 03:02:41 +04:00
|
|
|
return WAIT_FAILED;
|
2012-09-18 20:57:19 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
DWORD WaitForMultipleObjectsEx(DWORD nCount, const HANDLE* lpHandles, BOOL bWaitAll, DWORD dwMilliseconds, BOOL bAlertable)
|
|
|
|
{
|
2013-08-14 15:19:57 +04:00
|
|
|
fprintf(stderr, "[ERROR] %s: Function not implemented.\n", __func__);
|
2013-08-13 16:05:38 +04:00
|
|
|
assert(0);
|
2012-09-18 20:57:19 +04:00
|
|
|
return 0;
|
|
|
|
}
|
2012-09-18 23:51:33 +04:00
|
|
|
|
|
|
|
DWORD SignalObjectAndWait(HANDLE hObjectToSignal, HANDLE hObjectToWaitOn, DWORD dwMilliseconds, BOOL bAlertable)
|
|
|
|
{
|
2013-08-14 15:19:57 +04:00
|
|
|
fprintf(stderr, "[ERROR] %s: Function not implemented.\n", __func__);
|
2013-08-13 16:05:38 +04:00
|
|
|
assert(0);
|
2012-09-18 23:51:33 +04:00
|
|
|
return 0;
|
|
|
|
}
|
2012-09-19 01:33:52 +04:00
|
|
|
|
|
|
|
#endif
|