flizzer_tracker.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316
  1. #include <stdio.h>
  2. #include <furi.h>
  3. #include <gui/gui.h>
  4. #include <notification/notification_messages.h>
  5. #include <input/input.h>
  6. #include <furi_hal.h>
  7. #include <u8g2_glue.h>
  8. #include <stm32wbxx_ll_tim.h>
  9. #include "flizzer_tracker_hal.h"
  10. /*
  11. Fontname: -Raccoon-Fixed4x6-Medium-R-Normal--6-60-75-75-P-40-ISO10646-1
  12. Copyright:
  13. Glyphs: 95/203
  14. BBX Build Mode: 0
  15. */
  16. const uint8_t u8g2_font_tom_thumb_4x6_tr[725] U8G2_FONT_SECTION("u8g2_font_tom_thumb_4x6_tr") =
  17. "_\0\2\2\2\3\3\4\4\3\6\0\377\5\377\5\0\0\352\1\330\2\270 \5\340\315\0!\6\265\310"
  18. "\254\0\42\6\213\313$\25#\10\227\310\244\241\206\12$\10\227\310\215\70b\2%\10\227\310d\324F\1"
  19. "&\10\227\310(\65R\22'\5\251\313\10(\6\266\310\251\62)\10\226\310\304\224\24\0*\6\217\312\244"
  20. "\16+\7\217\311\245\225\0,\6\212\310)\0-\5\207\312\14.\5\245\310\4/\7\227\310Ve\4\60"
  21. "\7\227\310-k\1\61\6\226\310\255\6\62\10\227\310h\220\312\1\63\11\227\310h\220\62X\0\64\10\227"
  22. "\310$\65b\1\65\10\227\310\214\250\301\2\66\10\227\310\315\221F\0\67\10\227\310\314TF\0\70\10\227"
  23. "\310\214\64\324\10\71\10\227\310\214\64\342\2:\6\255\311\244\0;\7\222\310e\240\0<\10\227\310\246\32"
  24. "d\20=\6\217\311l\60>\11\227\310d\220A*\1\77\10\227\310\314\224a\2@\10\227\310UC\3"
  25. "\1A\10\227\310UC\251\0B\10\227\310\250\264\322\2C\7\227\310\315\32\10D\10\227\310\250d-\0"
  26. "E\10\227\310\214\70\342\0F\10\227\310\214\70b\4G\10\227\310\315\221\222\0H\10\227\310$\65\224\12"
  27. "I\7\227\310\254X\15J\7\227\310\226\252\2K\10\227\310$\265\222\12L\7\227\310\304\346\0M\10\227"
  28. "\310\244\61\224\12N\10\227\310\244q\250\0O\7\227\310UV\5P\10\227\310\250\264b\4Q\10\227\310"
  29. "Uj$\1R\10\227\310\250\64V\1S\10\227\310m\220\301\2T\7\227\310\254\330\2U\7\227\310$"
  30. "W\22V\10\227\310$\253L\0W\10\227\310$\65\206\12X\10\227\310$\325R\1Y\10\227\310$U"
  31. "V\0Z\7\227\310\314T\16[\7\227\310\214X\16\134\10\217\311d\220A\0]\7\227\310\314r\4^"
  32. "\5\213\313\65_\5\207\310\14`\6\212\313\304\0a\7\223\310\310\65\2b\10\227\310D\225\324\2c\7"
  33. "\223\310\315\14\4d\10\227\310\246\245\222\0e\6\223\310\235\2f\10\227\310\246\264b\2g\10\227\307\35"
  34. "\61%\0h\10\227\310D\225\254\0i\6\265\310\244\1j\10\233\307f\30U\5k\10\227\310\304\264T"
  35. "\1l\7\227\310\310\326\0m\7\223\310<R\0n\7\223\310\250d\5o\7\223\310U\252\2p\10\227"
  36. "\307\250\244V\4q\10\227\307-\225d\0r\6\223\310\315\22s\10\223\310\215\70\22\0t\10\227\310\245"
  37. "\25\243\0u\7\223\310$+\11v\10\223\310$\65R\2w\7\223\310\244q\4x\7\223\310\244\62\25"
  38. "y\11\227\307$\225dJ\0z\7\223\310\254\221\6{\10\227\310\251\32D\1|\6\265\310(\1}\11"
  39. "\227\310\310\14RR\0~\6\213\313\215\4\0\0\0\4\377\377\0";
  40. typedef enum {
  41. EventTypeTick,
  42. EventTypeInput,
  43. } EventType;
  44. typedef struct {
  45. EventType type;
  46. InputEvent input;
  47. } FlizzerTrackerEvent;
  48. typedef struct
  49. {
  50. bool stop;
  51. uint32_t counter;
  52. uint32_t counter_2;
  53. NotificationApp* notification;
  54. SoundEngine sound_engine;
  55. uint32_t frequency;
  56. uint8_t current_waveform_index;
  57. uint16_t pw;
  58. } FlizzerTrackerApp;
  59. #define FURI_HAL_SPEAKER_TIMER TIM2
  60. #define FURI_HAL_SPEAKER_CHANNEL LL_TIM_CHANNEL_CH1
  61. void timer_draw_callback(void* ctx)
  62. {
  63. if(LL_TIM_IsActiveFlag_UPDATE(TIM2))
  64. {
  65. LL_TIM_ClearFlag_UPDATE(TIM2);
  66. }
  67. FlizzerTrackerApp* tracker = (FlizzerTrackerApp*)ctx;
  68. tracker->counter++;
  69. //return;
  70. }
  71. void timer_draw_callback_2(void* ctx)
  72. {
  73. if(LL_TIM_IsActiveFlag_UPDATE(TIM1))
  74. {
  75. LL_TIM_ClearFlag_UPDATE(TIM1);
  76. }
  77. FlizzerTrackerApp* tracker = (FlizzerTrackerApp*)ctx;
  78. tracker->counter_2++;
  79. //return;
  80. }
  81. static void sound_engine_dma_isr(void* ctx)
  82. {
  83. FlizzerTrackerApp* tracker = (FlizzerTrackerApp*)ctx;
  84. // half of transfer
  85. if(LL_DMA_IsActiveFlag_HT1(DMA1))
  86. {
  87. LL_DMA_ClearFlag_HT1(DMA1);
  88. // fill first half of buffer
  89. sound_engine_fill_buffer(&tracker->sound_engine, tracker->sound_engine.audio_buffer, tracker->sound_engine.audio_buffer_size / 2);
  90. tracker->counter++;
  91. }
  92. // transfer complete
  93. if(LL_DMA_IsActiveFlag_TC1(DMA1))
  94. {
  95. LL_DMA_ClearFlag_TC1(DMA1);
  96. // fill second half of buffer
  97. sound_engine_fill_buffer(&tracker->sound_engine, &tracker->sound_engine.audio_buffer[tracker->sound_engine.audio_buffer_size / 2], tracker->sound_engine.audio_buffer_size / 2); //&app->sample_buffer[index]
  98. tracker->counter++;
  99. }
  100. }
  101. const char* wave_names[5] =
  102. {
  103. "NONE",
  104. "NOISE",
  105. "PULSE",
  106. "TRIANGLE",
  107. "SAWTOOTH",
  108. };
  109. static void draw_callback(Canvas* canvas, void* ctx)
  110. {
  111. FlizzerTrackerApp* tracker = (FlizzerTrackerApp*)ctx;
  112. canvas_clear(canvas);
  113. canvas_set_custom_font(canvas, u8g2_font_tom_thumb_4x6_tr);
  114. char buffer[30] = {0};
  115. snprintf(buffer, 20, "FREQUENCY: %ld Hz", tracker->frequency);
  116. canvas_draw_str(canvas, 0, 10, buffer);
  117. snprintf(buffer, 20, "WAVEFORM: %s", wave_names[tracker->current_waveform_index]);
  118. canvas_draw_str(canvas, 0, 20, buffer);
  119. snprintf(buffer, 20, "PULSE WIDTH: $%03X", tracker->pw);
  120. canvas_draw_str(canvas, 0, 30, buffer);
  121. }
  122. static void input_callback(InputEvent* input_event, void* ctx)
  123. {
  124. // Проверяем, что контекст не нулевой
  125. furi_assert(ctx);
  126. FuriMessageQueue* event_queue = ctx;
  127. FlizzerTrackerEvent event = {.type = EventTypeInput, .input = *input_event};
  128. furi_message_queue_put(event_queue, &event, FuriWaitForever);
  129. }
  130. const uint8_t waveforms[5] =
  131. {
  132. SE_WAVEFORM_NONE,
  133. SE_WAVEFORM_NOISE,
  134. SE_WAVEFORM_PULSE,
  135. SE_WAVEFORM_TRIANGLE,
  136. SE_WAVEFORM_SAW,
  137. };
  138. int32_t flizzer_tracker_app(void* p)
  139. {
  140. UNUSED(p);
  141. // Текущее событие типа кастомного типа FlizzerTrackerEvent
  142. FlizzerTrackerEvent event;
  143. // Очередь событий на 8 элементов размера FlizzerTrackerEvent
  144. FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(FlizzerTrackerEvent));
  145. FlizzerTrackerApp* tracker = malloc(sizeof(FlizzerTrackerApp));
  146. //direct_draw_run(tracker);
  147. // Создаем новый view port
  148. ViewPort* view_port = view_port_alloc();
  149. // Создаем callback отрисовки, без контекста
  150. view_port_draw_callback_set(view_port, draw_callback, tracker);
  151. // Создаем callback нажатий на клавиши, в качестве контекста передаем
  152. // нашу очередь сообщений, чтоб запихивать в неё эти события
  153. view_port_input_callback_set(view_port, input_callback, event_queue);
  154. // Создаем GUI приложения
  155. Gui* gui = furi_record_open(RECORD_GUI);
  156. // Подключаем view port к GUI в полноэкранном режиме
  157. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  158. tracker->notification = furi_record_open(RECORD_NOTIFICATION);
  159. notification_message(tracker->notification, &sequence_display_backlight_enforce_on);
  160. furi_hal_interrupt_set_isr_ex(FuriHalInterruptIdDma1Ch1, 13, sound_engine_dma_isr, tracker);
  161. //sound_engine_init(&tracker->sound_engine, 44100, false, 4096);
  162. sound_engine_init(&tracker->sound_engine, 44100, true, 4096);
  163. tracker->sound_engine.channel[0].waveform = SE_WAVEFORM_NOISE;
  164. tracker->sound_engine.channel[0].pw = 0x200;
  165. tracker->frequency = 440;
  166. tracker->current_waveform_index = 1;
  167. sound_engine_set_channel_frequency(&tracker->sound_engine, &tracker->sound_engine.channel[0], 440 * 1024);
  168. sound_engine_start();
  169. // Бесконечный цикл обработки очереди событий
  170. while(1)
  171. {
  172. // Выбираем событие из очереди в переменную event (ждем бесконечно долго, если очередь пуста)
  173. // и проверяем, что у нас получилось это сделать
  174. furi_check(furi_message_queue_get(event_queue, &event, FuriWaitForever) == FuriStatusOk);
  175. // Наше событие — это нажатие кнопки
  176. if(event.type == EventTypeInput)
  177. {
  178. // Если нажата кнопка "назад", то выходим из цикла, а следовательно и из приложения
  179. if(event.input.key == InputKeyBack && event.input.type == InputTypeShort)
  180. {
  181. break;
  182. }
  183. if(event.input.key == InputKeyUp && event.input.type == InputTypeShort)
  184. {
  185. tracker->frequency += 50;
  186. sound_engine_set_channel_frequency(&tracker->sound_engine, &tracker->sound_engine.channel[0], tracker->frequency * 1024);
  187. //break;
  188. }
  189. if(event.input.key == InputKeyDown && event.input.type == InputTypeShort)
  190. {
  191. if(tracker->frequency > 50)
  192. {
  193. tracker->frequency -= 50;
  194. }
  195. sound_engine_set_channel_frequency(&tracker->sound_engine, &tracker->sound_engine.channel[0], tracker->frequency * 1024);
  196. //break;
  197. }
  198. if(event.input.key == InputKeyRight && event.input.type == InputTypeShort)
  199. {
  200. if(tracker->current_waveform_index < 4)
  201. {
  202. tracker->current_waveform_index++;
  203. }
  204. tracker->sound_engine.channel[0].waveform = waveforms[tracker->current_waveform_index];
  205. //break;
  206. }
  207. if(event.input.key == InputKeyLeft && event.input.type == InputTypeShort)
  208. {
  209. if(tracker->current_waveform_index > 0)
  210. {
  211. tracker->current_waveform_index--;
  212. }
  213. tracker->sound_engine.channel[0].waveform = waveforms[tracker->current_waveform_index];
  214. //break;
  215. }
  216. if(event.input.key == InputKeyOk && event.input.type == InputTypeShort)
  217. {
  218. if(tracker->pw + 0x100 < 0xFFF)
  219. {
  220. tracker->pw += 0x100;
  221. }
  222. else
  223. {
  224. tracker->pw = 0x100;
  225. }
  226. tracker->sound_engine.channel[0].waveform = waveforms[tracker->current_waveform_index];
  227. tracker->sound_engine.channel[0].pw = tracker->pw;
  228. //break;
  229. }
  230. }
  231. }
  232. //furi_hal_interrupt_set_isr_ex(FuriHalInterruptIdTIM2, 14, NULL, NULL);
  233. //furi_hal_interrupt_set_isr_ex(FuriHalInterruptIdTim1UpTim16, 15, NULL, NULL);
  234. sound_engine_stop();
  235. sound_engine_deinit(&tracker->sound_engine);
  236. FURI_CRITICAL_ENTER();
  237. LL_TIM_DeInit(TIM1);
  238. LL_TIM_DeInit(TIM2);
  239. LL_TIM_DeInit(TIM16);
  240. FURI_CRITICAL_EXIT();
  241. notification_message(tracker->notification, &sequence_display_backlight_enforce_auto);
  242. furi_record_close(RECORD_NOTIFICATION);
  243. // Специальная очистка памяти, занимаемой очередью
  244. furi_message_queue_free(event_queue);
  245. // Чистим созданные объекты, связанные с интерфейсом
  246. gui_remove_view_port(gui, view_port);
  247. view_port_free(view_port);
  248. furi_record_close(RECORD_GUI);
  249. free(tracker);
  250. return 0;
  251. }