blob: 169f03b0023d73762c956361ba062d6a1f08db16 [file] [log] [blame]
Jorim Jaggife762342016-10-13 14:33:27 +02001/*
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
Wale Ogunwale59507092018-10-29 09:00:30 -070017package com.android.server.wm;
Jorim Jaggife762342016-10-13 14:33:27 +020018
Wale Ogunwaledfb7fb22017-06-23 14:52:40 -070019import static android.os.Trace.TRACE_TAG_ACTIVITY_MANAGER;
David Stevens9440dc82017-03-16 19:00:20 -070020import static android.view.Display.DEFAULT_DISPLAY;
Wale Ogunwale9e4f3e02018-05-17 09:35:39 -070021import static android.view.WindowManager.TRANSIT_FLAG_KEYGUARD_GOING_AWAY_NO_ANIMATION;
22import static android.view.WindowManager.TRANSIT_FLAG_KEYGUARD_GOING_AWAY_TO_SHADE;
23import static android.view.WindowManager.TRANSIT_FLAG_KEYGUARD_GOING_AWAY_WITH_WALLPAPER;
24import static android.view.WindowManager.TRANSIT_KEYGUARD_GOING_AWAY;
25import static android.view.WindowManager.TRANSIT_KEYGUARD_OCCLUDE;
26import static android.view.WindowManager.TRANSIT_KEYGUARD_UNOCCLUDE;
27import static android.view.WindowManager.TRANSIT_UNSET;
Adrian Roose99bc052017-11-20 17:55:31 +010028import static android.view.WindowManagerPolicyConstants.KEYGUARD_GOING_AWAY_FLAG_NO_WINDOW_ANIMATIONS;
29import static android.view.WindowManagerPolicyConstants.KEYGUARD_GOING_AWAY_FLAG_TO_SHADE;
30import static android.view.WindowManagerPolicyConstants.KEYGUARD_GOING_AWAY_FLAG_WITH_WALLPAPER;
wilsonshih0299c8a2018-08-24 15:52:57 +080031
Issei Suzukid6eb5a22019-02-20 23:08:03 +010032import static com.android.server.am.KeyguardControllerProto.AOD_SHOWING;
wilsonshih0299c8a2018-08-24 15:52:57 +080033import static com.android.server.am.KeyguardControllerProto.KEYGUARD_OCCLUDED_STATES;
Yi Jin6c6e9ca2018-03-20 16:53:35 -070034import static com.android.server.am.KeyguardControllerProto.KEYGUARD_SHOWING;
wilsonshih0299c8a2018-08-24 15:52:57 +080035import static com.android.server.am.KeyguardOccludedProto.DISPLAY_ID;
36import static com.android.server.am.KeyguardOccludedProto.KEYGUARD_OCCLUDED;
wilsonshihe7903ea2018-09-26 16:17:59 +080037import static com.android.server.wm.ActivityStackSupervisor.PRESERVE_WINDOWS;
38import static com.android.server.wm.ActivityTaskManagerDebugConfig.TAG_ATM;
39import static com.android.server.wm.ActivityTaskManagerDebugConfig.TAG_WITH_CLASS_NAME;
Jorim Jaggife762342016-10-13 14:33:27 +020040
Jorim Jaggi241ae102016-11-02 21:57:33 -070041import android.os.IBinder;
42import android.os.RemoteException;
Wale Ogunwaledfb7fb22017-06-23 14:52:40 -070043import android.os.Trace;
Jorim Jaggi241ae102016-11-02 21:57:33 -070044import android.util.Slog;
wilsonshih0299c8a2018-08-24 15:52:57 +080045import android.util.SparseArray;
Steven Timotius4346f0a2017-09-12 11:07:21 -070046import android.util.proto.ProtoOutputStream;
Adrian Roose99bc052017-11-20 17:55:31 +010047
Jorim Jaggi241ae102016-11-02 21:57:33 -070048import com.android.internal.policy.IKeyguardDismissCallback;
Adrian Roose99bc052017-11-20 17:55:31 +010049import com.android.server.policy.WindowManagerPolicy;
Wale Ogunwale9e4f3e02018-05-17 09:35:39 -070050import com.android.server.wm.ActivityTaskManagerInternal.SleepToken;
Adrian Roose99bc052017-11-20 17:55:31 +010051
Jorim Jaggi8d786932016-10-26 19:08:36 -070052import java.io.PrintWriter;
Jorim Jaggife762342016-10-13 14:33:27 +020053
54/**
55 * Controls Keyguard occluding, dismissing and transitions depending on what kind of activities are
56 * currently visible.
57 * <p>
58 * Note that everything in this class should only be accessed with the AM lock being held.
59 */
60class KeyguardController {
61
Wale Ogunwale98875612018-10-12 07:53:02 -070062 private static final String TAG = TAG_WITH_CLASS_NAME ? "KeyguardController" : TAG_ATM;
Jorim Jaggi241ae102016-11-02 21:57:33 -070063
Jorim Jaggife762342016-10-13 14:33:27 +020064 private final ActivityStackSupervisor mStackSupervisor;
65 private WindowManagerService mWindowManager;
Jorim Jaggife762342016-10-13 14:33:27 +020066 private boolean mKeyguardShowing;
Lucas Dupin47a65c72018-02-15 14:16:18 -080067 private boolean mAodShowing;
Jorim Jaggi8d786932016-10-26 19:08:36 -070068 private boolean mKeyguardGoingAway;
Jorim Jaggi07961872016-11-23 11:28:57 +010069 private boolean mDismissalRequested;
wilsonshihe7903ea2018-09-26 16:17:59 +080070 private int[] mSecondaryDisplayIdsShowing;
Jorim Jaggife762342016-10-13 14:33:27 +020071 private int mBeforeUnoccludeTransit;
72 private int mVisibilityTransactionDepth;
wilsonshih0299c8a2018-08-24 15:52:57 +080073 private final SparseArray<KeyguardDisplayState> mDisplayStates = new SparseArray<>();
74 private final ActivityTaskManagerService mService;
Wale Ogunwaled32da472018-11-16 07:19:28 -080075 private RootActivityContainer mRootActivityContainer;
Jorim Jaggife762342016-10-13 14:33:27 +020076
Wale Ogunwalef6733932018-06-27 05:14:34 -070077 KeyguardController(ActivityTaskManagerService service,
Jorim Jaggife762342016-10-13 14:33:27 +020078 ActivityStackSupervisor stackSupervisor) {
79 mService = service;
80 mStackSupervisor = stackSupervisor;
81 }
82
83 void setWindowManager(WindowManagerService windowManager) {
84 mWindowManager = windowManager;
Wale Ogunwaled32da472018-11-16 07:19:28 -080085 mRootActivityContainer = mService.mRootActivityContainer;
Jorim Jaggife762342016-10-13 14:33:27 +020086 }
87
88 /**
Lucas Dupin47a65c72018-02-15 14:16:18 -080089 * @return true if either Keyguard or AOD are showing, not going away, and not being occluded
Issei Suzukid6eb5a22019-02-20 23:08:03 +010090 * on the given display, false otherwise.
Lucas Dupin47a65c72018-02-15 14:16:18 -080091 */
92 boolean isKeyguardOrAodShowing(int displayId) {
wilsonshih0299c8a2018-08-24 15:52:57 +080093 return (mKeyguardShowing || mAodShowing) && !mKeyguardGoingAway
wilsonshih498a4b82018-12-11 16:10:16 +080094 && !isDisplayOccluded(displayId);
Lucas Dupin47a65c72018-02-15 14:16:18 -080095 }
96
97 /**
Louis Chang3319d022019-03-12 11:53:14 +080098 * @return {@code true} for default display when AOD is showing. Otherwise, same as
99 * {@link #isKeyguardOrAodShowing(int)}
Issei Suzukid6eb5a22019-02-20 23:08:03 +0100100 * TODO(b/125198167): Replace isKeyguardOrAodShowing() by this logic.
101 */
102 boolean isKeyguardUnoccludedOrAodShowing(int displayId) {
Louis Chang3319d022019-03-12 11:53:14 +0800103 if (displayId == DEFAULT_DISPLAY && mAodShowing) {
104 return true;
105 }
106 return isKeyguardOrAodShowing(displayId);
Issei Suzukid6eb5a22019-02-20 23:08:03 +0100107 }
108
109 /**
David Stevens53a39ea2017-08-23 18:41:49 -0700110 * @return true if Keyguard is showing, not going away, and not being occluded on the given
111 * display, false otherwise
Jorim Jaggife762342016-10-13 14:33:27 +0200112 */
David Stevens53a39ea2017-08-23 18:41:49 -0700113 boolean isKeyguardShowing(int displayId) {
wilsonshih498a4b82018-12-11 16:10:16 +0800114 return mKeyguardShowing && !mKeyguardGoingAway && !isDisplayOccluded(displayId);
Jorim Jaggife762342016-10-13 14:33:27 +0200115 }
116
117 /**
118 * @return true if Keyguard is either showing or occluded, but not going away
119 */
120 boolean isKeyguardLocked() {
121 return mKeyguardShowing && !mKeyguardGoingAway;
122 }
123
124 /**
Bryce Lee271617a2018-03-15 10:39:12 -0700125 * @return {@code true} if the keyguard is going away, {@code false} otherwise.
126 */
127 boolean isKeyguardGoingAway() {
128 // Also check keyguard showing in case value is stale.
129 return mKeyguardGoingAway && mKeyguardShowing;
130 }
131
132 /**
Jorim Jaggife762342016-10-13 14:33:27 +0200133 * Update the Keyguard showing state.
134 */
wilsonshih177261f2019-02-22 12:02:18 +0800135 void setKeyguardShown(boolean keyguardShowing, boolean aodShowing) {
Adrian Roos6ec76b72018-04-25 14:01:11 +0200136 // If keyguard is going away, but SystemUI aborted the transition, need to reset state.
wilsonshih2bc846c2019-05-02 14:35:21 +0800137 final boolean keyguardChanged = keyguardShowing != mKeyguardShowing
138 || mKeyguardGoingAway && keyguardShowing;
139 final boolean aodChanged = aodShowing != mAodShowing;
140 if (!keyguardChanged && !aodChanged) {
Jorim Jaggife762342016-10-13 14:33:27 +0200141 return;
142 }
Lucas Dupin47a65c72018-02-15 14:16:18 -0800143 mKeyguardShowing = keyguardShowing;
144 mAodShowing = aodShowing;
chaviw0e9fb132018-06-05 16:29:13 -0700145 mWindowManager.setAodShowing(aodShowing);
wilsonshih2bc846c2019-05-02 14:35:21 +0800146
147 if (keyguardChanged) {
148 // Irrelevant to AOD.
David Stevens53a39ea2017-08-23 18:41:49 -0700149 dismissDockedStackIfNeeded();
Andrii Kulian0d595f32018-02-21 15:47:33 -0800150 setKeyguardGoingAway(false);
Lucas Dupin47a65c72018-02-15 14:16:18 -0800151 if (keyguardShowing) {
David Stevens53a39ea2017-08-23 18:41:49 -0700152 mDismissalRequested = false;
153 }
Jorim Jaggife762342016-10-13 14:33:27 +0200154 }
wilsonshih2bc846c2019-05-02 14:35:21 +0800155 // TODO(b/113840485): Check usage for non-default display
156 mWindowManager.setKeyguardOrAodShowingOnDefaultDisplay(
157 isKeyguardOrAodShowing(DEFAULT_DISPLAY));
158
159 // Update the sleep token first such that ensureActivitiesVisible has correct sleep token
160 // state when evaluating visibilities.
David Stevens9440dc82017-03-16 19:00:20 -0700161 updateKeyguardSleepToken();
wilsonshih2bc846c2019-05-02 14:35:21 +0800162 mRootActivityContainer.ensureActivitiesVisible(null, 0, !PRESERVE_WINDOWS);
Jorim Jaggife762342016-10-13 14:33:27 +0200163 }
164
165 /**
166 * Called when Keyguard is going away.
167 *
Adrian Roose99bc052017-11-20 17:55:31 +0100168 * @param flags See {@link WindowManagerPolicy#KEYGUARD_GOING_AWAY_FLAG_TO_SHADE}
Jorim Jaggife762342016-10-13 14:33:27 +0200169 * etc.
170 */
171 void keyguardGoingAway(int flags) {
Wale Ogunwaledfb7fb22017-06-23 14:52:40 -0700172 if (!mKeyguardShowing) {
173 return;
174 }
175 Trace.traceBegin(TRACE_TAG_ACTIVITY_MANAGER, "keyguardGoingAway");
176 mWindowManager.deferSurfaceLayout();
177 try {
178 setKeyguardGoingAway(true);
Wale Ogunwale3a256e62018-12-06 14:41:18 -0800179 mRootActivityContainer.getDefaultDisplay().mDisplayContent
lumark588a3e82018-07-20 18:53:54 +0800180 .prepareAppTransition(TRANSIT_KEYGUARD_GOING_AWAY,
181 false /* alwaysKeepCurrent */, convertTransitFlags(flags),
182 false /* forceOverride */);
David Stevens9440dc82017-03-16 19:00:20 -0700183 updateKeyguardSleepToken();
Jorim Jaggife762342016-10-13 14:33:27 +0200184
Wale Ogunwaledfb7fb22017-06-23 14:52:40 -0700185 // Some stack visibility might change (e.g. docked stack)
Wale Ogunwaled32da472018-11-16 07:19:28 -0800186 mRootActivityContainer.resumeFocusedStacksTopActivities();
187 mRootActivityContainer.ensureActivitiesVisible(null, 0, !PRESERVE_WINDOWS);
188 mRootActivityContainer.addStartingWindowsForVisibleActivities(
189 true /* taskSwitch */);
Wale Ogunwaledfb7fb22017-06-23 14:52:40 -0700190 mWindowManager.executeAppTransition();
191 } finally {
192 Trace.traceBegin(TRACE_TAG_ACTIVITY_MANAGER, "keyguardGoingAway: surfaceLayout");
193 mWindowManager.continueSurfaceLayout();
194 Trace.traceEnd(TRACE_TAG_ACTIVITY_MANAGER);
195
196 Trace.traceEnd(TRACE_TAG_ACTIVITY_MANAGER);
Jorim Jaggife762342016-10-13 14:33:27 +0200197 }
198 }
199
Lucas Dupinc80c67e2017-12-04 14:29:10 -0800200 void dismissKeyguard(IBinder token, IKeyguardDismissCallback callback, CharSequence message) {
Jorim Jaggi241ae102016-11-02 21:57:33 -0700201 final ActivityRecord activityRecord = ActivityRecord.forTokenLocked(token);
202 if (activityRecord == null || !activityRecord.visibleIgnoringKeyguard) {
203 failCallback(callback);
204 return;
205 }
Jorim Jaggid7214892017-07-18 14:05:19 +0200206 Slog.i(TAG, "Activity requesting to dismiss Keyguard: " + activityRecord);
chaviw59b98852017-06-13 12:05:44 -0700207
208 // If the client has requested to dismiss the keyguard and the Activity has the flag to
209 // turn the screen on, wakeup the screen if it's the top Activity.
210 if (activityRecord.getTurnScreenOnFlag() && activityRecord.isTopRunningActivity()) {
211 mStackSupervisor.wakeUp("dismissKeyguard");
212 }
213
Lucas Dupinc80c67e2017-12-04 14:29:10 -0800214 mWindowManager.dismissKeyguard(callback, message);
Jorim Jaggi241ae102016-11-02 21:57:33 -0700215 }
216
Wale Ogunwalebfa81ad2017-05-24 15:14:42 -0700217 private void setKeyguardGoingAway(boolean keyguardGoingAway) {
218 mKeyguardGoingAway = keyguardGoingAway;
219 mWindowManager.setKeyguardGoingAway(keyguardGoingAway);
220 }
221
Jorim Jaggi241ae102016-11-02 21:57:33 -0700222 private void failCallback(IKeyguardDismissCallback callback) {
223 try {
224 callback.onDismissError();
225 } catch (RemoteException e) {
226 Slog.w(TAG, "Failed to call callback", e);
227 }
228 }
229
Jorim Jaggife762342016-10-13 14:33:27 +0200230 private int convertTransitFlags(int keyguardGoingAwayFlags) {
231 int result = 0;
232 if ((keyguardGoingAwayFlags & KEYGUARD_GOING_AWAY_FLAG_TO_SHADE) != 0) {
233 result |= TRANSIT_FLAG_KEYGUARD_GOING_AWAY_TO_SHADE;
234 }
235 if ((keyguardGoingAwayFlags & KEYGUARD_GOING_AWAY_FLAG_NO_WINDOW_ANIMATIONS) != 0) {
236 result |= TRANSIT_FLAG_KEYGUARD_GOING_AWAY_NO_ANIMATION;
237 }
238 if ((keyguardGoingAwayFlags & KEYGUARD_GOING_AWAY_FLAG_WITH_WALLPAPER) != 0) {
239 result |= TRANSIT_FLAG_KEYGUARD_GOING_AWAY_WITH_WALLPAPER;
240 }
241 return result;
242 }
243
244 /**
245 * Starts a batch of visibility updates.
246 */
247 void beginActivityVisibilityUpdate() {
248 mVisibilityTransactionDepth++;
249 }
250
251 /**
252 * Ends a batch of visibility updates. After all batches are done, this method makes sure to
253 * update lockscreen occluded/dismiss state if needed.
254 */
255 void endActivityVisibilityUpdate() {
256 mVisibilityTransactionDepth--;
257 if (mVisibilityTransactionDepth == 0) {
258 visibilitiesUpdated();
259 }
260 }
261
Jorim Jaggie69c9312016-10-31 18:24:38 -0700262 /**
263 * @return True if we may show an activity while Keyguard is showing because we are in the
264 * process of dismissing it anyways, false otherwise.
265 */
Jorim Jaggi07961872016-11-23 11:28:57 +0100266 boolean canShowActivityWhileKeyguardShowing(ActivityRecord r, boolean dismissKeyguard) {
267
268 // Allow to show it when we are about to dismiss Keyguard. This isn't allowed if r is
269 // already the dismissing activity, in which case we don't allow it to repeatedly dismiss
270 // Keyguard.
Lucas Dupin47a65c72018-02-15 14:16:18 -0800271 return dismissKeyguard && canDismissKeyguard() && !mAodShowing
wilsonshih0299c8a2018-08-24 15:52:57 +0800272 && (mDismissalRequested
Arthur Hungcf58b332019-04-17 16:07:10 +0800273 || (r.canShowWhenLocked()
274 && getDisplay(r.getDisplayId()).mDismissingKeyguardActivity != r));
Jorim Jaggi07961872016-11-23 11:28:57 +0100275 }
276
277 /**
278 * @return True if we may show an activity while Keyguard is occluded, false otherwise.
279 */
280 boolean canShowWhileOccluded(boolean dismissKeyguard, boolean showWhenLocked) {
Brad Stenning2bdc21e2019-03-11 14:33:22 -0700281 return showWhenLocked || dismissKeyguard
282 && !mWindowManager.isKeyguardSecure(mService.getCurrentUserId());
Jorim Jaggie69c9312016-10-31 18:24:38 -0700283 }
284
Jorim Jaggife762342016-10-13 14:33:27 +0200285 private void visibilitiesUpdated() {
wilsonshih0299c8a2018-08-24 15:52:57 +0800286 boolean requestDismissKeyguard = false;
Wale Ogunwaled32da472018-11-16 07:19:28 -0800287 for (int displayNdx = mRootActivityContainer.getChildCount() - 1;
288 displayNdx >= 0; displayNdx--) {
289 final ActivityDisplay display = mRootActivityContainer.getChildAt(displayNdx);
wilsonshih0299c8a2018-08-24 15:52:57 +0800290 final KeyguardDisplayState state = getDisplay(display.mDisplayId);
291 state.visibilitiesUpdated(this, display);
292 requestDismissKeyguard |= state.mRequestDismissKeyguard;
Jorim Jaggife762342016-10-13 14:33:27 +0200293 }
wilsonshih0299c8a2018-08-24 15:52:57 +0800294
295 // Dismissing Keyguard happens globally using the information from all displays.
296 if (requestDismissKeyguard) {
Jorim Jaggife762342016-10-13 14:33:27 +0200297 handleDismissKeyguard();
298 }
299 }
300
301 /**
302 * Called when occluded state changed.
303 */
Issei Suzuki62356a22019-04-11 16:46:37 +0200304 private void handleOccludedChanged(int displayId) {
305 // TODO(b/113840485): Handle app transition for individual display, and apply occluded
306 // state change to secondary displays.
307 // For now, only default display fully supports occluded change. Other displays only
308 // updates keygaurd sleep token on that display.
309 if (displayId != DEFAULT_DISPLAY) {
310 updateKeyguardSleepToken(displayId);
311 return;
312 }
313
wilsonshih0299c8a2018-08-24 15:52:57 +0800314 mWindowManager.onKeyguardOccludedChanged(isDisplayOccluded(DEFAULT_DISPLAY));
Jorim Jaggife762342016-10-13 14:33:27 +0200315 if (isKeyguardLocked()) {
316 mWindowManager.deferSurfaceLayout();
317 try {
Wale Ogunwale3a256e62018-12-06 14:41:18 -0800318 mRootActivityContainer.getDefaultDisplay().mDisplayContent
lumark588a3e82018-07-20 18:53:54 +0800319 .prepareAppTransition(resolveOccludeTransit(),
320 false /* alwaysKeepCurrent */, 0 /* flags */,
321 true /* forceOverride */);
Issei Suzuki62356a22019-04-11 16:46:37 +0200322 updateKeyguardSleepToken(DEFAULT_DISPLAY);
Wale Ogunwaled32da472018-11-16 07:19:28 -0800323 mRootActivityContainer.ensureActivitiesVisible(null, 0, !PRESERVE_WINDOWS);
Jorim Jaggife762342016-10-13 14:33:27 +0200324 mWindowManager.executeAppTransition();
325 } finally {
326 mWindowManager.continueSurfaceLayout();
327 }
328 }
329 dismissDockedStackIfNeeded();
330 }
331
332 /**
wilsonshih0299c8a2018-08-24 15:52:57 +0800333 * Called when somebody wants to dismiss the Keyguard via the flag.
Jorim Jaggife762342016-10-13 14:33:27 +0200334 */
335 private void handleDismissKeyguard() {
Jorim Jaggi07961872016-11-23 11:28:57 +0100336 // We only allow dismissing Keyguard via the flag when Keyguard is secure for legacy
337 // reasons, because that's how apps used to dismiss Keyguard in the secure case. In the
338 // insecure case, we actually show it on top of the lockscreen. See #canShowWhileOccluded.
Brad Stenning2bdc21e2019-03-11 14:33:22 -0700339 if (!mWindowManager.isKeyguardSecure(mService.getCurrentUserId())) {
Wale Ogunwaled32da472018-11-16 07:19:28 -0800340 return;
341 }
Jorim Jaggife762342016-10-13 14:33:27 +0200342
Wale Ogunwaled32da472018-11-16 07:19:28 -0800343 mWindowManager.dismissKeyguard(null /* callback */, null /* message */);
344 mDismissalRequested = true;
345
346 // If we are about to unocclude the Keyguard, but we can dismiss it without security,
347 // we immediately dismiss the Keyguard so the activity gets shown without a flicker.
Wale Ogunwale3a256e62018-12-06 14:41:18 -0800348 final DisplayContent dc =
349 mRootActivityContainer.getDefaultDisplay().mDisplayContent;
Wale Ogunwaled32da472018-11-16 07:19:28 -0800350 if (mKeyguardShowing && canDismissKeyguard()
Wale Ogunwale3a256e62018-12-06 14:41:18 -0800351 && dc.mAppTransition.getAppTransition() == TRANSIT_KEYGUARD_UNOCCLUDE) {
352 dc.prepareAppTransition(mBeforeUnoccludeTransit, false /* alwaysKeepCurrent */,
Wale Ogunwaled32da472018-11-16 07:19:28 -0800353 0 /* flags */, true /* forceOverride */);
354 mRootActivityContainer.ensureActivitiesVisible(null, 0, !PRESERVE_WINDOWS);
355 mWindowManager.executeAppTransition();
Jorim Jaggife762342016-10-13 14:33:27 +0200356 }
357 }
358
wilsonshih0299c8a2018-08-24 15:52:57 +0800359 private boolean isDisplayOccluded(int displayId) {
360 return getDisplay(displayId).mOccluded;
361 }
362
Jorim Jaggife762342016-10-13 14:33:27 +0200363 /**
364 * @return true if Keyguard can be currently dismissed without entering credentials.
365 */
Andrii Kulianfc8f82b2017-01-26 13:17:27 -0800366 boolean canDismissKeyguard() {
Brad Stenning2bdc21e2019-03-11 14:33:22 -0700367 return mWindowManager.isKeyguardTrusted()
368 || !mWindowManager.isKeyguardSecure(mService.getCurrentUserId());
Jorim Jaggife762342016-10-13 14:33:27 +0200369 }
370
371 private int resolveOccludeTransit() {
Wale Ogunwale3a256e62018-12-06 14:41:18 -0800372 final DisplayContent dc =
373 mService.mRootActivityContainer.getDefaultDisplay().mDisplayContent;
Jorim Jaggife762342016-10-13 14:33:27 +0200374 if (mBeforeUnoccludeTransit != TRANSIT_UNSET
Wale Ogunwale3a256e62018-12-06 14:41:18 -0800375 && dc.mAppTransition.getAppTransition() == TRANSIT_KEYGUARD_UNOCCLUDE
wilsonshih0299c8a2018-08-24 15:52:57 +0800376 // TODO(b/113840485): Handle app transition for individual display.
377 && isDisplayOccluded(DEFAULT_DISPLAY)) {
Jorim Jaggife762342016-10-13 14:33:27 +0200378
379 // Reuse old transit in case we are occluding Keyguard again, meaning that we never
380 // actually occclude/unocclude Keyguard, but just run a normal transition.
381 return mBeforeUnoccludeTransit;
wilsonshih0299c8a2018-08-24 15:52:57 +0800382 // TODO(b/113840485): Handle app transition for individual display.
383 } else if (!isDisplayOccluded(DEFAULT_DISPLAY)) {
Jorim Jaggife762342016-10-13 14:33:27 +0200384
385 // Save transit in case we dismiss/occlude Keyguard shortly after.
Wale Ogunwale3a256e62018-12-06 14:41:18 -0800386 mBeforeUnoccludeTransit = dc.mAppTransition.getAppTransition();
Jorim Jaggife762342016-10-13 14:33:27 +0200387 return TRANSIT_KEYGUARD_UNOCCLUDE;
388 } else {
389 return TRANSIT_KEYGUARD_OCCLUDE;
390 }
391 }
392
393 private void dismissDockedStackIfNeeded() {
wilsonshih0299c8a2018-08-24 15:52:57 +0800394 // TODO(b/113840485): Handle docked stack for individual display.
395 if (mKeyguardShowing && isDisplayOccluded(DEFAULT_DISPLAY)) {
Jorim Jaggife762342016-10-13 14:33:27 +0200396 // The lock screen is currently showing, but is occluded by a window that can
397 // show on top of the lock screen. In this can we want to dismiss the docked
398 // stack since it will be complicated/risky to try to put the activity on top
399 // of the lock screen in the right fullscreen configuration.
Wale Ogunwaled32da472018-11-16 07:19:28 -0800400 final ActivityStack stack =
401 mRootActivityContainer.getDefaultDisplay().getSplitScreenPrimaryStack();
Wale Ogunwale9dcf9462017-09-19 15:13:01 -0700402 if (stack == null) {
403 return;
404 }
405 mStackSupervisor.moveTasksToFullscreenStackLocked(stack,
Andrii Kulian52d255c2018-07-13 11:32:19 -0700406 stack.isFocusedStackOnDisplay());
Jorim Jaggife762342016-10-13 14:33:27 +0200407 }
408 }
Jorim Jaggi8d786932016-10-26 19:08:36 -0700409
David Stevens9440dc82017-03-16 19:00:20 -0700410 private void updateKeyguardSleepToken() {
Wale Ogunwaled32da472018-11-16 07:19:28 -0800411 for (int displayNdx = mRootActivityContainer.getChildCount() - 1;
412 displayNdx >= 0; displayNdx--) {
413 final ActivityDisplay display = mRootActivityContainer.getChildAt(displayNdx);
Issei Suzuki62356a22019-04-11 16:46:37 +0200414 updateKeyguardSleepToken(display.mDisplayId);
415 }
416 }
417
418 private void updateKeyguardSleepToken(int displayId) {
419 final KeyguardDisplayState state = getDisplay(displayId);
420 if (isKeyguardUnoccludedOrAodShowing(displayId) && state.mSleepToken == null) {
421 state.acquiredSleepToken();
422 } else if (!isKeyguardUnoccludedOrAodShowing(displayId) && state.mSleepToken != null) {
423 state.releaseSleepToken();
wilsonshih0299c8a2018-08-24 15:52:57 +0800424 }
425 }
426
427 private KeyguardDisplayState getDisplay(int displayId) {
wilsonshihe7903ea2018-09-26 16:17:59 +0800428 KeyguardDisplayState state = mDisplayStates.get(displayId);
429 if (state == null) {
430 state = new KeyguardDisplayState(mService, displayId);
431 mDisplayStates.append(displayId, state);
wilsonshih0299c8a2018-08-24 15:52:57 +0800432 }
wilsonshihe7903ea2018-09-26 16:17:59 +0800433 return state;
wilsonshih0299c8a2018-08-24 15:52:57 +0800434 }
435
436 void onDisplayRemoved(int displayId) {
wilsonshihe7903ea2018-09-26 16:17:59 +0800437 final KeyguardDisplayState state = mDisplayStates.get(displayId);
438 if (state != null) {
439 state.onRemoved();
wilsonshih0299c8a2018-08-24 15:52:57 +0800440 mDisplayStates.remove(displayId);
441 }
442 }
443
444 /** Represents Keyguard state per individual display. */
445 private static class KeyguardDisplayState {
446 private final int mDisplayId;
447 private boolean mOccluded;
448 private ActivityRecord mDismissingKeyguardActivity;
449 private boolean mRequestDismissKeyguard;
450 private final ActivityTaskManagerService mService;
451 private SleepToken mSleepToken;
452
453 KeyguardDisplayState(ActivityTaskManagerService service, int displayId) {
454 mService = service;
455 mDisplayId = displayId;
456 }
457
458 void onRemoved() {
459 mDismissingKeyguardActivity = null;
460 releaseSleepToken();
461 }
462
463 void acquiredSleepToken() {
464 if (mSleepToken == null) {
465 mSleepToken = mService.acquireSleepToken("keyguard", mDisplayId);
466 }
467 }
468
469 void releaseSleepToken() {
470 if (mSleepToken != null) {
471 mSleepToken.release();
472 mSleepToken = null;
473 }
474 }
475
476 void visibilitiesUpdated(KeyguardController controller, ActivityDisplay display) {
477 final boolean lastOccluded = mOccluded;
478 final ActivityRecord lastDismissActivity = mDismissingKeyguardActivity;
479 mRequestDismissKeyguard = false;
480 mOccluded = false;
481 mDismissingKeyguardActivity = null;
482
wilsonshih9a10e9d2019-01-11 14:39:27 +0800483 final ActivityStack stack = getStackForControllingOccluding(display);
484 if (stack != null) {
485 final ActivityRecord topDismissing = stack.getTopDismissingKeyguardActivity();
486 mOccluded = stack.topActivityOccludesKeyguard() || (topDismissing != null
487 && stack.topRunningActivityLocked() == topDismissing
488 && controller.canShowWhileOccluded(
wilsonshih0299c8a2018-08-24 15:52:57 +0800489 true /* dismissKeyguard */,
490 false /* showWhenLocked */));
wilsonshih9a10e9d2019-01-11 14:39:27 +0800491 if (stack.getTopDismissingKeyguardActivity() != null) {
492 mDismissingKeyguardActivity = stack.getTopDismissingKeyguardActivity();
wilsonshih0299c8a2018-08-24 15:52:57 +0800493 }
wilsonshih498a4b82018-12-11 16:10:16 +0800494 // FLAG_CAN_SHOW_WITH_INSECURE_KEYGUARD only apply for secondary display.
495 if (mDisplayId != DEFAULT_DISPLAY) {
496 mOccluded |= stack.canShowWithInsecureKeyguard()
497 && controller.canDismissKeyguard();
498 }
wilsonshih0299c8a2018-08-24 15:52:57 +0800499 }
wilsonshih498a4b82018-12-11 16:10:16 +0800500 // TODO(b/123372519): isShowingDream can only works on default display.
501 if (mDisplayId == DEFAULT_DISPLAY) {
502 mOccluded |= controller.mWindowManager.isShowingDream();
503 }
wilsonshih0299c8a2018-08-24 15:52:57 +0800504
Issei Suzuki62356a22019-04-11 16:46:37 +0200505 if (lastOccluded != mOccluded) {
506 controller.handleOccludedChanged(mDisplayId);
wilsonshih0299c8a2018-08-24 15:52:57 +0800507 }
508 if (lastDismissActivity != mDismissingKeyguardActivity && !mOccluded
509 && mDismissingKeyguardActivity != null
Brad Stenning2bdc21e2019-03-11 14:33:22 -0700510 && controller.mWindowManager.isKeyguardSecure(
511 controller.mService.getCurrentUserId())) {
wilsonshih0299c8a2018-08-24 15:52:57 +0800512 mRequestDismissKeyguard = true;
513 }
514 }
515
wilsonshih9a10e9d2019-01-11 14:39:27 +0800516 /**
517 * Gets the stack used to check the occluded state.
518 * <p>
519 * Only the top non-pinned activity of the focusable stack on each display can control its
520 * occlusion state.
521 */
522 private ActivityStack getStackForControllingOccluding(ActivityDisplay display) {
523 for (int stackNdx = display.getChildCount() - 1; stackNdx >= 0; --stackNdx) {
524 final ActivityStack stack = display.getChildAt(stackNdx);
525 if (stack != null && stack.isFocusableAndVisible()
526 && !stack.inPinnedWindowingMode()) {
527 return stack;
528 }
529 }
530 return null;
531 }
532
wilsonshih0299c8a2018-08-24 15:52:57 +0800533 void dumpStatus(PrintWriter pw, String prefix) {
534 final StringBuilder sb = new StringBuilder();
535 sb.append(prefix);
536 sb.append(" Occluded=").append(mOccluded)
537 .append(" DismissingKeyguardActivity=")
538 .append(mDismissingKeyguardActivity)
539 .append(" at display=")
540 .append(mDisplayId);
541 pw.println(sb.toString());
542 }
543
544 void writeToProto(ProtoOutputStream proto, long fieldId) {
545 final long token = proto.start(fieldId);
546 proto.write(DISPLAY_ID, mDisplayId);
547 proto.write(KEYGUARD_OCCLUDED, mOccluded);
548 proto.end(token);
David Stevens9440dc82017-03-16 19:00:20 -0700549 }
550 }
551
Jorim Jaggi8d786932016-10-26 19:08:36 -0700552 void dump(PrintWriter pw, String prefix) {
553 pw.println(prefix + "KeyguardController:");
554 pw.println(prefix + " mKeyguardShowing=" + mKeyguardShowing);
Lucas Dupin47a65c72018-02-15 14:16:18 -0800555 pw.println(prefix + " mAodShowing=" + mAodShowing);
Jorim Jaggi8d786932016-10-26 19:08:36 -0700556 pw.println(prefix + " mKeyguardGoingAway=" + mKeyguardGoingAway);
wilsonshih0299c8a2018-08-24 15:52:57 +0800557 dumpDisplayStates(pw, prefix);
Jorim Jaggi07961872016-11-23 11:28:57 +0100558 pw.println(prefix + " mDismissalRequested=" + mDismissalRequested);
Jorim Jaggi8d786932016-10-26 19:08:36 -0700559 pw.println(prefix + " mVisibilityTransactionDepth=" + mVisibilityTransactionDepth);
560 }
Steven Timotius4346f0a2017-09-12 11:07:21 -0700561
562 void writeToProto(ProtoOutputStream proto, long fieldId) {
563 final long token = proto.start(fieldId);
Issei Suzukid6eb5a22019-02-20 23:08:03 +0100564 proto.write(AOD_SHOWING, mAodShowing);
Steven Timotius4346f0a2017-09-12 11:07:21 -0700565 proto.write(KEYGUARD_SHOWING, mKeyguardShowing);
wilsonshih0299c8a2018-08-24 15:52:57 +0800566 writeDisplayStatesToProto(proto, KEYGUARD_OCCLUDED_STATES);
Steven Timotius4346f0a2017-09-12 11:07:21 -0700567 proto.end(token);
568 }
wilsonshih0299c8a2018-08-24 15:52:57 +0800569
570 private void dumpDisplayStates(PrintWriter pw, String prefix) {
571 for (int i = 0; i < mDisplayStates.size(); i++) {
572 mDisplayStates.valueAt(i).dumpStatus(pw, prefix);
573 }
574 }
575
576 private void writeDisplayStatesToProto(ProtoOutputStream proto, long fieldId) {
577 for (int i = 0; i < mDisplayStates.size(); i++) {
578 mDisplayStates.valueAt(i).writeToProto(proto, fieldId);
579 }
580 }
Jorim Jaggife762342016-10-13 14:33:27 +0200581}