Files
safe_numerics/test/test_multiply_auto.cpp
2015-08-02 11:30:14 -07:00

276 lines
8.9 KiB
C++

// 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 <iostream>
#include <exception>
#include <cxxabi.h>
#include "../include/safe_integer.hpp"
#include "../include/automatic.hpp"
template <class T>
using safe_t = boost::numeric::safe<
T,
boost::numeric::automatic
>;
template<class T1, class T2>
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<safe_t<T1> >::value, "safe_t not safe!");
safe_t<T1> t1 = v1;
using result_type = decltype(t1 * v2);
result_type result;
static_assert(
is_safe<result_type>::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<safe_t<T2> >::value, "safe_t not safe!");
safe_t<T2> t2 = v2;
using result_type = decltype(v1 * t2);
result_type result;
static_assert(
is_safe<result_type>::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> t1 = v1;
safe_t<T2> t2 = v2;
using result_type = decltype(t1 * t2);
result_type result;
static_assert(
is_safe<result_type>::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 ;
}