xref: /freebsd/share/man/man9/scheduler.9 (revision 1b6c76a2fe091c74f08427e6c870851025a9cf67)
1.\" Copyright (c) 2000-2001 John H. Baldwin <jhb@FreeBSD.org>
2.\" All rights reserved.
3.\"
4.\" Redistribution and use in source and binary forms, with or without
5.\" modification, are permitted provided that the following conditions
6.\" are met:
7.\" 1. Redistributions of source code must retain the above copyright
8.\"    notice, this list of conditions and the following disclaimer.
9.\" 2. Redistributions in binary form must reproduce the above copyright
10.\"    notice, this list of conditions and the following disclaimer in the
11.\"    documentation and/or other materials provided with the distribution.
12.\"
13.\" THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY EXPRESS OR
14.\" IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
15.\" OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
16.\" IN NO EVENT SHALL THE DEVELOPERS BE LIABLE FOR ANY DIRECT, INDIRECT,
17.\" INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
18.\" NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
19.\" DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
20.\" THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
21.\" (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
22.\" THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
23.\"
24.\" $FreeBSD$
25.\"
26.Dd November 3, 2000
27.Dt SCHEDULER 9
28.Os
29.Sh NAME
30.Nm curpriority_cmp ,
31.Nm maybe_resched ,
32.Nm resetpriority ,
33.Nm roundrobin ,
34.Nm roundrobin_interval ,
35.Nm sched_setup ,
36.Nm schedclock ,
37.Nm schedcpu ,
38.Nm setrunnable ,
39.Nm updatepri
40.Nd perform round-robin scheduling of runnable processes
41.Sh SYNOPSIS
42.Fd #include <sys/param.h>
43.Fd #include <sys/proc.h>
44.Ft int
45.Fn curpriority_cmp "struct proc *p"
46.Ft void
47.Fn maybe_resched "struct proc *chk"
48.Ft void
49.Fn propagate_priority "struct proc *p"
50.Ft void
51.Fn resetpriority "struct proc *p"
52.Ft void
53.Fn roundrobin "void *arg"
54.Ft int
55.Fn roundrobin_interval "void"
56.Ft void
57.Fn sched_setup "void *dummy"
58.Ft void
59.Fn schedclock "struct proc *p"
60.Ft void
61.Fn schedcpu "void *arg"
62.Ft void
63.Fn setrunnable "struct proc *p"
64.Ft void
65.Fn updatepri "struct proc *p"
66.Sh DESCRIPTION
67Each process has three different priorities stored in
68.Vt "struct proc" :
69.Va p_usrpri ,
70.Va p_nativepri ,
71and
72.Va p_priority .
73.Pp
74The
75.Va p_usrpri
76member is the user priority of the process calculated from a process'
77estimated CPU time and nice level.
78.Pp
79The
80.Va p_nativepri
81member is the saved priority used by
82.Fn propagate_priority .
83When a process obtains a mutex, its priority is saved in
84.Va p_nativepri .
85While it holds the mutex, the process's priority may be bumped by another
86process that blocks on the mutex.
87When the process releases the mutex, then its priority is restored to the
88priority saved in
89.Va p_nativepri .
90.Pp
91The
92.Va p_priority
93member is the actual priority of the process and is used to determine what
94.Xr runqueue 9
95it runs on, for example.
96.Pp
97The
98.Fn curpriority_cmp
99function compares the cached priority of the currently running process with
100process
101.Fa p .
102If the currently running process has a higher priority, then it will return
103a value less than zero.
104If the current process has a lower priority, then it will return a value
105greater than zero.
106If the current process has the same priority as
107.Fa p ,
108then
109.Fn curpriority_cmp
110will return zero.
111The cached priority of the currently running process is updated when a process
112resumes from
113.Xr tsleep 9
114or returns to userland in
115.Fn userret
116and is stored in the private variable
117.Va curpriority .
118.Pp
119The
120.Fn maybe_resched
121function compares the priorities of the current process and process
122.Fa chk .
123If
124.Fa chk
125has a higher priority than the current process, then a context switch is
126needed, and
127.Fn need_resched
128is called to force a reschedule on the next return to userland.
129.Pp
130The
131.Fn propagate_priority
132looks at the process that owns the mutex
133.Fa p
134is blocked on.
135That process's priority is bumped to the priority of
136.Fa p
137if needed.
138If the process is currently running, then the function returns.
139If the process is on a
140.Xr runqueue 9 ,
141then the process is moved to the appropriate
142.Xr runqueue 9
143for its new priority.
144If the process is blocked on a mutex, its position in the list of
145processes blocked on the mutex in question is updated to reflect its new
146priority.
147Then, the function repeats the procedure using the process that owns the
148mutex just encountered.
149Note that an process's priorities are only bumped to the priority of the
150original process
151.Fa p ,
152not to the priority of the previously encountered process.
153.Pp
154The
155.Fn resetpriority
156function recomputes the user priority of the process
157.Fa p
158stored in
159.Va p_usrpri
160and calls
161.Fn maybe_resched
162to force a reschedule if needed.
163.Pp
164The
165.Fn roundrobin
166function is used as a
167.Xr timeout 9
168function to force a reschedule every
169.Va sched_quantum
170ticks.
171.Pp
172The
173.Fn roundrobin_interval
174function simply returns the number of clock ticks in between reschedules
175triggered by
176.Fn roundrobin .
177Thus, all it does is return the current value of
178.Va sched_quantum .
179.Pp
180The
181.Fn sched_setup
182function is a
183.Xr SYSINIT 9
184that is called to start up the callout driven scheduler functions.
185It just calls the
186.Fn roundrobin
187and
188.Fn schedcpu
189functions for the first time.
190After the initial call, the two functions will propagate themselves by
191registering their callout event again at the completion of the respective
192function.
193.Pp
194The
195.Fn schedclock
196function is called by
197.Fn statclock
198to adjust the priority of the currently running process.
199It updates the process's estimated CPU time and then adjusts the priority via
200.Fn resetpriority .
201.Pp
202The
203.Fn schedcpu
204function updates all process priorities.
205First, it updates statistics that track how long processes have been in various
206process states.
207Secondly, it updates the estimated CPU time for the current process such
208that about 90% of the CPU usage is forgotten in 5 * load average seconds.
209For example, if the load average is 2.00,
210then at least 90% of the estimated CPU time for the process should be based
211on the amount of CPU time the process has had in the last 10 seconds.
212It then recomputes the priority of the process and moves it to the
213appropriate
214.Xr runqueue 9
215if necessary.
216Thirdly, it updates the %CPU estimate used by utilities such as
217.Xr ps 1
218and
219.Xr top 1
220so that 95% of the CPU usage is forgotten in 60 seconds.
221Once all process priorities have been updated,
222.Fn schedcpu
223calls
224.Fn vmmeter
225to update various other statistics including the load average.
226Finally, it schedules itself to run again in
227.Va hz
228clock ticks.
229.Pp
230The
231.Fn setrunnable
232function is used to change a process's state to be runnable.
233The process is placed on a
234.Xr runqueue 9
235if needed, and the swapper process is woken up and told to swap the process in
236if the process is swapped out.
237If the process has been asleep for at least one run of
238.Fn schedcpu ,
239then
240.Fn updatepri
241is used to adjust the priority of the process.
242.Pp
243The
244.Fn updatepri
245function is used to adjust the priority of a process that has been asleep.
246It retroactively decays the estimated CPU time of the process for each
247.Fn schedcpu
248event that the process was asleep.
249Finally, it calls
250.Fn resetpriority
251to adjust the priority of the process.
252.Sh SEE ALSO
253.Xr mi_switch 9 ,
254.Xr runqueue 9 ,
255.Xr sleepqueue 9 ,
256.Xr tsleep 9
257.Sh BUGS
258The
259.Va curpriority
260variable really should be per-CPU.
261In addition,
262.Fn maybe_resched
263should compare the priority of
264.Fa chk
265with that of each CPU, and then send an IPI to the processor with the lowest
266priority to trigger a reschedule if needed.
267.Pp
268Priority propagation is broken and is thus disabled by default.
269The
270.Va p_nativepri
271variable is only updated if a process does not obtain a sleep mutex on the
272first try.
273Also, if a process obtains more than one sleep mutex in this manner, and
274had its priority bumped in between, then
275.Va p_nativepri
276will be clobbered.
277