1 /* 2 * Copyright (C) 2009 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 package com.android.internal.os; 18 19 import android.content.Context; 20 import android.content.Intent; 21 import android.content.IntentFilter; 22 import android.content.pm.PackageManager; 23 import android.content.res.Resources; 24 import android.hardware.SensorManager; 25 import android.net.ConnectivityManager; 26 import android.os.BatteryStats; 27 import android.os.BatteryStats.Uid; 28 import android.os.Bundle; 29 import android.os.MemoryFile; 30 import android.os.Parcel; 31 import android.os.ParcelFileDescriptor; 32 import android.os.Process; 33 import android.os.RemoteException; 34 import android.os.ServiceManager; 35 import android.os.SystemClock; 36 import android.os.UserHandle; 37 import android.text.format.DateUtils; 38 import android.util.ArrayMap; 39 import android.util.Log; 40 import android.util.SparseArray; 41 import android.util.SparseLongArray; 42 43 import com.android.internal.annotations.VisibleForTesting; 44 import com.android.internal.app.IBatteryStats; 45 import com.android.internal.os.BatterySipper.DrainType; 46 import com.android.internal.util.ArrayUtils; 47 48 import java.io.File; 49 import java.io.FileInputStream; 50 import java.io.FileOutputStream; 51 import java.io.IOException; 52 import java.util.ArrayList; 53 import java.util.Collections; 54 import java.util.Comparator; 55 import java.util.List; 56 import java.util.Locale; 57 58 /** 59 * A helper class for retrieving the power usage information for all applications and services. 60 * 61 * The caller must initialize this class as soon as activity object is ready to use (for example, in 62 * onAttach() for Fragment), call create() in onCreate() and call destroy() in onDestroy(). 63 */ 64 public class BatteryStatsHelper { 65 static final boolean DEBUG = false; 66 67 private static final String TAG = BatteryStatsHelper.class.getSimpleName(); 68 69 private static BatteryStats sStatsXfer; 70 private static Intent sBatteryBroadcastXfer; 71 private static ArrayMap<File, BatteryStats> sFileXfer = new ArrayMap<>(); 72 73 final private Context mContext; 74 final private boolean mCollectBatteryBroadcast; 75 final private boolean mWifiOnly; 76 77 private IBatteryStats mBatteryInfo; 78 private BatteryStats mStats; 79 private Intent mBatteryBroadcast; 80 private PowerProfile mPowerProfile; 81 82 private String[] mSystemPackageArray; 83 private String[] mServicepackageArray; 84 private PackageManager mPackageManager; 85 86 /** 87 * List of apps using power. 88 */ 89 private final List<BatterySipper> mUsageList = new ArrayList<>(); 90 91 /** 92 * List of apps using wifi power. 93 */ 94 private final List<BatterySipper> mWifiSippers = new ArrayList<>(); 95 96 /** 97 * List of apps using bluetooth power. 98 */ 99 private final List<BatterySipper> mBluetoothSippers = new ArrayList<>(); 100 101 private final SparseArray<List<BatterySipper>> mUserSippers = new SparseArray<>(); 102 103 private final List<BatterySipper> mMobilemsppList = new ArrayList<>(); 104 105 private int mStatsType = BatteryStats.STATS_SINCE_CHARGED; 106 107 long mRawRealtimeUs; 108 long mRawUptimeUs; 109 long mBatteryRealtimeUs; 110 long mBatteryUptimeUs; 111 long mTypeBatteryRealtimeUs; 112 long mTypeBatteryUptimeUs; 113 long mBatteryTimeRemainingUs; 114 long mChargeTimeRemainingUs; 115 116 private long mStatsPeriod = 0; 117 118 // The largest entry by power. 119 private double mMaxPower = 1; 120 121 // The largest real entry by power (not undercounted or overcounted). 122 private double mMaxRealPower = 1; 123 124 // Total computed power. 125 private double mComputedPower; 126 private double mTotalPower; 127 private double mMinDrainedPower; 128 private double mMaxDrainedPower; 129 130 PowerCalculator mCpuPowerCalculator; 131 PowerCalculator mWakelockPowerCalculator; 132 MobileRadioPowerCalculator mMobileRadioPowerCalculator; 133 PowerCalculator mWifiPowerCalculator; 134 PowerCalculator mBluetoothPowerCalculator; 135 PowerCalculator mSensorPowerCalculator; 136 PowerCalculator mCameraPowerCalculator; 137 PowerCalculator mFlashlightPowerCalculator; 138 PowerCalculator mMemoryPowerCalculator; 139 140 boolean mHasWifiPowerReporting = false; 141 boolean mHasBluetoothPowerReporting = false; 142 checkWifiOnly(Context context)143 public static boolean checkWifiOnly(Context context) { 144 ConnectivityManager cm = (ConnectivityManager) context.getSystemService( 145 Context.CONNECTIVITY_SERVICE); 146 return !cm.isNetworkSupported(ConnectivityManager.TYPE_MOBILE); 147 } 148 checkHasWifiPowerReporting(BatteryStats stats, PowerProfile profile)149 public static boolean checkHasWifiPowerReporting(BatteryStats stats, PowerProfile profile) { 150 return stats.hasWifiActivityReporting() && 151 profile.getAveragePower(PowerProfile.POWER_WIFI_CONTROLLER_IDLE) != 0 && 152 profile.getAveragePower(PowerProfile.POWER_WIFI_CONTROLLER_RX) != 0 && 153 profile.getAveragePower(PowerProfile.POWER_WIFI_CONTROLLER_TX) != 0; 154 } 155 checkHasBluetoothPowerReporting(BatteryStats stats, PowerProfile profile)156 public static boolean checkHasBluetoothPowerReporting(BatteryStats stats, 157 PowerProfile profile) { 158 return stats.hasBluetoothActivityReporting() && 159 profile.getAveragePower(PowerProfile.POWER_BLUETOOTH_CONTROLLER_IDLE) != 0 && 160 profile.getAveragePower(PowerProfile.POWER_BLUETOOTH_CONTROLLER_RX) != 0 && 161 profile.getAveragePower(PowerProfile.POWER_BLUETOOTH_CONTROLLER_TX) != 0; 162 } 163 BatteryStatsHelper(Context context)164 public BatteryStatsHelper(Context context) { 165 this(context, true); 166 } 167 BatteryStatsHelper(Context context, boolean collectBatteryBroadcast)168 public BatteryStatsHelper(Context context, boolean collectBatteryBroadcast) { 169 this(context, collectBatteryBroadcast, checkWifiOnly(context)); 170 } 171 BatteryStatsHelper(Context context, boolean collectBatteryBroadcast, boolean wifiOnly)172 public BatteryStatsHelper(Context context, boolean collectBatteryBroadcast, boolean wifiOnly) { 173 mContext = context; 174 mCollectBatteryBroadcast = collectBatteryBroadcast; 175 mWifiOnly = wifiOnly; 176 mPackageManager = context.getPackageManager(); 177 178 final Resources resources = context.getResources(); 179 mSystemPackageArray = resources.getStringArray( 180 com.android.internal.R.array.config_batteryPackageTypeSystem); 181 mServicepackageArray = resources.getStringArray( 182 com.android.internal.R.array.config_batteryPackageTypeService); 183 } 184 storeStatsHistoryInFile(String fname)185 public void storeStatsHistoryInFile(String fname) { 186 synchronized (sFileXfer) { 187 File path = makeFilePath(mContext, fname); 188 sFileXfer.put(path, this.getStats()); 189 FileOutputStream fout = null; 190 try { 191 fout = new FileOutputStream(path); 192 Parcel hist = Parcel.obtain(); 193 getStats().writeToParcelWithoutUids(hist, 0); 194 byte[] histData = hist.marshall(); 195 fout.write(histData); 196 } catch (IOException e) { 197 Log.w(TAG, "Unable to write history to file", e); 198 } finally { 199 if (fout != null) { 200 try { 201 fout.close(); 202 } catch (IOException e) { 203 } 204 } 205 } 206 } 207 } 208 statsFromFile(Context context, String fname)209 public static BatteryStats statsFromFile(Context context, String fname) { 210 synchronized (sFileXfer) { 211 File path = makeFilePath(context, fname); 212 BatteryStats stats = sFileXfer.get(path); 213 if (stats != null) { 214 return stats; 215 } 216 FileInputStream fin = null; 217 try { 218 fin = new FileInputStream(path); 219 byte[] data = readFully(fin); 220 Parcel parcel = Parcel.obtain(); 221 parcel.unmarshall(data, 0, data.length); 222 parcel.setDataPosition(0); 223 return com.android.internal.os.BatteryStatsImpl.CREATOR.createFromParcel(parcel); 224 } catch (IOException e) { 225 Log.w(TAG, "Unable to read history to file", e); 226 } finally { 227 if (fin != null) { 228 try { 229 fin.close(); 230 } catch (IOException e) { 231 } 232 } 233 } 234 } 235 return getStats(IBatteryStats.Stub.asInterface( 236 ServiceManager.getService(BatteryStats.SERVICE_NAME))); 237 } 238 dropFile(Context context, String fname)239 public static void dropFile(Context context, String fname) { 240 makeFilePath(context, fname).delete(); 241 } 242 makeFilePath(Context context, String fname)243 private static File makeFilePath(Context context, String fname) { 244 return new File(context.getFilesDir(), fname); 245 } 246 247 /** Clears the current stats and forces recreating for future use. */ clearStats()248 public void clearStats() { 249 mStats = null; 250 } 251 getStats()252 public BatteryStats getStats() { 253 if (mStats == null) { 254 load(); 255 } 256 return mStats; 257 } 258 getBatteryBroadcast()259 public Intent getBatteryBroadcast() { 260 if (mBatteryBroadcast == null && mCollectBatteryBroadcast) { 261 load(); 262 } 263 return mBatteryBroadcast; 264 } 265 getPowerProfile()266 public PowerProfile getPowerProfile() { 267 return mPowerProfile; 268 } 269 create(BatteryStats stats)270 public void create(BatteryStats stats) { 271 mPowerProfile = new PowerProfile(mContext); 272 mStats = stats; 273 } 274 create(Bundle icicle)275 public void create(Bundle icicle) { 276 if (icicle != null) { 277 mStats = sStatsXfer; 278 mBatteryBroadcast = sBatteryBroadcastXfer; 279 } 280 mBatteryInfo = IBatteryStats.Stub.asInterface( 281 ServiceManager.getService(BatteryStats.SERVICE_NAME)); 282 mPowerProfile = new PowerProfile(mContext); 283 } 284 storeState()285 public void storeState() { 286 sStatsXfer = mStats; 287 sBatteryBroadcastXfer = mBatteryBroadcast; 288 } 289 makemAh(double power)290 public static String makemAh(double power) { 291 if (power == 0) return "0"; 292 293 final String format; 294 if (power < .00001) { 295 format = "%.8f"; 296 } else if (power < .0001) { 297 format = "%.7f"; 298 } else if (power < .001) { 299 format = "%.6f"; 300 } else if (power < .01) { 301 format = "%.5f"; 302 } else if (power < .1) { 303 format = "%.4f"; 304 } else if (power < 1) { 305 format = "%.3f"; 306 } else if (power < 10) { 307 format = "%.2f"; 308 } else if (power < 100) { 309 format = "%.1f"; 310 } else { 311 format = "%.0f"; 312 } 313 314 // Use English locale because this is never used in UI (only in checkin and dump). 315 return String.format(Locale.ENGLISH, format, power); 316 } 317 318 /** 319 * Refreshes the power usage list. 320 */ refreshStats(int statsType, int asUser)321 public void refreshStats(int statsType, int asUser) { 322 SparseArray<UserHandle> users = new SparseArray<>(1); 323 users.put(asUser, new UserHandle(asUser)); 324 refreshStats(statsType, users); 325 } 326 327 /** 328 * Refreshes the power usage list. 329 */ refreshStats(int statsType, List<UserHandle> asUsers)330 public void refreshStats(int statsType, List<UserHandle> asUsers) { 331 final int n = asUsers.size(); 332 SparseArray<UserHandle> users = new SparseArray<>(n); 333 for (int i = 0; i < n; ++i) { 334 UserHandle userHandle = asUsers.get(i); 335 users.put(userHandle.getIdentifier(), userHandle); 336 } 337 refreshStats(statsType, users); 338 } 339 340 /** 341 * Refreshes the power usage list. 342 */ refreshStats(int statsType, SparseArray<UserHandle> asUsers)343 public void refreshStats(int statsType, SparseArray<UserHandle> asUsers) { 344 refreshStats(statsType, asUsers, SystemClock.elapsedRealtime() * 1000, 345 SystemClock.uptimeMillis() * 1000); 346 } 347 refreshStats(int statsType, SparseArray<UserHandle> asUsers, long rawRealtimeUs, long rawUptimeUs)348 public void refreshStats(int statsType, SparseArray<UserHandle> asUsers, long rawRealtimeUs, 349 long rawUptimeUs) { 350 // Initialize mStats if necessary. 351 getStats(); 352 353 mMaxPower = 0; 354 mMaxRealPower = 0; 355 mComputedPower = 0; 356 mTotalPower = 0; 357 358 mUsageList.clear(); 359 mWifiSippers.clear(); 360 mBluetoothSippers.clear(); 361 mUserSippers.clear(); 362 mMobilemsppList.clear(); 363 364 if (mStats == null) { 365 return; 366 } 367 368 if (mCpuPowerCalculator == null) { 369 mCpuPowerCalculator = new CpuPowerCalculator(mPowerProfile); 370 } 371 mCpuPowerCalculator.reset(); 372 373 if (mMemoryPowerCalculator == null) { 374 mMemoryPowerCalculator = new MemoryPowerCalculator(mPowerProfile); 375 } 376 mMemoryPowerCalculator.reset(); 377 378 if (mWakelockPowerCalculator == null) { 379 mWakelockPowerCalculator = new WakelockPowerCalculator(mPowerProfile); 380 } 381 mWakelockPowerCalculator.reset(); 382 383 if (mMobileRadioPowerCalculator == null) { 384 mMobileRadioPowerCalculator = new MobileRadioPowerCalculator(mPowerProfile, mStats); 385 } 386 mMobileRadioPowerCalculator.reset(mStats); 387 388 // checkHasWifiPowerReporting can change if we get energy data at a later point, so 389 // always check this field. 390 final boolean hasWifiPowerReporting = checkHasWifiPowerReporting(mStats, mPowerProfile); 391 if (mWifiPowerCalculator == null || hasWifiPowerReporting != mHasWifiPowerReporting) { 392 mWifiPowerCalculator = hasWifiPowerReporting ? 393 new WifiPowerCalculator(mPowerProfile) : 394 new WifiPowerEstimator(mPowerProfile); 395 mHasWifiPowerReporting = hasWifiPowerReporting; 396 } 397 mWifiPowerCalculator.reset(); 398 399 final boolean hasBluetoothPowerReporting = checkHasBluetoothPowerReporting(mStats, 400 mPowerProfile); 401 if (mBluetoothPowerCalculator == null || 402 hasBluetoothPowerReporting != mHasBluetoothPowerReporting) { 403 mBluetoothPowerCalculator = new BluetoothPowerCalculator(mPowerProfile); 404 mHasBluetoothPowerReporting = hasBluetoothPowerReporting; 405 } 406 mBluetoothPowerCalculator.reset(); 407 408 if (mSensorPowerCalculator == null) { 409 mSensorPowerCalculator = new SensorPowerCalculator(mPowerProfile, 410 (SensorManager) mContext.getSystemService(Context.SENSOR_SERVICE)); 411 } 412 mSensorPowerCalculator.reset(); 413 414 if (mCameraPowerCalculator == null) { 415 mCameraPowerCalculator = new CameraPowerCalculator(mPowerProfile); 416 } 417 mCameraPowerCalculator.reset(); 418 419 if (mFlashlightPowerCalculator == null) { 420 mFlashlightPowerCalculator = new FlashlightPowerCalculator(mPowerProfile); 421 } 422 mFlashlightPowerCalculator.reset(); 423 424 mStatsType = statsType; 425 mRawUptimeUs = rawUptimeUs; 426 mRawRealtimeUs = rawRealtimeUs; 427 mBatteryUptimeUs = mStats.getBatteryUptime(rawUptimeUs); 428 mBatteryRealtimeUs = mStats.getBatteryRealtime(rawRealtimeUs); 429 mTypeBatteryUptimeUs = mStats.computeBatteryUptime(rawUptimeUs, mStatsType); 430 mTypeBatteryRealtimeUs = mStats.computeBatteryRealtime(rawRealtimeUs, mStatsType); 431 mBatteryTimeRemainingUs = mStats.computeBatteryTimeRemaining(rawRealtimeUs); 432 mChargeTimeRemainingUs = mStats.computeChargeTimeRemaining(rawRealtimeUs); 433 434 if (DEBUG) { 435 Log.d(TAG, "Raw time: realtime=" + (rawRealtimeUs / 1000) + " uptime=" 436 + (rawUptimeUs / 1000)); 437 Log.d(TAG, "Battery time: realtime=" + (mBatteryRealtimeUs / 1000) + " uptime=" 438 + (mBatteryUptimeUs / 1000)); 439 Log.d(TAG, "Battery type time: realtime=" + (mTypeBatteryRealtimeUs / 1000) + " uptime=" 440 + (mTypeBatteryUptimeUs / 1000)); 441 } 442 mMinDrainedPower = (mStats.getLowDischargeAmountSinceCharge() 443 * mPowerProfile.getBatteryCapacity()) / 100; 444 mMaxDrainedPower = (mStats.getHighDischargeAmountSinceCharge() 445 * mPowerProfile.getBatteryCapacity()) / 100; 446 447 processAppUsage(asUsers); 448 449 // Before aggregating apps in to users, collect all apps to sort by their ms per packet. 450 for (int i = 0; i < mUsageList.size(); i++) { 451 BatterySipper bs = mUsageList.get(i); 452 bs.computeMobilemspp(); 453 if (bs.mobilemspp != 0) { 454 mMobilemsppList.add(bs); 455 } 456 } 457 458 for (int i = 0; i < mUserSippers.size(); i++) { 459 List<BatterySipper> user = mUserSippers.valueAt(i); 460 for (int j = 0; j < user.size(); j++) { 461 BatterySipper bs = user.get(j); 462 bs.computeMobilemspp(); 463 if (bs.mobilemspp != 0) { 464 mMobilemsppList.add(bs); 465 } 466 } 467 } 468 Collections.sort(mMobilemsppList, new Comparator<BatterySipper>() { 469 @Override 470 public int compare(BatterySipper lhs, BatterySipper rhs) { 471 return Double.compare(rhs.mobilemspp, lhs.mobilemspp); 472 } 473 }); 474 475 processMiscUsage(); 476 477 Collections.sort(mUsageList); 478 479 // At this point, we've sorted the list so we are guaranteed the max values are at the top. 480 // We have only added real powers so far. 481 if (!mUsageList.isEmpty()) { 482 mMaxRealPower = mMaxPower = mUsageList.get(0).totalPowerMah; 483 final int usageListCount = mUsageList.size(); 484 for (int i = 0; i < usageListCount; i++) { 485 mComputedPower += mUsageList.get(i).totalPowerMah; 486 } 487 } 488 489 if (DEBUG) { 490 Log.d(TAG, "Accuracy: total computed=" + makemAh(mComputedPower) + ", min discharge=" 491 + makemAh(mMinDrainedPower) + ", max discharge=" + makemAh(mMaxDrainedPower)); 492 } 493 494 mTotalPower = mComputedPower; 495 if (mStats.getLowDischargeAmountSinceCharge() > 1) { 496 if (mMinDrainedPower > mComputedPower) { 497 double amount = mMinDrainedPower - mComputedPower; 498 mTotalPower = mMinDrainedPower; 499 BatterySipper bs = new BatterySipper(DrainType.UNACCOUNTED, null, amount); 500 501 // Insert the BatterySipper in its sorted position. 502 int index = Collections.binarySearch(mUsageList, bs); 503 if (index < 0) { 504 index = -(index + 1); 505 } 506 mUsageList.add(index, bs); 507 mMaxPower = Math.max(mMaxPower, amount); 508 } else if (mMaxDrainedPower < mComputedPower) { 509 double amount = mComputedPower - mMaxDrainedPower; 510 511 // Insert the BatterySipper in its sorted position. 512 BatterySipper bs = new BatterySipper(DrainType.OVERCOUNTED, null, amount); 513 int index = Collections.binarySearch(mUsageList, bs); 514 if (index < 0) { 515 index = -(index + 1); 516 } 517 mUsageList.add(index, bs); 518 mMaxPower = Math.max(mMaxPower, amount); 519 } 520 } 521 522 // Smear it! 523 final double hiddenPowerMah = removeHiddenBatterySippers(mUsageList); 524 final double totalRemainingPower = getTotalPower() - hiddenPowerMah; 525 if (Math.abs(totalRemainingPower) > 1e-3) { 526 for (int i = 0, size = mUsageList.size(); i < size; i++) { 527 final BatterySipper sipper = mUsageList.get(i); 528 if (!sipper.shouldHide) { 529 sipper.proportionalSmearMah = hiddenPowerMah 530 * ((sipper.totalPowerMah + sipper.screenPowerMah) 531 / totalRemainingPower); 532 sipper.sumPower(); 533 } 534 } 535 } 536 } 537 processAppUsage(SparseArray<UserHandle> asUsers)538 private void processAppUsage(SparseArray<UserHandle> asUsers) { 539 final boolean forAllUsers = (asUsers.get(UserHandle.USER_ALL) != null); 540 mStatsPeriod = mTypeBatteryRealtimeUs; 541 542 BatterySipper osSipper = null; 543 final SparseArray<? extends Uid> uidStats = mStats.getUidStats(); 544 final int NU = uidStats.size(); 545 for (int iu = 0; iu < NU; iu++) { 546 final Uid u = uidStats.valueAt(iu); 547 final BatterySipper app = new BatterySipper(BatterySipper.DrainType.APP, u, 0); 548 549 mCpuPowerCalculator.calculateApp(app, u, mRawRealtimeUs, mRawUptimeUs, mStatsType); 550 mWakelockPowerCalculator.calculateApp(app, u, mRawRealtimeUs, mRawUptimeUs, mStatsType); 551 mMobileRadioPowerCalculator.calculateApp(app, u, mRawRealtimeUs, mRawUptimeUs, 552 mStatsType); 553 mWifiPowerCalculator.calculateApp(app, u, mRawRealtimeUs, mRawUptimeUs, mStatsType); 554 mBluetoothPowerCalculator.calculateApp(app, u, mRawRealtimeUs, mRawUptimeUs, 555 mStatsType); 556 mSensorPowerCalculator.calculateApp(app, u, mRawRealtimeUs, mRawUptimeUs, mStatsType); 557 mCameraPowerCalculator.calculateApp(app, u, mRawRealtimeUs, mRawUptimeUs, mStatsType); 558 mFlashlightPowerCalculator.calculateApp(app, u, mRawRealtimeUs, mRawUptimeUs, 559 mStatsType); 560 561 final double totalPower = app.sumPower(); 562 if (DEBUG && totalPower != 0) { 563 Log.d(TAG, String.format("UID %d: total power=%s", u.getUid(), 564 makemAh(totalPower))); 565 } 566 567 // Add the app to the list if it is consuming power. 568 if (totalPower != 0 || u.getUid() == 0) { 569 // 570 // Add the app to the app list, WiFi, Bluetooth, etc, or into "Other Users" list. 571 // 572 final int uid = app.getUid(); 573 final int userId = UserHandle.getUserId(uid); 574 if (uid == Process.WIFI_UID) { 575 mWifiSippers.add(app); 576 } else if (uid == Process.BLUETOOTH_UID) { 577 mBluetoothSippers.add(app); 578 } else if (!forAllUsers && asUsers.get(userId) == null 579 && UserHandle.getAppId(uid) >= Process.FIRST_APPLICATION_UID) { 580 // We are told to just report this user's apps as one large entry. 581 List<BatterySipper> list = mUserSippers.get(userId); 582 if (list == null) { 583 list = new ArrayList<>(); 584 mUserSippers.put(userId, list); 585 } 586 list.add(app); 587 } else { 588 mUsageList.add(app); 589 } 590 591 if (uid == 0) { 592 osSipper = app; 593 } 594 } 595 } 596 597 if (osSipper != null) { 598 // The device has probably been awake for longer than the screen on 599 // time and application wake lock time would account for. Assign 600 // this remainder to the OS, if possible. 601 mWakelockPowerCalculator.calculateRemaining(osSipper, mStats, mRawRealtimeUs, 602 mRawUptimeUs, mStatsType); 603 osSipper.sumPower(); 604 } 605 } 606 addPhoneUsage()607 private void addPhoneUsage() { 608 long phoneOnTimeMs = mStats.getPhoneOnTime(mRawRealtimeUs, mStatsType) / 1000; 609 double phoneOnPower = mPowerProfile.getAveragePower(PowerProfile.POWER_RADIO_ACTIVE) 610 * phoneOnTimeMs / (60 * 60 * 1000); 611 if (phoneOnPower != 0) { 612 addEntry(BatterySipper.DrainType.PHONE, phoneOnTimeMs, phoneOnPower); 613 } 614 } 615 616 /** 617 * Screen power is the additional power the screen takes while the device is running. 618 */ addScreenUsage()619 private void addScreenUsage() { 620 double power = 0; 621 long screenOnTimeMs = mStats.getScreenOnTime(mRawRealtimeUs, mStatsType) / 1000; 622 power += screenOnTimeMs * mPowerProfile.getAveragePower(PowerProfile.POWER_SCREEN_ON); 623 final double screenFullPower = 624 mPowerProfile.getAveragePower(PowerProfile.POWER_SCREEN_FULL); 625 for (int i = 0; i < BatteryStats.NUM_SCREEN_BRIGHTNESS_BINS; i++) { 626 double screenBinPower = screenFullPower * (i + 0.5f) 627 / BatteryStats.NUM_SCREEN_BRIGHTNESS_BINS; 628 long brightnessTime = mStats.getScreenBrightnessTime(i, mRawRealtimeUs, mStatsType) 629 / 1000; 630 double p = screenBinPower * brightnessTime; 631 if (DEBUG && p != 0) { 632 Log.d(TAG, "Screen bin #" + i + ": time=" + brightnessTime 633 + " power=" + makemAh(p / (60 * 60 * 1000))); 634 } 635 power += p; 636 } 637 power /= (60 * 60 * 1000); // To hours 638 if (power != 0) { 639 addEntry(BatterySipper.DrainType.SCREEN, screenOnTimeMs, power); 640 } 641 } 642 addRadioUsage()643 private void addRadioUsage() { 644 BatterySipper radio = new BatterySipper(BatterySipper.DrainType.CELL, null, 0); 645 mMobileRadioPowerCalculator.calculateRemaining(radio, mStats, mRawRealtimeUs, mRawUptimeUs, 646 mStatsType); 647 radio.sumPower(); 648 if (radio.totalPowerMah > 0) { 649 mUsageList.add(radio); 650 } 651 } 652 aggregateSippers(BatterySipper bs, List<BatterySipper> from, String tag)653 private void aggregateSippers(BatterySipper bs, List<BatterySipper> from, String tag) { 654 for (int i = 0; i < from.size(); i++) { 655 BatterySipper wbs = from.get(i); 656 if (DEBUG) Log.d(TAG, tag + " adding sipper " + wbs + ": cpu=" + wbs.cpuTimeMs); 657 bs.add(wbs); 658 } 659 bs.computeMobilemspp(); 660 bs.sumPower(); 661 } 662 663 /** 664 * Calculate the baseline power usage for the device when it is in suspend and idle. 665 * The device is drawing POWER_CPU_IDLE power at its lowest power state. 666 * The device is drawing POWER_CPU_IDLE + POWER_CPU_AWAKE power when a wakelock is held. 667 */ addIdleUsage()668 private void addIdleUsage() { 669 final double suspendPowerMaMs = (mTypeBatteryRealtimeUs / 1000) * 670 mPowerProfile.getAveragePower(PowerProfile.POWER_CPU_IDLE); 671 final double idlePowerMaMs = (mTypeBatteryUptimeUs / 1000) * 672 mPowerProfile.getAveragePower(PowerProfile.POWER_CPU_AWAKE); 673 final double totalPowerMah = (suspendPowerMaMs + idlePowerMaMs) / (60 * 60 * 1000); 674 if (DEBUG && totalPowerMah != 0) { 675 Log.d(TAG, "Suspend: time=" + (mTypeBatteryRealtimeUs / 1000) 676 + " power=" + makemAh(suspendPowerMaMs / (60 * 60 * 1000))); 677 Log.d(TAG, "Idle: time=" + (mTypeBatteryUptimeUs / 1000) 678 + " power=" + makemAh(idlePowerMaMs / (60 * 60 * 1000))); 679 } 680 681 if (totalPowerMah != 0) { 682 addEntry(BatterySipper.DrainType.IDLE, mTypeBatteryRealtimeUs / 1000, totalPowerMah); 683 } 684 } 685 686 /** 687 * We do per-app blaming of WiFi activity. If energy info is reported from the controller, 688 * then only the WiFi process gets blamed here since we normalize power calculations and 689 * assign all the power drain to apps. If energy info is not reported, we attribute the 690 * difference between total running time of WiFi for all apps and the actual running time 691 * of WiFi to the WiFi subsystem. 692 */ addWiFiUsage()693 private void addWiFiUsage() { 694 BatterySipper bs = new BatterySipper(DrainType.WIFI, null, 0); 695 mWifiPowerCalculator.calculateRemaining(bs, mStats, mRawRealtimeUs, mRawUptimeUs, 696 mStatsType); 697 aggregateSippers(bs, mWifiSippers, "WIFI"); 698 if (bs.totalPowerMah > 0) { 699 mUsageList.add(bs); 700 } 701 } 702 703 /** 704 * Bluetooth usage is not attributed to any apps yet, so the entire blame goes to the 705 * Bluetooth Category. 706 */ addBluetoothUsage()707 private void addBluetoothUsage() { 708 BatterySipper bs = new BatterySipper(BatterySipper.DrainType.BLUETOOTH, null, 0); 709 mBluetoothPowerCalculator.calculateRemaining(bs, mStats, mRawRealtimeUs, mRawUptimeUs, 710 mStatsType); 711 aggregateSippers(bs, mBluetoothSippers, "Bluetooth"); 712 if (bs.totalPowerMah > 0) { 713 mUsageList.add(bs); 714 } 715 } 716 addUserUsage()717 private void addUserUsage() { 718 for (int i = 0; i < mUserSippers.size(); i++) { 719 final int userId = mUserSippers.keyAt(i); 720 BatterySipper bs = new BatterySipper(DrainType.USER, null, 0); 721 bs.userId = userId; 722 aggregateSippers(bs, mUserSippers.valueAt(i), "User"); 723 mUsageList.add(bs); 724 } 725 } 726 addMemoryUsage()727 private void addMemoryUsage() { 728 BatterySipper memory = new BatterySipper(DrainType.MEMORY, null, 0); 729 mMemoryPowerCalculator.calculateRemaining(memory, mStats, mRawRealtimeUs, mRawUptimeUs, 730 mStatsType); 731 memory.sumPower(); 732 if (memory.totalPowerMah > 0) { 733 mUsageList.add(memory); 734 } 735 } 736 processMiscUsage()737 private void processMiscUsage() { 738 addUserUsage(); 739 addPhoneUsage(); 740 addScreenUsage(); 741 addWiFiUsage(); 742 addBluetoothUsage(); 743 addMemoryUsage(); 744 addIdleUsage(); // Not including cellular idle power 745 // Don't compute radio usage if it's a wifi-only device 746 if (!mWifiOnly) { 747 addRadioUsage(); 748 } 749 } 750 addEntry(DrainType drainType, long time, double power)751 private BatterySipper addEntry(DrainType drainType, long time, double power) { 752 BatterySipper bs = new BatterySipper(drainType, null, 0); 753 bs.usagePowerMah = power; 754 bs.usageTimeMs = time; 755 bs.sumPower(); 756 mUsageList.add(bs); 757 return bs; 758 } 759 getUsageList()760 public List<BatterySipper> getUsageList() { 761 return mUsageList; 762 } 763 getMobilemsppList()764 public List<BatterySipper> getMobilemsppList() { 765 return mMobilemsppList; 766 } 767 getStatsPeriod()768 public long getStatsPeriod() { 769 return mStatsPeriod; 770 } 771 getStatsType()772 public int getStatsType() { 773 return mStatsType; 774 } 775 getMaxPower()776 public double getMaxPower() { 777 return mMaxPower; 778 } 779 getMaxRealPower()780 public double getMaxRealPower() { 781 return mMaxRealPower; 782 } 783 getTotalPower()784 public double getTotalPower() { 785 return mTotalPower; 786 } 787 getComputedPower()788 public double getComputedPower() { 789 return mComputedPower; 790 } 791 getMinDrainedPower()792 public double getMinDrainedPower() { 793 return mMinDrainedPower; 794 } 795 getMaxDrainedPower()796 public double getMaxDrainedPower() { 797 return mMaxDrainedPower; 798 } 799 readFully(FileInputStream stream)800 public static byte[] readFully(FileInputStream stream) throws java.io.IOException { 801 return readFully(stream, stream.available()); 802 } 803 readFully(FileInputStream stream, int avail)804 public static byte[] readFully(FileInputStream stream, int avail) throws java.io.IOException { 805 int pos = 0; 806 byte[] data = new byte[avail]; 807 while (true) { 808 int amt = stream.read(data, pos, data.length - pos); 809 //Log.i("foo", "Read " + amt + " bytes at " + pos 810 // + " of avail " + data.length); 811 if (amt <= 0) { 812 //Log.i("foo", "**** FINISHED READING: pos=" + pos 813 // + " len=" + data.length); 814 return data; 815 } 816 pos += amt; 817 avail = stream.available(); 818 if (avail > data.length - pos) { 819 byte[] newData = new byte[pos + avail]; 820 System.arraycopy(data, 0, newData, 0, pos); 821 data = newData; 822 } 823 } 824 } 825 826 /** 827 * Mark the {@link BatterySipper} that we should hide and smear the screen usage based on 828 * foreground activity time. 829 * 830 * @param sippers sipper list that need to check and remove 831 * @return the total power of the hidden items of {@link BatterySipper} 832 * for proportional smearing 833 */ removeHiddenBatterySippers(List<BatterySipper> sippers)834 public double removeHiddenBatterySippers(List<BatterySipper> sippers) { 835 double proportionalSmearPowerMah = 0; 836 BatterySipper screenSipper = null; 837 for (int i = sippers.size() - 1; i >= 0; i--) { 838 final BatterySipper sipper = sippers.get(i); 839 sipper.shouldHide = shouldHideSipper(sipper); 840 if (sipper.shouldHide) { 841 if (sipper.drainType != BatterySipper.DrainType.OVERCOUNTED 842 && sipper.drainType != BatterySipper.DrainType.SCREEN 843 && sipper.drainType != BatterySipper.DrainType.UNACCOUNTED) { 844 // Don't add it if it is overcounted, unaccounted or screen 845 proportionalSmearPowerMah += sipper.totalPowerMah; 846 } 847 } 848 849 if (sipper.drainType == BatterySipper.DrainType.SCREEN) { 850 screenSipper = sipper; 851 } 852 } 853 854 smearScreenBatterySipper(sippers, screenSipper); 855 856 return proportionalSmearPowerMah; 857 } 858 859 /** 860 * Smear the screen on power usage among {@code sippers}, based on ratio of foreground activity 861 * time. 862 */ smearScreenBatterySipper(List<BatterySipper> sippers, BatterySipper screenSipper)863 public void smearScreenBatterySipper(List<BatterySipper> sippers, BatterySipper screenSipper) { 864 final long rawRealtimeMs = SystemClock.elapsedRealtime(); 865 long totalActivityTimeMs = 0; 866 final SparseLongArray activityTimeArray = new SparseLongArray(); 867 for (int i = 0, size = sippers.size(); i < size; i++) { 868 final BatteryStats.Uid uid = sippers.get(i).uidObj; 869 if (uid != null) { 870 final long timeMs = getForegroundActivityTotalTimeMs(uid, rawRealtimeMs); 871 activityTimeArray.put(uid.getUid(), timeMs); 872 totalActivityTimeMs += timeMs; 873 } 874 } 875 876 if (totalActivityTimeMs >= 10 * DateUtils.MINUTE_IN_MILLIS) { 877 final double screenPowerMah = screenSipper.totalPowerMah; 878 for (int i = 0, size = sippers.size(); i < size; i++) { 879 final BatterySipper sipper = sippers.get(i); 880 sipper.screenPowerMah = screenPowerMah * activityTimeArray.get(sipper.getUid(), 0) 881 / totalActivityTimeMs; 882 } 883 } 884 } 885 886 /** 887 * Check whether we should hide the battery sipper. 888 */ shouldHideSipper(BatterySipper sipper)889 public boolean shouldHideSipper(BatterySipper sipper) { 890 final BatterySipper.DrainType drainType = sipper.drainType; 891 892 return drainType == BatterySipper.DrainType.IDLE 893 || drainType == BatterySipper.DrainType.CELL 894 || drainType == BatterySipper.DrainType.SCREEN 895 || drainType == BatterySipper.DrainType.UNACCOUNTED 896 || drainType == BatterySipper.DrainType.OVERCOUNTED 897 || isTypeService(sipper) 898 || isTypeSystem(sipper); 899 } 900 901 /** 902 * Check whether {@code sipper} is type service 903 */ isTypeService(BatterySipper sipper)904 public boolean isTypeService(BatterySipper sipper) { 905 final String[] packages = mPackageManager.getPackagesForUid(sipper.getUid()); 906 if (packages == null) { 907 return false; 908 } 909 910 for (String packageName : packages) { 911 if (ArrayUtils.contains(mServicepackageArray, packageName)) { 912 return true; 913 } 914 } 915 916 return false; 917 } 918 919 /** 920 * Check whether {@code sipper} is type system 921 */ isTypeSystem(BatterySipper sipper)922 public boolean isTypeSystem(BatterySipper sipper) { 923 final int uid = sipper.uidObj == null ? -1 : sipper.getUid(); 924 sipper.mPackages = mPackageManager.getPackagesForUid(uid); 925 // Classify all the sippers to type system if the range of uid is 0...FIRST_APPLICATION_UID 926 if (uid >= Process.ROOT_UID && uid < Process.FIRST_APPLICATION_UID) { 927 return true; 928 } else if (sipper.mPackages != null) { 929 for (final String packageName : sipper.mPackages) { 930 if (ArrayUtils.contains(mSystemPackageArray, packageName)) { 931 return true; 932 } 933 } 934 } 935 936 return false; 937 } 938 939 @VisibleForTesting getForegroundActivityTotalTimeMs(BatteryStats.Uid uid, long rawRealtimeMs)940 public long getForegroundActivityTotalTimeMs(BatteryStats.Uid uid, long rawRealtimeMs) { 941 final BatteryStats.Timer timer = uid.getForegroundActivityTimer(); 942 if (timer != null) { 943 return timer.getTotalTimeLocked(rawRealtimeMs, BatteryStats.STATS_SINCE_CHARGED); 944 } 945 946 return 0; 947 } 948 949 @VisibleForTesting setPackageManager(PackageManager packageManager)950 public void setPackageManager(PackageManager packageManager) { 951 mPackageManager = packageManager; 952 } 953 954 @VisibleForTesting setSystemPackageArray(String[] array)955 public void setSystemPackageArray(String[] array) { 956 mSystemPackageArray = array; 957 } 958 959 @VisibleForTesting setServicePackageArray(String[] array)960 public void setServicePackageArray(String[] array) { 961 mServicepackageArray = array; 962 } 963 load()964 private void load() { 965 if (mBatteryInfo == null) { 966 return; 967 } 968 mStats = getStats(mBatteryInfo); 969 if (mCollectBatteryBroadcast) { 970 mBatteryBroadcast = mContext.registerReceiver(null, 971 new IntentFilter(Intent.ACTION_BATTERY_CHANGED)); 972 } 973 } 974 getStats(IBatteryStats service)975 private static BatteryStatsImpl getStats(IBatteryStats service) { 976 try { 977 ParcelFileDescriptor pfd = service.getStatisticsStream(); 978 if (pfd != null) { 979 try (FileInputStream fis = new ParcelFileDescriptor.AutoCloseInputStream(pfd)) { 980 byte[] data = readFully(fis, MemoryFile.getSize(pfd.getFileDescriptor())); 981 Parcel parcel = Parcel.obtain(); 982 parcel.unmarshall(data, 0, data.length); 983 parcel.setDataPosition(0); 984 BatteryStatsImpl stats = com.android.internal.os.BatteryStatsImpl.CREATOR 985 .createFromParcel(parcel); 986 return stats; 987 } catch (IOException e) { 988 Log.w(TAG, "Unable to read statistics stream", e); 989 } 990 } 991 } catch (RemoteException e) { 992 Log.w(TAG, "RemoteException:", e); 993 } 994 return new BatteryStatsImpl(); 995 } 996 } 997