The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 1 | /* |
| 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 | |
| 17 | package com.android.server.am; |
| 18 | |
Jorim Jaggi | e2ad37f | 2018-01-22 22:41:22 +0100 | [diff] [blame] | 19 | import static android.app.ActivityManager.START_SUCCESS; |
Wale Ogunwale | e23149f | 2015-03-06 15:39:44 -0800 | [diff] [blame] | 20 | import static com.android.server.am.ActivityManagerDebugConfig.TAG_AM; |
| 21 | import static com.android.server.am.ActivityManagerDebugConfig.TAG_WITH_CLASS_NAME; |
| 22 | |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 23 | import android.app.ActivityManager; |
Jorim Jaggi | 4d8d32c | 2018-01-19 15:57:41 +0100 | [diff] [blame] | 24 | import android.app.ActivityOptions; |
Suchi Amalapurapu | 1ccac75 | 2009-06-12 10:09:58 -0700 | [diff] [blame] | 25 | import android.content.IIntentSender; |
| 26 | import android.content.IIntentReceiver; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 27 | import android.app.PendingIntent; |
| 28 | import android.content.Intent; |
| 29 | import android.os.Binder; |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 30 | import android.os.Bundle; |
Dianne Hackborn | bcbcaa7 | 2009-09-10 10:54:46 -0700 | [diff] [blame] | 31 | import android.os.IBinder; |
Dianne Hackborn | f66adfd | 2017-04-13 11:01:48 -0700 | [diff] [blame] | 32 | import android.os.RemoteCallbackList; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 33 | import android.os.RemoteException; |
Craig Mautner | 5f2bb4c | 2015-03-12 16:10:27 -0700 | [diff] [blame] | 34 | import android.os.TransactionTooLargeException; |
Dianne Hackborn | 50cdf7c3 | 2012-09-23 17:08:57 -0700 | [diff] [blame] | 35 | import android.os.UserHandle; |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 36 | import android.util.ArrayMap; |
Joe Onorato | 8a9b220 | 2010-02-26 18:56:32 -0800 | [diff] [blame] | 37 | import android.util.Slog; |
Dianne Hackborn | bc02a39 | 2016-06-02 17:15:08 -0700 | [diff] [blame] | 38 | import android.util.TimeUtils; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 39 | |
Dianne Hackborn | f66adfd | 2017-04-13 11:01:48 -0700 | [diff] [blame] | 40 | import com.android.internal.os.IResultReceiver; |
Craig Mautner | b916836 | 2015-02-26 20:40:19 -0800 | [diff] [blame] | 41 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 42 | import java.io.PrintWriter; |
| 43 | import java.lang.ref.WeakReference; |
Rubin Xu | f24d606 | 2016-07-20 17:34:50 +0100 | [diff] [blame] | 44 | import java.util.Objects; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 45 | |
Dianne Hackborn | be4e6aa | 2013-06-07 13:25:29 -0700 | [diff] [blame] | 46 | final class PendingIntentRecord extends IIntentSender.Stub { |
Wale Ogunwale | e23149f | 2015-03-06 15:39:44 -0800 | [diff] [blame] | 47 | private static final String TAG = TAG_WITH_CLASS_NAME ? "PendingIntentRecord" : TAG_AM; |
| 48 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 49 | final ActivityManagerService owner; |
| 50 | final Key key; |
| 51 | final int uid; |
| 52 | final WeakReference<PendingIntentRecord> ref; |
| 53 | boolean sent = false; |
| 54 | boolean canceled = false; |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 55 | private ArrayMap<IBinder, Long> whitelistDuration; |
Dianne Hackborn | f66adfd | 2017-04-13 11:01:48 -0700 | [diff] [blame] | 56 | private RemoteCallbackList<IResultReceiver> mCancelCallbacks; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 57 | |
Dianne Hackborn | 1d442e0 | 2009-04-20 18:14:05 -0700 | [diff] [blame] | 58 | String stringName; |
Dianne Hackborn | a1f1a3c | 2014-02-24 18:12:28 -0800 | [diff] [blame] | 59 | String lastTagPrefix; |
| 60 | String lastTag; |
Wale Ogunwale | e23149f | 2015-03-06 15:39:44 -0800 | [diff] [blame] | 61 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 62 | final static class Key { |
| 63 | final int type; |
| 64 | final String packageName; |
Dianne Hackborn | 01e4cfc | 2010-06-24 15:07:24 -0700 | [diff] [blame] | 65 | final ActivityRecord activity; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 66 | final String who; |
| 67 | final int requestCode; |
| 68 | final Intent requestIntent; |
| 69 | final String requestResolvedType; |
Jorim Jaggi | 4d8d32c | 2018-01-19 15:57:41 +0100 | [diff] [blame] | 70 | final SafeActivityOptions options; |
Dianne Hackborn | 621e17d | 2010-11-22 15:59:56 -0800 | [diff] [blame] | 71 | Intent[] allIntents; |
| 72 | String[] allResolvedTypes; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 73 | final int flags; |
| 74 | final int hashCode; |
Amith Yamasani | 4ea6069 | 2012-08-28 14:34:53 -0700 | [diff] [blame] | 75 | final int userId; |
Felipe Leme | a1b79bf | 2016-05-24 13:06:54 -0700 | [diff] [blame] | 76 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 77 | private static final int ODD_PRIME_NUMBER = 37; |
Felipe Leme | a1b79bf | 2016-05-24 13:06:54 -0700 | [diff] [blame] | 78 | |
Dianne Hackborn | 01e4cfc | 2010-06-24 15:07:24 -0700 | [diff] [blame] | 79 | Key(int _t, String _p, ActivityRecord _a, String _w, |
Jorim Jaggi | 4d8d32c | 2018-01-19 15:57:41 +0100 | [diff] [blame] | 80 | int _r, Intent[] _i, String[] _it, int _f, SafeActivityOptions _o, int _userId) { |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 81 | type = _t; |
| 82 | packageName = _p; |
| 83 | activity = _a; |
| 84 | who = _w; |
| 85 | requestCode = _r; |
Dianne Hackborn | 621e17d | 2010-11-22 15:59:56 -0800 | [diff] [blame] | 86 | requestIntent = _i != null ? _i[_i.length-1] : null; |
| 87 | requestResolvedType = _it != null ? _it[_it.length-1] : null; |
| 88 | allIntents = _i; |
| 89 | allResolvedTypes = _it; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 90 | flags = _f; |
Dianne Hackborn | 7a2195c | 2012-03-19 17:38:00 -0700 | [diff] [blame] | 91 | options = _o; |
Amith Yamasani | 4ea6069 | 2012-08-28 14:34:53 -0700 | [diff] [blame] | 92 | userId = _userId; |
| 93 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 94 | int hash = 23; |
| 95 | hash = (ODD_PRIME_NUMBER*hash) + _f; |
| 96 | hash = (ODD_PRIME_NUMBER*hash) + _r; |
Amith Yamasani | 4ea6069 | 2012-08-28 14:34:53 -0700 | [diff] [blame] | 97 | hash = (ODD_PRIME_NUMBER*hash) + _userId; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 98 | if (_w != null) { |
| 99 | hash = (ODD_PRIME_NUMBER*hash) + _w.hashCode(); |
| 100 | } |
| 101 | if (_a != null) { |
| 102 | hash = (ODD_PRIME_NUMBER*hash) + _a.hashCode(); |
| 103 | } |
Dianne Hackborn | 621e17d | 2010-11-22 15:59:56 -0800 | [diff] [blame] | 104 | if (requestIntent != null) { |
| 105 | hash = (ODD_PRIME_NUMBER*hash) + requestIntent.filterHashCode(); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 106 | } |
Dianne Hackborn | 621e17d | 2010-11-22 15:59:56 -0800 | [diff] [blame] | 107 | if (requestResolvedType != null) { |
| 108 | hash = (ODD_PRIME_NUMBER*hash) + requestResolvedType.hashCode(); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 109 | } |
Rubin Xu | f24d606 | 2016-07-20 17:34:50 +0100 | [diff] [blame] | 110 | hash = (ODD_PRIME_NUMBER*hash) + (_p != null ? _p.hashCode() : 0); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 111 | hash = (ODD_PRIME_NUMBER*hash) + _t; |
| 112 | hashCode = hash; |
Joe Onorato | 8a9b220 | 2010-02-26 18:56:32 -0800 | [diff] [blame] | 113 | //Slog.i(ActivityManagerService.TAG, this + " hashCode=0x" |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 114 | // + Integer.toHexString(hashCode)); |
| 115 | } |
Felipe Leme | a1b79bf | 2016-05-24 13:06:54 -0700 | [diff] [blame] | 116 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 117 | public boolean equals(Object otherObj) { |
| 118 | if (otherObj == null) { |
| 119 | return false; |
| 120 | } |
| 121 | try { |
| 122 | Key other = (Key)otherObj; |
| 123 | if (type != other.type) { |
| 124 | return false; |
| 125 | } |
Amith Yamasani | 4ea6069 | 2012-08-28 14:34:53 -0700 | [diff] [blame] | 126 | if (userId != other.userId){ |
| 127 | return false; |
| 128 | } |
Rubin Xu | f24d606 | 2016-07-20 17:34:50 +0100 | [diff] [blame] | 129 | if (!Objects.equals(packageName, other.packageName)) { |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 130 | return false; |
| 131 | } |
| 132 | if (activity != other.activity) { |
| 133 | return false; |
| 134 | } |
Rubin Xu | f24d606 | 2016-07-20 17:34:50 +0100 | [diff] [blame] | 135 | if (!Objects.equals(who, other.who)) { |
| 136 | return false; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 137 | } |
| 138 | if (requestCode != other.requestCode) { |
| 139 | return false; |
| 140 | } |
| 141 | if (requestIntent != other.requestIntent) { |
| 142 | if (requestIntent != null) { |
| 143 | if (!requestIntent.filterEquals(other.requestIntent)) { |
| 144 | return false; |
| 145 | } |
| 146 | } else if (other.requestIntent != null) { |
| 147 | return false; |
| 148 | } |
| 149 | } |
Rubin Xu | f24d606 | 2016-07-20 17:34:50 +0100 | [diff] [blame] | 150 | if (!Objects.equals(requestResolvedType, other.requestResolvedType)) { |
| 151 | return false; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 152 | } |
| 153 | if (flags != other.flags) { |
| 154 | return false; |
| 155 | } |
| 156 | return true; |
| 157 | } catch (ClassCastException e) { |
| 158 | } |
| 159 | return false; |
| 160 | } |
| 161 | |
| 162 | public int hashCode() { |
| 163 | return hashCode; |
| 164 | } |
Craig Mautner | 5f2bb4c | 2015-03-12 16:10:27 -0700 | [diff] [blame] | 165 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 166 | public String toString() { |
| 167 | return "Key{" + typeName() + " pkg=" + packageName |
Dianne Hackborn | c3b91fd | 2010-02-23 17:25:30 -0800 | [diff] [blame] | 168 | + " intent=" |
Dianne Hackborn | 90c52de | 2011-09-23 12:57:44 -0700 | [diff] [blame] | 169 | + (requestIntent != null |
Dianne Hackborn | 21c241e | 2012-03-08 13:57:23 -0800 | [diff] [blame] | 170 | ? requestIntent.toShortString(false, true, false, false) : "<null>") |
Amith Yamasani | 4ea6069 | 2012-08-28 14:34:53 -0700 | [diff] [blame] | 171 | + " flags=0x" + Integer.toHexString(flags) + " u=" + userId + "}"; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 172 | } |
Craig Mautner | 5f2bb4c | 2015-03-12 16:10:27 -0700 | [diff] [blame] | 173 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 174 | String typeName() { |
| 175 | switch (type) { |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 176 | case ActivityManager.INTENT_SENDER_ACTIVITY: |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 177 | return "startActivity"; |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 178 | case ActivityManager.INTENT_SENDER_BROADCAST: |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 179 | return "broadcastIntent"; |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 180 | case ActivityManager.INTENT_SENDER_SERVICE: |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 181 | return "startService"; |
Christopher Tate | 08992ac | 2017-03-21 11:37:06 -0700 | [diff] [blame] | 182 | case ActivityManager.INTENT_SENDER_FOREGROUND_SERVICE: |
| 183 | return "startForegroundService"; |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 184 | case ActivityManager.INTENT_SENDER_ACTIVITY_RESULT: |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 185 | return "activityResult"; |
| 186 | } |
| 187 | return Integer.toString(type); |
| 188 | } |
| 189 | } |
Craig Mautner | 5f2bb4c | 2015-03-12 16:10:27 -0700 | [diff] [blame] | 190 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 191 | PendingIntentRecord(ActivityManagerService _owner, Key _k, int _u) { |
| 192 | owner = _owner; |
| 193 | key = _k; |
| 194 | uid = _u; |
| 195 | ref = new WeakReference<PendingIntentRecord>(this); |
| 196 | } |
| 197 | |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 198 | void setWhitelistDurationLocked(IBinder whitelistToken, long duration) { |
| 199 | if (duration > 0) { |
| 200 | if (whitelistDuration == null) { |
| 201 | whitelistDuration = new ArrayMap<>(); |
| 202 | } |
| 203 | whitelistDuration.put(whitelistToken, duration); |
| 204 | } else if (whitelistDuration != null) { |
| 205 | whitelistDuration.remove(whitelistToken); |
| 206 | if (whitelistDuration.size() <= 0) { |
| 207 | whitelistDuration = null; |
| 208 | } |
| 209 | |
| 210 | } |
Felipe Leme | a1b79bf | 2016-05-24 13:06:54 -0700 | [diff] [blame] | 211 | this.stringName = null; |
| 212 | } |
| 213 | |
Dianne Hackborn | f66adfd | 2017-04-13 11:01:48 -0700 | [diff] [blame] | 214 | public void registerCancelListenerLocked(IResultReceiver receiver) { |
| 215 | if (mCancelCallbacks == null) { |
| 216 | mCancelCallbacks = new RemoteCallbackList<>(); |
| 217 | } |
| 218 | mCancelCallbacks.register(receiver); |
| 219 | } |
| 220 | |
| 221 | public void unregisterCancelListenerLocked(IResultReceiver receiver) { |
Makoto Onuki | 8a0319a | 2018-04-25 16:31:05 -0700 | [diff] [blame] | 222 | if (mCancelCallbacks == null) { |
| 223 | return; // Already unregistered or detached. |
| 224 | } |
Dianne Hackborn | f66adfd | 2017-04-13 11:01:48 -0700 | [diff] [blame] | 225 | mCancelCallbacks.unregister(receiver); |
| 226 | if (mCancelCallbacks.getRegisteredCallbackCount() <= 0) { |
| 227 | mCancelCallbacks = null; |
| 228 | } |
| 229 | } |
| 230 | |
| 231 | public RemoteCallbackList<IResultReceiver> detachCancelListenersLocked() { |
| 232 | RemoteCallbackList<IResultReceiver> listeners = mCancelCallbacks; |
| 233 | mCancelCallbacks = null; |
| 234 | return listeners; |
| 235 | } |
| 236 | |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 237 | public void send(int code, Intent intent, String resolvedType, IBinder whitelistToken, |
Dianne Hackborn | 0c4e6a8 | 2016-05-13 17:37:08 -0700 | [diff] [blame] | 238 | IIntentReceiver finishedReceiver, String requiredPermission, Bundle options) { |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 239 | sendInner(code, intent, resolvedType, whitelistToken, finishedReceiver, |
Andrii Kulian | b1cdb10 | 2017-07-13 15:33:06 -0700 | [diff] [blame] | 240 | requiredPermission, null, null, 0, 0, 0, options); |
Dianne Hackborn | bcbcaa7 | 2009-09-10 10:54:46 -0700 | [diff] [blame] | 241 | } |
Craig Mautner | 5f2bb4c | 2015-03-12 16:10:27 -0700 | [diff] [blame] | 242 | |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 243 | public int sendWithResult(int code, Intent intent, String resolvedType, IBinder whitelistToken, |
| 244 | IIntentReceiver finishedReceiver, String requiredPermission, Bundle options) { |
| 245 | return sendInner(code, intent, resolvedType, whitelistToken, finishedReceiver, |
Andrii Kulian | b1cdb10 | 2017-07-13 15:33:06 -0700 | [diff] [blame] | 246 | requiredPermission, null, null, 0, 0, 0, options); |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 247 | } |
| 248 | |
| 249 | int sendInner(int code, Intent intent, String resolvedType, IBinder whitelistToken, |
| 250 | IIntentReceiver finishedReceiver, |
Craig Mautner | 5f2bb4c | 2015-03-12 16:10:27 -0700 | [diff] [blame] | 251 | String requiredPermission, IBinder resultTo, String resultWho, int requestCode, |
Andrii Kulian | b1cdb10 | 2017-07-13 15:33:06 -0700 | [diff] [blame] | 252 | int flagsMask, int flagsValues, Bundle options) { |
Jeff Sharkey | f0ec2e0 | 2016-03-21 12:37:54 -0600 | [diff] [blame] | 253 | if (intent != null) intent.setDefusable(true); |
| 254 | if (options != null) options.setDefusable(true); |
| 255 | |
| 256 | synchronized (owner) { |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 257 | if (!canceled) { |
| 258 | sent = true; |
| 259 | if ((key.flags&PendingIntent.FLAG_ONE_SHOT) != 0) { |
| 260 | owner.cancelIntentSenderLocked(this, true); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 261 | } |
Svetoslav | b0a7839 | 2015-04-10 17:25:35 -0700 | [diff] [blame] | 262 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 263 | Intent finalIntent = key.requestIntent != null |
| 264 | ? new Intent(key.requestIntent) : new Intent(); |
Svetoslav | b0a7839 | 2015-04-10 17:25:35 -0700 | [diff] [blame] | 265 | |
| 266 | final boolean immutable = (key.flags & PendingIntent.FLAG_IMMUTABLE) != 0; |
| 267 | if (!immutable) { |
| 268 | if (intent != null) { |
| 269 | int changes = finalIntent.fillIn(intent, key.flags); |
| 270 | if ((changes & Intent.FILL_IN_DATA) == 0) { |
| 271 | resolvedType = key.requestResolvedType; |
| 272 | } |
| 273 | } else { |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 274 | resolvedType = key.requestResolvedType; |
| 275 | } |
Svetoslav | b0a7839 | 2015-04-10 17:25:35 -0700 | [diff] [blame] | 276 | flagsMask &= ~Intent.IMMUTABLE_FLAGS; |
| 277 | flagsValues &= flagsMask; |
| 278 | finalIntent.setFlags((finalIntent.getFlags() & ~flagsMask) | flagsValues); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 279 | } else { |
| 280 | resolvedType = key.requestResolvedType; |
| 281 | } |
Craig Mautner | 5f2bb4c | 2015-03-12 16:10:27 -0700 | [diff] [blame] | 282 | |
Amith Yamasani | 83c994f | 2017-05-01 14:29:19 -0700 | [diff] [blame] | 283 | final int callingUid = Binder.getCallingUid(); |
| 284 | final int callingPid = Binder.getCallingPid(); |
| 285 | |
Jorim Jaggi | 12d38a8 | 2018-04-26 18:51:16 +0200 | [diff] [blame] | 286 | // Extract options before clearing calling identity |
| 287 | SafeActivityOptions mergedOptions = key.options; |
| 288 | if (mergedOptions == null) { |
| 289 | mergedOptions = SafeActivityOptions.fromBundle(options); |
| 290 | } else { |
| 291 | mergedOptions.setCallerOptions(ActivityOptions.fromBundle(options)); |
| 292 | } |
| 293 | |
Amith Yamasani | 83c994f | 2017-05-01 14:29:19 -0700 | [diff] [blame] | 294 | final long origId = Binder.clearCallingIdentity(); |
| 295 | |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 296 | if (whitelistDuration != null) { |
| 297 | Long duration = whitelistDuration.get(whitelistToken); |
| 298 | if (duration != null) { |
| 299 | int procState = owner.getUidState(callingUid); |
| 300 | if (!ActivityManager.isProcStateBackground(procState)) { |
| 301 | StringBuilder tag = new StringBuilder(64); |
| 302 | tag.append("pendingintent:"); |
| 303 | UserHandle.formatUid(tag, callingUid); |
| 304 | tag.append(":"); |
| 305 | if (finalIntent.getAction() != null) { |
| 306 | tag.append(finalIntent.getAction()); |
| 307 | } else if (finalIntent.getComponent() != null) { |
| 308 | finalIntent.getComponent().appendShortString(tag); |
| 309 | } else if (finalIntent.getData() != null) { |
| 310 | tag.append(finalIntent.getData()); |
| 311 | } |
| 312 | owner.tempWhitelistForPendingIntentLocked(callingPid, |
| 313 | callingUid, uid, duration, tag.toString()); |
| 314 | } else { |
| 315 | Slog.w(TAG, "Not doing whitelist " + this + ": caller state=" |
| 316 | + procState); |
| 317 | } |
Dianne Hackborn | e4d1a2e | 2017-04-14 17:57:33 -0700 | [diff] [blame] | 318 | } |
Dianne Hackborn | e4d1a2e | 2017-04-14 17:57:33 -0700 | [diff] [blame] | 319 | } |
| 320 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 321 | boolean sendFinish = finishedReceiver != null; |
Dianne Hackborn | 50cdf7c3 | 2012-09-23 17:08:57 -0700 | [diff] [blame] | 322 | int userId = key.userId; |
| 323 | if (userId == UserHandle.USER_CURRENT) { |
Fyodor Kupolov | 1b3edac | 2017-09-19 15:48:06 -0700 | [diff] [blame] | 324 | userId = owner.mUserController.getCurrentOrTargetUserId(); |
Dianne Hackborn | 50cdf7c3 | 2012-09-23 17:08:57 -0700 | [diff] [blame] | 325 | } |
Jorim Jaggi | e2ad37f | 2018-01-22 22:41:22 +0100 | [diff] [blame] | 326 | int res = START_SUCCESS; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 327 | switch (key.type) { |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 328 | case ActivityManager.INTENT_SENDER_ACTIVITY: |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 329 | try { |
Makoto Onuki | c00ea71 | 2018-04-13 12:06:39 -0700 | [diff] [blame] | 330 | // Note when someone has a pending intent, even from different |
| 331 | // users, then there's no need to ensure the calling user matches |
| 332 | // the target user, so validateIncomingUser is always false below. |
| 333 | |
Dianne Hackborn | 621e17d | 2010-11-22 15:59:56 -0800 | [diff] [blame] | 334 | if (key.allIntents != null && key.allIntents.length > 1) { |
| 335 | Intent[] allIntents = new Intent[key.allIntents.length]; |
| 336 | String[] allResolvedTypes = new String[key.allIntents.length]; |
| 337 | System.arraycopy(key.allIntents, 0, allIntents, 0, |
| 338 | key.allIntents.length); |
| 339 | if (key.allResolvedTypes != null) { |
| 340 | System.arraycopy(key.allResolvedTypes, 0, allResolvedTypes, 0, |
| 341 | key.allResolvedTypes.length); |
| 342 | } |
| 343 | allIntents[allIntents.length-1] = finalIntent; |
| 344 | allResolvedTypes[allResolvedTypes.length-1] = resolvedType; |
Makoto Onuki | c00ea71 | 2018-04-13 12:06:39 -0700 | [diff] [blame] | 345 | |
Jorim Jaggi | e2ad37f | 2018-01-22 22:41:22 +0100 | [diff] [blame] | 346 | res = owner.getActivityStartController().startActivitiesInPackage( |
| 347 | uid, key.packageName, allIntents, allResolvedTypes, |
Makoto Onuki | 7041c4b | 2018-02-06 13:36:34 -0800 | [diff] [blame] | 348 | resultTo, mergedOptions, userId, |
Makoto Onuki | c00ea71 | 2018-04-13 12:06:39 -0700 | [diff] [blame] | 349 | false /* validateIncomingUser */); |
Dianne Hackborn | 621e17d | 2010-11-22 15:59:56 -0800 | [diff] [blame] | 350 | } else { |
Jorim Jaggi | e2ad37f | 2018-01-22 22:41:22 +0100 | [diff] [blame] | 351 | res = owner.getActivityStartController().startActivityInPackage(uid, |
Jorim Jaggi | 4d8d32c | 2018-01-19 15:57:41 +0100 | [diff] [blame] | 352 | callingPid, callingUid, key.packageName, finalIntent, |
| 353 | resolvedType, resultTo, resultWho, requestCode, 0, |
Makoto Onuki | c00ea71 | 2018-04-13 12:06:39 -0700 | [diff] [blame] | 354 | mergedOptions, userId, null, "PendingIntentRecord", |
| 355 | false /* validateIncomingUser */); |
Dianne Hackborn | 621e17d | 2010-11-22 15:59:56 -0800 | [diff] [blame] | 356 | } |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 357 | } catch (RuntimeException e) { |
Wale Ogunwale | e23149f | 2015-03-06 15:39:44 -0800 | [diff] [blame] | 358 | Slog.w(TAG, "Unable to send startActivity intent", e); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 359 | } |
| 360 | break; |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 361 | case ActivityManager.INTENT_SENDER_ACTIVITY_RESULT: |
Andrii Kulian | 02b7a83 | 2016-10-06 23:11:56 -0700 | [diff] [blame] | 362 | final ActivityStack stack = key.activity.getStack(); |
| 363 | if (stack != null) { |
| 364 | stack.sendActivityResultLocked(-1, key.activity, key.who, |
| 365 | key.requestCode, code, finalIntent); |
Wale Ogunwale | 7d70117 | 2015-03-11 15:36:30 -0700 | [diff] [blame] | 366 | } |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 367 | break; |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 368 | case ActivityManager.INTENT_SENDER_BROADCAST: |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 369 | try { |
| 370 | // If a completion callback has been requested, require |
| 371 | // that the broadcast be delivered synchronously |
Amith Yamasani | 83b6ef0 | 2014-11-07 15:34:04 -0800 | [diff] [blame] | 372 | int sent = owner.broadcastIntentInPackage(key.packageName, uid, |
Dianne Hackborn | a750a63 | 2015-06-16 17:18:23 -0700 | [diff] [blame] | 373 | finalIntent, resolvedType, finishedReceiver, code, null, null, |
| 374 | requiredPermission, options, (finishedReceiver != null), |
| 375 | false, userId); |
Amith Yamasani | 83b6ef0 | 2014-11-07 15:34:04 -0800 | [diff] [blame] | 376 | if (sent == ActivityManager.BROADCAST_SUCCESS) { |
| 377 | sendFinish = false; |
| 378 | } |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 379 | } catch (RuntimeException e) { |
Wale Ogunwale | e23149f | 2015-03-06 15:39:44 -0800 | [diff] [blame] | 380 | Slog.w(TAG, "Unable to send startActivity intent", e); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 381 | } |
| 382 | break; |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 383 | case ActivityManager.INTENT_SENDER_SERVICE: |
Christopher Tate | 08992ac | 2017-03-21 11:37:06 -0700 | [diff] [blame] | 384 | case ActivityManager.INTENT_SENDER_FOREGROUND_SERVICE: |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 385 | try { |
Christopher Tate | 08992ac | 2017-03-21 11:37:06 -0700 | [diff] [blame] | 386 | owner.startServiceInPackage(uid, finalIntent, resolvedType, |
| 387 | key.type == ActivityManager.INTENT_SENDER_FOREGROUND_SERVICE, |
| 388 | key.packageName, userId); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 389 | } catch (RuntimeException e) { |
Wale Ogunwale | e23149f | 2015-03-06 15:39:44 -0800 | [diff] [blame] | 390 | Slog.w(TAG, "Unable to send startService intent", e); |
Dianne Hackborn | 0c4e6a8 | 2016-05-13 17:37:08 -0700 | [diff] [blame] | 391 | } catch (TransactionTooLargeException e) { |
| 392 | res = ActivityManager.START_CANCELED; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 393 | } |
| 394 | break; |
| 395 | } |
Wale Ogunwale | e23149f | 2015-03-06 15:39:44 -0800 | [diff] [blame] | 396 | |
Dianne Hackborn | 0c4e6a8 | 2016-05-13 17:37:08 -0700 | [diff] [blame] | 397 | if (sendFinish && res != ActivityManager.START_CANCELED) { |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 398 | try { |
| 399 | finishedReceiver.performReceive(new Intent(finalIntent), 0, |
Dianne Hackborn | 20e8098 | 2012-08-31 19:00:44 -0700 | [diff] [blame] | 400 | null, null, false, false, key.userId); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 401 | } catch (RemoteException e) { |
| 402 | } |
| 403 | } |
Dianne Hackborn | 0c4e6a8 | 2016-05-13 17:37:08 -0700 | [diff] [blame] | 404 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 405 | Binder.restoreCallingIdentity(origId); |
Dianne Hackborn | 0c4e6a8 | 2016-05-13 17:37:08 -0700 | [diff] [blame] | 406 | |
| 407 | return res; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 408 | } |
| 409 | } |
Dianne Hackborn | a4972e9 | 2012-03-14 10:38:05 -0700 | [diff] [blame] | 410 | return ActivityManager.START_CANCELED; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 411 | } |
Craig Mautner | df88d73 | 2014-01-27 09:21:32 -0800 | [diff] [blame] | 412 | |
| 413 | @Override |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 414 | protected void finalize() throws Throwable { |
Dianne Hackborn | 9e0f5d9 | 2010-02-22 15:05:42 -0800 | [diff] [blame] | 415 | try { |
| 416 | if (!canceled) { |
| 417 | owner.mHandler.sendMessage(owner.mHandler.obtainMessage( |
| 418 | ActivityManagerService.FINALIZE_PENDING_INTENT_MSG, this)); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 419 | } |
Dianne Hackborn | 9e0f5d9 | 2010-02-22 15:05:42 -0800 | [diff] [blame] | 420 | } finally { |
| 421 | super.finalize(); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 422 | } |
| 423 | } |
| 424 | |
Dianne Hackborn | 9e0f5d9 | 2010-02-22 15:05:42 -0800 | [diff] [blame] | 425 | public void completeFinalize() { |
| 426 | synchronized(owner) { |
| 427 | WeakReference<PendingIntentRecord> current = |
| 428 | owner.mIntentSenderRecords.get(key); |
| 429 | if (current == ref) { |
| 430 | owner.mIntentSenderRecords.remove(key); |
| 431 | } |
| 432 | } |
| 433 | } |
Felipe Leme | a1b79bf | 2016-05-24 13:06:54 -0700 | [diff] [blame] | 434 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 435 | void dump(PrintWriter pw, String prefix) { |
Dianne Hackborn | 1d442e0 | 2009-04-20 18:14:05 -0700 | [diff] [blame] | 436 | pw.print(prefix); pw.print("uid="); pw.print(uid); |
| 437 | pw.print(" packageName="); pw.print(key.packageName); |
| 438 | pw.print(" type="); pw.print(key.typeName()); |
| 439 | pw.print(" flags=0x"); pw.println(Integer.toHexString(key.flags)); |
| 440 | if (key.activity != null || key.who != null) { |
| 441 | pw.print(prefix); pw.print("activity="); pw.print(key.activity); |
| 442 | pw.print(" who="); pw.println(key.who); |
| 443 | } |
| 444 | if (key.requestCode != 0 || key.requestResolvedType != null) { |
| 445 | pw.print(prefix); pw.print("requestCode="); pw.print(key.requestCode); |
| 446 | pw.print(" requestResolvedType="); pw.println(key.requestResolvedType); |
| 447 | } |
Dianne Hackborn | c3b91fd | 2010-02-23 17:25:30 -0800 | [diff] [blame] | 448 | if (key.requestIntent != null) { |
| 449 | pw.print(prefix); pw.print("requestIntent="); |
Dianne Hackborn | 21c241e | 2012-03-08 13:57:23 -0800 | [diff] [blame] | 450 | pw.println(key.requestIntent.toShortString(false, true, true, true)); |
Dianne Hackborn | c3b91fd | 2010-02-23 17:25:30 -0800 | [diff] [blame] | 451 | } |
Dianne Hackborn | 1d442e0 | 2009-04-20 18:14:05 -0700 | [diff] [blame] | 452 | if (sent || canceled) { |
| 453 | pw.print(prefix); pw.print("sent="); pw.print(sent); |
| 454 | pw.print(" canceled="); pw.println(canceled); |
| 455 | } |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 456 | if (whitelistDuration != null) { |
Dianne Hackborn | bc02a39 | 2016-06-02 17:15:08 -0700 | [diff] [blame] | 457 | pw.print(prefix); |
| 458 | pw.print("whitelistDuration="); |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 459 | for (int i = 0; i < whitelistDuration.size(); i++) { |
| 460 | if (i != 0) { |
| 461 | pw.print(", "); |
| 462 | } |
| 463 | pw.print(Integer.toHexString(System.identityHashCode(whitelistDuration.keyAt(i)))); |
| 464 | pw.print(":"); |
| 465 | TimeUtils.formatDuration(whitelistDuration.valueAt(i), pw); |
| 466 | } |
Dianne Hackborn | bc02a39 | 2016-06-02 17:15:08 -0700 | [diff] [blame] | 467 | pw.println(); |
| 468 | } |
Dianne Hackborn | f66adfd | 2017-04-13 11:01:48 -0700 | [diff] [blame] | 469 | if (mCancelCallbacks != null) { |
| 470 | pw.print(prefix); pw.println("mCancelCallbacks:"); |
| 471 | for (int i = 0; i < mCancelCallbacks.getRegisteredCallbackCount(); i++) { |
| 472 | pw.print(prefix); pw.print(" #"); pw.print(i); pw.print(": "); |
| 473 | pw.println(mCancelCallbacks.getRegisteredCallbackItem(i)); |
| 474 | } |
| 475 | } |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 476 | } |
| 477 | |
| 478 | public String toString() { |
Dianne Hackborn | 1d442e0 | 2009-04-20 18:14:05 -0700 | [diff] [blame] | 479 | if (stringName != null) { |
| 480 | return stringName; |
| 481 | } |
| 482 | StringBuilder sb = new StringBuilder(128); |
| 483 | sb.append("PendingIntentRecord{"); |
| 484 | sb.append(Integer.toHexString(System.identityHashCode(this))); |
| 485 | sb.append(' '); |
| 486 | sb.append(key.packageName); |
| 487 | sb.append(' '); |
| 488 | sb.append(key.typeName()); |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 489 | if (whitelistDuration != null) { |
Dianne Hackborn | bc02a39 | 2016-06-02 17:15:08 -0700 | [diff] [blame] | 490 | sb.append( " (whitelist: "); |
Dianne Hackborn | 9830552 | 2017-05-05 17:53:53 -0700 | [diff] [blame] | 491 | for (int i = 0; i < whitelistDuration.size(); i++) { |
| 492 | if (i != 0) { |
| 493 | sb.append(","); |
| 494 | } |
| 495 | sb.append(Integer.toHexString(System.identityHashCode(whitelistDuration.keyAt(i)))); |
| 496 | sb.append(":"); |
| 497 | TimeUtils.formatDuration(whitelistDuration.valueAt(i), sb); |
| 498 | } |
Dianne Hackborn | bc02a39 | 2016-06-02 17:15:08 -0700 | [diff] [blame] | 499 | sb.append(")"); |
Felipe Leme | a1b79bf | 2016-05-24 13:06:54 -0700 | [diff] [blame] | 500 | } |
Dianne Hackborn | 1d442e0 | 2009-04-20 18:14:05 -0700 | [diff] [blame] | 501 | sb.append('}'); |
| 502 | return stringName = sb.toString(); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 503 | } |
| 504 | } |