STM32筆記(三)ADC、DMA、USART的綜合練習_第1頁
STM32筆記(三)ADC、DMA、USART的綜合練習_第2頁
STM32筆記(三)ADC、DMA、USART的綜合練習_第3頁
STM32筆記(三)ADC、DMA、USART的綜合練習_第4頁
STM32筆記(三)ADC、DMA、USART的綜合練習_第5頁
已閱讀5頁,還剩6頁未讀 繼續免費閱讀

下載本文檔

版權說明:本文檔由用戶提供并上傳,收益歸屬內容提供方,若內容存在侵權,請進行舉報或認領

文檔簡介

STM32筆記(三)ADC、DMA、USART的綜合練習這是一個綜合的例子,演示了ADC模塊、DMA模塊和USART模塊的基本使用。

我們在這里設置ADC為連續轉換模式,常規轉換序列中有兩路轉換通道,分別是ADC_CH10(PC0)和ADC_CH16(片內溫度傳感器)。因為使用了自動多通道轉換,數據的取出工作最適合使用DMA方式取出,so,我們在內存里開辟了一個u16AD_Value[2]數組,并設置了相應的DMA模塊,使ADC在每個通道轉換結束后啟動DMA傳輸,其緩沖區數據量為2個HalfWord,使兩路通道的轉換結果自動的分別落到AD_Value[0]和AD_Value[1]中。

然后,在主函數里,就無需手動啟動AD轉換,等待轉換結束,再取結果了。我們可以在主函數里隨時取AD_Value中的數值,那里永遠都是最新的AD轉換結果。

如果我們定義一個更大的AD_Value數組,并調整DMA的傳輸數據量(BufferSize)可以實現AD結果的循環隊列存儲,從而可以進行各種數字濾波算法。

接著,取到轉換結果后,根據V=(AD_Value/4096)*Vref+的公式可以算出相應通道的電壓值,也可以根據

T(℃)=

(1.43-Vad)/34*10^(-6)+25的算法,得到片內溫度傳感器的測量溫度值了。

通過重新定義putchar函數,及包含"stdio.h"頭文件,我們可以方便的使用標準C的庫函數printf(),實現串口通信。

相關的官方例程,可以參考FWLibV2.0的ADC\ADC1_DMA和USART\printf兩個目錄下的代碼。

本代碼例子是基于萬利199的開發板EK-STM32F實現,CPU=STM32F103VBT6

/******************************************************************************

*本文件實現ADC模塊的基本功能

*設置ADC1的常規轉換序列包含CH10和CH16(片內溫度傳感器)

*設置了連續轉換模式,并使用DMA傳輸

*AD轉換值被放在了AD_Value[2]數組內,[0]保存CH0結果,[1]保存CH16結果

*GetVolt函數計算[0]的值對應的電壓值(放大100倍,保留2位小數)

*GetTemp函數計算[1]的值對應的溫度值,計算公式在相應函數內有說明

*

作者:jjldc(九九)

*******************************************************************************/

/*Includes------------------------------------------------------------------*/

#include"stm32f10x_lib.h"

#include"stdio.h"

/*Privatetypedef-----------------------------------------------------------*/

/*Privatedefine------------------------------------------------------------*/

#defineADC1_DR_Address

((u32)0x4001244C)

/*Privatemacro-------------------------------------------------------------*/

/*Privatevariables---------------------------------------------------------*/

vu16AD_Value[2];

vu16i=0;

s16

Temp;

u16

Volt;

/*Privatefunctionprototypes-----------------------------------------------*/

voidRCC_Configuration(void);

voidGPIO_Configuration(void);

voidNVIC_Configuration(void);

voidUSART1_Configuration(void);

voidADC1_Configuration(void);

voidDMA_Configuration(void);

intfputc(intch,FILE*f);

voidDelay(void);

u16GetTemp(u16advalue);

u16GetVolt(u16advalue);

/*Privatefunctions---------------------------------------------------------*/

/*******************************************************************************

*FunctionName

:main

*Description

:Mainprogram.

*Input

:None

*Output

:None

*Return

:None

*******************************************************************************/

intmain(void)

{

RCC_Configuration();

GPIO_Configuration();

NVIC_Configuration();

USART1_Configuration();

DMA_Configuration();

ADC1_Configuration();

//啟動第一次AD轉換

ADC_SoftwareStartConvCmd(ADC1,ENABLE);

//因為已經配置好了DMA,接下來AD自動連續轉換,結果自動保存在AD_Value處

while(1)

{

Delay();

Temp=GetTemp(AD_Value[1]);

Volt=GetVolt(AD_Value[0]);

USART_SendData(USART1,0x0c);

//清屏

//注意,USART_SendData函數不檢查是否發送完成

//等待發送完成

while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);

printf("電壓:%d.%d\t溫度:%d.%d℃\r\n",\

Volt/100,Volt%100,Temp/100,Temp%100);

}

}

/*******************************************************************************

*FunctionName

:重定義系統putchar函數intfputc(intch,FILE*f)

*Description

:串口發一個字節

*Input

:intch,FILE*f

*Output

:

*Return

:intch

*******************************************************************************/

intfputc(intch,FILE*f)

{

//USART_SendData(USART1,(u8)ch);

USART1->DR=(u8)ch;

/*Loopuntiltheendoftransmission*/

while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET)

{

}

returnch;

}

/*******************************************************************************

*FunctionName

:Delay

*Description

:延時函數

*Input

:None

*Output

:None

*Return

:None

*******************************************************************************/

voidDelay(void)

{

u32i;

for(i=0;i<0x4f0000;i++);

return;

}

/*******************************************************************************

*FunctionName

:GetTemp

*Description

:根據ADC結果計算溫度

*Input

:u16advalue

*Output

:

*Return

:u16temp

*******************************************************************************/

u16GetTemp(u16advalue)

{

u32Vtemp_sensor;

s32Current_Temp;

//

ADC轉換結束以后,讀取ADC_DR寄存器中的結果,轉換溫度值計算公式如下:

//

V25-VSENSE

//

T(℃)=------------

+25

//

Avg_Slope

//

V25:

溫度傳感器在25℃時的輸出電壓,典型值1.43V。

//

VSENSE:溫度傳感器的當前輸出電壓,與ADC_DR寄存器中的結果ADC_ConvertedValue之間的轉換關系為:

//

ADC_ConvertedValue*Vdd

//

VSENSE=--------------------------

//

Vdd_convert_value(0xFFF)

//

Avg_Slope:溫度傳感器輸出電壓和溫度的關聯參數,典型值4.3mV/℃。

Vtemp_sensor=advalue*330/4096;

Current_Temp=(s32)(143-Vtemp_sensor)*10000/43+2500;

return(s16)Current_Temp;

}

/*******************************************************************************

*FunctionName

:GetVolt

*Description

:根據ADC結果計算電壓

*Input

:u16advalue

*Output

:

*Return

:u16temp

*******************************************************************************/

u16GetVolt(u16advalue)

{

return(u16)(advalue*330/4096);

}

/*******************************************************************************

*FunctionName

:RCC_Configuration

*Description

:系統時鐘設置

*Input

:None

*Output

:None

*Return

:None

*******************************************************************************/

voidRCC_Configuration(void)

{

ErrorStatusHSEStartUpStatus;

//使能外部晶振

RCC_HSEConfig(RCC_HSE_ON);

//等待外部晶振穩定

HSEStartUpStatus=RCC_WaitForHSEStartUp();

//如果外部晶振啟動成功,則進行下一步操作

if(HSEStartUpStatus==SUCCESS)

{

//設置HCLK(AHB時鐘)=SYSCLK

RCC_HCLKConfig(RCC_SYSCLK_Div1);

//PCLK1(APB1)=HCLK/2

RCC_PCLK1Config(RCC_HCLK_Div2);

//PCLK2(APB2)=HCLK

RCC_PCLK2Config(RCC_HCLK_Div1);

//設置ADC時鐘頻率

RCC_ADCCLKConfig(RCC_PCLK2_Div2);

//FLASH時序控制

//推薦值:SYSCLK=0~24MHz

Latency=0

//

SYSCLK=24~48MHz

Latency=1

//

SYSCLK=48~72MHz

Latency=2

FLASH_SetLatency(FLASH_Latency_2);

//開啟FLASH預取指功能

FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);

//PLL設置SYSCLK/1*9=8*1*9=72MHz

RCC_PLLConfig(RCC_PLLSource_HSE_Div1,RCC_PLLMul_9);

//啟動PLL

RCC_PLLCmd(ENABLE);

//等待PLL穩定

while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY)==RESET);

//系統時鐘SYSCLK來自PLL輸出

RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);

//切換時鐘后等待系統時鐘穩定

while(RCC_GetSYSCLKSource()!=0x08);

}

//下面是給各模塊開啟時鐘

//啟動GPIO

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB|\

RCC_APB2Periph_GPIOC|RCC_APB2Periph_GPIOD,\

ENABLE);

//啟動AFIO

RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);

//啟動USART1

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);

//啟動DMA時鐘

RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);

//啟動ADC1時鐘

RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);

}

/*******************************************************************************

*FunctionName

:GPIO_Configuration

*Description

:GPIO設置

*Input

:None

*Output

:None

*Return

:None

*******************************************************************************/

voidGPIO_Configuration(void)

{

GPIO_InitTypeDefGPIO_InitStructure;

//PC口4567腳設置GPIO輸出,推挽2M

GPIO_InitStructure.GPIO_Pin=GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7;

GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;

GPIO_InitStructure.GPIO_Speed=GPIO_Speed_2MHz;

GPIO_Init(GPIOC,&GPIO_InitStructure);

//KEY2KEY3JOYKEY

//位于PD口的3411-15腳,使能設置為輸入

GPIO_InitStructure.GPIO_Pin=GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_11|GPIO_Pin_12|\

GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15;

GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING;

GPIO_Init(GPIOD,&GPIO_InitStructure);

//USART1_TX

GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;

GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;

GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;

GPIO_Init(GPIOA,&GPIO_InitStructure);

//USART1_RX

GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10;

GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING;

GPIO_Init(GPIOA,&GPIO_InitStructure);

//ADC_CH10-->PC0

GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10;

GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AIN;

GPIO_Init(GPIOC,&GPIO_InitStructure);

}

/*******************************************************************************

*FunctionName

:NVIC_Configuration

*Description

:NVIC設置

*Input

:None

*Output

:None

*Return

:None

*******************************************************************************/

voidNVIC_Configuration(void)

{

NVIC_InitTypeDefNVIC_InitStructure;

#ifdef

VECT_TAB_RAM

//SettheVectorTablebaselocationat0x20000000

NVIC_SetVectorTable(NVIC_VectTab_RAM,0x0);

#else

/*VECT_TAB_FLASH

*/

//SettheVectorTablebaselocationat0x08000000

NVIC_SetVectorTable(NVIC_VectTab_FLASH,0x0);

#endif

//設置NVIC優先級分組為Group2:0-3搶占式優先級,0-3的響應式優先級

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);

//串口中斷打開

NVIC_InitStructure.NVIC_IRQChannel=USART1_IRQChannel;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0;

NVIC_InitStructure.NVIC_IRQChannelSubPriority=1;

NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;

NVIC_Init(&NVIC_InitStructure);

}

/*******************************************************************************

*FunctionName

:USART1_Configuration

*Description

:NUSART1設置

*Input

:None

*Output

:None

*Return

:None

*******************************************************************************/

voidUSART1_Configuration(void)

{

USART_InitTypeDefUSART_InitStructure;

USART_InitStructure.USART_BaudRate=19200;

USART_InitStructure.USART_WordLength=USART_WordLength_8b;

USART_InitStructure.USART_StopBits=USART_StopBits_1;

USART_InitStructure.USART_Parity=USART_Parity_No;

USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None;

USART_InitStructure.USART_Mode=USART_Mode_Tx|USART_Mode_Rx;

USART_Init(USART1,&USART_InitStructure);

USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);

USART_Cmd(USART1,ENABLE);

}

/*******************************************************************************

*FunctionName

:ADC1_Configuration

*Description

:ADC1設置(包括ADC模塊配置和自校準)

*Input

:None

*Output

:None

*Return

:None

*******************************************************************************/

voidADC1_Configuration(void)

{

ADC_InitTypeDefADC_InitStructure;

ADC_InitStructure.ADC_Mode=ADC_Mode_Independent;

ADC_InitStructure.ADC_ScanConvMode=ENABLE;

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);

//ADC內置溫度傳感器使能(要使用片內溫度傳感器,切忌要開啟它)

ADC_TempSensorVrefintCmd(ENABLE);

//常規轉換序列1:通道10

ADC_RegularChannelConfig(ADC1,ADC_Channel_10,1,ADC_SampleTime_13Cycles5);

//常規轉換序列2:通道16(內部溫度傳感器),采樣時間>2.2us,(239cycles)

ADC_RegularChannelConfig(ADC1,ADC_Channel_16,2,ADC_SampleTime_239Cycles5);

//EnableADC1

ADC_Cmd(ADC1,ENABLE);

//開啟ADC的DMA支持(要實現DMA功能,還需獨立配置DMA通道等參數)

ADC_DMACmd(ADC1,ENABLE);

//下面是ADC自動校準,開機后需執行一次,保證精度

//EnableADC1resetcalibarationregister

ADC_ResetCalibration(ADC1);

//ChecktheendofADC1resetcalibrationregister

while(ADC_GetResetCalibrationStatus(ADC1));

//StartADC1calibaration

ADC_StartCalibration(ADC1);

//ChecktheendofADC1calibration

whil

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯系上傳者。文件的所有權益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
  • 4. 未經權益所有人同意不得將文件中的內容挪作商業或盈利用途。
  • 5. 人人文庫網僅提供信息存儲空間,僅對用戶上傳內容的表現方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
  • 6. 下載文件中如有侵權或不適當內容,請與我們聯系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論