STM32的ADC編程方法


  記錄一下STM32的ADC編程方法!

        前面已經學習了DMA,知道如何使用DMA去減小CPU的負擔,這里的ADC轉換也來使用DMA---這個也是STM32的ADC轉換最常見的方式。

 

---第一步是---了解STM32的ADC對應的GPIO口----如下圖---不用記住,可以查詢,我是將它剪下來粘貼到書本的相應章節----!

 

---第二步是---配置相應ADC轉換的GPIO口----這里使用PC0--PC1

static void ADC1_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;

RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);  //打開DMA1的時鍾
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC, ENABLE);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 |GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;  //模擬輸入
GPIO_Init(GPIOC, &GPIO_InitStructure);
}

---第三步是---配置ADC的DMA----配置ADC通道等---

#define ADC1_DR_Address    ((u32)0x40012400+0x4c)  //外設地址
__IO uint16_t ADC_ConvertedValue[2];  //內存數組

 

static void ADC1_Mode_Config(void)
{
DMA_InitTypeDef DMA_InitStructure;
ADC_InitTypeDef ADC_InitStructure;

DMA_DeInit(DMA1_Channel1);  
//---------------ADC的DMA配置--------------------
DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;  //ADC1地址---代表ADC1保存轉換值的寄存器
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue;  //內存地址---用來保存DMA傳輸過來的ADC轉換值----后面直接使用的變量地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;  //外設為數據源
DMA_InitStructure.DMA_BufferSize = 2;  //傳輸總數據---2通道需要傳輸2個數據
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  //外設地址固定
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  //內存地址自增---總體表示始終從外設ADC1地址處取值---依次保存到連續的兩個內存變量中---
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;   //外設傳輸數據單元---半字16位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;  //內存傳輸數據單元---半字16位
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;  //循環模式---2個數據依次循環接收從外設ADC1傳輸過來的ADC值---
DMA_InitStructure.DMA_Priority = DMA_Priority_High;  //高優先級
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  //禁止內存傳內存
DMA_Init(DMA1_Channel1, &DMA_InitStructure);

DMA_Cmd(DMA1_Channel1, ENABLE);  //再次打開DMA1

        //------------ADC模式配置------------------------
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;  //獨立模式----還有很多模式---這個比較常見
ADC_InitStructure.ADC_ScanConvMode = ENABLE ;   //掃描模式---采集多通道使用----本程序采集2通道---所以掃描模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;  //連續轉換模式---不難理解---就是不停地采集---一次接一次
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;  //不使用外部觸發轉換---觸發分為外部觸發---比如中斷與定時器。軟件觸發---后面有專用函數
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;   //采集的數據右對齊---方便計算
ADC_InitStructure.ADC_NbrOfChannel = 2;  //總共需要轉換的通道個數---這里2個
ADC_Init(ADC1, &ADC_InitStructure);

RCC_ADCCLKConfig(RCC_PCLK2_Div8);  //配置ADC轉換時鍾---PCLK2的8分頻
        //下面這個函數比較重要----配置ADC的通道與采樣周期---前面說的PC0與PC1對應的ADC通道分別是--10與11。采集周期也有幾種。
ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 1, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 2, ADC_SampleTime_55Cycles5);

ADC_DMACmd(ADC1, ENABLE);  //打開DMA1的ADC1
ADC_Cmd(ADC1, ENABLE);  //打開ADC1

ADC_ResetCalibration(ADC1);  //復位校准寄存器
while(ADC_GetResetCalibrationStatus(ADC1));  //等待校准寄存器復位完成

ADC_StartCalibration(ADC1);  //ADC校准
while(ADC_GetCalibrationStatus(ADC1));  //校准完成

ADC_SoftwareStartConvCmd(ADC1, ENABLE);  //軟件觸發轉換
}

---第四部分是---在硬件上使用了一個通道切換芯片----CD4052----由PC2---PC3控制通道的選擇CD4052切換控制GPIO配置----

void GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  //推完輸出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOC, &GPIO_InitStructure);
}

---第五部分是---主函數-----

extern __IO uint16_t ADC_ConvertedValue[2];  //聲明外部變量
uint16_t My_ADC[2];  //求平均值

int main(void)
{
u8 i,led=0x01;

USART1_Config();


ADC1_GPIO_Config();
ADC1_Mode_Config();


while (1)
{
My_ADC[0]=0;
My_ADC[1]=0;


for(i=0;i<10;i++)  
{
                        My_ADC[0]+=ADC_ConvertedValue[0];
My_ADC[1]+=ADC_ConvertedValue[1];
               }
My_ADC[0]=My_ADC[0]/10;   //采集10次求平均值
My_ADC[1]=My_ADC[1]/10;

ADC_ConvertedValueLocal =(float) My_ADC[0]/4096*3.3;   //轉換為電壓值

printf("\r\n The current AD---0 value = 0x%04X \n", My_ADC[0]); 
printf("The current AD---0 value = %f V \n",ADC_ConvertedValueLocal); 

ADC_ConvertedValueLocal =(float) My_ADC[1]/4096*3.3; 

printf("The current AD---1 value = 0x%04X \n", My_ADC[1]); 
printf("The current AD---1 value = %f V \n",ADC_ConvertedValueLocal);  
}
}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM