“顶嵌杯”决赛第2题公布

题目名称:模拟Modbus协议

Description

在嵌入式系统开发中,Modbus协议是工业控制系统中广泛应用的一种协议。本题用来简单模拟Modbus协议,只需根据条件生成符合该协议的数据帧,并解析所获取的数据。
假设设备使用的协议发送数据格式如下:

<SlaveAddress, 1 Byte> <Function, 1 Byte> <Start Address, 2 Bytes> <NumberofBytes, 2 Bytes> <Checksum, 2 Bytes>
其中前四项将在输入条件中给出,最后一项为CRC校验和,需根据前四项的数据,按照CRC算法进行计算。注意数据的长度,多于1byte的高位在前,低位在后。该CRC校验算法的描述如下:

1)
CRC赋值0xFFFF

2)
取初始信息的第一个字节(8位)与CRC进行异或运算,将结果赋给CRC

3)
CRC数据右移一位,最前位(左边)补0

4)
如果右移前,CRC最低位(最右端)为1,则将右移后的CRC0xA001进行异或运算,且将结果赋给CRC。否则,跳过此步。

5)
重复348次(即右边8位)。

6)
对初始信息的下一个字节,同样执行2345步,直到信息中所有字节都执行了同样的步骤。

7)
将此时得到的CRC值的高8位和低8位交换,即得到CRC校验和。


对应的接收格式如下:

<SlaveAddress,1Byte> <Function,1Byte> <NumberofBytes,1Byte> <DataIEEE32,xByte> <Checksum,2Bytes>
其中DataIEEE32为一个或多个按IEEE754标准定义的32位浮点数,具体的数据长度由NumberofBytes项来决定(比如NumberofBytes4,则DataIEEE32项为4 bytes,正好表示一个浮点数;如为8,则DataIEEE32项为8 bytes,可表示两个浮点数)。本题要求编程实现从IEEE32数据(如“420B 999A ”)到浮点数(如34.9)的转换,从而解析出浮点数值。


提示:你可以根据IEEE754标准自行设计转换算法;或者直接利用C语言float类型的实现特性:x86 linux下,gcc编译器将C语言代码“float f = 34.9;”编译成汇编代码“movl $0x420b 999a , -4(%ebp)” AT&T x86汇编格式),也就是说,单精度浮点数34.9在内存中就是由整数0x420b 999a 来表示的,你可以利用这一特性来完成转换。

Input

输入包含多组数据,以EOF结束
每组数据共两行。

第一行共四个十进制整数,分别为协议格式要求的:<SlaveAddress, 1 Byte>,<Function, 1 Byte>,<Start Address, 2 Bytes>,<NumberofBytes, 2 Bytes>,以逗号“,”分开。

如:
1,4,40,2
其中:1SlaveAddress4Function40Start Address2NumberofBytes

第二行为符合接收格式的数据帧(16进制表示),需从其中解析所接收的数据,其长度小于64个字符,浮点数数据最多为4个(即DataIEEE32数据项最多为32bytes)。

: 010404420B 999A 7405
其中:01SlaveAddress04Function04NumberofBytes 420B 999A DataIEEE327405Checksum

Output

每组数据输出共两行。
第一行:根据输入结果的第一行,输出完整的符合该协议发送格式的数据帧,数据用16进制大写表示,每部分的长度都要求符合协议格式,比如Start Address项如果不到2 bytes,则需要在左边补零。

如: 010400280002F 1C
3
其中:01SlaveAddress04 Function0028Start Address0002NumberofBytesF 1C 3Checksum


第二行:根据输入结果的第二行,依次解析IEEE32数据,将其转换成浮点数并打印结果(小数点后保留一位)。解析之前需检查CRC校验和,如校验失败则直接打印CRC_ERROR。如有多个数据,用逗号分隔。

如:
34.9
该浮点值为420B 999A 所对应的值。

Sample Input

1,4,40,2 

010404420B 999A 7405 

1,4,40,2

010404420B 999A 7404

2,4,383,4

02040841CC 0000477F 2100DF85

 

 

Sample Output

010400280002F 1C 3

34.9

010400280002F 1C 3

CRC_ERROR

0204017F 0004C 1DE

25.5,65313.0

Source  (参考代码)

#include <stdio.h>

#include <string.h>

 

#define MAX_RECEIVE_LEN 65

 

int char2int(char c)

{

       if(c >= '0' && c <= '9') return c - '0';

       return c - 'A' + 10;

}

 

unsigned short cal_crc(unsigned char* p,int len)

{

       unsigned short ret = 0xFFFF;

       int i = 0,k = 0;

       for(;i < len;++i)

       {

              ret ^= p[i];

              for(k = 0;k < 8;++k)

              {

                     ret = (ret&0x01)?((ret>>1)^0xA001):(ret>>1);

              }

       }

       ret = ((ret&0x00FF)<<8)|((ret&0xFF00)>>8);

       return ret;

}

 

int main()

{

       int saveadd = 0,func = 0,startadd = 0,num = 0,i = 0,len = 0,k = 0;

       unsigned short val = 0;

       unsigned char receive[MAX_RECEIVE_LEN];

       unsigned char receivetmp[MAX_RECEIVE_LEN];

       unsigned char str[MAX_RECEIVE_LEN];

       unsigned char fstr[4];

       float * pf = NULL;

       while(EOF != scanf("%d,%d,%d,%d",&saveadd,&func,&startadd,&num))

       {

              str[0] = saveadd;str[1] = func;

              str[3] = startadd&0xFF;startadd >>= 8;str[2] = startadd&0xFF;

              str[5] = num&0xFF;num >>= 8;str[4] = num&0xFF;

              val = cal_crc(str,6);

              str[7] = val&0xFF;val >>= 8;str[6] = val&0xFF;

              for(i = 0;i < 8;++i) printf("%02X",str[i]);

              printf("/n");

              scanf("%s",receivetmp);

              len = strlen((char*)receivetmp)/2;

              for(i = 0;i < len;++i)

              {

                     receive[i] = (unsigned char)(char2int(receivetmp[i*2])*16+char2int(receivetmp[i*2+1]));

              }

              val = cal_crc(receive,len-2);

              if((val&0xFF) != receive[len-1]||((val>>8)&0xFF) != receive[len-2]) { printf("CRC_ERROR/n");continue; }

              k = 2;num = receive[k]/4;++k;

              for(i = 0;i < num;++i)

              {

                     fstr[0] = receive[k+3];fstr[1] = receive[k+2];

                     fstr[2] = receive[k+1];fstr[3] = receive[k];

                     pf = (float*)(fstr);

                     printf("% .1f ",*pf);

                     if(i != num - 1) printf(",");

                     k += 4;

              }

              if(0 != num) printf("/n");

       }

       return 0;

}

 

注:该代码来自参赛选手北京瑞星研发工程师曾剑杰,被“顶嵌杯”专家评审组公认为符合工程化的高质量代码。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值