女人自慰AV免费观看内涵网,日韩国产剧情在线观看网址,神马电影网特片网,最新一级电影欧美,在线观看亚洲欧美日韩,黄色视频在线播放免费观看,ABO涨奶期羡澄,第一导航fulione,美女主播操b

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]&&ipargs=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

    文章

    11049

    瀏覽量

    216151
  • STM32
    +關注

    關注

    2290

    文章

    11018

    瀏覽量

    362705
  • 串口
    +關注

    關注

    14

    文章

    1586

    瀏覽量

    79051

原文標題:STM32如何高效接收串口數據?

文章出處:【微信號:mcu168,微信公眾號:硬件攻城獅】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦
    熱點推薦

    STM32F407驕陽電機版用DMA雙緩存接收串口數據時,上電第一次接收區是memory1而不是memory0?為什么?

    STM32 F407驕陽電機版用DMA雙緩存接收串口數據時,上電第一次接收區是memory1而不是memory0?
    發表于 06-12 07:15

    【RA-Eco-RA4M2開發板評測】使用RA4M2的串口實現任意任意類型任意長度的數據接收,并將接收到的數據顯示在串口助手上

    使用RA4M2的串口實現任意任意類型任意長度的數據接收,并將接收到的任意類型任意長度的數據顯示在串口
    發表于 05-04 14:11

    STM32串口下載軟件(FLYMCU)

    STM32串口下載軟件(FLYMCU),經典版本,親試可用。
    發表于 04-09 15:59 ?2次下載

    STM32F411RE NUCLEO UART串口通信無法接收是什么原因引起的?

    STM32F411RE NUCLEO UART串口通信無法接收
    發表于 03-13 08:00

    STM32F427串口接收和發送中斷同時使能,為什么會出現接收中斷丟數的情況?

    STM32F427芯片,針對UART7開啟串口接收緩存區非空中斷RXNE和串口傳輸完成中斷TC. 1.單測試收發都沒有任何問題。 2.將串口
    發表于 03-11 07:05

    STM32H7打開DCache后,出現了串口接收信息為空的現象,是哪里出了問題?

    我首先打開了串口一的DMA接受數據,是可以的,接下來配置LWIP網絡,出現了串口接收信息為空的現象,然后我逐步排查,發現了在打開DCache后串口
    發表于 03-10 08:25

    【代碼分享】基于樂鑫ESP32的串口不定長數據接收方法

    【代碼分享】基于樂鑫ESP32的串口不定長數據接收方法
    的頭像 發表于 11-15 01:02 ?1346次閱讀
    【代碼分享】基于樂鑫ESP32的<b class='flag-5'>串口</b>不定長<b class='flag-5'>數據</b><b class='flag-5'>接收</b>方法

    STM32CUBEMX(8)--USART通過定時器中斷方式接收不定長數據

    概述 本文利用中斷實現串口不定長接收(非DMA),使用HAL庫,將接收數據打印出去。 DMA接收請查看: https://blog.csd
    發表于 09-06 16:48

    STM32CUBEMX(6)--移植雅特力AT32F403AVGT7,雙串口通過DMA方式接收不定長數據

    每個外設都需要實現自己的本地數據存儲)相比,DMA解決方案在硅片成本和功耗方面的成本較低。 根據使用的產品型號的不同,有一個或兩個DMA模塊。 本篇文章主要介紹如何使用STM32CubeMX實現串口
    發表于 09-06 16:37

    STM32CUBEMX(2)--USART通過DMA方式接收不定長數據

    (\"數據內容:\"); for(int i=0;i<Rx_len;i++) { printf(\"%c\",ReceiveBuff);//向串口打印接收
    發表于 09-04 11:48

    stm32串口燒錄怎么設置

    準備工作 確保您擁有STM32開發板和相應的硬件設備,如USB轉串口模塊。 安裝STM32CubeMX和STM32CubeProgrammer軟件,這些是ST官方提供的工具,用于配置和
    的頭像 發表于 08-22 09:33 ?3062次閱讀

    19.4-STM32接收數據-狀態顯示在屏幕 openMV尋跡與小車控制 Openmv+STM32F103C8T6視覺巡線小車

    19.4-STM32接收數據-狀態顯示在屏幕 openMV尋跡與小車控制 Openmv+STM32F103C8T6視覺巡線小車
    的頭像 發表于 08-20 11:12 ?1531次閱讀
    19.4-<b class='flag-5'>STM32</b><b class='flag-5'>接收</b><b class='flag-5'>數據</b>-狀態顯示在屏幕 openMV尋跡與小車控制 Openmv+<b class='flag-5'>STM32</b>F103C8T6視覺巡線小車

    STM32G030F6用串口中斷函數接收數據,發送數據就死機怎么解決?

    平臺介紹: 芯片是使用的STM32G030F6,系統是rt-thread nano-v3.1.5, 使用rtthread studio + cubemx生成工程項目 問題描述:想使用串口中斷方式去
    發表于 07-11 06:44

    STM32L0使用stop模式下,9600波特率,低功耗串口接收數據,發現偶爾丟失第一個字節,什么原因?

    請給一個STM32L0系列的低功耗串口的在stop模式下工作的例程。 我們在使用stop模式下,9600波特率,低功耗串口接收數據,發現偶爾
    發表于 07-05 07:54

    ESP32串口接收和發送數據,會有延遲是怎么回事?

    測試環境:串口波特率600bps,串口接收256字節數據。 在調試串口時,發現發送多于120字節時,串口
    發表于 06-26 08:07