xref: /dflybsd-src/lib/libthread_xu/thread/thr_suspend_np.c (revision 940be950819fa932cd401a01f1182bf686a2e61e)
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