2 * Copyright (c) 2000-2007 Apple Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
29 * @OSF_FREE_COPYRIGHT@
32 * Copyright (c) 1993 The University of Utah and
33 * the Center for Software Science (CSS). All rights reserved.
35 * Permission to use, copy, modify and distribute this software and its
36 * documentation is hereby granted, provided that both the copyright
37 * notice and this permission notice appear in all copies of the
38 * software, derivative works or modified versions, and any portions
39 * thereof, and that both notices appear in supporting documentation.
41 * THE UNIVERSITY OF UTAH AND CSS ALLOW FREE USE OF THIS SOFTWARE IN ITS "AS
42 * IS" CONDITION. THE UNIVERSITY OF UTAH AND CSS DISCLAIM ANY LIABILITY OF
43 * ANY KIND FOR ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
45 * CSS requests users of this software to return to css-dist@cs.utah.edu any
46 * improvements that they make and grant CSS redistribution rights.
48 * Author: Bryan Ford, University of Utah CSS
50 * Thread management routines
52 #include <mach/mach_types.h>
53 #include <mach/kern_return.h>
54 #include <mach/alert.h>
56 #include <mach/thread_act_server.h>
58 #include <kern/kern_types.h>
60 #include <kern/mach_param.h>
61 #include <kern/zalloc.h>
62 #include <kern/extmod_statistics.h>
63 #include <kern/thread.h>
64 #include <kern/task.h>
65 #include <kern/sched_prim.h>
66 #include <kern/misc_protos.h>
67 #include <kern/assert.h>
68 #include <kern/exception.h>
69 #include <kern/ipc_mig.h>
70 #include <kern/ipc_tt.h>
71 #include <kern/machine.h>
73 #include <kern/syscall_subr.h>
74 #include <kern/sync_lock.h>
75 #include <kern/processor.h>
76 #include <kern/timer.h>
77 #include <kern/affinity.h>
81 #include <security/mac_mach_internal.h>
83 void act_abort(thread_t
);
84 void install_special_handler_locked(thread_t
);
85 void special_handler_continue(void);
88 * Internal routine to mark a thread as started.
89 * Always called with the thread mutex locked.
91 * Note: function intentionally declared with the noinline attribute to
92 * prevent multiple declaration of probe symbols in this file; we would
93 * prefer "#pragma noinline", but gcc does not support it.
94 * PR-6385749 -- the lwp-start probe should fire from within the context
95 * of the newly created thread. Commented out for now, in case we
96 * turn it into a dead code probe.
99 thread_start_internal(
102 clear_wait(thread
, THREAD_AWAKENED
);
103 thread
->started
= TRUE
;
104 // DTRACE_PROC1(lwp__start, thread_t, thread);
108 * Internal routine to terminate a thread.
109 * Sometimes called with task already locked.
112 thread_terminate_internal(
115 kern_return_t result
= KERN_SUCCESS
;
117 thread_mtx_lock(thread
);
119 if (thread
->active
) {
120 thread
->active
= FALSE
;
125 clear_wait(thread
, THREAD_INTERRUPTED
);
127 thread_start_internal(thread
);
131 result
= KERN_TERMINATED
;
133 if (thread
->affinity_set
!= NULL
)
134 thread_affinity_terminate(thread
);
136 thread_mtx_unlock(thread
);
138 if (thread
!= current_thread() && result
== KERN_SUCCESS
)
139 thread_wait(thread
, FALSE
);
145 * Terminate a thread.
151 kern_return_t result
;
153 if (thread
== THREAD_NULL
)
154 return (KERN_INVALID_ARGUMENT
);
156 if ( thread
->task
== kernel_task
&&
157 thread
!= current_thread() )
158 return (KERN_FAILURE
);
160 result
= thread_terminate_internal(thread
);
163 * If a kernel thread is terminating itself, force an AST here.
164 * Kernel threads don't normally pass through the AST checking
165 * code - and all threads finish their own termination in the
166 * special handler APC.
168 if (thread
->task
== kernel_task
) {
169 ml_set_interrupts_enabled(FALSE
);
170 ast_taken(AST_APC
, TRUE
);
171 panic("thread_terminate");
178 * Suspend execution of the specified thread.
179 * This is a recursive-style suspension of the thread, a count of
180 * suspends is maintained.
182 * Called with thread mutex held.
186 register thread_t thread
)
188 if (thread
->suspend_count
++ == 0) {
189 install_special_handler(thread
);
191 thread_wakeup_one(&thread
->suspend_count
);
196 * Decrement internal suspension count, setting thread
197 * runnable when count falls to zero.
199 * Called with thread mutex held.
203 register thread_t thread
)
205 if ( thread
->suspend_count
> 0 &&
206 --thread
->suspend_count
== 0 ) {
208 thread_wakeup_one(&thread
->suspend_count
);
210 thread_start_internal(thread
);
217 register thread_t thread
)
219 thread_t self
= current_thread();
220 kern_return_t result
= KERN_SUCCESS
;
222 if (thread
== THREAD_NULL
|| thread
->task
== kernel_task
)
223 return (KERN_INVALID_ARGUMENT
);
225 thread_mtx_lock(thread
);
227 if (thread
->active
) {
228 if ( thread
->user_stop_count
++ == 0 &&
229 thread
->suspend_count
++ == 0 ) {
230 install_special_handler(thread
);
232 thread_wakeup_one(&thread
->suspend_count
);
236 result
= KERN_TERMINATED
;
238 thread_mtx_unlock(thread
);
240 if (thread
!= self
&& result
== KERN_SUCCESS
)
241 thread_wait(thread
, FALSE
);
248 register thread_t thread
)
250 kern_return_t result
= KERN_SUCCESS
;
252 if (thread
== THREAD_NULL
|| thread
->task
== kernel_task
)
253 return (KERN_INVALID_ARGUMENT
);
255 thread_mtx_lock(thread
);
257 if (thread
->active
) {
258 if (thread
->user_stop_count
> 0) {
259 if ( --thread
->user_stop_count
== 0 &&
260 --thread
->suspend_count
== 0 ) {
262 thread_wakeup_one(&thread
->suspend_count
);
264 thread_start_internal(thread
);
269 result
= KERN_FAILURE
;
272 result
= KERN_TERMINATED
;
274 thread_mtx_unlock(thread
);
280 * thread_depress_abort:
282 * Prematurely abort priority depression if there is one.
285 thread_depress_abort(
286 register thread_t thread
)
288 kern_return_t result
;
290 if (thread
== THREAD_NULL
)
291 return (KERN_INVALID_ARGUMENT
);
293 thread_mtx_lock(thread
);
296 result
= thread_depress_abort_internal(thread
);
298 result
= KERN_TERMINATED
;
300 thread_mtx_unlock(thread
);
307 * Indicate that the activation should run its
308 * special handler to detect a condition.
310 * Called with thread mutex held.
316 spl_t s
= splsched();
320 if (!(thread
->sched_flags
& TH_SFLAG_ABORT
)) {
321 thread
->sched_flags
|= TH_SFLAG_ABORT
;
322 install_special_handler_locked(thread
);
325 thread
->sched_flags
&= ~TH_SFLAG_ABORTSAFELY
;
327 thread_unlock(thread
);
333 register thread_t thread
)
335 kern_return_t result
= KERN_SUCCESS
;
337 if (thread
== THREAD_NULL
)
338 return (KERN_INVALID_ARGUMENT
);
340 thread_mtx_lock(thread
);
342 if (thread
->active
) {
344 clear_wait(thread
, THREAD_INTERRUPTED
);
347 result
= KERN_TERMINATED
;
349 thread_mtx_unlock(thread
);
358 kern_return_t result
= KERN_SUCCESS
;
360 if (thread
== THREAD_NULL
)
361 return (KERN_INVALID_ARGUMENT
);
363 thread_mtx_lock(thread
);
365 if (thread
->active
) {
366 spl_t s
= splsched();
369 if (!thread
->at_safe_point
||
370 clear_wait_internal(thread
, THREAD_INTERRUPTED
) != KERN_SUCCESS
) {
371 if (!(thread
->sched_flags
& TH_SFLAG_ABORT
)) {
372 thread
->sched_flags
|= TH_SFLAG_ABORTED_MASK
;
373 install_special_handler_locked(thread
);
376 thread_unlock(thread
);
380 result
= KERN_TERMINATED
;
382 thread_mtx_unlock(thread
);
387 /*** backward compatibility hacks ***/
388 #include <mach/thread_info.h>
389 #include <mach/thread_special_ports.h>
390 #include <ipc/ipc_port.h>
395 thread_flavor_t flavor
,
396 thread_info_t thread_info_out
,
397 mach_msg_type_number_t
*thread_info_count
)
399 kern_return_t result
;
401 if (thread
== THREAD_NULL
)
402 return (KERN_INVALID_ARGUMENT
);
404 thread_mtx_lock(thread
);
406 if (thread
->active
|| thread
->inspection
)
407 result
= thread_info_internal(
408 thread
, flavor
, thread_info_out
, thread_info_count
);
410 result
= KERN_TERMINATED
;
412 thread_mtx_unlock(thread
);
419 register thread_t thread
,
421 thread_state_t state
, /* pointer to OUT array */
422 mach_msg_type_number_t
*state_count
) /*IN/OUT*/
424 kern_return_t result
= KERN_SUCCESS
;
426 if (thread
== THREAD_NULL
)
427 return (KERN_INVALID_ARGUMENT
);
429 thread_mtx_lock(thread
);
431 if (thread
->active
) {
432 if (thread
!= current_thread()) {
435 thread_mtx_unlock(thread
);
437 if (thread_stop(thread
, FALSE
)) {
438 thread_mtx_lock(thread
);
439 result
= machine_thread_get_state(
440 thread
, flavor
, state
, state_count
);
441 thread_unstop(thread
);
444 thread_mtx_lock(thread
);
445 result
= KERN_ABORTED
;
448 thread_release(thread
);
451 result
= machine_thread_get_state(
452 thread
, flavor
, state
, state_count
);
454 else if (thread
->inspection
)
456 result
= machine_thread_get_state(
457 thread
, flavor
, state
, state_count
);
460 result
= KERN_TERMINATED
;
462 thread_mtx_unlock(thread
);
468 * Change thread's machine-dependent state. Called with nothing
469 * locked. Returns same way.
472 thread_set_state_internal(
473 register thread_t thread
,
475 thread_state_t state
,
476 mach_msg_type_number_t state_count
,
479 kern_return_t result
= KERN_SUCCESS
;
481 if (thread
== THREAD_NULL
)
482 return (KERN_INVALID_ARGUMENT
);
484 thread_mtx_lock(thread
);
486 if (thread
->active
) {
487 if (thread
!= current_thread()) {
490 thread_mtx_unlock(thread
);
492 if (thread_stop(thread
, TRUE
)) {
493 thread_mtx_lock(thread
);
494 result
= machine_thread_set_state(
495 thread
, flavor
, state
, state_count
);
496 thread_unstop(thread
);
499 thread_mtx_lock(thread
);
500 result
= KERN_ABORTED
;
503 thread_release(thread
);
506 result
= machine_thread_set_state(
507 thread
, flavor
, state
, state_count
);
510 result
= KERN_TERMINATED
;
512 if ((result
== KERN_SUCCESS
) && from_user
)
513 extmod_statistics_incr_thread_set_state(thread
);
515 thread_mtx_unlock(thread
);
520 /* No prototype, since thread_act_server.h has the _from_user version if KERNEL_SERVER */
523 register thread_t thread
,
525 thread_state_t state
,
526 mach_msg_type_number_t state_count
);
530 register thread_t thread
,
532 thread_state_t state
,
533 mach_msg_type_number_t state_count
)
535 return thread_set_state_internal(thread
, flavor
, state
, state_count
, FALSE
);
539 thread_set_state_from_user(
540 register thread_t thread
,
542 thread_state_t state
,
543 mach_msg_type_number_t state_count
)
545 return thread_set_state_internal(thread
, flavor
, state
, state_count
, TRUE
);
549 * Kernel-internal "thread" interfaces used outside this file:
552 /* Initialize (or re-initialize) a thread state. Called from execve
553 * with nothing locked, returns same way.
556 thread_state_initialize(
557 register thread_t thread
)
559 kern_return_t result
= KERN_SUCCESS
;
561 if (thread
== THREAD_NULL
)
562 return (KERN_INVALID_ARGUMENT
);
564 thread_mtx_lock(thread
);
566 if (thread
->active
) {
567 if (thread
!= current_thread()) {
570 thread_mtx_unlock(thread
);
572 if (thread_stop(thread
, TRUE
)) {
573 thread_mtx_lock(thread
);
574 result
= machine_thread_state_initialize( thread
);
575 thread_unstop(thread
);
578 thread_mtx_lock(thread
);
579 result
= KERN_ABORTED
;
582 thread_release(thread
);
585 result
= machine_thread_state_initialize( thread
);
588 result
= KERN_TERMINATED
;
590 thread_mtx_unlock(thread
);
598 register thread_t target
)
600 thread_t self
= current_thread();
601 kern_return_t result
= KERN_SUCCESS
;
603 if (target
== THREAD_NULL
|| target
== self
)
604 return (KERN_INVALID_ARGUMENT
);
606 thread_mtx_lock(target
);
608 if (target
->active
) {
611 thread_mtx_unlock(target
);
613 if (thread_stop(target
, TRUE
)) {
614 thread_mtx_lock(target
);
615 result
= machine_thread_dup(self
, target
);
616 if (self
->affinity_set
!= AFFINITY_SET_NULL
)
617 thread_affinity_dup(self
, target
);
618 thread_unstop(target
);
621 thread_mtx_lock(target
);
622 result
= KERN_ABORTED
;
625 thread_release(target
);
628 result
= KERN_TERMINATED
;
630 thread_mtx_unlock(target
);
639 * Set the status of the specified thread.
640 * Called with (and returns with) no locks held.
644 register thread_t thread
,
646 thread_state_t tstate
,
647 mach_msg_type_number_t count
)
650 return (thread_set_state(thread
, flavor
, tstate
, count
));
656 * Get the status of the specified thread.
660 register thread_t thread
,
662 thread_state_t tstate
,
663 mach_msg_type_number_t
*count
)
665 return (thread_get_state(thread
, flavor
, tstate
, count
));
669 * Change thread's machine-dependent userspace TSD base.
670 * Called with nothing locked. Returns same way.
675 mach_vm_offset_t tsd_base
)
677 kern_return_t result
= KERN_SUCCESS
;
679 if (thread
== THREAD_NULL
)
680 return (KERN_INVALID_ARGUMENT
);
682 thread_mtx_lock(thread
);
684 if (thread
->active
) {
685 if (thread
!= current_thread()) {
688 thread_mtx_unlock(thread
);
690 if (thread_stop(thread
, TRUE
)) {
691 thread_mtx_lock(thread
);
692 result
= machine_thread_set_tsd_base(thread
, tsd_base
);
693 thread_unstop(thread
);
696 thread_mtx_lock(thread
);
697 result
= KERN_ABORTED
;
700 thread_release(thread
);
703 result
= machine_thread_set_tsd_base(thread
, tsd_base
);
706 result
= KERN_TERMINATED
;
708 thread_mtx_unlock(thread
);
714 * install_special_handler:
716 * Install the special returnhandler that handles suspension and
717 * termination, if it hasn't been installed already.
719 * Called with the thread mutex held.
722 install_special_handler(
725 spl_t s
= splsched();
728 install_special_handler_locked(thread
);
729 thread_unlock(thread
);
734 * install_special_handler_locked:
736 * Do the work of installing the special_handler.
738 * Called with the thread mutex and scheduling lock held.
741 install_special_handler_locked(
746 * Temporarily undepress, so target has
747 * a chance to do locking required to
748 * block itself in special_handler().
750 if (thread
->sched_flags
& TH_SFLAG_DEPRESSED_MASK
)
751 thread_recompute_sched_pri(thread
, TRUE
);
753 thread_ast_set(thread
, AST_APC
);
755 if (thread
== current_thread())
756 ast_propagate(thread
->ast
);
758 processor_t processor
= thread
->last_processor
;
760 if ( processor
!= PROCESSOR_NULL
&&
761 processor
->state
== PROCESSOR_RUNNING
&&
762 processor
->active_thread
== thread
)
763 cause_ast_check(processor
);
768 * Activation control support routines internal to this file:
773 * special_handler_continue
775 * Continuation routine for the special handler blocks. It checks
776 * to see whether there has been any new suspensions. If so, it
777 * installs the special handler again. Otherwise, it checks to see
778 * if the current depression needs to be re-instated (it may have
779 * been temporarily removed in order to get to this point in a hurry).
782 special_handler_continue(void)
784 thread_t thread
= current_thread();
786 thread_mtx_lock(thread
);
788 if (thread
->suspend_count
> 0)
789 install_special_handler(thread
);
791 spl_t s
= splsched();
794 if (thread
->sched_flags
& TH_SFLAG_DEPRESSED_MASK
) {
795 processor_t myprocessor
= thread
->last_processor
;
797 thread
->sched_pri
= DEPRESSPRI
;
799 KERNEL_DEBUG_CONSTANT(MACHDBG_CODE(DBG_MACH_SCHED
, MACH_SCHED_CHANGE_PRIORITY
),
800 (uintptr_t)thread_tid(thread
),
803 0, /* eventually, 'reason' */
806 myprocessor
->current_pri
= thread
->sched_pri
;
808 thread_unlock(thread
);
812 thread_mtx_unlock(thread
);
814 thread_exception_return();
819 * special_handler - handles suspension, termination. Called
820 * with nothing locked. Returns (if it returns) the same way.
828 thread_mtx_lock(thread
);
832 thread
->sched_flags
&= ~TH_SFLAG_ABORTED_MASK
;
833 thread_unlock(thread
);
837 * If we're suspended, go to sleep and wait for someone to wake us up.
839 if (thread
->active
) {
840 if (thread
->suspend_count
> 0) {
841 assert_wait(&thread
->suspend_count
, THREAD_ABORTSAFE
);
842 thread_mtx_unlock(thread
);
843 thread_block((thread_continue_t
)special_handler_continue
);
848 thread_mtx_unlock(thread
);
850 thread_terminate_self();
854 thread_mtx_unlock(thread
);
857 /* Prototype, see justification above */
862 thread_state_t state
,
863 mach_msg_type_number_t count
);
869 thread_state_t state
,
870 mach_msg_type_number_t count
)
872 if (thread
== current_thread())
873 return (KERN_INVALID_ARGUMENT
);
875 return (thread_set_state(thread
, flavor
, state
, count
));
880 act_set_state_from_user(
883 thread_state_t state
,
884 mach_msg_type_number_t count
)
886 if (thread
== current_thread())
887 return (KERN_INVALID_ARGUMENT
);
889 return (thread_set_state_from_user(thread
, flavor
, state
, count
));
897 thread_state_t state
,
898 mach_msg_type_number_t
*count
)
900 if (thread
== current_thread())
901 return (KERN_INVALID_ARGUMENT
);
903 return (thread_get_state(thread
, flavor
, state
, count
));
911 spl_t s
= splsched();
913 if (thread
== current_thread()) {
914 thread_ast_set(thread
, ast
);
915 ast_propagate(thread
->ast
);
917 processor_t processor
;
920 thread_ast_set(thread
, ast
);
921 processor
= thread
->last_processor
;
922 if ( processor
!= PROCESSOR_NULL
&&
923 processor
->state
== PROCESSOR_RUNNING
&&
924 processor
->active_thread
== thread
)
925 cause_ast_check(processor
);
926 thread_unlock(thread
);
936 act_set_ast( thread
, AST_BSD
);
944 if (thread
!= current_thread())
945 if( !ml_get_interrupts_enabled() )
946 panic("unsafe act_set_kperf operation");
948 act_set_ast( thread
, AST_KPERF
);
956 act_set_ast( thread
, AST_MACF
);
961 set_astledger(thread_t thread
)
963 act_set_ast(thread
, AST_LEDGER
);