]> rtime.felk.cvut.cz Git - l4.git/blob - l4/pkg/uclibc/lib/contrib/uclibc/test/nptl/tst-mutex9.c
update
[l4.git] / l4 / pkg / uclibc / lib / contrib / uclibc / test / nptl / tst-mutex9.c
1 /* Copyright (C) 2003, 2004 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
3    Contributed by Ulrich Drepper <drepper@redhat.com>, 2003.
4
5    The GNU C Library is free software; you can redistribute it and/or
6    modify it under the terms of the GNU Lesser General Public
7    License as published by the Free Software Foundation; either
8    version 2.1 of the License, or (at your option) any later version.
9
10    The GNU C Library 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 GNU
13    Lesser General Public License for more details.
14
15    You should have received a copy of the GNU Lesser General Public
16    License along with the GNU C Library; if not, write to the Free
17    Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
18    02111-1307 USA.  */
19
20 #include <errno.h>
21 #include <pthread.h>
22 #include <stdio.h>
23 #include <string.h>
24 #include <unistd.h>
25 #include <sys/mman.h>
26 #include <sys/wait.h>
27 #include <stdint.h>
28 #include <stdlib.h>
29 #include <sys/time.h>
30
31 int gettimeofday(struct timeval *tv, struct timezone *tz);
32
33
34 static int
35 do_test (void)
36 {
37   size_t ps = sysconf (_SC_PAGESIZE);
38   char tmpfname[] = "/tmp/tst-mutex9.XXXXXX";
39   char data[ps];
40   void *mem;
41   int fd;
42   pthread_mutex_t *m;
43   pthread_mutexattr_t a;
44   pid_t pid;
45   char *p;
46
47   fd = mkstemp (tmpfname);
48   if (fd == -1)
49     {
50       printf ("cannot open temporary file: %m\n");
51       return 1;
52     }
53
54   /* Make sure it is always removed.  */
55   unlink (tmpfname);
56
57   /* Create one page of data.  */
58   memset (data, '\0', ps);
59
60   /* Write the data to the file.  */
61   if (write (fd, data, ps) != (ssize_t) ps)
62     {
63       puts ("short write");
64       return 1;
65     }
66
67   mem = mmap (NULL, ps, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
68   if (mem == MAP_FAILED)
69     {
70       printf ("mmap failed: %m\n");
71       return 1;
72     }
73
74   m = (pthread_mutex_t *) (((uintptr_t) mem + __alignof (pthread_mutex_t))
75                            & ~(__alignof (pthread_mutex_t) - 1));
76   p = (char *) (m + 1);
77
78   if (pthread_mutexattr_init (&a) != 0)
79     {
80       puts ("mutexattr_init failed");
81       return 1;
82     }
83
84   if (pthread_mutexattr_setpshared (&a, PTHREAD_PROCESS_SHARED) != 0)
85     {
86       puts ("mutexattr_setpshared failed");
87       return 1;
88     }
89
90   if (pthread_mutexattr_settype (&a, PTHREAD_MUTEX_RECURSIVE) != 0)
91     {
92       puts ("mutexattr_settype failed");
93       return 1;
94     }
95
96   if (pthread_mutex_init (m, &a) != 0)
97     {
98       puts ("mutex_init failed");
99       return 1;
100     }
101
102   if (pthread_mutex_lock (m) != 0)
103     {
104       puts ("mutex_lock failed");
105       return 1;
106     }
107
108   if (pthread_mutexattr_destroy (&a) != 0)
109     {
110       puts ("mutexattr_destroy failed");
111       return 1;
112     }
113
114   puts ("going to fork now");
115   pid = fork ();
116   if (pid == -1)
117     {
118       puts ("fork failed");
119       return 1;
120     }
121   else if (pid == 0)
122     {
123       if (pthread_mutex_trylock (m) == 0)
124         {
125           puts ("child: mutex_trylock succeeded");
126           exit (1);
127         }
128
129       if (pthread_mutex_unlock (m) == 0)
130         {
131           puts ("child: mutex_unlock succeeded");
132           exit (1);
133         }
134
135       struct timeval tv;
136       gettimeofday (&tv, NULL);
137       struct timespec ts;
138       TIMEVAL_TO_TIMESPEC (&tv, &ts);
139       ts.tv_nsec += 500000000;
140       if (ts.tv_nsec >= 1000000000)
141         {
142           ++ts.tv_sec;
143           ts.tv_nsec -= 1000000000;
144         }
145
146       int e = pthread_mutex_timedlock (m, &ts);
147       if (e == 0)
148         {
149           puts ("child: mutex_timedlock succeeded");
150           exit (1);
151         }
152       if (e != ETIMEDOUT)
153         {
154           puts ("child: mutex_timedlock didn't time out");
155           exit (1);
156         }
157
158       alarm (1);
159
160       pthread_mutex_lock (m);
161
162       puts ("child: mutex_lock returned");
163
164       exit (0);
165     }
166
167   sleep (2);
168
169   int status;
170   if (TEMP_FAILURE_RETRY (waitpid (pid, &status, 0)) != pid)
171     {
172       puts ("waitpid failed");
173       return 1;
174     }
175   if (! WIFSIGNALED (status))
176     {
177       puts ("child not killed by signal");
178       return 1;
179     }
180   if (WTERMSIG (status) != SIGALRM)
181     {
182       puts ("child not killed by SIGALRM");
183       return 1;
184     }
185
186   return 0;
187 }
188
189 #define TIMEOUT 3
190 #define TEST_FUNCTION do_test ()
191 #include "../test-skeleton.c"