irda.c 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. #include "irda.h"
  2. #include "furi/check.h"
  3. #include "common/irda_common_i.h"
  4. #include "irda_protocol_defs_i.h"
  5. #include <stdbool.h>
  6. #include <stdint.h>
  7. #include <stdlib.h>
  8. #include <furi.h>
  9. #include "irda_i.h"
  10. #include <furi-hal-irda.h>
  11. typedef struct {
  12. IrdaAlloc alloc;
  13. IrdaDecode decode;
  14. IrdaDecoderReset reset;
  15. IrdaFree free;
  16. IrdaDecoderCheckReady check_ready;
  17. } IrdaDecoders;
  18. typedef struct {
  19. IrdaAlloc alloc;
  20. IrdaEncode encode;
  21. IrdaEncoderReset reset;
  22. IrdaFree free;
  23. } IrdaEncoders;
  24. struct IrdaDecoderHandler {
  25. void** ctx;
  26. };
  27. struct IrdaEncoderHandler {
  28. void* handler;
  29. const IrdaEncoders* encoder;
  30. };
  31. typedef struct {
  32. IrdaEncoders encoder;
  33. IrdaDecoders decoder;
  34. IrdaGetProtocolSpec get_protocol_spec;
  35. } IrdaEncoderDecoder;
  36. static const IrdaEncoderDecoder irda_encoder_decoder[] = {
  37. {
  38. .decoder = {
  39. .alloc = irda_decoder_nec_alloc,
  40. .decode = irda_decoder_nec_decode,
  41. .reset = irda_decoder_nec_reset,
  42. .free = irda_decoder_nec_free},
  43. .encoder = {
  44. .alloc = irda_encoder_nec_alloc,
  45. .encode = irda_encoder_nec_encode,
  46. .reset = irda_encoder_nec_reset,
  47. .free = irda_encoder_nec_free},
  48. .get_protocol_spec = irda_nec_get_spec,
  49. },
  50. {
  51. .decoder = {
  52. .alloc = irda_decoder_samsung32_alloc,
  53. .decode = irda_decoder_samsung32_decode,
  54. .reset = irda_decoder_samsung32_reset,
  55. .free = irda_decoder_samsung32_free},
  56. .encoder = {
  57. .alloc = irda_encoder_samsung32_alloc,
  58. .encode = irda_encoder_samsung32_encode,
  59. .reset = irda_encoder_samsung32_reset,
  60. .free = irda_encoder_samsung32_free},
  61. .get_protocol_spec = irda_samsung32_get_spec,
  62. },
  63. {
  64. .decoder = {
  65. .alloc = irda_decoder_rc5_alloc,
  66. .decode = irda_decoder_rc5_decode,
  67. .reset = irda_decoder_rc5_reset,
  68. .free = irda_decoder_rc5_free},
  69. .encoder = {
  70. .alloc = irda_encoder_rc5_alloc,
  71. .encode = irda_encoder_rc5_encode,
  72. .reset = irda_encoder_rc5_reset,
  73. .free = irda_encoder_rc5_free},
  74. .get_protocol_spec = irda_rc5_get_spec,
  75. },
  76. {
  77. .decoder = {
  78. .alloc = irda_decoder_rc6_alloc,
  79. .decode = irda_decoder_rc6_decode,
  80. .reset = irda_decoder_rc6_reset,
  81. .free = irda_decoder_rc6_free},
  82. .encoder = {
  83. .alloc = irda_encoder_rc6_alloc,
  84. .encode = irda_encoder_rc6_encode,
  85. .reset = irda_encoder_rc6_reset,
  86. .free = irda_encoder_rc6_free},
  87. .get_protocol_spec = irda_rc6_get_spec,
  88. },
  89. {
  90. .decoder = {
  91. .alloc = irda_decoder_sirc_alloc,
  92. .decode = irda_decoder_sirc_decode,
  93. .reset = irda_decoder_sirc_reset,
  94. .check_ready = irda_decoder_sirc_check_ready,
  95. .free = irda_decoder_sirc_free},
  96. .encoder = {
  97. .alloc = irda_encoder_sirc_alloc,
  98. .encode = irda_encoder_sirc_encode,
  99. .reset = irda_encoder_sirc_reset,
  100. .free = irda_encoder_sirc_free},
  101. .get_protocol_spec = irda_sirc_get_spec,
  102. },
  103. };
  104. static int irda_find_index_by_protocol(IrdaProtocol protocol);
  105. static const IrdaProtocolSpecification* irda_get_spec_by_protocol(IrdaProtocol protocol);
  106. const IrdaMessage* irda_decode(IrdaDecoderHandler* handler, bool level, uint32_t duration) {
  107. furi_assert(handler);
  108. IrdaMessage* message = NULL;
  109. IrdaMessage* result = NULL;
  110. for (int i = 0; i < COUNT_OF(irda_encoder_decoder); ++i) {
  111. if (irda_encoder_decoder[i].decoder.decode) {
  112. message = irda_encoder_decoder[i].decoder.decode(handler->ctx[i], level, duration);
  113. if (!result && message) {
  114. result = message;
  115. }
  116. }
  117. }
  118. return result;
  119. }
  120. IrdaDecoderHandler* irda_alloc_decoder(void) {
  121. IrdaDecoderHandler* handler = furi_alloc(sizeof(IrdaDecoderHandler));
  122. handler->ctx = furi_alloc(sizeof(void*) * COUNT_OF(irda_encoder_decoder));
  123. for (int i = 0; i < COUNT_OF(irda_encoder_decoder); ++i) {
  124. handler->ctx[i] = 0;
  125. if (irda_encoder_decoder[i].decoder.alloc)
  126. handler->ctx[i] = irda_encoder_decoder[i].decoder.alloc();
  127. }
  128. irda_reset_decoder(handler);
  129. return handler;
  130. }
  131. void irda_free_decoder(IrdaDecoderHandler* handler) {
  132. furi_assert(handler);
  133. furi_assert(handler->ctx);
  134. for (int i = 0; i < COUNT_OF(irda_encoder_decoder); ++i) {
  135. if (irda_encoder_decoder[i].decoder.free)
  136. irda_encoder_decoder[i].decoder.free(handler->ctx[i]);
  137. }
  138. free(handler->ctx);
  139. free(handler);
  140. }
  141. void irda_reset_decoder(IrdaDecoderHandler* handler) {
  142. for (int i = 0; i < COUNT_OF(irda_encoder_decoder); ++i) {
  143. if (irda_encoder_decoder[i].decoder.reset)
  144. irda_encoder_decoder[i].decoder.reset(handler->ctx[i]);
  145. }
  146. }
  147. const IrdaMessage* irda_check_decoder_ready(IrdaDecoderHandler* handler) {
  148. furi_assert(handler);
  149. IrdaMessage* message = NULL;
  150. IrdaMessage* result = NULL;
  151. for (int i = 0; i < COUNT_OF(irda_encoder_decoder); ++i) {
  152. if (irda_encoder_decoder[i].decoder.check_ready) {
  153. message = irda_encoder_decoder[i].decoder.check_ready(handler->ctx[i]);
  154. if (!result && message) {
  155. result = message;
  156. }
  157. }
  158. }
  159. return result;
  160. }
  161. IrdaEncoderHandler* irda_alloc_encoder(void) {
  162. IrdaEncoderHandler* handler = furi_alloc(sizeof(IrdaEncoderHandler));
  163. handler->handler = NULL;
  164. handler->encoder = NULL;
  165. return handler;
  166. }
  167. void irda_free_encoder(IrdaEncoderHandler* handler) {
  168. furi_assert(handler);
  169. const IrdaEncoders* encoder = handler->encoder;
  170. if (encoder || handler->handler) {
  171. furi_assert(encoder);
  172. furi_assert(handler->handler);
  173. furi_assert(encoder->free);
  174. encoder->free(handler->handler);
  175. }
  176. free(handler);
  177. }
  178. static int irda_find_index_by_protocol(IrdaProtocol protocol) {
  179. for (int i = 0; i < COUNT_OF(irda_encoder_decoder); ++i) {
  180. if (irda_encoder_decoder[i].get_protocol_spec(protocol)) {
  181. return i;
  182. }
  183. }
  184. return -1;
  185. }
  186. void irda_reset_encoder(IrdaEncoderHandler* handler, const IrdaMessage* message) {
  187. furi_assert(handler);
  188. furi_assert(message);
  189. int index = irda_find_index_by_protocol(message->protocol);
  190. furi_check(index >= 0);
  191. const IrdaEncoders* required_encoder = &irda_encoder_decoder[index].encoder;
  192. furi_assert(required_encoder);
  193. furi_assert(required_encoder->reset);
  194. furi_assert(required_encoder->alloc);
  195. /* Realloc encoder if different protocol set */
  196. if (required_encoder != handler->encoder) {
  197. if (handler->handler != NULL) {
  198. furi_assert(handler->encoder->free);
  199. handler->encoder->free(handler->handler);
  200. }
  201. handler->encoder = required_encoder;
  202. handler->handler = handler->encoder->alloc();
  203. }
  204. handler->encoder->reset(handler->handler, message);
  205. }
  206. IrdaStatus irda_encode(IrdaEncoderHandler* handler, uint32_t* duration, bool* level) {
  207. furi_assert(handler);
  208. furi_assert(duration);
  209. furi_assert(level);
  210. const IrdaEncoders* encoder = handler->encoder;
  211. furi_assert(encoder);
  212. furi_assert(encoder->encode);
  213. IrdaStatus status = encoder->encode(handler->handler, duration, level);
  214. furi_assert(status != IrdaStatusError);
  215. return status;
  216. }
  217. bool irda_is_protocol_valid(IrdaProtocol protocol) {
  218. return irda_find_index_by_protocol(protocol) >= 0;
  219. }
  220. IrdaProtocol irda_get_protocol_by_name(const char* protocol_name) {
  221. for (IrdaProtocol protocol = 0; protocol < IrdaProtocolMAX; ++protocol) {
  222. const char* name = irda_get_protocol_name(protocol);
  223. if (!strcmp(name, protocol_name))
  224. return protocol;
  225. }
  226. return IrdaProtocolUnknown;
  227. }
  228. static const IrdaProtocolSpecification* irda_get_spec_by_protocol(IrdaProtocol protocol) {
  229. int index = irda_find_index_by_protocol(protocol);
  230. furi_check(index >= 0);
  231. const IrdaProtocolSpecification* spec = irda_encoder_decoder[index].get_protocol_spec(protocol);
  232. furi_assert(spec);
  233. return spec;
  234. }
  235. const char* irda_get_protocol_name(IrdaProtocol protocol) {
  236. return irda_get_spec_by_protocol(protocol)->name;
  237. }
  238. uint8_t irda_get_protocol_address_length(IrdaProtocol protocol) {
  239. return irda_get_spec_by_protocol(protocol)->address_length;
  240. }
  241. uint8_t irda_get_protocol_command_length(IrdaProtocol protocol) {
  242. return irda_get_spec_by_protocol(protocol)->command_length;
  243. }
  244. uint32_t irda_get_protocol_frequency(IrdaProtocol protocol) {
  245. return irda_get_spec_by_protocol(protocol)->frequency;
  246. }
  247. float irda_get_protocol_duty_cycle(IrdaProtocol protocol) {
  248. return irda_get_spec_by_protocol(protocol)->duty_cycle;
  249. }