#ifndef BOOST_CORE_LIGHTWEIGHT_TEST_HPP #define BOOST_CORE_LIGHTWEIGHT_TEST_HPP // MS compatible compilers support #pragma once #if defined(_MSC_VER) # pragma once #endif // // boost/core/lightweight_test.hpp - lightweight test library // // Copyright (c) 2002, 2009, 2014 Peter Dimov // Copyright (2) Beman Dawes 2010, 2011 // Copyright (3) Ion Gaztanaga 2013 // // 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 // #include #include #include #include #include #include #include // IDE's like Visual Studio perform better if output goes to std::cout or // some other stream, so allow user to configure output stream: #ifndef BOOST_LIGHTWEIGHT_TEST_OSTREAM # define BOOST_LIGHTWEIGHT_TEST_OSTREAM std::cerr #endif namespace boost { namespace detail { struct report_errors_reminder { bool called_report_errors_function; report_errors_reminder() : called_report_errors_function(false) {} ~report_errors_reminder() { BOOST_ASSERT(called_report_errors_function); // verify report_errors() was called } }; inline report_errors_reminder& report_errors_remind() { static report_errors_reminder r; return r; } inline int & test_errors() { static int x = 0; report_errors_remind(); return x; } inline void test_failed_impl(char const * expr, char const * file, int line, char const * function) { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): test '" << expr << "' failed in function '" << function << "'" << std::endl; ++test_errors(); } inline void error_impl(char const * msg, char const * file, int line, char const * function) { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): " << msg << " in function '" << function << "'" << std::endl; ++test_errors(); } inline void throw_failed_impl(char const * excep, char const * file, int line, char const * function) { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): Exception '" << excep << "' not thrown in function '" << function << "'" << std::endl; ++test_errors(); } // In the comparisons below, it is possible that T and U are signed and unsigned integer types, which generates warnings in some compilers. // A cleaner fix would require common_type trait or some meta-programming, which would introduce a dependency on Boost.TypeTraits. To avoid // the dependency we just disable the warnings. #if defined(_MSC_VER) # pragma warning(push) # pragma warning(disable: 4389) #elif defined(__clang__) && defined(__has_warning) # if __has_warning("-Wsign-compare") # pragma clang diagnostic push //# pragma clang diagnostic ignored "-Wsign-compare" # endif #elif defined(__GNUC__) && !(defined(__INTEL_COMPILER) || defined(__ICL) || defined(__ICC) || defined(__ECC)) && (__GNUC__ * 100 + __GNUC_MINOR__) >= 406 # pragma GCC diagnostic push //# pragma GCC diagnostic ignored "-Wsign-compare" #endif // specialize test output for char pointers to avoid printing as cstring template inline const T& test_output_impl(const T& v) { return v; } inline const void* test_output_impl(const char* v) { return v; } inline const void* test_output_impl(const unsigned char* v) { return v; } inline const void* test_output_impl(const signed char* v) { return v; } inline const void* test_output_impl(char* v) { return v; } inline const void* test_output_impl(unsigned char* v) { return v; } inline const void* test_output_impl(signed char* v) { return v; } template inline const void* test_output_impl(T volatile* v) { return const_cast(v); } #if !defined( BOOST_NO_CXX11_NULLPTR ) inline const void* test_output_impl(std::nullptr_t) { return nullptr; } #endif template inline void test_eq_impl( char const * expr1, char const * expr2, char const * file, int line, char const * function, T const & t, U const & u ) { if( t == u ) { report_errors_remind(); } else { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): test '" << expr1 << " == " << expr2 << "' failed in function '" << function << "': " << "'" << test_output_impl(t) << "' != '" << test_output_impl(u) << "'" << std::endl; ++test_errors(); } } template inline void test_ne_impl( char const * expr1, char const * expr2, char const * file, int line, char const * function, T const & t, U const & u ) { if( t != u ) { report_errors_remind(); } else { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): test '" << expr1 << " != " << expr2 << "' failed in function '" << function << "': " << "'" << test_output_impl(t) << "' == '" << test_output_impl(u) << "'" << std::endl; ++test_errors(); } } template inline void test_lt_impl( char const * expr1, char const * expr2, char const * file, int line, char const * function, T const & t, U const & u ) { if( t < u ) { report_errors_remind(); } else { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): test '" << expr1 << " < " << expr2 << "' failed in function '" << function << "': " << "'" << test_output_impl(t) << "' >= '" << test_output_impl(u) << "'" << std::endl; ++test_errors(); } } template inline void test_le_impl( char const * expr1, char const * expr2, char const * file, int line, char const * function, T const & t, U const & u ) { if( t <= u ) { report_errors_remind(); } else { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): test '" << expr1 << " <= " << expr2 << "' failed in function '" << function << "': " << "'" << test_output_impl(t) << "' > '" << test_output_impl(u) << "'" << std::endl; ++test_errors(); } } template inline void test_gt_impl( char const * expr1, char const * expr2, char const * file, int line, char const * function, T const & t, U const & u ) { if( t > u ) { report_errors_remind(); } else { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): test '" << expr1 << " > " << expr2 << "' failed in function '" << function << "': " << "'" << test_output_impl(t) << "' <= '" << test_output_impl(u) << "'" << std::endl; ++test_errors(); } } template inline void test_ge_impl( char const * expr1, char const * expr2, char const * file, int line, char const * function, T const & t, U const & u ) { if( t >= u ) { report_errors_remind(); } else { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): test '" << expr1 << " >= " << expr2 << "' failed in function '" << function << "': " << "'" << test_output_impl(t) << "' < '" << test_output_impl(u) << "'" << std::endl; ++test_errors(); } } inline void test_cstr_eq_impl( char const * expr1, char const * expr2, char const * file, int line, char const * function, char const * const t, char const * const u ) { if( std::strcmp(t, u) == 0 ) { report_errors_remind(); } else { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): test '" << expr1 << " == " << expr2 << "' failed in function '" << function << "': " << "'" << t << "' != '" << u << "'" << std::endl; ++test_errors(); } } inline void test_cstr_ne_impl( char const * expr1, char const * expr2, char const * file, int line, char const * function, char const * const t, char const * const u ) { if( std::strcmp(t, u) != 0 ) { report_errors_remind(); } else { BOOST_LIGHTWEIGHT_TEST_OSTREAM << file << "(" << line << "): test '" << expr1 << " == " << expr2 << "' failed in function '" << function << "': " << "'" << t << "' == '" << u << "'" << std::endl; ++test_errors(); } } template void test_all_eq_impl(FormattedOutputFunction& output, char const * file, int line, char const * function, InputIterator1 first_begin, InputIterator1 first_end, InputIterator2 second_begin, InputIterator2 second_end) { InputIterator1 first_it = first_begin; InputIterator2 second_it = second_begin; typename std::iterator_traits::difference_type first_index = 0; typename std::iterator_traits::difference_type second_index = 0; std::size_t error_count = 0; const std::size_t max_count = 8; do { while ((first_it != first_end) && (second_it != second_end) && (*first_it == *second_it)) { ++first_it; ++second_it; ++first_index; ++second_index; } if ((first_it == first_end) || (second_it == second_end)) { break; // do-while } if (error_count == 0) { output << file << "(" << line << "): Container contents differ in function '" << function << "':"; } else if (error_count >= max_count) { output << " ..."; break; } output << " [" << first_index << "] '" << test_output_impl(*first_it) << "' != '" << test_output_impl(*second_it) << "'"; ++first_it; ++second_it; ++first_index; ++second_index; ++error_count; } while (first_it != first_end); first_index += std::distance(first_it, first_end); second_index += std::distance(second_it, second_end); if (first_index != second_index) { if (error_count == 0) { output << file << "(" << line << "): Container sizes differ in function '" << function << "': size(" << first_index << ") != size(" << second_index << ")"; } else { output << " [*] size(" << first_index << ") != size(" << second_index << ")"; } ++error_count; } if (error_count == 0) { boost::detail::report_errors_remind(); } else { output << std::endl; ++boost::detail::test_errors(); } } template void test_all_with_impl(FormattedOutputFunction& output, char const * file, int line, char const * function, InputIterator1 first_begin, InputIterator1 first_end, InputIterator2 second_begin, InputIterator2 second_end, BinaryPredicate predicate) { InputIterator1 first_it = first_begin; InputIterator2 second_it = second_begin; typename std::iterator_traits::difference_type first_index = 0; typename std::iterator_traits::difference_type second_index = 0; std::size_t error_count = 0; const std::size_t max_count = 8; do { while ((first_it != first_end) && (second_it != second_end) && predicate(*first_it, *second_it)) { ++first_it; ++second_it; ++first_index; ++second_index; } if ((first_it == first_end) || (second_it == second_end)) { break; // do-while } if (error_count == 0) { output << file << "(" << line << "): Container contents differ in function '" << function << "':"; } else if (error_count >= max_count) { output << " ..."; break; } output << " [" << first_index << "]"; ++first_it; ++second_it; ++first_index; ++second_index; ++error_count; } while (first_it != first_end); first_index += std::distance(first_it, first_end); second_index += std::distance(second_it, second_end); if (first_index != second_index) { if (error_count == 0) { output << file << "(" << line << "): Container sizes differ in function '" << function << "': size(" << first_index << ") != size(" << second_index << ")"; } else { output << " [*] size(" << first_index << ") != size(" << second_index << ")"; } ++error_count; } if (error_count == 0) { report_errors_remind(); } else { output << std::endl; ++test_errors(); } } #if defined(_MSC_VER) # pragma warning(pop) #elif defined(__clang__) && defined(__has_warning) # if __has_warning("-Wsign-compare") # pragma clang diagnostic pop # endif #elif defined(__GNUC__) && !(defined(__INTEL_COMPILER) || defined(__ICL) || defined(__ICC) || defined(__ECC)) && (__GNUC__ * 100 + __GNUC_MINOR__) >= 406 # pragma GCC diagnostic pop #endif } // namespace detail inline int report_errors() { boost::detail::report_errors_remind().called_report_errors_function = true; int errors = boost::detail::test_errors(); if( errors == 0 ) { BOOST_LIGHTWEIGHT_TEST_OSTREAM << "No errors detected." << std::endl; return 0; } else { BOOST_LIGHTWEIGHT_TEST_OSTREAM << errors << " error" << (errors == 1? "": "s") << " detected." << std::endl; return 1; } } } // namespace boost #define BOOST_TEST(expr) ((expr)? (void)0: ::boost::detail::test_failed_impl(#expr, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION)) #define BOOST_TEST_NOT(expr) BOOST_TEST(!(expr)) #define BOOST_ERROR(msg) ( ::boost::detail::error_impl(msg, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION) ) #define BOOST_TEST_EQ(expr1,expr2) ( ::boost::detail::test_eq_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2) ) #define BOOST_TEST_NE(expr1,expr2) ( ::boost::detail::test_ne_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2) ) #define BOOST_TEST_LT(expr1,expr2) ( ::boost::detail::test_lt_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2) ) #define BOOST_TEST_LE(expr1,expr2) ( ::boost::detail::test_le_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2) ) #define BOOST_TEST_GT(expr1,expr2) ( ::boost::detail::test_gt_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2) ) #define BOOST_TEST_GE(expr1,expr2) ( ::boost::detail::test_ge_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2) ) #define BOOST_TEST_CSTR_EQ(expr1,expr2) ( ::boost::detail::test_cstr_eq_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2) ) #define BOOST_TEST_CSTR_NE(expr1,expr2) ( ::boost::detail::test_cstr_ne_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2) ) #define BOOST_TEST_ALL_EQ(begin1, end1, begin2, end2) ( ::boost::detail::test_all_eq_impl(BOOST_LIGHTWEIGHT_TEST_OSTREAM, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, begin1, end1, begin2, end2) ) #define BOOST_TEST_ALL_WITH(begin1, end1, begin2, end2, predicate) ( ::boost::detail::test_all_with_impl(BOOST_LIGHTWEIGHT_TEST_OSTREAM, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, begin1, end1, begin2, end2, predicate) ) #ifndef BOOST_NO_EXCEPTIONS #define BOOST_TEST_THROWS( EXPR, EXCEP ) \ try { \ EXPR; \ ::boost::detail::throw_failed_impl \ (#EXCEP, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION); \ } \ catch(EXCEP const&) { \ } \ catch(...) { \ ::boost::detail::throw_failed_impl \ (#EXCEP, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION); \ } \ // #else #define BOOST_TEST_THROWS( EXPR, EXCEP ) #endif #endif // #ifndef BOOST_CORE_LIGHTWEIGHT_TEST_HPP