/* * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.app; import static android.app.admin.DevicePolicyResources.Drawables.Style.SOLID_COLORED; import static android.app.admin.DevicePolicyResources.Drawables.Style.SOLID_NOT_COLORED; import static android.app.admin.DevicePolicyResources.Drawables.WORK_PROFILE_ICON; import static android.app.admin.DevicePolicyResources.Drawables.WORK_PROFILE_ICON_BADGE; import static android.app.admin.DevicePolicyResources.UNDEFINED; import static android.content.pm.Checksum.TYPE_PARTIAL_MERKLE_ROOT_1M_SHA256; import static android.content.pm.Checksum.TYPE_PARTIAL_MERKLE_ROOT_1M_SHA512; import static android.content.pm.Checksum.TYPE_WHOLE_MD5; import static android.content.pm.Checksum.TYPE_WHOLE_MERKLE_ROOT_4K_SHA256; import static android.content.pm.Checksum.TYPE_WHOLE_SHA1; import static android.content.pm.Checksum.TYPE_WHOLE_SHA256; import static android.content.pm.Checksum.TYPE_WHOLE_SHA512; import android.annotation.CallbackExecutor; import android.annotation.DrawableRes; import android.annotation.NonNull; import android.annotation.Nullable; import android.annotation.StringRes; import android.annotation.UserIdInt; import android.annotation.XmlRes; import android.app.admin.DevicePolicyManager; import android.app.role.RoleManager; import android.compat.annotation.UnsupportedAppUsage; import android.content.ComponentName; import android.content.ContentResolver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.IntentSender; import android.content.pm.ActivityInfo; import android.content.pm.ApkChecksum; import android.content.pm.ApplicationInfo; import android.content.pm.ArchivedPackageInfo; import android.content.pm.ChangedPackages; import android.content.pm.Checksum; import android.content.pm.ComponentInfo; import android.content.pm.FeatureInfo; import android.content.pm.IOnChecksumsReadyListener; import android.content.pm.IPackageDataObserver; import android.content.pm.IPackageDeleteObserver; import android.content.pm.IPackageManager; import android.content.pm.IPackageMoveObserver; import android.content.pm.IPackageStatsObserver; import android.content.pm.InstallSourceInfo; import android.content.pm.InstantAppInfo; import android.content.pm.InstrumentationInfo; import android.content.pm.IntentFilterVerificationInfo; import android.content.pm.KeySet; import android.content.pm.ModuleInfo; import android.content.pm.PackageInfo; import android.content.pm.PackageInstaller; import android.content.pm.PackageItemInfo; import android.content.pm.PackageManager; import android.content.pm.ParceledListSlice; import android.content.pm.PermissionGroupInfo; import android.content.pm.PermissionInfo; import android.content.pm.ProviderInfo; import android.content.pm.ResolveInfo; import android.content.pm.ServiceInfo; import android.content.pm.SharedLibraryInfo; import android.content.pm.SuspendDialogInfo; import android.content.pm.VerifierDeviceIdentity; import android.content.pm.VersionedPackage; import android.content.pm.dex.ArtManager; import android.content.pm.parsing.ApkLiteParseUtils; import android.content.res.ApkAssets; import android.content.res.Configuration; import android.content.res.Resources; import android.content.res.TypedArray; import android.content.res.XmlResourceParser; import android.graphics.Bitmap; import android.graphics.Canvas; import android.graphics.Rect; import android.graphics.drawable.BitmapDrawable; import android.graphics.drawable.Drawable; import android.graphics.drawable.LayerDrawable; import android.os.Build; import android.os.Bundle; import android.os.Handler; import android.os.IRemoteCallback; import android.os.Looper; import android.os.Message; import android.os.ParcelFileDescriptor; import android.os.ParcelableException; import android.os.PersistableBundle; import android.os.Process; import android.os.RemoteException; import android.os.StrictMode; import android.os.SystemProperties; import android.os.UserHandle; import android.os.UserManager; import android.os.storage.StorageManager; import android.os.storage.VolumeInfo; import android.permission.PermissionControllerManager; import android.permission.PermissionManager; import android.provider.Settings; import android.system.ErrnoException; import android.system.Os; import android.system.OsConstants; import android.system.StructStat; import android.text.TextUtils; import android.util.ArrayMap; import android.util.ArraySet; import android.util.AttributeSet; import android.util.LauncherIcons; import android.util.Log; import android.util.Slog; import android.util.Xml; import com.android.internal.annotations.GuardedBy; import com.android.internal.annotations.Immutable; import com.android.internal.annotations.VisibleForTesting; import com.android.internal.os.SomeArgs; import com.android.internal.util.UserIcons; import dalvik.system.VMRuntime; import libcore.util.EmptyArray; import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlPullParserException; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.lang.ref.WeakReference; import java.security.cert.Certificate; import java.security.cert.CertificateEncodingException; import java.security.cert.X509Certificate; import java.util.ArrayList; import java.util.Arrays; import java.util.Collections; import java.util.Iterator; import java.util.List; import java.util.Objects; import java.util.Set; import java.util.concurrent.Executor; import java.util.function.Consumer; import java.util.function.Function; /** @hide */ public class ApplicationPackageManager extends PackageManager { private static final String TAG = "ApplicationPackageManager"; private static final boolean DEBUG_ICONS = false; private static final int DEFAULT_EPHEMERAL_COOKIE_MAX_SIZE_BYTES = 16384; // 16KB // Default flags to use with PackageManager when no flags are given. private static final int sDefaultFlags = GET_SHARED_LIBRARY_FILES; /** Default set of checksums - includes all available checksums. * @see PackageManager#requestChecksums */ private static final int DEFAULT_CHECKSUMS = TYPE_WHOLE_MERKLE_ROOT_4K_SHA256 | TYPE_WHOLE_MD5 | TYPE_WHOLE_SHA1 | TYPE_WHOLE_SHA256 | TYPE_WHOLE_SHA512 | TYPE_PARTIAL_MERKLE_ROOT_1M_SHA256 | TYPE_PARTIAL_MERKLE_ROOT_1M_SHA512; // Name of the resource which provides background permission button string public static final String APP_PERMISSION_BUTTON_ALLOW_ALWAYS = "app_permission_button_allow_always"; // Name of the package which the permission controller's resources are in. public static final String PERMISSION_CONTROLLER_RESOURCE_PACKAGE = "com.android.permissioncontroller"; private volatile UserManager mUserManager; private volatile PermissionManager mPermissionManager; private volatile PackageInstaller mInstaller; private volatile ArtManager mArtManager; private volatile DevicePolicyManager mDevicePolicyManager; private volatile String mPermissionsControllerPackageName; @GuardedBy("mDelegates") private final ArrayList mDelegates = new ArrayList<>(); @NonNull @GuardedBy("mPackageMonitorCallbacks") private final ArraySet mPackageMonitorCallbacks = new ArraySet<>(); UserManager getUserManager() { if (mUserManager == null) { mUserManager = UserManager.get(mContext); } return mUserManager; } DevicePolicyManager getDevicePolicyManager() { if (mDevicePolicyManager == null) { mDevicePolicyManager = mContext.getSystemService(DevicePolicyManager.class); } return mDevicePolicyManager; } private PermissionManager getPermissionManager() { if (mPermissionManager == null) { mPermissionManager = mContext.getSystemService(PermissionManager.class); } return mPermissionManager; } @Override public int getUserId() { return mContext.getUserId(); } @Override public PackageInfo getPackageInfo(String packageName, int flags) throws NameNotFoundException { return getPackageInfo(packageName, PackageInfoFlags.of(flags)); } @Override public PackageInfo getPackageInfo(String packageName, PackageInfoFlags flags) throws NameNotFoundException { return getPackageInfoAsUser(packageName, flags, getUserId()); } @Override public PackageInfo getPackageInfo(VersionedPackage versionedPackage, int flags) throws NameNotFoundException { return getPackageInfo(versionedPackage, PackageInfoFlags.of(flags)); } @Override public PackageInfo getPackageInfo(VersionedPackage versionedPackage, PackageInfoFlags flags) throws NameNotFoundException { final int userId = getUserId(); try { PackageInfo pi = mPM.getPackageInfoVersioned(versionedPackage, updateFlagsForPackage(flags.getValue(), userId), userId); if (pi != null) { return pi; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException(versionedPackage.toString()); } @Override public PackageInfo getPackageInfoAsUser(String packageName, int flags, int userId) throws NameNotFoundException { return getPackageInfoAsUser(packageName, PackageInfoFlags.of(flags), userId); } @Override public PackageInfo getPackageInfoAsUser(String packageName, PackageInfoFlags flags, int userId) throws NameNotFoundException { PackageInfo pi = getPackageInfoAsUserCached( packageName, updateFlagsForPackage(flags.getValue(), userId), userId); if (pi == null) { throw new NameNotFoundException(packageName); } return pi; } @Override public String[] currentToCanonicalPackageNames(String[] names) { try { return mPM.currentToCanonicalPackageNames(names); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public String[] canonicalToCurrentPackageNames(String[] names) { try { return mPM.canonicalToCurrentPackageNames(names); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public Intent getLaunchIntentForPackage(String packageName) { // First see if the package has an INFO activity; the existence of // such an activity is implied to be the desired front-door for the // overall package (such as if it has multiple launcher entries). Intent intentToResolve = new Intent(Intent.ACTION_MAIN); intentToResolve.addCategory(Intent.CATEGORY_INFO); intentToResolve.setPackage(packageName); List ris = queryIntentActivities(intentToResolve, 0); // Otherwise, try to find a main launcher activity. if (ris == null || ris.size() <= 0) { // reuse the intent instance intentToResolve.removeCategory(Intent.CATEGORY_INFO); intentToResolve.addCategory(Intent.CATEGORY_LAUNCHER); intentToResolve.setPackage(packageName); ris = queryIntentActivities(intentToResolve, 0); } if (ris == null || ris.size() <= 0) { return null; } Intent intent = new Intent(intentToResolve); intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); intent.setClassName(ris.get(0).activityInfo.packageName, ris.get(0).activityInfo.name); return intent; } @Override public Intent getLeanbackLaunchIntentForPackage(String packageName) { return getLaunchIntentForPackageAndCategory(packageName, Intent.CATEGORY_LEANBACK_LAUNCHER); } @Override public Intent getCarLaunchIntentForPackage(String packageName) { return getLaunchIntentForPackageAndCategory(packageName, Intent.CATEGORY_CAR_LAUNCHER); } private Intent getLaunchIntentForPackageAndCategory(String packageName, String category) { // Try to find a main launcher activity for the given categories. Intent intentToResolve = new Intent(Intent.ACTION_MAIN); intentToResolve.addCategory(category); intentToResolve.setPackage(packageName); List ris = queryIntentActivities(intentToResolve, 0); if (ris == null || ris.size() <= 0) { return null; } Intent intent = new Intent(intentToResolve); intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); intent.setClassName(ris.get(0).activityInfo.packageName, ris.get(0).activityInfo.name); return intent; } @Override public @NonNull IntentSender getLaunchIntentSenderForPackage(@NonNull String packageName) { try { return mPM.getLaunchIntentSenderForPackage(packageName, mContext.getPackageName(), mContext.getAttributionTag(), getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public int[] getPackageGids(String packageName) throws NameNotFoundException { return getPackageGids(packageName, 0); } @Override public int[] getPackageGids(String packageName, int flags) throws NameNotFoundException { return getPackageGids(packageName, PackageInfoFlags.of(flags)); } @Override public int[] getPackageGids(String packageName, PackageInfoFlags flags) throws NameNotFoundException { final int userId = getUserId(); try { int[] gids = mPM.getPackageGids(packageName, updateFlagsForPackage(flags.getValue(), userId), userId); if (gids != null) { return gids; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException(packageName); } @Override public int getPackageUid(String packageName, int flags) throws NameNotFoundException { return getPackageUid(packageName, PackageInfoFlags.of(flags)); } @Override public int getPackageUid(String packageName, PackageInfoFlags flags) throws NameNotFoundException { return getPackageUidAsUser(packageName, flags, getUserId()); } @Override public int getPackageUidAsUser(String packageName, int userId) throws NameNotFoundException { return getPackageUidAsUser(packageName, 0, userId); } @Override public int getPackageUidAsUser(String packageName, int flags, int userId) throws NameNotFoundException { return getPackageUidAsUser(packageName, PackageInfoFlags.of(flags), userId); } @Override public int getPackageUidAsUser(String packageName, PackageInfoFlags flags, int userId) throws NameNotFoundException { try { int uid = mPM.getPackageUid(packageName, updateFlagsForPackage(flags.getValue(), userId), userId); if (uid >= 0) { return uid; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException(packageName); } @Override @SuppressWarnings("unchecked") public List getAllPermissionGroups(int flags) { return getPermissionManager().getAllPermissionGroups(flags); } @Override public PermissionGroupInfo getPermissionGroupInfo(String groupName, int flags) throws NameNotFoundException { final PermissionGroupInfo permissionGroupInfo = getPermissionManager() .getPermissionGroupInfo(groupName, flags); if (permissionGroupInfo == null) { throw new NameNotFoundException(groupName); } return permissionGroupInfo; } @Override public PermissionInfo getPermissionInfo(String permName, int flags) throws NameNotFoundException { final PermissionInfo permissionInfo = getPermissionManager().getPermissionInfo(permName, flags); if (permissionInfo == null) { throw new NameNotFoundException(permName); } return permissionInfo; } @Override @SuppressWarnings("unchecked") public List queryPermissionsByGroup(String groupName, int flags) throws NameNotFoundException { final List permissionInfos = getPermissionManager().queryPermissionsByGroup( groupName, flags); if (permissionInfos == null) { throw new NameNotFoundException(groupName); } return permissionInfos; } @Override public void getPlatformPermissionsForGroup(@NonNull String permissionGroupName, @NonNull @CallbackExecutor Executor executor, @NonNull Consumer> callback) { final PermissionControllerManager permissionControllerManager = mContext.getSystemService( PermissionControllerManager.class); permissionControllerManager.getPlatformPermissionsForGroup(permissionGroupName, executor, callback); } @Override public void getGroupOfPlatformPermission(@NonNull String permissionName, @NonNull @CallbackExecutor Executor executor, @NonNull Consumer callback) { final PermissionControllerManager permissionControllerManager = mContext.getSystemService( PermissionControllerManager.class); permissionControllerManager.getGroupOfPlatformPermission(permissionName, executor, callback); } @Override public boolean arePermissionsIndividuallyControlled() { return mContext.getResources().getBoolean( com.android.internal.R.bool.config_permissionsIndividuallyControlled); } @Override public boolean isWirelessConsentModeEnabled() { return mContext.getResources().getBoolean( com.android.internal.R.bool.config_wirelessConsentRequired); } @Override public ApplicationInfo getApplicationInfo(String packageName, int flags) throws NameNotFoundException { return getApplicationInfo(packageName, ApplicationInfoFlags.of(flags)); } @Override public ApplicationInfo getApplicationInfo(String packageName, ApplicationInfoFlags flags) throws NameNotFoundException { return getApplicationInfoAsUser(packageName, flags, getUserId()); } @Override public ApplicationInfo getApplicationInfoAsUser(String packageName, int flags, int userId) throws NameNotFoundException { return getApplicationInfoAsUser(packageName, ApplicationInfoFlags.of(flags), userId); } @Override public ApplicationInfo getApplicationInfoAsUser(String packageName, ApplicationInfoFlags flags, int userId) throws NameNotFoundException { ApplicationInfo ai = getApplicationInfoAsUserCached( packageName, updateFlagsForApplication(flags.getValue(), userId), userId); if (ai == null) { throw new NameNotFoundException(packageName); } return maybeAdjustApplicationInfo(ai); } private static ApplicationInfo maybeAdjustApplicationInfo(ApplicationInfo info) { // If we're dealing with a multi-arch application that has both // 32 and 64 bit shared libraries, we might need to choose the secondary // depending on what the current runtime's instruction set is. if (info.primaryCpuAbi != null && info.secondaryCpuAbi != null) { final String runtimeIsa = VMRuntime.getRuntime().vmInstructionSet(); // Get the instruction set that the libraries of secondary Abi is supported. // In presence of a native bridge this might be different than the one secondary Abi used. String secondaryIsa = VMRuntime.getInstructionSet(info.secondaryCpuAbi); final String secondaryDexCodeIsa = SystemProperties.get("ro.dalvik.vm.isa." + secondaryIsa); secondaryIsa = secondaryDexCodeIsa.isEmpty() ? secondaryIsa : secondaryDexCodeIsa; // If the runtimeIsa is the same as the primary isa, then we do nothing. // Everything will be set up correctly because info.nativeLibraryDir will // correspond to the right ISA. if (runtimeIsa.equals(secondaryIsa)) { ApplicationInfo modified = new ApplicationInfo(info); modified.nativeLibraryDir = info.secondaryNativeLibraryDir; return modified; } } return info; } @Override public int getTargetSdkVersion(@NonNull String packageName) throws NameNotFoundException { try { int version = mPM.getTargetSdkVersion(packageName); if (version != -1) { return version; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new PackageManager.NameNotFoundException(packageName); } @Override public ActivityInfo getActivityInfo(ComponentName className, int flags) throws NameNotFoundException { return getActivityInfo(className, ComponentInfoFlags.of(flags)); } @Override public ActivityInfo getActivityInfo(ComponentName className, ComponentInfoFlags flags) throws NameNotFoundException { final int userId = getUserId(); try { ActivityInfo ai = mPM.getActivityInfo(className, updateFlagsForComponent(flags.getValue(), userId, null), userId); if (ai != null) { return ai; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException(className.toString()); } @Override public ActivityInfo getReceiverInfo(ComponentName className, int flags) throws NameNotFoundException { return getReceiverInfo(className, ComponentInfoFlags.of(flags)); } @Override public ActivityInfo getReceiverInfo(ComponentName className, ComponentInfoFlags flags) throws NameNotFoundException { final int userId = getUserId(); try { ActivityInfo ai = mPM.getReceiverInfo(className, updateFlagsForComponent(flags.getValue(), userId, null), userId); if (ai != null) { return ai; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException(className.toString()); } @Override public ServiceInfo getServiceInfo(ComponentName className, int flags) throws NameNotFoundException { return getServiceInfo(className, ComponentInfoFlags.of(flags)); } @Override public ServiceInfo getServiceInfo(ComponentName className, ComponentInfoFlags flags) throws NameNotFoundException { final int userId = getUserId(); try { ServiceInfo si = mPM.getServiceInfo(className, updateFlagsForComponent(flags.getValue(), userId, null), userId); if (si != null) { return si; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException(className.toString()); } @Override public ProviderInfo getProviderInfo(ComponentName className, int flags) throws NameNotFoundException { return getProviderInfo(className, ComponentInfoFlags.of(flags)); } @Override public ProviderInfo getProviderInfo(ComponentName className, ComponentInfoFlags flags) throws NameNotFoundException { final int userId = getUserId(); try { ProviderInfo pi = mPM.getProviderInfo(className, updateFlagsForComponent(flags.getValue(), userId, null), userId); if (pi != null) { return pi; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException(className.toString()); } @Override public String[] getSystemSharedLibraryNames() { try { return mPM.getSystemSharedLibraryNames(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @Override public @NonNull List getSharedLibraries(int flags) { return this.getSharedLibraries(PackageInfoFlags.of(flags)); } /** @hide * @param flags */ @Override public @NonNull List getSharedLibraries(PackageInfoFlags flags) { return getSharedLibrariesAsUser(flags, getUserId()); } /** @hide */ @Override public @NonNull List getSharedLibrariesAsUser(int flags, int userId) { return getSharedLibrariesAsUser(PackageInfoFlags.of(flags), userId); } /** @hide */ @Override @SuppressWarnings("unchecked") public @NonNull List getSharedLibrariesAsUser(PackageInfoFlags flags, int userId) { try { ParceledListSlice sharedLibs = mPM.getSharedLibraries( mContext.getOpPackageName(), flags.getValue(), userId); if (sharedLibs == null) { return Collections.emptyList(); } return sharedLibs.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @NonNull @Override public List getDeclaredSharedLibraries(@NonNull String packageName, int flags) { return getDeclaredSharedLibraries(packageName, PackageInfoFlags.of(flags)); } @NonNull @Override @SuppressWarnings("unchecked") public List getDeclaredSharedLibraries(@NonNull String packageName, PackageInfoFlags flags) { try { ParceledListSlice sharedLibraries = mPM.getDeclaredSharedLibraries( packageName, flags.getValue(), mContext.getUserId()); return sharedLibraries != null ? sharedLibraries.getList() : Collections.emptyList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @Override public @NonNull String getServicesSystemSharedLibraryPackageName() { try { return mPM.getServicesSystemSharedLibraryPackageName(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** * @hide */ public @NonNull String getSharedSystemSharedLibraryPackageName() { try { return mPM.getSharedSystemSharedLibraryPackageName(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public ChangedPackages getChangedPackages(int sequenceNumber) { try { return mPM.getChangedPackages(sequenceNumber, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override @SuppressWarnings("unchecked") public FeatureInfo[] getSystemAvailableFeatures() { try { ParceledListSlice parceledList = mPM.getSystemAvailableFeatures(); if (parceledList == null) { return new FeatureInfo[0]; } final List list = parceledList.getList(); final FeatureInfo[] res = new FeatureInfo[list.size()]; for (int i = 0; i < res.length; i++) { res[i] = list.get(i); } return res; } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean hasSystemFeature(String name) { return hasSystemFeature(name, 0); } /** * Identifies a single hasSystemFeature query. */ @Immutable private static final class HasSystemFeatureQuery { public final String name; public final int version; public HasSystemFeatureQuery(String n, int v) { name = n; version = v; } @Override public String toString() { return String.format("HasSystemFeatureQuery(name=\"%s\", version=%d)", name, version); } @Override public boolean equals(@Nullable Object o) { if (o instanceof HasSystemFeatureQuery) { HasSystemFeatureQuery r = (HasSystemFeatureQuery) o; return Objects.equals(name, r.name) && version == r.version; } else { return false; } } @Override public int hashCode() { return Objects.hashCode(name) * 13 + version; } } // Make this cache relatively large. There are many system features and // none are ever invalidated. MPTS tests suggests that the cache should // hold at least 150 entries. private final static PropertyInvalidatedCache mHasSystemFeatureCache = new PropertyInvalidatedCache( 256, "cache_key.has_system_feature") { @Override public Boolean recompute(HasSystemFeatureQuery query) { try { return ActivityThread.currentActivityThread().getPackageManager(). hasSystemFeature(query.name, query.version); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } }; @Override public boolean hasSystemFeature(String name, int version) { return mHasSystemFeatureCache.query(new HasSystemFeatureQuery(name, version)); } /** @hide */ public void disableHasSystemFeatureCache() { mHasSystemFeatureCache.disableLocal(); } /** @hide */ public static void invalidateHasSystemFeatureCache() { mHasSystemFeatureCache.invalidateCache(); } @Override public int checkPermission(String permName, String pkgName) { return getPermissionManager().checkPackageNamePermission(permName, pkgName, mContext.getDeviceId(), getUserId()); } @Override public boolean isPermissionRevokedByPolicy(String permName, String pkgName) { return getPermissionManager().isPermissionRevokedByPolicy(pkgName, permName); } /** * @hide */ @Override public String getPermissionControllerPackageName() { if (mPermissionsControllerPackageName == null) { try { mPermissionsControllerPackageName = mPM.getPermissionControllerPackageName(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } return mPermissionsControllerPackageName; } /** * @hide */ @Override public String getSdkSandboxPackageName() { try { return mPM.getSdkSandboxPackageName(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean addPermission(PermissionInfo info) { return getPermissionManager().addPermission(info, false); } @Override public boolean addPermissionAsync(PermissionInfo info) { return getPermissionManager().addPermission(info, true); } @Override public void removePermission(String name) { getPermissionManager().removePermission(name); } @Override public void grantRuntimePermission(String packageName, String permissionName, UserHandle user) { getPermissionManager().grantRuntimePermission(packageName, permissionName, user); } @Override public void revokeRuntimePermission(String packageName, String permName, UserHandle user) { revokeRuntimePermission(packageName, permName, user, null); } @Override public void revokeRuntimePermission(String packageName, String permName, UserHandle user, String reason) { getPermissionManager().revokeRuntimePermission(packageName, permName, user, reason); } @Override public int getPermissionFlags(String permName, String packageName, UserHandle user) { return getPermissionManager().getPermissionFlags(packageName, permName, user); } @Override public void updatePermissionFlags(String permName, String packageName, int flagMask, int flagValues, UserHandle user) { getPermissionManager().updatePermissionFlags(packageName, permName, flagMask, flagValues, user); } @Override public @NonNull Set getWhitelistedRestrictedPermissions( @NonNull String packageName, @PermissionWhitelistFlags int flags) { return getPermissionManager().getAllowlistedRestrictedPermissions(packageName, flags); } @Override public boolean addWhitelistedRestrictedPermission(@NonNull String packageName, @NonNull String permName, @PermissionWhitelistFlags int flags) { return getPermissionManager().addAllowlistedRestrictedPermission(packageName, permName, flags); } @Override public boolean setAutoRevokeWhitelisted(@NonNull String packageName, boolean whitelisted) { return getPermissionManager().setAutoRevokeExempted(packageName, whitelisted); } @Override public boolean isAutoRevokeWhitelisted(@NonNull String packageName) { return getPermissionManager().isAutoRevokeExempted(packageName); } @Override public boolean removeWhitelistedRestrictedPermission(@NonNull String packageName, @NonNull String permName, @PermissionWhitelistFlags int flags) { return getPermissionManager().removeAllowlistedRestrictedPermission(packageName, permName, flags); } @Override @UnsupportedAppUsage public boolean shouldShowRequestPermissionRationale(String permName) { return getPermissionManager().shouldShowRequestPermissionRationale(permName); } @Override public Intent buildRequestPermissionsIntent(@NonNull String[] permissions) { Intent intent = super.buildRequestPermissionsIntent(permissions); intent.putExtra(EXTRA_REQUEST_PERMISSIONS_DEVICE_ID, mContext.getDeviceId()); return intent; } @Override public CharSequence getBackgroundPermissionOptionLabel() { try { String permissionController = getPermissionControllerPackageName(); Context context = mContext.createPackageContext(permissionController, 0); int textId = context.getResources().getIdentifier(APP_PERMISSION_BUTTON_ALLOW_ALWAYS, "string", PERMISSION_CONTROLLER_RESOURCE_PACKAGE); if (textId != 0) { return context.getText(textId); } } catch (NameNotFoundException e) { Log.e(TAG, "Permission controller not found.", e); } return ""; } @Override public int checkSignatures(String pkg1, String pkg2) { try { return mPM.checkSignatures(pkg1, pkg2, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public int checkSignatures(int uid1, int uid2) { try { return mPM.checkUidSignatures(uid1, uid2); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean hasSigningCertificate( String packageName, byte[] certificate, @CertificateInputType int type) { try { return mPM.hasSigningCertificate(packageName, certificate, type); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean hasSigningCertificate( int uid, byte[] certificate, @CertificateInputType int type) { try { return mPM.hasUidSigningCertificate(uid, certificate, type); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } private static List encodeCertificates(List certs) throws CertificateEncodingException { if (certs == null) { return null; } List result = new ArrayList<>(certs.size()); for (Certificate cert : certs) { if (!(cert instanceof X509Certificate)) { throw new CertificateEncodingException("Only X509 certificates supported."); } result.add(cert.getEncoded()); } return result; } @Override public void requestChecksums(@NonNull String packageName, boolean includeSplits, @Checksum.TypeMask int required, @NonNull List trustedInstallers, @NonNull OnChecksumsReadyListener onChecksumsReadyListener) throws CertificateEncodingException, NameNotFoundException { Objects.requireNonNull(packageName); Objects.requireNonNull(onChecksumsReadyListener); Objects.requireNonNull(trustedInstallers); if (trustedInstallers == TRUST_ALL) { trustedInstallers = null; } else if (trustedInstallers == TRUST_NONE) { trustedInstallers = Collections.emptyList(); } else if (trustedInstallers.isEmpty()) { throw new IllegalArgumentException( "trustedInstallers has to be one of TRUST_ALL/TRUST_NONE or a non-empty " + "list of certificates."); } try { IOnChecksumsReadyListener onChecksumsReadyListenerDelegate = new IOnChecksumsReadyListener.Stub() { @Override public void onChecksumsReady(List checksums) throws RemoteException { onChecksumsReadyListener.onChecksumsReady(checksums); } }; mPM.requestPackageChecksums(packageName, includeSplits, DEFAULT_CHECKSUMS, required, encodeCertificates(trustedInstallers), onChecksumsReadyListenerDelegate, getUserId()); } catch (ParcelableException e) { e.maybeRethrow(PackageManager.NameNotFoundException.class); throw new RuntimeException(e); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** * Wrap the cached value in a class that does deep compares on string * arrays. The comparison is needed only for the verification mode of * PropertyInvalidatedCache; this mode is only enabled for debugging. * The return result is an array of strings but the order in the array * is not important. To properly compare two arrays, the arrays are * sorted before the comparison. */ private static class GetPackagesForUidResult { private final String [] mValue; GetPackagesForUidResult(String []s) { mValue = s; } public String[] value() { return mValue; } @Override public String toString() { return Arrays.toString(mValue); } @Override public int hashCode() { return Arrays.hashCode(mValue); } /** * Arrays.sort() throws an NPE if passed a null pointer, so nulls * are handled first. */ @Override public boolean equals(@Nullable Object o) { if (o instanceof GetPackagesForUidResult) { String [] r = ((GetPackagesForUidResult) o).mValue; String [] l = mValue; if ((r == null) != (l == null)) { return false; } else if (r == null) { return true; } // Both arrays are non-null. Sort before comparing. Arrays.sort(r); Arrays.sort(l); return Arrays.equals(l, r); } else { return false; } } } private static final String CACHE_KEY_PACKAGES_FOR_UID_PROPERTY = "cache_key.get_packages_for_uid"; private static final PropertyInvalidatedCache mGetPackagesForUidCache = new PropertyInvalidatedCache( 32, CACHE_KEY_PACKAGES_FOR_UID_PROPERTY) { @Override public GetPackagesForUidResult recompute(Integer uid) { try { return new GetPackagesForUidResult( ActivityThread.currentActivityThread(). getPackageManager().getPackagesForUid(uid)); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public String queryToString(Integer uid) { return String.format("uid=%d", uid.intValue()); } }; @Override public String[] getPackagesForUid(int uid) { return mGetPackagesForUidCache.query(uid).value(); } /** @hide */ public static void disableGetPackagesForUidCache() { mGetPackagesForUidCache.disableLocal(); } /** @hide */ public static void invalidateGetPackagesForUidCache() { PropertyInvalidatedCache.invalidateCache(CACHE_KEY_PACKAGES_FOR_UID_PROPERTY); } @Override public String getNameForUid(int uid) { try { return mPM.getNameForUid(uid); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public String[] getNamesForUids(int[] uids) { try { return mPM.getNamesForUids(uids); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public int getUidForSharedUser(String sharedUserName) throws NameNotFoundException { try { int uid = mPM.getUidForSharedUser(sharedUserName); if (uid != Process.INVALID_UID) { return uid; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException("No shared userid for user:"+sharedUserName); } @Override public List getInstalledModules(int flags) { try { return mPM.getInstalledModules(flags); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public ModuleInfo getModuleInfo(String packageName, int flags) throws NameNotFoundException { try { ModuleInfo mi = mPM.getModuleInfo(packageName, flags); if (mi != null) { return mi; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException("No module info for package: " + packageName); } @SuppressWarnings("unchecked") @Override public List getInstalledPackages(int flags) { return getInstalledPackages(PackageInfoFlags.of(flags)); } @SuppressWarnings("unchecked") @Override public List getInstalledPackages(PackageInfoFlags flags) { return getInstalledPackagesAsUser(flags, getUserId()); } /** @hide */ @Override @SuppressWarnings("unchecked") public List getInstalledPackagesAsUser(int flags, int userId) { return getInstalledPackagesAsUser(PackageInfoFlags.of(flags), userId); } /** @hide */ @Override @SuppressWarnings("unchecked") public List getInstalledPackagesAsUser(PackageInfoFlags flags, int userId) { try { ParceledListSlice parceledList = mPM.getInstalledPackages(updateFlagsForPackage(flags.getValue(), userId), userId); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override @NonNull public PersistableBundle getAppMetadata(@NonNull String packageName) throws NameNotFoundException { PersistableBundle appMetadata = null; ParcelFileDescriptor pfd = null; try { pfd = mPM.getAppMetadataFd(packageName, getUserId()); } catch (ParcelableException e) { e.maybeRethrow(NameNotFoundException.class); throw new RuntimeException(e); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } if (pfd != null) { try (InputStream inputStream = new ParcelFileDescriptor.AutoCloseInputStream(pfd)) { appMetadata = PersistableBundle.readFromStream(inputStream); } catch (IOException e) { throw new RuntimeException(e); } } return appMetadata != null ? appMetadata : new PersistableBundle(); } @Override public @AppMetadataSource int getAppMetadataSource(@NonNull String packageName) throws NameNotFoundException { Objects.requireNonNull(packageName, "packageName cannot be null"); int source = PackageManager.APP_METADATA_SOURCE_UNKNOWN; try { source = mPM.getAppMetadataSource(packageName, getUserId()); } catch (ParcelableException e) { e.maybeRethrow(NameNotFoundException.class); throw new RuntimeException(e); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } return source; } @SuppressWarnings("unchecked") @Override public List getPackagesHoldingPermissions(String[] permissions, int flags) { return this.getPackagesHoldingPermissions(permissions, PackageInfoFlags.of(flags)); } @SuppressWarnings("unchecked") @Override public List getPackagesHoldingPermissions(String[] permissions, PackageInfoFlags flags) { final int userId = getUserId(); try { ParceledListSlice parceledList = mPM.getPackagesHoldingPermissions(permissions, updateFlagsForPackage(flags.getValue(), userId), userId); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @SuppressWarnings("unchecked") @Override public List getInstalledApplications(int flags) { return getInstalledApplicationsAsUser(flags, getUserId()); } @Override public List getInstalledApplications(ApplicationInfoFlags flags) { return getInstalledApplicationsAsUser(flags, getUserId()); } /** @hide */ @SuppressWarnings("unchecked") @Override public List getInstalledApplicationsAsUser(int flags, int userId) { return getInstalledApplicationsAsUser(ApplicationInfoFlags.of(flags), userId); } /** @hide */ @SuppressWarnings("unchecked") @Override public List getInstalledApplicationsAsUser(ApplicationInfoFlags flags, int userId) { try { ParceledListSlice parceledList = mPM.getInstalledApplications(updateFlagsForApplication( flags.getValue(), userId), userId); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @SuppressWarnings("unchecked") @Override public List getInstantApps() { try { ParceledListSlice slice = mPM.getInstantApps(getUserId()); if (slice != null) { return slice.getList(); } return Collections.emptyList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @Override public Drawable getInstantAppIcon(String packageName) { try { Bitmap bitmap = mPM.getInstantAppIcon(packageName, getUserId()); if (bitmap != null) { return new BitmapDrawable(null, bitmap); } return null; } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean isInstantApp() { return isInstantApp(mContext.getPackageName()); } @Override public boolean isInstantApp(String packageName) { try { return mPM.isInstantApp(packageName, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } public int getInstantAppCookieMaxBytes() { return Settings.Global.getInt(mContext.getContentResolver(), Settings.Global.EPHEMERAL_COOKIE_MAX_SIZE_BYTES, DEFAULT_EPHEMERAL_COOKIE_MAX_SIZE_BYTES); } @Override public int getInstantAppCookieMaxSize() { return getInstantAppCookieMaxBytes(); } @Override public @NonNull byte[] getInstantAppCookie() { try { final byte[] cookie = mPM.getInstantAppCookie(mContext.getPackageName(), getUserId()); if (cookie != null) { return cookie; } else { return EmptyArray.BYTE; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void clearInstantAppCookie() { updateInstantAppCookie(null); } @Override public void updateInstantAppCookie(@NonNull byte[] cookie) { if (cookie != null && cookie.length > getInstantAppCookieMaxBytes()) { throw new IllegalArgumentException("instant cookie longer than " + getInstantAppCookieMaxBytes()); } try { mPM.setInstantAppCookie(mContext.getPackageName(), cookie, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) @Override public boolean setInstantAppCookie(@NonNull byte[] cookie) { try { return mPM.setInstantAppCookie(mContext.getPackageName(), cookie, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public ResolveInfo resolveActivity(Intent intent, int flags) { return resolveActivity(intent, ResolveInfoFlags.of(flags)); } @Override public ResolveInfo resolveActivity(Intent intent, ResolveInfoFlags flags) { return resolveActivityAsUser(intent, flags, getUserId()); } @Override public ResolveInfo resolveActivityAsUser(Intent intent, int flags, int userId) { return resolveActivityAsUser(intent, ResolveInfoFlags.of(flags), userId); } @Override public ResolveInfo resolveActivityAsUser(Intent intent, ResolveInfoFlags flags, int userId) { try { return mPM.resolveIntent( intent, intent.resolveTypeIfNeeded(mContext.getContentResolver()), updateFlagsForComponent(flags.getValue(), userId, intent), userId); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public List queryIntentActivities(Intent intent, int flags) { return queryIntentActivities(intent, ResolveInfoFlags.of(flags)); } @Override public List queryIntentActivities(Intent intent, ResolveInfoFlags flags) { return queryIntentActivitiesAsUser(intent, flags, getUserId()); } /** @hide Same as above but for a specific user */ @Override public List queryIntentActivitiesAsUser(Intent intent, int flags, int userId) { return queryIntentActivitiesAsUser(intent, ResolveInfoFlags.of(flags), userId); } /** @hide Same as above but for a specific user */ @Override @SuppressWarnings("unchecked") public List queryIntentActivitiesAsUser(Intent intent, ResolveInfoFlags flags, int userId) { try { ParceledListSlice parceledList = mPM.queryIntentActivities( intent, intent.resolveTypeIfNeeded(mContext.getContentResolver()), updateFlagsForComponent(flags.getValue(), userId, intent), userId); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public List queryIntentActivityOptions(ComponentName caller, Intent[] specifics, Intent intent, int flags) { return queryIntentActivityOptions(caller, specifics == null ? null : new ArrayList<>(Arrays.asList(specifics)), intent, ResolveInfoFlags.of(flags)); } @Override @SuppressWarnings("unchecked") public List queryIntentActivityOptions(ComponentName caller, List specifics, Intent intent, ResolveInfoFlags flags) { final int userId = getUserId(); final ContentResolver resolver = mContext.getContentResolver(); String[] specificTypes = null; if (specifics != null) { final int numSpecifics = specifics.size(); for (int i = 0; i < numSpecifics; i++) { Intent sp = specifics.get(i); if (sp != null) { String t = sp.resolveTypeIfNeeded(resolver); if (t != null) { if (specificTypes == null) { specificTypes = new String[numSpecifics]; } specificTypes[i] = t; } } } } try { ParceledListSlice parceledList = mPM.queryIntentActivityOptions( caller, specifics == null ? null : specifics.toArray(new Intent[0]), specificTypes, intent, intent.resolveTypeIfNeeded(resolver), updateFlagsForComponent(flags.getValue(), userId, intent), userId); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** * @hide */ @Override public List queryBroadcastReceiversAsUser(Intent intent, int flags, int userId) { return queryBroadcastReceiversAsUser(intent, ResolveInfoFlags.of(flags), userId); } /** * @hide */ @Override @SuppressWarnings("unchecked") public List queryBroadcastReceiversAsUser(Intent intent, ResolveInfoFlags flags, int userId) { try { ParceledListSlice parceledList = mPM.queryIntentReceivers( intent, intent.resolveTypeIfNeeded(mContext.getContentResolver()), updateFlagsForComponent(flags.getValue(), userId, intent), userId); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public List queryBroadcastReceivers(Intent intent, int flags) { return queryBroadcastReceivers(intent, ResolveInfoFlags.of(flags)); } @Override public List queryBroadcastReceivers(Intent intent, ResolveInfoFlags flags) { return queryBroadcastReceiversAsUser(intent, flags, getUserId()); } @Override public ResolveInfo resolveServiceAsUser(Intent intent, int flags, @UserIdInt int userId) { return resolveServiceAsUser(intent, ResolveInfoFlags.of(flags), userId); } @Override public ResolveInfo resolveServiceAsUser(Intent intent, ResolveInfoFlags flags, @UserIdInt int userId) { try { return mPM.resolveService( intent, intent.resolveTypeIfNeeded(mContext.getContentResolver()), updateFlagsForComponent(flags.getValue(), userId, intent), userId); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public ResolveInfo resolveService(Intent intent, int flags) { return resolveService(intent, ResolveInfoFlags.of(flags)); } @Override public ResolveInfo resolveService(Intent intent, ResolveInfoFlags flags) { return resolveServiceAsUser(intent, flags, getUserId()); } @Override public List queryIntentServicesAsUser(Intent intent, int flags, int userId) { return queryIntentServicesAsUser(intent, ResolveInfoFlags.of(flags), userId); } @Override @SuppressWarnings("unchecked") public List queryIntentServicesAsUser(Intent intent, ResolveInfoFlags flags, int userId) { try { ParceledListSlice parceledList = mPM.queryIntentServices( intent, intent.resolveTypeIfNeeded(mContext.getContentResolver()), updateFlagsForComponent(flags.getValue(), userId, intent), userId); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public List queryIntentServices(Intent intent, int flags) { return queryIntentServices(intent, ResolveInfoFlags.of(flags)); } @Override public List queryIntentServices(Intent intent, ResolveInfoFlags flags) { return queryIntentServicesAsUser(intent, flags, getUserId()); } @Override public List queryIntentContentProvidersAsUser( Intent intent, int flags, int userId) { return queryIntentContentProvidersAsUser(intent, ResolveInfoFlags.of(flags), userId); } @Override @SuppressWarnings("unchecked") public List queryIntentContentProvidersAsUser( Intent intent, ResolveInfoFlags flags, int userId) { try { ParceledListSlice parceledList = mPM.queryIntentContentProviders( intent, intent.resolveTypeIfNeeded(mContext.getContentResolver()), updateFlagsForComponent(flags.getValue(), userId, intent), userId); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public List queryIntentContentProviders(Intent intent, int flags) { return queryIntentContentProviders(intent, ResolveInfoFlags.of(flags)); } @Override public List queryIntentContentProviders(Intent intent, ResolveInfoFlags flags) { return queryIntentContentProvidersAsUser(intent, flags, getUserId()); } @Override public ProviderInfo resolveContentProvider(String name, int flags) { return resolveContentProvider(name, ComponentInfoFlags.of(flags)); } @Override public ProviderInfo resolveContentProvider(String name, ComponentInfoFlags flags) { return resolveContentProviderAsUser(name, flags, getUserId()); } /** @hide **/ @Override public ProviderInfo resolveContentProviderAsUser(String name, int flags, int userId) { return resolveContentProviderAsUser(name, ComponentInfoFlags.of(flags), userId); } /** @hide **/ @Override public ProviderInfo resolveContentProviderAsUser(String name, ComponentInfoFlags flags, int userId) { try { return mPM.resolveContentProvider(name, updateFlagsForComponent(flags.getValue(), userId, null), userId); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public List queryContentProviders(String processName, int uid, int flags) { return queryContentProviders(processName, uid, ComponentInfoFlags.of(flags)); } @Override public List queryContentProviders(String processName, int uid, ComponentInfoFlags flags) { return queryContentProviders(processName, uid, flags, null); } @Override public List queryContentProviders(String processName, int uid, int flags, String metaDataKey) { return queryContentProviders(processName, uid, ComponentInfoFlags.of(flags), metaDataKey); } @Override @SuppressWarnings("unchecked") public List queryContentProviders(String processName, int uid, ComponentInfoFlags flags, String metaDataKey) { try { ParceledListSlice slice = mPM.queryContentProviders(processName, uid, updateFlagsForComponent(flags.getValue(), UserHandle.getUserId(uid), null), metaDataKey); return slice != null ? slice.getList() : Collections.emptyList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public InstrumentationInfo getInstrumentationInfo( ComponentName className, int flags) throws NameNotFoundException { try { InstrumentationInfo ii = mPM.getInstrumentationInfoAsUser( className, flags, getUserId()); if (ii != null) { return ii; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException(className.toString()); } @Override @SuppressWarnings("unchecked") public List queryInstrumentation( String targetPackage, int flags) { try { ParceledListSlice parceledList = mPM.queryInstrumentationAsUser(targetPackage, flags, getUserId()); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Nullable @Override public Drawable getDrawable(String packageName, @DrawableRes int resId, @Nullable ApplicationInfo appInfo) { final ResourceName name = new ResourceName(packageName, resId); final Drawable cachedIcon = getCachedIcon(name); if (cachedIcon != null) { return cachedIcon; } if (appInfo == null) { try { appInfo = getApplicationInfo(packageName, sDefaultFlags); } catch (NameNotFoundException e) { return null; } } if (resId != 0) { try { final Resources r = getResourcesForApplication(appInfo); final Drawable dr = r.getDrawable(resId, null); if (dr != null) { putCachedIcon(name, dr); } if (false) { RuntimeException e = new RuntimeException("here"); e.fillInStackTrace(); Log.w(TAG, "Getting drawable 0x" + Integer.toHexString(resId) + " from package " + packageName + ": app scale=" + r.getCompatibilityInfo().applicationScale + ", caller scale=" + mContext.getResources() .getCompatibilityInfo().applicationScale, e); } if (DEBUG_ICONS) { Log.v(TAG, "Getting drawable 0x" + Integer.toHexString(resId) + " from " + r + ": " + dr); } return dr; } catch (NameNotFoundException e) { Log.w("PackageManager", "Failure retrieving resources for " + appInfo.packageName); } catch (Resources.NotFoundException e) { Log.w("PackageManager", "Failure retrieving resources for " + appInfo.packageName + ": " + e.getMessage()); } catch (Exception e) { // If an exception was thrown, fall through to return // default icon. Log.w("PackageManager", "Failure retrieving icon 0x" + Integer.toHexString(resId) + " in package " + packageName, e); } } return null; } @Override public Drawable getActivityIcon(ComponentName activityName) throws NameNotFoundException { return getActivityInfo(activityName, sDefaultFlags).loadIcon(this); } @Override public Drawable getActivityIcon(Intent intent) throws NameNotFoundException { if (intent.getComponent() != null) { return getActivityIcon(intent.getComponent()); } ResolveInfo info = resolveActivity(intent, MATCH_DEFAULT_ONLY); if (info != null) { return info.activityInfo.loadIcon(this); } throw new NameNotFoundException(intent.toUri(0)); } @Override public Drawable getDefaultActivityIcon() { return mContext.getDrawable(com.android.internal.R.drawable.sym_def_app_icon); } @Override public Drawable getApplicationIcon(ApplicationInfo info) { return info.loadIcon(this); } @Override public Drawable getApplicationIcon(String packageName) throws NameNotFoundException { return getApplicationIcon(getApplicationInfo(packageName, sDefaultFlags)); } @Override public Drawable getActivityBanner(ComponentName activityName) throws NameNotFoundException { return getActivityInfo(activityName, sDefaultFlags).loadBanner(this); } @Override public Drawable getActivityBanner(Intent intent) throws NameNotFoundException { if (intent.getComponent() != null) { return getActivityBanner(intent.getComponent()); } ResolveInfo info = resolveActivity( intent, MATCH_DEFAULT_ONLY); if (info != null) { return info.activityInfo.loadBanner(this); } throw new NameNotFoundException(intent.toUri(0)); } @Override public Drawable getApplicationBanner(ApplicationInfo info) { return info.loadBanner(this); } @Override public Drawable getApplicationBanner(String packageName) throws NameNotFoundException { return getApplicationBanner(getApplicationInfo(packageName, sDefaultFlags)); } @Override public Drawable getActivityLogo(ComponentName activityName) throws NameNotFoundException { return getActivityInfo(activityName, sDefaultFlags).loadLogo(this); } @Override public Drawable getActivityLogo(Intent intent) throws NameNotFoundException { if (intent.getComponent() != null) { return getActivityLogo(intent.getComponent()); } ResolveInfo info = resolveActivity(intent, MATCH_DEFAULT_ONLY); if (info != null) { return info.activityInfo.loadLogo(this); } throw new NameNotFoundException(intent.toUri(0)); } @Override public Drawable getApplicationLogo(ApplicationInfo info) { return info.loadLogo(this); } @Override public Drawable getApplicationLogo(String packageName) throws NameNotFoundException { return getApplicationLogo(getApplicationInfo(packageName, sDefaultFlags)); } @Override public Drawable getUserBadgedIcon(Drawable icon, UserHandle user) { if (!hasUserBadge(user.getIdentifier())) { return icon; } final Drawable badgeForeground = getDevicePolicyManager().getResources().getDrawable( getUpdatableUserIconBadgeId(user), SOLID_COLORED, () -> getDefaultUserIconBadge(user)); Drawable badge = new LauncherIcons(mContext).getBadgeDrawable( badgeForeground, getUserBadgeColor(user, false)); return getBadgedDrawable(icon, badge, null, true); } private String getUpdatableUserIconBadgeId(UserHandle user) { return getUserManager().isManagedProfile(user.getIdentifier()) ? WORK_PROFILE_ICON_BADGE : UNDEFINED; } private Drawable getDefaultUserIconBadge(UserHandle user) { return mContext.getDrawable(getUserManager().getUserIconBadgeResId(user.getIdentifier())); } @Override public Drawable getUserBadgedDrawableForDensity(Drawable drawable, UserHandle user, Rect badgeLocation, int badgeDensity) { Drawable badgeDrawable = getUserBadgeForDensity(user, badgeDensity); if (badgeDrawable == null) { return drawable; } return getBadgedDrawable(drawable, badgeDrawable, badgeLocation, true); } /** * Returns the color of the user's actual badge (not the badge's shadow). * @param checkTheme whether to check the theme to determine the badge color. This should be * true if the background is determined by the theme. Otherwise, if * checkTheme is false, returns the color assuming a light background. */ private int getUserBadgeColor(UserHandle user, boolean checkTheme) { if (checkTheme && mContext.getResources().getConfiguration().isNightModeActive()) { return getUserManager().getUserBadgeDarkColor(user.getIdentifier()); } return getUserManager().getUserBadgeColor(user.getIdentifier()); } @Override public Drawable getUserBadgeForDensity(UserHandle user, int density) { // This is part of the shadow, not the main color, and is not actually corp-specific. Drawable badgeColor = getProfileIconForDensity(user, com.android.internal.R.drawable.ic_corp_badge_color, density); if (badgeColor == null) { return null; } final Drawable badgeForeground = getDevicePolicyManager().getResources() .getDrawableForDensity( getUpdatableUserBadgeId(user), SOLID_COLORED, density, () -> getDefaultUserBadgeForDensity(user, density)); badgeForeground.setTint(getUserBadgeColor(user, false)); Drawable badge = new LayerDrawable(new Drawable[] {badgeColor, badgeForeground }); return badge; } private String getUpdatableUserBadgeId(UserHandle user) { return getUserManager().isManagedProfile(user.getIdentifier()) ? WORK_PROFILE_ICON : UNDEFINED; } private Drawable getDefaultUserBadgeForDensity(UserHandle user, int density) { return getDrawableForDensity( getUserManager().getUserBadgeResId(user.getIdentifier()), density); } /** * Returns the badge color based on whether device has dark theme enabled or not. */ @Override public Drawable getUserBadgeForDensityNoBackground(UserHandle user, int density) { if (!hasUserBadge(user.getIdentifier())) { return null; } final Drawable badge = getDevicePolicyManager().getResources().getDrawableForDensity( getUpdatableUserBadgeId(user), SOLID_NOT_COLORED, density, () -> getDefaultUserBadgeNoBackgroundForDensity(user, density)); if (badge != null) { badge.setTint(getUserBadgeColor(user, true)); } return badge; } private Drawable getDefaultUserBadgeNoBackgroundForDensity(UserHandle user, int density) { return getDrawableForDensity( getUserManager().getUserBadgeNoBackgroundResId(user.getIdentifier()), density); } private Drawable getDrawableForDensity(int drawableId, int density) { if (density <= 0) { density = mContext.getResources().getDisplayMetrics().densityDpi; } return mContext.getResources().getDrawableForDensity(drawableId, density); } private Drawable getProfileIconForDensity(UserHandle user, int drawableId, int density) { if (hasUserBadge(user.getIdentifier())) { return getDrawableForDensity(drawableId, density); } return null; } @Override public CharSequence getUserBadgedLabel(CharSequence label, UserHandle user) { return getUserManager().getBadgedLabelForUser(label, user); } @Override public Resources getResourcesForActivity(ComponentName activityName) throws NameNotFoundException { return getResourcesForApplication( getActivityInfo(activityName, sDefaultFlags).applicationInfo); } @Override public Resources getResourcesForApplication(@NonNull ApplicationInfo app) throws NameNotFoundException { return getResourcesForApplication(app, null); } @Override public Resources getResourcesForApplication(@NonNull ApplicationInfo app, @Nullable Configuration configuration) throws NameNotFoundException { if (app.packageName.equals("system")) { Context sysuiContext = mContext.mMainThread.getSystemUiContext(); if (configuration != null) { sysuiContext = sysuiContext.createConfigurationContext(configuration); } return sysuiContext.getResources(); } final boolean sameUid = (app.uid == Process.myUid()); final Resources r = mContext.mMainThread.getTopLevelResources( sameUid ? app.sourceDir : app.publicSourceDir, sameUid ? app.splitSourceDirs : app.splitPublicSourceDirs, app.resourceDirs, app.overlayPaths, app.sharedLibraryFiles, mContext.mPackageInfo, configuration); if (r != null) { return r; } throw new NameNotFoundException("Unable to open " + app.publicSourceDir); } @Override public Resources getResourcesForApplication(String appPackageName) throws NameNotFoundException { return getResourcesForApplication( getApplicationInfo(appPackageName, sDefaultFlags)); } /** @hide */ @Override public Resources getResourcesForApplicationAsUser(String appPackageName, int userId) throws NameNotFoundException { if (userId < 0) { throw new IllegalArgumentException( "Call does not support special user #" + userId); } if ("system".equals(appPackageName)) { return mContext.mMainThread.getSystemUiContext().getResources(); } try { ApplicationInfo ai = mPM.getApplicationInfo(appPackageName, sDefaultFlags, userId); if (ai != null) { return getResourcesForApplication(ai); } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } throw new NameNotFoundException("Package " + appPackageName + " doesn't exist"); } volatile int mCachedSafeMode = -1; @Override public boolean isSafeMode() { try { if (mCachedSafeMode < 0) { mCachedSafeMode = mPM.isSafeMode() ? 1 : 0; } return mCachedSafeMode != 0; } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void addOnPermissionsChangeListener(OnPermissionsChangedListener listener) { getPermissionManager().addOnPermissionsChangeListener(listener); } @Override public void removeOnPermissionsChangeListener(OnPermissionsChangedListener listener) { getPermissionManager().removeOnPermissionsChangeListener(listener); } @UnsupportedAppUsage static void configurationChanged() { synchronized (sSync) { sIconCache.clear(); sStringCache.clear(); } } @UnsupportedAppUsage protected ApplicationPackageManager(ContextImpl context, IPackageManager pm) { mContext = context; mPM = pm; } /** * Update given flags when being used to request {@link PackageInfo}. */ private long updateFlagsForPackage(long flags, int userId) { if ((flags & (GET_ACTIVITIES | GET_RECEIVERS | GET_SERVICES | GET_PROVIDERS)) != 0) { // Caller is asking for component details, so they'd better be // asking for specific Direct Boot matching behavior if ((flags & (MATCH_DIRECT_BOOT_UNAWARE | MATCH_DIRECT_BOOT_AWARE | MATCH_DIRECT_BOOT_AUTO)) == 0) { onImplicitDirectBoot(userId); } } return flags; } /** * Update given flags when being used to request {@link ApplicationInfo}. */ private long updateFlagsForApplication(long flags, int userId) { return updateFlagsForPackage(flags, userId); } /** * Update given flags when being used to request {@link ComponentInfo}. */ private long updateFlagsForComponent(@ComponentInfoFlagsBits long flags, int userId, Intent intent) { if (intent != null) { if ((intent.getFlags() & Intent.FLAG_DIRECT_BOOT_AUTO) != 0) { flags |= MATCH_DIRECT_BOOT_AUTO; } } // Caller is asking for component details, so they'd better be // asking for specific Direct Boot matching behavior if ((flags & (MATCH_DIRECT_BOOT_UNAWARE | MATCH_DIRECT_BOOT_AWARE | MATCH_DIRECT_BOOT_AUTO)) == 0) { onImplicitDirectBoot(userId); } return flags; } private void onImplicitDirectBoot(int userId) { // Only report if someone is relying on implicit behavior while the user // is locked; code running when unlocked is going to see both aware and // unaware components. if (StrictMode.vmImplicitDirectBootEnabled()) { // We can cache the unlocked state for the userId we're running as, // since any relocking of that user will always result in our // process being killed to release any CE FDs we're holding onto. if (userId == UserHandle.myUserId()) { if (mUserUnlocked) { return; } else if (mContext.getSystemService(UserManager.class) .isUserUnlockingOrUnlocked(userId)) { mUserUnlocked = true; } else { StrictMode.onImplicitDirectBoot(); } } else if (!mContext.getSystemService(UserManager.class) .isUserUnlockingOrUnlocked(userId)) { StrictMode.onImplicitDirectBoot(); } } } @Nullable private Drawable getCachedIcon(@NonNull ResourceName name) { synchronized (sSync) { final WeakReference wr = sIconCache.get(name); if (DEBUG_ICONS) Log.v(TAG, "Get cached weak drawable ref for " + name + ": " + wr); if (wr != null) { // we have the activity final Drawable.ConstantState state = wr.get(); if (state != null) { if (DEBUG_ICONS) { Log.v(TAG, "Get cached drawable state for " + name + ": " + state); } // Note: It's okay here to not use the newDrawable(Resources) variant // of the API. The ConstantState comes from a drawable that was // originally created by passing the proper app Resources instance // which means the state should already contain the proper // resources specific information (like density.) See // BitmapDrawable.BitmapState for instance. return state.newDrawable(); } // our entry has been purged sIconCache.remove(name); } } return null; } private void putCachedIcon(@NonNull ResourceName name, @NonNull Drawable dr) { synchronized (sSync) { sIconCache.put(name, new WeakReference<>(dr.getConstantState())); if (DEBUG_ICONS) Log.v(TAG, "Added cached drawable state for " + name + ": " + dr); } } static void handlePackageBroadcast(int cmd, String[] pkgList, boolean hasPkgInfo) { boolean immediateGc = false; if (cmd == ApplicationThreadConstants.EXTERNAL_STORAGE_UNAVAILABLE) { immediateGc = true; } if (pkgList != null && (pkgList.length > 0)) { boolean needCleanup = false; for (String ssp : pkgList) { synchronized (sSync) { for (int i=sIconCache.size()-1; i>=0; i--) { ResourceName nm = sIconCache.keyAt(i); if (nm.packageName.equals(ssp)) { //Log.i(TAG, "Removing cached drawable for " + nm); sIconCache.removeAt(i); needCleanup = true; } } for (int i=sStringCache.size()-1; i>=0; i--) { ResourceName nm = sStringCache.keyAt(i); if (nm.packageName.equals(ssp)) { //Log.i(TAG, "Removing cached string for " + nm); sStringCache.removeAt(i); needCleanup = true; } } } } if (needCleanup || hasPkgInfo) { if (immediateGc) { // Schedule an immediate gc. Runtime.getRuntime().gc(); } else { ActivityThread.currentActivityThread().scheduleGcIdler(); } } } } private static final class ResourceName { final String packageName; final int iconId; ResourceName(String _packageName, int _iconId) { packageName = _packageName; iconId = _iconId; } ResourceName(ApplicationInfo aInfo, int _iconId) { this(aInfo.packageName, _iconId); } ResourceName(ComponentInfo cInfo, int _iconId) { this(cInfo.applicationInfo.packageName, _iconId); } ResourceName(ResolveInfo rInfo, int _iconId) { this(rInfo.activityInfo.applicationInfo.packageName, _iconId); } @Override public boolean equals(@Nullable Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; ResourceName that = (ResourceName) o; if (iconId != that.iconId) return false; return !(packageName != null ? !packageName.equals(that.packageName) : that.packageName != null); } @Override public int hashCode() { int result; result = packageName.hashCode(); result = 31 * result + iconId; return result; } @Override public String toString() { return "{ResourceName " + packageName + " / " + iconId + "}"; } } private CharSequence getCachedString(ResourceName name) { synchronized (sSync) { WeakReference wr = sStringCache.get(name); if (wr != null) { // we have the activity CharSequence cs = wr.get(); if (cs != null) { return cs; } // our entry has been purged sStringCache.remove(name); } } return null; } private void putCachedString(ResourceName name, CharSequence cs) { synchronized (sSync) { sStringCache.put(name, new WeakReference(cs)); } } @Override public CharSequence getText(String packageName, @StringRes int resid, ApplicationInfo appInfo) { ResourceName name = new ResourceName(packageName, resid); CharSequence text = getCachedString(name); if (text != null) { return text; } if (appInfo == null) { try { appInfo = getApplicationInfo(packageName, sDefaultFlags); } catch (NameNotFoundException e) { return null; } } try { Resources r = getResourcesForApplication(appInfo); text = r.getText(resid); putCachedString(name, text); return text; } catch (NameNotFoundException e) { Log.w("PackageManager", "Failure retrieving resources for " + appInfo.packageName); } catch (RuntimeException e) { // If an exception was thrown, fall through to return // default icon. Log.w("PackageManager", "Failure retrieving text 0x" + Integer.toHexString(resid) + " in package " + packageName, e); } return null; } @Override public XmlResourceParser getXml(String packageName, @XmlRes int resid, ApplicationInfo appInfo) { if (appInfo == null) { try { appInfo = getApplicationInfo(packageName, sDefaultFlags); } catch (NameNotFoundException e) { return null; } } try { Resources r = getResourcesForApplication(appInfo); return r.getXml(resid); } catch (RuntimeException e) { // If an exception was thrown, fall through to return // default icon. Log.w("PackageManager", "Failure retrieving xml 0x" + Integer.toHexString(resid) + " in package " + packageName, e); } catch (NameNotFoundException e) { Log.w("PackageManager", "Failure retrieving resources for " + appInfo.packageName); } return null; } @Override public CharSequence getApplicationLabel(ApplicationInfo info) { return info.loadLabel(this); } @Override public int installExistingPackage(String packageName) throws NameNotFoundException { return installExistingPackage(packageName, INSTALL_REASON_UNKNOWN); } @Override public int installExistingPackage(String packageName, int installReason) throws NameNotFoundException { return installExistingPackageAsUser(packageName, installReason, getUserId()); } @Override public int installExistingPackageAsUser(String packageName, int userId) throws NameNotFoundException { return installExistingPackageAsUser(packageName, INSTALL_REASON_UNKNOWN, userId); } private int installExistingPackageAsUser(String packageName, int installReason, int userId) throws NameNotFoundException { try { int res = mPM.installExistingPackageAsUser(packageName, userId, INSTALL_ALL_WHITELIST_RESTRICTED_PERMISSIONS, installReason, null); if (res == INSTALL_FAILED_INVALID_URI) { throw new NameNotFoundException("Package " + packageName + " doesn't exist"); } return res; } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void verifyPendingInstall(int id, int response) { try { mPM.verifyPendingInstall(id, response); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void extendVerificationTimeout(int id, int verificationCodeAtTimeout, long millisecondsToDelay) { try { mPM.extendVerificationTimeout(id, verificationCodeAtTimeout, millisecondsToDelay); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void verifyIntentFilter(int id, int verificationCode, List failedDomains) { try { mPM.verifyIntentFilter(id, verificationCode, failedDomains); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public int getIntentVerificationStatusAsUser(String packageName, int userId) { try { return mPM.getIntentVerificationStatus(packageName, userId); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean updateIntentVerificationStatusAsUser(String packageName, int status, int userId) { try { return mPM.updateIntentVerificationStatus(packageName, status, userId); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override @SuppressWarnings("unchecked") public List getIntentFilterVerifications(String packageName) { try { ParceledListSlice parceledList = mPM.getIntentFilterVerifications(packageName); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override @SuppressWarnings("unchecked") public List getAllIntentFilters(String packageName) { try { ParceledListSlice parceledList = mPM.getAllIntentFilters(packageName); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public String getDefaultBrowserPackageNameAsUser(int userId) { RoleManager roleManager = mContext.getSystemService(RoleManager.class); return roleManager.getBrowserRoleHolder(userId); } @Override public boolean setDefaultBrowserPackageNameAsUser(String packageName, int userId) { RoleManager roleManager = mContext.getSystemService(RoleManager.class); return roleManager.setBrowserRoleHolder(packageName, userId); } @Override public void setInstallerPackageName(String targetPackage, String installerPackageName) { try { mPM.setInstallerPackageName(targetPackage, installerPackageName); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void setUpdateAvailable(String packageName, boolean updateAvailable) { try { mPM.setUpdateAvailable(packageName, updateAvailable); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public String getInstallerPackageName(String packageName) { try { return mPM.getInstallerPackageName(packageName); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override @NonNull public InstallSourceInfo getInstallSourceInfo(String packageName) throws NameNotFoundException { final InstallSourceInfo installSourceInfo; try { installSourceInfo = mPM.getInstallSourceInfo(packageName, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } if (installSourceInfo == null) { throw new NameNotFoundException(packageName); } return installSourceInfo; } @Override public boolean isAppArchivable(String packageName) throws NameNotFoundException { try { Objects.requireNonNull(packageName); return mPM.isAppArchivable(packageName, new UserHandle(getUserId())); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public int getMoveStatus(int moveId) { try { return mPM.getMoveStatus(moveId); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void registerMoveCallback(MoveCallback callback, Handler handler) { synchronized (mDelegates) { final MoveCallbackDelegate delegate = new MoveCallbackDelegate(callback, handler.getLooper()); try { mPM.registerMoveCallback(delegate); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } mDelegates.add(delegate); } } @Override public void unregisterMoveCallback(MoveCallback callback) { synchronized (mDelegates) { for (Iterator i = mDelegates.iterator(); i.hasNext();) { final MoveCallbackDelegate delegate = i.next(); if (delegate.mCallback == callback) { try { mPM.unregisterMoveCallback(delegate); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } i.remove(); } } } } @Override public int movePackage(String packageName, VolumeInfo vol) { try { final String volumeUuid; if (VolumeInfo.ID_PRIVATE_INTERNAL.equals(vol.id)) { volumeUuid = StorageManager.UUID_PRIVATE_INTERNAL; } else if (vol.isPrimaryPhysical()) { volumeUuid = StorageManager.UUID_PRIMARY_PHYSICAL; } else { volumeUuid = Objects.requireNonNull(vol.fsUuid); } return mPM.movePackage(packageName, volumeUuid); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) public @Nullable VolumeInfo getPackageCurrentVolume(ApplicationInfo app) { final StorageManager storage = mContext.getSystemService(StorageManager.class); return getPackageCurrentVolume(app, storage); } @VisibleForTesting protected @Nullable VolumeInfo getPackageCurrentVolume(ApplicationInfo app, StorageManager storage) { if (app.isInternal()) { return storage.findVolumeById(VolumeInfo.ID_PRIVATE_INTERNAL); } else { return storage.findVolumeByUuid(app.volumeUuid); } } @Override public @NonNull List getPackageCandidateVolumes(ApplicationInfo app) { final StorageManager storageManager = mContext.getSystemService(StorageManager.class); return getPackageCandidateVolumes(app, storageManager, mPM); } @VisibleForTesting protected @NonNull List getPackageCandidateVolumes(ApplicationInfo app, StorageManager storageManager, IPackageManager pm) { final VolumeInfo currentVol = getPackageCurrentVolume(app, storageManager); final List vols = storageManager.getVolumes(); final List candidates = new ArrayList<>(); for (VolumeInfo vol : vols) { if (Objects.equals(vol, currentVol) || isPackageCandidateVolume(mContext, app, vol, pm)) { candidates.add(vol); } } return candidates; } @VisibleForTesting protected boolean isForceAllowOnExternal(Context context) { return Settings.Global.getInt( context.getContentResolver(), Settings.Global.FORCE_ALLOW_ON_EXTERNAL, 0) != 0; } @VisibleForTesting protected boolean isAllow3rdPartyOnInternal(Context context) { return context.getResources().getBoolean( com.android.internal.R.bool.config_allow3rdPartyAppOnInternal); } private boolean isPackageCandidateVolume( ContextImpl context, ApplicationInfo app, VolumeInfo vol, IPackageManager pm) { final boolean forceAllowOnExternal = isForceAllowOnExternal(context); if (VolumeInfo.ID_PRIVATE_INTERNAL.equals(vol.getId())) { return app.isSystemApp() || isAllow3rdPartyOnInternal(context); } // System apps and apps demanding internal storage can't be moved // anywhere else if (app.isSystemApp()) { return false; } if (!forceAllowOnExternal && (app.installLocation == PackageInfo.INSTALL_LOCATION_INTERNAL_ONLY || app.installLocation == PackageInfo.INSTALL_LOCATION_UNSPECIFIED)) { return false; } // Gotta be able to write there if (!vol.isMountedWritable()) { return false; } // Moving into an ASEC on public primary is only option internal if (vol.isPrimaryPhysical()) { return app.isInternal(); } // Some apps can't be moved. (e.g. device admins) try { if (pm.isPackageDeviceAdminOnAnyUser(app.packageName)) { return false; } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } // Otherwise we can move to any private volume return (vol.getType() == VolumeInfo.TYPE_PRIVATE); } @Override public int movePrimaryStorage(VolumeInfo vol) { try { final String volumeUuid; if (VolumeInfo.ID_PRIVATE_INTERNAL.equals(vol.id)) { volumeUuid = StorageManager.UUID_PRIVATE_INTERNAL; } else if (vol.isPrimaryPhysical()) { volumeUuid = StorageManager.UUID_PRIMARY_PHYSICAL; } else { volumeUuid = Objects.requireNonNull(vol.fsUuid); } return mPM.movePrimaryStorage(volumeUuid); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public @Nullable VolumeInfo getPrimaryStorageCurrentVolume() { final StorageManager storage = mContext.getSystemService(StorageManager.class); final String volumeUuid = storage.getPrimaryStorageUuid(); return storage.findVolumeByQualifiedUuid(volumeUuid); } @Override public @NonNull List getPrimaryStorageCandidateVolumes() { final StorageManager storage = mContext.getSystemService(StorageManager.class); final VolumeInfo currentVol = getPrimaryStorageCurrentVolume(); final List vols = storage.getVolumes(); final List candidates = new ArrayList<>(); if (Objects.equals(StorageManager.UUID_PRIMARY_PHYSICAL, storage.getPrimaryStorageUuid()) && currentVol != null) { // TODO: support moving primary physical to emulated volume candidates.add(currentVol); } else { for (VolumeInfo vol : vols) { if (Objects.equals(vol, currentVol) || isPrimaryStorageCandidateVolume(vol)) { candidates.add(vol); } } } return candidates; } private static boolean isPrimaryStorageCandidateVolume(VolumeInfo vol) { // Private internal is always an option if (VolumeInfo.ID_PRIVATE_INTERNAL.equals(vol.getId())) { return true; } // Gotta be able to write there if (!vol.isMountedWritable()) { return false; } // We can move to any private volume return (vol.getType() == VolumeInfo.TYPE_PRIVATE); } @Override @UnsupportedAppUsage public void deletePackage(String packageName, IPackageDeleteObserver observer, int flags) { deletePackageAsUser(packageName, observer, flags, getUserId()); } @Override public void deletePackageAsUser(String packageName, IPackageDeleteObserver observer, int flags, int userId) { try { mPM.deletePackageAsUser(packageName, VERSION_CODE_HIGHEST, observer, userId, flags); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void clearApplicationUserData(String packageName, IPackageDataObserver observer) { try { mPM.clearApplicationUserData(packageName, observer, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void deleteApplicationCacheFiles(String packageName, IPackageDataObserver observer) { try { mPM.deleteApplicationCacheFiles(packageName, observer); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void deleteApplicationCacheFilesAsUser(String packageName, int userId, IPackageDataObserver observer) { try { mPM.deleteApplicationCacheFilesAsUser(packageName, userId, observer); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void freeStorageAndNotify(String volumeUuid, long idealStorageSize, IPackageDataObserver observer) { try { mPM.freeStorageAndNotify(volumeUuid, idealStorageSize, 0, observer); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void freeStorage(String volumeUuid, long freeStorageSize, IntentSender pi) { try { mPM.freeStorage(volumeUuid, freeStorageSize, 0, pi); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public String[] setDistractingPackageRestrictions(String[] packages, int distractionFlags) { try { return mPM.setDistractingPackageRestrictionsAsUser(packages, distractionFlags, mContext.getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public String[] setPackagesSuspended(String[] packageNames, boolean suspended, PersistableBundle appExtras, PersistableBundle launcherExtras, String dialogMessage) { final SuspendDialogInfo dialogInfo = !TextUtils.isEmpty(dialogMessage) ? new SuspendDialogInfo.Builder().setMessage(dialogMessage).build() : null; return setPackagesSuspended(packageNames, suspended, appExtras, launcherExtras, dialogInfo, 0); } @Override public String[] setPackagesSuspended(String[] packageNames, boolean suspended, PersistableBundle appExtras, PersistableBundle launcherExtras, SuspendDialogInfo dialogInfo) { return setPackagesSuspended(packageNames, suspended, appExtras, launcherExtras, dialogInfo, 0); } @Override public String[] setPackagesSuspended(String[] packageNames, boolean suspended, PersistableBundle appExtras, PersistableBundle launcherExtras, SuspendDialogInfo dialogInfo, int flags) { try { return mPM.setPackagesSuspendedAsUser(packageNames, suspended, appExtras, launcherExtras, dialogInfo, flags, mContext.getOpPackageName(), UserHandle.myUserId() /* suspendingUserId */, getUserId() /* targetUserId */); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public String[] getUnsuspendablePackages(String[] packageNames) { try { return mPM.getUnsuspendablePackagesForUser(packageNames, mContext.getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public Bundle getSuspendedPackageAppExtras() { try { return mPM.getSuspendedPackageAppExtras(mContext.getOpPackageName(), getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public String getSuspendingPackage(String suspendedPackage) { try { return mPM.getSuspendingPackage(suspendedPackage, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean isPackageSuspendedForUser(String packageName, int userId) { try { return mPM.isPackageSuspendedForUser(packageName, userId); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @Override public boolean isPackageSuspended(String packageName) throws NameNotFoundException { try { return isPackageSuspendedForUser(packageName, getUserId()); } catch (IllegalArgumentException ie) { throw new NameNotFoundException(packageName); } } @Override public boolean isPackageSuspended() { return isPackageSuspendedForUser(mContext.getOpPackageName(), getUserId()); } @Override public boolean isPackageQuarantined(@NonNull String packageName) throws NameNotFoundException { try { return mPM.isPackageQuarantinedForUser(packageName, getUserId()); } catch (IllegalArgumentException ie) { throw new NameNotFoundException(packageName); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean isPackageStopped(@NonNull String packageName) throws NameNotFoundException { try { return mPM.isPackageStoppedForUser(packageName, getUserId()); } catch (IllegalArgumentException ie) { throw new NameNotFoundException(packageName); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @Override public void setApplicationCategoryHint(String packageName, int categoryHint) { try { mPM.setApplicationCategoryHint(packageName, categoryHint, mContext.getOpPackageName()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override @UnsupportedAppUsage public void getPackageSizeInfoAsUser(String packageName, int userHandle, IPackageStatsObserver observer) { final String msg = "Shame on you for calling the hidden API " + "getPackageSizeInfoAsUser(). Shame!"; if (mContext.getApplicationInfo().targetSdkVersion >= Build.VERSION_CODES.O) { throw new UnsupportedOperationException(msg); } else if (observer != null) { Log.d(TAG, msg); try { observer.onGetStatsCompleted(null, false); } catch (RemoteException ignored) { } } } @Override public void addPackageToPreferred(String packageName) { Log.w(TAG, "addPackageToPreferred() is a no-op"); } @Override public void removePackageFromPreferred(String packageName) { Log.w(TAG, "removePackageFromPreferred() is a no-op"); } @Override public List getPreferredPackages(int flags) { Log.w(TAG, "getPreferredPackages() is a no-op"); return Collections.emptyList(); } @Override public void addPreferredActivity(IntentFilter filter, int match, ComponentName[] set, ComponentName activity) { try { mPM.addPreferredActivity(filter, match, set, activity, getUserId(), false); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void addPreferredActivityAsUser(IntentFilter filter, int match, ComponentName[] set, ComponentName activity, int userId) { try { mPM.addPreferredActivity(filter, match, set, activity, userId, false); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void replacePreferredActivity(IntentFilter filter, int match, ComponentName[] set, ComponentName activity) { try { mPM.replacePreferredActivity(filter, match, set, activity, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void replacePreferredActivityAsUser(IntentFilter filter, int match, ComponentName[] set, ComponentName activity, int userId) { try { mPM.replacePreferredActivity(filter, match, set, activity, userId); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void clearPackagePreferredActivities(String packageName) { try { mPM.clearPackagePreferredActivities(packageName); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void addUniquePreferredActivity(@NonNull IntentFilter filter, int match, @Nullable ComponentName[] set, @NonNull ComponentName activity) { try { mPM.addPreferredActivity(filter, match, set, activity, getUserId(), true); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public int getPreferredActivities(List outFilters, List outActivities, String packageName) { try { return mPM.getPreferredActivities(outFilters, outActivities, packageName); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public ComponentName getHomeActivities(List outActivities) { try { return mPM.getHomeActivities(outActivities); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void setSyntheticAppDetailsActivityEnabled(String packageName, boolean enabled) { try { ComponentName componentName = new ComponentName(packageName, APP_DETAILS_ACTIVITY_CLASS_NAME); mPM.setComponentEnabledSetting(componentName, enabled ? COMPONENT_ENABLED_STATE_DEFAULT : COMPONENT_ENABLED_STATE_DISABLED, DONT_KILL_APP, getUserId(), mContext.getOpPackageName()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean getSyntheticAppDetailsActivityEnabled(String packageName) { try { ComponentName componentName = new ComponentName(packageName, APP_DETAILS_ACTIVITY_CLASS_NAME); int state = mPM.getComponentEnabledSetting(componentName, getUserId()); return state == COMPONENT_ENABLED_STATE_ENABLED || state == COMPONENT_ENABLED_STATE_DEFAULT; } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void setComponentEnabledSetting(ComponentName componentName, int newState, int flags) { try { mPM.setComponentEnabledSetting(componentName, newState, flags, getUserId(), mContext.getOpPackageName()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void setComponentEnabledSettings(List settings) { try { mPM.setComponentEnabledSettings(settings, getUserId(), mContext.getOpPackageName()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public int getComponentEnabledSetting(ComponentName componentName) { try { return mPM.getComponentEnabledSetting(componentName, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void setApplicationEnabledSetting(String packageName, int newState, int flags) { try { mPM.setApplicationEnabledSetting(packageName, newState, flags, getUserId(), mContext.getOpPackageName()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public int getApplicationEnabledSetting(String packageName) { try { return mPM.getApplicationEnabledSetting(packageName, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void flushPackageRestrictionsAsUser(int userId) { try { mPM.flushPackageRestrictionsAsUser(userId); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean setApplicationHiddenSettingAsUser(String packageName, boolean hidden, UserHandle user) { try { return mPM.setApplicationHiddenSettingAsUser(packageName, hidden, user.getIdentifier()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean getApplicationHiddenSettingAsUser(String packageName, UserHandle user) { try { return mPM.getApplicationHiddenSettingAsUser(packageName, user.getIdentifier()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @Override public void setSystemAppState(String packageName, @SystemAppState int state) { try { switch (state) { case SYSTEM_APP_STATE_HIDDEN_UNTIL_INSTALLED_HIDDEN: mPM.setSystemAppHiddenUntilInstalled(packageName, true); break; case SYSTEM_APP_STATE_HIDDEN_UNTIL_INSTALLED_VISIBLE: mPM.setSystemAppHiddenUntilInstalled(packageName, false); break; case SYSTEM_APP_STATE_INSTALLED: mPM.setSystemAppInstallState(packageName, true, getUserId()); break; case SYSTEM_APP_STATE_UNINSTALLED: mPM.setSystemAppInstallState(packageName, false, getUserId()); break; default: } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @Override public KeySet getKeySetByAlias(String packageName, String alias) { Objects.requireNonNull(packageName); Objects.requireNonNull(alias); try { return mPM.getKeySetByAlias(packageName, alias); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @Override public KeySet getSigningKeySet(String packageName) { Objects.requireNonNull(packageName); try { return mPM.getSigningKeySet(packageName); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @Override public boolean isSignedBy(String packageName, KeySet ks) { Objects.requireNonNull(packageName); Objects.requireNonNull(ks); try { return mPM.isPackageSignedByKeySet(packageName, ks); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** @hide */ @Override public boolean isSignedByExactly(String packageName, KeySet ks) { Objects.requireNonNull(packageName); Objects.requireNonNull(ks); try { return mPM.isPackageSignedByKeySetExactly(packageName, ks); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** * @hide */ @Override public VerifierDeviceIdentity getVerifierDeviceIdentity() { try { return mPM.getVerifierDeviceIdentity(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public boolean isUpgrade() { return isDeviceUpgrading(); } @Override public boolean isDeviceUpgrading() { try { return mPM.isDeviceUpgrading(); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public PackageInstaller getPackageInstaller() { if (mInstaller == null) { try { mInstaller = new PackageInstaller(mPM.getPackageInstaller(), mContext.getPackageName(), mContext.getAttributionTag(), getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } return mInstaller; } @Override public boolean isPackageAvailable(String packageName) { try { return mPM.isPackageAvailable(packageName, getUserId()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** * @hide */ @Override public void addCrossProfileIntentFilter(IntentFilter filter, int sourceUserId, int targetUserId, int flags) { try { mPM.addCrossProfileIntentFilter(filter, mContext.getOpPackageName(), sourceUserId, targetUserId, flags); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** * @hide */ @Override public boolean removeCrossProfileIntentFilter(IntentFilter filter, int sourceUserId, int targetUserId, int flags) { try { return mPM.removeCrossProfileIntentFilter(filter, mContext.getOpPackageName(), sourceUserId, targetUserId, flags); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** * @hide */ @Override public void clearCrossProfileIntentFilters(int sourceUserId) { try { mPM.clearCrossProfileIntentFilters(sourceUserId, mContext.getOpPackageName()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** * @hide */ public Drawable loadItemIcon(PackageItemInfo itemInfo, ApplicationInfo appInfo) { Drawable dr = loadUnbadgedItemIcon(itemInfo, appInfo); if (itemInfo.showUserIcon != UserHandle.USER_NULL) { return dr; } return getUserBadgedIcon(dr, new UserHandle(getUserId())); } /** * @hide */ public Drawable loadUnbadgedItemIcon(@NonNull PackageItemInfo itemInfo, @Nullable ApplicationInfo appInfo) { if (itemInfo.showUserIcon != UserHandle.USER_NULL) { // Indicates itemInfo is for a different user (e.g. a profile's parent), so use a // generic user icon (users generally lack permission to view each other's actual icons) int targetUserId = itemInfo.showUserIcon; return UserIcons.getDefaultUserIcon( mContext.getResources(), targetUserId, /* light= */ false); } Drawable dr = null; if (itemInfo.packageName != null) { if (itemInfo.isArchived) { dr = getArchivedAppIcon(itemInfo.packageName); } else { dr = getDrawable(itemInfo.packageName, itemInfo.icon, appInfo); } } if (dr == null && itemInfo != appInfo && appInfo != null) { dr = loadUnbadgedItemIcon(appInfo, appInfo); } if (dr == null) { dr = itemInfo.loadDefaultIcon(this); } return dr; } private Drawable getBadgedDrawable(Drawable drawable, Drawable badgeDrawable, Rect badgeLocation, boolean tryBadgeInPlace) { final int badgedWidth = drawable.getIntrinsicWidth(); final int badgedHeight = drawable.getIntrinsicHeight(); final boolean canBadgeInPlace = tryBadgeInPlace && (drawable instanceof BitmapDrawable) && ((BitmapDrawable) drawable).getBitmap().isMutable(); final Bitmap bitmap; if (canBadgeInPlace) { bitmap = ((BitmapDrawable) drawable).getBitmap(); } else { bitmap = Bitmap.createBitmap(badgedWidth, badgedHeight, Bitmap.Config.ARGB_8888); } Canvas canvas = new Canvas(bitmap); if (!canBadgeInPlace) { drawable.setBounds(0, 0, badgedWidth, badgedHeight); drawable.draw(canvas); } if (badgeLocation != null) { if (badgeLocation.left < 0 || badgeLocation.top < 0 || badgeLocation.width() > badgedWidth || badgeLocation.height() > badgedHeight) { throw new IllegalArgumentException("Badge location " + badgeLocation + " not in badged drawable bounds " + new Rect(0, 0, badgedWidth, badgedHeight)); } badgeDrawable.setBounds(0, 0, badgeLocation.width(), badgeLocation.height()); canvas.save(); canvas.translate(badgeLocation.left, badgeLocation.top); badgeDrawable.draw(canvas); canvas.restore(); } else { badgeDrawable.setBounds(0, 0, badgedWidth, badgedHeight); badgeDrawable.draw(canvas); } if (!canBadgeInPlace) { BitmapDrawable mergedDrawable = new BitmapDrawable(mContext.getResources(), bitmap); if (drawable instanceof BitmapDrawable) { BitmapDrawable bitmapDrawable = (BitmapDrawable) drawable; mergedDrawable.setTargetDensity(bitmapDrawable.getBitmap().getDensity()); } return mergedDrawable; } return drawable; } private boolean hasUserBadge(int userId) { return getUserManager().hasBadge(userId); } /** * @hide */ @Override public int getInstallReason(String packageName, UserHandle user) { try { return mPM.getInstallReason(packageName, user.getIdentifier()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } /** {@hide} */ private static class MoveCallbackDelegate extends IPackageMoveObserver.Stub implements Handler.Callback { private static final int MSG_CREATED = 1; private static final int MSG_STATUS_CHANGED = 2; final MoveCallback mCallback; final Handler mHandler; public MoveCallbackDelegate(MoveCallback callback, Looper looper) { mCallback = callback; mHandler = new Handler(looper, this); } @Override public boolean handleMessage(Message msg) { switch (msg.what) { case MSG_CREATED: { final SomeArgs args = (SomeArgs) msg.obj; mCallback.onCreated(args.argi1, (Bundle) args.arg2); args.recycle(); return true; } case MSG_STATUS_CHANGED: { final SomeArgs args = (SomeArgs) msg.obj; mCallback.onStatusChanged(args.argi1, args.argi2, (long) args.arg3); args.recycle(); return true; } } return false; } @Override public void onCreated(int moveId, Bundle extras) { final SomeArgs args = SomeArgs.obtain(); args.argi1 = moveId; args.arg2 = extras; mHandler.obtainMessage(MSG_CREATED, args).sendToTarget(); } @Override public void onStatusChanged(int moveId, int status, long estMillis) { final SomeArgs args = SomeArgs.obtain(); args.argi1 = moveId; args.argi2 = status; args.arg3 = estMillis; mHandler.obtainMessage(MSG_STATUS_CHANGED, args).sendToTarget(); } } private final ContextImpl mContext; @UnsupportedAppUsage private final IPackageManager mPM; /** Assume locked until we hear otherwise */ private volatile boolean mUserUnlocked = false; private static final Object sSync = new Object(); private static ArrayMap> sIconCache = new ArrayMap>(); private static ArrayMap> sStringCache = new ArrayMap>(); @Override public boolean canRequestPackageInstalls() { try { return mPM.canRequestPackageInstalls(mContext.getPackageName(), getUserId()); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public ComponentName getInstantAppResolverSettingsComponent() { try { return mPM.getInstantAppResolverSettingsComponent(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public ComponentName getInstantAppInstallerComponent() { try { return mPM.getInstantAppInstallerComponent(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public String getInstantAppAndroidId(String packageName, UserHandle user) { try { return mPM.getInstantAppAndroidId(packageName, user.getIdentifier()); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } private static class DexModuleRegisterResult { final String dexModulePath; final boolean success; final String message; private DexModuleRegisterResult(String dexModulePath, boolean success, String message) { this.dexModulePath = dexModulePath; this.success = success; this.message = message; } } private static class DexModuleRegisterCallbackDelegate extends android.content.pm.IDexModuleRegisterCallback.Stub implements Handler.Callback { private static final int MSG_DEX_MODULE_REGISTERED = 1; private final DexModuleRegisterCallback callback; private final Handler mHandler; DexModuleRegisterCallbackDelegate(@NonNull DexModuleRegisterCallback callback) { this.callback = callback; mHandler = new Handler(Looper.getMainLooper(), this); } @Override public void onDexModuleRegistered(@NonNull String dexModulePath, boolean success, @Nullable String message)throws RemoteException { mHandler.obtainMessage(MSG_DEX_MODULE_REGISTERED, new DexModuleRegisterResult(dexModulePath, success, message)).sendToTarget(); } @Override public boolean handleMessage(Message msg) { if (msg.what != MSG_DEX_MODULE_REGISTERED) { return false; } DexModuleRegisterResult result = (DexModuleRegisterResult)msg.obj; callback.onDexModuleRegistered(result.dexModulePath, result.success, result.message); return true; } } @Override public void registerDexModule(@NonNull String dexModule, @Nullable DexModuleRegisterCallback callback) { // Create the callback delegate to be passed to package manager service. DexModuleRegisterCallbackDelegate callbackDelegate = null; if (callback != null) { callbackDelegate = new DexModuleRegisterCallbackDelegate(callback); } // Check if this is a shared module by looking if the others can read it. boolean isSharedModule = false; try { StructStat stat = Os.stat(dexModule); if ((OsConstants.S_IROTH & stat.st_mode) != 0) { isSharedModule = true; } } catch (ErrnoException e) { if (callbackDelegate != null) { callback.onDexModuleRegistered(dexModule, false, "Could not get stat the module file: " + e.getMessage()); } return; } // Invoke the package manager service. try { mPM.registerDexModule(mContext.getPackageName(), dexModule, isSharedModule, callbackDelegate); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public CharSequence getHarmfulAppWarning(String packageName) { try { return mPM.getHarmfulAppWarning(packageName, getUserId()); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public void setHarmfulAppWarning(String packageName, CharSequence warning) { try { mPM.setHarmfulAppWarning(packageName, warning, getUserId()); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public ArtManager getArtManager() { if (mArtManager == null) { try { mArtManager = new ArtManager(mContext, mPM.getArtManager()); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } return mArtManager; } @Override public String getDefaultTextClassifierPackageName() { try { return mPM.getDefaultTextClassifierPackageName(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public String getSystemTextClassifierPackageName() { try { return mPM.getSystemTextClassifierPackageName(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public String getAttentionServicePackageName() { try { return mPM.getAttentionServicePackageName(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public String getRotationResolverPackageName() { try { return mPM.getRotationResolverPackageName(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public String getWellbeingPackageName() { try { return mPM.getWellbeingPackageName(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public String getAppPredictionServicePackageName() { try { return mPM.getAppPredictionServicePackageName(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public String getSystemCaptionsServicePackageName() { try { return mPM.getSystemCaptionsServicePackageName(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public String getSetupWizardPackageName() { try { return mPM.getSetupWizardPackageName(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public String getIncidentReportApproverPackageName() { try { return mPM.getIncidentReportApproverPackageName(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public boolean isPackageStateProtected(String packageName, int userId) { try { return mPM.isPackageStateProtected(packageName, userId); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } public void sendDeviceCustomizationReadyBroadcast() { try { mPM.sendDeviceCustomizationReadyBroadcast(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public boolean isAutoRevokeWhitelisted() { try { return mPM.isAutoRevokeWhitelisted(mContext.getPackageName()); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public void setMimeGroup(String mimeGroup, Set mimeTypes) { try { mPM.setMimeGroup(mContext.getPackageName(), mimeGroup, new ArrayList<>(mimeTypes)); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @NonNull @Override public Set getMimeGroup(String group) { try { List mimeGroup = mPM.getMimeGroup(mContext.getPackageName(), group); return new ArraySet<>(mimeGroup); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public Property getProperty(String propertyName, String packageName) throws NameNotFoundException { Objects.requireNonNull(packageName); Objects.requireNonNull(propertyName); return getPropertyAsUser(propertyName, packageName, null /* className */, getUserId()); } @Override public Property getProperty(String propertyName, ComponentName component) throws NameNotFoundException { Objects.requireNonNull(component); Objects.requireNonNull(propertyName); return getPropertyAsUser(propertyName, component.getPackageName(), component.getClassName(), getUserId()); } @Override public Property getPropertyAsUser(@NonNull String propertyName, @NonNull String packageName, @Nullable String className, int userId) throws NameNotFoundException { Objects.requireNonNull(packageName); Objects.requireNonNull(propertyName); try { final Property property = mPM.getPropertyAsUser(propertyName, packageName, className, userId); if (property == null) { throw new NameNotFoundException(); } return property; } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public List queryApplicationProperty(String propertyName) { Objects.requireNonNull(propertyName); try { final ParceledListSlice parceledList = mPM.queryProperty(propertyName, TYPE_APPLICATION); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public List queryActivityProperty(String propertyName) { Objects.requireNonNull(propertyName); try { final ParceledListSlice parceledList = mPM.queryProperty(propertyName, TYPE_ACTIVITY); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public List queryProviderProperty(String propertyName) { Objects.requireNonNull(propertyName); try { final ParceledListSlice parceledList = mPM.queryProperty(propertyName, TYPE_PROVIDER); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public List queryReceiverProperty(String propertyName) { Objects.requireNonNull(propertyName); try { final ParceledListSlice parceledList = mPM.queryProperty(propertyName, TYPE_RECEIVER); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public List queryServiceProperty(String propertyName) { Objects.requireNonNull(propertyName); try { final ParceledListSlice parceledList = mPM.queryProperty(propertyName, TYPE_SERVICE); if (parceledList == null) { return Collections.emptyList(); } return parceledList.getList(); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public boolean canPackageQuery(@NonNull String sourcePackageName, @NonNull String targetPackageName) throws NameNotFoundException { Objects.requireNonNull(sourcePackageName); Objects.requireNonNull(targetPackageName); return canPackageQuery(sourcePackageName, new String[]{targetPackageName})[0]; } @Override @NonNull public boolean[] canPackageQuery(@NonNull String sourcePackageName, @NonNull String[] targetPackageNames) throws NameNotFoundException { Objects.requireNonNull(sourcePackageName); Objects.requireNonNull(targetPackageNames); try { return mPM.canPackageQuery(sourcePackageName, targetPackageNames, getUserId()); } catch (ParcelableException e) { e.maybeRethrow(PackageManager.NameNotFoundException.class); throw new RuntimeException(e); } catch (RemoteException re) { throw re.rethrowAsRuntimeException(); } } @Override public void makeUidVisible(int recipientUid, int visibleUid) { try { mPM.makeUidVisible(recipientUid, visibleUid); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public @Nullable ArchivedPackageInfo getArchivedPackage(@NonNull String packageName) { try { var parcel = mPM.getArchivedPackage(packageName, mContext.getUserId()); if (parcel == null) { return null; } return new ArchivedPackageInfo(parcel); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public boolean canUserUninstall(String packageName, UserHandle user) { try { return mPM.getBlockUninstallForUser(packageName, user.getIdentifier()); } catch (RemoteException e) { throw e.rethrowAsRuntimeException(); } } @Override public boolean shouldShowNewAppInstalledNotification() { return Settings.Global.getInt(mContext.getContentResolver(), Settings.Global.SHOW_NEW_APP_INSTALLED_NOTIFICATION_ENABLED, 0) == 1; } @Override public void relinquishUpdateOwnership(String targetPackage) { Objects.requireNonNull(targetPackage); try { mPM.relinquishUpdateOwnership(targetPackage); } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void registerPackageMonitorCallback(@NonNull IRemoteCallback callback, int userId) { Objects.requireNonNull(callback); try { mPM.registerPackageMonitorCallback(callback, userId); synchronized (mPackageMonitorCallbacks) { if (mPackageMonitorCallbacks.contains(callback)) { throw new IllegalStateException( "registerPackageMonitorCallback: callback already registered: " + callback); } mPackageMonitorCallbacks.add(callback); } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Override public void unregisterPackageMonitorCallback(@NonNull IRemoteCallback callback) { Objects.requireNonNull(callback); try { mPM.unregisterPackageMonitorCallback(callback); synchronized (mPackageMonitorCallbacks) { mPackageMonitorCallbacks.remove(callback); } } catch (RemoteException e) { throw e.rethrowFromSystemServer(); } } @Nullable private Drawable getArchivedAppIcon(String packageName) { try { Bitmap archivedAppIcon = mPM.getArchivedAppIcon(packageName, new UserHandle(getUserId()), mContext.getPackageName()); if (archivedAppIcon == null) { return null; } return new BitmapDrawable(null, archivedAppIcon); } catch (RemoteException e) { Slog.e(TAG, "Failed to retrieve archived app icon: " + e.getMessage()); return null; } } @Override public T parseAndroidManifest(@NonNull File apkFile, @NonNull Function parserFunction) throws IOException { Objects.requireNonNull(apkFile, "apkFile cannot be null"); Objects.requireNonNull(parserFunction, "parserFunction cannot be null"); try (XmlResourceParser xmlResourceParser = getAndroidManifestParser(apkFile)) { return parserFunction.apply(xmlResourceParser); } catch (IOException e) { Log.w(TAG, "Failed to get the android manifest parser", e); throw e; } } private static XmlResourceParser getAndroidManifestParser(@NonNull File apkFile) throws IOException { ApkAssets apkAssets = null; try { apkAssets = ApkAssets.loadFromPath(apkFile.getAbsolutePath()); return apkAssets.openXml(ApkLiteParseUtils.ANDROID_MANIFEST_FILENAME); } finally { if (apkAssets != null) { try { apkAssets.close(); } catch (Throwable ignored) { Log.w(TAG, "Failed to close apkAssets", ignored); } } } } @Override public TypedArray extractPackageItemInfoAttributes(PackageItemInfo info, String name, String rootTag, int[] attributes) { if (info == null || info.metaData == null) { return null; } try (XmlResourceParser parser = info.loadXmlMetaData(this, name)) { if (parser == null) { Log.w(TAG, "No " + name + " metadata"); return null; } final AttributeSet attrs = Xml.asAttributeSet(parser); while (true) { final int type = parser.next(); if (type == XmlPullParser.END_DOCUMENT || type == XmlPullParser.START_TAG) { break; } } if (!TextUtils.equals(parser.getName(), rootTag)) { Log.w(TAG, "Metadata does not start with " + name + " tag"); return null; } return getResourcesForApplication(info.getApplicationInfo()) .obtainAttributes(attrs, attributes); } catch (PackageManager.NameNotFoundException | IOException | XmlPullParserException e) { Log.e(TAG, "Error parsing: " + info.packageName, e); return null; } } }