* winsup.api/checksignal.c: Ditto. * winsup.api/crlf.c: Ditto. * winsup.api/devzero.c: Ditto. * winsup.api/iospeed.c: Ditto. * winsup.api/mmaptest01.c: Ditto. * winsup.api/mmaptest02.c: Ditto. * winsup.api/mmaptest03.c: Ditto. * winsup.api/mmaptest04.c: Ditto. * winsup.api/nullgetcwd.c: Ditto. * winsup.api/sigchld.c: Ditto. * winsup.api/signal-into-win32-api.c: Ditto. * winsup.api/systemcall.c: Ditto. * winsup.api/waitpid.c: Ditto. * winsup.api/pthread/mainthreadexits.c: Ditto. * winsup.api/pthread/test.h: Ditto. * winsup.api/pthread/threadidafterfork.c: Ditto. * Makefile.in: Remove cygrun.exe from RUNTIME since it is built here now.
		
			
				
	
	
		
			191 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			191 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include <sys/types.h>
 | |
| #include <sys/mman.h>
 | |
| #include <unistd.h>
 | |
| #include <fcntl.h>
 | |
| #include <signal.h>
 | |
| #include <setjmp.h>
 | |
| #include <stdlib.h>
 | |
| #include <stdio.h>
 | |
| #include <errno.h>
 | |
| #include <string.h>
 | |
| 
 | |
| sigset_t unblock_sigsegv;
 | |
| jmp_buf r;
 | |
| size_t pg;
 | |
| 
 | |
| int fd;
 | |
| 
 | |
| /* Checks behaviour of anonymous mmap.
 | |
| 
 | |
|    test_1: If we map a 2-page region and unmap its second page, the first page
 | |
|    must remain.
 | |
| 
 | |
|    test_2: If we map a 2-page region and unmap its first page, the second
 | |
|    page must remain.
 | |
| 
 | |
|    test_3: If we map two consecutive 1-page regions and unmap them both with
 | |
|    one munmap, both must go away.
 | |
| */
 | |
| 
 | |
| void
 | |
| perror_exit (const char *str, int code)
 | |
| {
 | |
|   printf ("%s: %s\n", str, strerror (errno));
 | |
|   exit (code);
 | |
| }
 | |
| 
 | |
| void
 | |
| anonmap_init ()
 | |
| {
 | |
|   sigemptyset (&unblock_sigsegv);
 | |
|   sigaddset (&unblock_sigsegv, SIGSEGV);
 | |
|   pg = getpagesize ();
 | |
|   fd = open ("/dev/zero", O_RDWR);
 | |
| }
 | |
| 
 | |
| char *
 | |
| anonmap (size_t size)
 | |
| {
 | |
|   return (char *) mmap (0, size, PROT_READ|PROT_WRITE, MAP_PRIVATE, fd, 0);
 | |
| }
 | |
| 
 | |
| void
 | |
| anonfree (char *loc, size_t size)
 | |
| {
 | |
|   munmap (loc, size);
 | |
| }
 | |
|      
 | |
| void
 | |
| sigsegv (int unused)
 | |
| {
 | |
|   sigprocmask (SIG_UNBLOCK, &unblock_sigsegv, 0);
 | |
|   longjmp (r, 1);
 | |
| }
 | |
| 
 | |
| int
 | |
| compare_pointers (const void *a, const void *b)
 | |
| {
 | |
|   const char *x = *(const char *const *)a;
 | |
|   const char *y = *(const char *const *)b;
 | |
| 
 | |
|   if (x > y)
 | |
|     return 1;
 | |
|   if (x < y)
 | |
|     return -1;
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| void
 | |
| test_1 ()
 | |
| {
 | |
|   char *x = anonmap (pg * 2);
 | |
|   if (x == (char *)MAP_FAILED)
 | |
|     perror_exit ("test 1 mmap", 1);
 | |
| 
 | |
|   signal (SIGSEGV, sigsegv);
 | |
|   if (setjmp (r))
 | |
|     perror_exit ("test 1 fault", 2);
 | |
| 
 | |
|   x[0] = 1;
 | |
|   x[pg] = 1;
 | |
| 
 | |
|   anonfree (x + pg, pg);
 | |
|   x[0] = 2;
 | |
| 
 | |
|   if (setjmp (r) == 0)
 | |
|     {
 | |
|       x[pg] = 1;
 | |
|       perror_exit ("test 1 no fault", 3);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void
 | |
| test_2 ()
 | |
| {
 | |
|   char *x = anonmap (pg * 2);
 | |
|   if (x == (char *)MAP_FAILED)
 | |
|     perror_exit ("test 2 mmap", 4);
 | |
| 
 | |
|   signal (SIGSEGV, sigsegv);
 | |
|   if (setjmp (r))
 | |
|     perror_exit ("test 2 fault", 5);
 | |
| 
 | |
|   x[0] = 1;
 | |
|   x[pg] = 1;
 | |
| 
 | |
|   anonfree (x, pg);
 | |
|   x[pg] = 2;
 | |
| 
 | |
|   if (setjmp (r) == 0)
 | |
|     {
 | |
|       x[0] = 1;
 | |
|       perror_exit ("test 2 no fault", 6);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void
 | |
| test_3 ()
 | |
| {
 | |
|   char *x[10];
 | |
|   char *y;
 | |
|   int i;
 | |
| 
 | |
|   /* There's no way to guarantee we get consecutive pages from the OS.  The
 | |
|      approach taken here is to allocate ten of them, sort the list, and
 | |
|      look for consecutive pages.  */
 | |
|   for (i = 0; i < 10; i++)
 | |
|     {
 | |
|       x[i] = anonmap (pg);
 | |
|       if (x[i] == (char *)MAP_FAILED)
 | |
| 	perror_exit ("test 3 mmap 1", 7);
 | |
|     }
 | |
|   qsort (x, 10, sizeof (char *), compare_pointers);
 | |
| 
 | |
|   y = 0;
 | |
|   for (i = 0; i < 9; i++)
 | |
|     if (x[i] + pg == x[i+1])
 | |
|       {
 | |
| 	y = x[i];
 | |
| 	break;
 | |
|       }
 | |
|   if (y == 0)
 | |
|     {
 | |
|       fputs ("test 3: couldn't get two consecutive pages, giving up\n", stdout);
 | |
|       exit (65);
 | |
|     }
 | |
|   
 | |
|   signal (SIGSEGV, sigsegv);
 | |
|   if (setjmp (r))
 | |
|     perror_exit ("test 3 fault", 8);
 | |
| 
 | |
|   y[0] = 1;
 | |
|   y[pg] = 1;
 | |
| 
 | |
|   anonfree (y, pg * 2);
 | |
| 
 | |
|   if (setjmp (r) == 0)
 | |
|     {
 | |
|       y[0] = 1;
 | |
|       perror_exit ("test 3 no fault 1", 9);
 | |
|     }
 | |
|   
 | |
|   signal (SIGSEGV, sigsegv);
 | |
|   if (setjmp (r) == 0)
 | |
|     {
 | |
|       y[pg] = 1;
 | |
|       perror_exit ("test 3 no fault 2", 10);
 | |
|     }
 | |
| }
 | |
| 
 | |
| int
 | |
| main ()
 | |
| {
 | |
|   anonmap_init();
 | |
| 
 | |
|   test_1();
 | |
|   test_2();
 | |
|   test_3();
 | |
| 
 | |
|   exit(0);
 | |
| }
 |