前言
本篇记录关于32的ADC的DMA采集,包含ADC外设配置思路、DMA配置思路,以及标准库和HAL库两个版本的程序实现。
一、ADC配置思路
ADC配置需要考虑哪些呢?首先最先想到的是应该是它需要IO做模拟输入,它有分辨率,采样时间等等的要求,若是多个ADC通道采集,那么还需要考虑不同通道的顺序转换,因为几个通道共用一个ADC外设的DR寄存器。 按照惯例,要使用片上外设,首先是使能相应外设的时钟,使能跟外设有关的IO时钟,配置IO口,配置ADC外设。接着在采集模拟信号之前,别忘了使能ADC,开启转换。
二、DMA配置思路
DMA就是一个直接内存访问控制器,关于它的作用,通俗地讲就是无需CPU就能实现将数据从外设搬到内存;将数据从内存搬到外设;将数据从一个内存搬到另一个内存。那么既然是片上的控制器,使用之前,首先是使能相应的时钟,配置DMA控制器,然后使能控制器让其按配置参数工作。DMA是怎么将ADC的采集数据搬到内存中的呢,什么时候搬?只有在规则通道转换完成之后ADC才会发送DMA请求,这时DMA就开始接锅啦。
三、ADC+DMA程序实现
1.标准库版本
这里我使用的是stm32F103C8T6来写个测试程序。步骤如下:
1.1 配置相关IO口。
我使用ADC1组的IN0~4,IN8,IN9这7个通道来采集模拟信号。对应的引脚是PA0–PA4,PB0,PB1。配置如下:
static void ADC1_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 ;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOB, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin =
GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3| GPIO_Pin_4;
GPIO_Init(GPIOA, &GPIO_InitStructure);
}
1.2 配置ADC
static void ADC1_init(void)
{
ADC_InitTypeDef ADC_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_ScanConvMode = ENABLE ;
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfChannel = 7;
ADC_Init(ADC1, &ADC_InitStructure);
RCC_ADCCLKConfig(RCC_PCLK2_Div6);
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_RegularChannelConfig(ADC1, ADC_Channel_4, 5, ADC_SampleTime_239Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 6, ADC_SampleTime_239Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_9, 7, ADC_SampleTime_239Cycles5);
ADC_Cmd(ADC1, ENABLE);
ADC_ResetCalibration(ADC1);
while(ADC_GetResetCalibrationStatus(ADC1));
ADC_StartCalibration(ADC1);
while(ADC_GetCalibrationStatus(ADC1));
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}
1.3 配置DMA
从参考手册上面了解到ADC1的请求信号通过DMA1通道1传入,如下: 所以配置目标是DMA1的通道1,如下:
static void DMA1_init(void)
{
DMA_InitTypeDef DMA_InitStructure;
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
DMA_DeInit(DMA1_Channel1);
DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue[0];
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize = 7;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
DMA_Cmd(DMA1_Channel1, ENABLE);
}
1.4 开始ADC+DMA采集
extern void ADC_DMA_Start(void)
{
ADC1_GPIO_Config();
ADC1_init();
DMA1_init();
ADC_DMACmd(ADC1, ENABLE);
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}
2.HAL库版本
2.1 cubemx配置
推荐大家使用CUBEMX直接配置ADC,我这里使用STM32F407单片机做下示例,下面是配置流程: 选择Analog中的ADC组,然后选择想要的通道,我选了6个通道,接下来就是ADC外设的参数选择,最后一步是DMA的设置参数选择。 上面的第一个参数Mode选独立模式,即选中的ADC组跟其他ADC组是独立开来的; 第二个是ADC驱动时钟配置,STM32F4的ADC最大时钟不能超过36Mhz(F1的不能超过14Mhz),而ADC是挂载在APB2总线上的,总线最大时钟频率是84MHz,这里选个4分频; 接下来是设置分辨率,对齐模式。由于用到DMA,所以这里使能扫描模式和使能连续转换模式,扫描选中的通道和每个通道连续转换。使能ADC的DMA请求。设置单通道转换完成置位EOC。 接着,设置规则通道的转换通道个数,采样时间,以及顺序。 ADC的参数设置完毕,接下来只需转到DMA Settings添加DMA的请求信号,注意这里将DMA请求模式设置为循环Circular,从外设到内存。
2.2 代码移植
一切准备就绪,不用管NVIC Settings,点击生成代码。生成的代码如下:
static void MX_ADC1_Init(void)
{
ADC_ChannelConfTypeDef sConfig = {0};
hadc1.Instance = ADC1;
hadc1.Init.ClockPrescaler = ADC_CLOCK_SYNC_PCLK_DIV4;
hadc1.Init.Resolution = ADC_RESOLUTION_12B;
hadc1.Init.ScanConvMode = ENABLE;
hadc1.Init.ContinuousConvMode = ENABLE;
hadc1.Init.DiscontinuousConvMode = DISABLE;
hadc1.Init.ExternalTrigConvEdge = ADC_EXTERNALTRIGCONVEDGE_NONE;
hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
hadc1.Init.NbrOfConversion = 6;
hadc1.Init.DMAContinuousRequests = ENABLE;
hadc1.Init.EOCSelection = ADC_EOC_SINGLE_CONV;
while(HAL_ADC_Init(&hadc1) != HAL_OK);
sConfig.Channel = ADC_CHANNEL_10;
sConfig.Rank = 1;
sConfig.SamplingTime = ADC_SAMPLETIME_480CYCLES;
while(HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK);
sConfig.Channel = ADC_CHANNEL_11;
sConfig.Rank = 2;
sConfig.SamplingTime = ADC_SAMPLETIME_480CYCLES;
while(HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK);
sConfig.Channel = ADC_CHANNEL_12;
sConfig.Rank = 3;
while(HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK);
sConfig.Channel = ADC_CHANNEL_13;
sConfig.Rank = 4;
while(HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK);
sConfig.Channel = ADC_CHANNEL_14;
sConfig.Rank = 5;
while(HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK);
sConfig.Channel = ADC_CHANNEL_15;
sConfig.Rank = 6;
while(HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK);
}
将上面的代码移植到自己的工程中,这里没贴出ADC句柄的定义。再在自己的工程中移植ADC的回调函数MSP,代码如下,有趣的是HAL库用__HAL_LINKDMA宏将外设和DMA连接起来了。
void HAL_ADC_MspInit(ADC_HandleTypeDef* hadc)
{
GPIO_InitTypeDef GPIO_InitStruct = {0};
if(hadc->Instance==ADC1)
{
__HAL_RCC_ADC1_CLK_ENABLE();
__HAL_RCC_GPIOC_CLK_ENABLE();
GPIO_InitStruct.Pin = GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_2|GPIO_PIN_3
|GPIO_PIN_4|GPIO_PIN_5;
GPIO_InitStruct.Mode = GPIO_MODE_ANALOG;
GPIO_InitStruct.Pull = GPIO_NOPULL;
HAL_GPIO_Init(GPIOC, &GPIO_InitStruct);
hdma_adc1.Instance = DMA2_Stream4;
hdma_adc1.Init.Channel = DMA_CHANNEL_0;
hdma_adc1.Init.Direction = DMA_PERIPH_TO_MEMORY;
hdma_adc1.Init.PeriphInc = DMA_PINC_DISABLE;
hdma_adc1.Init.MemInc = DMA_MINC_ENABLE;
hdma_adc1.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;
hdma_adc1.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
hdma_adc1.Init.Mode = DMA_CIRCULAR;
hdma_adc1.Init.Priority = DMA_PRIORITY_LOW;
hdma_adc1.Init.FIFOMode = DMA_FIFOMODE_DISABLE;
while (HAL_DMA_Init(&hdma_adc1) != HAL_OK);
__HAL_LINKDMA(hadc,DMA_Handle,hdma_adc1);
}
}
最后开启ADC+DMA的采集:
extern void ADC_Init(void)
{
__HAL_RCC_DMA2_CLK_ENABLE();
MX_ADC1_Init();
HAL_ADC_Start_DMA(&hadc1, (uint32_t*)ADC_convert_result, 6);
}
2.3 结果
|