Skip to content
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

Merged
merged 1 commit into from
Aug 17, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions ChangeLog.md
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,10 @@ See docs/process.md for more on how version tagging works.

2.0.27
------
- Added some support for signal handling libc functions (raise, kill,
sigaction, sigpending, etc). We still don't have a way to deliver signals from
the outside but these at least now work for sending signals to the current
thread (JS context) (#14883).
Comment on lines +23 to +26
Copy link
Collaborator

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).

Copy link
Collaborator Author

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

- Added `EM_ASYNC_JS` macro - similar to `EM_JS`, but allows using `await`
inside the JS block and automatically integrates with Asyncify without
the need for listing the declared function in `ASYNCIFY_IMPORTS` (#9709).
Expand Down
20 changes: 20 additions & 0 deletions src/library.js
Original file line number Diff line number Diff line change
Expand Up @@ -2619,6 +2619,20 @@ LibraryManager.library = {
return 0;
},

// http://pubs.opengroup.org/onlinepubs/000095399/functions/alarm.html
alarm__deps: ['raise'],
alarm: function(seconds) {
setTimeout(function() {
_raise({{{ cDefine('SIGALRM') }}});
}, seconds*1000);
},

// Helper for raise() to avoid signature mismatch failures:
// https://github.com/emscripten-core/posixtestsuite/issues/6
__call_sighandler: function(fp, sig) {
{{{ makeDynCall('vi', 'fp') }}}(sig);
},

// ==========================================================================
// emscripten.h
// ==========================================================================
Expand Down Expand Up @@ -3681,6 +3695,12 @@ LibraryManager.library = {
},
#endif

$safeSetTimeout__deps: ['$callUserCallback',
#if !MINIMAL_RUNTIME
'$runtimeKeepalivePush',
'$runtimeKeepalivePop',
#endif
],
$safeSetTimeout: function(func, timeout) {
{{{ runtimeKeepalivePush() }}}
return setTimeout(function() {
Expand Down
85 changes: 0 additions & 85 deletions src/library_signals.js

This file was deleted.

1 change: 0 additions & 1 deletion src/modules.js
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,6 @@ var LibraryManager = {
'library_formatString.js',
'library_math.js',
'library_path.js',
'library_signals.js',
'library_syscall.js',
'library_html5.js',
'library_stack_trace.js',
Expand Down
19 changes: 19 additions & 0 deletions system/lib/libc/kill.c
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;
}
2 changes: 1 addition & 1 deletion system/lib/libc/musl/include/signal.h
Original file line number Diff line number Diff line change
Expand Up @@ -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 */
Copy link
Collaborator

Choose a reason for hiding this comment

The 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 SIG_IGN is 1.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The 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 SIG_IGN could produce false positives.

Can you find the fix the code that contains that assumption? ..

Could it be that not all object files were re-built?

Copy link
Collaborator

@hoodmane hoodmane Jun 8, 2022

Choose a reason for hiding this comment

The 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?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The 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.

Copy link
Collaborator

Choose a reason for hiding this comment

The 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 -2's. I suspect the Python interpreter interns 1 in some special way which makes the wrong code work by chance when the value in question is 1.

Copy link
Collaborator

@hoodmane hoodmane Jun 8, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Okay I will apply the patch in @tiran's comment there.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks @sbc100

Copy link
Contributor

Choose a reason for hiding this comment

The 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.

Copy link
Collaborator

Choose a reason for hiding this comment

The 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;

Expand Down
70 changes: 70 additions & 0 deletions system/lib/libc/pthread_sigmask.c
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;
}
39 changes: 39 additions & 0 deletions system/lib/libc/raise.c
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
Copy link
Member

Choose a reason for hiding this comment

The 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?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The 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;
}
29 changes: 29 additions & 0 deletions system/lib/libc/sigaction.c
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);
29 changes: 29 additions & 0 deletions system/lib/libc/sigtimedwait.c
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;
}
3 changes: 3 additions & 0 deletions tests/test_core.py
Original file line number Diff line number Diff line change
Expand Up @@ -5319,6 +5319,9 @@ def test_fs_64bit(self):
def test_sigalrm(self):
self.do_runf(test_file('test_sigalrm.c'), 'Received alarm!')

def test_signals(self):
self.do_core_test(test_file('test_signals.c'))

@no_windows('https://github.com/emscripten-core/emscripten/issues/8882')
def test_unistd_access(self):
self.uses_es6 = True
Expand Down
10 changes: 4 additions & 6 deletions tests/test_posixtest.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,8 +29,6 @@ class posixtest(RunnerCore):

def filter_tests(all_tests):
pthread_tests = [t for t in all_tests if t.startswith('pthread_')]
# filter out some tests we don't support
pthread_tests = [t for t in pthread_tests if not t.startswith('pthread_sigmask')]
return pthread_tests


Expand All @@ -56,15 +54,15 @@ def get_pthread_tests():
'test_pthread_atfork_3_2': 'fork() and multiple processes are not supported',
'test_pthread_atfork_4_1': 'fork() and multiple processes are not supported',
'test_pthread_kill_1_1': 'signals are not supported',
'test_pthread_create_1_5': 'semaphores are not supported',
'test_pthread_create_1_5': 'fork() and multiple processes are not supported',
'test_pthread_exit_6_1': 'lacking necessary mmap() support',
'test_pthread_spin_lock_1_1': 'signals are not supported',
'test_pthread_mutex_lock_5_1': 'signals are not supported',
'test_pthread_mutexattr_settype_2_1': 'interrupting pthread_mutex_lock wait via SIGALRM is not supported',
'test_pthread_spin_lock_3_1': 'signals are not supported',
'test_pthread_create_14_1': 'signals are not supported',
'test_pthread_detach_4_3': 'signals are not supported',
'test_pthread_join_6_3': 'signals are not supported',
'test_pthread_create_14_1': 'creates too many threads',
'test_pthread_detach_4_3': 'creates too many threads',
'test_pthread_join_6_3': 'creates too many threads',
'test_pthread_barrier_wait_3_2': 'signals are not supported',
'test_pthread_cond_broadcast_1_2': 'tries to create 10,0000 threads, then depends on fork()',
}
Expand Down
Loading