折騰了兩天ADC多通道采樣,采樣的結(jié)果都很亂,完全不是預(yù)期值。在amobbs求助也沒有找到結(jié)果。于是決定從簡單開始,一步步折騰著破ADC。
【ADC試驗(yàn)1實(shí)驗(yàn)說明】
1、這個實(shí)驗(yàn)僅僅是初始化一個ADC,對其輸入進(jìn)行采樣。
2、使用STM32F4的ADC1進(jìn)行采樣,采樣值不輸出之在編譯器里邊觀察。
3、使用ST外設(shè)庫進(jìn)行實(shí)驗(yàn)
4、本實(shí)驗(yàn)只為采集到數(shù)據(jù),采樣周期、采樣間隔設(shè)置為最大。
【ADC試驗(yàn)1實(shí)驗(yàn)結(jié)果】
成功采集到了ADC1,通道1引腳PA1上的輸入。數(shù)據(jù)穩(wěn)定不跳變。
【ADC試驗(yàn)1實(shí)驗(yàn)步驟】
1、首先懷疑是工程中使用的USART、EXTI什么的影響了ADC的。重建工程,加入ST外設(shè)庫,添加引用位置。這一步就不說了。
2、開啟GPIOA、ADC時鐘。因?yàn)槭褂肁DC1的通道1,對應(yīng)的PA1引腳作為輸入。
ADC掛接在APB2時鐘上,GPIOA掛接在AHB1時鐘上。所以要開啟這兩個時鐘。
代碼如下:
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
3、輸入引腳配置
輸入端口PA,引腳1.模擬輸入,引腳時鐘100M
//PA1 PA2 PA3,模擬輸入
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_InitStructure.GPIO_Pin = GPIO_PinSource1 | GPIO_PinSource2 | GPIO_PinSource3;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
GPIO_Init(GPIOA,&GPIO_InitStructure);
4、ADC通用的初始化
這是F4系列新出來的東西,與F1不同。這里通過庫函數(shù)配置一個新增的寄存器ADC_CCR,這個配置將影響片上所有的ADC。
//ADC通用配置
ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled; //不用DMA
ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent; //獨(dú)立采樣
ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div8; //速度盡可能低
ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_20Cycles;
ADC_CommonInit(&ADC_CommonInitStructure);
5、ADC1初始化
之后對單獨(dú)的ADC1進(jìn)行配置,設(shè)定一些參數(shù)。最后的ADC_RegularChannelConfig完成通道選擇和掃描順序設(shè)置
主要參數(shù):連續(xù)采樣/數(shù)據(jù)右對齊/不用外部觸發(fā)/采樣通道數(shù)1/12位精度
//ADC1配置
ADC_InitStructyre.ADC_ContinuousConvMode = ENABLE;
ADC_InitStructyre.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructyre.ADC_ExternalTrigConv = ADC_ExternalTrigConvEdge_None;
ADC_InitStructyre.ADC_NbrOfConversion = 1; //通道數(shù)
ADC_InitStructyre.ADC_Resolution = ADC_Resolution_12b;
ADC_InitStructyre.ADC_ScanConvMode = ENABLE;
ADC_Init(ADC1,&ADC_InitStructyre);
ADC_Cmd(ADC1,ENABLE);
ADC_RegularChannelConfig(ADC1,ADC_Channel_1,1,ADC_SampleTime_144Cycles);
6、手動開始ADC采樣,由于配置為連續(xù)掃描所有只需要開啟一次即可:
使用函數(shù)ADC_SoftwareStartConv(ADC1);完成
7、獲取ADC采樣數(shù)據(jù):
可以通過讀ADC的DR或者使用庫函數(shù)ADC_GetConversionValue來完成。我使用的是庫函數(shù)。
adcvalue1 = ADC_GetConversionValue(ADC1);
adcvota = adcvalue1 *3300 / 0xfff;
8、最后加上串口功能,讓數(shù)據(jù)回顯
while(1)
{
adcvalue1 = ADC_GetConversionValue(ADC1);
adcvota = adcvalue1 *3300 / 0xfff;
for(i = 0;i<10000;i++)
{
sum += adcvota;
if(i ==9999)
{
avgvota = sum/10000;
sum = 0;
printf("avg vota is: %drn",avgvota);
}
}
}
9、完整的代碼:
#include
#include "stm32f4xx_rcc.h"
#include "stm32f4xx_gpio.h"
#include "stm32f4xx_adc.h"
#include "stm32f4xx_conf.h"#include "usart.h"
#include
NVIC_InitTypeDef NVIC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
ADC_CommonInitTypeDef ADC_CommonInitStructure;
ADC_InitTypeDef ADC_InitStructyre;int adcvalue1,adcvalue2,adcvalue3;
int adcvota ;int i,sum,avgvota;
int main()
{
USART_Config();
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
//PA1 PA2 PA3,模擬輸入
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
// GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_Pin = GPIO_PinSource1 | GPIO_PinSource2 | GPIO_PinSource3;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
GPIO_Init(GPIOA,&GPIO_InitStructure);
//ADC通用配置
ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled; //不用DMA
ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent; //獨(dú)立采樣
ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div8; //速度盡可能低
ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_20Cycles;
ADC_CommonInit(&ADC_CommonInitStructure);
//ADC1配置
ADC_InitStructyre.ADC_ContinuousConvMode = ENABLE;
ADC_InitStructyre.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructyre.ADC_ExternalTrigConv = ADC_ExternalTrigConvEdge_None;
ADC_InitStructyre.ADC_NbrOfConversion = 1;
ADC_InitStructyre.ADC_Resolution = ADC_Resolution_12b;
ADC_InitStructyre.ADC_ScanConvMode = ENABLE;
ADC_Init(ADC1,&ADC_InitStructyre);
ADC_Cmd(ADC1,ENABLE);
ADC_RegularChannelConfig(ADC1,ADC_Channel_1,1,ADC_SampleTime_144Cycles);
ADC_SoftwareStartConv(ADC1);
sum = 0;
while(1)
{
adcvalue1 = ADC_GetConversionValue(ADC1);
adcvota = adcvalue1 *3300 / 0xfff;
for(i = 0;i<10000;i++)
{
sum += adcvota;
if(i ==9999)
{
avgvota = sum/10000;
sum = 0;
printf("avg vota is: %drn",avgvota);
}
}
}
}
//重定義printf
int fputc(int ch,FILE *f)
{
//檢查發(fā)送寄存器為空之后再往里邊放數(shù)據(jù)
while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET)
{}
//USART發(fā)送一個字符
USART_SendData(USART1, (uint8_t) ch);
return ch;
}