本帖最后由 芯圣电子官方QQ 于 2023-7-25 09:14 编辑
非常高兴在第八波的抽奖中成为幸运者。等待几天后收到了芯圣寄出的开发板。
由于以前买过芯圣的HC89S105开发板,所以已经安装过了相关驱动。拿到HC89F0541开发板后什么也不需要做了,直接开始用Keil编程测试。本次测试主要使用了定时器、外部中断及串口模块。程序直接使用它例程中的外部中断程序和原C51开发板中的部分程序,根据本次测试的具体情况做了修改移植。
将以前买的C51学习板拿出来,连上红外接收头的相关线路。
用USB转串口连上PC
没有使用LCD1602来显示,主要怕线太多了。直接使用串口传输数据到串口助手就可以了。
一切就绪后,用遥控板向红外接收头发送数据,这是串口助手收到的数据,一切正常。
很高兴正好赶上这次活动,**芯圣越做越好。很喜欢它们的PTM功能,使用起来非常方便。
源码如下:main.c (移植于0541开发板中的外部中断程序,对串口波特率进行了相应修改)
/**
* ************************************************************************************
* 上海芯圣电子股份有限公司
* www.holychip.cn
* ************************************************************************************
* @Examle Version V1.0.1.0
* [url=home.php?mod=space&uid=1102]@demo[/url] Version V1.0.4.0
* [url=home.php?mod=space&uid=212281]@date[/url] 2018.5.10
* ************************************************************************************
* 模块性能介绍
* 1、MCU总共提供了26个外部中断,可以设置4级中断的优先级。
* 2、有4个外部中断向量入口,INT0/INT1分别有一个独立的中断向量入口,INT2-7共用一个中断
* 向量入口,INT8-17共用一个中断向量入口。
* 3、INT0-17中断都可以设定4种触发方式,分别为上升沿、下降沿、双沿以及低电平。
* 4、INT0、INT1中断标志位硬件自动清0,INT2-INT17需软件清0。
* 5、P2口中断仅可以设置为下降沿中断,并且只有一个标志位。
* ************************************************************************************
* 应用注意事项
* 1、只要相应的EINTx(x=2...15)被允许,对应的中断标志就可能被置1,反之,对应标志不会被
置1。
* 2、外部中断在使用查询方式时,无法正常清除中断标志位。可以通过先禁止中断使能位再清除
* 中断标志,在完成清除中断标志位后再使能外部中断来进行正常的外部中断产生查询。使用
* 中断方式时,无该问题。
* ************************************************************************************
* 客户服务
* 感谢您使用我们的单片机,若发现错误或对函数的使用存在疑问。请添加上海芯圣电子官方QQ群
* ****************************技术支持群:201030494***********************************
* ************************************************************************************
**/
#define ALLOCATE_EXTERN
#include "HC89F0541.h"
#include "ir.h"
/***************************************************************************************
* 实始化UART1的要关配置
***************************************************************************************/
void UART1_Init()
{
P1M6 = 0xC2; //P16设置为推挽输出
P1M7 = 0x62; //P17设置为上拉输出
TXD_MAP = 0x16; //TXD映射P16
RXD_MAP = 0x17; //RXD映射P17
T4CON = 0x06; //T4工作模式:UART1波特率发生器
TH4 = 0xFF;
TL4 = 0x98; //波特率9600
SCON2 = 0x02; //8位UART,波特率可变
//SCON = 0x10; //允许串行接收
//IE |= 0x10; //使能串口中断
}
void INT0_Init()
{
P0M0 = 0x69; //P00设置为带SMT上拉输入
PITS0 |= 0x01; //INT0下降沿
IE |= 0x01; //打开INT0中断
}
void TIMER0_Init()
{
TMOD |= 0x00; //定时器0工作方式0,16位自动重装
TCON1 |= 0x01; //使用fcpu / 1
TH0 = 0xEE; //根据红外程序和0541定时器模块进行了修改,278us一次中断
TL0 = 0xA0;
ET0 = 1; //定时器0中断
TR0 = 1; //启动定时器0
}
void main()
{
uchar i;
/************************************系统初始化****************************************/
CLKSWR = 0x51; //选择内部高频RC为系统时钟,内部高频RC 2分频,Fosc=16MHz
CLKDIV = 0x01; //Fosc 1分频得到Fcpu,Fcpu=16MHz
INT0_Init();
UART1_Init();
TIMER0_Init();
EA = 1; //打开总中断
while(1)
{
if(IRok) //外部中断处理完接收的32位红外数据
{
IRok = 0;
IRcordpro(); //处理字节
if(IRpro_ok) //如何成功处理,然后通过UART发送出去
{
IRpro_ok = 0;
for(i=0;i<4;i++)
{
SBUF = IRcord[i];
while(!TI);
TI = 0;
}
}
}
}
}
红外处理程序:ir.h (移植于原开发板中的部分程序)
/*====================================
硬件接口位声明
====================================*/
uchar IRtime; //检测红外高电平持续时间(脉宽)
uchar IRcord[4]; //此数组用于储存分离出来的4个字节的数据(用户码2个字节+键值码2个字节)
uchar IRdata[33]; //此数组用于储存红外的33位数据(第一位为引导码用户码16+键值码16)
bit IRpro_ok, IRok; //第一个用于红外接收4个字节完毕。IRok用为检测脉宽完毕
void time0() interrupt TIMER0_VECTOR //定义定时器0
{
IRtime++; //检测脉宽,1次为278us
}
/***************************************************************************************
* @说明 INT0中断服务函数 用于处理红外的各个电平
* @参数 无
* @返回值 无
* @注 无
***************************************************************************************/
void int0() interrupt INT0_VECTOR //定义外部中断0
{
static uchar i; // 声明静态变量(在跳出函数后在回来执行的时候不会丢失数值)i用于把33次高电平的持续时间存入IRdata
static bit startflag; //开始储存脉宽标志位
if(startflag) //开始接收脉宽检测
{
if( (IRtime < 53) && (IRtime >= 32) ) /*判断是否是引导码,底电平9000us+高4500us
这个自己可以算我以11.0592来算了NEC协议的引导码低8000-10000+高4000-5000
如果已经接收了引导码那么i不会被置0就会开始依次存入脉宽*/
i = 0; //如果是引导码那么执行i=0把他存到IRdata的第一个位
IRdata[i] = IRtime; //以T0的溢出次数来计算脉宽,把这个时间存到数组里面到后面判断
IRtime = 0; //计数清零,下一个下降沿的时候在存入脉宽
i++; //计数脉宽存入的次数
if(i == 33) //如果存入34次 数组的下标是从0开始i等于33表示执行了34次
{
IRok = 1; //那么表示脉宽检测完毕
i = 0; //把脉宽计数清零准备下次存入
}
}
else
{
IRtime = 0; //引导码开始进入把脉宽计数清零开始计数
startflag = 1; //开始处理标志位置1
}
}
void IRcordpro() //提取它的33次脉宽进行数据解码
{
uchar i, j, k, cord, value; /*i用于处理4个字节,j用于处理一个字节中每一位,k用于33次脉宽中的哪一位
cord用于取出脉宽的时间判断是否符合1的脉宽时间*/
k = 1; //从第一位脉宽开始取,丢弃引导码脉宽
for(i = 0; i < 4; i++)
{
for(j = 0; j < 8; j++)
{
cord = IRdata[k]; //把脉宽存入cord
if(cord > 5) //如果脉宽大于我11.0592的t0溢出率为约278us*5=1390那么判断为1
value = value | 0x80; /*接收的时候是先接收最低位,
把最低位先放到value的最高位在和0x08按位或一下
这样不会改变valua的其他位的数值只会让他最高位为1*/
if(j < 7)
{
value = value >> 1; //value位左移依次接收8位数据。
}
k++; //每执行一次脉宽位加1
}
IRcord[i] = value; //每处理完一个字节把它放入IRcord数组中。
value = 0; //清零value方便下次在存入数据
}
IRpro_ok = 1; //接收完4个字节后IRpro ok置1表示红外解码完成
}
|