irda.c 7.5 KB

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