STM32的ADC转换最常见的方式

发布时间:2023-08-23  

这里的ADC转换也来使用DMA---这个也是STM32的ADC转换最常见的方式。


第一步是了解STM32的ADC对应的GPIO口如下图不用记住,可以查询,我是将它剪下来粘贴到书本的相应章节!


第二步是配置相应ADC转换的GPIO口这里使用PC0--PC1

static void ADC1_GPIO_Config(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //打开DMA1的时钟

RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 “ RCC_APB2Periph_GPIOC, ENABLE);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 |GPIO_Pin_1;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入

GPIO_Init(GPIOC, &GPIO_InitStructure);

}

第三步是-配置ADC的DMA配置ADC通道等---

#define ADC1_DR_Address ((u32)0x40012400+0x4c) //外设地址

__IO uint16_t ADC_ConvertedValue[2];//内存数组

staTIc void ADC1_Mode_Config(void)

{

DMA_InitTypeDef DMA_InitStructure;

ADC_InitTypeDef ADC_InitStructure;

DMA_DeInit(DMA1_Channel1);

//---------------ADC的DMA配置--------------------

DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address; //ADC1地址---代表ADC1保存转换值的寄存器

DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue;//内存地址---用来保存DMA传输过来的ADC转换值----后面直接使用的变量地址

DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //外设为数据源

DMA_InitStructure.DMA_BufferSize = 2; //传输总数据---2通道需要传输2个数据

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设地址固定

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;//内存地址自增---总体表示始终从外设ADC1地址处取值---依次保存到连续的两个内存变量中---

DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;//外设传输数据单元---半字16位

DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //内存传输数据单元---半字16位

DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //循环模式---2个数据依次循环接收从外设ADC1传输过来的ADC值---

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); //再次打开DMA1

//------------ADC模式配置------------------------

ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//独立模式----还有很多模式---这个比较常见

ADC_InitStructure.ADC_ScanConvMode = ENABLE ; //扫描模式---采集多通道使用----本程序采集2通道---所以扫描模式

ADC_InitStructure.ADC_ConTInuousConvMode = ENABLE; //连续转换模式---不难理解---就是不停地采集---一次接一次

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //不使用外部触发转换---触发分为外部触发---比如中断与定时器。软件触发---后面有专用函数

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //采集的数据右对齐---方便计算

ADC_InitStructure.ADC_NbrOfChannel = 2; //总共需要转换的通道个数---这里2个

ADC_Init(ADC1, &ADC_InitStructure);

RCC_ADCCLKConfig(RCC_PCLK2_Div8); //配置ADC转换时钟---PCLK2的8分频

//下面这个函数比较重要----配置ADC的通道与采样周期---前面说的PC0与PC1对应的ADC通道分别是--10与11。采集周期也有几种。

ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 1, ADC_SampleTIme_55Cycles5);

ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 2, ADC_SampleTIme_55Cycles5);

ADC_DMACmd(ADC1, ENABLE); //打开DMA1的ADC1

ADC_Cmd(ADC1, ENABLE); //打开ADC1

ADC_ResetCalibration(ADC1); //复位校准寄存器

while(ADC_GetResetCalibrationStatus(ADC1));//等待校准寄存器复位完成

ADC_StartCalibration(ADC1);//ADC校准

while(ADC_GetCalibrationStatus(ADC1));//校准完成

ADC_SoftwareStartConvCmd(ADC1, ENABLE);//软件触发转换

}

---第四部分是---在硬件上使用了一个通道切换芯片----CD4052----由PC2---PC3控制通道的选择CD4052切换控制GPIO配置----

void GPIO_Config(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推完输出

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init(GPIOC, &GPIO_InitStructure);

}

---第五部分是---主函数-----

extern __IO uint16_t ADC_ConvertedValue[2];//声明外部变量

uint16_t My_ADC[2]; //求平均值

int main(void)

{

u8 i,led=0x01;

USART1_Config();

ADC1_GPIO_Config();

ADC1_Mode_Config();

while (1)

{

My_ADC[0]=0;

My_ADC[1]=0;

for(i=0;i《10;i++)

{

My_ADC[0]+=ADC_ConvertedValue[0];

My_ADC[1]+=ADC_ConvertedValue[1];

}

My_ADC[0]=My_ADC[0]/10; //采集10次求平均值

My_ADC[1]=My_ADC[1]/10;

ADC_ConvertedValueLocal =(float) My_ADC[0]/4096*3.3; //转换为电压值

printf(” The current AD---0 value = 0x%04X “, My_ADC[0]);

printf(”The current AD---0 value = %f V “,ADC_ConvertedValueLocal);

ADC_ConvertedValueLocal =(float) My_ADC[1]/4096*3.3;

printf(”The current AD---1 value = 0x%04X “, My_ADC[1]);

printf(”The current AD---1 value = %f V “,ADC_ConvertedValueLocal);


文章来源于:电子工程世界    原文链接
本站所有转载文章系出于传递更多信息之目的,且明确注明来源,不希望被转载的媒体或个人可与我们联系,我们将立即进行删除处理。

我们与500+贴片厂合作,完美满足客户的定制需求。为品牌提供定制化的推广方案、专属产品特色页,多渠道推广,SEM/SEO精准营销以及与公众号的联合推广...详细>>

利用葫芦芯平台的卓越技术服务和新产品推广能力,原厂代理能轻松打入消费物联网(IOT)、信息与通信(ICT)、汽车及新能源汽车、工业自动化及工业物联网、装备及功率电子...详细>>

充分利用其强大的电子元器件采购流量,创新性地为这些物料提供了一个全新的窗口。我们的高效数字营销技术,不仅可以助你轻松识别与连接到需求方,更能够极大地提高“闲置物料”的处理能力,通过葫芦芯平台...详细>>

我们的目标很明确:构建一个全方位的半导体产业生态系统。成为一家全球领先的半导体互联网生态公司。目前,我们已成功打造了智能汽车、智能家居、大健康医疗、机器人和材料等五大生态领域。更为重要的是...详细>>

我们深知加工与定制类服务商的价值和重要性,因此,我们倾力为您提供最顶尖的营销资源。在我们的平台上,您可以直接接触到100万的研发工程师和采购工程师,以及10万的活跃客户群体...详细>>

凭借我们强大的专业流量和尖端的互联网数字营销技术,我们承诺为原厂提供免费的产品资料推广服务。无论是最新的资讯、技术动态还是创新产品,都可以通过我们的平台迅速传达给目标客户...详细>>

我们不止于将线索转化为潜在客户。葫芦芯平台致力于形成业务闭环,从引流、宣传到最终销售,全程跟进,确保每一个potential lead都得到妥善处理,从而大幅提高转化率。不仅如此...详细>>