随着单片机的使用日益频繁,用其作前置机进行采集和通信也常见于各种应用,一般是利用前置机采集各种终端数据后进行处理、存储,再主动或被动上报给管理站。这种情况下下,采集会需要一个串口,上报又需要另一个串口,这就要求单片机具有双串口的功能,但我们知道一般的51系列只提供一个串口,那么另一个串口只能靠程序模拟。
本文所说的模拟串口, 就是利用51的两个输入输出引脚P1.0和P1.1,置1或0分别代表高低电平,也就是串口通信中所说的位,如起始位用低电平,则将其置0,停止位为高电平,则将其置1,各种数据位和校验位则根据情况置1或置0。
以11.0592MHz的晶振为例,通过定时计数器0产生中断信号来模拟串口电平,下面附上具体源代码。
/*
************************************************************************************
Fuction:使用51单片机GPIO口模拟串口通信,通过定时计数器0来产生中断信号
Software Designer:Jason
************************************************************************************
*/
#include <reg52.h>
sbit P1_0 = P1^0;
sbit P1_1 = P1^1;
#define RXD P1_0
#define TXD P1_1
unsigned char flag;
void init();
void send_byte(unsigned char);
unsigned char rec_byte();
void wait_int();
//将从PC机串口接收到的数据原封不动回传给PC机
void main()
{
unsigned char temp;
init();
while(1)
{
if(RI == 1)
{
RI = 0;
temp = rec_byte();
send_byte(temp);
while(!TI);
TI = 0;
}
}
}
//端口及中断初始化
void init()
{
TMOD = 0x02; //定时器0,方式2
TH0 = 0xfd; //波特率9600
TL0 = TH0;
TR0 = 0; //在发送或接收时打开
ET0 = 1; //允许定时器0中断
EA =1; //允许所有中断
}
//通过串口发送一个字节数据
void send_byte(unsigned char dat)
{
unsigned char i=8;
TR0 = 1; //开启T0中断
TXD = 0; //发送起始位0
wait_int();
while(i--) //发送8位数据
{
TXD = (bit)(dat & 0x01);
wait_int();
dat = dat>>1;
}
TXD = 1; //发送停止位1
wait_int();
TR0 = 0; //关闭T0中断
}
//通过串口接收一个字节数据
unsigned char rec_byte()
{
unsigned char dat=0;
unsigned char i=8;
TR0 = 1; //开启T0中断
wait_int(); //等过起始位电平
while(i--) //接收8位数据
{
dat = dat<<1;
if(RXD)
dat |= 0x80;
wait_int();
}
wait_int(); //等过停止位电平
TR0 = 0; //关闭T0中断
return dat;
}
//等待中断到来
void wait_int()
{
while(!flag);
flag = 0;
}
//中断服务程序
void timer0() interrupt 1
{
flag = 1;
}
本文介绍了一种方法,通过51单片机的GPIO口模拟串口通信,以应对单片机只有一个串口的情况。利用P1.0和P1.1引脚的高低电平变化来模拟串口的位信号,如起始位、数据位、校验位和停止位。并以11.0592MHz晶振为例,展示了如何使用定时计数器0产生中断信号来实现这一功能。
2万+

被折叠的 条评论
为什么被折叠?



