本帖最后由 海洋漁業 于 2019-2-10 00:05 編輯
相信很多朋友在學習串口通訊的時候,因為晶振是12Mhz而不能設置4800bps以上的波特率,我在學習中也是如此,網上也沒有什么靠譜的方法。
作為一個新時代的青少年,通過不懈的學習(其實是看器件手冊無意間看到的)終于有了突破。
1:晶振為11.0592Mhz
我們知道T1可以作為波特率發生器,也知道波特率計算公式(方式1、3):2的SMOD的次方除以32再乘T1溢出率(晶振頻率除以12乘(256-TH1));
即:波特率=(2的SMOD次方/32)X晶振頻率/[12x(256-TH1)]
我們可以輕松的算出TH1和TL1內的值,列如:
計算9600bps ,設SMOD=1;
9600bps=(2/32)X[11059200/12X(256-TH1)]
算得(256-TH1)=6,則TH1=250 (0XFA);
計算波特率為9600、串口工作方式為方式1的程序:
void init_serial()
{
TMOD=0X20;//設置T1工作方式為8位自動重裝
SCON=0X50;//設置串口工作方式為10位異步收發器
PCON=0X80;//SMOD置為1
TH1=(256-6);//6的由來[(11059200/12)/16]/9600 (9600為所求波特率)
TL1=(256-6);
EA=1;//開總中斷
ES=1;//開串口中斷
TR1=1;//啟動T1
}
2:晶振為12Mhz
如果我們繼續將T1作為波特率發生器,那么計算的公式也沒變化。
列如:
波特率為2400,SMOD=1;
2400bps=(2/32)X[12000000/12X(256-TH1)]
算得(256-TH1)=26.041666,是個無限循環小數,如果取26帶進公式里算,波特率四舍五入后是2404,誤差不是很大。
- void init_serial()
- {
- TMOD=0X20;//設置T1工作方式為8位自動重裝
- SCON=0X50;//設置串口工作方式為10位異步收發器
- PCON=0X80;//SMOD置為1
- TH1=(256-26);
- TL1=(256-26);
- EA=1;//開總中斷
- ES=1;//開串口中斷
- TR1=1;//啟動T1
- }
復制代碼
如果4800bps,SMOD=1;
4800bps=(2/32)X[12000000/12X(256-TH1)]
算得(256-TH1)=13.0208333,也是個無限循環小數,取13,算出波特率約為4808,誤差也不大。
- void init_serial()
- {
- TMOD=0X20;//設置T1工作方式為8位自動重裝
- SCON=0X50;//設置串口工作方式為10位異步收發器
- PCON=0X80;//SMOD置為1
- TH1=(256-13);
- TL1=(256-13);
- EA=1;//開放總中斷
- ES=1;//開放串口中斷
- TR1=1;//啟動T1
- }
復制代碼
那么,就T1作為波特率發生器就夠了嗎?不行,高波特率誤差大。
如:9600bps,SMOD=1;
9600bps=(2/32)X[12000000/12X(256-TH1)]
算得(256-TH1)=6.510416(6的循環)
如果我們取7,則計算出來的波特率為8928.571428(571428的循環),取6,波特率則為10416.66666(6的循環),誤差太大。
這該怎么辦呢? 有兩種方法:1.換晶振(不推薦) 2.使用T2作為波特率發生器(推薦) 以下是有關T2的器件手冊截圖
T2
這里著重注意TCLK與RCLK,手冊上的描述是:RCLK=0時,將定時器1的溢出率脈沖作為串口模式1和模式3的接受時鐘,TCLK=0時,將定時器1的溢出率脈沖作為串口模式1和模式3的(這個“的”是我加上的,不知道為什么前面有后面沒有)發送時鐘。
這兩句話講的已經夠清楚了,我們只把RCLK置為1,那我們的接受波特率就是(2的SMOD的次方/32)X(T2溢出率);如果我們只把TCLK置為1,那么我們的發送波特率就為(2的SMOD的次方/32)X(T2溢出率);如果我們將RCLK與TCLK都置為1,那么我們的波特率就是(2的SMOD的次方/32)X(T2溢出率)。
T2溢出率和T1溢出率不同它的計算公式為:晶振頻率/[2X(65536-RCAP2H RCAP2L)],此外T2是一個16位的定時/計數器,不像T0、T1既是8位又是16位。
使用T2作為波特率發生器的波特率計算公式為:(2的SMOD的次方/32)X(T2溢出率)
即波特率=(2的SMOD的次方/32)X晶振頻率/[2X(65536-RCAP2H RCAP2L)];
按照T2的波特率計算公式我們就可以隨著自己的想法設置波特率了,列如我們要設置9600bps,SMOD=1:
將9600bps代入波特率=(2的SMOD的次方/32)X晶振頻率/[2X(65536-RCAP2H RCAP2L)],
得:
9600bps=(2/32)X12000000/[2X(65536-RCAP2H RCAP2L)]
算得(65536-RCAP2H RCAP2L)=39;
所以有
- void init_serial()//設值9600bps的程序,大家可以試試
- {
- SCON=0X50; //設置串口工作模式為10位異步收發器
- T2CON=0X34; //RCLK=1,TCLK=1,TR2=1
- PCON=0X80; //SMOD=1
- RCAP2H=(65536-39)/256; // 裝入計算好的值,這是我喜歡的懶方法
- RCAP2L=(65536-39)%256;//同上,不過這是對(65536-39)求余,前者是求模(也就是求商)
- ES=1;//開串口中斷
- EA=1;//開總中斷
- }
復制代碼 本人屬于發貼新手和單片機新手,如有錯誤請您委婉點地指出。
感謝您的瀏覽,同時也要謝謝審核我的管理員,真是麻煩您了!
【這是我第一次發稿】
|