channel_numeric_operations.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. //
  2. // Copyright 2019 Mateusz Loskot <mateusz at loskot dot net>
  3. //
  4. // Distributed under the Boost Software License, Version 1.0
  5. // See accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt
  7. //
  8. #include <boost/gil.hpp>
  9. #include <boost/gil/extension/numeric/channel_numeric_operations.hpp>
  10. #include <tuple>
  11. #include <type_traits>
  12. #define BOOST_TEST_MODULE test_ext_numeric_pixel_numeric_operations
  13. #include "unit_test.hpp"
  14. #include "unit_test_utility.hpp"
  15. #include "core/channel/test_fixture.hpp"
  16. namespace gil = boost::gil;
  17. namespace fixture = boost::gil::test::fixture;
  18. BOOST_AUTO_TEST_SUITE(channel_plus_t)
  19. BOOST_AUTO_TEST_CASE_TEMPLATE(plus_integer_same_types, channel_t, fixture::channel_integer_types)
  20. {
  21. gil::channel_plus_t<channel_t, channel_t, channel_t> f;
  22. BOOST_TEST(f(0, 0) == channel_t(0));
  23. BOOST_TEST(f(100, 27) == channel_t(127));
  24. }
  25. BOOST_AUTO_TEST_CASE_TEMPLATE(plus_integer_mixed_types, channel_t, fixture::channel_integer_types)
  26. {
  27. {
  28. using channel1_t = channel_t;
  29. using channel2_t = std::uint8_t; // duplicates only one of fixture::channel_integer_types
  30. gil::channel_plus_t<channel1_t, channel2_t, channel1_t> f;
  31. BOOST_TEST(f(0, 0) == channel1_t(0));
  32. BOOST_TEST(f(100, 27) == channel_t(127));
  33. }
  34. {
  35. using channel1_t = std::uint8_t; // duplicates only one of fixture::channel_integer_types
  36. using channel2_t = channel_t;
  37. gil::channel_plus_t<channel1_t, channel2_t, channel2_t> f;
  38. BOOST_TEST(f(0, 0) == channel2_t(0));
  39. BOOST_TEST(f(100, 27) == channel_t(127));
  40. }
  41. }
  42. BOOST_AUTO_TEST_CASE_TEMPLATE(plus_integer_signed_types_with_overflow, channel_t, fixture::channel_integer_signed_types)
  43. {
  44. // Signed integer overflow is UB, so just check addition does not yield mathematically
  45. // expected value but is constrained by the range of representable values for given type.
  46. auto const max_value = gil::channel_traits<channel_t>::max_value();
  47. gil::channel_plus_t<channel_t, channel_t, channel_t> f;
  48. BOOST_TEST(f(max_value, 1) != std::int64_t(max_value) + 1);
  49. BOOST_TEST(f(max_value, max_value) != std::int64_t(max_value) + max_value);
  50. }
  51. BOOST_AUTO_TEST_CASE_TEMPLATE(plus_integer_unsigned_types_with_wraparound, channel_t, fixture::channel_integer_unsigned_types)
  52. {
  53. // The C Standard, 6.2.5, paragraph 9 [ISO/IEC 9899:2011], states:
  54. // A computation involving unsigned operands can never overflow, because a result that
  55. // cannot be represented by the resulting unsigned integer type is reduced modulo the number
  56. // that is one greater than the largest value that can be represented by the resulting type.
  57. auto const max_value = gil::channel_traits<channel_t>::max_value();
  58. auto const min_value = gil::channel_traits<channel_t>::min_value();
  59. gil::channel_plus_t<channel_t, channel_t, channel_t> f;
  60. BOOST_TEST(f(max_value, 1) == min_value);
  61. BOOST_TEST(f(max_value, max_value) == max_value - 1);
  62. }
  63. BOOST_AUTO_TEST_SUITE_END() // channel_plus_t
  64. BOOST_AUTO_TEST_SUITE(channel_minus_t)
  65. BOOST_AUTO_TEST_CASE_TEMPLATE(minus_integer_same_types, channel_t, fixture::channel_integer_types)
  66. {
  67. gil::channel_minus_t<channel_t, channel_t, channel_t> f;
  68. BOOST_TEST(f(0, 0) == channel_t(0));
  69. BOOST_TEST(f(100, 27) == channel_t(73));
  70. }
  71. BOOST_AUTO_TEST_CASE_TEMPLATE(minus_integer_mixed_types, channel_t, fixture::channel_integer_types)
  72. {
  73. {
  74. using channel1_t = channel_t;
  75. using channel2_t = std::uint8_t; // duplicates only one of fixture::channel_integer_types
  76. gil::channel_minus_t<channel1_t, channel2_t, channel1_t> f;
  77. BOOST_TEST(f(0, 0) == channel1_t(0));
  78. BOOST_TEST(f(100, 27) == channel_t(73));
  79. }
  80. {
  81. using channel1_t = std::uint8_t; // duplicates only one of fixture::channel_integer_types
  82. using channel2_t = channel_t;
  83. gil::channel_minus_t<channel1_t, channel2_t, channel2_t> f;
  84. BOOST_TEST(f(0, 0) == channel2_t(0));
  85. BOOST_TEST(f(100, 27) == channel_t(73));
  86. }
  87. }
  88. BOOST_AUTO_TEST_SUITE_END() // channel_minus_t
  89. BOOST_AUTO_TEST_SUITE(channel_multiplies_t)
  90. BOOST_AUTO_TEST_CASE_TEMPLATE(multiplies_integer_same_types, channel_t, fixture::channel_integer_types)
  91. {
  92. gil::channel_multiplies_t<channel_t, channel_t, channel_t> f;
  93. BOOST_TEST(f(0, 0) == channel_t(0));
  94. BOOST_TEST(f(1, 1) == channel_t(1));
  95. BOOST_TEST(f(4, 2) == channel_t(8));
  96. }
  97. BOOST_AUTO_TEST_CASE_TEMPLATE(multiplies_integer_mixed_types, channel_t, fixture::channel_integer_types)
  98. {
  99. {
  100. using channel1_t = channel_t;
  101. using channel2_t = std::uint8_t; // duplicates only one of fixture::channel_integer_types
  102. gil::channel_multiplies_t<channel1_t, channel2_t, channel1_t> f;
  103. BOOST_TEST(f(0, 0) == channel1_t(0));
  104. BOOST_TEST(f(4, 2) == channel_t(8));
  105. }
  106. {
  107. using channel1_t = std::uint8_t; // duplicates only one of fixture::channel_integer_types
  108. using channel2_t = channel_t;
  109. gil::channel_multiplies_t<channel1_t, channel2_t, channel2_t> f;
  110. BOOST_TEST(f(0, 0) == channel2_t(0));
  111. BOOST_TEST(f(4, 2) == channel_t(8));
  112. }
  113. }
  114. BOOST_AUTO_TEST_SUITE_END() // channel_multiplies_t
  115. BOOST_AUTO_TEST_SUITE(channel_divides_t)
  116. BOOST_AUTO_TEST_CASE_TEMPLATE(divides_integer_same_types, channel_t, fixture::channel_integer_types)
  117. {
  118. gil::channel_divides_t<channel_t, channel_t, channel_t> f;
  119. BOOST_TEST(f(0, 1) == channel_t(0));
  120. BOOST_TEST(f(1, 1) == channel_t(1));
  121. BOOST_TEST(f(4, 2) == channel_t(2));
  122. }
  123. BOOST_AUTO_TEST_CASE_TEMPLATE(divides_integer_mixed_types, channel_t, fixture::channel_integer_types)
  124. {
  125. {
  126. using channel1_t = channel_t;
  127. using channel2_t = std::uint8_t; // duplicates only one of fixture::channel_integer_types
  128. gil::channel_divides_t<channel1_t, channel2_t, channel1_t> f;
  129. BOOST_TEST(f(0, 1) == channel1_t(0));
  130. BOOST_TEST(f(4, 2) == channel_t(2));
  131. }
  132. {
  133. using channel1_t = std::uint8_t; // duplicates only one of fixture::channel_integer_types
  134. using channel2_t = channel_t;
  135. gil::channel_divides_t<channel1_t, channel2_t, channel2_t> f;
  136. BOOST_TEST(f(0, 1) == channel2_t(0));
  137. BOOST_TEST(f(4, 2) == channel_t(2));
  138. }
  139. }
  140. BOOST_AUTO_TEST_SUITE_END() // channel_divides_t
  141. BOOST_AUTO_TEST_SUITE(channel_plus_scalar_t)
  142. BOOST_AUTO_TEST_CASE_TEMPLATE(plus_scalar_integer_same_types, channel_t, fixture::channel_integer_types)
  143. {
  144. gil::channel_plus_scalar_t<channel_t, int, channel_t> f;
  145. BOOST_TEST(f(0, 0) == channel_t(0));
  146. BOOST_TEST(f(100, 27) == channel_t(127));
  147. }
  148. BOOST_AUTO_TEST_CASE_TEMPLATE(plus_scalar_integer_mixed_types, channel_t, fixture::channel_integer_types)
  149. {
  150. using channel_result_t = std::uint8_t;
  151. gil::channel_plus_scalar_t<channel_t, int, channel_result_t> f;
  152. BOOST_TEST(f(0, 0) == channel_result_t(0));
  153. BOOST_TEST(f(100, 27) == channel_result_t(127));
  154. }
  155. BOOST_AUTO_TEST_SUITE_END() // channel_plus_scalar_t
  156. BOOST_AUTO_TEST_SUITE(channel_minus_scalar_t)
  157. BOOST_AUTO_TEST_CASE_TEMPLATE(minus_scalar_integer_same_types, channel_t, fixture::channel_integer_types)
  158. {
  159. gil::channel_minus_scalar_t<channel_t, int, channel_t> f;
  160. BOOST_TEST(f(0, 0) == channel_t(0));
  161. BOOST_TEST(f(100, 27) == channel_t(73));
  162. }
  163. BOOST_AUTO_TEST_CASE_TEMPLATE(minus_scalar_integer_mixed_types, channel_t, fixture::channel_integer_types)
  164. {
  165. using channel_result_t = std::uint8_t;
  166. gil::channel_minus_scalar_t<channel_t, int, std::uint8_t> f;
  167. BOOST_TEST(f(0, 0) == channel_result_t(0));
  168. BOOST_TEST(f(100, 27) == channel_result_t(73));
  169. }
  170. BOOST_AUTO_TEST_SUITE_END() // channel_minus_scalar_t
  171. BOOST_AUTO_TEST_SUITE(channel_multiplies_scalar_t)
  172. BOOST_AUTO_TEST_CASE_TEMPLATE(multiplies_scalar_integer_same_types, channel_t, fixture::channel_integer_types)
  173. {
  174. gil::channel_multiplies_scalar_t<channel_t, channel_t, channel_t> f;
  175. BOOST_TEST(f(0, 0) == channel_t(0));
  176. BOOST_TEST(f(1, 1) == channel_t(1));
  177. BOOST_TEST(f(4, 2) == channel_t(8));
  178. }
  179. BOOST_AUTO_TEST_CASE_TEMPLATE(multiplies_scalar_integer_mixed_types, channel_t, fixture::channel_integer_types)
  180. {
  181. using channel_result_t = std::uint8_t;
  182. gil::channel_multiplies_scalar_t<channel_t, int, channel_result_t> f;
  183. BOOST_TEST(f(0, 0) == channel_result_t(0));
  184. BOOST_TEST(f(4, 2) == channel_result_t(8));
  185. }
  186. BOOST_AUTO_TEST_SUITE_END() // channel_multiplies_scalar_t
  187. BOOST_AUTO_TEST_SUITE(channel_divides_scalar_t)
  188. BOOST_AUTO_TEST_CASE_TEMPLATE(divides_scalar_integer_same_types, channel_t, fixture::channel_integer_types)
  189. {
  190. gil::channel_divides_scalar_t<channel_t, channel_t, channel_t> f;
  191. BOOST_TEST(f(0, 1) == channel_t(0));
  192. BOOST_TEST(f(1, 1) == channel_t(1));
  193. BOOST_TEST(f(4, 2) == channel_t(2));
  194. }
  195. BOOST_AUTO_TEST_CASE_TEMPLATE(divides_scalar_integer_mixed_types, channel_t, fixture::channel_integer_types)
  196. {
  197. using channel_result_t = std::uint8_t; // duplicates only one of fixture::channel_integer_types
  198. gil::channel_divides_scalar_t<channel_t, int, channel_result_t> f;
  199. BOOST_TEST(f(0, 1) == channel_t(0));
  200. BOOST_TEST(f(4, 2) == channel_t(2));
  201. }
  202. BOOST_AUTO_TEST_SUITE_END() // channel_divides_scalar_t
  203. BOOST_AUTO_TEST_SUITE(channel_halves_t)
  204. BOOST_AUTO_TEST_CASE_TEMPLATE(halves_integer_same_types, channel_t, fixture::channel_integer_types)
  205. {
  206. gil::channel_halves_t<channel_t> f;
  207. {
  208. channel_t c(0);
  209. f(c);
  210. BOOST_TEST(c == channel_t(0));
  211. }
  212. {
  213. channel_t c(2);
  214. f(c);
  215. BOOST_TEST(c == channel_t(1));
  216. }
  217. {
  218. channel_t c(4);
  219. f(c);
  220. BOOST_TEST(c == channel_t(2));
  221. }
  222. }
  223. BOOST_AUTO_TEST_SUITE_END() // channel_halves_t
  224. BOOST_AUTO_TEST_SUITE(channel_zeros_t)
  225. BOOST_AUTO_TEST_CASE_TEMPLATE(zeros_integer_same_types, channel_t, fixture::channel_integer_types)
  226. {
  227. gil::channel_zeros_t<channel_t> f;
  228. {
  229. channel_t c(0);
  230. f(c);
  231. BOOST_TEST(c == channel_t(0));
  232. }
  233. {
  234. channel_t c(2);
  235. f(c);
  236. BOOST_TEST(c == channel_t(0));
  237. }
  238. {
  239. channel_t c(4);
  240. f(c);
  241. BOOST_TEST(c == channel_t(0));
  242. }
  243. }
  244. BOOST_AUTO_TEST_SUITE_END() // channel_zeros_t
  245. BOOST_AUTO_TEST_SUITE(channel_assigns_t)
  246. BOOST_AUTO_TEST_CASE_TEMPLATE(assigns_integer_same_types, channel_t, fixture::channel_integer_types)
  247. {
  248. gil::channel_assigns_t<channel_t, channel_t> f;
  249. {
  250. channel_t c1(10);
  251. channel_t c2(20);
  252. f(c1, c2);
  253. BOOST_TEST(c2 == c1);
  254. }
  255. }
  256. BOOST_AUTO_TEST_SUITE_END() // channel_assigns_t