找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1108|回复: 4
收起左侧

关于单片机RS485程序的问题

[复制链接]
回帖奖励 3 黑币 回复本帖可获得 3 黑币奖励! 每人限 1 次
ID:253579 发表于 2022-2-19 11:31 | 显示全部楼层 |阅读模式
关于手把手教51的教程里,关于485那章程序有个疑问

这个串口驱动函数里 len = UartRead(buf, sizeof(buf)-2); //将接收到的命令读取到缓冲区中这行为什么要把sizeof(buf)-2呢


/* 串口驱动函数,监测数据帧的接收,调度功能函数,需在主循环中调用 */
void UartDriver()
{
    unsigned char len;
    unsigned char pdata buf[40];

    if (flagFrame) //有命令到达时,读取处理该命令
    {
        flagFrame = 0;
        len = UartRead(buf, sizeof(buf)-2); //将接收到的命令读取到缓冲区中
        UartAction(buf, len);  //传递数据帧,调用动作执行函数
    }
}


以下是单片机原代码
  1. #include <reg52.h>
  2. #include <intrins.h>

  3. sbit RS485_DIR = P1^7;  //RS485方向选择引脚

  4. bit flagFrame = 0;  //帧接收完成标志,即接收到一帧新数据
  5. bit flagTxd = 0;    //单字节发送完成标志,用来替代TXD中断标志位
  6. unsigned char cntRxd = 0;   //接收字节计数器
  7. unsigned char pdata bufRxd[64];  //接收字节缓冲区

  8. extern void UartAction(unsigned char *buf, unsigned char len);

  9. /* 串口配置函数,baud-通信波特率 */
  10. void ConfigUART(unsigned int baud)
  11. {
  12.     RS485_DIR = 0; //RS485设置为接收方向
  13.     SCON  = 0x50;  //配置串口为模式1
  14.     TMOD &= 0x0F;  //清零T1的控制位
  15.     TMOD |= 0x20;  //配置T1为模式2
  16.     TH1 = 256 - (11059200/12/32)/baud;  //计算T1重载值
  17.     TL1 = TH1;     //初值等于重载值
  18.     ET1 = 0;       //禁止T1中断
  19.     ES  = 1;       //使能串口中断
  20.     TR1 = 1;       //启动T1
  21. }
  22. /* 软件延时函数,延时时间(t*10)us */
  23. void DelayX10us(unsigned char t)
  24. {
  25.     do {
  26.         _nop_();
  27.         _nop_();
  28.         _nop_();
  29.         _nop_();
  30.         _nop_();
  31.         _nop_();
  32.         _nop_();
  33.         _nop_();
  34.     } while (--t);
  35. }
  36. /* 串口数据写入,即串口发送函数,buf-待发送数据的指针,len-指定的发送长度 */
  37. void UartWrite(unsigned char *buf, unsigned char len)
  38. {
  39.     RS485_DIR = 1;  //RS485设置为发送
  40.     while (len--)   //循环发送所有字节
  41.     {
  42.         flagTxd = 0;      //清零发送标志
  43.         SBUF = *buf++;    //发送一个字节数据
  44.         while (!flagTxd); //等待该字节发送完成
  45.     }
  46.     DelayX10us(5);  //等待最后的停止位完成,延时时间由波特率决定
  47.     RS485_DIR = 0;  //RS485设置为接收
  48. }
  49. /* 串口数据读取函数,buf-接收指针,len-指定的读取长度,返回值-实际读到的长度 */
  50. unsigned char UartRead(unsigned char *buf, unsigned char len)
  51. {
  52.     unsigned char i;

  53.     if (len > cntRxd)  //指定读取长度大于实际接收到的数据长度时,
  54.     {                  //读取长度设置为实际接收到的数据长度
  55.         len = cntRxd;
  56.     }
  57.     for (i=0; i<len; i++)  //拷贝接收到的数据到接收指针上
  58.     {
  59.         *buf++ = bufRxd[i];
  60.     }
  61.     cntRxd = 0;  //接收计数器清零

  62.     return len;  //返回实际读取长度
  63. }
  64. /* 串口接收监控,由空闲时间判定帧结束,需在定时中断中调用,ms-定时间隔 */
  65. void UartRxMonitor(unsigned char ms)
  66. {
  67.     static unsigned char cntbkp = 0;
  68.     static unsigned char idletmr = 0;

  69.     if (cntRxd > 0)  //接收计数器大于零时,监控总线空闲时间
  70.     {
  71.         if (cntbkp != cntRxd)  //接收计数器改变,即刚接收到数据时,清零空闲计时
  72.         {
  73.             cntbkp = cntRxd;
  74.             idletmr = 0;
  75.         }
  76.         else                   //接收计数器未改变,即总线空闲时,累积空闲时间
  77.         {
  78.             if (idletmr < 30)  //空闲计时小于30ms时,持续累加
  79.             {
  80.                 idletmr += ms;
  81.                 if (idletmr >= 30)  //空闲时间达到30ms时,即判定为一帧接收完毕
  82.                 {
  83.                     flagFrame = 1;  //设置帧接收完成标志
  84.                 }
  85.             }
  86.         }
  87.     }
  88.     else
  89.     {
  90.         cntbkp = 0;
  91.     }
  92. }
  93. /* 串口驱动函数,监测数据帧的接收,调度功能函数,需在主循环中调用 */
  94. void UartDriver()
  95. {
  96.     unsigned char len;
  97.     unsigned char pdata buf[40];

  98.     if (flagFrame) //有命令到达时,读取处理该命令
  99.     {
  100.         flagFrame = 0;
  101.         len = UartRead(buf, sizeof(buf)-2); //将接收到的命令读取到缓冲区中
  102.         UartAction(buf, len);  //传递数据帧,调用动作执行函数
  103.     }
  104. }
  105. /* 串口中断服务函数 */
  106. void InterruptUART() interrupt 4
  107. {
  108.     if (RI)  //接收到新字节
  109.     {
  110.         RI = 0;  //清零接收中断标志位
  111.         if (cntRxd < sizeof(bufRxd)) //接收缓冲区尚未用完时,
  112.         {                            //保存接收字节,并递增计数器
  113.             bufRxd[cntRxd++] = SBUF;
  114.         }
  115.     }
  116.     if (TI)  //字节发送完毕
  117.     {
  118.         TI = 0;   //清零发送中断标志位
  119.         flagTxd = 1;  //设置字节发送完成标志
  120.     }
  121. }

复制代码

回复

使用道具 举报

ID:161164 发表于 2022-2-19 15:17 | 显示全部楼层
因为Modbus的尾2Byte是CRC验证用的
回复

使用道具 举报

ID:1003077 发表于 2022-2-19 19:54 | 显示全部楼层
就是读回来数据不做CRC校验,2个字节不用
回复

使用道具 举报

ID:293363 发表于 2022-2-19 21:34 | 显示全部楼层
就是不对数据进行CRC校验
回复

使用道具 举报

ID:1003226 发表于 2022-2-23 10:37 | 显示全部楼层
crc校验的关系
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表