[AT32F423] AT32F423多路ADC测量

[复制链接]
 楼主| lulugl 发表于 2023-11-5 17:47 | 显示全部楼层 |阅读模式

【AT32F423ADC简介】
ADC 是一个将模拟输入信号转换为 12 位、10 位、8 位或 6 位的数字信号的外设。采样率最高可达 5.33MSPS。多达 26 个通道源可进行采样及转换。
【AT32F423ADC主要特征】
模拟方面有以下特征:
  • 支持分辨率 12 位、10 位、8 位或 6 位的转换
  • 自校准时间: 205 个 ADC 时钟周期
  • ADC 转换时间
  • 快速通道: ADC时钟80MHz,分辨率12位时转换时间为0.1875 μs(5.33MSps)
  • 慢速通道: ADC时钟80MHz,分辨率12位时转换时间为0.2375 μs(4.21MSps)
  • 快速通道: ADC时钟80MHz,分辨率6位时转换时间为0.1126 μs(8.88MSps)
  • 慢速通道: ADC时钟80MHz,分辨率6位时转换时间为0.1626 μs(6.15MSps)

ADC 供电要求:2.4V 到 3.6V 
ADC 输入范围:VREF- ≤ VIN ≤ VREF+ 数字控制方面有以下特征:
  • 通道管理区分优先权不同的普通通道与抢占通道
  • 普通通道与抢占通道具备各自独立的触发侦测电路
  • 各通道均可独立配置采样时间
  • 转换顺序管理支持多种不同的多通道转换
  • 过采样器:硬件过采样最高可实现 16 位分辨率
  • 可选择的数据对齐方式
  • 可配置的电压监测边界
  • 支持 DMA 传输的普通通道数据
  • 可设定以下事件发生时响应中断


― 普通通道转换数据溢出
― 抢占通道组转换结束
― 普通通道转换结束

― 电压监测超出范围
【通道管理】
模拟信号通道输入 每个 ADC 拥有多达 26 个模拟信号通道输入,以 ADC_INx 表示,x=0 至 17、20 至 27。ADC_IN0 至 ADC_IN15 为外部模拟输入,ADC_IN16 为内部温度传感器,ADC_IN17 为内部参考电压,ADC_IN20 至 ADC_IN27 为外部模拟输入。
通道转换
AT32F423系列 技术手册2023.08.02 第 342 页 版本 2.02 转换区分为普通通道转换与抢占通道转换,抢占通道的转换优先权高于普通通道。 抢占通道触发若发生于普通通道转换途中,优先进行抢占通道的转换,普通通道于抢占通道转换结束后重 新开始转换被打断的通道。普通通道触发若发生于抢占通道转换途中,普通通道的转换会等待抢占通道转 换完成后才开始。
将通道(ADC_INx)编排进普通通道序列(ADC_OSQx)以及抢占通道序列(ADC_PSQ),相同通道可 重复编排,序列总数由 OCLEN 与 PCLEN 定义,接着即可启动普通通道转换或抢占通道转换。
【ADC操作流程】
ADC 的基础操作流程如下图所示,建议第一次上电后进行校准,以提升采样与转换准确度。待校准完成 后可靠触发引起 ADC 采样转换,转换结束后即可读取数据。
bf124ffe3d91b60fde8d63ff917ff54d
【代码的实现】
此次实验转换,我选择了6个输入源,分别为ADC1的4-9共6个通道。对应的IO分别为PA4-7,PB0-1,对应为ADC1的通道4-9。
gpio的配置,配置IO为模式为GPIO_MODE_ANALOG。同时使能PA、PB端口的时钟,具体代码如下:
  1. static void gpio_config(void)
  2. {
  3.   gpio_init_type gpio_initstructure;
  4.   crm_periph_clock_enable(CRM_GPIOA_PERIPH_CLOCK, TRUE);
  5.   crm_periph_clock_enable(CRM_GPIOB_PERIPH_CLOCK, TRUE);

  6.   gpio_default_para_init(&gpio_initstructure);

  7.   /* config adc pin as analog input mode */
  8.   gpio_initstructure.gpio_mode = GPIO_MODE_ANALOG;
  9.   gpio_initstructure.gpio_pins = GPIO_PINS_4 | GPIO_PINS_5 | GPIO_PINS_6 | GPIO_PINS_7;
  10.   gpio_init(GPIOA, &gpio_initstructure);

  11.   gpio_initstructure.gpio_mode = GPIO_MODE_ANALOG;
  12.   gpio_initstructure.gpio_pins = GPIO_PINS_0 | GPIO_PINS_1;
  13.   gpio_init(GPIOB, &gpio_initstructure);
  14. }

【ADC的配置】
首先使能ADC1的时钟源CRM_ADC1_PERIPH_CLOCK,选择时钟源为CRM_ADC_CLOCK_SOURCE_HCLK;分频为ADC_HCLK_DIV_4。使能DMA,转换结果右对齐,连续转换。配置6个通道。具体代码如下:
  1. /**
  2.   * [url=home.php?mod=space&uid=247401]@brief[/url]  adc configuration.
  3.   * @param  none
  4.   * @retval none
  5.   */
  6. static void adc_config(void)
  7. {
  8.   adc_common_config_type adc_common_struct;
  9.   adc_base_config_type adc_base_struct;
  10.   crm_periph_clock_enable(CRM_ADC1_PERIPH_CLOCK, TRUE);
  11.   nvic_irq_enable(ADC1_IRQn, 0, 0);
  12.   crm_adc_clock_select(CRM_ADC_CLOCK_SOURCE_HCLK);

  13.   adc_common_default_para_init(&adc_common_struct);

  14.   /* config division,adcclk is division by hclk */
  15.   adc_common_struct.div = ADC_HCLK_DIV_4;

  16.   /* config inner temperature sensor and vintrv */
  17.   adc_common_struct.tempervintrv_state = FALSE;

  18.   adc_common_config(&adc_common_struct);

  19.   adc_base_default_para_init(&adc_base_struct);

  20.   adc_base_struct.sequence_mode = TRUE;
  21.   adc_base_struct.repeat_mode = FALSE;
  22.   adc_base_struct.data_align = ADC_RIGHT_ALIGNMENT;
  23.   adc_base_struct.ordinary_channel_length = 6;
  24.   adc_base_config(ADC1, &adc_base_struct);
  25.   adc_resolution_set(ADC1, ADC_RESOLUTION_12B);

  26.   /* config ordinary channel */
  27.   adc_ordinary_channel_set(ADC1, ADC_CHANNEL_4, 1, ADC_SAMPLETIME_6_5);
  28.   adc_ordinary_channel_set(ADC1, ADC_CHANNEL_5, 2, ADC_SAMPLETIME_6_5);
  29.   adc_ordinary_channel_set(ADC1, ADC_CHANNEL_6, 3, ADC_SAMPLETIME_6_5);
  30.         adc_ordinary_channel_set(ADC1, ADC_CHANNEL_7, 4, ADC_SAMPLETIME_6_5);
  31.   adc_ordinary_channel_set(ADC1, ADC_CHANNEL_8, 5, ADC_SAMPLETIME_6_5);
  32.   adc_ordinary_channel_set(ADC1, ADC_CHANNEL_9, 6, ADC_SAMPLETIME_6_5);

  33.   /* config ordinary trigger source and trigger edge */
  34.   adc_ordinary_conversion_trigger_set(ADC1, ADC_ORDINARY_TRIG_TMR1CH1, ADC_ORDINARY_TRIG_EDGE_NONE);

  35.   /* config dma mode,it's not useful when common dma mode is use */
  36.   adc_dma_mode_enable(ADC1, TRUE);

  37.   /* config dma request repeat,it's not useful when common dma mode is use */
  38.   adc_dma_request_repeat_enable(ADC1, TRUE);

  39.   /* enable adc overflow interrupt */
  40.   adc_interrupt_enable(ADC1, ADC_OCCO_INT, TRUE);

  41.   /* set oversampling ratio and shift */
  42.   adc_oversample_ratio_shift_set(ADC1, ADC_OVERSAMPLE_RATIO_8, ADC_OVERSAMPLE_SHIFT_3);
  43.   /* disable ordinary oversampling trigger mode */
  44.   adc_ordinary_oversample_trig_enable(ADC1, FALSE);
  45.   /* set ordinary oversample restart mode */
  46.   adc_ordinary_oversample_restart_set(ADC1, ADC_OVERSAMPLE_CONTINUE);
  47.   /* enable ordinary oversampling */
  48.   adc_ordinary_oversample_enable(ADC1, TRUE);



  49.   /* adc enable */
  50.   adc_enable(ADC1, TRUE);
  51.   while(adc_flag_get(ADC1, ADC_RDY_FLAG) == RESET);

  52.   /* adc calibration */
  53.   adc_calibration_init(ADC1);
  54.   while(adc_calibration_init_status_get(ADC1));
  55.   adc_calibration_start(ADC1);
  56.   while(adc_calibration_status_get(ADC1));
  57. }

【DMA配置】
使用能dma通道1的时钟,使能DMA1的中断,设置dma的缓冲为30,即是6个通道接收5次数据。设置地址为结构体adc1_ordinary_valuetab的地址,使用DMA1中断,具体代码如下:
  1. /**
  2.   * @brief  dma configuration.
  3.   * @param  none
  4.   * @retval none
  5.   */
  6. static void dma_config(void)
  7. {
  8.   dma_init_type dma_init_struct;
  9.   crm_periph_clock_enable(CRM_DMA1_PERIPH_CLOCK, TRUE);
  10.   nvic_irq_enable(DMA1_Channel1_IRQn, 0, 0);

  11.   dma_reset(DMA1_CHANNEL1);
  12.   dma_default_para_init(&dma_init_struct);
  13.   dma_init_struct.buffer_size = 30;
  14.   dma_init_struct.direction = DMA_DIR_PERIPHERAL_TO_MEMORY;
  15.   dma_init_struct.memory_base_addr = (uint32_t)adc1_ordinary_valuetab;
  16.   dma_init_struct.memory_data_width = DMA_MEMORY_DATA_WIDTH_HALFWORD;
  17.   dma_init_struct.memory_inc_enable = TRUE;
  18.   dma_init_struct.peripheral_base_addr = (uint32_t)&(ADC1->odt);
  19.   dma_init_struct.peripheral_data_width = DMA_PERIPHERAL_DATA_WIDTH_HALFWORD;
  20.   dma_init_struct.peripheral_inc_enable = FALSE;
  21.   dma_init_struct.priority = DMA_PRIORITY_HIGH;
  22.   dma_init_struct.loop_mode_enable = TRUE;
  23.   dma_init(DMA1_CHANNEL1, &dma_init_struct);

  24.   dmamux_enable(DMA1, TRUE);
  25.   dmamux_init(DMA1MUX_CHANNEL1, DMAMUX_DMAREQ_ID_ADC1);

  26.   /* enable dma transfer complete interrupt */
  27.   dma_interrupt_enable(DMA1_CHANNEL1, DMA_FDT_INT, TRUE);
  28.   dma_channel_enable(DMA1_CHANNEL1, TRUE);
  29. }

at32f423_int.c中,更新两个标志,同时清除中断标志:
  1. /**
  2.   * @brief  this function handles dma1_channel1 handler.
  3.   * @param  none
  4.   * @retval none
  5.   */
  6. void DMA1_Channel1_IRQHandler(void)
  7. {
  8.   if(dma_flag_get(DMA1_FDT1_FLAG) != RESET)
  9.   {
  10.     dma_flag_clear(DMA1_FDT1_FLAG);
  11.     dma_trans_complete_flag = 1;
  12.   }
  13. }

  14. /**
  15.   * @brief  this function handles adc1_2_3 handler.
  16.   * @param  none
  17.   * @retval none
  18.   */
  19. void ADC1_IRQHandler(void)
  20. {
  21.   if(adc_flag_get(ADC1, ADC_OCCO_FLAG) != RESET)
  22.   {
  23.     adc_flag_clear(ADC1, ADC_OCCO_FLAG);
  24.     adc1_overflow_flag++;
  25.   }

  26. }

5、采集处理:转换5次后,把结构体中的数据取出来,并取平均值,然后显示到OLED上,代码如下:
  1. int main(void)
  2. {
  3.   __IO uint32_t index = 0;
  4.         uint32_t val[6]={0};
  5.         uint8_t show_str[20] = {0};
  6.   nvic_priority_group_config(NVIC_PRIORITY_GROUP_4);

  7.   /* config the system clock */
  8.   system_clock_config();

  9.   /* init at start board */
  10.   at32_board_init();
  11.   at32_led_off(LED2);
  12.   at32_led_off(LED3);
  13.   at32_led_off(LED4);
  14.   uart_print_init(115200);
  15.   gpio_config();
  16.   dma_config();
  17.   adc_config();
  18.   printf("tmr_trigger_automatic_preempt \r\n");


  19.   at32_led_on(LED2);
  20.         OLED_Init();
  21.         OLED_Clear(1);
  22.         GUI_ShowCHinese(24,16,16,"雅特力科技",0);
  23.         GUI_ShowString(30,32,"AT32F423",16,1);
  24.         OLED_Display();
  25.   while(1)
  26.   {
  27.                  /* adc1 software trigger start conversion */
  28.   for(index = 0; index < 5; index++)
  29.   {
  30.     adc_ordinary_software_trigger_enable(ADC1, TRUE);
  31.                
  32.     delay_ms(100);
  33.   }
  34.   if((dma_trans_complete_flag == 0) || (adc1_overflow_flag != 0))
  35.   {
  36.     /* printf flag when error occur */
  37.     at32_led_on(LED3);
  38.     at32_led_on(LED4);
  39.     printf("error occur\r\n");
  40.     printf("adc1_overflow_flag = %d\r\n",adc1_overflow_flag);
  41.     printf("dma_trans_complete_flag = %d\r\n",dma_trans_complete_flag);
  42.   }
  43.   else
  44.   {
  45.     /* printf data when conversion end without error */
  46.     printf("conversion end without error\r\n");
  47.     for(index = 0; index < 5; index++)
  48.     {
  49.       printf("adc1_ordinary_valuetab[%d][0] = 0x%x\r\n", index, adc1_ordinary_valuetab[index][0]);
  50.       printf("adc1_ordinary_valuetab[%d][1] = 0x%x\r\n", index, adc1_ordinary_valuetab[index][1]);
  51.       printf("adc1_ordinary_valuetab[%d][2] = 0x%x\r\n", index, adc1_ordinary_valuetab[index][2]);
  52.       printf("adc1_ordinary_valuetab[%d][3] = 0x%x\r\n", index, adc1_ordinary_valuetab[index][3]);
  53.       printf("adc1_ordinary_valuetab[%d][4] = 0x%x\r\n", index, adc1_ordinary_valuetab[index][4]);
  54.       printf("adc1_ordinary_valuetab[%d][5] = 0x%x\r\n", index, adc1_ordinary_valuetab[index][5]);
  55.       printf("\r\n");
  56.                         val[0] += adc1_ordinary_valuetab[index][0];
  57.                         val[1] += adc1_ordinary_valuetab[index][1];
  58.                         val[2] += adc1_ordinary_valuetab[index][2];
  59.                         val[3] += adc1_ordinary_valuetab[index][3];
  60.                         val[4] += adc1_ordinary_valuetab[index][4];
  61.                         val[5] += adc1_ordinary_valuetab[index][5];
  62.     }
  63.   }
  64.         OLED_Clear(1);
  65.         GUI_ShowCHinese(24,0,16,"雅特力科技",1);
  66.         sprintf(show_str,"1:%1.2fV",((float)(val[0]/5) * 3.3)/4095);
  67.         GUI_ShowString(0,16,show_str,16,0);
  68.         sprintf(show_str,"2:%1.2fV",((float)(val[1]/5) * 3.3)/4095);
  69.         GUI_ShowString(0,32,show_str,16,0);
  70.         sprintf(show_str,"3:%1.2fV",((float)(val[2]/5) * 3.3)/4095);
  71.         GUI_ShowString(0,48,show_str,16,0);
  72.         sprintf(show_str,"4:%1.2fV",((float)(val[3]/5) * 3.3)/4095);
  73.         GUI_ShowString(63,16,show_str,16,0);
  74.         sprintf(show_str,"5:%1.2fV",((float)(val[4]/5) * 3.3)/4095);
  75.         GUI_ShowString(63,32,show_str,16,0);
  76.         sprintf(show_str,"6:%1.2fV",((float)(val[5]/5) * 3.3)/4095);
  77.         GUI_ShowString(63,48,show_str,16,0);
  78.         OLED_Display();
  79.         val[0] = 0;
  80.         val[1] = 0;
  81.         val[2] = 0;
  82.         val[3] = 0;
  83.         val[4] = 0;
  84.         val[5] = 0;
  85.         dma_trans_complete_flag = 0;
  86.        
  87.   }
  88. }
【实验效果】
电压测量.jpg
https://www.bilibili.com/video/BV1Dc41197Ak/?vd_source=e1bd226340c8b87027d5dcfc6b0c3344

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

本版积分规则

180

主题

830

帖子

12

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