原文來自
http://www.51hei.com/bbs/dpj-39885-1.html
今天說一下STM32單片機的接收不定長度字節數據的方法。由於STM32單片機帶IDLE中斷,所以利用這個中斷,可以接收不定長字節的數據,由於STM32屬於ARM單片機,所以這篇文章的方法也適合其他的ARM單片機。
IDLE中斷什麼時候發生?
IDLE就是串口收到一幀數據後,發生的中斷。什麼是一幀數據呢?比如說給單片機一次發來1個字節,或者一次發來8個字節,這些一次發來的數據,就稱爲一幀數據,也可以叫做一包數據。
如何判斷一幀數據結束,就是我們今天討論的問題。因爲很多項目中都要用到這個,因爲只有接收到一幀數據以後,你纔可以判斷這次收了幾個字節和每個字節的內容是否符合協議要求。
看了前面IDLE中斷的定義,你就會明白了,一幀數據結束後,就會產生IDLE中斷。這個中斷真是太TMD有用了。省去了好多判斷的麻煩。
如何配置好IDLE中斷?
下面我們就配置好串口IDLE中斷吧。
這是串口CR1寄存器,其中,對bit4寫1開啓IDLE中斷,對bit5寫1開啓接收數據中斷。(注意:不同系列的STM32,對應的寄存器位可能不同)
(RXNE中斷和IDLE中斷的區別?
當接收到1個字節,就會產生RXNE中斷,當接收到一幀數據,就會產生IDLE中斷。比如給單片機一次性發送了8個字節,就會產生8次RXNE中斷,1次IDLE中斷。)
這是狀態寄存器,當串口接收到數據時,bit5就會自動變成1,當接收完一幀數據後,bit4就會變成1.
需要注意的是,在中斷函數裏面,需要把對應的位清0,否則會影響下一次數據的接收。比如RXNE接收數據中斷,只要把接收到的一個字節讀出來,就會清除這個中斷。IDLE中斷,如何是F0系列的單片機,需要用ICR寄存器來清除,如果是F1系列的單片機,清除方法是“先讀SR寄存器,再讀DR寄存器”。(我怎麼知道?手冊上寫的)
下面以STM32F103爲例給出源程序。
我們先來看程序中的主要部分。
串口初始化函數片段
%E7%A9%BA%E9%97%B2%E4%B8%AD%E6%96%AD.png (17.99 KB, 下載次數: 45)
下載附件 保存到相冊
2015-10-29 23:11 上傳
如果你原來的串口初始化函數具有打開串口接收中斷的話,實際上就是在初始化函數中多了一條打開空閒中斷的語句。
串口中斷函數
串口中斷函數裏面,最重要的兩條語句,就是上圖中圈出來的兩條語句。第一條語句用來判斷是否接收到1個字節,第二條語句用來判斷是否接收到1幀數據。(是不是感覺超級方便?媽媽再也不用擔心我如何判斷是否接收完1幀數據了。)
主函數
我寫的這個主函數,是用來驗證接收的正確性的。RxCounter表示的是這一幀數據有幾個字節,接收完一幀數據,會在中斷函數裏面把ReceiveState置1,然後,通過串口把接收到的數據發送回串口。這樣,既驗證了接收了多少字節的正確性,又驗證了接收到的數據是否正確。
上圖是結果驗證。
點擊下載源程序: STM32串口接收不定長數據程序.zip (2.25 MB, 下載次數: 112)
我是瑞生,毫無保留的給大家透露電子設計經驗,不定時分享實用的落地的電子設計技巧,希望能夠幫助到大家。
/**
******************************************************************************
* @file 串口接收不定長字節數據
* @author 瑞生
* @version V1.0
* @date 2015.10.23
* @brief Main program body
******************************************************************************
******************************************************************************
*/
/* Includes ------------------------------------------------------------------*/
#include "stm32f10x.h"
#include "uart.h"
volatile uint8_t aRxBuffer[100]={0x00};
volatile uint8_t RxCounter=0;
volatile uint8_t ReceiveState=0;
/**
* @brief Main program.
* @param None
* @retval None
*/
int main(void)
{
uint8_t i=0;
USART1_Init();
while (1)
{
if(ReceiveState==1)//如果接收到1幀數據
{
ReceiveState=0;
i=0;
while(RxCounter--)// 把接收到數據發送回串口
{
USART_SendData(USART1, aRxBuffer[i++]);
while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
}
RxCounter=0;
}
}
}
複製代碼