1 /* 2 * Copyright (C) 2019 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 android.app.cts; 18 19 import static android.app.ActivityManager.PROCESS_CAPABILITY_ALL; 20 import static android.app.ActivityManager.PROCESS_CAPABILITY_FOREGROUND_CAMERA; 21 import static android.app.ActivityManager.PROCESS_CAPABILITY_FOREGROUND_MICROPHONE; 22 import static android.app.ActivityManager.PROCESS_CAPABILITY_NETWORK; 23 import static android.app.ActivityManager.PROCESS_CAPABILITY_NONE; 24 import static android.app.stubs.LocalForegroundService.ACTION_START_FGS_RESULT; 25 import static android.app.stubs.LocalForegroundServiceLocation.ACTION_START_FGSL_RESULT; 26 import static android.os.PowerExemptionManager.REASON_PUSH_MESSAGING; 27 import static android.os.PowerExemptionManager.REASON_PUSH_MESSAGING_OVER_QUOTA; 28 import static android.os.PowerExemptionManager.REASON_UNKNOWN; 29 import static android.os.PowerExemptionManager.TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_ALLOWED; 30 import static android.os.PowerExemptionManager.TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_NOT_ALLOWED; 31 import static android.os.PowerExemptionManager.TEMPORARY_ALLOW_LIST_TYPE_NONE; 32 33 import static com.android.compatibility.common.util.SystemUtil.runWithShellPermissionIdentity; 34 35 import static junit.framework.Assert.assertNotNull; 36 import static junit.framework.Assert.assertTrue; 37 import static junit.framework.Assert.fail; 38 39 import android.accessibilityservice.AccessibilityService; 40 import android.app.ActivityManager; 41 import android.app.BroadcastOptions; 42 import android.app.ForegroundServiceStartNotAllowedException; 43 import android.app.Instrumentation; 44 import android.app.cts.android.app.cts.tools.WaitForBroadcast; 45 import android.app.cts.android.app.cts.tools.WatchUidRunner; 46 import android.app.stubs.CommandReceiver; 47 import android.app.stubs.LocalForegroundService; 48 import android.app.stubs.LocalForegroundServiceLocation; 49 import android.content.ComponentName; 50 import android.content.Context; 51 import android.content.Intent; 52 import android.content.ServiceConnection; 53 import android.content.pm.ApplicationInfo; 54 import android.content.pm.ServiceInfo; 55 import android.os.Bundle; 56 import android.os.IBinder; 57 import android.os.PowerExemptionManager; 58 import android.os.SystemClock; 59 import android.permission.cts.PermissionUtils; 60 import android.platform.test.annotations.AsbSecurityTest; 61 import android.provider.DeviceConfig; 62 import android.provider.Settings; 63 64 import androidx.test.InstrumentationRegistry; 65 import androidx.test.ext.junit.runners.AndroidJUnit4; 66 import androidx.test.filters.LargeTest; 67 68 import com.android.compatibility.common.util.SystemUtil; 69 70 import org.junit.After; 71 import org.junit.Before; 72 import org.junit.Ignore; 73 import org.junit.Test; 74 import org.junit.runner.RunWith; 75 76 @RunWith(AndroidJUnit4.class) 77 public class ActivityManagerFgsBgStartTest { 78 private static final String TAG = ActivityManagerFgsBgStartTest.class.getName(); 79 80 private static final String STUB_PACKAGE_NAME = "android.app.stubs"; 81 private static final String PACKAGE_NAME_APP1 = "com.android.app1"; 82 private static final String PACKAGE_NAME_APP2 = "com.android.app2"; 83 private static final String PACKAGE_NAME_APP3 = "com.android.app3"; 84 85 private static final String KEY_DEFAULT_FGS_STARTS_RESTRICTION_ENABLED = 86 "default_fgs_starts_restriction_enabled"; 87 88 private static final String KEY_FGS_START_FOREGROUND_TIMEOUT = 89 "fgs_start_foreground_timeout"; 90 91 private static final String KEY_PUSH_MESSAGING_OVER_QUOTA_BEHAVIOR = 92 "push_messaging_over_quota_behavior"; 93 94 private static final int DEFAULT_FGS_START_FOREGROUND_TIMEOUT_MS = 10 * 1000; 95 96 public static final Integer LOCAL_SERVICE_PROCESS_CAPABILITY = new Integer( 97 PROCESS_CAPABILITY_FOREGROUND_CAMERA 98 | PROCESS_CAPABILITY_FOREGROUND_MICROPHONE 99 | PROCESS_CAPABILITY_NETWORK); 100 101 private static final int WAITFOR_MSEC = 10000; 102 103 private static final int TEMP_ALLOWLIST_DURATION_MS = 2000; 104 105 private static final String[] PACKAGE_NAMES = { 106 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, PACKAGE_NAME_APP3 107 }; 108 109 private Context mContext; 110 private Instrumentation mInstrumentation; 111 private Context mTargetContext; 112 113 private int mOrigDeviceDemoMode = 0; 114 115 @Before setUp()116 public void setUp() throws Exception { 117 mInstrumentation = InstrumentationRegistry.getInstrumentation(); 118 mContext = mInstrumentation.getContext(); 119 mTargetContext = mInstrumentation.getTargetContext(); 120 for (int i = 0; i < PACKAGE_NAMES.length; ++i) { 121 CtsAppTestUtils.makeUidIdle(mInstrumentation, PACKAGE_NAMES[i]); 122 // The manifest file gives test app SYSTEM_ALERT_WINDOW permissions, which also exempt 123 // the app from BG-FGS-launch restriction. Remove SYSTEM_ALERT_WINDOW permission to test 124 // other BG-FGS-launch exemptions. 125 allowBgActivityStart(PACKAGE_NAMES[i], false); 126 } 127 CtsAppTestUtils.turnScreenOn(mInstrumentation, mContext); 128 cleanupResiduals(); 129 enableFgsRestriction(true, true, null); 130 } 131 132 @After tearDown()133 public void tearDown() throws Exception { 134 for (int i = 0; i < PACKAGE_NAMES.length; ++i) { 135 CtsAppTestUtils.makeUidIdle(mInstrumentation, PACKAGE_NAMES[i]); 136 allowBgActivityStart(PACKAGE_NAMES[i], true); 137 } 138 cleanupResiduals(); 139 enableFgsRestriction(true, true, null); 140 for (String packageName: PACKAGE_NAMES) { 141 resetFgsRestriction(packageName); 142 } 143 } 144 cleanupResiduals()145 private void cleanupResiduals() { 146 // Stop all the packages to avoid residual impact 147 final ActivityManager am = mContext.getSystemService(ActivityManager.class); 148 for (int i = 0; i < PACKAGE_NAMES.length; i++) { 149 final String pkgName = PACKAGE_NAMES[i]; 150 SystemUtil.runWithShellPermissionIdentity(() -> { 151 am.forceStopPackage(pkgName); 152 }); 153 } 154 // Make sure we are in Home screen 155 mInstrumentation.getUiAutomation().performGlobalAction( 156 AccessibilityService.GLOBAL_ACTION_HOME); 157 } 158 159 /** 160 * APP1 is in BG state, it can start FGSL, but it won't get location capability. 161 * APP1 is in TOP state, it gets location capability. 162 * @throws Exception 163 */ 164 @Test testFgsLocationStartFromBG()165 public void testFgsLocationStartFromBG() throws Exception { 166 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 167 PACKAGE_NAME_APP1, 0); 168 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 169 WAITFOR_MSEC); 170 171 try { 172 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 173 waiter.prepare(ACTION_START_FGSL_RESULT); 174 // APP1 is in BG state, Start FGSL in APP1, it won't get location capability. 175 Bundle bundle = new Bundle(); 176 bundle.putInt(LocalForegroundServiceLocation.EXTRA_FOREGROUND_SERVICE_TYPE, 177 ServiceInfo.FOREGROUND_SERVICE_TYPE_LOCATION); 178 // start FGSL. 179 enableFgsRestriction(false, true, null); 180 CommandReceiver.sendCommand(mContext, 181 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE_LOCATION, 182 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, bundle); 183 // APP1 is in FGS state, but won't get location capability. 184 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 185 WatchUidRunner.STATE_FG_SERVICE, 186 new Integer(PROCESS_CAPABILITY_NETWORK)); 187 waiter.doWait(WAITFOR_MSEC); 188 // stop FGSL 189 CommandReceiver.sendCommand(mContext, 190 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE_LOCATION, 191 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 192 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 193 WatchUidRunner.STATE_CACHED_EMPTY, 194 new Integer(PROCESS_CAPABILITY_NONE)); 195 196 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 197 waiter.prepare(ACTION_START_FGS_RESULT); 198 // APP1 is in FGS state, 199 CommandReceiver.sendCommand(mContext, 200 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 201 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, bundle); 202 // start FGSL in app1, it won't get location capability. 203 CommandReceiver.sendCommand(mContext, 204 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE_LOCATION, 205 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, bundle); 206 // APP1 is in STATE_FG_SERVICE, but won't get location capability. 207 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 208 WatchUidRunner.STATE_FG_SERVICE, 209 new Integer(PROCESS_CAPABILITY_NETWORK)); 210 waiter.doWait(WAITFOR_MSEC); 211 // stop FGS. 212 CommandReceiver.sendCommand(mContext, 213 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 214 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 215 // stop FGSL. 216 CommandReceiver.sendCommand(mContext, 217 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE_LOCATION, 218 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 219 220 // Put APP1 in TOP state, now it gets location capability (because the TOP process 221 // gets all while-in-use permission (not from FGSL). 222 allowBgActivityStart(PACKAGE_NAME_APP1, true); 223 CommandReceiver.sendCommand(mContext, 224 CommandReceiver.COMMAND_START_ACTIVITY, 225 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 226 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 227 WatchUidRunner.STATE_TOP, 228 new Integer(PROCESS_CAPABILITY_ALL)); 229 230 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 231 waiter.prepare(ACTION_START_FGSL_RESULT); 232 // APP1 is in TOP state, start the FGSL in APP1, it will get location capability. 233 CommandReceiver.sendCommand(mContext, 234 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE_LOCATION, 235 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, bundle); 236 // Stop the activity. 237 CommandReceiver.sendCommand(mContext, 238 CommandReceiver.COMMAND_STOP_ACTIVITY, 239 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 240 // The FGSL still has location capability because it is started from TOP. 241 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 242 WatchUidRunner.STATE_FG_SERVICE, 243 new Integer(PROCESS_CAPABILITY_ALL)); 244 waiter.doWait(WAITFOR_MSEC); 245 // Stop FGSL. 246 CommandReceiver.sendCommand(mContext, 247 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE_LOCATION, 248 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 249 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 250 WatchUidRunner.STATE_CACHED_EMPTY, 251 new Integer(PROCESS_CAPABILITY_NONE)); 252 } finally { 253 uid1Watcher.finish(); 254 } 255 } 256 257 /** 258 * APP1 is in BG state, it can start FGSL in APP2, but the FGS won't get location 259 * capability. 260 * APP1 is in TOP state, it can start FGSL in APP2, FGSL gets location capability. 261 * @throws Exception 262 */ 263 @Test testFgsLocationStartFromBGTwoProcesses()264 public void testFgsLocationStartFromBGTwoProcesses() throws Exception { 265 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 266 PACKAGE_NAME_APP1, 0); 267 ApplicationInfo app2Info = mContext.getPackageManager().getApplicationInfo( 268 PACKAGE_NAME_APP2, 0); 269 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 270 WAITFOR_MSEC); 271 WatchUidRunner uid2Watcher = new WatchUidRunner(mInstrumentation, app2Info.uid, 272 WAITFOR_MSEC); 273 274 try { 275 // APP1 is in BG state, start FGSL in APP2. 276 Bundle bundle = new Bundle(); 277 bundle.putInt(LocalForegroundServiceLocation.EXTRA_FOREGROUND_SERVICE_TYPE, 278 ServiceInfo.FOREGROUND_SERVICE_TYPE_LOCATION); 279 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 280 waiter.prepare(ACTION_START_FGSL_RESULT); 281 enableFgsRestriction(false, true, null); 282 CommandReceiver.sendCommand(mContext, 283 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE_LOCATION, 284 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, bundle); 285 // APP2 won't have location capability because APP1 is not in TOP state. 286 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 287 WatchUidRunner.STATE_FG_SERVICE, 288 new Integer(PROCESS_CAPABILITY_NETWORK)); 289 waiter.doWait(WAITFOR_MSEC); 290 291 CommandReceiver.sendCommand(mContext, 292 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE_LOCATION, 293 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 294 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 295 WatchUidRunner.STATE_CACHED_EMPTY, 296 new Integer(PROCESS_CAPABILITY_NONE)); 297 298 // Put APP1 in TOP state 299 allowBgActivityStart(PACKAGE_NAME_APP1, true); 300 CommandReceiver.sendCommand(mContext, 301 CommandReceiver.COMMAND_START_ACTIVITY, 302 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 303 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 304 WatchUidRunner.STATE_TOP, 305 new Integer(PROCESS_CAPABILITY_ALL)); 306 307 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 308 waiter.prepare(ACTION_START_FGSL_RESULT); 309 // From APP1, start FGSL in APP2. 310 CommandReceiver.sendCommand(mContext, 311 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE_LOCATION, 312 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, bundle); 313 // Now APP2 gets location capability. 314 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 315 WatchUidRunner.STATE_FG_SERVICE, 316 new Integer(PROCESS_CAPABILITY_ALL)); 317 waiter.doWait(WAITFOR_MSEC); 318 319 CommandReceiver.sendCommand(mContext, 320 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE_LOCATION, 321 PACKAGE_NAME_APP2, PACKAGE_NAME_APP2, 0, null); 322 323 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 324 WatchUidRunner.STATE_CACHED_EMPTY, 325 new Integer(PROCESS_CAPABILITY_NONE)); 326 327 CommandReceiver.sendCommand(mContext, 328 CommandReceiver.COMMAND_STOP_ACTIVITY, 329 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 330 331 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 332 WatchUidRunner.STATE_CACHED_EMPTY, 333 new Integer(PROCESS_CAPABILITY_NONE)); 334 } finally { 335 uid1Watcher.finish(); 336 uid2Watcher.finish(); 337 } 338 } 339 340 /** 341 * APP1 is in BG state, by a PendingIntent, it can start FGSL in APP2, 342 * but the FGS won't get location capability. 343 * APP1 is in TOP state, by a PendingIntent, it can start FGSL in APP2, 344 * FGSL gets location capability. 345 * @throws Exception 346 */ 347 @Test testFgsLocationPendingIntent()348 public void testFgsLocationPendingIntent() throws Exception { 349 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 350 PACKAGE_NAME_APP1, 0); 351 ApplicationInfo app2Info = mContext.getPackageManager().getApplicationInfo( 352 PACKAGE_NAME_APP2, 0); 353 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 354 WAITFOR_MSEC); 355 WatchUidRunner uid2Watcher = new WatchUidRunner(mInstrumentation, app2Info.uid, 356 WAITFOR_MSEC); 357 358 try { 359 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 360 waiter.prepare(ACTION_START_FGSL_RESULT); 361 // APP1 is in BG state, start FGSL in APP2. 362 enableFgsRestriction(false, true, null); 363 CommandReceiver.sendCommand(mContext, 364 CommandReceiver.COMMAND_CREATE_FGSL_PENDING_INTENT, 365 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 366 CommandReceiver.sendCommand(mContext, 367 CommandReceiver.COMMAND_SEND_FGSL_PENDING_INTENT, 368 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 369 // APP2 won't have location capability. 370 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 371 WatchUidRunner.STATE_FG_SERVICE, 372 new Integer(PROCESS_CAPABILITY_NETWORK)); 373 waiter.doWait(WAITFOR_MSEC); 374 // Stop FGSL in APP2. 375 CommandReceiver.sendCommand(mContext, 376 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE_LOCATION, 377 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 378 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 379 WatchUidRunner.STATE_CACHED_EMPTY, 380 new Integer(PROCESS_CAPABILITY_NONE)); 381 382 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 383 waiter.prepare(ACTION_START_FGS_RESULT); 384 // Put APP1 in FGS state, start FGSL in APP2. 385 CommandReceiver.sendCommand(mContext, 386 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 387 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 388 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 389 WatchUidRunner.STATE_FG_SERVICE, 390 new Integer(PROCESS_CAPABILITY_NETWORK)); 391 waiter.doWait(WAITFOR_MSEC); 392 CommandReceiver.sendCommand(mContext, 393 CommandReceiver.COMMAND_CREATE_FGSL_PENDING_INTENT, 394 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 395 396 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 397 waiter.prepare(ACTION_START_FGSL_RESULT); 398 CommandReceiver.sendCommand(mContext, 399 CommandReceiver.COMMAND_SEND_FGSL_PENDING_INTENT, 400 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 401 // APP2 won't have location capability. 402 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 403 WatchUidRunner.STATE_FG_SERVICE, 404 new Integer(PROCESS_CAPABILITY_NETWORK)); 405 waiter.doWait(WAITFOR_MSEC); 406 // stop FGSL in APP2. 407 CommandReceiver.sendCommand(mContext, 408 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE_LOCATION, 409 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 410 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 411 WatchUidRunner.STATE_CACHED_EMPTY, 412 new Integer(PROCESS_CAPABILITY_NONE)); 413 414 // put APP1 in TOP state, start FGSL in APP2. 415 allowBgActivityStart(PACKAGE_NAME_APP1, true); 416 CommandReceiver.sendCommand(mContext, 417 CommandReceiver.COMMAND_START_ACTIVITY, 418 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 419 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 420 WatchUidRunner.STATE_TOP, 421 new Integer(PROCESS_CAPABILITY_ALL)); 422 CommandReceiver.sendCommand(mContext, 423 CommandReceiver.COMMAND_CREATE_FGSL_PENDING_INTENT, 424 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 425 426 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 427 waiter.prepare(ACTION_START_FGSL_RESULT); 428 CommandReceiver.sendCommand(mContext, 429 CommandReceiver.COMMAND_SEND_FGSL_PENDING_INTENT, 430 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 431 // APP2 now have location capability (because APP1 is TOP) 432 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 433 WatchUidRunner.STATE_FG_SERVICE, 434 new Integer(PROCESS_CAPABILITY_ALL)); 435 waiter.doWait(WAITFOR_MSEC); 436 437 // stop FGSL in APP2. 438 CommandReceiver.sendCommand(mContext, 439 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE_LOCATION, 440 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 441 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 442 WatchUidRunner.STATE_CACHED_EMPTY, 443 new Integer(PROCESS_CAPABILITY_NONE)); 444 445 // stop FGS in APP1, 446 CommandReceiver.sendCommand(mContext, 447 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 448 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 449 // stop TOP activity in APP1. 450 CommandReceiver.sendCommand(mContext, 451 CommandReceiver.COMMAND_STOP_ACTIVITY, 452 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 453 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 454 WatchUidRunner.STATE_CACHED_EMPTY, 455 new Integer(PROCESS_CAPABILITY_NONE)); 456 } finally { 457 uid1Watcher.finish(); 458 uid2Watcher.finish(); 459 } 460 } 461 462 /** 463 * Test a FGS start by bind from BG does not get get while-in-use capability. 464 * @throws Exception 465 */ 466 @Test 467 @AsbSecurityTest(cveBugId = 173516292) testFgsLocationStartFromBGWithBind()468 public void testFgsLocationStartFromBGWithBind() throws Exception { 469 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 470 PACKAGE_NAME_APP1, 0); 471 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 472 WAITFOR_MSEC); 473 474 try { 475 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 476 waiter.prepare(ACTION_START_FGSL_RESULT); 477 // APP1 is in BG state, bind FGSL in APP1 first. 478 enableFgsRestriction(false, true, null); 479 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_BIND_FOREGROUND_SERVICE, 480 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 481 Bundle bundle = new Bundle(); 482 bundle.putInt(LocalForegroundServiceLocation.EXTRA_FOREGROUND_SERVICE_TYPE, 483 ServiceInfo.FOREGROUND_SERVICE_TYPE_LOCATION); 484 // Then start FGSL in APP1, it won't get location capability. 485 CommandReceiver.sendCommand(mContext, 486 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE_LOCATION, 487 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, bundle); 488 489 // APP1 is in FGS state, but won't get location capability. 490 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 491 WatchUidRunner.STATE_FG_SERVICE, 492 new Integer(PROCESS_CAPABILITY_NETWORK)); 493 waiter.doWait(WAITFOR_MSEC); 494 495 // unbind service. 496 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_UNBIND_SERVICE, 497 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 498 // stop FGSL 499 CommandReceiver.sendCommand(mContext, 500 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE_LOCATION, 501 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 502 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 503 WatchUidRunner.STATE_CACHED_EMPTY, 504 new Integer(PROCESS_CAPABILITY_NONE)); 505 } finally { 506 uid1Watcher.finish(); 507 } 508 } 509 510 @Test testUpdateUidProcState()511 public void testUpdateUidProcState() throws Exception { 512 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 513 PACKAGE_NAME_APP1, 0); 514 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 515 WAITFOR_MSEC); 516 ApplicationInfo app2Info = mContext.getPackageManager().getApplicationInfo( 517 PACKAGE_NAME_APP2, 0); 518 WatchUidRunner uid2Watcher = new WatchUidRunner(mInstrumentation, app2Info.uid, 519 WAITFOR_MSEC); 520 ApplicationInfo app3Info = mContext.getPackageManager().getApplicationInfo( 521 PACKAGE_NAME_APP3, 0); 522 WatchUidRunner uid3Watcher = new WatchUidRunner(mInstrumentation, app3Info.uid, 523 WAITFOR_MSEC); 524 525 try { 526 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 527 waiter.prepare(ACTION_START_FGS_RESULT); 528 529 enableFgsRestriction(false, true, null); 530 531 // START FGS in APP2. 532 CommandReceiver.sendCommand(mContext, 533 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 534 PACKAGE_NAME_APP2, PACKAGE_NAME_APP2, 0, null); 535 // APP2 proc state is 4. 536 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 537 WatchUidRunner.STATE_FG_SERVICE); 538 waiter.doWait(WAITFOR_MSEC); 539 540 // APP2 binds to APP1. 541 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_BIND_SERVICE, 542 PACKAGE_NAME_APP2, PACKAGE_NAME_APP1, Context.BIND_INCLUDE_CAPABILITIES, null); 543 // APP1 gets proc state 4. 544 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 545 WatchUidRunner.STATE_FG_SERVICE); 546 547 // Start activity in APP3, this put APP3 in TOP state. 548 allowBgActivityStart(PACKAGE_NAME_APP3, true); 549 CommandReceiver.sendCommand(mContext, 550 CommandReceiver.COMMAND_START_ACTIVITY, 551 PACKAGE_NAME_APP3, PACKAGE_NAME_APP3, 0, null); 552 // APP3 gets proc state 2. 553 uid3Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_TOP); 554 555 // APP3 repeatedly bind/unbind with APP2, observer APP1 proc state change. 556 // Observe updateUidProcState() call latency. 557 for (int i = 0; i < 10; ++i) { 558 // APP3 bind to APP2 559 CommandReceiver.sendCommand(mContext, 560 CommandReceiver.COMMAND_BIND_SERVICE, 561 PACKAGE_NAME_APP3, PACKAGE_NAME_APP2, Context.BIND_INCLUDE_CAPABILITIES, 562 null); 563 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_BOUND_TOP); 564 565 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_UNBIND_SERVICE, 566 PACKAGE_NAME_APP3, PACKAGE_NAME_APP2, 0, null); 567 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 568 } 569 570 // unbind service. 571 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_STOP_ACTIVITY, 572 PACKAGE_NAME_APP3, PACKAGE_NAME_APP3, 0, null); 573 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_UNBIND_SERVICE, 574 PACKAGE_NAME_APP3, PACKAGE_NAME_APP2, 0, null); 575 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_UNBIND_SERVICE, 576 PACKAGE_NAME_APP2, PACKAGE_NAME_APP1, 0, null); 577 CommandReceiver.sendCommand(mContext, 578 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 579 PACKAGE_NAME_APP2, PACKAGE_NAME_APP2, 0, null); 580 581 } finally { 582 uid1Watcher.finish(); 583 uid2Watcher.finish(); 584 uid3Watcher.finish(); 585 allowBgActivityStart(PACKAGE_NAME_APP3, false); 586 } 587 } 588 589 /** 590 * Test FGS background startForeground() restriction, use DeviceConfig to turn on restriction. 591 * @throws Exception 592 */ 593 @Test testFgsStartFromBG1()594 public void testFgsStartFromBG1() throws Exception { 595 testFgsStartFromBG(true); 596 } 597 598 /** 599 * Test FGS background startForeground() restriction, use AppCompat CHANGE ID to turn on 600 * restriction. 601 * @throws Exception 602 */ 603 @Test testFgsStartFromBG2()604 public void testFgsStartFromBG2() throws Exception { 605 testFgsStartFromBG(false); 606 } 607 testFgsStartFromBG(boolean useDeviceConfig)608 private void testFgsStartFromBG(boolean useDeviceConfig) throws Exception { 609 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 610 PACKAGE_NAME_APP1, 0); 611 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 612 WAITFOR_MSEC); 613 try { 614 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 615 waiter.prepare(ACTION_START_FGS_RESULT); 616 // disable the FGS background startForeground() restriction. 617 enableFgsRestriction(false, true, null); 618 enableFgsRestriction(false, useDeviceConfig, PACKAGE_NAME_APP1); 619 // APP1 is in BG state, Start FGS in APP1. 620 CommandReceiver.sendCommand(mContext, 621 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 622 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 623 // APP1 is in STATE_FG_SERVICE. 624 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 625 waiter.doWait(WAITFOR_MSEC); 626 // stop FGS. 627 CommandReceiver.sendCommand(mContext, 628 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 629 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 630 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 631 632 // Enable the FGS background startForeground() restriction. 633 allowBgActivityStart(PACKAGE_NAME_APP1, false); 634 enableFgsRestriction(true, true, null); 635 enableFgsRestriction(true, useDeviceConfig, PACKAGE_NAME_APP1); 636 // Start FGS in BG state. 637 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 638 waiter.prepare(ACTION_START_FGS_RESULT); 639 CommandReceiver.sendCommand(mContext, 640 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 641 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 642 // APP1 does not enter FGS state 643 try { 644 waiter.doWait(WAITFOR_MSEC); 645 fail("Service should not enter foreground service state"); 646 } catch (Exception e) { 647 } 648 649 // Put APP1 in TOP state. 650 allowBgActivityStart(PACKAGE_NAME_APP1, true); 651 CommandReceiver.sendCommand(mContext, 652 CommandReceiver.COMMAND_START_ACTIVITY, 653 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 654 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_TOP); 655 allowBgActivityStart(PACKAGE_NAME_APP1, false); 656 657 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 658 waiter.prepare(ACTION_START_FGS_RESULT); 659 // Now it can start FGS. 660 CommandReceiver.sendCommand(mContext, 661 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 662 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 663 // Stop activity. 664 CommandReceiver.sendCommand(mContext, 665 CommandReceiver.COMMAND_STOP_ACTIVITY, 666 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 667 // FGS is still running. 668 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 669 waiter.doWait(WAITFOR_MSEC); 670 // Stop the FGS. 671 CommandReceiver.sendCommand(mContext, 672 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 673 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 674 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 675 } finally { 676 uid1Watcher.finish(); 677 } 678 } 679 680 /** 681 * Test a FGS can start from a process that is at BOUND_TOP state. 682 * @throws Exception 683 */ 684 @Test testFgsStartFromBoundTopState()685 public void testFgsStartFromBoundTopState() throws Exception { 686 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 687 PACKAGE_NAME_APP1, 0); 688 ApplicationInfo app2Info = mContext.getPackageManager().getApplicationInfo( 689 PACKAGE_NAME_APP2, 0); 690 ApplicationInfo app3Info = mContext.getPackageManager().getApplicationInfo( 691 PACKAGE_NAME_APP3, 0); 692 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 693 WAITFOR_MSEC); 694 WatchUidRunner uid2Watcher = new WatchUidRunner(mInstrumentation, app2Info.uid, 695 WAITFOR_MSEC); 696 WatchUidRunner uid3Watcher = new WatchUidRunner(mInstrumentation, app3Info.uid, 697 WAITFOR_MSEC); 698 try { 699 // Enable the FGS background startForeground() restriction. 700 enableFgsRestriction(true, true, null); 701 702 // Put APP1 in TOP state. 703 allowBgActivityStart(PACKAGE_NAME_APP1, true); 704 CommandReceiver.sendCommand(mContext, 705 CommandReceiver.COMMAND_START_ACTIVITY, 706 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 707 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_TOP); 708 709 // APP1 bound to service in APP2, APP2 get BOUND_TOP state. 710 CommandReceiver.sendCommand(mContext, 711 CommandReceiver.COMMAND_BIND_SERVICE, 712 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 713 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_BOUND_TOP); 714 715 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 716 waiter.prepare(ACTION_START_FGS_RESULT); 717 // APP2 can start FGS in APP3. 718 CommandReceiver.sendCommand(mContext, 719 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 720 PACKAGE_NAME_APP2, PACKAGE_NAME_APP3, 0, null); 721 uid3Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 722 waiter.doWait(WAITFOR_MSEC); 723 724 // Stop activity. 725 CommandReceiver.sendCommand(mContext, 726 CommandReceiver.COMMAND_STOP_ACTIVITY, 727 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 728 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 729 // unbind service. 730 CommandReceiver.sendCommand(mContext, 731 CommandReceiver.COMMAND_UNBIND_SERVICE, 732 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 733 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 734 // Stop the FGS. 735 CommandReceiver.sendCommand(mContext, 736 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 737 PACKAGE_NAME_APP2, PACKAGE_NAME_APP3, 0, null); 738 uid3Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 739 } finally { 740 uid1Watcher.finish(); 741 uid2Watcher.finish(); 742 uid3Watcher.finish(); 743 } 744 } 745 746 /** 747 * Test a FGS can start from a process that is at FOREGROUND_SERVICE state. 748 * @throws Exception 749 */ 750 @Test testFgsStartFromFgsState()751 public void testFgsStartFromFgsState() throws Exception { 752 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 753 PACKAGE_NAME_APP1, 0); 754 ApplicationInfo app2Info = mContext.getPackageManager().getApplicationInfo( 755 PACKAGE_NAME_APP2, 0); 756 ApplicationInfo app3Info = mContext.getPackageManager().getApplicationInfo( 757 PACKAGE_NAME_APP3, 0); 758 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 759 WAITFOR_MSEC); 760 WatchUidRunner uid2Watcher = new WatchUidRunner(mInstrumentation, app2Info.uid, 761 WAITFOR_MSEC); 762 WatchUidRunner uid3Watcher = new WatchUidRunner(mInstrumentation, app3Info.uid, 763 WAITFOR_MSEC); 764 try { 765 // Enable the FGS background startForeground() restriction. 766 enableFgsRestriction(true, true, null); 767 768 // Put APP1 in TOP state. 769 allowBgActivityStart(PACKAGE_NAME_APP1, true); 770 CommandReceiver.sendCommand(mContext, 771 CommandReceiver.COMMAND_START_ACTIVITY, 772 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 773 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_TOP); 774 775 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 776 waiter.prepare(ACTION_START_FGS_RESULT); 777 // APP1 can start FGS in APP2, APP2 gets FOREGROUND_SERVICE state. 778 CommandReceiver.sendCommand(mContext, 779 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 780 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 781 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 782 waiter.doWait(WAITFOR_MSEC); 783 784 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 785 waiter.prepare(ACTION_START_FGS_RESULT); 786 // APP2 can start FGS in APP3. 787 CommandReceiver.sendCommand(mContext, 788 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 789 PACKAGE_NAME_APP2, PACKAGE_NAME_APP3, 0, null); 790 uid3Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 791 waiter.doWait(WAITFOR_MSEC); 792 793 // Stop activity in APP1. 794 CommandReceiver.sendCommand(mContext, 795 CommandReceiver.COMMAND_STOP_ACTIVITY, 796 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 797 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 798 // Stop FGS in APP2. 799 CommandReceiver.sendCommand(mContext, 800 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 801 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 802 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 803 // Stop FGS in APP3. 804 CommandReceiver.sendCommand(mContext, 805 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 806 PACKAGE_NAME_APP3, PACKAGE_NAME_APP3, 0, null); 807 uid3Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 808 } finally { 809 uid1Watcher.finish(); 810 uid2Watcher.finish(); 811 uid3Watcher.finish(); 812 } 813 } 814 815 /** 816 * When the service is started by bindService() command, test when BG-FGS-launch 817 * restriction is disabled, FGS can start from background. 818 * @throws Exception 819 */ 820 @Test testFgsStartFromBGWithBind()821 public void testFgsStartFromBGWithBind() throws Exception { 822 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 823 PACKAGE_NAME_APP1, 0); 824 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 825 WAITFOR_MSEC); 826 827 try { 828 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 829 waiter.prepare(ACTION_START_FGSL_RESULT); 830 // APP1 is in BG state, bind FGSL in APP1 first. 831 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_BIND_FOREGROUND_SERVICE, 832 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 833 // Then start FGSL in APP1 834 enableFgsRestriction(false, true, null); 835 CommandReceiver.sendCommand(mContext, 836 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE_LOCATION, 837 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 838 // APP1 is in FGS state 839 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 840 waiter.doWait(WAITFOR_MSEC); 841 842 // stop FGS 843 CommandReceiver.sendCommand(mContext, 844 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE_LOCATION, 845 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 846 // unbind service. 847 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_UNBIND_SERVICE, 848 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 849 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 850 } finally { 851 uid1Watcher.finish(); 852 } 853 } 854 855 /** 856 * When the service is started by bindService() command, test when BG-FGS-launch 857 * restriction is enabled, FGS can NOT start from background. 858 * @throws Exception 859 */ 860 @Test testFgsStartFromBGWithBindWithRestriction()861 public void testFgsStartFromBGWithBindWithRestriction() throws Exception { 862 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 863 PACKAGE_NAME_APP1, 0); 864 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 865 WAITFOR_MSEC); 866 867 try { 868 enableFgsRestriction(true, true, null); 869 // APP1 is in BG state, bind FGSL in APP1 first. 870 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_BIND_FOREGROUND_SERVICE, 871 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 872 // Then start FGS in APP1 873 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 874 waiter.prepare(ACTION_START_FGS_RESULT); 875 CommandReceiver.sendCommand(mContext, 876 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 877 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 878 // APP1 does not enter FGS state 879 try { 880 waiter.doWait(WAITFOR_MSEC); 881 fail("Service should not enter foreground service state"); 882 } catch (Exception e) { 883 } 884 885 // stop FGS 886 CommandReceiver.sendCommand(mContext, 887 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 888 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 889 // unbind service. 890 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_UNBIND_SERVICE, 891 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 892 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 893 } finally { 894 uid1Watcher.finish(); 895 } 896 } 897 898 /** 899 * Test BIND_ALLOW_BACKGROUND_ACTIVITY_STARTS flag. 900 * Shell has START_ACTIVITIES_FROM_BACKGROUND permission, it can use this bind flag to 901 * pass BG-Activity-launch ability to APP2, then APP2 can start APP2 FGS from background. 902 */ 903 @Test testFgsBindingFlagActivity()904 public void testFgsBindingFlagActivity() throws Exception { 905 testFgsBindingFlag(Context.BIND_ALLOW_BACKGROUND_ACTIVITY_STARTS); 906 } 907 908 /** 909 * Test BIND_ALLOW_FOREGROUND_SERVICE_STARTS_FROM_BACKGROUND flag. 910 * Shell has START_FOREGROUND_SERVICES_FROM_BACKGROUND permission, it can use this bind flag to 911 * pass BG-FGS-launch ability to APP2, then APP2 can start APP3 FGS from background. 912 */ 913 @Test testFgsBindingFlagFGS()914 public void testFgsBindingFlagFGS() throws Exception { 915 testFgsBindingFlag(Context.BIND_ALLOW_FOREGROUND_SERVICE_STARTS_FROM_BACKGROUND); 916 } 917 918 /** 919 * Test no binding flag. 920 * Shell has START_FOREGROUND_SERVICES_FROM_BACKGROUND permission, without any bind flag, 921 * the BG-FGS-launch ability can be passed to APP2 by service binding, then APP2 can start 922 * APP3 FGS from background. 923 */ 924 @Test testFgsBindingFlagNone()925 public void testFgsBindingFlagNone() throws Exception { 926 testFgsBindingFlag(0); 927 } 928 testFgsBindingFlag(int bindingFlag)929 private void testFgsBindingFlag(int bindingFlag) throws Exception { 930 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 931 PACKAGE_NAME_APP1, 0); 932 ApplicationInfo app2Info = mContext.getPackageManager().getApplicationInfo( 933 PACKAGE_NAME_APP2, 0); 934 ApplicationInfo app3Info = mContext.getPackageManager().getApplicationInfo( 935 PACKAGE_NAME_APP3, 0); 936 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 937 WAITFOR_MSEC); 938 WatchUidRunner uid2Watcher = new WatchUidRunner(mInstrumentation, app2Info.uid, 939 WAITFOR_MSEC); 940 WatchUidRunner uid3Watcher = new WatchUidRunner(mInstrumentation, app3Info.uid, 941 WAITFOR_MSEC); 942 try { 943 // Enable the FGS background startForeground() restriction. 944 enableFgsRestriction(true, true, null); 945 946 // testapp is in background. 947 // testapp binds to service in APP2, APP2 still in background state. 948 final Intent intent = new Intent().setClassName( 949 PACKAGE_NAME_APP2, "android.app.stubs.LocalService"); 950 951 /* 952 final ServiceConnection connection = new ServiceConnection() { 953 @Override 954 public void onServiceConnected(ComponentName name, IBinder service) { 955 } 956 @Override 957 public void onServiceDisconnected(ComponentName name) { 958 } 959 }; 960 runWithShellPermissionIdentity(() -> { 961 mTargetContext.bindService(intent, connection, 962 Context.BIND_AUTO_CREATE | Context.BIND_WAIVE_PRIORITY); 963 }); 964 965 // APP2 can not start FGS in APP3. 966 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 967 waiter.prepare(ACTION_START_FGS_RESULT); 968 CommandReceiver.sendCommand(mContext, 969 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 970 PACKAGE_NAME_APP2, PACKAGE_NAME_APP3, 0, null); 971 try { 972 waiter.doWait(WAITFOR_MSEC); 973 fail("Service should not enter foreground service state"); 974 } catch (Exception e) { 975 } 976 977 // testapp unbind service in APP2. 978 runWithShellPermissionIdentity(() -> mTargetContext.unbindService(connection)); 979 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 980 */ 981 982 // testapp is in background. 983 // testapp binds to service in APP2 using the binding flag. 984 // APP2 still in background state. 985 final ServiceConnection connection2 = new ServiceConnection() { 986 @Override 987 public void onServiceConnected(ComponentName name, IBinder service) { 988 } 989 @Override 990 public void onServiceDisconnected(ComponentName name) { 991 } 992 }; 993 runWithShellPermissionIdentity(() -> mTargetContext.bindService(intent, connection2, 994 Context.BIND_AUTO_CREATE | Context.BIND_WAIVE_PRIORITY 995 | bindingFlag)); 996 997 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 998 waiter.prepare(ACTION_START_FGS_RESULT); 999 // Because the binding flag, 1000 // APP2 can start FGS from background. 1001 CommandReceiver.sendCommand(mContext, 1002 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1003 PACKAGE_NAME_APP2, PACKAGE_NAME_APP3, 0, null); 1004 uid3Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1005 waiter.doWait(WAITFOR_MSEC); 1006 1007 // testapp unbind service in APP2. 1008 runWithShellPermissionIdentity(() -> mTargetContext.unbindService(connection2)); 1009 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 1010 // Stop the FGS in APP3. 1011 CommandReceiver.sendCommand(mContext, 1012 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1013 PACKAGE_NAME_APP3, PACKAGE_NAME_APP3, 0, null); 1014 uid3Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 1015 } finally { 1016 uid1Watcher.finish(); 1017 uid2Watcher.finish(); 1018 uid3Watcher.finish(); 1019 } 1020 } 1021 1022 /** 1023 * Test a FGS can start from BG if the app has SYSTEM_ALERT_WINDOW permission. 1024 */ 1025 @Test testFgsStartSystemAlertWindow()1026 public void testFgsStartSystemAlertWindow() throws Exception { 1027 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1028 PACKAGE_NAME_APP1, 0); 1029 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1030 WAITFOR_MSEC); 1031 try { 1032 // Enable the FGS background startForeground() restriction. 1033 enableFgsRestriction(true, true, null); 1034 // Start FGS in BG state. 1035 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1036 waiter.prepare(ACTION_START_FGS_RESULT); 1037 CommandReceiver.sendCommand(mContext, 1038 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1039 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1040 // APP1 does not enter FGS state 1041 try { 1042 waiter.doWait(WAITFOR_MSEC); 1043 fail("Service should not enter foreground service state"); 1044 } catch (Exception e) { 1045 } 1046 1047 PermissionUtils.grantPermission( 1048 PACKAGE_NAME_APP1, android.Manifest.permission.SYSTEM_ALERT_WINDOW); 1049 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1050 waiter.prepare(ACTION_START_FGS_RESULT); 1051 // Now it can start FGS. 1052 CommandReceiver.sendCommand(mContext, 1053 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1054 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1055 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1056 waiter.doWait(WAITFOR_MSEC); 1057 // Stop the FGS. 1058 CommandReceiver.sendCommand(mContext, 1059 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1060 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1061 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 1062 } finally { 1063 uid1Watcher.finish(); 1064 } 1065 } 1066 1067 /** 1068 * Test a FGS can start from BG if the device is in retail demo mode. 1069 */ 1070 @Test 1071 // Change Settings.Global.DEVICE_DEMO_MODE on device may trigger other listener and put 1072 // the device in undesired state, for example, the battery charge level is set to 35% 1073 // permanently, ignore this test for now. 1074 @Ignore testFgsStartRetailDemoMode()1075 public void testFgsStartRetailDemoMode() throws Exception { 1076 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1077 PACKAGE_NAME_APP1, 0); 1078 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1079 WAITFOR_MSEC); 1080 runWithShellPermissionIdentity(()-> { 1081 mOrigDeviceDemoMode = Settings.Global.getInt(mContext.getContentResolver(), 1082 Settings.Global.DEVICE_DEMO_MODE, 0); }); 1083 1084 try { 1085 // Enable the FGS background startForeground() restriction. 1086 enableFgsRestriction(true, true, null); 1087 // Start FGS in BG state. 1088 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1089 waiter.prepare(ACTION_START_FGS_RESULT); 1090 CommandReceiver.sendCommand(mContext, 1091 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1092 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1093 // APP1 does not enter FGS state 1094 try { 1095 waiter.doWait(WAITFOR_MSEC); 1096 fail("Service should not enter foreground service state"); 1097 } catch (Exception e) { 1098 } 1099 1100 runWithShellPermissionIdentity(()-> { 1101 Settings.Global.putInt(mContext.getContentResolver(), 1102 Settings.Global.DEVICE_DEMO_MODE, 1); }); 1103 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1104 waiter.prepare(ACTION_START_FGS_RESULT); 1105 // Now it can start FGS. 1106 CommandReceiver.sendCommand(mContext, 1107 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1108 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1109 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1110 waiter.doWait(WAITFOR_MSEC); 1111 // Stop the FGS. 1112 CommandReceiver.sendCommand(mContext, 1113 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1114 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1115 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 1116 } finally { 1117 uid1Watcher.finish(); 1118 runWithShellPermissionIdentity(()-> { 1119 Settings.Global.putInt(mContext.getContentResolver(), 1120 Settings.Global.DEVICE_DEMO_MODE, mOrigDeviceDemoMode); }); 1121 } 1122 } 1123 1124 // At Context.startForegroundService() or Service.startForeground() calls, if the FGS is 1125 // restricted by background restriction and the app's targetSdkVersion is at least S, the 1126 // framework throws a ForegroundServiceStartNotAllowedException with error message. 1127 @Test 1128 @Ignore("The instrumentation is allowed to star FGS, it does not throw the exception") testFgsStartFromBGException()1129 public void testFgsStartFromBGException() throws Exception { 1130 ForegroundServiceStartNotAllowedException expectedException = null; 1131 final Intent intent = new Intent().setClassName( 1132 PACKAGE_NAME_APP1, "android.app.stubs.LocalForegroundService"); 1133 try { 1134 allowBgActivityStart("android.app.stubs", false); 1135 enableFgsRestriction(true, true, null); 1136 mContext.startForegroundService(intent); 1137 } catch (ForegroundServiceStartNotAllowedException e) { 1138 expectedException = e; 1139 } finally { 1140 mContext.stopService(intent); 1141 allowBgActivityStart("android.app.stubs", true); 1142 } 1143 String expectedMessage = "mAllowStartForeground false"; 1144 assertNotNull(expectedException); 1145 assertTrue(expectedException.getMessage().contains(expectedMessage)); 1146 } 1147 1148 /** 1149 * Test a FGS can start from BG if the app is in the DeviceIdleController's AllowList. 1150 */ 1151 @Test testFgsStartAllowList()1152 public void testFgsStartAllowList() throws Exception { 1153 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1154 PACKAGE_NAME_APP1, 0); 1155 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1156 WAITFOR_MSEC); 1157 try { 1158 // Enable the FGS background startForeground() restriction. 1159 enableFgsRestriction(true, true, null); 1160 // Start FGS in BG state. 1161 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1162 waiter.prepare(ACTION_START_FGS_RESULT); 1163 CommandReceiver.sendCommand(mContext, 1164 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1165 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1166 // APP1 does not enter FGS state 1167 try { 1168 waiter.doWait(WAITFOR_MSEC); 1169 fail("Service should not enter foreground service state"); 1170 } catch (Exception e) { 1171 } 1172 1173 // Add package to AllowList. 1174 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1175 "dumpsys deviceidle whitelist +" + PACKAGE_NAME_APP1); 1176 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1177 waiter.prepare(ACTION_START_FGS_RESULT); 1178 // Now it can start FGS. 1179 CommandReceiver.sendCommand(mContext, 1180 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1181 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1182 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1183 waiter.doWait(WAITFOR_MSEC); 1184 // Stop the FGS. 1185 CommandReceiver.sendCommand(mContext, 1186 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1187 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1188 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 1189 } finally { 1190 uid1Watcher.finish(); 1191 // Remove package from AllowList. 1192 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1193 "dumpsys deviceidle whitelist -" + PACKAGE_NAME_APP1); 1194 } 1195 } 1196 1197 /** 1198 * Test temp allowlist types in BroadcastOptions. 1199 */ 1200 @Test testTempAllowListType()1201 public void testTempAllowListType() throws Exception { 1202 testTempAllowListTypeInternal(TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_NOT_ALLOWED); 1203 testTempAllowListTypeInternal(TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_ALLOWED); 1204 } 1205 testTempAllowListTypeInternal(int type)1206 private void testTempAllowListTypeInternal(int type) throws Exception { 1207 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1208 PACKAGE_NAME_APP1, 0); 1209 ApplicationInfo app2Info = mContext.getPackageManager().getApplicationInfo( 1210 PACKAGE_NAME_APP2, 0); 1211 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1212 WAITFOR_MSEC); 1213 WatchUidRunner uid2Watcher = new WatchUidRunner(mInstrumentation, app2Info.uid, 1214 WAITFOR_MSEC); 1215 try { 1216 // Enable the FGS background startForeground() restriction. 1217 enableFgsRestriction(true, true, null); 1218 // Start FGS in BG state. 1219 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1220 waiter.prepare(ACTION_START_FGS_RESULT); 1221 CommandReceiver.sendCommand(mContext, 1222 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1223 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 1224 // APP1 does not enter FGS state 1225 try { 1226 waiter.doWait(WAITFOR_MSEC); 1227 fail("Service should not enter foreground service state"); 1228 } catch (Exception e) { 1229 } 1230 1231 // Now it can start FGS. 1232 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1233 waiter.prepare(ACTION_START_FGS_RESULT); 1234 runWithShellPermissionIdentity(()-> { 1235 final BroadcastOptions options = BroadcastOptions.makeBasic(); 1236 // setTemporaryAppAllowlist API requires 1237 // START_FOREGROUND_SERVICES_FROM_BACKGROUND permission. 1238 options.setTemporaryAppAllowlist(TEMP_ALLOWLIST_DURATION_MS, type, REASON_UNKNOWN, 1239 ""); 1240 // Must use Shell to issue this command because Shell has 1241 // START_FOREGROUND_SERVICES_FROM_BACKGROUND permission. 1242 CommandReceiver.sendCommandWithBroadcastOptions(mContext, 1243 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1244 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null, 1245 options.toBundle()); 1246 }); 1247 if (type == TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_ALLOWED) { 1248 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1249 waiter.doWait(WAITFOR_MSEC); 1250 // Stop the FGS. 1251 CommandReceiver.sendCommand(mContext, 1252 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1253 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 1254 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 1255 WatchUidRunner.STATE_CACHED_EMPTY); 1256 } else if (type == TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_NOT_ALLOWED) { 1257 // APP1 does not enter FGS state 1258 try { 1259 waiter.doWait(WAITFOR_MSEC); 1260 fail("Service should not enter foreground service state"); 1261 } catch (Exception e) { 1262 } 1263 } 1264 } finally { 1265 uid1Watcher.finish(); 1266 uid2Watcher.finish(); 1267 // Sleep 10 seconds to let the temp allowlist expire so it won't affect next test case. 1268 SystemClock.sleep(TEMP_ALLOWLIST_DURATION_MS); 1269 } 1270 1271 } 1272 1273 /** 1274 * Test a FGS can start from BG if the process had a visible activity recently. 1275 */ 1276 @LargeTest 1277 @Test testVisibleActivityGracePeriod()1278 public void testVisibleActivityGracePeriod() throws Exception { 1279 ApplicationInfo app2Info = mContext.getPackageManager().getApplicationInfo( 1280 PACKAGE_NAME_APP2, 0); 1281 WatchUidRunner uid2Watcher = new WatchUidRunner(mInstrumentation, app2Info.uid, 1282 WAITFOR_MSEC); 1283 final String namespaceActivityManager = "activity_manager"; 1284 final String keyFgToBgFgsGraceDuration = "fg_to_bg_fgs_grace_duration"; 1285 final long[] curFgToBgFgsGraceDuration = {-1}; 1286 try { 1287 // Enable the FGS background startForeground() restriction. 1288 enableFgsRestriction(true, true, null); 1289 // Allow bg actvity start from APP1. 1290 allowBgActivityStart(PACKAGE_NAME_APP1, true); 1291 1292 SystemUtil.runWithShellPermissionIdentity(() -> { 1293 curFgToBgFgsGraceDuration[0] = DeviceConfig.getInt( 1294 namespaceActivityManager, 1295 keyFgToBgFgsGraceDuration, -1); 1296 DeviceConfig.setProperty(namespaceActivityManager, 1297 keyFgToBgFgsGraceDuration, 1298 Long.toString(WAITFOR_MSEC), false); 1299 }); 1300 1301 testVisibleActivityGracePeriodInternal(uid2Watcher, "KEYCODE_HOME"); 1302 testVisibleActivityGracePeriodInternal(uid2Watcher, "KEYCODE_BACK"); 1303 } finally { 1304 uid2Watcher.finish(); 1305 // Remove package from AllowList. 1306 allowBgActivityStart(PACKAGE_NAME_APP1, false); 1307 if (curFgToBgFgsGraceDuration[0] >= 0) { 1308 SystemUtil.runWithShellPermissionIdentity(() -> { 1309 DeviceConfig.setProperty(namespaceActivityManager, 1310 keyFgToBgFgsGraceDuration, 1311 Long.toString(curFgToBgFgsGraceDuration[0]), false); 1312 }); 1313 } else { 1314 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1315 "device_config delete " + namespaceActivityManager 1316 + " " + keyFgToBgFgsGraceDuration); 1317 } 1318 } 1319 } 1320 testVisibleActivityGracePeriodInternal(WatchUidRunner uidWatcher, String keyCode)1321 private void testVisibleActivityGracePeriodInternal(WatchUidRunner uidWatcher, String keyCode) 1322 throws Exception { 1323 testVisibleActivityGracePeriodInternal(uidWatcher, keyCode, null, 1324 () -> uidWatcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 1325 WatchUidRunner.STATE_FG_SERVICE), true); 1326 1327 testVisibleActivityGracePeriodInternal(uidWatcher, keyCode, 1328 () -> SystemClock.sleep(WAITFOR_MSEC + 2000), // Wait for the grace period to expire 1329 () -> { 1330 try { 1331 uidWatcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 1332 WatchUidRunner.STATE_FG_SERVICE); 1333 fail("Service should not enter foreground service state"); 1334 } catch (Exception e) { 1335 // Expected. 1336 } 1337 }, false); 1338 } 1339 testVisibleActivityGracePeriodInternal(WatchUidRunner uidWatcher, String keyCode, Runnable prep, Runnable verifier, boolean stopFgs)1340 private void testVisibleActivityGracePeriodInternal(WatchUidRunner uidWatcher, 1341 String keyCode, Runnable prep, Runnable verifier, boolean stopFgs) throws Exception { 1342 // Put APP2 in TOP state. 1343 CommandReceiver.sendCommand(mContext, 1344 CommandReceiver.COMMAND_START_ACTIVITY, 1345 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 1346 uidWatcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_TOP); 1347 1348 // Take a nap to wait for the UI to settle down. 1349 SystemClock.sleep(2000); 1350 1351 // Now inject key event. 1352 CtsAppTestUtils.executeShellCmd(mInstrumentation, "input keyevent " + keyCode); 1353 1354 // It should go to the cached state. 1355 uidWatcher.waitFor(WatchUidRunner.CMD_CACHED, null); 1356 1357 if (prep != null) { 1358 prep.run(); 1359 } 1360 1361 // Start FGS from APP2. 1362 CommandReceiver.sendCommand(mContext, 1363 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1364 PACKAGE_NAME_APP2, PACKAGE_NAME_APP2, 0, null); 1365 1366 if (verifier != null) { 1367 verifier.run(); 1368 } 1369 1370 if (stopFgs) { 1371 // Stop the FGS. 1372 CommandReceiver.sendCommand(mContext, 1373 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1374 PACKAGE_NAME_APP2, PACKAGE_NAME_APP2, 0, null); 1375 uidWatcher.waitFor(WatchUidRunner.CMD_CACHED, null); 1376 } 1377 } 1378 1379 /** 1380 * After background service is started, after 10 seconds timeout, the startForeground() can 1381 * succeed or not depends on the service's app proc state. 1382 * Test starService() -> startForeground() 1383 */ 1384 @Test testStartForegroundTimeout()1385 public void testStartForegroundTimeout() throws Exception { 1386 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1387 PACKAGE_NAME_APP1, 0); 1388 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1389 WAITFOR_MSEC); 1390 try { 1391 // Enable the FGS background startForeground() restriction. 1392 enableFgsRestriction(true, true, null); 1393 setFgsStartForegroundTimeout(DEFAULT_FGS_START_FOREGROUND_TIMEOUT_MS); 1394 1395 // Put app to a TOP proc state. 1396 allowBgActivityStart(PACKAGE_NAME_APP1, true); 1397 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_START_ACTIVITY, 1398 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1399 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_TOP); 1400 allowBgActivityStart(PACKAGE_NAME_APP1, false); 1401 1402 // start background service. 1403 Bundle extras = LocalForegroundService.newCommand( 1404 LocalForegroundService.COMMAND_START_NO_FOREGROUND); 1405 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_START_SERVICE, 1406 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, extras); 1407 1408 // stop the activity. 1409 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_STOP_ACTIVITY, 1410 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1411 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_SERVICE); 1412 1413 // Sleep after the timeout DEFAULT_FGS_START_FOREGROUND_TIMEOUT_MS 1414 SystemClock.sleep(DEFAULT_FGS_START_FOREGROUND_TIMEOUT_MS + 1000); 1415 1416 extras = LocalForegroundService.newCommand( 1417 LocalForegroundService.COMMAND_START_FOREGROUND); 1418 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_START_SERVICE, 1419 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, extras); 1420 // APP1 does not enter FGS state 1421 // startForeground() is called after 10 seconds FgsStartForegroundTimeout. 1422 try { 1423 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1424 fail("Service should not enter foreground service state"); 1425 } catch (Exception e) { 1426 } 1427 1428 // Put app to a TOP proc state. 1429 allowBgActivityStart(PACKAGE_NAME_APP1, true); 1430 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_START_ACTIVITY, 1431 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1432 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 1433 WatchUidRunner.STATE_TOP, new Integer(PROCESS_CAPABILITY_ALL)); 1434 allowBgActivityStart(PACKAGE_NAME_APP1, false); 1435 1436 // Call startForeground(). 1437 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1438 waiter.prepare(ACTION_START_FGS_RESULT); 1439 extras = LocalForegroundService.newCommand( 1440 LocalForegroundService.COMMAND_START_FOREGROUND); 1441 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_START_SERVICE, 1442 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, extras); 1443 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_STOP_ACTIVITY, 1444 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1445 1446 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1447 waiter.doWait(WAITFOR_MSEC); 1448 1449 // Stop the FGS. 1450 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_STOP_SERVICE, 1451 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1452 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY, 1453 new Integer(PROCESS_CAPABILITY_NONE)); 1454 } finally { 1455 uid1Watcher.finish(); 1456 setFgsStartForegroundTimeout(DEFAULT_FGS_START_FOREGROUND_TIMEOUT_MS); 1457 } 1458 } 1459 1460 /** 1461 * After startForeground() and stopForeground(), the second startForeground() can succeed or not 1462 * depends on the service's app proc state. 1463 * Test startForegroundService() -> startForeground() -> stopForeground() -> startForeground(). 1464 */ 1465 @Test testSecondStartForeground()1466 public void testSecondStartForeground() throws Exception { 1467 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1468 PACKAGE_NAME_APP1, 0); 1469 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1470 WAITFOR_MSEC); 1471 try { 1472 // Enable the FGS background startForeground() restriction. 1473 enableFgsRestriction(true, true, null); 1474 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1475 waiter.prepare(ACTION_START_FGS_RESULT); 1476 // bypass bg-service-start restriction. 1477 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1478 "dumpsys deviceidle whitelist +" + PACKAGE_NAME_APP1); 1479 // start foreground service. 1480 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1481 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1482 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1483 waiter.doWait(WAITFOR_MSEC); 1484 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1485 "dumpsys deviceidle whitelist -" + PACKAGE_NAME_APP1); 1486 1487 // stopForeground() 1488 Bundle extras = LocalForegroundService.newCommand( 1489 LocalForegroundService.COMMAND_STOP_FOREGROUND_REMOVE_NOTIFICATION); 1490 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_START_SERVICE, 1491 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, extras); 1492 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_SERVICE, 1493 new Integer(PROCESS_CAPABILITY_NONE)); 1494 1495 // startForeground() again. 1496 extras = LocalForegroundService.newCommand( 1497 LocalForegroundService.COMMAND_START_FOREGROUND); 1498 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_START_SERVICE, 1499 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, extras); 1500 try { 1501 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1502 fail("Service should not enter foreground service state"); 1503 } catch (Exception e) { 1504 } 1505 1506 // Put app to a TOP proc state. 1507 allowBgActivityStart(PACKAGE_NAME_APP1, true); 1508 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_START_ACTIVITY, 1509 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1510 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_TOP, 1511 new Integer(PROCESS_CAPABILITY_ALL)); 1512 allowBgActivityStart(PACKAGE_NAME_APP1, false); 1513 1514 // Call startForeground() second time. 1515 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1516 waiter.prepare(ACTION_START_FGS_RESULT); 1517 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1518 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1519 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_STOP_ACTIVITY, 1520 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1521 1522 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE, 1523 LOCAL_SERVICE_PROCESS_CAPABILITY); 1524 waiter.doWait(WAITFOR_MSEC); 1525 1526 // Stop the FGS. 1527 CommandReceiver.sendCommand(mContext, CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1528 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1529 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY, 1530 new Integer(PROCESS_CAPABILITY_NONE)); 1531 } finally { 1532 uid1Watcher.finish(); 1533 } 1534 } 1535 1536 /** 1537 * Test OP_ACTIVATE_VPN and OP_ACTIVATE_PLATFORM_VPN are exempted from BG-FGS-launch 1538 * restriction. 1539 * @throws Exception 1540 */ 1541 @Test testFgsStartVpn()1542 public void testFgsStartVpn() throws Exception { 1543 testFgsStartVpnInternal("ACTIVATE_VPN"); 1544 testFgsStartVpnInternal("ACTIVATE_PLATFORM_VPN"); 1545 } 1546 testFgsStartVpnInternal(String vpnAppOp)1547 private void testFgsStartVpnInternal(String vpnAppOp) throws Exception { 1548 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1549 PACKAGE_NAME_APP1, 0); 1550 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1551 WAITFOR_MSEC); 1552 try { 1553 // Enable the FGS background startForeground() restriction. 1554 enableFgsRestriction(true, true, null); 1555 // Start FGS in BG state. 1556 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1557 waiter.prepare(ACTION_START_FGS_RESULT); 1558 CommandReceiver.sendCommand(mContext, 1559 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1560 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1561 // APP1 does not enter FGS state 1562 try { 1563 waiter.doWait(WAITFOR_MSEC); 1564 fail("Service should not enter foreground service state"); 1565 } catch (Exception e) { 1566 } 1567 1568 setAppOp(PACKAGE_NAME_APP1, vpnAppOp, true); 1569 1570 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1571 waiter.prepare(ACTION_START_FGS_RESULT); 1572 // Now it can start FGS. 1573 CommandReceiver.sendCommand(mContext, 1574 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1575 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1576 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1577 waiter.doWait(WAITFOR_MSEC); 1578 // Stop the FGS. 1579 CommandReceiver.sendCommand(mContext, 1580 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1581 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1582 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 1583 } finally { 1584 uid1Watcher.finish(); 1585 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1586 "appops reset " + PACKAGE_NAME_APP1); 1587 } 1588 } 1589 1590 /** 1591 * The default behavior for temp allowlist reasonCode REASON_PUSH_MESSAGING_OVER_QUOTA 1592 * is TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_NOT_ALLOWED (not allowed to start FGS). But 1593 * the behavior can be changed by device config command. There are three possible values: 1594 * {@link TEMPORARY_ALLOW_LIST_TYPE_NONE} (-1): 1595 * not temp allowlisted. 1596 * {@link TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_ALLOWED} (0): 1597 * temp allowlisted and allow FGS. 1598 * {@link TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_NOT_ALLOWED} (1): 1599 * temp allowlisted, not allow FGS. 1600 * @throws Exception 1601 */ 1602 @Test testPushMessagingOverQuota()1603 public void testPushMessagingOverQuota() throws Exception { 1604 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1605 PACKAGE_NAME_APP1, 0); 1606 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1607 WAITFOR_MSEC); 1608 final int defaultBehavior = getPushMessagingOverQuotaBehavior(); 1609 try { 1610 // Enable the FGS background startForeground() restriction. 1611 enableFgsRestriction(true, true, null); 1612 // Default behavior is TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_NOT_ALLOWED. 1613 setPushMessagingOverQuotaBehavior( 1614 TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_NOT_ALLOWED); 1615 // Start FGS in BG state. 1616 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1617 waiter.prepare(ACTION_START_FGS_RESULT); 1618 CommandReceiver.sendCommand(mContext, 1619 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1620 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1621 // APP1 does not enter FGS state 1622 try { 1623 waiter.doWait(WAITFOR_MSEC); 1624 fail("Service should not enter foreground service state"); 1625 } catch (Exception e) { 1626 } 1627 1628 setPushMessagingOverQuotaBehavior(TEMPORARY_ALLOW_LIST_TYPE_NONE); 1629 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1630 waiter.prepare(ACTION_START_FGS_RESULT); 1631 runWithShellPermissionIdentity(() -> { 1632 mContext.getSystemService(PowerExemptionManager.class).addToTemporaryAllowList( 1633 PACKAGE_NAME_APP1, PowerExemptionManager.REASON_PUSH_MESSAGING_OVER_QUOTA, 1634 "", TEMP_ALLOWLIST_DURATION_MS); 1635 }); 1636 CommandReceiver.sendCommand(mContext, 1637 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1638 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1639 // APP1 does not enter FGS state 1640 try { 1641 waiter.doWait(WAITFOR_MSEC); 1642 fail("Service should not enter foreground service state"); 1643 } catch (Exception e) { 1644 } 1645 1646 // Change behavior to TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_ALLOWED. 1647 setPushMessagingOverQuotaBehavior( 1648 TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_ALLOWED); 1649 runWithShellPermissionIdentity(() -> { 1650 mContext.getSystemService(PowerExemptionManager.class).addToTemporaryAllowList( 1651 PACKAGE_NAME_APP1, PowerExemptionManager.REASON_PUSH_MESSAGING_OVER_QUOTA, 1652 "", TEMP_ALLOWLIST_DURATION_MS); 1653 }); 1654 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1655 waiter.prepare(ACTION_START_FGS_RESULT); 1656 // Now it can start FGS. 1657 CommandReceiver.sendCommand(mContext, 1658 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1659 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1660 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1661 waiter.doWait(WAITFOR_MSEC); 1662 // Stop the FGS. 1663 CommandReceiver.sendCommand(mContext, 1664 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1665 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1666 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 1667 } finally { 1668 uid1Watcher.finish(); 1669 // Change back to default behavior. 1670 setPushMessagingOverQuotaBehavior(defaultBehavior); 1671 // allow temp allowlist to expire. 1672 SystemClock.sleep(TEMP_ALLOWLIST_DURATION_MS); 1673 } 1674 } 1675 1676 /** 1677 * Test temp allowlist reasonCode in BroadcastOptions. 1678 * When REASON_PUSH_MESSAGING_OVER_QUOTA, DeviceIdleController changes temp allowlist type to 1679 * TEMPORARY_ALLOWLIST_TYPE_FOREGROUND_SERVICE_NOT_ALLOWED so FGS start is not allowed. 1680 * When REASON_DENIED (-1), DeviceIdleController changes temp allowlist type to 1681 * TEMPORARY_ALLOWLIST_TYPE_NONE, the temp allowlist itself is not allowed. 1682 * All other reason codes, DeviceIdleController does not change temp allowlist type. 1683 */ 1684 @Test testTempAllowListReasonCode()1685 public void testTempAllowListReasonCode() throws Exception { 1686 // FGS start is temp allowed. 1687 testTempAllowListReasonCodeInternal(REASON_PUSH_MESSAGING); 1688 // FGS start is not allowed. 1689 testTempAllowListReasonCodeInternal(REASON_PUSH_MESSAGING_OVER_QUOTA); 1690 // Temp allowlist itself is not allowed. REASON_DENIED is not exposed in 1691 // PowerExemptionManager, just use its value "-1" here. 1692 testTempAllowListReasonCodeInternal(-1); 1693 } 1694 testTempAllowListReasonCodeInternal(int reasonCode)1695 private void testTempAllowListReasonCodeInternal(int reasonCode) throws Exception { 1696 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1697 PACKAGE_NAME_APP1, 0); 1698 ApplicationInfo app2Info = mContext.getPackageManager().getApplicationInfo( 1699 PACKAGE_NAME_APP2, 0); 1700 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1701 WAITFOR_MSEC); 1702 WatchUidRunner uid2Watcher = new WatchUidRunner(mInstrumentation, app2Info.uid, 1703 WAITFOR_MSEC); 1704 final int defaultBehavior = getPushMessagingOverQuotaBehavior(); 1705 try { 1706 setPushMessagingOverQuotaBehavior( 1707 TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_NOT_ALLOWED); 1708 // Enable the FGS background startForeground() restriction. 1709 enableFgsRestriction(true, true, null); 1710 // Now it can start FGS. 1711 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1712 waiter.prepare(ACTION_START_FGS_RESULT); 1713 runWithShellPermissionIdentity(()-> { 1714 final BroadcastOptions options = BroadcastOptions.makeBasic(); 1715 // setTemporaryAppAllowlist API requires 1716 // START_FOREGROUND_SERVICES_FROM_BACKGROUND permission. 1717 options.setTemporaryAppAllowlist(TEMP_ALLOWLIST_DURATION_MS, 1718 TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_ALLOWED, reasonCode, 1719 ""); 1720 // Must use Shell to issue this command because Shell has 1721 // START_FOREGROUND_SERVICES_FROM_BACKGROUND permission. 1722 CommandReceiver.sendCommandWithBroadcastOptions(mContext, 1723 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1724 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null, 1725 options.toBundle()); 1726 }); 1727 if (reasonCode == REASON_PUSH_MESSAGING) { 1728 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1729 waiter.doWait(WAITFOR_MSEC); 1730 // Stop the FGS. 1731 CommandReceiver.sendCommand(mContext, 1732 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1733 PACKAGE_NAME_APP1, PACKAGE_NAME_APP2, 0, null); 1734 uid2Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, 1735 WatchUidRunner.STATE_CACHED_EMPTY); 1736 } else if (reasonCode == REASON_PUSH_MESSAGING_OVER_QUOTA) { 1737 // APP1 does not enter FGS state 1738 try { 1739 waiter.doWait(WAITFOR_MSEC); 1740 fail("Service should not enter foreground service state"); 1741 } catch (Exception e) { 1742 } 1743 } 1744 } finally { 1745 uid1Watcher.finish(); 1746 uid2Watcher.finish(); 1747 setPushMessagingOverQuotaBehavior(defaultBehavior); 1748 // Sleep to let the temp allowlist expire so it won't affect next test case. 1749 SystemClock.sleep(TEMP_ALLOWLIST_DURATION_MS); 1750 } 1751 } 1752 1753 /** 1754 * Test default_input_method is exempted from BG-FGS-start restriction. 1755 * @throws Exception 1756 */ 1757 @Test testFgsStartInputMethod()1758 public void testFgsStartInputMethod() throws Exception { 1759 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1760 PACKAGE_NAME_APP1, 0); 1761 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1762 WAITFOR_MSEC); 1763 final String defaultInputMethod = CtsAppTestUtils.executeShellCmd(mInstrumentation, 1764 "settings get --user current secure default_input_method"); 1765 try { 1766 // Enable the FGS background startForeground() restriction. 1767 enableFgsRestriction(true, true, null); 1768 // Start FGS in BG state. 1769 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1770 waiter.prepare(ACTION_START_FGS_RESULT); 1771 CommandReceiver.sendCommand(mContext, 1772 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1773 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1774 // APP1 does not enter FGS state 1775 try { 1776 waiter.doWait(WAITFOR_MSEC); 1777 fail("Service should not enter foreground service state"); 1778 } catch (Exception e) { 1779 } 1780 1781 // Change default_input_method to PACKAGE_NAME_APP1. 1782 final ComponentName cn = new ComponentName(PACKAGE_NAME_APP1, "xxx"); 1783 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1784 "settings put --user current secure default_input_method " 1785 + cn.flattenToShortString()); 1786 1787 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1788 waiter.prepare(ACTION_START_FGS_RESULT); 1789 // Now it can start FGS. 1790 CommandReceiver.sendCommand(mContext, 1791 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1792 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1793 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1794 waiter.doWait(WAITFOR_MSEC); 1795 // Stop the FGS. 1796 CommandReceiver.sendCommand(mContext, 1797 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1798 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1799 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 1800 } finally { 1801 uid1Watcher.finish(); 1802 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1803 "settings put --user current secure default_input_method " 1804 + defaultInputMethod); 1805 } 1806 } 1807 1808 /** 1809 * When PowerExemptionManager.addToTemporaryAllowList() is called more than one time, the second 1810 * call can extend the duration of the first call if the first call has not expired yet. 1811 * @throws Exception 1812 */ 1813 @Test testOverlappedTempAllowList()1814 public void testOverlappedTempAllowList() throws Exception { 1815 ApplicationInfo app1Info = mContext.getPackageManager().getApplicationInfo( 1816 PACKAGE_NAME_APP1, 0); 1817 WatchUidRunner uid1Watcher = new WatchUidRunner(mInstrumentation, app1Info.uid, 1818 WAITFOR_MSEC); 1819 try { 1820 // Enable the FGS background startForeground() restriction. 1821 enableFgsRestriction(true, true, null); 1822 // Start FGS in BG state. 1823 WaitForBroadcast waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1824 waiter.prepare(ACTION_START_FGS_RESULT); 1825 CommandReceiver.sendCommand(mContext, 1826 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1827 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1828 // APP1 does not enter FGS state 1829 try { 1830 waiter.doWait(WAITFOR_MSEC); 1831 fail("Service should not enter foreground service state"); 1832 } catch (Exception e) { 1833 } 1834 1835 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1836 waiter.prepare(ACTION_START_FGS_RESULT); 1837 runWithShellPermissionIdentity(() -> { 1838 mContext.getSystemService(PowerExemptionManager.class).addToTemporaryAllowList( 1839 PACKAGE_NAME_APP1, PowerExemptionManager.REASON_PUSH_MESSAGING, 1840 "", 10000); 1841 }); 1842 1843 SystemClock.sleep(5000); 1844 runWithShellPermissionIdentity(() -> { 1845 mContext.getSystemService(PowerExemptionManager.class).addToTemporaryAllowList( 1846 PACKAGE_NAME_APP1, PowerExemptionManager.REASON_PUSH_MESSAGING, 1847 "", 10000); 1848 }); 1849 SystemClock.sleep(5000); 1850 1851 // The first addToTemporaryAllowList()'s 10000ms duration has expired. 1852 // Now FGS start is allowed by second addToTemporaryAllowList()'s 10000ms duration. 1853 waiter = new WaitForBroadcast(mInstrumentation.getTargetContext()); 1854 waiter.prepare(ACTION_START_FGS_RESULT); 1855 // Now it can start FGS. 1856 CommandReceiver.sendCommand(mContext, 1857 CommandReceiver.COMMAND_START_FOREGROUND_SERVICE, 1858 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1859 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_FG_SERVICE); 1860 waiter.doWait(WAITFOR_MSEC); 1861 // Stop the FGS. 1862 CommandReceiver.sendCommand(mContext, 1863 CommandReceiver.COMMAND_STOP_FOREGROUND_SERVICE, 1864 PACKAGE_NAME_APP1, PACKAGE_NAME_APP1, 0, null); 1865 uid1Watcher.waitFor(WatchUidRunner.CMD_PROCSTATE, WatchUidRunner.STATE_CACHED_EMPTY); 1866 } finally { 1867 uid1Watcher.finish(); 1868 // allow temp allowlist to expire. 1869 SystemClock.sleep(5000); 1870 } 1871 } 1872 1873 /** 1874 * Turn on the FGS BG-launch restriction. DeviceConfig can turn on restriction on the whole 1875 * device (across all apps). AppCompat can turn on restriction on a single app package. 1876 * @param enable true to turn on restriction, false to turn off. 1877 * @param useDeviceConfig true to use DeviceConfig, false to use AppCompat CHANGE ID. 1878 * @param packageName the packageName if using AppCompat CHANGE ID. 1879 * @throws Exception 1880 */ enableFgsRestriction(boolean enable, boolean useDeviceConfig, String packageName)1881 private void enableFgsRestriction(boolean enable, boolean useDeviceConfig, String packageName) 1882 throws Exception { 1883 if (useDeviceConfig) { 1884 runWithShellPermissionIdentity(() -> { 1885 DeviceConfig.setProperty("activity_manager", 1886 KEY_DEFAULT_FGS_STARTS_RESTRICTION_ENABLED, 1887 Boolean.toString(enable), false); 1888 } 1889 ); 1890 } else { 1891 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1892 "am compat " + (enable ? "enable" : "disable") 1893 + " FGS_BG_START_RESTRICTION_CHANGE_ID " + packageName); 1894 } 1895 } 1896 1897 /** 1898 * Clean up the FGS BG-launch restriction. 1899 * @param packageName the packageName that will have its changeid override reset. 1900 * @throws Exception 1901 */ resetFgsRestriction(String packageName)1902 private void resetFgsRestriction(String packageName) 1903 throws Exception { 1904 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1905 "am compat reset FGS_BG_START_RESTRICTION_CHANGE_ID " + packageName); 1906 } 1907 1908 /** 1909 * SYSTEM_ALERT_WINDOW permission will allow both BG-activity start and BG-FGS start. 1910 * Some cases we want to grant this permission to allow activity start to bring the app up to 1911 * TOP state. 1912 * Some cases we want to revoke this permission to test other BG-FGS-launch exemptions. 1913 * @param packageName 1914 * @param allow 1915 * @throws Exception 1916 */ allowBgActivityStart(String packageName, boolean allow)1917 private void allowBgActivityStart(String packageName, boolean allow) throws Exception { 1918 if (allow) { 1919 PermissionUtils.grantPermission( 1920 packageName, android.Manifest.permission.SYSTEM_ALERT_WINDOW); 1921 } else { 1922 PermissionUtils.revokePermission( 1923 packageName, android.Manifest.permission.SYSTEM_ALERT_WINDOW); 1924 } 1925 } 1926 setFgsStartForegroundTimeout(int timeoutMs)1927 private void setFgsStartForegroundTimeout(int timeoutMs) throws Exception { 1928 runWithShellPermissionIdentity(() -> { 1929 DeviceConfig.setProperty("activity_manager", 1930 KEY_FGS_START_FOREGROUND_TIMEOUT, 1931 Integer.toString(timeoutMs), false); 1932 } 1933 ); 1934 } 1935 setAppOp(String packageName, String opStr, boolean allow)1936 private void setAppOp(String packageName, String opStr, boolean allow) throws Exception { 1937 CtsAppTestUtils.executeShellCmd(mInstrumentation, 1938 "appops set " + packageName + " " + opStr + " " 1939 + (allow ? "allow" : "deny")); 1940 } 1941 setPushMessagingOverQuotaBehavior( int type)1942 private void setPushMessagingOverQuotaBehavior( 1943 /* @PowerExemptionManager.TempAllowListType */ int type) throws Exception { 1944 runWithShellPermissionIdentity(() -> { 1945 DeviceConfig.setProperty("activity_manager", 1946 KEY_PUSH_MESSAGING_OVER_QUOTA_BEHAVIOR, 1947 Integer.toString(type), false); 1948 } 1949 ); 1950 // Sleep 2 seconds to allow the device config change to be applied. 1951 SystemClock.sleep(2000); 1952 } 1953 getPushMessagingOverQuotaBehavior()1954 private int getPushMessagingOverQuotaBehavior() throws Exception { 1955 final String defaultBehaviorStr = CtsAppTestUtils.executeShellCmd(mInstrumentation, 1956 "device_config get activity_manager " 1957 + KEY_PUSH_MESSAGING_OVER_QUOTA_BEHAVIOR).trim(); 1958 int defaultBehavior = TEMPORARY_ALLOW_LIST_TYPE_FOREGROUND_SERVICE_NOT_ALLOWED; 1959 if (!defaultBehaviorStr.equals("null")) { 1960 try { 1961 defaultBehavior = Integer.parseInt(defaultBehaviorStr); 1962 } catch (NumberFormatException e) { 1963 } 1964 } 1965 return defaultBehavior; 1966 } 1967 } 1968