/* * Copyright (c) 2018-2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include __BEGIN_DECLS // // /!\ This structure is accessed inside setjmp.S, keep both files in sync! // struct __jmp_buf { #if defined(__x86_64__) uint64_t rbx; uint64_t r12; uint64_t r13; uint64_t r14; uint64_t r15; uint64_t rbp; uint64_t rsp; uint64_t rip; #elif defined(__aarch64__) // FIXME: This is likely incorrect. uint64_t regs[22]; #elif defined(__riscv) && __riscv_xlen == 64 uint64_t s[12]; uint64_t fs[12]; uint64_t sp; uint64_t ra; #else # error "Unknown architecture" #endif int did_save_signal_mask; sigset_t saved_signal_mask; }; typedef struct __jmp_buf jmp_buf[1]; typedef struct __jmp_buf sigjmp_buf[1]; /** * Since setjmp.h may be included by ports written in C, we need to guard this. */ #ifdef __cplusplus # if defined(__x86_64__) static_assert(sizeof(struct __jmp_buf) == 72, "struct __jmp_buf unsynchronized with x86_64/setjmp.S"); # elif defined(__aarch64__) static_assert(sizeof(struct __jmp_buf) == 184, "struct __jmp_buf unsynchronized with aarch64/setjmp.S"); # elif defined(__riscv) && __riscv_xlen == 64 static_assert(sizeof(struct __jmp_buf) == 216, "struct __jmp_buf unsynchronized with riscv64/setjmp.S"); # else # error "Unknown architecture" # endif #endif /** * 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. */ int setjmp(jmp_buf); __attribute__((noreturn)) void longjmp(jmp_buf, int val); int sigsetjmp(sigjmp_buf, int savesigs); __attribute__((noreturn)) void siglongjmp(sigjmp_buf, int val); /** * _setjmp() and _longjmp() are specified as behaving the exactly the same as * setjmp() and longjmp(), except they are not supposed to modify the signal mask. * * Our implementations already follow this restriction, so we just map them directly * to the same functions. * * https://pubs.opengroup.org/onlinepubs/9699969599/functions/_setjmp.html */ int _setjmp(jmp_buf); __attribute__((noreturn)) void _longjmp(jmp_buf, int val); __END_DECLS