1 /* sigaltstack wrappers.
2 Copyright (C) 2019-2022 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
9
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
14
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, see
17 <https://www.gnu.org/licenses/>. */
18
19 #include <support/xsignal.h>
20 #include <support/support.h>
21 #include <support/xunistd.h>
22 #include <support/check.h>
23
24 #include <stdlib.h>
25 #include <string.h>
26 #include <sys/mman.h>
27 #include <sys/param.h> /* roundup, MAX */
28
29 #ifndef MAP_NORESERVE
30 # define MAP_NORESERVE 0
31 #endif
32 #ifndef MAP_STACK
33 # define MAP_STACK 0
34 #endif
35
36 /* The "cookie" returned by xalloc_sigstack points to one of these
37 structures. */
38 struct sigstack_desc
39 {
40 struct support_stack stack;
41 stack_t alt_stack; /* The address and size of the stack itself. */
42 stack_t old_stack; /* The previous signal stack. */
43 };
44
45 void *
xalloc_sigstack(size_t size)46 xalloc_sigstack (size_t size)
47 {
48 struct sigstack_desc *desc = xmalloc (sizeof (struct sigstack_desc));
49 desc->stack = support_stack_alloc (size);
50 desc->alt_stack.ss_sp = desc->stack.stack;
51 desc->alt_stack.ss_flags = 0;
52 desc->alt_stack.ss_size = desc->stack.size;
53
54 if (sigaltstack (&desc->alt_stack, &desc->old_stack))
55 FAIL_EXIT1 ("sigaltstack (new stack: sp=%p, size=%zu, flags=%u): %m\n",
56 desc->alt_stack.ss_sp, desc->alt_stack.ss_size,
57 desc->alt_stack.ss_flags);
58
59 return desc;
60 }
61
62 void
xfree_sigstack(void * stack)63 xfree_sigstack (void *stack)
64 {
65 struct sigstack_desc *desc = stack;
66
67 if (sigaltstack (&desc->old_stack, 0))
68 FAIL_EXIT1 ("sigaltstack (restore old stack: sp=%p, size=%zu, flags=%u): "
69 "%m\n", desc->old_stack.ss_sp, desc->old_stack.ss_size,
70 desc->old_stack.ss_flags);
71 support_stack_free (&desc->stack);
72 free (desc);
73 }
74
75 void
xget_sigstack_location(const void * stack,unsigned char ** addrp,size_t * sizep)76 xget_sigstack_location (const void *stack, unsigned char **addrp, size_t *sizep)
77 {
78 const struct sigstack_desc *desc = stack;
79 *addrp = desc->alt_stack.ss_sp;
80 *sizep = desc->alt_stack.ss_size;
81 }
82