前面一节,我们讲到用ADC单通道采集电压数据,但是只能够实现一个通道的电压数据采集,现在我们需要多个ADC通道采集不同传感器的数据,就需要开启STM32的ADC多通道数据了,接下来我们放代码

ADCM.C

#include "ADCM.h"

//

//本程序只供学习使用,未经作者许可,不得用于其它任何用途

//STM32C8T6开发板

//ADC多通道采集DMA

//闲鱼关注:强哥电子701;CSDN:Dandelion701:B站:Dandelion701

//修改日期:2024.1.17

//

__IO uint16_t ADC_ConvertedValue[NOFCHANEL]={0,0};

static void ADCx_GPIO_Config(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

// 打开 ADC IO端口时钟

ADC_GPIO_APBxClock_FUN ( ADC_GPIO_CLK, ENABLE );

// 配置 ADC IO 引脚模式

GPIO_InitStructure.GPIO_Pin = ADC_PIN1|ADC_PIN2;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;

// 初始化 ADC IO

GPIO_Init(ADC_PORT, &GPIO_InitStructure);

}

static void ADCx_Mode_Config(void)

{

DMA_InitTypeDef DMA_InitStructure;

ADC_InitTypeDef ADC_InitStructure;

// 打开DMA时钟

RCC_AHBPeriphClockCmd(ADC_DMA_CLK, ENABLE);

// 打开ADC时钟

ADC_APBxClock_FUN ( ADC_CLK, ENABLE );

// 复位DMA控制器

DMA_DeInit(ADC_DMA_CHANNEL);

// 配置 DMA 初始化结构体

// 外设基址为:ADC 数据寄存器地址

DMA_InitStructure.DMA_PeripheralBaseAddr = ( u32 ) ( & ( ADC_x->DR ) );

// 存储器地址

DMA_InitStructure.DMA_MemoryBaseAddr = (u32)ADC_ConvertedValue;

// 数据源来自外设

DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;

// 缓冲区大小,应该等于数据目的地的大小

DMA_InitStructure.DMA_BufferSize = NOFCHANEL;

// 外设寄存器只有一个,地址不用递增

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 传输通道优先级为高,当使用一个DMA通道时,优先级设置不影响

DMA_InitStructure.DMA_Priority = DMA_Priority_High;

// 禁止存储器到存储器模式,因为是从外设到存储器

DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;

// 初始化DMA

DMA_Init(ADC_DMA_CHANNEL, &DMA_InitStructure);

// 使能 DMA 通道

DMA_Cmd(ADC_DMA_CHANNEL , ENABLE);

// ADC 模式配置

// 只使用一个ADC,属于单模式

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 = NOFCHANEL;

// 初始化ADC

ADC_Init(ADC_x, &ADC_InitStructure);

// 配置ADC时钟N狿CLK2的8分频,即9MHz

RCC_ADCCLKConfig(RCC_PCLK2_Div8);

// 配置ADC 通道的转换顺序和采样时间

ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL1, 1, ADC_SampleTime_55Cycles5);

ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL2, 2, ADC_SampleTime_55Cycles5);

// 使能ADC DMA 请求

ADC_DMACmd(ADC_x, ENABLE);

// 开启ADC ,并开始转换

ADC_Cmd(ADC_x, ENABLE);

// 初始化ADC 校准寄存器

ADC_ResetCalibration(ADC_x);

// 等待校准寄存器初始化完成

while(ADC_GetResetCalibrationStatus(ADC_x));

// ADC开始校准

ADC_StartCalibration(ADC_x);

// 等待校准完成

while(ADC_GetCalibrationStatus(ADC_x));

// 由于没有采用外部触发,所以使用软件触发ADC转换

ADC_SoftwareStartConvCmd(ADC_x, ENABLE);

}

void ADCx_Init(void)

{

ADCx_GPIO_Config();

ADCx_Mode_Config();

}

ADCM.h

#ifndef __ADCM_H

#define __ADCM_H

#include "stm32f10x.h"

// 注意:用作ADC采集的IO必须没有复用,否则采集电压会有影响

/********************ADC1输入通道(引脚)配置**************************/

#define ADC_APBxClock_FUN RCC_APB2PeriphClockCmd

#define ADC_CLK RCC_APB2Periph_ADC1

#define ADC_GPIO_APBxClock_FUN RCC_APB2PeriphClockCmd

#define ADC_GPIO_CLK RCC_APB2Periph_GPIOA

#define ADC_PORT GPIOA

// 注意

// 1-PC0 在IOT里面接的是蜂鸣器,默认被拉低

// 2-PC0 在MINI里面接的是SPI FLASH的 片选,默认被拉高

// 所以 PC0 做 ADC 转换通道的时候,结果可能会有误差

// 转换通道个数

#define NOFCHANEL 2

#define ADC_PIN1 GPIO_Pin_0

#define ADC_CHANNEL1 ADC_Channel_0

#define ADC_PIN2 GPIO_Pin_1

#define ADC_CHANNEL2 ADC_Channel_1

// ADC1 对应 DMA1通道1,ADC3对应DMA2通道5,ADC2没有DMA功能

#define ADC_x ADC1

#define ADC_DMA_CHANNEL DMA1_Channel1

#define ADC_DMA_CLK RCC_AHBPeriph_DMA1

void ADCx_Init(void);

#endif

main

#include "ADCM.h"

int i=0;

u8 string[30] = {0},string1[30] = {0};

float ADC_ConvertedValueLocal[NOFCHANEL];

extern __IO uint16_t ADC_ConvertedValue[NOFCHANEL];

void Use_MQ_3(void)

{

ADC_ConvertedValueLocal[0] =(float) ADC_ConvertedValue[0]*125/4095;

sprintf((char *)string,"MQ3:%f",ADC_ConvertedValueLocal[0]);

OLED_ShowString(1,40,string,8,1);

}

void Use_MQ_2(void)

{

ADC_ConvertedValueLocal[1] =(float) ADC_ConvertedValue[1]*99/4096;

sprintf((char *)string,"MQ2:%f",ADC_ConvertedValueLocal[1]);

OLED_ShowString(1,50,string,8,1);

}

int main(void)

{

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先

delay_init();//延时函数初始化

uart_init(115200);//串口2

LED_Init();

OLED_Init();//oled显示屏初始化

OLED_ColorTurn(0); //0正常显示,1 反色显示

OLED_DisplayTurn(0); //屏幕旋转180度调整

ADCx_Init();//ADC通道初始化

while(1)

{

OLED_Refresh();//oled刷新

Use_MQ_3();

Use_MQ_2();

}

}

推荐阅读

评论可见,请评论后查看内容,谢谢!!!
 您阅读本篇文章共花了: