1、開發平臺
開發環境:Keil MDK 5.14;
MCU:STM32F407ZET6;
STM32F4xx固件庫:STM32F4xx_DSP_StdPeriph_Lib_V1.4.0;
串口調試助手;
2、問題描述
在測試用STM32F4xx芯片的串口USART1以DMA方式進行RS485收發通訊時,出現數據字節丟失的現象,一般丟失1~2個字節。
出現問題時測試的簡單收發機制:使能串口USART1的DMA收發功能,開啟了DMA發送完成中斷和USART1空閑中斷。通過串口調試助手發送N個字節給MCU,當MCU產生USART1空閑中斷時,在USART1空閑中斷服務程序中將DMA接收到的N個字節數據從接收緩存拷貝到發送緩存,準備好數據后,RS485切換為發送模式,通過啟動一次DMA發送,將N個字節數據原樣回送到串口調試助手。最后,在DMA發送完成中斷服務程序中判斷到有DMA發送完成標志TCIF7置位時,立即將RS485再次切換為接收模式。
3、原因分析
在STM32F4xx英文參考手冊(RM0090)中,USART章節的使用DMA發送小節給出了如下時序圖:
由圖可見,當DMA將第3個字節Frame 3寫到USART數據寄存器USART_DR時,TX線上才剛準備出現第2個字節Frame 2的時序,并且DMA發送完成中斷標志在TX線還未出現第2個字節Frame 2時序時就由硬件置1了,所以,如果軟件中在DMA發送完成中斷服務程序中檢測到DMA TCIF標志置1后馬上將RS485切換為接收模式,則后面的字節數據將會丟失。
所以,需要讓數據字節不丟失的話,必須讓所有字節(包括字節的停止位)在TX線上穩定發送完成后,才能將RS485切換為接收模式。
4、解決方法
如上圖所示,有一個關鍵點是:當所有字節(包括字節的停止位)在TX線上穩定發送完成后,串口發送完成標志(TC flag)置1。所以,有兩個解決方法:
方法一:用DMA發送完成中斷,不用USART1發送完成中斷。在DMA發送完成中斷服務程序中檢測到有TCIF7置1時,再等待USART1發送完成標志TC置1,直到USART1發送完成標志TC置1后,清零USART1發送完成標志TC,然后再將RS485切換為接收模式。
方法二:用USART1發送完成中斷,不用DMA發送完成中斷。在USART1中斷服務程序USART1_IRQHandler()中,檢測到有USART1發送完成標志TC置1時,清零USART1發送完成標志TC,并且要清零DMA發送完成標志DMA_FLAG_TCIF7,最好同時清零DMA_FLAG_FEIF7、DMA_FLAG_DMEIF7、DMA_FLAG_TEIF7 、DMA_FLAG_HTIF7,然后再將RS485切換為接收模式。
5、參考源代碼
Usart.h頭文件
/*----------------------------------------------------------------------------------------------------
*Copyright: SXD Tech. Co., Ltd.
*開發 環境: Keil MDK 5.14 && STM32F407ZET6
*文件 名稱: USART串行通信驅動頭文件
*作 者: 順信德
*版 本: V1.0
*日 期: 2018-2-6
*說 明:
*修改 日志: (1)
----------------------------------------------------------------------------------------------------*/
#ifndef _USART_H_
#define _USART_H_
#include "Global.h"
/*---------------------------------------------宏定義(S)---------------------------------------------*/
#define RS485_Recv(); {PFout(11)=0;} //SP485接收模式,低電平有效
#define RS485_Send(); {PFout(11)=1;} //SP485發送模式,高電平有效
#define USART1_SEND_MAXLEN 512 /*串口1最大發送字節長度*/
#define USART1_RECV_MAXLEN 512 /*串口1最大接收字節長度*/
/*---------------------------------------------宏定義(E)---------------------------------------------*/
/*--------------------------------------------端口定義(S)--------------------------------------------*/
/*--------------------------------------------端口定義(E)--------------------------------------------*/
/*--------------------------------------------變量聲明(S)--------------------------------------------*/
extern u32 G_u32RS485BaudRate; //RS485通訊波特率
extern u8 G_u8Usart1SendBuf[USART1_SEND_MAXLEN]; //發送數據緩沖區
extern u8 G_u8Usart1RecvBuf[USART1_RECV_MAXLEN]; //接收數據緩沖區
extern u16 G_u16CommRecvLen; //通訊接收的一幀數據長度
/*--------------------------------------------變量聲明(E)--------------------------------------------*/
/*--------------------------------------------函數聲明(S)--------------------------------------------*/
extern void USART1_Init(u32 baud); //USART1串口初始化
extern void USART_DMA_SendStart(DMA_Stream_TypeDef *DMA_Streamx, u16 m_u16SendCnt); //串口USART1啟動一次DMA傳輸
extern void USART1_DMA_SendNByte(u8 *m_pSendBuf, u16 m_u16SendCnt); //串口USART1以DMA方式發送多字節
/*--------------------------------------------函數聲明(E)--------------------------------------------*/
#endif
Usart.c源文件
方法一:用DMA發送完成中斷
/*----------------------------------------------------------------------------------------------------
*Copyright: SXD Tech. Co., Ltd.
*開發 環境: Keil MDK 5.14 && STM32F407ZET6
*文件 名稱: USART串行通信驅動源文件
*作 者: 順信德
*版 本: V1.0
*日 期: 2018-2-6
*說 明:
*修改 日志: (1)
----------------------------------------------------------------------------------------------------*/
#include "Usart.h"
/*--------------------------------------------變量定義(S)--------------------------------------------*/
u32 G_u32RS485BaudRate = 9600; //RS485通訊波特率
u8 G_u8Usart1SendBuf[USART1_SEND_MAXLEN]={0,}; //發送數據緩沖區
u8 G_u8Usart1RecvBuf[USART1_RECV_MAXLEN]={0,}; //接收數據緩沖區
u16 G_u16CommRecvLen=0; //通訊接收的一幀數據長度
/*--------------------------------------------變量定義(E)--------------------------------------------*/
/*--------------------------------------------函數聲明(S)--------------------------------------------*/
void USART1_Init(u32 baud); //USART1串口初始化
void USART_DMA_SendStart(DMA_Stream_TypeDef *DMA_Streamx, u16 m_u16SendCnt); //串口USART1啟動一次DMA傳輸
void USART1_DMA_SendNByte(u8 *m_pSendBuf, u16 m_u16SendCnt); //串口USART1以DMA方式發送多字節
/*--------------------------------------------函數聲明(E)--------------------------------------------*/
/*----------------------------------------------------------------------------------------------------
*函數名稱:void USART1_Init(u32 baud)
*函數功能:USART1串口初始化函數
*入口參數:u32 baud - 波特率(單位bps)
*出口參數:無
*說 明:用于RS485通信;
----------------------------------------------------------------------------------------------------*/
void USART1_Init(u32 baud)
{
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
DMA_InitTypeDef DMA_InitStructure;
u16 mid_u16RetryCnt = 0;
USART_DeInit(USART1);
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); //使能GPIOA時鐘
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); //使能USART1時鐘
//USART1對應引腳復用映射
GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_USART1); //GPIOA9復用為USART1_TX
GPIO_PinAFConfig(GPIOA, GPIO_PinSource10, GPIO_AF_USART1); //GPIOA10復用為USART1_RX
//USART1端口配置
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10; //GPIOA9與GPIOA10
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; //復用功能
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_25MHz; //速度25MHz
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //推挽輸出
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; //上拉
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PA9,PA10
//USART1初始化設置
USART_InitStructure.USART_BaudRate = baud; //波特率設置
USART_InitStructure.USART_WordLength = USART_WordLength_8b; //字長為8位數據格式
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); //初始化USART1
USART_Cmd(USART1, ENABLE); //使能USART1
USART_ClearFlag(USART1, USART_FLAG_TC); //清除發送完成標志
while(USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET); //等待空閑幀發送完成后再清零發送完成標志
USART_ClearFlag(USART1, USART_FLAG_TC); //清除發送完成標志
USART_ITConfig(USART1, USART_IT_TC, DISABLE); //禁止USART1傳輸完成中斷
USART_ITConfig(USART1, USART_IT_RXNE, DISABLE); //禁止USART1接收不為空中斷
USART_ITConfig(USART1, USART_IT_TXE, DISABLE); //禁止USART1發送空中斷
USART_ITConfig(USART1, USART_IT_IDLE, ENABLE); //開啟USART1空閑中斷
//USART1 NVIC配置
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; //串口1中斷通道
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3; //搶占優先級3
NVIC_InitStructure.NVIC_IRQChannelSubPriority =3; //子優先級3
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
NVIC_Init(&NVIC_InitStructure);
USART_DMACmd(USART1, USART_DMAReq_Tx, ENABLE); //使能串口1的DMA發送
USART_DMACmd(USART1, USART_DMAReq_Rx, ENABLE); //使能串口1的DMA接收
// - USART1發送DMA配置
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2,ENABLE); //DMA2時鐘使能
DMA_DeInit(DMA2_Stream7);
while ((DMA_GetCmdStatus(DMA2_Stream7) != DISABLE) && (mid_u16RetryCnt++ < 500)); //等待DMA可配置
//配置DMA2_Stream7
DMA_InitStructure.DMA_Channel = DMA_Channel_4; //通道選擇
DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&USART1->DR; //DMA外設地址
DMA_InitStructure.DMA_Memory0BaseAddr = (u32)G_u8Usart1SendBuf; //DMA 存儲器0地址
DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral; //存儲器到外設模式
DMA_InitStructure.DMA_BufferSize = USART1_SEND_MAXLEN; //數據傳輸量
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外設非增量模式
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //存儲器增量模式
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; //外設數據長度:8位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //存儲器數據長度:8位
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //使用普通模式
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //中等優先級
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single; //存儲器突發單次傳輸
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single; //外設突發單次傳輸
DMA_Init(DMA2_Stream7, &DMA_InitStructure); //初始化DMA Stream
//DMA2_Stream7的NVIC配置
NVIC_InitStructure.NVIC_IRQChannel = DMA2_Stream7_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
DMA_ClearITPendingBit(DMA2_Stream7, DMA_IT_TCIF7); //清除DMA發送完成中斷標志
DMA_ITConfig(DMA2_Stream7, DMA_IT_TC, ENABLE); //使能DMA發送完成中斷
DMA_Cmd(DMA2_Stream7, ENABLE); //使能DMA2_Stream7
// - USART1接收DMA配置
mid_u16RetryCnt = 0;
DMA_DeInit(DMA2_Stream5);
while ((DMA_GetCmdStatus(DMA2_Stream5) != DISABLE) && (mid_u16RetryCnt++ < 500)); //等待DMA可配置
//配置DMA2_Stream5
DMA_InitStructure.DMA_Channel = DMA_Channel_4; //通道選擇
DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&USART1->DR; //DMA外設地址
DMA_InitStructure.DMA_Memory0BaseAddr = (u32)G_u8Usart1RecvBuf; //DMA 存儲器0地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory; //外設到存儲器模式
DMA_InitStructure.DMA_BufferSize = USART1_RECV_MAXLEN; //數據傳輸量
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外設非增量模式
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //存儲器增量模式
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; //外設數據長度:8位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //存儲器數據長度:8位
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //使用普通模式
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //中等優先級
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single; //存儲器突發單次傳輸
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single; //外設突發單次傳輸
DMA_Init(DMA2_Stream5, &DMA_InitStructure); //初始化DMA Stream
DMA_Cmd(DMA2_Stream5, ENABLE); //使能DMA2_Stream5
}
/*--------------------------------------------------------------------------------------
函數名稱:void USART_DMA_SendStart(DMA_Stream_TypeDef *DMA_Streamx, u16 m_u16SendCnt)
函數功能:串口USART1啟動一次DMA傳輸函數
入口參數:DMA_Stream_TypeDef DMA_Streamx - DMA數據流(DMA1_Stream0~7/DMA2_Stream0~7);
u16 m_u16SendCnt - 待傳輸數據字節數
出口參數:無
說 明:無
---------------------------------------------------------------------------------------*/
void USART_DMA_SendStart(DMA_Stream_TypeDef *DMA_Streamx, u16 m_u16SendCnt)
{
u16 l_u16RetryCnt = 0;
DMA_Cmd(DMA_Streamx, DISABLE); //關閉DMA傳輸
while ((DMA_GetCmdStatus(DMA_Streamx) != DISABLE) && (l_u16RetryCnt++ < 500)); //等待DMA可配置
DMA_SetCurrDataCounter(DMA_Streamx, m_u16SendCnt); //數據傳輸量
DMA_Cmd(DMA_Streamx, ENABLE); //開啟DMA傳輸
}
/*--------------------------------------------------------------------------------------
函數名稱:void USART1_DMA_SendNByte(u8 *m_pSendBuf, u16 m_u16SendCnt)
函數功能:串口USART1以DMA方式發送多字節函數
入口參數:u8 *m_pSendBuf - 待發送數據緩存, u16 m_u16SendCnt - 待發送數據個數
出口參數:無
說 明:無
---------------------------------------------------------------------------------------*/
void USART1_DMA_SendNByte(u8 *m_pSendBuf, u16 m_u16SendCnt)
{
memcpy(G_u8Usart1SendBuf, m_pSendBuf, m_u16SendCnt);
USART_DMA_SendStart(DMA2_Stream7, m_u16SendCnt); //啟動一次DMA傳輸
}
/*--------------------------------------------------------------------------------------
函數名稱:void DMA2_Stream7_IRQHandler(void)
函數功能:串口USART1以DMA方式發送完成中斷服務程序
入口參數:無
出口參數:無
說 明:無
---------------------------------------------------------------------------------------*/
void DMA2_Stream7_IRQHandler(void)
{
if(DMA_GetFlagStatus(DMA2_Stream7, DMA_FLAG_TCIF7) != RESET) //DMA發送完成?
{
DMA_ClearFlag(DMA2_Stream7, DMA_FLAG_TCIF7 | DMA_FLAG_FEIF7 |
DMA_FLAG_DMEIF7 | DMA_FLAG_TEIF7 | DMA_FLAG_HTIF7); //清除標志位
while(!USART_GetFlagStatus(USART1, USART_FLAG_TC)); //等待USART1發送完成標志TC置1
USART_ClearFlag(USART1, USART_FLAG_TC); //清除發送完成標志
RS485_Recv(); //切換為RS485接收模式
}
}
/*--------------------------------------------------------------------------------------
函數名稱:void USART1_IRQHandler(void)
函數功能:USART串口1中斷服務程序
入口參數:無
出口參數:無
說 明:無
---------------------------------------------------------------------------------------*/
void USART1_IRQHandler(void)
{
u16 l_u16Temp = 0;
if(USART_GetITStatus(USART1, USART_IT_IDLE) != RESET) //若有空閑中斷
{
DMA_Cmd(DMA2_Stream5, DISABLE); //關閉DMA2_Stream5,防止處理期間有數據
DMA_ClearFlag(DMA2_Stream5, DMA_FLAG_TCIF5 | DMA_FLAG_FEIF5 |
DMA_FLAG_DMEIF5 | DMA_FLAG_TEIF5 | DMA_FLAG_HTIF5); //清除標志位
//清除USART總線空閑中斷標志(只要讀USART1->SR和USART1->DR即可)
l_u16Temp = USART1->SR;
l_u16Temp = USART1->DR;
G_u16CommRecvLen = USART1_RECV_MAXLEN - DMA_GetCurrDataCounter(DMA2_Stream5); //求出接收到數據的字節數
if(G_u16CommRecvLen <= USART1_RECV_MAXLEN)
{
RS485_Send(); //RS485發送模式
USART1_DMA_SendNByte(G_u8Usart1RecvBuf, G_u16CommRecvLen); //回送接收到的數據
}
DMA_SetCurrDataCounter(DMA2_Stream5, USART1_RECV_MAXLEN); //設置傳輸數據長度
DMA_Cmd(DMA2_Stream5, ENABLE); //使能DMA2_Stream5
}
}
方法二:用USART1發送完成中斷
/*----------------------------------------------------------------------------------------------------
*Copyright: SXD Tech. Co., Ltd.
*開發 環境: Keil MDK 5.14 && STM32F407ZET6
*文件 名稱: USART串行通信驅動源文件
*作 者: 順信德
*版 本: V1.0
*日 期: 2018-2-6
*說 明:
*修改 日志: (1)
----------------------------------------------------------------------------------------------------*/
#include "Usart.h"
/*--------------------------------------------變量定義(S)--------------------------------------------*/
u32 G_u32RS485BaudRate = 9600; //RS485通訊波特率
u8 G_u8Usart1SendBuf[USART1_SEND_MAXLEN]={0,}; //發送數據緩沖區
u8 G_u8Usart1RecvBuf[USART1_RECV_MAXLEN]={0,}; //接收數據緩沖區
u16 G_u16CommRecvLen=0; //通訊接收的一幀數據長度
/*--------------------------------------------變量定義(E)--------------------------------------------*/
/*--------------------------------------------函數聲明(S)--------------------------------------------*/
void USART1_Init(u32 baud); //USART1串口初始化
void USART_DMA_SendStart(DMA_Stream_TypeDef *DMA_Streamx, u16 m_u16SendCnt); //串口USART1啟動一次DMA傳輸
void USART1_DMA_SendNByte(u8 *m_pSendBuf, u16 m_u16SendCnt); //串口USART1以DMA方式發送多字節
/*--------------------------------------------函數聲明(E)--------------------------------------------*/
/*----------------------------------------------------------------------------------------------------
*函數名稱:void USART1_Init(u32 baud)
*函數功能:USART1串口初始化函數
*入口參數:u32 baud - 波特率(單位bps)
*出口參數:無
*說 明:用于RS485通信;
----------------------------------------------------------------------------------------------------*/
void USART1_Init(u32 baud)
{
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
DMA_InitTypeDef DMA_InitStructure;
u16 mid_u16RetryCnt = 0;
USART_DeInit(USART1);
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); //使能GPIOA時鐘
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); //使能USART1時鐘
//USART1對應引腳復用映射
GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_USART1); //GPIOA9復用為USART1_TX
GPIO_PinAFConfig(GPIOA, GPIO_PinSource10, GPIO_AF_USART1); //GPIOA10復用為USART1_RX
//USART1端口配置
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10; //GPIOA9與GPIOA10
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; //復用功能
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_25MHz; //速度25MHz
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //推挽輸出
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; //上拉
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PA9,PA10
//USART1初始化設置
USART_InitStructure.USART_BaudRate = baud; //波特率設置
USART_InitStructure.USART_WordLength = USART_WordLength_8b; //字長為8位數據格式
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); //初始化USART1
USART_Cmd(USART1, ENABLE); //使能USART1
USART_ClearFlag(USART1, USART_FLAG_TC); //清除發送完成標志
while(USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET); //等待空閑幀發送完成后再清零發送完成標志
USART_ClearFlag(USART1, USART_FLAG_TC); //清除發送完成標志
USART_ITConfig(USART1, USART_IT_TC, ENABLE); //使能USART1傳輸完成中斷
USART_ITConfig(USART1, USART_IT_RXNE, DISABLE); //禁止USART1接收不為空中斷
USART_ITConfig(USART1, USART_IT_TXE, DISABLE); //禁止USART1發送空中斷
USART_ITConfig(USART1, USART_IT_IDLE, ENABLE); //開啟USART1空閑中斷
//USART1 NVIC配置
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; //串口1中斷通道
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3; //搶占優先級3
NVIC_InitStructure.NVIC_IRQChannelSubPriority =3; //子優先級3
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
NVIC_Init(&NVIC_InitStructure);
USART_DMACmd(USART1, USART_DMAReq_Tx, ENABLE); //使能串口1的DMA發送
USART_DMACmd(USART1, USART_DMAReq_Rx, ENABLE); //使能串口1的DMA接收
// - USART1發送DMA配置
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2,ENABLE); //DMA2時鐘使能
DMA_DeInit(DMA2_Stream7);
while ((DMA_GetCmdStatus(DMA2_Stream7) != DISABLE) && (mid_u16RetryCnt++ < 500)); //等待DMA可配置
//配置DMA2_Stream7
DMA_InitStructure.DMA_Channel = DMA_Channel_4; //通道選擇
DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&USART1->DR; //DMA外設地址
DMA_InitStructure.DMA_Memory0BaseAddr = (u32)G_u8Usart1SendBuf; //DMA 存儲器0地址
DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral; //存儲器到外設模式
DMA_InitStructure.DMA_BufferSize = USART1_SEND_MAXLEN; //數據傳輸量
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外設非增量模式
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //存儲器增量模式
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; //外設數據長度:8位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //存儲器數據長度:8位
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //使用普通模式
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //中等優先級
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single; //存儲器突發單次傳輸
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single; //外設突發單次傳輸
DMA_Init(DMA2_Stream7, &DMA_InitStructure); //初始化DMA Stream
DMA_Cmd(DMA2_Stream7, ENABLE); //使能DMA2_Stream7
// - USART1接收DMA配置
mid_u16RetryCnt = 0;
DMA_DeInit(DMA2_Stream5);
while ((DMA_GetCmdStatus(DMA2_Stream5) != DISABLE) && (mid_u16RetryCnt++ < 500)); //等待DMA可配置
//配置DMA2_Stream5
DMA_InitStructure.DMA_Channel = DMA_Channel_4; //通道選擇
DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&USART1->DR; //DMA外設地址
DMA_InitStructure.DMA_Memory0BaseAddr = (u32)G_u8Usart1RecvBuf; //DMA 存儲器0地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory; //外設到存儲器模式
DMA_InitStructure.DMA_BufferSize = USART1_RECV_MAXLEN; //數據傳輸量
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外設非增量模式
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //存儲器增量模式
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; //外設數據長度:8位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //存儲器數據長度:8位
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //使用普通模式
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //中等優先級
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single; //存儲器突發單次傳輸
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single; //外設突發單次傳輸
DMA_Init(DMA2_Stream5, &DMA_InitStructure); //初始化DMA Stream
DMA_Cmd(DMA2_Stream5, ENABLE); //使能DMA2_Stream5
}
/*--------------------------------------------------------------------------------------
函數名稱:void USART_DMA_SendStart(DMA_Stream_TypeDef *DMA_Streamx, u16 m_u16SendCnt)
函數功能:串口USART1啟動一次DMA傳輸函數
入口參數:DMA_Stream_TypeDef DMA_Streamx - DMA數據流(DMA1_Stream0~7/DMA2_Stream0~7);
u16 m_u16SendCnt - 待傳輸數據字節數
出口參數:無
說 明:無
---------------------------------------------------------------------------------------*/
void USART_DMA_SendStart(DMA_Stream_TypeDef *DMA_Streamx, u16 m_u16SendCnt)
{
u16 l_u16RetryCnt = 0;
DMA_Cmd(DMA_Streamx, DISABLE); //關閉DMA傳輸
while ((DMA_GetCmdStatus(DMA_Streamx) != DISABLE) && (l_u16RetryCnt++ < 500)); //等待DMA可配置
DMA_SetCurrDataCounter(DMA_Streamx, m_u16SendCnt); //數據傳輸量
DMA_Cmd(DMA_Streamx, ENABLE); //開啟DMA傳輸
}
/*--------------------------------------------------------------------------------------
函數名稱:void USART1_DMA_SendNByte(u8 *m_pSendBuf, u16 m_u16SendCnt)
函數功能:串口USART1以DMA方式發送多字節函數
入口參數:u8 *m_pSendBuf - 待發送數據緩存, u16 m_u16SendCnt - 待發送數據個數
出口參數:無
說 明:無
---------------------------------------------------------------------------------------*/
void USART1_DMA_SendNByte(u8 *m_pSendBuf, u16 m_u16SendCnt)
{
memcpy(G_u8Usart1SendBuf, m_pSendBuf, m_u16SendCnt);
USART_DMA_SendStart(DMA2_Stream7, m_u16SendCnt); //啟動一次DMA傳輸
}
/*--------------------------------------------------------------------------------------
函數名稱:void USART1_IRQHandler(void)
函數功能:USART串口1中斷服務程序
入口參數:無
出口參數:無
說 明:無
---------------------------------------------------------------------------------------*/
void USART1_IRQHandler(void)
{
u16 l_u16Temp = 0;
if(USART_GetITStatus(USART1, USART_IT_IDLE) != RESET) //若有空閑中斷
{
DMA_Cmd(DMA2_Stream5, DISABLE); //關閉DMA2_Stream5,防止處理期間有數據
DMA_ClearFlag(DMA2_Stream5, DMA_FLAG_TCIF5 | DMA_FLAG_FEIF5 |
DMA_FLAG_DMEIF5 | DMA_FLAG_TEIF5 | DMA_FLAG_HTIF5);//清零標志位
//清除USART總線空閑中斷標志(只要讀USART1->SR和USART1->DR即可)
l_u16Temp = USART1->SR;
l_u16Temp = USART1->DR;
G_u16CommRecvLen = USART1_RECV_MAXLEN - DMA_GetCurrDataCounter(DMA2_Stream5); //求出接收到數據的字節數
if(G_u16CommRecvLen <= USART1_RECV_MAXLEN)
{
RS485_Send(); //RS485發送模式
USART1_DMA_SendNByte(G_u8Usart1RecvBuf, G_u16CommRecvLen);
}
DMA_SetCurrDataCounter(DMA2_Stream5, USART1_RECV_MAXLEN); //設置傳輸數據長度
DMA_Cmd(DMA2_Stream5, ENABLE); //使能DMA2_Stream5
}
if(USART_GetITStatus(USART1, USART_IT_TC) != RESET) //若有發送完成中斷
{
USART_ClearITPendingBit(USART1, USART_IT_TC); //清除USART1發送完成中斷標志
DMA_ClearFlag(DMA2_Stream7, DMA_FLAG_TCIF7 | DMA_FLAG_FEIF7 |
DMA_FLAG_DMEIF7 | DMA_FLAG_TEIF7 | DMA_FLAG_HTIF7);//清零標志位
RS485_Recv(); //切換為RS485接收模式
}
}
6、聲明
本程序的收發機制只是簡單的處理機制,只是為了說明解決數據丟失字節問題的方法,對進行快速大數據通訊時會出現亂碼。所以,用于實際項目中,需對此程序的收發處理機制進行重新設計。兩種方法中,個人認為方法二更好,因為方法一在中斷里面等待白白耗費了時間。
上一篇:STM32串口接收使用DMA雙緩沖
下一篇:STM32串口通信USART(二)---DMA方式
推薦閱讀
史海拾趣
背景:在電子產品行業,產品品質和安全合規性至關重要。GardTec深知這一點,始終將產品品質放在首位。
發展:公司建立了嚴格的質量控制體系,確保每一件產品都符合最高的品質標準。同時,GardTec還積極遵循國際安全標準和法規要求,如REACH/RoHS等,確保產品的合規性。
影響:高品質的產品和嚴格的合規性要求,使得GardTec在電子行業中贏得了客戶的信賴和好評,進一步提升了公司的品牌形象和市場競爭力。
隨著技術的不斷成熟,DCX-CHOL Enterprises的產品逐漸在市場上樹立了良好的口碑。公司敏銳地捕捉到智能家居、物聯網等新興領域對電子產品的巨大需求,于是開始將產品線向這些領域拓展。通過與行業領軍企業的合作,DCX-CHOL Enterprises成功地將產品打入這些新興市場,市場份額逐年上升。同時,公司還積極開拓國際市場,將產品出口到多個國家和地區,進一步擴大了公司的知名度和影響力。
在市場競爭日益激烈的背景下,ADPOW公司深知品質是生存和發展的關鍵。因此,公司建立了嚴格的質量管理體系,從原材料采購到產品出廠的每一個環節都進行嚴格把控。同時,公司注重品牌建設,通過廣告宣傳、參加展會等方式提升品牌知名度和美譽度。這些舉措有效提升了公司的市場競爭力。
GSME Electronics深知品質是企業發展的生命線。因此,公司積極尋求并通過了ISO9001:2000質量管理體系、ISO14001環境管理體系以及IECQ QC080000危害物質流程管理體系等三項認證。同時,公司還遵循歐盟ROHS指令,通過了無鉛、汞、無公害認證,確保產品符合國際環保標準。這些舉措不僅提升了公司的市場競爭力,也贏得了國內外客戶的廣泛認可。
DBM REFLEX公司成立于本世紀初,初期以提供光學組件的小規模定制服務為主。創始人對光學技術的深刻理解和對市場的精準把握,使公司迅速在光學器件領域嶄露頭角。隨著技術的不斷積累和市場需求的增長,DBM REFLEX逐漸擴大了產品線,開始為LED市場設計、鑄模和生產高質量的光學器件。
集成電路是一種采用特殊工藝,將晶體管、電阻、電容等元件集成在硅基片上而形成的具有一定功能的器件,英文為縮寫為IC,也俗稱芯片。集成電路是六十年代出現的,當時只集成了十幾個元器件。 后來集成度越來越高,也有了今天的P-III。 集成電路根 ...… 查看全部問答∨ |
|
每日無線詞匯----射線跟蹤模型 Ray Tracing Model(zt) 類比:每天有成千上萬的人從北京出發去往全國各地,假若現在想知道每天有多少乘客從北京出發到上海。理論上我們只要把每天從北京到上海所有可能的交通工具包括飛機、火車、汽車所能運輸的人加起來便可以了。但是你有可能少考慮一部分人,他們可能跑 ...… 查看全部問答∨ |
|
在程序的前面用 m_hled=CreateFile(_T(\"CIS1:\"),GENERIC_READ|GENERIC_WRITE,0,NULL,OPEN_EXISTING,0,0);//打開驅動 ret = DeviceIoControl(m_hled,CAM_IOCTL_SAMSUNG_PREVIEW_START,NULL,NULL,NULL,NULL,NULL,NULL); 在開發板上LCD已經實時顯 ...… 查看全部問答∨ |
如題“ 求友善之臂SBC2410光盤資料包含vivi等內核源碼” 在線等待,急用........ 哪個兄弟有, 懇請您給我發一個 ,在下不勝感激, 資料內容比較大,所以請您發到我的郵箱:804763993@qq.com 請注明您的eeworld號 我給你再追加100分 謝謝.....… 查看全部問答∨ |
最近將wince6從 R2升級到了R3, 發現ie在打開網頁后出現Data Abort. 網頁還是能打開。但是CPU占用率變化很大。經常上升到%100. 從MS的官方論壇發現了打了R3 09年12月份的補丁是必然會出現Data Abort的, 但是我安裝的是R3沒有添加任何補丁的也會出現 ...… 查看全部問答∨ |
|
各位大俠 ,有誰知道指示表全自動檢定儀會用到MSP430的那一款單片機呀? 有知道的請聯系我0755-28168418 QQ:754181079… 查看全部問答∨ |