模拟Modbus协议问题

问题:

在嵌入式系统开发中,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]);
    }
}

时间: 2024-10-17 23:02:08

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

基于AVR128的简单Modbus协议实现

Modbus通讯协议是由Modicon公司在1979年开发的,应用于工业现场控制的总线协议.Modbus通讯系统包括带有可编程控制的芯片节点和公共传输线组成,其目的是用于多节点数据的采集和监控.Modbus协议采用主从模式,通讯系统中有一个主机对多个节点从机进行监控,从机节点最多支持247个.每个从机均有自己独立的从机地址,而且改地址能够被主机识别. 能够支持Modbus协议的通讯系统有RS-232,RS-422,RS-485等.同时Modbus协议具有标准.开放.免费.帧格式简单等特点而被广大

基于AVR128单纯Modbus协议实施

Modbus通信协议Modicon公司1979在发展中,适用于工业现场总线协议控制.Modbus通信系统包含芯片的节点,并与组合物可编程控制的公共传输线,它的目的是收集和监视多个节点的数据.Modbus协议采用主从模式,通信系统具有多个节点的从一台主机机监视器.最多支持从节点247个.每一个从机均有自己独立的从机地址.并且改地址可以被主机识别. 可以支持Modbus协议的通讯系统有RS-232.RS-422,RS-485等.同一时候Modbus协议具有标准.开放.免费.帧格式简单等特点而被广大p

modbus协议(2)

上一篇介绍了modbus协议的基本概念,这一篇主要介绍最近做的一个小项目:STM3210ZET6与昆仑屏(TPC)的通信.在该项目中最关键的技术就是下位机modbus协议的解析. 首先介绍下昆仑屏(TPC),项目中用到的触摸屏采用的RS232接口,modbus协议. 采用的驱动是:莫迪康ModbusRTU:本驱动支持 01.02.03.04.05.06.15.16 常用功能码. 本驱动构件支持的寄存器及功能码说明如下: 1.设备构件参数设置: "莫迪康 ModbusRTU"子设备参数设

modbus协议之串行链路

目录 modbus协议之串行链路 一.modbus简介 二.modbus消息桢 三.modus设备地址与功能码 附录一.CRC校验与LRC校验 附录二.功能码 其它参考文档 modbus协议之串行链路 一.modbus简介 Modbus是一种串行通信协议,是Modicon公司(现在的施耐德电气Schneider Electric)于1979年为使用可编程逻辑控制(PLC)通信而发表.Modbus已经成为工业领域通信协议的业界标准(De facto),并且现在是工业电子设备之间常用的连接方式. M

用IO模拟串口协议发送数据

<pre name="code" class="cpp">//文件usend.h #ifndef _USEND_H_ #define _USEND_H_ //====红外接收相关定义============================= #define PuTx_High (P_uTx = 1) //数据高 #define PuTx_Low (P_uTx = 0) //数据低 #define V_SendDatNum 6//6 //发送数据字节数 /

简单Modbus协议数据源工具实现(一)WinForm

这是一个学习C#.Winform的自我回顾过程,用来发现存在的不足,也为了推动自己继续学习. 大学通信专业毕业之后,进入了一家电力科技公司从事软件开发工作,主要用的是Delphi语言进行电力通信协议的上位机开发.因为上位机需要与下位机通信才好进行测试,而事实上没有那么多现成的装置给你借用调试,加上公司慢慢的开始推行C#/WPF来做一些定制软件,所以想学习一下C#,刚好现在也有一个自身的需求出现--上位机程序调试困难,所以就从最易入手的winform程序切入,慢慢的加深对于C#语言的理解.于是就打

模拟SPI协议时序

SPI是串行外设接口总线,摩托罗拉公司开发的一种全双工,同步通信总线,有四线制和三线制. 在单片机系统应用中,单片机常常是被用来当做主机(MASTER),外围器件被当做从机(SLAVE). 所以,在以下的介绍中,都是默认单片机是主机模式进行说明的. SPI总线相对于IIC总线,无总裁机制,无应答机制. SPI常用的四线制分别是,MISO(主入从出).MOSI(主出从入).SCK(同步时钟线).CS(片选线,也有是NSS). 主从机之间典型的接线方式如下所示: 按照时钟线的时钟极性(CPOL)和相

MODBUS协议相关代码(CRC验证 客户端程序)

Modbus协议是一种已广泛应用于当今工业控制领域的通用通讯协议.通过此协议,控制器相互之间.或控制器经由网络(如以太网)可以和其它设备之间进行通信.Modbus协议使用的是主从通讯技术,即由主设备主动查询和操作从设备.一般将主控设备方所使用的协议称为Modbus Master,从设备方使用的协议称为Modbus Slave.典型的主设备包括工控机和工业控制器等:典型的从设备如PLC可编程控制器等.Modbus通讯物理接口可以选用串口(包括RS232和RS485),也可以选择以太网口. 1.十六

RTU的优势与Modbus协议介绍

RTU是REMOTE TERMINAL UNIT 的简称,即远方数据终端,用于监视.控制与数据采集的应用.具有遥测.遥信.遥调.遥控功能.RTU功能必须确保两种功能:1,有数据传输功能.2,有采集和控制功能. 经过多年的PLC+DTU在工业自动化应用中已经普遍采用这种方式在进行数据的在线监测和远程控制.当RTU面世后,RTU集成的A/D和I/O采集功能已经可以达到取代部分PLC功能了. 1,在一些相对简单的温度.压力.湿度.水位.烟雾等传感器的数据采集监测,完全已经可以通过RTU取代前端早期的P