1 # Check prerequisites for compiling lib/c-stack.c.
3 # Copyright (C) 2002 Free Software Foundation, Inc.
5 # This program is free software; you can redistribute it and/or modify
6 # it under the terms of the GNU General Public License as published by
7 # the Free Software Foundation; either version 2, or (at your option)
10 # This program 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
13 # GNU General Public License for more details.
15 # You should have received a copy of the GNU General Public License
16 # along with this program; if not, write to the Free Software
17 # Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
20 # Written by Paul Eggert.
22 AC_DEFUN([AC_SYS_XSI_STACK_OVERFLOW_HEURISTIC],
23 [# for STACK_DIRECTION
24 AC_REQUIRE([AC_FUNC_ALLOCA])
26 AC_CACHE_CHECK([for working C stack overflow detection],
27 ac_cv_sys_xsi_stack_overflow_heuristic,
35 char buffer[SIGSTKSZ];
39 } alternate_signal_stack;
42 # define find_stack_direction(ptr) STACK_DIRECTION
45 find_stack_direction (char const *addr)
48 return (! addr ? find_stack_direction (&dummy)
49 : addr < &dummy ? 1 : -1);
54 segv_handler (int signo, siginfo_t *info, void *context)
56 if (0 < info->si_code)
58 ucontext_t const *user_context = context;
59 char const *stack_min = user_context->uc_stack.ss_sp;
60 size_t stack_size = user_context->uc_stack.ss_size;
61 char const *faulting_address = info->si_addr;
62 size_t s = faulting_address - stack_min;
63 size_t page_size = sysconf (_SC_PAGESIZE);
64 if (find_stack_direction (0) < 0)
66 if (s < stack_size + page_size)
81 st.ss_sp = alternate_signal_stack.buffer;
82 st.ss_size = sizeof alternate_signal_stack.buffer;
83 r = sigaltstack (&st, 0);
87 sigemptyset (&act.sa_mask);
88 act.sa_flags = SA_NODEFER | SA_ONSTACK | SA_RESETHAND | SA_SIGINFO;
89 act.sa_sigaction = segv_handler;
90 return sigaction (SIGSEGV, &act, 0);
98 return *p + recurse (array);
105 return recurse ("\1");
108 [ac_cv_sys_xsi_stack_overflow_heuristic=yes],
109 [ac_cv_sys_xsi_stack_overflow_heuristic=no],
110 [ac_cv_sys_xsi_stack_overflow_heuristic=cross-compiling])])
112 if test $ac_cv_sys_xsi_stack_overflow_heuristic = yes; then
113 AC_DEFINE(HAVE_XSI_STACK_OVERFLOW_HEURISTIC, 1,
114 [Define to 1 if extending the stack slightly past the limit causes
115 a SIGSEGV, and an alternate stack can be established with sigaltstack,
116 and the signal handler is passed a context that specifies the
117 run time stack. This behavior is defined by POSIX 1003.1-2001
118 with the X/Open System Interface (XSI) option
119 and is a standardized way to implement a SEGV-based stack
120 overflow detection heuristic.])
124 AC_DEFUN([jm_PREREQ_C_STACK],
125 [AC_REQUIRE([AC_SYS_XSI_STACK_OVERFLOW_HEURISTIC])
127 # for STACK_DIRECTION
128 AC_REQUIRE([AC_FUNC_ALLOCA])
130 AC_CHECK_FUNCS(getcontext sigaltstack)
131 AC_CHECK_DECLS([getcontext], , , [#include <ucontext.h>])
132 AC_CHECK_DECLS([sigaltstack], , , [#include <signal.h>])
134 AC_CHECK_HEADERS(sys/resource.h ucontext.h unistd.h)
136 AC_CHECK_TYPES([stack_t], , , [#include <signal.h>])])