1 /**
2  * Copyright (C) 2016 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 package com.android.server.vr;
17 
18 import android.annotation.NonNull;
19 import android.app.ActivityManager;
20 import android.content.ComponentName;
21 import android.content.ContentResolver;
22 import android.content.Context;
23 import android.content.Intent;
24 import android.content.pm.PackageManager;
25 import android.content.pm.ResolveInfo;
26 import android.content.pm.ServiceInfo;
27 import android.os.Handler;
28 import android.os.Looper;
29 import android.os.UserHandle;
30 import android.os.UserManager;
31 import android.provider.Settings;
32 import android.text.TextUtils;
33 import android.util.ArraySet;
34 import android.util.Slog;
35 import android.util.SparseArray;
36 
37 import com.android.internal.content.PackageMonitor;
38 import com.android.server.vr.SettingsObserver.SettingChangeListener;
39 
40 import java.util.Collection;
41 import java.util.List;
42 import java.util.Set;
43 
44 /**
45  * Detects changes in packages, settings, and current users that may affect whether components
46  * implementing a given service can be run.
47  *
48  * @hide
49  */
50 public class EnabledComponentsObserver implements SettingChangeListener {
51 
52     private static final String TAG = EnabledComponentsObserver.class.getSimpleName();
53     private static final String ENABLED_SERVICES_SEPARATOR = ":";
54 
55     public static final int NO_ERROR = 0;
56     public static final int DISABLED = -1;
57     public static final int NOT_INSTALLED = -2;
58 
59     private final Object mLock;
60     private final Context mContext;
61     private final String mSettingName;
62     private final String mServiceName;
63     private final String mServicePermission;
64     private final SparseArray<ArraySet<ComponentName>> mInstalledSet = new SparseArray<>();
65     private final SparseArray<ArraySet<ComponentName>> mEnabledSet = new SparseArray<>();
66     private final Set<EnabledComponentChangeListener> mEnabledComponentListeners = new ArraySet<>();
67 
68     /**
69      * Implement this to receive callbacks when relevant changes to the allowed components occur.
70      */
71     public interface EnabledComponentChangeListener {
72 
73         /**
74          * Called when a change in the allowed components occurs.
75          */
onEnabledComponentChanged()76         void onEnabledComponentChanged();
77     }
78 
EnabledComponentsObserver(@onNull Context context, @NonNull String settingName, @NonNull String servicePermission, @NonNull String serviceName, @NonNull Object lock, @NonNull Collection<EnabledComponentChangeListener> listeners)79     private EnabledComponentsObserver(@NonNull Context context, @NonNull String settingName,
80             @NonNull String servicePermission, @NonNull String serviceName, @NonNull Object lock,
81             @NonNull Collection<EnabledComponentChangeListener> listeners) {
82         mLock = lock;
83         mContext = context;
84         mSettingName = settingName;
85         mServiceName = serviceName;
86         mServicePermission = servicePermission;
87         mEnabledComponentListeners.addAll(listeners);
88     }
89 
90     /**
91      * Create a EnabledComponentObserver instance.
92      *
93      * @param context the context to query for changes.
94      * @param handler a handler to receive lifecycle events from system services on.
95      * @param settingName the name of a setting to monitor for a list of enabled components.
96      * @param looper a {@link Looper} to use for receiving package callbacks.
97      * @param servicePermission the permission required by the components to be bound.
98      * @param serviceName the intent action implemented by the tracked components.
99      * @param lock a lock object used to guard instance state in all callbacks and method calls.
100      * @return an EnableComponentObserver instance.
101      */
build(@onNull Context context, @NonNull Handler handler, @NonNull String settingName, @NonNull Looper looper, @NonNull String servicePermission, @NonNull String serviceName, @NonNull final Object lock, @NonNull Collection<EnabledComponentChangeListener> listeners)102     public static EnabledComponentsObserver build(@NonNull Context context,
103             @NonNull Handler handler, @NonNull String settingName, @NonNull Looper looper,
104             @NonNull String servicePermission, @NonNull String serviceName,
105             @NonNull final Object lock,
106             @NonNull Collection<EnabledComponentChangeListener> listeners) {
107 
108         SettingsObserver s = SettingsObserver.build(context, handler, settingName);
109 
110         final EnabledComponentsObserver o = new EnabledComponentsObserver(context, settingName,
111                 servicePermission, serviceName, lock, listeners);
112 
113         PackageMonitor packageMonitor = new PackageMonitor(true) {
114             @Override
115             public void onSomePackagesChanged() {
116                 o.onPackagesChanged();
117 
118             }
119 
120             @Override
121             public void onPackageDisappeared(String packageName, int reason) {
122                 o.onPackagesChanged();
123 
124             }
125 
126             @Override
127             public void onPackageModified(String packageName) {
128                 o.onPackagesChanged();
129 
130             }
131 
132             @Override
133             public boolean onHandleForceStop(Intent intent, String[] packages, int uid,
134                     boolean doit) {
135                 o.onPackagesChanged();
136 
137                 return super.onHandleForceStop(intent, packages, uid, doit);
138             }
139         };
140 
141         packageMonitor.register(context, looper, UserHandle.ALL, true);
142 
143         s.addListener(o);
144 
145         return o;
146 
147     }
148 
onPackagesChanged()149     public void onPackagesChanged() {
150         rebuildAll();
151     }
152 
153     @Override
onSettingChanged()154     public void onSettingChanged() {
155         rebuildAll();
156     }
157 
158     @Override
onSettingRestored(String prevValue, String newValue, int userId)159     public void onSettingRestored(String prevValue, String newValue, int userId) {
160         rebuildAll();
161     }
162 
onUsersChanged()163     public void onUsersChanged() {
164         rebuildAll();
165     }
166 
167     /**
168      * Rebuild the sets of allowed components for each current user profile.
169      */
rebuildAll()170     public void rebuildAll() {
171         synchronized (mLock) {
172             mInstalledSet.clear();
173             mEnabledSet.clear();
174             final int[] userIds = getCurrentProfileIds();
175             for (int i : userIds) {
176                 ArraySet<ComponentName> implementingPackages = loadComponentNamesForUser(i);
177                 ArraySet<ComponentName> packagesFromSettings =
178                         loadComponentNamesFromSetting(mSettingName, i);
179                 packagesFromSettings.retainAll(implementingPackages);
180 
181                 mInstalledSet.put(i, implementingPackages);
182                 mEnabledSet.put(i, packagesFromSettings);
183 
184             }
185         }
186         sendSettingChanged();
187     }
188 
189     /**
190      * Check whether a given component is present and enabled for the given user.
191      *
192      * @param component the component to check.
193      * @param userId the user ID for the component to check.
194      * @return {@code true} if present and enabled.
195      */
isValid(ComponentName component, int userId)196     public int isValid(ComponentName component, int userId) {
197         synchronized (mLock) {
198             ArraySet<ComponentName> installedComponents = mInstalledSet.get(userId);
199             if (installedComponents == null || !installedComponents.contains(component)) {
200                 return NOT_INSTALLED;
201             }
202             ArraySet<ComponentName> validComponents = mEnabledSet.get(userId);
203             if (validComponents == null || !validComponents.contains(component)) {
204                 return DISABLED;
205             }
206             return NO_ERROR;
207         }
208     }
209 
210     /**
211      * Return all VrListenerService components installed for this user.
212      *
213      * @param userId ID of the user to check.
214      * @return a set of {@link ComponentName}s.
215      */
getInstalled(int userId)216     public ArraySet<ComponentName> getInstalled(int userId) {
217         synchronized (mLock) {
218             ArraySet<ComponentName> ret = mInstalledSet.get(userId);
219             if (ret == null) {
220                 return new ArraySet<ComponentName>();
221             }
222             return ret;
223         }
224     }
225 
226     /**
227      * Return all VrListenerService components enabled for this user.
228      *
229      * @param userId ID of the user to check.
230      * @return a set of {@link ComponentName}s.
231      */
getEnabled(int userId)232     public ArraySet<ComponentName> getEnabled(int userId) {
233         synchronized (mLock) {
234             ArraySet<ComponentName> ret = mEnabledSet.get(userId);
235             if (ret == null) {
236                 return new ArraySet<ComponentName>();
237             }
238             return ret;
239 
240         }
241     }
242 
getCurrentProfileIds()243     private int[] getCurrentProfileIds() {
244         UserManager userManager = (UserManager) mContext.getSystemService(Context.USER_SERVICE);
245         if (userManager == null) {
246             return null;
247         }
248         return userManager.getEnabledProfileIds(ActivityManager.getCurrentUser());
249     }
250 
loadComponentNames(PackageManager pm, int userId, String serviceName, String permissionName)251     public static ArraySet<ComponentName> loadComponentNames(PackageManager pm, int userId,
252             String serviceName, String permissionName) {
253 
254         ArraySet<ComponentName> installed = new ArraySet<>();
255         Intent queryIntent = new Intent(serviceName);
256         List<ResolveInfo> installedServices = pm.queryIntentServicesAsUser(
257                 queryIntent,
258                 PackageManager.GET_SERVICES | PackageManager.GET_META_DATA |
259                                     PackageManager.MATCH_DIRECT_BOOT_AWARE |
260                                     PackageManager.MATCH_DIRECT_BOOT_UNAWARE,
261                 userId);
262         if (installedServices != null) {
263             for (int i = 0, count = installedServices.size(); i < count; i++) {
264                 ResolveInfo resolveInfo = installedServices.get(i);
265                 ServiceInfo info = resolveInfo.serviceInfo;
266 
267                 ComponentName component = new ComponentName(info.packageName, info.name);
268                 if (!permissionName.equals(info.permission)) {
269                     Slog.w(TAG, "Skipping service " + info.packageName + "/" + info.name
270                             + ": it does not require the permission "
271                             + permissionName);
272                     continue;
273                 }
274                 installed.add(component);
275             }
276         }
277         return installed;
278     }
279 
loadComponentNamesForUser(int userId)280     private ArraySet<ComponentName> loadComponentNamesForUser(int userId) {
281         return loadComponentNames(mContext.getPackageManager(), userId, mServiceName,
282                 mServicePermission);
283     }
284 
loadComponentNamesFromSetting(String settingName, int userId)285     private ArraySet<ComponentName> loadComponentNamesFromSetting(String settingName,
286             int userId) {
287         final ContentResolver cr = mContext.getContentResolver();
288         String settingValue = Settings.Secure.getStringForUser(
289                 cr,
290                 settingName,
291                 userId);
292         if (TextUtils.isEmpty(settingValue))
293             return new ArraySet<>();
294         String[] restored = settingValue.split(ENABLED_SERVICES_SEPARATOR);
295         ArraySet<ComponentName> result = new ArraySet<>(restored.length);
296         for (int i = 0; i < restored.length; i++) {
297             ComponentName value = ComponentName.unflattenFromString(restored[i]);
298             if (null != value) {
299                 result.add(value);
300             }
301         }
302         return result;
303     }
304 
sendSettingChanged()305     private void sendSettingChanged() {
306         for (EnabledComponentChangeListener l : mEnabledComponentListeners) {
307             l.onEnabledComponentChanged();
308         }
309     }
310 
311 }
312