使用51的定時器模擬串口
另外STC的手冊上面說的8位自動重載定時器竟然TMOD等于6!導(dǎo)致我搞了一下午都調(diào)不通,沒想到老妖寫錯了……定時器重載模式應(yīng)該是=2.我現(xiàn)在想把TMOD寄存器的O去掉,再用這個詞問候他們?nèi)遥。?/p>本文引用地址:http://m.butianyuan.cn/article/201611/323424.htm
順便說一句,1T的51單片機足夠在定時器里實現(xiàn)全雙工,這就是為什么老妖說真實波特率要/3的原因。有RCNT=3這個語句在,所以可以實現(xiàn)同時接收發(fā)送……老妖寫程序不寫注釋是出了名的……但是至于普通的12T單片機。如果也用老妖得的程序,要注意定時器模式?jīng)]有16位自動重裝模式。如果/3的話,9600就成3200的波特率了……還是非標(biāo)準(zhǔn)的,這個程序在89C52上最高波特率只能到9600,繼續(xù)往上加就嚴(yán)重誤碼……所以模擬串口還是用1T單片機或者12T降3倍速吧,(但是速度能卡的可怕,我感覺9600都卡)。要不然誤碼率能搞死人。
對了,順便說明一下,這個程序在keil c51上通不過,因為Keil內(nèi)置putchar……會提示重復(fù)定義。能過keil的程序請參考http://www.51hei.com/mcu/1541.html,而且似乎keil的printf函數(shù)有毛病……
#include#include #include void WaitTF0(){while(!TF0);TF0=0;}void WByte(BYTE out){//發(fā)送啟始位BYTE i=8;BYTE tmp=out;TR0=1;//開定時器TX1=0;WaitTF0();//發(fā)送8位數(shù)據(jù)位while(i--){TX1=(tmp&0x01); //先傳低位tmp=tmp>>1;WaitTF0();}//發(fā)送校驗位(無)//發(fā)送結(jié)束位TX1=1;WaitTF0();TR0=0;} void putchar(char ch){WByte(ch);}BYTE RByte(){BYTE in=0;BYTE cnt;while(RX1==1);//等待RXD變低,啟動定時器,這個是阻塞模式TR0=1;//同步開定時器WaitTF0();//等到周期過去for(cnt=0;cnt<8;cnt++){in=in >>1;//從高移到低if(RX1==1) in = in | 0x80;//如果RXD=1,則最高置位WaitTF0();//等待一位過去}//等待停止位//WaitTF0();TR0=0;//關(guān)閉定時器return in;}void Init_VSer(){//指令周期=(1000000/Baud)/(12/SysClock)//定時器值=0x100-指令周期TMOD |=0x02; //計數(shù)器0,方式2TH0=0xa0; TL0=0xa0; TR0=0; //停止計數(shù)//TF0=0; //ET0=0; //T1中斷關(guān)//EA=0; //總中斷關(guān)}void main(){BYTE a;Init_VSer();printf("Hello,world!");while(1){a=RByte();//阻塞模式putchar(a);}}
順便附上老妖的減掉三倍速的1T單片機模擬串口程序。穩(wěn)定得多,當(dāng)然效率也低的多。那個是靠定時器中斷實現(xiàn)的。不過至少不會象在12T上到主程序幾乎跑不動的地步。強烈要求老妖出有串口的C版本!
/*----------------------------------------------------*//* --- STC MCU International Limited -----------------*//* --- 演示STC 15 系列單片機利用定時器1實現(xiàn)模擬串口功能-----*//* --- Mobile: (86)13922809991 -----------------------*//* --- Fax: 86-755-82905966 --------------------------*//* --- Tel: 86-755-82948412 --------------------------*//* --- Web: www.STCMCU.com ---------------------------*//* 如果要在程序中使用或在文章中引用該程序, -----------------*//* 請在程序中或文章中注明使用了STC的資料及程序 -------------*//*----------------------------------------------------*/#include "reg51.h"http://define baudrate const//BAUD = 256 - SYSclk/3/BAUDRATE/M (1T:M=1; 12T:M=12)//NOTE: (SYSclk/3/BAUDRATE) must be greater than 98, (RECOMMEND GREATER THAN 110)//開始一直看不懂為什么要除三//#define BAUD 0xF400 // 1200bps @ 11.0592MHz//#define BAUD 0xFA00 // 2400bps @ 11.0592MHz//#define BAUD 0xFD00 // 4800bps @ 11.0592MHz//#define BAUD 0xFE80 // 9600bps @ 11.0592MHz//#define BAUD 0xFF40 //19200bps @ 11.0592MHz//#define BAUD 0xFFA0 //38400bps @ 11.0592MHz//#define BAUD 0xEC00 // 1200bps @ 18.432MHz//#define BAUD 0xF600 // 2400bps @ 18.432MHz//#define BAUD 0xFB00 // 4800bps @ 18.432MHz//#define BAUD 0xFD80 // 9600bps @ 18.432MHz//#define BAUD 0xFEC0 //19200bps @ 18.432MHz#define BAUD 0xFF60 //38400bps @ 18.432MHz//#define BAUD 0xE800 // 1200bps @ 22.1184MHz//#define BAUD 0xF400 // 2400bps @ 22.1184MHz//#define BAUD 0xFA00 // 4800bps @ 22.1184MHz//#define BAUD 0xFD00 // 9600bps @ 22.1184MHz//#define BAUD 0xFE80 //19200bps @ 22.1184MHz//#define BAUD 0xFF40 //38400bps @ 22.1184MHz//#define BAUD 0xFF80 //57600bps @ 22.1184MHz//define UART TX/RX porttypedef bit BOOL;typedef unsigned char BYTE;typedef unsigned int WORD;BYTE TBUF,RBUF;BYTE TDAT,RDAT;BYTE TCNT,RCNT;BYTE TBIT,RBIT;BOOL TING,RING;BOOL TEND,REND;void UART_INIT();BYTE t, r;BYTE buf[16];void main(){TMOD = 0x00;//timer1 in 16-bit auto reload modeAUXR = 0x40;//timer1 working at 1T modeTL1 = BAUD;//initial timer1 and set reload valueTH1 = BAUD>>8;TR1 = 1;//timer1 start running[原文有錯誤],感覺這個代碼不像老妖寫的。注釋都英文……ET1 = 1;//enable timer1 interruptPT1 = 1;//improve timer1 interrupt priorityEA = 1;//open global interrupt switchUART_INIT();while (1){//users functionif (REND){REND = 0;buf[r++ & 0x0f] = RBUF;}if (TEND){if (t != r){TEND = 0;TBUF = buf[t++ & 0x0f];TING = 1;}}}}
評論