![]() |
發布時間: 2024-10-21 16:10
正文摘要:/** **************************************************************************************************** * @file adc.c * @author ... |
調整了以下幾點代碼,看可否解決DMA傳輸數據為0的問題。修改點包括: 1. **確保DMA和ADC的初始化順序正確**。 2. **添加調試日志,監控DMA傳輸狀態和數據**。 3. **確保DMA中斷優先級和中斷處理函數工作正常**。 4. **保證ADC DMA啟動后的數據讀取流程正確**。 ### 修改后的代碼如下: ```c #include "./BSP/ADC/adc.h" #include "./SYSTEM/delay/delay.h" #include "./SYSTEM/usart/usart.h" ADC_HandleTypeDef g_adc_handle; /* ADC句柄 */ DMA_HandleTypeDef g_dma_adc_handle; /* 與ADC關聯的DMA句柄 */ uint8_t g_adc_dma_sta = 0; /* DMA傳輸狀態標志, 0,未完成; 1, 已完成 */ uint16_t g_adc_value[ADC_CH_NUM * ADC_COLL] = {0}; /* 存儲ADC原始值 */ uint16_t g_adc_val[ADC_CH_NUM] = {0}; /* 存儲處理后的ADC平均值 */ /********************************************************************/ /** * @brief ADC初始化函數 * @retval 無 */ void adc_init(void) { ADC_ChannelConfTypeDef sConfig = {0}; g_adc_handle.Instance = ADC_ADCX; g_adc_handle.Init.ContinuousConvMode = ENABLE; g_adc_handle.Init.DataAlign = ADC_DATAALIGN_RIGHT; g_adc_handle.Init.DiscontinuousConvMode = DISABLE; g_adc_handle.Init.ExternalTrigConv = ADC_SOFTWARE_START; g_adc_handle.Init.NbrOfConversion = ADC_CH_NUM; g_adc_handle.Init.ScanConvMode = ENABLE; HAL_ADC_Init(&g_adc_handle); HAL_ADCEx_Calibration_Start(&g_adc_handle); // ADC校準 sConfig.Channel = ADC_ADCX_CH4; sConfig.Rank = 1; sConfig.SamplingTime = ADC_SAMPLETIME_239CYCLES_5; HAL_ADC_ConfigChannel(&g_adc_handle, &sConfig); sConfig.Channel = ADC_ADCX_CH5; sConfig.Rank = 2; sConfig.SamplingTime = ADC_SAMPLETIME_239CYCLES_5; HAL_ADC_ConfigChannel(&g_adc_handle, &sConfig); } /********************************************************************/ /** * @brief DMA初始化函數 * @retval 無 */ void adc_dma_init(void) { if ((uint32_t)ADC_ADCX_DMACx > (uint32_t)DMA1_Channel7) /* 大于DMA1_Channel7, 則為DMA2的通道了 */ { __HAL_RCC_DMA2_CLK_ENABLE(); /* DMA2時鐘使能 */ } else { __HAL_RCC_DMA1_CLK_ENABLE(); /* DMA1時鐘使能 */ } g_dma_adc_handle.Instance = DMA1_Channel1; g_dma_adc_handle.Init.Direction = DMA_PERIPH_TO_MEMORY; g_dma_adc_handle.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD; g_dma_adc_handle.Init.MemInc = DMA_MINC_ENABLE; g_dma_adc_handle.Init.Mode = DMA_CIRCULAR; // 設置為循環模式,確保DMA持續工作 g_dma_adc_handle.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD; g_dma_adc_handle.Init.PeriphInc = DMA_PINC_DISABLE; g_dma_adc_handle.Init.Priority = DMA_PRIORITY_MEDIUM; HAL_DMA_Init(&g_dma_adc_handle); __HAL_LINKDMA(&g_adc_handle, DMA_Handle, g_dma_adc_handle); // 關聯DMA和ADC } /********************************************************************/ /** * @brief DMA傳輸完成中斷處理函數 * @retval 無 */ void ADC_ADCX_DMASX_IRQHandler(void) { HAL_DMA_IRQHandler(&g_dma_adc_handle); g_adc_dma_sta = 1; // 標記DMA傳輸完成 } /********************************************************************/ /** * @brief ADC轉換完成回調函數 * @retval 無 */ void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef *hadc) { if (hadc->Instance == ADC_ADCX) { HAL_ADC_Stop_DMA(hadc); // 停止DMA,防止數據被覆蓋 calc_adc_val(g_adc_val); // 計算ADC平均值 g_adc_dma_sta = 1; // 標記DMA傳輸完成 HAL_ADC_Start_DMA(hadc, (uint32_t *)g_adc_value, ADC_CH_NUM * ADC_COLL); // 重新啟動DMA } } /********************************************************************/ /** * @brief 計算ADC的平均值(濾波) * @param *p : 存放ADC值的指針地址 * @retval 無 */ void calc_adc_val(uint16_t *p) { uint32_t temp[ADC_CH_NUM] = {0}; /* 緩存數組 */ for (int i = 0; i < ADC_COLL; i++) { for (int j = 0; j < ADC_CH_NUM; j++) { temp[j] += g_adc_value[j + i * ADC_CH_NUM]; /* 累加 */ } } for (int j = 0; j < ADC_CH_NUM; j++) { p[j] = temp[j] / ADC_COLL; /* 計算平均值 */ } } int main(void) { HAL_Init(); /* 初始化HAL庫 */ sys_stm32_clock_init(RCC_PLL_MUL9); /* 設置時鐘, 72Mhz */ delay_init(72); /* 延時初始化 */ usart_init(115200); /* 串口初始化為115200 */ adc_init(); // 初始化ADC adc_dma_init(); // 初始化DMA // 啟動DMA if (HAL_ADC_Start_DMA(&g_adc_handle, (uint32_t *)g_adc_value, ADC_CH_NUM * ADC_COLL) != HAL_OK) { printf("ADC DMA start failed\r\n"); } while (1) { if (g_adc_dma_sta) // DMA傳輸完成 { g_adc_dma_sta = 0; // 重置標志位 printf("ADC Value Channel 1: %d, Channel 2: %d\r\n", g_adc_val[0], g_adc_val[1]); } } } ``` ### 程序修改了: 1. **DMA初始化**:確保正確初始化DMA,并關聯到ADC句柄。 2. **DMA模式**:將DMA模式設置為 `DMA_CIRCULAR`,這樣DMA會持續工作,而不需要手動重新啟動。 3. **中斷處理**:在 `ADC_ADCX_DMASX_IRQHandler()` 中設置DMA傳輸完成標志位,并在 `HAL_ADC_ConvCpltCallback()` 中處理傳輸完成的邏輯。 4. **ADC轉換完成處理**:在回調函數 `HAL_ADC_ConvCpltCallback()` 中停止DMA、計算ADC平均值,并重新啟動DMA。 5. **主循環**:通過輪詢 `g_adc_dma_sta` 檢測DMA傳輸完成,并在主循環中打印ADC轉換后的數據。 ### 調試提示: - 在 `HAL_ADC_Start_DMA()` 后面添加錯誤處理,確保DMA和ADC啟動正常。 - 你可以通過串口日志查看 `g_adc_val[]` 是否有正確的值輸出。如果還存在問題,可以進一步調試DMA的中斷是否正確觸發。 |