標(biāo)題:
STM32 adc多通道采集數(shù)據(jù)存儲(chǔ)并DMA傳輸問題
[打印本頁]
作者:
尤曉權(quán)
時(shí)間:
2020-3-20 11:34
標(biāo)題:
STM32 adc多通道采集數(shù)據(jù)存儲(chǔ)并DMA傳輸問題
adc開啟8個(gè)通道采集數(shù)據(jù),之后用DMA將數(shù)據(jù)傳送到數(shù)組value[8][10]中,現(xiàn)在我很疑惑,這個(gè)二維數(shù)組中的前8個(gè)數(shù)據(jù)是不是DMA從ADC->DR中回傳的八個(gè)數(shù)據(jù)的第一次采樣值?
單片機(jī)源程序如下:
#define Channel_Num 8 //ADC的通道數(shù),使用8個(gè)通道
#define Sample_Num 10 //采樣次數(shù),使用平均濾波,采樣10次取均值
//#define DMA_BUFF_SIZE Channel_Num*Sample_Num*3
u16 AD_Value[8][10];// __attribute__((at(0X68000000)));//行優(yōu)先存儲(chǔ),DMA第一次傳輸?shù)?個(gè)數(shù)據(jù)為ADC各個(gè)通道的第一次采樣數(shù)據(jù)
u16 Actual_AD_Value[8];
void ADC_GPIO_init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_GPIOC, ENABLE );
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模擬輸入引腳
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模擬輸入引腳
GPIO_Init(GPIOC, &GPIO_InitStructure);
}
//adc1 8通道循環(huán)采集配置
void Adc_DMA_Init(void)
{
ADC_InitTypeDef ADC_InitStructure;
DMA_InitTypeDef DMA_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 , ENABLE ); //使能ADC1通道時(shí)鐘
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA傳輸
RCC_ADCCLKConfig(RCC_PCLK2_Div6); //設(shè)置ADC時(shí)鐘分頻因子6,72M/6=12M,ADC最大時(shí)間不能超過14M
DMA_DeInit(DMA1_Channel1); //將DMA的通道1寄存器重設(shè)為缺省值
DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&ADC1->DR; //DMA外設(shè)基地址
DMA_InitStructure.DMA_MemoryBaseAddr =(u32)&AD_Value; //DMA內(nèi)存基地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //數(shù)據(jù)傳輸方向,從外設(shè)讀取發(fā)送到內(nèi)存
DMA_InitStructure.DMA_BufferSize = (Channel_Num*Sample_Num+2)*2; //DMA通道的DMA緩存的大小//Channel_Num*Sample_Num
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外設(shè)地址寄存器不變
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //內(nèi)存地址寄存器遞增
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //外設(shè)數(shù)據(jù)寬度為16位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //數(shù)據(jù)寬度為16位
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //工作在循環(huán)模式DMA_Mode_Normal DMA_Mode_Circular
DMA_InitStructure.DMA_Priority = DMA_Priority_High; //DMA通道1擁有高優(yōu)先級(jí)
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //DMA通道x沒有設(shè)置為內(nèi)存到內(nèi)存?zhèn)鬏?br />
DMA_Init(DMA1_Channel1, &DMA_InitStructure); //根據(jù)DMA_InitStruct中指定的參數(shù)初始化DMA的通道USART1_Tx_DMA_Channel所標(biāo)識(shí)的寄存器
DMA_Cmd(DMA1_Channel1,ENABLE);
ADC_DeInit(ADC1); //復(fù)位ADC1
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在獨(dú)立模式
ADC_InitStructure.ADC_ScanConvMode = ENABLE; //模數(shù)轉(zhuǎn)換工作在掃描模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; //模數(shù)轉(zhuǎn)換工作在連續(xù)轉(zhuǎn)換模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //轉(zhuǎn)換由軟件而不是外部觸發(fā)啟動(dòng)
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC數(shù)據(jù)右對(duì)齊
ADC_InitStructure.ADC_NbrOfChannel = 8; //順序進(jìn)行規(guī)則轉(zhuǎn)換的ADC通道的數(shù)目
ADC_Init(ADC1, &ADC_InitStructure); //根據(jù)ADC_InitStruct中指定的參數(shù)初始化外設(shè)ADCx的寄存器
//設(shè)置指定ADC的規(guī)則組通道,8個(gè)序列,采樣時(shí)間
ADC_RegularChannelConfig(ADC1,ADC_Channel_0, 1,ADC_SampleTime_239Cycles5);
ADC_RegularChannelConfig(ADC1,ADC_Channel_1, 2,ADC_SampleTime_239Cycles5);
ADC_RegularChannelConfig(ADC1,ADC_Channel_2, 3,ADC_SampleTime_239Cycles5);
ADC_RegularChannelConfig(ADC1,ADC_Channel_3, 4,ADC_SampleTime_239Cycles5);
ADC_RegularChannelConfig(ADC1,ADC_Channel_10, 5,ADC_SampleTime_239Cycles5);
ADC_RegularChannelConfig(ADC1,ADC_Channel_11, 6,ADC_SampleTime_239Cycles5);
ADC_RegularChannelConfig(ADC1,ADC_Channel_12, 7,ADC_SampleTime_239Cycles5);
ADC_RegularChannelConfig(ADC1,ADC_Channel_13, 8,ADC_SampleTime_239Cycles5);
ADC_DMACmd(ADC1, ENABLE);
ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1
ADC_ResetCalibration(ADC1); //使能復(fù)位校準(zhǔn)
while(ADC_GetResetCalibrationStatus(ADC1)); //等待復(fù)位校準(zhǔn)結(jié)束
ADC_StartCalibration(ADC1); //開啟AD校準(zhǔn)
while(ADC_GetCalibrationStatus(ADC1)); //等待校準(zhǔn)結(jié)束
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的軟件轉(zhuǎn)換啟動(dòng)功能
}
//獲得每個(gè)通道連續(xù)采集10的的AD平均值
void Get_Adc(void)
{
u16 sum=0;
u8 i=0,count=0;
for(i=0;i<Channel_Num;i++)
{
for(count=0;count<Sample_Num;count++)
{
sum+=AD_Value[i][count];
}
Actual_AD_Value[i]=sum/Sample_Num;
}
sum=0;
}
float GET_Voltage(u16 AD_values)
{
return (float)(AD_values*3.3/4096);
}
復(fù)制代碼
作者:
尤曉權(quán)
時(shí)間:
2020-3-20 13:22
我還想問一下,這個(gè)程序,在電壓處理那塊void Get_Adc(void)中,Actual_AD_Value[0]沒有值,也沒查到原因,請(qǐng)哥哥指導(dǎo)下們
歡迎光臨 (http://www.torrancerestoration.com/bbs/)
Powered by Discuz! X3.1