工程可見Github<傳送門>
?導(dǎo)讀:《藍(lán)橋杯嵌入式組》專欄文章是博主2019年參加藍(lán)橋杯的嵌入式組比賽所做的學(xué)習(xí)筆記,在當(dāng)年的比賽中,由于忙于準(zhǔn)備考研及保研相關(guān)工作,博主僅僅參加了當(dāng)年的省賽,并獲得了省賽一等獎(jiǎng)的成績。成績雖談不上最好,但至少問心無愧。如今2021年回頭再看該系列文章,仍然感觸頗多。為了能更好地幫助到單片機(jī)初學(xué)者,今年特地抽出時(shí)間對(duì)當(dāng)年的文章邏輯和結(jié)構(gòu)進(jìn)行重構(gòu),以達(dá)到初學(xué)者快速上手的目的。需要指出的是,由于本人水平有限,如有錯(cuò)誤還請(qǐng)讀者指出,非常感謝。那么,接下來讓我們一起開始愉快的學(xué)習(xí)吧。
一、主要代碼
main.c
/*******************************************************************************
* 文件名:main.c
* 描 述:
* 作 者:CLAY
* 版本號(hào):v1.0.0
* 日 期: 2019年2月17日
* 備 注:修改后的LCD例程
* 滑變對(duì)應(yīng)的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,別忘了這個(gè)初始化
E2Write(0x01, 0xAA);//在0x01這個(gè)地址寫下,0xAA這個(gè)數(shù)據(jù)
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轉(zhuǎn)換結(jié)果
sprintf((char *)str, "ADC = %.2f V", dat_ADC * 3.3 / 4096);//12位ADC對(duì)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時(shí)鐘和ADC1時(shí)鐘
RCC_ADCCLKConfig(RCC_PCLK2_Div6); //ADC時(shí)鐘最大14M,至少進(jìn)行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獨(dú)立模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE;//單通道
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//單次轉(zhuǎn)換
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//軟件觸發(fā)
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//數(shù)據(jù)右對(duì)齊
ADC_InitStructure.ADC_NbrOfChannel = 1;//通道數(shù)目1
ADC_Init(ADC1, &ADC_InitStructure);
ADC_Cmd(ADC1, ENABLE);//使能ADC
//ADC校準(zhǔn)(校準(zhǔn)前必須先使能ADC)
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é)束
}
u16 Get_ADC(void)
{
u16 tmp;
ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 1, ADC_SampleTime_239Cycles5);//轉(zhuǎn)換時(shí)間239.5個(gè)周期
ADC_SoftwareStartConvCmd(ADC1, ENABLE);//軟件觸發(fā)
while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC) == 0);//等待轉(zhuǎn)換完成
tmp = ADC_GetConversionValue(ADC1);//獲取轉(zhuǎn)換結(jié)果(讀取轉(zhuǎn)換結(jié)果會(huì)自動(dòng)清0-EOC標(biāo)志位)
ADC_SoftwareStartConvCmd(ADC1, DISABLE);//啟動(dòng)下一次軟件轉(zhuǎn)換
return tmp;
}
adc.h
#ifndef _ADC_H
#define _ADC_H
void ADC1_Init(void);
u16 Get_ADC(void);
#endif
二、需要注意的地方
2.1、參考例程
路徑為:...\嵌入式設(shè)計(jì)與開發(fā)項(xiàng)目加密資料\嵌入式設(shè)計(jì)與開發(fā)\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)
會(huì)出現(xiàn)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")
重復(fù)定義的錯(cuò)誤,所以這里寫成ADC1_Init(void)
2.3、校準(zhǔn)前必須先開啟ADC!
2.4、移植程序需要改的幾個(gè)地方
①、時(shí)鐘配置函數(shù) 我們使用的ADC_IN8在PB0,所以要修改 RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC, ENABLE);
為GPIOB
因?yàn)锳DC最大時(shí)鐘14M,而APB2最大72M,所以至少需要經(jīng)過6分頻! RCC_ADCCLKConfig(RCC_PCLK2_Div4);
改為 RCC_PCLK2_Div6
②、GPIO配置 使用的是PB0,所以GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
需要改為GPIO_Pin_0
,同時(shí)最后的結(jié)構(gòu)體初始化,也要把GPIO_Init(GPIOC, &GPIO_InitStructure);
改為GPIOB
。
③、ADC配置 ADC_InitStructure.ADC_ScanConvMode = ENABLE; ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
掃描模式和連續(xù)轉(zhuǎn)換模式都需要改為DISABLE
...
④、規(guī)則組轉(zhuǎn)換配置函數(shù) ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 1, ADC_SampleTime_55Cycles5);
需要改成通道8
,轉(zhuǎn)換時(shí)間是239.5
個(gè)周期ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 1, ADC_SampleTime_239Cycles5);
⑤、轉(zhuǎn)換完成后數(shù)據(jù)處理
while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == 0); tmp = ADC_GetConversionValue(ADC1);
這兩個(gè)函數(shù)是等待轉(zhuǎn)換完成并獲取轉(zhuǎn)換結(jié)果的函數(shù),例程中沒有,需要在stm32f10x_adc.h中找到函數(shù)定義。
轉(zhuǎn)換完成標(biāo)志,對(duì)應(yīng)ADC_FLAG參數(shù),如有遺忘可繼續(xù)F12追根溯源到.c文件查看
接著,光標(biāo)定位到IS_ADC_GET_FLAG
點(diǎn)擊F12
即可找到。
⑥、切記獲取到的ADC值,進(jìn)行* 3.3 / 4096
進(jìn)行量化, 并且最后的值存入一個(gè)float類型中
結(jié)語:以上就是本篇文章的全部內(nèi)容啦,希望大家可以多多支持我的原創(chuàng)文章。如有錯(cuò)誤,請(qǐng)及時(shí)指正,非常感謝。