TRGO控制ADC多通道+DMA转运

# 01.描述

- 该段程序意在可以快速复制使用ADC多通道采集,优势是使用定时器触发ADC的4个通道进行采集,适用于时序要求高的情况,使用的STM32F103C8T6,最终保存在AD_Value数组里面。要注意的是没有ADC_ExternalTrigConv_T2_TRGO,搞了半天的定时器2的TRGO,结果是不支持。

​#include "stm32f10x.h"                  // Device header
#include "ad.h"

uint16_t AD_Value[4]; // 存放AD转换结果的全局数组
void TIM3_TRGO_Init(void)
{
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct = {0};
    TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;
    TIM_TimeBaseInitStruct.TIM_Period = 99; // 10ms
    TIM_TimeBaseInitStruct.TIM_Prescaler = 71;
    TIM_TimeBaseInitStruct.TIM_RepetitionCounter = 0;
    TIM_TimeBaseInit(TIM3, &TIM_TimeBaseInitStruct);
    TIM_SelectOutputTrigger(TIM3, TIM_TRGOSource_Update);
    TIM_Cmd(TIM3, ENABLE);
}

void ADC1_Init()
{
    TIM3_TRGO_Init(); // 定时器初始化
    ADC_InitTypeDef ADC_InitStructure={0};
    GPIO_InitTypeDef GPIO_InitStructure={0};
    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); // 开启DMA1时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_ADC1, ENABLE); // 开GPIOA和ADC1
    RCC_ADCCLKConfig(RCC_PCLK2_Div6); // 分频因子改为6
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3; // PA0-PA3引脚
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; // 模拟输入引脚
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
    ADC_InitStructure.ADC_ScanConvMode = ENABLE;
    ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; // 非连续转换模式
    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T3_TRGO; // 使用TIM3的TRGO信号触发
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; // ADC数据右对齐
    ADC_InitStructure.ADC_NbrOfChannel = 4; // 顺序进行规则转换的ADC通道的数目
    ADC_Init(ADC1, &ADC_InitStructure); // 根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器
    /* DMA初始化 */
    DMA_InitTypeDef DMA_InitStructure; // 定义结构体变量
    DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR; // 外设基地址,给定形参AddrA
    DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; // 外设数据宽度,选择半字,对应16为的ADC数据寄存器
    DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; // 外设地址自增,选择失能,始终以ADC数据寄存器为源
    DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)AD_Value; // 存储器基地址,给定存放AD转换结果的全局数组AD_Value
    DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; // 存储器数据宽度,选择半字,与源数据宽度对应
    DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; // 存储器地址自增,选择使能,每次转运后,数组移到下一个位置
    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; // 数据传输方向,选择由外设到存储器,ADC数据寄存器转到数组
    DMA_InitStructure.DMA_BufferSize = 4; // 转运的数据大小(转运次数),与ADC通道数一致
    DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; // 模式,选择正常模式,每次触发转换一次
    DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; // 存储器到存储器,选择失能,数据由ADC外设触发转运到存储器
    DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; // 优先级,选择中等
    DMA_Init(DMA1_Channel1, &DMA_InitStructure); // 将结构体变量交给DMA_Init,配置DMA1的通道1
    /* DMA和ADC使能 */
    DMA_Cmd(DMA1_Channel1, ENABLE); // DMA1的通道1使能
    ADC_DMACmd(ADC1, ENABLE); // ADC1触发DMA1的信号使能
    ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_239Cycles5);
    ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_239Cycles5);
    ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_239Cycles5);
    ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_239Cycles5);
    ADC_ExternalTrigConvCmd(ADC1, ENABLE);
    ADC_Cmd(ADC1, ENABLE); // 闭合ADC的总开关
    ADC_ResetCalibration(ADC1); // 使能复位校准  
    while(ADC_GetResetCalibrationStatus(ADC1)); // 等待复位校准结束
    ADC_StartCalibration(ADC1); // 开启AD校准
    while(ADC_GetCalibrationStatus(ADC1)); // 等待校准结束
    //有的说要加有的说不加,实测不加是没问题的,但是加了也不影响
    ADC_SoftwareStartConvCmd(ADC1, ENABLE); // 使能指定的ADC1的软件转换启动功能
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值