找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1241|回复: 1
收起左侧

stm32F407单片机实现RS485通讯程序,同时modbus从机测试,用串口3实现

[复制链接]
ID:291668 发表于 2023-3-30 22:45 | 显示全部楼层 |阅读模式
pcb1.png
自己画的板子,用串口3 (PD8,PD9实现,PB15收发选择电平)。

stm32F407单片机源程序如下:
  1. /*485-2 */
  2. #define RS485_RE_GPIO_PORT                  GPIOB
  3. #define RS485_RE_GPIO_PIN                   SYS_GPIO_PIN15
  4. #define RS485_RE_GPIO_CLK_ENABLE()          do{ RCC->AHB1ENR |= 1 << 1; }while(0)   /* PB口时钟使能 */

  5. #define RS485_TX_GPIO_PORT                  GPIOD
  6. #define RS485_TX_GPIO_PIN                   SYS_GPIO_PIN8
  7. #define RS485_TX_GPIO_AF                    7                                       /* AF功能选择 */
  8. #define RS485_TX_GPIO_CLK_ENABLE()          do{ RCC->AHB1ENR |= 1 << 3; }while(0)   /* PD口时钟使能 */

  9. #define RS485_RX_GPIO_PORT                  GPIOD
  10. #define RS485_RX_GPIO_PIN                   SYS_GPIO_PIN9
  11. #define RS485_RX_GPIO_AF                    7                                       /* AF功能选择 */
  12. #define RS485_RX_GPIO_CLK_ENABLE()          do{ RCC->AHB1ENR |= 1 << 3; }while(0)   /* PD口时钟使能 */

  13. #define RS485_UX                            USART3
  14. #define RS485_UX_IRQn                       USART3_IRQn
  15. #define RS485_UX_IRQHandler                 USART3_IRQHandler
  16. #define RS485_UX_CLK_ENABLE()               do{ RCC->APB1ENR |= 1 << 18; }while(0)  /* USART3 时钟使能 */
  17. /*485-2 END */
  18. /* 控制RS485_RE脚, 控制RS485发送/接收状态
  19. * RS485_RE = 0, 进入接收模式
  20. * RS485_RE = 1, 进入发送模式
  21. */
  22. #define RS485_RE(x)     sys_gpio_pin_set(RS485_RE_GPIO_PORT, RS485_RE_GPIO_PIN, x)


  23. #define RS485_REC_LEN               64          /* 定义最大接收字节数 64 */
  24. #define RS485_EN_RX                 1           /* 使能(1)/禁止(0)RS485接收 */


  25. extern uint8_t g_RS485_rx_buf[RS485_REC_LEN];   /* 接收缓冲,最大RS485_REC_LEN个字节 */
  26. extern uint8_t g_RS485_rx_cnt;                  /* 接收数据长度 */


  27. void rs485_init(uint32_t sclk, uint32_t baudrate);  /* RS485初始化 */
  28. void rs485_send_data(uint8_t *buf, uint8_t len);    /* RS485发送数据 */
  29. void rs485_receive_data(uint8_t *buf, uint8_t *len);/* RS485接收数据 */
  30. extern void UartDriver(void); /* 485接收数据处理 */

  31. #endif
  32. //串口驱动函数,检测数据帧的接收,调度功能函数,需在主循环中调用
  33. void UartDriver(void)
  34. {
  35.         unsigned char i=0,cnt;
  36.         unsigned int crc;
  37.         unsigned char crch,crcl;
  38.         static uint8_t len;
  39.         static uint8_t buf[60];
  40.         if(flagFrame)            //帧接收完成标志,即接收到一帧新数据
  41.         {
  42.                 flagFrame=0;           //帧接收完成标志清零
  43.                 len = UartRead(buf,sizeof(buf));   //将接收到的命令读到缓冲区中
  44.                 if(buf[0]==0x01)                   //判断地址是不是0x01
  45.                 {
  46.                         crc=GetCRC16(buf,len-2);       //计算CRC校验值,出去CRC校验值
  47.                         crch=crc>>8;                                    //crc高位
  48.                         crcl=crc&0xFF;                                        //crc低位
  49.                         if((buf[len-2]==crch)&&(buf[len-1]==crcl))  //判断CRC校验是否正确
  50.                         {
  51.                                 switch (buf[1])  //按功能码执行操作
  52.                                 {
  53.                                         case 0x03:     //读数据
  54.                                                 if((buf[2]==0x00)&&(buf[3]<=0x05))  //寄存器地址支持0x0000~0x0005
  55.                                                 {
  56.                                                         
  57.                                                         if(buf[3]<=0x04)
  58.                                                         {
  59.                                                                 i=buf[3];//提取寄存器地址
  60.                                                                 cnt=buf[5];  //提取待读取的寄存器数量
  61.                                                                 buf[2]=cnt*2;  //读取数据的字节数,为寄存器*2,因modbus定义的寄存器为16位
  62.                                                                 len=3;                                                        
  63.                                                                 while(cnt--)
  64.                                                                 {
  65.                                                                         buf[len++]=0x00;                                //寄存器高字节补0
  66.                                                                         buf[len++]=regGroup[i++];                //低字节
  67.                                                         }
  68.                                                         
  69.                                                 }
  70.                                                         break;
  71.                                         }
  72.                                                 else  //寄存器地址不被支持时,返回错误码
  73.                                                 {   
  74.                                                         buf[1]=0x83;  //功能码最高位置1
  75.                                                         buf[2]=0x02;  //设置异常码为02-无效地址
  76.                                                         len=3;
  77.                                                         break;
  78.                                                 }
  79.                                         case 0x06:           //写入单个寄存器
  80.                                                 if((buf[2]==0x00)&&(buf[3]<=0x05))   //寄存器地址支持0x0000-0x0005
  81.                                                 {
  82.                                                         if(buf[3]<=0x04)
  83.                                                         {
  84.                                                                 i=buf[3];                                //提取寄存器地址
  85.                                                                 regGroup[i]=buf[5];                //保存寄存器数据
  86.                                                                 LED0(0);
  87.                                                         }
  88.                                                         len -=2;                 //长度-2以重新计算CRC并返回原帧
  89.                                                         break;
  90.                                                 }
  91.                                                 else  
  92.                                                 {                                                        //寄存器地址不被支持,返回错误码
  93.                                                         buf[1]=0x86;           //功能码最高位置1
  94.                                                         buf[2]=0x02;           //设置异常码为02-无效地址
  95.                                                         len=3;
  96.                                                         break;
  97.                                         }
  98.                                         default:    //其他不支持的功能码
  99.                                                     buf[1]=0x80;     //功能码最高位置1
  100.                                                         buf[2]=0x01;     //设置异常码为01—无效功能
  101.                                                         len=3;
  102.                                                         break;
  103.                                 }
  104.                             crc=GetCRC16(buf,len);           //计算CRC校验值
  105.                                 buf[len++]=crc>>8;           //CRC高字节
  106.                                 buf[len++]=crc&0xff;        //CRC低字节
  107. //                                rs485_UartWrite(buf,len);  //发送响应帧
  108.                 rs485_send_data(buf,len);       //发送响应帧
  109.                         }
  110.                 }
  111.         }
  112. }

  113.                                 
  114. void UartRxMonitor(uint8_t ms) //串口接收监控
  115. {
  116.         static uint8_t USART3_RX_BKP=0;  //定义USART2_RC_BKP暂时存储数据长度与实际长度比较
  117.         static uint8_t idletmr=0;        //定义监控时间
  118.         if(g_RS485_rx_cnt>0)//接收计数器大于零时,监控总线空闲时间
  119.         {
  120.                 if(USART3_RX_BKP!=g_RS485_rx_cnt) //接收计数器改变,即刚接收到数据时,清零空闲计时
  121.                 {
  122.                         USART3_RX_BKP=g_RS485_rx_cnt;  //赋值操作,将实际长度给USART2_RX_BKP
  123.                         idletmr=0;                    //将监控时间清零
  124.                 }
  125.                 else                              ////接收计数器未改变,即总线空闲时,累计空闲时间
  126.                 {
  127.                         //如果在一帧数据完成之前有超过3.5个字节时间的停顿,接收设备将刷新当前的消息并假定下一个字节是一个新的数据帧的开始
  128.                         if(idletmr<5)                  //空闲时间小于1ms时,持续累加
  129.                         {
  130.                                 idletmr +=ms;
  131.                                 if(idletmr>=5)             //空闲时间达到1ms时,即判定为1帧接收完毕
  132.                                 {
  133.                                         flagFrame=1;//设置命令到达标志,帧接收完毕标志
  134.                                 }
  135.                         }
  136.                 }
  137.         }
  138.         else
  139.         {
  140.                 USART3_RX_BKP=0;
  141.         }
  142. }

  143.                         



  144. /**
  145. * @brief       RS485发送len个字节
  146. * @param       buf     : 发送区首地址
  147. * @param       len     : 发送的字节数(为了和本代码的接收匹配,这里建议不要超过 RS485_REC_LEN 个字节)
  148. * @retval      无
  149. */
  150. void rs485_send_data(uint8_t *buf, uint8_t len)
  151. {
  152.     uint8_t t;
  153.     RS485_RE(1);    /* 进入发送模式 */

  154.     for (t = 0; t < len; t++)   /* 循环发送数据 */
  155.     {
  156.         while ((RS485_UX->SR & 0X40) == 0); /* 等待发送结束 */

  157.         RS485_UX->DR = buf[t];
  158.     }

  159.     while ((RS485_UX->SR & 0X40) == 0);     /* 等待发送结束 */

  160.     g_RS485_rx_cnt = 0;
  161.     RS485_RE(0);    /* 进入接收模式 */
  162. }

  163. /**
  164. * @brief       RS485查询接收到的数据
  165. * @param       buf     : 接收缓冲区首地址
  166. * @param       len     : 接收到的数据长度
  167. *   @arg               0   , 表示没有接收到任何数据
  168. *   @arg               其他, 表示接收到的数据长度
  169. * @retval      无
  170. */
  171. void rs485_receive_data(uint8_t *buf, uint8_t *len)
  172. {
  173.     uint8_t rxlen = g_RS485_rx_cnt;
  174.     uint8_t i = 0;
  175.     *len = 0;       /* 默认为0 */
  176.     delay_ms(10);   /* 等待10ms,连续超过10ms没有接收到一个数据,则认为接收结束 */

  177.     if (rxlen == g_RS485_rx_cnt && rxlen)   /* 接收到了数据,且接收完成了 */
  178.     {
  179.         for (i = 0; i < rxlen; i++)
  180.         {
  181.             buf[i] = g_RS485_rx_buf[i];
  182.         }

  183.         *len = g_RS485_rx_cnt;  /* 记录本次数据长度 */
  184.         g_RS485_rx_cnt = 0;     /* 清零 */
  185.     }
  186. }
复制代码

Keil代码下载: 测试 485-2(串口3)modbus rtu从机实验.7z (216.49 KB, 下载次数: 55)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:291668 发表于 2023-4-2 16:15 | 显示全部楼层
485通讯是半双工的,注意收发控制引脚的电平切换。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表