[AT32F403/403A] 403A-定时器触发双ADC-同步规则模式,跑不通

[复制链接]
 楼主| 寻觅左岸 发表于 2021-5-18 13:52 | 显示全部楼层 |阅读模式
  1. #include "dual_adc_test.h"
  2. #include "at32f4xx.h"
  3. #include "stdio.h"
  4. #include "at32_board.h"

  5. /************************************************
  6. ALIENTEK 战舰AT32F403开发板实验
  7. ADC 实验(ADC1/2/3同时使用)
  8. 说明1:ADC1和ADC2采用同步规则模式,
  9. 说明2:ADC配置为扫描模式、非连续转换、软触发、右对齐、使用DMA模式、通道采样时间1.5个周期
  10. 说明3:ADC1规则组定义两个通道(通道0、通道2),ADC2定义两个通道(通道1、通道3)
  11. 说明4:ADC1通过DMA1的通道1传输数据,ADC2利用双ADC由ADC1的DMA1的通道1传输数据,
  12. 说明5:所有ADC转换数据均在DMA1传输完成中断内进行打印

  13. ************************************************/
  14. #define ADC1_DR_Address    ((uint32_t)0x4001244C)
  15. #define ADC3_DR_Address    ((uint32_t)0x40013C4C)
  16. #define dual_adc_ADC_size     10

  17. #define M 128
  18. #define N 8
  19. uint16_t   value[N][M];
  20. uint32_t ADC_ConvertedValue1;  
  21. uint32_t ADC_ConvertedValue2;

  22. ADC_InitType ADC_InitStructure;
  23. DMA_InitType DMA_InitStructure;
  24. TMR_TimerBaseInitType TMR_TimerBaseInitStructure;
  25. TMR_OCInitType TMR_OCInitStructure;

  26. __IO uint32_t dual_adc_ADC_DualConvertedValueTab[dual_adc_ADC_size];

  27. void dual_adc_RCC_Configuration(void)
  28. {
  29.         /* ADCCLK = PCLK2/6 =120/6=20M */
  30.         RCC_ADCCLKConfig(RCC_APB2CLK_Div6);

  31.         /* Enable DMA1 clock */
  32.         RCC_AHBPeriphClockCmd(RCC_AHBPERIPH_DMA1, ENABLE);

  33.         /* Enable ADC1, ADC2 and GPIOC clock */
  34.         RCC_APB2PeriphClockCmd(        RCC_APB2PERIPH_ADC1 | RCC_APB2PERIPH_ADC2 | RCC_APB2PERIPH_GPIOA, ENABLE);
  35. }
  36. void dual_adc_GPIO_Configuration(void)
  37. {
  38.         GPIO_InitType GPIO_InitStructure;

  39.         /* Configure PA0, PA1 PA2, PA3
  40.         as analog input ----------------------------------------------------------*/
  41.         GPIO_StructInit(&GPIO_InitStructure);
  42.         GPIO_InitStructure.GPIO_Pins = GPIO_Pins_0|GPIO_Pins_1 | GPIO_Pins_2 | GPIO_Pins_3;
  43.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_ANALOG;
  44.         GPIO_Init(GPIOA, &GPIO_InitStructure);
  45. }
  46. void dual_adc_NVIC_Configuration(void)
  47. {
  48.         NVIC_InitType NVIC_InitStructure;

  49.     NVIC_InitStructure.NVIC_IRQChannel = ADC1_2_IRQn;
  50.         NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
  51.         NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
  52.         NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  53.         NVIC_Init(&NVIC_InitStructure);

  54.         NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel1_IRQn;
  55.         NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
  56.         NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
  57.         NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  58.         NVIC_Init(&NVIC_InitStructure);
  59. }
  60. void dual_adc_DMA_Configuration(void)
  61. {
  62.         /* DMA1 channel1 configuration ----------------------------------------------*/
  63.         DMA_Reset(DMA1_Channel1);
  64.         DMA_DefaultInitParaConfig(&DMA_InitStructure);
  65.         DMA_InitStructure.DMA_PeripheralBaseAddr    = ADC1_DR_Address;
  66.         DMA_InitStructure.DMA_MemoryBaseAddr        = (uint32_t)dual_adc_ADC_DualConvertedValueTab;
  67.         DMA_InitStructure.DMA_Direction             = DMA_DIR_PERIPHERALSRC;
  68.         DMA_InitStructure.DMA_BufferSize            = 2;//pa0 pa2 两个通道
  69.         DMA_InitStructure.DMA_PeripheralInc         = DMA_PERIPHERALINC_DISABLE;
  70.         DMA_InitStructure.DMA_MemoryInc             = DMA_MEMORYINC_ENABLE;
  71.         DMA_InitStructure.DMA_PeripheralDataWidth   = DMA_PERIPHERALDATAWIDTH_WORD;//一定要word,因为RM手册有说ADC2
  72.         DMA_InitStructure.DMA_MemoryDataWidth       = DMA_MEMORYDATAWIDTH_WORD;
  73.         DMA_InitStructure.DMA_Mode                  = DMA_MODE_CIRCULAR;
  74.         DMA_InitStructure.DMA_Priority              = DMA_PRIORITY_HIGH;
  75.         DMA_InitStructure.DMA_MTOM                  = DMA_MEMTOMEM_DISABLE;
  76.         DMA_Init(DMA1_Channel1, &DMA_InitStructure);
  77.    
  78.     DMA_INTConfig(DMA1_Channel1, DMA_INT_TC, ENABLE);//实际应用不加DMA中断,这里只做打印测试
  79.    
  80.         /* Enable DMA1 Channel1 */
  81.         DMA_ChannelEnable(DMA1_Channel1, ENABLE);
  82. }
  83. void dual_adc_Config_Configuration(void)
  84. {
  85.     //ADC_Reset(ADC1);
  86.     //ADC_Reset(ADC2);
  87.         /* ADC1 configuration ------------------------------------------------------*/
  88.         ADC_StructInit(&ADC_InitStructure);
  89.         ADC_InitStructure.ADC_Mode              = ADC_Mode_RegSimult;
  90.         ADC_InitStructure.ADC_ScanMode          = ENABLE;
  91.         ADC_InitStructure.ADC_ContinuousMode    = DISABLE;
  92.         ADC_InitStructure.ADC_ExternalTrig      = ADC_ExternalTrig_TMR1_CC3;//TIM1-CC3触发
  93.         ADC_InitStructure.ADC_DataAlign         = ADC_DataAlign_Right;
  94.         ADC_InitStructure.ADC_NumOfChannel      = 2;
  95.         ADC_Init(ADC1, &ADC_InitStructure);
  96.         /* ADC1 regular channels configuration */
  97.         ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_1_5);    //PA0
  98.         ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 2, ADC_SampleTime_1_5);    //PA2
  99.     ADC_ExternalTrigConvCtrl(ADC1, ENABLE);
  100.      
  101.         /* ADC2 configuration ------------------------------------------------------*/
  102.         ADC_StructInit(&ADC_InitStructure);
  103.         ADC_InitStructure.ADC_Mode              = ADC_Mode_RegSimult;
  104.         ADC_InitStructure.ADC_ScanMode          = ENABLE;
  105.         ADC_InitStructure.ADC_ContinuousMode    = DISABLE;
  106.         ADC_InitStructure.ADC_ExternalTrig      = ADC_ExternalTrig_TMR1_CC3;//TIM1-CC3触发
  107.         ADC_InitStructure.ADC_DataAlign         = ADC_DataAlign_Right;
  108.         ADC_InitStructure.ADC_NumOfChannel      = 2;
  109.         ADC_Init(ADC2, &ADC_InitStructure);
  110.         /* ADC2 regular channels configuration */
  111.         ADC_RegularChannelConfig(ADC2, ADC_Channel_1, 1, ADC_SampleTime_1_5);//PA1
  112.         ADC_RegularChannelConfig(ADC2, ADC_Channel_3, 2, ADC_SampleTime_1_5);//PA3
  113.         ADC_ExternalTrigConvCtrl(ADC2, ENABLE);

  114.     ADC_DMACtrl(ADC1, ENABLE);
  115.         /* Enable ADC1 */
  116.         ADC_Ctrl(ADC1, ENABLE);
  117.    
  118.         /* Enable ADC2 */
  119.         ADC_Ctrl(ADC2, ENABLE);//根据移植手册,需要把这句话放校准前面,不然程序会卡在CAL异常,实测必须放前面
  120.        
  121.         //ADC_Ctrl(ADC1, ENABLE);  
  122.         ADC_RstCalibration(ADC1);
  123.         while(ADC_GetResetCalibrationStatus(ADC1));
  124.         ADC_StartCalibration(ADC1);
  125.         while(ADC_GetCalibrationStatus(ADC1));

  126.         //ADC_Ctrl(ADC2, ENABLE);
  127.         ADC_RstCalibration(ADC2);
  128.         while(ADC_GetResetCalibrationStatus(ADC2));
  129.         ADC_StartCalibration(ADC2);
  130.         while(ADC_GetCalibrationStatus(ADC2));
  131.    
  132.    
  133. }
  134. //定时器1-cc3的设置触发
  135. void dual_adc_tim_init(void)
  136. {
  137.     /* Time Base configuration */
  138.         TMR_TimeBaseStructInit(&TMR_TimerBaseInitStructure);
  139.         TMR_TimerBaseInitStructure.TMR_Period            = (1500000/15000)-1;                                           // 15KHz      
  140.         TMR_TimerBaseInitStructure.TMR_DIV               = (SystemCoreClock/1500000)-1;        // timer1_clk=1.5MHz      
  141.         TMR_TimerBaseInitStructure.TMR_CounterMode = TMR_CounterDIR_Up;
  142.         TMR_TimerBaseInitStructure.TMR_ClockDivision = TMR_CKD_DIV1;
  143.         TMR_TimerBaseInitStructure.TMR_RepetitionCounter = 0;
  144.         TMR_TimeBaseInit(TMR1,&TMR_TimerBaseInitStructure);

  145.     TMR_OCStructInit(&TMR_OCInitStructure);
  146.         TMR_OCInitStructure.TMR_Pulse = 10;
  147.         TMR_OCInitStructure.TMR_OCMode = TMR_OCMode_PWM1;
  148.         TMR_OCInitStructure.TMR_OCIdleState = TMR_OCIdleState_Reset;
  149.         TMR_OCInitStructure.TMR_OCNIdleState = TMR_OCNIdleState_Reset;
  150.         TMR_OCInitStructure.TMR_OCPolarity = TMR_OCPolarity_High;
  151.         TMR_OCInitStructure.TMR_OCNPolarity = TMR_OCNPolarity_High;
  152.         TMR_OCInitStructure.TMR_OutputState = TMR_OutputState_Enable;
  153.         TMR_OCInitStructure.TMR_OutputNState = TMR_OutputNState_Disable;
  154.         TMR_OC3Init(TMR1,&TMR_OCInitStructure);
  155.        
  156.         TMR_CtrlPWMOutputs(TMR1,ENABLE);
  157.        
  158.         TMR_Cmd(TMR1,ENABLE);   
  159. }

  160. u16 Get_Adc(u8 ch)   
  161. {
  162.     ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_13_5);

  163.         ADC_SoftwareStartConvCtrl(ADC1, ENABLE);                //使能指定的ADC1的软件转换启动功能       
  164.          
  165.         while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EC ));//等待转换结束

  166.         return ADC_GetConversionValue(ADC1);        //返回最近一次ADC1规则组的转换结果
  167. }
  168. void dual_adc_read_adc(void)
  169. {
  170.     //uint16_t adc1_val[2];
  171.     //
  172.     //uint16_t adc2_val[2];
  173.     //ADC_SoftwareStartConvCtrl(ADC1, ENABLE);
  174.     //ADC_SoftwareStartConvCtrl(ADC2, ENABLE);
  175.     //
  176.     //adc1_val[0] =dual_adc_ADC_DualConvertedValueTab[0]&0xffff;//A1
  177.         //adc1_val[1] =(dual_adc_ADC_DualConvertedValueTab[0]>>16)&0xffff;
  178.     //
  179.         //
  180.         //adc2_val[0] =(int)(dual_adc_ADC_DualConvertedValueTab[0]>>16);
  181.         //adc2_val[1] =(int)(dual_adc_ADC_DualConvertedValueTab[1]>>16);
  182.     //
  183.     //printf("adc1-0:%d\n\r",adc1_val[0] );
  184.     //printf("adc1-1:%d\n\r",adc1_val[1]);
  185.         int i;
  186.      printf("\r\n 采样开始\r\n");
  187.          for(i=0;i<M;i++)
  188.                 {
  189.                         /* Start ADC1 Software Conversion  启动ADC1软件转换 */                
  190. //                        ADC_SoftwareStartConvCtrl(ADC1, ENABLE);  //开始转换
  191. //                       
  192. //            //ADC_GetDualModeConversionValue
  193. //            
  194. //                        ADC_ConvertedValue1=ADC_GetConversionValue(ADC1);
  195. //                       
  196. //                       
  197. //                        value[0][0] = (ADC_ConvertedValue1&0xffff); //获取ADC的值
  198. //                       value[1][1] = ((ADC_ConvertedValue1>> 16)&0xffff);  //获取ADC的值

  199.                         ADC_ConvertedValue1 = (u32)Get_Adc(ADC_Channel_0);
  200.                         ADC_ConvertedValue2 = (u32)Get_Adc(ADC_Channel_2);
  201.                        
  202.                         printf("\r\n 编号%d  \t 编号%d  \t AD值: %d\r\n", 0,i,  ADC_ConvertedValue1);
  203.                         printf("\r\n 编号%d  \t 编号%d  \t AD值: %d\r\n", 1,i,  ADC_ConvertedValue2);
  204.                                        
  205.                 }
  206.                  printf("\r\n 采样结束\r\n");
  207. }

  208. void DMA1_Channel1_IRQHandler(void)
  209. {
  210.         if ( DMA_GetITStatus(DMA1_INT_TC1) == SET)
  211.         {
  212.                 printf("DMA1_Enter_IRQ\r\n");
  213.         printf("ADC1_Channel0_Value=%x\r\n", dual_adc_ADC_DualConvertedValueTab[0] & 0xFFFF);
  214.                 printf("ADC2_Channel2_Value=%x\r\n", dual_adc_ADC_DualConvertedValueTab[0] >> 16);
  215.                 printf("ADC1_Channel1_Value=%x\r\n", dual_adc_ADC_DualConvertedValueTab[1] & 0xFFFF);
  216.                 printf("ADC2_Channel3_Value=%x\r\n", dual_adc_ADC_DualConvertedValueTab[1] >> 16);
  217.                 DMA_ClearITPendingBit(DMA1_INT_TC1);
  218.         }
  219. }
  220. void ADC1_2_IRQHandler(void)
  221. {
  222.         if(ADC_GetINTStatus(ADC1, ADC_INT_EC) == SET)
  223.         {
  224.                 printf("ADC1 ADC_GetConversionValue: %d\r\n", ADC_GetConversionValue(ADC1));
  225.                 ADC_ClearINTPendingBit(ADC1, ADC_INT_JEC);
  226.         }
  227.         if(ADC_GetINTStatus(ADC2, ADC_INT_EC) == SET)
  228.         {
  229.                 printf("ADC2 ADC_GetConversionValue: %d\r\n", ADC_GetConversionValue(ADC2));
  230.                 ADC_ClearINTPendingBit(ADC1, ADC_INT_EC);//这里为什么demo是ADC1
  231.         }
  232. }
  233. void dual_adc_main_init(void)//main函数初始化
  234. {
  235.     dual_adc_RCC_Configuration();
  236.    
  237.     dual_adc_GPIO_Configuration();
  238.    
  239.     dual_adc_NVIC_Configuration();
  240.    
  241.     dual_adc_DMA_Configuration();
  242.    
  243.     dual_adc_Config_Configuration();   
  244.    
  245.     dual_adc_tim_init();   

  246.     printf("Test_Start\r\n");   
  247.     Delay_ms(1000);   
  248. }


 楼主| 寻觅左岸 发表于 2021-5-18 13:55 | 显示全部楼层
采集的双ADC模式下,ADC1和ADC的通道都不对,这个代码是根据AT的文档FAQ0023修改的
hoop 发表于 2021-5-18 18:03 | 显示全部楼层
本帖最后由 hoop 于 2021-5-18 18:04 编辑

ADC1和ADC的通道都不对——是数据出现错位了吗?还是AD值不准有一定偏差?
如下几点代码改动建议
1)配置顺序做一下调整,先初始化TMR(但不使能),再初始化DMA和ADC,最后再使能TMR——因为TMR初始化时会输出一次非期望的触发事件
2)ADC2的ADC_InitStructure.ADC_ExternalTrig      = ADC_ExternalTrig_TMR1_CC3;修改为ADC_ExternalTrig_None——双ADC模式下要禁止从ADC的触发,不然容易丢失同步规则
3)增加TMR的CH3使能,即在TMR使能的地方增加代码TMR_CCxCmd(TMR1,TMR_Channel_3,ENABLE);   
4)如果应用需要TMR触发转换的话,除非调试定位问题,否则就不要再写软触发命令ADC_SoftwareStartConvCtrl。
勇敢的大白菜 发表于 2021-5-18 23:38 | 显示全部楼层
感谢楼主的分享,这么长的代码。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

8

主题

15

帖子

1

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