【STC15】STC15单片机获取红外解码从软串口输出
【STC15】STC15单片机获取红外解码从软串口输出
- 单片机型号:
STC15F2K60S2
- 采用外部晶振,频率:
11.0592
MHz - 红外接收头,数据引脚接
P3.2
这里软串口实现主要是位处理时间一定要准确才行。
以Keil编程环境为例,如果是在SDCC编译环境(Platform IO)下,这个时间又要重新计算,具体参考:《【STC单片机】在不同编译环境下波特率位时间计算差异》
void BitTime(void) //位时间函数
{uint i;i = ((MAIN_Fosc / 100) * 104) / 130000 - 1; //根据主时钟来计算位时间while(--i);}
在采用外部晶振频率
11.0592MHz
不变的情况下,如果修改代码中的MAIN_Fosc
参数为12000000ul
那么修改参数如下:
void BitTime(void) //位时间函数
{uint i;i = ((MAIN_Fosc / 100) * 104) / 140000 - 1; //根据主时钟来计算位时间while(--i);}
这个位处理时间不是很好调。
实例代码
/************* 功能说明 **************红外接收程序。适用于市场上用量最大的HT6121/6122及其兼容IC的编码。对于用户码与User_code定义不同的遥控器,程序会将用户码一起从串口输出。使用模拟串口发送监控显示编码,显示内容为ASCII码和中文。本接收程序基于状态机的方式,占用CPU的时间非常少。HEX文件在本目录的/list里面。******************************************//************* 用户系统配置 **************/#define MAIN_Fosc 11059200uL //定义主时钟, 模拟串口和红外接收会自动适应。5~36MHZ#define D_TIMER0 125 //选择定时器时间, us, 红外接收要求在60us~250us之间#define User_code 0xFD02 //定义红外接收用户码/************* 以下宏定义用户请勿修改 **************/
#include "reg51.H"
#include <intrins.H>
#define uchar unsigned char
#define uint unsigned int#define freq_base (MAIN_Fosc / 1200)
#define Timer0_Reload (65536 - (D_TIMER0 * freq_base / 10000))sbit P10 = P1^0;/************* 本地常量声明 **************//************* 本地变量声明 **************/
sbit P_TXD1 = P3^1; //定义模拟串口发送脚,打印信息用
sbit P_IR_RX = P3^2; //定义红外接收输入端口bit P_IR_RX_temp; //Last sample
bit B_IR_Sync; //已收到同步标志
uchar IR_SampleCnt; //采样计数
uchar IR_BitCnt; //编码位数
uchar IR_UserH; //用户码(地址)高字节
uchar IR_UserL; //用户码(地址)低字节
uchar IR_data; //数据原码
uchar IR_DataShit; //数据反码bit B_IrUserErr; //User code error flag
bit B_IR_Press; //Key press flag,include repeat key.
uchar IR_code; //IR code 红外键码/************* 本地函数声明 **************/
void BitTime(void);
void Tx1Send(uchar dat);
uchar HEX2ASCII(uchar dat);
void InitTimer(void);
void PrintString(unsigned char code *puts);/************* 外部函数和变量声明 *****************//********************* 主函数 *************************/
void main(void)
{InitTimer(); //初始化TimerPrintString("****** STC系列MCU红外接收程序 2022-07-08 ******\r\n"); //上电后串口发送一条提示信息while(1){if(B_IR_Press) //有IR键按下{PrintString("红外码: 0x"); //提示红外键码Tx1Send(HEX2ASCII(IR_code >> 4)); //键码高半字节Tx1Send(HEX2ASCII(IR_code)); //键码低半字节if(B_IrUserErr) //用户码错误,则发送用户码{Tx1Send(' '); //发空格Tx1Send(' '); //发空格PrintString("用户码: 0x"); //提示用户码Tx1Send(HEX2ASCII(IR_UserH >> 4)); //用户码高字节的高半字节Tx1Send(HEX2ASCII(IR_UserH)); //用户码高字节的低半字节Tx1Send(HEX2ASCII(IR_UserL >> 4)); //用户码低字节的高半字节Tx1Send(HEX2ASCII(IR_UserL)); //用户码低字节的低半字节}Tx1Send(0x0d); //发回车Tx1Send(0x0a); //发回车B_IR_Press = 0; //清除IR键按下标志}}
}/********************* 十六进制转ASCII函数 *************************/
uchar HEX2ASCII(uchar dat)
{dat &= 0x0f;if(dat <= 9) return (dat + '0'); //数字0~9return (dat - 10 + 'A'); //字母A~F
}//*******************************************************************
//*********************** IR Remote Module **************************//*********************** IR Remote Module **************************
//this programme is used for Receive IR Remote (HT6121).//data format: Synchro,AddressH,AddressL,data,/data, (total 32 bit).//send a frame(85ms), pause 23ms, send synchro of another frame, pause 94ms//data rate: 108ms/Frame//Synchro:low=9ms,high=4.5/2.25ms,low=0.5626ms
//Bit0:high=0.5626ms,low=0.5626ms
//Bit1:high=1.6879ms,low=0.5626ms
//frame space = 23 ms or 96 ms/******************** 红外采样时间宏定义, 用户不要随意修改 *******************/#if ((D_TIMER0 <= 250) && (D_TIMER0 >= 60))#define D_IR_sample D_TIMER0 //定义采样时间,在60us~250us之间
#endif#define D_IR_SYNC_MAX (15000/D_IR_sample) //SYNC max time
#define D_IR_SYNC_MIN (9700 /D_IR_sample) //SYNC min time
#define D_IR_SYNC_DIVIDE (12375/D_IR_sample) //decide data 0 or 1
#define D_IR_DATA_MAX (3000 /D_IR_sample) //data max time
#define D_IR_DATA_MIN (600 /D_IR_sample) //data min time
#define D_IR_DATA_DIVIDE (1687 /D_IR_sample) //decide data 0 or 1
#define D_IR_BIT_NUMBER 32 //bit number//*******************************************************************************************
//**************************** IR RECEIVE MODULE ********************************************void IR_RX_HT6121(void)
{uchar SampleTime;IR_SampleCnt++; //Sample + 1F0 = P_IR_RX_temp; //Save Last sample statusP_IR_RX_temp = P_IR_RX; //Read current statusif(F0 && !P_IR_RX_temp) //Last sample is high,and current sample is low, so is fall edge{SampleTime = IR_SampleCnt; //get the sample timeIR_SampleCnt = 0; //Clear the sample counterif(SampleTime > D_IR_SYNC_MAX) B_IR_Sync = 0; //large the Maxim SYNC time, then errorelse if(SampleTime >= D_IR_SYNC_MIN) //SYNC{if(SampleTime >= D_IR_SYNC_DIVIDE){B_IR_Sync = 1; //has received SYNCIR_BitCnt = D_IR_BIT_NUMBER; //Load bit number}}else if(B_IR_Sync) //has received SYNC{if(SampleTime > D_IR_DATA_MAX) B_IR_Sync=0; //data samlpe time to largeelse{IR_DataShit >>= 1; //data shift right 1 bitif(SampleTime >= D_IR_DATA_DIVIDE) IR_DataShit |= 0x80; //devide data 0 or 1if(--IR_BitCnt == 0) //bit number is over?{B_IR_Sync = 0; //Clear SYNCif(~IR_DataShit == IR_data) //判断数据正反码{if((IR_UserH == (User_code / 256)) &&IR_UserL == (User_code % 256))B_IrUserErr = 0; //User code is righeelse B_IrUserErr = 1; //user code is wrongIR_code = IR_data;B_IR_Press = 1; //数据有效}}else if((IR_BitCnt & 7)== 0) //one byte receive{IR_UserL = IR_UserH; //Save the User code high byteIR_UserH = IR_data; //Save the User code low byteIR_data = IR_DataShit; //Save the IR data byte}}}}
}/**************** Timer初始化函数 ******************************/
void InitTimer(void)
{TMOD = 0; //for STC15Fxxx系列 Timer0 as 16bit reload timer.TH0 = Timer0_Reload / 256;TL0 = Timer0_Reload % 256;ET0 = 1;TR0 = 1;EA = 1;
}/********************** Timer0中断函数************************/
void timer0 (void) interrupt 1
{IR_RX_HT6121();
}/********************** 模拟串口相关函数************************/void BitTime(void) //位时间函数
{uint i;i = ((MAIN_Fosc / 100) * 104) / 130000 - 1; //根据主时钟来计算位时间while(--i);}//模拟串口发送
void Tx1Send(uchar dat) //9600,N,8,1 发送一个字节
{uchar i;EA = 0;P_TXD1 = 0;BitTime();for(i=0; i<8; i++){if(dat & 1) P_TXD1 = 1;else P_TXD1 = 0;dat >>= 1;BitTime();}P_TXD1 = 1;EA = 1;BitTime();BitTime();
}void PrintString(unsigned char code *puts) //发送一串字符串
{for (; *puts != 0; puts++) Tx1Send(*puts); //遇到停止符0结束
}
- 在使用红外遥控按键下按键时的串口打印信息
程序源码
链接:https://pan.baidu.com/s/18tQojIUvf5KbqwW8WKA4Wg
提取码:9e5e
你们一直抱怨这个地方,但是你们却没有勇气走出这里。. ——《飞越疯人院》 |
---|
【STC15】STC15单片机获取红外解码从软串口输出相关推荐
- STC单片机获取红外解码从串口输出
STC单片机获取红外解码从串口输出 串口数据演示 所使用的红外接收头VS1838 VS1838引脚定义 电气特性 所使用的红外按键板 红外遥控接收模块 VS/HX1838 相关的资料: https:/ ...
- STC15系列单片机外设参考资料
STC15系列单片机外设参考资料 今天在调试串口的时候遇到大坑,特此将STC15系列相关的资料粘贴出来说明一下. 在研究STC15W408AS 用定时器1,想用它作为串口1的波特率发生器,死活测试不成 ...
- 求义隆单片机c语言红外解码程序,只用一个单片机定时器的红外接收解码程序...
/****************<51单片机轻松入门-基于STC15W4K系列>配套例程 ************* ★★★★★★★★★★★★★★★★★★★★★★★★ <51单片机 ...
- 自制电脑红外遥控接收器(PC软解码)
网上有很多介绍红外遥控接收器制作的文章,但其中大部分是用单片进行红外解码,然后再通过串口或USB把解码后的按键信息传入到PC的.这样的电路制作起来,不仅造价相对偏高,而且需要对单片编程,这会令大部分软 ...
- STC15系列单片机利用EEPROM统计开机次数
STC15系列单片机利用EEPROM统计开机次数
- STC15系列单片机通过串口多字节数据读写EEPROM操作
STC15系列单片机通过串口多字节数据读写EEPROM操作
- STC15系列单片机EEPROM读写示例
STC15系列单片机EEPROM读写示例
- STM32单片机(12) 红外信号接收解码(外部中断)
[转载请注明出处:http://blog.csdn.net/leytton/article/details/38687537] 本程序主要利用外部中断,实现红外遥控器信号接收解码,并利用串口通信把编码 ...
- 【VSCode PlatformIO】STC15系列单片机开发专用头文件内容分享
[VSCode PlatformIO]STC15系列单片机开发专用头文件内容分享 如果使用VSCode PlatformIO开发STC单片机,所提供的型号的头文件很有限,除非自己添加,像STC系列单片 ...
最新文章
- 前端代码规范(es6,eslint,vue)
- Form 中FND_Message的用法总结
- creo减速器建模实例_Proe/Creo建模实例教程----乌比莫斯之环教程(表达式扫描特征)...
- C语言试题六十八之请编写函数实现亲密数
- Spring配置文件简介
- IAR切BANK--BANK说明
- 论文浅尝 | 采用成对编码的图卷积网络用于知识图谱补全
- 怎么判断两颗二叉树相等
- 相对定位(HTML、CSS)
- 一款 macOS 全能效率神器
- GPG入门 - 练习笔记
- MUR6060PT-ASEMI快恢复二极管MUR6060PT
- 程序员们,千万不要接私活!
- 人工智能时代的幼儿教育软件
- 模拟电路设计的九段境界
- linux网页制作教程,linux:.htaccess文件使用教程
- Activity(一)—— Activity的启动流程
- rk3568-Android11-韦根输入驱动
- 飞凌嵌入式全志A40i开发板试用体验之 称重系统-开发环境搭建(1)
- 用Python写3D游戏,太赞了
热门文章
- SpringBoot核心-Actuator
- STM32与树莓派:嵌入式系统开发与教育计算的区别
- Android屏幕待机时间的获取和设置
- 网络编程---Ip和端口号
- Nacos在Windows10启动失败问题解决办法
- 刻苦学习Java第八天【打卡贴】
- jupyter notebook中,对tensorflow运行时经常出现“The kernel appears to have died. It will restart automatically”
- Mybatis怎么怎么获取刚insert后生成的ID(探索selectKey)
- 达梦数据库使用案例实践
- 黑马程序员云道页面—巩固html和css(新特性没有使用)