// Copyright (c) 2012 Robert Ramey // // 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/safe_integer.hpp" #include "../include/automatic.hpp" template using safe_t = boost::numeric::safe< T, boost::numeric::automatic >; template bool test_multiply_auto( T1 v1, T2 v2, const char *av1, const char *av2, char expected_result ){ using namespace boost::numeric; auto unsafe_result = v1 + v2; { std::cout << "testing safe<" << av1 << "> * " << av2 << " -> "; static_assert(is_safe >::value, "safe_t not safe!"); safe_t t1 = v1; using result_type = decltype(t1 * v2); result_type result; static_assert( is_safe::value, "Expression failed to return safe type" ); try{ result = t1 * v2; std::cout << std::hex << result << "(" << std::dec << result << ")" << std::endl; if(expected_result == 'x'){ const std::type_info & ti = typeid(result); int status; std::cout << "*** failed to detect error in multiplication " << abi::__cxa_demangle(ti.name(),0,0,&status) << '\n' << std::endl; try{ t1 * v2; } catch(std::exception){} assert(result != unsafe_result); return false; } } catch(std::exception){ std::cout << std::hex << result << "(" << std::dec << result << ")" << std::endl; if(expected_result == '.'){ const std::type_info & ti = typeid(result); int status; std::cout << "*** erroneously detected error in multiplication " << abi::__cxa_demangle(ti.name(),0,0,&status) << '\n' << std::endl; try{ t1 * v2; } catch(std::exception){} assert(result == unsafe_result); return false; } } } { std::cout << "testing " << av1 << " * " << "safe<"<< av2 << "> -> "; static_assert(is_safe >::value, "safe_t not safe!"); safe_t t2 = v2; using result_type = decltype(v1 * t2); result_type result; static_assert( is_safe::value, "Expression failed to return safe type" ); try{ result = v1 * t2; std::cout << std::hex << result << "(" << std::dec << result << ")" << std::endl; if(expected_result == 'x'){ const std::type_info & ti = typeid(result); int status; std::cout << "*** failed to detect error in multiplication " << abi::__cxa_demangle(ti.name(),0,0,&status) << '\n' << std::endl; try{ v1 * t2; } catch(std::exception){} assert(result != unsafe_result); return false; } } catch(std::exception){ std::cout << std::hex << result << "(" << std::dec << result << ")" << std::endl; if(expected_result == '.'){ const std::type_info & ti = typeid(result); int status; std::cout << "*** erroneously detected error in multiplication " << abi::__cxa_demangle(ti.name(),0,0,&status) << '\n' << std::endl; try{ v1 * t2; } catch(std::exception){} assert(result == unsafe_result); return false; } } } { std::cout << "testing safe<" << av1 << "> * safe<" << av2 << "> -> "; safe_t t1 = v1; safe_t t2 = v2; using result_type = decltype(t1 * t2); result_type result; static_assert( is_safe::value, "Expression failed to return safe type" ); try{ result = t1 * t2; std::cout << std::hex << result << "(" << std::dec << result << ")" << std::endl; if(expected_result == 'x'){ const std::type_info & ti = typeid(result); int status; std::cout << "*** failed to detect error in multiplication " << abi::__cxa_demangle(ti.name(),0,0,&status) << '\n' << std::endl; try{ t1 * t2; } catch(std::exception){} assert(result != unsafe_result); return false; } } catch(std::exception){ std::cout << std::hex << result << "(" << std::dec << result << ")" << std::endl; if(expected_result == '.'){ const std::type_info & ti = typeid(result); int status; std::cout << "*** erroneously detected error in multiplication " << abi::__cxa_demangle(ti.name(),0,0,&status) << '\n' << std::endl; try{ t1 * t2; } catch(std::exception){} assert(result == unsafe_result); return false; } } } return true; // correct result } #include "test.hpp" #include "test_values.hpp" // note: same test matrix as used in test_checked. Here we test all combinations // safe and unsafe integers. in test_checked we test all combinations of // integer primitives const char *test_multiplication_result[VALUE_ARRAY_SIZE] = { // 0 0 0 0 // 01234567012345670123456701234567 // 01234567890123456789012345678901 /* 0*/ "................................", /* 1*/ ".............xx..............xxx", /* 2*/ ".............xx.............xxxx", /* 3*/ "..............x.............xxxx", /* 4*/ "................................", /* 5*/ ".............xx..............xxx", /* 6*/ ".............xx.............xxxx", /* 7*/ "..............x.............xxxx", /* 8*/ "................................", /* 9*/ ".............xx..............xxx", /*10*/ ".............xx.............xxxx", /*11*/ "..............x.............xxxx", /*12*/ "................................", /*13*/ ".xx..xx..xx..xx..xxx.xxx.xxx.xxx", /*14*/ ".xxx.xxx.xxx.xxx.xxx.xxx.xxxxxxx", /*15*/ "..............x.............xxxx", // 0 0 0 0 // 01234567012345670123456701234567 // 01234567890123456789012345678901 /*16*/ "................................", /*17*/ ".............xx..............xxx", /*18*/ ".............xx..............xxx", /*19*/ ".............xx..............xxx", /*20*/ "................................", /*21*/ ".............xx..............xxx", /*22*/ ".............xx..............xxx", /*23*/ ".............xx..............xxx", /*24*/ "................................", /*25*/ ".............xx..............xxx", /*26*/ ".............xx..............xxx", /*27*/ ".............xx..............xxx", /*28*/ "..xx..xx..xx..xx................", /*29*/ ".xxx.xxx.xxx.xxx.xxx.xxx.xxx.xxx", /*30*/ ".xxx.xxx.xxx.xxx.xxx.xxx.xxx.xxx", /*31*/ ".xxx.xxx.xxx.xxx.xxx.xxx.xxx.xxx" }; #define TEST_PRINT(i1, i2) \ (std::cout << i1 << ',' << i2 << ' ' \ << test_multiplication_result[i1][i2] \ << std::endl); #define TEST_IMPL(v1, v2, result) \ rval &= test_multiply_auto( \ v1, \ v2, \ BOOST_PP_STRINGIZE(v1), \ BOOST_PP_STRINGIZE(v2), \ result \ ); /**/ #define TESTX(value_index1, value_index2) \ TEST_PRINT(value_index1, value_index2); \ TEST_IMPL( \ BOOST_PP_ARRAY_ELEM(value_index1, VALUES), \ BOOST_PP_ARRAY_ELEM(value_index2, VALUES), \ test_multiplication_result[value_index1][value_index2] \ ) /**/ int main(int argc, char * argv[]){ // sanity check on test matrix - should be symetrical for(int i = 0; i < VALUE_ARRAY_SIZE; ++i) for(int j = i + 1; j < VALUE_ARRAY_SIZE; ++j) if(test_multiplication_result[i][j] != test_multiplication_result[j][i]){ std::cout << i << ',' << j << std::endl; return 1; } bool rval = true; TEST_EACH_VALUE_PAIR return ! rval ; }