|
我剛學(xué)SMT32。我自己編寫了雙通道非DMA采集ADC,和6通道DMA采集ADC。雙通道的只會采集PA1,沒有采集PA0。我想可能是覆蓋數(shù)據(jù)的原因。而6通道DMA的采集結(jié)果全是0。大佬能看下我的程序錯在哪里嘛。
這是雙通道的。
電路圖.png (59.21 KB, 下載次數(shù): 30)
下載附件
這是我的電路圖。
2020-6-9 16:38 上傳
編譯可以通過,沒有錯誤。但是雙通道仿真時,只能采集PA1的值。而多通道DMA采集時,全是0.大佬能看下問題出在哪里嘛。
這是adc.c
#include "adc.h"
void ADC1_GPIO_Config(void){
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_ADC1, ENABLE); //使能ADC1,GPIOC時鐘
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1; //
//GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//為什么沒有配置這個????
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模擬輸入
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PC4
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 ; //
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模擬輸入
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PC4
}
void ADC_Config(void)
{
ADC_InitTypeDef ADC_InitStructure;//ADC結(jié)構(gòu)體變量//注意在一個語句快內(nèi)變量的聲明要放在可執(zhí)行語句的前面,否則出錯,因此要放在ADC1_GPIO_Config();前面
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//ADC1和ADC2工作在獨立模式
ADC_InitStructure.ADC_ScanConvMode = ENABLE; //使能掃描
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;//ADC轉(zhuǎn)換工作在連續(xù)模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//由軟件控制轉(zhuǎn)換,不使用外部觸發(fā)
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//轉(zhuǎn)換數(shù)據(jù)右對齊
ADC_InitStructure.ADC_NbrOfChannel = 1;//轉(zhuǎn)換通道為1
ADC_Init(ADC1, &ADC_InitStructure); //初始化ADC
ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 2, ADC_SampleTime_55Cycles5);
//ADC1選擇信道14,音序等級1,采樣時間55.5個周期
// ADC_DMACmd(ADC1, ENABLE);//使能ADC1模塊DMA
ADC_Cmd(ADC1, ENABLE);//使能ADC1
ADC_ITConfig(ADC1, ADC_IT_EOC, ENABLE);
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
// ADC_ResetCalibration(ADC1); //重置.(復(fù)位).ADC1校準(zhǔn)寄存器
// while(ADC_GetResetCalibrationStatus(ADC1));//等待ADC1校準(zhǔn)重置完成
// ADC_StartCalibration(ADC1);//開始ADC1校準(zhǔn)
// while(ADC_GetCalibrationStatus(ADC1));
} //等待ADC1校準(zhǔn)完成
/* ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能ADC1軟件開始轉(zhuǎn)換*/
這是adc.h
#ifndef __ADC_H
#define __ADC_H
#include "sys.h"
void Adc_Init(void);
void ADC1_GPIO_Config(void);
void ADC_Config(void);
#endif
這是main.c
#include "stm32f10x.h"
#include "bsp-lcd1602.h"
#include "delay.h"
#include "sys.h"
#include "adc.h"
void main (void)
{int a,c,d,b,e,f;
float temp;float ADC_ConvertedValue;float temp1;
delay_init(); //延時函數(shù)初始化
LCD1602_Init();
ADC1_GPIO_Config();
ADC_Config();
while(1)
{ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_239Cycles5 );
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));
ADC_ConvertedValue=ADC_GetConversionValue(ADC1);
temp=(float)ADC_ConvertedValue*(3.4/4096);
a=temp/1;
c=temp*10;
d=c%10;
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 2, ADC_SampleTime_239Cycles5 );
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));
ADC_ConvertedValue=ADC_GetConversionValue(ADC1);
temp1=(float)ADC_ConvertedValue*(3.4/4096);
b=temp/1;
e=temp*10;
f=c%10;
LCD_ShowNum(11,0,a);
LCD_ShowNum(13,0,d);
LCD_ShowNum(1,0,b);
LCD_ShowNum(2,0,f);
}
}
這是多通道DMA的adc.c
#include "adc.h"
__IO uint16_t ADC_ConvertedValue[NOFCHANEL]={0,0,0,0,0,0};
/**
* @brief ADC GPIO 初始化
* @param 無
* @retval 無
*/
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|
ADC_PIN3|
ADC_PIN4|
ADC_PIN5|
ADC_PIN6;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
// 初始化 ADC IO
GPIO_Init(ADC_PORT, &GPIO_InitStructure);
}
/**
* @brief 配置ADC工作模式
* @param 無
* @retval 無
*/
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 );
// 復(fù)位DMA控制器
DMA_DeInit(ADC_DMA_CHANNEL);
// 配置 DMA 初始化結(jié)構(gòu)體
// 外設(shè)基址為:ADC 數(shù)據(jù)寄存器地址
DMA_InitStructure.DMA_PeripheralBaseAddr = ( u32 ) ( & ( ADC_x->DR ) );
// 存儲器地址
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)ADC_ConvertedValue;
// 數(shù)據(jù)源來自外設(shè)
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
// 緩沖區(qū)大小,應(yīng)該等于數(shù)據(jù)目的地的大小
DMA_InitStructure.DMA_BufferSize = NOFCHANEL;
// 外設(shè)寄存器只有一個,地址不用遞增
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
// 存儲器地址遞增
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
// 外設(shè)數(shù)據(jù)大小為半字,即兩個字節(jié)
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
// 內(nèi)存數(shù)據(jù)大小也為半字,跟外設(shè)數(shù)據(jù)大小相同
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
// 循環(huán)傳輸模式
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
// DMA 傳輸通道優(yōu)先級為高,當(dāng)使用一個DMA通道時,優(yōu)先級設(shè)置不影響
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
// 禁止存儲器到存儲器模式,因為是從外設(shè)到存儲器
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 ;
// 連續(xù)轉(zhuǎn)換模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
// 不用外部觸發(fā)轉(zhuǎn)換,軟件開啟即可
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
// 轉(zhuǎn)換結(jié)果右對齊
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
// 轉(zhuǎn)換通道個數(shù)
ADC_InitStructure.ADC_NbrOfChannel = NOFCHANEL;
// 初始化ADC
ADC_Init(ADC_x, &ADC_InitStructure);
// 配置ADC時鐘N狿CLK2的8分頻,即9MHz
RCC_ADCCLKConfig(RCC_PCLK2_Div8);
// 配置ADC 通道的轉(zhuǎn)換順序和采樣時間
ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL1, 1, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL2, 2, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL3, 3, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL4, 4, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL5, 5, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL6, 6, ADC_SampleTime_55Cycles5);
// 使能ADC DMA 請求
ADC_DMACmd(ADC_x, ENABLE);
// 開啟ADC ,并開始轉(zhuǎn)換
ADC_Cmd(ADC_x, ENABLE);
// 初始化ADC 校準(zhǔn)寄存器
ADC_ResetCalibration(ADC_x);
// 等待校準(zhǔn)寄存器初始化完成
while(ADC_GetResetCalibrationStatus(ADC_x));
// ADC開始校準(zhǔn)
ADC_StartCalibration(ADC_x);
// 等待校準(zhǔn)完成
while(ADC_GetCalibrationStatus(ADC_x));
// 由于沒有采用外部觸發(fā),所以使用軟件觸發(fā)ADC轉(zhuǎn)換
ADC_SoftwareStartConvCmd(ADC_x, ENABLE);
}
/**
* @brief ADC初始化
* @param 無
* @retval 無
*/
void ADCx_Init(void)
{
ADCx_GPIO_Config();
ADCx_Mode_Config();
}
這是adc.h
#ifndef __ADC_H
#define __ADC_H
#include "sys.h"
#include "stm32f10x.h"
// 注意:用作ADC采集的IO必須沒有復(fù)用,否則采集電壓會有影響
/********************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 在指南者里面接的是蜂鳴器,默認(rèn)被拉低
// 2-PC0 在指南者里面接的是SPI FLASH的 片選,默認(rèn)被拉高
// 所以 PC0 做 ADC 轉(zhuǎn)換通道的時候,結(jié)果可能會有誤差
// 轉(zhuǎn)換通道個數(shù)
#define NOFCHANEL 6
#define ADC_PIN1 GPIO_Pin_0
#define ADC_CHANNEL1 ADC_Channel_0
#define ADC_PIN2 GPIO_Pin_1
#define ADC_CHANNEL2 ADC_Channel_1
#define ADC_PIN3 GPIO_Pin_2
#define ADC_CHANNEL3 ADC_Channel_2
#define ADC_PIN4 GPIO_Pin_3
#define ADC_CHANNEL4 ADC_Channel_3
#define ADC_PIN5 GPIO_Pin_4
#define ADC_CHANNEL5 ADC_Channel_4
#define ADC_PIN6 GPIO_Pin_5
#define ADC_CHANNEL6 ADC_Channel_5
// ADC1 對應(yīng) DMA1通道1,ADC3對應(yīng)DMA2通道5,ADC2沒有DMA功能
#define ADC_x ADC1
#define ADC_DMA_CHANNEL DMA1_Channel1
#define ADC_DMA_CLK RCC_AHBPeriph_DMA1
/**************************函數(shù)聲明********************************/
void ADCx_Init (void);
void ADCx_GPIO_Config(void);
void ADCx_Mode_Config(void);
#endif /* __ADC_H */
這是main.c
#include "stm32f10x.h"
#include "bsp-lcd1602.h"
#include "delay.h"
#include "sys.h"
#include "adc.h"
// ADC1轉(zhuǎn)換的電壓值通過MDA方式傳到SRAM
extern __IO uint16_t ADC_ConvertedValue[NOFCHANEL];
// 局部變量,用于保存轉(zhuǎn)換計算后的電壓值
float ADC_ConvertedValueLocal[NOFCHANEL];
/*
// 軟件延時
void Delay(__IO uint32_t nCount)
{
for(; nCount != 0; nCount--);
}
\
**/
int main(void)
{
int a,b,c,d,f;volatile int e;
// ADC 初始化
/*ADCx_Init();*/
delay_init(); //延時函數(shù)初始化
LCD1602_Init();
ADCx_GPIO_Config;
ADCx_Mode_Config;
while (1)
{
ADC_ConvertedValueLocal[0] =(float) ADC_ConvertedValue[0]/4096*3.3;
ADC_ConvertedValueLocal[1] =(float) ADC_ConvertedValue[1]/4096*3.3;
ADC_ConvertedValueLocal[2] =(float) ADC_ConvertedValue[2]/4096*3.3;
ADC_ConvertedValueLocal[3] =(float) ADC_ConvertedValue[3]/4096*3.3;
ADC_ConvertedValueLocal[4] =(float) ADC_ConvertedValue[4]/4096*3.3;
ADC_ConvertedValueLocal[5] =(float) ADC_ConvertedValue[5]/4096*3.3;
a=ADC_ConvertedValueLocal[0]/1;
b=ADC_ConvertedValueLocal[1]/1;
c=ADC_ConvertedValueLocal[2]/1;
d=ADC_ConvertedValueLocal[3]/1;
e=ADC_ConvertedValueLocal[4]/1;
f=ADC_ConvertedValueLocal[5]/1;
LCD_ShowNum(1,0,a);
LCD_ShowNum(2,0,b);
LCD_ShowNum(3,0,c);
LCD_ShowNum(4,0,d);
LCD_ShowNum(5,0,e);
LCD_ShowNum(6,0,f);
}
}
|
|