source: doc/theses/andrew_beach_MMath/features.tex @ 3b69398

Last change on this file since 3b69398 was 166b384, checked in by Andrew Beach <ajbeach@…>, 3 years ago

Andrew MMath: Added the missing front matter and corrected a few spelling/grammar mistakes.

  • Property mode set to 100644
File size: 38.1 KB
RevLine 
[4706098c]1\chapter{Exception Features}
[553f8abe]2\label{c:features}
[4706098c]3
[4aba055]4This chapter covers the design and user interface of the \CFA EHM
5and begins with a general overview of EHMs. It is not a strict
6definition of all EHMs nor an exhaustive list of all possible features.
[9cdfa5fb]7However, it does cover the most common structure and features found in them.
[f6106a6]8
[4aba055]9\section{Overview of EHMs}
[4260566]10% We should cover what is an exception handling mechanism and what is an
11% exception before this. Probably in the introduction. Some of this could
12% move there.
[4aba055]13\subsection{Raise / Handle}
[4260566]14An exception operation has two main parts: raise and handle.
[6071efc]15These terms are sometimes known as throw and catch but this work uses
[4260566]16throw/catch as a particular kind of raise/handle.
[4aba055]17These are the two parts that the user writes and may
[e3984a68]18be the only two pieces of the EHM that have any syntax in a language.
[4260566]19
[4aba055]20\paragraph{Raise}
[e3984a68]21The raise is the starting point for exception handling,
22by raising an exception, which passes it to
[f6106a6]23the EHM.
[4260566]24
[f6106a6]25Some well known examples include the @throw@ statements of \Cpp and Java and
[e3984a68]26the \code{Python}{raise} statement of Python. In real systems, a raise may
27perform some other work (such as memory management) but for the
[299b8b2]28purposes of this overview that can be ignored.
[4260566]29
[4aba055]30\paragraph{Handle}
[e3984a68]31The primary purpose of an EHM is to run some user code to handle a raised
32exception. This code is given, along with some other information,
33in a handler.
[f6106a6]34
35A handler has three common features: the previously mentioned user code, a
[e3984a68]36region of code it guards and an exception label/condition that matches
37against the raised exception.
[4aba055]38Only raises inside the guarded region and raising exceptions that match the
[f6106a6]39label can be handled by a given handler.
[6071efc]40If multiple handlers could can handle an exception,
[e3984a68]41EHMs define a rule to pick one, such as ``best match" or ``first found".
[4260566]42
[f6106a6]43The @try@ statements of \Cpp, Java and Python are common examples. All three
[9cdfa5fb]44also show another common feature of handlers: they are grouped by the guarded
[f42a6b8]45region.
[f6106a6]46
[4aba055]47\subsection{Propagation}
[de47a9d]48After an exception is raised comes what is usually the biggest step for the
[e3984a68]49EHM: finding and setting up the handler for execution.
50The propagation from raise to
[f6106a6]51handler can be broken up into three different tasks: searching for a handler,
[21f2e92]52matching against the handler and installing the handler.
[de47a9d]53
[4aba055]54\paragraph{Searching}
[f6106a6]55The EHM begins by searching for handlers that might be used to handle
[e3984a68]56the exception.
57The search will find handlers that have the raise site in their guarded
[f6106a6]58region.
[4aba055]59The search includes handlers in the current function, as well as any in
60callers on the stack that have the function call in their guarded region.
[f6106a6]61
[4aba055]62\paragraph{Matching}
[e3984a68]63Each handler found is with the raised exception. The exception
64label defines a condition that is used with the exception and decides if
[f6106a6]65there is a match or not.
[e3984a68]66%
[4aba055]67In languages where the first match is used, this step is intertwined with
[e3984a68]68searching; a match check is performed immediately after the search finds
69a handler.
[4260566]70
[4aba055]71\paragraph{Installing}
[e3984a68]72After a handler is chosen, it must be made ready to run.
[f6106a6]73The implementation can vary widely to fit with the rest of the
[de47a9d]74design of the EHM. The installation step might be trivial or it could be
[4260566]75the most expensive step in handling an exception. The latter tends to be the
76case when stack unwinding is involved.
[de47a9d]77
[6071efc]78If a matching handler is not guaranteed to be found, the EHM needs a
[e3984a68]79different course of action for this case.
[4aba055]80This situation only occurs with unchecked exceptions as checked exceptions
[f42a6b8]81(such as in Java) can make the guarantee.
[e3984a68]82The unhandled action is usually very general, such as aborting the program.
[4260566]83
[4aba055]84\paragraph{Hierarchy}
[f6106a6]85A common way to organize exceptions is in a hierarchical structure.
[166b384]86This pattern comes from object-oriented languages where the
[4260566]87exception hierarchy is a natural extension of the object hierarchy.
88
[e3984a68]89Consider the following exception hierarchy:
[4706098c]90\begin{center}
[6a8208cb]91\input{exception-hierarchy}
[4706098c]92\end{center}
[4aba055]93A handler labeled with any given exception can handle exceptions of that
[4260566]94type or any child type of that exception. The root of the exception hierarchy
[f42a6b8]95(here \code{C}{exception}) acts as a catch-all, leaf types catch single types
[4260566]96and the exceptions in the middle can be used to catch different groups of
97related exceptions.
98
99This system has some notable advantages, such as multiple levels of grouping,
[f42a6b8]100the ability for libraries to add new exception types and the isolation
[f6106a6]101between different sub-hierarchies.
102This design is used in \CFA even though it is not a object-orientated
[9cdfa5fb]103language, so different tools are used to create the hierarchy.
[4260566]104
105% Could I cite the rational for the Python IO exception rework?
106
[4aba055]107\subsection{Completion}
[6071efc]108After the handler has finished, the entire exception operation has to complete
[f6106a6]109and continue executing somewhere else. This step is usually simple,
110both logically and in its implementation, as the installation of the handler
111is usually set up to do most of the work.
[de47a9d]112
[e3984a68]113The EHM can return control to many different places, where
[4aba055]114the most common are after the handler definition (termination)
115and after the raise (resumption).
[4260566]116
[4aba055]117\subsection{Communication}
[887fc79]118For effective exception handling, additional information is often passed
[4aba055]119from the raise to the handler and back again.
[9cdfa5fb]120So far, only communication of the exception's identity is covered.
[e3984a68]121A common communication method for adding information to an exception
122is putting fields into the exception instance
[4aba055]123and giving the handler access to them.
[e3984a68]124% You can either have pointers/references in the exception, or have p/rs to
125% the exception when it doesn't have to be copied.
126Passing references or pointers allows data at the raise location to be
127updated, passing information in both directions.
[4260566]128
129\section{Virtuals}
[3b8acfb]130\label{s:virtuals}
[6aa84e0]131A common feature in many programming languages is a tool to pair code
132(behaviour) with data.
[166b384]133In \CFA, this is done with the virtual system,
[6aa84e0]134which allow type information to be abstracted away, recovered and allow
135operations to be performed on the abstract objects.
136
[f6106a6]137Virtual types and casts are not part of \CFA's EHM nor are they required for
[e3984a68]138an EHM.
139However, one of the best ways to support an exception hierarchy
[4aba055]140is via a virtual hierarchy and dispatch system.
[f42a6b8]141Ideally, the virtual system would have been part of \CFA before the work
[a6c45c6]142on exception handling began, but unfortunately it was not.
[4aba055]143Hence, only the features and framework needed for the EHM were
[e3984a68]144designed and implemented for this thesis.
145Other features were considered to ensure that
[4aba055]146the structure could accommodate other desirable features in the future
[e3984a68]147but are not implemented.
148The rest of this section only discusses the implemented subset of the
[f42a6b8]149virtual system design.
[4260566]150
151The virtual system supports multiple ``trees" of types. Each tree is
152a simple hierarchy with a single root type. Each type in a tree has exactly
[f6106a6]153one parent -- except for the root type which has zero parents -- and any
[4260566]154number of children.
155Any type that belongs to any of these trees is called a virtual type.
156% A type's ancestors are its parent and its parent's ancestors.
157% The root type has no ancestors.
[4aba055]158% A type's descendants are its children and its children's descendants.
[4260566]159
[9cdfa5fb]160For the purposes of illustration, a proposed, but unimplemented, syntax
[13afd0c]161will be used. Each virtual type is represented by a trait with an annotation
[e3984a68]162that makes it a virtual type. This annotation is empty for a root type, which
163creates a new tree:
164\begin{cfa}
165trait root_type(T) virtual() {}
166\end{cfa}
167The annotation may also refer to any existing virtual type to make this new
168type a child of that type and part of the same tree. The parent may itself
169be a child or a root type and may have any number of existing children.
[cd03b76d]170
171% OK, for some reason the b and t positioning options are reversed here.
172\begin{minipage}[b]{0.6\textwidth}
[e3984a68]173\begin{cfa}
174trait child_a(T) virtual(root_type) {}
175trait grandchild(T) virtual(child_a) {}
176trait child_b(T) virtual(root_type) {}
177\end{cfa}
[cd03b76d]178\end{minipage}
179\begin{minipage}{0.4\textwidth}
180\begin{center}
181\input{virtual-tree}
182\end{center}
183\end{minipage}
[4aba055]184
[9cdfa5fb]185Every virtual type also has a list of virtual members and a unique id.
186Both are stored in a virtual table.
[e3984a68]187Every instance of a virtual type also has a pointer to a virtual table stored
188in it, although there is no per-type virtual table as in many other languages.
[4260566]189
[9cdfa5fb]190The list of virtual members is accumulated from the root type down the tree.
191Every virtual type
[e3984a68]192inherits the list of virtual members from its parent and may add more
193virtual members to the end of the list which are passed on to its children.
194Again, using the unimplemented syntax this might look like:
195\begin{cfa}
196trait root_type(T) virtual() {
197        const char * to_string(T const & this);
198        unsigned int size;
199}
200
201trait child_type(T) virtual(root_type) {
202        char * irrelevant_function(int, char);
203}
204\end{cfa}
205% Consider adding a diagram, but we might be good with the explanation.
206
[9cdfa5fb]207As @child_type@ is a child of @root_type@, it has the virtual members of
[e3984a68]208@root_type@ (@to_string@ and @size@) as well as the one it declared
[13afd0c]209(@irrelevant_function@).
[e3984a68]210
211It is important to note that these are virtual members, and may contain   
212arbitrary fields, functions or otherwise.
213The names ``size" and ``align" are reserved for the size and alignment of the
214virtual type, and are always automatically initialized as such.
[9cdfa5fb]215The other special case is uses of the trait's polymorphic argument
[e3984a68]216(@T@ in the example), which are always updated to refer to the current
[9cdfa5fb]217virtual type. This allows functions that refer to the polymorphic argument
[e3984a68]218to act as traditional virtual methods (@to_string@ in the example), as the
219object can always be passed to a virtual method in its virtual table.
[4260566]220
[9cdfa5fb]221Up until this point, the virtual system is similar to ones found in
222object-oriented languages, but this is where \CFA diverges.
[e3984a68]223Objects encapsulate a single set of methods in each type,
224universally across the entire program,
225and indeed all programs that use that type definition.
226The only way to change any method is to inherit and define a new type with
227its own universal implementation. In this sense,
228these object-oriented types are ``closed" and cannot be altered.
229% Because really they are class oriented.
230
231In \CFA, types do not encapsulate any code.
[9cdfa5fb]232Whether or not a type satisfies any given assertion, and hence any trait, is
[e3984a68]233context sensitive. Types can begin to satisfy a trait, stop satisfying it or
234satisfy the same trait at any lexical location in the program.
[9cdfa5fb]235In this sense, a type's implementation in the set of functions and variables
[e3984a68]236that allow it to satisfy a trait is ``open" and can change
237throughout the program.
[4aba055]238This capability means it is impossible to pick a single set of functions
[e3984a68]239that represent a type's implementation across a program.
[f6106a6]240
241\CFA side-steps this issue by not having a single virtual table for each
[4aba055]242type. A user can define virtual tables that are filled in at their
243declaration and given a name. Anywhere that name is visible, even if it is
[e3984a68]244defined locally inside a function (although in this case the user must ensure
245it outlives any objects that use it), it can be used.
[4aba055]246Specifically, a virtual type is ``bound" to a virtual table that
[08e75215]247sets the virtual members for that object. The virtual members can be accessed
248through the object.
[4706098c]249
[ed4d7c1]250This means virtual tables are declared and named in \CFA.
251They are declared as variables, using the type
252@vtable(VIRTUAL_TYPE)@ and any valid name. For example:
253\begin{cfa}
254vtable(virtual_type_name) table_name;
255\end{cfa}
256
[9cdfa5fb]257Like any variable, they may be forward declared with the @extern@ keyword.
[ed4d7c1]258Forward declaring virtual tables is relatively common.
259Many virtual types have an ``obvious" implementation that works in most
260cases.
261A pattern that has appeared in the early work using virtuals is to
262implement a virtual table with the the obvious definition and place a forward
263declaration of it in the header beside the definition of the virtual type.
264
265Even on the full declaration, no initializer should be used.
266Initialization is automatic.
267The type id and special virtual members ``size" and ``align" only depend on
[9cdfa5fb]268the virtual type, which is fixed given the type of the virtual table, and
[ed4d7c1]269so the compiler fills in a fixed value.
[9cdfa5fb]270The other virtual members are resolved using the best match to the member's
[ed4d7c1]271name and type, in the same context as the virtual table is declared using
272\CFA's normal resolution rules.
273
[9cdfa5fb]274While much of the virtual infrastructure has been created,
275it is currently only used
[4706098c]276internally for exception handling. The only user-level feature is the virtual
[21f2e92]277cast, which is the same as the \Cpp \code{C++}{dynamic_cast}.
[7eb6eb5]278\label{p:VirtualCast}
[4706098c]279\begin{cfa}
[4a36b344]280(virtual TYPE)EXPRESSION
[4706098c]281\end{cfa}
[29c9b23]282Note, the syntax and semantics matches a C-cast, rather than the function-like
283\Cpp syntax for special casts. Both the type of @EXPRESSION@ and @TYPE@ must be
[9cdfa5fb]284pointers to virtual types.
[de47a9d]285The cast dynamically checks if the @EXPRESSION@ type is the same or a sub-type
[29c9b23]286of @TYPE@, and if true, returns a pointer to the
[4706098c]287@EXPRESSION@ object, otherwise it returns @0p@ (null pointer).
[9cdfa5fb]288This allows the expression to be used as both a cast and a type check.
[4706098c]289
[ed4d7c1]290\section{Exceptions}
291
292The syntax for declaring an exception is the same as declaring a structure
[9cdfa5fb]293except the keyword:
[ed4d7c1]294\begin{cfa}
295exception TYPE_NAME {
296        FIELDS
297};
298\end{cfa}
299
[9cdfa5fb]300Fields are filled in the same way as a structure as well. However, an extra
[13afd0c]301field is added that contains the pointer to the virtual table.
302It must be explicitly initialized by the user when the exception is
[ed4d7c1]303constructed.
304
305Here is an example of declaring an exception type along with a virtual table,
306assuming the exception has an ``obvious" implementation and a default
307virtual table makes sense.
308
309\begin{minipage}[t]{0.4\textwidth}
[9cdfa5fb]310Header (.hfa):
[ed4d7c1]311\begin{cfa}
312exception Example {
313        int data;
314};
315
316extern vtable(Example)
317        example_base_vtable;
318\end{cfa}
319\end{minipage}
320\begin{minipage}[t]{0.6\textwidth}
[9cdfa5fb]321Implementation (.cfa):
[ed4d7c1]322\begin{cfa}
323vtable(Example) example_base_vtable
324\end{cfa}
325\vfil
326\end{minipage}
327
328%\subsection{Exception Details}
[13afd0c]329This is the only interface needed when raising and handling exceptions.
[9cdfa5fb]330However, it is actually a shorthand for a more complex
331trait-based interface.
[4a36b344]332
[13afd0c]333The language views exceptions through a series of traits.
334If a type satisfies them, then it can be used as an exception. The following
[4706098c]335is the base trait all exceptions need to match.
336\begin{cfa}
337trait is_exception(exceptT &, virtualT &) {
[a6c45c6]338        // Numerous imaginary assertions.
[02b73ea]339};
[4706098c]340\end{cfa}
[13afd0c]341The trait is defined over two types: the exception type and the virtual table
[4aba055]342type. Each exception type should have a single virtual table type.
343There are no actual assertions in this trait because the trait system
344cannot express them yet (adding such assertions would be part of
[a6c45c6]345completing the virtual system). The imaginary assertions would probably come
346from a trait defined by the virtual system, and state that the exception type
[9cdfa5fb]347is a virtual type,
348that that the type is a descendant of @exception_t@ (the base exception type)
[e3984a68]349and allow the user to find the virtual table type.
[29c9b23]350
351% I did have a note about how it is the programmer's responsibility to make
352% sure the function is implemented correctly. But this is true of every
[de47a9d]353% similar system I know of (except Agda's I guess) so I took it out.
354
[f6106a6]355There are two more traits for exceptions defined as follows:
[4706098c]356\begin{cfa}
[02b73ea]357trait is_termination_exception(
[4706098c]358                exceptT &, virtualT & | is_exception(exceptT, virtualT)) {
[29c9b23]359        void defaultTerminationHandler(exceptT &);
[02b73ea]360};
361
362trait is_resumption_exception(
[4706098c]363                exceptT &, virtualT & | is_exception(exceptT, virtualT)) {
[29c9b23]364        void defaultResumptionHandler(exceptT &);
[02b73ea]365};
[4706098c]366\end{cfa}
[9cdfa5fb]367Both traits ensure a pair of types is an exception type and
368its virtual table type,
[f6106a6]369and defines one of the two default handlers. The default handlers are used
[9cdfa5fb]370as fallbacks and are discussed in detail in \autoref{s:ExceptionHandling}.
[de47a9d]371
[f6106a6]372However, all three of these traits can be tricky to use directly.
373While there is a bit of repetition required,
[de47a9d]374the largest issue is that the virtual table type is mangled and not in a user
[9cdfa5fb]375facing way. So, these three macros are provided to wrap these traits to
[f6106a6]376simplify referring to the names:
[f42a6b8]377@IS_EXCEPTION@, @IS_TERMINATION_EXCEPTION@ and @IS_RESUMPTION_EXCEPTION@.
[1830a86]378
[f6106a6]379All three take one or two arguments. The first argument is the name of the
380exception type. The macro passes its unmangled and mangled form to the trait.
[1830a86]381The second (optional) argument is a parenthesized list of polymorphic
[f6106a6]382arguments. This argument is only used with polymorphic exceptions and the
[9cdfa5fb]383list is passed to both types.
[f6106a6]384In the current set-up, the two types always have the same polymorphic
[9cdfa5fb]385arguments, so these macros can be used without losing flexibility.
[29c9b23]386
[9cdfa5fb]387For example, consider a function that is polymorphic over types that have a
[29c9b23]388defined arithmetic exception:
389\begin{cfa}
[de47a9d]390forall(Num | IS_EXCEPTION(Arithmetic, (Num)))
[29c9b23]391void some_math_function(Num & left, Num & right);
392\end{cfa}
[4706098c]393
[1830a86]394\section{Exception Handling}
[f6106a6]395\label{s:ExceptionHandling}
[4aba055]396As stated,
[21f2e92]397\CFA provides two kinds of exception handling: termination and resumption.
[f6106a6]398These twin operations are the core of \CFA's exception handling mechanism.
[e3984a68]399This section covers the general patterns shared by the two operations and
[9cdfa5fb]400then goes on to cover the details of each individual operation.
[de47a9d]401
[f6106a6]402Both operations follow the same set of steps.
[e3984a68]403First, a user raises an exception.
404Second, the exception propagates up the stack, searching for a handler.
405Third, if a handler is found, the exception is caught and the handler is run.
[4aba055]406After that control continues at a raise-dependent location.
[e3984a68]407As an alternate to the third step,
408if a handler is not found, a default handler is run and, if it returns,
409then control
[4aba055]410continues after the raise.
[f6106a6]411
[e3984a68]412The differences between the two operations include how propagation is
[13afd0c]413performed, where execution continues after an exception is handled
[e3984a68]414and which default handler is run.
[1830a86]415
[4706098c]416\subsection{Termination}
417\label{s:Termination}
[e3984a68]418Termination handling is the familiar kind of handling
[9cdfa5fb]419used in most programming
[1830a86]420languages with exception handling.
[4aba055]421It is a dynamic, non-local goto. If the raised exception is matched and
422handled, the stack is unwound and control (usually) continues in the function
[f6106a6]423on the call stack that defined the handler.
424Termination is commonly used when an error has occurred and recovery is
425impossible locally.
[1830a86]426
427% (usually) Control can continue in the current function but then a different
428% control flow construct should be used.
[4706098c]429
[f6106a6]430A termination raise is started with the @throw@ statement:
[4706098c]431\begin{cfa}
[4a36b344]432throw EXPRESSION;
[4706098c]433\end{cfa}
[29c9b23]434The expression must return a reference to a termination exception, where the
[f6106a6]435termination exception is any type that satisfies the trait
436@is_termination_exception@ at the call site.
[4aba055]437Through \CFA's trait system, the trait functions are implicitly passed into the
[e3984a68]438throw code for use by the EHM.
[f6106a6]439A new @defaultTerminationHandler@ can be defined in any scope to
[e3984a68]440change the throw's behaviour when a handler is not found (see below).
[de47a9d]441
[4aba055]442The throw copies the provided exception into managed memory to ensure
[21f2e92]443the exception is not destroyed if the stack is unwound.
[f6106a6]444It is the user's responsibility to ensure the original exception is cleaned
[4aba055]445up whether the stack is unwound or not. Allocating it on the stack is
[f6106a6]446usually sufficient.
[de47a9d]447
[4aba055]448% How to say propagation starts, its first sub-step is the search.
449Then propagation starts with the search. \CFA uses a ``first match" rule so
[e3984a68]450matching is performed with the copied exception as the search key.
451It starts from the raise site and proceeds towards base of the stack,
[1830a86]452from callee to caller.
[e3984a68]453At each stack frame, a check is made for termination handlers defined by the
[1830a86]454@catch@ clauses of a @try@ statement.
[4706098c]455\begin{cfa}
[4a36b344]456try {
[4706098c]457        GUARDED_BLOCK
[f6106a6]458} catch (EXCEPTION_TYPE$\(_1\)$ * [NAME$\(_1\)$]) {
[4706098c]459        HANDLER_BLOCK$\(_1\)$
[f6106a6]460} catch (EXCEPTION_TYPE$\(_2\)$ * [NAME$\(_2\)$]) {
[4706098c]461        HANDLER_BLOCK$\(_2\)$
[4a36b344]462}
[4706098c]463\end{cfa}
[4aba055]464When viewed on its own, a try statement simply executes the statements
[e3984a68]465in the \snake{GUARDED_BLOCK} and when those are finished,
[4aba055]466the try statement finishes.
[de47a9d]467
468However, while the guarded statements are being executed, including any
[4aba055]469invoked functions, all the handlers in these statements are included in the
470search path.
[e3984a68]471Hence, if a termination exception is raised, these handlers may be matched
[4aba055]472against the exception and may handle it.
[f6106a6]473
474Exception matching checks the handler in each catch clause in the order
[4aba055]475they appear, top to bottom. If the representation of the raised exception type
[e3984a68]476is the same or a descendant of @EXCEPTION_TYPE@$_i$, then @NAME@$_i$
[21f2e92]477(if provided) is
478bound to a pointer to the exception and the statements in @HANDLER_BLOCK@$_i$
479are executed. If control reaches the end of the handler, the exception is
[de47a9d]480freed and control continues after the try statement.
[4706098c]481
[e3984a68]482If no termination handler is found during the search, then the default handler
483(\defaultTerminationHandler) visible at the raise statement is called.
484Through \CFA's trait system the best match at the raise statement is used.
[4aba055]485This function is run and is passed the copied exception.
[e3984a68]486If the default handler finishes, control continues after the raise statement.
[1830a86]487
[f6106a6]488There is a global @defaultTerminationHandler@ that is polymorphic over all
[4aba055]489termination exception types.
[f6106a6]490The global default termination handler performs a cancellation
[e3984a68]491(as described in \vref{s:Cancellation})
492on the current stack with the copied exception.
493Since it is so general, a more specific handler can be defined,
494overriding the default behaviour for the specific exception types.
[6aa84e0]495
496For example, consider an error reading a configuration file.
[9411cf0]497This is most likely a problem with the configuration file (@config_error@),
498but the function could have been passed the wrong file name (@arg_error@).
[6aa84e0]499In this case the function could raise one exception and then, if it is
500unhandled, raise the other.
501This is not usual behaviour for either exception so changing the
502default handler will be done locally:
503\begin{cfa}
504{
505        void defaultTerminationHandler(config_error &) {
506                throw (arg_error){arg_vt};
507        }
508        throw (config_error){config_vt};
509}
510\end{cfa}
[4706098c]511
512\subsection{Resumption}
513\label{s:Resumption}
514
[e3984a68]515Resumption exception handling is less familar form of exception handling,
516but is
[f6106a6]517just as old~\cite{Goodenough75} and is simpler in many ways.
518It is a dynamic, non-local function call. If the raised exception is
[e3984a68]519matched, a closure is taken from up the stack and executed,
[4aba055]520after which the raising function continues executing.
521The common uses for resumption exceptions include
522potentially repairable errors, where execution can continue in the same
523function once the error is corrected, and
524ignorable events, such as logging where nothing needs to happen and control
[e3984a68]525should always continue from the raise site.
526
527Except for the changes to fit into that pattern, resumption exception
528handling is symmetric with termination exception handling, by design
529(see \autoref{s:Termination}).
[8483c39a]530
[4706098c]531A resumption raise is started with the @throwResume@ statement:
532\begin{cfa}
[4a36b344]533throwResume EXPRESSION;
[4706098c]534\end{cfa}
[cd03b76d]535% The new keywords are currently ``experimental" and not used in this work.
[e3984a68]536It works much the same way as the termination raise, except the
537type must satisfy the \snake{is_resumption_exception} that uses the
538default handler: \defaultResumptionHandler.
539This can be specialized for particular exception types.
540
541At run-time, no exception copy is made. Since
542resumption does not unwind the stack nor otherwise remove values from the
543current scope, there is no need to manage memory to keep the exception
544allocated.
545
546Then propagation starts with the search,
547following the same search path as termination,
548from the raise site to the base of stack and top of try statement to bottom.
549However, the handlers on try statements are defined by @catchResume@ clauses.
[4706098c]550\begin{cfa}
[4a36b344]551try {
[4706098c]552        GUARDED_BLOCK
[f6106a6]553} catchResume (EXCEPTION_TYPE$\(_1\)$ * [NAME$\(_1\)$]) {
[4706098c]554        HANDLER_BLOCK$\(_1\)$
[f6106a6]555} catchResume (EXCEPTION_TYPE$\(_2\)$ * [NAME$\(_2\)$]) {
[4706098c]556        HANDLER_BLOCK$\(_2\)$
[4a36b344]557}
[4706098c]558\end{cfa}
[f42a6b8]559Note that termination handlers and resumption handlers may be used together
[f6106a6]560in a single try statement, intermixing @catch@ and @catchResume@ freely.
[4aba055]561Each type of handler only interacts with exceptions from the matching
562kind of raise.
[e3984a68]563Like @catch@ clauses, @catchResume@ clauses have no effect if an exception
564is not raised.
[f42a6b8]565
[e3984a68]566The matching rules are exactly the same as well.
567The first major difference here is that after
568@EXCEPTION_TYPE@$_i$ is matched and @NAME@$_i$ is bound to the exception,
569@HANDLER_BLOCK@$_i$ is executed right away without first unwinding the stack.
[9cdfa5fb]570After the block has finished running, control jumps to the raise site, where
[e3984a68]571the just handled exception came from, and continues executing after it,
572not after the try statement.
[6aa84e0]573
574For instance, a resumption used to send messages to the logger may not
575need to be handled at all. Putting the following default handler
[9411cf0]576at the global scope can make handling that exception optional by default.
[6aa84e0]577\begin{cfa}
578void defaultResumptionHandler(log_message &) {
579    // Nothing, it is fine not to handle logging.
580}
581// ... No change at raise sites. ...
582throwResume (log_message){strlit_log, "Begin event processing."}
583\end{cfa}
[1830a86]584
[f6106a6]585\subsubsection{Resumption Marking}
[df24d37]586\label{s:ResumptionMarking}
[1830a86]587A key difference between resumption and termination is that resumption does
[e3984a68]588not unwind the stack. A side effect is that, when a handler is matched
589and run, its try block (the guarded statements) and every try statement
[9cdfa5fb]590searched before it are still on the stack. Their presence can lead to
[cd03b76d]591the recursive resumption problem.\cite{Buhr00a}
592% Other possible citation is MacLaren77, but the form is different.
[1830a86]593
594The recursive resumption problem is any situation where a resumption handler
595ends up being called while it is running.
596Consider a trivial case:
597\begin{cfa}
598try {
599        throwResume (E &){};
600} catchResume(E *) {
601        throwResume (E &){};
602}
603\end{cfa}
[4aba055]604When this code is executed, the guarded @throwResume@ starts a
605search and matches the handler in the @catchResume@ clause. This
[e3984a68]606call is placed on the stack above the try-block.
607Now the second raise in the handler searches the same try block,
608matches again and then puts another instance of the
[4aba055]609same handler on the stack leading to infinite recursion.
[1830a86]610
[f42a6b8]611While this situation is trivial and easy to avoid, much more complex cycles
612can form with multiple handlers and different exception types.
[e3984a68]613To prevent all of these cases, each try statement is ``marked" from the
614time the exception search reaches it to either when a handler completes
615handling that exception or when the search reaches the base
[4aba055]616of the stack.
617While a try statement is marked, its handlers are never matched, effectively
[21f2e92]618skipping over it to the next try statement.
[4a36b344]619
[6a8208cb]620\begin{center}
621\input{stack-marking}
622\end{center}
[de47a9d]623
[9cdfa5fb]624There are other sets of marking rules that could be used.
625For instance, marking just the handlers that caught the exception
[4aba055]626would also prevent recursive resumption.
[9cdfa5fb]627However, the rules selected mirror what happens with termination,
[e3984a68]628so this reduces the amount of rules and patterns a programmer has to know.
[4706098c]629
[e3984a68]630The marked try statements are the ones that would be removed from
631the stack for a termination exception, \ie those on the stack
[4aba055]632between the handler and the raise statement.
633This symmetry applies to the default handler as well, as both kinds of
634default handlers are run at the raise statement, rather than (physically
635or logically) at the bottom of the stack.
636% In early development having the default handler happen after
637% unmarking was just more useful. We assume that will continue.
[4706098c]638
639\section{Conditional Catch}
[de47a9d]640Both termination and resumption handler clauses can be given an additional
641condition to further control which exceptions they handle:
[4706098c]642\begin{cfa}
[f6106a6]643catch (EXCEPTION_TYPE * [NAME] ; CONDITION)
[4706098c]644\end{cfa}
645First, the same semantics is used to match the exception type. Second, if the
646exception matches, @CONDITION@ is executed. The condition expression may
[de47a9d]647reference all names in scope at the beginning of the try block and @NAME@
[1c1c180]648introduced in the handler clause. If the condition is true, then the handler
[1830a86]649matches. Otherwise, the exception search continues as if the exception type
650did not match.
[f6106a6]651
[4aba055]652The condition matching allows finer matching by checking
[f6106a6]653more kinds of information than just the exception type.
[4706098c]654\begin{cfa}
655try {
[f6106a6]656        handle1 = open( f1, ... );
657        handle2 = open( f2, ... );
658        handle3 = open( f3, ... );
[4706098c]659        ...
[de47a9d]660} catch( IOFailure * f ; fd( f ) == f1 ) {
[f6106a6]661        // Only handle IO failure for f1.
662} catch( IOFailure * f ; fd( f ) == f3 ) {
663        // Only handle IO failure for f3.
[4706098c]664}
[e3984a68]665// Handle a failure relating to f2 further down the stack.
[4706098c]666\end{cfa}
[9cdfa5fb]667In this example, the file that experienced the IO error is used to decide
[f6106a6]668which handler should be run, if any at all.
669
670\begin{comment}
671% I know I actually haven't got rid of them yet, but I'm going to try
672% to write it as if I had and see if that makes sense:
673\section{Reraising}
674\label{s:Reraising}
[4706098c]675Within the handler block or functions called from the handler block, it is
676possible to reraise the most recently caught exception with @throw@ or
[1830a86]677@throwResume@, respectively.
[4706098c]678\begin{cfa}
[29c9b23]679try {
680        ...
681} catch( ... ) {
[1830a86]682        ... throw;
[4706098c]683} catchResume( ... ) {
[1830a86]684        ... throwResume;
[4706098c]685}
686\end{cfa}
687The only difference between a raise and a reraise is that reraise does not
688create a new exception; instead it continues using the current exception, \ie
689no allocation and copy. However the default handler is still set to the one
690visible at the raise point, and hence, for termination could refer to data that
691is part of an unwound stack frame. To prevent this problem, a new default
692handler is generated that does a program-level abort.
[f6106a6]693\end{comment}
694
695\subsection{Comparison with Reraising}
[9cdfa5fb]696In languages without conditional catch -- that is, no ability to match an
697exception based on something other than its type -- it can be mimicked
[e3984a68]698by matching all exceptions of the right type, checking any additional
699conditions inside the handler and re-raising the exception if it does not
700match those.
701
702Here is a minimal example comparing both patterns, using @throw;@
[9cdfa5fb]703(no operand) to start a re-raise.
[e3984a68]704\begin{center}
705\begin{tabular}{l r}
[f6106a6]706\begin{cfa}
707try {
[f42a6b8]708    do_work_may_throw();
[e3984a68]709} catch(exception_t * exc ;
710                can_handle(exc)) {
[f42a6b8]711    handle(exc);
[f6106a6]712}
713
[e3984a68]714
715
716\end{cfa}
717&
[f6106a6]718\begin{cfa}
719try {
[f42a6b8]720    do_work_may_throw();
[e3984a68]721} catch(exception_t * exc) {
[f42a6b8]722    if (can_handle(exc)) {
723        handle(exc);
724    } else {
725        throw;
726    }
[f6106a6]727}
728\end{cfa}
[e3984a68]729\end{tabular}
730\end{center}
[9cdfa5fb]731At first glance, catch-and-reraise may appear to just be a quality-of-life
[e3984a68]732feature, but there are some significant differences between the two
[9cdfa5fb]733strategies.
[e3984a68]734
735A simple difference that is more important for \CFA than many other languages
[9cdfa5fb]736is that the raise site changes with a re-raise, but does not with a
[e3984a68]737conditional catch.
738This is important in \CFA because control returns to the raise site to run
[9cdfa5fb]739the per-site default handler. Because of this, only a conditional catch can
[e3984a68]740allow the original raise to continue.
741
742The more complex issue comes from the difference in how conditional
743catches and re-raises handle multiple handlers attached to a single try
744statement. A conditional catch will continue checking later handlers while
745a re-raise will skip them.
746If the different handlers could handle some of the same exceptions,
747translating a try statement that uses one to use the other can quickly
748become non-trivial:
749
750\noindent
751Original, with conditional catch:
752\begin{cfa}
753...
754} catch (an_exception * e ; check_a(e)) {
755        handle_a(e);
756} catch (exception_t * e ; check_b(e)) {
757        handle_b(e);
758}
759\end{cfa}
760Translated, with re-raise:
761\begin{cfa}
762...
763} catch (exception_t * e) {
764        an_exception * an_e = (virtual an_exception *)e;
765        if (an_e && check_a(an_e)) {
766                handle_a(an_e);
767        } else if (check_b(e)) {
768                handle_b(e);
769        } else {
770                throw;
771        }
772}
773\end{cfa}
774(There is a simpler solution if @handle_a@ never raises exceptions,
775using nested try statements.)
776
777% } catch (an_exception * e ; check_a(e)) {
778%     handle_a(e);
779% } catch (exception_t * e ; !(virtual an_exception *)e && check_b(e)) {
780%     handle_b(e);
781% }
[4aba055]782%
[e3984a68]783% } catch (an_exception * e)
784%   if (check_a(e)) {
785%     handle_a(e);
786%   } else throw;
787% } catch (exception_t * e)
788%   if (check_b(e)) {
789%     handle_b(e);
790%   } else throw;
791% }
[9cdfa5fb]792In similar simple examples, translating from re-raise to conditional catch
793takes less code but it does not have a general, trivial solution either.
[e3984a68]794
795So, given that the two patterns do not trivially translate into each other,
796it becomes a matter of which on should be encouraged and made the default.
[9cdfa5fb]797From the premise that if a handler could handle an exception then it
[e3984a68]798should, it follows that checking as many handlers as possible is preferred.
[9cdfa5fb]799So, conditional catch and checking later handlers is a good default.
[4a36b344]800
801\section{Finally Clauses}
[f6106a6]802\label{s:FinallyClauses}
[9cdfa5fb]803Finally clauses are used to perform unconditional cleanup when leaving a
[f6106a6]804scope and are placed at the end of a try statement after any handler clauses:
[4706098c]805\begin{cfa}
[4a36b344]806try {
[4706098c]807        GUARDED_BLOCK
[29c9b23]808} ... // any number or kind of handler clauses
809... finally {
[4706098c]810        FINALLY_BLOCK
[4a36b344]811}
[4706098c]812\end{cfa}
[29c9b23]813The @FINALLY_BLOCK@ is executed when the try statement is removed from the
[1830a86]814stack, including when the @GUARDED_BLOCK@ finishes, any termination handler
[f42a6b8]815finishes or during an unwind.
[29c9b23]816The only time the block is not executed is if the program is exited before
[1830a86]817the stack is unwound.
[4706098c]818
819Execution of the finally block should always finish, meaning control runs off
[f6106a6]820the end of the block. This requirement ensures control always continues as if
[9cdfa5fb]821the finally clause is not present, \ie finally is for cleanup, not changing
[f6106a6]822control flow.
823Because of this requirement, local control flow out of the finally block
[1c1c180]824is forbidden. The compiler precludes any @break@, @continue@, @fallthru@ or
[4706098c]825@return@ that causes control to leave the finally block. Other ways to leave
[9cdfa5fb]826the finally block, such as a @longjmp@ or termination are much harder to check,
827and at best require additional run-time overhead, and so are only
[1830a86]828discouraged.
829
[9cdfa5fb]830Not all languages with unwinding have finally clauses. Notably, \Cpp does
[e3984a68]831without it as destructors, and the RAII design pattern, serve a similar role.
832Although destructors and finally clauses can be used for the same cases,
[4aba055]833they have their own strengths, similar to top-level function and lambda
834functions with closures.
[e3984a68]835Destructors take more work to create, but if there is clean-up code
836that needs to be run every time a type is used, they are much easier
[9cdfa5fb]837to set up for each use. % It's automatic.
838On the other hand, finally clauses capture the local context, so are easy to
839use when the cleanup is not dependent on the type of a variable or requires
[4aba055]840information from multiple variables.
[4a36b344]841
842\section{Cancellation}
[f6106a6]843\label{s:Cancellation}
[de47a9d]844Cancellation is a stack-level abort, which can be thought of as as an
[f6106a6]845uncatchable termination. It unwinds the entire current stack, and if
[9cdfa5fb]846possible, forwards the cancellation exception to a different stack.
[4706098c]847
[29c9b23]848Cancellation is not an exception operation like termination or resumption.
[4706098c]849There is no special statement for starting a cancellation; instead the standard
[9cdfa5fb]850library function @cancel_stack@ is called, passing an exception. Unlike a
851raise, this exception is not used in matching, only to pass information about
[4706098c]852the cause of the cancellation.
[e3984a68]853Finally, as no handler is provided, there is no default handler.
[4706098c]854
[9cdfa5fb]855After @cancel_stack@ is called, the exception is copied into the EHM's memory
[4aba055]856and the current stack is unwound.
857The behaviour after that depends on the kind of stack being cancelled.
[a6c45c6]858
859\paragraph{Main Stack}
[9cdfa5fb]860The main stack is the one used by
861the program's main function at the start of execution,
[f6106a6]862and is the only stack in a sequential program.
[9cdfa5fb]863After the main stack is unwound, there is a program-level abort.
[f6106a6]864
[e3984a68]865The first reason for this behaviour is for sequential programs where there
[9cdfa5fb]866is only one stack, and hence no stack to pass information to.
[e3984a68]867Second, even in concurrent programs, the main stack has no dependency
868on another stack and no reliable way to find another living stack.
869Finally, keeping the same behaviour in both sequential and concurrent
870programs is simple and easy to understand.
[4706098c]871
[a6c45c6]872\paragraph{Thread Stack}
[f6106a6]873A thread stack is created for a \CFA @thread@ object or object that satisfies
874the @is_thread@ trait.
[4aba055]875After a thread stack is unwound, the exception is stored until another
[f6106a6]876thread attempts to join with it. Then the exception @ThreadCancelled@,
877which stores a reference to the thread and to the exception passed to the
[4aba055]878cancellation, is reported from the join to the joining thread.
[f6106a6]879There is one difference between an explicit join (with the @join@ function)
880and an implicit join (from a destructor call). The explicit join takes the
881default handler (@defaultResumptionHandler@) from its calling context while
[9cdfa5fb]882the implicit join provides its own, which does a program abort if the
[f6106a6]883@ThreadCancelled@ exception cannot be handled.
884
[4aba055]885The communication and synchronization are done here because threads only have
886two structural points (not dependent on user-code) where
887communication/synchronization happens: start and join.
[f6106a6]888Since a thread must be running to perform a cancellation (and cannot be
889cancelled from another stack), the cancellation must be after start and
[4aba055]890before the join, so join is used.
[f6106a6]891
892% TODO: Find somewhere to discuss unwind collisions.
893The difference between the explicit and implicit join is for safety and
894debugging. It helps prevent unwinding collisions by avoiding throwing from
895a destructor and prevents cascading the error across multiple threads if
896the user is not equipped to deal with it.
[33e1c91]897It is always possible to add an explicit join if that is the desired behaviour.
898
899With explicit join and a default handler that triggers a cancellation, it is
[e3984a68]900possible to cascade an error across any number of threads,
901alternating between the resumption (possibly termination) and cancellation,
902cleaning up each
[33e1c91]903in turn, until the error is handled or the main thread is reached.
[f6106a6]904
[a6c45c6]905\paragraph{Coroutine Stack}
[f6106a6]906A coroutine stack is created for a @coroutine@ object or object that
907satisfies the @is_coroutine@ trait.
[4aba055]908After a coroutine stack is unwound, control returns to the @resume@ function
909that most recently resumed it. @resume@ reports a
[814f87d]910@CoroutineCancelled@ exception, which contains a reference to the cancelled
[f6106a6]911coroutine and the exception used to cancel it.
[4aba055]912The @resume@ function also takes the \defaultResumptionHandler{} from the
[21f2e92]913caller's context and passes it to the internal report.
[f6106a6]914
[e3984a68]915A coroutine only knows of two other coroutines,
916its starter and its last resumer.
[4aba055]917The starter has a much more distant connection, while the last resumer just
[f6106a6]918(in terms of coroutine state) called resume on this coroutine, so the message
919is passed to the latter.
[33e1c91]920
921With a default handler that triggers a cancellation, it is possible to
[e3984a68]922cascade an error across any number of coroutines,
923alternating between the resumption (possibly termination) and cancellation,
924cleaning up each in turn,
[33e1c91]925until the error is handled or a thread stack is reached.
Note: See TracBrowser for help on using the repository browser.