blob: 37a549a960a792cca347f489449603098e7995d5 [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2006 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
17package com.android.server.am;
18
Jorim Jaggi0a932142016-02-01 17:42:25 -080019import android.annotation.Nullable;
Craig Mautnerb0f7dc72013-04-01 16:34:45 -070020import android.app.Activity;
Craig Mautner9db9a0b2013-04-29 17:05:56 -070021import android.app.ActivityManager;
Wale Ogunwale3797c222015-10-27 14:21:58 -070022import android.app.ActivityManager.StackId;
Craig Mautner648f69b2014-09-18 14:16:26 -070023import android.app.ActivityManager.TaskDescription;
Wale Ogunwaleff3c66c2015-11-18 19:22:49 -080024import android.app.ActivityManager.TaskThumbnail;
Winsonc809cbb2015-11-02 12:06:15 -080025import android.app.ActivityManager.TaskThumbnailInfo;
Craig Mautnerb0f7dc72013-04-01 16:34:45 -070026import android.app.ActivityOptions;
Dianne Hackborn885fbe52014-08-23 15:23:58 -070027import android.app.AppGlobals;
Jorim Jaggi0a932142016-02-01 17:42:25 -080028import android.app.IActivityManager;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080029import android.content.ComponentName;
30import android.content.Intent;
31import android.content.pm.ActivityInfo;
Dianne Hackborn885fbe52014-08-23 15:23:58 -070032import android.content.pm.ApplicationInfo;
33import android.content.pm.IPackageManager;
34import android.content.pm.PackageManager;
Wale Ogunwalee4a0c572015-06-30 08:40:31 -070035import android.content.res.Configuration;
Craig Mautner9db9a0b2013-04-29 17:05:56 -070036import android.graphics.Bitmap;
Winsonc809cbb2015-11-02 12:06:15 -080037import android.graphics.Point;
Wale Ogunwalee4a0c572015-06-30 08:40:31 -070038import android.graphics.Rect;
Wale Ogunwale0fc365c2015-05-25 19:35:42 -070039import android.os.Debug;
Craig Mautnerc0ffce52014-07-01 12:38:52 -070040import android.os.ParcelFileDescriptor;
Dianne Hackborn885fbe52014-08-23 15:23:58 -070041import android.os.RemoteException;
Dianne Hackbornf02b60a2012-08-16 10:48:27 -070042import android.os.UserHandle;
Dianne Hackborn91097de2014-04-04 18:02:06 -070043import android.service.voice.IVoiceInteractionSession;
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -070044import android.util.DisplayMetrics;
Dianne Hackborn7f96b792012-05-29 18:46:45 -070045import android.util.Slog;
Suprabh Shukla23593142015-11-03 17:31:15 -080046
Dianne Hackborn91097de2014-04-04 18:02:06 -070047import com.android.internal.app.IVoiceInteractor;
Craig Mautner21d24a22014-04-23 11:45:37 -070048import com.android.internal.util.XmlUtils;
Suprabh Shukla23593142015-11-03 17:31:15 -080049
Craig Mautner21d24a22014-04-23 11:45:37 -070050import org.xmlpull.v1.XmlPullParser;
51import org.xmlpull.v1.XmlPullParserException;
52import org.xmlpull.v1.XmlSerializer;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080053
Craig Mautnerc0ffce52014-07-01 12:38:52 -070054import java.io.File;
Craig Mautner21d24a22014-04-23 11:45:37 -070055import java.io.IOException;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080056import java.io.PrintWriter;
Craig Mautner5d9c7be2013-02-15 14:02:56 -080057import java.util.ArrayList;
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -070058import java.util.Objects;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080059
Jorim Jaggi0a932142016-02-01 17:42:25 -080060import static android.app.ActivityManager.StackId.DOCKED_STACK_ID;
61import static android.app.ActivityManager.StackId.FREEFORM_WORKSPACE_STACK_ID;
62import static android.app.ActivityManager.StackId.FULLSCREEN_WORKSPACE_STACK_ID;
63import static android.app.ActivityManager.StackId.HOME_STACK_ID;
64import static android.app.ActivityManager.StackId.INVALID_STACK_ID;
65import static android.app.ActivityManager.StackId.PINNED_STACK_ID;
66import static android.content.Intent.FLAG_ACTIVITY_NEW_DOCUMENT;
67import static android.content.Intent.FLAG_ACTIVITY_RETAIN_IN_RECENTS;
68import static android.content.pm.ActivityInfo.LOCK_TASK_LAUNCH_MODE_ALWAYS;
69import static android.content.pm.ActivityInfo.LOCK_TASK_LAUNCH_MODE_DEFAULT;
70import static android.content.pm.ActivityInfo.LOCK_TASK_LAUNCH_MODE_IF_WHITELISTED;
71import static android.content.pm.ActivityInfo.LOCK_TASK_LAUNCH_MODE_NEVER;
72import static android.content.pm.ActivityInfo.RESIZE_MODE_CROP_WINDOWS;
Wale Ogunwaled829d362016-02-10 19:24:49 -080073import static android.content.pm.ActivityInfo.RESIZE_MODE_FORCE_RESIZEABLE;
Jorim Jaggi0a932142016-02-01 17:42:25 -080074import static android.content.pm.ActivityInfo.RESIZE_MODE_RESIZEABLE;
75import static android.content.pm.ActivityInfo.RESIZE_MODE_UNRESIZEABLE;
76import static android.content.pm.ApplicationInfo.PRIVATE_FLAG_PRIVILEGED;
77import static com.android.server.am.ActivityManagerDebugConfig.DEBUG_ADD_REMOVE;
78import static com.android.server.am.ActivityManagerDebugConfig.DEBUG_LOCKTASK;
79import static com.android.server.am.ActivityManagerDebugConfig.DEBUG_RECENTS;
80import static com.android.server.am.ActivityManagerDebugConfig.DEBUG_TASKS;
81import static com.android.server.am.ActivityManagerDebugConfig.POSTFIX_ADD_REMOVE;
82import static com.android.server.am.ActivityManagerDebugConfig.POSTFIX_LOCKTASK;
83import static com.android.server.am.ActivityManagerDebugConfig.POSTFIX_RECENTS;
84import static com.android.server.am.ActivityManagerDebugConfig.POSTFIX_TASKS;
85import static com.android.server.am.ActivityManagerDebugConfig.TAG_AM;
86import static com.android.server.am.ActivityManagerDebugConfig.TAG_WITH_CLASS_NAME;
87import static com.android.server.am.ActivityManagerService.LOCK_SCREEN_SHOWN;
88import static com.android.server.am.ActivityRecord.APPLICATION_ACTIVITY_TYPE;
89import static com.android.server.am.ActivityRecord.HOME_ACTIVITY_TYPE;
90import static com.android.server.am.ActivityRecord.RECENTS_ACTIVITY_TYPE;
91
Craig Mautnerc0ffce52014-07-01 12:38:52 -070092final class TaskRecord {
Wale Ogunwalee23149f2015-03-06 15:39:44 -080093 private static final String TAG = TAG_WITH_CLASS_NAME ? "TaskRecord" : TAG_AM;
Wale Ogunwale0fc365c2015-05-25 19:35:42 -070094 private static final String TAG_ADD_REMOVE = TAG + POSTFIX_ADD_REMOVE;
Wale Ogunwaleee006da2015-03-30 14:49:25 -070095 private static final String TAG_RECENTS = TAG + POSTFIX_RECENTS;
Craig Mautnere0570202015-05-13 13:06:11 -070096 private static final String TAG_LOCKTASK = TAG + POSTFIX_LOCKTASK;
Wale Ogunwaleee006da2015-03-30 14:49:25 -070097 private static final String TAG_TASKS = TAG + POSTFIX_TASKS;
Wale Ogunwalee23149f2015-03-06 15:39:44 -080098
Wale Ogunwale18795a22014-12-03 11:38:33 -080099 static final String ATTR_TASKID = "task_id";
Craig Mautner21d24a22014-04-23 11:45:37 -0700100 private static final String TAG_INTENT = "intent";
101 private static final String TAG_AFFINITYINTENT = "affinity_intent";
Wale Ogunwale18795a22014-12-03 11:38:33 -0800102 static final String ATTR_REALACTIVITY = "real_activity";
Andrei Stingaceanu4ccec532016-01-13 12:10:21 +0000103 static final String ATTR_REALACTIVITY_SUSPENDED = "real_activity_suspended";
Craig Mautner21d24a22014-04-23 11:45:37 -0700104 private static final String ATTR_ORIGACTIVITY = "orig_activity";
Stefan Kuhnee88d1e52015-05-18 10:33:45 -0700105 private static final String TAG_ACTIVITY = "activity";
Craig Mautner21d24a22014-04-23 11:45:37 -0700106 private static final String ATTR_AFFINITY = "affinity";
Dianne Hackborn79228822014-09-16 11:11:23 -0700107 private static final String ATTR_ROOT_AFFINITY = "root_affinity";
Craig Mautner21d24a22014-04-23 11:45:37 -0700108 private static final String ATTR_ROOTHASRESET = "root_has_reset";
Dianne Hackborn13420f22014-07-18 15:43:56 -0700109 private static final String ATTR_AUTOREMOVERECENTS = "auto_remove_recents";
Craig Mautner21d24a22014-04-23 11:45:37 -0700110 private static final String ATTR_ASKEDCOMPATMODE = "asked_compat_mode";
111 private static final String ATTR_USERID = "user_id";
Wale Ogunwalef80170f2016-02-04 15:12:29 -0800112 private static final String ATTR_USER_SETUP_COMPLETE = "user_setup_complete";
Dianne Hackborn885fbe52014-08-23 15:23:58 -0700113 private static final String ATTR_EFFECTIVE_UID = "effective_uid";
Craig Mautner21d24a22014-04-23 11:45:37 -0700114 private static final String ATTR_TASKTYPE = "task_type";
Winson Chungffa2ec62014-07-03 15:54:42 -0700115 private static final String ATTR_FIRSTACTIVETIME = "first_active_time";
Winson Chungf1fbd772014-06-24 18:06:58 -0700116 private static final String ATTR_LASTACTIVETIME = "last_active_time";
Craig Mautner21d24a22014-04-23 11:45:37 -0700117 private static final String ATTR_LASTDESCRIPTION = "last_description";
118 private static final String ATTR_LASTTIMEMOVED = "last_time_moved";
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700119 private static final String ATTR_NEVERRELINQUISH = "never_relinquish_identity";
Wale Ogunwale18795a22014-12-03 11:38:33 -0800120 static final String ATTR_TASK_AFFILIATION = "task_affiliation";
Craig Mautnera228ae92014-07-09 05:44:55 -0700121 private static final String ATTR_PREV_AFFILIATION = "prev_affiliation";
122 private static final String ATTR_NEXT_AFFILIATION = "next_affiliation";
Winson Chungec396d62014-08-06 17:08:00 -0700123 private static final String ATTR_TASK_AFFILIATION_COLOR = "task_affiliation_color";
Craig Mautnerdc00cbe2014-07-20 17:48:47 -0700124 private static final String ATTR_CALLING_UID = "calling_uid";
125 private static final String ATTR_CALLING_PACKAGE = "calling_package";
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800126 private static final String ATTR_RESIZE_MODE = "resize_mode";
Wale Ogunwalea7afaa22015-05-01 20:39:18 -0700127 private static final String ATTR_PRIVILEGED = "privileged";
Wale Ogunwale706ed792015-08-02 10:29:44 -0700128 private static final String ATTR_NON_FULLSCREEN_BOUNDS = "non_fullscreen_bounds";
Craig Mautner21d24a22014-04-23 11:45:37 -0700129
130 private static final String TASK_THUMBNAIL_SUFFIX = "_task_thumbnail";
131
Wale Ogunwale18795a22014-12-03 11:38:33 -0800132 static final int INVALID_TASK_ID = -1;
133
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800134 final int taskId; // Unique identifier for this task.
Dianne Hackborn79228822014-09-16 11:11:23 -0700135 String affinity; // The affinity name for this task, or null; may change identity.
136 String rootAffinity; // Initial base affinity, or null; does not change from initial root.
Dianne Hackborn91097de2014-04-04 18:02:06 -0700137 final IVoiceInteractionSession voiceSession; // Voice interaction session driving task
138 final IVoiceInteractor voiceInteractor; // Associated interactor to provide to app
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800139 Intent intent; // The original intent that started the task.
140 Intent affinityIntent; // Intent of affinity-moved activity that started this task.
Dianne Hackborn885fbe52014-08-23 15:23:58 -0700141 int effectiveUid; // The current effective uid of the identity of this task.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800142 ComponentName origActivity; // The non-alias activity component of the intent.
143 ComponentName realActivity; // The actual activity component that started the task.
Andrei Stingaceanu4ccec532016-01-13 12:10:21 +0000144 boolean realActivitySuspended; // True if the actual activity component that started the
145 // task is suspended.
Winson Chungffa2ec62014-07-03 15:54:42 -0700146 long firstActiveTime; // First time this task was active.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800147 long lastActiveTime; // Last time this task was active, including sleep.
Dianne Hackborn852975d2014-08-22 17:42:43 -0700148 boolean inRecents; // Actually in the recents list?
149 boolean isAvailable; // Is the activity available to be launched?
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800150 boolean rootWasReset; // True if the intent at the root of the task had
151 // the FLAG_ACTIVITY_RESET_TASK_IF_NEEDED flag.
Dianne Hackborn13420f22014-07-18 15:43:56 -0700152 boolean autoRemoveRecents; // If true, we should automatically remove the task from
153 // recents when activity finishes
Dianne Hackborn36cd41f2011-05-25 21:00:46 -0700154 boolean askedCompatMode;// Have asked the user about compat mode for this task.
Dianne Hackbornd38aed82014-06-10 21:36:35 -0700155 boolean hasBeenVisible; // Set if any activities in the task have been visible to the user.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800156
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700157 String stringName; // caching of toString() result.
Dianne Hackborn9da2d402012-03-15 13:43:08 -0700158 int userId; // user for which this task was created
Wale Ogunwalef80170f2016-02-04 15:12:29 -0800159 boolean mUserSetupComplete; // The user set-up is complete as of the last time the task activity
160 // was changed.
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800161
162 int numFullscreen; // Number of fullscreen activities.
163
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800164 int mResizeMode; // The resize mode of this task and its activities.
165 // Based on the {@link ActivityInfo#resizeMode} of the root activity.
Jorim Jaggi8202b2a2016-02-03 19:24:31 -0800166 boolean mTemporarilyUnresizable; // Separate flag from mResizeMode used to suppress resize
167 // changes on a temporary basis.
Craig Mautner15df08a2015-04-01 12:17:18 -0700168 int mLockTaskMode; // Which tasklock mode to launch this task in. One of
169 // ActivityManager.LOCK_TASK_LAUNCH_MODE_*
Wale Ogunwalea7afaa22015-05-01 20:39:18 -0700170 private boolean mPrivileged; // The root activity application of this task holds
171 // privileged permissions.
172
Craig Mautner15df08a2015-04-01 12:17:18 -0700173 /** Can't be put in lockTask mode. */
174 final static int LOCK_TASK_AUTH_DONT_LOCK = 0;
Benjamin Franz469dd582015-06-09 14:24:36 +0100175 /** Can enter app pinning with user approval. Can never start over existing lockTask task. */
Craig Mautner15df08a2015-04-01 12:17:18 -0700176 final static int LOCK_TASK_AUTH_PINNABLE = 1;
177 /** Starts in LOCK_TASK_MODE_LOCKED automatically. Can start over existing lockTask task. */
178 final static int LOCK_TASK_AUTH_LAUNCHABLE = 2;
Benjamin Franz469dd582015-06-09 14:24:36 +0100179 /** Can enter lockTask without user approval. Can start over existing lockTask task. */
Craig Mautner15df08a2015-04-01 12:17:18 -0700180 final static int LOCK_TASK_AUTH_WHITELISTED = 3;
Benjamin Franz469dd582015-06-09 14:24:36 +0100181 /** Priv-app that starts in LOCK_TASK_MODE_LOCKED automatically. Can start over existing
182 * lockTask task. */
183 final static int LOCK_TASK_AUTH_LAUNCHABLE_PRIV = 4;
Craig Mautner15df08a2015-04-01 12:17:18 -0700184 int mLockTaskAuth = LOCK_TASK_AUTH_PINNABLE;
185
186 int mLockTaskUid = -1; // The uid of the application that called startLockTask().
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800187
Winson Chung03a9bae2014-05-02 09:56:12 -0700188 // This represents the last resolved activity values for this task
189 // NOTE: This value needs to be persisted with each task
Craig Mautner648f69b2014-09-18 14:16:26 -0700190 TaskDescription lastTaskDescription = new TaskDescription();
Winson Chung03a9bae2014-05-02 09:56:12 -0700191
Craig Mautnerd2328952013-03-05 12:46:26 -0800192 /** List of all activities in the task arranged in history order */
Craig Mautner21d24a22014-04-23 11:45:37 -0700193 final ArrayList<ActivityRecord> mActivities;
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800194
Craig Mautnerd2328952013-03-05 12:46:26 -0800195 /** Current stack */
196 ActivityStack stack;
197
Craig Mautner2c1faed2013-07-23 12:56:02 -0700198 /** Takes on same set of values as ActivityRecord.mActivityType */
Craig Mautner21d24a22014-04-23 11:45:37 -0700199 int taskType;
Craig Mautner1602ec22013-05-12 10:24:27 -0700200
Craig Mautner21d24a22014-04-23 11:45:37 -0700201 /** Takes on same value as first root activity */
202 boolean isPersistable = false;
Craig Mautnerffcfcaa2014-06-05 09:54:38 -0700203 int maxRecents;
Craig Mautner21d24a22014-04-23 11:45:37 -0700204
205 /** Only used for persistable tasks, otherwise 0. The last time this task was moved. Used for
206 * determining the order when restoring. Sign indicates whether last task movement was to front
207 * (positive) or back (negative). Absolute value indicates time. */
208 long mLastTimeMoved = System.currentTimeMillis();
209
Craig Mautner84984fa2014-06-19 11:19:20 -0700210 /** Indication of what to run next when task exits. Use ActivityRecord types.
211 * ActivityRecord.APPLICATION_ACTIVITY_TYPE indicates to resume the task below this one in the
212 * task stack. */
213 private int mTaskToReturnTo = APPLICATION_ACTIVITY_TYPE;
Craig Mautnerae7ecab2013-09-18 11:48:14 -0700214
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700215 /** If original intent did not allow relinquishing task identity, save that information */
216 boolean mNeverRelinquishIdentity = true;
217
Craig Mautner362449a2014-06-20 14:04:39 -0700218 // Used in the unique case where we are clearing the task in order to reuse it. In that case we
219 // do not want to delete the stack when the task goes empty.
Filip Gruszczynskibe9dabd2016-01-19 12:23:10 -0800220 private boolean mReuseTask = false;
Craig Mautner362449a2014-06-20 14:04:39 -0700221
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700222 private Bitmap mLastThumbnail; // Last thumbnail captured for this item.
Wale Ogunwalebe23ff42014-10-21 16:29:51 -0700223 private final File mLastThumbnailFile; // File containing last thumbnail.
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700224 private final String mFilename;
Winsonc809cbb2015-11-02 12:06:15 -0800225 private TaskThumbnailInfo mLastThumbnailInfo;
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700226 CharSequence lastDescription; // Last description captured for this item.
227
Craig Mautnera228ae92014-07-09 05:44:55 -0700228 int mAffiliatedTaskId; // taskId of parent affiliation or self if no parent.
Winson Chungec396d62014-08-06 17:08:00 -0700229 int mAffiliatedTaskColor; // color of the parent task affiliation.
Craig Mautnera228ae92014-07-09 05:44:55 -0700230 TaskRecord mPrevAffiliate; // previous task in affiliated chain.
Wale Ogunwale18795a22014-12-03 11:38:33 -0800231 int mPrevAffiliateTaskId = INVALID_TASK_ID; // previous id for persistence.
Craig Mautnera228ae92014-07-09 05:44:55 -0700232 TaskRecord mNextAffiliate; // next task in affiliated chain.
Wale Ogunwale18795a22014-12-03 11:38:33 -0800233 int mNextAffiliateTaskId = INVALID_TASK_ID; // next id for persistence.
Craig Mautnera228ae92014-07-09 05:44:55 -0700234
Craig Mautnerdc00cbe2014-07-20 17:48:47 -0700235 // For relaunching the task from recents as though it was launched by the original launcher.
236 int mCallingUid;
237 String mCallingPackage;
238
Craig Mautner21d24a22014-04-23 11:45:37 -0700239 final ActivityManagerService mService;
240
Wale Ogunwalee4a0c572015-06-30 08:40:31 -0700241 // Whether or not this task covers the entire screen; by default tasks are fullscreen.
242 boolean mFullscreen = true;
243
244 // Bounds of the Task. null for fullscreen tasks.
245 Rect mBounds = null;
Jorim Jaggi82c9dc92016-02-05 15:10:33 -0800246 private final Rect mTmpStableBounds = new Rect();
247 private final Rect mTmpNonDecorBounds = new Rect();
Wale Ogunwale9a08f822016-02-17 19:03:58 -0800248 private final Rect mTmpRect = new Rect();
Jorim Jaggi0a932142016-02-01 17:42:25 -0800249 private final Rect mTmpRect2 = new Rect();
250
Wale Ogunwale706ed792015-08-02 10:29:44 -0700251 // Last non-fullscreen bounds the task was launched in or resized to.
252 // The information is persisted and used to determine the appropriate stack to launch the
253 // task into on restore.
254 Rect mLastNonFullscreenBounds = null;
Filip Gruszczynskid2f1d942015-10-14 15:10:12 -0700255 // Minimal size for width/height of this task when it's resizeable. -1 means it should use the
256 // default minimal size.
257 final int mMinimalSize;
Wale Ogunwalee4a0c572015-06-30 08:40:31 -0700258
Chong Zhangfdcc4d42015-10-14 16:50:12 -0700259 // Ranking (from top) of this task among all visible tasks. (-1 means it's not visible)
260 // This number will be assigned when we evaluate OOM scores for all visible tasks.
261 int mLayerRank = -1;
262
Wale Ogunwalee4a0c572015-06-30 08:40:31 -0700263 Configuration mOverrideConfig = Configuration.EMPTY;
264
Craig Mautner21d24a22014-04-23 11:45:37 -0700265 TaskRecord(ActivityManagerService service, int _taskId, ActivityInfo info, Intent _intent,
Dianne Hackborn91097de2014-04-04 18:02:06 -0700266 IVoiceInteractionSession _voiceSession, IVoiceInteractor _voiceInteractor) {
Craig Mautner21d24a22014-04-23 11:45:37 -0700267 mService = service;
Craig Mautnera228ae92014-07-09 05:44:55 -0700268 mFilename = String.valueOf(_taskId) + TASK_THUMBNAIL_SUFFIX +
269 TaskPersister.IMAGE_EXTENSION;
Suprabh Shukla23593142015-11-03 17:31:15 -0800270 userId = UserHandle.getUserId(info.applicationInfo.uid);
271 mLastThumbnailFile = new File(TaskPersister.getUserImagesDir(userId), mFilename);
Winsonc809cbb2015-11-02 12:06:15 -0800272 mLastThumbnailInfo = new TaskThumbnailInfo();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800273 taskId = _taskId;
Craig Mautnera228ae92014-07-09 05:44:55 -0700274 mAffiliatedTaskId = _taskId;
Dianne Hackborn91097de2014-04-04 18:02:06 -0700275 voiceSession = _voiceSession;
276 voiceInteractor = _voiceInteractor;
Dianne Hackborn852975d2014-08-22 17:42:43 -0700277 isAvailable = true;
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800278 mActivities = new ArrayList<>();
Craig Mautner15df08a2015-04-01 12:17:18 -0700279 mCallingUid = info.applicationInfo.uid;
280 mCallingPackage = info.packageName;
Martijn Coenend4a69702014-06-30 11:12:17 -0700281 setIntent(_intent, info);
Filip Gruszczynskid2f1d942015-10-14 15:10:12 -0700282 mMinimalSize = info != null && info.layout != null ? info.layout.minimalSize : -1;
Craig Mautner21d24a22014-04-23 11:45:37 -0700283 }
284
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700285 TaskRecord(ActivityManagerService service, int _taskId, ActivityInfo info, Intent _intent,
Winsonc809cbb2015-11-02 12:06:15 -0800286 TaskDescription _taskDescription, TaskThumbnailInfo thumbnailInfo) {
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700287 mService = service;
288 mFilename = String.valueOf(_taskId) + TASK_THUMBNAIL_SUFFIX +
289 TaskPersister.IMAGE_EXTENSION;
Suprabh Shukla23593142015-11-03 17:31:15 -0800290 userId = UserHandle.getUserId(info.applicationInfo.uid);
291 mLastThumbnailFile = new File(TaskPersister.getUserImagesDir(userId), mFilename);
Winsonc809cbb2015-11-02 12:06:15 -0800292 mLastThumbnailInfo = thumbnailInfo;
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700293 taskId = _taskId;
294 mAffiliatedTaskId = _taskId;
295 voiceSession = null;
296 voiceInteractor = null;
Dianne Hackborn852975d2014-08-22 17:42:43 -0700297 isAvailable = true;
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800298 mActivities = new ArrayList<>();
Craig Mautner15df08a2015-04-01 12:17:18 -0700299 mCallingUid = info.applicationInfo.uid;
300 mCallingPackage = info.packageName;
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700301 setIntent(_intent, info);
302
303 taskType = ActivityRecord.APPLICATION_ACTIVITY_TYPE;
304 isPersistable = true;
Dianne Hackborn852975d2014-08-22 17:42:43 -0700305 // Clamp to [1, max].
306 maxRecents = Math.min(Math.max(info.maxRecents, 1),
307 ActivityManager.getMaxAppRecentsLimitStatic());
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700308
309 taskType = APPLICATION_ACTIVITY_TYPE;
310 mTaskToReturnTo = HOME_ACTIVITY_TYPE;
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700311 lastTaskDescription = _taskDescription;
Filip Gruszczynskid2f1d942015-10-14 15:10:12 -0700312 mMinimalSize = info != null && info.layout != null ? info.layout.minimalSize : -1;
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700313 }
314
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800315 private TaskRecord(ActivityManagerService service, int _taskId, Intent _intent,
316 Intent _affinityIntent, String _affinity, String _rootAffinity,
317 ComponentName _realActivity, ComponentName _origActivity, boolean _rootWasReset,
318 boolean _autoRemoveRecents, boolean _askedCompatMode, int _taskType, int _userId,
319 int _effectiveUid, String _lastDescription, ArrayList<ActivityRecord> activities,
320 long _firstActiveTime, long _lastActiveTime, long lastTimeMoved,
321 boolean neverRelinquishIdentity, TaskDescription _lastTaskDescription,
Winsonc809cbb2015-11-02 12:06:15 -0800322 TaskThumbnailInfo lastThumbnailInfo, int taskAffiliation, int prevTaskId,
323 int nextTaskId, int taskAffiliationColor, int callingUid, String callingPackage,
Wale Ogunwalef80170f2016-02-04 15:12:29 -0800324 int resizeMode, boolean privileged, boolean _realActivitySuspended,
325 boolean userSetupComplete) {
Craig Mautner21d24a22014-04-23 11:45:37 -0700326 mService = service;
Craig Mautnera228ae92014-07-09 05:44:55 -0700327 mFilename = String.valueOf(_taskId) + TASK_THUMBNAIL_SUFFIX +
328 TaskPersister.IMAGE_EXTENSION;
Suprabh Shukla23593142015-11-03 17:31:15 -0800329 mLastThumbnailFile = new File(TaskPersister.getUserImagesDir(_userId), mFilename);
Winsonc809cbb2015-11-02 12:06:15 -0800330 mLastThumbnailInfo = lastThumbnailInfo;
Craig Mautner21d24a22014-04-23 11:45:37 -0700331 taskId = _taskId;
332 intent = _intent;
333 affinityIntent = _affinityIntent;
334 affinity = _affinity;
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800335 rootAffinity = _rootAffinity;
Craig Mautner21d24a22014-04-23 11:45:37 -0700336 voiceSession = null;
337 voiceInteractor = null;
338 realActivity = _realActivity;
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800339 realActivitySuspended = _realActivitySuspended;
Craig Mautner21d24a22014-04-23 11:45:37 -0700340 origActivity = _origActivity;
341 rootWasReset = _rootWasReset;
Dianne Hackborn852975d2014-08-22 17:42:43 -0700342 isAvailable = true;
Dianne Hackborn13420f22014-07-18 15:43:56 -0700343 autoRemoveRecents = _autoRemoveRecents;
Craig Mautner21d24a22014-04-23 11:45:37 -0700344 askedCompatMode = _askedCompatMode;
345 taskType = _taskType;
Craig Mautner84984fa2014-06-19 11:19:20 -0700346 mTaskToReturnTo = HOME_ACTIVITY_TYPE;
Craig Mautner21d24a22014-04-23 11:45:37 -0700347 userId = _userId;
Wale Ogunwalef80170f2016-02-04 15:12:29 -0800348 mUserSetupComplete = userSetupComplete;
Dianne Hackborn885fbe52014-08-23 15:23:58 -0700349 effectiveUid = _effectiveUid;
Winson Chungffa2ec62014-07-03 15:54:42 -0700350 firstActiveTime = _firstActiveTime;
Winson Chungf1fbd772014-06-24 18:06:58 -0700351 lastActiveTime = _lastActiveTime;
Craig Mautner21d24a22014-04-23 11:45:37 -0700352 lastDescription = _lastDescription;
353 mActivities = activities;
354 mLastTimeMoved = lastTimeMoved;
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700355 mNeverRelinquishIdentity = neverRelinquishIdentity;
Winson Chung2cb86c72014-06-25 12:03:30 -0700356 lastTaskDescription = _lastTaskDescription;
Craig Mautnera228ae92014-07-09 05:44:55 -0700357 mAffiliatedTaskId = taskAffiliation;
Winson Chungec396d62014-08-06 17:08:00 -0700358 mAffiliatedTaskColor = taskAffiliationColor;
Craig Mautnera228ae92014-07-09 05:44:55 -0700359 mPrevAffiliateTaskId = prevTaskId;
360 mNextAffiliateTaskId = nextTaskId;
Craig Mautnerdc00cbe2014-07-20 17:48:47 -0700361 mCallingUid = callingUid;
362 mCallingPackage = callingPackage;
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800363 mResizeMode = resizeMode;
Wale Ogunwalea7afaa22015-05-01 20:39:18 -0700364 mPrivileged = privileged;
Suprabh Shukla74f9f162015-11-06 15:39:42 -0800365 ActivityInfo info = (mActivities.size() > 0) ? mActivities.get(0).info : null;
Filip Gruszczynskid2f1d942015-10-14 15:10:12 -0700366 mMinimalSize = info != null && info.layout != null ? info.layout.minimalSize : -1;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800367 }
368
369 void touchActiveTime() {
Craig Mautner494f6bdd2014-07-21 11:17:46 -0700370 lastActiveTime = System.currentTimeMillis();
Winson Chungffa2ec62014-07-03 15:54:42 -0700371 if (firstActiveTime == 0) {
372 firstActiveTime = lastActiveTime;
373 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800374 }
Craig Mautner9db9a0b2013-04-29 17:05:56 -0700375
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800376 long getInactiveDuration() {
Craig Mautner494f6bdd2014-07-21 11:17:46 -0700377 return System.currentTimeMillis() - lastActiveTime;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800378 }
Craig Mautner9db9a0b2013-04-29 17:05:56 -0700379
Winson Chungfee26772014-08-05 12:21:52 -0700380 /** Sets the original intent, and the calling uid and package. */
381 void setIntent(ActivityRecord r) {
Winson Chungfee26772014-08-05 12:21:52 -0700382 mCallingUid = r.launchedFromUid;
383 mCallingPackage = r.launchedFromPackage;
Craig Mautner15df08a2015-04-01 12:17:18 -0700384 setIntent(r.intent, r.info);
Winson Chungfee26772014-08-05 12:21:52 -0700385 }
386
387 /** Sets the original intent, _without_ updating the calling uid or package. */
388 private void setIntent(Intent _intent, ActivityInfo info) {
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700389 if (intent == null) {
390 mNeverRelinquishIdentity =
391 (info.flags & ActivityInfo.FLAG_RELINQUISH_TASK_IDENTITY) == 0;
392 } else if (mNeverRelinquishIdentity) {
393 return;
394 }
395
396 affinity = info.taskAffinity;
Dianne Hackborn79228822014-09-16 11:11:23 -0700397 if (intent == null) {
398 // If this task already has an intent associated with it, don't set the root
399 // affinity -- we don't want it changing after initially set, but the initially
400 // set value may be null.
401 rootAffinity = affinity;
402 }
Dianne Hackborn885fbe52014-08-23 15:23:58 -0700403 effectiveUid = info.applicationInfo.uid;
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700404 stringName = null;
Dianne Hackbornf5b86712011-12-05 17:42:41 -0800405
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800406 if (info.targetActivity == null) {
Dianne Hackbornf5b86712011-12-05 17:42:41 -0800407 if (_intent != null) {
408 // If this Intent has a selector, we want to clear it for the
409 // recent task since it is not relevant if the user later wants
410 // to re-launch the app.
Dianne Hackbornd367ca82012-05-07 15:49:39 -0700411 if (_intent.getSelector() != null || _intent.getSourceBounds() != null) {
Dianne Hackbornf5b86712011-12-05 17:42:41 -0800412 _intent = new Intent(_intent);
413 _intent.setSelector(null);
Dianne Hackbornd367ca82012-05-07 15:49:39 -0700414 _intent.setSourceBounds(null);
Dianne Hackbornf5b86712011-12-05 17:42:41 -0800415 }
416 }
Wale Ogunwaleee006da2015-03-30 14:49:25 -0700417 if (DEBUG_TASKS) Slog.v(TAG_TASKS, "Setting Intent of " + this + " to " + _intent);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800418 intent = _intent;
419 realActivity = _intent != null ? _intent.getComponent() : null;
420 origActivity = null;
421 } else {
422 ComponentName targetComponent = new ComponentName(
423 info.packageName, info.targetActivity);
424 if (_intent != null) {
425 Intent targetIntent = new Intent(_intent);
426 targetIntent.setComponent(targetComponent);
Dianne Hackbornf5b86712011-12-05 17:42:41 -0800427 targetIntent.setSelector(null);
Dianne Hackbornd367ca82012-05-07 15:49:39 -0700428 targetIntent.setSourceBounds(null);
Wale Ogunwaleee006da2015-03-30 14:49:25 -0700429 if (DEBUG_TASKS) Slog.v(TAG_TASKS,
Dianne Hackborn7f96b792012-05-29 18:46:45 -0700430 "Setting Intent of " + this + " to target " + targetIntent);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800431 intent = targetIntent;
432 realActivity = targetComponent;
433 origActivity = _intent.getComponent();
434 } else {
435 intent = null;
436 realActivity = targetComponent;
437 origActivity = new ComponentName(info.packageName, info.name);
438 }
439 }
Amith Yamasani742a6712011-05-04 14:49:28 -0700440
Craig Mautner47b20ba2014-09-17 17:23:44 -0700441 final int intentFlags = intent == null ? 0 : intent.getFlags();
442 if ((intentFlags & Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED) != 0) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800443 // Once we are set to an Intent with this flag, we count this
444 // task as having a true root activity.
445 rootWasReset = true;
446 }
Amith Yamasani742a6712011-05-04 14:49:28 -0700447
Dianne Hackborn09233282014-04-30 11:33:59 -0700448 userId = UserHandle.getUserId(info.applicationInfo.uid);
Wale Ogunwalef80170f2016-02-04 15:12:29 -0800449 mUserSetupComplete = mService.mUserController.isUserSetupCompleteLocked(userId);
Craig Mautner41db4a72014-05-07 17:20:56 -0700450 if ((info.flags & ActivityInfo.FLAG_AUTO_REMOVE_FROM_RECENTS) != 0) {
Dianne Hackborn13420f22014-07-18 15:43:56 -0700451 // If the activity itself has requested auto-remove, then just always do it.
452 autoRemoveRecents = true;
Wale Ogunwale843bfb92015-03-27 11:06:48 -0700453 } else if ((intentFlags & (FLAG_ACTIVITY_NEW_DOCUMENT | FLAG_ACTIVITY_RETAIN_IN_RECENTS))
454 == FLAG_ACTIVITY_NEW_DOCUMENT) {
Dianne Hackborn13420f22014-07-18 15:43:56 -0700455 // If the caller has not asked for the document to be retained, then we may
456 // want to turn on auto-remove, depending on whether the target has set its
457 // own document launch mode.
458 if (info.documentLaunchMode != ActivityInfo.DOCUMENT_LAUNCH_NONE) {
459 autoRemoveRecents = false;
460 } else {
461 autoRemoveRecents = true;
462 }
463 } else {
464 autoRemoveRecents = false;
Craig Mautner41db4a72014-05-07 17:20:56 -0700465 }
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800466 mResizeMode = info.resizeMode;
Craig Mautner15df08a2015-04-01 12:17:18 -0700467 mLockTaskMode = info.lockTaskLaunchMode;
Wale Ogunwalea7afaa22015-05-01 20:39:18 -0700468 mPrivileged = (info.applicationInfo.privateFlags & PRIVATE_FLAG_PRIVILEGED) != 0;
Craig Mautner15df08a2015-04-01 12:17:18 -0700469 setLockTaskAuth();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800470 }
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800471
Craig Mautner84984fa2014-06-19 11:19:20 -0700472 void setTaskToReturnTo(int taskToReturnTo) {
Wale Ogunwale673cbd22016-01-30 18:30:55 -0800473 mTaskToReturnTo = (taskToReturnTo == RECENTS_ACTIVITY_TYPE)
474 ? HOME_ACTIVITY_TYPE : taskToReturnTo;
Craig Mautner84984fa2014-06-19 11:19:20 -0700475 }
476
477 int getTaskToReturnTo() {
478 return mTaskToReturnTo;
479 }
480
Craig Mautnera228ae92014-07-09 05:44:55 -0700481 void setPrevAffiliate(TaskRecord prevAffiliate) {
482 mPrevAffiliate = prevAffiliate;
Wale Ogunwale18795a22014-12-03 11:38:33 -0800483 mPrevAffiliateTaskId = prevAffiliate == null ? INVALID_TASK_ID : prevAffiliate.taskId;
Craig Mautnera228ae92014-07-09 05:44:55 -0700484 }
485
486 void setNextAffiliate(TaskRecord nextAffiliate) {
487 mNextAffiliate = nextAffiliate;
Wale Ogunwale18795a22014-12-03 11:38:33 -0800488 mNextAffiliateTaskId = nextAffiliate == null ? INVALID_TASK_ID : nextAffiliate.taskId;
Craig Mautnera228ae92014-07-09 05:44:55 -0700489 }
490
491 // Close up recents linked list.
492 void closeRecentsChain() {
493 if (mPrevAffiliate != null) {
494 mPrevAffiliate.setNextAffiliate(mNextAffiliate);
495 }
496 if (mNextAffiliate != null) {
497 mNextAffiliate.setPrevAffiliate(mPrevAffiliate);
498 }
499 setPrevAffiliate(null);
500 setNextAffiliate(null);
501 }
502
Winson Chung740c3ac2014-11-12 16:14:38 -0800503 void removedFromRecents() {
Dianne Hackborn852975d2014-08-22 17:42:43 -0700504 disposeThumbnail();
505 closeRecentsChain();
506 if (inRecents) {
507 inRecents = false;
Winson Chung740c3ac2014-11-12 16:14:38 -0800508 mService.notifyTaskPersisterLocked(this, false);
Dianne Hackborn852975d2014-08-22 17:42:43 -0700509 }
510 }
511
Craig Mautnera228ae92014-07-09 05:44:55 -0700512 void setTaskToAffiliateWith(TaskRecord taskToAffiliateWith) {
513 closeRecentsChain();
514 mAffiliatedTaskId = taskToAffiliateWith.mAffiliatedTaskId;
Winson Chungec396d62014-08-06 17:08:00 -0700515 mAffiliatedTaskColor = taskToAffiliateWith.mAffiliatedTaskColor;
Craig Mautnera228ae92014-07-09 05:44:55 -0700516 // Find the end
517 while (taskToAffiliateWith.mNextAffiliate != null) {
518 final TaskRecord nextRecents = taskToAffiliateWith.mNextAffiliate;
519 if (nextRecents.mAffiliatedTaskId != mAffiliatedTaskId) {
520 Slog.e(TAG, "setTaskToAffiliateWith: nextRecents=" + nextRecents + " affilTaskId="
521 + nextRecents.mAffiliatedTaskId + " should be " + mAffiliatedTaskId);
522 if (nextRecents.mPrevAffiliate == taskToAffiliateWith) {
523 nextRecents.setPrevAffiliate(null);
524 }
525 taskToAffiliateWith.setNextAffiliate(null);
526 break;
527 }
528 taskToAffiliateWith = nextRecents;
529 }
530 taskToAffiliateWith.setNextAffiliate(this);
531 setPrevAffiliate(taskToAffiliateWith);
532 setNextAffiliate(null);
533 }
534
Winson Chung096f36b2014-08-20 15:39:01 -0700535 /**
Winsonc809cbb2015-11-02 12:06:15 -0800536 * Sets the last thumbnail with the current task bounds and the system orientation.
Winson Chung096f36b2014-08-20 15:39:01 -0700537 * @return whether the thumbnail was set
538 */
Winsonc809cbb2015-11-02 12:06:15 -0800539 boolean setLastThumbnailLocked(Bitmap thumbnail) {
540 final Configuration serviceConfig = mService.mConfiguration;
541 int taskWidth = 0;
542 int taskHeight = 0;
543 if (mBounds != null) {
544 // Non-fullscreen tasks
545 taskWidth = mBounds.width();
546 taskHeight = mBounds.height();
547 } else if (stack != null) {
548 // Fullscreen tasks
549 final Point displaySize = new Point();
550 stack.getDisplaySize(displaySize);
551 taskWidth = displaySize.x;
552 taskHeight = displaySize.y;
553 } else {
554 Slog.e(TAG, "setLastThumbnailLocked() called on Task without stack");
555 }
556 return setLastThumbnailLocked(thumbnail, taskWidth, taskHeight, serviceConfig.orientation);
557 }
558
559 /**
560 * Sets the last thumbnail with the current task bounds.
561 * @return whether the thumbnail was set
562 */
563 private boolean setLastThumbnailLocked(Bitmap thumbnail, int taskWidth, int taskHeight,
564 int screenOrientation) {
Winson Chung096f36b2014-08-20 15:39:01 -0700565 if (mLastThumbnail != thumbnail) {
566 mLastThumbnail = thumbnail;
Winsonc809cbb2015-11-02 12:06:15 -0800567 mLastThumbnailInfo.taskWidth = taskWidth;
568 mLastThumbnailInfo.taskHeight = taskHeight;
569 mLastThumbnailInfo.screenOrientation = screenOrientation;
Winson Chung096f36b2014-08-20 15:39:01 -0700570 if (thumbnail == null) {
571 if (mLastThumbnailFile != null) {
572 mLastThumbnailFile.delete();
573 }
574 } else {
Suprabh Shukla09a88f52015-12-02 14:36:31 -0800575 mService.mRecentTasks.saveImage(thumbnail, mLastThumbnailFile.getAbsolutePath());
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700576 }
Winson Chung096f36b2014-08-20 15:39:01 -0700577 return true;
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700578 }
Winson Chung096f36b2014-08-20 15:39:01 -0700579 return false;
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700580 }
581
582 void getLastThumbnail(TaskThumbnail thumbs) {
583 thumbs.mainThumbnail = mLastThumbnail;
Winsonc809cbb2015-11-02 12:06:15 -0800584 thumbs.thumbnailInfo = mLastThumbnailInfo;
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700585 thumbs.thumbnailFileDescriptor = null;
Craig Mautnerf4f8bb72014-07-29 10:41:40 -0700586 if (mLastThumbnail == null) {
Suprabh Shukla09a88f52015-12-02 14:36:31 -0800587 thumbs.mainThumbnail = mService.mRecentTasks.getImageFromWriteQueue(
Suprabh Shukla23593142015-11-03 17:31:15 -0800588 mLastThumbnailFile.getAbsolutePath());
Craig Mautnerf4f8bb72014-07-29 10:41:40 -0700589 }
Winson Chung096f36b2014-08-20 15:39:01 -0700590 // Only load the thumbnail file if we don't have a thumbnail
591 if (thumbs.mainThumbnail == null && mLastThumbnailFile.exists()) {
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700592 try {
593 thumbs.thumbnailFileDescriptor = ParcelFileDescriptor.open(mLastThumbnailFile,
594 ParcelFileDescriptor.MODE_READ_ONLY);
595 } catch (IOException e) {
Dianne Hackborn9844d292013-10-04 16:44:22 -0700596 }
597 }
598 }
599
Winsonc809cbb2015-11-02 12:06:15 -0800600 /**
601 * Removes in-memory thumbnail data when the max number of in-memory task thumbnails is reached.
602 */
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700603 void freeLastThumbnail() {
604 mLastThumbnail = null;
605 }
606
Winsonc809cbb2015-11-02 12:06:15 -0800607 /**
608 * Removes all associated thumbnail data when a task is removed or pruned from recents.
609 */
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700610 void disposeThumbnail() {
Winsonc6a2da02015-11-11 18:11:59 -0800611 mLastThumbnailInfo.reset();
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700612 mLastThumbnail = null;
613 lastDescription = null;
614 }
615
Winson Chung1147c402014-05-14 11:05:00 -0700616 /** Returns the intent for the root activity for this task */
617 Intent getBaseIntent() {
618 return intent != null ? intent : affinityIntent;
619 }
620
Winson Chung3b3f4642014-04-22 10:08:18 -0700621 /** Returns the first non-finishing activity from the root. */
622 ActivityRecord getRootActivity() {
623 for (int i = 0; i < mActivities.size(); i++) {
624 final ActivityRecord r = mActivities.get(i);
625 if (r.finishing) {
626 continue;
627 }
628 return r;
629 }
630 return null;
631 }
632
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800633 ActivityRecord getTopActivity() {
634 for (int i = mActivities.size() - 1; i >= 0; --i) {
635 final ActivityRecord r = mActivities.get(i);
636 if (r.finishing) {
637 continue;
638 }
639 return r;
640 }
641 return null;
642 }
643
Filip Gruszczynski3e85ba22015-10-05 22:48:30 -0700644 ActivityRecord topRunningActivityLocked() {
Wale Ogunwale7d701172015-03-11 15:36:30 -0700645 if (stack != null) {
646 for (int activityNdx = mActivities.size() - 1; activityNdx >= 0; --activityNdx) {
647 ActivityRecord r = mActivities.get(activityNdx);
Filip Gruszczynski3e85ba22015-10-05 22:48:30 -0700648 if (!r.finishing && stack.okToShowLocked(r)) {
Wale Ogunwale7d701172015-03-11 15:36:30 -0700649 return r;
650 }
Craig Mautner6b74cb52013-09-27 17:02:21 -0700651 }
652 }
653 return null;
654 }
655
Wale Ogunwale6d42df62015-11-05 09:50:55 -0800656 void setFrontOfTask() {
657 setFrontOfTask(null);
658 }
659
Craig Mautner3b475fe2013-12-16 15:58:31 -0800660 /** Call after activity movement or finish to make sure that frontOfTask is set correctly */
Wale Ogunwale6d42df62015-11-05 09:50:55 -0800661 void setFrontOfTask(ActivityRecord newTop) {
662 // If a top candidate is suggested by the caller, go ahead and use it and mark all others
663 // as not front. This is needed in situations where the current front activity in the
664 // task isn't finished yet and we want to set the front to the activity moved to the front
665 // of the task.
666 boolean foundFront = newTop != null ? true : false;
667
Craig Mautner3b475fe2013-12-16 15:58:31 -0800668 final int numActivities = mActivities.size();
Craig Mautner704e40b2013-12-18 16:43:51 -0800669 for (int activityNdx = 0; activityNdx < numActivities; ++activityNdx) {
Craig Mautner3b475fe2013-12-16 15:58:31 -0800670 final ActivityRecord r = mActivities.get(activityNdx);
671 if (foundFront || r.finishing) {
672 r.frontOfTask = false;
673 } else {
674 r.frontOfTask = true;
675 // Set frontOfTask false for every following activity.
676 foundFront = true;
677 }
678 }
Craig Mautner9587ee02014-06-23 15:00:10 +0000679 if (!foundFront && numActivities > 0) {
680 // All activities of this task are finishing. As we ought to have a frontOfTask
681 // activity, make the bottom activity front.
682 mActivities.get(0).frontOfTask = true;
683 }
Wale Ogunwale6d42df62015-11-05 09:50:55 -0800684 if (newTop != null) {
685 newTop.frontOfTask = true;
686 }
Craig Mautner3b475fe2013-12-16 15:58:31 -0800687 }
688
Craig Mautnerde4ef022013-04-07 19:01:33 -0700689 /**
Craig Mautner3b475fe2013-12-16 15:58:31 -0800690 * Reorder the history stack so that the passed activity is brought to the front.
Craig Mautnerde4ef022013-04-07 19:01:33 -0700691 */
692 final void moveActivityToFrontLocked(ActivityRecord newTop) {
Wale Ogunwale0fc365c2015-05-25 19:35:42 -0700693 if (DEBUG_ADD_REMOVE) Slog.i(TAG_ADD_REMOVE,
694 "Removing and adding activity " + newTop
695 + " to stack at top callers=" + Debug.getCallers(4));
Craig Mautnerde4ef022013-04-07 19:01:33 -0700696
Craig Mautnerde4ef022013-04-07 19:01:33 -0700697 mActivities.remove(newTop);
698 mActivities.add(newTop);
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700699 updateEffectiveIntent();
Craig Mautner3b475fe2013-12-16 15:58:31 -0800700
Wale Ogunwale6d42df62015-11-05 09:50:55 -0800701 setFrontOfTask(newTop);
Craig Mautnerde4ef022013-04-07 19:01:33 -0700702 }
703
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800704 void addActivityAtBottom(ActivityRecord r) {
Craig Mautner77878772013-03-04 19:46:24 -0800705 addActivityAtIndex(0, r);
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800706 }
707
708 void addActivityToTop(ActivityRecord r) {
Craig Mautner1602ec22013-05-12 10:24:27 -0700709 addActivityAtIndex(mActivities.size(), r);
710 }
711
712 void addActivityAtIndex(int index, ActivityRecord r) {
Craig Mautner6170f732013-04-02 13:05:23 -0700713 // Remove r first, and if it wasn't already in the list and it's fullscreen, count it.
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800714 if (!mActivities.remove(r) && r.fullscreen) {
715 // Was not previously in list.
716 numFullscreen++;
717 }
Craig Mautner2c1faed2013-07-23 12:56:02 -0700718 // Only set this based on the first activity
719 if (mActivities.isEmpty()) {
Craig Mautner21d24a22014-04-23 11:45:37 -0700720 taskType = r.mActivityType;
721 isPersistable = r.isPersistable();
Craig Mautnerdc00cbe2014-07-20 17:48:47 -0700722 mCallingUid = r.launchedFromUid;
723 mCallingPackage = r.launchedFromPackage;
Dianne Hackborn852975d2014-08-22 17:42:43 -0700724 // Clamp to [1, max].
725 maxRecents = Math.min(Math.max(r.info.maxRecents, 1),
726 ActivityManager.getMaxAppRecentsLimitStatic());
Craig Mautner2c1faed2013-07-23 12:56:02 -0700727 } else {
728 // Otherwise make all added activities match this one.
Craig Mautner21d24a22014-04-23 11:45:37 -0700729 r.mActivityType = taskType;
Craig Mautner78733002013-06-10 13:54:49 -0700730 }
Craig Mautner77878772013-03-04 19:46:24 -0800731 mActivities.add(index, r);
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700732 updateEffectiveIntent();
Craig Mautner21d24a22014-04-23 11:45:37 -0700733 if (r.isPersistable()) {
734 mService.notifyTaskPersisterLocked(this, false);
735 }
Craig Mautner77878772013-03-04 19:46:24 -0800736 }
737
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800738 /** @return true if this was the last activity in the task */
739 boolean removeActivity(ActivityRecord r) {
740 if (mActivities.remove(r) && r.fullscreen) {
741 // Was previously in list.
742 numFullscreen--;
743 }
Craig Mautner21d24a22014-04-23 11:45:37 -0700744 if (r.isPersistable()) {
745 mService.notifyTaskPersisterLocked(this, false);
746 }
Craig Mautner41326202014-06-20 14:38:21 -0700747 if (mActivities.isEmpty()) {
Craig Mautner5afcd4d2014-06-21 18:11:33 -0700748 return !mReuseTask;
Craig Mautner41326202014-06-20 14:38:21 -0700749 }
750 updateEffectiveIntent();
751 return false;
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800752 }
753
Craig Mautner41db4a72014-05-07 17:20:56 -0700754 boolean autoRemoveFromRecents() {
Dianne Hackbornd38aed82014-06-10 21:36:35 -0700755 // We will automatically remove the task either if it has explicitly asked for
756 // this, or it is empty and has never contained an activity that got shown to
757 // the user.
Dianne Hackborn13420f22014-07-18 15:43:56 -0700758 return autoRemoveRecents || (mActivities.isEmpty() && !hasBeenVisible);
Craig Mautner41db4a72014-05-07 17:20:56 -0700759 }
760
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700761 /**
762 * Completely remove all activities associated with an existing
763 * task starting at a specified index.
764 */
765 final void performClearTaskAtIndexLocked(int activityNdx) {
Craig Mautner1602ec22013-05-12 10:24:27 -0700766 int numActivities = mActivities.size();
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700767 for ( ; activityNdx < numActivities; ++activityNdx) {
Craig Mautner1602ec22013-05-12 10:24:27 -0700768 final ActivityRecord r = mActivities.get(activityNdx);
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700769 if (r.finishing) {
770 continue;
771 }
Craig Mautner21d24a22014-04-23 11:45:37 -0700772 if (stack == null) {
773 // Task was restored from persistent storage.
774 r.takeFromHistory();
775 mActivities.remove(activityNdx);
776 --activityNdx;
777 --numActivities;
Todd Kennedy539db512014-12-15 09:57:55 -0800778 } else if (stack.finishActivityLocked(
779 r, Activity.RESULT_CANCELED, null, "clear-task-index", false)) {
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700780 --activityNdx;
781 --numActivities;
782 }
783 }
784 }
785
786 /**
787 * Completely remove all activities associated with an existing task.
788 */
789 final void performClearTaskLocked() {
Craig Mautner362449a2014-06-20 14:04:39 -0700790 mReuseTask = true;
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700791 performClearTaskAtIndexLocked(0);
Craig Mautner362449a2014-06-20 14:04:39 -0700792 mReuseTask = false;
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700793 }
794
Filip Gruszczynskibe9dabd2016-01-19 12:23:10 -0800795 ActivityRecord performClearTaskForReuseLocked(ActivityRecord newR, int launchFlags) {
796 mReuseTask = true;
797 final ActivityRecord result = performClearTaskLocked(newR, launchFlags);
798 mReuseTask = false;
799 return result;
800 }
801
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700802 /**
803 * Perform clear operation as requested by
804 * {@link Intent#FLAG_ACTIVITY_CLEAR_TOP}: search from the top of the
805 * stack to the given task, then look for
806 * an instance of that activity in the stack and, if found, finish all
807 * activities on top of it and return the instance.
808 *
809 * @param newR Description of the new activity being started.
810 * @return Returns the old activity that should be continued to be used,
811 * or null if none was found.
812 */
813 final ActivityRecord performClearTaskLocked(ActivityRecord newR, int launchFlags) {
Craig Mautner1602ec22013-05-12 10:24:27 -0700814 int numActivities = mActivities.size();
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700815 for (int activityNdx = numActivities - 1; activityNdx >= 0; --activityNdx) {
Craig Mautner1602ec22013-05-12 10:24:27 -0700816 ActivityRecord r = mActivities.get(activityNdx);
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700817 if (r.finishing) {
818 continue;
819 }
820 if (r.realActivity.equals(newR.realActivity)) {
821 // Here it is! Now finish everything in front...
Craig Mautner1602ec22013-05-12 10:24:27 -0700822 final ActivityRecord ret = r;
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700823
824 for (++activityNdx; activityNdx < numActivities; ++activityNdx) {
Craig Mautner1602ec22013-05-12 10:24:27 -0700825 r = mActivities.get(activityNdx);
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700826 if (r.finishing) {
827 continue;
828 }
829 ActivityOptions opts = r.takeOptionsLocked();
830 if (opts != null) {
831 ret.updateOptionsLocked(opts);
832 }
Wale Ogunwale7d701172015-03-11 15:36:30 -0700833 if (stack != null && stack.finishActivityLocked(
Todd Kennedy539db512014-12-15 09:57:55 -0800834 r, Activity.RESULT_CANCELED, null, "clear-task-stack", false)) {
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700835 --activityNdx;
836 --numActivities;
837 }
838 }
839
840 // Finally, if this is a normal launch mode (that is, not
841 // expecting onNewIntent()), then we will finish the current
842 // instance of the activity so a new fresh one can be started.
843 if (ret.launchMode == ActivityInfo.LAUNCH_MULTIPLE
844 && (launchFlags & Intent.FLAG_ACTIVITY_SINGLE_TOP) == 0) {
845 if (!ret.finishing) {
Wale Ogunwale7d701172015-03-11 15:36:30 -0700846 if (stack != null) {
847 stack.finishActivityLocked(
848 ret, Activity.RESULT_CANCELED, null, "clear-task-top", false);
849 }
Craig Mautnerb0f7dc72013-04-01 16:34:45 -0700850 return null;
851 }
852 }
853
854 return ret;
855 }
856 }
857
858 return null;
859 }
860
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700861 public TaskThumbnail getTaskThumbnailLocked() {
Craig Mautner21d24a22014-04-23 11:45:37 -0700862 if (stack != null) {
863 final ActivityRecord resumedActivity = stack.mResumedActivity;
864 if (resumedActivity != null && resumedActivity.task == this) {
Winson8b1871d2015-11-20 09:56:20 -0800865 final Bitmap thumbnail = stack.screenshotActivitiesLocked(resumedActivity);
Winsonc809cbb2015-11-02 12:06:15 -0800866 setLastThumbnailLocked(thumbnail);
Craig Mautner21d24a22014-04-23 11:45:37 -0700867 }
Craig Mautner9db9a0b2013-04-29 17:05:56 -0700868 }
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700869 final TaskThumbnail taskThumbnail = new TaskThumbnail();
870 getLastThumbnail(taskThumbnail);
871 return taskThumbnail;
Craig Mautner9db9a0b2013-04-29 17:05:56 -0700872 }
873
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700874 public void removeTaskActivitiesLocked() {
875 // Just remove the entire task.
876 performClearTaskAtIndexLocked(0);
Craig Mautner9db9a0b2013-04-29 17:05:56 -0700877 }
878
Craig Mautner432f64e2015-05-20 14:59:57 -0700879 String lockTaskAuthToString() {
880 switch (mLockTaskAuth) {
881 case LOCK_TASK_AUTH_DONT_LOCK: return "LOCK_TASK_AUTH_DONT_LOCK";
882 case LOCK_TASK_AUTH_PINNABLE: return "LOCK_TASK_AUTH_PINNABLE";
883 case LOCK_TASK_AUTH_LAUNCHABLE: return "LOCK_TASK_AUTH_LAUNCHABLE";
884 case LOCK_TASK_AUTH_WHITELISTED: return "LOCK_TASK_AUTH_WHITELISTED";
Benjamin Franz469dd582015-06-09 14:24:36 +0100885 case LOCK_TASK_AUTH_LAUNCHABLE_PRIV: return "LOCK_TASK_AUTH_LAUNCHABLE_PRIV";
Craig Mautner432f64e2015-05-20 14:59:57 -0700886 default: return "unknown=" + mLockTaskAuth;
887 }
888 }
889
Craig Mautner15df08a2015-04-01 12:17:18 -0700890 void setLockTaskAuth() {
Benjamin Franz469dd582015-06-09 14:24:36 +0100891 if (!mPrivileged &&
892 (mLockTaskMode == LOCK_TASK_LAUNCH_MODE_ALWAYS ||
893 mLockTaskMode == LOCK_TASK_LAUNCH_MODE_NEVER)) {
894 // Non-priv apps are not allowed to use always or never, fall back to default
895 mLockTaskMode = LOCK_TASK_LAUNCH_MODE_DEFAULT;
896 }
Craig Mautner15df08a2015-04-01 12:17:18 -0700897 switch (mLockTaskMode) {
898 case LOCK_TASK_LAUNCH_MODE_DEFAULT:
899 mLockTaskAuth = isLockTaskWhitelistedLocked() ?
900 LOCK_TASK_AUTH_WHITELISTED : LOCK_TASK_AUTH_PINNABLE;
901 break;
902
903 case LOCK_TASK_LAUNCH_MODE_NEVER:
Benjamin Franz469dd582015-06-09 14:24:36 +0100904 mLockTaskAuth = LOCK_TASK_AUTH_DONT_LOCK;
Craig Mautner15df08a2015-04-01 12:17:18 -0700905 break;
906
907 case LOCK_TASK_LAUNCH_MODE_ALWAYS:
Benjamin Franz469dd582015-06-09 14:24:36 +0100908 mLockTaskAuth = LOCK_TASK_AUTH_LAUNCHABLE_PRIV;
Craig Mautner15df08a2015-04-01 12:17:18 -0700909 break;
910
911 case LOCK_TASK_LAUNCH_MODE_IF_WHITELISTED:
912 mLockTaskAuth = isLockTaskWhitelistedLocked() ?
913 LOCK_TASK_AUTH_LAUNCHABLE : LOCK_TASK_AUTH_PINNABLE;
914 break;
915 }
Craig Mautner432f64e2015-05-20 14:59:57 -0700916 if (DEBUG_LOCKTASK) Slog.d(TAG_LOCKTASK, "setLockTaskAuth: task=" + this +
917 " mLockTaskAuth=" + lockTaskAuthToString());
Craig Mautner15df08a2015-04-01 12:17:18 -0700918 }
919
920 boolean isLockTaskWhitelistedLocked() {
Benjamin Franz6fd84cc2015-08-06 18:09:03 +0100921 String pkg = (realActivity != null) ? realActivity.getPackageName() : null;
922 if (pkg == null) {
Craig Mautner15df08a2015-04-01 12:17:18 -0700923 return false;
924 }
925 String[] packages = mService.mLockTaskPackages.get(userId);
926 if (packages == null) {
927 return false;
928 }
929 for (int i = packages.length - 1; i >= 0; --i) {
Benjamin Franz6fd84cc2015-08-06 18:09:03 +0100930 if (pkg.equals(packages[i])) {
Craig Mautner15df08a2015-04-01 12:17:18 -0700931 return true;
932 }
933 }
934 return false;
935 }
Wale Ogunwale74e26592016-02-05 11:48:37 -0800936
Craig Mautnera82aa092013-09-13 15:34:08 -0700937 boolean isHomeTask() {
Craig Mautner84984fa2014-06-19 11:19:20 -0700938 return taskType == HOME_ACTIVITY_TYPE;
Craig Mautnera82aa092013-09-13 15:34:08 -0700939 }
940
Wale Ogunwale74e26592016-02-05 11:48:37 -0800941 boolean isRecentsTask() {
942 return taskType == RECENTS_ACTIVITY_TYPE;
943 }
944
Craig Mautner86d67a42013-05-14 10:34:38 -0700945 boolean isApplicationTask() {
Craig Mautner84984fa2014-06-19 11:19:20 -0700946 return taskType == APPLICATION_ACTIVITY_TYPE;
947 }
948
949 boolean isOverHomeStack() {
950 return mTaskToReturnTo == HOME_ACTIVITY_TYPE || mTaskToReturnTo == RECENTS_ACTIVITY_TYPE;
Craig Mautner1602ec22013-05-12 10:24:27 -0700951 }
952
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800953 boolean isResizeable() {
954 return !isHomeTask() && (mService.mForceResizableActivities
Jorim Jaggi8202b2a2016-02-03 19:24:31 -0800955 || ActivityInfo.isResizeableMode(mResizeMode)) && !mTemporarilyUnresizable;
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800956 }
957
958 boolean inCropWindowsResizeMode() {
959 return !isResizeable() && mResizeMode == RESIZE_MODE_CROP_WINDOWS;
960 }
961
Wale Ogunwale513346d2016-01-27 10:55:01 -0800962 boolean canGoInDockedStack() {
963 return isResizeable() || inCropWindowsResizeMode();
964 }
965
Craig Mautner525f3d92013-05-07 14:01:50 -0700966 /**
967 * Find the activity in the history stack within the given task. Returns
968 * the index within the history at which it's found, or < 0 if not found.
969 */
970 final ActivityRecord findActivityInHistoryLocked(ActivityRecord r) {
971 final ComponentName realActivity = r.realActivity;
972 for (int activityNdx = mActivities.size() - 1; activityNdx >= 0; --activityNdx) {
973 ActivityRecord candidate = mActivities.get(activityNdx);
974 if (candidate.finishing) {
975 continue;
976 }
977 if (candidate.realActivity.equals(realActivity)) {
978 return candidate;
979 }
980 }
981 return null;
982 }
983
Winson Chunga449dc02014-05-16 11:15:04 -0700984 /** Updates the last task description values. */
985 void updateTaskDescription() {
986 // Traverse upwards looking for any break between main task activities and
987 // utility activities.
988 int activityNdx;
989 final int numActivities = mActivities.size();
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700990 final boolean relinquish = numActivities == 0 ? false :
991 (mActivities.get(0).info.flags & ActivityInfo.FLAG_RELINQUISH_TASK_IDENTITY) != 0;
Winson Chunga449dc02014-05-16 11:15:04 -0700992 for (activityNdx = Math.min(numActivities, 1); activityNdx < numActivities;
Craig Mautner21d24a22014-04-23 11:45:37 -0700993 ++activityNdx) {
Winson Chunga449dc02014-05-16 11:15:04 -0700994 final ActivityRecord r = mActivities.get(activityNdx);
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700995 if (relinquish && (r.info.flags & ActivityInfo.FLAG_RELINQUISH_TASK_IDENTITY) == 0) {
996 // This will be the top activity for determining taskDescription. Pre-inc to
997 // overcome initial decrement below.
998 ++activityNdx;
999 break;
1000 }
Winson Chunga449dc02014-05-16 11:15:04 -07001001 if (r.intent != null &&
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001002 (r.intent.getFlags() & Intent.FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET) != 0) {
Winson Chunga449dc02014-05-16 11:15:04 -07001003 break;
1004 }
1005 }
1006 if (activityNdx > 0) {
1007 // Traverse downwards starting below break looking for set label, icon.
1008 // Note that if there are activities in the task but none of them set the
1009 // recent activity values, then we do not fall back to the last set
1010 // values in the TaskRecord.
1011 String label = null;
Craig Mautner648f69b2014-09-18 14:16:26 -07001012 String iconFilename = null;
Winson Chunga449dc02014-05-16 11:15:04 -07001013 int colorPrimary = 0;
Winson Chung1af8eda2016-02-05 17:55:56 +00001014 int colorBackground = 0;
Winson Chunga449dc02014-05-16 11:15:04 -07001015 for (--activityNdx; activityNdx >= 0; --activityNdx) {
1016 final ActivityRecord r = mActivities.get(activityNdx);
1017 if (r.taskDescription != null) {
1018 if (label == null) {
1019 label = r.taskDescription.getLabel();
1020 }
Craig Mautner648f69b2014-09-18 14:16:26 -07001021 if (iconFilename == null) {
1022 iconFilename = r.taskDescription.getIconFilename();
Winson Chunga449dc02014-05-16 11:15:04 -07001023 }
1024 if (colorPrimary == 0) {
1025 colorPrimary = r.taskDescription.getPrimaryColor();
Winson Chunga449dc02014-05-16 11:15:04 -07001026 }
Winson Chung1af8eda2016-02-05 17:55:56 +00001027 if (colorBackground == 0) {
1028 colorBackground = r.taskDescription.getBackgroundColor();
1029 }
Winson Chunga449dc02014-05-16 11:15:04 -07001030 }
1031 }
Winson Chung1af8eda2016-02-05 17:55:56 +00001032 lastTaskDescription = new TaskDescription(label, null, iconFilename, colorPrimary,
1033 colorBackground);
Winson Chungec396d62014-08-06 17:08:00 -07001034 // Update the task affiliation color if we are the parent of the group
1035 if (taskId == mAffiliatedTaskId) {
1036 mAffiliatedTaskColor = lastTaskDescription.getPrimaryColor();
1037 }
Winson Chunga449dc02014-05-16 11:15:04 -07001038 }
1039 }
1040
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001041 int findEffectiveRootIndex() {
Craig Mautner4767f4b2014-09-18 15:38:33 -07001042 int effectiveNdx = 0;
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001043 final int topActivityNdx = mActivities.size() - 1;
Dianne Hackborn39569af2014-09-23 10:56:58 -07001044 for (int activityNdx = 0; activityNdx <= topActivityNdx; ++activityNdx) {
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001045 final ActivityRecord r = mActivities.get(activityNdx);
1046 if (r.finishing) {
1047 continue;
1048 }
Craig Mautner4767f4b2014-09-18 15:38:33 -07001049 effectiveNdx = activityNdx;
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001050 if ((r.info.flags & ActivityInfo.FLAG_RELINQUISH_TASK_IDENTITY) == 0) {
1051 break;
1052 }
1053 }
Craig Mautner4767f4b2014-09-18 15:38:33 -07001054 return effectiveNdx;
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001055 }
1056
1057 void updateEffectiveIntent() {
1058 final int effectiveRootIndex = findEffectiveRootIndex();
1059 final ActivityRecord r = mActivities.get(effectiveRootIndex);
Winson Chungfee26772014-08-05 12:21:52 -07001060 setIntent(r);
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001061 }
1062
Craig Mautner21d24a22014-04-23 11:45:37 -07001063 void saveToXml(XmlSerializer out) throws IOException, XmlPullParserException {
Wale Ogunwaleee006da2015-03-30 14:49:25 -07001064 if (DEBUG_RECENTS) Slog.i(TAG_RECENTS, "Saving task=" + this);
Craig Mautner21d24a22014-04-23 11:45:37 -07001065
1066 out.attribute(null, ATTR_TASKID, String.valueOf(taskId));
1067 if (realActivity != null) {
1068 out.attribute(null, ATTR_REALACTIVITY, realActivity.flattenToShortString());
1069 }
Andrei Stingaceanu4ccec532016-01-13 12:10:21 +00001070 out.attribute(null, ATTR_REALACTIVITY_SUSPENDED, String.valueOf(realActivitySuspended));
Craig Mautner21d24a22014-04-23 11:45:37 -07001071 if (origActivity != null) {
1072 out.attribute(null, ATTR_ORIGACTIVITY, origActivity.flattenToShortString());
1073 }
Dianne Hackborn79228822014-09-16 11:11:23 -07001074 // Write affinity, and root affinity if it is different from affinity.
1075 // We use the special string "@" for a null root affinity, so we can identify
1076 // later whether we were given a root affinity or should just make it the
1077 // same as the affinity.
Craig Mautner21d24a22014-04-23 11:45:37 -07001078 if (affinity != null) {
1079 out.attribute(null, ATTR_AFFINITY, affinity);
Dianne Hackborn79228822014-09-16 11:11:23 -07001080 if (!affinity.equals(rootAffinity)) {
1081 out.attribute(null, ATTR_ROOT_AFFINITY, rootAffinity != null ? rootAffinity : "@");
1082 }
1083 } else if (rootAffinity != null) {
1084 out.attribute(null, ATTR_ROOT_AFFINITY, rootAffinity != null ? rootAffinity : "@");
Craig Mautner21d24a22014-04-23 11:45:37 -07001085 }
1086 out.attribute(null, ATTR_ROOTHASRESET, String.valueOf(rootWasReset));
Dianne Hackborn13420f22014-07-18 15:43:56 -07001087 out.attribute(null, ATTR_AUTOREMOVERECENTS, String.valueOf(autoRemoveRecents));
Craig Mautner21d24a22014-04-23 11:45:37 -07001088 out.attribute(null, ATTR_ASKEDCOMPATMODE, String.valueOf(askedCompatMode));
1089 out.attribute(null, ATTR_USERID, String.valueOf(userId));
Wale Ogunwalef80170f2016-02-04 15:12:29 -08001090 out.attribute(null, ATTR_USER_SETUP_COMPLETE, String.valueOf(mUserSetupComplete));
Dianne Hackborn885fbe52014-08-23 15:23:58 -07001091 out.attribute(null, ATTR_EFFECTIVE_UID, String.valueOf(effectiveUid));
Craig Mautner21d24a22014-04-23 11:45:37 -07001092 out.attribute(null, ATTR_TASKTYPE, String.valueOf(taskType));
Winson Chungffa2ec62014-07-03 15:54:42 -07001093 out.attribute(null, ATTR_FIRSTACTIVETIME, String.valueOf(firstActiveTime));
Winson Chungf1fbd772014-06-24 18:06:58 -07001094 out.attribute(null, ATTR_LASTACTIVETIME, String.valueOf(lastActiveTime));
Craig Mautner21d24a22014-04-23 11:45:37 -07001095 out.attribute(null, ATTR_LASTTIMEMOVED, String.valueOf(mLastTimeMoved));
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001096 out.attribute(null, ATTR_NEVERRELINQUISH, String.valueOf(mNeverRelinquishIdentity));
Craig Mautner21d24a22014-04-23 11:45:37 -07001097 if (lastDescription != null) {
1098 out.attribute(null, ATTR_LASTDESCRIPTION, lastDescription.toString());
1099 }
Winson Chung2cb86c72014-06-25 12:03:30 -07001100 if (lastTaskDescription != null) {
Craig Mautner648f69b2014-09-18 14:16:26 -07001101 lastTaskDescription.saveToXml(out);
Winson Chung2cb86c72014-06-25 12:03:30 -07001102 }
Winsonc809cbb2015-11-02 12:06:15 -08001103 mLastThumbnailInfo.saveToXml(out);
Winson Chungec396d62014-08-06 17:08:00 -07001104 out.attribute(null, ATTR_TASK_AFFILIATION_COLOR, String.valueOf(mAffiliatedTaskColor));
Craig Mautnera228ae92014-07-09 05:44:55 -07001105 out.attribute(null, ATTR_TASK_AFFILIATION, String.valueOf(mAffiliatedTaskId));
1106 out.attribute(null, ATTR_PREV_AFFILIATION, String.valueOf(mPrevAffiliateTaskId));
1107 out.attribute(null, ATTR_NEXT_AFFILIATION, String.valueOf(mNextAffiliateTaskId));
Craig Mautnerdc00cbe2014-07-20 17:48:47 -07001108 out.attribute(null, ATTR_CALLING_UID, String.valueOf(mCallingUid));
1109 out.attribute(null, ATTR_CALLING_PACKAGE, mCallingPackage == null ? "" : mCallingPackage);
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001110 out.attribute(null, ATTR_RESIZE_MODE, String.valueOf(mResizeMode));
Wale Ogunwalea7afaa22015-05-01 20:39:18 -07001111 out.attribute(null, ATTR_PRIVILEGED, String.valueOf(mPrivileged));
Wale Ogunwale706ed792015-08-02 10:29:44 -07001112 if (mLastNonFullscreenBounds != null) {
1113 out.attribute(
1114 null, ATTR_NON_FULLSCREEN_BOUNDS, mLastNonFullscreenBounds.flattenToString());
Wale Ogunwalee4a0c572015-06-30 08:40:31 -07001115 }
Winson Chung2cb86c72014-06-25 12:03:30 -07001116
Craig Mautner21d24a22014-04-23 11:45:37 -07001117 if (affinityIntent != null) {
1118 out.startTag(null, TAG_AFFINITYINTENT);
1119 affinityIntent.saveToXml(out);
1120 out.endTag(null, TAG_AFFINITYINTENT);
1121 }
1122
1123 out.startTag(null, TAG_INTENT);
1124 intent.saveToXml(out);
1125 out.endTag(null, TAG_INTENT);
1126
1127 final ArrayList<ActivityRecord> activities = mActivities;
1128 final int numActivities = activities.size();
1129 for (int activityNdx = 0; activityNdx < numActivities; ++activityNdx) {
1130 final ActivityRecord r = activities.get(activityNdx);
Craig Mautner43e52ed2014-06-16 17:18:52 -07001131 if (r.info.persistableMode == ActivityInfo.PERSIST_ROOT_ONLY || !r.isPersistable() ||
Wale Ogunwale843bfb92015-03-27 11:06:48 -07001132 ((r.intent.getFlags() & FLAG_ACTIVITY_NEW_DOCUMENT
1133 | FLAG_ACTIVITY_RETAIN_IN_RECENTS) == FLAG_ACTIVITY_NEW_DOCUMENT) &&
Craig Mautner43e52ed2014-06-16 17:18:52 -07001134 activityNdx > 0) {
Craig Mautnerf357c0c2014-06-09 09:23:27 -07001135 // Stop at first non-persistable or first break in task (CLEAR_WHEN_TASK_RESET).
Craig Mautner21d24a22014-04-23 11:45:37 -07001136 break;
1137 }
1138 out.startTag(null, TAG_ACTIVITY);
1139 r.saveToXml(out);
1140 out.endTag(null, TAG_ACTIVITY);
1141 }
Craig Mautner21d24a22014-04-23 11:45:37 -07001142 }
1143
1144 static TaskRecord restoreFromXml(XmlPullParser in, ActivityStackSupervisor stackSupervisor)
1145 throws IOException, XmlPullParserException {
1146 Intent intent = null;
1147 Intent affinityIntent = null;
Winsonc809cbb2015-11-02 12:06:15 -08001148 ArrayList<ActivityRecord> activities = new ArrayList<>();
Craig Mautner21d24a22014-04-23 11:45:37 -07001149 ComponentName realActivity = null;
Andrei Stingaceanu4ccec532016-01-13 12:10:21 +00001150 boolean realActivitySuspended = false;
Craig Mautner21d24a22014-04-23 11:45:37 -07001151 ComponentName origActivity = null;
1152 String affinity = null;
Dianne Hackborn79228822014-09-16 11:11:23 -07001153 String rootAffinity = null;
1154 boolean hasRootAffinity = false;
Craig Mautner21d24a22014-04-23 11:45:37 -07001155 boolean rootHasReset = false;
Dianne Hackborn13420f22014-07-18 15:43:56 -07001156 boolean autoRemoveRecents = false;
Craig Mautner21d24a22014-04-23 11:45:37 -07001157 boolean askedCompatMode = false;
1158 int taskType = ActivityRecord.APPLICATION_ACTIVITY_TYPE;
Craig Mautner21d24a22014-04-23 11:45:37 -07001159 int userId = 0;
Wale Ogunwalef80170f2016-02-04 15:12:29 -08001160 boolean userSetupComplete = true;
Dianne Hackborn885fbe52014-08-23 15:23:58 -07001161 int effectiveUid = -1;
Craig Mautner21d24a22014-04-23 11:45:37 -07001162 String lastDescription = null;
Winson Chungffa2ec62014-07-03 15:54:42 -07001163 long firstActiveTime = -1;
Winson Chung2cb86c72014-06-25 12:03:30 -07001164 long lastActiveTime = -1;
Craig Mautner21d24a22014-04-23 11:45:37 -07001165 long lastTimeOnTop = 0;
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001166 boolean neverRelinquishIdentity = true;
Stefan Kuhnee88d1e52015-05-18 10:33:45 -07001167 int taskId = INVALID_TASK_ID;
Craig Mautner21d24a22014-04-23 11:45:37 -07001168 final int outerDepth = in.getDepth();
Craig Mautner648f69b2014-09-18 14:16:26 -07001169 TaskDescription taskDescription = new TaskDescription();
Winsonc809cbb2015-11-02 12:06:15 -08001170 TaskThumbnailInfo thumbnailInfo = new TaskThumbnailInfo();
Wale Ogunwale18795a22014-12-03 11:38:33 -08001171 int taskAffiliation = INVALID_TASK_ID;
Winson Chungec396d62014-08-06 17:08:00 -07001172 int taskAffiliationColor = 0;
Wale Ogunwale18795a22014-12-03 11:38:33 -08001173 int prevTaskId = INVALID_TASK_ID;
1174 int nextTaskId = INVALID_TASK_ID;
Craig Mautnerdc00cbe2014-07-20 17:48:47 -07001175 int callingUid = -1;
1176 String callingPackage = "";
Wale Ogunwaled829d362016-02-10 19:24:49 -08001177 int resizeMode = RESIZE_MODE_FORCE_RESIZEABLE;
Wale Ogunwalea7afaa22015-05-01 20:39:18 -07001178 boolean privileged = false;
Wale Ogunwalee4a0c572015-06-30 08:40:31 -07001179 Rect bounds = null;
Craig Mautner21d24a22014-04-23 11:45:37 -07001180
1181 for (int attrNdx = in.getAttributeCount() - 1; attrNdx >= 0; --attrNdx) {
1182 final String attrName = in.getAttributeName(attrNdx);
1183 final String attrValue = in.getAttributeValue(attrNdx);
Stefan Kuhnee88d1e52015-05-18 10:33:45 -07001184 if (TaskPersister.DEBUG) Slog.d(TaskPersister.TAG, "TaskRecord: attribute name=" +
1185 attrName + " value=" + attrValue);
Craig Mautner21d24a22014-04-23 11:45:37 -07001186 if (ATTR_TASKID.equals(attrName)) {
Wale Ogunwale18795a22014-12-03 11:38:33 -08001187 if (taskId == INVALID_TASK_ID) taskId = Integer.valueOf(attrValue);
Craig Mautner21d24a22014-04-23 11:45:37 -07001188 } else if (ATTR_REALACTIVITY.equals(attrName)) {
1189 realActivity = ComponentName.unflattenFromString(attrValue);
Andrei Stingaceanu4ccec532016-01-13 12:10:21 +00001190 } else if (ATTR_REALACTIVITY_SUSPENDED.equals(attrName)) {
1191 realActivitySuspended = Boolean.valueOf(attrValue);
Craig Mautner21d24a22014-04-23 11:45:37 -07001192 } else if (ATTR_ORIGACTIVITY.equals(attrName)) {
1193 origActivity = ComponentName.unflattenFromString(attrValue);
1194 } else if (ATTR_AFFINITY.equals(attrName)) {
1195 affinity = attrValue;
Dianne Hackborn79228822014-09-16 11:11:23 -07001196 } else if (ATTR_ROOT_AFFINITY.equals(attrName)) {
1197 rootAffinity = attrValue;
1198 hasRootAffinity = true;
Craig Mautner21d24a22014-04-23 11:45:37 -07001199 } else if (ATTR_ROOTHASRESET.equals(attrName)) {
1200 rootHasReset = Boolean.valueOf(attrValue);
Dianne Hackborn13420f22014-07-18 15:43:56 -07001201 } else if (ATTR_AUTOREMOVERECENTS.equals(attrName)) {
1202 autoRemoveRecents = Boolean.valueOf(attrValue);
Craig Mautner21d24a22014-04-23 11:45:37 -07001203 } else if (ATTR_ASKEDCOMPATMODE.equals(attrName)) {
1204 askedCompatMode = Boolean.valueOf(attrValue);
1205 } else if (ATTR_USERID.equals(attrName)) {
1206 userId = Integer.valueOf(attrValue);
Wale Ogunwalef80170f2016-02-04 15:12:29 -08001207 } else if (ATTR_USER_SETUP_COMPLETE.equals(attrName)) {
1208 userSetupComplete = Boolean.valueOf(attrValue);
Dianne Hackborn885fbe52014-08-23 15:23:58 -07001209 } else if (ATTR_EFFECTIVE_UID.equals(attrName)) {
1210 effectiveUid = Integer.valueOf(attrValue);
Craig Mautner21d24a22014-04-23 11:45:37 -07001211 } else if (ATTR_TASKTYPE.equals(attrName)) {
1212 taskType = Integer.valueOf(attrValue);
Winson Chungffa2ec62014-07-03 15:54:42 -07001213 } else if (ATTR_FIRSTACTIVETIME.equals(attrName)) {
1214 firstActiveTime = Long.valueOf(attrValue);
Winson Chungf1fbd772014-06-24 18:06:58 -07001215 } else if (ATTR_LASTACTIVETIME.equals(attrName)) {
1216 lastActiveTime = Long.valueOf(attrValue);
Craig Mautner21d24a22014-04-23 11:45:37 -07001217 } else if (ATTR_LASTDESCRIPTION.equals(attrName)) {
1218 lastDescription = attrValue;
1219 } else if (ATTR_LASTTIMEMOVED.equals(attrName)) {
1220 lastTimeOnTop = Long.valueOf(attrValue);
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001221 } else if (ATTR_NEVERRELINQUISH.equals(attrName)) {
1222 neverRelinquishIdentity = Boolean.valueOf(attrValue);
Winsonc809cbb2015-11-02 12:06:15 -08001223 } else if (attrName.startsWith(TaskThumbnailInfo.ATTR_TASK_THUMBNAILINFO_PREFIX)) {
1224 thumbnailInfo.restoreFromXml(attrName, attrValue);
Craig Mautner648f69b2014-09-18 14:16:26 -07001225 } else if (attrName.startsWith(TaskDescription.ATTR_TASKDESCRIPTION_PREFIX)) {
1226 taskDescription.restoreFromXml(attrName, attrValue);
Craig Mautnera228ae92014-07-09 05:44:55 -07001227 } else if (ATTR_TASK_AFFILIATION.equals(attrName)) {
1228 taskAffiliation = Integer.valueOf(attrValue);
1229 } else if (ATTR_PREV_AFFILIATION.equals(attrName)) {
1230 prevTaskId = Integer.valueOf(attrValue);
1231 } else if (ATTR_NEXT_AFFILIATION.equals(attrName)) {
1232 nextTaskId = Integer.valueOf(attrValue);
Winson Chungec396d62014-08-06 17:08:00 -07001233 } else if (ATTR_TASK_AFFILIATION_COLOR.equals(attrName)) {
1234 taskAffiliationColor = Integer.valueOf(attrValue);
Craig Mautnerdc00cbe2014-07-20 17:48:47 -07001235 } else if (ATTR_CALLING_UID.equals(attrName)) {
1236 callingUid = Integer.valueOf(attrValue);
1237 } else if (ATTR_CALLING_PACKAGE.equals(attrName)) {
1238 callingPackage = attrValue;
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001239 } else if (ATTR_RESIZE_MODE.equals(attrName)) {
1240 resizeMode = Integer.valueOf(attrValue);
Wale Ogunwaled829d362016-02-10 19:24:49 -08001241 resizeMode = (resizeMode == RESIZE_MODE_CROP_WINDOWS)
1242 ? RESIZE_MODE_FORCE_RESIZEABLE : resizeMode;
Wale Ogunwalea7afaa22015-05-01 20:39:18 -07001243 } else if (ATTR_PRIVILEGED.equals(attrName)) {
1244 privileged = Boolean.valueOf(attrValue);
Wale Ogunwale706ed792015-08-02 10:29:44 -07001245 } else if (ATTR_NON_FULLSCREEN_BOUNDS.equals(attrName)) {
Wale Ogunwalee4a0c572015-06-30 08:40:31 -07001246 bounds = Rect.unflattenFromString(attrValue);
Craig Mautner21d24a22014-04-23 11:45:37 -07001247 } else {
1248 Slog.w(TAG, "TaskRecord: Unknown attribute=" + attrName);
1249 }
1250 }
1251
1252 int event;
1253 while (((event = in.next()) != XmlPullParser.END_DOCUMENT) &&
Ben Kwa8814cf42015-07-08 10:54:56 -07001254 (event != XmlPullParser.END_TAG || in.getDepth() >= outerDepth)) {
Craig Mautner21d24a22014-04-23 11:45:37 -07001255 if (event == XmlPullParser.START_TAG) {
1256 final String name = in.getName();
Stefan Kuhnee88d1e52015-05-18 10:33:45 -07001257 if (TaskPersister.DEBUG) Slog.d(TaskPersister.TAG, "TaskRecord: START_TAG name=" +
1258 name);
Craig Mautner21d24a22014-04-23 11:45:37 -07001259 if (TAG_AFFINITYINTENT.equals(name)) {
1260 affinityIntent = Intent.restoreFromXml(in);
1261 } else if (TAG_INTENT.equals(name)) {
1262 intent = Intent.restoreFromXml(in);
1263 } else if (TAG_ACTIVITY.equals(name)) {
Wale Ogunwale18795a22014-12-03 11:38:33 -08001264 ActivityRecord activity = ActivityRecord.restoreFromXml(in, stackSupervisor);
Stefan Kuhnee88d1e52015-05-18 10:33:45 -07001265 if (TaskPersister.DEBUG) Slog.d(TaskPersister.TAG, "TaskRecord: activity=" +
1266 activity);
Craig Mautner21d24a22014-04-23 11:45:37 -07001267 if (activity != null) {
1268 activities.add(activity);
1269 }
1270 } else {
1271 Slog.e(TAG, "restoreTask: Unexpected name=" + name);
1272 XmlUtils.skipCurrentTag(in);
1273 }
1274 }
1275 }
Dianne Hackborn79228822014-09-16 11:11:23 -07001276 if (!hasRootAffinity) {
1277 rootAffinity = affinity;
1278 } else if ("@".equals(rootAffinity)) {
1279 rootAffinity = null;
1280 }
Dianne Hackborn885fbe52014-08-23 15:23:58 -07001281 if (effectiveUid <= 0) {
1282 Intent checkIntent = intent != null ? intent : affinityIntent;
1283 effectiveUid = 0;
1284 if (checkIntent != null) {
1285 IPackageManager pm = AppGlobals.getPackageManager();
1286 try {
1287 ApplicationInfo ai = pm.getApplicationInfo(
1288 checkIntent.getComponent().getPackageName(),
1289 PackageManager.GET_UNINSTALLED_PACKAGES
1290 | PackageManager.GET_DISABLED_COMPONENTS, userId);
1291 if (ai != null) {
1292 effectiveUid = ai.uid;
1293 }
1294 } catch (RemoteException e) {
1295 }
1296 }
1297 Slog.w(TAG, "Updating task #" + taskId + " for " + checkIntent
1298 + ": effectiveUid=" + effectiveUid);
1299 }
1300
Craig Mautner21d24a22014-04-23 11:45:37 -07001301 final TaskRecord task = new TaskRecord(stackSupervisor.mService, taskId, intent,
Dianne Hackborn79228822014-09-16 11:11:23 -07001302 affinityIntent, affinity, rootAffinity, realActivity, origActivity, rootHasReset,
Dianne Hackborn885fbe52014-08-23 15:23:58 -07001303 autoRemoveRecents, askedCompatMode, taskType, userId, effectiveUid, lastDescription,
1304 activities, firstActiveTime, lastActiveTime, lastTimeOnTop, neverRelinquishIdentity,
Winsonc809cbb2015-11-02 12:06:15 -08001305 taskDescription, thumbnailInfo, taskAffiliation, prevTaskId, nextTaskId,
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001306 taskAffiliationColor, callingUid, callingPackage, resizeMode, privileged,
Wale Ogunwalef80170f2016-02-04 15:12:29 -08001307 realActivitySuspended, userSetupComplete);
Filip Gruszczynskiaff7f132015-09-02 17:21:21 -07001308 task.updateOverrideConfiguration(bounds);
Craig Mautner21d24a22014-04-23 11:45:37 -07001309
1310 for (int activityNdx = activities.size() - 1; activityNdx >=0; --activityNdx) {
Craig Mautnerc0ffce52014-07-01 12:38:52 -07001311 activities.get(activityNdx).task = task;
Craig Mautner21d24a22014-04-23 11:45:37 -07001312 }
1313
Wale Ogunwaleee006da2015-03-30 14:49:25 -07001314 if (DEBUG_RECENTS) Slog.d(TAG_RECENTS, "Restored task=" + task);
Craig Mautner21d24a22014-04-23 11:45:37 -07001315 return task;
1316 }
1317
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001318 private void adjustForMinimalTaskDimensions(Rect bounds) {
1319 if (bounds == null) {
1320 return;
1321 }
1322 final int minimalSize = mMinimalSize == -1
1323 ? mService.mStackSupervisor.mDefaultMinimalSizeOfResizeableTask : mMinimalSize;
1324 final boolean adjustWidth = minimalSize > bounds.width();
1325 final boolean adjustHeight = minimalSize > bounds.height();
1326 if (!(adjustWidth || adjustHeight)) {
1327 return;
1328 }
1329
1330 if (adjustWidth) {
1331 if (mBounds != null && bounds.right == mBounds.right) {
1332 bounds.left = bounds.right - minimalSize;
1333 } else {
1334 // Either left bounds match, or neither match, or the previous bounds were
1335 // fullscreen and we default to keeping left.
1336 bounds.right = bounds.left + minimalSize;
1337 }
1338 }
1339 if (adjustHeight) {
1340 if (mBounds != null && bounds.bottom == mBounds.bottom) {
1341 bounds.top = bounds.bottom - minimalSize;
1342 } else {
1343 // Either top bounds match, or neither match, or the previous bounds were
1344 // fullscreen and we default to keeping top.
1345 bounds.bottom = bounds.top + minimalSize;
1346 }
1347 }
1348 }
1349
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001350 /**
1351 * Update task's override configuration based on the bounds.
Jorim Jaggi0a932142016-02-01 17:42:25 -08001352 * @param bounds The bounds of the task.
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001353 * @return Update configuration or null if there is no change.
1354 */
Filip Gruszczynskiaff7f132015-09-02 17:21:21 -07001355 Configuration updateOverrideConfiguration(Rect bounds) {
Jorim Jaggi0a932142016-02-01 17:42:25 -08001356 return updateOverrideConfiguration(bounds, null /* insetBounds */);
1357 }
1358
1359 /**
1360 * Update task's override configuration based on the bounds.
1361 * @param bounds The bounds of the task.
1362 * @param insetBounds The bounds used to calculate the system insets, which is used here to
1363 * subtract the navigation bar/status bar size from the screen size reported
1364 * to the application. See {@link IActivityManager#resizeDockedStack}.
1365 * @return Update configuration or null if there is no change.
1366 */
1367 Configuration updateOverrideConfiguration(Rect bounds, @Nullable Rect insetBounds) {
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001368 if (Objects.equals(mBounds, bounds)) {
1369 return null;
1370 }
Wale Ogunwale5f986092015-12-04 15:35:38 -08001371 final Configuration oldConfig = mOverrideConfig;
1372 final boolean oldFullscreen = mFullscreen;
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001373
1374 mFullscreen = bounds == null;
Wale Ogunwale706ed792015-08-02 10:29:44 -07001375 if (mFullscreen) {
Wale Ogunwale3797c222015-10-27 14:21:58 -07001376 if (mBounds != null && StackId.persistTaskBounds(stack.mStackId)) {
Wale Ogunwale706ed792015-08-02 10:29:44 -07001377 mLastNonFullscreenBounds = mBounds;
1378 }
1379 mBounds = null;
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001380 mOverrideConfig = Configuration.EMPTY;
Wale Ogunwale706ed792015-08-02 10:29:44 -07001381 } else {
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001382 mTmpRect.set(bounds);
1383 adjustForMinimalTaskDimensions(mTmpRect);
Chong Zhangb15758a2015-11-17 12:12:03 -08001384 if (mBounds == null) {
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001385 mBounds = new Rect(mTmpRect);
Chong Zhangb15758a2015-11-17 12:12:03 -08001386 } else {
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001387 mBounds.set(mTmpRect);
Chong Zhangb15758a2015-11-17 12:12:03 -08001388 }
Wale Ogunwale3797c222015-10-27 14:21:58 -07001389 if (stack == null || StackId.persistTaskBounds(stack.mStackId)) {
Wale Ogunwaleb34a7ad2015-08-14 11:05:30 -07001390 mLastNonFullscreenBounds = mBounds;
1391 }
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001392 mOverrideConfig = calculateOverrideConfig(mTmpRect, insetBounds);
Wale Ogunwale706ed792015-08-02 10:29:44 -07001393 }
Wale Ogunwale5f986092015-12-04 15:35:38 -08001394
1395 if (mFullscreen != oldFullscreen) {
Wale Ogunwale22e25262016-02-01 10:32:02 -08001396 mService.mStackSupervisor.scheduleReportMultiWindowChanged(this);
Wale Ogunwale5f986092015-12-04 15:35:38 -08001397 }
1398
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001399 return !mOverrideConfig.equals(oldConfig) ? mOverrideConfig : null;
Wale Ogunwalee4a0c572015-06-30 08:40:31 -07001400 }
1401
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001402 private void subtractNonDecorInsets(Rect inOutBounds, Rect inInsetBounds) {
1403 mTmpRect2.set(inInsetBounds);
1404 mService.mWindowManager.subtractNonDecorInsets(mTmpRect2);
1405 int leftInset = mTmpRect2.left - inInsetBounds.left;
1406 int topInset = mTmpRect2.top - inInsetBounds.top;
1407 int rightInset = inInsetBounds.right - mTmpRect2.right;
1408 int bottomInset = inInsetBounds.bottom - mTmpRect2.bottom;
1409 inOutBounds.inset(leftInset, topInset, rightInset, bottomInset);
1410 }
1411
Jorim Jaggi0a932142016-02-01 17:42:25 -08001412 private void subtractStableInsets(Rect inOutBounds, Rect inInsetBounds) {
1413 mTmpRect2.set(inInsetBounds);
1414 mService.mWindowManager.subtractStableInsets(mTmpRect2);
1415 int leftInset = mTmpRect2.left - inInsetBounds.left;
1416 int topInset = mTmpRect2.top - inInsetBounds.top;
1417 int rightInset = inInsetBounds.right - mTmpRect2.right;
1418 int bottomInset = inInsetBounds.bottom - mTmpRect2.bottom;
1419 inOutBounds.inset(leftInset, topInset, rightInset, bottomInset);
1420 }
1421
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001422 private Configuration calculateOverrideConfig(Rect bounds, Rect insetBounds) {
1423 mTmpNonDecorBounds.set(bounds);
1424 mTmpStableBounds.set(bounds);
1425 subtractNonDecorInsets(
1426 mTmpNonDecorBounds, insetBounds != null ? insetBounds : bounds);
1427 subtractStableInsets(
1428 mTmpStableBounds, insetBounds != null ? insetBounds : bounds);
1429
1430 // For calculating screenWidthDp, screenWidthDp, we use the stable inset screen area,
1431 // i.e. the screen area without the system bars.
Jorim Jaggia95ca8d2016-01-15 22:54:46 -08001432 final Configuration serviceConfig = mService.mConfiguration;
1433 final Configuration config = new Configuration(Configuration.EMPTY);
1434 // TODO(multidisplay): Update Dp to that of display stack is on.
1435 final float density = serviceConfig.densityDpi * DisplayMetrics.DENSITY_DEFAULT_SCALE;
1436 config.screenWidthDp =
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001437 Math.min((int)(mTmpStableBounds.width() / density), serviceConfig.screenWidthDp);
Jorim Jaggia95ca8d2016-01-15 22:54:46 -08001438 config.screenHeightDp =
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001439 Math.min((int)(mTmpStableBounds.height() / density), serviceConfig.screenHeightDp);
1440 config.smallestScreenWidthDp = Math.min(config.screenWidthDp, config.screenHeightDp);
1441
1442 // TODO: Orientation?
Jorim Jaggia95ca8d2016-01-15 22:54:46 -08001443 config.orientation = (config.screenWidthDp <= config.screenHeightDp)
1444 ? Configuration.ORIENTATION_PORTRAIT
1445 : Configuration.ORIENTATION_LANDSCAPE;
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001446
1447 // For calculating screen layout, we need to use the non-decor inset screen area for the
1448 // calculation for compatibility reasons, i.e. screen area without system bars that could
1449 // never go away in Honeycomb.
1450 final int compatScreenWidthDp = (int)(mTmpNonDecorBounds.width() / density);
1451 final int compatScreenHeightDp = (int)(mTmpNonDecorBounds.height() / density);
Jorim Jaggia95ca8d2016-01-15 22:54:46 -08001452 final int sl = Configuration.resetScreenLayout(serviceConfig.screenLayout);
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001453 final int longSize = Math.max(compatScreenHeightDp, compatScreenWidthDp);
1454 final int shortSize = Math.min(compatScreenHeightDp, compatScreenWidthDp);
Jorim Jaggi0a932142016-02-01 17:42:25 -08001455 config.screenLayout = Configuration.reduceScreenLayout(sl, longSize, shortSize);
Jorim Jaggia95ca8d2016-01-15 22:54:46 -08001456 return config;
1457 }
1458
1459 /**
1460 * Using the existing configuration {@param config}, creates a new task override config such
1461 * that all the fields that are usually set in an override config are set to the ones in
1462 * {@param config}.
1463 */
1464 Configuration extractOverrideConfig(Configuration config) {
1465 final Configuration extracted = new Configuration(Configuration.EMPTY);
1466 extracted.screenWidthDp = config.screenWidthDp;
1467 extracted.screenHeightDp = config.screenHeightDp;
1468 extracted.smallestScreenWidthDp = config.smallestScreenWidthDp;
1469 extracted.orientation = config.orientation;
1470 extracted.screenLayout = config.screenLayout;
1471 return extracted;
1472 }
1473
Filip Gruszczynskidce2d162016-01-12 15:40:13 -08001474 Rect updateOverrideConfigurationFromLaunchBounds() {
1475 final Rect bounds = validateBounds(getLaunchBounds());
1476 updateOverrideConfiguration(bounds);
1477 return bounds;
1478 }
1479
1480 static Rect validateBounds(Rect bounds) {
1481 if (bounds != null && bounds.isEmpty()) {
1482 Slog.wtf(TAG, "Received strange task bounds: " + bounds, new Throwable());
1483 return null;
1484 }
1485 return bounds;
1486 }
1487
Wale Ogunwale935e5022015-11-10 12:36:10 -08001488 /** Updates the task's bounds and override configuration to match what is expected for the
1489 * input stack. */
1490 void updateOverrideConfigurationForStack(ActivityStack inStack) {
1491 if (stack != null && stack == inStack) {
1492 return;
1493 }
1494
1495 if (inStack.mStackId == FREEFORM_WORKSPACE_STACK_ID) {
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001496 if (!isResizeable()) {
Wale Ogunwale935e5022015-11-10 12:36:10 -08001497 throw new IllegalArgumentException("Can not position non-resizeable task="
1498 + this + " in stack=" + inStack);
1499 }
1500 if (mBounds != null) {
1501 return;
1502 }
1503 if (mLastNonFullscreenBounds != null) {
1504 updateOverrideConfiguration(mLastNonFullscreenBounds);
1505 } else {
1506 inStack.layoutTaskInStack(this, null);
1507 }
1508 } else {
1509 updateOverrideConfiguration(inStack.mBounds);
1510 }
1511 }
1512
Chong Zhang0fa656b2015-08-31 15:17:21 -07001513 /**
1514 * Returns the correct stack to use based on task type and currently set bounds,
1515 * regardless of the focused stack and current stack association of the task.
1516 * The task will be moved (and stack focus changed) later if necessary.
1517 */
1518 int getLaunchStackId() {
1519 if (!isApplicationTask()) {
Wale Ogunwale706ed792015-08-02 10:29:44 -07001520 return HOME_STACK_ID;
1521 }
Chong Zhang0fa656b2015-08-31 15:17:21 -07001522 if (mBounds != null) {
Wale Ogunwale706ed792015-08-02 10:29:44 -07001523 return FREEFORM_WORKSPACE_STACK_ID;
1524 }
1525 return FULLSCREEN_WORKSPACE_STACK_ID;
1526 }
1527
1528 /** Returns the bounds that should be used to launch this task. */
1529 Rect getLaunchBounds() {
Chong Zhang75b37202015-12-04 14:16:36 -08001530 // If we're over lockscreen, forget about stack bounds and use fullscreen.
1531 if (mService.mLockScreenShown == LOCK_SCREEN_SHOWN) {
1532 return null;
1533 }
1534
Chong Zhang7d5f5102016-01-13 10:29:24 -08001535 if (stack == null) {
1536 return null;
1537 }
1538
1539 final int stackId = stack.mStackId;
1540 if (stackId == HOME_STACK_ID
1541 || stackId == FULLSCREEN_WORKSPACE_STACK_ID
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001542 || (stackId == DOCKED_STACK_ID && !isResizeable())) {
1543 return isResizeable() ? stack.mBounds : null;
Wale Ogunwale3797c222015-10-27 14:21:58 -07001544 } else if (!StackId.persistTaskBounds(stackId)) {
Wale Ogunwaleb34a7ad2015-08-14 11:05:30 -07001545 return stack.mBounds;
Wale Ogunwale706ed792015-08-02 10:29:44 -07001546 }
1547 return mLastNonFullscreenBounds;
1548 }
1549
Wale Ogunwale39381972015-12-17 17:15:29 -08001550 boolean canMatchRootAffinity() {
1551 // We don't allow root affinity matching on the pinned stack as no other task should
1552 // be launching in it based on affinity.
1553 return rootAffinity != null && (stack == null || stack.mStackId != PINNED_STACK_ID);
1554 }
1555
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001556 void dump(PrintWriter pw, String prefix) {
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001557 pw.print(prefix); pw.print("userId="); pw.print(userId);
Dianne Hackborn885fbe52014-08-23 15:23:58 -07001558 pw.print(" effectiveUid="); UserHandle.formatUid(pw, effectiveUid);
1559 pw.print(" mCallingUid="); UserHandle.formatUid(pw, mCallingUid);
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001560 pw.print(" mCallingPackage="); pw.println(mCallingPackage);
Dianne Hackborn79228822014-09-16 11:11:23 -07001561 if (affinity != null || rootAffinity != null) {
1562 pw.print(prefix); pw.print("affinity="); pw.print(affinity);
1563 if (affinity == null || !affinity.equals(rootAffinity)) {
1564 pw.print(" root="); pw.println(rootAffinity);
1565 } else {
1566 pw.println();
1567 }
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001568 }
Dianne Hackborn91097de2014-04-04 18:02:06 -07001569 if (voiceSession != null || voiceInteractor != null) {
1570 pw.print(prefix); pw.print("VOICE: session=0x");
1571 pw.print(Integer.toHexString(System.identityHashCode(voiceSession)));
1572 pw.print(" interactor=0x");
1573 pw.println(Integer.toHexString(System.identityHashCode(voiceInteractor)));
1574 }
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001575 if (intent != null) {
1576 StringBuilder sb = new StringBuilder(128);
1577 sb.append(prefix); sb.append("intent={");
Dianne Hackborn21c241e2012-03-08 13:57:23 -08001578 intent.toShortString(sb, false, true, false, true);
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001579 sb.append('}');
1580 pw.println(sb.toString());
1581 }
1582 if (affinityIntent != null) {
1583 StringBuilder sb = new StringBuilder(128);
1584 sb.append(prefix); sb.append("affinityIntent={");
Dianne Hackborn21c241e2012-03-08 13:57:23 -08001585 affinityIntent.toShortString(sb, false, true, false, true);
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001586 sb.append('}');
1587 pw.println(sb.toString());
1588 }
1589 if (origActivity != null) {
1590 pw.print(prefix); pw.print("origActivity=");
1591 pw.println(origActivity.flattenToShortString());
1592 }
1593 if (realActivity != null) {
1594 pw.print(prefix); pw.print("realActivity=");
1595 pw.println(realActivity.flattenToShortString());
1596 }
Dianne Hackborn852975d2014-08-22 17:42:43 -07001597 if (autoRemoveRecents || isPersistable || taskType != 0 || mTaskToReturnTo != 0
1598 || numFullscreen != 0) {
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001599 pw.print(prefix); pw.print("autoRemoveRecents="); pw.print(autoRemoveRecents);
Dianne Hackborn852975d2014-08-22 17:42:43 -07001600 pw.print(" isPersistable="); pw.print(isPersistable);
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001601 pw.print(" numFullscreen="); pw.print(numFullscreen);
1602 pw.print(" taskType="); pw.print(taskType);
1603 pw.print(" mTaskToReturnTo="); pw.println(mTaskToReturnTo);
1604 }
Craig Mautner432f64e2015-05-20 14:59:57 -07001605 if (rootWasReset || mNeverRelinquishIdentity || mReuseTask
1606 || mLockTaskAuth != LOCK_TASK_AUTH_PINNABLE) {
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001607 pw.print(prefix); pw.print("rootWasReset="); pw.print(rootWasReset);
1608 pw.print(" mNeverRelinquishIdentity="); pw.print(mNeverRelinquishIdentity);
Craig Mautner432f64e2015-05-20 14:59:57 -07001609 pw.print(" mReuseTask="); pw.print(mReuseTask);
1610 pw.print(" mLockTaskAuth="); pw.println(lockTaskAuthToString());
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001611 }
Wale Ogunwale18795a22014-12-03 11:38:33 -08001612 if (mAffiliatedTaskId != taskId || mPrevAffiliateTaskId != INVALID_TASK_ID
1613 || mPrevAffiliate != null || mNextAffiliateTaskId != INVALID_TASK_ID
1614 || mNextAffiliate != null) {
Dianne Hackborn852975d2014-08-22 17:42:43 -07001615 pw.print(prefix); pw.print("affiliation="); pw.print(mAffiliatedTaskId);
1616 pw.print(" prevAffiliation="); pw.print(mPrevAffiliateTaskId);
1617 pw.print(" (");
1618 if (mPrevAffiliate == null) {
1619 pw.print("null");
1620 } else {
1621 pw.print(Integer.toHexString(System.identityHashCode(mPrevAffiliate)));
1622 }
1623 pw.print(") nextAffiliation="); pw.print(mNextAffiliateTaskId);
1624 pw.print(" (");
1625 if (mNextAffiliate == null) {
1626 pw.print("null");
1627 } else {
1628 pw.print(Integer.toHexString(System.identityHashCode(mNextAffiliate)));
1629 }
1630 pw.println(")");
1631 }
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001632 pw.print(prefix); pw.print("Activities="); pw.println(mActivities);
Dianne Hackborn852975d2014-08-22 17:42:43 -07001633 if (!askedCompatMode || !inRecents || !isAvailable) {
1634 pw.print(prefix); pw.print("askedCompatMode="); pw.print(askedCompatMode);
1635 pw.print(" inRecents="); pw.print(inRecents);
1636 pw.print(" isAvailable="); pw.println(isAvailable);
Dianne Hackborn36cd41f2011-05-25 21:00:46 -07001637 }
Craig Mautnerc0ffce52014-07-01 12:38:52 -07001638 pw.print(prefix); pw.print("lastThumbnail="); pw.print(mLastThumbnail);
Dianne Hackborn852975d2014-08-22 17:42:43 -07001639 pw.print(" lastThumbnailFile="); pw.println(mLastThumbnailFile);
1640 if (lastDescription != null) {
1641 pw.print(prefix); pw.print("lastDescription="); pw.println(lastDescription);
1642 }
Wale Ogunwale069fbe42015-07-29 11:38:01 -07001643 if (stack != null) {
1644 pw.print(prefix); pw.print("stackId="); pw.println(stack.mStackId);
1645 }
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001646 pw.print(prefix + "hasBeenVisible=" + hasBeenVisible);
1647 pw.print(" mResizeMode=" + ActivityInfo.resizeModeToString(mResizeMode));
1648 pw.print(" isResizeable=" + isResizeable());
1649 pw.print(" firstActiveTime=" + lastActiveTime);
1650 pw.print(" lastActiveTime=" + lastActiveTime);
1651 pw.println(" (inactive for " + (getInactiveDuration() / 1000) + "s)");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001652 }
1653
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001654 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001655 public String toString() {
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001656 StringBuilder sb = new StringBuilder(128);
Craig Mautnerde4ef022013-04-07 19:01:33 -07001657 if (stringName != null) {
1658 sb.append(stringName);
1659 sb.append(" U=");
1660 sb.append(userId);
Wale Ogunwale6c5eb1c2015-11-10 07:52:22 -08001661 sb.append(" StackId=");
1662 sb.append(stack != null ? stack.mStackId : INVALID_STACK_ID);
Craig Mautnerde4ef022013-04-07 19:01:33 -07001663 sb.append(" sz=");
1664 sb.append(mActivities.size());
1665 sb.append('}');
1666 return sb.toString();
1667 }
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001668 sb.append("TaskRecord{");
1669 sb.append(Integer.toHexString(System.identityHashCode(this)));
1670 sb.append(" #");
1671 sb.append(taskId);
1672 if (affinity != null) {
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001673 sb.append(" A=");
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001674 sb.append(affinity);
1675 } else if (intent != null) {
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001676 sb.append(" I=");
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001677 sb.append(intent.getComponent().flattenToShortString());
1678 } else if (affinityIntent != null) {
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001679 sb.append(" aI=");
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001680 sb.append(affinityIntent.getComponent().flattenToShortString());
1681 } else {
1682 sb.append(" ??");
1683 }
Craig Mautnerde4ef022013-04-07 19:01:33 -07001684 stringName = sb.toString();
1685 return toString();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001686 }
1687}