diff --git a/services/samgr/native/include/schedule/system_ability_state_context.h b/services/samgr/native/include/schedule/system_ability_state_context.h index 88fb3503c842398cf8dcab2a235469b3542339d3..f185920ccf2c4ec2e02dac7ac1abbd89b1133744 100644 --- a/services/samgr/native/include/schedule/system_ability_state_context.h +++ b/services/samgr/native/include/schedule/system_ability_state_context.h @@ -86,6 +86,7 @@ struct SystemAbilityContext { std::shared_ptr unloadRequest; int32_t systemAbilityId = -1; int32_t delayUnloadTime = 0; + int64_t lastStartTime = 0; SystemAbilityState state = SystemAbilityState::NOT_LOADED; PendingEvent pendingEvent = PendingEvent::NO_EVENT; bool isAutoRestart = false; diff --git a/services/samgr/native/include/schedule/system_ability_state_scheduler.h b/services/samgr/native/include/schedule/system_ability_state_scheduler.h index a1c4646254f13ca53f7bba4d9fc343fa48e4c018..dee24b721fe73d09d51c7d2dbc2848e4cd48612a 100644 --- a/services/samgr/native/include/schedule/system_ability_state_scheduler.h +++ b/services/samgr/native/include/schedule/system_ability_state_scheduler.h @@ -65,6 +65,7 @@ public: int32_t CheckStartEnableOnce(const OnDemandEvent& event, const SaControlInfo& saControl, sptr callback); int32_t CheckStopEnableOnce(const OnDemandEvent& event, const SaControlInfo& saControl); + void UpdateLimitDelayUnloadTime(int32_t systemAbilityId); private: void InitStateContext(const std::list& saProfiles); diff --git a/services/samgr/native/source/schedule/system_ability_state_scheduler.cpp b/services/samgr/native/source/schedule/system_ability_state_scheduler.cpp index 7e1e8125edc46ad17bc9a150007aeaf56c8f6c0b..801d2ca4263e0a54fc37244c591771a25638a942 100644 --- a/services/samgr/native/source/schedule/system_ability_state_scheduler.cpp +++ b/services/samgr/native/source/schedule/system_ability_state_scheduler.cpp @@ -39,6 +39,8 @@ constexpr int32_t MAX_SUBSCRIBE_COUNT = 256; constexpr int32_t UNLOAD_TIMEOUT_TIME = 5 * 1000; constexpr const char* LOCAL_DEVICE = "local"; constexpr int32_t MAX_DELAY_TIME = 5 * 60 * 1000; +constexpr int32_t MAX_DURATION = 10 * 60 * 1000; // ms +constexpr int32_t ONCE_DELAY_TIME = 10 * 1000; // ms constexpr const char* CANCEL_UNLOAD = "cancelUnload"; constexpr const char* KEY_EVENT_ID = "eventId"; constexpr const char* KEY_NAME = "name"; @@ -173,6 +175,26 @@ bool SystemAbilityStateScheduler::GetSystemAbilityContext(int32_t systemAbilityI return true; } +void SystemAbilityStateScheduler::UpdateLimitDelayUnloadTime(int32_t systemAbilityId) +{ + std::shared_ptr abilityContext; + if (!GetSystemAbilityContext(systemAbilityId, abilityContext)) { + return; + } + std::lock_guard autoLock(abilityContext->ownProcessContext->processLock); + if (abilityContext->lastStartTime != 0) { + int64_t begin = abilityContext->lastStartTime; + int64_t end = GetTickCount(); + if (end - begin <= MAX_DURATION) { + int64_t onceDelayTime = abilityContext->delayUnloadTime; + onceDelayTime += ONCE_DELAY_TIME; + abilityContext->delayUnloadTime = LimitDelayUnloadTime(onceDelayTime); + HILOGI("DelayUnloadTime is %{public}d, SA:%{public}d", abilityContext->delayUnloadTime, systemAbilityId); + } + } + abilityContext->lastStartTime = GetTickCount(); +} + bool SystemAbilityStateScheduler::GetSystemProcessContext(const std::u16string& processName, std::shared_ptr& processContext) { diff --git a/services/samgr/native/source/system_ability_manager.cpp b/services/samgr/native/source/system_ability_manager.cpp index bb93a68f642de454614b625067391ad5cdc3de04..1e1aea5b33a87b6025d292c2e0f8d5d3d124bab8 100644 --- a/services/samgr/native/source/system_ability_manager.cpp +++ b/services/samgr/native/source/system_ability_manager.cpp @@ -1065,6 +1065,7 @@ int32_t SystemAbilityManager::AddSystemAbility(int32_t systemAbilityId, const sp HILOGE("abilityStateScheduler is nullptr"); return ERR_INVALID_VALUE; } + abilityStateScheduler_->UpdateLimitDelayUnloadTime(systemAbilityId); SystemAbilityInvalidateCache(systemAbilityId); abilityStateScheduler_->SendAbilityStateEvent(systemAbilityId, AbilityStateEvent::ABILITY_LOAD_SUCCESS_EVENT); SendSystemAbilityAddedMsg(systemAbilityId, ability); diff --git a/services/samgr/native/test/unittest/src/system_ability_state_scheduler_test.cpp b/services/samgr/native/test/unittest/src/system_ability_state_scheduler_test.cpp index 5f461ffeca78df0d942f71a71ba92bc5d3780af2..5fb9d916d726c568e7dbc85c65651e018ac083fb 100644 --- a/services/samgr/native/test/unittest/src/system_ability_state_scheduler_test.cpp +++ b/services/samgr/native/test/unittest/src/system_ability_state_scheduler_test.cpp @@ -44,6 +44,8 @@ constexpr int32_t MAX_DELAY_TIME_TEST = 5 * 60 * 1000; constexpr int32_t BEYOND_DELAY_TIME_TEST = 5 * 60 * 1000 + 1; constexpr int32_t TEST_SYSTEM_ABILITY1 = 1491; constexpr int32_t TEST_SYSTEM_ABILITY2 = 1492; +constexpr int32_t ONCE_DELAY_TIME = 10 * 1000; //ms +constexpr int32_t MAX_DURATION = 11 * 60 * 1000; // ms const std::u16string process = u"test"; const std::u16string process_invalid = u"test_invalid"; const std::string LOCAL_DEVICE = "local"; @@ -2454,6 +2456,75 @@ HWTEST_F(SystemAbilityStateSchedulerTest, LimitDelayUnloadTime002, TestSize.Leve EXPECT_EQ(ret, MAX_DELAY_TIME_TEST); } +/** + * @tc.name: UpdateLimitDelayUnloadTime001 + * @tc.desc: test UpdateLimitDelayUnloadTime with saId invalid + * @tc.type: FUNC + * @tc.require: I6FDNZ + */ + +HWTEST_F(SystemAbilityStateSchedulerTest, UpdateLimitDelayUnloadTime001, TestSize.Level3) +{ + std::shared_ptr systemAbilityStateScheduler = + std::make_shared(); + std::shared_ptr abilityContext = std::make_shared(); + abilityContext->systemAbilityId = -1; + abilityContext->lastStartTime = 0; + systemAbilityStateScheduler->UpdateLimitDelayUnloadTime(abilityContext->systemAbilityId); + EXPECT_EQ(abilityContext->lastStartTime, 0); +} + +/** + * @tc.name: UpdateLimitDelayUnloadTime002 + * @tc.desc: test UpdateLimitDelayUnloadTime with lastStartTime not equal to 0 + * @tc.type: FUNC + * @tc.require: I6FDNZ + */ + +HWTEST_F(SystemAbilityStateSchedulerTest, UpdateLimitDelayUnloadTime002, TestSize.Level3) +{ + std::shared_ptr systemAbilityStateScheduler = + std::make_shared(); + int said = 401; + std::list saProfiles; + systemAbilityStateScheduler->Init(saProfiles); + std::shared_ptr abilityContext = std::make_shared(); + std::shared_ptr systemProcessContext = std::make_shared(); + systemAbilityStateScheduler->abilityContextMap_.clear(); + abilityContext->ownProcessContext = systemProcessContext; + systemAbilityStateScheduler->abilityContextMap_[said] = abilityContext; + abilityContext->lastStartTime = GetTickCount(); + int32_t delaytime = abilityContext->delayUnloadTime; + systemAbilityStateScheduler->UpdateLimitDelayUnloadTime(said); + delaytime += ONCE_DELAY_TIME; + EXPECT_EQ(delaytime, abilityContext->delayUnloadTime); +} + +/** + * @tc.name: UpdateLimitDelayUnloadTime002 + * @tc.desc: test UpdateLimitDelayUnloadTime with duation greater than duration + * @tc.type: FUNC + * @tc.require: I6FDNZ + */ + +HWTEST_F(SystemAbilityStateSchedulerTest, UpdateLimitDelayUnloadTime003, TestSize.Level3) +{ + std::shared_ptr systemAbilityStateScheduler = + std::make_shared(); + int said = 401; + std::list saProfiles; + systemAbilityStateScheduler->Init(saProfiles); + std::shared_ptr abilityContext = std::make_shared(); + std::shared_ptr systemProcessContext = std::make_shared(); + systemAbilityStateScheduler->abilityContextMap_.clear(); + abilityContext->ownProcessContext = systemProcessContext; + systemAbilityStateScheduler->abilityContextMap_[said] = abilityContext; + abilityContext->lastStartTime = GetTickCount() - MAX_DURATION; + int32_t delaytime = abilityContext->delayUnloadTime; + systemAbilityStateScheduler->UpdateLimitDelayUnloadTime(said); + EXPECT_EQ(delaytime, abilityContext->delayUnloadTime); +} + /** * @tc.name: PostUnloadTimeoutTask001 * @tc.desc: test PostUnloadTimeoutTask with state is SystemProcessState::STOPPING