工程可見Github<傳送門>
?導讀:《藍橋杯嵌入式組》專欄文章是博主2019年參加藍橋杯的嵌入式組比賽所做的學習筆記,在當年的比賽中,由于忙于準備考研及保研相關工作,博主僅僅參加了當年的省賽,并獲得了省賽一等獎的成績。成績雖談不上最好,但至少問心無愧。如今2021年回頭再看該系列文章,仍然感觸頗多。為了能更好地幫助到單片機初學者,今年特地抽出時間對當年的文章邏輯和結構進行重構,以達到初學者快速上手的目的。需要指出的是,由于本人水平有限,如有錯誤還請讀者指出,非常感謝。那么,接下來讓我們一起開始愉快的學習吧。
一、主要代碼
main.c
/*******************************************************************************
* 文件名:main.c
* 描 述:
* 作 者:CLAY
* 版本號:v1.0.0
* 日 期: 2019年2月17日
* 備 注:修改后的LCD例程
* 滑變對應的PB0(ADC1_IN8)AD值,量化為0~3.3V顯示在LCD上
*******************************************************************************
*/
#include "stm32f10x.h"
#include "lcd.h"
#include "e2prom.h"
#include "stdio.h"
#include "i2c.h"
#include "adc.h"
u32 TimingDelay = 0;
void Delay_Ms(u32 nTime);
//Main Body
int main(void)
{
u8 str[20];
u8 dat;
float dat_ADC;
STM3210B_LCD_Init();
LCD_Clear(Blue);
LCD_SetBackColor(Blue);
LCD_SetTextColor(White);
SysTick_Config(SystemCoreClock/1000);
i2c_init();//FUCK,別忘了這個初始化
E2Write(0x01, 0xAA);//在0x01這個地址寫下,0xAA這個數據
Delay_Ms(5);
dat = E2Read(0x01);
LCD_ClearLine(Line5);
sprintf((char *)str, "data = %d", dat);
LCD_DisplayStringLine(Line5, str);
ADC1_Init();
while(1)
{
dat_ADC = Get_ADC();//獲取ADC轉換結果
sprintf((char *)str, "ADC = %.2f V", dat_ADC * 3.3 / 4096);//12位ADC對3.3V量化
LCD_DisplayStringLine(Line2, str);
Delay_Ms(500);
}
}
//
void Delay_Ms(u32 nTime)
{
TimingDelay = nTime;
while(TimingDelay != 0);
}
adc.c
#include "stm32f10x.h"
void ADC1_Init(void)
{
ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOB, ENABLE);//開啟GPIO時鐘和ADC1時鐘
RCC_ADCCLKConfig(RCC_PCLK2_Div6); //ADC時鐘最大14M,至少進行6分頻!!!
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;//選中PB0引腳
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;//模擬輸入
GPIO_Init(GPIOB, &GPIO_InitStructure);
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//ADC獨立模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE;//單通道
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//單次轉換
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//軟件觸發
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//數據右對齊
ADC_InitStructure.ADC_NbrOfChannel = 1;//通道數目1
ADC_Init(ADC1, &ADC_InitStructure);
ADC_Cmd(ADC1, ENABLE);//使能ADC
//ADC校準(校準前必須先使能ADC)
ADC_ResetCalibration(ADC1);//使能復位校準
while(ADC_GetResetCalibrationStatus(ADC1));//等待復位校準結束
ADC_StartCalibration(ADC1);//開啟AD校準
while(ADC_GetCalibrationStatus(ADC1));//等待校準結束
}
u16 Get_ADC(void)
{
u16 tmp;
ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 1, ADC_SampleTime_239Cycles5);//轉換時間239.5個周期
ADC_SoftwareStartConvCmd(ADC1, ENABLE);//軟件觸發
while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC) == 0);//等待轉換完成
tmp = ADC_GetConversionValue(ADC1);//獲取轉換結果(讀取轉換結果會自動清0-EOC標志位)
ADC_SoftwareStartConvCmd(ADC1, DISABLE);//啟動下一次軟件轉換
return tmp;
}
adc.h
#ifndef _ADC_H
#define _ADC_H
void ADC1_Init(void);
u16 Get_ADC(void);
#endif
二、需要注意的地方
2.1、參考例程
路徑為:...\嵌入式設計與開發項目加密資料\嵌入式設計與開發\STM32固件庫v3.5\stm32f10x_stdperiph_lib\STM32F10x_StdPeriph_Lib_V3.5.0\Project\STM32F10x_StdPeriph_Examples\ADC\ADC1_DMA\main.c
2.2、不要寫成ADC_Init(void)
會出現error: #147-D: declaration is incompatible with "void ADC_Init(ADC_TypeDef *, ADC_InitTypeDef *)" (declared at line 429 of "..\Libraries\STM32F10x_StdPeriph_Driver\inc\stm32f10x_adc.h")
重復定義的錯誤,所以這里寫成ADC1_Init(void)
2.3、校準前必須先開啟ADC!
2.4、移植程序需要改的幾個地方
①、時鐘配置函數 我們使用的ADC_IN8在PB0,所以要修改 RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC, ENABLE);
為GPIOB
因為ADC最大時鐘14M,而APB2最大72M,所以至少需要經過6分頻! RCC_ADCCLKConfig(RCC_PCLK2_Div4);
改為 RCC_PCLK2_Div6
②、GPIO配置 使用的是PB0,所以GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
需要改為GPIO_Pin_0
,同時最后的結構體初始化,也要把GPIO_Init(GPIOC, &GPIO_InitStructure);
改為GPIOB
。
③、ADC配置 ADC_InitStructure.ADC_ScanConvMode = ENABLE; ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
掃描模式和連續轉換模式都需要改為DISABLE
...
④、規則組轉換配置函數 ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 1, ADC_SampleTime_55Cycles5);
需要改成通道8
,轉換時間是239.5
個周期ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 1, ADC_SampleTime_239Cycles5);
⑤、轉換完成后數據處理
while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == 0); tmp = ADC_GetConversionValue(ADC1);
這兩個函數是等待轉換完成并獲取轉換結果的函數,例程中沒有,需要在stm32f10x_adc.h中找到函數定義。
轉換完成標志,對應ADC_FLAG參數,如有遺忘可繼續F12追根溯源到.c文件查看
接著,光標定位到IS_ADC_GET_FLAG
點擊F12
即可找到。
⑥、切記獲取到的ADC值,進行* 3.3 / 4096
進行量化, 并且最后的值存入一個float類型中
結語:以上就是本篇文章的全部內容啦,希望大家可以多多支持我的原創文章。如有錯誤,請及時指正,非常感謝。