1*c9945492SAndroid Build Coastguard Worker #include "pthread_impl.h"
2*c9945492SAndroid Build Coastguard Worker #include <semaphore.h>
3*c9945492SAndroid Build Coastguard Worker #include <string.h>
4*c9945492SAndroid Build Coastguard Worker
dummy_0(void)5*c9945492SAndroid Build Coastguard Worker static void dummy_0(void)
6*c9945492SAndroid Build Coastguard Worker {
7*c9945492SAndroid Build Coastguard Worker }
8*c9945492SAndroid Build Coastguard Worker
9*c9945492SAndroid Build Coastguard Worker weak_alias(dummy_0, __tl_lock);
10*c9945492SAndroid Build Coastguard Worker weak_alias(dummy_0, __tl_unlock);
11*c9945492SAndroid Build Coastguard Worker
12*c9945492SAndroid Build Coastguard Worker static int target_tid;
13*c9945492SAndroid Build Coastguard Worker static void (*callback)(void *), *context;
14*c9945492SAndroid Build Coastguard Worker static sem_t target_sem, caller_sem, exit_sem;
15*c9945492SAndroid Build Coastguard Worker
dummy(void * p)16*c9945492SAndroid Build Coastguard Worker static void dummy(void *p)
17*c9945492SAndroid Build Coastguard Worker {
18*c9945492SAndroid Build Coastguard Worker }
19*c9945492SAndroid Build Coastguard Worker
handler(int sig)20*c9945492SAndroid Build Coastguard Worker static void handler(int sig)
21*c9945492SAndroid Build Coastguard Worker {
22*c9945492SAndroid Build Coastguard Worker if (__pthread_self()->tid != target_tid) return;
23*c9945492SAndroid Build Coastguard Worker
24*c9945492SAndroid Build Coastguard Worker int old_errno = errno;
25*c9945492SAndroid Build Coastguard Worker
26*c9945492SAndroid Build Coastguard Worker /* Inform caller we have received signal and wait for
27*c9945492SAndroid Build Coastguard Worker * the caller to let us make the callback. */
28*c9945492SAndroid Build Coastguard Worker sem_post(&caller_sem);
29*c9945492SAndroid Build Coastguard Worker sem_wait(&target_sem);
30*c9945492SAndroid Build Coastguard Worker
31*c9945492SAndroid Build Coastguard Worker callback(context);
32*c9945492SAndroid Build Coastguard Worker
33*c9945492SAndroid Build Coastguard Worker /* Inform caller we've complered the callback and wait
34*c9945492SAndroid Build Coastguard Worker * for the caller to release us to return. */
35*c9945492SAndroid Build Coastguard Worker sem_post(&caller_sem);
36*c9945492SAndroid Build Coastguard Worker sem_wait(&exit_sem);
37*c9945492SAndroid Build Coastguard Worker
38*c9945492SAndroid Build Coastguard Worker /* Inform caller we are returning and state is destroyable. */
39*c9945492SAndroid Build Coastguard Worker sem_post(&caller_sem);
40*c9945492SAndroid Build Coastguard Worker
41*c9945492SAndroid Build Coastguard Worker errno = old_errno;
42*c9945492SAndroid Build Coastguard Worker }
43*c9945492SAndroid Build Coastguard Worker
__synccall(void (* func)(void *),void * ctx)44*c9945492SAndroid Build Coastguard Worker void __synccall(void (*func)(void *), void *ctx)
45*c9945492SAndroid Build Coastguard Worker {
46*c9945492SAndroid Build Coastguard Worker sigset_t oldmask;
47*c9945492SAndroid Build Coastguard Worker int cs, i, r;
48*c9945492SAndroid Build Coastguard Worker struct sigaction sa = { .sa_flags = SA_RESTART | SA_ONSTACK, .sa_handler = handler };
49*c9945492SAndroid Build Coastguard Worker pthread_t self = __pthread_self(), td;
50*c9945492SAndroid Build Coastguard Worker int count = 0;
51*c9945492SAndroid Build Coastguard Worker
52*c9945492SAndroid Build Coastguard Worker /* Blocking signals in two steps, first only app-level signals
53*c9945492SAndroid Build Coastguard Worker * before taking the lock, then all signals after taking the lock,
54*c9945492SAndroid Build Coastguard Worker * is necessary to achieve AS-safety. Blocking them all first would
55*c9945492SAndroid Build Coastguard Worker * deadlock if multiple threads called __synccall. Waiting to block
56*c9945492SAndroid Build Coastguard Worker * any until after the lock would allow re-entry in the same thread
57*c9945492SAndroid Build Coastguard Worker * with the lock already held. */
58*c9945492SAndroid Build Coastguard Worker __block_app_sigs(&oldmask);
59*c9945492SAndroid Build Coastguard Worker __tl_lock();
60*c9945492SAndroid Build Coastguard Worker __block_all_sigs(0);
61*c9945492SAndroid Build Coastguard Worker pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &cs);
62*c9945492SAndroid Build Coastguard Worker
63*c9945492SAndroid Build Coastguard Worker sem_init(&target_sem, 0, 0);
64*c9945492SAndroid Build Coastguard Worker sem_init(&caller_sem, 0, 0);
65*c9945492SAndroid Build Coastguard Worker sem_init(&exit_sem, 0, 0);
66*c9945492SAndroid Build Coastguard Worker
67*c9945492SAndroid Build Coastguard Worker if (!libc.threads_minus_1 || __syscall(SYS_gettid) != self->tid)
68*c9945492SAndroid Build Coastguard Worker goto single_threaded;
69*c9945492SAndroid Build Coastguard Worker
70*c9945492SAndroid Build Coastguard Worker callback = func;
71*c9945492SAndroid Build Coastguard Worker context = ctx;
72*c9945492SAndroid Build Coastguard Worker
73*c9945492SAndroid Build Coastguard Worker /* Block even implementation-internal signals, so that nothing
74*c9945492SAndroid Build Coastguard Worker * interrupts the SIGSYNCCALL handlers. The main possible source
75*c9945492SAndroid Build Coastguard Worker * of trouble is asynchronous cancellation. */
76*c9945492SAndroid Build Coastguard Worker memset(&sa.sa_mask, -1, sizeof sa.sa_mask);
77*c9945492SAndroid Build Coastguard Worker __libc_sigaction(SIGSYNCCALL, &sa, 0);
78*c9945492SAndroid Build Coastguard Worker
79*c9945492SAndroid Build Coastguard Worker
80*c9945492SAndroid Build Coastguard Worker for (td=self->next; td!=self; td=td->next) {
81*c9945492SAndroid Build Coastguard Worker target_tid = td->tid;
82*c9945492SAndroid Build Coastguard Worker while ((r = -__syscall(SYS_tkill, td->tid, SIGSYNCCALL)) == EAGAIN);
83*c9945492SAndroid Build Coastguard Worker if (r) {
84*c9945492SAndroid Build Coastguard Worker /* If we failed to signal any thread, nop out the
85*c9945492SAndroid Build Coastguard Worker * callback to abort the synccall and just release
86*c9945492SAndroid Build Coastguard Worker * any threads already caught. */
87*c9945492SAndroid Build Coastguard Worker callback = func = dummy;
88*c9945492SAndroid Build Coastguard Worker break;
89*c9945492SAndroid Build Coastguard Worker }
90*c9945492SAndroid Build Coastguard Worker sem_wait(&caller_sem);
91*c9945492SAndroid Build Coastguard Worker count++;
92*c9945492SAndroid Build Coastguard Worker }
93*c9945492SAndroid Build Coastguard Worker target_tid = 0;
94*c9945492SAndroid Build Coastguard Worker
95*c9945492SAndroid Build Coastguard Worker /* Serialize execution of callback in caught threads, or just
96*c9945492SAndroid Build Coastguard Worker * release them all if synccall is being aborted. */
97*c9945492SAndroid Build Coastguard Worker for (i=0; i<count; i++) {
98*c9945492SAndroid Build Coastguard Worker sem_post(&target_sem);
99*c9945492SAndroid Build Coastguard Worker sem_wait(&caller_sem);
100*c9945492SAndroid Build Coastguard Worker }
101*c9945492SAndroid Build Coastguard Worker
102*c9945492SAndroid Build Coastguard Worker sa.sa_handler = SIG_IGN;
103*c9945492SAndroid Build Coastguard Worker __libc_sigaction(SIGSYNCCALL, &sa, 0);
104*c9945492SAndroid Build Coastguard Worker
105*c9945492SAndroid Build Coastguard Worker single_threaded:
106*c9945492SAndroid Build Coastguard Worker func(ctx);
107*c9945492SAndroid Build Coastguard Worker
108*c9945492SAndroid Build Coastguard Worker /* Only release the caught threads once all threads, including the
109*c9945492SAndroid Build Coastguard Worker * caller, have returned from the callback function. */
110*c9945492SAndroid Build Coastguard Worker for (i=0; i<count; i++)
111*c9945492SAndroid Build Coastguard Worker sem_post(&exit_sem);
112*c9945492SAndroid Build Coastguard Worker for (i=0; i<count; i++)
113*c9945492SAndroid Build Coastguard Worker sem_wait(&caller_sem);
114*c9945492SAndroid Build Coastguard Worker
115*c9945492SAndroid Build Coastguard Worker sem_destroy(&caller_sem);
116*c9945492SAndroid Build Coastguard Worker sem_destroy(&target_sem);
117*c9945492SAndroid Build Coastguard Worker sem_destroy(&exit_sem);
118*c9945492SAndroid Build Coastguard Worker
119*c9945492SAndroid Build Coastguard Worker pthread_setcancelstate(cs, 0);
120*c9945492SAndroid Build Coastguard Worker __tl_unlock();
121*c9945492SAndroid Build Coastguard Worker __restore_sigs(&oldmask);
122*c9945492SAndroid Build Coastguard Worker }
123