4-2 STM32F405--ADC单通道(连续转换、非扫描、软件触发)

功能描述:使用STM32F405RGT6的ADC1的通道4对输入到PA4上的模拟电压进行模数转换,并在OLED上显示ADC的原始采样值和实际电压值。ADC配置为独立模式、单通道、连续转换、软件触发、非扫描模式。

  • main.c文件内容如下
#include "stm32f4xx.h"                  // Device header
#include "OLED.h"
#include "MyADC.h"

uint16_t ADC_Value;
double Voltage;

int main(void)
{
	MyADC_Init();
	OLED_Init();
	
	OLED_Clear();	//清屏OLED
	OLED_Refresh();	//刷新OLED
	
	while(1)
	{
		ADC_Value = MyADC_GetValue();					//获取ADC采样值
		Voltage = ((double)ADC_Value/4096)*3.3;			//计算对应的电压
		OLED_Printf(0,0,8,"ADC_Value = %d",ADC_Value);	//显示ADC采样值
		OLED_Printf(0,16,8,"Voltage = %.2lfV",Voltage);	//显示电压值
		OLED_Refresh();									//刷新OLED
	}
}
  • MyADC.c文件内容如下
#include "MyADC.h"           

/**
  * @摘要  		初始化ADC和相应的GPIO
  * @参数  		无
  * @返回值  	无
  * @说明  		无
  */
void MyADC_Init(void)
{
	//开启GPIOA的时钟
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
	
	//定义GPIO初始化结构体
	GPIO_InitTypeDef GPIO_InitStructure;
	//初始化上述结构体
	GPIO_StructInit(&GPIO_InitStructure);
	//配置GPIO模式为模拟输入模式
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
	//配置GPIO输出类型为推挽输出(此处无用)
	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
	//指定GPIO引脚为Pin4
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
	//配置GPIO输入类型为浮空输入(此处无用)
	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
	//配置GPIO的速度为快速50MHZ
	GPIO_InitStructure.GPIO_Speed = GPIO_Fast_Speed;
	//初始化对应的GPIO
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	//开启ADC1的时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
	
	//定义所有ADC共用配置的初始化结构体
	ADC_CommonInitTypeDef ADC_CommonInitStructure;
	//初始化上述结构体
	ADC_CommonStructInit(&ADC_CommonInitStructure);
	//配置为独立模式
	ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;
	//配置ADC时钟的预分频器的值为4,即21MHZ(<30MHZ)
	ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4;
	//失能DMA
	ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled;
	//配置多ADC模式下的两个ADC采样间隔(此处无用)
	ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_8Cycles;
	//初始化所有ADC共用配置
	ADC_CommonInit(&ADC_CommonInitStructure);
	
	//定义ADC初始化结构体
	ADC_InitTypeDef ADC_InitStructure;
	//初始化上述结构体
	ADC_StructInit(&ADC_InitStructure);
	//配置ADC分辨率为12位
	ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;
	//配置ADC为连续转换模式
	ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
	//配置ADC为非扫描模式
	ADC_InitStructure.ADC_ScanConvMode = DISABLE;
	//配置ADC数据对齐方式为右对齐
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
	//配置ADC外部触发为无,即关闭外部触发
	ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;
	//配置ADC外部触发源为TIM2的TRGO输出(此处无用)
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T2_TRGO;
	//配置ADC转换通道数为1(此处无用)
	ADC_InitStructure.ADC_NbrOfConversion = 1;
	//初始化对应的ADC
	ADC_Init(ADC1, &ADC_InitStructure);
	
	//配置每一个规则通道的排序和采样周期
	ADC_RegularChannelConfig(ADC1, ADC_Channel_4, 1, ADC_SampleTime_56Cycles);
	
	//开启ADC1,此时ADC1进入等待触发的状态
	ADC_Cmd(ADC1, ENABLE);
	
	//软件触发ADC1,此时ADC1进入采样转换状态
	ADC_SoftwareStartConv(ADC1);
}

/**
  * @摘要  		软件触发ADC并读取采样值
  * @参数  		无
  * @返回值  	ADC的采样值
  * @说明  		无
  */
uint16_t MyADC_GetValue(void)
{
	//等待ADC1转换完成
	while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC) == RESET);
	//转换完成后,ADC1又进入采样转换状态
	/*	
	ADC_DR寄存器里的值每隔一个ADC采样转换周期(这里是3.2us左右)就更新,上述标志位就被置1,
	所以当下一步读取了ADC_DR寄存器里的值后硬件会将上述标志位置0,但马上(3.2us)又会置1,
	故此时上述等待语句意义不大,可以省略
	*/
	//返回ADC1的采样值
	return ADC_GetConversionValue(ADC1);
}
  • MyADC.h文件内容如下
#ifndef __MYADC_H
#define __MYADC_H

#include "stm32f4xx.h"                  // Device header

void MyADC_Init(void);
uint16_t MyADC_GetValue(void);

#endif

 

 

以下是使用HAL库在STM32F405上配置ADC1多通道单次转换DMA方式的代码: 首先,需要在CubeMX中启用ADC和DMA,并将ADC配置为多通道单次转换模式。在ADC设置中,选择“多通道”并选择要转换的通道。在DMA设置中,选择“循环模式”和“内存递增模式”,并将数据大小设置为16位。 然后,在主函数中初始化ADC和DMA,并启动转换: ```c /* ADC and DMA initialization */ ADC_HandleTypeDef hadc1; DMA_HandleTypeDef hdma_adc1; void MX_ADC1_Init(void) { hadc1.Instance = ADC1; hadc1.Init.ClockPrescaler = ADC_CLOCK_SYNC_PCLK_DIV4; hadc1.Init.Resolution = ADC_RESOLUTION_12B; hadc1.Init.ScanConvMode = ENABLE; hadc1.Init.ContinuousConvMode = DISABLE; hadc1.Init.DiscontinuousConvMode = DISABLE; hadc1.Init.NbrOfDiscConversion = 0; hadc1.Init.ExternalTrigConvEdge = ADC_EXTERNALTRIGCONVEDGE_NONE; hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START; hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT; hadc1.Init.NbrOfConversion = 2; hadc1.Init.DMAContinuousRequests = ENABLE; hadc1.Init.EOCSelection = ADC_EOC_SINGLE_CONV; if (HAL_ADC_Init(&hadc1) != HAL_OK) { Error_Handler(); } ADC_ChannelConfTypeDef sConfig = {0}; sConfig.Channel = ADC_CHANNEL_0; sConfig.Rank = 1; sConfig.SamplingTime = ADC_SAMPLETIME_84CYCLES; if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK) { Error_Handler(); } sConfig.Channel = ADC_CHANNEL_1; sConfig.Rank = 2; if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK) { Error_Handler(); } } void MX_DMA_Init(void) { /* DMA controller clock enable */ __HAL_RCC_DMA2_CLK_ENABLE(); /* DMA interrupt init */ /* DMA2_Stream0_IRQn interrupt configuration */ HAL_NVIC_SetPriority(DMA2_Stream0_IRQn, 0, 0); HAL_NVIC_EnableIRQ(DMA2_Stream0_IRQn); } int main(void) { /* Initialize peripherals */ HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_ADC1_Init(); MX_DMA_Init(); /* Start ADC conversion */ HAL_ADC_Start_DMA(&hadc1, (uint32_t*)adc_values, 2); while (1) { /* Main loop */ } } ``` 在转换完成后,DMA中断将被触发。在该中断处理程序中,可以读取ADC值并进行进一步处理: ```c /* DMA interrupt handler */ void DMA2_Stream0_IRQHandler(void) { HAL_DMA_IRQHandler(&hdma_adc1); /* Process ADC values */ uint16_t value0 = adc_values[0]; uint16_t value1 = adc_values[1]; /* ... */ } ``` 这里的“adc_values”是一个uint16_t类型的数组,大小为2,用于存储ADC转换结果。在ADC转换完成后,DMA将这些值存储在该数组中。可以使用这些值进行进一步的处理,例如将它们发送到计算机或用于控制外设。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值