1 /* 2 * Copyright (C) 2020 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 17 package com.android.server.pm; 18 19 import static android.content.pm.PackageManager.TYPE_ACTIVITY; 20 import static android.content.pm.PackageManager.TYPE_APPLICATION; 21 import static android.content.pm.PackageManager.TYPE_PROVIDER; 22 import static android.content.pm.PackageManager.TYPE_RECEIVER; 23 import static android.content.pm.PackageManager.TYPE_SERVICE; 24 25 import android.annotation.NonNull; 26 import android.annotation.Nullable; 27 import android.content.pm.PackageManager; 28 import android.content.pm.PackageManager.Property; 29 import android.content.pm.PackageManager.PropertyLocation; 30 import android.os.Binder; 31 import android.os.UserHandle; 32 import android.util.ArrayMap; 33 34 import com.android.internal.pm.pkg.component.ParsedComponent; 35 import com.android.server.pm.pkg.AndroidPackage; 36 37 import java.util.ArrayList; 38 import java.util.Iterator; 39 import java.util.List; 40 import java.util.Map; 41 import java.util.Objects; 42 import java.util.function.Predicate; 43 44 /** 45 * Manages properties defined within a package using the <property> tag. 46 */ 47 public class PackageProperty { 48 /** 49 * Mapping of property name to all defined defined properties. 50 * <p>This is a mapping of property name --> package map. The package 51 * map is a mapping of package name -> list of properties. 52 */ 53 private ArrayMap<String, ArrayMap<String, ArrayList<Property>>> mApplicationProperties; 54 private ArrayMap<String, ArrayMap<String, ArrayList<Property>>> mActivityProperties; 55 private ArrayMap<String, ArrayMap<String, ArrayList<Property>>> mProviderProperties; 56 private ArrayMap<String, ArrayMap<String, ArrayList<Property>>> mReceiverProperties; 57 private ArrayMap<String, ArrayMap<String, ArrayList<Property>>> mServiceProperties; 58 59 /** 60 * If the provided component is {@code null}, returns the property defined on the 61 * application. Otherwise, returns the property defined on the component. 62 */ getProperty(@onNull String propertyName, @NonNull String packageName, @Nullable String className)63 public Property getProperty(@NonNull String propertyName, @NonNull String packageName, 64 @Nullable String className) { 65 if (className == null) { 66 return getApplicationProperty(propertyName, packageName); 67 } 68 return getComponentProperty(propertyName, packageName, className); 69 } 70 71 /** 72 * Returns all properties defined at the given location. 73 * <p>Valid locations are {@link PackageManager#TYPE_APPLICATION}, 74 * {@link PackageManager#TYPE_ACTIVITY}, {@link PackageManager#TYPE_PROVIDER}, 75 * {@link PackageManager#TYPE_RECEIVER}, or {@link PackageManager#TYPE_SERVICE}. 76 */ queryProperty(@onNull String propertyName, @PropertyLocation int componentType, Predicate<String> filter)77 public List<Property> queryProperty(@NonNull String propertyName, 78 @PropertyLocation int componentType, Predicate<String> filter) { 79 final ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyMap; 80 if (componentType == TYPE_APPLICATION) { 81 propertyMap = mApplicationProperties; 82 } else if (componentType == TYPE_ACTIVITY) { 83 propertyMap = mActivityProperties; 84 } else if (componentType == TYPE_PROVIDER) { 85 propertyMap = mProviderProperties; 86 } else if (componentType == TYPE_RECEIVER) { 87 propertyMap = mReceiverProperties; 88 } else if (componentType == TYPE_SERVICE) { 89 propertyMap = mServiceProperties; 90 } else { 91 propertyMap = null; 92 } 93 if (propertyMap == null) { 94 return null; 95 } 96 final ArrayMap<String, ArrayList<Property>> packagePropertyMap = 97 propertyMap.get(propertyName); 98 if (packagePropertyMap == null) { 99 return null; 100 } 101 final int callingUid = Binder.getCallingUid(); 102 final int callingUserId = UserHandle.getCallingUserId(); 103 final int mapSize = packagePropertyMap.size(); 104 final List<Property> result = new ArrayList<>(mapSize); 105 for (int i = 0; i < mapSize; i++) { 106 final String packageName = packagePropertyMap.keyAt(i); 107 if (filter.test(packageName)) { 108 continue; 109 } 110 result.addAll(packagePropertyMap.valueAt(i)); 111 } 112 return result; 113 } 114 115 /** Adds all properties defined for the given package */ addAllProperties(AndroidPackage pkg)116 void addAllProperties(AndroidPackage pkg) { 117 mApplicationProperties = addProperties(pkg.getProperties(), mApplicationProperties); 118 mActivityProperties = addComponentProperties(pkg.getActivities(), mActivityProperties); 119 mProviderProperties = addComponentProperties(pkg.getProviders(), mProviderProperties); 120 mReceiverProperties = addComponentProperties(pkg.getReceivers(), mReceiverProperties); 121 mServiceProperties = addComponentProperties(pkg.getServices(), mServiceProperties); 122 } 123 124 /** Adds all properties defined for the given package */ removeAllProperties(AndroidPackage pkg)125 void removeAllProperties(AndroidPackage pkg) { 126 mApplicationProperties = removeProperties(pkg.getProperties(), mApplicationProperties); 127 mActivityProperties = removeComponentProperties(pkg.getActivities(), mActivityProperties); 128 mProviderProperties = removeComponentProperties(pkg.getProviders(), mProviderProperties); 129 mReceiverProperties = removeComponentProperties(pkg.getReceivers(), mReceiverProperties); 130 mServiceProperties = removeComponentProperties(pkg.getServices(), mServiceProperties); 131 } 132 133 /** Add the properties defined on the given components to the property collection */ 134 private static <T extends ParsedComponent> addComponentProperties( @onNull List<T> components, @Nullable ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyCollection)135 ArrayMap<String, ArrayMap<String, ArrayList<Property>>> addComponentProperties( 136 @NonNull List<T> components, 137 @Nullable ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyCollection) { 138 ArrayMap<String, ArrayMap<String, ArrayList<Property>>> returnCollection = 139 propertyCollection; 140 final int componentsSize = components.size(); 141 for (int i = 0; i < componentsSize; i++) { 142 final Map<String, Property> properties = components.get(i).getProperties(); 143 if (properties.size() == 0) { 144 continue; 145 } 146 returnCollection = addProperties(properties, returnCollection); 147 } 148 return returnCollection; 149 } 150 151 /** Add the given properties to the property collection */ addProperties( @onNull Map<String, Property> properties, @Nullable ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyCollection)152 private static ArrayMap<String, ArrayMap<String, ArrayList<Property>>> addProperties( 153 @NonNull Map<String, Property> properties, 154 @Nullable ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyCollection) { 155 if (properties.size() == 0) { 156 return propertyCollection; 157 } 158 final ArrayMap<String, ArrayMap<String, ArrayList<Property>>> returnCollection = 159 propertyCollection == null ? new ArrayMap<>(10) : propertyCollection; 160 final Iterator<Property> iter = properties.values().iterator(); 161 while (iter.hasNext()) { 162 final Property property = iter.next(); 163 final String propertyName = property.getName(); 164 final String packageName = property.getPackageName(); 165 ArrayMap<String, ArrayList<Property>> propertyMap = returnCollection.get(propertyName); 166 if (propertyMap == null) { 167 propertyMap = new ArrayMap<>(); 168 returnCollection.put(propertyName, propertyMap); 169 } 170 ArrayList<Property> packageProperties = propertyMap.get(packageName); 171 if (packageProperties == null) { 172 packageProperties = new ArrayList<>(properties.size()); 173 propertyMap.put(packageName, packageProperties); 174 } 175 packageProperties.add(property); 176 } 177 return returnCollection; 178 } 179 180 /** Removes the properties defined on the given components from the property collection */ 181 private static <T extends ParsedComponent> removeComponentProperties( @onNull List<T> components, @Nullable ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyCollection)182 ArrayMap<String, ArrayMap<String, ArrayList<Property>>> removeComponentProperties( 183 @NonNull List<T> components, 184 @Nullable ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyCollection) { 185 ArrayMap<String, ArrayMap<String, ArrayList<Property>>> returnCollection = 186 propertyCollection; 187 final int componentsSize = components.size(); 188 for (int i = 0; returnCollection != null && i < componentsSize; i++) { 189 final Map<String, Property> properties = components.get(i).getProperties(); 190 if (properties.size() == 0) { 191 continue; 192 } 193 returnCollection = removeProperties(properties, returnCollection); 194 } 195 return returnCollection; 196 } 197 198 /** Removes the given properties from the property collection */ removeProperties( @onNull Map<String, Property> properties, @Nullable ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyCollection)199 private static ArrayMap<String, ArrayMap<String, ArrayList<Property>>> removeProperties( 200 @NonNull Map<String, Property> properties, 201 @Nullable ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyCollection) { 202 if (propertyCollection == null) { 203 return null; 204 } 205 final Iterator<Property> iter = properties.values().iterator(); 206 while (iter.hasNext()) { 207 final Property property = iter.next(); 208 final String propertyName = property.getName(); 209 final String packageName = property.getPackageName(); 210 ArrayMap<String, ArrayList<Property>> propertyMap = 211 propertyCollection.get(propertyName); 212 if (propertyMap == null) { 213 // error 214 continue; 215 } 216 ArrayList<Property> packageProperties = propertyMap.get(packageName); 217 if (packageProperties == null) { 218 //error 219 continue; 220 } 221 packageProperties.remove(property); 222 223 // clean up empty structures 224 if (packageProperties.size() == 0) { 225 propertyMap.remove(packageName); 226 } 227 if (propertyMap.size() == 0) { 228 propertyCollection.remove(propertyName); 229 } 230 } 231 if (propertyCollection.size() == 0) { 232 return null; 233 } 234 return propertyCollection; 235 } 236 getProperty(String propertyName, String packageName, String className, ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyMap)237 private static Property getProperty(String propertyName, String packageName, String className, 238 ArrayMap<String, ArrayMap<String, ArrayList<Property>>> propertyMap) { 239 final ArrayMap<String, ArrayList<Property>> packagePropertyMap = 240 propertyMap.get(propertyName); 241 if (packagePropertyMap == null) { 242 return null; 243 } 244 final List<Property> propertyList = packagePropertyMap.get(packageName); 245 if (propertyList == null) { 246 return null; 247 } 248 for (int i = propertyList.size() - 1; i >= 0; i--) { 249 final Property property = propertyList.get(i); 250 if (Objects.equals(className, property.getClassName())) { 251 return property; 252 } 253 } 254 return null; 255 } 256 getComponentProperty( String propertyName, String packageName, String className)257 private Property getComponentProperty( 258 String propertyName, String packageName, String className) { 259 Property property = null; 260 if (property == null && mActivityProperties != null) { 261 property = getProperty(propertyName, packageName, className, mActivityProperties); 262 } 263 if (property == null && mProviderProperties != null) { 264 property = getProperty(propertyName, packageName, className, mProviderProperties); 265 } 266 if (property == null && mReceiverProperties != null) { 267 property = getProperty(propertyName, packageName, className, mReceiverProperties); 268 } 269 if (property == null && mServiceProperties != null) { 270 property = getProperty(propertyName, packageName, className, mServiceProperties); 271 } 272 return property; 273 } 274 getApplicationProperty(String propertyName, String packageName)275 private Property getApplicationProperty(String propertyName, String packageName) { 276 final ArrayMap<String, ArrayList<Property>> packagePropertyMap = 277 mApplicationProperties != null ? mApplicationProperties.get(propertyName) : null; 278 if (packagePropertyMap == null) { 279 return null; 280 } 281 final List<Property> propertyList = packagePropertyMap.get(packageName); 282 if (propertyList == null) { 283 return null; 284 } 285 return propertyList.get(0); 286 } 287 } 288