用定时器 T0 或 T1 模拟串行口程序

相关主题
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

/*************************************************************************

用定时器T0 或T1 模拟串行口程序。

最高波特率(12 clock):

本程序收、发波特率相同。

11.059MHz -- 最高波特率收: 9600, 最低波特率:300

30.000MHz -- 最高波特率收: 28800 最低波特率:300

40.000MHz -- 最高波特率收: 38400 最低波特率:300

...

使用说明:

1. 本程序使用一个定时器和任意2 个I/O 口模拟一个串行口。

2. 1位起始位,8位数据位,1位停止位。发数据位时先发低位。

3. 支持半双工通讯。收、发波特率相同。

4. 应把定时器中断优先级设置为最高级。

5. 本程序每接收一个字节后就把它放到一个队列缓冲区中(也可使用环行缓冲区),

待缓冲区满后,将缓冲区中的内容原样发回。这是为了测试多字节连续收发的

能力和简化程序。实际应用中应防止缓冲区溢出。

6. 由接收转换到发送时要先调用soft_send_enable ();

由发送转换到接收时要先调用 soft_receive_enable ()。

7. 发送最后一个字节后如果要立刻转为接收,必须等待最后一个字节后发送完毕 w hile ( rs_f_TI == 0) ; // 等待最后一个字节发送完毕

**************************************************************************

编程说明:

----------------

发送:

由接收转换到发送时要先调用soft_send_enable (), 它为发送做初始化的工作。

以后就可以调用rs_send_byte () 启动发送一个字节的过程。

发送口平时为高电平,rs_send_byte ()函数使发送口变为低电平开始发送起始位; 同时设置和启动定时器,为发送数据位在预定的时刻产生定时器中断。发送数据位和停止位都在定时器的中断服务程序中进行。

中断服务程序中处理4 种情况:发送数据位、发送停止位、发送完毕、处理错误。----------------

接收:

由发送转换到接收时要先调用soft_receive_enable (), 它为接收做初始化的工

作。定时器以3 到4 倍波特率的频率产生中断(参见rs_TE ST0 的定义)检测P C 机发送的起始位。一旦检测到起始位,立刻把定时器产生中断的频率调整到与波特率相同,准备在下一个定时器中断中接收第1 个数据位。

中断服务程序中处理以下情况:

1. 收到的是P C 机发送的起始位: 调整定时器产生中断的频率与波特率相同。

2. 收到第8 位数据位: 存储接收到的字节。

3. 收到第1--7 位数据位: 存储到收、发移位暂存器。

4. 收到停止位: 调用soft_recei v e_enable(),检测PC 机发出的下一个起始位。

5. 处理出错的情况。

**************************************************************************/

#i nclude

sfr16 DP TR = 0x82;

typedef unsigned char INT8U;

typedef unsigned int INT16U;

#define YE S 1

#define NO 0

//定义使用哪个定时器, 只可定义一个

//#define TIMER_0

#define TIMER_1

//定义串口收、发送管脚。

sbit rs_TXD = P2^1;

sbit rs_RXD = P2^0;

//根据定时器确定参数

#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; //中断允许位为E T0

sbit IP_PTx = IP^1; //中断优先级

sfr rs_timerL = 0x8A; //TL0

sfr rs_timerH = 0x8C; //TH0

#endif

#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; //中断允许位为E T1

sbit IP_PTx = IP^4; //中断优先级

sfr rs_timerL = 0x8B; //TL1

sfr rs_timerH = 0x8D; //TH1

#endif

INT8U bdata rs_BUF; //串行收、发时用的移位暂存器。

sbit rs_BUF_bit7 = rs_BUF^7; //移位暂存器的最高位。

INT8U rs_shift_count; //移位计数器。

INT8U bdata rsFlags;

sbit rs_f_TI = rsFlags^0; //0:正在发送; 1: 一个字符完毕

sbit rs_f_RI_enable = rsFlags^1; //0:禁止接收; 1:允许接收

sbit rs_f_TI_enable = rsFlags^2; //0:禁止发送; 1:允许发送

//选择以下一个晶体频率

//#define Fosc 6000000 //6MHz

#define Fosc 11059200 //11.059MHz

//#define Fosc 12000000

//#define Fosc 18432000

//#define Fosc 20000000

//#define Fosc 24000000

//#define Fosc 30000000

//#define Fosc 40000000

//选择以下一个波特率:

//#efine Baud 300 //11.059MHz时,baud 最低为300

//#define Baud 1200

//#define Baud 2400

//#define Baud 4800

#define Baud 9600

//#define Baud 14400

//#define Baud 19200

//#define Baud 28800

//#define Baud 38400

//#define Baud 57600

//收、发一位所需定时器计数

#define rs_FULL_BIT0 ((Fosc/12) / Baud)

#define rs_FULL_BIT (65536 - rs_FULL_BIT0)

#define rs_FULL_BIT_H rs_FULL_BIT >> 8 //收、发一位所需定时器计数高位#define rs_FULL_BIT_L (rs_FULL_BIT & 0x00FF) //收、发一位所需定时器计数低位

//检测起始位的时间间隔所需定时器计数

#define rs_TE ST0 rs_FULL_BIT0 / 4 //波特率较低时可以除以3 或除以2

相关文档
最新文档