menu.cpp 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. #include "stm32g4xx_hal.h"
  2. #include "cmsis_os.h"
  3. #include "menu_tree.h"
  4. #include "control.h"
  5. #include "uart_bridge.h"
  6. #include <stdio.h>
  7. #define DBG if(0)
  8. #define MON_DELAY 100
  9. osMessageQId menu_event;
  10. osMutexId menu_mutex;
  11. osThreadId menu_update_handle;
  12. osThreadId menu_control_handle;
  13. menuItem* selectedMenuItem;
  14. bool update_flag = false;
  15. static void vUpdate(void const *params);
  16. static osStatus menu_block(uint32_t ms);
  17. static void menu_unblock(void);
  18. //
  19. void menu_init(void)
  20. {
  21. // Очередь для управляющих сигналов меню
  22. osMessageQDef(vButEventQueue, 1, uint8_t);
  23. menu_event = osMessageCreate(osMessageQ(vButEventQueue), NULL);
  24. // Mutex для блокировки меню
  25. osMutexDef(menu_mut);
  26. menu_mutex = osMutexCreate(osMutex(menu_mut));
  27. // События. Сигналы энкодера
  28. osThreadDef(control_thr, vControl, osPriorityNormal, 0, 2*configMINIMAL_STACK_SIZE);
  29. menu_control_handle = osThreadCreate(osThread(control_thr), NULL);
  30. // Обновление меню
  31. osThreadDef(menu_up_thr, vUpdate, osPriorityNormal, 0, 4*configMINIMAL_STACK_SIZE);
  32. menu_update_handle = osThreadCreate(osThread(menu_up_thr), NULL);
  33. // Начальный экран
  34. selectedMenuItem = (menuItem *) &m_main;
  35. (*(selectedMenuItem->func_draw))(selectedMenuItem->Select);
  36. }
  37. //
  38. void menu_send_event(control_t but)
  39. {
  40. osMessagePut(menu_event, but, 0);
  41. }
  42. //
  43. static void vControl(void const *params)
  44. {
  45. osEvent event;
  46. for (;;)
  47. {
  48. event = osMessageGet(menu_event, 100);
  49. if (event.status != osEventMessage)
  50. continue;
  51. switch (event.value.v)
  52. {
  53. case CONTROL_FWD :
  54. DBG printf("Encoder forward\r\n");
  55. if (menu_block(0) == osOK) {
  56. button_right();
  57. menu_unblock();
  58. }
  59. break;
  60. case CONTROL_BACK :
  61. DBG printf("Encoder back\r\n");
  62. if (menu_block(0) == osOK) {
  63. button_left();
  64. menu_unblock();
  65. }
  66. break;
  67. case CONTROL_PUSH :
  68. DBG printf("Encoder push");
  69. if (menu_block(0) == osOK) {
  70. menuFocus();
  71. menu_unblock();
  72. }
  73. break;
  74. }
  75. }
  76. }
  77. //
  78. static void vUpdate(void const *params)
  79. {
  80. for (;;)
  81. {
  82. osDelay(MON_DELAY);
  83. #if 0
  84. // Имитация изменения счетчиков для теста
  85. inc_raw_counters();
  86. #endif
  87. // Текущий экран меню нужно перерисовать
  88. if (update_flag)
  89. {
  90. (*(selectedMenuItem->func_draw))(selectedMenuItem->Select);
  91. }
  92. }
  93. }
  94. //
  95. static osStatus menu_block(uint32_t ms)
  96. {
  97. return osMutexWait(menu_mutex, ms);
  98. }
  99. //
  100. static void menu_unblock(void)
  101. {
  102. osMutexRelease(menu_mutex);
  103. }