当前位置:   article > 正文

STM32F103C8T6的MODBUS-RTU通讯(485通讯)_stm32 modbus

stm32 modbus

1.功能简介

本次实验是使用STM32F103C8T6单片机的MODBUS-RTU通讯,通过串口助手的调试来获取寄存器的值并可以修改寄存器的值。

2.源码与资料(附有视频讲解)!!!!!!!

包括源码(KEIL5)和调试助手,看不懂程序还有老师视频手把手教学还等什么,还不心动吗!!!快上车吧!!!!!

链接:https://pan.baidu.com/s/1cWhJge_cqRfNOzAsogcqrw?pwd=NNNN 
提取码:NNNN

本次实验应用的程序打开“源码资料”文件夹——打开“MODBUS源码资料”——打开“MODBUS资料”——打开“STM32-MODBUS程序”文件夹——打开“MODBUS从机成功文件夹”——打开“MODBUS从机2.0”程序即可

3.所需元器件

元器件清单

                                                                 实物图

                      1                              2                                3                                       4

4.接线

4.1USB转串口线下载线

GND接GND

RXD接PA9

TXD接PA10

3V3接3.3V接口

4.2USB转485线

USB口与电脑的USB接口相连

端子部分

   图一                      

在端子部分的接线只需要T/R+和T/R-这两个端口,这两个端口根据图二的标识显示分别为A+与B-,这两个端子需要接到RS485转TTL串口模块的A与B接口即可。

4.3RS485转TTL串口模块

                       

图中上面的A与B接口如同上文所说接相同的A+与B-即可。

图中下面部分的接口如下所示

VCC-单片机3.3v接口

GND-单片机GND接口

TXD-PA3接口

RXD-PA2接口

GND-单品机3.3V接口

注意!注意!在通讯时不要只接USB转485的线,记得要同时接上USB转TTL串口的线,不然单片机没有供电!!!!!

5.源码解析

 5.1定时器

   5.1.1Timer.C

   定时器2初始化

  1. #include "timer.h"
  2. void Timer2_Init() //1ms产生1次更新事件
  3. {
  4. TIM_TimeBaseInitTypeDef timer;
  5. RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);
  6. TIM_DeInit(TIM2);
  7. timer.TIM_Period=1000-1;// 1ms
  8. timer.TIM_Prescaler=72-1;// 72M/72=1MHZ-->1us
  9. timer.TIM_ClockDivision=TIM_CKD_DIV1;
  10. timer.TIM_CounterMode=TIM_CounterMode_Up;
  11. TIM_TimeBaseInit(TIM2,&timer);
  12. TIM_Cmd(TIM2,ENABLE);
  13. TIM_ITConfig(TIM2, TIM_IT_Update,ENABLE);
  14. }

5.1.2Timer.h

  1. #ifndef _timer_
  2. #define _timer_
  3. #include "stm32f10x_conf.h"
  4. void Timer2_Init(void);
  5. #endif

 在主函数中写入了定时器2的中断服务子函数,1ms一次中断。

  1. void TIM2_IRQHandler()//定时器2的中断服务子函数 1ms一次中断
  2. {
  3. u8 st;
  4. st= TIM_GetFlagStatus(TIM2, TIM_FLAG_Update);
  5. if(st==SET)
  6. {
  7. TIM_ClearFlag(TIM2, TIM_FLAG_Update);
  8. if(modbus.timrun!=0)
  9. {
  10. modbus.timout++;
  11. if(modbus.timout>=8) //间隔时间达到了时间
  12. {
  13. modbus.timrun=0;//关闭定时器--停止定时
  14. modbus.reflag=1; //收到一帧数据
  15. }
  16. }
  17. }
  18. }

5.2串口初始化

在串口的口选择了PA2与PA3引脚

  1. void RS485_Init()
  2. {
  3. USART_InitTypeDef USART_InitStructure;
  4. GPIO_InitTypeDef GPIO_InitStructure;
  5. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOC|RCC_APB2Periph_AFIO, ENABLE);
  6. RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);
  7. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
  8. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  9. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//GPIO_Mode_IN_FLOATING;//GPIO_Mode_AF_OD;//
  10. GPIO_Init(GPIOA, &GPIO_InitStructure);
  11. RS485_RT_0; //使MAX485芯片处于接收状态
  12. //USART1_TX PB.10
  13. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
  14. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  15. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;//GPIO_Mode_Out_PP;//GPIO_Mode_IN_FLOATING;//GPIO_Mode_AF_OD;//
  16. GPIO_Init(GPIOA, &GPIO_InitStructure);
  17. //USART1_RX PB.11
  18. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
  19. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//GPIO_Mode_IPU;//
  20. GPIO_Init(GPIOA, &GPIO_InitStructure);
  21. //Usart1 NVIC ??
  22. USART_InitStructure.USART_BaudRate = 9600;//?????9600;
  23. USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  24. USART_InitStructure.USART_StopBits = USART_StopBits_1;
  25. USART_InitStructure.USART_Parity = USART_Parity_No;
  26. USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  27. USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
  28. USART_DeInit(USART2);
  29. USART_Init(USART2, &USART_InitStructure);
  30. USART_ITConfig(USART2,USART_IT_RXNE,ENABLE);
  31. USART_Cmd(USART2, ENABLE);
  32. USART_ClearFlag(USART2,USART_FLAG_TC );
  33. // GetAdd_rs485();
  34. // RS485_IsrInit(); //485?????
  35. }

5.3CRC校验

 CRC校验码

  1. //==========================================
  2. #include "modbusCRC.h"
  3. /* CRC 高位字节值表 */
  4. const uchar auchCRCHi[] = {
  5. 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
  6. 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
  7. 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
  8. 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
  9. 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
  10. 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
  11. 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
  12. 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
  13. 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
  14. 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
  15. 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
  16. 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
  17. 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
  18. 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
  19. 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
  20. 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
  21. 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
  22. 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
  23. 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
  24. 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
  25. 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
  26. 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
  27. 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
  28. 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
  29. 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
  30. 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40
  31. } ;
  32. /* CRC低位字节值表*/
  33. const uchar auchCRCLo[] = {
  34. 0x00, 0xC0, 0xC1, 0x01, 0xC3, 0x03, 0x02, 0xC2, 0xC6, 0x06,
  35. 0x07, 0xC7, 0x05, 0xC5, 0xC4, 0x04, 0xCC, 0x0C, 0x0D, 0xCD,
  36. 0x0F, 0xCF, 0xCE, 0x0E, 0x0A, 0xCA, 0xCB, 0x0B, 0xC9, 0x09,
  37. 0x08, 0xC8, 0xD8, 0x18, 0x19, 0xD9, 0x1B, 0xDB, 0xDA, 0x1A,
  38. 0x1E, 0xDE, 0xDF, 0x1F, 0xDD, 0x1D, 0x1C, 0xDC, 0x14, 0xD4,
  39. 0xD5, 0x15, 0xD7, 0x17, 0x16, 0xD6, 0xD2, 0x12, 0x13, 0xD3,
  40. 0x11, 0xD1, 0xD0, 0x10, 0xF0, 0x30, 0x31, 0xF1, 0x33, 0xF3,
  41. 0xF2, 0x32, 0x36, 0xF6, 0xF7, 0x37, 0xF5, 0x35, 0x34, 0xF4,
  42. 0x3C, 0xFC, 0xFD, 0x3D, 0xFF, 0x3F, 0x3E, 0xFE, 0xFA, 0x3A,
  43. 0x3B, 0xFB, 0x39, 0xF9, 0xF8, 0x38, 0x28, 0xE8, 0xE9, 0x29,
  44. 0xEB, 0x2B, 0x2A, 0xEA, 0xEE, 0x2E, 0x2F, 0xEF, 0x2D, 0xED,
  45. 0xEC, 0x2C, 0xE4, 0x24, 0x25, 0xE5, 0x27, 0xE7, 0xE6, 0x26,
  46. 0x22, 0xE2, 0xE3, 0x23, 0xE1, 0x21, 0x20, 0xE0, 0xA0, 0x60,
  47. 0x61, 0xA1, 0x63, 0xA3, 0xA2, 0x62, 0x66, 0xA6, 0xA7, 0x67,
  48. 0xA5, 0x65, 0x64, 0xA4, 0x6C, 0xAC, 0xAD, 0x6D, 0xAF, 0x6F,
  49. 0x6E, 0xAE, 0xAA, 0x6A, 0x6B, 0xAB, 0x69, 0xA9, 0xA8, 0x68,
  50. 0x78, 0xB8, 0xB9, 0x79, 0xBB, 0x7B, 0x7A, 0xBA, 0xBE, 0x7E,
  51. 0x7F, 0xBF, 0x7D, 0xBD, 0xBC, 0x7C, 0xB4, 0x74, 0x75, 0xB5,
  52. 0x77, 0xB7, 0xB6, 0x76, 0x72, 0xB2, 0xB3, 0x73, 0xB1, 0x71,
  53. 0x70, 0xB0, 0x50, 0x90, 0x91, 0x51, 0x93, 0x53, 0x52, 0x92,
  54. 0x96, 0x56, 0x57, 0x97, 0x55, 0x95, 0x94, 0x54, 0x9C, 0x5C,
  55. 0x5D, 0x9D, 0x5F, 0x9F, 0x9E, 0x5E, 0x5A, 0x9A, 0x9B, 0x5B,
  56. 0x99, 0x59, 0x58, 0x98, 0x88, 0x48, 0x49, 0x89, 0x4B, 0x8B,
  57. 0x8A, 0x4A, 0x4E, 0x8E, 0x8F, 0x4F, 0x8D, 0x4D, 0x4C, 0x8C,
  58. 0x44, 0x84, 0x85, 0x45, 0x87, 0x47, 0x46, 0x86, 0x82, 0x42,
  59. 0x43, 0x83, 0x41, 0x81, 0x80, 0x40
  60. } ;
  61. /******************************************************************
  62. 功能: CRC16校验
  63. 输入:
  64. 输出:
  65. ******************************************************************/
  66. uint crc16( uchar *puchMsg, uint usDataLen )
  67. {
  68. uchar uchCRCHi = 0xFF ; // 高CRC字节初始化
  69. uchar uchCRCLo = 0xFF ; // 低CRC 字节初始化
  70. unsigned long uIndex ; // CRC循环中的索引
  71. while ( usDataLen-- ) // 传输消息缓冲区
  72. {
  73. uIndex = uchCRCHi ^ *puchMsg++ ; // 计算CRC
  74. uchCRCHi = uchCRCLo ^ auchCRCHi[uIndex] ;
  75. uchCRCLo = auchCRCLo[uIndex] ;
  76. }
  77. return ( uchCRCHi << 8 | uchCRCLo ) ;
  78. }

 5.4MODBUS功能码与发送函数

    5.3.1MODBUS.C

     主要的功能代码都在这个函数,其中modbus.init()函数中的设备地址为4串口助手中的调试要记得改(如下图所示原来是1),不然调试不通!!!

另外两个函数则是写了我们在本次实验所需用到的两个功能码03功能码和06功能码。

MODBUS-RTU通讯共有如下图所示的功能码,在这次通讯中我们只需要读保持寄存器和写单个寄存器所以只用到了03和06功能码

                        

最后一个函数则是单片机接收和写入数据的函数。具体函数的原理我就不过多赘述了,大家可以去资料里的视频中去了解。

  1. #include "modbus.h"
  2. #include "modbus_uart.h"
  3. #include "modbusCRC.h"
  4. MODBUS modbus;
  5. extern u16 Reg[];
  6. /*
  7. 因为波特率 9600
  8. 1位数据的时间为 1000000us/9600bit/s=104us
  9. 一个字节为 104us*10位 =1040us
  10. 所以 MODBUS确定一个数据帧完成的时间为 1040us*3.5=3.64ms ->10ms
  11. */
  12. void Mosbus_Init()
  13. {
  14. modbus.myadd=4; //本从设备的地址
  15. modbus.timrun=0; //MODbus定时器停止计时
  16. RS485_Init();
  17. }
  18. void Modbud_fun3() //3号功能码处理 ---主机要读取本从机的寄存器
  19. {
  20. u16 Regadd;
  21. u16 Reglen;
  22. u16 byte;
  23. u16 i,j;
  24. u16 crc;
  25. Regadd=modbus.rcbuf[2]*256+modbus.rcbuf[3]; //得到要读取的寄存器的首地址
  26. Reglen=modbus.rcbuf[4]*256+modbus.rcbuf[5]; //得到要读取的寄存器的数量
  27. i=0;
  28. modbus.Sendbuf[i++]=modbus.myadd;//本设备地址
  29. modbus.Sendbuf[i++]=0x03; //功能码
  30. byte=Reglen*2; //要返回的数据字节数
  31. //modbus.Sendbuf[i++]=byte/256; //
  32. modbus.Sendbuf[i++]=byte%256;
  33. for(j=0;j<Reglen;j++)
  34. {
  35. modbus.Sendbuf[i++]=Reg[Regadd+j]/256;
  36. modbus.Sendbuf[i++]=Reg[Regadd+j]%256;
  37. }
  38. crc=crc16(modbus.Sendbuf,i);
  39. modbus.Sendbuf[i++]=crc/256; //
  40. modbus.Sendbuf[i++]=crc%256;
  41. RS485_RT_1; //
  42. for(j=0;j<i;j++)
  43. {
  44. RS485_byte(modbus.Sendbuf[j]);
  45. }
  46. RS485_RT_0; //
  47. }
  48. void Modbud_fun6() //6号功能码处理
  49. {
  50. u16 Regadd;
  51. u16 val;
  52. u16 i,crc,j;
  53. i=0;
  54. Regadd=modbus.rcbuf[2]*256+modbus.rcbuf[3]; //得到要修改的地址
  55. val=modbus.rcbuf[4]*256+modbus.rcbuf[5]; //修改后的值
  56. Reg[Regadd]=val; //修改本设备相应的寄存器
  57. //以下为回应主机
  58. modbus.Sendbuf[i++]=modbus.myadd;//本设备地址
  59. modbus.Sendbuf[i++]=0x06; //功能码
  60. modbus.Sendbuf[i++]=Regadd/256;
  61. modbus.Sendbuf[i++]=Regadd%256;
  62. modbus.Sendbuf[i++]=val/256;
  63. modbus.Sendbuf[i++]=val%256;
  64. crc=crc16(modbus.Sendbuf,i);
  65. modbus.Sendbuf[i++]=crc/256; //
  66. modbus.Sendbuf[i++]=crc%256;
  67. RS485_RT_1; //
  68. for(j=0;j<i;j++)
  69. {
  70. RS485_byte(modbus.Sendbuf[j]);
  71. }
  72. RS485_RT_0; //
  73. }
  74. void Mosbus_Event()
  75. {
  76. u16 crc;
  77. u16 rccrc;
  78. if(modbus.reflag==0) //没有收到MODbus的数据包
  79. {
  80. return ;
  81. }
  82. crc= crc16(&modbus.rcbuf[0], modbus.recount-2); //计算校验码
  83. rccrc=modbus.rcbuf[modbus.recount-2]*256 + modbus.rcbuf[modbus.recount-1]; //收到的校验码
  84. if(crc == rccrc) //数据包符号CRC校验规则
  85. {
  86. if(modbus.rcbuf[0] == modbus.myadd) //确认数据包是否是发给本设备的
  87. {
  88. switch(modbus.rcbuf[1]) //分析功能码
  89. {
  90. case 0: break;
  91. case 1: break;
  92. case 2: break;
  93. case 3: Modbud_fun3(); break; //3号功能码处理
  94. case 4: break;
  95. case 5: break;
  96. case 6: Modbud_fun6(); break; //6号功能码处理
  97. case 7: break;
  98. //....
  99. }
  100. }
  101. else if(modbus.rcbuf[0] == 0) //广播地址
  102. {
  103. }
  104. }
  105. modbus.recount=0; //
  106. modbus.reflag=0;
  107. }

5.3.2MODBUS.H

  1. #ifndef _modbus_
  2. #define _modbus_
  3. #include "stm32f10x_conf.h"
  4. #define RS485_RT_1 GPIO_SetBits(GPIOA, GPIO_Pin_5) //485发送状态
  5. #define RS485_RT_0 GPIO_ResetBits(GPIOA, GPIO_Pin_5) //485置接收状态
  6. typedef struct
  7. {
  8. u8 myadd;//本设备的地址
  9. u8 rcbuf[100]; //MODBUS接收缓冲区
  10. u16 timout;//MODbus的数据断续时间
  11. u8 recount;//MODbus端口已经收到的数据个数
  12. u8 timrun;//MODbus定时器是否计时的标志
  13. u8 reflag;//收到一帧数据的标志
  14. u8 Sendbuf[100]; //MODbus发送缓冲区
  15. }MODBUS;
  16. extern MODBUS modbus;
  17. void Mosbus_Init(void);
  18. void Mosbus_Event(void);
  19. #endif

5.5主函数

其中数组Reg[ ]中的值就是我们所需读写的值。

  1. //#include "stm32f10x.h" // Device header
  2. #include "timer.h"
  3. #include "modbus_uart.h"
  4. #include "modbus.h"
  5. u16 Reg[]={0x0000, //本设备寄存器中的值
  6. 0x0001,
  7. 0x0002,
  8. 0x0007,
  9. 0x0004,
  10. 0x0005,
  11. 0x0006,
  12. 0x0007,
  13. 0x0008,
  14. 0x0009,
  15. 0x000A,
  16. };
  17. void delay(u32 x)
  18. {
  19. while(x--);
  20. }
  21. void Isr_Init()
  22. {
  23. NVIC_InitTypeDef isr;
  24. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1); // a bbb
  25. isr.NVIC_IRQChannel=TIM2_IRQn;
  26. isr.NVIC_IRQChannelCmd=ENABLE;
  27. isr.NVIC_IRQChannelPreemptionPriority=1;
  28. isr.NVIC_IRQChannelSubPriority=2;
  29. NVIC_Init(&isr); //
  30. isr.NVIC_IRQChannel=USART2_IRQn;
  31. isr.NVIC_IRQChannelCmd=ENABLE;
  32. isr.NVIC_IRQChannelPreemptionPriority=1;
  33. isr.NVIC_IRQChannelSubPriority=0;
  34. NVIC_Init(&isr); //
  35. }
  36. void TIM2_IRQHandler()//定时器2的中断服务子函数 1ms一次中断
  37. {
  38. u8 st;
  39. st= TIM_GetFlagStatus(TIM2, TIM_FLAG_Update);
  40. if(st==SET)
  41. {
  42. TIM_ClearFlag(TIM2, TIM_FLAG_Update);
  43. if(modbus.timrun!=0)
  44. {
  45. modbus.timout++;
  46. if(modbus.timout>=8) //间隔时间达到了时间
  47. {
  48. modbus.timrun=0;//关闭定时器--停止定时
  49. modbus.reflag=1; //收到一帧数据
  50. }
  51. }
  52. }
  53. }
  54. int main()
  55. {
  56. Timer2_Init();
  57. Mosbus_Init();
  58. Isr_Init();
  59. while(1)
  60. {
  61. Mosbus_Event(); //处理MODbus数据
  62. // RS485_byte('B');
  63. }
  64. }

 6串口助手调试

先在串口助手选对应的COM口并打开串口,不知道具体是哪个COM口的可以在设备管理器中找到。

                                  

之后将串口调试助手的设备地址改为4。

之后即可开始读写操作

读操作例:

寄存器地址就是主函数中的REG数组,地址0就是第1个数,这里的地址3就是第4个数0x07,而数量就是一次性要读出多少个数组。

读出之后发送端会显示如下图所示,04则是之前的设备地址,03是功能码,后面则是两位为一组,00 03 则表示寄存器地址是3,00 01则表示要读出的数量,最后面两位则是CRC校验码。

如果通讯成功,在下面的接收区则会接收到单片机发来的数据,如果没有接收到数据则要检查接线是否错误。

在上面的写寄存器区也是相同的操作,不过的是一次只能写入单个寄存器,同样如果写入成功则会接收到单片机发回的值。

如果操作成功大家可以反复先读取再写入然后再次读取去验证数值是否已经被修改完成。

                                                                                                                                            By:凌浩

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/很楠不爱3/article/detail/489265
推荐阅读
相关标签
  

闽ICP备14008679号