当前位置:首页 > 公众号精选 > 嵌入式云IOT技术圈
[导读]原版Marlin固件硬件平台基于arduino,采用C++类对串口操作函数函数进行了封装,代码注释中介绍了这些函数的功能。

原版Marlin固件硬件平台基于arduino,采用C++类对串口操作函数函数进行了封装,代码注释中介绍了这些函数的功能。MarlinSerial.h文件中类的定义,此处的类只保留的框架结构,留存的这些函数基本上是要一直到STM32平台要实现的函数。

class MarlinSerial //: public Stream
{
  public:
    MarlinSerial();
    void begin(long); //串口初始化设置,配置串口波特率
    void end();       //禁止串口传输函数
    int peek(void);   //读串口缓存中下一字节的数据(字符型),但不从内部缓存中删除该数据。
    int read(void);   //读取串口数据,一次读一个字符,读完后删除已读数据
    void flush(void); //等待输出数据传送完毕
    int available(void);//返回的是缓冲区准确的可读字节数
    void checkRx(void)
};
extern MarlinSerial MSerial; //外部声明,实例化一个串口对象MSerial

MarlinSerial.cpp文件中定义了具体函数的实现方式,通过实例化的对象便可以操作这些串口函数 。

循环队列简介

该串口操作函数用到了数据结构中循环队列的算法,下面先介绍一下循环队列:

//定义队列 #define MaxSize  50  //定义队列中元素的最大个数 typedef struct
{
    int  data[MaxSize]; //存放队列元素
    int front, rear; //队头指针和队尾指针
}SqQueue

把存储队列元素的表从逻辑上看成一个环,称为循环队列。当队首指针Q.font = MaxSize-1后再前进一个位置就会自动到0,这就可以利用除法取余运算来实现。

具体循环队列的实现请参考数据结构 循环队列部分。(后面整理这一部分)

为什么要在串口接收部分创建环形缓冲区?

(引用)串口数据处理机制是数据接收并原样回发的机制是:成功接收到一个数据,触发进入中断, 在中断函数中将数据读取出来,然后立即处理。这一种数据处理机制是“非缓冲中断方式”,虽然这种数 据处理方式不消耗时间,但是这种数据处理方式严重的缺点是:数据无缓冲区,如果先前接收的的 数据如果尚未发送完成(处理完成),然后串口又接收到新的数据,新接收的数据就会把尚未处理 的数据覆盖,从而导致“数据丢包”。串口接收部分创建环形缓冲区便可以很好的避免因收发速度不 一致产生的数据丢包。

串口缓冲区的实现

接下来具体分析下Marlin串口缓冲区的实现(下面分析的代码为移植到STM32上的实现代码,原理一致。):

.h头文件

#define RX_BUFFER_SIZE 128   //定义串口缓冲区的大小 //定义环形缓冲区结构体
typerdef struct
{
  unsigned char buffer[RX_BUFFER_SIZE];  //存放接收到的字符
  int head;  //队头指针
  int tail;    //队尾指针
}ring_buffer;

注意:这里的头和尾的定义恰与循环队列里面的头和尾定义相反,在理解上将head当作rear,将tail当作front即可

.c文件

ring_buffer rx_buffer  =  { { 0 }, 0, 0 }; //定义结构体类型的接收缓冲区并初始化
void store_char(unsigned char c)  //将接收到的数据存入缓冲区
{
  int i = (unsigned int)(rx_buffer.head + 1) % RX_BUFFER_SIZE;
  //如果我们应该存储的接收到的字符的位置刚好在尾端的前面
  //(意味着头部将要进入尾端的当前位置),这样将会溢出缓冲区,
  //因此我们不该存入这个字符或使这个头前进 if (i != rx_buffer.tail)  //缓冲区没有存满
  {
    rx_buffer.buffer[rx_buffer.head] = c;
    rx_buffer.head = i;
  }
}
unsigned int MSerial_available(void)  //返回串口缓存区中数据的个数
{ return (unsigned int)(RX_BUFFER_SIZE + rx_buffer.head

   - rx_buffer.tail) % RX_BUFFER_SIZE;
  }

uint8_t MSerial_peek(void)
{ if (rx_buffer.head == rx_buffer.tail)
    { return 0;
    } else { return rx_buffer.buffer[rx_buffer.tail];
    }
}

uint8_t Mserial_read(void)  //按存入顺序逐个读取缓冲区的数据
{
  uint8_t c;
   /*如果头不是在尾的前面,将收不到任何字符*/ if (rx_buffer.head == rx_buffer.tail)
 { return 0;
  } else {
    c = rx_buffer.buffer[rx_buffer.tail];
    rx_buffer.tail = (unsigned int)(rx_buffer.tail + 1) % RX_BUFFER_SIZE; return c;
  }
}
void MSerial_flush(void)  //等待串口数据传送完毕
{
  // RX
 //不要颠倒这个否则可能会有一些问题,如果接收中断发生在读
 //取rx_buffer_head之后但在写入rx_buffer_tail之前
 //之前的rx_buffer_head值可能被写到rx_buffer_tail
 //使它呈现缓冲区是满的而非空的状态*/
  rx_buffer.head = rx_buffer.tail;
}

后面还有有什么不太理解,可以检索“循环队列” 、“串口环形缓冲区”等关键字来增进理解。

免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭