Home | History | Annotate | Download | only in wm
      1 /*
      2  * Copyright (C) 2007 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.server.wm;
     18 
     19 import android.Manifest;
     20 import android.animation.ValueAnimator;
     21 import android.annotation.IntDef;
     22 import android.annotation.NonNull;
     23 import android.annotation.Nullable;
     24 import android.app.ActivityManagerInternal;
     25 import android.app.ActivityManagerNative;
     26 import android.app.AppOpsManager;
     27 import android.app.IActivityManager;
     28 import android.app.Notification;
     29 import android.app.NotificationManager;
     30 import android.app.PendingIntent;
     31 import android.app.admin.DevicePolicyManager;
     32 import android.content.BroadcastReceiver;
     33 import android.content.ContentResolver;
     34 import android.content.Context;
     35 import android.content.Intent;
     36 import android.content.IntentFilter;
     37 import android.content.pm.ActivityInfo;
     38 import android.content.pm.PackageManager;
     39 import android.content.res.CompatibilityInfo;
     40 import android.content.res.Configuration;
     41 import android.database.ContentObserver;
     42 import android.graphics.Bitmap;
     43 import android.graphics.PixelFormat;
     44 import android.graphics.Matrix;
     45 import android.graphics.Point;
     46 import android.graphics.Rect;
     47 import android.graphics.RectF;
     48 import android.graphics.Region;
     49 import android.hardware.display.DisplayManager;
     50 import android.hardware.display.DisplayManagerInternal;
     51 import android.hardware.input.InputManager;
     52 import android.net.Uri;
     53 import android.os.Binder;
     54 import android.os.Build;
     55 import android.os.Bundle;
     56 import android.os.Debug;
     57 import android.os.Handler;
     58 import android.os.IBinder;
     59 import android.os.IRemoteCallback;
     60 import android.os.Looper;
     61 import android.os.Message;
     62 import android.os.Parcel;
     63 import android.os.ParcelFileDescriptor;
     64 import android.os.PowerManager;
     65 import android.os.PowerManagerInternal;
     66 import android.os.Process;
     67 import android.os.RemoteException;
     68 import android.os.ServiceManager;
     69 import android.os.StrictMode;
     70 import android.os.SystemClock;
     71 import android.os.SystemProperties;
     72 import android.os.SystemService;
     73 import android.os.Trace;
     74 import android.os.UserHandle;
     75 import android.os.WorkSource;
     76 import android.provider.Settings;
     77 import android.util.ArraySet;
     78 import android.util.DisplayMetrics;
     79 import android.util.EventLog;
     80 import android.util.Log;
     81 import android.util.Pair;
     82 import android.util.Slog;
     83 import android.util.SparseArray;
     84 import android.util.SparseIntArray;
     85 import android.util.TimeUtils;
     86 import android.util.TypedValue;
     87 import android.view.AppTransitionAnimationSpec;
     88 import android.view.Choreographer;
     89 import android.view.Display;
     90 import android.view.DisplayInfo;
     91 import android.view.Gravity;
     92 import android.view.PointerIcon;
     93 import android.view.IAppTransitionAnimationSpecsFuture;
     94 import android.view.IApplicationToken;
     95 import android.view.IDockedStackListener;
     96 import android.view.IInputFilter;
     97 import android.view.IOnKeyguardExitResult;
     98 import android.view.IRotationWatcher;
     99 import android.view.IWindow;
    100 import android.view.IWindowId;
    101 import android.view.IWindowManager;
    102 import android.view.IWindowSession;
    103 import android.view.IWindowSessionCallback;
    104 import android.view.InputChannel;
    105 import android.view.InputDevice;
    106 import android.view.InputEvent;
    107 import android.view.InputEventReceiver;
    108 import android.view.KeyEvent;
    109 import android.view.MagnificationSpec;
    110 import android.view.MotionEvent;
    111 import android.view.Surface;
    112 import android.view.Surface.OutOfResourcesException;
    113 import android.view.SurfaceControl;
    114 import android.view.SurfaceSession;
    115 import android.view.View;
    116 import android.view.WindowContentFrameStats;
    117 import android.view.WindowManager;
    118 import android.view.WindowManager.LayoutParams;
    119 import android.view.WindowManagerGlobal;
    120 import android.view.WindowManagerInternal;
    121 import android.view.WindowManagerPolicy;
    122 import android.view.WindowManagerPolicy.PointerEventListener;
    123 import android.view.animation.Animation;
    124 import android.view.inputmethod.InputMethodManagerInternal;
    125 
    126 import com.android.internal.R;
    127 import com.android.internal.app.IAssistScreenshotReceiver;
    128 import com.android.internal.os.IResultReceiver;
    129 import com.android.internal.policy.IShortcutService;
    130 import com.android.internal.util.ArrayUtils;
    131 import com.android.internal.util.FastPrintWriter;
    132 import com.android.internal.view.IInputContext;
    133 import com.android.internal.view.IInputMethodClient;
    134 import com.android.internal.view.IInputMethodManager;
    135 import com.android.internal.view.WindowManagerPolicyThread;
    136 import com.android.server.AttributeCache;
    137 import com.android.server.DisplayThread;
    138 import com.android.server.EventLogTags;
    139 import com.android.server.FgThread;
    140 import com.android.server.LocalServices;
    141 import com.android.server.UiThread;
    142 import com.android.server.Watchdog;
    143 import com.android.server.input.InputManagerService;
    144 import com.android.server.policy.PhoneWindowManager;
    145 import com.android.server.power.ShutdownThread;
    146 
    147 import java.io.BufferedWriter;
    148 import java.io.DataInputStream;
    149 import java.io.File;
    150 import java.io.FileDescriptor;
    151 import java.io.FileInputStream;
    152 import java.io.FileNotFoundException;
    153 import java.io.IOException;
    154 import java.io.OutputStream;
    155 import java.io.OutputStreamWriter;
    156 import java.io.PrintWriter;
    157 import java.io.StringWriter;
    158 import java.lang.annotation.Retention;
    159 import java.lang.annotation.RetentionPolicy;
    160 import java.net.Socket;
    161 import java.text.DateFormat;
    162 import java.util.ArrayList;
    163 import java.util.Arrays;
    164 import java.util.Date;
    165 import java.util.HashMap;
    166 import java.util.Iterator;
    167 import java.util.List;
    168 
    169 import static android.app.ActivityManager.DOCKED_STACK_CREATE_MODE_TOP_OR_LEFT;
    170 import static android.app.ActivityManager.StackId.DOCKED_STACK_ID;
    171 import static android.app.ActivityManager.StackId.FREEFORM_WORKSPACE_STACK_ID;
    172 import static android.app.ActivityManager.StackId.PINNED_STACK_ID;
    173 import static android.app.StatusBarManager.DISABLE_MASK;
    174 import static android.content.pm.ActivityInfo.SCREEN_ORIENTATION_BEHIND;
    175 import static android.content.pm.ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED;
    176 import static android.view.WindowManager.DOCKED_BOTTOM;
    177 import static android.view.WindowManager.DOCKED_INVALID;
    178 import static android.view.WindowManager.DOCKED_TOP;
    179 import static android.view.WindowManager.LayoutParams.FIRST_APPLICATION_WINDOW;
    180 import static android.view.WindowManager.LayoutParams.FIRST_SUB_WINDOW;
    181 import static android.view.WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM;
    182 import static android.view.WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON;
    183 import static android.view.WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
    184 import static android.view.WindowManager.LayoutParams.FLAG_SECURE;
    185 import static android.view.WindowManager.LayoutParams.FLAG_SHOW_WALLPAPER;
    186 import static android.view.WindowManager.LayoutParams.INPUT_FEATURE_NO_INPUT_CHANNEL;
    187 import static android.view.WindowManager.LayoutParams.LAST_APPLICATION_WINDOW;
    188 import static android.view.WindowManager.LayoutParams.LAST_SUB_WINDOW;
    189 import static android.view.WindowManager.LayoutParams.PRIVATE_FLAG_COMPATIBLE_WINDOW;
    190 import static android.view.WindowManager.LayoutParams.TYPE_ACCESSIBILITY_OVERLAY;
    191 import static android.view.WindowManager.LayoutParams.TYPE_APPLICATION;
    192 import static android.view.WindowManager.LayoutParams.TYPE_APPLICATION_STARTING;
    193 import static android.view.WindowManager.LayoutParams.TYPE_BASE_APPLICATION;
    194 import static android.view.WindowManager.LayoutParams.TYPE_DRAWN_APPLICATION;
    195 import static android.view.WindowManager.LayoutParams.TYPE_BOOT_PROGRESS;
    196 import static android.view.WindowManager.LayoutParams.TYPE_DOCK_DIVIDER;
    197 import static android.view.WindowManager.LayoutParams.TYPE_DREAM;
    198 import static android.view.WindowManager.LayoutParams.TYPE_INPUT_METHOD;
    199 import static android.view.WindowManager.LayoutParams.TYPE_INPUT_METHOD_DIALOG;
    200 import static android.view.WindowManager.LayoutParams.TYPE_NAVIGATION_BAR;
    201 import static android.view.WindowManager.LayoutParams.TYPE_PRIVATE_PRESENTATION;
    202 import static android.view.WindowManager.LayoutParams.TYPE_QS_DIALOG;
    203 import static android.view.WindowManager.LayoutParams.TYPE_STATUS_BAR;
    204 import static android.view.WindowManager.LayoutParams.TYPE_VOICE_INTERACTION;
    205 import static android.view.WindowManager.LayoutParams.TYPE_WALLPAPER;
    206 import static android.view.WindowManagerGlobal.RELAYOUT_DEFER_SURFACE_DESTROY;
    207 import static android.view.WindowManagerGlobal.RELAYOUT_RES_SURFACE_CHANGED;
    208 import static android.view.WindowManagerPolicy.FINISH_LAYOUT_REDO_WALLPAPER;
    209 import static android.view.WindowManagerPolicy.TRANSIT_EXIT;
    210 import static android.view.WindowManagerPolicy.TRANSIT_PREVIEW_DONE;
    211 import static com.android.server.wm.AppWindowAnimator.PROLONG_ANIMATION_AT_END;
    212 import static com.android.server.wm.AppWindowAnimator.PROLONG_ANIMATION_AT_START;
    213 import static com.android.server.wm.DragResizeMode.DRAG_RESIZE_MODE_DOCKED_DIVIDER;
    214 import static com.android.server.wm.DragResizeMode.DRAG_RESIZE_MODE_FREEFORM;
    215 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG;
    216 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_ADD_REMOVE;
    217 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_ANIM;
    218 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_APP_ORIENTATION;
    219 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_APP_TRANSITIONS;
    220 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_BOOT;
    221 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_CONFIGURATION;
    222 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_DISPLAY;
    223 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_DRAG;
    224 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_FOCUS;
    225 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_FOCUS_LIGHT;
    226 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_INPUT_METHOD;
    227 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_KEYGUARD;
    228 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_KEEP_SCREEN_ON;
    229 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_LAYOUT;
    230 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_ORIENTATION;
    231 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_RESIZE;
    232 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_SCREENSHOT;
    233 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_SCREEN_ON;
    234 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_STACK;
    235 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_STARTING_WINDOW;
    236 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_SURFACE_TRACE;
    237 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_TASK_POSITIONING;
    238 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_TOKEN_MOVEMENT;
    239 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_VISIBILITY;
    240 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_WALLPAPER_LIGHT;
    241 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_WINDOW_MOVEMENT;
    242 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_WINDOW_TRACE;
    243 import static com.android.server.wm.WindowManagerDebugConfig.SHOW_LIGHT_TRANSACTIONS;
    244 import static com.android.server.wm.WindowManagerDebugConfig.SHOW_STACK_CRAWLS;
    245 import static com.android.server.wm.WindowManagerDebugConfig.SHOW_SURFACE_ALLOC;
    246 import static com.android.server.wm.WindowManagerDebugConfig.SHOW_TRANSACTIONS;
    247 import static com.android.server.wm.WindowManagerDebugConfig.SHOW_VERBOSE_TRANSACTIONS;
    248 import static com.android.server.wm.WindowManagerDebugConfig.TAG_KEEP_SCREEN_ON;
    249 import static com.android.server.wm.WindowManagerDebugConfig.TAG_WITH_CLASS_NAME;
    250 import static com.android.server.wm.WindowManagerDebugConfig.TAG_WM;
    251 import static com.android.server.wm.WindowStateAnimator.DRAW_PENDING;
    252 
    253 /** {@hide} */
    254 public class WindowManagerService extends IWindowManager.Stub
    255         implements Watchdog.Monitor, WindowManagerPolicy.WindowManagerFuncs {
    256     private static final String TAG = TAG_WITH_CLASS_NAME ? "WindowManagerService" : TAG_WM;
    257 
    258     static final int LAYOUT_REPEAT_THRESHOLD = 4;
    259 
    260     static final boolean PROFILE_ORIENTATION = false;
    261     static final boolean localLOGV = DEBUG;
    262 
    263     /** How much to multiply the policy's type layer, to reserve room
    264      * for multiple windows of the same type and Z-ordering adjustment
    265      * with TYPE_LAYER_OFFSET. */
    266     static final int TYPE_LAYER_MULTIPLIER = 10000;
    267 
    268     /** Offset from TYPE_LAYER_MULTIPLIER for moving a group of windows above
    269      * or below others in the same layer. */
    270     static final int TYPE_LAYER_OFFSET = 1000;
    271 
    272     /** How much to increment the layer for each window, to reserve room
    273      * for effect surfaces between them.
    274      */
    275     static final int WINDOW_LAYER_MULTIPLIER = 5;
    276 
    277     /**
    278      * Dim surface layer is immediately below target window.
    279      */
    280     static final int LAYER_OFFSET_DIM = 1;
    281 
    282     /**
    283      * Animation thumbnail is as far as possible below the window above
    284      * the thumbnail (or in other words as far as possible above the window
    285      * below it).
    286      */
    287     static final int LAYER_OFFSET_THUMBNAIL = WINDOW_LAYER_MULTIPLIER - 1;
    288 
    289     /** The maximum length we will accept for a loaded animation duration:
    290      * this is 10 seconds.
    291      */
    292     static final int MAX_ANIMATION_DURATION = 10 * 1000;
    293 
    294     /** Amount of time (in milliseconds) to delay before declaring a window freeze timeout. */
    295     static final int WINDOW_FREEZE_TIMEOUT_DURATION = 2000;
    296 
    297     /** Amount of time (in milliseconds) to delay before declaring a seamless rotation timeout. */
    298     static final int SEAMLESS_ROTATION_TIMEOUT_DURATION = 2000;
    299 
    300     /** Amount of time (in milliseconds) to delay before declaring a window replacement timeout. */
    301     static final int WINDOW_REPLACEMENT_TIMEOUT_DURATION = 2000;
    302 
    303     /** Amount of time to allow a last ANR message to exist before freeing the memory. */
    304     static final int LAST_ANR_LIFETIME_DURATION_MSECS = 2 * 60 * 60 * 1000; // Two hours
    305     /**
    306      * If true, the window manager will do its own custom freezing and general
    307      * management of the screen during rotation.
    308      */
    309     static final boolean CUSTOM_SCREEN_ROTATION = true;
    310 
    311     // Maximum number of milliseconds to wait for input devices to be enumerated before
    312     // proceding with safe mode detection.
    313     private static final int INPUT_DEVICES_READY_FOR_SAFE_MODE_DETECTION_TIMEOUT_MILLIS = 1000;
    314 
    315     // Default input dispatching timeout in nanoseconds.
    316     static final long DEFAULT_INPUT_DISPATCHING_TIMEOUT_NANOS = 5000 * 1000000L;
    317 
    318     // Poll interval in milliseconds for watching boot animation finished.
    319     private static final int BOOT_ANIMATION_POLL_INTERVAL = 200;
    320 
    321     // The name of the boot animation service in init.rc.
    322     private static final String BOOT_ANIMATION_SERVICE = "bootanim";
    323 
    324     static final int UPDATE_FOCUS_NORMAL = 0;
    325     static final int UPDATE_FOCUS_WILL_ASSIGN_LAYERS = 1;
    326     static final int UPDATE_FOCUS_PLACING_SURFACES = 2;
    327     static final int UPDATE_FOCUS_WILL_PLACE_SURFACES = 3;
    328 
    329     private static final String SYSTEM_SECURE = "ro.secure";
    330     private static final String SYSTEM_DEBUGGABLE = "ro.debuggable";
    331 
    332     private static final String DENSITY_OVERRIDE = "ro.config.density_override";
    333     private static final String SIZE_OVERRIDE = "ro.config.size_override";
    334 
    335     private static final int MAX_SCREENSHOT_RETRIES = 3;
    336 
    337     private static final String PROPERTY_EMULATOR_CIRCULAR = "ro.emulator.circular";
    338 
    339     // Used to indicate that if there is already a transition set, it should be preserved when
    340     // trying to apply a new one.
    341     private static final boolean ALWAYS_KEEP_CURRENT = true;
    342 
    343     private static final float DRAG_SHADOW_ALPHA_TRANSPARENT = .7071f;
    344 
    345     private static final String PROPERTY_BUILD_DATE_UTC = "ro.build.date.utc";
    346 
    347     // Enums for animation scale update types.
    348     @Retention(RetentionPolicy.SOURCE)
    349     @IntDef({WINDOW_ANIMATION_SCALE, TRANSITION_ANIMATION_SCALE, ANIMATION_DURATION_SCALE})
    350     private @interface UpdateAnimationScaleMode {};
    351     private static final int WINDOW_ANIMATION_SCALE = 0;
    352     private static final int TRANSITION_ANIMATION_SCALE = 1;
    353     private static final int ANIMATION_DURATION_SCALE = 2;
    354 
    355     final private KeyguardDisableHandler mKeyguardDisableHandler;
    356 
    357     final BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() {
    358         @Override
    359         public void onReceive(Context context, Intent intent) {
    360             final String action = intent.getAction();
    361             if (DevicePolicyManager.ACTION_DEVICE_POLICY_MANAGER_STATE_CHANGED.equals(action)) {
    362                 mKeyguardDisableHandler.sendEmptyMessage(
    363                     KeyguardDisableHandler.KEYGUARD_POLICY_CHANGED);
    364             }
    365         }
    366     };
    367     final WindowSurfacePlacer mWindowPlacerLocked;
    368 
    369     /**
    370      * Current user when multi-user is enabled. Don't show windows of
    371      * non-current user. Also see mCurrentProfileIds.
    372      */
    373     int mCurrentUserId;
    374     /**
    375      * Users that are profiles of the current user. These are also allowed to show windows
    376      * on the current user.
    377      */
    378     int[] mCurrentProfileIds = new int[] {};
    379 
    380     final Context mContext;
    381 
    382     final boolean mHaveInputMethods;
    383 
    384     final boolean mHasPermanentDpad;
    385     final long mDrawLockTimeoutMillis;
    386     final boolean mAllowAnimationsInLowPowerMode;
    387 
    388     final boolean mAllowBootMessages;
    389 
    390     final boolean mLimitedAlphaCompositing;
    391 
    392     final WindowManagerPolicy mPolicy = new PhoneWindowManager();
    393 
    394     final IActivityManager mActivityManager;
    395     final ActivityManagerInternal mAmInternal;
    396 
    397     final AppOpsManager mAppOps;
    398 
    399     final DisplaySettings mDisplaySettings;
    400 
    401     /**
    402      * All currently active sessions with clients.
    403      */
    404     final ArraySet<Session> mSessions = new ArraySet<>();
    405 
    406     /**
    407      * Mapping from an IWindow IBinder to the server's Window object.
    408      * This is also used as the lock for all of our state.
    409      * NOTE: Never call into methods that lock ActivityManagerService while holding this object.
    410      */
    411     final HashMap<IBinder, WindowState> mWindowMap = new HashMap<>();
    412 
    413     /**
    414      * Mapping from a token IBinder to a WindowToken object.
    415      */
    416     final HashMap<IBinder, WindowToken> mTokenMap = new HashMap<>();
    417 
    418     /**
    419      * List of window tokens that have finished starting their application,
    420      * and now need to have the policy remove their windows.
    421      */
    422     final ArrayList<AppWindowToken> mFinishedStarting = new ArrayList<>();
    423 
    424     /**
    425      * List of window tokens that have finished drawing their own windows and
    426      * no longer need to show any saved surfaces. Windows that's still showing
    427      * saved surfaces will be cleaned up after next animation pass.
    428      */
    429     final ArrayList<AppWindowToken> mFinishedEarlyAnim = new ArrayList<>();
    430 
    431     /**
    432      * List of app window tokens that are waiting for replacing windows. If the
    433      * replacement doesn't come in time the stale windows needs to be disposed of.
    434      */
    435     final ArrayList<AppWindowToken> mReplacingWindowTimeouts = new ArrayList<>();
    436 
    437     /**
    438      * The input consumer added to the window manager which consumes input events to windows below
    439      * it.
    440      */
    441     InputConsumerImpl mInputConsumer;
    442 
    443     /**
    444      * The input consumer added to the window manager before all wallpaper windows.
    445      */
    446     InputConsumerImpl mWallpaperInputConsumer;
    447 
    448     /**
    449      * Windows that are being resized.  Used so we can tell the client about
    450      * the resize after closing the transaction in which we resized the
    451      * underlying surface.
    452      */
    453     final ArrayList<WindowState> mResizingWindows = new ArrayList<>();
    454 
    455     /**
    456      * Windows whose animations have ended and now must be removed.
    457      */
    458     final ArrayList<WindowState> mPendingRemove = new ArrayList<>();
    459 
    460     /**
    461      * Used when processing mPendingRemove to avoid working on the original array.
    462      */
    463     WindowState[] mPendingRemoveTmp = new WindowState[20];
    464 
    465     /**
    466      * Windows whose surface should be destroyed.
    467      */
    468     final ArrayList<WindowState> mDestroySurface = new ArrayList<>();
    469 
    470     /**
    471      * Windows with a preserved surface waiting to be destroyed. These windows
    472      * are going through a surface change. We keep the old surface around until
    473      * the first frame on the new surface finishes drawing.
    474      */
    475     final ArrayList<WindowState> mDestroyPreservedSurface = new ArrayList<>();
    476 
    477     /**
    478      * Windows that have lost input focus and are waiting for the new
    479      * focus window to be displayed before they are told about this.
    480      */
    481     ArrayList<WindowState> mLosingFocus = new ArrayList<>();
    482 
    483     /**
    484      * This is set when we have run out of memory, and will either be an empty
    485      * list or contain windows that need to be force removed.
    486      */
    487     final ArrayList<WindowState> mForceRemoves = new ArrayList<>();
    488 
    489     /**
    490      * Windows that clients are waiting to have drawn.
    491      */
    492     ArrayList<WindowState> mWaitingForDrawn = new ArrayList<>();
    493     /**
    494      * And the callback to make when they've all been drawn.
    495      */
    496     Runnable mWaitingForDrawnCallback;
    497 
    498     /**
    499      * Used when rebuilding window list to keep track of windows that have
    500      * been removed.
    501      */
    502     WindowState[] mRebuildTmp = new WindowState[20];
    503 
    504     /**
    505      * Stores for each user whether screencapture is disabled
    506      * This array is essentially a cache for all userId for
    507      * {@link android.app.admin.DevicePolicyManager#getScreenCaptureDisabled}
    508      */
    509     SparseArray<Boolean> mScreenCaptureDisabled = new SparseArray<>();
    510 
    511     IInputMethodManager mInputMethodManager;
    512 
    513     AccessibilityController mAccessibilityController;
    514 
    515     final SurfaceSession mFxSession;
    516     Watermark mWatermark;
    517     StrictModeFlash mStrictModeFlash;
    518     CircularDisplayMask mCircularDisplayMask;
    519     EmulatorDisplayOverlay mEmulatorDisplayOverlay;
    520 
    521     final float[] mTmpFloats = new float[9];
    522     final Rect mTmpRect = new Rect();
    523     final Rect mTmpRect2 = new Rect();
    524     final Rect mTmpRect3 = new Rect();
    525     final RectF mTmpRectF = new RectF();
    526 
    527     final Matrix mTmpTransform = new Matrix();
    528 
    529     boolean mDisplayReady;
    530     boolean mSafeMode;
    531     boolean mDisplayEnabled = false;
    532     boolean mSystemBooted = false;
    533     boolean mForceDisplayEnabled = false;
    534     boolean mShowingBootMessages = false;
    535     boolean mBootAnimationStopped = false;
    536 
    537     // Following variables are for debugging screen wakelock only.
    538     WindowState mLastWakeLockHoldingWindow = null;
    539     WindowState mLastWakeLockObscuringWindow = null;
    540 
    541     /** Dump of the windows and app tokens at the time of the last ANR. Cleared after
    542      * LAST_ANR_LIFETIME_DURATION_MSECS */
    543     String mLastANRState;
    544 
    545     /** All DisplayContents in the world, kept here */
    546     SparseArray<DisplayContent> mDisplayContents = new SparseArray<>(2);
    547 
    548     int mRotation = 0;
    549     int mLastOrientation = ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED;
    550     boolean mAltOrientation = false;
    551 
    552     private boolean mKeyguardWaitingForActivityDrawn;
    553 
    554     int mDockedStackCreateMode = DOCKED_STACK_CREATE_MODE_TOP_OR_LEFT;
    555     Rect mDockedStackCreateBounds;
    556 
    557     private final SparseIntArray mTmpTaskIds = new SparseIntArray();
    558 
    559     private final ArrayList<Integer> mChangedStackList = new ArrayList();
    560 
    561     boolean mForceResizableTasks = false;
    562 
    563     int getDragLayerLocked() {
    564         return mPolicy.windowTypeToLayerLw(LayoutParams.TYPE_DRAG) * TYPE_LAYER_MULTIPLIER
    565                 + TYPE_LAYER_OFFSET;
    566     }
    567 
    568     class RotationWatcher {
    569         IRotationWatcher watcher;
    570         IBinder.DeathRecipient deathRecipient;
    571         RotationWatcher(IRotationWatcher w, IBinder.DeathRecipient d) {
    572             watcher = w;
    573             deathRecipient = d;
    574         }
    575     }
    576     ArrayList<RotationWatcher> mRotationWatchers = new ArrayList<>();
    577     int mDeferredRotationPauseCount;
    578 
    579     int mSystemDecorLayer = 0;
    580     final Rect mScreenRect = new Rect();
    581 
    582     boolean mDisplayFrozen = false;
    583     long mDisplayFreezeTime = 0;
    584     int mLastDisplayFreezeDuration = 0;
    585     Object mLastFinishedFreezeSource = null;
    586     boolean mWaitingForConfig = false;
    587 
    588     final static int WINDOWS_FREEZING_SCREENS_NONE = 0;
    589     final static int WINDOWS_FREEZING_SCREENS_ACTIVE = 1;
    590     final static int WINDOWS_FREEZING_SCREENS_TIMEOUT = 2;
    591     int mWindowsFreezingScreen = WINDOWS_FREEZING_SCREENS_NONE;
    592 
    593     boolean mClientFreezingScreen = false;
    594     int mAppsFreezingScreen = 0;
    595     int mLastWindowForcedOrientation = ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED;
    596     int mLastKeyguardForcedOrientation = ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED;
    597 
    598     int mLayoutSeq = 0;
    599 
    600     // Last systemUiVisibility we received from status bar.
    601     int mLastStatusBarVisibility = 0;
    602     // Last systemUiVisibility we dispatched to windows.
    603     int mLastDispatchedSystemUiVisibility = 0;
    604 
    605     // State while inside of layoutAndPlaceSurfacesLocked().
    606     boolean mFocusMayChange;
    607 
    608     Configuration mCurConfiguration = new Configuration();
    609 
    610     // This is held as long as we have the screen frozen, to give us time to
    611     // perform a rotation animation when turning off shows the lock screen which
    612     // changes the orientation.
    613     private final PowerManager.WakeLock mScreenFrozenLock;
    614 
    615     final AppTransition mAppTransition;
    616     boolean mSkipAppTransitionAnimation = false;
    617 
    618     final ArraySet<AppWindowToken> mOpeningApps = new ArraySet<>();
    619     final ArraySet<AppWindowToken> mClosingApps = new ArraySet<>();
    620 
    621     boolean mIsTouchDevice;
    622 
    623     final DisplayMetrics mDisplayMetrics = new DisplayMetrics();
    624     final DisplayMetrics mRealDisplayMetrics = new DisplayMetrics();
    625     final DisplayMetrics mTmpDisplayMetrics = new DisplayMetrics();
    626     final DisplayMetrics mCompatDisplayMetrics = new DisplayMetrics();
    627 
    628     final H mH = new H();
    629 
    630     final Choreographer mChoreographer = Choreographer.getInstance();
    631 
    632     WindowState mCurrentFocus = null;
    633     WindowState mLastFocus = null;
    634 
    635     /** This just indicates the window the input method is on top of, not
    636      * necessarily the window its input is going to. */
    637     WindowState mInputMethodTarget = null;
    638 
    639     /** If true hold off on modifying the animation layer of mInputMethodTarget */
    640     boolean mInputMethodTargetWaitingAnim;
    641 
    642     WindowState mInputMethodWindow = null;
    643     final ArrayList<WindowState> mInputMethodDialogs = new ArrayList<>();
    644 
    645     /** Temporary list for comparison. Always clear this after use so we don't end up with
    646      * orphaned windows references */
    647     final ArrayList<WindowState> mTmpWindows = new ArrayList<>();
    648 
    649     boolean mHardKeyboardAvailable;
    650     WindowManagerInternal.OnHardKeyboardStatusChangeListener mHardKeyboardStatusChangeListener;
    651     SettingsObserver mSettingsObserver;
    652 
    653     private final class SettingsObserver extends ContentObserver {
    654         private final Uri mDisplayInversionEnabledUri =
    655                 Settings.Secure.getUriFor(Settings.Secure.ACCESSIBILITY_DISPLAY_INVERSION_ENABLED);
    656         private final Uri mWindowAnimationScaleUri =
    657                 Settings.Global.getUriFor(Settings.Global.WINDOW_ANIMATION_SCALE);
    658         private final Uri mTransitionAnimationScaleUri =
    659                 Settings.Global.getUriFor(Settings.Global.TRANSITION_ANIMATION_SCALE);
    660         private final Uri mAnimationDurationScaleUri =
    661                 Settings.Global.getUriFor(Settings.Global.ANIMATOR_DURATION_SCALE);
    662 
    663         public SettingsObserver() {
    664             super(new Handler());
    665             ContentResolver resolver = mContext.getContentResolver();
    666             resolver.registerContentObserver(mDisplayInversionEnabledUri, false, this,
    667                     UserHandle.USER_ALL);
    668             resolver.registerContentObserver(mWindowAnimationScaleUri, false, this,
    669                     UserHandle.USER_ALL);
    670             resolver.registerContentObserver(mTransitionAnimationScaleUri, false, this,
    671                     UserHandle.USER_ALL);
    672             resolver.registerContentObserver(mAnimationDurationScaleUri, false, this,
    673                     UserHandle.USER_ALL);
    674         }
    675 
    676         @Override
    677         public void onChange(boolean selfChange, Uri uri) {
    678             if (uri == null) {
    679                 return;
    680             }
    681 
    682             if (mDisplayInversionEnabledUri.equals(uri)) {
    683                 updateCircularDisplayMaskIfNeeded();
    684             } else {
    685                 @UpdateAnimationScaleMode
    686                 final int mode;
    687                 if (mWindowAnimationScaleUri.equals(uri)) {
    688                     mode = WINDOW_ANIMATION_SCALE;
    689                 } else if (mTransitionAnimationScaleUri.equals(uri)) {
    690                     mode = TRANSITION_ANIMATION_SCALE;
    691                 } else if (mAnimationDurationScaleUri.equals(uri)) {
    692                     mode = ANIMATION_DURATION_SCALE;
    693                 } else {
    694                     // Ignoring unrecognized content changes
    695                     return;
    696                 }
    697                 Message m = mH.obtainMessage(H.UPDATE_ANIMATION_SCALE, mode, 0);
    698                 mH.sendMessage(m);
    699             }
    700         }
    701     }
    702 
    703     WallpaperController mWallpaperControllerLocked;
    704 
    705     final WindowLayersController mLayersController;
    706 
    707     boolean mAnimateWallpaperWithTarget;
    708 
    709     AppWindowToken mFocusedApp = null;
    710 
    711     PowerManager mPowerManager;
    712     PowerManagerInternal mPowerManagerInternal;
    713 
    714     float mWindowAnimationScaleSetting = 1.0f;
    715     float mTransitionAnimationScaleSetting = 1.0f;
    716     float mAnimatorDurationScaleSetting = 1.0f;
    717     boolean mAnimationsDisabled = false;
    718 
    719     final InputManagerService mInputManager;
    720     final DisplayManagerInternal mDisplayManagerInternal;
    721     final DisplayManager mDisplayManager;
    722     final Display[] mDisplays;
    723 
    724     // Who is holding the screen on.
    725     Session mHoldingScreenOn;
    726     PowerManager.WakeLock mHoldingScreenWakeLock;
    727 
    728     boolean mTurnOnScreen;
    729 
    730     // Whether or not a layout can cause a wake up when theater mode is enabled.
    731     boolean mAllowTheaterModeWakeFromLayout;
    732 
    733     TaskPositioner mTaskPositioner;
    734     DragState mDragState = null;
    735 
    736     // For frozen screen animations.
    737     int mExitAnimId, mEnterAnimId;
    738 
    739     boolean mAnimationScheduled;
    740 
    741     /** Skip repeated AppWindowTokens initialization. Note that AppWindowsToken's version of this
    742      * is a long initialized to Long.MIN_VALUE so that it doesn't match this value on startup. */
    743     int mTransactionSequence;
    744 
    745     final WindowAnimator mAnimator;
    746 
    747     private final BoundsAnimationController mBoundsAnimationController;
    748 
    749     SparseArray<Task> mTaskIdToTask = new SparseArray<>();
    750 
    751     /** All of the TaskStacks in the window manager, unordered. For an ordered list call
    752      * DisplayContent.getStacks(). */
    753     SparseArray<TaskStack> mStackIdToStack = new SparseArray<>();
    754 
    755     private final PointerEventDispatcher mPointerEventDispatcher;
    756 
    757     private WindowContentFrameStats mTempWindowRenderStats;
    758 
    759     final class DragInputEventReceiver extends InputEventReceiver {
    760         // Set, if stylus button was down at the start of the drag.
    761         private boolean mStylusButtonDownAtStart;
    762         // Indicates the first event to check for button state.
    763         private boolean mIsStartEvent = true;
    764 
    765         public DragInputEventReceiver(InputChannel inputChannel, Looper looper) {
    766             super(inputChannel, looper);
    767         }
    768 
    769         @Override
    770         public void onInputEvent(InputEvent event) {
    771             boolean handled = false;
    772             try {
    773                 if (event instanceof MotionEvent
    774                         && (event.getSource() & InputDevice.SOURCE_CLASS_POINTER) != 0
    775                         && mDragState != null) {
    776                     final MotionEvent motionEvent = (MotionEvent)event;
    777                     boolean endDrag = false;
    778                     final float newX = motionEvent.getRawX();
    779                     final float newY = motionEvent.getRawY();
    780                     final boolean isStylusButtonDown =
    781                             (motionEvent.getButtonState() & MotionEvent.BUTTON_STYLUS_PRIMARY) != 0;
    782 
    783                     if (mIsStartEvent) {
    784                         if (isStylusButtonDown) {
    785                             // First event and the button was down, check for the button being
    786                             // lifted in the future, if that happens we'll drop the item.
    787                             mStylusButtonDownAtStart = true;
    788                         }
    789                         mIsStartEvent = false;
    790                     }
    791 
    792                     switch (motionEvent.getAction()) {
    793                     case MotionEvent.ACTION_DOWN: {
    794                         if (DEBUG_DRAG) {
    795                             Slog.w(TAG_WM, "Unexpected ACTION_DOWN in drag layer");
    796                         }
    797                     } break;
    798 
    799                     case MotionEvent.ACTION_MOVE: {
    800                         if (mStylusButtonDownAtStart && !isStylusButtonDown) {
    801                             if (DEBUG_DRAG) Slog.d(TAG_WM, "Button no longer pressed; dropping at "
    802                                     + newX + "," + newY);
    803                             synchronized (mWindowMap) {
    804                                 endDrag = mDragState.notifyDropLw(newX, newY);
    805                             }
    806                         } else {
    807                             synchronized (mWindowMap) {
    808                                 // move the surface and tell the involved window(s) where we are
    809                                 mDragState.notifyMoveLw(newX, newY);
    810                             }
    811                         }
    812                     } break;
    813 
    814                     case MotionEvent.ACTION_UP: {
    815                         if (DEBUG_DRAG) Slog.d(TAG_WM, "Got UP on move channel; dropping at "
    816                                 + newX + "," + newY);
    817                         synchronized (mWindowMap) {
    818                             endDrag = mDragState.notifyDropLw(newX, newY);
    819                         }
    820                     } break;
    821 
    822                     case MotionEvent.ACTION_CANCEL: {
    823                         if (DEBUG_DRAG) Slog.d(TAG_WM, "Drag cancelled!");
    824                         endDrag = true;
    825                     } break;
    826                     }
    827 
    828                     if (endDrag) {
    829                         if (DEBUG_DRAG) Slog.d(TAG_WM, "Drag ended; tearing down state");
    830                         // tell all the windows that the drag has ended
    831                         synchronized (mWindowMap) {
    832                             mDragState.endDragLw();
    833                         }
    834                         mStylusButtonDownAtStart = false;
    835                         mIsStartEvent = true;
    836                     }
    837 
    838                     handled = true;
    839                 }
    840             } catch (Exception e) {
    841                 Slog.e(TAG_WM, "Exception caught by drag handleMotion", e);
    842             } finally {
    843                 finishInputEvent(event, handled);
    844             }
    845         }
    846     }
    847 
    848     /**
    849      * Whether the UI is currently running in touch mode (not showing
    850      * navigational focus because the user is directly pressing the screen).
    851      */
    852     boolean mInTouchMode;
    853 
    854     private ViewServer mViewServer;
    855     final ArrayList<WindowChangeListener> mWindowChangeListeners = new ArrayList<>();
    856     boolean mWindowsChanged = false;
    857 
    858     public interface WindowChangeListener {
    859         public void windowsChanged();
    860         public void focusChanged();
    861     }
    862 
    863     final Configuration mTempConfiguration = new Configuration();
    864 
    865     // The desired scaling factor for compatible apps.
    866     float mCompatibleScreenScale;
    867 
    868     // If true, only the core apps and services are being launched because the device
    869     // is in a special boot mode, such as being encrypted or waiting for a decryption password.
    870     // For example, when this flag is true, there will be no wallpaper service.
    871     final boolean mOnlyCore;
    872 
    873     // List of clients without a transtiton animation that we notify once we are done transitioning
    874     // since they won't be notified through the app window animator.
    875     final List<IBinder> mNoAnimationNotifyOnTransitionFinished = new ArrayList<>();
    876 
    877     // List of displays to reconfigure after configuration changes.
    878     // Some of the information reported for a display is dependent on resources to do the right
    879     // calculations. For example, {@link DisplayInfo#smallestNominalAppWidth} and company are
    880     // dependent on the height and width of the status and nav bar which change depending on the
    881     // current configuration.
    882     private final DisplayContentList mReconfigureOnConfigurationChanged = new DisplayContentList();
    883 
    884     /** Listener to notify activity manager about app transitions. */
    885     private final WindowManagerInternal.AppTransitionListener mActivityManagerAppTransitionNotifier
    886             = new WindowManagerInternal.AppTransitionListener() {
    887 
    888         @Override
    889         public void onAppTransitionCancelledLocked() {
    890             mH.sendEmptyMessage(H.NOTIFY_APP_TRANSITION_CANCELLED);
    891         }
    892 
    893         @Override
    894         public void onAppTransitionFinishedLocked(IBinder token) {
    895             mH.sendEmptyMessage(H.NOTIFY_APP_TRANSITION_FINISHED);
    896             AppWindowToken atoken = findAppWindowToken(token);
    897             if (atoken == null) {
    898                 return;
    899             }
    900             if (atoken.mLaunchTaskBehind) {
    901                 try {
    902                     mActivityManager.notifyLaunchTaskBehindComplete(atoken.token);
    903                 } catch (RemoteException e) {
    904                 }
    905                 atoken.mLaunchTaskBehind = false;
    906             } else {
    907                 atoken.updateReportedVisibilityLocked();
    908                 if (atoken.mEnteringAnimation) {
    909                     atoken.mEnteringAnimation = false;
    910                     try {
    911                         mActivityManager.notifyEnterAnimationComplete(atoken.token);
    912                     } catch (RemoteException e) {
    913                     }
    914                 }
    915             }
    916         }
    917     };
    918 
    919     public static WindowManagerService main(final Context context,
    920             final InputManagerService im,
    921             final boolean haveInputMethods, final boolean showBootMsgs,
    922             final boolean onlyCore) {
    923         final WindowManagerService[] holder = new WindowManagerService[1];
    924         DisplayThread.getHandler().runWithScissors(new Runnable() {
    925             @Override
    926             public void run() {
    927                 holder[0] = new WindowManagerService(context, im,
    928                         haveInputMethods, showBootMsgs, onlyCore);
    929             }
    930         }, 0);
    931         return holder[0];
    932     }
    933 
    934     private void initPolicy() {
    935         UiThread.getHandler().runWithScissors(new Runnable() {
    936             @Override
    937             public void run() {
    938                 WindowManagerPolicyThread.set(Thread.currentThread(), Looper.myLooper());
    939 
    940                 mPolicy.init(mContext, WindowManagerService.this, WindowManagerService.this);
    941             }
    942         }, 0);
    943     }
    944 
    945     private WindowManagerService(Context context, InputManagerService inputManager,
    946             boolean haveInputMethods, boolean showBootMsgs, boolean onlyCore) {
    947         mContext = context;
    948         mHaveInputMethods = haveInputMethods;
    949         mAllowBootMessages = showBootMsgs;
    950         mOnlyCore = onlyCore;
    951         mLimitedAlphaCompositing = context.getResources().getBoolean(
    952                 com.android.internal.R.bool.config_sf_limitedAlpha);
    953         mHasPermanentDpad = context.getResources().getBoolean(
    954                 com.android.internal.R.bool.config_hasPermanentDpad);
    955         mInTouchMode = context.getResources().getBoolean(
    956                 com.android.internal.R.bool.config_defaultInTouchMode);
    957         mDrawLockTimeoutMillis = context.getResources().getInteger(
    958                 com.android.internal.R.integer.config_drawLockTimeoutMillis);
    959         mAllowAnimationsInLowPowerMode = context.getResources().getBoolean(
    960                 com.android.internal.R.bool.config_allowAnimationsInLowPowerMode);
    961         mInputManager = inputManager; // Must be before createDisplayContentLocked.
    962         mDisplayManagerInternal = LocalServices.getService(DisplayManagerInternal.class);
    963         mDisplaySettings = new DisplaySettings();
    964         mDisplaySettings.readSettingsLocked();
    965 
    966         mWallpaperControllerLocked = new WallpaperController(this);
    967         mWindowPlacerLocked = new WindowSurfacePlacer(this);
    968         mLayersController = new WindowLayersController(this);
    969 
    970         LocalServices.addService(WindowManagerPolicy.class, mPolicy);
    971 
    972         mPointerEventDispatcher = new PointerEventDispatcher(mInputManager.monitorInput(TAG_WM));
    973 
    974         mFxSession = new SurfaceSession();
    975         mDisplayManager = (DisplayManager)context.getSystemService(Context.DISPLAY_SERVICE);
    976         mDisplays = mDisplayManager.getDisplays();
    977         for (Display display : mDisplays) {
    978             createDisplayContentLocked(display);
    979         }
    980 
    981         mKeyguardDisableHandler = new KeyguardDisableHandler(mContext, mPolicy);
    982 
    983         mPowerManager = (PowerManager)context.getSystemService(Context.POWER_SERVICE);
    984         mPowerManagerInternal = LocalServices.getService(PowerManagerInternal.class);
    985         mPowerManagerInternal.registerLowPowerModeObserver(
    986                 new PowerManagerInternal.LowPowerModeListener() {
    987             @Override
    988             public void onLowPowerModeChanged(boolean enabled) {
    989                 synchronized (mWindowMap) {
    990                     if (mAnimationsDisabled != enabled && !mAllowAnimationsInLowPowerMode) {
    991                         mAnimationsDisabled = enabled;
    992                         dispatchNewAnimatorScaleLocked(null);
    993                     }
    994                 }
    995             }
    996         });
    997         mAnimationsDisabled = mPowerManagerInternal.getLowPowerModeEnabled();
    998         mScreenFrozenLock = mPowerManager.newWakeLock(
    999                 PowerManager.PARTIAL_WAKE_LOCK, "SCREEN_FROZEN");
   1000         mScreenFrozenLock.setReferenceCounted(false);
   1001 
   1002         mAppTransition = new AppTransition(context, this);
   1003         mAppTransition.registerListenerLocked(mActivityManagerAppTransitionNotifier);
   1004 
   1005         mBoundsAnimationController =
   1006                 new BoundsAnimationController(mAppTransition, UiThread.getHandler());
   1007 
   1008         mActivityManager = ActivityManagerNative.getDefault();
   1009         mAmInternal = LocalServices.getService(ActivityManagerInternal.class);
   1010         mAppOps = (AppOpsManager)context.getSystemService(Context.APP_OPS_SERVICE);
   1011         AppOpsManager.OnOpChangedInternalListener opListener =
   1012                 new AppOpsManager.OnOpChangedInternalListener() {
   1013                     @Override public void onOpChanged(int op, String packageName) {
   1014                         updateAppOpsState();
   1015                     }
   1016                 };
   1017         mAppOps.startWatchingMode(AppOpsManager.OP_SYSTEM_ALERT_WINDOW, null, opListener);
   1018         mAppOps.startWatchingMode(AppOpsManager.OP_TOAST_WINDOW, null, opListener);
   1019 
   1020         // Get persisted window scale setting
   1021         mWindowAnimationScaleSetting = Settings.Global.getFloat(context.getContentResolver(),
   1022                 Settings.Global.WINDOW_ANIMATION_SCALE, mWindowAnimationScaleSetting);
   1023         mTransitionAnimationScaleSetting = Settings.Global.getFloat(context.getContentResolver(),
   1024                 Settings.Global.TRANSITION_ANIMATION_SCALE, mTransitionAnimationScaleSetting);
   1025         setAnimatorDurationScale(Settings.Global.getFloat(context.getContentResolver(),
   1026                 Settings.Global.ANIMATOR_DURATION_SCALE, mAnimatorDurationScaleSetting));
   1027 
   1028         // Track changes to DevicePolicyManager state so we can enable/disable keyguard.
   1029         IntentFilter filter = new IntentFilter();
   1030         filter.addAction(DevicePolicyManager.ACTION_DEVICE_POLICY_MANAGER_STATE_CHANGED);
   1031         mContext.registerReceiver(mBroadcastReceiver, filter);
   1032 
   1033         mSettingsObserver = new SettingsObserver();
   1034 
   1035         mHoldingScreenWakeLock = mPowerManager.newWakeLock(
   1036                 PowerManager.SCREEN_BRIGHT_WAKE_LOCK | PowerManager.ON_AFTER_RELEASE, TAG_WM);
   1037         mHoldingScreenWakeLock.setReferenceCounted(false);
   1038 
   1039         mAnimator = new WindowAnimator(this);
   1040 
   1041         mAllowTheaterModeWakeFromLayout = context.getResources().getBoolean(
   1042                 com.android.internal.R.bool.config_allowTheaterModeWakeFromWindowLayout);
   1043 
   1044 
   1045         LocalServices.addService(WindowManagerInternal.class, new LocalService());
   1046         initPolicy();
   1047 
   1048         // Add ourself to the Watchdog monitors.
   1049         Watchdog.getInstance().addMonitor(this);
   1050 
   1051         SurfaceControl.openTransaction();
   1052         try {
   1053             createWatermarkInTransaction();
   1054         } finally {
   1055             SurfaceControl.closeTransaction();
   1056         }
   1057 
   1058         showEmulatorDisplayOverlayIfNeeded();
   1059     }
   1060 
   1061     public InputMonitor getInputMonitor() {
   1062         return mInputMonitor;
   1063     }
   1064 
   1065     @Override
   1066     public boolean onTransact(int code, Parcel data, Parcel reply, int flags)
   1067             throws RemoteException {
   1068         try {
   1069             return super.onTransact(code, data, reply, flags);
   1070         } catch (RuntimeException e) {
   1071             // The window manager only throws security exceptions, so let's
   1072             // log all others.
   1073             if (!(e instanceof SecurityException)) {
   1074                 Slog.wtf(TAG_WM, "Window Manager Crash", e);
   1075             }
   1076             throw e;
   1077         }
   1078     }
   1079 
   1080     private void placeWindowAfter(WindowState pos, WindowState window) {
   1081         final WindowList windows = pos.getWindowList();
   1082         final int i = windows.indexOf(pos);
   1083         if (DEBUG_FOCUS || DEBUG_WINDOW_MOVEMENT || DEBUG_ADD_REMOVE) Slog.v(
   1084             TAG_WM, "Adding window " + window + " at "
   1085             + (i+1) + " of " + windows.size() + " (after " + pos + ")");
   1086         windows.add(i+1, window);
   1087         mWindowsChanged = true;
   1088     }
   1089 
   1090     private void placeWindowBefore(WindowState pos, WindowState window) {
   1091         final WindowList windows = pos.getWindowList();
   1092         int i = windows.indexOf(pos);
   1093         if (DEBUG_FOCUS || DEBUG_WINDOW_MOVEMENT || DEBUG_ADD_REMOVE) Slog.v(
   1094             TAG_WM, "Adding window " + window + " at "
   1095             + i + " of " + windows.size() + " (before " + pos + ")");
   1096         if (i < 0) {
   1097             Slog.w(TAG_WM, "placeWindowBefore: Unable to find " + pos + " in " + windows);
   1098             i = 0;
   1099         }
   1100         windows.add(i, window);
   1101         mWindowsChanged = true;
   1102     }
   1103 
   1104     //This method finds out the index of a window that has the same app token as
   1105     //win. used for z ordering the windows in mWindows
   1106     private int findIdxBasedOnAppTokens(WindowState win) {
   1107         WindowList windows = win.getWindowList();
   1108         for(int j = windows.size() - 1; j >= 0; j--) {
   1109             WindowState wentry = windows.get(j);
   1110             if(wentry.mAppToken == win.mAppToken) {
   1111                 return j;
   1112             }
   1113         }
   1114         return -1;
   1115     }
   1116 
   1117     /**
   1118      * Return the list of Windows from the passed token on the given Display.
   1119      * @param token The token with all the windows.
   1120      * @param displayContent The display we are interested in.
   1121      * @return List of windows from token that are on displayContent.
   1122      */
   1123     private WindowList getTokenWindowsOnDisplay(WindowToken token, DisplayContent displayContent) {
   1124         final WindowList windowList = new WindowList();
   1125         final int count = token.windows.size();
   1126         for (int i = 0; i < count; i++) {
   1127             final WindowState win = token.windows.get(i);
   1128             if (win.getDisplayContent() == displayContent) {
   1129                 windowList.add(win);
   1130             }
   1131         }
   1132         return windowList;
   1133     }
   1134 
   1135     /**
   1136      * Recursive search through a WindowList and all of its windows' children.
   1137      * @param targetWin The window to search for.
   1138      * @param windows The list to search.
   1139      * @return The index of win in windows or of the window that is an ancestor of win.
   1140      */
   1141     private int indexOfWinInWindowList(WindowState targetWin, WindowList windows) {
   1142         for (int i = windows.size() - 1; i >= 0; i--) {
   1143             final WindowState w = windows.get(i);
   1144             if (w == targetWin) {
   1145                 return i;
   1146             }
   1147             if (!w.mChildWindows.isEmpty()) {
   1148                 if (indexOfWinInWindowList(targetWin, w.mChildWindows) >= 0) {
   1149                     return i;
   1150                 }
   1151             }
   1152         }
   1153         return -1;
   1154     }
   1155 
   1156     private int addAppWindowToListLocked(final WindowState win) {
   1157         final DisplayContent displayContent = win.getDisplayContent();
   1158         if (displayContent == null) {
   1159             // It doesn't matter this display is going away.
   1160             return 0;
   1161         }
   1162         final IWindow client = win.mClient;
   1163         final WindowToken token = win.mToken;
   1164 
   1165         final WindowList windows = displayContent.getWindowList();
   1166         WindowList tokenWindowList = getTokenWindowsOnDisplay(token, displayContent);
   1167         int tokenWindowsPos = 0;
   1168         if (!tokenWindowList.isEmpty()) {
   1169             return addAppWindowToTokenListLocked(win, token, windows, tokenWindowList);
   1170         }
   1171 
   1172         // No windows from this token on this display
   1173         if (localLOGV) Slog.v(TAG_WM, "Figuring out where to add app window " + client.asBinder()
   1174                 + " (token=" + token + ")");
   1175         // Figure out where the window should go, based on the
   1176         // order of applications.
   1177         WindowState pos = null;
   1178 
   1179         final ArrayList<Task> tasks = displayContent.getTasks();
   1180         int taskNdx;
   1181         int tokenNdx = -1;
   1182         for (taskNdx = tasks.size() - 1; taskNdx >= 0; --taskNdx) {
   1183             AppTokenList tokens = tasks.get(taskNdx).mAppTokens;
   1184             for (tokenNdx = tokens.size() - 1; tokenNdx >= 0; --tokenNdx) {
   1185                 final AppWindowToken t = tokens.get(tokenNdx);
   1186                 if (t == token) {
   1187                     --tokenNdx;
   1188                     if (tokenNdx < 0) {
   1189                         --taskNdx;
   1190                         if (taskNdx >= 0) {
   1191                             tokenNdx = tasks.get(taskNdx).mAppTokens.size() - 1;
   1192                         }
   1193                     }
   1194                     break;
   1195                 }
   1196 
   1197                 // We haven't reached the token yet; if this token
   1198                 // is not going to the bottom and has windows on this display, we can
   1199                 // use it as an anchor for when we do reach the token.
   1200                 tokenWindowList = getTokenWindowsOnDisplay(t, displayContent);
   1201                 if (!t.sendingToBottom && tokenWindowList.size() > 0) {
   1202                     pos = tokenWindowList.get(0);
   1203                 }
   1204             }
   1205             if (tokenNdx >= 0) {
   1206                 // early exit
   1207                 break;
   1208             }
   1209         }
   1210 
   1211         // We now know the index into the apps.  If we found
   1212         // an app window above, that gives us the position; else
   1213         // we need to look some more.
   1214         if (pos != null) {
   1215             // Move behind any windows attached to this one.
   1216             WindowToken atoken = mTokenMap.get(pos.mClient.asBinder());
   1217             if (atoken != null) {
   1218                 tokenWindowList =
   1219                         getTokenWindowsOnDisplay(atoken, displayContent);
   1220                 final int NC = tokenWindowList.size();
   1221                 if (NC > 0) {
   1222                     WindowState bottom = tokenWindowList.get(0);
   1223                     if (bottom.mSubLayer < 0) {
   1224                         pos = bottom;
   1225                     }
   1226                 }
   1227             }
   1228             placeWindowBefore(pos, win);
   1229             return tokenWindowsPos;
   1230         }
   1231 
   1232         // Continue looking down until we find the first
   1233         // token that has windows on this display.
   1234         for ( ; taskNdx >= 0; --taskNdx) {
   1235             AppTokenList tokens = tasks.get(taskNdx).mAppTokens;
   1236             for ( ; tokenNdx >= 0; --tokenNdx) {
   1237                 final AppWindowToken t = tokens.get(tokenNdx);
   1238                 tokenWindowList = getTokenWindowsOnDisplay(t, displayContent);
   1239                 final int NW = tokenWindowList.size();
   1240                 if (NW > 0) {
   1241                     pos = tokenWindowList.get(NW-1);
   1242                     break;
   1243                 }
   1244             }
   1245             if (tokenNdx >= 0) {
   1246                 // found
   1247                 break;
   1248             }
   1249         }
   1250 
   1251         if (pos != null) {
   1252             // Move in front of any windows attached to this
   1253             // one.
   1254             WindowToken atoken = mTokenMap.get(pos.mClient.asBinder());
   1255             if (atoken != null) {
   1256                 final int NC = atoken.windows.size();
   1257                 if (NC > 0) {
   1258                     WindowState top = atoken.windows.get(NC-1);
   1259                     if (top.mSubLayer >= 0) {
   1260                         pos = top;
   1261                     }
   1262                 }
   1263             }
   1264             placeWindowAfter(pos, win);
   1265             return tokenWindowsPos;
   1266         }
   1267 
   1268         // Just search for the start of this layer.
   1269         final int myLayer = win.mBaseLayer;
   1270         int i;
   1271         for (i = windows.size() - 1; i >= 0; --i) {
   1272             WindowState w = windows.get(i);
   1273             // Dock divider shares the base layer with application windows, but we want to always
   1274             // keep it above the application windows. The sharing of the base layer is intended
   1275             // for window animations, which need to be above the dock divider for the duration
   1276             // of the animation.
   1277             if (w.mBaseLayer <= myLayer && w.mAttrs.type != TYPE_DOCK_DIVIDER) {
   1278                 break;
   1279             }
   1280         }
   1281         if (DEBUG_FOCUS || DEBUG_WINDOW_MOVEMENT || DEBUG_ADD_REMOVE) Slog.v(TAG_WM,
   1282                 "Based on layer: Adding window " + win + " at " + (i + 1) + " of "
   1283                         + windows.size());
   1284         windows.add(i + 1, win);
   1285         mWindowsChanged = true;
   1286         return tokenWindowsPos;
   1287     }
   1288 
   1289     private int addAppWindowToTokenListLocked(WindowState win, WindowToken token,
   1290             WindowList windows, WindowList tokenWindowList) {
   1291         int tokenWindowsPos;
   1292         // If this application has existing windows, we
   1293         // simply place the new window on top of them... but
   1294         // keep the starting window on top.
   1295         if (win.mAttrs.type == TYPE_BASE_APPLICATION) {
   1296             // Base windows go behind everything else.
   1297             WindowState lowestWindow = tokenWindowList.get(0);
   1298             placeWindowBefore(lowestWindow, win);
   1299             tokenWindowsPos = indexOfWinInWindowList(lowestWindow, token.windows);
   1300         } else {
   1301             AppWindowToken atoken = win.mAppToken;
   1302             final int windowListPos = tokenWindowList.size();
   1303             WindowState lastWindow = tokenWindowList.get(windowListPos - 1);
   1304             if (atoken != null && lastWindow == atoken.startingWindow) {
   1305                 placeWindowBefore(lastWindow, win);
   1306                 tokenWindowsPos = indexOfWinInWindowList(lastWindow, token.windows);
   1307             } else {
   1308                 int newIdx = findIdxBasedOnAppTokens(win);
   1309                 //there is a window above this one associated with the same
   1310                 //apptoken note that the window could be a floating window
   1311                 //that was created later or a window at the top of the list of
   1312                 //windows associated with this token.
   1313                 if (DEBUG_FOCUS || DEBUG_WINDOW_MOVEMENT || DEBUG_ADD_REMOVE) Slog.v(TAG_WM,
   1314                         "not Base app: Adding window " + win + " at " + (newIdx + 1) + " of "
   1315                                 + windows.size());
   1316                 windows.add(newIdx + 1, win);
   1317                 if (newIdx < 0) {
   1318                     // No window from token found on win's display.
   1319                     tokenWindowsPos = 0;
   1320                 } else {
   1321                     tokenWindowsPos = indexOfWinInWindowList(
   1322                             windows.get(newIdx), token.windows) + 1;
   1323                 }
   1324                 mWindowsChanged = true;
   1325             }
   1326         }
   1327         return tokenWindowsPos;
   1328     }
   1329 
   1330     private void addFreeWindowToListLocked(final WindowState win) {
   1331         final WindowList windows = win.getWindowList();
   1332 
   1333         // Figure out where window should go, based on layer.
   1334         final int myLayer = win.mBaseLayer;
   1335         int i;
   1336         for (i = windows.size() - 1; i >= 0; i--) {
   1337             final WindowState otherWin = windows.get(i);
   1338             if (otherWin.getBaseType() != TYPE_WALLPAPER && otherWin.mBaseLayer <= myLayer) {
   1339                 // Wallpaper wanders through the window list, for example to position itself
   1340                 // directly behind keyguard. Because of this it will break the ordering based on
   1341                 // WindowState.mBaseLayer. There might windows with higher mBaseLayer behind it and
   1342                 // we don't want the new window to appear above them. An example of this is adding
   1343                 // of the docked stack divider. Consider a scenario with the following ordering (top
   1344                 // to bottom): keyguard, wallpaper, assist preview, apps. We want the dock divider
   1345                 // to land below the assist preview, so the dock divider must ignore the wallpaper,
   1346                 // with which it shares the base layer.
   1347                 break;
   1348             }
   1349         }
   1350         i++;
   1351         if (DEBUG_FOCUS || DEBUG_WINDOW_MOVEMENT || DEBUG_ADD_REMOVE) Slog.v(TAG_WM,
   1352                 "Free window: Adding window " + win + " at " + i + " of " + windows.size());
   1353         windows.add(i, win);
   1354         mWindowsChanged = true;
   1355     }
   1356 
   1357     private void addAttachedWindowToListLocked(final WindowState win, boolean addToToken) {
   1358         final WindowToken token = win.mToken;
   1359         final DisplayContent displayContent = win.getDisplayContent();
   1360         if (displayContent == null) {
   1361             return;
   1362         }
   1363         final WindowState attached = win.mAttachedWindow;
   1364 
   1365         WindowList tokenWindowList = getTokenWindowsOnDisplay(token, displayContent);
   1366 
   1367         // Figure out this window's ordering relative to the window
   1368         // it is attached to.
   1369         final int NA = tokenWindowList.size();
   1370         final int sublayer = win.mSubLayer;
   1371         int largestSublayer = Integer.MIN_VALUE;
   1372         WindowState windowWithLargestSublayer = null;
   1373         int i;
   1374         for (i = 0; i < NA; i++) {
   1375             WindowState w = tokenWindowList.get(i);
   1376             final int wSublayer = w.mSubLayer;
   1377             if (wSublayer >= largestSublayer) {
   1378                 largestSublayer = wSublayer;
   1379                 windowWithLargestSublayer = w;
   1380             }
   1381             if (sublayer < 0) {
   1382                 // For negative sublayers, we go below all windows
   1383                 // in the same sublayer.
   1384                 if (wSublayer >= sublayer) {
   1385                     if (addToToken) {
   1386                         if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM, "Adding " + win + " to " + token);
   1387                         token.windows.add(i, win);
   1388                     }
   1389                     placeWindowBefore(wSublayer >= 0 ? attached : w, win);
   1390                     break;
   1391                 }
   1392             } else {
   1393                 // For positive sublayers, we go above all windows
   1394                 // in the same sublayer.
   1395                 if (wSublayer > sublayer) {
   1396                     if (addToToken) {
   1397                         if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM, "Adding " + win + " to " + token);
   1398                         token.windows.add(i, win);
   1399                     }
   1400                     placeWindowBefore(w, win);
   1401                     break;
   1402                 }
   1403             }
   1404         }
   1405         if (i >= NA) {
   1406             if (addToToken) {
   1407                 if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM, "Adding " + win + " to " + token);
   1408                 token.windows.add(win);
   1409             }
   1410             if (sublayer < 0) {
   1411                 placeWindowBefore(attached, win);
   1412             } else {
   1413                 placeWindowAfter(largestSublayer >= 0
   1414                                  ? windowWithLargestSublayer
   1415                                  : attached,
   1416                                  win);
   1417             }
   1418         }
   1419     }
   1420 
   1421     private void addWindowToListInOrderLocked(final WindowState win, boolean addToToken) {
   1422         if (DEBUG_FOCUS) Slog.d(TAG_WM, "addWindowToListInOrderLocked: win=" + win +
   1423                 " Callers=" + Debug.getCallers(4));
   1424         if (win.mAttachedWindow == null) {
   1425             final WindowToken token = win.mToken;
   1426             int tokenWindowsPos = 0;
   1427             if (token.appWindowToken != null) {
   1428                 tokenWindowsPos = addAppWindowToListLocked(win);
   1429             } else {
   1430                 addFreeWindowToListLocked(win);
   1431             }
   1432             if (addToToken) {
   1433                 if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM, "Adding " + win + " to " + token);
   1434                 token.windows.add(tokenWindowsPos, win);
   1435             }
   1436         } else {
   1437             addAttachedWindowToListLocked(win, addToToken);
   1438         }
   1439 
   1440         final AppWindowToken appToken = win.mAppToken;
   1441         if (appToken != null) {
   1442             if (addToToken) {
   1443                 appToken.addWindow(win);
   1444             }
   1445         }
   1446     }
   1447 
   1448     static boolean canBeImeTarget(WindowState w) {
   1449         final int fl = w.mAttrs.flags
   1450                 & (FLAG_NOT_FOCUSABLE|FLAG_ALT_FOCUSABLE_IM);
   1451         final int type = w.mAttrs.type;
   1452         if (fl == 0 || fl == (FLAG_NOT_FOCUSABLE|FLAG_ALT_FOCUSABLE_IM)
   1453                 || type == TYPE_APPLICATION_STARTING) {
   1454             if (DEBUG_INPUT_METHOD) {
   1455                 Slog.i(TAG_WM, "isVisibleOrAdding " + w + ": " + w.isVisibleOrAdding());
   1456                 if (!w.isVisibleOrAdding()) {
   1457                     Slog.i(TAG_WM, "  mSurfaceController=" + w.mWinAnimator.mSurfaceController
   1458                             + " relayoutCalled=" + w.mRelayoutCalled
   1459                             + " viewVis=" + w.mViewVisibility
   1460                             + " policyVis=" + w.mPolicyVisibility
   1461                             + " policyVisAfterAnim=" + w.mPolicyVisibilityAfterAnim
   1462                             + " attachHid=" + w.mAttachedHidden
   1463                             + " exiting=" + w.mAnimatingExit + " destroying=" + w.mDestroying);
   1464                     if (w.mAppToken != null) {
   1465                         Slog.i(TAG_WM, "  mAppToken.hiddenRequested=" + w.mAppToken.hiddenRequested);
   1466                     }
   1467                 }
   1468             }
   1469             return w.isVisibleOrAdding();
   1470         }
   1471         return false;
   1472     }
   1473 
   1474     /**
   1475      * Dig through the WindowStates and find the one that the Input Method will target.
   1476      * @param willMove
   1477      * @return The index+1 in mWindows of the discovered target.
   1478      */
   1479     int findDesiredInputMethodWindowIndexLocked(boolean willMove) {
   1480         // TODO(multidisplay): Needs some serious rethought when the target and IME are not on the
   1481         // same display. Or even when the current IME/target are not on the same screen as the next
   1482         // IME/target. For now only look for input windows on the main screen.
   1483         WindowList windows = getDefaultWindowListLocked();
   1484         WindowState w = null;
   1485         int i;
   1486         for (i = windows.size() - 1; i >= 0; --i) {
   1487             WindowState win = windows.get(i);
   1488 
   1489             if (DEBUG_INPUT_METHOD && willMove) Slog.i(TAG_WM, "Checking window @" + i
   1490                     + " " + win + " fl=0x" + Integer.toHexString(win.mAttrs.flags));
   1491             if (canBeImeTarget(win)) {
   1492                 w = win;
   1493                 //Slog.i(TAG_WM, "Putting input method here!");
   1494 
   1495                 // Yet more tricksyness!  If this window is a "starting"
   1496                 // window, we do actually want to be on top of it, but
   1497                 // it is not -really- where input will go.  So if the caller
   1498                 // is not actually looking to move the IME, look down below
   1499                 // for a real window to target...
   1500                 if (!willMove
   1501                         && w.mAttrs.type == TYPE_APPLICATION_STARTING
   1502                         && i > 0) {
   1503                     WindowState wb = windows.get(i-1);
   1504                     if (wb.mAppToken == w.mAppToken && canBeImeTarget(wb)) {
   1505                         i--;
   1506                         w = wb;
   1507                     }
   1508                 }
   1509                 break;
   1510             }
   1511         }
   1512 
   1513         // Now w is either mWindows[0] or an IME (or null if mWindows is empty).
   1514 
   1515         if (DEBUG_INPUT_METHOD && willMove) Slog.v(TAG_WM, "Proposed new IME target: " + w);
   1516 
   1517         // Now, a special case -- if the last target's window is in the
   1518         // process of exiting, and is above the new target, keep on the
   1519         // last target to avoid flicker.  Consider for example a Dialog with
   1520         // the IME shown: when the Dialog is dismissed, we want to keep
   1521         // the IME above it until it is completely gone so it doesn't drop
   1522         // behind the dialog or its full-screen scrim.
   1523         final WindowState curTarget = mInputMethodTarget;
   1524         if (curTarget != null
   1525                 && curTarget.isDisplayedLw()
   1526                 && curTarget.isClosing()
   1527                 && (w == null || curTarget.mWinAnimator.mAnimLayer > w.mWinAnimator.mAnimLayer)) {
   1528             if (DEBUG_INPUT_METHOD) Slog.v(TAG_WM, "Current target higher, not changing");
   1529             return windows.indexOf(curTarget) + 1;
   1530         }
   1531 
   1532         if (DEBUG_INPUT_METHOD) Slog.v(TAG_WM, "Desired input method target="
   1533                 + w + " willMove=" + willMove);
   1534 
   1535         if (willMove && w != null) {
   1536             AppWindowToken token = curTarget == null ? null : curTarget.mAppToken;
   1537             if (token != null) {
   1538 
   1539                 // Now some fun for dealing with window animations that
   1540                 // modify the Z order.  We need to look at all windows below
   1541                 // the current target that are in this app, finding the highest
   1542                 // visible one in layering.
   1543                 WindowState highestTarget = null;
   1544                 int highestPos = 0;
   1545                 if (token.mAppAnimator.animating || token.mAppAnimator.animation != null) {
   1546                     WindowList curWindows = curTarget.getWindowList();
   1547                     int pos = curWindows.indexOf(curTarget);
   1548                     while (pos >= 0) {
   1549                         WindowState win = curWindows.get(pos);
   1550                         if (win.mAppToken != token) {
   1551                             break;
   1552                         }
   1553                         if (!win.mRemoved) {
   1554                             if (highestTarget == null || win.mWinAnimator.mAnimLayer >
   1555                                     highestTarget.mWinAnimator.mAnimLayer) {
   1556                                 highestTarget = win;
   1557                                 highestPos = pos;
   1558                             }
   1559                         }
   1560                         pos--;
   1561                     }
   1562                 }
   1563 
   1564                 if (highestTarget != null) {
   1565                     if (DEBUG_INPUT_METHOD) Slog.v(TAG_WM, mAppTransition + " " + highestTarget
   1566                             + " animating=" + highestTarget.mWinAnimator.isAnimationSet()
   1567                             + " layer=" + highestTarget.mWinAnimator.mAnimLayer
   1568                             + " new layer=" + w.mWinAnimator.mAnimLayer);
   1569 
   1570                     if (mAppTransition.isTransitionSet()) {
   1571                         // If we are currently setting up for an animation,
   1572                         // hold everything until we can find out what will happen.
   1573                         mInputMethodTargetWaitingAnim = true;
   1574                         mInputMethodTarget = highestTarget;
   1575                         return highestPos + 1;
   1576                     } else if (highestTarget.mWinAnimator.isAnimationSet() &&
   1577                             highestTarget.mWinAnimator.mAnimLayer > w.mWinAnimator.mAnimLayer) {
   1578                         // If the window we are currently targeting is involved
   1579                         // with an animation, and it is on top of the next target
   1580                         // we will be over, then hold off on moving until
   1581                         // that is done.
   1582                         mInputMethodTargetWaitingAnim = true;
   1583                         mInputMethodTarget = highestTarget;
   1584                         return highestPos + 1;
   1585                     }
   1586                 }
   1587             }
   1588         }
   1589 
   1590         //Slog.i(TAG_WM, "Placing input method @" + (i+1));
   1591         if (w != null) {
   1592             if (willMove) {
   1593                 if (DEBUG_INPUT_METHOD) Slog.w(TAG_WM, "Moving IM target from " + curTarget + " to "
   1594                         + w + (SHOW_STACK_CRAWLS ? " Callers=" + Debug.getCallers(4) : ""));
   1595                 mInputMethodTarget = w;
   1596                 mInputMethodTargetWaitingAnim = false;
   1597                 if (w.mAppToken != null) {
   1598                     mLayersController.setInputMethodAnimLayerAdjustment(
   1599                             w.mAppToken.mAppAnimator.animLayerAdjustment);
   1600                 } else {
   1601                     mLayersController.setInputMethodAnimLayerAdjustment(0);
   1602                 }
   1603             }
   1604 
   1605             // If the docked divider is visible, we still need to go through this whole
   1606             // excercise to find the appropriate input method target (used for animations
   1607             // and dialog adjustments), but for purposes of Z ordering we simply wish to
   1608             // place it above the docked divider. Unless it is already above the divider.
   1609             WindowState dockedDivider = w.mDisplayContent.mDividerControllerLocked.getWindow();
   1610             if (dockedDivider != null && dockedDivider.isVisibleLw()) {
   1611                 int dividerIndex = windows.indexOf(dockedDivider);
   1612                 if (dividerIndex > 0 && dividerIndex > i) {
   1613                     return dividerIndex + 1;
   1614                 }
   1615             }
   1616             return i+1;
   1617         }
   1618         if (willMove) {
   1619             if (DEBUG_INPUT_METHOD) Slog.w(TAG_WM, "Moving IM target from " + curTarget + " to null."
   1620                     + (SHOW_STACK_CRAWLS ? " Callers=" + Debug.getCallers(4) : ""));
   1621             mInputMethodTarget = null;
   1622             mLayersController.setInputMethodAnimLayerAdjustment(0);
   1623         }
   1624         return -1;
   1625     }
   1626 
   1627     void addInputMethodWindowToListLocked(WindowState win) {
   1628         int pos = findDesiredInputMethodWindowIndexLocked(true);
   1629         if (pos >= 0) {
   1630             win.mTargetAppToken = mInputMethodTarget.mAppToken;
   1631             if (DEBUG_WINDOW_MOVEMENT || DEBUG_ADD_REMOVE) Slog.v(
   1632                     TAG_WM, "Adding input method window " + win + " at " + pos);
   1633             // TODO(multidisplay): IMEs are only supported on the default display.
   1634             getDefaultWindowListLocked().add(pos, win);
   1635             mWindowsChanged = true;
   1636             moveInputMethodDialogsLocked(pos + 1);
   1637             return;
   1638         }
   1639         win.mTargetAppToken = null;
   1640         addWindowToListInOrderLocked(win, true);
   1641         moveInputMethodDialogsLocked(pos);
   1642     }
   1643 
   1644     private int tmpRemoveWindowLocked(int interestingPos, WindowState win) {
   1645         WindowList windows = win.getWindowList();
   1646         int wpos = windows.indexOf(win);
   1647         if (wpos >= 0) {
   1648             if (wpos < interestingPos) interestingPos--;
   1649             if (DEBUG_WINDOW_MOVEMENT) Slog.v(TAG_WM, "Temp removing at " + wpos + ": " + win);
   1650             windows.remove(wpos);
   1651             mWindowsChanged = true;
   1652             int NC = win.mChildWindows.size();
   1653             while (NC > 0) {
   1654                 NC--;
   1655                 WindowState cw = win.mChildWindows.get(NC);
   1656                 int cpos = windows.indexOf(cw);
   1657                 if (cpos >= 0) {
   1658                     if (cpos < interestingPos) interestingPos--;
   1659                     if (DEBUG_WINDOW_MOVEMENT) Slog.v(TAG_WM, "Temp removing child at "
   1660                             + cpos + ": " + cw);
   1661                     windows.remove(cpos);
   1662                 }
   1663             }
   1664         }
   1665         return interestingPos;
   1666     }
   1667 
   1668     private void reAddWindowToListInOrderLocked(WindowState win) {
   1669         addWindowToListInOrderLocked(win, false);
   1670         // This is a hack to get all of the child windows added as well
   1671         // at the right position.  Child windows should be rare and
   1672         // this case should be rare, so it shouldn't be that big a deal.
   1673         WindowList windows = win.getWindowList();
   1674         int wpos = windows.indexOf(win);
   1675         if (wpos >= 0) {
   1676             if (DEBUG_WINDOW_MOVEMENT) Slog.v(TAG_WM, "ReAdd removing from " + wpos + ": " + win);
   1677             windows.remove(wpos);
   1678             mWindowsChanged = true;
   1679             reAddWindowLocked(wpos, win);
   1680         }
   1681     }
   1682 
   1683     void logWindowList(final WindowList windows, String prefix) {
   1684         int N = windows.size();
   1685         while (N > 0) {
   1686             N--;
   1687             Slog.v(TAG_WM, prefix + "#" + N + ": " + windows.get(N));
   1688         }
   1689     }
   1690 
   1691     void moveInputMethodDialogsLocked(int pos) {
   1692         ArrayList<WindowState> dialogs = mInputMethodDialogs;
   1693 
   1694         // TODO(multidisplay): IMEs are only supported on the default display.
   1695         WindowList windows = getDefaultWindowListLocked();
   1696         final int N = dialogs.size();
   1697         if (DEBUG_INPUT_METHOD) Slog.v(TAG_WM, "Removing " + N + " dialogs w/pos=" + pos);
   1698         for (int i=0; i<N; i++) {
   1699             pos = tmpRemoveWindowLocked(pos, dialogs.get(i));
   1700         }
   1701         if (DEBUG_INPUT_METHOD) {
   1702             Slog.v(TAG_WM, "Window list w/pos=" + pos);
   1703             logWindowList(windows, "  ");
   1704         }
   1705 
   1706         if (pos >= 0) {
   1707             final AppWindowToken targetAppToken = mInputMethodTarget.mAppToken;
   1708             // Skip windows owned by the input method.
   1709             if (mInputMethodWindow != null) {
   1710                 while (pos < windows.size()) {
   1711                     WindowState wp = windows.get(pos);
   1712                     if (wp == mInputMethodWindow || wp.mAttachedWindow == mInputMethodWindow) {
   1713                         pos++;
   1714                         continue;
   1715                     }
   1716                     break;
   1717                 }
   1718             }
   1719             if (DEBUG_INPUT_METHOD) Slog.v(TAG_WM, "Adding " + N + " dialogs at pos=" + pos);
   1720             for (int i=0; i<N; i++) {
   1721                 WindowState win = dialogs.get(i);
   1722                 win.mTargetAppToken = targetAppToken;
   1723                 pos = reAddWindowLocked(pos, win);
   1724             }
   1725             if (DEBUG_INPUT_METHOD) {
   1726                 Slog.v(TAG_WM, "Final window list:");
   1727                 logWindowList(windows, "  ");
   1728             }
   1729             return;
   1730         }
   1731         for (int i=0; i<N; i++) {
   1732             WindowState win = dialogs.get(i);
   1733             win.mTargetAppToken = null;
   1734             reAddWindowToListInOrderLocked(win);
   1735             if (DEBUG_INPUT_METHOD) {
   1736                 Slog.v(TAG_WM, "No IM target, final list:");
   1737                 logWindowList(windows, "  ");
   1738             }
   1739         }
   1740     }
   1741 
   1742     boolean moveInputMethodWindowsIfNeededLocked(boolean needAssignLayers) {
   1743         final WindowState imWin = mInputMethodWindow;
   1744         final int DN = mInputMethodDialogs.size();
   1745         if (imWin == null && DN == 0) {
   1746             return false;
   1747         }
   1748 
   1749         // TODO(multidisplay): IMEs are only supported on the default display.
   1750         WindowList windows = getDefaultWindowListLocked();
   1751 
   1752         int imPos = findDesiredInputMethodWindowIndexLocked(true);
   1753         if (imPos >= 0) {
   1754             // In this case, the input method windows are to be placed
   1755             // immediately above the window they are targeting.
   1756 
   1757             // First check to see if the input method windows are already
   1758             // located here, and contiguous.
   1759             final int N = windows.size();
   1760             WindowState firstImWin = imPos < N
   1761                     ? windows.get(imPos) : null;
   1762 
   1763             // Figure out the actual input method window that should be
   1764             // at the bottom of their stack.
   1765             WindowState baseImWin = imWin != null
   1766                     ? imWin : mInputMethodDialogs.get(0);
   1767             if (baseImWin.mChildWindows.size() > 0) {
   1768                 WindowState cw = baseImWin.mChildWindows.get(0);
   1769                 if (cw.mSubLayer < 0) baseImWin = cw;
   1770             }
   1771 
   1772             if (firstImWin == baseImWin) {
   1773                 // The windows haven't moved...  but are they still contiguous?
   1774                 // First find the top IM window.
   1775                 int pos = imPos+1;
   1776                 while (pos < N) {
   1777                     if (!(windows.get(pos)).mIsImWindow) {
   1778                         break;
   1779                     }
   1780                     pos++;
   1781                 }
   1782                 pos++;
   1783                 // Now there should be no more input method windows above.
   1784                 while (pos < N) {
   1785                     if ((windows.get(pos)).mIsImWindow) {
   1786                         break;
   1787                     }
   1788                     pos++;
   1789                 }
   1790                 if (pos >= N) {
   1791                     // Z order is good.
   1792                     // The IM target window may be changed, so update the mTargetAppToken.
   1793                     if (imWin != null) {
   1794                         imWin.mTargetAppToken = mInputMethodTarget.mAppToken;
   1795                     }
   1796                     return false;
   1797                 }
   1798             }
   1799 
   1800             if (imWin != null) {
   1801                 if (DEBUG_INPUT_METHOD) {
   1802                     Slog.v(TAG_WM, "Moving IM from " + imPos);
   1803                     logWindowList(windows, "  ");
   1804                 }
   1805                 imPos = tmpRemoveWindowLocked(imPos, imWin);
   1806                 if (DEBUG_INPUT_METHOD) {
   1807                     Slog.v(TAG_WM, "List after removing with new pos " + imPos + ":");
   1808                     logWindowList(windows, "  ");
   1809                 }
   1810                 imWin.mTargetAppToken = mInputMethodTarget.mAppToken;
   1811                 reAddWindowLocked(imPos, imWin);
   1812                 if (DEBUG_INPUT_METHOD) {
   1813                     Slog.v(TAG_WM, "List after moving IM to " + imPos + ":");
   1814                     logWindowList(windows, "  ");
   1815                 }
   1816                 if (DN > 0) moveInputMethodDialogsLocked(imPos+1);
   1817             } else {
   1818                 moveInputMethodDialogsLocked(imPos);
   1819             }
   1820 
   1821         } else {
   1822             // In this case, the input method windows go in a fixed layer,
   1823             // because they aren't currently associated with a focus window.
   1824 
   1825             if (imWin != null) {
   1826                 if (DEBUG_INPUT_METHOD) Slog.v(TAG_WM, "Moving IM from " + imPos);
   1827                 tmpRemoveWindowLocked(0, imWin);
   1828                 imWin.mTargetAppToken = null;
   1829                 reAddWindowToListInOrderLocked(imWin);
   1830                 if (DEBUG_INPUT_METHOD) {
   1831                     Slog.v(TAG_WM, "List with no IM target:");
   1832                     logWindowList(windows, "  ");
   1833                 }
   1834                 if (DN > 0) moveInputMethodDialogsLocked(-1);
   1835             } else {
   1836                 moveInputMethodDialogsLocked(-1);
   1837             }
   1838 
   1839         }
   1840 
   1841         if (needAssignLayers) {
   1842             mLayersController.assignLayersLocked(windows);
   1843         }
   1844 
   1845         return true;
   1846     }
   1847 
   1848     private static boolean excludeWindowTypeFromTapOutTask(int windowType) {
   1849         switch (windowType) {
   1850             case TYPE_STATUS_BAR:
   1851             case TYPE_NAVIGATION_BAR:
   1852             case TYPE_INPUT_METHOD_DIALOG:
   1853                 return true;
   1854         }
   1855         return false;
   1856     }
   1857 
   1858     public int addWindow(Session session, IWindow client, int seq,
   1859             WindowManager.LayoutParams attrs, int viewVisibility, int displayId,
   1860             Rect outContentInsets, Rect outStableInsets, Rect outOutsets,
   1861             InputChannel outInputChannel) {
   1862         int[] appOp = new int[1];
   1863         int res = mPolicy.checkAddPermission(attrs, appOp);
   1864         if (res != WindowManagerGlobal.ADD_OKAY) {
   1865             return res;
   1866         }
   1867 
   1868         boolean reportNewConfig = false;
   1869         WindowState attachedWindow = null;
   1870         long origId;
   1871         final int type = attrs.type;
   1872 
   1873         synchronized(mWindowMap) {
   1874             if (!mDisplayReady) {
   1875                 throw new IllegalStateException("Display has not been initialialized");
   1876             }
   1877 
   1878             final DisplayContent displayContent = getDisplayContentLocked(displayId);
   1879             if (displayContent == null) {
   1880                 Slog.w(TAG_WM, "Attempted to add window to a display that does not exist: "
   1881                         + displayId + ".  Aborting.");
   1882                 return WindowManagerGlobal.ADD_INVALID_DISPLAY;
   1883             }
   1884             if (!displayContent.hasAccess(session.mUid)) {
   1885                 Slog.w(TAG_WM, "Attempted to add window to a display for which the application "
   1886                         + "does not have access: " + displayId + ".  Aborting.");
   1887                 return WindowManagerGlobal.ADD_INVALID_DISPLAY;
   1888             }
   1889 
   1890             if (mWindowMap.containsKey(client.asBinder())) {
   1891                 Slog.w(TAG_WM, "Window " + client + " is already added");
   1892                 return WindowManagerGlobal.ADD_DUPLICATE_ADD;
   1893             }
   1894 
   1895             if (type >= FIRST_SUB_WINDOW && type <= LAST_SUB_WINDOW) {
   1896                 attachedWindow = windowForClientLocked(null, attrs.token, false);
   1897                 if (attachedWindow == null) {
   1898                     Slog.w(TAG_WM, "Attempted to add window with token that is not a window: "
   1899                           + attrs.token + ".  Aborting.");
   1900                     return WindowManagerGlobal.ADD_BAD_SUBWINDOW_TOKEN;
   1901                 }
   1902                 if (attachedWindow.mAttrs.type >= FIRST_SUB_WINDOW
   1903                         && attachedWindow.mAttrs.type <= LAST_SUB_WINDOW) {
   1904                     Slog.w(TAG_WM, "Attempted to add window with token that is a sub-window: "
   1905                             + attrs.token + ".  Aborting.");
   1906                     return WindowManagerGlobal.ADD_BAD_SUBWINDOW_TOKEN;
   1907                 }
   1908             }
   1909 
   1910             if (type == TYPE_PRIVATE_PRESENTATION && !displayContent.isPrivate()) {
   1911                 Slog.w(TAG_WM, "Attempted to add private presentation window to a non-private display.  Aborting.");
   1912                 return WindowManagerGlobal.ADD_PERMISSION_DENIED;
   1913             }
   1914 
   1915             boolean addToken = false;
   1916             WindowToken token = mTokenMap.get(attrs.token);
   1917             AppWindowToken atoken = null;
   1918             if (token == null) {
   1919                 if (type >= FIRST_APPLICATION_WINDOW && type <= LAST_APPLICATION_WINDOW) {
   1920                     Slog.w(TAG_WM, "Attempted to add application window with unknown token "
   1921                           + attrs.token + ".  Aborting.");
   1922                     return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1923                 }
   1924                 if (type == TYPE_INPUT_METHOD) {
   1925                     Slog.w(TAG_WM, "Attempted to add input method window with unknown token "
   1926                           + attrs.token + ".  Aborting.");
   1927                     return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1928                 }
   1929                 if (type == TYPE_VOICE_INTERACTION) {
   1930                     Slog.w(TAG_WM, "Attempted to add voice interaction window with unknown token "
   1931                           + attrs.token + ".  Aborting.");
   1932                     return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1933                 }
   1934                 if (type == TYPE_WALLPAPER) {
   1935                     Slog.w(TAG_WM, "Attempted to add wallpaper window with unknown token "
   1936                           + attrs.token + ".  Aborting.");
   1937                     return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1938                 }
   1939                 if (type == TYPE_DREAM) {
   1940                     Slog.w(TAG_WM, "Attempted to add Dream window with unknown token "
   1941                           + attrs.token + ".  Aborting.");
   1942                     return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1943                 }
   1944                 if (type == TYPE_QS_DIALOG) {
   1945                     Slog.w(TAG_WM, "Attempted to add QS dialog window with unknown token "
   1946                           + attrs.token + ".  Aborting.");
   1947                     return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1948                 }
   1949                 if (type == TYPE_ACCESSIBILITY_OVERLAY) {
   1950                     Slog.w(TAG_WM, "Attempted to add Accessibility overlay window with unknown token "
   1951                             + attrs.token + ".  Aborting.");
   1952                     return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1953                 }
   1954                 token = new WindowToken(this, attrs.token, -1, false);
   1955                 addToken = true;
   1956             } else if (type >= FIRST_APPLICATION_WINDOW && type <= LAST_APPLICATION_WINDOW) {
   1957                 atoken = token.appWindowToken;
   1958                 if (atoken == null) {
   1959                     Slog.w(TAG_WM, "Attempted to add window with non-application token "
   1960                           + token + ".  Aborting.");
   1961                     return WindowManagerGlobal.ADD_NOT_APP_TOKEN;
   1962                 } else if (atoken.removed) {
   1963                     Slog.w(TAG_WM, "Attempted to add window with exiting application token "
   1964                           + token + ".  Aborting.");
   1965                     return WindowManagerGlobal.ADD_APP_EXITING;
   1966                 }
   1967                 if (type == TYPE_APPLICATION_STARTING && atoken.firstWindowDrawn) {
   1968                     // No need for this guy!
   1969                     if (DEBUG_STARTING_WINDOW || localLOGV) Slog.v(
   1970                             TAG_WM, "**** NO NEED TO START: " + attrs.getTitle());
   1971                     return WindowManagerGlobal.ADD_STARTING_NOT_NEEDED;
   1972                 }
   1973             } else if (type == TYPE_INPUT_METHOD) {
   1974                 if (token.windowType != TYPE_INPUT_METHOD) {
   1975                     Slog.w(TAG_WM, "Attempted to add input method window with bad token "
   1976                             + attrs.token + ".  Aborting.");
   1977                       return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1978                 }
   1979             } else if (type == TYPE_VOICE_INTERACTION) {
   1980                 if (token.windowType != TYPE_VOICE_INTERACTION) {
   1981                     Slog.w(TAG_WM, "Attempted to add voice interaction window with bad token "
   1982                             + attrs.token + ".  Aborting.");
   1983                       return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1984                 }
   1985             } else if (type == TYPE_WALLPAPER) {
   1986                 if (token.windowType != TYPE_WALLPAPER) {
   1987                     Slog.w(TAG_WM, "Attempted to add wallpaper window with bad token "
   1988                             + attrs.token + ".  Aborting.");
   1989                       return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1990                 }
   1991             } else if (type == TYPE_DREAM) {
   1992                 if (token.windowType != TYPE_DREAM) {
   1993                     Slog.w(TAG_WM, "Attempted to add Dream window with bad token "
   1994                             + attrs.token + ".  Aborting.");
   1995                       return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   1996                 }
   1997             } else if (type == TYPE_ACCESSIBILITY_OVERLAY) {
   1998                 if (token.windowType != TYPE_ACCESSIBILITY_OVERLAY) {
   1999                     Slog.w(TAG_WM, "Attempted to add Accessibility overlay window with bad token "
   2000                             + attrs.token + ".  Aborting.");
   2001                     return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   2002                 }
   2003             } else if (type == TYPE_QS_DIALOG) {
   2004                 if (token.windowType != TYPE_QS_DIALOG) {
   2005                     Slog.w(TAG_WM, "Attempted to add QS dialog window with bad token "
   2006                             + attrs.token + ".  Aborting.");
   2007                     return WindowManagerGlobal.ADD_BAD_APP_TOKEN;
   2008                 }
   2009             } else if (token.appWindowToken != null) {
   2010                 Slog.w(TAG_WM, "Non-null appWindowToken for system window of type=" + type);
   2011                 // It is not valid to use an app token with other system types; we will
   2012                 // instead make a new token for it (as if null had been passed in for the token).
   2013                 attrs.token = null;
   2014                 token = new WindowToken(this, null, -1, false);
   2015                 addToken = true;
   2016             }
   2017 
   2018             WindowState win = new WindowState(this, session, client, token,
   2019                     attachedWindow, appOp[0], seq, attrs, viewVisibility, displayContent);
   2020             if (win.mDeathRecipient == null) {
   2021                 // Client has apparently died, so there is no reason to
   2022                 // continue.
   2023                 Slog.w(TAG_WM, "Adding window client " + client.asBinder()
   2024                         + " that is dead, aborting.");
   2025                 return WindowManagerGlobal.ADD_APP_EXITING;
   2026             }
   2027 
   2028             if (win.getDisplayContent() == null) {
   2029                 Slog.w(TAG_WM, "Adding window to Display that has been removed.");
   2030                 return WindowManagerGlobal.ADD_INVALID_DISPLAY;
   2031             }
   2032 
   2033             mPolicy.adjustWindowParamsLw(win.mAttrs);
   2034             win.setShowToOwnerOnlyLocked(mPolicy.checkShowToOwnerOnly(attrs));
   2035 
   2036             res = mPolicy.prepareAddWindowLw(win, attrs);
   2037             if (res != WindowManagerGlobal.ADD_OKAY) {
   2038                 return res;
   2039             }
   2040 
   2041             final boolean openInputChannels = (outInputChannel != null
   2042                     && (attrs.inputFeatures & INPUT_FEATURE_NO_INPUT_CHANNEL) == 0);
   2043             if  (openInputChannels) {
   2044                 win.openInputChannel(outInputChannel);
   2045             }
   2046 
   2047             // From now on, no exceptions or errors allowed!
   2048 
   2049             res = WindowManagerGlobal.ADD_OKAY;
   2050 
   2051             if (excludeWindowTypeFromTapOutTask(type)) {
   2052                 displayContent.mTapExcludedWindows.add(win);
   2053             }
   2054 
   2055             origId = Binder.clearCallingIdentity();
   2056 
   2057             if (addToken) {
   2058                 mTokenMap.put(attrs.token, token);
   2059             }
   2060             win.attach();
   2061             mWindowMap.put(client.asBinder(), win);
   2062             if (win.mAppOp != AppOpsManager.OP_NONE) {
   2063                 int startOpResult = mAppOps.startOpNoThrow(win.mAppOp, win.getOwningUid(),
   2064                         win.getOwningPackage());
   2065                 if ((startOpResult != AppOpsManager.MODE_ALLOWED) &&
   2066                         (startOpResult != AppOpsManager.MODE_DEFAULT)) {
   2067                     win.setAppOpVisibilityLw(false);
   2068                 }
   2069             }
   2070 
   2071             if (type == TYPE_APPLICATION_STARTING && token.appWindowToken != null) {
   2072                 token.appWindowToken.startingWindow = win;
   2073                 if (DEBUG_STARTING_WINDOW) Slog.v (TAG_WM, "addWindow: " + token.appWindowToken
   2074                         + " startingWindow=" + win);
   2075             }
   2076 
   2077             boolean imMayMove = true;
   2078 
   2079             if (type == TYPE_INPUT_METHOD) {
   2080                 win.mGivenInsetsPending = true;
   2081                 mInputMethodWindow = win;
   2082                 addInputMethodWindowToListLocked(win);
   2083                 imMayMove = false;
   2084             } else if (type == TYPE_INPUT_METHOD_DIALOG) {
   2085                 mInputMethodDialogs.add(win);
   2086                 addWindowToListInOrderLocked(win, true);
   2087                 moveInputMethodDialogsLocked(findDesiredInputMethodWindowIndexLocked(true));
   2088                 imMayMove = false;
   2089             } else {
   2090                 addWindowToListInOrderLocked(win, true);
   2091                 if (type == TYPE_WALLPAPER) {
   2092                     mWallpaperControllerLocked.clearLastWallpaperTimeoutTime();
   2093                     displayContent.pendingLayoutChanges |= FINISH_LAYOUT_REDO_WALLPAPER;
   2094                 } else if ((attrs.flags&FLAG_SHOW_WALLPAPER) != 0) {
   2095                     displayContent.pendingLayoutChanges |= FINISH_LAYOUT_REDO_WALLPAPER;
   2096                 } else if (mWallpaperControllerLocked.isBelowWallpaperTarget(win)) {
   2097                     // If there is currently a wallpaper being shown, and
   2098                     // the base layer of the new window is below the current
   2099                     // layer of the target window, then adjust the wallpaper.
   2100                     // This is to avoid a new window being placed between the
   2101                     // wallpaper and its target.
   2102                     displayContent.pendingLayoutChanges |= FINISH_LAYOUT_REDO_WALLPAPER;
   2103                 }
   2104             }
   2105 
   2106             // If the window is being added to a task that's docked but non-resizeable,
   2107             // we need to update this new window's scroll position when it's added.
   2108             win.applyScrollIfNeeded();
   2109 
   2110             // If the window is being added to a stack that's currently adjusted for IME,
   2111             // make sure to apply the same adjust to this new window.
   2112             win.applyAdjustForImeIfNeeded();
   2113 
   2114             if (type == TYPE_DOCK_DIVIDER) {
   2115                 getDefaultDisplayContentLocked().getDockedDividerController().setWindow(win);
   2116             }
   2117 
   2118             final WindowStateAnimator winAnimator = win.mWinAnimator;
   2119             winAnimator.mEnterAnimationPending = true;
   2120             winAnimator.mEnteringAnimation = true;
   2121             // Check if we need to prepare a transition for replacing window first.
   2122             if (atoken != null && !prepareWindowReplacementTransition(atoken)) {
   2123                 // If not, check if need to set up a dummy transition during display freeze
   2124                 // so that the unfreeze wait for the apps to draw. This might be needed if
   2125                 // the app is relaunching.
   2126                 prepareNoneTransitionForRelaunching(atoken);
   2127             }
   2128 
   2129             if (displayContent.isDefaultDisplay) {
   2130                 final DisplayInfo displayInfo = displayContent.getDisplayInfo();
   2131                 final Rect taskBounds;
   2132                 if (atoken != null && atoken.mTask != null) {
   2133                     taskBounds = mTmpRect;
   2134                     atoken.mTask.getBounds(mTmpRect);
   2135                 } else {
   2136                     taskBounds = null;
   2137                 }
   2138                 if (mPolicy.getInsetHintLw(win.mAttrs, taskBounds, mRotation,
   2139                         displayInfo.logicalWidth, displayInfo.logicalHeight, outContentInsets,
   2140                         outStableInsets, outOutsets)) {
   2141                     res |= WindowManagerGlobal.ADD_FLAG_ALWAYS_CONSUME_NAV_BAR;
   2142                 }
   2143             } else {
   2144                 outContentInsets.setEmpty();
   2145                 outStableInsets.setEmpty();
   2146             }
   2147 
   2148             if (mInTouchMode) {
   2149                 res |= WindowManagerGlobal.ADD_FLAG_IN_TOUCH_MODE;
   2150             }
   2151             if (win.mAppToken == null || !win.mAppToken.clientHidden) {
   2152                 res |= WindowManagerGlobal.ADD_FLAG_APP_VISIBLE;
   2153             }
   2154 
   2155             mInputMonitor.setUpdateInputWindowsNeededLw();
   2156 
   2157             boolean focusChanged = false;
   2158             if (win.canReceiveKeys()) {
   2159                 focusChanged = updateFocusedWindowLocked(UPDATE_FOCUS_WILL_ASSIGN_LAYERS,
   2160                         false /*updateInputWindows*/);
   2161                 if (focusChanged) {
   2162                     imMayMove = false;
   2163                 }
   2164             }
   2165 
   2166             if (imMayMove) {
   2167                 moveInputMethodWindowsIfNeededLocked(false);
   2168             }
   2169 
   2170             mLayersController.assignLayersLocked(displayContent.getWindowList());
   2171             // Don't do layout here, the window must call
   2172             // relayout to be displayed, so we'll do it there.
   2173 
   2174             if (focusChanged) {
   2175                 mInputMonitor.setInputFocusLw(mCurrentFocus, false /*updateInputWindows*/);
   2176             }
   2177             mInputMonitor.updateInputWindowsLw(false /*force*/);
   2178 
   2179             if (localLOGV || DEBUG_ADD_REMOVE) Slog.v(TAG_WM, "addWindow: New client "
   2180                     + client.asBinder() + ": window=" + win + " Callers=" + Debug.getCallers(5));
   2181 
   2182             if (win.isVisibleOrAdding() && updateOrientationFromAppTokensLocked(false)) {
   2183                 reportNewConfig = true;
   2184             }
   2185             if (attrs.removeTimeoutMilliseconds > 0) {
   2186                 mH.sendMessageDelayed(
   2187                         mH.obtainMessage(H.WINDOW_REMOVE_TIMEOUT, win),
   2188                         attrs.removeTimeoutMilliseconds);
   2189             }
   2190         }
   2191 
   2192         if (reportNewConfig) {
   2193             sendNewConfiguration();
   2194         }
   2195 
   2196         Binder.restoreCallingIdentity(origId);
   2197 
   2198         return res;
   2199     }
   2200 
   2201     /**
   2202      * Returns true if we're done setting up any transitions.
   2203      */
   2204     private boolean prepareWindowReplacementTransition(AppWindowToken atoken) {
   2205         atoken.clearAllDrawn();
   2206         WindowState replacedWindow = null;
   2207         for (int i = atoken.windows.size() - 1; i >= 0 && replacedWindow == null; i--) {
   2208             WindowState candidate = atoken.windows.get(i);
   2209             if (candidate.mAnimatingExit && candidate.mWillReplaceWindow
   2210                     && candidate.mAnimateReplacingWindow) {
   2211                 replacedWindow = candidate;
   2212             }
   2213         }
   2214         if (replacedWindow == null) {
   2215             // We expect to already receive a request to remove the old window. If it did not
   2216             // happen, let's just simply add a window.
   2217             return false;
   2218         }
   2219         // We use the visible frame, because we want the animation to morph the window from what
   2220         // was visible to the user to the final destination of the new window.
   2221         Rect frame = replacedWindow.mVisibleFrame;
   2222         // We treat this as if this activity was opening, so we can trigger the app transition
   2223         // animation and piggy-back on existing transition animation infrastructure.
   2224         mOpeningApps.add(atoken);
   2225         prepareAppTransition(AppTransition.TRANSIT_ACTIVITY_RELAUNCH, ALWAYS_KEEP_CURRENT);
   2226         mAppTransition.overridePendingAppTransitionClipReveal(frame.left, frame.top,
   2227                 frame.width(), frame.height());
   2228         executeAppTransition();
   2229         return true;
   2230     }
   2231 
   2232     private void prepareNoneTransitionForRelaunching(AppWindowToken atoken) {
   2233         // Set up a none-transition and add the app to opening apps, so that the display
   2234         // unfreeze wait for the apps to be drawn.
   2235         // Note that if the display unfroze already because app unfreeze timed out,
   2236         // we don't set up the transition anymore and just let it go.
   2237         if (mDisplayFrozen && !mOpeningApps.contains(atoken) && atoken.isRelaunching()) {
   2238             mOpeningApps.add(atoken);
   2239             prepareAppTransition(AppTransition.TRANSIT_NONE, !ALWAYS_KEEP_CURRENT);
   2240             executeAppTransition();
   2241         }
   2242     }
   2243 
   2244     /**
   2245      * Returns whether screen capture is disabled for all windows of a specific user.
   2246      */
   2247     boolean isScreenCaptureDisabledLocked(int userId) {
   2248         Boolean disabled = mScreenCaptureDisabled.get(userId);
   2249         if (disabled == null) {
   2250             return false;
   2251         }
   2252         return disabled;
   2253     }
   2254 
   2255     boolean isSecureLocked(WindowState w) {
   2256         if ((w.mAttrs.flags&WindowManager.LayoutParams.FLAG_SECURE) != 0) {
   2257             return true;
   2258         }
   2259         if (isScreenCaptureDisabledLocked(UserHandle.getUserId(w.mOwnerUid))) {
   2260             return true;
   2261         }
   2262         return false;
   2263     }
   2264 
   2265     /**
   2266      * Set mScreenCaptureDisabled for specific user
   2267      */
   2268     @Override
   2269     public void setScreenCaptureDisabled(int userId, boolean disabled) {
   2270         int callingUid = Binder.getCallingUid();
   2271         if (callingUid != Process.SYSTEM_UID) {
   2272             throw new SecurityException("Only system can call setScreenCaptureDisabled.");
   2273         }
   2274 
   2275         synchronized(mWindowMap) {
   2276             mScreenCaptureDisabled.put(userId, disabled);
   2277             // Update secure surface for all windows belonging to this user.
   2278             for (int displayNdx = mDisplayContents.size() - 1; displayNdx >= 0; --displayNdx) {
   2279                 WindowList windows = mDisplayContents.valueAt(displayNdx).getWindowList();
   2280                 for (int winNdx = windows.size() - 1; winNdx >= 0; --winNdx) {
   2281                     final WindowState win = windows.get(winNdx);
   2282                     if (win.mHasSurface && userId == UserHandle.getUserId(win.mOwnerUid)) {
   2283                         win.mWinAnimator.setSecureLocked(disabled);
   2284                     }
   2285                 }
   2286             }
   2287         }
   2288     }
   2289 
   2290     private void setupWindowForRemoveOnExit(WindowState win) {
   2291         win.mRemoveOnExit = true;
   2292         win.setDisplayLayoutNeeded();
   2293         // Request a focus update as this window's input channel is already gone. Otherwise
   2294         // we could have no focused window in input manager.
   2295         final boolean focusChanged = updateFocusedWindowLocked(
   2296                 UPDATE_FOCUS_WILL_PLACE_SURFACES, false /*updateInputWindows*/);
   2297         mWindowPlacerLocked.performSurfacePlacement();
   2298         if (focusChanged) {
   2299             mInputMonitor.updateInputWindowsLw(false /*force*/);
   2300         }
   2301     }
   2302 
   2303     public void removeWindow(Session session, IWindow client) {
   2304         synchronized(mWindowMap) {
   2305             WindowState win = windowForClientLocked(session, client, false);
   2306             if (win == null) {
   2307                 return;
   2308             }
   2309             removeWindowLocked(win);
   2310         }
   2311     }
   2312 
   2313     void removeWindowLocked(WindowState win) {
   2314         removeWindowLocked(win, false);
   2315     }
   2316 
   2317     void removeWindowLocked(WindowState win, boolean keepVisibleDeadWindow) {
   2318         win.mWindowRemovalAllowed = true;
   2319         if (DEBUG_ADD_REMOVE) Slog.v(TAG,
   2320                 "removeWindowLocked: " + win + " callers=" + Debug.getCallers(4));
   2321 
   2322         final boolean startingWindow = win.mAttrs.type == TYPE_APPLICATION_STARTING;
   2323         if (startingWindow) {
   2324             if (DEBUG_STARTING_WINDOW) Slog.d(TAG_WM, "Starting window removed " + win);
   2325         }
   2326 
   2327         if (localLOGV || DEBUG_FOCUS || DEBUG_FOCUS_LIGHT && win == mCurrentFocus) Slog.v(
   2328                 TAG_WM, "Remove " + win + " client="
   2329                 + Integer.toHexString(System.identityHashCode(win.mClient.asBinder()))
   2330                 + ", surfaceController=" + win.mWinAnimator.mSurfaceController + " Callers="
   2331                 + Debug.getCallers(4));
   2332 
   2333         final long origId = Binder.clearCallingIdentity();
   2334 
   2335         win.disposeInputChannel();
   2336 
   2337         if (DEBUG_APP_TRANSITIONS) Slog.v(TAG_WM,
   2338                 "Remove " + win + ": mSurfaceController=" + win.mWinAnimator.mSurfaceController
   2339                 + " mAnimatingExit=" + win.mAnimatingExit
   2340                 + " mRemoveOnExit=" + win.mRemoveOnExit
   2341                 + " mHasSurface=" + win.mHasSurface
   2342                 + " surfaceShowing=" + win.mWinAnimator.getShown()
   2343                 + " isAnimationSet=" + win.mWinAnimator.isAnimationSet()
   2344                 + " app-animation="
   2345                 + (win.mAppToken != null ? win.mAppToken.mAppAnimator.animation : null)
   2346                 + " mWillReplaceWindow=" + win.mWillReplaceWindow
   2347                 + " inPendingTransaction="
   2348                 + (win.mAppToken != null ? win.mAppToken.inPendingTransaction : false)
   2349                 + " mDisplayFrozen=" + mDisplayFrozen
   2350                 + " callers=" + Debug.getCallers(6));
   2351         // Visibility of the removed window. Will be used later to update orientation later on.
   2352         boolean wasVisible = false;
   2353         // First, see if we need to run an animation. If we do, we have to hold off on removing the
   2354         // window until the animation is done. If the display is frozen, just remove immediately,
   2355         // since the animation wouldn't be seen.
   2356         if (win.mHasSurface && okToDisplay()) {
   2357             final AppWindowToken appToken = win.mAppToken;
   2358             if (win.mWillReplaceWindow) {
   2359                 // This window is going to be replaced. We need to keep it around until the new one
   2360                 // gets added, then we will get rid of this one.
   2361                 if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM, "Preserving " + win + " until the new one is "
   2362                         + "added");
   2363                 // TODO: We are overloading mAnimatingExit flag to prevent the window state from
   2364                 // been removed. We probably need another flag to indicate that window removal
   2365                 // should be deffered vs. overloading the flag that says we are playing an exit
   2366                 // animation.
   2367                 win.mAnimatingExit = true;
   2368                 win.mReplacingRemoveRequested = true;
   2369                 Binder.restoreCallingIdentity(origId);
   2370                 return;
   2371             }
   2372 
   2373             if (win.isAnimatingWithSavedSurface() && !appToken.allDrawnExcludingSaved) {
   2374                 // We started enter animation early with a saved surface, now the app asks to remove
   2375                 // this window. If we remove it now and the app is not yet drawn, we'll show a
   2376                 // flicker. Delay the removal now until it's really drawn.
   2377                 if (DEBUG_ADD_REMOVE) {
   2378                     Slog.d(TAG_WM, "removeWindowLocked: delay removal of " + win
   2379                             + " due to early animation");
   2380                 }
   2381                 // Do not set mAnimatingExit to true here, it will cause the surface to be hidden
   2382                 // immediately after the enter animation is done. If the app is not yet drawn then
   2383                 // it will show up as a flicker.
   2384                 setupWindowForRemoveOnExit(win);
   2385                 Binder.restoreCallingIdentity(origId);
   2386                 return;
   2387             }
   2388             // If we are not currently running the exit animation, we need to see about starting one
   2389             wasVisible = win.isWinVisibleLw();
   2390 
   2391             if (keepVisibleDeadWindow) {
   2392                 if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM,
   2393                         "Not removing " + win + " because app died while it's visible");
   2394 
   2395                 win.mAppDied = true;
   2396                 win.setDisplayLayoutNeeded();
   2397                 mWindowPlacerLocked.performSurfacePlacement();
   2398 
   2399                 // Set up a replacement input channel since the app is now dead.
   2400                 // We need to catch tapping on the dead window to restart the app.
   2401                 win.openInputChannel(null);
   2402                 mInputMonitor.updateInputWindowsLw(true /*force*/);
   2403 
   2404                 Binder.restoreCallingIdentity(origId);
   2405                 return;
   2406             }
   2407 
   2408             final WindowStateAnimator winAnimator = win.mWinAnimator;
   2409             if (wasVisible) {
   2410                 final int transit = (!startingWindow) ? TRANSIT_EXIT : TRANSIT_PREVIEW_DONE;
   2411 
   2412                 // Try starting an animation.
   2413                 if (winAnimator.applyAnimationLocked(transit, false)) {
   2414                     win.mAnimatingExit = true;
   2415                 }
   2416                 //TODO (multidisplay): Magnification is supported only for the default display.
   2417                 if (mAccessibilityController != null
   2418                         && win.getDisplayId() == Display.DEFAULT_DISPLAY) {
   2419                     mAccessibilityController.onWindowTransitionLocked(win, transit);
   2420                 }
   2421             }
   2422             final boolean isAnimating =
   2423                     winAnimator.isAnimationSet() && !winAnimator.isDummyAnimation();
   2424             final boolean lastWindowIsStartingWindow = startingWindow && appToken != null
   2425                     && appToken.allAppWindows.size() == 1;
   2426             // We delay the removal of a window if it has a showing surface that can be used to run
   2427             // exit animation and it is marked as exiting.
   2428             // Also, If isn't the an animating starting window that is the last window in the app.
   2429             // We allow the removal of the non-animating starting window now as there is no
   2430             // additional window or animation that will trigger its removal.
   2431             if (winAnimator.getShown() && win.mAnimatingExit
   2432                     && (!lastWindowIsStartingWindow || isAnimating)) {
   2433                 // The exit animation is running or should run... wait for it!
   2434                 if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM,
   2435                         "Not removing " + win + " due to exit animation ");
   2436                 setupWindowForRemoveOnExit(win);
   2437                 if (appToken != null) {
   2438                     appToken.updateReportedVisibilityLocked();
   2439                 }
   2440                 Binder.restoreCallingIdentity(origId);
   2441                 return;
   2442             }
   2443         }
   2444 
   2445         removeWindowInnerLocked(win);
   2446         // Removing a visible window will effect the computed orientation
   2447         // So just update orientation if needed.
   2448         if (wasVisible && updateOrientationFromAppTokensLocked(false)) {
   2449             mH.sendEmptyMessage(H.SEND_NEW_CONFIGURATION);
   2450         }
   2451         updateFocusedWindowLocked(UPDATE_FOCUS_NORMAL, true /*updateInputWindows*/);
   2452         Binder.restoreCallingIdentity(origId);
   2453     }
   2454 
   2455     void removeWindowInnerLocked(WindowState win) {
   2456         if (win.mRemoved) {
   2457             // Nothing to do.
   2458             if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM,
   2459                     "removeWindowInnerLocked: " + win + " Already removed...");
   2460             return;
   2461         }
   2462 
   2463         for (int i = win.mChildWindows.size() - 1; i >= 0; i--) {
   2464             WindowState cwin = win.mChildWindows.get(i);
   2465             Slog.w(TAG_WM, "Force-removing child win " + cwin + " from container " + win);
   2466             removeWindowInnerLocked(cwin);
   2467         }
   2468 
   2469         win.mRemoved = true;
   2470 
   2471         if (mInputMethodTarget == win) {
   2472             moveInputMethodWindowsIfNeededLocked(false);
   2473         }
   2474 
   2475         if (false) {
   2476             RuntimeException e = new RuntimeException("here");
   2477             e.fillInStackTrace();
   2478             Slog.w(TAG_WM, "Removing window " + win, e);
   2479         }
   2480 
   2481         final int type = win.mAttrs.type;
   2482         if (excludeWindowTypeFromTapOutTask(type)) {
   2483             final DisplayContent displaycontent = win.getDisplayContent();
   2484             displaycontent.mTapExcludedWindows.remove(win);
   2485         }
   2486         mPolicy.removeWindowLw(win);
   2487         win.removeLocked();
   2488 
   2489         if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM, "removeWindowInnerLocked: " + win);
   2490         mWindowMap.remove(win.mClient.asBinder());
   2491         if (win.mAppOp != AppOpsManager.OP_NONE) {
   2492             mAppOps.finishOp(win.mAppOp, win.getOwningUid(), win.getOwningPackage());
   2493         }
   2494 
   2495         mPendingRemove.remove(win);
   2496         mResizingWindows.remove(win);
   2497         mWindowsChanged = true;
   2498         if (DEBUG_WINDOW_MOVEMENT) Slog.v(TAG_WM, "Final remove of window: " + win);
   2499 
   2500         if (mInputMethodWindow == win) {
   2501             mInputMethodWindow = null;
   2502         } else if (win.mAttrs.type == TYPE_INPUT_METHOD_DIALOG) {
   2503             mInputMethodDialogs.remove(win);
   2504         }
   2505 
   2506         final WindowToken token = win.mToken;
   2507         final AppWindowToken atoken = win.mAppToken;
   2508         if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM, "Removing " + win + " from " + token);
   2509         token.windows.remove(win);
   2510         if (atoken != null) {
   2511             atoken.allAppWindows.remove(win);
   2512         }
   2513         if (localLOGV) Slog.v(
   2514                 TAG_WM, "**** Removing window " + win + ": count="
   2515                 + token.windows.size());
   2516         if (token.windows.size() == 0) {
   2517             if (!token.explicit) {
   2518                 mTokenMap.remove(token.token);
   2519             } else if (atoken != null) {
   2520                 atoken.firstWindowDrawn = false;
   2521                 atoken.clearAllDrawn();
   2522             }
   2523         }
   2524 
   2525         if (atoken != null) {
   2526             if (atoken.startingWindow == win) {
   2527                 if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM, "Notify removed startingWindow " + win);
   2528                 scheduleRemoveStartingWindowLocked(atoken);
   2529             } else
   2530             if (atoken.allAppWindows.size() == 0 && atoken.startingData != null) {
   2531                 // If this is the last window and we had requested a starting
   2532                 // transition window, well there is no point now.
   2533                 if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM, "Nulling last startingWindow");
   2534                 atoken.startingData = null;
   2535             } else if (atoken.allAppWindows.size() == 1 && atoken.startingView != null) {
   2536                 // If this is the last window except for a starting transition
   2537                 // window, we need to get rid of the starting transition.
   2538                 scheduleRemoveStartingWindowLocked(atoken);
   2539             }
   2540         }
   2541 
   2542         if (type == TYPE_WALLPAPER) {
   2543             mWallpaperControllerLocked.clearLastWallpaperTimeoutTime();
   2544             getDefaultDisplayContentLocked().pendingLayoutChanges |=
   2545                     WindowManagerPolicy.FINISH_LAYOUT_REDO_WALLPAPER;
   2546         } else if ((win.mAttrs.flags&FLAG_SHOW_WALLPAPER) != 0) {
   2547             getDefaultDisplayContentLocked().pendingLayoutChanges |=
   2548                     WindowManagerPolicy.FINISH_LAYOUT_REDO_WALLPAPER;
   2549         }
   2550 
   2551         final WindowList windows = win.getWindowList();
   2552         if (windows != null) {
   2553             windows.remove(win);
   2554             if (!mWindowPlacerLocked.isInLayout()) {
   2555                 mLayersController.assignLayersLocked(windows);
   2556                 win.setDisplayLayoutNeeded();
   2557                 mWindowPlacerLocked.performSurfacePlacement();
   2558                 if (win.mAppToken != null) {
   2559                     win.mAppToken.updateReportedVisibilityLocked();
   2560                 }
   2561             }
   2562         }
   2563 
   2564         mInputMonitor.updateInputWindowsLw(true /*force*/);
   2565     }
   2566 
   2567     public void updateAppOpsState() {
   2568         synchronized(mWindowMap) {
   2569             final int numDisplays = mDisplayContents.size();
   2570             for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   2571                 final WindowList windows = mDisplayContents.valueAt(displayNdx).getWindowList();
   2572                 final int numWindows = windows.size();
   2573                 for (int winNdx = 0; winNdx < numWindows; ++winNdx) {
   2574                     final WindowState win = windows.get(winNdx);
   2575                     if (win.mAppOp != AppOpsManager.OP_NONE) {
   2576                         final int mode = mAppOps.checkOpNoThrow(win.mAppOp, win.getOwningUid(),
   2577                                 win.getOwningPackage());
   2578                         win.setAppOpVisibilityLw(mode == AppOpsManager.MODE_ALLOWED ||
   2579                                 mode == AppOpsManager.MODE_DEFAULT);
   2580                     }
   2581                 }
   2582             }
   2583         }
   2584     }
   2585 
   2586     static void logSurface(WindowState w, String msg, boolean withStackTrace) {
   2587         String str = "  SURFACE " + msg + ": " + w;
   2588         if (withStackTrace) {
   2589             logWithStack(TAG, str);
   2590         } else {
   2591             Slog.i(TAG_WM, str);
   2592         }
   2593     }
   2594 
   2595     static void logSurface(SurfaceControl s, String title, String msg) {
   2596         String str = "  SURFACE " + s + ": " + msg + " / " + title;
   2597         Slog.i(TAG_WM, str);
   2598     }
   2599 
   2600     static void logWithStack(String tag, String s) {
   2601         RuntimeException e = null;
   2602         if (SHOW_STACK_CRAWLS) {
   2603             e = new RuntimeException();
   2604             e.fillInStackTrace();
   2605         }
   2606         Slog.i(tag, s, e);
   2607     }
   2608 
   2609     void setTransparentRegionWindow(Session session, IWindow client, Region region) {
   2610         long origId = Binder.clearCallingIdentity();
   2611         try {
   2612             synchronized (mWindowMap) {
   2613                 WindowState w = windowForClientLocked(session, client, false);
   2614                 if (SHOW_TRANSACTIONS) WindowManagerService.logSurface(w,
   2615                         "transparentRegionHint=" + region, false);
   2616 
   2617                 if ((w != null) && w.mHasSurface) {
   2618                     w.mWinAnimator.setTransparentRegionHintLocked(region);
   2619                 }
   2620             }
   2621         } finally {
   2622             Binder.restoreCallingIdentity(origId);
   2623         }
   2624     }
   2625 
   2626     void setInsetsWindow(Session session, IWindow client, int touchableInsets, Rect contentInsets,
   2627             Rect visibleInsets, Region touchableRegion) {
   2628         long origId = Binder.clearCallingIdentity();
   2629         try {
   2630             synchronized (mWindowMap) {
   2631                 WindowState w = windowForClientLocked(session, client, false);
   2632                 if (DEBUG_LAYOUT) Slog.d(TAG, "setInsetsWindow " + w
   2633                         + ", contentInsets=" + w.mGivenContentInsets + " -> " + contentInsets
   2634                         + ", visibleInsets=" + w.mGivenVisibleInsets + " -> " + visibleInsets
   2635                         + ", touchableRegion=" + w.mGivenTouchableRegion + " -> " + touchableRegion
   2636                         + ", touchableInsets " + w.mTouchableInsets + " -> " + touchableInsets);
   2637                 if (w != null) {
   2638                     w.mGivenInsetsPending = false;
   2639                     w.mGivenContentInsets.set(contentInsets);
   2640                     w.mGivenVisibleInsets.set(visibleInsets);
   2641                     w.mGivenTouchableRegion.set(touchableRegion);
   2642                     w.mTouchableInsets = touchableInsets;
   2643                     if (w.mGlobalScale != 1) {
   2644                         w.mGivenContentInsets.scale(w.mGlobalScale);
   2645                         w.mGivenVisibleInsets.scale(w.mGlobalScale);
   2646                         w.mGivenTouchableRegion.scale(w.mGlobalScale);
   2647                     }
   2648                     w.setDisplayLayoutNeeded();
   2649                     mWindowPlacerLocked.performSurfacePlacement();
   2650                 }
   2651             }
   2652         } finally {
   2653             Binder.restoreCallingIdentity(origId);
   2654         }
   2655     }
   2656 
   2657     public void getWindowDisplayFrame(Session session, IWindow client,
   2658             Rect outDisplayFrame) {
   2659         synchronized(mWindowMap) {
   2660             WindowState win = windowForClientLocked(session, client, false);
   2661             if (win == null) {
   2662                 outDisplayFrame.setEmpty();
   2663                 return;
   2664             }
   2665             outDisplayFrame.set(win.mDisplayFrame);
   2666         }
   2667     }
   2668 
   2669     public void onRectangleOnScreenRequested(IBinder token, Rect rectangle) {
   2670         synchronized (mWindowMap) {
   2671             if (mAccessibilityController != null) {
   2672                 WindowState window = mWindowMap.get(token);
   2673                 //TODO (multidisplay): Magnification is supported only for the default display.
   2674                 if (window != null && window.getDisplayId() == Display.DEFAULT_DISPLAY) {
   2675                     mAccessibilityController.onRectangleOnScreenRequestedLocked(rectangle);
   2676                 }
   2677             }
   2678         }
   2679     }
   2680 
   2681     public IWindowId getWindowId(IBinder token) {
   2682         synchronized (mWindowMap) {
   2683             WindowState window = mWindowMap.get(token);
   2684             return window != null ? window.mWindowId : null;
   2685         }
   2686     }
   2687 
   2688     public void pokeDrawLock(Session session, IBinder token) {
   2689         synchronized (mWindowMap) {
   2690             WindowState window = windowForClientLocked(session, token, false);
   2691             if (window != null) {
   2692                 window.pokeDrawLockLw(mDrawLockTimeoutMillis);
   2693             }
   2694         }
   2695     }
   2696 
   2697     void repositionChild(Session session, IWindow client,
   2698             int left, int top, int right, int bottom,
   2699             long frameNumber, Rect outFrame) {
   2700         Trace.traceBegin(Trace.TRACE_TAG_WINDOW_MANAGER, "repositionChild");
   2701         long origId = Binder.clearCallingIdentity();
   2702 
   2703         try {
   2704             synchronized(mWindowMap) {
   2705                 WindowState win = windowForClientLocked(session, client, false);
   2706                 if (win == null) {
   2707                     return;
   2708                 }
   2709                 if (win.mAttachedWindow == null) {
   2710                     throw new IllegalArgumentException(
   2711                             "repositionChild called but window is not"
   2712                             + "attached to a parent win=" + win);
   2713                 }
   2714 
   2715                 win.mAttrs.x = left;
   2716                 win.mAttrs.y = top;
   2717                 win.mAttrs.width = right - left;
   2718                 win.mAttrs.height = bottom - top;
   2719                 win.setWindowScale(win.mRequestedWidth, win.mRequestedHeight);
   2720 
   2721                 if (win.mHasSurface) {
   2722                     if (SHOW_TRANSACTIONS) {
   2723                         Slog.i(TAG_WM, ">>> OPEN TRANSACTION repositionChild");
   2724                     }
   2725 
   2726                     SurfaceControl.openTransaction();
   2727 
   2728                     try {
   2729 
   2730                         win.applyGravityAndUpdateFrame(win.mContainingFrame, win.mDisplayFrame);
   2731                         win.mWinAnimator.computeShownFrameLocked();
   2732 
   2733                         win.mWinAnimator.setSurfaceBoundariesLocked(false);
   2734 
   2735                         if (frameNumber > 0) {
   2736                             win.mWinAnimator.deferTransactionUntilParentFrame(frameNumber);
   2737                         }
   2738 
   2739                     } finally {
   2740                         SurfaceControl.closeTransaction();
   2741                         if (SHOW_TRANSACTIONS) {
   2742                             Slog.i(TAG_WM, "<<< CLOSE TRANSACTION repositionChild");
   2743                         }
   2744                     }
   2745                 }
   2746 
   2747                 outFrame = win.mCompatFrame;
   2748             }
   2749         } finally {
   2750             Binder.restoreCallingIdentity(origId);
   2751             Trace.traceEnd(Trace.TRACE_TAG_WINDOW_MANAGER);
   2752         }
   2753     }
   2754 
   2755     public int relayoutWindow(Session session, IWindow client, int seq,
   2756             WindowManager.LayoutParams attrs, int requestedWidth,
   2757             int requestedHeight, int viewVisibility, int flags,
   2758             Rect outFrame, Rect outOverscanInsets, Rect outContentInsets,
   2759             Rect outVisibleInsets, Rect outStableInsets, Rect outOutsets, Rect outBackdropFrame,
   2760             Configuration outConfig, Surface outSurface) {
   2761         int result = 0;
   2762         boolean configChanged;
   2763         boolean hasStatusBarPermission =
   2764                 mContext.checkCallingOrSelfPermission(android.Manifest.permission.STATUS_BAR)
   2765                         == PackageManager.PERMISSION_GRANTED;
   2766 
   2767         long origId = Binder.clearCallingIdentity();
   2768         synchronized(mWindowMap) {
   2769             WindowState win = windowForClientLocked(session, client, false);
   2770             if (win == null) {
   2771                 return 0;
   2772             }
   2773 
   2774             WindowStateAnimator winAnimator = win.mWinAnimator;
   2775             if (viewVisibility != View.GONE) {
   2776                 win.setRequestedSize(requestedWidth, requestedHeight);
   2777             }
   2778 
   2779             int attrChanges = 0;
   2780             int flagChanges = 0;
   2781             if (attrs != null) {
   2782                 mPolicy.adjustWindowParamsLw(attrs);
   2783                 // if they don't have the permission, mask out the status bar bits
   2784                 if (seq == win.mSeq) {
   2785                     int systemUiVisibility = attrs.systemUiVisibility
   2786                             | attrs.subtreeSystemUiVisibility;
   2787                     if ((systemUiVisibility & DISABLE_MASK) != 0) {
   2788                         if (!hasStatusBarPermission) {
   2789                             systemUiVisibility &= ~DISABLE_MASK;
   2790                         }
   2791                     }
   2792                     win.mSystemUiVisibility = systemUiVisibility;
   2793                 }
   2794                 if (win.mAttrs.type != attrs.type) {
   2795                     throw new IllegalArgumentException(
   2796                             "Window type can not be changed after the window is added.");
   2797                 }
   2798 
   2799                 // Odd choice but less odd than embedding in copyFrom()
   2800                 if ((attrs.privateFlags & WindowManager.LayoutParams.PRIVATE_FLAG_PRESERVE_GEOMETRY)
   2801                         != 0) {
   2802                     attrs.x = win.mAttrs.x;
   2803                     attrs.y = win.mAttrs.y;
   2804                     attrs.width = win.mAttrs.width;
   2805                     attrs.height = win.mAttrs.height;
   2806                 }
   2807 
   2808                 flagChanges = win.mAttrs.flags ^= attrs.flags;
   2809                 attrChanges = win.mAttrs.copyFrom(attrs);
   2810                 if ((attrChanges & (WindowManager.LayoutParams.LAYOUT_CHANGED
   2811                         | WindowManager.LayoutParams.SYSTEM_UI_VISIBILITY_CHANGED)) != 0) {
   2812                     win.mLayoutNeeded = true;
   2813                 }
   2814             }
   2815 
   2816             if (DEBUG_LAYOUT) Slog.v(TAG_WM, "Relayout " + win + ": viewVisibility=" + viewVisibility
   2817                     + " req=" + requestedWidth + "x" + requestedHeight + " " + win.mAttrs);
   2818             winAnimator.mSurfaceDestroyDeferred = (flags & RELAYOUT_DEFER_SURFACE_DESTROY) != 0;
   2819             win.mEnforceSizeCompat =
   2820                     (win.mAttrs.privateFlags & PRIVATE_FLAG_COMPATIBLE_WINDOW) != 0;
   2821             if ((attrChanges & WindowManager.LayoutParams.ALPHA_CHANGED) != 0) {
   2822                 winAnimator.mAlpha = attrs.alpha;
   2823             }
   2824             win.setWindowScale(win.mRequestedWidth, win.mRequestedHeight);
   2825 
   2826             if (win.mAttrs.surfaceInsets.left != 0
   2827                     || win.mAttrs.surfaceInsets.top != 0
   2828                     || win.mAttrs.surfaceInsets.right != 0
   2829                     || win.mAttrs.surfaceInsets.bottom != 0) {
   2830                 winAnimator.setOpaqueLocked(false);
   2831             }
   2832 
   2833             boolean imMayMove = (flagChanges & (FLAG_ALT_FOCUSABLE_IM | FLAG_NOT_FOCUSABLE)) != 0;
   2834             final boolean isDefaultDisplay = win.isDefaultDisplay();
   2835             boolean focusMayChange = isDefaultDisplay && (win.mViewVisibility != viewVisibility
   2836                     || ((flagChanges & FLAG_NOT_FOCUSABLE) != 0)
   2837                     || (!win.mRelayoutCalled));
   2838 
   2839             boolean wallpaperMayMove = win.mViewVisibility != viewVisibility
   2840                     && (win.mAttrs.flags & FLAG_SHOW_WALLPAPER) != 0;
   2841             wallpaperMayMove |= (flagChanges & FLAG_SHOW_WALLPAPER) != 0;
   2842             if ((flagChanges & FLAG_SECURE) != 0 && winAnimator.mSurfaceController != null) {
   2843                 winAnimator.mSurfaceController.setSecure(isSecureLocked(win));
   2844             }
   2845 
   2846             win.mRelayoutCalled = true;
   2847             win.mInRelayout = true;
   2848 
   2849             final int oldVisibility = win.mViewVisibility;
   2850             win.mViewVisibility = viewVisibility;
   2851             if (DEBUG_SCREEN_ON) {
   2852                 RuntimeException stack = new RuntimeException();
   2853                 stack.fillInStackTrace();
   2854                 Slog.i(TAG_WM, "Relayout " + win + ": oldVis=" + oldVisibility
   2855                         + " newVis=" + viewVisibility, stack);
   2856             }
   2857             if (viewVisibility == View.VISIBLE &&
   2858                     (win.mAppToken == null || !win.mAppToken.clientHidden)) {
   2859                 result = relayoutVisibleWindow(outConfig, result, win, winAnimator, attrChanges,
   2860                         oldVisibility);
   2861                 try {
   2862                     result = createSurfaceControl(outSurface, result, win, winAnimator);
   2863                 } catch (Exception e) {
   2864                     mInputMonitor.updateInputWindowsLw(true /*force*/);
   2865 
   2866                     Slog.w(TAG_WM, "Exception thrown when creating surface for client "
   2867                              + client + " (" + win.mAttrs.getTitle() + ")",
   2868                              e);
   2869                     Binder.restoreCallingIdentity(origId);
   2870                     return 0;
   2871                 }
   2872                 if ((result & WindowManagerGlobal.RELAYOUT_RES_FIRST_TIME) != 0) {
   2873                     focusMayChange = isDefaultDisplay;
   2874                 }
   2875                 if (win.mAttrs.type == TYPE_INPUT_METHOD && mInputMethodWindow == null) {
   2876                     mInputMethodWindow = win;
   2877                     imMayMove = true;
   2878                 }
   2879                 win.adjustStartingWindowFlags();
   2880             } else {
   2881                 winAnimator.mEnterAnimationPending = false;
   2882                 winAnimator.mEnteringAnimation = false;
   2883                 final boolean usingSavedSurfaceBeforeVisible =
   2884                         oldVisibility != View.VISIBLE && win.isAnimatingWithSavedSurface();
   2885                 if (DEBUG_APP_TRANSITIONS || DEBUG_ANIM) {
   2886                     if (winAnimator.hasSurface() && !win.mAnimatingExit
   2887                             && usingSavedSurfaceBeforeVisible) {
   2888                         Slog.d(TAG, "Ignoring layout to invisible when using saved surface " + win);
   2889                     }
   2890                 }
   2891 
   2892                 if (winAnimator.hasSurface() && !win.mAnimatingExit
   2893                         && !usingSavedSurfaceBeforeVisible) {
   2894                     if (DEBUG_VISIBILITY) Slog.i(TAG_WM, "Relayout invis " + win
   2895                             + ": mAnimatingExit=" + win.mAnimatingExit);
   2896                     // If we are not currently running the exit animation, we
   2897                     // need to see about starting one.
   2898                     // We don't want to animate visibility of windows which are pending
   2899                     // replacement. In the case of activity relaunch child windows
   2900                     // could request visibility changes as they are detached from the main
   2901                     // application window during the tear down process. If we satisfied
   2902                     // these visibility changes though, we would cause a visual glitch
   2903                     // hiding the window before it's replacement was available.
   2904                     // So we just do nothing on our side.
   2905                     if (!win.mWillReplaceWindow) {
   2906                         focusMayChange = tryStartExitingAnimation(
   2907                                 win, winAnimator, isDefaultDisplay, focusMayChange);
   2908                     }
   2909                     result |= RELAYOUT_RES_SURFACE_CHANGED;
   2910                 }
   2911                 final WindowSurfaceController surfaceController = winAnimator.mSurfaceController;
   2912                 if (viewVisibility == View.VISIBLE && surfaceController != null) {
   2913                     // We already told the client to go invisible, but the message may not be
   2914                     // handled yet, or it might want to draw a last frame. If we already have a
   2915                     // surface, let the client use that, but don't create new surface at this point.
   2916                     surfaceController.getSurface(outSurface);
   2917                 } else {
   2918                     if (DEBUG_VISIBILITY) Slog.i(TAG_WM, "Releasing surface in: " + win);
   2919 
   2920                     try {
   2921                         Trace.traceBegin(Trace.TRACE_TAG_WINDOW_MANAGER, "wmReleaseOutSurface_"
   2922                                 + win.mAttrs.getTitle());
   2923                         outSurface.release();
   2924                     } finally {
   2925                         Trace.traceEnd(Trace.TRACE_TAG_WINDOW_MANAGER);
   2926                     }
   2927                 }
   2928             }
   2929 
   2930             if (focusMayChange) {
   2931                 //System.out.println("Focus may change: " + win.mAttrs.getTitle());
   2932                 if (updateFocusedWindowLocked(UPDATE_FOCUS_WILL_PLACE_SURFACES,
   2933                         false /*updateInputWindows*/)) {
   2934                     imMayMove = false;
   2935                 }
   2936                 //System.out.println("Relayout " + win + ": focus=" + mCurrentFocus);
   2937             }
   2938 
   2939             // updateFocusedWindowLocked() already assigned layers so we only need to
   2940             // reassign them at this point if the IM window state gets shuffled
   2941             boolean toBeDisplayed = (result & WindowManagerGlobal.RELAYOUT_RES_FIRST_TIME) != 0;
   2942             if (imMayMove && (moveInputMethodWindowsIfNeededLocked(false) || toBeDisplayed)) {
   2943                 // Little hack here -- we -should- be able to rely on the
   2944                 // function to return true if the IME has moved and needs
   2945                 // its layer recomputed.  However, if the IME was hidden
   2946                 // and isn't actually moved in the list, its layer may be
   2947                 // out of data so we make sure to recompute it.
   2948                 mLayersController.assignLayersLocked(win.getWindowList());
   2949             }
   2950 
   2951             if (wallpaperMayMove) {
   2952                 getDefaultDisplayContentLocked().pendingLayoutChanges |=
   2953                         WindowManagerPolicy.FINISH_LAYOUT_REDO_WALLPAPER;
   2954             }
   2955 
   2956             win.setDisplayLayoutNeeded();
   2957             win.mGivenInsetsPending = (flags&WindowManagerGlobal.RELAYOUT_INSETS_PENDING) != 0;
   2958             configChanged = updateOrientationFromAppTokensLocked(false);
   2959             mWindowPlacerLocked.performSurfacePlacement();
   2960             if (toBeDisplayed && win.mIsWallpaper) {
   2961                 DisplayInfo displayInfo = getDefaultDisplayInfoLocked();
   2962                 mWallpaperControllerLocked.updateWallpaperOffset(
   2963                         win, displayInfo.logicalWidth, displayInfo.logicalHeight, false);
   2964             }
   2965             if (win.mAppToken != null) {
   2966                 win.mAppToken.updateReportedVisibilityLocked();
   2967             }
   2968             if (winAnimator.mReportSurfaceResized) {
   2969                 winAnimator.mReportSurfaceResized = false;
   2970                 result |= WindowManagerGlobal.RELAYOUT_RES_SURFACE_RESIZED;
   2971             }
   2972             if (mPolicy.isNavBarForcedShownLw(win)) {
   2973                 result |= WindowManagerGlobal.RELAYOUT_RES_CONSUME_ALWAYS_NAV_BAR;
   2974             }
   2975             if (!win.isGoneForLayoutLw()) {
   2976                 win.mResizedWhileGone = false;
   2977             }
   2978             outFrame.set(win.mCompatFrame);
   2979             outOverscanInsets.set(win.mOverscanInsets);
   2980             outContentInsets.set(win.mContentInsets);
   2981             outVisibleInsets.set(win.mVisibleInsets);
   2982             outStableInsets.set(win.mStableInsets);
   2983             outOutsets.set(win.mOutsets);
   2984             outBackdropFrame.set(win.getBackdropFrame(win.mFrame));
   2985             if (localLOGV) Slog.v(
   2986                 TAG_WM, "Relayout given client " + client.asBinder()
   2987                 + ", requestedWidth=" + requestedWidth
   2988                 + ", requestedHeight=" + requestedHeight
   2989                 + ", viewVisibility=" + viewVisibility
   2990                 + "\nRelayout returning frame=" + outFrame
   2991                 + ", surface=" + outSurface);
   2992 
   2993             if (localLOGV || DEBUG_FOCUS) Slog.v(
   2994                 TAG_WM, "Relayout of " + win + ": focusMayChange=" + focusMayChange);
   2995 
   2996             result |= mInTouchMode ? WindowManagerGlobal.RELAYOUT_RES_IN_TOUCH_MODE : 0;
   2997 
   2998             mInputMonitor.updateInputWindowsLw(true /*force*/);
   2999 
   3000             if (DEBUG_LAYOUT) {
   3001                 Slog.v(TAG_WM, "Relayout complete " + win + ": outFrame=" + outFrame.toShortString());
   3002             }
   3003             win.mInRelayout = false;
   3004         }
   3005 
   3006         if (configChanged) {
   3007             sendNewConfiguration();
   3008         }
   3009         Binder.restoreCallingIdentity(origId);
   3010         return result;
   3011     }
   3012 
   3013     private boolean tryStartExitingAnimation(WindowState win, WindowStateAnimator winAnimator,
   3014             boolean isDefaultDisplay, boolean focusMayChange) {
   3015         // Try starting an animation; if there isn't one, we
   3016         // can destroy the surface right away.
   3017         int transit = WindowManagerPolicy.TRANSIT_EXIT;
   3018         if (win.mAttrs.type == TYPE_APPLICATION_STARTING) {
   3019             transit = WindowManagerPolicy.TRANSIT_PREVIEW_DONE;
   3020         }
   3021         if (win.isWinVisibleLw() && winAnimator.applyAnimationLocked(transit, false)) {
   3022             focusMayChange = isDefaultDisplay;
   3023             win.mAnimatingExit = true;
   3024             win.mWinAnimator.mAnimating = true;
   3025         } else if (win.mWinAnimator.isAnimationSet()) {
   3026             // Currently in a hide animation... turn this into
   3027             // an exit.
   3028             win.mAnimatingExit = true;
   3029             win.mWinAnimator.mAnimating = true;
   3030         } else if (mWallpaperControllerLocked.isWallpaperTarget(win)) {
   3031             // If the wallpaper is currently behind this
   3032             // window, we need to change both of them inside
   3033             // of a transaction to avoid artifacts.
   3034             win.mAnimatingExit = true;
   3035             win.mWinAnimator.mAnimating = true;
   3036         } else {
   3037             if (mInputMethodWindow == win) {
   3038                 mInputMethodWindow = null;
   3039             }
   3040             win.destroyOrSaveSurface();
   3041         }
   3042         //TODO (multidisplay): Magnification is supported only for the default
   3043         if (mAccessibilityController != null
   3044                 && win.getDisplayId() == Display.DEFAULT_DISPLAY) {
   3045             mAccessibilityController.onWindowTransitionLocked(win, transit);
   3046         }
   3047         return focusMayChange;
   3048     }
   3049 
   3050     private int createSurfaceControl(Surface outSurface, int result, WindowState win,
   3051             WindowStateAnimator winAnimator) {
   3052         if (!win.mHasSurface) {
   3053             result |= RELAYOUT_RES_SURFACE_CHANGED;
   3054         }
   3055         WindowSurfaceController surfaceController = winAnimator.createSurfaceLocked();
   3056         if (surfaceController != null) {
   3057             surfaceController.getSurface(outSurface);
   3058             if (SHOW_TRANSACTIONS) Slog.i(TAG_WM, "  OUT SURFACE " + outSurface + ": copied");
   3059         } else {
   3060             // For some reason there isn't a surface.  Clear the
   3061             // caller's object so they see the same state.
   3062             Slog.w(TAG_WM, "Failed to create surface control for " + win);
   3063             outSurface.release();
   3064         }
   3065         return result;
   3066     }
   3067 
   3068     private int relayoutVisibleWindow(Configuration outConfig, int result, WindowState win,
   3069             WindowStateAnimator winAnimator, int attrChanges, int oldVisibility) {
   3070         result |= !win.isVisibleLw() ? WindowManagerGlobal.RELAYOUT_RES_FIRST_TIME : 0;
   3071         if (win.mAnimatingExit) {
   3072             Slog.d(TAG, "relayoutVisibleWindow: " + win + " mAnimatingExit=true, mRemoveOnExit="
   3073                     + win.mRemoveOnExit + ", mDestroying=" + win.mDestroying);
   3074 
   3075             winAnimator.cancelExitAnimationForNextAnimationLocked();
   3076             win.mAnimatingExit = false;
   3077         }
   3078         if (win.mDestroying) {
   3079             win.mDestroying = false;
   3080             mDestroySurface.remove(win);
   3081         }
   3082         if (oldVisibility == View.GONE) {
   3083             winAnimator.mEnterAnimationPending = true;
   3084         }
   3085 
   3086         win.mLastVisibleLayoutRotation = mRotation;
   3087 
   3088         winAnimator.mEnteringAnimation = true;
   3089         if ((result & WindowManagerGlobal.RELAYOUT_RES_FIRST_TIME) != 0) {
   3090             win.prepareWindowToDisplayDuringRelayout(outConfig);
   3091         }
   3092         if ((attrChanges & LayoutParams.FORMAT_CHANGED) != 0) {
   3093             // If the format can't be changed in place, preserve the old surface until the app draws
   3094             // on the new one. This prevents blinking when we change elevation of freeform and
   3095             // pinned windows.
   3096             if (!winAnimator.tryChangeFormatInPlaceLocked()) {
   3097                 winAnimator.preserveSurfaceLocked();
   3098                 result |= RELAYOUT_RES_SURFACE_CHANGED
   3099                         | WindowManagerGlobal.RELAYOUT_RES_FIRST_TIME;
   3100             }
   3101         }
   3102 
   3103         // If we're starting a drag-resize, we'll be changing the surface size as well as
   3104         // notifying the client to render to with an offset from the surface's top-left.
   3105         if (win.isDragResizeChanged() || win.isResizedWhileNotDragResizing()) {
   3106             win.setDragResizing();
   3107             win.setResizedWhileNotDragResizing(false);
   3108             // We can only change top level windows to the full-screen surface when
   3109             // resizing (as we only have one full-screen surface). So there is no need
   3110             // to preserve and destroy windows which are attached to another, they
   3111             // will keep their surface and its size may change over time.
   3112             if (win.mHasSurface && win.mAttachedWindow == null) {
   3113                 winAnimator.preserveSurfaceLocked();
   3114                 result |= WindowManagerGlobal.RELAYOUT_RES_FIRST_TIME;
   3115             }
   3116         }
   3117         final boolean freeformResizing = win.isDragResizing()
   3118                 && win.getResizeMode() == DRAG_RESIZE_MODE_FREEFORM;
   3119         final boolean dockedResizing = win.isDragResizing()
   3120                 && win.getResizeMode() == DRAG_RESIZE_MODE_DOCKED_DIVIDER;
   3121         result |= freeformResizing ? WindowManagerGlobal.RELAYOUT_RES_DRAG_RESIZING_FREEFORM : 0;
   3122         result |= dockedResizing ? WindowManagerGlobal.RELAYOUT_RES_DRAG_RESIZING_DOCKED : 0;
   3123         if (win.isAnimatingWithSavedSurface()) {
   3124             // If we're animating with a saved surface now, request client to report draw.
   3125             // We still need to know when the real thing is drawn.
   3126             result |= WindowManagerGlobal.RELAYOUT_RES_FIRST_TIME;
   3127         }
   3128         return result;
   3129     }
   3130 
   3131     public void performDeferredDestroyWindow(Session session, IWindow client) {
   3132         long origId = Binder.clearCallingIdentity();
   3133 
   3134         try {
   3135             synchronized (mWindowMap) {
   3136                 WindowState win = windowForClientLocked(session, client, false);
   3137                 if (win == null || win.mWillReplaceWindow) {
   3138                     return;
   3139                 }
   3140 
   3141                 win.mWinAnimator.destroyDeferredSurfaceLocked();
   3142             }
   3143         } finally {
   3144             Binder.restoreCallingIdentity(origId);
   3145         }
   3146     }
   3147 
   3148     public boolean outOfMemoryWindow(Session session, IWindow client) {
   3149         long origId = Binder.clearCallingIdentity();
   3150 
   3151         try {
   3152             synchronized (mWindowMap) {
   3153                 WindowState win = windowForClientLocked(session, client, false);
   3154                 if (win == null) {
   3155                     return false;
   3156                 }
   3157                 return reclaimSomeSurfaceMemoryLocked(win.mWinAnimator, "from-client", false);
   3158             }
   3159         } finally {
   3160             Binder.restoreCallingIdentity(origId);
   3161         }
   3162     }
   3163 
   3164     public void finishDrawingWindow(Session session, IWindow client) {
   3165         final long origId = Binder.clearCallingIdentity();
   3166         try {
   3167             synchronized (mWindowMap) {
   3168                 WindowState win = windowForClientLocked(session, client, false);
   3169                 if (DEBUG_ADD_REMOVE) Slog.d(TAG_WM, "finishDrawingWindow: " + win + " mDrawState="
   3170                         + (win != null ? win.mWinAnimator.drawStateToString() : "null"));
   3171                 if (win != null && win.mWinAnimator.finishDrawingLocked()) {
   3172                     if ((win.mAttrs.flags & FLAG_SHOW_WALLPAPER) != 0) {
   3173                         getDefaultDisplayContentLocked().pendingLayoutChanges |=
   3174                                 WindowManagerPolicy.FINISH_LAYOUT_REDO_WALLPAPER;
   3175                     }
   3176                     win.setDisplayLayoutNeeded();
   3177                     mWindowPlacerLocked.requestTraversal();
   3178                 }
   3179             }
   3180         } finally {
   3181             Binder.restoreCallingIdentity(origId);
   3182         }
   3183     }
   3184 
   3185     private boolean applyAnimationLocked(AppWindowToken atoken, WindowManager.LayoutParams lp,
   3186             int transit, boolean enter, boolean isVoiceInteraction) {
   3187         // Only apply an animation if the display isn't frozen.  If it is
   3188         // frozen, there is no reason to animate and it can cause strange
   3189         // artifacts when we unfreeze the display if some different animation
   3190         // is running.
   3191         Trace.traceBegin(Trace.TRACE_TAG_WINDOW_MANAGER, "WM#applyAnimationLocked");
   3192         if (okToDisplay()) {
   3193             DisplayInfo displayInfo = getDefaultDisplayInfoLocked();
   3194             final int width = displayInfo.appWidth;
   3195             final int height = displayInfo.appHeight;
   3196             if (DEBUG_APP_TRANSITIONS || DEBUG_ANIM) Slog.v(TAG_WM,
   3197                     "applyAnimation: atoken=" + atoken);
   3198 
   3199             // Determine the visible rect to calculate the thumbnail clip
   3200             final WindowState win = atoken.findMainWindow();
   3201             final Rect frame = new Rect(0, 0, width, height);
   3202             final Rect displayFrame = new Rect(0, 0,
   3203                     displayInfo.logicalWidth, displayInfo.logicalHeight);
   3204             final Rect insets = new Rect();
   3205             Rect surfaceInsets = null;
   3206             final boolean freeform = win != null && win.inFreeformWorkspace();
   3207             if (win != null) {
   3208                 // Containing frame will usually cover the whole screen, including dialog windows.
   3209                 // For freeform workspace windows it will not cover the whole screen and it also
   3210                 // won't exactly match the final freeform window frame (e.g. when overlapping with
   3211                 // the status bar). In that case we need to use the final frame.
   3212                 if (freeform) {
   3213                     frame.set(win.mFrame);
   3214                 } else {
   3215                     frame.set(win.mContainingFrame);
   3216                 }
   3217                 surfaceInsets = win.getAttrs().surfaceInsets;
   3218                 insets.set(win.mContentInsets);
   3219             }
   3220 
   3221             if (atoken.mLaunchTaskBehind) {
   3222                 // Differentiate the two animations. This one which is briefly on the screen
   3223                 // gets the !enter animation, and the other activity which remains on the
   3224                 // screen gets the enter animation. Both appear in the mOpeningApps set.
   3225                 enter = false;
   3226             }
   3227             if (DEBUG_APP_TRANSITIONS) Slog.d(TAG_WM, "Loading animation for app transition."
   3228                     + " transit=" + AppTransition.appTransitionToString(transit) + " enter=" + enter
   3229                     + " frame=" + frame + " insets=" + insets + " surfaceInsets=" + surfaceInsets);
   3230             Animation a = mAppTransition.loadAnimation(lp, transit, enter, mCurConfiguration.uiMode,
   3231                     mCurConfiguration.orientation, frame, displayFrame, insets, surfaceInsets,
   3232                     isVoiceInteraction, freeform, atoken.mTask.mTaskId);
   3233             if (a != null) {
   3234                 if (DEBUG_ANIM) logWithStack(TAG, "Loaded animation " + a + " for " + atoken);
   3235                 final int containingWidth = frame.width();
   3236                 final int containingHeight = frame.height();
   3237                 atoken.mAppAnimator.setAnimation(a, containingWidth, containingHeight,
   3238                         mAppTransition.canSkipFirstFrame(), mAppTransition.getAppStackClipMode());
   3239             }
   3240         } else {
   3241             atoken.mAppAnimator.clearAnimation();
   3242         }
   3243         Trace.traceEnd(Trace.TRACE_TAG_WINDOW_MANAGER);
   3244 
   3245         return atoken.mAppAnimator.animation != null;
   3246     }
   3247 
   3248     // -------------------------------------------------------------
   3249     // Application Window Tokens
   3250     // -------------------------------------------------------------
   3251 
   3252     public void validateAppTokens(int stackId, List<TaskGroup> tasks) {
   3253         synchronized (mWindowMap) {
   3254             int t = tasks.size() - 1;
   3255             if (t < 0) {
   3256                 Slog.w(TAG_WM, "validateAppTokens: empty task list");
   3257                 return;
   3258             }
   3259 
   3260             TaskGroup task = tasks.get(0);
   3261             int taskId = task.taskId;
   3262             Task targetTask = mTaskIdToTask.get(taskId);
   3263             DisplayContent displayContent = targetTask.getDisplayContent();
   3264             if (displayContent == null) {
   3265                 Slog.w(TAG_WM, "validateAppTokens: no Display for taskId=" + taskId);
   3266                 return;
   3267             }
   3268 
   3269             final ArrayList<Task> localTasks = mStackIdToStack.get(stackId).getTasks();
   3270             int taskNdx;
   3271             for (taskNdx = localTasks.size() - 1; taskNdx >= 0 && t >= 0; --taskNdx, --t) {
   3272                 AppTokenList localTokens = localTasks.get(taskNdx).mAppTokens;
   3273                 task = tasks.get(t);
   3274                 List<IApplicationToken> tokens = task.tokens;
   3275 
   3276                 DisplayContent lastDisplayContent = displayContent;
   3277                 displayContent = mTaskIdToTask.get(taskId).getDisplayContent();
   3278                 if (displayContent != lastDisplayContent) {
   3279                     Slog.w(TAG_WM, "validateAppTokens: displayContent changed in TaskGroup list!");
   3280                     return;
   3281                 }
   3282 
   3283                 int tokenNdx;
   3284                 int v;
   3285                 for (tokenNdx = localTokens.size() - 1, v = task.tokens.size() - 1;
   3286                         tokenNdx >= 0 && v >= 0; ) {
   3287                     final AppWindowToken atoken = localTokens.get(tokenNdx);
   3288                     if (atoken.removed) {
   3289                         --tokenNdx;
   3290                         continue;
   3291                     }
   3292                     if (tokens.get(v) != atoken.token) {
   3293                         break;
   3294                     }
   3295                     --tokenNdx;
   3296                     v--;
   3297                 }
   3298 
   3299                 if (tokenNdx >= 0 || v >= 0) {
   3300                     break;
   3301                 }
   3302             }
   3303 
   3304             if (taskNdx >= 0 || t >= 0) {
   3305                 Slog.w(TAG_WM, "validateAppTokens: Mismatch! ActivityManager=" + tasks);
   3306                 Slog.w(TAG_WM, "validateAppTokens: Mismatch! WindowManager=" + localTasks);
   3307                 Slog.w(TAG_WM, "validateAppTokens: Mismatch! Callers=" + Debug.getCallers(4));
   3308             }
   3309         }
   3310     }
   3311 
   3312     public void validateStackOrder(Integer[] remoteStackIds) {
   3313         // TODO:
   3314     }
   3315 
   3316     private boolean checkCallingPermission(String permission, String func) {
   3317         // Quick check: if the calling permission is me, it's all okay.
   3318         if (Binder.getCallingPid() == Process.myPid()) {
   3319             return true;
   3320         }
   3321 
   3322         if (mContext.checkCallingPermission(permission)
   3323                 == PackageManager.PERMISSION_GRANTED) {
   3324             return true;
   3325         }
   3326         String msg = "Permission Denial: " + func + " from pid="
   3327                 + Binder.getCallingPid()
   3328                 + ", uid=" + Binder.getCallingUid()
   3329                 + " requires " + permission;
   3330         Slog.w(TAG_WM, msg);
   3331         return false;
   3332     }
   3333 
   3334     boolean okToDisplay() {
   3335         return !mDisplayFrozen && mDisplayEnabled && mPolicy.isScreenOn();
   3336     }
   3337 
   3338     AppWindowToken findAppWindowToken(IBinder token) {
   3339         WindowToken wtoken = mTokenMap.get(token);
   3340         if (wtoken == null) {
   3341             return null;
   3342         }
   3343         return wtoken.appWindowToken;
   3344     }
   3345 
   3346     @Override
   3347     public void addWindowToken(IBinder token, int type) {
   3348         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   3349                 "addWindowToken()")) {
   3350             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   3351         }
   3352 
   3353         synchronized(mWindowMap) {
   3354             WindowToken wtoken = mTokenMap.get(token);
   3355             if (wtoken != null) {
   3356                 Slog.w(TAG_WM, "Attempted to add existing input method token: " + token);
   3357                 return;
   3358             }
   3359             wtoken = new WindowToken(this, token, type, true);
   3360             mTokenMap.put(token, wtoken);
   3361             if (type == TYPE_WALLPAPER) {
   3362                 mWallpaperControllerLocked.addWallpaperToken(wtoken);
   3363             }
   3364         }
   3365     }
   3366 
   3367     @Override
   3368     public void removeWindowToken(IBinder token) {
   3369         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   3370                 "removeWindowToken()")) {
   3371             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   3372         }
   3373 
   3374         final long origId = Binder.clearCallingIdentity();
   3375         synchronized(mWindowMap) {
   3376             DisplayContent displayContent = null;
   3377             WindowToken wtoken = mTokenMap.remove(token);
   3378             if (wtoken != null) {
   3379                 boolean delayed = false;
   3380                 if (!wtoken.hidden) {
   3381                     final int N = wtoken.windows.size();
   3382                     boolean changed = false;
   3383 
   3384                     for (int i=0; i<N; i++) {
   3385                         WindowState win = wtoken.windows.get(i);
   3386                         displayContent = win.getDisplayContent();
   3387 
   3388                         if (win.mWinAnimator.isAnimationSet()) {
   3389                             delayed = true;
   3390                         }
   3391 
   3392                         if (win.isVisibleNow()) {
   3393                             win.mWinAnimator.applyAnimationLocked(WindowManagerPolicy.TRANSIT_EXIT,
   3394                                     false);
   3395                             //TODO (multidisplay): Magnification is supported only for the default
   3396                             if (mAccessibilityController != null && win.isDefaultDisplay()) {
   3397                                 mAccessibilityController.onWindowTransitionLocked(win,
   3398                                         WindowManagerPolicy.TRANSIT_EXIT);
   3399                             }
   3400                             changed = true;
   3401                             if (displayContent != null) {
   3402                                 displayContent.layoutNeeded = true;
   3403                             }
   3404                         }
   3405                     }
   3406 
   3407                     wtoken.hidden = true;
   3408 
   3409                     if (changed) {
   3410                         mWindowPlacerLocked.performSurfacePlacement();
   3411                         updateFocusedWindowLocked(UPDATE_FOCUS_NORMAL,
   3412                                 false /*updateInputWindows*/);
   3413                     }
   3414 
   3415                     if (delayed && displayContent != null) {
   3416                         displayContent.mExitingTokens.add(wtoken);
   3417                     } else if (wtoken.windowType == TYPE_WALLPAPER) {
   3418                         mWallpaperControllerLocked.removeWallpaperToken(wtoken);
   3419                     }
   3420                 } else if (wtoken.windowType == TYPE_WALLPAPER) {
   3421                     mWallpaperControllerLocked.removeWallpaperToken(wtoken);
   3422                 }
   3423 
   3424                 mInputMonitor.updateInputWindowsLw(true /*force*/);
   3425             } else {
   3426                 Slog.w(TAG_WM, "Attempted to remove non-existing token: " + token);
   3427             }
   3428         }
   3429         Binder.restoreCallingIdentity(origId);
   3430     }
   3431 
   3432     private Task createTaskLocked(int taskId, int stackId, int userId, AppWindowToken atoken,
   3433             Rect bounds, Configuration config) {
   3434         if (DEBUG_STACK) Slog.i(TAG_WM, "createTaskLocked: taskId=" + taskId + " stackId=" + stackId
   3435                 + " atoken=" + atoken + " bounds=" + bounds);
   3436         final TaskStack stack = mStackIdToStack.get(stackId);
   3437         if (stack == null) {
   3438             throw new IllegalArgumentException("addAppToken: invalid stackId=" + stackId);
   3439         }
   3440         EventLog.writeEvent(EventLogTags.WM_TASK_CREATED, taskId, stackId);
   3441         Task task = new Task(taskId, stack, userId, this, bounds, config);
   3442         mTaskIdToTask.put(taskId, task);
   3443         stack.addTask(task, !atoken.mLaunchTaskBehind /* toTop */, atoken.showForAllUsers);
   3444         return task;
   3445     }
   3446 
   3447     @Override
   3448     public void addAppToken(int addPos, IApplicationToken token, int taskId, int stackId,
   3449             int requestedOrientation, boolean fullscreen, boolean showForAllUsers, int userId,
   3450             int configChanges, boolean voiceInteraction, boolean launchTaskBehind,
   3451             Rect taskBounds, Configuration config, int taskResizeMode, boolean alwaysFocusable,
   3452             boolean homeTask, int targetSdkVersion, int rotationAnimationHint) {
   3453         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   3454                 "addAppToken()")) {
   3455             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   3456         }
   3457 
   3458         // Get the dispatching timeout here while we are not holding any locks so that it
   3459         // can be cached by the AppWindowToken.  The timeout value is used later by the
   3460         // input dispatcher in code that does hold locks.  If we did not cache the value
   3461         // here we would run the chance of introducing a deadlock between the window manager
   3462         // (which holds locks while updating the input dispatcher state) and the activity manager
   3463         // (which holds locks while querying the application token).
   3464         long inputDispatchingTimeoutNanos;
   3465         try {
   3466             inputDispatchingTimeoutNanos = token.getKeyDispatchingTimeout() * 1000000L;
   3467         } catch (RemoteException ex) {
   3468             Slog.w(TAG_WM, "Could not get dispatching timeout.", ex);
   3469             inputDispatchingTimeoutNanos = DEFAULT_INPUT_DISPATCHING_TIMEOUT_NANOS;
   3470         }
   3471 
   3472         synchronized(mWindowMap) {
   3473             AppWindowToken atoken = findAppWindowToken(token.asBinder());
   3474             if (atoken != null) {
   3475                 Slog.w(TAG_WM, "Attempted to add existing app token: " + token);
   3476                 return;
   3477             }
   3478             atoken = new AppWindowToken(this, token, voiceInteraction);
   3479             atoken.inputDispatchingTimeoutNanos = inputDispatchingTimeoutNanos;
   3480             atoken.appFullscreen = fullscreen;
   3481             atoken.showForAllUsers = showForAllUsers;
   3482             atoken.targetSdk = targetSdkVersion;
   3483             atoken.requestedOrientation = requestedOrientation;
   3484             atoken.layoutConfigChanges = (configChanges &
   3485                     (ActivityInfo.CONFIG_SCREEN_SIZE | ActivityInfo.CONFIG_ORIENTATION)) != 0;
   3486             atoken.mLaunchTaskBehind = launchTaskBehind;
   3487             atoken.mAlwaysFocusable = alwaysFocusable;
   3488             if (DEBUG_TOKEN_MOVEMENT || DEBUG_ADD_REMOVE) Slog.v(TAG_WM, "addAppToken: " + atoken
   3489                     + " to stack=" + stackId + " task=" + taskId + " at " + addPos);
   3490             atoken.mRotationAnimationHint = rotationAnimationHint;
   3491 
   3492             Task task = mTaskIdToTask.get(taskId);
   3493             if (task == null) {
   3494                 task = createTaskLocked(taskId, stackId, userId, atoken, taskBounds, config);
   3495             }
   3496             task.addAppToken(addPos, atoken, taskResizeMode, homeTask);
   3497 
   3498             mTokenMap.put(token.asBinder(), atoken);
   3499 
   3500             // Application tokens start out hidden.
   3501             atoken.hidden = true;
   3502             atoken.hiddenRequested = true;
   3503         }
   3504     }
   3505 
   3506     @Override
   3507     public void setAppTask(IBinder token, int taskId, int stackId, Rect taskBounds,
   3508             Configuration config, int taskResizeMode, boolean homeTask) {
   3509         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   3510                 "setAppTask()")) {
   3511             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   3512         }
   3513 
   3514         synchronized(mWindowMap) {
   3515             final AppWindowToken atoken = findAppWindowToken(token);
   3516             if (atoken == null) {
   3517                 Slog.w(TAG_WM, "Attempted to set task id of non-existing app token: " + token);
   3518                 return;
   3519             }
   3520             final Task oldTask = atoken.mTask;
   3521             oldTask.removeAppToken(atoken);
   3522 
   3523             Task newTask = mTaskIdToTask.get(taskId);
   3524             if (newTask == null) {
   3525                 newTask = createTaskLocked(
   3526                         taskId, stackId, oldTask.mUserId, atoken, taskBounds, config);
   3527             }
   3528             newTask.addAppToken(Integer.MAX_VALUE /* at top */, atoken, taskResizeMode, homeTask);
   3529         }
   3530     }
   3531 
   3532     public int getOrientationLocked() {
   3533         if (mDisplayFrozen) {
   3534             if (mLastWindowForcedOrientation != SCREEN_ORIENTATION_UNSPECIFIED) {
   3535                 if (DEBUG_ORIENTATION) Slog.v(TAG_WM,
   3536                         "Display is frozen, return " + mLastWindowForcedOrientation);
   3537                 // If the display is frozen, some activities may be in the middle
   3538                 // of restarting, and thus have removed their old window.  If the
   3539                 // window has the flag to hide the lock screen, then the lock screen
   3540                 // can re-appear and inflict its own orientation on us.  Keep the
   3541                 // orientation stable until this all settles down.
   3542                 return mLastWindowForcedOrientation;
   3543             } else if (mPolicy.isKeyguardLocked()) {
   3544                 // Use the last orientation the while the display is frozen with the
   3545                 // keyguard locked. This could be the keyguard forced orientation or
   3546                 // from a SHOW_WHEN_LOCKED window. We don't want to check the show when
   3547                 // locked window directly though as things aren't stable while
   3548                 // the display is frozen, for example the window could be momentarily unavailable
   3549                 // due to activity relaunch.
   3550                 if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Display is frozen while keyguard locked, "
   3551                         + "return " + mLastOrientation);
   3552                 return mLastOrientation;
   3553             }
   3554         } else {
   3555             // TODO(multidisplay): Change to the correct display.
   3556             final WindowList windows = getDefaultWindowListLocked();
   3557             for (int pos = windows.size() - 1; pos >= 0; --pos) {
   3558                 WindowState win = windows.get(pos);
   3559                 if (win.mAppToken != null) {
   3560                     // We hit an application window. so the orientation will be determined by the
   3561                     // app window. No point in continuing further.
   3562                     break;
   3563                 }
   3564                 if (!win.isVisibleLw() || !win.mPolicyVisibilityAfterAnim) {
   3565                     continue;
   3566                 }
   3567                 int req = win.mAttrs.screenOrientation;
   3568                 if(req == SCREEN_ORIENTATION_UNSPECIFIED || req == SCREEN_ORIENTATION_BEHIND) {
   3569                     continue;
   3570                 }
   3571 
   3572                 if (DEBUG_ORIENTATION) Slog.v(TAG_WM, win + " forcing orientation to " + req);
   3573                 if (mPolicy.isKeyguardHostWindow(win.mAttrs)) {
   3574                     mLastKeyguardForcedOrientation = req;
   3575                 }
   3576                 return (mLastWindowForcedOrientation = req);
   3577             }
   3578             mLastWindowForcedOrientation = SCREEN_ORIENTATION_UNSPECIFIED;
   3579 
   3580             if (mPolicy.isKeyguardLocked()) {
   3581                 // The screen is locked and no top system window is requesting an orientation.
   3582                 // Return either the orientation of the show-when-locked app (if there is any) or
   3583                 // the orientation of the keyguard. No point in searching from the rest of apps.
   3584                 WindowState winShowWhenLocked = (WindowState) mPolicy.getWinShowWhenLockedLw();
   3585                 AppWindowToken appShowWhenLocked = winShowWhenLocked == null ?
   3586                         null : winShowWhenLocked.mAppToken;
   3587                 if (appShowWhenLocked != null) {
   3588                     int req = appShowWhenLocked.requestedOrientation;
   3589                     if (req == SCREEN_ORIENTATION_BEHIND) {
   3590                         req = mLastKeyguardForcedOrientation;
   3591                     }
   3592                     if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Done at " + appShowWhenLocked
   3593                             + " -- show when locked, return " + req);
   3594                     return req;
   3595                 }
   3596                 if (DEBUG_ORIENTATION) Slog.v(TAG_WM,
   3597                         "No one is requesting an orientation when the screen is locked");
   3598                 return mLastKeyguardForcedOrientation;
   3599             }
   3600         }
   3601 
   3602         // Top system windows are not requesting an orientation. Start searching from apps.
   3603         return getAppSpecifiedOrientation();
   3604     }
   3605 
   3606     private int getAppSpecifiedOrientation() {
   3607         int lastOrientation = SCREEN_ORIENTATION_UNSPECIFIED;
   3608         boolean findingBehind = false;
   3609         boolean lastFullscreen = false;
   3610         DisplayContent displayContent = getDefaultDisplayContentLocked();
   3611         final ArrayList<Task> tasks = displayContent.getTasks();
   3612         final boolean inMultiWindow = isStackVisibleLocked(DOCKED_STACK_ID)
   3613                 || isStackVisibleLocked(FREEFORM_WORKSPACE_STACK_ID);
   3614         final boolean dockMinimized =
   3615                 getDefaultDisplayContentLocked().mDividerControllerLocked.isMinimizedDock();
   3616         for (int taskNdx = tasks.size() - 1; taskNdx >= 0; --taskNdx) {
   3617             AppTokenList tokens = tasks.get(taskNdx).mAppTokens;
   3618             final int firstToken = tokens.size() - 1;
   3619             for (int tokenNdx = firstToken; tokenNdx >= 0; --tokenNdx) {
   3620                 final AppWindowToken atoken = tokens.get(tokenNdx);
   3621 
   3622                 if (DEBUG_APP_ORIENTATION) Slog.v(TAG_WM, "Checking app orientation: " + atoken);
   3623 
   3624                 // if we're about to tear down this window and not seek for
   3625                 // the behind activity, don't use it for orientation
   3626                 if (!findingBehind && !atoken.hidden && atoken.hiddenRequested) {
   3627                     if (DEBUG_ORIENTATION) Slog.v(TAG_WM,
   3628                             "Skipping " + atoken + " -- going to hide");
   3629                     continue;
   3630                 }
   3631 
   3632                 if (tokenNdx == firstToken) {
   3633                     // If we have hit a new Task, and the bottom of the previous group didn't
   3634                     // explicitly say to use the orientation behind it, and the last app was
   3635                     // full screen, then we'll stick with the user's orientation.
   3636                     if (lastOrientation != SCREEN_ORIENTATION_BEHIND && lastFullscreen) {
   3637                         if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Done at " + atoken
   3638                                 + " -- end of group, return " + lastOrientation);
   3639                         return lastOrientation;
   3640                     }
   3641                 }
   3642 
   3643                 // We ignore any hidden applications on the top.
   3644                 if (atoken.hiddenRequested) {
   3645                     if (DEBUG_ORIENTATION) Slog.v(TAG_WM,
   3646                             "Skipping " + atoken + " -- hidden on top");
   3647                     continue;
   3648                 }
   3649 
   3650                 // No app except the home app may specify the screen orientation in multi-window,
   3651                 // and only if the docked stack is minimized to avoid weirdness when home task
   3652                 // temporarily gets moved to the front.
   3653                 if (inMultiWindow && (!atoken.mTask.isHomeTask() || !dockMinimized)) {
   3654                     continue;
   3655                 }
   3656 
   3657                 if (tokenNdx == 0) {
   3658                     // Last token in this task.
   3659                     lastOrientation = atoken.requestedOrientation;
   3660                 }
   3661 
   3662                 int or = atoken.requestedOrientation;
   3663                 // If this application is fullscreen, and didn't explicitly say
   3664                 // to use the orientation behind it, then just take whatever
   3665                 // orientation it has and ignores whatever is under it.
   3666                 lastFullscreen = atoken.appFullscreen;
   3667                 if (lastFullscreen && or != SCREEN_ORIENTATION_BEHIND) {
   3668                     if (DEBUG_ORIENTATION) Slog.v(TAG_WM,
   3669                             "Done at " + atoken + " -- full screen, return " + or);
   3670                     return or;
   3671                 }
   3672                 // If this application has requested an explicit orientation, then use it.
   3673                 if (or != SCREEN_ORIENTATION_UNSPECIFIED && or != SCREEN_ORIENTATION_BEHIND) {
   3674                     if (DEBUG_ORIENTATION) Slog.v(TAG_WM,
   3675                             "Done at " + atoken + " -- explicitly set, return " + or);
   3676                     return or;
   3677                 }
   3678                 findingBehind |= (or == SCREEN_ORIENTATION_BEHIND);
   3679             }
   3680         }
   3681         if (DEBUG_ORIENTATION) Slog.v(TAG_WM,
   3682                 "No app is requesting an orientation, return " + mLastOrientation);
   3683         // The next app has not been requested to be visible, so we keep the current orientation
   3684         // to prevent freezing/unfreezing the display too early unless we are in multi-window, in
   3685         // which we don't let the app customize the orientation unless it was the home task that
   3686         // is handled above.
   3687         return inMultiWindow ? SCREEN_ORIENTATION_UNSPECIFIED : mLastOrientation;
   3688     }
   3689 
   3690     @Override
   3691     public Configuration updateOrientationFromAppTokens(
   3692             Configuration currentConfig, IBinder freezeThisOneIfNeeded) {
   3693         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   3694                 "updateOrientationFromAppTokens()")) {
   3695             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   3696         }
   3697 
   3698         Configuration config = null;
   3699         long ident = Binder.clearCallingIdentity();
   3700 
   3701         synchronized(mWindowMap) {
   3702             config = updateOrientationFromAppTokensLocked(currentConfig,
   3703                     freezeThisOneIfNeeded);
   3704         }
   3705 
   3706         Binder.restoreCallingIdentity(ident);
   3707         return config;
   3708     }
   3709 
   3710     private Configuration updateOrientationFromAppTokensLocked(
   3711             Configuration currentConfig, IBinder freezeThisOneIfNeeded) {
   3712         if (!mDisplayReady) {
   3713             return null;
   3714         }
   3715         Configuration config = null;
   3716 
   3717         if (updateOrientationFromAppTokensLocked(false)) {
   3718             if (freezeThisOneIfNeeded != null) {
   3719                 AppWindowToken atoken = findAppWindowToken(freezeThisOneIfNeeded);
   3720                 if (atoken != null) {
   3721                     startAppFreezingScreenLocked(atoken);
   3722                 }
   3723             }
   3724             config = computeNewConfigurationLocked();
   3725 
   3726         } else if (currentConfig != null) {
   3727             // No obvious action we need to take, but if our current
   3728             // state mismatches the activity manager's, update it,
   3729             // disregarding font scale, which should remain set to
   3730             // the value of the previous configuration.
   3731             mTempConfiguration.setToDefaults();
   3732             mTempConfiguration.updateFrom(currentConfig);
   3733             computeScreenConfigurationLocked(mTempConfiguration);
   3734             if (currentConfig.diff(mTempConfiguration) != 0) {
   3735                 mWaitingForConfig = true;
   3736                 final DisplayContent displayContent = getDefaultDisplayContentLocked();
   3737                 displayContent.layoutNeeded = true;
   3738                 int anim[] = new int[2];
   3739                 if (displayContent.isDimming()) {
   3740                     anim[0] = anim[1] = 0;
   3741                 } else {
   3742                     mPolicy.selectRotationAnimationLw(anim);
   3743                 }
   3744                 startFreezingDisplayLocked(false, anim[0], anim[1]);
   3745                 config = new Configuration(mTempConfiguration);
   3746             }
   3747         }
   3748 
   3749         return config;
   3750     }
   3751 
   3752     /*
   3753      * Determine the new desired orientation of the display, returning
   3754      * a non-null new Configuration if it has changed from the current
   3755      * orientation.  IF TRUE IS RETURNED SOMEONE MUST CALL
   3756      * setNewConfiguration() TO TELL THE WINDOW MANAGER IT CAN UNFREEZE THE
   3757      * SCREEN.  This will typically be done for you if you call
   3758      * sendNewConfiguration().
   3759      *
   3760      * The orientation is computed from non-application windows first. If none of
   3761      * the non-application windows specify orientation, the orientation is computed from
   3762      * application tokens.
   3763      * @see android.view.IWindowManager#updateOrientationFromAppTokens(
   3764      * android.os.IBinder)
   3765      */
   3766     boolean updateOrientationFromAppTokensLocked(boolean inTransaction) {
   3767         long ident = Binder.clearCallingIdentity();
   3768         try {
   3769             int req = getOrientationLocked();
   3770             if (req != mLastOrientation) {
   3771                 mLastOrientation = req;
   3772                 //send a message to Policy indicating orientation change to take
   3773                 //action like disabling/enabling sensors etc.,
   3774                 mPolicy.setCurrentOrientationLw(req);
   3775                 if (updateRotationUncheckedLocked(inTransaction)) {
   3776                     // changed
   3777                     return true;
   3778                 }
   3779             }
   3780 
   3781             return false;
   3782         } finally {
   3783             Binder.restoreCallingIdentity(ident);
   3784         }
   3785     }
   3786 
   3787     @Override
   3788     public int[] setNewConfiguration(Configuration config) {
   3789         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   3790                 "setNewConfiguration()")) {
   3791             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   3792         }
   3793 
   3794         synchronized(mWindowMap) {
   3795             if (mWaitingForConfig) {
   3796                 mWaitingForConfig = false;
   3797                 mLastFinishedFreezeSource = "new-config";
   3798             }
   3799             boolean configChanged = mCurConfiguration.diff(config) != 0;
   3800             if (!configChanged) {
   3801                 return null;
   3802             }
   3803             prepareFreezingAllTaskBounds();
   3804             mCurConfiguration = new Configuration(config);
   3805             return onConfigurationChanged();
   3806         }
   3807     }
   3808 
   3809     @Override
   3810     public Rect getBoundsForNewConfiguration(int stackId) {
   3811         synchronized(mWindowMap) {
   3812             final TaskStack stack = mStackIdToStack.get(stackId);
   3813             final Rect outBounds = new Rect();
   3814             stack.getBoundsForNewConfiguration(outBounds);
   3815             return outBounds;
   3816         }
   3817     }
   3818 
   3819     private void prepareFreezingAllTaskBounds() {
   3820         for (int i = mDisplayContents.size() - 1; i >= 0; i--) {
   3821             ArrayList<TaskStack> stacks = mDisplayContents.valueAt(i).getStacks();
   3822             for (int stackNdx = stacks.size() - 1; stackNdx >= 0; --stackNdx) {
   3823                 final TaskStack stack = stacks.get(stackNdx);
   3824                 stack.prepareFreezingTaskBounds();
   3825             }
   3826         }
   3827 
   3828     }
   3829     private int[] onConfigurationChanged() {
   3830         mPolicy.onConfigurationChanged();
   3831 
   3832         final DisplayContent defaultDisplayContent = getDefaultDisplayContentLocked();
   3833         if (!mReconfigureOnConfigurationChanged.contains(defaultDisplayContent)) {
   3834             // The default display size information is heavily dependent on the resources in the
   3835             // current configuration, so we need to reconfigure it everytime the configuration
   3836             // changes. See {@link PhoneWindowManager#setInitialDisplaySize}...sigh...
   3837             mReconfigureOnConfigurationChanged.add(defaultDisplayContent);
   3838         }
   3839         for (int i = mReconfigureOnConfigurationChanged.size() - 1; i >= 0; i--) {
   3840             reconfigureDisplayLocked(mReconfigureOnConfigurationChanged.remove(i));
   3841         }
   3842 
   3843         defaultDisplayContent.getDockedDividerController().onConfigurationChanged();
   3844         mChangedStackList.clear();
   3845         for (int stackNdx = mStackIdToStack.size() - 1; stackNdx >= 0; stackNdx--) {
   3846             final TaskStack stack = mStackIdToStack.valueAt(stackNdx);
   3847             if (stack.onConfigurationChanged()) {
   3848                 mChangedStackList.add(stack.mStackId);
   3849             }
   3850         }
   3851         return mChangedStackList.isEmpty() ?
   3852                 null : ArrayUtils.convertToIntArray(mChangedStackList);
   3853     }
   3854 
   3855     @Override
   3856     public void setAppOrientation(IApplicationToken token, int requestedOrientation) {
   3857         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   3858                 "setAppOrientation()")) {
   3859             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   3860         }
   3861 
   3862         synchronized(mWindowMap) {
   3863             AppWindowToken atoken = findAppWindowToken(token.asBinder());
   3864             if (atoken == null) {
   3865                 Slog.w(TAG_WM, "Attempted to set orientation of non-existing app token: " + token);
   3866                 return;
   3867             }
   3868 
   3869             atoken.requestedOrientation = requestedOrientation;
   3870         }
   3871     }
   3872 
   3873     @Override
   3874     public int getAppOrientation(IApplicationToken token) {
   3875         synchronized(mWindowMap) {
   3876             AppWindowToken wtoken = findAppWindowToken(token.asBinder());
   3877             if (wtoken == null) {
   3878                 return ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED;
   3879             }
   3880 
   3881             return wtoken.requestedOrientation;
   3882         }
   3883     }
   3884 
   3885     void setFocusTaskRegionLocked() {
   3886         if (mFocusedApp != null) {
   3887             final Task task = mFocusedApp.mTask;
   3888             final DisplayContent displayContent = task.getDisplayContent();
   3889             if (displayContent != null) {
   3890                 displayContent.setTouchExcludeRegion(task);
   3891             }
   3892         }
   3893     }
   3894 
   3895     @Override
   3896     public void setFocusedApp(IBinder token, boolean moveFocusNow) {
   3897         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   3898                 "setFocusedApp()")) {
   3899             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   3900         }
   3901 
   3902         synchronized(mWindowMap) {
   3903             final AppWindowToken newFocus;
   3904             if (token == null) {
   3905                 if (DEBUG_FOCUS_LIGHT) Slog.v(TAG_WM, "Clearing focused app, was " + mFocusedApp);
   3906                 newFocus = null;
   3907             } else {
   3908                 newFocus = findAppWindowToken(token);
   3909                 if (newFocus == null) {
   3910                     Slog.w(TAG_WM, "Attempted to set focus to non-existing app token: " + token);
   3911                 }
   3912                 if (DEBUG_FOCUS_LIGHT) Slog.v(TAG_WM, "Set focused app to: " + newFocus
   3913                         + " old focus=" + mFocusedApp + " moveFocusNow=" + moveFocusNow);
   3914             }
   3915 
   3916             final boolean changed = mFocusedApp != newFocus;
   3917             if (changed) {
   3918                 mFocusedApp = newFocus;
   3919                 mInputMonitor.setFocusedAppLw(newFocus);
   3920                 setFocusTaskRegionLocked();
   3921             }
   3922 
   3923             if (moveFocusNow && changed) {
   3924                 final long origId = Binder.clearCallingIdentity();
   3925                 updateFocusedWindowLocked(UPDATE_FOCUS_NORMAL, true /*updateInputWindows*/);
   3926                 Binder.restoreCallingIdentity(origId);
   3927             }
   3928         }
   3929     }
   3930 
   3931     /**
   3932      * @param transit What kind of transition is happening. Use one of the constants
   3933      *                AppTransition.TRANSIT_*.
   3934      * @param alwaysKeepCurrent If true and a transition is already set, new transition will NOT
   3935      *                          be set.
   3936      */
   3937     @Override
   3938     public void prepareAppTransition(int transit, boolean alwaysKeepCurrent) {
   3939         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   3940                 "prepareAppTransition()")) {
   3941             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   3942         }
   3943         synchronized(mWindowMap) {
   3944             boolean prepared = mAppTransition.prepareAppTransitionLocked(
   3945                     transit, alwaysKeepCurrent);
   3946             if (prepared && okToDisplay()) {
   3947                 mSkipAppTransitionAnimation = false;
   3948             }
   3949         }
   3950     }
   3951 
   3952     @Override
   3953     public int getPendingAppTransition() {
   3954         return mAppTransition.getAppTransition();
   3955     }
   3956 
   3957     @Override
   3958     public void overridePendingAppTransition(String packageName,
   3959             int enterAnim, int exitAnim, IRemoteCallback startedCallback) {
   3960         synchronized(mWindowMap) {
   3961             mAppTransition.overridePendingAppTransition(packageName, enterAnim, exitAnim,
   3962                     startedCallback);
   3963         }
   3964     }
   3965 
   3966     @Override
   3967     public void overridePendingAppTransitionScaleUp(int startX, int startY, int startWidth,
   3968             int startHeight) {
   3969         synchronized(mWindowMap) {
   3970             mAppTransition.overridePendingAppTransitionScaleUp(startX, startY, startWidth,
   3971                     startHeight);
   3972         }
   3973     }
   3974 
   3975     @Override
   3976     public void overridePendingAppTransitionClipReveal(int startX, int startY,
   3977             int startWidth, int startHeight) {
   3978         synchronized(mWindowMap) {
   3979             mAppTransition.overridePendingAppTransitionClipReveal(startX, startY, startWidth,
   3980                     startHeight);
   3981         }
   3982     }
   3983 
   3984     @Override
   3985     public void overridePendingAppTransitionThumb(Bitmap srcThumb, int startX,
   3986             int startY, IRemoteCallback startedCallback, boolean scaleUp) {
   3987         synchronized(mWindowMap) {
   3988             mAppTransition.overridePendingAppTransitionThumb(srcThumb, startX, startY,
   3989                     startedCallback, scaleUp);
   3990         }
   3991     }
   3992 
   3993     @Override
   3994     public void overridePendingAppTransitionAspectScaledThumb(Bitmap srcThumb, int startX,
   3995             int startY, int targetWidth, int targetHeight, IRemoteCallback startedCallback,
   3996             boolean scaleUp) {
   3997         synchronized(mWindowMap) {
   3998             mAppTransition.overridePendingAppTransitionAspectScaledThumb(srcThumb, startX, startY,
   3999                     targetWidth, targetHeight, startedCallback, scaleUp);
   4000         }
   4001     }
   4002 
   4003     @Override
   4004     public void overridePendingAppTransitionMultiThumb(AppTransitionAnimationSpec[] specs,
   4005             IRemoteCallback onAnimationStartedCallback, IRemoteCallback onAnimationFinishedCallback,
   4006             boolean scaleUp) {
   4007         synchronized (mWindowMap) {
   4008             mAppTransition.overridePendingAppTransitionMultiThumb(specs, onAnimationStartedCallback,
   4009                     onAnimationFinishedCallback, scaleUp);
   4010             prolongAnimationsFromSpecs(specs, scaleUp);
   4011 
   4012         }
   4013     }
   4014 
   4015     void prolongAnimationsFromSpecs(@NonNull AppTransitionAnimationSpec[] specs, boolean scaleUp) {
   4016         // This is used by freeform <-> recents windows transition. We need to synchronize
   4017         // the animation with the appearance of the content of recents, so we will make
   4018         // animation stay on the first or last frame a little longer.
   4019         mTmpTaskIds.clear();
   4020         for (int i = specs.length - 1; i >= 0; i--) {
   4021             mTmpTaskIds.put(specs[i].taskId, 0);
   4022         }
   4023         for (final WindowState win : mWindowMap.values()) {
   4024             final Task task = win.getTask();
   4025             if (task != null && mTmpTaskIds.get(task.mTaskId, -1) != -1
   4026                     && task.inFreeformWorkspace()) {
   4027                 final AppWindowToken appToken = win.mAppToken;
   4028                 if (appToken != null && appToken.mAppAnimator != null) {
   4029                     appToken.mAppAnimator.startProlongAnimation(scaleUp ?
   4030                             PROLONG_ANIMATION_AT_START : PROLONG_ANIMATION_AT_END);
   4031                 }
   4032             }
   4033         }
   4034     }
   4035 
   4036     @Override
   4037     public void overridePendingAppTransitionInPlace(String packageName, int anim) {
   4038         synchronized(mWindowMap) {
   4039             mAppTransition.overrideInPlaceAppTransition(packageName, anim);
   4040         }
   4041     }
   4042 
   4043     @Override
   4044     public void overridePendingAppTransitionMultiThumbFuture(
   4045             IAppTransitionAnimationSpecsFuture specsFuture, IRemoteCallback callback,
   4046             boolean scaleUp) {
   4047         synchronized(mWindowMap) {
   4048             mAppTransition.overridePendingAppTransitionMultiThumbFuture(specsFuture, callback,
   4049                     scaleUp);
   4050         }
   4051     }
   4052 
   4053     @Override
   4054     public void endProlongedAnimations() {
   4055         synchronized (mWindowMap) {
   4056             for (final WindowState win : mWindowMap.values()) {
   4057                 final AppWindowToken appToken = win.mAppToken;
   4058                 if (appToken != null && appToken.mAppAnimator != null) {
   4059                     appToken.mAppAnimator.endProlongedAnimation();
   4060                 }
   4061             }
   4062             mAppTransition.notifyProlongedAnimationsEnded();
   4063         }
   4064     }
   4065 
   4066     @Override
   4067     public void executeAppTransition() {
   4068         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   4069                 "executeAppTransition()")) {
   4070             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   4071         }
   4072 
   4073         synchronized(mWindowMap) {
   4074             if (DEBUG_APP_TRANSITIONS) Slog.w(TAG_WM, "Execute app transition: " + mAppTransition
   4075                     + " Callers=" + Debug.getCallers(5));
   4076             if (mAppTransition.isTransitionSet()) {
   4077                 mAppTransition.setReady();
   4078                 final long origId = Binder.clearCallingIdentity();
   4079                 try {
   4080                     mWindowPlacerLocked.performSurfacePlacement();
   4081                 } finally {
   4082                     Binder.restoreCallingIdentity(origId);
   4083                 }
   4084             }
   4085         }
   4086     }
   4087 
   4088     @Override
   4089     public boolean setAppStartingWindow(IBinder token, String pkg,
   4090             int theme, CompatibilityInfo compatInfo,
   4091             CharSequence nonLocalizedLabel, int labelRes, int icon, int logo,
   4092             int windowFlags, IBinder transferFrom, boolean createIfNeeded) {
   4093         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   4094                 "setAppStartingWindow()")) {
   4095             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   4096         }
   4097 
   4098         synchronized(mWindowMap) {
   4099             if (DEBUG_STARTING_WINDOW) Slog.v(
   4100                     TAG_WM, "setAppStartingWindow: token=" + token + " pkg=" + pkg
   4101                     + " transferFrom=" + transferFrom);
   4102 
   4103             AppWindowToken wtoken = findAppWindowToken(token);
   4104             if (wtoken == null) {
   4105                 Slog.w(TAG_WM, "Attempted to set icon of non-existing app token: " + token);
   4106                 return false;
   4107             }
   4108 
   4109             // If the display is frozen, we won't do anything until the
   4110             // actual window is displayed so there is no reason to put in
   4111             // the starting window.
   4112             if (!okToDisplay()) {
   4113                 return false;
   4114             }
   4115 
   4116             if (wtoken.startingData != null) {
   4117                 return false;
   4118             }
   4119 
   4120             // If this is a translucent window, then don't
   4121             // show a starting window -- the current effect (a full-screen
   4122             // opaque starting window that fades away to the real contents
   4123             // when it is ready) does not work for this.
   4124             if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM, "Checking theme of starting window: 0x"
   4125                     + Integer.toHexString(theme));
   4126             if (theme != 0) {
   4127                 AttributeCache.Entry ent = AttributeCache.instance().get(pkg, theme,
   4128                         com.android.internal.R.styleable.Window, mCurrentUserId);
   4129                 if (ent == null) {
   4130                     // Whoops!  App doesn't exist.  Um.  Okay.  We'll just
   4131                     // pretend like we didn't see that.
   4132                     return false;
   4133                 }
   4134                 final boolean windowIsTranslucent = ent.array.getBoolean(
   4135                         com.android.internal.R.styleable.Window_windowIsTranslucent, false);
   4136                 final boolean windowIsFloating = ent.array.getBoolean(
   4137                         com.android.internal.R.styleable.Window_windowIsFloating, false);
   4138                 final boolean windowShowWallpaper = ent.array.getBoolean(
   4139                         com.android.internal.R.styleable.Window_windowShowWallpaper, false);
   4140                 final boolean windowDisableStarting = ent.array.getBoolean(
   4141                         com.android.internal.R.styleable.Window_windowDisablePreview, false);
   4142                 if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM, "Translucent=" + windowIsTranslucent
   4143                         + " Floating=" + windowIsFloating
   4144                         + " ShowWallpaper=" + windowShowWallpaper);
   4145                 if (windowIsTranslucent) {
   4146                     return false;
   4147                 }
   4148                 if (windowIsFloating || windowDisableStarting) {
   4149                     return false;
   4150                 }
   4151                 if (windowShowWallpaper) {
   4152                     if (mWallpaperControllerLocked.getWallpaperTarget() == null) {
   4153                         // If this theme is requesting a wallpaper, and the wallpaper
   4154                         // is not currently visible, then this effectively serves as
   4155                         // an opaque window and our starting window transition animation
   4156                         // can still work.  We just need to make sure the starting window
   4157                         // is also showing the wallpaper.
   4158                         windowFlags |= FLAG_SHOW_WALLPAPER;
   4159                     } else {
   4160                         return false;
   4161                     }
   4162                 }
   4163             }
   4164 
   4165             if (transferStartingWindow(transferFrom, wtoken)) {
   4166                 return true;
   4167             }
   4168 
   4169             // There is no existing starting window, and the caller doesn't
   4170             // want us to create one, so that's it!
   4171             if (!createIfNeeded) {
   4172                 return false;
   4173             }
   4174 
   4175             if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM, "Creating StartingData");
   4176             wtoken.startingData = new StartingData(pkg, theme, compatInfo, nonLocalizedLabel,
   4177                     labelRes, icon, logo, windowFlags);
   4178             Message m = mH.obtainMessage(H.ADD_STARTING, wtoken);
   4179             // Note: we really want to do sendMessageAtFrontOfQueue() because we
   4180             // want to process the message ASAP, before any other queued
   4181             // messages.
   4182             if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM, "Enqueueing ADD_STARTING");
   4183             mH.sendMessageAtFrontOfQueue(m);
   4184         }
   4185         return true;
   4186     }
   4187 
   4188     private boolean transferStartingWindow(IBinder transferFrom, AppWindowToken wtoken) {
   4189         if (transferFrom == null) {
   4190             return false;
   4191         }
   4192         AppWindowToken ttoken = findAppWindowToken(transferFrom);
   4193         if (ttoken == null) {
   4194             return false;
   4195         }
   4196         WindowState startingWindow = ttoken.startingWindow;
   4197         if (startingWindow != null && ttoken.startingView != null) {
   4198             // In this case, the starting icon has already been displayed, so start
   4199             // letting windows get shown immediately without any more transitions.
   4200             mSkipAppTransitionAnimation = true;
   4201 
   4202             if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM,
   4203                     "Moving existing starting " + startingWindow + " from " + ttoken
   4204                             + " to " + wtoken);
   4205             final long origId = Binder.clearCallingIdentity();
   4206 
   4207             // Transfer the starting window over to the new token.
   4208             wtoken.startingData = ttoken.startingData;
   4209             wtoken.startingView = ttoken.startingView;
   4210             wtoken.startingDisplayed = ttoken.startingDisplayed;
   4211             ttoken.startingDisplayed = false;
   4212             wtoken.startingWindow = startingWindow;
   4213             wtoken.reportedVisible = ttoken.reportedVisible;
   4214             ttoken.startingData = null;
   4215             ttoken.startingView = null;
   4216             ttoken.startingWindow = null;
   4217             ttoken.startingMoved = true;
   4218             startingWindow.mToken = wtoken;
   4219             startingWindow.mRootToken = wtoken;
   4220             startingWindow.mAppToken = wtoken;
   4221 
   4222             if (DEBUG_WINDOW_MOVEMENT || DEBUG_ADD_REMOVE || DEBUG_STARTING_WINDOW) {
   4223                 Slog.v(TAG_WM, "Removing starting window: " + startingWindow);
   4224             }
   4225             startingWindow.getWindowList().remove(startingWindow);
   4226             mWindowsChanged = true;
   4227             if (DEBUG_ADD_REMOVE) Slog.v(TAG_WM,
   4228                     "Removing starting " + startingWindow + " from " + ttoken);
   4229             ttoken.windows.remove(startingWindow);
   4230             ttoken.allAppWindows.remove(startingWindow);
   4231             addWindowToListInOrderLocked(startingWindow, true);
   4232 
   4233             // Propagate other interesting state between the
   4234             // tokens.  If the old token is displayed, we should
   4235             // immediately force the new one to be displayed.  If
   4236             // it is animating, we need to move that animation to
   4237             // the new one.
   4238             if (ttoken.allDrawn) {
   4239                 wtoken.allDrawn = true;
   4240                 wtoken.deferClearAllDrawn = ttoken.deferClearAllDrawn;
   4241             }
   4242             if (ttoken.firstWindowDrawn) {
   4243                 wtoken.firstWindowDrawn = true;
   4244             }
   4245             if (!ttoken.hidden) {
   4246                 wtoken.hidden = false;
   4247                 wtoken.hiddenRequested = false;
   4248             }
   4249             if (wtoken.clientHidden != ttoken.clientHidden) {
   4250                 wtoken.clientHidden = ttoken.clientHidden;
   4251                 wtoken.sendAppVisibilityToClients();
   4252             }
   4253             ttoken.mAppAnimator.transferCurrentAnimation(
   4254                     wtoken.mAppAnimator, startingWindow.mWinAnimator);
   4255 
   4256             updateFocusedWindowLocked(UPDATE_FOCUS_WILL_PLACE_SURFACES,
   4257                     true /*updateInputWindows*/);
   4258             getDefaultDisplayContentLocked().layoutNeeded = true;
   4259             mWindowPlacerLocked.performSurfacePlacement();
   4260             Binder.restoreCallingIdentity(origId);
   4261             return true;
   4262         } else if (ttoken.startingData != null) {
   4263             // The previous app was getting ready to show a
   4264             // starting window, but hasn't yet done so.  Steal it!
   4265             if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM, "Moving pending starting from " + ttoken
   4266                     + " to " + wtoken);
   4267             wtoken.startingData = ttoken.startingData;
   4268             ttoken.startingData = null;
   4269             ttoken.startingMoved = true;
   4270             Message m = mH.obtainMessage(H.ADD_STARTING, wtoken);
   4271             // Note: we really want to do sendMessageAtFrontOfQueue() because we
   4272             // want to process the message ASAP, before any other queued
   4273             // messages.
   4274             mH.sendMessageAtFrontOfQueue(m);
   4275             return true;
   4276         }
   4277         final AppWindowAnimator tAppAnimator = ttoken.mAppAnimator;
   4278         final AppWindowAnimator wAppAnimator = wtoken.mAppAnimator;
   4279         if (tAppAnimator.thumbnail != null) {
   4280             // The old token is animating with a thumbnail, transfer that to the new token.
   4281             if (wAppAnimator.thumbnail != null) {
   4282                 wAppAnimator.thumbnail.destroy();
   4283             }
   4284             wAppAnimator.thumbnail = tAppAnimator.thumbnail;
   4285             wAppAnimator.thumbnailLayer = tAppAnimator.thumbnailLayer;
   4286             wAppAnimator.thumbnailAnimation = tAppAnimator.thumbnailAnimation;
   4287             tAppAnimator.thumbnail = null;
   4288         }
   4289         return false;
   4290     }
   4291 
   4292     public void removeAppStartingWindow(IBinder token) {
   4293         synchronized (mWindowMap) {
   4294             final AppWindowToken wtoken = mTokenMap.get(token).appWindowToken;
   4295             scheduleRemoveStartingWindowLocked(wtoken);
   4296         }
   4297     }
   4298 
   4299     public void setAppFullscreen(IBinder token, boolean toOpaque) {
   4300         synchronized (mWindowMap) {
   4301             AppWindowToken atoken = findAppWindowToken(token);
   4302             if (atoken != null) {
   4303                 atoken.appFullscreen = toOpaque;
   4304                 setWindowOpaqueLocked(token, toOpaque);
   4305                 mWindowPlacerLocked.requestTraversal();
   4306             }
   4307         }
   4308     }
   4309 
   4310     public void setWindowOpaque(IBinder token, boolean isOpaque) {
   4311         synchronized (mWindowMap) {
   4312             setWindowOpaqueLocked(token, isOpaque);
   4313         }
   4314     }
   4315 
   4316     public void setWindowOpaqueLocked(IBinder token, boolean isOpaque) {
   4317         AppWindowToken wtoken = findAppWindowToken(token);
   4318         if (wtoken != null) {
   4319             WindowState win = wtoken.findMainWindow();
   4320             if (win != null) {
   4321                 win.mWinAnimator.setOpaqueLocked(isOpaque);
   4322             }
   4323         }
   4324     }
   4325 
   4326     boolean setTokenVisibilityLocked(AppWindowToken wtoken, WindowManager.LayoutParams lp,
   4327             boolean visible, int transit, boolean performLayout, boolean isVoiceInteraction) {
   4328         boolean delayed = false;
   4329 
   4330         if (wtoken.clientHidden == visible) {
   4331             wtoken.clientHidden = !visible;
   4332             wtoken.sendAppVisibilityToClients();
   4333         }
   4334 
   4335         // Allow for state changes and animation to be applied if:
   4336         // * token is transitioning visibility state
   4337         // * or the token was marked as hidden and is exiting before we had a chance to play the
   4338         // transition animation
   4339         // * or this is an opening app and windows are being replaced.
   4340         boolean visibilityChanged = false;
   4341         if (wtoken.hidden == visible || (wtoken.hidden && wtoken.mIsExiting) ||
   4342                 (visible && wtoken.waitingForReplacement())) {
   4343             boolean changed = false;
   4344             if (DEBUG_APP_TRANSITIONS) Slog.v(
   4345                 TAG_WM, "Changing app " + wtoken + " hidden=" + wtoken.hidden
   4346                 + " performLayout=" + performLayout);
   4347 
   4348             boolean runningAppAnimation = false;
   4349 
   4350             if (transit != AppTransition.TRANSIT_UNSET) {
   4351                 if (wtoken.mAppAnimator.animation == AppWindowAnimator.sDummyAnimation) {
   4352                     wtoken.mAppAnimator.setNullAnimation();
   4353                 }
   4354                 if (applyAnimationLocked(wtoken, lp, transit, visible, isVoiceInteraction)) {
   4355                     delayed = runningAppAnimation = true;
   4356                 }
   4357                 WindowState window = wtoken.findMainWindow();
   4358                 //TODO (multidisplay): Magnification is supported only for the default display.
   4359                 if (window != null && mAccessibilityController != null
   4360                         && window.getDisplayId() == Display.DEFAULT_DISPLAY) {
   4361                     mAccessibilityController.onAppWindowTransitionLocked(window, transit);
   4362                 }
   4363                 changed = true;
   4364             }
   4365 
   4366             final int windowsCount = wtoken.allAppWindows.size();
   4367             for (int i = 0; i < windowsCount; i++) {
   4368                 WindowState win = wtoken.allAppWindows.get(i);
   4369                 if (win == wtoken.startingWindow) {
   4370                     // Starting window that's exiting will be removed when the animation
   4371                     // finishes. Mark all relevant flags for that finishExit will proceed
   4372                     // all the way to actually remove it.
   4373                     if (!visible && win.isVisibleNow() && wtoken.mAppAnimator.isAnimating()) {
   4374                         win.mAnimatingExit = true;
   4375                         win.mRemoveOnExit = true;
   4376                         win.mWindowRemovalAllowed = true;
   4377                     }
   4378                     continue;
   4379                 }
   4380 
   4381                 //Slog.i(TAG_WM, "Window " + win + ": vis=" + win.isVisible());
   4382                 //win.dump("  ");
   4383                 if (visible) {
   4384                     if (!win.isVisibleNow()) {
   4385                         if (!runningAppAnimation) {
   4386                             win.mWinAnimator.applyAnimationLocked(
   4387                                     WindowManagerPolicy.TRANSIT_ENTER, true);
   4388                             //TODO (multidisplay): Magnification is supported only for the default
   4389                             if (mAccessibilityController != null
   4390                                     && win.getDisplayId() == Display.DEFAULT_DISPLAY) {
   4391                                 mAccessibilityController.onWindowTransitionLocked(win,
   4392                                         WindowManagerPolicy.TRANSIT_ENTER);
   4393                             }
   4394                         }
   4395                         changed = true;
   4396                         win.setDisplayLayoutNeeded();
   4397                     }
   4398                 } else if (win.isVisibleNow()) {
   4399                     if (!runningAppAnimation) {
   4400                         win.mWinAnimator.applyAnimationLocked(
   4401                                 WindowManagerPolicy.TRANSIT_EXIT, false);
   4402                         //TODO (multidisplay): Magnification is supported only for the default
   4403                         if (mAccessibilityController != null
   4404                                 && win.getDisplayId() == Display.DEFAULT_DISPLAY) {
   4405                             mAccessibilityController.onWindowTransitionLocked(win,
   4406                                     WindowManagerPolicy.TRANSIT_EXIT);
   4407                         }
   4408                     }
   4409                     changed = true;
   4410                     win.setDisplayLayoutNeeded();
   4411                 }
   4412             }
   4413 
   4414             wtoken.hidden = wtoken.hiddenRequested = !visible;
   4415             visibilityChanged = true;
   4416             if (!visible) {
   4417                 unsetAppFreezingScreenLocked(wtoken, true, true);
   4418             } else {
   4419                 // If we are being set visible, and the starting window is
   4420                 // not yet displayed, then make sure it doesn't get displayed.
   4421                 WindowState swin = wtoken.startingWindow;
   4422                 if (swin != null && !swin.isDrawnLw()) {
   4423                     swin.mPolicyVisibility = false;
   4424                     swin.mPolicyVisibilityAfterAnim = false;
   4425                  }
   4426             }
   4427 
   4428             if (DEBUG_APP_TRANSITIONS) Slog.v(TAG_WM, "setTokenVisibilityLocked: " + wtoken
   4429                       + ": hidden=" + wtoken.hidden + " hiddenRequested="
   4430                       + wtoken.hiddenRequested);
   4431 
   4432             if (changed) {
   4433                 mInputMonitor.setUpdateInputWindowsNeededLw();
   4434                 if (performLayout) {
   4435                     updateFocusedWindowLocked(UPDATE_FOCUS_WILL_PLACE_SURFACES,
   4436                             false /*updateInputWindows*/);
   4437                     mWindowPlacerLocked.performSurfacePlacement();
   4438                 }
   4439                 mInputMonitor.updateInputWindowsLw(false /*force*/);
   4440             }
   4441         }
   4442 
   4443         if (wtoken.mAppAnimator.animation != null) {
   4444             delayed = true;
   4445         }
   4446 
   4447         for (int i = wtoken.allAppWindows.size() - 1; i >= 0 && !delayed; i--) {
   4448             if (wtoken.allAppWindows.get(i).mWinAnimator.isWindowAnimationSet()) {
   4449                 delayed = true;
   4450             }
   4451         }
   4452 
   4453         if (visibilityChanged) {
   4454             if (visible && !delayed) {
   4455                 // The token was made immediately visible, there will be no entrance animation.
   4456                 // We need to inform the client the enter animation was finished.
   4457                 wtoken.mEnteringAnimation = true;
   4458                 mActivityManagerAppTransitionNotifier.onAppTransitionFinishedLocked(wtoken.token);
   4459             }
   4460 
   4461             if (!mClosingApps.contains(wtoken) && !mOpeningApps.contains(wtoken)) {
   4462                 // The token is not closing nor opening, so even if there is an animation set, that
   4463                 // doesn't mean that it goes through the normal app transition cycle so we have
   4464                 // to inform the docked controller about visibility change.
   4465                 getDefaultDisplayContentLocked().getDockedDividerController()
   4466                         .notifyAppVisibilityChanged();
   4467             }
   4468         }
   4469 
   4470         return delayed;
   4471     }
   4472 
   4473     void updateTokenInPlaceLocked(AppWindowToken wtoken, int transit) {
   4474         if (transit != AppTransition.TRANSIT_UNSET) {
   4475             if (wtoken.mAppAnimator.animation == AppWindowAnimator.sDummyAnimation) {
   4476                 wtoken.mAppAnimator.setNullAnimation();
   4477             }
   4478             applyAnimationLocked(wtoken, null, transit, false, false);
   4479         }
   4480     }
   4481 
   4482     @Override
   4483     public void notifyAppResumed(IBinder token, boolean wasStopped, boolean allowSavedSurface) {
   4484         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   4485                 "notifyAppResumed()")) {
   4486             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   4487         }
   4488 
   4489         synchronized(mWindowMap) {
   4490             final AppWindowToken wtoken;
   4491             wtoken = findAppWindowToken(token);
   4492             if (wtoken == null) {
   4493                 Slog.w(TAG_WM, "Attempted to notify resumed of non-existing app token: " + token);
   4494                 return;
   4495             }
   4496             wtoken.notifyAppResumed(wasStopped, allowSavedSurface);
   4497         }
   4498     }
   4499 
   4500     @Override
   4501     public void notifyAppStopped(IBinder token) {
   4502         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   4503                 "notifyAppStopped()")) {
   4504             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   4505         }
   4506 
   4507         synchronized(mWindowMap) {
   4508             final AppWindowToken wtoken;
   4509             wtoken = findAppWindowToken(token);
   4510             if (wtoken == null) {
   4511                 Slog.w(TAG_WM, "Attempted to notify stopped of non-existing app token: " + token);
   4512                 return;
   4513             }
   4514             wtoken.notifyAppStopped();
   4515         }
   4516     }
   4517 
   4518     @Override
   4519     public void setAppVisibility(IBinder token, boolean visible) {
   4520         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   4521                 "setAppVisibility()")) {
   4522             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   4523         }
   4524 
   4525         AppWindowToken wtoken;
   4526 
   4527         synchronized(mWindowMap) {
   4528             wtoken = findAppWindowToken(token);
   4529             if (wtoken == null) {
   4530                 Slog.w(TAG_WM, "Attempted to set visibility of non-existing app token: " + token);
   4531                 return;
   4532             }
   4533 
   4534             if (DEBUG_APP_TRANSITIONS || DEBUG_ORIENTATION) Slog.v(TAG_WM, "setAppVisibility(" +
   4535                     token + ", visible=" + visible + "): " + mAppTransition +
   4536                     " hidden=" + wtoken.hidden + " hiddenRequested=" +
   4537                     wtoken.hiddenRequested + " Callers=" + Debug.getCallers(6));
   4538 
   4539             mOpeningApps.remove(wtoken);
   4540             mClosingApps.remove(wtoken);
   4541             wtoken.waitingToShow = false;
   4542             wtoken.hiddenRequested = !visible;
   4543 
   4544             if (!visible) {
   4545                 // If the app is dead while it was visible, we kept its dead window on screen.
   4546                 // Now that the app is going invisible, we can remove it. It will be restarted
   4547                 // if made visible again.
   4548                 wtoken.removeAllDeadWindows();
   4549                 wtoken.setVisibleBeforeClientHidden();
   4550             } else if (visible) {
   4551                 if (!mAppTransition.isTransitionSet() && mAppTransition.isReady()) {
   4552                     // Add the app mOpeningApps if transition is unset but ready. This means
   4553                     // we're doing a screen freeze, and the unfreeze will wait for all opening
   4554                     // apps to be ready.
   4555                     mOpeningApps.add(wtoken);
   4556                 }
   4557                 wtoken.startingMoved = false;
   4558                 // If the token is currently hidden (should be the common case), or has been
   4559                 // stopped, then we need to set up to wait for its windows to be ready.
   4560                 if (wtoken.hidden || wtoken.mAppStopped) {
   4561                     wtoken.clearAllDrawn();
   4562 
   4563                     // If the app was already visible, don't reset the waitingToShow state.
   4564                     if (wtoken.hidden) {
   4565                         wtoken.waitingToShow = true;
   4566                     }
   4567 
   4568                     if (wtoken.clientHidden) {
   4569                         // In the case where we are making an app visible
   4570                         // but holding off for a transition, we still need
   4571                         // to tell the client to make its windows visible so
   4572                         // they get drawn.  Otherwise, we will wait on
   4573                         // performing the transition until all windows have
   4574                         // been drawn, they never will be, and we are sad.
   4575                         wtoken.clientHidden = false;
   4576                         wtoken.sendAppVisibilityToClients();
   4577                     }
   4578                 }
   4579                 wtoken.requestUpdateWallpaperIfNeeded();
   4580 
   4581                 if (DEBUG_ADD_REMOVE) Slog.v(
   4582                         TAG_WM, "No longer Stopped: " + wtoken);
   4583                 wtoken.mAppStopped = false;
   4584             }
   4585 
   4586             // If we are preparing an app transition, then delay changing
   4587             // the visibility of this token until we execute that transition.
   4588             if (okToDisplay() && mAppTransition.isTransitionSet()) {
   4589                 // A dummy animation is a placeholder animation which informs others that an
   4590                 // animation is going on (in this case an application transition). If the animation
   4591                 // was transferred from another application/animator, no dummy animator should be
   4592                 // created since an animation is already in progress.
   4593                 if (wtoken.mAppAnimator.usingTransferredAnimation
   4594                         && wtoken.mAppAnimator.animation == null) {
   4595                     Slog.wtf(TAG_WM, "Will NOT set dummy animation on: " + wtoken
   4596                             + ", using null transfered animation!");
   4597                 }
   4598                 if (!wtoken.mAppAnimator.usingTransferredAnimation &&
   4599                         (!wtoken.startingDisplayed || mSkipAppTransitionAnimation)) {
   4600                     if (DEBUG_APP_TRANSITIONS) Slog.v(
   4601                             TAG_WM, "Setting dummy animation on: " + wtoken);
   4602                     wtoken.mAppAnimator.setDummyAnimation();
   4603                 }
   4604                 wtoken.inPendingTransaction = true;
   4605                 if (visible) {
   4606                     mOpeningApps.add(wtoken);
   4607                     wtoken.mEnteringAnimation = true;
   4608                 } else {
   4609                     mClosingApps.add(wtoken);
   4610                     wtoken.mEnteringAnimation = false;
   4611                 }
   4612                 if (mAppTransition.getAppTransition() == AppTransition.TRANSIT_TASK_OPEN_BEHIND) {
   4613                     // We're launchingBehind, add the launching activity to mOpeningApps.
   4614                     final WindowState win =
   4615                             findFocusedWindowLocked(getDefaultDisplayContentLocked());
   4616                     if (win != null) {
   4617                         final AppWindowToken focusedToken = win.mAppToken;
   4618                         if (focusedToken != null) {
   4619                             if (DEBUG_APP_TRANSITIONS) Slog.d(TAG_WM, "TRANSIT_TASK_OPEN_BEHIND, " +
   4620                                     " adding " + focusedToken + " to mOpeningApps");
   4621                             // Force animation to be loaded.
   4622                             focusedToken.hidden = true;
   4623                             mOpeningApps.add(focusedToken);
   4624                         }
   4625                     }
   4626                 }
   4627                 return;
   4628             }
   4629 
   4630             final long origId = Binder.clearCallingIdentity();
   4631             wtoken.inPendingTransaction = false;
   4632             setTokenVisibilityLocked(wtoken, null, visible, AppTransition.TRANSIT_UNSET,
   4633                     true, wtoken.voiceInteraction);
   4634             wtoken.updateReportedVisibilityLocked();
   4635             Binder.restoreCallingIdentity(origId);
   4636         }
   4637     }
   4638 
   4639     void unsetAppFreezingScreenLocked(AppWindowToken wtoken,
   4640             boolean unfreezeSurfaceNow, boolean force) {
   4641         if (wtoken.mAppAnimator.freezingScreen) {
   4642             if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Clear freezing of " + wtoken
   4643                     + " force=" + force);
   4644             final int N = wtoken.allAppWindows.size();
   4645             boolean unfrozeWindows = false;
   4646             for (int i=0; i<N; i++) {
   4647                 WindowState w = wtoken.allAppWindows.get(i);
   4648                 if (w.mAppFreezing) {
   4649                     w.mAppFreezing = false;
   4650                     if (w.mHasSurface && !w.mOrientationChanging
   4651                             && mWindowsFreezingScreen != WINDOWS_FREEZING_SCREENS_TIMEOUT) {
   4652                         if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "set mOrientationChanging of " + w);
   4653                         w.mOrientationChanging = true;
   4654                         mWindowPlacerLocked.mOrientationChangeComplete = false;
   4655                     }
   4656                     w.mLastFreezeDuration = 0;
   4657                     unfrozeWindows = true;
   4658                     w.setDisplayLayoutNeeded();
   4659                 }
   4660             }
   4661             if (force || unfrozeWindows) {
   4662                 if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "No longer freezing: " + wtoken);
   4663                 wtoken.mAppAnimator.freezingScreen = false;
   4664                 wtoken.mAppAnimator.lastFreezeDuration = (int)(SystemClock.elapsedRealtime()
   4665                         - mDisplayFreezeTime);
   4666                 mAppsFreezingScreen--;
   4667                 mLastFinishedFreezeSource = wtoken;
   4668             }
   4669             if (unfreezeSurfaceNow) {
   4670                 if (unfrozeWindows) {
   4671                     mWindowPlacerLocked.performSurfacePlacement();
   4672                 }
   4673                 stopFreezingDisplayLocked();
   4674             }
   4675         }
   4676     }
   4677 
   4678     private void startAppFreezingScreenLocked(AppWindowToken wtoken) {
   4679         if (DEBUG_ORIENTATION) logWithStack(TAG, "Set freezing of " + wtoken.appToken + ": hidden="
   4680                 + wtoken.hidden + " freezing=" + wtoken.mAppAnimator.freezingScreen);
   4681         if (!wtoken.hiddenRequested) {
   4682             if (!wtoken.mAppAnimator.freezingScreen) {
   4683                 wtoken.mAppAnimator.freezingScreen = true;
   4684                 wtoken.mAppAnimator.lastFreezeDuration = 0;
   4685                 mAppsFreezingScreen++;
   4686                 if (mAppsFreezingScreen == 1) {
   4687                     startFreezingDisplayLocked(false, 0, 0);
   4688                     mH.removeMessages(H.APP_FREEZE_TIMEOUT);
   4689                     mH.sendEmptyMessageDelayed(H.APP_FREEZE_TIMEOUT, 2000);
   4690                 }
   4691             }
   4692             final int N = wtoken.allAppWindows.size();
   4693             for (int i=0; i<N; i++) {
   4694                 WindowState w = wtoken.allAppWindows.get(i);
   4695                 w.mAppFreezing = true;
   4696             }
   4697         }
   4698     }
   4699 
   4700     @Override
   4701     public void startAppFreezingScreen(IBinder token, int configChanges) {
   4702         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   4703                 "setAppFreezingScreen()")) {
   4704             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   4705         }
   4706 
   4707         synchronized(mWindowMap) {
   4708             if (configChanges == 0 && okToDisplay()) {
   4709                 if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Skipping set freeze of " + token);
   4710                 return;
   4711             }
   4712 
   4713             AppWindowToken wtoken = findAppWindowToken(token);
   4714             if (wtoken == null || wtoken.appToken == null) {
   4715                 Slog.w(TAG_WM, "Attempted to freeze screen with non-existing app token: " + wtoken);
   4716                 return;
   4717             }
   4718             final long origId = Binder.clearCallingIdentity();
   4719             startAppFreezingScreenLocked(wtoken);
   4720             Binder.restoreCallingIdentity(origId);
   4721         }
   4722     }
   4723 
   4724     @Override
   4725     public void stopAppFreezingScreen(IBinder token, boolean force) {
   4726         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   4727                 "setAppFreezingScreen()")) {
   4728             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   4729         }
   4730 
   4731         synchronized(mWindowMap) {
   4732             AppWindowToken wtoken = findAppWindowToken(token);
   4733             if (wtoken == null || wtoken.appToken == null) {
   4734                 return;
   4735             }
   4736             final long origId = Binder.clearCallingIdentity();
   4737             if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Clear freezing of " + token
   4738                     + ": hidden=" + wtoken.hidden + " freezing=" + wtoken.mAppAnimator.freezingScreen);
   4739             unsetAppFreezingScreenLocked(wtoken, true, force);
   4740             Binder.restoreCallingIdentity(origId);
   4741         }
   4742     }
   4743 
   4744     @Override
   4745     public void removeAppToken(IBinder token) {
   4746         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   4747                 "removeAppToken()")) {
   4748             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   4749         }
   4750 
   4751         AppWindowToken wtoken = null;
   4752         AppWindowToken startingToken = null;
   4753         boolean delayed = false;
   4754 
   4755         final long origId = Binder.clearCallingIdentity();
   4756         synchronized(mWindowMap) {
   4757             WindowToken basewtoken = mTokenMap.remove(token);
   4758             if (basewtoken != null && (wtoken=basewtoken.appWindowToken) != null) {
   4759                 if (DEBUG_APP_TRANSITIONS) Slog.v(TAG_WM, "Removing app token: " + wtoken);
   4760                 delayed = setTokenVisibilityLocked(wtoken, null, false,
   4761                         AppTransition.TRANSIT_UNSET, true, wtoken.voiceInteraction);
   4762                 wtoken.inPendingTransaction = false;
   4763                 mOpeningApps.remove(wtoken);
   4764                 wtoken.waitingToShow = false;
   4765                 if (mClosingApps.contains(wtoken)) {
   4766                     delayed = true;
   4767                 } else if (mAppTransition.isTransitionSet()) {
   4768                     mClosingApps.add(wtoken);
   4769                     delayed = true;
   4770                 }
   4771                 if (DEBUG_APP_TRANSITIONS) Slog.v(
   4772                         TAG_WM, "Removing app " + wtoken + " delayed=" + delayed
   4773                         + " animation=" + wtoken.mAppAnimator.animation
   4774                         + " animating=" + wtoken.mAppAnimator.animating);
   4775                 if (DEBUG_ADD_REMOVE || DEBUG_TOKEN_MOVEMENT) Slog.v(TAG_WM, "removeAppToken: "
   4776                         + wtoken + " delayed=" + delayed + " Callers=" + Debug.getCallers(4));
   4777                 final TaskStack stack = wtoken.mTask.mStack;
   4778                 if (delayed && !wtoken.allAppWindows.isEmpty()) {
   4779                     // set the token aside because it has an active animation to be finished
   4780                     if (DEBUG_ADD_REMOVE || DEBUG_TOKEN_MOVEMENT) Slog.v(TAG_WM,
   4781                             "removeAppToken make exiting: " + wtoken);
   4782                     stack.mExitingAppTokens.add(wtoken);
   4783                     wtoken.mIsExiting = true;
   4784                 } else {
   4785                     // Make sure there is no animation running on this token,
   4786                     // so any windows associated with it will be removed as
   4787                     // soon as their animations are complete
   4788                     wtoken.mAppAnimator.clearAnimation();
   4789                     wtoken.mAppAnimator.animating = false;
   4790                     wtoken.removeAppFromTaskLocked();
   4791                 }
   4792 
   4793                 wtoken.removed = true;
   4794                 if (wtoken.startingData != null) {
   4795                     startingToken = wtoken;
   4796                 }
   4797                 unsetAppFreezingScreenLocked(wtoken, true, true);
   4798                 if (mFocusedApp == wtoken) {
   4799                     if (DEBUG_FOCUS_LIGHT) Slog.v(TAG_WM, "Removing focused app token:" + wtoken);
   4800                     mFocusedApp = null;
   4801                     updateFocusedWindowLocked(UPDATE_FOCUS_NORMAL, true /*updateInputWindows*/);
   4802                     mInputMonitor.setFocusedAppLw(null);
   4803                 }
   4804             } else {
   4805                 Slog.w(TAG_WM, "Attempted to remove non-existing app token: " + token);
   4806             }
   4807 
   4808             if (!delayed && wtoken != null) {
   4809                 wtoken.updateReportedVisibilityLocked();
   4810             }
   4811 
   4812             // Will only remove if startingToken non null.
   4813             scheduleRemoveStartingWindowLocked(startingToken);
   4814         }
   4815         Binder.restoreCallingIdentity(origId);
   4816 
   4817     }
   4818 
   4819     void scheduleRemoveStartingWindowLocked(AppWindowToken wtoken) {
   4820         if (wtoken == null) {
   4821             return;
   4822         }
   4823         if (mH.hasMessages(H.REMOVE_STARTING, wtoken)) {
   4824             // Already scheduled.
   4825             return;
   4826         }
   4827 
   4828         if (wtoken.startingWindow == null) {
   4829             if (wtoken.startingData != null) {
   4830                 // Starting window has not been added yet, but it is scheduled to be added.
   4831                 // Go ahead and cancel the request.
   4832                 if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM,
   4833                         "Clearing startingData for token=" + wtoken);
   4834                 wtoken.startingData = null;
   4835             }
   4836             return;
   4837         }
   4838 
   4839         if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM, Debug.getCallers(1) +
   4840                 ": Schedule remove starting " + wtoken + (wtoken != null ?
   4841                 " startingWindow=" + wtoken.startingWindow : ""));
   4842         Message m = mH.obtainMessage(H.REMOVE_STARTING, wtoken);
   4843         mH.sendMessage(m);
   4844     }
   4845 
   4846     void dumpAppTokensLocked() {
   4847         final int numStacks = mStackIdToStack.size();
   4848         for (int stackNdx = 0; stackNdx < numStacks; ++stackNdx) {
   4849             final TaskStack stack = mStackIdToStack.valueAt(stackNdx);
   4850             Slog.v(TAG_WM, "  Stack #" + stack.mStackId + " tasks from bottom to top:");
   4851             final ArrayList<Task> tasks = stack.getTasks();
   4852             final int numTasks = tasks.size();
   4853             for (int taskNdx = 0; taskNdx < numTasks; ++taskNdx) {
   4854                 final Task task = tasks.get(taskNdx);
   4855                 Slog.v(TAG_WM, "    Task #" + task.mTaskId + " activities from bottom to top:");
   4856                 AppTokenList tokens = task.mAppTokens;
   4857                 final int numTokens = tokens.size();
   4858                 for (int tokenNdx = 0; tokenNdx < numTokens; ++tokenNdx) {
   4859                     Slog.v(TAG_WM, "      activity #" + tokenNdx + ": " + tokens.get(tokenNdx).token);
   4860                 }
   4861             }
   4862         }
   4863     }
   4864 
   4865     void dumpWindowsLocked() {
   4866         final int numDisplays = mDisplayContents.size();
   4867         for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   4868             final DisplayContent displayContent = mDisplayContents.valueAt(displayNdx);
   4869             Slog.v(TAG_WM, " Display #" + displayContent.getDisplayId());
   4870             final WindowList windows = displayContent.getWindowList();
   4871             for (int winNdx = windows.size() - 1; winNdx >= 0; --winNdx) {
   4872                 Slog.v(TAG_WM, "  #" + winNdx + ": " + windows.get(winNdx));
   4873             }
   4874         }
   4875     }
   4876 
   4877     private final int reAddWindowLocked(int index, WindowState win) {
   4878         final WindowList windows = win.getWindowList();
   4879         // Adding child windows relies on mChildWindows being ordered by mSubLayer.
   4880         final int NCW = win.mChildWindows.size();
   4881         boolean winAdded = false;
   4882         for (int j=0; j<NCW; j++) {
   4883             WindowState cwin = win.mChildWindows.get(j);
   4884             if (!winAdded && cwin.mSubLayer >= 0) {
   4885                 if (DEBUG_WINDOW_MOVEMENT) Slog.v(TAG_WM, "Re-adding child window at "
   4886                         + index + ": " + cwin);
   4887                 win.mRebuilding = false;
   4888                 windows.add(index, win);
   4889                 index++;
   4890                 winAdded = true;
   4891             }
   4892             if (DEBUG_WINDOW_MOVEMENT) Slog.v(TAG_WM, "Re-adding window at "
   4893                     + index + ": " + cwin);
   4894             cwin.mRebuilding = false;
   4895             windows.add(index, cwin);
   4896             index++;
   4897         }
   4898         if (!winAdded) {
   4899             if (DEBUG_WINDOW_MOVEMENT) Slog.v(TAG_WM, "Re-adding window at "
   4900                     + index + ": " + win);
   4901             win.mRebuilding = false;
   4902             windows.add(index, win);
   4903             index++;
   4904         }
   4905         mWindowsChanged = true;
   4906         return index;
   4907     }
   4908 
   4909     private final int reAddAppWindowsLocked(final DisplayContent displayContent, int index,
   4910                                             WindowToken token) {
   4911         final int NW = token.windows.size();
   4912         for (int i=0; i<NW; i++) {
   4913             final WindowState win = token.windows.get(i);
   4914             final DisplayContent winDisplayContent = win.getDisplayContent();
   4915             if (winDisplayContent == displayContent || winDisplayContent == null) {
   4916                 win.mDisplayContent = displayContent;
   4917                 index = reAddWindowLocked(index, win);
   4918             }
   4919         }
   4920         return index;
   4921     }
   4922 
   4923 
   4924     void moveStackWindowsLocked(DisplayContent displayContent) {
   4925         final WindowList windows = displayContent.getWindowList();
   4926         mTmpWindows.addAll(windows);
   4927 
   4928         rebuildAppWindowListLocked(displayContent);
   4929 
   4930         // Set displayContent.layoutNeeded if window order changed.
   4931         final int tmpSize = mTmpWindows.size();
   4932         final int winSize = windows.size();
   4933         int tmpNdx = 0, winNdx = 0;
   4934         while (tmpNdx < tmpSize && winNdx < winSize) {
   4935             // Skip over all exiting windows, they've been moved out of order.
   4936             WindowState tmp;
   4937             do {
   4938                 tmp = mTmpWindows.get(tmpNdx++);
   4939             } while (tmpNdx < tmpSize && tmp.mAppToken != null && tmp.mAppToken.mIsExiting);
   4940 
   4941             WindowState win;
   4942             do {
   4943                 win = windows.get(winNdx++);
   4944             } while (winNdx < winSize && win.mAppToken != null && win.mAppToken.mIsExiting);
   4945 
   4946             if (tmp != win) {
   4947                 // Window order changed.
   4948                 displayContent.layoutNeeded = true;
   4949                 break;
   4950             }
   4951         }
   4952         if (tmpNdx != winNdx) {
   4953             // One list was different from the other.
   4954             displayContent.layoutNeeded = true;
   4955         }
   4956         mTmpWindows.clear();
   4957 
   4958         if (!updateFocusedWindowLocked(UPDATE_FOCUS_WILL_PLACE_SURFACES,
   4959                 false /*updateInputWindows*/)) {
   4960             mLayersController.assignLayersLocked(displayContent.getWindowList());
   4961         }
   4962 
   4963         mInputMonitor.setUpdateInputWindowsNeededLw();
   4964         mWindowPlacerLocked.performSurfacePlacement();
   4965         mInputMonitor.updateInputWindowsLw(false /*force*/);
   4966         //dump();
   4967     }
   4968 
   4969     public void moveTaskToTop(int taskId) {
   4970         final long origId = Binder.clearCallingIdentity();
   4971         try {
   4972             synchronized(mWindowMap) {
   4973                 Task task = mTaskIdToTask.get(taskId);
   4974                 if (task == null) {
   4975                     // Normal behavior, addAppToken will be called next and task will be created.
   4976                     return;
   4977                 }
   4978                 final TaskStack stack = task.mStack;
   4979                 final DisplayContent displayContent = task.getDisplayContent();
   4980                 displayContent.moveStack(stack, true);
   4981                 if (displayContent.isDefaultDisplay) {
   4982                     final TaskStack homeStack = displayContent.getHomeStack();
   4983                     if (homeStack != stack) {
   4984                         // When a non-home stack moves to the top, the home stack moves to the
   4985                         // bottom.
   4986                         displayContent.moveStack(homeStack, false);
   4987                     }
   4988                 }
   4989                 stack.moveTaskToTop(task);
   4990                 if (mAppTransition.isTransitionSet()) {
   4991                     task.setSendingToBottom(false);
   4992                 }
   4993                 moveStackWindowsLocked(displayContent);
   4994             }
   4995         } finally {
   4996             Binder.restoreCallingIdentity(origId);
   4997         }
   4998     }
   4999 
   5000     public void moveTaskToBottom(int taskId) {
   5001         final long origId = Binder.clearCallingIdentity();
   5002         try {
   5003             synchronized(mWindowMap) {
   5004                 Task task = mTaskIdToTask.get(taskId);
   5005                 if (task == null) {
   5006                     Slog.e(TAG_WM, "moveTaskToBottom: taskId=" + taskId
   5007                             + " not found in mTaskIdToTask");
   5008                     return;
   5009                 }
   5010                 final TaskStack stack = task.mStack;
   5011                 stack.moveTaskToBottom(task);
   5012                 if (mAppTransition.isTransitionSet()) {
   5013                     task.setSendingToBottom(true);
   5014                 }
   5015                 moveStackWindowsLocked(stack.getDisplayContent());
   5016             }
   5017         } finally {
   5018             Binder.restoreCallingIdentity(origId);
   5019         }
   5020     }
   5021 
   5022     boolean isStackVisibleLocked(int stackId) {
   5023         final TaskStack stack = mStackIdToStack.get(stackId);
   5024         return (stack != null && stack.isVisibleLocked());
   5025     }
   5026 
   5027     public void setDockedStackCreateState(int mode, Rect bounds) {
   5028         synchronized (mWindowMap) {
   5029             setDockedStackCreateStateLocked(mode, bounds);
   5030         }
   5031     }
   5032 
   5033     void setDockedStackCreateStateLocked(int mode, Rect bounds) {
   5034         mDockedStackCreateMode = mode;
   5035         mDockedStackCreateBounds = bounds;
   5036     }
   5037 
   5038     /**
   5039      * Create a new TaskStack and place it on a DisplayContent.
   5040      * @param stackId The unique identifier of the new stack.
   5041      * @param displayId The unique identifier of the DisplayContent.
   5042      * @param onTop If true the stack will be place at the top of the display,
   5043      *              else at the bottom
   5044      * @return The initial bounds the stack was created with. null means fullscreen.
   5045      */
   5046     public Rect attachStack(int stackId, int displayId, boolean onTop) {
   5047         final long origId = Binder.clearCallingIdentity();
   5048         try {
   5049             synchronized (mWindowMap) {
   5050                 final DisplayContent displayContent = mDisplayContents.get(displayId);
   5051                 boolean attachedToDisplay = false;
   5052                 if (displayContent != null) {
   5053                     TaskStack stack = mStackIdToStack.get(stackId);
   5054                     if (stack == null) {
   5055                         if (DEBUG_STACK) Slog.d(TAG_WM, "attachStack: stackId=" + stackId);
   5056 
   5057                         stack = displayContent.getStackById(stackId);
   5058                         if (stack != null) {
   5059                             // It's already attached to the display. Detach and re-attach
   5060                             // because onTop might change, and be sure to clear mDeferDetach!
   5061                             displayContent.detachStack(stack);
   5062                             stack.mDeferDetach = false;
   5063                             attachedToDisplay = true;
   5064                         } else {
   5065                             stack = new TaskStack(this, stackId);
   5066                         }
   5067 
   5068                         mStackIdToStack.put(stackId, stack);
   5069                         if (stackId == DOCKED_STACK_ID) {
   5070                             getDefaultDisplayContentLocked().mDividerControllerLocked
   5071                                     .notifyDockedStackExistsChanged(true);
   5072                         }
   5073                     }
   5074                     if (!attachedToDisplay) {
   5075                         stack.attachDisplayContent(displayContent);
   5076                     }
   5077                     displayContent.attachStack(stack, onTop);
   5078                     if (stack.getRawFullscreen()) {
   5079                         return null;
   5080                     }
   5081                     Rect bounds = new Rect();
   5082                     stack.getRawBounds(bounds);
   5083                     return bounds;
   5084                 }
   5085             }
   5086         } finally {
   5087             Binder.restoreCallingIdentity(origId);
   5088         }
   5089         return null;
   5090     }
   5091 
   5092     void detachStackLocked(DisplayContent displayContent, TaskStack stack) {
   5093         displayContent.detachStack(stack);
   5094         stack.detachDisplay();
   5095         if (stack.mStackId == DOCKED_STACK_ID) {
   5096             getDefaultDisplayContentLocked().mDividerControllerLocked
   5097                     .notifyDockedStackExistsChanged(false);
   5098         }
   5099     }
   5100 
   5101     public void detachStack(int stackId) {
   5102         synchronized (mWindowMap) {
   5103             TaskStack stack = mStackIdToStack.get(stackId);
   5104             if (stack != null) {
   5105                 final DisplayContent displayContent = stack.getDisplayContent();
   5106                 if (displayContent != null) {
   5107                     if (stack.isAnimating()) {
   5108                         stack.mDeferDetach = true;
   5109                         return;
   5110                     }
   5111                     detachStackLocked(displayContent, stack);
   5112                 }
   5113             }
   5114         }
   5115     }
   5116 
   5117     public void removeStack(int stackId) {
   5118         synchronized (mWindowMap) {
   5119             mStackIdToStack.remove(stackId);
   5120         }
   5121     }
   5122 
   5123     public void removeTask(int taskId) {
   5124         synchronized (mWindowMap) {
   5125             Task task = mTaskIdToTask.get(taskId);
   5126             if (task == null) {
   5127                 if (DEBUG_STACK) Slog.i(TAG_WM, "removeTask: could not find taskId=" + taskId);
   5128                 return;
   5129             }
   5130             task.removeLocked();
   5131         }
   5132     }
   5133 
   5134     @Override
   5135     public void cancelTaskWindowTransition(int taskId) {
   5136         synchronized (mWindowMap) {
   5137             Task task = mTaskIdToTask.get(taskId);
   5138             if (task != null) {
   5139                 task.cancelTaskWindowTransition();
   5140             }
   5141         }
   5142     }
   5143 
   5144     @Override
   5145     public void cancelTaskThumbnailTransition(int taskId) {
   5146         synchronized (mWindowMap) {
   5147             Task task = mTaskIdToTask.get(taskId);
   5148             if (task != null) {
   5149                 task.cancelTaskThumbnailTransition();
   5150             }
   5151         }
   5152     }
   5153 
   5154     public void addTask(int taskId, int stackId, boolean toTop) {
   5155         synchronized (mWindowMap) {
   5156             if (DEBUG_STACK) Slog.i(TAG_WM, "addTask: adding taskId=" + taskId
   5157                     + " to " + (toTop ? "top" : "bottom"));
   5158             Task task = mTaskIdToTask.get(taskId);
   5159             if (task == null) {
   5160                 if (DEBUG_STACK) Slog.i(TAG_WM, "addTask: could not find taskId=" + taskId);
   5161                 return;
   5162             }
   5163             TaskStack stack = mStackIdToStack.get(stackId);
   5164             stack.addTask(task, toTop);
   5165             final DisplayContent displayContent = stack.getDisplayContent();
   5166             displayContent.layoutNeeded = true;
   5167             mWindowPlacerLocked.performSurfacePlacement();
   5168         }
   5169     }
   5170 
   5171     public void moveTaskToStack(int taskId, int stackId, boolean toTop) {
   5172         synchronized (mWindowMap) {
   5173             if (DEBUG_STACK) Slog.i(TAG_WM, "moveTaskToStack: moving taskId=" + taskId
   5174                     + " to stackId=" + stackId + " at " + (toTop ? "top" : "bottom"));
   5175             Task task = mTaskIdToTask.get(taskId);
   5176             if (task == null) {
   5177                 if (DEBUG_STACK) Slog.i(TAG_WM, "moveTaskToStack: could not find taskId=" + taskId);
   5178                 return;
   5179             }
   5180             TaskStack stack = mStackIdToStack.get(stackId);
   5181             if (stack == null) {
   5182                 if (DEBUG_STACK) Slog.i(TAG_WM, "moveTaskToStack: could not find stackId=" + stackId);
   5183                 return;
   5184             }
   5185             task.moveTaskToStack(stack, toTop);
   5186             final DisplayContent displayContent = stack.getDisplayContent();
   5187             displayContent.layoutNeeded = true;
   5188             mWindowPlacerLocked.performSurfacePlacement();
   5189         }
   5190     }
   5191 
   5192     public void getStackDockedModeBounds(int stackId, Rect bounds, boolean ignoreVisibility) {
   5193         synchronized (mWindowMap) {
   5194             final TaskStack stack = mStackIdToStack.get(stackId);
   5195             if (stack != null) {
   5196                 stack.getStackDockedModeBoundsLocked(bounds, ignoreVisibility);
   5197                 return;
   5198             }
   5199             bounds.setEmpty();
   5200         }
   5201     }
   5202 
   5203     @Override
   5204     public void getStackBounds(int stackId, Rect bounds) {
   5205         synchronized (mWindowMap) {
   5206             final TaskStack stack = mStackIdToStack.get(stackId);
   5207             if (stack != null) {
   5208                 stack.getBounds(bounds);
   5209                 return;
   5210             }
   5211             bounds.setEmpty();
   5212         }
   5213     }
   5214 
   5215     /**
   5216      * Re-sizes a stack and its containing tasks.
   5217      * @param stackId Id of stack to resize.
   5218      * @param bounds New stack bounds. Passing in null sets the bounds to fullscreen.
   5219      * @param configs Configurations for tasks in the resized stack, keyed by task id.
   5220      * @param taskBounds Bounds for tasks in the resized stack, keyed by task id.
   5221      * @return True if the stack is now fullscreen.
   5222      * */
   5223     public boolean resizeStack(int stackId, Rect bounds,
   5224             SparseArray<Configuration> configs, SparseArray<Rect> taskBounds,
   5225             SparseArray<Rect> taskTempInsetBounds) {
   5226         synchronized (mWindowMap) {
   5227             final TaskStack stack = mStackIdToStack.get(stackId);
   5228             if (stack == null) {
   5229                 throw new IllegalArgumentException("resizeStack: stackId " + stackId
   5230                         + " not found.");
   5231             }
   5232             if (stack.setBounds(bounds, configs, taskBounds, taskTempInsetBounds)
   5233                     && stack.isVisibleLocked()) {
   5234                 stack.getDisplayContent().layoutNeeded = true;
   5235                 mWindowPlacerLocked.performSurfacePlacement();
   5236             }
   5237             return stack.getRawFullscreen();
   5238         }
   5239     }
   5240 
   5241     public void prepareFreezingTaskBounds(int stackId) {
   5242         synchronized (mWindowMap) {
   5243             final TaskStack stack = mStackIdToStack.get(stackId);
   5244             if (stack == null) {
   5245                 throw new IllegalArgumentException("prepareFreezingTaskBounds: stackId " + stackId
   5246                         + " not found.");
   5247             }
   5248             stack.prepareFreezingTaskBounds();
   5249         }
   5250     }
   5251 
   5252     public void positionTaskInStack(int taskId, int stackId, int position, Rect bounds,
   5253             Configuration config) {
   5254         synchronized (mWindowMap) {
   5255             if (DEBUG_STACK) Slog.i(TAG_WM, "positionTaskInStack: positioning taskId=" + taskId
   5256                     + " in stackId=" + stackId + " at " + position);
   5257             Task task = mTaskIdToTask.get(taskId);
   5258             if (task == null) {
   5259                 if (DEBUG_STACK) Slog.i(TAG_WM,
   5260                         "positionTaskInStack: could not find taskId=" + taskId);
   5261                 return;
   5262             }
   5263             TaskStack stack = mStackIdToStack.get(stackId);
   5264             if (stack == null) {
   5265                 if (DEBUG_STACK) Slog.i(TAG_WM,
   5266                         "positionTaskInStack: could not find stackId=" + stackId);
   5267                 return;
   5268             }
   5269             task.positionTaskInStack(stack, position, bounds, config);
   5270             final DisplayContent displayContent = stack.getDisplayContent();
   5271             displayContent.layoutNeeded = true;
   5272             mWindowPlacerLocked.performSurfacePlacement();
   5273         }
   5274     }
   5275 
   5276     /**
   5277      * Re-sizes the specified task and its containing windows.
   5278      * Returns a {@link Configuration} object that contains configurations settings
   5279      * that should be overridden due to the operation.
   5280      */
   5281     public void resizeTask(int taskId, Rect bounds, Configuration configuration,
   5282             boolean relayout, boolean forced) {
   5283         synchronized (mWindowMap) {
   5284             Task task = mTaskIdToTask.get(taskId);
   5285             if (task == null) {
   5286                 throw new IllegalArgumentException("resizeTask: taskId " + taskId
   5287                         + " not found.");
   5288             }
   5289 
   5290             if (task.resizeLocked(bounds, configuration, forced) && relayout) {
   5291                 task.getDisplayContent().layoutNeeded = true;
   5292                 mWindowPlacerLocked.performSurfacePlacement();
   5293             }
   5294         }
   5295     }
   5296 
   5297     /**
   5298      * Puts a specific task into docked drag resizing mode. See {@link DragResizeMode}.
   5299      *
   5300      * @param taskId The id of the task to put into drag resize mode.
   5301      * @param resizing Whether to put the task into drag resize mode.
   5302      */
   5303     public void setTaskDockedResizing(int taskId, boolean resizing) {
   5304         synchronized (mWindowMap) {
   5305             Task task = mTaskIdToTask.get(taskId);
   5306             if (task == null) {
   5307                 Slog.w(TAG, "setTaskDockedResizing: taskId " + taskId + " not found.");
   5308                 return;
   5309             }
   5310             task.setDragResizing(resizing, DRAG_RESIZE_MODE_DOCKED_DIVIDER);
   5311         }
   5312     }
   5313 
   5314     public void scrollTask(int taskId, Rect bounds) {
   5315         synchronized (mWindowMap) {
   5316             Task task = mTaskIdToTask.get(taskId);
   5317             if (task == null) {
   5318                 throw new IllegalArgumentException("scrollTask: taskId " + taskId
   5319                         + " not found.");
   5320             }
   5321 
   5322             if (task.scrollLocked(bounds)) {
   5323                 task.getDisplayContent().layoutNeeded = true;
   5324                 mInputMonitor.setUpdateInputWindowsNeededLw();
   5325                 mWindowPlacerLocked.performSurfacePlacement();
   5326             }
   5327         }
   5328     }
   5329     /**
   5330      * Starts deferring layout passes. Useful when doing multiple changes but to optimize
   5331      * performance, only one layout pass should be done. This can be called multiple times, and
   5332      * layouting will be resumed once the last caller has called {@link #continueSurfaceLayout}
   5333      */
   5334     public void deferSurfaceLayout() {
   5335         synchronized (mWindowMap) {
   5336             mWindowPlacerLocked.deferLayout();
   5337         }
   5338     }
   5339 
   5340     /**
   5341      * Resumes layout passes after deferring them. See {@link #deferSurfaceLayout()}
   5342      */
   5343     public void continueSurfaceLayout() {
   5344         synchronized (mWindowMap) {
   5345             mWindowPlacerLocked.continueLayout();
   5346         }
   5347     }
   5348 
   5349     public void getTaskBounds(int taskId, Rect bounds) {
   5350         synchronized (mWindowMap) {
   5351             Task task = mTaskIdToTask.get(taskId);
   5352             if (task != null) {
   5353                 task.getBounds(bounds);
   5354                 return;
   5355             }
   5356             bounds.setEmpty();
   5357         }
   5358     }
   5359 
   5360     /** Return true if the input task id represents a valid window manager task. */
   5361     public boolean isValidTaskId(int taskId) {
   5362         synchronized (mWindowMap) {
   5363             return mTaskIdToTask.get(taskId) != null;
   5364         }
   5365     }
   5366 
   5367     // -------------------------------------------------------------
   5368     // Misc IWindowSession methods
   5369     // -------------------------------------------------------------
   5370 
   5371     @Override
   5372     public void startFreezingScreen(int exitAnim, int enterAnim) {
   5373         if (!checkCallingPermission(android.Manifest.permission.FREEZE_SCREEN,
   5374                 "startFreezingScreen()")) {
   5375             throw new SecurityException("Requires FREEZE_SCREEN permission");
   5376         }
   5377 
   5378         synchronized(mWindowMap) {
   5379             if (!mClientFreezingScreen) {
   5380                 mClientFreezingScreen = true;
   5381                 final long origId = Binder.clearCallingIdentity();
   5382                 try {
   5383                     startFreezingDisplayLocked(false, exitAnim, enterAnim);
   5384                     mH.removeMessages(H.CLIENT_FREEZE_TIMEOUT);
   5385                     mH.sendEmptyMessageDelayed(H.CLIENT_FREEZE_TIMEOUT, 5000);
   5386                 } finally {
   5387                     Binder.restoreCallingIdentity(origId);
   5388                 }
   5389             }
   5390         }
   5391     }
   5392 
   5393     @Override
   5394     public void stopFreezingScreen() {
   5395         if (!checkCallingPermission(android.Manifest.permission.FREEZE_SCREEN,
   5396                 "stopFreezingScreen()")) {
   5397             throw new SecurityException("Requires FREEZE_SCREEN permission");
   5398         }
   5399 
   5400         synchronized(mWindowMap) {
   5401             if (mClientFreezingScreen) {
   5402                 mClientFreezingScreen = false;
   5403                 mLastFinishedFreezeSource = "client";
   5404                 final long origId = Binder.clearCallingIdentity();
   5405                 try {
   5406                     stopFreezingDisplayLocked();
   5407                 } finally {
   5408                     Binder.restoreCallingIdentity(origId);
   5409                 }
   5410             }
   5411         }
   5412     }
   5413 
   5414     @Override
   5415     public void disableKeyguard(IBinder token, String tag) {
   5416         if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.DISABLE_KEYGUARD)
   5417             != PackageManager.PERMISSION_GRANTED) {
   5418             throw new SecurityException("Requires DISABLE_KEYGUARD permission");
   5419         }
   5420         // If this isn't coming from the system then don't allow disabling the lockscreen
   5421         // to bypass security.
   5422         if (Binder.getCallingUid() != Process.SYSTEM_UID && isKeyguardSecure()) {
   5423             Log.d(TAG_WM, "current mode is SecurityMode, ignore disableKeyguard");
   5424             return;
   5425         }
   5426 
   5427         // If this isn't coming from the current user, ignore it.
   5428         if (Binder.getCallingUserHandle().getIdentifier() != mCurrentUserId) {
   5429             Log.d(TAG_WM, "non-current user, ignore disableKeyguard");
   5430             return;
   5431         }
   5432 
   5433         if (token == null) {
   5434             throw new IllegalArgumentException("token == null");
   5435         }
   5436 
   5437         mKeyguardDisableHandler.sendMessage(mKeyguardDisableHandler.obtainMessage(
   5438                 KeyguardDisableHandler.KEYGUARD_DISABLE, new Pair<IBinder, String>(token, tag)));
   5439     }
   5440 
   5441     @Override
   5442     public void reenableKeyguard(IBinder token) {
   5443         if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.DISABLE_KEYGUARD)
   5444             != PackageManager.PERMISSION_GRANTED) {
   5445             throw new SecurityException("Requires DISABLE_KEYGUARD permission");
   5446         }
   5447 
   5448         if (token == null) {
   5449             throw new IllegalArgumentException("token == null");
   5450         }
   5451 
   5452         mKeyguardDisableHandler.sendMessage(mKeyguardDisableHandler.obtainMessage(
   5453                 KeyguardDisableHandler.KEYGUARD_REENABLE, token));
   5454     }
   5455 
   5456     /**
   5457      * @see android.app.KeyguardManager#exitKeyguardSecurely
   5458      */
   5459     @Override
   5460     public void exitKeyguardSecurely(final IOnKeyguardExitResult callback) {
   5461         if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.DISABLE_KEYGUARD)
   5462             != PackageManager.PERMISSION_GRANTED) {
   5463             throw new SecurityException("Requires DISABLE_KEYGUARD permission");
   5464         }
   5465 
   5466         if (callback == null) {
   5467             throw new IllegalArgumentException("callback == null");
   5468         }
   5469 
   5470         mPolicy.exitKeyguardSecurely(new WindowManagerPolicy.OnKeyguardExitResult() {
   5471             @Override
   5472             public void onKeyguardExitResult(boolean success) {
   5473                 try {
   5474                     callback.onKeyguardExitResult(success);
   5475                 } catch (RemoteException e) {
   5476                     // Client has died, we don't care.
   5477                 }
   5478             }
   5479         });
   5480     }
   5481 
   5482     @Override
   5483     public boolean inKeyguardRestrictedInputMode() {
   5484         return mPolicy.inKeyguardRestrictedKeyInputMode();
   5485     }
   5486 
   5487     @Override
   5488     public boolean isKeyguardLocked() {
   5489         return mPolicy.isKeyguardLocked();
   5490     }
   5491 
   5492     @Override
   5493     public boolean isKeyguardSecure() {
   5494         int userId = UserHandle.getCallingUserId();
   5495         long origId = Binder.clearCallingIdentity();
   5496         try {
   5497             return mPolicy.isKeyguardSecure(userId);
   5498         } finally {
   5499             Binder.restoreCallingIdentity(origId);
   5500         }
   5501     }
   5502 
   5503     @Override
   5504     public void dismissKeyguard() {
   5505         if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.DISABLE_KEYGUARD)
   5506                 != PackageManager.PERMISSION_GRANTED) {
   5507             throw new SecurityException("Requires DISABLE_KEYGUARD permission");
   5508         }
   5509         synchronized(mWindowMap) {
   5510             mPolicy.dismissKeyguardLw();
   5511         }
   5512     }
   5513 
   5514     @Override
   5515     public void keyguardGoingAway(int flags) {
   5516         if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.DISABLE_KEYGUARD)
   5517                 != PackageManager.PERMISSION_GRANTED) {
   5518             throw new SecurityException("Requires DISABLE_KEYGUARD permission");
   5519         }
   5520         if (DEBUG_KEYGUARD) Slog.d(TAG_WM,
   5521                 "keyguardGoingAway: flags=0x" + Integer.toHexString(flags));
   5522         synchronized (mWindowMap) {
   5523             mAnimator.mKeyguardGoingAway = true;
   5524             mAnimator.mKeyguardGoingAwayFlags = flags;
   5525             mWindowPlacerLocked.requestTraversal();
   5526         }
   5527     }
   5528 
   5529     public void keyguardWaitingForActivityDrawn() {
   5530         if (DEBUG_KEYGUARD) Slog.d(TAG_WM, "keyguardWaitingForActivityDrawn");
   5531         synchronized (mWindowMap) {
   5532             mKeyguardWaitingForActivityDrawn = true;
   5533         }
   5534     }
   5535 
   5536     public void notifyActivityDrawnForKeyguard() {
   5537         if (DEBUG_KEYGUARD) Slog.d(TAG_WM, "notifyActivityDrawnForKeyguard: waiting="
   5538                 + mKeyguardWaitingForActivityDrawn + " Callers=" + Debug.getCallers(5));
   5539         synchronized (mWindowMap) {
   5540             if (mKeyguardWaitingForActivityDrawn) {
   5541                 mPolicy.notifyActivityDrawnForKeyguardLw();
   5542                 mKeyguardWaitingForActivityDrawn = false;
   5543             }
   5544         }
   5545     }
   5546 
   5547     void showGlobalActions() {
   5548         mPolicy.showGlobalActions();
   5549     }
   5550 
   5551     @Override
   5552     public void closeSystemDialogs(String reason) {
   5553         synchronized(mWindowMap) {
   5554             final int numDisplays = mDisplayContents.size();
   5555             for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   5556                 final WindowList windows = mDisplayContents.valueAt(displayNdx).getWindowList();
   5557                 final int numWindows = windows.size();
   5558                 for (int winNdx = 0; winNdx < numWindows; ++winNdx) {
   5559                     final WindowState w = windows.get(winNdx);
   5560                     if (w.mHasSurface) {
   5561                         try {
   5562                             w.mClient.closeSystemDialogs(reason);
   5563                         } catch (RemoteException e) {
   5564                         }
   5565                     }
   5566                 }
   5567             }
   5568         }
   5569     }
   5570 
   5571     static float fixScale(float scale) {
   5572         if (scale < 0) scale = 0;
   5573         else if (scale > 20) scale = 20;
   5574         return Math.abs(scale);
   5575     }
   5576 
   5577     @Override
   5578     public void setAnimationScale(int which, float scale) {
   5579         if (!checkCallingPermission(android.Manifest.permission.SET_ANIMATION_SCALE,
   5580                 "setAnimationScale()")) {
   5581             throw new SecurityException("Requires SET_ANIMATION_SCALE permission");
   5582         }
   5583 
   5584         scale = fixScale(scale);
   5585         switch (which) {
   5586             case 0: mWindowAnimationScaleSetting = scale; break;
   5587             case 1: mTransitionAnimationScaleSetting = scale; break;
   5588             case 2: mAnimatorDurationScaleSetting = scale; break;
   5589         }
   5590 
   5591         // Persist setting
   5592         mH.sendEmptyMessage(H.PERSIST_ANIMATION_SCALE);
   5593     }
   5594 
   5595     @Override
   5596     public void setAnimationScales(float[] scales) {
   5597         if (!checkCallingPermission(android.Manifest.permission.SET_ANIMATION_SCALE,
   5598                 "setAnimationScale()")) {
   5599             throw new SecurityException("Requires SET_ANIMATION_SCALE permission");
   5600         }
   5601 
   5602         if (scales != null) {
   5603             if (scales.length >= 1) {
   5604                 mWindowAnimationScaleSetting = fixScale(scales[0]);
   5605             }
   5606             if (scales.length >= 2) {
   5607                 mTransitionAnimationScaleSetting = fixScale(scales[1]);
   5608             }
   5609             if (scales.length >= 3) {
   5610                 mAnimatorDurationScaleSetting = fixScale(scales[2]);
   5611                 dispatchNewAnimatorScaleLocked(null);
   5612             }
   5613         }
   5614 
   5615         // Persist setting
   5616         mH.sendEmptyMessage(H.PERSIST_ANIMATION_SCALE);
   5617     }
   5618 
   5619     private void setAnimatorDurationScale(float scale) {
   5620         mAnimatorDurationScaleSetting = scale;
   5621         ValueAnimator.setDurationScale(scale);
   5622     }
   5623 
   5624     public float getWindowAnimationScaleLocked() {
   5625         return mAnimationsDisabled ? 0 : mWindowAnimationScaleSetting;
   5626     }
   5627 
   5628     public float getTransitionAnimationScaleLocked() {
   5629         return mAnimationsDisabled ? 0 : mTransitionAnimationScaleSetting;
   5630     }
   5631 
   5632     @Override
   5633     public float getAnimationScale(int which) {
   5634         switch (which) {
   5635             case 0: return mWindowAnimationScaleSetting;
   5636             case 1: return mTransitionAnimationScaleSetting;
   5637             case 2: return mAnimatorDurationScaleSetting;
   5638         }
   5639         return 0;
   5640     }
   5641 
   5642     @Override
   5643     public float[] getAnimationScales() {
   5644         return new float[] { mWindowAnimationScaleSetting, mTransitionAnimationScaleSetting,
   5645                 mAnimatorDurationScaleSetting };
   5646     }
   5647 
   5648     @Override
   5649     public float getCurrentAnimatorScale() {
   5650         synchronized(mWindowMap) {
   5651             return mAnimationsDisabled ? 0 : mAnimatorDurationScaleSetting;
   5652         }
   5653     }
   5654 
   5655     void dispatchNewAnimatorScaleLocked(Session session) {
   5656         mH.obtainMessage(H.NEW_ANIMATOR_SCALE, session).sendToTarget();
   5657     }
   5658 
   5659     @Override
   5660     public void registerPointerEventListener(PointerEventListener listener) {
   5661         mPointerEventDispatcher.registerInputEventListener(listener);
   5662     }
   5663 
   5664     @Override
   5665     public void unregisterPointerEventListener(PointerEventListener listener) {
   5666         mPointerEventDispatcher.unregisterInputEventListener(listener);
   5667     }
   5668 
   5669     // Called by window manager policy. Not exposed externally.
   5670     @Override
   5671     public int getLidState() {
   5672         int sw = mInputManager.getSwitchState(-1, InputDevice.SOURCE_ANY,
   5673                 InputManagerService.SW_LID);
   5674         if (sw > 0) {
   5675             // Switch state: AKEY_STATE_DOWN or AKEY_STATE_VIRTUAL.
   5676             return LID_CLOSED;
   5677         } else if (sw == 0) {
   5678             // Switch state: AKEY_STATE_UP.
   5679             return LID_OPEN;
   5680         } else {
   5681             // Switch state: AKEY_STATE_UNKNOWN.
   5682             return LID_ABSENT;
   5683         }
   5684     }
   5685 
   5686     // Called by window manager policy. Not exposed externally.
   5687     @Override
   5688     public void lockDeviceNow() {
   5689         lockNow(null);
   5690     }
   5691 
   5692     // Called by window manager policy. Not exposed externally.
   5693     @Override
   5694     public int getCameraLensCoverState() {
   5695         int sw = mInputManager.getSwitchState(-1, InputDevice.SOURCE_ANY,
   5696                 InputManagerService.SW_CAMERA_LENS_COVER);
   5697         if (sw > 0) {
   5698             // Switch state: AKEY_STATE_DOWN or AKEY_STATE_VIRTUAL.
   5699             return CAMERA_LENS_COVERED;
   5700         } else if (sw == 0) {
   5701             // Switch state: AKEY_STATE_UP.
   5702             return CAMERA_LENS_UNCOVERED;
   5703         } else {
   5704             // Switch state: AKEY_STATE_UNKNOWN.
   5705             return CAMERA_LENS_COVER_ABSENT;
   5706         }
   5707     }
   5708 
   5709     // Called by window manager policy.  Not exposed externally.
   5710     @Override
   5711     public void switchInputMethod(boolean forwardDirection) {
   5712         final InputMethodManagerInternal inputMethodManagerInternal =
   5713                 LocalServices.getService(InputMethodManagerInternal.class);
   5714         if (inputMethodManagerInternal != null) {
   5715             inputMethodManagerInternal.switchInputMethod(forwardDirection);
   5716         }
   5717     }
   5718 
   5719     // Called by window manager policy.  Not exposed externally.
   5720     @Override
   5721     public void shutdown(boolean confirm) {
   5722         ShutdownThread.shutdown(mContext, PowerManager.SHUTDOWN_USER_REQUESTED, confirm);
   5723     }
   5724 
   5725     // Called by window manager policy.  Not exposed externally.
   5726     @Override
   5727     public void reboot(boolean confirm) {
   5728         ShutdownThread.reboot(mContext, PowerManager.SHUTDOWN_USER_REQUESTED, confirm);
   5729     }
   5730 
   5731     // Called by window manager policy.  Not exposed externally.
   5732     @Override
   5733     public void rebootSafeMode(boolean confirm) {
   5734         ShutdownThread.rebootSafeMode(mContext, confirm);
   5735     }
   5736 
   5737     public void setCurrentProfileIds(final int[] currentProfileIds) {
   5738         synchronized (mWindowMap) {
   5739             mCurrentProfileIds = currentProfileIds;
   5740         }
   5741     }
   5742 
   5743     public void setCurrentUser(final int newUserId, final int[] currentProfileIds) {
   5744         synchronized (mWindowMap) {
   5745             mCurrentUserId = newUserId;
   5746             mCurrentProfileIds = currentProfileIds;
   5747             mAppTransition.setCurrentUser(newUserId);
   5748             mPolicy.setCurrentUserLw(newUserId);
   5749 
   5750             // If keyguard was disabled, re-enable it
   5751             // TODO: Keep track of keyguardEnabled state per user and use here...
   5752             // e.g. enabled = mKeyguardDisableHandler.getEnabledStateForUser(newUserId);
   5753             mPolicy.enableKeyguard(true);
   5754 
   5755             // Hide windows that should not be seen by the new user.
   5756             final int numDisplays = mDisplayContents.size();
   5757             for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   5758                 final DisplayContent displayContent = mDisplayContents.valueAt(displayNdx);
   5759                 displayContent.switchUserStacks();
   5760                 rebuildAppWindowListLocked(displayContent);
   5761             }
   5762             mWindowPlacerLocked.performSurfacePlacement();
   5763 
   5764             // Notify whether the docked stack exists for the current user
   5765             final DisplayContent displayContent = getDefaultDisplayContentLocked();
   5766             displayContent.mDividerControllerLocked
   5767                     .notifyDockedStackExistsChanged(hasDockedTasksForUser(newUserId));
   5768 
   5769             // If the display is already prepared, update the density.
   5770             // Otherwise, we'll update it when it's prepared.
   5771             if (mDisplayReady) {
   5772                 final int forcedDensity = getForcedDisplayDensityForUserLocked(newUserId);
   5773                 final int targetDensity = forcedDensity != 0 ? forcedDensity
   5774                         : displayContent.mInitialDisplayDensity;
   5775                 setForcedDisplayDensityLocked(displayContent, targetDensity);
   5776             }
   5777         }
   5778     }
   5779 
   5780     /**
   5781      * Returns whether there is a docked task for the current user.
   5782      */
   5783     boolean hasDockedTasksForUser(int userId) {
   5784         final TaskStack stack = mStackIdToStack.get(DOCKED_STACK_ID);
   5785         if (stack == null) {
   5786             return false;
   5787         }
   5788 
   5789         final ArrayList<Task> tasks = stack.getTasks();
   5790         boolean hasUserTask = false;
   5791         for (int i = tasks.size() - 1; i >= 0 && !hasUserTask; i--) {
   5792             final Task task = tasks.get(i);
   5793             hasUserTask = (task.mUserId == userId);
   5794         }
   5795         return hasUserTask;
   5796     }
   5797 
   5798     /* Called by WindowState */
   5799     boolean isCurrentProfileLocked(int userId) {
   5800         if (userId == mCurrentUserId) return true;
   5801         for (int i = 0; i < mCurrentProfileIds.length; i++) {
   5802             if (mCurrentProfileIds[i] == userId) return true;
   5803         }
   5804         return false;
   5805     }
   5806 
   5807     public void enableScreenAfterBoot() {
   5808         synchronized(mWindowMap) {
   5809             if (DEBUG_BOOT) {
   5810                 RuntimeException here = new RuntimeException("here");
   5811                 here.fillInStackTrace();
   5812                 Slog.i(TAG_WM, "enableScreenAfterBoot: mDisplayEnabled=" + mDisplayEnabled
   5813                         + " mForceDisplayEnabled=" + mForceDisplayEnabled
   5814                         + " mShowingBootMessages=" + mShowingBootMessages
   5815                         + " mSystemBooted=" + mSystemBooted, here);
   5816             }
   5817             if (mSystemBooted) {
   5818                 return;
   5819             }
   5820             mSystemBooted = true;
   5821             hideBootMessagesLocked();
   5822             // If the screen still doesn't come up after 30 seconds, give
   5823             // up and turn it on.
   5824             mH.sendEmptyMessageDelayed(H.BOOT_TIMEOUT, 30*1000);
   5825         }
   5826 
   5827         mPolicy.systemBooted();
   5828 
   5829         performEnableScreen();
   5830     }
   5831 
   5832     @Override
   5833     public void enableScreenIfNeeded() {
   5834         synchronized (mWindowMap) {
   5835             enableScreenIfNeededLocked();
   5836         }
   5837     }
   5838 
   5839     void enableScreenIfNeededLocked() {
   5840         if (DEBUG_BOOT) {
   5841             RuntimeException here = new RuntimeException("here");
   5842             here.fillInStackTrace();
   5843             Slog.i(TAG_WM, "enableScreenIfNeededLocked: mDisplayEnabled=" + mDisplayEnabled
   5844                     + " mForceDisplayEnabled=" + mForceDisplayEnabled
   5845                     + " mShowingBootMessages=" + mShowingBootMessages
   5846                     + " mSystemBooted=" + mSystemBooted, here);
   5847         }
   5848         if (mDisplayEnabled) {
   5849             return;
   5850         }
   5851         if (!mSystemBooted && !mShowingBootMessages) {
   5852             return;
   5853         }
   5854         mH.sendEmptyMessage(H.ENABLE_SCREEN);
   5855     }
   5856 
   5857     public void performBootTimeout() {
   5858         synchronized(mWindowMap) {
   5859             if (mDisplayEnabled) {
   5860                 return;
   5861             }
   5862             Slog.w(TAG_WM, "***** BOOT TIMEOUT: forcing display enabled");
   5863             mForceDisplayEnabled = true;
   5864         }
   5865         performEnableScreen();
   5866     }
   5867 
   5868     private boolean checkWaitingForWindowsLocked() {
   5869 
   5870         boolean haveBootMsg = false;
   5871         boolean haveApp = false;
   5872         // if the wallpaper service is disabled on the device, we're never going to have
   5873         // wallpaper, don't bother waiting for it
   5874         boolean haveWallpaper = false;
   5875         boolean wallpaperEnabled = mContext.getResources().getBoolean(
   5876                 com.android.internal.R.bool.config_enableWallpaperService)
   5877                 && !mOnlyCore;
   5878         boolean haveKeyguard = true;
   5879         // TODO(multidisplay): Expand to all displays?
   5880         final WindowList windows = getDefaultWindowListLocked();
   5881         final int N = windows.size();
   5882         for (int i=0; i<N; i++) {
   5883             WindowState w = windows.get(i);
   5884             if (w.isVisibleLw() && !w.mObscured && !w.isDrawnLw()) {
   5885                 return true;
   5886             }
   5887             if (w.isDrawnLw()) {
   5888                 if (w.mAttrs.type == TYPE_BOOT_PROGRESS) {
   5889                     haveBootMsg = true;
   5890                 } else if (w.mAttrs.type == TYPE_APPLICATION
   5891                         || w.mAttrs.type == TYPE_DRAWN_APPLICATION) {
   5892                     haveApp = true;
   5893                 } else if (w.mAttrs.type == TYPE_WALLPAPER) {
   5894                     haveWallpaper = true;
   5895                 } else if (w.mAttrs.type == TYPE_STATUS_BAR) {
   5896                     haveKeyguard = mPolicy.isKeyguardDrawnLw();
   5897                 }
   5898             }
   5899         }
   5900 
   5901         if (DEBUG_SCREEN_ON || DEBUG_BOOT) {
   5902             Slog.i(TAG_WM, "******** booted=" + mSystemBooted + " msg=" + mShowingBootMessages
   5903                     + " haveBoot=" + haveBootMsg + " haveApp=" + haveApp
   5904                     + " haveWall=" + haveWallpaper + " wallEnabled=" + wallpaperEnabled
   5905                     + " haveKeyguard=" + haveKeyguard);
   5906         }
   5907 
   5908         // If we are turning on the screen to show the boot message,
   5909         // don't do it until the boot message is actually displayed.
   5910         if (!mSystemBooted && !haveBootMsg) {
   5911             return true;
   5912         }
   5913 
   5914         // If we are turning on the screen after the boot is completed
   5915         // normally, don't do so until we have the application and
   5916         // wallpaper.
   5917         if (mSystemBooted && ((!haveApp && !haveKeyguard) ||
   5918                 (wallpaperEnabled && !haveWallpaper))) {
   5919             return true;
   5920         }
   5921 
   5922         return false;
   5923     }
   5924 
   5925     public void performEnableScreen() {
   5926         synchronized(mWindowMap) {
   5927             if (DEBUG_BOOT) Slog.i(TAG_WM, "performEnableScreen: mDisplayEnabled=" + mDisplayEnabled
   5928                     + " mForceDisplayEnabled=" + mForceDisplayEnabled
   5929                     + " mShowingBootMessages=" + mShowingBootMessages
   5930                     + " mSystemBooted=" + mSystemBooted
   5931                     + " mOnlyCore=" + mOnlyCore,
   5932                     new RuntimeException("here").fillInStackTrace());
   5933             if (mDisplayEnabled) {
   5934                 return;
   5935             }
   5936             if (!mSystemBooted && !mShowingBootMessages) {
   5937                 return;
   5938             }
   5939 
   5940             // Don't enable the screen until all existing windows have been drawn.
   5941             if (!mForceDisplayEnabled && checkWaitingForWindowsLocked()) {
   5942                 return;
   5943             }
   5944 
   5945             if (!mBootAnimationStopped) {
   5946                 // Do this one time.
   5947                 Trace.asyncTraceBegin(Trace.TRACE_TAG_WINDOW_MANAGER, "Stop bootanim", 0);
   5948                 try {
   5949                     IBinder surfaceFlinger = ServiceManager.getService("SurfaceFlinger");
   5950                     if (surfaceFlinger != null) {
   5951                         //Slog.i(TAG_WM, "******* TELLING SURFACE FLINGER WE ARE BOOTED!");
   5952                         Parcel data = Parcel.obtain();
   5953                         data.writeInterfaceToken("android.ui.ISurfaceComposer");
   5954                         surfaceFlinger.transact(IBinder.FIRST_CALL_TRANSACTION, // BOOT_FINISHED
   5955                                 data, null, 0);
   5956                         data.recycle();
   5957                     }
   5958                 } catch (RemoteException ex) {
   5959                     Slog.e(TAG_WM, "Boot completed: SurfaceFlinger is dead!");
   5960                 }
   5961                 mBootAnimationStopped = true;
   5962             }
   5963 
   5964             if (!mForceDisplayEnabled && !checkBootAnimationCompleteLocked()) {
   5965                 if (DEBUG_BOOT) Slog.i(TAG_WM, "performEnableScreen: Waiting for anim complete");
   5966                 return;
   5967             }
   5968 
   5969             EventLog.writeEvent(EventLogTags.WM_BOOT_ANIMATION_DONE, SystemClock.uptimeMillis());
   5970             Trace.asyncTraceEnd(Trace.TRACE_TAG_WINDOW_MANAGER, "Stop bootanim", 0);
   5971             mDisplayEnabled = true;
   5972             if (DEBUG_SCREEN_ON || DEBUG_BOOT) Slog.i(TAG_WM, "******************** ENABLING SCREEN!");
   5973 
   5974             // Enable input dispatch.
   5975             mInputMonitor.setEventDispatchingLw(mEventDispatchingEnabled);
   5976         }
   5977 
   5978         try {
   5979             mActivityManager.bootAnimationComplete();
   5980         } catch (RemoteException e) {
   5981         }
   5982 
   5983         mPolicy.enableScreenAfterBoot();
   5984 
   5985         // Make sure the last requested orientation has been applied.
   5986         updateRotationUnchecked(false, false);
   5987     }
   5988 
   5989     private boolean checkBootAnimationCompleteLocked() {
   5990         if (SystemService.isRunning(BOOT_ANIMATION_SERVICE)) {
   5991             mH.removeMessages(H.CHECK_IF_BOOT_ANIMATION_FINISHED);
   5992             mH.sendEmptyMessageDelayed(H.CHECK_IF_BOOT_ANIMATION_FINISHED,
   5993                     BOOT_ANIMATION_POLL_INTERVAL);
   5994             if (DEBUG_BOOT) Slog.i(TAG_WM, "checkBootAnimationComplete: Waiting for anim complete");
   5995             return false;
   5996         }
   5997         if (DEBUG_BOOT) Slog.i(TAG_WM, "checkBootAnimationComplete: Animation complete!");
   5998         return true;
   5999     }
   6000 
   6001     public void showBootMessage(final CharSequence msg, final boolean always) {
   6002         boolean first = false;
   6003         synchronized(mWindowMap) {
   6004             if (DEBUG_BOOT) {
   6005                 RuntimeException here = new RuntimeException("here");
   6006                 here.fillInStackTrace();
   6007                 Slog.i(TAG_WM, "showBootMessage: msg=" + msg + " always=" + always
   6008                         + " mAllowBootMessages=" + mAllowBootMessages
   6009                         + " mShowingBootMessages=" + mShowingBootMessages
   6010                         + " mSystemBooted=" + mSystemBooted, here);
   6011             }
   6012             if (!mAllowBootMessages) {
   6013                 return;
   6014             }
   6015             if (!mShowingBootMessages) {
   6016                 if (!always) {
   6017                     return;
   6018                 }
   6019                 first = true;
   6020             }
   6021             if (mSystemBooted) {
   6022                 return;
   6023             }
   6024             mShowingBootMessages = true;
   6025             mPolicy.showBootMessage(msg, always);
   6026         }
   6027         if (first) {
   6028             performEnableScreen();
   6029         }
   6030     }
   6031 
   6032     public void hideBootMessagesLocked() {
   6033         if (DEBUG_BOOT) {
   6034             RuntimeException here = new RuntimeException("here");
   6035             here.fillInStackTrace();
   6036             Slog.i(TAG_WM, "hideBootMessagesLocked: mDisplayEnabled=" + mDisplayEnabled
   6037                     + " mForceDisplayEnabled=" + mForceDisplayEnabled
   6038                     + " mShowingBootMessages=" + mShowingBootMessages
   6039                     + " mSystemBooted=" + mSystemBooted, here);
   6040         }
   6041         if (mShowingBootMessages) {
   6042             mShowingBootMessages = false;
   6043             mPolicy.hideBootMessages();
   6044         }
   6045     }
   6046 
   6047     @Override
   6048     public void setInTouchMode(boolean mode) {
   6049         synchronized(mWindowMap) {
   6050             mInTouchMode = mode;
   6051         }
   6052     }
   6053 
   6054     private void updateCircularDisplayMaskIfNeeded() {
   6055         // we're fullscreen and not hosted in an ActivityView
   6056         if (mContext.getResources().getConfiguration().isScreenRound()
   6057                 && mContext.getResources().getBoolean(
   6058                 com.android.internal.R.bool.config_windowShowCircularMask)) {
   6059             final int currentUserId;
   6060             synchronized(mWindowMap) {
   6061                 currentUserId = mCurrentUserId;
   6062             }
   6063             // Device configuration calls for a circular display mask, but we only enable the mask
   6064             // if the accessibility color inversion feature is disabled, as the inverted mask
   6065             // causes artifacts.
   6066             int inversionState = Settings.Secure.getIntForUser(mContext.getContentResolver(),
   6067                     Settings.Secure.ACCESSIBILITY_DISPLAY_INVERSION_ENABLED, 0, currentUserId);
   6068             int showMask = (inversionState == 1) ? 0 : 1;
   6069             Message m = mH.obtainMessage(H.SHOW_CIRCULAR_DISPLAY_MASK);
   6070             m.arg1 = showMask;
   6071             mH.sendMessage(m);
   6072         }
   6073     }
   6074 
   6075     public void showEmulatorDisplayOverlayIfNeeded() {
   6076         if (mContext.getResources().getBoolean(
   6077                 com.android.internal.R.bool.config_windowEnableCircularEmulatorDisplayOverlay)
   6078                 && SystemProperties.getBoolean(PROPERTY_EMULATOR_CIRCULAR, false)
   6079                 && Build.IS_EMULATOR) {
   6080             mH.sendMessage(mH.obtainMessage(H.SHOW_EMULATOR_DISPLAY_OVERLAY));
   6081         }
   6082     }
   6083 
   6084     public void showCircularMask(boolean visible) {
   6085         synchronized(mWindowMap) {
   6086 
   6087             if (SHOW_LIGHT_TRANSACTIONS) Slog.i(TAG_WM,
   6088                     ">>> OPEN TRANSACTION showCircularMask(visible=" + visible + ")");
   6089             SurfaceControl.openTransaction();
   6090             try {
   6091                 if (visible) {
   6092                     // TODO(multi-display): support multiple displays
   6093                     if (mCircularDisplayMask == null) {
   6094                         int screenOffset = mContext.getResources().getInteger(
   6095                                 com.android.internal.R.integer.config_windowOutsetBottom);
   6096                         int maskThickness = mContext.getResources().getDimensionPixelSize(
   6097                                 com.android.internal.R.dimen.circular_display_mask_thickness);
   6098 
   6099                         mCircularDisplayMask = new CircularDisplayMask(
   6100                                 getDefaultDisplayContentLocked().getDisplay(),
   6101                                 mFxSession,
   6102                                 mPolicy.windowTypeToLayerLw(
   6103                                         WindowManager.LayoutParams.TYPE_POINTER)
   6104                                         * TYPE_LAYER_MULTIPLIER + 10, screenOffset, maskThickness);
   6105                     }
   6106                     mCircularDisplayMask.setVisibility(true);
   6107                 } else if (mCircularDisplayMask != null) {
   6108                     mCircularDisplayMask.setVisibility(false);
   6109                     mCircularDisplayMask = null;
   6110                 }
   6111             } finally {
   6112                 SurfaceControl.closeTransaction();
   6113                 if (SHOW_LIGHT_TRANSACTIONS) Slog.i(TAG_WM,
   6114                         "<<< CLOSE TRANSACTION showCircularMask(visible=" + visible + ")");
   6115             }
   6116         }
   6117     }
   6118 
   6119     public void showEmulatorDisplayOverlay() {
   6120         synchronized(mWindowMap) {
   6121 
   6122             if (SHOW_LIGHT_TRANSACTIONS) Slog.i(TAG_WM,
   6123                     ">>> OPEN TRANSACTION showEmulatorDisplayOverlay");
   6124             SurfaceControl.openTransaction();
   6125             try {
   6126                 if (mEmulatorDisplayOverlay == null) {
   6127                     mEmulatorDisplayOverlay = new EmulatorDisplayOverlay(
   6128                             mContext,
   6129                             getDefaultDisplayContentLocked().getDisplay(),
   6130                             mFxSession,
   6131                             mPolicy.windowTypeToLayerLw(
   6132                                     WindowManager.LayoutParams.TYPE_POINTER)
   6133                                     * TYPE_LAYER_MULTIPLIER + 10);
   6134                 }
   6135                 mEmulatorDisplayOverlay.setVisibility(true);
   6136             } finally {
   6137                 SurfaceControl.closeTransaction();
   6138                 if (SHOW_LIGHT_TRANSACTIONS) Slog.i(TAG_WM,
   6139                         "<<< CLOSE TRANSACTION showEmulatorDisplayOverlay");
   6140             }
   6141         }
   6142     }
   6143 
   6144     // TODO: more accounting of which pid(s) turned it on, keep count,
   6145     // only allow disables from pids which have count on, etc.
   6146     @Override
   6147     public void showStrictModeViolation(boolean on) {
   6148         int pid = Binder.getCallingPid();
   6149         mH.sendMessage(mH.obtainMessage(H.SHOW_STRICT_MODE_VIOLATION, on ? 1 : 0, pid));
   6150     }
   6151 
   6152     private void showStrictModeViolation(int arg, int pid) {
   6153         final boolean on = arg != 0;
   6154         synchronized(mWindowMap) {
   6155             // Ignoring requests to enable the red border from clients
   6156             // which aren't on screen.  (e.g. Broadcast Receivers in
   6157             // the background..)
   6158             if (on) {
   6159                 boolean isVisible = false;
   6160                 final int numDisplays = mDisplayContents.size();
   6161                 for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   6162                     final WindowList windows = mDisplayContents.valueAt(displayNdx).getWindowList();
   6163                     final int numWindows = windows.size();
   6164                     for (int winNdx = 0; winNdx < numWindows; ++winNdx) {
   6165                         final WindowState ws = windows.get(winNdx);
   6166                         if (ws.mSession.mPid == pid && ws.isVisibleLw()) {
   6167                             isVisible = true;
   6168                             break;
   6169                         }
   6170                     }
   6171                 }
   6172                 if (!isVisible) {
   6173                     return;
   6174                 }
   6175             }
   6176 
   6177             if (SHOW_VERBOSE_TRANSACTIONS) Slog.i(TAG_WM,
   6178                     ">>> OPEN TRANSACTION showStrictModeViolation");
   6179             SurfaceControl.openTransaction();
   6180             try {
   6181                 // TODO(multi-display): support multiple displays
   6182                 if (mStrictModeFlash == null) {
   6183                     mStrictModeFlash = new StrictModeFlash(
   6184                             getDefaultDisplayContentLocked().getDisplay(), mFxSession);
   6185                 }
   6186                 mStrictModeFlash.setVisibility(on);
   6187             } finally {
   6188                 SurfaceControl.closeTransaction();
   6189                 if (SHOW_VERBOSE_TRANSACTIONS) Slog.i(TAG_WM,
   6190                         "<<< CLOSE TRANSACTION showStrictModeViolation");
   6191             }
   6192         }
   6193     }
   6194 
   6195     @Override
   6196     public void setStrictModeVisualIndicatorPreference(String value) {
   6197         SystemProperties.set(StrictMode.VISUAL_PROPERTY, value);
   6198     }
   6199 
   6200     private static void convertCropForSurfaceFlinger(Rect crop, int rot, int dw, int dh) {
   6201         if (rot == Surface.ROTATION_90) {
   6202             final int tmp = crop.top;
   6203             crop.top = dw - crop.right;
   6204             crop.right = crop.bottom;
   6205             crop.bottom = dw - crop.left;
   6206             crop.left = tmp;
   6207         } else if (rot == Surface.ROTATION_180) {
   6208             int tmp = crop.top;
   6209             crop.top = dh - crop.bottom;
   6210             crop.bottom = dh - tmp;
   6211             tmp = crop.right;
   6212             crop.right = dw - crop.left;
   6213             crop.left = dw - tmp;
   6214         } else if (rot == Surface.ROTATION_270) {
   6215             final int tmp = crop.top;
   6216             crop.top = crop.left;
   6217             crop.left = dh - crop.bottom;
   6218             crop.bottom = crop.right;
   6219             crop.right = dh - tmp;
   6220         }
   6221     }
   6222 
   6223     @Override
   6224     public Bitmap screenshotWallpaper() {
   6225         if (!checkCallingPermission(Manifest.permission.READ_FRAME_BUFFER,
   6226                 "screenshotWallpaper()")) {
   6227             throw new SecurityException("Requires READ_FRAME_BUFFER permission");
   6228         }
   6229         try {
   6230             Trace.traceBegin(Trace.TRACE_TAG_WINDOW_MANAGER, "screenshotWallpaper");
   6231             return screenshotApplicationsInner(null, Display.DEFAULT_DISPLAY, -1, -1, true, 1f,
   6232                     Bitmap.Config.ARGB_8888, true);
   6233         } finally {
   6234             Trace.traceEnd(Trace.TRACE_TAG_WINDOW_MANAGER);
   6235         }
   6236     }
   6237 
   6238     /**
   6239      * Takes a snapshot of the screen.  In landscape mode this grabs the whole screen.
   6240      * In portrait mode, it grabs the upper region of the screen based on the vertical dimension
   6241      * of the target image.
   6242      */
   6243     @Override
   6244     public boolean requestAssistScreenshot(final IAssistScreenshotReceiver receiver) {
   6245         if (!checkCallingPermission(Manifest.permission.READ_FRAME_BUFFER,
   6246                 "requestAssistScreenshot()")) {
   6247             throw new SecurityException("Requires READ_FRAME_BUFFER permission");
   6248         }
   6249 
   6250         FgThread.getHandler().post(new Runnable() {
   6251             @Override
   6252             public void run() {
   6253                 Bitmap bm = screenshotApplicationsInner(null, Display.DEFAULT_DISPLAY, -1, -1,
   6254                         true, 1f, Bitmap.Config.ARGB_8888, false);
   6255                 try {
   6256                     receiver.send(bm);
   6257                 } catch (RemoteException e) {
   6258                 }
   6259             }
   6260         });
   6261 
   6262         return true;
   6263     }
   6264 
   6265     /**
   6266      * Takes a snapshot of the screen.  In landscape mode this grabs the whole screen.
   6267      * In portrait mode, it grabs the full screenshot.
   6268      *
   6269      * @param displayId the Display to take a screenshot of.
   6270      * @param width the width of the target bitmap
   6271      * @param height the height of the target bitmap
   6272      * @param frameScale the scale to apply to the frame, only used when width = -1 and height = -1
   6273      */
   6274     @Override
   6275     public Bitmap screenshotApplications(IBinder appToken, int displayId, int width, int height,
   6276             float frameScale) {
   6277         if (!checkCallingPermission(Manifest.permission.READ_FRAME_BUFFER,
   6278                 "screenshotApplications()")) {
   6279             throw new SecurityException("Requires READ_FRAME_BUFFER permission");
   6280         }
   6281         try {
   6282             Trace.traceBegin(Trace.TRACE_TAG_WINDOW_MANAGER, "screenshotApplications");
   6283             return screenshotApplicationsInner(appToken, displayId, width, height, false,
   6284                     frameScale, Bitmap.Config.RGB_565, false);
   6285         } finally {
   6286             Trace.traceEnd(Trace.TRACE_TAG_WINDOW_MANAGER);
   6287         }
   6288     }
   6289 
   6290     /**
   6291      * Takes a snapshot of the screen.  In landscape mode this grabs the whole screen.
   6292      * In portrait mode, it grabs the full screenshot.
   6293      *
   6294      * @param displayId the Display to take a screenshot of.
   6295      * @param width the width of the target bitmap
   6296      * @param height the height of the target bitmap
   6297      * @param includeFullDisplay true if the screen should not be cropped before capture
   6298      * @param frameScale the scale to apply to the frame, only used when width = -1 and height = -1
   6299      * @param config of the output bitmap
   6300      * @param wallpaperOnly true if only the wallpaper layer should be included in the screenshot
   6301      */
   6302     Bitmap screenshotApplicationsInner(IBinder appToken, int displayId, int width, int height,
   6303             boolean includeFullDisplay, float frameScale, Bitmap.Config config,
   6304             boolean wallpaperOnly) {
   6305         final DisplayContent displayContent;
   6306         synchronized(mWindowMap) {
   6307             displayContent = getDisplayContentLocked(displayId);
   6308             if (displayContent == null) {
   6309                 if (DEBUG_SCREENSHOT) Slog.i(TAG_WM, "Screenshot of " + appToken
   6310                         + ": returning null. No Display for displayId=" + displayId);
   6311                 return null;
   6312             }
   6313         }
   6314         final DisplayInfo displayInfo = displayContent.getDisplayInfo();
   6315         int dw = displayInfo.logicalWidth;
   6316         int dh = displayInfo.logicalHeight;
   6317         if (dw == 0 || dh == 0) {
   6318             if (DEBUG_SCREENSHOT) Slog.i(TAG_WM, "Screenshot of " + appToken
   6319                     + ": returning null. logical widthxheight=" + dw + "x" + dh);
   6320             return null;
   6321         }
   6322 
   6323         Bitmap bm = null;
   6324 
   6325         int maxLayer = 0;
   6326         final Rect frame = new Rect();
   6327         final Rect stackBounds = new Rect();
   6328 
   6329         boolean screenshotReady;
   6330         int minLayer;
   6331         if (appToken == null && !wallpaperOnly) {
   6332             screenshotReady = true;
   6333             minLayer = 0;
   6334         } else {
   6335             screenshotReady = false;
   6336             minLayer = Integer.MAX_VALUE;
   6337         }
   6338 
   6339         WindowState appWin = null;
   6340 
   6341         boolean includeImeInScreenshot;
   6342         synchronized(mWindowMap) {
   6343             final AppWindowToken imeTargetAppToken =
   6344                     mInputMethodTarget != null ? mInputMethodTarget.mAppToken : null;
   6345             // We only include the Ime in the screenshot if the app we are screenshoting is the IME
   6346             // target and isn't in multi-window mode. We don't screenshot the IME in multi-window
   6347             // mode because the frame of the IME might not overlap with that of the app.
   6348             // E.g. IME target app at the top in split-screen mode and the IME at the bottom
   6349             // overlapping with the bottom app.
   6350             includeImeInScreenshot = imeTargetAppToken != null
   6351                     && imeTargetAppToken.appToken != null
   6352                     && imeTargetAppToken.appToken.asBinder() == appToken
   6353                     && !mInputMethodTarget.isInMultiWindowMode();
   6354         }
   6355 
   6356         final int aboveAppLayer = (mPolicy.windowTypeToLayerLw(TYPE_APPLICATION) + 1)
   6357                 * TYPE_LAYER_MULTIPLIER + TYPE_LAYER_OFFSET;
   6358 
   6359         synchronized(mWindowMap) {
   6360             // Figure out the part of the screen that is actually the app.
   6361             appWin = null;
   6362             final WindowList windows = displayContent.getWindowList();
   6363             for (int i = windows.size() - 1; i >= 0; i--) {
   6364                 WindowState ws = windows.get(i);
   6365                 if (!ws.mHasSurface) {
   6366                     continue;
   6367                 }
   6368                 if (ws.mLayer >= aboveAppLayer) {
   6369                     continue;
   6370                 }
   6371                 if (wallpaperOnly && !ws.mIsWallpaper) {
   6372                     continue;
   6373                 }
   6374                 if (ws.mIsImWindow) {
   6375                     if (!includeImeInScreenshot) {
   6376                         continue;
   6377                     }
   6378                 } else if (ws.mIsWallpaper) {
   6379                     // If this is the wallpaper layer and we're only looking for the wallpaper layer
   6380                     // then the target window state is this one.
   6381                     if (wallpaperOnly) {
   6382                         appWin = ws;
   6383                     }
   6384 
   6385                     if (appWin == null) {
   6386                         // We have not ran across the target window yet, so it is probably
   6387                         // behind the wallpaper. This can happen when the keyguard is up and
   6388                         // all windows are moved behind the wallpaper. We don't want to
   6389                         // include the wallpaper layer in the screenshot as it will coverup
   6390                         // the layer of the target window.
   6391                         continue;
   6392                     }
   6393                     // Fall through. The target window is in front of the wallpaper. For this
   6394                     // case we want to include the wallpaper layer in the screenshot because
   6395                     // the target window might have some transparent areas.
   6396                 } else if (appToken != null) {
   6397                     if (ws.mAppToken == null || ws.mAppToken.token != appToken) {
   6398                         // This app window is of no interest if it is not associated with the
   6399                         // screenshot app.
   6400                         continue;
   6401                     }
   6402                     appWin = ws;
   6403                 }
   6404 
   6405                 // Include this window.
   6406 
   6407                 final WindowStateAnimator winAnim = ws.mWinAnimator;
   6408                 int layer = winAnim.mSurfaceController.getLayer();
   6409                 if (maxLayer < layer) {
   6410                     maxLayer = layer;
   6411                 }
   6412                 if (minLayer > layer) {
   6413                     minLayer = layer;
   6414                 }
   6415 
   6416                 // Don't include wallpaper in bounds calculation
   6417                 if (!includeFullDisplay && !ws.mIsWallpaper) {
   6418                     final Rect wf = ws.mFrame;
   6419                     final Rect cr = ws.mContentInsets;
   6420                     int left = wf.left + cr.left;
   6421                     int top = wf.top + cr.top;
   6422                     int right = wf.right - cr.right;
   6423                     int bottom = wf.bottom - cr.bottom;
   6424                     frame.union(left, top, right, bottom);
   6425                     ws.getVisibleBounds(stackBounds);
   6426                     if (!Rect.intersects(frame, stackBounds)) {
   6427                         // Set frame empty if there's no intersection.
   6428                         frame.setEmpty();
   6429                     }
   6430                 }
   6431 
   6432                 final boolean foundTargetWs =
   6433                         (ws.mAppToken != null && ws.mAppToken.token == appToken)
   6434                         || (appWin != null && wallpaperOnly);
   6435                 if (foundTargetWs && ws.isDisplayedLw() && winAnim.getShown()) {
   6436                     screenshotReady = true;
   6437                 }
   6438 
   6439                 if (ws.isObscuringFullscreen(displayInfo)){
   6440                     break;
   6441                 }
   6442             }
   6443 
   6444             if (appToken != null && appWin == null) {
   6445                 // Can't find a window to snapshot.
   6446                 if (DEBUG_SCREENSHOT) Slog.i(TAG_WM,
   6447                         "Screenshot: Couldn't find a surface matching " + appToken);
   6448                 return null;
   6449             }
   6450 
   6451             if (!screenshotReady) {
   6452                 Slog.i(TAG_WM, "Failed to capture screenshot of " + appToken +
   6453                         " appWin=" + (appWin == null ? "null" : (appWin + " drawState=" +
   6454                         appWin.mWinAnimator.mDrawState)));
   6455                 return null;
   6456             }
   6457 
   6458             // Screenshot is ready to be taken. Everything from here below will continue
   6459             // through the bottom of the loop and return a value. We only stay in the loop
   6460             // because we don't want to release the mWindowMap lock until the screenshot is
   6461             // taken.
   6462 
   6463             if (maxLayer == 0) {
   6464                 if (DEBUG_SCREENSHOT) Slog.i(TAG_WM, "Screenshot of " + appToken
   6465                         + ": returning null maxLayer=" + maxLayer);
   6466                 return null;
   6467             }
   6468 
   6469             if (!includeFullDisplay) {
   6470                 // Constrain frame to the screen size.
   6471                 if (!frame.intersect(0, 0, dw, dh)) {
   6472                     frame.setEmpty();
   6473                 }
   6474             } else {
   6475                 // Caller just wants entire display.
   6476                 frame.set(0, 0, dw, dh);
   6477             }
   6478             if (frame.isEmpty()) {
   6479                 return null;
   6480             }
   6481 
   6482             if (width < 0) {
   6483                 width = (int) (frame.width() * frameScale);
   6484             }
   6485             if (height < 0) {
   6486                 height = (int) (frame.height() * frameScale);
   6487             }
   6488 
   6489             // Tell surface flinger what part of the image to crop. Take the top
   6490             // right part of the application, and crop the larger dimension to fit.
   6491             Rect crop = new Rect(frame);
   6492             if (width / (float) frame.width() < height / (float) frame.height()) {
   6493                 int cropWidth = (int)((float)width / (float)height * frame.height());
   6494                 crop.right = crop.left + cropWidth;
   6495             } else {
   6496                 int cropHeight = (int)((float)height / (float)width * frame.width());
   6497                 crop.bottom = crop.top + cropHeight;
   6498             }
   6499 
   6500             // The screenshot API does not apply the current screen rotation.
   6501             int rot = getDefaultDisplayContentLocked().getDisplay().getRotation();
   6502 
   6503             if (rot == Surface.ROTATION_90 || rot == Surface.ROTATION_270) {
   6504                 rot = (rot == Surface.ROTATION_90) ? Surface.ROTATION_270 : Surface.ROTATION_90;
   6505             }
   6506 
   6507             // Surfaceflinger is not aware of orientation, so convert our logical
   6508             // crop to surfaceflinger's portrait orientation.
   6509             convertCropForSurfaceFlinger(crop, rot, dw, dh);
   6510 
   6511             if (DEBUG_SCREENSHOT) {
   6512                 Slog.i(TAG_WM, "Screenshot: " + dw + "x" + dh + " from " + minLayer + " to "
   6513                         + maxLayer + " appToken=" + appToken);
   6514                 for (int i = 0; i < windows.size(); i++) {
   6515                     WindowState win = windows.get(i);
   6516                     WindowSurfaceController controller = win.mWinAnimator.mSurfaceController;
   6517                     Slog.i(TAG_WM, win + ": " + win.mLayer
   6518                             + " animLayer=" + win.mWinAnimator.mAnimLayer
   6519                             + " surfaceLayer=" + ((controller == null)
   6520                                 ? "null" : controller.getLayer()));
   6521                 }
   6522             }
   6523 
   6524             ScreenRotationAnimation screenRotationAnimation =
   6525                     mAnimator.getScreenRotationAnimationLocked(Display.DEFAULT_DISPLAY);
   6526             final boolean inRotation = screenRotationAnimation != null &&
   6527                     screenRotationAnimation.isAnimating();
   6528             if (DEBUG_SCREENSHOT && inRotation) Slog.v(TAG_WM,
   6529                     "Taking screenshot while rotating");
   6530 
   6531             // We force pending transactions to flush before taking
   6532             // the screenshot by pushing an empty synchronous transaction.
   6533             SurfaceControl.openTransaction();
   6534             SurfaceControl.closeTransactionSync();
   6535 
   6536             bm = SurfaceControl.screenshot(crop, width, height, minLayer, maxLayer,
   6537                     inRotation, rot);
   6538             if (bm == null) {
   6539                 Slog.w(TAG_WM, "Screenshot failure taking screenshot for (" + dw + "x" + dh
   6540                         + ") to layer " + maxLayer);
   6541                 return null;
   6542             }
   6543         }
   6544 
   6545         if (DEBUG_SCREENSHOT) {
   6546             // TEST IF IT's ALL BLACK
   6547             int[] buffer = new int[bm.getWidth() * bm.getHeight()];
   6548             bm.getPixels(buffer, 0, bm.getWidth(), 0, 0, bm.getWidth(), bm.getHeight());
   6549             boolean allBlack = true;
   6550             final int firstColor = buffer[0];
   6551             for (int i = 0; i < buffer.length; i++) {
   6552                 if (buffer[i] != firstColor) {
   6553                     allBlack = false;
   6554                     break;
   6555                 }
   6556             }
   6557             if (allBlack) {
   6558                 Slog.i(TAG_WM, "Screenshot " + appWin + " was monochrome(" +
   6559                         Integer.toHexString(firstColor) + ")! mSurfaceLayer=" +
   6560                         (appWin != null ?
   6561                                 appWin.mWinAnimator.mSurfaceController.getLayer() : "null") +
   6562                         " minLayer=" + minLayer + " maxLayer=" + maxLayer);
   6563             }
   6564         }
   6565 
   6566         // Create a copy of the screenshot that is immutable and backed in ashmem.
   6567         // This greatly reduces the overhead of passing the bitmap between processes.
   6568         Bitmap ret = bm.createAshmemBitmap(config);
   6569         bm.recycle();
   6570         return ret;
   6571     }
   6572 
   6573     /**
   6574      * Freeze rotation changes.  (Enable "rotation lock".)
   6575      * Persists across reboots.
   6576      * @param rotation The desired rotation to freeze to, or -1 to use the
   6577      * current rotation.
   6578      */
   6579     @Override
   6580     public void freezeRotation(int rotation) {
   6581         if (!checkCallingPermission(android.Manifest.permission.SET_ORIENTATION,
   6582                 "freezeRotation()")) {
   6583             throw new SecurityException("Requires SET_ORIENTATION permission");
   6584         }
   6585         if (rotation < -1 || rotation > Surface.ROTATION_270) {
   6586             throw new IllegalArgumentException("Rotation argument must be -1 or a valid "
   6587                     + "rotation constant.");
   6588         }
   6589 
   6590         if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "freezeRotation: mRotation=" + mRotation);
   6591 
   6592         long origId = Binder.clearCallingIdentity();
   6593         try {
   6594             mPolicy.setUserRotationMode(WindowManagerPolicy.USER_ROTATION_LOCKED,
   6595                     rotation == -1 ? mRotation : rotation);
   6596         } finally {
   6597             Binder.restoreCallingIdentity(origId);
   6598         }
   6599 
   6600         updateRotationUnchecked(false, false);
   6601     }
   6602 
   6603     /**
   6604      * Thaw rotation changes.  (Disable "rotation lock".)
   6605      * Persists across reboots.
   6606      */
   6607     @Override
   6608     public void thawRotation() {
   6609         if (!checkCallingPermission(android.Manifest.permission.SET_ORIENTATION,
   6610                 "thawRotation()")) {
   6611             throw new SecurityException("Requires SET_ORIENTATION permission");
   6612         }
   6613 
   6614         if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "thawRotation: mRotation=" + mRotation);
   6615 
   6616         long origId = Binder.clearCallingIdentity();
   6617         try {
   6618             mPolicy.setUserRotationMode(WindowManagerPolicy.USER_ROTATION_FREE,
   6619                     777); // rot not used
   6620         } finally {
   6621             Binder.restoreCallingIdentity(origId);
   6622         }
   6623 
   6624         updateRotationUnchecked(false, false);
   6625     }
   6626 
   6627     /**
   6628      * Recalculate the current rotation.
   6629      *
   6630      * Called by the window manager policy whenever the state of the system changes
   6631      * such that the current rotation might need to be updated, such as when the
   6632      * device is docked or rotated into a new posture.
   6633      */
   6634     @Override
   6635     public void updateRotation(boolean alwaysSendConfiguration, boolean forceRelayout) {
   6636         updateRotationUnchecked(alwaysSendConfiguration, forceRelayout);
   6637     }
   6638 
   6639     /**
   6640      * Temporarily pauses rotation changes until resumed.
   6641      *
   6642      * This can be used to prevent rotation changes from occurring while the user is
   6643      * performing certain operations, such as drag and drop.
   6644      *
   6645      * This call nests and must be matched by an equal number of calls to
   6646      * {@link #resumeRotationLocked}.
   6647      */
   6648     void pauseRotationLocked() {
   6649         mDeferredRotationPauseCount += 1;
   6650     }
   6651 
   6652     /**
   6653      * Resumes normal rotation changes after being paused.
   6654      */
   6655     void resumeRotationLocked() {
   6656         if (mDeferredRotationPauseCount > 0) {
   6657             mDeferredRotationPauseCount -= 1;
   6658             if (mDeferredRotationPauseCount == 0) {
   6659                 boolean changed = updateRotationUncheckedLocked(false);
   6660                 if (changed) {
   6661                     mH.sendEmptyMessage(H.SEND_NEW_CONFIGURATION);
   6662                 }
   6663             }
   6664         }
   6665     }
   6666 
   6667     public void updateRotationUnchecked(boolean alwaysSendConfiguration, boolean forceRelayout) {
   6668         if(DEBUG_ORIENTATION) Slog.v(TAG_WM, "updateRotationUnchecked("
   6669                    + "alwaysSendConfiguration=" + alwaysSendConfiguration + ")");
   6670 
   6671         long origId = Binder.clearCallingIdentity();
   6672         boolean changed;
   6673         synchronized(mWindowMap) {
   6674             changed = updateRotationUncheckedLocked(false);
   6675             if (!changed || forceRelayout) {
   6676                 getDefaultDisplayContentLocked().layoutNeeded = true;
   6677                 mWindowPlacerLocked.performSurfacePlacement();
   6678             }
   6679         }
   6680 
   6681         if (changed || alwaysSendConfiguration) {
   6682             sendNewConfiguration();
   6683         }
   6684 
   6685         Binder.restoreCallingIdentity(origId);
   6686     }
   6687 
   6688 
   6689     // TODO(multidisplay): Rotate any display?
   6690     /**
   6691      * Updates the current rotation.
   6692      *
   6693      * Returns true if the rotation has been changed.  In this case YOU
   6694      * MUST CALL sendNewConfiguration() TO UNFREEZE THE SCREEN.
   6695      */
   6696     public boolean updateRotationUncheckedLocked(boolean inTransaction) {
   6697         if (mDeferredRotationPauseCount > 0) {
   6698             // Rotation updates have been paused temporarily.  Defer the update until
   6699             // updates have been resumed.
   6700             if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Deferring rotation, rotation is paused.");
   6701             return false;
   6702         }
   6703 
   6704         ScreenRotationAnimation screenRotationAnimation =
   6705                 mAnimator.getScreenRotationAnimationLocked(Display.DEFAULT_DISPLAY);
   6706         if (screenRotationAnimation != null && screenRotationAnimation.isAnimating()) {
   6707             // Rotation updates cannot be performed while the previous rotation change
   6708             // animation is still in progress.  Skip this update.  We will try updating
   6709             // again after the animation is finished and the display is unfrozen.
   6710             if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Deferring rotation, animation in progress.");
   6711             return false;
   6712         }
   6713 
   6714         if (!mDisplayEnabled) {
   6715             // No point choosing a rotation if the display is not enabled.
   6716             if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Deferring rotation, display is not enabled.");
   6717             return false;
   6718         }
   6719 
   6720         // TODO: Implement forced rotation changes.
   6721         //       Set mAltOrientation to indicate that the application is receiving
   6722         //       an orientation that has different metrics than it expected.
   6723         //       eg. Portrait instead of Landscape.
   6724 
   6725         int rotation = mPolicy.rotationForOrientationLw(mLastOrientation, mRotation);
   6726         boolean altOrientation = !mPolicy.rotationHasCompatibleMetricsLw(
   6727                 mLastOrientation, rotation);
   6728 
   6729         if (DEBUG_ORIENTATION) {
   6730             Slog.v(TAG_WM, "Selected orientation "
   6731                     + mLastOrientation + ", got rotation " + rotation
   6732                     + " which has " + (altOrientation ? "incompatible" : "compatible")
   6733                     + " metrics");
   6734         }
   6735 
   6736         if (mRotation == rotation && mAltOrientation == altOrientation) {
   6737             // No change.
   6738             return false;
   6739         }
   6740 
   6741         if (DEBUG_ORIENTATION) {
   6742             Slog.v(TAG_WM,
   6743                 "Rotation changed to " + rotation + (altOrientation ? " (alt)" : "")
   6744                 + " from " + mRotation + (mAltOrientation ? " (alt)" : "")
   6745                 + ", lastOrientation=" + mLastOrientation);
   6746         }
   6747 
   6748         int oldRotation = mRotation;
   6749 
   6750         mRotation = rotation;
   6751         mAltOrientation = altOrientation;
   6752         mPolicy.setRotationLw(mRotation);
   6753 
   6754         mWindowsFreezingScreen = WINDOWS_FREEZING_SCREENS_ACTIVE;
   6755         mH.removeMessages(H.WINDOW_FREEZE_TIMEOUT);
   6756         mH.sendEmptyMessageDelayed(H.WINDOW_FREEZE_TIMEOUT, WINDOW_FREEZE_TIMEOUT_DURATION);
   6757         mWaitingForConfig = true;
   6758         final DisplayContent displayContent = getDefaultDisplayContentLocked();
   6759         displayContent.layoutNeeded = true;
   6760         final int[] anim = new int[2];
   6761         if (displayContent.isDimming()) {
   6762             anim[0] = anim[1] = 0;
   6763         } else {
   6764             mPolicy.selectRotationAnimationLw(anim);
   6765         }
   6766         boolean rotateSeamlessly = mPolicy.shouldRotateSeamlessly(oldRotation, mRotation);
   6767         final WindowList windows = displayContent.getWindowList();
   6768         // We can't rotate seamlessly while an existing seamless rotation is still
   6769         // waiting on windows to finish drawing.
   6770         if (rotateSeamlessly) {
   6771             for (int i = windows.size() - 1; i >= 0; i--) {
   6772                 WindowState w = windows.get(i);
   6773                 if (w.mSeamlesslyRotated) {
   6774                     rotateSeamlessly = false;
   6775                     break;
   6776                 }
   6777                 // In what can only be called an unfortunate workaround we require
   6778                 // seamlessly rotated child windows to have the TRANSFORM_TO_DISPLAY_INVERSE
   6779                 // flag. Due to limitations in the client API, there is no way for
   6780                 // the client to set this flag in a race free fashion. If we seamlessly rotate
   6781                 // a window which does not have this flag, but then gains it, we will get
   6782                 // an incorrect visual result (rotated viewfinder). This means if we want to
   6783                 // support seamlessly rotating windows which could gain this flag, we can't
   6784                 // rotate windows without it. This limits seamless rotation in N to camera framework
   6785                 // users, windows without children, and native code. This is unfortunate but
   6786                 // having the camera work is our primary goal.
   6787                 if (w.isChildWindow() & w.isVisibleNow() &&
   6788                         !w.mWinAnimator.mSurfaceController.getTransformToDisplayInverse()) {
   6789                     rotateSeamlessly = false;
   6790                 }
   6791             }
   6792         }
   6793 
   6794         if (!rotateSeamlessly) {
   6795             startFreezingDisplayLocked(inTransaction, anim[0], anim[1]);
   6796             // startFreezingDisplayLocked can reset the ScreenRotationAnimation.
   6797             screenRotationAnimation =
   6798                 mAnimator.getScreenRotationAnimationLocked(Display.DEFAULT_DISPLAY);
   6799         } else {
   6800             // The screen rotation animation uses a screenshot to freeze the screen
   6801             // while windows resize underneath.
   6802             // When we are rotating seamlessly, we allow the elements to transition
   6803             // to their rotated state independently and without a freeze required.
   6804             screenRotationAnimation = null;
   6805         }
   6806 
   6807         // We need to update our screen size information to match the new rotation. If the rotation
   6808         // has actually changed then this method will return true and, according to the comment at
   6809         // the top of the method, the caller is obligated to call computeNewConfigurationLocked().
   6810         // By updating the Display info here it will be available to
   6811         // computeScreenConfigurationLocked later.
   6812         updateDisplayAndOrientationLocked(mCurConfiguration.uiMode);
   6813 
   6814         final DisplayInfo displayInfo = displayContent.getDisplayInfo();
   6815         if (!inTransaction) {
   6816             if (SHOW_TRANSACTIONS) {
   6817                 Slog.i(TAG_WM, ">>> OPEN TRANSACTION setRotationUnchecked");
   6818             }
   6819             SurfaceControl.openTransaction();
   6820         }
   6821         try {
   6822             // NOTE: We disable the rotation in the emulator because
   6823             //       it doesn't support hardware OpenGL emulation yet.
   6824             if (CUSTOM_SCREEN_ROTATION && screenRotationAnimation != null
   6825                     && screenRotationAnimation.hasScreenshot()) {
   6826                 if (screenRotationAnimation.setRotationInTransaction(
   6827                         rotation, mFxSession,
   6828                         MAX_ANIMATION_DURATION, getTransitionAnimationScaleLocked(),
   6829                         displayInfo.logicalWidth, displayInfo.logicalHeight)) {
   6830                     scheduleAnimationLocked();
   6831                 }
   6832             }
   6833 
   6834             if (rotateSeamlessly) {
   6835                 for (int i = windows.size() - 1; i >= 0; i--) {
   6836                     WindowState w = windows.get(i);
   6837                     w.mWinAnimator.seamlesslyRotateWindow(oldRotation, mRotation);
   6838                 }
   6839             }
   6840 
   6841             mDisplayManagerInternal.performTraversalInTransactionFromWindowManager();
   6842         } finally {
   6843             if (!inTransaction) {
   6844                 SurfaceControl.closeTransaction();
   6845                 if (SHOW_LIGHT_TRANSACTIONS) {
   6846                     Slog.i(TAG_WM, "<<< CLOSE TRANSACTION setRotationUnchecked");
   6847                 }
   6848             }
   6849         }
   6850 
   6851         for (int i = windows.size() - 1; i >= 0; i--) {
   6852             WindowState w = windows.get(i);
   6853             // Discard surface after orientation change, these can't be reused.
   6854             if (w.mAppToken != null) {
   6855                 w.mAppToken.destroySavedSurfaces();
   6856             }
   6857             if (w.mHasSurface && !rotateSeamlessly) {
   6858                 if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Set mOrientationChanging of " + w);
   6859                 w.mOrientationChanging = true;
   6860                 mWindowPlacerLocked.mOrientationChangeComplete = false;
   6861                 w.mLastFreezeDuration = 0;
   6862             }
   6863         }
   6864         if (rotateSeamlessly) {
   6865             mH.removeMessages(H.SEAMLESS_ROTATION_TIMEOUT);
   6866             mH.sendEmptyMessageDelayed(H.SEAMLESS_ROTATION_TIMEOUT, SEAMLESS_ROTATION_TIMEOUT_DURATION);
   6867         }
   6868 
   6869         for (int i=mRotationWatchers.size()-1; i>=0; i--) {
   6870             try {
   6871                 mRotationWatchers.get(i).watcher.onRotationChanged(rotation);
   6872             } catch (RemoteException e) {
   6873             }
   6874         }
   6875 
   6876         // TODO (multidisplay): Magnification is supported only for the default display.
   6877         // Announce rotation only if we will not animate as we already have the
   6878         // windows in final state. Otherwise, we make this call at the rotation end.
   6879         if (screenRotationAnimation == null && mAccessibilityController != null
   6880                 && displayContent.getDisplayId() == Display.DEFAULT_DISPLAY) {
   6881             mAccessibilityController.onRotationChangedLocked(getDefaultDisplayContentLocked(),
   6882                     rotation);
   6883         }
   6884 
   6885         return true;
   6886     }
   6887 
   6888     @Override
   6889     public int getRotation() {
   6890         return mRotation;
   6891     }
   6892 
   6893     @Override
   6894     public boolean isRotationFrozen() {
   6895         return mPolicy.getUserRotationMode() == WindowManagerPolicy.USER_ROTATION_LOCKED;
   6896     }
   6897 
   6898     @Override
   6899     public int watchRotation(IRotationWatcher watcher) {
   6900         final IBinder watcherBinder = watcher.asBinder();
   6901         IBinder.DeathRecipient dr = new IBinder.DeathRecipient() {
   6902             @Override
   6903             public void binderDied() {
   6904                 synchronized (mWindowMap) {
   6905                     for (int i=0; i<mRotationWatchers.size(); i++) {
   6906                         if (watcherBinder == mRotationWatchers.get(i).watcher.asBinder()) {
   6907                             RotationWatcher removed = mRotationWatchers.remove(i);
   6908                             IBinder binder = removed.watcher.asBinder();
   6909                             if (binder != null) {
   6910                                 binder.unlinkToDeath(this, 0);
   6911                             }
   6912                             i--;
   6913                         }
   6914                     }
   6915                 }
   6916             }
   6917         };
   6918 
   6919         synchronized (mWindowMap) {
   6920             try {
   6921                 watcher.asBinder().linkToDeath(dr, 0);
   6922                 mRotationWatchers.add(new RotationWatcher(watcher, dr));
   6923             } catch (RemoteException e) {
   6924                 // Client died, no cleanup needed.
   6925             }
   6926 
   6927             return mRotation;
   6928         }
   6929     }
   6930 
   6931     @Override
   6932     public void removeRotationWatcher(IRotationWatcher watcher) {
   6933         final IBinder watcherBinder = watcher.asBinder();
   6934         synchronized (mWindowMap) {
   6935             for (int i=0; i<mRotationWatchers.size(); i++) {
   6936                 RotationWatcher rotationWatcher = mRotationWatchers.get(i);
   6937                 if (watcherBinder == rotationWatcher.watcher.asBinder()) {
   6938                     RotationWatcher removed = mRotationWatchers.remove(i);
   6939                     IBinder binder = removed.watcher.asBinder();
   6940                     if (binder != null) {
   6941                         binder.unlinkToDeath(removed.deathRecipient, 0);
   6942                     }
   6943                     i--;
   6944                 }
   6945             }
   6946         }
   6947     }
   6948 
   6949     /**
   6950      * Apps that use the compact menu panel (as controlled by the panelMenuIsCompact
   6951      * theme attribute) on devices that feature a physical options menu key attempt to position
   6952      * their menu panel window along the edge of the screen nearest the physical menu key.
   6953      * This lowers the travel distance between invoking the menu panel and selecting
   6954      * a menu option.
   6955      *
   6956      * This method helps control where that menu is placed. Its current implementation makes
   6957      * assumptions about the menu key and its relationship to the screen based on whether
   6958      * the device's natural orientation is portrait (width < height) or landscape.
   6959      *
   6960      * The menu key is assumed to be located along the bottom edge of natural-portrait
   6961      * devices and along the right edge of natural-landscape devices. If these assumptions
   6962      * do not hold for the target device, this method should be changed to reflect that.
   6963      *
   6964      * @return A {@link Gravity} value for placing the options menu window
   6965      */
   6966     @Override
   6967     public int getPreferredOptionsPanelGravity() {
   6968         synchronized (mWindowMap) {
   6969             final int rotation = getRotation();
   6970 
   6971             // TODO(multidisplay): Assume that such devices physical keys are on the main screen.
   6972             final DisplayContent displayContent = getDefaultDisplayContentLocked();
   6973             if (displayContent.mInitialDisplayWidth < displayContent.mInitialDisplayHeight) {
   6974                 // On devices with a natural orientation of portrait
   6975                 switch (rotation) {
   6976                     default:
   6977                     case Surface.ROTATION_0:
   6978                         return Gravity.CENTER_HORIZONTAL | Gravity.BOTTOM;
   6979                     case Surface.ROTATION_90:
   6980                         return Gravity.RIGHT | Gravity.BOTTOM;
   6981                     case Surface.ROTATION_180:
   6982                         return Gravity.CENTER_HORIZONTAL | Gravity.BOTTOM;
   6983                     case Surface.ROTATION_270:
   6984                         return Gravity.START | Gravity.BOTTOM;
   6985                 }
   6986             }
   6987 
   6988             // On devices with a natural orientation of landscape
   6989             switch (rotation) {
   6990                 default:
   6991                 case Surface.ROTATION_0:
   6992                     return Gravity.RIGHT | Gravity.BOTTOM;
   6993                 case Surface.ROTATION_90:
   6994                     return Gravity.CENTER_HORIZONTAL | Gravity.BOTTOM;
   6995                 case Surface.ROTATION_180:
   6996                     return Gravity.START | Gravity.BOTTOM;
   6997                 case Surface.ROTATION_270:
   6998                     return Gravity.CENTER_HORIZONTAL | Gravity.BOTTOM;
   6999             }
   7000         }
   7001     }
   7002 
   7003     /**
   7004      * Starts the view server on the specified port.
   7005      *
   7006      * @param port The port to listener to.
   7007      *
   7008      * @return True if the server was successfully started, false otherwise.
   7009      *
   7010      * @see com.android.server.wm.ViewServer
   7011      * @see com.android.server.wm.ViewServer#VIEW_SERVER_DEFAULT_PORT
   7012      */
   7013     @Override
   7014     public boolean startViewServer(int port) {
   7015         if (isSystemSecure()) {
   7016             return false;
   7017         }
   7018 
   7019         if (!checkCallingPermission(Manifest.permission.DUMP, "startViewServer")) {
   7020             return false;
   7021         }
   7022 
   7023         if (port < 1024) {
   7024             return false;
   7025         }
   7026 
   7027         if (mViewServer != null) {
   7028             if (!mViewServer.isRunning()) {
   7029                 try {
   7030                     return mViewServer.start();
   7031                 } catch (IOException e) {
   7032                     Slog.w(TAG_WM, "View server did not start");
   7033                 }
   7034             }
   7035             return false;
   7036         }
   7037 
   7038         try {
   7039             mViewServer = new ViewServer(this, port);
   7040             return mViewServer.start();
   7041         } catch (IOException e) {
   7042             Slog.w(TAG_WM, "View server did not start");
   7043         }
   7044         return false;
   7045     }
   7046 
   7047     private boolean isSystemSecure() {
   7048         return "1".equals(SystemProperties.get(SYSTEM_SECURE, "1")) &&
   7049                 "0".equals(SystemProperties.get(SYSTEM_DEBUGGABLE, "0"));
   7050     }
   7051 
   7052     /**
   7053      * Stops the view server if it exists.
   7054      *
   7055      * @return True if the server stopped, false if it wasn't started or
   7056      *         couldn't be stopped.
   7057      *
   7058      * @see com.android.server.wm.ViewServer
   7059      */
   7060     @Override
   7061     public boolean stopViewServer() {
   7062         if (isSystemSecure()) {
   7063             return false;
   7064         }
   7065 
   7066         if (!checkCallingPermission(Manifest.permission.DUMP, "stopViewServer")) {
   7067             return false;
   7068         }
   7069 
   7070         if (mViewServer != null) {
   7071             return mViewServer.stop();
   7072         }
   7073         return false;
   7074     }
   7075 
   7076     /**
   7077      * Indicates whether the view server is running.
   7078      *
   7079      * @return True if the server is running, false otherwise.
   7080      *
   7081      * @see com.android.server.wm.ViewServer
   7082      */
   7083     @Override
   7084     public boolean isViewServerRunning() {
   7085         if (isSystemSecure()) {
   7086             return false;
   7087         }
   7088 
   7089         if (!checkCallingPermission(Manifest.permission.DUMP, "isViewServerRunning")) {
   7090             return false;
   7091         }
   7092 
   7093         return mViewServer != null && mViewServer.isRunning();
   7094     }
   7095 
   7096     /**
   7097      * Lists all availble windows in the system. The listing is written in the
   7098      * specified Socket's output stream with the following syntax:
   7099      * windowHashCodeInHexadecimal windowName
   7100      * Each line of the ouput represents a different window.
   7101      *
   7102      * @param client The remote client to send the listing to.
   7103      * @return False if an error occured, true otherwise.
   7104      */
   7105     boolean viewServerListWindows(Socket client) {
   7106         if (isSystemSecure()) {
   7107             return false;
   7108         }
   7109 
   7110         boolean result = true;
   7111 
   7112         WindowList windows = new WindowList();
   7113         synchronized (mWindowMap) {
   7114             //noinspection unchecked
   7115             final int numDisplays = mDisplayContents.size();
   7116             for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   7117                 final DisplayContent displayContent = mDisplayContents.valueAt(displayNdx);
   7118                 windows.addAll(displayContent.getWindowList());
   7119             }
   7120         }
   7121 
   7122         BufferedWriter out = null;
   7123 
   7124         // Any uncaught exception will crash the system process
   7125         try {
   7126             OutputStream clientStream = client.getOutputStream();
   7127             out = new BufferedWriter(new OutputStreamWriter(clientStream), 8 * 1024);
   7128 
   7129             final int count = windows.size();
   7130             for (int i = 0; i < count; i++) {
   7131                 final WindowState w = windows.get(i);
   7132                 out.write(Integer.toHexString(System.identityHashCode(w)));
   7133                 out.write(' ');
   7134                 out.append(w.mAttrs.getTitle());
   7135                 out.write('\n');
   7136             }
   7137 
   7138             out.write("DONE.\n");
   7139             out.flush();
   7140         } catch (Exception e) {
   7141             result = false;
   7142         } finally {
   7143             if (out != null) {
   7144                 try {
   7145                     out.close();
   7146                 } catch (IOException e) {
   7147                     result = false;
   7148                 }
   7149             }
   7150         }
   7151 
   7152         return result;
   7153     }
   7154 
   7155     // TODO(multidisplay): Extend to multiple displays.
   7156     /**
   7157      * Returns the focused window in the following format:
   7158      * windowHashCodeInHexadecimal windowName
   7159      *
   7160      * @param client The remote client to send the listing to.
   7161      * @return False if an error occurred, true otherwise.
   7162      */
   7163     boolean viewServerGetFocusedWindow(Socket client) {
   7164         if (isSystemSecure()) {
   7165             return false;
   7166         }
   7167 
   7168         boolean result = true;
   7169 
   7170         WindowState focusedWindow = getFocusedWindow();
   7171 
   7172         BufferedWriter out = null;
   7173 
   7174         // Any uncaught exception will crash the system process
   7175         try {
   7176             OutputStream clientStream = client.getOutputStream();
   7177             out = new BufferedWriter(new OutputStreamWriter(clientStream), 8 * 1024);
   7178 
   7179             if(focusedWindow != null) {
   7180                 out.write(Integer.toHexString(System.identityHashCode(focusedWindow)));
   7181                 out.write(' ');
   7182                 out.append(focusedWindow.mAttrs.getTitle());
   7183             }
   7184             out.write('\n');
   7185             out.flush();
   7186         } catch (Exception e) {
   7187             result = false;
   7188         } finally {
   7189             if (out != null) {
   7190                 try {
   7191                     out.close();
   7192                 } catch (IOException e) {
   7193                     result = false;
   7194                 }
   7195             }
   7196         }
   7197 
   7198         return result;
   7199     }
   7200 
   7201     /**
   7202      * Sends a command to a target window. The result of the command, if any, will be
   7203      * written in the output stream of the specified socket.
   7204      *
   7205      * The parameters must follow this syntax:
   7206      * windowHashcode extra
   7207      *
   7208      * Where XX is the length in characeters of the windowTitle.
   7209      *
   7210      * The first parameter is the target window. The window with the specified hashcode
   7211      * will be the target. If no target can be found, nothing happens. The extra parameters
   7212      * will be delivered to the target window and as parameters to the command itself.
   7213      *
   7214      * @param client The remote client to sent the result, if any, to.
   7215      * @param command The command to execute.
   7216      * @param parameters The command parameters.
   7217      *
   7218      * @return True if the command was successfully delivered, false otherwise. This does
   7219      *         not indicate whether the command itself was successful.
   7220      */
   7221     boolean viewServerWindowCommand(Socket client, String command, String parameters) {
   7222         if (isSystemSecure()) {
   7223             return false;
   7224         }
   7225 
   7226         boolean success = true;
   7227         Parcel data = null;
   7228         Parcel reply = null;
   7229 
   7230         BufferedWriter out = null;
   7231 
   7232         // Any uncaught exception will crash the system process
   7233         try {
   7234             // Find the hashcode of the window
   7235             int index = parameters.indexOf(' ');
   7236             if (index == -1) {
   7237                 index = parameters.length();
   7238             }
   7239             final String code = parameters.substring(0, index);
   7240             int hashCode = (int) Long.parseLong(code, 16);
   7241 
   7242             // Extract the command's parameter after the window description
   7243             if (index < parameters.length()) {
   7244                 parameters = parameters.substring(index + 1);
   7245             } else {
   7246                 parameters = "";
   7247             }
   7248 
   7249             final WindowState window = findWindow(hashCode);
   7250             if (window == null) {
   7251                 return false;
   7252             }
   7253 
   7254             data = Parcel.obtain();
   7255             data.writeInterfaceToken("android.view.IWindow");
   7256             data.writeString(command);
   7257             data.writeString(parameters);
   7258             data.writeInt(1);
   7259             ParcelFileDescriptor.fromSocket(client).writeToParcel(data, 0);
   7260 
   7261             reply = Parcel.obtain();
   7262 
   7263             final IBinder binder = window.mClient.asBinder();
   7264             // TODO: GET THE TRANSACTION CODE IN A SAFER MANNER
   7265             binder.transact(IBinder.FIRST_CALL_TRANSACTION, data, reply, 0);
   7266 
   7267             reply.readException();
   7268 
   7269             if (!client.isOutputShutdown()) {
   7270                 out = new BufferedWriter(new OutputStreamWriter(client.getOutputStream()));
   7271                 out.write("DONE\n");
   7272                 out.flush();
   7273             }
   7274 
   7275         } catch (Exception e) {
   7276             Slog.w(TAG_WM, "Could not send command " + command + " with parameters " + parameters, e);
   7277             success = false;
   7278         } finally {
   7279             if (data != null) {
   7280                 data.recycle();
   7281             }
   7282             if (reply != null) {
   7283                 reply.recycle();
   7284             }
   7285             if (out != null) {
   7286                 try {
   7287                     out.close();
   7288                 } catch (IOException e) {
   7289 
   7290                 }
   7291             }
   7292         }
   7293 
   7294         return success;
   7295     }
   7296 
   7297     public void addWindowChangeListener(WindowChangeListener listener) {
   7298         synchronized(mWindowMap) {
   7299             mWindowChangeListeners.add(listener);
   7300         }
   7301     }
   7302 
   7303     public void removeWindowChangeListener(WindowChangeListener listener) {
   7304         synchronized(mWindowMap) {
   7305             mWindowChangeListeners.remove(listener);
   7306         }
   7307     }
   7308 
   7309     private void notifyWindowsChanged() {
   7310         WindowChangeListener[] windowChangeListeners;
   7311         synchronized(mWindowMap) {
   7312             if(mWindowChangeListeners.isEmpty()) {
   7313                 return;
   7314             }
   7315             windowChangeListeners = new WindowChangeListener[mWindowChangeListeners.size()];
   7316             windowChangeListeners = mWindowChangeListeners.toArray(windowChangeListeners);
   7317         }
   7318         int N = windowChangeListeners.length;
   7319         for(int i = 0; i < N; i++) {
   7320             windowChangeListeners[i].windowsChanged();
   7321         }
   7322     }
   7323 
   7324     private void notifyFocusChanged() {
   7325         WindowChangeListener[] windowChangeListeners;
   7326         synchronized(mWindowMap) {
   7327             if(mWindowChangeListeners.isEmpty()) {
   7328                 return;
   7329             }
   7330             windowChangeListeners = new WindowChangeListener[mWindowChangeListeners.size()];
   7331             windowChangeListeners = mWindowChangeListeners.toArray(windowChangeListeners);
   7332         }
   7333         int N = windowChangeListeners.length;
   7334         for(int i = 0; i < N; i++) {
   7335             windowChangeListeners[i].focusChanged();
   7336         }
   7337     }
   7338 
   7339     private WindowState findWindow(int hashCode) {
   7340         if (hashCode == -1) {
   7341             // TODO(multidisplay): Extend to multiple displays.
   7342             return getFocusedWindow();
   7343         }
   7344 
   7345         synchronized (mWindowMap) {
   7346             final int numDisplays = mDisplayContents.size();
   7347             for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   7348                 final WindowList windows = mDisplayContents.valueAt(displayNdx).getWindowList();
   7349                 final int numWindows = windows.size();
   7350                 for (int winNdx = 0; winNdx < numWindows; ++winNdx) {
   7351                     final WindowState w = windows.get(winNdx);
   7352                     if (System.identityHashCode(w) == hashCode) {
   7353                         return w;
   7354                     }
   7355                 }
   7356             }
   7357         }
   7358 
   7359         return null;
   7360     }
   7361 
   7362     /*
   7363      * Instruct the Activity Manager to fetch the current configuration and broadcast
   7364      * that to config-changed listeners if appropriate.
   7365      */
   7366     void sendNewConfiguration() {
   7367         try {
   7368             mActivityManager.updateConfiguration(null);
   7369         } catch (RemoteException e) {
   7370         }
   7371     }
   7372 
   7373     public Configuration computeNewConfiguration() {
   7374         synchronized (mWindowMap) {
   7375             return computeNewConfigurationLocked();
   7376         }
   7377     }
   7378 
   7379     private Configuration computeNewConfigurationLocked() {
   7380         if (!mDisplayReady) {
   7381             return null;
   7382         }
   7383         Configuration config = new Configuration();
   7384         config.fontScale = 0;
   7385         computeScreenConfigurationLocked(config);
   7386         return config;
   7387     }
   7388 
   7389     private void adjustDisplaySizeRanges(DisplayInfo displayInfo, int rotation, int uiMode,
   7390             int dw, int dh) {
   7391         // TODO: Multidisplay: for now only use with default display.
   7392         final int width = mPolicy.getConfigDisplayWidth(dw, dh, rotation, uiMode);
   7393         if (width < displayInfo.smallestNominalAppWidth) {
   7394             displayInfo.smallestNominalAppWidth = width;
   7395         }
   7396         if (width > displayInfo.largestNominalAppWidth) {
   7397             displayInfo.largestNominalAppWidth = width;
   7398         }
   7399         final int height = mPolicy.getConfigDisplayHeight(dw, dh, rotation, uiMode);
   7400         if (height < displayInfo.smallestNominalAppHeight) {
   7401             displayInfo.smallestNominalAppHeight = height;
   7402         }
   7403         if (height > displayInfo.largestNominalAppHeight) {
   7404             displayInfo.largestNominalAppHeight = height;
   7405         }
   7406     }
   7407 
   7408     private int reduceConfigLayout(int curLayout, int rotation, float density,
   7409             int dw, int dh, int uiMode) {
   7410         // TODO: Multidisplay: for now only use with default display.
   7411         // Get the app screen size at this rotation.
   7412         int w = mPolicy.getNonDecorDisplayWidth(dw, dh, rotation, uiMode);
   7413         int h = mPolicy.getNonDecorDisplayHeight(dw, dh, rotation, uiMode);
   7414 
   7415         // Compute the screen layout size class for this rotation.
   7416         int longSize = w;
   7417         int shortSize = h;
   7418         if (longSize < shortSize) {
   7419             int tmp = longSize;
   7420             longSize = shortSize;
   7421             shortSize = tmp;
   7422         }
   7423         longSize = (int)(longSize/density);
   7424         shortSize = (int)(shortSize/density);
   7425         return Configuration.reduceScreenLayout(curLayout, longSize, shortSize);
   7426     }
   7427 
   7428     private void computeSizeRangesAndScreenLayout(DisplayInfo displayInfo, boolean rotated,
   7429                   int uiMode, int dw, int dh, float density, Configuration outConfig) {
   7430         // TODO: Multidisplay: for now only use with default display.
   7431 
   7432         // We need to determine the smallest width that will occur under normal
   7433         // operation.  To this, start with the base screen size and compute the
   7434         // width under the different possible rotations.  We need to un-rotate
   7435         // the current screen dimensions before doing this.
   7436         int unrotDw, unrotDh;
   7437         if (rotated) {
   7438             unrotDw = dh;
   7439             unrotDh = dw;
   7440         } else {
   7441             unrotDw = dw;
   7442             unrotDh = dh;
   7443         }
   7444         displayInfo.smallestNominalAppWidth = 1<<30;
   7445         displayInfo.smallestNominalAppHeight = 1<<30;
   7446         displayInfo.largestNominalAppWidth = 0;
   7447         displayInfo.largestNominalAppHeight = 0;
   7448         adjustDisplaySizeRanges(displayInfo, Surface.ROTATION_0, uiMode, unrotDw, unrotDh);
   7449         adjustDisplaySizeRanges(displayInfo, Surface.ROTATION_90, uiMode, unrotDh, unrotDw);
   7450         adjustDisplaySizeRanges(displayInfo, Surface.ROTATION_180, uiMode, unrotDw, unrotDh);
   7451         adjustDisplaySizeRanges(displayInfo, Surface.ROTATION_270, uiMode, unrotDh, unrotDw);
   7452         int sl = Configuration.resetScreenLayout(outConfig.screenLayout);
   7453         sl = reduceConfigLayout(sl, Surface.ROTATION_0, density, unrotDw, unrotDh, uiMode);
   7454         sl = reduceConfigLayout(sl, Surface.ROTATION_90, density, unrotDh, unrotDw, uiMode);
   7455         sl = reduceConfigLayout(sl, Surface.ROTATION_180, density, unrotDw, unrotDh, uiMode);
   7456         sl = reduceConfigLayout(sl, Surface.ROTATION_270, density, unrotDh, unrotDw, uiMode);
   7457         outConfig.smallestScreenWidthDp = (int)(displayInfo.smallestNominalAppWidth / density);
   7458         outConfig.screenLayout = sl;
   7459     }
   7460 
   7461     private int reduceCompatConfigWidthSize(int curSize, int rotation, int uiMode,
   7462             DisplayMetrics dm, int dw, int dh) {
   7463         // TODO: Multidisplay: for now only use with default display.
   7464         dm.noncompatWidthPixels = mPolicy.getNonDecorDisplayWidth(dw, dh, rotation, uiMode);
   7465         dm.noncompatHeightPixels = mPolicy.getNonDecorDisplayHeight(dw, dh, rotation, uiMode);
   7466         float scale = CompatibilityInfo.computeCompatibleScaling(dm, null);
   7467         int size = (int)(((dm.noncompatWidthPixels / scale) / dm.density) + .5f);
   7468         if (curSize == 0 || size < curSize) {
   7469             curSize = size;
   7470         }
   7471         return curSize;
   7472     }
   7473 
   7474     private int computeCompatSmallestWidth(boolean rotated, int uiMode, DisplayMetrics dm, int dw, int dh) {
   7475         // TODO: Multidisplay: for now only use with default display.
   7476         mTmpDisplayMetrics.setTo(dm);
   7477         final DisplayMetrics tmpDm = mTmpDisplayMetrics;
   7478         final int unrotDw, unrotDh;
   7479         if (rotated) {
   7480             unrotDw = dh;
   7481             unrotDh = dw;
   7482         } else {
   7483             unrotDw = dw;
   7484             unrotDh = dh;
   7485         }
   7486         int sw = reduceCompatConfigWidthSize(0, Surface.ROTATION_0, uiMode, tmpDm, unrotDw, unrotDh);
   7487         sw = reduceCompatConfigWidthSize(sw, Surface.ROTATION_90, uiMode, tmpDm, unrotDh, unrotDw);
   7488         sw = reduceCompatConfigWidthSize(sw, Surface.ROTATION_180, uiMode, tmpDm, unrotDw, unrotDh);
   7489         sw = reduceCompatConfigWidthSize(sw, Surface.ROTATION_270, uiMode, tmpDm, unrotDh, unrotDw);
   7490         return sw;
   7491     }
   7492 
   7493     /** Do not call if mDisplayReady == false */
   7494     DisplayInfo updateDisplayAndOrientationLocked(int uiMode) {
   7495         // TODO(multidisplay): For now, apply Configuration to main screen only.
   7496         final DisplayContent displayContent = getDefaultDisplayContentLocked();
   7497 
   7498         // Use the effective "visual" dimensions based on current rotation
   7499         final boolean rotated = (mRotation == Surface.ROTATION_90
   7500                 || mRotation == Surface.ROTATION_270);
   7501         final int realdw = rotated ?
   7502                 displayContent.mBaseDisplayHeight : displayContent.mBaseDisplayWidth;
   7503         final int realdh = rotated ?
   7504                 displayContent.mBaseDisplayWidth : displayContent.mBaseDisplayHeight;
   7505         int dw = realdw;
   7506         int dh = realdh;
   7507 
   7508         if (mAltOrientation) {
   7509             if (realdw > realdh) {
   7510                 // Turn landscape into portrait.
   7511                 int maxw = (int)(realdh/1.3f);
   7512                 if (maxw < realdw) {
   7513                     dw = maxw;
   7514                 }
   7515             } else {
   7516                 // Turn portrait into landscape.
   7517                 int maxh = (int)(realdw/1.3f);
   7518                 if (maxh < realdh) {
   7519                     dh = maxh;
   7520                 }
   7521             }
   7522         }
   7523 
   7524         // Update application display metrics.
   7525         final int appWidth = mPolicy.getNonDecorDisplayWidth(dw, dh, mRotation, uiMode);
   7526         final int appHeight = mPolicy.getNonDecorDisplayHeight(dw, dh, mRotation, uiMode);
   7527         final DisplayInfo displayInfo = displayContent.getDisplayInfo();
   7528         displayInfo.rotation = mRotation;
   7529         displayInfo.logicalWidth = dw;
   7530         displayInfo.logicalHeight = dh;
   7531         displayInfo.logicalDensityDpi = displayContent.mBaseDisplayDensity;
   7532         displayInfo.appWidth = appWidth;
   7533         displayInfo.appHeight = appHeight;
   7534         displayInfo.getLogicalMetrics(mRealDisplayMetrics,
   7535                 CompatibilityInfo.DEFAULT_COMPATIBILITY_INFO, null);
   7536         displayInfo.getAppMetrics(mDisplayMetrics);
   7537         if (displayContent.mDisplayScalingDisabled) {
   7538             displayInfo.flags |= Display.FLAG_SCALING_DISABLED;
   7539         } else {
   7540             displayInfo.flags &= ~Display.FLAG_SCALING_DISABLED;
   7541         }
   7542 
   7543         mDisplayManagerInternal.setDisplayInfoOverrideFromWindowManager(
   7544                 displayContent.getDisplayId(), displayInfo);
   7545 
   7546         displayContent.mBaseDisplayRect.set(0, 0, dw, dh);
   7547         if (false) {
   7548             Slog.i(TAG_WM, "Set app display size: " + appWidth + " x " + appHeight);
   7549         }
   7550 
   7551         mCompatibleScreenScale = CompatibilityInfo.computeCompatibleScaling(mDisplayMetrics,
   7552                 mCompatDisplayMetrics);
   7553         return displayInfo;
   7554     }
   7555 
   7556     /** Do not call if mDisplayReady == false */
   7557     void computeScreenConfigurationLocked(Configuration config) {
   7558         final DisplayInfo displayInfo = updateDisplayAndOrientationLocked(
   7559                 config.uiMode);
   7560 
   7561         final int dw = displayInfo.logicalWidth;
   7562         final int dh = displayInfo.logicalHeight;
   7563         config.orientation = (dw <= dh) ? Configuration.ORIENTATION_PORTRAIT :
   7564                 Configuration.ORIENTATION_LANDSCAPE;
   7565         config.screenWidthDp =
   7566                 (int)(mPolicy.getConfigDisplayWidth(dw, dh, mRotation, config.uiMode) /
   7567                         mDisplayMetrics.density);
   7568         config.screenHeightDp =
   7569                 (int)(mPolicy.getConfigDisplayHeight(dw, dh, mRotation, config.uiMode) /
   7570                         mDisplayMetrics.density);
   7571         final boolean rotated = (mRotation == Surface.ROTATION_90
   7572                 || mRotation == Surface.ROTATION_270);
   7573 
   7574         computeSizeRangesAndScreenLayout(displayInfo, rotated, config.uiMode, dw, dh,
   7575                 mDisplayMetrics.density, config);
   7576 
   7577         config.screenLayout = (config.screenLayout & ~Configuration.SCREENLAYOUT_ROUND_MASK)
   7578                 | ((displayInfo.flags & Display.FLAG_ROUND) != 0
   7579                         ? Configuration.SCREENLAYOUT_ROUND_YES
   7580                         : Configuration.SCREENLAYOUT_ROUND_NO);
   7581 
   7582         config.compatScreenWidthDp = (int)(config.screenWidthDp / mCompatibleScreenScale);
   7583         config.compatScreenHeightDp = (int)(config.screenHeightDp / mCompatibleScreenScale);
   7584         config.compatSmallestScreenWidthDp = computeCompatSmallestWidth(rotated, config.uiMode,
   7585                 mDisplayMetrics, dw, dh);
   7586         config.densityDpi = displayInfo.logicalDensityDpi;
   7587 
   7588         // Update the configuration based on available input devices, lid switch,
   7589         // and platform configuration.
   7590         config.touchscreen = Configuration.TOUCHSCREEN_NOTOUCH;
   7591         config.keyboard = Configuration.KEYBOARD_NOKEYS;
   7592         config.navigation = Configuration.NAVIGATION_NONAV;
   7593 
   7594         int keyboardPresence = 0;
   7595         int navigationPresence = 0;
   7596         final InputDevice[] devices = mInputManager.getInputDevices();
   7597         final int len = devices.length;
   7598         for (int i = 0; i < len; i++) {
   7599             InputDevice device = devices[i];
   7600             if (!device.isVirtual()) {
   7601                 final int sources = device.getSources();
   7602                 final int presenceFlag = device.isExternal() ?
   7603                         WindowManagerPolicy.PRESENCE_EXTERNAL :
   7604                                 WindowManagerPolicy.PRESENCE_INTERNAL;
   7605 
   7606                 if (mIsTouchDevice) {
   7607                     if ((sources & InputDevice.SOURCE_TOUCHSCREEN) ==
   7608                             InputDevice.SOURCE_TOUCHSCREEN) {
   7609                         config.touchscreen = Configuration.TOUCHSCREEN_FINGER;
   7610                     }
   7611                 } else {
   7612                     config.touchscreen = Configuration.TOUCHSCREEN_NOTOUCH;
   7613                 }
   7614 
   7615                 if ((sources & InputDevice.SOURCE_TRACKBALL) == InputDevice.SOURCE_TRACKBALL) {
   7616                     config.navigation = Configuration.NAVIGATION_TRACKBALL;
   7617                     navigationPresence |= presenceFlag;
   7618                 } else if ((sources & InputDevice.SOURCE_DPAD) == InputDevice.SOURCE_DPAD
   7619                         && config.navigation == Configuration.NAVIGATION_NONAV) {
   7620                     config.navigation = Configuration.NAVIGATION_DPAD;
   7621                     navigationPresence |= presenceFlag;
   7622                 }
   7623 
   7624                 if (device.getKeyboardType() == InputDevice.KEYBOARD_TYPE_ALPHABETIC) {
   7625                     config.keyboard = Configuration.KEYBOARD_QWERTY;
   7626                     keyboardPresence |= presenceFlag;
   7627                 }
   7628             }
   7629         }
   7630 
   7631         if (config.navigation == Configuration.NAVIGATION_NONAV && mHasPermanentDpad) {
   7632             config.navigation = Configuration.NAVIGATION_DPAD;
   7633             navigationPresence |= WindowManagerPolicy.PRESENCE_INTERNAL;
   7634         }
   7635 
   7636         // Determine whether a hard keyboard is available and enabled.
   7637         boolean hardKeyboardAvailable = config.keyboard != Configuration.KEYBOARD_NOKEYS;
   7638         if (hardKeyboardAvailable != mHardKeyboardAvailable) {
   7639             mHardKeyboardAvailable = hardKeyboardAvailable;
   7640             mH.removeMessages(H.REPORT_HARD_KEYBOARD_STATUS_CHANGE);
   7641             mH.sendEmptyMessage(H.REPORT_HARD_KEYBOARD_STATUS_CHANGE);
   7642         }
   7643 
   7644         // Let the policy update hidden states.
   7645         config.keyboardHidden = Configuration.KEYBOARDHIDDEN_NO;
   7646         config.hardKeyboardHidden = Configuration.HARDKEYBOARDHIDDEN_NO;
   7647         config.navigationHidden = Configuration.NAVIGATIONHIDDEN_NO;
   7648         mPolicy.adjustConfigurationLw(config, keyboardPresence, navigationPresence);
   7649     }
   7650 
   7651     void notifyHardKeyboardStatusChange() {
   7652         final boolean available;
   7653         final WindowManagerInternal.OnHardKeyboardStatusChangeListener listener;
   7654         synchronized (mWindowMap) {
   7655             listener = mHardKeyboardStatusChangeListener;
   7656             available = mHardKeyboardAvailable;
   7657         }
   7658         if (listener != null) {
   7659             listener.onHardKeyboardStatusChange(available);
   7660         }
   7661     }
   7662 
   7663     boolean startMovingTask(IWindow window, float startX, float startY) {
   7664         WindowState win = null;
   7665         synchronized (mWindowMap) {
   7666             win = windowForClientLocked(null, window, false);
   7667             // win shouldn't be null here, pass it down to startPositioningLocked
   7668             // to get warning if it's null.
   7669             if (!startPositioningLocked(win, false /*resize*/, startX, startY)) {
   7670                 return false;
   7671             }
   7672         }
   7673         try {
   7674             mActivityManager.setFocusedTask(win.getTask().mTaskId);
   7675         } catch(RemoteException e) {}
   7676         return true;
   7677     }
   7678 
   7679     private void startScrollingTask(DisplayContent displayContent, int startX, int startY) {
   7680         if (DEBUG_TASK_POSITIONING) Slog.d(TAG_WM,
   7681                 "startScrollingTask: " + "{" + startX + ", " + startY + "}");
   7682 
   7683         Task task = null;
   7684         synchronized (mWindowMap) {
   7685             int taskId = displayContent.taskIdFromPoint(startX, startY);
   7686             if (taskId >= 0) {
   7687                 task = mTaskIdToTask.get(taskId);
   7688             }
   7689             if (task == null || !task.isDockedInEffect() || !startPositioningLocked(
   7690                     task.getTopVisibleAppMainWindow(), false /*resize*/, startX, startY)) {
   7691                 return;
   7692             }
   7693         }
   7694         try {
   7695             mActivityManager.setFocusedTask(task.mTaskId);
   7696         } catch(RemoteException e) {}
   7697     }
   7698 
   7699     private void handleTapOutsideTask(DisplayContent displayContent, int x, int y) {
   7700         int taskId = -1;
   7701         synchronized (mWindowMap) {
   7702             final Task task = displayContent.findTaskForControlPoint(x, y);
   7703             if (task != null) {
   7704                 if (!startPositioningLocked(
   7705                         task.getTopVisibleAppMainWindow(), true /*resize*/, x, y)) {
   7706                     return;
   7707                 }
   7708                 taskId = task.mTaskId;
   7709             } else {
   7710                 taskId = displayContent.taskIdFromPoint(x, y);
   7711             }
   7712         }
   7713         if (taskId >= 0) {
   7714             try {
   7715                 mActivityManager.setFocusedTask(taskId);
   7716             } catch(RemoteException e) {}
   7717         }
   7718     }
   7719 
   7720     private boolean startPositioningLocked(
   7721             WindowState win, boolean resize, float startX, float startY) {
   7722         if (DEBUG_TASK_POSITIONING) Slog.d(TAG_WM, "startPositioningLocked: "
   7723             + "win=" + win + ", resize=" + resize + ", {" + startX + ", " + startY + "}");
   7724 
   7725         if (win == null || win.getAppToken() == null) {
   7726             Slog.w(TAG_WM, "startPositioningLocked: Bad window " + win);
   7727             return false;
   7728         }
   7729         if (win.mInputChannel == null) {
   7730             Slog.wtf(TAG_WM, "startPositioningLocked: " + win + " has no input channel, "
   7731                     + " probably being removed");
   7732             return false;
   7733         }
   7734 
   7735         final DisplayContent displayContent = win.getDisplayContent();
   7736         if (displayContent == null) {
   7737             Slog.w(TAG_WM, "startPositioningLocked: Invalid display content " + win);
   7738             return false;
   7739         }
   7740 
   7741         Display display = displayContent.getDisplay();
   7742         mTaskPositioner = new TaskPositioner(this);
   7743         mTaskPositioner.register(display);
   7744         mInputMonitor.updateInputWindowsLw(true /*force*/);
   7745 
   7746         // We need to grab the touch focus so that the touch events during the
   7747         // resizing/scrolling are not sent to the app. 'win' is the main window
   7748         // of the app, it may not have focus since there might be other windows
   7749         // on top (eg. a dialog window).
   7750         WindowState transferFocusFromWin = win;
   7751         if (mCurrentFocus != null && mCurrentFocus != win
   7752                 && mCurrentFocus.mAppToken == win.mAppToken) {
   7753             transferFocusFromWin = mCurrentFocus;
   7754         }
   7755         if (!mInputManager.transferTouchFocus(
   7756                 transferFocusFromWin.mInputChannel, mTaskPositioner.mServerChannel)) {
   7757             Slog.e(TAG_WM, "startPositioningLocked: Unable to transfer touch focus");
   7758             mTaskPositioner.unregister();
   7759             mTaskPositioner = null;
   7760             mInputMonitor.updateInputWindowsLw(true /*force*/);
   7761             return false;
   7762         }
   7763 
   7764         mTaskPositioner.startDragLocked(win, resize, startX, startY);
   7765         return true;
   7766     }
   7767 
   7768     private void finishPositioning() {
   7769         if (DEBUG_TASK_POSITIONING) {
   7770             Slog.d(TAG_WM, "finishPositioning");
   7771         }
   7772         synchronized (mWindowMap) {
   7773             if (mTaskPositioner != null) {
   7774                 mTaskPositioner.unregister();
   7775                 mTaskPositioner = null;
   7776                 mInputMonitor.updateInputWindowsLw(true /*force*/);
   7777             }
   7778         }
   7779     }
   7780 
   7781     void adjustForImeIfNeeded(final DisplayContent displayContent) {
   7782         final WindowState imeWin = mInputMethodWindow;
   7783         final boolean imeVisible = imeWin != null && imeWin.isVisibleLw() && imeWin.isDisplayedLw()
   7784                 && !displayContent.mDividerControllerLocked.isImeHideRequested();
   7785         final boolean dockVisible = isStackVisibleLocked(DOCKED_STACK_ID);
   7786         final TaskStack imeTargetStack = getImeFocusStackLocked();
   7787         final int imeDockSide = (dockVisible && imeTargetStack != null) ?
   7788                 imeTargetStack.getDockSide() : DOCKED_INVALID;
   7789         final boolean imeOnTop = (imeDockSide == DOCKED_TOP);
   7790         final boolean imeOnBottom = (imeDockSide == DOCKED_BOTTOM);
   7791         final boolean dockMinimized = displayContent.mDividerControllerLocked.isMinimizedDock();
   7792         final int imeHeight = mPolicy.getInputMethodWindowVisibleHeightLw();
   7793         final boolean imeHeightChanged = imeVisible &&
   7794                 imeHeight != displayContent.mDividerControllerLocked.getImeHeightAdjustedFor();
   7795 
   7796         // The divider could be adjusted for IME position, or be thinner than usual,
   7797         // or both. There are three possible cases:
   7798         // - If IME is visible, and focus is on top, divider is not moved for IME but thinner.
   7799         // - If IME is visible, and focus is on bottom, divider is moved for IME and thinner.
   7800         // - If IME is not visible, divider is not moved and is normal width.
   7801 
   7802         if (imeVisible && dockVisible && (imeOnTop || imeOnBottom) && !dockMinimized) {
   7803             final ArrayList<TaskStack> stacks = displayContent.getStacks();
   7804             for (int i = stacks.size() - 1; i >= 0; --i) {
   7805                 final TaskStack stack = stacks.get(i);
   7806                 final boolean isDockedOnBottom = stack.getDockSide() == DOCKED_BOTTOM;
   7807                 if (stack.isVisibleLocked() && (imeOnBottom || isDockedOnBottom)) {
   7808                     stack.setAdjustedForIme(imeWin, imeOnBottom && imeHeightChanged);
   7809                 } else {
   7810                     stack.resetAdjustedForIme(false);
   7811                 }
   7812             }
   7813             displayContent.mDividerControllerLocked.setAdjustedForIme(
   7814                     imeOnBottom /*ime*/, true /*divider*/, true /*animate*/, imeWin, imeHeight);
   7815         } else {
   7816             final ArrayList<TaskStack> stacks = displayContent.getStacks();
   7817             for (int i = stacks.size() - 1; i >= 0; --i) {
   7818                 final TaskStack stack = stacks.get(i);
   7819                 stack.resetAdjustedForIme(!dockVisible);
   7820             }
   7821             displayContent.mDividerControllerLocked.setAdjustedForIme(
   7822                     false /*ime*/, false /*divider*/, dockVisible /*animate*/, imeWin, imeHeight);
   7823         }
   7824     }
   7825 
   7826     // -------------------------------------------------------------
   7827     // Drag and drop
   7828     // -------------------------------------------------------------
   7829 
   7830     IBinder prepareDragSurface(IWindow window, SurfaceSession session,
   7831             int flags, int width, int height, Surface outSurface) {
   7832         if (DEBUG_DRAG) {
   7833             Slog.d(TAG_WM, "prepare drag surface: w=" + width + " h=" + height
   7834                     + " flags=" + Integer.toHexString(flags) + " win=" + window
   7835                     + " asbinder=" + window.asBinder());
   7836         }
   7837 
   7838         final int callerPid = Binder.getCallingPid();
   7839         final int callerUid = Binder.getCallingUid();
   7840         final long origId = Binder.clearCallingIdentity();
   7841         IBinder token = null;
   7842 
   7843         try {
   7844             synchronized (mWindowMap) {
   7845                 try {
   7846                     if (mDragState == null) {
   7847                         // TODO(multi-display): support other displays
   7848                         final DisplayContent displayContent = getDefaultDisplayContentLocked();
   7849                         final Display display = displayContent.getDisplay();
   7850 
   7851                         SurfaceControl surface = new SurfaceControl(session, "drag surface",
   7852                                 width, height, PixelFormat.TRANSLUCENT, SurfaceControl.HIDDEN);
   7853                         surface.setLayerStack(display.getLayerStack());
   7854                         float alpha = 1;
   7855                         if ((flags & View.DRAG_FLAG_OPAQUE) == 0) {
   7856                             alpha = DRAG_SHADOW_ALPHA_TRANSPARENT;
   7857                         }
   7858                         surface.setAlpha(alpha);
   7859 
   7860                         if (SHOW_TRANSACTIONS) Slog.i(TAG_WM, "  DRAG "
   7861                                 + surface + ": CREATE");
   7862                         outSurface.copyFrom(surface);
   7863                         final IBinder winBinder = window.asBinder();
   7864                         token = new Binder();
   7865                         mDragState = new DragState(this, token, surface, flags, winBinder);
   7866                         mDragState.mPid = callerPid;
   7867                         mDragState.mUid = callerUid;
   7868                         mDragState.mOriginalAlpha = alpha;
   7869                         token = mDragState.mToken = new Binder();
   7870 
   7871                         // 5 second timeout for this window to actually begin the drag
   7872                         mH.removeMessages(H.DRAG_START_TIMEOUT, winBinder);
   7873                         Message msg = mH.obtainMessage(H.DRAG_START_TIMEOUT, winBinder);
   7874                         mH.sendMessageDelayed(msg, 5000);
   7875                     } else {
   7876                         Slog.w(TAG_WM, "Drag already in progress");
   7877                     }
   7878                 } catch (OutOfResourcesException e) {
   7879                     Slog.e(TAG_WM, "Can't allocate drag surface w=" + width + " h=" + height, e);
   7880                     if (mDragState != null) {
   7881                         mDragState.reset();
   7882                         mDragState = null;
   7883                     }
   7884                 }
   7885             }
   7886         } finally {
   7887             Binder.restoreCallingIdentity(origId);
   7888         }
   7889 
   7890         return token;
   7891     }
   7892 
   7893     // -------------------------------------------------------------
   7894     // Input Events and Focus Management
   7895     // -------------------------------------------------------------
   7896 
   7897     final InputMonitor mInputMonitor = new InputMonitor(this);
   7898     private boolean mEventDispatchingEnabled;
   7899 
   7900     @Override
   7901     public void pauseKeyDispatching(IBinder _token) {
   7902         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   7903                 "pauseKeyDispatching()")) {
   7904             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   7905         }
   7906 
   7907         synchronized (mWindowMap) {
   7908             WindowToken token = mTokenMap.get(_token);
   7909             if (token != null) {
   7910                 mInputMonitor.pauseDispatchingLw(token);
   7911             }
   7912         }
   7913     }
   7914 
   7915     @Override
   7916     public void resumeKeyDispatching(IBinder _token) {
   7917         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   7918                 "resumeKeyDispatching()")) {
   7919             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   7920         }
   7921 
   7922         synchronized (mWindowMap) {
   7923             WindowToken token = mTokenMap.get(_token);
   7924             if (token != null) {
   7925                 mInputMonitor.resumeDispatchingLw(token);
   7926             }
   7927         }
   7928     }
   7929 
   7930     @Override
   7931     public void setEventDispatching(boolean enabled) {
   7932         if (!checkCallingPermission(android.Manifest.permission.MANAGE_APP_TOKENS,
   7933                 "setEventDispatching()")) {
   7934             throw new SecurityException("Requires MANAGE_APP_TOKENS permission");
   7935         }
   7936 
   7937         synchronized (mWindowMap) {
   7938             mEventDispatchingEnabled = enabled;
   7939             if (mDisplayEnabled) {
   7940                 mInputMonitor.setEventDispatchingLw(enabled);
   7941             }
   7942         }
   7943     }
   7944 
   7945     private WindowState getFocusedWindow() {
   7946         synchronized (mWindowMap) {
   7947             return getFocusedWindowLocked();
   7948         }
   7949     }
   7950 
   7951     private WindowState getFocusedWindowLocked() {
   7952         return mCurrentFocus;
   7953     }
   7954 
   7955     TaskStack getImeFocusStackLocked() {
   7956         // Don't use mCurrentFocus.getStack() because it returns home stack for system windows.
   7957         // Also don't use mInputMethodTarget's stack, because some window with FLAG_NOT_FOCUSABLE
   7958         // and FLAG_ALT_FOCUSABLE_IM flags both set might be set to IME target so they're moved
   7959         // to make room for IME, but the window is not the focused window that's taking input.
   7960         return (mFocusedApp != null && mFocusedApp.mTask != null) ?
   7961                 mFocusedApp.mTask.mStack : null;
   7962     }
   7963 
   7964     private void showAuditSafeModeNotification() {
   7965         PendingIntent pendingIntent = PendingIntent.getActivity(mContext, 0,
   7966                 new Intent(Intent.ACTION_VIEW,
   7967                            Uri.parse("https://support.google.com/nexus/answer/2852139")), 0);
   7968 
   7969         String title = mContext.getString(R.string.audit_safemode_notification);
   7970 
   7971         Notification notification = new Notification.Builder(mContext)
   7972                 .setSmallIcon(com.android.internal.R.drawable.stat_sys_warning)
   7973                 .setWhen(0)
   7974                 .setOngoing(true)
   7975                 .setTicker(title)
   7976                 .setLocalOnly(true)
   7977                 .setPriority(Notification.PRIORITY_HIGH)
   7978                 .setVisibility(Notification.VISIBILITY_PUBLIC)
   7979                 .setColor(mContext.getColor(
   7980                         com.android.internal.R.color.system_notification_accent_color))
   7981                 .setContentTitle(title)
   7982                 .setContentText(mContext.getString(R.string.audit_safemode_notification_details))
   7983                 .setContentIntent(pendingIntent)
   7984                 .build();
   7985 
   7986         NotificationManager notificationManager = (NotificationManager) mContext
   7987                 .getSystemService(Context.NOTIFICATION_SERVICE);
   7988 
   7989         notificationManager.notifyAsUser(null, R.string.audit_safemode_notification, notification,
   7990                 UserHandle.ALL);
   7991     }
   7992 
   7993     public boolean detectSafeMode() {
   7994         if (!mInputMonitor.waitForInputDevicesReady(
   7995                 INPUT_DEVICES_READY_FOR_SAFE_MODE_DETECTION_TIMEOUT_MILLIS)) {
   7996             Slog.w(TAG_WM, "Devices still not ready after waiting "
   7997                    + INPUT_DEVICES_READY_FOR_SAFE_MODE_DETECTION_TIMEOUT_MILLIS
   7998                    + " milliseconds before attempting to detect safe mode.");
   7999         }
   8000 
   8001         if (Settings.Global.getInt(
   8002                 mContext.getContentResolver(), Settings.Global.SAFE_BOOT_DISALLOWED, 0) != 0) {
   8003             return false;
   8004         }
   8005 
   8006         int menuState = mInputManager.getKeyCodeState(-1, InputDevice.SOURCE_ANY,
   8007                 KeyEvent.KEYCODE_MENU);
   8008         int sState = mInputManager.getKeyCodeState(-1, InputDevice.SOURCE_ANY, KeyEvent.KEYCODE_S);
   8009         int dpadState = mInputManager.getKeyCodeState(-1, InputDevice.SOURCE_DPAD,
   8010                 KeyEvent.KEYCODE_DPAD_CENTER);
   8011         int trackballState = mInputManager.getScanCodeState(-1, InputDevice.SOURCE_TRACKBALL,
   8012                 InputManagerService.BTN_MOUSE);
   8013         int volumeDownState = mInputManager.getKeyCodeState(-1, InputDevice.SOURCE_ANY,
   8014                 KeyEvent.KEYCODE_VOLUME_DOWN);
   8015         mSafeMode = menuState > 0 || sState > 0 || dpadState > 0 || trackballState > 0
   8016                 || volumeDownState > 0;
   8017         try {
   8018             if (SystemProperties.getInt(ShutdownThread.REBOOT_SAFEMODE_PROPERTY, 0) != 0
   8019                     || SystemProperties.getInt(ShutdownThread.RO_SAFEMODE_PROPERTY, 0) != 0) {
   8020                 int auditSafeMode = SystemProperties.getInt(ShutdownThread.AUDIT_SAFEMODE_PROPERTY, 0);
   8021 
   8022                 if (auditSafeMode == 0) {
   8023                     mSafeMode = true;
   8024                     SystemProperties.set(ShutdownThread.REBOOT_SAFEMODE_PROPERTY, "");
   8025                 } else {
   8026                     // stay in safe mode until we have updated to a newer build
   8027                     int buildDate = SystemProperties.getInt(PROPERTY_BUILD_DATE_UTC, 0);
   8028 
   8029                     if (auditSafeMode >= buildDate) {
   8030                         mSafeMode = true;
   8031                         showAuditSafeModeNotification();
   8032                     } else {
   8033                         SystemProperties.set(ShutdownThread.REBOOT_SAFEMODE_PROPERTY, "");
   8034                         SystemProperties.set(ShutdownThread.AUDIT_SAFEMODE_PROPERTY, "");
   8035                     }
   8036                 }
   8037             }
   8038         } catch (IllegalArgumentException e) {
   8039         }
   8040         if (mSafeMode) {
   8041             Log.i(TAG_WM, "SAFE MODE ENABLED (menu=" + menuState + " s=" + sState
   8042                     + " dpad=" + dpadState + " trackball=" + trackballState + ")");
   8043             SystemProperties.set(ShutdownThread.RO_SAFEMODE_PROPERTY, "1");
   8044         } else {
   8045             Log.i(TAG_WM, "SAFE MODE not enabled");
   8046         }
   8047         mPolicy.setSafeMode(mSafeMode);
   8048         return mSafeMode;
   8049     }
   8050 
   8051     public void displayReady() {
   8052         for (Display display : mDisplays) {
   8053             displayReady(display.getDisplayId());
   8054         }
   8055 
   8056         synchronized(mWindowMap) {
   8057             final DisplayContent displayContent = getDefaultDisplayContentLocked();
   8058             readForcedDisplayPropertiesLocked(displayContent);
   8059             mDisplayReady = true;
   8060         }
   8061 
   8062         try {
   8063             mActivityManager.updateConfiguration(null);
   8064         } catch (RemoteException e) {
   8065         }
   8066 
   8067         synchronized(mWindowMap) {
   8068             mIsTouchDevice = mContext.getPackageManager().hasSystemFeature(
   8069                     PackageManager.FEATURE_TOUCHSCREEN);
   8070             configureDisplayPolicyLocked(getDefaultDisplayContentLocked());
   8071         }
   8072 
   8073         try {
   8074             mActivityManager.updateConfiguration(null);
   8075         } catch (RemoteException e) {
   8076         }
   8077 
   8078         updateCircularDisplayMaskIfNeeded();
   8079     }
   8080 
   8081     private void displayReady(int displayId) {
   8082         synchronized(mWindowMap) {
   8083             final DisplayContent displayContent = getDisplayContentLocked(displayId);
   8084             if (displayContent != null) {
   8085                 mAnimator.addDisplayLocked(displayId);
   8086                 displayContent.initializeDisplayBaseInfo();
   8087                 if (displayContent.mTapDetector != null) {
   8088                     displayContent.mTapDetector.init();
   8089                 }
   8090             }
   8091         }
   8092     }
   8093 
   8094     public void systemReady() {
   8095         mPolicy.systemReady();
   8096     }
   8097 
   8098     // -------------------------------------------------------------
   8099     // Async Handler
   8100     // -------------------------------------------------------------
   8101 
   8102     final class H extends Handler {
   8103         public static final int REPORT_FOCUS_CHANGE = 2;
   8104         public static final int REPORT_LOSING_FOCUS = 3;
   8105         public static final int DO_TRAVERSAL = 4;
   8106         public static final int ADD_STARTING = 5;
   8107         public static final int REMOVE_STARTING = 6;
   8108         public static final int FINISHED_STARTING = 7;
   8109         public static final int REPORT_APPLICATION_TOKEN_WINDOWS = 8;
   8110         public static final int REPORT_APPLICATION_TOKEN_DRAWN = 9;
   8111         public static final int WINDOW_FREEZE_TIMEOUT = 11;
   8112 
   8113         public static final int APP_TRANSITION_TIMEOUT = 13;
   8114         public static final int PERSIST_ANIMATION_SCALE = 14;
   8115         public static final int FORCE_GC = 15;
   8116         public static final int ENABLE_SCREEN = 16;
   8117         public static final int APP_FREEZE_TIMEOUT = 17;
   8118         public static final int SEND_NEW_CONFIGURATION = 18;
   8119         public static final int REPORT_WINDOWS_CHANGE = 19;
   8120         public static final int DRAG_START_TIMEOUT = 20;
   8121         public static final int DRAG_END_TIMEOUT = 21;
   8122         public static final int REPORT_HARD_KEYBOARD_STATUS_CHANGE = 22;
   8123         public static final int BOOT_TIMEOUT = 23;
   8124         public static final int WAITING_FOR_DRAWN_TIMEOUT = 24;
   8125         public static final int SHOW_STRICT_MODE_VIOLATION = 25;
   8126         public static final int DO_ANIMATION_CALLBACK = 26;
   8127 
   8128         public static final int DO_DISPLAY_ADDED = 27;
   8129         public static final int DO_DISPLAY_REMOVED = 28;
   8130         public static final int DO_DISPLAY_CHANGED = 29;
   8131 
   8132         public static final int CLIENT_FREEZE_TIMEOUT = 30;
   8133         public static final int TAP_OUTSIDE_TASK = 31;
   8134         public static final int NOTIFY_ACTIVITY_DRAWN = 32;
   8135 
   8136         public static final int ALL_WINDOWS_DRAWN = 33;
   8137 
   8138         public static final int NEW_ANIMATOR_SCALE = 34;
   8139 
   8140         public static final int SHOW_CIRCULAR_DISPLAY_MASK = 35;
   8141         public static final int SHOW_EMULATOR_DISPLAY_OVERLAY = 36;
   8142 
   8143         public static final int CHECK_IF_BOOT_ANIMATION_FINISHED = 37;
   8144         public static final int RESET_ANR_MESSAGE = 38;
   8145         public static final int WALLPAPER_DRAW_PENDING_TIMEOUT = 39;
   8146 
   8147         public static final int FINISH_TASK_POSITIONING = 40;
   8148 
   8149         public static final int UPDATE_DOCKED_STACK_DIVIDER = 41;
   8150 
   8151         public static final int RESIZE_STACK = 42;
   8152         public static final int RESIZE_TASK = 43;
   8153 
   8154         public static final int TWO_FINGER_SCROLL_START = 44;
   8155 
   8156         public static final int WINDOW_REPLACEMENT_TIMEOUT = 46;
   8157 
   8158         public static final int NOTIFY_APP_TRANSITION_STARTING = 47;
   8159         public static final int NOTIFY_APP_TRANSITION_CANCELLED = 48;
   8160         public static final int NOTIFY_APP_TRANSITION_FINISHED = 49;
   8161         public static final int NOTIFY_STARTING_WINDOW_DRAWN = 50;
   8162         public static final int UPDATE_ANIMATION_SCALE = 51;
   8163         public static final int WINDOW_REMOVE_TIMEOUT = 52;
   8164         public static final int NOTIFY_DOCKED_STACK_MINIMIZED_CHANGED = 53;
   8165         public static final int SEAMLESS_ROTATION_TIMEOUT = 54;
   8166 
   8167         /**
   8168          * Used to denote that an integer field in a message will not be used.
   8169          */
   8170         public static final int UNUSED = 0;
   8171 
   8172         @Override
   8173         public void handleMessage(Message msg) {
   8174             if (DEBUG_WINDOW_TRACE) {
   8175                 Slog.v(TAG_WM, "handleMessage: entry what=" + msg.what);
   8176             }
   8177             switch (msg.what) {
   8178                 case REPORT_FOCUS_CHANGE: {
   8179                     WindowState lastFocus;
   8180                     WindowState newFocus;
   8181 
   8182                     AccessibilityController accessibilityController = null;
   8183 
   8184                     synchronized(mWindowMap) {
   8185                         // TODO(multidisplay): Accessibility supported only of default desiplay.
   8186                         if (mAccessibilityController != null && getDefaultDisplayContentLocked()
   8187                                 .getDisplayId() == Display.DEFAULT_DISPLAY) {
   8188                             accessibilityController = mAccessibilityController;
   8189                         }
   8190 
   8191                         lastFocus = mLastFocus;
   8192                         newFocus = mCurrentFocus;
   8193                         if (lastFocus == newFocus) {
   8194                             // Focus is not changing, so nothing to do.
   8195                             return;
   8196                         }
   8197                         mLastFocus = newFocus;
   8198                         if (DEBUG_FOCUS_LIGHT) Slog.i(TAG_WM, "Focus moving from " + lastFocus +
   8199                                 " to " + newFocus);
   8200                         if (newFocus != null && lastFocus != null
   8201                                 && !newFocus.isDisplayedLw()) {
   8202                             //Slog.i(TAG_WM, "Delaying loss of focus...");
   8203                             mLosingFocus.add(lastFocus);
   8204                             lastFocus = null;
   8205                         }
   8206                     }
   8207 
   8208                     // First notify the accessibility manager for the change so it has
   8209                     // the windows before the newly focused one starts firing eventgs.
   8210                     if (accessibilityController != null) {
   8211                         accessibilityController.onWindowFocusChangedNotLocked();
   8212                     }
   8213 
   8214                     //System.out.println("Changing focus from " + lastFocus
   8215                     //                   + " to " + newFocus);
   8216                     if (newFocus != null) {
   8217                         if (DEBUG_FOCUS_LIGHT) Slog.i(TAG_WM, "Gaining focus: " + newFocus);
   8218                         newFocus.reportFocusChangedSerialized(true, mInTouchMode);
   8219                         notifyFocusChanged();
   8220                     }
   8221 
   8222                     if (lastFocus != null) {
   8223                         if (DEBUG_FOCUS_LIGHT) Slog.i(TAG_WM, "Losing focus: " + lastFocus);
   8224                         lastFocus.reportFocusChangedSerialized(false, mInTouchMode);
   8225                     }
   8226                 } break;
   8227 
   8228                 case REPORT_LOSING_FOCUS: {
   8229                     ArrayList<WindowState> losers;
   8230 
   8231                     synchronized(mWindowMap) {
   8232                         losers = mLosingFocus;
   8233                         mLosingFocus = new ArrayList<WindowState>();
   8234                     }
   8235 
   8236                     final int N = losers.size();
   8237                     for (int i=0; i<N; i++) {
   8238                         if (DEBUG_FOCUS_LIGHT) Slog.i(TAG_WM, "Losing delayed focus: " +
   8239                                 losers.get(i));
   8240                         losers.get(i).reportFocusChangedSerialized(false, mInTouchMode);
   8241                     }
   8242                 } break;
   8243 
   8244                 case DO_TRAVERSAL: {
   8245                     synchronized(mWindowMap) {
   8246                         mWindowPlacerLocked.performSurfacePlacement();
   8247                     }
   8248                 } break;
   8249 
   8250                 case ADD_STARTING: {
   8251                     final AppWindowToken wtoken = (AppWindowToken)msg.obj;
   8252                     final StartingData sd = wtoken.startingData;
   8253 
   8254                     if (sd == null) {
   8255                         // Animation has been canceled... do nothing.
   8256                         return;
   8257                     }
   8258 
   8259                     if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM, "Add starting "
   8260                             + wtoken + ": pkg=" + sd.pkg);
   8261 
   8262                     View view = null;
   8263                     try {
   8264                         final Configuration overrideConfig = wtoken != null && wtoken.mTask != null
   8265                                 ? wtoken.mTask.mOverrideConfig : null;
   8266                         view = mPolicy.addStartingWindow(wtoken.token, sd.pkg, sd.theme,
   8267                             sd.compatInfo, sd.nonLocalizedLabel, sd.labelRes, sd.icon, sd.logo,
   8268                             sd.windowFlags, overrideConfig);
   8269                     } catch (Exception e) {
   8270                         Slog.w(TAG_WM, "Exception when adding starting window", e);
   8271                     }
   8272 
   8273                     if (view != null) {
   8274                         boolean abort = false;
   8275 
   8276                         synchronized(mWindowMap) {
   8277                             if (wtoken.removed || wtoken.startingData == null) {
   8278                                 // If the window was successfully added, then
   8279                                 // we need to remove it.
   8280                                 if (wtoken.startingWindow != null) {
   8281                                     if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM,
   8282                                             "Aborted starting " + wtoken
   8283                                             + ": removed=" + wtoken.removed
   8284                                             + " startingData=" + wtoken.startingData);
   8285                                     wtoken.startingWindow = null;
   8286                                     wtoken.startingData = null;
   8287                                     abort = true;
   8288                                 }
   8289                             } else {
   8290                                 wtoken.startingView = view;
   8291                             }
   8292                             if (DEBUG_STARTING_WINDOW && !abort) Slog.v(TAG_WM,
   8293                                     "Added starting " + wtoken
   8294                                     + ": startingWindow="
   8295                                     + wtoken.startingWindow + " startingView="
   8296                                     + wtoken.startingView);
   8297                         }
   8298 
   8299                         if (abort) {
   8300                             try {
   8301                                 mPolicy.removeStartingWindow(wtoken.token, view);
   8302                             } catch (Exception e) {
   8303                                 Slog.w(TAG_WM, "Exception when removing starting window", e);
   8304                             }
   8305                         }
   8306                     }
   8307                 } break;
   8308 
   8309                 case REMOVE_STARTING: {
   8310                     final AppWindowToken wtoken = (AppWindowToken)msg.obj;
   8311                     IBinder token = null;
   8312                     View view = null;
   8313                     synchronized (mWindowMap) {
   8314                         if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM, "Remove starting "
   8315                                 + wtoken + ": startingWindow="
   8316                                 + wtoken.startingWindow + " startingView="
   8317                                 + wtoken.startingView);
   8318                         if (wtoken.startingWindow != null) {
   8319                             view = wtoken.startingView;
   8320                             token = wtoken.token;
   8321                             wtoken.startingData = null;
   8322                             wtoken.startingView = null;
   8323                             wtoken.startingWindow = null;
   8324                             wtoken.startingDisplayed = false;
   8325                         }
   8326                     }
   8327                     if (view != null) {
   8328                         try {
   8329                             mPolicy.removeStartingWindow(token, view);
   8330                         } catch (Exception e) {
   8331                             Slog.w(TAG_WM, "Exception when removing starting window", e);
   8332                         }
   8333                     }
   8334                 } break;
   8335 
   8336                 case FINISHED_STARTING: {
   8337                     IBinder token = null;
   8338                     View view = null;
   8339                     while (true) {
   8340                         synchronized (mWindowMap) {
   8341                             final int N = mFinishedStarting.size();
   8342                             if (N <= 0) {
   8343                                 break;
   8344                             }
   8345                             AppWindowToken wtoken = mFinishedStarting.remove(N-1);
   8346 
   8347                             if (DEBUG_STARTING_WINDOW) Slog.v(TAG_WM,
   8348                                     "Finished starting " + wtoken
   8349                                     + ": startingWindow=" + wtoken.startingWindow
   8350                                     + " startingView=" + wtoken.startingView);
   8351 
   8352                             if (wtoken.startingWindow == null) {
   8353                                 continue;
   8354                             }
   8355 
   8356                             view = wtoken.startingView;
   8357                             token = wtoken.token;
   8358                             wtoken.startingData = null;
   8359                             wtoken.startingView = null;
   8360                             wtoken.startingWindow = null;
   8361                             wtoken.startingDisplayed = false;
   8362                         }
   8363 
   8364                         try {
   8365                             mPolicy.removeStartingWindow(token, view);
   8366                         } catch (Exception e) {
   8367                             Slog.w(TAG_WM, "Exception when removing starting window", e);
   8368                         }
   8369                     }
   8370                 } break;
   8371 
   8372                 case REPORT_APPLICATION_TOKEN_DRAWN: {
   8373                     final AppWindowToken wtoken = (AppWindowToken)msg.obj;
   8374 
   8375                     try {
   8376                         if (DEBUG_VISIBILITY) Slog.v(
   8377                                 TAG_WM, "Reporting drawn in " + wtoken);
   8378                         wtoken.appToken.windowsDrawn();
   8379                     } catch (RemoteException ex) {
   8380                     }
   8381                 } break;
   8382 
   8383                 case REPORT_APPLICATION_TOKEN_WINDOWS: {
   8384                     final AppWindowToken wtoken = (AppWindowToken)msg.obj;
   8385 
   8386                     boolean nowVisible = msg.arg1 != 0;
   8387                     boolean nowGone = msg.arg2 != 0;
   8388 
   8389                     try {
   8390                         if (DEBUG_VISIBILITY) Slog.v(
   8391                                 TAG_WM, "Reporting visible in " + wtoken
   8392                                 + " visible=" + nowVisible
   8393                                 + " gone=" + nowGone);
   8394                         if (nowVisible) {
   8395                             wtoken.appToken.windowsVisible();
   8396                         } else {
   8397                             wtoken.appToken.windowsGone();
   8398                         }
   8399                     } catch (RemoteException ex) {
   8400                     }
   8401                 } break;
   8402 
   8403                 case WINDOW_FREEZE_TIMEOUT: {
   8404                     // TODO(multidisplay): Can non-default displays rotate?
   8405                     synchronized (mWindowMap) {
   8406                         Slog.w(TAG_WM, "Window freeze timeout expired.");
   8407                         mWindowsFreezingScreen = WINDOWS_FREEZING_SCREENS_TIMEOUT;
   8408                         final WindowList windows = getDefaultWindowListLocked();
   8409                         int i = windows.size();
   8410                         while (i > 0) {
   8411                             i--;
   8412                             WindowState w = windows.get(i);
   8413                             if (w.mOrientationChanging) {
   8414                                 w.mOrientationChanging = false;
   8415                                 w.mLastFreezeDuration = (int)(SystemClock.elapsedRealtime()
   8416                                         - mDisplayFreezeTime);
   8417                                 Slog.w(TAG_WM, "Force clearing orientation change: " + w);
   8418                             }
   8419                         }
   8420                         mWindowPlacerLocked.performSurfacePlacement();
   8421                     }
   8422                     break;
   8423                 }
   8424 
   8425                 case APP_TRANSITION_TIMEOUT: {
   8426                     synchronized (mWindowMap) {
   8427                         if (mAppTransition.isTransitionSet() || !mOpeningApps.isEmpty()
   8428                                     || !mClosingApps.isEmpty()) {
   8429                             if (DEBUG_APP_TRANSITIONS) Slog.v(TAG_WM, "*** APP TRANSITION TIMEOUT."
   8430                                     + " isTransitionSet()=" + mAppTransition.isTransitionSet()
   8431                                     + " mOpeningApps.size()=" + mOpeningApps.size()
   8432                                     + " mClosingApps.size()=" + mClosingApps.size());
   8433                             mAppTransition.setTimeout();
   8434                             mWindowPlacerLocked.performSurfacePlacement();
   8435                         }
   8436                     }
   8437                     break;
   8438                 }
   8439 
   8440                 case PERSIST_ANIMATION_SCALE: {
   8441                     Settings.Global.putFloat(mContext.getContentResolver(),
   8442                             Settings.Global.WINDOW_ANIMATION_SCALE, mWindowAnimationScaleSetting);
   8443                     Settings.Global.putFloat(mContext.getContentResolver(),
   8444                             Settings.Global.TRANSITION_ANIMATION_SCALE,
   8445                             mTransitionAnimationScaleSetting);
   8446                     Settings.Global.putFloat(mContext.getContentResolver(),
   8447                             Settings.Global.ANIMATOR_DURATION_SCALE, mAnimatorDurationScaleSetting);
   8448                     break;
   8449                 }
   8450 
   8451                 case UPDATE_ANIMATION_SCALE: {
   8452                     @UpdateAnimationScaleMode
   8453                     final int mode = msg.arg1;
   8454                     switch (mode) {
   8455                         case WINDOW_ANIMATION_SCALE: {
   8456                             mWindowAnimationScaleSetting = Settings.Global.getFloat(
   8457                                     mContext.getContentResolver(),
   8458                                     Settings.Global.WINDOW_ANIMATION_SCALE,
   8459                                     mWindowAnimationScaleSetting);
   8460                             break;
   8461                         }
   8462                         case TRANSITION_ANIMATION_SCALE: {
   8463                             mTransitionAnimationScaleSetting = Settings.Global.getFloat(
   8464                                     mContext.getContentResolver(),
   8465                                     Settings.Global.TRANSITION_ANIMATION_SCALE,
   8466                                     mTransitionAnimationScaleSetting);
   8467                             break;
   8468                         }
   8469                         case ANIMATION_DURATION_SCALE: {
   8470                             mAnimatorDurationScaleSetting = Settings.Global.getFloat(
   8471                                     mContext.getContentResolver(),
   8472                                     Settings.Global.ANIMATOR_DURATION_SCALE,
   8473                                     mAnimatorDurationScaleSetting);
   8474                             dispatchNewAnimatorScaleLocked(null);
   8475                             break;
   8476                         }
   8477                     }
   8478                     break;
   8479                 }
   8480 
   8481                 case FORCE_GC: {
   8482                     synchronized (mWindowMap) {
   8483                         // Since we're holding both mWindowMap and mAnimator we don't need to
   8484                         // hold mAnimator.mLayoutToAnim.
   8485                         if (mAnimator.isAnimating() || mAnimationScheduled) {
   8486                             // If we are animating, don't do the gc now but
   8487                             // delay a bit so we don't interrupt the animation.
   8488                             sendEmptyMessageDelayed(H.FORCE_GC, 2000);
   8489                             return;
   8490                         }
   8491                         // If we are currently rotating the display, it will
   8492                         // schedule a new message when done.
   8493                         if (mDisplayFrozen) {
   8494                             return;
   8495                         }
   8496                     }
   8497                     Runtime.getRuntime().gc();
   8498                     break;
   8499                 }
   8500 
   8501                 case ENABLE_SCREEN: {
   8502                     performEnableScreen();
   8503                     break;
   8504                 }
   8505 
   8506                 case APP_FREEZE_TIMEOUT: {
   8507                     synchronized (mWindowMap) {
   8508                         Slog.w(TAG_WM, "App freeze timeout expired.");
   8509                         mWindowsFreezingScreen = WINDOWS_FREEZING_SCREENS_TIMEOUT;
   8510                         final int numStacks = mStackIdToStack.size();
   8511                         for (int stackNdx = 0; stackNdx < numStacks; ++stackNdx) {
   8512                             final TaskStack stack = mStackIdToStack.valueAt(stackNdx);
   8513                             final ArrayList<Task> tasks = stack.getTasks();
   8514                             for (int taskNdx = tasks.size() - 1; taskNdx >= 0; --taskNdx) {
   8515                                 AppTokenList tokens = tasks.get(taskNdx).mAppTokens;
   8516                                 for (int tokenNdx = tokens.size() - 1; tokenNdx >= 0; --tokenNdx) {
   8517                                     AppWindowToken tok = tokens.get(tokenNdx);
   8518                                     if (tok.mAppAnimator.freezingScreen) {
   8519                                         Slog.w(TAG_WM, "Force clearing freeze: " + tok);
   8520                                         unsetAppFreezingScreenLocked(tok, true, true);
   8521                                     }
   8522                                 }
   8523                             }
   8524                         }
   8525                     }
   8526                     break;
   8527                 }
   8528 
   8529                 case CLIENT_FREEZE_TIMEOUT: {
   8530                     synchronized (mWindowMap) {
   8531                         if (mClientFreezingScreen) {
   8532                             mClientFreezingScreen = false;
   8533                             mLastFinishedFreezeSource = "client-timeout";
   8534                             stopFreezingDisplayLocked();
   8535                         }
   8536                     }
   8537                     break;
   8538                 }
   8539 
   8540                 case SEND_NEW_CONFIGURATION: {
   8541                     removeMessages(SEND_NEW_CONFIGURATION);
   8542                     sendNewConfiguration();
   8543                     break;
   8544                 }
   8545 
   8546                 case REPORT_WINDOWS_CHANGE: {
   8547                     if (mWindowsChanged) {
   8548                         synchronized (mWindowMap) {
   8549                             mWindowsChanged = false;
   8550                         }
   8551                         notifyWindowsChanged();
   8552                     }
   8553                     break;
   8554                 }
   8555 
   8556                 case DRAG_START_TIMEOUT: {
   8557                     IBinder win = (IBinder)msg.obj;
   8558                     if (DEBUG_DRAG) {
   8559                         Slog.w(TAG_WM, "Timeout starting drag by win " + win);
   8560                     }
   8561                     synchronized (mWindowMap) {
   8562                         // !!! TODO: ANR the app that has failed to start the drag in time
   8563                         if (mDragState != null) {
   8564                             mDragState.unregister();
   8565                             mInputMonitor.updateInputWindowsLw(true /*force*/);
   8566                             mDragState.reset();
   8567                             mDragState = null;
   8568                         }
   8569                     }
   8570                     break;
   8571                 }
   8572 
   8573                 case DRAG_END_TIMEOUT: {
   8574                     IBinder win = (IBinder)msg.obj;
   8575                     if (DEBUG_DRAG) {
   8576                         Slog.w(TAG_WM, "Timeout ending drag to win " + win);
   8577                     }
   8578                     synchronized (mWindowMap) {
   8579                         // !!! TODO: ANR the drag-receiving app
   8580                         if (mDragState != null) {
   8581                             mDragState.mDragResult = false;
   8582                             mDragState.endDragLw();
   8583                         }
   8584                     }
   8585                     break;
   8586                 }
   8587 
   8588                 case REPORT_HARD_KEYBOARD_STATUS_CHANGE: {
   8589                     notifyHardKeyboardStatusChange();
   8590                     break;
   8591                 }
   8592 
   8593                 case BOOT_TIMEOUT: {
   8594                     performBootTimeout();
   8595                     break;
   8596                 }
   8597 
   8598                 case WAITING_FOR_DRAWN_TIMEOUT: {
   8599                     Runnable callback = null;
   8600                     synchronized (mWindowMap) {
   8601                         Slog.w(TAG_WM, "Timeout waiting for drawn: undrawn=" + mWaitingForDrawn);
   8602                         mWaitingForDrawn.clear();
   8603                         callback = mWaitingForDrawnCallback;
   8604                         mWaitingForDrawnCallback = null;
   8605                     }
   8606                     if (callback != null) {
   8607                         callback.run();
   8608                     }
   8609                     break;
   8610                 }
   8611 
   8612                 case SHOW_STRICT_MODE_VIOLATION: {
   8613                     showStrictModeViolation(msg.arg1, msg.arg2);
   8614                     break;
   8615                 }
   8616 
   8617                 case SHOW_CIRCULAR_DISPLAY_MASK: {
   8618                     showCircularMask(msg.arg1 == 1);
   8619                     break;
   8620                 }
   8621 
   8622                 case SHOW_EMULATOR_DISPLAY_OVERLAY: {
   8623                     showEmulatorDisplayOverlay();
   8624                     break;
   8625                 }
   8626 
   8627                 case DO_ANIMATION_CALLBACK: {
   8628                     try {
   8629                         ((IRemoteCallback)msg.obj).sendResult(null);
   8630                     } catch (RemoteException e) {
   8631                     }
   8632                     break;
   8633                 }
   8634 
   8635                 case DO_DISPLAY_ADDED:
   8636                     handleDisplayAdded(msg.arg1);
   8637                     break;
   8638 
   8639                 case DO_DISPLAY_REMOVED:
   8640                     synchronized (mWindowMap) {
   8641                         handleDisplayRemovedLocked(msg.arg1);
   8642                     }
   8643                     break;
   8644 
   8645                 case DO_DISPLAY_CHANGED:
   8646                     synchronized (mWindowMap) {
   8647                         handleDisplayChangedLocked(msg.arg1);
   8648                     }
   8649                     break;
   8650 
   8651                 case TWO_FINGER_SCROLL_START: {
   8652                     startScrollingTask((DisplayContent)msg.obj, msg.arg1, msg.arg2);
   8653                 }
   8654                 break;
   8655 
   8656                 case TAP_OUTSIDE_TASK: {
   8657                     handleTapOutsideTask((DisplayContent)msg.obj, msg.arg1, msg.arg2);
   8658                 }
   8659                 break;
   8660 
   8661                 case FINISH_TASK_POSITIONING: {
   8662                     finishPositioning();
   8663                 }
   8664                 break;
   8665 
   8666                 case NOTIFY_ACTIVITY_DRAWN:
   8667                     try {
   8668                         mActivityManager.notifyActivityDrawn((IBinder) msg.obj);
   8669                     } catch (RemoteException e) {
   8670                     }
   8671                     break;
   8672                 case ALL_WINDOWS_DRAWN: {
   8673                     Runnable callback;
   8674                     synchronized (mWindowMap) {
   8675                         callback = mWaitingForDrawnCallback;
   8676                         mWaitingForDrawnCallback = null;
   8677                     }
   8678                     if (callback != null) {
   8679                         callback.run();
   8680                     }
   8681                 }
   8682                 case NEW_ANIMATOR_SCALE: {
   8683                     float scale = getCurrentAnimatorScale();
   8684                     ValueAnimator.setDurationScale(scale);
   8685                     Session session = (Session)msg.obj;
   8686                     if (session != null) {
   8687                         try {
   8688                             session.mCallback.onAnimatorScaleChanged(scale);
   8689                         } catch (RemoteException e) {
   8690                         }
   8691                     } else {
   8692                         ArrayList<IWindowSessionCallback> callbacks
   8693                                 = new ArrayList<IWindowSessionCallback>();
   8694                         synchronized (mWindowMap) {
   8695                             for (int i=0; i<mSessions.size(); i++) {
   8696                                 callbacks.add(mSessions.valueAt(i).mCallback);
   8697                             }
   8698 
   8699                         }
   8700                         for (int i=0; i<callbacks.size(); i++) {
   8701                             try {
   8702                                 callbacks.get(i).onAnimatorScaleChanged(scale);
   8703                             } catch (RemoteException e) {
   8704                             }
   8705                         }
   8706                     }
   8707                 }
   8708                 break;
   8709                 case CHECK_IF_BOOT_ANIMATION_FINISHED: {
   8710                     final boolean bootAnimationComplete;
   8711                     synchronized (mWindowMap) {
   8712                         if (DEBUG_BOOT) Slog.i(TAG_WM, "CHECK_IF_BOOT_ANIMATION_FINISHED:");
   8713                         bootAnimationComplete = checkBootAnimationCompleteLocked();
   8714                     }
   8715                     if (bootAnimationComplete) {
   8716                         performEnableScreen();
   8717                     }
   8718                 }
   8719                 break;
   8720                 case RESET_ANR_MESSAGE: {
   8721                     synchronized (mWindowMap) {
   8722                         mLastANRState = null;
   8723                     }
   8724                 }
   8725                 break;
   8726                 case WALLPAPER_DRAW_PENDING_TIMEOUT: {
   8727                     synchronized (mWindowMap) {
   8728                         if (mWallpaperControllerLocked.processWallpaperDrawPendingTimeout()) {
   8729                             mWindowPlacerLocked.performSurfacePlacement();
   8730                         }
   8731                     }
   8732                 }
   8733                 case UPDATE_DOCKED_STACK_DIVIDER: {
   8734                     synchronized (mWindowMap) {
   8735                         final DisplayContent displayContent = getDefaultDisplayContentLocked();
   8736                         displayContent.getDockedDividerController().reevaluateVisibility(false);
   8737                         adjustForImeIfNeeded(displayContent);
   8738                     }
   8739                 }
   8740                 break;
   8741                 case RESIZE_TASK: {
   8742                     try {
   8743                         mActivityManager.resizeTask(msg.arg1, (Rect) msg.obj, msg.arg2);
   8744                     } catch (RemoteException e) {
   8745                         // This will not happen since we are in the same process.
   8746                     }
   8747                 }
   8748                 break;
   8749                 case RESIZE_STACK: {
   8750                     try {
   8751                         mActivityManager.resizeStack(
   8752                                 msg.arg1, (Rect) msg.obj, msg.arg2 == 1, false, false, -1);
   8753                     } catch (RemoteException e) {
   8754                         // This will not happen since we are in the same process.
   8755                     }
   8756                 }
   8757                 break;
   8758                 case WINDOW_REPLACEMENT_TIMEOUT: {
   8759                     synchronized (mWindowMap) {
   8760                         for (int i = mReplacingWindowTimeouts.size() - 1; i >= 0; i--) {
   8761                             final AppWindowToken token = mReplacingWindowTimeouts.get(i);
   8762                             token.clearTimedoutReplacesLocked();
   8763                         }
   8764                         mReplacingWindowTimeouts.clear();
   8765                     }
   8766                 }
   8767                 case NOTIFY_APP_TRANSITION_STARTING: {
   8768                     mAmInternal.notifyAppTransitionStarting(msg.arg1);
   8769                 }
   8770                 break;
   8771                 case NOTIFY_APP_TRANSITION_CANCELLED: {
   8772                     mAmInternal.notifyAppTransitionCancelled();
   8773                 }
   8774                 break;
   8775                 case NOTIFY_APP_TRANSITION_FINISHED: {
   8776                     mAmInternal.notifyAppTransitionFinished();
   8777                 }
   8778                 break;
   8779                 case NOTIFY_STARTING_WINDOW_DRAWN: {
   8780                     mAmInternal.notifyStartingWindowDrawn();
   8781                 }
   8782                 break;
   8783                 case WINDOW_REMOVE_TIMEOUT: {
   8784                     final WindowState window = (WindowState) msg.obj;
   8785                     synchronized(mWindowMap) {
   8786                         // TODO: This is all about fixing b/21693547
   8787                         // where partially initialized Toasts get stuck
   8788                         // around and keep the screen on. We'd like
   8789                         // to just remove the toast...but this can cause clients
   8790                         // who miss the timeout due to normal circumstances (e.g.
   8791                         // running under debugger) to crash (b/29105388). The windows will
   8792                         // eventually be removed when the client process finishes.
   8793                         // The best we can do for now is remove the FLAG_KEEP_SCREEN_ON
   8794                         // and prevent the symptoms of b/21693547.
   8795                         window.mAttrs.flags &= ~FLAG_KEEP_SCREEN_ON;
   8796                         window.setDisplayLayoutNeeded();
   8797                         mWindowPlacerLocked.performSurfacePlacement();
   8798                     }
   8799                 }
   8800                 break;
   8801                 case NOTIFY_DOCKED_STACK_MINIMIZED_CHANGED: {
   8802                     mAmInternal.notifyDockedStackMinimizedChanged(msg.arg1 == 1);
   8803                 }
   8804                 break;
   8805                 case SEAMLESS_ROTATION_TIMEOUT: {
   8806                     // Rotation only supported on primary display.
   8807                     // TODO(multi-display)
   8808                     synchronized(mWindowMap) {
   8809                         final DisplayContent displayContent = getDefaultDisplayContentLocked();
   8810                         final WindowList windows = displayContent.getWindowList();
   8811                         boolean layoutNeeded = false;
   8812                         for (int i = windows.size() - 1; i >= 0; i--) {
   8813                             WindowState w = windows.get(i);
   8814                             if (w.mSeamlesslyRotated) {
   8815                                 layoutNeeded = true;
   8816                                 w.setDisplayLayoutNeeded();
   8817                             }
   8818                             w.mSeamlesslyRotated = false;
   8819                         }
   8820                         if (layoutNeeded) {
   8821                             mWindowPlacerLocked.performSurfacePlacement();
   8822                         }
   8823                     }
   8824                 }
   8825                 break;
   8826             }
   8827             if (DEBUG_WINDOW_TRACE) {
   8828                 Slog.v(TAG_WM, "handleMessage: exit");
   8829             }
   8830         }
   8831     }
   8832 
   8833     void destroyPreservedSurfaceLocked() {
   8834         for (int i = mDestroyPreservedSurface.size() - 1; i >= 0 ; i--) {
   8835             final WindowState w = mDestroyPreservedSurface.get(i);
   8836             w.mWinAnimator.destroyPreservedSurfaceLocked();
   8837         }
   8838         mDestroyPreservedSurface.clear();
   8839     }
   8840 
   8841     void stopUsingSavedSurfaceLocked() {
   8842         for (int i = mFinishedEarlyAnim.size() - 1; i >= 0 ; i--) {
   8843             final AppWindowToken wtoken = mFinishedEarlyAnim.get(i);
   8844             wtoken.stopUsingSavedSurfaceLocked();
   8845         }
   8846         mFinishedEarlyAnim.clear();
   8847     }
   8848 
   8849     // -------------------------------------------------------------
   8850     // IWindowManager API
   8851     // -------------------------------------------------------------
   8852 
   8853     @Override
   8854     public IWindowSession openSession(IWindowSessionCallback callback, IInputMethodClient client,
   8855             IInputContext inputContext) {
   8856         if (client == null) throw new IllegalArgumentException("null client");
   8857         if (inputContext == null) throw new IllegalArgumentException("null inputContext");
   8858         Session session = new Session(this, callback, client, inputContext);
   8859         return session;
   8860     }
   8861 
   8862     @Override
   8863     public boolean inputMethodClientHasFocus(IInputMethodClient client) {
   8864         synchronized (mWindowMap) {
   8865             // The focus for the client is the window immediately below
   8866             // where we would place the input method window.
   8867             int idx = findDesiredInputMethodWindowIndexLocked(false);
   8868             if (idx > 0) {
   8869                 // TODO(multidisplay): IMEs are only supported on the default display.
   8870                 WindowState imFocus = getDefaultWindowListLocked().get(idx-1);
   8871                 if (DEBUG_INPUT_METHOD) {
   8872                     Slog.i(TAG_WM, "Desired input method target: " + imFocus);
   8873                     Slog.i(TAG_WM, "Current focus: " + mCurrentFocus);
   8874                     Slog.i(TAG_WM, "Last focus: " + mLastFocus);
   8875                 }
   8876                 if (imFocus != null) {
   8877                     // This may be a starting window, in which case we still want
   8878                     // to count it as okay.
   8879                     if (imFocus.mAttrs.type == LayoutParams.TYPE_APPLICATION_STARTING
   8880                             && imFocus.mAppToken != null) {
   8881                         // The client has definitely started, so it really should
   8882                         // have a window in this app token.  Let's look for it.
   8883                         for (int i=0; i<imFocus.mAppToken.windows.size(); i++) {
   8884                             WindowState w = imFocus.mAppToken.windows.get(i);
   8885                             if (w != imFocus) {
   8886                                 Log.i(TAG_WM, "Switching to real app window: " + w);
   8887                                 imFocus = w;
   8888                                 break;
   8889                             }
   8890                         }
   8891                     }
   8892                     if (DEBUG_INPUT_METHOD) {
   8893                         Slog.i(TAG_WM, "IM target client: " + imFocus.mSession.mClient);
   8894                         if (imFocus.mSession.mClient != null) {
   8895                             Slog.i(TAG_WM, "IM target client binder: "
   8896                                     + imFocus.mSession.mClient.asBinder());
   8897                             Slog.i(TAG_WM, "Requesting client binder: " + client.asBinder());
   8898                         }
   8899                     }
   8900                     if (imFocus.mSession.mClient != null &&
   8901                             imFocus.mSession.mClient.asBinder() == client.asBinder()) {
   8902                         return true;
   8903                     }
   8904                 }
   8905             }
   8906 
   8907             // Okay, how about this...  what is the current focus?
   8908             // It seems in some cases we may not have moved the IM
   8909             // target window, such as when it was in a pop-up window,
   8910             // so let's also look at the current focus.  (An example:
   8911             // go to Gmail, start searching so the keyboard goes up,
   8912             // press home.  Sometimes the IME won't go down.)
   8913             // Would be nice to fix this more correctly, but it's
   8914             // way at the end of a release, and this should be good enough.
   8915             if (mCurrentFocus != null && mCurrentFocus.mSession.mClient != null
   8916                     && mCurrentFocus.mSession.mClient.asBinder() == client.asBinder()) {
   8917                 return true;
   8918             }
   8919         }
   8920         return false;
   8921     }
   8922 
   8923     @Override
   8924     public void getInitialDisplaySize(int displayId, Point size) {
   8925         synchronized (mWindowMap) {
   8926             final DisplayContent displayContent = getDisplayContentLocked(displayId);
   8927             if (displayContent != null && displayContent.hasAccess(Binder.getCallingUid())) {
   8928                 size.x = displayContent.mInitialDisplayWidth;
   8929                 size.y = displayContent.mInitialDisplayHeight;
   8930             }
   8931         }
   8932     }
   8933 
   8934     @Override
   8935     public void getBaseDisplaySize(int displayId, Point size) {
   8936         synchronized (mWindowMap) {
   8937             final DisplayContent displayContent = getDisplayContentLocked(displayId);
   8938             if (displayContent != null && displayContent.hasAccess(Binder.getCallingUid())) {
   8939                 size.x = displayContent.mBaseDisplayWidth;
   8940                 size.y = displayContent.mBaseDisplayHeight;
   8941             }
   8942         }
   8943     }
   8944 
   8945     @Override
   8946     public void setForcedDisplaySize(int displayId, int width, int height) {
   8947         if (mContext.checkCallingOrSelfPermission(
   8948                 android.Manifest.permission.WRITE_SECURE_SETTINGS) !=
   8949                 PackageManager.PERMISSION_GRANTED) {
   8950             throw new SecurityException("Must hold permission " +
   8951                     android.Manifest.permission.WRITE_SECURE_SETTINGS);
   8952         }
   8953         if (displayId != Display.DEFAULT_DISPLAY) {
   8954             throw new IllegalArgumentException("Can only set the default display");
   8955         }
   8956         final long ident = Binder.clearCallingIdentity();
   8957         try {
   8958             synchronized(mWindowMap) {
   8959                 // Set some sort of reasonable bounds on the size of the display that we
   8960                 // will try to emulate.
   8961                 final int MIN_WIDTH = 200;
   8962                 final int MIN_HEIGHT = 200;
   8963                 final int MAX_SCALE = 2;
   8964                 final DisplayContent displayContent = getDisplayContentLocked(displayId);
   8965                 if (displayContent != null) {
   8966                     width = Math.min(Math.max(width, MIN_WIDTH),
   8967                             displayContent.mInitialDisplayWidth * MAX_SCALE);
   8968                     height = Math.min(Math.max(height, MIN_HEIGHT),
   8969                             displayContent.mInitialDisplayHeight * MAX_SCALE);
   8970                     setForcedDisplaySizeLocked(displayContent, width, height);
   8971                     Settings.Global.putString(mContext.getContentResolver(),
   8972                             Settings.Global.DISPLAY_SIZE_FORCED, width + "," + height);
   8973                 }
   8974             }
   8975         } finally {
   8976             Binder.restoreCallingIdentity(ident);
   8977         }
   8978     }
   8979 
   8980     @Override
   8981     public void setForcedDisplayScalingMode(int displayId, int mode) {
   8982         if (mContext.checkCallingOrSelfPermission(
   8983                 android.Manifest.permission.WRITE_SECURE_SETTINGS) !=
   8984                 PackageManager.PERMISSION_GRANTED) {
   8985             throw new SecurityException("Must hold permission " +
   8986                     android.Manifest.permission.WRITE_SECURE_SETTINGS);
   8987         }
   8988         if (displayId != Display.DEFAULT_DISPLAY) {
   8989             throw new IllegalArgumentException("Can only set the default display");
   8990         }
   8991         final long ident = Binder.clearCallingIdentity();
   8992         try {
   8993             synchronized(mWindowMap) {
   8994                 final DisplayContent displayContent = getDisplayContentLocked(displayId);
   8995                 if (displayContent != null) {
   8996                     if (mode < 0 || mode > 1) {
   8997                         mode = 0;
   8998                     }
   8999                     setForcedDisplayScalingModeLocked(displayContent, mode);
   9000                     Settings.Global.putInt(mContext.getContentResolver(),
   9001                             Settings.Global.DISPLAY_SCALING_FORCE, mode);
   9002                 }
   9003             }
   9004         } finally {
   9005             Binder.restoreCallingIdentity(ident);
   9006         }
   9007     }
   9008 
   9009     private void setForcedDisplayScalingModeLocked(DisplayContent displayContent, int mode) {
   9010         Slog.i(TAG_WM, "Using display scaling mode: " + (mode == 0 ? "auto" : "off"));
   9011         displayContent.mDisplayScalingDisabled = (mode != 0);
   9012         reconfigureDisplayLocked(displayContent);
   9013     }
   9014 
   9015     private void readForcedDisplayPropertiesLocked(final DisplayContent displayContent) {
   9016         // Display size.
   9017         String sizeStr = Settings.Global.getString(mContext.getContentResolver(),
   9018                 Settings.Global.DISPLAY_SIZE_FORCED);
   9019         if (sizeStr == null || sizeStr.length() == 0) {
   9020             sizeStr = SystemProperties.get(SIZE_OVERRIDE, null);
   9021         }
   9022         if (sizeStr != null && sizeStr.length() > 0) {
   9023             final int pos = sizeStr.indexOf(',');
   9024             if (pos > 0 && sizeStr.lastIndexOf(',') == pos) {
   9025                 int width, height;
   9026                 try {
   9027                     width = Integer.parseInt(sizeStr.substring(0, pos));
   9028                     height = Integer.parseInt(sizeStr.substring(pos+1));
   9029                     if (displayContent.mBaseDisplayWidth != width
   9030                             || displayContent.mBaseDisplayHeight != height) {
   9031                         Slog.i(TAG_WM, "FORCED DISPLAY SIZE: " + width + "x" + height);
   9032                         displayContent.mBaseDisplayWidth = width;
   9033                         displayContent.mBaseDisplayHeight = height;
   9034                     }
   9035                 } catch (NumberFormatException ex) {
   9036                 }
   9037             }
   9038         }
   9039 
   9040         // Display density.
   9041         final int density = getForcedDisplayDensityForUserLocked(mCurrentUserId);
   9042         if (density != 0) {
   9043             displayContent.mBaseDisplayDensity = density;
   9044         }
   9045 
   9046         // Display scaling mode.
   9047         int mode = Settings.Global.getInt(mContext.getContentResolver(),
   9048                 Settings.Global.DISPLAY_SCALING_FORCE, 0);
   9049         if (mode != 0) {
   9050             Slog.i(TAG_WM, "FORCED DISPLAY SCALING DISABLED");
   9051             displayContent.mDisplayScalingDisabled = true;
   9052         }
   9053     }
   9054 
   9055     // displayContent must not be null
   9056     private void setForcedDisplaySizeLocked(DisplayContent displayContent, int width, int height) {
   9057         Slog.i(TAG_WM, "Using new display size: " + width + "x" + height);
   9058         displayContent.mBaseDisplayWidth = width;
   9059         displayContent.mBaseDisplayHeight = height;
   9060         reconfigureDisplayLocked(displayContent);
   9061     }
   9062 
   9063     @Override
   9064     public void clearForcedDisplaySize(int displayId) {
   9065         if (mContext.checkCallingOrSelfPermission(
   9066                 android.Manifest.permission.WRITE_SECURE_SETTINGS) !=
   9067                 PackageManager.PERMISSION_GRANTED) {
   9068             throw new SecurityException("Must hold permission " +
   9069                     android.Manifest.permission.WRITE_SECURE_SETTINGS);
   9070         }
   9071         if (displayId != Display.DEFAULT_DISPLAY) {
   9072             throw new IllegalArgumentException("Can only set the default display");
   9073         }
   9074         final long ident = Binder.clearCallingIdentity();
   9075         try {
   9076             synchronized(mWindowMap) {
   9077                 final DisplayContent displayContent = getDisplayContentLocked(displayId);
   9078                 if (displayContent != null) {
   9079                     setForcedDisplaySizeLocked(displayContent, displayContent.mInitialDisplayWidth,
   9080                             displayContent.mInitialDisplayHeight);
   9081                     Settings.Global.putString(mContext.getContentResolver(),
   9082                             Settings.Global.DISPLAY_SIZE_FORCED, "");
   9083                 }
   9084             }
   9085         } finally {
   9086             Binder.restoreCallingIdentity(ident);
   9087         }
   9088     }
   9089 
   9090     @Override
   9091     public int getInitialDisplayDensity(int displayId) {
   9092         synchronized (mWindowMap) {
   9093             final DisplayContent displayContent = getDisplayContentLocked(displayId);
   9094             if (displayContent != null && displayContent.hasAccess(Binder.getCallingUid())) {
   9095                 return displayContent.mInitialDisplayDensity;
   9096             }
   9097         }
   9098         return -1;
   9099     }
   9100 
   9101     @Override
   9102     public int getBaseDisplayDensity(int displayId) {
   9103         synchronized (mWindowMap) {
   9104             final DisplayContent displayContent = getDisplayContentLocked(displayId);
   9105             if (displayContent != null && displayContent.hasAccess(Binder.getCallingUid())) {
   9106                 return displayContent.mBaseDisplayDensity;
   9107             }
   9108         }
   9109         return -1;
   9110     }
   9111 
   9112     @Override
   9113     public void setForcedDisplayDensity(int displayId, int density) {
   9114         if (mContext.checkCallingOrSelfPermission(
   9115                 android.Manifest.permission.WRITE_SECURE_SETTINGS) !=
   9116                 PackageManager.PERMISSION_GRANTED) {
   9117             throw new SecurityException("Must hold permission " +
   9118                     android.Manifest.permission.WRITE_SECURE_SETTINGS);
   9119         }
   9120         if (displayId != Display.DEFAULT_DISPLAY) {
   9121             throw new IllegalArgumentException("Can only set the default display");
   9122         }
   9123         final long ident = Binder.clearCallingIdentity();
   9124         try {
   9125             synchronized(mWindowMap) {
   9126                 final DisplayContent displayContent = getDisplayContentLocked(displayId);
   9127                 if (displayContent != null) {
   9128                     setForcedDisplayDensityLocked(displayContent, density);
   9129                     Settings.Secure.putStringForUser(mContext.getContentResolver(),
   9130                             Settings.Secure.DISPLAY_DENSITY_FORCED,
   9131                             Integer.toString(density), mCurrentUserId);
   9132                 }
   9133             }
   9134         } finally {
   9135             Binder.restoreCallingIdentity(ident);
   9136         }
   9137     }
   9138 
   9139     @Override
   9140     public void clearForcedDisplayDensity(int displayId) {
   9141         if (mContext.checkCallingOrSelfPermission(
   9142                 android.Manifest.permission.WRITE_SECURE_SETTINGS) !=
   9143                 PackageManager.PERMISSION_GRANTED) {
   9144             throw new SecurityException("Must hold permission " +
   9145                     android.Manifest.permission.WRITE_SECURE_SETTINGS);
   9146         }
   9147         if (displayId != Display.DEFAULT_DISPLAY) {
   9148             throw new IllegalArgumentException("Can only set the default display");
   9149         }
   9150         final long ident = Binder.clearCallingIdentity();
   9151         try {
   9152             synchronized(mWindowMap) {
   9153                 final DisplayContent displayContent = getDisplayContentLocked(displayId);
   9154                 if (displayContent != null) {
   9155                     setForcedDisplayDensityLocked(displayContent,
   9156                             displayContent.mInitialDisplayDensity);
   9157                     Settings.Secure.putStringForUser(mContext.getContentResolver(),
   9158                             Settings.Secure.DISPLAY_DENSITY_FORCED, "", mCurrentUserId);
   9159                 }
   9160             }
   9161         } finally {
   9162             Binder.restoreCallingIdentity(ident);
   9163         }
   9164     }
   9165 
   9166     /**
   9167      * @param userId the ID of the user
   9168      * @return the forced display density for the specified user, if set, or
   9169      *         {@code 0} if not set
   9170      */
   9171     private int getForcedDisplayDensityForUserLocked(int userId) {
   9172         String densityStr = Settings.Secure.getStringForUser(mContext.getContentResolver(),
   9173                 Settings.Secure.DISPLAY_DENSITY_FORCED, userId);
   9174         if (densityStr == null || densityStr.length() == 0) {
   9175             densityStr = SystemProperties.get(DENSITY_OVERRIDE, null);
   9176         }
   9177         if (densityStr != null && densityStr.length() > 0) {
   9178             try {
   9179                 return Integer.parseInt(densityStr);
   9180             } catch (NumberFormatException ex) {
   9181             }
   9182         }
   9183         return 0;
   9184     }
   9185 
   9186     /**
   9187      * Forces the given display to the use the specified density.
   9188      *
   9189      * @param displayContent the display to modify
   9190      * @param density the density in DPI to use
   9191      */
   9192     private void setForcedDisplayDensityLocked(@NonNull DisplayContent displayContent,
   9193             int density) {
   9194         displayContent.mBaseDisplayDensity = density;
   9195         reconfigureDisplayLocked(displayContent);
   9196     }
   9197 
   9198     // displayContent must not be null
   9199     private void reconfigureDisplayLocked(DisplayContent displayContent) {
   9200         // TODO: Multidisplay: for now only use with default display.
   9201         if (!mDisplayReady) {
   9202             return;
   9203         }
   9204         configureDisplayPolicyLocked(displayContent);
   9205         displayContent.layoutNeeded = true;
   9206 
   9207         boolean configChanged = updateOrientationFromAppTokensLocked(false);
   9208         mTempConfiguration.setToDefaults();
   9209         mTempConfiguration.updateFrom(mCurConfiguration);
   9210         computeScreenConfigurationLocked(mTempConfiguration);
   9211         configChanged |= mCurConfiguration.diff(mTempConfiguration) != 0;
   9212 
   9213         if (configChanged) {
   9214             mWaitingForConfig = true;
   9215             startFreezingDisplayLocked(false, 0, 0);
   9216             mH.sendEmptyMessage(H.SEND_NEW_CONFIGURATION);
   9217             if (!mReconfigureOnConfigurationChanged.contains(displayContent)) {
   9218                 mReconfigureOnConfigurationChanged.add(displayContent);
   9219             }
   9220         }
   9221 
   9222         mWindowPlacerLocked.performSurfacePlacement();
   9223     }
   9224 
   9225     private void configureDisplayPolicyLocked(DisplayContent displayContent) {
   9226         mPolicy.setInitialDisplaySize(displayContent.getDisplay(),
   9227                 displayContent.mBaseDisplayWidth,
   9228                 displayContent.mBaseDisplayHeight,
   9229                 displayContent.mBaseDisplayDensity);
   9230 
   9231         DisplayInfo displayInfo = displayContent.getDisplayInfo();
   9232         mPolicy.setDisplayOverscan(displayContent.getDisplay(),
   9233                 displayInfo.overscanLeft, displayInfo.overscanTop,
   9234                 displayInfo.overscanRight, displayInfo.overscanBottom);
   9235     }
   9236 
   9237     @Override
   9238     public void setOverscan(int displayId, int left, int top, int right, int bottom) {
   9239         if (mContext.checkCallingOrSelfPermission(
   9240                 android.Manifest.permission.WRITE_SECURE_SETTINGS) !=
   9241                 PackageManager.PERMISSION_GRANTED) {
   9242             throw new SecurityException("Must hold permission " +
   9243                     android.Manifest.permission.WRITE_SECURE_SETTINGS);
   9244         }
   9245         final long ident = Binder.clearCallingIdentity();
   9246         try {
   9247             synchronized(mWindowMap) {
   9248                 DisplayContent displayContent = getDisplayContentLocked(displayId);
   9249                 if (displayContent != null) {
   9250                     setOverscanLocked(displayContent, left, top, right, bottom);
   9251                 }
   9252             }
   9253         } finally {
   9254             Binder.restoreCallingIdentity(ident);
   9255         }
   9256     }
   9257 
   9258     private void setOverscanLocked(DisplayContent displayContent,
   9259             int left, int top, int right, int bottom) {
   9260         final DisplayInfo displayInfo = displayContent.getDisplayInfo();
   9261         displayInfo.overscanLeft = left;
   9262         displayInfo.overscanTop = top;
   9263         displayInfo.overscanRight = right;
   9264         displayInfo.overscanBottom = bottom;
   9265 
   9266         mDisplaySettings.setOverscanLocked(displayInfo.uniqueId, displayInfo.name, left, top,
   9267                 right, bottom);
   9268         mDisplaySettings.writeSettingsLocked();
   9269 
   9270         reconfigureDisplayLocked(displayContent);
   9271     }
   9272 
   9273     // -------------------------------------------------------------
   9274     // Internals
   9275     // -------------------------------------------------------------
   9276 
   9277     final WindowState windowForClientLocked(Session session, IWindow client,
   9278             boolean throwOnError) {
   9279         return windowForClientLocked(session, client.asBinder(), throwOnError);
   9280     }
   9281 
   9282     final WindowState windowForClientLocked(Session session, IBinder client,
   9283             boolean throwOnError) {
   9284         WindowState win = mWindowMap.get(client);
   9285         if (localLOGV) Slog.v(
   9286             TAG_WM, "Looking up client " + client + ": " + win);
   9287         if (win == null) {
   9288             RuntimeException ex = new IllegalArgumentException(
   9289                     "Requested window " + client + " does not exist");
   9290             if (throwOnError) {
   9291                 throw ex;
   9292             }
   9293             Slog.w(TAG_WM, "Failed looking up window", ex);
   9294             return null;
   9295         }
   9296         if (session != null && win.mSession != session) {
   9297             RuntimeException ex = new IllegalArgumentException(
   9298                     "Requested window " + client + " is in session " +
   9299                     win.mSession + ", not " + session);
   9300             if (throwOnError) {
   9301                 throw ex;
   9302             }
   9303             Slog.w(TAG_WM, "Failed looking up window", ex);
   9304             return null;
   9305         }
   9306 
   9307         return win;
   9308     }
   9309 
   9310     final void rebuildAppWindowListLocked() {
   9311         rebuildAppWindowListLocked(getDefaultDisplayContentLocked());
   9312     }
   9313 
   9314     private void rebuildAppWindowListLocked(final DisplayContent displayContent) {
   9315         final WindowList windows = displayContent.getWindowList();
   9316         int NW = windows.size();
   9317         int i;
   9318         int lastBelow = -1;
   9319         int numRemoved = 0;
   9320 
   9321         if (mRebuildTmp.length < NW) {
   9322             mRebuildTmp = new WindowState[NW+10];
   9323         }
   9324 
   9325         // First remove all existing app windows.
   9326         i=0;
   9327         while (i < NW) {
   9328             WindowState w = windows.get(i);
   9329             if (w.mAppToken != null) {
   9330                 WindowState win = windows.remove(i);
   9331                 win.mRebuilding = true;
   9332                 mRebuildTmp[numRemoved] = win;
   9333                 mWindowsChanged = true;
   9334                 if (DEBUG_WINDOW_MOVEMENT) Slog.v(TAG_WM, "Rebuild removing window: " + win);
   9335                 NW--;
   9336                 numRemoved++;
   9337                 continue;
   9338             } else if (lastBelow == i-1) {
   9339                 if (w.mAttrs.type == TYPE_WALLPAPER) {
   9340                     lastBelow = i;
   9341                 }
   9342             }
   9343             i++;
   9344         }
   9345 
   9346         // Keep whatever windows were below the app windows still below,
   9347         // by skipping them.
   9348         lastBelow++;
   9349         i = lastBelow;
   9350 
   9351         // First add all of the exiting app tokens...  these are no longer
   9352         // in the main app list, but still have windows shown.  We put them
   9353         // in the back because now that the animation is over we no longer
   9354         // will care about them.
   9355         final ArrayList<TaskStack> stacks = displayContent.getStacks();
   9356         final int numStacks = stacks.size();
   9357         for (int stackNdx = 0; stackNdx < numStacks; ++stackNdx) {
   9358             AppTokenList exitingAppTokens = stacks.get(stackNdx).mExitingAppTokens;
   9359             int NT = exitingAppTokens.size();
   9360             for (int j = 0; j < NT; j++) {
   9361                 i = reAddAppWindowsLocked(displayContent, i, exitingAppTokens.get(j));
   9362             }
   9363         }
   9364 
   9365         // And add in the still active app tokens in Z order.
   9366         for (int stackNdx = 0; stackNdx < numStacks; ++stackNdx) {
   9367             final ArrayList<Task> tasks = stacks.get(stackNdx).getTasks();
   9368             final int numTasks = tasks.size();
   9369             for (int taskNdx = 0; taskNdx < numTasks; ++taskNdx) {
   9370                 final AppTokenList tokens = tasks.get(taskNdx).mAppTokens;
   9371                 final int numTokens = tokens.size();
   9372                 for (int tokenNdx = 0; tokenNdx < numTokens; ++tokenNdx) {
   9373                     final AppWindowToken wtoken = tokens.get(tokenNdx);
   9374                     if (wtoken.mIsExiting && !wtoken.waitingForReplacement()) {
   9375                         continue;
   9376                     }
   9377                     i = reAddAppWindowsLocked(displayContent, i, wtoken);
   9378                 }
   9379             }
   9380         }
   9381 
   9382         i -= lastBelow;
   9383         if (i != numRemoved) {
   9384             displayContent.layoutNeeded = true;
   9385             Slog.w(TAG_WM, "On display=" + displayContent.getDisplayId() + " Rebuild removed "
   9386                     + numRemoved + " windows but added " + i + " rebuildAppWindowListLocked() "
   9387                     + " callers=" + Debug.getCallers(10));
   9388             for (i = 0; i < numRemoved; i++) {
   9389                 WindowState ws = mRebuildTmp[i];
   9390                 if (ws.mRebuilding) {
   9391                     StringWriter sw = new StringWriter();
   9392                     PrintWriter pw = new FastPrintWriter(sw, false, 1024);
   9393                     ws.dump(pw, "", true);
   9394                     pw.flush();
   9395                     Slog.w(TAG_WM, "This window was lost: " + ws);
   9396                     Slog.w(TAG_WM, sw.toString());
   9397                     ws.mWinAnimator.destroySurfaceLocked();
   9398                 }
   9399             }
   9400             Slog.w(TAG_WM, "Current app token list:");
   9401             dumpAppTokensLocked();
   9402             Slog.w(TAG_WM, "Final window list:");
   9403             dumpWindowsLocked();
   9404         }
   9405         Arrays.fill(mRebuildTmp, null);
   9406     }
   9407 
   9408     void makeWindowFreezingScreenIfNeededLocked(WindowState w) {
   9409         // If the screen is currently frozen or off, then keep
   9410         // it frozen/off until this window draws at its new
   9411         // orientation.
   9412         if (!okToDisplay() && mWindowsFreezingScreen != WINDOWS_FREEZING_SCREENS_TIMEOUT) {
   9413             if (DEBUG_ORIENTATION) Slog.v(TAG_WM, "Changing surface while display frozen: " + w);
   9414             w.mOrientationChanging = true;
   9415             w.mLastFreezeDuration = 0;
   9416             mWindowPlacerLocked.mOrientationChangeComplete = false;
   9417             if (mWindowsFreezingScreen == WINDOWS_FREEZING_SCREENS_NONE) {
   9418                 mWindowsFreezingScreen = WINDOWS_FREEZING_SCREENS_ACTIVE;
   9419                 // XXX should probably keep timeout from
   9420                 // when we first froze the display.
   9421                 mH.removeMessages(H.WINDOW_FREEZE_TIMEOUT);
   9422                 mH.sendEmptyMessageDelayed(H.WINDOW_FREEZE_TIMEOUT,
   9423                         WINDOW_FREEZE_TIMEOUT_DURATION);
   9424             }
   9425         }
   9426     }
   9427 
   9428     /**
   9429      * @return bitmap indicating if another pass through layout must be made.
   9430      */
   9431     int handleAnimatingStoppedAndTransitionLocked() {
   9432         int changes = 0;
   9433 
   9434         mAppTransition.setIdle();
   9435 
   9436         for (int i = mNoAnimationNotifyOnTransitionFinished.size() - 1; i >= 0; i--) {
   9437             final IBinder token = mNoAnimationNotifyOnTransitionFinished.get(i);
   9438             mAppTransition.notifyAppTransitionFinishedLocked(token);
   9439         }
   9440         mNoAnimationNotifyOnTransitionFinished.clear();
   9441 
   9442         mWallpaperControllerLocked.hideDeferredWallpapersIfNeeded();
   9443 
   9444         // Restore window app tokens to the ActivityManager views
   9445         ArrayList<TaskStack> stacks = getDefaultDisplayContentLocked().getStacks();
   9446         for (int stackNdx = stacks.size() - 1; stackNdx >= 0; --stackNdx) {
   9447             final ArrayList<Task> tasks = stacks.get(stackNdx).getTasks();
   9448             for (int taskNdx = tasks.size() - 1; taskNdx >= 0; --taskNdx) {
   9449                 final AppTokenList tokens = tasks.get(taskNdx).mAppTokens;
   9450                 for (int tokenNdx = tokens.size() - 1; tokenNdx >= 0; --tokenNdx) {
   9451                     tokens.get(tokenNdx).sendingToBottom = false;
   9452                 }
   9453             }
   9454         }
   9455         rebuildAppWindowListLocked();
   9456 
   9457         changes |= PhoneWindowManager.FINISH_LAYOUT_REDO_LAYOUT;
   9458         if (DEBUG_WALLPAPER_LIGHT) Slog.v(TAG_WM,
   9459                 "Wallpaper layer changed: assigning layers + relayout");
   9460         moveInputMethodWindowsIfNeededLocked(true);
   9461         mWindowPlacerLocked.mWallpaperMayChange = true;
   9462         // Since the window list has been rebuilt, focus might
   9463         // have to be recomputed since the actual order of windows
   9464         // might have changed again.
   9465         mFocusMayChange = true;
   9466 
   9467         return changes;
   9468     }
   9469 
   9470     void updateResizingWindows(final WindowState w) {
   9471         final WindowStateAnimator winAnimator = w.mWinAnimator;
   9472         if (w.mHasSurface && w.mLayoutSeq == mLayoutSeq && !w.isGoneForLayoutLw()) {
   9473             final Task task = w.getTask();
   9474             // In the case of stack bound animations, the window frames
   9475             // will update (unlike other animations which just modifiy
   9476             // various transformation properties). We don't want to
   9477             // notify the client of frame changes in this case. Not only
   9478             // is it a lot of churn, but the frame may not correspond
   9479             // to the surface size or the onscreen area at various
   9480             // phases in the animation, and the client will become
   9481             // sad and confused.
   9482             if (task != null && task.mStack.getBoundsAnimating()) {
   9483                 return;
   9484             }
   9485             w.setReportResizeHints();
   9486             boolean configChanged = w.isConfigChanged();
   9487             if (DEBUG_CONFIGURATION && configChanged) {
   9488                 Slog.v(TAG_WM, "Win " + w + " config changed: "
   9489                         + mCurConfiguration);
   9490             }
   9491             final boolean dragResizingChanged = w.isDragResizeChanged()
   9492                     && !w.isDragResizingChangeReported();
   9493 
   9494             if (localLOGV) Slog.v(TAG_WM, "Resizing " + w
   9495                     + ": configChanged=" + configChanged
   9496                     + " dragResizingChanged=" + dragResizingChanged
   9497                     + " last=" + w.mLastFrame + " frame=" + w.mFrame);
   9498 
   9499             // We update mLastFrame always rather than in the conditional with the
   9500             // last inset variables, because mFrameSizeChanged only tracks the
   9501             // width and height changing.
   9502             w.mLastFrame.set(w.mFrame);
   9503 
   9504             if (w.mContentInsetsChanged
   9505                     || w.mVisibleInsetsChanged
   9506                     || winAnimator.mSurfaceResized
   9507                     || w.mOutsetsChanged
   9508                     || w.mFrameSizeChanged
   9509                     || configChanged
   9510                     || dragResizingChanged
   9511                     || !w.isResizedWhileNotDragResizingReported()) {
   9512                 if (DEBUG_RESIZE || DEBUG_ORIENTATION) {
   9513                     Slog.v(TAG_WM, "Resize reasons for w=" + w + ": "
   9514                             + " contentInsetsChanged=" + w.mContentInsetsChanged
   9515                             + " " + w.mContentInsets.toShortString()
   9516                             + " visibleInsetsChanged=" + w.mVisibleInsetsChanged
   9517                             + " " + w.mVisibleInsets.toShortString()
   9518                             + " stableInsetsChanged=" + w.mStableInsetsChanged
   9519                             + " " + w.mStableInsets.toShortString()
   9520                             + " outsetsChanged=" + w.mOutsetsChanged
   9521                             + " " + w.mOutsets.toShortString()
   9522                             + " surfaceResized=" + winAnimator.mSurfaceResized
   9523                             + " configChanged=" + configChanged
   9524                             + " dragResizingChanged=" + dragResizingChanged
   9525                             + " resizedWhileNotDragResizingReported="
   9526                             + w.isResizedWhileNotDragResizingReported());
   9527                 }
   9528 
   9529                 // If it's a dead window left on screen, and the configuration changed,
   9530                 // there is nothing we can do about it. Remove the window now.
   9531                 if (w.mAppToken != null && w.mAppDied) {
   9532                     w.mAppToken.removeAllDeadWindows();
   9533                     return;
   9534                 }
   9535 
   9536                 w.mLastOverscanInsets.set(w.mOverscanInsets);
   9537                 w.mLastContentInsets.set(w.mContentInsets);
   9538                 w.mLastVisibleInsets.set(w.mVisibleInsets);
   9539                 w.mLastStableInsets.set(w.mStableInsets);
   9540                 w.mLastOutsets.set(w.mOutsets);
   9541                 makeWindowFreezingScreenIfNeededLocked(w);
   9542                 // If the orientation is changing, or we're starting or ending
   9543                 // a drag resizing action, then we need to hold off on unfreezing
   9544                 // the display until this window has been redrawn; to do that,
   9545                 // we need to go through the process of getting informed by the
   9546                 // application when it has finished drawing.
   9547                 if (w.mOrientationChanging || dragResizingChanged
   9548                         || w.isResizedWhileNotDragResizing()) {
   9549                     if (DEBUG_SURFACE_TRACE || DEBUG_ANIM || DEBUG_ORIENTATION || DEBUG_RESIZE) {
   9550                         Slog.v(TAG_WM, "Orientation or resize start waiting for draw"
   9551                                 + ", mDrawState=DRAW_PENDING in " + w
   9552                                 + ", surfaceController " + winAnimator.mSurfaceController);
   9553                     }
   9554                     winAnimator.mDrawState = DRAW_PENDING;
   9555                     if (w.mAppToken != null) {
   9556                         w.mAppToken.clearAllDrawn();
   9557                     }
   9558                 }
   9559                 if (!mResizingWindows.contains(w)) {
   9560                     if (DEBUG_RESIZE || DEBUG_ORIENTATION) Slog.v(TAG_WM,
   9561                             "Resizing window " + w);
   9562                     mResizingWindows.add(w);
   9563                 }
   9564             } else if (w.mOrientationChanging) {
   9565                 if (w.isDrawnLw()) {
   9566                     if (DEBUG_ORIENTATION) Slog.v(TAG_WM,
   9567                             "Orientation not waiting for draw in "
   9568                             + w + ", surfaceController " + winAnimator.mSurfaceController);
   9569                     w.mOrientationChanging = false;
   9570                     w.mLastFreezeDuration = (int)(SystemClock.elapsedRealtime()
   9571                             - mDisplayFreezeTime);
   9572                 }
   9573             }
   9574         }
   9575     }
   9576 
   9577     void checkDrawnWindowsLocked() {
   9578         if (mWaitingForDrawn.isEmpty() || mWaitingForDrawnCallback == null) {
   9579             return;
   9580         }
   9581         for (int j = mWaitingForDrawn.size() - 1; j >= 0; j--) {
   9582             WindowState win = mWaitingForDrawn.get(j);
   9583             if (DEBUG_SCREEN_ON) Slog.i(TAG_WM, "Waiting for drawn " + win +
   9584                     ": removed=" + win.mRemoved + " visible=" + win.isVisibleLw() +
   9585                     " mHasSurface=" + win.mHasSurface +
   9586                     " drawState=" + win.mWinAnimator.mDrawState);
   9587             if (win.mRemoved || !win.mHasSurface || !win.mPolicyVisibility) {
   9588                 // Window has been removed or hidden; no draw will now happen, so stop waiting.
   9589                 if (DEBUG_SCREEN_ON) Slog.w(TAG_WM, "Aborted waiting for drawn: " + win);
   9590                 mWaitingForDrawn.remove(win);
   9591             } else if (win.hasDrawnLw()) {
   9592                 // Window is now drawn (and shown).
   9593                 if (DEBUG_SCREEN_ON) Slog.d(TAG_WM, "Window drawn win=" + win);
   9594                 mWaitingForDrawn.remove(win);
   9595             }
   9596         }
   9597         if (mWaitingForDrawn.isEmpty()) {
   9598             if (DEBUG_SCREEN_ON) Slog.d(TAG_WM, "All windows drawn!");
   9599             mH.removeMessages(H.WAITING_FOR_DRAWN_TIMEOUT);
   9600             mH.sendEmptyMessage(H.ALL_WINDOWS_DRAWN);
   9601         }
   9602     }
   9603 
   9604     void setHoldScreenLocked(final Session newHoldScreen) {
   9605         final boolean hold = newHoldScreen != null;
   9606 
   9607         if (hold && mHoldingScreenOn != newHoldScreen) {
   9608             mHoldingScreenWakeLock.setWorkSource(new WorkSource(newHoldScreen.mUid));
   9609         }
   9610         mHoldingScreenOn = newHoldScreen;
   9611 
   9612         final boolean state = mHoldingScreenWakeLock.isHeld();
   9613         if (hold != state) {
   9614             if (hold) {
   9615                 if (DEBUG_KEEP_SCREEN_ON) {
   9616                     Slog.d(TAG_KEEP_SCREEN_ON, "Acquiring screen wakelock due to " +
   9617                             mWindowPlacerLocked.mHoldScreenWindow);
   9618                 }
   9619                 mLastWakeLockHoldingWindow = mWindowPlacerLocked.mHoldScreenWindow;
   9620                 mLastWakeLockObscuringWindow = null;
   9621                 mHoldingScreenWakeLock.acquire();
   9622                 mPolicy.keepScreenOnStartedLw();
   9623             } else {
   9624                 if (DEBUG_KEEP_SCREEN_ON) {
   9625                     Slog.d(TAG_KEEP_SCREEN_ON, "Releasing screen wakelock, obscured by " +
   9626                             mWindowPlacerLocked.mObsuringWindow);
   9627                 }
   9628                 mLastWakeLockHoldingWindow = null;
   9629                 mLastWakeLockObscuringWindow = mWindowPlacerLocked.mObsuringWindow;
   9630                 mPolicy.keepScreenOnStoppedLw();
   9631                 mHoldingScreenWakeLock.release();
   9632             }
   9633         }
   9634     }
   9635 
   9636     void requestTraversal() {
   9637         synchronized (mWindowMap) {
   9638             mWindowPlacerLocked.requestTraversal();
   9639         }
   9640     }
   9641 
   9642     /** Note that Locked in this case is on mLayoutToAnim */
   9643     void scheduleAnimationLocked() {
   9644         if (!mAnimationScheduled) {
   9645             mAnimationScheduled = true;
   9646             mChoreographer.postFrameCallback(mAnimator.mAnimationFrameCallback);
   9647         }
   9648     }
   9649 
   9650     boolean needsLayout() {
   9651         final int numDisplays = mDisplayContents.size();
   9652         for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   9653             final DisplayContent displayContent = mDisplayContents.valueAt(displayNdx);
   9654             if (displayContent.layoutNeeded) {
   9655                 return true;
   9656             }
   9657         }
   9658         return false;
   9659     }
   9660 
   9661     /** If a window that has an animation specifying a colored background and the current wallpaper
   9662      * is visible, then the color goes *below* the wallpaper so we don't cause the wallpaper to
   9663      * suddenly disappear. */
   9664     int adjustAnimationBackground(WindowStateAnimator winAnimator) {
   9665         WindowList windows = winAnimator.mWin.getWindowList();
   9666         for (int i = windows.size() - 1; i >= 0; --i) {
   9667             WindowState testWin = windows.get(i);
   9668             if (testWin.mIsWallpaper && testWin.isVisibleNow()) {
   9669                 return testWin.mWinAnimator.mAnimLayer;
   9670             }
   9671         }
   9672         return winAnimator.mAnimLayer;
   9673     }
   9674 
   9675     boolean reclaimSomeSurfaceMemoryLocked(WindowStateAnimator winAnimator, String operation,
   9676                                            boolean secure) {
   9677         final WindowSurfaceController surfaceController = winAnimator.mSurfaceController;
   9678         boolean leakedSurface = false;
   9679         boolean killedApps = false;
   9680 
   9681         EventLog.writeEvent(EventLogTags.WM_NO_SURFACE_MEMORY, winAnimator.mWin.toString(),
   9682                 winAnimator.mSession.mPid, operation);
   9683 
   9684         final long callingIdentity = Binder.clearCallingIdentity();
   9685         try {
   9686             // There was some problem...   first, do a sanity check of the window list to make sure
   9687             // we haven't left any dangling surfaces around.
   9688 
   9689             Slog.i(TAG_WM, "Out of memory for surface!  Looking for leaks...");
   9690             final int numDisplays = mDisplayContents.size();
   9691             for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   9692                 final WindowList windows = mDisplayContents.valueAt(displayNdx).getWindowList();
   9693                 final int numWindows = windows.size();
   9694                 for (int winNdx = 0; winNdx < numWindows; ++winNdx) {
   9695                     final WindowState ws = windows.get(winNdx);
   9696                     final WindowStateAnimator wsa = ws.mWinAnimator;
   9697                     if (wsa.mSurfaceController == null) {
   9698                         continue;
   9699                     }
   9700                     if (!mSessions.contains(wsa.mSession)) {
   9701                         Slog.w(TAG_WM, "LEAKED SURFACE (session doesn't exist): "
   9702                                 + ws + " surface=" + wsa.mSurfaceController
   9703                                 + " token=" + ws.mToken
   9704                                 + " pid=" + ws.mSession.mPid
   9705                                 + " uid=" + ws.mSession.mUid);
   9706                         wsa.destroySurface();
   9707                         mForceRemoves.add(ws);
   9708                         leakedSurface = true;
   9709                     } else if (ws.mAppToken != null && ws.mAppToken.clientHidden) {
   9710                         Slog.w(TAG_WM, "LEAKED SURFACE (app token hidden): "
   9711                                 + ws + " surface=" + wsa.mSurfaceController
   9712                                 + " token=" + ws.mAppToken
   9713                                 + " saved=" + ws.hasSavedSurface());
   9714                         if (SHOW_TRANSACTIONS) logSurface(ws, "LEAK DESTROY", false);
   9715                         wsa.destroySurface();
   9716                         leakedSurface = true;
   9717                     }
   9718                 }
   9719             }
   9720 
   9721             if (!leakedSurface) {
   9722                 Slog.w(TAG_WM, "No leaked surfaces; killing applicatons!");
   9723                 SparseIntArray pidCandidates = new SparseIntArray();
   9724                 for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   9725                     final WindowList windows = mDisplayContents.valueAt(displayNdx).getWindowList();
   9726                     final int numWindows = windows.size();
   9727                     for (int winNdx = 0; winNdx < numWindows; ++winNdx) {
   9728                         final WindowState ws = windows.get(winNdx);
   9729                         if (mForceRemoves.contains(ws)) {
   9730                             continue;
   9731                         }
   9732                         WindowStateAnimator wsa = ws.mWinAnimator;
   9733                         if (wsa.mSurfaceController != null) {
   9734                             pidCandidates.append(wsa.mSession.mPid, wsa.mSession.mPid);
   9735                         }
   9736                     }
   9737                     if (pidCandidates.size() > 0) {
   9738                         int[] pids = new int[pidCandidates.size()];
   9739                         for (int i=0; i<pids.length; i++) {
   9740                             pids[i] = pidCandidates.keyAt(i);
   9741                         }
   9742                         try {
   9743                             if (mActivityManager.killPids(pids, "Free memory", secure)) {
   9744                                 killedApps = true;
   9745                             }
   9746                         } catch (RemoteException e) {
   9747                         }
   9748                     }
   9749                 }
   9750             }
   9751 
   9752             if (leakedSurface || killedApps) {
   9753                 // We managed to reclaim some memory, so get rid of the trouble
   9754                 // surface and ask the app to request another one.
   9755                 Slog.w(TAG_WM, "Looks like we have reclaimed some memory, clearing surface for retry.");
   9756                 if (surfaceController != null) {
   9757                     if (SHOW_TRANSACTIONS || SHOW_SURFACE_ALLOC) logSurface(winAnimator.mWin,
   9758                             "RECOVER DESTROY", false);
   9759                     winAnimator.destroySurface();
   9760                     scheduleRemoveStartingWindowLocked(winAnimator.mWin.mAppToken);
   9761                 }
   9762 
   9763                 try {
   9764                     winAnimator.mWin.mClient.dispatchGetNewSurface();
   9765                 } catch (RemoteException e) {
   9766                 }
   9767             }
   9768         } finally {
   9769             Binder.restoreCallingIdentity(callingIdentity);
   9770         }
   9771 
   9772         return leakedSurface || killedApps;
   9773     }
   9774 
   9775     boolean updateFocusedWindowLocked(int mode, boolean updateInputWindows) {
   9776         WindowState newFocus = computeFocusedWindowLocked();
   9777         if (mCurrentFocus != newFocus) {
   9778             Trace.traceBegin(Trace.TRACE_TAG_WINDOW_MANAGER, "wmUpdateFocus");
   9779             // This check makes sure that we don't already have the focus
   9780             // change message pending.
   9781             mH.removeMessages(H.REPORT_FOCUS_CHANGE);
   9782             mH.sendEmptyMessage(H.REPORT_FOCUS_CHANGE);
   9783             // TODO(multidisplay): Focused windows on default display only.
   9784             final DisplayContent displayContent = getDefaultDisplayContentLocked();
   9785             final boolean imWindowChanged = moveInputMethodWindowsIfNeededLocked(
   9786                     mode != UPDATE_FOCUS_WILL_ASSIGN_LAYERS
   9787                             && mode != UPDATE_FOCUS_WILL_PLACE_SURFACES);
   9788             if (imWindowChanged) {
   9789                 displayContent.layoutNeeded = true;
   9790                 newFocus = computeFocusedWindowLocked();
   9791             }
   9792 
   9793             if (DEBUG_FOCUS_LIGHT || localLOGV) Slog.v(TAG_WM, "Changing focus from " +
   9794                     mCurrentFocus + " to " + newFocus + " Callers=" + Debug.getCallers(4));
   9795             final WindowState oldFocus = mCurrentFocus;
   9796             mCurrentFocus = newFocus;
   9797             mLosingFocus.remove(newFocus);
   9798 
   9799             int focusChanged = mPolicy.focusChangedLw(oldFocus, newFocus);
   9800 
   9801             if (imWindowChanged && oldFocus != mInputMethodWindow) {
   9802                 // Focus of the input method window changed. Perform layout if needed.
   9803                 if (mode == UPDATE_FOCUS_PLACING_SURFACES) {
   9804                     mWindowPlacerLocked.performLayoutLockedInner(displayContent, true /*initial*/,
   9805                             updateInputWindows);
   9806                     focusChanged &= ~WindowManagerPolicy.FINISH_LAYOUT_REDO_LAYOUT;
   9807                 } else if (mode == UPDATE_FOCUS_WILL_PLACE_SURFACES) {
   9808                     // Client will do the layout, but we need to assign layers
   9809                     // for handleNewWindowLocked() below.
   9810                     mLayersController.assignLayersLocked(displayContent.getWindowList());
   9811                 }
   9812             }
   9813 
   9814             if ((focusChanged & WindowManagerPolicy.FINISH_LAYOUT_REDO_LAYOUT) != 0) {
   9815                 // The change in focus caused us to need to do a layout.  Okay.
   9816                 displayContent.layoutNeeded = true;
   9817                 if (mode == UPDATE_FOCUS_PLACING_SURFACES) {
   9818                     mWindowPlacerLocked.performLayoutLockedInner(displayContent, true /*initial*/,
   9819                             updateInputWindows);
   9820                 }
   9821             }
   9822 
   9823             if (mode != UPDATE_FOCUS_WILL_ASSIGN_LAYERS) {
   9824                 // If we defer assigning layers, then the caller is responsible for
   9825                 // doing this part.
   9826                 mInputMonitor.setInputFocusLw(mCurrentFocus, updateInputWindows);
   9827             }
   9828 
   9829             adjustForImeIfNeeded(displayContent);
   9830 
   9831             Trace.traceEnd(Trace.TRACE_TAG_WINDOW_MANAGER);
   9832             return true;
   9833         }
   9834         return false;
   9835     }
   9836 
   9837     private WindowState computeFocusedWindowLocked() {
   9838         final int displayCount = mDisplayContents.size();
   9839         for (int i = 0; i < displayCount; i++) {
   9840             final DisplayContent displayContent = mDisplayContents.valueAt(i);
   9841             WindowState win = findFocusedWindowLocked(displayContent);
   9842             if (win != null) {
   9843                 return win;
   9844             }
   9845         }
   9846         return null;
   9847     }
   9848 
   9849     WindowState findFocusedWindowLocked(DisplayContent displayContent) {
   9850         final WindowList windows = displayContent.getWindowList();
   9851         for (int i = windows.size() - 1; i >= 0; i--) {
   9852             final WindowState win = windows.get(i);
   9853 
   9854             if (localLOGV || DEBUG_FOCUS) Slog.v(
   9855                 TAG_WM, "Looking for focus: " + i
   9856                 + " = " + win
   9857                 + ", flags=" + win.mAttrs.flags
   9858                 + ", canReceive=" + win.canReceiveKeys());
   9859 
   9860             if (!win.canReceiveKeys()) {
   9861                 continue;
   9862             }
   9863 
   9864             AppWindowToken wtoken = win.mAppToken;
   9865 
   9866             // If this window's application has been removed, just skip it.
   9867             if (wtoken != null && (wtoken.removed || wtoken.sendingToBottom)) {
   9868                 if (DEBUG_FOCUS) Slog.v(TAG_WM, "Skipping " + wtoken + " because "
   9869                         + (wtoken.removed ? "removed" : "sendingToBottom"));
   9870                 continue;
   9871             }
   9872 
   9873             // Descend through all of the app tokens and find the first that either matches
   9874             // win.mAppToken (return win) or mFocusedApp (return null).
   9875             if (wtoken != null && win.mAttrs.type != TYPE_APPLICATION_STARTING &&
   9876                     mFocusedApp != null) {
   9877                 ArrayList<Task> tasks = displayContent.getTasks();
   9878                 for (int taskNdx = tasks.size() - 1; taskNdx >= 0; --taskNdx) {
   9879                     AppTokenList tokens = tasks.get(taskNdx).mAppTokens;
   9880                     int tokenNdx = tokens.size() - 1;
   9881                     for ( ; tokenNdx >= 0; --tokenNdx) {
   9882                         final AppWindowToken token = tokens.get(tokenNdx);
   9883                         if (wtoken == token) {
   9884                             break;
   9885                         }
   9886                         if (mFocusedApp == token && token.windowsAreFocusable()) {
   9887                             // Whoops, we are below the focused app whose windows are focusable...
   9888                             // No focus for you!!!
   9889                             if (localLOGV || DEBUG_FOCUS_LIGHT) Slog.v(TAG_WM,
   9890                                     "findFocusedWindow: Reached focused app=" + mFocusedApp);
   9891                             return null;
   9892                         }
   9893                     }
   9894                     if (tokenNdx >= 0) {
   9895                         // Early exit from loop, must have found the matching token.
   9896                         break;
   9897                     }
   9898                 }
   9899             }
   9900 
   9901             if (DEBUG_FOCUS_LIGHT) Slog.v(TAG_WM, "findFocusedWindow: Found new focus @ " + i +
   9902                         " = " + win);
   9903             return win;
   9904         }
   9905 
   9906         if (DEBUG_FOCUS_LIGHT) Slog.v(TAG_WM, "findFocusedWindow: No focusable windows.");
   9907         return null;
   9908     }
   9909 
   9910     private void startFreezingDisplayLocked(boolean inTransaction, int exitAnim, int enterAnim) {
   9911         if (mDisplayFrozen) {
   9912             return;
   9913         }
   9914 
   9915         if (!mDisplayReady || !mPolicy.isScreenOn()) {
   9916             // No need to freeze the screen before the system is ready or if
   9917             // the screen is off.
   9918             return;
   9919         }
   9920 
   9921         if (DEBUG_ORIENTATION) Slog.d(TAG_WM,
   9922                 "startFreezingDisplayLocked: inTransaction=" + inTransaction
   9923                 + " exitAnim=" + exitAnim + " enterAnim=" + enterAnim
   9924                 + " called by " + Debug.getCallers(8));
   9925         mScreenFrozenLock.acquire();
   9926 
   9927         mDisplayFrozen = true;
   9928         mDisplayFreezeTime = SystemClock.elapsedRealtime();
   9929         mLastFinishedFreezeSource = null;
   9930 
   9931         mInputMonitor.freezeInputDispatchingLw();
   9932 
   9933         // Clear the last input window -- that is just used for
   9934         // clean transitions between IMEs, and if we are freezing
   9935         // the screen then the whole world is changing behind the scenes.
   9936         mPolicy.setLastInputMethodWindowLw(null, null);
   9937 
   9938         if (mAppTransition.isTransitionSet()) {
   9939             mAppTransition.freeze();
   9940         }
   9941 
   9942         if (PROFILE_ORIENTATION) {
   9943             File file = new File("/data/system/frozen");
   9944             Debug.startMethodTracing(file.toString(), 8 * 1024 * 1024);
   9945         }
   9946 
   9947         if (CUSTOM_SCREEN_ROTATION) {
   9948             mExitAnimId = exitAnim;
   9949             mEnterAnimId = enterAnim;
   9950             final DisplayContent displayContent = getDefaultDisplayContentLocked();
   9951             final int displayId = displayContent.getDisplayId();
   9952             ScreenRotationAnimation screenRotationAnimation =
   9953                     mAnimator.getScreenRotationAnimationLocked(displayId);
   9954             if (screenRotationAnimation != null) {
   9955                 screenRotationAnimation.kill();
   9956             }
   9957 
   9958             // Check whether the current screen contains any secure content.
   9959             boolean isSecure = false;
   9960             final WindowList windows = getDefaultWindowListLocked();
   9961             final int N = windows.size();
   9962             for (int i = 0; i < N; i++) {
   9963                 WindowState ws = windows.get(i);
   9964                 if (ws.isOnScreen() && (ws.mAttrs.flags & FLAG_SECURE) != 0) {
   9965                     isSecure = true;
   9966                     break;
   9967                 }
   9968             }
   9969 
   9970             // TODO(multidisplay): rotation on main screen only.
   9971             displayContent.updateDisplayInfo();
   9972             screenRotationAnimation = new ScreenRotationAnimation(mContext, displayContent,
   9973                     mFxSession, inTransaction, mPolicy.isDefaultOrientationForced(), isSecure);
   9974             mAnimator.setScreenRotationAnimationLocked(displayId, screenRotationAnimation);
   9975         }
   9976     }
   9977 
   9978     void stopFreezingDisplayLocked() {
   9979         if (!mDisplayFrozen) {
   9980             return;
   9981         }
   9982 
   9983         if (mWaitingForConfig || mAppsFreezingScreen > 0
   9984                 || mWindowsFreezingScreen == WINDOWS_FREEZING_SCREENS_ACTIVE
   9985                 || mClientFreezingScreen || !mOpeningApps.isEmpty()) {
   9986             if (DEBUG_ORIENTATION) Slog.d(TAG_WM,
   9987                 "stopFreezingDisplayLocked: Returning mWaitingForConfig=" + mWaitingForConfig
   9988                 + ", mAppsFreezingScreen=" + mAppsFreezingScreen
   9989                 + ", mWindowsFreezingScreen=" + mWindowsFreezingScreen
   9990                 + ", mClientFreezingScreen=" + mClientFreezingScreen
   9991                 + ", mOpeningApps.size()=" + mOpeningApps.size());
   9992             return;
   9993         }
   9994 
   9995         if (DEBUG_ORIENTATION) Slog.d(TAG_WM,
   9996                 "stopFreezingDisplayLocked: Unfreezing now");
   9997 
   9998         mDisplayFrozen = false;
   9999         mLastDisplayFreezeDuration = (int)(SystemClock.elapsedRealtime() - mDisplayFreezeTime);
   10000         StringBuilder sb = new StringBuilder(128);
   10001         sb.append("Screen frozen for ");
   10002         TimeUtils.formatDuration(mLastDisplayFreezeDuration, sb);
   10003         if (mLastFinishedFreezeSource != null) {
   10004             sb.append(" due to ");
   10005             sb.append(mLastFinishedFreezeSource);
   10006         }
   10007         Slog.i(TAG_WM, sb.toString());
   10008         mH.removeMessages(H.APP_FREEZE_TIMEOUT);
   10009         mH.removeMessages(H.CLIENT_FREEZE_TIMEOUT);
   10010         if (PROFILE_ORIENTATION) {
   10011             Debug.stopMethodTracing();
   10012         }
   10013 
   10014         boolean updateRotation = false;
   10015 
   10016         final DisplayContent displayContent = getDefaultDisplayContentLocked();
   10017         final int displayId = displayContent.getDisplayId();
   10018         ScreenRotationAnimation screenRotationAnimation =
   10019                 mAnimator.getScreenRotationAnimationLocked(displayId);
   10020         if (CUSTOM_SCREEN_ROTATION && screenRotationAnimation != null
   10021                 && screenRotationAnimation.hasScreenshot()) {
   10022             if (DEBUG_ORIENTATION) Slog.i(TAG_WM, "**** Dismissing screen rotation animation");
   10023             // TODO(multidisplay): rotation on main screen only.
   10024             DisplayInfo displayInfo = displayContent.getDisplayInfo();
   10025             // Get rotation animation again, with new top window
   10026             boolean isDimming = displayContent.isDimming();
   10027             if (!mPolicy.validateRotationAnimationLw(mExitAnimId, mEnterAnimId, isDimming)) {
   10028                 mExitAnimId = mEnterAnimId = 0;
   10029             }
   10030             if (screenRotationAnimation.dismiss(mFxSession, MAX_ANIMATION_DURATION,
   10031                     getTransitionAnimationScaleLocked(), displayInfo.logicalWidth,
   10032                         displayInfo.logicalHeight, mExitAnimId, mEnterAnimId)) {
   10033                 scheduleAnimationLocked();
   10034             } else {
   10035                 screenRotationAnimation.kill();
   10036                 mAnimator.setScreenRotationAnimationLocked(displayId, null);
   10037                 updateRotation = true;
   10038             }
   10039         } else {
   10040             if (screenRotationAnimation != null) {
   10041                 screenRotationAnimation.kill();
   10042                 mAnimator.setScreenRotationAnimationLocked(displayId, null);
   10043             }
   10044             updateRotation = true;
   10045         }
   10046 
   10047         mInputMonitor.thawInputDispatchingLw();
   10048 
   10049         boolean configChanged;
   10050 
   10051         // While the display is frozen we don't re-compute the orientation
   10052         // to avoid inconsistent states.  However, something interesting
   10053         // could have actually changed during that time so re-evaluate it
   10054         // now to catch that.
   10055         configChanged = updateOrientationFromAppTokensLocked(false);
   10056 
   10057         // A little kludge: a lot could have happened while the
   10058         // display was frozen, so now that we are coming back we
   10059         // do a gc so that any remote references the system
   10060         // processes holds on others can be released if they are
   10061         // no longer needed.
   10062         mH.removeMessages(H.FORCE_GC);
   10063         mH.sendEmptyMessageDelayed(H.FORCE_GC, 2000);
   10064 
   10065         mScreenFrozenLock.release();
   10066 
   10067         if (updateRotation) {
   10068             if (DEBUG_ORIENTATION) Slog.d(TAG_WM, "Performing post-rotate rotation");
   10069             configChanged |= updateRotationUncheckedLocked(false);
   10070         }
   10071 
   10072         if (configChanged) {
   10073             mH.sendEmptyMessage(H.SEND_NEW_CONFIGURATION);
   10074         }
   10075     }
   10076 
   10077     static int getPropertyInt(String[] tokens, int index, int defUnits, int defDps,
   10078             DisplayMetrics dm) {
   10079         if (index < tokens.length) {
   10080             String str = tokens[index];
   10081             if (str != null && str.length() > 0) {
   10082                 try {
   10083                     int val = Integer.parseInt(str);
   10084                     return val;
   10085                 } catch (Exception e) {
   10086                 }
   10087             }
   10088         }
   10089         if (defUnits == TypedValue.COMPLEX_UNIT_PX) {
   10090             return defDps;
   10091         }
   10092         int val = (int)TypedValue.applyDimension(defUnits, defDps, dm);
   10093         return val;
   10094     }
   10095 
   10096     void createWatermarkInTransaction() {
   10097         if (mWatermark != null) {
   10098             return;
   10099         }
   10100 
   10101         File file = new File("/system/etc/setup.conf");
   10102         FileInputStream in = null;
   10103         DataInputStream ind = null;
   10104         try {
   10105             in = new FileInputStream(file);
   10106             ind = new DataInputStream(in);
   10107             String line = ind.readLine();
   10108             if (line != null) {
   10109                 String[] toks = line.split("%");
   10110                 if (toks != null && toks.length > 0) {
   10111                     mWatermark = new Watermark(getDefaultDisplayContentLocked().getDisplay(),
   10112                             mRealDisplayMetrics, mFxSession, toks);
   10113                 }
   10114             }
   10115         } catch (FileNotFoundException e) {
   10116         } catch (IOException e) {
   10117         } finally {
   10118             if (ind != null) {
   10119                 try {
   10120                     ind.close();
   10121                 } catch (IOException e) {
   10122                 }
   10123             } else if (in != null) {
   10124                 try {
   10125                     in.close();
   10126                 } catch (IOException e) {
   10127                 }
   10128             }
   10129         }
   10130     }
   10131 
   10132     @Override
   10133     public void statusBarVisibilityChanged(int visibility) {
   10134         if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.STATUS_BAR)
   10135                 != PackageManager.PERMISSION_GRANTED) {
   10136             throw new SecurityException("Caller does not hold permission "
   10137                     + android.Manifest.permission.STATUS_BAR);
   10138         }
   10139 
   10140         synchronized (mWindowMap) {
   10141             mLastStatusBarVisibility = visibility;
   10142             visibility = mPolicy.adjustSystemUiVisibilityLw(visibility);
   10143             updateStatusBarVisibilityLocked(visibility);
   10144         }
   10145     }
   10146 
   10147     // TOOD(multidisplay): StatusBar on multiple screens?
   10148     boolean updateStatusBarVisibilityLocked(int visibility) {
   10149         if (mLastDispatchedSystemUiVisibility == visibility) {
   10150             return false;
   10151         }
   10152         final int globalDiff = (visibility ^ mLastDispatchedSystemUiVisibility)
   10153                 // We are only interested in differences of one of the
   10154                 // clearable flags...
   10155                 & View.SYSTEM_UI_CLEARABLE_FLAGS
   10156                 // ...if it has actually been cleared.
   10157                 & ~visibility;
   10158 
   10159         mLastDispatchedSystemUiVisibility = visibility;
   10160         mInputManager.setSystemUiVisibility(visibility);
   10161         final WindowList windows = getDefaultWindowListLocked();
   10162         final int N = windows.size();
   10163         for (int i = 0; i < N; i++) {
   10164             WindowState ws = windows.get(i);
   10165             try {
   10166                 int curValue = ws.mSystemUiVisibility;
   10167                 int diff = (curValue ^ visibility) & globalDiff;
   10168                 int newValue = (curValue&~diff) | (visibility&diff);
   10169                 if (newValue != curValue) {
   10170                     ws.mSeq++;
   10171                     ws.mSystemUiVisibility = newValue;
   10172                 }
   10173                 if (newValue != curValue || ws.mAttrs.hasSystemUiListeners) {
   10174                     ws.mClient.dispatchSystemUiVisibilityChanged(ws.mSeq,
   10175                             visibility, newValue, diff);
   10176                 }
   10177             } catch (RemoteException e) {
   10178                 // so sorry
   10179             }
   10180         }
   10181         return true;
   10182     }
   10183 
   10184     @Override
   10185     public void reevaluateStatusBarVisibility() {
   10186         synchronized (mWindowMap) {
   10187             int visibility = mPolicy.adjustSystemUiVisibilityLw(mLastStatusBarVisibility);
   10188             if (updateStatusBarVisibilityLocked(visibility)) {
   10189                 mWindowPlacerLocked.requestTraversal();
   10190             }
   10191         }
   10192     }
   10193 
   10194     private static final class HideNavInputConsumer extends InputConsumerImpl
   10195             implements WindowManagerPolicy.InputConsumer {
   10196         private final InputEventReceiver mInputEventReceiver;
   10197 
   10198         HideNavInputConsumer(WindowManagerService service, Looper looper,
   10199                              InputEventReceiver.Factory inputEventReceiverFactory) {
   10200             super(service, "input consumer", null);
   10201             mInputEventReceiver = inputEventReceiverFactory.createInputEventReceiver(
   10202                     mClientChannel, looper);
   10203         }
   10204 
   10205         @Override
   10206         public void dismiss() {
   10207             if (mService.removeInputConsumer()) {
   10208                 synchronized (mService.mWindowMap) {
   10209                     mInputEventReceiver.dispose();
   10210                     disposeChannelsLw();
   10211                 }
   10212             }
   10213         }
   10214     }
   10215 
   10216     @Override
   10217     public WindowManagerPolicy.InputConsumer addInputConsumer(Looper looper,
   10218             InputEventReceiver.Factory inputEventReceiverFactory) {
   10219         synchronized (mWindowMap) {
   10220             HideNavInputConsumer inputConsumerImpl = new HideNavInputConsumer(
   10221                     this, looper, inputEventReceiverFactory);
   10222             mInputConsumer = inputConsumerImpl;
   10223             mInputMonitor.updateInputWindowsLw(true);
   10224             return inputConsumerImpl;
   10225         }
   10226     }
   10227 
   10228     boolean removeInputConsumer() {
   10229         synchronized (mWindowMap) {
   10230             if (mInputConsumer != null) {
   10231                 mInputConsumer = null;
   10232                 mInputMonitor.updateInputWindowsLw(true);
   10233                 return true;
   10234             }
   10235             return false;
   10236         }
   10237     }
   10238 
   10239     @Override
   10240     public void createWallpaperInputConsumer(InputChannel inputChannel) {
   10241         synchronized (mWindowMap) {
   10242             mWallpaperInputConsumer = new InputConsumerImpl(this, "wallpaper input", inputChannel);
   10243             mWallpaperInputConsumer.mWindowHandle.hasWallpaper = true;
   10244             mInputMonitor.updateInputWindowsLw(true);
   10245         }
   10246     }
   10247 
   10248     @Override
   10249     public void removeWallpaperInputConsumer() {
   10250         synchronized (mWindowMap) {
   10251             if (mWallpaperInputConsumer != null) {
   10252                 mWallpaperInputConsumer.disposeChannelsLw();
   10253                 mWallpaperInputConsumer = null;
   10254                 mInputMonitor.updateInputWindowsLw(true);
   10255             }
   10256         }
   10257     }
   10258 
   10259     @Override
   10260     public boolean hasNavigationBar() {
   10261         return mPolicy.hasNavigationBar();
   10262     }
   10263 
   10264     @Override
   10265     public void lockNow(Bundle options) {
   10266         mPolicy.lockNow(options);
   10267     }
   10268 
   10269     public void showRecentApps(boolean fromHome) {
   10270         mPolicy.showRecentApps(fromHome);
   10271     }
   10272 
   10273     @Override
   10274     public boolean isSafeModeEnabled() {
   10275         return mSafeMode;
   10276     }
   10277 
   10278     @Override
   10279     public boolean clearWindowContentFrameStats(IBinder token) {
   10280         if (!checkCallingPermission(Manifest.permission.FRAME_STATS,
   10281                 "clearWindowContentFrameStats()")) {
   10282             throw new SecurityException("Requires FRAME_STATS permission");
   10283         }
   10284         synchronized (mWindowMap) {
   10285             WindowState windowState = mWindowMap.get(token);
   10286             if (windowState == null) {
   10287                 return false;
   10288             }
   10289             WindowSurfaceController surfaceController = windowState.mWinAnimator.mSurfaceController;
   10290             if (surfaceController == null) {
   10291                 return false;
   10292             }
   10293             return surfaceController.clearWindowContentFrameStats();
   10294         }
   10295     }
   10296 
   10297     @Override
   10298     public WindowContentFrameStats getWindowContentFrameStats(IBinder token) {
   10299         if (!checkCallingPermission(Manifest.permission.FRAME_STATS,
   10300                 "getWindowContentFrameStats()")) {
   10301             throw new SecurityException("Requires FRAME_STATS permission");
   10302         }
   10303         synchronized (mWindowMap) {
   10304             WindowState windowState = mWindowMap.get(token);
   10305             if (windowState == null) {
   10306                 return null;
   10307             }
   10308             WindowSurfaceController surfaceController = windowState.mWinAnimator.mSurfaceController;
   10309             if (surfaceController == null) {
   10310                 return null;
   10311             }
   10312             if (mTempWindowRenderStats == null) {
   10313                 mTempWindowRenderStats = new WindowContentFrameStats();
   10314             }
   10315             WindowContentFrameStats stats = mTempWindowRenderStats;
   10316             if (!surfaceController.getWindowContentFrameStats(stats)) {
   10317                 return null;
   10318             }
   10319             return stats;
   10320         }
   10321     }
   10322 
   10323     public void notifyAppRelaunching(IBinder token) {
   10324         synchronized (mWindowMap) {
   10325             AppWindowToken appWindow = findAppWindowToken(token);
   10326             if (appWindow != null) {
   10327                 appWindow.startRelaunching();
   10328             }
   10329         }
   10330     }
   10331 
   10332     public void notifyAppRelaunchingFinished(IBinder token) {
   10333         synchronized (mWindowMap) {
   10334             AppWindowToken appWindow = findAppWindowToken(token);
   10335             if (appWindow != null) {
   10336                 appWindow.finishRelaunching();
   10337             }
   10338         }
   10339     }
   10340 
   10341     public void notifyAppRelaunchesCleared(IBinder token) {
   10342         synchronized (mWindowMap) {
   10343             final AppWindowToken appWindow = findAppWindowToken(token);
   10344             if (appWindow != null) {
   10345                 appWindow.clearRelaunching();
   10346             }
   10347         }
   10348     }
   10349 
   10350     @Override
   10351     public int getDockedDividerInsetsLw() {
   10352         return getDefaultDisplayContentLocked().getDockedDividerController().getContentInsets();
   10353     }
   10354 
   10355     void dumpPolicyLocked(PrintWriter pw, String[] args, boolean dumpAll) {
   10356         pw.println("WINDOW MANAGER POLICY STATE (dumpsys window policy)");
   10357         mPolicy.dump("    ", pw, args);
   10358     }
   10359 
   10360     void dumpAnimatorLocked(PrintWriter pw, String[] args, boolean dumpAll) {
   10361         pw.println("WINDOW MANAGER ANIMATOR STATE (dumpsys window animator)");
   10362         mAnimator.dumpLocked(pw, "    ", dumpAll);
   10363     }
   10364 
   10365     void dumpTokensLocked(PrintWriter pw, boolean dumpAll) {
   10366         pw.println("WINDOW MANAGER TOKENS (dumpsys window tokens)");
   10367         if (!mTokenMap.isEmpty()) {
   10368             pw.println("  All tokens:");
   10369             Iterator<WindowToken> it = mTokenMap.values().iterator();
   10370             while (it.hasNext()) {
   10371                 WindowToken token = it.next();
   10372                 pw.print("  "); pw.print(token);
   10373                 if (dumpAll) {
   10374                     pw.println(':');
   10375                     token.dump(pw, "    ");
   10376                 } else {
   10377                     pw.println();
   10378                 }
   10379             }
   10380         }
   10381         mWallpaperControllerLocked.dumpTokens(pw, "  ", dumpAll);
   10382         if (!mFinishedStarting.isEmpty()) {
   10383             pw.println();
   10384             pw.println("  Finishing start of application tokens:");
   10385             for (int i=mFinishedStarting.size()-1; i>=0; i--) {
   10386                 WindowToken token = mFinishedStarting.get(i);
   10387                 pw.print("  Finished Starting #"); pw.print(i);
   10388                         pw.print(' '); pw.print(token);
   10389                 if (dumpAll) {
   10390                     pw.println(':');
   10391                     token.dump(pw, "    ");
   10392                 } else {
   10393                     pw.println();
   10394                 }
   10395             }
   10396         }
   10397         if (!mOpeningApps.isEmpty() || !mClosingApps.isEmpty()) {
   10398             pw.println();
   10399             if (mOpeningApps.size() > 0) {
   10400                 pw.print("  mOpeningApps="); pw.println(mOpeningApps);
   10401             }
   10402             if (mClosingApps.size() > 0) {
   10403                 pw.print("  mClosingApps="); pw.println(mClosingApps);
   10404             }
   10405         }
   10406     }
   10407 
   10408     void dumpSessionsLocked(PrintWriter pw, boolean dumpAll) {
   10409         pw.println("WINDOW MANAGER SESSIONS (dumpsys window sessions)");
   10410         for (int i=0; i<mSessions.size(); i++) {
   10411             Session s = mSessions.valueAt(i);
   10412             pw.print("  Session "); pw.print(s); pw.println(':');
   10413             s.dump(pw, "    ");
   10414         }
   10415     }
   10416 
   10417     void dumpDisplayContentsLocked(PrintWriter pw, boolean dumpAll) {
   10418         pw.println("WINDOW MANAGER DISPLAY CONTENTS (dumpsys window displays)");
   10419         if (mDisplayReady) {
   10420             final int numDisplays = mDisplayContents.size();
   10421             for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   10422                 final DisplayContent displayContent = mDisplayContents.valueAt(displayNdx);
   10423                 displayContent.dump("  ", pw);
   10424             }
   10425         } else {
   10426             pw.println("  NO DISPLAY");
   10427         }
   10428     }
   10429 
   10430     void dumpWindowsLocked(PrintWriter pw, boolean dumpAll,
   10431             ArrayList<WindowState> windows) {
   10432         pw.println("WINDOW MANAGER WINDOWS (dumpsys window windows)");
   10433         dumpWindowsNoHeaderLocked(pw, dumpAll, windows);
   10434     }
   10435 
   10436     void dumpWindowsNoHeaderLocked(PrintWriter pw, boolean dumpAll,
   10437             ArrayList<WindowState> windows) {
   10438         final int numDisplays = mDisplayContents.size();
   10439         for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   10440             final WindowList windowList = mDisplayContents.valueAt(displayNdx).getWindowList();
   10441             for (int winNdx = windowList.size() - 1; winNdx >= 0; --winNdx) {
   10442                 final WindowState w = windowList.get(winNdx);
   10443                 if (windows == null || windows.contains(w)) {
   10444                     pw.print("  Window #"); pw.print(winNdx); pw.print(' ');
   10445                             pw.print(w); pw.println(":");
   10446                     w.dump(pw, "    ", dumpAll || windows != null);
   10447                 }
   10448             }
   10449         }
   10450         if (mInputMethodDialogs.size() > 0) {
   10451             pw.println();
   10452             pw.println("  Input method dialogs:");
   10453             for (int i=mInputMethodDialogs.size()-1; i>=0; i--) {
   10454                 WindowState w = mInputMethodDialogs.get(i);
   10455                 if (windows == null || windows.contains(w)) {
   10456                     pw.print("  IM Dialog #"); pw.print(i); pw.print(": "); pw.println(w);
   10457                 }
   10458             }
   10459         }
   10460         if (mPendingRemove.size() > 0) {
   10461             pw.println();
   10462             pw.println("  Remove pending for:");
   10463             for (int i=mPendingRemove.size()-1; i>=0; i--) {
   10464                 WindowState w = mPendingRemove.get(i);
   10465                 if (windows == null || windows.contains(w)) {
   10466                     pw.print("  Remove #"); pw.print(i); pw.print(' ');
   10467                             pw.print(w);
   10468                     if (dumpAll) {
   10469                         pw.println(":");
   10470                         w.dump(pw, "    ", true);
   10471                     } else {
   10472                         pw.println();
   10473                     }
   10474                 }
   10475             }
   10476         }
   10477         if (mForceRemoves != null && mForceRemoves.size() > 0) {
   10478             pw.println();
   10479             pw.println("  Windows force removing:");
   10480             for (int i=mForceRemoves.size()-1; i>=0; i--) {
   10481                 WindowState w = mForceRemoves.get(i);
   10482                 pw.print("  Removing #"); pw.print(i); pw.print(' ');
   10483                         pw.print(w);
   10484                 if (dumpAll) {
   10485                     pw.println(":");
   10486                     w.dump(pw, "    ", true);
   10487                 } else {
   10488                     pw.println();
   10489                 }
   10490             }
   10491         }
   10492         if (mDestroySurface.size() > 0) {
   10493             pw.println();
   10494             pw.println("  Windows waiting to destroy their surface:");
   10495             for (int i=mDestroySurface.size()-1; i>=0; i--) {
   10496                 WindowState w = mDestroySurface.get(i);
   10497                 if (windows == null || windows.contains(w)) {
   10498                     pw.print("  Destroy #"); pw.print(i); pw.print(' ');
   10499                             pw.print(w);
   10500                     if (dumpAll) {
   10501                         pw.println(":");
   10502                         w.dump(pw, "    ", true);
   10503                     } else {
   10504                         pw.println();
   10505                     }
   10506                 }
   10507             }
   10508         }
   10509         if (mLosingFocus.size() > 0) {
   10510             pw.println();
   10511             pw.println("  Windows losing focus:");
   10512             for (int i=mLosingFocus.size()-1; i>=0; i--) {
   10513                 WindowState w = mLosingFocus.get(i);
   10514                 if (windows == null || windows.contains(w)) {
   10515                     pw.print("  Losing #"); pw.print(i); pw.print(' ');
   10516                             pw.print(w);
   10517                     if (dumpAll) {
   10518                         pw.println(":");
   10519                         w.dump(pw, "    ", true);
   10520                     } else {
   10521                         pw.println();
   10522                     }
   10523                 }
   10524             }
   10525         }
   10526         if (mResizingWindows.size() > 0) {
   10527             pw.println();
   10528             pw.println("  Windows waiting to resize:");
   10529             for (int i=mResizingWindows.size()-1; i>=0; i--) {
   10530                 WindowState w = mResizingWindows.get(i);
   10531                 if (windows == null || windows.contains(w)) {
   10532                     pw.print("  Resizing #"); pw.print(i); pw.print(' ');
   10533                             pw.print(w);
   10534                     if (dumpAll) {
   10535                         pw.println(":");
   10536                         w.dump(pw, "    ", true);
   10537                     } else {
   10538                         pw.println();
   10539                     }
   10540                 }
   10541             }
   10542         }
   10543         if (mWaitingForDrawn.size() > 0) {
   10544             pw.println();
   10545             pw.println("  Clients waiting for these windows to be drawn:");
   10546             for (int i=mWaitingForDrawn.size()-1; i>=0; i--) {
   10547                 WindowState win = mWaitingForDrawn.get(i);
   10548                 pw.print("  Waiting #"); pw.print(i); pw.print(' '); pw.print(win);
   10549             }
   10550         }
   10551         pw.println();
   10552         pw.print("  mCurConfiguration="); pw.println(this.mCurConfiguration);
   10553         pw.print("  mHasPermanentDpad="); pw.println(mHasPermanentDpad);
   10554         pw.print("  mCurrentFocus="); pw.println(mCurrentFocus);
   10555         if (mLastFocus != mCurrentFocus) {
   10556             pw.print("  mLastFocus="); pw.println(mLastFocus);
   10557         }
   10558         pw.print("  mFocusedApp="); pw.println(mFocusedApp);
   10559         if (mInputMethodTarget != null) {
   10560             pw.print("  mInputMethodTarget="); pw.println(mInputMethodTarget);
   10561         }
   10562         pw.print("  mInTouchMode="); pw.print(mInTouchMode);
   10563                 pw.print(" mLayoutSeq="); pw.println(mLayoutSeq);
   10564         pw.print("  mLastDisplayFreezeDuration=");
   10565                 TimeUtils.formatDuration(mLastDisplayFreezeDuration, pw);
   10566                 if ( mLastFinishedFreezeSource != null) {
   10567                     pw.print(" due to ");
   10568                     pw.print(mLastFinishedFreezeSource);
   10569                 }
   10570                 pw.println();
   10571         pw.print("  mLastWakeLockHoldingWindow=");pw.print(mLastWakeLockHoldingWindow);
   10572                 pw.print(" mLastWakeLockObscuringWindow="); pw.print(mLastWakeLockObscuringWindow);
   10573                 pw.println();
   10574 
   10575         mInputMonitor.dump(pw, "  ");
   10576 
   10577         if (dumpAll) {
   10578             pw.print("  mSystemDecorLayer="); pw.print(mSystemDecorLayer);
   10579                     pw.print(" mScreenRect="); pw.println(mScreenRect.toShortString());
   10580             if (mLastStatusBarVisibility != 0) {
   10581                 pw.print("  mLastStatusBarVisibility=0x");
   10582                         pw.println(Integer.toHexString(mLastStatusBarVisibility));
   10583             }
   10584             if (mInputMethodWindow != null) {
   10585                 pw.print("  mInputMethodWindow="); pw.println(mInputMethodWindow);
   10586             }
   10587             mWindowPlacerLocked.dump(pw, "  ");
   10588             mWallpaperControllerLocked.dump(pw, "  ");
   10589             mLayersController.dump(pw, "  ");
   10590             pw.print("  mSystemBooted="); pw.print(mSystemBooted);
   10591                     pw.print(" mDisplayEnabled="); pw.println(mDisplayEnabled);
   10592             if (needsLayout()) {
   10593                 pw.print("  layoutNeeded on displays=");
   10594                 for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   10595                     final DisplayContent displayContent = mDisplayContents.valueAt(displayNdx);
   10596                     if (displayContent.layoutNeeded) {
   10597                         pw.print(displayContent.getDisplayId());
   10598                     }
   10599                 }
   10600                 pw.println();
   10601             }
   10602             pw.print("  mTransactionSequence="); pw.println(mTransactionSequence);
   10603             pw.print("  mDisplayFrozen="); pw.print(mDisplayFrozen);
   10604                     pw.print(" windows="); pw.print(mWindowsFreezingScreen);
   10605                     pw.print(" client="); pw.print(mClientFreezingScreen);
   10606                     pw.print(" apps="); pw.print(mAppsFreezingScreen);
   10607                     pw.print(" waitingForConfig="); pw.println(mWaitingForConfig);
   10608             pw.print("  mRotation="); pw.print(mRotation);
   10609                     pw.print(" mAltOrientation="); pw.println(mAltOrientation);
   10610             pw.print("  mLastWindowForcedOrientation="); pw.print(mLastWindowForcedOrientation);
   10611                     pw.print(" mLastOrientation="); pw.println(mLastOrientation);
   10612             pw.print("  mDeferredRotationPauseCount="); pw.println(mDeferredRotationPauseCount);
   10613             pw.print("  Animation settings: disabled="); pw.print(mAnimationsDisabled);
   10614                     pw.print(" window="); pw.print(mWindowAnimationScaleSetting);
   10615                     pw.print(" transition="); pw.print(mTransitionAnimationScaleSetting);
   10616                     pw.print(" animator="); pw.println(mAnimatorDurationScaleSetting);
   10617             pw.print(" mSkipAppTransitionAnimation=");pw.println(mSkipAppTransitionAnimation);
   10618             pw.println("  mLayoutToAnim:");
   10619             mAppTransition.dump(pw, "    ");
   10620         }
   10621     }
   10622 
   10623     boolean dumpWindows(PrintWriter pw, String name, String[] args,
   10624             int opti, boolean dumpAll) {
   10625         WindowList windows = new WindowList();
   10626         if ("apps".equals(name) || "visible".equals(name) || "visible-apps".equals(name)) {
   10627             final boolean appsOnly = name.contains("apps");
   10628             final boolean visibleOnly = name.contains("visible");
   10629             synchronized(mWindowMap) {
   10630                 if (appsOnly) {
   10631                     dumpDisplayContentsLocked(pw, true);
   10632                 }
   10633 
   10634                 final int numDisplays = mDisplayContents.size();
   10635                 for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   10636                     final WindowList windowList =
   10637                             mDisplayContents.valueAt(displayNdx).getWindowList();
   10638                     for (int winNdx = windowList.size() - 1; winNdx >= 0; --winNdx) {
   10639                         final WindowState w = windowList.get(winNdx);
   10640                         if ((!visibleOnly || w.mWinAnimator.getShown())
   10641                                 && (!appsOnly || w.mAppToken != null)) {
   10642                             windows.add(w);
   10643                         }
   10644                     }
   10645                 }
   10646             }
   10647         } else {
   10648             int objectId = 0;
   10649             // See if this is an object ID.
   10650             try {
   10651                 objectId = Integer.parseInt(name, 16);
   10652                 name = null;
   10653             } catch (RuntimeException e) {
   10654             }
   10655             synchronized(mWindowMap) {
   10656                 final int numDisplays = mDisplayContents.size();
   10657                 for (int displayNdx = 0; displayNdx < numDisplays; ++displayNdx) {
   10658                     final WindowList windowList =
   10659                             mDisplayContents.valueAt(displayNdx).getWindowList();
   10660                     for (int winNdx = windowList.size() - 1; winNdx >= 0; --winNdx) {
   10661                         final WindowState w = windowList.get(winNdx);
   10662                         if (name != null) {
   10663                             if (w.mAttrs.getTitle().toString().contains(name)) {
   10664                                 windows.add(w);
   10665                             }
   10666                         } else if (System.identityHashCode(w) == objectId) {
   10667                             windows.add(w);
   10668                         }
   10669                     }
   10670                 }
   10671             }
   10672         }
   10673 
   10674         if (windows.size() <= 0) {
   10675             return false;
   10676         }
   10677 
   10678         synchronized(mWindowMap) {
   10679             dumpWindowsLocked(pw, dumpAll, windows);
   10680         }
   10681         return true;
   10682     }
   10683 
   10684     void dumpLastANRLocked(PrintWriter pw) {
   10685         pw.println("WINDOW MANAGER LAST ANR (dumpsys window lastanr)");
   10686         if (mLastANRState == null) {
   10687             pw.println("  <no ANR has occurred since boot>");
   10688         } else {
   10689             pw.println(mLastANRState);
   10690         }
   10691     }
   10692 
   10693     /**
   10694      * Saves information about the state of the window manager at
   10695      * the time an ANR occurred before anything else in the system changes
   10696      * in response.
   10697      *
   10698      * @param appWindowToken The application that ANR'd, may be null.
   10699      * @param windowState The window that ANR'd, may be null.
   10700      * @param reason The reason for the ANR, may be null.
   10701      */
   10702     public void saveANRStateLocked(AppWindowToken appWindowToken, WindowState windowState,
   10703             String reason) {
   10704         StringWriter sw = new StringWriter();
   10705         PrintWriter pw = new FastPrintWriter(sw, false, 1024);
   10706         pw.println("  ANR time: " + DateFormat.getInstance().format(new Date()));
   10707         if (appWindowToken != null) {
   10708             pw.println("  Application at fault: " + appWindowToken.stringName);
   10709         }
   10710         if (windowState != null) {
   10711             pw.println("  Window at fault: " + windowState.mAttrs.getTitle());
   10712         }
   10713         if (reason != null) {
   10714             pw.println("  Reason: " + reason);
   10715         }
   10716         pw.println();
   10717         dumpWindowsNoHeaderLocked(pw, true, null);
   10718         pw.println();
   10719         pw.println("Last ANR continued");
   10720         dumpDisplayContentsLocked(pw, true);
   10721         pw.close();
   10722         mLastANRState = sw.toString();
   10723 
   10724         mH.removeMessages(H.RESET_ANR_MESSAGE);
   10725         mH.sendEmptyMessageDelayed(H.RESET_ANR_MESSAGE, LAST_ANR_LIFETIME_DURATION_MSECS);
   10726     }
   10727 
   10728     @Override
   10729     public void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
   10730         if (mContext.checkCallingOrSelfPermission("android.permission.DUMP")
   10731                 != PackageManager.PERMISSION_GRANTED) {
   10732             pw.println("Permission Denial: can't dump WindowManager from from pid="
   10733                     + Binder.getCallingPid()
   10734                     + ", uid=" + Binder.getCallingUid());
   10735             return;
   10736         }
   10737 
   10738         boolean dumpAll = false;
   10739 
   10740         int opti = 0;
   10741         while (opti < args.length) {
   10742             String opt = args[opti];
   10743             if (opt == null || opt.length() <= 0 || opt.charAt(0) != '-') {
   10744                 break;
   10745             }
   10746             opti++;
   10747             if ("-a".equals(opt)) {
   10748                 dumpAll = true;
   10749             } else if ("-h".equals(opt)) {
   10750                 pw.println("Window manager dump options:");
   10751                 pw.println("  [-a] [-h] [cmd] ...");
   10752                 pw.println("  cmd may be one of:");
   10753                 pw.println("    l[astanr]: last ANR information");
   10754                 pw.println("    p[policy]: policy state");
   10755                 pw.println("    a[animator]: animator state");
   10756                 pw.println("    s[essions]: active sessions");
   10757                 pw.println("    surfaces: active surfaces (debugging enabled only)");
   10758                 pw.println("    d[isplays]: active display contents");
   10759                 pw.println("    t[okens]: token list");
   10760                 pw.println("    w[indows]: window list");
   10761                 pw.println("  cmd may also be a NAME to dump windows.  NAME may");
   10762                 pw.println("    be a partial substring in a window name, a");
   10763                 pw.println("    Window hex object identifier, or");
   10764                 pw.println("    \"all\" for all windows, or");
   10765                 pw.println("    \"visible\" for the visible windows.");
   10766                 pw.println("    \"visible-apps\" for the visible app windows.");
   10767                 pw.println("  -a: include all available server state.");
   10768                 return;
   10769             } else {
   10770                 pw.println("Unknown argument: " + opt + "; use -h for help");
   10771             }
   10772         }
   10773 
   10774         // Is the caller requesting to dump a particular piece of data?
   10775         if (opti < args.length) {
   10776             String cmd = args[opti];
   10777             opti++;
   10778             if ("lastanr".equals(cmd) || "l".equals(cmd)) {
   10779                 synchronized(mWindowMap) {
   10780                     dumpLastANRLocked(pw);
   10781                 }
   10782                 return;
   10783             } else if ("policy".equals(cmd) || "p".equals(cmd)) {
   10784                 synchronized(mWindowMap) {
   10785                     dumpPolicyLocked(pw, args, true);
   10786                 }
   10787                 return;
   10788             } else if ("animator".equals(cmd) || "a".equals(cmd)) {
   10789                 synchronized(mWindowMap) {
   10790                     dumpAnimatorLocked(pw, args, true);
   10791                 }
   10792                 return;
   10793             } else if ("sessions".equals(cmd) || "s".equals(cmd)) {
   10794                 synchronized(mWindowMap) {
   10795                     dumpSessionsLocked(pw, true);
   10796                 }
   10797                 return;
   10798             } else if ("surfaces".equals(cmd)) {
   10799                 synchronized(mWindowMap) {
   10800                     WindowSurfaceController.SurfaceTrace.dumpAllSurfaces(pw, null);
   10801                 }
   10802                 return;
   10803             } else if ("displays".equals(cmd) || "d".equals(cmd)) {
   10804                 synchronized(mWindowMap) {
   10805                     dumpDisplayContentsLocked(pw, true);
   10806                 }
   10807                 return;
   10808             } else if ("tokens".equals(cmd) || "t".equals(cmd)) {
   10809                 synchronized(mWindowMap) {
   10810                     dumpTokensLocked(pw, true);
   10811                 }
   10812                 return;
   10813             } else if ("windows".equals(cmd) || "w".equals(cmd)) {
   10814                 synchronized(mWindowMap) {
   10815                     dumpWindowsLocked(pw, true, null);
   10816                 }
   10817                 return;
   10818             } else if ("all".equals(cmd) || "a".equals(cmd)) {
   10819                 synchronized(mWindowMap) {
   10820                     dumpWindowsLocked(pw, true, null);
   10821                 }
   10822                 return;
   10823             } else {
   10824                 // Dumping a single name?
   10825                 if (!dumpWindows(pw, cmd, args, opti, dumpAll)) {
   10826                     pw.println("Bad window command, or no windows match: " + cmd);
   10827                     pw.println("Use -h for help.");
   10828                 }
   10829                 return;
   10830             }
   10831         }
   10832 
   10833         synchronized(mWindowMap) {
   10834             pw.println();
   10835             if (dumpAll) {
   10836                 pw.println("-------------------------------------------------------------------------------");
   10837             }
   10838             dumpLastANRLocked(pw);
   10839             pw.println();
   10840             if (dumpAll) {
   10841                 pw.println("-------------------------------------------------------------------------------");
   10842             }
   10843             dumpPolicyLocked(pw, args, dumpAll);
   10844             pw.println();
   10845             if (dumpAll) {
   10846                 pw.println("-------------------------------------------------------------------------------");
   10847             }
   10848             dumpAnimatorLocked(pw, args, dumpAll);
   10849             pw.println();
   10850             if (dumpAll) {
   10851                 pw.println("-------------------------------------------------------------------------------");
   10852             }
   10853             dumpSessionsLocked(pw, dumpAll);
   10854             pw.println();
   10855             if (dumpAll) {
   10856                 pw.println("-------------------------------------------------------------------------------");
   10857             }
   10858             WindowSurfaceController.SurfaceTrace.dumpAllSurfaces(pw, dumpAll ?
   10859                     "-------------------------------------------------------------------------------"
   10860                     : null);
   10861             pw.println();
   10862             if (dumpAll) {
   10863                 pw.println("-------------------------------------------------------------------------------");
   10864             }
   10865             dumpDisplayContentsLocked(pw, dumpAll);
   10866             pw.println();
   10867             if (dumpAll) {
   10868                 pw.println("-------------------------------------------------------------------------------");
   10869             }
   10870             dumpTokensLocked(pw, dumpAll);
   10871             pw.println();
   10872             if (dumpAll) {
   10873                 pw.println("-------------------------------------------------------------------------------");
   10874             }
   10875             dumpWindowsLocked(pw, dumpAll, null);
   10876         }
   10877     }
   10878 
   10879     // Called by the heartbeat to ensure locks are not held indefnitely (for deadlock detection).
   10880     @Override
   10881     public void monitor() {
   10882         synchronized (mWindowMap) { }
   10883     }
   10884 
   10885     private DisplayContent newDisplayContentLocked(final Display display) {
   10886         DisplayContent displayContent = new DisplayContent(display, this);
   10887         final int displayId = display.getDisplayId();
   10888         if (DEBUG_DISPLAY) Slog.v(TAG_WM, "Adding display=" + display);
   10889         mDisplayContents.put(displayId, displayContent);
   10890 
   10891         DisplayInfo displayInfo = displayContent.getDisplayInfo();
   10892         final Rect rect = new Rect();
   10893         mDisplaySettings.getOverscanLocked(displayInfo.name, displayInfo.uniqueId, rect);
   10894         displayInfo.overscanLeft = rect.left;
   10895         displayInfo.overscanTop = rect.top;
   10896         displayInfo.overscanRight = rect.right;
   10897         displayInfo.overscanBottom = rect.bottom;
   10898         mDisplayManagerInternal.setDisplayInfoOverrideFromWindowManager(displayId, displayInfo);
   10899         configureDisplayPolicyLocked(displayContent);
   10900 
   10901         // TODO: Create an input channel for each display with touch capability.
   10902         if (displayId == Display.DEFAULT_DISPLAY) {
   10903             displayContent.mTapDetector = new TaskTapPointerEventListener(this, displayContent);
   10904             registerPointerEventListener(displayContent.mTapDetector);
   10905             registerPointerEventListener(mMousePositionTracker);
   10906         }
   10907 
   10908         return displayContent;
   10909     }
   10910 
   10911     public void createDisplayContentLocked(final Display display) {
   10912         if (display == null) {
   10913             throw new IllegalArgumentException("getDisplayContent: display must not be null");
   10914         }
   10915         getDisplayContentLocked(display.getDisplayId());
   10916     }
   10917 
   10918     /**
   10919      * Retrieve the DisplayContent for the specified displayId. Will create a new DisplayContent if
   10920      * there is a Display for the displayId.
   10921      * @param displayId The display the caller is interested in.
   10922      * @return The DisplayContent associated with displayId or null if there is no Display for it.
   10923      */
   10924     public DisplayContent getDisplayContentLocked(final int displayId) {
   10925         DisplayContent displayContent = mDisplayContents.get(displayId);
   10926         if (displayContent == null) {
   10927             final Display display = mDisplayManager.getDisplay(displayId);
   10928             if (display != null) {
   10929                 displayContent = newDisplayContentLocked(display);
   10930             }
   10931         }
   10932         return displayContent;
   10933     }
   10934 
   10935     // There is an inherent assumption that this will never return null.
   10936     public DisplayContent getDefaultDisplayContentLocked() {
   10937         return getDisplayContentLocked(Display.DEFAULT_DISPLAY);
   10938     }
   10939 
   10940     public WindowList getDefaultWindowListLocked() {
   10941         return getDefaultDisplayContentLocked().getWindowList();
   10942     }
   10943 
   10944     public DisplayInfo getDefaultDisplayInfoLocked() {
   10945         return getDefaultDisplayContentLocked().getDisplayInfo();
   10946     }
   10947 
   10948     /**
   10949      * Return the list of WindowStates associated on the passed display.
   10950      * @param display The screen to return windows from.
   10951      * @return The list of WindowStates on the screen, or null if the there is no screen.
   10952      */
   10953     public WindowList getWindowListLocked(final Display display) {
   10954         return getWindowListLocked(display.getDisplayId());
   10955     }
   10956 
   10957     /**
   10958      * Return the list of WindowStates associated on the passed display.
   10959      * @param displayId The screen to return windows from.
   10960      * @return The list of WindowStates on the screen, or null if the there is no screen.
   10961      */
   10962     public WindowList getWindowListLocked(final int displayId) {
   10963         final DisplayContent displayContent = getDisplayContentLocked(displayId);
   10964         return displayContent != null ? displayContent.getWindowList() : null;
   10965     }
   10966 
   10967     public void onDisplayAdded(int displayId) {
   10968         mH.sendMessage(mH.obtainMessage(H.DO_DISPLAY_ADDED, displayId, 0));
   10969     }
   10970 
   10971     public void handleDisplayAdded(int displayId) {
   10972         synchronized (mWindowMap) {
   10973             final Display display = mDisplayManager.getDisplay(displayId);
   10974             if (display != null) {
   10975                 createDisplayContentLocked(display);
   10976                 displayReady(displayId);
   10977             }
   10978             mWindowPlacerLocked.requestTraversal();
   10979         }
   10980     }
   10981 
   10982     public void onDisplayRemoved(int displayId) {
   10983         mH.sendMessage(mH.obtainMessage(H.DO_DISPLAY_REMOVED, displayId, 0));
   10984     }
   10985 
   10986     private void handleDisplayRemovedLocked(int displayId) {
   10987         final DisplayContent displayContent = getDisplayContentLocked(displayId);
   10988         if (displayContent != null) {
   10989             if (displayContent.isAnimating()) {
   10990                 displayContent.mDeferredRemoval = true;
   10991                 return;
   10992             }
   10993             if (DEBUG_DISPLAY) Slog.v(TAG_WM, "Removing display=" + displayContent);
   10994             mDisplayContents.delete(displayId);
   10995             displayContent.close();
   10996             if (displayId == Display.DEFAULT_DISPLAY) {
   10997                 unregisterPointerEventListener(displayContent.mTapDetector);
   10998                 unregisterPointerEventListener(mMousePositionTracker);
   10999             }
   11000         }
   11001         mAnimator.removeDisplayLocked(displayId);
   11002         mWindowPlacerLocked.requestTraversal();
   11003     }
   11004 
   11005     public void onDisplayChanged(int displayId) {
   11006         mH.sendMessage(mH.obtainMessage(H.DO_DISPLAY_CHANGED, displayId, 0));
   11007     }
   11008 
   11009     private void handleDisplayChangedLocked(int displayId) {
   11010         final DisplayContent displayContent = getDisplayContentLocked(displayId);
   11011         if (displayContent != null) {
   11012             displayContent.updateDisplayInfo();
   11013         }
   11014         mWindowPlacerLocked.requestTraversal();
   11015     }
   11016 
   11017     @Override
   11018     public Object getWindowManagerLock() {
   11019         return mWindowMap;
   11020     }
   11021 
   11022     /**
   11023      * Hint to a token that its activity will relaunch, which will trigger removal and addition of
   11024      * a window.
   11025      * @param token Application token for which the activity will be relaunched.
   11026      */
   11027     public void setReplacingWindow(IBinder token, boolean animate) {
   11028         AppWindowToken appWindowToken = null;
   11029         synchronized (mWindowMap) {
   11030             appWindowToken = findAppWindowToken(token);
   11031             if (appWindowToken == null || !appWindowToken.isVisible()) {
   11032                 Slog.w(TAG_WM, "Attempted to set replacing window on non-existing app token "
   11033                         + token);
   11034                 return;
   11035             }
   11036             appWindowToken.setReplacingWindows(animate);
   11037         }
   11038     }
   11039 
   11040     /**
   11041      * Hint to a token that its windows will be replaced across activity relaunch.
   11042      * The windows would otherwise be removed  shortly following this as the
   11043      * activity is torn down.
   11044      * @param token Application token for which the activity will be relaunched.
   11045      * @param childrenOnly Whether to mark only child windows for replacement
   11046      *                     (for the case where main windows are being preserved/
   11047      *                     reused rather than replaced).
   11048      *
   11049      */
   11050     public void setReplacingWindows(IBinder token, boolean childrenOnly) {
   11051         AppWindowToken appWindowToken = null;
   11052         synchronized (mWindowMap) {
   11053             appWindowToken = findAppWindowToken(token);
   11054             if (appWindowToken == null || !appWindowToken.isVisible()) {
   11055                 Slog.w(TAG_WM, "Attempted to set replacing window on non-existing app token "
   11056                         + token);
   11057                 return;
   11058             }
   11059 
   11060             if (childrenOnly) {
   11061                 appWindowToken.setReplacingChildren();
   11062             } else {
   11063                 appWindowToken.setReplacingWindows(false /* animate */);
   11064             }
   11065 
   11066             scheduleClearReplacingWindowIfNeeded(token, true /* replacing */);
   11067         }
   11068     }
   11069 
   11070     /**
   11071      * If we're replacing the window, schedule a timer to clear the replaced window
   11072      * after a timeout, in case the replacing window is not coming.
   11073      *
   11074      * If we're not replacing the window, clear the replace window settings of the app.
   11075      *
   11076      * @param token Application token for the activity whose window might be replaced.
   11077      * @param replacing Whether the window is being replaced or not.
   11078      */
   11079     public void scheduleClearReplacingWindowIfNeeded(IBinder token, boolean replacing) {
   11080         AppWindowToken appWindowToken = null;
   11081         synchronized (mWindowMap) {
   11082             appWindowToken = findAppWindowToken(token);
   11083             if (appWindowToken == null) {
   11084                 Slog.w(TAG_WM, "Attempted to reset replacing window on non-existing app token "
   11085                         + token);
   11086                 return;
   11087             }
   11088             if (replacing) {
   11089                 scheduleReplacingWindowTimeouts(appWindowToken);
   11090             } else {
   11091                 appWindowToken.resetReplacingWindows();
   11092             }
   11093         }
   11094     }
   11095 
   11096     void scheduleReplacingWindowTimeouts(AppWindowToken appWindowToken) {
   11097         if (!mReplacingWindowTimeouts.contains(appWindowToken)) {
   11098             mReplacingWindowTimeouts.add(appWindowToken);
   11099         }
   11100         mH.removeMessages(H.WINDOW_REPLACEMENT_TIMEOUT);
   11101         mH.sendEmptyMessageDelayed(
   11102                 H.WINDOW_REPLACEMENT_TIMEOUT, WINDOW_REPLACEMENT_TIMEOUT_DURATION);
   11103     }
   11104 
   11105     @Override
   11106     public int getDockedStackSide() {
   11107         synchronized (mWindowMap) {
   11108             final TaskStack dockedStack = getDefaultDisplayContentLocked()
   11109                     .getDockedStackVisibleForUserLocked();
   11110             return dockedStack == null ? DOCKED_INVALID : dockedStack.getDockSide();
   11111         }
   11112     }
   11113 
   11114     @Override
   11115     public void setDockedStackResizing(boolean resizing) {
   11116         synchronized (mWindowMap) {
   11117             getDefaultDisplayContentLocked().getDockedDividerController().setResizing(resizing);
   11118             requestTraversal();
   11119         }
   11120     }
   11121 
   11122     @Override
   11123     public void setDockedStackDividerTouchRegion(Rect touchRegion) {
   11124         synchronized (mWindowMap) {
   11125             getDefaultDisplayContentLocked().getDockedDividerController()
   11126                     .setTouchRegion(touchRegion);
   11127             setFocusTaskRegionLocked();
   11128         }
   11129     }
   11130 
   11131     @Override
   11132     public void setResizeDimLayer(boolean visible, int targetStackId, float alpha) {
   11133         synchronized (mWindowMap) {
   11134             getDefaultDisplayContentLocked().getDockedDividerController().setResizeDimLayer(
   11135                     visible, targetStackId, alpha);
   11136         }
   11137     }
   11138 
   11139     public void animateResizePinnedStack(final Rect bounds, final int animationDuration) {
   11140         synchronized (mWindowMap) {
   11141             final TaskStack stack = mStackIdToStack.get(PINNED_STACK_ID);
   11142             if (stack == null) {
   11143                 Slog.w(TAG, "animateResizePinnedStack: stackId " + PINNED_STACK_ID + " not found.");
   11144                 return;
   11145             }
   11146             final Rect originalBounds = new Rect();
   11147             stack.getBounds(originalBounds);
   11148             UiThread.getHandler().post(new Runnable() {
   11149                 @Override
   11150                 public void run() {
   11151                     mBoundsAnimationController.animateBounds(
   11152                             stack, originalBounds, bounds, animationDuration);
   11153                 }
   11154             });
   11155         }
   11156     }
   11157 
   11158     public void setTaskResizeable(int taskId, int resizeMode) {
   11159         synchronized (mWindowMap) {
   11160             final Task task = mTaskIdToTask.get(taskId);
   11161             if (task != null) {
   11162                 task.setResizeable(resizeMode);
   11163             }
   11164         }
   11165     }
   11166 
   11167     public void setForceResizableTasks(boolean forceResizableTasks) {
   11168         synchronized (mWindowMap) {
   11169             mForceResizableTasks = forceResizableTasks;
   11170         }
   11171     }
   11172 
   11173     static int dipToPixel(int dip, DisplayMetrics displayMetrics) {
   11174         return (int)TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dip, displayMetrics);
   11175     }
   11176 
   11177     @Override
   11178     public void registerDockedStackListener(IDockedStackListener listener) {
   11179         if (!checkCallingPermission(android.Manifest.permission.REGISTER_WINDOW_MANAGER_LISTENERS,
   11180                 "registerDockedStackListener()")) {
   11181             return;
   11182         }
   11183         // TODO(multi-display): The listener is registered on the default display only.
   11184         getDefaultDisplayContentLocked().mDividerControllerLocked.registerDockedStackListener(
   11185                 listener);
   11186     }
   11187 
   11188     @Override
   11189     public void requestAppKeyboardShortcuts(IResultReceiver receiver, int deviceId) {
   11190         try {
   11191             WindowState focusedWindow = getFocusedWindow();
   11192             if (focusedWindow != null && focusedWindow.mClient != null) {
   11193                 getFocusedWindow().mClient.requestAppKeyboardShortcuts(receiver, deviceId);
   11194             }
   11195         } catch (RemoteException e) {
   11196         }
   11197     }
   11198 
   11199     @Override
   11200     public void getStableInsets(Rect outInsets) throws RemoteException {
   11201         synchronized (mWindowMap) {
   11202             getStableInsetsLocked(outInsets);
   11203         }
   11204     }
   11205 
   11206     void getStableInsetsLocked(Rect outInsets) {
   11207         final DisplayInfo di = getDefaultDisplayInfoLocked();
   11208         mPolicy.getStableInsetsLw(di.rotation, di.logicalWidth, di.logicalHeight, outInsets);
   11209     }
   11210 
   11211     private void getNonDecorInsetsLocked(Rect outInsets) {
   11212         final DisplayInfo di = getDefaultDisplayInfoLocked();
   11213         mPolicy.getNonDecorInsetsLw(di.rotation, di.logicalWidth, di.logicalHeight, outInsets);
   11214     }
   11215 
   11216     /**
   11217      * Intersects the specified {@code inOutBounds} with the display frame that excludes the stable
   11218      * inset areas.
   11219      *
   11220      * @param inOutBounds The inOutBounds to subtract the stable inset areas from.
   11221      */
   11222     public void subtractStableInsets(Rect inOutBounds) {
   11223         synchronized (mWindowMap) {
   11224             getStableInsetsLocked(mTmpRect2);
   11225             final DisplayInfo di = getDefaultDisplayInfoLocked();
   11226             mTmpRect.set(0, 0, di.logicalWidth, di.logicalHeight);
   11227             subtractInsets(mTmpRect, mTmpRect2, inOutBounds);
   11228         }
   11229     }
   11230 
   11231     /**
   11232      * Intersects the specified {@code inOutBounds} with the display frame that excludes
   11233      * areas that could never be removed in Honeycomb. See
   11234      * {@link WindowManagerPolicy#getNonDecorInsetsLw}.
   11235      *
   11236      * @param inOutBounds The inOutBounds to subtract the inset areas from.
   11237      */
   11238     public void subtractNonDecorInsets(Rect inOutBounds) {
   11239         synchronized (mWindowMap) {
   11240             getNonDecorInsetsLocked(mTmpRect2);
   11241             final DisplayInfo di = getDefaultDisplayInfoLocked();
   11242             mTmpRect.set(0, 0, di.logicalWidth, di.logicalHeight);
   11243             subtractInsets(mTmpRect, mTmpRect2, inOutBounds);
   11244         }
   11245     }
   11246 
   11247     void subtractInsets(Rect display, Rect insets, Rect inOutBounds) {
   11248         mTmpRect3.set(display);
   11249         mTmpRect3.inset(insets);
   11250         inOutBounds.intersect(mTmpRect3);
   11251     }
   11252 
   11253     /**
   11254      * Calculates the smallest width for a task given the {@param bounds}. It does that by iterating
   11255      * across all screen orientations, and returns the minimum of the task width taking into account
   11256      * that the bounds might change because the snap algorithm snaps to a different value.
   11257      *
   11258      * @return the smallest width to be used in the Configuration, in dips
   11259      */
   11260     public int getSmallestWidthForTaskBounds(Rect bounds) {
   11261         synchronized (mWindowMap) {
   11262             return getDefaultDisplayContentLocked().getDockedDividerController()
   11263                     .getSmallestWidthDpForBounds(bounds);
   11264         }
   11265     }
   11266 
   11267     private MousePositionTracker mMousePositionTracker = new MousePositionTracker();
   11268 
   11269     private static class MousePositionTracker implements PointerEventListener {
   11270         private boolean mLatestEventWasMouse;
   11271         private float mLatestMouseX;
   11272         private float mLatestMouseY;
   11273 
   11274         void updatePosition(float x, float y) {
   11275             synchronized (this) {
   11276                 mLatestEventWasMouse = true;
   11277                 mLatestMouseX = x;
   11278                 mLatestMouseY = y;
   11279             }
   11280         }
   11281 
   11282         @Override
   11283         public void onPointerEvent(MotionEvent motionEvent) {
   11284             if (motionEvent.isFromSource(InputDevice.SOURCE_MOUSE)) {
   11285                 updatePosition(motionEvent.getRawX(), motionEvent.getRawY());
   11286             } else {
   11287                 synchronized (this) {
   11288                     mLatestEventWasMouse = false;
   11289                 }
   11290             }
   11291         }
   11292     };
   11293 
   11294     void updatePointerIcon(IWindow client) {
   11295         float mouseX, mouseY;
   11296 
   11297         synchronized(mMousePositionTracker) {
   11298             if (!mMousePositionTracker.mLatestEventWasMouse) {
   11299                 return;
   11300             }
   11301             mouseX = mMousePositionTracker.mLatestMouseX;
   11302             mouseY = mMousePositionTracker.mLatestMouseY;
   11303         }
   11304 
   11305         synchronized (mWindowMap) {
   11306             if (mDragState != null) {
   11307                 // Drag cursor overrides the app cursor.
   11308                 return;
   11309             }
   11310             WindowState callingWin = windowForClientLocked(null, client, false);
   11311             if (callingWin == null) {
   11312                 Slog.w(TAG_WM, "Bad requesting window " + client);
   11313                 return;
   11314             }
   11315             final DisplayContent displayContent = callingWin.getDisplayContent();
   11316             if (displayContent == null) {
   11317                 return;
   11318             }
   11319             WindowState windowUnderPointer =
   11320                     displayContent.getTouchableWinAtPointLocked(mouseX, mouseY);
   11321             if (windowUnderPointer != callingWin) {
   11322                 return;
   11323             }
   11324             try {
   11325                 windowUnderPointer.mClient.updatePointerIcon(
   11326                         windowUnderPointer.translateToWindowX(mouseX),
   11327                         windowUnderPointer.translateToWindowY(mouseY));
   11328             } catch (RemoteException e) {
   11329                 Slog.w(TAG_WM, "unable to update pointer icon");
   11330             }
   11331         }
   11332     }
   11333 
   11334     void restorePointerIconLocked(DisplayContent displayContent, float latestX, float latestY) {
   11335         // Mouse position tracker has not been getting updates while dragging, update it now.
   11336         mMousePositionTracker.updatePosition(latestX, latestY);
   11337 
   11338         WindowState windowUnderPointer =
   11339                 displayContent.getTouchableWinAtPointLocked(latestX, latestY);
   11340         if (windowUnderPointer != null) {
   11341             try {
   11342                 windowUnderPointer.mClient.updatePointerIcon(
   11343                         windowUnderPointer.translateToWindowX(latestX),
   11344                         windowUnderPointer.translateToWindowY(latestY));
   11345             } catch (RemoteException e) {
   11346                 Slog.w(TAG_WM, "unable to restore pointer icon");
   11347             }
   11348         } else {
   11349             InputManager.getInstance().setPointerIconType(PointerIcon.TYPE_DEFAULT);
   11350         }
   11351     }
   11352 
   11353     @Override
   11354     public void registerShortcutKey(long shortcutCode, IShortcutService shortcutKeyReceiver)
   11355             throws RemoteException {
   11356         if (!checkCallingPermission(Manifest.permission.REGISTER_WINDOW_MANAGER_LISTENERS,
   11357                 "registerShortcutKey")) {
   11358             throw new SecurityException(
   11359                     "Requires REGISTER_WINDOW_MANAGER_LISTENERS permission");
   11360         }
   11361         mPolicy.registerShortcutKey(shortcutCode, shortcutKeyReceiver);
   11362     }
   11363 
   11364     private final class LocalService extends WindowManagerInternal {
   11365         @Override
   11366         public void requestTraversalFromDisplayManager() {
   11367             requestTraversal();
   11368         }
   11369 
   11370         @Override
   11371         public void setMagnificationSpec(MagnificationSpec spec) {
   11372             synchronized (mWindowMap) {
   11373                 if (mAccessibilityController != null) {
   11374                     mAccessibilityController.setMagnificationSpecLocked(spec);
   11375                 } else {
   11376                     throw new IllegalStateException("Magnification callbacks not set!");
   11377                 }
   11378             }
   11379             if (Binder.getCallingPid() != android.os.Process.myPid()) {
   11380                 spec.recycle();
   11381             }
   11382         }
   11383 
   11384         @Override
   11385         public void getMagnificationRegion(@NonNull Region magnificationRegion) {
   11386             synchronized (mWindowMap) {
   11387                 if (mAccessibilityController != null) {
   11388                     mAccessibilityController.getMagnificationRegionLocked(magnificationRegion);
   11389                 } else {
   11390                     throw new IllegalStateException("Magnification callbacks not set!");
   11391                 }
   11392             }
   11393         }
   11394 
   11395         @Override
   11396         public MagnificationSpec getCompatibleMagnificationSpecForWindow(IBinder windowToken) {
   11397             synchronized (mWindowMap) {
   11398                 WindowState windowState = mWindowMap.get(windowToken);
   11399                 if (windowState == null) {
   11400                     return null;
   11401                 }
   11402                 MagnificationSpec spec = null;
   11403                 if (mAccessibilityController != null) {
   11404                     spec = mAccessibilityController.getMagnificationSpecForWindowLocked(windowState);
   11405                 }
   11406                 if ((spec == null || spec.isNop()) && windowState.mGlobalScale == 1.0f) {
   11407                     return null;
   11408                 }
   11409                 spec = (spec == null) ? MagnificationSpec.obtain() : MagnificationSpec.obtain(spec);
   11410                 spec.scale *= windowState.mGlobalScale;
   11411                 return spec;
   11412             }
   11413         }
   11414 
   11415         @Override
   11416         public void setMagnificationCallbacks(@Nullable MagnificationCallbacks callbacks) {
   11417             synchronized (mWindowMap) {
   11418                 if (mAccessibilityController == null) {
   11419                     mAccessibilityController = new AccessibilityController(
   11420                             WindowManagerService.this);
   11421                 }
   11422                 mAccessibilityController.setMagnificationCallbacksLocked(callbacks);
   11423                 if (!mAccessibilityController.hasCallbacksLocked()) {
   11424                     mAccessibilityController = null;
   11425                 }
   11426             }
   11427         }
   11428 
   11429         @Override
   11430         public void setWindowsForAccessibilityCallback(WindowsForAccessibilityCallback callback) {
   11431             synchronized (mWindowMap) {
   11432                 if (mAccessibilityController == null) {
   11433                     mAccessibilityController = new AccessibilityController(
   11434                             WindowManagerService.this);
   11435                 }
   11436                 mAccessibilityController.setWindowsForAccessibilityCallback(callback);
   11437                 if (!mAccessibilityController.hasCallbacksLocked()) {
   11438                     mAccessibilityController = null;
   11439                 }
   11440             }
   11441         }
   11442 
   11443         @Override
   11444         public void setInputFilter(IInputFilter filter) {
   11445             mInputManager.setInputFilter(filter);
   11446         }
   11447 
   11448         @Override
   11449         public IBinder getFocusedWindowToken() {
   11450             synchronized (mWindowMap) {
   11451                 WindowState windowState = getFocusedWindowLocked();
   11452                 if (windowState != null) {
   11453                     return windowState.mClient.asBinder();
   11454                 }
   11455                 return null;
   11456             }
   11457         }
   11458 
   11459         @Override
   11460         public boolean isKeyguardLocked() {
   11461             return WindowManagerService.this.isKeyguardLocked();
   11462         }
   11463 
   11464         @Override
   11465         public void showGlobalActions() {
   11466             WindowManagerService.this.showGlobalActions();
   11467         }
   11468 
   11469         @Override
   11470         public void getWindowFrame(IBinder token, Rect outBounds) {
   11471             synchronized (mWindowMap) {
   11472                 WindowState windowState = mWindowMap.get(token);
   11473                 if (windowState != null) {
   11474                     outBounds.set(windowState.mFrame);
   11475                 } else {
   11476                     outBounds.setEmpty();
   11477                 }
   11478             }
   11479         }
   11480 
   11481         @Override
   11482         public void waitForAllWindowsDrawn(Runnable callback, long timeout) {
   11483             boolean allWindowsDrawn = false;
   11484             synchronized (mWindowMap) {
   11485                 mWaitingForDrawnCallback = callback;
   11486                 final WindowList windows = getDefaultWindowListLocked();
   11487                 for (int winNdx = windows.size() - 1; winNdx >= 0; --winNdx) {
   11488                     final WindowState win = windows.get(winNdx);
   11489                     final boolean isForceHiding = mPolicy.isForceHiding(win.mAttrs);
   11490                     final boolean keyguard = mPolicy.isKeyguardHostWindow(win.mAttrs);
   11491                     if (win.isVisibleLw()
   11492                             && (win.mAppToken != null || isForceHiding || keyguard)) {
   11493                         win.mWinAnimator.mDrawState = DRAW_PENDING;
   11494                         // Force add to mResizingWindows.
   11495                         win.mLastContentInsets.set(-1, -1, -1, -1);
   11496                         mWaitingForDrawn.add(win);
   11497 
   11498                         // No need to wait for the windows below Keyguard.
   11499                         if (isForceHiding) {
   11500                             break;
   11501                         }
   11502                     }
   11503                 }
   11504                 mWindowPlacerLocked.requestTraversal();
   11505                 mH.removeMessages(H.WAITING_FOR_DRAWN_TIMEOUT);
   11506                 if (mWaitingForDrawn.isEmpty()) {
   11507                     allWindowsDrawn = true;
   11508                 } else {
   11509                     mH.sendEmptyMessageDelayed(H.WAITING_FOR_DRAWN_TIMEOUT, timeout);
   11510                     checkDrawnWindowsLocked();
   11511                 }
   11512             }
   11513             if (allWindowsDrawn) {
   11514                 callback.run();
   11515             }
   11516         }
   11517 
   11518         @Override
   11519         public void addWindowToken(IBinder token, int type) {
   11520             WindowManagerService.this.addWindowToken(token, type);
   11521         }
   11522 
   11523         @Override
   11524         public void removeWindowToken(IBinder token, boolean removeWindows) {
   11525             synchronized(mWindowMap) {
   11526                 if (removeWindows) {
   11527                     WindowToken wtoken = mTokenMap.remove(token);
   11528                     if (wtoken != null) {
   11529                         wtoken.removeAllWindows();
   11530                     }
   11531                 }
   11532                 WindowManagerService.this.removeWindowToken(token);
   11533             }
   11534         }
   11535 
   11536         @Override
   11537         public void registerAppTransitionListener(AppTransitionListener listener) {
   11538             synchronized (mWindowMap) {
   11539                 mAppTransition.registerListenerLocked(listener);
   11540             }
   11541         }
   11542 
   11543         @Override
   11544         public int getInputMethodWindowVisibleHeight() {
   11545             synchronized (mWindowMap) {
   11546                 return mPolicy.getInputMethodWindowVisibleHeightLw();
   11547             }
   11548         }
   11549 
   11550         @Override
   11551         public void saveLastInputMethodWindowForTransition() {
   11552             synchronized (mWindowMap) {
   11553                 if (mInputMethodWindow != null) {
   11554                     mPolicy.setLastInputMethodWindowLw(mInputMethodWindow, mInputMethodTarget);
   11555                 }
   11556             }
   11557         }
   11558 
   11559         @Override
   11560         public void clearLastInputMethodWindowForTransition() {
   11561             synchronized (mWindowMap) {
   11562                 mPolicy.setLastInputMethodWindowLw(null, null);
   11563             }
   11564         }
   11565 
   11566         @Override
   11567         public boolean isHardKeyboardAvailable() {
   11568             synchronized (mWindowMap) {
   11569                 return mHardKeyboardAvailable;
   11570             }
   11571         }
   11572 
   11573         @Override
   11574         public void setOnHardKeyboardStatusChangeListener(
   11575                 OnHardKeyboardStatusChangeListener listener) {
   11576             synchronized (mWindowMap) {
   11577                 mHardKeyboardStatusChangeListener = listener;
   11578             }
   11579         }
   11580 
   11581         @Override
   11582         public boolean isStackVisible(int stackId) {
   11583             synchronized (mWindowMap) {
   11584                 return WindowManagerService.this.isStackVisibleLocked(stackId);
   11585             }
   11586         }
   11587 
   11588         @Override
   11589         public boolean isDockedDividerResizing() {
   11590             synchronized (mWindowMap) {
   11591                 return getDefaultDisplayContentLocked().getDockedDividerController().isResizing();
   11592             }
   11593         }
   11594     }
   11595 }
   11596