51单片机拥有一个半双工串口,分别关联P30 P31,使用串口需要注意一下几个寄存器
1.TMOD
串口的始终来源有四种模式,其中,移位寄存器模式不是标准串口,另外两个都是九位数据的通讯模式,一般我们不采用,一般我们采用以为起始一位停止
八位数据的模式来进行串口通讯,这个时候需要定时器1来提供串口波特率时钟
具体说明如下
串口波特率计算方式如下
其中SMOD是波特率倍增位,一般很少使用
T1溢出率计算
此处的12是因为51的周期是时钟周期分频12之后的(具体分频做什么,主要是取指,译码运行等一系列工作),由此可以计算出波特率关系
使用串口需要这几个步骤
1.TMOD设置TMOD,为方式2,无门控 0x20
2.设置TH1 TL1波特率初值(SMOD不设置的情况下为0,所以一般忽略这一步,如果设置了,计算时记得加上)
3.启动定时器TR1
4.设置串口的工作模式,选择工作模式1 设置SM0 SM1
5.根据需要启动REN,使能接收(也可以不使能,查询法,不过会丢数据)
6.根据需要使能中断ES,EA
7.如果使能了中断,记得编写中断处理函数
注意:
如果使能了接收中断,那么在串口中断中,要判断到底是发生了接收中断还是发送中断,因为两个中断共用中断源,依靠TI RI识别
51单片机使用PRINTF
keil内置了编写好的printf函数库,默认情况下,直接包含<stdio.h>就可以使用,但是,这个printf和串口中断最好不要一起工作,也就是说printf工作的时候串口中断发生,会使得printf很慢,因为printf发送的时候会频繁进入中断,有一个比较好的办法是重载putchar,将之前putchar判定TI的位置修改为自己的判定.
另外,在中断中不要使用printf,这会导致系统库函数printf出现重入,系统挂掉就很正常(类似于malloc也不要用)
示例程序如下
#include "uart.h"
u8 Uart0_Send_Ready;
char putchar (char c)
{
if (c == '\n')
{
while (!Uart0_Send_Ready);
Uart0_Send_Ready = 0;
SBUF = 0x0d; /* output CR */
}
while (!Uart0_Send_Ready);
Uart0_Send_Ready = 0;
return (SBUF = c);
}
//51单片机最好使用9600波特率,安全稳定
#define BAUD 9600
#define INIT_VALUE 256-(XTAL/(384*BAUD))
static txSendOver = 0;
void UartInit(void)//default baud is 9600 is nore safe baud
{
u8 value = TMOD;
value &= 0x0f;
value |= 0x20;//使用模式2
TMOD = value;
TH1 = 0xfd;//
TL1 = 0xfd;
//不打开中断
TR1 = 1;
SM0 = 0;//8λuart
SM1 = 1;
REN = 1;//使能接收
EA=1; //打开总中断
ES=1; //打开串口中断
}
void UartSendChar(u8 value)
{
txSendOver = 1;
SBUF = value;
while(txSendOver);
}
void UartSendBuffer(u8* buffer,u8 length)
{
u8 i = 0;
for(i = 0; i < length; i++)
{
UartSendChar(*(buffer+i));
}
}
void UartInt() interrupt 4
{
u8 dat;
if(TI == 1)
{
Uart0_Send_Ready = 1;
TI = 0;
txSendOver = 0;
}
if(RI == 1)
{
RI = 0;
dat=SBUF;
}
}
记得自己添加stdio.h的文件包含