Add some basic signal support.

It only works for sending a signal to a process that's in userspace code.

We implement reception by synthesizing a PUSHA+PUSHF in the receiving process
(operating on values in the TSS.)
The TSS CS:EIP is then rerouted to the signal handler and a tiny return
trampoline is constructed in a dedicated region in the receiving process.

Also hacked up /bin/kill to be able to send arbitrary signals (kill -N PID)
This commit is contained in:
Andreas Kling 2018-11-06 10:46:40 +01:00
commit 153ea704af
Notes: sideshowbarker 2024-07-19 18:33:15 +09:00
13 changed files with 240 additions and 30 deletions

View file

@ -6,6 +6,7 @@
#include <LibC/stdlib.h>
#include <LibC/utsname.h>
#include <LibC/pwd.h>
#include <signal.h>
#include <AK/FileSystemPath.h>
struct GlobalState {
@ -32,6 +33,29 @@ static int sh_pwd(int, const char**)
return 0;
}
void did_receive_signal(int signum)
{
printf("\nMy word, I've received a signal with number %d\n", signum);
//exit(0);
}
static int sh_busy(int, const char**)
{
struct sigaction sa;
sa.sa_handler = did_receive_signal;
sa.sa_flags = 0;
sa.sa_mask = 0;
sa.sa_restorer = nullptr;
int rc = sigaction(SIGUSR1, &sa, nullptr);
assert(rc == 0);
printf("listening for SIGUSR1 while looping in userspace...\n");
for (;;) {
for (volatile int i = 0; i < 100000; ++i)
;
}
return 0;
}
static int sh_fork(int, const char**)
{
pid_t pid = fork();
@ -147,6 +171,10 @@ static bool handle_builtin(int argc, const char** argv, int& retval)
retval = sh_fef(argc, argv);
return true;
}
if (!strcmp(argv[0], "busy")) {
retval = sh_busy(argc, argv);
return true;
}
if (!strcmp(argv[0], "wt")) {
retval = sh_wt(argc, argv);
return true;