新聞中心

詳解RS485通信

作者: 時間:2016-12-16 來源:網(wǎng)絡 收藏

  在進行 RS485 實驗中,我們通信用的引腳必須是 P3.0 和 P3.1,此外還有一個方向控制引腳,我們使用杜邦線將其連接到 P1.7 上去。RS485 的另外一端,大家可以使用一個 USB轉 RS485 模塊,用雙絞線把開發(fā)板和模塊上的 A 和 B 分別對應連起來,USB 那頭插入電腦,然后就可以進行通信了。

  學習了第 13 章實用的串口通信方法和程序后,做這種串口通信的方法就很簡單了,基本是一致的。我們使用實用串口通信例程的思路,做了一個簡單的程序,通過串口調試助手下發(fā)任意個字符,單片機接收到后在末尾添加“回車+換行”符后再送回,在調試助手上重新顯示出來,先把程序貼出來。

  程序中需要注意的一點是:因為平常都是將 MAX485 設置為接收狀態(tài),只有在發(fā)送數(shù)據(jù)的時候才將 MAX485 改為發(fā)送狀態(tài),所以在 UartWrite()函數(shù)開頭將 MAX485 方向引腳拉高,函數(shù)退出前再拉低。但是這里有一個細節(jié),就是單片機的發(fā)送和接收中斷產(chǎn)生的時刻都是在停止位的一半上,也就是說每當停止位傳送了一半的時候,RI 或 TI 就已經(jīng)置位并且馬上進入中斷(如果中斷使能的話)函數(shù)了,接收的時候自然不會存在問題,但發(fā)送的時候就不一樣了:當緊接著向 SBUF 寫入一個字節(jié)數(shù)據(jù)時,UART 硬件會在完成上一個停止位的發(fā)送后,再開始新字節(jié)的發(fā)送,但如果此時不是繼續(xù)發(fā)送下一個字節(jié),而是已經(jīng)發(fā)送完畢了,要停止發(fā)送并將 MAX485 方向引腳拉低以使 MAX485 重新處于接收狀態(tài)時就有問題了,因為這時候最后的這個停止位實際只發(fā)送了一半,還沒有完全完成,所以就有了 UartWrite()函數(shù)內DelayX10us(5)這個操作,這是人為的增加了 50us 的延時,這 50us 的時間正好讓剩下的一半停止位完成,那么這個時間自然就是由通信波特率決定的了,為波特率周期的一半。

  /****************************RS485.c 文件程序源代碼*****************************/

view plaincopy to clipboardprint?
  1. #include
  2. #include
  3. sbitRS485_DIR=P1^7;//RS485方向選擇引腳
  4. bitflagFrame=0;//幀接收完成標志,即接收到一幀新數(shù)據(jù)
  5. bitflagTxd=0;//單字節(jié)發(fā)送完成標志,用來替代TXD中斷標志位
  6. unsignedcharcntRxd=0;//接收字節(jié)計數(shù)器
  7. unsignedcharpdatabufRxd[64];//接收字節(jié)緩沖區(qū)
  8. externvoidUartAction(unsignedchar*buf,unsignedcharlen);
  9. /*串口配置函數(shù),baud-通信波特率*/
  10. voidConfigUART(unsignedintbaud){
  11. RS485_DIR=0;//RS485設置為接收方向
  12. SCON=0x50;//配置串口為模式1
  13. TMOD&=0x0F;//清零T1的控制位
  14. TMOD|=0x20;//配置T1為模式2
  15. TH1=256-(11059200/12/32)/baud;//計算T1重載值
  16. TL1=TH1;//初值等于重載值
  17. ET1=0;//禁止T1中斷
  18. ES=1;//使能串口中斷
  19. TR1=1;//啟動T1
  20. }
  21. /*軟件延時函數(shù),延時時間(t*10)us*/
  22. voidDelayX10us(unsignedchart){
  23. do{
  24. _nop_();
  25. _nop_();
  26. _nop_();
  27. _nop_();
  28. _nop_();
  29. _nop_();
  30. _nop_();
  31. _nop_();
  32. }while(--t);
  33. }
  34. /*串口數(shù)據(jù)寫入,即串口發(fā)送函數(shù),buf-待發(fā)送數(shù)據(jù)的指針,len-指定的發(fā)送長度*/
  35. voidUartWrite(unsignedchar*buf,unsignedcharlen){
  36. RS485_DIR=1;//RS485設置為發(fā)送
  37. while(len--){//循環(huán)發(fā)送所有字節(jié)
  38. flagTxd=0;//清零發(fā)送標志
  39. SBUF=*buf++;//發(fā)送一個字節(jié)數(shù)據(jù)
  40. while(!flagTxd);//等待該字節(jié)發(fā)送完成
  41. }
  42. DelayX10us(5);//等待最后的停止位完成,延時時間由波特率決定
  43. RS485_DIR=0;//RS485設置為接收
  44. }
  45. /*串口數(shù)據(jù)讀取函數(shù),buf-接收指針,len-指定的讀取長度,返回值-實際讀到的長度*/
  46. unsignedcharUartRead(unsignedchar*buf,unsignedcharlen){
  47. unsignedchari;
  48. //指定讀取長度大于實際接收到的數(shù)據(jù)長度時,
  49. //讀取長度設置為實際接收到的數(shù)據(jù)長度
  50. if(len>cntRxd){
  51. len=cntRxd;
  52. }
  53. for(i=0;i
  54. *buf++=bufRxd[i];
  55. }
  56. cntRxd=0;//接收計數(shù)器清零
  57. returnlen;//返回實際讀取長度
  58. }
  59. /*串口接收監(jiān)控,由空閑時間判定幀結束,需在定時中斷中調用,ms-定時間隔*/
  60. voidUartRxMonitor(unsignedcharms){
  61. staticunsignedcharcntbkp=0;
  62. staticunsignedcharidletmr=0;
  63. if(cntRxd>0){//接收計數(shù)器大于零時,監(jiān)控總線空閑時間
  64. if(cntbkp!=cntRxd){//接收計數(shù)器改變,即剛接收到數(shù)據(jù)時,清零空閑計時
  65. cntbkp=cntRxd;
  66. idletmr=0;
  67. }else{//接收計數(shù)器未改變,即總線空閑時,累積空閑時間
  68. if(idletmr<30){//空閑計時小于30ms時,持續(xù)累加
  69. idletmr+=ms;
  70. if(idletmr>=30){//空閑時間達到30ms時,即判定為一幀接收完畢
  71. flagFrame=1;//設置幀接收完成標志
  72. }
  73. }
  74. }
  75. }else{
  76. cntbkp=0;
  77. }
  78. }
  79. /*串口驅動函數(shù),監(jiān)測數(shù)據(jù)幀的接收,調度功能函數(shù),需在主循環(huán)中調用*/
  80. voidUartDriver(){
  81. unsignedcharlen;
  82. unsignedcharpdatabuf[40];
  83. if(flagFrame){//有命令到達時,讀取處理該命令
  84. flagFrame=0;
  85. len=UartRead(buf,sizeof(buf)-2);//將接收到的命令讀取到緩沖區(qū)中
  86. UartAction(buf,len);//傳遞數(shù)據(jù)幀,調用動作執(zhí)行函數(shù)
  87. }
  88. }
  89. /*串口中斷服務函數(shù)*/
  90. voidInterruptUART()interrupt4{
  91. if(RI){//接收到新字節(jié)
  92. RI=0;//清零接收中斷標志位
  93. //接收緩沖區(qū)尚未用完時,保存接收字節(jié),并遞增計數(shù)器
  94. if(cntRxd
  95. bufRxd[cntRxd++]=SBUF;
  96. }
  97. }
  98. if(TI){//字節(jié)發(fā)送完畢
  99. TI=0;//清零發(fā)送中斷標志位
  100. flagTxd=1;//設置字節(jié)發(fā)送完成標志
  101. }
  102. }



關鍵詞: RS485通

評論


技術專區(qū)

關閉