1 /*
2  * Copyright (c) 2021-2022 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 "entities/phone_entity.h"
17 
18 #include <cinttypes>
19 
20 #include "battery_stats_service.h"
21 #include "stats_log.h"
22 
23 namespace OHOS {
24 namespace PowerMgr {
25 namespace {
26 }
27 
PhoneEntity()28 PhoneEntity::PhoneEntity()
29 {
30     consumptionType_ = BatteryStatsInfo::CONSUMPTION_TYPE_PHONE;
31 }
32 
GetActiveTimeMs(StatsUtils::StatsType statsType,int16_t level)33 int64_t PhoneEntity::GetActiveTimeMs(StatsUtils::StatsType statsType, int16_t level)
34 {
35     int64_t activeTimeMs = StatsUtils::DEFAULT_VALUE;
36     switch (statsType) {
37         case StatsUtils::STATS_TYPE_PHONE_ACTIVE: {
38             if (level != StatsUtils::INVALID_VALUE) {
39                 auto iter = phoneOnTimerMap_.find(level);
40                 if (iter != phoneOnTimerMap_.end() && iter->second != nullptr) {
41                     activeTimeMs = iter->second->GetRunningTimeMs();
42                     STATS_HILOGD(COMP_SVC, "Get phone on time: %{public}" PRId64 "ms of signal level: %{public}d",
43                         activeTimeMs, level);
44                     break;
45                 }
46                 STATS_HILOGD(COMP_SVC, "No phone on timer found, return 0");
47                 break;
48             }
49             activeTimeMs = GetTotalTimeMs(StatsUtils::STATS_TYPE_PHONE_ACTIVE);
50             STATS_HILOGD(COMP_SVC, "Get phone on total time: %{public}" PRId64 "ms", activeTimeMs);
51             break;
52         }
53         case StatsUtils::STATS_TYPE_PHONE_DATA: {
54             if (level != StatsUtils::INVALID_VALUE) {
55                 auto iter = phoneDataTimerMap_.find(level);
56                 if (iter != phoneDataTimerMap_.end() && iter->second != nullptr) {
57                     activeTimeMs = iter->second->GetRunningTimeMs();
58                     STATS_HILOGD(COMP_SVC, "Get phone data time: %{public}" PRId64 "ms of signal level: %{public}d",
59                         activeTimeMs, level);
60                     break;
61                 }
62                 STATS_HILOGD(COMP_SVC, "No phone data timer found, return 0");
63                 break;
64             }
65             activeTimeMs = GetTotalTimeMs(StatsUtils::STATS_TYPE_PHONE_DATA);
66             STATS_HILOGD(COMP_SVC, "Get phone data total time: %{public}" PRId64 "ms", activeTimeMs);
67             break;
68         }
69         default:
70             break;
71     }
72     return activeTimeMs;
73 }
74 
GetTotalTimeMs(StatsUtils::StatsType statsType)75 int64_t PhoneEntity::GetTotalTimeMs(StatsUtils::StatsType statsType)
76 {
77     int64_t totalTimeMs = StatsUtils::DEFAULT_VALUE;
78     switch (statsType) {
79         case StatsUtils::STATS_TYPE_PHONE_ACTIVE: {
80             for (auto timerIter : phoneOnTimerMap_) {
81                 totalTimeMs += timerIter.second->GetRunningTimeMs();
82             }
83             break;
84         }
85         case StatsUtils::STATS_TYPE_PHONE_DATA: {
86             for (auto timerIter : phoneDataTimerMap_) {
87                 totalTimeMs += timerIter.second->GetRunningTimeMs();
88             }
89             break;
90         }
91         default:
92             break;
93     }
94     return totalTimeMs;
95 }
96 
Calculate(int32_t uid)97 void PhoneEntity::Calculate(int32_t uid)
98 {
99     auto bss = BatteryStatsService::GetInstance();
100     // Calculate phone on power
101     double phoneOnPowerMah = StatsUtils::DEFAULT_VALUE;
102     for (int32_t i = 0; i < StatsUtils::RADIO_SIGNAL_BIN; i++) {
103         auto phoneOnAverageMa =
104             bss->GetBatteryStatsParser()->GetAveragePowerMa(StatsUtils::CURRENT_RADIO_ON, i);
105         auto phoneOnLevelTimeMs = GetActiveTimeMs(StatsUtils::STATS_TYPE_PHONE_ACTIVE, i);
106         double phoneOnLevelPowerMah = phoneOnAverageMa * phoneOnLevelTimeMs / StatsUtils::MS_IN_HOUR;
107         phoneOnPowerMah += phoneOnLevelPowerMah;
108     }
109 
110     // Calculate phone data power
111     double phoneDataPowerMah = StatsUtils::DEFAULT_VALUE;
112     for (int32_t i = 0; i < StatsUtils::RADIO_SIGNAL_BIN; i++) {
113         auto phoneDataAverageMa =
114             bss->GetBatteryStatsParser()->GetAveragePowerMa(StatsUtils::CURRENT_RADIO_DATA, i);
115         auto phoneDataLevelTimeMs = GetActiveTimeMs(StatsUtils::STATS_TYPE_PHONE_DATA, i);
116         double phoneDataLevelPowerMah = phoneDataAverageMa * phoneDataLevelTimeMs / StatsUtils::MS_IN_HOUR;
117         phoneDataPowerMah += phoneDataLevelPowerMah;
118     }
119     phonePowerMah_ = phoneOnPowerMah + phoneDataPowerMah;
120     totalPowerMah_ += phonePowerMah_;
121     std::shared_ptr<BatteryStatsInfo> statsInfo = std::make_shared<BatteryStatsInfo>();
122     statsInfo->SetConsumptioType(BatteryStatsInfo::CONSUMPTION_TYPE_PHONE);
123     statsInfo->SetPower(phonePowerMah_);
124     statsInfoList_.push_back(statsInfo);
125     STATS_HILOGD(COMP_SVC, "Calculate phone active power consumption: %{public}lfmAh", phonePowerMah_);
126 }
127 
GetEntityPowerMah(int32_t uidOrUserId)128 double PhoneEntity::GetEntityPowerMah(int32_t uidOrUserId)
129 {
130     return phonePowerMah_;
131 }
132 
GetStatsPowerMah(StatsUtils::StatsType statsType,int32_t uid)133 double PhoneEntity::GetStatsPowerMah(StatsUtils::StatsType statsType, int32_t uid)
134 {
135     return phonePowerMah_;
136 }
137 
GetOrCreateTimer(StatsUtils::StatsType statsType,int16_t level)138 std::shared_ptr<StatsHelper::ActiveTimer> PhoneEntity::GetOrCreateTimer(StatsUtils::StatsType statsType, int16_t level)
139 {
140     std::shared_ptr<StatsHelper::ActiveTimer> timer = nullptr;
141     if (level <= StatsUtils::INVALID_VALUE || level > StatsUtils::RADIO_SIGNAL_BIN) {
142         STATS_HILOGD(COMP_SVC, "Illegal signal level");
143         return timer;
144     }
145 
146     switch (statsType) {
147         case StatsUtils::STATS_TYPE_PHONE_ACTIVE: {
148             auto onIter = phoneOnTimerMap_.find(level);
149             if (onIter != phoneOnTimerMap_.end()) {
150                 STATS_HILOGD(COMP_SVC, "Get phone on timer for level: %{public}d", level);
151                 timer = onIter->second;
152                 break;
153             }
154             STATS_HILOGD(COMP_SVC, "Create phone on timer for level: %{public}d", level);
155             std::shared_ptr<StatsHelper::ActiveTimer> phoneOnTimer = std::make_shared<StatsHelper::ActiveTimer>();
156             phoneOnTimerMap_.insert(
157                 std::pair<int32_t, std::shared_ptr<StatsHelper::ActiveTimer>>(level, phoneOnTimer));
158             timer = phoneOnTimer;
159             break;
160         }
161         case StatsUtils::STATS_TYPE_PHONE_DATA: {
162             auto dataIter = phoneDataTimerMap_.find(level);
163             if (dataIter != phoneDataTimerMap_.end()) {
164                 STATS_HILOGD(COMP_SVC, "Get phone data timer for level: %{public}d", level);
165                 timer = dataIter->second;
166                 break;
167             }
168             STATS_HILOGD(COMP_SVC, "Create phone data timer for level: %{public}d", level);
169             std::shared_ptr<StatsHelper::ActiveTimer> phoneDataTimer = std::make_shared<StatsHelper::ActiveTimer>();
170             phoneDataTimerMap_.insert(
171                 std::pair<int32_t, std::shared_ptr<StatsHelper::ActiveTimer>>(level, phoneDataTimer));
172             timer = phoneDataTimer;
173             break;
174         }
175         default:
176             STATS_HILOGW(COMP_SVC, "Create phone timer failed");
177             break;
178     }
179     return timer;
180 }
181 
Reset()182 void PhoneEntity::Reset()
183 {
184     // Reset app Phone total power consumption
185     phonePowerMah_ = StatsUtils::DEFAULT_VALUE;
186 
187     // Reset Phone on timer
188     for (auto& iter : phoneOnTimerMap_) {
189         if (iter.second) {
190             iter.second->Reset();
191         }
192     }
193 
194     // Reset Phone data timer
195     for (auto& iter : phoneDataTimerMap_) {
196         if (iter.second) {
197             iter.second->Reset();
198         }
199     }
200 }
201 
DumpInfo(std::string & result,int32_t uid)202 void PhoneEntity::DumpInfo(std::string& result, int32_t uid)
203 {
204     int64_t phoneOnTime = StatsUtils::DEFAULT_VALUE;
205     int64_t phoneDataTime = StatsUtils::DEFAULT_VALUE;
206     for (int32_t i = 0; i < StatsUtils::RADIO_SIGNAL_BIN; i++) {
207         phoneOnTime += GetActiveTimeMs(StatsUtils::STATS_TYPE_PHONE_ACTIVE, i);
208         phoneDataTime += GetActiveTimeMs(StatsUtils::STATS_TYPE_PHONE_DATA, i);
209     }
210 
211     result.append("Phone dump:\n")
212         .append("Phone active time: ")
213         .append(ToString(phoneOnTime))
214         .append("ms\n")
215         .append("Phone data time: ")
216         .append(ToString(phoneDataTime))
217         .append("ms")
218         .append("\n");
219 }
220 } // namespace PowerMgr
221 } // namespace OHOS