irda_common_encoder.c 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. #include "furi/check.h"
  2. #include "irda.h"
  3. #include "irda_common_i.h"
  4. #include <stdbool.h>
  5. #include <furi.h>
  6. #include "irda_i.h"
  7. /*
  8. *
  9. * 3:
  10. * even_timing = 0
  11. * level = 0 ^ 1 = 1
  12. * 4:
  13. * even_timing = 1
  14. * level = 1 ^ 1 = 0
  15. * ++timing;
  16. *
  17. *
  18. * 0 1 2 | 3 4 |
  19. * _____-------_____---___
  20. */
  21. IrdaStatus irda_common_encode_manchester(IrdaCommonEncoder* encoder, uint32_t* duration, bool* level) {
  22. furi_assert(encoder);
  23. furi_assert(duration);
  24. furi_assert(level);
  25. const IrdaTimings* timings = &encoder->protocol->timings;
  26. uint8_t index = encoder->bits_encoded / 8;
  27. uint8_t shift = encoder->bits_encoded % 8; // LSB first
  28. bool logic_value = !!(encoder->data[index] & (0x01 << shift));
  29. bool inverse = encoder->protocol->manchester_inverse_level;
  30. bool even_timing = !(encoder->timings_encoded % 2);
  31. *level = even_timing ^ logic_value ^ inverse;
  32. *duration = timings->bit1_mark;
  33. if (even_timing) /* start encoding from space */
  34. ++encoder->bits_encoded;
  35. ++encoder->timings_encoded;
  36. bool finish = (encoder->bits_encoded == encoder->protocol->databit_len);
  37. finish |= (encoder->bits_encoded == (encoder->protocol->databit_len-1)) && *level && !even_timing;
  38. return finish ? IrdaStatusDone : IrdaStatusOk;
  39. }
  40. IrdaStatus irda_common_encode_pdwm(IrdaCommonEncoder* encoder, uint32_t* duration, bool* level) {
  41. furi_assert(encoder);
  42. furi_assert(duration);
  43. furi_assert(level);
  44. const IrdaTimings* timings = &encoder->protocol->timings;
  45. uint8_t index = encoder->bits_encoded / 8;
  46. uint8_t shift = encoder->bits_encoded % 8; // LSB first
  47. bool logic_value = !!(encoder->data[index] & (0x01 << shift));
  48. // stop bit
  49. if (encoder->bits_encoded == encoder->protocol->databit_len) {
  50. *duration = timings->bit1_mark;
  51. *level = true;
  52. ++encoder->timings_encoded;
  53. return IrdaStatusDone;
  54. }
  55. if (encoder->timings_encoded % 2) { /* start encoding from space */
  56. *duration = logic_value ? timings->bit1_mark : timings->bit0_mark;
  57. *level = true;
  58. } else {
  59. *duration = logic_value ? timings->bit1_space : timings->bit0_space;
  60. *level = false;
  61. ++encoder->bits_encoded;
  62. }
  63. ++encoder->timings_encoded;
  64. return IrdaStatusOk;
  65. }
  66. IrdaStatus irda_common_encode(IrdaCommonEncoder* encoder, uint32_t* duration, bool* level) {
  67. furi_assert(encoder);
  68. furi_assert(duration);
  69. furi_assert(level);
  70. IrdaStatus status = IrdaStatusOk;
  71. const IrdaTimings* timings = &encoder->protocol->timings;
  72. switch (encoder->state) {
  73. case IrdaCommonEncoderStateSpace:
  74. *duration = encoder->protocol->timings.silence_time;
  75. *level = false;
  76. status = IrdaStatusOk;
  77. encoder->state = IrdaCommonEncoderStatePreamble;
  78. ++encoder->timings_encoded;
  79. break;
  80. case IrdaCommonEncoderStatePreamble:
  81. if (timings->preamble_mark) {
  82. if (encoder->timings_encoded == 1) {
  83. *duration = timings->preamble_mark;
  84. *level = true;
  85. } else {
  86. *duration = timings->preamble_space;
  87. *level = false;
  88. encoder->state = IrdaCommonEncoderStateEncode;
  89. }
  90. ++encoder->timings_encoded;
  91. break;
  92. } else {
  93. encoder->state = IrdaCommonEncoderStateEncode;
  94. }
  95. /* FALLTHROUGH */
  96. case IrdaCommonEncoderStateEncode:
  97. status = encoder->protocol->encode(encoder, duration, level);
  98. if (status == IrdaStatusDone) {
  99. if (encoder->protocol->encode_repeat) {
  100. encoder->state = IrdaCommonEncoderStateEncodeRepeat;
  101. } else {
  102. encoder->timings_encoded = 0;
  103. encoder->bits_encoded = 0;
  104. encoder->switch_detect = 0;
  105. encoder->state = IrdaCommonEncoderStateSpace;
  106. }
  107. }
  108. break;
  109. case IrdaCommonEncoderStateEncodeRepeat:
  110. status = encoder->protocol->encode_repeat(encoder, duration, level);
  111. break;
  112. }
  113. return status;
  114. }
  115. void* irda_common_encoder_alloc(const IrdaCommonProtocolSpec* protocol) {
  116. furi_assert(protocol);
  117. uint32_t alloc_size = sizeof(IrdaCommonEncoder)
  118. + protocol->databit_len / 8
  119. + !!(protocol->databit_len % 8);
  120. IrdaCommonEncoder* encoder = furi_alloc(alloc_size);
  121. memset(encoder, 0, alloc_size);
  122. encoder->protocol = protocol;
  123. return encoder;
  124. }
  125. void irda_common_encoder_free(IrdaCommonEncoder* encoder) {
  126. furi_assert(encoder);
  127. free(encoder);
  128. }
  129. void irda_common_encoder_reset(IrdaCommonEncoder* encoder) {
  130. furi_assert(encoder);
  131. encoder->timings_encoded = 0;
  132. encoder->bits_encoded = 0;
  133. encoder->state = IrdaCommonEncoderStateSpace;
  134. encoder->switch_detect = 0;
  135. uint8_t bytes_to_clear = encoder->protocol->databit_len / 8
  136. + !!(encoder->protocol->databit_len % 8);
  137. memset(encoder->data, 0, bytes_to_clear);
  138. }
  139. void irda_common_encoder_set_context(void* decoder, void* context) {
  140. IrdaCommonEncoder* common_encoder = decoder;
  141. common_encoder->context = context;
  142. }