Blame winpr/libwinpr/synch/wait.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * WinPR: Windows Portable Runtime
Packit 1fb8d4
 * Synchronization Functions
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit 1fb8d4
 * Copyright 2014 Hardening <contact@hardening-consulting.com>
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_CONFIG_H
Packit 1fb8d4
#include "config.h"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_UNISTD_H
Packit 1fb8d4
#include <unistd.h>
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_POLL_H
Packit 1fb8d4
#include <poll.h>
Packit 1fb8d4
#else
Packit 1fb8d4
#ifndef _WIN32
Packit 1fb8d4
#include <sys/select.h>
Packit 1fb8d4
#endif
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <assert.h>
Packit 1fb8d4
#include <errno.h>
Packit 1fb8d4
Packit 1fb8d4
#include <winpr/crt.h>
Packit 1fb8d4
#include <winpr/synch.h>
Packit 1fb8d4
#include <winpr/platform.h>
Packit 1fb8d4
Packit 1fb8d4
#include "synch.h"
Packit 1fb8d4
#include "../thread/thread.h"
Packit 1fb8d4
#include <winpr/thread.h>
Packit 1fb8d4
#include <winpr/debug.h>
Packit 1fb8d4
Packit 1fb8d4
#include "../log.h"
Packit 1fb8d4
#define TAG WINPR_TAG("sync.wait")
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * WaitForSingleObject
Packit 1fb8d4
 * WaitForSingleObjectEx
Packit 1fb8d4
 * WaitForMultipleObjectsEx
Packit 1fb8d4
 * SignalObjectAndWait
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifndef _WIN32
Packit 1fb8d4
Packit 1fb8d4
#include <stdlib.h>
Packit 1fb8d4
#include <time.h>
Packit 1fb8d4
#include <sys/time.h>
Packit 1fb8d4
#include <sys/wait.h>
Packit 1fb8d4
Packit 1fb8d4
#include "../handle/handle.h"
Packit 1fb8d4
Packit 1fb8d4
#include "../pipe/pipe.h"
Packit 1fb8d4
Packit 1fb8d4
/* clock_gettime is not implemented on OSX prior to 10.12 */
Packit 1fb8d4
#ifdef __MACH__
Packit 1fb8d4
Packit 1fb8d4
#include <mach/mach_time.h>
Packit 1fb8d4
Packit 1fb8d4
#ifndef CLOCK_REALTIME
Packit 1fb8d4
#define CLOCK_REALTIME 0
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#ifndef CLOCK_MONOTONIC
Packit 1fb8d4
#define CLOCK_MONOTONIC 0
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
/* clock_gettime is not implemented on OSX prior to 10.12 */
Packit 1fb8d4
int _mach_clock_gettime(int clk_id, struct timespec* t);
Packit 1fb8d4
Packit Service 5a9772
int _mach_clock_gettime(int clk_id, struct timespec* t)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT64 time;
Packit 1fb8d4
	double seconds;
Packit 1fb8d4
	double nseconds;
Packit 1fb8d4
	mach_timebase_info_data_t timebase;
Packit 1fb8d4
	mach_timebase_info(&timebase);
Packit 1fb8d4
	time = mach_absolute_time();
Packit Service 5a9772
	nseconds = ((double)time * (double)timebase.numer) / ((double)timebase.denom);
Packit Service 5a9772
	seconds = ((double)time * (double)timebase.numer) / ((double)timebase.denom * 1e9);
Packit 1fb8d4
	t->tv_sec = seconds;
Packit 1fb8d4
	t->tv_nsec = nseconds;
Packit 1fb8d4
	return 0;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/* if clock_gettime is declared, then __CLOCK_AVAILABILITY will be defined */
Packit 1fb8d4
#ifdef __CLOCK_AVAILABILITY
Packit 1fb8d4
/* If we compiled with Mac OSX 10.12 or later, then clock_gettime will be declared
Packit 1fb8d4
 * * but it may be NULL at runtime. So we need to check before using it. */
Packit 1fb8d4
int _mach_safe_clock_gettime(int clk_id, struct timespec* t);
Packit 1fb8d4
Packit Service 5a9772
int _mach_safe_clock_gettime(int clk_id, struct timespec* t)
Packit 1fb8d4
{
Packit 1fb8d4
	if (clock_gettime)
Packit 1fb8d4
	{
Packit 1fb8d4
		return clock_gettime(clk_id, t);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return _mach_clock_gettime(clk_id, t);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
#define clock_gettime _mach_safe_clock_gettime
Packit 1fb8d4
#else
Packit 1fb8d4
#define clock_gettime _mach_clock_gettime
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit Service 5a9772
static long long ts_difftime(const struct timespec* o, const struct timespec* n)
Packit 1fb8d4
{
Packit 1fb8d4
	long long oldValue = o->tv_sec * 1000000000LL + o->tv_nsec;
Packit 1fb8d4
	long long newValue = n->tv_sec * 1000000000LL + n->tv_nsec;
Packit 1fb8d4
	return newValue - oldValue;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/* Drop in replacement for pthread_mutex_timedlock
Packit 1fb8d4
 */
Packit 1fb8d4
#if !defined(HAVE_PTHREAD_MUTEX_TIMEDLOCK)
Packit 1fb8d4
#include <pthread.h>
Packit 1fb8d4
Packit 1fb8d4
static int pthread_mutex_timedlock(pthread_mutex_t* mutex, const struct timespec* timeout)
Packit 1fb8d4
{
Packit 1fb8d4
	struct timespec timenow;
Packit 1fb8d4
	struct timespec sleepytime;
Packit 1fb8d4
	unsigned long long diff;
Packit 1fb8d4
	int retcode;
Packit 1fb8d4
	/* This is just to avoid a completely busy wait */
Packit 1fb8d4
	clock_gettime(CLOCK_MONOTONIC, &timenow);
Packit 1fb8d4
	diff = ts_difftime(&timenow, timeout);
Packit 1fb8d4
	sleepytime.tv_sec = diff / 1000000000LL;
Packit 1fb8d4
	sleepytime.tv_nsec = diff % 1000000000LL;
Packit 1fb8d4
Packit 1fb8d4
	while ((retcode = pthread_mutex_trylock(mutex)) == EBUSY)
Packit 1fb8d4
	{
Packit 1fb8d4
		clock_gettime(CLOCK_MONOTONIC, &timenow);
Packit 1fb8d4
Packit 1fb8d4
		if (ts_difftime(timeout, &timenow) >= 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			return ETIMEDOUT;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		nanosleep(&sleepytime, NULL);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return retcode;
Packit 1fb8d4
}
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_POLL_H
Packit 1fb8d4
static DWORD handle_mode_to_pollevent(ULONG mode)
Packit 1fb8d4
{
Packit 1fb8d4
	DWORD event = 0;
Packit 1fb8d4
Packit 1fb8d4
	if (mode & WINPR_FD_READ)
Packit 1fb8d4
		event |= POLLIN;
Packit 1fb8d4
Packit 1fb8d4
	if (mode & WINPR_FD_WRITE)
Packit 1fb8d4
		event |= POLLOUT;
Packit 1fb8d4
Packit 1fb8d4
	return event;
Packit 1fb8d4
}
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
static void ts_add_ms(struct timespec* ts, DWORD dwMilliseconds)
Packit 1fb8d4
{
Packit 1fb8d4
	ts->tv_sec += dwMilliseconds / 1000L;
Packit 1fb8d4
	ts->tv_nsec += (dwMilliseconds % 1000L) * 1000000L;
Packit 1fb8d4
	ts->tv_sec += ts->tv_nsec / 1000000000L;
Packit 1fb8d4
	ts->tv_nsec = ts->tv_nsec % 1000000000L;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static int waitOnFd(int fd, ULONG mode, DWORD dwMilliseconds)
Packit 1fb8d4
{
Packit 1fb8d4
	int status;
Packit 1fb8d4
#ifdef HAVE_POLL_H
Packit 1fb8d4
	struct pollfd pollfds;
Packit 1fb8d4
	pollfds.fd = fd;
Packit 1fb8d4
	pollfds.events = handle_mode_to_pollevent(mode);
Packit 1fb8d4
	pollfds.revents = 0;
Packit 1fb8d4
Packit 1fb8d4
	do
Packit 1fb8d4
	{
Packit 1fb8d4
		status = poll(&pollfds, 1, dwMilliseconds);
Packit Service 5a9772
	} while ((status < 0) && (errno == EINTR));
Packit 1fb8d4
Packit 1fb8d4
#else
Packit 1fb8d4
	struct timeval timeout;
Packit 1fb8d4
	fd_set rfds, wfds;
Packit 1fb8d4
	fd_set* prfds = NULL;
Packit 1fb8d4
	fd_set* pwfds = NULL;
Packit 1fb8d4
	fd_set* pefds = NULL;
Packit 1fb8d4
	FD_ZERO(&rfds);
Packit 1fb8d4
	FD_SET(fd, &rfds);
Packit 1fb8d4
	FD_ZERO(&wfds);
Packit 1fb8d4
	FD_SET(fd, &wfds);
Packit 1fb8d4
	ZeroMemory(&timeout, sizeof(timeout));
Packit 1fb8d4
Packit 1fb8d4
	if (mode & WINPR_FD_READ)
Packit 1fb8d4
		prfds = &rfd;;
Packit 1fb8d4
Packit 1fb8d4
	if (mode & WINPR_FD_WRITE)
Packit 1fb8d4
		pwfds = &wfd;;
Packit 1fb8d4
Packit 1fb8d4
	if ((dwMilliseconds != INFINITE) && (dwMilliseconds != 0))
Packit 1fb8d4
	{
Packit 1fb8d4
		timeout.tv_sec = dwMilliseconds / 1000;
Packit 1fb8d4
		timeout.tv_usec = (dwMilliseconds % 1000) * 1000;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	do
Packit 1fb8d4
	{
Packit Service 5a9772
		status =
Packit Service 5a9772
		    select(fd + 1, prfds, pwfds, pefds, (dwMilliseconds == INFINITE) ? NULL : &timeout);
Packit Service 5a9772
	} while (status < 0 && (errno == EINTR));
Packit 1fb8d4
Packit 1fb8d4
#endif
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
DWORD WaitForSingleObject(HANDLE hHandle, DWORD dwMilliseconds)
Packit 1fb8d4
{
Packit 1fb8d4
	ULONG Type;
Packit 1fb8d4
	WINPR_HANDLE* Object;
Packit 1fb8d4
Packit 1fb8d4
	if (!winpr_Handle_GetInfo(hHandle, &Type, &Object))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "invalid hHandle.");
Packit 1fb8d4
		SetLastError(ERROR_INVALID_HANDLE);
Packit 1fb8d4
		return WAIT_FAILED;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (Type == HANDLE_TYPE_PROCESS)
Packit 1fb8d4
	{
Packit 1fb8d4
		WINPR_PROCESS* process;
Packit Service 5a9772
		process = (WINPR_PROCESS*)Object;
Packit 1fb8d4
Packit 1fb8d4
		if (process->pid != waitpid(process->pid, &(process->status), 0))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_ERR(TAG, "waitpid failure [%d] %s", errno, strerror(errno));
Packit 1fb8d4
			SetLastError(ERROR_INTERNAL_ERROR);
Packit 1fb8d4
			return WAIT_FAILED;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit Service 5a9772
		process->dwExitCode = (DWORD)process->status;
Packit 1fb8d4
		return WAIT_OBJECT_0;
Packit 1fb8d4
	}
Packit 1fb8d4
	else if (Type == HANDLE_TYPE_MUTEX)
Packit 1fb8d4
	{
Packit 1fb8d4
		WINPR_MUTEX* mutex;
Packit Service 5a9772
		mutex = (WINPR_MUTEX*)Object;
Packit 1fb8d4
Packit 1fb8d4
		if (dwMilliseconds != INFINITE)
Packit 1fb8d4
		{
Packit 1fb8d4
			int status;
Packit 1fb8d4
			struct timespec timeout;
Packit 1fb8d4
			clock_gettime(CLOCK_MONOTONIC, &timeout);
Packit 1fb8d4
			ts_add_ms(&timeout, dwMilliseconds);
Packit 1fb8d4
			status = pthread_mutex_timedlock(&mutex->mutex, &timeout);
Packit 1fb8d4
Packit 1fb8d4
			if (ETIMEDOUT == status)
Packit 1fb8d4
				return WAIT_TIMEOUT;
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			pthread_mutex_lock(&mutex->mutex);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		return WAIT_OBJECT_0;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		int status;
Packit 1fb8d4
		int fd = winpr_Handle_getFd(Object);
Packit 1fb8d4
Packit 1fb8d4
		if (fd < 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_ERR(TAG, "winpr_Handle_getFd did not return a fd!");
Packit 1fb8d4
			SetLastError(ERROR_INVALID_HANDLE);
Packit 1fb8d4
			return WAIT_FAILED;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		status = waitOnFd(fd, Object->Mode, dwMilliseconds);
Packit 1fb8d4
Packit 1fb8d4
		if (status < 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_ERR(TAG, "waitOnFd() failure [%d] %s", errno, strerror(errno));
Packit 1fb8d4
			SetLastError(ERROR_INTERNAL_ERROR);
Packit 1fb8d4
			return WAIT_FAILED;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (status != 1)
Packit 1fb8d4
			return WAIT_TIMEOUT;
Packit 1fb8d4
Packit 1fb8d4
		return winpr_Handle_cleanup(Object);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	SetLastError(ERROR_INTERNAL_ERROR);
Packit 1fb8d4
	return WAIT_FAILED;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
DWORD WaitForSingleObjectEx(HANDLE hHandle, DWORD dwMilliseconds, BOOL bAlertable)
Packit 1fb8d4
{
Packit Service 5a9772
	if (bAlertable)
Packit Service 5a9772
	{
Packit Service 5a9772
		/* TODO: Implement */
Packit Service 5a9772
		WLog_ERR(TAG, "%s: Not implemented: bAlertable", __FUNCTION__);
Packit Service 5a9772
		return WAIT_FAILED;
Packit Service 5a9772
	}
Packit Service 5a9772
	return WaitForSingleObject(hHandle, dwMilliseconds);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
DWORD WaitForMultipleObjects(DWORD nCount, const HANDLE* lpHandles, BOOL bWaitAll,
Packit 1fb8d4
                             DWORD dwMilliseconds)
Packit 1fb8d4
{
Packit 1fb8d4
	struct timespec starttime;
Packit 1fb8d4
	struct timespec timenow;
Packit 1fb8d4
	unsigned long long diff;
Packit 1fb8d4
	DWORD signalled;
Packit 1fb8d4
	DWORD polled;
Packit 1fb8d4
	DWORD* poll_map = NULL;
Packit 1fb8d4
	BOOL* signalled_idx = NULL;
Packit 1fb8d4
	int fd = -1;
Packit 1fb8d4
	DWORD index;
Packit 1fb8d4
	int status;
Packit 1fb8d4
	ULONG Type;
Packit 1fb8d4
	BOOL signal_handled = FALSE;
Packit 1fb8d4
	WINPR_HANDLE* Object;
Packit 1fb8d4
#ifdef HAVE_POLL_H
Packit 1fb8d4
	struct pollfd* pollfds;
Packit 1fb8d4
#else
Packit 1fb8d4
	int maxfd;
Packit 1fb8d4
	fd_set rfds;
Packit 1fb8d4
	fd_set wfds;
Packit 1fb8d4
	struct timeval timeout;
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
	if (!nCount || (nCount > MAXIMUM_WAIT_OBJECTS))
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_ERR(TAG, "invalid handles count(%" PRIu32 ")", nCount);
Packit 1fb8d4
		return WAIT_FAILED;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (bWaitAll)
Packit 1fb8d4
	{
Packit 1fb8d4
		signalled_idx = alloca(nCount * sizeof(BOOL));
Packit 1fb8d4
		memset(signalled_idx, FALSE, nCount * sizeof(BOOL));
Packit 1fb8d4
		poll_map = alloca(nCount * sizeof(DWORD));
Packit 1fb8d4
		memset(poll_map, 0, nCount * sizeof(DWORD));
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_POLL_H
Packit 1fb8d4
	pollfds = alloca(nCount * sizeof(struct pollfd));
Packit 1fb8d4
#endif
Packit 1fb8d4
	signalled = 0;
Packit 1fb8d4
Packit 1fb8d4
	do
Packit 1fb8d4
	{
Packit 1fb8d4
#ifndef HAVE_POLL_H
Packit 1fb8d4
		fd_set* prfds = NULL;
Packit 1fb8d4
		fd_set* pwfds = NULL;
Packit 1fb8d4
		maxfd = 0;
Packit 1fb8d4
		FD_ZERO(&rfds);
Packit 1fb8d4
		FD_ZERO(&wfds);
Packit 1fb8d4
		ZeroMemory(&timeout, sizeof(timeout));
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
		if (bWaitAll && (dwMilliseconds != INFINITE))
Packit 1fb8d4
			clock_gettime(CLOCK_MONOTONIC, &starttime);
Packit 1fb8d4
Packit 1fb8d4
		polled = 0;
Packit 1fb8d4
Packit 1fb8d4
		for (index = 0; index < nCount; index++)
Packit 1fb8d4
		{
Packit 1fb8d4
			if (bWaitAll)
Packit 1fb8d4
			{
Packit 1fb8d4
				if (signalled_idx[index])
Packit 1fb8d4
					continue;
Packit 1fb8d4
Packit 1fb8d4
				poll_map[polled] = index;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (!winpr_Handle_GetInfo(lpHandles[index], &Type, &Object))
Packit 1fb8d4
			{
Packit 1fb8d4
				WLog_ERR(TAG, "invalid event file descriptor");
Packit 1fb8d4
				SetLastError(ERROR_INVALID_HANDLE);
Packit 1fb8d4
				return WAIT_FAILED;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			fd = winpr_Handle_getFd(Object);
Packit 1fb8d4
Packit 1fb8d4
			if (fd == -1)
Packit 1fb8d4
			{
Packit 1fb8d4
				WLog_ERR(TAG, "invalid file descriptor");
Packit 1fb8d4
				SetLastError(ERROR_INVALID_HANDLE);
Packit 1fb8d4
				return WAIT_FAILED;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_POLL_H
Packit 1fb8d4
			pollfds[polled].fd = fd;
Packit 1fb8d4
			pollfds[polled].events = handle_mode_to_pollevent(Object->Mode);
Packit 1fb8d4
			pollfds[polled].revents = 0;
Packit 1fb8d4
#else
Packit 1fb8d4
Packit 1fb8d4
			if (Object->Mode & WINPR_FD_READ)
Packit Service 5a9772
			{
Packit Service 5a9772
				FD_SET(fd, &rfds);
Packit 1fb8d4
				prfds = &rfd;;
Packit Service 5a9772
			}
Packit 1fb8d4
Packit 1fb8d4
			if (Object->Mode & WINPR_FD_WRITE)
Packit Service 5a9772
			{
Packit Service 5a9772
				FD_SET(fd, &wfds);
Packit 1fb8d4
				pwfds = &wfd;;
Packit Service 5a9772
			}
Packit 1fb8d4
Packit 1fb8d4
			if (fd > maxfd)
Packit 1fb8d4
				maxfd = fd;
Packit 1fb8d4
Packit 1fb8d4
#endif
Packit 1fb8d4
			polled++;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_POLL_H
Packit 1fb8d4
Packit 1fb8d4
		do
Packit 1fb8d4
		{
Packit 1fb8d4
			status = poll(pollfds, polled, dwMilliseconds);
Packit Service 5a9772
		} while (status < 0 && errno == EINTR);
Packit 1fb8d4
Packit 1fb8d4
#else
Packit 1fb8d4
Packit 1fb8d4
		if ((dwMilliseconds != INFINITE) && (dwMilliseconds != 0))
Packit 1fb8d4
		{
Packit 1fb8d4
			timeout.tv_sec = dwMilliseconds / 1000;
Packit 1fb8d4
			timeout.tv_usec = (dwMilliseconds % 1000) * 1000;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		do
Packit 1fb8d4
		{
Packit Service 5a9772
			status =
Packit Service 5a9772
			    select(maxfd + 1, prfds, pwfds, 0, (dwMilliseconds == INFINITE) ? NULL : &timeout);
Packit Service 5a9772
		} while (status < 0 && errno == EINTR);
Packit 1fb8d4
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
		if (status < 0)
Packit 1fb8d4
		{
Packit 1fb8d4
#ifdef HAVE_POLL_H
Packit Service 5a9772
			WLog_ERR(TAG, "poll() handle %d (%" PRIu32 ") failure [%d] %s", index, nCount, errno,
Packit 1fb8d4
			         strerror(errno));
Packit 1fb8d4
#else
Packit Service 5a9772
			WLog_ERR(TAG, "select() handle %d (%" PRIu32 ") failure [%d] %s", index, nCount, errno,
Packit 1fb8d4
			         strerror(errno));
Packit 1fb8d4
#endif
Packit 1fb8d4
			winpr_log_backtrace(TAG, WLOG_ERROR, 20);
Packit 1fb8d4
			SetLastError(ERROR_INTERNAL_ERROR);
Packit 1fb8d4
			return WAIT_FAILED;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (status == 0)
Packit 1fb8d4
			return WAIT_TIMEOUT;
Packit 1fb8d4
Packit 1fb8d4
		if (bWaitAll && (dwMilliseconds != INFINITE))
Packit 1fb8d4
		{
Packit 1fb8d4
			clock_gettime(CLOCK_MONOTONIC, &timenow);
Packit 1fb8d4
			diff = ts_difftime(&timenow, &starttime);
Packit 1fb8d4
Packit 1fb8d4
			if (diff / 1000 > dwMilliseconds)
Packit 1fb8d4
				return WAIT_TIMEOUT;
Packit 1fb8d4
			else
Packit 1fb8d4
				dwMilliseconds -= (diff / 1000);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		signal_handled = FALSE;
Packit 1fb8d4
Packit 1fb8d4
		for (index = 0; index < polled; index++)
Packit 1fb8d4
		{
Packit 1fb8d4
			DWORD idx;
Packit 1fb8d4
			BOOL signal_set = FALSE;
Packit 1fb8d4
Packit 1fb8d4
			if (bWaitAll)
Packit 1fb8d4
				idx = poll_map[index];
Packit 1fb8d4
			else
Packit 1fb8d4
				idx = index;
Packit 1fb8d4
Packit 1fb8d4
			if (!winpr_Handle_GetInfo(lpHandles[idx], &Type, &Object))
Packit 1fb8d4
			{
Packit 1fb8d4
				WLog_ERR(TAG, "invalid hHandle.");
Packit 1fb8d4
				SetLastError(ERROR_INVALID_HANDLE);
Packit 1fb8d4
				return WAIT_FAILED;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			fd = winpr_Handle_getFd(lpHandles[idx]);
Packit 1fb8d4
Packit 1fb8d4
			if (fd == -1)
Packit 1fb8d4
			{
Packit 1fb8d4
				WLog_ERR(TAG, "invalid file descriptor");
Packit 1fb8d4
				SetLastError(ERROR_INVALID_HANDLE);
Packit 1fb8d4
				return WAIT_FAILED;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_POLL_H
Packit 1fb8d4
			signal_set = pollfds[index].revents & pollfds[index].events;
Packit 1fb8d4
#else
Packit 1fb8d4
Packit 1fb8d4
			if (Object->Mode & WINPR_FD_READ)
Packit 1fb8d4
				signal_set = FD_ISSET(fd, &rfds) ? 1 : 0;
Packit 1fb8d4
Packit 1fb8d4
			if (Object->Mode & WINPR_FD_WRITE)
Packit 1fb8d4
				signal_set |= FD_ISSET(fd, &wfds) ? 1 : 0;
Packit 1fb8d4
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
			if (signal_set)
Packit 1fb8d4
			{
Packit 1fb8d4
				DWORD rc = winpr_Handle_cleanup(lpHandles[idx]);
Packit 1fb8d4
Packit 1fb8d4
				if (rc != WAIT_OBJECT_0)
Packit 1fb8d4
					return rc;
Packit 1fb8d4
Packit 1fb8d4
				if (bWaitAll)
Packit 1fb8d4
				{
Packit 1fb8d4
					signalled_idx[idx] = TRUE;
Packit 1fb8d4
Packit 1fb8d4
					/* Continue checks from last position. */
Packit 1fb8d4
					for (; signalled < nCount; signalled++)
Packit 1fb8d4
					{
Packit 1fb8d4
						if (!signalled_idx[signalled])
Packit 1fb8d4
							break;
Packit 1fb8d4
					}
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				if (!bWaitAll)
Packit 1fb8d4
					return (WAIT_OBJECT_0 + index);
Packit 1fb8d4
Packit 1fb8d4
				if (signalled >= nCount)
Packit 1fb8d4
					return (WAIT_OBJECT_0);
Packit 1fb8d4
Packit 1fb8d4
				signal_handled = TRUE;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit Service 5a9772
	} while (bWaitAll || !signal_handled);
Packit 1fb8d4
Packit 1fb8d4
	WLog_ERR(TAG, "failed (unknown error)");
Packit 1fb8d4
	SetLastError(ERROR_INTERNAL_ERROR);
Packit 1fb8d4
	return WAIT_FAILED;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
DWORD WaitForMultipleObjectsEx(DWORD nCount, const HANDLE* lpHandles, BOOL bWaitAll,
Packit 1fb8d4
                               DWORD dwMilliseconds, BOOL bAlertable)
Packit 1fb8d4
{
Packit Service 5a9772
	if (bAlertable)
Packit Service 5a9772
	{
Packit Service 5a9772
		/* TODO: Implement */
Packit Service 5a9772
		WLog_ERR(TAG, "%s: Not implemented: bAlertable", __FUNCTION__);
Packit Service 5a9772
		return WAIT_FAILED;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit 1fb8d4
	return WaitForMultipleObjects(nCount, lpHandles, bWaitAll, dwMilliseconds);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
DWORD SignalObjectAndWait(HANDLE hObjectToSignal, HANDLE hObjectToWaitOn, DWORD dwMilliseconds,
Packit 1fb8d4
                          BOOL bAlertable)
Packit 1fb8d4
{
Packit 1fb8d4
	WLog_ERR(TAG, "%s: Not implemented.", __FUNCTION__);
Packit 1fb8d4
	SetLastError(ERROR_NOT_SUPPORTED);
Packit 1fb8d4
	return WAIT_FAILED;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
#endif