262
|
1 |
(*<*)
|
|
2 |
theory Paper
|
272
|
3 |
imports CpsG ExtGG (* "~~/src/HOL/Library/LaTeXsugar" *) 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
|
|
14 |
birthtime ("inception") and
|
|
15 |
If ("(\<^raw:\textrm{>if\<^raw:}> (_)/ \<^raw:\textrm{>then\<^raw:}> (_)/ \<^raw:\textrm{>else\<^raw:}> (_))" 10) and
|
|
16 |
DUMMY ("\<^raw:\mbox{$\_\!\_$}>")
|
262
|
17 |
(*>*)
|
|
18 |
|
|
19 |
section {* Introduction *}
|
|
20 |
|
|
21 |
text {*
|
284
|
22 |
Many real-time systems need to support threads involving priorities and
|
267
|
23 |
locking of resources. Locking of resources ensures mutual exclusion
|
275
|
24 |
when accessing shared data or devices that cannot be
|
284
|
25 |
preempted. Priorities allow scheduling of threads that need to
|
275
|
26 |
finish their work within deadlines. Unfortunately, both features
|
|
27 |
can interact in subtle ways leading to a problem, called
|
284
|
28 |
\emph{Priority Inversion}. Suppose three threads having priorities
|
|
29 |
$H$(igh), $M$(edium) and $L$(ow). We would expect that the thread
|
|
30 |
$H$ blocks any other thread with lower priority and itself cannot
|
|
31 |
be blocked by any thread with lower priority. Alas, in a naive
|
275
|
32 |
implementation of resource looking and priorities this property can
|
|
33 |
be violated. Even worse, $H$ can be delayed indefinitely by
|
284
|
34 |
threads with lower priorities. For this let $L$ be in the
|
275
|
35 |
possession of a lock for a resource that also $H$ needs. $H$ must
|
|
36 |
therefore wait for $L$ to exit the critical section and release this
|
|
37 |
lock. The problem is that $L$ might in turn be blocked by any
|
284
|
38 |
thread with priority $M$, and so $H$ sits there potentially waiting
|
|
39 |
indefinitely. Since $H$ is blocked by threads with lower
|
275
|
40 |
priorities, the problem is called Priority Inversion. It was first
|
277
|
41 |
described in \cite{Lampson80} in the context of the
|
275
|
42 |
Mesa programming language designed for concurrent programming.
|
265
|
43 |
|
273
|
44 |
If the problem of Priority Inversion is ignored, real-time systems
|
267
|
45 |
can become unpredictable and resulting bugs can be hard to diagnose.
|
|
46 |
The classic example where this happened is the software that
|
284
|
47 |
controlled the Mars Pathfinder mission in 1997 \cite{Reeves98}.
|
|
48 |
Once the spacecraft landed, the software shut down at irregular
|
|
49 |
intervals leading to loss of project time as normal operation of the
|
|
50 |
craft could only resume the next day (the mission and data already
|
|
51 |
collected were fortunately not lost, because of a clever system
|
|
52 |
design). The reason for the shutdowns was that the scheduling
|
|
53 |
software fell victim of Priority Inversion: a low priority thread
|
|
54 |
locking a resource prevented a high priority thread from running in
|
|
55 |
time leading to a system reset. Once the problem was found, it was
|
|
56 |
rectified by enabling the \emph{Priority Inheritance Protocol} (PIP)
|
|
57 |
\cite{Sha90}\footnote{Sha et al.~call it the \emph{Basic Priority
|
|
58 |
Inheritance Protocol} \cite{Sha90} and others sometimes call it
|
|
59 |
also \emph{Priority Boosting}.} in the scheduling software.
|
262
|
60 |
|
284
|
61 |
The idea behind PIP is to let the thread $L$ temporarily inherit
|
|
62 |
the high priority from $H$ until $L$ leaves the critical section by
|
|
63 |
unlocking the resource. This solves the problem of $H$ having to
|
|
64 |
wait indefinitely, because $L$ cannot be blocked by threads having
|
|
65 |
priority $M$. While a few other solutions exist for the Priority
|
|
66 |
Inversion problem, PIP is one that is widely deployed and
|
|
67 |
implemented. This includes VxWorks (a proprietary real-time OS used
|
|
68 |
in the Mars Pathfinder mission, in Boeing's 787 Dreamliner, Honda's
|
|
69 |
ASIMO robot, etc.), but also the POSIX 1003.1c Standard realised for
|
|
70 |
example in libraries for FreeBSD, Solaris and Linux.
|
274
|
71 |
|
284
|
72 |
One advantage of PIP is that increasing the priority of a thread
|
275
|
73 |
can be dynamically calculated by the scheduler. This is in contrast
|
277
|
74 |
to, for example, \emph{Priority Ceiling} \cite{Sha90}, another
|
|
75 |
solution to the Priority Inversion problem, which requires static
|
284
|
76 |
analysis of the program in order to prevent Priority
|
|
77 |
Inversion. However, there has also been strong criticism against
|
|
78 |
PIP. For instance, PIP cannot prevent deadlocks when lock
|
|
79 |
dependencies are circular, and also blocking times can be
|
|
80 |
substantial (more than just the duration of a critical section).
|
|
81 |
Though, most criticism against PIP centres around unreliable
|
|
82 |
implementations and PIP being too complicated and too inefficient.
|
|
83 |
For example, Yodaiken writes in \cite{Yodaiken02}:
|
274
|
84 |
|
|
85 |
\begin{quote}
|
|
86 |
\it{}``Priority inheritance is neither efficient nor reliable. Implementations
|
|
87 |
are either incomplete (and unreliable) or surprisingly complex and intrusive.''
|
|
88 |
\end{quote}
|
273
|
89 |
|
274
|
90 |
\noindent
|
275
|
91 |
He suggests to avoid PIP altogether by not allowing critical
|
277
|
92 |
sections to be preempted. While this might have been a reasonable
|
|
93 |
solution in 2002, in our modern multiprocessor world, this seems out
|
278
|
94 |
of date. The reason is that this precludes other high-priority
|
284
|
95 |
threads from running even when they do not make any use of the locked
|
278
|
96 |
resource.
|
|
97 |
|
|
98 |
However, there is clearly a need for investigating correct
|
|
99 |
algorithms for PIP. A few specifications for PIP exist (in English)
|
|
100 |
and also a few high-level descriptions of implementations (e.g.~in
|
|
101 |
the textbook \cite[Section 5.6.5]{Vahalia96}), but they help little
|
|
102 |
with actual implementations. That this is a problem in practise is
|
283
|
103 |
proved by an email from Baker, who wrote on 13 July 2009 on the Linux
|
278
|
104 |
Kernel mailing list:
|
274
|
105 |
|
|
106 |
\begin{quote}
|
275
|
107 |
\it{}``I observed in the kernel code (to my disgust), the Linux PIP
|
|
108 |
implementation is a nightmare: extremely heavy weight, involving
|
|
109 |
maintenance of a full wait-for graph, and requiring updates for a
|
|
110 |
range of events, including priority changes and interruptions of
|
|
111 |
wait operations.''
|
274
|
112 |
\end{quote}
|
|
113 |
|
|
114 |
\noindent
|
277
|
115 |
The criticism by Yodaiken, Baker and others suggests to us to look
|
|
116 |
again at PIP from a more abstract level (but still concrete enough
|
|
117 |
to inform an implementation) and makes PIP an ideal candidate for a
|
|
118 |
formal verification. One reason, of course, is that the original
|
284
|
119 |
presentation of PIP~\cite{Sha90}, despite being informally
|
283
|
120 |
``proved'' correct, is actually \emph{flawed}.
|
|
121 |
|
|
122 |
Yodaiken \cite{Yodaiken02} points to a subtlety that had been
|
|
123 |
overlooked in the informal proof by Sha et al. They specify in
|
284
|
124 |
\cite{Sha90} that after the thread (whose priority has been raised)
|
283
|
125 |
completes its critical section and releases the lock, it ``returns
|
|
126 |
to its original priority level.'' This leads them to believe that an
|
284
|
127 |
implementation of PIP is ``rather straightforward''~\cite{Sha90}.
|
|
128 |
Unfortunately, as Yodaiken points out, this behaviour is too
|
|
129 |
simplistic. Consider the case where the low priority thread $L$
|
|
130 |
locks \emph{two} resources, and two high-priority threads $H$ and
|
283
|
131 |
$H'$ each wait for one of them. If $L$ then releases one resource
|
|
132 |
so that $H$, say, can proceed, then we still have Priority Inversion
|
|
133 |
with $H'$ (which waits for the other resource). The correct
|
|
134 |
behaviour for $L$ is to revert to the highest remaining priority of
|
284
|
135 |
the threads that it blocks. The advantage of formalising the
|
|
136 |
correctness of a high-level specification of PIP in a theorem prover
|
|
137 |
is that such issues clearly show up and cannot be overlooked as in
|
|
138 |
informal reasoning (since we have to analyse all possible behaviours
|
|
139 |
of threads, i.e.~\emph{traces}, that could possibly happen).
|
274
|
140 |
|
279
|
141 |
There have been earlier formal investigations into PIP, but ...\cite{Faria08}
|
284
|
142 |
|
|
143 |
vt (valid trace) was introduced earlier, cite
|
|
144 |
|
|
145 |
distributed PIP
|
280
|
146 |
*}
|
278
|
147 |
|
283
|
148 |
section {* Formal Model of the Priority Inheritance Protocol *}
|
267
|
149 |
|
280
|
150 |
text {*
|
284
|
151 |
We follow the original work by Sha et al.~\cite{Sha90} by modelling
|
|
152 |
first a classical single CPU system where only one \emph{thread} is
|
|
153 |
active at any given moment. We shall discuss later how to lift this
|
|
154 |
restriction. Our model of PIP is based on Paulson's inductive
|
|
155 |
approach to protocol verification \cite{Paulson98}, where the
|
|
156 |
\emph{state} of a system is given by a list of events that
|
|
157 |
happened so far. \emph{Events} fall into four categories defined as
|
|
158 |
the datatype
|
283
|
159 |
|
|
160 |
\begin{isabelle}\ \ \ \ \ %%%
|
284
|
161 |
\mbox{\begin{tabular}{r@ {\hspace{2mm}}c@ {\hspace{2mm}}l@ {\hspace{7mm}}l}
|
|
162 |
\isacommand{datatype} event
|
|
163 |
& @{text "="} & @{term "Create thread priority"}\\
|
|
164 |
& @{text "|"} & @{term "Exit thread"} \\
|
|
165 |
& @{text "|"} & @{term "Set thread priority"} & {\rm reset of the proprity for} @{text thread}\\
|
|
166 |
& @{text "|"} & @{term "P thread cs"} & {\rm request of resource} @{text "cs"} {\rm by} @{text "thread"}\\
|
|
167 |
& @{text "|"} & @{term "V thread cs"} & {\rm release of resource} @{text "cs"} {\rm by} @{text "thread"}
|
|
168 |
\end{tabular}}
|
|
169 |
\end{isabelle}
|
|
170 |
|
|
171 |
\noindent
|
|
172 |
whereby threads, priorities and (critical) resources are represented
|
|
173 |
as natural numbers. As in Paulson's work, we need to define
|
|
174 |
functions that allow one to make some observations about states.
|
|
175 |
One is the ``live'' threads we have seen so far in a state:
|
|
176 |
|
|
177 |
\begin{isabelle}\ \ \ \ \ %%%
|
|
178 |
\mbox{\begin{tabular}{lcl}
|
|
179 |
@{thm (lhs) threads.simps(1)} & @{text "\<equiv>"} &
|
|
180 |
@{thm (rhs) threads.simps(1)}\\
|
|
181 |
@{thm (lhs) threads.simps(2)[where thread="th"]} & @{text "\<equiv>"} &
|
|
182 |
@{thm (rhs) threads.simps(2)[where thread="th"]}\\
|
|
183 |
@{thm (lhs) threads.simps(3)[where thread="th"]} & @{text "\<equiv>"} &
|
|
184 |
@{thm (rhs) threads.simps(3)[where thread="th"]}\\
|
|
185 |
@{term "threads (DUMMY#s)"} & @{text "\<equiv>"} & @{term "threads s"}\\
|
|
186 |
\end{tabular}}
|
283
|
187 |
\end{isabelle}
|
|
188 |
|
|
189 |
\noindent
|
284
|
190 |
Another is that given a thread @{text "th"}, what is the original priority was at
|
|
191 |
its inception.
|
|
192 |
|
|
193 |
\begin{isabelle}\ \ \ \ \ %%%
|
|
194 |
\mbox{\begin{tabular}{lcl}
|
|
195 |
@{thm (lhs) original_priority.simps(1)[where thread="th"]} & @{text "\<equiv>"} &
|
|
196 |
@{thm (rhs) original_priority.simps(1)[where thread="th"]}\\
|
|
197 |
@{thm (lhs) original_priority.simps(2)[where thread="th" and thread'="th'"]} & @{text "\<equiv>"} &
|
|
198 |
@{thm (rhs) original_priority.simps(2)[where thread="th" and thread'="th'"]}\\
|
|
199 |
@{thm (lhs) original_priority.simps(3)[where thread="th" and thread'="th'"]} & @{text "\<equiv>"} &
|
|
200 |
@{thm (rhs) original_priority.simps(3)[where thread="th" and thread'="th'"]}\\
|
|
201 |
@{term "original_priority th (DUMMY#s)"} & @{text "\<equiv>"} & @{term "original_priority th s"}\\
|
|
202 |
\end{tabular}}
|
|
203 |
\end{isabelle}
|
|
204 |
|
|
205 |
\noindent
|
|
206 |
In this definition we set @{text 0} as the default priority for
|
|
207 |
threads that have not (yet) been created. The last function we need
|
|
208 |
calculates the ``time'', or index, at which a process was created.
|
|
209 |
|
|
210 |
\begin{isabelle}\ \ \ \ \ %%%
|
|
211 |
\mbox{\begin{tabular}{lcl}
|
|
212 |
@{thm (lhs) birthtime.simps(1)[where thread="th"]} & @{text "\<equiv>"} &
|
|
213 |
@{thm (rhs) birthtime.simps(1)[where thread="th"]}\\
|
|
214 |
@{thm (lhs) birthtime.simps(2)[where thread="th" and thread'="th'"]} & @{text "\<equiv>"} &
|
|
215 |
@{thm (rhs) birthtime.simps(2)[where thread="th" and thread'="th'"]}\\
|
|
216 |
@{thm (lhs) birthtime.simps(3)[where thread="th" and thread'="th'"]} & @{text "\<equiv>"} &
|
|
217 |
@{thm (rhs) birthtime.simps(3)[where thread="th" and thread'="th'"]}\\
|
|
218 |
@{term "birthtime th (DUMMY#s)"} & @{text "\<equiv>"} & @{term "birthtime th s"}\\
|
|
219 |
\end{tabular}}
|
|
220 |
\end{isabelle}
|
|
221 |
|
|
222 |
\begin{center}
|
|
223 |
threads, original-priority, birth time, precedence.
|
|
224 |
\end{center}
|
|
225 |
|
|
226 |
|
|
227 |
|
|
228 |
resources
|
|
229 |
|
|
230 |
step relation:
|
|
231 |
|
|
232 |
\begin{center}
|
|
233 |
\begin{tabular}{c}
|
|
234 |
@{thm[mode=Rule] thread_create[where thread=th]}\hspace{1cm}
|
|
235 |
@{thm[mode=Rule] thread_exit[where thread=th]}\medskip\\
|
|
236 |
|
|
237 |
@{thm[mode=Rule] thread_P[where thread=th]}\medskip\\
|
|
238 |
@{thm[mode=Rule] thread_V[where thread=th]}\\
|
|
239 |
\end{tabular}
|
|
240 |
\end{center}
|
|
241 |
|
|
242 |
valid state:
|
|
243 |
|
|
244 |
\begin{center}
|
|
245 |
\begin{tabular}{c}
|
|
246 |
@{thm[mode=Axiom] vt_nil[where cs=step]}\hspace{1cm}
|
|
247 |
@{thm[mode=Rule] vt_cons[where cs=step]}
|
|
248 |
\end{tabular}
|
|
249 |
\end{center}
|
|
250 |
|
274
|
251 |
|
280
|
252 |
To define events, the identifiers of {\em threads},
|
|
253 |
{\em priority} and {\em critical resources } (abbreviated as @{text "cs"})
|
|
254 |
need to be represented. All three are represetned using standard
|
|
255 |
Isabelle/HOL type @{typ "nat"}:
|
|
256 |
*}
|
273
|
257 |
|
280
|
258 |
text {*
|
|
259 |
\bigskip
|
284
|
260 |
The priority inversion phenomenon was first published in
|
|
261 |
\cite{Lampson80}. The two protocols widely used to eliminate
|
|
262 |
priority inversion, namely PI (Priority Inheritance) and PCE
|
|
263 |
(Priority Ceiling Emulation), were proposed in \cite{Sha90}. PCE is
|
|
264 |
less convenient to use because it requires static analysis of
|
|
265 |
programs. Therefore, PI is more commonly used in
|
|
266 |
practice\cite{locke-july02}. However, as pointed out in the
|
|
267 |
literature, the analysis of priority inheritance protocol is quite
|
|
268 |
subtle\cite{yodaiken-july02}. A formal analysis will certainly be
|
|
269 |
helpful for us to understand and correctly implement PI. All
|
|
270 |
existing formal analysis of PI
|
|
271 |
\cite{conf/fase/JahierHR09,WellingsBSB07,Faria08} are based on the
|
|
272 |
model checking technology. Because of the state explosion problem,
|
|
273 |
model check is much like an exhaustive testing of finite models with
|
|
274 |
limited size. The results obtained can not be safely generalized to
|
|
275 |
models with arbitrarily large size. Worse still, since model
|
|
276 |
checking is fully automatic, it give little insight on why the
|
|
277 |
formal model is correct. It is therefore definitely desirable to
|
|
278 |
analyze PI using theorem proving, which gives more general results
|
|
279 |
as well as deeper insight. And this is the purpose of this paper
|
|
280 |
which gives a formal analysis of PI in the interactive theorem
|
|
281 |
prover Isabelle using Higher Order Logic (HOL). The formalization
|
262
|
282 |
focuses on on two issues:
|
|
283 |
|
|
284 |
\begin{enumerate}
|
|
285 |
\item The correctness of the protocol model itself. A series of desirable properties is
|
|
286 |
derived until we are fully convinced that the formal model of PI does
|
|
287 |
eliminate priority inversion. And a better understanding of PI is so obtained
|
|
288 |
in due course. For example, we find through formalization that the choice of
|
|
289 |
next thread to take hold when a
|
|
290 |
resource is released is irrelevant for the very basic property of PI to hold.
|
|
291 |
A point never mentioned in literature.
|
|
292 |
\item The correctness of the implementation. A series of properties is derived the meaning
|
|
293 |
of which can be used as guidelines on how PI can be implemented efficiently and correctly.
|
|
294 |
\end{enumerate}
|
|
295 |
|
|
296 |
The rest of the paper is organized as follows: Section \ref{overview} gives an overview
|
|
297 |
of PI. Section \ref{model} introduces the formal model of PI. Section \ref{general}
|
|
298 |
discusses a series of basic properties of PI. Section \ref{extension} shows formally
|
|
299 |
how priority inversion is controlled by PI. Section \ref{implement} gives properties
|
|
300 |
which can be used for guidelines of implementation. Section \ref{related} discusses
|
|
301 |
related works. Section \ref{conclusion} concludes the whole paper.
|
265
|
302 |
|
273
|
303 |
The basic priority inheritance protocol has two problems:
|
|
304 |
|
|
305 |
It does not prevent a deadlock from happening in a program with circular lock dependencies.
|
|
306 |
|
|
307 |
A chain of blocking may be formed; blocking duration can be substantial, though bounded.
|
|
308 |
|
265
|
309 |
|
|
310 |
Contributions
|
|
311 |
|
|
312 |
Despite the wide use of Priority Inheritance Protocol in real time operating
|
|
313 |
system, it's correctness has never been formally proved and mechanically checked.
|
|
314 |
All existing verification are based on model checking technology. Full automatic
|
|
315 |
verification gives little help to understand why the protocol is correct.
|
|
316 |
And results such obtained only apply to models of limited size.
|
|
317 |
This paper presents a formal verification based on theorem proving.
|
|
318 |
Machine checked formal proof does help to get deeper understanding. We found
|
|
319 |
the fact which is not mentioned in the literature, that the choice of next
|
|
320 |
thread to take over when an critical resource is release does not affect the correctness
|
|
321 |
of the protocol. The paper also shows how formal proof can help to construct
|
|
322 |
correct and efficient implementation.\bigskip
|
|
323 |
|
262
|
324 |
*}
|
|
325 |
|
|
326 |
section {* An overview of priority inversion and priority inheritance \label{overview} *}
|
|
327 |
|
|
328 |
text {*
|
|
329 |
|
|
330 |
Priority inversion refers to the phenomenon when a thread with high priority is blocked
|
|
331 |
by a thread with low priority. Priority happens when the high priority thread requests
|
|
332 |
for some critical resource already taken by the low priority thread. Since the high
|
|
333 |
priority thread has to wait for the low priority thread to complete, it is said to be
|
|
334 |
blocked by the low priority thread. Priority inversion might prevent high priority
|
|
335 |
thread from fulfill its task in time if the duration of priority inversion is indefinite
|
|
336 |
and unpredictable. Indefinite priority inversion happens when indefinite number
|
|
337 |
of threads with medium priorities is activated during the period when the high
|
|
338 |
priority thread is blocked by the low priority thread. Although these medium
|
|
339 |
priority threads can not preempt the high priority thread directly, they are able
|
|
340 |
to preempt the low priority threads and cause it to stay in critical section for
|
|
341 |
an indefinite long duration. In this way, the high priority thread may be blocked indefinitely.
|
|
342 |
|
|
343 |
Priority inheritance is one protocol proposed to avoid indefinite priority inversion.
|
|
344 |
The basic idea is to let the high priority thread donate its priority to the low priority
|
|
345 |
thread holding the critical resource, so that it will not be preempted by medium priority
|
|
346 |
threads. The thread with highest priority will not be blocked unless it is requesting
|
|
347 |
some critical resource already taken by other threads. Viewed from a different angle,
|
|
348 |
any thread which is able to block the highest priority threads must already hold some
|
|
349 |
critical resource. Further more, it must have hold some critical resource at the
|
|
350 |
moment the highest priority is created, otherwise, it may never get change to run and
|
|
351 |
get hold. Since the number of such resource holding lower priority threads is finite,
|
|
352 |
if every one of them finishes with its own critical section in a definite duration,
|
|
353 |
the duration the highest priority thread is blocked is definite as well. The key to
|
|
354 |
guarantee lower priority threads to finish in definite is to donate them the highest
|
|
355 |
priority. In such cases, the lower priority threads is said to have inherited the
|
|
356 |
highest priority. And this explains the name of the protocol:
|
|
357 |
{\em Priority Inheritance} and how Priority Inheritance prevents indefinite delay.
|
|
358 |
|
|
359 |
The objectives of this paper are:
|
|
360 |
\begin{enumerate}
|
|
361 |
\item Build the above mentioned idea into formal model and prove a series of properties
|
|
362 |
until we are convinced that the formal model does fulfill the original idea.
|
|
363 |
\item Show how formally derived properties can be used as guidelines for correct
|
|
364 |
and efficient implementation.
|
|
365 |
\end{enumerate}
|
|
366 |
The proof is totally formal in the sense that every detail is reduced to the
|
|
367 |
very first principles of Higher Order Logic. The nature of interactive theorem
|
|
368 |
proving is for the human user to persuade computer program to accept its arguments.
|
|
369 |
A clear and simple understanding of the problem at hand is both a prerequisite and a
|
|
370 |
byproduct of such an effort, because everything has finally be reduced to the very
|
|
371 |
first principle to be checked mechanically. The former intuitive explanation of
|
|
372 |
Priority Inheritance is just such a byproduct.
|
|
373 |
*}
|
|
374 |
|
|
375 |
section {* Formal model of Priority Inheritance \label{model} *}
|
|
376 |
text {*
|
|
377 |
\input{../../generated/PrioGDef}
|
|
378 |
*}
|
|
379 |
|
|
380 |
section {* General properties of Priority Inheritance \label{general} *}
|
264
|
381 |
|
|
382 |
text {*
|
|
383 |
The following are several very basic prioprites:
|
|
384 |
\begin{enumerate}
|
|
385 |
\item All runing threads must be ready (@{text "runing_ready"}):
|
|
386 |
@{thm[display] "runing_ready"}
|
|
387 |
\item All ready threads must be living (@{text "readys_threads"}):
|
|
388 |
@{thm[display] "readys_threads"}
|
|
389 |
\item There are finite many living threads at any moment (@{text "finite_threads"}):
|
|
390 |
@{thm[display] "finite_threads"}
|
|
391 |
\item Every waiting queue does not contain duplcated elements (@{text "wq_distinct"}):
|
|
392 |
@{thm[display] "wq_distinct"}
|
|
393 |
\item All threads in waiting queues are living threads (@{text "wq_threads"}):
|
|
394 |
@{thm[display] "wq_threads"}
|
|
395 |
\item The event which can get a thread into waiting queue must be @{term "P"}-events
|
|
396 |
(@{text "block_pre"}):
|
|
397 |
@{thm[display] "block_pre"}
|
|
398 |
\item A thread may never wait for two different critical resources
|
|
399 |
(@{text "waiting_unique"}):
|
|
400 |
@{thm[display] waiting_unique[of _ _ "cs\<^isub>1" "cs\<^isub>2"]}
|
|
401 |
\item Every resource can only be held by one thread
|
|
402 |
(@{text "held_unique"}):
|
|
403 |
@{thm[display] held_unique[of _ "th\<^isub>1" _ "th\<^isub>2"]}
|
|
404 |
\item Every living thread has an unique precedence
|
|
405 |
(@{text "preced_unique"}):
|
|
406 |
@{thm[display] preced_unique[of "th\<^isub>1" _ "th\<^isub>2"]}
|
|
407 |
\end{enumerate}
|
|
408 |
*}
|
|
409 |
|
|
410 |
text {* \noindent
|
|
411 |
The following lemmas show how RAG is changed with the execution of events:
|
|
412 |
\begin{enumerate}
|
|
413 |
\item Execution of @{term "Set"} does not change RAG (@{text "depend_set_unchanged"}):
|
|
414 |
@{thm[display] depend_set_unchanged}
|
|
415 |
\item Execution of @{term "Create"} does not change RAG (@{text "depend_create_unchanged"}):
|
|
416 |
@{thm[display] depend_create_unchanged}
|
|
417 |
\item Execution of @{term "Exit"} does not change RAG (@{text "depend_exit_unchanged"}):
|
|
418 |
@{thm[display] depend_exit_unchanged}
|
|
419 |
\item Execution of @{term "P"} (@{text "step_depend_p"}):
|
|
420 |
@{thm[display] step_depend_p}
|
|
421 |
\item Execution of @{term "V"} (@{text "step_depend_v"}):
|
|
422 |
@{thm[display] step_depend_v}
|
|
423 |
\end{enumerate}
|
|
424 |
*}
|
|
425 |
|
|
426 |
text {* \noindent
|
|
427 |
These properties are used to derive the following important results about RAG:
|
|
428 |
\begin{enumerate}
|
|
429 |
\item RAG is loop free (@{text "acyclic_depend"}):
|
|
430 |
@{thm [display] acyclic_depend}
|
|
431 |
\item RAGs are finite (@{text "finite_depend"}):
|
|
432 |
@{thm [display] finite_depend}
|
|
433 |
\item Reverse paths in RAG are well founded (@{text "wf_dep_converse"}):
|
|
434 |
@{thm [display] wf_dep_converse}
|
|
435 |
\item The dependence relation represented by RAG has a tree structure (@{text "unique_depend"}):
|
|
436 |
@{thm [display] unique_depend[of _ _ "n\<^isub>1" "n\<^isub>2"]}
|
|
437 |
\item All threads in RAG are living threads
|
|
438 |
(@{text "dm_depend_threads"} and @{text "range_in"}):
|
|
439 |
@{thm [display] dm_depend_threads range_in}
|
|
440 |
\end{enumerate}
|
|
441 |
*}
|
|
442 |
|
|
443 |
text {* \noindent
|
|
444 |
The following lemmas show how every node in RAG can be chased to ready threads:
|
|
445 |
\begin{enumerate}
|
|
446 |
\item Every node in RAG can be chased to a ready thread (@{text "chain_building"}):
|
|
447 |
@{thm [display] chain_building[rule_format]}
|
|
448 |
\item The ready thread chased to is unique (@{text "dchain_unique"}):
|
|
449 |
@{thm [display] dchain_unique[of _ _ "th\<^isub>1" "th\<^isub>2"]}
|
|
450 |
\end{enumerate}
|
|
451 |
*}
|
|
452 |
|
|
453 |
text {* \noindent
|
|
454 |
Properties about @{term "next_th"}:
|
|
455 |
\begin{enumerate}
|
|
456 |
\item The thread taking over is different from the thread which is releasing
|
|
457 |
(@{text "next_th_neq"}):
|
|
458 |
@{thm [display] next_th_neq}
|
|
459 |
\item The thread taking over is unique
|
|
460 |
(@{text "next_th_unique"}):
|
|
461 |
@{thm [display] next_th_unique[of _ _ _ "th\<^isub>1" "th\<^isub>2"]}
|
|
462 |
\end{enumerate}
|
|
463 |
*}
|
|
464 |
|
|
465 |
text {* \noindent
|
|
466 |
Some deeper results about the system:
|
|
467 |
\begin{enumerate}
|
|
468 |
\item There can only be one running thread (@{text "runing_unique"}):
|
|
469 |
@{thm [display] runing_unique[of _ "th\<^isub>1" "th\<^isub>2"]}
|
|
470 |
\item The maximum of @{term "cp"} and @{term "preced"} are equal (@{text "max_cp_eq"}):
|
|
471 |
@{thm [display] max_cp_eq}
|
|
472 |
\item There must be one ready thread having the max @{term "cp"}-value
|
|
473 |
(@{text "max_cp_readys_threads"}):
|
|
474 |
@{thm [display] max_cp_readys_threads}
|
|
475 |
\end{enumerate}
|
|
476 |
*}
|
|
477 |
|
|
478 |
text {* \noindent
|
|
479 |
The relationship between the count of @{text "P"} and @{text "V"} and the number of
|
|
480 |
critical resources held by a thread is given as follows:
|
|
481 |
\begin{enumerate}
|
|
482 |
\item The @{term "V"}-operation decreases the number of critical resources
|
|
483 |
one thread holds (@{text "cntCS_v_dec"})
|
|
484 |
@{thm [display] cntCS_v_dec}
|
|
485 |
\item The number of @{text "V"} never exceeds the number of @{text "P"}
|
|
486 |
(@{text "cnp_cnv_cncs"}):
|
|
487 |
@{thm [display] cnp_cnv_cncs}
|
|
488 |
\item The number of @{text "V"} equals the number of @{text "P"} when
|
|
489 |
the relevant thread is not living:
|
|
490 |
(@{text "cnp_cnv_eq"}):
|
|
491 |
@{thm [display] cnp_cnv_eq}
|
|
492 |
\item When a thread is not living, it does not hold any critical resource
|
|
493 |
(@{text "not_thread_holdents"}):
|
|
494 |
@{thm [display] not_thread_holdents}
|
|
495 |
\item When the number of @{text "P"} equals the number of @{text "V"}, the relevant
|
|
496 |
thread does not hold any critical resource, therefore no thread can depend on it
|
|
497 |
(@{text "count_eq_dependents"}):
|
|
498 |
@{thm [display] count_eq_dependents}
|
|
499 |
\end{enumerate}
|
|
500 |
*}
|
262
|
501 |
|
|
502 |
section {* Key properties \label{extension} *}
|
|
503 |
|
264
|
504 |
(*<*)
|
|
505 |
context extend_highest_gen
|
|
506 |
begin
|
|
507 |
(*>*)
|
|
508 |
|
|
509 |
text {*
|
|
510 |
The essential of {\em Priority Inheritance} is to avoid indefinite priority inversion. For this
|
|
511 |
purpose, we need to investigate what happens after one thread takes the highest precedence.
|
|
512 |
A locale is used to describe such a situation, which assumes:
|
|
513 |
\begin{enumerate}
|
|
514 |
\item @{term "s"} is a valid state (@{text "vt_s"}):
|
|
515 |
@{thm vt_s}.
|
|
516 |
\item @{term "th"} is a living thread in @{term "s"} (@{text "threads_s"}):
|
|
517 |
@{thm threads_s}.
|
|
518 |
\item @{term "th"} has the highest precedence in @{term "s"} (@{text "highest"}):
|
|
519 |
@{thm highest}.
|
|
520 |
\item The precedence of @{term "th"} is @{term "Prc prio tm"} (@{text "preced_th"}):
|
|
521 |
@{thm preced_th}.
|
|
522 |
\end{enumerate}
|
|
523 |
*}
|
|
524 |
|
|
525 |
text {* \noindent
|
|
526 |
Under these assumptions, some basic priority can be derived for @{term "th"}:
|
|
527 |
\begin{enumerate}
|
|
528 |
\item The current precedence of @{term "th"} equals its own precedence (@{text "eq_cp_s_th"}):
|
|
529 |
@{thm [display] eq_cp_s_th}
|
|
530 |
\item The current precedence of @{term "th"} is the highest precedence in
|
|
531 |
the system (@{text "highest_cp_preced"}):
|
|
532 |
@{thm [display] highest_cp_preced}
|
|
533 |
\item The precedence of @{term "th"} is the highest precedence
|
|
534 |
in the system (@{text "highest_preced_thread"}):
|
|
535 |
@{thm [display] highest_preced_thread}
|
|
536 |
\item The current precedence of @{term "th"} is the highest current precedence
|
|
537 |
in the system (@{text "highest'"}):
|
|
538 |
@{thm [display] highest'}
|
|
539 |
\end{enumerate}
|
|
540 |
*}
|
|
541 |
|
|
542 |
text {* \noindent
|
|
543 |
To analysis what happens after state @{term "s"} a sub-locale is defined, which
|
|
544 |
assumes:
|
|
545 |
\begin{enumerate}
|
|
546 |
\item @{term "t"} is a valid extension of @{term "s"} (@{text "vt_t"}): @{thm vt_t}.
|
|
547 |
\item Any thread created in @{term "t"} has priority no higher than @{term "prio"}, therefore
|
|
548 |
its precedence can not be higher than @{term "th"}, therefore
|
|
549 |
@{term "th"} remain to be the one with the highest precedence
|
|
550 |
(@{text "create_low"}):
|
|
551 |
@{thm [display] create_low}
|
|
552 |
\item Any adjustment of priority in
|
|
553 |
@{term "t"} does not happen to @{term "th"} and
|
|
554 |
the priority set is no higher than @{term "prio"}, therefore
|
|
555 |
@{term "th"} remain to be the one with the highest precedence (@{text "set_diff_low"}):
|
|
556 |
@{thm [display] set_diff_low}
|
|
557 |
\item Since we are investigating what happens to @{term "th"}, it is assumed
|
|
558 |
@{term "th"} does not exit during @{term "t"} (@{text "exit_diff"}):
|
|
559 |
@{thm [display] exit_diff}
|
|
560 |
\end{enumerate}
|
|
561 |
*}
|
|
562 |
|
|
563 |
text {* \noindent
|
|
564 |
All these assumptions are put into a predicate @{term "extend_highest_gen"}.
|
|
565 |
It can be proved that @{term "extend_highest_gen"} holds
|
|
566 |
for any moment @{text "i"} in it @{term "t"} (@{text "red_moment"}):
|
|
567 |
@{thm [display] red_moment}
|
|
568 |
|
|
569 |
From this, an induction principle can be derived for @{text "t"}, so that
|
|
570 |
properties already derived for @{term "t"} can be applied to any prefix
|
|
571 |
of @{text "t"} in the proof of new properties
|
|
572 |
about @{term "t"} (@{text "ind"}):
|
|
573 |
\begin{center}
|
|
574 |
@{thm[display] ind}
|
|
575 |
\end{center}
|
|
576 |
|
|
577 |
The following properties can be proved about @{term "th"} in @{term "t"}:
|
|
578 |
\begin{enumerate}
|
|
579 |
\item In @{term "t"}, thread @{term "th"} is kept live and its
|
|
580 |
precedence is preserved as well
|
|
581 |
(@{text "th_kept"}):
|
|
582 |
@{thm [display] th_kept}
|
|
583 |
\item In @{term "t"}, thread @{term "th"}'s precedence is always the maximum among
|
|
584 |
all living threads
|
|
585 |
(@{text "max_preced"}):
|
|
586 |
@{thm [display] max_preced}
|
|
587 |
\item In @{term "t"}, thread @{term "th"}'s current precedence is always the maximum precedence
|
|
588 |
among all living threads
|
|
589 |
(@{text "th_cp_max_preced"}):
|
|
590 |
@{thm [display] th_cp_max_preced}
|
|
591 |
\item In @{term "t"}, thread @{term "th"}'s current precedence is always the maximum current
|
|
592 |
precedence among all living threads
|
|
593 |
(@{text "th_cp_max"}):
|
|
594 |
@{thm [display] th_cp_max}
|
|
595 |
\item In @{term "t"}, thread @{term "th"}'s current precedence equals its precedence at moment
|
|
596 |
@{term "s"}
|
|
597 |
(@{text "th_cp_preced"}):
|
|
598 |
@{thm [display] th_cp_preced}
|
|
599 |
\end{enumerate}
|
|
600 |
*}
|
|
601 |
|
|
602 |
text {* \noindent
|
266
|
603 |
The main theorem of this part is to characterizing the running thread during @{term "t"}
|
264
|
604 |
(@{text "runing_inversion_2"}):
|
|
605 |
@{thm [display] runing_inversion_2}
|
|
606 |
According to this, if a thread is running, it is either @{term "th"} or was
|
|
607 |
already live and held some resource
|
|
608 |
at moment @{text "s"} (expressed by: @{text "cntV s th' < cntP s th'"}).
|
|
609 |
|
|
610 |
Since there are only finite many threads live and holding some resource at any moment,
|
|
611 |
if every such thread can release all its resources in finite duration, then after finite
|
|
612 |
duration, none of them may block @{term "th"} anymore. So, no priority inversion may happen
|
|
613 |
then.
|
|
614 |
*}
|
|
615 |
|
|
616 |
(*<*)
|
|
617 |
end
|
|
618 |
(*>*)
|
|
619 |
|
262
|
620 |
section {* Properties to guide implementation \label{implement} *}
|
|
621 |
|
264
|
622 |
text {*
|
266
|
623 |
The properties (especially @{text "runing_inversion_2"}) convinced us that the model defined
|
|
624 |
in Section \ref{model} does prevent indefinite priority inversion and therefore fulfills
|
264
|
625 |
the fundamental requirement of Priority Inheritance protocol. Another purpose of this paper
|
266
|
626 |
is to show how this model can be used to guide a concrete implementation. As discussed in
|
276
|
627 |
Section 5.6.5 of \cite{Vahalia96}, the implementation of Priority Inheritance in Solaris
|
266
|
628 |
uses sophisticated linking data structure. Except discussing two scenarios to show how
|
|
629 |
the data structure should be manipulated, a lot of details of the implementation are missing.
|
|
630 |
In \cite{Faria08,conf/fase/JahierHR09,WellingsBSB07} the protocol is described formally
|
|
631 |
using different notations, but little information is given on how this protocol can be
|
|
632 |
implemented efficiently, especially there is no information on how these data structure
|
|
633 |
should be manipulated.
|
|
634 |
|
|
635 |
Because the scheduling of threads is based on current precedence,
|
|
636 |
the central issue in implementation of Priority Inheritance is how to compute the precedence
|
|
637 |
correctly and efficiently. As long as the precedence is correct, it is very easy to
|
|
638 |
modify the scheduling algorithm to select the correct thread to execute.
|
|
639 |
|
|
640 |
First, it can be proved that the computation of current precedence @{term "cp"} of a threads
|
|
641 |
only involves its children (@{text "cp_rec"}):
|
|
642 |
@{thm [display] cp_rec}
|
|
643 |
where @{term "children s th"} represents the set of children of @{term "th"} in the current
|
|
644 |
RAG:
|
|
645 |
\[
|
|
646 |
@{thm (lhs) children_def} @{text "\<equiv>"} @{thm (rhs) children_def}
|
|
647 |
\]
|
|
648 |
where the definition of @{term "child"} is:
|
|
649 |
\[ @{thm (lhs) child_def} @{text "\<equiv>"} @{thm (rhs) child_def}
|
|
650 |
\]
|
|
651 |
|
|
652 |
The aim of this section is to fill the missing details of how current precedence should
|
|
653 |
be changed with the happening of events, with each event type treated by one subsection,
|
|
654 |
where the computation of @{term "cp"} uses lemma @{text "cp_rec"}.
|
|
655 |
*}
|
|
656 |
|
|
657 |
subsection {* Event @{text "Set th prio"} *}
|
|
658 |
|
|
659 |
(*<*)
|
|
660 |
context step_set_cps
|
|
661 |
begin
|
|
662 |
(*>*)
|
|
663 |
|
|
664 |
text {*
|
|
665 |
The context under which event @{text "Set th prio"} happens is formalized as follows:
|
|
666 |
\begin{enumerate}
|
|
667 |
\item The formation of @{term "s"} (@{text "s_def"}): @{thm s_def}.
|
|
668 |
\item State @{term "s"} is a valid state (@{text "vt_s"}): @{thm vt_s}. This implies
|
|
669 |
event @{text "Set th prio"} is eligible to happen under state @{term "s'"} and
|
|
670 |
state @{term "s'"} is a valid state.
|
|
671 |
\end{enumerate}
|
264
|
672 |
*}
|
|
673 |
|
266
|
674 |
text {* \noindent
|
|
675 |
Under such a context, we investigated how the current precedence @{term "cp"} of
|
|
676 |
threads change from state @{term "s'"} to @{term "s"} and obtained the following
|
|
677 |
conclusions:
|
|
678 |
\begin{enumerate}
|
|
679 |
%% \item The RAG does not change (@{text "eq_dep"}): @{thm "eq_dep"}.
|
|
680 |
\item All threads with no dependence relation with thread @{term "th"} have their
|
|
681 |
@{term "cp"}-value unchanged (@{text "eq_cp"}):
|
|
682 |
@{thm [display] eq_cp}
|
|
683 |
This lemma implies the @{term "cp"}-value of @{term "th"}
|
|
684 |
and those threads which have a dependence relation with @{term "th"} might need
|
|
685 |
to be recomputed. The way to do this is to start from @{term "th"}
|
|
686 |
and follow the @{term "depend"}-chain to recompute the @{term "cp"}-value of every
|
|
687 |
encountered thread using lemma @{text "cp_rec"}.
|
|
688 |
Since the @{term "depend"}-relation is loop free, this procedure
|
|
689 |
can always stop. The the following lemma shows this procedure actually could stop earlier.
|
|
690 |
\item The following two lemma shows, if a thread the re-computation of which
|
|
691 |
gives an unchanged @{term "cp"}-value, the procedure described above can stop.
|
|
692 |
\begin{enumerate}
|
|
693 |
\item Lemma @{text "eq_up_self"} shows if the re-computation of
|
|
694 |
@{term "th"}'s @{term "cp"} gives the same result, the procedure can stop:
|
|
695 |
@{thm [display] eq_up_self}
|
|
696 |
\item Lemma @{text "eq_up"}) shows if the re-computation at intermediate threads
|
|
697 |
gives unchanged result, the procedure can stop:
|
|
698 |
@{thm [display] eq_up}
|
|
699 |
\end{enumerate}
|
|
700 |
\end{enumerate}
|
|
701 |
*}
|
|
702 |
|
|
703 |
(*<*)
|
|
704 |
end
|
|
705 |
(*>*)
|
264
|
706 |
|
272
|
707 |
subsection {* Event @{text "V th cs"} *}
|
|
708 |
|
|
709 |
(*<*)
|
|
710 |
context step_v_cps_nt
|
|
711 |
begin
|
|
712 |
(*>*)
|
|
713 |
|
|
714 |
text {*
|
|
715 |
The context under which event @{text "V th cs"} happens is formalized as follows:
|
|
716 |
\begin{enumerate}
|
|
717 |
\item The formation of @{term "s"} (@{text "s_def"}): @{thm s_def}.
|
|
718 |
\item State @{term "s"} is a valid state (@{text "vt_s"}): @{thm vt_s}. This implies
|
|
719 |
event @{text "V th cs"} is eligible to happen under state @{term "s'"} and
|
|
720 |
state @{term "s'"} is a valid state.
|
|
721 |
\end{enumerate}
|
|
722 |
*}
|
|
723 |
|
|
724 |
text {* \noindent
|
|
725 |
Under such a context, we investigated how the current precedence @{term "cp"} of
|
|
726 |
threads change from state @{term "s'"} to @{term "s"}.
|
|
727 |
|
|
728 |
|
|
729 |
Two subcases are considerted,
|
|
730 |
where the first is that there exits @{term "th'"}
|
|
731 |
such that
|
|
732 |
@{thm [display] nt}
|
|
733 |
holds, which means there exists a thread @{term "th'"} to take over
|
|
734 |
the resource release by thread @{term "th"}.
|
|
735 |
In this sub-case, the following results are obtained:
|
|
736 |
\begin{enumerate}
|
|
737 |
\item The change of RAG is given by lemma @{text "depend_s"}:
|
|
738 |
@{thm [display] "depend_s"}
|
|
739 |
which shows two edges are removed while one is added. These changes imply how
|
|
740 |
the current precedences should be re-computed.
|
|
741 |
\item First all threads different from @{term "th"} and @{term "th'"} have their
|
|
742 |
@{term "cp"}-value kept, therefore do not need a re-computation
|
|
743 |
(@{text "cp_kept"}): @{thm [display] cp_kept}
|
|
744 |
This lemma also implies, only the @{term "cp"}-values of @{term "th"} and @{term "th'"}
|
|
745 |
need to be recomputed.
|
|
746 |
\end{enumerate}
|
|
747 |
*}
|
|
748 |
|
|
749 |
(*<*)
|
|
750 |
end
|
|
751 |
|
|
752 |
context step_v_cps_nnt
|
|
753 |
begin
|
|
754 |
(*>*)
|
|
755 |
|
|
756 |
text {*
|
|
757 |
The other sub-case is when for all @{text "th'"}
|
|
758 |
@{thm [display] nnt}
|
|
759 |
holds, no such thread exists. The following results can be obtained for this
|
|
760 |
sub-case:
|
|
761 |
\begin{enumerate}
|
|
762 |
\item The change of RAG is given by lemma @{text "depend_s"}:
|
|
763 |
@{thm [display] depend_s}
|
|
764 |
which means only one edge is removed.
|
|
765 |
\item In this case, no re-computation is needed (@{text "eq_cp"}):
|
|
766 |
@{thm [display] eq_cp}
|
|
767 |
\end{enumerate}
|
|
768 |
*}
|
|
769 |
|
|
770 |
(*<*)
|
|
771 |
end
|
|
772 |
(*>*)
|
|
773 |
|
|
774 |
|
|
775 |
subsection {* Event @{text "P th cs"} *}
|
|
776 |
|
|
777 |
(*<*)
|
|
778 |
context step_P_cps_e
|
|
779 |
begin
|
|
780 |
(*>*)
|
|
781 |
|
|
782 |
text {*
|
|
783 |
The context under which event @{text "P th cs"} happens is formalized as follows:
|
|
784 |
\begin{enumerate}
|
|
785 |
\item The formation of @{term "s"} (@{text "s_def"}): @{thm s_def}.
|
|
786 |
\item State @{term "s"} is a valid state (@{text "vt_s"}): @{thm vt_s}. This implies
|
|
787 |
event @{text "P th cs"} is eligible to happen under state @{term "s'"} and
|
|
788 |
state @{term "s'"} is a valid state.
|
|
789 |
\end{enumerate}
|
|
790 |
|
|
791 |
This case is further divided into two sub-cases. The first is when @{thm ee} holds.
|
|
792 |
The following results can be obtained:
|
|
793 |
\begin{enumerate}
|
|
794 |
\item One edge is added to the RAG (@{text "depend_s"}):
|
|
795 |
@{thm [display] depend_s}
|
|
796 |
\item No re-computation is needed (@{text "eq_cp"}):
|
|
797 |
@{thm [display] eq_cp}
|
|
798 |
\end{enumerate}
|
|
799 |
*}
|
|
800 |
|
|
801 |
(*<*)
|
|
802 |
end
|
|
803 |
|
|
804 |
context step_P_cps_ne
|
|
805 |
begin
|
|
806 |
(*>*)
|
|
807 |
|
|
808 |
text {*
|
|
809 |
The second is when @{thm ne} holds.
|
|
810 |
The following results can be obtained:
|
|
811 |
\begin{enumerate}
|
|
812 |
\item One edge is added to the RAG (@{text "depend_s"}):
|
|
813 |
@{thm [display] depend_s}
|
|
814 |
\item Threads with no dependence relation with @{term "th"} do not need a re-computation
|
|
815 |
of their @{term "cp"}-values (@{text "eq_cp"}):
|
|
816 |
@{thm [display] eq_cp}
|
|
817 |
This lemma implies all threads with a dependence relation with @{term "th"} may need
|
|
818 |
re-computation.
|
|
819 |
\item Similar to the case of @{term "Set"}, the computation procedure could stop earlier
|
|
820 |
(@{text "eq_up"}):
|
|
821 |
@{thm [display] eq_up}
|
|
822 |
\end{enumerate}
|
|
823 |
|
|
824 |
*}
|
|
825 |
|
|
826 |
(*<*)
|
|
827 |
end
|
|
828 |
(*>*)
|
|
829 |
|
|
830 |
subsection {* Event @{text "Create th prio"} *}
|
|
831 |
|
|
832 |
(*<*)
|
|
833 |
context step_create_cps
|
|
834 |
begin
|
|
835 |
(*>*)
|
|
836 |
|
|
837 |
text {*
|
|
838 |
The context under which event @{text "Create th prio"} happens is formalized as follows:
|
|
839 |
\begin{enumerate}
|
|
840 |
\item The formation of @{term "s"} (@{text "s_def"}): @{thm s_def}.
|
|
841 |
\item State @{term "s"} is a valid state (@{text "vt_s"}): @{thm vt_s}. This implies
|
|
842 |
event @{text "Create th prio"} is eligible to happen under state @{term "s'"} and
|
|
843 |
state @{term "s'"} is a valid state.
|
|
844 |
\end{enumerate}
|
|
845 |
The following results can be obtained under this context:
|
|
846 |
\begin{enumerate}
|
|
847 |
\item The RAG does not change (@{text "eq_dep"}):
|
|
848 |
@{thm [display] eq_dep}
|
|
849 |
\item All threads other than @{term "th"} do not need re-computation (@{text "eq_cp"}):
|
|
850 |
@{thm [display] eq_cp}
|
|
851 |
\item The @{term "cp"}-value of @{term "th"} equals its precedence
|
|
852 |
(@{text "eq_cp_th"}):
|
|
853 |
@{thm [display] eq_cp_th}
|
|
854 |
\end{enumerate}
|
|
855 |
|
|
856 |
*}
|
|
857 |
|
|
858 |
|
|
859 |
(*<*)
|
|
860 |
end
|
|
861 |
(*>*)
|
|
862 |
|
|
863 |
subsection {* Event @{text "Exit th"} *}
|
|
864 |
|
|
865 |
(*<*)
|
|
866 |
context step_exit_cps
|
|
867 |
begin
|
|
868 |
(*>*)
|
|
869 |
|
|
870 |
text {*
|
|
871 |
The context under which event @{text "Exit th"} happens is formalized as follows:
|
|
872 |
\begin{enumerate}
|
|
873 |
\item The formation of @{term "s"} (@{text "s_def"}): @{thm s_def}.
|
|
874 |
\item State @{term "s"} is a valid state (@{text "vt_s"}): @{thm vt_s}. This implies
|
|
875 |
event @{text "Exit th"} is eligible to happen under state @{term "s'"} and
|
|
876 |
state @{term "s'"} is a valid state.
|
|
877 |
\end{enumerate}
|
|
878 |
The following results can be obtained under this context:
|
|
879 |
\begin{enumerate}
|
|
880 |
\item The RAG does not change (@{text "eq_dep"}):
|
|
881 |
@{thm [display] eq_dep}
|
|
882 |
\item All threads other than @{term "th"} do not need re-computation (@{text "eq_cp"}):
|
|
883 |
@{thm [display] eq_cp}
|
|
884 |
\end{enumerate}
|
|
885 |
Since @{term th} does not live in state @{term "s"}, there is no need to compute
|
|
886 |
its @{term cp}-value.
|
|
887 |
*}
|
|
888 |
|
|
889 |
(*<*)
|
|
890 |
end
|
|
891 |
(*>*)
|
|
892 |
|
|
893 |
|
262
|
894 |
section {* Related works \label{related} *}
|
|
895 |
|
|
896 |
text {*
|
|
897 |
\begin{enumerate}
|
|
898 |
\item {\em Integrating Priority Inheritance Algorithms in the Real-Time Specification for Java}
|
|
899 |
\cite{WellingsBSB07} models and verifies the combination of Priority Inheritance (PI) and
|
|
900 |
Priority Ceiling Emulation (PCE) protocols in the setting of Java virtual machine
|
|
901 |
using extended Timed Automata(TA) formalism of the UPPAAL tool. Although a detailed
|
|
902 |
formal model of combined PI and PCE is given, the number of properties is quite
|
|
903 |
small and the focus is put on the harmonious working of PI and PCE. Most key features of PI
|
|
904 |
(as well as PCE) are not shown. Because of the limitation of the model checking technique
|
|
905 |
used there, properties are shown only for a small number of scenarios. Therefore,
|
|
906 |
the verification does not show the correctness of the formal model itself in a
|
|
907 |
convincing way.
|
|
908 |
\item {\em Formal Development of Solutions for Real-Time Operating Systems with TLA+/TLC}
|
|
909 |
\cite{Faria08}. A formal model of PI is given in TLA+. Only 3 properties are shown
|
|
910 |
for PI using model checking. The limitation of model checking is intrinsic to the work.
|
|
911 |
\item {\em Synchronous modeling and validation of priority inheritance schedulers}
|
|
912 |
\cite{conf/fase/JahierHR09}. Gives a formal model
|
|
913 |
of PI and PCE in AADL (Architecture Analysis \& Design Language) and checked
|
|
914 |
several properties using model checking. The number of properties shown there is
|
|
915 |
less than here and the scale is also limited by the model checking technique.
|
|
916 |
\item {\em The Priority Ceiling Protocol: Formalization and Analysis Using PVS}
|
|
917 |
\cite{dutertre99b}. Formalized another protocol for Priority Inversion in the
|
|
918 |
interactive theorem proving system PVS.
|
|
919 |
\end{enumerate}
|
|
920 |
|
|
921 |
|
|
922 |
There are several works on inversion avoidance:
|
|
923 |
\begin{enumerate}
|
|
924 |
\item {\em Solving the group priority inversion problem in a timed asynchronous system}
|
|
925 |
\cite{Wang:2002:SGP}. The notion of Group Priority Inversion is introduced. The main
|
|
926 |
strategy is still inversion avoidance. The method is by reordering requests
|
|
927 |
in the setting of Client-Server.
|
|
928 |
\item {\em A Formalization of Priority Inversion} \cite{journals/rts/BabaogluMS93}.
|
|
929 |
Formalized the notion of Priority
|
|
930 |
Inversion and proposes methods to avoid it.
|
|
931 |
\end{enumerate}
|
|
932 |
|
|
933 |
{\em Examples of inaccurate specification of the protocol ???}.
|
|
934 |
|
|
935 |
*}
|
|
936 |
|
|
937 |
section {* Conclusions \label{conclusion} *}
|
|
938 |
|
|
939 |
(*<*)
|
|
940 |
end
|
|
941 |
(*>*) |