xref: /freebsd/contrib/sendmail/libsm/exc.html (revision 1c05a6ea6b849ff95e539c31adea887c644a6a01)
1<html>
2<head>
3    <title>libsm : Exception Handling</title>
4</head>
5<body>
6
7<a href="index.html">Back to libsm overview</a>
8
9<center>
10    <h1> libsm : Exception Handling </h1>
11    <br> $Id: exc.html,v 1.13 2006-06-20 17:18:16 ca Exp $
12</center>
13
14<h2> Introduction </h2>
15
16The exception handling package provides the facilities that
17functions in libsm use to report errors.
18Here are the basic concepts:
19
20<ol>
21<li>
22    When a function detects an exceptional condition at the library level,
23    it does not print an error message, or call syslog, or
24    exit the program.  Instead, it reports the error back to its
25    caller, and lets the caller decide what to do.
26    This improves modularity, because error handling is separated
27    from error reporting.
28    <p>
29<li>
30    Errors are not represented by a single integer error code,
31    because then you can't represent everything that an error handler
32    might need to know about an error by a single integer.
33    Instead, errors are represented by exception objects.
34    An exception object contains an exception code and an array
35    of zero or more exception arguments.
36    The exception code is a string that specifies what kind of exception
37    this is, and the arguments may be integers, strings or exception objects.
38    <p>
39<li>
40    Errors are not reported using a special return value,
41    because if you religiously check for error returns from every
42    function call that could fail, then most of your code ends up being
43    error handling code.  Errors are reported by raising an exception.
44    When an exception is raised, we unwind the call stack
45    until we find an exception handler.  If the exception is
46    not handled, then we print the exception on stderr and
47    exit the program.
48</ol>
49
50<h2> Synopsis </h2>
51
52<pre>
53#include &lt;sm/exc.h&gt;
54
55typedef struct sm_exc_type SM_EXC_TYPE_T;
56typedef struct sm_exc SM_EXC_T;
57typedef union sm_val SM_VAL_T;
58
59/*
60**  Exception types
61*/
62
63extern const char SmExcTypeMagic[];
64
65struct sm_exc_type
66{
67	const char	*sm_magic;
68	const char	*etype_category;
69	const char	*etype_argformat;
70	void 		(*etype_print)(SM_EXC_T *exc, SM_FILE_T *stream);
71	const char	*etype_printcontext;
72};
73
74extern const SM_EXC_TYPE_T SmEtypeOs;
75extern const SM_EXC_TYPE_T SmEtypeErr;
76
77void
78sm_etype_printf(
79	SM_EXC_T *exc,
80	SM_FILE_T *stream);
81
82/*
83**  Exception objects
84*/
85
86extern const char SmExcMagic[];
87
88union sm_val
89{
90	int		v_int;
91	long		v_long;
92	char		*v_str;
93	SM_EXC_T	*v_exc;
94};
95
96struct sm_exc
97{
98	const char		*sm_magic;
99	size_t			exc_refcount;
100	const SM_EXC_TYPE_T	*exc_type;
101	SM_VAL_T		*exc_argv;
102};
103
104SM_EXC_T *
105sm_exc_new_x(
106	const SM_EXC_TYPE_T *type,
107	...);
108
109SM_EXC_T *
110sm_exc_addref(
111	SM_EXC_T *exc);
112
113void
114sm_exc_free(
115	SM_EXC_T *exc);
116
117bool
118sm_exc_match(
119	SM_EXC_T *exc,
120	const char *pattern);
121
122void
123sm_exc_print(
124	SM_EXC_T *exc,
125	SM_FILE_T *stream);
126
127void
128sm_exc_write(
129	SM_EXC_T *exc,
130	SM_FILE_T *stream);
131
132void
133sm_exc_raise_x(
134	SM_EXC_T *exc);
135
136void
137sm_exc_raisenew_x(
138	const SM_EXC_TYPE_T *type,
139	...);
140
141/*
142**  Ensure that cleanup code is executed,
143**  and/or handle an exception.
144*/
145SM_TRY
146	Block of code that may raise an exception.
147SM_FINALLY
148	Cleanup code that may raise an exception.
149	This clause is guaranteed to be executed even if an exception is
150	raised by the SM_TRY clause or by an earlier SM_FINALLY clause.
151	You may have 0 or more SM_FINALLY clauses.
152SM_EXCEPT(exc, pattern)
153	Exception handling code, triggered by an exception
154	whose category matches 'pattern'.
155	You may have 0 or more SM_EXCEPT clauses.
156SM_END_TRY
157</pre>
158
159<h2> Overview </h2>
160
161    An exception is an object which represents an exceptional condition,
162    which might be an error condition like "out of memory", or might be
163    a condition like "end of file".
164<p>
165    Functions in libsm report errors and other unusual conditions by
166    raising an exception, rather than by returning an error code or
167    setting a global variable such as errno.  If a libsm function is
168    capable of raising an exception, its name ends in "_x".
169    (We do not raise an exception when a bug is detected in the
170    program; instead, we terminate the program using <tt>sm_abort</tt>.
171    See <a href="assert.html">the assertion package</a>
172    for details.)
173<p>
174    When you are using the libsm exception handling package,
175    you are using a new programming paradigm.
176    You will need to abandon some of the programming idioms
177    you are accustomed to, and switch to new idioms.
178    Here is an overview of some of these idioms.
179<ol>
180<li>
181	When a function is unable to complete its task because
182	of an exceptional condition, it reports this condition
183	by raising an exception.
184	<p>
185	Here is an example of how to construct an exception object
186	and raise an exception.
187	In this example, we convert a Unix system error into an exception.
188<blockquote><pre>
189fd = open(path, O_RDONLY);
190if (fd == -1)
191	sm_exc_raise_x(sm_exc_new_x(&SmEtypeOs, errno, "open", "%s", path));
192</pre></blockquote>
193
194	Because the idiom <tt>sm_exc_raise_x(sm_exc_new_x(...))</tt>
195	is so common, it can be abbreviated as <tt>sm_exc_raisenew_x(...)</tt>.
196<p>
197<li>
198	When you detect an error at the application level,
199	you don't call a function like BSD's <tt>errx</tt>,
200	which prints an error message on stderr and exits the program.
201	Instead, you raise an exception.
202	This causes cleanup code in surrounding exception handlers
203	to be run before the program exits.
204	For example, instead of this:
205<blockquote><pre>
206errx(1, "%s:%d: syntax error", filename, lineno);
207</pre></blockquote>
208
209	use this:
210
211<blockquote><pre>
212sm_exc_raisenew_x(&SmEtypeErr, "%s:%d: syntax error", filename, lineno);
213</pre></blockquote>
214
215	The latter code raises an exception, unwinding the call stack
216	and executing cleanup code.
217	If the exception is not handled, then the exception is printed
218	to stderr and the program exits.
219	The end result is substantially the same as a call to <tt>errx</tt>.
220<p>
221<li>
222        The SM_TRY ... SM_FINALLY ... control structure
223	ensures that cleanup code is executed and resources are released
224	in the presence of exceptions.
225<p>
226	For example, suppose that you have written the following code:
227
228<blockquote><pre>
229rpool = sm_rpool_new_x(&SmRpoolRoot, 0);
230... some code ...
231sm_rpool_free_x(rpool);
232</pre></blockquote>
233
234	If any of the functions called within "... some code ..." have
235	names ending in _x, then it is possible that an exception will be
236	raised, and if that happens, then "rpool" will not be freed.
237	And that's a bug.  To fix this bug, change your code so it looks
238	like this:
239
240<blockquote><pre>
241rpool = sm_rpool_new_x(&SmRpoolRoot, 0);
242SM_TRY
243	... some code that can raise an exception ...
244SM_FINALLY
245	sm_rpool_free_x(rpool);
246SM_END_TRY
247</pre></blockquote>
248
249<li>
250	The SM_TRY ... SM_EXCEPT ... control structure handles an exception.
251	Unhandled exceptions terminate the program.
252	For example, here is a simple exception handler
253	that traps all exceptions, and prints the exceptions:
254
255<blockquote><pre>
256SM_TRY
257	/* code that can raise an exception */
258	...
259SM_EXCEPT(exc, "*")
260	/* catch all exceptions */
261	sm_exc_print(exc, stderr);
262SM_END_TRY
263</pre></blockquote>
264
265    Exceptions are reference counted.  The SM_END_TRY macro contains a
266    call to sm_exc_free, so you don't normally need to worry about freeing
267    an exception after handling it.  In the rare case that you want an
268    exception to outlive an exception handler, then you increment its
269    reference count by calling sm_exc_addref.
270<p>
271<li>
272    The second argument of the SM_EXCEPT macro is a glob pattern
273    which specifies the types of exceptions that are to be handled.
274    For example, you might want to handle an end-of-file exception
275    differently from other exceptions.
276    Here's how you do that:
277
278<blockquote><pre>
279SM_TRY
280	/* code that might raise end-of-file, or some other exception */
281	...
282SM_EXCEPT(exc, "E:sm.eof")
283	/* what to do if end-of-file is encountered */
284	...
285SM_EXCEPT(exc, "*")
286	/* what to do if some other exception is raised */
287	...
288SM_END_TRY
289</pre></blockquote>
290</ol>
291
292<h2> Exception Values </h2>
293
294In traditional C code, errors are usually denoted by a single integer,
295such as errno.  In practice, errno does not carry enough information
296to describe everything that an error handler might want to know about
297an error.  And the scheme is not very extensible: if several different
298packages want to add additional error codes, it is hard to avoid
299collisions.
300
301<p>
302In libsm, an exceptional condition is described
303by an object of type SM_EXC_T.
304An exception object is created by specifying an exception type
305and a list of exception arguments.
306
307<p>
308The exception arguments are an array of zero or more values.
309The values may be a mixture of ints, longs, strings, and exceptions.
310In the SM_EXC_T structure, the argument vector is represented
311by <tt>SM_VAL_T&nbsp;*exc_argv</tt>, where <tt>SM_VAL_T</tt>
312is a union of the possible argument types.
313The number and types of exception arguments is determined by
314the exception type.
315
316<p>
317An exception type is a statically initialized const object
318of type SM_EXC_TYPE_T, which has the following members:
319
320<dl>
321<dt>
322<tt> const char *sm_magic </tt>
323<dd>
324	A pointer to <tt>SmExcTypeMagic</tt>.
325	<p>
326<dt>
327<tt> const char *etype_category </tt>
328<dd>
329	This is a string of the form
330	<tt>"</tt><i>class</i><tt>:</tt><i>name</i><tt>"</tt>.
331	<p>
332	The <i>class</i> is used to assign the exception type to
333	one of a number of broad categories of exceptions on which an
334	exception handler might want to discriminate.
335	I suspect that what we want is a hierarchical taxonomy,
336	but I don't have a full design for this yet.
337	For now, I am recommending the following classes:
338	<dl>
339	<dt><tt>"F"</tt>
340	<dd>A fatal error has occurred.
341	    This is an error that prevents the application
342	    from making any further progress, so the only
343	    recourse is to raise an exception, execute cleanup code
344	    as the stack is unwound, then exit the application.
345	    The out-of-memory exception raised by sm_malloc_x
346	    has category "F:sm.heap" because sendmail commits suicide
347	    (after logging the error and cleaning up) when it runs out
348	    of memory.
349
350	<dt><tt>"E"</tt>
351	<dd>The function could not complete its task because an error occurred.
352	    (It might be useful to define subclasses of this category,
353	    in which case our taxonony becomes a tree, and 'F' becomes
354	    a subclass of 'E'.)
355
356	<dt><tt>"J"</tt>
357	<dd>This exception is being raised in order to effect a
358	    non-local jump.  No error has occurred; we are just
359	    performing the non-local equivalent of a <tt>continue</tt>,
360	    <tt>break</tt> or <tt>return</tt>.
361
362	<dt><tt>"S"</tt>
363	<dd>The function was interrupted by a signal.
364	    Signals are not errors because they occur asynchronously,
365	    and they are semantically unrelated to the function that
366	    happens to be executing when the signal arrives.
367	    Note that it is extremely dangerous to raise an exception
368	    from a signal handler.  For example, if you are in the middle
369	    of a call to malloc, you might corrupt the heap.
370	</dl>
371	Eric's libsm paper defines <tt>"W"</tt>, <tt>"D"</tt> and <tt>"I"</tt>
372	for Warning, Debug and Informational:
373	I suspect these categories only make sense in the context of
374	Eric's 1985 exception handling system which allowed you to
375	raise conditions without terminating the calling function.
376	<p>
377	The <i>name</i> uniquely identifies the exception type.
378	I recommend a string of the form
379	<i>library</i><tt>.</tt><i>package</i><tt>.</tt><i>detail</i>.
380	<p>
381<dt>
382<tt> const char *etype_argformat </tt>
383<dd>
384	This is an array of single character codes.
385	Each code indicates the type of one of the exception arguments.
386	<tt>sm_exc_new_x</tt> uses this string to decode its variable
387	argument list into an exception argument vector.
388	The following type codes are supported:
389	<dl>
390	<dt><tt>i</tt>
391	<dd>
392		The exception argument has type <tt>int</tt>.
393	<dt><tt>l</tt>
394	<dd>
395		The exception argument has type <tt>long</tt>.
396	<dt><tt>e</tt>
397	<dd>
398		The exception argument has type <tt>SM_EXC_T*</tt>.
399		The value may either be <tt>NULL</tt> or a pointer
400		to an exception.  The pointer value is simply copied
401		into the exception argument vector.
402	<dt><tt>s</tt>
403	<dd>
404		The exception argument has type <tt>char*</tt>.
405		The value may either be <tt>NULL</tt> or a pointer
406		to a character string.  In the latter case,
407		<tt>sm_exc_new_x</tt> will make a copy of the string.
408	<dt><tt>r</tt>
409	<dd>
410		The exception argument has type <tt>char*</tt>.
411		<tt>sm_exc_new_x</tt> will read a printf-style
412		format string argument followed by a list of printf
413		arguments from its variable argument list, and convert
414		these into a string.
415		This type code can only occur as the last element
416		of <tt>exc_argformat</tt>.
417	</dl>
418	<p>
419<dt>
420<tt> void (*etype_print)(SM_EXC_T *exc, SM_FILE_T *stream) </tt>
421<dd>
422	This function prints an exception of the specified type
423	onto an output stream.
424	The final character printed is not a newline.
425</dl>
426
427<h2> Standard Exceptions and Exception Types </h2>
428
429Libsm defines one standard exception value, <tt>SmHeapOutOfMemory</tt>.
430This is a statically initialized const variable, because it seems
431like a bad idea to dynamically allocate an exception object to
432report a low memory condition.
433This exception has category <tt>"F:sm.heap"</tt>.
434If you need to, you can explicitly raise this exception
435with <tt>sm_exc_raise_x(&SmHeapOutOfMemory)</tt>.
436
437<p>
438Statically initialized exception values cannot contain any
439run-time parameters, so the normal case is to dynamically allocate
440a new exception object whenever you raise an exception.
441Before you can create an exception, you need an exception type.
442Libsm defines the following standard exception types.
443
444<dl>
445<dt>
446<tt> SmEtypeOs </tt>
447<dd>
448	This represents a generic operating system error.
449	The category is <tt>"E:sm.os"</tt>.
450	The argformat is <tt>"isr"</tt>,
451	where argv[0] is the value of <tt>errno</tt>
452	after a system call has failed,
453	argv[1] is the name of the function (usually a system call) that failed,
454	and argv[2] is either <tt>NULL</tt>
455	or a character string which describes some of the arguments
456	to the failing system call (usually it is just a file name).
457	Here's an example of raising an exception:
458
459<blockquote><pre>
460fd = open(filename, O_RDONLY);
461if (fd == -1)
462	sm_exc_raisenew_x(&SmEtypeOs, errno, "open", "%s", filename);
463</pre></blockquote>
464
465	If errno is ENOENT and filename is "/etc/mail/snedmail.cf",
466	then the exception raised by the above code will be printed as
467
468<blockquote><pre>
469/etc/mail/snedmail.cf: open failed: No such file or directory
470</pre></blockquote>
471
472<dt>
473<tt> SmEtypeErr </tt>
474<dd>
475	This represents a generic error.
476	The category is <tt>"E:sm.err"</tt>,
477	and the argformat is <tt>"r"</tt>.
478	You can use it
479	in application contexts where you are raising an exception
480	for the purpose of terminating the program.
481	You know the exception won't be handled,
482	so you don't need to worry about packaging the error for
483	later analysis by an exception handler.
484	All you need to specify is the message string that
485	will be printed to stderr before the program exits.
486	For example,
487
488<blockquote><pre>
489sm_exc_raisenew_x(&SmEtypeErr, "name lookup failed: %s", name);
490</pre></blockquote>
491</dl>
492
493<h2> Custom Exception Types </h2>
494
495If you are writing a library package, and you need to raise
496exceptions that are not standard Unix system errors,
497then you need to define one or more new exception types.
498
499<p>
500Every new exception type needs a print function.
501The standard print function <tt>sm_etype_printf</tt>
502is all you need in the majority of cases.
503It prints the <tt>etype_printcontext</tt> string of the exception type,
504substituting occurrences of %0 through %9 with the corresponding
505exception argument.
506If exception argument 3 is an int or long,
507then %3 will print the argument in decimal,
508and %o3 or %x3 will print it in octal or hex.
509
510<p>
511In the following example, I will assume that your library
512package implements regular expressions, and can raise 5 different exceptions.
513When compiling a regular expression, 3 different syntax errors
514can be reported:
515<ul>
516<li>unbalanced parenthesis
517<li>unbalanced bracket
518<li>missing argument for repetition operator
519</ul>
520Whenever one of these errors is reported, you will also report
521the index of the character within the regex string at which the
522syntax error was detected.
523The fourth exception is raised if a compiled regular expression
524is invalid: this exception has no arguments.
525The fifth exception is raised if the package runs out of memory:
526for this, you use the standard <tt>SmHeapOutOfMemory</tt> exception.
527
528<p>
529The obvious approach is to define 4 separate exception types.
530Here they are:
531
532<blockquote><pre>
533/* print a regular expression syntax error */
534void
535rx_esyntax_print(SM_EXC_T *exc, SM_FILE_T *stream)
536{
537	sm_io_fprintf(stream, "rx syntax error at character %d: %s",
538		exc-&gt;exc_argv[0].v_int,
539		exc-&gt;exc_type-&gt;etype_printcontext);
540}
541SM_EXC_TYPE_T RxSyntaxParen = {
542	SmExcTypeMagic,
543	"E:mylib.rx.syntax.paren",
544	"i",
545	rx_esyntax_print,
546	"unbalanced parenthesis"
547};
548SM_EXC_TYPE_T RxSyntaxBracket = {
549	SmExcTypeMagic,
550	"E:mylib.rx.syntax.bracket",
551	"i",
552	rx_esyntax_print,
553	"unbalanced bracket"
554};
555SM_EXC_TYPE_T RxSyntaxMissingArg = {
556	SmExcTypeMagic,
557	"E:mylib.rx.syntax.missingarg",
558	"i",
559	rx_esyntax_print,
560	"missing argument for repetition operator"
561};
562
563SM_EXC_TYPE_T RxRunCorrupt = {
564	SmExcTypeMagic,
565	"E:mylib.rx.run.corrupt",
566	"",
567	sm_etype_printf,
568	"rx runtime error: compiled regular expression is corrupt"
569};
570</pre></blockquote>
571
572<p>
573With the above definitions, you can raise a syntax error reporting
574an unbalanced parenthesis at string offset <tt>i</tt> using:
575<blockquote><pre>
576sm_exc_raisenew_x(&RxSyntaxParen, i);
577</pre></blockquote>
578
579If <tt>i==42</tt> then this exception will be printed as:
580<blockquote><pre>
581rx syntax error at character 42: unbalanced parenthesis
582</pre></blockquote>
583
584An exception handler can provide special handling for regular
585expression syntax errors using this code:
586<blockquote><pre>
587SM_TRY
588	... code that might raise an exception ...
589SM_EXCEPT(exc, "E:mylib.rx.syntax.*")
590	int i = exc-&gt;exc_argv[0].v_int;
591	... handle a regular expression syntax error ...
592SM_END_TRY
593</pre></blockquote>
594
595<p>
596External requirements may force you to define an integer code
597for each error reported by your package.  Or you may be wrapping
598an existing package that works this way.  In this case, it might
599make sense to define a single exception type, patterned after SmEtypeOs,
600and include the integer code as an exception argument.
601
602<p>
603Your package might intercept an exception E generated by a lower
604level package, and then reclassify it as a different expression E'.
605For example, a package for reading a configuration file might
606reclassify one of the regular expression syntax errors from the
607previous example as a configuration file syntax error.
608When you do this, the new exception E' should include the original
609exception E as an exception parameter, and the print function for
610exception E' should print the high level description of the exception
611(eg, "syntax error in configuration file %s at line %d\n"),
612then print the subexception that is stored as an exception parameter.
613
614<h2> Function Reference </h2>
615
616<dl>
617<dt>
618<tt> SM_EXC_T *sm_exc_new_x(const SM_EXC_TYPE_T *type, ...) </tt>
619<dd>
620	Create a new exception.  Raise an exception on heap exhaustion.
621	The new exception has a reference count of 1.
622	<p>
623
624	A list of zero or more exception arguments follows the exception type;
625	these are copied into the new exception object.
626	The number and types of these arguments is determined
627	by <tt>type-&gt;etype_argformat</tt>.
628	<p>
629
630	Note that there is no rpool argument to sm_exc_new_x.
631	Exceptions are allocated directly from the heap.
632	This is because exceptions are normally raised at low levels
633	of abstraction and handled at high levels.  Because the low
634	level code typically has no idea of how or at what level the
635	exception will be handled, it also has no idea of which resource
636	pool, if any, should own the exception.
637	<p>
638<dt>
639<tt> SM_EXC_T *sm_exc_addref(SM_EXC_T *exc) </tt>
640<dd>
641	Increment the reference count of an exception.
642	Return the first argument.
643	<p>
644<dt>
645<tt> void sm_exc_free(SM_EXC_T *exc) </tt>
646<dd>
647	Decrement the reference count of an exception.
648	If it reaches 0, free the exception object.
649	<p>
650<dt>
651<tt> bool sm_exc_match(SM_EXC_T *exc, const char *pattern) </tt>
652<dd>
653	Compare the exception's category to the specified glob pattern,
654	return true if they match.
655	<p>
656<dt>
657<tt> void sm_exc_print(SM_EXC_T *exc, SM_FILE_T *stream) </tt>
658<dd>
659	Print the exception on the stream
660	as a sequence of one or more newline terminated lines.
661	<p>
662<dt>
663<tt> void sm_exc_write(SM_EXC_T *exc, SM_FILE_T *stream) </tt>
664<dd>
665	Write the exception on the stream without a terminating newline.
666	<p>
667<dt>
668<tt> void sm_exc_raise_x(SM_EXC_T *exc) </tt>
669<dd>
670	Raise the exception.  This function does not return to its caller.
671	<p>
672<dt>
673<tt> void sm_exc_raisenew_x(const SM_EXC_TYPE_T *type, ...) </tt>
674<dd>
675	A short form for <tt>sm_exc_raise_x(sm_exc_new_x(type,...))</tt>.
676</dl>
677
678<h2> Macro Reference </h2>
679
680The SM_TRY ... SM_END_TRY control structure
681ensures that cleanup code is executed in the presence of exceptions,
682and permits exceptions to be handled.
683
684<blockquote><pre>
685SM_TRY
686	A block of code that may raise an exception.
687SM_FINALLY
688	Cleanup code that may raise an exception.
689	This code is guaranteed to be executed whether or not
690	an exception was raised by a previous clause.
691	You may have 0 or more SM_FINALLY clauses.
692SM_EXCEPT(e, pat)
693	Exception handling code, which is triggered by an exception
694	whose category matches the glob pattern 'pat'.
695	The exception value is bound to the local variable 'e'.
696	You may have 0 or more SM_EXCEPT clauses.
697SM_END_TRY
698</pre></blockquote>
699
700First, the SM_TRY clause is executed, then each SM_FINALLY clause is
701executed in sequence.
702If one or more of these clauses was terminated by an exception,
703then the first such exception is remembered, and the other exceptions
704are lost.
705
706If no exception was raised, then we are done.
707
708Otherwise, each of the SM_EXCEPT clauses is examined in sequence.
709and the first SM_EXCEPT clause whose pattern argument matches the exception
710(see <tt>sm_exc_match</tt>) is executed.
711If none of the SM_EXCEPT clauses matched the exception, or if there are
712no SM_EXCEPT clauses, then the remembered exception is re-raised.
713
714<p>
715SM_TRY .. SM_END_TRY clauses may be nested arbitrarily.
716
717<p>
718It is illegal to jump out of a SM_TRY or SM_FINALLY clause
719using goto, break, continue, return or longjmp.
720If you do this, you will corrupt the internal exception handling stack.
721You can't use <tt>break</tt> or <tt>continue</tt> in an SM_EXCEPT clause;
722these are reserved for use by the implementation.
723It is legal to jump out of an SM_EXCEPT clause using goto or return;
724however, in this case, you must take responsibility
725for freeing the exception object.
726
727<p>
728The SM_TRY and SM_FINALLY macros contain calls to setjmp,
729and consequently, they suffer from the limitations imposed on setjmp
730by the C standard.
731Suppose you declare an auto variable <tt>i</tt> outside of a
732SM_TRY ... SM_END_TRY statement, initializing it to 0.
733Then you modify <tt>i</tt> inside of a SM_TRY or SM_FINALLY clause,
734setting it to 1.
735If you reference <tt>i</tt> in a different SM_FINALLY clause, or in
736an SM_EXCEPT clause, then it is implementation dependent whether <tt>i</tt>
737will be 0 or 1, unless you have declared <tt>i</tt> to be <tt>volatile</tt>.
738
739<blockquote><pre>
740int volatile i = 0;
741
742SM_TRY
743	i = 1;
744	...
745SM_FINALLY
746	/* the following reference to i only works if i is declared volatile */
747	use(i);
748	...
749SM_EXCEPT(exc, "*")
750	/* the following reference to i only works if i is declared volatile */
751	use(i);
752	...
753SM_END_TRY
754</pre></blockquote>
755
756</body>
757</html>
758