infrared_test.c 18 KB

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