久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 3968|回復: 3
收起左側

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, 下載次數: 78)

評分

參與人數 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

回復

使用道具 舉報

ID:291668 發表于 2023-4-2 16:15 | 顯示全部樓層
485通訊是半雙工的,注意收發控制引腳的電平切換。
回復

使用道具 舉報

ID:146782 發表于 2024-9-2 16:19 | 顯示全部樓層
好東西!我來下一個
回復

使用道具 舉報

ID:111490 發表于 2024-9-18 08:22 | 顯示全部樓層
學習!!!
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 午夜影院普通用户体验区 | 国产aⅴ精品 | 91玖玖 | 精品成人佐山爱一区二区 | 亚洲综合五月天婷婷 | 亚洲精品一区二区三区蜜桃久 | 日本xx视频免费观看 | 国产xxxx在线 | 成人伊人 | 四虎在线观看 | 亚洲第一福利视频 | 午夜久久久| 玩丰满女领导对白露脸hd | 蜜月va乱码一区二区三区 | 亚洲欧美在线免费观看 | 国产精品久久久久久久久 | 欧美亚洲国语精品一区二区 | 狠狠干在线 | 国产精品久久a | 一本一道久久a久久精品蜜桃 | 伊人伊成久久人综合网站 | 一级黄色短片 | 久久成人综合 | 免费一级片 | 91精品国产乱码久久久久久久久 | 色婷婷国产精品 | 亚洲精品一区二区在线观看 | 色狠狠桃花综合 | 国产精品久久久久久网站 | 国产视频一区二区 | 国产一区二区三区在线 | 久久久久久久一区 | 日韩欧美在线观看 | 一区二区三区国产好的精 | 日韩精品在线免费 | 日韩欧美国产精品 | 亚洲精品9999 | 日本欧美在线观看视频 | 一级黄色片在线免费观看 | 日韩欧美手机在线 | 亚洲精品日韩欧美 |