cosmopolitan/libc/calls/sigtimedwait.c
2024-09-15 01:18:27 -07:00

88 lines
4.2 KiB
C

/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
│ vi: set et ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi │
╞══════════════════════════════════════════════════════════════════════════════╡
│ Copyright 2022 Justine Alexandra Roberts Tunney │
│ │
│ Permission to use, copy, modify, and/or distribute this software for │
│ any purpose with or without fee is hereby granted, provided that the │
│ above copyright notice and this permission notice appear in all copies. │
│ │
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
│ PERFORMANCE OF THIS SOFTWARE. │
╚─────────────────────────────────────────────────────────────────────────────*/
#include "libc/calls/sigtimedwait.h"
#include "libc/calls/cp.internal.h"
#include "libc/calls/sigtimedwait.internal.h"
#include "libc/calls/struct/siginfo.internal.h"
#include "libc/calls/struct/sigset.internal.h"
#include "libc/calls/struct/timespec.internal.h"
#include "libc/dce.h"
#include "libc/intrin/strace.h"
#include "libc/str/str.h"
#include "libc/sysv/errfuns.h"
int sys_sigtimedwait_nt(const sigset_t *, siginfo_t *, const struct timespec *);
/**
* Waits for signal synchronously, w/ timeout.
*
* This function does not change the thread signal mask. Signals that
* aren't masked, which aren't in `set`, will be handled normally, in
* which case this function will raise `EINTR`.
*
* This function silently ignores attempts to synchronously wait for
* SIGTHR which is used internally by the POSIX threads implementation.
*
* @param set is signals for which we'll be waiting
* @param opt_info if not null shall receive info about signal
* @param opt_timeout is relative deadline and null means wait forever
* @return signal number on success, or -1 w/ errno
* @raise EINTR if an asynchronous signal was delivered instead
* @raise ECANCELED if thread was cancelled in masked mode
* @raise EINVAL if nanoseconds parameter was out of range
* @raise EAGAIN if timeout elapsed
* @raise ENOSYS on XNU, OpenBSD, and Metal
* @raise EFAULT if invalid memory was supplied
* @cancelationpoint
* @norestart
*/
int sigtimedwait(const sigset_t *set, siginfo_t *opt_info,
const struct timespec *opt_timeout) {
int rc;
char strsig[21];
struct timespec ts;
union siginfo_meta si = {0};
BEGIN_CANCELATION_POINT;
// validate timeout
if (opt_timeout && opt_timeout->tv_nsec >= 1000000000ull) {
rc = einval();
} else if (IsLinux() || IsFreebsd() || IsNetbsd()) {
if (opt_timeout) {
// 1. Linux needs its size parameter
// 2. NetBSD modifies timeout argument
ts = *opt_timeout;
rc = sys_sigtimedwait(set, &si, &ts, 8);
} else {
rc = sys_sigtimedwait(set, &si, 0, 8);
}
if (rc != -1 && opt_info)
__siginfo2cosmo(opt_info, &si);
} else if (IsWindows()) {
rc = sys_sigtimedwait_nt(set, opt_info, opt_timeout);
} else {
rc = enosys();
}
END_CANCELATION_POINT;
STRACE("sigtimedwait(%s, [%s], %s) → %s% m", DescribeSigset(0, set),
DescribeSiginfo(rc, opt_info), DescribeTimespec(0, opt_timeout),
strsignal_r(rc, strsig));
return rc;
}