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 auto g_statsService = DelayedStatsSpSingleton<BatteryStatsService>::GetInstance();
27 }
28
PhoneEntity()29 PhoneEntity::PhoneEntity()
30 {
31 consumptionType_ = BatteryStatsInfo::CONSUMPTION_TYPE_PHONE;
32 }
33
GetActiveTimeMs(StatsUtils::StatsType statsType,int16_t level)34 int64_t PhoneEntity::GetActiveTimeMs(StatsUtils::StatsType statsType, int16_t level)
35 {
36 int64_t activeTimeMs = StatsUtils::DEFAULT_VALUE;
37 switch (statsType) {
38 case StatsUtils::STATS_TYPE_PHONE_ACTIVE: {
39 if (level != StatsUtils::INVALID_VALUE) {
40 auto iter = phoneOnTimerMap_.find(level);
41 if (iter != phoneOnTimerMap_.end() && iter->second != nullptr) {
42 activeTimeMs = iter->second->GetRunningTimeMs();
43 STATS_HILOGD(COMP_SVC, "Get phone on time: %{public}" PRId64 "ms of signal level: %{public}d",
44 activeTimeMs, level);
45 break;
46 }
47 STATS_HILOGD(COMP_SVC, "No phone on timer found, return 0");
48 break;
49 }
50 activeTimeMs = GetTotalTimeMs(StatsUtils::STATS_TYPE_PHONE_ACTIVE);
51 STATS_HILOGD(COMP_SVC, "Get phone on total time: %{public}" PRId64 "ms", activeTimeMs);
52 break;
53 }
54 case StatsUtils::STATS_TYPE_PHONE_DATA: {
55 if (level != StatsUtils::INVALID_VALUE) {
56 auto iter = phoneDataTimerMap_.find(level);
57 if (iter != phoneDataTimerMap_.end() && iter->second != nullptr) {
58 activeTimeMs = iter->second->GetRunningTimeMs();
59 STATS_HILOGD(COMP_SVC, "Get phone data time: %{public}" PRId64 "ms of signal level: %{public}d",
60 activeTimeMs, level);
61 break;
62 }
63 STATS_HILOGD(COMP_SVC, "No phone data timer found, return 0");
64 break;
65 }
66 activeTimeMs = GetTotalTimeMs(StatsUtils::STATS_TYPE_PHONE_DATA);
67 STATS_HILOGD(COMP_SVC, "Get phone data total time: %{public}" PRId64 "ms", activeTimeMs);
68 break;
69 }
70 default:
71 break;
72 }
73 return activeTimeMs;
74 }
75
GetTotalTimeMs(StatsUtils::StatsType statsType)76 int64_t PhoneEntity::GetTotalTimeMs(StatsUtils::StatsType statsType)
77 {
78 int64_t totalTimeMs = StatsUtils::DEFAULT_VALUE;
79 switch (statsType) {
80 case StatsUtils::STATS_TYPE_PHONE_ACTIVE: {
81 for (auto timerIter : phoneOnTimerMap_) {
82 totalTimeMs += timerIter.second->GetRunningTimeMs();
83 }
84 break;
85 }
86 case StatsUtils::STATS_TYPE_PHONE_DATA: {
87 for (auto timerIter : phoneDataTimerMap_) {
88 totalTimeMs += timerIter.second->GetRunningTimeMs();
89 }
90 break;
91 }
92 default:
93 break;
94 }
95 return totalTimeMs;
96 }
97
Calculate(int32_t uid)98 void PhoneEntity::Calculate(int32_t uid)
99 {
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 g_statsService->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 g_statsService->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