⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 io to uart.lst

📁 基于51的简单交通灯模拟,对那些刚刚入门的朋友可能有所帮助
💻 LST
📖 第 1 页 / 共 2 页
字号:
C51 COMPILER V7.06   IO_TO_UART                                                            08/22/2008 23:17:02 PAGE 1   


C51 COMPILER V7.06, COMPILATION OF MODULE IO_TO_UART
OBJECT MODULE PLACED IN .\IO to UART.obj
COMPILER INVOKED BY: f:\Keil\C51\BIN\C51.EXE ..\..\IO口转串口程序\IO to UART.c BROWSE DEBUG OBJECTEXTEND PRINT(.\IO to U
                    -ART.lst) OBJECT(.\IO to UART.obj)

stmt level    source

   1          //普通IO模拟串口收发程序 
   2          //程序匠人 发表于 2008-2-13 5:48:00  阅读全文(422) | 回复(0) | 引用通告(0) | 编辑 
   3           
   4          /*************************************************************************
   5          用定时器 T0 或 T1 模拟串行口程序。
   6          最高波特率(12 clock):
   7              本程序收、发波特率相同。 
   8          
   9              11.059MHz -- 最高波特率 收: 9600,  最低波特率:300   
  10              30.000MHz -- 最高波特率 收: 28800  最低波特率:300   
  11              40.000MHz -- 最高波特率 收: 38400  最低波特率:300   
  12                 ...
  13          使用说明:
  14              1. 本程序使用一个定时器和任意 2 个 I/O 口模拟一个串行口。
  15              2. 1位起始位,8位数据位,1位停止位。发数据位时先发低位。
  16              3. 支持半双工通讯。收、发波特率相同。
  17              4. 应把定时器中断优先级设置为最高级。
  18              5. 本程序每接收一个字节后就把它放到一个队列缓冲区中(也可使用环行缓冲区),
  19                 待缓冲区满后,将缓冲区中的内容原样发回。这是为了测试多字节连续收发的
  20                 能力和简化程序。实际应用中应防止缓冲区溢出。  
  21              6. 由接收转换到发送时要先调用  soft_send_enable ();
  22                 由发送转换到接收时要先调用  soft_receive_enable ()。
  23              7. 发送最后一个字节后如果要立刻转为接收,必须等待最后一个字节后发送完毕
  24                     while ( rs_f_TI == 0)  ;  // 等待最后一个字节发送完毕
  25          **************************************************************************
  26          编程说明:
  27          ----------------
  28          发送:
  29              由接收转换到发送时要先调用 soft_send_enable (), 它为发送做初始化的工作。
  30          以后就可以调用 rs_send_byte () 启动发送一个字节的过程。
  31              发送口平时为高电平,rs_send_byte ()函数使发送口变为低电平开始发送起始位; 
  32          同时设置和启动定时器,为发送数据位在预定的时刻产生定时器中断。发送数据位和
  33          停止位都在定时器的中断服务程序中进行。
  34              中断服务程序中处理 4 种情况:发送数据位、发送停止位、发送完毕、处理错误。
  35          ----------------
  36          接收:
  37              由发送转换到接收时要先调用 soft_receive_enable (), 它为接收做初始化的工
  38          作。定时器以 3 到 4 倍波特率的频率产生中断(参见 rs_TEST0 的定义)检测 PC
  39          机发送的起始位。一旦检测到起始位,立刻把定时器产生中断的频率调整到与波特率
  40          相同,准备在下一个定时器中断中接收第 1 个数据位。
  41              中断服务程序中处理以下情况:
  42              1. 收到的是 PC 机发送的起始位: 调整定时器产生中断的频率与波特率相同。
  43              2. 收到第 8 位数据位: 存储接收到的字节。
  44              3. 收到第 1--7 位数据位: 存储到收、发移位暂存器。
  45              4. 收到停止位: 调用 soft_receive_enable(),检测 PC 机发出的下一个起始位。 
  46              5. 处理出错的情况。
  47          **************************************************************************/
  48          #include<reg52.h>
  49          sfr16 DPTR = 0x82;
  50          
  51          typedef unsigned char INT8U;
  52          typedef unsigned int INT16U;
  53          
  54          #define YES   1
C51 COMPILER V7.06   IO_TO_UART                                                            08/22/2008 23:17:02 PAGE 2   

  55          #define NO    0
  56          
  57          //定义使用哪个定时器, 只可定义一个   
  58          //#define TIMER_0
  59          #define TIMER_1
  60          
  61          //定义串口收、发送管脚。  
  62          sbit rs_TXD = P2^1;
  63          sbit rs_RXD = P2^0;
  64          
  65          //根据定时器确定参数  
  66          #ifdef TIMER_0
                  #define TMOD_AND_WORD   0xF0;
                  #define TMOD_TIME_MODE  0x01;
                  #define TMOD_COUNT_MODE 0x05;      //设置计数模式位  
                  sbit  TCON_ENABLE_TIMER = TCON^4;
                  sbit  TCON_TFx = TCON^5;           //中断标志位  
                  sbit  IE_ETx = IE^1;               //中断允许位为 ET0  
                  sbit  IP_PTx = IP^1;               //中断优先级  
              
                  sfr rs_timerL = 0x8A;              //TL0 
                  sfr rs_timerH = 0x8C;              //TH0 
              #endif
  78          
  79          #ifdef TIMER_1
  80              #define TMOD_AND_WORD   0x0F;
  81              #define TMOD_TIME_MODE  0x10;
  82              #define TMOD_COUNT_MODE 0x50;      //设置计数模式位  
  83              sbit  TCON_ENABLE_TIMER = TCON^6;  //
  84              sbit  TCON_TFx = TCON^7;           //中断标志位  
  85              sbit  IE_ETx = IE^3;               //中断允许位为 ET1   
  86              sbit  IP_PTx = IP^4;               //中断优先级  
  87          
  88              sfr rs_timerL = 0x8B;              //TL1
  89              sfr rs_timerH = 0x8D;              //TH1  
  90          #endif
  91          
  92          INT8U   bdata rs_BUF;                  //串行收、发时用的移位暂存器。   
  93          sbit rs_BUF_bit7 = rs_BUF^7;        //移位暂存器的最高位。 
  94          INT8U   rs_shift_count;                //移位计数器。  
  95          
  96          INT8U bdata rsFlags;
  97          sbit rs_f_TI        = rsFlags^0;    //0:正在发送; 1: 一个字符完毕  
  98          sbit rs_f_RI_enable = rsFlags^1;   //0:禁止接收; 1:允许接收  
  99          sbit rs_f_TI_enable = rsFlags^2;   //0:禁止发送; 1:允许发送  
 100          
 101          //选择以下一个晶体频率
 102          //#define Fosc 6000000                 //6MHz 
 103          #define Fosc 11059200                  //11.059MHz 
 104          //#define Fosc 12000000
 105          //#define Fosc 18432000
 106          //#define Fosc 20000000
 107          //#define Fosc 24000000
 108          //#define Fosc 30000000
 109          //#define Fosc 40000000
 110          
 111          //选择以下一个波特率:
 112          //#efine Baud 300                      //11.059MHz时,baud 最低为 300   
 113          //#define Baud 1200
 114          //#define Baud 2400
 115          //#define Baud 4800
 116          #define Baud 9600
C51 COMPILER V7.06   IO_TO_UART                                                            08/22/2008 23:17:02 PAGE 3   

 117          //#define Baud 14400
 118          //#define Baud 19200
 119          //#define Baud 28800
 120          //#define Baud 38400
 121          //#define Baud 57600
 122          
 123          //收、发一位所需定时器计数   
 124          #define rs_FULL_BIT0 ((Fosc/12) / Baud)
 125          #define rs_FULL_BIT (65536 - rs_FULL_BIT0)
 126          #define rs_FULL_BIT_H rs_FULL_BIT >> 8        //收、发一位所需定时器计数高位   
 127          #define rs_FULL_BIT_L (rs_FULL_BIT & 0x00FF)  //收、发一位所需定时器计数低位   
 128          
 129          //检测起始位的时间间隔所需定时器计数    
 130          #define rs_TEST0 rs_FULL_BIT0 / 4             //波特率较低时可以除以 3 或除以 2    
 131          #define rs_TEST ((~rs_TEST0))
 132          #define rs_TEST_H rs_TEST >> 8                //高位  
 133          #define rs_TEST_L rs_TEST & 0x00FF            //低位  
 134          
 135          //发送起始位所需定时器总计数   
 136          #define rs_START_BIT 0xFFFF - (Fosc/12/Baud) + 0x28
 137          #define rs_START_BIT_H rs_START_BIT >> 8      //发送起始位所需定时器计数高位    
 138          #define rs_START_BIT_L rs_START_BIT & 0x00FF  //发送起始位所需定时器计数低位   
 139          
 140          #define rs_RECEIVE_MAX   128                  //最大接收长度  
 141          INT8U idata rs232buffer[rs_RECEIVE_MAX];      //收、发缓冲区
 142          INT16U ReceivePoint;                       //接收数据存储指针  
 143          
 144          void soft_rs232_interrupt( void );
 145          
 146          #ifdef TIMER_0
                  void timer0 (void) interrupt 1 using 3
                  {
                      if (rs_RXD == 0 | rs_shift_count > 0)
                      { soft_rs232_interrupt(); }
                      else
                      {
                          rs_timerH = rs_TEST_H;
                          rs_timerL = rs_TEST_L;
                      }
                  }
              #endif
 158          
 159          #ifdef TIMER_1
 160              void timer1 (void) interrupt 3 using 3
 161              {
 162   1              if (rs_RXD == 0 | rs_shift_count > 0)
 163   1              { soft_rs232_interrupt(); }
 164   1              else
 165   1              {
 166   2                  rs_timerH = rs_TEST_H;
 167   2                  rs_timerL = rs_TEST_L;
 168   2              }
 169   1          }
 170          #endif
 171          /***************************************/
 172          
 173          void soft_rs232_init (void)            //串口初始化  
 174          {
 175   1          TCON_ENABLE_TIMER = 0;             //停止定时器  
 176   1          TMOD &= TMOD_AND_WORD;
 177   1          TMOD |= TMOD_TIME_MODE;
 178   1          rs_RXD = 1;                        //接收脚置成高电平  
C51 COMPILER V7.06   IO_TO_UART                                                            08/22/2008 23:17:02 PAGE 4   

 179   1          rs_TXD = 1;                        //发射脚置成高电平  
 180   1          IP_PTx = 1;                        //置中断优先级为高  
 181   1          IE_ETx = 1;                        //允许定时器中断    

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -