1 /* 2 * Copyright (C) 2008 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.systemui.power; 18 19 import android.content.BroadcastReceiver; 20 import android.content.ContentResolver; 21 import android.content.Context; 22 import android.content.Intent; 23 import android.content.IntentFilter; 24 import android.content.pm.ActivityInfo; 25 import android.content.res.Configuration; 26 import android.database.ContentObserver; 27 import android.os.BatteryManager; 28 import android.os.Handler; 29 import android.os.IThermalEventListener; 30 import android.os.IThermalService; 31 import android.os.PowerManager; 32 import android.os.RemoteException; 33 import android.os.ServiceManager; 34 import android.os.SystemClock; 35 import android.os.Temperature; 36 import android.os.UserHandle; 37 import android.provider.Settings; 38 import android.text.format.DateUtils; 39 import android.util.Log; 40 import android.util.Slog; 41 42 import com.android.internal.annotations.VisibleForTesting; 43 import com.android.settingslib.fuelgauge.Estimate; 44 import com.android.settingslib.utils.ThreadUtils; 45 import com.android.systemui.Dependency; 46 import com.android.systemui.R; 47 import com.android.systemui.SystemUI; 48 import com.android.systemui.broadcast.BroadcastDispatcher; 49 import com.android.systemui.statusbar.CommandQueue; 50 import com.android.systemui.statusbar.phone.StatusBar; 51 52 import java.io.FileDescriptor; 53 import java.io.PrintWriter; 54 import java.time.Duration; 55 import java.util.Arrays; 56 import java.util.concurrent.Future; 57 58 import javax.inject.Inject; 59 import javax.inject.Singleton; 60 61 import dagger.Lazy; 62 63 @Singleton 64 public class PowerUI extends SystemUI implements CommandQueue.Callbacks { 65 66 static final String TAG = "PowerUI"; 67 static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG); 68 private static final long TEMPERATURE_INTERVAL = 30 * DateUtils.SECOND_IN_MILLIS; 69 private static final long TEMPERATURE_LOGGING_INTERVAL = DateUtils.HOUR_IN_MILLIS; 70 private static final int MAX_RECENT_TEMPS = 125; // TEMPERATURE_LOGGING_INTERVAL plus a buffer 71 static final long THREE_HOURS_IN_MILLIS = DateUtils.HOUR_IN_MILLIS * 3; 72 private static final int CHARGE_CYCLE_PERCENT_RESET = 45; 73 private static final long SIX_HOURS_MILLIS = Duration.ofHours(6).toMillis(); 74 public static final int NO_ESTIMATE_AVAILABLE = -1; 75 private static final String BOOT_COUNT_KEY = "boot_count"; 76 private static final String PREFS = "powerui_prefs"; 77 78 private final Handler mHandler = new Handler(); 79 @VisibleForTesting 80 final Receiver mReceiver = new Receiver(); 81 82 private PowerManager mPowerManager; 83 private WarningsUI mWarnings; 84 private InattentiveSleepWarningView mOverlayView; 85 private final Configuration mLastConfiguration = new Configuration(); 86 private int mPlugType = 0; 87 private int mInvalidCharger = 0; 88 private EnhancedEstimates mEnhancedEstimates; 89 private Future mLastShowWarningTask; 90 private boolean mEnableSkinTemperatureWarning; 91 private boolean mEnableUsbTemperatureAlarm; 92 93 private int mLowBatteryAlertCloseLevel; 94 private final int[] mLowBatteryReminderLevels = new int[2]; 95 96 private long mScreenOffTime = -1; 97 98 @VisibleForTesting boolean mLowWarningShownThisChargeCycle; 99 @VisibleForTesting boolean mSevereWarningShownThisChargeCycle; 100 @VisibleForTesting BatteryStateSnapshot mCurrentBatteryStateSnapshot; 101 @VisibleForTesting BatteryStateSnapshot mLastBatteryStateSnapshot; 102 @VisibleForTesting IThermalService mThermalService; 103 104 @VisibleForTesting int mBatteryLevel = 100; 105 @VisibleForTesting int mBatteryStatus = BatteryManager.BATTERY_STATUS_UNKNOWN; 106 107 private IThermalEventListener mSkinThermalEventListener; 108 private IThermalEventListener mUsbThermalEventListener; 109 private final BroadcastDispatcher mBroadcastDispatcher; 110 private final CommandQueue mCommandQueue; 111 private final Lazy<StatusBar> mStatusBarLazy; 112 113 @Inject PowerUI(Context context, BroadcastDispatcher broadcastDispatcher, CommandQueue commandQueue, Lazy<StatusBar> statusBarLazy)114 public PowerUI(Context context, BroadcastDispatcher broadcastDispatcher, 115 CommandQueue commandQueue, Lazy<StatusBar> statusBarLazy) { 116 super(context); 117 mBroadcastDispatcher = broadcastDispatcher; 118 mCommandQueue = commandQueue; 119 mStatusBarLazy = statusBarLazy; 120 } 121 start()122 public void start() { 123 mPowerManager = (PowerManager) mContext.getSystemService(Context.POWER_SERVICE); 124 mScreenOffTime = mPowerManager.isScreenOn() ? -1 : SystemClock.elapsedRealtime(); 125 mWarnings = Dependency.get(WarningsUI.class); 126 mEnhancedEstimates = Dependency.get(EnhancedEstimates.class); 127 mLastConfiguration.setTo(mContext.getResources().getConfiguration()); 128 129 ContentObserver obs = new ContentObserver(mHandler) { 130 @Override 131 public void onChange(boolean selfChange) { 132 updateBatteryWarningLevels(); 133 } 134 }; 135 final ContentResolver resolver = mContext.getContentResolver(); 136 resolver.registerContentObserver(Settings.Global.getUriFor( 137 Settings.Global.LOW_POWER_MODE_TRIGGER_LEVEL), 138 false, obs, UserHandle.USER_ALL); 139 updateBatteryWarningLevels(); 140 mReceiver.init(); 141 142 // Check to see if we need to let the user know that the phone previously shut down due 143 // to the temperature being too high. 144 showWarnOnThermalShutdown(); 145 146 // Register an observer to configure mEnableSkinTemperatureWarning and perform the 147 // registration of skin thermal event listener upon Settings change. 148 resolver.registerContentObserver( 149 Settings.Global.getUriFor(Settings.Global.SHOW_TEMPERATURE_WARNING), 150 false /*notifyForDescendants*/, 151 new ContentObserver(mHandler) { 152 @Override 153 public void onChange(boolean selfChange) { 154 doSkinThermalEventListenerRegistration(); 155 } 156 }); 157 // Register an observer to configure mEnableUsbTemperatureAlarm and perform the 158 // registration of usb thermal event listener upon Settings change. 159 resolver.registerContentObserver( 160 Settings.Global.getUriFor(Settings.Global.SHOW_USB_TEMPERATURE_ALARM), 161 false /*notifyForDescendants*/, 162 new ContentObserver(mHandler) { 163 @Override 164 public void onChange(boolean selfChange) { 165 doUsbThermalEventListenerRegistration(); 166 } 167 }); 168 initThermalEventListeners(); 169 mCommandQueue.addCallback(this); 170 } 171 172 @Override onConfigurationChanged(Configuration newConfig)173 protected void onConfigurationChanged(Configuration newConfig) { 174 final int mask = ActivityInfo.CONFIG_MCC | ActivityInfo.CONFIG_MNC; 175 176 // Safe to modify mLastConfiguration here as it's only updated by the main thread (here). 177 if ((mLastConfiguration.updateFrom(newConfig) & mask) != 0) { 178 mHandler.post(this::initThermalEventListeners); 179 } 180 } 181 updateBatteryWarningLevels()182 void updateBatteryWarningLevels() { 183 int critLevel = mContext.getResources().getInteger( 184 com.android.internal.R.integer.config_criticalBatteryWarningLevel); 185 int warnLevel = mContext.getResources().getInteger( 186 com.android.internal.R.integer.config_lowBatteryWarningLevel); 187 188 if (warnLevel < critLevel) { 189 warnLevel = critLevel; 190 } 191 192 mLowBatteryReminderLevels[0] = warnLevel; 193 mLowBatteryReminderLevels[1] = critLevel; 194 mLowBatteryAlertCloseLevel = mLowBatteryReminderLevels[0] 195 + mContext.getResources().getInteger( 196 com.android.internal.R.integer.config_lowBatteryCloseWarningBump); 197 } 198 199 /** 200 * Buckets the battery level. 201 * 202 * The code in this function is a little weird because I couldn't comprehend 203 * the bucket going up when the battery level was going down. --joeo 204 * 205 * 1 means that the battery is "ok" 206 * 0 means that the battery is between "ok" and what we should warn about. 207 * less than 0 means that the battery is low 208 */ findBatteryLevelBucket(int level)209 private int findBatteryLevelBucket(int level) { 210 if (level >= mLowBatteryAlertCloseLevel) { 211 return 1; 212 } 213 if (level > mLowBatteryReminderLevels[0]) { 214 return 0; 215 } 216 final int N = mLowBatteryReminderLevels.length; 217 for (int i=N-1; i>=0; i--) { 218 if (level <= mLowBatteryReminderLevels[i]) { 219 return -1-i; 220 } 221 } 222 throw new RuntimeException("not possible!"); 223 } 224 225 @VisibleForTesting 226 final class Receiver extends BroadcastReceiver { 227 228 private boolean mHasReceivedBattery = false; 229 init()230 public void init() { 231 // Register for Intent broadcasts for... 232 IntentFilter filter = new IntentFilter(); 233 filter.addAction(PowerManager.ACTION_POWER_SAVE_MODE_CHANGED); 234 filter.addAction(Intent.ACTION_BATTERY_CHANGED); 235 filter.addAction(Intent.ACTION_SCREEN_OFF); 236 filter.addAction(Intent.ACTION_SCREEN_ON); 237 filter.addAction(Intent.ACTION_USER_SWITCHED); 238 mBroadcastDispatcher.registerReceiverWithHandler(this, filter, mHandler); 239 // Force get initial values. Relying on Sticky behavior until API for getting info. 240 if (!mHasReceivedBattery) { 241 // Get initial state 242 Intent intent = mContext.registerReceiver( 243 null, 244 new IntentFilter(Intent.ACTION_BATTERY_CHANGED) 245 ); 246 if (intent != null) { 247 onReceive(mContext, intent); 248 } 249 } 250 } 251 252 @Override onReceive(Context context, Intent intent)253 public void onReceive(Context context, Intent intent) { 254 String action = intent.getAction(); 255 if (PowerManager.ACTION_POWER_SAVE_MODE_CHANGED.equals(action)) { 256 ThreadUtils.postOnBackgroundThread(() -> { 257 if (mPowerManager.isPowerSaveMode()) { 258 mWarnings.dismissLowBatteryWarning(); 259 } 260 }); 261 } else if (Intent.ACTION_BATTERY_CHANGED.equals(action)) { 262 mHasReceivedBattery = true; 263 final int oldBatteryLevel = mBatteryLevel; 264 mBatteryLevel = intent.getIntExtra(BatteryManager.EXTRA_LEVEL, 100); 265 final int oldBatteryStatus = mBatteryStatus; 266 mBatteryStatus = intent.getIntExtra(BatteryManager.EXTRA_STATUS, 267 BatteryManager.BATTERY_STATUS_UNKNOWN); 268 final int oldPlugType = mPlugType; 269 mPlugType = intent.getIntExtra(BatteryManager.EXTRA_PLUGGED, 1); 270 final int oldInvalidCharger = mInvalidCharger; 271 mInvalidCharger = intent.getIntExtra(BatteryManager.EXTRA_INVALID_CHARGER, 0); 272 mLastBatteryStateSnapshot = mCurrentBatteryStateSnapshot; 273 274 final boolean plugged = mPlugType != 0; 275 final boolean oldPlugged = oldPlugType != 0; 276 277 int oldBucket = findBatteryLevelBucket(oldBatteryLevel); 278 int bucket = findBatteryLevelBucket(mBatteryLevel); 279 280 if (DEBUG) { 281 Slog.d(TAG, "buckets ....." + mLowBatteryAlertCloseLevel 282 + " .. " + mLowBatteryReminderLevels[0] 283 + " .. " + mLowBatteryReminderLevels[1]); 284 Slog.d(TAG, "level " + oldBatteryLevel + " --> " + mBatteryLevel); 285 Slog.d(TAG, "status " + oldBatteryStatus + " --> " + mBatteryStatus); 286 Slog.d(TAG, "plugType " + oldPlugType + " --> " + mPlugType); 287 Slog.d(TAG, "invalidCharger " + oldInvalidCharger + " --> " + mInvalidCharger); 288 Slog.d(TAG, "bucket " + oldBucket + " --> " + bucket); 289 Slog.d(TAG, "plugged " + oldPlugged + " --> " + plugged); 290 } 291 292 mWarnings.update(mBatteryLevel, bucket, mScreenOffTime); 293 if (oldInvalidCharger == 0 && mInvalidCharger != 0) { 294 Slog.d(TAG, "showing invalid charger warning"); 295 mWarnings.showInvalidChargerWarning(); 296 return; 297 } else if (oldInvalidCharger != 0 && mInvalidCharger == 0) { 298 mWarnings.dismissInvalidChargerWarning(); 299 } else if (mWarnings.isInvalidChargerWarningShowing()) { 300 // if invalid charger is showing, don't show low battery 301 if (DEBUG) { 302 Slog.d(TAG, "Bad Charger"); 303 } 304 return; 305 } 306 307 // Show the correct version of low battery warning if needed 308 if (mLastShowWarningTask != null) { 309 mLastShowWarningTask.cancel(true); 310 if (DEBUG) { 311 Slog.d(TAG, "cancelled task"); 312 } 313 } 314 mLastShowWarningTask = ThreadUtils.postOnBackgroundThread(() -> { 315 maybeShowBatteryWarningV2( 316 plugged, bucket); 317 }); 318 319 } else if (Intent.ACTION_SCREEN_OFF.equals(action)) { 320 mScreenOffTime = SystemClock.elapsedRealtime(); 321 } else if (Intent.ACTION_SCREEN_ON.equals(action)) { 322 mScreenOffTime = -1; 323 } else if (Intent.ACTION_USER_SWITCHED.equals(action)) { 324 mWarnings.userSwitched(); 325 } else { 326 Slog.w(TAG, "unknown intent: " + intent); 327 } 328 } 329 } 330 maybeShowBatteryWarningV2(boolean plugged, int bucket)331 protected void maybeShowBatteryWarningV2(boolean plugged, int bucket) { 332 final boolean hybridEnabled = mEnhancedEstimates.isHybridNotificationEnabled(); 333 final boolean isPowerSaverMode = mPowerManager.isPowerSaveMode(); 334 335 // Stick current battery state into an immutable container to determine if we should show 336 // a warning. 337 if (DEBUG) { 338 Slog.d(TAG, "evaluating which notification to show"); 339 } 340 if (hybridEnabled) { 341 if (DEBUG) { 342 Slog.d(TAG, "using hybrid"); 343 } 344 Estimate estimate = refreshEstimateIfNeeded(); 345 mCurrentBatteryStateSnapshot = new BatteryStateSnapshot(mBatteryLevel, isPowerSaverMode, 346 plugged, bucket, mBatteryStatus, mLowBatteryReminderLevels[1], 347 mLowBatteryReminderLevels[0], estimate.getEstimateMillis(), 348 estimate.getAverageDischargeTime(), 349 mEnhancedEstimates.getSevereWarningThreshold(), 350 mEnhancedEstimates.getLowWarningThreshold(), estimate.isBasedOnUsage(), 351 mEnhancedEstimates.getLowWarningEnabled()); 352 } else { 353 if (DEBUG) { 354 Slog.d(TAG, "using standard"); 355 } 356 mCurrentBatteryStateSnapshot = new BatteryStateSnapshot(mBatteryLevel, isPowerSaverMode, 357 plugged, bucket, mBatteryStatus, mLowBatteryReminderLevels[1], 358 mLowBatteryReminderLevels[0]); 359 } 360 361 mWarnings.updateSnapshot(mCurrentBatteryStateSnapshot); 362 if (mCurrentBatteryStateSnapshot.isHybrid()) { 363 maybeShowHybridWarning(mCurrentBatteryStateSnapshot, mLastBatteryStateSnapshot); 364 } else { 365 maybeShowBatteryWarning(mCurrentBatteryStateSnapshot, mLastBatteryStateSnapshot); 366 } 367 } 368 369 // updates the time estimate if we don't have one or battery level has changed. 370 @VisibleForTesting refreshEstimateIfNeeded()371 Estimate refreshEstimateIfNeeded() { 372 if (mLastBatteryStateSnapshot == null 373 || mLastBatteryStateSnapshot.getTimeRemainingMillis() == NO_ESTIMATE_AVAILABLE 374 || mBatteryLevel != mLastBatteryStateSnapshot.getBatteryLevel()) { 375 final Estimate estimate = mEnhancedEstimates.getEstimate(); 376 if (DEBUG) { 377 Slog.d(TAG, "updated estimate: " + estimate.getEstimateMillis()); 378 } 379 return estimate; 380 } 381 return new Estimate(mLastBatteryStateSnapshot.getTimeRemainingMillis(), 382 mLastBatteryStateSnapshot.isBasedOnUsage(), 383 mLastBatteryStateSnapshot.getAverageTimeToDischargeMillis()); 384 } 385 386 @VisibleForTesting maybeShowHybridWarning(BatteryStateSnapshot currentSnapshot, BatteryStateSnapshot lastSnapshot)387 void maybeShowHybridWarning(BatteryStateSnapshot currentSnapshot, 388 BatteryStateSnapshot lastSnapshot) { 389 // if we are now over 45% battery & 6 hours remaining so we can trigger hybrid 390 // notification again 391 final long timeRemainingMillis = currentSnapshot.getTimeRemainingMillis(); 392 if (currentSnapshot.getBatteryLevel() >= CHARGE_CYCLE_PERCENT_RESET 393 && (timeRemainingMillis > SIX_HOURS_MILLIS 394 || timeRemainingMillis == NO_ESTIMATE_AVAILABLE)) { 395 mLowWarningShownThisChargeCycle = false; 396 mSevereWarningShownThisChargeCycle = false; 397 if (DEBUG) { 398 Slog.d(TAG, "Charge cycle reset! Can show warnings again"); 399 } 400 } 401 402 final boolean playSound = currentSnapshot.getBucket() != lastSnapshot.getBucket() 403 || lastSnapshot.getPlugged(); 404 405 if (shouldShowHybridWarning(currentSnapshot)) { 406 mWarnings.showLowBatteryWarning(playSound); 407 // mark if we've already shown a warning this cycle. This will prevent the notification 408 // trigger from spamming users by only showing low/critical warnings once per cycle 409 if ((timeRemainingMillis != NO_ESTIMATE_AVAILABLE 410 && timeRemainingMillis <= currentSnapshot.getSevereThresholdMillis()) 411 || currentSnapshot.getBatteryLevel() 412 <= currentSnapshot.getSevereLevelThreshold()) { 413 mSevereWarningShownThisChargeCycle = true; 414 mLowWarningShownThisChargeCycle = true; 415 if (DEBUG) { 416 Slog.d(TAG, "Severe warning marked as shown this cycle"); 417 } 418 } else { 419 Slog.d(TAG, "Low warning marked as shown this cycle"); 420 mLowWarningShownThisChargeCycle = true; 421 } 422 } else if (shouldDismissHybridWarning(currentSnapshot)) { 423 if (DEBUG) { 424 Slog.d(TAG, "Dismissing warning"); 425 } 426 mWarnings.dismissLowBatteryWarning(); 427 } else { 428 if (DEBUG) { 429 Slog.d(TAG, "Updating warning"); 430 } 431 mWarnings.updateLowBatteryWarning(); 432 } 433 } 434 435 @VisibleForTesting shouldShowHybridWarning(BatteryStateSnapshot snapshot)436 boolean shouldShowHybridWarning(BatteryStateSnapshot snapshot) { 437 if (snapshot.getPlugged() 438 || snapshot.getBatteryStatus() == BatteryManager.BATTERY_STATUS_UNKNOWN) { 439 Slog.d(TAG, "can't show warning due to - plugged: " + snapshot.getPlugged() 440 + " status unknown: " 441 + (snapshot.getBatteryStatus() == BatteryManager.BATTERY_STATUS_UNKNOWN)); 442 return false; 443 } 444 445 final long timeRemainingMillis = snapshot.getTimeRemainingMillis(); 446 // Only show the low warning if enabled once per charge cycle & no battery saver 447 final boolean canShowWarning = snapshot.isLowWarningEnabled() 448 && !mLowWarningShownThisChargeCycle && !snapshot.isPowerSaver() 449 && ((timeRemainingMillis != NO_ESTIMATE_AVAILABLE 450 && timeRemainingMillis < snapshot.getLowThresholdMillis()) 451 || snapshot.getBatteryLevel() <= snapshot.getLowLevelThreshold()); 452 453 // Only show the severe warning once per charge cycle 454 final boolean canShowSevereWarning = !mSevereWarningShownThisChargeCycle 455 && ((timeRemainingMillis != NO_ESTIMATE_AVAILABLE 456 && timeRemainingMillis < snapshot.getSevereThresholdMillis()) 457 || snapshot.getBatteryLevel() <= snapshot.getSevereLevelThreshold()); 458 459 final boolean canShow = canShowWarning || canShowSevereWarning; 460 461 if (DEBUG) { 462 Slog.d(TAG, "Enhanced trigger is: " + canShow + "\nwith battery snapshot:" 463 + " mLowWarningShownThisChargeCycle: " + mLowWarningShownThisChargeCycle 464 + " mSevereWarningShownThisChargeCycle: " + mSevereWarningShownThisChargeCycle 465 + "\n" + snapshot.toString()); 466 } 467 return canShow; 468 } 469 470 @VisibleForTesting shouldDismissHybridWarning(BatteryStateSnapshot snapshot)471 boolean shouldDismissHybridWarning(BatteryStateSnapshot snapshot) { 472 return snapshot.getPlugged() 473 || snapshot.getTimeRemainingMillis() > snapshot.getLowThresholdMillis(); 474 } 475 maybeShowBatteryWarning( BatteryStateSnapshot currentSnapshot, BatteryStateSnapshot lastSnapshot)476 protected void maybeShowBatteryWarning( 477 BatteryStateSnapshot currentSnapshot, 478 BatteryStateSnapshot lastSnapshot) { 479 final boolean playSound = currentSnapshot.getBucket() != lastSnapshot.getBucket() 480 || lastSnapshot.getPlugged(); 481 482 if (shouldShowLowBatteryWarning(currentSnapshot, lastSnapshot)) { 483 mWarnings.showLowBatteryWarning(playSound); 484 } else if (shouldDismissLowBatteryWarning(currentSnapshot, lastSnapshot)) { 485 mWarnings.dismissLowBatteryWarning(); 486 } else { 487 mWarnings.updateLowBatteryWarning(); 488 } 489 } 490 491 @VisibleForTesting shouldShowLowBatteryWarning( BatteryStateSnapshot currentSnapshot, BatteryStateSnapshot lastSnapshot)492 boolean shouldShowLowBatteryWarning( 493 BatteryStateSnapshot currentSnapshot, 494 BatteryStateSnapshot lastSnapshot) { 495 return !currentSnapshot.getPlugged() 496 && !currentSnapshot.isPowerSaver() 497 && (((currentSnapshot.getBucket() < lastSnapshot.getBucket() 498 || lastSnapshot.getPlugged()) 499 && currentSnapshot.getBucket() < 0)) 500 && currentSnapshot.getBatteryStatus() != BatteryManager.BATTERY_STATUS_UNKNOWN; 501 } 502 503 @VisibleForTesting shouldDismissLowBatteryWarning( BatteryStateSnapshot currentSnapshot, BatteryStateSnapshot lastSnapshot)504 boolean shouldDismissLowBatteryWarning( 505 BatteryStateSnapshot currentSnapshot, 506 BatteryStateSnapshot lastSnapshot) { 507 return currentSnapshot.isPowerSaver() 508 || currentSnapshot.getPlugged() 509 || (currentSnapshot.getBucket() > lastSnapshot.getBucket() 510 && currentSnapshot.getBucket() > 0); 511 } 512 initThermalEventListeners()513 private void initThermalEventListeners() { 514 doSkinThermalEventListenerRegistration(); 515 doUsbThermalEventListenerRegistration(); 516 } 517 518 @VisibleForTesting doSkinThermalEventListenerRegistration()519 synchronized void doSkinThermalEventListenerRegistration() { 520 final boolean oldEnableSkinTemperatureWarning = mEnableSkinTemperatureWarning; 521 boolean ret = false; 522 523 mEnableSkinTemperatureWarning = Settings.Global.getInt(mContext.getContentResolver(), 524 Settings.Global.SHOW_TEMPERATURE_WARNING, 525 mContext.getResources().getInteger(R.integer.config_showTemperatureWarning)) != 0; 526 527 if (mEnableSkinTemperatureWarning != oldEnableSkinTemperatureWarning) { 528 try { 529 if (mSkinThermalEventListener == null) { 530 mSkinThermalEventListener = new SkinThermalEventListener(); 531 } 532 if (mThermalService == null) { 533 mThermalService = IThermalService.Stub.asInterface( 534 ServiceManager.getService(Context.THERMAL_SERVICE)); 535 } 536 if (mEnableSkinTemperatureWarning) { 537 ret = mThermalService.registerThermalEventListenerWithType( 538 mSkinThermalEventListener, Temperature.TYPE_SKIN); 539 } else { 540 ret = mThermalService.unregisterThermalEventListener(mSkinThermalEventListener); 541 } 542 } catch (RemoteException e) { 543 Slog.e(TAG, "Exception while (un)registering skin thermal event listener.", e); 544 } 545 546 if (!ret) { 547 mEnableSkinTemperatureWarning = !mEnableSkinTemperatureWarning; 548 Slog.e(TAG, "Failed to register or unregister skin thermal event listener."); 549 } 550 } 551 } 552 553 @VisibleForTesting doUsbThermalEventListenerRegistration()554 synchronized void doUsbThermalEventListenerRegistration() { 555 final boolean oldEnableUsbTemperatureAlarm = mEnableUsbTemperatureAlarm; 556 boolean ret = false; 557 558 mEnableUsbTemperatureAlarm = Settings.Global.getInt(mContext.getContentResolver(), 559 Settings.Global.SHOW_USB_TEMPERATURE_ALARM, 560 mContext.getResources().getInteger(R.integer.config_showUsbPortAlarm)) != 0; 561 562 if (mEnableUsbTemperatureAlarm != oldEnableUsbTemperatureAlarm) { 563 try { 564 if (mUsbThermalEventListener == null) { 565 mUsbThermalEventListener = new UsbThermalEventListener(); 566 } 567 if (mThermalService == null) { 568 mThermalService = IThermalService.Stub.asInterface( 569 ServiceManager.getService(Context.THERMAL_SERVICE)); 570 } 571 if (mEnableUsbTemperatureAlarm) { 572 ret = mThermalService.registerThermalEventListenerWithType( 573 mUsbThermalEventListener, Temperature.TYPE_USB_PORT); 574 } else { 575 ret = mThermalService.unregisterThermalEventListener(mUsbThermalEventListener); 576 } 577 } catch (RemoteException e) { 578 Slog.e(TAG, "Exception while (un)registering usb thermal event listener.", e); 579 } 580 581 if (!ret) { 582 mEnableUsbTemperatureAlarm = !mEnableUsbTemperatureAlarm; 583 Slog.e(TAG, "Failed to register or unregister usb thermal event listener."); 584 } 585 } 586 } 587 showWarnOnThermalShutdown()588 private void showWarnOnThermalShutdown() { 589 int bootCount = -1; 590 int lastReboot = mContext.getSharedPreferences(PREFS, 0).getInt(BOOT_COUNT_KEY, -1); 591 try { 592 bootCount = Settings.Global.getInt(mContext.getContentResolver(), 593 Settings.Global.BOOT_COUNT); 594 } catch (Settings.SettingNotFoundException e) { 595 Slog.e(TAG, "Failed to read system boot count from Settings.Global.BOOT_COUNT"); 596 } 597 // Only show the thermal shutdown warning when there is a thermal reboot. 598 if (bootCount > lastReboot) { 599 mContext.getSharedPreferences(PREFS, 0).edit().putInt(BOOT_COUNT_KEY, 600 bootCount).apply(); 601 if (mPowerManager.getLastShutdownReason() 602 == PowerManager.SHUTDOWN_REASON_THERMAL_SHUTDOWN) { 603 mWarnings.showThermalShutdownWarning(); 604 } 605 } 606 } 607 608 @Override showInattentiveSleepWarning()609 public void showInattentiveSleepWarning() { 610 if (mOverlayView == null) { 611 mOverlayView = new InattentiveSleepWarningView(mContext); 612 } 613 614 mOverlayView.show(); 615 } 616 617 @Override dismissInattentiveSleepWarning(boolean animated)618 public void dismissInattentiveSleepWarning(boolean animated) { 619 if (mOverlayView != null) { 620 mOverlayView.dismiss(animated); 621 } 622 } 623 dump(FileDescriptor fd, PrintWriter pw, String[] args)624 public void dump(FileDescriptor fd, PrintWriter pw, String[] args) { 625 pw.print("mLowBatteryAlertCloseLevel="); 626 pw.println(mLowBatteryAlertCloseLevel); 627 pw.print("mLowBatteryReminderLevels="); 628 pw.println(Arrays.toString(mLowBatteryReminderLevels)); 629 pw.print("mBatteryLevel="); 630 pw.println(Integer.toString(mBatteryLevel)); 631 pw.print("mBatteryStatus="); 632 pw.println(Integer.toString(mBatteryStatus)); 633 pw.print("mPlugType="); 634 pw.println(Integer.toString(mPlugType)); 635 pw.print("mInvalidCharger="); 636 pw.println(Integer.toString(mInvalidCharger)); 637 pw.print("mScreenOffTime="); 638 pw.print(mScreenOffTime); 639 if (mScreenOffTime >= 0) { 640 pw.print(" ("); 641 pw.print(SystemClock.elapsedRealtime() - mScreenOffTime); 642 pw.print(" ago)"); 643 } 644 pw.println(); 645 pw.print("soundTimeout="); 646 pw.println(Settings.Global.getInt(mContext.getContentResolver(), 647 Settings.Global.LOW_BATTERY_SOUND_TIMEOUT, 0)); 648 pw.print("bucket: "); 649 pw.println(Integer.toString(findBatteryLevelBucket(mBatteryLevel))); 650 pw.print("mEnableSkinTemperatureWarning="); 651 pw.println(mEnableSkinTemperatureWarning); 652 pw.print("mEnableUsbTemperatureAlarm="); 653 pw.println(mEnableUsbTemperatureAlarm); 654 mWarnings.dump(pw); 655 } 656 657 /** 658 * The interface to allow PowerUI to communicate with whatever implementation of WarningsUI 659 * is being used by the system. 660 */ 661 public interface WarningsUI { 662 663 /** 664 * Updates battery and screen info for determining whether to trigger battery warnings or 665 * not. 666 * @param batteryLevel The current battery level 667 * @param bucket The current battery bucket 668 * @param screenOffTime How long the screen has been off in millis 669 */ update(int batteryLevel, int bucket, long screenOffTime)670 void update(int batteryLevel, int bucket, long screenOffTime); 671 dismissLowBatteryWarning()672 void dismissLowBatteryWarning(); 673 showLowBatteryWarning(boolean playSound)674 void showLowBatteryWarning(boolean playSound); 675 dismissInvalidChargerWarning()676 void dismissInvalidChargerWarning(); 677 showInvalidChargerWarning()678 void showInvalidChargerWarning(); 679 updateLowBatteryWarning()680 void updateLowBatteryWarning(); 681 isInvalidChargerWarningShowing()682 boolean isInvalidChargerWarningShowing(); 683 dismissHighTemperatureWarning()684 void dismissHighTemperatureWarning(); 685 showHighTemperatureWarning()686 void showHighTemperatureWarning(); 687 688 /** 689 * Display USB port overheat alarm 690 */ showUsbHighTemperatureAlarm()691 void showUsbHighTemperatureAlarm(); 692 showThermalShutdownWarning()693 void showThermalShutdownWarning(); 694 dump(PrintWriter pw)695 void dump(PrintWriter pw); 696 userSwitched()697 void userSwitched(); 698 699 /** 700 * Updates the snapshot of battery state used for evaluating battery warnings 701 * @param snapshot object containing relevant values for making battery warning decisions. 702 */ updateSnapshot(BatteryStateSnapshot snapshot)703 void updateSnapshot(BatteryStateSnapshot snapshot); 704 } 705 706 // Skin thermal event received from thermal service manager subsystem 707 @VisibleForTesting 708 final class SkinThermalEventListener extends IThermalEventListener.Stub { notifyThrottling(Temperature temp)709 @Override public void notifyThrottling(Temperature temp) { 710 int status = temp.getStatus(); 711 712 if (status >= Temperature.THROTTLING_EMERGENCY) { 713 if (!mStatusBarLazy.get().isDeviceInVrMode()) { 714 mWarnings.showHighTemperatureWarning(); 715 Slog.d(TAG, "SkinThermalEventListener: notifyThrottling was called " 716 + ", current skin status = " + status 717 + ", temperature = " + temp.getValue()); 718 } 719 } else { 720 mWarnings.dismissHighTemperatureWarning(); 721 } 722 } 723 } 724 725 // Usb thermal event received from thermal service manager subsystem 726 @VisibleForTesting 727 final class UsbThermalEventListener extends IThermalEventListener.Stub { notifyThrottling(Temperature temp)728 @Override public void notifyThrottling(Temperature temp) { 729 int status = temp.getStatus(); 730 731 if (status >= Temperature.THROTTLING_EMERGENCY) { 732 mWarnings.showUsbHighTemperatureAlarm(); 733 Slog.d(TAG, "UsbThermalEventListener: notifyThrottling was called " 734 + ", current usb port status = " + status 735 + ", temperature = " + temp.getValue()); 736 } 737 } 738 } 739 } 740