GPIO_reader.c 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. #include <furi.h>
  2. #include <gui/gui.h>
  3. #include <input/input.h>
  4. #include <stdlib.h>
  5. #include "GPIO_reader_item.h"
  6. typedef enum {
  7. EventTypeTick,
  8. EventTypeKey,
  9. } EventType;
  10. typedef struct {
  11. EventType type;
  12. InputEvent input;
  13. } PluginEvent;
  14. typedef struct {
  15. int pin;
  16. int pullMode;
  17. FuriMutex* mutex;
  18. } PluginState;
  19. static void render_callback(Canvas* const canvas, void* ctx) {
  20. const PluginState* plugin_state = ctx;
  21. furi_mutex_acquire(plugin_state->mutex, FuriWaitForever);
  22. canvas_set_font(canvas, FontPrimary);
  23. canvas_draw_str_aligned(
  24. canvas,
  25. canvas_width(canvas) / 2,
  26. canvas_height(canvas) / 10,
  27. AlignCenter,
  28. AlignCenter,
  29. "GPIO reader");
  30. canvas_set_font(canvas, FontSecondary);
  31. canvas_draw_str_aligned(
  32. canvas,
  33. canvas_width(canvas) / 2,
  34. canvas_height(canvas) / 10 * 3,
  35. AlignCenter,
  36. AlignCenter,
  37. gpio_item_get_pin_name(plugin_state->pin));
  38. canvas_draw_str_aligned(
  39. canvas,
  40. canvas_width(canvas) / 2,
  41. canvas_height(canvas) / 10 * 5,
  42. AlignCenter,
  43. AlignCenter,
  44. gpio_item_get_pull_mode(plugin_state->pullMode));
  45. canvas_set_font(canvas, FontPrimary);
  46. canvas_draw_str_aligned(
  47. canvas,
  48. canvas_width(canvas) / 2,
  49. canvas_height(canvas) / 10 * 8,
  50. AlignCenter,
  51. AlignCenter,
  52. gpio_item_get_pin_level(plugin_state->pin));
  53. furi_mutex_release(plugin_state->mutex);
  54. }
  55. static void input_callback(InputEvent* input_event, FuriMessageQueue* event_queue) {
  56. furi_assert(event_queue);
  57. PluginEvent event = {.type = EventTypeKey, .input = *input_event};
  58. furi_message_queue_put(event_queue, &event, FuriWaitForever);
  59. }
  60. static void GPIO_reader_state_init(PluginState* const plugin_state) {
  61. plugin_state->pin = 0;
  62. plugin_state->pullMode = 0;
  63. gpio_item_configure_pin(plugin_state->pin, plugin_state->pullMode);
  64. }
  65. int32_t GPIO_reader_app(void* p) {
  66. UNUSED(p);
  67. FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
  68. PluginState* plugin_state = malloc(sizeof(PluginState));
  69. GPIO_reader_state_init(plugin_state);
  70. plugin_state->mutex = furi_mutex_alloc(FuriMutexTypeNormal);
  71. if(!plugin_state->mutex) {
  72. FURI_LOG_E("GPIO_reader", "cannot create mutex\r\n");
  73. free(plugin_state);
  74. return 255;
  75. }
  76. // Set system callbacks
  77. ViewPort* view_port = view_port_alloc();
  78. view_port_draw_callback_set(view_port, render_callback, plugin_state);
  79. view_port_input_callback_set(view_port, input_callback, event_queue);
  80. // Open GUI and register view_port
  81. Gui* gui = furi_record_open("gui");
  82. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  83. PluginEvent event;
  84. for(bool processing = true; processing;) {
  85. FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
  86. furi_mutex_acquire(plugin_state->mutex, FuriWaitForever);
  87. if(event_status == FuriStatusOk) {
  88. // press events
  89. if(event.type == EventTypeKey) {
  90. if(event.input.type == InputTypePress || event.input.type == InputTypeRepeat) {
  91. switch(event.input.key) {
  92. case InputKeyRight:
  93. plugin_state->pin = (plugin_state->pin + 1) % GPIO_ITEM_COUNT;
  94. gpio_item_configure_pin(plugin_state->pin, plugin_state->pullMode);
  95. break;
  96. case InputKeyLeft:
  97. plugin_state->pin =
  98. (plugin_state->pin - 1 + GPIO_ITEM_COUNT) % GPIO_ITEM_COUNT;
  99. gpio_item_configure_pin(plugin_state->pin, plugin_state->pullMode);
  100. break;
  101. case InputKeyUp:
  102. plugin_state->pullMode = (plugin_state->pullMode + 1) % GPIO_PULL_COUNT;
  103. gpio_item_configure_pin(plugin_state->pin, plugin_state->pullMode);
  104. break;
  105. case InputKeyDown:
  106. plugin_state->pullMode =
  107. (plugin_state->pullMode - 1 + GPIO_PULL_COUNT) % GPIO_PULL_COUNT;
  108. gpio_item_configure_pin(plugin_state->pin, plugin_state->pullMode);
  109. break;
  110. case InputKeyBack:
  111. processing = false;
  112. break;
  113. default:
  114. break;
  115. }
  116. }
  117. }
  118. // } else {
  119. // FURI_LOG_D("GPIO_reader", "FuriMessageQueue: event timeout");
  120. // event timeout
  121. }
  122. view_port_update(view_port);
  123. furi_mutex_release(plugin_state->mutex);
  124. }
  125. view_port_enabled_set(view_port, false);
  126. gui_remove_view_port(gui, view_port);
  127. furi_record_close("gui");
  128. view_port_free(view_port);
  129. furi_mutex_free(plugin_state->mutex);
  130. furi_message_queue_free(event_queue);
  131. free(plugin_state);
  132. return 0;
  133. }