利用外部中断实现清零_利用外部中断实现模拟串口数据的收发-模拟电子-

本文详细介绍了如何使用51单片机通过外部中断模拟串口进行数据收发,强调了在退出中断时清零中断标志位的重要性。同时,建议在实际项目中优先使用内置串口,若使用模拟串口,数据包应尽量短,并避免与动态数码管显示方案结合,以防不稳定。文中提供了硬件平台设置、功能实现、源代码讲解以及关键延时参数的调整方法。
摘要由CSDN通过智能技术生成

这一节注意四个知识点:

第一个:如何利用外部中断实现模拟串口数据的收发。

第二个:在退出外部中断函数时,必须通过软件把外部中断标志位IE0清零,否则在接收到的数据包最后面会多收到一个无效的字节0xFF。

第三个:实际做项目的时候,尽量利用单片机内部自带的集成串口,不到万不得已尽量不要用自制的模拟串口,如果非要用本节讲的模拟串口,那么一次接收的数据包不要太长,尽可能越短越好,因为自己做的模拟串口在稳定性上肯定比不上单片机自带的串口。这种模拟串口在批量生产时容易因为晶振的误差,以及外界各地温度的温差而影响产品的一致性,是有隐患的。

第四个:用模拟串口时,尽量不要选用动态数码管的显示方案,因为单片机在收发串口数据时,只能专心干一件事,此时不能中途被动态数码管扫描程序占用。而动态数码管得不到均匀扫描,就会产生略微闪烁的现象瑕疵。

具体内容,请看源代码讲解。

(1)硬件平台:

基于朱兆祺51单片机学习板。当把程序下载到单片机之后,要做以下跳线处理:

单片机原来的P3.1引脚是TI串口输出引脚,P3.0是RI串口输入引脚,分别把P3.1和P3.0的黄颜色跳冒去掉,同时也把外部中断0的引脚P3.2和一根IO口P1.0引脚的换颜色跳冒去掉,把P3.2跳冒的右针连接到P3.0跳冒的左针,作为模拟串口的接收数据线。把P1.0跳冒的右针连接到P3.1跳冒的左针,作为模拟串口的发送数据线。

(2)实现功能:

波特率是:9600 。

通过电脑串口调试助手模拟上位机,往单片机任意发送一串不超过10个的数据包,单片机如实地返回接收到的整包数据给上位机。

例如:

(a)上位机发送数据:01 02 03 04 05 06 07 08 09 0A

单片机返回: 01 02 03 04 05 06 07 08 09 0A

(b)上位机发送数据: 05 07 EE A8 F9

单片机返回: 05 07 EE A8 F9

(3)源代码讲解如下:

#include "REG52.H"

#define const_voice_short 40 //蜂鸣器短叫的持续时间

#define const_rc_size 20 //接收串口中断数据的缓冲区数组大小

#define const_receive_time 5 //如果超过这个时间没有串口数据过来,就认为一串数据已经全部接收完,这个时间根据实际情况来调整大小

/* 注释一:

* 以下时序脉冲延时参数我是在keil uVision2 平台下,Memory Model在small模式,Code Rom Size在Large模式下编译的,

* 如果在不同keil版本,不同的模式下,编译出来的程序有可能此参数会不一样。

* 以下的时序脉冲延时参数是需要一步一步慢慢调的。我一开始的时候先编写一个简单的发送数据测试程序,

* 先确调试出合适的发送时序延时数据。然后再编写串口接收数据的程序,从而调试出接收时序的延时参数。

* 比如:我第一步发送数据的测试程序是这样的:

void main()

{

initial_myself();

delay_long(100);

initial_peripheral();

while(1)

{

// usart_service(); //串口服务程序

eusart_send(0x08); //测试程序,让它不断发送数据给上位机观察,确保发送延时时序的参数准确性

delay_long(300);

eusart_send(0xE5); //测试程序,让它不断发送数据给上位机观察,确保发送延时时序的参数准确性

delay_long(300);

}

}

*/

#define const_t_1 10 //发送时序延时1 第一步先调出此数据

#define const_t_2 9 //发送时序延时2 第一步先调出此数据

#define const_r_1 7 //接收时序延时1 第二步再调出此数据

#define const_r_2 9 //接收时序延时2 第二步再调出此数据

void initial_myself(void);

void initial_peripheral(void);

void delay_long(unsigned int uiDelaylong);

void delay_short(unsigned int uiDelayShort);

void delay_minimum(unsigned char ucDelayMinimum); //细分度最小的延时,用char类型一个字节

void T0_time(void); //定时中断函数

void INT0_int(void); //外部0中断函数,在本系统中是模拟串口的接收中断函数。

void usart_service(void); //串口服务程序,在main函数里

void eusart_send(unsigned char ucSendData);

unsigned char read_eusart_byte();//从串口读一个字节

sbit beep_dr=P2^7; //蜂鸣器的驱动IO口

sbit ti_dr=P1^0; //模拟串口发送数据的IO口

sbit ri_sr=P3^2; //模拟串口接收数据的IO口 也是外部中断0的复用IO口

unsigned int uiSendCnt=0; //用来识别串口是否接收完一串数据的计时器

unsigned char ucSendLock=1; //串口服务程序的自锁变量,每次接收完一串数据只处理一次

unsigned int uiRcregTotal=0; //代表当前缓冲区已经接收了多少个数据

unsigned char ucRcregBuf[const_rc_size]; //接收串口中断数据的缓冲区数组

unsigned char ucTest=0;

void main()

{

initial_myself();

delay_long(100);

initial_peripheral();

while(1)

{

usart_service(); //串口服务程序

}

}

void usart_service(void) //串口服务程序,在main函数里

{

unsigned char i=0;

if(uiSendCnt>=const_receive_time&&ucSendLock==1) //说明超过了一定的时间内,再也没有新数据从串口来

{

ucSendLock=0; //处理一次就锁起来,不用每次都进来,除非有新接收的数据

//下面的代码进入数据协议解析和数据处理的阶段

for(i=0;i

{

eusart_send(ucRcregBuf[i]);

}

uiRcregTotal=0; //清空缓冲的下标,方便下次重新从0下标开始接受新数据

}

}

//往串口发送一个字节

void eusart_send(unsigned char ucSendData) //往上位机发送一个字节的函数

{

unsigned char i=8;

EA=0; //关总中断

ti_dr=0; //发送启始位

delay_minimum(const_t_1); //发送时序延时1 delay_minimum是本程序细分度最小的延时

while(i--)

{

ti_dr=ucSendData&0x01; //先传低位

delay_minimum(const_t_2); //发送时序延时2 delay_minimum是本程序细分度最小的延时

ucSendData=ucSendData>>1;

}

ti_dr=1; //发送结束位

delay_short(400); //每个字节之间的延时,这里非常关键,也是最容易出错的地方。延时的大小请根据实际项目来调整

EA=1; //开总中断

}

//从串口读取一个字节

unsigned char read_eusart_byte()

{

unsigned char ucReadData=0;

unsigned char i=8;

delay_minimum(const_r_1); //接收时序延时1 。作用是等过起始位 delay_minimum是本程序细分度最小的延时

while(i--)

{

ucReadData >>=1;

if(ri_sr==1)

{

ucReadData|=0x80; //先收低位

}

if(ri_sr==0) //此处空指令,是为了让驱动时序的时间保持一致性

{

;

}

delay_minimum(const_r_2); //接收时序延时2 delay_minimum是本程序细分度最小的延时

}

return ucReadData;

}

void T0_time(void) interrupt 1 //定时中断

{

TF0=0; //清除中断标志

TR0=0; //关中断

if(uiSendCnt

{

uiSendCnt++; //表面上这个数据不断累加,但是在串口中断里,每接收一个字节它都会被清零,除非这个中间没有串口数据过来

ucSendLock=1; //开自锁标志

}

TH0=0xfe; //重装初始值(65535-500)=65035=0xfe0b

TL0=0x0b;

TR0=1; //开中断

}

void INT0_int(void) interrupt 0 //INT0外部中断函数

{

EX0=0; //禁止外部0中断 这个只是我个人的编程习惯,也可以不关闭

++uiRcregTotal;

if(uiRcregTotal>const_rc_size) //超过缓冲区

{

uiRcregTotal=const_rc_size;

}

ucRcregBuf[uiRcregTotal-1]=read_eusart_byte(); //将串口接收到的数据缓存到接收缓冲区里

uiSendCnt=0; //及时喂狗,虽然main函数那边不断在累加,但是只要串口的数据还没发送完毕,那么它永远也长不大,因为每个中断都被清零。

/* 注释二:

* 注意,此处必须把IE0中断标志清零,否则在接收到的数据包最后面会多收到一个无效的字节0xFF。

*/

IE0=0; //外部中断0标志位清零,必须的!

EX0=1; //打开外部0中断

}

void delay_long(unsigned int uiDelayLong)

{

unsigned int i;

unsigned int j;

for(i=0;i

{

for(j=0;j<500;j++) //内嵌循环的空指令数量

{

; //一个分号相当于执行一条空语句

}

}

}

void delay_short(unsigned int uiDelayShort)

{

unsigned int i;

for(i=0;i

{

; //一个分号相当于执行一条空语句

}

}

/* 注释三:

* 由于IO口模拟的串口时序要求很高,所以用的延时函数尽可能细分度越高越好,以下用一个字节的延时计时器

*/

void delay_minimum(unsigned char ucDelayMinimum) //细分度最小的延时,用char类型一个字节

{

unsigned char i;

for(i=0;i

{

; //一个分号相当于执行一条空语句

}

}

void initial_myself(void) //第一区 初始化单片机

{

beep_dr=1; //用PNP三极管控制蜂鸣器,输出高电平时不叫。

//配置定时器

TMOD=0x01; //设置定时器0为工作方式1

TH0=0xfe; //重装初始值(65535-500)=65035=0xfe0b

TL0=0x0b;

}

void initial_peripheral(void) //第二区 初始化外围

{

EX0=1; //允许外部中断0

IT0=1; //下降沿触发外部中断0 如果是0代表低电平中断

IP=0x01; //设置外部中断0为最高优先级,可以打断低优先级中断服务。实现中断嵌套功能

EA=1; //开总中断

ET0=1; //允许定时中断

TR0=1; //启动定时中断

}

查看评论

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,这是一个更具挑战性的任务,需要使用软件延时来代替定时器和中断的功能。 1. 硬件连接 将TCS3200模块连接到51单片机上,具体连接方式可以参考TCS3200模块的说明书。其中,S0和S1分别接P1.0和P1.1,OUT接P1.2,OE接P1.3,VCC接VCC,GND接GND。 2. 软件设计 首先,我们需要定义一些常量和变量: ```c #define FREQ_2_PERCENT(x) (100 - (x) / 10.24) // 将频率转换为百分比 sbit S0 = P1 ^ 0; sbit S1 = P1 ^ 1; sbit OUT = P1 ^ 2; sbit OE = P1 ^ 3; // TCS3200模块的控制引脚 unsigned char red = 0, green = 0, blue = 0; // 存储RGB三个颜色的值 unsigned char uart_buf[4] = {0}; // 存储通过串口发送的数据 ``` 然后,我们需要编写一些函数来控制TCS3200模块: ```c void TCS3200_Setup(unsigned char mode) { switch (mode) { case 0: // 关闭输出 OE = 1; break; case 1: // 输出频率2% S0 = 0; S1 = 0; break; case 2: // 输出频率20% S0 = 1; S1 = 0; break; case 3: // 输出频率100% S0 = 0; S1 = 1; break; case 4: // 输出脉冲计数 S0 = 1; S1 = 1; break; } } unsigned int TCS3200_Read(unsigned char mode) { unsigned int val = 0; switch (mode) { case 1: // 读取红色分量 TCS3200_Setup(1); break; case 2: // 读取绿色分量 TCS3200_Setup(2); break; case 3: // 读取蓝色分量 TCS3200_Setup(3); break; case 4: // 读取总频率 TCS3200_Setup(4); break; } OE = 0; for (int i = 0; i < 100; i++) // 延时等待输出稳定 { _nop_(); } val = 0; for (int i = 0; i < 32; i++) // 读取32个脉冲 { val <<= 1; OUT = 1; for (int j = 0; j < 10; j++) // 每个脉冲的持续时间为10us { _nop_(); } val |= OUT; OUT = 0; for (int j = 0; j < 10; j++) // 两个脉冲之间的间隔时间为10us { _nop_(); } } OE = 1; return val; } ``` 最后,我们需要编写主函数: ```c void main() { while (1) { red = FREQ_2_PERCENT(TCS3200_Read(1)); green = FREQ_2_PERCENT(TCS3200_Read(2)); blue = FREQ_2_PERCENT(TCS3200_Read(3)); uart_buf[0] = red; uart_buf[1] = green; uart_buf[2] = blue; uart_buf[3] = '\n'; // 将RGB三个颜色的值存储到uart_buf中,并在最后加上换行符 for (int i = 0; i < 4; i++) { SBUF = uart_buf[i]; while (!TI) ; TI = 0; } // 通过串口发送数据 } } ``` 在主函数中,我们首先读取TCS3200模块输出的红、绿、蓝三个颜色的频率,并将其转换为百分比。然后,将这些值存储到uart_buf中,并通过串口发送出去。 注意,我们这里没有使用定时器和中断,而是通过软件延时来控制TCS3200模块的输出。在TCS3200_Read函数中,我们首先等待TCS3200模块输出稳定,然后通过循环读取32个脉冲的状态来计算出TCS3200模块输出的频率。 希望这个代码能够对你有所帮助!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值