spawn: Use special invocation for <spawn.h> on OS/2 kLIBC.
[gnulib.git] / tests / zerosize-ptr.h
blob45a1729d3e57c418cc850c8a8ce68720b01d0bde
1 /* Return a pointer to a zero-size object in memory.
2 Copyright (C) 2009-2021 Free Software Foundation, Inc.
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 3 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. If not, see <https://www.gnu.org/licenses/>. */
17 /* ISO C 99 does not allow memcmp(), memchr() etc. to be invoked with a NULL
18 argument. Therefore this file produces a non-NULL pointer which cannot
19 be dereferenced, if possible. */
21 /* On Android, when targeting Android 4.4 or older with a GCC toolchain,
22 prevent a compilation error
23 "error: call to 'mmap' declared with attribute error: mmap is not
24 available with _FILE_OFFSET_BITS=64 when using GCC until android-21.
25 Either raise your minSdkVersion, disable _FILE_OFFSET_BITS=64, or
26 switch to Clang."
27 The files that we access in this compilation unit are less than 2 GB
28 large. */
29 #if defined __ANDROID__
30 # undef _FILE_OFFSET_BITS
31 # undef __USE_FILE_OFFSET64
32 #endif
34 #include <stdlib.h>
36 /* Test whether mmap() and mprotect() are available.
37 We don't use HAVE_MMAP, because AC_FUNC_MMAP would not define it on HP-UX.
38 HAVE_MPROTECT is not enough, because mingw does not have mmap() but has an
39 mprotect() function in libgcc.a. */
40 #if HAVE_SYS_MMAN_H && HAVE_MPROTECT
41 # include <fcntl.h>
42 # include <unistd.h>
43 # include <sys/types.h>
44 # include <sys/mman.h>
45 /* Define MAP_FILE when it isn't otherwise. */
46 # ifndef MAP_FILE
47 # define MAP_FILE 0
48 # endif
49 #endif
51 /* Return a pointer to a zero-size object in memory (that is, actually, a
52 pointer to a page boundary where the previous page is readable and writable
53 and the next page is neither readable not writable), if possible.
54 Return NULL otherwise. */
56 static void *
57 zerosize_ptr (void)
59 /* Use mmap and mprotect when they exist. Don't test HAVE_MMAP, because it is
60 not defined on HP-UX 11 (since it does not support MAP_FIXED). */
61 #if HAVE_SYS_MMAN_H && HAVE_MPROTECT
62 # if HAVE_MAP_ANONYMOUS
63 const int flags = MAP_ANONYMOUS | MAP_PRIVATE;
64 const int fd = -1;
65 # else /* !HAVE_MAP_ANONYMOUS */
66 const int flags = MAP_FILE | MAP_PRIVATE;
67 int fd = open ("/dev/zero", O_RDONLY, 0666);
68 if (fd >= 0)
69 # endif
71 int pagesize = getpagesize ();
72 char *two_pages =
73 (char *) mmap (NULL, 2 * pagesize, PROT_READ | PROT_WRITE,
74 flags, fd, 0);
75 if (two_pages != (char *)(-1)
76 && mprotect (two_pages + pagesize, pagesize, PROT_NONE) == 0)
77 return two_pages + pagesize;
79 #endif
80 return NULL;