问题:

在嵌入式系统开发中,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,则将右移后的CRC与0xA001进行异或运算,且将结果赋给CRC。否则,跳过此步。
5)重复3,4步8次(即右边8位)。
6)对初始信息的下一个字节,同样执行2,3,4,5步,直到信息中所有字节都执行了同样的步骤。
7)将此时得到的CRC值的高8位和低8位交换,即得到CRC校验和。

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

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

Input
输入包含多组数据,以EOF结束
每组数据共两行。
第一行共四个十进制整数,分别为协议格式要求的:<SlaveAddress, 1 Byte>,<Function, 1 Byte>,<Start Address, 2 Bytes>,<NumberofBytes, 2 Bytes>,以逗号“,”分开。
如:1,4,40,2
其中:1为SlaveAddress;4为Function;40为Start Address;2为NumberofBytes。
第二行为符合接收格式的数据帧(16进制表示),需从其中解析所接收的数据,其长度小于64个字符,浮点数数据最多为4个(即DataIEEE32数据项最多为32bytes)。
如: 010404420B999A7405
其中:01为SlaveAddress;04为Function;04为NumberofBytes; 420B999A 为DataIEEE32;7405为Checksum。

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

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

Sample Input
1,4,40,2  
010404420B999A7405  
1,4,40,2
010404420B999A7404
2,4,383,4
02040841CC0000477F2100DF85

Sample Output
010400280002F1C3
34.9
010400280002F1C3
CRC_ERROR
0204017F0004C1DE
25.5,65313.0

回答:

#include <string.h>

char strout[100], recv[100], recv_hd[100];
unsigned short slaveAdd, func, nb, startAdd;
unsigned short crc;
float result[5];
char *pt;

do_cyc(unsigned short data)
{
    int i;
    unsigned short tmp;

crc = data ^ crc;
    for (i=0 ; i<8 ; i++)
    {
        tmp = crc & (0x0001);
        crc = crc >> 1;
        if (tmp == 1)
        {
            crc = 0xa001 ^ crc;
        }
    }
}

unsigned short get_crc()
{
    crc = 0xffff;

do_cyc(slaveAdd & 0x00ff);
    do_cyc(func & 0x00ff);
    do_cyc((startAdd & 0xff00) >> 8);
    do_cyc(startAdd & 0x00ff);
    do_cyc((nb & 0xff00) >> 8);
    do_cyc(nb & 0x00ff);

unsigned short swap;
    swap = crc;
    crc = (crc >> 8) & 0x00ff;
    crc = crc | ((swap << 8) & 0xff00);
}

float conv(int data)
{
    float ret;
    memcpy(&ret, &data, sizeof(int));
    return ret;
}

void pre_handle()
{
    int i, data;
    pt = recv;
    crc = 0xffff;

strncpy(recv_hd, pt, 2);
    recv_hd[2] = '\0';
    sscanf(recv_hd, "%hX", &slaveAdd);
    pt += 2;
    do_cyc(slaveAdd & 0x00ff);

strncpy(recv_hd, pt, 2);
    recv_hd[2] = '\0';
    sscanf(recv_hd, "%hX", &func);
    pt += 2;
    do_cyc(func & 0x00ff);

strncpy(recv_hd, pt, 2);
    recv_hd[2] = '\0';
    sscanf(recv_hd, "%hX", &nb);
    pt += 2;
    do_cyc(nb & 0x00ff);

for (i=0 ; i<nb/4 ; i++)
    {
        strncpy(recv_hd, pt, 8);
        sscanf(recv_hd, "%X", &data);
        result[i] = conv(data);
        do_cyc((unsigned short)((data & 0xff000000) >> 24));
        do_cyc((unsigned short)((data & 0x00ff0000) >> 16));
        do_cyc((unsigned short)((data & 0x0000ff00) >> 8));
        do_cyc((unsigned short)(data & 0x000000ff));
        pt += 8;
    }
    unsigned short swap;
    swap = crc;
    crc = (crc >> 8) & 0x00ff;
    crc = crc | ((swap << 8) & 0xff00);
}

int main(int argc, char *argv[])
{
    char cmp[10];

while (EOF != scanf("%hd,%hd,%hd,%hd", &slaveAdd, &func, &startAdd, &nb))
    {
        get_crc();
        sprintf(strout, "%02X%02X%04X%04X%04X", slaveAdd, func, startAdd, nb, crc);
        printf("%s\n", strout);
        scanf("%s", recv);
        pre_handle(recv);
        sprintf(cmp, "%hX", crc);
        if (0 != strcmp(cmp, pt))
        {
            printf("CRC_ERROR\n");
            continue;
        }
        int i;
        for (i=0 ; i<nb/4-1 ; i++)
            printf("%.1f,", result[i]);
        printf("%.1f\n", result[i]);
    }
}

模拟Modbus协议问题的更多相关文章

  1. Socket编程之聊天程序 - 模拟Fins/ModBus协议通信过程

    设备控制软件编程涉及到的基本通信方式主要有TCP/IP与串口,用到的数据通信协议有Fins与ModBus. 更高级别的通信如.net中的Remoting与WCF在进行C/S架构软件开发时会采用. 本篇 ...

  2. 基于AVR128单纯Modbus协议实施

    Modbus通信协议Modicon公司1979在发展中,适用于工业现场总线协议控制.Modbus通信系统包含芯片的节点,并与组合物可编程控制的公共传输线,它的目的是收集和监视多个节点的数据.Modbu ...

  3. RS485通信和Modbus协议(转)

    转自:http://www.51hei.com/bbs/dpj-23230-1.html 在工业控制.电力通讯.智能仪表等领域,通常情况下是采用串口通信的方式进行数据交换.最初采用的方式是RS232接 ...

  4. 工控安全入门(一)—— Modbus协议

    modbus基础知识 modbus协议最初是由Modicon公司在1971年推出的全球第一款真正意义上用于工业现场的总线协议,最初是为了实现串行通信,运用在串口(如RS232.RS485等)传输上的, ...

  5. Modbus协议和应用开发介绍

    因业务需要了解Modbus协议的使用,因此对Modbus的协议,以及相应的C#处理应用进行了解,针对协议的几种方式(RTU.ASCII.TCPIP)进行了封装,以及对Modbus的各种功能码的特点进行 ...

  6. modbus协议开关量采集模块

    modbus协议开关量采集模块是指的使用Modbus协议的进行信号的采集与控制的一种设备. Modbus 协议设备都具有唯一的 Modbus 地址,众山 DTU 默认 Modbus 地址为 100,用 ...

  7. modbus协议讲义

        Modbus 一个工业上常用的通讯协议.一种通讯约定.Modbus协议包括RTU.ASCII.TCP.其中MODBUS-RTU最常用,比较简单,在单片机上很容易实现.虽然RTU比较简单,但是看 ...

  8. Android 模拟HTTP协议的编码问题 Android默认编码UTF-8

    Android通过GET和POST方法请求服务器和浏览器请求的过程是不一样的. 浏览器请求服务器的时候会先将中文进行UTF-8编码,然后再发送到服务器端. Android编程下我们需要通过URLEnc ...

  9. 各种非标232,485协议,自定义协议转modbus协议模块定制开发,各种流量计协议转modbus,

    工业现场经常会碰到通过485或者232采集各类仪表数据,但是很多早期的仪表和设备不支持标准modbus协议,而是采用自定义的协议,这些协议数据由plc或者dcs系统来实现采集,不仅费时麻烦,而且不方便 ...

随机推荐

  1. C和指针笔记 3.6链接属性

    链接属性决定如何处理在不同文件中出现的标识符.标识符的作用域也它的链接属性有关,但这两个属性并不相同. 没有链接属性的标识符(none)总是被当作单独的个体,也就是说该标识符的多个声明被当作独立不同的 ...

  2. 用yo命令创建项目

    1,npm install -g yo 安装yeoman 2,npm install -g generator-webapp 安装项目脚手架(生成器) 如果安装angular的项目后面则是genera ...

  3. Xcode中文乱码问题

    老师给拷贝的程序用Xcode打开中文显示是乱码,而预览里面是正常显示的,Xcode默认编码UTF-8没错的,怎么办呢? 解决办法:用自带的文本编辑器打开,全选,复制,Xcode中打开文件,粘贴,ok~ ...

  4. Thread锁 Monitor类、Lock关键字和Mutex类

    Monitor 类锁定一个对象 当多线程公用一个对象时,也会出现和公用代码类似的问题,这种问题就不应该使用lock关键字了,这里需要用到System.Threading中的一个类Monitor,我们可 ...

  5. Swift3.0 iOS获取当前时间 - 年月日时分秒星期

    Swift3.0 iOS获取当前时间 - 年月日时分秒星期func getTimes() -> [Int] { var timers: [Int] = [] // 返回的数组 let calen ...

  6. Spring中Bean的命名问题(id和name区别)及ref和idref之间的区别

    Spring中Bean的命名 1.每个Bean可以有一个id属性,并可以根据该id在IoC容器中查找该Bean,该id属性值必须在IoC容器中唯一: 2.可以不指定id属性,只指定全限定类名,如: & ...

  7. Sublime 将 Tab 转为空格

    最近在使用 vue-cli 搭建项目,但每次用 Hbuilder 编写 vue 文件的时候,如果存在<script>部分就会报错,错误信息大意是说空格有问题.仔细研究了之后才知道,这是因为 ...

  8. [转]C# WinForm treeview checkbox----递归算法利用

    在平常开发中,treeview的节点显示checkbox,若节点存在几级时,往往希望,选中父节点后,其子节点都要选中,如何实现勒,请看 using System; using System.Colle ...

  9. Qt——树的搜索

    一.Qt中的树 Qt中树的实现有两种方式.第一种是使用Qt提供的QTreeWidget,很多函数都封装好,比较方便:另一种是通过QTreeView实现,设置它的数据模型,比如使用QStandardIt ...

  10. 前端开发工程师:网易web前端课程,价值1499元【无水印版】

    这套网上的朋友购买分享给我的,特此分享~ 让大家都受益 早日成为强大的web前端开发工程师!!赶紧回复下载吧 下载地址:http://fu83.cn/thread-172-1-1.html