You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

test_mat_pixel_affine.cpp 16 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495
  1. // Copyright 2020 Tencent
  2. // SPDX-License-Identifier: BSD-3-Clause
  3. #include "mat.h"
  4. #include "prng.h"
  5. #include <string.h>
  6. static struct prng_rand_t g_prng_rand_state;
  7. #define SRAND(seed) prng_srand(seed, &g_prng_rand_state)
  8. #define RAND() prng_rand(&g_prng_rand_state)
  9. static ncnn::Mat generate_ncnn_logo(int w, int h)
  10. {
  11. // clang-format off
  12. // *INDENT-OFF*
  13. static const unsigned char ncnn_logo_data[16][16] =
  14. {
  15. {245, 245, 33, 245, 245, 245, 245, 245, 245, 245, 245, 245, 245, 33, 245, 245},
  16. {245, 33, 33, 33, 245, 245, 245, 245, 245, 245, 245, 245, 33, 33, 33, 245},
  17. {245, 33, 158, 158, 33, 245, 245, 245, 245, 245, 245, 33, 158, 158, 33, 245},
  18. { 33, 117, 158, 224, 158, 33, 245, 245, 245, 245, 33, 158, 224, 158, 117, 33},
  19. { 33, 117, 224, 224, 224, 66, 33, 33, 33, 33, 66, 224, 224, 224, 117, 33},
  20. { 33, 189, 224, 224, 224, 224, 224, 224, 224, 224, 224, 224, 224, 224, 189, 33},
  21. { 33, 224, 224, 224, 224, 224, 224, 224, 224, 224, 224, 224, 224, 224, 224, 33},
  22. { 33, 224, 224, 97, 97, 97, 97, 224, 224, 97, 97, 97, 97, 224, 224, 33},
  23. { 33, 224, 224, 97, 33, 0, 189, 224, 224, 97, 0, 33, 97, 224, 224, 33},
  24. { 33, 224, 224, 97, 33, 0, 189, 224, 224, 97, 0, 33, 97, 224, 224, 33},
  25. { 33, 224, 224, 97, 97, 97, 97, 224, 224, 97, 189, 189, 97, 224, 224, 33},
  26. { 33, 66, 66, 66, 224, 224, 224, 224, 224, 224, 224, 224, 66, 66, 66, 33},
  27. { 66, 158, 158, 66, 66, 224, 224, 224, 224, 224, 224, 66, 158, 158, 66, 66},
  28. { 66, 158, 158, 208, 66, 224, 224, 224, 224, 224, 224, 66, 158, 158, 208, 66},
  29. { 66, 224, 202, 158, 66, 224, 224, 224, 224, 224, 224, 66, 224, 202, 158, 66},
  30. { 66, 158, 224, 158, 66, 224, 224, 224, 224, 224, 224, 66, 158, 224, 158, 66}
  31. };
  32. // *INDENT-ON*
  33. // clang-format on
  34. ncnn::Mat m(w, h, (size_t)1, 1);
  35. resize_bilinear_c1((const unsigned char*)ncnn_logo_data, 16, 16, m, w, h);
  36. return m;
  37. }
  38. static ncnn::Mat RandomMat(int w, int h, int elempack)
  39. {
  40. ncnn::Mat image = generate_ncnn_logo(w, h);
  41. ncnn::Mat m(w, h, 1, (size_t)elempack, elempack);
  42. for (int i = 0; i < h; i++)
  43. {
  44. unsigned char* p = m.row<unsigned char>(i);
  45. const unsigned char* pb = image.row<const unsigned char>(i);
  46. for (int j = 0; j < w; j++)
  47. {
  48. for (int k = 0; k < elempack; k++)
  49. {
  50. p[k] = pb[0];
  51. }
  52. p += elempack;
  53. pb += 1;
  54. }
  55. }
  56. return m;
  57. }
  58. static bool NearlyEqual(unsigned char a, unsigned char b)
  59. {
  60. return abs(a - b) <= 10;
  61. }
  62. static int CompareNearlyEqual(const ncnn::Mat& a, const ncnn::Mat& b)
  63. {
  64. for (int i = 0; i < a.h; i++)
  65. {
  66. const unsigned char* pa = a.row<const unsigned char>(i);
  67. const unsigned char* pb = b.row<const unsigned char>(i);
  68. for (int j = 0; j < a.w; j++)
  69. {
  70. for (int k = 0; k < a.elempack; k++)
  71. {
  72. if (!NearlyEqual(pa[k], pb[k]))
  73. {
  74. fprintf(stderr, "value not match at h:%d w:%d [%d] expect %d but got %d\n", i, j, k, pa[k], pb[k]);
  75. return -1;
  76. }
  77. }
  78. pa += a.elempack;
  79. pb += a.elempack;
  80. }
  81. }
  82. return 0;
  83. }
  84. static int test_mat_pixel_affine_a(int w, int h)
  85. {
  86. for (int c = 1; c <= 4; c++)
  87. {
  88. ncnn::Mat a0 = RandomMat(w, h, c);
  89. float tm[6];
  90. float tm_inv[6];
  91. ncnn::get_rotation_matrix(10.f, 0.15f, w / 2, h / 2, tm);
  92. ncnn::invert_affine_transform(tm, tm_inv);
  93. ncnn::Mat a1(w / 2, h / 2, (size_t)c, c);
  94. ncnn::Mat a2 = a0.clone();
  95. if (c == 1)
  96. {
  97. ncnn::warpaffine_bilinear_c1(a0, w, h, a1, w / 2, h / 2, tm, 0);
  98. ncnn::warpaffine_bilinear_c1(a1, w / 2, h / 2, a2, w, h, tm_inv, -233);
  99. }
  100. if (c == 2)
  101. {
  102. ncnn::warpaffine_bilinear_c2(a0, w, h, a1, w / 2, h / 2, tm, 0);
  103. ncnn::warpaffine_bilinear_c2(a1, w / 2, h / 2, a2, w, h, tm_inv, -233);
  104. }
  105. if (c == 3)
  106. {
  107. ncnn::warpaffine_bilinear_c3(a0, w, h, a1, w / 2, h / 2, tm, 0);
  108. ncnn::warpaffine_bilinear_c3(a1, w / 2, h / 2, a2, w, h, tm_inv, -233);
  109. }
  110. if (c == 4)
  111. {
  112. ncnn::warpaffine_bilinear_c4(a0, w, h, a1, w / 2, h / 2, tm, 0);
  113. ncnn::warpaffine_bilinear_c4(a1, w / 2, h / 2, a2, w, h, tm_inv, -233);
  114. }
  115. if (CompareNearlyEqual(a0, a2) != 0)
  116. {
  117. fprintf(stderr, "test_mat_pixel_affine_a failed w=%d h=%d c=%d\n", w, h, c);
  118. return -1;
  119. }
  120. }
  121. return 0;
  122. }
  123. static int test_mat_pixel_affine_b(int w, int h)
  124. {
  125. for (int c = 1; c <= 4; c++)
  126. {
  127. ncnn::Mat a0 = RandomMat(w, h, c);
  128. float tm[6];
  129. float tm_inv[6];
  130. ncnn::get_rotation_matrix(20.f, 0.25f, w / 4, h / 4, tm);
  131. ncnn::invert_affine_transform(tm, tm_inv);
  132. ncnn::Mat a1(w / 4, h / 4, (size_t)c, c);
  133. ncnn::Mat a2 = a0.clone();
  134. if (c == 1)
  135. {
  136. ncnn::warpaffine_bilinear_c1(a0, w, h, a1, w / 4, h / 4, tm, 0);
  137. ncnn::warpaffine_bilinear_c1(a1, w / 4, h / 4, a2, w, h, tm_inv, -233);
  138. }
  139. if (c == 2)
  140. {
  141. ncnn::warpaffine_bilinear_c2(a0, w, h, a1, w / 4, h / 4, tm, 0);
  142. ncnn::warpaffine_bilinear_c2(a1, w / 4, h / 4, a2, w, h, tm_inv, -233);
  143. }
  144. if (c == 3)
  145. {
  146. ncnn::warpaffine_bilinear_c3(a0, w, h, a1, w / 4, h / 4, tm, 0);
  147. ncnn::warpaffine_bilinear_c3(a1, w / 4, h / 4, a2, w, h, tm_inv, -233);
  148. }
  149. if (c == 4)
  150. {
  151. ncnn::warpaffine_bilinear_c4(a0, w, h, a1, w / 4, h / 4, tm, 0);
  152. ncnn::warpaffine_bilinear_c4(a1, w / 4, h / 4, a2, w, h, tm_inv, -233);
  153. }
  154. if (CompareNearlyEqual(a0, a2) != 0)
  155. {
  156. fprintf(stderr, "test_mat_pixel_affine_b failed w=%d h=%d c=%d\n", w, h, c);
  157. return -1;
  158. }
  159. }
  160. return 0;
  161. }
  162. static int test_mat_pixel_affine_c(int w, int h)
  163. {
  164. for (int c = 1; c <= 4; c++)
  165. {
  166. ncnn::Mat a0 = RandomMat(w, h, c);
  167. float tm[6];
  168. float tm_inv[6];
  169. ncnn::get_rotation_matrix(-30.f, 0.6f, w / 2, h / 2, tm);
  170. ncnn::invert_affine_transform(tm, tm_inv);
  171. ncnn::Mat a1(w / 2, h / 2, (size_t)c, c);
  172. ncnn::Mat a2 = a0.clone();
  173. if (c == 1)
  174. {
  175. ncnn::warpaffine_bilinear_c1(a0, w, h, a1, w / 2, h / 2, tm, 0);
  176. ncnn::warpaffine_bilinear_c1(a1, w / 2, h / 2, a2, w, h, tm_inv, -233);
  177. }
  178. if (c == 2)
  179. {
  180. ncnn::warpaffine_bilinear_c2(a0, w, h, a1, w / 2, h / 2, tm, 0);
  181. ncnn::warpaffine_bilinear_c2(a1, w / 2, h / 2, a2, w, h, tm_inv, -233);
  182. }
  183. if (c == 3)
  184. {
  185. ncnn::warpaffine_bilinear_c3(a0, w, h, a1, w / 2, h / 2, tm, 0);
  186. ncnn::warpaffine_bilinear_c3(a1, w / 2, h / 2, a2, w, h, tm_inv, -233);
  187. }
  188. if (c == 4)
  189. {
  190. ncnn::warpaffine_bilinear_c4(a0, w, h, a1, w / 2, h / 2, tm, 0);
  191. ncnn::warpaffine_bilinear_c4(a1, w / 2, h / 2, a2, w, h, tm_inv, -233);
  192. }
  193. if (CompareNearlyEqual(a0, a2) != 0)
  194. {
  195. fprintf(stderr, "test_mat_pixel_affine_c failed w=%d h=%d c=%d\n", w, h, c);
  196. return -1;
  197. }
  198. }
  199. return 0;
  200. }
  201. static int test_mat_pixel_affine_d(int w, int h)
  202. {
  203. for (int c = 1; c <= 4; c++)
  204. {
  205. ncnn::Mat a0 = RandomMat(w, h, c);
  206. const float points_from[4] = {w / 8.f, h / 8.f, w / 8.f + 1.f, h / 8.f + 3.f};
  207. const float points_to[4] = {w / 2.f, h / 2.f, w / 2.f + 2.f, h / 2.f};
  208. float tm[6];
  209. float tm_inv[6];
  210. ncnn::get_affine_transform(points_from, points_to, 2, tm);
  211. ncnn::invert_affine_transform(tm, tm_inv);
  212. ncnn::Mat a1(w / 4, h / 4, (size_t)c, c);
  213. ncnn::Mat a2 = a0.clone();
  214. if (c == 1)
  215. {
  216. ncnn::warpaffine_bilinear_c1(a0, w, h, a1, w / 4, h / 4, tm, 0);
  217. ncnn::warpaffine_bilinear_c1(a1, w / 4, h / 4, a2, w, h, tm_inv, -233);
  218. }
  219. if (c == 2)
  220. {
  221. ncnn::warpaffine_bilinear_c2(a0, w, h, a1, w / 4, h / 4, tm, 0);
  222. ncnn::warpaffine_bilinear_c2(a1, w / 4, h / 4, a2, w, h, tm_inv, -233);
  223. }
  224. if (c == 3)
  225. {
  226. ncnn::warpaffine_bilinear_c3(a0, w, h, a1, w / 4, h / 4, tm, 0);
  227. ncnn::warpaffine_bilinear_c3(a1, w / 4, h / 4, a2, w, h, tm_inv, -233);
  228. }
  229. if (c == 4)
  230. {
  231. ncnn::warpaffine_bilinear_c4(a0, w, h, a1, w / 4, h / 4, tm, 0);
  232. ncnn::warpaffine_bilinear_c4(a1, w / 4, h / 4, a2, w, h, tm_inv, -233);
  233. }
  234. if (CompareNearlyEqual(a0, a2) != 0)
  235. {
  236. fprintf(stderr, "test_mat_pixel_affine_d failed w=%d h=%d c=%d\n", w, h, c);
  237. return -1;
  238. }
  239. }
  240. return 0;
  241. }
  242. static int test_mat_pixel_affine_e(int w, int h)
  243. {
  244. for (int c = 1; c <= 4; c++)
  245. {
  246. ncnn::Mat a0 = RandomMat(w, h, c);
  247. float tm[6];
  248. float tm_inv[6];
  249. ncnn::get_rotation_matrix(-180.f, 0.5f, w / 2, h / 2, tm);
  250. ncnn::invert_affine_transform(tm, tm_inv);
  251. ncnn::Mat a1(w, h, (size_t)c, c);
  252. ncnn::Mat a2 = a0.clone();
  253. if (c == 1)
  254. {
  255. ncnn::warpaffine_bilinear_c1(a0, w, h, a1, w, h, tm, 0);
  256. ncnn::warpaffine_bilinear_c1(a1, w, h, a2, w, h, tm_inv, -233);
  257. }
  258. if (c == 2)
  259. {
  260. ncnn::warpaffine_bilinear_c2(a0, w, h, a1, w, h, tm, 0);
  261. ncnn::warpaffine_bilinear_c2(a1, w, h, a2, w, h, tm_inv, -233);
  262. }
  263. if (c == 3)
  264. {
  265. ncnn::warpaffine_bilinear_c3(a0, w, h, a1, w, h, tm, 0);
  266. ncnn::warpaffine_bilinear_c3(a1, w, h, a2, w, h, tm_inv, -233);
  267. }
  268. if (c == 4)
  269. {
  270. ncnn::warpaffine_bilinear_c4(a0, w, h, a1, w, h, tm, 0);
  271. ncnn::warpaffine_bilinear_c4(a1, w, h, a2, w, h, tm_inv, -233);
  272. }
  273. if (CompareNearlyEqual(a0, a2) != 0)
  274. {
  275. fprintf(stderr, "test_mat_pixel_affine_e failed w=%d h=%d c=%d\n", w, h, c);
  276. return -1;
  277. }
  278. }
  279. return 0;
  280. }
  281. static int test_mat_pixel_affine_f(int w, int h)
  282. {
  283. for (int c = 1; c <= 4; c++)
  284. {
  285. ncnn::Mat a0 = RandomMat(w, h, c);
  286. float tm[6];
  287. float tm_inv[6];
  288. ncnn::get_rotation_matrix(0.002f, 1.0f, w / 2, h / 2, tm);
  289. ncnn::invert_affine_transform(tm, tm_inv);
  290. ncnn::Mat a1(w * 2, h * 2, (size_t)c, c);
  291. ncnn::Mat a2 = a0.clone();
  292. if (c == 1)
  293. {
  294. ncnn::warpaffine_bilinear_c1(a0, w, h, a1, w * 2, h * 2, tm, 0);
  295. ncnn::warpaffine_bilinear_c1(a1, w * 2, h * 2, a2, w, h, tm_inv, -233);
  296. }
  297. if (c == 2)
  298. {
  299. ncnn::warpaffine_bilinear_c2(a0, w, h, a1, w * 2, h * 2, tm, 0);
  300. ncnn::warpaffine_bilinear_c2(a1, w * 2, h * 2, a2, w, h, tm_inv, -233);
  301. }
  302. if (c == 3)
  303. {
  304. ncnn::warpaffine_bilinear_c3(a0, w, h, a1, w * 2, h * 2, tm, 0);
  305. ncnn::warpaffine_bilinear_c3(a1, w * 2, h * 2, a2, w, h, tm_inv, -233);
  306. }
  307. if (c == 4)
  308. {
  309. ncnn::warpaffine_bilinear_c4(a0, w, h, a1, w * 2, h * 2, tm, 0);
  310. ncnn::warpaffine_bilinear_c4(a1, w * 2, h * 2, a2, w, h, tm_inv, -233);
  311. }
  312. if (CompareNearlyEqual(a0, a2) != 0)
  313. {
  314. fprintf(stderr, "test_mat_pixel_affine_f failed w=%d h=%d c=%d\n", w, h, c);
  315. return -1;
  316. }
  317. }
  318. return 0;
  319. }
  320. static int test_mat_pixel_affine_g(int w, int h)
  321. {
  322. for (int c = 1; c <= 4; c++)
  323. {
  324. ncnn::Mat a0 = RandomMat(w, h, c);
  325. float tm[6];
  326. float tm_inv[6];
  327. ncnn::get_rotation_matrix(0.f, 0.5f, w / 2 + 4, h / 2 - 3, tm);
  328. ncnn::invert_affine_transform(tm, tm_inv);
  329. ncnn::Mat a1(w, h, (size_t)c, c);
  330. ncnn::Mat a2 = a0.clone();
  331. if (c == 1)
  332. {
  333. ncnn::warpaffine_bilinear_c1(a0, w, h, a1, w, h, tm, 0);
  334. ncnn::warpaffine_bilinear_c1(a1, w, h, a2, w, h, tm_inv, -233);
  335. }
  336. if (c == 2)
  337. {
  338. ncnn::warpaffine_bilinear_c2(a0, w, h, a1, w, h, tm, 0);
  339. ncnn::warpaffine_bilinear_c2(a1, w, h, a2, w, h, tm_inv, -233);
  340. }
  341. if (c == 3)
  342. {
  343. ncnn::warpaffine_bilinear_c3(a0, w, h, a1, w, h, tm, 0);
  344. ncnn::warpaffine_bilinear_c3(a1, w, h, a2, w, h, tm_inv, -233);
  345. }
  346. if (c == 4)
  347. {
  348. ncnn::warpaffine_bilinear_c4(a0, w, h, a1, w, h, tm, 0);
  349. ncnn::warpaffine_bilinear_c4(a1, w, h, a2, w, h, tm_inv, -233);
  350. }
  351. if (CompareNearlyEqual(a0, a2) != 0)
  352. {
  353. fprintf(stderr, "test_mat_pixel_affine_g failed w=%d h=%d c=%d\n", w, h, c);
  354. return -1;
  355. }
  356. }
  357. return 0;
  358. }
  359. static int test_mat_pixel_affine_0()
  360. {
  361. return 0
  362. || test_mat_pixel_affine_a(60, 70)
  363. || test_mat_pixel_affine_b(60, 70)
  364. || test_mat_pixel_affine_c(60, 70)
  365. || test_mat_pixel_affine_d(60, 70)
  366. || test_mat_pixel_affine_e(60, 70)
  367. || test_mat_pixel_affine_f(60, 70)
  368. || test_mat_pixel_affine_g(60, 70)
  369. || test_mat_pixel_affine_a(120, 160)
  370. || test_mat_pixel_affine_b(120, 160)
  371. || test_mat_pixel_affine_c(120, 160)
  372. || test_mat_pixel_affine_d(120, 160)
  373. || test_mat_pixel_affine_e(120, 160)
  374. || test_mat_pixel_affine_f(120, 160)
  375. || test_mat_pixel_affine_g(120, 160)
  376. || test_mat_pixel_affine_a(220, 330)
  377. || test_mat_pixel_affine_b(220, 330)
  378. || test_mat_pixel_affine_c(220, 330)
  379. || test_mat_pixel_affine_d(220, 330)
  380. || test_mat_pixel_affine_e(220, 330)
  381. || test_mat_pixel_affine_f(220, 330)
  382. || test_mat_pixel_affine_g(220, 330);
  383. }
  384. static int test_mat_pixel_affine_yuv420sp(int w, int h)
  385. {
  386. ncnn::Mat a0(w, h * 3 / 2, (size_t)1u, 1);
  387. ncnn::Mat a0_y = RandomMat(w, h, 1);
  388. ncnn::Mat a0_uv = RandomMat(w / 2, h / 2, 2);
  389. memcpy(a0, a0_y, w * h);
  390. memcpy((unsigned char*)a0 + w * h, a0_uv, w * h / 2);
  391. float tm[6];
  392. float tm_inv[6];
  393. ncnn::get_rotation_matrix(-70.f, 0.3f, w / 2, h / 2, tm);
  394. ncnn::invert_affine_transform(tm, tm_inv);
  395. ncnn::Mat a1(w / 2, (h / 2) * 3 / 2, (size_t)1u, 1);
  396. ncnn::Mat a2 = a0.clone();
  397. ncnn::warpaffine_bilinear_yuv420sp(a0, w, h, a1, w / 2, h / 2, tm, 0);
  398. ncnn::warpaffine_bilinear_yuv420sp(a1, w / 2, h / 2, a2, w, h, tm_inv, -233);
  399. // Y
  400. if (CompareNearlyEqual(ncnn::Mat(w, h, (unsigned char*)a0, (size_t)1u, 1), ncnn::Mat(w, h, (unsigned char*)a2, (size_t)1u, 1)) != 0)
  401. {
  402. fprintf(stderr, "test_mat_pixel_affine_yuv420sp Y failed w=%d h=%d\n", w, h);
  403. return -1;
  404. }
  405. // UV
  406. if (CompareNearlyEqual(ncnn::Mat(w / 2, h / 2, (unsigned char*)a0 + w * h, (size_t)2u, 2), ncnn::Mat(w / 2, h / 2, (unsigned char*)a2 + w * h, (size_t)2u, 2)) != 0)
  407. {
  408. fprintf(stderr, "test_mat_pixel_affine_yuv420sp UV failed w=%d h=%d\n", w, h);
  409. return -1;
  410. }
  411. return 0;
  412. }
  413. static int test_mat_pixel_affine_1()
  414. {
  415. return 0
  416. || test_mat_pixel_affine_yuv420sp(40, 40)
  417. || test_mat_pixel_affine_yuv420sp(120, 160)
  418. || test_mat_pixel_affine_yuv420sp(220, 340);
  419. }
  420. int main()
  421. {
  422. SRAND(7767517);
  423. return test_mat_pixel_affine_0() || test_mat_pixel_affine_1();
  424. }