blob: efd9907df9d329f9186f9eb631b397633ff90b61 [file] [log] [blame]
Todd Poynor752faf22013-06-12 13:25:59 -07001/*
2 * Copyright (C) 2013 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "healthd"
18
Yabin Cuie98e1772016-02-17 12:21:34 -080019#include <healthd/healthd.h>
20#include <healthd/BatteryMonitor.h>
Todd Poynor752faf22013-06-12 13:25:59 -070021
22#include <dirent.h>
23#include <errno.h>
24#include <fcntl.h>
25#include <stdio.h>
26#include <stdlib.h>
Mark Salyzynacb1ddf2015-07-23 09:22:50 -070027#include <sys/types.h>
Todd Poynor752faf22013-06-12 13:25:59 -070028#include <unistd.h>
Thierry Strudelf73de6f2019-01-11 17:09:20 -080029
30#include <algorithm>
James Hawkins588a2ca2016-02-18 14:52:46 -080031#include <memory>
Yifan Hong1d4368b2019-10-07 11:18:04 -070032#include <optional>
Mark Salyzynacb1ddf2015-07-23 09:22:50 -070033
Yifan Hongb99d15c2022-03-01 12:12:34 -080034#include <aidl/android/hardware/health/HealthInfo.h>
Michael Scott3217c5c2016-06-05 11:20:13 -070035#include <android-base/file.h>
Elliott Hughesda46b392016-10-11 17:09:00 -070036#include <android-base/parseint.h>
Michael Scott3217c5c2016-06-05 11:20:13 -070037#include <android-base/strings.h>
Yifan Hong1d4368b2019-10-07 11:18:04 -070038#include <android/hardware/health/2.1/types.h>
Yifan Hongb99d15c2022-03-01 12:12:34 -080039#include <android/hardware/health/translate-ndk.h>
Todd Poynor752faf22013-06-12 13:25:59 -070040#include <batteryservice/BatteryService.h>
41#include <cutils/klog.h>
Todd Poynor3db03a52014-05-21 16:28:13 -070042#include <cutils/properties.h>
Todd Poynorc133b712013-08-14 17:39:13 -070043#include <utils/Errors.h>
Todd Poynor752faf22013-06-12 13:25:59 -070044#include <utils/String8.h>
45#include <utils/Vector.h>
46
47#define POWER_SUPPLY_SUBSYSTEM "power_supply"
48#define POWER_SUPPLY_SYSFS_PATH "/sys/class/" POWER_SUPPLY_SUBSYSTEM
Ruchi Kandoia78fc232014-07-10 15:06:21 -070049#define FAKE_BATTERY_CAPACITY 42
50#define FAKE_BATTERY_TEMPERATURE 424
Ruchi Kandoi5c09ec12016-02-25 16:19:30 -080051#define MILLION 1.0e6
Badhri Jagan Sridharan40e1df42015-10-27 10:43:53 -070052#define DEFAULT_VBUS_VOLTAGE 5000000
Todd Poynor752faf22013-06-12 13:25:59 -070053
Yifan Hong1d4368b2019-10-07 11:18:04 -070054using HealthInfo_1_0 = android::hardware::health::V1_0::HealthInfo;
55using HealthInfo_2_0 = android::hardware::health::V2_0::HealthInfo;
56using HealthInfo_2_1 = android::hardware::health::V2_1::HealthInfo;
Yifan Hongb99d15c2022-03-01 12:12:34 -080057using aidl::android::hardware::health::BatteryCapacityLevel;
Jack Wue561d032022-11-24 12:19:41 +080058using aidl::android::hardware::health::BatteryChargingPolicy;
59using aidl::android::hardware::health::BatteryChargingState;
Yifan Hongb99d15c2022-03-01 12:12:34 -080060using aidl::android::hardware::health::BatteryHealth;
Jack Wue561d032022-11-24 12:19:41 +080061using aidl::android::hardware::health::BatteryHealthData;
David Andersond5ed26a2023-12-08 15:12:24 -080062using aidl::android::hardware::health::BatteryPartStatus;
Yifan Hongb99d15c2022-03-01 12:12:34 -080063using aidl::android::hardware::health::BatteryStatus;
64using aidl::android::hardware::health::HealthInfo;
65
66namespace {
67
68// Translate from AIDL back to HIDL definition for getHealthInfo_*_* calls.
69// Skips storageInfo and diskStats.
70void translateToHidl(const ::aidl::android::hardware::health::HealthInfo& in,
71 ::android::hardware::health::V1_0::HealthInfo* out) {
72 out->chargerAcOnline = in.chargerAcOnline;
73 out->chargerUsbOnline = in.chargerUsbOnline;
74 out->chargerWirelessOnline = in.chargerWirelessOnline;
75 out->maxChargingCurrent = in.maxChargingCurrentMicroamps;
76 out->maxChargingVoltage = in.maxChargingVoltageMicrovolts;
77 out->batteryStatus =
78 static_cast<::android::hardware::health::V1_0::BatteryStatus>(in.batteryStatus);
79 out->batteryHealth =
80 static_cast<::android::hardware::health::V1_0::BatteryHealth>(in.batteryHealth);
81 out->batteryPresent = in.batteryPresent;
82 out->batteryLevel = in.batteryLevel;
83 out->batteryVoltage = in.batteryVoltageMillivolts;
84 out->batteryTemperature = in.batteryTemperatureTenthsCelsius;
85 out->batteryCurrent = in.batteryCurrentMicroamps;
86 out->batteryCycleCount = in.batteryCycleCount;
87 out->batteryFullCharge = in.batteryFullChargeUah;
88 out->batteryChargeCounter = in.batteryChargeCounterUah;
89 out->batteryTechnology = in.batteryTechnology;
90}
91
92void translateToHidl(const ::aidl::android::hardware::health::HealthInfo& in,
93 ::android::hardware::health::V2_0::HealthInfo* out) {
94 translateToHidl(in, &out->legacy);
95 out->batteryCurrentAverage = in.batteryCurrentAverageMicroamps;
96 // Skip storageInfo and diskStats
97}
98
99void translateToHidl(const ::aidl::android::hardware::health::HealthInfo& in,
100 ::android::hardware::health::V2_1::HealthInfo* out) {
101 translateToHidl(in, &out->legacy);
102 out->batteryCapacityLevel = static_cast<android::hardware::health::V2_1::BatteryCapacityLevel>(
103 in.batteryCapacityLevel);
104 out->batteryChargeTimeToFullNowSeconds = in.batteryChargeTimeToFullNowSeconds;
105 out->batteryFullChargeDesignCapacityUah = in.batteryFullChargeDesignCapacityUah;
106}
107
108} // namespace
Yifan Hong1d4368b2019-10-07 11:18:04 -0700109
Todd Poynor752faf22013-06-12 13:25:59 -0700110namespace android {
111
Yifan Hong1d4368b2019-10-07 11:18:04 -0700112template <typename T>
113struct SysfsStringEnumMap {
Mark Salyzyn6f5b47f2014-05-15 15:00:59 -0700114 const char* s;
Yifan Hong1d4368b2019-10-07 11:18:04 -0700115 T val;
Todd Poynor752faf22013-06-12 13:25:59 -0700116};
117
Yifan Hong1d4368b2019-10-07 11:18:04 -0700118template <typename T>
119static std::optional<T> mapSysfsString(const char* str, SysfsStringEnumMap<T> map[]) {
Todd Poynor752faf22013-06-12 13:25:59 -0700120 for (int i = 0; map[i].s; i++)
121 if (!strcmp(str, map[i].s))
122 return map[i].val;
123
Yifan Hong1d4368b2019-10-07 11:18:04 -0700124 return std::nullopt;
Yabin Cuidb04a492016-02-16 17:19:23 -0800125}
126
Yifan Hongb99d15c2022-03-01 12:12:34 -0800127static void initHealthInfo(HealthInfo* health_info) {
Bart Van Assche024e18f2022-02-24 21:22:07 +0000128 *health_info = {
129 .batteryCapacityLevel = BatteryCapacityLevel::UNSUPPORTED,
130 .batteryChargeTimeToFullNowSeconds =
131 (int64_t)HealthInfo::BATTERY_CHARGE_TIME_TO_FULL_NOW_SECONDS_UNSUPPORTED,
132 .batteryStatus = BatteryStatus::UNKNOWN,
133 .batteryHealth = BatteryHealth::UNKNOWN,
134 };
Yifan Hong6cabe9b2019-11-05 17:04:50 -0800135}
136
Todd Poynore030a102018-01-19 14:03:59 -0800137BatteryMonitor::BatteryMonitor()
138 : mHealthdConfig(nullptr),
139 mBatteryDevicePresent(false),
140 mBatteryFixedCapacity(0),
Yifan Hong1d4368b2019-10-07 11:18:04 -0700141 mBatteryFixedTemperature(0),
Jack Wub57f68a2023-02-04 19:56:06 +0800142 mBatteryHealthStatus(BatteryMonitor::BH_UNKNOWN),
Yifan Hongb99d15c2022-03-01 12:12:34 -0800143 mHealthInfo(std::make_unique<HealthInfo>()) {
Yifan Hong6cabe9b2019-11-05 17:04:50 -0800144 initHealthInfo(mHealthInfo.get());
145}
Yifan Hong1d4368b2019-10-07 11:18:04 -0700146
147BatteryMonitor::~BatteryMonitor() {}
148
Yifan Hongb99d15c2022-03-01 12:12:34 -0800149HealthInfo_1_0 BatteryMonitor::getHealthInfo_1_0() const {
150 HealthInfo_1_0 health_info_1_0;
151 translateToHidl(*mHealthInfo, &health_info_1_0);
152 return health_info_1_0;
Yabin Cuidb04a492016-02-16 17:19:23 -0800153}
154
Yifan Hongb99d15c2022-03-01 12:12:34 -0800155HealthInfo_2_0 BatteryMonitor::getHealthInfo_2_0() const {
156 HealthInfo_2_0 health_info_2_0;
157 translateToHidl(*mHealthInfo, &health_info_2_0);
158 return health_info_2_0;
Hridya Valsaraju7fa72252018-01-12 17:44:33 -0800159}
160
Yifan Hongb99d15c2022-03-01 12:12:34 -0800161HealthInfo_2_1 BatteryMonitor::getHealthInfo_2_1() const {
162 HealthInfo_2_1 health_info_2_1;
163 translateToHidl(*mHealthInfo, &health_info_2_1);
164 return health_info_2_1;
165}
166
167const HealthInfo& BatteryMonitor::getHealthInfo() const {
Yifan Hong1d4368b2019-10-07 11:18:04 -0700168 return *mHealthInfo;
169}
170
171BatteryStatus getBatteryStatus(const char* status) {
172 static SysfsStringEnumMap<BatteryStatus> batteryStatusMap[] = {
173 {"Unknown", BatteryStatus::UNKNOWN},
174 {"Charging", BatteryStatus::CHARGING},
175 {"Discharging", BatteryStatus::DISCHARGING},
176 {"Not charging", BatteryStatus::NOT_CHARGING},
177 {"Full", BatteryStatus::FULL},
178 {NULL, BatteryStatus::UNKNOWN},
Todd Poynor752faf22013-06-12 13:25:59 -0700179 };
180
Yifan Hong1d4368b2019-10-07 11:18:04 -0700181 auto ret = mapSysfsString(status, batteryStatusMap);
182 if (!ret) {
Todd Poynor752faf22013-06-12 13:25:59 -0700183 KLOG_WARNING(LOG_TAG, "Unknown battery status '%s'\n", status);
Yifan Hong1d4368b2019-10-07 11:18:04 -0700184 *ret = BatteryStatus::UNKNOWN;
Todd Poynor752faf22013-06-12 13:25:59 -0700185 }
186
Yifan Hong1d4368b2019-10-07 11:18:04 -0700187 return *ret;
Todd Poynor752faf22013-06-12 13:25:59 -0700188}
189
Stephane Lee86f9f6a2019-12-19 15:09:41 -0800190BatteryCapacityLevel getBatteryCapacityLevel(const char* capacityLevel) {
191 static SysfsStringEnumMap<BatteryCapacityLevel> batteryCapacityLevelMap[] = {
192 {"Unknown", BatteryCapacityLevel::UNKNOWN},
193 {"Critical", BatteryCapacityLevel::CRITICAL},
194 {"Low", BatteryCapacityLevel::LOW},
195 {"Normal", BatteryCapacityLevel::NORMAL},
196 {"High", BatteryCapacityLevel::HIGH},
197 {"Full", BatteryCapacityLevel::FULL},
Stephane Lee06846042020-02-12 17:00:24 -0800198 {NULL, BatteryCapacityLevel::UNSUPPORTED},
Stephane Lee86f9f6a2019-12-19 15:09:41 -0800199 };
200
201 auto ret = mapSysfsString(capacityLevel, batteryCapacityLevelMap);
202 if (!ret) {
Stephane Lee06846042020-02-12 17:00:24 -0800203 KLOG_WARNING(LOG_TAG, "Unsupported battery capacity level '%s'\n", capacityLevel);
204 *ret = BatteryCapacityLevel::UNSUPPORTED;
Stephane Lee86f9f6a2019-12-19 15:09:41 -0800205 }
206
207 return *ret;
208}
209
Yifan Hong1d4368b2019-10-07 11:18:04 -0700210BatteryHealth getBatteryHealth(const char* status) {
211 static SysfsStringEnumMap<BatteryHealth> batteryHealthMap[] = {
212 {"Unknown", BatteryHealth::UNKNOWN},
213 {"Good", BatteryHealth::GOOD},
214 {"Overheat", BatteryHealth::OVERHEAT},
215 {"Dead", BatteryHealth::DEAD},
216 {"Over voltage", BatteryHealth::OVER_VOLTAGE},
217 {"Unspecified failure", BatteryHealth::UNSPECIFIED_FAILURE},
218 {"Cold", BatteryHealth::COLD},
219 // battery health values from JEITA spec
220 {"Warm", BatteryHealth::GOOD},
221 {"Cool", BatteryHealth::GOOD},
222 {"Hot", BatteryHealth::OVERHEAT},
223 {NULL, BatteryHealth::UNKNOWN},
Todd Poynor752faf22013-06-12 13:25:59 -0700224 };
225
Yifan Hong1d4368b2019-10-07 11:18:04 -0700226 auto ret = mapSysfsString(status, batteryHealthMap);
227 if (!ret) {
Todd Poynor752faf22013-06-12 13:25:59 -0700228 KLOG_WARNING(LOG_TAG, "Unknown battery health '%s'\n", status);
Yifan Hong1d4368b2019-10-07 11:18:04 -0700229 *ret = BatteryHealth::UNKNOWN;
Todd Poynor752faf22013-06-12 13:25:59 -0700230 }
231
Yifan Hong1d4368b2019-10-07 11:18:04 -0700232 return *ret;
Todd Poynor752faf22013-06-12 13:25:59 -0700233}
234
Jack Wub57f68a2023-02-04 19:56:06 +0800235BatteryHealth getBatteryHealthStatus(int status) {
236 BatteryHealth value;
237
238 if (status == BatteryMonitor::BH_NOMINAL)
239 value = BatteryHealth::GOOD;
240 else if (status == BatteryMonitor::BH_MARGINAL)
241 value = BatteryHealth::FAIR;
242 else if (status == BatteryMonitor::BH_NEEDS_REPLACEMENT)
243 value = BatteryHealth::DEAD;
244 else if (status == BatteryMonitor::BH_FAILED)
245 value = BatteryHealth::UNSPECIFIED_FAILURE;
Jack Wucf996f32023-04-13 19:37:46 +0800246 else if (status == BatteryMonitor::BH_NOT_AVAILABLE)
247 value = BatteryHealth::NOT_AVAILABLE;
Jack Wu8231c3f2023-05-19 14:31:53 +0800248 else if (status == BatteryMonitor::BH_INCONSISTENT)
249 value = BatteryHealth::INCONSISTENT;
Jack Wub57f68a2023-02-04 19:56:06 +0800250 else
251 value = BatteryHealth::UNKNOWN;
252
253 return value;
254}
255
Jack Wue561d032022-11-24 12:19:41 +0800256BatteryChargingPolicy getBatteryChargingPolicy(const char* chargingPolicy) {
257 static SysfsStringEnumMap<BatteryChargingPolicy> batteryChargingPolicyMap[] = {
258 {"0", BatteryChargingPolicy::INVALID}, {"1", BatteryChargingPolicy::DEFAULT},
259 {"2", BatteryChargingPolicy::LONG_LIFE}, {"3", BatteryChargingPolicy::ADAPTIVE},
260 {NULL, BatteryChargingPolicy::DEFAULT},
261 };
262
263 auto ret = mapSysfsString(chargingPolicy, batteryChargingPolicyMap);
264 if (!ret) {
265 *ret = BatteryChargingPolicy::DEFAULT;
266 }
267
268 return *ret;
269}
270
271BatteryChargingState getBatteryChargingState(const char* chargingState) {
272 static SysfsStringEnumMap<BatteryChargingState> batteryChargingStateMap[] = {
273 {"0", BatteryChargingState::INVALID}, {"1", BatteryChargingState::NORMAL},
274 {"2", BatteryChargingState::TOO_COLD}, {"3", BatteryChargingState::TOO_HOT},
275 {"4", BatteryChargingState::LONG_LIFE}, {"5", BatteryChargingState::ADAPTIVE},
276 {NULL, BatteryChargingState::NORMAL},
277 };
278
279 auto ret = mapSysfsString(chargingState, batteryChargingStateMap);
280 if (!ret) {
281 *ret = BatteryChargingState::NORMAL;
282 }
283
284 return *ret;
285}
286
Bart Van Assche095c9442022-03-02 17:36:34 +0000287static int readFromFile(const String8& path, std::string* buf) {
Bart Van Assche5a7e5082022-02-24 21:40:15 +0000288 buf->clear();
Steven Moreland2aac3352017-03-10 22:31:08 -0800289 if (android::base::ReadFileToString(path.c_str(), buf)) {
Michael Scott3217c5c2016-06-05 11:20:13 -0700290 *buf = android::base::Trim(*buf);
Todd Poynor752faf22013-06-12 13:25:59 -0700291 }
Michael Scott3217c5c2016-06-05 11:20:13 -0700292 return buf->length();
Todd Poynor752faf22013-06-12 13:25:59 -0700293}
294
Jack Wue561d032022-11-24 12:19:41 +0800295static bool writeToFile(const String8& path, int32_t in_value) {
296 return android::base::WriteStringToFile(std::to_string(in_value), path.c_str());
297}
298
Bart Van Assche095c9442022-03-02 17:36:34 +0000299static BatteryMonitor::PowerSupplyType readPowerSupplyType(const String8& path) {
Yifan Hong1d4368b2019-10-07 11:18:04 -0700300 static SysfsStringEnumMap<int> supplyTypeMap[] = {
Bart Van Assche095c9442022-03-02 17:36:34 +0000301 {"Unknown", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_UNKNOWN},
302 {"Battery", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_BATTERY},
303 {"UPS", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_AC},
304 {"Mains", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_AC},
305 {"USB", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_USB},
306 {"USB_DCP", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_AC},
307 {"USB_HVDCP", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_AC},
308 {"USB_CDP", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_AC},
309 {"USB_ACA", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_AC},
310 {"USB_C", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_AC},
311 {"USB_PD", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_AC},
312 {"USB_PD_DRP", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_USB},
313 {"Wireless", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_WIRELESS},
314 {"Dock", BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_DOCK},
Yifan Hong1d4368b2019-10-07 11:18:04 -0700315 {NULL, 0},
Todd Poynor752faf22013-06-12 13:25:59 -0700316 };
Yifan Hong1d4368b2019-10-07 11:18:04 -0700317 std::string buf;
Todd Poynor752faf22013-06-12 13:25:59 -0700318
Bart Van Assche095c9442022-03-02 17:36:34 +0000319 if (readFromFile(path, &buf) <= 0) {
320 return BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_UNKNOWN;
321 }
Todd Poynor752faf22013-06-12 13:25:59 -0700322
Yifan Hong1d4368b2019-10-07 11:18:04 -0700323 auto ret = mapSysfsString(buf.c_str(), supplyTypeMap);
John Stultz47a6bf02019-11-06 00:23:34 +0000324 if (!ret) {
Michael Scott3217c5c2016-06-05 11:20:13 -0700325 KLOG_WARNING(LOG_TAG, "Unknown power supply type '%s'\n", buf.c_str());
Bart Van Assche095c9442022-03-02 17:36:34 +0000326 *ret = BatteryMonitor::ANDROID_POWER_SUPPLY_TYPE_UNKNOWN;
Johan Redestig32828612016-02-03 13:45:54 +0100327 }
Todd Poynor752faf22013-06-12 13:25:59 -0700328
Yifan Hong1d4368b2019-10-07 11:18:04 -0700329 return static_cast<BatteryMonitor::PowerSupplyType>(*ret);
Todd Poynor752faf22013-06-12 13:25:59 -0700330}
331
Bart Van Assche095c9442022-03-02 17:36:34 +0000332static bool getBooleanField(const String8& path) {
Michael Scott3217c5c2016-06-05 11:20:13 -0700333 std::string buf;
Todd Poynor752faf22013-06-12 13:25:59 -0700334 bool value = false;
Michael Scott3217c5c2016-06-05 11:20:13 -0700335
336 if (readFromFile(path, &buf) > 0)
337 if (buf[0] != '0')
Todd Poynor752faf22013-06-12 13:25:59 -0700338 value = true;
Todd Poynor752faf22013-06-12 13:25:59 -0700339
340 return value;
341}
342
Bart Van Assche095c9442022-03-02 17:36:34 +0000343static int getIntField(const String8& path) {
Michael Scott3217c5c2016-06-05 11:20:13 -0700344 std::string buf;
Todd Poynor752faf22013-06-12 13:25:59 -0700345 int value = 0;
Michael Scott3217c5c2016-06-05 11:20:13 -0700346
347 if (readFromFile(path, &buf) > 0)
Elliott Hughesda46b392016-10-11 17:09:00 -0700348 android::base::ParseInt(buf, &value);
Michael Scott3217c5c2016-06-05 11:20:13 -0700349
Todd Poynor752faf22013-06-12 13:25:59 -0700350 return value;
351}
352
Bart Van Assche095c9442022-03-02 17:36:34 +0000353static bool isScopedPowerSupply(const char* name) {
Kazuhiro Inaba8e4d9822019-06-12 13:46:08 +0900354 constexpr char kScopeDevice[] = "Device";
355
356 String8 path;
357 path.appendFormat("%s/%s/scope", POWER_SUPPLY_SYSFS_PATH, name);
358 std::string scope;
359 return (readFromFile(path, &scope) > 0 && scope == kScopeDevice);
360}
361
Yifan Hong1353e702019-10-07 10:41:30 -0700362void BatteryMonitor::updateValues(void) {
Yifan Hong6cabe9b2019-11-05 17:04:50 -0800363 initHealthInfo(mHealthInfo.get());
Yifan Hong1d4368b2019-10-07 11:18:04 -0700364
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000365 if (!mHealthdConfig->batteryPresentPath.empty())
Yifan Hongb99d15c2022-03-01 12:12:34 -0800366 mHealthInfo->batteryPresent = getBooleanField(mHealthdConfig->batteryPresentPath);
Todd Poynor752faf22013-06-12 13:25:59 -0700367 else
Yifan Hongb99d15c2022-03-01 12:12:34 -0800368 mHealthInfo->batteryPresent = mBatteryDevicePresent;
Todd Poynor752faf22013-06-12 13:25:59 -0700369
Yifan Hongb99d15c2022-03-01 12:12:34 -0800370 mHealthInfo->batteryLevel = mBatteryFixedCapacity
371 ? mBatteryFixedCapacity
372 : getIntField(mHealthdConfig->batteryCapacityPath);
373 mHealthInfo->batteryVoltageMillivolts = getIntField(mHealthdConfig->batteryVoltagePath) / 1000;
Todd Poynorb45f1f52013-07-30 18:57:16 -0700374
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000375 if (!mHealthdConfig->batteryCurrentNowPath.empty())
Yifan Hongb99d15c2022-03-01 12:12:34 -0800376 mHealthInfo->batteryCurrentMicroamps = getIntField(mHealthdConfig->batteryCurrentNowPath);
Ruchi Kandoicc338802015-08-24 13:01:16 -0700377
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000378 if (!mHealthdConfig->batteryFullChargePath.empty())
Yifan Hongb99d15c2022-03-01 12:12:34 -0800379 mHealthInfo->batteryFullChargeUah = getIntField(mHealthdConfig->batteryFullChargePath);
Ruchi Kandoicc338802015-08-24 13:01:16 -0700380
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000381 if (!mHealthdConfig->batteryCycleCountPath.empty())
Yifan Hongb99d15c2022-03-01 12:12:34 -0800382 mHealthInfo->batteryCycleCount = getIntField(mHealthdConfig->batteryCycleCountPath);
Ruchi Kandoicc338802015-08-24 13:01:16 -0700383
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000384 if (!mHealthdConfig->batteryChargeCounterPath.empty())
Yifan Hongb99d15c2022-03-01 12:12:34 -0800385 mHealthInfo->batteryChargeCounterUah =
386 getIntField(mHealthdConfig->batteryChargeCounterPath);
Ruchi Kandoi3f9886b2016-04-07 12:34:40 -0700387
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000388 if (!mHealthdConfig->batteryCurrentAvgPath.empty())
Yifan Hongb99d15c2022-03-01 12:12:34 -0800389 mHealthInfo->batteryCurrentAverageMicroamps =
Yifan Hong35cb0832019-10-07 13:58:29 -0700390 getIntField(mHealthdConfig->batteryCurrentAvgPath);
391
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000392 if (!mHealthdConfig->batteryChargeTimeToFullNowPath.empty())
Stephane Lee86f9f6a2019-12-19 15:09:41 -0800393 mHealthInfo->batteryChargeTimeToFullNowSeconds =
394 getIntField(mHealthdConfig->batteryChargeTimeToFullNowPath);
395
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000396 if (!mHealthdConfig->batteryFullChargeDesignCapacityUahPath.empty())
Stephane Lee1c108ed2020-02-10 18:23:57 -0800397 mHealthInfo->batteryFullChargeDesignCapacityUah =
398 getIntField(mHealthdConfig->batteryFullChargeDesignCapacityUahPath);
Yifan Hong35cb0832019-10-07 13:58:29 -0700399
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000400 if (!mHealthdConfig->batteryHealthStatusPath.empty())
Jack Wub57f68a2023-02-04 19:56:06 +0800401 mBatteryHealthStatus = getIntField(mHealthdConfig->batteryHealthStatusPath);
402
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000403 if (!mHealthdConfig->batteryStateOfHealthPath.empty())
AleX Pelosiff708922023-02-17 01:39:21 +0000404 mHealthInfo->batteryHealthData->batteryStateOfHealth =
405 getIntField(mHealthdConfig->batteryStateOfHealthPath);
406
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000407 if (!mHealthdConfig->batteryManufacturingDatePath.empty())
Jack Wue561d032022-11-24 12:19:41 +0800408 mHealthInfo->batteryHealthData->batteryManufacturingDateSeconds =
409 getIntField(mHealthdConfig->batteryManufacturingDatePath);
410
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000411 if (!mHealthdConfig->batteryFirstUsageDatePath.empty())
Jack Wue561d032022-11-24 12:19:41 +0800412 mHealthInfo->batteryHealthData->batteryFirstUsageSeconds =
413 getIntField(mHealthdConfig->batteryFirstUsageDatePath);
414
Yifan Hongb99d15c2022-03-01 12:12:34 -0800415 mHealthInfo->batteryTemperatureTenthsCelsius =
416 mBatteryFixedTemperature ? mBatteryFixedTemperature
417 : getIntField(mHealthdConfig->batteryTemperaturePath);
Todd Poynor752faf22013-06-12 13:25:59 -0700418
Michael Scott3217c5c2016-06-05 11:20:13 -0700419 std::string buf;
Todd Poynor752faf22013-06-12 13:25:59 -0700420
Stephane Lee86f9f6a2019-12-19 15:09:41 -0800421 if (readFromFile(mHealthdConfig->batteryCapacityLevelPath, &buf) > 0)
422 mHealthInfo->batteryCapacityLevel = getBatteryCapacityLevel(buf.c_str());
423
Michael Scott3217c5c2016-06-05 11:20:13 -0700424 if (readFromFile(mHealthdConfig->batteryStatusPath, &buf) > 0)
Yifan Hongb99d15c2022-03-01 12:12:34 -0800425 mHealthInfo->batteryStatus = getBatteryStatus(buf.c_str());
Todd Poynor752faf22013-06-12 13:25:59 -0700426
Jack Wub57f68a2023-02-04 19:56:06 +0800427 // Backward compatible with android.hardware.health V1
428 if (mBatteryHealthStatus < BatteryMonitor::BH_MARGINAL) {
429 if (readFromFile(mHealthdConfig->batteryHealthPath, &buf) > 0)
430 mHealthInfo->batteryHealth = getBatteryHealth(buf.c_str());
431 } else {
432 mHealthInfo->batteryHealth = getBatteryHealthStatus(mBatteryHealthStatus);
433 }
Todd Poynor752faf22013-06-12 13:25:59 -0700434
Michael Scott3217c5c2016-06-05 11:20:13 -0700435 if (readFromFile(mHealthdConfig->batteryTechnologyPath, &buf) > 0)
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000436 mHealthInfo->batteryTechnology = buf;
Todd Poynor752faf22013-06-12 13:25:59 -0700437
Jack Wue561d032022-11-24 12:19:41 +0800438 if (readFromFile(mHealthdConfig->chargingPolicyPath, &buf) > 0)
439 mHealthInfo->chargingPolicy = getBatteryChargingPolicy(buf.c_str());
440
441 if (readFromFile(mHealthdConfig->chargingStatePath, &buf) > 0)
442 mHealthInfo->chargingState = getBatteryChargingState(buf.c_str());
443
Badhri Jagan Sridharan40e1df42015-10-27 10:43:53 -0700444 double MaxPower = 0;
Todd Poynor752faf22013-06-12 13:25:59 -0700445
ShevT9d98a6a2018-07-26 11:47:47 +0300446 for (size_t i = 0; i < mChargerNames.size(); i++) {
Todd Poynor752faf22013-06-12 13:25:59 -0700447 String8 path;
Tomasz Wasilczyk18b74612023-08-10 23:29:50 +0000448 path.appendFormat("%s/%s/online", POWER_SUPPLY_SYSFS_PATH, mChargerNames[i].c_str());
Michael Scott3217c5c2016-06-05 11:20:13 -0700449 if (getIntField(path)) {
450 path.clear();
Tomasz Wasilczyk18b74612023-08-10 23:29:50 +0000451 path.appendFormat("%s/%s/type", POWER_SUPPLY_SYSFS_PATH, mChargerNames[i].c_str());
Michael Scott3217c5c2016-06-05 11:20:13 -0700452 switch(readPowerSupplyType(path)) {
453 case ANDROID_POWER_SUPPLY_TYPE_AC:
Yifan Hongb99d15c2022-03-01 12:12:34 -0800454 mHealthInfo->chargerAcOnline = true;
Michael Scott3217c5c2016-06-05 11:20:13 -0700455 break;
456 case ANDROID_POWER_SUPPLY_TYPE_USB:
Yifan Hongb99d15c2022-03-01 12:12:34 -0800457 mHealthInfo->chargerUsbOnline = true;
Michael Scott3217c5c2016-06-05 11:20:13 -0700458 break;
459 case ANDROID_POWER_SUPPLY_TYPE_WIRELESS:
Yifan Hongb99d15c2022-03-01 12:12:34 -0800460 mHealthInfo->chargerWirelessOnline = true;
Michael Scott3217c5c2016-06-05 11:20:13 -0700461 break;
Jack Wu06b90412021-12-15 20:40:21 +0800462 case ANDROID_POWER_SUPPLY_TYPE_DOCK:
Yifan Hongb99d15c2022-03-01 12:12:34 -0800463 mHealthInfo->chargerDockOnline = true;
Jack Wu06b90412021-12-15 20:40:21 +0800464 break;
Michael Scott3217c5c2016-06-05 11:20:13 -0700465 default:
Jack Wu06b90412021-12-15 20:40:21 +0800466 path.clear();
467 path.appendFormat("%s/%s/is_dock", POWER_SUPPLY_SYSFS_PATH,
Tomasz Wasilczyk18b74612023-08-10 23:29:50 +0000468 mChargerNames[i].c_str());
469 if (access(path.c_str(), R_OK) == 0)
Yifan Hongb99d15c2022-03-01 12:12:34 -0800470 mHealthInfo->chargerDockOnline = true;
471 else
Jack Wu06b90412021-12-15 20:40:21 +0800472 KLOG_WARNING(LOG_TAG, "%s: Unknown power supply type\n",
Tomasz Wasilczyk18b74612023-08-10 23:29:50 +0000473 mChargerNames[i].c_str());
Michael Scott3217c5c2016-06-05 11:20:13 -0700474 }
475 path.clear();
476 path.appendFormat("%s/%s/current_max", POWER_SUPPLY_SYSFS_PATH,
Tomasz Wasilczyk18b74612023-08-10 23:29:50 +0000477 mChargerNames[i].c_str());
478 int ChargingCurrent = (access(path.c_str(), R_OK) == 0) ? getIntField(path) : 0;
Badhri Jagan Sridharan40e1df42015-10-27 10:43:53 -0700479
Dmitry Shmidt9f6b80c2016-06-20 12:58:37 -0700480 path.clear();
481 path.appendFormat("%s/%s/voltage_max", POWER_SUPPLY_SYSFS_PATH,
Tomasz Wasilczyk18b74612023-08-10 23:29:50 +0000482 mChargerNames[i].c_str());
Badhri Jagan Sridharan40e1df42015-10-27 10:43:53 -0700483
Dmitry Shmidt9f6b80c2016-06-20 12:58:37 -0700484 int ChargingVoltage =
Tomasz Wasilczyk18b74612023-08-10 23:29:50 +0000485 (access(path.c_str(), R_OK) == 0) ? getIntField(path) : DEFAULT_VBUS_VOLTAGE;
Badhri Jagan Sridharan40e1df42015-10-27 10:43:53 -0700486
Dmitry Shmidt9f6b80c2016-06-20 12:58:37 -0700487 double power = ((double)ChargingCurrent / MILLION) *
488 ((double)ChargingVoltage / MILLION);
489 if (MaxPower < power) {
Yifan Hongb99d15c2022-03-01 12:12:34 -0800490 mHealthInfo->maxChargingCurrentMicroamps = ChargingCurrent;
491 mHealthInfo->maxChargingVoltageMicrovolts = ChargingVoltage;
Dmitry Shmidt9f6b80c2016-06-20 12:58:37 -0700492 MaxPower = power;
Todd Poynor752faf22013-06-12 13:25:59 -0700493 }
494 }
495 }
Yifan Hong1353e702019-10-07 10:41:30 -0700496}
Todd Poynor752faf22013-06-12 13:25:59 -0700497
Bart Van Assche095c9442022-03-02 17:36:34 +0000498static void doLogValues(const HealthInfo& props, const struct healthd_config& healthd_config) {
Yifan Hong1353e702019-10-07 10:41:30 -0700499 char dmesgline[256];
500 size_t len;
501 if (props.batteryPresent) {
502 snprintf(dmesgline, sizeof(dmesgline), "battery l=%d v=%d t=%s%d.%d h=%d st=%d",
Yifan Hongb99d15c2022-03-01 12:12:34 -0800503 props.batteryLevel, props.batteryVoltageMillivolts,
504 props.batteryTemperatureTenthsCelsius < 0 ? "-" : "",
505 abs(props.batteryTemperatureTenthsCelsius / 10),
506 abs(props.batteryTemperatureTenthsCelsius % 10), props.batteryHealth,
507 props.batteryStatus);
Todd Poynorb45f1f52013-07-30 18:57:16 -0700508
Yifan Hong1353e702019-10-07 10:41:30 -0700509 len = strlen(dmesgline);
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000510 if (!healthd_config.batteryCurrentNowPath.empty()) {
Yifan Hong1353e702019-10-07 10:41:30 -0700511 len += snprintf(dmesgline + len, sizeof(dmesgline) - len, " c=%d",
Yifan Hongb99d15c2022-03-01 12:12:34 -0800512 props.batteryCurrentMicroamps);
Todd Poynor10b235e2013-08-07 15:25:14 -0700513 }
514
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000515 if (!healthd_config.batteryFullChargePath.empty()) {
Yifan Hong1353e702019-10-07 10:41:30 -0700516 len += snprintf(dmesgline + len, sizeof(dmesgline) - len, " fc=%d",
Yifan Hongb99d15c2022-03-01 12:12:34 -0800517 props.batteryFullChargeUah);
Yifan Hong1353e702019-10-07 10:41:30 -0700518 }
Mark Salyzynacb1ddf2015-07-23 09:22:50 -0700519
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000520 if (!healthd_config.batteryCycleCountPath.empty()) {
Yifan Hong1353e702019-10-07 10:41:30 -0700521 len += snprintf(dmesgline + len, sizeof(dmesgline) - len, " cc=%d",
522 props.batteryCycleCount);
523 }
524 } else {
525 len = snprintf(dmesgline, sizeof(dmesgline), "battery none");
Todd Poynorb45f1f52013-07-30 18:57:16 -0700526 }
527
Yifan Hongb99d15c2022-03-01 12:12:34 -0800528 snprintf(dmesgline + len, sizeof(dmesgline) - len, " chg=%s%s%s%s",
Yifan Hong1353e702019-10-07 10:41:30 -0700529 props.chargerAcOnline ? "a" : "", props.chargerUsbOnline ? "u" : "",
Yifan Hongb99d15c2022-03-01 12:12:34 -0800530 props.chargerWirelessOnline ? "w" : "", props.chargerDockOnline ? "d" : "");
Yifan Hong1353e702019-10-07 10:41:30 -0700531
532 KLOG_WARNING(LOG_TAG, "%s\n", dmesgline);
533}
534
Bart Van Assche095c9442022-03-02 17:36:34 +0000535void BatteryMonitor::logValues(const HealthInfo_2_1& health_info,
536 const struct healthd_config& healthd_config) {
537 HealthInfo aidl_health_info;
538 (void)android::h2a::translate(health_info, &aidl_health_info);
539 doLogValues(aidl_health_info, healthd_config);
540}
541
542void BatteryMonitor::logValues(void) {
543 doLogValues(*mHealthInfo, *mHealthdConfig);
544}
545
Yifan Hong1353e702019-10-07 10:41:30 -0700546bool BatteryMonitor::isChargerOnline() {
Yifan Hongb99d15c2022-03-01 12:12:34 -0800547 const HealthInfo& props = *mHealthInfo;
Jack Wu06b90412021-12-15 20:40:21 +0800548 return props.chargerAcOnline | props.chargerUsbOnline | props.chargerWirelessOnline |
Yifan Hongb99d15c2022-03-01 12:12:34 -0800549 props.chargerDockOnline;
Todd Poynor752faf22013-06-12 13:25:59 -0700550}
551
Yabin Cuiaedf6032016-02-19 18:03:23 -0800552int BatteryMonitor::getChargeStatus() {
Yifan Hong1d4368b2019-10-07 11:18:04 -0700553 BatteryStatus result = BatteryStatus::UNKNOWN;
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000554 if (!mHealthdConfig->batteryStatusPath.empty()) {
Michael Scott3217c5c2016-06-05 11:20:13 -0700555 std::string buf;
556 if (readFromFile(mHealthdConfig->batteryStatusPath, &buf) > 0)
557 result = getBatteryStatus(buf.c_str());
Yabin Cuiaedf6032016-02-19 18:03:23 -0800558 }
Yifan Hong1d4368b2019-10-07 11:18:04 -0700559 return static_cast<int>(result);
Yabin Cuiaedf6032016-02-19 18:03:23 -0800560}
561
Jack Wue561d032022-11-24 12:19:41 +0800562status_t BatteryMonitor::setChargingPolicy(int value) {
563 status_t ret = NAME_NOT_FOUND;
564 bool result;
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000565 if (!mHealthdConfig->chargingPolicyPath.empty()) {
Jack Wue561d032022-11-24 12:19:41 +0800566 result = writeToFile(mHealthdConfig->chargingPolicyPath, value);
567 if (!result) {
568 KLOG_WARNING(LOG_TAG, "setChargingPolicy fail\n");
569 ret = BAD_VALUE;
570 } else {
571 ret = OK;
572 }
573 }
574 return ret;
575}
576
577int BatteryMonitor::getChargingPolicy() {
578 BatteryChargingPolicy result = BatteryChargingPolicy::DEFAULT;
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000579 if (!mHealthdConfig->chargingPolicyPath.empty()) {
Jack Wue561d032022-11-24 12:19:41 +0800580 std::string buf;
581 if (readFromFile(mHealthdConfig->chargingPolicyPath, &buf) > 0)
582 result = getBatteryChargingPolicy(buf.c_str());
583 }
584 return static_cast<int>(result);
585}
586
587int BatteryMonitor::getBatteryHealthData(int id) {
588 if (id == BATTERY_PROP_MANUFACTURING_DATE) {
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000589 if (!mHealthdConfig->batteryManufacturingDatePath.empty())
Jack Wue561d032022-11-24 12:19:41 +0800590 return getIntField(mHealthdConfig->batteryManufacturingDatePath);
591 }
592 if (id == BATTERY_PROP_FIRST_USAGE_DATE) {
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000593 if (!mHealthdConfig->batteryFirstUsageDatePath.empty())
Jack Wue561d032022-11-24 12:19:41 +0800594 return getIntField(mHealthdConfig->batteryFirstUsageDatePath);
595 }
AleX Pelosiff708922023-02-17 01:39:21 +0000596 if (id == BATTERY_PROP_STATE_OF_HEALTH) {
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000597 if (!mHealthdConfig->batteryStateOfHealthPath.empty())
AleX Pelosiff708922023-02-17 01:39:21 +0000598 return getIntField(mHealthdConfig->batteryStateOfHealthPath);
599 }
David Andersond5ed26a2023-12-08 15:12:24 -0800600 if (id == BATTERY_PROP_PART_STATUS) {
601 return static_cast<int>(BatteryPartStatus::UNSUPPORTED);
602 }
Jack Wue561d032022-11-24 12:19:41 +0800603 return 0;
604}
605
Todd Poynorc133b712013-08-14 17:39:13 -0700606status_t BatteryMonitor::getProperty(int id, struct BatteryProperty *val) {
607 status_t ret = BAD_VALUE;
Jin Qian72adf112017-02-02 17:31:13 -0800608 std::string buf;
Todd Poynorc133b712013-08-14 17:39:13 -0700609
Todd Poynor8f132af2014-05-08 17:15:45 -0700610 val->valueInt64 = LONG_MIN;
611
Todd Poynorc133b712013-08-14 17:39:13 -0700612 switch(id) {
613 case BATTERY_PROP_CHARGE_COUNTER:
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000614 if (!mHealthdConfig->batteryChargeCounterPath.empty()) {
Todd Poynor8f132af2014-05-08 17:15:45 -0700615 val->valueInt64 =
Todd Poynorc133b712013-08-14 17:39:13 -0700616 getIntField(mHealthdConfig->batteryChargeCounterPath);
Elliott Hughes643268f2018-10-08 11:10:11 -0700617 ret = OK;
Todd Poynorc133b712013-08-14 17:39:13 -0700618 } else {
619 ret = NAME_NOT_FOUND;
620 }
621 break;
622
623 case BATTERY_PROP_CURRENT_NOW:
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000624 if (!mHealthdConfig->batteryCurrentNowPath.empty()) {
Todd Poynor8f132af2014-05-08 17:15:45 -0700625 val->valueInt64 =
Todd Poynorc133b712013-08-14 17:39:13 -0700626 getIntField(mHealthdConfig->batteryCurrentNowPath);
Elliott Hughes643268f2018-10-08 11:10:11 -0700627 ret = OK;
Todd Poynorc133b712013-08-14 17:39:13 -0700628 } else {
629 ret = NAME_NOT_FOUND;
630 }
631 break;
632
Todd Poynorbc102112013-08-27 18:11:49 -0700633 case BATTERY_PROP_CURRENT_AVG:
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000634 if (!mHealthdConfig->batteryCurrentAvgPath.empty()) {
Todd Poynor8f132af2014-05-08 17:15:45 -0700635 val->valueInt64 =
Todd Poynorbc102112013-08-27 18:11:49 -0700636 getIntField(mHealthdConfig->batteryCurrentAvgPath);
Elliott Hughes643268f2018-10-08 11:10:11 -0700637 ret = OK;
Todd Poynorbc102112013-08-27 18:11:49 -0700638 } else {
639 ret = NAME_NOT_FOUND;
640 }
641 break;
642
Paul Lawrence347c8de2014-03-19 15:04:40 -0700643 case BATTERY_PROP_CAPACITY:
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000644 if (!mHealthdConfig->batteryCapacityPath.empty()) {
Todd Poynor8f132af2014-05-08 17:15:45 -0700645 val->valueInt64 =
Paul Lawrence347c8de2014-03-19 15:04:40 -0700646 getIntField(mHealthdConfig->batteryCapacityPath);
Elliott Hughes643268f2018-10-08 11:10:11 -0700647 ret = OK;
Paul Lawrence347c8de2014-03-19 15:04:40 -0700648 } else {
649 ret = NAME_NOT_FOUND;
650 }
651 break;
652
Todd Poynor8f132af2014-05-08 17:15:45 -0700653 case BATTERY_PROP_ENERGY_COUNTER:
Todd Poynore14b37e2014-05-20 13:54:40 -0700654 if (mHealthdConfig->energyCounter) {
655 ret = mHealthdConfig->energyCounter(&val->valueInt64);
656 } else {
657 ret = NAME_NOT_FOUND;
658 }
Todd Poynor8f132af2014-05-08 17:15:45 -0700659 break;
660
Jin Qian72adf112017-02-02 17:31:13 -0800661 case BATTERY_PROP_BATTERY_STATUS:
Todd Poynore030a102018-01-19 14:03:59 -0800662 val->valueInt64 = getChargeStatus();
Elliott Hughes643268f2018-10-08 11:10:11 -0700663 ret = OK;
Jin Qian72adf112017-02-02 17:31:13 -0800664 break;
665
Jack Wue561d032022-11-24 12:19:41 +0800666 case BATTERY_PROP_CHARGING_POLICY:
667 val->valueInt64 = getChargingPolicy();
668 ret = OK;
669 break;
670
671 case BATTERY_PROP_MANUFACTURING_DATE:
672 val->valueInt64 = getBatteryHealthData(BATTERY_PROP_MANUFACTURING_DATE);
673 ret = OK;
674 break;
675
676 case BATTERY_PROP_FIRST_USAGE_DATE:
677 val->valueInt64 = getBatteryHealthData(BATTERY_PROP_FIRST_USAGE_DATE);
678 ret = OK;
679 break;
680
AleX Pelosiff708922023-02-17 01:39:21 +0000681 case BATTERY_PROP_STATE_OF_HEALTH:
682 val->valueInt64 = getBatteryHealthData(BATTERY_PROP_STATE_OF_HEALTH);
683 ret = OK;
684 break;
685
David Andersond5ed26a2023-12-08 15:12:24 -0800686 case BATTERY_PROP_PART_STATUS:
687 val->valueInt64 = getBatteryHealthData(BATTERY_PROP_PART_STATUS);
688 ret = OK;
689 break;
690
Todd Poynorc133b712013-08-14 17:39:13 -0700691 default:
692 break;
693 }
694
Todd Poynorc133b712013-08-14 17:39:13 -0700695 return ret;
696}
697
David Andersond5ed26a2023-12-08 15:12:24 -0800698status_t BatteryMonitor::getSerialNumber(std::optional<std::string>* out) {
699 *out = std::nullopt;
700 return OK;
701}
702
Todd Poynor020369d2013-09-18 20:09:33 -0700703void BatteryMonitor::dumpState(int fd) {
704 int v;
705 char vs[128];
Yifan Hongb99d15c2022-03-01 12:12:34 -0800706 const HealthInfo& props = *mHealthInfo;
Todd Poynor020369d2013-09-18 20:09:33 -0700707
Jack Wu06b90412021-12-15 20:40:21 +0800708 snprintf(vs, sizeof(vs),
709 "ac: %d usb: %d wireless: %d dock: %d current_max: %d voltage_max: %d\n",
710 props.chargerAcOnline, props.chargerUsbOnline, props.chargerWirelessOnline,
Yifan Hongb99d15c2022-03-01 12:12:34 -0800711 props.chargerDockOnline, props.maxChargingCurrentMicroamps,
712 props.maxChargingVoltageMicrovolts);
Todd Poynor020369d2013-09-18 20:09:33 -0700713 write(fd, vs, strlen(vs));
714 snprintf(vs, sizeof(vs), "status: %d health: %d present: %d\n",
715 props.batteryStatus, props.batteryHealth, props.batteryPresent);
716 write(fd, vs, strlen(vs));
Yifan Hongb99d15c2022-03-01 12:12:34 -0800717 snprintf(vs, sizeof(vs), "level: %d voltage: %d temp: %d\n", props.batteryLevel,
718 props.batteryVoltageMillivolts, props.batteryTemperatureTenthsCelsius);
Todd Poynor020369d2013-09-18 20:09:33 -0700719 write(fd, vs, strlen(vs));
720
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000721 if (!mHealthdConfig->batteryCurrentNowPath.empty()) {
Todd Poynor020369d2013-09-18 20:09:33 -0700722 v = getIntField(mHealthdConfig->batteryCurrentNowPath);
723 snprintf(vs, sizeof(vs), "current now: %d\n", v);
724 write(fd, vs, strlen(vs));
725 }
726
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000727 if (!mHealthdConfig->batteryCurrentAvgPath.empty()) {
Todd Poynor020369d2013-09-18 20:09:33 -0700728 v = getIntField(mHealthdConfig->batteryCurrentAvgPath);
729 snprintf(vs, sizeof(vs), "current avg: %d\n", v);
730 write(fd, vs, strlen(vs));
731 }
732
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000733 if (!mHealthdConfig->batteryChargeCounterPath.empty()) {
Todd Poynor020369d2013-09-18 20:09:33 -0700734 v = getIntField(mHealthdConfig->batteryChargeCounterPath);
735 snprintf(vs, sizeof(vs), "charge counter: %d\n", v);
736 write(fd, vs, strlen(vs));
737 }
Ruchi Kandoicc338802015-08-24 13:01:16 -0700738
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000739 if (!mHealthdConfig->batteryCurrentNowPath.empty()) {
Yifan Hongb99d15c2022-03-01 12:12:34 -0800740 snprintf(vs, sizeof(vs), "current now: %d\n", props.batteryCurrentMicroamps);
Ruchi Kandoicc338802015-08-24 13:01:16 -0700741 write(fd, vs, strlen(vs));
742 }
743
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000744 if (!mHealthdConfig->batteryCycleCountPath.empty()) {
Ruchi Kandoicc338802015-08-24 13:01:16 -0700745 snprintf(vs, sizeof(vs), "cycle count: %d\n", props.batteryCycleCount);
746 write(fd, vs, strlen(vs));
747 }
748
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000749 if (!mHealthdConfig->batteryFullChargePath.empty()) {
Yifan Hongb99d15c2022-03-01 12:12:34 -0800750 snprintf(vs, sizeof(vs), "Full charge: %d\n", props.batteryFullChargeUah);
Ruchi Kandoicc338802015-08-24 13:01:16 -0700751 write(fd, vs, strlen(vs));
752 }
Todd Poynor020369d2013-09-18 20:09:33 -0700753}
754
Todd Poynorc7464c92013-09-10 12:40:00 -0700755void BatteryMonitor::init(struct healthd_config *hc) {
Todd Poynor752faf22013-06-12 13:25:59 -0700756 String8 path;
Todd Poynor3db03a52014-05-21 16:28:13 -0700757 char pval[PROPERTY_VALUE_MAX];
Todd Poynor752faf22013-06-12 13:25:59 -0700758
Todd Poynorf5d30122013-08-12 17:03:35 -0700759 mHealthdConfig = hc;
James Hawkins588a2ca2016-02-18 14:52:46 -0800760 std::unique_ptr<DIR, decltype(&closedir)> dir(opendir(POWER_SUPPLY_SYSFS_PATH), closedir);
Todd Poynor752faf22013-06-12 13:25:59 -0700761 if (dir == NULL) {
762 KLOG_ERROR(LOG_TAG, "Could not open %s\n", POWER_SUPPLY_SYSFS_PATH);
763 } else {
764 struct dirent* entry;
765
James Hawkins588a2ca2016-02-18 14:52:46 -0800766 while ((entry = readdir(dir.get()))) {
Todd Poynor752faf22013-06-12 13:25:59 -0700767 const char* name = entry->d_name;
768
769 if (!strcmp(name, ".") || !strcmp(name, ".."))
770 continue;
771
Bart Van Assche25b2a8d2022-02-24 21:51:34 +0000772 std::vector<String8>::iterator itIgnoreName =
773 find(hc->ignorePowerSupplyNames.begin(), hc->ignorePowerSupplyNames.end(),
774 String8(name));
Thierry Strudelf73de6f2019-01-11 17:09:20 -0800775 if (itIgnoreName != hc->ignorePowerSupplyNames.end())
776 continue;
777
Todd Poynor752faf22013-06-12 13:25:59 -0700778 // Look for "type" file in each subdirectory
779 path.clear();
780 path.appendFormat("%s/%s/type", POWER_SUPPLY_SYSFS_PATH, name);
781 switch(readPowerSupplyType(path)) {
782 case ANDROID_POWER_SUPPLY_TYPE_AC:
783 case ANDROID_POWER_SUPPLY_TYPE_USB:
784 case ANDROID_POWER_SUPPLY_TYPE_WIRELESS:
Jack Wu06b90412021-12-15 20:40:21 +0800785 case ANDROID_POWER_SUPPLY_TYPE_DOCK:
Todd Poynor752faf22013-06-12 13:25:59 -0700786 path.clear();
787 path.appendFormat("%s/%s/online", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk18b74612023-08-10 23:29:50 +0000788 if (access(path.c_str(), R_OK) == 0) mChargerNames.add(String8(name));
Todd Poynor752faf22013-06-12 13:25:59 -0700789 break;
790
791 case ANDROID_POWER_SUPPLY_TYPE_BATTERY:
Kazuhiro Inaba8e4d9822019-06-12 13:46:08 +0900792 // Some devices expose the battery status of sub-component like
793 // stylus. Such a device-scoped battery info needs to be skipped
794 // in BatteryMonitor, which is intended to report the status of
795 // the battery supplying the power to the whole system.
796 if (isScopedPowerSupply(name)) continue;
Todd Poynor6dcc45e2013-10-21 20:26:25 -0700797 mBatteryDevicePresent = true;
798
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000799 if (mHealthdConfig->batteryStatusPath.empty()) {
Todd Poynor752faf22013-06-12 13:25:59 -0700800 path.clear();
Todd Poynorf5d30122013-08-12 17:03:35 -0700801 path.appendFormat("%s/%s/status", POWER_SUPPLY_SYSFS_PATH,
802 name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000803 if (access(path.c_str(), R_OK) == 0) mHealthdConfig->batteryStatusPath = path;
Todd Poynor752faf22013-06-12 13:25:59 -0700804 }
805
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000806 if (mHealthdConfig->batteryHealthPath.empty()) {
Todd Poynor752faf22013-06-12 13:25:59 -0700807 path.clear();
Todd Poynorf5d30122013-08-12 17:03:35 -0700808 path.appendFormat("%s/%s/health", POWER_SUPPLY_SYSFS_PATH,
809 name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000810 if (access(path.c_str(), R_OK) == 0) mHealthdConfig->batteryHealthPath = path;
Todd Poynor752faf22013-06-12 13:25:59 -0700811 }
812
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000813 if (mHealthdConfig->batteryPresentPath.empty()) {
Todd Poynorf5d30122013-08-12 17:03:35 -0700814 path.clear();
815 path.appendFormat("%s/%s/present", POWER_SUPPLY_SYSFS_PATH,
816 name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000817 if (access(path.c_str(), R_OK) == 0) mHealthdConfig->batteryPresentPath = path;
Todd Poynorf5d30122013-08-12 17:03:35 -0700818 }
819
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000820 if (mHealthdConfig->batteryCapacityPath.empty()) {
Todd Poynorf5d30122013-08-12 17:03:35 -0700821 path.clear();
822 path.appendFormat("%s/%s/capacity", POWER_SUPPLY_SYSFS_PATH,
823 name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000824 if (access(path.c_str(), R_OK) == 0) mHealthdConfig->batteryCapacityPath = path;
Todd Poynorf5d30122013-08-12 17:03:35 -0700825 }
826
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000827 if (mHealthdConfig->batteryVoltagePath.empty()) {
Todd Poynorf5d30122013-08-12 17:03:35 -0700828 path.clear();
829 path.appendFormat("%s/%s/voltage_now",
830 POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000831 if (access(path.c_str(), R_OK) == 0) {
Todd Poynorf5d30122013-08-12 17:03:35 -0700832 mHealthdConfig->batteryVoltagePath = path;
Todd Poynorf5d30122013-08-12 17:03:35 -0700833 }
834 }
835
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000836 if (mHealthdConfig->batteryFullChargePath.empty()) {
Ruchi Kandoicc338802015-08-24 13:01:16 -0700837 path.clear();
838 path.appendFormat("%s/%s/charge_full",
839 POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000840 if (access(path.c_str(), R_OK) == 0)
Ruchi Kandoicc338802015-08-24 13:01:16 -0700841 mHealthdConfig->batteryFullChargePath = path;
842 }
843
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000844 if (mHealthdConfig->batteryCurrentNowPath.empty()) {
Todd Poynorf5d30122013-08-12 17:03:35 -0700845 path.clear();
846 path.appendFormat("%s/%s/current_now",
847 POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000848 if (access(path.c_str(), R_OK) == 0)
Todd Poynorf5d30122013-08-12 17:03:35 -0700849 mHealthdConfig->batteryCurrentNowPath = path;
850 }
851
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000852 if (mHealthdConfig->batteryCycleCountPath.empty()) {
Ruchi Kandoicc338802015-08-24 13:01:16 -0700853 path.clear();
854 path.appendFormat("%s/%s/cycle_count",
855 POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000856 if (access(path.c_str(), R_OK) == 0)
Ruchi Kandoicc338802015-08-24 13:01:16 -0700857 mHealthdConfig->batteryCycleCountPath = path;
858 }
859
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000860 if (mHealthdConfig->batteryCapacityLevelPath.empty()) {
Stephane Lee86f9f6a2019-12-19 15:09:41 -0800861 path.clear();
862 path.appendFormat("%s/%s/capacity_level", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000863 if (access(path.c_str(), R_OK) == 0) {
864 mHealthdConfig->batteryCapacityLevelPath = path;
865 }
Stephane Lee86f9f6a2019-12-19 15:09:41 -0800866 }
867
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000868 if (mHealthdConfig->batteryChargeTimeToFullNowPath.empty()) {
Stephane Lee86f9f6a2019-12-19 15:09:41 -0800869 path.clear();
870 path.appendFormat("%s/%s/time_to_full_now", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000871 if (access(path.c_str(), R_OK) == 0)
Stephane Lee86f9f6a2019-12-19 15:09:41 -0800872 mHealthdConfig->batteryChargeTimeToFullNowPath = path;
873 }
874
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000875 if (mHealthdConfig->batteryFullChargeDesignCapacityUahPath.empty()) {
Stephane Lee1c108ed2020-02-10 18:23:57 -0800876 path.clear();
877 path.appendFormat("%s/%s/charge_full_design", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000878 if (access(path.c_str(), R_OK) == 0)
Stephane Lee1c108ed2020-02-10 18:23:57 -0800879 mHealthdConfig->batteryFullChargeDesignCapacityUahPath = path;
880 }
881
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000882 if (mHealthdConfig->batteryCurrentAvgPath.empty()) {
Todd Poynorbc102112013-08-27 18:11:49 -0700883 path.clear();
884 path.appendFormat("%s/%s/current_avg",
885 POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000886 if (access(path.c_str(), R_OK) == 0)
Todd Poynorbc102112013-08-27 18:11:49 -0700887 mHealthdConfig->batteryCurrentAvgPath = path;
888 }
889
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000890 if (mHealthdConfig->batteryChargeCounterPath.empty()) {
Todd Poynorf5d30122013-08-12 17:03:35 -0700891 path.clear();
892 path.appendFormat("%s/%s/charge_counter",
893 POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000894 if (access(path.c_str(), R_OK) == 0)
Todd Poynorf5d30122013-08-12 17:03:35 -0700895 mHealthdConfig->batteryChargeCounterPath = path;
896 }
897
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000898 if (mHealthdConfig->batteryTemperaturePath.empty()) {
Todd Poynorf5d30122013-08-12 17:03:35 -0700899 path.clear();
900 path.appendFormat("%s/%s/temp", POWER_SUPPLY_SYSFS_PATH,
901 name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000902 if (access(path.c_str(), R_OK) == 0) {
Todd Poynorf5d30122013-08-12 17:03:35 -0700903 mHealthdConfig->batteryTemperaturePath = path;
Todd Poynorf5d30122013-08-12 17:03:35 -0700904 }
905 }
906
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000907 if (mHealthdConfig->batteryTechnologyPath.empty()) {
Todd Poynorf5d30122013-08-12 17:03:35 -0700908 path.clear();
909 path.appendFormat("%s/%s/technology",
910 POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000911 if (access(path.c_str(), R_OK) == 0)
Todd Poynorf5d30122013-08-12 17:03:35 -0700912 mHealthdConfig->batteryTechnologyPath = path;
913 }
914
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000915 if (mHealthdConfig->batteryStateOfHealthPath.empty()) {
Jack Wue561d032022-11-24 12:19:41 +0800916 path.clear();
917 path.appendFormat("%s/%s/state_of_health", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000918 if (access(path.c_str(), R_OK) == 0) {
Jack Wue561d032022-11-24 12:19:41 +0800919 mHealthdConfig->batteryStateOfHealthPath = path;
920 } else {
921 path.clear();
922 path.appendFormat("%s/%s/health_index", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000923 if (access(path.c_str(), R_OK) == 0)
Jack Wue561d032022-11-24 12:19:41 +0800924 mHealthdConfig->batteryStateOfHealthPath = path;
925 }
926 }
927
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000928 if (mHealthdConfig->batteryHealthStatusPath.empty()) {
Jack Wub57f68a2023-02-04 19:56:06 +0800929 path.clear();
930 path.appendFormat("%s/%s/health_status", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000931 if (access(path.c_str(), R_OK) == 0) {
932 mHealthdConfig->batteryHealthStatusPath = path;
933 }
Jack Wub57f68a2023-02-04 19:56:06 +0800934 }
935
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000936 if (mHealthdConfig->batteryManufacturingDatePath.empty()) {
Jack Wue561d032022-11-24 12:19:41 +0800937 path.clear();
938 path.appendFormat("%s/%s/manufacturing_date", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000939 if (access(path.c_str(), R_OK) == 0)
Jack Wue561d032022-11-24 12:19:41 +0800940 mHealthdConfig->batteryManufacturingDatePath = path;
941 }
942
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000943 if (mHealthdConfig->batteryFirstUsageDatePath.empty()) {
Jack Wue561d032022-11-24 12:19:41 +0800944 path.clear();
945 path.appendFormat("%s/%s/first_usage_date", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000946 if (access(path.c_str(), R_OK) == 0) {
947 mHealthdConfig->batteryFirstUsageDatePath = path;
948 }
Jack Wue561d032022-11-24 12:19:41 +0800949 }
950
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000951 if (mHealthdConfig->chargingStatePath.empty()) {
Jack Wue561d032022-11-24 12:19:41 +0800952 path.clear();
953 path.appendFormat("%s/%s/charging_state", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000954 if (access(path.c_str(), R_OK) == 0) mHealthdConfig->chargingStatePath = path;
Jack Wue561d032022-11-24 12:19:41 +0800955 }
956
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000957 if (mHealthdConfig->chargingPolicyPath.empty()) {
Jack Wue561d032022-11-24 12:19:41 +0800958 path.clear();
959 path.appendFormat("%s/%s/charging_policy", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk2b1a0592023-09-12 15:26:15 +0000960 if (access(path.c_str(), R_OK) == 0) mHealthdConfig->chargingPolicyPath = path;
Jack Wue561d032022-11-24 12:19:41 +0800961 }
962
Todd Poynor752faf22013-06-12 13:25:59 -0700963 break;
964
965 case ANDROID_POWER_SUPPLY_TYPE_UNKNOWN:
966 break;
967 }
Jack Wu06b90412021-12-15 20:40:21 +0800968
969 // Look for "is_dock" file
970 path.clear();
971 path.appendFormat("%s/%s/is_dock", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk18b74612023-08-10 23:29:50 +0000972 if (access(path.c_str(), R_OK) == 0) {
Jack Wu06b90412021-12-15 20:40:21 +0800973 path.clear();
974 path.appendFormat("%s/%s/online", POWER_SUPPLY_SYSFS_PATH, name);
Tomasz Wasilczyk18b74612023-08-10 23:29:50 +0000975 if (access(path.c_str(), R_OK) == 0) mChargerNames.add(String8(name));
Jack Wu06b90412021-12-15 20:40:21 +0800976 }
Todd Poynor752faf22013-06-12 13:25:59 -0700977 }
Todd Poynor752faf22013-06-12 13:25:59 -0700978 }
979
Ian Pedowitz585ab652015-10-12 19:01:00 -0700980 // Typically the case for devices which do not have a battery and
981 // and are always plugged into AC mains.
Todd Poynor6dcc45e2013-10-21 20:26:25 -0700982 if (!mBatteryDevicePresent) {
Todd Poynorebeb0c02014-09-23 14:54:24 -0700983 KLOG_WARNING(LOG_TAG, "No battery devices found\n");
Todd Poynor6dcc45e2013-10-21 20:26:25 -0700984 hc->periodic_chores_interval_fast = -1;
985 hc->periodic_chores_interval_slow = -1;
986 } else {
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000987 if (mHealthdConfig->batteryStatusPath.empty())
Todd Poynor6dcc45e2013-10-21 20:26:25 -0700988 KLOG_WARNING(LOG_TAG, "BatteryStatusPath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000989 if (mHealthdConfig->batteryHealthPath.empty())
Todd Poynor6dcc45e2013-10-21 20:26:25 -0700990 KLOG_WARNING(LOG_TAG, "BatteryHealthPath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000991 if (mHealthdConfig->batteryPresentPath.empty())
Todd Poynor6dcc45e2013-10-21 20:26:25 -0700992 KLOG_WARNING(LOG_TAG, "BatteryPresentPath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000993 if (mHealthdConfig->batteryCapacityPath.empty())
Todd Poynor6dcc45e2013-10-21 20:26:25 -0700994 KLOG_WARNING(LOG_TAG, "BatteryCapacityPath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000995 if (mHealthdConfig->batteryVoltagePath.empty())
Todd Poynor6dcc45e2013-10-21 20:26:25 -0700996 KLOG_WARNING(LOG_TAG, "BatteryVoltagePath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000997 if (mHealthdConfig->batteryTemperaturePath.empty())
Todd Poynor6dcc45e2013-10-21 20:26:25 -0700998 KLOG_WARNING(LOG_TAG, "BatteryTemperaturePath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +0000999 if (mHealthdConfig->batteryTechnologyPath.empty())
Todd Poynor6dcc45e2013-10-21 20:26:25 -07001000 KLOG_WARNING(LOG_TAG, "BatteryTechnologyPath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001001 if (mHealthdConfig->batteryCurrentNowPath.empty())
Ruchi Kandoicc338802015-08-24 13:01:16 -07001002 KLOG_WARNING(LOG_TAG, "BatteryCurrentNowPath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001003 if (mHealthdConfig->batteryFullChargePath.empty())
Ruchi Kandoicc338802015-08-24 13:01:16 -07001004 KLOG_WARNING(LOG_TAG, "BatteryFullChargePath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001005 if (mHealthdConfig->batteryCycleCountPath.empty())
Ruchi Kandoicc338802015-08-24 13:01:16 -07001006 KLOG_WARNING(LOG_TAG, "BatteryCycleCountPath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001007 if (mHealthdConfig->batteryCapacityLevelPath.empty())
Stephane Lee86f9f6a2019-12-19 15:09:41 -08001008 KLOG_WARNING(LOG_TAG, "batteryCapacityLevelPath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001009 if (mHealthdConfig->batteryChargeTimeToFullNowPath.empty())
Stephane Lee86f9f6a2019-12-19 15:09:41 -08001010 KLOG_WARNING(LOG_TAG, "batteryChargeTimeToFullNowPath. not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001011 if (mHealthdConfig->batteryFullChargeDesignCapacityUahPath.empty())
Stephane Lee1c108ed2020-02-10 18:23:57 -08001012 KLOG_WARNING(LOG_TAG, "batteryFullChargeDesignCapacityUahPath. not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001013 if (mHealthdConfig->batteryStateOfHealthPath.empty())
Jack Wue561d032022-11-24 12:19:41 +08001014 KLOG_WARNING(LOG_TAG, "batteryStateOfHealthPath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001015 if (mHealthdConfig->batteryHealthStatusPath.empty())
Jack Wub57f68a2023-02-04 19:56:06 +08001016 KLOG_WARNING(LOG_TAG, "batteryHealthStatusPath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001017 if (mHealthdConfig->batteryManufacturingDatePath.empty())
Jack Wue561d032022-11-24 12:19:41 +08001018 KLOG_WARNING(LOG_TAG, "batteryManufacturingDatePath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001019 if (mHealthdConfig->batteryFirstUsageDatePath.empty())
Jack Wue561d032022-11-24 12:19:41 +08001020 KLOG_WARNING(LOG_TAG, "batteryFirstUsageDatePath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001021 if (mHealthdConfig->chargingStatePath.empty())
Jack Wue561d032022-11-24 12:19:41 +08001022 KLOG_WARNING(LOG_TAG, "chargingStatePath not found\n");
Tomasz Wasilczykf5971292023-08-14 18:18:26 +00001023 if (mHealthdConfig->chargingPolicyPath.empty())
Jack Wue561d032022-11-24 12:19:41 +08001024 KLOG_WARNING(LOG_TAG, "chargingPolicyPath not found\n");
Todd Poynor6dcc45e2013-10-21 20:26:25 -07001025 }
Todd Poynor3db03a52014-05-21 16:28:13 -07001026
Ruchi Kandoia78fc232014-07-10 15:06:21 -07001027 if (property_get("ro.boot.fake_battery", pval, NULL) > 0
1028 && strtol(pval, NULL, 10) != 0) {
1029 mBatteryFixedCapacity = FAKE_BATTERY_CAPACITY;
1030 mBatteryFixedTemperature = FAKE_BATTERY_TEMPERATURE;
1031 }
Todd Poynor752faf22013-06-12 13:25:59 -07001032}
1033
1034}; // namespace android