Blame winpr/libwinpr/synch/wait.c

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