2020-01-18 08:38:21 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
|
|
*
|
2021-04-22 08:24:48 +00:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 08:38:21 +00:00
|
|
|
*/
|
|
|
|
|
2018-11-05 15:40:48 +00:00
|
|
|
#pragma once
|
|
|
|
|
2020-09-18 07:49:51 +00:00
|
|
|
#include <bits/stdint.h>
|
2019-02-26 13:05:28 +00:00
|
|
|
#include <signal.h>
|
|
|
|
#include <stdbool.h>
|
2019-05-28 09:53:16 +00:00
|
|
|
#include <sys/cdefs.h>
|
|
|
|
#include <sys/types.h>
|
2018-11-05 15:40:48 +00:00
|
|
|
|
|
|
|
__BEGIN_DECLS
|
|
|
|
|
2021-08-13 22:32:19 +00:00
|
|
|
//
|
|
|
|
// /!\ This structure is accessed inside setjmp.S, keep both files in sync!
|
|
|
|
//
|
|
|
|
|
2019-02-26 13:05:28 +00:00
|
|
|
struct __jmp_buf {
|
2021-07-01 10:04:20 +00:00
|
|
|
#ifdef __i386__
|
2021-08-13 22:32:19 +00:00
|
|
|
uint32_t ebx;
|
|
|
|
uint32_t esi;
|
|
|
|
uint32_t edi;
|
|
|
|
uint32_t ebp;
|
|
|
|
uint32_t esp;
|
|
|
|
uint32_t eip;
|
2021-07-01 10:04:20 +00:00
|
|
|
#elif __x86_64__
|
2021-08-13 22:32:19 +00:00
|
|
|
uint64_t rbx;
|
|
|
|
uint64_t r12;
|
|
|
|
uint64_t r13;
|
|
|
|
uint64_t r14;
|
|
|
|
uint64_t r15;
|
|
|
|
uint64_t rbp;
|
|
|
|
uint64_t rsp;
|
|
|
|
uint64_t rip;
|
2021-07-01 10:04:20 +00:00
|
|
|
#else
|
|
|
|
# error
|
|
|
|
#endif
|
2021-08-13 22:32:19 +00:00
|
|
|
int did_save_signal_mask;
|
2019-02-26 13:05:28 +00:00
|
|
|
sigset_t saved_signal_mask;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct __jmp_buf jmp_buf[1];
|
2019-11-17 20:01:12 +00:00
|
|
|
typedef struct __jmp_buf sigjmp_buf[1];
|
2018-11-05 15:40:48 +00:00
|
|
|
|
2021-08-25 15:11:34 +00:00
|
|
|
#ifdef __i386__
|
|
|
|
static_assert(sizeof(struct __jmp_buf) == 32, "struct __jmp_buf unsynchronized with i386/setjmp.S");
|
|
|
|
#elif __x86_64__
|
|
|
|
static_assert(sizeof(struct __jmp_buf) == 72, "struct __jmp_buf unsynchronized with x86_64/setjmp.S");
|
|
|
|
#else
|
|
|
|
# error
|
|
|
|
#endif
|
|
|
|
|
2021-08-13 22:32:19 +00:00
|
|
|
/**
|
|
|
|
* Calling conventions mandates that sigsetjmp() cannot call setjmp(),
|
|
|
|
* otherwise the restored calling environment will not be the original caller's
|
|
|
|
* but sigsetjmp()'s and we'll return to the wrong call site on siglongjmp().
|
|
|
|
*
|
|
|
|
* The setjmp(), sigsetjmp() and longjmp() functions have to be implemented in
|
|
|
|
* assembly because they touch the call stack and registers in non-portable
|
|
|
|
* ways. However, we *can* implement siglongjmp() as a standard C function.
|
|
|
|
*/
|
|
|
|
|
2018-11-05 15:40:48 +00:00
|
|
|
int setjmp(jmp_buf);
|
2021-08-13 22:32:19 +00:00
|
|
|
__attribute__((noreturn)) void longjmp(jmp_buf, int val);
|
2018-11-05 15:40:48 +00:00
|
|
|
|
2019-11-17 20:01:12 +00:00
|
|
|
int sigsetjmp(sigjmp_buf, int savesigs);
|
2021-08-13 22:32:19 +00:00
|
|
|
__attribute__((noreturn)) void siglongjmp(sigjmp_buf, int val);
|
2019-02-26 13:05:28 +00:00
|
|
|
|
2018-11-05 15:40:48 +00:00
|
|
|
__END_DECLS
|