operators.C 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. // -*- mode: c++; coding: utf-8 -*-
  2. /// @file operators.C
  3. /// @brief Tests for operators on ra:: expr templates.
  4. // (c) Daniel Llorens - 2014-2015
  5. // This library is free software; you can redistribute it and/or modify it under
  6. // the terms of the GNU Lesser General Public License as published by the Free
  7. // Software Foundation; either version 3 of the License, or (at your option) any
  8. // later version.
  9. #include "ra/complex.H"
  10. #include "ra/test.H"
  11. #include "ra/mpdebug.H"
  12. #include "ra/operators.H"
  13. #include "ra/io.H"
  14. #include "ra/big.H"
  15. #include "ra/wedge.H"
  16. using std::cout, std::endl;
  17. using real = double;
  18. using complex = std::complex<double>;
  19. int main()
  20. {
  21. TestRecorder tr;
  22. tr.section("unary ops");
  23. {
  24. #define DEF_TEST_UNARY_OP(OP) \
  25. auto test = [&tr](auto token, auto x, auto y, auto && vx, auto && vy, real err) \
  26. { \
  27. using T = decltype(token); \
  28. using TY = decltype(OP(std::declval<T>())); \
  29. tr.info("scalar-scalar").test_abs_error(OP(T(x)), TY(y), err); \
  30. tr.info("array(0)-scalar").test_abs_error(OP(ra::Unique<T, 0>(x)), TY(y), err); \
  31. tr.info("array(var)-scalar").test_abs_error(OP(ra::Unique<T>(x)), TY(y), err); \
  32. tr.info("array(1)-array(1)").test_abs_error(OP(vx), vy, err); \
  33. };
  34. {
  35. DEF_TEST_UNARY_OP(abs);
  36. test(int(), -3, 3, ra::Unique<int, 1>{1, -3, -2}, ra::Unique<int, 1>{1, 3, 2}, 0.);
  37. test(real(), -3, 3, ra::Unique<real, 1>{1, -3, -2}, ra::Unique<real, 1>{1, 3, 2}, 0.);
  38. test(float(), -3, 3, ra::Unique<float, 1>{1, -3, -2}, ra::Unique<float, 1>{1, 3, 2}, 0.);
  39. test(complex(), -3, 3, ra::Unique<complex, 1>{1, -3, -2}, ra::Unique<complex, 1>{1, 3, 2}, 0.);
  40. }
  41. #define TEST_UNARY_OP_CR(OP, ri, ro, ci, co, err) \
  42. { \
  43. DEF_TEST_UNARY_OP(OP); \
  44. test(real(), ri, ro, ra::Unique<real, 1>{ri, ri, ri}, ra::Unique<complex, 1>{ro, ro, ro}, err); \
  45. test(complex(), ci, co, ra::Unique<complex, 1>{ci, ci}, ra::Unique<complex, 1>{co, co}, err); \
  46. }
  47. TEST_UNARY_OP_CR(conj, 1., 1., complex(1., 2.), complex(1., -2), 0.);
  48. TEST_UNARY_OP_CR(cos, 0., 1., complex(0, 0), complex(1., 0.), 0.);
  49. TEST_UNARY_OP_CR(sin, 1.57079632679489661, 1., complex(1.57079632679489661, 0), complex(1., 0.), 0.);
  50. TEST_UNARY_OP_CR(exp, 0., 1., complex(0, 0), complex(1., 0.), 0.);
  51. TEST_UNARY_OP_CR(sqrt, 4., 2., complex(-1, 0), complex(0., 1.), 1e-16);
  52. TEST_UNARY_OP_CR(xI, 4., complex(0, 4.), complex(1., -2.), complex(2., 1.), 0.);
  53. #undef TEST_UNARY_OP_CR
  54. #undef DEF_TEST_UNARY_OP
  55. // TODO merge with DEF_TEST_UNARY_OP
  56. tr.info("odd").test_eq(ra::Unique<bool, 1> {true, false, true, true}, odd(ra::Unique<int, 1> {1, 2, 3, -1}));
  57. }
  58. tr.section("check decay of rank 0 Containers/Slices w/ operators");
  59. {
  60. {
  61. auto test = [&tr](auto && a)
  62. {
  63. tr.test_eq(12, a*4.);
  64. auto b = a();
  65. static_assert(std::is_same_v<int, decltype(b)>, "unexpected b non-decay to real");
  66. static_assert(std::is_same_v<real, decltype(b*4.)>, "expected b decay to real");
  67. static_assert(std::is_same_v<real, decltype(4.*b)>, "expected b decay to real");
  68. tr.test_eq(12., b*4.);
  69. tr.test_eq(12., 4.*b);
  70. static_assert(std::is_same_v<real, decltype(a*4.)>, "expected a decay to real");
  71. static_assert(std::is_same_v<real, decltype(4.*a)>, "expected a decay to real");
  72. tr.test_eq(12., a*4.);
  73. tr.test_eq(12., 4.*a);
  74. };
  75. test(ra::Small<int>(3));
  76. test(ra::Unique<int, 0>({}, 3));
  77. }
  78. {
  79. ra::Small<int, 3> a { 1, 2, 3 };
  80. ra::Small<int> b { 5 };
  81. a *= b;
  82. tr.test_eq(a[0], 5);
  83. tr.test_eq(a[1], 10);
  84. tr.test_eq(a[2], 15);
  85. }
  86. {
  87. ra::Small<int> a { 3 };
  88. ra::Small<int> b { 2 };
  89. auto c = a*b;
  90. static_assert(std::is_same_v<int, decltype(a*b)>, "expected a, b decay to real"); \
  91. tr.test_eq(c, 6);
  92. }
  93. }
  94. tr.section("lvalue-rvalue operators I");
  95. {
  96. ra::Unique<complex, 1> a({3}, 0.);
  97. imag_part(a) = ra::Unique<real, 1> { 7., 2., 3. }; // TODO operator=(initializer_list) ?
  98. real_part(a) = -imag_part(ra::Unique<complex, 1> { xI(7.), xI(2.), xI(3.) })+1;
  99. tr.test_eq(ra::Unique<complex, 1> {{-6., 7.}, {-1., 2.}, {-2., 3.}}, a);
  100. }
  101. tr.section("lvalue-rvalue operators II [ma115]");
  102. {
  103. ra::Small<std::complex<double>, 2, 2> A = {{1., 2.}, {3., 4.}};
  104. imag_part(A) = -2*real_part(A);
  105. cout << A << endl;
  106. tr.test_eq(ra::Small<std::complex<double>, 2, 2> {{{1., -2.}, {2., -4.}}, {{3., -6.}, {4, -8.}}}, A);
  107. }
  108. tr.section("operators with Unique");
  109. {
  110. ra::Unique<int, 2> a({3, 2}, { 1, 2, 3, 20, 5, 6 });
  111. ra::Unique<int, 1> b({3}, { 10, 20, 30 });
  112. #define TESTSUM(expr) \
  113. tr.test_eq(expr, ra::Small<int, 3, 2> {11, 12, 23, 40, 35, 36});
  114. TESTSUM(ra::expr([](int a, int b) { return a + b; }, a.iter(), b.iter()));
  115. TESTSUM(a.iter() + b.iter());
  116. TESTSUM(a+b);
  117. #undef TESTSUM
  118. #define TESTEQ(expr) \
  119. tr.test_eq(expr, ra::Small<bool, 3, 2> {false, false, false, true, false, false});
  120. TESTEQ(a==b);
  121. TESTEQ(!(a!=b));
  122. #undef TESTEQ
  123. }
  124. tr.section("operators with View");
  125. {
  126. {
  127. ra::Unique<complex, 2> const a({2, 3}, {1, 2, 3, 4, 5, 6});
  128. {
  129. auto a0 = a(0);
  130. tr.test_eq(ra::Small<real, 3>{.5, 1., 1.5}, 0.5*a0);
  131. }
  132. {
  133. auto a0 = a.at(ra::Small<int, 1> { 0 }); // BUG Not sure this is what I want
  134. tr.test_eq(ra::Small<real, 3>{.5, 1., 1.5}, 0.5*a0);
  135. }
  136. }
  137. {
  138. ra::Unique<complex, 1> const a({3}, {1, 2, 3});
  139. {
  140. auto a0 = a(0);
  141. tr.test_eq(0.5, 0.5*a0);
  142. }
  143. {
  144. auto a0 = a.at(ra::Small<int, 1> { 0 }); // BUG Not sure this is what I want, see above
  145. tr.test_eq(2.1, 2.1*a0);
  146. tr.test_eq(0.5, 0.5*a0);
  147. tr.test_eq(0.5, complex(0.5)*a0);
  148. }
  149. }
  150. }
  151. tr.section("operators with Small");
  152. {
  153. ra::Small<int, 3> a { 1, 2, 3 };
  154. ra::Small<int, 3> b { 1, 2, 4 };
  155. tr.test_eq(ra::Small<int, 3> {2, 4, 7}, ra::expr([](int a, int b) { return a + b; }, a.iter(), b.iter()));
  156. tr.test_eq(ra::Small<int, 3> {2, 4, 7}, (a.iter() + b.iter()));
  157. tr.test_eq(ra::Small<int, 3> {2, 4, 7}, a+b);
  158. }
  159. tr.section("constructors from expr"); // TODO For all other Container types.
  160. {
  161. {
  162. // TODO Systematic init-from-expr tests (every expr type vs every container type) with operators.H included.
  163. ra::Unique<int, 1> a({3}, { 1, 2, 3 });
  164. ra::Unique<int, 1> b({3}, { 10, 20, 30 });
  165. ra::Unique<int, 1> c(a.iter() + b.iter());
  166. tr.test_eq(ra::Small<int, 3> {11, 22, 33}, c);
  167. }
  168. {
  169. ra::Unique<int, 2> a({3, 2}, 77);
  170. tr.test_eq(a, ra::Small<int, 3, 2> {77, 77, 77, 77, 77, 77});
  171. }
  172. {
  173. ra::Unique<int, 2> a({3, 2}, ra::cast<int>(ra::_0-ra::_1));
  174. tr.test_eq(ra::Small<int, 3, 2> {0, -1, 1, 0, 2, 1}, a);
  175. }
  176. }
  177. tr.section("mixed ra-type / foreign-scalar operations");
  178. {
  179. ra::Unique<int, 2> a({3, 2}, { 1, 2, 3, 20, 5, 6 });
  180. ra::Small<int, 3, 2> ref {4, 5, 6, 23, 8, 9};
  181. tr.test_eq(ref, ra::expr([](int a, int b) { return a + b; }, ra::start(a), ra::start(3)));
  182. tr.test_eq(ref, ra::start(a) + ra::start(3));
  183. tr.test_eq(ref, a+3);
  184. }
  185. // These are rather different because they have to be defined in-class.
  186. tr.section("constructors & assignment operators with expr rhs"); // TODO use TestRecorder::test_eq().
  187. {
  188. real check0[6] = { 0, -1, 1, 0, 2, 1 };
  189. real check1[6] = { 4, 3, 5, 4, 6, 5 };
  190. real check2[6] = { 8, 6, 10, 8, 12, 10 };
  191. auto test = [&](auto && a)
  192. {
  193. tr.test(std::equal(a.begin(), a.end(), check0));
  194. a += 4;
  195. tr.test(std::equal(a.begin(), a.end(), check1));
  196. a += a;
  197. tr.test(std::equal(a.begin(), a.end(), check2));
  198. };
  199. test(ra::Unique<int, 2>({3, 2}, ra::cast<int>(ra::_0-ra::_1)));
  200. test(ra::Small<int, 3, 2>(ra::cast<int>(ra::_0-ra::_1)));
  201. }
  202. tr.section("assignment ops with ra::scalar [ra21]");
  203. {
  204. ra::Small<real, 2> a { 0, 0 };
  205. ra::Big<ra::Small<real, 2>, 1> b { {1, 10}, {2, 20}, {3, 30} };
  206. // use scalar to match 1 (a) vs 3 (b) instead of 2 vs 3.
  207. ra::scalar(a) += b;
  208. tr.test_eq(ra::Small<real, 2> { 6, 60 }, a);
  209. }
  210. tr.section("pack operator");
  211. {
  212. ra::Small<real, 6> a = { 0, -1, 1, 0, 2, 1 };
  213. ra::Small<int, 6> b = { 4, 3, 5, 4, 6, 5 };
  214. ra::Big<std::tuple<real, int>, 1> x = ra::pack<std::tuple<real, int> >(a, b); // TODO kinda redundant...
  215. tr.test_eq(a, map([](auto && x) -> decltype(auto) { return std::get<0>(x); }, x));
  216. tr.test_eq(b, map([](auto && x) -> decltype(auto) { return std::get<1>(x); }, x));
  217. }
  218. tr.section("pack operator as ref");
  219. {
  220. using T = std::tuple<real, int>;
  221. ra::Big<T> x { T(0., 1), T(2., 3), T(4., 5) };
  222. ra::Small<real, 3> a = -99.;
  223. ra::Small<int, 3> b = -77;
  224. ra::pack<std::tuple<real &, int &> >(a, b) = x;
  225. tr.test_eq(ra::Small<real, 3> {0., 2., 4.}, a);
  226. tr.test_eq(ra::Small<int, 3> {1, 3, 5}, b);
  227. }
  228. tr.section("operator= for View, Container. Cf test/ownership.C");
  229. {
  230. real check5[6] = { 5, 5, 5, 5, 5, 5 };
  231. real check9[6] = { 9, 9, 9, 9, 9, 9 };
  232. ra::Unique<int, 2> a({3, 2}, 7);
  233. ra::Unique<int, 2> b({3, 2}, 5);
  234. ra::View<int, 2> c = a();
  235. ra::View<int, 2> d = b();
  236. c = d;
  237. tr.test(std::equal(a.begin(), a.end(), check5));
  238. ra::Unique<int, 2> t({2, 3}, 9);
  239. c = transpose({1, 0}, t);
  240. tr.test(std::equal(a.begin(), a.end(), check9));
  241. a = d;
  242. tr.test(std::equal(a.begin(), a.end(), check5));
  243. ra::Unique<int, 2> e = d;
  244. tr.test(std::equal(e.begin(), e.end(), check5));
  245. }
  246. tr.section("operator= for Dynamic");
  247. {
  248. ra::Unique<int, 1> a({7}, 7);
  249. ra::Small<ra::dim_t, 3> i { 2, 3, 5 };
  250. ra::Small<int, 3> b { 22, 33, 55 };
  251. ra::expr([&a](ra::dim_t i) -> decltype(auto) { return a(i); }, ra::start(i)) = b;
  252. int checka[] = { 7, 7, 22, 33, 7, 55, 7 };
  253. tr.test(std::equal(checka, checka+7, a.begin()));
  254. }
  255. tr.section("wedge");
  256. {
  257. {
  258. ra::Small<real, 3> a {1, 2, 3};
  259. ra::Small<real, 3> b {4, 5, 7};
  260. ra::Small<real, 3> c;
  261. fun::Wedge<3, 1, 1>::product(a, b, c);
  262. tr.test_eq(ra::Small<real, 3> {-1, 5, -3}, c);
  263. }
  264. {
  265. ra::Small<real, 1> a {2};
  266. ra::Small<real, 1> b {3};
  267. ra::Small<real, 1> r;
  268. fun::Wedge<1, 0, 0>::product(a, b, r);
  269. tr.test_eq(6, r[0]);
  270. tr.test_eq(6, wedge<1, 0, 0>(ra::Small<real, 1>{2}, ra::Small<real, 1>{3}));
  271. tr.test_eq(6, wedge<1, 0, 0>(ra::Small<real, 1>{2}, 3.));
  272. tr.test_eq(6, wedge<1, 0, 0>(2., ra::Small<real, 1>{3}));
  273. tr.test_eq(6, wedge<1, 0, 0>(2., 3));
  274. }
  275. }
  276. tr.section("hodge / hodgex");
  277. {
  278. ra::Small<real, 3> a {1, 2, 3};
  279. ra::Small<real, 3> c;
  280. fun::hodgex<3, 1>(a, c);
  281. tr.test_eq(a, c);
  282. auto d = fun::hodge<3, 1>(a);
  283. tr.test_eq(a, d);
  284. }
  285. tr.section("index");
  286. {
  287. {
  288. ra::Big<real, 1> a {1, 2, 3, -4, 9, 9, 8};
  289. tr.test_eq(3, index(a<0));
  290. tr.test_eq(-1, index(a>100));
  291. }
  292. {
  293. ra::Big<real> a {1, 2, 3, -4, 9, 9, 8};
  294. tr.test_eq(4, index(abs(a)>4));
  295. }
  296. }
  297. tr.section("lexicographical_compare");
  298. {
  299. ra::Big<int, 3> a({10, 2, 2}, {0, 0, 1, 3, 0, 1, 3, 3, 0, 2, 3, 0, 3, 1, 2, 1, 1, 1, 3, 1, 0, 3, 2, 2, 2, 3, 1, 2, 2, 0, 0, 1, 0, 1, 1, 1, 3, 0, 2, 1});
  300. ra::Big<int, 1> i = ra::iota(a.size(0));
  301. std::sort(i.data(), i.data()+i.size(),
  302. [&a](int i, int j)
  303. {
  304. return lexicographical_compare(a(i), a(j));
  305. });
  306. tr.test_eq(ra::start({0, 8, 1, 2, 5, 4, 7, 6, 9, 3}), i);
  307. }
  308. return tr.summary();
  309. }