娇小w搡bbbb搡bbb,《第一次の人妻》,中国成熟妇女毛茸茸,边啃奶头边躁狠狠躁视频免费观看

【GD32H757Z海棠派開發板使用手冊】第十講 USART-中斷串口收發實驗

發布者:LovingLife2023最新更新時間:2024-12-13 來源: elecfans關鍵字:USART  中斷  串口收發 手機看文章 掃描二維碼
隨時隨地手機看文章

10.1實驗內容

通過本實驗主要學習以下內容:

  • 使用中斷進行串口收發

10.2實驗原理

10.2.1串口寄存器介紹

串口有幾個非常重要的寄存器需要讀者理解。

數據接收寄存器(USART_RDATA)

wKgaomZFbTOAXPmzAAAwMAxUG-I889.png

數據發送寄存器(USART_RDATA)

wKgZomZFbUCAHOXZAAAvvuPzh50222.png

發送時,除了發送數據寄存器,還有一個移位寄存器,當數據寫入數據寄存器中,移位寄存器空閑的情況下,數據從數據寄存器中轉移到移位寄存器,移位寄存器按照低bit——高bit的順序將數據移位到IO口上。

接收時,接收到的數據保存在數據接收寄存器中,CPU或DMA可以從該寄存器中讀接收到的數據。

狀態寄存器(USART_STAT )

wKgZomZFbUuAFDIOAABSvI18o7s614.png

我們需要特別理解TBE、TC、RBNE、IDLE、OREE這幾位。

  1. TBE(發送空):這個位置“1”表示現在可以往數據寄存器中寫數據了,當移位寄存器空閑時,寫入到數據寄存器中的數據則會轉移到移位寄存器中,串口開始對外發送數據;

  2. TC(發送完成):發送數據時,當數據寄存器和移位寄存器都為空時,表示所有的數據都已經完成了,則TC置“1”,所以當連續發數據時,最后一個字節從移位寄存器中發送完,TC才會置起。

  3. RBNE(接受非空):當串口接受到一個字節數據,RBNE置“1”,此時CPU可以去數據寄存器中取數據,當使用了DMA接受,DMA自動將數據寄存器中數據搬走,當數據寄存器數據被讀走/搬走,RBNE位自動清“0”;

  4. IDLE(空閑):該標志位用于檢測接受空閑,當串口接受最后一個字節后,再往后一個字節時間內,沒有接受到新的數據,則該位置“1”;

IDLE一般用于串口DMA接受中,DMA接受中,MCU無法知道發送方的數據個數,所以可以通過判斷IDLE位(或IDLE中斷)來判斷發送方一幀數據發送結束了。

5. OREE(溢出錯誤):當RBNE置位的情況,又接收到一個字節數據,則OREE位置“1”。

以上就是串口寄存器的介紹。本實驗就是使用TBE中斷和RBNE中斷來實現中斷收發數據,實驗原理是RBNE中斷用來接受數據,IDLE中斷用于判斷發送方數據結束,TBE中斷用于發送數據。


10.3硬件設計


本實驗使用P1接口的PA9和PA10實現串口功能,硬件設計請見上一章。


10.4代碼解析


10.4.1串口中斷發送函數


在driver_uart.c中定義了串口中斷發送函數:


C

Drv_Err driver_uart_int_transmit(typdef_uart_struct *uartx,uint8_t *pbuff,uint16_t length)

{

__IO uint64_t timeout = driver_tick;

while(uartx->uart_control.Com_Flag.Bits.SendState==1){

if((timeout+UART_TIMEOUT_MS) <= driver_tick) {              

uartx->uart_control.Com_Flag.Bits.SendState=0;

return DRV_ERROR;

}

}


uartx->uart_control.Com_Flag.Bits.SendSuccess=0;

uartx->uart_control.Com_Flag.Bits.SendState=1;

uartx->uart_control.p_Send=pbuff;

uartx->uart_control.SendSize=length;

uartx->uart_control.SendCount=0;


usart_flag_clear(uartx->uart_x,USART_FLAG_TC);

usart_interrupt_enable(uartx->uart_x,USART_INT_TBE);


return DRV_SUCCESS;

}

10.4.2串口中斷接受函數


在driver_uart.c中定義了串口中斷接受函數:


C

Drv_Err driver_uart_int_receive(typdef_uart_struct *uartx,uint8_t *pbuff,uint16_t length)

{

__IO uint64_t timeout = driver_tick;

while(uartx->uart_control.Com_Flag.Bits.RecState==1){

if((timeout+UART_TIMEOUT_MS) <= driver_tick) {              

uartx->uart_control.Com_Flag.Bits.RecState=0;

return DRV_ERROR;

}

}


if(usart_flag_get(uartx->uart_x,USART_FLAG_ORERR))

{

usart_flag_clear(uartx->uart_x,USART_FLAG_ORERR);

}


uartx->uart_control.Com_Flag.Bits.RecSuccess=0;

uartx->uart_control.Com_Flag.Bits.RecState=1;

uartx->uart_control.p_Rec=pbuff;

uartx->uart_control.RecSize=length;

uartx->uart_control.RecCount=0;


usart_flag_clear(uartx->uart_x,USART_FLAG_IDLE);


usart_interrupt_enable(uartx->uart_x,USART_INT_RBNE);

usart_interrupt_enable(uartx->uart_x,USART_INT_IDLE);


return DRV_SUCCESS;

}

10.4.3main函數實現


以下為main函數代碼:


C

int main(void)

{

//延時、共用驅動部分初始化

driver_init();

//初始化UART為中斷模式,注冊接受完成(IDLE)回調函數

BOARD_UART.uart_mode_tx=MODE_INT;

BOARD_UART.uart_mode_rx=MODE_INT;

BOARD_UART.uart_idle_callback=user_receive_complete_callback;

bsp_uart_init(&BOARD_UART);


bsp_led_init(&LED2);

bsp_led_init(&LED1);

bsp_led_on(&LED2);

bsp_led_off(&LED1);

//使能UART中斷

nvic_irq_enable(USART0_IRQn,2,0);

delay_ms(100);

printf_log('uart interrupt mode sends and receives loopback packets of indefinite length.rn');

//啟動UART中斷接受,最長100byte

driver_uart_int_receive(&BOARD_UART,uart_rec_buff,100);


while (1)

{

//查詢到接受完成回調函數標志

if(uart_receive_complete_flag==SET)

{

uart_receive_complete_flag=RESET;

//啟動中斷方式發送剛接受到的數據

driver_uart_int_transmit(&BOARD_UART,uart_send_buff,uart_receive_count);

printf_log('rn The received data is %srn',uart_send_buff);

memset(uart_send_buff,0,100);

}

}

}

本例程main函數首先進行了延時函數初始化,再初始化UART為中斷模式,接著配置串口BOARD_UART,開啟串口中斷NVIC,這里使用到了IDLE中斷,TBE中斷和RBNE中斷,然后配置串口D中斷接受,最長100個字節,所以我們可以給串口發送100個字節以下長度的數據。在while(1)循環中循環查詢uart_receive_complete_flag標志位,當該標志位為“SET”時,表示IDLE中斷被觸發,一幀數據接受完,最后將接收到的幀數據通過中斷發送方式原封不動發送到串口上。


10.4.4中斷函數


在bsp_uart.c中定義了串口中斷處理函數


C

void USART0_IRQHandler(void)

{

driver_uart_int_handler(&BOARD_UART);

}

在driver_uart.c中定義了driver_uart_int_handler函數:


C

Drv_Err driver_uart_int_handler(typdef_uart_struct *uartx)

{

Drv_Err uart_state=DRV_SUCCESS;

if(usart_interrupt_flag_get(uartx->uart_x,USART_INT_FLAG_RBNE)!=RESET)

{

if(uartx->uart_control.RecCount < uartx->uart_control.RecSize){

uartx->uart_control.p_Rec[uartx->uart_control.RecCount]=usart_data_receive(uartx->uart_x);

uartx->uart_control.RecCount++;

}

else{

usart_data_receive(uartx->uart_x);

uart_state=DRV_ERROR;

//err 溢出

}

if(uartx->uart_rbne_callback!=NULL){

uartx->uart_rbne_callback(uartx);

}

//callback

if(uartx->uart_control.RecCount == uartx->uart_control.RecSize){

uartx->uart_control.Com_Flag.Bits.RecSuccess=1;

uartx->uart_control.Com_Flag.Bits.RecState=0;

uartx->uart_control.RecCount=0;

}

}

if(usart_interrupt_flag_get(uartx->uart_x,USART_INT_FLAG_IDLE)!=RESET)

{

usart_interrupt_flag_clear(uartx->uart_x,USART_INT_FLAG_IDLE);

if( (uartx->uart_mode_rx==MODE_INT && uartx->uart_control.RecCount>0)

||(uartx->uart_mode_rx==MODE_DMA && dma_transfer_number_get(uartx->uart_rx_dma->dmax,uartx->uart_rx_dma->dma_chx)!=uartx->uart_control.RecSize))

{

uartx->uart_control.Com_Flag.Bits.RecSuccess=1;

uartx->uart_control.Com_Flag.Bits.RecState=0;


if(uartx->uart_mode_rx==MODE_DMA){

uartx->uart_control.RecCount=uartx->uart_control.RecSize-dma_transfer_number_get(uartx->uart_rx_dma->dmax,uartx->uart_rx_dma->dma_chx);

}

//callback

if(uartx->uart_idle_callback!=NULL){

uartx->uart_idle_callback(uartx);

}

}


}


if(usart_interrupt_flag_get(uartx->uart_x,USART_INT_FLAG_TBE)!=RESET)

{

usart_data_transmit(uartx->uart_x,uartx->uart_control.p_Send[uartx->uart_control.SendCount]);

uartx->uart_control.SendCount++;


if(uartx->uart_tbe_callback!=NULL){

uartx->uart_tbe_callback(uartx);

}


if(uartx->uart_control.SendCount >= uartx->uart_control.SendSize)

{

uartx->uart_control.SendCount=0;

usart_interrupt_disable(uartx->uart_x, USART_INT_TBE);

usart_interrupt_enable(uartx->uart_x, USART_INT_TC);

}

}


if(usart_interrupt_flag_get(uartx->uart_x,USART_INT_FLAG_TC)!=RESET)

{

usart_interrupt_disable(uartx->uart_x, USART_INT_TC);

usart_flag_clear(uartx->uart_x,USART_FLAG_TC);


if( !(uartx->uart_mode_rx==MODE_DMA && dma_transfer_number_get(uartx->uart_tx_dma->dmax,uartx->uart_tx_dma->dma_chx)!=0) )

{

uartx->uart_control.Com_Flag.Bits.SendSuccess=1;

uartx->uart_control.Com_Flag.Bits.SendState=0;


if(uartx->uart_tc_callback!=NULL){

uartx->uart_tc_callback(uartx);

}


uartx->uart_control.SendCount=0;

}

}


if(usart_flag_get(uartx->uart_x,USART_FLAG_ORERR)==SET)

{

usart_flag_clear(uartx->uart_x,USART_FLAG_ORERR);

uart_state=DRV_ERROR;

}


return uart_state;


}

10.5實驗結果


使用串口調試助手發送一幀數據到MCU,MCU會將這幀數據回發到串口調試助手中。


wKgZomZCzhqAd1-uAAAEQNN_QRI502.pngwKgZomZCziKAWcJsAAAIXPeWoWw536.png

關鍵字:USART  中斷  串口收發 引用地址:【GD32H757Z海棠派開發板使用手冊】第十講 USART-中斷串口收發實驗

上一篇:【GD32F470紫藤派開發板使用手冊】第七講 FWDG-看門狗實驗
下一篇:【GD32F470紫藤派開發板使用手冊】第六講 TIMER_3路PWM輸出實驗

推薦閱讀最新更新時間:2025-04-23 21:49

STM32 串口例程之查詢收發
有了STM32,使用串口簡直就是玩游戲的感覺。這里鄙人就不談STM32串口的happy了,直接上代碼,讀者從代碼中體會樂趣。發送接受均采用查詢方式,串口調試工具使用超級終端或者這貨--SecureCRT 5.5(傳說比超級終端還超級),在中端里面輸入什么,同步接受與發送,感覺就像是在文本框里面打字。 還是甩一張工程結構圖正面: 代碼鄙人偷懶,所有代碼在main.c里面一鍋煮:
[單片機]
基于STM32F103ZE的USART2 端口時鐘
USART1程序很多,看看USART2程序。注意紅色部分,首先要使能相關端口的時鐘。 因為USART2可以映射成不同的端口,所以要端口映射。 結合你自己的開發板,設置正確的映射口.(正點原子的是PA03PA02.) STM32F103ZE默認的USART2就是PA03PA02,如果配置映射,USART2端口是PD05PD06了。 程序在正點原子開發板SRAM中調試通過,開發環境IAR5.30. void USART2_Initialise( u32 bound ) { GPIO_InitTypeDef GPIO_InitStructure;//I/O?ú3?ê??ˉ?á11ì? NVIC_InitTypeDef NVIC_
[單片機]
基于STM32F103ZE的<font color='red'>USART</font>2 端口時鐘
STM32CubeMX學習教程之六:USART串口輸出和printf( )重定向到串口
軟件: STM32CubeMX V4.25.0 System Workbench V2.4 固件庫版本: STM32Cube FW_F1 V1.6.1 硬件: OneNet 麒麟座V2.3 在STM32CubeMX中新建項目,選擇正確的MCU型號 首先設置RCC和SYS,如下圖 然后根據板子實際情況設置時鐘(麒麟座外部晶振是12M,STM32F103x的最高主頻是72M),如下圖 配置USART1為Asychronous模式 可以看到PA9和PA10被配置為了USART1_TX, USART1_RX。 UART1配置用默認的115200,8, None和1就好。 Project -
[單片機]
STM32CubeMX學習教程之六:<font color='red'>USART</font><font color='red'>串口</font>輸出和printf( )重定向到<font color='red'>串口</font>
AVR單片機USART串口通信基本應用
/********************************************* File name : usart.c Chip type : ATmega16 Program type : Application Clock frequency : 11.0592MHz Memory model : Small 注意:發送采用查詢方式,接收采用中斷方式,在初始化程序中,需要將TXD設置成輸出方式,RXD設置成輸入方式,并使RXD的內部上拉有效 *********************************************/ #include mega16.
[單片機]
STM32---DMA(USART)的演示
 這里有個小小的例子,來演示DMA模塊與系統程序并行工作。   用串口以低波特率發送一個10K的數據,花費近10s時間,此時按照以往方法,CPU要不斷等待數據發送、送數據;或者送數據、進中斷、送數據,處理起來比較消耗時間。   使用了DMA功能以后,用戶程序中只需配置好DMA,開啟傳輸后,再也不需要操心,10K數據完成后會有標志位或中斷產生,期間可以做任何想做的事,非常方便。   這個是相應的代碼例子,基于STM32F103VBT6 /****************************************************************************** * 本文件實現串口發送功能(通過重構p
[單片機]
S3C2440 中斷
  CPU和外設構成了計算機系統,CPU和外設之間通過總線進行連接,用于數據通信和控制,CPU管理監視計算機系統中所有硬件,通常以兩種方式來對硬件進行管理監視:   查詢方式:CPU不停的去查詢每一個硬件的當前狀態,根據硬件的狀態決定處理與否。好比是工廠里的檢查員,不停的檢查各個崗位工作狀態,發現情況及時處理。這種方式實現起來簡單,通常用在只有少量外設硬件的系統中,如果一個計算機系統中有很多硬件,這種方式無疑是耗時,低效的,同時還大量占用CPU資源,并且對多任務系統反應遲鈍。   中斷方式:當某個硬件產生需要CPU處理的事件時,主動通過一根信號線“告知”CPU,同時設置某個寄存器里對應的位,CPU一旦發現這根信號線上的電平有變化,
[單片機]
S3C2440 <font color='red'>中斷</font>
在51單片機中,中斷函數與一般函數的調用區別?
中斷函數的調用過程類似于一般函數調用,區別在于何時調用一般函數在程序中是事先安排好的;而何時調用中斷函數事先卻無法確定,因為中斷的發生是由外部因素決定的,程序中無法事先安排調用語句。因此,調用中斷函數的過程是由硬件自動完成的。 參考資料: (1)王靜霞等. 單片機應用計數(C語言版)第3版 . 北京,電子工業出版社,2015:166.
[單片機]
s3c2440裸機-異常中斷5-irq定時器中斷
之前講過s3c2440時鐘體系,看了時鐘體系再來看定時器中斷會更好的結合運用所學知識點。 S3c2440共有2種定時器: 1.Watchdog看門狗定時器 2.PWM脈沖可調制定時器 下面詳細介紹2種定時器的原理,來了解定時器是如何產生定時器中斷的。 1. Watchdog看門狗定時器 1)Watchdog看門狗定時器原理 Watchdog定時器的原理很簡單,寄存器很少,框圖如下: 1.定時器,定時器那肯定是需要用到時鐘的,從框圖中可以看到Watchdog定時器采用的時鐘源是PCLK,從s3c2440時鐘體系中也可以體現出來,接的是APB總線。 2.然后到達一個8bit的分頻器,可以通過配置WTCON 來設置分頻器的預設
[單片機]
小廣播
設計資源 培訓 開發板 精華推薦

最新單片機文章
何立民專欄 單片機及嵌入式寶典

北京航空航天大學教授,20余年來致力于單片機與嵌入式系統推廣工作。

隨便看看

 
EEWorld訂閱號

 
EEWorld服務號

 
汽車開發圈

 
機器人開發圈

電子工程世界版權所有 京ICP證060456號 京ICP備10001474號-1 電信業務審批[2006]字第258號函 京公網安備 11010802033920號 Copyright ? 2005-2025 EEWORLD.com.cn, Inc. All rights reserved
主站蜘蛛池模板: 凌海市| 涿州市| 鹤壁市| 手游| 抚宁县| 鹤庆县| 龙门县| 房山区| 墨玉县| 湄潭县| 北宁市| 平乡县| 商河县| 方正县| 盘山县| 乐陵市| 南投市| 张家口市| 横峰县| 武功县| 百色市| 湟源县| 津南区| 玉门市| 沭阳县| 泉州市| 牟定县| 综艺| 张家口市| 科技| 无锡市| 大渡口区| 红安县| 浑源县| 钟祥市| 长治县| 澄城县| 桃江县| 句容市| 永济市| 徐水县|