1 /*
2 * Copyright (c) 2024 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 "ability_cache_manager.h"
17
18 #include "hilog_tag_wrapper.h"
19
20 namespace OHOS {
21 namespace AAFwk {
22 const std::string FRS_APP_INDEX = "ohos.extra.param.key.frs_index";
23 const std::string FRS_BUNDLE_NAME = "com.ohos.formrenderservice";
24
AbilityCacheManager()25 AbilityCacheManager::AbilityCacheManager() {}
26
~AbilityCacheManager()27 AbilityCacheManager::~AbilityCacheManager() {}
28
GetInstance()29 AbilityCacheManager &AbilityCacheManager::GetInstance()
30 {
31 static AbilityCacheManager abilityRecMgr;
32 return abilityRecMgr;
33 }
34
Init(uint32_t devCapacity,uint32_t procCapacity)35 void AbilityCacheManager::Init(uint32_t devCapacity, uint32_t procCapacity)
36 {
37 devLruCapacity_ = devCapacity;
38 procLruCapacity_ = procCapacity;
39 }
40
RemoveAbilityRecInDevList(std::shared_ptr<AbilityRecord> abilityRecord)41 void AbilityCacheManager::RemoveAbilityRecInDevList(std::shared_ptr<AbilityRecord> abilityRecord)
42 {
43 if (abilityRecord == nullptr) {
44 return;
45 }
46 auto it = devRecLru_.begin();
47 uint32_t accessTokenId = abilityRecord->GetApplicationInfo().accessTokenId;
48 while (it != devRecLru_.end()) {
49 if ((*it) && (*it)->GetRecordId() == abilityRecord->GetRecordId()) {
50 devRecLru_.erase(it);
51 devLruCnt_--;
52 return;
53 } else {
54 it++;
55 }
56 }
57 }
58
RemoveAbilityRecInProcList(std::shared_ptr<AbilityRecord> abilityRecord)59 void AbilityCacheManager::RemoveAbilityRecInProcList(std::shared_ptr<AbilityRecord> abilityRecord)
60 {
61 if (abilityRecord == nullptr) {
62 return;
63 }
64 uint32_t accessTokenId = abilityRecord->GetApplicationInfo().accessTokenId;
65 auto findProcInfo = procLruMap_.find(accessTokenId);
66 if (findProcInfo == procLruMap_.end()) {
67 TAG_LOGE(AAFwkTag::SERVICE_EXT, "not find record");
68 return;
69 }
70 auto it = findProcInfo->second.recList.begin();
71
72 while (it != findProcInfo->second.recList.end()) {
73 if ((*it) && (*it)->GetRecordId() == abilityRecord->GetRecordId()) {
74 findProcInfo->second.recList.erase(it);
75 findProcInfo->second.cnt--;
76 if (findProcInfo->second.cnt == 0) {
77 procLruMap_.erase(findProcInfo);
78 }
79 return;
80 } else {
81 it++;
82 }
83 }
84 }
85
AddToProcLru(std::shared_ptr<AbilityRecord> abilityRecord)86 std::shared_ptr<AbilityRecord> AbilityCacheManager::AddToProcLru(std::shared_ptr<AbilityRecord> abilityRecord)
87 {
88 if (abilityRecord == nullptr) {
89 return nullptr;
90 }
91 auto findProcInfo = procLruMap_.find(abilityRecord->GetApplicationInfo().accessTokenId);
92 if (findProcInfo == procLruMap_.end()) {
93 std::list<std::shared_ptr<AbilityRecord>> recList;
94 ProcRecordsInfo procRecInfo = {recList, 1};
95 procRecInfo.recList.push_back(abilityRecord);
96 procLruMap_[abilityRecord->GetApplicationInfo().accessTokenId] = procRecInfo;
97 return nullptr;
98 }
99 if (findProcInfo->second.cnt == procLruCapacity_) {
100 RemoveAbilityRecInDevList(findProcInfo->second.recList.front());
101 std::shared_ptr<AbilityRecord> rec = findProcInfo->second.recList.front();
102 findProcInfo->second.recList.pop_front();
103 findProcInfo->second.recList.push_back(abilityRecord);
104 return rec;
105 }
106 findProcInfo->second.cnt++;
107 findProcInfo->second.recList.push_back(abilityRecord);
108 return nullptr;
109 }
110
AddToDevLru(std::shared_ptr<AbilityRecord> abilityRecord,std::shared_ptr<AbilityRecord> rec)111 std::shared_ptr<AbilityRecord> AbilityCacheManager::AddToDevLru(std::shared_ptr<AbilityRecord> abilityRecord,
112 std::shared_ptr<AbilityRecord> rec)
113 {
114 if (rec != nullptr) {
115 devRecLru_.push_back(abilityRecord);
116 devLruCnt_++;
117 return rec;
118 }
119 if (devLruCnt_ == devLruCapacity_ && devLruCnt_ > 0) {
120 rec = devRecLru_.front();
121 RemoveAbilityRecInProcList(rec);
122 devRecLru_.pop_front();
123 devLruCnt_--;
124 }
125 devRecLru_.push_back(abilityRecord);
126 devLruCnt_++;
127 return rec;
128 }
129
Put(std::shared_ptr<AbilityRecord> abilityRecord)130 std::shared_ptr<AbilityRecord> AbilityCacheManager::Put(std::shared_ptr<AbilityRecord> abilityRecord)
131 {
132 if (abilityRecord == nullptr) {
133 TAG_LOGE(AAFwkTag::SERVICE_EXT, "null abilityRecord");
134 return nullptr;
135 }
136 TAG_LOGD(AAFwkTag::SERVICE_EXT, "Put the ability to lru, service:%{public}s, extension type %{public}d",
137 abilityRecord->GetURI().c_str(), abilityRecord->GetAbilityInfo().extensionAbilityType);
138 std::lock_guard<std::mutex> lock(mutex_);
139 std::shared_ptr<AbilityRecord> rec = AddToProcLru(abilityRecord);
140 return AddToDevLru(abilityRecord, rec);
141 }
142
Remove(std::shared_ptr<AbilityRecord> abilityRecord)143 void AbilityCacheManager::Remove(std::shared_ptr<AbilityRecord> abilityRecord)
144 {
145 if (abilityRecord == nullptr) {
146 TAG_LOGE(AAFwkTag::SERVICE_EXT, "null abilityRecord");
147 return;
148 }
149 TAG_LOGD(AAFwkTag::SERVICE_EXT, "Remove the ability from lru, service:%{public}s, extension type %{public}d",
150 abilityRecord->GetURI().c_str(), abilityRecord->GetAbilityInfo().extensionAbilityType);
151 std::lock_guard<std::mutex> lock(mutex_);
152 RemoveAbilityRecInProcList(abilityRecord);
153 RemoveAbilityRecInDevList(abilityRecord);
154 }
155
IsRecInfoSame(const AbilityRequest & abilityRequest,std::shared_ptr<AbilityRecord> abilityRecord)156 bool AbilityCacheManager::IsRecInfoSame(const AbilityRequest& abilityRequest,
157 std::shared_ptr<AbilityRecord> abilityRecord)
158 {
159 return abilityRecord != nullptr &&
160 abilityRequest.abilityInfo.moduleName == abilityRecord->GetAbilityInfo().moduleName &&
161 abilityRequest.want.GetElement().GetAbilityName() == abilityRecord->GetWant().GetElement().GetAbilityName();
162 }
163
GetAbilityRecInProcList(const AbilityRequest & abilityRequest)164 std::shared_ptr<AbilityRecord> AbilityCacheManager::GetAbilityRecInProcList(const AbilityRequest &abilityRequest)
165 {
166 auto findProcInfo = procLruMap_.find(abilityRequest.appInfo.accessTokenId);
167 if (findProcInfo == procLruMap_.end()) {
168 TAG_LOGE(AAFwkTag::SERVICE_EXT, "can't found bundleName");
169 return nullptr;
170 }
171 ProcRecordsInfo &procRecordsInfo = findProcInfo->second;
172 auto recIter = procRecordsInfo.recList.begin();
173 while (recIter != procRecordsInfo.recList.end()) {
174 if (IsRecInfoSame(abilityRequest, *recIter)) {
175 std::shared_ptr<AbilityRecord> abilityRecord = *recIter;
176 procRecordsInfo.recList.erase(recIter);
177 procRecordsInfo.cnt--;
178 return abilityRecord;
179 }
180 recIter++;
181 }
182 TAG_LOGD(AAFwkTag::SERVICE_EXT, "Can't found the abilityRecord in process list for get.");
183 return nullptr;
184 }
185
Get(const AbilityRequest & abilityRequest)186 std::shared_ptr<AbilityRecord> AbilityCacheManager::Get(const AbilityRequest& abilityRequest)
187 {
188 TAG_LOGD(AAFwkTag::SERVICE_EXT, "Get the ability from lru, service:%{public}s, extension type %{public}d",
189 abilityRequest.abilityInfo.uri.c_str(), abilityRequest.abilityInfo.extensionAbilityType);
190 std::lock_guard<std::mutex> lock(mutex_);
191 std::shared_ptr<AbilityRecord> abilityRecord = GetAbilityRecInProcList(abilityRequest);
192 if (abilityRecord == nullptr) {
193 TAG_LOGD(AAFwkTag::SERVICE_EXT, "Can't found the abilityRecord for get.");
194 return nullptr;
195 }
196 RemoveAbilityRecInDevList(abilityRecord);
197 return abilityRecord;
198 }
199
FindRecordByToken(const sptr<IRemoteObject> & token)200 std::shared_ptr<AbilityRecord> AbilityCacheManager::FindRecordByToken(const sptr<IRemoteObject> &token)
201 {
202 if (token == nullptr) {
203 TAG_LOGE(AAFwkTag::SERVICE_EXT, "null token");
204 return nullptr;
205 }
206 std::lock_guard<std::mutex> lock(mutex_);
207 auto it = devRecLru_.begin();
208 while (it != devRecLru_.end()) {
209 if (*it) {
210 sptr<IRemoteObject> srcToken = (*it)->GetToken();
211 if (srcToken == token) {
212 std::shared_ptr<AbilityRecord> &abilityRecord = *it;
213 TAG_LOGD(AAFwkTag::SERVICE_EXT,
214 "Find the ability by token from lru, service:%{public}s, extension type %{public}d",
215 abilityRecord->GetURI().c_str(), abilityRecord->GetAbilityInfo().extensionAbilityType);
216 return abilityRecord;
217 }
218 }
219 it++;
220 }
221 return nullptr;
222 }
223
GetAbilityList()224 std::list<std::shared_ptr<AbilityRecord>> AbilityCacheManager::GetAbilityList()
225 {
226 std::lock_guard<std::mutex> lock(mutex_);
227 return devRecLru_;
228 }
229
FindRecordBySessionId(const std::string & assertSessionId)230 std::shared_ptr<AbilityRecord> AbilityCacheManager::FindRecordBySessionId(const std::string &assertSessionId)
231 {
232 std::lock_guard<std::mutex> lock(mutex_);
233 auto it = devRecLru_.begin();
234 while (it != devRecLru_.end()) {
235 if (!*it) {
236 it++;
237 continue;
238 }
239 auto assertSessionStr = (*it)->GetWant().GetStringParam(Want::PARAM_ASSERT_FAULT_SESSION_ID);
240 if (assertSessionStr == assertSessionId) {
241 std::shared_ptr<AbilityRecord> &abilityRecord = *it;
242 TAG_LOGD(AAFwkTag::SERVICE_EXT,
243 "Find the ability by sessionId from lru, service:%{public}s, extension type %{public}d",
244 abilityRecord->GetURI().c_str(), abilityRecord->GetAbilityInfo().extensionAbilityType);
245 return abilityRecord;
246 } else {
247 it++;
248 }
249 }
250 return nullptr;
251 }
252
FindRecordByServiceKey(const std::string & serviceKey)253 std::shared_ptr<AbilityRecord> AbilityCacheManager::FindRecordByServiceKey(const std::string &serviceKey)
254 {
255 std::lock_guard<std::mutex> lock(mutex_);
256 auto it = devRecLru_.begin();
257 while (it != devRecLru_.end()) {
258 if (!*it) {
259 it++;
260 continue;
261 }
262 std::string curServiceKey = (*it)->GetURI();
263 if (FRS_BUNDLE_NAME == (*it)->GetAbilityInfo().bundleName) {
264 curServiceKey = curServiceKey + std::to_string((*it)->GetWant().GetIntParam(FRS_APP_INDEX, 0));
265 }
266 if (curServiceKey.compare(serviceKey) == 0) {
267 std::shared_ptr<AbilityRecord> &abilityRecord = *it;
268 TAG_LOGD(AAFwkTag::SERVICE_EXT,
269 "Find the ability by serviceKey from lru, service:%{public}s, extension type %{public}d",
270 abilityRecord->GetURI().c_str(), abilityRecord->GetAbilityInfo().extensionAbilityType);
271 return abilityRecord;
272 } else {
273 it++;
274 }
275 }
276 return nullptr;
277 }
278
RemoveLauncherDeathRecipient()279 void AbilityCacheManager::RemoveLauncherDeathRecipient()
280 {
281 std::lock_guard<std::mutex> lock(mutex_);
282 auto it = devRecLru_.begin();
283 while (it != devRecLru_.end()) {
284 auto targetExtension = *it;
285 if (targetExtension != nullptr && targetExtension->GetAbilityInfo().type == AbilityType::EXTENSION &&
286 ((targetExtension->GetAbilityInfo().name == AbilityConfig::LAUNCHER_ABILITY_NAME &&
287 targetExtension->GetAbilityInfo().bundleName == AbilityConfig::LAUNCHER_BUNDLE_NAME) ||
288 targetExtension->IsSceneBoard())) {
289 targetExtension->RemoveAbilityDeathRecipient();
290 return;
291 }
292 it++;
293 }
294 }
295
SignRestartAppFlag(int32_t uid,const std::string & instanceKey)296 void AbilityCacheManager::SignRestartAppFlag(int32_t uid, const std::string &instanceKey)
297 {
298 std::lock_guard<std::mutex> lock(mutex_);
299 auto it = devRecLru_.begin();
300 while (it != devRecLru_.end()) {
301 auto abilityRecord = *it;
302 if (abilityRecord != nullptr && abilityRecord->GetUid() == uid &&
303 abilityRecord->GetInstanceKey() == instanceKey) {
304 abilityRecord->SetRestartAppFlag(true);
305 }
306 it++;
307 }
308 }
309
DeleteInvalidServiceRecord(const std::string & bundleName)310 void AbilityCacheManager::DeleteInvalidServiceRecord(const std::string &bundleName)
311 {
312 std::lock_guard<std::mutex> lock(mutex_);
313 auto it = devRecLru_.begin();
314 while (it != devRecLru_.end()) {
315 auto abilityRecord = *it;
316 if (abilityRecord != nullptr && abilityRecord->GetApplicationInfo().bundleName == bundleName) {
317 RemoveAbilityRecInProcList(abilityRecord);
318 RemoveAbilityRecInDevList(abilityRecord);
319 }
320 it++;
321 }
322 }
323 } // namespace AAFwk
324 } // namespace OHOS