可乐丸子 发表于 2018-5-4 11:36:41

多路开关定时问题求助

准备做个esp8266 soc的四路定时开关 四路定时互不影响 可以同时定时 最好加入周一到周七自由选择,下面三个文件应该如何修改gizwits_product.h


#ifndef _GIZWITS_PRODUCT_H_
#define _GIZWITS_PRODUCT_H_

#ifdef __cplusplus
extern "C" {
#endif

#include "gizwits_protocol.h"
#include "gagent_soc.h"
#define ON               1
#define OFF               0
extern bool STA;
extern uint32_t time_data;//用于存储定时数据
extern _tm ntptimeStr;
/**
* MCU software version number
*/
#define SOFTWARE_VERSION                        "03030000"

/**
* MCU hardware version number
*/
#define HARDWARE_VERSION                        "03000001"

/**
* Gagent minor version number for OTA upgrade
* OTA hardware version number: 00ESP826
* OTA software version number: 040206xx // "xx" is version number defaults to "25", consistent with the Gagent library version
*/
#define SDK_VERSION                           "25"   

#ifndef SOFTWARE_VERSION
    #error "no define SOFTWARE_VERSION"
#endif

#ifndef HARDWARE_VERSION
    #error "no define HARDWARE_VERSION"
#endif


/** User area The current device state structure */
extern dataPoint_t currentDataPoint;

/** User area data point initialization */
void userInit(void);

/** User area device status detection */
void userHandle(void);

void getntpTimerFunc(void);
/** User zone event handling */
int8_t gizwitsEventProcess(eventInfo_t * info, uint8_t * data, uint32_t len);

#ifdef __cplusplus
}
#endif
gizwits_product.c
#include <stdio.h>
#include <string.h>
#include "gizwits_product.h"
#include "driver/hal_key.h"
/*
LOCAL os_timer_t time_low_sz;                                                 ///< 用户定时器结构

//关闭定时器
//参数一:要关闭的定时器
os_timer_disarm(&time_low_sz);

//设置定时器
//参数一:要设置的定时器;参数二:定时器到时间要执行的回调函数;参数三:回调函数的参数
os_timer_setfn(&time_low_sz, (os_timer_func_t *)time_low, NULL);

//使能(启动)定时器
//参数一:要使能的定时器;参数二:定时时间(单位:ms);参数三:是否重复执行,即执行完1次
os_timer_arm(&time_low_sz, 1000, 1);

//倒计时函数
void ICACHE_FLASH_ATTR time_low(void)
{



        currentDataPoint.valuecountdown = time_data;//上报当前剩余时间
}
*/

/** 用户区域当前设备状态结构 */
dataPoint_t currentDataPoint;
bool STA={0}; //用于存储5个开关的状态
uint32_t time_data={0};//用于存储定时数据0.定时时1.定时分   2.倒计时(分钟)3.定时开:ON定时关关:OFF   4。需要定时的开关01234

_tm ntptimeStr;//时间结构体返回临时值

/**@name Gizwits 用户界面
* @{
*/
/**
* @brief 事件处理接口
* 描述:
* 1. 用户可以自定义wifi模块状态的变化
* 2. 用户可以在事件处理逻辑中添加数据点,比如调用相关硬件外设操作界面。
* @param info: 事件队列
* @param data: 协议数据
* @param len: 协议数据长度
* @return NULL
* @ref gizwits_protocol.h
*/
int8_t ICACHE_FLASH_ATTR gizwitsEventProcess(eventInfo_t *info, uint8_t *data, uint32_t len)
{
    uint8_t i = 0;
    dataPoint_t * dataPointPtr = (dataPoint_t *)data;
    moduleStatusInfo_t * wifiData = (moduleStatusInfo_t *)data;

    if((NULL == info) || (NULL == data))
    {
      GIZWITS_LOG("!!! gizwitsEventProcess Error \n");
      return -1;
    }

    for(i = 0; i < info->num; i++)
    {
      switch(info->event)
      {
      case EVENT_sw1 :
                        currentDataPoint.valuesw1 = dataPointPtr->valuesw1;
                        GIZWITS_LOG("Evt: EVENT_sw1 %d \n", currentDataPoint.valuesw1);
                        if(0x01 == currentDataPoint.valuesw1)
                        {
                                STA=1;//用户处理
                        }
                        else
                        {
                                STA=0;//用户处理
                        }
                        break;
                case EVENT_sw2 :
                        currentDataPoint.valuesw2 = dataPointPtr->valuesw2;
                        GIZWITS_LOG("Evt: EVENT_sw2 %d \n", currentDataPoint.valuesw2);
                        if(0x01 == currentDataPoint.valuesw2)
                        {
                                STA=1;//用户处理
                        }
                        else
                        {
                                STA=0;//用户处理
                        }
                        break;
                case EVENT_sw3 :
                        currentDataPoint.valuesw3 = dataPointPtr->valuesw3;
                        GIZWITS_LOG("Evt: EVENT_sw3 %d \n", currentDataPoint.valuesw3);
                        if(0x01 == currentDataPoint.valuesw3)
                        {
                                STA=1;//用户处理
                        }
                        else
                        {
                                STA=0;//用户处理
                        }
                        break;
                case EVENT_sw4 :
                        currentDataPoint.valuesw4 = dataPointPtr->valuesw4;
                        GIZWITS_LOG("Evt: EVENT_sw4 %d \n", currentDataPoint.valuesw4);
                        if(0x01 == currentDataPoint.valuesw4)
                        {
                                STA=1;//用户处理
                        }
                        else
                        {
                                STA=0;//用户处理
                        }
                        break;

      case EVENT_ok :
            currentDataPoint.valueok = dataPointPtr->valueok;
            GIZWITS_LOG("Evt: EVENT_ok %d \n", currentDataPoint.valueok);
            if(0x01 == currentDataPoint.valueok)
            {
                    STA=0;//用户处理
            }
            else
            {
                    STA=1;//用户处理
            }
            break;

      case EVENT_sw:
            currentDataPoint.valuesw = dataPointPtr->valuesw;
            GIZWITS_LOG("Evt: EVENT_sw %d\n", currentDataPoint.valuesw);
            switch(currentDataPoint.valuesw)
            {
            case sw_VALUE0://开关1
                    time_data=0;//用户处理
                break;
            case sw_VALUE1://开关2
                    time_data=1;//用户处理
                break;
            case sw_VALUE2://开关3
                    time_data=2;//用户处理
                break;
            case sw_VALUE3://开关4
                    time_data=3;//用户处理
                break;
            case sw_VALUE4://所有开关
                    time_data=4;//用户处理
                                break;
            default:
                break;
            }
            break;

      case EVENT_mode:
            currentDataPoint.valuemode = dataPointPtr->valuemode;
            GIZWITS_LOG("Evt: EVENT_mode %d\n", currentDataPoint.valuemode);
            switch(currentDataPoint.valuemode)
            {
            case mode_VALUE0:
                    time_data=1;//用户处理
                break;
            case mode_VALUE1:
                    time_data=0;//用户处理
                break;
            default:
                break;
            }
            break;

      case EVENT_hour:
            currentDataPoint.valuehour= dataPointPtr->valuehour;
            GIZWITS_LOG("Evt:EVENT_hour %d\n",currentDataPoint.valuehour);
            time_data=currentDataPoint.valuehour;//用户处理
            break;

      case EVENT_minute:
            currentDataPoint.valueminute= dataPointPtr->valueminute;
            GIZWITS_LOG("Evt:EVENT_minute %d\n",currentDataPoint.valueminute);
            time_data=currentDataPoint.valueminute;//用户处理
            break;

      case EVENT_countdown:
            currentDataPoint.valuecountdown= dataPointPtr->valuecountdown;
            GIZWITS_LOG("Evt:EVENT_countdown %d\n",currentDataPoint.valuecountdown);
            time_data=currentDataPoint.valuecountdown * 60;//用户处理,转换成秒钟
            break;

      case WIFI_SOFTAP://WIFI处于SOFTAP模式,热点模式,Access Point,提供无线接入服务,允许其它无线设备接入,提供数据访问,一般的无线路由/网桥工作在该模式下。AP和AP之间允许相互连接 Sta模式;
            break;
      case WIFI_AIRLINK://WIFI处于AIRLINK模式,广播模式
            break;
      case WIFI_STATION://WIFI处于STATION模式,类似于无线终端,sta本身并不接受无线的接入,它可以连接到AP,一般无线网卡即工作在该模式。
            break;
      case WIFI_CON_ROUTER://wifi连接路由器
            GIZWITS_LOG("@@@@ connected router\n");
            break;
      case WIFI_DISCON_ROUTER://WiFi断开路由器
            GIZWITS_LOG("@@@@ disconnected router\n");
            break;
      case WIFI_CON_M2M://WIFI连接M2M
            GIZWITS_LOG("@@@@ connected m2m\n");
                        setConnectM2MStatus(0x01);
            break;
      case WIFI_DISCON_M2M://WIFI断开M2M
            GIZWITS_LOG("@@@@ disconnected m2m\n");
                        setConnectM2MStatus(0x00);
            break;
      case WIFI_RSSI://wifi接收信号的强度
            GIZWITS_LOG("@@@@ RSSI %d\n", wifiData->rssi);
            break;
      case TRANSPARENT_DATA://透明数据传输
            GIZWITS_LOG("TRANSPARENT_DATA \n");
            //用户处理 , 读取数据 , 大小
            break;
      case MODULE_INFO://模块信息
            GIZWITS_LOG("MODULE INFO ...\n");
            break;
            
      default:
            break;
      }
    }
    system_os_post(USER_TASK_PRIO_2, SIG_UPGRADE_DATA, 0);
    return 0;
}


/**
* 用户网络时间获取与数据处理
* 这里用户除了需要获取网络时间以外,还要创建相应的任务
* @参数 无
* @返回 无
*/

void ICACHE_FLASH_ATTR getntpTimerFunc(void)
{
        //倒计时缓存数据
        bool swls1 = {0};//临时缓存倒计时
        bool swls2 = {0};//临时缓存定时
        uint32_t swls = {0};//缓存需要设置的开关标志
/******************倒计时************************
******倒计时处理,定时器可以实现掉网情况下也能执行,还可******************
******以通过网络时间判定,触发了之后保存现在时间,计算出******************
******要相应的时间,然后判断当前时间是否达到了控制的时间******************
******网络处理方法请自行参考上述流程写*************************/
        if((time_data!=0) && (STA==1) && (swls1==0))
        {
                swls1=1;
                swls1=1;
                STA=0;//确定开关也要进行清位上报

        }
        if((time_data!=0) && (swls1==1))//如果倒计时值大于0,而起OK开关按下,倒计时不创建多任务,创建多任务之后无法看到剩余时间
        {
                if(swls1==1)
                {
                        swls=time_data;//缓存要使用的开关标志
                        swls=time_data;//缓存需要设置的开关标志
                        time_data=0;//云端清位
                        time_data=0;//云端清位
                        swls1=0;
                }
                        time_data--;//由于此函数是1秒执行一次,所以每秒减1就行了
        }
                else//倒计时结束了
                {
                        swls1=0;
                        //开关控制
                        switch(swls)
                        {
                                case 0 :
                                {
                                        if(swls==0) STA=0;//设置第一个开关的状态
                                                else STA=1;//设置第一个开关的状态
                                }
                                        break;
                                case 1 :
                                {
                                        if(swls==0) STA=0;//设置第二个开关的状态
                                                else STA=1;//设置第一个开关的状态
                                }
                                        break;
                                case 2 :
                                {
                                        if(swls==0) STA=0;//设置第三个开关的状态
                                                else STA=1;//设置第一个开关的状态
                                }
                                        break;
                                case 3 :
                                {
                                        if(swls==0) STA=0;//设置第四个开关的状态
                                                else STA=1;//设置第一个开关的状态
                                }
                                        break;
                                case 4 :
                                {
                                        if(swls==0) //设置全部开关的状态
                                        {
                                                STA=0;
                                                STA=0;
                                                STA=0;
                                                STA=0;
                                        }
                                                else
                                                {
                                                        STA=1;
                                                        STA=1;
                                                        STA=1;
                                                        STA=1;
                                                }
                                }
                                        break;
                                default:
                                        break;
                        }
                }
/******************倒计时处理结束************************/

/******************定时预约处理*************************/
/*currentDataPoint.valuehour = time_data;
currentDataPoint.valueminute = time_data;*/
        gagentGetNTP(&ntptimeStr);
    os_printf("gagentntp secon: %d",ntptimeStr.second);//打印时间
        if(((time_data != 0) || (time_data != 0)) && (STA==1) && (swls2==0))
        {
                swls2=1;
                swls2=1;
                STA=0;//确定开关也要进行清位上报
                currentDataPoint.valuehour = 0;//云端清除
                currentDataPoint.valueminute = 0;//云端清除
        }
    if(((time_data != 0) || (time_data != 0)) && (swls2==1) )
    {
                if(swls2==1)
                {
                        swls=time_data;//缓存要使用的开关标志
                        swls=time_data;//缓存需要设置的开关标志
                        time_data=0;//云端清位
                        time_data=0;//云端清位
                        swls2=0;
                }
            if(ntptimeStr.hour == time_data)//时
            {
                    if(ntptimeStr.minute == time_data)//分
                    {
                            swls2=0;
                                //开关控制
                                switch(swls)
                                {
                                        case 0 :
                                        {
                                                if(swls==0) STA=0;//设置第一个开关的状态
                                                        else STA=1;//设置第一个开关的状态
                                        }
                                                break;
                                        case 1 :
                                        {
                                                if(swls==0) STA=0;//设置第二个开关的状态
                                                        else STA=1;//设置第一个开关的状态
                                        }
                                                break;
                                        case 2 :
                                        {
                                                if(swls==0) STA=0;//设置第三个开关的状态
                                                        else STA=1;//设置第一个开关的状态
                                        }
                                                break;
                                        case 3 :
                                        {
                                                if(swls==0) STA=0;//设置第四个开关的状态
                                                        else STA=1;//设置第一个开关的状态
                                        }
                                                break;
                                        case 4 :
                                        {
                                                if(swls==0) //设置全部开关的状态
                                                {
                                                        STA=0;
                                                        STA=0;
                                                        STA=0;
                                                        STA=0;
                                                }
                                                        else
                                                        {
                                                                STA=1;
                                                                STA=1;
                                                                STA=1;
                                                                STA=1;
                                                        }
                                        }
                                                break;
                                        default:
                                                break;
                                }

                    }
            }
    }
}

/**
* 用户数据采集
* 这里用户除了需要实现数据点采集以外的数据采集外,还可以自定义采集频率和设计数据滤波算法。
* @参数 无
* @返回 无
*/
void ICACHE_FLASH_ATTR userHandle(void)
{
        //GPIO开关控制,此处为低电平有效控制,若改为高电平,则不需要取反
        GPIO_OUTPUT_SET(GPIO_ID_PIN(12), !STA);
        GPIO_OUTPUT_SET(GPIO_ID_PIN(13), !STA);
        GPIO_OUTPUT_SET(GPIO_ID_PIN(14), !STA);
        GPIO_OUTPUT_SET(GPIO_ID_PIN(15), !STA);

        //数据上报,可适当增加定时上报时间,节约芯片上传产生不必要的负荷
        currentDataPoint.valuesw1 = STA;
        currentDataPoint.valuesw2 = STA;
        currentDataPoint.valuesw3 = STA;
        currentDataPoint.valuesw4 = STA;
        currentDataPoint.valueok= STA;
        if(time_data<60 && time_data>0) currentDataPoint.valuecountdown =1;//不到1分钟按照1分钟计算
        else currentDataPoint.valuecountdown = time_data/60;//秒钟转换成分钟
    system_os_post(USER_TASK_PRIO_2, SIG_UPGRADE_DATA, 0);
}

/**
* 数据点初始化功能
* 在函数中完成初始用户相关数据
* @参数 无
* @返回 无
* @note 开发人员可以在这个函数中添加一个数据点状态初始化值。
*/
void ICACHE_FLASH_ATTR userInit(void)
{
    gizMemset((uint8_t *)¤tDataPoint, 0, sizeof(dataPoint_t));

        /** 警告!!!因变量的初始化,必须在数据范围 **/
    //以下数据点都不具备掉电存储的初始化
        currentDataPoint.valuesw1 = 0;
        currentDataPoint.valuesw2 = 0;
        currentDataPoint.valuesw3 = 0;
        currentDataPoint.valuesw4 = 0;
        currentDataPoint.valueok = 0;
        currentDataPoint.valuesw = 0;
        currentDataPoint.valuemode = 0;
        currentDataPoint.valuehour = 0;
        currentDataPoint.valueminute = 0;
}user_main.c
#include "ets_sys.h"
#include "osapi.h"
#include "user_interface.h"
#include "gagent_soc.h"
#include "user_devicefind.h"
#include "user_webserver.h"
#include "gizwits_product.h"
#include "driver/hal_key.h"
#if ESP_PLATFORM
#include "user_esp_platform.h"
#endif

#ifdef SERVER_SSL_ENABLE
#include "ssl/cert.h"
#include "ssl/private_key.h"
#else
#ifdef CLIENT_SSL_ENABLE
unsigned char *default_certificate;
unsigned int default_certificate_len = 0;
unsigned char *default_private_key;
unsigned int default_private_key_len = 0;
#endif
#endif

/**@name 用户定时器相关参数
* @{
*/
#define USER_TIME_MS 50                                                           ///< 50毫秒时间
LOCAL os_timer_t userTimer;                                                 ///< 用户定时器结构

#define NTP_TIME_1000MS 1000                                                           ///< 1000毫秒时间
LOCAL os_timer_t ntpTimer1000ms;                                                 ///< 用户定时器结构
/**@} */

/**@name 按键的相关定义
* @{
*/
#define GPIO_KEY_NUM                            4                           ///< 定义按键成员的总数。
#define KEY_0_IO_MUX                            PERIPHS_IO_MUX_GPIO2_U      ///< ESP8266 GPIO 功能
#define KEY_0_IO_NUM                            2                           ///< ESP8266 GPIO 数
#define KEY_0_IO_FUNC                           FUNC_GPIO2                  ///< ESP8266 GPIO 名称
#define KEY_1_IO_MUX                            PERIPHS_IO_MUX_GPIO0_U       ///< ESP8266 GPIO 功能
#define KEY_1_IO_NUM                            0                        ///< ESP8266 GPIO 数
#define KEY_1_IO_FUNC                           FUNC_GPIO0               ///< ESP8266 GPIO 名称
#define KEY_2_IO_MUX                            PERIPHS_IO_MUX_GPIO4_U      ///< ESP8266 GPIO 功能
#define KEY_2_IO_NUM                            4                           ///< ESP8266 GPIO 数
#define KEY_2_IO_FUNC                           FUNC_GPIO4                  ///< ESP8266 GPIO 名称
#define KEY_3_IO_MUX                            PERIPHS_IO_MUX_GPIO5_U       ///< ESP8266 GPIO 功能
#define KEY_3_IO_NUM                            5                        ///< ESP8266 GPIO 数
#define KEY_3_IO_FUNC                           FUNC_GPIO5               ///< ESP8266 GPIO 名称
LOCAL key_typedef_t * singleKey;                              ///< 定义单个按键成员数组指针
LOCAL keys_typedef_t keys;                                                ///< 定义总体按键模块结构指针。
/**@} */

/*key1按键短按处理*/
LOCAL void ICACHE_FLASH_ATTR key1ShortPress(void)
{
        STA=!STA;
}
/*key1按键长按处理*/
LOCAL void ICACHE_FLASH_ATTR key1LongPress(void)
{
        gizwitsSetMode(WIFI_SOFTAP_MODE);
        //gizwitsSetMode(WIFI_AIRLINK_MODE);
}
/*key2按键短按处理*/
LOCAL void ICACHE_FLASH_ATTR key2ShortPress(void)
{
        STA=!STA;
}
/* key2按键长按处理*/
LOCAL void ICACHE_FLASH_ATTR key2LongPress(void)
{
        gizwitsSetMode(WIFI_SOFTAP_MODE);
}
/*key3按键短按处理*/
LOCAL void ICACHE_FLASH_ATTR key3ShortPress(void)
{
        STA=!STA;
}
/*key3按键长按处理*/
LOCAL void ICACHE_FLASH_ATTR key3LongPress(void)
{
        gizwitsSetMode(WIFI_SOFTAP_MODE);
}
//key4按键短按处理*
LOCAL void ICACHE_FLASH_ATTR key4ShortPress(void)
{
        STA=!STA;
}
//key4按键长按处理*
LOCAL void ICACHE_FLASH_ATTR key4LongPress(void)
{
        gizwitsSetMode(WIFI_SOFTAP_MODE);
}
/**
* 按键初始化
* @参数 无
* @返回 无
*/
LOCAL void ICACHE_FLASH_ATTR keyInit(void)
{
    singleKey = keyInitOne(KEY_0_IO_NUM, KEY_0_IO_MUX, KEY_0_IO_FUNC,
                              key1LongPress, key1ShortPress);
    singleKey = keyInitOne(KEY_1_IO_NUM, KEY_1_IO_MUX, KEY_1_IO_FUNC,
                              key2LongPress, key2ShortPress);
    singleKey = keyInitOne(KEY_2_IO_NUM, KEY_2_IO_MUX, KEY_2_IO_FUNC,
                              key3LongPress, key3ShortPress);
    singleKey = keyInitOne(KEY_3_IO_NUM, KEY_3_IO_MUX, KEY_3_IO_FUNC,
                              key4LongPress, key4ShortPress);
    keys.singleKey = singleKey;
    keyParaInit(&keys);
    PIN_FUNC_SELECT(PERIPHS_IO_MUX_MTDI_U, FUNC_GPIO12);//开关1
    PIN_FUNC_SELECT(PERIPHS_IO_MUX_MTCK_U, FUNC_GPIO13);//开关2
    PIN_FUNC_SELECT(PERIPHS_IO_MUX_MTMS_U, FUNC_GPIO14);//开关3
    PIN_FUNC_SELECT(PERIPHS_IO_MUX_MTDO_U, FUNC_GPIO15);//开关4
}

/**
* @brief user_rf_cal_sector_set
* 在Flash中使用第636扇区的(2544k ~ 2548k)存储rf_cal参数
* @参数 无
* @返回 无
*/
uint32_t ICACHE_FLASH_ATTR user_rf_cal_sector_set()
{
    return 636;
}

/**
* @brief 程序的入口函数
* 在函数中完成与用户相关的初始化
* @参数 无
* @返回 无
*/
void ICACHE_FLASH_ATTR user_init(void)
{
    uint32_t system_free_size = 0;

    wifi_station_set_auto_connect(1);
    wifi_set_sleep_type(NONE_SLEEP_T);//设置无休眠模式
    espconn_tcp_set_max_con(10);
    uart_init_3(9600,115200);
    UART_SetPrintPort(1);
    GIZWITS_LOG( "---------------SDK version:%s--------------\n", system_get_sdk_version());
    GIZWITS_LOG( "system_get_free_heap_size=%d\n",system_get_free_heap_size());

    struct rst_info *rtc_info = system_get_rst_info();
    GIZWITS_LOG( "reset reason: %x\n", rtc_info->reason);
    if (rtc_info->reason == REASON_WDT_RST ||
      rtc_info->reason == REASON_EXCEPTION_RST ||
      rtc_info->reason == REASON_SOFT_WDT_RST)
    {
      if (rtc_info->reason == REASON_EXCEPTION_RST)
      {
            GIZWITS_LOG("Fatal exception (%d):\n", rtc_info->exccause);
      }
      GIZWITS_LOG( "epc1=0x%08x, epc2=0x%08x, epc3=0x%08x, excvaddr=0x%08x, depc=0x%08x\n",
                rtc_info->epc1, rtc_info->epc2, rtc_info->epc3, rtc_info->excvaddr, rtc_info->depc);
    }

    if (system_upgrade_userbin_check() == UPGRADE_FW_BIN1)
    {
      GIZWITS_LOG( "---UPGRADE_FW_BIN1---\n");
    }
    else if (system_upgrade_userbin_check() == UPGRADE_FW_BIN2)
    {
      GIZWITS_LOG( "---UPGRADE_FW_BIN2---\n");
    }

    keyInit();
   
    userInit();

    gizwitsInit();

    //使用定时
    os_timer_disarm(&userTimer);
    os_timer_setfn(&userTimer, (os_timer_func_t *)userHandle, NULL);
    os_timer_arm(&userTimer, USER_TIME_MS, 1);

    //定时获取网络时间以及处理函数
    os_timer_disarm(&ntpTimer1000ms);
    os_timer_setfn(&ntpTimer1000ms, (os_timer_func_t *)getntpTimerFunc, NULL);
    os_timer_arm(&ntpTimer1000ms, NTP_TIME_1000MS, 1);

    GIZWITS_LOG("--- system_free_size = %d ---\n", system_get_free_heap_size());
}



海底沙 发表于 2018-5-8 07:46:38

我从来都不修改gizwits_product.h,所有定义直接写在gizwits_product.c。

烽烽 发表于 2018-5-15 14:27:26

我觉得用一个定时器,4个uint32的全局变量作为定时器计数值就可以四路定时开关 四路定时互不影响 可以同时定时。当然我们还有APPSDK也有定时任务
页: [1]
查看完整版本: 多路开关定时问题求助