51单片机串口通信控制LED(hex)-苏荣意

该文详细介绍了如何通过串口通信控制51单片机的LED,包括串口初始化设置,如SCON和PCON寄存器的配置,以及波特率的计算。同时,文章提到了定时器T1在方式2下用于生成波特率,以及中断服务程序中的RI和TI标志位的处理。在实际应用中,通过发送不同十六进制值来控制LED的状态。
摘要由CSDN通过智能技术生成

基于串口通信控制51单片机LED

所需用到的文件

串口初始化

 51单片机默认的是12分频,所以不用配置AUXR寄存器,高级的单片机要调分频系数的时候,可能会用到AUXR寄存器。

  • ①:SCON、PCON的配置。SCON:用于设定串行口的工作方式,这里选择方式1(SM0=0,SM1=1)10位UART(8位数据,1位起始位,1位停止位),波特率可变。REN是允许串行接收控制位,这里是发送,所以不需要置1。SCON=0x40。

 TI:发送中断标志位。在方式 0 时,当串行发送第 8 位数据结束时,或在其它方式,串行发送停止位的开始时,由内部硬件使 TI 置 1,向 CPU 发中断申请。 在中断服务程序中,必须用软件将其清 0,取消此中断申请。

RI:接收中断标志位。在方式 0 时,当串行接收第 8 位数据结束时,或在其 它方式,串行接收停止位的中间时,由内部硬件使 RI 置 1,向 CPU 发中断申请。 也必须在中断服务程序中,用软件将其清 0,取消此中断申请。

PCON:令SMOD=0,其他位不变。PCON&=0x7f。

SMOD:波特率倍增位。在串口方式 1、方式 2、方式 3 时,波特率与 SMOD 有 关,当 SMOD=1 时,波特率提高一倍。复位时,SMOD=0。

  • ②波特率的确定。用8位自动重装(方式2)定时器T1的溢出率来产生波特率。例如:波特率为9600,T1初始值怎么算?首先,算出溢出频率=波特率x16x2=0.3072Mhz,从而得到T1溢出一次的时间=1÷溢出频率 =3.2552us;假设晶振为12Mhz,那么计数一次的时间=1/(12M)x12=1us,所以计数次数=溢出一次的时间÷计数一次的时间=3.2552。发现计数次数不是整数,这是因为晶振频率为12Mhz,存在误差,如果晶振频率为11.0592Mhz,就没有误差,初始值就是2^8-3=253,化成16进制就是0xfd,所以初始值TH1=0xfd,重装值TL1=0xfd。TMOD&=0x0f,TMOD|=0x20。打开T1,TR1=1。
  • ③中断的开放。因为是用T1的溢出得到波特率,所以要禁止T1中断。ET1=0。

 uart.c  

首先配置串口:

①使用定时器1-方式2-"8位自动重置定时模式(TMOD=0x02)"

②串口控制寄存器为方式1-"8步异步(不同波特率)通信方式,波特率可变(SCON=0x50)"

③不加倍速率(PCON&=0x7F)

波特率为9600(SCON方式1的波特率计算公式为:2^SMOD*11.0592MHz/(256-TH0)*12*32=波特率,简化为 波特率=(2^SMOD/32)*(T1溢出率) )

#include "uart.h"
 
void Uart_Init()  //初始化串口uart
{
    TMOD=0x20;
    SCON=0x50;
    PCON&=0x7F;
    TH1=0xfd;   //9600波特率下的初值
    TL1=0xfd;   // TL1= TH1
    ET1=0;   //禁止定时器1被中断
    TR1=1;   //开启定时器1
    ES=1;   //开启串口中断
    EA=1;   //开启总中断
}
 
void Uart_Send_Byte(unsigned char uartData)  //发送一个字节Byte
 
{ //SBUF此时为发送寄存器,需要将要发送的数据uartData存储在SBUF中
    SBUF=uartData;     //作为发送时的SBUF将自动发出存储在其中的数据(uartData)至接收区缓存
    while(!TI);     //发送完成时TI将置1,需要软件清0
    TI=0;     //软件清0
}
main.c 

两个标志位:

RI(接收完成标志位)-dat=SBUF

TI(发送完成标志位)-SBUF=dat

#include <reg52.h>
#include "uart.h"
 
void main()
{
	Uart_Init();     //串口初始化
	P2=0x00;     //将有8个led灯的特殊寄存器置0(我这里置0为LED亮,这一步是检查LED是否正常工作)
	while(1)
	{
	}
}
 
void Uart_Service() interrupt 4  //开启串口通信中断(由RI TI触发)
{
	unsigned char dat;
	/*数据存在SBUF,使用"Uart_Send_Byte(dat)-发送寄存器SBUF"将会自动发出SBUF(SBUF=dat)*/
	while(!RI);    //当接收到发送缓存器的一个字节的数据后时,RI置1
	dat=SBUF;      //此时可将SBUF存储到dat中(dat为一个字节,相当于dat读取接收区SBUF)
	P2=dat;        //再将dat值赋给P2-有8个led灯的特殊寄存器,即可实现控制LED
	Uart_Send_Byte(dat);     //这里是发送一个字节dat
	RI=0;     //手动将RI置0
}

发送十六进制0x00(0b00000000)

发送十六进制0xaa(0b10101010)  

发送十六进制0xFF(0b11111111) 

物联网应用技术2班-20210320069-苏荣意 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值