Linux驱动开发-编写NEC红外线协议解码驱动

「这是我参与2022首次更文挑战的第22天,活动详情查看:2022首次更文挑战

1. 红外线知识点介绍

在光谱中波长自760nm至400um的电磁波称为红外线,它是一种不可见光。目前几乎所有的视频和音频设备都可以通过红外遥控的方式进行遥控,比如电视机、空调、影碟机等,都可以见到红外遥控的影子。这种技术应用广泛,相应的应用器件都十分廉价,因此红外遥控是我们日常设备控制的理想方式。

红外线遥控是目前使用最广泛的一种通信和遥控手段。由于红外线遥控装置具有体积小、功耗低、功能强、 成本低等特点,因而,继彩电、录像机之后,在录音机、音响设备、空凋机以及玩具等其它小型电器装置上也纷 纷采用红外线遥控。工业设备中,在高压、辐射、有毒气体、粉尘等环境下,采用红外线遥控不仅完全可靠而且 能有效地隔离电气干扰。

NEC协议是众多红外线协议中的一种,以前广泛用在电视机,投影仪设备里,很早之前经常说的万能电视遥控器就是NEC协议的。

当前文章就介绍如何在Linux下通过红外线接收模块,编写一个NEC协议的红外线解码驱动,解析遥控器传输过来的各种控制指令,完成对应的动作响应;驱动里用到了外部中断接收数据,通过定时器计算间隔时间完成解码。

NEC协议的特点如下:

单个码一共分为5各部分(没有算重复码): 引导码+用户码+用户反码+按键码+按键反码

对于接收方_引导码: 9ms的低电平+4.5ms的高电平。

接收的数据是0: 560us低电平+560us高电平

接收的数据是1: 560us低电平+1680us高电平

image-20220106112451549

image-20220106111805216

2. 硬件环境

当前开发板采用友善之臂的Tiny4412,CPU是三星的EXYNOS4412,最高主频为1.5GHZ,Linux内核版本是3.5。

下面是红外线接收模块原理图:

image-20220106111542287

通过杜邦线接在开发板的中断输入脚上: (GPX1_0接口上,第9个排针。)

image-20220106112325813

驱动安装后,解码的效果:

image-20220106112527780

3. 案例代码

驱动代码思路: 采用外部中断接收NEC的数据,在工作队列里完成协议解析,最终通过printk打印出来。

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/interrupt.h>
#include <linux/gpio.h>
#include <mach/gpio.h>
#include <plat/gpio-cfg.h>
#include <linux/delay.h>
#include <linux/workqueue.h>static int irq;
#define NEC_INFRARED_GPIO EXYNOS4_GPX1(0)
static struct work_struct work;/*获取高电平持续时间--us单位*/
static u32 GetTimeH(void)
{
    ktime_t my_time,my_time2;
    unsigned int i,j;
    my_time=ktime_get(); //获取当前时间
    i=ktime_to_us(my_time); //转 us
    while(gpio_get_value(NEC_INFRARED_GPIO)){}
    my_time2=ktime_get(); //获取当前时间
    j=ktime_to_us(my_time2); //转 us
    return j-i;
}
​
/*获取低电平持续时间--us单位*/
static u32 GetTimeL(void)
{
    ktime_t my_time,my_time2;
    unsigned int i,j;
    my_time=ktime_get(); //获取当前时间
    i=ktime_to_us(my_time); //转 us
    while(gpio_get_value(NEC_INFRARED_GPIO)==0){}
    my_time2=ktime_get(); //获取当前时间
    j=ktime_to_us(my_time2); //转 us
    return j-i;
}
​
/*
工作函数
*/
static u8 buf[4];//[0]用户码  [1]用户反码 [2]按键码  [3]按键反码
static void new_work_func(struct work_struct *work)
{
    u8 data=0;
    u32 time_us;
    /*1. 判断引导码*/
    time_us=GetTimeL();
    if(time_us>12000 || time_us <7000)return;  //标准9000
    time_us=GetTimeH();
    if(time_us>6000 || time_us <3000)return;  //标准4500
    /*2. 接收32位数据*/
    int i,j;
    for(i=0;i<4;i++)
    {
        for(j=0;j<8;j++)
        {
            time_us=GetTimeL();
            if(time_us>700 || time_us <400)return;  //标准560
            time_us=GetTimeH();
            if(time_us<700 && time_us>400)  // 0 :标准560
            {
                data<<=1;
            }
            else if(time_us<1800 && time_us>1500)  // 1 :标准1680
            {
                data<<=1;
                data|=0x01;
            }
            else
            {
                return;
            }
        }
        buf[i]=data;
    }
    printk("用户码:%d,按键码:%d\n",buf[0],buf[2]);
}
​
/*中断服务函数*/
irqreturn_t nec_irq_handler_func(int irq, void *dev)
{
    /*添加工作到工作队列*/
    schedule_work(&work);
    return IRQ_HANDLED;
}
​
static int __init tiny4412_hello_module_init(void)
{
    /*初始化工作函数*/
    INIT_WORK(&work,new_work_func);
    /*1. 获取中断号*/
    irq=gpio_to_irq(NEC_INFRARED_GPIO);
    /*2. 注册中断*/
    request_irq(irq,nec_irq_handler_func,IRQF_TRIGGER_FALLING,"tinyy412_nec",NULL);
​
    printk("驱动测试: 驱动安装成功\n");
    return 0;
}
​
static void __exit tiny4412_hello_module_cleanup(void)
{
    free_irq(irq,NULL);
    printk("驱动测试: 驱动卸载成功\n");
}
​
module_init(tiny4412_hello_module_init);    /*驱动入口--安装驱动的时候执行*/
module_exit(tiny4412_hello_module_cleanup); /*驱动出口--卸载驱动的时候执行*/MODULE_LICENSE("GPL");  /*设置模块的许可证--GPL*/复制代码

猜你喜欢

转载自juejin.im/post/7062243125019803661