4134 lines
146 KiB
Java
4134 lines
146 KiB
Java
/*
|
|
* 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<MoveCallbackDelegate> mDelegates = new ArrayList<>();
|
|
|
|
@NonNull
|
|
@GuardedBy("mPackageMonitorCallbacks")
|
|
private final ArraySet<IRemoteCallback> 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<ResolveInfo> 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<ResolveInfo> 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<PermissionGroupInfo> 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<PermissionInfo> queryPermissionsByGroup(String groupName, int flags)
|
|
throws NameNotFoundException {
|
|
final List<PermissionInfo> 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<List<String>> 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<String> 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<SharedLibraryInfo> getSharedLibraries(int flags) {
|
|
return this.getSharedLibraries(PackageInfoFlags.of(flags));
|
|
}
|
|
|
|
/** @hide
|
|
* @param flags */
|
|
@Override
|
|
public @NonNull List<SharedLibraryInfo> getSharedLibraries(PackageInfoFlags flags) {
|
|
return getSharedLibrariesAsUser(flags, getUserId());
|
|
}
|
|
|
|
/** @hide */
|
|
@Override
|
|
public @NonNull List<SharedLibraryInfo> getSharedLibrariesAsUser(int flags, int userId) {
|
|
return getSharedLibrariesAsUser(PackageInfoFlags.of(flags), userId);
|
|
}
|
|
|
|
/** @hide */
|
|
@Override
|
|
@SuppressWarnings("unchecked")
|
|
public @NonNull List<SharedLibraryInfo> getSharedLibrariesAsUser(PackageInfoFlags flags,
|
|
int userId) {
|
|
try {
|
|
ParceledListSlice<SharedLibraryInfo> 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<SharedLibraryInfo> getDeclaredSharedLibraries(@NonNull String packageName,
|
|
int flags) {
|
|
return getDeclaredSharedLibraries(packageName, PackageInfoFlags.of(flags));
|
|
}
|
|
|
|
@NonNull
|
|
@Override
|
|
@SuppressWarnings("unchecked")
|
|
public List<SharedLibraryInfo> getDeclaredSharedLibraries(@NonNull String packageName,
|
|
PackageInfoFlags flags) {
|
|
try {
|
|
ParceledListSlice<SharedLibraryInfo> 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<FeatureInfo> parceledList =
|
|
mPM.getSystemAvailableFeatures();
|
|
if (parceledList == null) {
|
|
return new FeatureInfo[0];
|
|
}
|
|
final List<FeatureInfo> 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<HasSystemFeatureQuery, Boolean>
|
|
mHasSystemFeatureCache =
|
|
new PropertyInvalidatedCache<HasSystemFeatureQuery, Boolean>(
|
|
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<String> 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<byte[]> encodeCertificates(List<Certificate> certs) throws
|
|
CertificateEncodingException {
|
|
if (certs == null) {
|
|
return null;
|
|
}
|
|
List<byte[]> 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<Certificate> 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<ApkChecksum> 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<Integer, GetPackagesForUidResult>
|
|
mGetPackagesForUidCache =
|
|
new PropertyInvalidatedCache<Integer, GetPackagesForUidResult>(
|
|
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<ModuleInfo> 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<PackageInfo> getInstalledPackages(int flags) {
|
|
return getInstalledPackages(PackageInfoFlags.of(flags));
|
|
}
|
|
|
|
@SuppressWarnings("unchecked")
|
|
@Override
|
|
public List<PackageInfo> getInstalledPackages(PackageInfoFlags flags) {
|
|
return getInstalledPackagesAsUser(flags, getUserId());
|
|
}
|
|
|
|
/** @hide */
|
|
@Override
|
|
@SuppressWarnings("unchecked")
|
|
public List<PackageInfo> getInstalledPackagesAsUser(int flags, int userId) {
|
|
return getInstalledPackagesAsUser(PackageInfoFlags.of(flags), userId);
|
|
}
|
|
|
|
/** @hide */
|
|
@Override
|
|
@SuppressWarnings("unchecked")
|
|
public List<PackageInfo> getInstalledPackagesAsUser(PackageInfoFlags flags, int userId) {
|
|
try {
|
|
ParceledListSlice<PackageInfo> 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<PackageInfo> getPackagesHoldingPermissions(String[] permissions, int flags) {
|
|
return this.getPackagesHoldingPermissions(permissions, PackageInfoFlags.of(flags));
|
|
}
|
|
|
|
@SuppressWarnings("unchecked")
|
|
@Override
|
|
public List<PackageInfo> getPackagesHoldingPermissions(String[] permissions,
|
|
PackageInfoFlags flags) {
|
|
final int userId = getUserId();
|
|
try {
|
|
ParceledListSlice<PackageInfo> 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<ApplicationInfo> getInstalledApplications(int flags) {
|
|
return getInstalledApplicationsAsUser(flags, getUserId());
|
|
}
|
|
|
|
@Override
|
|
public List<ApplicationInfo> getInstalledApplications(ApplicationInfoFlags flags) {
|
|
return getInstalledApplicationsAsUser(flags, getUserId());
|
|
}
|
|
|
|
/** @hide */
|
|
@SuppressWarnings("unchecked")
|
|
@Override
|
|
public List<ApplicationInfo> getInstalledApplicationsAsUser(int flags, int userId) {
|
|
return getInstalledApplicationsAsUser(ApplicationInfoFlags.of(flags), userId);
|
|
}
|
|
|
|
/** @hide */
|
|
@SuppressWarnings("unchecked")
|
|
@Override
|
|
public List<ApplicationInfo> getInstalledApplicationsAsUser(ApplicationInfoFlags flags,
|
|
int userId) {
|
|
try {
|
|
ParceledListSlice<ApplicationInfo> 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<InstantAppInfo> getInstantApps() {
|
|
try {
|
|
ParceledListSlice<InstantAppInfo> 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<ResolveInfo> queryIntentActivities(Intent intent, int flags) {
|
|
return queryIntentActivities(intent, ResolveInfoFlags.of(flags));
|
|
}
|
|
|
|
@Override
|
|
public List<ResolveInfo> queryIntentActivities(Intent intent, ResolveInfoFlags flags) {
|
|
return queryIntentActivitiesAsUser(intent, flags, getUserId());
|
|
}
|
|
|
|
/** @hide Same as above but for a specific user */
|
|
@Override
|
|
public List<ResolveInfo> 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<ResolveInfo> queryIntentActivitiesAsUser(Intent intent, ResolveInfoFlags flags,
|
|
int userId) {
|
|
try {
|
|
ParceledListSlice<ResolveInfo> 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<ResolveInfo> 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<ResolveInfo> queryIntentActivityOptions(ComponentName caller,
|
|
List<Intent> 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<ResolveInfo> 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<ResolveInfo> queryBroadcastReceiversAsUser(Intent intent, int flags, int userId) {
|
|
return queryBroadcastReceiversAsUser(intent, ResolveInfoFlags.of(flags), userId);
|
|
}
|
|
|
|
/**
|
|
* @hide
|
|
*/
|
|
@Override
|
|
@SuppressWarnings("unchecked")
|
|
public List<ResolveInfo> queryBroadcastReceiversAsUser(Intent intent, ResolveInfoFlags flags,
|
|
int userId) {
|
|
try {
|
|
ParceledListSlice<ResolveInfo> 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<ResolveInfo> queryBroadcastReceivers(Intent intent, int flags) {
|
|
return queryBroadcastReceivers(intent, ResolveInfoFlags.of(flags));
|
|
}
|
|
|
|
@Override
|
|
public List<ResolveInfo> 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<ResolveInfo> queryIntentServicesAsUser(Intent intent, int flags, int userId) {
|
|
return queryIntentServicesAsUser(intent, ResolveInfoFlags.of(flags), userId);
|
|
}
|
|
|
|
@Override
|
|
@SuppressWarnings("unchecked")
|
|
public List<ResolveInfo> queryIntentServicesAsUser(Intent intent, ResolveInfoFlags flags,
|
|
int userId) {
|
|
try {
|
|
ParceledListSlice<ResolveInfo> 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<ResolveInfo> queryIntentServices(Intent intent, int flags) {
|
|
return queryIntentServices(intent, ResolveInfoFlags.of(flags));
|
|
}
|
|
|
|
@Override
|
|
public List<ResolveInfo> queryIntentServices(Intent intent, ResolveInfoFlags flags) {
|
|
return queryIntentServicesAsUser(intent, flags, getUserId());
|
|
}
|
|
|
|
@Override
|
|
public List<ResolveInfo> queryIntentContentProvidersAsUser(
|
|
Intent intent, int flags, int userId) {
|
|
return queryIntentContentProvidersAsUser(intent, ResolveInfoFlags.of(flags), userId);
|
|
}
|
|
|
|
@Override
|
|
@SuppressWarnings("unchecked")
|
|
public List<ResolveInfo> queryIntentContentProvidersAsUser(
|
|
Intent intent, ResolveInfoFlags flags, int userId) {
|
|
try {
|
|
ParceledListSlice<ResolveInfo> 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<ResolveInfo> queryIntentContentProviders(Intent intent, int flags) {
|
|
return queryIntentContentProviders(intent, ResolveInfoFlags.of(flags));
|
|
}
|
|
|
|
@Override
|
|
public List<ResolveInfo> 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<ProviderInfo> queryContentProviders(String processName, int uid, int flags) {
|
|
return queryContentProviders(processName, uid, ComponentInfoFlags.of(flags));
|
|
}
|
|
|
|
@Override
|
|
public List<ProviderInfo> queryContentProviders(String processName, int uid,
|
|
ComponentInfoFlags flags) {
|
|
return queryContentProviders(processName, uid, flags, null);
|
|
}
|
|
|
|
@Override
|
|
public List<ProviderInfo> queryContentProviders(String processName,
|
|
int uid, int flags, String metaDataKey) {
|
|
return queryContentProviders(processName, uid, ComponentInfoFlags.of(flags), metaDataKey);
|
|
}
|
|
|
|
@Override
|
|
@SuppressWarnings("unchecked")
|
|
public List<ProviderInfo> queryContentProviders(String processName,
|
|
int uid, ComponentInfoFlags flags, String metaDataKey) {
|
|
try {
|
|
ParceledListSlice<ProviderInfo> 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<InstrumentationInfo> queryInstrumentation(
|
|
String targetPackage, int flags) {
|
|
try {
|
|
ParceledListSlice<InstrumentationInfo> 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<Drawable.ConstantState> 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<CharSequence> 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<CharSequence>(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<String> 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<IntentFilterVerificationInfo> getIntentFilterVerifications(String packageName) {
|
|
try {
|
|
ParceledListSlice<IntentFilterVerificationInfo> parceledList =
|
|
mPM.getIntentFilterVerifications(packageName);
|
|
if (parceledList == null) {
|
|
return Collections.emptyList();
|
|
}
|
|
return parceledList.getList();
|
|
} catch (RemoteException e) {
|
|
throw e.rethrowFromSystemServer();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
@SuppressWarnings("unchecked")
|
|
public List<IntentFilter> getAllIntentFilters(String packageName) {
|
|
try {
|
|
ParceledListSlice<IntentFilter> 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<MoveCallbackDelegate> 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<VolumeInfo> getPackageCandidateVolumes(ApplicationInfo app) {
|
|
final StorageManager storageManager = mContext.getSystemService(StorageManager.class);
|
|
return getPackageCandidateVolumes(app, storageManager, mPM);
|
|
}
|
|
|
|
@VisibleForTesting
|
|
protected @NonNull List<VolumeInfo> getPackageCandidateVolumes(ApplicationInfo app,
|
|
StorageManager storageManager, IPackageManager pm) {
|
|
final VolumeInfo currentVol = getPackageCurrentVolume(app, storageManager);
|
|
final List<VolumeInfo> vols = storageManager.getVolumes();
|
|
final List<VolumeInfo> 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<VolumeInfo> getPrimaryStorageCandidateVolumes() {
|
|
final StorageManager storage = mContext.getSystemService(StorageManager.class);
|
|
final VolumeInfo currentVol = getPrimaryStorageCurrentVolume();
|
|
final List<VolumeInfo> vols = storage.getVolumes();
|
|
final List<VolumeInfo> 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<PackageInfo> 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<IntentFilter> outFilters,
|
|
List<ComponentName> outActivities, String packageName) {
|
|
try {
|
|
return mPM.getPreferredActivities(outFilters, outActivities, packageName);
|
|
} catch (RemoteException e) {
|
|
throw e.rethrowFromSystemServer();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public ComponentName getHomeActivities(List<ResolveInfo> 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<ComponentEnabledSetting> 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<ResourceName, WeakReference<Drawable.ConstantState>> sIconCache
|
|
= new ArrayMap<ResourceName, WeakReference<Drawable.ConstantState>>();
|
|
private static ArrayMap<ResourceName, WeakReference<CharSequence>> sStringCache
|
|
= new ArrayMap<ResourceName, WeakReference<CharSequence>>();
|
|
|
|
@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<String> mimeTypes) {
|
|
try {
|
|
mPM.setMimeGroup(mContext.getPackageName(), mimeGroup, new ArrayList<>(mimeTypes));
|
|
} catch (RemoteException e) {
|
|
throw e.rethrowAsRuntimeException();
|
|
}
|
|
}
|
|
|
|
@NonNull
|
|
@Override
|
|
public Set<String> getMimeGroup(String group) {
|
|
try {
|
|
List<String> 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<Property> queryApplicationProperty(String propertyName) {
|
|
Objects.requireNonNull(propertyName);
|
|
try {
|
|
final ParceledListSlice<Property> parceledList =
|
|
mPM.queryProperty(propertyName, TYPE_APPLICATION);
|
|
if (parceledList == null) {
|
|
return Collections.emptyList();
|
|
}
|
|
return parceledList.getList();
|
|
} catch (RemoteException e) {
|
|
throw e.rethrowAsRuntimeException();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public List<Property> queryActivityProperty(String propertyName) {
|
|
Objects.requireNonNull(propertyName);
|
|
try {
|
|
final ParceledListSlice<Property> parceledList =
|
|
mPM.queryProperty(propertyName, TYPE_ACTIVITY);
|
|
if (parceledList == null) {
|
|
return Collections.emptyList();
|
|
}
|
|
return parceledList.getList();
|
|
} catch (RemoteException e) {
|
|
throw e.rethrowAsRuntimeException();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public List<Property> queryProviderProperty(String propertyName) {
|
|
Objects.requireNonNull(propertyName);
|
|
try {
|
|
final ParceledListSlice<Property> parceledList =
|
|
mPM.queryProperty(propertyName, TYPE_PROVIDER);
|
|
if (parceledList == null) {
|
|
return Collections.emptyList();
|
|
}
|
|
return parceledList.getList();
|
|
} catch (RemoteException e) {
|
|
throw e.rethrowAsRuntimeException();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public List<Property> queryReceiverProperty(String propertyName) {
|
|
Objects.requireNonNull(propertyName);
|
|
try {
|
|
final ParceledListSlice<Property> parceledList =
|
|
mPM.queryProperty(propertyName, TYPE_RECEIVER);
|
|
if (parceledList == null) {
|
|
return Collections.emptyList();
|
|
}
|
|
return parceledList.getList();
|
|
} catch (RemoteException e) {
|
|
throw e.rethrowAsRuntimeException();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public List<Property> queryServiceProperty(String propertyName) {
|
|
Objects.requireNonNull(propertyName);
|
|
try {
|
|
final ParceledListSlice<Property> 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> T parseAndroidManifest(@NonNull File apkFile,
|
|
@NonNull Function<XmlResourceParser, T> 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;
|
|
}
|
|
}
|
|
}
|