你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

【经验分享】STM32F103单片机modbus通信示例

[复制链接]
STMCU小助手 发布时间:2022-3-20 21:44
      前两天在研究STM32F103单片机的串口空闲中断时,突然想起来Modbus通信非常适合用空闲中断来处理。先看看Modbus RTU模式下的通信规范。

X}VKZB)JFT7OP1MK]D~`QKS.png

       可以看到Modbus RTU通信模式下,数据的开始和结束是由空闲字符间隔时间来区分的,而STM32F103单片机自带串口空闲模式检测。

}(H52OS$DGQ%ZC95O5%4X3R.png

        在通常情况下Modbus通信一帧数据的检测可以用时间判断,不停的去读取接收数据的长度是否发送变化,如果在一定时间内,接收数据的长度没有发生变化,就认为一帧数据结束完毕。

        通过时间来判断一帧数据是否接收完成,会导致单片机不停的进入中断。而STM32F103单片机自带了空闲数据检测功能,这个功能是由单片机内部的硬件完成的,不需要程序去处理,这样单片机的就有机会去执行更多了任务了。为了让单片机在处理Modbus RTU通信时更简单快速,可以用 串口DMA加上空闲中断的方式来实现。

        下面直接用代码说明具体的实现过程,首先初始化串口。

  1. #define UART2_DMA  1                                                                                                                                                                //使用串口2  DMA传输
  2. #define  CHECK_NONE_ONE_STOP    1 //无校验位  1个停止位  1有效  0 无效
  3. #define  CHECK_NONE_TWO_STOP    0 //无校验位  2个停止位  1有效  0 无效
  4. #define  CHECK_EVEN    0          //偶数校验   1有效  0 无效
  5. #define  CHECK_ODD     0          //奇数校验   1有效  0 无效


  6. void  uart2_init( u16 baud )
  7. {
  8.     GPIO_InitTypeDef GPIO_InitStructure;
  9.     USART_InitTypeDef USART_InitStructure;
  10.     NVIC_InitTypeDef  NVIC_InitStructure;

  11.     RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOA, ENABLE );
  12.     RCC_APB1PeriphClockCmd( RCC_APB1Periph_USART2, ENABLE );

  13.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
  14.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;                                                                                                 //推挽复用模式
  15.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  16.     GPIO_Init( GPIOA, &GPIO_InitStructure );

  17.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
  18.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;                                                                         //浮空输入模式
  19.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  20.     GPIO_Init( GPIOA, &GPIO_InitStructure );

  21.     NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;
  22.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x02;
  23.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x02;
  24.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  25.     NVIC_Init( &NVIC_InitStructure );

  26.     USART_InitStructure.USART_BaudRate = baud;
  27.     USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  28.     USART_InitStructure.USART_StopBits = USART_StopBits_1;
  29.     USART_InitStructure.USART_Parity = USART_Parity_No;
  30.                
  31. #if(CHECK_EVEN == 1)                                                                                                                                                                                                                          //如果定义了偶校验  数据位长度要改为9位
  32.     USART_InitStructure.USART_WordLength = USART_WordLength_9b;
  33.     USART_InitStructure.USART_Parity = USART_Parity_Even;
  34. #endif

  35. #if(CHECK_ODD == 1)                                                                                                                                                                                                                          //如果定义了奇校验  数据位长度要改为9位
  36.     USART_InitStructure.USART_WordLength = USART_WordLength_9b;
  37.     USART_InitStructure.USART_Parity = USART_Parity_Odd;
  38. #endif

  39. #if(CHECK_NONE_ONE_STOP==1)                                                                                                                                                                                           //停止位为 一位
  40.     USART_InitStructure.USART_StopBits = USART_StopBits_1;
  41. #endif

  42. #if(CHECK_NONE_TWO_STOP==1)                                                                                                                                                                                                 //停止位为 两位                                                                                                                                       
  43.     USART_InitStructure.USART_StopBits = USART_StopBits_2;
  44. #endif               
  45.                
  46.                
  47.     USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
  48.     USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;

  49.     USART_Init( USART2, &USART_InitStructure );
  50.                
  51. #if (UART2_DMA == 1)
  52.     USART_ITConfig( USART2, USART_IT_IDLE, ENABLE );                                                                                         //使能串口空闲中断
  53.     USART_DMACmd( USART2, USART_DMAReq_Rx, ENABLE );                                                                                         //使能串口2 DMA接收
  54.     uartDMA_Init();                                                                                                                                                                                                                                 //初始化 DMA
  55. #else
  56.     USART_ITConfig( USART2, USART_IT_RXNE, ENABLE );                                                                                   //使能串口RXNE接收中断
  57. #endif
  58.     USART_Cmd( USART2, ENABLE );                                                                                                                                                                         //使能串口2
  59.                
  60. //RXNE中断和IDLE中断的区别?
  61. //当接收到1个字节,就会产生RXNE中断,当接收到一帧数据,就会产生IDLE中断。比如给单片机一次性发送了8个字节,就会产生8次RXNE中断,1次IDLE中断。
  62. }
复制代码

        在Modbus通信时,需要对每个字节的数据进行校验,所以这里使用宏定义来初始化串口数据使用的是那种校验方式。如果需要偶校验就将CHECK_EVEN 设置为1,如果需要奇校验就将CHECK_ODD设置为1,如果不需要校验位,就将这两个都设置为0。在串口初始化的时候,需要开启空闲中断和DMA功能。

        下来初始化 串口2的DMA功能

  1. void uartDMA_Init( void )
  2. {
  3.     DMA_InitTypeDef  DMA_IniStructure;

  4.     RCC_AHBPeriphClockCmd( RCC_AHBPeriph_DMA1, ENABLE );                                                                          //使能DMA时钟

  5.     DMA_DeInit( DMA1_Channel6 );                                                                                                                                                             //DMA1通道6对应 USART2_RX
  6.     DMA_IniStructure.DMA_PeripheralBaseAddr = ( u32 )&USART2->DR;                                         //DMA外设usart基地址
  7.     DMA_IniStructure.DMA_MemoryBaseAddr = ( u32 )dma_rec_buff;                                                  //DMA内存基地址
  8.     DMA_IniStructure.DMA_DIR = DMA_DIR_PeripheralSRC;                                                                                         //数据传输方向,从外设读取发送到内存
  9.     DMA_IniStructure.DMA_BufferSize = DMA_REC_LEN;                                                                                                 //DMA通道的DMA缓存的大小  也就是 DMA一次传输的字节数
  10.     DMA_IniStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;                                 //外设地址寄存器不变
  11.     DMA_IniStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;                                                                 //数据缓冲区地址递增
  12.     DMA_IniStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; //外设以字节为单位搬运
  13.     DMA_IniStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;                                  //数据缓冲区以字节为单位搬入
  14.     DMA_IniStructure.DMA_Mode = DMA_Mode_Normal;                                                                                                         //工作在正常缓存模式
  15.     DMA_IniStructure.DMA_Priority = DMA_Priority_Medium;                                                                         //DMA通道 x拥有中优先级
  16.     DMA_IniStructure.DMA_M2M = DMA_M2M_Disable;                                                                                                                 //DMA通道x没有设置为内存到内存传输

  17.     DMA_Init( DMA1_Channel6, &DMA_IniStructure );
  18.     DMA_Cmd( DMA1_Channel6, ENABLE );
  19. }
复制代码

GT9L~X56176YOAZQS$LM5QR.png

        串口2的接收对应的是DMA的通道6,所以DMA的通道选择通道6,然后将串口2的数据寄存器作为外设基地址,将数组dma_rec_buff作为DMA的内存地址。外设地址不自增,内存地址自增。这样串口2每接收一个字节的数据,DMA就将接收到的数据存入数组dma_rec_buff中,同时指向数组的地址增加一位,这样串口2接收到的数据就会被一直存放在数组dma_rec_buff中。

        这里要注意一点,数组的缓存区要大于串口2接收到的最长数据个数,否则如果接收到串口数据比较长,而数组比较小,数组中的数据就会被覆盖,导致接收到的数据错误。

     接下来就是串口中断的实现了

  1. void USART2_IRQHandler( void )
  2. {
  3.     u8 tem = 0;
  4. #if (UART2_DMA == 1)                        //如果使能了 UART2_DMA 则使用串口空闲中断和 DMA功能接收数据
  5.         if( USART_GetITStatus( USART2, USART_IT_IDLE ) != RESET )                                                                                   //空闲中断  一帧数据发送完成
  6.     {
  7.         USART_ReceiveData( USART2 );                                                                                                                                                                                 //读取数据注意:这句必须要,否则不能够清除空闲中断标志位。
  8.         DMA_Cmd( DMA1_Channel6, DISABLE );                                                                                                                                                         //关闭 DMA 防止后续数据干扰
  9.         uart2_rec_cnt = DMA_REC_LEN - DMA_GetCurrDataCounter( DMA1_Channel6 );  //DMA接收缓冲区数据长度减去当前 DMA传输通道中剩余单元数量就是已经接收到的数据数量

  10.         copy_data( dma_rec_buff, uart2_rec_cnt );                                                                                                                    //备份数据
  11.         receiveOK_flag = 1;                                                                                                                                                                                                                        //置位数据接收完成标志位
  12.         USART_ClearITPendingBit( USART2, USART_IT_IDLE );                                                                                                 //清除空闲中断标志位
  13.         myDMA_Enable( DMA1_Channel6 );                                                                                                                                                                         //重新恢复 DMA等待下一次接收
  14.     }
  15. #else                                                                                //如果未使能 UART2_DMA 则通过常规方式接收数据  每接收到一个字节就会进入一次中断
  16.     if( USART_GetITStatus( USART2, USART_IT_RXNE ) != RESET )                                                 //接收中断
  17.     {
  18.         tem = USART_ReceiveData( USART2 );
  19.         USART_SendData( USART2, tem );
  20.     }
  21. #endif
  22. }
复制代码

        串口进入空闲中断后,说明一帧数据已经接收完成。关闭DMA通道,开始处理数据。获取DMA中接收到的数据长度,然后将数据拷贝到备份数组中,然后置接收完成标志位。开启 DMA进行下一次数据接收。

        在主程序中如果检测到了数据接收完成标志时,就开始处理接收到的数据。这样就可以将串口接收数据和处理数据分开,在处理上一笔数据的时候,就可以继续接收下一笔数据。提高了系统的实时性。

        这里如果直接使用DMA接收数组去处理数据的话,有可能会出现,旧的一笔数据未处理完成时,DMA接收到了新的一笔数据,然后存储到了 DMA缓存数组中。这样在处理数据过程中,旧的数据就会被新的数据覆盖掉,导致系统异常。如果Modbus的通信速率没有那么快,在下一次数据来临之前,系统完全有时间处理完上一笔数据,那么直接使用DMA的缓存数据也是可以的。

       不过上面这种将数据拷贝倒其他数据中处理的方法也是有风险的。比如第一笔数据还没处理完成,第二笔数据已经接收完成,此时进入串口空闲中断,系统依然会将第二笔数据拷贝到备份数组中去。这样备份数组中的数据依然会被覆盖。但是这种情况如果出现的话,就说明接收数据的频率高于系统处理数据的频率,系统本身就存在风险,需要重新设计。

     接收数据完成后,就需要在主程序中调用数据处理函数。

  1. int main(void)
  2. {
  3.     u8  j = 0;
  4.         
  5.         NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
  6.         
  7.     delay_init();       //延时函数初始化
  8.     LED_Init();         //初始化与LED连接的硬件接口
  9.     uart2_init(9600);
  10.     while(1)
  11.     {

  12.         if( receiveOK_flag )                                //一帧数据接收完成后开始处理数据
  13.         {
  14.             receiveOK_flag = 0;
  15.             DisposeReceive();                                //处理modbus通信
  16.         }
  17.                         
  18.         j++;
  19.         if(j > 50)
  20.         {
  21.             j = 0;
  22.             LED = !LED;
  23.         }
  24.         delay_ms(10);
  25.     }
  26. }
复制代码

最后是Modbus的协议解析
  1. void DisposeReceive( void )
  2. {
  3.     u16 CRC16 = 0, CRC16Temp = 0;
  4.     if( data_backup[0] == SlaveID )                                                                         //地址等于本机地址 地址范围:1 - 32
  5.     {
  6.         CRC16 = App_Tab_Get_CRC16( data_backup, dataLen_backup - 2 );                                  //CRC校验 低字节在前 高字节在后 高字节为报文最后一个字节
  7.         CRC16Temp = ( ( u16 )( data_backup[dataLen_backup - 1] << 8 ) | data_backup[dataLen_backup - 2] );
  8.         if( CRC16 != CRC16Temp )
  9.         {
  10.             err = 4;                                                                                               //CRC校验错误
  11.         }
  12.         StartRegAddr = ( u16 )( data_backup[2] << 8 ) | data_backup[3];
  13.         if( StartRegAddr > (HoldRegStartAddr + HoldRegCount - 1) )
  14.         {
  15.             err = 2;                                                                                               //起始地址不在规定范围内 00 - 07    1 - 8号通道
  16.         }
  17.         if( err == 0 )
  18.         {
  19.             switch( data_backup[1] )                                                                        //功能码
  20.             {
  21.                 case 3:                                                                                            //读多个寄存器
  22.                 {
  23.                     Modbus_03_Slave();
  24.                     break;
  25.                 }
  26.                 case 6:                                                                                            //写单个寄存器
  27.                 {
  28.                     Modbus_06_Slave();
  29.                     break;
  30.                 }
  31.                 case 16:                                                                                           //写多个寄存器
  32.                 {
  33.                     Modbus_16_Slave();
  34.                     break;
  35.                 }
  36.                 default:
  37.                 {
  38.                     err = 1;                                                                                       //不支持该功能码
  39.                     break;
  40.                 }
  41.             }
  42.         }
  43.         if( err > 0 )
  44.         {
  45.             SendBuf[0] = data_backup[0];
  46.             SendBuf[1] = data_backup[1] | 0x80;
  47.             SendBuf[2] = err;                                                                                      //发送错误代码
  48.             CRC16Temp = App_Tab_Get_CRC16( SendBuf, 3 );                                                           //计算CRC校验值
  49.             SendBuf[3] = CRC16Temp & 0xFF;                                                                         //CRC低位
  50.             SendBuf[4] = ( CRC16Temp >> 8 );                                                                       //CRC高位
  51.             uart2_Send( SendBuf, 5 );                                       
  52.             err = 0;                                                                                               //发送完数据后清除错误标志
  53.         }
  54.     }
  55. }

  56. /*
  57. 函数功能:读保持寄存器  03
  58. 主站请求报文:      0x01 0x03   0x0000  0x0001  0x840A     读从0开始的1个保持寄存器
  59. 从站正常响应报文:  0x01 0x03   0x02    0x09C4  0xBF87     读到的2字节数据为 0x09C4
  60. */
  61. void Modbus_03_Slave( void )
  62. {
  63.     u16 RegNum = 0;
  64.     u16 CRC16Temp = 0;
  65.     u8 i = 0;
  66.     RegNum = ( u16 )( data_backup[4] << 8 ) | data_backup[5];                                                //获取寄存器数量
  67.     if( ( StartRegAddr + RegNum ) <= (HoldRegStartAddr + HoldRegCount) )      //寄存器地址+寄存器数量 在规定范围内 <=8
  68.     {
  69.         SendBuf[0] = data_backup[0];                                                                                                                                                                        //地址        
  70.         SendBuf[1] = data_backup[1];                                                                                                                                                                        //功能码
  71.         SendBuf[2] = RegNum * 2;                                                                                                                                                                                        //返回字节数量
  72.         for( i = 0; i < RegNum; i++ )                                                                             //循环读取保持寄存器内的值
  73.         {
  74.             SendBuf[3 + i * 2] = HoldReg[StartRegAddr * 2 + i * 2];
  75.             SendBuf[4 + i * 2] = HoldReg[StartRegAddr * 2 + i * 2 + 1];
  76.         }
  77.         CRC16Temp = App_Tab_Get_CRC16( SendBuf, RegNum * 2 + 3 );                                                  //获取CRC校验值
  78.         SendBuf[RegNum * 2 + 3] = CRC16Temp & 0xFF;                                                                //CRC低位
  79.         SendBuf[RegNum * 2 + 4] = ( CRC16Temp >> 8 );                                                              //CRC高位
  80.         uart2_Send( SendBuf, RegNum * 2 + 5 );
  81.     }
  82.     else
  83.     {
  84.         err = 3;                                                                                                   //寄存器数量不在规定范围内
  85.     }
  86. }
  87. /*
  88. 函数功能:写单个保持寄存器 06
  89. 主站请求报文:      0x01 0x06    0x0000  0x1388    0x849C   写0号寄存器的值为0x1388
  90. 从站正常响应报文:  0x01 0x06    0x0000  0x1388    0x849C    0号寄存器的值为0x1388
  91. */
  92. void Modbus_06_Slave( void )
  93. {
  94.     u16  RegValue = 0;
  95.     u16 CRC16Temp = 0;
  96.     RegValue = ( u16 )( data_backup[4] << 8 ) | data_backup[5];                                              //获取寄存器值
  97.     if( RegValue <= HoldMaxValue )                                                  //寄存器值不超过1000
  98.     {
  99.         HoldReg[StartRegAddr * 2] = data_backup[4];                                                         //存储寄存器值
  100.         HoldReg[StartRegAddr * 2 + 1] = data_backup[5];
  101.         SendBuf[0] = data_backup[0];                                                                                                                                                                        //地址
  102.         SendBuf[1] = data_backup[1];                                                                                                                                                                        //功能码
  103.         SendBuf[2] = data_backup[2];                                                                                                                                                                        //地址高位
  104.         SendBuf[3] = data_backup[3];                                                                                                                                                                        //地址低位
  105.         SendBuf[4] = data_backup[4];                                                                                                                                                                        //值高位
  106.         SendBuf[5] = data_backup[5];                                                                                                                                                                        //值低位
  107.         CRC16Temp = App_Tab_Get_CRC16( SendBuf, 6 );                                                              //获取CRC校验值
  108.         SendBuf[6] = CRC16Temp & 0xFF;                                                                            //CRC低位
  109.         SendBuf[7] = ( CRC16Temp >> 8 );                                                                          //CRC高位
  110.         uart2_Send( SendBuf, 8 );
  111.     }
  112.     else
  113.     {
  114.         err =  3;                                                                                                  //寄存器数值不在规定范围内
  115.     }
  116. }
  117. /*
  118. 函数功能:写多个连续保持寄存器值 16
  119. 主站请求报文:       0x01 0x10    0x7540  0x0002  0x04  0x0000 0x2710    0xB731  写从0x7540地址开始的2个保持寄存器值 共4字节
  120. 从站正常响应报文:   0x01 0x10    0x7540  0x0002  0x5A10                         写从0x7540地址开始的2个保持寄存器值
  121. */
  122. void Modbus_16_Slave( void )
  123. {
  124.     u16 RegNum = 0;
  125.     u16 CRC16Temp = 0;
  126.     u8 i = 0;
  127.     RegNum = ( u16 )( data_backup[4] << 8 ) | data_backup[5];                                                //获取寄存器数量
  128.     if( ( StartRegAddr + RegNum ) <= (HoldRegStartAddr + HoldRegCount) )             //寄存器地址+寄存器数量 在规定范围内 <=8
  129.     {
  130.         for( i = 0; i < RegNum; i++ )                                                                              //存储寄存器设置值
  131.         {
  132.             HoldReg[StartRegAddr * 2 + i * 2] = data_backup[i * 2 + 7];
  133.             HoldReg[StartRegAddr * 2 + 1 + i * 2] = data_backup[i * 2 + 8];
  134.         }
  135.         SendBuf[0] = data_backup[0];                                                                                                                                                                        //起始地址
  136.         SendBuf[1] = data_backup[1];                                                                                                                                                                        //功能码
  137.         SendBuf[2] = data_backup[2];                                                                                                                                                                        //地址高位
  138.         SendBuf[3] = data_backup[3];                                                                                                                                                                        //地址低位
  139.         SendBuf[4] = data_backup[4];                                                                                                                                                                        //寄存器数量高位
  140.         SendBuf[5] = data_backup[5];                                                                                                                                                                        //寄存器数量低位
  141.         CRC16Temp = App_Tab_Get_CRC16( SendBuf, 6 );                                                        //获取CRC校验值
  142.         SendBuf[6] = CRC16Temp & 0xFF;                                                                      //CRC低位
  143.         SendBuf[7] = ( CRC16Temp >> 8 );                                                                    //CRC高位
  144.         uart2_Send( SendBuf, 8 );
  145.     }
  146.     else
  147.     {
  148.         err = 3;                                                                                                   //寄存器数量不在规定范围内
  149.     }
  150. }
复制代码

Modbus的执行在这里只实现了最常用的三个。

HYI%R~(IXBRGFE3Y}7K{U$U.png

各个功能码的实现原理在这里就不说了,直接通过代码中和注释就可以看明白了。

下面贴出串口和modbus的完整代码

uart2.h

  1. #ifndef __UART2_H
  2. #define __UART2_H
  3. #include "sys.h"

  4. #define DMA_REC_LEN 50                                                                                                                                                        //DMA数据接收缓冲区

  5. void  uart2_init(u16 baud);
  6. void uartDMA_Init( void );
  7. void myDMA_Enable( DMA_Channel_TypeDef*DMA_CHx );
  8. void uart2_Send( u8 *buf, u16 len );
  9. void copy_data( u8 *buf, u16 len );
  10. #endif
复制代码

uart2.c

  1. //串口2空闲中断 + DMA数据传输
  2. #include "uart2.h"
  3. #define UART2_DMA  1                                                                                                                                                                //使用串口2  DMA传输
  4. #define  CHECK_NONE_ONE_STOP    1 //无校验位  1个停止位  1有效  0 无效
  5. #define  CHECK_NONE_TWO_STOP    0 //无校验位  2个停止位  1有效  0 无效
  6. #define  CHECK_EVEN    0          //偶数校验   1有效  0 无效
  7. #define  CHECK_ODD     0          //奇数校验   1有效  0 无效

  8. u8 dma_rec_buff[DMA_REC_LEN] = {0};
  9. u16 uart2_rec_cnt = 0;                                                                                                                                                        //串口接收数据长度

  10. u8 data_backup[DMA_REC_LEN] = {0};                                                                                                         //数据备份
  11. u16 dataLen_backup = 0;                                                                                                                                                        //长度备份
  12. _Bool receiveOK_flag = 0;                                                                                                                                                //接收完成标志位

  13. /*
  14. 空闲中断是什么意思呢?
  15. 指的是当总线接收数据时,一旦数据流断了,此时总线没有接收传输,处于空闲状态,IDLE就会置1,产生空闲中断;又有数据发送时,IDLE位就会置0;
  16. 注意:置1之后它不会自动清0,也不会因为状态位是1而一直产生中断,它只有0跳变到1时才会产生,也可以理解为上升沿触发。
  17. 所以,为确保下次空闲中断正常进行,需要在中断服务函数发送任意数据来清除标志位。
  18. */

  19. void  uart2_init( u16 baud )
  20. {
  21.     GPIO_InitTypeDef GPIO_InitStructure;
  22.     USART_InitTypeDef USART_InitStructure;
  23.     NVIC_InitTypeDef  NVIC_InitStructure;

  24.     RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOA, ENABLE );
  25.     RCC_APB1PeriphClockCmd( RCC_APB1Periph_USART2, ENABLE );

  26.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
  27.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;                                                                                                 //推挽复用模式
  28.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  29.     GPIO_Init( GPIOA, &GPIO_InitStructure );

  30.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
  31.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;                                                                         //浮空输入模式
  32.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  33.     GPIO_Init( GPIOA, &GPIO_InitStructure );

  34.     NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;
  35.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x02;
  36.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x02;
  37.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  38.     NVIC_Init( &NVIC_InitStructure );

  39.     USART_InitStructure.USART_BaudRate = baud;
  40.     USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  41.     USART_InitStructure.USART_StopBits = USART_StopBits_1;
  42.     USART_InitStructure.USART_Parity = USART_Parity_No;
  43.                
  44. #if(CHECK_EVEN == 1)                                                                                                                                                                                                                          //如果定义了偶校验  数据位长度要改为9位
  45.     USART_InitStructure.USART_WordLength = USART_WordLength_9b;
  46.     USART_InitStructure.USART_Parity = USART_Parity_Even;
  47. #endif

  48. #if(CHECK_ODD == 1)                                                                                                                                                                                                                          //如果定义了奇校验  数据位长度要改为9位
  49.     USART_InitStructure.USART_WordLength = USART_WordLength_9b;
  50.     USART_InitStructure.USART_Parity = USART_Parity_Odd;
  51. #endif

  52. #if(CHECK_NONE_ONE_STOP==1)                                                                                                                                                                                           //停止位为 一位
  53.     USART_InitStructure.USART_StopBits = USART_StopBits_1;
  54. #endif

  55. #if(CHECK_NONE_TWO_STOP==1)                                                                                                                                                                                                 //停止位为 两位                                                                                                                                       
  56.     USART_InitStructure.USART_StopBits = USART_StopBits_2;
  57. #endif               
  58.                
  59.                
  60.     USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
  61.     USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;

  62.     USART_Init( USART2, &USART_InitStructure );
  63.                
  64. #if (UART2_DMA == 1)
  65.     USART_ITConfig( USART2, USART_IT_IDLE, ENABLE );                                                                                         //使能串口空闲中断
  66.     USART_DMACmd( USART2, USART_DMAReq_Rx, ENABLE );                                                                                         //使能串口2 DMA接收
  67.     uartDMA_Init();                                                                                                                                                                                                                                 //初始化 DMA
  68. #else
  69.     USART_ITConfig( USART2, USART_IT_RXNE, ENABLE );                                                                                   //使能串口RXNE接收中断
  70. #endif
  71.     USART_Cmd( USART2, ENABLE );                                                                                                                                                                         //使能串口2
  72.                
  73. //RXNE中断和IDLE中断的区别?
  74. //当接收到1个字节,就会产生RXNE中断,当接收到一帧数据,就会产生IDLE中断。比如给单片机一次性发送了8个字节,就会产生8次RXNE中断,1次IDLE中断。
  75. }

  76. void uartDMA_Init( void )
  77. {
  78.     DMA_InitTypeDef  DMA_IniStructure;

  79.     RCC_AHBPeriphClockCmd( RCC_AHBPeriph_DMA1, ENABLE );                                                                          //使能DMA时钟

  80.     DMA_DeInit( DMA1_Channel6 );                                                                                                                                                             //DMA1通道6对应 USART2_RX
  81.     DMA_IniStructure.DMA_PeripheralBaseAddr = ( u32 )&USART2->DR;                                         //DMA外设usart基地址
  82.     DMA_IniStructure.DMA_MemoryBaseAddr = ( u32 )dma_rec_buff;                                                  //DMA内存基地址
  83.     DMA_IniStructure.DMA_DIR = DMA_DIR_PeripheralSRC;                                                                                         //数据传输方向,从外设读取发送到内存
  84.     DMA_IniStructure.DMA_BufferSize = DMA_REC_LEN;                                                                                                 //DMA通道的DMA缓存的大小  也就是 DMA一次传输的字节数
  85.     DMA_IniStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;                                 //外设地址寄存器不变
  86.     DMA_IniStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;                                                                 //数据缓冲区地址递增
  87.     DMA_IniStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; //外设以字节为单位搬运
  88.     DMA_IniStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;                                  //数据缓冲区以字节为单位搬入
  89.     DMA_IniStructure.DMA_Mode = DMA_Mode_Normal;                                                                                                         //工作在正常缓存模式
  90.     DMA_IniStructure.DMA_Priority = DMA_Priority_Medium;                                                                         //DMA通道 x拥有中优先级
  91.     DMA_IniStructure.DMA_M2M = DMA_M2M_Disable;                                                                                                                 //DMA通道x没有设置为内存到内存传输

  92.     DMA_Init( DMA1_Channel6, &DMA_IniStructure );
  93.     DMA_Cmd( DMA1_Channel6, ENABLE );
  94. }

  95. //重新恢复DMA指针
  96. void myDMA_Enable( DMA_Channel_TypeDef*DMA_CHx )
  97. {
  98.     DMA_Cmd( DMA_CHx, DISABLE );                                                                                                                                                                   //关闭DMA1所指示的通道
  99.     DMA_SetCurrDataCounter( DMA_CHx, DMA_REC_LEN );                                                                                                 //DMA通道的DMA缓存的大小
  100.     DMA_Cmd( DMA_CHx, ENABLE );                                                                                                                                                                           //DMA1所指示的通道
  101. }
  102. //发送len个字节
  103. //buf:发送区首地址
  104. //len:发送的字节数
  105. void uart2_Send( u8 *buf, u16 len )
  106. {
  107.     u16 t;
  108.     for( t = 0; t < len; t++ )                                                                                                                                                                                 //循环发送数据
  109.     {
  110.         while( USART_GetFlagStatus( USART2, USART_FLAG_TC ) == RESET );
  111.         USART_SendData( USART2, buf[t] );
  112.     }
  113.     while( USART_GetFlagStatus( USART2, USART_FLAG_TC ) == RESET );
  114. }

  115. //备份接收到的数据
  116. void copy_data( u8 *buf, u16 len )
  117. {
  118.     u16 t;
  119.     dataLen_backup = len;                                                                                                                                                                                                         //保存数据长度
  120.     for( t = 0; t < len; t++ )
  121.     {
  122.         data_backup[t] = buf[t];                                                                                                                                                                         //备份接收到的数据,防止在处理数据过程中接收到新数据,将旧数据覆盖掉。
  123.     }
  124. }
  125. // Modbus RTU 通信协议
  126. //地址码 功能码 寄存器高位 寄存器低位  数据高位  数据低位 CRC高位 CRC低位
  127. //01     03      00          00         03         e8      xx        xx
  128. //利用空闲中断接收串口不定长数据
  129. //串口接收一组数据结束后才会进入中断,在数据发送过程中,已经接收到的数据将会被存入DMA的缓冲区中
  130. void USART2_IRQHandler( void )
  131. {
  132.     u8 tem = 0;
  133. #if (UART2_DMA == 1)                        //如果使能了 UART2_DMA 则使用串口空闲中断和 DMA功能接收数据
  134.         if( USART_GetITStatus( USART2, USART_IT_IDLE ) != RESET )                                                                                   //空闲中断  一帧数据发送完成
  135.     {
  136.         USART_ReceiveData( USART2 );                                                                                                                                                                                 //读取数据注意:这句必须要,否则不能够清除空闲中断标志位。
  137.         DMA_Cmd( DMA1_Channel6, DISABLE );                                                                                                                                                         //关闭 DMA 防止后续数据干扰
  138.         uart2_rec_cnt = DMA_REC_LEN - DMA_GetCurrDataCounter( DMA1_Channel6 );  //DMA接收缓冲区数据长度减去当前 DMA传输通道中剩余单元数量就是已经接收到的数据数量
  139.         //uart2_Send( dma_rec_buff, uart2_rec_cnt );                                                                                                                //发送接收到的数据
  140.                           copy_data( dma_rec_buff, uart2_rec_cnt );                                                                                                                    //备份数据
  141.         receiveOK_flag = 1;                                                                                                                                                                                                                        //置位数据接收完成标志位
  142.         USART_ClearITPendingBit( USART2, USART_IT_IDLE );                                                                                                 //清除空闲中断标志位
  143.         myDMA_Enable( DMA1_Channel6 );                                                                                                                                                                         //重新恢复 DMA等待下一次接收
  144.     }
  145. #else                                                                                //如果未使能 UART2_DMA 则通过常规方式接收数据  每接收到一个字节就会进入一次中断
  146.     if( USART_GetITStatus( USART2, USART_IT_RXNE ) != RESET )                                                 //接收中断
  147.     {
  148.         tem = USART_ReceiveData( USART2 );
  149.         USART_SendData( USART2, tem );
  150.     }
  151. #endif
  152. }
复制代码

modbus.h

  1. #ifndef __MODBUS_H
  2. #define __MODBUS_H
  3. #include "sys.h"


  4. #define SlaveID  0x01     //从机地址

  5. void DisposeReceive( void );
  6. void Modbus_03_Slave(void);
  7. void Modbus_06_Slave(void);
  8. void Modbus_16_Slave(void);
  9. #endif
复制代码

modbus.c

  1. #include "modbus.h"
  2. #include "crc16.h"
  3. #include "uart2.h"

  4. #define HoldRegStartAddr    0x0000                                                                                                                                                                                //保持寄存器起始地址
  5. #define HoldRegCount        8                                                                                                                                                                                                        //保持寄存器数量
  6. #define HoldMaxValue        1000                                                                                                                                                    //寄存器最大值

  7. extern u8 data_backup[DMA_REC_LEN];                                                                                                                                                                   //modbus接收到数据备份
  8. extern u16 dataLen_backup;                                                                                                                                                                                                                //modbus接收数据长度备份


  9. u8 SendBuf[DMA_REC_LEN] = {0};
  10. u16 StartRegAddr = HoldRegStartAddr;
  11. u8 err = 0;                                                                                                        //错误代码

  12. u8 HoldReg[HoldRegCount*2] = {1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8}; //存储8路数据值


  13. /*
  14. 错误代码说明:
  15. 0x01  不是支持的功能码
  16. 0x02  起始地址不在规定范围内
  17. 0x03  寄存器数量不在规定范围内
  18. 0x04  数据校验错误
  19. */
  20. //处理接收到的数据
  21. // 接收: [地址][功能码][起始地址高][起始地址低][总寄存器数高][总寄存器数低][CRC低][CRC高]
  22. void DisposeReceive( void )
  23. {
  24.     u16 CRC16 = 0, CRC16Temp = 0;
  25.     if( data_backup[0] == SlaveID )                                                                         //地址等于本机地址 地址范围:1 - 32
  26.     {
  27.         CRC16 = App_Tab_Get_CRC16( data_backup, dataLen_backup - 2 );                                  //CRC校验 低字节在前 高字节在后 高字节为报文最后一个字节
  28.         CRC16Temp = ( ( u16 )( data_backup[dataLen_backup - 1] << 8 ) | data_backup[dataLen_backup - 2] );
  29.         if( CRC16 != CRC16Temp )
  30.         {
  31.             err = 4;                                                                                               //CRC校验错误
  32.         }
  33.         StartRegAddr = ( u16 )( data_backup[2] << 8 ) | data_backup[3];
  34.         if( StartRegAddr > (HoldRegStartAddr + HoldRegCount - 1) )
  35.         {
  36.             err = 2;                                                                                               //起始地址不在规定范围内 00 - 07    1 - 8号通道
  37.         }
  38.         if( err == 0 )
  39.         {
  40.             switch( data_backup[1] )                                                                        //功能码
  41.             {
  42.                 case 3:                                                                                            //读多个寄存器
  43.                 {
  44.                     Modbus_03_Slave();
  45.                     break;
  46.                 }
  47.                 case 6:                                                                                            //写单个寄存器
  48.                 {
  49.                     Modbus_06_Slave();
  50.                     break;
  51.                 }
  52.                 case 16:                                                                                           //写多个寄存器
  53.                 {
  54.                     Modbus_16_Slave();
  55.                     break;
  56.                 }
  57.                 default:
  58.                 {
  59.                     err = 1;                                                                                       //不支持该功能码
  60.                     break;
  61.                 }
  62.             }
  63.         }
  64.         if( err > 0 )
  65.         {
  66.             SendBuf[0] = data_backup[0];
  67.             SendBuf[1] = data_backup[1] | 0x80;
  68.             SendBuf[2] = err;                                                                                      //发送错误代码
  69.             CRC16Temp = App_Tab_Get_CRC16( SendBuf, 3 );                                                           //计算CRC校验值
  70.             SendBuf[3] = CRC16Temp & 0xFF;                                                                         //CRC低位
  71.             SendBuf[4] = ( CRC16Temp >> 8 );                                                                       //CRC高位
  72.             uart2_Send( SendBuf, 5 );                                       
  73.             err = 0;                                                                                               //发送完数据后清除错误标志
  74.         }
  75.     }
  76. }

  77. /*
  78. 函数功能:读保持寄存器  03
  79. 主站请求报文:      0x01 0x03   0x0000  0x0001  0x840A     读从0开始的1个保持寄存器
  80. 从站正常响应报文:  0x01 0x03   0x02    0x09C4  0xBF87     读到的2字节数据为 0x09C4
  81. */
  82. void Modbus_03_Slave( void )
  83. {
  84.     u16 RegNum = 0;
  85.     u16 CRC16Temp = 0;
  86.     u8 i = 0;
  87.     RegNum = ( u16 )( data_backup[4] << 8 ) | data_backup[5];                                                //获取寄存器数量
  88.     if( ( StartRegAddr + RegNum ) <= (HoldRegStartAddr + HoldRegCount) )      //寄存器地址+寄存器数量 在规定范围内 <=8
  89.     {
  90.         SendBuf[0] = data_backup[0];                                                                                                                                                                        //地址        
  91.         SendBuf[1] = data_backup[1];                                                                                                                                                                        //功能码
  92.         SendBuf[2] = RegNum * 2;                                                                                                                                                                                        //返回字节数量
  93.         for( i = 0; i < RegNum; i++ )                                                                             //循环读取保持寄存器内的值
  94.         {
  95.             SendBuf[3 + i * 2] = HoldReg[StartRegAddr * 2 + i * 2];
  96.             SendBuf[4 + i * 2] = HoldReg[StartRegAddr * 2 + i * 2 + 1];
  97.         }
  98.         CRC16Temp = App_Tab_Get_CRC16( SendBuf, RegNum * 2 + 3 );                                                  //获取CRC校验值
  99.         SendBuf[RegNum * 2 + 3] = CRC16Temp & 0xFF;                                                                //CRC低位
  100.         SendBuf[RegNum * 2 + 4] = ( CRC16Temp >> 8 );                                                              //CRC高位
  101.         uart2_Send( SendBuf, RegNum * 2 + 5 );
  102.     }
  103.     else
  104.     {
  105.         err = 3;                                                                                                   //寄存器数量不在规定范围内
  106.     }
  107. }
  108. /*
  109. 函数功能:写单个保持寄存器 06
  110. 主站请求报文:      0x01 0x06    0x0000  0x1388    0x849C   写0号寄存器的值为0x1388
  111. 从站正常响应报文:  0x01 0x06    0x0000  0x1388    0x849C    0号寄存器的值为0x1388
  112. */
  113. void Modbus_06_Slave( void )
  114. {
  115.     u16  RegValue = 0;
  116.     u16 CRC16Temp = 0;
  117.     RegValue = ( u16 )( data_backup[4] << 8 ) | data_backup[5];                                              //获取寄存器值
  118.     if( RegValue <= HoldMaxValue )                                                  //寄存器值不超过1000
  119.     {
  120.         HoldReg[StartRegAddr * 2] = data_backup[4];                                                         //存储寄存器值
  121.         HoldReg[StartRegAddr * 2 + 1] = data_backup[5];
  122.         SendBuf[0] = data_backup[0];                                                                                                                                                                        //地址
  123.         SendBuf[1] = data_backup[1];                                                                                                                                                                        //功能码
  124.         SendBuf[2] = data_backup[2];                                                                                                                                                                        //地址高位
  125.         SendBuf[3] = data_backup[3];                                                                                                                                                                        //地址低位
  126.         SendBuf[4] = data_backup[4];                                                                                                                                                                        //值高位
  127.         SendBuf[5] = data_backup[5];                                                                                                                                                                        //值低位
  128.         CRC16Temp = App_Tab_Get_CRC16( SendBuf, 6 );                                                              //获取CRC校验值
  129.         SendBuf[6] = CRC16Temp & 0xFF;                                                                            //CRC低位
  130.         SendBuf[7] = ( CRC16Temp >> 8 );                                                                          //CRC高位
  131.         uart2_Send( SendBuf, 8 );
  132.     }
  133.     else
  134.     {
  135.         err =  3;                                                                                                  //寄存器数值不在规定范围内
  136.     }
  137. }
  138. /*
  139. 函数功能:写多个连续保持寄存器值 16
  140. 主站请求报文:       0x01 0x10    0x7540  0x0002  0x04  0x0000 0x2710    0xB731  写从0x7540地址开始的2个保持寄存器值 共4字节
  141. 从站正常响应报文:   0x01 0x10    0x7540  0x0002  0x5A10                         写从0x7540地址开始的2个保持寄存器值
  142. */
  143. void Modbus_16_Slave( void )
  144. {
  145.     u16 RegNum = 0;
  146.     u16 CRC16Temp = 0;
  147.     u8 i = 0;
  148.     RegNum = ( u16 )( data_backup[4] << 8 ) | data_backup[5];                                                //获取寄存器数量
  149.     if( ( StartRegAddr + RegNum ) <= (HoldRegStartAddr + HoldRegCount) )             //寄存器地址+寄存器数量 在规定范围内 <=8
  150.     {
  151.         for( i = 0; i < RegNum; i++ )                                                                              //存储寄存器设置值
  152.         {
  153.             HoldReg[StartRegAddr * 2 + i * 2] = data_backup[i * 2 + 7];
  154.             HoldReg[StartRegAddr * 2 + 1 + i * 2] = data_backup[i * 2 + 8];
  155.         }
  156.         SendBuf[0] = data_backup[0];                                                                                                                                                                        //起始地址
  157.         SendBuf[1] = data_backup[1];                                                                                                                                                                        //功能码
  158.         SendBuf[2] = data_backup[2];                                                                                                                                                                        //地址高位
  159.         SendBuf[3] = data_backup[3];                                                                                                                                                                        //地址低位
  160.         SendBuf[4] = data_backup[4];                                                                                                                                                                        //寄存器数量高位
  161.         SendBuf[5] = data_backup[5];                                                                                                                                                                        //寄存器数量低位
  162.         CRC16Temp = App_Tab_Get_CRC16( SendBuf, 6 );                                                        //获取CRC校验值
  163.         SendBuf[6] = CRC16Temp & 0xFF;                                                                      //CRC低位
  164.         SendBuf[7] = ( CRC16Temp >> 8 );                                                                    //CRC高位
  165.         uart2_Send( SendBuf, 8 );
  166.     }
  167.     else
  168.     {
  169.         err = 3;                                                                                                   //寄存器数量不在规定范围内
  170.     }
  171. }
复制代码

测试效果

测试功能码 0x03  读一个或者多个保持寄存器值

G[DIAGIW8)$K9KXN[LUJ8W8.png

测试功能码 0x06 写单个保持寄存值

5R6${[O14XV}RWG2%ELMRN2.png

测试功能码 0x10 写多个保持寄存器值

WJLI9948FBP1D%XE**{%QP.png

再次读取所有寄存器值

$%K_$S@0%F]LU5H~YQ$T6$N.png

说明寄存器值修改成功。


收藏 1 评论0 发布时间:2022-3-20 21:44

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版