STM32 MODBUS通讯源码 stm32 modbus 主机
yuyutoo 2024-12-19 17:33 6 浏览 0 评论
最近几年受口罩影响,工控产品的价格以是水涨船高。尤其是PLC,不管是高端还是低端的PLC,价格是早已翻了好几倍。价格高不说而且还缺货,所以很多公司为了降低成本,保证能够有充足的货源,都采取了用STM32自制电路板来替代。虽然STM32的价格也是翻了好几倍,但是这也只是单个芯片的价格而已,如果用产品去相比PLC仍然还是具有相当大的空间。其实如果没有复杂的总线控制的需求,只是简单的完成PLC的IO及通讯功能,实现起来还是挺简单的。IO我就不讲了,因为那确实太简单。下面分享给大家的是MODBUS-RTU的通讯源码,希望能给搞工控的朋友一些帮助。
【下面是MODBUS-RTU调度程序】
void ModbusRTU(u8 com,u8 *data, u16 len)
{
if(data[0]==1&&IF_ModbusCRC16(com,data,len )==1)
{
switch(data[1])//根据不同的功能码进行处理
{
case 01:{CMD01(com,data,len);break;}//读输出开关量
case 02:{CMD02(com,data,len);break;}//读输入开关量
case 03:{CMD03(com,data,len);break;}//读多个寄存器
case 05:{CMD05(com,data,len);break;}//写单个输出开关量
case 06:{CMD06(com,data,len);break;}//写单个寄存器
case 15:{CMD0F(com,data,len);break;}//写多个输出开关量
case 16:{CMD10(com,data,len);break;}//写多个寄存器
}
}
}
u8 M[1000];
u8 X[1000];
u8 Y[1000];
u16 D[1000];
u8 IS8BIT =0;
u16 OUTIM_A=0;
u16 OUTIM_B=0;
void SEND_DAT(u8 com,u8 *data, u16 len)//将数据从指定的串口发送出去
{
if(com==2){USART2_SendTxt(data,len);}
if(com==3){USART3_SendTxt(data,len);}
if(com==4){USART4_SendTxt(data,len);}
if(com==5){USART5_SendTxt(data,len);}
}
【下面是MODBUS-RTU从站模式】
void CMD01(u8 com,u8 *data, u16 len)
{
u8 U8BIT=0;
u8 U8BYTE=0;
u8 RDT[200];
int i=0;
int LEN=data[4];
int STARTADDR=data[2];
LEN=(LEN<<8) + data[5];
STARTADDR=(STARTADDR<<8)+data[3];
RDT[0]=data[0];
RDT[1]=data[1];
RDT[2]=LEN/8;
if(LEN%8){RDT[2]=RDT[2]+1;}
for(i=0;i<LEN;i++)
{
U8BIT=i%8;
if(M[STARTADDR+i])
{
U8BYTE=1;
RDT[3+i/8]=RDT[3+i/8]|(U8BYTE<<U8BIT);
}
else
{
U8BYTE=254;
RDT[3+i/8]=RDT[3+i/8]&(U8BYTE<<U8BIT);
}
}
ModbusCRC16(com,RDT,RDT[2]+3 );
RDT[4+LEN]=COMCRC16[com][0];
RDT[5+LEN]=COMCRC16[com][1];
SEND_DAT(com,RDT,LEN+5);
}
void CMD02(u8 com,u8 *data, u16 len)
{
}
void CMD03(u8 com,u8 *data, u16 len)
{
u8 RDT[200];
int i=0;
int LEN=data[5]*2;
int STARTADDR=data[2];
STARTADDR=(STARTADDR<<8)+data[3];
RDT[0]=data[0];
RDT[1]=data[1];
RDT[2]=LEN;
for(i=0;i<LEN;i++)
{
if(IS8BIT==0){if((i%2)==0){RDT[3+i]=D[STARTADDR+i/2]>>8;}else{RDT[3+i]=D[STARTADDR+i/2];}}
else{RDT[3+i]=D[STARTADDR+i];}
}
ModbusCRC16(com,RDT,LEN+3 );
RDT[3+LEN]=COMCRC16[com][0];
RDT[4+LEN]=COMCRC16[com][1];
SEND_DAT(com,RDT,LEN+5);
}
void CMD05(u8 com,u8 *data, u16 len)
{
u8 RDT[200];
int i=0;
int STARTADDR=data[2];
STARTADDR=(STARTADDR<<8)+data[3];
RDT[0]=data[0];
RDT[1]=data[1];
RDT[2]=data[2];
RDT[3]=data[3];
RDT[4]=data[4];
RDT[5]=data[5];
RDT[6]=data[6];
RDT[7]=data[7];
M[STARTADDR+0]=data[5]?1:0;
SEND_DAT(com,RDT,8);
}
void CMD06(u8 com,u8 *data, u16 len)
{
u8 RDT[200];
int i=0;
int STARTADDR=data[2];
STARTADDR=(STARTADDR<<8)+data[3];
RDT[0]=data[0];
RDT[1]=data[1];
RDT[2]=data[2];
RDT[3]=data[3];
RDT[4]=data[4];
RDT[5]=data[5];
RDT[6]=data[6];
RDT[7]=data[7];
if(IS8BIT==0)
{
D[STARTADDR]=data[4];
D[STARTADDR]=(D[STARTADDR]<<8)+data[5];
}
else
{
D[STARTADDR+0]=data[4];
D[STARTADDR+1]=data[5];
}
SEND_DAT(com,RDT,8);
}
void CMD0F(u8 com,u8 *data, u16 len)
{
u8 U8BIT=0;
u8 U8BYTE=0;
u8 RDT[200];
int i=0;
int LEN=data[4];
int STARTADDR=data[2];
LEN=(LEN<<8)+data[5];
STARTADDR=(STARTADDR<<8)+data[3];
RDT[0]=data[0];
RDT[1]=data[1];
RDT[2]=data[2];
RDT[3]=data[3];
RDT[4]=data[4];
RDT[5]=data[5];
for(i=0;i<LEN;i++)
{
U8BIT=i%8;
U8BYTE=1<<U8BIT;
M[STARTADDR+i]=(data[7+i/8]&U8BYTE)?1:0;
}
ModbusCRC16(com,RDT,6);
RDT[6]=COMCRC16[com][0];
RDT[7]=COMCRC16[com][1];
SEND_DAT(com,RDT,8);
}
void CMD10(u8 com,u8 *data, u16 len)//写多个寄存器
{
u8 RDT[200];
int i=0;
int LEN=data[6];
int STARTADDR=data[2];
STARTADDR=(STARTADDR<<8)+data[3];
RDT[0]=data[0];
RDT[1]=data[1];
RDT[2]=data[2];
RDT[3]=data[3];
RDT[4]=data[4];
RDT[5]=data[5];
for(i=0;i<LEN;i++)
{
if(IS8BIT==0)
{
if((i%2)==0){D[STARTADDR+i/2]=data[7+i];D[STARTADDR+i/2]=D[STARTADDR+i/2]<<8;}else{D[STARTADDR+i/2]+=data[7+i];}
}
else
{
D[STARTADDR+i]=data[7+i];
}
}
ModbusCRC16(com,RDT,6);
RDT[6]=COMCRC16[com][0];
RDT[7]=COMCRC16[com][1];
SEND_DAT(com,RDT,8);
}
【下面是CRC效验程序】
#include "MODBUS_CRC16.h"
u8 COM2CRC16[2];
u8 COM3CRC16[2];
u16 IX, IY;
//CRC校验查表用参数 /* CRC 高位字节值表*/
static u8 auchCRCHi[] =
{
0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x01,0xC0,
0x80,0x41,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,
0x00,0xC1,0x81,0x40,0x00,0xC1,0x81,0x40,0x01,0xC0,
0x80,0x41,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,
0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x00,0xC1,
0x81,0x40,0x01,0xC0,0x80,0x41,0x01,0xC0,0x80,0x41,
0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x00,0xC1,
0x81,0x40,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,
0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x01,0xC0,
0x80,0x41,0x00,0xC1,0x81,0x40,0x00,0xC1,0x81,0x40,
0x01,0xC0,0x80,0x41,0x01,0xC0,0x80,0x41,0x00,0xC1,
0x81,0x40,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,
0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x01,0xC0,
0x80,0x41,0x00,0xC1,0x81,0x40,0x00,0xC1,0x81,0x40,
0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x01,0xC0,
0x80,0x41,0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,
0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x01,0xC0,
0x80,0x41,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,
0x00,0xC1,0x81,0x40,0x00,0xC1,0x81,0x40,0x01,0xC0,
0x80,0x41,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,
0x01,0xC0,0x80,0x41,0x00,0xC1,0x81,0x40,0x01,0xC0,
0x80,0x41,0x00,0xC1,0x81,0x40,0x00,0xC1,0x81,0x40,
0x01,0xC0,0x80,0x41,0x01,0xC0,0x80,0x41,0x00,0xC1,
0x81,0x40,0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,
0x00,0xC1,0x81,0x40,0x01,0xC0,0x80,0x41,0x01,0xC0,
0x80,0x41,0x00,0xC1,0x81,0x40
};
/* CRC低位字节值表*/
static u8 auchCRCLo[] =
{
0x00,0xC0,0xC1,0x01,0xC3,0x03,0x02,0xC2,0xC6,0x06,
0x07,0xC7,0x05,0xC5,0xC4,0x04,0xCC,0x0C,0x0D,0xCD,
0x0F,0xCF,0xCE,0x0E,0x0A,0xCA,0xCB,0x0B,0xC9,0x09,
0x08,0xC8,0xD8,0x18,0x19,0xD9,0x1B,0xDB,0xDA,0x1A,
0x1E,0xDE,0xDF,0x1F,0xDD,0x1D,0x1C,0xDC,0x14,0xD4,
0xD5,0x15,0xD7,0x17,0x16,0xD6,0xD2,0x12,0x13,0xD3,
0x11,0xD1,0xD0,0x10,0xF0,0x30,0x31,0xF1,0x33,0xF3,
0xF2,0x32,0x36,0xF6,0xF7,0x37,0xF5,0x35,0x34,0xF4,
0x3C,0xFC,0xFD,0x3D,0xFF,0x3F,0x3E,0xFE,0xFA,0x3A,
0x3B,0xFB,0x39,0xF9,0xF8,0x38,0x28,0xE8,0xE9,0x29,
0xEB,0x2B,0x2A,0xEA,0xEE,0x2E,0x2F,0xEF,0x2D,0xED,
0xEC,0x2C,0xE4,0x24,0x25,0xE5,0x27,0xE7,0xE6,0x26,
0x22,0xE2,0xE3,0x23,0xE1,0x21,0x20,0xE0,0xA0,0x60,
0x61,0xA1,0x63,0xA3,0xA2,0x62,0x66,0xA6,0xA7,0x67,
0xA5,0x65,0x64,0xA4,0x6C,0xAC,0xAD,0x6D,0xAF,0x6F,
0x6E,0xAE,0xAA,0x6A,0x6B,0xAB,0x69,0xA9,0xA8,0x68,
0x78,0xB8,0xB9,0x79,0xBB,0x7B,0x7A,0xBA,0xBE,0x7E,
0x7F,0xBF,0x7D,0xBD,0xBC,0x7C,0xB4,0x74,0x75,0xB5,
0x77,0xB7,0xB6,0x76,0x72,0xB2,0xB3,0x73,0xB1,0x71,
0x70,0xB0,0x50,0x90,0x91,0x51,0x93,0x53,0x52,0x92,
0x96,0x56,0x57,0x97,0x55,0x95,0x94,0x54,0x9C,0x5C,
0x5D,0x9D,0x5F,0x9F,0x9E,0x5E,0x5A,0x9A,0x9B,0x5B,
0x99,0x59,0x58,0x98,0x88,0x48,0x49,0x89,0x4B,0x8B,
0x8A,0x4A,0x4E,0x8E,0x8F,0x4F,0x8D,0x4D,0x4C,0x8C,
0x44,0x84,0x85,0x45,0x87,0x47,0x46,0x86,0x82,0x42,
0x43,0x83,0x41,0x81,0x80,0x40
};
void ModbusCRC16(int com,u8 *data, u16 len)
{
u8 L=(u8)len;
u8 uchCRCHi=0xFF; /*高CRC字节初始化*/
u8 uchCRCLo=0xFF; /*低CRC字节初始化*/
u8 uIndex; /*CRC循环中的索引*/
while(L--) /* 传输消息缓冲区*/
{
uIndex = uchCRCHi ^ *data++; /* 计算CRC */
uchCRCHi = uchCRCLo ^ auchCRCHi[uIndex];
uchCRCLo = auchCRCLo[uIndex];
}
if(com==2)
{
COM2CRC16[0] =uchCRCHi;//高位置
COM2CRC16[1] =uchCRCLo; //低位置
}
else
{
COM3CRC16[0] =uchCRCHi;//高位置
COM3CRC16[1] =uchCRCLo; //低位置
}
}
u8 IF_ModbusCRC16(int com,u8 *data, u16 len )
{
u8 CRC16[2];
if(len>4)
{
ModbusCRC16(com,data,len-2);
if(com==2)
{
CRC16[0]=COM2CRC16[0];//高位置
CRC16[1]=COM2CRC16[1];//低位置
}
else
{
CRC16[0]=COM3CRC16[0];//高位置
CRC16[1]=COM3CRC16[1];//低位置
}
if(data[len-2]==CRC16[0]&&data[len-1]==CRC16[1])return 1;else return 0;
}
else{return 0;}
}
相关推荐
- 建筑福利-pdf转dwg格式转换器,再也不用描图-极客青年
-
作为一名经常熬夜画图的建筑狗或者cad用户,你体验过pdf图纸描图到cad吗?前几天一个老同学找我,说他的毕业设计需要我帮忙,发给我一份pdf图纸文件,问我怎么把pdf图纸转换成dwg格式。机智的我灵...
- 想学 HTML,不知从何入手?看完这篇文章你就知道了
-
很多人都说HTML是一门很简单的语言,看看书,看看视频就能读懂。但是,如果你完全没有接触过,就想通过看一遍教程,背背标签,想要完全了解HTML,真的有点太天真了。HTML中文...
- 「前端」HTML之结构
-
今天继续为大家分享前端的知识,如果对前端比较感兴趣的小伙伴,可以关注我,我会更大家继续分享更多与前端相关的内容,当然如果内容中又不当的或者文字错误的,欢迎大家在评论区留言,我会及时修改纠正。1.初识H...
- 手把手教你使用Python网络爬虫下载一本小说(附源码)
-
大家好,我是Python进阶者。前言前几天【磐奚鸟】大佬在群里分享了一个抓取小说的代码,感觉还是蛮不错的,这里分享给大家学习。...
- 用于处理pdf文件格式的转换器
-
在上传过程中如果单个文件太大则容易中断,而且文件太大的话对与存储也有些弊端。那么我们应该想到将文件进行压缩(注意这里压缩指的是不改变文件格式的压缩,而不是用变成压缩文件。这里就将以下用专门的软件压缩P...
- 乐书:在线 Kindle 电子书制作和转换工具
-
之前Kindle伴侣曾推荐过可以在Windows和Mac系统平台上运行的kindle电子书制作软件Sigil(教程),用它可以制作出高质量的的ePub格式电子书,当然最后还需要通...
- 付费文档怎么下载?教你5种方法,任意下载全网资源
-
网上查资料的时候,经常遇到需要注册登录或者付费的才能复制或者是下载,遇到这种情况大多数人都会选择重新查。...
- 捡来的知识!3种方法随便复制网页内容,白嫖真香呀
-
网上的资源真的多,所以许多人常常会从网上找资料。我们看到感兴趣的内容,第一时间可能会想要收入囊中。比如说截个图啊,或者挑选有意思的句子复制粘贴,记录下来。可是,有些时候,却会遇到这样的情况:1、内容不...
- AI的使用,生成HTML网页。
-
利用deepseek,豆包,kimi以及通义千问,写入相同的需求。【写一个网页,实现抽奖功能,点击“开始”,按键显示“停止”,姓名开始显示在屏幕上,人员包括:“张三”,“里斯”,“Bool”,“流水废...
- pdf转换成jpg转换器 4.1 官方正式版
-
pdf转换成jpg工具软件简介pdf转换成jpg转换器是一款界面简洁,操作方便的pdf转换成jpg转换器。pdf转换成jpg转换器可以将PDF文档转换为JPG,BMP,GIF,PNG,TIF图片文件。...
- 办公必备的office转换成pdf转换器怎么用?
-
2016-02-2415:53:37南方报道网评论(我要点评)字体刚从校园走出社会,对于快节奏的办公环境,难免会觉得有些吃力。在起步阶段力求将手头上的事情按时完工不出错,但是渐渐的你会发现,别人只...
- 为什么PDF转Word大多要收费?
-
PDF转Word大多都要收费?并非主要是因为技术上的难度,而是基于多方面的商业和版权考虑的,下面给大家浅分析下原因:...
- 如何用python生成简单的html report报告
-
前提:用python写了一个简单的log分析,主要也就是查询一些key,value出来,后面也可以根据需求增加。查询出来后,为了好看,搞个html表格来显示。需要的组件:jinja2flask...
- 学用系列|如何搞定word批量替换修改和格式转换?这里一站搞定
-
想必不少朋友都会碰到批量修改word文档内容、压缩文档图片、文件格式转换等重复性文档处理工作的需要,今天胖胖老师就推荐给大家一个免费工具XCLWinKits,一站搞定你所有的需要。什么是XCLWinK...
- 这款PDF文档转换神器,能帮你解决PDF使用中的许多难点
-
不管是平时的学习还是工作,相信许多朋友都经常接触PDF文件。可以说,PDF文件在我们的日常办公学习过程中的重要性和Word文档一样重要。在之前的更新中,小编介绍了几款非常不错的PDF文档格式转换软件,...
你 发表评论:
欢迎- 一周热门
-
-
前端面试:iframe 的优缺点? iframe有那些缺点
-
带斜线的表头制作好了,如何填充内容?这几种方法你更喜欢哪个?
-
漫学笔记之PHP.ini常用的配置信息
-
其实模版网站在开发工作中很重要,推荐几个参考站给大家
-
推荐7个模板代码和其他游戏源码下载的网址
-
[干货] JAVA - JVM - 2 内存两分 [干货]+java+-+jvm+-+2+内存两分吗
-
正在学习使用python搭建自动化测试框架?这个系统包你可能会用到
-
织梦(Dedecms)建站教程 织梦建站详细步骤
-
【开源分享】2024PHP在线客服系统源码(搭建教程+终身使用)
-
2024PHP在线客服系统源码+完全开源 带详细搭建教程
-
- 最近发表
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)