# The test program for the next two tests is the same except for one # set of ifdefs. cat >ct-mmap.inc <<'EOF' #include #include #include #include #include #include #if !defined (MAP_ANONYMOUS) && defined (MAP_ANON) # define MAP_ANONYMOUS MAP_ANON #endif /* This mess was copied from the GNU getpagesize.h. */ #ifndef HAVE_GETPAGESIZE # ifdef HAVE_UNISTD_H # include # endif /* Assume that all systems that can run configure have sys/param.h. */ # ifndef HAVE_SYS_PARAM_H # define HAVE_SYS_PARAM_H 1 # endif # ifdef _SC_PAGESIZE # define getpagesize() sysconf(_SC_PAGESIZE) # else /* no _SC_PAGESIZE */ # ifdef HAVE_SYS_PARAM_H # include # ifdef EXEC_PAGESIZE # define getpagesize() EXEC_PAGESIZE # else /* no EXEC_PAGESIZE */ # ifdef NBPG # define getpagesize() NBPG * CLSIZE # ifndef CLSIZE # define CLSIZE 1 # endif /* no CLSIZE */ # else /* no NBPG */ # ifdef NBPC # define getpagesize() NBPC # else /* no NBPC */ # ifdef PAGESIZE # define getpagesize() PAGESIZE # endif /* PAGESIZE */ # endif /* no NBPC */ # endif /* no NBPG */ # endif /* no EXEC_PAGESIZE */ # else /* no HAVE_SYS_PARAM_H */ # define getpagesize() 8192 /* punt totally */ # endif /* no HAVE_SYS_PARAM_H */ # endif /* no _SC_PAGESIZE */ #endif /* no HAVE_GETPAGESIZE */ #ifndef MAP_FAILED # define MAP_FAILED -1 #endif #undef perror_exit #define perror_exit(str, val) \ do { perror(str); exit(val); } while (0) /* Some versions of cygwin mmap require that munmap is called with the same parameters as mmap. GCC expects that this is not the case. Test for various forms of this problem. Warning - icky signal games. */ static sigset_t unblock_sigsegv; static jmp_buf r; static size_t pg; static int devzero; static char * anonmap (size) size_t size; { #ifdef USE_MAP_ANON return (char *) mmap (0, size, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0); #else return (char *) mmap (0, size, PROT_READ|PROT_WRITE, MAP_PRIVATE, devzero, 0); #endif } static void sigsegv (unused) int unused; { sigprocmask (SIG_UNBLOCK, &unblock_sigsegv, 0); longjmp (r, 1); } /* Basic functionality test. */ void test_0 () { char *x = anonmap (pg); if (x == (char *) MAP_FAILED) perror_exit("test 0 mmap", 2); *(int *)x += 1; if (munmap(x, pg) < 0) perror_exit("test 0 munmap", 3); } /* 1. If we map a 2-page region and unmap its second page, the first page must remain. */ static void test_1 () { char *x = anonmap (pg * 2); if (x == (char *)MAP_FAILED) perror_exit ("test 1 mmap", 4); signal (SIGSEGV, sigsegv); if (setjmp (r)) perror_exit ("test 1 fault", 5); x[0] = 1; x[pg] = 1; if (munmap (x + pg, pg) < 0) perror_exit ("test 1 munmap 1", 6); x[0] = 2; if (setjmp (r) == 0) { x[pg] = 1; perror_exit ("test 1 no fault", 7); } if (munmap (x, pg) < 0) perror_exit ("test 1 munmap 2", 8); } /* 2. If we map a 2-page region and unmap its first page, the second page must remain. */ static void test_2 () { char *x = anonmap (pg * 2); if (x == (char *)MAP_FAILED) perror_exit ("test 2 mmap", 9); signal (SIGSEGV, sigsegv); if (setjmp (r)) perror_exit ("test 2 fault", 10); x[0] = 1; x[pg] = 1; if (munmap (x, pg) < 0) perror_exit ("test 2 munmap 1", 11); x[pg] = 2; if (setjmp (r) == 0) { x[0] = 1; perror_exit ("test 2 no fault", 12); } if (munmap (x+pg, pg) < 0) perror_exit ("test 2 munmap 2", 13); } /* 3. If we map two adjacent 1-page regions and unmap them both with one munmap, both must go away. Getting two adjacent 1-page regions with two mmap calls is slightly tricky. All OS's tested skip over already-allocated blocks; therefore we have been careful to unmap all allocated regions in previous tests. HP/UX allocates pages backward in memory. No OS has yet been observed to be so perverse as to leave unmapped space between consecutive calls to mmap. */ static void test_3 () { char *x, *y, *z; x = anonmap (pg); if (x == (char *)MAP_FAILED) perror_exit ("test 3 mmap 1", 14); y = anonmap (pg); if (y == (char *)MAP_FAILED) perror_exit ("test 3 mmap 2", 15); if (y != x + pg) { if (y == x - pg) z = y, y = x, x = z; else { fprintf (stderr, "test 3 nonconsecutive pages - %lx, %lx\n", (unsigned long)x, (unsigned long)y); exit (16); } } signal (SIGSEGV, sigsegv); if (setjmp (r)) perror_exit ("test 3 fault", 17); x[0] = 1; y[0] = 1; if (munmap (x, pg*2) < 0) perror_exit ("test 3 munmap", 18); if (setjmp (r) == 0) { x[0] = 1; perror_exit ("test 3 no fault 1", 19); } signal (SIGSEGV, sigsegv); if (setjmp (r) == 0) { y[0] = 1; perror_exit ("test 3 no fault 2", 20); } } int main () { sigemptyset (&unblock_sigsegv); sigaddset (&unblock_sigsegv, SIGSEGV); pg = getpagesize (); #ifndef USE_MAP_ANON devzero = open ("/dev/zero", O_RDWR); if (devzero < 0) perror_exit ("open /dev/zero", 1); #endif test_0(); test_1(); test_2(); test_3(); exit(0); } EOF echo $ac_n "checking for working mmap from /dev/zero""... $ac_c" 1>&6 echo "configure:4276: checking for working mmap from /dev/zero" >&5 if eval "test \"`echo '$''{'ac_cv_func_mmap_dev_zero'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else if test "$cross_compiling" = yes; then # If this is not cygwin, and /dev/zero is a character device, it's probably # safe to assume it works. case "$host_os" in cygwin* | win32 | pe | mingw* ) ac_cv_func_mmap_dev_zero=buggy ;; * ) if test -c /dev/zero then ac_cv_func_mmap_dev_zero=yes else ac_cv_func_mmap_dev_zero=no fi ;; esac else cat > conftest.$ac_ext <&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext} && (./conftest; exit) 2>/dev/null then ac_cv_func_mmap_dev_zero=yes else echo "configure: failed program was:" >&5 cat conftest.$ac_ext >&5 rm -fr conftest* if test $? -lt 4 then ac_cv_func_mmap_dev_zero=no else ac_cv_func_mmap_dev_zero=buggy fi fi rm -fr conftest* fi fi echo "$ac_t""$ac_cv_func_mmap_dev_zero" 1>&6 if test $ac_cv_func_mmap_dev_zero = yes; then cat >> confdefs.h <<\EOF #define HAVE_MMAP_DEV_ZERO 1 EOF fi echo $ac_n "checking for working mmap with MAP_ANON(YMOUS)""... $ac_c" 1>&6 echo "configure:4323: checking for working mmap with MAP_ANON(YMOUS)" >&5 if eval "test \"`echo '$''{'ac_cv_func_mmap_anon'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else if test "$cross_compiling" = yes; then # Unlike /dev/zero, it is not safe to assume MAP_ANON(YMOUS) works # just because it's there. Some SCO Un*xen define it but don't implement it. ac_cv_func_mmap_anon=no else cat > conftest.$ac_ext <&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext} && (./conftest; exit) 2>/dev/null then ac_cv_func_mmap_anon=yes else echo "configure: failed program was:" >&5 cat conftest.$ac_ext >&5 rm -fr conftest* if test $? -lt 4 then ac_cv_func_mmap_anon=no else ac_cv_func_mmap_anon=buggy fi fi rm -fr conftest* fi fi