infrared_decoder_nec.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. #include "common/infrared_common_i.h"
  2. #include "infrared.h"
  3. #include "infrared_protocol_defs_i.h"
  4. #include <stdbool.h>
  5. #include <stdint.h>
  6. #include <furi.h>
  7. #include "../infrared_i.h"
  8. InfraredMessage* infrared_decoder_nec_check_ready(void* ctx) {
  9. return infrared_common_decoder_check_ready(ctx);
  10. }
  11. bool infrared_decoder_nec_interpret(InfraredCommonDecoder* decoder) {
  12. furi_assert(decoder);
  13. bool result = false;
  14. if(decoder->databit_cnt == 32) {
  15. uint8_t address = decoder->data[0];
  16. uint8_t address_inverse = decoder->data[1];
  17. uint8_t command = decoder->data[2];
  18. uint8_t command_inverse = decoder->data[3];
  19. uint8_t inverse_command_inverse = (uint8_t)~command_inverse;
  20. uint8_t inverse_address_inverse = (uint8_t)~address_inverse;
  21. if((command == inverse_command_inverse) && (address == inverse_address_inverse)) {
  22. decoder->message.protocol = InfraredProtocolNEC;
  23. decoder->message.address = address;
  24. decoder->message.command = command;
  25. decoder->message.repeat = false;
  26. result = true;
  27. } else {
  28. decoder->message.protocol = InfraredProtocolNECext;
  29. decoder->message.address = decoder->data[0] | (decoder->data[1] << 8);
  30. decoder->message.command = decoder->data[2] | (decoder->data[3] << 8);
  31. decoder->message.repeat = false;
  32. result = true;
  33. }
  34. } else if(decoder->databit_cnt == 42) {
  35. uint32_t* data1 = (void*)decoder->data;
  36. uint16_t* data2 = (void*)(data1 + 1);
  37. uint16_t address = *data1 & 0x1FFF;
  38. uint16_t address_inverse = (*data1 >> 13) & 0x1FFF;
  39. uint16_t command = ((*data1 >> 26) & 0x3F) | ((*data2 & 0x3) << 6);
  40. uint16_t command_inverse = (*data2 >> 2) & 0xFF;
  41. if((address == (~address_inverse & 0x1FFF)) && (command == (~command_inverse & 0xFF))) {
  42. decoder->message.protocol = InfraredProtocolNEC42;
  43. decoder->message.address = address;
  44. decoder->message.command = command;
  45. decoder->message.repeat = false;
  46. result = true;
  47. } else {
  48. decoder->message.protocol = InfraredProtocolNEC42ext;
  49. decoder->message.address = address | (address_inverse << 13);
  50. decoder->message.command = command | (command_inverse << 8);
  51. decoder->message.repeat = false;
  52. result = true;
  53. }
  54. }
  55. return result;
  56. }
  57. // timings start from Space (delay between message and repeat)
  58. InfraredStatus infrared_decoder_nec_decode_repeat(InfraredCommonDecoder* decoder) {
  59. furi_assert(decoder);
  60. float preamble_tolerance = decoder->protocol->timings.preamble_tolerance;
  61. uint32_t bit_tolerance = decoder->protocol->timings.bit_tolerance;
  62. InfraredStatus status = InfraredStatusError;
  63. if(decoder->timings_cnt < 4) return InfraredStatusOk;
  64. if((decoder->timings[0] > INFRARED_NEC_REPEAT_PAUSE_MIN) &&
  65. (decoder->timings[0] < INFRARED_NEC_REPEAT_PAUSE_MAX) &&
  66. MATCH_TIMING(decoder->timings[1], INFRARED_NEC_REPEAT_MARK, preamble_tolerance) &&
  67. MATCH_TIMING(decoder->timings[2], INFRARED_NEC_REPEAT_SPACE, preamble_tolerance) &&
  68. MATCH_TIMING(decoder->timings[3], decoder->protocol->timings.bit1_mark, bit_tolerance)) {
  69. status = InfraredStatusReady;
  70. decoder->timings_cnt = 0;
  71. } else {
  72. status = InfraredStatusError;
  73. }
  74. return status;
  75. }
  76. void* infrared_decoder_nec_alloc(void) {
  77. return infrared_common_decoder_alloc(&protocol_nec);
  78. }
  79. InfraredMessage* infrared_decoder_nec_decode(void* decoder, bool level, uint32_t duration) {
  80. return infrared_common_decode(decoder, level, duration);
  81. }
  82. void infrared_decoder_nec_free(void* decoder) {
  83. infrared_common_decoder_free(decoder);
  84. }
  85. void infrared_decoder_nec_reset(void* decoder) {
  86. infrared_common_decoder_reset(decoder);
  87. }