author | urbanc |
Sun, 12 Feb 2012 23:25:49 +0000 | |
changeset 305 | 463bed130705 |
parent 304 | bd05c5011c0f |
child 306 | 5113aa1ae69a |
permissions | -rwxr-xr-x |
262 | 1 |
(*<*) |
2 |
theory Paper |
|
301 | 3 |
imports "../CpsG" "../ExtGG" "~~/src/HOL/Library/LaTeXsugar" |
262 | 4 |
begin |
266 | 5 |
ML {* |
273 | 6 |
open Printer; |
272 | 7 |
show_question_marks_default := false; |
266 | 8 |
*} |
284 | 9 |
|
10 |
notation (latex output) |
|
11 |
Cons ("_::_" [78,77] 73) and |
|
12 |
vt ("valid'_state") and |
|
13 |
runing ("running") and |
|
286 | 14 |
birthtime ("last'_set") and |
284 | 15 |
If ("(\<^raw:\textrm{>if\<^raw:}> (_)/ \<^raw:\textrm{>then\<^raw:}> (_)/ \<^raw:\textrm{>else\<^raw:}> (_))" 10) and |
286 | 16 |
Prc ("'(_, _')") and |
287 | 17 |
holding ("holds") and |
18 |
waiting ("waits") and |
|
290 | 19 |
Th ("T") and |
20 |
Cs ("C") and |
|
287 | 21 |
readys ("ready") and |
290 | 22 |
depend ("RAG") and |
23 |
preced ("prec") and |
|
24 |
cpreced ("cprec") and |
|
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
25 |
dependents ("dependants") and |
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
26 |
cp ("cprec") and |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
27 |
holdents ("resources") and |
299 | 28 |
original_priority ("priority") and |
284 | 29 |
DUMMY ("\<^raw:\mbox{$\_\!\_$}>") |
262 | 30 |
(*>*) |
31 |
||
32 |
section {* Introduction *} |
|
33 |
||
34 |
text {* |
|
284 | 35 |
Many real-time systems need to support threads involving priorities and |
267 | 36 |
locking of resources. Locking of resources ensures mutual exclusion |
275 | 37 |
when accessing shared data or devices that cannot be |
284 | 38 |
preempted. Priorities allow scheduling of threads that need to |
275 | 39 |
finish their work within deadlines. Unfortunately, both features |
40 |
can interact in subtle ways leading to a problem, called |
|
284 | 41 |
\emph{Priority Inversion}. Suppose three threads having priorities |
42 |
$H$(igh), $M$(edium) and $L$(ow). We would expect that the thread |
|
43 |
$H$ blocks any other thread with lower priority and itself cannot |
|
44 |
be blocked by any thread with lower priority. Alas, in a naive |
|
275 | 45 |
implementation of resource looking and priorities this property can |
46 |
be violated. Even worse, $H$ can be delayed indefinitely by |
|
284 | 47 |
threads with lower priorities. For this let $L$ be in the |
275 | 48 |
possession of a lock for a resource that also $H$ needs. $H$ must |
49 |
therefore wait for $L$ to exit the critical section and release this |
|
50 |
lock. The problem is that $L$ might in turn be blocked by any |
|
284 | 51 |
thread with priority $M$, and so $H$ sits there potentially waiting |
52 |
indefinitely. Since $H$ is blocked by threads with lower |
|
275 | 53 |
priorities, the problem is called Priority Inversion. It was first |
277 | 54 |
described in \cite{Lampson80} in the context of the |
275 | 55 |
Mesa programming language designed for concurrent programming. |
265 | 56 |
|
273 | 57 |
If the problem of Priority Inversion is ignored, real-time systems |
267 | 58 |
can become unpredictable and resulting bugs can be hard to diagnose. |
59 |
The classic example where this happened is the software that |
|
284 | 60 |
controlled the Mars Pathfinder mission in 1997 \cite{Reeves98}. |
61 |
Once the spacecraft landed, the software shut down at irregular |
|
62 |
intervals leading to loss of project time as normal operation of the |
|
63 |
craft could only resume the next day (the mission and data already |
|
64 |
collected were fortunately not lost, because of a clever system |
|
65 |
design). The reason for the shutdowns was that the scheduling |
|
66 |
software fell victim of Priority Inversion: a low priority thread |
|
67 |
locking a resource prevented a high priority thread from running in |
|
68 |
time leading to a system reset. Once the problem was found, it was |
|
69 |
rectified by enabling the \emph{Priority Inheritance Protocol} (PIP) |
|
70 |
\cite{Sha90}\footnote{Sha et al.~call it the \emph{Basic Priority |
|
286 | 71 |
Inheritance Protocol} \cite{Sha90} and others sometimes also call it |
72 |
\emph{Priority Boosting}.} in the scheduling software. |
|
262 | 73 |
|
284 | 74 |
The idea behind PIP is to let the thread $L$ temporarily inherit |
286 | 75 |
the high priority from $H$ until $L$ leaves the critical section |
284 | 76 |
unlocking the resource. This solves the problem of $H$ having to |
77 |
wait indefinitely, because $L$ cannot be blocked by threads having |
|
78 |
priority $M$. While a few other solutions exist for the Priority |
|
79 |
Inversion problem, PIP is one that is widely deployed and |
|
80 |
implemented. This includes VxWorks (a proprietary real-time OS used |
|
81 |
in the Mars Pathfinder mission, in Boeing's 787 Dreamliner, Honda's |
|
82 |
ASIMO robot, etc.), but also the POSIX 1003.1c Standard realised for |
|
83 |
example in libraries for FreeBSD, Solaris and Linux. |
|
274 | 84 |
|
284 | 85 |
One advantage of PIP is that increasing the priority of a thread |
275 | 86 |
can be dynamically calculated by the scheduler. This is in contrast |
277 | 87 |
to, for example, \emph{Priority Ceiling} \cite{Sha90}, another |
88 |
solution to the Priority Inversion problem, which requires static |
|
284 | 89 |
analysis of the program in order to prevent Priority |
90 |
Inversion. However, there has also been strong criticism against |
|
91 |
PIP. For instance, PIP cannot prevent deadlocks when lock |
|
92 |
dependencies are circular, and also blocking times can be |
|
93 |
substantial (more than just the duration of a critical section). |
|
94 |
Though, most criticism against PIP centres around unreliable |
|
95 |
implementations and PIP being too complicated and too inefficient. |
|
96 |
For example, Yodaiken writes in \cite{Yodaiken02}: |
|
274 | 97 |
|
98 |
\begin{quote} |
|
99 |
\it{}``Priority inheritance is neither efficient nor reliable. Implementations |
|
100 |
are either incomplete (and unreliable) or surprisingly complex and intrusive.'' |
|
101 |
\end{quote} |
|
273 | 102 |
|
274 | 103 |
\noindent |
275 | 104 |
He suggests to avoid PIP altogether by not allowing critical |
286 | 105 |
sections to be preempted. Unfortunately, this solution does not |
304 | 106 |
help in real-time systems with hard deadlines for high-priority |
107 |
threads. |
|
278 | 108 |
|
286 | 109 |
In our opinion, there is clearly a need for investigating correct |
278 | 110 |
algorithms for PIP. A few specifications for PIP exist (in English) |
111 |
and also a few high-level descriptions of implementations (e.g.~in |
|
112 |
the textbook \cite[Section 5.6.5]{Vahalia96}), but they help little |
|
113 |
with actual implementations. That this is a problem in practise is |
|
283 | 114 |
proved by an email from Baker, who wrote on 13 July 2009 on the Linux |
278 | 115 |
Kernel mailing list: |
274 | 116 |
|
117 |
\begin{quote} |
|
275 | 118 |
\it{}``I observed in the kernel code (to my disgust), the Linux PIP |
119 |
implementation is a nightmare: extremely heavy weight, involving |
|
120 |
maintenance of a full wait-for graph, and requiring updates for a |
|
121 |
range of events, including priority changes and interruptions of |
|
122 |
wait operations.'' |
|
274 | 123 |
\end{quote} |
124 |
||
125 |
\noindent |
|
277 | 126 |
The criticism by Yodaiken, Baker and others suggests to us to look |
127 |
again at PIP from a more abstract level (but still concrete enough |
|
286 | 128 |
to inform an implementation), and makes PIP an ideal candidate for a |
277 | 129 |
formal verification. One reason, of course, is that the original |
284 | 130 |
presentation of PIP~\cite{Sha90}, despite being informally |
283 | 131 |
``proved'' correct, is actually \emph{flawed}. |
132 |
||
133 |
Yodaiken \cite{Yodaiken02} points to a subtlety that had been |
|
134 |
overlooked in the informal proof by Sha et al. They specify in |
|
284 | 135 |
\cite{Sha90} that after the thread (whose priority has been raised) |
283 | 136 |
completes its critical section and releases the lock, it ``returns |
137 |
to its original priority level.'' This leads them to believe that an |
|
284 | 138 |
implementation of PIP is ``rather straightforward''~\cite{Sha90}. |
139 |
Unfortunately, as Yodaiken points out, this behaviour is too |
|
140 |
simplistic. Consider the case where the low priority thread $L$ |
|
141 |
locks \emph{two} resources, and two high-priority threads $H$ and |
|
300 | 142 |
$H'$ each wait for one of them. If $L$ releases one resource |
283 | 143 |
so that $H$, say, can proceed, then we still have Priority Inversion |
144 |
with $H'$ (which waits for the other resource). The correct |
|
145 |
behaviour for $L$ is to revert to the highest remaining priority of |
|
284 | 146 |
the threads that it blocks. The advantage of formalising the |
147 |
correctness of a high-level specification of PIP in a theorem prover |
|
148 |
is that such issues clearly show up and cannot be overlooked as in |
|
149 |
informal reasoning (since we have to analyse all possible behaviours |
|
300 | 150 |
of threads, i.e.~\emph{traces}, that could possibly happen).\medskip |
274 | 151 |
|
300 | 152 |
\noindent |
301 | 153 |
{\bf Contributions:} There have been earlier formal investigations |
304 | 154 |
into PIP \cite{Faria08,Jahier09,Wellings07}, but they employ model |
155 |
checking techniques. This paper presents a formalised and |
|
156 |
mechanically checked proof for the correctness of PIP (to our |
|
305 | 157 |
knowledge the first one; the earlier informal proof by Sha et |
304 | 158 |
al.~\cite{Sha90} is flawed). In contrast to model checking, our |
159 |
formalisation provides insight into why PIP is correct and allows us |
|
160 |
to prove stronger properties that, as we will show, inform an |
|
161 |
implementation. For example, we found by ``playing'' with the formalisation |
|
162 |
that the choice of the next thread to take over a lock when a |
|
305 | 163 |
resource is released is irrelevant for PIP being correct. Something |
164 |
which has not been mentioned in the relevant literature. |
|
280 | 165 |
*} |
278 | 166 |
|
283 | 167 |
section {* Formal Model of the Priority Inheritance Protocol *} |
267 | 168 |
|
280 | 169 |
text {* |
286 | 170 |
The Priority Inheritance Protocol, short PIP, is a scheduling |
171 |
algorithm for a single-processor system.\footnote{We shall come back |
|
172 |
later to the case of PIP on multi-processor systems.} Our model of |
|
173 |
PIP is based on Paulson's inductive approach to protocol |
|
174 |
verification \cite{Paulson98}, where the \emph{state} of a system is |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
175 |
given by a list of events that happened so far. \emph{Events} of PIP fall |
290 | 176 |
into five categories defined as the datatype: |
283 | 177 |
|
178 |
\begin{isabelle}\ \ \ \ \ %%% |
|
284 | 179 |
\mbox{\begin{tabular}{r@ {\hspace{2mm}}c@ {\hspace{2mm}}l@ {\hspace{7mm}}l} |
180 |
\isacommand{datatype} event |
|
181 |
& @{text "="} & @{term "Create thread priority"}\\ |
|
182 |
& @{text "|"} & @{term "Exit thread"} \\ |
|
286 | 183 |
& @{text "|"} & @{term "Set thread priority"} & {\rm reset of the priority for} @{text thread}\\ |
284 | 184 |
& @{text "|"} & @{term "P thread cs"} & {\rm request of resource} @{text "cs"} {\rm by} @{text "thread"}\\ |
185 |
& @{text "|"} & @{term "V thread cs"} & {\rm release of resource} @{text "cs"} {\rm by} @{text "thread"} |
|
186 |
\end{tabular}} |
|
187 |
\end{isabelle} |
|
188 |
||
189 |
\noindent |
|
286 | 190 |
whereby threads, priorities and (critical) resources are represented |
191 |
as natural numbers. The event @{term Set} models the situation that |
|
192 |
a thread obtains a new priority given by the programmer or |
|
193 |
user (for example via the {\tt nice} utility under UNIX). As in Paulson's work, we |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
194 |
need to define functions that allow us to make some observations |
297 | 195 |
about states. One, called @{term threads}, calculates the set of |
293 | 196 |
``live'' threads that we have seen so far: |
284 | 197 |
|
198 |
\begin{isabelle}\ \ \ \ \ %%% |
|
199 |
\mbox{\begin{tabular}{lcl} |
|
200 |
@{thm (lhs) threads.simps(1)} & @{text "\<equiv>"} & |
|
201 |
@{thm (rhs) threads.simps(1)}\\ |
|
202 |
@{thm (lhs) threads.simps(2)[where thread="th"]} & @{text "\<equiv>"} & |
|
203 |
@{thm (rhs) threads.simps(2)[where thread="th"]}\\ |
|
204 |
@{thm (lhs) threads.simps(3)[where thread="th"]} & @{text "\<equiv>"} & |
|
205 |
@{thm (rhs) threads.simps(3)[where thread="th"]}\\ |
|
206 |
@{term "threads (DUMMY#s)"} & @{text "\<equiv>"} & @{term "threads s"}\\ |
|
207 |
\end{tabular}} |
|
283 | 208 |
\end{isabelle} |
209 |
||
210 |
\noindent |
|
299 | 211 |
In this definition @{term "DUMMY # DUMMY"} stands for list-cons. |
290 | 212 |
Another function calculates the priority for a thread @{text "th"}, which is |
213 |
defined as |
|
284 | 214 |
|
215 |
\begin{isabelle}\ \ \ \ \ %%% |
|
216 |
\mbox{\begin{tabular}{lcl} |
|
217 |
@{thm (lhs) original_priority.simps(1)[where thread="th"]} & @{text "\<equiv>"} & |
|
218 |
@{thm (rhs) original_priority.simps(1)[where thread="th"]}\\ |
|
219 |
@{thm (lhs) original_priority.simps(2)[where thread="th" and thread'="th'"]} & @{text "\<equiv>"} & |
|
220 |
@{thm (rhs) original_priority.simps(2)[where thread="th" and thread'="th'"]}\\ |
|
221 |
@{thm (lhs) original_priority.simps(3)[where thread="th" and thread'="th'"]} & @{text "\<equiv>"} & |
|
222 |
@{thm (rhs) original_priority.simps(3)[where thread="th" and thread'="th'"]}\\ |
|
223 |
@{term "original_priority th (DUMMY#s)"} & @{text "\<equiv>"} & @{term "original_priority th s"}\\ |
|
224 |
\end{tabular}} |
|
225 |
\end{isabelle} |
|
226 |
||
227 |
\noindent |
|
228 |
In this definition we set @{text 0} as the default priority for |
|
229 |
threads that have not (yet) been created. The last function we need |
|
285 | 230 |
calculates the ``time'', or index, at which time a process had its |
290 | 231 |
priority last set. |
284 | 232 |
|
233 |
\begin{isabelle}\ \ \ \ \ %%% |
|
234 |
\mbox{\begin{tabular}{lcl} |
|
235 |
@{thm (lhs) birthtime.simps(1)[where thread="th"]} & @{text "\<equiv>"} & |
|
236 |
@{thm (rhs) birthtime.simps(1)[where thread="th"]}\\ |
|
237 |
@{thm (lhs) birthtime.simps(2)[where thread="th" and thread'="th'"]} & @{text "\<equiv>"} & |
|
238 |
@{thm (rhs) birthtime.simps(2)[where thread="th" and thread'="th'"]}\\ |
|
239 |
@{thm (lhs) birthtime.simps(3)[where thread="th" and thread'="th'"]} & @{text "\<equiv>"} & |
|
240 |
@{thm (rhs) birthtime.simps(3)[where thread="th" and thread'="th'"]}\\ |
|
241 |
@{term "birthtime th (DUMMY#s)"} & @{text "\<equiv>"} & @{term "birthtime th s"}\\ |
|
242 |
\end{tabular}} |
|
243 |
\end{isabelle} |
|
286 | 244 |
|
245 |
\noindent |
|
287 | 246 |
In this definition @{term "length s"} stands for the length of the list |
247 |
of events @{text s}. Again the default value in this function is @{text 0} |
|
248 |
for threads that have not been created yet. A \emph{precedence} of a thread @{text th} in a |
|
290 | 249 |
state @{text s} is the pair of natural numbers defined as |
284 | 250 |
|
286 | 251 |
\begin{isabelle}\ \ \ \ \ %%% |
290 | 252 |
@{thm preced_def[where thread="th"]} |
286 | 253 |
\end{isabelle} |
254 |
||
255 |
\noindent |
|
287 | 256 |
The point of precedences is to schedule threads not according to priorities (because what should |
286 | 257 |
we do in case two threads have the same priority), but according to precedences. |
290 | 258 |
Precedences allow us to always discriminate between two threads with equal priority by |
296 | 259 |
taking into account the time when the priority was last set. We order precedences so |
286 | 260 |
that threads with the same priority get a higher precedence if their priority has been |
293 | 261 |
set earlier, since for such threads it is more urgent to finish their work. In an implementation |
262 |
this choice would translate to a quite natural FIFO-scheduling of processes with |
|
286 | 263 |
the same priority. |
264 |
||
265 |
Next, we introduce the concept of \emph{waiting queues}. They are |
|
266 |
lists of threads associated with every resource. The first thread in |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
267 |
this list (i.e.~the head, or short @{term hd}) is chosen to be the one |
290 | 268 |
that is in possession of the |
286 | 269 |
``lock'' of the corresponding resource. We model waiting queues as |
293 | 270 |
functions, below abbreviated as @{text wq}. They take a resource as |
271 |
argument and return a list of threads. This allows us to define |
|
290 | 272 |
when a thread \emph{holds}, respectively \emph{waits} for, a |
293 | 273 |
resource @{text cs} given a waiting queue function @{text wq}. |
287 | 274 |
|
275 |
\begin{isabelle}\ \ \ \ \ %%% |
|
276 |
\begin{tabular}{@ {}l} |
|
290 | 277 |
@{thm cs_holding_def[where thread="th"]}\\ |
278 |
@{thm cs_waiting_def[where thread="th"]} |
|
287 | 279 |
\end{tabular} |
280 |
\end{isabelle} |
|
281 |
||
282 |
\noindent |
|
283 |
In this definition we assume @{text "set"} converts a list into a set. |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
284 |
At the beginning, that is in the state where no thread is created yet, |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
285 |
the waiting queue function will be the function that returns the |
293 | 286 |
empty list for every resource. |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
287 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
288 |
\begin{isabelle}\ \ \ \ \ %%% |
301 | 289 |
@{abbrev all_unlocked}\hfill\numbered{allunlocked} |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
290 |
\end{isabelle} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
291 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
292 |
\noindent |
290 | 293 |
Using @{term "holding"} and @{term waiting}, we can introduce \emph{Resource Allocation Graphs} |
294 |
(RAG), which represent the dependencies between threads and resources. |
|
295 |
We represent RAGs as relations using pairs of the form |
|
296 |
||
297 |
\begin{isabelle}\ \ \ \ \ %%% |
|
298 |
@{term "(Th th, Cs cs)"} \hspace{5mm}{\rm and}\hspace{5mm} |
|
299 |
@{term "(Cs cs, Th th)"} |
|
300 |
\end{isabelle} |
|
301 |
||
302 |
\noindent |
|
303 |
where the first stands for a \emph{waiting edge} and the second for a |
|
304 |
\emph{holding edge} (@{term Cs} and @{term Th} are constructors of a |
|
305 |
datatype for vertices). Given a waiting queue function, a RAG is defined |
|
306 |
as |
|
307 |
||
308 |
\begin{isabelle}\ \ \ \ \ %%% |
|
309 |
@{thm cs_depend_def} |
|
310 |
\end{isabelle} |
|
311 |
||
312 |
\noindent |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
313 |
Given three threads and three resources, an instance of a RAG is as follows: |
290 | 314 |
|
315 |
\begin{center} |
|
297 | 316 |
\newcommand{\fnt}{\fontsize{7}{8}\selectfont} |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
317 |
\begin{tikzpicture}[scale=1] |
297 | 318 |
%%\draw[step=2mm] (-3,2) grid (1,-1); |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
319 |
|
297 | 320 |
\node (A) at (0,0) [draw, rounded corners=1mm, rectangle, very thick] {@{text "th\<^isub>0"}}; |
321 |
\node (B) at (2,0) [draw, circle, very thick, inner sep=0.4mm] {@{text "cs\<^isub>1"}}; |
|
322 |
\node (C) at (4,0.7) [draw, rounded corners=1mm, rectangle, very thick] {@{text "th\<^isub>1"}}; |
|
323 |
\node (D) at (4,-0.7) [draw, rounded corners=1mm, rectangle, very thick] {@{text "th\<^isub>2"}}; |
|
324 |
\node (E) at (6,-0.7) [draw, circle, very thick, inner sep=0.4mm] {@{text "cs\<^isub>2"}}; |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
325 |
\node (E1) at (6, 0.2) [draw, circle, very thick, inner sep=0.4mm] {@{text "cs\<^isub>3"}}; |
297 | 326 |
\node (F) at (8,-0.7) [draw, rounded corners=1mm, rectangle, very thick] {@{text "th\<^isub>3"}}; |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
327 |
|
300 | 328 |
\draw [<-,line width=0.6mm] (A) to node [pos=0.54,sloped,above=-0.5mm] {\fnt{}holding} (B); |
297 | 329 |
\draw [->,line width=0.6mm] (C) to node [pos=0.4,sloped,above=-0.5mm] {\fnt{}waiting} (B); |
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
330 |
\draw [->,line width=0.6mm] (D) to node [pos=0.4,sloped,below=-0.5mm] {\fnt{}waiting} (B); |
300 | 331 |
\draw [<-,line width=0.6mm] (D) to node [pos=0.54,sloped,below=-0.5mm] {\fnt{}holding} (E); |
332 |
\draw [<-,line width=0.6mm] (D) to node [pos=0.54,sloped,above=-0.5mm] {\fnt{}holding} (E1); |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
333 |
\draw [->,line width=0.6mm] (F) to node [pos=0.45,sloped,below=-0.5mm] {\fnt{}waiting} (E); |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
334 |
\end{tikzpicture} |
290 | 335 |
\end{center} |
336 |
||
337 |
\noindent |
|
296 | 338 |
The use of relations for representing RAGs allows us to conveniently define |
290 | 339 |
the notion of the \emph{dependants} of a thread. This is defined as |
340 |
||
341 |
\begin{isabelle}\ \ \ \ \ %%% |
|
342 |
@{thm cs_dependents_def} |
|
343 |
\end{isabelle} |
|
344 |
||
345 |
\noindent |
|
296 | 346 |
This definition needs to account for all threads that wait for a thread to |
290 | 347 |
release a resource. This means we need to include threads that transitively |
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
348 |
wait for a resource being released (in the picture above this means the dependants |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
349 |
of @{text "th\<^isub>0"} are @{text "th\<^isub>1"} and @{text "th\<^isub>2"}, but also @{text "th\<^isub>3"}, |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
350 |
which cannot make any progress unless @{text "th\<^isub>2"} makes progress, which |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
351 |
in turn needs to wait for @{text "th\<^isub>1"} to finish). If there is a circle in a RAG, then clearly |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
352 |
we have a deadlock. Therefore when a thread requests a resource, |
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
353 |
we must ensure that the resulting RAG is not circular. |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
354 |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
355 |
Next we introduce the notion of the \emph{current precedence} of a thread @{text th} in a |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
356 |
state @{text s}. It is defined as |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
357 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
358 |
\begin{isabelle}\ \ \ \ \ %%% |
299 | 359 |
@{thm cpreced_def2}\hfill\numbered{cpreced} |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
360 |
\end{isabelle} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
361 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
362 |
\noindent |
293 | 363 |
While the precedence @{term prec} of a thread is determined by the programmer |
364 |
(for example when the thread is |
|
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
365 |
created), the point of the current precedence is to let scheduler increase this |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
366 |
priority, if needed according to PIP. Therefore the current precedence of @{text th} is |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
367 |
given as the maximum of the precedence @{text th} has in state @{text s} \emph{and} all |
296 | 368 |
processes that are dependants of @{text th}. Since the notion @{term "dependants"} is |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
369 |
defined as the transitive closure of all dependent threads, we deal correctly with the |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
370 |
problem in the algorithm by Sha et al.~\cite{Sha90} where a priority of a thread is |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
371 |
lowered prematurely. |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
372 |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
373 |
The next function, called @{term schs}, defines the behaviour of the scheduler. It will be defined |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
374 |
by recursion on the state (a list of events); @{term "schs"} returns a \emph{schedule state}, which |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
375 |
we represent as a record consisting of two |
296 | 376 |
functions: |
293 | 377 |
|
378 |
\begin{isabelle}\ \ \ \ \ %%% |
|
379 |
@{text "\<lparr>wq_fun, cprec_fun\<rparr>"} |
|
380 |
\end{isabelle} |
|
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
381 |
|
294
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
382 |
\noindent |
299 | 383 |
The first function is a waiting queue function (that is it takes a resource @{text "cs"} and returns the |
296 | 384 |
corresponding list of threads that wait for it), the second is a function that takes |
299 | 385 |
a thread and returns its current precedence (see \eqref{cpreced}). We assume the usual getter and |
296 | 386 |
setter methods for such records. |
294
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
387 |
|
301 | 388 |
In the initial state, the scheduler starts with all resources unlocked (see \eqref{allunlocked}) and the |
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
389 |
current precedence of every thread is initialised with @{term "Prc 0 0"}; that means |
299 | 390 |
\mbox{@{abbrev initial_cprec}}. Therefore |
296 | 391 |
we have |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
392 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
393 |
\begin{isabelle}\ \ \ \ \ %%% |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
394 |
\begin{tabular}{@ {}l} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
395 |
@{thm (lhs) schs.simps(1)} @{text "\<equiv>"}\\ |
294
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
396 |
\hspace{5mm}@{term "(|wq_fun = all_unlocked, cprec_fun = (\<lambda>_::thread. Prc 0 0)|)"} |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
397 |
\end{tabular} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
398 |
\end{isabelle} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
399 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
400 |
\noindent |
296 | 401 |
The cases for @{term Create}, @{term Exit} and @{term Set} are also straightforward: |
402 |
we calculate the waiting queue function of the (previous) state @{text s}; |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
403 |
this waiting queue function @{text wq} is unchanged in the next schedule state---because |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
404 |
none of these events lock or release any resources; |
296 | 405 |
for calculating the next @{term "cprec_fun"}, we use @{text wq} and the function |
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
406 |
@{term cpreced}. This gives the following three clauses for @{term schs}: |
290 | 407 |
|
408 |
\begin{isabelle}\ \ \ \ \ %%% |
|
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
409 |
\begin{tabular}{@ {}l} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
410 |
@{thm (lhs) schs.simps(2)} @{text "\<equiv>"}\\ |
294
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
411 |
\hspace{5mm}@{text "let"} @{text "wq = wq_fun (schs s)"} @{text "in"}\\ |
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
412 |
\hspace{8mm}@{term "(|wq_fun = wq\<iota>, cprec_fun = cpreced wq\<iota> (Create th prio # s)|)"}\smallskip\\ |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
413 |
@{thm (lhs) schs.simps(3)} @{text "\<equiv>"}\\ |
294
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
414 |
\hspace{5mm}@{text "let"} @{text "wq = wq_fun (schs s)"} @{text "in"}\\ |
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
415 |
\hspace{8mm}@{term "(|wq_fun = wq\<iota>, cprec_fun = cpreced wq\<iota> (Exit th # s)|)"}\smallskip\\ |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
416 |
@{thm (lhs) schs.simps(4)} @{text "\<equiv>"}\\ |
294
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
417 |
\hspace{5mm}@{text "let"} @{text "wq = wq_fun (schs s)"} @{text "in"}\\ |
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
418 |
\hspace{8mm}@{term "(|wq_fun = wq\<iota>, cprec_fun = cpreced wq\<iota> (Set th prio # s)|)"} |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
419 |
\end{tabular} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
420 |
\end{isabelle} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
421 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
422 |
\noindent |
300 | 423 |
More interesting are the cases when a resource, say @{text cs}, is locked or released. In these cases |
424 |
we need to calculate a new waiting queue function. For the event @{term "P th cs"}, we have to update |
|
296 | 425 |
the function so that the new thread list for @{text cs} is old thread list plus the thread @{text th} |
300 | 426 |
appended to the end of that list (remember the head of this list is seen to be in the possession of this |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
427 |
resource). |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
428 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
429 |
\begin{isabelle}\ \ \ \ \ %%% |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
430 |
\begin{tabular}{@ {}l} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
431 |
@{thm (lhs) schs.simps(5)} @{text "\<equiv>"}\\ |
294
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
432 |
\hspace{5mm}@{text "let"} @{text "wq = wq_fun (schs s)"} @{text "in"}\\ |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
433 |
\hspace{5mm}@{text "let"} @{text "new_wq = wq(cs := (wq cs @ [th]))"} @{text "in"}\\ |
294
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
434 |
\hspace{8mm}@{term "(|wq_fun = new_wq, cprec_fun = cpreced new_wq (P th cs # s)|)"} |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
435 |
\end{tabular} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
436 |
\end{isabelle} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
437 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
438 |
\noindent |
300 | 439 |
The clause for event @{term "V th cs"} is similar, except that we need to update the waiting queue function |
301 | 440 |
so that the thread that possessed the lock is deleted from the corresponding thread list. For this |
441 |
list transformation, we use |
|
296 | 442 |
the auxiliary function @{term release}. A simple version of @{term release} would |
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
443 |
just delete this thread and return the rest, namely |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
444 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
445 |
\begin{isabelle}\ \ \ \ \ %%% |
296 | 446 |
\begin{tabular}{@ {}lcl} |
447 |
@{term "release []"} & @{text "\<equiv>"} & @{term "[]"}\\ |
|
448 |
@{term "release (DUMMY # qs)"} & @{text "\<equiv>"} & @{term "qs"}\\ |
|
449 |
\end{tabular} |
|
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
450 |
\end{isabelle} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
451 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
452 |
\noindent |
300 | 453 |
In practice, however, often the thread with the highest precedence in the list will get the |
296 | 454 |
lock next. We have implemented this choice, but later found out that the choice |
300 | 455 |
of which thread is chosen next is actually irrelevant for the correctness of PIP. |
296 | 456 |
Therefore we prove the stronger result where @{term release} is defined as |
457 |
||
458 |
\begin{isabelle}\ \ \ \ \ %%% |
|
459 |
\begin{tabular}{@ {}lcl} |
|
460 |
@{term "release []"} & @{text "\<equiv>"} & @{term "[]"}\\ |
|
461 |
@{term "release (DUMMY # qs)"} & @{text "\<equiv>"} & @{term "SOME qs'. distinct qs' \<and> set qs' = set qs"}\\ |
|
462 |
\end{tabular} |
|
463 |
\end{isabelle} |
|
464 |
||
465 |
\noindent |
|
466 |
@{text "SOME"} stands for Hilbert's epsilon and implements an arbitrary |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
467 |
choice for the next waiting list. It just has to be a list of distinctive threads and |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
468 |
contain the same elements as @{text "qs"}. This gives for @{term V} the clause: |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
469 |
|
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
470 |
\begin{isabelle}\ \ \ \ \ %%% |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
471 |
\begin{tabular}{@ {}l} |
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
472 |
@{thm (lhs) schs.simps(6)} @{text "\<equiv>"}\\ |
294
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
473 |
\hspace{5mm}@{text "let"} @{text "wq = wq_fun (schs s)"} @{text "in"}\\ |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
474 |
\hspace{5mm}@{text "let"} @{text "new_wq = release (wq cs)"} @{text "in"}\\ |
294
bc5bf9e9ada2
renamed waiting_queue -> wq_fun; cur_preced -> cprec_fun
urbanc
parents:
293
diff
changeset
|
475 |
\hspace{8mm}@{term "(|wq_fun = new_wq, cprec_fun = cpreced new_wq (V th cs # s)|)"} |
291
5ef9f6ebe827
more on paper; modified schs functions; it is still compatible with the old definition
urbanc
parents:
290
diff
changeset
|
476 |
\end{tabular} |
290 | 477 |
\end{isabelle} |
478 |
||
300 | 479 |
Having the scheduler function @{term schs} at our disposal, we can ``lift'', or |
480 |
overload, the notions |
|
481 |
@{term waiting}, @{term holding}, @{term depend} and @{term cp} to operate on states only. |
|
286 | 482 |
|
483 |
\begin{isabelle}\ \ \ \ \ %%% |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
484 |
\begin{tabular}{@ {}rcl} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
485 |
@{thm (lhs) s_holding_abv} & @{text "\<equiv>"} & @{thm (rhs) s_holding_abv}\\ |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
486 |
@{thm (lhs) s_waiting_abv} & @{text "\<equiv>"} & @{thm (rhs) s_waiting_abv}\\ |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
487 |
@{thm (lhs) s_depend_abv} & @{text "\<equiv>"} & @{thm (rhs) s_depend_abv}\\ |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
488 |
@{thm (lhs) cp_def} & @{text "\<equiv>"} & @{thm (rhs) cp_def} |
287 | 489 |
\end{tabular} |
490 |
\end{isabelle} |
|
491 |
||
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
492 |
\noindent |
300 | 493 |
With these abbreviations we can introduce |
494 |
the notion of threads being @{term readys} in a state (i.e.~threads |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
495 |
that do not wait for any resource) and the running thread. |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
496 |
|
287 | 497 |
\begin{isabelle}\ \ \ \ \ %%% |
498 |
\begin{tabular}{@ {}l} |
|
499 |
@{thm readys_def}\\ |
|
500 |
@{thm runing_def}\\ |
|
286 | 501 |
\end{tabular} |
502 |
\end{isabelle} |
|
284 | 503 |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
504 |
\noindent |
299 | 505 |
In this definition @{term "f ` S"} stands for the set @{text S} under the image of the |
506 |
function @{text f}. |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
507 |
Note that in the initial case, that is where the list of events is empty, the set |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
508 |
@{term threads} is empty and therefore there is no thread ready nor a running. |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
509 |
If there is one or more threads ready, then there can only be \emph{one} thread |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
510 |
running, namely the one whose current precedence is equal to the maximum of all ready |
304 | 511 |
threads. We use the set-comprehension to capture both possibilities. |
300 | 512 |
We can now also define the set of resources that are locked by a thread in any |
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
513 |
given state. |
284 | 514 |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
515 |
\begin{isabelle}\ \ \ \ \ %%% |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
516 |
@{thm holdents_def} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
517 |
\end{isabelle} |
284 | 518 |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
519 |
\noindent |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
520 |
These resources are given by the holding edges in the RAG. |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
521 |
|
304 | 522 |
Finally we can define what a \emph{valid state} is in our PIP. For |
523 |
example we cannot expect to be able to exit a thread, if it was not |
|
524 |
created yet. These validity constraints are characterised by the |
|
525 |
inductive predicate @{term "step"}. We give five inference rules |
|
526 |
relating a state and an event that can happen next. |
|
284 | 527 |
|
528 |
\begin{center} |
|
529 |
\begin{tabular}{c} |
|
530 |
@{thm[mode=Rule] thread_create[where thread=th]}\hspace{1cm} |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
531 |
@{thm[mode=Rule] thread_exit[where thread=th]} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
532 |
\end{tabular} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
533 |
\end{center} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
534 |
|
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
535 |
\noindent |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
536 |
The first rule states that a thread can only be created, if it does not yet exists. |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
537 |
Similarly, the second rule states that a thread can only be terminated if it was |
301 | 538 |
running and does not lock any resources anymore (to simplify ). The event @{text Set} can happen |
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
539 |
if the corresponding thread is running. |
284 | 540 |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
541 |
\begin{center} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
542 |
@{thm[mode=Rule] thread_set[where thread=th]} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
543 |
\end{center} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
544 |
|
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
545 |
\noindent |
301 | 546 |
If a thread wants to lock a resource, then the thread needs to be |
547 |
running and also we have to make sure that the resource lock does |
|
548 |
not lead to a cycle in the RAG. In practice, ensuring the latter is |
|
549 |
of course the responsibility of the programmer. Here in our formal |
|
550 |
model we just exclude such problematic cases in order to make |
|
551 |
some meaningful statements about PIP.\footnote{This situation is |
|
552 |
similar to the infamous occurs check in Prolog: in order to say |
|
553 |
anything meaningful about unification, we need to perform an occurs |
|
554 |
check; but in practice the occurs check is ommited and the |
|
555 |
responsibility to avoid problems rests with the programmer.} |
|
556 |
Similarly, if a thread wants to release a lock on a resource, then |
|
557 |
it must be running and in the possession of that lock. This is |
|
558 |
formally given by the last two inference rules of @{term step}. |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
559 |
|
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
560 |
\begin{center} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
561 |
\begin{tabular}{c} |
284 | 562 |
@{thm[mode=Rule] thread_P[where thread=th]}\medskip\\ |
563 |
@{thm[mode=Rule] thread_V[where thread=th]}\\ |
|
564 |
\end{tabular} |
|
565 |
\end{center} |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
566 |
|
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
567 |
\noindent |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
568 |
A valid state of PIP can then be conveniently be defined as follows: |
284 | 569 |
|
570 |
\begin{center} |
|
571 |
\begin{tabular}{c} |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
572 |
@{thm[mode=Axiom] vt_nil}\hspace{1cm} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
573 |
@{thm[mode=Rule] vt_cons} |
284 | 574 |
\end{tabular} |
575 |
\end{center} |
|
576 |
||
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
577 |
\noindent |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
578 |
This completes our formal model of PIP. In the next section we present |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
579 |
properties that show our version of PIP is correct. |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
580 |
*} |
274 | 581 |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
582 |
section {* Correctness Proof *} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
583 |
|
301 | 584 |
|
585 |
(*<*) |
|
586 |
context extend_highest_gen |
|
587 |
begin |
|
588 |
(*>*) |
|
589 |
||
590 |
print_locale extend_highest_gen |
|
591 |
thm extend_highest_gen_def |
|
592 |
thm extend_highest_gen_axioms_def |
|
593 |
thm highest_gen_def |
|
594 |
text {* |
|
595 |
Main lemma |
|
596 |
||
597 |
\begin{enumerate} |
|
598 |
\item @{term "s"} is a valid state (@{text "vt_s"}): |
|
599 |
@{thm vt_s}. |
|
600 |
\item @{term "th"} is a living thread in @{term "s"} (@{text "threads_s"}): |
|
601 |
@{thm threads_s}. |
|
602 |
\item @{term "th"} has the highest precedence in @{term "s"} (@{text "highest"}): |
|
603 |
@{thm highest}. |
|
604 |
\item The precedence of @{term "th"} is @{term "Prc prio tm"} (@{text "preced_th"}): |
|
605 |
@{thm preced_th}. |
|
304 | 606 |
|
607 |
\item @{term "t"} is a valid extension of @{term "s"} (@{text "vt_t"}): @{thm vt_t}. |
|
608 |
\item Any thread created in @{term "t"} has priority no higher than @{term "prio"}, therefore |
|
609 |
its precedence can not be higher than @{term "th"}, therefore |
|
610 |
@{term "th"} remain to be the one with the highest precedence |
|
611 |
(@{text "create_low"}): |
|
612 |
@{thm [display] create_low} |
|
613 |
\item Any adjustment of priority in |
|
614 |
@{term "t"} does not happen to @{term "th"} and |
|
615 |
the priority set is no higher than @{term "prio"}, therefore |
|
616 |
@{term "th"} remain to be the one with the highest precedence (@{text "set_diff_low"}): |
|
617 |
@{thm [display] set_diff_low} |
|
618 |
\item Since we are investigating what happens to @{term "th"}, it is assumed |
|
619 |
@{term "th"} does not exit during @{term "t"} (@{text "exit_diff"}): |
|
620 |
@{thm [display] exit_diff} |
|
301 | 621 |
\end{enumerate} |
622 |
||
623 |
\begin{lemma} |
|
624 |
@{thm[mode=IfThen] moment_blocked} |
|
625 |
\end{lemma} |
|
626 |
||
627 |
\begin{theorem} |
|
628 |
@{thm[mode=IfThen] runing_inversion_2} |
|
629 |
\end{theorem} |
|
630 |
||
304 | 631 |
\begin{theorem} |
632 |
@{thm[mode=IfThen] runing_inversion_3} |
|
633 |
\end{theorem} |
|
301 | 634 |
|
635 |
||
636 |
||
637 |
TO DO |
|
638 |
||
639 |
*} |
|
640 |
||
641 |
(*<*) |
|
642 |
end |
|
643 |
(*>*) |
|
298
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
644 |
|
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
645 |
section {* Properties for an Implementation *} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
646 |
|
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
647 |
text {* TO DO *} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
648 |
|
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
649 |
section {* Conclusion *} |
f2e0d031a395
completed model section; vt has only state as argument
urbanc
parents:
297
diff
changeset
|
650 |
|
300 | 651 |
text {* |
652 |
The Priority Inheritance Protocol is a classic textbook algorithm |
|
653 |
used in real-time systems in order to avoid the problem of Priority |
|
654 |
Inversion. |
|
655 |
||
301 | 656 |
A clear and simple understanding of the problem at hand is both a |
657 |
prerequisite and a byproduct of such an effort, because everything |
|
658 |
has finally be reduced to the very first principle to be checked |
|
659 |
mechanically. |
|
660 |
||
304 | 661 |
Our formalisation and the one presented |
662 |
in \cite{Wang09} are the only ones that employ Paulson's method for |
|
663 |
verifying protocols which are \emph{not} security related. |
|
664 |
||
300 | 665 |
TO DO |
666 |
||
301 | 667 |
no clue about multi-processor case according to \cite{Steinberg10} |
668 |
||
300 | 669 |
*} |
273 | 670 |
|
280 | 671 |
text {* |
672 |
\bigskip |
|
284 | 673 |
The priority inversion phenomenon was first published in |
674 |
\cite{Lampson80}. The two protocols widely used to eliminate |
|
675 |
priority inversion, namely PI (Priority Inheritance) and PCE |
|
676 |
(Priority Ceiling Emulation), were proposed in \cite{Sha90}. PCE is |
|
677 |
less convenient to use because it requires static analysis of |
|
678 |
programs. Therefore, PI is more commonly used in |
|
679 |
practice\cite{locke-july02}. However, as pointed out in the |
|
680 |
literature, the analysis of priority inheritance protocol is quite |
|
681 |
subtle\cite{yodaiken-july02}. A formal analysis will certainly be |
|
682 |
helpful for us to understand and correctly implement PI. All |
|
683 |
existing formal analysis of PI |
|
304 | 684 |
\cite{Jahier09,Wellings07,Faria08} are based on the |
284 | 685 |
model checking technology. Because of the state explosion problem, |
686 |
model check is much like an exhaustive testing of finite models with |
|
687 |
limited size. The results obtained can not be safely generalized to |
|
688 |
models with arbitrarily large size. Worse still, since model |
|
689 |
checking is fully automatic, it give little insight on why the |
|
690 |
formal model is correct. It is therefore definitely desirable to |
|
691 |
analyze PI using theorem proving, which gives more general results |
|
692 |
as well as deeper insight. And this is the purpose of this paper |
|
693 |
which gives a formal analysis of PI in the interactive theorem |
|
694 |
prover Isabelle using Higher Order Logic (HOL). The formalization |
|
262 | 695 |
focuses on on two issues: |
696 |
||
697 |
\begin{enumerate} |
|
698 |
\item The correctness of the protocol model itself. A series of desirable properties is |
|
699 |
derived until we are fully convinced that the formal model of PI does |
|
700 |
eliminate priority inversion. And a better understanding of PI is so obtained |
|
701 |
in due course. For example, we find through formalization that the choice of |
|
702 |
next thread to take hold when a |
|
703 |
resource is released is irrelevant for the very basic property of PI to hold. |
|
704 |
A point never mentioned in literature. |
|
705 |
\item The correctness of the implementation. A series of properties is derived the meaning |
|
706 |
of which can be used as guidelines on how PI can be implemented efficiently and correctly. |
|
707 |
\end{enumerate} |
|
708 |
||
709 |
The rest of the paper is organized as follows: Section \ref{overview} gives an overview |
|
710 |
of PI. Section \ref{model} introduces the formal model of PI. Section \ref{general} |
|
711 |
discusses a series of basic properties of PI. Section \ref{extension} shows formally |
|
712 |
how priority inversion is controlled by PI. Section \ref{implement} gives properties |
|
713 |
which can be used for guidelines of implementation. Section \ref{related} discusses |
|
714 |
related works. Section \ref{conclusion} concludes the whole paper. |
|
265 | 715 |
|
273 | 716 |
The basic priority inheritance protocol has two problems: |
717 |
||
718 |
It does not prevent a deadlock from happening in a program with circular lock dependencies. |
|
719 |
||
720 |
A chain of blocking may be formed; blocking duration can be substantial, though bounded. |
|
721 |
||
265 | 722 |
|
723 |
Contributions |
|
724 |
||
725 |
Despite the wide use of Priority Inheritance Protocol in real time operating |
|
726 |
system, it's correctness has never been formally proved and mechanically checked. |
|
727 |
All existing verification are based on model checking technology. Full automatic |
|
728 |
verification gives little help to understand why the protocol is correct. |
|
729 |
And results such obtained only apply to models of limited size. |
|
730 |
This paper presents a formal verification based on theorem proving. |
|
731 |
Machine checked formal proof does help to get deeper understanding. We found |
|
732 |
the fact which is not mentioned in the literature, that the choice of next |
|
733 |
thread to take over when an critical resource is release does not affect the correctness |
|
734 |
of the protocol. The paper also shows how formal proof can help to construct |
|
735 |
correct and efficient implementation.\bigskip |
|
736 |
||
262 | 737 |
*} |
738 |
||
739 |
section {* An overview of priority inversion and priority inheritance \label{overview} *} |
|
740 |
||
741 |
text {* |
|
742 |
||
743 |
Priority inversion refers to the phenomenon when a thread with high priority is blocked |
|
744 |
by a thread with low priority. Priority happens when the high priority thread requests |
|
745 |
for some critical resource already taken by the low priority thread. Since the high |
|
746 |
priority thread has to wait for the low priority thread to complete, it is said to be |
|
747 |
blocked by the low priority thread. Priority inversion might prevent high priority |
|
748 |
thread from fulfill its task in time if the duration of priority inversion is indefinite |
|
749 |
and unpredictable. Indefinite priority inversion happens when indefinite number |
|
750 |
of threads with medium priorities is activated during the period when the high |
|
751 |
priority thread is blocked by the low priority thread. Although these medium |
|
752 |
priority threads can not preempt the high priority thread directly, they are able |
|
753 |
to preempt the low priority threads and cause it to stay in critical section for |
|
754 |
an indefinite long duration. In this way, the high priority thread may be blocked indefinitely. |
|
755 |
||
756 |
Priority inheritance is one protocol proposed to avoid indefinite priority inversion. |
|
757 |
The basic idea is to let the high priority thread donate its priority to the low priority |
|
758 |
thread holding the critical resource, so that it will not be preempted by medium priority |
|
759 |
threads. The thread with highest priority will not be blocked unless it is requesting |
|
760 |
some critical resource already taken by other threads. Viewed from a different angle, |
|
761 |
any thread which is able to block the highest priority threads must already hold some |
|
762 |
critical resource. Further more, it must have hold some critical resource at the |
|
763 |
moment the highest priority is created, otherwise, it may never get change to run and |
|
764 |
get hold. Since the number of such resource holding lower priority threads is finite, |
|
765 |
if every one of them finishes with its own critical section in a definite duration, |
|
766 |
the duration the highest priority thread is blocked is definite as well. The key to |
|
767 |
guarantee lower priority threads to finish in definite is to donate them the highest |
|
768 |
priority. In such cases, the lower priority threads is said to have inherited the |
|
769 |
highest priority. And this explains the name of the protocol: |
|
770 |
{\em Priority Inheritance} and how Priority Inheritance prevents indefinite delay. |
|
771 |
||
772 |
The objectives of this paper are: |
|
773 |
\begin{enumerate} |
|
774 |
\item Build the above mentioned idea into formal model and prove a series of properties |
|
775 |
until we are convinced that the formal model does fulfill the original idea. |
|
776 |
\item Show how formally derived properties can be used as guidelines for correct |
|
777 |
and efficient implementation. |
|
778 |
\end{enumerate} |
|
779 |
The proof is totally formal in the sense that every detail is reduced to the |
|
780 |
very first principles of Higher Order Logic. The nature of interactive theorem |
|
781 |
proving is for the human user to persuade computer program to accept its arguments. |
|
782 |
A clear and simple understanding of the problem at hand is both a prerequisite and a |
|
783 |
byproduct of such an effort, because everything has finally be reduced to the very |
|
784 |
first principle to be checked mechanically. The former intuitive explanation of |
|
785 |
Priority Inheritance is just such a byproduct. |
|
786 |
*} |
|
787 |
||
788 |
section {* Formal model of Priority Inheritance \label{model} *} |
|
789 |
text {* |
|
790 |
\input{../../generated/PrioGDef} |
|
791 |
*} |
|
792 |
||
793 |
section {* General properties of Priority Inheritance \label{general} *} |
|
264 | 794 |
|
795 |
text {* |
|
796 |
The following are several very basic prioprites: |
|
797 |
\begin{enumerate} |
|
798 |
\item All runing threads must be ready (@{text "runing_ready"}): |
|
799 |
@{thm[display] "runing_ready"} |
|
800 |
\item All ready threads must be living (@{text "readys_threads"}): |
|
801 |
@{thm[display] "readys_threads"} |
|
802 |
\item There are finite many living threads at any moment (@{text "finite_threads"}): |
|
803 |
@{thm[display] "finite_threads"} |
|
804 |
\item Every waiting queue does not contain duplcated elements (@{text "wq_distinct"}): |
|
805 |
@{thm[display] "wq_distinct"} |
|
806 |
\item All threads in waiting queues are living threads (@{text "wq_threads"}): |
|
807 |
@{thm[display] "wq_threads"} |
|
808 |
\item The event which can get a thread into waiting queue must be @{term "P"}-events |
|
809 |
(@{text "block_pre"}): |
|
810 |
@{thm[display] "block_pre"} |
|
811 |
\item A thread may never wait for two different critical resources |
|
812 |
(@{text "waiting_unique"}): |
|
813 |
@{thm[display] waiting_unique[of _ _ "cs\<^isub>1" "cs\<^isub>2"]} |
|
814 |
\item Every resource can only be held by one thread |
|
815 |
(@{text "held_unique"}): |
|
816 |
@{thm[display] held_unique[of _ "th\<^isub>1" _ "th\<^isub>2"]} |
|
817 |
\item Every living thread has an unique precedence |
|
818 |
(@{text "preced_unique"}): |
|
819 |
@{thm[display] preced_unique[of "th\<^isub>1" _ "th\<^isub>2"]} |
|
820 |
\end{enumerate} |
|
821 |
*} |
|
822 |
||
823 |
text {* \noindent |
|
824 |
The following lemmas show how RAG is changed with the execution of events: |
|
825 |
\begin{enumerate} |
|
826 |
\item Execution of @{term "Set"} does not change RAG (@{text "depend_set_unchanged"}): |
|
827 |
@{thm[display] depend_set_unchanged} |
|
828 |
\item Execution of @{term "Create"} does not change RAG (@{text "depend_create_unchanged"}): |
|
829 |
@{thm[display] depend_create_unchanged} |
|
830 |
\item Execution of @{term "Exit"} does not change RAG (@{text "depend_exit_unchanged"}): |
|
831 |
@{thm[display] depend_exit_unchanged} |
|
832 |
\item Execution of @{term "P"} (@{text "step_depend_p"}): |
|
833 |
@{thm[display] step_depend_p} |
|
834 |
\item Execution of @{term "V"} (@{text "step_depend_v"}): |
|
835 |
@{thm[display] step_depend_v} |
|
836 |
\end{enumerate} |
|
837 |
*} |
|
838 |
||
839 |
text {* \noindent |
|
840 |
These properties are used to derive the following important results about RAG: |
|
841 |
\begin{enumerate} |
|
842 |
\item RAG is loop free (@{text "acyclic_depend"}): |
|
843 |
@{thm [display] acyclic_depend} |
|
844 |
\item RAGs are finite (@{text "finite_depend"}): |
|
845 |
@{thm [display] finite_depend} |
|
846 |
\item Reverse paths in RAG are well founded (@{text "wf_dep_converse"}): |
|
847 |
@{thm [display] wf_dep_converse} |
|
848 |
\item The dependence relation represented by RAG has a tree structure (@{text "unique_depend"}): |
|
849 |
@{thm [display] unique_depend[of _ _ "n\<^isub>1" "n\<^isub>2"]} |
|
850 |
\item All threads in RAG are living threads |
|
851 |
(@{text "dm_depend_threads"} and @{text "range_in"}): |
|
852 |
@{thm [display] dm_depend_threads range_in} |
|
853 |
\end{enumerate} |
|
854 |
*} |
|
855 |
||
856 |
text {* \noindent |
|
857 |
The following lemmas show how every node in RAG can be chased to ready threads: |
|
858 |
\begin{enumerate} |
|
859 |
\item Every node in RAG can be chased to a ready thread (@{text "chain_building"}): |
|
860 |
@{thm [display] chain_building[rule_format]} |
|
861 |
\item The ready thread chased to is unique (@{text "dchain_unique"}): |
|
862 |
@{thm [display] dchain_unique[of _ _ "th\<^isub>1" "th\<^isub>2"]} |
|
863 |
\end{enumerate} |
|
864 |
*} |
|
865 |
||
866 |
text {* \noindent |
|
867 |
Properties about @{term "next_th"}: |
|
868 |
\begin{enumerate} |
|
869 |
\item The thread taking over is different from the thread which is releasing |
|
870 |
(@{text "next_th_neq"}): |
|
871 |
@{thm [display] next_th_neq} |
|
872 |
\item The thread taking over is unique |
|
873 |
(@{text "next_th_unique"}): |
|
874 |
@{thm [display] next_th_unique[of _ _ _ "th\<^isub>1" "th\<^isub>2"]} |
|
875 |
\end{enumerate} |
|
876 |
*} |
|
877 |
||
878 |
text {* \noindent |
|
879 |
Some deeper results about the system: |
|
880 |
\begin{enumerate} |
|
881 |
\item There can only be one running thread (@{text "runing_unique"}): |
|
882 |
@{thm [display] runing_unique[of _ "th\<^isub>1" "th\<^isub>2"]} |
|
883 |
\item The maximum of @{term "cp"} and @{term "preced"} are equal (@{text "max_cp_eq"}): |
|
884 |
@{thm [display] max_cp_eq} |
|
885 |
\item There must be one ready thread having the max @{term "cp"}-value |
|
886 |
(@{text "max_cp_readys_threads"}): |
|
887 |
@{thm [display] max_cp_readys_threads} |
|
888 |
\end{enumerate} |
|
889 |
*} |
|
890 |
||
891 |
text {* \noindent |
|
892 |
The relationship between the count of @{text "P"} and @{text "V"} and the number of |
|
893 |
critical resources held by a thread is given as follows: |
|
894 |
\begin{enumerate} |
|
895 |
\item The @{term "V"}-operation decreases the number of critical resources |
|
896 |
one thread holds (@{text "cntCS_v_dec"}) |
|
897 |
@{thm [display] cntCS_v_dec} |
|
898 |
\item The number of @{text "V"} never exceeds the number of @{text "P"} |
|
899 |
(@{text "cnp_cnv_cncs"}): |
|
900 |
@{thm [display] cnp_cnv_cncs} |
|
901 |
\item The number of @{text "V"} equals the number of @{text "P"} when |
|
902 |
the relevant thread is not living: |
|
903 |
(@{text "cnp_cnv_eq"}): |
|
904 |
@{thm [display] cnp_cnv_eq} |
|
905 |
\item When a thread is not living, it does not hold any critical resource |
|
906 |
(@{text "not_thread_holdents"}): |
|
907 |
@{thm [display] not_thread_holdents} |
|
908 |
\item When the number of @{text "P"} equals the number of @{text "V"}, the relevant |
|
909 |
thread does not hold any critical resource, therefore no thread can depend on it |
|
910 |
(@{text "count_eq_dependents"}): |
|
911 |
@{thm [display] count_eq_dependents} |
|
912 |
\end{enumerate} |
|
913 |
*} |
|
262 | 914 |
|
915 |
section {* Key properties \label{extension} *} |
|
916 |
||
264 | 917 |
(*<*) |
918 |
context extend_highest_gen |
|
919 |
begin |
|
920 |
(*>*) |
|
921 |
||
922 |
text {* |
|
923 |
The essential of {\em Priority Inheritance} is to avoid indefinite priority inversion. For this |
|
924 |
purpose, we need to investigate what happens after one thread takes the highest precedence. |
|
925 |
A locale is used to describe such a situation, which assumes: |
|
926 |
\begin{enumerate} |
|
927 |
\item @{term "s"} is a valid state (@{text "vt_s"}): |
|
928 |
@{thm vt_s}. |
|
929 |
\item @{term "th"} is a living thread in @{term "s"} (@{text "threads_s"}): |
|
930 |
@{thm threads_s}. |
|
931 |
\item @{term "th"} has the highest precedence in @{term "s"} (@{text "highest"}): |
|
932 |
@{thm highest}. |
|
933 |
\item The precedence of @{term "th"} is @{term "Prc prio tm"} (@{text "preced_th"}): |
|
934 |
@{thm preced_th}. |
|
935 |
\end{enumerate} |
|
936 |
*} |
|
937 |
||
938 |
text {* \noindent |
|
939 |
Under these assumptions, some basic priority can be derived for @{term "th"}: |
|
940 |
\begin{enumerate} |
|
941 |
\item The current precedence of @{term "th"} equals its own precedence (@{text "eq_cp_s_th"}): |
|
942 |
@{thm [display] eq_cp_s_th} |
|
943 |
\item The current precedence of @{term "th"} is the highest precedence in |
|
944 |
the system (@{text "highest_cp_preced"}): |
|
945 |
@{thm [display] highest_cp_preced} |
|
946 |
\item The precedence of @{term "th"} is the highest precedence |
|
947 |
in the system (@{text "highest_preced_thread"}): |
|
948 |
@{thm [display] highest_preced_thread} |
|
949 |
\item The current precedence of @{term "th"} is the highest current precedence |
|
950 |
in the system (@{text "highest'"}): |
|
951 |
@{thm [display] highest'} |
|
952 |
\end{enumerate} |
|
953 |
*} |
|
954 |
||
955 |
text {* \noindent |
|
956 |
To analysis what happens after state @{term "s"} a sub-locale is defined, which |
|
957 |
assumes: |
|
958 |
\begin{enumerate} |
|
959 |
\item @{term "t"} is a valid extension of @{term "s"} (@{text "vt_t"}): @{thm vt_t}. |
|
960 |
\item Any thread created in @{term "t"} has priority no higher than @{term "prio"}, therefore |
|
961 |
its precedence can not be higher than @{term "th"}, therefore |
|
962 |
@{term "th"} remain to be the one with the highest precedence |
|
963 |
(@{text "create_low"}): |
|
964 |
@{thm [display] create_low} |
|
965 |
\item Any adjustment of priority in |
|
966 |
@{term "t"} does not happen to @{term "th"} and |
|
967 |
the priority set is no higher than @{term "prio"}, therefore |
|
968 |
@{term "th"} remain to be the one with the highest precedence (@{text "set_diff_low"}): |
|
969 |
@{thm [display] set_diff_low} |
|
970 |
\item Since we are investigating what happens to @{term "th"}, it is assumed |
|
971 |
@{term "th"} does not exit during @{term "t"} (@{text "exit_diff"}): |
|
972 |
@{thm [display] exit_diff} |
|
973 |
\end{enumerate} |
|
974 |
*} |
|
975 |
||
976 |
text {* \noindent |
|
977 |
All these assumptions are put into a predicate @{term "extend_highest_gen"}. |
|
978 |
It can be proved that @{term "extend_highest_gen"} holds |
|
979 |
for any moment @{text "i"} in it @{term "t"} (@{text "red_moment"}): |
|
980 |
@{thm [display] red_moment} |
|
981 |
||
982 |
From this, an induction principle can be derived for @{text "t"}, so that |
|
983 |
properties already derived for @{term "t"} can be applied to any prefix |
|
984 |
of @{text "t"} in the proof of new properties |
|
985 |
about @{term "t"} (@{text "ind"}): |
|
986 |
\begin{center} |
|
987 |
@{thm[display] ind} |
|
988 |
\end{center} |
|
989 |
||
990 |
The following properties can be proved about @{term "th"} in @{term "t"}: |
|
991 |
\begin{enumerate} |
|
992 |
\item In @{term "t"}, thread @{term "th"} is kept live and its |
|
993 |
precedence is preserved as well |
|
994 |
(@{text "th_kept"}): |
|
995 |
@{thm [display] th_kept} |
|
996 |
\item In @{term "t"}, thread @{term "th"}'s precedence is always the maximum among |
|
997 |
all living threads |
|
998 |
(@{text "max_preced"}): |
|
999 |
@{thm [display] max_preced} |
|
1000 |
\item In @{term "t"}, thread @{term "th"}'s current precedence is always the maximum precedence |
|
1001 |
among all living threads |
|
1002 |
(@{text "th_cp_max_preced"}): |
|
1003 |
@{thm [display] th_cp_max_preced} |
|
1004 |
\item In @{term "t"}, thread @{term "th"}'s current precedence is always the maximum current |
|
1005 |
precedence among all living threads |
|
1006 |
(@{text "th_cp_max"}): |
|
1007 |
@{thm [display] th_cp_max} |
|
1008 |
\item In @{term "t"}, thread @{term "th"}'s current precedence equals its precedence at moment |
|
1009 |
@{term "s"} |
|
1010 |
(@{text "th_cp_preced"}): |
|
1011 |
@{thm [display] th_cp_preced} |
|
1012 |
\end{enumerate} |
|
1013 |
*} |
|
1014 |
||
1015 |
text {* \noindent |
|
266 | 1016 |
The main theorem of this part is to characterizing the running thread during @{term "t"} |
264 | 1017 |
(@{text "runing_inversion_2"}): |
1018 |
@{thm [display] runing_inversion_2} |
|
1019 |
According to this, if a thread is running, it is either @{term "th"} or was |
|
1020 |
already live and held some resource |
|
1021 |
at moment @{text "s"} (expressed by: @{text "cntV s th' < cntP s th'"}). |
|
1022 |
||
1023 |
Since there are only finite many threads live and holding some resource at any moment, |
|
1024 |
if every such thread can release all its resources in finite duration, then after finite |
|
1025 |
duration, none of them may block @{term "th"} anymore. So, no priority inversion may happen |
|
1026 |
then. |
|
1027 |
*} |
|
1028 |
||
1029 |
(*<*) |
|
1030 |
end |
|
1031 |
(*>*) |
|
1032 |
||
262 | 1033 |
section {* Properties to guide implementation \label{implement} *} |
1034 |
||
264 | 1035 |
text {* |
266 | 1036 |
The properties (especially @{text "runing_inversion_2"}) convinced us that the model defined |
1037 |
in Section \ref{model} does prevent indefinite priority inversion and therefore fulfills |
|
264 | 1038 |
the fundamental requirement of Priority Inheritance protocol. Another purpose of this paper |
266 | 1039 |
is to show how this model can be used to guide a concrete implementation. As discussed in |
276 | 1040 |
Section 5.6.5 of \cite{Vahalia96}, the implementation of Priority Inheritance in Solaris |
266 | 1041 |
uses sophisticated linking data structure. Except discussing two scenarios to show how |
1042 |
the data structure should be manipulated, a lot of details of the implementation are missing. |
|
304 | 1043 |
In \cite{Faria08,Jahier09,Wellings07} the protocol is described formally |
266 | 1044 |
using different notations, but little information is given on how this protocol can be |
1045 |
implemented efficiently, especially there is no information on how these data structure |
|
1046 |
should be manipulated. |
|
1047 |
||
1048 |
Because the scheduling of threads is based on current precedence, |
|
1049 |
the central issue in implementation of Priority Inheritance is how to compute the precedence |
|
1050 |
correctly and efficiently. As long as the precedence is correct, it is very easy to |
|
1051 |
modify the scheduling algorithm to select the correct thread to execute. |
|
1052 |
||
1053 |
First, it can be proved that the computation of current precedence @{term "cp"} of a threads |
|
1054 |
only involves its children (@{text "cp_rec"}): |
|
1055 |
@{thm [display] cp_rec} |
|
1056 |
where @{term "children s th"} represents the set of children of @{term "th"} in the current |
|
1057 |
RAG: |
|
1058 |
\[ |
|
1059 |
@{thm (lhs) children_def} @{text "\<equiv>"} @{thm (rhs) children_def} |
|
1060 |
\] |
|
1061 |
where the definition of @{term "child"} is: |
|
1062 |
\[ @{thm (lhs) child_def} @{text "\<equiv>"} @{thm (rhs) child_def} |
|
1063 |
\] |
|
1064 |
||
1065 |
The aim of this section is to fill the missing details of how current precedence should |
|
1066 |
be changed with the happening of events, with each event type treated by one subsection, |
|
1067 |
where the computation of @{term "cp"} uses lemma @{text "cp_rec"}. |
|
1068 |
*} |
|
1069 |
||
1070 |
subsection {* Event @{text "Set th prio"} *} |
|
1071 |
||
1072 |
(*<*) |
|
1073 |
context step_set_cps |
|
1074 |
begin |
|
1075 |
(*>*) |
|
1076 |
||
1077 |
text {* |
|
1078 |
The context under which event @{text "Set th prio"} happens is formalized as follows: |
|
1079 |
\begin{enumerate} |
|
1080 |
\item The formation of @{term "s"} (@{text "s_def"}): @{thm s_def}. |
|
1081 |
\item State @{term "s"} is a valid state (@{text "vt_s"}): @{thm vt_s}. This implies |
|
1082 |
event @{text "Set th prio"} is eligible to happen under state @{term "s'"} and |
|
1083 |
state @{term "s'"} is a valid state. |
|
1084 |
\end{enumerate} |
|
264 | 1085 |
*} |
1086 |
||
266 | 1087 |
text {* \noindent |
1088 |
Under such a context, we investigated how the current precedence @{term "cp"} of |
|
1089 |
threads change from state @{term "s'"} to @{term "s"} and obtained the following |
|
1090 |
conclusions: |
|
1091 |
\begin{enumerate} |
|
1092 |
%% \item The RAG does not change (@{text "eq_dep"}): @{thm "eq_dep"}. |
|
1093 |
\item All threads with no dependence relation with thread @{term "th"} have their |
|
1094 |
@{term "cp"}-value unchanged (@{text "eq_cp"}): |
|
1095 |
@{thm [display] eq_cp} |
|
1096 |
This lemma implies the @{term "cp"}-value of @{term "th"} |
|
1097 |
and those threads which have a dependence relation with @{term "th"} might need |
|
1098 |
to be recomputed. The way to do this is to start from @{term "th"} |
|
1099 |
and follow the @{term "depend"}-chain to recompute the @{term "cp"}-value of every |
|
1100 |
encountered thread using lemma @{text "cp_rec"}. |
|
1101 |
Since the @{term "depend"}-relation is loop free, this procedure |
|
1102 |
can always stop. The the following lemma shows this procedure actually could stop earlier. |
|
1103 |
\item The following two lemma shows, if a thread the re-computation of which |
|
1104 |
gives an unchanged @{term "cp"}-value, the procedure described above can stop. |
|
1105 |
\begin{enumerate} |
|
1106 |
\item Lemma @{text "eq_up_self"} shows if the re-computation of |
|
1107 |
@{term "th"}'s @{term "cp"} gives the same result, the procedure can stop: |
|
1108 |
@{thm [display] eq_up_self} |
|
1109 |
\item Lemma @{text "eq_up"}) shows if the re-computation at intermediate threads |
|
1110 |
gives unchanged result, the procedure can stop: |
|
1111 |
@{thm [display] eq_up} |
|
1112 |
\end{enumerate} |
|
1113 |
\end{enumerate} |
|
1114 |
*} |
|
1115 |
||
1116 |
(*<*) |
|
1117 |
end |
|
1118 |
(*>*) |
|
264 | 1119 |
|
272 | 1120 |
subsection {* Event @{text "V th cs"} *} |
1121 |
||
1122 |
(*<*) |
|
1123 |
context step_v_cps_nt |
|
1124 |
begin |
|
1125 |
(*>*) |
|
1126 |
||
1127 |
text {* |
|
1128 |
The context under which event @{text "V th cs"} happens is formalized as follows: |
|
1129 |
\begin{enumerate} |
|
1130 |
\item The formation of @{term "s"} (@{text "s_def"}): @{thm s_def}. |
|
1131 |
\item State @{term "s"} is a valid state (@{text "vt_s"}): @{thm vt_s}. This implies |
|
1132 |
event @{text "V th cs"} is eligible to happen under state @{term "s'"} and |
|
1133 |
state @{term "s'"} is a valid state. |
|
1134 |
\end{enumerate} |
|
1135 |
*} |
|
1136 |
||
1137 |
text {* \noindent |
|
1138 |
Under such a context, we investigated how the current precedence @{term "cp"} of |
|
1139 |
threads change from state @{term "s'"} to @{term "s"}. |
|
1140 |
||
1141 |
||
1142 |
Two subcases are considerted, |
|
1143 |
where the first is that there exits @{term "th'"} |
|
1144 |
such that |
|
1145 |
@{thm [display] nt} |
|
1146 |
holds, which means there exists a thread @{term "th'"} to take over |
|
1147 |
the resource release by thread @{term "th"}. |
|
1148 |
In this sub-case, the following results are obtained: |
|
1149 |
\begin{enumerate} |
|
1150 |
\item The change of RAG is given by lemma @{text "depend_s"}: |
|
1151 |
@{thm [display] "depend_s"} |
|
1152 |
which shows two edges are removed while one is added. These changes imply how |
|
1153 |
the current precedences should be re-computed. |
|
1154 |
\item First all threads different from @{term "th"} and @{term "th'"} have their |
|
1155 |
@{term "cp"}-value kept, therefore do not need a re-computation |
|
1156 |
(@{text "cp_kept"}): @{thm [display] cp_kept} |
|
1157 |
This lemma also implies, only the @{term "cp"}-values of @{term "th"} and @{term "th'"} |
|
1158 |
need to be recomputed. |
|
1159 |
\end{enumerate} |
|
1160 |
*} |
|
1161 |
||
1162 |
(*<*) |
|
1163 |
end |
|
1164 |
||
1165 |
context step_v_cps_nnt |
|
1166 |
begin |
|
1167 |
(*>*) |
|
1168 |
||
1169 |
text {* |
|
1170 |
The other sub-case is when for all @{text "th'"} |
|
1171 |
@{thm [display] nnt} |
|
1172 |
holds, no such thread exists. The following results can be obtained for this |
|
1173 |
sub-case: |
|
1174 |
\begin{enumerate} |
|
1175 |
\item The change of RAG is given by lemma @{text "depend_s"}: |
|
1176 |
@{thm [display] depend_s} |
|
1177 |
which means only one edge is removed. |
|
1178 |
\item In this case, no re-computation is needed (@{text "eq_cp"}): |
|
1179 |
@{thm [display] eq_cp} |
|
1180 |
\end{enumerate} |
|
1181 |
*} |
|
1182 |
||
1183 |
(*<*) |
|
1184 |
end |
|
1185 |
(*>*) |
|
1186 |
||
1187 |
||
1188 |
subsection {* Event @{text "P th cs"} *} |
|
1189 |
||
1190 |
(*<*) |
|
1191 |
context step_P_cps_e |
|
1192 |
begin |
|
1193 |
(*>*) |
|
1194 |
||
1195 |
text {* |
|
1196 |
The context under which event @{text "P th cs"} happens is formalized as follows: |
|
1197 |
\begin{enumerate} |
|
1198 |
\item The formation of @{term "s"} (@{text "s_def"}): @{thm s_def}. |
|
1199 |
\item State @{term "s"} is a valid state (@{text "vt_s"}): @{thm vt_s}. This implies |
|
1200 |
event @{text "P th cs"} is eligible to happen under state @{term "s'"} and |
|
1201 |
state @{term "s'"} is a valid state. |
|
1202 |
\end{enumerate} |
|
1203 |
||
1204 |
This case is further divided into two sub-cases. The first is when @{thm ee} holds. |
|
1205 |
The following results can be obtained: |
|
1206 |
\begin{enumerate} |
|
1207 |
\item One edge is added to the RAG (@{text "depend_s"}): |
|
1208 |
@{thm [display] depend_s} |
|
1209 |
\item No re-computation is needed (@{text "eq_cp"}): |
|
1210 |
@{thm [display] eq_cp} |
|
1211 |
\end{enumerate} |
|
1212 |
*} |
|
1213 |
||
1214 |
(*<*) |
|
1215 |
end |
|
1216 |
||
1217 |
context step_P_cps_ne |
|
1218 |
begin |
|
1219 |
(*>*) |
|
1220 |
||
1221 |
text {* |
|
1222 |
The second is when @{thm ne} holds. |
|
1223 |
The following results can be obtained: |
|
1224 |
\begin{enumerate} |
|
1225 |
\item One edge is added to the RAG (@{text "depend_s"}): |
|
1226 |
@{thm [display] depend_s} |
|
1227 |
\item Threads with no dependence relation with @{term "th"} do not need a re-computation |
|
1228 |
of their @{term "cp"}-values (@{text "eq_cp"}): |
|
1229 |
@{thm [display] eq_cp} |
|
1230 |
This lemma implies all threads with a dependence relation with @{term "th"} may need |
|
1231 |
re-computation. |
|
1232 |
\item Similar to the case of @{term "Set"}, the computation procedure could stop earlier |
|
1233 |
(@{text "eq_up"}): |
|
1234 |
@{thm [display] eq_up} |
|
1235 |
\end{enumerate} |
|
1236 |
||
1237 |
*} |
|
1238 |
||
1239 |
(*<*) |
|
1240 |
end |
|
1241 |
(*>*) |
|
1242 |
||
1243 |
subsection {* Event @{text "Create th prio"} *} |
|
1244 |
||
1245 |
(*<*) |
|
1246 |
context step_create_cps |
|
1247 |
begin |
|
1248 |
(*>*) |
|
1249 |
||
1250 |
text {* |
|
1251 |
The context under which event @{text "Create th prio"} happens is formalized as follows: |
|
1252 |
\begin{enumerate} |
|
1253 |
\item The formation of @{term "s"} (@{text "s_def"}): @{thm s_def}. |
|
1254 |
\item State @{term "s"} is a valid state (@{text "vt_s"}): @{thm vt_s}. This implies |
|
1255 |
event @{text "Create th prio"} is eligible to happen under state @{term "s'"} and |
|
1256 |
state @{term "s'"} is a valid state. |
|
1257 |
\end{enumerate} |
|
1258 |
The following results can be obtained under this context: |
|
1259 |
\begin{enumerate} |
|
1260 |
\item The RAG does not change (@{text "eq_dep"}): |
|
1261 |
@{thm [display] eq_dep} |
|
1262 |
\item All threads other than @{term "th"} do not need re-computation (@{text "eq_cp"}): |
|
1263 |
@{thm [display] eq_cp} |
|
1264 |
\item The @{term "cp"}-value of @{term "th"} equals its precedence |
|
1265 |
(@{text "eq_cp_th"}): |
|
1266 |
@{thm [display] eq_cp_th} |
|
1267 |
\end{enumerate} |
|
1268 |
||
1269 |
*} |
|
1270 |
||
1271 |
||
1272 |
(*<*) |
|
1273 |
end |
|
1274 |
(*>*) |
|
1275 |
||
1276 |
subsection {* Event @{text "Exit th"} *} |
|
1277 |
||
1278 |
(*<*) |
|
1279 |
context step_exit_cps |
|
1280 |
begin |
|
1281 |
(*>*) |
|
1282 |
||
1283 |
text {* |
|
1284 |
The context under which event @{text "Exit th"} happens is formalized as follows: |
|
1285 |
\begin{enumerate} |
|
1286 |
\item The formation of @{term "s"} (@{text "s_def"}): @{thm s_def}. |
|
1287 |
\item State @{term "s"} is a valid state (@{text "vt_s"}): @{thm vt_s}. This implies |
|
1288 |
event @{text "Exit th"} is eligible to happen under state @{term "s'"} and |
|
1289 |
state @{term "s'"} is a valid state. |
|
1290 |
\end{enumerate} |
|
1291 |
The following results can be obtained under this context: |
|
1292 |
\begin{enumerate} |
|
1293 |
\item The RAG does not change (@{text "eq_dep"}): |
|
1294 |
@{thm [display] eq_dep} |
|
1295 |
\item All threads other than @{term "th"} do not need re-computation (@{text "eq_cp"}): |
|
1296 |
@{thm [display] eq_cp} |
|
1297 |
\end{enumerate} |
|
1298 |
Since @{term th} does not live in state @{term "s"}, there is no need to compute |
|
1299 |
its @{term cp}-value. |
|
1300 |
*} |
|
1301 |
||
1302 |
(*<*) |
|
1303 |
end |
|
1304 |
(*>*) |
|
1305 |
||
1306 |
||
262 | 1307 |
section {* Related works \label{related} *} |
1308 |
||
1309 |
text {* |
|
1310 |
\begin{enumerate} |
|
1311 |
\item {\em Integrating Priority Inheritance Algorithms in the Real-Time Specification for Java} |
|
304 | 1312 |
\cite{Wellings07} models and verifies the combination of Priority Inheritance (PI) and |
262 | 1313 |
Priority Ceiling Emulation (PCE) protocols in the setting of Java virtual machine |
1314 |
using extended Timed Automata(TA) formalism of the UPPAAL tool. Although a detailed |
|
1315 |
formal model of combined PI and PCE is given, the number of properties is quite |
|
1316 |
small and the focus is put on the harmonious working of PI and PCE. Most key features of PI |
|
1317 |
(as well as PCE) are not shown. Because of the limitation of the model checking technique |
|
1318 |
used there, properties are shown only for a small number of scenarios. Therefore, |
|
1319 |
the verification does not show the correctness of the formal model itself in a |
|
1320 |
convincing way. |
|
1321 |
\item {\em Formal Development of Solutions for Real-Time Operating Systems with TLA+/TLC} |
|
1322 |
\cite{Faria08}. A formal model of PI is given in TLA+. Only 3 properties are shown |
|
1323 |
for PI using model checking. The limitation of model checking is intrinsic to the work. |
|
1324 |
\item {\em Synchronous modeling and validation of priority inheritance schedulers} |
|
304 | 1325 |
\cite{Jahier09}. Gives a formal model |
262 | 1326 |
of PI and PCE in AADL (Architecture Analysis \& Design Language) and checked |
1327 |
several properties using model checking. The number of properties shown there is |
|
1328 |
less than here and the scale is also limited by the model checking technique. |
|
1329 |
\item {\em The Priority Ceiling Protocol: Formalization and Analysis Using PVS} |
|
1330 |
\cite{dutertre99b}. Formalized another protocol for Priority Inversion in the |
|
1331 |
interactive theorem proving system PVS. |
|
1332 |
\end{enumerate} |
|
1333 |
||
1334 |
||
1335 |
There are several works on inversion avoidance: |
|
1336 |
\begin{enumerate} |
|
1337 |
\item {\em Solving the group priority inversion problem in a timed asynchronous system} |
|
1338 |
\cite{Wang:2002:SGP}. The notion of Group Priority Inversion is introduced. The main |
|
1339 |
strategy is still inversion avoidance. The method is by reordering requests |
|
1340 |
in the setting of Client-Server. |
|
1341 |
\item {\em A Formalization of Priority Inversion} \cite{journals/rts/BabaogluMS93}. |
|
1342 |
Formalized the notion of Priority |
|
1343 |
Inversion and proposes methods to avoid it. |
|
1344 |
\end{enumerate} |
|
1345 |
||
1346 |
{\em Examples of inaccurate specification of the protocol ???}. |
|
1347 |
||
1348 |
*} |
|
1349 |
||
1350 |
section {* Conclusions \label{conclusion} *} |
|
1351 |
||
286 | 1352 |
text {* |
1353 |
The work in this paper only deals with single CPU configurations. The |
|
1354 |
"one CPU" assumption is essential for our formalisation, because the |
|
1355 |
main lemma fails in multi-CPU configuration. The lemma says that any |
|
1356 |
runing thead must be the one with the highest prioirty or already held |
|
1357 |
some resource when the highest priority thread was initiated. When |
|
1358 |
there are multiple CPUs, it may well be the case that a threads did |
|
1359 |
not hold any resource when the highest priority thread was initiated, |
|
1360 |
but that thread still runs after that moment on a separate CPU. In |
|
1361 |
this way, the main lemma does not hold anymore. |
|
1362 |
||
1363 |
||
1364 |
There are some works deals with priority inversion in multi-CPU |
|
1365 |
configurations[???], but none of them have given a formal correctness |
|
1366 |
proof. The extension of our formal proof to deal with multi-CPU |
|
1367 |
configurations is not obvious. One possibility, as suggested in paper |
|
1368 |
[???], is change our formal model (the defiintion of "schs") to give |
|
1369 |
the released resource to the thread with the highest prioirty. In this |
|
1370 |
way, indefinite prioirty inversion can be avoided, but for a quite |
|
1371 |
different reason from the one formalized in this paper (because the |
|
1372 |
"mail lemma" will be different). This means a formal correctness proof |
|
1373 |
for milt-CPU configuration would be quite different from the one given |
|
1374 |
in this paper. The solution of prioirty inversion problem in mult-CPU |
|
1375 |
configurations is a different problem which needs different solutions |
|
1376 |
which is outside the scope of this paper. |
|
1377 |
||
1378 |
*} |
|
1379 |
||
262 | 1380 |
(*<*) |
1381 |
end |
|
1382 |
(*>*) |