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

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

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

FIFO隊(duì)列的串口數(shù)據(jù)收發(fā)方法

FPGA之家 ? 來源:FPGA之家 ? 作者:FPGA之家 ? 2022-07-12 09:16 ? 次閱讀

01

前言

嵌入式軟件的開發(fā)中,串口是十分常用且基礎(chǔ)的功能。在需要批量發(fā)送數(shù)據(jù)的場合,可以使用while循環(huán)等待發(fā)送完成標(biāo)志位的方式,但是這種方式會占據(jù)主循環(huán),影響效率。也可以采用dma的方式,但是dma在發(fā)送數(shù)據(jù)時(shí)非常高效,但是批量接收數(shù)據(jù)時(shí),就很不靈活,特別是一些在串口數(shù)據(jù)中解析某種協(xié)議格式時(shí),很不方便。下面介紹一種利用串口中斷結(jié)合FIFO隊(duì)列的串口數(shù)據(jù)收發(fā)方法,結(jié)合了不阻塞批量發(fā)與靈活接收的優(yōu)點(diǎn),特別適用于串口協(xié)議收發(fā)的使用場景。

02

FIFO隊(duì)列

FIFO是英文First In First Out 的縮寫,是一種先進(jìn)先出的數(shù)據(jù)緩存器,順序?qū)懭霐?shù)據(jù),順序的讀出數(shù)據(jù),其數(shù)據(jù)地址由內(nèi)部讀寫指針自動加1完成。相比于一個(gè)同等緩存大小的數(shù)值,F(xiàn)IFO就是多管理了一個(gè)先進(jìn)先出的功能,方便串口數(shù)據(jù)的存入和讀出。

Fifo在帶操作系統(tǒng)嵌入式軟件中都有現(xiàn)成的實(shí)現(xiàn),但是在基礎(chǔ)的嵌入軟件中,我們可以自己實(shí)現(xiàn)一個(gè)。

#define UART1_IN_FIFO_SIZE   100 //接收串口隊(duì)列的深度#define UART1_OUT_FIFO_SIZE 250 //發(fā)送串口隊(duì)列的深度
//頭文件函數(shù)列表FIFO_EXT u8 uart1infifo_data[UART1_IN_FIFO_SIZE];#define uart1infifo_count (uart1infifo_GetCount())FIFO_EXT u16 uart1infifo_front;FIFO_EXT u16 uart1infifo_rear;FIFO_EXT void uart1infifo_Clear(void);FIFO_EXT void uart1infifo_DataIn(u8 d);FIFO_EXT u8 uart1infifo_DataOut(void);FIFO_EXT u16 uart1infifo_GetSpace(void);FIFO_EXT u16 uart1infifo_GetCount(void);
//獲取串口1接收隊(duì)列緩存數(shù)u16 uart1infifo_GetCount(void){  u16 countR,countF;  countR = uart1infifo_rear;  countF = uart1infifo_front;  if (countR >=  countF)  {    return(countR - countF);  }  else  {    return(UART1_IN_FIFO_SIZE + countR - countF);  }}//清空串口1接收隊(duì)列void uart1infifo_Clear(void){  uart1infifo_front = UART1_IN_FIFO_SIZE -1;  uart1infifo_rear = uart1infifo_front;//  uart1infifo_count = 0;}//串口1接收隊(duì)列入數(shù)據(jù)void uart1infifo_DataIn(u8 d){  if (uart1infifo_count < UART1_IN_FIFO_SIZE)  {    uart1infifo_rear = (uart1infifo_rear +1) % UART1_IN_FIFO_SIZE;      uart1infifo_data[uart1infifo_rear] = d;  }}//串口1接收隊(duì)列出數(shù)據(jù)u8 uart1infifo_DataOut(void){  if (uart1infifo_rear !=  uart1infifo_front)  {    uart1infifo_front = (uart1infifo_front +1) % UART1_IN_FIFO_SIZE;    return(uart1infifo_data[uart1infifo_front]);  }  else  {    return(0xff);  }}

為了節(jié)省篇幅,串口1發(fā)送隊(duì)列就不詳細(xì)描述了,在接收隊(duì)列的基礎(chǔ)上稍加修改即可。

03

中斷收發(fā)串口

//串口發(fā)送函數(shù) void SendDataToUart1(u8 * pData, u16 len){  u8i;  //串口發(fā)送隊(duì)列將慢,等待一下數(shù)據(jù)發(fā)送  while(1){  if(uart1outfifo_GetSpace()>len+5)  {        break;      }      else      {        i = 0;      }    }    USART_ITConfig(USART1, USART_IT_TXE, DISABLE);  //關(guān)閉中斷,防止隊(duì)列的進(jìn)出會同時(shí)進(jìn)行    while (len --)    {          uart1outfifo_DataIn(*pData);      pData ++;    }      USART_ITConfig(USART1, USART_IT_TXE, ENABLE);}
//串口處理函數(shù)void USART1_IRQHandler(void){  if (USART_GetFlagStatus(USART1, USART_FLAG_RXNE))  {    uart1infifo_DataIn(USART_ReceiveData(USART1));//接收數(shù)據(jù)并放入串口接收隊(duì)列  //串口數(shù)據(jù)處理flag  }  else if (USART_GetFlagStatus(USART1, USART_FLAG_TXE))  {    if (uart1outfifo_count > 0)    {      USART_SendData(USART1, uart1outfifo_DataOut());//發(fā)隊(duì)列取出數(shù)據(jù)放入串口發(fā)送寄存器}else {      USART_ITConfig(USART1, USART_IT_TXE, DISABLE);    }  }}

04

串口數(shù)據(jù)處理

不定長數(shù)據(jù)包超時(shí)處理

在上節(jié)的“串口數(shù)據(jù)處理flag”處,加入超時(shí)的標(biāo)記g_uartTimeOut = n;并在定時(shí)器中斷中倒計(jì)時(shí)g_uartTimeOut,減到0后,產(chǎn)生數(shù)據(jù)包處理標(biāo)志gb_needDealUartPkg = 1。主循環(huán)掃到gb_needDealUartPkg是1后,讀出uart1infifo中的全部數(shù)據(jù)進(jìn)行解包處理。

不定長數(shù)據(jù)包按內(nèi)容格式處理

?在上節(jié)的“串口數(shù)據(jù)處理flag”處,加入比對數(shù)據(jù)包格式的函數(shù),當(dāng)格式滿足要求時(shí),將整個(gè)數(shù)據(jù)包存入數(shù)據(jù)包隊(duì)列(參照前面的串口數(shù)據(jù)接收函數(shù),寫一個(gè)接收隊(duì)列,接收的數(shù)據(jù)為數(shù)據(jù)包結(jié)構(gòu)體)。主循環(huán)掃描數(shù)據(jù)包隊(duì)列的緩存數(shù),有就去處理。

定長數(shù)據(jù)包處理

主循環(huán)中掃描uart1infifo_count,當(dāng)達(dá)到定長后,讀出uart1infifo中的定長數(shù)據(jù)進(jìn)行解包處理。

原文標(biāo)題:嵌入式軟件中的串口收發(fā)隊(duì)列設(shè)計(jì)方法

文章出處:【微信公眾號:FPGA之家】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

審核編輯:彭靜

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

    關(guān)注

    37

    文章

    6687

    瀏覽量

    123140
  • fifo
    +關(guān)注

    關(guān)注

    3

    文章

    386

    瀏覽量

    43496
  • 串口數(shù)據(jù)
    +關(guān)注

    關(guān)注

    0

    文章

    33

    瀏覽量

    13858

原文標(biāo)題:嵌入式軟件中的串口收發(fā)隊(duì)列設(shè)計(jì)方法

文章出處:【微信號:zhuyandz,微信公眾號:FPGA之家】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    tlv320aic3016配置好以后然后打開中斷允許,沒有錄音數(shù)據(jù)的輸入,到隊(duì)列fifo的數(shù)是0嗎?

    配置好以后然后打開中斷允許如果沒有錄音數(shù)據(jù)的輸入,那么到隊(duì)列fifo的數(shù)是0嗎,level的值會變化嗎?
    發(fā)表于 10-28 07:11

    嵌入式環(huán)形隊(duì)列與消息隊(duì)列的實(shí)現(xiàn)原理

    嵌入式環(huán)形隊(duì)列,也稱為環(huán)形緩沖區(qū)或循環(huán)隊(duì)列,是一種先進(jìn)先出(FIFO)的數(shù)據(jù)結(jié)構(gòu),用于在固定大小的存儲區(qū)域中高效地存儲和訪問數(shù)據(jù)。其主要特點(diǎn)
    的頭像 發(fā)表于 09-02 15:29 ?293次閱讀

    玩轉(zhuǎn)RT-Thread之消息隊(duì)列的應(yīng)用

    在嵌入式系統(tǒng)開發(fā)中,實(shí)時(shí)處理串口和ADC數(shù)據(jù)是一項(xiàng)重要的任務(wù)。本文將介紹如何在RT-Thread實(shí)時(shí)操作系統(tǒng)中,利用消息隊(duì)列來同時(shí)處理來自串口和ADC的
    的頭像 發(fā)表于 07-23 08:11 ?521次閱讀
    玩轉(zhuǎn)RT-Thread之消息<b class='flag-5'>隊(duì)列</b>的應(yīng)用

    EPS32-C3使用USB串口收發(fā)數(shù)據(jù),在發(fā)送長度為64的數(shù)據(jù)時(shí),上位機(jī)串口工具無法接收到數(shù)據(jù)怎么解決?

    EPS32-C3 使用USB串口收發(fā)數(shù)據(jù) 結(jié)果發(fā)現(xiàn)使用int usb_serial_jtag_write_bytes(const void* src, size_t size, TickType_t
    發(fā)表于 06-06 06:12

    同步FIFO和異步FIFO區(qū)別介紹

    1. FIFO簡介 FIFO是一種先進(jìn)先出數(shù)據(jù)緩存器,它與普通存儲器的區(qū)別是沒有外部讀寫地址線,使用起來非常簡單,缺點(diǎn)是只能順序讀寫,而不能隨機(jī)讀寫。 2. 使用場景 數(shù)據(jù)緩沖:也就是
    的頭像 發(fā)表于 06-04 14:27 ?1203次閱讀
    同步<b class='flag-5'>FIFO</b>和異步<b class='flag-5'>FIFO</b>區(qū)別介紹

    STM32串口1同時(shí)收發(fā)數(shù)據(jù)卡在回調(diào)函數(shù)中怎么解決?

    只能正常收發(fā)幾次,卡住后進(jìn)入不了user_main_program 同時(shí)使用了串口2連接esp8266進(jìn)行數(shù)據(jù)收發(fā)
    發(fā)表于 05-29 06:34

    freertos串口接收數(shù)據(jù)后如何發(fā)送給任務(wù)?

    正在學(xué)freertos。串口中斷接收一幀數(shù)據(jù)后,放到數(shù)組里,如何將數(shù)據(jù)發(fā)送給任務(wù)呢? 如果用消息隊(duì)列,是否建立的消息隊(duì)列需要是數(shù)組類型的?
    發(fā)表于 04-18 06:36

    TC397 asclin做串口收發(fā)數(shù)據(jù)有亂碼怎么解決?

    在做串口測試時(shí),將rxfifo的中斷模式設(shè)置成BATCH模式,發(fā)現(xiàn)每次進(jìn)入中斷后從FIFO中讀出的數(shù)據(jù)偶爾會有亂碼,請問各位大神又遇到過類似問題的嗎
    發(fā)表于 02-06 08:27

    TC1728如何用FIFO發(fā)送和接收數(shù)據(jù),如何配置和調(diào)用?

    配置好并不能收到數(shù)據(jù),請問前面如何配置才能收發(fā)數(shù)據(jù)? 如果要用FIFO方式收發(fā)一個(gè)報(bào)文如 Data1[8] ={0x00
    發(fā)表于 02-04 09:33

    串口中斷函數(shù)中,接收的數(shù)據(jù)量已經(jīng)超過了FIFO的長度,會不會造成數(shù)據(jù)丟失呀?

    如果在串口中斷函數(shù)中,正在讀緩沖區(qū)的中的數(shù)據(jù)(還沒有讀完),這個(gè)時(shí)候,串口又接收到新的數(shù)據(jù),接收的數(shù)據(jù)量已經(jīng)超過了
    發(fā)表于 01-17 08:14

    如何用DMA的方式讀寫串口數(shù)據(jù)?

    串口接收(或者發(fā)送)的數(shù)據(jù)比較大,如果用中斷逐字節(jié)從FIFO數(shù)據(jù)太耗時(shí)了,光盤里的找不到有關(guān)DMA讀寫串口的demo
    發(fā)表于 01-17 07:35

    labview 隊(duì)列最前端插入的應(yīng)用

    LabVIEW是一種用于實(shí)時(shí)測試、測量和控制系統(tǒng)的高級系統(tǒng)設(shè)計(jì)軟件。它采用了數(shù)據(jù)流編程方式,提供了一種直觀、可視化的方法來構(gòu)建復(fù)雜的測試和測量應(yīng)用程序。其中一個(gè)重要的功能是隊(duì)列,它可以在軟件設(shè)計(jì)中
    的頭像 發(fā)表于 01-08 11:45 ?1110次閱讀

    labview怎么清除串口緩存的數(shù)據(jù)

    LabVIEW 是一款功能強(qiáng)大的圖形化編程軟件,常用于控制、監(jiān)測和數(shù)據(jù)采集等應(yīng)用。當(dāng)我們使用串口進(jìn)行數(shù)據(jù)通信時(shí),有時(shí)會遇到串口緩存的數(shù)據(jù)無法
    的頭像 發(fā)表于 01-08 11:30 ?3391次閱讀

    labview隊(duì)列有什么實(shí)際作用

    LabVIEW隊(duì)列是一種數(shù)據(jù)結(jié)構(gòu),常用于解決多任務(wù)并發(fā)處理的問題。它被廣泛應(yīng)用于科學(xué)研究、工程項(xiàng)目和自動化控制等領(lǐng)域。在LabVIEW中,隊(duì)列提供了一種高效、方便的方式來處理不同任務(wù)之間的數(shù)據(jù)
    的頭像 發(fā)表于 01-05 16:42 ?1447次閱讀

    RT-Thread Nano入門:串口接收與消息隊(duì)列

    本文主要介紹怎么用RT-Thread Nano的消息隊(duì)列方式實(shí)現(xiàn)串口數(shù)據(jù)接收,結(jié)合串口接收中斷和空閑中斷,接收上位機(jī)發(fā)來的一幀數(shù)據(jù)。
    的頭像 發(fā)表于 11-22 11:07 ?3708次閱讀
    RT-Thread Nano入門:<b class='flag-5'>串口</b>接收與消息<b class='flag-5'>隊(duì)列</b>