通信按照传统的理解就是信息的传输与交换。对于单片机来说,通信则与传感器、存储芯片、外围控制芯片等技术紧密结合,成为整个单片机系统的“神经中枢”。没有通信,单片机所实现的功能仅仅局限于单片机本身,就无法通过其他设备获得有用信息,也无法将自己产生的信息告诉其它设备。如果单片机通信没处理好的话,它和外围器件的合作程度就受到限制,最终整个系统也无法完成强大的功能,由此可见单片机通信技术的重要性。
UART(Universal Asynchronous Receiver/Transmitter,即通用异步收发器)串行通信是单片机最常用的一种通信技术,通常用于单片机和电脑之间以及单片机和单片机之间的通信。
11.1 串行通信的初步认识
通信按照基本类型可以分为并行通信和串行通信。并行通信时数据的各个位同时传送,可以实现字节为单位通信,但是因为通信线多占用资源多,成本高。比如我们前边用到的
P0 = 0xfe;
一次给
P0
的
8
个
IO
口分别赋值,同时进行信号输出,类似于有
8
个车道同时可以过去
8
辆车一样,这种形式就是并行的,我们习惯上还称
P0
、
P1
、
P2
和
P3
为
51
单片机的
4
组并行总线。
而串行通信,就如同一条车道,一次只能一辆车过去,如果一个
0xfe
这样一个字节的数据要传输过去的话,假如低位在前高位在后,那发送方式就是
0-1-1-1-1-1-1-1-1
,一位一位的发送出去的,要发送
8
次才能发送完一个字节。
在我们的
STC89C52
上,有两个引脚,是专门用来做
UART
串口通信的,一个是
P3.0
一个是
P3.1
,还分别有另外的名字叫做
RXD
和
TXD
,这两个引脚是专门用来进行
UART
通信的,如果我们两个单片机进行
UART
串口通信的话,那基本的演示图如图
11-1
所示。
图
11-1
单片机之间
UART
通信示意图
图中,GND
表示单片机系统电源的参考地,TXD
是串行发送引脚,
RXD
是串行接收引脚。两个单片机之间要通信,首先电源基准得一样,所以我们要把两个单片机的
GND
相互连起来,然后单片机
1
的
TXD
引脚接到单片机
2
的
RXD
引脚上,即此路为单片机1
发送而单片机
2
接收的通道,单片机
1
的
RXD
引脚接到单片机
2
的
TXD
引脚上,即此路为单片机2
发送而单片机
2
接收的通道。这个示意图就体现了两个单片机各自收发信息的过程。
当单片机
1
想给单片机
2
发送数据时,比如发送一个
0xE4
这个数据,用二进制形式表示就是0b11100100,在
UART
通信过程中,是低位先发,高位后发的原则,那么就让
TXD
首先拉低电平,持续一段时间,发送一位
0
,然后继续拉低,再持续一段时间,又发送了一位
0
,然后拉高电平,持续一段时间,发了一位
1......
一直到把
8
位二进制数字
0b11100100
全部发送完毕。这里就牵扯到了一个问题,就是持续的这“一段时间”到底是多久?从这里引入我们通信中的另外重要概念——波特率,也叫做比特率。
波特率就是发送一位二进制数据的速率,习惯上用
baud
表示,即我们发送一位数据的持续时间
=1/baud
。在通信之前,单片机
1
和单片机
2
首先都要明确的约定好他们之间的通信波特率,必须保持一致,收发双方才能正常实现通信,这一点大家一定要记清楚。
约定好速度后,我们还要考虑第二个问题,数据什么时候是起始,什么时候是结束呢?不管是提前接收还是延迟接收,数据都会接收错误。在
UART串行通信的时候,一个字节是
8
位,规定当没有通信信号发生时,通信线路保持高电平,当要发送数据之前,先发一位
0
表示起始位,然后发送
8
位数据位,数据位是先低后高的顺序,数据位发完后再发一位
1
表示停止位。这样本来要发送一个字节
8
位数据,而实际上我们一共发送了
10
位,多出来的两位其中一位起始位,一位停止位。而接收方呢,原本一直保持的高电平,一旦检测到来了一位低电平,那就知道了要开始准备接收数据了,接收到
8
位数据位后,然后检测到停止位,再准备下一个数据的接收了。我们图示看一下,如图
11-2
所示。
图
11-2
串口数据发送示意图
像我们的图
11-2
串口数据发送示意图,实际上是一个时域示意图,就是信号随着时间变化的对应关系。比如在单片机的发送引脚上,左边的是先发生的,右边的是后发生的,数据位的切换时间就是波特率分之一秒,如果能够理解时域的概念,后边很多通信的时序图就很容易理解了。
11.2 串行RS232通信接口
在我们的台式电脑上,有一个
9
针的串行接口,这个串行接口叫做RS232
接口,它和
UART通信有关联,但是由于现在笔记本电脑都不带这种
9
针串口了,所以和单片机通信越来越趋向于使用
USB
虚拟的串口和单片机通信,因此这一节的内容作为了解内容,大家知道有这么回事就行。
我们先来认识一下这个标准串口,串口分为
9
针的和
9
孔的,习惯上我们也称之为公头和母头,如图
11-3
所示。
图
11-3 RS232
通信接口
RS232
接口一共有
9
个引脚,分别定义是:
1
、载波检测
(DCD)
;
2
、接收数据
(RXD)
;
3
、发送数据
(TXD)
;
4
、数据终端准备好
(DTR)
;
5
、信号地线
(SG)
;
6
、数据准备好
(DSR)
;
7
、请求发送
(RTS)
;
8
、清除发送
(CTS)
;
9
、振铃提示
(RI)
。我们要让这个串口和我们单片机进行通信,我们只需要关心其中的
2
脚
(RXD)
,
3
脚
(TXD)
和
5
脚
(GND)
。
虽然这三个脚的名字和我们单片机上的串口名字一样,但是却不能直接和单片机对连直接通信,这是为什么呢?随着我们了解的内容越来越多,我们得慢慢知道,不是所有的电路都是5V代表高电平而0V代表低电平的。对于RS232标准来说,它是个反逻辑,也叫做负逻辑。为何叫负逻辑?它的
TXD
和
RXD
的电压,
-3V
到
-15V
代表是
1
,
3-15V
之间的电压代表是
0
。低电平代表的是
1
,而高电平代表的是
0
,所以称之为负逻辑。因此电脑的
9
针
232
串口是不能和单片机直接连接的,需要用一个转换芯片
MAX232
来完成,如图
11-4
所示。
图
11-4 MAX232
转接图
这个芯片就可以实现把标准RS232
串口电平转换成我们单片机能够识别和承受的UART 0V/5V
电平标准。从这里大家似乎慢慢有点明白了,其实RS232
串口和
UART
串口,他们的协议类型是一样,只是电平不同而已,而
MAX232
这个芯片起到的就是中间人的作用,他把
UART电平转换成RS232电平,也把RS232电平转换成
UART电平,从而实现标准RS232
接口和单片机
UART
之间的通信连接。
11.3 USB转串口通信
随着技术的发展,工业上还有RS232
串口通信的大量使用,但是商业技术的应用上,已经慢慢的使用
USB
转
UART
技术取代了RS232
串口,绝大多数笔记本电脑已经没有串口这个东西了,那我们要实现单片机和电脑之间的通信该如何办呢?
我们只需要在我们电路上添加一个
USB
转串口芯片,就可以成功实现
USB
通信协议和标准
UART
串行通信协议的转换,在我们的开发板上,我们使用的是
CH340T
这个芯片,如图
11-5
所示。
图
11-5 USB
转串口电路
左侧J2
是一组跳线的组合,大家可以在我们板子左下角的跳线位置找到,我们是把3脚和5脚、4脚和6脚通过跳线帽短接到一起。右侧的
CH340T这个电路很简单,把电源电路,晶振电路接好后,
6
脚和
7
脚的DP
和
DM分别接
USB
口的
2
个数据引脚上去,
3
脚和
4
脚通过跳线接到了我们单片机的
TXD
和
RXD
上去。
CH340T
的电路里
3
脚位置加了个
4148
的二极管,是一个小技巧。因为我们的
STC89C52RC
这个单片机下载程序需要冷启动,就是先点下载后上电,上电瞬间单片机会先检测需要不需要下载程序。虽然单片机的
VCC
是由开关来控制,但是由于
CH340T的
3
脚是输出引脚,如果没有此二极管,开关后级单片机在断电的情况下,
CH340T的
3
脚和单片机的
P3.0(即RXD)引脚连在一起,有电流会通过这个引脚流入后级电路并且给后级的电容充电,造成后级有一定幅度的电压,这个电压值虽然只有两三伏左右,但是可能会影响到我们的冷启动。加了二极管后,一方面不影响通信,另外一个方面还可以消除这种问题。这个地方可以暂时作为了解,大家如果自己做这块电路,可以参考一下。
11.4 IO口模拟UART串口通信
为了让大家充分理解
UART
串口通信的原理,我们先用
P3.0
和
P3.1
这两个当做
IO
口来进行模拟实际串口通信的过程,原理搞懂后,我们再使用寄存器配置实现串口通信过程。
对于
UART
串口波特率,常用的值是
300
、
600
、
1200
、
2400
、
4800
、
9600
、
14400
、
19200
、
28800
、
38400
、
57600
、
115200
、
128000
、
256000
等速率。
IO
口模拟
UART
串行通信程序是一个简单的演示程序,我们使用串口调试助手下发一个数据,数据加
1
后,再自动返回。串口调试助手,在我们进行全板子测试视频的时候,大家已经见过,这里我们直接使用
STC-ISP
软件自带的串口调试助手,
可到www.51hei.com去下载此程序,先把串口调试助手使用给大家说一下,如图
11-6
所示。第一步要选择串口助手菜单,第二步选择十六进制显示,第三步选择十六进制发送,第四步选择
COM
口,这个
COM
口要和自己电脑设备管理器里的那个
COM
口一致,波特率是我们程序设定好的选择,我们程序中让一个数据位持续时间是
1/9600
秒,那这个地方选择波特率就是选
9600
,校验位选
N
,数据位
8
,停止位
1
。
图
11-6
串口调试助手示意图
串口调试助手的实质就是我们利用电脑上的
UART
通信接口,通过这个
UART
接口发送数据给我们的单片机,也可以把我们的单片机发送的数据接收到这个调试助手界面上。
因为初次接触通信方面的技术,所以我对这个程序进行一下解释,大家可以边看我的解释边看程序,把底层原理先彻底弄懂。
变量定义部分就不用说了,直接看
main
主函数。首先是对通信的波特率的设定,在这里我们配置的波特率是
9600
,那么串口调试助手也得是
9600
。配置波特率的时候,我们用的是定时器
0
的模式
2
。模式
2
中,不再是
TH0
代表高
8
位,
TL0
代表低
8
位了,而只有
TL0
在进行计数了。当
TL0
溢出后,不仅仅会让
TF0
变
1
,而且还会将
TH0
中的内容重新自动装到
TL0
中。这样有一个好处,我们可以把我们想要的定时器初值提前存在
TH0
中,当
TL0
溢出后,
TH0
自动把初值就重新送入
TL0
了,全自动的,不需要程序上再给
TL0
重新赋值了,配置方式很简单,大家可以自己看下程序并且计算一下初值。
波特率设置好以后,打开中断,然后等待接收串口调试助手下发的数据。接收数据的时候,首先要进行低电平检测 while (PIN_RXD),若没有低电平则说明没有数据,一旦检测到低电平,就进入启动接收函数StartRXD()。接收函数最开始启动半个波特率周期,初学可能这里不是很明白。大家回头看一下我们的图11-2
里边的串口数据示意图,信号在数据位电平变化的时候去读,因为时序上的误差以及信号稳定性的问题很容易读错数据,所以我们希望在信号最稳定的时候去读数据。除了信号变化的那个沿的位置外,其他位置都很稳定,那么我们现在就约定在信号中间位置去读取电平状态,这样能够保证我们信号读的是对的。
一旦读到了起始信号,我们就把当前状态设定成接受状态,并且打开定时器中断,第一次是半个周期进入中断后,对起始位进行二次判断一下,确认一下起始位是低电平,而不是一个干扰信号。以后每经过
9600
分之一秒进入一次中断,并且把这个引脚的状态读到RxdBuf里边。等待接收完毕之后,我们再把这个RxdBuf加1
,再通过
TXD
引脚发送出去,同样需要先发一位起始位,然后发
8
个数据位,再发结束位,发送完毕后,程序运行到while (PIN_RXD),等待第二轮信号接收的开始。
#include
sbit PIN_RXD = P3^0; //
接收引脚定义
sbit PIN_TXD = P3^1; //
发送引脚定义
bit RxdOrTxd = 0; //
指示当前状态为接收还是发送
bit RxdEnd = 0; //
接收结束标志
bit TxdEnd = 0; //
发送结束标志
unsigned char RxdBuf = 0; //
接收缓冲器
unsigned char TxdBuf = 0; //
发送缓冲器
void ConfigUART(unsigned int baud);
void StartTXD(unsigned char dat);
void StartRXD();
void main ()
{
ConfigUART(9600); //
配置波特率为
9600
EA = 1; //
开总中断
while(1)
{
while (PIN_RXD); //
等待接收引脚出现低电平,即起始位
StartRXD(); //
启动接收
while (!RxdEnd); //
等待接收完成
StartTXD(RxdBuf+1); //
接收到的数据
+1
后,发送回去
while (!TxdEnd); //
等待发送完成
}
}
void ConfigUART(unsigned int baud) //
串口配置函数,
baud
为波特率
{
TMOD &= 0xF0; //
清零
T0
的控制位
TMOD |= 0x02; //
配置
T0
为模式
2
TH0 = 256 - (11059200/12) / baud; //
计算
T0
重载值
}
void StartRXD() //
启动串行接收
{
TL0 = 256 - ((256-TH0) >> 1); //
接收启动时的
T0
定时为半个波特率周期
ET0 = 1; //
使能
T0
中断
TR0 = 1; //
启动
T0
RxdEnd = 0; //
清零接收结束标志
RxdOrTxd = 0; //
设置当前状态为接收
}
void StartTXD(unsigned char dat) //
启动串行发送,
dat
为待发送字节数据
{
TxdBuf = dat; //
待发送数据保存到发送缓冲器
TL0 = TH0; //T0
计数初值为重载值
ET0 = 1; //
使能
T0
中断
TR0 = 1; //
启动
T0
PIN_TXD = 0; //
发送起始位
TxdEnd = 0; //
清零发送结束标志
RxdOrTxd = 1; //
设置当前状态为发送
}
void InterruptTimer0() interrupt 1 //T0
中断服务函数,处理串行发送和接收
{
static unsigned char cnt = 0; //bit
计数器,记录当前正在处理的位
if (RxdOrTxd) //
串行发送处理
{
cnt++;
if (cnt <= 8) //
低位在先依次发送
8bit
数据位
{
PIN_TXD = TxdBuf & 0x01;
TxdBuf >>= 1;
}
else if (cnt == 9) //
发送停止位
{
PIN_TXD = 1;
}
else //
发送结束
{
cnt = 0; //
复位
bit
计数器
TR0 = 0; //
关闭
T0
TxdEnd = 1; //
置发送结束标志
}
}
else //
串行接收处理
{
if (cnt == 0) //
处理起始位
{
if (!PIN_RXD) //
起始位为
0
时,清零接收缓冲器,准备接收数据位
{
RxdBuf = 0;
cnt++;
}
else //
起始位不为
0
时,中止接收
{
TR0 = 0; //
关闭
T0
}
}
else if (cnt <= 8) //
处理
8
位数据位
{
RxdBuf >>= 1; //
低位在先,所以将之前接收的位向右移
if (PIN_RXD) //
接收脚为
1
时,缓冲器最高位置
1
;为
0
时不处理即仍保持移位后的
0
{
RxdBuf |= 0x80;
}
cnt++;
}
else //
停止位处理
{
cnt = 0; //
复位
bit
计数器
TR0 = 0; //
关闭
T0
if (PIN_RXD) //
停止位为
1
时,方能认为数据有效
{
RxdEnd = 1; //
置接收结束标志
}
}
}
}
同学们通过学习我们的程序,也慢慢感受到了,程序的延时部分已经不再使用简单的
delay
来完成了,我们要通过我们的程序编写积累,慢慢提高自己灵活运用定时器的能力。一个小小的定时器,可以帮我们完成很多很多工作。
11.5 UART串口通信的基本应用
11.5.1 通信的三种基本类型
我们常用的通信通常可以分为单工、半双工、全双工通信。
单工就是指只允许一方向另外一方传送信息,而另一方不能回传信息。比如我们的电视遥控器,我们的收音机广播等,都是单工通信技术。
半双工是指数据可以在双方之间相互传播,但是同一时刻只能其中一方发给另外一方,比如我们的对讲机就是典型的半双工。
全双工通信就发送数据的同时也能够接受数据,两者同步进行,就如同我们的电话一样,我们说话的同时也可以听到对方的声音。
11.5.2 UART模块介绍
IO
口模拟串口通信,大家了解了串口通信的实质,但是我们的单片机程序却需要不停的检测扫描单片机
IO
口收到的数据,大量占用了
CPU
资源。这时候就会有聪明人想了,其实我们不是很关心通信的过程,我们只需要一个通信的结果,最终得到接收到的数据就行了。这样我们可以在单片机内部做一个硬件模块,让他自动接收数据,接收完了,通知我们一下就可以了,我们的
51
单片机内部就存在这样一个
UART
模块,要正确使用它,当然还得先把对应的特殊功能寄存器配置好。
51
单片机的
UART
串行口的结构由串行口控制寄存器
SCON
、发送和接收电路三部分构成,先来了解一下串口控制寄存器
SCON
。
表11-1 SCON--串行控制寄存器的位分配
(
地址:98H)
可位寻址;复位值:
0x00
;复位源:任何复位
位
|
7
|
6
|
5
|
4
|
3
|
2
|
1
|
0
|
符号
|
SM0
|
SM1
|
SM2
|
REN
|
TB8
|
RB8
|
TI
|
RI
|
表11-2 SCON--串行控制寄存器的位描述
|
|
|
|
|
这两位共同决定了串口通信的模式0
到模式
3
共
4
种模式。我们最常用的就是模式1
,也就是
SM0=0
,
SM1=1
,下边我们重点就讲模式
1
,其他模式从略。
|
|
|
|
|
多机通信控制位(
很少用
)
,模式
1
直接清零。
|
|
|
使能串行接收。由软件置位使能接收,软件清零则禁止接收
|
|
|
模式2
和
3
中将要发送的第
9
位数据
(
很少用
)
|
|
|
模式2
和
3
中接收第
9
位数据
(
很少用
)
,模式
1
用来接收停止位
|
|
|
发送中断标志位,模式1
下,在数据位最后一位发送结束,开始发送停止位时由硬件自动置1
,必须通过软件清零。也就是说,再发送前我们清零
TI
,发送数据,数据发送到停止位时,
TI
硬件置
1
,方便我们
CPU
查询发送完毕状态。
|
|
|
接收中断标志位,当接收电路接收到停止位的中间位置时,RI
由硬件置
1
。也就是说,接收数据之前我们必须清零
RI
,接受数据到停止位的中间位置时,
RI
硬件置
1
,方便我们
CPU
查询到接收状态。
|
前边学了那么多寄存器的配置,相信
SCON
这个地方,对于大多数同学来说已经不是难点了,应该能看懂并且可以自己配置了。对于串口的四种模式,模式
1
是最常用的,就是我们前边提到的
1
位起始位,
8
位数据位和
1
位结束位。因为我们的教程不同于教科书,只要有的功能都一一介绍,我们只介绍实用的技术,所以其他
3
种模式,真正遇到需要使用的时候大家再去查资料就行。
在我们使用
IO
口模拟串口通信的时候,我们串口的波特率是使用定时器
0
的中断体现出来的。在实际串口模块中,有一个专门的波特率发生器用来控制发送数据的速度和读取接收数据的速度。对于
STC89C52RC
单片机来讲,这个波特率发生器只能由定时器
1
或定时器
2
产生,而不能由定时器
0
产生,这和我们模拟的通信是完全不同的概念。
如果用定时器
2
,需要配置额外的寄存器,默认是使用定时器
1
的,我们本章内容主要是使用定时器
1
作为波特率发生器来讲解,方式
1
下的波特率发生器必须使用定时器
1
的模式
2
,也就是自动重装载模式,定时器的初值具体的计算公式是:
TH1 = TL1 = 256 - 晶振值/12 /2/16 /波特率
和波特率有关的还有一个寄存器,是一个电源管理寄存器PCON,他的最高位可以把波特率提高一倍,也就是如果写PCON |=0x80
以后,计算公式就成了
TH1 = TL1 = 256 - 晶振值/12 /16 /波特率
数字的含义这里解释一下,
256
是
8
位数据的溢出值,也就是
TL1
的溢出值,
11059200
就是我们板子上单片机的晶振,
12
是说
1
个机器周期是
12
个时钟周期,值得关注的是这个
16
,重点说明。我们在
IO
口模拟串口通信接收数据的时候,我们采集的是这一位数据的中间位置,而实际上串口模块比我们模拟的要复杂和精确一些。他采取的方式是把一位信号采集
16
次,其中第
7
、
8
、
9
次取出来,这三次中其中两次如果是高电平,那么就认定这一位数据是
1
,如果两次是低电平,那么就认定这一位是
0
,这样一旦受到意外干扰读错一次数据,也依然可以保证最终数据的正确性。
了解了串口采集模式,在这里要给大家留一个思考题。“晶振值
/12/2/16/
波特率”这个地方计算的时候,出现不能除尽,或者出现小数怎么办,允许出现多大的偏差?把这部分理解了,也就理解了我们的晶振为何使用
11.0592M
了。
串口通信的发送和接收电路,我们主要了解一下他们在物理上有
2
个名字相同的
SBUF
寄存器,他们的地址也都是
99H
,但是一个用来做发送缓冲,一个用来做接收缓冲。意思就是说,有
2
个房间,两个房间的门牌号是一样的,其中一个只出人不进人,另外一个只进人不出人,这样的话,我们就可以实现
UART
的全双工通信,相互之间不会产生干扰。但是在逻辑上呢,我们每次只操作
SBUF
,单片机会自动根据对它执行的是“读”还是“写”操作来选择是接收
SBUF
还是发送
SBUF
,后边通过程序,我们就会彻底了解这个问题。
11.5.3 UART串口程序
一般情况下,我们编写串口通信程序的基本步骤如下所示:
1、配置串口为模式
1
。
2、配置定时器
T1
为模式
2
,即自动重装模式。
3、确定波特率大小,计算定时器
TH1
和
TL1
的初值,如果有需要可以使用
PCON
进行波特率加倍。
4、打开定时器控制寄存器
TR1
,让定时器跑起来。
这个地方还要特别注意一下,就是在使用
T1
做波特率发生器的时候,千万不要再使能
T1
的中断了。
我们先来看一下由
IO
口模拟串口通信直接改为使用硬件
UART
模块时程序代码,看看程序是不是简单了很多,因为大部分的工作硬件模块都替我们做了。程序功能和
IO
口模拟的是完全一样的。
#include
void ConfigUART(unsigned int baud);
void main ()
{
ConfigUART(9600); //
配置波特率为
9600
while(1)
{
while (!RI); //
等待接收完成
RI = 0; //
清零接收中断标志位
SBUF = SBUF + 1; //
接收到的数据
+1
后,发送回去;
//等号左边的SBUF
实际上就是发送
SBUF
,因为对它的操作是“写”;
//
等号右边的是接收
SBUF
,因为对它的操作是“读”。
while (!TI); //
等待发送完成
TI = 0; //
清零发送中断标志位
}
}
void ConfigUART(unsigned int baud) //
串口配置函数,
baud
为波特率
{
SCON = 0x50; //
配置串口为模式
1
TMOD &= 0x0F; //
清零
T1
的控制位
TMOD |= 0x20; //
配置
T1
为模式
2
TH1 = 256 - (11059200/12/32) / baud; //
计算
T1
重载值
TL1 = TH1; //
初值等于重载值
ET1 = 0; //
禁止
T1
中断
TR1 = 1; //
启动
T1
}
当然了,这个程序还是在主循环里等待接收中断标志位和发送中断标志位的方法来编写的,而实际工程开发中,当然就不能这么干了,所以就用到了串口中断,来看一下程序。
#include
void ConfigUART(unsigned int baud);
void main ()
{
ConfigUART(9600); //
配置波特率为
9600
while(1);
}
void ConfigUART(unsigned int baud) //
串口配置函数,
baud
为波特率
{
SCON = 0x50; //
配置串口为模式
1
TMOD &= 0x0F; //
清零
T1
的控制位
TMOD |= 0x20; //
配置
T1
为模式
2
TH1 = 256 - (11059200/12/32) / baud; //
计算
T1
重载值
TL1 = TH1; //
初值等于重载值
ET1 = 0; //
禁止
T1
中断
TR1 = 1; //
启动
T1
ES = 1; //
打开串口中断
EA = 1; //
打开总中断
}
void InterruptUART() interrupt 4
{
if (RI) //
接收到字节
{
RI = 0; //
手动清零接收中断标志位
SBUF = SBUF + 1;//
接收数据
+1
发回去,左边为发送SBUF
,右边为接收
SBUF
。
}
if (TI) //
字节发送完毕
{
TI = 0; //
手动清零发送中断标志位
}
}
大家可以试验一下试试,看看是不是和前边用
IO
口模拟通信实现的效果一致,而主循环却完全空出来了,我们就可以随意添加其它功能代码进去。
11.6 字符和数据之间的转换
我们学串口通信的应用主要是实现单片机和电脑之间的信息互发,可以用电脑控制单片机的一些信息,可以把单片机的一些信息状况发给电脑上的软件。下面我们就做一个简单的例程,实现单片机串口调试助手发送的数据,在我们开发板上的数码管上显示出来。
#include
sbit ADDR3 = P1^3; //LED
选择地址线
3
sbit ENLED = P1^4; //LED
总使能引脚
unsigned char code LedChar[] = { //
数码管显示字符转换表
0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,
0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
};
unsigned char LedBuff[6] = { //
数码管
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF
};
unsigned char T0RH = 0; //T0
重载值的高字节
unsigned char T0RL = 0; //T0
重载值的低字节
unsigned char RxdByte = 0; //
串口接收到的字节
void ConfigTimer0(unsigned int ms);
void ConfigUART(unsigned int baud);
void main ()
{
P0 = 0xFF; //P0
口初始化
ADDR3 = 1; //
选择数码管
ENLED = 0; //LED
总使能
EA = 1; //
开总中断
ConfigTimer0(1); //
配置
T0
定时
1ms
ConfigUART(9600); //
配置波特率为
9600
while(1)
{ //
将接收字节在数码管上以十六进制形式显示出来
LedBuff[0] = LedChar[RxdByte & 0x0F];
LedBuff[1] = LedChar[RxdByte >> 4];
}
}
void ConfigTimer0(unsigned int ms) //T0
配置函数
{
unsigned long tmp;
tmp = 11059200 / 12; //
定时器计数频率
tmp = (tmp * ms) / 1000; //
计算所需的计数值
tmp = 65536 - tmp; //
计算定时器重载值
tmp = tmp + 31; //
修正中断响应延时造成的误差
T0RH = (unsigned char)(tmp >> 8); //
定时器重载值拆分为高低字节
T0RL = (unsigned char)tmp;
TMOD &= 0xF0; //
清零
T0
的控制位
TMOD |= 0x01; //
配置
T0
为模式
1
TH0 = T0RH; //
加载
T0
重载值
TL0 = T0RL;
ET0 = 1; //
使能
T0
中断
TR0 = 1; //
启动
T0
}
void ConfigUART(unsigned int baud) //
串口配置函数,
baud
为波特率
{
SCON = 0x50; //
配置串口为模式
1
TMOD &= 0x0F; //
清零
T1
的控制位
TMOD |= 0x20; //
配置
T1
为模式
2
TH1 = 256 - (11059200/12/32) / baud; //
计算
T1
重载值
TL1 = TH1; //
初值等于重载值
ET1 = 0; //
禁止
T1
中断
ES = 1; //
使能串口中断
TR1 = 1; //
启动
T1
}
void LedScan() //LED
显示扫描函数
{
static unsigned char index = 0;
P0 = 0xFF; //
关闭所有段选位,显示消隐
P1 = (P1 & 0xF8) | index; //
位选索引值赋值到
P1
口低
3
位
P0 = LedBuff[index]; //
相应显示缓冲区的值赋值到
P0
口
if (index < 5) //
位选索引
0-5
循环,因有
6
个数码管
index++;
else
index = 0;
}
void InterruptTimer0() interrupt 1 //T0
中断服务函数
{
TH0 = T0RH; //
定时器重新加载重载值
TL0 = T0RL;
LedScan(); //LED
扫描显示
}
void InterruptUART() interrupt 4
{
if (RI) //
接收到字节
{
RI = 0; //
手动清零接收中断标志位
RxdByte = SBUF; //
接收到的数据保存到接收字节变量中
SBUF = RxdByte; //
接收到的数据又直接发回,这叫回显
-"echo"
,以提示用户输入的信息是否已正确接收
}
if (TI) //
字节发送完毕
{
TI = 0; //
手动清零发送中断标志位
}
}
大家在做这个实验的时候,有个小问题要注意一下。因为我们
STC89C52RC
下载程序是使用了
UART
串口下载,下载完程序后,程序运行起来了,可是下载软件最后还会通过串口发送一些额外的数据,所以程序刚下载进去不是显示
00
,而可能是其他数据。大家只要把开关关闭,重新打开一次就好了。
细心的同学可能会发现,在串口调试助手发送选项和接收选项处,还有个“字符格式发送”和“字符格式显示”,这是什么意思呢?
先抛开我们使用的汉字不谈,那么我们常用的字符就包含了
0~9
的数字、
A~Z/a~z
的字母、还有各种标点符号等。那么在单片机系统里面我们怎么来表示它们呢?
ASCII
码(American Standard Code for Information Interchange
,即美国信息互换标准代码)可以完成这个使命:我们知道,在单片机中一个字节的数据可以有
0
~
255
共
256
个值,我们取其中的
0
~
127
共
128
个值赋予了它另外一层涵义,即让它们分别来代表一个常用字符,其具体的对应关系如下表。
表
11-3 ASCII
表
这样我们就在常用字符和字节数据之间建立了一一对应的关系,那么现在一个字节就既可以代表一个整数又可以代表一个字符了,但它本质上只是一个字节的数据,而我们赋予了它不同的涵义,什么时候赋予它那种涵义就看编程者的意图了。
ASCII
码在单片机系统中应用非常广泛,我们后续的课程也会经常使用到它,下面我们来对它做一个直观的认识,同学们一定要深刻理解其本质。
对照上述表格,我们就可以实现字符和数字之间的转换了,比如还是这个程序,我们发送的时候改成字符格式发送,接收还是用十六进制接收,这样接收和数码管好做一下对比。
我们用字符格式发送一个小写的
a
,返回一个十六进制的
0x61
,数码管上显示的也是
61
,
ASCII
码表里字符
a
对应十进制是
97
,等于十六进制的
0x61
;我们再用字符格式发送一个数字
1
,返回一个十六进制的
0x31
,数码管上显示的也是
31
,
ASCII
表里字符
1
对应的十进制是
49
,等于十六进制的
0x31
。这下大家就该清楚了:所谓的十六进制发送和十六进制接收,都是按字节数据的真实值进行的;而字符格式发送和字符格式接收,是按
ASCII
码表中字符形式进行的,但它实际上最终传输的还是一个字节数据。这个表格,当然不需要大家去记住,理解它,用的时候过来查就行了。
通信的学习,不像前边控制部分那么直观了,通信部分我们的程序只能获得一个结果,而其过程我们却无法直接看到,所以慢慢的可能大家就会知道有示波器和逻辑分析仪这类测量仪器。如果学校实验室或者公司里有示波器或者逻辑分析仪这类仪器,可以拿过来抓一下串口波形,直观的了解一下。如果暂时还没有这些仪器,先知道这么回事,有条件再说。因为工具类的东西有的比较昂贵,有条件可以尽量使用学校或者公司的。在这里我用一款简易的逻辑分析仪把串口通信的波形抓出来给大家看一下,大家了解一下即可,如图
11-7
所示。
图11-7
逻辑分析仪串口数据示意图
分析仪和示波器的作用,就是把通信过程的波形抓出来进行分析。先大概说一下波形的意思。波形左边是低位,右边是高位,上边这个波形是电脑发送给单片机的,下边这个波形是单片机回发给电脑的。以上边的波形为例,左边第一位是起始位
0
,从低位到高位依次是
10001100
,顺序倒一下,就是数据
0x31
,也就是
ASCII
码表里的‘
1
’。大家可以注意到分析仪在每个数据位都给标了一个白色的点,表示是数据,起始位和无数据的时候都没有这个白点。时间标
T1
和
T2
的差值在右边显示出来是
0.102ms
,大概是
9600
分之一,稍微有点偏差,在容许范围内即可。通过图
11-7
,我们可以清晰的了解了串口通信的收发的详细过程。
那我们这里再来了解一下,如果我们使用串口调试助手,用字符格式直接发送一个“
12
”,我们在我们的数码管上应该显示什么呢?串口调试助手应该返回什么呢?经过试验发现,我们数码管显示的是
32
,而串口调试助手返回十六进制显示的是
31
、
32
两个数据,如图
11-8
所示。
图
11-8
串口调试助手数据显示
我们用逻辑分析仪把这个数据抓出来看一下,如图
11-9
所示。
图
11-9
逻辑分析仪抓取数据
对于
ASCII
码表来说,数字本身是字符而非数据,所以如果发送“
12
”的话,实际上是是分别发送了“
1
”和“
2
”两个字符,单片机呢,先收到第一个字符“
1
”,在数码管上会显示出
31
这个对应数字,但是瞬间马上就又收到了“
2
”这个字符,数码管瞬间从
31
变成了
32
,而我们视觉上呢,根本是没有办法发现这种快速变化的,所以我们感觉数码管直接显示的是
32
。