2  * Copyright (c) 2000-2009 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@ 
  32  * Mach Operating System 
  33  * Copyright (c) 1991,1990,1989,1988,1987 Carnegie Mellon University 
  34  * All Rights Reserved. 
  36  * Permission to use, copy, modify and distribute this software and its 
  37  * documentation is hereby granted, provided that both the copyright 
  38  * notice and this permission notice appear in all copies of the 
  39  * software, derivative works or modified versions, and any portions 
  40  * thereof, and that both notices appear in supporting documentation. 
  42  * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS" 
  43  * CONDITION.  CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR 
  44  * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE. 
  46  * Carnegie Mellon requests users of this software to return to 
  48  *  Software Distribution Coordinator  or  Software.Distribution@CS.CMU.EDU 
  49  *  School of Computer Science 
  50  *  Carnegie Mellon University 
  51  *  Pittsburgh PA 15213-3890 
  53  * any improvements or extensions that they make and grant Carnegie Mellon 
  54  * the rights to redistribute these changes. 
  60  *      Author: Avadis Tevanian, Jr. 
  66 #include <mach/boolean.h> 
  67 #include <mach/kern_return.h> 
  68 #include <mach/machine.h> 
  69 #include <kern/host.h> 
  70 #include <kern/mach_param.h> 
  71 #include <kern/sched.h> 
  73 #include <kern/thread.h> 
  74 #include <kern/processor.h> 
  75 #include <machine/machparam.h> 
  78  *      thread_quantum_expire: 
  80  *      Recalculate the quantum and priority for a thread. 
  86 thread_quantum_expire( 
  87         timer_call_param_t      p0
, 
  88         timer_call_param_t      p1
) 
  90         processor_t                     processor 
= p0
; 
  97          *      Check for fail-safe trip. 
  99         if (!(thread
->sched_mode 
& (TH_MODE_TIMESHARE
|TH_MODE_PROMOTED
))) { 
 100                 uint64_t                        new_computation
; 
 102                 new_computation 
= processor
->quantum_end
; 
 103                 new_computation 
-= thread
->computation_epoch
; 
 104                 if (new_computation 
+ thread
->computation_metered 
> 
 105                                                                                         max_unsafe_computation
) { 
 107                         if (thread
->sched_mode 
& TH_MODE_REALTIME
) { 
 108                                 thread
->priority 
= DEPRESSPRI
; 
 110                                 thread
->safe_mode 
|= TH_MODE_REALTIME
; 
 111                                 thread
->sched_mode 
&= ~TH_MODE_REALTIME
; 
 116                         thread
->safe_release 
= sched_tick 
+ sched_safe_duration
; 
 117                         thread
->sched_mode 
|= (TH_MODE_FAILSAFE
|TH_MODE_TIMESHARE
); 
 122          *      Recompute scheduled priority if appropriate. 
 124         if (thread
->sched_stamp 
!= sched_tick
) 
 125                 update_priority(thread
); 
 127         if (thread
->sched_mode 
& TH_MODE_TIMESHARE
) { 
 128                 register uint32_t       delta
; 
 130                 thread_timer_delta(thread
, delta
); 
 133                  *      Accumulate timesharing usage only 
 134                  *      during contention for processor 
 137                 if (thread
->pri_shift 
< INT8_MAX
) 
 138                         thread
->sched_usage 
+= delta
; 
 140                 thread
->cpu_delta 
+= delta
; 
 143                  * Adjust the scheduled priority if 
 144                  * the thread has not been promoted 
 145                  * and is not depressed. 
 147                 if (    !(thread
->sched_mode 
& TH_MODE_PROMOTED
)        && 
 148                                 !(thread
->sched_mode 
& TH_MODE_ISDEPRESSED
)             ) 
 149                         compute_my_priority(thread
); 
 152         processor
->current_pri 
= thread
->sched_pri
; 
 155          *      This quantum is up, give this thread another. 
 157         if (first_timeslice(processor
)) 
 158                 processor
->timeslice
--; 
 160         thread_quantum_init(thread
); 
 161         processor
->quantum_end 
+= thread
->current_quantum
; 
 162         timer_call_enter1(&processor
->quantum_timer
, 
 163                                                         thread
, processor
->quantum_end
); 
 166          *      Context switch check. 
 168         if ((preempt 
= csw_check(processor
)) != AST_NONE
) 
 171                 processor_set_t         pset 
= processor
->processor_set
; 
 175                 pset_pri_hint(pset
, processor
, processor
->current_pri
); 
 176                 pset_count_hint(pset
, processor
, processor
->runq
.count
); 
 181         thread_unlock(thread
); 
 185  *      Define shifts for simulating (5/8) ** n 
 187  *      Shift structures for holding update shifts.  Actual computation 
 188  *      is  usage = (usage >> shift1) +/- (usage >> abs(shift2))  where the 
 189  *      +/- is determined by the sign of shift 2. 
 196 #define SCHED_DECAY_TICKS       32 
 197 static struct shift_data        sched_decay_shifts
[SCHED_DECAY_TICKS
] = { 
 198         {1,1},{1,3},{1,-3},{2,-7},{3,5},{3,-5},{4,-8},{5,7}, 
 199         {5,-7},{6,-10},{7,10},{7,-9},{8,-11},{9,12},{9,-11},{10,-13}, 
 200         {11,14},{11,-13},{12,-15},{13,17},{13,-15},{14,-17},{15,19},{16,18}, 
 201         {16,-19},{17,22},{18,20},{18,-20},{19,26},{20,22},{20,-22},{21,-27} 
 205  *      do_priority_computation: 
 207  *      Calculate the timesharing priority based upon usage and load. 
 209 #ifdef CONFIG_EMBEDDED 
 211 #define do_priority_computation(thread, pri)                                                    \ 
 213         (pri) = (thread)->priority              /* start with base priority */          \ 
 214             - ((thread)->sched_usage >> (thread)->pri_shift);                           \ 
 215         if ((pri) < MAXPRI_THROTTLE) {                                                                          \ 
 216                 if ((thread)->task->max_priority > MAXPRI_THROTTLE)                             \ 
 217                         (pri) = MAXPRI_THROTTLE;                                                                        \ 
 219                         if ((pri) < MINPRI_USER)                                                                        \ 
 220                                 (pri) = MINPRI_USER;                                                                    \ 
 222         if ((pri) > MAXPRI_KERNEL)                                                                                      \ 
 223                 (pri) = MAXPRI_KERNEL;                                                                                  \ 
 228 #define do_priority_computation(thread, pri)                                                    \ 
 230         (pri) = (thread)->priority              /* start with base priority */          \ 
 231             - ((thread)->sched_usage >> (thread)->pri_shift);                           \ 
 232         if ((pri) < MINPRI_USER)                                                                                        \ 
 233                 (pri) = MINPRI_USER;                                                                                    \ 
 235         if ((pri) > MAXPRI_KERNEL)                                                                                      \ 
 236                 (pri) = MAXPRI_KERNEL;                                                                                  \ 
 244  *      Set the base priority of the thread 
 245  *      and reset its scheduled priority. 
 247  *      Called with the thread locked. 
 251         register thread_t       thread
, 
 252         register int            priority
) 
 254         thread
->priority 
= priority
; 
 255         compute_priority(thread
, FALSE
); 
 261  *      Reset the scheduled priority of the thread 
 262  *      according to its base priority if the 
 263  *      thread has not been promoted or depressed. 
 265  *      Called with the thread locked. 
 269         register thread_t       thread
, 
 270         boolean_t                       override_depress
) 
 272         register int            priority
; 
 274         if (    !(thread
->sched_mode 
& TH_MODE_PROMOTED
)                        && 
 275                         (!(thread
->sched_mode 
& TH_MODE_ISDEPRESSED
)    || 
 276                                  override_depress                                                       
)               ) { 
 277                 if (thread
->sched_mode 
& TH_MODE_TIMESHARE
) 
 278                         do_priority_computation(thread
, priority
); 
 280                         priority 
= thread
->priority
; 
 282                 set_sched_pri(thread
, priority
); 
 287  *      compute_my_priority: 
 289  *      Reset the scheduled priority for 
 290  *      a timesharing thread. 
 292  *      Only for use on the current thread 
 293  *      if timesharing and not depressed. 
 295  *      Called with the thread locked. 
 299         register thread_t       thread
) 
 301         register int            priority
; 
 303         do_priority_computation(thread
, priority
); 
 304         assert(thread
->runq 
== PROCESSOR_NULL
); 
 305         thread
->sched_pri 
= priority
; 
 311  *      Perform housekeeping operations driven by scheduler tick. 
 313  *      Called with the thread locked. 
 317         register thread_t       thread
) 
 319         register unsigned       ticks
; 
 320         register uint32_t       delta
; 
 322         ticks 
= sched_tick 
- thread
->sched_stamp
; 
 324         thread
->sched_stamp 
+= ticks
; 
 325         thread
->pri_shift 
= sched_pri_shift
; 
 328          *      Gather cpu usage data. 
 330         thread_timer_delta(thread
, delta
); 
 331         if (ticks 
< SCHED_DECAY_TICKS
) { 
 332                 register struct shift_data      
*shiftp
; 
 335                  *      Accumulate timesharing usage only 
 336                  *      during contention for processor 
 339                 if (thread
->pri_shift 
< INT8_MAX
) 
 340                         thread
->sched_usage 
+= delta
; 
 342                 thread
->cpu_usage 
+= delta 
+ thread
->cpu_delta
; 
 343                 thread
->cpu_delta 
= 0; 
 345                 shiftp 
= &sched_decay_shifts
[ticks
]; 
 346                 if (shiftp
->shift2 
> 0) { 
 348                                                 (thread
->cpu_usage 
>> shiftp
->shift1
) + 
 349                                                 (thread
->cpu_usage 
>> shiftp
->shift2
); 
 350                     thread
->sched_usage 
= 
 351                                                 (thread
->sched_usage 
>> shiftp
->shift1
) + 
 352                                                 (thread
->sched_usage 
>> shiftp
->shift2
); 
 356                                                 (thread
->cpu_usage 
>> shiftp
->shift1
) - 
 357                                                 (thread
->cpu_usage 
>> -(shiftp
->shift2
)); 
 358                     thread
->sched_usage 
= 
 359                                                 (thread
->sched_usage 
>> shiftp
->shift1
) - 
 360                                                 (thread
->sched_usage 
>> -(shiftp
->shift2
)); 
 364                 thread
->cpu_usage 
= thread
->cpu_delta 
= 0; 
 365                 thread
->sched_usage 
= 0; 
 369          *      Check for fail-safe release. 
 371         if (    (thread
->sched_mode 
& TH_MODE_FAILSAFE
)         && 
 372                         thread
->sched_stamp 
>= thread
->safe_release             
) { 
 373                 if (!(thread
->safe_mode 
& TH_MODE_TIMESHARE
)) { 
 374                         if (thread
->safe_mode 
& TH_MODE_REALTIME
) { 
 375                                 thread
->priority 
= BASEPRI_RTQUEUES
; 
 377                                 thread
->sched_mode 
|= TH_MODE_REALTIME
; 
 380                         thread
->sched_mode 
&= ~TH_MODE_TIMESHARE
; 
 382                         if ((thread
->state 
& (TH_RUN
|TH_IDLE
)) == TH_RUN
) 
 385                         if (!(thread
->sched_mode 
& TH_MODE_ISDEPRESSED
)) 
 386                                 set_sched_pri(thread
, thread
->priority
); 
 389                 thread
->safe_mode 
= 0; 
 390                 thread
->sched_mode 
&= ~TH_MODE_FAILSAFE
; 
 394          *      Recompute scheduled priority if appropriate. 
 396         if (    (thread
->sched_mode 
& TH_MODE_TIMESHARE
)        && 
 397                         !(thread
->sched_mode 
& TH_MODE_PROMOTED
)        && 
 398                         !(thread
->sched_mode 
& TH_MODE_ISDEPRESSED
)             ) { 
 399                 register int            new_pri
; 
 401                 do_priority_computation(thread
, new_pri
); 
 402                 if (new_pri 
!= thread
->sched_pri
) { 
 403                         boolean_t               removed 
= run_queue_remove(thread
); 
 405                         thread
->sched_pri 
= new_pri
; 
 407                                 thread_setrun(thread
, SCHED_TAILQ
);