app.c 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. #include <furi.h>
  2. #include <furi_hal.h>
  3. #include <furi_hal_pwm.h>
  4. #include <input/input.h>
  5. #include <gui/gui.h>
  6. #include <stdlib.h>
  7. #include <stm32wbxx_ll_tim.h>
  8. #include <stm32wbxx_ll_lptim.h>
  9. #include <stm32wbxx_ll_rcc.h>
  10. // ./fbt.cmd launch_app APPSRC=servotester
  11. #define SCREEN_XRES 128
  12. #define SCREEN_YRES 64
  13. #define MIN_ANGLE 0
  14. #define MAX_ANGLE 180
  15. #define FREQ 50
  16. typedef enum Mode {
  17. Manual = 1,
  18. Center = 2,
  19. Sweep = 3,
  20. } Mode;
  21. typedef struct App {
  22. Gui* gui;
  23. ViewPort* view_port;
  24. FuriMessageQueue* event_queue;
  25. int running;
  26. Mode mode;
  27. uint8_t angle;
  28. FuriHalPwmOutputId ch_prev;
  29. FuriHalPwmOutputId ch;
  30. } App;
  31. void input_callback(InputEvent* input_event, void* ctx) {
  32. App* app = ctx;
  33. furi_message_queue_put(app->event_queue, input_event, FuriWaitForever);
  34. }
  35. uint32_t angle_to_compare(uint8_t angle) {
  36. /*
  37. 1%: freq_div: 1280000, prescaler: 19, period: 64000, compare: 640
  38. 3%: freq_div: 1280000, prescaler: 19, period: 64000, compare: 1920
  39. 5%: freq_div: 1280000, prescaler: 19, period: 64000, compare: 3200
  40. 7.5%: freq_div: 1280000, prescaler: 19, period: 64000, compare: 4800
  41. 7%: freq_div: 1280000, prescaler: 19, period: 64000, compare: 4480
  42. 10%: freq_div: 1280000, prescaler: 19, period: 64000, compare: 6400
  43. 13%: freq_div: 1280000, prescaler: 19, period: 64000, compare: 8320
  44. 15%: freq_div: 1280000, prescaler: 19, period: 64000, compare: 9600
  45. 20%: freq_div: 1280000, prescaler: 19, period: 64000, compare: 12800
  46. */
  47. uint32_t min_compare = 1920;
  48. uint32_t max_compare = 8320;
  49. if(angle == MIN_ANGLE) return min_compare;
  50. if(angle == MAX_ANGLE) return max_compare;
  51. return min_compare + floor(((float)angle / (float)MAX_ANGLE) * (max_compare - min_compare));
  52. }
  53. void render_callback(Canvas* const canvas, void* ctx) {
  54. App* app = ctx;
  55. canvas_set_color(canvas, ColorWhite);
  56. canvas_draw_box(canvas, 0, 0, SCREEN_XRES - 1, SCREEN_YRES - 1);
  57. canvas_set_color(canvas, ColorBlack);
  58. canvas_set_font(canvas, FontSecondary);
  59. uint8_t mode_base_x = 20;
  60. canvas_draw_str(canvas, mode_base_x, 10, "Mode");
  61. canvas_draw_str(canvas, mode_base_x, 30, "Manual");
  62. canvas_draw_str(canvas, mode_base_x, 45, "Center");
  63. canvas_draw_str(canvas, mode_base_x, 60, "Sweep");
  64. uint8_t selector_base_x = 5;
  65. uint8_t r = 4;
  66. uint8_t correction = r;
  67. if(app->mode == Manual) {
  68. canvas_draw_disc(canvas, selector_base_x, 30 - correction, r);
  69. } else {
  70. canvas_draw_circle(canvas, selector_base_x, 30 - correction, r);
  71. }
  72. if(app->mode == Center) {
  73. canvas_draw_disc(canvas, selector_base_x, 45 - correction, r);
  74. } else {
  75. canvas_draw_circle(canvas, selector_base_x, 45 - correction, r);
  76. }
  77. if(app->mode == Sweep) {
  78. canvas_draw_disc(canvas, selector_base_x, 60 - correction, r);
  79. } else {
  80. canvas_draw_circle(canvas, selector_base_x, 60 - correction, r);
  81. }
  82. uint8_t angle_base_x = 70;
  83. canvas_draw_str(canvas, angle_base_x, 10, "Angle");
  84. canvas_draw_circle(canvas, angle_base_x + 18, 21, 2);
  85. char angle_str[32];
  86. snprintf(angle_str, sizeof(angle_str), "%3u", app->angle);
  87. canvas_draw_str(canvas, angle_base_x, 30, angle_str);
  88. }
  89. App* app_alloc() {
  90. App* app = malloc(sizeof(App));
  91. app->gui = furi_record_open(RECORD_GUI);
  92. app->view_port = view_port_alloc();
  93. view_port_draw_callback_set(app->view_port, render_callback, app);
  94. view_port_input_callback_set(app->view_port, input_callback, app);
  95. gui_add_view_port(app->gui, app->view_port, GuiLayerFullscreen);
  96. app->event_queue = furi_message_queue_alloc(8, sizeof(InputEvent));
  97. app->running = 1;
  98. app->mode = Manual;
  99. app->ch = FuriHalPwmOutputIdTim1PA7; // A7
  100. app->angle = 0;
  101. return app;
  102. }
  103. void app_free(App* app) {
  104. furi_assert(app);
  105. furi_hal_pwm_stop(app->ch);
  106. view_port_enabled_set(app->view_port, false);
  107. gui_remove_view_port(app->gui, app->view_port);
  108. view_port_free(app->view_port);
  109. furi_record_close(RECORD_GUI);
  110. furi_message_queue_free(app->event_queue);
  111. app->gui = NULL;
  112. free(app);
  113. }
  114. void update_state(App* app, InputKey key) {
  115. if(key == InputKeyUp) {
  116. if(app->mode == Manual) {
  117. app->angle = MIN_ANGLE;
  118. app->mode = Sweep;
  119. } else if(app->mode == Center) {
  120. app->mode = Manual;
  121. } else if(app->mode == Sweep) {
  122. app->mode = Center;
  123. }
  124. }
  125. if(key == InputKeyDown) {
  126. if(app->mode == Manual) {
  127. app->mode = Center;
  128. } else if(app->mode == Center) {
  129. app->angle = MIN_ANGLE;
  130. app->mode = Sweep;
  131. } else if(app->mode == Sweep) {
  132. app->mode = Manual;
  133. }
  134. }
  135. if(app->mode == Manual) {
  136. if(key == InputKeyRight) {
  137. if(app->angle < MAX_ANGLE) {
  138. app->angle = app->angle + 10;
  139. }
  140. }
  141. if(key == InputKeyLeft) {
  142. if(app->angle > MIN_ANGLE) {
  143. app->angle = app->angle - 10;
  144. }
  145. }
  146. }
  147. }
  148. void custom_pwm_set_params(uint32_t freq, uint32_t compare) {
  149. furi_assert(freq > 0);
  150. uint32_t freq_div = 64000000LU / freq;
  151. uint32_t prescaler = freq_div / 0x10000LU;
  152. uint32_t period = freq_div / (prescaler + 1);
  153. // uint32_t compare = period * duty / 100;
  154. LL_TIM_SetPrescaler(TIM1, prescaler);
  155. LL_TIM_SetAutoReload(TIM1, period - 1);
  156. LL_TIM_OC_SetCompareCH1(TIM1, compare);
  157. }
  158. void tick(void* ctx) {
  159. App* app = ctx;
  160. if(!app->running) return;
  161. if(app->mode != Sweep) return;
  162. if(app->angle == MAX_ANGLE) {
  163. app->angle = MIN_ANGLE;
  164. } else if(app->angle == MIN_ANGLE) {
  165. app->angle = MAX_ANGLE;
  166. }
  167. custom_pwm_set_params(FREQ, angle_to_compare(app->angle));
  168. }
  169. int32_t servotester_app_entry(void* p) {
  170. UNUSED(p);
  171. App* app = app_alloc();
  172. furi_hal_pwm_start(app->ch, FREQ, angle_to_compare(app->angle));
  173. custom_pwm_set_params(FREQ, angle_to_compare(app->angle));
  174. FuriTimer* timer = furi_timer_alloc(tick, FuriTimerTypePeriodic, app);
  175. furi_timer_start(timer, furi_kernel_get_tick_frequency() / 1);
  176. InputEvent input;
  177. while(app->running) {
  178. FuriStatus qstat = furi_message_queue_get(app->event_queue, &input, 100);
  179. if(qstat == FuriStatusOk) {
  180. if(input.key == InputKeyBack) {
  181. app->running = 0;
  182. // TODO: handle InputTypeLong
  183. } else if(input.type == InputTypePress) {
  184. update_state(app, input.key);
  185. if(!app->running) {
  186. break;
  187. }
  188. switch(app->mode) {
  189. case Manual:
  190. custom_pwm_set_params(FREQ, angle_to_compare(app->angle));
  191. break;
  192. case Center:
  193. app->angle = 90;
  194. custom_pwm_set_params(FREQ, angle_to_compare(app->angle));
  195. break;
  196. case Sweep:
  197. // handled in ticks
  198. break;
  199. }
  200. }
  201. }
  202. }
  203. furi_timer_free(timer);
  204. app_free(app);
  205. return 0;
  206. }