1 /* Copyright (C) 2003-2023 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
9 The GNU C Library 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 GNU
12 Lesser General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, see
16 <https://www.gnu.org/licenses/>. */
28 #include <support/check.h>
29 #include <support/timespec.h>
30 #include <support/xunistd.h>
31 #include <support/xthread.h>
38 /* A bogus clock value that tells run_test to use pthread_mutex_timedlock
39 rather than pthread_mutex_clocklock. */
40 #define CLOCK_USE_TIMEDLOCK (-1)
43 do_test_clock (clockid_t clockid
, int tmo_result
)
45 const clockid_t clockid_for_get
=
46 (clockid
== CLOCK_USE_TIMEDLOCK
) ? CLOCK_REALTIME
: clockid
;
47 size_t ps
= sysconf (_SC_PAGESIZE
);
48 char tmpfname
[] = "/tmp/tst-mutex9.XXXXXX";
53 pthread_mutexattr_t a
;
56 fd
= mkstemp (tmpfname
);
58 FAIL_EXIT1 ("cannot open temporary file: %m\n");
60 /* Make sure it is always removed. */
63 /* Create one page of data. */
64 memset (data
, '\0', ps
);
66 /* Write the data to the file. */
67 xwrite (fd
, data
, ps
);
69 mem
= xmmap (NULL
, ps
, PROT_READ
| PROT_WRITE
, MAP_SHARED
, fd
);
71 m
= (pthread_mutex_t
*) (((uintptr_t) mem
+ __alignof (pthread_mutex_t
))
72 & ~(__alignof (pthread_mutex_t
) - 1));
74 TEST_COMPARE (pthread_mutexattr_init (&a
), 0);
76 TEST_COMPARE (pthread_mutexattr_setpshared (&a
, PTHREAD_PROCESS_SHARED
), 0);
78 TEST_COMPARE (pthread_mutexattr_settype (&a
, PTHREAD_MUTEX_RECURSIVE
), 0);
81 TEST_COMPARE (pthread_mutexattr_setprotocol (&a
, PTHREAD_PRIO_INHERIT
), 0);
82 #elif defined ENABLE_PP
83 TEST_COMPARE (pthread_mutexattr_setprotocol (&a
, PTHREAD_PRIO_PROTECT
), 0);
84 TEST_COMPARE (pthread_mutexattr_setprioceiling (&a
, 6), 0);
88 if ((e
= pthread_mutex_init (m
, &a
)) != 0)
92 FAIL_UNSUPPORTED ("PI mutexes unsupported");
94 FAIL_EXIT1 ("mutex_init failed");
97 TEST_COMPARE (pthread_mutex_lock (m
), 0);
99 TEST_COMPARE (pthread_mutexattr_destroy (&a
), 0);
101 puts ("going to fork now");
105 if (pthread_mutex_trylock (m
) == 0)
106 FAIL_EXIT1 ("child: mutex_trylock succeeded");
108 if (pthread_mutex_unlock (m
) == 0)
109 FAIL_EXIT1 ("child: mutex_unlock succeeded");
111 const struct timespec ts
= timespec_add (xclock_now (clockid_for_get
),
112 make_timespec (0, 500000000));
114 if (clockid
== CLOCK_USE_TIMEDLOCK
)
115 TEST_COMPARE (pthread_mutex_timedlock (m
, &ts
), tmo_result
);
117 TEST_COMPARE (pthread_mutex_clocklock (m
, clockid
, &ts
), tmo_result
);
121 pthread_mutex_lock (m
);
123 puts ("child: mutex_lock returned");
131 if (TEMP_FAILURE_RETRY (waitpid (pid
, &status
, 0)) != pid
)
132 FAIL_EXIT1 ("waitpid failed");
133 if (! WIFSIGNALED (status
))
134 FAIL_EXIT1 ("child not killed by signal");
135 TEST_COMPARE (WTERMSIG (status
), SIGALRM
);
145 int monotonic_result
=
147 support_mutex_pi_monotonic () ? ETIMEDOUT
: EINVAL
;
152 do_test_clock (CLOCK_USE_TIMEDLOCK
, ETIMEDOUT
);
153 do_test_clock (CLOCK_REALTIME
, ETIMEDOUT
);
154 do_test_clock (CLOCK_MONOTONIC
, monotonic_result
);
158 #include <support/test-driver.c>