• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2021-2023 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 #include "updater_main.h"
16 #include <chrono>
17 #include <dirent.h>
18 #include <fcntl.h>
19 #include <getopt.h>
20 #include <libgen.h>
21 #include <string>
22 #include <sys/mount.h>
23 #include <sys/reboot.h>
24 #include <sys/stat.h>
25 #include <sys/statvfs.h>
26 #include <sys/syscall.h>
27 #include <thread>
28 #include <unistd.h>
29 #include <vector>
30 #include "applypatch/partition_record.h"
31 #include "cert_verify.h"
32 #include "flashd/flashd.h"
33 #include "fs_manager/mount.h"
34 #include "include/updater/updater.h"
35 #include "json_node.h"
36 #include "language/language_ui.h"
37 #include "log/dump.h"
38 #include "log/log.h"
39 #include "misc_info/misc_info.h"
40 #include "package/pkg_manager.h"
41 #include "pkg_manager.h"
42 #include "pkg_utils.h"
43 #include "ptable_parse/ptable_process.h"
44 #include "sdcard_update/sdcard_update.h"
45 #include "securec.h"
46 #include "updater/updater_const.h"
47 #include "updater/hwfault_retry.h"
48 #include "updater/updater_preprocess.h"
49 #include "updater_ui_stub.h"
50 #include "utils.h"
51 #include "factory_reset/factory_reset.h"
52 #include "write_state/write_state.h"
53 
54 namespace Updater {
55 using Utils::String2Int;
56 using namespace Hpackage;
57 using namespace Updater::Utils;
58 using namespace std::literals::chrono_literals;
59 
60 [[maybe_unused]] constexpr int DISPLAY_TIME = 1000 * 1000;
61 constexpr struct option OPTIONS[] = {
62     { "update_package", required_argument, nullptr, 0 },
63     { "retry_count", required_argument, nullptr, 0 },
64     { "factory_wipe_data", no_argument, nullptr, 0 },
65     { "user_wipe_data", no_argument, nullptr, 0 },
66     { "sdcard_update", no_argument, nullptr, 0 },
67     { "upgraded_pkg_num", required_argument, nullptr, 0 },
68     { "force_update_action", required_argument, nullptr, 0 },
69     { "night_update", no_argument, nullptr, 0 },
70     { USB_MODE, no_argument, nullptr, 0 },
71     { "UPDATE:MAINIMG", no_argument, nullptr, 0 },
72     { "UPDATE:SD", no_argument, nullptr, 0 },
73     { nullptr, 0, nullptr, 0 },
74 };
75 constexpr float VERIFY_PERCENT = 0.05;
76 
FactoryReset(FactoryResetMode mode,const std::string & path)77 int FactoryReset(FactoryResetMode mode, const std::string &path)
78 {
79     UpdaterInit::GetInstance().InvokeEvent(FACTORY_RESET_INIT_EVENT);
80     auto ret = FactoryResetProcess::GetInstance().FactoryResetFunc(mode, path);
81     if (ret == 0) {
82         LOG(INFO) << "restorecon for " << path;
83         RestoreconPath(path); // restore selinux context for /data after factory reset success
84     }
85     return ret;
86 }
87 
OtaUpdatePreCheck(PkgManager::PkgManagerPtr pkgManager,const std::string & packagePath)88 static int OtaUpdatePreCheck(PkgManager::PkgManagerPtr pkgManager, const std::string &packagePath)
89 {
90     UPDATER_INIT_RECORD;
91     if (pkgManager == nullptr) {
92         LOG(ERROR) << "pkgManager is nullptr";
93         UPDATER_LAST_WORD(PKG_INVALID_FILE);
94         return UPDATE_CORRUPT;
95     }
96     char realPath[PATH_MAX + 1] = {0};
97     if (realpath(packagePath.c_str(), realPath) == nullptr) {
98         LOG(ERROR) << "realpath error";
99         UPDATER_LAST_WORD(PKG_INVALID_FILE);
100         return PKG_INVALID_FILE;
101     }
102     if (access(realPath, F_OK) != 0) {
103         LOG(ERROR) << "package does not exist!";
104         UPDATER_LAST_WORD(PKG_INVALID_FILE);
105         return PKG_INVALID_FILE;
106     }
107 
108     int32_t ret = pkgManager->VerifyOtaPackage(packagePath);
109     if (ret != PKG_SUCCESS) {
110         LOG(INFO) << "VerifyOtaPackage fail ret :"<< ret;
111         UPDATER_LAST_WORD(ret);
112         return ret;
113     }
114 
115     return PKG_SUCCESS;
116 }
117 
UpdatePreCheck(UpdaterParams & upParams,const std::string pkgPath)118 static UpdaterStatus UpdatePreCheck(UpdaterParams &upParams, const std::string pkgPath)
119 {
120     if (PreProcess::GetInstance().DoUpdateAuth(pkgPath) != 0) {
121         UPDATER_LAST_WORD(UPDATE_ERROR);
122         return UPDATE_ERROR;
123     }
124 
125     PkgManager::PkgManagerPtr pkgManager = PkgManager::CreatePackageInstance();
126     if (GetUpdatePackageInfo(pkgManager, pkgPath) != PKG_SUCCESS) {
127         PkgManager::ReleasePackageInstance(pkgManager);
128         LOG(ERROR) << "Verify update bin file Fail!";
129         UPDATER_LAST_WORD(UPDATE_ERROR);
130         return UPDATE_ERROR;
131     }
132     if (PreProcess::GetInstance().DoUpdatePreProcess(upParams, pkgManager) != PKG_SUCCESS) {
133         PkgManager::ReleasePackageInstance(pkgManager);
134         LOG(ERROR) << "Version Check Fail!";
135         UPDATER_LAST_WORD(UPDATE_ERROR);
136         return UPDATE_ERROR;
137     }
138     PkgManager::ReleasePackageInstance(pkgManager);
139     return UPDATE_SUCCESS;
140 }
141 
VerifyPackages(UpdaterParams & upParams)142 static UpdaterStatus VerifyPackages(UpdaterParams &upParams)
143 {
144     LOG(INFO) << "Verify packages start...";
145     UPDATER_UI_INSTANCE.ShowProgressPage();
146     UPDATER_UI_INSTANCE.ShowUpdInfo(TR(UPD_VERIFYPKG));
147 
148     if (upParams.callbackProgress == nullptr) {
149         UPDATER_UI_INSTANCE.ShowUpdInfo(TR(UPD_VERIFYPKGFAIL), true);
150         UPDATER_LAST_WORD(UPDATE_CORRUPT);
151         return UPDATE_CORRUPT;
152     }
153     upParams.callbackProgress(0.0);
154     upParams.installTime.resize(upParams.updatePackage.size(), std::chrono::duration<double>(0));
155     ReadInstallTime(upParams);
156     for (unsigned int i = upParams.pkgLocation; i < upParams.updatePackage.size(); i++) {
157         LOG(INFO) << "Verify package:" << upParams.updatePackage[i];
158         auto startTime = std::chrono::system_clock::now();
159         PkgManager::PkgManagerPtr manager = PkgManager::CreatePackageInstance();
160         int32_t verifyret = OtaUpdatePreCheck(manager, upParams.updatePackage[i]);
161         PkgManager::ReleasePackageInstance(manager);
162 
163         if (verifyret != PKG_SUCCESS || UpdatePreCheck(upParams, upParams.updatePackage[i]) != UPDATE_SUCCESS) {
164             upParams.pkgLocation = i;
165             UPDATER_UI_INSTANCE.ShowUpdInfo(TR(UPD_VERIFYPKGFAIL), true);
166             auto endTime = std::chrono::system_clock::now();
167             upParams.installTime[i] = endTime - startTime;
168             UPDATER_LAST_WORD(UPDATE_CORRUPT);
169             return UPDATE_CORRUPT;
170         }
171         auto endTime = std::chrono::system_clock::now();
172         upParams.installTime[i] = endTime - startTime;
173     }
174 
175     ProgressSmoothHandler(0, static_cast<int>(VERIFY_PERCENT * FULL_PERCENT_PROGRESS));
176     LOG(INFO) << "Verify packages successfull...";
177     return UPDATE_SUCCESS;
178 }
179 
GetBatteryCapacity(int & capacity)180 bool GetBatteryCapacity(int &capacity)
181 {
182     const static std::vector<const char *> vec = {
183         "/sys/class/power_supply/battery/capacity",
184         "/sys/class/power_supply/Battery/capacity"
185     };
186     for (auto &it : vec) {
187         std::ifstream ifs { it };
188         if (!ifs.is_open()) {
189             continue;
190         }
191 
192         int tmpCapacity = 0;
193         ifs >> tmpCapacity;
194         if ((ifs.fail()) || (ifs.bad())) {
195             continue;
196         }
197 
198         capacity = tmpCapacity;
199         return true;
200     }
201 
202     return false;
203 }
204 
IsBatteryCapacitySufficient()205 bool IsBatteryCapacitySufficient()
206 {
207     if (Utils::CheckUpdateMode(OTA_MODE)) {
208         LOG(INFO) << "this is OTA update, on need to determine the battery";
209         return true;
210     }
211     static constexpr auto levelIdx = "lowBatteryLevel";
212     static constexpr auto jsonPath = "/etc/product_cfg.json";
213 
214     int capacity = 0;
215     bool ret = GetBatteryCapacity(capacity);
216     if (!ret) {
217         return true; /* maybe no battery or err value return default true */
218     }
219 
220     JsonNode node { Fs::path { jsonPath }};
221     auto item = node[levelIdx].As<int>();
222     if (!item.has_value()) {
223         return true; /* maybe no value return default true */
224     }
225 
226     int lowLevel = *item;
227     if (lowLevel > 100 || lowLevel < 0) { /* full percent is 100 */
228         LOG(ERROR) << "load battery level error:" << lowLevel;
229         return false; /* config err not allow to update */
230     }
231 
232     LOG(INFO) << "current capacity:" << capacity << ", low level:" << lowLevel;
233 
234     return capacity >= lowLevel;
235 }
236 
InstallUpdaterPackage(UpdaterParams & upParams,PkgManager::PkgManagerPtr manager)237 UpdaterStatus InstallUpdaterPackage(UpdaterParams &upParams, PkgManager::PkgManagerPtr manager)
238 {
239     UpdaterStatus status = UPDATE_UNKNOWN;
240     STAGE(UPDATE_STAGE_BEGIN) << "Install package";
241     if (upParams.retryCount == 0) {
242         // First time enter updater, record retryCount in case of abnormal reset.
243         if (!PartitionRecord::GetInstance().ClearRecordPartitionOffset()) {
244             LOG(ERROR) << "ClearRecordPartitionOffset failed";
245             UPDATER_LAST_WORD(UPDATE_ERROR);
246             return UPDATE_ERROR;
247         }
248         SetMessageToMisc(upParams.miscCmd, upParams.retryCount + 1, "retry_count");
249     }
250     if (upParams.updateMode == SDCARD_UPDATE) {
251         status = DoInstallUpdaterPackage(manager, upParams, SDCARD_UPDATE);
252     } else {
253         status = DoInstallUpdaterPackage(manager, upParams, HOTA_UPDATE);
254     }
255     if (status != UPDATE_SUCCESS) {
256         UPDATER_UI_INSTANCE.Sleep(UI_SHOW_DURATION);
257         UPDATER_UI_INSTANCE.ShowLog(TR(LOG_UPDFAIL));
258         STAGE(UPDATE_STAGE_FAIL) << "Install failed";
259         if (status == UPDATE_RETRY) {
260             HwFaultRetry::GetInstance().DoRetryAction();
261             UPDATER_UI_INSTANCE.ShowFailedPage();
262         }
263     } else {
264         LOG(INFO) << "Install package success.";
265         STAGE(UPDATE_STAGE_SUCCESS) << "Install package";
266     }
267     return status;
268 }
269 
CalcProgress(const UpdaterParams & upParams,std::vector<double> & pkgStartPosition,double & updateStartPosition)270 static UpdaterStatus CalcProgress(const UpdaterParams &upParams,
271     std::vector<double> &pkgStartPosition, double &updateStartPosition)
272 {
273     UPDATER_INIT_RECORD;
274     int64_t allPkgSize = 0;
275     std::vector<int64_t> everyPkgSize;
276     if (upParams.pkgLocation == upParams.updatePackage.size()) {
277         updateStartPosition = VERIFY_PERCENT;
278         return UPDATE_SUCCESS;
279     }
280     for (const auto &path : upParams.updatePackage) {
281         char realPath[PATH_MAX + 1] = {0};
282         if (realpath(path.c_str(), realPath) == nullptr) {
283             LOG(WARNING) << "Can not find updatePackage : " << path;
284             everyPkgSize.push_back(0);
285             continue;
286         }
287         struct stat st {};
288         if (stat(realPath, &st) == 0) {
289             everyPkgSize.push_back(st.st_size);
290             allPkgSize += st.st_size;
291             LOG(INFO) << "pkg " << path << " size is:" << st.st_size;
292         }
293     }
294     pkgStartPosition.push_back(VERIFY_PERCENT);
295     if (allPkgSize == 0) {
296         LOG(ERROR) << "All packages's size is 0.";
297         UPDATER_LAST_WORD(UPDATE_ERROR);
298         return UPDATE_ERROR;
299     }
300     int64_t startSize = 0;
301     for (auto size : everyPkgSize) {
302         startSize += size;
303         float percent = static_cast<double>(startSize) / static_cast<double>(allPkgSize) + VERIFY_PERCENT;
304         percent = (percent > 1.0) ? 1.0 : percent; // 1.0 : 100%
305         LOG(INFO) << "percent is:" << percent;
306         pkgStartPosition.push_back(percent);
307     }
308 
309     updateStartPosition = pkgStartPosition[upParams.pkgLocation];
310     return UPDATE_SUCCESS;
311 }
312 
PreUpdatePackages(UpdaterParams & upParams)313 static UpdaterStatus PreUpdatePackages(UpdaterParams &upParams)
314 {
315     UPDATER_INIT_RECORD;
316     LOG(INFO) << "start to update packages, start index:" << upParams.pkgLocation;
317 
318     UpdaterStatus status = UPDATE_UNKNOWN;
319 
320     upParams.installTime.resize(upParams.updatePackage.size(), std::chrono::duration<double>(0));
321     if (SetupPartitions() != 0) {
322         UPDATER_UI_INSTANCE.ShowUpdInfo(TR(UPD_SETPART_FAIL), true);
323         UPDATER_LAST_WORD(UPDATE_ERROR);
324         return UPDATE_ERROR;
325     }
326     const std::string resultPath = std::string(UPDATER_PATH) + "/" + std::string(UPDATER_RESULT_FILE);
327     if (access(resultPath.c_str(), F_OK) != -1) {
328         (void)DeleteFile(resultPath);
329         LOG(INFO) << "delete last upgrade file";
330     }
331 
332     if(upParams.pkgLocation == upParams.updatePackage.size()) {
333         LOG(WARNING) << "all package has been upgraded, skip pre process";
334         return UPDATE_SUCCESS;
335     }
336 
337     // verify packages first
338     if (VerifyPackages(upParams) != UPDATE_SUCCESS) {
339         UPDATER_LAST_WORD(UPDATE_CORRUPT);
340         return UPDATE_CORRUPT; // verify package failed must return UPDATE_CORRUPT, ux need it !!!
341     }
342 
343     // Only handle UPATE_ERROR and UPDATE_SUCCESS here.Let package verify handle others.
344     if (IsSpaceCapacitySufficient(upParams) == UPDATE_ERROR) {
345         UPDATER_LAST_WORD(status);
346         return status;
347     }
348     if (upParams.retryCount == 0 && !IsBatteryCapacitySufficient()) {
349         UPDATER_UI_INSTANCE.ShowUpdInfo(TR(LOG_LOWPOWER));
350         UPDATER_UI_INSTANCE.Sleep(UI_SHOW_DURATION);
351         UPDATER_LAST_WORD(UPDATE_ERROR);
352         LOG(ERROR) << "Battery is not sufficient for install package.";
353         return UPDATE_SKIP;
354     }
355 
356 #ifdef UPDATER_USE_PTABLE
357     if (!PtablePreProcess::GetInstance().DoPtableProcess(upParams)) {
358         LOG(ERROR) << "DoPtableProcess failed";
359         UPDATER_LAST_WORD(UPDATE_ERROR);
360         return UPDATE_ERROR;
361     }
362 #endif
363     return UPDATE_SUCCESS;
364 }
365 
DoInstallPackages(UpdaterParams & upParams,std::vector<double> & pkgStartPosition)366 static UpdaterStatus DoInstallPackages(UpdaterParams &upParams, std::vector<double> &pkgStartPosition)
367 {
368     UpdaterStatus status = UPDATE_UNKNOWN;
369     if (upParams.pkgLocation == upParams.updatePackage.size()) {
370         LOG(WARNING) << "all packages has been installed, directly return success";
371         upParams.callbackProgress(FULL_PERCENT_PROGRESS);
372         return UPDATE_SUCCESS;
373     }
374     for (; upParams.pkgLocation < upParams.updatePackage.size(); upParams.pkgLocation++) {
375         PkgManager::PkgManagerPtr manager = PkgManager::CreatePackageInstance();
376         auto startTime = std::chrono::system_clock::now();
377         upParams.currentPercentage = pkgStartPosition[upParams.pkgLocation + 1] -
378             pkgStartPosition[upParams.pkgLocation];
379         upParams.initialProgress = pkgStartPosition[upParams.pkgLocation];
380         LOG(INFO) << "InstallUpdaterPackage pkg is " << upParams.updatePackage[upParams.pkgLocation] <<
381             " percent:" << upParams.initialProgress << "~" << pkgStartPosition[upParams.pkgLocation + 1];
382 
383         status = InstallUpdaterPackage(upParams, manager);
384         auto endTime = std::chrono::system_clock::now();
385         upParams.installTime[upParams.pkgLocation] = upParams.installTime[upParams.pkgLocation] + endTime - startTime;
386         WriteInstallTime(upParams);
387         if (status != UPDATE_SUCCESS) {
388             LOG(ERROR) << "InstallUpdaterPackage failed! Pkg is " << upParams.updatePackage[upParams.pkgLocation];
389             if (!CheckResultFail()) {
390                 UPDATER_LAST_WORD(status);
391             }
392             PkgManager::ReleasePackageInstance(manager);
393             return status;
394         }
395         ProgressSmoothHandler(
396             static_cast<int>(upParams.initialProgress * FULL_PERCENT_PROGRESS +
397             upParams.currentPercentage * GetTmpProgressValue()),
398             static_cast<int>(pkgStartPosition[upParams.pkgLocation + 1] * FULL_PERCENT_PROGRESS));
399         SetMessageToMisc(upParams.miscCmd, upParams.pkgLocation + 1, "upgraded_pkg_num");
400         PkgManager::ReleasePackageInstance(manager);
401     }
402     return status;
403 }
404 
DoUpdatePackages(UpdaterParams & upParams)405 UpdaterStatus DoUpdatePackages(UpdaterParams &upParams)
406 {
407     UPDATER_INIT_RECORD;
408     UpdaterStatus status = UPDATE_UNKNOWN;
409     std::vector<double> pkgStartPosition {};
410     double updateStartPosition = 0.0;
411     status = CalcProgress(upParams, pkgStartPosition, updateStartPosition);
412     if (status != UPDATE_SUCCESS) {
413         UPDATER_LAST_WORD(status);
414         return status;
415     }
416     for (unsigned int i = 0; i < upParams.updatePackage.size(); i++) {
417         LOG(INFO) << "package " << i << ":" << upParams.updatePackage[i] <<
418             " precent:" << upParams.currentPercentage;
419     }
420     if (upParams.callbackProgress == nullptr) {
421         LOG(ERROR) << "CallbackProgress is nullptr";
422         return UPDATE_CORRUPT;
423     }
424     upParams.callbackProgress(updateStartPosition * FULL_PERCENT_PROGRESS);
425     status = DoInstallPackages(upParams, pkgStartPosition);
426     if (status != UPDATE_SUCCESS) {
427         UPDATER_LAST_WORD(status);
428         return status;
429     }
430     if (upParams.forceUpdate) {
431         UPDATER_UI_INSTANCE.ShowLogRes(TR(LABEL_UPD_OK_SHUTDOWN));
432     }
433     UPDATER_UI_INSTANCE.ShowSuccessPage();
434     return status;
435 }
436 
PostUpdatePackages(UpdaterParams & upParams,bool updateResult)437 static void PostUpdatePackages(UpdaterParams &upParams, bool updateResult)
438 {
439     std::string writeBuffer;
440     std::string buf;
441     std::string time;
442     if (!updateResult) {
443         const std::string resultPath = std::string(UPDATER_PATH) + "/" + std::string(UPDATER_RESULT_FILE);
444         std::ifstream fin {resultPath};
445         if (!fin.is_open() || !std::getline(fin, buf)) {
446             LOG(ERROR) << "read result file error " << resultPath;
447             buf = "fail|";
448         }
449     } else {
450         buf = "pass|";
451         upParams.pkgLocation = upParams.pkgLocation == 0 ? upParams.pkgLocation : (upParams.pkgLocation - 1);
452     }
453 
454     for (unsigned int i = 0; i < upParams.pkgLocation; i++) {
455         time = DurationToString(upParams.installTime, i);
456         writeBuffer += upParams.updatePackage[i] + "|pass||install_time=" + time + "|\n";
457     }
458     time = DurationToString(upParams.installTime, upParams.pkgLocation);
459 
460     writeBuffer += upParams.updatePackage[upParams.pkgLocation] + "|" + buf + "|install_time=" + time + "|\n";
461     for (unsigned int i = upParams.pkgLocation + 1; i < upParams.updatePackage.size(); i++) {
462         writeBuffer += upParams.updatePackage[i] + "\n";
463     }
464     if (writeBuffer != "") {
465         writeBuffer.pop_back();
466     }
467     LOG(INFO) << "post over, writeBuffer = " << writeBuffer;
468     WriteDumpResult(writeBuffer);
469     DeleteInstallTimeFile();
470 }
471 
UpdaterFromSdcard(UpdaterParams & upParams)472 UpdaterStatus UpdaterFromSdcard(UpdaterParams &upParams)
473 {
474     upParams.callbackProgress = [] (float value) { UPDATER_UI_INSTANCE.ShowProgress(value); };
475     SetMessageToMisc(upParams.miscCmd, 0, "sdcard_update");
476     if (CheckSdcardPkgs(upParams) != UPDATE_SUCCESS) {
477         LOG(ERROR) << "can not find sdcard packages";
478         return UPDATE_ERROR;
479     }
480     upParams.installTime.resize(upParams.updatePackage.size(), std::chrono::duration<double>(0));
481     // verify packages first
482     if (upParams.retryCount == 0 && !IsBatteryCapacitySufficient()) {
483         UPDATER_UI_INSTANCE.ShowUpdInfo(TR(LOG_LOWPOWER));
484         UPDATER_UI_INSTANCE.Sleep(UI_SHOW_DURATION);
485         LOG(ERROR) << "Battery is not sufficient for install package.";
486         return UPDATE_SKIP;
487     }
488 
489     if (VerifyPackages(upParams) != UPDATE_SUCCESS) {
490         return UPDATE_CORRUPT; // verify package failed must return UPDATE_CORRUPT, ux need it !!!
491     }
492 #ifdef UPDATER_USE_PTABLE
493     if (!PtablePreProcess::GetInstance().DoPtableProcess(upParams)) {
494         LOG(ERROR) << "DoPtableProcess failed";
495         return UPDATE_ERROR;
496     }
497 #endif
498     upParams.initialProgress += VERIFY_PERCENT;
499     upParams.currentPercentage -= VERIFY_PERCENT;
500 
501     STAGE(UPDATE_STAGE_BEGIN) << "UpdaterFromSdcard";
502     LOG(INFO) << "UpdaterFromSdcard start, sdcard updaterPath : " << upParams.updatePackage[upParams.pkgLocation];
503     UPDATER_UI_INSTANCE.ShowLog(TR(LOG_SDCARD_NOTMOVE));
504     return DoUpdatePackages(upParams);
505 }
506 
InstallUpdaterPackages(UpdaterParams & upParams)507 UpdaterStatus InstallUpdaterPackages(UpdaterParams &upParams)
508 {
509     UpdaterStatus status = PreUpdatePackages(upParams);
510     if (status == UPDATE_SUCCESS) {
511         status = DoUpdatePackages(upParams);
512     }
513     PostUpdatePackages(upParams, status == UPDATE_SUCCESS);
514     return status;
515 }
516 
StartUpdaterEntry(UpdaterParams & upParams)517 UpdaterStatus StartUpdaterEntry(UpdaterParams &upParams)
518 {
519     UpdaterStatus status = UPDATE_UNKNOWN;
520     status = PreStartUpdaterEntry(upParams, status);
521     if (status != UPDATE_SUCCESS) {
522         LOG(ERROR) << "PreStartUpdaterEntry failed";
523         return status;
524     }
525 
526     status = DoUpdaterEntry(upParams);
527     if (status != UPDATE_SUCCESS) {
528         LOG(WARNING) << "DoUpdaterEntry failed";
529     }
530 
531     status = PostStartUpdaterEntry(upParams, status);
532     if (status != UPDATE_SUCCESS) {
533         LOG(ERROR) << "PostStartUpdaterEntry failed";
534     }
535     return status;
536 }
537 
DoUpdaterEntry(UpdaterParams & upParams)538 UpdaterStatus DoUpdaterEntry(UpdaterParams &upParams)
539 {
540     UpdaterStatus status = UPDATE_UNKNOWN;
541     if (upParams.updateMode == SDCARD_UPDATE) {
542         LOG(INFO) << "start sdcard update";
543         UPDATER_UI_INSTANCE.ShowProgressPage();
544         status = UpdaterFromSdcard(upParams);
545         return status;
546     } else if (upParams.updatePackage.size() > 0) {
547         UPDATER_UI_INSTANCE.ShowProgressPage();
548         status = InstallUpdaterPackages(upParams);
549     } else if (upParams.factoryWipeData) {
550         UPDATER_UI_INSTANCE.ShowProgressPage();
551         LOG(INFO) << "Factory level FactoryReset begin";
552         status = UPDATE_SUCCESS;
553 #if !defined(UPDATER_UT) && defined(UPDATER_UI_SUPPORT)
554         DoProgress();
555 #endif
556         if (FactoryReset(FACTORY_WIPE_DATA, "/data") != 0) {
557             LOG(ERROR) << "FactoryReset factory level failed";
558             status = UPDATE_ERROR;
559         }
560         UPDATER_UI_INSTANCE.ShowLogRes(
561             (status != UPDATE_SUCCESS) ? TR(LOGRES_FACTORY_FAIL) : TR(LOGRES_FACTORY_DONE));
562         UpdaterInit::GetInstance().InvokeEvent(UPDATER_RPMB_DATA_CLEAR_EVENT);
563     } else if (upParams.userWipeData) {
564         UPDATER_UI_INSTANCE.ShowProgressPage();
565         LOG(INFO) << "User level FactoryReset begin";
566         status = UPDATE_SUCCESS;
567 #if !defined(UPDATER_UT) && defined(UPDATER_UI_SUPPORT)
568         DoProgress();
569 #endif
570         if (FactoryReset(USER_WIPE_DATA, "/data") != 0) {
571             LOG(ERROR) << "FactoryReset user level failed";
572             status = UPDATE_ERROR;
573         }
574         if (status != UPDATE_SUCCESS) {
575             UPDATER_UI_INSTANCE.ShowLogRes(TR(LOGRES_WIPE_FAIL));
576         } else {
577             UpdaterInit::GetInstance().InvokeEvent(UPDATER_RPMB_DATA_CLEAR_EVENT);
578             UPDATER_UI_INSTANCE.ShowSuccessPage();
579             UPDATER_UI_INSTANCE.ShowLogRes(TR(LOGRES_WIPE_FINISH));
580             ClearUpdaterParaMisc();
581             std::this_thread::sleep_for(std::chrono::milliseconds(UI_SHOW_DURATION));
582         }
583     }
584     return status;
585 }
586 
InitOptionsFuncTab(char * & optarg,PackageUpdateMode & mode,UpdaterParams & upParams)587 std::unordered_map<std::string, std::function<void ()>> InitOptionsFuncTab(char* &optarg,
588     PackageUpdateMode &mode, UpdaterParams &upParams)
589 {
590     std::unordered_map<std::string, std::function<void ()>> optionsFuncTab {
591         {"update_package", [&]() -> void
592         {
593             upParams.updatePackage.push_back(optarg);
594             (void)UPDATER_UI_INSTANCE.SetMode(UPDATERMODE_OTA);
595             mode = HOTA_UPDATE;
596         }},
597         {"retry_count", [&]() -> void
598         {
599             upParams.retryCount = atoi(optarg);
600             HwFaultRetry::GetInstance().SetRetryCount(upParams.retryCount);
601         }},
602         {"factory_wipe_data", [&]() -> void
603         {
604             (void)UPDATER_UI_INSTANCE.SetMode(UPDATERMODE_REBOOTFACTORYRST);
605             upParams.factoryWipeData = true;
606             upParams.factoryReset = true;
607         }},
608         {"user_wipe_data", [&]() -> void
609         {
610             (void)UPDATER_UI_INSTANCE.SetMode(UPDATERMODE_REBOOTFACTORYRST);
611             upParams.userWipeData = true;
612         }},
613         {"upgraded_pkg_num", [&]() -> void
614         {
615             upParams.pkgLocation = static_cast<unsigned int>(atoi(optarg));
616         }},
617         {"sdcard_update", [&]() -> void
618         {
619             upParams.updateMode = SDCARD_UPDATE;
620         }},
621         {"UPDATE:MAINIMG", [&]() -> void
622         {
623             upParams.updateMode = SDCARD_UPDATE;
624             upParams.mainUpdate = true;
625         }},
626         {"UPDATE:SD", [&]() -> void
627         {
628             upParams.updateMode = SDCARD_UPDATE;
629             upParams.sdUpdate = true;
630         }},
631         {"force_update_action", [&]() -> void
632         {
633             if (std::string(optarg) == POWEROFF) {
634                 upParams.forceUpdate = true;
635             }
636         }},
637         {"night_update", [&]() -> void
638         {
639             (void)UPDATER_UI_INSTANCE.SetMode(UPDATERMODE_NIGHTUPDATE);
640             upParams.forceReboot = true;
641         }}
642     };
643     return optionsFuncTab;
644 }
645 
IsSupportOption(const std::string & option)646 __attribute__((weak)) bool IsSupportOption([[maybe_unused]] const std::string &option)
647 {
648     LOG(INFO) << "option: " << option;
649     return false;
650 }
651 
ProcessOtherOption(const std::string & option,UpdaterParams & upParams)652 __attribute__((weak)) UpdaterStatus ProcessOtherOption([[maybe_unused]] const std::string &option,
653     [[maybe_unused]] UpdaterParams &upParams)
654 {
655     return UPDATE_UNKNOWN;
656 }
657 
StartUpdater(const std::vector<std::string> & args,char ** argv,PackageUpdateMode & mode,UpdaterParams & upParams)658 static UpdaterStatus StartUpdater(const std::vector<std::string> &args,
659     char **argv, PackageUpdateMode &mode, UpdaterParams &upParams)
660 {
661     std::vector<char *> extractedArgs;
662     int rc;
663     int optionIndex;
664     auto optionsFuncTab = InitOptionsFuncTab(optarg, mode, upParams);
665 
666     for (const auto &arg : args) {
667         extractedArgs.push_back(const_cast<char *>(arg.c_str()));
668     }
669     extractedArgs.push_back(nullptr);
670     extractedArgs.insert(extractedArgs.begin(), argv[0]);
671     while ((rc = getopt_long(extractedArgs.size() - 1, extractedArgs.data(), "", OPTIONS, &optionIndex)) != -1) {
672         switch (rc) {
673             case 0: {
674                 std::string option = OPTIONS[optionIndex].name;
675                 LOG(INFO) << "option: " << option;
676                 if (optionsFuncTab.find(option) != optionsFuncTab.end()) {
677                     auto optionsFunc = optionsFuncTab.at(option);
678                     optionsFunc();
679                 } else if (IsSupportOption(option)) {
680                     return ProcessOtherOption(option, upParams);
681                 }
682                 break;
683             }
684             default:
685                 LOG(ERROR) << "Invalid argument.";
686                 break;
687         }
688     }
689     optind = 1;
690     if (upParams.pkgLocation == 0) {
691         DeleteInstallTimeFile();
692     }
693     // Sanity checks
694     if (upParams.updateMode == SDCARD_UPDATE) {
695         (void)UPDATER_UI_INSTANCE.SetMode(UPDATERMODE_SDCARD);
696         mode = SDCARD_UPDATE;
697     }
698     if (upParams.factoryWipeData && upParams.userWipeData) {
699         LOG(WARNING) << "Factory level reset and user level reset both set. use user level reset.";
700         upParams.factoryWipeData = false;
701     }
702     return StartUpdaterEntry(upParams);
703 }
704 
705 // add updater mode
706 REGISTER_MODE(Updater, "updater.hdc.configfs");
707 
UpdaterMain(int argc,char ** argv)708 int UpdaterMain(int argc, char **argv)
709 {
710     [[maybe_unused]] UpdaterStatus status = UPDATE_UNKNOWN;
711     UpdaterParams upParams;
712     upParams.callbackProgress = [] (float value) { UPDATER_UI_INSTANCE.ShowProgress(value); };
713     UpdaterInit::GetInstance().InvokeEvent(UPDATER_PRE_INIT_EVENT);
714     std::vector<std::string> args = ParseParams(argc, argv);
715 
716     LOG(INFO) << "Ready to start";
717 #if !defined(UPDATER_UT) && defined(UPDATER_UI_SUPPORT)
718     UPDATER_UI_INSTANCE.InitEnv();
719 #endif
720     UpdaterInit::GetInstance().InvokeEvent(UPDATER_INIT_EVENT);
721     PackageUpdateMode mode = UNKNOWN_UPDATE;
722     status = StartUpdater(args, argv, mode, upParams);
723 #if !defined(UPDATER_UT) && defined(UPDATER_UI_SUPPORT)
724     UPDATER_UI_INSTANCE.Sleep(UI_SHOW_DURATION);
725     if (status != UPDATE_SUCCESS && status != UPDATE_SKIP) {
726         if (mode == HOTA_UPDATE) {
727             UPDATER_UI_INSTANCE.ShowFailedPage();
728             UpdaterInit::GetInstance().InvokeEvent(UPDATER_POST_INIT_EVENT);
729             if (upParams.forceReboot) {
730                 Utils::UsSleep(5 * DISPLAY_TIME); // 5 : 5s
731                 PostUpdater(true);
732                 Utils::UpdaterDoReboot("");
733                 return 0;
734             }
735         } else if (mode == SDCARD_UPDATE) {
736             UPDATER_UI_INSTANCE.ShowLogRes(
737                 status == UPDATE_CORRUPT ? TR(LOGRES_VERIFY_FAILED) : TR(LOGRES_UPDATE_FAILED));
738             UPDATER_UI_INSTANCE.ShowFailedPage();
739             Utils::UsSleep(5 * DISPLAY_TIME); // 5 : 5s
740             UPDATER_UI_INSTANCE.ShowMainpage();
741         } else if (upParams.userWipeData || upParams.factoryWipeData) {
742             UPDATER_UI_INSTANCE.ShowFailedPage();
743         } else {
744             UPDATER_UI_INSTANCE.ShowMainpage();
745             UPDATER_UI_INSTANCE.Sleep(50); /* wait for page flush 50ms */
746             UPDATER_UI_INSTANCE.SaveScreen();
747         }
748         // Wait for user input
749         while (true) {
750             Utils::UsSleep(DISPLAY_TIME);
751         }
752         return 0;
753     }
754 #endif
755     PostUpdater(true);
756     upParams.forceUpdate || upParams.factoryReset ? Utils::DoShutdown() : Utils::UpdaterDoReboot("");
757     return 0;
758 }
759 } // Updater
760