// $Id$ // ============================================================================ /** * @file CORBA_macros.h * * Writing code that is portable between platforms with or without * native C++ exceptions is hard. The following macros offer some * help on this task, mostly oriented to making the ORB code and the * IDL generated code portable. * * @author Nanbor Wang * @author Based on the original implementation by * @author Aniruddha Gokhale * @author Carlos O'Ryan , et al. */ // ============================================================================ // Macros for handling CORBA exceptions. #ifndef ACE_CORBA_MACROS_H #define ACE_CORBA_MACROS_H #include "ace/pre.h" # if !defined (ACE_LACKS_PRAGMA_ONCE) # pragma once # endif /* ACE_LACKS_PRAGMA_ONCE */ #include "ace/OS.h" // All these macros assume the CORBA::Environment variable used to pass // in/out the exception is call ACE_TRY_ENV. Below is the name we use // in TAO (The ACE ORB.) Most other ORB's have their own naming // convention. You can redefine ACE_TRY_ENV to change the default name // ACE_ADOPT_CORBA_ENV allows the use of non-standard name within a // scope. #if !defined (ACE_TRY_ENV) # define ACE_TRY_ENV _ACE_CORBA_Environment_variable #endif /* ACE_TRY_ENV */ // This is the exception caught by ACE_CATCHANY. #if !defined (ACE_ANY_EXCEPTION) # define ACE_ANY_EXCEPTION ex #endif /* ACE_ANY_EXCEPTION */ // By default, if the compiler support native exception handling, assume // CORBA also support native exception handling. But it can be disabled // by defining ACE_CORBA_HAS_EXCEPTIONS=0. // If the compiler does not support exceptions handling, make sure native // exception handling is disabled. #if defined (ACE_HAS_EXCEPTIONS) # if defined (ACE_CORBA_HAS_EXCEPTIONS) # if (ACE_CORBA_HAS_EXCEPTIONS == 0) # undef ACE_CORBA_HAS_EXCEPTIONS # endif /* ACE_CORBA_HAS_EXCEPTIONS == 0 */ # else /* !ACE_CORBA_HAS_EXCEPTIONS */ # define ACE_CORBA_HAS_EXCEPTIONS # endif /* ACE_CORBA_HAS_EXCEPTIONS */ #else # if defined (ACE_CORBA_HAS_EXCEPTIONS) # undef ACE_CORBA_HAS_EXCEPTIONS # endif /* ACE_CORBA_HAS_EXCEPTIONS */ #endif /* ACE_HAS_EXCEPTIONS */ #define ACE_DECLARE_NEW_CORBA_ENV \ CORBA::Environment ACE_TRY_ENV #if defined (ACE_CORBA_HAS_EXCEPTIONS) // ----------------------------------------------------------------- # define ACE_ADOPT_CORBA_ENV(ENV) ACE_UNUSED_ARG(ENV) // No need to check. Native exceptions handle the control // flow automatically when an exception occurs. # define ACE_CHECK // Used then the function requires a return value. # define ACE_CHECK_RETURN(RETV) // ACE_THROW_INT should not be used by the user. # define ACE_THROW_INT(EXCEPTION) \ do { \ ACE_UNUSED_ARG(ACE_TRY_ENV); \ throw EXCEPTION; \ } while (0) // Throwing an exception is easy. These two macros should _NOT_ be // used within try blocks. # define ACE_THROW(EXCEPTION) \ do { \ ACE_UNUSED_ARG(ACE_TRY_ENV); \ throw EXCEPTION; \ } while (0) // Throwing an exception when the function reqires a return value. # if defined (WIN32) || defined (__HP_aCC) # define ACE_THROW_RETURN(EXCEPTION, RETV) \ do \ { \ ACE_UNUSED_ARG(ACE_TRY_ENV); \ throw EXCEPTION; \ return RETV; \ } while (0) # else /* WIN32 */ # define ACE_THROW_RETURN(EXCEPTION,RETV) \ do { \ ACE_UNUSED_ARG(ACE_TRY_ENV); \ throw EXCEPTION; \ } while (0) # endif /* WIN32 */ // For compilers with native exceptions, we can simply use // try to try. ;-) do {} while (0) is required to avoid // compilation warnings. # define ACE_TRY \ do \ { \ try \ { # define ACE_TRY_NEW_ENV \ do \ { \ CORBA::Environment ACE_TRY_ENV; \ try \ { # define ACE_TRY_EX(LABEL) \ do \ { \ try \ { // No need to check for exceptions within try block for compilers with // native exceptions. # define ACE_TRY_CHECK # define ACE_TRY_CHECK_EX(LABEL) // Likewise, throwing exceptions within try blocks is easy. # define ACE_TRY_THROW(EXCEPTION) throw EXCEPTION # define ACE_TRY_THROW_EX(EXCEPTION,LABEL) throw EXCEPTION // Same thing for catch. # define ACE_CATCH(EXCEPTION,VAR) \ } \ catch (EXCEPTION & VAR) \ { \ ACE_UNUSED_ARG (VAR); # define ACE_CATCHANY \ ACE_CATCH(CORBA::Exception, ACE_ANY_EXCEPTION) # define ACE_CATCHALL \ } \ catch (...) \ { # if defined (ACE_HAS_DEPRECATED_ACE_RETHROW) # define ACE_RETHROW throw # endif /* ACE_HAS_DEPRECATED_ACE_RETHROW */ // Rethrowing the exception from catch blocks. # define ACE_RE_THROW throw # define ACE_RE_THROW_EX(LABEL) throw // Close the catch block. # define ACE_ENDTRY \ } \ } while (0) #else /* ! ACE_CORBA_HAS_EXCEPTIONS */ // ----------------------------------------------------------------- // To handle compilers without native exceptions, things get a bit // hairy. Exceptions are simulated using CORBA::Environment. // The trick here is to make sure the flow-of-control can simulate // the case when native exceptions occur... # define ACE_ADOPT_CORBA_ENV(ENV) CORBA::Environment &ACE_TRY_ENV = ENV // Follow every statement that could throw exceptions with ACE_CHECK // or ACE_CHECK_ENV. These two macros should _NOT_ be used within // try blocks. Use ACE_TRY_CHECK or ACE_TRY_CHECK_EX instead. # define ACE_CHECK \ if (ACE_TRY_ENV . exception () != 0) \ return // When function requires a return value # define ACE_CHECK_RETURN(RETV) \ if (ACE_TRY_ENV . exception () != 0) \ return RETV // ACE_THROW_INT should not be used by the user. # define ACE_THROW_INT(EXCEPTION) ACE_TRY_ENV.exception (new EXCEPTION) // Throwing exceptions will inevitably cause an return from the current // function. These two macros should _NOT_ be used within try blocks. // Use ACE_TRY_THROW or ACE_TRY_THROW_EX instead. # define ACE_THROW(EXCEPTION) \ do \ { \ ACE_TRY_ENV.exception (new EXCEPTION); \ return; \ } while (0) # define ACE_THROW_RETURN(EXCEPTION,RETV) \ do \ { \ ACE_TRY_ENV.exception (new EXCEPTION); \ return RETV; \ } while (0) // ACE_TRY sets up flags to control program flow. ACE_TRY_FLAG acts // like a one-shot flip-flop. When an exception occured (detect it // using ACE_TRY_CHECK,) ACE_TRY_FLAG will be reset and the control // goes back into ACE_TRY_LABEL. Since ACE_TRY_FLAG is reset, the try // block won't get executed again and the control proceeds to the following // catch blocks. ACE_EXECTION_NOT_CAUGHT flag is used to prevent // catching an exception twice. This macro assumes there's already an // CORBA:;Environment variable ACE_TRY_ENV defined (which, should be // the most usual case.) # define ACE_TRY \ do { \ int ACE_TRY_FLAG = 1; \ int ACE_EXCEPTION_NOT_CAUGHT = 1; \ ACE_TRY_LABEL: \ if (ACE_TRY_FLAG) \ do { // ACE_TRY_NEW_ENV functions like the macro ACE_TRY but defines a // new CORBA::Environment variable ACE_TRY_ENV. It is most often // used in the outer most function where no ACE_TRY_ENV is available. # define ACE_TRY_NEW_ENV \ do { \ CORBA::Environment ACE_TRY_ENV;\ int ACE_TRY_FLAG = 1; \ int ACE_EXCEPTION_NOT_CAUGHT = 1; \ ACE_TRY_LABEL: \ if (ACE_TRY_FLAG) \ do { // ACE_TRY_EX works exactly like ACE_TRY macro except the lable used // in the try block is customizable to avoid name clashing. It should // be used when, nested try blocks or, multiple try blocks are required // in the same function. # define ACE_TRY_EX(LABEL) \ do { \ int ACE_TRY_FLAG = 1; \ int ACE_EXCEPTION_NOT_CAUGHT = 1; \ ACE_TRY_LABEL ## LABEL: \ if (ACE_TRY_FLAG) \ do { // Check for exceptions within try blocks. # define ACE_TRY_CHECK \ { \ if (ACE_TRY_ENV.exception () != 0) \ { \ ACE_TRY_FLAG = 0; \ goto ACE_TRY_LABEL; \ } \ } // Checking exception within EX try blocks. # define ACE_TRY_CHECK_EX(LABEL) \ { \ if (ACE_TRY_ENV.exception () != 0) \ { \ ACE_TRY_FLAG = 0; \ goto ACE_TRY_LABEL ## LABEL; \ } \ } // Throwing exception within TRY blocks. # define ACE_TRY_THROW(EXCEPTION) \ { \ ACE_TRY_ENV.exception (new EXCEPTION); \ ACE_TRY_FLAG = 0; \ goto ACE_TRY_LABEL; \ } # define ACE_TRY_THROW_EX(EXCEPTION,LABEL) \ { \ ACE_TRY_ENV.exception (new EXCEPTION); \ ACE_TRY_FLAG = 0; \ goto ACE_TRY_LABEL ## LABEL; \ } // When exceptions occur or try block finishes execution without // exception, control will continue in the catch block. This macro // first check if there's non-caught exception we are waiting for // left. It all the conditions met, we have caught an exception. // It then reset the ACE_EXCEPTION_NOT_CAUGHT to prevent subsequent // catch blocks catch the same exception again and extract out the // underlying exception in the ACE_TRY_ENV. We also make a copy // of ACE_TRY_ENV in ACE_CAUGHT_ENV in case we want to rethrow the // exception. ACE_TRY_ENV is cleared out after the exception is // caught so you should not use ACE_TRY_ENV within the catch block. // (In fact, you should use the exception directly.) # define ACE_CATCH(TYPE,VAR) \ } while (0); \ do \ if (ACE_TRY_ENV.exception () != 0 && ACE_EXCEPTION_NOT_CAUGHT && \ TYPE::_downcast(ACE_TRY_ENV.exception ()) != 0) \ { \ CORBA::Environment ACE_CAUGHT_ENV = ACE_TRY_ENV;\ ACE_EXCEPTION_NOT_CAUGHT = 0; \ TYPE &VAR = *TYPE::_downcast (ACE_CAUGHT_ENV.exception ()); \ ACE_UNUSED_ARG (VAR); \ ACE_TRY_ENV.clear (); // ACE_CATCHANY uses ACE_CATCH to catch all CORBA exceptions. # define ACE_CATCHANY ACE_CATCH (CORBA::Exception, ACE_ANY_EXCEPTION) // Since there's no other exception for compilers without exception // support, we simply catch all CORBA exceptions for ACE_CATCHALL. # define ACE_CATCHALL ACE_CATCHANY # if defined (ACE_HAS_DEPRECATED_ACE_RETHROW) # define ACE_RETHROW \ do \ ACE_TRY_ENV = ACE_CAUGHT_ENV; \ while (0) # endif /* ACE_HAS_DEPRECATED_ACE_RETHROW */ // Rethrowing exception within catch blocks. Notice that we depends // on the ACE_CHECK/ACE_CHECK_RETURN following the ACE_ENDTRY, or // ACE_TRY_CHECK/ ACE_TRY_CHECK_EX following the ACE_ENDTRY if the // catch block is within another try block, to do the "Right // Thing[TM]." # define ACE_RE_THROW \ do {\ ACE_TRY_ENV = ACE_CAUGHT_ENV; \ goto ACE_TRY_LABEL; \ } while (0) # define ACE_RE_THROW_EX(LABEL) \ do {\ ACE_TRY_ENV = ACE_CAUGHT_ENV; \ goto ACE_TRY_LABEL ## LABEL; \ } while (0) // Close the try block. Notice that since exception may not get // caught, and exceptions can also be rethrown from the catch block, // it's always a good idea to follow ACE_ENDTRY with ACE_CHECK or // ACE_TRY_CHECK (depending on the context.) # define ACE_ENDTRY \ } while (0); \ } while (0) #endif /* ! ACE_CORBA_HAS_EXCEPTIONS */ // ACE_HAS_EXCEPTIONS is not the same as ACE_NEW_THROWS_EXCEPTIONS. #if defined(ACE_NEW_THROWS_EXCEPTIONS) # define ACE_NEW_THROW_EX(POINTER,CONSTRUCTOR,EXCEPTION) \ do { try { POINTER = new CONSTRUCTOR; } \ catch (ACE_bad_alloc) { errno = ENOMEM; ACE_THROW_INT (EXCEPTION); } \ } while (0) // The following ACE_NEW_THROW* macros are to be deprecated soon. // -------------------- Start Deprecated -------------------- # define ACE_NEW_THROW(POINTER,CONSTRUCTOR,EXCEPTION) \ do { try { POINTER = new CONSTRUCTOR; } \ catch (ACE_bad_alloc) { errno = ENOMEM; TAO_THROW (EXCEPTION); } \ } while (0) # define ACE_NEW_THROW_RETURN(POINTER,CONSTRUCTOR,EXCEPTION,RET_VAL) \ do { try { POINTER = new CONSTRUCTOR; } \ catch (ACE_bad_alloc) { errno = ENOMEM; TAO_THROW_RETURN (EXCEPTION,RET_VAL); } \ } while (0) # define ACE_NEW_TRY_THROW(POINTER,CONSTRUCTOR,EXCEPTION) \ do { try { POINTER = new CONSTRUCTOR; } \ catch (ACE_bad_alloc) { errno = ENOMEM; TAO_TRY_THROW (EXCEPTION); } \ } while (0) // -------------------- End Deprecated -------------------- #else # define ACE_NEW_THROW_EX(POINTER,CONSTRUCTOR,EXCEPTION) \ do { POINTER = new CONSTRUCTOR; \ if (POINTER == 0) { errno = ENOMEM; ACE_THROW_INT (EXCEPTION); } \ } while (0) // The following ACE_NEW_THROW* macros are to be deprecated soon. // -------------------- Start Deprecated -------------------- # define ACE_NEW_THROW(POINTER,CONSTRUCTOR,EXCEPTION) \ do { POINTER = new CONSTRUCTOR; \ if (POINTER == 0) { errno = ENOMEM; TAO_THROW (EXCEPTION); } \ } while (0) # define ACE_NEW_THROW_RETURN(POINTER,CONSTRUCTOR,EXCEPTION,RET_VAL) \ do { POINTER = new CONSTRUCTOR; \ if (POINTER == 0)\ { errno = ENOMEM; TAO_THROW_RETURN (EXCEPTION,RET_VAL); } \ } while (0) # define ACE_NEW_TRY_THROW(POINTER,CONSTRUCTOR,EXCEPTION) \ do { POINTER = new CONSTRUCTOR; \ if (POINTER == 0) { errno = ENOMEM; TAO_TRY_THROW (EXCEPTION); } \ } while (0) // -------------------- End Deprecated -------------------- #endif /* ACE_NEW_THROWS_EXCEPTIONS */ # define ACE_GUARD_THROW_EX(MUTEX,OBJ,LOCK,EXCEPTION) \ ACE_Guard< MUTEX > OBJ (LOCK); \ if (OBJ.locked () == 0) ACE_THROW_INT (EXCEPTION); # define ACE_READ_GUARD_THROW_EX(MUTEX,OBJ,LOCK,EXCEPTION) \ ACE_Read_Guard< MUTEX > OBJ (LOCK); \ if (OBJ.locked () == 0) ACE_THROW_INT (EXCEPTION); # define ACE_WRITE_GUARD_THROW_EX(MUTEX,OBJ,LOCK,EXCEPTION) \ ACE_Write_Guard< MUTEX > OBJ (LOCK); \ if (OBJ.locked () == 0) ACE_THROW_INT (EXCEPTION); // The following ACE_GUARD_THROW* macros are to be deprecated soon. // -------------------- Start Deprecated -------------------- # define ACE_GUARD_THROW(MUTEX,OBJ,LOCK,EXCEPTION) \ ACE_Guard< MUTEX > OBJ (LOCK); \ if (OBJ.locked () == 0) TAO_THROW (EXCEPTION); # define ACE_GUARD_THROW_RETURN(MUTEX,OBJ,LOCK,EXCEPTION,RETURN) \ ACE_Guard< MUTEX > OBJ (LOCK); \ if (OBJ.locked () == 0) TAO_THROW_RETURN (EXCEPTION, RETURN); // -------------------- End Deprecation -------------------- // ============================================================ // Print out a TAO exception. This is not CORBA compliant. # define ACE_PRINT_TAO_EXCEPTION(EX,INFO) \ EX._tao_print_exception (INFO) // Print out a CORBA exception. There is not portable way to // dump a CORBA exception. If you are using other ORB implementation, // redefine the macro to get what you want. # if !defined ACE_PRINT_EXCEPTION # define ACE_PRINT_EXCEPTION(EX,INFO) ACE_PRINT_TAO_EXCEPTION(EX,INFO) # endif /* ACE_PRINT_EXCEPTION */ #include "ace/post.h" #endif /* ACE_CORBA_MACROS_H */