历史上的今天

今天是:2024年09月15日(星期日)

2021年09月15日 | STM8学习笔记---Modbus通信协议简单移植

发布者:breakthrough2 来源: eefocus关键字:STM8  Modbus  通信协议  移植 手机看文章 扫描二维码
随时随地手机看文章

Modbus是一种串行通信协议,在工业中应用是比较广泛的。关于Modbus的介绍网上资料很多,这里就不细说了。刚开始接触的时候看Modbus的介绍,光是协议的介绍有几百页,还有各种命令,各种链路层的应用,看了几天,越看越糊涂,越看越不会用。   


最后在单片机上移植成功后才感觉Modbus协议没那么复杂,如果刚开始学的时候,没必要把Modbus协议中每个功能都去了解。就把它当做简单的串口协议,只使用最简单的几个命令就行了。熟悉之后再慢慢了解其他功能。


下面就从单片机串口通信角度去理解Modbus协议,及如何将协议移植到单片机上。


先看看Modbus的协议

从大的方面来讲,协议总共由4部分组成: 地址、功能、数据、校验。


地址1个字节,也就是设备的地址范围是 0 --- 255。


功能码也就是命令,也是一个字节,范围是0---255。


数据位在不同的情况下有不同的长度。


校验位一般用的是CRC校验


下来看看功能码都有哪些

常用的功能码有表格上面的这些,可以理解为一个数字代表的一种命令。给单片机移植时用03、06、16这三个命令就够用了。


这里面读线圈、写单个线圈、写单个寄存器等等,到底什么是线圈?什么是寄存器?这些都是什么意思?


简单的理解线圈就是位操作。比如说单片机控制了8路的继电器输出,为了方便表示继电器的状态,就用8个位来表示8个继电器的状态,比如0表示继电器断开,1表示继电器吸合。这样0x00就表示8路继电器全部断开,0xFF表示8路继电器全部吸合。


寄存器是字节操作,比如传感器采集温度的时候用一个字节表示当前温度,比如当前温度28℃,就用0x1C表示。

如果理解不了寄存器和线圈的含义就不用管它,就把他当做一个命令来看,在单片机中使用时03、06、16这三个命令就能满足基本需求,下面就单独分析一下这三个命令的含义。


03是读多个保持寄存器值,读取的个数可以设置,比如有8组温度传感器采集数据,要读取温度值,可以一组一组去读,也可以一次性读多个值,读取的个数自己设置。


先看看03的命令格式

请求就是单片机主机发送数据,正常响应就是主机发送的命令格式正确时,从机回复的数据。当主机发送的数据从机不能正确识别时,从机要返回异常响应数据,告诉主机发送的命令有错误。

这里解释一下命令里面各个位的含义,这里是采集8组温度传感器的数值,假如一个从机有8路温度传感器,这个从机的地址就定义为0x01,这个地址根据实际项目可以自己定义。功能码为0x03,这里使用Modbus规定的功能码,意思是读多个寄存器。起始地址为两个字节,表示从第几个温度传感器开始读取数据,寄存器数量也为两个字节,表示要读取几个温度传感器的值。由于只有8路温度传感器,所以起始地址的范围就是 0x0000 ---- 0x0007。寄存器数量的范围为0x0001---0x0008,最少要读取一个寄存器的值,最多读8个寄存器的值。最后就是CRC校验, CRC具体的校验方式这里不用关心,使用的时候直接调用校验函数就行。


这里要注意请求数据的时候要发送起始地址和请求数量,而返回数据的时候就没有请求地址了,只有发送的寄存器字节数。


比如现在要读取第一个温度传感器的值,那么请求数据格式如下:


从站地址     功能码   起始地址高位  起始地址低位  寄存器数量高位  寄存器数量低位  CRC校验高位  CRC校验低位


0x01             0x03            0x00              0x00                   0x00                   0x01                 xx                   xx


从0地址开始,读取1个寄存器的值,也就是读取第一个温度传感器的值。


正常响应返回数据格式如下


从站地址     功能码     字节数   寄存器数量高位  寄存器数量低位  CRC校验高位  CRC校验低位


0x01              0x03      0x02            0x00                  0x1E                  XX                      XX


读取到了2个字节寄存器的值,寄存器值为 0x001E, 0x001E对应的十进制数为30,说明第一个温度传感器的温度值为30℃。


那么异常响应是什么情况下会用到?假如请求数据发送的是读取第9个温度传感器的值,从机接收到数据后发现没有第9个传感器,说明主机发送的地址值超过范围了,那么从机这时就要给主机发送异常响应。常用的异常响应码有下面几种

从异常响应码中可以看出来,地址值不在范围内的异常码为0x02,Modbus规定返回异常响应时,差错码的值为功能码的值加上0x80,当前功能码为0x03,所以返回的差错码数值为0x83,差错码数值为0x02。


请求数据:


从站地址     功能码   起始地址高位  起始地址低位  寄存器数量高位  寄存器数量低位  CRC校验高位  CRC校验低位


0x01             0x03            0x00              0x09                   0x00                   0x01                 xx                   xx


异常响应:


从站地址    差错码   异常码   CRC校验   


0x01            0x83         0x02    xx


再看一个读取多个寄存器值的示例:

下面在看0x06写单个保持寄存器,就是给一个指定的寄存器中写入数据。通信格式如下:

通信示例如下:

可以看到写单个保持寄存器的请求命令和正常响应命令是完全相同的,这个就更好理解了。这块要注意 差错码的值为功能码的值加上0x80,当前功能码为0x06,所以返回的差错码数值为0x86。


下来在看看16(0x10)写多个保持寄存器,写多个保存寄存器和读多个寄存器基本一样,只不过一个是读,一个是写。

这块要注意 差错码的值为功能码的值加上0x80,当前功能码为0x10,所以返回的差错码数值为0x90。


通信示例如下:

响应命令只返回写的寄存器数量,而不返回写的寄存器值,这个和写单个寄存器是不同的。


通过上面的分析对Modbus就会有个大概的了解了,它也没有想得那么复杂。


下面就看看用代码如何实现上面这3个命令的功能。


首先看串口发送和接收代码的实现


#include "uart.h"

#include "stdio.h"

#include "main.h"

 

u8 ReceiveBuf[MaxDataLen] = {0};

u8 RecIndexLen = 0;

void Uart1_IO_Init( void )

{

    PD_DDR |= ( 1 << 5 ); //输出模式 TXD

    PD_CR1 |= ( 1 << 5 ); //推挽输出

    PD_DDR &= ~( 1 << 6 ); //输入模式 RXD

    PD_CR1 &= ~( 1 << 6 ); //浮空输入

}

//波特率最大可以设置为38400

void Uart1_Init( unsigned int baudrate )

{

    unsigned int baud;

    baud = 16000000 / baudrate;

    Uart1_IO_Init();

    UART1_CR1 = 0;

    UART1_CR2 = 0;

    UART1_CR3 = 0;

    UART1_BRR2 = ( unsigned char )( ( baud & 0xf000 ) >> 8 ) | ( ( unsigned char )( baud & 0x000f ) );

    UART1_BRR1 = ( ( unsigned char )( ( baud & 0x0ff0 ) >> 4 ) );

    UART1_CR2_bit.REN = 1;        //接收使能

    UART1_CR2_bit.TEN = 1;        //发送使能

    UART1_CR2_bit.RIEN = 1;       //接收中断使能

}

//阻塞式发送函数

void SendChar( unsigned char dat )

{

    while( ( UART1_SR & 0x80 ) == 0x00 ); //发送数据寄存器

    UART1_DR = dat;

}

//发送一组数据

void Uart1_Send( unsigned char* DataAdd, unsigned char len )

{

    unsigned char i;

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

    {

        SendChar( DataAdd[i] );

    }

    //SendChar(0x0d);     //发送回车换行,测试用

    //SendChar(0x0a);

}

//接收中断函数 中断号18

#pragma vector = 20             // IAR中的中断号,要在STVD中的中断号上加2

__interrupt void UART1_Handle( void )

{

    u8 res = 0;

    res = UART1_DR;

    ReceiveBuf[RecIndexLen++] = res;

    return;

}

          串口代码和常规的用法是一样的,初始化IO口和波特率,然后用中断接收数据,ReceiveBuf数组用来存放接收的数据,RecIndexLen用来统计接收数据的长度。


         一组数据接收完毕之后,调用数据处理函数,来处理接收到的数据。


//处理接收到的数据

// 接收: [地址][功能码][起始地址高][起始地址低][总寄存器数高][总寄存器数低][CRC低][CRC高]

void DisposeReceive( void )

{

    u16 CRC16 = 0, CRC16Temp = 0;

    if( ReceiveBuf[0] == SlaveID )                                 //地址等于本机地址 地址范围:1 - 32

    {

        CRC16 = App_Tab_Get_CRC16( ReceiveBuf, RecIndexLen - 2 );  //CRC校验 低字节在前 高字节在后 高字节为报文最后一个字节

        CRC16Temp = ( ( u16 )( ReceiveBuf[RecIndexLen - 1] << 8 ) | ReceiveBuf[RecIndexLen - 2] );

        if( CRC16 != CRC16Temp )

        {

            err = 4;                                               //CRC校验错误

        }

        StartRegAddr = ( u16 )( ReceiveBuf[2] << 8 ) | ReceiveBuf[3];

        if( StartRegAddr > 0x07 )

        {

            err = 2;                                               //起始地址不在规定范围内 00 - 07    1 - 8号通道

        }

        if( err == 0 )

        {

            switch( ReceiveBuf[1] )                                //功能码

            {

                case 3:                                            //读多个寄存器

                {

                    Modbus_03_Slave();

                    break;

                }

                case 6:                                            //写单个寄存器

                {

                    Modbus_06_Slave();

                    break;

                }

                case 16:                                           //写多个寄存器

                {

                    Modbus_16_Slave();

                    break;

                }

                default:

                {

                    err = 1;                                       //不支持该功能码

                    break;

                }

            }

        }

        if( err > 0 )

        {

            SendBuf[0] = ReceiveBuf[0];

            SendBuf[1] = ReceiveBuf[1] | 0x80;

            SendBuf[2] = err;                                      //发送错误代码

            CRC16Temp = App_Tab_Get_CRC16( SendBuf, 3 );           //计算CRC校验值

[1] [2] [3]
关键字:STM8  Modbus  通信协议  移植 引用地址:STM8学习笔记---Modbus通信协议简单移植

上一篇:STM8S003F3 使用定时器来计算方波周期的方法
下一篇:STM8学习笔记---ADC多通道采样

推荐阅读

   还有不到一周,苹果将召开秋季新品发布会,属于网友们的历时一年的“新iPhone发布会”也将迎来落幕,虽然苹果官方没有传出什么关于新iPhone的消息,但是树大招风,从供应链以及分析师透露的只言片语可以确定,今年发布三款新iPhone几乎是确认无疑的事。按照惯例,今年苹果发布的三款新iPhone将命名为iPhone 9、iPhone Xs和iPhone Xs Plus。其中i...
摩托罗拉今天在巴西市场推出智能手机新品——Moto E7 Plus,该机搭载水滴屏设计,后置双摄模组,定位于入门级产品,售价为 1499 雷亚尔(约 1925.1 元)。  配置方面,Moto E7 Plus 搭载骁龙 460 处理器,辅以 4GB RAM+64GB ROM,并支持 microSD 卡扩展。  其他方面,内置 5000 毫安时电池,支持 10W 充电;运行 Androi...
郭明錤表示,最新调查指出,苹果已解决面板模组品质问题,立讯目前正在增加与复制产线,并预计在 9 月中下旬开始大量出货。报告预计 Apple Watch Series 7 与总 Apple Watch 出货量在 2021 年将分别显著增长至 1400–1600 万块与 4000–4500 万块。  另外,郭明錤也对 Apple Watch Series 8 和未来的 AirPods 进行了简要...

史海拾趣

问答坊 | AI 解惑

GSM手机射频测试指导

本帖最后由 jameswangsynnex 于 2015-3-3 19:58 编辑 GSM手机射频测试指导 …

查看全部问答∨

扬声器阻抗特性测试

为什么,我在测试测量时从低频到高频逐渐改变信号发生器输出信号频率测试扬声器阻抗特性时,发现在20-20khz电压值随频率增大而增大,不知道是方法不对还是具体操作有误…

查看全部问答∨

【藏书阁】电路原理 (上、下册)江辑光

目录: 上册: 第1章 电路元件的电路定律 1.1 电路和电路模型 1.2 电流、电压、电动势 1.3 电路元件的功率 1.4 电阻元件 1.5 电感元件 1.6 电容元件 1.7 电源元件 1.8 受控电源 1.9 基尔霍夫定律 习题 第2章 简单电阻电路 ...…

查看全部问答∨

偶新来的,还请各位帅哥多关照啊。

一个人在沙漠行走,终于找到一片绿洲。 纯散分。…

查看全部问答∨

[大连] 高新招聘对日嵌入式开发工程师 开发课长

大家好。我是Yufy。 有几个新的工作机会刚刚open,想在这发布一下,看看有没有朋友愿意去试试。 工作地点:大连 因为客户的要求,公司的名称和薪水情况不方便公开透露,请大家谅解。不过这个公司薪水福利很好(年假,住房公积金等),如果有 ...…

查看全部问答∨

大家放松一下做几道面试题!

1 int a; int *p; p=&a; *p=0x500; a=(int)(*(&p)); a=(int)(&(*p)); if(a==(int)p) printf("equal!"); else printf("not equal!"); 输出什么? 2 void foo(void) {         unsigned int a=6; ...…

查看全部问答∨

哈哈,超级高兴,老大表扬我喽!!

近来公司网络很不稳定,时不时出现所有人不能上网的现象,我在其中一台电脑上装上了赛门铁克的网络安全特警NIS2007,马上2007就发现了一个ARP病毒,我把所有的电脑都装上了NIS2007,老大很高兴,当即批准购买8套正版NIS2007。…

查看全部问答∨

是不是Verilog现在比VDHL用的广的多

RT.我是学生 上次做实验的时候.老师跟我说.以后你们工作了.公司要求都是用Verilog.学校非得教VDHL,..没办法. 想问问各位工作了的前辈们.真的是这个样子么? 还有.现在学校真的太落伍了.51单片机和dsp还在用汇编讲..这些底层的东西.现在工作中真的 ...…

查看全部问答∨

PWM Buck circuit 电容 电压求解

本帖最后由 dontium 于 2015-1-23 13:21 编辑 问题描述写在附件的图片上。望高人指点迷津。谢谢! 实际测量一个样品,Vp=2.84V。 [ 本帖最后由 y651848590 于 2011-7-14 21:42 编辑 ] …

查看全部问答∨

电动消防车

本帖最后由 paulhyde 于 2014-9-15 04:05 编辑 2012年大学生电子设计竞赛黑龙江赛区TI杯竞赛题A题: 电动消防车(本科)一、任务设计制作一个电动消防车,能到消防场地任意地点进行灭火作业。以蜡烛模拟火源,火源随机分布在场地中,消防场地如图1 ...…

查看全部问答∨
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved