0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

STM32如何高效接收串口数据?

GReq_mcu168 来源:玩转单片机 作者:玩转单片机 2020-12-09 16:48 次阅读

硬件stm32f103cbt6
软件:STM32F10x_StdPeriph_Lib_V3.5.0

DMA,直接内存存取,可以用它的双手释放CPU的灵魂,所以,本文通过USART3进行串口收发,接受使用DMA的方式,无需CPU进行干预,当接受完成之后,数据可以直接从内存的缓冲区读取,从而减少了CPU的压力。

具体的代码实现如下:

usart_driver.h 封装了接口,数据接收回调函数类型,基本数据结构等;

usart_driver.c 函数原型实现,中断服务函数实现等;

拷贝这两个文件即可,可以根据目录下的参考用例,进行初始化。

头文件usart_driver.h已经声明了外部函数可能用到的接口;

USART3_DR的地址

因为USART3接收到数据会存在DR寄存器中,而DMA控制器则负责将该寄存器中的内容一一搬运到内存的缓冲区中(比如你定义的某个数组中),所以这里需要告诉DMA控制去哪里搬运,因此需要设置USART3_DR的总线地址。

USART3的基址如下图所示;

9037015a-2e2b-11eb-a64d-12bb97331649.png

USART3的基址

DR寄存器的偏移地址如下图所示;

9087af60-2e2b-11eb-a64d-12bb97331649.png

DR偏移地址

所以最终地址为:0x40004800 + 0x004#define USART_DR_Base 0x40004804

DMA的通道

因为有很多外设都可以使用DMA,比如ADCI2C,SPI等等,所以,不同的外设就要选择属于自己的DMA通道,查找参考手册;

90a75b8a-2e2b-11eb-a64d-12bb97331649.png

DMA通道

因此USART3_RX在这里会使用DMA1的通道3,这都是硬件上已经预先分配好的,我们需要遵循这个规则。所以在代码中我们做出相应的定义;如下所示;

#defineUSART_Rx_DMA_ChannelDMA1_Channel3

DMA的中断

DMA支持三种中断:传输过半,传输完成,传输出错;

90e44446-2e2b-11eb-a64d-12bb97331649.png

DMA中断

因此在使用是相当安全也相当灵活,而本文只是用了传输完成中断;如下定义了,传输完成中断的标志位,DMA1_FLAG_TC3也就对应了图中的TCIF;

#defineUSART_Rx_DMA_FLAGDMA1_FLAG_TC3

USART接收回调函数

在STM32的HAL中封装了大量外设的回调函数,使用起来十分方便,但是标准库中则没有这样的做法,但是这里我们可以自己实现,rx_cbk就是回调,即串口数据接收完成就会执行已经注册的回调函数;

typedefvoid(*rx_cbk)(void*args);

通过使用接口usart_set_rx_cbk进行回调函数的注册,pargs为将传递的参数指针;

voidusart_set_rx_cbk(uart_mod_t*pmod,rx_cbkpfunc,void*pargs);

头文件源码

#ifndefUSART_DRIVER_H #defineUSART_DRIVER_H #include #include /*Privatefunctionprototypes-----------------------------------------------*/ #defineUSE_MICROLIB_USART1 #ifUSE_MICROLIB_USART #ifdef__GNUC__ /*WithGCC/RAISONANCE,smallprintf(optionLDLinker->Libraries->Smallprintf setto'Yes')calls__io_putchar()*/ #definePUTCHAR_PROTOTYPEint__io_putchar(intch) #else #definePUTCHAR_PROTOTYPEintfputc(intch,FILE*f) //#defineGETCHAR_PROTOTYPEintfgetc(FILE*f) #endif/*__GNUC__*/ externPUTCHAR_PROTOTYPE; #else #endif //default8N1 #defineCOM_PORTUSART3 #defineTX_PINGPIO_Pin_10 #defineRX_PINGPIO_Pin_11 #defineBAUDRATE115200 #defineIRQ_UART_PRE3 #defineIRQ_UART_SUB3 #defineUSART_Rx_DMA_ChannelDMA1_Channel3 #defineUSART_Rx_DMA_FLAGDMA1_FLAG_TC3 #defineUSART_DR_Base0x40004804 #defineUSART_BUF_SIZE((uint16_t)16) typedefvoid(*rx_cbk)(void*args); structuart_mod{ uint8_trx_buf[USART_BUF_SIZE]; uint8_trx_dat_len; uint8_thead; uint8_ttail; void(*init)(void); void*pargs; rx_cbkpfunc_rx_cbk; }; typedefstructuart_moduart_mod_t; externuart_mod_tuser_uart_mod; voidusart_init(void); voidusart_set_rx_cbk(uart_mod_t*pmod,rx_cbkpfunc,void*pargs); voidusart_send_char(charch); voidusart_test_echo(void); uint8_tusart_recv_char(void); intusart_printf(constchar*fmt,...); //externGETCHAR_PROTOTYPE; #endif

DMA的基本配置

串口接收DMA的配置在函数dma_init中;

staticvoiddma_init(void)

已经定义了数据缓冲区,如下:

uint8_tRxBuffer[USART_BUF_SIZE]={0};

因此需要在DMA的配置中设置USART_DR的地址,和数据缓冲区的地址,以及两者的大小;还有就是数据流向;

寄存器流向内存;

内存流向寄存器;这个需要搞清楚;相关配置如下所示;

DMA_InitStructure.DMA_PeripheralBaseAddr=USART_DR_Base; DMA_InitStructure.DMA_MemoryBaseAddr=(uint32_t)RxBuffer; DMA_InitStructure.DMA_BufferSize=USART_BUF_SIZE; DMA_InitStructure.DMA_DIR=DMA_DIR_PeripheralSRC;

注意:DMA_DIR_PeripheralSRC表示,外设作为源地址,数据是从外设寄存器流向内存,即DMA会把数据从地址USART_DR_Base搬运到RxBuffer去。如果这个地方搞错,会导致RxBuffer始终没有你想要的数据。

环形队列接收数据

线性缓冲区会因为缓冲器接收数据已满导致无法继续接收的问题;而环形队列进行接收的话,会自动进行覆盖,这样一来,在读取数据的时候,也要配置一个环形队列进行数据处理,下面的配置是把DMA配置为循环模式;

DMA_InitStructure.DMA_Mode=DMA_Mode_Circular;

在结构体user_uart_mod中,则用两个变量分别指向队首head和队尾tail;具体数据的读取在函数USART3_IRQHandler中,会把数据从内存的RxBuffer读取到结构体user_uart_mod的成员变量rx_buf中;最终调用回调函数。

函数原型

usart_driver.c

#include #include #include"stm32f10x_usart.h" #include"usart_driver.h" uint8_tRxBuffer[USART_BUF_SIZE]={0}; uart_mod_tuser_uart_mod={ .rx_dat_len=0, .head=0, .tail=0, .pfunc_rx_cbk=NULL, .pargs=NULL }; staticUSART_InitTypeDefUSART_InitStructure; staticvoidrcc_init(void){ RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE); /*EnableGPIOclock*/ RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB |RCC_APB2Periph_AFIO,ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3,ENABLE); } staticvoidgpio_init(void){ GPIO_InitTypeDefGPIO_InitStructure; /*ConfigureUSARTTxasalternatefunctionpush-pull*/ GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Pin=TX_PIN; GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; GPIO_Init(GPIOB,&GPIO_InitStructure); /*ConfigureUSARTRxasinputfloating*/ GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING; GPIO_InitStructure.GPIO_Pin=RX_PIN; GPIO_Init(GPIOB,&GPIO_InitStructure); } staticvoiddma_init(void){ DMA_InitTypeDefDMA_InitStructure; /*USARTy_Tx_DMA_Channel(triggeredbyUSARTyTxevent)Config*/ DMA_DeInit(USART_Rx_DMA_Channel); DMA_InitStructure.DMA_PeripheralBaseAddr=USART_DR_Base; DMA_InitStructure.DMA_MemoryBaseAddr=(uint32_t)RxBuffer; //DMA_InitStructure.DMA_DIR=DMA_DIR_PeripheralDST; DMA_InitStructure.DMA_DIR=DMA_DIR_PeripheralSRC; DMA_InitStructure.DMA_BufferSize=USART_BUF_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(USART_Rx_DMA_Channel,&DMA_InitStructure); } staticvoidirq_init(void){ NVIC_InitTypeDefNVIC_InitStructure; /*EnabletheUSART3_IRQnInterrupt*/ NVIC_InitStructure.NVIC_IRQChannel=USART3_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=IRQ_UART_PRE; NVIC_InitStructure.NVIC_IRQChannelSubPriority=IRQ_UART_SUB; NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE; NVIC_Init(&NVIC_InitStructure); } voidusart_send_char(charch){ /*Loopuntiltheendoftransmission*/ //while(USART_GetFlagStatus(COM_PORT,USART_FLAG_TC)==RESET){} while((COM_PORT->SR&USART_FLAG_TC)!=USART_FLAG_TC){ } USART_SendData(COM_PORT,(uint8_t)ch); } uint8_tusart_recv_char(){ /*WaitthebyteisentirelyreceivedbyUSARTy*/ //while(USART_GetFlagStatus(COM_PORT,USART_FLAG_RXNE)==RESET){} while((COM_PORT->SR&USART_FLAG_RXNE)!=USART_FLAG_RXNE){ } /*StorethereceivedbyteintheRxBuffer1*/ return(uint8_t)USART_ReceiveData(COM_PORT); } intusart_printf(constchar*fmt,...) { uint8_ti=0; uint8_tusart_tx_buf[128]={0}; va_listap; va_start(ap,fmt); vsprintf((char*)usart_tx_buf,fmt,ap); va_end(ap); while(usart_tx_buf[i]&&i< 128){   usart_send_char(usart_tx_buf[i]);      i++;  }      usart_send_char('');  return 0; } void usart_test_echo(){  uint8_t tmp_dat = 0xff;  tmp_dat = usart_recv_char();  usart_send_char(tmp_dat); } void usart_init(void){  rcc_init ();  gpio_init ();  irq_init();    /* USARTx configured as follow:   - BaudRate = 115200 baud     - Word Length = 8 Bits   - One Stop Bit   - No parity   - Hardware flow control disabled (RTS and CTS signals)   - Receive and transmit enabled  */  USART_InitStructure.USART_BaudRate = BAUDRATE;  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 configuration */  USART_Init(COM_PORT, &USART_InitStructure);  USART_ITConfig(COM_PORT, USART_IT_IDLE, ENABLE);  //USART_ITConfig(COM_PORT, USART_IT_RXNE, ENABLE);  /* Enable USART */  USART_Cmd(COM_PORT, ENABLE);    USART_DMACmd(COM_PORT,USART_DMAReq_Rx, ENABLE);  dma_init();  DMA_ITConfig(USART_Rx_DMA_Channel, DMA_IT_TC, ENABLE);   DMA_ITConfig(USART_Rx_DMA_Channel, DMA_IT_TE, ENABLE);  DMA_Cmd(USART_Rx_DMA_Channel, ENABLE);  } void usart_set_rx_cbk(uart_mod_t *pmod, rx_cbk pfunc,void *pargs){  pmod->pargs=pargs; pmod->pfunc_rx_cbk=pfunc; } voidDMA1_Channel3_IRQHandler(void){ if(DMA_GetITStatus(USART_Rx_DMA_FLAG)==SET){ DMA_ClearITPendingBit(USART_Rx_DMA_FLAG); } } /** *@briefThisfunctionhandlesUSART3globalinterruptrequest. *@paramNone *@retvalNone */ voidUSART3_IRQHandler(void) { uint8_tbuf[USART_BUF_SIZE]; uint16_trect_len=0; if(USART_GetITStatus(COM_PORT,USART_IT_IDLE)!=RESET) { uint8_ti=0; USART_ReceiveData(COM_PORT); user_uart_mod.head=USART_BUF_SIZE-DMA_GetCurrDataCounter(USART_Rx_DMA_Channel); //fifoisnotfull while(user_uart_mod.head%USART_BUF_SIZE!=user_uart_mod.tail%USART_BUF_SIZE){ user_uart_mod.rx_buf[i++]=RxBuffer[user_uart_mod.tail++%USART_BUF_SIZE]; } user_uart_mod.rx_dat_len=i; //DMA_Cmd(USART_Rx_DMA_Channel,ENABLE); if(user_uart_mod.pfunc_rx_cbk!=NULL){ user_uart_mod.pfunc_rx_cbk(user_uart_mod.pargs); } } USART_ClearITPendingBit(COM_PORT,USART_IT_IDLE); //USART_ClearITPendingBit(COM_PORT,USART_IT_RXNE); } #ifUSE_MICROLIB_USART /** *@briefRetargetstheClibraryprintffunctiontotheUSART. *@paramNone *@retvalNone */ PUTCHAR_PROTOTYPE { /*Placeyourimplementationoffputchere*/ /*e.g.writeacharactertotheUSART*/ USART_SendData(COM_PORT,(uint8_t)ch); /*Loopuntiltheendoftransmission*/ while(USART_GetFlagStatus(COM_PORT,USART_FLAG_TC)==RESET) {} returnch; } #else #pragmaimport(__use_no_semihosting) struct__FILE { inthandle; }; FILE__stdout; int_sys_exit(intx) { x=x; return0; } intfputc(intch,FILE*f) { /*Placeyourimplementationoffputchere*/ /*e.g.writeacharactertotheUSART*/ USART_SendData(COM_PORT,(uint8_t)ch); /*Loopuntiltheendoftransmission*/ while(USART_GetFlagStatus(COM_PORT,USART_FLAG_TC)==RESET) {} returnch; } #endif

参考用例

这里需要调用usart_init,并设置回调函数,如果不设置,则不会执行回调。

voidmotor_get_cmd_from_uart(void*pargs){ if(pargs==NULL){ return; } uart_mod_t*p=pargs; if(p->rx_dat_len>0&&p->rx_dat_len==PACKAGE_SIZE){ if(p->rx_buf[0]==PACKAGE_HEAD &&p->rx_buf[PACKAGE_SIZE-1]==PACKAGE_TAIL){ user_cmd_mod.head=p->rx_buf[0]; user_cmd_mod.cmd.value_n[0]=p->rx_buf[1]; user_cmd_mod.cmd.value_n[1]=p->rx_buf[2]; user_cmd_mod.option=p->rx_buf[3]; user_cmd_mod.data.value_n[0]=p->rx_buf[4]; user_cmd_mod.data.value_n[1]=p->rx_buf[5]; user_cmd_mod.data.value_n[2]=p->rx_buf[6]; user_cmd_mod.data.value_n[3]=p->rx_buf[7]; user_cmd_mod.tail=p->rx_buf[PACKAGE_SIZE-1]; user_cmd_mod.process_flag=1; } } p->rx_dat_len=0; } intmain(void){ usart_init(); usart_set_rx_cbk(&user_uart_mod,motor_get_cmd_from_uart,&user_uart_mod); }

总结

本文简单介绍了基于STM32基于DMA,利用串口空闲中断进行串口数据接收的具体配置和实现方法,代码基于标准库3.5版本;
因为标准库ST目前已经不再更新,并且ST提供了cubemx工具可以进行基于HAL库和LL库的外设快速配置,从而简化大量工作;当然为了不掉头发感觉撸寄存器也不错,最终适合自己的才是最好的。


责任编辑:lq

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

    关注

    68

    文章

    10824

    浏览量

    211110
  • STM32
    +关注

    关注

    2265

    文章

    10870

    浏览量

    354729
  • 串口
    +关注

    关注

    14

    文章

    1543

    浏览量

    76168

原文标题:STM32如何高效接收串口数据?

文章出处:【微信号:mcu168,微信公众号:硬件攻城狮】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    【代码分享】基于乐鑫ESP32的串口不定长数据接收方法

    【代码分享】基于乐鑫ESP32的串口不定长数据接收方法
    的头像 发表于 11-15 01:02 131次阅读
    【代码分享】基于乐鑫ESP32的<b class='flag-5'>串口</b>不定长<b class='flag-5'>数据</b><b class='flag-5'>接收</b>方法

    STM32G030F6用串口中断函数接收数据,发送数据就死机怎么解决?

    平台介绍: 芯片是使用的STM32G030F6,系统是rt-thread nano-v3.1.5, 使用rtthread studio + cubemx生成工程项目 问题描述:想使用串口中断方式去
    发表于 07-11 06:44

    求助,使用STM32G030C8T6的串口接收问题求解

    目前使用STM32G030的芯片做了一个485接收从机,由主机STM32F407的芯片发送数据去询问从机当前数据,但是在使用过程中遇见了一个
    发表于 05-28 06:11

    STM32F0xx_HAL_Driver库的串口接收数据个数,是不是只能写成1,一个一个数据接收

    在使用STM32F0xx_HAL_Driver库进行串口接收数据 HAL_StatusTypeDef HAL_UART_Receive( UART_HandleTypeDef *hu
    发表于 05-14 06:39

    stm32 usb转串口,android接收有数据丢失要如何处理?

    我用pc上的串门专家可以准确的收取到stm32通过usb转串口发送的数据,但是在android上用apk接收时会有数据的丢失。这是什么问题啊
    发表于 05-13 08:45

    单片机在串口发送数据时可以接收串口助手发的数据,为什么只能接收两字节?

    最近做串口通信,单片机在串口发送数据时可以接收串口助手发的数据,不过只能
    发表于 05-08 07:52

    STM32 hal库无法接收串口数据是怎么回事?

    型号:STM32L432KC官方开发板 串口使用的是板载的USB下载口以及串口 用法: 在串口初始化后,while(1)之前调用 接收中断
    发表于 04-29 06:11

    Python怎么读取STM32串口数据

    =ser.readlines()print(s) 可是什么都读取不了。如果用买的STM32开发板送的串口助手,能接收数据。板子上烧录的printf输出程序。 请问各位大神,Pytho
    发表于 04-24 07:30

    Freertos利用STM32 HAL做串口接收,中断突然失效的原因?

    Freertos 利用STM32 HAL做串口接收,发送数据正常,没有通过中断处理,接收通过中断接收
    发表于 04-07 07:21

    stm32串口接收中断触发原理

    配置串口接收中断使能:在初始化串口时,需要设置相应的控制寄存器来使能串口接收中断。这通常可以通过设置相应的标志位或使用特定的寄存器位来完成。
    发表于 01-17 15:42 7855次阅读
    <b class='flag-5'>stm32</b><b class='flag-5'>串口</b><b class='flag-5'>接收</b>中断触发原理

    labview串口接收数据怎么不丢帧

    LabVIEW是一款功能强大的图形化开发环境,广泛应用于数据采集与处理、自动化控制系统等领域。在串口通信应用中,如何保证数据的稳定传输是一个重要的问题。本文将详细介绍LabVIEW串口
    的头像 发表于 01-08 11:38 2018次阅读

    stm32怎么读取串口发来的指令

    读取串口发来的指令是嵌入式系统中一项常见的任务,特别是在与外部设备进行通信时。在STM32系列微控制器中,提供了多个串口接口(USART、UART等),可以用于读取和处理串口
    的头像 发表于 01-07 17:08 2606次阅读

    单片机串口通信的接收与发送

    的原理。串口通信是通过发送和接收两根线来实现的,分别为发送线(Tx)和接收线(Rx)。当单片机发送数据时,数据通过发送线发送出去,而当外部设
    的头像 发表于 12-20 14:03 3481次阅读

    stm32虚拟串口接收数据处理

    STM32中使用虚拟串口时,我们需要处理接收数据接收到的数据可能是来自其他设备的传感器
    的头像 发表于 12-20 11:22 2178次阅读

    智能车ROS与STM32串口通信代码

    Stm32_Serial; // 声明串口对象 并且在类的定义中,声明两个结构体,用来存储接收和要发送的数据 RECEIVE_DATA Receive_Data ; //The se
    的头像 发表于 11-26 17:47 1128次阅读