[AT32F435] ADC双Buffer

[复制链接]
 楼主| 我想看大海 发表于 2024-11-16 18:28 | 显示全部楼层 |阅读模式
功能简介  EDMA具备双缓冲模式功能,ADC配合这个功能使用,可有效提高数据转换效率。
本案例将以该EDMA的双缓冲功能为基础,示范如何配置使用ADC的快速转换。


资源准备
1) 硬件环境: 对应产品型号的AT-START BOARD
PA4——3.3V
PA5——0V
PA6——1.5V左右

配置流程
 配置ADC使用的GPIO
 配置用于普通通道数据传输的DMA
 ADC相关配置及设定
 普通通道软触发
 获取转换数据

GPIO配置函数代码
  1. static void gpio_config(void)
  2. {
  3. gpio_init_type gpio_initstructure;
  4. crm_periph_clock_enable(CRM_GPIOA_PERIPH_CLOCK, TRUE);
  5. gpio_default_para_init(&gpio_initstructure);
  6. /* config adc pin as analog input mode */
  7. gpio_initstructure.gpio_mode = GPIO_MODE_ANALOG;
  8. gpio_initstructure.gpio_pins = GPIO_PINS_4 | GPIO_PINS_5 | GPIO_PINS_6;
  9. gpio_init(GPIOA, &gpio_initstructure);
  10. }   
DMA配置函数代码
  1. static void edma_config(void)
  2. {
  3. edma_init_type edma_init_struct;
  4. crm_periph_clock_enable(CRM_EDMA_PERIPH_CLOCK, TRUE);
  5. nvic_irq_enable(EDMA_Stream1_IRQn, 0, 0);
  6. edma_reset(EDMA_STREAM1);
  7.   edma_default_para_init(&edma_init_struct);
  8.   edma_init_struct.buffer_size = 3;
  9.   edma_init_struct.direction = EDMA_DIR_PERIPHERAL_TO_MEMORY;
  10.   edma_init_struct.memory0_base_addr = (uint32_t)adc1_ordinary_valuetab;
  11.   edma_init_struct.memory_burst_mode = EDMA_MEMORY_SINGLE;
  12.   edma_init_struct.memory_data_width = EDMA_MEMORY_DATA_WIDTH_HALFWORD;
  13.   edma_init_struct.memory_inc_enable = TRUE;
  14.   edma_init_struct.peripheral_base_addr = (uint32_t)&(ADC1->odt);
  15.   edma_init_struct.peripheral_burst_mode = EDMA_PERIPHERAL_SINGLE;
  16.   edma_init_struct.peripheral_data_width = EDMA_PERIPHERAL_DATA_WIDTH_HALFWORD;
  17.   edma_init_struct.peripheral_inc_enable = FALSE;
  18.   edma_init_struct.priority = EDMA_PRIORITY_VERY_HIGH;
  19.   edma_init_struct.loop_mode_enable = FALSE;
  20.   edma_init_struct.fifo_threshold = EDMA_FIFO_THRESHOLD_1QUARTER;
  21.   edma_init_struct.fifo_mode_enable = FALSE;
  22.   edma_init(EDMA_STREAM1, &edma_init_struct);

  23.   /* edmamux init and enable */
  24.   edmamux_enable(TRUE);
  25.   edmamux_init(EDMAMUX_CHANNEL1, EDMAMUX_DMAREQ_ID_ADC1);

  26.   /* config double buffer mode */
  27.   edma_double_buffer_mode_init(EDMA_STREAM1, (uint32_t)double_adc1_ordinary_valuetab,
  28. EDMA_MEMORY_0);

  29.   /* enable the double memory mode */
  30.   edma_double_buffer_mode_enable(EDMA_STREAM1, TRUE);

  31.   /* enable edma full data transfer intterrupt */
  32.   edma_interrupt_enable(EDMA_STREAM1, EDMA_FDT_INT, TRUE);
  33.   edma_stream_enable(EDMA_STREAM1, TRUE);
  34. }   
ADC配置函数代码
  1. static void adc_config(void)
  2. {
  3.   adc_common_config_type adc_common_struct;
  4.   adc_base_config_type adc_base_struct;
  5.   crm_periph_clock_enable(CRM_ADC1_PERIPH_CLOCK, TRUE);
  6.   nvic_irq_enable(ADC1_2_3_IRQn, 0, 0);

  7.   adc_common_default_para_init(&adc_common_struct);

  8.   /* config combine mode */
  9. adc_common_struct.combine_mode = ADC_INDEPENDENT_MODE;

  10.   /* config division,adcclk is division by hclk */
  11.   adc_common_struct.div = ADC_HCLK_DIV_17;

  12.   /* config common dma mode,it's not useful in independent mode */
  13.   adc_common_struct.common_dma_mode = ADC_COMMON_DMAMODE_DISABLE;

  14.   /* config common dma request repeat */
  15.   adc_common_struct.common_dma_request_repeat_state = FALSE;

  16.   /* config adjacent adc sampling interval,it's useful for ordinary shifting mode */
  17.   adc_common_struct.sampling_interval = ADC_SAMPLING_INTERVAL_5CYCLES;

  18.   /* config inner temperature sensor and vintrv */
  19.   adc_common_struct.tempervintrv_state = FALSE;

  20.   /* config voltage battery */
  21.   adc_common_struct.vbat_state = FALSE;
  22.   adc_common_config(&adc_common_struct);

  23.   adc_base_default_para_init(&adc_base_struct);

  24.   adc_base_struct.sequence_mode = TRUE;
  25.   adc_base_struct.repeat_mode = FALSE;
  26.   adc_base_struct.data_align = ADC_RIGHT_ALIGNMENT;
  27.   adc_base_struct.ordinary_channel_length = 3;
  28.   adc_base_config(ADC1, &adc_base_struct);
  29.    
  30.   adc_resolution_set(ADC1, ADC_RESOLUTION_12B);

  31.   /* config ordinary channel */
  32.   adc_ordinary_channel_set(ADC1, ADC_CHANNEL_4, 1, ADC_SAMPLETIME_640_5);
  33.   adc_ordinary_channel_set(ADC1, ADC_CHANNEL_5, 2, ADC_SAMPLETIME_640_5);
  34.   adc_ordinary_channel_set(ADC1, ADC_CHANNEL_6, 3, ADC_SAMPLETIME_640_5);

  35.   /* config ordinary trigger source and trigger edge */
  36.   adc_ordinary_conversion_trigger_set(ADC1, ADC_ORDINARY_TRIG_TMR1CH1,
  37. ADC_ORDINARY_TRIG_EDGE_NONE);

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

  40.   /* config dma request repeat,it's not useful when common dma mode is use */
  41.   adc_dma_request_repeat_enable(ADC1, TRUE);
  42. /* enable adc overflow interrupt */
  43.   adc_interrupt_enable(ADC1, ADC_OCCO_INT, TRUE);

  44.   /* adc enable */
  45.   adc_enable(ADC1, TRUE);
  46.   while(adc_flag_get(ADC1, ADC_RDY_FLAG) == RESET);

  47.   /* adc calibration */
  48.   adc_calibration_init(ADC1);
  49.   while(adc_calibration_init_status_get(ADC1));
  50.   adc_calibration_start(ADC1);
  51.   while(adc_calibration_status_get(ADC1));
  52. }   
中断服务函数代码
  1. /* 获取普通通道数据传输完成状态 */
  2. void EDMA_Stream1_IRQHandler(void)
  3. {
  4.   if(edma_flag_get(EDMA_FDT1_FLAG) != RESET)
  5.   {
  6.     if(edma_memory_target_get(EDMA_STREAM1))
  7.     {
  8.       double_buffer_is_useful = 0;
  9.     }
  10.     else
  11.     {
  12.       double_buffer_is_useful = 1;
  13.     }
  14.     edma_flag_clear(EDMA_FDT1_FLAG);
  15.     edma_trans_complete_flag = 1;
  16.   }
  17. }
  18. /* 获取ADC的溢出状态信息 */
  19. void ADC1_2_3_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. }
main函数代码

  1. int main(void)
  2. {
  3.   __IO uint32_t index = 0;
  4.   nvic_priority_group_config(NVIC_PRIORITY_GROUP_4);
  5. /* config the system clock */
  6.   system_clock_config();

  7.   /* init at start board */
  8.   at32_board_init();
  9.   at32_led_off(LED2);
  10.   at32_led_off(LED3);
  11.   at32_led_off(LED4);
  12.   usart1_config(115200);
  13.   gpio_config();
  14.   edma_config();
  15.   adc_config();
  16.   printf("edma_double_buffer \r\n");
  17.   while(1)
  18.   {
  19.     adc_ordinary_software_trigger_enable(ADC1, TRUE);
  20.     while(edma_trans_complete_flag == 0);
  21.     edma_trans_complete_flag = 0;
  22.     if(adc1_overflow_flag != 0)
  23.     {
  24.       /* printf flag when error occur */
  25.       at32_led_on(LED4);
  26.       printf("error occur\r\n");
  27.       printf("adc1_overflow_flag = %d\r\n",adc1_overflow_flag);
  28.     }
  29.     else
  30.     {
  31.       printf("conversion end without error\r\n");
  32.       /* printf data when conversion end without error */
  33.       if(double_buffer_is_useful == 1)
  34.       {
  35.         printf("double_adc1_ordinary_valuetab[0] = 0x%x\r\n", double_adc1_ordinary_valuetab[0]);
  36.         printf("double_adc1_ordinary_valuetab[1] = 0x%x\r\n", double_adc1_ordinary_valuetab[1]);
  37.         printf("double_adc1_ordinary_valuetab[2] = 0x%x\r\n", double_adc1_ordinary_valuetab[2]);
  38.       }
  39.       else
  40.       {
  41.         printf("adc1_ordinary_valuetab[0] = 0x%x\r\n", adc1_ordinary_valuetab[0]);
  42.         printf("adc1_ordinary_valuetab[1] = 0x%x\r\n", adc1_ordinary_valuetab[1]);
  43.         printf("adc1_ordinary_valuetab[2] = 0x%x\r\n", adc1_ordinary_valuetab[2]);
  44.       }
  45.     }
  46.     printf("\r\n");
  47.     delay_sec(1);
  48.   }
  49. }  
实验效果  可通过串口打印查看实现效果,如下图所示,ADC转换数据被硬件循环的存储到EDMA设定的两个buffer 内了。
串口配置
Baud rate:115200
Stop bits:1
Data bits:8
Parity:None

ADC双Buffer实验结果
3498673873c23630a.png


黑心单片机 发表于 2024-11-16 19:26 | 显示全部楼层
这个代码挺详细的,就是没注释太少了,基础差,看的有点费劲。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

36

主题

282

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部

36

主题

282

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部