scope.c 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. #include <furi.h>
  2. #include <furi_hal.h>
  3. #include <furi_hal_resources.h>
  4. #include <gui/gui.h>
  5. #include <input/input.h>
  6. #include "stm32wbxx_hal.h"
  7. #include "stm32wbxx_nucleo.h"
  8. #include "stm32wbxx_ll_adc.h"
  9. #include "stm32wbxx_ll_dma.h"
  10. #include "scope_icons.h"
  11. #define DIGITAL_SCALE_12BITS ((uint32_t) 0xFFF)
  12. #define VAR_CONVERTED_DATA_INIT_VALUE (DIGITAL_SCALE_12BITS + 1)
  13. #define VAR_CONVERTED_DATA_INIT_VALUE_16BITS (0xFFFF + 1U)
  14. #define __ADC_CALC_DATA_VOLTAGE(__VREFANALOG_VOLTAGE__, __ADC_DATA__) \
  15. ((__ADC_DATA__) * (__VREFANALOG_VOLTAGE__) / DIGITAL_SCALE_12BITS)
  16. #define VDDA_APPLI ((uint32_t)3300)
  17. ADC_HandleTypeDef hadc1 = { 0 };
  18. /* Variables for ADC conversion data */
  19. __IO uint16_t uhADCxConvertedData = VAR_CONVERTED_DATA_INIT_VALUE; /* ADC group regular conversion data */
  20. /* Variables for ADC conversion data computation to physical values */
  21. uint16_t uhADCxConvertedData_Voltage_mVolt = 0; /* Value of voltage calculated from ADC conversion data (unit: mV) */
  22. void Error_Handler()
  23. {
  24. while (1) {
  25. }
  26. }
  27. static void MX_ADC1_Init(void)
  28. {
  29. hadc1.Instance = ADC1;
  30. RCC_PeriphCLKInitTypeDef PeriphClkInitStruct = { 0 };
  31. PeriphClkInitStruct.PeriphClockSelection = RCC_PERIPHCLK_ADC;
  32. PeriphClkInitStruct.AdcClockSelection = RCC_ADCCLKSOURCE_PLL;
  33. if (HAL_RCCEx_PeriphCLKConfig(&PeriphClkInitStruct) != HAL_OK) {
  34. Error_Handler();
  35. }
  36. /* Peripheral clock enable */
  37. // __HAL_RCC_ADC_CLK_ENABLE();
  38. //HAL_Init();
  39. //SystemClock_Config();
  40. __HAL_RCC_ADC_CLK_ENABLE();
  41. __HAL_RCC_GPIOA_CLK_ENABLE();
  42. __HAL_RCC_GPIOB_CLK_ENABLE();
  43. __HAL_RCC_GPIOC_CLK_ENABLE();
  44. // __HAL_RCC_ADC_FORCE_RESET();
  45. // __HAL_RCC_ADC_RELEASE_RESET();
  46. //furi_hal_gpio_init_ex(&gpio_ext_pc0, GpioModeAnalog, GpioPullNo, GpioSpeedVeryHigh, GpioAltFnUnused);
  47. GPIO_InitTypeDef GPIO_initStructre = { 0, 0, 0, 0, 0 };
  48. GPIO_initStructre.Pin = GPIO_PIN_0; // Provide input to channel 10 of ADC i.e GPIO Pin 0 of Port C
  49. GPIO_initStructre.Mode = GPIO_MODE_ANALOG; //GPIO Pin as analog Mode
  50. GPIO_initStructre.Pull = GPIO_NOPULL;
  51. HAL_GPIO_Init(GPIOC, &GPIO_initStructre); // GPIO Initialization
  52. hadc1.Init.ClockPrescaler = ADC_CLOCK_SYNC_PCLK_DIV4;
  53. hadc1.Init.Resolution = ADC_RESOLUTION_12B;
  54. hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
  55. hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
  56. hadc1.Init.EOCSelection = ADC_EOC_SINGLE_CONV; //ADC_EOC_SEQ_CONV ; //ADC_EOC_SINGLE_CONV;
  57. hadc1.Init.LowPowerAutoWait = DISABLE;
  58. hadc1.Init.ContinuousConvMode = DISABLE;
  59. //hadc1.Init.ContinuousConvMode = ENABLE;
  60. hadc1.Init.NbrOfConversion = 1;
  61. hadc1.Init.DiscontinuousConvMode = DISABLE;
  62. hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
  63. hadc1.Init.ExternalTrigConvEdge = ADC_EXTERNALTRIGCONVEDGE_NONE;
  64. hadc1.Init.DMAContinuousRequests = DISABLE;
  65. hadc1.Init.Overrun = ADC_OVR_DATA_OVERWRITTEN;
  66. hadc1.Init.OversamplingMode = DISABLE;
  67. /*
  68. hadc1.Init.Oversampling.Ratio = ADC_OVERSAMPLING_RATIO_16;
  69. hadc1.Init.Oversampling.RightBitShift = ADC_RIGHTBITSHIFT_4;
  70. hadc1.Init.Oversampling.TriggeredMode = ADC_TRIGGEREDMODE_SINGLE_TRIGGER;
  71. hadc1.Init.Oversampling.OversamplingStopReset = ADC_REGOVERSAMPLING_CONTINUED_MODE;
  72. */
  73. if (HAL_ADC_Init(&hadc1) != HAL_OK) {
  74. Error_Handler();
  75. }
  76. }
  77. typedef struct {
  78. uint8_t x, y;
  79. } ImagePosition;
  80. uint16_t val1;
  81. ADC_HandleTypeDef hadc1;
  82. static ImagePosition image_position = {.x = 0,.y = 0 };
  83. void assert_failed(uint8_t * file, uint32_t line)
  84. {
  85. UNUSED(file);
  86. UNUSED(line);
  87. /* USER CODE BEGIN 6 */
  88. /* User can add his own implementation to report the file name and line number,
  89. ex: printf("Wrong parameters value: file %s on line %d", file, line) */
  90. /* Infinite loop */
  91. while (1) {
  92. }
  93. /* USER CODE END 6 */
  94. }
  95. // Screen is 128x64 px
  96. static void app_draw_callback(Canvas * canvas, void *ctx)
  97. {
  98. UNUSED(ctx);
  99. //canvas_clear(canvas);
  100. //canvas_draw_icon(canvas, image_position.x % 128, image_position.y % 64, &I_dolphin_71x25);
  101. //char *hello = "hello";
  102. char buf[50];
  103. snprintf(buf, 50, "%d", val1);
  104. canvas_draw_str(canvas, 10, 10, buf);
  105. //canvas_draw_dot(canvas, image_position.x % 128, image_position.y % 64);
  106. canvas_draw_line(canvas, 0, 0, 0, 63);
  107. canvas_draw_line(canvas, 0, 63, 128, 63);
  108. }
  109. static void app_input_callback(InputEvent * input_event, void *ctx)
  110. {
  111. furi_assert(ctx);
  112. FuriMessageQueue *event_queue = ctx;
  113. furi_message_queue_put(event_queue, input_event, FuriWaitForever);
  114. }
  115. int32_t scope_main(void *p)
  116. {
  117. UNUSED(p);
  118. //UNUSED(MX_DMA_Init);
  119. //UNUSED(MX_GPIO_Init);
  120. FuriMessageQueue *event_queue =
  121. furi_message_queue_alloc(8, sizeof(InputEvent));
  122. /*
  123. HAL_SYSCFG_VREFBUF_VoltageScalingConfig(SYSCFG_VREFBUF_VOLTAGE_SCALE1);
  124. HAL_SYSCFG_EnableVREFBUF();
  125. */
  126. VREFBUF->CSR |= VREFBUF_CSR_ENVR;
  127. VREFBUF->CSR &= ~VREFBUF_CSR_HIZ;
  128. VREFBUF->CSR |= VREFBUF_CSR_VRS;
  129. while (!(VREFBUF->CSR & VREFBUF_CSR_VRR)) {
  130. };
  131. MX_ADC1_Init();
  132. //Initialize LED on board
  133. if (HAL_ADCEx_Calibration_Start(&hadc1, ADC_SINGLE_ENDED) != HAL_OK) {
  134. Error_Handler();
  135. }
  136. int i = 0;
  137. // Configure view port
  138. ViewPort *view_port = view_port_alloc();
  139. view_port_draw_callback_set(view_port, app_draw_callback, view_port);
  140. view_port_input_callback_set(view_port, app_input_callback,
  141. event_queue);
  142. // Register view port in GUI
  143. Gui *gui = furi_record_open(RECORD_GUI);
  144. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  145. InputEvent event;
  146. bool running = true;
  147. while (running) {
  148. if (furi_message_queue_get(event_queue, &event, 100) ==
  149. FuriStatusOk) {
  150. if ((event.type == InputTypePress)
  151. || (event.type == InputTypeRepeat)) {
  152. switch (event.key) {
  153. case InputKeyLeft:
  154. image_position.x -= 2;
  155. break;
  156. case InputKeyRight:
  157. image_position.x += 2;
  158. break;
  159. case InputKeyUp:
  160. image_position.y -= 2;
  161. break;
  162. case InputKeyDown:
  163. image_position.y += 2;
  164. break;
  165. default:
  166. running = false;
  167. break;
  168. }
  169. }
  170. }
  171. ADC_ChannelConfTypeDef sConfig = { 0 };
  172. sConfig.Channel = ADC_CHANNEL_1;
  173. sConfig.Rank = ADC_REGULAR_RANK_1;
  174. sConfig.SamplingTime = ADC_SAMPLETIME_640CYCLES_5; ////ADC_SAMPLETIME_640CYCLES_5;
  175. sConfig.SingleDiff = ADC_SINGLE_ENDED;
  176. sConfig.OffsetNumber = ADC_OFFSET_NONE;
  177. sConfig.Offset = 0;
  178. if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK) {
  179. Error_Handler();
  180. }
  181. /* Start ADC group regular conversion */
  182. if (HAL_ADC_Start(&hadc1) != HAL_OK) {
  183. /* ADC conversion start error */
  184. Error_Handler();
  185. }
  186. /* Wait till conversion is done */
  187. if (HAL_ADC_PollForConversion(&hadc1, 1000) != HAL_OK) {
  188. /* End Of Conversion flag not set on time */
  189. Error_Handler();
  190. } else {
  191. /* Retrieve ADC conversion data */
  192. uhADCxConvertedData = HAL_ADC_GetValue(&hadc1);
  193. /* Computation of ADC conversions raw data to physical values */
  194. /* using helper macro. */
  195. uhADCxConvertedData_Voltage_mVolt =
  196. __ADC_CALC_DATA_VOLTAGE(VDDA_APPLI, uhADCxConvertedData);
  197. i++;
  198. val1 = uhADCxConvertedData_Voltage_mVolt; // + i;
  199. }
  200. /* Start ADC group regular conversion */
  201. if (HAL_ADC_Stop(&hadc1) != HAL_OK) {
  202. /* ADC conversion start error */
  203. Error_Handler();
  204. }
  205. view_port_update(view_port);
  206. }
  207. view_port_enabled_set(view_port, false);
  208. gui_remove_view_port(gui, view_port);
  209. view_port_free(view_port);
  210. furi_message_queue_free(event_queue);
  211. furi_record_close(RECORD_GUI);
  212. return 0;
  213. }