ESP8266 WiFi物联网智能插座—下位机软件实现

news2024/9/26 5:22:30

目录

1、软件架构

2、开发环境 

3、软件功能

4、程序设计

4.1、初始化

4.2、主循环状态机

4.3、初始化模式

4.4、配置模式

4.5、运行模式

4.6、重启模式

4.7、升级模式

5、程序功能特点

5.1、日志管理

5.2、数据缓存队列


本篇博文开始讲解下位机插座节点的MCU软件程序是如何实现。

1、软件架构

下位机软件架构采用前后台控制系统,使用状态机思维实现程序设计。

2、开发环境 

开发环境使用Arduino IDE,IDE安装过程可参见:https://handsome-man.blog.csdn.net/article/details/121195905

智能插座的控制器是ESP8266,需要在IDE中安装该开发包,如下图所示:

3、软件功能

下位机软件整功能如下图所示:

4、程序设计

4.1、初始化

节点上电后会执行初始化,初始化程序顺序执行,代码如下所示:

  Init_Log();
  Log.verboseln("config start!");

  Log.verboseln("init IO");
  Init_IO();
  Log.verboseln("IO OK!");

  Log.verboseln("init EEPROM");
  Init_EEPROM();
  if(Device_VariableInitial(MODE1) == STATUS_SUCCESS)
  {
    Log.verboseln("EEPROM OK!");
  }
  else
  {
    Log.errorln("EEPROM ERROR!");
  }
  
  Log.verboseln("init data queue");
  Init_queue();
  Log.verboseln("data queue OK!");
  
  Log.verboseln("init WiFi and server");
  if(Init_WIFI() == STATUS_SUCCESS)
  {
    Log.verboseln("WiFi and server OK!");
  }
  else
  {
    Log.errorln("WiFi and server ERROR!");
  }
  
  Log.verboseln("init time");
  Init_Time();
  Log.verboseln("time OK!");

  Log.verboseln("init electrical parameter");
  // Init_BL0942();  // 串口初始化时,已经初始化波特率
  Log.verboseln("electrical parameter OK!");

  Log.verboseln("config end!");
  
  program_state.run_state = INIT_STATE;

初始化时候有两点需要注意:

1、节点的日志打印和采集电参数据使用同一路UART,在正式版本软件中,为了避免出现数据错乱的问题,需要将日志打印功能关闭,使#define LOG_OFF 0

 /**
 ******************************************************************************
 ** \brief  初始化log日志模块
 **
 ** \param  无
 **
 ** \retval 无
 **
 ******************************************************************************/
void Init_Log(void)
{
  Serial.begin(4800, SERIAL_8N1);  // 4800bps 无校验
  Serial.println();
  
  Log.setPrefix(printPrefix); // set prefix similar to NLog
  Log.setSuffix(printSuffix); // set suffix 
  Log.begin(LOG_LEVEL_VERBOSE, &Serial);
  Log.setShowLevel(false);    // Do not show loglevel, we will do this in the prefix

  #if LOG_OFF
    DeInit_Log();
  #endif
}

 2、E2PROM使用ESP8266内置的Flash模拟。默认情况下,每次线烧程序、OTA升级程序,这部分存储的配置并不会覆盖或者更新,只有上位机下发更新配置参数才会修改。如果想线烧程序更改配置,需要先将标志位#define DEVICE_FLAG   0XAA55修改成非0XAA55的其他数值。

4.2、主循环状态机

在主循环中使用1ms周期调度维护软件状态机,节点运行有5种状态模式:初始化模式、配置模式、运行模式、重启模式和升级模式。节点默认处于运行模式,代码如下所示:

    switch(program_state.run_state)
    {
      // 初始化模式
      case INIT_STATE:
        Init_State();

        break;

      // 配置模式
      case CONFIG_STATE:
        Config_State();

        break;

      // 运行模式
      case RUN_STATE:
        Run_State();

        break;

      // 重启模式
      case RESET_STATE:
        Reset_State();

        break;

      // 升级模式
      case UPDATA_STATE:
        Updata_State();

        break;
    }

4.3、初始化模式

初始化模式中初始化一些变量数据。

初始化模式中,有一个机制,第一次连接立刻上传一次数据到服务器,否则就按照默认的60秒周期上报数据,第一次上报数据会很慢。代码如下所示:

 /**
 ******************************************************************************
 ** \brief  初始化状态逻辑
 **
 ** \param  无
 **
 ** \retval 无
 **
 ******************************************************************************/
void Init_State(void)
{
  wifi_send_data.device_head = DeviceParamSave.device_head + FUNCTION_ID1;
  wifi_send_data.device_id = DeviceParamSave.device_id;
  memcpy(&(wifi_send_data.software_version), &(DeviceParamSave.software_version), 15);
  memcpy(&(wifi_send_data.hardware_version), &(DeviceParamSave.hardware_version), 15);
  memcpy(&(wifi_send_data.release_time), &(DeviceParamSave.release_time), 10);
  wifi_send_data.upload_cycle = DeviceParamSave.upload_cycle;
  wifi_send_data.sample_cycle = DeviceParamSave.sample_cycle;

  program_state.run_state_time = (DeviceParamSave.upload_cycle * 1000);  // 第一次连接立刻上传一次数据到服务器
  
  program_state.run_state = RUN_STATE;
}

4.4、配置模式

配置模式可接收上位机下发的配置参数,存储到节点E2PROM中。

配置模式有超时机制,3分钟上位机未下发配置参数,自动跳转到运行模式。

更新配置参数后,由配置模式切换到重启模式,节电重启。

代码如下所示:

 /**
 ******************************************************************************
 ** \brief  配置状态逻辑
 **
 ** \param  无
 **
 ** \retval 无
 **
 ******************************************************************************/
void Config_State(void)
{
  program_state.config_state_time++;

  if(program_state.config_state_time >= CYCLE_TIME_180SEC)
  {
    LED_OFF;

    program_state.config_state_time = 0;

    program_state.run_state = RUN_STATE;

	  Log.warningln("config timeout");
    Log.warningln("switch run state");
  }

  // 处理WiFi接收的数据
  if(wifi_receive_flag == true)
  {
    if(receive_data[0] == DeviceParamSave.device_head + FUNCTION_ID4)
    {
      memcpy(&wifi_receive_config, receive_data, sizeof(ReceiveConfig_t));
  
      if((wifi_receive_config.device_old_head == (DeviceParamSave.device_head + FUNCTION_ID4)) && 
      (wifi_receive_config.device_old_id == (DeviceParamSave.device_id) || (wifi_receive_config.device_old_id == 0XFFFF)))
      {
        crc_temp = check_crc16((uint8_t *)&wifi_receive_config, wifi_receive_config.device_len - 2);
        
        if(wifi_receive_config.crc == crc_temp)
        {
          if(wifi_receive_config.device_config_type == 0) // 默认配置
          {
            Log.verboseln("default setting...");
            
            DeviceParamSave.device_flag = DEVICE_FLAG;

            if((wifi_receive_config.device_new_head != 0) && (wifi_receive_config.device_new_head !=  DeviceParamSave.device_head))
            {
              DeviceParamSave.device_head = wifi_receive_config.device_new_head;
            }
            else
            {
              Log.verboseln("DEVICE_HEAD 0 or invariant");
            }
            
            if((wifi_receive_config.device_new_id != 0) && (wifi_receive_config.device_new_id !=  DeviceParamSave.device_id))
            {
              DeviceParamSave.device_id = wifi_receive_config.device_new_id;
            }
            else
            {
              Log.verboseln("DEVICE_ID 0 or invariant");
            }

            if((strcmp(wifi_receive_config.software_version, "") != 0) && (strcmp(wifi_receive_config.software_version, DeviceParamSave.software_version) != 0))
            {
              memcpy(&(DeviceParamSave.software_version), &(wifi_receive_config.software_version), 15);
            }
            else
            {
              Log.verboseln("SW_VERSION null or invariant");
            }
            
            if((strcmp(wifi_receive_config.hardware_version, "") != 0) && (strcmp(wifi_receive_config.hardware_version, DeviceParamSave.hardware_version) != 0))
            {
              memcpy(&(DeviceParamSave.hardware_version), &(wifi_receive_config.hardware_version), 15);
            }
            else
            {
              Log.verboseln("HW_VERSION null or invariant");
            }

            if((strcmp(wifi_receive_config.release_time, "") != 0) && (strcmp(wifi_receive_config.release_time, DeviceParamSave.release_time) != 0))
            {
              memcpy(&(DeviceParamSave.release_time), &(wifi_receive_config.release_time), 10);
            }
            else
            {
              Log.verboseln("RELEASE_TIME null or invariant");
            }
            
            if((wifi_receive_config.upload_cycle != 0) && (wifi_receive_config.upload_cycle !=  DeviceParamSave.upload_cycle))
            {
              DeviceParamSave.upload_cycle = wifi_receive_config.upload_cycle;
            }
            else
            {
              Log.verboseln("UPLOAD_CYCLE 0 or invariant");
            }

            if((wifi_receive_config.sample_cycle != 0) && (wifi_receive_config.sample_cycle !=  DeviceParamSave.sample_cycle))
            {
              DeviceParamSave.sample_cycle = wifi_receive_config.sample_cycle;
            }
            else
            {
              Log.verboseln("SAMPLE_CYCLE 0 or invariant");
            }
            
            if((strcmp(wifi_receive_config.wifi_ssid, "") != 0) && (strcmp(wifi_receive_config.wifi_ssid, DeviceParamSave.wifi_ssid) != 0))
            {
              memcpy(&(DeviceParamSave.wifi_ssid), &(wifi_receive_config.wifi_ssid), 64);
            }
            else
            {
              Log.verboseln("WIFI_SSID null or invariant");
            }

            if((strcmp(wifi_receive_config.wifi_password, "") != 0) && (strcmp(wifi_receive_config.wifi_password, DeviceParamSave.wifi_password) != 0))
            {
              memcpy(&(DeviceParamSave.wifi_password), &(wifi_receive_config.wifi_password), 64);
            }
            else
            {
              Log.verboseln("WIFI_PASSWORD null or invariant");
            }

            if((strcmp(wifi_receive_config.server_ip, "") != 0) && (strcmp(wifi_receive_config.server_ip, DeviceParamSave.server_ip) != 0))
            {
              memcpy(&(DeviceParamSave.server_ip), &(wifi_receive_config.server_ip), 64);
            }
            else
            {
              Log.verboseln("SERVER_IP null or invariant");
            }

            if((wifi_receive_config.server_port != 0) && (wifi_receive_config.server_port !=  DeviceParamSave.server_port))
            {
              DeviceParamSave.server_port = wifi_receive_config.server_port;
            }
            else
            {
              Log.verboseln("SERVER_PORT 0 or invariant");
            }
          }
          else if(wifi_receive_config.device_config_type == 1) // 恢复出厂设置
          {
            Log.verboseln("factory data reset...");
            
            DeviceParamSave.device_flag = DEVICE_FLAG;
            DeviceParamSave.device_head = DEVICE_HEAD;
            DeviceParamSave.device_id = DEVICE_ID;
            memcpy(&(DeviceParamSave.software_version), SW_VERSION, strlen(SW_VERSION));
            memcpy(&(DeviceParamSave.hardware_version), HW_VERSION, strlen(HW_VERSION));
            memcpy(&(DeviceParamSave.release_time), RELEASE_TIME, strlen(RELEASE_TIME));
            DeviceParamSave.upload_cycle = UPLOAD_CYCLE;
            DeviceParamSave.sample_cycle = SAMPLE_CYCLE;
            memcpy(&(DeviceParamSave.wifi_ssid), WIFI_SSID, strlen(WIFI_SSID));
            memcpy(&(DeviceParamSave.wifi_password), WIFI_PASSWORD, strlen(WIFI_PASSWORD));
            memcpy(&(DeviceParamSave.server_ip), SERVER_IP, strlen(SERVER_IP));
            DeviceParamSave.server_port = SERVER_PORT;
          }

		      DeviceParamSave.crc = check_crc16((uint8_t *)&DeviceParamSave, sizeof(DeviceParamSave_t) - 2);
			
          Log.verboseln("DEVICE_HEAD:0X%X", DeviceParamSave.device_head);
          Log.verboseln("DEVICE_ID:0X%X", DeviceParamSave.device_id);
          Log.verboseln("SW_VERSION:%S", DeviceParamSave.software_version);
          Log.verboseln("HW_VERSION:%S", DeviceParamSave.hardware_version);
          Log.verboseln("RELEASE_TIME:%S", DeviceParamSave.release_time);
          Log.verboseln("UPLOAD_CYCLE:%d", DeviceParamSave.upload_cycle);
          Log.verboseln("SAMPLE_CYCLE:%d", DeviceParamSave.sample_cycle);
          Log.verboseln("WIFI_SSID:%S", DeviceParamSave.wifi_ssid);
          Log.verboseln("WIFI_PASSWORD:%S", DeviceParamSave.wifi_password);
          Log.verboseln("SERVER_IP:%S", DeviceParamSave.server_ip);
          Log.verboseln("SERVER_PORT:%d", DeviceParamSave.server_port);

          if(Device_SaveParam() == STATUS_SUCCESS)
          {
            // 成功响应
            wifi_send_state.state_id = ((wifi_receive_config.device_old_head - DeviceParamSave.device_head) << 4) + STATUS_SUCCESS;

            program_state.config_state_time = 0;
            program_state.run_state = RESET_STATE;  // 配置成功,重启节点
			      Log.verboseln("config successful");
            Log.verboseln("switch reset state");
          }
          else
          {
            // 失败响应
            wifi_send_state.state_id = ((wifi_receive_config.device_old_head - DeviceParamSave.device_head) << 4) + STATUS_ERROR;
			      Log.errorln("config fail");
          }
        }
        else
        {
          // 失败响应
          wifi_send_state.state_id = ((wifi_receive_config.device_old_head - DeviceParamSave.device_head) << 4) + STATUS_ERROR;
		      Log.errorln("verify error");
        }
      }
      else
      {
        // 失败响应
        wifi_send_state.state_id = ((wifi_receive_config.device_old_head - DeviceParamSave.device_head) << 4) + STATUS_ERROR;
		    Log.errorln("frame error");
      }
  
      // WiFi发送响应组包
      wifi_send_state.device_head = DeviceParamSave.device_head + FUNCTION_ID2;
      wifi_send_state.device_len = sizeof(SendState_t);
      wifi_send_state.device_id = DeviceParamSave.device_id;
      memcpy(&(wifi_send_state.software_version), &(DeviceParamSave.software_version), 15);
      memcpy(&(wifi_send_state.hardware_version), &(DeviceParamSave.hardware_version), 15);
      wifi_send_state.crc = check_crc16((uint8_t *)&wifi_send_state, wifi_send_state.device_len - 2);
  
      WIFI_send_data((char *)&wifi_send_state, wifi_send_state.device_len);
    }
    
    // 清除数据缓存
    memset(receive_data, 0, wifi_receive_config.device_len);  
    memset((char *)&wifi_send_state, 0, wifi_send_state.device_len);
    memset((char *)&wifi_receive_config, 0, wifi_receive_config.device_len);

    wifi_receive_flag = false;  // 处理完成后,方可接收WiFi新数据
  }
}

4.5、运行模式

只有在运行模式下,上位机才可以切换到配置模式、重启模式和升级模式,其他模式暂不支持远程控制模式切换。

运行模式下可周期上报节点数据,以及支持上位机控制继电器开关。

代码如下所示:

 /**
 ******************************************************************************
 ** \brief  运行状态逻辑
 **
 ** \param  无
 **
 ** \retval 无
 **
 ******************************************************************************/
void Run_State(void)
{
  program_state.run_state_time++;

  // 处理WiFi接收的数据
  if(wifi_receive_flag == true)
  {
    if(receive_data[0] == DeviceParamSave.device_head + FUNCTION_ID3)
    {
      memcpy(&wifi_receive_mode_data, receive_data, sizeof(ReceiveData_Mode_t));
     
      if((wifi_receive_mode_data.device_head == (DeviceParamSave.device_head + FUNCTION_ID3)) && 
      (wifi_receive_mode_data.device_id == (DeviceParamSave.device_id) || (wifi_receive_mode_data.device_id == 0XFFFF)))
      {
        crc_temp = check_crc16((uint8_t *)&wifi_receive_mode_data, wifi_receive_mode_data.device_len - 2);
  
        if(wifi_receive_mode_data.crc == crc_temp)
        {
          if(wifi_receive_mode_data.switch_mode == 0)
          {
            program_state.run_state = RUN_STATE;
            Log.verboseln("keep run state");
          }
          else if(wifi_receive_mode_data.switch_mode == 1)
          {
            RELAY_OFF;  // 进入配置模式,要断开继电器
            program_state.run_state = CONFIG_STATE;
            Log.verboseln("switch config state");
          }
          else if(wifi_receive_mode_data.switch_mode == 2)
          {
            RELAY_OFF;  // 进入升级模式,要断开继电器
            program_state.run_state = UPDATA_STATE;
            Log.verboseln("switch updata state");
          }
          else if(wifi_receive_mode_data.switch_mode == 3)
          {
            RELAY_OFF;  // 进入重启模式,要断开继电器
            program_state.run_state = RESET_STATE;
            Log.verboseln("switch reset state");
          }

          // 成功响应
          wifi_send_state.state_id = ((wifi_receive_mode_data.device_head - DeviceParamSave.device_head) << 4) + STATUS_SUCCESS;
        }
        else
        {
          // 失败响应
          wifi_send_state.state_id = ((wifi_receive_mode_data.device_head - DeviceParamSave.device_head) << 4) + STATUS_ERROR;
		      Log.errorln("verify error");
        }
      }
      else
      {
        // 失败响应
        wifi_send_state.state_id = ((wifi_receive_mode_data.device_head - DeviceParamSave.device_head) << 4) + STATUS_ERROR;
		    Log.errorln("frame error");
      }

      // WiFi发送响应组包
      wifi_send_state.device_head = DeviceParamSave.device_head + FUNCTION_ID2;
      wifi_send_state.device_len = sizeof(SendState_t);
      wifi_send_state.device_id = DeviceParamSave.device_id;
      memcpy(&(wifi_send_state.software_version), &(DeviceParamSave.software_version), 15);
      memcpy(&(wifi_send_state.hardware_version), &(DeviceParamSave.hardware_version), 15);
      wifi_send_state.crc = check_crc16((uint8_t *)&wifi_send_state, wifi_send_state.device_len - 2);

      WIFI_send_data((char *)&wifi_send_state, wifi_send_state.device_len);
    }

    if(receive_data[0] == DeviceParamSave.device_head + FUNCTION_ID5)
    {
      memcpy(&wifi_receive_control_data, receive_data, sizeof(ReceiveData_Control_t));
      
      if((wifi_receive_control_data.device_head == (DeviceParamSave.device_head + FUNCTION_ID5)) && 
      (wifi_receive_control_data.device_id == (DeviceParamSave.device_id) || (wifi_receive_control_data.device_id == 0XFFFF)))
      {
        crc_temp = check_crc16((uint8_t *)&wifi_receive_control_data, wifi_receive_control_data.device_len - 2);
  
        if(wifi_receive_control_data.crc == crc_temp)
        {
          if(wifi_receive_control_data.relay_state == 0)
          {
            RELAY_OFF;
          }
          else
          {
            RELAY_ON;
          }
  
          // 成功响应
          wifi_send_state.state_id = ((wifi_receive_control_data.device_head - DeviceParamSave.device_head) << 4) + STATUS_SUCCESS;
        }
        else
        {
          // 失败响应
          wifi_send_state.state_id = ((wifi_receive_control_data.device_head - DeviceParamSave.device_head) << 4) + STATUS_ERROR;
          Log.errorln("verify error");
        }
      }
      else
      {
        // 失败响应
        wifi_send_state.state_id = ((wifi_receive_control_data.device_head - DeviceParamSave.device_head) << 4) + STATUS_ERROR;
        Log.errorln("frame error");
      }

      // WiFi发送响应组包
      wifi_send_state.device_head = DeviceParamSave.device_head + FUNCTION_ID2;
      wifi_send_state.device_len = sizeof(SendState_t);
      wifi_send_state.device_id = DeviceParamSave.device_id;
      memcpy(&(wifi_send_state.software_version), &(DeviceParamSave.software_version), 15);
      memcpy(&(wifi_send_state.hardware_version), &(DeviceParamSave.hardware_version), 15);
      wifi_send_state.crc = check_crc16((uint8_t *)&wifi_send_state, wifi_send_state.device_len - 2);
      
      WIFI_send_data((char *)&wifi_send_state, wifi_send_state.device_len);
    }

    // 清除数据缓存
    memset(receive_data, 0, wifi_receive_control_data.device_len);  
    memset((char *)&wifi_send_state, 0, wifi_send_state.device_len);
    memset((char *)&wifi_receive_mode_data, 0, wifi_receive_mode_data.device_len);
    memset((char *)&wifi_receive_control_data, 0, wifi_receive_control_data.device_len);

    wifi_receive_flag = false;  // 处理完成后,方可接收WiFi新数据
  }

  // 采集电压、电流和电耗,统计设备有效运行时间
  // 在逻辑上设定,采样时间要小于等于上传云端时间
  // 此项目中采样周期必须设定为1秒
  if((program_state.run_state_time % DeviceParamSave.sample_cycle) == 0)
  {
    Updata_BL0942();

    wifi_send_data.voltage = getVoltage();     // 电压
    wifi_send_data.current = getCurrent();     // 电流
    wifi_send_data.power = getActivePower();   // 功率
    wifi_send_data.electricity = getEnergy();  // 电量

    if(wifi_send_data.power > 0.5)  // 功率大于0.5W,认为有负载
    {
      run_start_flag = true;  
    }
    else
    {
      run_start_flag = false; 
    }
  }

  // 上传数据到服务器
  if(program_state.run_state_time >= (DeviceParamSave.upload_cycle * 1000))
  {
    program_state.run_state_time = 0;  // 上传周期时间要大于采样周期时间

    hours = run_time_ms / 3600000;
    minutes = (run_time_ms % 3600000) / 60000;
    seconds = (run_time_ms % 60000) / 1000;
    time_data = String(hours) + '-' + String(minutes) + '-' + String(seconds);
    memcpy(&(wifi_send_data.run_time), time_data.c_str(), time_data.length());

    for(uint8_t i = time_data.length(); i < 12; i++)
    {
      wifi_send_data.run_time[i] = 0x00;
    }

    wifi_send_data.device_len = sizeof(SendData_t);
    wifi_send_data.crc = check_crc16((uint8_t *)&wifi_send_data, wifi_send_data.device_len - 2);

    WIFI_send_data((char *)&wifi_send_data, wifi_send_data.device_len);
  }
}

4.6、重启模式

确保缓存区数据都发送出去并且断开WiFi和服务器连接后节点重启。代码如下所示:

 /**
 ******************************************************************************
 ** \brief  重启状态逻辑
 **
 ** \param  无
 **
 ** \retval 无
 **
 ******************************************************************************/
void Reset_State(void)
{
  if(tk_queue_empty(&send_dataqueue) == true)  // 确保发送缓存区的数据都发送后才可以重启
  {
    delay(3000);  // 重启节点的ACK可能还未发送出去,需要有延时
    DeInit_WIFI();

    ESP.restart();
  }
}

4.7、升级模式

当所有发送缓存区的数据都发送完成后,才可以执行升级功能。

目前升级仅支持局域网升级,升级前节点会发送升级的IP和端口给上位机。

升级超时时间默认设置为180秒,超时后节点切换到重启模式。

代码如下所示:

 /**
 ******************************************************************************
 ** \brief  升级状态逻辑
 **
 ** \param  无
 **
 ** \retval 无
 **
 ******************************************************************************/
void Updata_State(void)
{
  static bool state_flag = false;

  if(Init_OTA() == STATUS_SUCCESS)
  {
    if(state_flag == false)
    {
      state_flag = true;

      // WiFi发送升级IP和端口
      wifi_send_updata.device_head = DeviceParamSave.device_head + FUNCTION_ID6;
      wifi_send_updata.device_len = sizeof(SendUpdata_t);
      wifi_send_updata.device_id = DeviceParamSave.device_id;
      memcpy(&(wifi_send_updata.software_version), &(DeviceParamSave.software_version), 15);
      memcpy(&(wifi_send_updata.hardware_version), &(DeviceParamSave.hardware_version), 15);
      memcpy(&(wifi_send_updata.updata_ip), ota_ip, strlen(ota_ip));
      wifi_send_updata.updata_port = OTA_PORT;
      wifi_send_updata.crc = check_crc16((uint8_t *)&wifi_send_updata, wifi_send_updata.device_len - 2);

      WIFI_send_data((char *)&wifi_send_updata, wifi_send_updata.device_len);

      memset((char *)&wifi_send_updata, 0, wifi_send_updata.device_len);
    }


    if(tk_queue_empty(&send_dataqueue) == true)  // 确保发送缓存区的数据都发送后才可以升级
    {
      OTA_updata();
    }
  }

  program_state.updata_state_time++;

  if(program_state.updata_state_time >= CYCLE_TIME_180SEC) 
  {
    LED_OFF;

    program_state.updata_state_time = 0;
    
    program_state.run_state = RESET_STATE;

    Log.warningln("updata timeout");
    Log.warningln("switch reset state");
  }
}

5、程序功能特点

5.1、日志管理

下位机支持日志管理,可自定义串口打印不同等级的日志。

不过打印日志的串口和驱动BL0942的串口共用一路,所以在发布正式程序时,需要屏蔽日志打印功能。

日志管理部分代码如下所示:

 /**
 ******************************************************************************
 ** \brief  初始化log日志模块
 **
 ** \param  无
 **
 ** \retval 无
 **
 ******************************************************************************/
void Init_Log(void)
{
  Serial.begin(4800, SERIAL_8N1);  // 4800bps 无校验
  Serial.println();
  
  Log.setPrefix(printPrefix); // set prefix similar to NLog
  Log.setSuffix(printSuffix); // set suffix 
  Log.begin(LOG_LEVEL_VERBOSE, &Serial);
  Log.setShowLevel(false);    // Do not show loglevel, we will do this in the prefix

  #if LOG_OFF
    DeInit_Log();
  #endif
}

5.2、数据缓存队列

发送和接收数据支持FIFO缓存方式写入和读取数据,可自定义缓存区大小。

本项目中程序基本是顺序结构运行,不存在外部中断和定时任务对数据的干扰,并且发送和接收数据的数据量也不是很大,即使暂不使用FIFO缓存也可以满足使用要求。

数据缓存部分代码如下所示:

 /**
 ******************************************************************************
 ** \brief  初始化数据缓存
 **
 ** \param  无
 **
 ** \retval 无
 **
 ******************************************************************************/
void Init_queue(void)
{
  // 清空缓冲区
  memset(send_dataqueue_pool, 0, SEND_DATAQUEUE_POOL_SIZE);
  memset(receive_dataqueue_pool, 0, RECEIVE_DATAQUEUE_POOL_SIZE);
  memset(serial_receive_dataqueue_pool, 0, SERIAL_RECEIVE_DATAQUEUE_POOL_SIZE);

  // 静态方式创建一个循环队列,存满不能再存
  tk_queue_init(&send_dataqueue, send_dataqueue_pool, sizeof(send_dataqueue_pool), sizeof(send_dataqueue_pool[0]), false);
  tk_queue_init(&receive_dataqueue, receive_dataqueue_pool, sizeof(receive_dataqueue_pool), sizeof(receive_dataqueue_pool[0]), false);
  tk_queue_init(&serial_receive_dataqueue, serial_receive_dataqueue_pool, sizeof(serial_receive_dataqueue_pool), sizeof(serial_receive_dataqueue_pool[0]), false);
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1078861.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

淘宝京东拼多多商品价格采集分析接口

淘宝、京东、拼多多商品价格采集分析接口的具体信息&#xff0c;但是可以为您提供一些参考内容&#xff1a; 登录京东、天猫、淘宝、拼多多、苏宁、国美、唯品会等电商平台&#xff0c;注册并获取开发者账号和API接口权限。通过开发者账号和API接口权限&#xff0c;访问京东、…

Java之实现添加文字水印、图片水印功能

文章目录 1 添加文字水印、图片水印1.1 给图片添加文字水印1.1.1 获取原图片对象信息1.1.1.1 读取本地图片1.1.1.2 读取网络图片 1.1.2 添加水印1.1.3 确定水印位置1.1.3.1 设置固定值1.1.3.2 根据原图大小进行设置 1.1.4 获取目标图片1.1.5 完整代码 1.2 java实现给图片添加图…

HomeView/主页 的实现

1. 创建数据模型 1.1 创建货币模型 CoinModel.swift import Foundation// GoinGecko API info /*URL:https://api.coingecko.com/api/v3/coins/markets?vs_currencyusd&ordermarket_cap_desc&per_page250&page1&sparklinetrue&price_change_percentage24…

论文中的小细节——为什么论文中总是写WX而不是XW?

这是最近一个师弟问我的问题&#xff0c;为什么在论文里面总是写的 Y W X YWX YWX&#xff0c;而不是 Y X W YXW YXW&#xff1f;就算有的时候需要转置 W W W 或者 X X X&#xff0c;都不写 Y X W YXW YXW&#xff1f; 这个问题我最开始回答的是这就是约定俗成的写法&…

新能源三电系统测试如何快速连接,避免多次插拔?安规测试电连接器的作用

新能源汽车行业高速发展&#xff0c;电机、电动、电池作为核心三部件&#xff0c;必须有严格的检测已确保产品性能、质量、安全性符合要求。其中会涉及多种测试&#xff0c;如&#xff1a;防水防尘/气密测试、EOL测试、DCR测试等&#xff0c;需要连接相应的检测设备仪器。如何快…

LLaVA:大型语言和视觉助手,图片识别和理解能力让人惊叹

01简介 视觉指令调整&#xff1a;针对多模式 GPT-4 级别功能而构建的大型语言和视觉助手。 视觉聊天&#xff1a;构建多模式 GPT-4 级聊天机器人构建了包含 30 个未见过的图像的评估数据集&#xff1a;每个图像都与三种类型的指令相关联&#xff1a;对话、详细描述和…

查找浏览器中保存的密码

edge浏览器 谷歌浏览器 设置-》自动填充密码-》点击密码管理工具 后就可以查看到浏览器保存的密码了

PyTorch 入门

一、说明 深度学习是机器学习的一个分支&#xff0c;其中编写的算法模仿人脑的功能。深度学习中最常用的库是 Tensorflow 和 PyTorch。由于有各种可用的深度学习框架&#xff0c;人们可能想知道何时使用 PyTorch。以下是人们更喜欢使用 Pytorch 来完成特定任务的原因。 Pytorch…

LeetCode918 环形子数组最大值

这里写自定义目录标题 题目&#xff1a; 思路&#xff1a; 1、破环成链。 由于是环形数组&#xff0c;则数组的末端会与开头相连呈环状。 只需要将长度为N的数组拷贝一次&#xff0c;拼接在原数组后&#xff0c;即得到长度为2N的数组&#xff0c;该数组即为环形数组。 如&am…

mars3d的api文档关于addDynamicPosition查找使用说明

示例链接&#xff1a;功能示例(Vue版) | Mars3D三维可视化平台 | 火星科技 api地址&#xff1a;Mars3D三维可视化平台 | 火星科技 说明&#xff1a; 1.用户反馈不知道如何搜索这个属性的用法 说明&#xff1a; 1. 示例代码中的graphic.addDynamicPosition()说明这个addDynam…

electron 升级 v22 遇到问题

Electron 漏洞 https://mp.weixin.qq.com/s/5LpSJb_5uV8EIDOl3fz9Tw 由于 23以上不在支持win 7 8 8.1 所以我选择安装 v22.3.24 electron 22.3.24 node-sass 6.0.1 sass-loader 10.4.1 对应的版本 npm i node-sass6.0.1 --sass_binary_sitehttps://npm.taobao.org/mirrors…

vscode安装svn扩展(windows)

一、安装 1.1 环境说明 操作系统 windows 10 1.2 安装过程 1. 安装svn 双击安装程序 点击next 继续next 继续next 点击Install 在弹出框中点击 是 开始安装进度&#xff0c;一会将安装成功 安装结束 右键菜单栏中已经有svn选项&#xff0c;并且能正常拉取以…

一区TOP期刊提出一种植被覆盖度估算新方法

研究背景&#xff1a;光合植被(PV)主要指绿色的叶片&#xff0c;而非光合植被(NPV)包括凋落物和茎等。利用遥感数据估算植被覆盖(FVC)主要有三种方法&#xff1a;1) 回归模型2&#xff09;光谱混合物分析(SMA) 3) 数据驱动的模型。在这三种方法中&#xff0c;回归模型因其简单性…

网络安全--APT技术、密码学

目录 1. 什么是APT&#xff1f; 2. APT 的攻击过程&#xff1f; 3. 详细说明APT的防御技术 4. 什么是对称加密&#xff1f; 5. 什么是非对称加密&#xff1f; 6. 私密性的密码学应用&#xff1f; 7. 非对称加密如何解决身份认证问题&#xff1f; 8. 如何解决公钥身份认…

这篇被吹爆了的职称评审个人专业技术业务工作总结范文模板,不看就是你的损失了

职称评审需要写个人工作总结吗&#xff1f;甘建二告诉你毫无疑问&#xff0c;肯定是必须要的&#xff0c;对于评职称的人来说&#xff0c;准备一篇几千字的工作技术总结&#xff0c;是必不可少的申报材料之一。甚至对于初级职称评审来说&#xff0c;个人专业技术总结决定了一个…

三战时区问题

一、前言 前两天测试向我反馈生产环境挖机进口日期页面上展示的比数据库中存储的早了一天&#xff0c;但在测试环境是正常的&#xff0c;并且其它时间字段都没有任何问题&#xff0c;很是奇怪&#xff0c;安排开发排查没有头绪&#xff0c;然后我自己登录生产环境确认LInux系统…

别再瞎考证了,从事网络安全工作,这五大证书是加分项!

其他行业&#xff0c;对证书的要求可能并不明显&#xff0c;但在信息安全类的岗位中&#xff0c;大部分招聘信息中明确标明持有NISP、CISP等网络安全相关资质证书优先等。因此想要快速进入网络安全行业&#xff0c;通过学习并考取相关证书&#xff0c;是一个很不错的途径。 那么…

flutter开发实战-Universal Links配置及flutter微信分享实现

flutter开发实战-Universal Links配置及flutter微信分享实现 在最近开发中碰到了需要实现微信分享&#xff0c;在iOS端需要配置UniversalLink&#xff0c;在分享使用fluwx插件来实现微信分享功能。 一、配置UniversalLink 1.1、什么是UniversalLink Universal link 是Apple…

Vue中的router路由的介绍(快速入门)

路由的介绍 文章目录 路由的介绍1、VueRouter的介绍2、VueRouter的使用&#xff08;52&#xff09;2.1、5个基础步骤(固定)2.2、两个核心步骤 3、组件存放的目录&#xff08;组件分类&#xff09; 生活中的路由&#xff1a;设备和ip的映射关系&#xff08;路由器&#xff09; V…

管网水位监测仪:井下水位监测的创新者

随着城市化进程的不断推进&#xff0c;城市排水管网系统的完善和安全运行越来越受到社会和政府的关注。近期&#xff0c;国家加大了对城市基础设施建设的投入力度&#xff0c;推动了排水管网系统的升级和改造。在这样的背景下&#xff0c;万宾科技推出了一款全新的产品——管网…