找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 45|回复: 0
打印 上一主题 下一主题
收起左侧

adc_dma进行采集时打印数据为0,不开dma数据能够正常显示,想问一下是什么原因

[复制链接]
跳转到指定楼层
楼主

  1. /**
  2. ****************************************************************************************************
  3. * @file        adc.c
  4. * @author      正点原子团队(ALIENTEK)
  5. * @version     V1.3
  6. * @date        2020-04-24
  7. * @brief       ADC 驱动代码
  8. * @license     Copyright (c) 2020-2032, 广州市星翼电子科技有限公司
  9. ****************************************************************************************************
  10. * @attention

  11. *
  12. * 修改说明
  13. * V1.0 20200423
  14. * 第一次发布
  15. * V1.1 20200423
  16. * 1,支持ADC单通道DMA采集
  17. * 2,新增adc_dma_init和adc_dma_enable函数.
  18. * V1.2 20200423
  19. * 1,支持ADC多通道DMA采集
  20. * 2,新增adc_nch_dma_init函数.
  21. * V1.3 20200424
  22. * 1,支持内部温度传感器温度采集
  23. * 2,新增adc_temperature_init和adc_get_temperature函数.
  24. *
  25. ****************************************************************************************************
  26. */

  27. #include "./BSP/ADC/adc.h"
  28. //#include "./BSP/DMA/dma.h"
  29. #include "./SYSTEM/delay/delay.h"
  30. #include "./SYSTEM/usart/usart.h"


  31. ADC_HandleTypeDef g_adc_handle;   /* ADC句柄 */
  32. DMA_HandleTypeDef g_dma_adc_handle;     /* 与ADC关联的DMA句柄 */
  33. uint8_t g_adc_dma_sta = 0;              /* DMA传输状态标志, 0,未完成; 1, 已完成 */


  34. uint16_t g_adc_value[ADC_CH_NUM * ADC_COLL] = {0};  /* 存储ADC原始值 */
  35. //float g_adc_u_value[ADC_CH_NUM] = {0};              /* 存储ADC转换后的电压值 */

  36. /********************************************************************/
  37. /**
  38. * @brief       ADC初始化函数
  39. *   @note      本函数支持ADC1/ADC2任意通道, 但是不支持ADC3
  40. *              我们使用12位精度, ADC采样时钟=12M, 转换时间为: 采样周期 + 12.5个ADC周期
  41. *              设置最大采样周期: 239.5, 则转换时间 = 252 个ADC周期 = 21us
  42. * @param       无
  43. * @retval      无
  44. */
  45. void adc_init(void)
  46. {
  47.    ADC_ChannelConfTypeDef sConfig = {0};
  48.          
  49.          g_adc_handle.Instance = ADC_ADCX;
  50.          g_adc_handle.Init.ContinuousConvMode = ENABLE;
  51.          g_adc_handle.Init.DataAlign = ADC_DATAALIGN_RIGHT;
  52.          g_adc_handle.Init.DiscontinuousConvMode = DISABLE;
  53.          g_adc_handle.Init.ExternalTrigConv = ADC_SOFTWARE_START;
  54.          g_adc_handle.Init.NbrOfConversion =  ADC_CH_NUM;
  55.          g_adc_handle.Init.NbrOfDiscConversion = 0;
  56.          g_adc_handle.Init.ScanConvMode = ENABLE;
  57.          HAL_ADC_Init(&g_adc_handle);
  58.          
  59.          HAL_ADCEx_Calibration_Start(&g_adc_handle);//ADC校准

  60.          
  61.          sConfig.Channel = ADC_ADCX_CH4;
  62.          sConfig.Rank = 1;
  63.          sConfig.SamplingTime = ADC_SAMPLETIME_239CYCLES_5;
  64.          HAL_ADC_ConfigChannel(&g_adc_handle,&sConfig);
  65.          
  66.          sConfig.Channel = ADC_ADCX_CH5;
  67.          sConfig.Rank = 2;
  68.          sConfig.SamplingTime = ADC_SAMPLETIME_239CYCLES_5;
  69.          HAL_ADC_ConfigChannel(&g_adc_handle,&sConfig);
  70.          
  71.         __HAL_LINKDMA(&g_adc_handle, DMA_Handle ,g_dma_adc_handle);
  72.          
  73.          
  74. //                        HAL_NVIC_SetPriority(DMA1_Channel1_IRQn,3,3);
  75. //                        HAL_NVIC_EnableIRQ(DMA1_Channel1_IRQn);
  76. //                        
  77. //      HAL_DMA_Start_IT(&g_dma_adc_handle, (uint32_t)&ADC1->DR, (uint32_t)g_adc_value, ADC_CH_NUM * ADC_COLL);
  78. //                        HAL_ADC_Start_DMA(&g_adc_handle,(uint32_t *)g_adc_value,ADC_CH_NUM * ADC_COLL);
  79. }



  80. void adc_dma_init(void)
  81. {
  82.     if ((uint32_t)ADC_ADCX_DMACx > (uint32_t)DMA1_Channel7)     /* 大于DMA1_Channel7, 则为DMA2的通道了 */
  83.     {
  84.         __HAL_RCC_DMA2_CLK_ENABLE();                            /* DMA2时钟使能 */
  85.     }
  86.     else
  87.     {
  88.         __HAL_RCC_DMA1_CLK_ENABLE();                            /* DMA1时钟使能 */
  89.     }
  90.         
  91.         g_dma_adc_handle.Instance = DMA1_Channel1;
  92.         g_dma_adc_handle.Init.Direction = DMA_PERIPH_TO_MEMORY;
  93.         g_dma_adc_handle.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
  94.         g_dma_adc_handle.Init.MemInc = DMA_MINC_ENABLE;
  95.         g_dma_adc_handle.Init.Mode = DMA_NORMAL;
  96.         g_dma_adc_handle.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;
  97.         g_dma_adc_handle.Init.PeriphInc =  DMA_PINC_DISABLE;
  98.         g_dma_adc_handle.Init.Priority = DMA_PRIORITY_MEDIUM;
  99.         HAL_DMA_Init(&g_dma_adc_handle);
  100.         
  101. //        __HAL_LINKDMA(&g_adc_handle, DMA_Handle ,g_dma_adc_handle);
  102.         
  103.                                 
  104.                         HAL_NVIC_SetPriority(DMA1_Channel1_IRQn,1,3);
  105.                         HAL_NVIC_EnableIRQ(DMA1_Channel1_IRQn);
  106.                         
  107. //      HAL_DMA_Start_IT(&g_dma_adc_handle, (uint32_t)&ADC1->DR, (uint32_t)g_adc_value, ADC_CH_NUM * ADC_COLL);
  108.                         HAL_ADC_Start_DMA(&g_adc_handle,(uint32_t *)g_adc_value,ADC_CH_NUM * ADC_COLL);
  109.                
  110. //                if (HAL_ADC_Start_DMA(&g_adc_handle, (uint32_t *)g_adc_value, ADC_CH_NUM * ADC_COLL) != HAL_OK)
  111. //{
  112. //    // 处理错误
  113. //        printf("error");
  114. //}

  115. }









  116. /**
  117. * @brief       ADC底层驱动,引脚配置,时钟使能
  118.                 此函数会被HAL_ADC_Init()调用
  119. * @param       hadc:ADC句柄
  120. * @retval      无
  121. */
  122. void HAL_ADC_MspInit(ADC_HandleTypeDef *hadc)
  123. {
  124.     if(hadc->Instance == ADC_ADCX)
  125.     {
  126.         GPIO_InitTypeDef GPIO_Initure;
  127.                         
  128.         ADC_ADCX_CHY_CLK_ENABLE();  //adc时钟
  129.                         
  130.         ADC_ADCX_CH4_GPIO_CLK_ENABLE(); /* 开启GPIO时钟 */
  131.         ADC_ADCX_CH5_GPIO_CLK_ENABLE(); /* 开启GPIO时钟 */
  132.                         

  133.         /* AD采集引脚模式设置,模拟输入 */
  134.         GPIO_Initure.Pin = ADC_ADCX_CH4_GPIO_PIN;        
  135.         GPIO_Initure.Mode = GPIO_MODE_ANALOG;
  136.         GPIO_Initure.Pull = GPIO_PULLUP;      
  137.         HAL_GPIO_Init(ADC_ADCX_CH4_GPIO_PORT, &GPIO_Initure);
  138.                         
  139.                                 /* AD采集引脚模式设置,模拟输入 */
  140.         GPIO_Initure.Pin = ADC_ADCX_CH5_GPIO_PIN;        
  141.         GPIO_Initure.Mode = GPIO_MODE_ANALOG;
  142.         GPIO_Initure.Pull = GPIO_PULLUP;      
  143.         HAL_GPIO_Init(ADC_ADCX_CH5_GPIO_PORT, &GPIO_Initure);
  144.                         
  145.                         
  146.                         //ADC时钟初始化
  147.                         RCC_PeriphCLKInitTypeDef adc_clk_init = {0};
  148.                         adc_clk_init.PeriphClockSelection = RCC_PERIPHCLK_ADC;
  149.                         adc_clk_init.AdcClockSelection = RCC_ADCPCLK2_DIV6;
  150.                         HAL_RCCEx_PeriphCLKConfig(&adc_clk_init);

  151.     }
  152. }




  153. void ADC_ADCX_DMASX_IRQHandler(void)
  154. {
  155.         HAL_DMA_IRQHandler(&g_dma_adc_handle);
  156.   g_adc_dma_sta = 1;  // 标记DMA传输完成
  157. }


  158. uint16_t g_adc_val[ADC_CH_NUM];


  159. void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef *hadc)
  160. {
  161.         uint32_t temp;
  162.         if(hadc->Instance == ADC_ADCX)
  163.         {
  164.                
  165. //                print_adc_values();
  166.                 HAL_ADC_Stop_DMA(hadc);
  167.                 calc_adc_val(g_adc_val);
  168.                 HAL_ADC_Start_DMA(&g_adc_handle,(uint32_t *)&g_adc_value,ADC_SUM);
  169.   }
  170. }


  171. //void print_adc_values(void)
  172. //{
  173. //    // 确保在打印之前进行数据校验,例如检查数据个数和有效性
  174. //    for (int i = 0; i < ADC_CH_NUM * ADC_COLL; i++)
  175. //    {
  176. //        // 确认数组中的数据是否有效
  177. //        if (g_adc_value[i] != 0) // 例如,不等于0表示有效
  178. //        {
  179. //            printf("g_adc_value[%d]: %u\r\n", i, g_adc_value[i]);
  180. //        }
  181. //    }
  182. //}




  183. /**
  184. * @brief       获取通道ch的转换值,取times次, 然后平均
  185. * @param       ch: 通道号, 0~17
  186. * @retval      通道ch的times次转换结果平均值
  187. */
  188. uint32_t adc_get_result_average(uint8_t ch)
  189. {
  190.         uint32_t temp_val = 0;
  191.         uint16_t t;
  192.         
  193.         for(t = ch; t < ADC_SUM; t += ADC_CH_NUM)
  194.         {
  195.                 temp_val += g_adc_value[t];
  196.         }
  197.         
  198.         return temp_val / ADC_COLL;
  199. }


  200. /**
  201. * @brief       计算ADC的平均值(滤波)
  202. * @param       * p :存放ADC值的指针地址
  203. * @note        此函数对电压、温度、电流对应的ADC值进行滤波
  204. * @retval      无
  205. */
  206. void calc_adc_val(uint16_t * p)
  207. {
  208.     uint32_t temp[ADC_CH_NUM] = {0,0};            /* 定义一个缓存数组 */
  209.     int i,j;
  210.     for(i = 0; i < ADC_COLL; i++)                   /* 循环采集ADC_COLL次数 */
  211.     {
  212.         for(j = 0; j < ADC_CH_NUM; j++)             /* 根据ADC通道数循环获取,并累加 */
  213.         {
  214.             temp[j] += g_adc_value[j+i*ADC_CH_NUM]; /* 将采集到的ADC值,各通道进行累加 */
  215.         }
  216.     }
  217.     for(j = 0; j < ADC_CH_NUM; j++)
  218.     {
  219.         temp[j] /= ADC_COLL;                        /* 获取平均值 */
  220.         p[j] = temp[j];                             /* 存到*p */
  221.     }
  222. }













  223. /**
  224. ****************************************************************************************************
  225. * @file        main.c
  226. * @author      正点原子团队(ALIENTEK)
  227. * @version     V1.0
  228. * @date        2020-04-24
  229. * @brief       SPI 实验
  230. * @license     Copyright (c) 2020-2032, 广州市星翼电子科技有限公司
  231. ****************************************************************************************************
  232. * @attention
  233. *
  234. * 实验平台:正点原子 STM32F103开发板
  235. *
  236. ****************************************************************************************************
  237. */

  238. #include "./SYSTEM/sys/sys.h"
  239. #include "./SYSTEM/usart/usart.h"
  240. #include "./SYSTEM/delay/delay.h"
  241. #include "./USMART/usmart.h"
  242. #include "./BSP/LED/led.h"
  243. #include "./BSP/LCD/lcd.h"
  244. #include "./BSP/KEY/key.h"
  245. #include "./BSP/NORFLASH/norflash.h"

  246. #include "./BSP/SPI/spi.h"
  247. #include "./BSP/ICM42688/icm42688.h"

  248. #include "./BSP/ESP8266USART/esp8266usart.h"
  249. #include "./BSP/ESP8266/esp8266.h"


  250. #include "./BSP/ADC/adc.h"


  251. //int time=50;
  252. int8_t accel1[3];
  253. int16_t accel[3];
  254. int16_t gyro[3];
  255. int16_t adc_result[ADC_CH_NUM];
  256. //uint8_t ret;

  257. uint8_t *frame = NULL;
  258. uint8_t result;

  259. extern uint16_t g_adc_value[ADC_CH_NUM * ADC_COLL] ;
  260. extern uint16_t g_adc_val[ADC_CH_NUM];
  261. extern uint8_t g_adc_dma_sta;
  262. extern ADC_HandleTypeDef g_adc_handle;   /* ADC句柄 */

  263. int main(void)
  264. {


  265.     HAL_Init();                         /* 初始化HAL库 */
  266.     sys_stm32_clock_init(RCC_PLL_MUL9); /* 设置时钟, 72Mhz */
  267.     delay_init(72);                     /* 延时初始化 */
  268.     usart_init(115200);                 /* 串口初始化为115200 */
  269. //    usmart_dev.init(72);                /* 初始化USMART */
  270. //    led_init();                         /* 初始化LED */

  271. //    key_init();                         /* 初始化按键 */

  272.         
  273.           spi2_init();
  274.           Icm42688_Init();
  275. //          ICM42688_Connect();
  276.         
  277.           esp8266_uart_init(115200);
  278.         
  279.           adc_init();
  280.     adc_dma_init();



  281. //

  282. //    // 等待DMA传输完成
  283. //    while (!g_adc_dma_sta)
  284. //    {
  285. //        // 等待DMA传输完成
  286. //        printf("Waiting for DMA transfer to complete...\r\n");
  287. //    }

  288. //    // 输出第一个ADC值
  289. //    printf("g_adc_value[0]:%d\r\n", g_adc_value[0]);






  290. //    ESP8266_EN(0);
  291. //                result = esp8266_Init(115200);
  292. //                printf("result:%d\r\n",result);
  293. //           result = esp8266_send_at_cmd("AT","OK",500);
  294. //                 printf("result:%d\r\n",result);
  295.         
  296.         
  297. //     连接wifi成功
  298. //           result = esp8266_join_ap("xiao","312312312");
  299. //                 printf("result:%d\r\n",result);
  300. //for(int i = 0; i < 2 ;i++)
  301. //{
  302. //   printf("g_adc_value[%d]:%u\r\n",i,g_adc_val[i]);
  303. //}
  304. //   printf("wraeyrut");
  305. //         printf("g_adc_value");
  306. //         printf("x,y:%u\r\n",g_adc_val[0]);


  307.     while (1)
  308.     {
  309.                         
  310.                 //多通道ADC转化,没有DMA
  311. //                uint16_t adcx;
  312. //    adcx = adc_get_result(adc_result);
  313. //                printf("adcx:%d,adcy:%d,\r\n",adc_result[0],adc_result[1]);
  314.                         

  315.                   printf("adcx:%d,adcy:%d\r\n",g_adc_val[0],g_adc_val[1]);        
  316.                         
  317.                         
  318.                         
  319.                         
  320.                         
  321.                         
  322. //                //测试串口3是否正常
  323. //                //测试串口3是否正常
  324. //                //测试串口3是否正常
  325. //                frame = esp8266_uart_rx_get_frame();
  326. //                if (frame != NULL) {
  327. //        // 处理接收到的数据,这里假设我们要打印接收到的数据
  328. //    esp8266_uart_printf("Received: %s\r\n", frame);

  329. //        // 处理完之后,重置接收状态
  330. //        esp8266_uart_rx_restart(); // 重新启动接收
  331. //    } else {
  332. //        // 如果没有接收到完整数据,可以选择处理超时或者其他逻辑
  333. //        esp8266_uart_printf("No complete frame received yet.\r\n");
  334. //    }
  335.                         
  336. //                          esp8266_uart_printf("AT\r\n");
  337. //                          printf("456\r\n");
  338. //                          delay_ms(1000);
  339. //                                frame = esp8266_uart_rx_get_frame();
  340. //                          printf("frame:%s\r\n",frame);
  341. //                          printf("123\r\n");
  342.                                 
  343.                         
  344.                         
  345.                         

  346.                         //测试ICM42688
  347. //                        ICM42688_Read_Data(accel,gyro);
  348. //                        printf("gyro_x:%d   gyro_y:%d   gyro_z:%d\r\n",gyro[0],gyro[1],gyro[2]);
  349. //                        printf("accel_x:%d  accel_y:%d  accel_z:%d\r\n",accel[0],accel[1],accel[2]);
  350. //                  delay_ms(1000);
  351.                         



  352.     }
  353. }
复制代码
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表