When an exception is raised, normal program execution is abandoned and control is transferred to an exception handler. The selection of this handler depends on whether the exception is raised during the execution of statements or during the elaboration of declarations.
References: declaration, elaboration, and 3.9, exception, exception handler, raising of exceptions, statement.
Rationale references: 14.2.1 Declaration of Exceptions, 14.2.7 Order of Exceptions, 14.3 Examples, 14.4 Tasks and Exceptions, 14.4.1 Exceptions During Task Activation, 14.4.2 Exceptions Raised During Communication Between Tasks, 14.4.3 Abnormal Situations in an Accept Statement, 14.4.4 Example of Exceptions in a Rendezvous, 14.5 Technical Issues
Style Guide references: 4.3.1 Using Exceptions to Help Define an Abstraction, 5.8.1 Handling Versus Avoiding Exceptions, 5.8.2 Handlers for others, 5.8.3 Propagation, 5.8.4 Localizing the Cause of an Exception, 6.2.2 Defensive Task Communication, 6.3.4 Abnormal Termination, 7.5 Exceptions, 8.2.7 Exceptions
Sub-topics:
The handling of an exception raised by the execution of a sequence of statements depends on whether the innermost frame or accept statement that encloses the sequence of statements is a frame or an accept statement. The case where an accept statement is innermost is described in section 11.5. The case where a frame is innermost is presented here.
Different actions take place, depending on whether or not this frame has a handler for the exception, and on whether the exception is raised in the sequence of statements of the frame or in that of an exception handler.
If an exception is raised in the sequence of statements of a frame that has a handler for the exception, execution of the sequence of statements of the frame is abandoned and control is transferred to the exception handler. The execution of the sequence of statements of the handler completes the execution of the frame (or its elaboration if the frame is a package body).
If an exception is raised in the sequence of statements of a frame that does not have a handler for the exception, execution of this sequence of statements is abandoned. The next action depends on the nature of the frame:
Finally, if an exception is raised in the sequence of statements of an exception handler, execution of this sequence of statements is abandoned. Subsequent actions (including propagation, if any) are as in the cases (a) to (d) above, depending on the nature of the frame.
Example:
function FACTORIAL (N : POSITIVE) return FLOAT is begin if N = 1 then return 1.0; else return FLOAT(N) * FACTORIAL(N-1); end if; exception when NUMERIC_ERROR => return FLOAT'SAFE_LARGE; end FACTORIAL;
If the multiplication raises NUMERIC_ERROR, then FLOAT'SAFE_LARGE is returned by the handler. This value will cause further NUMERIC_ERROR exceptions to be raised by the evaluation of the expression in each of the remaining invocations of the function, so that for large values of N the function will ultimately return the value FLOAT'SAFE_LARGE.
Example:
procedure P is ERROR : exception; procedure R; procedure Q is begin R; ... -- error situation (2) exception ... when ERROR => -- handler E2 ... end Q; procedure R is begin ... -- error situation (3) end R; begin ... -- error situation (1) Q; ... exception ... when ERROR => -- handler E1 ... end P;
The following situations can arise:
Note that in the third situation, the exception raised in R results in (indirectly) transferring control to a handler that is part of Q and hence not enclosed by R. Note also that if a handler were provided within R for the exception choice others, situation (3) would cause execution of this handler, rather than direct termination of R.
Lastly, if ERROR had been declared in R, rather than in P, the handlers E1 and E2 could not provide an explicit handler for ERROR since this identifier would not be visible within the bodies of P and Q. In situation (3), the exception could however be handled in Q by providing a handler for the exception choice others.
Notes:
The language does not define what happens when the execution of the main program is abandoned after an unhandled exception.
The predefined exceptions are those that can be propagated by the basic operations and the predefined operators.
The case of a frame that is a generic unit is already covered by the rules for subprogram and package bodies, since the sequence of statements of such a frame is not executed but is the template for the corresponding sequences of statements of the subprograms or packages obtained by generic instantiation.
References: accept statement, basic operation, block statement, body stub, completion, declarative item, declarative part, dependent task, elaboration, and 3.9, exception, exception handler, frame, generic instantiation, generic unit, library unit, main program, numeric_error exception, package, package body, predefined operator, procedure, sequence of statements, statement, subprogram, subprogram body, subprogram call, subunit, task, task body.
Style Guide references: 5.8.1 Handling Versus Avoiding Exceptions
If an exception is raised during the elaboration of the declarative part of a given frame, this elaboration is abandoned. The next action depends on the nature of the frame:
Example of an exception in the declarative part of a block statement (case (b)):
procedure P is ... begin declare N : INTEGER := F; -- the function F may raise ERROR begin ... exception when ERROR => -- handler E1 end; ... exception when ERROR => -- handler E2 end P; -- if the exception ERROR is raised in the declaration of N, it is handled by E2
References: activation, block statement, body stub, completed task, declarative item, declarative part, elaboration, and 3.9, exception, frame, library unit, main program, package body, package declaration, package specification, subprogram, subprogram body, subprogram call, subunit, task, task body, task declaration, tasking_error exception.
Style Guide references: 5.8.1 Handling Versus Avoiding Exceptions
Address any questions or comments to adainfo@sw-eng.falls-church.va.us.