STM32与宇电设备实现AI-BUS通讯

宇电的设备使用基于RS-485的自定义协议,协议本身比较简单,只有2条指令:

读:地址代号+52H82 +要读的参数代号+0+0+校验码

写:地址代号+43H67+要写的参数代号+写入数低字节+写入数高字节+校验码

校验码采用 16 位求和校验方式,其中读指令的校验码计算方法为:要读参数的代号×256+82+ADDR。

写指令的校验码计算方法为以下公式做 16 位二进制加法计算得出的余数(溢出部分不处理):要写的参数代号×256+67+要写的参数值+ADDR。

返回的数据格式更是固定的,无论是读还是写,仪表都返回以下10个字节数据:测量值 PV+给定值 SV+输出值 MV 及报警状态+所读/写参数值+校验码。

其中 PV、 SV 及所读参数值均各占 2 个字节,代表一个 16 位二进制有符号补码整数,低位字节在前,高位字节在后,整数无法表示小数点,要求用户在上位机处理; MV 占一个字节,按 8 位有符号二进制数格式,数值范围-110~+110,状态位占一个字节,校验码占 2 个字节,共 10 个字节。

而返回的校验码计算则是:PV+SV+(报警状态*256+MV)+参数值+ADDR。清楚协议的这些规则后,编写程序只是顺理成章的事。直接上代码:

/*读取目标设备的参数值*/
void ReadAiBusDeviceParameter(uint8_t deviceAddr,uint8_t paraAddr,void (*AiBusSendByte)(uint8_t *,uint16_t))
{
  uint8_t readCommand[INSTRUCTION_LENGTH];
  uint16_t index=0;
  readCommand[index++]=0x80+deviceAddr;
  readCommand[index++]=0x80+deviceAddr;
  readCommand[index++]=READ_INSTRUCTION;
  readCommand[index++]=paraAddr;
  readCommand[index++]=0x0;
  readCommand[index++]=0x0;

  uint16_t checkSum=(uint16_t)paraAddr*256+READ_INSTRUCTION+(uint16_t)deviceAddr;

  readCommand[index++]=checkSum;
  readCommand[index++]=(checkSum>>8);

  AiBusSendByte(readCommand,INSTRUCTION_LENGTH);
}

/*设置目标设备的参数值*/
void WriteAiBusDeviceParameter(uint8_t deviceAddr,uint8_t paraAddr,uint16_t data,void (*AiBusSendByte)(uint8_t *,uint16_t))
{
  uint8_t writeCommand[INSTRUCTION_LENGTH];
  uint16_t index=0;
  writeCommand[index++]=0x80+deviceAddr;
  writeCommand[index++]=0x80+deviceAddr;
  writeCommand[index++]=WRITE_INSTRUCTION;
  writeCommand[index++]=paraAddr;
  writeCommand[index++]=data;
  writeCommand[index++]=(data>>8);

  uint16_t checkSum=(uint16_t)paraAddr*256+WRITE_INSTRUCTION+(uint16_t)deviceAddr+data;

  writeCommand[index++]=checkSum;
  writeCommand[index++]=(checkSum>>8);

  AiBusSendByte(writeCommand,INSTRUCTION_LENGTH);
}

/*解析返回数据,返回值为读或者写的参数值*/
int ParsingReturnData(uint8_t *receiveData,uint16_t *returnData,uint8_t *deviceAddr,uint16_t deviceNum)
{
  int status=-1;

  uint16_t pValue=0;
  uint16_t sValue=0;
  uint16_t mValue=0;
  uint16_t alarmStatus=0;
  uint16_t paraValue=0;
  uint16_t checkSum=0;

  pValue=receiveData[0]+receiveData[1]*256;
  sValue=receiveData[2]+receiveData[3]*256;
  mValue=(uint16_t)receiveData[4];
  alarmStatus=(uint16_t)receiveData[5];
  paraValue=receiveData[6]+receiveData[7]*256;
  checkSum=receiveData[8]+receiveData[9]*256;

  uint16_t chk=pValue+sValue+alarmStatus*256+mValue+paraValue;

  for(int i=0;i<deviceNum;i++)
  {
    if(checkSum==chk+deviceAddr[i])
    {
      status=i;
      returnData[0]=pValue;
      returnData[1]=sValue;
      returnData[2]=mValue;
      returnData[3]=alarmStatus;
      returnData[4]=paraValue;
      break;
    }
  }
  return status;
}

欢迎关注:

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值