美文网首页服务端ESP8266 SDK 开发学习
Esp8266进阶之路16 esp8266的 基于Nonos移植

Esp8266进阶之路16 esp8266的 基于Nonos移植

作者: 半颗心脏 | 来源:发表于2018-03-09 11:30 被阅读408次


    一、前言。

    距离上篇的8266进阶博文有那么一段时间了,那么本文带来的是基于Nonos的红外线H1838的NEC协议的移植小案例,浏览博文前,需要知道以下常识:

    • 1、 红外遥控的原理是什么?
    • 2、红外遥控的协议有哪些?NEC?
    • 3、红外遥控的电路组成?8266的最小系统搭搭建!

    • 这里写图片描述

    二、本博文实现的现象。


    1. 实现的现象;


    遥控器来控制三盏灯的亮灭情况,同时用遥控器进去配网模式,使8266进去一键配网模式!<font color = red >有些人问我的这个哪里买的,本文购买的红外线整套元器件,博文后面有某宝购买链接。

    • 三盏灯(红灯、绿灯、蓝灯),分别连接8266的 <font color = red >GPIO12、GPIO13、GPIO15;

    • 红外接收模块1883的输出端接8266的 <font color = red >GPIO14</font> ,其余看图接 !


    这里写图片描述

    2. 使用说明;


    这里写图片描述

    3. 遥控器对应的码值;


    • 下面仅仅列了本文所需的按键的键值,其他请自行查询;

    按键 键值
    1 0x45
    2 0x46
    3 0x47
    9 0x09
    # 0x0d

    三、esp8266的GPIO口中断使用。


    • 接收短脚使用官方推荐的GPIO14 ,见下图;

    这里写图片描述
    • GPIO14中断初始化 ;

        //中断停止
        ETS_GPIO_INTR_DISABLE();
        //设置中断回调函数
        ETS_GPIO_INTR_ATTACH(hongwai, NULL);
        //设置短脚14,低电平触发;
        gpio_pin_intr_state_set(GPIO_ID_PIN(14), GPIO_PIN_INTR_LOLEVEL);
        //中断使能
        ETS_GPIO_INTR_ENABLE();
    

    四、中断回调函数。


    • 见下图可见 (图片来自网络), 当GPIO14为低电平时候,说明有信号过来了,所以我们上面代码设置为低电平触发中断回调,进一步的剖析电平状态:

    这里写图片描述
        //开始判断是否为NEC波形引导码的前9MS和后4.5MS
        uint32 gpio_status = GPIO_REG_READ(GPIO_STATUS_ADDRESS);
    
        if (gpio_status & BIT(hwx)) {
            GPIO_REG_WRITE(GPIO_STATUS_W1TC_ADDRESS, gpio_status & BIT(hwx));
            gpio_pin_intr_state_set(GPIO_ID_PIN(hwx), GPIO_PIN_INTR_DISABLE);
        }
    
        //计数时间清0
        while (!GPIO_INPUT_GET(hwx)) {
            os_delay_us(100);
            ly_ct++;
        }
        if (ly_ct < 80 || ly_ct > 100) {
            gpio_pin_intr_state_set(GPIO_ID_PIN(hwx), 4);
            return;
        }
    
        //NEC引导码前9MS,ly_ct约等于90,给个误差值,用80-100之间来判断
        ly_ct = 0;
        while (GPIO_INPUT_GET(hwx)) {
            os_delay_us(100);
            ly_ct++;
        }
        if (ly_ct < 40 || ly_ct > 50) {
            gpio_pin_intr_state_set(GPIO_ID_PIN(hwx), 4);
            return;
        }
    
        //NEC引导码引导码的后4.5MS,ly_ct约45
        for (i = 0; i < 4; i++) {
            for (ia = 0; ia < 8; ia++) {
    
                while (!GPIO_INPUT_GET(hwx));
                ly_ct = 0;
                while (GPIO_INPUT_GET(hwx)) {
                    os_delay_us(100);
                    ly_ct++;
                }
                if (ly_ct > 20) {
                    gpio_pin_intr_state_set(GPIO_ID_PIN(hwx), 4);
                    return;
                }
                ly_lhj[i] >>= 1;
                if (ly_ct > 10)
                    ly_lhj[i] |= 0x80; //时间量TH1高于10,即高于1MS判断为1
            }
        }
    
        //ly_lhj[2]就是我们要的数据,因为数组第3个就是上面的第三个数据码
        
        //重新设置中断
        gpio_pin_intr_state_set(GPIO_ID_PIN(hwx), GPIO_PIN_INTR_LOLEVEL);
    

    五、LED初始化以及回调函数。


        //LED初始化,默认全部灭
        
        PIN_FUNC_SELECT(PERIPHS_IO_MUX_MTDO_U, FUNC_GPIO15);
        GPIO_OUTPUT_SET(GPIO_ID_PIN(15), 1);
    
        PIN_FUNC_SELECT(PERIPHS_IO_MUX_MTCK_U, FUNC_GPIO13);
        GPIO_OUTPUT_SET(GPIO_ID_PIN(13), 1);
    
        PIN_FUNC_SELECT(PERIPHS_IO_MUX_MTDI_U, FUNC_GPIO12);
        GPIO_OUTPUT_SET(GPIO_ID_PIN(12), 1);
    
        //因为指示灯是连接到GPIO2,所以把其灭掉,否则会常亮;我也不知道为啥会常亮
        PIN_FUNC_SELECT(PERIPHS_IO_MUX_GPIO2_U, FUNC_GPIO2);
        GPIO_OUTPUT_SET(GPIO_ID_PIN(2), 1);
    

    • 回调函数:

    • <font color=red> 特别强调的是:1-9-1-9-1 才进去一键配网等待,我这利用一个标志位scFlag自加来做,当进去配网模式,全部灯会先亮再灭再亮!


    static int scFlag = 0;
    
        char tempData[8];
        os_sprintf(tempData, "%s", data);
        os_printf("redCallBackFun  : %s\n", tempData);
        os_printf("redCallBackFun scFlag : %d\n", scFlag);
    
        //键值判断
        if (os_strcmp(tempData, "45") == 0) {
            GPIO_OUTPUT_SET(GPIO_ID_PIN(12), 0);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(13), 1);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(15), 1);
            scFlag++;
            if (scFlag == 2 || scFlag == 4) {
                scFlag = 0;
            }
        } else if (os_strcmp(tempData, "46") == 0) {
            GPIO_OUTPUT_SET(GPIO_ID_PIN(13), 0);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(12), 1);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(15), 1);
            scFlag = 0;
        } else if (os_strcmp(tempData, "47") == 0) {
            GPIO_OUTPUT_SET(GPIO_ID_PIN(15), 0);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(12), 1);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(13), 1);
            scFlag = 0;
        } else if (os_strcmp(tempData, "9") == 0) {
            GPIO_OUTPUT_SET(GPIO_ID_PIN(12), 1);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(13), 1);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(15), 1);
            scFlag++;
            if (scFlag == 1 || scFlag == 3 || scFlag == 5) {
                scFlag = 0;
            }
        } else if (os_strcmp(tempData, "d") == 0) {
            scFlag = 0;
            GPIO_OUTPUT_SET(GPIO_ID_PIN(12), 0);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(13), 0);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(15), 0);
        }
    
        //标志位为5,说明进去一键配网等待,灯光闪烁
        if (scFlag == 5) {
            scFlag = 0;
    
            GPIO_OUTPUT_SET(GPIO_ID_PIN(12), 1);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(13), 1);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(15), 1);
            os_delay_us(60000);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(12), 0);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(13), 0);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(15), 0);
            os_delay_us(60000);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(12), 1);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(13), 1);
            GPIO_OUTPUT_SET(GPIO_ID_PIN(15), 1);
    
            os_printf("----------开始进去SmartConfig配网 \n\n-----");
            smartconfig_init();
        }
    
    

    六、编译烧录固件。


    • 编译成功;

    这里写图片描述
    • 烧录,注意我使用的是4M flash的8266-12f ,也就是4*8 =32Mbit的size;

    这里写图片描述
    • 串口打印:
    这里写图片描述

    七、后记;



    相关文章

      网友评论

        本文标题:Esp8266进阶之路16 esp8266的 基于Nonos移植

        本文链接:https://www.haomeiwen.com/subject/wkspfftx.html