STM32的USART發(fā)送數(shù)據(jù)時如何使用TXE和TC標(biāo)志
在USART的發(fā)送端有2個寄存器,一個是程序可以看到的USART_DR寄存器,另一個是程序看不到的移位寄存器,對應(yīng)USART數(shù)據(jù)發(fā)送有兩個標(biāo)志,一個是TXE=發(fā)送數(shù)據(jù)寄存器空,另一個是TC=發(fā)送結(jié)束。
當(dāng)USART_DR中的數(shù)據(jù)傳送到移位寄存器后,TXE被設(shè)置,此時移位寄存器開始向TX信號線按位傳輸數(shù)據(jù),但因為TDR已經(jīng)變空,程序可以把下一個要發(fā)送的字節(jié)(操作USART_DR)寫入TDR中,而不必等到移位寄存器中所有位發(fā)送結(jié)束,所有位發(fā)送結(jié)束時(送出停止位后)硬件會設(shè)置TC標(biāo)志。
另一方面,在剛剛初始化好USART還沒有發(fā)送任何數(shù)據(jù)時,也會有TXE標(biāo)志,因為這時發(fā)送數(shù)據(jù)寄存器是空的。TXEIE和TCIE的意義很簡單,TXEIE允許在TXE標(biāo)志為'1'時產(chǎn)生中斷,而TCIE允許在TC標(biāo)志為'1'時產(chǎn)生中斷。
至于什么時候使用哪個標(biāo)志,需要根據(jù)你的需要自己決定。但我認(rèn)為TXE允許程序有更充裕的時間填寫TDR寄存器,保證發(fā)送的數(shù)據(jù)流不間斷。TC可以讓程序知道發(fā)送結(jié)束的確切時間,有利于程序控制外部數(shù)據(jù)流的時序。
?
TXE--寫寄存器DR清零
RXNE--讀寄存器DR清零,也可軟件手動清零
?TC--? 讀/寫寄存器DR清零,也可軟件手動清零
?
先說TC。即Transmission Complete。發(fā)送一個字節(jié)后才進(jìn)入中斷,這里稱為“發(fā)送后中斷”。和原來8051的TI方式一樣,都是發(fā)送后才進(jìn)中斷,需要在發(fā)送函數(shù)中先發(fā)送一個字節(jié)觸發(fā)中斷。發(fā)送函數(shù)如下
/*******
功能:中斷方式發(fā)送字符串.采用判斷TC的方式.即 判斷 發(fā)送后中斷 位.
輸入:字符串的首地址
輸出:無
*******/
void USART_SendDataString( u8 *pData )
{
? ? pDataByte = pData;
??
? ? USART_ClearFlag(USART1, USART_FLAG_TC);//清除傳輸完成標(biāo)志位,否則可能會丟失第1個字節(jié)的數(shù)據(jù).網(wǎng)友提供.
? ?
? ? USART_SendData(USART1, *(pDataByte++) ); //必須要++,不然會把第一個字符t發(fā)送兩次
}
中斷處理函數(shù)如下
/********
* Function Name??: USART1_IRQHandler
* Description? ? : This function handles USART1 global interrupt request.
* Input? ?? ?? ? : None
* Output? ?? ?? ?: None
* Return? ?? ?? ?: None
*********/
void USART1_IRQHandler(void)
{
? ? if( USART_GetITStatus(USART1, USART_IT_TC) == SET??)
? ? {
? ?? ???if( *pDataByte == '