-
Notifications
You must be signed in to change notification settings - Fork 3.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add improved support for signals (raise, kill, sigaction, etc) #14883
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
This file was deleted.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
/* | ||
* Copyright 2021 The Emscripten Authors. All rights reserved. | ||
* Emscripten is available under two separate licenses, the MIT license and the | ||
* University of Illinois/NCSA Open Source License. Both these licenses can be | ||
* found in the LICENSE file. | ||
*/ | ||
|
||
#include <signal.h> | ||
#include <stdio.h> | ||
#include <unistd.h> | ||
#include <errno.h> | ||
|
||
int kill(pid_t pid, int sig) { | ||
if (pid == getpid()) { | ||
return raise(sig); | ||
} | ||
errno = EPERM; | ||
return -1; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -259,7 +259,7 @@ int sigandset(sigset_t *, const sigset_t *, const sigset_t *); | |
|
||
#define SIG_ERR ((void (*)(int))-1) | ||
#define SIG_DFL ((void (*)(int)) 0) | ||
#define SIG_IGN ((void (*)(int)) 1) | ||
#define SIG_IGN ((void (*)(int))-2) /* XXX EMSCRIPTEN: use -2 since 1 is a valid function address */ | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This change seems to be a regression for Python. Some code somewhere must assume that There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Reverting this could be tricky because, as the comment says, 1 is a valid function point in Wasm.. so comparing and incoming function pointer with Can you find the fix the code that contains that assumption? .. Could it be that not all object files were re-built? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yeah the rational for the change is clear. I am trying to hunt down what is going on. The error reproduces in our CI so it's definitely from a clean build. But we could patch it back in Pyodide if need be. @tiran Have you run into this problem? Is there a Python patch we can backport? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. As a last resort we could shift the range of valid functions pointers from a range that starts at 1 to range that starts at something higher, such as 16, but it seems like quite in intrusive change (and not 100% free at runtime)... so I would much rather find an fix offending code. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It looks to me like the Python signalmodule code has a bug that is fixed in v3.11: in v3.10 in compares the memory addresses of the Python objects, and it's returning False because they are two different There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Okay I will apply the patch in @tiran's comment there. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Thanks @sbc100 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yeah, I ran into the problem, too. PR python/cpython#31759 fixed the issue in 3.11 / main. I did not backport the fix to 3.10. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Backports cleanly though. Thanks @tiran it's interesting rediscovering your work like this. |
||
|
||
typedef int sig_atomic_t; | ||
|
||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,70 @@ | ||
/* | ||
* Copyright 2021 The Emscripten Authors. All rights reserved. | ||
* Emscripten is available under two separate licenses, the MIT license and the | ||
* University of Illinois/NCSA Open Source License. Both these licenses can be | ||
* found in the LICENSE file. | ||
*/ | ||
|
||
#define _GNU_SOURCE // for sigorset/sigandset | ||
#include <stdbool.h> | ||
#include <threads.h> | ||
#include <signal.h> | ||
#include <errno.h> | ||
#include "libc.h" | ||
|
||
#define SST_SIZE (_NSIG/8/sizeof(long)) | ||
|
||
static thread_local sigset_t __sig_mask; | ||
sigset_t __sig_pending; | ||
|
||
static int siginvertset(sigset_t *dest, const sigset_t *src) { | ||
unsigned long i = 0, *d = (void*) dest, *s = (void*) src; | ||
for(; i < SST_SIZE; i++) d[i] = ~s[i]; | ||
return 0; | ||
} | ||
|
||
bool __sig_is_blocked(int sig) { | ||
return sigismember(&__sig_mask, sig); | ||
} | ||
|
||
int pthread_sigmask(int how, const sigset_t *restrict set, sigset_t *restrict old) { | ||
if (old) { | ||
*old = __sig_mask; | ||
} | ||
|
||
switch (how) { | ||
case SIG_SETMASK: | ||
__sig_mask = *set; | ||
break; | ||
case SIG_BLOCK: | ||
sigorset(&__sig_mask, &__sig_mask, set); | ||
break; | ||
case SIG_UNBLOCK: { | ||
sigset_t tmp; | ||
siginvertset(&tmp, set); | ||
sigandset(&__sig_mask, &__sig_mask, &tmp); | ||
break; | ||
} | ||
default: | ||
return EINVAL; | ||
} | ||
|
||
// These two signals can never be blocked. | ||
sigdelset(&__sig_mask, SIGKILL); | ||
sigdelset(&__sig_mask, SIGSTOP); | ||
|
||
// Raise any pending signals that are now unblocked. | ||
for (int sig = 0; sig < _NSIG; sig++) { | ||
if (sigismember(&__sig_pending, sig) && !sigismember(&__sig_mask, sig)) { | ||
sigdelset(&__sig_pending, sig); | ||
raise(sig); | ||
} | ||
} | ||
|
||
return 0; | ||
} | ||
|
||
int sigpending(sigset_t *set) { | ||
*set = __sig_pending; | ||
return 0; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,39 @@ | ||
/* | ||
* Copyright 2021 The Emscripten Authors. All rights reserved. | ||
* Emscripten is available under two separate licenses, the MIT license and the | ||
* University of Illinois/NCSA Open Source License. Both these licenses can be | ||
* found in the LICENSE file. | ||
*/ | ||
|
||
#define _GNU_SOURCE // for sighandler_t | ||
#include <stdbool.h> | ||
#include <stddef.h> | ||
#include <signal.h> | ||
|
||
extern struct sigaction __sig_actions[_NSIG]; | ||
extern sigset_t __sig_pending; | ||
|
||
bool __sig_is_blocked(int sig); | ||
|
||
extern void __call_sighandler(sighandler_t handler, int sig); | ||
|
||
int raise(int sig) { | ||
if (__sig_is_blocked(sig)) { | ||
sigaddset(&__sig_pending, sig); | ||
return 0; | ||
} | ||
if (__sig_actions[sig].sa_flags & SA_SIGINFO) { | ||
siginfo_t t = {0}; | ||
__sig_actions[sig].sa_sigaction(sig, &t, NULL); | ||
} else { | ||
if (__sig_actions[sig].sa_handler == SIG_DFL || __sig_actions[sig].sa_handler == SIG_IGN) { | ||
|
||
return 0; | ||
} | ||
// Avoid a direct call to the handler, and instead call via JS so we can | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. should we add a test with a mismatched signature below, or is posixtestsuite sufficient? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think we can call posixtestsuite enough. |
||
// avoid strict signature checking. | ||
// https://github.com/emscripten-core/posixtestsuite/issues/6 | ||
__call_sighandler(__sig_actions[sig].sa_handler, sig); | ||
} | ||
return 0; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
/* | ||
* Copyright 2021 The Emscripten Authors. All rights reserved. | ||
* Emscripten is available under two separate licenses, the MIT license and the | ||
* University of Illinois/NCSA Open Source License. Both these licenses can be | ||
* found in the LICENSE file. | ||
*/ | ||
|
||
#include <stdio.h> | ||
#include <signal.h> | ||
#include <errno.h> | ||
#include "libc.h" | ||
|
||
struct sigaction __sig_actions[_NSIG]; | ||
|
||
int __sigaction(int sig, const struct sigaction *restrict sa, struct sigaction *restrict old) { | ||
if (sig < 0 || sig >= _NSIG) { | ||
errno = EINVAL; | ||
return -1; | ||
} | ||
|
||
if (old) { | ||
*old = __sig_actions[sig]; | ||
} | ||
|
||
__sig_actions[sig] = *sa; | ||
return 0; | ||
} | ||
|
||
weak_alias(__sigaction, sigaction); | ||
sbc100 marked this conversation as resolved.
Show resolved
Hide resolved
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
/* | ||
* Copyright 2021 The Emscripten Authors. All rights reserved. | ||
* Emscripten is available under two separate licenses, the MIT license and the | ||
* University of Illinois/NCSA Open Source License. Both these licenses can be | ||
* found in the LICENSE file. | ||
*/ | ||
|
||
#include <signal.h> | ||
#include <errno.h> | ||
#include "syscall.h" | ||
#include "libc.h" | ||
|
||
extern sigset_t __sig_pending; | ||
|
||
int sigtimedwait(const sigset_t *restrict mask, siginfo_t *restrict si, const struct timespec *restrict timeout) { | ||
for (int sig = 0; sig < _NSIG; sig++) { | ||
if (sigismember(mask, sig) && sigismember(&__sig_pending, sig)) { | ||
if (si) { | ||
siginfo_t t = {0}; | ||
*si = t; | ||
} | ||
sigdelset(&__sig_pending, sig); | ||
return sig; | ||
} | ||
} | ||
|
||
errno = EINVAL; | ||
return -1; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This changelog note should probably be moved to 2.0.28 (since 2.0.27 was already released).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oops, I guess we never mark 2.0.27 as released: #14898