ProviderMap.java revision 7d19e0242faac8017033dabb872cdf1542fa184c
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.am;
18
19import android.content.ComponentName;
20import android.os.Binder;
21import android.os.Process;
22import android.os.RemoteException;
23import android.os.UserId;
24import android.util.Slog;
25import android.util.SparseArray;
26
27import java.io.FileDescriptor;
28import java.io.IOException;
29import java.io.PrintWriter;
30import java.util.ArrayList;
31import java.util.HashMap;
32import java.util.Iterator;
33import java.util.Map;
34import java.util.Map.Entry;
35import java.util.Set;
36
37/**
38 * Keeps track of content providers by authority (name) and class. It separates the mapping by
39 * user and ones that are not user-specific (system providers).
40 */
41public class ProviderMap {
42
43    private static final String TAG = "ProviderMap";
44
45    private static final boolean DBG = false;
46
47    private final HashMap<String, ContentProviderRecord> mSingletonByName
48            = new HashMap<String, ContentProviderRecord>();
49    private final HashMap<ComponentName, ContentProviderRecord> mSingletonByClass
50            = new HashMap<ComponentName, ContentProviderRecord>();
51
52    private final SparseArray<HashMap<String, ContentProviderRecord>> mProvidersByNamePerUser
53            = new SparseArray<HashMap<String, ContentProviderRecord>>();
54    private final SparseArray<HashMap<ComponentName, ContentProviderRecord>> mProvidersByClassPerUser
55            = new SparseArray<HashMap<ComponentName, ContentProviderRecord>>();
56
57    ContentProviderRecord getProviderByName(String name) {
58        return getProviderByName(name, -1);
59    }
60
61    ContentProviderRecord getProviderByName(String name, int userId) {
62        if (DBG) {
63            Slog.i(TAG, "getProviderByName: " + name + " , callingUid = " + Binder.getCallingUid());
64        }
65        // Try to find it in the global list
66        ContentProviderRecord record = mSingletonByName.get(name);
67        if (record != null) {
68            return record;
69        }
70
71        // Check the current user's list
72        return getProvidersByName(userId).get(name);
73    }
74
75    ContentProviderRecord getProviderByClass(ComponentName name) {
76        return getProviderByClass(name, -1);
77    }
78
79    ContentProviderRecord getProviderByClass(ComponentName name, int userId) {
80        if (DBG) {
81            Slog.i(TAG, "getProviderByClass: " + name + ", callingUid = " + Binder.getCallingUid());
82        }
83        // Try to find it in the global list
84        ContentProviderRecord record = mSingletonByClass.get(name);
85        if (record != null) {
86            return record;
87        }
88
89        // Check the current user's list
90        return getProvidersByClass(userId).get(name);
91    }
92
93    void putProviderByName(String name, ContentProviderRecord record) {
94        if (DBG) {
95            Slog.i(TAG, "putProviderByName: " + name + " , callingUid = " + Binder.getCallingUid()
96                + ", record uid = " + record.appInfo.uid);
97        }
98        if (record.singleton) {
99            mSingletonByName.put(name, record);
100        } else {
101            final int userId = UserId.getUserId(record.appInfo.uid);
102            getProvidersByName(userId).put(name, record);
103        }
104    }
105
106    void putProviderByClass(ComponentName name, ContentProviderRecord record) {
107        if (DBG) {
108            Slog.i(TAG, "putProviderByClass: " + name + " , callingUid = " + Binder.getCallingUid()
109                + ", record uid = " + record.appInfo.uid);
110        }
111        if (record.singleton) {
112            mSingletonByClass.put(name, record);
113        } else {
114            final int userId = UserId.getUserId(record.appInfo.uid);
115            getProvidersByClass(userId).put(name, record);
116        }
117    }
118
119    void removeProviderByName(String name, int optionalUserId) {
120        if (mSingletonByName.containsKey(name)) {
121            if (DBG)
122                Slog.i(TAG, "Removing from globalByName name=" + name);
123            mSingletonByName.remove(name);
124        } else {
125            // TODO: Verify this works, i.e., the caller happens to be from the correct user
126            if (DBG)
127                Slog.i(TAG,
128                        "Removing from providersByName name=" + name + " user="
129                        + (optionalUserId == -1 ? Binder.getOrigCallingUser() : optionalUserId));
130            getProvidersByName(optionalUserId).remove(name);
131        }
132    }
133
134    void removeProviderByClass(ComponentName name, int optionalUserId) {
135        if (mSingletonByClass.containsKey(name)) {
136            if (DBG)
137                Slog.i(TAG, "Removing from globalByClass name=" + name);
138            mSingletonByClass.remove(name);
139        } else {
140            if (DBG)
141                Slog.i(TAG,
142                        "Removing from providersByClass name=" + name + " user="
143                        + (optionalUserId == -1 ? Binder.getOrigCallingUser() : optionalUserId));
144            getProvidersByClass(optionalUserId).remove(name);
145        }
146    }
147
148    private HashMap<String, ContentProviderRecord> getProvidersByName(int optionalUserId) {
149        final int userId = optionalUserId >= 0
150                ? optionalUserId : Binder.getOrigCallingUser();
151        final HashMap<String, ContentProviderRecord> map = mProvidersByNamePerUser.get(userId);
152        if (map == null) {
153            HashMap<String, ContentProviderRecord> newMap = new HashMap<String, ContentProviderRecord>();
154            mProvidersByNamePerUser.put(userId, newMap);
155            return newMap;
156        } else {
157            return map;
158        }
159    }
160
161    HashMap<ComponentName, ContentProviderRecord> getProvidersByClass(int optionalUserId) {
162        final int userId = optionalUserId >= 0
163                ? optionalUserId : Binder.getOrigCallingUser();
164        final HashMap<ComponentName, ContentProviderRecord> map = mProvidersByClassPerUser.get(userId);
165        if (map == null) {
166            HashMap<ComponentName, ContentProviderRecord> newMap = new HashMap<ComponentName, ContentProviderRecord>();
167            mProvidersByClassPerUser.put(userId, newMap);
168            return newMap;
169        } else {
170            return map;
171        }
172    }
173
174    private void dumpProvidersByClassLocked(PrintWriter pw, boolean dumpAll,
175            HashMap<ComponentName, ContentProviderRecord> map) {
176        Iterator<Map.Entry<ComponentName, ContentProviderRecord>> it = map.entrySet().iterator();
177        while (it.hasNext()) {
178            Map.Entry<ComponentName, ContentProviderRecord> e = it.next();
179            ContentProviderRecord r = e.getValue();
180            pw.print("  * ");
181            pw.println(r);
182            r.dump(pw, "    ", dumpAll);
183        }
184    }
185
186    private void dumpProvidersByNameLocked(PrintWriter pw,
187            HashMap<String, ContentProviderRecord> map) {
188        Iterator<Map.Entry<String, ContentProviderRecord>> it = map.entrySet().iterator();
189        while (it.hasNext()) {
190            Map.Entry<String, ContentProviderRecord> e = it.next();
191            ContentProviderRecord r = e.getValue();
192            pw.print("  ");
193            pw.print(e.getKey());
194            pw.print(": ");
195            pw.println(r.toShortString());
196        }
197    }
198
199    void dumpProvidersLocked(PrintWriter pw, boolean dumpAll) {
200        if (mSingletonByClass.size() > 0) {
201            pw.println("");
202            pw.println("  Published single-user content providers (by class):");
203            dumpProvidersByClassLocked(pw, dumpAll, mSingletonByClass);
204        }
205
206        pw.println("");
207        for (int i = 0; i < mProvidersByClassPerUser.size(); i++) {
208            HashMap<ComponentName, ContentProviderRecord> map = mProvidersByClassPerUser.valueAt(i);
209            pw.println("  Published user " + mProvidersByClassPerUser.keyAt(i)
210                    + " content providers (by class):");
211            dumpProvidersByClassLocked(pw, dumpAll, map);
212            pw.println(" ");
213        }
214
215        if (dumpAll) {
216            pw.println("");
217            pw.println("  Single-user authority to provider mappings:");
218            dumpProvidersByNameLocked(pw, mSingletonByName);
219
220            for (int i = 0; i < mProvidersByNamePerUser.size(); i++) {
221                pw.println("");
222                pw.println("  User " + mProvidersByNamePerUser.keyAt(i)
223                        + " authority to provider mappings:");
224                dumpProvidersByNameLocked(pw, mProvidersByNamePerUser.valueAt(i));
225            }
226        }
227    }
228
229    protected boolean dumpProvider(FileDescriptor fd, PrintWriter pw, String name, String[] args,
230            int opti, boolean dumpAll) {
231        ArrayList<ContentProviderRecord> providers = new ArrayList<ContentProviderRecord>();
232
233        if ("all".equals(name)) {
234            synchronized (this) {
235                for (ContentProviderRecord r1 : getProvidersByClass(-1).values()) {
236                    providers.add(r1);
237                }
238            }
239        } else {
240            ComponentName componentName = name != null
241                    ? ComponentName.unflattenFromString(name) : null;
242            int objectId = 0;
243            if (componentName == null) {
244                // Not a '/' separated full component name; maybe an object ID?
245                try {
246                    objectId = Integer.parseInt(name, 16);
247                    name = null;
248                    componentName = null;
249                } catch (RuntimeException e) {
250                }
251            }
252
253            synchronized (this) {
254                for (ContentProviderRecord r1 : getProvidersByClass(-1).values()) {
255                    if (componentName != null) {
256                        if (r1.name.equals(componentName)) {
257                            providers.add(r1);
258                        }
259                    } else if (name != null) {
260                        if (r1.name.flattenToString().contains(name)) {
261                            providers.add(r1);
262                        }
263                    } else if (System.identityHashCode(r1) == objectId) {
264                        providers.add(r1);
265                    }
266                }
267            }
268        }
269
270        if (providers.size() <= 0) {
271            return false;
272        }
273
274        boolean needSep = false;
275        for (int i=0; i<providers.size(); i++) {
276            if (needSep) {
277                pw.println();
278            }
279            needSep = true;
280            dumpProvider("", fd, pw, providers.get(i), args, dumpAll);
281        }
282        return true;
283    }
284
285    /**
286     * Invokes IApplicationThread.dumpProvider() on the thread of the specified provider if
287     * there is a thread associated with the provider.
288     */
289    private void dumpProvider(String prefix, FileDescriptor fd, PrintWriter pw,
290            final ContentProviderRecord r, String[] args, boolean dumpAll) {
291        String innerPrefix = prefix + "  ";
292        synchronized (this) {
293            pw.print(prefix); pw.print("PROVIDER ");
294                    pw.print(r);
295                    pw.print(" pid=");
296                    if (r.proc != null) pw.println(r.proc.pid);
297                    else pw.println("(not running)");
298            if (dumpAll) {
299                r.dump(pw, innerPrefix, true);
300            }
301        }
302        if (r.proc != null && r.proc.thread != null) {
303            pw.println("    Client:");
304            pw.flush();
305            try {
306                TransferPipe tp = new TransferPipe();
307                try {
308                    r.proc.thread.dumpProvider(
309                            tp.getWriteFd().getFileDescriptor(), r.provider.asBinder(), args);
310                    tp.setBufferPrefix("      ");
311                    // Short timeout, since blocking here can
312                    // deadlock with the application.
313                    tp.go(fd, 2000);
314                } finally {
315                    tp.kill();
316                }
317            } catch (IOException ex) {
318                pw.println("      Failure while dumping the provider: " + ex);
319            } catch (RemoteException ex) {
320                pw.println("      Got a RemoteException while dumping the service");
321            }
322        }
323    }
324}
325