1 /*
2  * Copyright (C) 2006 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;
18 
19 import android.annotation.NonNull;
20 import android.content.Intent;
21 import android.content.IntentFilter;
22 import android.net.Uri;
23 import android.util.ArrayMap;
24 import android.util.ArraySet;
25 import android.util.FastImmutableArraySet;
26 import android.util.Log;
27 import android.util.LogPrinter;
28 import android.util.MutableInt;
29 import android.util.PrintWriterPrinter;
30 import android.util.Printer;
31 import android.util.Slog;
32 import android.util.proto.ProtoOutputStream;
33 
34 import com.android.internal.util.FastPrintWriter;
35 
36 import java.io.PrintWriter;
37 import java.util.ArrayList;
38 import java.util.Arrays;
39 import java.util.Collections;
40 import java.util.Comparator;
41 import java.util.Iterator;
42 import java.util.List;
43 import java.util.Set;
44 
45 /**
46  * {@hide}
47  */
48 public abstract class IntentResolver<F, R extends Object> {
49     final private static String TAG = "IntentResolver";
50     final private static boolean DEBUG = false;
51     final private static boolean localLOGV = DEBUG || false;
52     final private static boolean localVerificationLOGV = DEBUG || false;
53 
addFilter(F f)54     public void addFilter(F f) {
55         IntentFilter intentFilter = getIntentFilter(f);
56         if (localLOGV) {
57             Slog.v(TAG, "Adding filter: " + f);
58             intentFilter.dump(new LogPrinter(Log.VERBOSE, TAG, Log.LOG_ID_SYSTEM), "      ");
59             Slog.v(TAG, "    Building Lookup Maps:");
60         }
61 
62         mFilters.add(f);
63         int numS = register_intent_filter(f, intentFilter.schemesIterator(),
64                 mSchemeToFilter, "      Scheme: ");
65         int numT = register_mime_types(f, "      Type: ");
66         if (numS == 0 && numT == 0) {
67             register_intent_filter(f, intentFilter.actionsIterator(),
68                     mActionToFilter, "      Action: ");
69         }
70         if (numT != 0) {
71             register_intent_filter(f, intentFilter.actionsIterator(),
72                     mTypedActionToFilter, "      TypedAction: ");
73         }
74     }
75 
filterEquals(IntentFilter f1, IntentFilter f2)76     public static boolean filterEquals(IntentFilter f1, IntentFilter f2) {
77         int s1 = f1.countActions();
78         int s2 = f2.countActions();
79         if (s1 != s2) {
80             return false;
81         }
82         for (int i=0; i<s1; i++) {
83             if (!f2.hasAction(f1.getAction(i))) {
84                 return false;
85             }
86         }
87         s1 = f1.countCategories();
88         s2 = f2.countCategories();
89         if (s1 != s2) {
90             return false;
91         }
92         for (int i=0; i<s1; i++) {
93             if (!f2.hasCategory(f1.getCategory(i))) {
94                 return false;
95             }
96         }
97         s1 = f1.countDataTypes();
98         s2 = f2.countDataTypes();
99         if (s1 != s2) {
100             return false;
101         }
102         for (int i=0; i<s1; i++) {
103             if (!f2.hasExactDataType(f1.getDataType(i))) {
104                 return false;
105             }
106         }
107         s1 = f1.countDataSchemes();
108         s2 = f2.countDataSchemes();
109         if (s1 != s2) {
110             return false;
111         }
112         for (int i=0; i<s1; i++) {
113             if (!f2.hasDataScheme(f1.getDataScheme(i))) {
114                 return false;
115             }
116         }
117         s1 = f1.countDataAuthorities();
118         s2 = f2.countDataAuthorities();
119         if (s1 != s2) {
120             return false;
121         }
122         for (int i=0; i<s1; i++) {
123             if (!f2.hasDataAuthority(f1.getDataAuthority(i))) {
124                 return false;
125             }
126         }
127         s1 = f1.countDataPaths();
128         s2 = f2.countDataPaths();
129         if (s1 != s2) {
130             return false;
131         }
132         for (int i=0; i<s1; i++) {
133             if (!f2.hasDataPath(f1.getDataPath(i))) {
134                 return false;
135             }
136         }
137         s1 = f1.countDataSchemeSpecificParts();
138         s2 = f2.countDataSchemeSpecificParts();
139         if (s1 != s2) {
140             return false;
141         }
142         for (int i=0; i<s1; i++) {
143             if (!f2.hasDataSchemeSpecificPart(f1.getDataSchemeSpecificPart(i))) {
144                 return false;
145             }
146         }
147         return true;
148     }
149 
collectFilters(F[] array, IntentFilter matching)150     private ArrayList<F> collectFilters(F[] array, IntentFilter matching) {
151         ArrayList<F> res = null;
152         if (array != null) {
153             for (int i=0; i<array.length; i++) {
154                 F cur = array[i];
155                 if (cur == null) {
156                     break;
157                 }
158                 if (filterEquals(getIntentFilter(cur), matching)) {
159                     if (res == null) {
160                         res = new ArrayList<>();
161                     }
162                     res.add(cur);
163                 }
164             }
165         }
166         return res;
167     }
168 
findFilters(IntentFilter matching)169     public ArrayList<F> findFilters(IntentFilter matching) {
170         if (matching.countDataSchemes() == 1) {
171             // Fast case.
172             return collectFilters(mSchemeToFilter.get(matching.getDataScheme(0)), matching);
173         } else if (matching.countDataTypes() != 0 && matching.countActions() == 1) {
174             // Another fast case.
175             return collectFilters(mTypedActionToFilter.get(matching.getAction(0)), matching);
176         } else if (matching.countDataTypes() == 0 && matching.countDataSchemes() == 0
177                 && matching.countActions() == 1) {
178             // Last fast case.
179             return collectFilters(mActionToFilter.get(matching.getAction(0)), matching);
180         } else {
181             ArrayList<F> res = null;
182             for (F cur : mFilters) {
183                 if (filterEquals(getIntentFilter(cur), matching)) {
184                     if (res == null) {
185                         res = new ArrayList<>();
186                     }
187                     res.add(cur);
188                 }
189             }
190             return res;
191         }
192     }
193 
removeFilter(F f)194     public void removeFilter(F f) {
195         removeFilterInternal(f);
196         mFilters.remove(f);
197     }
198 
removeFilterInternal(F f)199     protected void removeFilterInternal(F f) {
200         IntentFilter intentFilter = getIntentFilter(f);
201         if (localLOGV) {
202             Slog.v(TAG, "Removing filter: " + f);
203             intentFilter.dump(new LogPrinter(Log.VERBOSE, TAG, Log.LOG_ID_SYSTEM), "      ");
204             Slog.v(TAG, "    Cleaning Lookup Maps:");
205         }
206 
207         int numS = unregister_intent_filter(f, intentFilter.schemesIterator(),
208                 mSchemeToFilter, "      Scheme: ");
209         int numT = unregister_mime_types(f, "      Type: ");
210         if (numS == 0 && numT == 0) {
211             unregister_intent_filter(f, intentFilter.actionsIterator(),
212                     mActionToFilter, "      Action: ");
213         }
214         if (numT != 0) {
215             unregister_intent_filter(f, intentFilter.actionsIterator(),
216                     mTypedActionToFilter, "      TypedAction: ");
217         }
218     }
219 
dumpMap(PrintWriter out, String titlePrefix, String title, String prefix, ArrayMap<String, F[]> map, String packageName, boolean printFilter, boolean collapseDuplicates)220     boolean dumpMap(PrintWriter out, String titlePrefix, String title,
221             String prefix, ArrayMap<String, F[]> map, String packageName,
222             boolean printFilter, boolean collapseDuplicates) {
223         final String eprefix = prefix + "  ";
224         final String fprefix = prefix + "    ";
225         final ArrayMap<Object, MutableInt> found = new ArrayMap<>();
226         boolean printedSomething = false;
227         Printer printer = null;
228         for (int mapi=0; mapi<map.size(); mapi++) {
229             F[] a = map.valueAt(mapi);
230             final int N = a.length;
231             boolean printedHeader = false;
232             F filter;
233             if (collapseDuplicates && !printFilter) {
234                 found.clear();
235                 for (int i=0; i<N && (filter=a[i]) != null; i++) {
236                     if (packageName != null && !isPackageForFilter(packageName, filter)) {
237                         continue;
238                     }
239                     Object label = filterToLabel(filter);
240                     int index = found.indexOfKey(label);
241                     if (index < 0) {
242                         found.put(label, new MutableInt(1));
243                     } else {
244                         found.valueAt(index).value++;
245                     }
246                 }
247                 for (int i=0; i<found.size(); i++) {
248                     if (title != null) {
249                         out.print(titlePrefix); out.println(title);
250                         title = null;
251                     }
252                     if (!printedHeader) {
253                         out.print(eprefix); out.print(map.keyAt(mapi)); out.println(":");
254                         printedHeader = true;
255                     }
256                     printedSomething = true;
257                     dumpFilterLabel(out, fprefix, found.keyAt(i), found.valueAt(i).value);
258                 }
259             } else {
260                 for (int i=0; i<N && (filter=a[i]) != null; i++) {
261                     if (packageName != null && !isPackageForFilter(packageName, filter)) {
262                         continue;
263                     }
264                     if (title != null) {
265                         out.print(titlePrefix); out.println(title);
266                         title = null;
267                     }
268                     if (!printedHeader) {
269                         out.print(eprefix); out.print(map.keyAt(mapi)); out.println(":");
270                         printedHeader = true;
271                     }
272                     printedSomething = true;
273                     dumpFilter(out, fprefix, filter);
274                     if (printFilter) {
275                         if (printer == null) {
276                             printer = new PrintWriterPrinter(out);
277                         }
278                         getIntentFilter(filter).dump(printer, fprefix + "  ");
279                     }
280                 }
281             }
282         }
283         return printedSomething;
284     }
285 
writeProtoMap(ProtoOutputStream proto, long fieldId, ArrayMap<String, F[]> map)286     void writeProtoMap(ProtoOutputStream proto, long fieldId, ArrayMap<String, F[]> map) {
287         int N = map.size();
288         for (int mapi = 0; mapi < N; mapi++) {
289             long token = proto.start(fieldId);
290             proto.write(IntentResolverProto.ArrayMapEntry.KEY, map.keyAt(mapi));
291             for (F f : map.valueAt(mapi)) {
292                 if (f != null) {
293                     proto.write(IntentResolverProto.ArrayMapEntry.VALUES, f.toString());
294                 }
295             }
296             proto.end(token);
297         }
298     }
299 
dumpDebug(ProtoOutputStream proto, long fieldId)300     public void dumpDebug(ProtoOutputStream proto, long fieldId) {
301         long token = proto.start(fieldId);
302         writeProtoMap(proto, IntentResolverProto.FULL_MIME_TYPES, mTypeToFilter);
303         writeProtoMap(proto, IntentResolverProto.BASE_MIME_TYPES, mBaseTypeToFilter);
304         writeProtoMap(proto, IntentResolverProto.WILD_MIME_TYPES, mWildTypeToFilter);
305         writeProtoMap(proto, IntentResolverProto.SCHEMES, mSchemeToFilter);
306         writeProtoMap(proto, IntentResolverProto.NON_DATA_ACTIONS, mActionToFilter);
307         writeProtoMap(proto, IntentResolverProto.MIME_TYPED_ACTIONS, mTypedActionToFilter);
308         proto.end(token);
309     }
310 
dump(PrintWriter out, String title, String prefix, String packageName, boolean printFilter, boolean collapseDuplicates)311     public boolean dump(PrintWriter out, String title, String prefix, String packageName,
312             boolean printFilter, boolean collapseDuplicates) {
313         String innerPrefix = prefix + "  ";
314         String sepPrefix = "\n" + prefix;
315         String curPrefix = title + "\n" + prefix;
316         if (dumpMap(out, curPrefix, "Full MIME Types:", innerPrefix,
317                 mTypeToFilter, packageName, printFilter, collapseDuplicates)) {
318             curPrefix = sepPrefix;
319         }
320         if (dumpMap(out, curPrefix, "Base MIME Types:", innerPrefix,
321                 mBaseTypeToFilter, packageName, printFilter, collapseDuplicates)) {
322             curPrefix = sepPrefix;
323         }
324         if (dumpMap(out, curPrefix, "Wild MIME Types:", innerPrefix,
325                 mWildTypeToFilter, packageName, printFilter, collapseDuplicates)) {
326             curPrefix = sepPrefix;
327         }
328         if (dumpMap(out, curPrefix, "Schemes:", innerPrefix,
329                 mSchemeToFilter, packageName, printFilter, collapseDuplicates)) {
330             curPrefix = sepPrefix;
331         }
332         if (dumpMap(out, curPrefix, "Non-Data Actions:", innerPrefix,
333                 mActionToFilter, packageName, printFilter, collapseDuplicates)) {
334             curPrefix = sepPrefix;
335         }
336         if (dumpMap(out, curPrefix, "MIME Typed Actions:", innerPrefix,
337                 mTypedActionToFilter, packageName, printFilter, collapseDuplicates)) {
338             curPrefix = sepPrefix;
339         }
340         return curPrefix == sepPrefix;
341     }
342 
343     private class IteratorWrapper implements Iterator<F> {
344         private final Iterator<F> mI;
345         private F mCur;
346 
IteratorWrapper(Iterator<F> it)347         IteratorWrapper(Iterator<F> it) {
348             mI = it;
349         }
350 
hasNext()351         public boolean hasNext() {
352             return mI.hasNext();
353         }
354 
next()355         public F next() {
356             return (mCur = mI.next());
357         }
358 
remove()359         public void remove() {
360             if (mCur != null) {
361                 removeFilterInternal(mCur);
362             }
363             mI.remove();
364         }
365 
366     }
367 
368     /**
369      * Returns an iterator allowing filters to be removed.
370      */
filterIterator()371     public Iterator<F> filterIterator() {
372         return new IteratorWrapper(mFilters.iterator());
373     }
374 
375     /**
376      * Returns a read-only set of the filters.
377      */
filterSet()378     public Set<F> filterSet() {
379         return Collections.unmodifiableSet(mFilters);
380     }
381 
queryIntentFromList(Intent intent, String resolvedType, boolean defaultOnly, ArrayList<F[]> listCut, int userId)382     public List<R> queryIntentFromList(Intent intent, String resolvedType, boolean defaultOnly,
383             ArrayList<F[]> listCut, int userId) {
384         ArrayList<R> resultList = new ArrayList<R>();
385 
386         final boolean debug = localLOGV ||
387                 ((intent.getFlags() & Intent.FLAG_DEBUG_LOG_RESOLUTION) != 0);
388 
389         FastImmutableArraySet<String> categories = getFastIntentCategories(intent);
390         final String scheme = intent.getScheme();
391         int N = listCut.size();
392         for (int i = 0; i < N; ++i) {
393             buildResolveList(intent, categories, debug, defaultOnly, resolvedType, scheme,
394                     listCut.get(i), resultList, userId);
395         }
396         filterResults(resultList);
397         sortResults(resultList);
398         return resultList;
399     }
400 
queryIntent(Intent intent, String resolvedType, boolean defaultOnly, int userId)401     public List<R> queryIntent(Intent intent, String resolvedType, boolean defaultOnly,
402             int userId) {
403         String scheme = intent.getScheme();
404 
405         ArrayList<R> finalList = new ArrayList<R>();
406 
407         final boolean debug = localLOGV ||
408                 ((intent.getFlags() & Intent.FLAG_DEBUG_LOG_RESOLUTION) != 0);
409 
410         if (debug) Slog.v(
411             TAG, "Resolving type=" + resolvedType + " scheme=" + scheme
412             + " defaultOnly=" + defaultOnly + " userId=" + userId + " of " + intent);
413 
414         F[] firstTypeCut = null;
415         F[] secondTypeCut = null;
416         F[] thirdTypeCut = null;
417         F[] schemeCut = null;
418 
419         // If the intent includes a MIME type, then we want to collect all of
420         // the filters that match that MIME type.
421         if (resolvedType != null) {
422             int slashpos = resolvedType.indexOf('/');
423             if (slashpos > 0) {
424                 final String baseType = resolvedType.substring(0, slashpos);
425                 if (!baseType.equals("*")) {
426                     if (resolvedType.length() != slashpos+2
427                             || resolvedType.charAt(slashpos+1) != '*') {
428                         // Not a wild card, so we can just look for all filters that
429                         // completely match or wildcards whose base type matches.
430                         firstTypeCut = mTypeToFilter.get(resolvedType);
431                         if (debug) Slog.v(TAG, "First type cut: " + Arrays.toString(firstTypeCut));
432                         secondTypeCut = mWildTypeToFilter.get(baseType);
433                         if (debug) Slog.v(TAG, "Second type cut: "
434                                 + Arrays.toString(secondTypeCut));
435                     } else {
436                         // We can match anything with our base type.
437                         firstTypeCut = mBaseTypeToFilter.get(baseType);
438                         if (debug) Slog.v(TAG, "First type cut: " + Arrays.toString(firstTypeCut));
439                         secondTypeCut = mWildTypeToFilter.get(baseType);
440                         if (debug) Slog.v(TAG, "Second type cut: "
441                                 + Arrays.toString(secondTypeCut));
442                     }
443                     // Any */* types always apply, but we only need to do this
444                     // if the intent type was not already */*.
445                     thirdTypeCut = mWildTypeToFilter.get("*");
446                     if (debug) Slog.v(TAG, "Third type cut: " + Arrays.toString(thirdTypeCut));
447                 } else if (intent.getAction() != null) {
448                     // The intent specified any type ({@literal *}/*).  This
449                     // can be a whole heck of a lot of things, so as a first
450                     // cut let's use the action instead.
451                     firstTypeCut = mTypedActionToFilter.get(intent.getAction());
452                     if (debug) Slog.v(TAG, "Typed Action list: " + Arrays.toString(firstTypeCut));
453                 }
454             }
455         }
456 
457         // If the intent includes a data URI, then we want to collect all of
458         // the filters that match its scheme (we will further refine matches
459         // on the authority and path by directly matching each resulting filter).
460         if (scheme != null) {
461             schemeCut = mSchemeToFilter.get(scheme);
462             if (debug) Slog.v(TAG, "Scheme list: " + Arrays.toString(schemeCut));
463         }
464 
465         // If the intent does not specify any data -- either a MIME type or
466         // a URI -- then we will only be looking for matches against empty
467         // data.
468         if (resolvedType == null && scheme == null && intent.getAction() != null) {
469             firstTypeCut = mActionToFilter.get(intent.getAction());
470             if (debug) Slog.v(TAG, "Action list: " + Arrays.toString(firstTypeCut));
471         }
472 
473         FastImmutableArraySet<String> categories = getFastIntentCategories(intent);
474         if (firstTypeCut != null) {
475             buildResolveList(intent, categories, debug, defaultOnly, resolvedType,
476                     scheme, firstTypeCut, finalList, userId);
477         }
478         if (secondTypeCut != null) {
479             buildResolveList(intent, categories, debug, defaultOnly, resolvedType,
480                     scheme, secondTypeCut, finalList, userId);
481         }
482         if (thirdTypeCut != null) {
483             buildResolveList(intent, categories, debug, defaultOnly, resolvedType,
484                     scheme, thirdTypeCut, finalList, userId);
485         }
486         if (schemeCut != null) {
487             buildResolveList(intent, categories, debug, defaultOnly, resolvedType,
488                     scheme, schemeCut, finalList, userId);
489         }
490         filterResults(finalList);
491         sortResults(finalList);
492 
493         if (debug) {
494             Slog.v(TAG, "Final result list:");
495             for (int i=0; i<finalList.size(); i++) {
496                 Slog.v(TAG, "  " + finalList.get(i));
497             }
498         }
499         return finalList;
500     }
501 
502     /**
503      * Control whether the given filter is allowed to go into the result
504      * list.  Mainly intended to prevent adding multiple filters for the
505      * same target object.
506      */
allowFilterResult(F filter, List<R> dest)507     protected boolean allowFilterResult(F filter, List<R> dest) {
508         return true;
509     }
510 
511     /**
512      * Returns whether the object associated with the given filter is
513      * "stopped", that is whether it should not be included in the result
514      * if the intent requests to excluded stopped objects.
515      */
isFilterStopped(F filter, int userId)516     protected boolean isFilterStopped(F filter, int userId) {
517         return false;
518     }
519 
520     /**
521      * Returns whether the given filter is "verified" that is whether it has been verified against
522      * its data URIs.
523      *
524      * The verification would happen only and only if the Intent action is
525      * {@link android.content.Intent#ACTION_VIEW} and the Intent category is
526      * {@link android.content.Intent#CATEGORY_BROWSABLE} and the Intent data scheme
527      * is "http" or "https".
528      *
529      * @see android.content.IntentFilter#setAutoVerify(boolean)
530      * @see android.content.IntentFilter#getAutoVerify()
531      */
isFilterVerified(F filter)532     protected boolean isFilterVerified(F filter) {
533         return getIntentFilter(filter).isVerified();
534     }
535 
536     /**
537      * Returns whether this filter is owned by this package. This must be
538      * implemented to provide correct filtering of Intents that have
539      * specified a package name they are to be delivered to.
540      */
isPackageForFilter(String packageName, F filter)541     protected abstract boolean isPackageForFilter(String packageName, F filter);
542 
newArray(int size)543     protected abstract F[] newArray(int size);
544 
545     @SuppressWarnings("unchecked")
newResult(F filter, int match, int userId)546     protected R newResult(F filter, int match, int userId) {
547         return (R)filter;
548     }
549 
550     @SuppressWarnings("unchecked")
sortResults(List<R> results)551     protected void sortResults(List<R> results) {
552         Collections.sort(results, mResolvePrioritySorter);
553     }
554 
555     /**
556      * Apply filtering to the results. This happens before the results are sorted.
557      */
filterResults(List<R> results)558     protected void filterResults(List<R> results) {
559     }
560 
dumpFilter(PrintWriter out, String prefix, F filter)561     protected void dumpFilter(PrintWriter out, String prefix, F filter) {
562         out.print(prefix); out.println(filter);
563     }
564 
filterToLabel(F filter)565     protected Object filterToLabel(F filter) {
566         return "IntentFilter";
567     }
568 
dumpFilterLabel(PrintWriter out, String prefix, Object label, int count)569     protected void dumpFilterLabel(PrintWriter out, String prefix, Object label, int count) {
570         out.print(prefix); out.print(label); out.print(": "); out.println(count);
571     }
572 
addFilter(ArrayMap<String, F[]> map, String name, F filter)573     private final void addFilter(ArrayMap<String, F[]> map, String name, F filter) {
574         F[] array = map.get(name);
575         if (array == null) {
576             array = newArray(2);
577             map.put(name,  array);
578             array[0] = filter;
579         } else {
580             final int N = array.length;
581             int i = N;
582             while (i > 0 && array[i-1] == null) {
583                 i--;
584             }
585             if (i < N) {
586                 array[i] = filter;
587             } else {
588                 F[] newa = newArray((N*3)/2);
589                 System.arraycopy(array, 0, newa, 0, N);
590                 newa[N] = filter;
591                 map.put(name, newa);
592             }
593         }
594     }
595 
register_mime_types(F filter, String prefix)596     private final int register_mime_types(F filter, String prefix) {
597         final Iterator<String> i = getIntentFilter(filter).typesIterator();
598         if (i == null) {
599             return 0;
600         }
601 
602         int num = 0;
603         while (i.hasNext()) {
604             String name = i.next();
605             num++;
606             if (localLOGV) Slog.v(TAG, prefix + name);
607             String baseName = name;
608             final int slashpos = name.indexOf('/');
609             if (slashpos > 0) {
610                 baseName = name.substring(0, slashpos).intern();
611             } else {
612                 name = name + "/*";
613             }
614 
615             addFilter(mTypeToFilter, name, filter);
616 
617             if (slashpos > 0) {
618                 addFilter(mBaseTypeToFilter, baseName, filter);
619             } else {
620                 addFilter(mWildTypeToFilter, baseName, filter);
621             }
622         }
623 
624         return num;
625     }
626 
unregister_mime_types(F filter, String prefix)627     private final int unregister_mime_types(F filter, String prefix) {
628         final Iterator<String> i = getIntentFilter(filter).typesIterator();
629         if (i == null) {
630             return 0;
631         }
632 
633         int num = 0;
634         while (i.hasNext()) {
635             String name = i.next();
636             num++;
637             if (localLOGV) Slog.v(TAG, prefix + name);
638             String baseName = name;
639             final int slashpos = name.indexOf('/');
640             if (slashpos > 0) {
641                 baseName = name.substring(0, slashpos).intern();
642             } else {
643                 name = name + "/*";
644             }
645 
646             remove_all_objects(mTypeToFilter, name, filter);
647 
648             if (slashpos > 0) {
649                 remove_all_objects(mBaseTypeToFilter, baseName, filter);
650             } else {
651                 remove_all_objects(mWildTypeToFilter, baseName, filter);
652             }
653         }
654         return num;
655     }
656 
register_intent_filter(F filter, Iterator<String> i, ArrayMap<String, F[]> dest, String prefix)657     protected final int register_intent_filter(F filter, Iterator<String> i,
658             ArrayMap<String, F[]> dest, String prefix) {
659         if (i == null) {
660             return 0;
661         }
662 
663         int num = 0;
664         while (i.hasNext()) {
665             String name = i.next();
666             num++;
667             if (localLOGV) Slog.v(TAG, prefix + name);
668             addFilter(dest, name, filter);
669         }
670         return num;
671     }
672 
unregister_intent_filter(F filter, Iterator<String> i, ArrayMap<String, F[]> dest, String prefix)673     protected final int unregister_intent_filter(F filter, Iterator<String> i,
674             ArrayMap<String, F[]> dest, String prefix) {
675         if (i == null) {
676             return 0;
677         }
678 
679         int num = 0;
680         while (i.hasNext()) {
681             String name = i.next();
682             num++;
683             if (localLOGV) Slog.v(TAG, prefix + name);
684             remove_all_objects(dest, name, filter);
685         }
686         return num;
687     }
688 
remove_all_objects(ArrayMap<String, F[]> map, String name, F object)689     private final void remove_all_objects(ArrayMap<String, F[]> map, String name,
690             F object) {
691         F[] array = map.get(name);
692         if (array != null) {
693             int LAST = array.length-1;
694             while (LAST >= 0 && array[LAST] == null) {
695                 LAST--;
696             }
697             for (int idx=LAST; idx>=0; idx--) {
698                 F arrayValue = array[idx];
699                 if (arrayValue != null && getIntentFilter(arrayValue) == getIntentFilter(object)) {
700                     final int remain = LAST - idx;
701                     if (remain > 0) {
702                         System.arraycopy(array, idx+1, array, idx, remain);
703                     }
704                     array[LAST] = null;
705                     LAST--;
706                 }
707             }
708             if (LAST < 0) {
709                 map.remove(name);
710             } else if (LAST < (array.length/2)) {
711                 F[] newa = newArray(LAST+2);
712                 System.arraycopy(array, 0, newa, 0, LAST+1);
713                 map.put(name, newa);
714             }
715         }
716     }
717 
getFastIntentCategories(Intent intent)718     private static FastImmutableArraySet<String> getFastIntentCategories(Intent intent) {
719         final Set<String> categories = intent.getCategories();
720         if (categories == null) {
721             return null;
722         }
723         return new FastImmutableArraySet<String>(categories.toArray(new String[categories.size()]));
724     }
725 
buildResolveList(Intent intent, FastImmutableArraySet<String> categories, boolean debug, boolean defaultOnly, String resolvedType, String scheme, F[] src, List<R> dest, int userId)726     private void buildResolveList(Intent intent, FastImmutableArraySet<String> categories,
727             boolean debug, boolean defaultOnly, String resolvedType, String scheme,
728             F[] src, List<R> dest, int userId) {
729         final String action = intent.getAction();
730         final Uri data = intent.getData();
731         final String packageName = intent.getPackage();
732 
733         final boolean excludingStopped = intent.isExcludingStopped();
734 
735         final Printer logPrinter;
736         final PrintWriter logPrintWriter;
737         if (debug) {
738             logPrinter = new LogPrinter(Log.VERBOSE, TAG, Log.LOG_ID_SYSTEM);
739             logPrintWriter = new FastPrintWriter(logPrinter);
740         } else {
741             logPrinter = null;
742             logPrintWriter = null;
743         }
744 
745         final int N = src != null ? src.length : 0;
746         boolean hasNonDefaults = false;
747         int i;
748         F filter;
749         for (i=0; i<N && (filter=src[i]) != null; i++) {
750             int match;
751             if (debug) Slog.v(TAG, "Matching against filter " + filter);
752 
753             if (excludingStopped && isFilterStopped(filter, userId)) {
754                 if (debug) {
755                     Slog.v(TAG, "  Filter's target is stopped; skipping");
756                 }
757                 continue;
758             }
759 
760             // Is delivery being limited to filters owned by a particular package?
761             if (packageName != null && !isPackageForFilter(packageName, filter)) {
762                 if (debug) {
763                     Slog.v(TAG, "  Filter is not from package " + packageName + "; skipping");
764                 }
765                 continue;
766             }
767 
768             // Are we verified ?
769             IntentFilter intentFilter = getIntentFilter(filter);
770             if (intentFilter.getAutoVerify()) {
771                 if (localVerificationLOGV || debug) {
772                     Slog.v(TAG, "  Filter verified: " + isFilterVerified(filter));
773                     int authorities = intentFilter.countDataAuthorities();
774                     for (int z = 0; z < authorities; z++) {
775                         Slog.v(TAG, "   " + intentFilter.getDataAuthority(z)
776                                 .getHost());
777                     }
778                 }
779             }
780 
781             // Do we already have this one?
782             if (!allowFilterResult(filter, dest)) {
783                 if (debug) {
784                     Slog.v(TAG, "  Filter's target already added");
785                 }
786                 continue;
787             }
788 
789             match = intentFilter.match(action, resolvedType, scheme, data, categories, TAG);
790             if (match >= 0) {
791                 if (debug) Slog.v(TAG, "  Filter matched!  match=0x" +
792                         Integer.toHexString(match) + " hasDefault="
793                         + intentFilter.hasCategory(Intent.CATEGORY_DEFAULT));
794                 if (!defaultOnly || intentFilter.hasCategory(Intent.CATEGORY_DEFAULT)) {
795                     final R oneResult = newResult(filter, match, userId);
796                     if (debug) Slog.v(TAG, "    Created result: " + oneResult);
797                     if (oneResult != null) {
798                         dest.add(oneResult);
799                         if (debug) {
800                             dumpFilter(logPrintWriter, "    ", filter);
801                             logPrintWriter.flush();
802                             intentFilter.dump(logPrinter, "    ");
803                         }
804                     }
805                 } else {
806                     hasNonDefaults = true;
807                 }
808             } else {
809                 if (debug) {
810                     String reason;
811                     switch (match) {
812                         case IntentFilter.NO_MATCH_ACTION: reason = "action"; break;
813                         case IntentFilter.NO_MATCH_CATEGORY: reason = "category"; break;
814                         case IntentFilter.NO_MATCH_DATA: reason = "data"; break;
815                         case IntentFilter.NO_MATCH_TYPE: reason = "type"; break;
816                         default: reason = "unknown reason"; break;
817                     }
818                     Slog.v(TAG, "  Filter did not match: " + reason);
819                 }
820             }
821         }
822 
823         if (debug && hasNonDefaults) {
824             if (dest.size() == 0) {
825                 Slog.v(TAG, "resolveIntent failed: found match, but none with CATEGORY_DEFAULT");
826             } else if (dest.size() > 1) {
827                 Slog.v(TAG, "resolveIntent: multiple matches, only some with CATEGORY_DEFAULT");
828             }
829         }
830     }
831 
832     // Sorts a List of IntentFilter objects into descending priority order.
833     @SuppressWarnings("rawtypes")
834     private static final Comparator mResolvePrioritySorter = new Comparator() {
835         public int compare(Object o1, Object o2) {
836             final int q1 = ((IntentFilter) o1).getPriority();
837             final int q2 = ((IntentFilter) o2).getPriority();
838             return (q1 > q2) ? -1 : ((q1 < q2) ? 1 : 0);
839         }
840     };
841 
842     /**
843      * All filters that have been registered.
844      */
845     protected final ArraySet<F> mFilters = new ArraySet<F>();
846 
847     /**
848      * All of the MIME types that have been registered, such as "image/jpeg",
849      * "image/*", or "{@literal *}/*".
850      */
851     private final ArrayMap<String, F[]> mTypeToFilter = new ArrayMap<String, F[]>();
852 
853     /**
854      * The base names of all of all fully qualified MIME types that have been
855      * registered, such as "image" or "*".  Wild card MIME types such as
856      * "image/*" will not be here.
857      */
858     private final ArrayMap<String, F[]> mBaseTypeToFilter = new ArrayMap<String, F[]>();
859 
860     /**
861      * The base names of all of the MIME types with a sub-type wildcard that
862      * have been registered.  For example, a filter with "image/*" will be
863      * included here as "image" but one with "image/jpeg" will not be
864      * included here.  This also includes the "*" for the "{@literal *}/*"
865      * MIME type.
866      */
867     private final ArrayMap<String, F[]> mWildTypeToFilter = new ArrayMap<String, F[]>();
868 
869     /**
870      * All of the URI schemes (such as http) that have been registered.
871      */
872     private final ArrayMap<String, F[]> mSchemeToFilter = new ArrayMap<String, F[]>();
873 
874     /**
875      * All of the actions that have been registered, but only those that did
876      * not specify data.
877      */
878     private final ArrayMap<String, F[]> mActionToFilter = new ArrayMap<String, F[]>();
879 
880     /**
881      * All of the actions that have been registered and specified a MIME type.
882      */
883     private final ArrayMap<String, F[]> mTypedActionToFilter = new ArrayMap<String, F[]>();
884 
885     /**
886      * Rather than refactoring the entire class, this allows the input {@link F} to be a type
887      * other than {@link IntentFilter}, transforming it whenever necessary. It is valid to use
888      * {@link IntentFilter} directly as {@link F} and just return {@param input}.
889      */
getIntentFilter(@onNull F input)890     protected abstract IntentFilter getIntentFilter(@NonNull F input);
891 }
892