
/***************************************************** modbus-rtu 通讯规约 通讯方式:rs-485 半双功* M# t1 k C8 j 校验方式:crc16: U' d ^0 }/ n- O 停止位:2位 编写:孙可 编程思路: 1.串口中断允许自动接收总线上的信息,当接收的字节后超过3.5个字节时间没有新的字节认为本次接收完成,接收完成标志置1;如果接收完成标志已经置1又有数据进来则丢弃新来的数据。 2.串口接收数据的处理, 当接收完成标志置1进入接收数据处理, (1)首先判断接收的第一位数据与本机地址是否相同,如果不相同清空接收缓存不发送任何信息; (2)接收的第一位数据与本机地址相同,则对接收缓存中的数据进行crc16校验,如果接收的校验位与本校验结果不相同清空接收缓存不发送任何信息;(3)如果crc16校验正确则根据数据串中的命令码进行相应的处理。5 p* x1 d3 _1 c1 t ******************************************************/9 L1 y& g! G' `# E! \( o% O) Q4 G2 \ #include "modbus.h"# w8 T4 P) M; b% N1 W( b u8 Com0_id = 0x05;//本机串口0的通讯地址. O9 x, x7 H, p8 C( o; O. X# M u8 Uart0_rev_buff[100];//com0串口接收缓冲区 u8 Uart0_send_buff[100];//com0串口发送缓冲区 vu8 Uart0_rev_count;2 T# q1 x( ~( ], E9 U vs8 Uart0_send_counter = 0;, u% l* N+ u+ I( v vu8 Uart0_rev_comflag; vu8 Crc_counter = 0;//com0校验计数器( Z& Q6 d6 ~) ]; R% b vu8 *Uart0_send_pointer = Uart0_send_buff;//com0串口发送指针 vu16 Mkgz_bz = 0;//模块故障标志1:输入异常,2:过压,3:欠压,4:过温 vu16 Out_current = 50;//输出电流0 d) w1 q1 l- K8 z vu16 Out_voltage = 240;//输出电压 vu16 Mkzt_bz = 0;//模块状态标志0 @" i6 D7 Q6 I1 K, b vu16 OutX_current = 1000;//输出限流 vu16 Jc_voltage = 2530;//均充电压% M3 L& Z- p! h; @! Z& h0 g- k vu16 Fc_voltage = 2400;//浮充电压3 [' s u; O+ C6 k vu16 user_day = 1825;//使用天数 7 {7 D! c9 B# e* A7 A4 d4 o void Delay(vu32 nCount); unsigned short getCRC16(volatile unsigned char *ptr,unsigned char len) ; void mov_data(u8 a[100],u8 b[100],u8 c);+ w$ M% D5 x6 H5 N ? void Modbus_Function_3(void); " }9 X1 Y7 o8 h+ T# O$ V void Modbus_Function_6(void); ( a" z8 l! H- n% r /***************************************1 N5 w0 R( a- U) }/ X4 C% O$ v 函数名称:crc16校验) @, k& f3 P+ c9 v% w 函数功能:crc16校验/ _0 v( |- r, S4 y 函数输入:字节指针*ptr,数据长度len- k$ }! K# s! r6 I6 b1 P 函数返回:双字节crc8 l( E+ V% v9 W4 w% W 函数编写:孙可 编写日期:2008年6月9日 函数版本:v0.2 ****************************************/ unsigned short getCRC16(volatile unsigned char *ptr,unsigned char len) { ; k `& Q( f" @ unsigned char i; unsigned short crc = 0xFFFF; * h3 H2 m |0 r" k3 _ if(len==0) {3 l" A) P: i; F' y& q% Q% I6 k r) f len = 1;% v. V5 ^7 J9 p$ q& p } $ C9 e% K+ V- v# h# _' \ while(len--) { crc ^= *ptr; 5 T( I( F) J, A# L/ Y for(i=0; i>= 1; crc ^= 0xA001; * P/ ^3 e% b. o- o } else * |% r0 h$ f$ \& o4 c {8 ~2 `9 W! d9 S crc >>= 1; } / Y0 D1 `( @& Z* n9 v' Y2 m } ptr++; + u, R1 x/ x7 g5 n } ' ]( k8 z1 F1 B! w8 m& I' {1 }& r) K return(crc); 6 Y5 U4 ?0 F- r. U* Z } /*************************************** 块数据复制数据函数% ~. L2 r7 ?2 H L# ~, E 功能:把数组a的c个数据复制到数组b中% }" r+ A* [( D 输入:指针a,指针b,数据个数c1 J) @4 n/ V6 D1 T4 [* x 返回:无8 P% ^1 W9 B1 }. ~% p7 d* J 编写:孙可 编写日期:2008年3月28日 版本:v0.1$ i3 w) F, \% L, X# M ****************************************/ U2 Q' T3 Z4 W* M% U1 z void mov_data(u8 a[100],u8 b[100],u8 c); D% D9 {8 v$ b d* K# E" z { u8 i;" E" c7 M3 A6 \" B+ ?# i: Z for(i=c; i>0; i--) { a = b; }! N+ f0 E- j3 z" W } /////////////////////////////////////////////////////////////////////// 9 N" p( A* W' B7 Y: L7 l void Modbus_Function_3(void) 4 S/ L" f" O* R W6 e { / Y5 r1 ?! i5 F u16 tempdress = 0; u8 i = 3; * G6 C& H3 S( ]/ w+ S u16 crcresult; tempdress = (Uart0_rev_buff[2] = 0x0120) & (tempdress + Uart0_rev_buff[5] < 0x0132))2 y, @0 z9 U9 e3 a, f { Uart0_send_buff[0] = Com0_id; Uart0_send_buff[1] = 0x03; Uart0_send_buff[2] = 2 * Uart0_rev_buff[5]; ' ?8 w2 {- @" s1 ~; i. | Uart0_send_counter = 2 * Uart0_rev_buff[5] + 3; switch(tempdress)- A! C }9 s4 H; x+ F, `9 w( B9 f {. l7 G$ A% a4 _' a0 | case 0x0120:% \) F$ P4 ?% b0 t, e, L {0 g6 V" i' Y0 W2 s/ u7 y" E0 l9 ~) A Uart0_send_buff = Mkgz_bz & 0xff;) r5 }5 }$ [8 J' C i++;+ h7 y# \7 p/ a( h Uart0_send_buff = (Mkgz_bz >> 8) & 0xff;5 f, J$ ^7 j/ `5 Q i++; }//后面不放break的目的是继续往下执行. g( }' ]/ _' P( b9 n% m4 g case 0x0122:5 G- }* h( i2 E {) p5 u, d+ [3 e Uart0_send_buff = Out_voltage & 0xff; i++;" U) M% {+ l9 U" D Uart0_send_buff = (Out_voltage >> 8) & 0xff;9 c0 n$ w/ q" w) ^" D i++; } case 0x0124:) g1 E9 s; a w3 s {+ N5 b a# j; g# k Uart0_send_buff = Out_current & 0xff; i++; Uart0_send_buff = (Out_current >> 8) & 0xff; i++; 1 i9 t1 Z' E2 y6 q, s! E U }8 a. y& t; I" d# n! R- f case 0x0126: { Uart0_send_buff = Mkzt_bz & 0xff; i++; Uart0_send_buff = (Mkzt_bz >> 8) & 0xff;7 l+ U! h0 |: n8 s i++; 0 m+ Q, [. O( m) t } case 0x0128://这个地址是备用的里面的数据没有意义7 u' D6 w5 k) q5 S {- h3 t% o' `) F) b% v Uart0_send_buff = 0x00; i++; Uart0_send_buff = 0x00; i++; 4 ]) X8 M) q$ C3 c1 L }3 @$ S2 y5 j* Q& b* ^# K9 o case 0x012A:6 T7 ?2 t& D: \9 A2 E* k' E { Uart0_send_buff = OutX_current & 0xff;# V* o e' O* x O i++; Uart0_send_buff = (OutX_current >> 8) & 0xff; i++;# [3 j: {' G0 ]. {$ j } case 0x012C:9 N2 C" W$ y! a1 S. P { Uart0_send_buff = Jc_voltage & 0xff; i++; Uart0_send_buff = (Jc_voltage >> 8) & 0xff; i++;* f! n/ ^! P) S& S" I v9 v }" L5 @1 h9 p& T/ a; N case 0x012E:) u) R& {, T9 G# @/ ~ {( z2 l. [, ~8 N5 i4 Y5 Z Uart0_send_buff = Fc_voltage & 0xff;, `" F6 R6 S0 C! C$ y! ^ i++; Uart0_send_buff = (Fc_voltage >> 8) & 0xff; i++;1 n% m. }0 i3 K" u# K) a }) v+ n. I; p3 m case 0x0130: { Uart0_send_buff = 0x00;8 z# d2 e$ p1 [ i++;0 R1 w5 q5 {, e7 n* R( z/ s Uart0_send_buff = 0x00; i++; }8 n7 b8 w3 e* i$ [ [+ F. m } , F! m, N" n7 @5 c/ q //UCSRB |= (1 |
NUCLEO-H723ZG开发板试用 ——串口点灯测试
OpenBLT移植到STM32F103战舰开发板上适用于所有STM32F103系列的Bootloader
MultiButton移植到STM32F103战舰开发板
汇编浮点库qfplib移植STM32F769I-DISCO开发板与硬件浮点运算性能测试对比
【STM32MP157】从ST官方例程中分析RPMsg-TTY/SDB核间通信的使用方法
S2LP 的 sniff 模式如何同时满足通讯和低功耗要求
双证齐发!移远通信通过ISO 26262功能安全流程认证及产品认证
意法半导体 STHV200超声波 IC单片简化设计
了解NFC技术优势与应用场景
基于STM32软件实现底层函数经验分享—IIC通信
RE:STM32F的232口简易MODBUS-RTU通讯程序
RE:STM32F的232口简易MODBUS-RTU通讯程序
RE:STM32F的232口简易MODBUS-RTU通讯程序
RE:STM32F的232口简易MODBUS-RTU通讯程序
回复:STM32F的232口简易MODBUS-RTU通讯程序
为方便沟通,特此留下QQ1024829053
RE:STM32F的232口简易MODBUS-RTU通讯程序
{
u8 i;/ A: a. e3 \8 @1 w" q( q
for(i=c; i>0; i--)
{
a = b;) Z% G/ @$ B* }! A! T8 p4 {! M
}; H1 n) ^5 O/ K5 r/ q
}( J, {0 P8 D# T% Y% T/ Q
这个函数有问题吧? a、b分别是数组名称,能直接用a=b吗?