远程固件升级服务(第三方服务器,使用libfota2扩展库)

一、FOTA 概述

FOTA 即远程升级功能,此功能可以让客户在不方便大量线刷升级(设备不在身边/量产 PCB 没引出 USB/需要大批量进行功能升级)的情况下,快速进行底层固件/脚本/脚本 + 底层固件的远程更新。

LuatOS 开发模式下,固件分为两部分:core 和 script

远程升级时:升级 script 和 core+script ,仅 script 脚本升级时为全量升级,core+script 都升级时为差分升级

远程升级时:可以仅升级 script;也可以同时升级 core+script

支持 iot 平台升级和自建第三方服务器(HTTP)升级

二、演示功能概述

本文将详细讲述如何使用自建服务器服务器进行远程升级。另外此教程演示了自建服务器的三种升级相关场景:

(1) fota 升级简单演示:使用 iot 服务器进行远程升级功能模块,简单升级演示;

(2) tcp 服务器下发升级指令:通过 tcp 服务器下发升级指令(指令格式使用 json 字符串,包含是否升级参数),控制设备使用 fota 功能模块;

(3) psm 低功耗 fota:低功耗 fota 功能模块,此场景是针对 psm 状态下升级没完成就进入休眠导致升级失败的情况写的一个示例。

三、演示硬件环境

1、Air780EPM V1.3 版本开发板一块 + 可上网的 sim 卡一张 +4g 天线一根 + 网线一根:

  • sim 卡插入开发板的 sim 卡槽
  • 天线装到开发板上
  • 网线一端插入开发板网口,另外一端连接可以上外网的路由器网口

2、TYPE-C USB 数据线一根 ,Air780EPM V1.3 版本开发板和数据线的硬件接线方式为:

  • Air780EPM V1.3 版本开发板通过 TYPE-C USB 口供电;(外部供电/USB 供电 拨动开关 拨到 USB 供电一端)
  • TYPE-C USB 数据线直接插到核心板的 TYPE-C USB 座子,另外一端连接电脑 USB 口;

四、软件环境

在开始实践本示例之前,先筹备一下软件环境:

  1. 烧录工具: Luatools 工具;
  2. 内核固件:Air780EPM V2012 版本固件(理论上,2025 年 8 月 10 日之后发布的固件都可以)
  3. LuatOS 需要的脚本和资源文件

准备好软件环境之后,将本篇文章中演示使用的项目文件烧录到 Air780EPM 开发板中。

4.为了方便测试,提供了免费的不可商用的 TCP/UDP web 测试工具:

五、API 接口说明

libfota2 - fota 升级 v2

六、自有服务器 FOTA

FOTA 有多种方式,可以使用iot 平台进行升级,也可以使用用户自建平台升级,可以只升级 core,可以只升级用户脚本,也可以 core+ 脚本一起升级,接下来介绍自建服务器的升级。

6.1 网卡切换

fota 功能需要联网去下载升级包,所以需要选择一种联网方式:

netdrv_device:配置连接外网使用的网卡,目前支持以下四种选择(三选一)

(1) netdrv_4g:4G 网卡

(2) netdrv_eth_spi:通过 SPI 外挂 CH390H 芯片的以太网卡

(3) netdrv_multiple:支持以上三种网卡,可以配置三种网卡的优先级

在 netdrv_device.lua 文件中选择一种使用即可。

6.2 云平台配置

每一家的云平台都不一样,但是本质上都一样,都是给模块下发正确的升级文件。建议平台做一些验证,比如 imei 验证、版本号验证,循环请求验证,这样能够有效的避免因为升级包下载不对导致的模块循环升级,不跑代码其他逻辑,引起的假死机现象。

6.3 升级包制作并上传

自建服务器可以使用此服务器来做测试 ,该服务器只能上传文件,上传后可以获得一个下载链接,复制的链接用作自建服务器的 url。

所以我们在最开始先要生成一个链接,步骤如下:在进行脚本改动前,需要在你的电脑上随便生成一个.bin 后缀,大小不为 0 的文件(可以随便往里面写点什么),然后将这个 bin 文件上传到模块请求的地址去,目的是为了获取升级的 url。

6.3.1 单脚本升级
6.3.1.1 旧版本软件本地烧录

如果用户只是新增一些自己的脚本逻辑,没有更新底层,可以选择仅脚本升级

1.脚本中修改 url 为自己的升级 url,然后打开 luatools 的项目管理界面,点击生成量产文件,默认放在 luatools 根目录下的"SOC 量产及远程升级文件\Air8000"目录下

注:一定要注意 url 前的"###"标志位,如果不加###,模块进行 HTTP 请求时,会在请求头里加上 version、project_key,imei,firmware_name 等参数

2.将对应生成的 2010 版本,001.000.000 的 SOC 的固件烧录到模组中。

6.3.1.2 新版本软件级包制作并上传

1.因为模块烧录的是 001.000.000 版本,所以我们需要给脚本里的版本号改一下,改为 001.000.001 版本

2.再将脚本中增加几行打印(为了模拟用户修改脚本的动作)

3.然后重新生成一次量产固件:

4.将升级文件改名成上面 url 中的文件名字 FOTA2_DEMO_001.000.001_LuatOS-SoC_V2010_Air780EPM_1.soc 。然后上传到服务器对应的地址。

6.3.2 含 core 升级

注意:对于不同后缀的固件,不能直接升级,分区有差异,强行升级可能无法启动。比如:LuatOS-SoC_V2012_Air780EPM_1.soc 只能升级LuatOS-SoC_V2014_Air780EPM_1.soc 后缀的固件,不能升级 LuatOS-SoC_V2014_Air780EPM_2.soc 后缀的固件。

6.3.2.1 旧版本软件本地烧录

每一次 core 的升级都会带来一些网络上的优化(例如信号差时的网络稳定性)以及一些 bug 修复,所以在发布新版本以后,用户可以先测试下 core 对自己脚本有无明显影响或性能提升,然后进行远程 FOTA。

1.本次选择脚本版本是 001.000.000,升级到 001.000.001;固件版本从 2010 升级到 2012 版本,

按照生成量产固件的步骤,001.000.000 版本脚本搭配 V2010 版本固件生成量产文件如下图:

2.将对应生成的 2010 版本,001.000.000 的 SOC 的固件烧录到模组中。

6.3.2.2 新版本软件级包制作并上传

1.修改脚本版本后以及固件版本后重新生成量产固件如下:

需要注意的是:生成了这个版本以后,再去脚本中改动脚本版本号为 001.000.002(只要前后两位有一位大于 1,一位大于 0 即可(当然第一位大于 1 的时候,整个版本号自然大于 001.000.001),如 002.000.200;001.000.001;030.000.311 等)

脚本版本号分为 A.B.C 三段;

因为历史原因,中间这一段 B 没有任何意义,但是必须存在;

假设旧的脚本版本号为 A1.B1.C1,新的脚本版本号为 A2.B2.C2;

当新旧脚本版本号满足以下任何一种条件时,版本号允许升级:

(1)、A2 等于 A1,同时 C2 大于 C1;

(2)、A2 大于 A1,同时 C2 大于或者等于 C1;

(3)、B1 B2 无意义

2.接下来就是针对这两个量产文件,制作一个差分文件,用来远程升级(注:远程升级中 core 为差分,脚本为全量升级)

点击到 luatools 的主界面,依次点击图中蓝框所示意的地方(注:必须使用 luatools_3.0.9 及其以上版本,要不差分包升级的时候可能会出问题)

3.按下图所示选择低版本以及高版本的固件,然后点击开始执行即可,如果不想输出的差分包在 luatools 根目录下,可以自行选择一个输出路径

4.在你选择的目录下看到如下所示,带着脚本的 PROJECT core 版本号 脚本版本号的 bin 后缀的差分文件。

5.将升级文件上传完成以后,为了防止模块当前固件不是最早的版本,可以点击 luatools 主界面右边的"下载固件"选择最早的 001.000.000 版本,按住 boot 重启模块,然后点击下载,将最早的固件下载进模块里即可

6、将升级文件改名成上面 url 中的文件名字 FOTA2_DEMO_001.000.001_LuatOS-SoC_V2012_Air780EPM_1.soc 。然后上传到服务器对应的地址。

7、将对应生成的 2010 版本,001.000.000 的 SOC 的量产固件烧录到模组中。

6.4 示例结果展示

本 demo 将会演示三种场景下的 fota 升级,场景在 main.lua 中切换。

6.4.1 场景一:fota 升级简单演示

主要代码如下:

-- 升级结果的回调函数 -- 功能:获取fota的回调函数 -- 参数: -- result:number类型 -- 0表示成功 -- 1表示连接失败 -- 2表示url错误 -- 3表示服务器断开 -- 4表示接收报文错误 -- 5表示使用iot平台VERSION需要使用 xxx.yyy.zzz形式 local function fota_cb(ret) log.info("fota", ret) if ret == 0 then log.info("升级包下载成功,重启模块") rtos.reboot() elseif ret == 1 then log.info("连接失败", "请检查url拼写或服务器配置(是否为内网)") elseif ret == 2 then log.info("url错误", "检查url拼写") elseif ret == 3 then log.info("服务器断开", "检查服务器白名单配置") elseif ret == 4 then log.error("FOTA 失败", "原因可能有:\n" .. "1) 服务器返回 200/206 但报文体为空(0 字节)—— 通常是升级包文件缺失或 URL 指向空文件;\n" .. "2) 服务器返回 4xx/5xx 等异常状态码 —— 请确认升级包已上传、URL 正确、鉴权信息有效;\n".. "3) 已经是最新版本,无需升级" ) elseif ret == 5 then log.info("版本号书写错误", "iot平台版本号需要使用xxx.yyy.zzz形式") else log.info("不是上面几种情况 ret为", ret) end end -- 使用第三方服务器,配置ota_opts参数 --[[ -- opts参数说明, 所有参数都是可选的 -- 1. opts.url string 升级所需要的URL, 若使用合宙iot平台,则不需要填 -- 2. opts.version string 版本号, 默认是 BSP版本号.x.z格式 -- 3. opts.timeout int 请求超时时间, 默认300000毫秒,单位毫秒 -- 4. opts.project_key string 合宙IOT平台的项目key, 默认取全局变量PRODUCT_KEY. 自建服务器不用填 -- 5. opts.imei string 设备识别码, 默认取IMEI(Cat.1模块)或WLAN MAC地址(wifi模块)或MCU唯一ID -- 6. opts.firmware_name string 固件名称,默认是 _G.PROJECT.. "_LuatOS-SoC_" .. rtos.bsp() -- 7. opts.server_cert string 服务器证书, 默认不使用 -- 8. opts.client_cert string 客户端证书, 默认不使用 -- 9. opts.client_key string 客户端私钥, 默认不使用 -- 10. opts.client_password string 客户端私钥口令, 默认不使用 -- 11. opts.method string 请求方法, 默认是GET -- 12. opts.headers table 额外添加的请求头,默认不需要 -- 13. opts.body string 额外添加的请求body,默认不需要 ]] local opts = { url = "###http://airtest.openluat.com:2900/download/FOTA2_DEMO_2012.001.001_LuatOS-SoC_Air8000.bin", -- 合宙IOT平台的默认升级URL, 不填就是这个默认值 -- 如果是自建的OTA服务器, 则需要填写正确的URL, 例如 http://192.168.1.5:8000/update -- 如果自建OTA服务器,且url包含全部参数,不需要额外添加参数, 请在url前面添加 ### -- 如果不加###,则默认会上传如下参数 -- 1. opts.version string 版本号, 默认是 BSP版本号.x.z格式 -- 2. opts.timeout int 请求超时时间, 默认300000毫秒,单位毫秒 -- 3. opts.project_key string 合宙IOT平台的项目key, 默认取全局变量PRODUCT_KEY. 自建服务器不用填 -- 4. opts.imei string 设备识别码, 默认取IMEI(Cat.1模块)或WLAN MAC地址(wifi模块)或MCU唯一ID -- 5. opts.firmware_name string 底层版本号 -- 请求的版本号, 合宙IOT有一套版本号体系,不传就是合宙规则, 自建服务器的话当然是自行约定版本号了 -- version = "" -- 其他更多参数, 请查阅libfota2的文档 https://wiki.luatos.com/api/libs/libfota2.html } function fota_task_func() -- 如果当前时间点设置的默认网卡还没有连接成功,一直在这里循环等待 while not socket.adapter(socket.dft()) do log.warn("fota_task_func", "wait IP_READY", socket.dft()) -- 在此处阻塞等待默认网卡连接成功的消息"IP_READY" -- 或者等待1秒超时退出阻塞等待状态; -- 注意:此处的1000毫秒超时不要修改的更长; -- 因为当使用libnetif.set_priority_order配置多个网卡连接外网的优先级时,会隐式的修改默认使用的网卡 -- 当libnetif.set_priority_order的调用时序和此处的socket.adapter(socket.dft())判断时序有可能不匹配 -- 此处的1秒,能够保证,即使时序不匹配,也能1秒钟退出阻塞状态,再去判断socket.adapter(socket.dft()) sys.waitUntil("IP_READY", 1000) end -- 检测到了IP_READY消息 log.info("fota_task_func", "recv IP_READY", socket.dft()) ----这个判断是提醒要设置url的,且不要使用本文中的测试服务器,实际生产请删除 -- if not opts.url or string.find(opts.url,"airtest.openluat.com") then -- while 1 do -- sys.wait(1000) -- log.info("fota", "当前URL",opts.url,"请修改正确的url") -- end -- end log.info("开始检查升级") libfota2.request(fota_cb, opts) end --创建并且启动一个task --运行这个task的主函数fota_task_func sys.taskInit(fota_task_func) -- 演示定时自动升级, 每隔4小时自动检查一次 sys.timerLoopStart(libfota2.request, 4 * 3600000, fota_cb, opts)

升级中:模块请求升级,下载完升级包以后会进行 MD5 验证升级包有无问题,如果没问题,就会启动重启程序,然后进行升级工作

升级后:升级成功后按我们之前的脚本,每隔 5S,打印一次"降功耗,找合宙"以及当前脚本版本号,可以看出,当前版本号已经由原来的 001.000.000 变为了 001.000.001,固件版本从 2010 变为 2012。

6.4.2 场景二:tcp 服务器下发升级指令

通过 tcp 服务器下发升级指令(指令格式使用 json 字符串,包含是否升级参数),控制设备使用 fota 功能模块。此场景下,设备会先连接到 TCP 服务器,等待服务器下发升级指令。在 air_srv_fota.lua 中会有一个参数控制,防止升级过程中重复下载升级包。

在 tcp_self_main.lua 中配置 tcp 服务器信息:

升级前:设备连接到 TCP 服务器,等待指令下发:{"fota": "true", "url": "

升级中:模组收到服务器下发的升级指令后,设备开始下载升级包并进行验证

升级后:升级包下载完成后设备重启升级,并循环打印新的版本号信息

6.4.3 场景三:psm 低功耗 fota

此场景是防止 psm 状态下升级没完成就进入休眠导致升级失败的情况写的一个例子。需要注意的是要等待升级成功后再去进入休眠。

主要代码如下:

-- 升级结果的回调函数 -- 功能:获取fota的回调函数 -- 参数: -- result:number类型 -- 0表示成功 -- 1表示连接失败 -- 2表示url错误 -- 3表示服务器断开 -- 4表示接收报文错误 -- 5表示使用iot平台VERSION需要使用 xxx.yyy.zzz形式 local function fota_cb(ret) log.info("fota", ret) --升级结束,触发升级回调,发布消息升级结束,可以进入休眠模式 sys.publish("FOTA_END") if ret == 0 then log.info("升级包下载成功,重启模块") rtos.reboot() elseif ret == 1 then log.info("连接失败", "请检查url拼写或服务器配置(是否为内网)") elseif ret == 2 then log.info("url错误", "检查url拼写") elseif ret == 3 then log.info("服务器断开", "检查服务器白名单配置") elseif ret == 4 then log.error("FOTA 失败", "原因可能有:\n" .. "1) 服务器返回 200/206 但报文体为空(0 字节)—— 通常是升级包文件缺失或 URL 指向空文件;\n" .. "2) 服务器返回 4xx/5xx 等异常状态码 —— 请确认升级包已上传、URL 正确、鉴权信息有效;\n".. "3) 已经是最新版本,无需升级" ) elseif ret == 5 then log.info("版本号书写错误", "iot平台版本号需要使用xxx.yyy.zzz形式") else log.info("不是上面几种情况 ret为", ret) end end -- 使用合宙iot平台进行升级, 支持自定义参数, 也可以不配置,如果要配置参数可以参考此链接https://docs.openluat.com/osapi/ext/libfota2/ local opts = { url = "###http://airtest.openluat.com:2900/download/FOTA2_DEMO_2012.001.001_LuatOS-SoC_Air8000.bin", -- 合宙IOT平台的默认升级URL, 不填就是这个默认值 -- 如果是自建的OTA服务器, 则需要填写正确的URL, 例如 http://192.168.1.5:8000/update -- 如果自建OTA服务器,且url包含全部参数,不需要额外添加参数, 请在url前面添加 ### -- 如果不加###,则默认会上传如下参数 -- 1. opts.version string 版本号, 默认是 BSP版本号.x.z格式 -- 2. opts.timeout int 请求超时时间, 默认300000毫秒,单位毫秒 -- 3. opts.project_key string 合宙IOT平台的项目key, 默认取全局变量PRODUCT_KEY. 自建服务器不用填 -- 4. opts.imei string 设备识别码, 默认取IMEI(Cat.1模块)或WLAN MAC地址(wifi模块)或MCU唯一ID -- 5. opts.firmware_name string 底层版本号 -- 请求的版本号, 合宙IOT有一套版本号体系,不传就是合宙规则, 自建服务器的话当然是自行约定版本号了 -- version = "" -- 其他更多参数, 请查阅libfota2的文档 https://wiki.luatos.com/api/libs/libfota2.html } function psm_fota_task_func() -- 如果是被定时器唤醒,因为上次进入PSM+时是开启了飞行模式,所以在唤醒后第一时间关闭飞行模式。 mobile.flymode(0, false) log.info("开始测试PSM+模式功耗。") -- 如果当前时间点设置的默认网卡还没有连接成功,一直在这里循环等待 while not socket.adapter(socket.dft()) do log.warn("fota_task_func", "wait IP_READY", socket.dft()) -- 在此处阻塞等待默认网卡连接成功的消息"IP_READY" -- 或者等待1秒超时退出阻塞等待状态; -- 注意:此处的1000毫秒超时不要修改的更长; -- 因为当使用libnetif.set_priority_order配置多个网卡连接外网的优先级时,会隐式的修改默认使用的网卡 -- 当libnetif.set_priority_order的调用时序和此处的socket.adapter(socket.dft())判断时序有可能不匹配 -- 此处的1秒,能够保证,即使时序不匹配,也能1秒钟退出阻塞状态,再去判断socket.adapter(socket.dft()) sys.waitUntil("IP_READY", 1000) end -- 检测到了IP_READY消息 log.info("fota_task_func", "recv IP_READY", socket.dft()) log.info("开始检查升级") libfota2.request(fota_cb, opts) -- 打印版本号, 方便看版本号变化, 非必须 log.info("fota", "脚本版本号", VERSION, "core版本号", rtos.version()) -- 等待下载升级包结束, 发布消息"FOTA_END", -- 如果15秒内没有收到消息,则15秒的时长到达后进入PSM+模式。 -- 需要注意的是在fota_cb回调函数中,升级包下载成功后,会立马重启并升级模组。如果还有其他事情要做不想立马重启升级,需自行决定reboot的时机 -- 升级包下载成功后,本demo默认是立即自动重启并且将升级包更新到模组中,更新成功后,会再次走到这里 -- 再次走到这里后,合宙iot平台会返回“已经是最新版本,不需要升级”,fota_cb回调函数中会发布消息"FOTA_END" -- 至此,才会继续向下执行代码,进入PSM+模式 sys.waitUntil("FOTA_END", 15000) log.info("升级结束,进入PSM模式") -- 关闭gps备电以及gsensor供电使能,防止休眠模式下漏电导致功耗增加 gpio.close(24) -- 定时检查升级 (每4小时唤醒一次) pm.dtimerStart(2, 4 * 3600000) -- 启动飞行模式,规避可能会出现的网络问题 mobile.flymode(0, true) -- 进入PSM模式 pm.power(pm.WORK_MODE, 3) -- 防御机制:15秒后如果未进入PSM则重启 sys.wait(15000) log.info("进入PSM+失败,重启") rtos.reboot() end sys.taskInit(psm_fota_task_func)

升级前:设备会在开机的时候去请求下配置的 url 的升级包,定期唤醒检查是否有升级任务

在此过程中模组会保持唤醒状态并等待升级包下载成功并校验通过,以及检查版本是否一致。如果版本不一致或升级包检验没通过则进入休眠等待下次唤醒检查升级包。没问题的话升级包下载成功后会重启并升级,升级流程如下:

升级后:升级完成后设备显示新的版本号,并重新去请求升级包,没有新的升级包则会进入 PSM 状态

七、常见问题与注意事项

7.1 注意事项:

1、版本号格式:使用 IoT 平台时,项目的 VERSION 必须为 xxx.yyy.zzz 的三段数字格式(如 "001.000.001"),否则平台版本比对可能出错。

2、PRODUCT_KEY:使用IoT 平台时,必须在 main.lua 中正确定义全局变量 PRODUCT_KEY,其值需从 IoT 平台的项目中获取。

3、重启时机:下载升级包成功(result 为 0)后,通常需要调用 rtos.reboot() 重启设备以更新。你可以根据需要延迟重启。

4、自建服务器规则

需要升级时,服务器应返回 HTTP 200,消息体为升级文件内容。

无需升级时,服务器应返回 HTTP 300 或以上的状态码。

5、固件类型:使用合宙 IoT 平台进行脚本升级时,使用 Luatools 生成的 .bin 量产文件。

7.2 为什么升级后我的模块没有任何反应了,像是变砖一样

有多种可能,

7.2.1 检查脚本

首先先检查下用户自己的脚本,有可能是引起重启/死机的代码写在了最前面,例如新加的某个值或者函数为 nil 但是还是去做了些加减乘除或者判断大小的逻辑。可以直接本地烧录下新版本的 core+ 脚本验证,如果有 fskv 等用到 flash 的代码,可能需要仔细检查才能排除问题,比如下载的时候勾选如下图所示的两个选项。

7.2.2 检查 core

如果是仅脚本升级,但是没注意使用了新 core 中才有的接口,就有可能引起循环重启,如果重启在代码最开头,模块可能来不及打印任何日志就重启了,可以直接本地烧录下新版本的 core+ 脚本验证,如果有 fskv 等用到 flash 的代码,可能需要仔细检查。

7.3 检查过脚本和 core,没问题,为什么会循环升级 6 次以后禁止升级

检查下升级包是否正常,有时候因为人员误操作,经常会出现旧脚本 + 新 core 或者新脚本 + 旧 core 的意外组合,

例如:

本来应该如下表描述的一样

操作人员失误后变成了如下

然后误操作旧版本(1) 和误操作新版本(1)进行差分,这样虽然脚本版本号旧版本大于了新版本,但是 core 的旧版本小于新版本,所以升级平台依旧认为是依次有效的升级,下发了升级包。

升级完成后,模块内部脚本版本号变成了 001.000.000 core 版本号为 V2008,下次模块请求升级的时候,当前固件上报的脚本版本号(001.000.000)依旧小于云平台存储的脚本版本号(001.000.001),然后继续下发升级包,就这么循环 6 次,然后触发 iot 平台的禁止升级规则

在正确生成差分包,并且上传成功后,可以在 iot 平台里解除禁止升级的限制

在"我的设备"中选择升级 imei 所在的项目,然后点击右边的"解除禁止升级",

确定“导致设备循环升级的异常”已经处理完成后,点击确定解除,即可解除限制升级

7.4 我想在服务器发送特定的字符串如"update"时再触发升级,应该怎么做

只需要在你希望的升级升级逻辑后面加上升级语句即可,例如 mqtt 的 demo 里增加几句话

elseif event == "recv" then libfota2 = require "libfota2" log.info("mqtt", "downlink", "topic", data, "payload", payload) --假定mqtt发过来的字符串为"update"就启动升级 if payload =="update" then libfota2.request(fota_cb, ota_opts) end sys.publish("mqtt_payload", data, payload)

又或者希望按键升级:

--这里假定使用GPIO0进行按键升级 gpio.debounce(0, 3000, 1) gpio.setup(0, function() libfota2.request(fota_cb, ota_opts) end, gpio.PULLUP)

7.5 如何处理同个项目外面有多个版本设备的升级情况

7.5.1 场景1:多种不同内核固件版本都要升级为最新版本内核固件+最新脚本

需要对每个版本都生成对应的差分包操作步骤: 现有A1、A2、A3三种内核固件需升级为B+新脚本: - 生成3个升级包:A1→B差分包+新脚本A2→B差分包+新脚本A3→B差分包+新脚本- 在合宙IoT平台创建3个升级配置,分别对应三类设备IMEI

7.5.2 场景2:多种不同内核固件版本+不同版本脚本都要升级为最新版本脚本,既只升级脚本。

操作步骤: 1. 准备完整的最新脚本(无需内核差分包) 2. 在IoT平台创建一个升级配置,指定所有需升级设备IMEI列表

7.5.3 升级规则说明
  • 内核固件:仅支持差分升级
  • 脚本:支持全量升级,可一次性完成

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

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

相关文章

/var/run/php-fpm.sock = 127.0.0.1:9000?

不,/var/run/php-fpm.sock ≠ 127.0.0.1:9000。 二者是 PHP-FPM 与 Web 服务器通信的两种不同传输机制: /var/run/php-fpm.sock 是 Unix Domain Socket(UDS)127.0.0.1:9000 是 TCP Socket 虽然功能相同(传递 FastCGI 请…

13.3GB《ArcGIS Pro 地理信息系统应用与实践》配套练习数据

前段时间有朋友问,哪里有《ArcGIS Pro 地理信息系统应用与实践》这本书的配套练习数据? 其实,我们早前也买过这本书,但此书确实没有附光盘,也没有附上配套练习数据的下载地址。 当我们拿到这本书的之后,也…

事件驱动架构设计思路和笔记

事件驱动架构完整学习指南 作者: smj 日期: 2025-01-20 版本: V1.0 项目: HCMS-WMS 机器人管理模块 V3.0📚 目录核心概念三大核心类详解完整工作流程代码实战示例设计模式深入性能优化要点常见问题FAQ扩展知识�…

2026年苏州正规灵活用工企业排行榜推荐

2026年苏州正规灵活用工企业排行榜推荐行业痛点分析当前,灵活用工领域面临诸多技术挑战。一方面,企业在招聘和管理临时工、合同工时,往往难以实现高效匹配和精细化管理;另一方面,结算流程复杂且存在合规风险。数据显示…

基于springboot的少数民族交流论坛

运行效果:https://lunwen.yeel.cn/view.php?id=5808 基于springboot的少数民族交流论坛摘要:随着互联网技术的快速发展,我国少数民族交流日益频繁,传统的交流方式已无法满足现代交流需求。本文以Spring Boot框架为…

基于SpringBoot的5种签到打卡设计思路及实现方案

签到打卡的多样性需求 在我们的日常开发工作中,经常会遇到各种签到打卡的需求:日常签到:用户每天签到获取积分奖励 活动签到:线下活动参与者扫码签到 考勤打卡:员工上下班打卡记录 位置打卡:基于地理位置的打卡签…

WX-0813 AI 降噪回音消除模组

一、产品概述WX-0813 是一款高集成度一体化语音处理模组,专为解决音频设备开发中的噪音干扰、回音叠加、功放适配三大核心痛点设计。模组集成 AI 降噪(ENC)、全双工回音消除(AEC)、双声道 5W 数字功放三大功能&#xf…

伊朗离我们的距离,竟然比上海飞深圳还近!

提到伊朗,不少人觉得它是遥远中东的神秘国度,却不知它与中国的距离远比想象中更近。 中国新疆塔什库尔干县到伊朗最东边境仅1200多公里,比上海飞深圳的距离还要短。 当下的伊朗正面临内忧外患的动荡局势,但这份动荡从未蔓延至中…

【收藏】提示词工程:大模型应用的核心技术,小白程序员必学的提效秘籍

提示词工程是打通大模型应用落地的核心技术,更是程序员、职场人与大模型高效对话的“通关密钥”。想要让大模型精准完成任务,而非输出“答非所问”的内容,就必须掌握科学设计提示词的方法——用清晰的指令、完整的上下文引导模型,…

昆明市英语雅思培训辅导机构推荐,2026权威出国雅思课程中心学校口碑排行榜 - 苏木2025

随着留学热度持续攀升,雅思考试已成为昆明学子通往世界名校的重要门槛,然而在雅思培训选课过程中,多数考生深陷“优质机构难甄别、提分技巧难掌握、个性化方案缺失”的困境。据云南省教育国际交流协会数据显示,202…

免费的考试软件哪个好?实测指南与避坑技巧

无论是学生自测复盘、教师布置随堂测验,还是企业开展基础内训考核,很多人都想找到一款好用的免费考试软件,却屡屡踩坑:要么功能残缺,连主观题都无法支持;要么弹窗广告泛滥,严重干扰答题节奏;要么稳定性极差,答题…

全球主流双转盘高内涵成像分析系统品牌有哪些?一文掌握双转盘共聚焦高内涵分析系统怎么选择 - 品牌推荐大师1

高内涵成像分析系统是一种结合了高分辨率成像和多参数分析技术的先进工具,广泛应用于生命科学研究、药物开发和细胞生物学等领域。随着技术的不断进步,双转盘共聚焦高内涵分析系统因其高速度、高灵敏度和低光毒性的特…

广西北部湾,将成为未来最繁忙的航运地!

作为西部陆海新通道的关键枢纽,广西北部湾港已经迈入到了千万标箱的大港行列。 由钦州、北海、防城三大港口组成的北部湾港,正通过智能化、自动化转型升级,为这条国际大通道持续注入强劲动力。 2025年截至12月30日,班列开行量达…

深度可分离卷积:轻量化模型与FPGA加速的黄金技术

在深度学习与硬件加速的交叉领域,“轻量化”与“高性能”始终是一对核心矛盾。尤其是在FPGA、嵌入式单片机等资源有限的平台上,传统卷积神经网络的庞大计算量和参数量往往成为落地阻碍。而深度可分离卷积(Depthwise…

2025年国内知名的防雨柜供应商排行榜,智能控制台/卡口监控杆/控制台定做/室外防雨箱/消防中心控制台厂家排行榜 - 品牌推荐师

行业背景:防雨柜市场进入精细化竞争阶段 随着工业4.0与智慧城市建设的推进,防雨柜作为保障户外设备稳定运行的核心装备,市场需求呈现爆发式增长。据行业白皮书统计,2024年国内防雨柜市场规模突破120亿元,年复合增…

医用离心机哪个厂商产品安全性高,安信实验仪器值得关注 - 工业品牌热点

在医疗诊断、生物科研与药物研发的核心环节中,医用离心机作为样本分离提纯的关键设备,其安全性、可靠性与适配性直接关系到实验数据准确性与临床诊断效率。面对市场上良莠不齐的医用离心机产品,医疗机构与科研单位如…

昆明市英语雅思培训辅导机构推荐、2026权威出国雅思课程中心学校口碑排行榜 - 苏木2025

随着昆明国际化进程的加速,雅思考试已成为众多学子出国深造的必经之路,然而在雅思培训市场中,考生普遍面临诸多困境:优质教育机构筛选困难、选课缺乏科学指引、提分技巧掌握不足、个性化备考方案缺失,如何在繁杂的…

2026年电力服务企业甄选:电力维保 / 设计 / 电气试验 / 检修 / 工程实力榜单 - 深度智识库

在新型电力系统建设全面提速、“双碳” 目标持续深化的 2026 年,电力维保、电力设计、电气试验、电力检修、电力工程的服务品质,直接决定着工业生产、市政建设、新能源项目等各类用电场景的安全稳定与节能增效水平。…

秦岭为脊,黄河为脉,陕西是中华文明的根!

陕西是中华文明的重要发祥地之一,文化积淀深厚,历史上先后有十四个朝代在此建都,留下丰富的文物古迹。 多年来,陕西先后发掘遗址千余处,出土了大量珍贵文物和科研标本,既有百万年前的人类遗迹,…