Chris@102
|
1 // boost/endian/detail/lightweight_test.hpp --------------------------------------------//
|
Chris@102
|
2
|
Chris@102
|
3 #ifndef BOOST_ENDIAN_LIGHTWEIGHT_TEST_HPP
|
Chris@102
|
4 #define BOOST_ENDIAN_LIGHTWEIGHT_TEST_HPP
|
Chris@102
|
5
|
Chris@102
|
6 // MS compatible compilers support #pragma once
|
Chris@102
|
7
|
Chris@102
|
8 #if defined(_MSC_VER)
|
Chris@102
|
9 # pragma once
|
Chris@102
|
10 #endif
|
Chris@102
|
11
|
Chris@102
|
12 //
|
Chris@102
|
13 // Copyright (c) 2002, 2009, 2014 Peter Dimov
|
Chris@102
|
14 // Copyright (2) Beman Dawes 2010, 2011, 2015
|
Chris@102
|
15 // Copyright (3) Ion Gaztanaga 2013
|
Chris@102
|
16 //
|
Chris@102
|
17 // Distributed under the Boost Software License, Version 1.0.
|
Chris@102
|
18 // See http://www.boost.org/LICENSE_1_0.txt
|
Chris@102
|
19 //
|
Chris@102
|
20
|
Chris@102
|
21 #include <boost/assert.hpp>
|
Chris@102
|
22 #include <boost/current_function.hpp>
|
Chris@102
|
23 #include <boost/core/no_exceptions_support.hpp>
|
Chris@102
|
24 #include <cstring> // for memcmp
|
Chris@102
|
25 #include <iostream>
|
Chris@102
|
26
|
Chris@102
|
27 // IDE's like Visual Studio perform better if output goes to std::cout or
|
Chris@102
|
28 // some other stream, so allow user to configure output stream:
|
Chris@102
|
29 #ifndef BOOST_LIGHTWEIGHT_TEST_OSTREAM
|
Chris@102
|
30 # define BOOST_LIGHTWEIGHT_TEST_OSTREAM std::cerr
|
Chris@102
|
31 #endif
|
Chris@102
|
32
|
Chris@102
|
33 namespace boost
|
Chris@102
|
34 {
|
Chris@102
|
35 namespace endian
|
Chris@102
|
36 {
|
Chris@102
|
37 namespace detail
|
Chris@102
|
38 {
|
Chris@102
|
39
|
Chris@102
|
40 struct report_errors_reminder
|
Chris@102
|
41 {
|
Chris@102
|
42 bool called_report_errors_function;
|
Chris@102
|
43
|
Chris@102
|
44 report_errors_reminder() : called_report_errors_function(false) {}
|
Chris@102
|
45
|
Chris@102
|
46 ~report_errors_reminder()
|
Chris@102
|
47 {
|
Chris@102
|
48 BOOST_ASSERT(called_report_errors_function); // verify report_errors() was called
|
Chris@102
|
49 }
|
Chris@102
|
50 };
|
Chris@102
|
51
|
Chris@102
|
52 inline report_errors_reminder& report_errors_remind()
|
Chris@102
|
53 {
|
Chris@102
|
54 static report_errors_reminder r;
|
Chris@102
|
55 return r;
|
Chris@102
|
56 }
|
Chris@102
|
57
|
Chris@102
|
58 inline int & test_errors()
|
Chris@102
|
59 {
|
Chris@102
|
60 static int x = 0;
|
Chris@102
|
61 report_errors_remind();
|
Chris@102
|
62 return x;
|
Chris@102
|
63 }
|
Chris@102
|
64
|
Chris@102
|
65 inline void test_failed_impl(char const * expr, char const * file, int line, char const * function)
|
Chris@102
|
66 {
|
Chris@102
|
67 BOOST_LIGHTWEIGHT_TEST_OSTREAM
|
Chris@102
|
68 << file << "(" << line << "): test '" << expr << "' failed in function '"
|
Chris@102
|
69 << function << "'" << std::endl;
|
Chris@102
|
70 ++test_errors();
|
Chris@102
|
71 }
|
Chris@102
|
72
|
Chris@102
|
73 inline void error_impl(char const * msg, char const * file, int line, char const * function)
|
Chris@102
|
74 {
|
Chris@102
|
75 BOOST_LIGHTWEIGHT_TEST_OSTREAM
|
Chris@102
|
76 << file << "(" << line << "): " << msg << " in function '"
|
Chris@102
|
77 << function << "'" << std::endl;
|
Chris@102
|
78 ++test_errors();
|
Chris@102
|
79 }
|
Chris@102
|
80
|
Chris@102
|
81 inline void throw_failed_impl(char const * excep, char const * file, int line, char const * function)
|
Chris@102
|
82 {
|
Chris@102
|
83 BOOST_LIGHTWEIGHT_TEST_OSTREAM
|
Chris@102
|
84 << file << "(" << line << "): Exception '" << excep << "' not thrown in function '"
|
Chris@102
|
85 << function << "'" << std::endl;
|
Chris@102
|
86 ++test_errors();
|
Chris@102
|
87 }
|
Chris@102
|
88
|
Chris@102
|
89 template<class T, class U> inline void test_eq_impl( char const * expr1, char const * expr2,
|
Chris@102
|
90 char const * file, int line, char const * function, T const & t, U const & u )
|
Chris@102
|
91 {
|
Chris@102
|
92 if( t == u )
|
Chris@102
|
93 {
|
Chris@102
|
94 report_errors_remind();
|
Chris@102
|
95 }
|
Chris@102
|
96 else
|
Chris@102
|
97 {
|
Chris@102
|
98 BOOST_LIGHTWEIGHT_TEST_OSTREAM
|
Chris@102
|
99 << file << "(" << line << "): test '" << expr1 << " == " << expr2
|
Chris@102
|
100 << "' failed in function '" << function << "': "
|
Chris@102
|
101 << "'" << t << "' != '" << u << "'" << std::endl;
|
Chris@102
|
102 ++test_errors();
|
Chris@102
|
103 }
|
Chris@102
|
104 }
|
Chris@102
|
105
|
Chris@102
|
106 template<class T, class U> inline void test_ne_impl( char const * expr1, char const * expr2,
|
Chris@102
|
107 char const * file, int line, char const * function, T const & t, U const & u )
|
Chris@102
|
108 {
|
Chris@102
|
109 if( t != u )
|
Chris@102
|
110 {
|
Chris@102
|
111 report_errors_remind();
|
Chris@102
|
112 }
|
Chris@102
|
113 else
|
Chris@102
|
114 {
|
Chris@102
|
115 BOOST_LIGHTWEIGHT_TEST_OSTREAM
|
Chris@102
|
116 << file << "(" << line << "): test '" << expr1 << " != " << expr2
|
Chris@102
|
117 << "' failed in function '" << function << "': "
|
Chris@102
|
118 << "'" << t << "' == '" << u << "'" << std::endl;
|
Chris@102
|
119 ++test_errors();
|
Chris@102
|
120 }
|
Chris@102
|
121 }
|
Chris@102
|
122
|
Chris@102
|
123 template <class T>
|
Chris@102
|
124 std::string to_hex(const T& x)
|
Chris@102
|
125 {
|
Chris@102
|
126 const char hex[] = { '0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f' };
|
Chris@102
|
127 std::string tmp;
|
Chris@102
|
128 const unsigned char* p = reinterpret_cast<const unsigned char*>(&x);
|
Chris@102
|
129 const unsigned char* e = p + sizeof(T);
|
Chris@102
|
130
|
Chris@102
|
131 for (; p < e; ++p)
|
Chris@102
|
132 {
|
Chris@102
|
133 tmp += hex[*p >> 4]; // high-order nibble
|
Chris@102
|
134 tmp += hex[*p & 0x0f]; // low-order nibble
|
Chris@102
|
135 }
|
Chris@102
|
136 return tmp;
|
Chris@102
|
137 }
|
Chris@102
|
138
|
Chris@102
|
139 template<class T, class U> inline bool test_memcmp_eq_impl(char const * expr1,
|
Chris@102
|
140 char const * expr2, char const * file, int line, char const * function, T const & t,
|
Chris@102
|
141 U const & u)
|
Chris@102
|
142 {
|
Chris@102
|
143 BOOST_ASSERT(sizeof(T) == sizeof(U));
|
Chris@102
|
144 if (sizeof(T) == sizeof(U)
|
Chris@102
|
145 && std::memcmp(&t, &u, sizeof(T)) == 0)
|
Chris@102
|
146 {
|
Chris@102
|
147 report_errors_remind();
|
Chris@102
|
148 return true;
|
Chris@102
|
149 }
|
Chris@102
|
150 else
|
Chris@102
|
151 {
|
Chris@102
|
152 BOOST_LIGHTWEIGHT_TEST_OSTREAM
|
Chris@102
|
153 << file << "(" << line << "): test 'std::memcmp(" << expr1 << ", " << expr2
|
Chris@102
|
154 << ") == 0' fails in function '" << function << "': "
|
Chris@102
|
155 << " with values '" << to_hex(t) << "' and '" << to_hex(u) << "'" << std::endl;
|
Chris@102
|
156 ++test_errors();
|
Chris@102
|
157 return false;
|
Chris@102
|
158 }
|
Chris@102
|
159 }
|
Chris@102
|
160
|
Chris@102
|
161 } // namespace detail
|
Chris@102
|
162
|
Chris@102
|
163 inline int report_errors()
|
Chris@102
|
164 {
|
Chris@102
|
165 boost::endian::detail::report_errors_remind().called_report_errors_function = true;
|
Chris@102
|
166
|
Chris@102
|
167 int errors = boost::endian::detail::test_errors();
|
Chris@102
|
168
|
Chris@102
|
169 if( errors == 0 )
|
Chris@102
|
170 {
|
Chris@102
|
171 BOOST_LIGHTWEIGHT_TEST_OSTREAM
|
Chris@102
|
172 << "No errors detected." << std::endl;
|
Chris@102
|
173 return 0;
|
Chris@102
|
174 }
|
Chris@102
|
175 else
|
Chris@102
|
176 {
|
Chris@102
|
177 BOOST_LIGHTWEIGHT_TEST_OSTREAM
|
Chris@102
|
178 << errors << " error" << (errors == 1? "": "s") << " detected." << std::endl;
|
Chris@102
|
179 return 1;
|
Chris@102
|
180 }
|
Chris@102
|
181 }
|
Chris@102
|
182
|
Chris@102
|
183 } // namespace endian
|
Chris@102
|
184 } // namespace boost
|
Chris@102
|
185
|
Chris@102
|
186 //////////////////////////////////////////////////////////////////////////////////////////
|
Chris@102
|
187 // TODO: Should all test macros return bool? See BOOST_TEST_MEM_EQ usage in fp_exaustive_test,cpp
|
Chris@102
|
188 //////////////////////////////////////////////////////////////////////////////////////////
|
Chris@102
|
189
|
Chris@102
|
190
|
Chris@102
|
191 #define BOOST_TEST(expr) \
|
Chris@102
|
192 ((expr)? (void)0: ::boost::endian::detail::test_failed_impl(#expr, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION))
|
Chris@102
|
193
|
Chris@102
|
194 #define BOOST_ERROR(msg) \
|
Chris@102
|
195 ( ::boost::endian::detail::error_impl(msg, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION) )
|
Chris@102
|
196
|
Chris@102
|
197 #define BOOST_TEST_EQ(expr1,expr2) \
|
Chris@102
|
198 ( ::boost::endian::detail::test_eq_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2) )
|
Chris@102
|
199 #define BOOST_TEST_NE(expr1,expr2) \
|
Chris@102
|
200 ( ::boost::endian::detail::test_ne_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2) )
|
Chris@102
|
201
|
Chris@102
|
202 #define BOOST_TEST_MEM_EQ(expr1,expr2) \
|
Chris@102
|
203 (::boost::endian::detail::test_memcmp_eq_impl(#expr1, #expr2, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION, expr1, expr2))
|
Chris@102
|
204
|
Chris@102
|
205 #ifndef BOOST_NO_EXCEPTIONS
|
Chris@102
|
206 #define BOOST_TEST_THROWS( EXPR, EXCEP ) \
|
Chris@102
|
207 try { \
|
Chris@102
|
208 EXPR; \
|
Chris@102
|
209 ::boost::detail::throw_failed_impl \
|
Chris@102
|
210 (#EXCEP, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION); \
|
Chris@102
|
211 } \
|
Chris@102
|
212 catch(EXCEP const&) { \
|
Chris@102
|
213 } \
|
Chris@102
|
214 catch(...) { \
|
Chris@102
|
215 ::boost::detail::throw_failed_impl \
|
Chris@102
|
216 (#EXCEP, __FILE__, __LINE__, BOOST_CURRENT_FUNCTION); \
|
Chris@102
|
217 } \
|
Chris@102
|
218 //
|
Chris@102
|
219 #else
|
Chris@102
|
220 #define BOOST_TEST_THROWS( EXPR, EXCEP )
|
Chris@102
|
221 #endif
|
Chris@102
|
222
|
Chris@102
|
223 #endif // #ifndef BOOST_ENDIAN_LIGHTWEIGHT_TEST_HPP
|