51 单片机UART 串口通信[ 方式1 ]实现

1  51串行口结构

51单片机的UART串行通信是基于其串行口的可编程硬件结构,只要用正确的程序代码通过初始化串行口对应寄存器的形式将其串行硬件结构初始化,再编写符合此串行口通信的程序代码便能够实现串行通信,其硬件结构决定了编程机制( 当然还要靠51芯片内CPU等机制 )。此结构具有UART( 通用异步收发器 )的全部功能,能同时进行数据的发送和接收,也可作为同步移位寄存器使用。此结构集成于单片机内部。

51 单片机UART 串口通信[ 方式1 ]实现_第1张图片

 

Figure1:51串行口结构

 

2  51串行口通信编程机制

决定编程机制的先决条件是51单片机的硬件结构及51芯片内部CPU的执行机制。根据串行通信口硬件结构,实现UART通信可以分为两步:

(1)初始化串行通信口( UART )

  •  配置SCON:设定串口方式( SM0SM1位配置4种方式 );串口是否要接收数据(REN )
  • 设定PCON寄存器的SMOD(若非串口方式0 )
  • 若串口为方式1和方式3,则需要配置TMOD:选择定时器1并配置其初值装载方式(MI M0 ),并设置TH1TL0的初值以确定通信的波特率(由波特率的计算公式 ),然后配置TCON:开启定时器1( TR1 )
  • 允许使用中断( ES ),开启UART中断( EA )

 可以这些寄存器的配置含义可以查看相应的知识点:中断配置IE寄存器、定时/计数方式配置寄存器TMOD、定时/计数控制寄存器及通信寄存器SCON寄存器笔记。

 

(2)程序实现:编写发送/接收数据代码及中断服务程序

注:TIRI的置位可查看SCON寄存器TI  RI被置位的条件

[1]发送的数据需要通过程序指令“SBUF =数据;”来将数据送往串行发送缓冲寄存器中,在此条语句后,要用循环判断SCON寄存器中的TI位是否被硬件置位1(表示发送完毕),等待发送完毕,TI置位被置为1时会进入串行中断服务程序[由此不能在串行中断服务程序里面将TI0,因为从循环里面进入中断后会返回到循环语句中来,所以要在串行中断服务程序之外将TI0,置0的目的是提供下一次数据发送成功的依据或进入中断服务器程序 ],需要用程序代码将TI重新置为0;然后再由CPU控制的时序将数据从TXD( P3.1 )脚发出,经过连接的串口线到从设备(接收数据的设备 )

[2]获取接收到数据时需要通过程序指令“存数据变量=SBUF;”将串行接收缓冲寄存器接收到的数据读取到程序中来,当接收到数据完毕后,RI会被硬件置位1,此时也会进入串行中断服务程序,需要用程序指令将RI重新置位0。当然接收数据是由RXD( P3.0 )引脚完成。

[3]串行中断服务程序的中断号为4,所以编写串行中断服务程序时需要有以下格式:

void 函数名() interrupt 4

{

         中断服务程序内容

}

串行中断服务函数当TIRI被硬件置位1时才被系统调用。TI被硬件置1的情况是,当串行发送缓冲寄存器内的数据被通过TXD( P3.1 )发送完毕时( 不同的串口方式置1条件不同 )不是指将数据存入串行发送缓冲器中时TI会被置1,同理RI被置1的情况是当RXD( P3.0 )接收到数据将此数据按位存入串行接收缓冲寄存器内完毕时( 不同串口方式置1的条件不同 )不是指从串行接收缓冲寄存器读完数据时RI会被置1。这个数据的长度和TI( RI )被置位1根据串行通信方式不同而不同。

 

对于用程序来实现51单片机中的串行通信还需要明确两点(与其它部分的联系):

(1)51单片机串行通信的方式1和方式3中,波特率由定时器1的溢出率决定。

(2)SBUF为特殊功能寄存器,它在程序中的两种不同的指令形式表示不同的含义,当在程序中用写指令“SBUF=a;”时,表示将a存入串行数据发送缓冲寄存器中;当在程序中使用读指令“a=SBUF;”时,表示向串行数据接收缓冲寄存器读数据并将得到数据赋值于a。接收器具有双缓冲结构,即在从接收寄存器中前一个已经收到的字节之前,便能接收第二个字节,如果第二个字节已经接收完毕,第一个字节还没有被读出,则第一个字节就损失( 被覆盖 )。而发送器中的数据是由CPU控制的,所以不必程序用户担忧。

 

3  程序实现串行通信

程序的大体框架为main主函数、串行中断服务程序、子函数(初始化串口、延时函数,字符串处理函数等)

 

(1)首先是用程序实现串行通信的初始化

//串口初始化

void UART_Init(void)

{

         SCON  = 0x40;                   //串口方式1

         PCON  = 0;                     //SMOD=0

         REN  = 1;                               //允许接收

        

         TMOD = 0x20;                            //定时器1定时方式2

         TR1  = 1;                                   //启动定时器

         ES  = 1;                                                              //UART中断

         EA  = 1;                                                              //中断使能

}

 

(2)用程序实现发数据

将要发送的数据赋值给SBUF即可。

SBUF = aChar;

while(TI == 0);//等待,直到发送成功

TI = 0;

这是发送一次数据(如一个字符,在串口发送方式1下占10个位)的代码,可单独写成函数。此种写法不可以将“TI = 0;”写在中断函数中,不然会造成死循环【执行一下代码便知】。

 

(3)在中断函数中接受数据

不管发送数据成功还是接受到数据都会进入到中断程序中,故而在中断程序中判断是否RI被置位为1,如是则表示收到了数据。

void UART_Receive_Int() interrupt 4

{       

         if(  RI == 1)

         {

                   RI     = 0;           

                   if(  index < 1 )

                   {

                           Receive[index] = SBUF;

                            index++;

                   }else{

                            index  = 0;

                   }                

         }       

}

然后只需要在主程序main函数中调用串行口初始化函数及用一定的逻辑调用数据发送函数即可。

 

(4)通过串口调试助手来观察程序执行效果

 

1】发数据显示

设置串口调试助手:

51 单片机UART 串口通信[ 方式1 ]实现_第2张图片

 

Figure2:向调试助手发数据显示

在串口调试助手中按照下划线部分设置后,结果显示区域就会显示由程序发将上来的数据。

 

【2】串口调试助手往单片机发送数据并再显示到串口调试界面之上

51 单片机UART 串口通信[ 方式1 ]实现_第3张图片

 

Figure3:调试助手向单片机发送数据显示界面

对串口调试助手的设置除了发送数据之上的设置之外还需要在表格中填入字符串,然后点击表格下画圈使之为“停止循环发送”,然后就可以将收到的数据同将要发送的数据一起发送到串口调试界面之上来。

 

此次笔记记录完毕。

你可能感兴趣的:(51 单片机UART 串口通信[ 方式1 ]实现)