irda_monitor.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. #include "gui/canvas.h"
  2. #include "irda.h"
  3. #include <stdio.h>
  4. #include <furi.h>
  5. #include <api-hal-irda.h>
  6. #include <api-hal.h>
  7. #include <notification/notification-messages.h>
  8. #include <gui/view_port.h>
  9. #include <gui/gui.h>
  10. #include <gui/elements.h>
  11. #define IRDA_TIMINGS_SIZE 700
  12. typedef struct {
  13. uint32_t timing_cnt;
  14. struct {
  15. uint8_t level;
  16. uint32_t duration;
  17. } timing[IRDA_TIMINGS_SIZE];
  18. } IrdaDelaysArray;
  19. typedef struct {
  20. IrdaDecoderHandler* handler;
  21. char display_text[64];
  22. osMessageQueueId_t event_queue;
  23. IrdaDelaysArray delays;
  24. } IrdaMonitor;
  25. static void irda_rx_callback(void* ctx, bool level, uint32_t duration) {
  26. IrdaMonitor* irda_monitor = (IrdaMonitor*)ctx;
  27. IrdaDelaysArray* delays = &irda_monitor->delays;
  28. if(delays->timing_cnt > 1) furi_assert(level != delays->timing[delays->timing_cnt - 1].level);
  29. delays->timing[delays->timing_cnt].level = level;
  30. delays->timing[delays->timing_cnt].duration = duration;
  31. delays->timing_cnt++; // Read-Modify-Write in ISR only: no need to add synchronization
  32. if(delays->timing_cnt >= IRDA_TIMINGS_SIZE) {
  33. delays->timing_cnt = 0;
  34. }
  35. }
  36. void irda_monitor_input_callback(InputEvent* input_event, void* ctx) {
  37. furi_assert(ctx);
  38. IrdaMonitor* irda_monitor = (IrdaMonitor*)ctx;
  39. if((input_event->type == InputTypeShort) && (input_event->key == InputKeyBack)) {
  40. osMessageQueuePut(irda_monitor->event_queue, input_event, 0, 0);
  41. }
  42. }
  43. static void irda_monitor_draw_callback(Canvas* canvas, void* ctx) {
  44. furi_assert(canvas);
  45. furi_assert(ctx);
  46. IrdaMonitor* irda_monitor = (IrdaMonitor*)ctx;
  47. canvas_clear(canvas);
  48. canvas_set_font(canvas, FontPrimary);
  49. elements_multiline_text_aligned(canvas, 64, 0, AlignCenter, AlignTop, "IRDA monitor\n");
  50. canvas_set_font(canvas, FontKeyboard);
  51. if(strlen(irda_monitor->display_text)) {
  52. elements_multiline_text_aligned(
  53. canvas, 64, 43, AlignCenter, AlignCenter, irda_monitor->display_text);
  54. }
  55. }
  56. int32_t irda_monitor_app(void* p) {
  57. (void)p;
  58. uint32_t counter = 0;
  59. uint32_t print_counter = 0;
  60. IrdaMonitor* irda_monitor = furi_alloc(sizeof(IrdaMonitor));
  61. irda_monitor->display_text[0] = 0;
  62. irda_monitor->event_queue = osMessageQueueNew(1, sizeof(InputEvent), NULL);
  63. ViewPort* view_port = view_port_alloc();
  64. IrdaDelaysArray* delays = &irda_monitor->delays;
  65. NotificationApp* notification = furi_record_open("notification");
  66. Gui* gui = furi_record_open("gui");
  67. view_port_draw_callback_set(view_port, irda_monitor_draw_callback, irda_monitor);
  68. view_port_input_callback_set(view_port, irda_monitor_input_callback, irda_monitor);
  69. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  70. api_hal_irda_rx_irq_init();
  71. api_hal_irda_rx_irq_set_callback(irda_rx_callback, irda_monitor);
  72. irda_monitor->handler = irda_alloc_decoder();
  73. while(1) {
  74. InputEvent event;
  75. if(osOK == osMessageQueueGet(irda_monitor->event_queue, &event, NULL, 50)) {
  76. if((event.type == InputTypeShort) && (event.key == InputKeyBack)) {
  77. break;
  78. }
  79. }
  80. if(counter != delays->timing_cnt) {
  81. notification_message(notification, &sequence_blink_blue_10);
  82. }
  83. for(; counter != delays->timing_cnt;) {
  84. const IrdaMessage* message = irda_decode(
  85. irda_monitor->handler,
  86. delays->timing[counter].level,
  87. delays->timing[counter].duration);
  88. ++counter;
  89. if(counter >= IRDA_TIMINGS_SIZE) counter = 0;
  90. if(message) {
  91. snprintf(
  92. irda_monitor->display_text,
  93. sizeof(irda_monitor->display_text),
  94. "%s\nA:0x%0*lX\nC:0x%0*lX\n%s\n",
  95. irda_get_protocol_name(message->protocol),
  96. irda_get_protocol_address_length(message->protocol),
  97. message->address,
  98. irda_get_protocol_command_length(message->protocol),
  99. message->command,
  100. message->repeat ? " R" : "");
  101. view_port_update(view_port);
  102. }
  103. size_t distance = (counter > print_counter) ?
  104. counter - print_counter :
  105. (counter + IRDA_TIMINGS_SIZE) - print_counter;
  106. if(message || (distance > (IRDA_TIMINGS_SIZE / 2))) {
  107. if(message) {
  108. printf(
  109. "== %s, A:0x%0*lX, C:0x%0*lX%s ==\r\n",
  110. irda_get_protocol_name(message->protocol),
  111. irda_get_protocol_address_length(message->protocol),
  112. message->address,
  113. irda_get_protocol_command_length(message->protocol),
  114. message->command,
  115. message->repeat ? " R" : "");
  116. } else {
  117. printf("== unknown data ==\r\n");
  118. snprintf(
  119. irda_monitor->display_text,
  120. sizeof(irda_monitor->display_text),
  121. "unknown data");
  122. view_port_update(view_port);
  123. }
  124. printf("{");
  125. while(print_counter != counter) {
  126. printf("%lu, ", delays->timing[print_counter].duration);
  127. ++print_counter;
  128. if(print_counter >= IRDA_TIMINGS_SIZE) {
  129. print_counter = 0;
  130. }
  131. }
  132. printf("\r\n};\r\n");
  133. }
  134. }
  135. }
  136. api_hal_irda_rx_irq_deinit();
  137. irda_free_decoder(irda_monitor->handler);
  138. osMessageQueueDelete(irda_monitor->event_queue);
  139. view_port_enabled_set(view_port, false);
  140. gui_remove_view_port(gui, view_port);
  141. view_port_free(view_port);
  142. furi_record_close("notification");
  143. furi_record_close("gui");
  144. free(irda_monitor);
  145. return 0;
  146. }