STM32 UART配置中断DMA传输

电子说

1.2w人已加入

描述

在处理UART数据的时候,DMA是一种非常灵活、高效的方式。

※补记:USART_DR 串口数据寄存器是一个双寄存器,包含了TDR和RDR,对它读操作,读取的是RDR寄存器的值,对它的写操作,实际上是写到TDR寄存器;当向该寄存器写数据的时候,串口就会自动发送,当收到收据的时候,也是存在该寄存器内。

数据存储器

数据存储器

关于读写:

__STATIC_INLINE uint8_t LL_USART_ReceiveData8(USART_TypeDef *USARTx);/*
读取接收寄存器USARTx_DR中8位数据,接收即所得。返回值最大为0xFF
若使能校验位,接收的最高位MSB将由硬件自动进行校验。
*/
__STATIC_INLINE uint16_t LL_USART_ReceiveData9(USART_TypeDef *USARTx);/*
读取接收寄存器USARTx_DR中9位数据(当字节长9,接收即所得)。返回值最大为0x1FF
若使能校验位,接收的最高位MSB将由硬件自动进行校验。
*/
__STATIC_INLINE void LL_USART_TransmitData9(USART_TypeDef *USARTx, uint16_t Value);/*
向发送寄存器写入9位数据。当使能校验位,发送时最高位MSB自动替换成校验值
*/
__STATIC_INLINE void LL_USART_TransmitData8(USART_TypeDef *USARTx, uint8_t Value)/*
向发送寄存器写入8位数据。当使能校验位,发送时最高位MSB自动替换成校验值
*/

·

API:

__STATIC_INLINE void LL_USART_EnableDMAReq_RX(USART_TypeDef *USARTx);/*
使能接收DMA,启用后DR有数据时将允许发送DMA请求;具体见示例用法
*/
__STATIC_INLINE void LL_USART_DisableDMAReq_RX(USART_TypeDef *USARTx);/*
禁用接收DMA
*/
__STATIC_INLINE uint32_t LL_USART_IsEnabledDMAReq_RX(USART_TypeDef *USARTx);/*
检测是否使能接收DMA
*/

__STATIC_INLINE void LL_USART_EnableDMAReq_TX(USART_TypeDef *USARTx);/*
使能发送DMA
*/
__STATIC_INLINE void LL_USART_DisableDMAReq_TX(USART_TypeDef *USARTx);/*
禁用发送DMA
*/
__STATIC_INLINE uint32_t LL_USART_IsEnabledDMAReq_TX(USART_TypeDef *USARTx);/*
检测是否使能发送DMA
*/
/**************************************************/
__STATIC_INLINE uint32_t LL_USART_DMA_GetRegAddr(USART_TypeDef *USARTx);/*
返回UART模块数据寄存器DR地址;无论是否启用DMA均可用
*/

/ 结力期末考分界线 */

配置使用DMA收发:

1.在CubeMX中:

数据存储器

数据存储器

数据存储器

+在项目设置中调整为使用LL

2.生成代码

此时在生成代码已实现了串口DMA的初始化设置并生成了相应的DMA中断句柄。接下来通过代码实现功能。示例为一个简单的功能,将串口接收到的数据再通过串口发出。

3.配置

①在main.h中定义全局变量test_data:

uint8_t test_data;

②在usart.c中引入变量

extern uint8_t test_data;

③在usart.c中进行设置

※通道配置为非循环模式时,传输结束后(即传输计数变为0)将不再产生DMA操作。要开始新的DMA传输,需要3个步骤:在关闭DMA通道的情况下,在DMA_CNDTRx寄存器中重新写入传输数目(有需要则需重新配置地址),然后重新开启DMA。

//RX DMA配置
 LL_DMA_SetPeriphAddress(DMA1, LL_DMA_CHANNEL_5,LL_USART_DMA_GetRegAddr(USART1));//连接外设寄存器USART1- >DR
 LL_DMA_SetMemoryAddress(DMA1, LL_DMA_CHANNEL_5, (uint32_t)&test_data);//连接数据存储地址
 LL_DMA_SetDataLength(DMA1, LL_DMA_CHANNEL_5, 1);/*设置传输的数据长度,由于是一字节一传所以此处为1,
 若数据为n字节则会在接收到的字节数量达到n的时候,才传输结束。*/
 LL_USART_EnableDMAReq_RX(USART1);//使能RX接收DMA
 LL_DMA_EnableIT_TC(DMA1, LL_DMA_CHANNEL_5);//使能DMA通道的传输完成中断功能;当传输量达到数据长度将引发中断
 LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_5);//开启DMA传输,此时若RX读取到数据写入DR将传输至MemoryAddress
 //
 //TX DMA配置
 LL_DMA_SetPeriphAddress(DMA1, LL_DMA_CHANNEL_4, LL_USART_DMA_GetRegAddr(USART1)); //连接外设寄存器USART1- >DR
 LL_DMA_SetMemoryAddress(DMA1, LL_DMA_CHANNEL_4, (uint32_t)&test_data); //连接数据存储地址
 LL_DMA_EnableIT_TC(DMA1, LL_DMA_CHANNEL_4); //使能DMA中断
 LL_USART_EnableDMAReq_TX(USART1); //使能TX接收DMA
 
  /* USER CODE END USART1_Init 1 */

④中断处理函数:

void DMA1_Channel5_IRQHandler(void)
{   //RX中断调用
  /* USER CODE BEGIN DMA1_Channel5_IRQn 0 */
 
    if(LL_DMA_IsActiveFlag_TC5(DMA1))//判断是否由DMA传输完成引发中断
 {
  LL_DMA_DisableChannel(DMA1, LL_DMA_CHANNEL_5);//关闭通道,以在之后开启新的DMA传输
  LL_DMA_SetMemoryAddress(DMA1, LL_DMA_CHANNEL_4, (uint32_t)&test_data);  //TX连接数据存储地址
  LL_DMA_SetDataLength(DMA1, LL_DMA_CHANNEL_4, 1);//设置TX数据长度
  LL_DMA_ClearFlag_TC5(DMA1);//清除中断标志,使能下一次RX中断
  
  LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_4);//从数据位置向TX启动一次DMA;传输完成后调用TX的DMA中断
 }
  /* USER CODE END DMA1_Channel5_IRQn 0 */

}
void DMA1_Channel4_IRQHandler(void)
{ //TX中断调用
  /* USER CODE BEGIN DMA1_Channel4_IRQn 0 */
    if(LL_DMA_IsActiveFlag_TC4(DMA1))//判断是否由DMA传输完成引发中断
 {
  LL_DMA_ClearFlag_TC4(DMA1); //清除中断标志
  LL_DMA_DisableChannel(DMA1, LL_DMA_CHANNEL_4); //关闭通道,以在之后开启新的DMA传输
  LL_DMA_SetMemoryAddress(DMA1, LL_DMA_CHANNEL_5, (uint32_t)&test_data);   //RX连接数据存储地址
  LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_5);//再次启动RX DMA功能
 }
  
  /* USER CODE END DMA1_Channel4_IRQn 0 */
}

以上代码即可实现功能。

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分