8051單片機使用定時器1工作在方式2的情況下作為串口波特率發(fā)生器,其波特率=(2smod/32)×(定時器T1溢出率),其中smod是PCON《7》,表示是否波特率加倍,F(xiàn)soc是系統(tǒng)的晶振大小。
波特率公式中:T1溢出率=溢出周期的倒數(shù);溢出周期=(256-TH1)×12/Fosc;
最終公式:
波特率:Baud=(2smod× Fsoc)/(32 ×12×(256-TH1))
我們一般不太關(guān)注波特率的計算,而是關(guān)心選用的傳輸速度(波特率)去反算定時器1(自動重裝模式)的初值(TH1),所以將上面的公式導(dǎo)一下,得到TH1的公式:
TH1=256-(Fsoc×2smod)/(12×32×Baud)
下面針對串口發(fā)送程序,水寒寫下例程,供大家參考。(我使用的是STC12C5A40S2調(diào)試,原則上在STC89C5x、AT89C5x等8051核心的單片機上都能夠成功,由于程序比較簡單,我沒有試,但應(yīng)該沒有問題的)
#include“Reg52.H”
請?zhí)崆坝嬎阋幌滤x晶振能達到的最高速度,波特率不能超過最高速度
(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”;
/***************函數(shù)聲明*******************/
voidInitUART(void);
voidSendOneByte(uint8);
voidSendrStr(constuint8*ptr);
/****************主函數(shù)********************/
voidmain(void)
{
uint8i=0;
InitUART();
while(str2[i]!=‘’)
{
SendOneByte(str2[i++]);
}
SendrStr(str1);
while(1);
}
/****************中斷服務(wù)函數(shù)***************/
voidUART_ISR(void)interrupt4
{
uint8RX_Data;
//只響應(yīng)“接收”中斷,“發(fā)送”中斷來了就直接抹掉
if(RI)
{
RI=0;//串口中斷標志不能自己清除,需要手動清除
RX_Data=SBUF;
SendOneByte(RX_Data);
}
else
TI=0;//串口發(fā)中斷是發(fā)送完緩沖區(qū)數(shù)據(jù)之后產(chǎn)生
}
/****************串口初始化函數(shù)*************/
voidInitUART(void)
{
TMOD=0x20;
SCON=0x50;
TH1=TC_VAL;
TL1=TH1;
PCON=0x80;//發(fā)送速率加倍
ES=1;
EA=1;
TR1=1;
}
/**************串口發(fā)送字符函數(shù)*************/
voidSendOneByte(uint8c)
{
ES=0;//禁止發(fā)送中斷
SBUF=c;
while(!TI);
TI=0;
ES=1;
}
/**************串口發(fā)送字符串函數(shù)*************/
voidSendrStr(constuint8*ptr)
{
do
{
SendOneByte(*ptr);
}while(*ptr++!=‘’);
}
責任編輯;zl
評論
查看更多