8051单片机使用定时器1工作在方式2的情况下作为串口波特率发生器,其波特率=(2smod /32)×(定时器T1溢出率),其中smod是PCON<7>,表示是否波特率加倍,Fsoc是系统的晶振大小。
波特率公式中:T1溢出率=溢出周期的倒数;溢出周期=(256-TH1)×12/Fosc;
最终公式:
波特率:Baud=(2smod × Fsoc)/(32 ×12×(256-TH1))
我们一般不太关注波特率的计算,而是关心选用的传输速度(波特率)去反算定时器1(自动重装模式)的初值(TH1),所以将上面的公式导一下,得到TH1的公式:
TH1=256-(Fsoc×2smod)/(12×32×Baud)
下面针对串口发送程序,写下例程,供大家参考。(我使用的是STC12C5A40S2调试,原则上在STC89C5x、AT89C5x等8051核心的单片机上都能够成功,由于程序比较简单,我没有试,但应该没有问题的)
1. #include "Reg52.H"
2. /*******************************************************************
3. 请提前计算一下所选晶振能达到的最高速度,波特率不能超过最高速度
4. (1) 波特率加倍(SMOD=1): Max_Baud = FOSC/12/16
5. (2) 波特率不加倍(SMOD=0):Max_Baud = FOSC/12/32
6. 例如:22.1184MHz晶振,波特率加倍时,最大波特率=22118400/12/16=115200
7. *******************************************************************/
8. #define FOSC 22118400 //振荡频率
9. #define BAUD 9600 //波特率
10. #define SMOD 1 //是否波特率加倍
11. #if SMOD
12. #define TC_VAL (256-FOSC/16/12/BAUD)
13. #else
14. #define TC_VAL (256-FOSC/32/12/BAUD)
15. #endif
16.
17. typedef unsigned char uint8;
18. typedef unsigned int uint16;
19.
20. code const char str1[] = "Ther string is transmitted from 80C51!rn";
21. code const char str2[] = "Author: xqlu(at)ysu.edu.cnrn";
22.
23. /***************函数声明*******************/
24. void InitUART(void);
25. void SendOneByte(uint8);
26. void SendrStr(const uint8 *ptr);
27.
28. /****************主函数********************/
29. void main(void)
30. {
31. uint8 i=0;
32. InitUART();
33.
34. while(str2[i]!='