新聞中心

EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > stm32的USART_IT_TXE和USART_IT_TC

stm32的USART_IT_TXE和USART_IT_TC

作者: 時(shí)間:2016-12-03 來(lái)源:網(wǎng)絡(luò) 收藏
一般來(lái)說(shuō),串口的發(fā)送中斷傳輸數(shù)據(jù),對(duì)于我目前的應(yīng)用來(lái)說(shuō),應(yīng)用的要求并不是很高,因此,因此就一直沒(méi)有對(duì)其進(jìn)行一個(gè)比較好的實(shí)驗(yàn)與認(rèn)識(shí)。然而,在一次串口程序升級(jí)(IAP)升級(jí)實(shí)驗(yàn)中,發(fā)現(xiàn)有人使用了這個(gè)發(fā)送中斷方式進(jìn)行的,所以特別的進(jìn)行了一個(gè)個(gè)步驟的實(shí)驗(yàn)來(lái)進(jìn)一步知道其運(yùn)行機(jī)理。

首先,串口發(fā)送數(shù)據(jù),可以使用的方式有:

本文引用地址:http://butianyuan.cn/article/201612/325196.htm

1、發(fā)送一個(gè)數(shù)據(jù),然后讀取USART_IT_TXE或者USART_IT_TC寄存器的狀態(tài)。

2、使用串口發(fā)送中斷

3、使用dma發(fā)送完成中斷

在效率上,肯定是3比較好。其次是2。

網(wǎng)上關(guān)于USART_IT_TXE和USART_IT_TC是怎么個(gè)用法,各家有各家的言論,在我這里只通過(guò)實(shí)驗(yàn)來(lái)了解,因?yàn)榫W(wǎng)上的很多人的講解是有一些歧義的。

本人用的芯片為stm32f103vet6,利用串口1進(jìn)行實(shí)驗(yàn),先貼上串口1的配置部分:

void USART1_Config(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

USART_InitTypeDef USART_InitStructure;

NVIC_InitTypeDef NVIC_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init(GPIOA, &GPIO_InitStructure);

/* Configure USART1 Rx (PA.10) as input floating */

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

GPIO_Init(GPIOA, &GPIO_InitStructure);

USART_InitStructure.USART_BaudRate = 115200;

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_Rx | USART_Mode_Tx;

USART_Init(USART1, &USART_InitStructure);

USART_Cmd(USART1, ENABLE);

NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

NVIC_Init( &NVIC_InitStructure );

USART_ITConfig(USART1, USART_IT_TXE, ENABLE);

USART_ITConfig(USART1, USART_IT_TC, ENABLE);

}

然后是中斷部分的實(shí)驗(yàn)代碼:

void USART1_IRQHandler(void)

{

if(SET == USART_GetITStatus(USART1, USART_IT_TXE))

{

USART_ClearITPendingBit(USART1, USART_IT_TXE);

if(有數(shù)據(jù))

{

}

else

{

USART_ITConfig(USART1, USART_IT_TXE, DISABLE);

USART_ITConfig(USART1, USART_IT_TC, ENABLE);

}

}

else if (USART_GetITStatus(USART1, USART_IT_TC) != RESET)

{

USART_ClearITPendingBit(USART1, USART_IT_TC);

USART_ITConfig(USART1, USART_IT_TC, DISABLE);

}

}

現(xiàn)在分析,在使能了串口1的功能后,明顯的是,串口就開(kāi)始進(jìn)入發(fā)送寄存器空狀態(tài),也就是說(shuō)if(SET == USART_GetITStatus(USART1, USART_IT_TXE))是成立的,如果沒(méi)有if判斷語(yǔ)句的話,也就是沒(méi)有if{}else{},那么就會(huì)一直在if(SET == USART_GetITStatus(USART1, USART_IT_TXE))中斷中,因此對(duì)于if(有數(shù)據(jù))就需要在if(有數(shù)據(jù)){

USART_SendData(USART1,數(shù)據(jù));}了,一直等到?jīng)]有數(shù)據(jù)了,就進(jìn)入關(guān)閉USART_GetITStatus(USART1, USART_IT_TC) != RESET與之同時(shí),需要開(kāi)啟USART_ITConfig(USART1, USART_IT_TC, ENABLE);讓其進(jìn)入傳輸完成中斷。有人說(shuō),這個(gè)中斷是發(fā)送一個(gè)字節(jié)完成后中斷,這種理解十分的讓人有歧義,應(yīng)該說(shuō),發(fā)送一個(gè)字節(jié)數(shù)據(jù)之后,如果后續(xù)還有數(shù)據(jù),這個(gè)中斷是不會(huì)進(jìn)去的,這個(gè)USART_GetITStatus(USART1, USART_IT_TC) != RESET;串口空閑相關(guān)。應(yīng)該稍微準(zhǔn)確的說(shuō)法是,發(fā)送一個(gè)字節(jié)數(shù)據(jù)完成后,串口處于空閑狀態(tài)了,這個(gè)時(shí)候,會(huì)進(jìn)入這個(gè)中斷。

好了,如果把串口初始化完成了,這個(gè)時(shí)候,串口發(fā)送中斷都被禁止了,怎么去發(fā)送數(shù)據(jù)呢?

發(fā)送數(shù)據(jù),直接把USART_ITConfig(USART1, USART_IT_TXE, ENABLE);顯然是不夠的,需要通過(guò)USART_SendData(USART1,數(shù)據(jù));然后開(kāi)啟中斷??梢詴呵铱醋魇羌せ畲冢沁^(guò)程應(yīng)該是數(shù)據(jù)到達(dá)寄存器后,很快為空,進(jìn)而產(chǎn)生if(SET == USART_GetITStatus(USART1, USART_IT_TXE))為真的條件,從而一切仍然能夠正常運(yùn)行了。

注意點(diǎn):

1、中斷流程

2、再次發(fā)送數(shù)據(jù),操作步驟。

綜上,一個(gè)串口中斷發(fā)送數(shù)據(jù)的程序就可以明顯展開(kāi)了。



關(guān)鍵詞: stm32USART_IT_TXEUSART_IT_T

評(píng)論


技術(shù)專(zhuān)區(qū)

關(guān)閉