RM新时代网站-首页

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

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

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

UART需要使用DMA發(fā)送嗎 ?

黃工的嵌入式技術(shù)圈 ? 來源:黃工的嵌入式技術(shù)圈 ? 2020-03-07 16:57 ? 次閱讀

DMA一種在嵌入式實時任務(wù)處理中常用的功能。

UART發(fā)送數(shù)據(jù)包,使用DMA方式能大量減輕CPU處理的時間,使其CPU資源不被大量浪費,尤其在UART收發(fā)大量數(shù)據(jù)包(如高頻率收發(fā)指令)時具有明顯優(yōu)勢。

Ⅰ簡述DMA

DMA:Direct Memory Access,直接內(nèi)存存取/訪問。簡單來說就是內(nèi)存RAM直接和其他設(shè)備(外設(shè))進行數(shù)據(jù)交互,而不需要CPU參與的一種控制器。

DMA它允許不同速度的硬件裝置來溝通,而不需要依賴于 CPU 的大量中斷負載。否則,CPU 需要從來源把每一片段的數(shù)據(jù)復(fù)制到暫存器,然后把它們再次寫回到新的地方。在這個時間中,CPU 對于其他的工作來說就無法使用。

ⅡDMA優(yōu)點

DMA在系統(tǒng)中的角色好比一個公司的員工,CPU好比是公司的老板。

老板想要寄送一個快遞到北京,只需要一個口令安排員工即可,具體填寫快遞單號、物流、派送等一系列工作老板不用關(guān)心。最后快遞被對方收到,通知一聲老板即可。

回到UART發(fā)送數(shù)據(jù),同樣的道理,CPU只需要簡單的操作(類似上面的“安排”),就可把一串?dāng)?shù)據(jù)包丟給DMA直接發(fā)送,最后發(fā)送完成,收到一個發(fā)送完成中斷,通知CPU發(fā)送完成即可。

說到這里相信大部分人都明白了,老板可以親自開車或者坐飛機送快遞,完成這件事情,但會耽擱老板很多時間。

同樣,如果我們使用UART自己發(fā)送,CPU就會不停仲裁發(fā)送結(jié)果,占據(jù)CPU大量資源。

RTOS中,特別是有大量任務(wù)需要處理的時候,UART使用DMA發(fā)送就會帶來很大方便。使用裸機運行的相同,尤為突出。

ⅢUART使用DMA發(fā)送配置

本文使用STM32F4 MCU、標準外設(shè)庫為例給大家簡單講述一下配置。


1.USART配置

USART(COM)宏定義:

/* COMM通信 */ #define COMM_COM USART2 #define COMM_COM_CLK RCC_APB1Periph_USART2 #define COMM_COM_TX_GPIO_CLK RCC_AHB1Periph_GPIOD //UART TX #define COMM_COM_TX_PIN GPIO_Pin_5 #define COMM_COM_TX_GPIO_PORT GPIOD #define COMM_COM_TX_SOURCE GPIO_PinSource5 #define COMM_COM_TX_AF GPIO_AF_USART2 #define COMM_COM_RX_GPIO_CLK RCC_AHB1Periph_GPIOD //UART RX #define COMM_COM_RX_PIN GPIO_Pin_6 #define COMM_COM_RX_GPIO_PORT GPIOD #define COMM_COM_RX_SOURCE GPIO_PinSource6 #define COMM_COM_RX_AF GPIO_AF_USART2 #define COMM_COM_IRQn USART2_IRQn #define COMM_COM_Priority 9 //優(yōu)先級 #define COMM_COM_BaudRate 115200 //波特率 #define COMM_COM_IRQHandler USART2_IRQHandler //中斷函數(shù)接口(見stm32f4xx_it.c)

USART配置:

/************************************************函數(shù)名稱 : USART_COMM_Configuration功 能 : 通信串口配置參 數(shù) : 無返 回 值 : 無作 者 : strongerHuang*************************************************/ void USART_COMM_Configuration(void){ GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; /* 時鐘配置 */ RCC_AHB1PeriphClockCmd(COMM_COM_TX_GPIO_CLK | COMM_COM_RX_GPIO_CLK, ENABLE); if((USART1 == COMM_COM) || (USART6 == COMM_COM)) RCC_APB2PeriphClockCmd(COMM_COM_CLK, ENABLE); else RCC_APB1PeriphClockCmd(COMM_COM_CLK, ENABLE); /* 復(fù)用配置 */ GPIO_PinAFConfig(COMM_COM_TX_GPIO_PORT, COMM_COM_TX_SOURCE, COMM_COM_TX_AF); GPIO_PinAFConfig(COMM_COM_RX_GPIO_PORT, COMM_COM_RX_SOURCE, COMM_COM_RX_AF); /* 引腳配置 */ GPIO_InitStructure.GPIO_Pin = COMM_COM_TX_PIN; //USART Tx GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; //復(fù)用模式 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; GPIO_Init(COMM_COM_TX_GPIO_PORT, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = COMM_COM_RX_PIN; //USART Rx GPIO_Init(COMM_COM_RX_GPIO_PORT, &GPIO_InitStructure); /* NVIC配置 */ NVIC_InitStructure.NVIC_IRQChannel = COMM_COM_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = COMM_COM_Priority; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); /* USART配置 */ USART_InitStructure.USART_BaudRate = COMM_COM_BaudRate; //波特率 USART_InitStructure.USART_WordLength = USART_WordLength_8b; //傳輸位數(shù) 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; //收發(fā)功能 USART_Init(COMM_COM, &USART_InitStructure); USART_ClearFlag(COMM_COM, USART_FLAG_RXNE | USART_FLAG_TC); USART_ITConfig(COMM_COM, USART_IT_RXNE, ENABLE); //接收中斷 USART_DMACmd(COMM_COM, USART_DMAReq_Tx, ENABLE); //使能DMA USART_Cmd(COMM_COM, ENABLE); //使能USART }

2.DMA配置

DMA宏定義:

/* COMM_DMA */ #define COMM_DR_ADDRESS ((uint32_t)USART2 + 0x04) #define COMM_DMA DMA1 #define COMM_DMA_CLK RCC_AHB1Periph_DMA1 #define COMM_TX_DMA_CHANNEL DMA_Channel_4 #define COMM_TX_DMA_STREAM DMA1_Stream6 #define COMM_TX_DMA_FLAG_TCIF DMA_FLAG_TCIF6 #define COMM_TX_DMA_IRQn DMA1_Stream6_IRQn #define COMM_TX_DMA_Priority 8 //優(yōu)先級 #define COMM_TX_DMA_IRQHandler DMA1_Stream6_IRQHandler //中斷函數(shù)接口(見stm32f4xx_it.c) #define COMM_TX_DMA_IT_TCIF DMA_IT_TCIF6

DMA配置:

/************************************************函數(shù)名稱 : USART_COMM_DMA_Configuration功 能 : 通信串口的DMA配置參 數(shù) : 無返 回 值 : 無作 者 : strongerHuang*************************************************/ void USART_COMM_DMA_Configuration(void){ DMA_InitTypeDef DMA_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; /* 使能時鐘 */ RCC_AHB1PeriphClockCmd(COMM_DMA_CLK, ENABLE); /* NVIC配置 */ NVIC_InitStructure.NVIC_IRQChannel = COMM_TX_DMA_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = COMM_TX_DMA_Priority; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); /* DMA配置 */ DMA_DeInit(COMM_TX_DMA_STREAM); DMA_InitStructure.DMA_Channel = COMM_TX_DMA_CHANNEL; //DMA通道 DMA_InitStructure.DMA_PeripheralBaseAddr = COMM_DR_ADDRESS; //外設(shè)地址 DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)0; //內(nèi)存地址(待傳入參數(shù)) DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral; //傳輸方向 DMA_InitStructure.DMA_BufferSize = 0; //傳輸長度(待傳入?yún)?shù)) DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外設(shè)遞增 DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //內(nèi)存遞增 DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //數(shù)據(jù)寬度 DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //循環(huán)模式 DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //優(yōu)先級 DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable; DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull; DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single; DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single; DMA_Init(COMM_TX_DMA_STREAM, &DMA_InitStructure); DMA_ClearFlag(COMM_TX_DMA_STREAM, COMM_TX_DMA_FLAG_TCIF); DMA_ITConfig(COMM_TX_DMA_STREAM, DMA_IT_TC, ENABLE); //使能DMA傳輸完成中斷 DMA_Cmd(COMM_TX_DMA_STREAM, DISABLE); //初始化禁止 } ⅣDMA發(fā)送UART數(shù)據(jù)包

DMA發(fā)送函數(shù):

/************************************************函數(shù)名稱 : COMM_SendBufByDMA功 能 : 通信串口通過DMA發(fā)送數(shù)據(jù)參 數(shù) : Buf ------ 數(shù)據(jù)(地址) Length --- 數(shù)據(jù)長度(字節(jié))返 回 值 : 無作 者 : strongerHuang*************************************************/ void COMM_SendBufByDMA(uint8_t *Buf, uint16_t Length){ DMA_Cmd(COMM_TX_DMA_STREAM, DISABLE); //關(guān)閉DMA //內(nèi)存地址 DMA_MemoryTargetConfig(COMM_TX_DMA_STREAM, (uint32_t)Buf, DMA_Memory_0); DMA_SetCurrDataCounter(COMM_TX_DMA_STREAM, Length); //設(shè)置DMA傳輸長度 DMA_Cmd(COMM_TX_DMA_STREAM, ENABLE); //使能DMA }

細心的朋友會發(fā)現(xiàn),這個發(fā)送函數(shù)其實很簡單,當(dāng)然,這里是使用STM32F4芯片,其他芯片也差不多,原理類似。 HAL庫同樣可以完成。

關(guān)于DMA發(fā)送完成中斷,可根據(jù)實際情況,如果使用RTOS,一般發(fā)送數(shù)據(jù)是一個任務(wù),這個任務(wù)會OS等待(檢測)發(fā)送完成信號(即DMA發(fā)送完成中斷)。

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

    關(guān)注

    68

    文章

    10854

    瀏覽量

    211574
  • uart
    +關(guān)注

    關(guān)注

    22

    文章

    1235

    瀏覽量

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

    關(guān)注

    3

    文章

    560

    瀏覽量

    100544
收藏 人收藏

    評論

    相關(guān)推薦

    MSPM0 UART通信中DMA和Ring Buffer環(huán)形緩沖的應(yīng)用

    電子發(fā)燒友網(wǎng)站提供《MSPM0 UART通信中DMA和Ring Buffer環(huán)形緩沖的應(yīng)用.pdf》資料免費下載
    發(fā)表于 09-05 11:01 ?0次下載
    MSPM0 <b class='flag-5'>UART</b>通信中<b class='flag-5'>DMA</b>和Ring Buffer環(huán)形緩沖的應(yīng)用

    RL78系列MCU DMAUART中的使用

    對于RL78系列的MCU,為了提高運行效率,減少CPU的占用,建議UART數(shù)據(jù)收發(fā)使用DMA功能。DMA(Direct Memory Access)是RL78 MCU內(nèi)置的一個控制器,能在支持
    的頭像 發(fā)表于 07-17 14:24 ?624次閱讀
    RL78系列MCU <b class='flag-5'>DMA</b>在<b class='flag-5'>UART</b>中的使用

    wifi的發(fā)送支持DMA嗎?

    wifi的發(fā)送支持DMA嗎? send(),這個函數(shù)是阻塞的嗎?怎么發(fā)送可以最大化發(fā)送頻率?我想在發(fā)送完成后處理一些東西,可不可以設(shè)置
    發(fā)表于 06-24 07:33

    在PSoC4中使用DMAUART遇到的疑問求解

    UART 發(fā)送一個字節(jié)。 512 字節(jié)后,DMA 應(yīng)產(chǎn)生中斷并斷開連接。 延遲 1 毫秒后,它應(yīng)該會再次發(fā)送,但我卻只看到每次突發(fā)發(fā)送 4
    發(fā)表于 05-29 07:32

    求助,關(guān)于stm32的HAL庫對UART采用DMA發(fā)送的地址位標志問題求解

    stm32 UART多機通訊可以采用地址喚醒模式,在發(fā)送地址字節(jié)時設(shè)置USART_DR第9位置1即可,原來的標準庫對這種模式的驅(qū)動沒問題,現(xiàn)在最新的HAL驅(qū)動庫對UART采用DMA
    發(fā)表于 05-09 07:01

    STM32F103 UART2通過DMA方式發(fā)送和接收,發(fā)不出去數(shù)據(jù)是怎么回事?

    發(fā)送 USART_DMACmd(USART2, USART_DMAReq_Rx, ENABLE); // 開啟串口DMA接收 } void DMA_Uart_Init(void
    發(fā)表于 04-30 06:17

    串口DMA其實一點兒都不難

    DMA一種在嵌入式實時任務(wù)處理中常用的功能。而UART發(fā)送數(shù)據(jù)包,使用DMA方式能大量減輕CPU處理的時間,使其CPU資源不被大量浪費,尤其在UAR
    的頭像 發(fā)表于 04-29 08:10 ?1978次閱讀
    串口<b class='flag-5'>DMA</b>其實一點兒都不難

    STM32H743 UART DMA數(shù)據(jù)發(fā)送報錯的原因?

    采用STM32H743芯片, 實現(xiàn)UART DMA收發(fā)數(shù)據(jù). 用圖形化的界面配置了設(shè)置, UART配置成DMA模式, 之后通過HAL_UART_Tr
    發(fā)表于 04-26 06:59

    USART3用DMA發(fā)送只能發(fā)送一次是什么原因?qū)е碌模?/a>

    )*/ DMA_InitStruct.DMA_MemoryBaseAddr = (u32)uart3.sendBuf; /*方向:從內(nèi)存到外設(shè)*/ DMA_InitStruct.DMA_DIR =
    發(fā)表于 04-19 08:21

    STM32 DMA雙緩沖發(fā)送不定長數(shù)據(jù)怎么解決?

    STM32 DMA雙緩沖發(fā)送不定長數(shù)據(jù),假如我需要發(fā)送100010個數(shù)據(jù),開辟了2塊1000數(shù)據(jù)區(qū),數(shù)據(jù)長度設(shè)置為1000,循環(huán)發(fā)送,最后一
    發(fā)表于 04-11 06:58

    使用F334的USART+DMA進行發(fā)送數(shù)據(jù)遇到的疑問求解

    最近在使用F334的USART+DMA進行發(fā)送數(shù)據(jù)的時候,遇到了一個很奇怪的問題。 在調(diào)用HAL_UART_Transmit_DMA(UART_HandleTypeDef *hua
    發(fā)表于 04-10 08:08

    求助,調(diào)用HAL_UART_Transmit_DMA函數(shù)連續(xù)發(fā)送時發(fā)現(xiàn)發(fā)不出去是為什么?

    我在調(diào)用HAL_UART_Transmit_DMA函數(shù)連續(xù)發(fā)送時發(fā)現(xiàn)發(fā)不出去(while(1)可以持續(xù)發(fā)送但可能丟了不少包),返回值為HAL_BUSY,條件即huart->gState
    發(fā)表于 04-08 06:01

    HAL_UART_Transmit_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)不能對pData賦值后發(fā)送的原因?

    在測試串口發(fā)送函數(shù)HAL_UART_Transmit_DMA(UART_HandleTypeDef huart, uint8_t pData, uint16_t Size) 時,發(fā)現(xiàn)不能對pData賦值后
    發(fā)表于 03-25 06:53

    使用UART07通過DMA進行數(shù)據(jù)發(fā)送,發(fā)出的數(shù)據(jù)不完整是怎么回事?

    使用UART07,通過DMA進行數(shù)據(jù)發(fā)送發(fā)送5個數(shù)據(jù):0x79,0x91,0x04,0x71,0x01 通過邏輯分析儀截取UART07_T
    發(fā)表于 02-21 08:30

    dma和串口直接發(fā)送的區(qū)別

    ,我們先來介紹一下DMA和串口直接發(fā)送的原理和工作方式。 DMA是一種高速數(shù)據(jù)傳輸技術(shù),它允許外設(shè)直接與內(nèi)存進行數(shù)據(jù)交換,而不需要CPU的介入。有了
    的頭像 發(fā)表于 01-07 17:43 ?2873次閱讀
    RM新时代网站-首页