irda_common_encoder.c 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  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 even_timing = !(encoder->timings_encoded % 2);
  30. *level = even_timing ^ logic_value;
  31. *duration = timings->bit1_mark;
  32. if (even_timing) /* start encoding from space */
  33. ++encoder->bits_encoded;
  34. ++encoder->timings_encoded;
  35. encoder->timings_sum += *duration;
  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. bool done = false;
  45. const IrdaTimings* timings = &encoder->protocol->timings;
  46. uint8_t index = encoder->bits_encoded / 8;
  47. uint8_t shift = encoder->bits_encoded % 8; // LSB first
  48. bool logic_value = !!(encoder->data[index] & (0x01 << shift));
  49. // stop bit
  50. if (encoder->bits_encoded == encoder->protocol->databit_len) {
  51. furi_assert(!encoder->protocol->no_stop_bit);
  52. *duration = timings->bit1_mark;
  53. *level = true;
  54. ++encoder->timings_encoded;
  55. encoder->timings_sum += *duration;
  56. return IrdaStatusDone;
  57. }
  58. if (encoder->timings_encoded % 2) { /* start encoding from space */
  59. *duration = logic_value ? timings->bit1_mark : timings->bit0_mark;
  60. *level = true;
  61. } else {
  62. *duration = logic_value ? timings->bit1_space : timings->bit0_space;
  63. *level = false;
  64. ++encoder->bits_encoded;
  65. }
  66. if ((encoder->bits_encoded == encoder->protocol->databit_len)
  67. && encoder->protocol->no_stop_bit) {
  68. done = true;
  69. }
  70. ++encoder->timings_encoded;
  71. encoder->timings_sum += *duration;
  72. return done ? IrdaStatusDone : IrdaStatusOk;
  73. }
  74. IrdaStatus irda_common_encode(IrdaCommonEncoder* encoder, uint32_t* duration, bool* level) {
  75. furi_assert(encoder);
  76. furi_assert(duration);
  77. furi_assert(level);
  78. IrdaStatus status = IrdaStatusOk;
  79. const IrdaTimings* timings = &encoder->protocol->timings;
  80. switch (encoder->state) {
  81. case IrdaCommonEncoderStateSilence:
  82. *duration = encoder->protocol->timings.silence_time;
  83. *level = false;
  84. status = IrdaStatusOk;
  85. encoder->state = IrdaCommonEncoderStatePreamble;
  86. ++encoder->timings_encoded;
  87. encoder->timings_sum = 0;
  88. break;
  89. case IrdaCommonEncoderStatePreamble:
  90. if (timings->preamble_mark) {
  91. if (encoder->timings_encoded == 1) {
  92. *duration = timings->preamble_mark;
  93. *level = true;
  94. } else {
  95. *duration = timings->preamble_space;
  96. *level = false;
  97. encoder->state = IrdaCommonEncoderStateEncode;
  98. }
  99. ++encoder->timings_encoded;
  100. encoder->timings_sum += *duration;
  101. break;
  102. } else {
  103. encoder->state = IrdaCommonEncoderStateEncode;
  104. }
  105. /* FALLTHROUGH */
  106. case IrdaCommonEncoderStateEncode:
  107. status = encoder->protocol->encode(encoder, duration, level);
  108. if (status == IrdaStatusDone) {
  109. if (encoder->protocol->encode_repeat) {
  110. encoder->state = IrdaCommonEncoderStateEncodeRepeat;
  111. } else {
  112. encoder->timings_encoded = 0;
  113. encoder->timings_sum = 0;
  114. encoder->bits_encoded = 0;
  115. encoder->switch_detect = 0;
  116. encoder->state = IrdaCommonEncoderStateSilence;
  117. }
  118. }
  119. break;
  120. case IrdaCommonEncoderStateEncodeRepeat:
  121. status = encoder->protocol->encode_repeat(encoder, duration, level);
  122. break;
  123. }
  124. return status;
  125. }
  126. void* irda_common_encoder_alloc(const IrdaCommonProtocolSpec* protocol) {
  127. furi_assert(protocol);
  128. uint32_t alloc_size = sizeof(IrdaCommonEncoder)
  129. + protocol->databit_len / 8
  130. + !!(protocol->databit_len % 8);
  131. IrdaCommonEncoder* encoder = furi_alloc(alloc_size);
  132. memset(encoder, 0, alloc_size);
  133. encoder->protocol = protocol;
  134. return encoder;
  135. }
  136. void irda_common_encoder_free(IrdaCommonEncoder* encoder) {
  137. furi_assert(encoder);
  138. free(encoder);
  139. }
  140. void irda_common_encoder_reset(IrdaCommonEncoder* encoder) {
  141. furi_assert(encoder);
  142. encoder->timings_encoded = 0;
  143. encoder->timings_sum = 0;
  144. encoder->bits_encoded = 0;
  145. encoder->state = IrdaCommonEncoderStateSilence;
  146. encoder->switch_detect = 0;
  147. uint8_t bytes_to_clear = encoder->protocol->databit_len / 8
  148. + !!(encoder->protocol->databit_len % 8);
  149. memset(encoder->data, 0, bytes_to_clear);
  150. }