1/*
2 * Copyright (C) 2012 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 java.io.FileDescriptor;
20import java.io.IOException;
21import java.io.PrintWriter;
22import java.util.ArrayList;
23import java.util.Collection;
24import java.util.HashMap;
25import java.util.HashSet;
26import java.util.Iterator;
27import java.util.List;
28
29import com.android.internal.os.BatteryStatsImpl;
30import com.android.server.am.ActivityManagerService.ItemMatcher;
31import com.android.server.am.ActivityManagerService.NeededUriGrants;
32
33import android.app.ActivityManager;
34import android.app.AppGlobals;
35import android.app.IApplicationThread;
36import android.app.IServiceConnection;
37import android.app.Notification;
38import android.app.PendingIntent;
39import android.app.Service;
40import android.content.ComponentName;
41import android.content.Context;
42import android.content.Intent;
43import android.content.pm.ApplicationInfo;
44import android.content.pm.PackageManager;
45import android.content.pm.ResolveInfo;
46import android.content.pm.ServiceInfo;
47import android.os.Binder;
48import android.os.IBinder;
49import android.os.Message;
50import android.os.Process;
51import android.os.RemoteException;
52import android.os.SystemClock;
53import android.os.UserHandle;
54import android.util.EventLog;
55import android.util.Log;
56import android.util.Slog;
57import android.util.SparseArray;
58import android.util.TimeUtils;
59
60public class ActiveServices {
61    static final boolean DEBUG_SERVICE = ActivityManagerService.DEBUG_SERVICE;
62    static final boolean DEBUG_SERVICE_EXECUTING = ActivityManagerService.DEBUG_SERVICE_EXECUTING;
63    static final boolean DEBUG_MU = ActivityManagerService.DEBUG_MU;
64    static final String TAG = ActivityManagerService.TAG;
65    static final String TAG_MU = ActivityManagerService.TAG_MU;
66
67    // How long we wait for a service to finish executing.
68    static final int SERVICE_TIMEOUT = 20*1000;
69
70    // How long a service needs to be running until restarting its process
71    // is no longer considered to be a relaunch of the service.
72    static final int SERVICE_RESTART_DURATION = 5*1000;
73
74    // How long a service needs to be running until it will start back at
75    // SERVICE_RESTART_DURATION after being killed.
76    static final int SERVICE_RESET_RUN_DURATION = 60*1000;
77
78    // Multiplying factor to increase restart duration time by, for each time
79    // a service is killed before it has run for SERVICE_RESET_RUN_DURATION.
80    static final int SERVICE_RESTART_DURATION_FACTOR = 4;
81
82    // The minimum amount of time between restarting services that we allow.
83    // That is, when multiple services are restarting, we won't allow each
84    // to restart less than this amount of time from the last one.
85    static final int SERVICE_MIN_RESTART_TIME_BETWEEN = 10*1000;
86
87    // Maximum amount of time for there to be no activity on a service before
88    // we consider it non-essential and allow its process to go on the
89    // LRU background list.
90    static final int MAX_SERVICE_INACTIVITY = 30*60*1000;
91
92    final ActivityManagerService mAm;
93
94    final ServiceMap mServiceMap = new ServiceMap();
95
96    /**
97     * All currently bound service connections.  Keys are the IBinder of
98     * the client's IServiceConnection.
99     */
100    final HashMap<IBinder, ArrayList<ConnectionRecord>> mServiceConnections
101            = new HashMap<IBinder, ArrayList<ConnectionRecord>>();
102
103    /**
104     * List of services that we have been asked to start,
105     * but haven't yet been able to.  It is used to hold start requests
106     * while waiting for their corresponding application thread to get
107     * going.
108     */
109    final ArrayList<ServiceRecord> mPendingServices
110            = new ArrayList<ServiceRecord>();
111
112    /**
113     * List of services that are scheduled to restart following a crash.
114     */
115    final ArrayList<ServiceRecord> mRestartingServices
116            = new ArrayList<ServiceRecord>();
117
118    /**
119     * List of services that are in the process of being stopped.
120     */
121    final ArrayList<ServiceRecord> mStoppingServices
122            = new ArrayList<ServiceRecord>();
123
124    static class ServiceMap {
125
126        private final SparseArray<HashMap<ComponentName, ServiceRecord>> mServicesByNamePerUser
127                = new SparseArray<HashMap<ComponentName, ServiceRecord>>();
128        private final SparseArray<HashMap<Intent.FilterComparison, ServiceRecord>>
129                mServicesByIntentPerUser = new SparseArray<
130                    HashMap<Intent.FilterComparison, ServiceRecord>>();
131
132        ServiceRecord getServiceByName(ComponentName name, int callingUser) {
133            // TODO: Deal with global services
134            if (DEBUG_MU)
135                Slog.v(TAG_MU, "getServiceByName(" + name + "), callingUser = " + callingUser);
136            return getServices(callingUser).get(name);
137        }
138
139        ServiceRecord getServiceByName(ComponentName name) {
140            return getServiceByName(name, -1);
141        }
142
143        ServiceRecord getServiceByIntent(Intent.FilterComparison filter, int callingUser) {
144            // TODO: Deal with global services
145            if (DEBUG_MU)
146                Slog.v(TAG_MU, "getServiceByIntent(" + filter + "), callingUser = " + callingUser);
147            return getServicesByIntent(callingUser).get(filter);
148        }
149
150        ServiceRecord getServiceByIntent(Intent.FilterComparison filter) {
151            return getServiceByIntent(filter, -1);
152        }
153
154        void putServiceByName(ComponentName name, int callingUser, ServiceRecord value) {
155            // TODO: Deal with global services
156            getServices(callingUser).put(name, value);
157        }
158
159        void putServiceByIntent(Intent.FilterComparison filter, int callingUser,
160                ServiceRecord value) {
161            // TODO: Deal with global services
162            getServicesByIntent(callingUser).put(filter, value);
163        }
164
165        void removeServiceByName(ComponentName name, int callingUser) {
166            // TODO: Deal with global services
167            ServiceRecord removed = getServices(callingUser).remove(name);
168            if (DEBUG_MU)
169                Slog.v(TAG, "removeServiceByName user=" + callingUser + " name=" + name
170                        + " removed=" + removed);
171        }
172
173        void removeServiceByIntent(Intent.FilterComparison filter, int callingUser) {
174            // TODO: Deal with global services
175            ServiceRecord removed = getServicesByIntent(callingUser).remove(filter);
176            if (DEBUG_MU)
177                Slog.v(TAG_MU, "removeServiceByIntent user=" + callingUser + " intent=" + filter
178                        + " removed=" + removed);
179        }
180
181        Collection<ServiceRecord> getAllServices(int callingUser) {
182            // TODO: Deal with global services
183            return getServices(callingUser).values();
184        }
185
186        private HashMap<ComponentName, ServiceRecord> getServices(int callingUser) {
187            HashMap<ComponentName, ServiceRecord> map = mServicesByNamePerUser.get(callingUser);
188            if (map == null) {
189                map = new HashMap<ComponentName, ServiceRecord>();
190                mServicesByNamePerUser.put(callingUser, map);
191            }
192            return map;
193        }
194
195        private HashMap<Intent.FilterComparison, ServiceRecord> getServicesByIntent(
196                int callingUser) {
197            HashMap<Intent.FilterComparison, ServiceRecord> map
198                    = mServicesByIntentPerUser.get(callingUser);
199            if (map == null) {
200                map = new HashMap<Intent.FilterComparison, ServiceRecord>();
201                mServicesByIntentPerUser.put(callingUser, map);
202            }
203            return map;
204        }
205    }
206
207    public ActiveServices(ActivityManagerService service) {
208        mAm = service;
209    }
210
211    ComponentName startServiceLocked(IApplicationThread caller,
212            Intent service, String resolvedType,
213            int callingPid, int callingUid, int userId) {
214        if (DEBUG_SERVICE) Slog.v(TAG, "startService: " + service
215                + " type=" + resolvedType + " args=" + service.getExtras());
216
217        if (caller != null) {
218            final ProcessRecord callerApp = mAm.getRecordForAppLocked(caller);
219            if (callerApp == null) {
220                throw new SecurityException(
221                        "Unable to find app for caller " + caller
222                        + " (pid=" + Binder.getCallingPid()
223                        + ") when starting service " + service);
224            }
225        }
226
227        ServiceLookupResult res =
228            retrieveServiceLocked(service, resolvedType,
229                    callingPid, callingUid, userId, true);
230        if (res == null) {
231            return null;
232        }
233        if (res.record == null) {
234            return new ComponentName("!", res.permission != null
235                    ? res.permission : "private to package");
236        }
237        ServiceRecord r = res.record;
238        NeededUriGrants neededGrants = mAm.checkGrantUriPermissionFromIntentLocked(
239                callingUid, r.packageName, service, service.getFlags(), null);
240        if (unscheduleServiceRestartLocked(r)) {
241            if (DEBUG_SERVICE) Slog.v(TAG, "START SERVICE WHILE RESTART PENDING: " + r);
242        }
243        r.startRequested = true;
244        r.callStart = false;
245        r.pendingStarts.add(new ServiceRecord.StartItem(r, false, r.makeNextStartId(),
246                service, neededGrants));
247        r.lastActivity = SystemClock.uptimeMillis();
248        synchronized (r.stats.getBatteryStats()) {
249            r.stats.startRunningLocked();
250        }
251        String error = bringUpServiceLocked(r, service.getFlags(), false);
252        if (error != null) {
253            return new ComponentName("!!", error);
254        }
255        return r.name;
256    }
257
258    private void stopServiceLocked(ServiceRecord service) {
259        synchronized (service.stats.getBatteryStats()) {
260            service.stats.stopRunningLocked();
261        }
262        service.startRequested = false;
263        service.callStart = false;
264        bringDownServiceLocked(service, false);
265    }
266
267    int stopServiceLocked(IApplicationThread caller, Intent service,
268            String resolvedType, int userId) {
269        if (DEBUG_SERVICE) Slog.v(TAG, "stopService: " + service
270                + " type=" + resolvedType);
271
272        final ProcessRecord callerApp = mAm.getRecordForAppLocked(caller);
273        if (caller != null && callerApp == null) {
274            throw new SecurityException(
275                    "Unable to find app for caller " + caller
276                    + " (pid=" + Binder.getCallingPid()
277                    + ") when stopping service " + service);
278        }
279
280        // If this service is active, make sure it is stopped.
281        ServiceLookupResult r = retrieveServiceLocked(service, resolvedType,
282                Binder.getCallingPid(), Binder.getCallingUid(), userId, false);
283        if (r != null) {
284            if (r.record != null) {
285                final long origId = Binder.clearCallingIdentity();
286                try {
287                    stopServiceLocked(r.record);
288                } finally {
289                    Binder.restoreCallingIdentity(origId);
290                }
291                return 1;
292            }
293            return -1;
294        }
295
296        return 0;
297    }
298
299    IBinder peekServiceLocked(Intent service, String resolvedType) {
300        ServiceLookupResult r = retrieveServiceLocked(service, resolvedType,
301                Binder.getCallingPid(), Binder.getCallingUid(),
302                UserHandle.getCallingUserId(), false);
303
304        IBinder ret = null;
305        if (r != null) {
306            // r.record is null if findServiceLocked() failed the caller permission check
307            if (r.record == null) {
308                throw new SecurityException(
309                        "Permission Denial: Accessing service " + r.record.name
310                        + " from pid=" + Binder.getCallingPid()
311                        + ", uid=" + Binder.getCallingUid()
312                        + " requires " + r.permission);
313            }
314            IntentBindRecord ib = r.record.bindings.get(r.record.intent);
315            if (ib != null) {
316                ret = ib.binder;
317            }
318        }
319
320        return ret;
321    }
322
323    boolean stopServiceTokenLocked(ComponentName className, IBinder token,
324            int startId) {
325        if (DEBUG_SERVICE) Slog.v(TAG, "stopServiceToken: " + className
326                + " " + token + " startId=" + startId);
327        ServiceRecord r = findServiceLocked(className, token, UserHandle.getCallingUserId());
328        if (r != null) {
329            if (startId >= 0) {
330                // Asked to only stop if done with all work.  Note that
331                // to avoid leaks, we will take this as dropping all
332                // start items up to and including this one.
333                ServiceRecord.StartItem si = r.findDeliveredStart(startId, false);
334                if (si != null) {
335                    while (r.deliveredStarts.size() > 0) {
336                        ServiceRecord.StartItem cur = r.deliveredStarts.remove(0);
337                        cur.removeUriPermissionsLocked();
338                        if (cur == si) {
339                            break;
340                        }
341                    }
342                }
343
344                if (r.getLastStartId() != startId) {
345                    return false;
346                }
347
348                if (r.deliveredStarts.size() > 0) {
349                    Slog.w(TAG, "stopServiceToken startId " + startId
350                            + " is last, but have " + r.deliveredStarts.size()
351                            + " remaining args");
352                }
353            }
354
355            synchronized (r.stats.getBatteryStats()) {
356                r.stats.stopRunningLocked();
357                r.startRequested = false;
358                r.callStart = false;
359            }
360            final long origId = Binder.clearCallingIdentity();
361            bringDownServiceLocked(r, false);
362            Binder.restoreCallingIdentity(origId);
363            return true;
364        }
365        return false;
366    }
367
368    public void setServiceForegroundLocked(ComponentName className, IBinder token,
369            int id, Notification notification, boolean removeNotification) {
370        final int userId = UserHandle.getCallingUserId();
371        final long origId = Binder.clearCallingIdentity();
372        try {
373            ServiceRecord r = findServiceLocked(className, token, userId);
374            if (r != null) {
375                if (id != 0) {
376                    if (notification == null) {
377                        throw new IllegalArgumentException("null notification");
378                    }
379                    if (r.foregroundId != id) {
380                        r.cancelNotification();
381                        r.foregroundId = id;
382                    }
383                    notification.flags |= Notification.FLAG_FOREGROUND_SERVICE;
384                    r.foregroundNoti = notification;
385                    r.isForeground = true;
386                    r.postNotification();
387                    if (r.app != null) {
388                        updateServiceForegroundLocked(r.app, true);
389                    }
390                } else {
391                    if (r.isForeground) {
392                        r.isForeground = false;
393                        if (r.app != null) {
394                            mAm.updateLruProcessLocked(r.app, false);
395                            updateServiceForegroundLocked(r.app, true);
396                        }
397                    }
398                    if (removeNotification) {
399                        r.cancelNotification();
400                        r.foregroundId = 0;
401                        r.foregroundNoti = null;
402                    }
403                }
404            }
405        } finally {
406            Binder.restoreCallingIdentity(origId);
407        }
408    }
409
410    private void updateServiceForegroundLocked(ProcessRecord proc, boolean oomAdj) {
411        boolean anyForeground = false;
412        for (ServiceRecord sr : proc.services) {
413            if (sr.isForeground) {
414                anyForeground = true;
415                break;
416            }
417        }
418        if (anyForeground != proc.foregroundServices) {
419            proc.foregroundServices = anyForeground;
420            if (oomAdj) {
421                mAm.updateOomAdjLocked();
422            }
423        }
424    }
425
426    int bindServiceLocked(IApplicationThread caller, IBinder token,
427            Intent service, String resolvedType,
428            IServiceConnection connection, int flags, int userId) {
429        if (DEBUG_SERVICE) Slog.v(TAG, "bindService: " + service
430                + " type=" + resolvedType + " conn=" + connection.asBinder()
431                + " flags=0x" + Integer.toHexString(flags));
432        final ProcessRecord callerApp = mAm.getRecordForAppLocked(caller);
433        if (callerApp == null) {
434            throw new SecurityException(
435                    "Unable to find app for caller " + caller
436                    + " (pid=" + Binder.getCallingPid()
437                    + ") when binding service " + service);
438        }
439
440        ActivityRecord activity = null;
441        if (token != null) {
442            activity = mAm.mMainStack.isInStackLocked(token);
443            if (activity == null) {
444                Slog.w(TAG, "Binding with unknown activity: " + token);
445                return 0;
446            }
447        }
448
449        int clientLabel = 0;
450        PendingIntent clientIntent = null;
451
452        if (callerApp.info.uid == Process.SYSTEM_UID) {
453            // Hacky kind of thing -- allow system stuff to tell us
454            // what they are, so we can report this elsewhere for
455            // others to know why certain services are running.
456            try {
457                clientIntent = (PendingIntent)service.getParcelableExtra(
458                        Intent.EXTRA_CLIENT_INTENT);
459            } catch (RuntimeException e) {
460            }
461            if (clientIntent != null) {
462                clientLabel = service.getIntExtra(Intent.EXTRA_CLIENT_LABEL, 0);
463                if (clientLabel != 0) {
464                    // There are no useful extras in the intent, trash them.
465                    // System code calling with this stuff just needs to know
466                    // this will happen.
467                    service = service.cloneFilter();
468                }
469            }
470        }
471
472        ServiceLookupResult res =
473            retrieveServiceLocked(service, resolvedType,
474                    Binder.getCallingPid(), Binder.getCallingUid(), userId, true);
475        if (res == null) {
476            return 0;
477        }
478        if (res.record == null) {
479            return -1;
480        }
481        ServiceRecord s = res.record;
482
483        final long origId = Binder.clearCallingIdentity();
484
485        try {
486            if (unscheduleServiceRestartLocked(s)) {
487                if (DEBUG_SERVICE) Slog.v(TAG, "BIND SERVICE WHILE RESTART PENDING: "
488                        + s);
489            }
490
491            AppBindRecord b = s.retrieveAppBindingLocked(service, callerApp);
492            ConnectionRecord c = new ConnectionRecord(b, activity,
493                    connection, flags, clientLabel, clientIntent);
494
495            IBinder binder = connection.asBinder();
496            ArrayList<ConnectionRecord> clist = s.connections.get(binder);
497            if (clist == null) {
498                clist = new ArrayList<ConnectionRecord>();
499                s.connections.put(binder, clist);
500            }
501            clist.add(c);
502            b.connections.add(c);
503            if (activity != null) {
504                if (activity.connections == null) {
505                    activity.connections = new HashSet<ConnectionRecord>();
506                }
507                activity.connections.add(c);
508            }
509            b.client.connections.add(c);
510            if ((c.flags&Context.BIND_ABOVE_CLIENT) != 0) {
511                b.client.hasAboveClient = true;
512            }
513            clist = mServiceConnections.get(binder);
514            if (clist == null) {
515                clist = new ArrayList<ConnectionRecord>();
516                mServiceConnections.put(binder, clist);
517            }
518            clist.add(c);
519
520            if ((flags&Context.BIND_AUTO_CREATE) != 0) {
521                s.lastActivity = SystemClock.uptimeMillis();
522                if (bringUpServiceLocked(s, service.getFlags(), false) != null) {
523                    return 0;
524                }
525            }
526
527            if (s.app != null) {
528                // This could have made the service more important.
529                mAm.updateOomAdjLocked(s.app);
530            }
531
532            if (DEBUG_SERVICE) Slog.v(TAG, "Bind " + s + " with " + b
533                    + ": received=" + b.intent.received
534                    + " apps=" + b.intent.apps.size()
535                    + " doRebind=" + b.intent.doRebind);
536
537            if (s.app != null && b.intent.received) {
538                // Service is already running, so we can immediately
539                // publish the connection.
540                try {
541                    c.conn.connected(s.name, b.intent.binder);
542                } catch (Exception e) {
543                    Slog.w(TAG, "Failure sending service " + s.shortName
544                            + " to connection " + c.conn.asBinder()
545                            + " (in " + c.binding.client.processName + ")", e);
546                }
547
548                // If this is the first app connected back to this binding,
549                // and the service had previously asked to be told when
550                // rebound, then do so.
551                if (b.intent.apps.size() == 1 && b.intent.doRebind) {
552                    requestServiceBindingLocked(s, b.intent, true);
553                }
554            } else if (!b.intent.requested) {
555                requestServiceBindingLocked(s, b.intent, false);
556            }
557        } finally {
558            Binder.restoreCallingIdentity(origId);
559        }
560
561        return 1;
562    }
563
564    void publishServiceLocked(ServiceRecord r, Intent intent, IBinder service) {
565        final long origId = Binder.clearCallingIdentity();
566        try {
567            if (DEBUG_SERVICE) Slog.v(TAG, "PUBLISHING " + r
568                    + " " + intent + ": " + service);
569            if (r != null) {
570                Intent.FilterComparison filter
571                        = new Intent.FilterComparison(intent);
572                IntentBindRecord b = r.bindings.get(filter);
573                if (b != null && !b.received) {
574                    b.binder = service;
575                    b.requested = true;
576                    b.received = true;
577                    if (r.connections.size() > 0) {
578                        Iterator<ArrayList<ConnectionRecord>> it
579                                = r.connections.values().iterator();
580                        while (it.hasNext()) {
581                            ArrayList<ConnectionRecord> clist = it.next();
582                            for (int i=0; i<clist.size(); i++) {
583                                ConnectionRecord c = clist.get(i);
584                                if (!filter.equals(c.binding.intent.intent)) {
585                                    if (DEBUG_SERVICE) Slog.v(
586                                            TAG, "Not publishing to: " + c);
587                                    if (DEBUG_SERVICE) Slog.v(
588                                            TAG, "Bound intent: " + c.binding.intent.intent);
589                                    if (DEBUG_SERVICE) Slog.v(
590                                            TAG, "Published intent: " + intent);
591                                    continue;
592                                }
593                                if (DEBUG_SERVICE) Slog.v(TAG, "Publishing to: " + c);
594                                try {
595                                    c.conn.connected(r.name, service);
596                                } catch (Exception e) {
597                                    Slog.w(TAG, "Failure sending service " + r.name +
598                                          " to connection " + c.conn.asBinder() +
599                                          " (in " + c.binding.client.processName + ")", e);
600                                }
601                            }
602                        }
603                    }
604                }
605
606                serviceDoneExecutingLocked(r, mStoppingServices.contains(r));
607            }
608        } finally {
609            Binder.restoreCallingIdentity(origId);
610        }
611    }
612
613    boolean unbindServiceLocked(IServiceConnection connection) {
614        IBinder binder = connection.asBinder();
615        if (DEBUG_SERVICE) Slog.v(TAG, "unbindService: conn=" + binder);
616        ArrayList<ConnectionRecord> clist = mServiceConnections.get(binder);
617        if (clist == null) {
618            Slog.w(TAG, "Unbind failed: could not find connection for "
619                  + connection.asBinder());
620            return false;
621        }
622
623        final long origId = Binder.clearCallingIdentity();
624        try {
625            while (clist.size() > 0) {
626                ConnectionRecord r = clist.get(0);
627                removeConnectionLocked(r, null, null);
628
629                if (r.binding.service.app != null) {
630                    // This could have made the service less important.
631                    mAm.updateOomAdjLocked(r.binding.service.app);
632                }
633            }
634        } finally {
635            Binder.restoreCallingIdentity(origId);
636        }
637
638        return true;
639    }
640
641    void unbindFinishedLocked(ServiceRecord r, Intent intent, boolean doRebind) {
642        final long origId = Binder.clearCallingIdentity();
643        try {
644            if (r != null) {
645                Intent.FilterComparison filter
646                        = new Intent.FilterComparison(intent);
647                IntentBindRecord b = r.bindings.get(filter);
648                if (DEBUG_SERVICE) Slog.v(TAG, "unbindFinished in " + r
649                        + " at " + b + ": apps="
650                        + (b != null ? b.apps.size() : 0));
651
652                boolean inStopping = mStoppingServices.contains(r);
653                if (b != null) {
654                    if (b.apps.size() > 0 && !inStopping) {
655                        // Applications have already bound since the last
656                        // unbind, so just rebind right here.
657                        requestServiceBindingLocked(r, b, true);
658                    } else {
659                        // Note to tell the service the next time there is
660                        // a new client.
661                        b.doRebind = true;
662                    }
663                }
664
665                serviceDoneExecutingLocked(r, inStopping);
666            }
667        } finally {
668            Binder.restoreCallingIdentity(origId);
669        }
670    }
671
672    private final ServiceRecord findServiceLocked(ComponentName name,
673            IBinder token, int userId) {
674        ServiceRecord r = mServiceMap.getServiceByName(name, userId);
675        return r == token ? r : null;
676    }
677
678    private final class ServiceLookupResult {
679        final ServiceRecord record;
680        final String permission;
681
682        ServiceLookupResult(ServiceRecord _record, String _permission) {
683            record = _record;
684            permission = _permission;
685        }
686    }
687
688    private class ServiceRestarter implements Runnable {
689        private ServiceRecord mService;
690
691        void setService(ServiceRecord service) {
692            mService = service;
693        }
694
695        public void run() {
696            synchronized(mAm) {
697                performServiceRestartLocked(mService);
698            }
699        }
700    }
701
702    private ServiceLookupResult retrieveServiceLocked(Intent service,
703            String resolvedType, int callingPid, int callingUid, int userId,
704            boolean createIfNeeded) {
705        ServiceRecord r = null;
706        if (DEBUG_SERVICE) Slog.v(TAG, "retrieveServiceLocked: " + service
707                + " type=" + resolvedType + " callingUid=" + callingUid);
708
709        userId = mAm.handleIncomingUser(callingPid, callingUid, userId,
710                false, true, "service", null);
711
712        if (service.getComponent() != null) {
713            r = mServiceMap.getServiceByName(service.getComponent(), userId);
714        }
715        if (r == null) {
716            Intent.FilterComparison filter = new Intent.FilterComparison(service);
717            r = mServiceMap.getServiceByIntent(filter, userId);
718        }
719        if (r == null) {
720            try {
721                ResolveInfo rInfo =
722                    AppGlobals.getPackageManager().resolveService(
723                                service, resolvedType,
724                                ActivityManagerService.STOCK_PM_FLAGS, userId);
725                ServiceInfo sInfo =
726                    rInfo != null ? rInfo.serviceInfo : null;
727                if (sInfo == null) {
728                    Slog.w(TAG, "Unable to start service " + service + " U=" + userId +
729                          ": not found");
730                    return null;
731                }
732                ComponentName name = new ComponentName(
733                        sInfo.applicationInfo.packageName, sInfo.name);
734                if (userId > 0) {
735                    if (mAm.isSingleton(sInfo.processName, sInfo.applicationInfo,
736                            sInfo.name, sInfo.flags)) {
737                        userId = 0;
738                    }
739                    sInfo = new ServiceInfo(sInfo);
740                    sInfo.applicationInfo = mAm.getAppInfoForUser(sInfo.applicationInfo, userId);
741                }
742                r = mServiceMap.getServiceByName(name, userId);
743                if (r == null && createIfNeeded) {
744                    Intent.FilterComparison filter = new Intent.FilterComparison(
745                            service.cloneFilter());
746                    ServiceRestarter res = new ServiceRestarter();
747                    BatteryStatsImpl.Uid.Pkg.Serv ss = null;
748                    BatteryStatsImpl stats = mAm.mBatteryStatsService.getActiveStatistics();
749                    synchronized (stats) {
750                        ss = stats.getServiceStatsLocked(
751                                sInfo.applicationInfo.uid, sInfo.packageName,
752                                sInfo.name);
753                    }
754                    r = new ServiceRecord(mAm, ss, name, filter, sInfo, res);
755                    res.setService(r);
756                    mServiceMap.putServiceByName(name, UserHandle.getUserId(r.appInfo.uid), r);
757                    mServiceMap.putServiceByIntent(filter, UserHandle.getUserId(r.appInfo.uid), r);
758
759                    // Make sure this component isn't in the pending list.
760                    int N = mPendingServices.size();
761                    for (int i=0; i<N; i++) {
762                        ServiceRecord pr = mPendingServices.get(i);
763                        if (pr.serviceInfo.applicationInfo.uid == sInfo.applicationInfo.uid
764                                && pr.name.equals(name)) {
765                            mPendingServices.remove(i);
766                            i--;
767                            N--;
768                        }
769                    }
770                }
771            } catch (RemoteException ex) {
772                // pm is in same process, this will never happen.
773            }
774        }
775        if (r != null) {
776            if (mAm.checkComponentPermission(r.permission,
777                    callingPid, callingUid, r.appInfo.uid, r.exported)
778                    != PackageManager.PERMISSION_GRANTED) {
779                if (!r.exported) {
780                    Slog.w(TAG, "Permission Denial: Accessing service " + r.name
781                            + " from pid=" + callingPid
782                            + ", uid=" + callingUid
783                            + " that is not exported from uid " + r.appInfo.uid);
784                    return new ServiceLookupResult(null, "not exported from uid "
785                            + r.appInfo.uid);
786                }
787                Slog.w(TAG, "Permission Denial: Accessing service " + r.name
788                        + " from pid=" + callingPid
789                        + ", uid=" + callingUid
790                        + " requires " + r.permission);
791                return new ServiceLookupResult(null, r.permission);
792            }
793            return new ServiceLookupResult(r, null);
794        }
795        return null;
796    }
797
798    private final void bumpServiceExecutingLocked(ServiceRecord r, String why) {
799        if (DEBUG_SERVICE) Log.v(TAG, ">>> EXECUTING "
800                + why + " of " + r + " in app " + r.app);
801        else if (DEBUG_SERVICE_EXECUTING) Log.v(TAG, ">>> EXECUTING "
802                + why + " of " + r.shortName);
803        long now = SystemClock.uptimeMillis();
804        if (r.executeNesting == 0 && r.app != null) {
805            if (r.app.executingServices.size() == 0) {
806                Message msg = mAm.mHandler.obtainMessage(
807                        ActivityManagerService.SERVICE_TIMEOUT_MSG);
808                msg.obj = r.app;
809                mAm.mHandler.sendMessageAtTime(msg, now+SERVICE_TIMEOUT);
810            }
811            r.app.executingServices.add(r);
812        }
813        r.executeNesting++;
814        r.executingStart = now;
815    }
816
817    private final boolean requestServiceBindingLocked(ServiceRecord r,
818            IntentBindRecord i, boolean rebind) {
819        if (r.app == null || r.app.thread == null) {
820            // If service is not currently running, can't yet bind.
821            return false;
822        }
823        if ((!i.requested || rebind) && i.apps.size() > 0) {
824            try {
825                bumpServiceExecutingLocked(r, "bind");
826                r.app.thread.scheduleBindService(r, i.intent.getIntent(), rebind);
827                if (!rebind) {
828                    i.requested = true;
829                }
830                i.hasBound = true;
831                i.doRebind = false;
832            } catch (RemoteException e) {
833                if (DEBUG_SERVICE) Slog.v(TAG, "Crashed while binding " + r);
834                return false;
835            }
836        }
837        return true;
838    }
839
840    private final boolean scheduleServiceRestartLocked(ServiceRecord r,
841            boolean allowCancel) {
842        boolean canceled = false;
843
844        final long now = SystemClock.uptimeMillis();
845
846        if ((r.serviceInfo.applicationInfo.flags
847                &ApplicationInfo.FLAG_PERSISTENT) == 0) {
848            long minDuration = SERVICE_RESTART_DURATION;
849            long resetTime = SERVICE_RESET_RUN_DURATION;
850
851            // Any delivered but not yet finished starts should be put back
852            // on the pending list.
853            final int N = r.deliveredStarts.size();
854            if (N > 0) {
855                for (int i=N-1; i>=0; i--) {
856                    ServiceRecord.StartItem si = r.deliveredStarts.get(i);
857                    si.removeUriPermissionsLocked();
858                    if (si.intent == null) {
859                        // We'll generate this again if needed.
860                    } else if (!allowCancel || (si.deliveryCount < ServiceRecord.MAX_DELIVERY_COUNT
861                            && si.doneExecutingCount < ServiceRecord.MAX_DONE_EXECUTING_COUNT)) {
862                        r.pendingStarts.add(0, si);
863                        long dur = SystemClock.uptimeMillis() - si.deliveredTime;
864                        dur *= 2;
865                        if (minDuration < dur) minDuration = dur;
866                        if (resetTime < dur) resetTime = dur;
867                    } else {
868                        Slog.w(TAG, "Canceling start item " + si.intent + " in service "
869                                + r.name);
870                        canceled = true;
871                    }
872                }
873                r.deliveredStarts.clear();
874            }
875
876            r.totalRestartCount++;
877            if (r.restartDelay == 0) {
878                r.restartCount++;
879                r.restartDelay = minDuration;
880            } else {
881                // If it has been a "reasonably long time" since the service
882                // was started, then reset our restart duration back to
883                // the beginning, so we don't infinitely increase the duration
884                // on a service that just occasionally gets killed (which is
885                // a normal case, due to process being killed to reclaim memory).
886                if (now > (r.restartTime+resetTime)) {
887                    r.restartCount = 1;
888                    r.restartDelay = minDuration;
889                } else {
890                    if ((r.serviceInfo.applicationInfo.flags
891                            &ApplicationInfo.FLAG_PERSISTENT) != 0) {
892                        // Services in peristent processes will restart much more
893                        // quickly, since they are pretty important.  (Think SystemUI).
894                        r.restartDelay += minDuration/2;
895                    } else {
896                        r.restartDelay *= SERVICE_RESTART_DURATION_FACTOR;
897                        if (r.restartDelay < minDuration) {
898                            r.restartDelay = minDuration;
899                        }
900                    }
901                }
902            }
903
904            r.nextRestartTime = now + r.restartDelay;
905
906            // Make sure that we don't end up restarting a bunch of services
907            // all at the same time.
908            boolean repeat;
909            do {
910                repeat = false;
911                for (int i=mRestartingServices.size()-1; i>=0; i--) {
912                    ServiceRecord r2 = mRestartingServices.get(i);
913                    if (r2 != r && r.nextRestartTime
914                            >= (r2.nextRestartTime-SERVICE_MIN_RESTART_TIME_BETWEEN)
915                            && r.nextRestartTime
916                            < (r2.nextRestartTime+SERVICE_MIN_RESTART_TIME_BETWEEN)) {
917                        r.nextRestartTime = r2.nextRestartTime + SERVICE_MIN_RESTART_TIME_BETWEEN;
918                        r.restartDelay = r.nextRestartTime - now;
919                        repeat = true;
920                        break;
921                    }
922                }
923            } while (repeat);
924
925        } else {
926            // Persistent processes are immediately restrted, so there is no
927            // reason to hold of on restarting their services.
928            r.totalRestartCount++;
929            r.restartCount = 0;
930            r.restartDelay = 0;
931            r.nextRestartTime = now;
932        }
933
934        if (!mRestartingServices.contains(r)) {
935            mRestartingServices.add(r);
936        }
937
938        r.cancelNotification();
939
940        mAm.mHandler.removeCallbacks(r.restarter);
941        mAm.mHandler.postAtTime(r.restarter, r.nextRestartTime);
942        r.nextRestartTime = SystemClock.uptimeMillis() + r.restartDelay;
943        Slog.w(TAG, "Scheduling restart of crashed service "
944                + r.shortName + " in " + r.restartDelay + "ms");
945        EventLog.writeEvent(EventLogTags.AM_SCHEDULE_SERVICE_RESTART,
946                r.userId, r.shortName, r.restartDelay);
947
948        return canceled;
949    }
950
951    final void performServiceRestartLocked(ServiceRecord r) {
952        if (!mRestartingServices.contains(r)) {
953            return;
954        }
955        bringUpServiceLocked(r, r.intent.getIntent().getFlags(), true);
956    }
957
958    private final boolean unscheduleServiceRestartLocked(ServiceRecord r) {
959        if (r.restartDelay == 0) {
960            return false;
961        }
962        r.resetRestartCounter();
963        mRestartingServices.remove(r);
964        mAm.mHandler.removeCallbacks(r.restarter);
965        return true;
966    }
967
968    private final String bringUpServiceLocked(ServiceRecord r,
969            int intentFlags, boolean whileRestarting) {
970        //Slog.i(TAG, "Bring up service:");
971        //r.dump("  ");
972
973        if (r.app != null && r.app.thread != null) {
974            sendServiceArgsLocked(r, false);
975            return null;
976        }
977
978        if (!whileRestarting && r.restartDelay > 0) {
979            // If waiting for a restart, then do nothing.
980            return null;
981        }
982
983        if (DEBUG_SERVICE) Slog.v(TAG, "Bringing up " + r + " " + r.intent);
984
985        // We are now bringing the service up, so no longer in the
986        // restarting state.
987        mRestartingServices.remove(r);
988
989        // Make sure that the user who owns this service is started.  If not,
990        // we don't want to allow it to run.
991        if (mAm.mStartedUsers.get(r.userId) == null) {
992            String msg = "Unable to launch app "
993                    + r.appInfo.packageName + "/"
994                    + r.appInfo.uid + " for service "
995                    + r.intent.getIntent() + ": user " + r.userId + " is stopped";
996            Slog.w(TAG, msg);
997            bringDownServiceLocked(r, true);
998            return msg;
999        }
1000
1001        // Service is now being launched, its package can't be stopped.
1002        try {
1003            AppGlobals.getPackageManager().setPackageStoppedState(
1004                    r.packageName, false, r.userId);
1005        } catch (RemoteException e) {
1006        } catch (IllegalArgumentException e) {
1007            Slog.w(TAG, "Failed trying to unstop package "
1008                    + r.packageName + ": " + e);
1009        }
1010
1011        final boolean isolated = (r.serviceInfo.flags&ServiceInfo.FLAG_ISOLATED_PROCESS) != 0;
1012        final String procName = r.processName;
1013        ProcessRecord app;
1014
1015        if (!isolated) {
1016            app = mAm.getProcessRecordLocked(procName, r.appInfo.uid);
1017            if (DEBUG_MU)
1018                Slog.v(TAG_MU, "bringUpServiceLocked: appInfo.uid=" + r.appInfo.uid + " app=" + app);
1019            if (app != null && app.thread != null) {
1020                try {
1021                    app.addPackage(r.appInfo.packageName);
1022                    realStartServiceLocked(r, app);
1023                    return null;
1024                } catch (RemoteException e) {
1025                    Slog.w(TAG, "Exception when starting service " + r.shortName, e);
1026                }
1027
1028                // If a dead object exception was thrown -- fall through to
1029                // restart the application.
1030            }
1031        } else {
1032            // If this service runs in an isolated process, then each time
1033            // we call startProcessLocked() we will get a new isolated
1034            // process, starting another process if we are currently waiting
1035            // for a previous process to come up.  To deal with this, we store
1036            // in the service any current isolated process it is running in or
1037            // waiting to have come up.
1038            app = r.isolatedProc;
1039        }
1040
1041        // Not running -- get it started, and enqueue this service record
1042        // to be executed when the app comes up.
1043        if (app == null) {
1044            if ((app=mAm.startProcessLocked(procName, r.appInfo, true, intentFlags,
1045                    "service", r.name, false, isolated)) == null) {
1046                String msg = "Unable to launch app "
1047                        + r.appInfo.packageName + "/"
1048                        + r.appInfo.uid + " for service "
1049                        + r.intent.getIntent() + ": process is bad";
1050                Slog.w(TAG, msg);
1051                bringDownServiceLocked(r, true);
1052                return msg;
1053            }
1054            if (isolated) {
1055                r.isolatedProc = app;
1056            }
1057        }
1058
1059        if (!mPendingServices.contains(r)) {
1060            mPendingServices.add(r);
1061        }
1062
1063        return null;
1064    }
1065
1066    private final void requestServiceBindingsLocked(ServiceRecord r) {
1067        Iterator<IntentBindRecord> bindings = r.bindings.values().iterator();
1068        while (bindings.hasNext()) {
1069            IntentBindRecord i = bindings.next();
1070            if (!requestServiceBindingLocked(r, i, false)) {
1071                break;
1072            }
1073        }
1074    }
1075
1076    private final void realStartServiceLocked(ServiceRecord r,
1077            ProcessRecord app) throws RemoteException {
1078        if (app.thread == null) {
1079            throw new RemoteException();
1080        }
1081        if (DEBUG_MU)
1082            Slog.v(TAG_MU, "realStartServiceLocked, ServiceRecord.uid = " + r.appInfo.uid
1083                    + ", ProcessRecord.uid = " + app.uid);
1084        r.app = app;
1085        r.restartTime = r.lastActivity = SystemClock.uptimeMillis();
1086
1087        app.services.add(r);
1088        bumpServiceExecutingLocked(r, "create");
1089        mAm.updateLruProcessLocked(app, true);
1090
1091        boolean created = false;
1092        try {
1093            mAm.mStringBuilder.setLength(0);
1094            r.intent.getIntent().toShortString(mAm.mStringBuilder, true, false, true, false);
1095            EventLog.writeEvent(EventLogTags.AM_CREATE_SERVICE,
1096                    r.userId, System.identityHashCode(r), r.shortName,
1097                    mAm.mStringBuilder.toString(), r.app.pid);
1098            synchronized (r.stats.getBatteryStats()) {
1099                r.stats.startLaunchedLocked();
1100            }
1101            mAm.ensurePackageDexOpt(r.serviceInfo.packageName);
1102            app.thread.scheduleCreateService(r, r.serviceInfo,
1103                    mAm.compatibilityInfoForPackageLocked(r.serviceInfo.applicationInfo));
1104            r.postNotification();
1105            created = true;
1106        } finally {
1107            if (!created) {
1108                app.services.remove(r);
1109                scheduleServiceRestartLocked(r, false);
1110            }
1111        }
1112
1113        requestServiceBindingsLocked(r);
1114
1115        // If the service is in the started state, and there are no
1116        // pending arguments, then fake up one so its onStartCommand() will
1117        // be called.
1118        if (r.startRequested && r.callStart && r.pendingStarts.size() == 0) {
1119            r.pendingStarts.add(new ServiceRecord.StartItem(r, false, r.makeNextStartId(),
1120                    null, null));
1121        }
1122
1123        sendServiceArgsLocked(r, true);
1124    }
1125
1126    private final void sendServiceArgsLocked(ServiceRecord r,
1127            boolean oomAdjusted) {
1128        final int N = r.pendingStarts.size();
1129        if (N == 0) {
1130            return;
1131        }
1132
1133        while (r.pendingStarts.size() > 0) {
1134            try {
1135                ServiceRecord.StartItem si = r.pendingStarts.remove(0);
1136                if (DEBUG_SERVICE) Slog.v(TAG, "Sending arguments to: "
1137                        + r + " " + r.intent + " args=" + si.intent);
1138                if (si.intent == null && N > 1) {
1139                    // If somehow we got a dummy null intent in the middle,
1140                    // then skip it.  DO NOT skip a null intent when it is
1141                    // the only one in the list -- this is to support the
1142                    // onStartCommand(null) case.
1143                    continue;
1144                }
1145                si.deliveredTime = SystemClock.uptimeMillis();
1146                r.deliveredStarts.add(si);
1147                si.deliveryCount++;
1148                if (si.neededGrants != null) {
1149                    mAm.grantUriPermissionUncheckedFromIntentLocked(si.neededGrants,
1150                            si.getUriPermissionsLocked());
1151                }
1152                bumpServiceExecutingLocked(r, "start");
1153                if (!oomAdjusted) {
1154                    oomAdjusted = true;
1155                    mAm.updateOomAdjLocked(r.app);
1156                }
1157                int flags = 0;
1158                if (si.deliveryCount > 1) {
1159                    flags |= Service.START_FLAG_RETRY;
1160                }
1161                if (si.doneExecutingCount > 0) {
1162                    flags |= Service.START_FLAG_REDELIVERY;
1163                }
1164                r.app.thread.scheduleServiceArgs(r, si.taskRemoved, si.id, flags, si.intent);
1165            } catch (RemoteException e) {
1166                // Remote process gone...  we'll let the normal cleanup take
1167                // care of this.
1168                if (DEBUG_SERVICE) Slog.v(TAG, "Crashed while scheduling start: " + r);
1169                break;
1170            } catch (Exception e) {
1171                Slog.w(TAG, "Unexpected exception", e);
1172                break;
1173            }
1174        }
1175    }
1176
1177    private final void bringDownServiceLocked(ServiceRecord r, boolean force) {
1178        //Slog.i(TAG, "Bring down service:");
1179        //r.dump("  ");
1180
1181        // Does it still need to run?
1182        if (!force && r.startRequested) {
1183            return;
1184        }
1185        if (r.connections.size() > 0) {
1186            if (!force) {
1187                // XXX should probably keep a count of the number of auto-create
1188                // connections directly in the service.
1189                Iterator<ArrayList<ConnectionRecord>> it = r.connections.values().iterator();
1190                while (it.hasNext()) {
1191                    ArrayList<ConnectionRecord> cr = it.next();
1192                    for (int i=0; i<cr.size(); i++) {
1193                        if ((cr.get(i).flags&Context.BIND_AUTO_CREATE) != 0) {
1194                            return;
1195                        }
1196                    }
1197                }
1198            }
1199
1200            // Report to all of the connections that the service is no longer
1201            // available.
1202            Iterator<ArrayList<ConnectionRecord>> it = r.connections.values().iterator();
1203            while (it.hasNext()) {
1204                ArrayList<ConnectionRecord> c = it.next();
1205                for (int i=0; i<c.size(); i++) {
1206                    ConnectionRecord cr = c.get(i);
1207                    // There is still a connection to the service that is
1208                    // being brought down.  Mark it as dead.
1209                    cr.serviceDead = true;
1210                    try {
1211                        cr.conn.connected(r.name, null);
1212                    } catch (Exception e) {
1213                        Slog.w(TAG, "Failure disconnecting service " + r.name +
1214                              " to connection " + c.get(i).conn.asBinder() +
1215                              " (in " + c.get(i).binding.client.processName + ")", e);
1216                    }
1217                }
1218            }
1219        }
1220
1221        // Tell the service that it has been unbound.
1222        if (r.bindings.size() > 0 && r.app != null && r.app.thread != null) {
1223            Iterator<IntentBindRecord> it = r.bindings.values().iterator();
1224            while (it.hasNext()) {
1225                IntentBindRecord ibr = it.next();
1226                if (DEBUG_SERVICE) Slog.v(TAG, "Bringing down binding " + ibr
1227                        + ": hasBound=" + ibr.hasBound);
1228                if (r.app != null && r.app.thread != null && ibr.hasBound) {
1229                    try {
1230                        bumpServiceExecutingLocked(r, "bring down unbind");
1231                        mAm.updateOomAdjLocked(r.app);
1232                        ibr.hasBound = false;
1233                        r.app.thread.scheduleUnbindService(r,
1234                                ibr.intent.getIntent());
1235                    } catch (Exception e) {
1236                        Slog.w(TAG, "Exception when unbinding service "
1237                                + r.shortName, e);
1238                        serviceDoneExecutingLocked(r, true);
1239                    }
1240                }
1241            }
1242        }
1243
1244        if (DEBUG_SERVICE) Slog.v(TAG, "Bringing down " + r + " " + r.intent);
1245        EventLog.writeEvent(EventLogTags.AM_DESTROY_SERVICE,
1246                r.userId, System.identityHashCode(r), r.shortName,
1247                (r.app != null) ? r.app.pid : -1);
1248
1249        mServiceMap.removeServiceByName(r.name, r.userId);
1250        mServiceMap.removeServiceByIntent(r.intent, r.userId);
1251        r.totalRestartCount = 0;
1252        unscheduleServiceRestartLocked(r);
1253
1254        // Also make sure it is not on the pending list.
1255        int N = mPendingServices.size();
1256        for (int i=0; i<N; i++) {
1257            if (mPendingServices.get(i) == r) {
1258                mPendingServices.remove(i);
1259                if (DEBUG_SERVICE) Slog.v(TAG, "Removed pending: " + r);
1260                i--;
1261                N--;
1262            }
1263        }
1264
1265        r.cancelNotification();
1266        r.isForeground = false;
1267        r.foregroundId = 0;
1268        r.foregroundNoti = null;
1269
1270        // Clear start entries.
1271        r.clearDeliveredStartsLocked();
1272        r.pendingStarts.clear();
1273
1274        if (r.app != null) {
1275            synchronized (r.stats.getBatteryStats()) {
1276                r.stats.stopLaunchedLocked();
1277            }
1278            r.app.services.remove(r);
1279            if (r.app.thread != null) {
1280                try {
1281                    bumpServiceExecutingLocked(r, "stop");
1282                    mStoppingServices.add(r);
1283                    mAm.updateOomAdjLocked(r.app);
1284                    r.app.thread.scheduleStopService(r);
1285                } catch (Exception e) {
1286                    Slog.w(TAG, "Exception when stopping service "
1287                            + r.shortName, e);
1288                    serviceDoneExecutingLocked(r, true);
1289                }
1290                updateServiceForegroundLocked(r.app, false);
1291            } else {
1292                if (DEBUG_SERVICE) Slog.v(
1293                    TAG, "Removed service that has no process: " + r);
1294            }
1295        } else {
1296            if (DEBUG_SERVICE) Slog.v(
1297                TAG, "Removed service that is not running: " + r);
1298        }
1299
1300        if (r.bindings.size() > 0) {
1301            r.bindings.clear();
1302        }
1303
1304        if (r.restarter instanceof ServiceRestarter) {
1305           ((ServiceRestarter)r.restarter).setService(null);
1306        }
1307    }
1308
1309    void removeConnectionLocked(
1310        ConnectionRecord c, ProcessRecord skipApp, ActivityRecord skipAct) {
1311        IBinder binder = c.conn.asBinder();
1312        AppBindRecord b = c.binding;
1313        ServiceRecord s = b.service;
1314        ArrayList<ConnectionRecord> clist = s.connections.get(binder);
1315        if (clist != null) {
1316            clist.remove(c);
1317            if (clist.size() == 0) {
1318                s.connections.remove(binder);
1319            }
1320        }
1321        b.connections.remove(c);
1322        if (c.activity != null && c.activity != skipAct) {
1323            if (c.activity.connections != null) {
1324                c.activity.connections.remove(c);
1325            }
1326        }
1327        if (b.client != skipApp) {
1328            b.client.connections.remove(c);
1329            if ((c.flags&Context.BIND_ABOVE_CLIENT) != 0) {
1330                b.client.updateHasAboveClientLocked();
1331            }
1332        }
1333        clist = mServiceConnections.get(binder);
1334        if (clist != null) {
1335            clist.remove(c);
1336            if (clist.size() == 0) {
1337                mServiceConnections.remove(binder);
1338            }
1339        }
1340
1341        if (b.connections.size() == 0) {
1342            b.intent.apps.remove(b.client);
1343        }
1344
1345        if (!c.serviceDead) {
1346            if (DEBUG_SERVICE) Slog.v(TAG, "Disconnecting binding " + b.intent
1347                    + ": shouldUnbind=" + b.intent.hasBound);
1348            if (s.app != null && s.app.thread != null && b.intent.apps.size() == 0
1349                    && b.intent.hasBound) {
1350                try {
1351                    bumpServiceExecutingLocked(s, "unbind");
1352                    mAm.updateOomAdjLocked(s.app);
1353                    b.intent.hasBound = false;
1354                    // Assume the client doesn't want to know about a rebind;
1355                    // we will deal with that later if it asks for one.
1356                    b.intent.doRebind = false;
1357                    s.app.thread.scheduleUnbindService(s, b.intent.intent.getIntent());
1358                } catch (Exception e) {
1359                    Slog.w(TAG, "Exception when unbinding service " + s.shortName, e);
1360                    serviceDoneExecutingLocked(s, true);
1361                }
1362            }
1363
1364            if ((c.flags&Context.BIND_AUTO_CREATE) != 0) {
1365                bringDownServiceLocked(s, false);
1366            }
1367        }
1368    }
1369
1370    void serviceDoneExecutingLocked(ServiceRecord r, int type, int startId, int res) {
1371        boolean inStopping = mStoppingServices.contains(r);
1372        if (r != null) {
1373            if (type == 1) {
1374                // This is a call from a service start...  take care of
1375                // book-keeping.
1376                r.callStart = true;
1377                switch (res) {
1378                    case Service.START_STICKY_COMPATIBILITY:
1379                    case Service.START_STICKY: {
1380                        // We are done with the associated start arguments.
1381                        r.findDeliveredStart(startId, true);
1382                        // Don't stop if killed.
1383                        r.stopIfKilled = false;
1384                        break;
1385                    }
1386                    case Service.START_NOT_STICKY: {
1387                        // We are done with the associated start arguments.
1388                        r.findDeliveredStart(startId, true);
1389                        if (r.getLastStartId() == startId) {
1390                            // There is no more work, and this service
1391                            // doesn't want to hang around if killed.
1392                            r.stopIfKilled = true;
1393                        }
1394                        break;
1395                    }
1396                    case Service.START_REDELIVER_INTENT: {
1397                        // We'll keep this item until they explicitly
1398                        // call stop for it, but keep track of the fact
1399                        // that it was delivered.
1400                        ServiceRecord.StartItem si = r.findDeliveredStart(startId, false);
1401                        if (si != null) {
1402                            si.deliveryCount = 0;
1403                            si.doneExecutingCount++;
1404                            // Don't stop if killed.
1405                            r.stopIfKilled = true;
1406                        }
1407                        break;
1408                    }
1409                    case Service.START_TASK_REMOVED_COMPLETE: {
1410                        // Special processing for onTaskRemoved().  Don't
1411                        // impact normal onStartCommand() processing.
1412                        r.findDeliveredStart(startId, true);
1413                        break;
1414                    }
1415                    default:
1416                        throw new IllegalArgumentException(
1417                                "Unknown service start result: " + res);
1418                }
1419                if (res == Service.START_STICKY_COMPATIBILITY) {
1420                    r.callStart = false;
1421                }
1422            }
1423            final long origId = Binder.clearCallingIdentity();
1424            serviceDoneExecutingLocked(r, inStopping);
1425            Binder.restoreCallingIdentity(origId);
1426        } else {
1427            Slog.w(TAG, "Done executing unknown service from pid "
1428                    + Binder.getCallingPid());
1429        }
1430    }
1431
1432    private void serviceDoneExecutingLocked(ServiceRecord r, boolean inStopping) {
1433        if (DEBUG_SERVICE) Slog.v(TAG, "<<< DONE EXECUTING " + r
1434                + ": nesting=" + r.executeNesting
1435                + ", inStopping=" + inStopping + ", app=" + r.app);
1436        else if (DEBUG_SERVICE_EXECUTING) Slog.v(TAG, "<<< DONE EXECUTING " + r.shortName);
1437        r.executeNesting--;
1438        if (r.executeNesting <= 0 && r.app != null) {
1439            if (DEBUG_SERVICE) Slog.v(TAG,
1440                    "Nesting at 0 of " + r.shortName);
1441            r.app.executingServices.remove(r);
1442            if (r.app.executingServices.size() == 0) {
1443                if (DEBUG_SERVICE || DEBUG_SERVICE_EXECUTING) Slog.v(TAG,
1444                        "No more executingServices of " + r.shortName);
1445                mAm.mHandler.removeMessages(ActivityManagerService.SERVICE_TIMEOUT_MSG, r.app);
1446            }
1447            if (inStopping) {
1448                if (DEBUG_SERVICE) Slog.v(TAG,
1449                        "doneExecuting remove stopping " + r);
1450                mStoppingServices.remove(r);
1451                r.bindings.clear();
1452            }
1453            mAm.updateOomAdjLocked(r.app);
1454        }
1455    }
1456
1457    boolean attachApplicationLocked(ProcessRecord proc, String processName) throws Exception {
1458        boolean didSomething = false;
1459        // Collect any services that are waiting for this process to come up.
1460        if (mPendingServices.size() > 0) {
1461            ServiceRecord sr = null;
1462            try {
1463                for (int i=0; i<mPendingServices.size(); i++) {
1464                    sr = mPendingServices.get(i);
1465                    if (proc != sr.isolatedProc && (proc.uid != sr.appInfo.uid
1466                            || !processName.equals(sr.processName))) {
1467                        continue;
1468                    }
1469
1470                    mPendingServices.remove(i);
1471                    i--;
1472                    realStartServiceLocked(sr, proc);
1473                    didSomething = true;
1474                }
1475            } catch (Exception e) {
1476                Slog.w(TAG, "Exception in new application when starting service "
1477                        + sr.shortName, e);
1478                throw e;
1479            }
1480        }
1481        // Also, if there are any services that are waiting to restart and
1482        // would run in this process, now is a good time to start them.  It would
1483        // be weird to bring up the process but arbitrarily not let the services
1484        // run at this point just because their restart time hasn't come up.
1485        if (mRestartingServices.size() > 0) {
1486            ServiceRecord sr = null;
1487            for (int i=0; i<mRestartingServices.size(); i++) {
1488                sr = mRestartingServices.get(i);
1489                if (proc != sr.isolatedProc && (proc.uid != sr.appInfo.uid
1490                        || !processName.equals(sr.processName))) {
1491                    continue;
1492                }
1493                mAm.mHandler.removeCallbacks(sr.restarter);
1494                mAm.mHandler.post(sr.restarter);
1495            }
1496        }
1497        return didSomething;
1498    }
1499
1500    void processStartTimedOutLocked(ProcessRecord proc) {
1501        for (int i=0; i<mPendingServices.size(); i++) {
1502            ServiceRecord sr = mPendingServices.get(i);
1503            if ((proc.uid == sr.appInfo.uid
1504                    && proc.processName.equals(sr.processName))
1505                    || sr.isolatedProc == proc) {
1506                Slog.w(TAG, "Forcing bringing down service: " + sr);
1507                sr.isolatedProc = null;
1508                mPendingServices.remove(i);
1509                i--;
1510                bringDownServiceLocked(sr, true);
1511            }
1512        }
1513    }
1514
1515    private boolean collectForceStopServicesLocked(String name, int userId,
1516            boolean evenPersistent, boolean doit,
1517            HashMap<ComponentName, ServiceRecord> services,
1518            ArrayList<ServiceRecord> result) {
1519        boolean didSomething = false;
1520        for (ServiceRecord service : services.values()) {
1521            if ((name == null || service.packageName.equals(name))
1522                    && (service.app == null || evenPersistent || !service.app.persistent)) {
1523                if (!doit) {
1524                    return true;
1525                }
1526                didSomething = true;
1527                Slog.i(TAG, "  Force stopping service " + service);
1528                if (service.app != null) {
1529                    service.app.removed = true;
1530                }
1531                service.app = null;
1532                service.isolatedProc = null;
1533                result.add(service);
1534            }
1535        }
1536        return didSomething;
1537    }
1538
1539    boolean forceStopLocked(String name, int userId, boolean evenPersistent, boolean doit) {
1540        boolean didSomething = false;
1541        ArrayList<ServiceRecord> services = new ArrayList<ServiceRecord>();
1542        if (userId == UserHandle.USER_ALL) {
1543            for (int i=0; i<mServiceMap.mServicesByNamePerUser.size(); i++) {
1544                didSomething |= collectForceStopServicesLocked(name, userId, evenPersistent,
1545                        doit, mServiceMap.mServicesByNamePerUser.valueAt(i), services);
1546                if (!doit && didSomething) {
1547                    return true;
1548                }
1549            }
1550        } else {
1551            HashMap<ComponentName, ServiceRecord> items
1552                    = mServiceMap.mServicesByNamePerUser.get(userId);
1553            if (items != null) {
1554                didSomething = collectForceStopServicesLocked(name, userId, evenPersistent,
1555                        doit, items, services);
1556            }
1557        }
1558
1559        int N = services.size();
1560        for (int i=0; i<N; i++) {
1561            bringDownServiceLocked(services.get(i), true);
1562        }
1563        return didSomething;
1564    }
1565
1566    void cleanUpRemovedTaskLocked(TaskRecord tr, ComponentName component, Intent baseIntent) {
1567        ArrayList<ServiceRecord> services = new ArrayList<ServiceRecord>();
1568        for (ServiceRecord sr : mServiceMap.getAllServices(tr.userId)) {
1569            if (sr.packageName.equals(component.getPackageName())) {
1570                services.add(sr);
1571            }
1572        }
1573
1574        // Take care of any running services associated with the app.
1575        for (int i=0; i<services.size(); i++) {
1576            ServiceRecord sr = services.get(i);
1577            if (sr.startRequested) {
1578                if ((sr.serviceInfo.flags&ServiceInfo.FLAG_STOP_WITH_TASK) != 0) {
1579                    Slog.i(TAG, "Stopping service " + sr.shortName + ": remove task");
1580                    stopServiceLocked(sr);
1581                } else {
1582                    sr.pendingStarts.add(new ServiceRecord.StartItem(sr, true,
1583                            sr.makeNextStartId(), baseIntent, null));
1584                    if (sr.app != null && sr.app.thread != null) {
1585                        sendServiceArgsLocked(sr, false);
1586                    }
1587                }
1588            }
1589        }
1590    }
1591
1592    final void killServicesLocked(ProcessRecord app,
1593            boolean allowRestart) {
1594        // Report disconnected services.
1595        if (false) {
1596            // XXX we are letting the client link to the service for
1597            // death notifications.
1598            if (app.services.size() > 0) {
1599                Iterator<ServiceRecord> it = app.services.iterator();
1600                while (it.hasNext()) {
1601                    ServiceRecord r = it.next();
1602                    if (r.connections.size() > 0) {
1603                        Iterator<ArrayList<ConnectionRecord>> jt
1604                                = r.connections.values().iterator();
1605                        while (jt.hasNext()) {
1606                            ArrayList<ConnectionRecord> cl = jt.next();
1607                            for (int i=0; i<cl.size(); i++) {
1608                                ConnectionRecord c = cl.get(i);
1609                                if (c.binding.client != app) {
1610                                    try {
1611                                        //c.conn.connected(r.className, null);
1612                                    } catch (Exception e) {
1613                                        // todo: this should be asynchronous!
1614                                        Slog.w(TAG, "Exception thrown disconnected servce "
1615                                              + r.shortName
1616                                              + " from app " + app.processName, e);
1617                                    }
1618                                }
1619                            }
1620                        }
1621                    }
1622                }
1623            }
1624        }
1625
1626        // Clean up any connections this application has to other services.
1627        if (app.connections.size() > 0) {
1628            Iterator<ConnectionRecord> it = app.connections.iterator();
1629            while (it.hasNext()) {
1630                ConnectionRecord r = it.next();
1631                removeConnectionLocked(r, app, null);
1632            }
1633        }
1634        app.connections.clear();
1635
1636        if (app.services.size() != 0) {
1637            // Any services running in the application need to be placed
1638            // back in the pending list.
1639            Iterator<ServiceRecord> it = app.services.iterator();
1640            while (it.hasNext()) {
1641                ServiceRecord sr = it.next();
1642                synchronized (sr.stats.getBatteryStats()) {
1643                    sr.stats.stopLaunchedLocked();
1644                }
1645                sr.app = null;
1646                sr.isolatedProc = null;
1647                sr.executeNesting = 0;
1648                if (mStoppingServices.remove(sr)) {
1649                    if (DEBUG_SERVICE) Slog.v(TAG, "killServices remove stopping " + sr);
1650                }
1651
1652                boolean hasClients = sr.bindings.size() > 0;
1653                if (hasClients) {
1654                    Iterator<IntentBindRecord> bindings
1655                            = sr.bindings.values().iterator();
1656                    while (bindings.hasNext()) {
1657                        IntentBindRecord b = bindings.next();
1658                        if (DEBUG_SERVICE) Slog.v(TAG, "Killing binding " + b
1659                                + ": shouldUnbind=" + b.hasBound);
1660                        b.binder = null;
1661                        b.requested = b.received = b.hasBound = false;
1662                    }
1663                }
1664
1665                if (sr.crashCount >= 2 && (sr.serviceInfo.applicationInfo.flags
1666                        &ApplicationInfo.FLAG_PERSISTENT) == 0) {
1667                    Slog.w(TAG, "Service crashed " + sr.crashCount
1668                            + " times, stopping: " + sr);
1669                    EventLog.writeEvent(EventLogTags.AM_SERVICE_CRASHED_TOO_MUCH,
1670                            sr.userId, sr.crashCount, sr.shortName, app.pid);
1671                    bringDownServiceLocked(sr, true);
1672                } else if (!allowRestart) {
1673                    bringDownServiceLocked(sr, true);
1674                } else {
1675                    boolean canceled = scheduleServiceRestartLocked(sr, true);
1676
1677                    // Should the service remain running?  Note that in the
1678                    // extreme case of so many attempts to deliver a command
1679                    // that it failed we also will stop it here.
1680                    if (sr.startRequested && (sr.stopIfKilled || canceled)) {
1681                        if (sr.pendingStarts.size() == 0) {
1682                            sr.startRequested = false;
1683                            if (!hasClients) {
1684                                // Whoops, no reason to restart!
1685                                bringDownServiceLocked(sr, true);
1686                            }
1687                        }
1688                    }
1689                }
1690            }
1691
1692            if (!allowRestart) {
1693                app.services.clear();
1694            }
1695        }
1696
1697        // Make sure we have no more records on the stopping list.
1698        int i = mStoppingServices.size();
1699        while (i > 0) {
1700            i--;
1701            ServiceRecord sr = mStoppingServices.get(i);
1702            if (sr.app == app) {
1703                mStoppingServices.remove(i);
1704                if (DEBUG_SERVICE) Slog.v(TAG, "killServices remove stopping " + sr);
1705            }
1706        }
1707
1708        app.executingServices.clear();
1709    }
1710
1711    ActivityManager.RunningServiceInfo makeRunningServiceInfoLocked(ServiceRecord r) {
1712        ActivityManager.RunningServiceInfo info =
1713            new ActivityManager.RunningServiceInfo();
1714        info.service = r.name;
1715        if (r.app != null) {
1716            info.pid = r.app.pid;
1717        }
1718        info.uid = r.appInfo.uid;
1719        info.process = r.processName;
1720        info.foreground = r.isForeground;
1721        info.activeSince = r.createTime;
1722        info.started = r.startRequested;
1723        info.clientCount = r.connections.size();
1724        info.crashCount = r.crashCount;
1725        info.lastActivityTime = r.lastActivity;
1726        if (r.isForeground) {
1727            info.flags |= ActivityManager.RunningServiceInfo.FLAG_FOREGROUND;
1728        }
1729        if (r.startRequested) {
1730            info.flags |= ActivityManager.RunningServiceInfo.FLAG_STARTED;
1731        }
1732        if (r.app != null && r.app.pid == ActivityManagerService.MY_PID) {
1733            info.flags |= ActivityManager.RunningServiceInfo.FLAG_SYSTEM_PROCESS;
1734        }
1735        if (r.app != null && r.app.persistent) {
1736            info.flags |= ActivityManager.RunningServiceInfo.FLAG_PERSISTENT_PROCESS;
1737        }
1738
1739        for (ArrayList<ConnectionRecord> connl : r.connections.values()) {
1740            for (int i=0; i<connl.size(); i++) {
1741                ConnectionRecord conn = connl.get(i);
1742                if (conn.clientLabel != 0) {
1743                    info.clientPackage = conn.binding.client.info.packageName;
1744                    info.clientLabel = conn.clientLabel;
1745                    return info;
1746                }
1747            }
1748        }
1749        return info;
1750    }
1751
1752    List<ActivityManager.RunningServiceInfo> getRunningServiceInfoLocked(int maxNum,
1753            int flags) {
1754        ArrayList<ActivityManager.RunningServiceInfo> res
1755                = new ArrayList<ActivityManager.RunningServiceInfo>();
1756
1757        final int uid = Binder.getCallingUid();
1758        final long ident = Binder.clearCallingIdentity();
1759        try {
1760            if (ActivityManager.checkUidPermission(
1761                    android.Manifest.permission.INTERACT_ACROSS_USERS_FULL,
1762                    uid) == PackageManager.PERMISSION_GRANTED) {
1763                int[] users = mAm.getUsersLocked();
1764                for (int ui=0; ui<users.length && res.size() < maxNum; ui++) {
1765                    if (mServiceMap.getAllServices(users[ui]).size() > 0) {
1766                        Iterator<ServiceRecord> it = mServiceMap.getAllServices(
1767                                users[ui]).iterator();
1768                        while (it.hasNext() && res.size() < maxNum) {
1769                            res.add(makeRunningServiceInfoLocked(it.next()));
1770                        }
1771                    }
1772                }
1773
1774                for (int i=0; i<mRestartingServices.size() && res.size() < maxNum; i++) {
1775                    ServiceRecord r = mRestartingServices.get(i);
1776                    ActivityManager.RunningServiceInfo info =
1777                            makeRunningServiceInfoLocked(r);
1778                    info.restarting = r.nextRestartTime;
1779                    res.add(info);
1780                }
1781            } else {
1782                int userId = UserHandle.getUserId(uid);
1783                if (mServiceMap.getAllServices(userId).size() > 0) {
1784                    Iterator<ServiceRecord> it
1785                            = mServiceMap.getAllServices(userId).iterator();
1786                    while (it.hasNext() && res.size() < maxNum) {
1787                        res.add(makeRunningServiceInfoLocked(it.next()));
1788                    }
1789                }
1790
1791                for (int i=0; i<mRestartingServices.size() && res.size() < maxNum; i++) {
1792                    ServiceRecord r = mRestartingServices.get(i);
1793                    if (r.userId == userId) {
1794                        ActivityManager.RunningServiceInfo info =
1795                                makeRunningServiceInfoLocked(r);
1796                        info.restarting = r.nextRestartTime;
1797                        res.add(info);
1798                    }
1799                }
1800            }
1801        } finally {
1802            Binder.restoreCallingIdentity(ident);
1803        }
1804
1805        return res;
1806    }
1807
1808    public PendingIntent getRunningServiceControlPanelLocked(ComponentName name) {
1809        int userId = UserHandle.getUserId(Binder.getCallingUid());
1810        ServiceRecord r = mServiceMap.getServiceByName(name, userId);
1811        if (r != null) {
1812            for (ArrayList<ConnectionRecord> conn : r.connections.values()) {
1813                for (int i=0; i<conn.size(); i++) {
1814                    if (conn.get(i).clientIntent != null) {
1815                        return conn.get(i).clientIntent;
1816                    }
1817                }
1818            }
1819        }
1820        return null;
1821    }
1822
1823    void serviceTimeout(ProcessRecord proc) {
1824        String anrMessage = null;
1825
1826        synchronized(this) {
1827            if (proc.executingServices.size() == 0 || proc.thread == null) {
1828                return;
1829            }
1830            long maxTime = SystemClock.uptimeMillis() - SERVICE_TIMEOUT;
1831            Iterator<ServiceRecord> it = proc.executingServices.iterator();
1832            ServiceRecord timeout = null;
1833            long nextTime = 0;
1834            while (it.hasNext()) {
1835                ServiceRecord sr = it.next();
1836                if (sr.executingStart < maxTime) {
1837                    timeout = sr;
1838                    break;
1839                }
1840                if (sr.executingStart > nextTime) {
1841                    nextTime = sr.executingStart;
1842                }
1843            }
1844            if (timeout != null && mAm.mLruProcesses.contains(proc)) {
1845                Slog.w(TAG, "Timeout executing service: " + timeout);
1846                anrMessage = "Executing service " + timeout.shortName;
1847            } else {
1848                Message msg = mAm.mHandler.obtainMessage(
1849                        ActivityManagerService.SERVICE_TIMEOUT_MSG);
1850                msg.obj = proc;
1851                mAm.mHandler.sendMessageAtTime(msg, nextTime+SERVICE_TIMEOUT);
1852            }
1853        }
1854
1855        if (anrMessage != null) {
1856            mAm.appNotResponding(proc, null, null, false, anrMessage);
1857        }
1858    }
1859
1860    /**
1861     * Prints a list of ServiceRecords (dumpsys activity services)
1862     */
1863    boolean dumpServicesLocked(FileDescriptor fd, PrintWriter pw, String[] args,
1864            int opti, boolean dumpAll, boolean dumpClient, String dumpPackage) {
1865        boolean needSep = false;
1866
1867        ItemMatcher matcher = new ItemMatcher();
1868        matcher.build(args, opti);
1869
1870        pw.println("ACTIVITY MANAGER SERVICES (dumpsys activity services)");
1871        try {
1872            int[] users = mAm.getUsersLocked();
1873            for (int user : users) {
1874                if (mServiceMap.getAllServices(user).size() > 0) {
1875                    boolean printed = false;
1876                    long nowReal = SystemClock.elapsedRealtime();
1877                    Iterator<ServiceRecord> it = mServiceMap.getAllServices(
1878                            user).iterator();
1879                    needSep = false;
1880                    while (it.hasNext()) {
1881                        ServiceRecord r = it.next();
1882                        if (!matcher.match(r, r.name)) {
1883                            continue;
1884                        }
1885                        if (dumpPackage != null && !dumpPackage.equals(r.appInfo.packageName)) {
1886                            continue;
1887                        }
1888                        if (!printed) {
1889                            if (user != 0) {
1890                                pw.println();
1891                            }
1892                            pw.println("  User " + user + " active services:");
1893                            printed = true;
1894                        }
1895                        if (needSep) {
1896                            pw.println();
1897                        }
1898                        pw.print("  * ");
1899                        pw.println(r);
1900                        if (dumpAll) {
1901                            r.dump(pw, "    ");
1902                            needSep = true;
1903                        } else {
1904                            pw.print("    app=");
1905                            pw.println(r.app);
1906                            pw.print("    created=");
1907                            TimeUtils.formatDuration(r.createTime, nowReal, pw);
1908                            pw.print(" started=");
1909                            pw.print(r.startRequested);
1910                            pw.print(" connections=");
1911                            pw.println(r.connections.size());
1912                            if (r.connections.size() > 0) {
1913                                pw.println("    Connections:");
1914                                for (ArrayList<ConnectionRecord> clist : r.connections.values()) {
1915                                    for (int i = 0; i < clist.size(); i++) {
1916                                        ConnectionRecord conn = clist.get(i);
1917                                        pw.print("      ");
1918                                        pw.print(conn.binding.intent.intent.getIntent()
1919                                                .toShortString(false, false, false, false));
1920                                        pw.print(" -> ");
1921                                        ProcessRecord proc = conn.binding.client;
1922                                        pw.println(proc != null ? proc.toShortString() : "null");
1923                                    }
1924                                }
1925                            }
1926                        }
1927                        if (dumpClient && r.app != null && r.app.thread != null) {
1928                            pw.println("    Client:");
1929                            pw.flush();
1930                            try {
1931                                TransferPipe tp = new TransferPipe();
1932                                try {
1933                                    r.app.thread.dumpService(tp.getWriteFd().getFileDescriptor(),
1934                                            r, args);
1935                                    tp.setBufferPrefix("      ");
1936                                    // Short timeout, since blocking here can
1937                                    // deadlock with the application.
1938                                    tp.go(fd, 2000);
1939                                } finally {
1940                                    tp.kill();
1941                                }
1942                            } catch (IOException e) {
1943                                pw.println("      Failure while dumping the service: " + e);
1944                            } catch (RemoteException e) {
1945                                pw.println("      Got a RemoteException while dumping the service");
1946                            }
1947                            needSep = true;
1948                        }
1949                    }
1950                    needSep = printed;
1951                }
1952            }
1953        } catch (Exception e) {
1954            Log.w(TAG, "Exception in dumpServicesLocked: " + e);
1955        }
1956
1957        if (mPendingServices.size() > 0) {
1958            boolean printed = false;
1959            for (int i=0; i<mPendingServices.size(); i++) {
1960                ServiceRecord r = mPendingServices.get(i);
1961                if (!matcher.match(r, r.name)) {
1962                    continue;
1963                }
1964                if (dumpPackage != null && !dumpPackage.equals(r.appInfo.packageName)) {
1965                    continue;
1966                }
1967                if (!printed) {
1968                    if (needSep) pw.println(" ");
1969                    needSep = true;
1970                    pw.println("  Pending services:");
1971                    printed = true;
1972                }
1973                pw.print("  * Pending "); pw.println(r);
1974                r.dump(pw, "    ");
1975            }
1976            needSep = true;
1977        }
1978
1979        if (mRestartingServices.size() > 0) {
1980            boolean printed = false;
1981            for (int i=0; i<mRestartingServices.size(); i++) {
1982                ServiceRecord r = mRestartingServices.get(i);
1983                if (!matcher.match(r, r.name)) {
1984                    continue;
1985                }
1986                if (dumpPackage != null && !dumpPackage.equals(r.appInfo.packageName)) {
1987                    continue;
1988                }
1989                if (!printed) {
1990                    if (needSep) pw.println(" ");
1991                    needSep = true;
1992                    pw.println("  Restarting services:");
1993                    printed = true;
1994                }
1995                pw.print("  * Restarting "); pw.println(r);
1996                r.dump(pw, "    ");
1997            }
1998            needSep = true;
1999        }
2000
2001        if (mStoppingServices.size() > 0) {
2002            boolean printed = false;
2003            for (int i=0; i<mStoppingServices.size(); i++) {
2004                ServiceRecord r = mStoppingServices.get(i);
2005                if (!matcher.match(r, r.name)) {
2006                    continue;
2007                }
2008                if (dumpPackage != null && !dumpPackage.equals(r.appInfo.packageName)) {
2009                    continue;
2010                }
2011                if (!printed) {
2012                    if (needSep) pw.println(" ");
2013                    needSep = true;
2014                    pw.println("  Stopping services:");
2015                    printed = true;
2016                }
2017                pw.print("  * Stopping "); pw.println(r);
2018                r.dump(pw, "    ");
2019            }
2020            needSep = true;
2021        }
2022
2023        if (dumpAll) {
2024            if (mServiceConnections.size() > 0) {
2025                boolean printed = false;
2026                Iterator<ArrayList<ConnectionRecord>> it
2027                        = mServiceConnections.values().iterator();
2028                while (it.hasNext()) {
2029                    ArrayList<ConnectionRecord> r = it.next();
2030                    for (int i=0; i<r.size(); i++) {
2031                        ConnectionRecord cr = r.get(i);
2032                        if (!matcher.match(cr.binding.service, cr.binding.service.name)) {
2033                            continue;
2034                        }
2035                        if (dumpPackage != null && (cr.binding.client == null
2036                                || !dumpPackage.equals(cr.binding.client.info.packageName))) {
2037                            continue;
2038                        }
2039                        if (!printed) {
2040                            if (needSep) pw.println(" ");
2041                            needSep = true;
2042                            pw.println("  Connection bindings to services:");
2043                            printed = true;
2044                        }
2045                        pw.print("  * "); pw.println(cr);
2046                        cr.dump(pw, "    ");
2047                    }
2048                }
2049                needSep = true;
2050            }
2051        }
2052
2053        return needSep;
2054    }
2055
2056    /**
2057     * There are three ways to call this:
2058     *  - no service specified: dump all the services
2059     *  - a flattened component name that matched an existing service was specified as the
2060     *    first arg: dump that one service
2061     *  - the first arg isn't the flattened component name of an existing service:
2062     *    dump all services whose component contains the first arg as a substring
2063     */
2064    protected boolean dumpService(FileDescriptor fd, PrintWriter pw, String name, String[] args,
2065            int opti, boolean dumpAll) {
2066        ArrayList<ServiceRecord> services = new ArrayList<ServiceRecord>();
2067
2068        synchronized (this) {
2069            int[] users = mAm.getUsersLocked();
2070            if ("all".equals(name)) {
2071                for (int user : users) {
2072                    for (ServiceRecord r1 : mServiceMap.getAllServices(user)) {
2073                        services.add(r1);
2074                    }
2075                }
2076            } else {
2077                ComponentName componentName = name != null
2078                        ? ComponentName.unflattenFromString(name) : null;
2079                int objectId = 0;
2080                if (componentName == null) {
2081                    // Not a '/' separated full component name; maybe an object ID?
2082                    try {
2083                        objectId = Integer.parseInt(name, 16);
2084                        name = null;
2085                        componentName = null;
2086                    } catch (RuntimeException e) {
2087                    }
2088                }
2089
2090                for (int user : users) {
2091                    for (ServiceRecord r1 : mServiceMap.getAllServices(user)) {
2092                        if (componentName != null) {
2093                            if (r1.name.equals(componentName)) {
2094                                services.add(r1);
2095                            }
2096                        } else if (name != null) {
2097                            if (r1.name.flattenToString().contains(name)) {
2098                                services.add(r1);
2099                            }
2100                        } else if (System.identityHashCode(r1) == objectId) {
2101                            services.add(r1);
2102                        }
2103                    }
2104                }
2105            }
2106        }
2107
2108        if (services.size() <= 0) {
2109            return false;
2110        }
2111
2112        boolean needSep = false;
2113        for (int i=0; i<services.size(); i++) {
2114            if (needSep) {
2115                pw.println();
2116            }
2117            needSep = true;
2118            dumpService("", fd, pw, services.get(i), args, dumpAll);
2119        }
2120        return true;
2121    }
2122
2123    /**
2124     * Invokes IApplicationThread.dumpService() on the thread of the specified service if
2125     * there is a thread associated with the service.
2126     */
2127    private void dumpService(String prefix, FileDescriptor fd, PrintWriter pw,
2128            final ServiceRecord r, String[] args, boolean dumpAll) {
2129        String innerPrefix = prefix + "  ";
2130        synchronized (this) {
2131            pw.print(prefix); pw.print("SERVICE ");
2132                    pw.print(r.shortName); pw.print(" ");
2133                    pw.print(Integer.toHexString(System.identityHashCode(r)));
2134                    pw.print(" pid=");
2135                    if (r.app != null) pw.println(r.app.pid);
2136                    else pw.println("(not running)");
2137            if (dumpAll) {
2138                r.dump(pw, innerPrefix);
2139            }
2140        }
2141        if (r.app != null && r.app.thread != null) {
2142            pw.print(prefix); pw.println("  Client:");
2143            pw.flush();
2144            try {
2145                TransferPipe tp = new TransferPipe();
2146                try {
2147                    r.app.thread.dumpService(tp.getWriteFd().getFileDescriptor(), r, args);
2148                    tp.setBufferPrefix(prefix + "    ");
2149                    tp.go(fd);
2150                } finally {
2151                    tp.kill();
2152                }
2153            } catch (IOException e) {
2154                pw.println(prefix + "    Failure while dumping the service: " + e);
2155            } catch (RemoteException e) {
2156                pw.println(prefix + "    Got a RemoteException while dumping the service");
2157            }
2158        }
2159    }
2160
2161}
2162