/try

Simple and clean exception handling in C

Primary LanguageCMIT LicenseMIT

try

Discord Channel

Simple and clean exception handling in C

example


  #include "try.h"

  #define OUTOFMEM    1
  #define WRONGINPUT  2
  #define INTERNALERR 3

  // Define a variable of type `try_t` for the use of the library
  try_t catch = 0;

  void some_other_func() 
  {
    ... code ...
     throw(WRONGINPUT);
    ... code ...
  }

  void a_function()
  {
    try {
       ... code ...
       if (something_failed) 
         throw(execption_num)  // exception_num must be between 1 and 65535 
   
       some_other_func();      // you can throw exceptions
                               // from other functions 
   
       ... code ...            // You MUST NEVER jump out of a try/catch block
                               // (via return, goto or break). Use throw() to
                               // ensure the right behaviour.
     }  
     catch(OUTOFMEM) {
       ... code ...
     }
     catch(WRONGINPUT) {
       ... code ...
     }
     catch() {                 // catch any other exception
       ... code ...            // otherwise the progam would abort.
     }
  }

Note that you can't put a try/catch block within a catch block, and you shouldn't, as this could be a sign that you are using try/catch to control the flow of your program rather than just handling exceptional events.

Of course, you can nest as many try/catch block in a try block; even indirectly through function calls.

Use try/catch in your program

  • Set your INCLUDE PATH so that try.h is reachable and include it from youre source files.

  • In one (and only one) of your source files define a global variable of type try_t and set it to 0 (I suggest using try_t catch = 0;) The library needs this definition to keep track of try blocks nesting it is not supposed to be used or changed by the programmer.

  • Alternatively, you can compile try.c and link against try.o

See the tests in the test directory for some example.

Reference

Try/Catch

  try {
     // Block of code that could potentially throw an exception
     throw(an_exception); // Directly or
     a_function();        // possibly generated
                          // in a function called by this block
  }
  catch(an_exception) { 
     // Block of code that is executed if the specified
     // exception is thrown within the try block
     if (something) leave(); // Only exit via `leave()`
     ... do something else ...
  }
  catch(check_function) { 
     // The speficied function (with signature int f(int x)) is
     // called passing the exception number as the argument.
     // The block of code is executed if the functions returns 
     // a non-zero value
     ... do something ...
  }
  catch() {
     // Block of code that is executed if no other handler
     // catches the exception. It is highly reccomended to 
     // ALWAYS have a catch() block; unhandled exceptions
     // will make the program abort().
  }

Remember to never leave a try/catch block with return, goto or break! If you really need to exit in an unstructured way, use leave() that must be outside any loop or switch block. It will move you to the end of the try/catch block.

Throwing exceptions.

  throw(exc [, ...])  Throws an exception (a positive integer).
                      The optional values are passed as part of the exception and 
                      can be retrieved through the `exception` object (see below).
                      This can be useful to specify further information about the exception.

  rethrow([...])      Only usable in a catch block to throw the same exception (possibly
                      with new additional information).

  exception           This object will help you getting the information about the
                      exception that had been thrown.
                      Default information are:
                        exception.exception_num  (int)    The exception thrown 
                        exception.file_name      (char *) The source file where it happened
                        exception.line_num       (int)    The source line number where it happened

                      Additional information can be specified by defining the macro `exception_info`
                      **before** including `try.h`.

                      Example:
                        #define exception_info  time_t time_stamp; int seq_num;
                        #include "try.h"
                                                 
                        if you throw this exception:

                           throw(ENOMEM, .time_stamp=time(NULL), global_seq++);

                        in the catch block, you can retrive the time stamp with 

                           exception.time_stamp

                        You can also only specify information by name:
                           throw(ENOMEM, .seq_num = global_seq++);

  tryabort()          Is a function that is invoked when an unhandled exception
                      is triggered. It can be useful for printing an informative
                      message or doing some cleanup before aborting the program.

                      The function can be easily overwritten:
 
                          #define tryabort() my_handler()
                          #include "try.h"
 
                          void my_handler()
                          {
                              fprintf(stderr,"Something wrong (%d) at %s:%d\n",
                                             exception.exception_num,
                                             exception.file_name,
                                             exception.line_num);
                          }