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

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

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

一個嚴謹?shù)腟TM32串口DMA發(fā)送&接收(1.5Mbps波特率)機制

jf_pJlTbmA9 ? 來源:Acuity ? 作者:Acuity ? 2023-09-18 10:58 ? 次閱讀

1、前言

直接存儲器訪問(Direct Memory Access),簡稱DMA。DMA是CPU一個用于數(shù)據(jù)從一個地址空間到另一地址空間“搬運”(拷貝)的組件,數(shù)據(jù)拷貝過程不需CPU干預,數(shù)據(jù)拷貝結(jié)束則通知CPU處理。因此,大量數(shù)據(jù)拷貝時,使用DMA可以釋放CPU資源。DMA數(shù)據(jù)拷貝過程,典型的有:

內(nèi)存—>內(nèi)存,內(nèi)存間拷貝

外設—>內(nèi)存,如uart、spi、i2c等總線接收數(shù)據(jù)過程

內(nèi)存—>外設,如uart、spi、i2c等總線發(fā)送數(shù)據(jù)過程

2、串口有必要使用DMA嗎

串口(uart)是一種低速的串行異步通信,適用于低速通信場景,通常使用的波特率小于或等于115200bps。對于小于或者等于115200bps波特率的,而且數(shù)據(jù)量不大的通信場景,一般沒必要使用DMA,或者說使用DMA并未能充分發(fā)揮出DMA的作用。

對于數(shù)量大,或者波特率提高時,必須使用DMA以釋放CPU資源,因為高波特率可能帶來這樣的問題:

對于發(fā)送,使用循環(huán)發(fā)送,可能阻塞線程,需要消耗大量CPU資源“搬運”數(shù)據(jù),浪費CPU

對于發(fā)送,使用中斷發(fā)送,不會阻塞線程,但需浪費大量中斷資源,CPU頻繁響應中斷;以115200bps波特率,1s傳輸11520字節(jié),大約69us需響應一次中斷,如波特率再提高,將消耗更多CPU資源

對于接收,如仍采用傳統(tǒng)的中斷模式接收,同樣會因為頻繁中斷導致消耗大量CPU資源

因此,高波特率場景下,串口非常有必要使用DMA。

3、實現(xiàn)方式

poYBAGEBT0eAXqCfAACLPEq_Ya0048.png 整體設計圖

4、STM32串口使用DMA

關(guān)于STM32串口使用DMA,不乏一些開發(fā)板例程及網(wǎng)絡上一些博主的使用教程。使用步驟、流程、配置基本大同小異,正確性也沒什么毛病,但都是一些基本的Demo例子,作為學習過程沒問題;實際項目使用缺乏嚴謹性,數(shù)據(jù)量大時可能導致數(shù)據(jù)異常。

測試平臺:

STM32F030C8T6

UART1/UART2

DMA1 Channel2—Channel5

ST標準庫

主頻48MHz(外部12MHz晶振)

poYBAGEBT0uAOcAWAAIhGF-ggs8007.png

5、串口DMA接收

5.1 基本流程

pYYBAGEBT0-ACz32AAB7B_VPMWM153.png 串口接收流程圖

5.2 相關(guān)配置

關(guān)鍵步驟

【1】初始化串口

【2】使能串口DMA接收模式,使能串口空閑中斷

【3】配置DMA參數(shù),使能DMA通道buf半滿(傳輸一半數(shù)據(jù))中斷、buf溢滿(傳輸數(shù)據(jù)完成)中斷

【1】第一步,DMA先將數(shù)據(jù)搬運到buf1,搬運完成通知CPU來拷貝buf1數(shù)據(jù)
【2】第二步,DMA將數(shù)據(jù)搬運到buf2,與CPU拷貝buf1數(shù)據(jù)不會沖突
【3】第三步,buf2數(shù)據(jù)搬運完成,通知CPU來拷貝buf2數(shù)據(jù)
【4】執(zhí)行完第三步,DMA返回執(zhí)行第一步,一直循環(huán)

poYBAGEBT1OAd_XjAACUQ09XfyA856.png 雙緩存DMA數(shù)據(jù)搬運過程

STM32F0系列DMA不支持雙緩存(以具體型號為準)機制,但提供了一個buf"半滿中斷",即是數(shù)據(jù)搬運到buf大小的一半時,可以產(chǎn)生一個中斷信號。基于這個機制,我們可以實現(xiàn)雙緩存功能,只需將buf空間開辟大一點即可。

【1】第一步,DMA將數(shù)據(jù)搬運完成buf的前一半時,產(chǎn)生“半滿中斷”,CPU來拷貝buf前半部分數(shù)據(jù)
【2】第二步,DMA繼續(xù)將數(shù)據(jù)搬運到buf的后半部分,與CPU拷貝buf前半部數(shù)據(jù)不會沖突
【3】第三步,buf后半部分數(shù)據(jù)搬運完成,觸發(fā)“溢滿中斷”,CPU來拷貝buf后半部分數(shù)據(jù)
【4】執(zhí)行完第三步,DMA返回執(zhí)行第一步,一直循環(huán)

pYYBAGEBT1iABDoHAACdB2Nhcl0137.png 使用半滿中斷DMA數(shù)據(jù)搬運過程

UART2 DMA模式接收配置代碼如下,與其他外設使用DMA的配置基本一致,留意關(guān)鍵配置:

串口接收,DMA通道工作模式設為連續(xù)模式

使能DMA通道接收buf半滿中斷、溢滿(傳輸完成)中斷

啟動DMA通道前清空相關(guān)狀態(tài)標識,防止首次傳輸錯亂數(shù)據(jù)

void bsp_uart2_dmarx_config(uint8_t *mem_addr, uint32_t mem_size)
{
  	DMA_InitTypeDef DMA_InitStructure;
	
	DMA_DeInit(DMA1_Channel5); 
	DMA_Cmd(DMA1_Channel5, DISABLE);
	DMA_InitStructure.DMA_PeripheralBaseAddr 	= (uint32_t) (USART2->RDR);/* UART2接收數(shù)據(jù)地址 */
	DMA_InitStructure.DMA_MemoryBaseAddr 		= (uint32_t)mem_addr; /* 接收buf */
	DMA_InitStructure.DMA_DIR 					= DMA_DIR_PeripheralSRC; 	/* 傳輸方向:外設->內(nèi)存 */
	DMA_InitStructure.DMA_BufferSize 			= mem_size; /* 接收buf大小 */
	DMA_InitStructure.DMA_PeripheralInc 		= DMA_PeripheralInc_Disable; 
	DMA_InitStructure.DMA_MemoryInc 			= DMA_MemoryInc_Enable; 
	DMA_InitStructure.DMA_PeripheralDataSize 	= DMA_PeripheralDataSize_Byte; 
	DMA_InitStructure.DMA_MemoryDataSize 		= DMA_MemoryDataSize_Byte;
	DMA_InitStructure.DMA_Mode 					= DMA_Mode_Circular; /* 連續(xù)模式 */
	DMA_InitStructure.DMA_Priority 				= DMA_Priority_VeryHigh; 
	DMA_InitStructure.DMA_M2M 					= DMA_M2M_Disable; 
	DMA_Init(DMA1_Channel5,  DMA_InitStructure); 
	DMA_ITConfig(DMA1_Channel5, DMA_IT_TC|DMA_IT_HT|DMA_IT_TE, ENABLE);/* 使能DMA半滿、溢滿、錯誤中斷 */
	DMA_ClearFlag(DMA1_IT_TC5);	/* 清除相關(guān)狀態(tài)標識 */
	DMA_ClearFlag(DMA1_IT_HT5);
	DMA_Cmd(DMA1_Channel5, ENABLE); 
}

DMA 錯誤中斷“DMA_IT_TE”,一般用于前期調(diào)試使用,用于檢查DMA出現(xiàn)錯誤的次數(shù),發(fā)布軟件可以不使能該中斷。

5.3 接收處理

基于上述描述機制,DMA方式接收串口數(shù)據(jù),有三種中斷場景需要CPU去將buf數(shù)據(jù)拷貝到fifo中,分別是:

DMA通道buf溢滿(傳輸完成)場景

DMA通道buf半滿場景

串口空閑中斷場景

前兩者場景,前面文章已經(jīng)描述。串口空閑中斷指的是,數(shù)據(jù)傳輸完成后,串口監(jiān)測到一段時間內(nèi)沒有數(shù)據(jù)進來,則觸發(fā)產(chǎn)生的中斷信號。

5.3 .1 接收數(shù)據(jù)大小

數(shù)據(jù)傳輸過程是隨機的,數(shù)據(jù)大小也是不定的,存在幾類情況:

數(shù)據(jù)剛好是DMA接收buf的整數(shù)倍,這是理想的狀態(tài)

數(shù)據(jù)量小于DMA接收buf或者小于接收buf的一半,此時會觸發(fā)串口空閑中斷

因此,我們需根據(jù)“DMA通道buf大小”、“DMA通道buf剩余空間大小”、“上一次接收的總數(shù)據(jù)大小”來計算當前接收的數(shù)據(jù)大小。

/* 獲取DMA通道接收buf剩余空間大小 */
uint16_t DMA_GetCurrDataCounter(DMA_Channel_TypeDef* DMAy_Channelx);

DMA通道buf溢滿場景計算

接收數(shù)據(jù)大小 = DMA通道buf大小 - 上一次接收的總數(shù)據(jù)大小

DMA通道buf溢滿中斷處理函數(shù):

void uart_dmarx_done_isr(uint8_t uart_id)
{
  	uint16_t recv_size;
	
	recv_size = s_uart_dev[uart_id].dmarx_buf_size - s_uart_dev[uart_id].last_dmarx_size;

	fifo_write( s_uart_dev[uart_id].rx_fifo, 
				   (const uint8_t *) (s_uart_dev[uart_id].dmarx_buf[s_uart_dev[uart_id].last_dmarx_size]), recv_size);

	s_uart_dev[uart_id].last_dmarx_size = 0;
}

DMA通道buf半滿場景計算

接收數(shù)據(jù)大小 = DMA通道接收總數(shù)據(jù)大小 - 上一次接收的總數(shù)據(jù)大小
DMA通道接收總數(shù)據(jù)大小 = DMA通道buf大小 - DMA通道buf剩余空間大小

DMA通道buf半滿中斷處理函數(shù):

void uart_dmarx_half_done_isr(uint8_t uart_id)
{
  	uint16_t recv_total_size;
  	uint16_t recv_size;
	
	if(uart_id == 0)
	{
	  	recv_total_size = s_uart_dev[uart_id].dmarx_buf_size - bsp_uart1_get_dmarx_buf_remain_size();
	}
	else if (uart_id == 1)
	{
		recv_total_size = s_uart_dev[uart_id].dmarx_buf_size - bsp_uart2_get_dmarx_buf_remain_size();
	}
	recv_size = recv_total_size - s_uart_dev[uart_id].last_dmarx_size;
	
	fifo_write( s_uart_dev[uart_id].rx_fifo, 
				   (const uint8_t *) (s_uart_dev[uart_id].dmarx_buf[s_uart_dev[uart_id].last_dmarx_size]), recv_size);
	s_uart_dev[uart_id].last_dmarx_size = recv_total_size;/* 記錄接收總數(shù)據(jù)大小 */
}

串口空閑中斷場景計算

串口空閑中斷場景的接收數(shù)據(jù)計算與“DMA通道buf半滿場景”計算方式是一樣的。

串口空閑中斷處理函數(shù):

void uart_dmarx_idle_isr(uint8_t uart_id)
{
  	uint16_t recv_total_size;
  	uint16_t recv_size;
	
	if(uart_id == 0)
	{
	  	recv_total_size = s_uart_dev[uart_id].dmarx_buf_size - bsp_uart1_get_dmarx_buf_remain_size();
	}
	else if (uart_id == 1)
	{
		recv_total_size = s_uart_dev[uart_id].dmarx_buf_size - bsp_uart2_get_dmarx_buf_remain_size();
	}
	recv_size = recv_total_size - s_uart_dev[uart_id].last_dmarx_size;
	s_UartTxRxCount[uart_id*2+1] += recv_size;
	fifo_write( s_uart_dev[uart_id].rx_fifo, 
				   (const uint8_t *) (s_uart_dev[uart_id].dmarx_buf[s_uart_dev[uart_id].last_dmarx_size]), recv_size);
	s_uart_dev[uart_id].last_dmarx_size = recv_total_size;
}

注:
串口空閑中斷處理函數(shù),除了將數(shù)據(jù)拷貝到串口接收fifo中,還可以增加特殊處理,如作為串口數(shù)據(jù)傳輸完成標識、不定長度數(shù)據(jù)處理等等。

5.3.2 接收數(shù)據(jù)偏移地址

將有效數(shù)據(jù)拷貝到fifo中,除了需知道有效數(shù)據(jù)大小外,還需知道數(shù)據(jù)存儲于DMA 接收buf的偏移地址。有效數(shù)據(jù)偏移地址只需記錄上一次接收的總大小即,可,在DMA通道buf全滿中斷處理函數(shù)將該值清零,因為下一次數(shù)據(jù)將從buf的開頭存儲。

在DMA通道buf溢滿中斷處理函數(shù)中將數(shù)據(jù)偏移地址清零:

void uart_dmarx_done_isr(uint8_t uart_id)
{
 	/* todo */
	s_uart_dev[uart_id].last_dmarx_size = 0;
}

5.4 應用讀取串口數(shù)據(jù)方法

經(jīng)過前面的處理步驟,已將串口數(shù)據(jù)拷貝至接收fifo,應用程序任務只需從fifo獲取數(shù)據(jù)進行處理。前提是,處理效率必須大于DAM接收搬運數(shù)據(jù)的效率,否則導致數(shù)據(jù)丟失或者被覆蓋處理。

6、串口DMA發(fā)送

6.1 基本流程

poYBAGEBT1yAR3c8AAAsb9y6mZo460.png 串口發(fā)送流程圖

5.2 相關(guān)配置

關(guān)鍵步驟

【1】初始化串口

【2】使能串口DMA發(fā)送模式

【3】配置DMA發(fā)送通道,這一步無需在初始化設置,有數(shù)據(jù)需要發(fā)送時才配置使能DMA發(fā)送通道

UART2 DMA模式發(fā)送配置代碼如下,與其他外設使用DMA的配置基本一致,留意關(guān)鍵配置:

串口發(fā)送是,DMA通道工作模式設為單次模式(正常模式),每次需要發(fā)送數(shù)據(jù)時重新配置DMA

使能DMA通道傳輸完成中斷,利用該中斷信息處理一些必要的任務,如清空發(fā)送狀態(tài)、啟動下一次傳輸

啟動DMA通道前清空相關(guān)狀態(tài)標識,防止首次傳輸錯亂數(shù)據(jù)

void bsp_uart2_dmatx_config(uint8_t *mem_addr, uint32_t mem_size)
{
  	DMA_InitTypeDef DMA_InitStructure;
	
	DMA_DeInit(DMA1_Channel4);
	DMA_Cmd(DMA1_Channel4, DISABLE);
	DMA_InitStructure.DMA_PeripheralBaseAddr 	= (uint32_t) (USART2->TDR);/* UART2發(fā)送數(shù)據(jù)地址 */
	DMA_InitStructure.DMA_MemoryBaseAddr 		= (uint32_t)mem_addr; 	/* 發(fā)送數(shù)據(jù)buf */
	DMA_InitStructure.DMA_DIR 					= DMA_DIR_PeripheralDST; 	/* 傳輸方向:內(nèi)存->外設 */
	DMA_InitStructure.DMA_BufferSize 			= mem_size; 			/* 發(fā)送數(shù)據(jù)buf大小 */
	DMA_InitStructure.DMA_PeripheralInc 		= DMA_PeripheralInc_Disable; 
	DMA_InitStructure.DMA_MemoryInc 			= DMA_MemoryInc_Enable; 
	DMA_InitStructure.DMA_PeripheralDataSize 	= DMA_PeripheralDataSize_Byte; 
	DMA_InitStructure.DMA_MemoryDataSize 		= DMA_MemoryDataSize_Byte;
	DMA_InitStructure.DMA_Mode 					= DMA_Mode_Normal; 		/* 單次模式 */
	DMA_InitStructure.DMA_Priority 				= DMA_Priority_High;	 
	DMA_InitStructure.DMA_M2M 					= DMA_M2M_Disable; 
	DMA_Init(DMA1_Channel4,  DMA_InitStructure);  
	DMA_ITConfig(DMA1_Channel4, DMA_IT_TC|DMA_IT_TE, ENABLE); /* 使能傳輸完成中斷、錯誤中斷 */
	DMA_ClearFlag(DMA1_IT_TC4);	/* 清除發(fā)送完成標識 */
	DMA_Cmd(DMA1_Channel4, ENABLE); /* 啟動DMA發(fā)送 */
}

5.3 發(fā)送處理

串口待發(fā)送數(shù)據(jù)存于發(fā)送fifo中,發(fā)送處理函數(shù)需要做的的任務就是循環(huán)查詢發(fā)送fifo是否存在數(shù)據(jù),如存在則將該數(shù)據(jù)拷貝到DMA發(fā)送buf中,然后啟動DMA傳輸。前提是需要等待上一次DMA傳輸完畢,即是DMA接收到DMA傳輸完成中斷信號"DMA_IT_TC"。

串口發(fā)送處理函數(shù):

void uart_poll_dma_tx(uint8_t uart_id)
{
  	uint16_t size = 0;
	
	if (0x01 == s_uart_dev[uart_id].status)
    {
        return;
    }
	size = fifo_read( s_uart_dev[uart_id].tx_fifo, s_uart_dev[uart_id].dmatx_buf,
					 s_uart_dev[uart_id].dmatx_buf_size);
	if (size != 0)
	{
        s_UartTxRxCount[uart_id*2+0] += size;
	  	if (uart_id == 0)
		{
            s_uart_dev[uart_id].status = 0x01;	/* DMA發(fā)送狀態(tài) */
		  	bsp_uart1_dmatx_config(s_uart_dev[uart_id].dmatx_buf, size);
		}
		else if (uart_id == 1)
		{
            s_uart_dev[uart_id].status = 0x01;	/* DMA發(fā)送狀態(tài),必須在使能DMA傳輸前置位,否則有可能DMA已經(jīng)傳輸并進入中斷 */
			bsp_uart2_dmatx_config(s_uart_dev[uart_id].dmatx_buf, size);
		}
	}
}

注意發(fā)送狀態(tài)標識,必須先置為“發(fā)送狀態(tài)”,然后啟動DMA 傳輸。如果步驟反過來,在傳輸數(shù)據(jù)量少時,DMA傳輸時間短,“DMA_IT_TC”中斷可能比“發(fā)送狀態(tài)標識置位”先執(zhí)行,導致程序誤判DMA一直處理發(fā)送狀態(tài)(發(fā)送標識無法被清除)。

注:
關(guān)于DMA發(fā)送數(shù)據(jù)啟動函數(shù),有些博客文章描述只需改變DMA發(fā)送buf的大小即可;經(jīng)過測試發(fā)現(xiàn),該方法在發(fā)送數(shù)據(jù)量較小時可行,數(shù)據(jù)量大后,導致發(fā)送失敗,而且不會觸發(fā)DMA發(fā)送完成中斷。因此,可靠辦法是:每次啟動DMA發(fā)送,重新配置DMA通道所有參數(shù)。該步驟只是配置寄存器過程,實質(zhì)上不會占用很多CPU執(zhí)行時間。

DMA傳輸完成中斷處理函數(shù):

void uart_dmatx_done_isr(uint8_t uart_id)
{
 	s_uart_dev[uart_id].status = 0;	/* 清空DMA發(fā)送狀態(tài)標識 */
}

上述串口發(fā)送處理函數(shù)可以在幾種情況調(diào)用:

主線程任務調(diào)用,前提是線程不能被其他任務阻塞,否則導致fifo溢出

void thread(void)
{
    uart_poll_dma_tx(DEV_UART1);
    uart_poll_dma_tx(DEV_UART2);
}

定時器中斷中調(diào)用

void TIMx_IRQHandler(void)
{
    uart_poll_dma_tx(DEV_UART1);
    uart_poll_dma_tx(DEV_UART2);
}

DMA通道傳輸完成中斷中調(diào)用

void DMA1_Channel4_5_IRQHandler(void)
{
	if(DMA_GetITStatus(DMA1_IT_TC4))
	{
		UartDmaSendDoneIsr(UART_2);
		DMA_ClearFlag(DMA1_FLAG_TC4);
		uart_poll_dma_tx(DEV_UART2);
	}
}

每次拷貝多少數(shù)據(jù)量到DMA發(fā)送buf:

關(guān)于這個問題,與具體應用場景有關(guān),遵循的原則就是:只要發(fā)送fifo的數(shù)據(jù)量大于等于DMA發(fā)送buf的大小,就應該填滿DMA發(fā)送buf,然后啟動DMA傳輸,這樣才能充分發(fā)揮會DMA性能。因此,需兼顧每次DMA傳輸?shù)男屎痛跀?shù)據(jù)流實時性,參考著幾類實現(xiàn):

周期查詢發(fā)送fifo數(shù)據(jù),啟動DMA傳輸,充分利用DMA發(fā)送效率,但可能降低串口數(shù)據(jù)流實時性

實時查詢發(fā)送fifo數(shù)據(jù),加上超時處理,理想的方法

在DMA傳輸完成中斷中處理,保證實時連續(xù)數(shù)據(jù)流

6、串口設備

6.1 數(shù)據(jù)結(jié)構(gòu)

/* 串口設備數(shù)據(jù)結(jié)構(gòu) */
typedef struct
{
	uint8_t status;			/* 發(fā)送狀態(tài) */
	_fifo_t tx_fifo;		/* 發(fā)送fifo */
	_fifo_t rx_fifo;		/* 接收fifo */
	uint8_t *dmarx_buf;		/* dma接收緩存 */
	uint16_t dmarx_buf_size;/* dma接收緩存大小*/
	uint8_t *dmatx_buf;		/* dma發(fā)送緩存 */
	uint16_t dmatx_buf_size;/* dma發(fā)送緩存大小 */
	uint16_t last_dmarx_size;/* dma上一次接收數(shù)據(jù)大小 */
}uart_device_t;

6.2 對外接口

/* 串口注冊初始化函數(shù) */
void uart_device_init(uint8_t uart_id)
{
  	if (uart_id == 1)
	{
		/* 配置串口2收發(fā)fifo */
		fifo_register( s_uart_dev[uart_id].tx_fifo,  s_uart2_tx_buf[0], 
                      sizeof(s_uart2_tx_buf), fifo_lock, fifo_unlock);
		fifo_register( s_uart_dev[uart_id].rx_fifo,  s_uart2_rx_buf[0], 
                      sizeof(s_uart2_rx_buf), fifo_lock, fifo_unlock);
		
		/* 配置串口2 DMA收發(fā)buf */
		s_uart_dev[uart_id].dmarx_buf =  s_uart2_dmarx_buf[0];
		s_uart_dev[uart_id].dmarx_buf_size = sizeof(s_uart2_dmarx_buf);
		s_uart_dev[uart_id].dmatx_buf =  s_uart2_dmatx_buf[0];
		s_uart_dev[uart_id].dmatx_buf_size = sizeof(s_uart2_dmatx_buf);
		bsp_uart2_dmarx_config(s_uart_dev[uart_id].dmarx_buf, 
							   sizeof(s_uart2_dmarx_buf));
		s_uart_dev[uart_id].status  = 0;
	}
}

/* 串口發(fā)送函數(shù) */
uint16_t uart_write(uint8_t uart_id, const uint8_t *buf, uint16_t size)
{
	return fifo_write( s_uart_dev[uart_id].tx_fifo, buf, size);
}

/* 串口讀取函數(shù) */
uint16_t uart_read(uint8_t uart_id, uint8_t *buf, uint16_t size)
{
	return fifo_read( s_uart_dev[uart_id].rx_fifo, buf, size);
}

7、相關(guān)文章

依賴的fifo參考該文章:

【1】通用環(huán)形緩沖區(qū)模塊

8、完整源碼

代碼倉庫:https://github.com/Prry/stm32f0-uart-dma

串口 DMA底層配置:

#include 
#include 
#include 
#include "stm32f0xx.h"
#include "bsp_uart.h"

/**
 * @brief  
 * @param  
 * @retval 
 */
static void bsp_uart1_gpio_init(void)
{
    GPIO_InitTypeDef    GPIO_InitStructure;
#if 0
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOB, ENABLE);
	
    GPIO_PinAFConfig(GPIOB, GPIO_PinSource6, GPIO_AF_0);
    GPIO_PinAFConfig(GPIOB, GPIO_PinSource7, GPIO_AF_0); 
	
	GPIO_InitStructure.GPIO_Pin 	= GPIO_Pin_6 | GPIO_Pin_7;
    GPIO_InitStructure.GPIO_Mode 	= GPIO_Mode_AF;
	GPIO_InitStructure.GPIO_OType 	= GPIO_OType_PP;
    GPIO_InitStructure.GPIO_Speed  	= GPIO_Speed_Level_3;
    GPIO_InitStructure.GPIO_PuPd 	= GPIO_PuPd_UP;
    GPIO_Init(GPIOB,  GPIO_InitStructure);
#else
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA, ENABLE);
	
    GPIO_PinAFConfig(GPIOB, GPIO_PinSource9, GPIO_AF_1);
    GPIO_PinAFConfig(GPIOB, GPIO_PinSource10, GPIO_AF_1); 
	
	GPIO_InitStructure.GPIO_Pin 	= GPIO_Pin_9 | GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode 	= GPIO_Mode_AF;
	GPIO_InitStructure.GPIO_OType 	= GPIO_OType_PP;
    GPIO_InitStructure.GPIO_Speed  	= GPIO_Speed_Level_3;
    GPIO_InitStructure.GPIO_PuPd 	= GPIO_PuPd_UP;
    GPIO_Init(GPIOA,  GPIO_InitStructure);
#endif
}

/**
 * @brief  
 * @param  
 * @retval 
 */
static void bsp_uart2_gpio_init(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOB, ENABLE);
	
	GPIO_PinAFConfig(GPIOA, GPIO_PinSource2, GPIO_AF_1);
	GPIO_PinAFConfig(GPIOA, GPIO_PinSource3, GPIO_AF_1);
	
	GPIO_InitStructure.GPIO_Pin   = GPIO_Pin_2 | GPIO_Pin_3;
	GPIO_InitStructure.GPIO_Mode  = GPIO_Mode_AF;
	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;
	GPIO_InitStructure.GPIO_PuPd  = GPIO_PuPd_UP;
	GPIO_Init(GPIOA,  GPIO_InitStructure);
}

/**
 * @brief  
 * @param  
 * @retval 
 */
void bsp_uart1_init(void)
{
	USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	
	bsp_uart1_gpio_init();
	
	/* 使能串口和DMA時鐘 */
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);
	
	USART_InitStructure.USART_BaudRate            = 57600;
	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_ITConfig(USART1, USART_IT_IDLE, ENABLE);	/* 使能空閑中斷 */
	USART_OverrunDetectionConfig(USART1, USART_OVRDetection_Disable);
	
	USART_Cmd(USART1, ENABLE);
	USART_DMACmd(USART1, USART_DMAReq_Rx|USART_DMAReq_Tx, ENABLE); /* 使能DMA收發(fā) */

	/* 串口中斷 */
	NVIC_InitStructure.NVIC_IRQChannel         = USART1_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelPriority = 2;
	NVIC_InitStructure.NVIC_IRQChannelCmd      = ENABLE;
	NVIC_Init( NVIC_InitStructure);

	/* DMA中斷 */
  	NVIC_InitStructure.NVIC_IRQChannel 		   = DMA1_Channel2_3_IRQn;       
  	NVIC_InitStructure.NVIC_IRQChannelPriority = 0; 
	NVIC_InitStructure.NVIC_IRQChannelCmd      = ENABLE;
  	NVIC_Init( NVIC_InitStructure);
}

/**
 * @brief  
 * @param  
 * @retval 
 */
void bsp_uart2_init(void)
{
	USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	
	bsp_uart2_gpio_init();
	
	/* 使能串口和DMA時鐘 */
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);

	USART_InitStructure.USART_BaudRate            = 57600;
	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(USART2,  USART_InitStructure);
	
	USART_ITConfig(USART2, USART_IT_IDLE, ENABLE);	/* 使能空閑中斷 */
	USART_OverrunDetectionConfig(USART2, USART_OVRDetection_Disable);
	
	USART_Cmd(USART2, ENABLE);
	USART_DMACmd(USART2, USART_DMAReq_Rx|USART_DMAReq_Tx, ENABLE); 	/* 使能DMA收發(fā) */

	/* 串口中斷 */
	NVIC_InitStructure.NVIC_IRQChannel         = USART2_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelPriority = 2;
	NVIC_InitStructure.NVIC_IRQChannelCmd      = ENABLE;
	NVIC_Init( NVIC_InitStructure);

	/* DMA中斷 */
	NVIC_InitStructure.NVIC_IRQChannel         = DMA1_Channel4_5_IRQn;       
  	NVIC_InitStructure.NVIC_IRQChannelPriority = 0; 
	NVIC_InitStructure.NVIC_IRQChannelCmd      = ENABLE;
  	NVIC_Init( NVIC_InitStructure);
}

void bsp_uart1_dmatx_config(uint8_t *mem_addr, uint32_t mem_size)
{
  	DMA_InitTypeDef DMA_InitStructure;
	
	DMA_DeInit(DMA1_Channel2);
	DMA_Cmd(DMA1_Channel2, DISABLE);
	DMA_InitStructure.DMA_PeripheralBaseAddr 	= (uint32_t) (USART1->TDR);
	DMA_InitStructure.DMA_MemoryBaseAddr 		= (uint32_t)mem_addr; 
	DMA_InitStructure.DMA_DIR 					= DMA_DIR_PeripheralDST; 	/* 傳輸方向:內(nèi)存->外設 */
	DMA_InitStructure.DMA_BufferSize 			= mem_size; 
	DMA_InitStructure.DMA_PeripheralInc 		= DMA_PeripheralInc_Disable; 
	DMA_InitStructure.DMA_MemoryInc 			= DMA_MemoryInc_Enable; 
	DMA_InitStructure.DMA_PeripheralDataSize 	= DMA_PeripheralDataSize_Byte; 
	DMA_InitStructure.DMA_MemoryDataSize 		= DMA_MemoryDataSize_Byte;
	DMA_InitStructure.DMA_Mode 					= DMA_Mode_Normal; 
	DMA_InitStructure.DMA_Priority 				= DMA_Priority_High; 
	DMA_InitStructure.DMA_M2M 					= DMA_M2M_Disable; 
	DMA_Init(DMA1_Channel2,  DMA_InitStructure);  
	DMA_ITConfig(DMA1_Channel2, DMA_IT_TC|DMA_IT_TE, ENABLE); 
	DMA_ClearFlag(DMA1_IT_TC2);	/* 清除發(fā)送完成標識 */
	DMA_Cmd(DMA1_Channel2, ENABLE); 
}

void bsp_uart1_dmarx_config(uint8_t *mem_addr, uint32_t mem_size)
{
  	DMA_InitTypeDef DMA_InitStructure;
	
	DMA_DeInit(DMA1_Channel3); 
	DMA_Cmd(DMA1_Channel3, DISABLE);
	DMA_InitStructure.DMA_PeripheralBaseAddr 	= (uint32_t) (USART1->RDR);
	DMA_InitStructure.DMA_MemoryBaseAddr 		= (uint32_t)mem_addr; 
	DMA_InitStructure.DMA_DIR 					= DMA_DIR_PeripheralSRC; 	/* 傳輸方向:外設->內(nèi)存 */
	DMA_InitStructure.DMA_BufferSize 			= mem_size; 
	DMA_InitStructure.DMA_PeripheralInc 		= DMA_PeripheralInc_Disable; 
	DMA_InitStructure.DMA_MemoryInc 			= DMA_MemoryInc_Enable; 
	DMA_InitStructure.DMA_PeripheralDataSize 	= DMA_PeripheralDataSize_Byte; 
	DMA_InitStructure.DMA_MemoryDataSize 		= DMA_MemoryDataSize_Byte;
	DMA_InitStructure.DMA_Mode 					= DMA_Mode_Circular; 
	DMA_InitStructure.DMA_Priority 				= DMA_Priority_VeryHigh; 
	DMA_InitStructure.DMA_M2M 					= DMA_M2M_Disable; 
	DMA_Init(DMA1_Channel3,  DMA_InitStructure); 
	DMA_ITConfig(DMA1_Channel3, DMA_IT_TC|DMA_IT_HT|DMA_IT_TE, ENABLE);/* 使能DMA半滿、全滿、錯誤中斷 */
	DMA_ClearFlag(DMA1_IT_TC3);
	DMA_ClearFlag(DMA1_IT_HT3);
	DMA_Cmd(DMA1_Channel3, ENABLE); 
}

uint16_t bsp_uart1_get_dmarx_buf_remain_size(void)
{
	return DMA_GetCurrDataCounter(DMA1_Channel3);	/* 獲取DMA接收buf剩余空間 */
}

void bsp_uart2_dmatx_config(uint8_t *mem_addr, uint32_t mem_size)
{
  	DMA_InitTypeDef DMA_InitStructure;
	
	DMA_DeInit(DMA1_Channel4);
	DMA_Cmd(DMA1_Channel4, DISABLE);
	DMA_InitStructure.DMA_PeripheralBaseAddr 	= (uint32_t) (USART2->TDR);
	DMA_InitStructure.DMA_MemoryBaseAddr 		= (uint32_t)mem_addr; 
	DMA_InitStructure.DMA_DIR 					= DMA_DIR_PeripheralDST; 	/* 傳輸方向:內(nèi)存->外設 */
	DMA_InitStructure.DMA_BufferSize 			= mem_size; 
	DMA_InitStructure.DMA_PeripheralInc 		= DMA_PeripheralInc_Disable; 
	DMA_InitStructure.DMA_MemoryInc 			= DMA_MemoryInc_Enable; 
	DMA_InitStructure.DMA_PeripheralDataSize 	= DMA_PeripheralDataSize_Byte; 
	DMA_InitStructure.DMA_MemoryDataSize 		= DMA_MemoryDataSize_Byte;
	DMA_InitStructure.DMA_Mode 					= DMA_Mode_Normal; 
	DMA_InitStructure.DMA_Priority 				= DMA_Priority_High; 
	DMA_InitStructure.DMA_M2M 					= DMA_M2M_Disable; 
	DMA_Init(DMA1_Channel4,  DMA_InitStructure);  
	DMA_ITConfig(DMA1_Channel4, DMA_IT_TC|DMA_IT_TE, ENABLE); 
	DMA_ClearFlag(DMA1_IT_TC4);	/* 清除發(fā)送完成標識 */
	DMA_Cmd(DMA1_Channel4, ENABLE); 
}

void bsp_uart2_dmarx_config(uint8_t *mem_addr, uint32_t mem_size)
{
  	DMA_InitTypeDef DMA_InitStructure;
	
	DMA_DeInit(DMA1_Channel5); 
	DMA_Cmd(DMA1_Channel5, DISABLE);
	DMA_InitStructure.DMA_PeripheralBaseAddr 	= (uint32_t) (USART2->RDR);
	DMA_InitStructure.DMA_MemoryBaseAddr 		= (uint32_t)mem_addr; 
	DMA_InitStructure.DMA_DIR 					= DMA_DIR_PeripheralSRC; 	/* 傳輸方向:外設->內(nèi)存 */
	DMA_InitStructure.DMA_BufferSize 			= mem_size; 
	DMA_InitStructure.DMA_PeripheralInc 		= DMA_PeripheralInc_Disable; 
	DMA_InitStructure.DMA_MemoryInc 			= DMA_MemoryInc_Enable; 
	DMA_InitStructure.DMA_PeripheralDataSize 	= DMA_PeripheralDataSize_Byte; 
	DMA_InitStructure.DMA_MemoryDataSize 		= DMA_MemoryDataSize_Byte;
	DMA_InitStructure.DMA_Mode 					= DMA_Mode_Circular; 
	DMA_InitStructure.DMA_Priority 				= DMA_Priority_VeryHigh; 
	DMA_InitStructure.DMA_M2M 					= DMA_M2M_Disable; 
	DMA_Init(DMA1_Channel5,  DMA_InitStructure); 
	DMA_ITConfig(DMA1_Channel5, DMA_IT_TC|DMA_IT_HT|DMA_IT_TE, ENABLE);/* 使能DMA半滿、全滿、錯誤中斷 */
	DMA_ClearFlag(DMA1_IT_TC5);
	DMA_ClearFlag(DMA1_IT_HT5);
	DMA_Cmd(DMA1_Channel5, ENABLE); 
}

uint16_t bsp_uart2_get_dmarx_buf_remain_size(void)
{
	return DMA_GetCurrDataCounter(DMA1_Channel5);	/* 獲取DMA接收buf剩余空間 */
}

壓力測試:

1.5Mbps波特率,串口助手每毫秒發(fā)送1k字節(jié)數(shù)據(jù),stm32f0 DMA接收數(shù)據(jù),再通過DMA發(fā)送回串口助手,毫無壓力。

1.5Mbps波特率,可傳輸大文件測試,將接收數(shù)據(jù)保存為文件,與源文件比較。

串口高波特率測試需要USB轉(zhuǎn)TLL工具及串口助手都支持才可行,推薦CP2102、FT232芯片的USB轉(zhuǎn)TTL工具。

poYBAGEBT2CAJZ0GAABRvHQJtpw822.png 1.5Mbps串口回環(huán)壓力測試

————————————————

原文鏈接:csdn
免責聲明:本文為轉(zhuǎn)載文章,轉(zhuǎn)載此文目的在于傳遞更多信息,版權(quán)歸原作者所有。本文所用視頻、圖片、文字如涉及作品版權(quán)問題,請聯(lián)系小編進行處理

審核編輯 黃宇

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

    關(guān)注

    68

    文章

    10780

    瀏覽量

    210488
  • STM32
    +關(guān)注

    關(guān)注

    2263

    文章

    10849

    瀏覽量

    353827
  • 串口
    +關(guān)注

    關(guān)注

    14

    文章

    1539

    瀏覽量

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

    關(guān)注

    3

    文章

    555

    瀏覽量

    100236
收藏 人收藏

    評論

    相關(guān)推薦

    STM32串口波特率大小計算案例

    STM32串口波特率通過USART_BRR進行設置,STM32波特率寄存器支持分數(shù)設置,以提高精確度。USART_BRR的前4位用于表
    的頭像 發(fā)表于 10-23 14:19 ?8105次閱讀
    <b class='flag-5'>STM32</b><b class='flag-5'>串口</b><b class='flag-5'>波特率</b>大小計算案例

    STM32串口發(fā)送數(shù)據(jù)和接收數(shù)據(jù)方式總結(jié)

    STM32串口發(fā)送數(shù)據(jù)和接收數(shù)據(jù)方式總結(jié)
    的頭像 發(fā)表于 09-19 09:14 ?6925次閱讀
    <b class='flag-5'>STM32</b><b class='flag-5'>串口</b><b class='flag-5'>發(fā)送</b>數(shù)據(jù)和<b class='flag-5'>接收</b>數(shù)據(jù)方式總結(jié)

    STM32F103串口發(fā)送數(shù)據(jù)后需要更改波特率應該用什么做判斷?

    STM32F103串口發(fā)送數(shù)據(jù)后,需要更改波特率,應該用什么做判斷?STM32F103串口
    發(fā)表于 09-04 09:33

    如何去計算STM32串口接收器和發(fā)送器的波特率

    如何去計算STM32串口接收器和發(fā)送器的波特率呢?設置波特率時的誤差該如何去計算呢?
    發(fā)表于 12-09 07:58

    STM32CubeMX的相關(guān)資料分享

    鏈接5中的代碼)1.C語言中volatile關(guān)鍵字的學習2.【STM32】HAL庫 STM32CubeMX教程十一—DMA (串口DMA
    發(fā)表于 01-14 07:10

    STM32串口自動識別波特率

    STM32串口自動識別波特率
    的頭像 發(fā)表于 03-04 09:53 ?9238次閱讀

    STM32串口DMA發(fā)送 接收1.5Mbps波特率機制

    等總線接收數(shù)據(jù)過程 內(nèi)存外設,如uart、spi、i2c等總線發(fā)送數(shù)據(jù)過程 2 串口有必要使用DMA串口
    的頭像 發(fā)表于 11-02 11:24 ?3467次閱讀
    <b class='flag-5'>一</b><b class='flag-5'>個</b><b class='flag-5'>STM32</b><b class='flag-5'>串口</b><b class='flag-5'>DMA</b><b class='flag-5'>發(fā)送</b> <b class='flag-5'>接收</b>(<b class='flag-5'>1.5Mbps</b><b class='flag-5'>波特率</b>)<b class='flag-5'>機制</b>

    STC12單片機雙串口共用獨立波特率發(fā)生器發(fā)生波特率

    ;uint Smog = 0 ;void Uart_Init(void){ EA=1; //打開總中斷 PCON &amp;amp;= 0x7F; //波特率不加倍 /* 定時器
    發(fā)表于 11-18 11:51 ?23次下載
    STC12單片機雙<b class='flag-5'>串口</b>共用獨立<b class='flag-5'>波特率</b>發(fā)生器發(fā)生<b class='flag-5'>波特率</b>

    串口波特率計算詳解

    USART_BRR,如下:????STM32串口波特率通過USART_BRR進行設置,STM32波特率寄存器支持分數(shù)設置,以提高精確度。U
    發(fā)表于 12-05 09:06 ?22次下載
    <b class='flag-5'>串口</b><b class='flag-5'>波特率</b>計算詳解

    STM32串口DMA發(fā)送數(shù)據(jù)

    ,省時省力,首先打開STMCubeMX新建工程,然后選擇對應的芯片2.外設配置設置RCC,設置高速外部時鐘HSE 選擇外部時鐘源3.串口配置點擊USART,mode設置為異步通信波特率
    發(fā)表于 12-07 10:36 ?22次下載
    <b class='flag-5'>STM32</b><b class='flag-5'>串口</b><b class='flag-5'>DMA</b><b class='flag-5'>發(fā)送</b>數(shù)據(jù)

    STM32踩坑:STM32串口發(fā)送亂碼問題

    STM32串口發(fā)送亂碼問題小編是嵌入式初學者,才學沒多久,將近兩個月的樣子,在學習過程中遇到了
    發(fā)表于 12-24 18:40 ?10次下載
    <b class='flag-5'>STM32</b>踩坑:<b class='flag-5'>STM32</b><b class='flag-5'>串口</b><b class='flag-5'>發(fā)送</b>亂碼問題

    STM32CUBEMX配置教程(九)STM32串口DMA收發(fā)數(shù)據(jù)

    ,因此寫下這系列教程以供記憶,順便讓我這個大萌新給廣大小萌新提供些學習幫助。此次工程效果:串口115200波特率,接收
    發(fā)表于 12-24 18:47 ?24次下載
    <b class='flag-5'>STM32</b>CUBEMX配置教程(九)<b class='flag-5'>STM32</b><b class='flag-5'>串口</b><b class='flag-5'>DMA</b>收發(fā)數(shù)據(jù)

    HAL庫STM32串口DMA不定長收發(fā),空閑中斷

    參考了鏈接5中的代碼)1.C語言中volatile關(guān)鍵字的學習2.【STM32】HAL庫 STM32CubeMX教程十一—DMA (串口DMA
    發(fā)表于 01-14 10:40 ?18次下載
    HAL庫<b class='flag-5'>STM32</b><b class='flag-5'>串口</b><b class='flag-5'>DMA</b>不定長收發(fā),空閑中斷

    STM32波特率計算詳解

    STM32串口波特率通過USART_BRR進行設置,STM32波特率寄存器支持分數(shù)設置,以提高精確度。USART_BRR的前4位用于
    發(fā)表于 02-11 10:46 ?20次下載
    <b class='flag-5'>STM32</b>下<b class='flag-5'>波特率</b>計算詳解

    STM32串口DMA接收發(fā)送

    串口(uart)是種低速的串行異步通信,適用于低速通信場景,通常使用的波特率小于或等于115200bps。對于小于或者等于115200bps波特率的,而且數(shù)據(jù)量不大的通信場景,
    的頭像 發(fā)表于 04-19 14:59 ?1w次閱讀