STM32入门(11)

STM32入门(11)

项目一:DMA+AD多通道

#include "stm32f10x.h"                    // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"


int main()
{
	
	// 模块初始化
	OLED_Init();                  // OLED初始化
	AD_Init();                    // AD初始化
	
	
	// 显示静态字符串
	OLED_ShowString(1,1,"AD0:"); 
	OLED_ShowString(2,1,"AD1:");
	OLED_ShowString(3,1,"AD2:"); 
	OLED_ShowString(4,1,"AD3:");
	

	while(1)
	{
		
		OLED_ShowNum(1, 5, AD_Value[0], 4);		//显示转换结果第0个数据
		OLED_ShowNum(2, 5, AD_Value[1], 4);		//显示转换结果第1个数据
		OLED_ShowNum(3, 5, AD_Value[2], 4);		//显示转换结果第2个数据
		OLED_ShowNum(4, 5, AD_Value[3], 4);		//显示转换结果第3个数据
		
		Delay_ms(100);         					// 延时100ms,手动增加一些转换的间隔时间 
	}
}

#include "stm32f10x.h"                  // Device header

uint16_t AD_Value[4];

/**
  * 函    数:AD初始化
  * 参    数:无
  * 返 回 值:无
  */
void AD_Init(void)
{
	// 开启时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);      // 开启ADC1的时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);     // 开启GPIOA的时钟
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);        // 开启DMA1的时钟
	
	// 设置ADC时钟
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);               // 选择时钟6分频,ADCCLK = 72MHz / 6 = 12MHz
	
	// GPIO初始化
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStructure);                  // 将PA0、PA1、PA2和PA3引脚初始化为模拟输入
	
	// 规则组通道配置
	ADC_RegularChannelConfig(ADC1,ADC_Channel_0,1,ADC_SampleTime_55Cycles5);  // 规则组序列1的位置,配置为通道0
	ADC_RegularChannelConfig(ADC1,ADC_Channel_1,2,ADC_SampleTime_55Cycles5);  // 规则组序列2的位置,配置为通道1
	ADC_RegularChannelConfig(ADC1,ADC_Channel_2,3,ADC_SampleTime_55Cycles5);  // 规则组序列3的位置,配置为通道2
	ADC_RegularChannelConfig(ADC1,ADC_Channel_3,4,ADC_SampleTime_55Cycles5);  // 规则组序列4的位置,配置为通道3
	
	// ADC初始化
	ADC_InitTypeDef ADC_InitStructure;                                  // 定义结构体变量
	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;					// 模式,选择独立模式,即单独使用ADC1
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;				// 数据对齐,选择右对齐
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;	// 外部触发,使用软件触发,不需要使用ADC1
    // 连续转换,使能,每转换一次规则组序列后立刻开始下一次转换
	ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;	
	ADC_InitStructure.ADC_ScanConvMode = ENABLE; // 扫描模式,使能,扫描规则组序列,扫描数量由ADC_NbrOfChannel确定	
	ADC_InitStructure.ADC_NbrOfChannel = 4;		 // 通道数,为4,扫描规则组的前4个通道
	ADC_Init(ADC1,&ADC_InitStructure);			 // 将结构体变量交给ADC_Init,配置为ADC1
	
	// DAM初始化
    // 定义结构体变量
	DMA_InitTypeDef DMA_InitStructure;                                           
    // 外设基地址,给定形参AddrA
	DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;              
    // 外设数据宽度,选择半字,对应16位的ADC数据寄存器
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;  
    // 外设地址自增,选择失能,始终以ADC数据寄存器为源
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
    // 存储器基地址,给定存放AD转换结果的全局数组AD_Value
	DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)AD_Value;
    // 存储器数据宽度,选择半字,与源数据宽度对应
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;	
    // 存储器地址自增,选择使能,每次转运后,数组移到下一个位置
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;	
    // 数据传输方向,选择由外设到存储器,ADC数据寄存器转到数组
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
    // 转运的数据大小(转运次数),与ADC通道数一致
	DMA_InitStructure.DMA_BufferSize = 4;
    // 模式,选择循环模式,与ADC的连续转换一致
	DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
    // 存储器到存储器,选择失能,数据由ADC外设触发转运到存储器
	DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;	
    // 优先级,选择中等
	DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;	
    // 将结构体变量交给DMA_Init,配置DMA1的通道1
	DMA_Init(DMA1_Channel1,&DMA_InitStructure);																	
	
	// DMA和ADC使能
	DMA_Cmd(DMA1_Channel1,ENABLE);     // DMA1的通道1使能
	ADC_DMACmd(ADC1,ENABLE);		   // ADC1触发DMA1的信号使能
	ADC_Cmd(ADC1,ENABLE);			   // ADC1使能
	
	// ADC校准
	ADC_ResetCalibration(ADC1);
	while (ADC_GetResetCalibrationStatus(ADC1) == SET);
	ADC_StartCalibration(ADC1);
	while (ADC_GetCalibrationStatus(ADC1) == SET);
	
	// ADC触发
	// 软件触发ADC开始工作,由于ADC处于连续转换模式,故触发一次后ADC就可以一直连续不断工作
	ADC_SoftwareStartConvCmd(ADC1,ENABLE);   
	
}

#ifndef __AD_H
#define __AD_H

extern uint16_t AD_Value[4];
void AD_Init(void);

#endif

posted @ 2026-03-16 21:25  Q&25  阅读(5)  评论(0)    收藏  举报