1570af302Sopenharmony_ci// commit 12e1e324683a1d381b7f15dd36c99b37dd44d940 2015-04-10
2570af302Sopenharmony_ci// robust mutex should work with detached threads too
3570af302Sopenharmony_ci#include <pthread.h>
4570af302Sopenharmony_ci#include <string.h>
5570af302Sopenharmony_ci#include <errno.h>
6570af302Sopenharmony_ci#include <time.h>
7570af302Sopenharmony_ci#include "test.h"
8570af302Sopenharmony_ci
9570af302Sopenharmony_ci#define TX(r,f,x) ( ((r)=(f))==x || \
10570af302Sopenharmony_ci (t_error(#f" failed: (pshared==%d) got %d \"%s\" want %d \"%s\"\n", pshared, r, strerror(r), x, strerror(x)), 0) )
11570af302Sopenharmony_ci#define T(r,f) TX(r,f,0)
12570af302Sopenharmony_ci
13570af302Sopenharmony_cistatic pthread_barrier_t barrier2;
14570af302Sopenharmony_cistatic int pshared;
15570af302Sopenharmony_ci
16570af302Sopenharmony_cistatic void *start_lock(void *arg)
17570af302Sopenharmony_ci{
18570af302Sopenharmony_ci	pthread_mutex_lock(arg);
19570af302Sopenharmony_ci	pthread_barrier_wait(&barrier2);
20570af302Sopenharmony_ci	return 0;
21570af302Sopenharmony_ci}
22570af302Sopenharmony_ci
23570af302Sopenharmony_cistatic void f()
24570af302Sopenharmony_ci{
25570af302Sopenharmony_ci	pthread_t td;
26570af302Sopenharmony_ci	int r;
27570af302Sopenharmony_ci	pthread_mutexattr_t mtx_a;
28570af302Sopenharmony_ci	pthread_mutex_t mtx;
29570af302Sopenharmony_ci	struct timespec ts;
30570af302Sopenharmony_ci
31570af302Sopenharmony_ci	T(r, pthread_barrier_init(&barrier2, 0, 2));
32570af302Sopenharmony_ci	T(r, pthread_mutexattr_init(&mtx_a));
33570af302Sopenharmony_ci	T(r, pthread_mutexattr_setrobust(&mtx_a, PTHREAD_MUTEX_ROBUST));
34570af302Sopenharmony_ci	if (pshared)
35570af302Sopenharmony_ci		T(r, pthread_mutexattr_setpshared(&mtx_a, PTHREAD_PROCESS_SHARED));
36570af302Sopenharmony_ci	T(r, pthread_mutex_init(&mtx, &mtx_a));
37570af302Sopenharmony_ci	T(r, pthread_create(&td, 0, start_lock, &mtx));
38570af302Sopenharmony_ci	T(r, pthread_detach(td));
39570af302Sopenharmony_ci	pthread_barrier_wait(&barrier2);
40570af302Sopenharmony_ci	pthread_barrier_destroy(&barrier2);
41570af302Sopenharmony_ci
42570af302Sopenharmony_ci	// enough time to ensure that the detached thread is dead
43570af302Sopenharmony_ci	clock_gettime(CLOCK_REALTIME, &ts);
44570af302Sopenharmony_ci	ts.tv_nsec += 100*1000*1000;
45570af302Sopenharmony_ci	if (ts.tv_nsec >= 1000*1000*1000) {
46570af302Sopenharmony_ci		ts.tv_sec++;
47570af302Sopenharmony_ci		ts.tv_nsec -= 1000*1000*1000;
48570af302Sopenharmony_ci	}
49570af302Sopenharmony_ci
50570af302Sopenharmony_ci	TX(r, pthread_mutex_timedlock(&mtx, &ts), EOWNERDEAD);
51570af302Sopenharmony_ci}
52570af302Sopenharmony_ci
53570af302Sopenharmony_ciint main(void)
54570af302Sopenharmony_ci{
55570af302Sopenharmony_ci	// test non-pshared and pshared robust mutexes as well
56570af302Sopenharmony_ci	return 0;
57570af302Sopenharmony_ci}
58