1 // boost/catch_exceptions.hpp -----------------------------------------------//
3 // Copyright Beman Dawes 1995-2001. Distributed under the Boost
4 // Software License, Version 1.0. (See accompanying file
5 // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
7 // See http://www.boost.org/libs/test for documentation.
10 // 13 Jun 01 report_exception() made inline. (John Maddock, Jesse Jones)
11 // 26 Feb 01 Numerous changes suggested during formal review. (Beman)
12 // 25 Jan 01 catch_exceptions.hpp code factored out of cpp_main.cpp.
13 // 22 Jan 01 Remove test_tools dependencies to reduce coupling.
14 // 5 Nov 00 Initial boost version (Beman Dawes)
16 #ifndef BOOST_CATCH_EXCEPTIONS_HPP
17 #define BOOST_CATCH_EXCEPTIONS_HPP
19 // header dependencies are deliberately restricted to the standard library
20 // to reduce coupling to other boost libraries.
21 #include <string> // for string
22 #include <new> // for bad_alloc
23 #include <typeinfo> // for bad_cast, bad_typeid
24 #include <exception> // for exception, bad_exception
25 #include <stdexcept> // for std exception hierarchy
26 #include <boost/cstdlib.hpp> // for exit codes
27 # if __GNUC__ != 2 || __GNUC_MINOR__ > 96
28 # include <ostream> // for ostream
30 # include <iostream> // workaround GNU missing ostream header
33 # if defined(__BORLANDC__) && (__BORLANDC__ <= 0x0551)
34 # define BOOST_BUILT_IN_EXCEPTIONS_MISSING_WHAT
37 #if defined(MPW_CPLUS) && (MPW_CPLUS <= 0x890)
38 # define BOOST_BUILT_IN_EXCEPTIONS_MISSING_WHAT
39 namespace std
{ class bad_typeid
{ }; }
47 // A separate reporting function was requested during formal review.
48 inline void report_exception( std::ostream
& os
,
49 const char * name
, const char * info
)
50 { os
<< "\n** uncaught exception: " << name
<< " " << info
<< std::endl
; }
53 // catch_exceptions ------------------------------------------------------//
55 template< class Generator
> // Generator is function object returning int
56 int catch_exceptions( Generator function_object
,
57 std::ostream
& out
, std::ostream
& err
)
59 int result
= 0; // quiet compiler warnings
60 bool exception_thrown
= true; // avoid setting result for each excptn type
62 #ifndef BOOST_NO_EXCEPTIONS
66 result
= function_object();
67 exception_thrown
= false;
68 #ifndef BOOST_NO_EXCEPTIONS
71 // As a result of hard experience with strangely interleaved output
72 // under some compilers, there is a lot of use of endl in the code below
73 // where a simple '\n' might appear to do.
75 // The rules for catch & arguments are a bit different from function
76 // arguments (ISO 15.3 paragraphs 18 & 19). Apparently const isn't
77 // required, but it doesn't hurt and some programmers ask for it.
79 catch ( const char * ex
)
80 { detail::report_exception( out
, "", ex
); }
81 catch ( const std::string
& ex
)
82 { detail::report_exception( out
, "", ex
.c_str() ); }
85 catch ( const std::bad_alloc
& ex
)
86 { detail::report_exception( out
, "std::bad_alloc:", ex
.what() ); }
88 # ifndef BOOST_BUILT_IN_EXCEPTIONS_MISSING_WHAT
89 catch ( const std::bad_cast
& ex
)
90 { detail::report_exception( out
, "std::bad_cast:", ex
.what() ); }
91 catch ( const std::bad_typeid
& ex
)
92 { detail::report_exception( out
, "std::bad_typeid:", ex
.what() ); }
94 catch ( const std::bad_cast
& )
95 { detail::report_exception( out
, "std::bad_cast", "" ); }
96 catch ( const std::bad_typeid
& )
97 { detail::report_exception( out
, "std::bad_typeid", "" ); }
100 catch ( const std::bad_exception
& ex
)
101 { detail::report_exception( out
, "std::bad_exception:", ex
.what() ); }
102 catch ( const std::domain_error
& ex
)
103 { detail::report_exception( out
, "std::domain_error:", ex
.what() ); }
104 catch ( const std::invalid_argument
& ex
)
105 { detail::report_exception( out
, "std::invalid_argument:", ex
.what() ); }
106 catch ( const std::length_error
& ex
)
107 { detail::report_exception( out
, "std::length_error:", ex
.what() ); }
108 catch ( const std::out_of_range
& ex
)
109 { detail::report_exception( out
, "std::out_of_range:", ex
.what() ); }
110 catch ( const std::range_error
& ex
)
111 { detail::report_exception( out
, "std::range_error:", ex
.what() ); }
112 catch ( const std::overflow_error
& ex
)
113 { detail::report_exception( out
, "std::overflow_error:", ex
.what() ); }
114 catch ( const std::underflow_error
& ex
)
115 { detail::report_exception( out
, "std::underflow_error:", ex
.what() ); }
116 catch ( const std::logic_error
& ex
)
117 { detail::report_exception( out
, "std::logic_error:", ex
.what() ); }
118 catch ( const std::runtime_error
& ex
)
119 { detail::report_exception( out
, "std::runtime_error:", ex
.what() ); }
120 catch ( const std::exception
& ex
)
121 { detail::report_exception( out
, "std::exception:", ex
.what() ); }
124 { detail::report_exception( out
, "unknown exception", "" ); }
125 #endif // BOOST_NO_EXCEPTIONS
127 if ( exception_thrown
) result
= boost::exit_exception_failure
;
129 if ( result
!= 0 && result
!= exit_success
)
131 out
<< std::endl
<< "**** returning with error code "
132 << result
<< std::endl
;
134 << "********** errors detected; see stdout for details ***********"
137 #if !defined(BOOST_NO_CPP_MAIN_SUCCESS_MESSAGE)
138 else { out
<< std::flush
<< "no errors detected" << std::endl
; }
141 } // catch_exceptions
145 #endif // BOOST_CATCH_EXCEPTIONS_HPP