這里有一點需要注意,《The Art of assembly language programming》中提到,發(fā)送寄存器為空時串口會產(chǎn)生中斷,但是只會產(chǎn)生一次,書中采用的解決辦法是通過IO端口直接寫入發(fā)送保持寄存器,此時發(fā)送完畢后串口便又會產(chǎn)生“發(fā)送保持寄存器為空”中斷。(就是每次發(fā)送前都設(shè)置中斷允許寄存器),從實際效果來看是可以的。
本文引用地址:
http://butianyuan.cn/article/201611/319243.htm
51單片機與串口通信(含代碼)
串口調(diào)試
1. 發(fā)送:向總線上發(fā)命令
2. 接收:從總線接收命令,并分析是地址還是數(shù)據(jù)。
3. 定時發(fā)送:從內(nèi)存中取數(shù)并向主機發(fā)送.
經(jīng)過調(diào)試,以上功能基本實現(xiàn),目前可以通過上位機對單片機進行實時控制。
程序如下:
//這是一個單片機C51串口接收(中斷)和發(fā)送例程,可以用來測試51單片機的中斷接收
//和查詢發(fā)送,另外我覺得發(fā)送沒有必要用中斷,因為程序的開銷是一樣的
#i nclude
#i nclude
#i nclude
#define INBUF_LEN 4 //數(shù)據(jù)長度
unsigned char inbuf1[INBUF_LEN];
unsigned char checksum,count3 , flag,temp,ch;
bit read_flag=0;
sbit cp=P1^1;
sbit DIR=P1^2;
int i;
unsigned int xdata *RAMDATA;
unsigned char a[6] ={0x11,0x22,0x33,0x44,0x55,0x66} ;
void init_serialcomm(void)
{
SCON=0x50; //在11.0592MHz下,設(shè)置串行口波特率為9600,方式1,并允許接收
PCON=0x00;
ES=1;
TMOD=0x21; //定時器工作于方式2,自動裝載方式
TH0=(65536-1000)%6;
TL0=(65536-1000)/256;
TL1=0xfd;
TH1=0xfd;
ET0=1;
TR0=1;
TR1=1;
// TI=0;
EA=1;
// TI=1;
RAMDATA=0x1F45;
}
void serial () interrupt 4 using 3
{
if(RI)
{ RI=0;
ch=SBUF;
TI=1; //置SBUF空
switch(ch)
{
case 0x01 :printf("A"); TI=0;break;
case 0x02 :printf("B"); TI=0;break;
case 0x03 :printf("C"); TI=0;break;
case 0x04 :printf("D"); TI=0;break;
default :printf("fg"); TI=0;break;
}
}
}
//向串口發(fā)送一個字符
void timer0() interrupt 1 using 3{
// char i;
flag++;
TH0=0x00;
TL0=0x00;
if(flag==10)
{// cp=!cp;
// for(i=0;i<6;i++)
P2=0x25;
TI=1;
temp=*RAMDATA;
printf("%c",temp);
TI=0;
// RAMDATA--;
flag=0;
}
}
//主程序
main()
{
init_serialcomm(); //初始化串口
//向6264中送數(shù)據(jù)
{
*RAMDATA=0x33;
}
while(1)
{
*RAMDATA=0x33;;
}
}
調(diào)試過程中遇到的問題:
1. 發(fā)送過程:在發(fā)送時必須保證TI=1:即發(fā)送緩沖器為空,否則將導(dǎo)致數(shù)據(jù)發(fā)不出去,如果想強制發(fā)送可以用:TI=1.具體發(fā)送數(shù)據(jù):利用printf(“akjdfaklfj”);函數(shù)直接發(fā)送即可。
2. 接收過程:在接收時多選用中斷方式,這樣可以節(jié)約CPU的時間,提高效率,
評論