171b3fa15SDavid Xu /*
271b3fa15SDavid Xu * Copyright (c) 1995-1998 John Birrell <jb@cimlogic.com.au>.
371b3fa15SDavid Xu * All rights reserved.
471b3fa15SDavid Xu *
571b3fa15SDavid Xu * Redistribution and use in source and binary forms, with or without
671b3fa15SDavid Xu * modification, are permitted provided that the following conditions
771b3fa15SDavid Xu * are met:
871b3fa15SDavid Xu * 1. Redistributions of source code must retain the above copyright
971b3fa15SDavid Xu * notice, this list of conditions and the following disclaimer.
1071b3fa15SDavid Xu * 2. Redistributions in binary form must reproduce the above copyright
1171b3fa15SDavid Xu * notice, this list of conditions and the following disclaimer in the
1271b3fa15SDavid Xu * documentation and/or other materials provided with the distribution.
13d3b15642Szrj * 3. Neither the name of the author nor the names of any co-contributors
1471b3fa15SDavid Xu * may be used to endorse or promote products derived from this software
1571b3fa15SDavid Xu * without specific prior written permission.
1671b3fa15SDavid Xu *
1771b3fa15SDavid Xu * THIS SOFTWARE IS PROVIDED BY JOHN BIRRELL AND CONTRIBUTORS ``AS IS'' AND
1871b3fa15SDavid Xu * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
1971b3fa15SDavid Xu * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2071b3fa15SDavid Xu * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
2171b3fa15SDavid Xu * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2271b3fa15SDavid Xu * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2371b3fa15SDavid Xu * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2471b3fa15SDavid Xu * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2571b3fa15SDavid Xu * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2671b3fa15SDavid Xu * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2771b3fa15SDavid Xu * SUCH DAMAGE.
2871b3fa15SDavid Xu *
2971b3fa15SDavid Xu */
309e2ee207SJoerg Sonnenberger
31fc71f871SDavid Xu #include "namespace.h"
3271b3fa15SDavid Xu #include <errno.h>
3371b3fa15SDavid Xu #include <pthread.h>
34fc71f871SDavid Xu #include <pthread_np.h>
35fc71f871SDavid Xu #include "un-namespace.h"
36fd4288bdSDavid Xu
3771b3fa15SDavid Xu #include "thr_private.h"
3871b3fa15SDavid Xu
39*940be950Szrj static int suspend_common(pthread_t, pthread_t, int);
4071b3fa15SDavid Xu
4171b3fa15SDavid Xu /* Suspend a thread: */
4271b3fa15SDavid Xu int
_pthread_suspend_np(pthread_t thread)4371b3fa15SDavid Xu _pthread_suspend_np(pthread_t thread)
4471b3fa15SDavid Xu {
45*940be950Szrj pthread_t curthread = tls_get_curthread();
4671b3fa15SDavid Xu int ret;
4771b3fa15SDavid Xu
4871b3fa15SDavid Xu /* Suspending the current thread doesn't make sense. */
499e2ee207SJoerg Sonnenberger if (thread == curthread)
5071b3fa15SDavid Xu ret = EDEADLK;
5171b3fa15SDavid Xu
5271b3fa15SDavid Xu /* Add a reference to the thread: */
5371b3fa15SDavid Xu else if ((ret = _thr_ref_add(curthread, thread, /*include dead*/0))
5471b3fa15SDavid Xu == 0) {
5571b3fa15SDavid Xu /* Lock the threads scheduling queue: */
5671b3fa15SDavid Xu THR_THREAD_LOCK(curthread, thread);
57fd4288bdSDavid Xu suspend_common(curthread, thread, 1);
5871b3fa15SDavid Xu /* Unlock the threads scheduling queue: */
5971b3fa15SDavid Xu THR_THREAD_UNLOCK(curthread, thread);
6071b3fa15SDavid Xu
6171b3fa15SDavid Xu /* Don't forget to remove the reference: */
6271b3fa15SDavid Xu _thr_ref_delete(curthread, thread);
6371b3fa15SDavid Xu }
6471b3fa15SDavid Xu return (ret);
6571b3fa15SDavid Xu }
6671b3fa15SDavid Xu
6771b3fa15SDavid Xu void
_pthread_suspend_all_np(void)6871b3fa15SDavid Xu _pthread_suspend_all_np(void)
6971b3fa15SDavid Xu {
70*940be950Szrj pthread_t curthread = tls_get_curthread();
71*940be950Szrj pthread_t thread;
72fd4288bdSDavid Xu int ret;
7371b3fa15SDavid Xu
7471b3fa15SDavid Xu THREAD_LIST_LOCK(curthread);
7571b3fa15SDavid Xu
7671b3fa15SDavid Xu TAILQ_FOREACH(thread, &_thread_list, tle) {
7771b3fa15SDavid Xu if (thread != curthread) {
7871b3fa15SDavid Xu THR_THREAD_LOCK(curthread, thread);
79fd4288bdSDavid Xu if (thread->state != PS_DEAD &&
80fd4288bdSDavid Xu !(thread->flags & THR_FLAGS_SUSPENDED))
81fd4288bdSDavid Xu thread->flags |= THR_FLAGS_NEED_SUSPEND;
82fd4288bdSDavid Xu THR_THREAD_UNLOCK(curthread, thread);
83fd4288bdSDavid Xu }
84fd4288bdSDavid Xu }
85fd4288bdSDavid Xu /* thr_kill(-1, SIGCANCEL); */
86fd4288bdSDavid Xu
87fd4288bdSDavid Xu restart:
88fd4288bdSDavid Xu TAILQ_FOREACH(thread, &_thread_list, tle) {
89fd4288bdSDavid Xu if (thread != curthread) {
90fd4288bdSDavid Xu /* First try to suspend the thread without waiting */
91fd4288bdSDavid Xu THR_THREAD_LOCK(curthread, thread);
92fd4288bdSDavid Xu ret = suspend_common(curthread, thread, 0);
93fd4288bdSDavid Xu if (ret == 0) {
94fd4288bdSDavid Xu /* Can not suspend, try to wait */
95fd4288bdSDavid Xu thread->refcount++;
96fd4288bdSDavid Xu THREAD_LIST_UNLOCK(curthread);
97fd4288bdSDavid Xu suspend_common(curthread, thread, 1);
98fd4288bdSDavid Xu THR_THREAD_UNLOCK(curthread, thread);
99fd4288bdSDavid Xu THREAD_LIST_LOCK(curthread);
100fd4288bdSDavid Xu _thr_ref_delete_unlocked(curthread, thread);
101fd4288bdSDavid Xu /*
102fd4288bdSDavid Xu * Because we were blocked, things may have
103fd4288bdSDavid Xu * been changed, we have to restart the
104fd4288bdSDavid Xu * process.
105fd4288bdSDavid Xu */
106fd4288bdSDavid Xu goto restart;
107fd4288bdSDavid Xu }
10871b3fa15SDavid Xu THR_THREAD_UNLOCK(curthread, thread);
10971b3fa15SDavid Xu }
11071b3fa15SDavid Xu }
11171b3fa15SDavid Xu
11271b3fa15SDavid Xu THREAD_LIST_UNLOCK(curthread);
11371b3fa15SDavid Xu }
11471b3fa15SDavid Xu
115fd4288bdSDavid Xu static int
suspend_common(pthread_t curthread,pthread_t thread,int waitok)116*940be950Szrj suspend_common(pthread_t curthread, pthread_t thread, int waitok)
11771b3fa15SDavid Xu {
118fd4288bdSDavid Xu umtx_t tmp;
119fd4288bdSDavid Xu
120fd4288bdSDavid Xu while (thread->state != PS_DEAD &&
121fd4288bdSDavid Xu !(thread->flags & THR_FLAGS_SUSPENDED)) {
12271b3fa15SDavid Xu thread->flags |= THR_FLAGS_NEED_SUSPEND;
123fd4288bdSDavid Xu tmp = thread->cycle;
12471b3fa15SDavid Xu _thr_send_sig(thread, SIGCANCEL);
1255a1048c8SDavid Xu THR_THREAD_UNLOCK(curthread, thread);
126fd4288bdSDavid Xu if (waitok) {
127fd4288bdSDavid Xu _thr_umtx_wait(&thread->cycle, tmp, NULL, 0);
128fd4288bdSDavid Xu THR_THREAD_LOCK(curthread, thread);
129fd4288bdSDavid Xu } else {
130fd4288bdSDavid Xu THR_THREAD_LOCK(curthread, thread);
131fd4288bdSDavid Xu return (0);
13271b3fa15SDavid Xu }
13371b3fa15SDavid Xu }
134fd4288bdSDavid Xu
135fd4288bdSDavid Xu return (1);
136fd4288bdSDavid Xu }
1375a1048c8SDavid Xu
1385a1048c8SDavid Xu __strong_reference(_pthread_suspend_np, pthread_suspend_np);
1395a1048c8SDavid Xu __strong_reference(_pthread_suspend_all_np, pthread_suspend_all_np);
140