RM新时代网站-首页

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

【GD32F303紅楓派開發(fā)板使用手冊】第十六講 USART-DMA串口收發(fā)實(shí)驗(yàn)

聚沃科技 ? 2024-06-15 09:54 ? 次閱讀
wKgaomZVdiiAfR9BAB3mDFhHnZc972.png

16.1實(shí)驗(yàn)內(nèi)容

通過本實(shí)驗(yàn)主要學(xué)習(xí)以下內(nèi)容:

16.2實(shí)驗(yàn)原理

16.2.1串口DMA工作原理

在前面ADC章節(jié)中,我們介紹了DMA的工作原理,這里就不多做介紹。從GD32F303用戶手冊中可以查到,各串口的TX和RX分別對應(yīng)DMA的不同通道,比如USART0的TX對應(yīng)DMA0的通道3,而RX對應(yīng)DMA0的通道4。

當(dāng)需要使用DMA發(fā)送時,需要配置DMA工作為內(nèi)存到外設(shè)的模式,DMA目標(biāo)地址需要設(shè)置為串口的數(shù)據(jù)寄存器,當(dāng)DMA使能后,一旦串口的TBE(發(fā)送空)標(biāo)志位為1,則DMA自動從內(nèi)存中搬運(yùn)數(shù)據(jù)到串口數(shù)據(jù)寄存器中。

當(dāng)需要使用DMA接受時,需要配置DMA工作為外設(shè)到內(nèi)存的模式,DMA的源地址需要設(shè)置為串口的數(shù)據(jù)寄存器,當(dāng)DMA使能,一旦串口收到一個字節(jié)數(shù)據(jù),RBNE(接受非空)標(biāo)志位為1,則DMA自動將數(shù)據(jù)寄存器中的數(shù)據(jù)搬運(yùn)到內(nèi)存中。

16.2.2串口寄存器介紹

串口有幾個非常重要的寄存器需要讀者理解,這里單獨(dú)用一個章節(jié)來介紹。

數(shù)據(jù)寄存器(USART_DATA)

wKgZomZCzTOAdz_OAAAuTeg2cA0146.png

該寄存器雖然只有一個,但內(nèi)部是映射為發(fā)送和接受兩個寄存器。

發(fā)送時,除了發(fā)送數(shù)據(jù)寄存器,還有一個移位寄存器,當(dāng)數(shù)據(jù)寫入數(shù)據(jù)寄存器中,移位寄存器空閑的情況下,數(shù)據(jù)從數(shù)據(jù)寄存器中轉(zhuǎn)移到移位寄存器,移位寄存器按照低bit——高bit的順序?qū)?shù)據(jù)移位到IO口上。

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

狀態(tài)寄存器0(USART_STAT0 )

wKgaomZCzUOAbIljAAA9A6YgC4w762.png

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

  1. TBE(發(fā)送空):這個位置“1”表示現(xiàn)在可以往數(shù)據(jù)寄存器中寫數(shù)據(jù)了,當(dāng)移位寄存器空閑時,寫入到數(shù)據(jù)寄存器中的數(shù)據(jù)則會轉(zhuǎn)移到移位寄存器中,串口開始對外發(fā)送數(shù)據(jù);
  2. TC(發(fā)送完成):發(fā)送數(shù)據(jù)時,當(dāng)數(shù)據(jù)寄存器和移位寄存器都為空時,表示所有的數(shù)據(jù)都已經(jīng)完成了,則TC置“1”,所以當(dāng)連續(xù)發(fā)數(shù)據(jù)時,最后一個字節(jié)從移位寄存器中發(fā)送完,TC才會置起。
  3. RBNE(接受非空):當(dāng)串口接受到一個字節(jié)數(shù)據(jù),RBNE置“1”,此時CPU可以去數(shù)據(jù)寄存器中取數(shù)據(jù),當(dāng)使用了DMA接受,DMA自動將數(shù)據(jù)寄存器中數(shù)據(jù)搬走,當(dāng)數(shù)據(jù)寄存器數(shù)據(jù)被讀走/搬走,RBNE位自動清“0”;
  4. IDLE(空閑):該標(biāo)志位用于檢測接受空閑,當(dāng)串口接受最后一個字節(jié)后,再往后一個字節(jié)時間內(nèi),沒有接受到新的數(shù)據(jù),則該位置“1”;

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

  1. OREE(溢出錯誤):當(dāng)RBNE置位的情況,又接收到一個字節(jié)數(shù)據(jù),則OREE位置“1”。

16.3硬件設(shè)計

本實(shí)驗(yàn)使用DMA進(jìn)行串口發(fā)送和接收,仍然使用USB轉(zhuǎn)UART接口,硬件設(shè)計見上一章。

16.4代碼解析

16.4.1串口DMA發(fā)送函數(shù)

在driver_uart.c中定義了串口DMA發(fā)送函數(shù)driver_uart_dma_transmit:

C
Drv_Err driver_uart_dma_transmit(typdef_uart_struct *uartx,uint8_t *pbuff,uint16_t length)
{
Drv_Err uart_state=DRV_ERROR;

uint32_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.SendSucess=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;
uart_state=driver_dma_flag_wait_timeout(uartx->uart_tx_dma,DMA_FLAG_FTF,SET);
usart_dma_transmit_config(uartx->uart_x,USART_DENT_DISABLE);
driver_dma_start(uartx->uart_tx_dma,pbuff,length);
usart_flag_clear(uartx->uart_x,USART_FLAG_TC);
usart_dma_transmit_config(uartx->uart_x,USART_DENT_ENABLE);
usart_interrupt_enable(uartx->uart_x,USART_INT_TC);
return uart_state;
}

16.4.2串口DMA接收函數(shù)

在driver_uart.c中定義了串口DMA接收函數(shù)driver_uart_dma_receive:

C
Drv_Err driver_uart_dma_receive(typdef_uart_struct *uartx,uint8_t *pbuff,uint16_t length)
{
Drv_Err uart_state=DRV_SUCCESS;
uint32_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;
}
}
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_dma_receive_config(uartx->uart_x,USART_DENR_DISABLE);
driver_dma_start(uartx->uart_rx_dma,pbuff,length);
USART_STAT0(uartx->uart_x);
usart_data_receive(uartx->uart_x);
usart_interrupt_flag_clear(uartx->uart_x,USART_INT_FLAG_IDLE);
usart_interrupt_enable(uartx->uart_x,USART_INT_IDLE);
usart_dma_receive_config(uartx->uart_x,USART_DENR_ENABLE);
return uart_state;
}

16.4.3main函數(shù)實(shí)現(xiàn)

以下為main函數(shù)代碼:

C
int main(void)
{
delay_init();

//初始化UART為DMA模式,注冊接受完成(IDLE)回調(diào)函數(shù)
BOARD_UART.uart_mode_tx=MODE_DMA;
BOARD_UART.uart_mode_rx=MODE_DMA;
BOARD_UART.uart_idle_callback=user_receive_complete_callback;
bsp_uart_init(&BOARD_UART);
nvic_irq_enable(USART0_IRQn,2,0);
delay_ms(1000);
printf("uart dma mode sends and receives loopback packets of indefinite length.\r\n");

//配置UART接受,最長100byte
driver_uart_dma_receive(&BOARD_UART,uart_rec_buff,100);

while (1)
{
//查詢到接受完成回調(diào)函數(shù)標(biāo)志
if(uart_receive_complete_flag==SET)
{
uart_receive_complete_flag=RESET;

//發(fā)送剛接受到的數(shù)據(jù)
driver_uart_dma_transmit(&BOARD_UART,uart_send_buff,uart_receive_count);
}
}
}

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

16.4.4中斷函數(shù)

在bsp_uart.c中定義了串口中斷處理函數(shù)

C
void USART0_IRQHandler(void)
{
driver_uart_int_handler(&BOARD_UART);
}

在driver_uart.c中定義了driver_uart_int_handler函數(shù):

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);
USART_STAT0(uartx->uart_x);
USART_DATA(uartx->uart_x);

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.SendSucess=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);
USART_STAT0(uartx->uart_x);
USART_DATA(uartx->uart_x);
uart_state=DRV_ERROR;
}

return uart_state;

}

16.5實(shí)驗(yàn)結(jié)果

使用USB-TypeC線,連接電腦和板上USB to UART口后,使用串口調(diào)試助手發(fā)送一幀數(shù)據(jù)到MCU,MCU會將這幀數(shù)據(jù)回發(fā)到串口調(diào)試助手中。

wKgZomZs8_GAHWEFAAAEB2USNVY682.pngwKgZomZs8_aAZSKSAAAIwyq5TP0056.png

教程GD32 MCU方案商聚沃科技原創(chuàng)發(fā)布,了解更多GD32 MCU教程,關(guān)注聚沃科技官網(wǎng)

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 單片機(jī)
    +關(guān)注

    關(guān)注

    6035

    文章

    44554

    瀏覽量

    634594
  • 嵌入式
    +關(guān)注

    關(guān)注

    5082

    文章

    19104

    瀏覽量

    304784
  • dma
    dma
    +關(guān)注

    關(guān)注

    3

    文章

    560

    瀏覽量

    100544
  • 開發(fā)板
    +關(guān)注

    關(guān)注

    25

    文章

    5032

    瀏覽量

    97371
  • USART
    +關(guān)注

    關(guān)注

    1

    文章

    195

    瀏覽量

    30834
收藏 人收藏

    評論

    相關(guān)推薦

    STM32CUBEMX開發(fā)GD32F303(8)----USART收發(fā)配置

    本章STM32CUBEMX配置STM32F103,并且在GD32F303中進(jìn)行開發(fā),同時通過GD32303C_START開發(fā)板內(nèi)進(jìn)行驗(yàn)證。
    的頭像 發(fā)表于 11-29 10:40 ?3169次閱讀
    STM32CUBEMX<b class='flag-5'>開發(fā)</b><b class='flag-5'>GD32F303</b>(8)----<b class='flag-5'>USART</b><b class='flag-5'>收發(fā)</b>配置

    STM32CUBEMX開發(fā)GD32F303(9)----USART通過DMA收發(fā)

    概述 本章STM32CUBEMX配置STM32F103,并且在GD32F303中進(jìn)行開發(fā),同時通過GD32303C_START開發(fā)板內(nèi)進(jìn)行驗(yàn)
    的頭像 發(fā)表于 11-29 11:15 ?2101次閱讀
    STM32CUBEMX<b class='flag-5'>開發(fā)</b><b class='flag-5'>GD32F303</b>(9)----<b class='flag-5'>USART</b>通過<b class='flag-5'>DMA</b><b class='flag-5'>收發(fā)</b>

    GD32F303】星空介紹

    一、開發(fā)板介紹星空GD開發(fā)板是由旗點(diǎn)科技推出的一款GD32開發(fā)板,板載
    發(fā)表于 09-11 17:55

    【星空GD32F303開發(fā)板試用體驗(yàn)】開箱+環(huán)境搭建

    Watchpoint: 4Load "G:\\星空GD32F303開發(fā)板\\03 例程\\Project\\04_USART_Printf\\MDK-ARM\\output
    發(fā)表于 10-18 14:15

    【星空GD32F303開發(fā)板試用體驗(yàn)】開箱+環(huán)境搭建

    完全,APM32F0xx_SDK解包,到星空GD32F303開發(fā)板\03 例程\Project\04_USART_printf\MDK-A
    發(fā)表于 11-02 15:36

    【星空GD32F303開發(fā)板試用體驗(yàn)】+板卡概覽

    本帖最后由 cooldog123pp 于 2021-11-6 21:07 編輯 星空GD開發(fā)板是由旗點(diǎn)科技推出的一款GD32開發(fā)板
    發(fā)表于 11-06 21:05

    星空GD32F303開發(fā)板的相關(guān)資料下載

    一、開發(fā)板介紹星空GD開發(fā)板是由旗點(diǎn)科技推出的一款GD32開發(fā)板,板載
    發(fā)表于 12-10 08:27

    GD32F303開發(fā)板介紹

    目錄如下,持續(xù)更新~~【1】星空GD32F303開發(fā)板介紹 與 文章目錄1. 串口基礎(chǔ)概念USART數(shù)據(jù)格式一般分為啟動位、數(shù)據(jù)幀、可能的
    發(fā)表于 01-17 08:06

    STM32CUBEMX開發(fā)GD32F303(11)----ADC在DMA模式下掃描多個通道

    本章STM32CUBEMX配置STM32F103,并且在GD32F303中進(jìn)行開發(fā),同時通過GD32303C_START開發(fā)板內(nèi)進(jìn)行驗(yàn)證。
    的頭像 發(fā)表于 11-30 10:59 ?2339次閱讀
    STM32CUBEMX<b class='flag-5'>開發(fā)</b><b class='flag-5'>GD32F303</b>(11)----ADC在<b class='flag-5'>DMA</b>模式下掃描多個通道

    GD32F303固件庫開發(fā)

    的可以加群申請:615061293 。 GD32F303固件庫開發(fā)(1)----前期準(zhǔn)備與燒錄 使用GDLINK、jlink、串口下載程序到GD芯片。 [https://blog.cs
    的頭像 發(fā)表于 07-27 09:27 ?1162次閱讀
    <b class='flag-5'>GD32F303</b>固件庫<b class='flag-5'>開發(fā)</b>

    GD32F303紅楓開發(fā)板使用手冊】第二 GPIO-流水燈實(shí)驗(yàn)

    GD32F303系列MCU最多可支持?112?個通用I/O?引腳(GPIO),分別為?PA0 ~ PA15,?PB0 ~ PB15,?PC0 ~ PC15,PD0 ~ PD15,?PE0
    的頭像 發(fā)表于 05-29 10:02 ?1630次閱讀
    【<b class='flag-5'>GD32F303</b><b class='flag-5'>紅楓</b><b class='flag-5'>派</b><b class='flag-5'>開發(fā)板</b><b class='flag-5'>使用手冊</b>】第二<b class='flag-5'>講</b> GPIO-流水燈<b class='flag-5'>實(shí)驗(yàn)</b>

    GD32F303紅楓開發(fā)板使用手冊】第三 GPIO-按鍵查詢檢測實(shí)驗(yàn)

    GD32F303系列MCU GPIO輸入配置結(jié)構(gòu)如下圖所示,輸入可配置上下拉電阻,通過施密特觸發(fā)器后可通過備用功能輸入或者通過輸入狀態(tài)寄存器進(jìn)行讀取。
    的頭像 發(fā)表于 05-30 10:02 ?852次閱讀
    【<b class='flag-5'>GD32F303</b><b class='flag-5'>紅楓</b><b class='flag-5'>派</b><b class='flag-5'>開發(fā)板</b><b class='flag-5'>使用手冊</b>】第三<b class='flag-5'>講</b> GPIO-按鍵查詢檢測<b class='flag-5'>實(shí)驗(yàn)</b>

    GD32F303紅楓開發(fā)板使用手冊】第五 FMC-片內(nèi)Flash擦寫讀實(shí)驗(yàn)

    MC即Flash控制器,其提供了片上Flash操作所需要的所有功能,在GD32F303系列MCU中,F(xiàn)lash前256K字節(jié)空間內(nèi),?CPU執(zhí)行指令零等待,具有相同主頻下最快的代碼執(zhí)行效率。FMC也
    的頭像 發(fā)表于 06-02 10:05 ?765次閱讀
    【<b class='flag-5'>GD32F303</b><b class='flag-5'>紅楓</b><b class='flag-5'>派</b><b class='flag-5'>開發(fā)板</b><b class='flag-5'>使用手冊</b>】第五<b class='flag-5'>講</b> FMC-片內(nèi)Flash擦寫讀<b class='flag-5'>實(shí)驗(yàn)</b>

    GD32F303紅楓開發(fā)板使用手冊第十 USART-中斷串口收發(fā)實(shí)驗(yàn)

    前面章節(jié)中我們已經(jīng)學(xué)習(xí)了串口的狀態(tài)標(biāo)志位,本實(shí)驗(yàn)就是使用TBE中斷和RBNE中斷來實(shí)現(xiàn)中斷收發(fā)數(shù)據(jù),實(shí)驗(yàn)原理是RBNE中斷用來接受數(shù)據(jù),IDLE中斷用于判斷發(fā)送方數(shù)據(jù)結(jié)束,TBE中斷用
    的頭像 發(fā)表于 06-17 09:53 ?1027次閱讀
    【<b class='flag-5'>GD32F303</b><b class='flag-5'>紅楓</b><b class='flag-5'>派</b><b class='flag-5'>開發(fā)板</b><b class='flag-5'>使用手冊</b>】<b class='flag-5'>第十</b>七<b class='flag-5'>講</b> <b class='flag-5'>USART</b>-中斷<b class='flag-5'>串口</b><b class='flag-5'>收發(fā)</b><b class='flag-5'>實(shí)驗(yàn)</b>

    GD32F303紅楓開發(fā)板使用手冊】第二十 SPI-SPI NAND FLASH讀寫實(shí)驗(yàn)

    通過本實(shí)驗(yàn)主要學(xué)習(xí)以下內(nèi)容: ?SPI通信協(xié)議,參考19.2.1東方紅開發(fā)板使用手冊 ?GD32F303 SPI操作方式,參考19.2.2東方紅
    的頭像 發(fā)表于 06-20 09:50 ?883次閱讀
    【<b class='flag-5'>GD32F303</b><b class='flag-5'>紅楓</b><b class='flag-5'>派</b><b class='flag-5'>開發(fā)板</b><b class='flag-5'>使用手冊</b>】第二十<b class='flag-5'>講</b> SPI-SPI NAND FLASH讀寫<b class='flag-5'>實(shí)驗(yàn)</b>
    RM新时代网站-首页