blob: 376326264d9c37b92e4e7fe041d4504fa1fa2dcc [file] [log] [blame]
/*
* Copyright (C) 2011 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 com.android.server.pm;
import android.annotation.NonNull;
import android.annotation.Nullable;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.content.pm.UserInfo;
import android.service.pm.PackageProto;
import android.util.ArrayMap;
import android.util.ArraySet;
import android.util.proto.ProtoOutputStream;
import com.android.internal.annotations.VisibleForTesting;
import com.android.server.pm.parsing.pkg.AndroidPackage;
import com.android.server.pm.permission.LegacyPermissionDataProvider;
import com.android.server.pm.permission.LegacyPermissionState;
import com.android.server.pm.pkg.PackageStateUnserialized;
import com.android.server.utils.SnapshotCache;
import java.io.File;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.UUID;
/**
* Settings data for a particular package we know about.
*/
public class PackageSetting extends PackageSettingBase {
@VisibleForTesting(visibility = VisibleForTesting.Visibility.PACKAGE)
public int appId;
/**
* This can be null whenever a physical APK on device is missing. This can be the result of
* removing an external storage device where the APK resides.
*
* This will result in the system reading the {@link PackageSetting} from disk, but without
* being able to parse the base APK's AndroidManifest.xml to read all of its metadata. The data
* that is written and read in {@link Settings} includes a minimal set of metadata needed to
* perform other checks in the system.
*
* This is important in order to enforce uniqueness within the system, as the package, even if
* on a removed storage device, is still considered installed. Another package of the same
* application ID or declaring the same permissions or similar cannot be installed.
*
* Re-attaching the storage device to make the APK available should allow the user to use the
* app once the device reboots or otherwise re-scans it.
*
* It is expected that all code that uses a {@link PackageSetting} understands this inner field
* may be null. Note that this relationship only works one way. It should not be possible to
* have an entry inside {@link PackageManagerService#mPackages} without a corresponding
* {@link PackageSetting} inside {@link Settings#mPackages}.
*
* @deprecated Use {@link #getPkg()}. The setter is favored to avoid unintended mutation.
*/
@Nullable
@Deprecated
public AndroidPackage pkg;
/**
* WARNING. The object reference is important. We perform integer equality and NOT
* object equality to check whether shared user settings are the same.
*/
SharedUserSetting sharedUser;
/**
* Temporary holding space for the shared user ID. While parsing package settings, the
* shared users tag may come after the packages. In this case, we must delay linking the
* shared user setting with the package setting. The shared user ID lets us link the
* two objects.
*/
private int sharedUserId;
/**
* Maps mime group name to the set of Mime types in a group. Mime groups declared
* by app are populated with empty sets at construction.
* Mime groups can not be created/removed at runtime, thus keys in this map should not change
*/
@Nullable
Map<String, ArraySet<String>> mimeGroups;
@NonNull
private PackageStateUnserialized pkgState = new PackageStateUnserialized();
@NonNull
private UUID mDomainSetId;
/**
* Snapshot support.
*/
private final SnapshotCache<PackageSetting> mSnapshot;
private SnapshotCache<PackageSetting> makeCache() {
return new SnapshotCache<PackageSetting>(this, this) {
@Override
public PackageSetting createSnapshot() {
return new PackageSetting(mSource, true);
}};
}
@VisibleForTesting(visibility = VisibleForTesting.Visibility.PACKAGE)
public PackageSetting(String name, String realName, @NonNull File codePath,
String legacyNativeLibraryPathString, String primaryCpuAbiString,
String secondaryCpuAbiString, String cpuAbiOverrideString,
long pVersionCode, int pkgFlags, int privateFlags,
int sharedUserId, String[] usesStaticLibraries,
long[] usesStaticLibrariesVersions, Map<String, ArraySet<String>> mimeGroups,
@NonNull UUID domainSetId) {
super(name, realName, codePath, legacyNativeLibraryPathString,
primaryCpuAbiString, secondaryCpuAbiString, cpuAbiOverrideString,
pVersionCode, pkgFlags, privateFlags,
usesStaticLibraries, usesStaticLibrariesVersions);
this.sharedUserId = sharedUserId;
mDomainSetId = domainSetId;
copyMimeGroups(mimeGroups);
mSnapshot = makeCache();
}
/**
* New instance of PackageSetting replicating the original settings.
* Note that it keeps the same PackageParser.Package instance.
*/
PackageSetting(PackageSetting orig) {
super(orig, orig.realName);
doCopy(orig);
mSnapshot = makeCache();
}
/**
* New instance of PackageSetting replicating the original settings, but, allows specifying
* a real package name.
* Note that it keeps the same PackageParser.Package instance.
*/
PackageSetting(PackageSetting orig, String realPkgName) {
super(orig, realPkgName);
doCopy(orig);
mSnapshot = makeCache();
}
/**
* Create a snapshot. The copy constructor is already in use and cannot be modified
* for this purpose.
*/
PackageSetting(PackageSetting orig, boolean snapshot) {
super(orig, snapshot);
// The existing doCopy() method cannot be used in here because sharedUser must be
// a snapshot, and not a reference. Also, the pkgState must be copied. However,
// this code should otherwise be kept in sync with doCopy().
appId = orig.appId;
pkg = orig.pkg;
sharedUser = orig.sharedUser == null ? null : orig.sharedUser.snapshot();
sharedUserId = orig.sharedUserId;
copyMimeGroups(orig.mimeGroups);
pkgState = orig.pkgState;
mDomainSetId = orig.getDomainSetId();
mSnapshot = new SnapshotCache.Sealed();
}
/**
* Return the package snapshot.
*/
public PackageSetting snapshot() {
return mSnapshot.snapshot();
}
/** @see #pkg **/
@Nullable
public AndroidPackage getPkg() {
return pkg;
}
public int getSharedUserId() {
if (sharedUser != null) {
return sharedUser.userId;
}
return sharedUserId;
}
public SharedUserSetting getSharedUser() {
return sharedUser;
}
@Override
public String toString() {
return "PackageSetting{"
+ Integer.toHexString(System.identityHashCode(this))
+ " " + name + "/" + appId + "}";
}
public void copyFrom(PackageSetting orig) {
super.copyFrom(orig);
doCopy(orig);
}
private void doCopy(PackageSetting orig) {
appId = orig.appId;
pkg = orig.pkg;
sharedUser = orig.sharedUser;
sharedUserId = orig.sharedUserId;
copyMimeGroups(orig.mimeGroups);
mDomainSetId = orig.getDomainSetId();
}
private void copyMimeGroups(@Nullable Map<String, ArraySet<String>> newMimeGroups) {
if (newMimeGroups == null) {
mimeGroups = null;
return;
}
mimeGroups = new ArrayMap<>(newMimeGroups.size());
for (String mimeGroup : newMimeGroups.keySet()) {
ArraySet<String> mimeTypes = newMimeGroups.get(mimeGroup);
if (mimeTypes != null) {
mimeGroups.put(mimeGroup, new ArraySet<>(mimeTypes));
} else {
mimeGroups.put(mimeGroup, new ArraySet<>());
}
}
}
/**
* Updates declared MIME groups, removing no longer declared groups
* and keeping previous state of MIME groups
*/
void updateMimeGroups(@Nullable Set<String> newMimeGroupNames) {
if (newMimeGroupNames == null) {
mimeGroups = null;
return;
}
if (mimeGroups == null) {
// set mimeGroups to empty map to avoid repeated null-checks in the next loop
mimeGroups = Collections.emptyMap();
}
ArrayMap<String, ArraySet<String>> updatedMimeGroups =
new ArrayMap<>(newMimeGroupNames.size());
for (String mimeGroup : newMimeGroupNames) {
if (mimeGroups.containsKey(mimeGroup)) {
updatedMimeGroups.put(mimeGroup, mimeGroups.get(mimeGroup));
} else {
updatedMimeGroups.put(mimeGroup, new ArraySet<>());
}
}
mimeGroups = updatedMimeGroups;
}
@Deprecated
@Override
public LegacyPermissionState getLegacyPermissionState() {
return (sharedUser != null)
? sharedUser.getLegacyPermissionState()
: super.getLegacyPermissionState();
}
public int getAppId() {
return appId;
}
public void setInstallPermissionsFixed(boolean fixed) {
installPermissionsFixed = fixed;
}
public boolean areInstallPermissionsFixed() {
return installPermissionsFixed;
}
public boolean isPrivileged() {
return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_PRIVILEGED) != 0;
}
public boolean isOem() {
return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_OEM) != 0;
}
public boolean isVendor() {
return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_VENDOR) != 0;
}
public boolean isProduct() {
return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_PRODUCT) != 0;
}
public boolean isSystemExt() {
return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_SYSTEM_EXT) != 0;
}
public boolean isOdm() {
return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_ODM) != 0;
}
public boolean isSystem() {
return (pkgFlags & ApplicationInfo.FLAG_SYSTEM) != 0;
}
@Override
public boolean isSharedUser() {
return sharedUser != null;
}
public boolean isMatch(int flags) {
if ((flags & PackageManager.MATCH_SYSTEM_ONLY) != 0) {
return isSystem();
}
return true;
}
public boolean setMimeGroup(String mimeGroup, List<String> mimeTypes) {
ArraySet<String> oldMimeTypes = getMimeGroupInternal(mimeGroup);
if (oldMimeTypes == null) {
throw new IllegalArgumentException("Unknown MIME group " + mimeGroup
+ " for package " + name);
}
ArraySet<String> newMimeTypes = new ArraySet<>(mimeTypes);
boolean hasChanges = !newMimeTypes.equals(oldMimeTypes);
mimeGroups.put(mimeGroup, newMimeTypes);
return hasChanges;
}
public List<String> getMimeGroup(String mimeGroup) {
ArraySet<String> mimeTypes = getMimeGroupInternal(mimeGroup);
if (mimeTypes == null) {
throw new IllegalArgumentException("Unknown MIME group " + mimeGroup
+ " for package " + name);
}
return new ArrayList<>(mimeTypes);
}
private ArraySet<String> getMimeGroupInternal(String mimeGroup) {
return mimeGroups != null ? mimeGroups.get(mimeGroup) : null;
}
public void dumpDebug(ProtoOutputStream proto, long fieldId, List<UserInfo> users,
LegacyPermissionDataProvider dataProvider) {
final long packageToken = proto.start(fieldId);
proto.write(PackageProto.NAME, (realName != null ? realName : name));
proto.write(PackageProto.UID, appId);
proto.write(PackageProto.VERSION_CODE, versionCode);
proto.write(PackageProto.INSTALL_TIME_MS, firstInstallTime);
proto.write(PackageProto.UPDATE_TIME_MS, lastUpdateTime);
proto.write(PackageProto.INSTALLER_NAME, installSource.installerPackageName);
if (pkg != null) {
proto.write(PackageProto.VERSION_STRING, pkg.getVersionName());
long splitToken = proto.start(PackageProto.SPLITS);
proto.write(PackageProto.SplitProto.NAME, "base");
proto.write(PackageProto.SplitProto.REVISION_CODE, pkg.getBaseRevisionCode());
proto.end(splitToken);
if (pkg.getSplitNames() != null) {
for (int i = 0; i < pkg.getSplitNames().length; i++) {
splitToken = proto.start(PackageProto.SPLITS);
proto.write(PackageProto.SplitProto.NAME, pkg.getSplitNames()[i]);
proto.write(PackageProto.SplitProto.REVISION_CODE,
pkg.getSplitRevisionCodes()[i]);
proto.end(splitToken);
}
}
long sourceToken = proto.start(PackageProto.INSTALL_SOURCE);
proto.write(PackageProto.InstallSourceProto.INITIATING_PACKAGE_NAME,
installSource.initiatingPackageName);
proto.write(PackageProto.InstallSourceProto.ORIGINATING_PACKAGE_NAME,
installSource.originatingPackageName);
proto.end(sourceToken);
}
proto.write(PackageProto.StatesProto.IS_LOADING, isPackageLoading());
writeUsersInfoToProto(proto, PackageProto.USERS);
writePackageUserPermissionsProto(proto, PackageProto.USER_PERMISSIONS, users, dataProvider);
proto.end(packageToken);
}
/**
* TODO (b/170263003) refactor to dump to permissiongr proto
* Dumps the permissions that are granted to users for this package.
*/
void writePackageUserPermissionsProto(ProtoOutputStream proto, long fieldId,
List<UserInfo> users, LegacyPermissionDataProvider dataProvider) {
Collection<LegacyPermissionState.PermissionState> runtimePermissionStates;
for (UserInfo user : users) {
final long permissionsToken = proto.start(PackageProto.USER_PERMISSIONS);
proto.write(PackageProto.UserPermissionsProto.ID, user.id);
runtimePermissionStates = dataProvider.getLegacyPermissionState(appId)
.getPermissionStates(user.id);
for (LegacyPermissionState.PermissionState permission : runtimePermissionStates) {
if (permission.isGranted()) {
proto.write(PackageProto.UserPermissionsProto.GRANTED_PERMISSIONS,
permission.getName());
}
}
proto.end(permissionsToken);
}
}
/** Updates all fields in the current setting from another. */
public void updateFrom(PackageSetting other) {
super.updateFrom(other);
appId = other.appId;
pkg = other.pkg;
sharedUserId = other.sharedUserId;
sharedUser = other.sharedUser;
mDomainSetId = other.mDomainSetId;
Set<String> mimeGroupNames = other.mimeGroups != null ? other.mimeGroups.keySet() : null;
updateMimeGroups(mimeGroupNames);
getPkgState().updateFrom(other.getPkgState());
}
@NonNull
public PackageStateUnserialized getPkgState() {
return pkgState;
}
@NonNull
public UUID getDomainSetId() {
return mDomainSetId;
}
public PackageSetting setDomainSetId(@NonNull UUID domainSetId) {
mDomainSetId = domainSetId;
return this;
}
}