• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2022-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 
16 #include "quick_fix_manager_client.h"
17 
18 #include "hilog_wrapper.h"
19 #include "hitrace_meter.h"
20 #include "if_system_ability_manager.h"
21 #include "iservice_registry.h"
22 #include "quick_fix_error_utils.h"
23 #include "quick_fix_load_callback.h"
24 #include "quick_fix_manager_proxy.h"
25 #include "system_ability_definition.h"
26 
27 namespace OHOS {
28 namespace AAFwk {
29 namespace {
30 const int LOAD_SA_TIMEOUT_MS = 4 * 1000;
31 } // namespace
32 
ApplyQuickFix(const std::vector<std::string> & quickFixFiles)33 int32_t QuickFixManagerClient::ApplyQuickFix(const std::vector<std::string> &quickFixFiles)
34 {
35     HITRACE_METER_NAME(HITRACE_TAG_ABILITY_MANAGER, __PRETTY_FUNCTION__);
36     HILOG_DEBUG("function called.");
37 
38     auto quickFixMgr = GetQuickFixMgrProxy();
39     if (quickFixMgr == nullptr) {
40         HILOG_ERROR("Get quick fix manager service failed.");
41         return QUICK_FIX_CONNECT_FAILED;
42     }
43 
44     return quickFixMgr->ApplyQuickFix(quickFixFiles);
45 }
46 
GetApplyedQuickFixInfo(const std::string & bundleName,ApplicationQuickFixInfo & quickFixInfo)47 int32_t QuickFixManagerClient::GetApplyedQuickFixInfo(const std::string &bundleName,
48     ApplicationQuickFixInfo &quickFixInfo)
49 {
50     HITRACE_METER_NAME(HITRACE_TAG_ABILITY_MANAGER, __PRETTY_FUNCTION__);
51     HILOG_DEBUG("function called.");
52 
53     auto quickFixMgr = GetQuickFixMgrProxy();
54     if (quickFixMgr == nullptr) {
55         HILOG_ERROR("Get quick fix manager service failed.");
56         return QUICK_FIX_CONNECT_FAILED;
57     }
58 
59     return quickFixMgr->GetApplyedQuickFixInfo(bundleName, quickFixInfo);
60 }
61 
GetQuickFixMgrProxy()62 sptr<IQuickFixManager> QuickFixManagerClient::GetQuickFixMgrProxy()
63 {
64     HILOG_DEBUG("function called.");
65     auto quickFixMgr = GetQuickFixMgr();
66     if (quickFixMgr != nullptr) {
67         HILOG_DEBUG("Quick fix manager has been started.");
68         return quickFixMgr;
69     }
70 
71     if (!LoadQuickFixMgrService()) {
72         HILOG_ERROR("Load quick fix manager service failed.");
73         return nullptr;
74     }
75 
76     quickFixMgr = GetQuickFixMgr();
77     if (quickFixMgr == nullptr || quickFixMgr->AsObject() == nullptr) {
78         HILOG_ERROR("Failed to get quick fix manager.");
79         return nullptr;
80     }
81 
82     auto self = weak_from_this();
83     const auto &onClearProxyCallback = [self](const wptr<IRemoteObject> &remote) {
84         auto impl = self.lock();
85         if (impl && impl->quickFixMgr_ == remote) {
86             impl->ClearProxy();
87         }
88     };
89 
90     sptr<QfmsDeathRecipient> recipient(new (std::nothrow) QfmsDeathRecipient(onClearProxyCallback));
91     quickFixMgr->AsObject()->AddDeathRecipient(recipient);
92 
93     HILOG_DEBUG("function finished.");
94     return quickFixMgr;
95 }
96 
RevokeQuickFix(const std::string & bundleName)97 int32_t QuickFixManagerClient::RevokeQuickFix(const std::string &bundleName)
98 {
99     HILOG_DEBUG("Function called.");
100 
101     auto quickFixMgr = GetQuickFixMgrProxy();
102     if (quickFixMgr == nullptr) {
103         HILOG_ERROR("Get quick fix manager service failed.");
104         return QUICK_FIX_CONNECT_FAILED;
105     }
106 
107     auto retval = quickFixMgr->RevokeQuickFix(bundleName);
108     HILOG_DEBUG("Function call end, retval is %{public}d.", retval);
109     return retval;
110 }
111 
ClearProxy()112 void QuickFixManagerClient::ClearProxy()
113 {
114     HILOG_DEBUG("function called.");
115     std::lock_guard<std::mutex> lock(mutex_);
116     quickFixMgr_ = nullptr;
117 }
118 
OnRemoteDied(const wptr<IRemoteObject> & remote)119 void QuickFixManagerClient::QfmsDeathRecipient::OnRemoteDied([[maybe_unused]] const wptr<IRemoteObject> &remote)
120 {
121     if (proxy_ != nullptr) {
122         HILOG_ERROR("quick fix manager service died.");
123         proxy_(remote);
124     }
125 }
126 
LoadQuickFixMgrService()127 bool QuickFixManagerClient::LoadQuickFixMgrService()
128 {
129     {
130         std::unique_lock<std::mutex> lock(loadSaMutex_);
131         loadSaFinished_ = false;
132     }
133 
134     auto systemAbilityMgr = SystemAbilityManagerClient::GetInstance().GetSystemAbilityManager();
135     if (systemAbilityMgr == nullptr) {
136         HILOG_ERROR("Failed to get SystemAbilityManager.");
137         return false;
138     }
139 
140     sptr<QuickFixLoadCallback> loadCallback = new (std::nothrow) QuickFixLoadCallback();
141     if (loadCallback == nullptr) {
142         HILOG_ERROR("Create load callback failed.");
143         return false;
144     }
145 
146     auto ret = systemAbilityMgr->LoadSystemAbility(QUICK_FIX_MGR_SERVICE_ID, loadCallback);
147     if (ret != 0) {
148         HILOG_ERROR("Load system ability %{public}d failed with %{public}d.", QUICK_FIX_MGR_SERVICE_ID, ret);
149         return false;
150     }
151 
152     {
153         std::unique_lock<std::mutex> lock(loadSaMutex_);
154         auto waitStatus = loadSaCondation_.wait_for(lock, std::chrono::milliseconds(LOAD_SA_TIMEOUT_MS),
155             [this]() {
156                 return loadSaFinished_;
157             });
158         if (!waitStatus) {
159             HILOG_ERROR("Wait for load sa timeout.");
160             return false;
161         }
162     }
163 
164     return true;
165 }
166 
SetQuickFixMgr(const sptr<IRemoteObject> & remoteObject)167 void QuickFixManagerClient::SetQuickFixMgr(const sptr<IRemoteObject> &remoteObject)
168 {
169     std::lock_guard<std::mutex> lock(mutex_);
170     quickFixMgr_ = iface_cast<IQuickFixManager>(remoteObject);
171 }
172 
GetQuickFixMgr()173 sptr<IQuickFixManager> QuickFixManagerClient::GetQuickFixMgr()
174 {
175     std::lock_guard<std::mutex> lock(mutex_);
176     return quickFixMgr_;
177 }
178 
OnLoadSystemAbilitySuccess(const sptr<IRemoteObject> & remoteObject)179 void QuickFixManagerClient::OnLoadSystemAbilitySuccess(const sptr<IRemoteObject> &remoteObject)
180 {
181     SetQuickFixMgr(remoteObject);
182     std::unique_lock<std::mutex> lock(loadSaMutex_);
183     loadSaFinished_ = true;
184     loadSaCondation_.notify_one();
185 }
186 
OnLoadSystemAbilityFail()187 void QuickFixManagerClient::OnLoadSystemAbilityFail()
188 {
189     SetQuickFixMgr(nullptr);
190     std::unique_lock<std::mutex> lock(loadSaMutex_);
191     loadSaFinished_ = true;
192     loadSaCondation_.notify_one();
193 }
194 }  // namespace AAFwk
195 }  // namespace OHOS
196