Bump to m4 1.4.19
[platform/upstream/m4.git] / m4 / malloc.m4
1 # malloc.m4 serial 27
2 dnl Copyright (C) 2007, 2009-2021 Free Software Foundation, Inc.
3 dnl This file is free software; the Free Software Foundation
4 dnl gives unlimited permission to copy and/or distribute it,
5 dnl with or without modifications, as long as this notice is preserved.
6
7 # This is adapted with modifications from upstream Autoconf here:
8 # https://git.savannah.gnu.org/cgit/autoconf.git/tree/lib/autoconf/functions.m4?id=v2.70#n949
9 AC_DEFUN([_AC_FUNC_MALLOC_IF],
10 [
11   AC_REQUIRE([AC_CANONICAL_HOST])dnl for cross-compiles
12   AC_CACHE_CHECK([whether malloc (0) returns nonnull],
13     [ac_cv_func_malloc_0_nonnull],
14     [AC_RUN_IFELSE(
15        [AC_LANG_PROGRAM(
16           [[#include <stdlib.h>
17           ]],
18           [[void *p = malloc (0);
19             int result = !p;
20             free (p);
21             return result;]])
22        ],
23        [ac_cv_func_malloc_0_nonnull=yes],
24        [ac_cv_func_malloc_0_nonnull=no],
25        [case "$host_os" in
26           # Guess yes on platforms where we know the result.
27           *-gnu* | freebsd* | netbsd* | openbsd* | bitrig* \
28           | gnu* | *-musl* | midnightbsd* \
29           | hpux* | solaris* | cygwin* | mingw* | msys* )
30             ac_cv_func_malloc_0_nonnull="guessing yes" ;;
31           # If we don't know, obey --enable-cross-guesses.
32           *) ac_cv_func_malloc_0_nonnull="$gl_cross_guess_normal" ;;
33         esac
34        ])
35     ])
36   AS_CASE([$ac_cv_func_malloc_0_nonnull], [*yes], [$1], [$2])
37 ])# _AC_FUNC_MALLOC_IF
38
39 # gl_FUNC_MALLOC_GNU
40 # ------------------
41 # Replace malloc if it is not compatible with GNU libc.
42 AC_DEFUN([gl_FUNC_MALLOC_GNU],
43 [
44   AC_REQUIRE([gl_STDLIB_H_DEFAULTS])
45   AC_REQUIRE([gl_FUNC_MALLOC_POSIX])
46   if test $REPLACE_MALLOC = 0; then
47     _AC_FUNC_MALLOC_IF([], [REPLACE_MALLOC=1])
48   fi
49 ])
50
51 # gl_FUNC_MALLOC_PTRDIFF
52 # ----------------------
53 # Test whether malloc (N) reliably fails when N exceeds PTRDIFF_MAX,
54 # and replace malloc otherwise.
55 AC_DEFUN([gl_FUNC_MALLOC_PTRDIFF],
56 [
57   AC_REQUIRE([gl_STDLIB_H_DEFAULTS])
58   AC_REQUIRE([gl_CHECK_MALLOC_PTRDIFF])
59   test "$gl_cv_malloc_ptrdiff" = yes || REPLACE_MALLOC=1
60 ])
61
62 # Test whether malloc, realloc, calloc refuse to create objects
63 # larger than what can be expressed in ptrdiff_t.
64 # Set gl_cv_func_malloc_gnu to yes or no accordingly.
65 AC_DEFUN([gl_CHECK_MALLOC_PTRDIFF],
66 [
67   AC_CACHE_CHECK([whether malloc is ptrdiff_t safe],
68     [gl_cv_malloc_ptrdiff],
69     [AC_COMPILE_IFELSE(
70        [AC_LANG_PROGRAM(
71           [[#include <stdint.h>
72           ]],
73           [[/* 64-bit ptrdiff_t is so wide that no practical platform
74                can exceed it.  */
75             #define WIDE_PTRDIFF (PTRDIFF_MAX >> 31 >> 31 != 0)
76
77             /* On rare machines where size_t fits in ptrdiff_t there
78                is no problem.  */
79             #define NARROW_SIZE (SIZE_MAX <= PTRDIFF_MAX)
80
81             /* glibc 2.30 and later malloc refuses to exceed ptrdiff_t
82                bounds even on 32-bit platforms.  We don't know which
83                non-glibc systems are safe.  */
84             #define KNOWN_SAFE (2 < __GLIBC__ + (30 <= __GLIBC_MINOR__))
85
86             #if WIDE_PTRDIFF || NARROW_SIZE || KNOWN_SAFE
87               return 0;
88             #else
89               #error "malloc might not be ptrdiff_t safe"
90               syntax error
91             #endif
92           ]])],
93        [gl_cv_malloc_ptrdiff=yes],
94        [gl_cv_malloc_ptrdiff=no])
95     ])
96 ])
97
98 # gl_FUNC_MALLOC_POSIX
99 # --------------------
100 # Test whether 'malloc' is POSIX compliant (sets errno to ENOMEM when it
101 # fails, and doesn't mess up with ptrdiff_t overflow), and replace
102 # malloc if it is not.
103 AC_DEFUN([gl_FUNC_MALLOC_POSIX],
104 [
105   AC_REQUIRE([gl_STDLIB_H_DEFAULTS])
106   AC_REQUIRE([gl_FUNC_MALLOC_PTRDIFF])
107   AC_REQUIRE([gl_CHECK_MALLOC_POSIX])
108   if test "$gl_cv_func_malloc_posix" = yes; then
109     AC_DEFINE([HAVE_MALLOC_POSIX], [1],
110       [Define if malloc, realloc, and calloc set errno on allocation failure.])
111   else
112     REPLACE_MALLOC=1
113   fi
114 ])
115
116 # Test whether malloc, realloc, calloc set errno to ENOMEM on failure.
117 # Set gl_cv_func_malloc_posix to yes or no accordingly.
118 AC_DEFUN([gl_CHECK_MALLOC_POSIX],
119 [
120   AC_REQUIRE([AC_CANONICAL_HOST])
121   AC_CACHE_CHECK([whether malloc, realloc, calloc set errno on failure],
122     [gl_cv_func_malloc_posix],
123     [
124       dnl It is too dangerous to try to allocate a large amount of memory:
125       dnl some systems go to their knees when you do that. So assume that
126       dnl all Unix implementations of the function set errno on failure,
127       dnl except on those platforms where we have seen 'test-malloc-gnu',
128       dnl 'test-realloc-gnu', 'test-calloc-gnu' fail.
129       case "$host_os" in
130         mingw*)
131           gl_cv_func_malloc_posix=no ;;
132         irix* | solaris*)
133           dnl On IRIX 6.5, the three functions return NULL with errno unset
134           dnl when the argument is larger than PTRDIFF_MAX.
135           dnl On Solaris 11.3, the three functions return NULL with errno set
136           dnl to EAGAIN, not ENOMEM, when the argument is larger than
137           dnl PTRDIFF_MAX.
138           dnl Here is a test program:
139 m4_divert_push([KILL])
140 #include <errno.h>
141 #include <stdio.h>
142 #include <stdlib.h>
143 #define ptrdiff_t long
144 #ifndef PTRDIFF_MAX
145 # define PTRDIFF_MAX ((ptrdiff_t) ((1UL << (8 * sizeof (ptrdiff_t) - 1)) - 1))
146 #endif
147
148 int main ()
149 {
150   void *p;
151
152   fprintf (stderr, "PTRDIFF_MAX = %lu\n", (unsigned long) PTRDIFF_MAX);
153
154   errno = 0;
155   p = malloc ((unsigned long) PTRDIFF_MAX + 1);
156   fprintf (stderr, "p=%p errno=%d\n", p, errno);
157
158   errno = 0;
159   p = calloc (PTRDIFF_MAX / 2 + 1, 2);
160   fprintf (stderr, "p=%p errno=%d\n", p, errno);
161
162   errno = 0;
163   p = realloc (NULL, (unsigned long) PTRDIFF_MAX + 1);
164   fprintf (stderr, "p=%p errno=%d\n", p, errno);
165
166   return 0;
167 }
168 m4_divert_pop([KILL])
169           gl_cv_func_malloc_posix=no ;;
170         *)
171           gl_cv_func_malloc_posix=yes ;;
172       esac
173     ])
174 ])