irda_decoder_encoder_test.c 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. #include <furi.h>
  2. #include "../minunit.h"
  3. #include "irda.h"
  4. #include "common/irda_common_i.h"
  5. #include "test_data/irda_nec_test_data.srcdata"
  6. #include "test_data/irda_necext_test_data.srcdata"
  7. #include "test_data/irda_samsung_test_data.srcdata"
  8. #include "test_data/irda_rc6_test_data.srcdata"
  9. #include "test_data/irda_rc5_test_data.srcdata"
  10. #define RUN_ENCODER(data, expected) \
  11. run_encoder((data), COUNT_OF(data), (expected), COUNT_OF(expected))
  12. #define RUN_DECODER(data, expected) \
  13. run_decoder((data), COUNT_OF(data), (expected), COUNT_OF(expected))
  14. static IrdaDecoderHandler* decoder_handler;
  15. static IrdaEncoderHandler* encoder_handler;
  16. static void test_setup(void) {
  17. decoder_handler = irda_alloc_decoder();
  18. encoder_handler = irda_alloc_encoder();
  19. }
  20. static void test_teardown(void) {
  21. irda_free_decoder(decoder_handler);
  22. irda_free_encoder(encoder_handler);
  23. }
  24. static void compare_message_results(
  25. const IrdaMessage* message_decoded,
  26. const IrdaMessage* message_expected) {
  27. mu_check(message_decoded->protocol == message_expected->protocol);
  28. mu_check(message_decoded->command == message_expected->command);
  29. mu_check(message_decoded->address == message_expected->address);
  30. mu_check(message_decoded->repeat == message_expected->repeat);
  31. }
  32. static void
  33. run_encoder_fill_array(IrdaEncoderHandler* handler, uint32_t* timings, uint32_t* timings_len) {
  34. uint32_t duration = 0;
  35. bool level = false; // start from space
  36. bool level_read;
  37. IrdaStatus status = IrdaStatusError;
  38. int i = 0;
  39. while(1) {
  40. status = irda_encode(handler, &duration, &level_read);
  41. if(level_read != level) {
  42. level = level_read;
  43. ++i;
  44. }
  45. timings[i] += duration;
  46. furi_assert((status == IrdaStatusOk) || (status == IrdaStatusDone));
  47. if(status == IrdaStatusDone) break;
  48. furi_assert(i < *timings_len);
  49. }
  50. *timings_len = i + 1;
  51. }
  52. // messages in input array for encoder should have one protocol
  53. static void run_encoder(
  54. const IrdaMessage input_messages[],
  55. uint32_t input_messages_len,
  56. const uint32_t expected_timings[],
  57. uint32_t expected_timings_len) {
  58. uint32_t* timings = 0;
  59. uint32_t timings_len = 0;
  60. uint32_t j = 0;
  61. for(uint32_t message_counter = 0; message_counter < input_messages_len; ++message_counter) {
  62. const IrdaMessage* message = &input_messages[message_counter];
  63. if(!message->repeat) {
  64. irda_reset_encoder(encoder_handler, message);
  65. }
  66. timings_len = 200;
  67. timings = furi_alloc(sizeof(uint32_t) * timings_len);
  68. run_encoder_fill_array(encoder_handler, timings, &timings_len);
  69. furi_assert(timings_len <= 200);
  70. for(int i = 0; i < timings_len; ++i, ++j) {
  71. mu_check(MATCH_BIT_TIMING(timings[i], expected_timings[j], 120));
  72. mu_assert(j < expected_timings_len, "encoded more timings than expected");
  73. }
  74. free(timings);
  75. }
  76. mu_assert(j == expected_timings_len, "encoded less timings than expected");
  77. }
  78. static void run_encoder_decoder(const IrdaMessage input_messages[], uint32_t input_messages_len) {
  79. uint32_t* timings = 0;
  80. uint32_t timings_len = 0;
  81. bool level = false;
  82. for(uint32_t message_counter = 0; message_counter < input_messages_len; ++message_counter) {
  83. const IrdaMessage* message_encoded = &input_messages[message_counter];
  84. if(!message_encoded->repeat) {
  85. irda_reset_encoder(encoder_handler, message_encoded);
  86. level = false;
  87. }
  88. timings_len = 200;
  89. timings = furi_alloc(sizeof(uint32_t) * timings_len);
  90. run_encoder_fill_array(encoder_handler, timings, &timings_len);
  91. furi_assert(timings_len <= 200);
  92. const IrdaMessage* message_decoded = 0;
  93. for(int i = 0; i < timings_len; ++i) {
  94. message_decoded = irda_decode(decoder_handler, level, timings[i]);
  95. if(i < timings_len - 1)
  96. mu_check(!message_decoded);
  97. else
  98. mu_check(message_decoded);
  99. level = !level;
  100. }
  101. if(message_decoded) {
  102. compare_message_results(message_decoded, message_encoded);
  103. } else {
  104. mu_check(0);
  105. }
  106. free(timings);
  107. }
  108. }
  109. static void run_decoder(
  110. const uint32_t* input_delays,
  111. uint32_t input_delays_len,
  112. const IrdaMessage* message_expected,
  113. uint32_t message_expected_len) {
  114. const IrdaMessage* message_decoded = 0;
  115. bool level = 0;
  116. uint32_t message_counter = 0;
  117. for(uint32_t i = 0; i < input_delays_len; ++i) {
  118. message_decoded = irda_decode(decoder_handler, level, input_delays[i]);
  119. if(message_decoded) {
  120. mu_assert(message_counter < message_expected_len, "decoded more than expected");
  121. if(message_counter >= message_expected_len) break;
  122. compare_message_results(message_decoded, &message_expected[message_counter]);
  123. ++message_counter;
  124. }
  125. level = !level;
  126. }
  127. mu_assert(message_counter == message_expected_len, "decoded less than expected");
  128. }
  129. MU_TEST(test_decoder_samsung32) {
  130. RUN_DECODER(test_decoder_samsung32_input1, test_decoder_samsung32_expected1);
  131. }
  132. MU_TEST(test_mix) {
  133. RUN_DECODER(test_decoder_rc5_input2, test_decoder_rc5_expected2);
  134. RUN_DECODER(test_decoder_necext_input1, test_decoder_necext_expected1);
  135. // can use encoder data for decoding, but can't do opposite
  136. RUN_DECODER(test_encoder_rc6_expected1, test_encoder_rc6_input1);
  137. RUN_DECODER(test_decoder_samsung32_input1, test_decoder_samsung32_expected1);
  138. RUN_DECODER(test_decoder_rc6_input1, test_decoder_rc6_expected1);
  139. RUN_DECODER(test_decoder_samsung32_input1, test_decoder_samsung32_expected1);
  140. RUN_DECODER(test_decoder_rc5_input1, test_decoder_rc5_expected1);
  141. RUN_DECODER(test_decoder_necext_input1, test_decoder_necext_expected1);
  142. RUN_DECODER(test_decoder_nec_input2, test_decoder_nec_expected2);
  143. RUN_DECODER(test_decoder_rc6_input1, test_decoder_rc6_expected1);
  144. RUN_DECODER(test_decoder_necext_input1, test_decoder_necext_expected1);
  145. RUN_DECODER(test_decoder_rc5_input5, test_decoder_rc5_expected5);
  146. RUN_DECODER(test_decoder_samsung32_input1, test_decoder_samsung32_expected1);
  147. }
  148. MU_TEST(test_decoder_nec1) {
  149. RUN_DECODER(test_decoder_nec_input1, test_decoder_nec_expected1);
  150. }
  151. MU_TEST(test_decoder_nec2) {
  152. RUN_DECODER(test_decoder_nec_input2, test_decoder_nec_expected2);
  153. }
  154. MU_TEST(test_decoder_unexpected_end_in_sequence) {
  155. // test_decoder_nec_input1 and test_decoder_nec_input2 shuts unexpected
  156. RUN_DECODER(test_decoder_nec_input1, test_decoder_nec_expected1);
  157. RUN_DECODER(test_decoder_nec_input1, test_decoder_nec_expected1);
  158. RUN_DECODER(test_decoder_nec_input2, test_decoder_nec_expected2);
  159. RUN_DECODER(test_decoder_nec_input2, test_decoder_nec_expected2);
  160. }
  161. MU_TEST(test_decoder_necext1) {
  162. RUN_DECODER(test_decoder_necext_input1, test_decoder_necext_expected1);
  163. RUN_DECODER(test_decoder_necext_input1, test_decoder_necext_expected1);
  164. }
  165. MU_TEST(test_decoder_rc5) {
  166. RUN_DECODER(test_decoder_rc5x_input1, test_decoder_rc5x_expected1);
  167. RUN_DECODER(test_decoder_rc5_input1, test_decoder_rc5_expected1);
  168. RUN_DECODER(test_decoder_rc5_input2, test_decoder_rc5_expected2);
  169. RUN_DECODER(test_decoder_rc5_input3, test_decoder_rc5_expected3);
  170. RUN_DECODER(test_decoder_rc5_input4, test_decoder_rc5_expected4);
  171. RUN_DECODER(test_decoder_rc5_input5, test_decoder_rc5_expected5);
  172. RUN_DECODER(test_decoder_rc5_input6, test_decoder_rc5_expected6);
  173. RUN_DECODER(test_decoder_rc5_input_all_repeats, test_decoder_rc5_expected_all_repeats);
  174. }
  175. MU_TEST(test_encoder_rc5x) {
  176. RUN_ENCODER(test_decoder_rc5x_expected1, test_decoder_rc5x_input1);
  177. }
  178. MU_TEST(test_encoder_rc5) {
  179. RUN_ENCODER(test_decoder_rc5_expected_all_repeats, test_decoder_rc5_input_all_repeats);
  180. }
  181. MU_TEST(test_decoder_rc6) {
  182. RUN_DECODER(test_decoder_rc6_input1, test_decoder_rc6_expected1);
  183. }
  184. MU_TEST(test_encoder_rc6) {
  185. RUN_ENCODER(test_encoder_rc6_input1, test_encoder_rc6_expected1);
  186. }
  187. MU_TEST(test_encoder_decoder_all) {
  188. run_encoder_decoder(test_nec_all, COUNT_OF(test_nec_all));
  189. run_encoder_decoder(test_necext_all, COUNT_OF(test_necext_all));
  190. run_encoder_decoder(test_samsung32_all, COUNT_OF(test_samsung32_all));
  191. run_encoder_decoder(test_rc6_all, COUNT_OF(test_rc6_all));
  192. run_encoder_decoder(test_rc5_all, COUNT_OF(test_rc5_all));
  193. }
  194. MU_TEST_SUITE(test_irda_decoder_encoder) {
  195. MU_SUITE_CONFIGURE(&test_setup, &test_teardown);
  196. MU_RUN_TEST(test_encoder_rc5x);
  197. MU_RUN_TEST(test_encoder_rc5);
  198. MU_RUN_TEST(test_decoder_rc5);
  199. MU_RUN_TEST(test_decoder_rc6);
  200. MU_RUN_TEST(test_encoder_rc6);
  201. MU_RUN_TEST(test_decoder_unexpected_end_in_sequence);
  202. MU_RUN_TEST(test_decoder_nec1);
  203. MU_RUN_TEST(test_decoder_nec2);
  204. MU_RUN_TEST(test_decoder_samsung32);
  205. MU_RUN_TEST(test_decoder_necext1);
  206. MU_RUN_TEST(test_mix);
  207. MU_RUN_TEST(test_encoder_decoder_all);
  208. }
  209. int run_minunit_test_irda_decoder_encoder() {
  210. MU_RUN_SUITE(test_irda_decoder_encoder);
  211. MU_REPORT();
  212. return MU_EXIT_CODE;
  213. }