1 /* 2 * Copyright (C) 2016 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.doze; 18 19 import static com.android.systemui.keyguard.WakefulnessLifecycle.WAKEFULNESS_AWAKE; 20 import static com.android.systemui.keyguard.WakefulnessLifecycle.WAKEFULNESS_WAKING; 21 22 import android.annotation.MainThread; 23 import android.content.res.Configuration; 24 import android.hardware.display.AmbientDisplayConfiguration; 25 import android.os.Trace; 26 import android.util.Log; 27 import android.view.Display; 28 29 import com.android.internal.util.Preconditions; 30 import com.android.systemui.dock.DockManager; 31 import com.android.systemui.doze.dagger.DozeScope; 32 import com.android.systemui.doze.dagger.WrappedService; 33 import com.android.systemui.keyguard.WakefulnessLifecycle; 34 import com.android.systemui.keyguard.WakefulnessLifecycle.Wakefulness; 35 import com.android.systemui.settings.UserTracker; 36 import com.android.systemui.statusbar.phone.DozeParameters; 37 import com.android.systemui.util.Assert; 38 import com.android.systemui.util.wakelock.WakeLock; 39 40 import java.io.PrintWriter; 41 import java.util.ArrayList; 42 43 import javax.inject.Inject; 44 45 /** 46 * Orchestrates all things doze. 47 * 48 * DozeMachine implements a state machine that orchestrates how the UI and triggers work and 49 * interfaces with the power and screen states. 50 * 51 * During state transitions and in certain states, DozeMachine holds a wake lock. 52 */ 53 @DozeScope 54 public class DozeMachine { 55 56 static final String TAG = "DozeMachine"; 57 static final boolean DEBUG = DozeService.DEBUG; 58 private final DozeLog mDozeLog; 59 private static final String REASON_CHANGE_STATE = "DozeMachine#requestState"; 60 private static final String REASON_HELD_FOR_STATE = "DozeMachine#heldForState"; 61 62 public enum State { 63 /** Default state. Transition to INITIALIZED to get Doze going. */ 64 UNINITIALIZED, 65 /** Doze components are set up. Followed by transition to DOZE or DOZE_AOD. */ 66 INITIALIZED, 67 /** Regular doze. Device is asleep and listening for pulse triggers. */ 68 DOZE, 69 /** Deep doze. Device is asleep and is not listening for pulse triggers. */ 70 DOZE_SUSPEND_TRIGGERS, 71 /** Always-on doze. Device is asleep, showing UI and listening for pulse triggers. */ 72 DOZE_AOD, 73 /** Pulse has been requested. Device is awake and preparing UI */ 74 DOZE_REQUEST_PULSE, 75 /** Pulse is showing. Device is awake and showing UI. */ 76 DOZE_PULSING, 77 /** Pulse is showing with bright wallpaper. Device is awake and showing UI. */ 78 DOZE_PULSING_BRIGHT, 79 /** Pulse is done showing. Followed by transition to DOZE or DOZE_AOD. */ 80 DOZE_PULSE_DONE, 81 /** Doze is done. DozeService is finished. */ 82 FINISH, 83 /** AOD, but the display is temporarily off. */ 84 DOZE_AOD_PAUSED, 85 /** AOD, prox is near, transitions to DOZE_AOD_PAUSED after a timeout. */ 86 DOZE_AOD_PAUSING, 87 /** Always-on doze. Device is awake, showing docking UI and listening for pulse triggers. */ 88 DOZE_AOD_DOCKED; 89 canPulse()90 boolean canPulse() { 91 switch (this) { 92 case DOZE: 93 case DOZE_AOD: 94 case DOZE_AOD_PAUSED: 95 case DOZE_AOD_PAUSING: 96 case DOZE_AOD_DOCKED: 97 return true; 98 default: 99 return false; 100 } 101 } 102 staysAwake()103 boolean staysAwake() { 104 switch (this) { 105 case DOZE_REQUEST_PULSE: 106 case DOZE_PULSING: 107 case DOZE_PULSING_BRIGHT: 108 case DOZE_AOD_DOCKED: 109 return true; 110 default: 111 return false; 112 } 113 } 114 isAlwaysOn()115 boolean isAlwaysOn() { 116 return this == DOZE_AOD || this == DOZE_AOD_DOCKED; 117 } 118 screenState(DozeParameters parameters)119 int screenState(DozeParameters parameters) { 120 switch (this) { 121 case UNINITIALIZED: 122 case INITIALIZED: 123 return parameters.shouldControlScreenOff() ? Display.STATE_ON 124 : Display.STATE_OFF; 125 case DOZE_REQUEST_PULSE: 126 return parameters.getDisplayNeedsBlanking() ? Display.STATE_OFF 127 : Display.STATE_ON; 128 case DOZE_AOD_PAUSED: 129 case DOZE: 130 case DOZE_SUSPEND_TRIGGERS: 131 return Display.STATE_OFF; 132 case DOZE_PULSING: 133 case DOZE_PULSING_BRIGHT: 134 case DOZE_AOD_DOCKED: 135 return Display.STATE_ON; 136 case DOZE_AOD: 137 case DOZE_AOD_PAUSING: 138 return Display.STATE_DOZE_SUSPEND; 139 default: 140 return Display.STATE_UNKNOWN; 141 } 142 } 143 } 144 145 private final Service mDozeService; 146 private final WakeLock mWakeLock; 147 private final AmbientDisplayConfiguration mAmbientDisplayConfig; 148 private final WakefulnessLifecycle mWakefulnessLifecycle; 149 private final DozeHost mDozeHost; 150 private final DockManager mDockManager; 151 private final Part[] mParts; 152 private final UserTracker mUserTracker; 153 154 private final ArrayList<State> mQueuedRequests = new ArrayList<>(); 155 private State mState = State.UNINITIALIZED; 156 private int mPulseReason; 157 private boolean mWakeLockHeldForCurrentState = false; 158 private int mUiModeType = Configuration.UI_MODE_TYPE_NORMAL; 159 160 @Inject DozeMachine(@rappedService Service service, AmbientDisplayConfiguration ambientDisplayConfig, WakeLock wakeLock, WakefulnessLifecycle wakefulnessLifecycle, DozeLog dozeLog, DockManager dockManager, DozeHost dozeHost, Part[] parts, UserTracker userTracker)161 public DozeMachine(@WrappedService Service service, 162 AmbientDisplayConfiguration ambientDisplayConfig, 163 WakeLock wakeLock, WakefulnessLifecycle wakefulnessLifecycle, 164 DozeLog dozeLog, DockManager dockManager, 165 DozeHost dozeHost, Part[] parts, UserTracker userTracker) { 166 mDozeService = service; 167 mAmbientDisplayConfig = ambientDisplayConfig; 168 mWakefulnessLifecycle = wakefulnessLifecycle; 169 mWakeLock = wakeLock; 170 mDozeLog = dozeLog; 171 mDockManager = dockManager; 172 mDozeHost = dozeHost; 173 mParts = parts; 174 mUserTracker = userTracker; 175 for (Part part : parts) { 176 part.setDozeMachine(this); 177 } 178 } 179 180 /** 181 * Clean ourselves up. 182 */ destroy()183 public void destroy() { 184 for (Part part : mParts) { 185 part.destroy(); 186 } 187 } 188 189 /** 190 * Notifies the {@link DozeMachine} that {@link Configuration} has changed. 191 */ onConfigurationChanged(Configuration newConfiguration)192 public void onConfigurationChanged(Configuration newConfiguration) { 193 int newUiModeType = newConfiguration.uiMode & Configuration.UI_MODE_TYPE_MASK; 194 if (mUiModeType == newUiModeType) return; 195 mUiModeType = newUiModeType; 196 for (Part part : mParts) { 197 part.onUiModeTypeChanged(mUiModeType); 198 } 199 } 200 201 /** 202 * Requests transitioning to {@code requestedState}. 203 * 204 * This can be called during a state transition, in which case it will be queued until all 205 * queued state transitions are done. 206 * 207 * A wake lock is held while the transition is happening. 208 * 209 * Note that {@link #transitionPolicy} can modify what state will be transitioned to. 210 */ 211 @MainThread requestState(State requestedState)212 public void requestState(State requestedState) { 213 Preconditions.checkArgument(requestedState != State.DOZE_REQUEST_PULSE); 214 requestState(requestedState, DozeLog.PULSE_REASON_NONE); 215 } 216 217 @MainThread requestPulse(int pulseReason)218 public void requestPulse(int pulseReason) { 219 // Must not be called during a transition. There's no inherent problem with that, 220 // but there's currently no need to execute from a transition and it simplifies the 221 // code to not have to worry about keeping the pulseReason in mQueuedRequests. 222 Preconditions.checkState(!isExecutingTransition()); 223 requestState(State.DOZE_REQUEST_PULSE, pulseReason); 224 } 225 226 /** 227 * @return true if {@link DozeMachine} is currently in either {@link State#UNINITIALIZED} 228 * or {@link State#FINISH} 229 */ isUninitializedOrFinished()230 public boolean isUninitializedOrFinished() { 231 return mState == State.UNINITIALIZED || mState == State.FINISH; 232 } 233 onScreenState(int state)234 void onScreenState(int state) { 235 mDozeLog.traceDisplayState(state); 236 for (Part part : mParts) { 237 part.onScreenState(state); 238 } 239 } 240 requestState(State requestedState, int pulseReason)241 private void requestState(State requestedState, int pulseReason) { 242 Assert.isMainThread(); 243 if (DEBUG) { 244 Log.i(TAG, "request: current=" + mState + " req=" + requestedState, 245 new Throwable("here")); 246 } 247 248 boolean runNow = !isExecutingTransition(); 249 mQueuedRequests.add(requestedState); 250 if (runNow) { 251 mWakeLock.acquire(REASON_CHANGE_STATE); 252 for (int i = 0; i < mQueuedRequests.size(); i++) { 253 // Transitions in Parts can call back into requestState, which will 254 // cause mQueuedRequests to grow. 255 transitionTo(mQueuedRequests.get(i), pulseReason); 256 } 257 mQueuedRequests.clear(); 258 mWakeLock.release(REASON_CHANGE_STATE); 259 } 260 } 261 262 /** 263 * @return the current state. 264 * 265 * This must not be called during a transition. 266 */ 267 @MainThread getState()268 public State getState() { 269 Assert.isMainThread(); 270 if (isExecutingTransition()) { 271 throw new IllegalStateException("Cannot get state because there were pending " 272 + "transitions: " + mQueuedRequests); 273 } 274 return mState; 275 } 276 277 /** 278 * @return the current pulse reason. 279 * 280 * This is only valid if the machine is currently in one of the pulse states. 281 */ 282 @MainThread getPulseReason()283 public int getPulseReason() { 284 Assert.isMainThread(); 285 Preconditions.checkState(mState == State.DOZE_REQUEST_PULSE 286 || mState == State.DOZE_PULSING 287 || mState == State.DOZE_PULSING_BRIGHT 288 || mState == State.DOZE_PULSE_DONE, "must be in pulsing state, but is " + mState); 289 return mPulseReason; 290 } 291 292 /** Requests the PowerManager to wake up now. 293 * @param reason {@link DozeLog.Reason} that woke up the device.*/ wakeUp(@ozeLog.Reason int reason)294 public void wakeUp(@DozeLog.Reason int reason) { 295 mDozeService.requestWakeUp(reason); 296 } 297 isExecutingTransition()298 public boolean isExecutingTransition() { 299 return !mQueuedRequests.isEmpty(); 300 } 301 transitionTo(State requestedState, int pulseReason)302 private void transitionTo(State requestedState, int pulseReason) { 303 State newState = transitionPolicy(requestedState); 304 305 if (DEBUG) { 306 Log.i(TAG, "transition: old=" + mState + " req=" + requestedState + " new=" + newState); 307 } 308 309 if (newState == mState) { 310 return; 311 } 312 313 validateTransition(newState); 314 315 State oldState = mState; 316 mState = newState; 317 318 mDozeLog.traceState(newState); 319 Trace.traceCounter(Trace.TRACE_TAG_APP, "doze_machine_state", newState.ordinal()); 320 321 updatePulseReason(newState, oldState, pulseReason); 322 performTransitionOnComponents(oldState, newState); 323 updateWakeLockState(newState); 324 325 resolveIntermediateState(newState); 326 } 327 updatePulseReason(State newState, State oldState, int pulseReason)328 private void updatePulseReason(State newState, State oldState, int pulseReason) { 329 if (newState == State.DOZE_REQUEST_PULSE) { 330 mPulseReason = pulseReason; 331 } else if (oldState == State.DOZE_PULSE_DONE) { 332 mPulseReason = DozeLog.PULSE_REASON_NONE; 333 } 334 } 335 performTransitionOnComponents(State oldState, State newState)336 private void performTransitionOnComponents(State oldState, State newState) { 337 for (Part p : mParts) { 338 p.transitionTo(oldState, newState); 339 } 340 mDozeLog.traceDozeStateSendComplete(newState); 341 342 if (newState == State.FINISH) { 343 mDozeService.finish(); 344 } 345 } 346 validateTransition(State newState)347 private void validateTransition(State newState) { 348 try { 349 switch (mState) { 350 case FINISH: 351 Preconditions.checkState(newState == State.FINISH); 352 break; 353 case UNINITIALIZED: 354 Preconditions.checkState(newState == State.INITIALIZED); 355 break; 356 } 357 switch (newState) { 358 case UNINITIALIZED: 359 throw new IllegalArgumentException("can't transition to UNINITIALIZED"); 360 case INITIALIZED: 361 Preconditions.checkState(mState == State.UNINITIALIZED); 362 break; 363 case DOZE_PULSING: 364 Preconditions.checkState(mState == State.DOZE_REQUEST_PULSE); 365 break; 366 case DOZE_PULSE_DONE: 367 Preconditions.checkState( 368 mState == State.DOZE_REQUEST_PULSE || mState == State.DOZE_PULSING 369 || mState == State.DOZE_PULSING_BRIGHT); 370 break; 371 default: 372 break; 373 } 374 } catch (RuntimeException e) { 375 throw new IllegalStateException("Illegal Transition: " + mState + " -> " + newState, e); 376 } 377 } 378 transitionPolicy(State requestedState)379 private State transitionPolicy(State requestedState) { 380 if (mState == State.FINISH) { 381 return State.FINISH; 382 } 383 if (mUiModeType == Configuration.UI_MODE_TYPE_CAR 384 && (requestedState.canPulse() || requestedState.staysAwake())) { 385 Log.i(TAG, "Doze is suppressed with all triggers disabled as car mode is active"); 386 mDozeLog.traceCarModeStarted(); 387 return State.DOZE_SUSPEND_TRIGGERS; 388 } 389 if (mDozeHost.isAlwaysOnSuppressed() && requestedState.isAlwaysOn()) { 390 Log.i(TAG, "Doze is suppressed by an app. Suppressing state: " + requestedState); 391 mDozeLog.traceAlwaysOnSuppressed(requestedState, "app"); 392 return State.DOZE; 393 } 394 if (mDozeHost.isPowerSaveActive() && requestedState.isAlwaysOn()) { 395 Log.i(TAG, "Doze is suppressed by battery saver. Suppressing state: " + requestedState); 396 mDozeLog.traceAlwaysOnSuppressed(requestedState, "batterySaver"); 397 return State.DOZE; 398 } 399 if ((mState == State.DOZE_AOD_PAUSED || mState == State.DOZE_AOD_PAUSING 400 || mState == State.DOZE_AOD || mState == State.DOZE 401 || mState == State.DOZE_AOD_DOCKED || mState == State.DOZE_SUSPEND_TRIGGERS) 402 && requestedState == State.DOZE_PULSE_DONE) { 403 Log.i(TAG, "Dropping pulse done because current state is already done: " + mState); 404 return mState; 405 } 406 if (requestedState == State.DOZE_REQUEST_PULSE && !mState.canPulse()) { 407 Log.i(TAG, "Dropping pulse request because current state can't pulse: " + mState); 408 return mState; 409 } 410 return requestedState; 411 } 412 updateWakeLockState(State newState)413 private void updateWakeLockState(State newState) { 414 boolean staysAwake = newState.staysAwake(); 415 if (mWakeLockHeldForCurrentState && !staysAwake) { 416 mWakeLock.release(REASON_HELD_FOR_STATE); 417 mWakeLockHeldForCurrentState = false; 418 } else if (!mWakeLockHeldForCurrentState && staysAwake) { 419 mWakeLock.acquire(REASON_HELD_FOR_STATE); 420 mWakeLockHeldForCurrentState = true; 421 } 422 } 423 resolveIntermediateState(State state)424 private void resolveIntermediateState(State state) { 425 switch (state) { 426 case INITIALIZED: 427 case DOZE_PULSE_DONE: 428 final State nextState; 429 @Wakefulness int wakefulness = mWakefulnessLifecycle.getWakefulness(); 430 if (state != State.INITIALIZED && (wakefulness == WAKEFULNESS_AWAKE 431 || wakefulness == WAKEFULNESS_WAKING)) { 432 nextState = State.FINISH; 433 } else if (mDockManager.isDocked()) { 434 nextState = mDockManager.isHidden() ? State.DOZE : State.DOZE_AOD_DOCKED; 435 } else if (mAmbientDisplayConfig.alwaysOnEnabled(mUserTracker.getUserId())) { 436 nextState = State.DOZE_AOD; 437 } else { 438 nextState = State.DOZE; 439 } 440 441 transitionTo(nextState, DozeLog.PULSE_REASON_NONE); 442 break; 443 default: 444 break; 445 } 446 } 447 448 /** Dumps the current state */ dump(PrintWriter pw)449 public void dump(PrintWriter pw) { 450 pw.print(" state="); pw.println(mState); 451 pw.print(" mUiModeType="); pw.println(mUiModeType); 452 pw.print(" wakeLockHeldForCurrentState="); pw.println(mWakeLockHeldForCurrentState); 453 pw.print(" wakeLock="); pw.println(mWakeLock); 454 pw.println("Parts:"); 455 for (Part p : mParts) { 456 p.dump(pw); 457 } 458 } 459 460 /** A part of the DozeMachine that needs to be notified about state changes. */ 461 public interface Part { 462 /** 463 * Transition from {@code oldState} to {@code newState}. 464 * 465 * This method is guaranteed to only be called while a wake lock is held. 466 */ transitionTo(State oldState, State newState)467 void transitionTo(State oldState, State newState); 468 469 /** Dump current state. For debugging only. */ dump(PrintWriter pw)470 default void dump(PrintWriter pw) {} 471 472 /** Give the Part a chance to clean itself up. */ destroy()473 default void destroy() {} 474 475 /** 476 * Alerts that the screenstate is being changed. 477 * Note: This may be called from within a call to transitionTo, so local DozeState may not 478 * be accurate nor match with the new displayState. 479 */ onScreenState(int displayState)480 default void onScreenState(int displayState) {} 481 482 /** Sets the {@link DozeMachine} when this Part is associated with one. */ setDozeMachine(DozeMachine dozeMachine)483 default void setDozeMachine(DozeMachine dozeMachine) {} 484 485 /** 486 * Notifies the Part about a change in {@link Configuration#uiMode}. 487 * 488 * @param newUiModeType {@link Configuration#UI_MODE_TYPE_NORMAL}, 489 * {@link Configuration#UI_MODE_TYPE_DESK}, 490 * {@link Configuration#UI_MODE_TYPE_CAR}, 491 * {@link Configuration#UI_MODE_TYPE_TELEVISION}, 492 * {@link Configuration#UI_MODE_TYPE_APPLIANCE}, 493 * {@link Configuration#UI_MODE_TYPE_WATCH}, 494 * or {@link Configuration#UI_MODE_TYPE_VR_HEADSET} 495 */ onUiModeTypeChanged(int newUiModeType)496 default void onUiModeTypeChanged(int newUiModeType) {} 497 } 498 499 /** A wrapper interface for {@link android.service.dreams.DreamService} */ 500 public interface Service { 501 /** Finish dreaming. */ finish()502 void finish(); 503 504 /** Request a display state. See {@link android.view.Display#STATE_DOZE}. */ setDozeScreenState(int state)505 void setDozeScreenState(int state); 506 507 /** Request waking up. */ requestWakeUp(@ozeLog.Reason int reason)508 void requestWakeUp(@DozeLog.Reason int reason); 509 510 /** Set screen brightness */ setDozeScreenBrightness(int brightness)511 void setDozeScreenBrightness(int brightness); 512 513 class Delegate implements Service { 514 private final Service mDelegate; 515 Delegate(Service delegate)516 public Delegate(Service delegate) { 517 mDelegate = delegate; 518 } 519 520 @Override finish()521 public void finish() { 522 mDelegate.finish(); 523 } 524 525 @Override setDozeScreenState(int state)526 public void setDozeScreenState(int state) { 527 mDelegate.setDozeScreenState(state); 528 } 529 530 @Override requestWakeUp(@ozeLog.Reason int reason)531 public void requestWakeUp(@DozeLog.Reason int reason) { 532 mDelegate.requestWakeUp(reason); 533 } 534 535 @Override setDozeScreenBrightness(int brightness)536 public void setDozeScreenBrightness(int brightness) { 537 mDelegate.setDozeScreenBrightness(brightness); 538 } 539 } 540 } 541 } 542