adc.c 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. #include "stm32f0xx_hal.h"
  2. #include "adc.h"
  3. #include "led.h"
  4. #include "tim.h"
  5. #include <stdio.h>
  6. #define VOLTAGE_FACTOR 0.000806 // 0.24194
  7. #define FREQ_FACTOR 0.000806
  8. ADC_HandleTypeDef AdcHandle;
  9. ADC_ChannelConfTypeDef sConfig;
  10. ADC_FILTER_t adc_bat;
  11. //ADC_FILTER_t adc_freq;
  12. adc_ufulter_t adc_freq;
  13. float bat_voltage = 0.0;
  14. uint16_t freq_voltage = 0.0;
  15. __IO uint16_t adc_values[2];
  16. //
  17. void adc_init(void)
  18. {
  19. GPIO_InitTypeDef GPIO_InitStruct = {0};
  20. __HAL_RCC_GPIOA_CLK_ENABLE();
  21. __HAL_RCC_ADC1_CLK_ENABLE();
  22. GPIO_InitStruct.Pin = GPIO_PIN_0 | GPIO_PIN_1;
  23. GPIO_InitStruct.Mode = GPIO_MODE_ANALOG;
  24. GPIO_InitStruct.Pull = GPIO_NOPULL;
  25. HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
  26. AdcHandle.Instance = ADC1;
  27. HAL_ADC_DeInit(&AdcHandle);
  28. AdcHandle.Init.ClockPrescaler = ADC_CLOCK_SYNC_PCLK_DIV2; /* Synchronous clock mode, input ADC clock with prscaler 2 */
  29. AdcHandle.Init.Resolution = ADC_RESOLUTION_12B; /* 12-bit resolution for converted data */
  30. AdcHandle.Init.DataAlign = ADC_DATAALIGN_RIGHT; /* Right-alignment for converted data */
  31. AdcHandle.Init.ScanConvMode = ADC_SCAN_DIRECTION_FORWARD; /* Sequencer disabled (ADC conversion on only 1 channel: channel set on rank 1) */
  32. AdcHandle.Init.EOCSelection = ADC_EOC_SINGLE_CONV; /* EOC flag picked-up to indicate conversion end */
  33. AdcHandle.Init.LowPowerAutoPowerOff = DISABLE;
  34. AdcHandle.Init.LowPowerAutoWait = DISABLE; /* Auto-delayed conversion feature disabled */
  35. AdcHandle.Init.ContinuousConvMode = DISABLE; /* Continuous mode enabled (automatic conversion restart after each conversion) */
  36. AdcHandle.Init.DiscontinuousConvMode = ENABLE; /* Parameter discarded because sequencer is disabled */
  37. AdcHandle.Init.ExternalTrigConv = ADC_SOFTWARE_START; /* Software start to trig the 1st conversion manually, without external event */
  38. AdcHandle.Init.ExternalTrigConvEdge = ADC_EXTERNALTRIGCONVEDGE_NONE; /* Parameter discarded because software trigger chosen */
  39. AdcHandle.Init.DMAContinuousRequests = ENABLE; /* ADC DMA continuous request to match with DMA circular mode */
  40. AdcHandle.Init.Overrun = ADC_OVR_DATA_OVERWRITTEN; /* DR register is overwritten with the last conversion result in case of overrun */
  41. AdcHandle.Init.SamplingTimeCommon = ADC_SAMPLETIME_28CYCLES_5;
  42. HAL_ADC_Init(&AdcHandle);
  43. sConfig.Channel = ADC_CHANNEL_0; /* Channel to be converted */
  44. sConfig.Rank = ADC_RANK_CHANNEL_NUMBER;
  45. HAL_ADC_ConfigChannel(&AdcHandle, &sConfig);
  46. sConfig.Channel = ADC_CHANNEL_1; /* Channel to be converted */
  47. HAL_ADC_ConfigChannel(&AdcHandle, &sConfig);
  48. HAL_ADCEx_Calibration_Start(&AdcHandle);
  49. adc_dma_init(&AdcHandle);
  50. //HAL_ADC_Start(&AdcHandle);
  51. }
  52. void adc_dma_init(ADC_HandleTypeDef *hadc)
  53. {
  54. static DMA_HandleTypeDef DmaHandle;
  55. __HAL_RCC_DMA1_CLK_ENABLE();
  56. DmaHandle.Instance = DMA1_Channel1;
  57. DmaHandle.Init.Direction = DMA_PERIPH_TO_MEMORY;
  58. DmaHandle.Init.PeriphInc = DMA_PINC_DISABLE;
  59. DmaHandle.Init.MemInc = DMA_MINC_ENABLE;
  60. DmaHandle.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD; /* Transfer from ADC by half-word to match with ADC configuration: ADC resolution 10 or 12 bits */
  61. DmaHandle.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD; /* Transfer to memory by half-word to match with buffer variable type: half-word */
  62. DmaHandle.Init.Mode = DMA_CIRCULAR; /* DMA in circular mode to match with ADC configuration: DMA continuous requests */
  63. DmaHandle.Init.Priority = DMA_PRIORITY_HIGH;
  64. HAL_DMA_DeInit(&DmaHandle);
  65. HAL_DMA_Init(&DmaHandle);
  66. __HAL_LINKDMA(hadc, DMA_Handle, DmaHandle);
  67. HAL_NVIC_SetPriority(DMA1_Channel1_IRQn, 1, 0);
  68. HAL_NVIC_EnableIRQ(DMA1_Channel1_IRQn);
  69. }
  70. //
  71. float adc_average(ADC_FILTER_t *flt, float new_value)
  72. {
  73. if (!(flt->head < ADC_FILTER_LEN))
  74. flt->head = 0;
  75. if (flt->count < ADC_FILTER_LEN)
  76. flt->count++;
  77. flt->sum -= flt->buf[flt->head];
  78. flt->sum += new_value;
  79. flt->buf[flt->head++] = new_value;
  80. return flt->sum/(flt->count);
  81. }
  82. //
  83. uint16_t adc_average_uint(adc_ufulter_t *flt, uint16_t new_value)
  84. {
  85. if (!(flt->head < ADC_FILTER_LEN))
  86. flt->head = 0;
  87. if (flt->count < ADC_FILTER_LEN)
  88. flt->count++;
  89. flt->sum -= flt->buf[flt->head];
  90. flt->sum += new_value;
  91. flt->buf[flt->head++] = new_value;
  92. return flt->sum/(flt->count);
  93. }
  94. //
  95. void adc_task(void)
  96. {
  97. HAL_ADC_Start_DMA(&AdcHandle, (uint32_t *)adc_values, 2);
  98. HAL_ADC_Start(&AdcHandle);
  99. #if 0
  100. bat_voltage = 2*adc_average(&adc_bat, HAL_ADC_GetValue(&AdcHandle) * ADC_FACTOR);
  101. if (bat_voltage > 3.4)
  102. led_but_on();
  103. else
  104. led_but_toggle();
  105. #endif
  106. }
  107. //
  108. void adc_print_data(void)
  109. {
  110. printf("%f %u\r\n", bat_voltage, freq_voltage);
  111. }
  112. void DMA1_Channel1_IRQHandler(void)
  113. {
  114. HAL_DMA_IRQHandler(AdcHandle.DMA_Handle);
  115. }
  116. void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef *AdcHandle)
  117. {
  118. bat_voltage = adc_average(&adc_bat, adc_values[0] * VOLTAGE_FACTOR);
  119. // Для получения значения в вольтах
  120. //freq_voltage = adc_average(&adc_freq, adc_values[1] * FREQ_FACTOR);
  121. freq_voltage = 4095 - adc_average_uint(&adc_freq, adc_values[1]);
  122. tim_set_string_freq();
  123. }