You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
388 lines
13 KiB
C++
388 lines
13 KiB
C++
#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 <iterator>
|
|
#include <boost/assert.hpp>
|
|
#include <boost/current_function.hpp>
|
|
#include <boost/core/no_exceptions_support.hpp>
|
|
#include <iostream>
|
|
#include <cstring>
|
|
|
|
// 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 <class T> 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<class T> inline const void* test_output_impl(T volatile* v) { return const_cast<T*>(v); }
|
|
|
|
template<class T, class U> 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<class T, class U> 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();
|
|
}
|
|
}
|
|
|
|
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<class FormattedOutputFunction, class InputIterator1, class InputIterator2>
|
|
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<InputIterator1>::difference_type first_index = 0;
|
|
typename std::iterator_traits<InputIterator2>::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<class FormattedOutputFunction, class InputIterator1, class InputIterator2, typename BinaryPredicate>
|
|
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<InputIterator1>::difference_type first_index = 0;
|
|
typename std::iterator_traits<InputIterator2>::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_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
|