occlum/test/signal/main.c

625 lines
19 KiB
C

#define _GNU_SOURCE
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <sys/wait.h>
#include <unistd.h>
#include <ucontext.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <spawn.h>
#include <assert.h>
#include <string.h>
#include <fcntl.h>
#include <signal.h>
#include <pthread.h>
#include <errno.h>
#include <time.h>
#include "test.h"
// ============================================================================
// Helper macros
// ============================================================================
// ============================================================================
// Helper functions
// ============================================================================
// ============================================================================
// Test sigprocmask
// ============================================================================
// Add a new macro to compare two sigset. Returns 0 iff the two sigset are equal.
// Musl libc defines sigset_t to 16 bytes, but on x86 only the first 8 bytes are
// meaningful. So this comparison only takes the first 8 bytes into account.
#define sigcmpset(a, b) memcmp((a), (b), 8)
int test_sigprocmask() {
int ret;
sigset_t new, old;
sigset_t expected_old;
// Check sigmask == []
if ((ret = sigprocmask(0, NULL, &old)) < 0) {
THROW_ERROR("sigprocmask failed unexpectedly");
}
sigemptyset(&expected_old);
if (sigcmpset(&old, &expected_old) != 0) {
THROW_ERROR("unexpected old sigset");
}
// SIG_BLOCK: [] --> [SIGSEGV]
sigemptyset(&new);
sigaddset(&new, SIGSEGV);
if ((ret = sigprocmask(SIG_BLOCK, &new, &old)) < 0) {
THROW_ERROR("sigprocmask failed unexpectedly");
}
sigemptyset(&expected_old);
if (sigcmpset(&old, &expected_old) != 0) {
THROW_ERROR("unexpected old sigset");
}
// SIG_SETMASK: [SIGSEGV] --> [SIGIO]
sigemptyset(&new);
sigaddset(&new, SIGIO);
if ((ret = sigprocmask(SIG_SETMASK, &new, &old)) < 0) {
THROW_ERROR("sigprocmask failed unexpectedly");
}
sigemptyset(&expected_old);
sigaddset(&expected_old, SIGSEGV);
if (sigcmpset(&old, &expected_old) != 0) {
THROW_ERROR("unexpected old sigset");
}
// SIG_UNBLOCK: [SIGIO] -> []
if ((ret = sigprocmask(SIG_UNBLOCK, &new, &old)) < 0) {
THROW_ERROR("sigprocmask failed unexpectedly");
}
sigemptyset(&expected_old);
sigaddset(&expected_old, SIGIO);
if (sigcmpset(&old, &expected_old) != 0) {
THROW_ERROR("unexpected old sigset");
}
// Check sigmask == []
if ((ret = sigprocmask(0, NULL, &old)) < 0) {
THROW_ERROR("sigprocmask failed unexpectedly");
}
sigemptyset(&expected_old);
if (sigcmpset(&old, &expected_old) != 0) {
THROW_ERROR("unexpected old sigset");
}
return 0;
}
// ============================================================================
// Test raise syscall and user-registered signal handlers
// ============================================================================
#define MAX_RECURSION_LEVEL 3
static void handle_sigio(int num, siginfo_t *info, void *context) {
static volatile int recursion_level = 0;
printf("Hello from SIGIO signal handler (recursion_level = %d)!\n", recursion_level);
recursion_level++;
if (recursion_level <= MAX_RECURSION_LEVEL) {
raise(SIGIO);
}
recursion_level--;
}
int test_raise() {
struct sigaction new_action, old_action;
memset(&new_action, 0, sizeof(struct sigaction));
memset(&old_action, 0, sizeof(struct sigaction));
new_action.sa_sigaction = handle_sigio;
new_action.sa_flags = SA_SIGINFO | SA_NODEFER;
if (sigaction(SIGIO, &new_action, &old_action) < 0) {
THROW_ERROR("registering new signal handler failed");
}
if (old_action.sa_handler != SIG_DFL) {
THROW_ERROR("unexpected old sig handler");
}
raise(SIGIO);
if (sigaction(SIGIO, &old_action, NULL) < 0) {
THROW_ERROR("restoring old signal handler failed");
}
return 0;
}
// ============================================================================
// Test abort, which uses SIGABRT behind the scene
// ============================================================================
int test_abort() {
pid_t child_pid;
char *child_argv[] = {"signal", "aborted_child", NULL};
int ret;
int status;
// Repeat multiple times to check that the resources of the killed child
// processes are indeed freed by the LibOS
for (int i = 0; i < 3; i++) {
ret = posix_spawn(&child_pid, "/bin/signal", NULL, NULL, child_argv, NULL);
if (ret < 0) {
THROW_ERROR("failed to spawn a child process\n");
}
ret = wait4(-1, &status, 0, NULL);
if (ret < 0) {
THROW_ERROR("failed to wait4 the child process\n");
}
if (!WIFSIGNALED(status) || WTERMSIG(status) != SIGABRT) {
THROW_ERROR("child process is expected to be killed by SIGILL\n");
}
}
return 0;
}
static int aborted_child() {
while (1) {
abort();
}
return 0;
}
// ============================================================================
// Test kill by sending SIGKILL to another process
// ============================================================================
int test_kill() {
pid_t child_pid;
char *child_argv[] = {"signal", "killed_child", NULL};
int ret;
int status;
// Repeat multiple times to check that the resources of the killed child
// processes are indeed freed by the LibOS
for (int i = 0; i < 3; i++) {
ret = posix_spawn(&child_pid, "/bin/signal", NULL, NULL, child_argv, NULL);
if (ret < 0) {
THROW_ERROR("failed to spawn a child process\n");
}
kill(child_pid, SIGKILL);
ret = wait4(-1, &status, 0, NULL);
if (ret < 0) {
THROW_ERROR("failed to wait4 the child process\n");
}
if (!WIFSIGNALED(status) || WTERMSIG(status) != SIGKILL) {
THROW_ERROR("child process is expected to be killed by SIGILL\n");
}
}
return 0;
}
// TODO: remove the use of getpid when we can deliver signals through interrupt
static int killed_child() {
while (1) {
getpid();
}
return 0;
}
// ============================================================================
// Test catching and handling hardware exception
// ============================================================================
static int SIGFPE_RESURSIVE_LEVEL = 0;
#define fxsave(addr) __asm __volatile("fxsave %0" : "=m" (*(addr)))
// Note: this function is fragile in the sense that compiler may not always
// emit the instruction pattern that triggers divide-by-zero as we expect.
// TODO: rewrite this in assembly
int div_maybe_zero(int x, int y) {
return x / y;
}
static void handle_sigfpe(int num, siginfo_t *info, void *_context) {
printf("SIGFPE Caught\n");
assert(num == SIGFPE);
assert(info->si_signo == SIGFPE);
char x[512] __attribute__((aligned(16))) = {};
char y[512] __attribute__((aligned(16))) = {};
SIGFPE_RESURSIVE_LEVEL ++;
ucontext_t *ucontext = _context;
mcontext_t *mcontext = &ucontext->uc_mcontext;
// The faulty instruction should be `idiv %esi` (f7 fe)
mcontext->gregs[REG_RIP] += 2;
fxsave(x);
// Try modify the floating point register
float a = 3.00001234567890123 / 2.001;
printf("a = %f\n", a);
float value = 3.14f;
__asm__ volatile (
"movss %[newval], %%xmm0" // Move new value to xmm0 register
:
: [newval] "m" (value) // Input constraint: value is a memory operand
: "%xmm0" // Use xmm0 register
);
fxsave(y);
// Make sure the floating point registers are modified
if (memcmp(x, y, 512) == 0) {
printf("floating point registers is not modified\n");
abort();
}
// Recursively trigger exception
if (SIGFPE_RESURSIVE_LEVEL < 4) {
volatile int c;
// Trigger divide-by-zero exception
int a = 1;
int b = 0;
c = div_maybe_zero(a, b);
}
return;
}
int test_handle_sigfpe() {
char *stack = malloc(SIGSTKSZ * 4);
if (stack == NULL) {
THROW_ERROR("stack allocation failed");
}
stack_t expected_ss = {
.ss_size = SIGSTKSZ * 4,
.ss_sp = stack,
.ss_flags = 0,
};
if (sigaltstack(&expected_ss, NULL) < 0) {
free(stack);
THROW_ERROR("failed to call sigaltstack");
}
// Set up a signal handler that handles divide-by-zero exception
struct sigaction new_action, old_action;
memset(&new_action, 0, sizeof(struct sigaction));
memset(&old_action, 0, sizeof(struct sigaction));
new_action.sa_sigaction = handle_sigfpe;
new_action.sa_flags = SA_SIGINFO | SA_NODEFER | SA_ONSTACK;
if (sigaction(SIGFPE, &new_action, &old_action) < 0) {
free(stack);
THROW_ERROR("registering new signal handler failed");
}
if (old_action.sa_handler != SIG_DFL) {
free(stack);
THROW_ERROR("unexpected old sig handler");
}
char x[512] __attribute__((aligned(16))) = {};
char y[512] __attribute__((aligned(16))) = {};
// Trigger divide-by-zero exception
int a = 1;
int b = 0;
// Use volatile to prevent compiler optimization
volatile int c;
fxsave(x);
c = div_maybe_zero(a, b);
fxsave(y);
if (memcmp(x, y, 512) != 0) {
free(stack);
THROW_ERROR("floating point registers are modified");
}
printf("Signal handler successfully jumped over the divide-by-zero instruction\n");
if (sigaction(SIGFPE, &old_action, NULL) < 0) {
free(stack);
THROW_ERROR("restoring old signal handler failed");
}
free(stack);
return 0;
}
// TODO: rewrite this in assembly
int read_maybe_null(int *p) {
return *p;
}
static void handle_sigsegv(int num, siginfo_t *info, void *_context) {
printf("SIGSEGV Caught\n");
assert(num == SIGSEGV);
assert(info->si_signo == SIGSEGV);
ucontext_t *ucontext = _context;
mcontext_t *mcontext = &ucontext->uc_mcontext;
// TODO: how long is the instruction?
// The faulty instruction should be `idiv %esi` (f7 fe)
mcontext->gregs[REG_RIP] += 2;
return;
}
int test_handle_sigsegv() {
// Set up a signal handler that handles divide-by-zero exception
struct sigaction new_action, old_action;
memset(&new_action, 0, sizeof(struct sigaction));
memset(&old_action, 0, sizeof(struct sigaction));
new_action.sa_sigaction = handle_sigsegv;
new_action.sa_flags = SA_SIGINFO;
if (sigaction(SIGSEGV, &new_action, &old_action) < 0) {
THROW_ERROR("registering new signal handler failed");
}
if (old_action.sa_handler != SIG_DFL) {
THROW_ERROR("unexpected old sig handler");
}
int *addr = NULL;
volatile int val = read_maybe_null(addr);
(void)val; // to suppress "unused variables" warning
printf("Signal handler successfully jumped over a null-dereferencing instruction\n");
void *ptr = mmap(NULL, 8192, PROT_READ | PROT_WRITE, MAP_ANONYMOUS | MAP_PRIVATE, -1, 0);
if (ptr == NULL) {
THROW_ERROR("mmap failure");
}
int ret = mprotect(ptr, 8192, PROT_NONE);
if (ret < 0) {
THROW_ERROR("mprotect failure");
}
val = read_maybe_null(ptr);
(void)val; // to suppress "unused variables" warning
printf("Signal handler successfully jumped over a PROT_NONE-visit instruction\n");
if (sigaction(SIGSEGV, &old_action, NULL) < 0) {
THROW_ERROR("restoring old signal handler failed");
}
return 0;
}
// ============================================================================
// Test handle signal on alternate signal stack
// ============================================================================
#define MAX_ALTSTACK_RECURSION_LEVEL 2
stack_t g_old_ss;
static void handle_sigpipe(int num, siginfo_t *info, void *context) {
static volatile int recursion_level = 0;
printf("Hello from SIGPIPE signal handler on the alternate signal stack (recursion_level = %d)\n",
recursion_level);
// save old_ss to check if we are on stack
stack_t old_ss;
sigaltstack(NULL, &old_ss);
g_old_ss = old_ss;
recursion_level++;
if (recursion_level <= MAX_ALTSTACK_RECURSION_LEVEL) {
raise(SIGPIPE);
}
recursion_level--;
}
int test_sigaltstack() {
char *stack = malloc(SIGSTKSZ);
if (stack == NULL) {
THROW_ERROR("stack allocation failed");
}
stack_t expected_ss = {
.ss_size = SIGSTKSZ,
.ss_sp = stack,
.ss_flags = 0,
};
if (sigaltstack(&expected_ss, NULL) < 0) {
free(stack);
THROW_ERROR("failed to call sigaltstack");
}
stack_t actual_ss;
if (sigaltstack(NULL, &actual_ss) < 0) {
free(stack);
THROW_ERROR("failed to call sigaltstack");
}
if (actual_ss.ss_size != expected_ss.ss_size
|| actual_ss.ss_sp != expected_ss.ss_sp
|| actual_ss.ss_flags != expected_ss.ss_flags) {
free(stack);
THROW_ERROR("failed to check the signal stack after set");
}
struct sigaction new_action, old_action;
memset(&new_action, 0, sizeof(struct sigaction));
memset(&old_action, 0, sizeof(struct sigaction));
new_action.sa_sigaction = handle_sigpipe;
new_action.sa_flags = SA_SIGINFO | SA_NODEFER | SA_ONSTACK;
if (sigaction(SIGPIPE, &new_action, &old_action) < 0) {
free(stack);
THROW_ERROR("registering new signal handler failed");
}
if (old_action.sa_handler != SIG_DFL) {
free(stack);
THROW_ERROR("unexpected old sig handler");
}
raise(SIGPIPE);
if (g_old_ss.ss_flags != SS_ONSTACK) {
free(stack);
THROW_ERROR("check stack flags failed");
}
if (sigaction(SIGPIPE, &old_action, NULL) < 0) {
free(stack);
THROW_ERROR("restoring old signal handler failed");
}
free(stack);
return 0;
}
// ============================================================================
// Test SIGCHLD signal
// ============================================================================
int sigchld = 0;
void proc_exit() {
sigchld = 1;
}
int test_sigchld() {
signal(SIGCHLD, proc_exit);
int ret, child_pid;
printf("Run a parent process has pid = %d and ppid = %d\n", getpid(), getppid());
ret = posix_spawn(&child_pid, "/bin/getpid", NULL, NULL, NULL, NULL);
if (ret < 0) {
printf("ERROR: failed to spawn a child process\n");
return -1;
}
printf("Spawn a new proces successfully (pid = %d)\n", child_pid);
wait(NULL);
if (sigchld == 0) { THROW_ERROR("Did not receive SIGCHLD"); }
return 0;
}
// ============================================================================
// Test sigtimedwait syscall
// ============================================================================
struct send_signal_data {
pthread_t target;
int signum;
struct timespec delay;
};
static void *send_signal_with_delay(void *_data) {
int ret;
struct send_signal_data *data = _data;
// Ensure data->delay time elapsed
while ((ret = nanosleep(&data->delay, NULL) < 0 && errno == EINTR)) ;
// Send the signal to the target thread
pthread_kill(data->target, data->signum);
free(data);
return NULL;
}
// Raise a signal for the current thread asynchronously by spawning another
// thread to send the signal to the parent thread after some specified delay.
// The delay is meant to ensure some operation in the parent thread is
// completed by the time the signal is sent.
static pthread_t raise_async(int signum, const struct timespec *delay) {
pthread_t thread;
int ret;
struct send_signal_data *data = malloc(sizeof(*data));
data->target = pthread_self();
data->signum = signum;
data->delay = *delay;
if ((ret = pthread_create(&thread, NULL, send_signal_with_delay, (void *)data)) < 0) {
printf("ERROR: pthread_create failed unexpectedly\n");
abort();
}
return thread;
}
int test_sigtimedwait() {
int ret;
siginfo_t info;
sigset_t new_mask, old_mask;
struct timespec delay, timeout;
// Update signal mask to block SIGIO
sigemptyset(&new_mask);
sigaddset(&new_mask, SIGIO);
if ((ret = sigprocmask(SIG_BLOCK, &new_mask, &old_mask)) < 0) {
THROW_ERROR("sigprocmask failed unexpectedly");
}
timeout.tv_sec = 1;
timeout.tv_nsec = 0;
// There is no pending signal, yet; so the syscall must return EAGAIN error
ret = sigtimedwait(&new_mask, &info, &timeout);
if (ret == 0 || (ret < 0 && errno != EAGAIN)) {
THROW_ERROR("sigprocmask must return with EAGAIN error");
}
// Let's generate a pending signal and then get it
raise(SIGIO);
if ((ret = sigtimedwait(&new_mask, &info, NULL)) < 0 || info.si_signo != SIGIO) {
THROW_ERROR("sigtimedwait should return the SIGIO");
}
// Now let's generate a pending signal in an async way. The pending signal
// does not exist yet at the time when sigtimedwait is called. So the
// current thread will be put to sleep and waken up until the
// asynchronously raised signal is sent to the current thread and becomes
// pending.
delay.tv_sec = 0;
delay.tv_nsec = 10 * 1000 * 1000; // 10ms
pthread_t thread = raise_async(SIGIO, &delay);
timeout.tv_sec = 0;
timeout.tv_nsec = 2 * delay.tv_nsec;
if ((ret = sigtimedwait(&new_mask, &info, &timeout)) < 0 || info.si_signo != SIGIO) {
THROW_ERROR("sigtimedwait should return the SIGIO");
}
// Restore the signal mask
if ((ret = sigprocmask(SIG_SETMASK, &old_mask, NULL)) < 0) {
THROW_ERROR("sigprocmask failed unexpectedly");
}
if (pthread_join(thread, NULL) != 0) {
THROW_ERROR("failed to join the thread");
}
return 0;
}
// ============================================================================
// Test suite main
// ============================================================================
static test_case_t test_cases[] = {
TEST_CASE(test_sigprocmask),
TEST_CASE(test_raise),
TEST_CASE(test_abort),
TEST_CASE(test_kill),
TEST_CASE(test_handle_sigfpe),
TEST_CASE(test_handle_sigsegv),
TEST_CASE(test_sigaltstack),
TEST_CASE(test_sigchld),
TEST_CASE(test_sigtimedwait),
};
int main(int argc, const char *argv[]) {
if (argc > 1) {
const char *cmd = argv[1];
if (strcmp(cmd, "aborted_child") == 0) {
return aborted_child();
} else if (strcmp(cmd, "killed_child") == 0) {
return killed_child();
} else {
fprintf(stderr, "ERROR: unknown command: %s\n", cmd);
return EXIT_FAILURE;
}
}
return test_suite_run(test_cases, ARRAY_SIZE(test_cases));
}