關(guān)于串口FIFO中斷
串口FIFO中斷有;RDA CTI
串口的接收模塊包括接收緩沖寄存器和移位寄存器。接收的數(shù)據(jù)進(jìn)入移位寄存器后經(jīng)移位處理并行傳入緩沖寄存器,事實(shí)上,UART的FIFO是一個(gè)硬件環(huán)形的緩沖隊(duì)列,物理上不可尋址,不可見(jiàn),僅U0RBR這個(gè)FIFO出口可見(jiàn)。U0RBR就是接收FIFO的第一位。FIFO的長(zhǎng)度是可設(shè)的,也叫觸發(fā)點(diǎn),低于這個(gè)長(zhǎng)度的字符串不會(huì)引起中斷,但在實(shí)際應(yīng)用中,不可能串口讀入的數(shù)據(jù)長(zhǎng)度總為觸發(fā)點(diǎn)值的整數(shù)倍,為此,引入了CTI即字符接收超時(shí)中斷,當(dāng)有不足觸發(fā)點(diǎn)值規(guī)定的字符串讀入時(shí),將引起中斷,其與串口的RDA中斷具有相同的優(yōu)先級(jí),并會(huì)同時(shí)被使能。
那么,LPC2000的UART機(jī)制是如何判斷串口讀入數(shù)據(jù)的一次性容量呢?如果接收FIFO里已經(jīng)有了1個(gè)字符,它可在一定的時(shí)間內(nèi)等待下一個(gè)字符的讀入,也就是說(shuō),不超過(guò)一定的時(shí)間就不會(huì)觸發(fā)CTI中斷,這個(gè)時(shí)間是在本次通訊協(xié)議設(shè)置的前提下,接收3.5到4.5個(gè)字符所用的時(shí)間。比如,需串口接收GPRS數(shù)傳狀態(tài)成功建立后的返回值“Ok_Info_WaitingForData”22個(gè)字符組成的字符串,F(xiàn)IFO觸發(fā)點(diǎn)設(shè)置為14,在前14個(gè)字符讀入之后,立即觸發(fā)RDA中斷(接收中斷),跳轉(zhuǎn)至RDA中斷服務(wù)子程序?qū)⑦@14個(gè)字符放入我所預(yù)設(shè)的緩沖區(qū)內(nèi),接著,后8個(gè)字符讀入,這時(shí)CPU并不立刻中斷,它需等待在本次串口通訊協(xié)議設(shè)置的前提下,接收3.5到4.5個(gè)字符所用的時(shí)間(需根據(jù)波特率和幀格式具體計(jì)算),這個(gè)時(shí)間一到,立刻觸發(fā)CTI中斷(超時(shí)中斷),換句話說(shuō),超過(guò)這個(gè)時(shí)間的等待,CPU就認(rèn)為一個(gè)完整的字符串已經(jīng)結(jié)束了,這才是字符串超時(shí)的真正含義。
CTI觸發(fā)的前提是接收FIFO里已經(jīng)有了1個(gè)字符,在等待下一個(gè)字符的進(jìn)入
用FIFO發(fā)送數(shù)據(jù)時(shí),需要保證接受端的FIFO不溢出,則發(fā)送過(guò)快,接受端FIFO滿后中斷末取走數(shù)據(jù),發(fā)送端就又發(fā)數(shù)據(jù)。其只能由發(fā)送端來(lái)保證
不過(guò)接受端FIFO有2個(gè)Buffer,循環(huán)接收.
3、個(gè)人覺(jué)得,當(dāng)FIFO的觸發(fā)深度設(shè)為1時(shí)和不用FIFO時(shí)的效果是一樣的。
4、串口發(fā)送時(shí),推薦第一個(gè)字符要在打開(kāi)串口中斷前送(不懂???),否則可能中斷只來(lái)一次。比如下面的這個(gè)用串口中斷發(fā)送字符串的函數(shù)為:
/****************************************************************************************************
** 函數(shù)名稱(chēng) :UART0_SendStr()
** 函數(shù)功能 :向串口發(fā)送字符串
** 入口參數(shù) :str 要發(fā)送的字符串的首地址指針
** 出口參數(shù) :無(wú)
*****************************************************************************************************
void UART0_SendStr( CHAR const *str)
{
Str_Send_P = str;
U0THR = *Str_Send_P++;
U0IER |= 0x02; //開(kāi)發(fā)送中斷
}
在中斷發(fā)送程序中的處理為:
SWITCH(IIR & 0x0e)
{
case 0x02: //利用串口發(fā)送中斷,發(fā)送字符串
if((*Str_Send_P)!='')
U0THR = *Str_Send_P++;
else
U0IER &= (~0x02); //關(guān)發(fā)送中斷
break;
這種做法的道理不太明白,但實(shí)驗(yàn)證明,不這么做就只發(fā)一次!~
擴(kuò)展閱讀:?jiǎn)纹瑱C(jī)串口通訊程序
編輯:admin 最后修改時(shí)間:2018-05-19