test_modulus.hpp 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. #ifndef BOOST_TEST_MODULUS_HPP
  2. #define BOOST_TEST_MODULUS_HPP
  3. // Copyright (c) 2015 Robert Ramey
  4. //
  5. // Distributed under the Boost Software License, Version 1.0. (See
  6. // accompanying file LICENSE_1_0.txt or copy at
  7. // http://www.boost.org/LICENSE_1_0.txt)
  8. #include <iostream>
  9. #include <boost/safe_numerics/safe_integer.hpp>
  10. #include <boost/safe_numerics/range_value.hpp>
  11. template<class T1, class T2>
  12. bool test_modulus(
  13. T1 v1,
  14. T2 v2,
  15. const char *av1,
  16. const char *av2,
  17. char expected_result
  18. ){
  19. std::cout << "testing"<< std::endl;
  20. {
  21. safe_t<T1> t1 = v1;
  22. using result_type = decltype(t1 % v2);
  23. std::cout << "safe<" << av1 << "> % " << av2 << " -> ";
  24. static_assert(
  25. boost::safe_numerics::is_safe<safe_t<T1> >::value,
  26. "safe_t not safe!"
  27. );
  28. static_assert(
  29. boost::safe_numerics::is_safe<result_type>::value,
  30. "Expression failed to return safe type"
  31. );
  32. try{
  33. // use auto to avoid checking assignment.
  34. auto result = t1 % v2;
  35. std::cout << make_result_display(result);
  36. if(expected_result == 'x'){
  37. std::cout
  38. << " ! = "<< av1 << " % " << av2
  39. << " failed to detect error in modulus"
  40. << std::endl;
  41. t1 % v2;
  42. return false;
  43. }
  44. std::cout << std::endl;
  45. }
  46. catch(const std::exception &){
  47. if(expected_result == '.'){
  48. std::cout
  49. << " == "<< av1 << " % " << av2
  50. << " erroneously detected error in modulus"
  51. << std::endl;
  52. try{
  53. t1 % v2;
  54. }
  55. catch(const std::exception &){}
  56. return false;
  57. }
  58. }
  59. }
  60. {
  61. safe_t<T2> t2 = v2;
  62. using result_type = decltype(v1 % t2);
  63. std::cout << av1 << " % " << "safe<" << av2 << "> -> ";
  64. static_assert(
  65. boost::safe_numerics::is_safe<safe_t<T2> >::value,
  66. "safe_t not safe!"
  67. );
  68. static_assert(
  69. boost::safe_numerics::is_safe<result_type>::value,
  70. "Expression failed to return safe type"
  71. );
  72. try{
  73. // use auto to avoid checking assignment.
  74. auto result = v1 % t2;
  75. std::cout << make_result_display(result);
  76. if(expected_result =='x'){
  77. std::cout
  78. << " ! = "<< av1 << " % " << av2
  79. << " failed to detect error in modulus"
  80. << std::endl;
  81. try{
  82. v1 % t2;
  83. }
  84. catch(const std::exception &){}
  85. return false;
  86. }
  87. std::cout << std::endl;
  88. }
  89. catch(const std::exception &){
  90. if(expected_result != 'x'){
  91. std::cout
  92. << " == "<< av1 << " % " << av2
  93. << " erroneously detected error in modulus"
  94. << std::endl;
  95. try{
  96. v1 % t2;
  97. }
  98. catch(const std::exception &){}
  99. return false;
  100. }
  101. }
  102. }
  103. {
  104. safe_t<T1> t1 = v1;
  105. safe_t<T2> t2 = v2;
  106. using result_type = decltype(t1 + t2);
  107. std::cout << "safe<" << av1 << "> % " << "safe<" << av2 << "> -> ";
  108. static_assert(
  109. boost::safe_numerics::is_safe<result_type>::value,
  110. "Expression failed to return safe type"
  111. );
  112. try{
  113. // use auto to avoid checking assignment.
  114. auto result = t1 % t2;
  115. std::cout << make_result_display(result);
  116. if(expected_result != '.'){
  117. std::cout
  118. << " ! = "<< av1 << " % " << av2
  119. << " failed to detect error in modulus"
  120. << std::endl;
  121. try{
  122. t1 % t2;
  123. }
  124. catch(const std::exception &){}
  125. return false;
  126. }
  127. std::cout << std::endl;
  128. }
  129. catch(const std::exception &){
  130. if(expected_result != 'x'){
  131. std::cout
  132. << " == "<< av1 << " % " << av2
  133. << " erroneously detected error in modulus"
  134. << std::endl;
  135. try{
  136. t1 % t2;
  137. }
  138. catch(const std::exception &){}
  139. return false;
  140. }
  141. }
  142. }
  143. return true;
  144. }
  145. #endif // BOOST_TEST_MODULUS