infrared_test.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521
  1. #include <furi.h>
  2. #include <flipper_format.h>
  3. #include <infrared.h>
  4. #include <common/infrared_common_i.h>
  5. #include "../minunit.h"
  6. #define IR_TEST_FILES_DIR EXT_PATH("unit_tests/infrared/")
  7. #define IR_TEST_FILE_PREFIX "test_"
  8. #define IR_TEST_FILE_SUFFIX ".irtest"
  9. typedef struct {
  10. InfraredDecoderHandler* decoder_handler;
  11. InfraredEncoderHandler* encoder_handler;
  12. string_t file_path;
  13. FlipperFormat* ff;
  14. } InfraredTest;
  15. static InfraredTest* test;
  16. static void infrared_test_alloc() {
  17. Storage* storage = furi_record_open(RECORD_STORAGE);
  18. test = malloc(sizeof(InfraredTest));
  19. test->decoder_handler = infrared_alloc_decoder();
  20. test->encoder_handler = infrared_alloc_encoder();
  21. test->ff = flipper_format_buffered_file_alloc(storage);
  22. string_init(test->file_path);
  23. }
  24. static void infrared_test_free() {
  25. furi_assert(test);
  26. infrared_free_decoder(test->decoder_handler);
  27. infrared_free_encoder(test->encoder_handler);
  28. flipper_format_free(test->ff);
  29. string_clear(test->file_path);
  30. furi_record_close(RECORD_STORAGE);
  31. free(test);
  32. test = NULL;
  33. }
  34. static bool infrared_test_prepare_file(const char* protocol_name) {
  35. string_t file_type;
  36. string_init(file_type);
  37. bool success = false;
  38. string_printf(
  39. test->file_path,
  40. "%s%s%s%s",
  41. IR_TEST_FILES_DIR,
  42. IR_TEST_FILE_PREFIX,
  43. protocol_name,
  44. IR_TEST_FILE_SUFFIX);
  45. do {
  46. uint32_t format_version;
  47. if(!flipper_format_buffered_file_open_existing(test->ff, string_get_cstr(test->file_path)))
  48. break;
  49. if(!flipper_format_read_header(test->ff, file_type, &format_version)) break;
  50. if(string_cmp_str(file_type, "IR tests file") || format_version != 1) break;
  51. success = true;
  52. } while(false);
  53. string_clear(file_type);
  54. return success;
  55. }
  56. static bool infrared_test_load_raw_signal(
  57. FlipperFormat* ff,
  58. const char* signal_name,
  59. uint32_t** timings,
  60. uint32_t* timings_count) {
  61. string_t buf;
  62. string_init(buf);
  63. bool success = false;
  64. do {
  65. bool is_name_found = false;
  66. for(; !is_name_found && flipper_format_read_string(ff, "name", buf);
  67. is_name_found = !string_cmp_str(buf, signal_name))
  68. ;
  69. if(!is_name_found) break;
  70. if(!flipper_format_read_string(ff, "type", buf) || string_cmp_str(buf, "raw")) break;
  71. if(!flipper_format_get_value_count(ff, "data", timings_count)) break;
  72. if(!*timings_count) break;
  73. *timings = malloc(*timings_count * sizeof(uint32_t*));
  74. if(!flipper_format_read_uint32(ff, "data", *timings, *timings_count)) {
  75. free(*timings);
  76. break;
  77. }
  78. success = true;
  79. } while(false);
  80. string_clear(buf);
  81. return success;
  82. }
  83. static bool infrared_test_read_message(FlipperFormat* ff, InfraredMessage* message) {
  84. string_t buf;
  85. string_init(buf);
  86. bool success = false;
  87. do {
  88. if(!flipper_format_read_string(ff, "protocol", buf)) break;
  89. message->protocol = infrared_get_protocol_by_name(string_get_cstr(buf));
  90. if(!infrared_is_protocol_valid(message->protocol)) break;
  91. if(!flipper_format_read_hex(ff, "address", (uint8_t*)&message->address, sizeof(uint32_t)))
  92. break;
  93. if(!flipper_format_read_hex(ff, "command", (uint8_t*)&message->command, sizeof(uint32_t)))
  94. break;
  95. if(!flipper_format_read_bool(ff, "repeat", &message->repeat, 1)) break;
  96. success = true;
  97. } while(false);
  98. string_clear(buf);
  99. return success;
  100. }
  101. static bool infrared_test_load_messages(
  102. FlipperFormat* ff,
  103. const char* signal_name,
  104. InfraredMessage** messages,
  105. uint32_t* messages_count) {
  106. string_t buf;
  107. string_init(buf);
  108. bool success = false;
  109. do {
  110. bool is_name_found = false;
  111. for(; !is_name_found && flipper_format_read_string(ff, "name", buf);
  112. is_name_found = !string_cmp_str(buf, signal_name))
  113. ;
  114. if(!is_name_found) break;
  115. if(!flipper_format_read_string(ff, "type", buf) || string_cmp_str(buf, "parsed_array"))
  116. break;
  117. if(!flipper_format_read_uint32(ff, "count", messages_count, 1)) break;
  118. if(!*messages_count) break;
  119. *messages = malloc(*messages_count * sizeof(InfraredMessage));
  120. uint32_t i;
  121. for(i = 0; i < *messages_count; ++i) {
  122. if(!infrared_test_read_message(ff, (*messages) + i)) {
  123. break;
  124. }
  125. }
  126. if(*messages_count != i) {
  127. free(*messages);
  128. break;
  129. }
  130. success = true;
  131. } while(false);
  132. string_clear(buf);
  133. return success;
  134. }
  135. static void infrared_test_compare_message_results(
  136. const InfraredMessage* message_decoded,
  137. const InfraredMessage* message_expected) {
  138. mu_check(message_decoded->protocol == message_expected->protocol);
  139. mu_check(message_decoded->command == message_expected->command);
  140. mu_check(message_decoded->address == message_expected->address);
  141. if((message_expected->protocol == InfraredProtocolSIRC) ||
  142. (message_expected->protocol == InfraredProtocolSIRC15) ||
  143. (message_expected->protocol == InfraredProtocolSIRC20)) {
  144. mu_check(message_decoded->repeat == false);
  145. } else {
  146. mu_check(message_decoded->repeat == message_expected->repeat);
  147. }
  148. }
  149. /* Encodes signal and merges same levels (high+high, low+low) */
  150. static void infrared_test_run_encoder_fill_array(
  151. InfraredEncoderHandler* handler,
  152. uint32_t* timings,
  153. uint32_t* timings_len,
  154. bool* start_level) {
  155. uint32_t duration = 0;
  156. bool level = false;
  157. bool level_read;
  158. InfraredStatus status = InfraredStatusError;
  159. size_t i = 0;
  160. bool first = true;
  161. while(1) {
  162. status = infrared_encode(handler, &duration, &level_read);
  163. if(first) {
  164. if(start_level) *start_level = level_read;
  165. first = false;
  166. timings[0] = 0;
  167. } else if(level_read != level) {
  168. ++i;
  169. furi_check(i < *timings_len);
  170. timings[i] = 0;
  171. }
  172. level = level_read;
  173. timings[i] += duration;
  174. furi_check((status == InfraredStatusOk) || (status == InfraredStatusDone));
  175. if(status == InfraredStatusDone) break;
  176. }
  177. *timings_len = i + 1;
  178. }
  179. // messages in input array for encoder should have one protocol
  180. static void infrared_test_run_encoder(InfraredProtocol protocol, uint32_t test_index) {
  181. uint32_t* timings;
  182. uint32_t timings_count = 200;
  183. uint32_t* expected_timings;
  184. uint32_t expected_timings_count;
  185. InfraredMessage* input_messages;
  186. uint32_t input_messages_count;
  187. string_t buf;
  188. string_init(buf);
  189. const char* protocol_name = infrared_get_protocol_name(protocol);
  190. mu_assert(infrared_test_prepare_file(protocol_name), "Failed to prepare test file");
  191. string_printf(buf, "encoder_input%d", test_index);
  192. mu_assert(
  193. infrared_test_load_messages(
  194. test->ff, string_get_cstr(buf), &input_messages, &input_messages_count),
  195. "Failed to load messages from file");
  196. string_printf(buf, "encoder_expected%d", test_index);
  197. mu_assert(
  198. infrared_test_load_raw_signal(
  199. test->ff, string_get_cstr(buf), &expected_timings, &expected_timings_count),
  200. "Failed to load raw signal from file");
  201. flipper_format_buffered_file_close(test->ff);
  202. string_clear(buf);
  203. uint32_t j = 0;
  204. timings = malloc(sizeof(uint32_t) * timings_count);
  205. for(uint32_t message_counter = 0; message_counter < input_messages_count; ++message_counter) {
  206. const InfraredMessage* message = &input_messages[message_counter];
  207. if(!message->repeat) {
  208. infrared_reset_encoder(test->encoder_handler, message);
  209. }
  210. timings_count = 200;
  211. infrared_test_run_encoder_fill_array(test->encoder_handler, timings, &timings_count, NULL);
  212. furi_check(timings_count <= 200);
  213. for(size_t i = 0; i < timings_count; ++i, ++j) {
  214. mu_check(MATCH_TIMING(timings[i], expected_timings[j], 120));
  215. mu_assert(j < expected_timings_count, "encoded more timings than expected");
  216. }
  217. }
  218. free(input_messages);
  219. free(expected_timings);
  220. free(timings);
  221. mu_assert(j == expected_timings_count, "encoded less timings than expected");
  222. }
  223. static void infrared_test_run_encoder_decoder(InfraredProtocol protocol, uint32_t test_index) {
  224. uint32_t* timings = 0;
  225. uint32_t timings_count = 200;
  226. InfraredMessage* input_messages;
  227. uint32_t input_messages_count;
  228. bool level = false;
  229. string_t buf;
  230. string_init(buf);
  231. timings = malloc(sizeof(uint32_t) * timings_count);
  232. const char* protocol_name = infrared_get_protocol_name(protocol);
  233. mu_assert(infrared_test_prepare_file(protocol_name), "Failed to prepare test file");
  234. string_printf(buf, "encoder_decoder_input%d", test_index);
  235. mu_assert(
  236. infrared_test_load_messages(
  237. test->ff, string_get_cstr(buf), &input_messages, &input_messages_count),
  238. "Failed to load messages from file");
  239. flipper_format_buffered_file_close(test->ff);
  240. string_clear(buf);
  241. for(uint32_t message_counter = 0; message_counter < input_messages_count; ++message_counter) {
  242. const InfraredMessage* message_encoded = &input_messages[message_counter];
  243. if(!message_encoded->repeat) {
  244. infrared_reset_encoder(test->encoder_handler, message_encoded);
  245. }
  246. timings_count = 200;
  247. infrared_test_run_encoder_fill_array(
  248. test->encoder_handler, timings, &timings_count, &level);
  249. furi_check(timings_count <= 200);
  250. const InfraredMessage* message_decoded = 0;
  251. for(size_t i = 0; i < timings_count; ++i) {
  252. message_decoded = infrared_decode(test->decoder_handler, level, timings[i]);
  253. if((i == timings_count - 2) && level && message_decoded) {
  254. /* In case we end with space timing - message can be decoded at last mark */
  255. break;
  256. } else if(i < timings_count - 1) {
  257. mu_check(!message_decoded);
  258. } else {
  259. if(!message_decoded) {
  260. message_decoded = infrared_check_decoder_ready(test->decoder_handler);
  261. }
  262. mu_check(message_decoded);
  263. }
  264. level = !level;
  265. }
  266. if(message_decoded) {
  267. infrared_test_compare_message_results(message_decoded, message_encoded);
  268. } else {
  269. mu_check(0);
  270. }
  271. }
  272. free(input_messages);
  273. free(timings);
  274. }
  275. static void infrared_test_run_decoder(InfraredProtocol protocol, uint32_t test_index) {
  276. uint32_t* timings;
  277. uint32_t timings_count;
  278. InfraredMessage* messages;
  279. uint32_t messages_count;
  280. string_t buf;
  281. string_init(buf);
  282. mu_assert(
  283. infrared_test_prepare_file(infrared_get_protocol_name(protocol)),
  284. "Failed to prepare test file");
  285. string_printf(buf, "decoder_input%d", test_index);
  286. mu_assert(
  287. infrared_test_load_raw_signal(test->ff, string_get_cstr(buf), &timings, &timings_count),
  288. "Failed to load raw signal from file");
  289. string_printf(buf, "decoder_expected%d", test_index);
  290. mu_assert(
  291. infrared_test_load_messages(test->ff, string_get_cstr(buf), &messages, &messages_count),
  292. "Failed to load messages from file");
  293. flipper_format_buffered_file_close(test->ff);
  294. string_clear(buf);
  295. InfraredMessage message_decoded_check_local;
  296. bool level = 0;
  297. uint32_t message_counter = 0;
  298. const InfraredMessage* message_decoded = 0;
  299. for(uint32_t i = 0; i < timings_count; ++i) {
  300. const InfraredMessage* message_decoded_check = 0;
  301. if(timings[i] > INFRARED_RAW_RX_TIMING_DELAY_US) {
  302. message_decoded_check = infrared_check_decoder_ready(test->decoder_handler);
  303. if(message_decoded_check) {
  304. /* infrared_decode() can reset message, but we have to call infrared_decode() to perform real
  305. * simulation: infrared_check() by timeout, then infrared_decode() when meet edge */
  306. message_decoded_check_local = *message_decoded_check;
  307. message_decoded_check = &message_decoded_check_local;
  308. }
  309. }
  310. message_decoded = infrared_decode(test->decoder_handler, level, timings[i]);
  311. if(message_decoded_check || message_decoded) {
  312. mu_assert(
  313. !(message_decoded_check && message_decoded),
  314. "both messages decoded: check_ready() and infrared_decode()");
  315. if(message_decoded_check) {
  316. message_decoded = message_decoded_check;
  317. }
  318. mu_assert(message_counter < messages_count, "decoded more than expected");
  319. infrared_test_compare_message_results(message_decoded, &messages[message_counter]);
  320. ++message_counter;
  321. }
  322. level = !level;
  323. }
  324. message_decoded = infrared_check_decoder_ready(test->decoder_handler);
  325. if(message_decoded) {
  326. infrared_test_compare_message_results(message_decoded, &messages[message_counter]);
  327. ++message_counter;
  328. }
  329. free(timings);
  330. free(messages);
  331. mu_assert(message_counter == messages_count, "decoded less than expected");
  332. }
  333. MU_TEST(infrared_test_decoder_samsung32) {
  334. infrared_test_run_decoder(InfraredProtocolSamsung32, 1);
  335. }
  336. MU_TEST(infrared_test_decoder_mixed) {
  337. infrared_test_run_decoder(InfraredProtocolRC5, 2);
  338. infrared_test_run_decoder(InfraredProtocolSIRC, 1);
  339. infrared_test_run_decoder(InfraredProtocolNECext, 1);
  340. infrared_test_run_decoder(InfraredProtocolRC6, 2);
  341. infrared_test_run_decoder(InfraredProtocolSamsung32, 1);
  342. infrared_test_run_decoder(InfraredProtocolRC6, 1);
  343. infrared_test_run_decoder(InfraredProtocolSamsung32, 1);
  344. infrared_test_run_decoder(InfraredProtocolRC5, 1);
  345. infrared_test_run_decoder(InfraredProtocolSIRC, 2);
  346. infrared_test_run_decoder(InfraredProtocolNECext, 1);
  347. infrared_test_run_decoder(InfraredProtocolSIRC, 4);
  348. infrared_test_run_decoder(InfraredProtocolNEC, 2);
  349. infrared_test_run_decoder(InfraredProtocolRC6, 1);
  350. infrared_test_run_decoder(InfraredProtocolNECext, 1);
  351. infrared_test_run_decoder(InfraredProtocolSIRC, 5);
  352. infrared_test_run_decoder(InfraredProtocolNEC, 3);
  353. infrared_test_run_decoder(InfraredProtocolRC5, 5);
  354. infrared_test_run_decoder(InfraredProtocolSamsung32, 1);
  355. infrared_test_run_decoder(InfraredProtocolSIRC, 3);
  356. }
  357. MU_TEST(infrared_test_decoder_nec) {
  358. infrared_test_run_decoder(InfraredProtocolNEC, 1);
  359. infrared_test_run_decoder(InfraredProtocolNEC, 2);
  360. infrared_test_run_decoder(InfraredProtocolNEC, 3);
  361. }
  362. MU_TEST(infrared_test_decoder_unexpected_end_in_sequence) {
  363. infrared_test_run_decoder(InfraredProtocolNEC, 1);
  364. infrared_test_run_decoder(InfraredProtocolNEC, 1);
  365. infrared_test_run_decoder(InfraredProtocolNEC, 2);
  366. infrared_test_run_decoder(InfraredProtocolNEC, 2);
  367. }
  368. MU_TEST(infrared_test_decoder_necext1) {
  369. infrared_test_run_decoder(InfraredProtocolNECext, 1);
  370. infrared_test_run_decoder(InfraredProtocolNECext, 1);
  371. }
  372. MU_TEST(infrared_test_decoder_long_packets_with_nec_start) {
  373. infrared_test_run_decoder(InfraredProtocolNEC42ext, 1);
  374. infrared_test_run_decoder(InfraredProtocolNEC42ext, 2);
  375. }
  376. MU_TEST(infrared_test_encoder_sirc) {
  377. infrared_test_run_encoder(InfraredProtocolSIRC, 1);
  378. infrared_test_run_encoder(InfraredProtocolSIRC, 2);
  379. }
  380. MU_TEST(infrared_test_decoder_sirc) {
  381. infrared_test_run_decoder(InfraredProtocolSIRC, 3);
  382. infrared_test_run_decoder(InfraredProtocolSIRC, 1);
  383. infrared_test_run_decoder(InfraredProtocolSIRC, 2);
  384. infrared_test_run_decoder(InfraredProtocolSIRC, 4);
  385. infrared_test_run_decoder(InfraredProtocolSIRC, 5);
  386. }
  387. MU_TEST(infrared_test_decoder_rc5) {
  388. infrared_test_run_decoder(InfraredProtocolRC5X, 1);
  389. infrared_test_run_decoder(InfraredProtocolRC5, 1);
  390. infrared_test_run_decoder(InfraredProtocolRC5, 2);
  391. infrared_test_run_decoder(InfraredProtocolRC5, 3);
  392. infrared_test_run_decoder(InfraredProtocolRC5, 4);
  393. infrared_test_run_decoder(InfraredProtocolRC5, 5);
  394. infrared_test_run_decoder(InfraredProtocolRC5, 6);
  395. infrared_test_run_decoder(InfraredProtocolRC5, 7);
  396. }
  397. MU_TEST(infrared_test_encoder_rc5x) {
  398. infrared_test_run_encoder(InfraredProtocolRC5X, 1);
  399. }
  400. MU_TEST(infrared_test_encoder_rc5) {
  401. infrared_test_run_encoder(InfraredProtocolRC5, 1);
  402. }
  403. MU_TEST(infrared_test_decoder_rc6) {
  404. infrared_test_run_decoder(InfraredProtocolRC6, 1);
  405. }
  406. MU_TEST(infrared_test_encoder_rc6) {
  407. infrared_test_run_encoder(InfraredProtocolRC6, 1);
  408. }
  409. MU_TEST(infrared_test_encoder_decoder_all) {
  410. infrared_test_run_encoder_decoder(InfraredProtocolNEC, 1);
  411. infrared_test_run_encoder_decoder(InfraredProtocolNECext, 1);
  412. infrared_test_run_encoder_decoder(InfraredProtocolNEC42, 1);
  413. infrared_test_run_encoder_decoder(InfraredProtocolNEC42ext, 1);
  414. infrared_test_run_encoder_decoder(InfraredProtocolSamsung32, 1);
  415. infrared_test_run_encoder_decoder(InfraredProtocolRC6, 1);
  416. infrared_test_run_encoder_decoder(InfraredProtocolRC5, 1);
  417. infrared_test_run_encoder_decoder(InfraredProtocolSIRC, 1);
  418. }
  419. MU_TEST_SUITE(infrared_test) {
  420. MU_SUITE_CONFIGURE(&infrared_test_alloc, &infrared_test_free);
  421. MU_RUN_TEST(infrared_test_encoder_sirc);
  422. MU_RUN_TEST(infrared_test_decoder_sirc);
  423. MU_RUN_TEST(infrared_test_encoder_rc5x);
  424. MU_RUN_TEST(infrared_test_encoder_rc5);
  425. MU_RUN_TEST(infrared_test_decoder_rc5);
  426. MU_RUN_TEST(infrared_test_decoder_rc6);
  427. MU_RUN_TEST(infrared_test_encoder_rc6);
  428. MU_RUN_TEST(infrared_test_decoder_unexpected_end_in_sequence);
  429. MU_RUN_TEST(infrared_test_decoder_long_packets_with_nec_start);
  430. MU_RUN_TEST(infrared_test_decoder_nec);
  431. MU_RUN_TEST(infrared_test_decoder_samsung32);
  432. MU_RUN_TEST(infrared_test_decoder_necext1);
  433. MU_RUN_TEST(infrared_test_decoder_mixed);
  434. MU_RUN_TEST(infrared_test_encoder_decoder_all);
  435. }
  436. int run_minunit_test_infrared() {
  437. MU_RUN_SUITE(infrared_test);
  438. return MU_EXIT_CODE;
  439. }