Sync usage with man page.
[netbsd-mini2440.git] / gnu / dist / diffutils / m4 / c-stack.m4
blob0599d5023ff9d16c730bcb6f8c8e8ec185e7758b
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)
8 # any later version.
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
18 # 02111-1307, USA.
20 AC_DEFUN([AC_SYS_XSI_STACK_OVERFLOW_HEURISTIC],
21   [# for STACK_DIRECTION
22    AC_REQUIRE([AC_FUNC_ALLOCA])
24    AC_CACHE_CHECK([for working C stack overflow detection],
25      ac_cv_sys_xsi_stack_overflow_heuristic,
26      [AC_TRY_RUN(
27         [
28          #include <signal.h>
29          #include <ucontext.h>
31          static union
32          {
33            char buffer[SIGSTKSZ];
34            long double ld;
35            uintmax_t u;
36            void *p;
37          } alternate_signal_stack;
39          #if STACK_DIRECTION
40          # define find_stack_direction(ptr) STACK_DIRECTION
41          #else
42          static int
43          find_stack_direction (char const *addr)
44          {
45            char dummy;
46            return (! addr ? find_stack_direction (&dummy)
47                    : addr < &dummy ? 1 : -1);
48          }
49          #endif
51          static void
52          segv_handler (int signo, siginfo_t *info, void *context)
53          {
54            if (0 < info->si_code)
55              {
56                ucontext_t const *user_context = context;
57                char const *stack_min = user_context->uc_stack.ss_sp;
58                size_t stack_size = user_context->uc_stack.ss_size;
59                char const *faulting_address = info->si_addr;
60                size_t s = faulting_address - stack_min;
61                size_t page_size = sysconf (_SC_PAGESIZE);
62                if (find_stack_direction (0) < 0)
63                  s += page_size;
64                if (s < stack_size + page_size)
65                  _exit (0);
66              }
68            _exit (1);
69          }
71          static int
72          c_stack_action (void)
73          {
74            stack_t st;
75            struct sigaction act;
76            int r;
78            st.ss_flags = 0;
79            st.ss_sp = alternate_signal_stack.buffer;
80            st.ss_size = sizeof alternate_signal_stack.buffer;
81            r = sigaltstack (&st, 0);
82            if (r != 0)
83              return r;
85            sigemptyset (&act.sa_mask);
86            act.sa_flags = SA_NODEFER | SA_ONSTACK | SA_RESETHAND | SA_SIGINFO;
87            act.sa_sigaction = segv_handler;
88            return sigaction (SIGSEGV, &act, 0);
89          }
91          static int
92          recurse (char *p)
93          {
94            char array[500];
95            array[0] = 1;
96            return *p + recurse (array);
97          }
99          int
100          main (void)
101          {
102            c_stack_action ();
103            return recurse ("\1");
104          }
105         ],
106         [ac_cv_sys_xsi_stack_overflow_heuristic=yes],
107         [ac_cv_sys_xsi_stack_overflow_heuristic=no],
108         [ac_cv_sys_xsi_stack_overflow_heuristic=cross-compiling])])
110    if test $ac_cv_sys_xsi_stack_overflow_heuristic = yes; then
111      AC_DEFINE(HAVE_XSI_STACK_OVERFLOW_HEURISTIC, 1,
112        [Define to 1 if extending the stack slightly past the limit causes
113         a SIGSEGV, and an alternate stack can be established with sigaltstack,
114         and the signal handler is passed a context that specifies the
115         run time stack.  This behavior is defined by POSIX 1003.1-2001
116         with the X/Open System Interface (XSI) option
117         and is a standardized way to implement a SEGV-based stack
118         overflow detection heuristic.])
119    fi])
122 AC_DEFUN([jm_PREREQ_C_STACK],
123   [AC_REQUIRE([jm_AC_TYPE_UINTMAX_T])
124    AC_REQUIRE([AC_SYS_XSI_STACK_OVERFLOW_HEURISTIC])
126    # for STACK_DIRECTION
127    AC_REQUIRE([AC_FUNC_ALLOCA])
129    AC_CHECK_HEADERS(unistd.h)
131    AC_CHECK_TYPES([siginfo_t, stack_t], , , [#include <signal.h>])])