blob: 327d1b8beeb12f58f40150b7b6b4207f0d45d9a6 [file] [log] [blame]
/*
* Copyright (C) 2012 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.content.pm;
import static android.content.pm.PackageManager.COMPONENT_ENABLED_STATE_DEFAULT;
import static android.content.pm.PackageManager.COMPONENT_ENABLED_STATE_DISABLED;
import static android.content.pm.PackageManager.COMPONENT_ENABLED_STATE_DISABLED_UNTIL_USED;
import static android.content.pm.PackageManager.COMPONENT_ENABLED_STATE_DISABLED_USER;
import static android.content.pm.PackageManager.COMPONENT_ENABLED_STATE_ENABLED;
import static android.content.pm.PackageManager.MATCH_DIRECT_BOOT_AWARE;
import static android.content.pm.PackageManager.MATCH_DIRECT_BOOT_UNAWARE;
import static android.content.pm.PackageManager.MATCH_DISABLED_COMPONENTS;
import static android.content.pm.PackageManager.MATCH_DISABLED_UNTIL_USED_COMPONENTS;
import static android.content.pm.PackageManager.MATCH_SYSTEM_ONLY;
import android.annotation.NonNull;
import android.annotation.Nullable;
import android.compat.annotation.UnsupportedAppUsage;
import android.content.ComponentName;
import android.content.pm.parsing.component.ParsedMainComponent;
import android.os.BaseBundle;
import android.os.Debug;
import android.os.PersistableBundle;
import android.text.TextUtils;
import android.util.ArrayMap;
import android.util.ArraySet;
import android.util.DebugUtils;
import android.util.Pair;
import android.util.Slog;
import com.android.internal.annotations.VisibleForTesting;
import com.android.internal.util.ArrayUtils;
import com.android.internal.util.CollectionUtils;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;
import org.xmlpull.v1.XmlSerializer;
import java.io.IOException;
import java.util.Arrays;
import java.util.LinkedHashSet;
import java.util.Map;
import java.util.Objects;
/**
* Per-user state information about a package.
* @hide
*/
public class PackageUserState {
private static final boolean DEBUG = false;
private static final String LOG_TAG = "PackageUserState";
public long ceDataInode;
public boolean installed;
public boolean stopped;
public boolean notLaunched;
public boolean hidden; // Is the app restricted by owner / admin
public int distractionFlags;
public boolean suspended;
public ArrayMap<String, SuspendParams> suspendParams; // Suspending package to suspend params
public boolean instantApp;
public boolean virtualPreload;
public int enabled;
public String lastDisableAppCaller;
public int domainVerificationStatus;
public int appLinkGeneration;
public int categoryHint = ApplicationInfo.CATEGORY_UNDEFINED;
public int installReason;
public @PackageManager.UninstallReason int uninstallReason;
public String harmfulAppWarning;
public ArraySet<String> disabledComponents;
public ArraySet<String> enabledComponents;
private String[] overlayPaths;
private ArrayMap<String, String[]> sharedLibraryOverlayPaths; // Lib name to overlay paths
private String[] cachedOverlayPaths;
@Nullable
private ArrayMap<ComponentName, Pair<String, Integer>> componentLabelIconOverrideMap;
@UnsupportedAppUsage
public PackageUserState() {
installed = true;
hidden = false;
suspended = false;
enabled = COMPONENT_ENABLED_STATE_DEFAULT;
domainVerificationStatus =
PackageManager.INTENT_FILTER_DOMAIN_VERIFICATION_STATUS_UNDEFINED;
installReason = PackageManager.INSTALL_REASON_UNKNOWN;
uninstallReason = PackageManager.UNINSTALL_REASON_UNKNOWN;
}
@VisibleForTesting
public PackageUserState(PackageUserState o) {
ceDataInode = o.ceDataInode;
installed = o.installed;
stopped = o.stopped;
notLaunched = o.notLaunched;
hidden = o.hidden;
distractionFlags = o.distractionFlags;
suspended = o.suspended;
suspendParams = new ArrayMap<>(o.suspendParams);
instantApp = o.instantApp;
virtualPreload = o.virtualPreload;
enabled = o.enabled;
lastDisableAppCaller = o.lastDisableAppCaller;
domainVerificationStatus = o.domainVerificationStatus;
appLinkGeneration = o.appLinkGeneration;
categoryHint = o.categoryHint;
installReason = o.installReason;
uninstallReason = o.uninstallReason;
disabledComponents = ArrayUtils.cloneOrNull(o.disabledComponents);
enabledComponents = ArrayUtils.cloneOrNull(o.enabledComponents);
overlayPaths =
o.overlayPaths == null ? null : Arrays.copyOf(o.overlayPaths, o.overlayPaths.length);
if (o.sharedLibraryOverlayPaths != null) {
sharedLibraryOverlayPaths = new ArrayMap<>(o.sharedLibraryOverlayPaths);
}
harmfulAppWarning = o.harmfulAppWarning;
if (o.componentLabelIconOverrideMap != null) {
this.componentLabelIconOverrideMap = new ArrayMap<>(o.componentLabelIconOverrideMap);
}
}
public String[] getOverlayPaths() {
return overlayPaths;
}
public void setOverlayPaths(String[] paths) {
overlayPaths = paths;
cachedOverlayPaths = null;
}
public Map<String, String[]> getSharedLibraryOverlayPaths() {
return sharedLibraryOverlayPaths;
}
public void setSharedLibraryOverlayPaths(String library, String[] paths) {
if (sharedLibraryOverlayPaths == null) {
sharedLibraryOverlayPaths = new ArrayMap<>();
}
sharedLibraryOverlayPaths.put(library, paths);
cachedOverlayPaths = null;
}
/**
* Overrides the non-localized label and icon of a component.
*
* @return true if the label or icon was changed.
*/
@VisibleForTesting(visibility = VisibleForTesting.Visibility.PACKAGE)
public boolean overrideLabelAndIcon(@NonNull ComponentName component,
@Nullable String nonLocalizedLabel, @Nullable Integer icon) {
String existingLabel = null;
Integer existingIcon = null;
if (componentLabelIconOverrideMap != null) {
Pair<String, Integer> pair = componentLabelIconOverrideMap.get(component);
if (pair != null) {
existingLabel = pair.first;
existingIcon = pair.second;
}
}
boolean changed = !TextUtils.equals(existingLabel, nonLocalizedLabel)
|| !Objects.equals(existingIcon, icon);
if (changed) {
if (nonLocalizedLabel == null && icon == null) {
componentLabelIconOverrideMap.remove(component);
if (componentLabelIconOverrideMap.isEmpty()) {
componentLabelIconOverrideMap = null;
}
} else {
if (componentLabelIconOverrideMap == null) {
componentLabelIconOverrideMap = new ArrayMap<>(1);
}
componentLabelIconOverrideMap.put(component, Pair.create(nonLocalizedLabel, icon));
}
}
return changed;
}
/**
* Clears all values previously set by {@link #overrideLabelAndIcon(ComponentName,
* String, Integer)}.
*
* This is done when the package is updated as the components and resource IDs may have changed.
*/
public void resetOverrideComponentLabelIcon() {
componentLabelIconOverrideMap = null;
}
@Nullable
public Pair<String, Integer> getOverrideLabelIconForComponent(ComponentName componentName) {
if (ArrayUtils.isEmpty(componentLabelIconOverrideMap)) {
return null;
}
return componentLabelIconOverrideMap.get(componentName);
}
/**
* Test if this package is installed.
*/
public boolean isAvailable(int flags) {
// True if it is installed for this user and it is not hidden. If it is hidden,
// still return true if the caller requested MATCH_UNINSTALLED_PACKAGES
final boolean matchAnyUser = (flags & PackageManager.MATCH_ANY_USER) != 0;
final boolean matchUninstalled = (flags & PackageManager.MATCH_UNINSTALLED_PACKAGES) != 0;
return matchAnyUser
|| (this.installed
&& (!this.hidden || matchUninstalled));
}
public boolean isMatch(ComponentInfo componentInfo, int flags) {
return isMatch(componentInfo.applicationInfo.isSystemApp(),
componentInfo.applicationInfo.enabled, componentInfo.enabled,
componentInfo.directBootAware, componentInfo.name, flags);
}
public boolean isMatch(boolean isSystem, boolean isPackageEnabled,
ParsedMainComponent component, int flags) {
return isMatch(isSystem, isPackageEnabled, component.isEnabled(),
component.isDirectBootAware(), component.getName(), flags);
}
/**
* Test if the given component is considered installed, enabled and a match
* for the given flags.
*
* <p>
* Expects at least one of {@link PackageManager#MATCH_DIRECT_BOOT_AWARE} and
* {@link PackageManager#MATCH_DIRECT_BOOT_UNAWARE} are specified in {@code flags}.
* </p>
*
*/
public boolean isMatch(boolean isSystem, boolean isPackageEnabled, boolean isComponentEnabled,
boolean isComponentDirectBootAware, String componentName, int flags) {
final boolean matchUninstalled = (flags & PackageManager.MATCH_KNOWN_PACKAGES) != 0;
if (!isAvailable(flags) && !(isSystem && matchUninstalled)) {
return reportIfDebug(false, flags);
}
if (!isEnabled(isPackageEnabled, isComponentEnabled, componentName, flags)) {
return reportIfDebug(false, flags);
}
if ((flags & MATCH_SYSTEM_ONLY) != 0) {
if (!isSystem) {
return reportIfDebug(false, flags);
}
}
final boolean matchesUnaware = ((flags & MATCH_DIRECT_BOOT_UNAWARE) != 0)
&& !isComponentDirectBootAware;
final boolean matchesAware = ((flags & MATCH_DIRECT_BOOT_AWARE) != 0)
&& isComponentDirectBootAware;
return reportIfDebug(matchesUnaware || matchesAware, flags);
}
public boolean reportIfDebug(boolean result, int flags) {
if (DEBUG && !result) {
Slog.i(LOG_TAG, "No match!; flags: "
+ DebugUtils.flagsToString(PackageManager.class, "MATCH_", flags) + " "
+ Debug.getCaller());
}
return result;
}
public boolean isEnabled(ComponentInfo componentInfo, int flags) {
return isEnabled(componentInfo.applicationInfo.enabled, componentInfo.enabled,
componentInfo.name, flags);
}
public boolean isEnabled(boolean isPackageEnabled,
ParsedMainComponent parsedComponent, int flags) {
return isEnabled(isPackageEnabled, parsedComponent.isEnabled(), parsedComponent.getName(),
flags);
}
/**
* Test if the given component is considered enabled.
*/
public boolean isEnabled(boolean isPackageEnabled, boolean isComponentEnabled,
String componentName, int flags) {
if ((flags & MATCH_DISABLED_COMPONENTS) != 0) {
return true;
}
// First check if the overall package is disabled; if the package is
// enabled then fall through to check specific component
switch (this.enabled) {
case COMPONENT_ENABLED_STATE_DISABLED:
case COMPONENT_ENABLED_STATE_DISABLED_USER:
return false;
case COMPONENT_ENABLED_STATE_DISABLED_UNTIL_USED:
if ((flags & MATCH_DISABLED_UNTIL_USED_COMPONENTS) == 0) {
return false;
}
// fallthrough
case COMPONENT_ENABLED_STATE_DEFAULT:
if (!isPackageEnabled) {
return false;
}
// fallthrough
case COMPONENT_ENABLED_STATE_ENABLED:
break;
}
// Check if component has explicit state before falling through to
// the manifest default
if (ArrayUtils.contains(this.enabledComponents, componentName)) {
return true;
}
if (ArrayUtils.contains(this.disabledComponents, componentName)) {
return false;
}
return isComponentEnabled;
}
public String[] getAllOverlayPaths() {
if (overlayPaths == null && sharedLibraryOverlayPaths == null) {
return null;
}
if (cachedOverlayPaths != null) {
return cachedOverlayPaths;
}
final LinkedHashSet<String> paths = new LinkedHashSet<>();
if (overlayPaths != null) {
final int N = overlayPaths.length;
for (int i = 0; i < N; i++) {
paths.add(overlayPaths[i]);
}
}
if (sharedLibraryOverlayPaths != null) {
for (String[] libOverlayPaths : sharedLibraryOverlayPaths.values()) {
if (libOverlayPaths != null) {
final int N = libOverlayPaths.length;
for (int i = 0; i < N; i++) {
paths.add(libOverlayPaths[i]);
}
}
}
}
cachedOverlayPaths = paths.toArray(new String[0]);
return cachedOverlayPaths;
}
@Override
final public boolean equals(Object obj) {
if (!(obj instanceof PackageUserState)) {
return false;
}
final PackageUserState oldState = (PackageUserState) obj;
if (ceDataInode != oldState.ceDataInode) {
return false;
}
if (installed != oldState.installed) {
return false;
}
if (stopped != oldState.stopped) {
return false;
}
if (notLaunched != oldState.notLaunched) {
return false;
}
if (hidden != oldState.hidden) {
return false;
}
if (distractionFlags != oldState.distractionFlags) {
return false;
}
if (suspended != oldState.suspended) {
return false;
}
if (suspended) {
if (!Objects.equals(suspendParams, oldState.suspendParams)) {
return false;
}
}
if (instantApp != oldState.instantApp) {
return false;
}
if (virtualPreload != oldState.virtualPreload) {
return false;
}
if (enabled != oldState.enabled) {
return false;
}
if ((lastDisableAppCaller == null && oldState.lastDisableAppCaller != null)
|| (lastDisableAppCaller != null
&& !lastDisableAppCaller.equals(oldState.lastDisableAppCaller))) {
return false;
}
if (domainVerificationStatus != oldState.domainVerificationStatus) {
return false;
}
if (appLinkGeneration != oldState.appLinkGeneration) {
return false;
}
if (categoryHint != oldState.categoryHint) {
return false;
}
if (installReason != oldState.installReason) {
return false;
}
if (uninstallReason != oldState.uninstallReason) {
return false;
}
if ((disabledComponents == null && oldState.disabledComponents != null)
|| (disabledComponents != null && oldState.disabledComponents == null)) {
return false;
}
if (disabledComponents != null) {
if (disabledComponents.size() != oldState.disabledComponents.size()) {
return false;
}
for (int i = disabledComponents.size() - 1; i >=0; --i) {
if (!oldState.disabledComponents.contains(disabledComponents.valueAt(i))) {
return false;
}
}
}
if ((enabledComponents == null && oldState.enabledComponents != null)
|| (enabledComponents != null && oldState.enabledComponents == null)) {
return false;
}
if (enabledComponents != null) {
if (enabledComponents.size() != oldState.enabledComponents.size()) {
return false;
}
for (int i = enabledComponents.size() - 1; i >=0; --i) {
if (!oldState.enabledComponents.contains(enabledComponents.valueAt(i))) {
return false;
}
}
}
if (harmfulAppWarning == null && oldState.harmfulAppWarning != null
|| (harmfulAppWarning != null
&& !harmfulAppWarning.equals(oldState.harmfulAppWarning))) {
return false;
}
return true;
}
@Override
public int hashCode() {
int hashCode = Long.hashCode(ceDataInode);
hashCode = 31 * hashCode + Boolean.hashCode(installed);
hashCode = 31 * hashCode + Boolean.hashCode(stopped);
hashCode = 31 * hashCode + Boolean.hashCode(notLaunched);
hashCode = 31 * hashCode + Boolean.hashCode(hidden);
hashCode = 31 * hashCode + distractionFlags;
hashCode = 31 * hashCode + Boolean.hashCode(suspended);
hashCode = 31 * hashCode + Objects.hashCode(suspendParams);
hashCode = 31 * hashCode + Boolean.hashCode(instantApp);
hashCode = 31 * hashCode + Boolean.hashCode(virtualPreload);
hashCode = 31 * hashCode + enabled;
hashCode = 31 * hashCode + Objects.hashCode(lastDisableAppCaller);
hashCode = 31 * hashCode + domainVerificationStatus;
hashCode = 31 * hashCode + appLinkGeneration;
hashCode = 31 * hashCode + categoryHint;
hashCode = 31 * hashCode + installReason;
hashCode = 31 * hashCode + uninstallReason;
hashCode = 31 * hashCode + Objects.hashCode(disabledComponents);
hashCode = 31 * hashCode + Objects.hashCode(enabledComponents);
hashCode = 31 * hashCode + Objects.hashCode(harmfulAppWarning);
return hashCode;
}
/**
* Container to describe suspension parameters.
*/
public static final class SuspendParams {
private static final String TAG_DIALOG_INFO = "dialog-info";
private static final String TAG_APP_EXTRAS = "app-extras";
private static final String TAG_LAUNCHER_EXTRAS = "launcher-extras";
public SuspendDialogInfo dialogInfo;
public PersistableBundle appExtras;
public PersistableBundle launcherExtras;
private SuspendParams() {
}
/**
* Returns a {@link SuspendParams} object with the given fields. Returns {@code null} if all
* the fields are {@code null}.
*
* @param dialogInfo
* @param appExtras
* @param launcherExtras
* @return A {@link SuspendParams} object or {@code null}.
*/
public static SuspendParams getInstanceOrNull(SuspendDialogInfo dialogInfo,
PersistableBundle appExtras, PersistableBundle launcherExtras) {
if (dialogInfo == null && appExtras == null && launcherExtras == null) {
return null;
}
final SuspendParams instance = new SuspendParams();
instance.dialogInfo = dialogInfo;
instance.appExtras = appExtras;
instance.launcherExtras = launcherExtras;
return instance;
}
@Override
public boolean equals(Object obj) {
if (this == obj) {
return true;
}
if (!(obj instanceof SuspendParams)) {
return false;
}
final SuspendParams other = (SuspendParams) obj;
if (!Objects.equals(dialogInfo, other.dialogInfo)) {
return false;
}
if (!BaseBundle.kindofEquals(appExtras, other.appExtras)) {
return false;
}
if (!BaseBundle.kindofEquals(launcherExtras, other.launcherExtras)) {
return false;
}
return true;
}
@Override
public int hashCode() {
int hashCode = Objects.hashCode(dialogInfo);
hashCode = 31 * hashCode + ((appExtras != null) ? appExtras.size() : 0);
hashCode = 31 * hashCode + ((launcherExtras != null) ? launcherExtras.size() : 0);
return hashCode;
}
/**
* Serializes this object into an xml format
* @param out the {@link XmlSerializer} object
* @throws IOException
*/
public void saveToXml(XmlSerializer out) throws IOException {
if (dialogInfo != null) {
out.startTag(null, TAG_DIALOG_INFO);
dialogInfo.saveToXml(out);
out.endTag(null, TAG_DIALOG_INFO);
}
if (appExtras != null) {
out.startTag(null, TAG_APP_EXTRAS);
try {
appExtras.saveToXml(out);
} catch (XmlPullParserException e) {
Slog.e(LOG_TAG, "Exception while trying to write appExtras."
+ " Will be lost on reboot", e);
}
out.endTag(null, TAG_APP_EXTRAS);
}
if (launcherExtras != null) {
out.startTag(null, TAG_LAUNCHER_EXTRAS);
try {
launcherExtras.saveToXml(out);
} catch (XmlPullParserException e) {
Slog.e(LOG_TAG, "Exception while trying to write launcherExtras."
+ " Will be lost on reboot", e);
}
out.endTag(null, TAG_LAUNCHER_EXTRAS);
}
}
/**
* Parses this object from the xml format. Returns {@code null} if no object related
* information could be read.
* @param in the reader
* @return
*/
public static SuspendParams restoreFromXml(XmlPullParser in) throws IOException {
SuspendDialogInfo readDialogInfo = null;
PersistableBundle readAppExtras = null;
PersistableBundle readLauncherExtras = null;
final int currentDepth = in.getDepth();
int type;
try {
while ((type = in.next()) != XmlPullParser.END_DOCUMENT
&& (type != XmlPullParser.END_TAG
|| in.getDepth() > currentDepth)) {
if (type == XmlPullParser.END_TAG
|| type == XmlPullParser.TEXT) {
continue;
}
switch (in.getName()) {
case TAG_DIALOG_INFO:
readDialogInfo = SuspendDialogInfo.restoreFromXml(in);
break;
case TAG_APP_EXTRAS:
readAppExtras = PersistableBundle.restoreFromXml(in);
break;
case TAG_LAUNCHER_EXTRAS:
readLauncherExtras = PersistableBundle.restoreFromXml(in);
break;
default:
Slog.w(LOG_TAG, "Unknown tag " + in.getName()
+ " in SuspendParams. Ignoring");
break;
}
}
} catch (XmlPullParserException e) {
Slog.e(LOG_TAG, "Exception while trying to parse SuspendParams,"
+ " some fields may default", e);
}
return getInstanceOrNull(readDialogInfo, readAppExtras, readLauncherExtras);
}
}
}