📄 io_soft_uart_use_timer.lst
字号:
C51 COMPILER V8.05a IO_SOFT_UART_USE_TIMER 10/20/2008 11:07:22 PAGE 1
C51 COMPILER V8.05a, COMPILATION OF MODULE IO_SOFT_UART_USE_TIMER
OBJECT MODULE PLACED IN IO_SOFT_UART_USE_TIMER.OBJ
COMPILER INVOKED BY: C:\Keil\C51\BIN\C51.EXE IO_SOFT_UART_USE_TIMER.c ROM(COMPACT) DEBUG OBJECTEXTEND
line level source
1 /* ----------------------- 版权声明 ----------------------------------
2 ------------------------------ 宏晶科技 2005/01/17 --------------------------
3 -------------- Tel: 0755-82948409 Fax:0755-82944243 -------------------
4 -------------- Mobile:13922805190 (姚永平) ----------------------------------
5 ------ Website:www.mcu-memory.com Email: support@dsp-memory.com --------
6 未经深圳市宏晶科技书面同意, 不得将本程序泄露、公开给第三方。
7 不得将本程序(或修改后的程序)使用在非深圳市宏晶科技销售的产品上。
8 客户产品上使用本程序时,客户产品的源程序中必须注明使用了深圳市宏晶科技的程序,
9 并保留如下内容:
10 ********************************************************************************
11 ------------------------------ 宏晶科技 2005/01/09 --------------------------
12 -------------- Tel: 0755-82948409 Fax:0755-82944243 -------------------
13 -------------- Mobile:13922805190 (姚永平) ----------------------------------
14 ------ Website:www.mcu-memory.com Email: support@dsp-memory.com --------
15 ********************************************************************************
16 --------------------本程序客户须认真消化,不提供技术支持---------------------- */
17
18 /*************************************************************************
19 用定时器 T0 或 T1 模拟串行口程序。
20 最高波特率(12 clock):
21 本程序收、发波特率相同。
22
23 11.059MHz -- 最高波特率 收: 9600, 最低波特率:300
24 30.000MHz -- 最高波特率 收: 28800 最低波特率:300
25 40.000MHz -- 最高波特率 收: 38400 最低波特率:300
26 ...
27 使用说明:
28 1. 本程序使用一个定时器和任意 2 个 I/O 口模拟一个串行口。
29 2. 1位起始位,8位数据位,1位停止位。发数据位时先发低位。
30 3. 支持半双工通讯。收、发波特率相同。
31 4. 应把定时器中断优先级设置为最高级。
32 5. 本程序每接收一个字节后就把它放到一个队列缓冲区中(也可使用环行缓冲区),
33 待缓冲区满后,将缓冲区中的内容原样发回。这是为了测试多字节连续收发的
34 能力和简化程序。实际应用中应防止缓冲区溢出。
35 6. 由接收转换到发送时要先调用 soft_send_enable ();
36 由发送转换到接收时要先调用 soft_receive_enable ()。
37 7. 发送最后一个字节后如果要立刻转为接收,必须等待最后一个字节后发送完毕
38 while ( rs_f_TI == 0) ; // 等待最后一个字节发送完毕
39 **************************************************************************
40 编程说明:
41 ----------------
42 发送:
43 由接收转换到发送时要先调用 soft_send_enable (), 它为发送做初始化的工作。
44 以后就可以调用 rs_send_byte () 启动发送一个字节的过程。
45 发送口平时为高电平,rs_send_byte ()函数使发送口变为低电平开始发送起始位;
46 同时设置和启动定时器,为发送数据位在预定的时刻产生定时器中断。发送数据位和
47 停止位都在定时器的中断服务程序中进行。
48 中断服务程序中处理 4 种情况:发送数据位、发送停止位、发送完毕、处理错误。
49 ----------------
50 接收:
51 由发送转换到接收时要先调用 soft_receive_enable (), 它为接收做初始化的工
52 作。定时器以 3 到 4 倍波特率的频率产生中断(参见 rs_TEST0 的定义)检测 PC
53 机发送的起始位。一旦检测到起始位,立刻把定时器产生中断的频率调整到与波特率
54 相同,准备在下一个定时器中断中接收第 1 个数据位。
55 中断服务程序中处理以下情况:
C51 COMPILER V8.05a IO_SOFT_UART_USE_TIMER 10/20/2008 11:07:22 PAGE 2
56 1. 收到的是 PC 机发送的起始位: 调整定时器产生中断的频率与波特率相同。
57 2. 收到第 8 位数据位: 存储接收到的字节。
58 3. 收到第 1--7 位数据位: 存储到收、发移位暂存器。
59 4. 收到停止位: 调用 soft_receive_enable(),检测 PC 机发出的下一个起始位。
60 5. 处理出错的情况。
61 **************************************************************************/
62
63 //sfr16 DPTR = 0x82;
64
65 #include "STC12C5410AD.h"
66 #include "app.h"
67 #include <string.h>
68 #include <stdio.h>
69
70 typedef unsigned char INT8U;
71 typedef unsigned int INT16U;
72
73 extern void UartPrintOut(unsigned char *pData,unsigned char size) ;
74
75 #define YES 1
76 #define NO 0
77
78 //定义使用哪个定时器, 只可定义一个
79 #define TIMER_0
80 // #define TIMER_1
81
82 //定义串口收、发送管脚。
83 sbit rs_TXD = P3^3;
84 sbit rs_RXD = P3^2;
85
86 //根据定时器确定参数
87 #ifdef TIMER_0
88 #define TMOD_AND_WORD 0xF0;
89 #define TMOD_TIME_MODE 0x01;
90 #define TMOD_COUNT_MODE 0x05; //设置计数模式位
91 sbit TCON_ENABLE_TIMER = TCON^4;
92 sbit TCON_TFx = TCON^5; //中断标志位
93 sbit IE_ETx = IE^1; //中断允许位为 ET0
94 sbit IP_PTx = IP^1; //中断优先级
95
96 sfr rs_timerL = 0x8A; //TL0
97 sfr rs_timerH = 0x8C; //TH0
98 #endif
99
100 #ifdef TIMER_1
#define TMOD_AND_WORD 0x0F;
#define TMOD_TIME_MODE 0x10;
#define TMOD_COUNT_MODE 0x50; //设置计数模式位
sbit TCON_ENABLE_TIMER = TCON^6; //
sbit TCON_TFx = TCON^7; //中断标志位
sbit IE_ETx = IE^3; //中断允许位为 ET1
sbit IP_PTx = IP^4; //中断优先级
sfr rs_timerL = 0x8B; //TL1
sfr rs_timerH = 0x8D; //TH1
#endif
112
113 INT8U bdata rs_BUF; //串行收、发时用的移位暂存器。
114 sbit rs_BUF_bit7 = rs_BUF^7; //移位暂存器的最高位。
115 INT8U rs_shift_count; //移位计数器。
116
117 INT8U bdata rsFlags;
C51 COMPILER V8.05a IO_SOFT_UART_USE_TIMER 10/20/2008 11:07:22 PAGE 3
118 sbit rs_f_TI = rsFlags^0; //0:正在发送; 1: 一个字符完毕
119 sbit rs_f_RI_enable = rsFlags^1; //0:禁止接收; 1:允许接收
120 sbit rs_f_TI_enable = rsFlags^2; //0:禁止发送; 1:允许发送
121
122 //选择以下一个晶体频率
123 //#define Fosc 6000000 //6MHz
124 #define Fosc 11059200 //11.059MHz
125 //#define Fosc 22118400 //22.1184MHz
126 //#define Fosc 12000000
127 //#define Fosc 18432000
128 //#define Fosc 20000000
129 //#define Fosc 24000000
130 //#define Fosc 30000000
131 //#define Fosc 40000000
132
133 //选择以下一个波特率:
134 //#efine Baud 300 //11.059MHz时,baud 最低为 300
135 #define Baud 1200
136 //#define Baud 2400
137 //#define Baud 4800
138 //#define Baud 9600
139 //#define Baud 14400
140 //#define Baud 19200
141 //#define Baud 28800
142 //#define Baud 38400
143 //#define Baud 57600
144
145 //收、发一位所需定时器计数
146 #define rs_FULL_BIT0 ((Fosc/12) / Baud)
147 #define rs_FULL_BIT (65536 - rs_FULL_BIT0)
148 #define rs_FULL_BIT_H rs_FULL_BIT >> 8 //收、发一位所需定时器计数高位
149 #define rs_FULL_BIT_L (rs_FULL_BIT & 0x00FF) //收、发一位所需定时器计数低位
150
151 //检测起始位的时间间隔所需定时器计数
152 #define rs_TEST0 rs_FULL_BIT0 / 4 //波特率较低时可以除以 3 或除以 2
153 #define rs_TEST ((~rs_TEST0))
154 #define rs_TEST_H rs_TEST >> 8 //高位
155 #define rs_TEST_L rs_TEST & 0x00FF //低位
156
157 //发送起始位所需定时器总计数
158 #define rs_START_BIT 0xFFFF - (Fosc/12/Baud) + 0x28
159 #define rs_START_BIT_H rs_START_BIT >> 8 //发送起始位所需定时器计数高位
160 #define rs_START_BIT_L rs_START_BIT & 0x00FF //发送起始位所需定时器计数低位
161
162 #define rs_RECEIVE_MAX 64 //最大接收长度
163 INT8U idata rs232buffer[rs_RECEIVE_MAX]; //收、发缓冲区
164 INT8U ReceivePoint=0; //接收数据存储指针
165
166 void soft_rs232_interrupt( void );
167
168 #ifdef TIMER_0
169 void timer0 (void) interrupt 1 using 3
170 {
171 1 if (rs_RXD == 0 | rs_shift_count > 0)
172 1 { soft_rs232_interrupt(); }
173 1 else
174 1 {
175 2 rs_timerH = rs_TEST_H;
176 2 rs_timerL = rs_TEST_L;
177 2 }
178 1 }
179 #endif
C51 COMPILER V8.05a IO_SOFT_UART_USE_TIMER 10/20/2008 11:07:22 PAGE 4
180
181 #ifdef TIMER_1
void timer1 (void) interrupt 3 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
193 /***************************************/
194
195
196
197 extern bit bdata phone_flag ; // 电话接口接收到电话号码完成标志;
198
199 void soft_rs232_init (void) //串口初始化
200 {
201 1 TCON_ENABLE_TIMER = 0; //停止定时器
202 1 TMOD &= TMOD_AND_WORD;
203 1 TMOD |= TMOD_TIME_MODE;
204 1 rs_RXD = 1; //接收脚置成高电平
205 1 rs_TXD = 1; //发射脚置成高电平
206 1 IP_PTx = 1; //置中断优先级为高
207 1 IE_ETx = 1; //允许定时器中断
208 1 }
209
210 void soft_receive_init() //监测起始位
211 {
212 1 TCON_ENABLE_TIMER = 0; //停止定时器
213 1 rs_timerH = rs_TEST_H;
214 1 rs_timerL = rs_TEST_L;
215 1 rs_shift_count = 0;
216 1 TCON_ENABLE_TIMER = 1; //启动定时器
217 1 }
218
219
220 void soft_receive_enable() //允许接收
221 {
222 1 rs_f_RI_enable = 1; //允许接收
223 1 rs_f_TI_enable = 0; //禁止发送
224 1 soft_receive_init(); //监测起始位, RXD 下降沿触发接收字节过程.
225 1 }
226
227 void soft_send_enable (void) //允许发送
228 {
229 1 TCON_ENABLE_TIMER = 0; //停止定时器
230 1 rs_f_TI_enable = 1; //允许发送
231 1 rs_f_RI_enable = 0; //禁止接收
232 1
233 1 rs_shift_count = 0; //清移位计数器
234 1 rs_f_TI = 1; //发送一个字符完毕标志
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -