422 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
	
		
		
			
		
	
	
			422 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
	
|  | /*=============================================================================
 | ||
|  |     Boost.Wave: A Standard compliant C++ preprocessor library | ||
|  | 
 | ||
|  |     http://www.boost.org/
 | ||
|  | 
 | ||
|  |     Copyright (c) 2001-2012 Hartmut Kaiser. Distributed under the Boost | ||
|  |     Software License, Version 1.0. (See accompanying file | ||
|  |     LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | ||
|  | =============================================================================*/ | ||
|  | 
 | ||
|  | #if !defined(CPP_EXCEPTIONS_HPP_5190E447_A781_4521_A275_5134FF9917D7_INCLUDED)
 | ||
|  | #define CPP_EXCEPTIONS_HPP_5190E447_A781_4521_A275_5134FF9917D7_INCLUDED
 | ||
|  | 
 | ||
|  | #include <exception>
 | ||
|  | #include <string>
 | ||
|  | #include <limits>
 | ||
|  | 
 | ||
|  | #include <boost/assert.hpp>
 | ||
|  | #include <boost/config.hpp>
 | ||
|  | #include <boost/throw_exception.hpp>
 | ||
|  | #include <boost/wave/wave_config.hpp>
 | ||
|  | #include <boost/wave/cpp_throw.hpp>
 | ||
|  | 
 | ||
|  | // this must occur after all of the includes and before any code appears
 | ||
|  | #ifdef BOOST_HAS_ABI_HEADERS
 | ||
|  | #include BOOST_ABI_PREFIX
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | namespace boost { | ||
|  | namespace wave { | ||
|  | 
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | // exception severity
 | ||
|  | namespace util { | ||
|  | 
 | ||
|  |     enum severity { | ||
|  |         severity_remark = 0, | ||
|  |         severity_warning, | ||
|  |         severity_error, | ||
|  |         severity_fatal, | ||
|  |         severity_commandline_error, | ||
|  |         last_severity_code = severity_commandline_error | ||
|  |     }; | ||
|  | 
 | ||
|  |     inline char const * | ||
|  |     get_severity(int level) | ||
|  |     { | ||
|  |         static char const *severity_text[] = | ||
|  |         { | ||
|  |             "remark",               // severity_remark
 | ||
|  |             "warning",              // severity_warning
 | ||
|  |             "error",                // severity_error
 | ||
|  |             "fatal error",          // severity_fatal
 | ||
|  |             "command line error"    // severity_commandline_error
 | ||
|  |         }; | ||
|  |         BOOST_ASSERT(severity_remark <= level && | ||
|  |             level <= last_severity_code); | ||
|  |         return severity_text[level]; | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | //  cpp_exception, the base class for all specific C preprocessor exceptions
 | ||
|  | class BOOST_SYMBOL_VISIBLE cpp_exception | ||
|  | :   public std::exception | ||
|  | { | ||
|  | public: | ||
|  |     cpp_exception(std::size_t line_, std::size_t column_, char const *filename_) throw() | ||
|  |     :   line(line_), column(column_) | ||
|  |     { | ||
|  |         unsigned int off = 0; | ||
|  |         while (off < sizeof(filename)-1 && *filename_) | ||
|  |             filename[off++] = *filename_++; | ||
|  |         filename[off] = 0; | ||
|  |     } | ||
|  |     ~cpp_exception() throw() {} | ||
|  | 
 | ||
|  |     virtual char const *what() const throw() = 0;           // to be overloaded
 | ||
|  |     virtual char const *description() const throw() = 0; | ||
|  |     virtual int get_errorcode() const throw() = 0; | ||
|  |     virtual int get_severity() const throw() = 0; | ||
|  |     virtual char const* get_related_name() const throw() = 0; | ||
|  |     virtual bool is_recoverable() const throw() = 0; | ||
|  | 
 | ||
|  |     std::size_t line_no() const throw() { return line; } | ||
|  |     std::size_t column_no() const throw() { return column; } | ||
|  |     char const *file_name() const throw() { return filename; } | ||
|  | 
 | ||
|  | protected: | ||
|  |     char filename[512]; | ||
|  |     std::size_t line; | ||
|  |     std::size_t column; | ||
|  | }; | ||
|  | 
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | // preprocessor error
 | ||
|  | class BOOST_SYMBOL_VISIBLE preprocess_exception : | ||
|  |     public cpp_exception | ||
|  | { | ||
|  | public: | ||
|  |     enum error_code { | ||
|  |         no_error = 0, | ||
|  |         unexpected_error, | ||
|  |         macro_redefinition, | ||
|  |         macro_insertion_error, | ||
|  |         bad_include_file, | ||
|  |         bad_include_statement, | ||
|  |         ill_formed_directive, | ||
|  |         error_directive, | ||
|  |         warning_directive, | ||
|  |         ill_formed_expression, | ||
|  |         missing_matching_if, | ||
|  |         missing_matching_endif, | ||
|  |         ill_formed_operator, | ||
|  |         bad_define_statement, | ||
|  |         bad_define_statement_va_args, | ||
|  |         too_few_macroarguments, | ||
|  |         too_many_macroarguments, | ||
|  |         empty_macroarguments, | ||
|  |         improperly_terminated_macro, | ||
|  |         bad_line_statement, | ||
|  |         bad_line_number, | ||
|  |         bad_line_filename, | ||
|  |         bad_undefine_statement, | ||
|  |         bad_macro_definition, | ||
|  |         illegal_redefinition, | ||
|  |         duplicate_parameter_name, | ||
|  |         invalid_concat, | ||
|  |         last_line_not_terminated, | ||
|  |         ill_formed_pragma_option, | ||
|  |         include_nesting_too_deep, | ||
|  |         misplaced_operator, | ||
|  |         alreadydefined_name, | ||
|  |         undefined_macroname, | ||
|  |         invalid_macroname, | ||
|  |         unexpected_qualified_name, | ||
|  |         division_by_zero, | ||
|  |         integer_overflow, | ||
|  |         illegal_operator_redefinition, | ||
|  |         ill_formed_integer_literal, | ||
|  |         ill_formed_character_literal, | ||
|  |         unbalanced_if_endif, | ||
|  |         character_literal_out_of_range, | ||
|  |         could_not_open_output_file, | ||
|  |         incompatible_config, | ||
|  |         ill_formed_pragma_message, | ||
|  |         pragma_message_directive, | ||
|  |         last_error_number = pragma_message_directive | ||
|  |     }; | ||
|  | 
 | ||
|  |     preprocess_exception(char const *what_, error_code code, std::size_t line_, | ||
|  |         std::size_t column_, char const *filename_) throw() | ||
|  |     :   cpp_exception(line_, column_, filename_), | ||
|  |         code(code) | ||
|  |     { | ||
|  |         unsigned int off = 0; | ||
|  |         while (off < sizeof(buffer) - 1 && *what_) | ||
|  |             buffer[off++] = *what_++; | ||
|  |         buffer[off] = 0; | ||
|  |     } | ||
|  |     ~preprocess_exception() throw() {} | ||
|  | 
 | ||
|  |     virtual char const *what() const throw() | ||
|  |     { | ||
|  |         return "boost::wave::preprocess_exception"; | ||
|  |     } | ||
|  |     virtual char const *description() const throw() | ||
|  |     { | ||
|  |         return buffer; | ||
|  |     } | ||
|  |     virtual int get_severity() const throw() | ||
|  |     { | ||
|  |         return severity_level(code); | ||
|  |     } | ||
|  |     virtual int get_errorcode() const throw() | ||
|  |     { | ||
|  |         return code; | ||
|  |     } | ||
|  |     virtual char const* get_related_name() const throw() | ||
|  |     { | ||
|  |         return "<unknown>"; | ||
|  |     } | ||
|  |     virtual bool is_recoverable() const throw() | ||
|  |     { | ||
|  |         switch (get_errorcode()) { | ||
|  |         // these are the exceptions thrown during processing not supposed to
 | ||
|  |         // produce any tokens on the context::iterator level
 | ||
|  |         case preprocess_exception::no_error:        // just a placeholder
 | ||
|  |         case preprocess_exception::macro_redefinition: | ||
|  |         case preprocess_exception::macro_insertion_error: | ||
|  |         case preprocess_exception::bad_macro_definition: | ||
|  |         case preprocess_exception::illegal_redefinition: | ||
|  |         case preprocess_exception::duplicate_parameter_name: | ||
|  |         case preprocess_exception::invalid_macroname: | ||
|  |         case preprocess_exception::bad_include_file: | ||
|  |         case preprocess_exception::bad_include_statement: | ||
|  |         case preprocess_exception::ill_formed_directive: | ||
|  |         case preprocess_exception::error_directive: | ||
|  |         case preprocess_exception::warning_directive: | ||
|  |         case preprocess_exception::ill_formed_expression: | ||
|  |         case preprocess_exception::missing_matching_if: | ||
|  |         case preprocess_exception::missing_matching_endif: | ||
|  |         case preprocess_exception::unbalanced_if_endif: | ||
|  |         case preprocess_exception::bad_define_statement: | ||
|  |         case preprocess_exception::bad_define_statement_va_args: | ||
|  |         case preprocess_exception::bad_line_statement: | ||
|  |         case preprocess_exception::bad_line_number: | ||
|  |         case preprocess_exception::bad_line_filename: | ||
|  |         case preprocess_exception::bad_undefine_statement: | ||
|  |         case preprocess_exception::division_by_zero: | ||
|  |         case preprocess_exception::integer_overflow: | ||
|  |         case preprocess_exception::ill_formed_integer_literal: | ||
|  |         case preprocess_exception::ill_formed_character_literal: | ||
|  |         case preprocess_exception::character_literal_out_of_range: | ||
|  |         case preprocess_exception::last_line_not_terminated: | ||
|  |         case preprocess_exception::include_nesting_too_deep: | ||
|  |         case preprocess_exception::illegal_operator_redefinition: | ||
|  |         case preprocess_exception::incompatible_config: | ||
|  |         case preprocess_exception::ill_formed_pragma_option: | ||
|  |         case preprocess_exception::ill_formed_pragma_message: | ||
|  |         case preprocess_exception::pragma_message_directive: | ||
|  |             return true; | ||
|  | 
 | ||
|  |         case preprocess_exception::unexpected_error: | ||
|  |         case preprocess_exception::ill_formed_operator: | ||
|  |         case preprocess_exception::too_few_macroarguments: | ||
|  |         case preprocess_exception::too_many_macroarguments: | ||
|  |         case preprocess_exception::empty_macroarguments: | ||
|  |         case preprocess_exception::improperly_terminated_macro: | ||
|  |         case preprocess_exception::invalid_concat: | ||
|  |         case preprocess_exception::could_not_open_output_file: | ||
|  |             break; | ||
|  |         } | ||
|  |         return false; | ||
|  |     } | ||
|  | 
 | ||
|  |     static char const *error_text(int code) | ||
|  |     { | ||
|  |     // error texts in this array must appear in the same order as the items in
 | ||
|  |     // the error enum above
 | ||
|  |         static char const *preprocess_exception_errors[] = { | ||
|  |             "no error",                                 // no_error
 | ||
|  |             "unexpected error (should not happen)",     // unexpected_error
 | ||
|  |             "illegal macro redefinition",               // macro_redefinition
 | ||
|  |             "macro definition failed (out of memory?)", // macro_insertion_error
 | ||
|  |             "could not find include file",              // bad_include_file
 | ||
|  |             "ill formed #include directive",            // bad_include_statement
 | ||
|  |             "ill formed preprocessor directive",        // ill_formed_directive
 | ||
|  |             "encountered #error directive or #pragma wave stop()", // error_directive
 | ||
|  |             "encountered #warning directive",           // warning_directive
 | ||
|  |             "ill formed preprocessor expression",       // ill_formed_expression
 | ||
|  |             "the #if for this directive is missing",    // missing_matching_if
 | ||
|  |             "detected at least one missing #endif directive",   // missing_matching_endif
 | ||
|  |             "ill formed preprocessing operator",        // ill_formed_operator
 | ||
|  |             "ill formed #define directive",             // bad_define_statement
 | ||
|  |             "__VA_ARGS__ can only appear in the " | ||
|  |             "expansion of a C99 variadic macro",        // bad_define_statement_va_args
 | ||
|  |             "too few macro arguments",                  // too_few_macroarguments
 | ||
|  |             "too many macro arguments",                 // too_many_macroarguments
 | ||
|  |             "empty macro arguments are not supported in pure C++ mode, " | ||
|  |             "use variadics mode to allow these",        // empty_macroarguments
 | ||
|  |             "improperly terminated macro invocation " | ||
|  |             "or replacement-list terminates in partial " | ||
|  |             "macro expansion (not supported yet)",      // improperly_terminated_macro
 | ||
|  |             "ill formed #line directive",               // bad_line_statement
 | ||
|  |             "line number argument of #line directive " | ||
|  |             "should consist out of decimal digits " | ||
|  |             "only and must be in range of [1..INT_MAX]", // bad_line_number
 | ||
|  |             "filename argument of #line directive should " | ||
|  |             "be a narrow string literal",               // bad_line_filename
 | ||
|  |             "#undef may not be used on this predefined name",   // bad_undefine_statement
 | ||
|  |             "invalid macro definition",                 // bad_macro_definition
 | ||
|  |             "this predefined name may not be redefined",        // illegal_redefinition
 | ||
|  |             "duplicate macro parameter name",           // duplicate_parameter_name
 | ||
|  |             "pasting the following two tokens does not " | ||
|  |             "give a valid preprocessing token",         // invalid_concat
 | ||
|  |             "last line of file ends without a newline", // last_line_not_terminated
 | ||
|  |             "unknown or illformed pragma option",       // ill_formed_pragma_option
 | ||
|  |             "include files nested too deep",            // include_nesting_too_deep
 | ||
|  |             "misplaced operator defined()",             // misplaced_operator
 | ||
|  |             "the name is already used in this scope as " | ||
|  |             "a macro or scope name",                    // alreadydefined_name
 | ||
|  |             "undefined macro or scope name may not be imported", // undefined_macroname
 | ||
|  |             "ill formed macro name",                    // invalid_macroname
 | ||
|  |             "qualified names are supported in C++11 mode only",  // unexpected_qualified_name
 | ||
|  |             "division by zero in preprocessor expression",       // division_by_zero
 | ||
|  |             "integer overflow in preprocessor expression",       // integer_overflow
 | ||
|  |             "this cannot be used as a macro name as it is " | ||
|  |             "an operator in C++",                       // illegal_operator_redefinition
 | ||
|  |             "ill formed integer literal or integer constant too large",   // ill_formed_integer_literal
 | ||
|  |             "ill formed character literal",             // ill_formed_character_literal
 | ||
|  |             "unbalanced #if/#endif in include file",    // unbalanced_if_endif
 | ||
|  |             "expression contains out of range character literal", // character_literal_out_of_range
 | ||
|  |             "could not open output file",               // could_not_open_output_file
 | ||
|  |             "incompatible state information",           // incompatible_config
 | ||
|  |             "illformed pragma message",                 // ill_formed_pragma_message
 | ||
|  |             "encountered #pragma message directive"     // pragma_message_directive
 | ||
|  |         }; | ||
|  |         BOOST_ASSERT(no_error <= code && code <= last_error_number); | ||
|  |         return preprocess_exception_errors[code]; | ||
|  |     } | ||
|  | 
 | ||
|  |     static util::severity severity_level(int code) | ||
|  |     { | ||
|  |         static util::severity preprocess_exception_severity[] = { | ||
|  |             util::severity_remark,             // no_error
 | ||
|  |             util::severity_fatal,              // unexpected_error
 | ||
|  |             util::severity_warning,            // macro_redefinition
 | ||
|  |             util::severity_fatal,              // macro_insertion_error
 | ||
|  |             util::severity_error,              // bad_include_file
 | ||
|  |             util::severity_error,              // bad_include_statement
 | ||
|  |             util::severity_error,              // ill_formed_directive
 | ||
|  |             util::severity_fatal,              // error_directive
 | ||
|  |             util::severity_warning,            // warning_directive
 | ||
|  |             util::severity_error,              // ill_formed_expression
 | ||
|  |             util::severity_error,              // missing_matching_if
 | ||
|  |             util::severity_error,              // missing_matching_endif
 | ||
|  |             util::severity_error,              // ill_formed_operator
 | ||
|  |             util::severity_error,              // bad_define_statement
 | ||
|  |             util::severity_error,              // bad_define_statement_va_args
 | ||
|  |             util::severity_warning,            // too_few_macroarguments
 | ||
|  |             util::severity_warning,            // too_many_macroarguments
 | ||
|  |             util::severity_warning,            // empty_macroarguments
 | ||
|  |             util::severity_error,              // improperly_terminated_macro
 | ||
|  |             util::severity_warning,            // bad_line_statement
 | ||
|  |             util::severity_warning,            // bad_line_number
 | ||
|  |             util::severity_warning,            // bad_line_filename
 | ||
|  |             util::severity_warning,            // bad_undefine_statement
 | ||
|  |             util::severity_commandline_error,  // bad_macro_definition
 | ||
|  |             util::severity_warning,            // illegal_redefinition
 | ||
|  |             util::severity_error,              // duplicate_parameter_name
 | ||
|  |             util::severity_error,              // invalid_concat
 | ||
|  |             util::severity_warning,            // last_line_not_terminated
 | ||
|  |             util::severity_warning,            // ill_formed_pragma_option
 | ||
|  |             util::severity_fatal,              // include_nesting_too_deep
 | ||
|  |             util::severity_error,              // misplaced_operator
 | ||
|  |             util::severity_error,              // alreadydefined_name
 | ||
|  |             util::severity_error,              // undefined_macroname
 | ||
|  |             util::severity_error,              // invalid_macroname
 | ||
|  |             util::severity_error,              // unexpected_qualified_name
 | ||
|  |             util::severity_fatal,              // division_by_zero
 | ||
|  |             util::severity_error,              // integer_overflow
 | ||
|  |             util::severity_error,              // illegal_operator_redefinition
 | ||
|  |             util::severity_error,              // ill_formed_integer_literal
 | ||
|  |             util::severity_error,              // ill_formed_character_literal
 | ||
|  |             util::severity_warning,            // unbalanced_if_endif
 | ||
|  |             util::severity_warning,            // character_literal_out_of_range
 | ||
|  |             util::severity_error,              // could_not_open_output_file
 | ||
|  |             util::severity_remark,             // incompatible_config
 | ||
|  |             util::severity_warning,            // ill_formed_pragma_message
 | ||
|  |             util::severity_remark,             // pragma_message_directive
 | ||
|  |         }; | ||
|  |         BOOST_ASSERT(no_error <= code && code <= last_error_number); | ||
|  |         return preprocess_exception_severity[code]; | ||
|  |     } | ||
|  |     static char const *severity_text(int code) | ||
|  |     { | ||
|  |         return util::get_severity(severity_level(code)); | ||
|  |     } | ||
|  | 
 | ||
|  | private: | ||
|  |     char buffer[512]; | ||
|  |     error_code code; | ||
|  | }; | ||
|  | 
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | //  Error during macro handling, this exception contains the related macro name
 | ||
|  | class BOOST_SYMBOL_VISIBLE macro_handling_exception : | ||
|  |     public preprocess_exception | ||
|  | { | ||
|  | public: | ||
|  |     macro_handling_exception(char const *what_, error_code code, std::size_t line_, | ||
|  |         std::size_t column_, char const *filename_, char const *macroname) throw() | ||
|  |     :   preprocess_exception(what_, code, line_, column_, filename_) | ||
|  |     { | ||
|  |         unsigned int off = 0; | ||
|  |         while (off < sizeof(name) && *macroname) | ||
|  |             name[off++] = *macroname++; | ||
|  |         name[off] = 0; | ||
|  |     } | ||
|  |     ~macro_handling_exception() throw() {} | ||
|  | 
 | ||
|  |     virtual char const *what() const throw() | ||
|  |     { | ||
|  |         return "boost::wave::macro_handling_exception"; | ||
|  |     } | ||
|  |     char const* get_related_name() const throw() | ||
|  |     { | ||
|  |         return name; | ||
|  |     } | ||
|  | 
 | ||
|  | private: | ||
|  |     char name[512]; | ||
|  | }; | ||
|  | 
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | //
 | ||
|  | //  The is_recoverable() function allows to decide, whether it is possible
 | ||
|  | //  simply to continue after a given exception was thrown by Wave.
 | ||
|  | //
 | ||
|  | //  This is kind of a hack to allow to recover from certain errors as long as
 | ||
|  | //  Wave doesn't provide better means of error recovery.
 | ||
|  | //
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | inline bool | ||
|  | is_recoverable(cpp_exception const& e) | ||
|  | { | ||
|  |     return e.is_recoverable(); | ||
|  | } | ||
|  | 
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////
 | ||
|  | }   // namespace wave
 | ||
|  | }   // namespace boost
 | ||
|  | 
 | ||
|  | // the suffix header occurs after all of the code
 | ||
|  | #ifdef BOOST_HAS_ABI_HEADERS
 | ||
|  | #include BOOST_ABI_SUFFIX
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | #endif // !defined(CPP_EXCEPTIONS_HPP_5190E447_A781_4521_A275_5134FF9917D7_INCLUDED)
 |