1/*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package com.android.server.pm;
18
19import android.content.pm.ApplicationInfo;
20import android.content.pm.PackageManager;
21import android.content.pm.PackageParser;
22import android.content.pm.UserInfo;
23import android.service.pm.PackageProto;
24import android.util.proto.ProtoOutputStream;
25
26import com.android.server.pm.permission.PermissionsState;
27
28import java.io.File;
29import java.util.List;
30
31/**
32 * Settings data for a particular package we know about.
33 */
34public final class PackageSetting extends PackageSettingBase {
35    int appId;
36    PackageParser.Package pkg;
37    /**
38     * WARNING. The object reference is important. We perform integer equality and NOT
39     * object equality to check whether shared user settings are the same.
40     */
41    SharedUserSetting sharedUser;
42    /**
43     * Temporary holding space for the shared user ID. While parsing package settings, the
44     * shared users tag may come after the packages. In this case, we must delay linking the
45     * shared user setting with the package setting. The shared user ID lets us link the
46     * two objects.
47     */
48    private int sharedUserId;
49
50    PackageSetting(String name, String realName, File codePath, File resourcePath,
51            String legacyNativeLibraryPathString, String primaryCpuAbiString,
52            String secondaryCpuAbiString, String cpuAbiOverrideString,
53            long pVersionCode, int pkgFlags, int privateFlags, String parentPackageName,
54            List<String> childPackageNames, int sharedUserId, String[] usesStaticLibraries,
55            long[] usesStaticLibrariesVersions) {
56        super(name, realName, codePath, resourcePath, legacyNativeLibraryPathString,
57                primaryCpuAbiString, secondaryCpuAbiString, cpuAbiOverrideString,
58                pVersionCode, pkgFlags, privateFlags, parentPackageName, childPackageNames,
59                usesStaticLibraries, usesStaticLibrariesVersions);
60        this.sharedUserId = sharedUserId;
61    }
62
63    /**
64     * New instance of PackageSetting replicating the original settings.
65     * Note that it keeps the same PackageParser.Package instance.
66     */
67    PackageSetting(PackageSetting orig) {
68        super(orig, orig.realName);
69        doCopy(orig);
70    }
71
72    /**
73     * New instance of PackageSetting replicating the original settings, but, allows specifying
74     * a real package name.
75     * Note that it keeps the same PackageParser.Package instance.
76     */
77    PackageSetting(PackageSetting orig, String realPkgName) {
78        super(orig, realPkgName);
79        doCopy(orig);
80    }
81
82    public int getSharedUserId() {
83        if (sharedUser != null) {
84            return sharedUser.userId;
85        }
86        return sharedUserId;
87    }
88
89    public SharedUserSetting getSharedUser() {
90        return sharedUser;
91    }
92
93    @Override
94    public String toString() {
95        return "PackageSetting{"
96            + Integer.toHexString(System.identityHashCode(this))
97            + " " + name + "/" + appId + "}";
98    }
99
100    public void copyFrom(PackageSetting orig) {
101        super.copyFrom(orig);
102        doCopy(orig);
103    }
104
105    private void doCopy(PackageSetting orig) {
106        appId = orig.appId;
107        pkg = orig.pkg;
108        sharedUser = orig.sharedUser;
109        sharedUserId = orig.sharedUserId;
110    }
111
112    @Override
113    public PermissionsState getPermissionsState() {
114        return (sharedUser != null)
115                ? sharedUser.getPermissionsState()
116                : super.getPermissionsState();
117    }
118
119    public PackageParser.Package getPackage() {
120        return pkg;
121    }
122
123    public int getAppId() {
124        return appId;
125    }
126
127    public void setInstallPermissionsFixed(boolean fixed) {
128        installPermissionsFixed = fixed;
129    }
130
131    public boolean areInstallPermissionsFixed() {
132        return installPermissionsFixed;
133    }
134
135    public boolean isPrivileged() {
136        return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_PRIVILEGED) != 0;
137    }
138
139    public boolean isOem() {
140        return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_OEM) != 0;
141    }
142
143    public boolean isVendor() {
144        return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_VENDOR) != 0;
145    }
146
147    public boolean isProduct() {
148        return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_PRODUCT) != 0;
149    }
150
151    public boolean isForwardLocked() {
152        return (pkgPrivateFlags & ApplicationInfo.PRIVATE_FLAG_FORWARD_LOCK) != 0;
153    }
154
155    public boolean isSystem() {
156        return (pkgFlags & ApplicationInfo.FLAG_SYSTEM) != 0;
157    }
158
159    public boolean isUpdatedSystem() {
160        return (pkgFlags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0;
161    }
162
163    @Override
164    public boolean isSharedUser() {
165        return sharedUser != null;
166    }
167
168    public boolean isMatch(int flags) {
169        if ((flags & PackageManager.MATCH_SYSTEM_ONLY) != 0) {
170            return isSystem();
171        }
172        return true;
173    }
174
175    public boolean hasChildPackages() {
176        return childPackageNames != null && !childPackageNames.isEmpty();
177    }
178
179    public void writeToProto(ProtoOutputStream proto, long fieldId, List<UserInfo> users) {
180        final long packageToken = proto.start(fieldId);
181        proto.write(PackageProto.NAME, (realName != null ? realName : name));
182        proto.write(PackageProto.UID, appId);
183        proto.write(PackageProto.VERSION_CODE, versionCode);
184        proto.write(PackageProto.VERSION_STRING, pkg.mVersionName);
185        proto.write(PackageProto.INSTALL_TIME_MS, firstInstallTime);
186        proto.write(PackageProto.UPDATE_TIME_MS, lastUpdateTime);
187        proto.write(PackageProto.INSTALLER_NAME, installerPackageName);
188
189        if (pkg != null) {
190            long splitToken = proto.start(PackageProto.SPLITS);
191            proto.write(PackageProto.SplitProto.NAME, "base");
192            proto.write(PackageProto.SplitProto.REVISION_CODE, pkg.baseRevisionCode);
193            proto.end(splitToken);
194            if (pkg.splitNames != null) {
195                for (int i = 0; i < pkg.splitNames.length; i++) {
196                    splitToken = proto.start(PackageProto.SPLITS);
197                    proto.write(PackageProto.SplitProto.NAME, pkg.splitNames[i]);
198                    proto.write(PackageProto.SplitProto.REVISION_CODE, pkg.splitRevisionCodes[i]);
199                    proto.end(splitToken);
200                }
201            }
202        }
203        writeUsersInfoToProto(proto, PackageProto.USERS);
204        proto.end(packageToken);
205    }
206}
207