2 * Copyright (c) 1993-1995, 1999-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 #include <mach/mach_types.h>
30 #include <mach/thread_act.h>
32 #include <kern/kern_types.h>
33 #include <kern/kalloc.h>
34 #include <kern/sched_prim.h>
35 #include <kern/clock.h>
36 #include <kern/task.h>
37 #include <kern/thread.h>
38 #include <kern/wait_queue.h>
40 #include <vm/vm_pageout.h>
42 #include <kern/thread_call.h>
43 #include <kern/call_entry.h>
45 #include <kern/timer_call.h>
47 #include <sys/kdebug.h>
49 #define internal_call_num 768
51 #define thread_call_thread_min 4
55 internal_call_storage
[internal_call_num
];
57 decl_simple_lock_data(static,thread_call_lock
)
61 thread_call_delaytimer
;
65 thread_call_xxx_queue
,
66 thread_call_pending_queue
, thread_call_delayed_queue
;
70 call_thread_waitqueue
;
74 activate_thread_awake
;
90 static __inline__ thread_call_t
91 _internal_call_allocate(void);
93 static __inline__
void
94 _internal_call_release(
98 static __inline__
void
99 _pending_call_enqueue(
102 _pending_call_dequeue(
105 _delayed_call_enqueue(
108 _delayed_call_dequeue(
112 static __inline__
void
113 _set_delayed_call_timer(
118 _remove_from_pending_queue(
119 thread_call_func_t func
,
120 thread_call_param_t param0
,
123 _remove_from_delayed_queue(
124 thread_call_func_t func
,
125 thread_call_param_t param0
,
130 _call_thread_wake(void);
134 _activate_thread(void);
138 timer_call_param_t p0
,
139 timer_call_param_t p1
142 #define qe(x) ((queue_entry_t)(x))
143 #define TC(x) ((thread_call_t)(x))
146 * Routine: thread_call_initialize [public]
148 * Description: Initialize this module, called
149 * early during system initialization.
151 * Preconditions: None.
153 * Postconditions: None.
157 thread_call_initialize(void)
159 kern_return_t result
;
164 simple_lock_init(&thread_call_lock
, 0);
167 simple_lock(&thread_call_lock
);
169 queue_init(&thread_call_pending_queue
);
170 queue_init(&thread_call_delayed_queue
);
172 queue_init(&thread_call_xxx_queue
);
174 call
= internal_call_storage
;
175 call
< &internal_call_storage
[internal_call_num
];
178 enqueue_tail(&thread_call_xxx_queue
, qe(call
));
181 timer_call_setup(&thread_call_delaytimer
, _delayed_call_timer
, NULL
);
183 wait_queue_init(&call_thread_waitqueue
, SYNC_POLICY_FIFO
);
184 thread_call_vars
.thread_lowat
= thread_call_thread_min
;
186 activate_thread_awake
= TRUE
;
188 simple_unlock(&thread_call_lock
);
191 result
= kernel_thread_start_priority((thread_continue_t
)_activate_thread
, NULL
, MAXPRI_KERNEL
- 2, &thread
);
192 if (result
!= KERN_SUCCESS
)
193 panic("thread_call_initialize");
195 thread_deallocate(thread
);
201 thread_call_func_t func
,
202 thread_call_param_t param0
205 call_entry_setup(call
, func
, param0
);
209 * Routine: _internal_call_allocate [private, inline]
211 * Purpose: Allocate an internal callout entry.
213 * Preconditions: thread_call_lock held.
215 * Postconditions: None.
218 static __inline__ thread_call_t
219 _internal_call_allocate(void)
223 if (queue_empty(&thread_call_xxx_queue
))
224 panic("_internal_call_allocate");
226 call
= TC(dequeue_head(&thread_call_xxx_queue
));
232 * Routine: _internal_call_release [private, inline]
234 * Purpose: Release an internal callout entry which
235 * is no longer pending (or delayed).
237 * Preconditions: thread_call_lock held.
239 * Postconditions: None.
244 _internal_call_release(
248 if ( call
>= internal_call_storage
&&
249 call
< &internal_call_storage
[internal_call_num
] )
250 enqueue_head(&thread_call_xxx_queue
, qe(call
));
254 * Routine: _pending_call_enqueue [private, inline]
256 * Purpose: Place an entry at the end of the
257 * pending queue, to be executed soon.
259 * Preconditions: thread_call_lock held.
261 * Postconditions: None.
266 _pending_call_enqueue(
270 enqueue_tail(&thread_call_pending_queue
, qe(call
));
271 if (++thread_call_vars
.pending_num
> thread_call_vars
.pending_hiwat
)
272 thread_call_vars
.pending_hiwat
= thread_call_vars
.pending_num
;
274 call
->state
= PENDING
;
278 * Routine: _pending_call_dequeue [private, inline]
280 * Purpose: Remove an entry from the pending queue,
281 * effectively unscheduling it.
283 * Preconditions: thread_call_lock held.
285 * Postconditions: None.
290 _pending_call_dequeue(
294 (void)remque(qe(call
));
295 thread_call_vars
.pending_num
--;
301 * Routine: _delayed_call_enqueue [private, inline]
303 * Purpose: Place an entry on the delayed queue,
304 * after existing entries with an earlier
305 * (or identical) deadline.
307 * Preconditions: thread_call_lock held.
309 * Postconditions: None.
314 _delayed_call_enqueue(
318 thread_call_t current
;
320 current
= TC(queue_first(&thread_call_delayed_queue
));
323 if ( queue_end(&thread_call_delayed_queue
, qe(current
)) ||
324 call
->deadline
< current
->deadline
) {
325 current
= TC(queue_prev(qe(current
)));
329 current
= TC(queue_next(qe(current
)));
332 insque(qe(call
), qe(current
));
333 if (++thread_call_vars
.delayed_num
> thread_call_vars
.delayed_hiwat
)
334 thread_call_vars
.delayed_hiwat
= thread_call_vars
.delayed_num
;
336 call
->state
= DELAYED
;
340 * Routine: _delayed_call_dequeue [private, inline]
342 * Purpose: Remove an entry from the delayed queue,
343 * effectively unscheduling it.
345 * Preconditions: thread_call_lock held.
347 * Postconditions: None.
352 _delayed_call_dequeue(
356 (void)remque(qe(call
));
357 thread_call_vars
.delayed_num
--;
363 * Routine: _set_delayed_call_timer [private]
365 * Purpose: Reset the timer so that it
366 * next expires when the entry is due.
368 * Preconditions: thread_call_lock held.
370 * Postconditions: None.
373 static __inline__
void
374 _set_delayed_call_timer(
378 timer_call_enter(&thread_call_delaytimer
, call
->deadline
);
382 * Routine: _remove_from_pending_queue [private]
384 * Purpose: Remove the first (or all) matching
385 * entries from the pending queue,
386 * effectively unscheduling them.
387 * Returns whether any matching entries
390 * Preconditions: thread_call_lock held.
392 * Postconditions: None.
397 _remove_from_pending_queue(
398 thread_call_func_t func
,
399 thread_call_param_t param0
,
403 boolean_t call_removed
= FALSE
;
406 call
= TC(queue_first(&thread_call_pending_queue
));
408 while (!queue_end(&thread_call_pending_queue
, qe(call
))) {
409 if ( call
->func
== func
&&
410 call
->param0
== param0
) {
411 thread_call_t next
= TC(queue_next(qe(call
)));
413 _pending_call_dequeue(call
);
415 _internal_call_release(call
);
424 call
= TC(queue_next(qe(call
)));
427 return (call_removed
);
431 * Routine: _remove_from_delayed_queue [private]
433 * Purpose: Remove the first (or all) matching
434 * entries from the delayed queue,
435 * effectively unscheduling them.
436 * Returns whether any matching entries
439 * Preconditions: thread_call_lock held.
441 * Postconditions: None.
446 _remove_from_delayed_queue(
447 thread_call_func_t func
,
448 thread_call_param_t param0
,
452 boolean_t call_removed
= FALSE
;
455 call
= TC(queue_first(&thread_call_delayed_queue
));
457 while (!queue_end(&thread_call_delayed_queue
, qe(call
))) {
458 if ( call
->func
== func
&&
459 call
->param0
== param0
) {
460 thread_call_t next
= TC(queue_next(qe(call
)));
462 _delayed_call_dequeue(call
);
464 _internal_call_release(call
);
473 call
= TC(queue_next(qe(call
)));
476 return (call_removed
);
480 * Routine: thread_call_func [public]
482 * Purpose: Schedule a function callout.
483 * Guarantees { function, argument }
484 * uniqueness if unique_call is TRUE.
486 * Preconditions: Callable from an interrupt context
489 * Postconditions: None.
494 thread_call_func_t func
,
495 thread_call_param_t param
,
496 boolean_t unique_call
503 simple_lock(&thread_call_lock
);
505 call
= TC(queue_first(&thread_call_pending_queue
));
507 while (unique_call
&& !queue_end(&thread_call_pending_queue
, qe(call
))) {
508 if ( call
->func
== func
&&
509 call
->param0
== param
) {
513 call
= TC(queue_next(qe(call
)));
516 if (!unique_call
|| queue_end(&thread_call_pending_queue
, qe(call
))) {
517 call
= _internal_call_allocate();
519 call
->param0
= param
;
522 _pending_call_enqueue(call
);
524 if (thread_call_vars
.active_num
<= 0)
528 simple_unlock(&thread_call_lock
);
533 * Routine: thread_call_func_delayed [public]
535 * Purpose: Schedule a function callout to
536 * occur at the stated time.
538 * Preconditions: Callable from an interrupt context
541 * Postconditions: None.
545 thread_call_func_delayed(
546 thread_call_func_t func
,
547 thread_call_param_t param
,
555 simple_lock(&thread_call_lock
);
557 call
= _internal_call_allocate();
559 call
->param0
= param
;
561 call
->deadline
= deadline
;
563 _delayed_call_enqueue(call
);
565 if (queue_first(&thread_call_delayed_queue
) == qe(call
))
566 _set_delayed_call_timer(call
);
568 simple_unlock(&thread_call_lock
);
573 * Routine: thread_call_func_cancel [public]
575 * Purpose: Unschedule a function callout.
576 * Removes one (or all)
577 * { function, argument }
578 * instance(s) from either (or both)
579 * the pending and the delayed queue,
580 * in that order. Returns a boolean
581 * indicating whether any calls were
584 * Preconditions: Callable from an interrupt context
587 * Postconditions: None.
591 thread_call_func_cancel(
592 thread_call_func_t func
,
593 thread_call_param_t param
,
601 simple_lock(&thread_call_lock
);
604 result
= _remove_from_pending_queue(func
, param
, cancel_all
) |
605 _remove_from_delayed_queue(func
, param
, cancel_all
);
607 result
= _remove_from_pending_queue(func
, param
, cancel_all
) ||
608 _remove_from_delayed_queue(func
, param
, cancel_all
);
610 simple_unlock(&thread_call_lock
);
617 * Routine: thread_call_allocate [public]
619 * Purpose: Allocate an external callout
622 * Preconditions: None.
624 * Postconditions: None.
628 thread_call_allocate(
629 thread_call_func_t func
,
630 thread_call_param_t param0
633 thread_call_t call
= (void *)kalloc(sizeof (thread_call_data_t
));
636 call
->param0
= param0
;
643 * Routine: thread_call_free [public]
645 * Purpose: Free an external callout
648 * Preconditions: None.
650 * Postconditions: None.
661 simple_lock(&thread_call_lock
);
663 if (call
->state
!= IDLE
) {
664 simple_unlock(&thread_call_lock
);
670 simple_unlock(&thread_call_lock
);
673 kfree(call
, sizeof (thread_call_data_t
));
679 * Routine: thread_call_enter [public]
681 * Purpose: Schedule an external callout
682 * entry to occur "soon". Returns a
683 * boolean indicating whether the call
684 * had been already scheduled.
686 * Preconditions: Callable from an interrupt context
689 * Postconditions: None.
697 boolean_t result
= TRUE
;
701 simple_lock(&thread_call_lock
);
703 if (call
->state
!= PENDING
) {
704 if (call
->state
== DELAYED
)
705 _delayed_call_dequeue(call
);
706 else if (call
->state
== IDLE
)
709 _pending_call_enqueue(call
);
711 if (thread_call_vars
.active_num
<= 0)
717 simple_unlock(&thread_call_lock
);
726 thread_call_param_t param1
729 boolean_t result
= TRUE
;
733 simple_lock(&thread_call_lock
);
735 if (call
->state
!= PENDING
) {
736 if (call
->state
== DELAYED
)
737 _delayed_call_dequeue(call
);
738 else if (call
->state
== IDLE
)
741 _pending_call_enqueue(call
);
743 if (thread_call_vars
.active_num
<= 0)
747 call
->param1
= param1
;
749 simple_unlock(&thread_call_lock
);
756 * Routine: thread_call_enter_delayed [public]
758 * Purpose: Schedule an external callout
759 * entry to occur at the stated time.
760 * Returns a boolean indicating whether
761 * the call had been already scheduled.
763 * Preconditions: Callable from an interrupt context
766 * Postconditions: None.
770 thread_call_enter_delayed(
775 boolean_t result
= TRUE
;
779 simple_lock(&thread_call_lock
);
781 if (call
->state
== PENDING
)
782 _pending_call_dequeue(call
);
783 else if (call
->state
== DELAYED
)
784 _delayed_call_dequeue(call
);
785 else if (call
->state
== IDLE
)
789 call
->deadline
= deadline
;
791 _delayed_call_enqueue(call
);
793 if (queue_first(&thread_call_delayed_queue
) == qe(call
))
794 _set_delayed_call_timer(call
);
796 simple_unlock(&thread_call_lock
);
803 thread_call_enter1_delayed(
805 thread_call_param_t param1
,
809 boolean_t result
= TRUE
;
813 simple_lock(&thread_call_lock
);
815 if (call
->state
== PENDING
)
816 _pending_call_dequeue(call
);
817 else if (call
->state
== DELAYED
)
818 _delayed_call_dequeue(call
);
819 else if (call
->state
== IDLE
)
822 call
->param1
= param1
;
823 call
->deadline
= deadline
;
825 _delayed_call_enqueue(call
);
827 if (queue_first(&thread_call_delayed_queue
) == qe(call
))
828 _set_delayed_call_timer(call
);
830 simple_unlock(&thread_call_lock
);
837 * Routine: thread_call_cancel [public]
839 * Purpose: Unschedule a callout entry.
840 * Returns a boolean indicating
841 * whether the call had actually
844 * Preconditions: Callable from an interrupt context
847 * Postconditions: None.
855 boolean_t result
= TRUE
;
859 simple_lock(&thread_call_lock
);
861 if (call
->state
== PENDING
)
862 _pending_call_dequeue(call
);
863 else if (call
->state
== DELAYED
)
864 _delayed_call_dequeue(call
);
868 simple_unlock(&thread_call_lock
);
875 * Routine: thread_call_is_delayed [public]
877 * Purpose: Returns a boolean indicating
878 * whether a call is currently scheduled
879 * to occur at a later time. Optionally
880 * returns the expiration time.
882 * Preconditions: Callable from an interrupt context
885 * Postconditions: None.
889 thread_call_is_delayed(
893 boolean_t result
= FALSE
;
897 simple_lock(&thread_call_lock
);
899 if (call
->state
== DELAYED
) {
900 if (deadline
!= NULL
)
901 *deadline
= call
->deadline
;
905 simple_unlock(&thread_call_lock
);
912 * Routine: _call_thread_wake [private, inline]
914 * Purpose: Wake a callout thread to service
915 * pending callout entries. May wake
916 * the activate thread in order to
917 * create additional callout threads.
919 * Preconditions: thread_call_lock held.
921 * Postconditions: None.
925 _call_thread_wake(void)
927 if (wait_queue_wakeup_one(&call_thread_waitqueue
, NULL
, THREAD_AWAKENED
) == KERN_SUCCESS
) {
928 thread_call_vars
.idle_thread_num
--;
930 if (++thread_call_vars
.active_num
> thread_call_vars
.active_hiwat
)
931 thread_call_vars
.active_hiwat
= thread_call_vars
.active_num
;
934 if (!activate_thread_awake
) {
935 thread_wakeup_one(&activate_thread_awake
);
936 activate_thread_awake
= TRUE
;
943 * Call out invoked by the scheduler.
949 __unused thread_t thread
)
951 simple_lock(&thread_call_lock
);
955 case SCHED_CALL_BLOCK
:
956 if (--thread_call_vars
.active_num
< thread_call_vars
.active_lowat
)
957 thread_call_vars
.active_lowat
= thread_call_vars
.active_num
;
959 if ( thread_call_vars
.active_num
<= 0 &&
960 thread_call_vars
.pending_num
> 0 )
964 case SCHED_CALL_UNBLOCK
:
965 if (++thread_call_vars
.active_num
> thread_call_vars
.active_hiwat
)
966 thread_call_vars
.active_hiwat
= thread_call_vars
.active_num
;
970 simple_unlock(&thread_call_lock
);
974 * Routine: _call_thread [private]
976 * Purpose: Executed by a callout thread.
978 * Preconditions: None.
980 * Postconditions: None.
985 _call_thread_continue(void)
987 thread_t self
= current_thread();
990 simple_lock(&thread_call_lock
);
992 thread_sched_call(self
, sched_call_thread
);
994 while (thread_call_vars
.pending_num
> 0) {
996 thread_call_func_t func
;
997 thread_call_param_t param0
, param1
;
999 call
= TC(dequeue_head(&thread_call_pending_queue
));
1000 thread_call_vars
.pending_num
--;
1003 param0
= call
->param0
;
1004 param1
= call
->param1
;
1008 _internal_call_release(call
);
1010 simple_unlock(&thread_call_lock
);
1013 KERNEL_DEBUG_CONSTANT(
1014 MACHDBG_CODE(DBG_MACH_SCHED
,MACH_CALLOUT
) | DBG_FUNC_NONE
,
1015 (int)func
, (int)param0
, (int)param1
, 0, 0);
1017 (*func
)(param0
, param1
);
1019 (void)thread_funnel_set(self
->funnel_lock
, FALSE
);
1022 simple_lock(&thread_call_lock
);
1025 thread_sched_call(self
, NULL
);
1027 if (--thread_call_vars
.active_num
< thread_call_vars
.active_lowat
)
1028 thread_call_vars
.active_lowat
= thread_call_vars
.active_num
;
1030 if (thread_call_vars
.idle_thread_num
< thread_call_vars
.thread_lowat
) {
1031 thread_call_vars
.idle_thread_num
++;
1033 wait_queue_assert_wait(&call_thread_waitqueue
, NULL
, THREAD_UNINT
, 0);
1035 simple_unlock(&thread_call_lock
);
1038 thread_block((thread_continue_t
)_call_thread_continue
);
1042 thread_call_vars
.thread_num
--;
1044 simple_unlock(&thread_call_lock
);
1047 thread_terminate(self
);
1055 _call_thread_continue();
1060 * Routine: _activate_thread [private]
1062 * Purpose: Executed by the activate thread.
1064 * Preconditions: None.
1066 * Postconditions: Never terminates.
1071 _activate_thread_continue(void)
1073 kern_return_t result
;
1077 simple_lock(&thread_call_lock
);
1079 while ( thread_call_vars
.active_num
<= 0 &&
1080 thread_call_vars
.pending_num
> 0 ) {
1082 if (++thread_call_vars
.active_num
> thread_call_vars
.active_hiwat
)
1083 thread_call_vars
.active_hiwat
= thread_call_vars
.active_num
;
1085 if (++thread_call_vars
.thread_num
> thread_call_vars
.thread_hiwat
)
1086 thread_call_vars
.thread_hiwat
= thread_call_vars
.thread_num
;
1088 simple_unlock(&thread_call_lock
);
1091 result
= kernel_thread_start_priority((thread_continue_t
)_call_thread
, NULL
, MAXPRI_KERNEL
- 1, &thread
);
1092 if (result
!= KERN_SUCCESS
)
1093 panic("activate_thread");
1095 thread_deallocate(thread
);
1098 simple_lock(&thread_call_lock
);
1101 assert_wait(&activate_thread_awake
, THREAD_INTERRUPTIBLE
);
1102 activate_thread_awake
= FALSE
;
1104 simple_unlock(&thread_call_lock
);
1107 thread_block((thread_continue_t
)_activate_thread_continue
);
1113 _activate_thread(void)
1115 thread_t self
= current_thread();
1117 self
->options
|= TH_OPT_VMPRIV
;
1118 vm_page_free_reserve(2); /* XXX */
1120 _activate_thread_continue();
1126 _delayed_call_timer(
1127 __unused timer_call_param_t p0
,
1128 __unused timer_call_param_t p1
1133 boolean_t new_pending
= FALSE
;
1137 simple_lock(&thread_call_lock
);
1139 clock_get_uptime(×tamp
);
1141 call
= TC(queue_first(&thread_call_delayed_queue
));
1143 while (!queue_end(&thread_call_delayed_queue
, qe(call
))) {
1144 if (call
->deadline
<= timestamp
) {
1145 _delayed_call_dequeue(call
);
1147 _pending_call_enqueue(call
);
1153 call
= TC(queue_first(&thread_call_delayed_queue
));
1156 if (!queue_end(&thread_call_delayed_queue
, qe(call
)))
1157 _set_delayed_call_timer(call
);
1159 if (new_pending
&& thread_call_vars
.active_num
<= 0)
1160 _call_thread_wake();
1162 simple_unlock(&thread_call_lock
);