自学内容网 自学内容网

51单片机嵌入式开发:13、STC89C52RC 之 RS232与电脑通讯


在这里插入图片描述

第十三节课,RS232与电脑通讯

1 概述

RS232(Recommended Standard 232)是一种常用的串行通信接口标准,用于在计算机和外部设备之间传输数据。RS232是由电子工业协会(Electronic Industries Association)制定的,最初是为了连接计算机和调制解调器之间进行数据通信而设计的。以下是关于RS232的一些基本介绍:
(1) 物理接口:RS232使用传统的串行通信方式,使用一对差分信号线进行数据传输。常见的RS232物理接口是使用DB-9或DB-25连接器,其中DB-9连接器具有9个引脚,而DB-25连接器具有25个引脚。
(2) TTL信号:RS232使用两个信号线(通常称为TX和RX)来传输数据。这些信号线与地之间的电压差表示二进制数据的逻辑状态。通常,高电平表示逻辑1,低电平表示逻辑0。RS232底层一般是由UART电路转化而来,如图所示。
在这里插入图片描述

(3) 通信速率:RS232支持多种通信速率(波特率),典型的速率包括2400、4800、9600、19200、38400、57600和115200等。通信双方必须在相同的波特率下进行通信才能正常传输数据。
(4) 数据格式:RS232通信使用异步串行传输方式,其中数据被分为连续的字节进行传输。每个字节包含起始位、数据位、可选的奇偶校验位和停止位。数据位的常见大小为8位。
(5) 控制信号:除了数据传输信号外,RS232还定义了一些控制信号,用于控制通信过程。其中包括RTS(请求发送)、CTS(发送就绪)、DTR(数据终端就绪)和DSR(数据设备就绪)等信号。
(6) 通信距离和线缆:由于RS232信号受到信号损耗和干扰的影响,其通信距离较短,通常不超过50英尺(15米)。常见的RS232线缆是使用三线链接,其中包括发送线、接收线和地线。
(7) 应用领域:RS232广泛应用于计算机和外部设备之间的串行通信,例如连接调制解调器、打印机、终端、传感器、工业设备和其他串口设备等。
需要注意的是,随着技术的发展,RS232逐渐被更快速、更先进的通信接口(如USB、Ethernet等)所取代,但在某些特定的应用场景中仍然广泛使用。

2 Uart介绍

2.1 概述

UART(Universal Asynchronous Receiver-Transmitter)是一种常见的串行通信接口,用于在计算机系统或微控制器与外部设备之间进行数据传输。UART是一个硬件模块,用于将并行数据转换为串行数据以进行传输,并将接收到的串行数据转换为并行数据。
以下是关于UART的一些基本信息:
(1) 功能:UART负责将并行数据转换为串行数据并发送,同时接收串行数据并将其转换为并行数据。它提供了一种异步通信的方式,其中数据被分为连续的字节进行传输。
(2) 数据传输:UART使用两个信号线进行数据传输:一个用于发送数据(TX,传输线路),另一个用于接收数据(RX,接收线路)。发送方将并行数据转换为连续的位流,并通过发送线路将其发送给接收方。接收方通过接收线路接收位流,并将其重新转换为并行数据。
(3) 波特率:UART通信使用波特率来定义数据传输的速率。波特率是指每秒传输的位数。通信双方必须在相同的波特率下进行通信才能正确传输和接收数据。
(4) 数据格式:UART通信中的数据帧通常由起始位、数据位、可选的奇偶校验位和停止位组成。起始位用于指示数据传输的开始,停止位用于指示数据传输的结束。数据位的大小一般为5、6、7或8位。
(5) 逻辑电平:UART通信使用逻辑电平来表示数据的逻辑状态。通常,高电平表示逻辑1,低电平表示逻辑0。逻辑电平的具体电压取决于特定的电平标准,如TTL(Transistor-Transistor Logic)或CMOS(Complementary Metal-Oxide-Semiconductor)。
(6) 缓冲区:UART通信中,发送方和接收方通常都具有缓冲区用于临时存储数据。发送方将数据写入发送缓冲区,而接收方从接收缓冲区读取接收到的数据。
(7) 应用领域:UART广泛应用于各种设备和系统中,包括嵌入式系统、传感器、通信模块、无线模块、显示器、调制解调器、打印机和许多其他外部设备。
需要注意的是,UART是一种硬件接口,与特定的通信协议无关。在实际应用中,UART通常与其他协议(如RS232、RS485等)结合使用,以实现特定的数据传输要求。

2.2 STC89C52UART介绍

STC89C52中的串行I/O端口与80C52中的串口兼容。它提供同步和异步通信模式。它作为通用异步收发器(UART)在三种全双工模式(模式1、2和3)下运行。异步传输和接收可以同时发生,并且具有不同的波特率串行I/O端口包括以下增强功能:
•帧错误检测
•自动地址识别
(1) UART模块:STC89C52芯片内部集成了一个UART模块,该模块提供了与外部设备进行串口通信的功能。
(2) 引脚:串口通信需要使用芯片上的特定引脚与外部设备连接。在STC89C52中,串口通信使用P3.0(RXD)和P3.1(TXD)引脚。其中,RXD引脚用于接收数据,TXD引脚用于发送数据。

在这里插入图片描述

(3) 波特率设置:UART通信的一个重要参数是波特率,它定义了数据传输的速率。在STC89C52中,可以通过设置相关的寄存器来配置波特率。常见的波特率设置可以是9600、19200、38400、57600等。
(4) 数据格式:UART通信涉及的数据格式包括数据位、奇偶校验和停止位等。在STC89C52中,可以通过设置寄存器来配置数据位数、奇偶校验和停止位数。常见的数据格式是8位数据位、无奇偶校验和1个停止位。
(5) 缓冲区:STC89C52的UART模块具有发送和接收缓冲区,用于临时存储数据。发送数据时,将数据写入发送缓冲区,然后UART模块将数据从缓冲区发送到TXD引脚。接收数据时,UART模块从RXD引脚接收数据,并将数据存储在接收缓冲区中,供单片机读取。
(6) 中断:STC89C52的UART模块支持中断机制,可以通过配置相关的中断寄存器来启用接收和发送中断。使用中断可以提高串口通信的效率和可靠性。
(7) 编程:使用STC89C52的串口通信功能,需要在单片机程序中配置相关的寄存器和处理接收和发送的数据。具体的编程方式和操作方法可以参考STC89C52的数据手册和开发工具的相关文档。

2.3 STC89C52 UART寄存器介绍

UART寄存器用于配置和控制UART通信的参数和操作。以下是STC89C52中与UART相关的一些常用寄存器:
(1) SCON(Synchronous Control Register):用于配置UART的工作模式和控制UART的操作。其中的位字段包括:
SM0和SM1:UART工作模式选择位,用于设置UART的工作模式(如波特率、数据位数、校验位等)。
REN:接收使能位,用于启用或禁用UART的接收功能。
TI:发送中断标志位,用于指示上一次发送是否完成。
RI:接收中断标志位,用于指示接收到数据。
(2) TMOD(Timer/Counter Mode Register):用于配置定时器/计数器的工作模式。其中的位字段包括:
T1M1和T1M0:定时器1工作模式选择位。
T0M1和T0M0:定时器0工作模式选择位。
TH1和TL1(Timer1 High Byte 和 Timer1 Low Byte):用于配置定时器1的初值,从而设置UART的波特率。
(3) PCON(Power Control Register):用于配置和控制电源管理功能。其中的位字段包括:
SMOD:串口模式选择位,用于控制UART的波特率加倍。
其他位字段用于配置和控制其他电源管理功能。
(4) SBUF(Serial Buffer Register):用于存储UART接收和发送的数据。
这些寄存器提供了对UART通信的配置和控制,您可以根据具体的需求和应用,对这些寄存器进行适当的设置和操作。

2.4 STC89C52 UART操作

(1)简单的接收发送程序

/******************************************************************/
void sys_uart_init(void)
{


SCON  = 0x50;        /* SCON: 模式 1, 8-bit UART, 使能接收         */
TMOD |= 0x20;               /* TMOD: timer 1, mode 2, 8-bit reload        */
TH1   = 0xFD;               /* TH1:  reload value for 9600 baud @ 11.0592MHz   */
TR1   = 1;                  /* TR1:  timer 1 run                          */
EA    = 1;                  /*打开总中断*/
ES    = 1;                  /*打开串口中断*/

}

/******************************************************************/
/*                  串口中断程序                                  */
/******************************************************************/


static unsigned char Temp = 0;          //定义临时变量 


void UART_isr(void) interrupt 4 //串行中断服务程序
{
if(RI)                        //判断是接收中断产生
{
RI=0;                      //标志位清零
Temp=SBUF;                 //读入缓冲区的值
P1=Temp;                   //把值输出到P1口,用于观察
SBUF=Temp;                 //把接收到的值再发回电脑端
}
if(TI)                        //如果是发送标志位,清零
{
TI=0;
}
}

(2)带有一定校验判断功能的用法

/******************************************************************/
/*                  串口中断程序                                  */
/******************************************************************/


static unsigned char uart_temp = 0;          //定义临时变量 
static unsigned char uart_cnt = 0;          //定义临时变量 


void UART_isr(void) interrupt 4 //串行中断服务程序
{
if(RI)                        //判断是接收中断产生
{
RI=0;                      //标志位清零
uart_temp=SBUF;                 //读入缓冲区的值
if(uart_cnt==0)
{
if(0x02 == uart_temp)
{
uart_cnt = 1;
}
else
{
uart_cnt = 0;
}
}
else if(uart_cnt==1)
{
if(0x05 == uart_temp)
{
uart_cnt = 2;
}
else
{
uart_cnt = 0;
}
}
else if(uart_cnt==2)
{
uart_cnt = 0;
P1=uart_temp;                   //把值输出到P1口,用于观察
SBUF=uart_temp;                 //把接收到的值再发回电脑端
}
else
{
uart_cnt = 0;
}
}
if(TI)                        //如果是发送标志位,清零
{
TI=0;
}
} 

(3)重定向使用printf
首先屏蔽uart中断,其次引用文件stdio.h文件,然后重定向putchar函数,至此,使用printf可以实现打印功能。

//c51_uart.c文件

#include "includes.h"



/*-----------------------------------------------
  名称:串口通信
  内容:连接好串口或者usb转串口至电脑,下载该程序,打开电源
        打开串口调试程序,将波特率设置为9600,无奇偶校验
        晶振11.0592MHz,发送和接收使用的格式相同,如都使用
        字符型格式,在发送框输入 hello,I Love MCU ,在接
        收框中同样可以看到相同字符,说明设置和通信正确
------------------------------------------------*/
                    


/******************************************************************/
void sys_uart_init(void)
{


SCON  = 0x50;        /* SCON: 模式 1, 8-bit UART, 使能接收         */
TMOD |= 0x20;               /* TMOD: timer 1, mode 2, 8-bit reload        */
TH1   = 0xFD;               /* TH1:  reload value for 9600 baud @ 11.0592MHz   */
TR1   = 1;                  /* TR1:  timer 1 run                          */
EA    = 1;                  /*打开总中断*/
//ES    = 1;                  /*打开串口中断*/

}


void Uart_SendChar(unsigned char  dat)
{
    SBUF = dat; 
    while(!TI); 
    TI = 0; 
}
 
char putchar(char c)//重定向
{
    Uart_SendChar(c);
    return c;

}

/******************************************************************/
/*                  串口中断程序                                  */
/******************************************************************/


static unsigned char uart_temp = 0;          //定义临时变量 
static unsigned char uart_cnt = 0;          //定义临时变量 


void UART_isr(void) interrupt 4 //串行中断服务程序
{
if(RI)                        //判断是接收中断产生
{
RI=0;                      //标志位清零
uart_temp=SBUF;                 //读入缓冲区的值
if(uart_cnt==0)
{
if(0x02 == uart_temp)
{
uart_cnt = 1;
}
else
{
uart_cnt = 0;
}
}
else if(uart_cnt==1)
{
if(0x05 == uart_temp)
{
uart_cnt = 2;
}
else
{
uart_cnt = 0;
}
}
else if(uart_cnt==2)
{
uart_cnt = 0;
P1=uart_temp;                   //把值输出到P1口,用于观察
SBUF=uart_temp;                 //把接收到的值再发回电脑端
}
else
{
uart_cnt = 0;
}
}
if(TI)                        //如果是发送标志位,清零
{
TI=0;
}
} 

在这里插入图片描述
(5)整体工程代码如下

//main.c文件

#include "includes.h"



/******************************************************************/
/*                    微秒延时函数  //10us                         */
/******************************************************************/
void delay_us(unsigned int us)//delay us
{
while(us--)
{
}
}

/******************************************************************/
/*                    微秒延时函数                                */
/******************************************************************/
void delay_ms(unsigned int Ms)//delay us
{
while(Ms--)
{
delay_us(100);
}
}

/*------------------------------------------------
                    延时子程序
------------------------------------------------*/
void delay(unsigned int cnt) 
{
 while(--cnt);
}




/*------------------------------------------------
                    主函数
------------------------------------------------*/
void main (void)
{
sys_timer_init();
sys_uart_init();

delay(10);
delay_ms(10);


sys_ledtube_on2();

sys_ledtube_on1();

//首先定义处于什么状态,
tx1838_type = 1;
printf("hello");
while (1)
{
//sys_keynum_ledon(9);
sys_tx1838_test();
}
}

//includes.h文件

#ifndef __INCLUDES_H__
#define __INCLUDES_H__

//#include<reg52.h> 

#include<intrins.h> //汇编指令_nop_
#include<stdio.h> //标准输入输出

//_nop_(); 产生一条NOP指令
//作用:对于延时很短的,要求在us级的,采用“_nop_”函数,这个函数相当汇编NOP指令,延时几微秒。
//NOP指令为单周期指令,可由晶振频率算出延时时间。

//8051 为每个机器周期 12 时钟
//对于12M晶振,延时1uS。
//11.0592M晶振,延时1.0851uS。

//对于延时比较长的,要求在大于10us,采用C51中的循环语句来实现。


//包含头文件,一般情况不需要改动,头文件包含特殊功能寄存器的定义
#include "STC89C5xRC_RDP.h"

//应用层头文件
//#include "c51_gpio.h"
#include "c51_ledtube.h"
//#include "c51_key.h"
#include "c51_timer.h"
//#include "c51_exit.h"
//#include "c51_lcd1602.h"
//#include "c51_iic.h"
#include "c51_tx1838.h"
#include "c51_uart.h"


extern void delay(unsigned int cnt);
extern void delay_us(unsigned int us);//delay us;
extern void delay_ms(unsigned int Ms);//delay Ms;


#endif

//c51_tx1838.h文件

#ifndef __C51_TX1838_H__
#define __C51_TX1838_H__


#define  NEC P32 //红外线接收头  

extern unsigned char tx1838_cnt;
extern unsigned char tx1838_type;
extern unsigned char tx1838_data[4];


extern void sys_tx1838_test(void);

#endif

//c51_timer.c文件

#include "includes.h"


void sys_timer_init(void)
{
sys_timer0_init();
sys_timer1_init();
sys_timer2_init();
sys_wdog_init();
clr_wdg();
}


/*------------------------------------------------
                    定时器初始化子程序
------------------------------------------------*/
void sys_timer0_init(void)
{
TMOD |= 0x01;  //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响     
TH0=0x00;      //给定初值,这里使用定时器最大值从0开始计数一直到65535溢出
TL0=0x00;
//EA=1;            //总中断打开 等最后一个中断打开
//ET0=1;           //定时器中断打开
TR0=1;           //定时器开关打开
}

/*------------------------------------------------
                    定时器初始化子程序
------------------------------------------------*/
void sys_timer1_init(void)
{
TMOD |= 0x20;  //使用模式2,     
TH1=0x05;      //给定初值,这里使用定时器最大值从5开始计数一直到255溢出
TL1=0x00;
//EA=1;            //总中断打开
//ET1=1;           //定时器中断打开

//TR1=1;           //定时器开关打开
}


/*------------------------------------------------
                    定时器初始化子程序
------------------------------------------------*/
void sys_timer2_init(void)
{
RCAP2H = 0/256;//
RCAP2L = 0/256;
//ET2=1;                     //打开定时器中断
//EA=1;                      //打开总中断
  
//TR2=1;                     //打开定时器开关
}




void sys_wdog_init(void)
{ 
//WDT_CONTR = 0x35;
}

void clr_wdg(void)
{
//WDT_CONTR = 0x35;
}


/*------------------------------------------------
                 定时器中断子程序
------------------------------------------------*/
void Timer0_isr(void) interrupt 1
{
TH0=0x00;  //重新赋值
TL0=0x00;

//sys_led_test1(); //流水灯操作
}


/*------------------------------------------------
                 定时器中断子程序
------------------------------------------------*/
void Timer1_isr(void) interrupt 3
{

//sys_led_test1(); //流水灯操作

}




/*------------------------------------------------
                 定时器中断子程序
------------------------------------------------*/
void Timer2_isr(void) interrupt 5//定时器2中断
{
    TF2=0;
    //sys_led_test1(); //流水灯操作
}

//c51_timer.h文件

#ifndef __C51_TIMER_H__
#define __C51_TIMER_H__




extern void sys_timer_init(void);
extern void sys_timer0_init(void);
extern void Timer0_isr(void);
extern void sys_timer1_init(void);
extern void Timer1_isr(void);
extern void sys_timer2_init(void);
extern void Timer2_isr(void);

extern void sys_wdog_init(void);
extern void clr_wdg(void);

#endif

//c51_ledtube.c文件

#include "includes.h"

// 显示段码值01234567,可对应原理图查看显示不同图形对应的引脚高点电平配置状态
unsigned char const EL[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,\
                   0x77,0x7c,0x39,0x5e,0x79,0x71};//0-F



/********************************************************
函数名称:sys_ledtube_on1
函数功能:点亮一个数码管全为亮起来
入口参数:
出口参数:
修    改:
内    容:
********************************************************/
void sys_ledtube_on1(void)
{
//根据原理图,将P0口全部输出高电平,P2选择0号数码管
P0=0xFF;//取显示数据,段码
P2=0;  //取位码
}

/********************************************************
函数名称:sys_ledtube_on2
函数功能:显示一组数据
入口参数:
出口参数:
修    改:
内    容:
********************************************************/
static unsigned char ledtube_cnt = 0;
void sys_ledtube_on2(void)
{
ledtube_cnt++;
if(ledtube_cnt>7)
{
ledtube_cnt = 0;
}
P0 = 0x00;//防止切换数码管瞬间有虚影出现
P2 = 0x00;
P0 = EL[ledtube_cnt];//取显示数据,段码
P2 = ledtube_cnt;  //取位码

//根据人眼适应虚影缓冲时间为50ms左右
//我们调整delay在500以下可以看到明显的看起来是一串数据一起显示
delay(100); 
}


/********************************************************
函数名称:sys_keynum_ledon
函数功能:显示按键数值
入口参数:按键数值
出口参数:
修    改:
内    容:
********************************************************/
void sys_keynum_ledon(unsigned char num)
{
//根据原理图,将P0口全部输出高电平,P2选择0号数码管
P0=EL[num];//取显示数据,段码
P2=0;  //取位码
}

//c51_ledtube.h文件

#ifndef __C51_LEDTUBE_H__
#define __C51_LEDTUBE_H__


extern unsigned char const EL[];

extern void sys_ledtube_on1(void);
extern void sys_ledtube_on2(void);

extern void sys_keynum_ledon(unsigned char num);


#endif

//c51_tx1838.c文件

#include "includes.h"


unsigned char tx1838_cnt = 0;
unsigned char tx1838_type = 0;
unsigned char tx1838_data[4] = {0};


void sys_tx1838_test(void)
{
unsigned int time = 0;
if(!NEC)
{
while(!NEC);//等待低电平结束
TH0 = 0;
TL0 = 0;
delay_us(1);
if(tx1838_type==4)
{
tx1838_type =1;
}
else
{
while(NEC)//等待数据位计时
{
if(TH0>30)
{
tx1838_type =1;
break;
}
}
}
time =(TH0<<8)+TL0; //取得脉冲宽度

switch(tx1838_type)
{
case 1:
{
if(time>3000 && time<7000) //接收到数据
{
tx1838_type = 2;

tx1838_cnt = 0;//接收位数量清0
tx1838_data[0] = 0;
tx1838_data[1] = 0;
tx1838_data[2] = 0;
tx1838_data[3] = 0;
}
else if(time>2000 && time<3000)//接收到重复码
{
tx1838_type = 3;
}
else
{
tx1838_type = 1;
}
break;
}
case 2:
{
tx1838_cnt ++ ;
if(time>168 && time<800) //接收到数据位为0的时间长度
{

}
else 
{
if(time>1100 && time<1800) //接收到数据位为1的时间长度
{
if(tx1838_cnt<=8)
{
tx1838_data[0] |= (1<<(tx1838_cnt-1));
}
else if(tx1838_cnt<=16)
{
tx1838_data[1] |= (1<<(tx1838_cnt-9));
}
else if(tx1838_cnt<=24)
{
tx1838_data[2] |= (1<<(tx1838_cnt-17));
}
else if(tx1838_cnt<=32)
{
tx1838_data[3] |= (1<<(tx1838_cnt-25));
}
else
{
tx1838_type = 1;

tx1838_cnt = 0;//接收位数量清0
tx1838_data[0] = 0;
tx1838_data[1] = 0;
tx1838_data[2] = 0;
tx1838_data[3] = 0;
}
}
else //重新解码 //接收到引导码或者结束码,或者接收到的是重复码,本章节不进行演示
{
tx1838_type = 1;

tx1838_cnt = 0;//接收位数量清0
tx1838_data[0] = 0;
tx1838_data[1] = 0;
tx1838_data[2] = 0;
tx1838_data[3] = 0;
}
}

if(tx1838_cnt>=32)
{
tx1838_type = 4;

switch(tx1838_data[3])//判断数码值
{
case 255:sys_keynum_ledon(0);break;//0 显示相应的按键值
case 254:sys_keynum_ledon(1);break;//1
case 253:sys_keynum_ledon(2);break;//2
case 252:sys_keynum_ledon(3);break;//3
case 251:sys_keynum_ledon(4);break;//4
case 250:sys_keynum_ledon(5);break;//5
case 249:sys_keynum_ledon(6);break;//6
case 248:sys_keynum_ledon(7);break;//7
case 247:sys_keynum_ledon(8);break;//8
case 246:sys_keynum_ledon(9);break;//9 显示相应的按键值

}
}
break;
}
case 3: //重复码
{
tx1838_type = 1;
sys_keynum_ledon(11);
break;
}
case 4: //结束码
{
tx1838_type = 1;

tx1838_cnt = 0;//接收位数量清0

break;
}
case 5:
{
break;
}
default:
tx1838_type = 1;
break;
}

}
}

//c51_uart.h文件

#ifndef __C51_UART_H__
#define __C51_UART_H__


extern void Uart_SendChar(unsigned char  dat);
extern char putchar(char c);//重定向

extern void sys_uart_init(void);
extern void UART_isr(void);


#endif

3 C51 UART总结

这些总结要点提供了关于C51中UART功能的基本概述。请注意,具体的UART配置和操作方法可能因不同的C51系列和具体的芯片型号而有所差异。为了准确使用C51的UART功能,请参考相应的芯片手册、官方文档或相关资料,以获取详细的UART配置和操作说明。


原文地址:https://blog.csdn.net/sinat_30943509/article/details/140571895

免责声明:本站文章内容转载自网络资源,如本站内容侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!