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