51单片机串口通信

目录

串口初始化

SCON

PCON

TMOD

波特率设置

代码实现

收发消息以及串口中断

主函数

完整代码 


最近在学习51单片机,需要做一个通过串口与MCU通信来控制LED亮灭的小项目。不多逼逼,直接开搞。

仿真电路图
Proteus8 仿真原理图

 

串口初始化

使用串口通信,需要对SCON、PCON和TMOD三个特殊寄存器进行配置。

SCON

串口工作方式寄存器SCON(98H)的结构如下表所示:

SCON
76543210
意义SM0SM1SM2RENTB8RB8TI

RI

SM2:多机通信控制位。

REN:串行接收允许位。置1表示允许,置0表示禁止。

TB8:使用奇偶校验时,发送数据校验位的存放位。

RB8:使用奇偶校验时,接收数据校验位的存放位。

TI:发送中断标志位。数据帧的停止位开始发送时,由硬件自动置1,需软件置0,方可继续发送下一帧。

RI:接收中断标志位。数据帧的停止位被接收到时,由硬件自动置1,需软件置0,方可继续接收下一帧。

SM0、SM1:串口工作方式。

51单片机通用异步收发器的工作方式表
SM0SM1方式说明波特率备注
000移位寄存器f_{osc}/12f_{osc}为晶振频率,51单片机的机器周期为12个时钟周期
01110位异步收发器可变1个起始位,1个停止位,8个数据位
10211位异步收发器f_{osc}/64f_{osc}/321个起始位,1个停止位。8个数据位,其中最后一位可作校验位
11311位异步收发器可变1个起始位,1个停止位。8个数据位,其中最后一位可作校验位

 

常用方式1。

51单片机UART方式1时序图
51单片机UART方式1时序图

PCON

PCON(97H)中只有最高位SMOD与串行通信有关。

PCON
76543210
意义SMOD-------

SMOD:波特率倍增标志位。

TMOD

TMOD(89H)用于设定计数/定时器的工作方式。

TMOD
76543210
意义GATEC/\overline{T}M1M0GATEC/\overline{T}M1M0

上表中,高四位控制T1,第四位控制T0。

  • GATE

0:仅由运行控制位TRx(x = 0,1,位于TCON中)来控制定时器/计数器运行。
1:用外中断引脚( INT0或 INT1)上的电平与运行控制位TRx共同来控制定时器/计数器运行。

  • C/\overline{T}

0:为定时器工作模式,对单片机的晶振频率12分频后的脉冲进行计数。
1:为计数器工作模式,计数器对外部输入引脚T0(P3.4)或T1(P3.5)的外部脉冲(负跳变)计数。

  • M1、M0
M1M0方式说明
00013位定时器/计数器
01116位定时器/计数器
102低8位作8位定时器/计数器,高8位存放自动装入低8位的初值
113T1停止计数,T0分为两个8位定时器/计数器

 

波特率设置

TCON采用方式1(SM0=0,SM1=1)。波特率由定时器T1设定得到。波特率的计算公式如下:

BaudRate=\frac{2^{SMOD}}{32}\times\frac{f_{osc}/12}{256-X}

X是定时器T1的初始值。

反过来,例如晶振频率为11.0592MHz时,要使波特率为9600,可使SMOD=0,定时器T1工作在方式2(TMOD=0x20),初始值应为:

\small X =256-\frac{2^{SMOD}}{32}\times\frac{f_{osc}/12}{BaudRate}=256-\frac{2^0}{32}\times\frac{11.0592\times10^6/12}{9600}=253=FDH

代码实现

void init_UART(void)
{
    SCON = 0x50;    // 设定串行工作方式1
    TMOD = 0x20;    // 设定定时器T1工作方式2 M1=1,M0=0
    TH1 = 0xfd;     // 设定定时器T1初始值设置为FDH
    TL1 = TH1;
    TR1 = 1;        // 使能定时器T1
    EA = 1;         // 使能总中断
    ES = 1;         // 使能串口中断
    PCON |= 0x80;   // SMOD置1,波特率倍增为19200
}

收发消息以及串口中断

// 串口中断函数
/*
每当串口收发中断发生,该函数被执行

当RI == 1,即接收完一个数据帧,接收中断发生时,
符合约定好的消息协议的消息会被写入全局字符串变量recv_buff中
该消息协议指接收到的字符串必须以字符'$'开始,以字符'#'结尾,中间部分为消息的内容,
写入recv_buff中,并将RI置0
接收到完整的符合协议的消息后,全局变量recved_whole置1

当TI == 1,即发送完一个数据帧,发送中断发生时,
将TI置0,并把TX总线忙状态清除,即全局变量TX_busy = 0
*/
void UART_interrupt() interrupt 4
{
    uchar byte_buff = 0;    // unsigned char 大小为1个字节
    static ucahr i = 0;
    static bit has_recved = 0;    // 接收到消息开始字符'$'时置1
    if(RI)
    {
        RI = 0;
        byte_buff = SBUF;
        if(byte_buff=='$' && !has_recved)
            has_recved = 1;
        else if(byte_buff=='#' && has_recved)
        {
            has_recved = 0;
            recv_buff[i] = 0;
            recved_whole = 1;
            i = 0;
        }
        else if(has_recved)
            recv_buff[i++] = byte_buff;  
    }
    else
    {
        TI = 0;
        TX_busy = 0;
    }
}


// 通过串口发送字符串
void sendString(uchar *str)
{
    while(*str!=0)
    {
        if(!TX_busy)    // 等待TX总线空闲,以进行下一个字符的发送
        {
            SBUF = *str;
            str++;
            TX_busy = 1;    // 置TX总线状态为忙,中断函数会把此值重新置0
        }
    }
}

主函数

要求根据上位机(PC)发送的指令来控制两个共阴极发光二极管LED_0、LED_1的亮灭,单片机做出响应回应:

上位机指令说明全局字符串变量recv_buff的值单片机回应
$00#将LED_0连接的IO口置0,点亮LED_000LED0 ON
$01#将LED_0连接的IO口置1,熄灭LED_001LED0 OFF
$10#将LED_1连接的IO口置0,点亮LED_110LED1 ON
$11#将LED_1连接的IO口置1,熄灭LED_111LED1 OFF
其他除上述指令以外的所有字符串 ERROR
void main()
{
    init_UART();
    while(1)
    {
        if(recved_whole)   // 是否已经接收到了一个完整的消息
        {
            if(recv_buff[0]=='0' && recv_buff[1]=='1' && recv_buff[2]==0)
            {
                LED_0 = 1;
                sendString("LED0 OFF");
            }
            else if(recv_buff[0]=='0' && recv_buff[1]=='0' && recv_buff[2]==0)
            {
                LED_0 = 0;
                sendString("LED0 ON");
            }
            else if(recv_buff[0]=='1' && recv_buff[1]=='1' && recv_buff[2]==0)
            {
                LED_1 = 1;
                sendString("LED1 OFF");
            }
            else if(recv_buff[0]=='1' && recv_buff[1]=='0' && recv_buff[2]==0)
            {
                LED_1 = 0;
                sendString("LED1 ON");
            }
            else
                sendString("ERROR");
            recved_whole = 0;
        }
    }
}

完整代码 

#include <reg52.h>

#define uchar unsigned char
#define uint unsigned int

sbit LED_0 = P1^0;            // 共阴极发光二极管 低电平点亮
sbit LED_1 = P1^1;
uchar recv_buff[10] = {0};    // 用于存储从PC接收到的字符串
bit recved_whole = 0;         // 接收到完整指令的标志位
bit TX_busy = 0;              // TX总线忙标志位

void init_UART();
void sendString(uchar*);

void main()
{
    init_UART();
    while(1)
    {
        if(recved_whole)
        {
            if(recv_buff[0]=='0' && recv_buff[1]=='1' && recv_buff[2]==0)
            {
                LED_0 = 1;
                sendString("LED0 OFF");
            }
            else if(recv_buff[0]=='0' && recv_buff[1]=='0' && recv_buff[2]==0)
            {
                LED_0 = 0;
                sendString("LED0 ON");
            }
            else if(recv_buff[0]=='1' && recv_buff[1]=='1' && recv_buff[2]==0)
            {
                LED_1 = 1;
                sendString("LED1 OFF");
            }
            else if(recv_buff[0]=='1' && recv_buff[1]=='0' && recv_buff[2]==0)
            {
                LED_1 = 0;
                sendString("LED1 ON");
            }
            else
                sendString("ERROR");
            recved_whole = 0;
        }
    }
}

void init_UART(void)
{
    SCON = 0x50;    // 串口工作方式1
    TMOD = 0x20;    // 定时器工作方式2
    TH1 = 0xfd;     // 定时器初始值设置为FDH
    TL1 = TH1;
    TR1 = 1;        // 使能定时器1
    EA = 1;         // 使能总中断
    ES = 1;         // 使能串口中断
    PCON |= 0x80;   // SMOD置1,波特率倍增为19200;否则波特率为9600
}

// 串口中断函数
void UART_interrupt() interrupt 4
{
    uchar byte_buff = 0;
    static uchar i = 0;
    static bit has_recved = 0;
    if(RI)
    {
        RI = 0;
        byte_buff = SBUF;
        if(byte_buff=='$' && !has_recved)
            has_recved = 1;
        else if(byte_buff=='#' && has_recved)
        {
            has_recved = 0;
            recv_buff[i] = 0;
            recved_whole = 1;
            i = 0;
        }
        else
            recv_buff[i++] = byte_buff;  
    }
    else if(has_recved)
    {
        TI = 0;
        TX_busy = 0;
    }
}

// 通过串口发送字符串
void sendString(uchar *str)
{
    while(*str!=0)
    {
        if(!TX_busy)
        {
            SBUF = *str;
            str++;
            TX_busy = 1;
        }
    }
}

 以上。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值