51单片机GPIO口模拟串口通信

 随着单片机的使用日益频繁,用其作前置机进行采集和通信也常见于各种应用,一般是利用前置机采集各种终端数据后进行处理、存储,再主动或被动上报给管理站。这种情况下下,采集会需要一个串口,上报又需要另一个串口,这就要求单片机具有双串口的功能,但我们知道一般的51系列只提供一个串口,那么另一个串口只能靠程序模拟。

        本文所说的模拟串口, 就是利用51的两个输入输出引脚P1.0和P1.1,置1或0分别代表高低电平,也就是串口通信中所说的位,如起始位用低电平,则将其置0,停止位为高电平,则将其置1,各种数据位和校验位则根据情况置1或置0。

        以11.0592MHz的晶振为例,通过定时计数器0产生中断信号来模拟串口电平,下面附上具体源代码。

[cpp]  view plain copy
  1. /* 
  2. ************************************************************************************ 
  3. Fuction:使用51单片机GPIO口模拟串口通信,通过定时计数器0来产生中断信号 
  4. Software Designer:Jason 
  5. ************************************************************************************ 
  6. */  
  7. #include <reg52.h>  
  8. sbit P1_0 = P1^0;  
  9. sbit P1_1 = P1^1;  
  10. #define RXD P1_0  
  11. #define TXD P1_1  
  12. unsigned char flag;  
  13. void init();  
  14. void send_byte(unsigned char);  
  15. unsigned char rec_byte();  
  16. void wait_int();  
  17. //将从PC机串口接收到的数据原封不动回传给PC机  
  18. void main()  
  19. {  
  20.     unsigned char temp;  
  21.     init();  
  22.     while(1)  
  23.     {  
  24.         if(RI == 1)  
  25.         {  
  26.             RI = 0;  
  27.             temp = rec_byte();  
  28.             send_byte(temp);  
  29.             while(!TI);  
  30.             TI = 0;  
  31.         }  
  32.     }  
  33. }  
  34. //端口及中断初始化  
  35. void init()  
  36. {  
  37.     TMOD = 0x02;    //定时器0,方式2  
  38.     TH0 = 0xfd;     //波特率9600  
  39.     TL0 = TH0;  
  40.     TR0 = 0;        //在发送或接收时打开  
  41.     ET0 = 1;        //允许定时器0中断  
  42.     EA =1;          //允许所有中断  
  43. }  
  44. //通过串口发送一个字节数据  
  45. void send_byte(unsigned char dat)  
  46. {  
  47.     unsigned char i=8;  
  48.     TR0 = 1;        //开启T0中断  
  49.     TXD = 0;        //发送起始位0  
  50.     wait_int();  
  51.     while(i--)      //发送8位数据  
  52.     {  
  53.         TXD = (bit)(dat & 0x01);  
  54.         wait_int();  
  55.         dat = dat>>1;  
  56.     }  
  57.     TXD = 1;    //发送停止位1  
  58.     wait_int();  
  59.     TR0 = 0;        //关闭T0中断  
  60. }  
  61. //通过串口接收一个字节数据  
  62. unsigned char rec_byte()  
  63. {  
  64.     unsigned char dat=0;  
  65.     unsigned char i=8;  
  66.     TR0 = 1;        //开启T0中断  
  67.     wait_int();     //等过起始位电平  
  68.     while(i--)      //接收8位数据  
  69.     {  
  70.         dat = dat<<1;  
  71.         if(RXD)  
  72.             dat |= 0x80;  
  73.         wait_int();  
  74.     }  
  75.     wait_int();     //等过停止位电平  
  76.     TR0 = 0;        //关闭T0中断  
  77.     return dat;  
  78. }  
  79. //等待中断到来  
  80. void wait_int()  
  81. {  
  82.     while(!flag);  
  83.     flag = 0;  
  84. }  
  85. //中断服务程序  
  86. void timer0() interrupt 1  
  87. {  
  88.     flag = 1;  
  89. }  

  • 0
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值