• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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