mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 22:22:45 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			91 lines
		
	
	
	
		
			2.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			91 lines
		
	
	
	
		
			2.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <bits/stdint.h>
 | |
| #include <signal.h>
 | |
| #include <stdbool.h>
 | |
| #include <sys/cdefs.h>
 | |
| #include <sys/types.h>
 | |
| 
 | |
| __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 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) == 120, "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
 | 
