luat_ota.c 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. #include "luat_base.h"
  2. #include "luat_ota.h"
  3. #include "luat_fs.h"
  4. #include "luat_timer.h"
  5. #include "luat_malloc.h"
  6. #include "luat_flash.h"
  7. #define LUAT_LOG_TAG "ota"
  8. #include "luat_log.h"
  9. int luat_bin_unpack(const char* path, int writeOut);
  10. int luat_ota_need_update(void) {
  11. return (luat_fs_fexist(UPDATE_BIN_PATH)) ? 1 : 0;
  12. }
  13. int luat_ota_need_rollback(void) {
  14. return (luat_fs_fexist(ROLLBACK_MARK_PATH)) ? 1 : 0;
  15. }
  16. int luat_ota_mark_rollback(void) {
  17. // 既然是异常退出,那肯定出错了!!!
  18. // 如果升级过, 那么就写入标志文件
  19. {
  20. if (luat_fs_fexist(UPDATE_MARK)) {
  21. FILE* fd = luat_fs_fopen("/rollback_mark", "wb");
  22. if (fd) {
  23. luat_fs_fclose(fd);
  24. }
  25. }
  26. else {
  27. // 没升级过, 那就是线刷, 不存在回滚
  28. }
  29. }
  30. return 0;
  31. }
  32. static void luat_bin_exec_update(void);
  33. static void luat_bin_exec_rollback(void);
  34. LUAT_WEAK void luat_ota_reboot(int timeout_ms) {
  35. if (timeout_ms > 0)
  36. luat_timer_mdelay(timeout_ms);
  37. luat_os_reboot(1);
  38. }
  39. void luat_ota_exec_update(void) {
  40. #if LUAT_OTA_MODE == 1
  41. luat_bin_exec_update();
  42. #elif LUAT_OTA_MODE == 2
  43. luat_db_exec_update();
  44. #endif
  45. }
  46. void luat_ota_exec_rollback(void) {
  47. #if LUAT_OTA_MODE == 1
  48. luat_bin_exec_rollback();
  49. #elif LUAT_OTA_MODE == 2
  50. luat_db_exec_rollback();
  51. #endif
  52. }
  53. int luat_ota_update_or_rollback(void) {
  54. if (luat_ota_need_update()) {
  55. luat_ota_exec_update();
  56. LLOGW("update: reboot at 5 secs");
  57. luat_ota_reboot(5000);
  58. return 1;
  59. }
  60. if (luat_ota_need_rollback()) {
  61. luat_ota_exec_rollback();
  62. LLOGW("rollback: reboot at 5 secs");
  63. luat_ota_reboot(5000);
  64. return 1;
  65. }
  66. return 0;
  67. }
  68. static void luat_bin_exec_update(void) {
  69. // 找到了, 检查一下大小
  70. LLOGI("found " UPDATE_BIN_PATH " ...");
  71. size_t fsize = luat_fs_fsize(UPDATE_BIN_PATH);
  72. if (fsize < 256) {
  73. // 太小了, 肯定不合法, 直接移除, 正常启动
  74. LLOGW(UPDATE_BIN_PATH " is too small, delete it");
  75. luat_fs_remove(UPDATE_BIN_PATH);
  76. return;
  77. }
  78. // 写入标志文件.
  79. // 必须提前写入, 即使解包失败, 仍标记为升级过,这样报错就能回滚
  80. LLOGI("write " UPDATE_MARK " first");
  81. FILE* fd = luat_fs_fopen(UPDATE_MARK, "wb");
  82. if (fd) {
  83. luat_fs_fclose(fd);
  84. // TODO 连标志文件都写入失败,怎么办?
  85. }
  86. // 检测升级包合法性
  87. if (luat_bin_unpack(UPDATE_BIN_PATH, 0) != LUA_OK) {
  88. LLOGE("%s is invaild!!", UPDATE_BIN_PATH);
  89. }
  90. else {
  91. // 开始解包升级文件
  92. if (luat_bin_unpack(UPDATE_BIN_PATH, 1) == LUA_OK) {
  93. LLOGI("update OK, remove " UPDATE_BIN_PATH);
  94. }
  95. else {
  96. LLOGW("update FAIL, remove " UPDATE_BIN_PATH);
  97. }
  98. }
  99. // 无论是否成功,都一定要删除升级文件, 防止升级死循环
  100. luat_fs_remove(UPDATE_BIN_PATH);
  101. }
  102. static void luat_bin_exec_rollback(void) {
  103. // 回滚文件存在,
  104. LLOGW("Found " ROLLBACK_MARK_PATH ", check rollback");
  105. // 首先,移除回滚标志, 防止重复N次的回滚
  106. luat_fs_remove("/rollback_mark"); // TODO 如果删除也失败呢?
  107. // 然后检查原始文件, flashx.bin
  108. if (!luat_fs_fexist(FLASHX_PATH)) {
  109. LLOGW("NOT " FLASHX_PATH " , can't rollback");
  110. return;
  111. }
  112. // 存在原始flashx.bin
  113. LLOGD("found " FLASHX_PATH ", unpack it");
  114. // 开始回滚操作
  115. if (luat_bin_unpack(FLASHX_PATH, 1) == LUA_OK) {
  116. LLOGI("rollback complete!");
  117. }
  118. else {
  119. LLOGE("rollback FAIL");
  120. }
  121. // 执行完成, 准备重启
  122. LLOGW("rollback: reboot at 5 secs");
  123. // 延迟5秒后,重启
  124. luat_timer_mdelay(5*1000);
  125. luat_os_reboot(0); // 重启
  126. }
  127. #include "luat_crypto.h"
  128. #include "luat_md5.h"
  129. #ifdef LUAT_USE_CRYPTO
  130. #include "mbedtls/md5.h"
  131. #undef luat_md5_init
  132. #undef luat_md5_update
  133. #undef luat_md5_finalize
  134. #if MBEDTLS_VERSION_NUMBER >= 0x03000000
  135. #define luat_md5_init mbedtls_md5_init
  136. #define luat_md5_starts mbedtls_md5_starts
  137. #define luat_md5_update mbedtls_md5_update
  138. #define luat_md5_finalize mbedtls_md5_finish
  139. #else
  140. #define luat_md5_init mbedtls_md5_init
  141. #define luat_md5_starts mbedtls_md5_starts_ret
  142. #define luat_md5_update mbedtls_md5_update_ret
  143. #define luat_md5_finalize mbedtls_md5_finish_ret
  144. #endif
  145. #endif
  146. void luat_str_fromhex(char* str, size_t len, char* buff);
  147. void luat_str_tohex(char* str, size_t len, char* buff);
  148. #define OTA_CHECK_BUFF_SIZE (64) // 超过64字节的话, luat_md5报错, 待查
  149. typedef struct ota_md5
  150. {
  151. uint8_t buff[OTA_CHECK_BUFF_SIZE];
  152. #ifdef LUAT_USE_CRYPTO
  153. mbedtls_md5_context context;
  154. #else
  155. struct md5_context context;
  156. #endif
  157. struct md5_digest digest;
  158. }ota_md5_t;
  159. int luat_ota_checkfile(const char* path) {
  160. int ret = 0;
  161. FILE * fd = luat_fs_fopen(path, "rb");
  162. if (fd == NULL) {
  163. LLOGE("no such file");
  164. return -1;
  165. }
  166. size_t binsize = luat_fs_fsize(path);
  167. if (binsize < 512 || binsize > 1024*1024) {
  168. luat_fs_fclose(fd);
  169. LLOGE("%s is too small/big %d", path, binsize);
  170. return -1;
  171. }
  172. ota_md5_t* ota = luat_heap_malloc(sizeof(ota_md5_t));
  173. if (ota == NULL) {
  174. luat_fs_fclose(fd);
  175. LLOGE("out of memory when check ota file md5");
  176. return -1;
  177. }
  178. unsigned int len = 0;
  179. int remain = binsize - 16;
  180. luat_md5_init(&ota->context);
  181. #ifdef LUAT_USE_CRYPTO
  182. luat_md5_starts(&ota->context);
  183. #endif
  184. while (remain > 0) {
  185. if (remain > OTA_CHECK_BUFF_SIZE) {
  186. len = luat_fs_fread(ota->buff, OTA_CHECK_BUFF_SIZE, 1, fd);
  187. }
  188. else {
  189. len = luat_fs_fread(ota->buff, remain, 1, fd);
  190. }
  191. //LLOGD("ota read %d byte", len);
  192. if (len == 0) {
  193. continue;
  194. }
  195. if (len < 0 || len > 512) {
  196. luat_heap_free(ota);
  197. luat_fs_fclose(fd);
  198. LLOGE("read file fail");
  199. return -1;
  200. }
  201. remain -= len;
  202. luat_md5_update(&ota->context, ota->buff, len);
  203. }
  204. luat_md5_finalize(&ota->context, &ota->digest);
  205. #ifdef LUAT_USE_CRYPTO
  206. mbedtls_md5_free(&ota->context);
  207. #endif
  208. // 应该还有16字节的md5
  209. //memset(ota->buff, 0, OTA_CHECK_BUFF_SIZE);
  210. luat_fs_fread(ota->buff, 16, 1, fd);
  211. // 读完就可以关了
  212. luat_fs_fclose(fd);
  213. // 判断一下md5
  214. // uint8_t *expect_md5 = ota->buff + 64;
  215. // uint8_t *face_md5 = ota->buff + 128;
  216. if (!memcmp(ota->buff, ota->digest.bytes, 16)) {
  217. LLOGD("ota file MD5 ok");
  218. ret = 0;
  219. }
  220. else {
  221. LLOGE("ota file MD5 FAIL");
  222. ret = -1;
  223. }
  224. luat_heap_free(ota);
  225. return ret;
  226. }
  227. #ifdef LUAT_USE_OTA
  228. int luat_ota(uint32_t luadb_addr){
  229. #ifdef LUAT_USE_ZLIB
  230. FILE *fd_out = NULL;
  231. FILE *fd_in = NULL;
  232. extern int zlib_decompress(FILE *source, FILE *dest);
  233. //检测是否有压缩升级文件
  234. if(luat_fs_fexist(UPDATE_TGZ_PATH)){
  235. LLOGI("found update.tgz, decompress ...");
  236. fd_in = luat_fs_fopen(UPDATE_TGZ_PATH, "r");
  237. if (fd_in == NULL){
  238. LLOGE("open the input file : %s error!", UPDATE_TGZ_PATH);
  239. goto _close_decompress;
  240. }
  241. luat_fs_remove(UPDATE_BIN_PATH);
  242. fd_out = luat_fs_fopen(UPDATE_BIN_PATH, "w+");
  243. if (fd_out == NULL){
  244. LLOGE("open the output file : %s error!", UPDATE_BIN_PATH);
  245. goto _close_decompress;
  246. }
  247. int ret = zlib_decompress(fd_in, fd_out);
  248. if (ret != 0){
  249. LLOGE("decompress file error!");
  250. }
  251. _close_decompress:
  252. if(fd_in != NULL){
  253. luat_fs_fclose(fd_in);
  254. }
  255. if(fd_out != NULL){
  256. luat_fs_fclose(fd_out);
  257. }
  258. //不论成功与否都删掉避免每次启动都执行一遍
  259. luat_fs_remove(UPDATE_TGZ_PATH);
  260. }
  261. #endif
  262. int ret = -1;
  263. //检测是否有升级文件
  264. if(luat_fs_fexist(UPDATE_BIN_PATH)){
  265. LLOGI("found update.bin, checking");
  266. if (luat_ota_checkfile(UPDATE_BIN_PATH) == 0) {
  267. LLOGI("update.bin ok, updating...");
  268. #define UPDATE_BUFF_SIZE 4096
  269. uint8_t* buff = luat_heap_malloc(UPDATE_BUFF_SIZE);
  270. int len = 0;
  271. int offset = 0;
  272. if (buff != NULL) {
  273. FILE* fd = luat_fs_fopen(UPDATE_BIN_PATH, "rb");
  274. if (fd){
  275. while (1) {
  276. memset(buff, 0, UPDATE_BUFF_SIZE);
  277. len = luat_fs_fread(buff, sizeof(uint8_t), UPDATE_BUFF_SIZE, fd);
  278. if (len < 1)
  279. break;
  280. luat_flash_erase(luadb_addr + offset, UPDATE_BUFF_SIZE);
  281. luat_flash_write(luadb_addr + offset, buff, UPDATE_BUFF_SIZE);
  282. offset += len;
  283. }
  284. }else{
  285. ret = -1;
  286. LLOGW("update.bin open error");
  287. }
  288. ret = 0;
  289. }
  290. }
  291. else {
  292. ret = -1;
  293. LLOGW("update.bin NOT ok, skip");
  294. }
  295. luat_fs_remove(UPDATE_BIN_PATH);
  296. }
  297. return ret;
  298. }
  299. #endif