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核心的单片机上都能够成功,由于程序比较简单,我没有试,但应该没有问题的)
#include“Reg52.H”
请提前计算一下所选晶振能达到的最高速度,波特率不能超过最高速度
(1)波特率加倍(SMOD=1):Max_Baud=FOSC/12/16
(2)波特率不加倍(SMOD=0):Max_Baud=FOSC/12/32
例如:22.1184MHz晶振,波特率加倍时,最大波特率=22118400/12/16=115200
#defineFOSC22118400//振荡频率
#defineBAUD9600//波特率
#defineSMOD1//是否波特率加倍
#ifSMOD
#defineTC_VAL(256-FOSC/16/12/BAUD)
#else
#defineTC_VAL(256-FOSC/32/12/BAUD)
#endif
typedefunsignedcharuint8;
typedefunsignedintuint16;
codeconstcharstr1[]=“Therstringistransmittedfrom80C51!rn”;
codeconstcharstr2[]=“Author:xqlu(at)ysu.edu.cnrn”;
/***************函数声明*******************/
voidInitUART(void);
voidSendOneByte(uint8);
voidSendrStr(constuint8*ptr);
/****************主函数********************/
voidmain(void)
{
uint8i=0;
InitUART();
while(str2[i]!=‘’)
{
SendOneByte(str2[i++]);
}
SendrStr(str1);
while(1);
}
/****************中断服务函数***************/
voidUART_ISR(void)interrupt4
{
uint8RX_Data;
//只响应“接收”中断,“发送”中断来了就直接抹掉
if(RI)
{
RI=0;//串口中断标志不能自己清除,需要手动清除
RX_Data=SBUF;
SendOneByte(RX_Data);
}
else
TI=0;//串口发中断是发送完缓冲区数据之后产生
}
/****************串口初始化函数*************/
voidInitUART(void)
{
TMOD=0x20;
SCON=0x50;
TH1=TC_VAL;
TL1=TH1;
PCON=0x80;//发送速率加倍
ES=1;
EA=1;
TR1=1;
}
/**************串口发送字符函数*************/
voidSendOneByte(uint8c)
{
ES=0;//禁止发送中断
SBUF=c;
while(!TI);
TI=0;
ES=1;
}
/**************串口发送字符串函数*************/
voidSendrStr(constuint8*ptr)
{
do
{
SendOneByte(*ptr);
}while(*ptr++!=‘’);
}