Home | History | Annotate | Download | only in am
      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 
     17 package com.android.server.am;
     18 
     19 import java.io.FileDescriptor;
     20 import java.io.IOException;
     21 import java.io.PrintWriter;
     22 import java.util.ArrayList;
     23 import java.util.HashSet;
     24 import java.util.Iterator;
     25 import java.util.List;
     26 
     27 import android.os.Handler;
     28 import android.os.Looper;
     29 import android.os.SystemProperties;
     30 import android.util.ArrayMap;
     31 import com.android.internal.app.ProcessStats;
     32 import com.android.internal.os.BatteryStatsImpl;
     33 import com.android.internal.os.TransferPipe;
     34 import com.android.server.am.ActivityManagerService.ItemMatcher;
     35 import com.android.server.am.ActivityManagerService.NeededUriGrants;
     36 
     37 import android.app.ActivityManager;
     38 import android.app.AppGlobals;
     39 import android.app.IApplicationThread;
     40 import android.app.IServiceConnection;
     41 import android.app.Notification;
     42 import android.app.PendingIntent;
     43 import android.app.Service;
     44 import android.content.ComponentName;
     45 import android.content.Context;
     46 import android.content.Intent;
     47 import android.content.pm.ApplicationInfo;
     48 import android.content.pm.PackageManager;
     49 import android.content.pm.ResolveInfo;
     50 import android.content.pm.ServiceInfo;
     51 import android.os.Binder;
     52 import android.os.IBinder;
     53 import android.os.Message;
     54 import android.os.Process;
     55 import android.os.RemoteException;
     56 import android.os.SystemClock;
     57 import android.os.UserHandle;
     58 import android.util.EventLog;
     59 import android.util.Slog;
     60 import android.util.SparseArray;
     61 import android.util.TimeUtils;
     62 
     63 public final class ActiveServices {
     64     static final boolean DEBUG_SERVICE = ActivityManagerService.DEBUG_SERVICE;
     65     static final boolean DEBUG_SERVICE_EXECUTING = ActivityManagerService.DEBUG_SERVICE_EXECUTING;
     66     static final boolean DEBUG_DELAYED_SERVICE = ActivityManagerService.DEBUG_SERVICE;
     67     static final boolean DEBUG_DELAYED_STATS = DEBUG_DELAYED_SERVICE;
     68     static final boolean DEBUG_MU = ActivityManagerService.DEBUG_MU;
     69     static final String TAG = ActivityManagerService.TAG;
     70     static final String TAG_MU = ActivityManagerService.TAG_MU;
     71 
     72     // How long we wait for a service to finish executing.
     73     static final int SERVICE_TIMEOUT = 20*1000;
     74 
     75     // How long we wait for a service to finish executing.
     76     static final int SERVICE_BACKGROUND_TIMEOUT = SERVICE_TIMEOUT * 10;
     77 
     78     // How long a service needs to be running until restarting its process
     79     // is no longer considered to be a relaunch of the service.
     80     static final int SERVICE_RESTART_DURATION = 1*1000;
     81 
     82     // How long a service needs to be running until it will start back at
     83     // SERVICE_RESTART_DURATION after being killed.
     84     static final int SERVICE_RESET_RUN_DURATION = 60*1000;
     85 
     86     // Multiplying factor to increase restart duration time by, for each time
     87     // a service is killed before it has run for SERVICE_RESET_RUN_DURATION.
     88     static final int SERVICE_RESTART_DURATION_FACTOR = 4;
     89 
     90     // The minimum amount of time between restarting services that we allow.
     91     // That is, when multiple services are restarting, we won't allow each
     92     // to restart less than this amount of time from the last one.
     93     static final int SERVICE_MIN_RESTART_TIME_BETWEEN = 10*1000;
     94 
     95     // Maximum amount of time for there to be no activity on a service before
     96     // we consider it non-essential and allow its process to go on the
     97     // LRU background list.
     98     static final int MAX_SERVICE_INACTIVITY = 30*60*1000;
     99 
    100     // How long we wait for a background started service to stop itself before
    101     // allowing the next pending start to run.
    102     static final int BG_START_TIMEOUT = 15*1000;
    103 
    104     final ActivityManagerService mAm;
    105 
    106     // Maximum number of services that we allow to start in the background
    107     // at the same time.
    108     final int mMaxStartingBackground;
    109 
    110     final SparseArray<ServiceMap> mServiceMap = new SparseArray<ServiceMap>();
    111 
    112     /**
    113      * All currently bound service connections.  Keys are the IBinder of
    114      * the client's IServiceConnection.
    115      */
    116     final ArrayMap<IBinder, ArrayList<ConnectionRecord>> mServiceConnections
    117             = new ArrayMap<IBinder, ArrayList<ConnectionRecord>>();
    118 
    119     /**
    120      * List of services that we have been asked to start,
    121      * but haven't yet been able to.  It is used to hold start requests
    122      * while waiting for their corresponding application thread to get
    123      * going.
    124      */
    125     final ArrayList<ServiceRecord> mPendingServices
    126             = new ArrayList<ServiceRecord>();
    127 
    128     /**
    129      * List of services that are scheduled to restart following a crash.
    130      */
    131     final ArrayList<ServiceRecord> mRestartingServices
    132             = new ArrayList<ServiceRecord>();
    133 
    134     /**
    135      * List of services that are in the process of being destroyed.
    136      */
    137     final ArrayList<ServiceRecord> mDestroyingServices
    138             = new ArrayList<ServiceRecord>();
    139 
    140     static final class DelayingProcess extends ArrayList<ServiceRecord> {
    141         long timeoout;
    142     }
    143 
    144     /**
    145      * Information about services for a single user.
    146      */
    147     class ServiceMap extends Handler {
    148         final int mUserId;
    149         final ArrayMap<ComponentName, ServiceRecord> mServicesByName
    150                 = new ArrayMap<ComponentName, ServiceRecord>();
    151         final ArrayMap<Intent.FilterComparison, ServiceRecord> mServicesByIntent
    152                 = new ArrayMap<Intent.FilterComparison, ServiceRecord>();
    153 
    154         final ArrayList<ServiceRecord> mDelayedStartList
    155                 = new ArrayList<ServiceRecord>();
    156         /* XXX eventually I'd like to have this based on processes instead of services.
    157          * That is, if we try to start two services in a row both running in the same
    158          * process, this should be one entry in mStartingBackground for that one process
    159          * that remains until all services in it are done.
    160         final ArrayMap<ProcessRecord, DelayingProcess> mStartingBackgroundMap
    161                 = new ArrayMap<ProcessRecord, DelayingProcess>();
    162         final ArrayList<DelayingProcess> mStartingProcessList
    163                 = new ArrayList<DelayingProcess>();
    164         */
    165 
    166         final ArrayList<ServiceRecord> mStartingBackground
    167                 = new ArrayList<ServiceRecord>();
    168 
    169         static final int MSG_BG_START_TIMEOUT = 1;
    170 
    171         ServiceMap(Looper looper, int userId) {
    172             super(looper);
    173             mUserId = userId;
    174         }
    175 
    176         @Override
    177         public void handleMessage(Message msg) {
    178             switch (msg.what) {
    179                 case MSG_BG_START_TIMEOUT: {
    180                     synchronized (mAm) {
    181                         rescheduleDelayedStarts();
    182                     }
    183                 } break;
    184             }
    185         }
    186 
    187         void ensureNotStartingBackground(ServiceRecord r) {
    188             if (mStartingBackground.remove(r)) {
    189                 if (DEBUG_DELAYED_STATS) Slog.v(TAG, "No longer background starting: " + r);
    190                 rescheduleDelayedStarts();
    191             }
    192             if (mDelayedStartList.remove(r)) {
    193                 if (DEBUG_DELAYED_STATS) Slog.v(TAG, "No longer delaying start: " + r);
    194             }
    195         }
    196 
    197         void rescheduleDelayedStarts() {
    198             removeMessages(MSG_BG_START_TIMEOUT);
    199             final long now = SystemClock.uptimeMillis();
    200             for (int i=0, N=mStartingBackground.size(); i<N; i++) {
    201                 ServiceRecord r = mStartingBackground.get(i);
    202                 if (r.startingBgTimeout <= now) {
    203                     Slog.i(TAG, "Waited long enough for: " + r);
    204                     mStartingBackground.remove(i);
    205                     N--;
    206                 }
    207             }
    208             while (mDelayedStartList.size() > 0
    209                     && mStartingBackground.size() < mMaxStartingBackground) {
    210                 ServiceRecord r = mDelayedStartList.remove(0);
    211                 if (DEBUG_DELAYED_STATS) Slog.v(TAG, "REM FR DELAY LIST (exec next): " + r);
    212                 if (r.pendingStarts.size() <= 0) {
    213                     Slog.w(TAG, "**** NO PENDING STARTS! " + r + " startReq=" + r.startRequested
    214                             + " delayedStop=" + r.delayedStop);
    215                 }
    216                 if (DEBUG_DELAYED_SERVICE) {
    217                     if (mDelayedStartList.size() > 0) {
    218                         Slog.v(TAG, "Remaining delayed list:");
    219                         for (int i=0; i<mDelayedStartList.size(); i++) {
    220                             Slog.v(TAG, "  #" + i + ": " + mDelayedStartList.get(i));
    221                         }
    222                     }
    223                 }
    224                 r.delayed = false;
    225                 startServiceInnerLocked(this, r.pendingStarts.get(0).intent, r, false, true);
    226             }
    227             if (mStartingBackground.size() > 0) {
    228                 ServiceRecord next = mStartingBackground.get(0);
    229                 long when = next.startingBgTimeout > now ? next.startingBgTimeout : now;
    230                 if (DEBUG_DELAYED_SERVICE) Slog.v(TAG, "Top bg start is " + next
    231                         + ", can delay others up to " + when);
    232                 Message msg = obtainMessage(MSG_BG_START_TIMEOUT);
    233                 sendMessageAtTime(msg, when);
    234             }
    235             if (mStartingBackground.size() < mMaxStartingBackground) {
    236                 mAm.backgroundServicesFinishedLocked(mUserId);
    237             }
    238         }
    239     }
    240 
    241     public ActiveServices(ActivityManagerService service) {
    242         mAm = service;
    243         int maxBg = 0;
    244         try {
    245             maxBg = Integer.parseInt(SystemProperties.get("ro.config.max_starting_bg", "0"));
    246         } catch(RuntimeException e) {
    247         }
    248         mMaxStartingBackground = maxBg > 0 ? maxBg : ActivityManager.isLowRamDeviceStatic() ? 1 : 3;
    249     }
    250 
    251     ServiceRecord getServiceByName(ComponentName name, int callingUser) {
    252         // TODO: Deal with global services
    253         if (DEBUG_MU)
    254             Slog.v(TAG_MU, "getServiceByName(" + name + "), callingUser = " + callingUser);
    255         return getServiceMap(callingUser).mServicesByName.get(name);
    256     }
    257 
    258     boolean hasBackgroundServices(int callingUser) {
    259         ServiceMap smap = mServiceMap.get(callingUser);
    260         return smap != null ? smap.mStartingBackground.size() >= mMaxStartingBackground : false;
    261     }
    262 
    263     private ServiceMap getServiceMap(int callingUser) {
    264         ServiceMap smap = mServiceMap.get(callingUser);
    265         if (smap == null) {
    266             smap = new ServiceMap(mAm.mHandler.getLooper(), callingUser);
    267             mServiceMap.put(callingUser, smap);
    268         }
    269         return smap;
    270     }
    271 
    272     ArrayMap<ComponentName, ServiceRecord> getServices(int callingUser) {
    273         return getServiceMap(callingUser).mServicesByName;
    274     }
    275 
    276     ComponentName startServiceLocked(IApplicationThread caller,
    277             Intent service, String resolvedType,
    278             int callingPid, int callingUid, int userId) {
    279         if (DEBUG_DELAYED_STATS) Slog.v(TAG, "startService: " + service
    280                 + " type=" + resolvedType + " args=" + service.getExtras());
    281 
    282         final boolean callerFg;
    283         if (caller != null) {
    284             final ProcessRecord callerApp = mAm.getRecordForAppLocked(caller);
    285             if (callerApp == null) {
    286                 throw new SecurityException(
    287                         "Unable to find app for caller " + caller
    288                         + " (pid=" + Binder.getCallingPid()
    289                         + ") when starting service " + service);
    290             }
    291             callerFg = callerApp.setSchedGroup != Process.THREAD_GROUP_BG_NONINTERACTIVE;
    292         } else {
    293             callerFg = true;
    294         }
    295 
    296 
    297         ServiceLookupResult res =
    298             retrieveServiceLocked(service, resolvedType,
    299                     callingPid, callingUid, userId, true, callerFg);
    300         if (res == null) {
    301             return null;
    302         }
    303         if (res.record == null) {
    304             return new ComponentName("!", res.permission != null
    305                     ? res.permission : "private to package");
    306         }
    307         ServiceRecord r = res.record;
    308         NeededUriGrants neededGrants = mAm.checkGrantUriPermissionFromIntentLocked(
    309                 callingUid, r.packageName, service, service.getFlags(), null);
    310         if (unscheduleServiceRestartLocked(r, callingUid, false)) {
    311             if (DEBUG_SERVICE) Slog.v(TAG, "START SERVICE WHILE RESTART PENDING: " + r);
    312         }
    313         r.lastActivity = SystemClock.uptimeMillis();
    314         r.startRequested = true;
    315         r.delayedStop = false;
    316         r.pendingStarts.add(new ServiceRecord.StartItem(r, false, r.makeNextStartId(),
    317                 service, neededGrants));
    318 
    319         final ServiceMap smap = getServiceMap(r.userId);
    320         boolean addToStarting = false;
    321         if (!callerFg && r.app == null && mAm.mStartedUsers.get(r.userId) != null) {
    322             ProcessRecord proc = mAm.getProcessRecordLocked(r.processName, r.appInfo.uid, false);
    323             if (proc == null || proc.curProcState > ActivityManager.PROCESS_STATE_RECEIVER) {
    324                 // If this is not coming from a foreground caller, then we may want
    325                 // to delay the start if there are already other background services
    326                 // that are starting.  This is to avoid process start spam when lots
    327                 // of applications are all handling things like connectivity broadcasts.
    328                 // We only do this for cached processes, because otherwise an application
    329                 // can have assumptions about calling startService() for a service to run
    330                 // in its own process, and for that process to not be killed before the
    331                 // service is started.  This is especially the case for receivers, which
    332                 // may start a service in onReceive() to do some additional work and have
    333                 // initialized some global state as part of that.
    334                 if (DEBUG_DELAYED_SERVICE) Slog.v(TAG, "Potential start delay of " + r + " in "
    335                         + proc);
    336                 if (r.delayed) {
    337                     // This service is already scheduled for a delayed start; just leave
    338                     // it still waiting.
    339                     if (DEBUG_DELAYED_STATS) Slog.v(TAG, "Continuing to delay: " + r);
    340                     return r.name;
    341                 }
    342                 if (smap.mStartingBackground.size() >= mMaxStartingBackground) {
    343                     // Something else is starting, delay!
    344                     Slog.i(TAG, "Delaying start of: " + r);
    345                     smap.mDelayedStartList.add(r);
    346                     r.delayed = true;
    347                     return r.name;
    348                 }
    349                 if (DEBUG_DELAYED_STATS) Slog.v(TAG, "Not delaying: " + r);
    350                 addToStarting = true;
    351             } else if (proc.curProcState >= ActivityManager.PROCESS_STATE_SERVICE) {
    352                 // We slightly loosen when we will enqueue this new service as a background
    353                 // starting service we are waiting for, to also include processes that are
    354                 // currently running other services or receivers.
    355                 addToStarting = true;
    356                 if (DEBUG_DELAYED_STATS) Slog.v(TAG, "Not delaying, but counting as bg: " + r);
    357             } else if (DEBUG_DELAYED_STATS) {
    358                 StringBuilder sb = new StringBuilder(128);
    359                 sb.append("Not potential delay (state=").append(proc.curProcState)
    360                         .append(' ').append(proc.adjType);
    361                 String reason = proc.makeAdjReason();
    362                 if (reason != null) {
    363                     sb.append(' ');
    364                     sb.append(reason);
    365                 }
    366                 sb.append("): ");
    367                 sb.append(r.toString());
    368                 Slog.v(TAG, sb.toString());
    369             }
    370         } else if (DEBUG_DELAYED_STATS) {
    371             if (callerFg) {
    372                 Slog.v(TAG, "Not potential delay (callerFg=" + callerFg + " uid="
    373                         + callingUid + " pid=" + callingPid + "): " + r);
    374             } else if (r.app != null) {
    375                 Slog.v(TAG, "Not potential delay (cur app=" + r.app + "): " + r);
    376             } else {
    377                 Slog.v(TAG, "Not potential delay (user " + r.userId + " not started): " + r);
    378             }
    379         }
    380 
    381         return startServiceInnerLocked(smap, service, r, callerFg, addToStarting);
    382     }
    383 
    384     ComponentName startServiceInnerLocked(ServiceMap smap, Intent service,
    385             ServiceRecord r, boolean callerFg, boolean addToStarting) {
    386         ProcessStats.ServiceState stracker = r.getTracker();
    387         if (stracker != null) {
    388             stracker.setStarted(true, mAm.mProcessStats.getMemFactorLocked(), r.lastActivity);
    389         }
    390         r.callStart = false;
    391         synchronized (r.stats.getBatteryStats()) {
    392             r.stats.startRunningLocked();
    393         }
    394         String error = bringUpServiceLocked(r, service.getFlags(), callerFg, false);
    395         if (error != null) {
    396             return new ComponentName("!!", error);
    397         }
    398 
    399         if (r.startRequested && addToStarting) {
    400             boolean first = smap.mStartingBackground.size() == 0;
    401             smap.mStartingBackground.add(r);
    402             r.startingBgTimeout = SystemClock.uptimeMillis() + BG_START_TIMEOUT;
    403             if (DEBUG_DELAYED_SERVICE) {
    404                 RuntimeException here = new RuntimeException("here");
    405                 here.fillInStackTrace();
    406                 Slog.v(TAG, "Starting background (first=" + first + "): " + r, here);
    407             } else if (DEBUG_DELAYED_STATS) {
    408                 Slog.v(TAG, "Starting background (first=" + first + "): " + r);
    409             }
    410             if (first) {
    411                 smap.rescheduleDelayedStarts();
    412             }
    413         } else if (callerFg) {
    414             smap.ensureNotStartingBackground(r);
    415         }
    416 
    417         return r.name;
    418     }
    419 
    420     private void stopServiceLocked(ServiceRecord service) {
    421         if (service.delayed) {
    422             // If service isn't actually running, but is is being held in the
    423             // delayed list, then we need to keep it started but note that it
    424             // should be stopped once no longer delayed.
    425             if (DEBUG_DELAYED_STATS) Slog.v(TAG, "Delaying stop of pending: " + service);
    426             service.delayedStop = true;
    427             return;
    428         }
    429         synchronized (service.stats.getBatteryStats()) {
    430             service.stats.stopRunningLocked();
    431         }
    432         service.startRequested = false;
    433         if (service.tracker != null) {
    434             service.tracker.setStarted(false, mAm.mProcessStats.getMemFactorLocked(),
    435                     SystemClock.uptimeMillis());
    436         }
    437         service.callStart = false;
    438         bringDownServiceIfNeededLocked(service, false, false);
    439     }
    440 
    441     int stopServiceLocked(IApplicationThread caller, Intent service,
    442             String resolvedType, int userId) {
    443         if (DEBUG_SERVICE) Slog.v(TAG, "stopService: " + service
    444                 + " type=" + resolvedType);
    445 
    446         final ProcessRecord callerApp = mAm.getRecordForAppLocked(caller);
    447         if (caller != null && callerApp == null) {
    448             throw new SecurityException(
    449                     "Unable to find app for caller " + caller
    450                     + " (pid=" + Binder.getCallingPid()
    451                     + ") when stopping service " + service);
    452         }
    453 
    454         // If this service is active, make sure it is stopped.
    455         ServiceLookupResult r = retrieveServiceLocked(service, resolvedType,
    456                 Binder.getCallingPid(), Binder.getCallingUid(), userId, false, false);
    457         if (r != null) {
    458             if (r.record != null) {
    459                 final long origId = Binder.clearCallingIdentity();
    460                 try {
    461                     stopServiceLocked(r.record);
    462                 } finally {
    463                     Binder.restoreCallingIdentity(origId);
    464                 }
    465                 return 1;
    466             }
    467             return -1;
    468         }
    469 
    470         return 0;
    471     }
    472 
    473     IBinder peekServiceLocked(Intent service, String resolvedType) {
    474         ServiceLookupResult r = retrieveServiceLocked(service, resolvedType,
    475                 Binder.getCallingPid(), Binder.getCallingUid(),
    476                 UserHandle.getCallingUserId(), false, false);
    477 
    478         IBinder ret = null;
    479         if (r != null) {
    480             // r.record is null if findServiceLocked() failed the caller permission check
    481             if (r.record == null) {
    482                 throw new SecurityException(
    483                         "Permission Denial: Accessing service " + r.record.name
    484                         + " from pid=" + Binder.getCallingPid()
    485                         + ", uid=" + Binder.getCallingUid()
    486                         + " requires " + r.permission);
    487             }
    488             IntentBindRecord ib = r.record.bindings.get(r.record.intent);
    489             if (ib != null) {
    490                 ret = ib.binder;
    491             }
    492         }
    493 
    494         return ret;
    495     }
    496 
    497     boolean stopServiceTokenLocked(ComponentName className, IBinder token,
    498             int startId) {
    499         if (DEBUG_SERVICE) Slog.v(TAG, "stopServiceToken: " + className
    500                 + " " + token + " startId=" + startId);
    501         ServiceRecord r = findServiceLocked(className, token, UserHandle.getCallingUserId());
    502         if (r != null) {
    503             if (startId >= 0) {
    504                 // Asked to only stop if done with all work.  Note that
    505                 // to avoid leaks, we will take this as dropping all
    506                 // start items up to and including this one.
    507                 ServiceRecord.StartItem si = r.findDeliveredStart(startId, false);
    508                 if (si != null) {
    509                     while (r.deliveredStarts.size() > 0) {
    510                         ServiceRecord.StartItem cur = r.deliveredStarts.remove(0);
    511                         cur.removeUriPermissionsLocked();
    512                         if (cur == si) {
    513                             break;
    514                         }
    515                     }
    516                 }
    517 
    518                 if (r.getLastStartId() != startId) {
    519                     return false;
    520                 }
    521 
    522                 if (r.deliveredStarts.size() > 0) {
    523                     Slog.w(TAG, "stopServiceToken startId " + startId
    524                             + " is last, but have " + r.deliveredStarts.size()
    525                             + " remaining args");
    526                 }
    527             }
    528 
    529             synchronized (r.stats.getBatteryStats()) {
    530                 r.stats.stopRunningLocked();
    531             }
    532             r.startRequested = false;
    533             if (r.tracker != null) {
    534                 r.tracker.setStarted(false, mAm.mProcessStats.getMemFactorLocked(),
    535                         SystemClock.uptimeMillis());
    536             }
    537             r.callStart = false;
    538             final long origId = Binder.clearCallingIdentity();
    539             bringDownServiceIfNeededLocked(r, false, false);
    540             Binder.restoreCallingIdentity(origId);
    541             return true;
    542         }
    543         return false;
    544     }
    545 
    546     public void setServiceForegroundLocked(ComponentName className, IBinder token,
    547             int id, Notification notification, boolean removeNotification) {
    548         final int userId = UserHandle.getCallingUserId();
    549         final long origId = Binder.clearCallingIdentity();
    550         try {
    551             ServiceRecord r = findServiceLocked(className, token, userId);
    552             if (r != null) {
    553                 if (id != 0) {
    554                     if (notification == null) {
    555                         throw new IllegalArgumentException("null notification");
    556                     }
    557                     if (r.foregroundId != id) {
    558                         r.cancelNotification();
    559                         r.foregroundId = id;
    560                     }
    561                     notification.flags |= Notification.FLAG_FOREGROUND_SERVICE;
    562                     r.foregroundNoti = notification;
    563                     r.isForeground = true;
    564                     r.postNotification();
    565                     if (r.app != null) {
    566                         updateServiceForegroundLocked(r.app, true);
    567                     }
    568                     getServiceMap(r.userId).ensureNotStartingBackground(r);
    569                 } else {
    570                     if (r.isForeground) {
    571                         r.isForeground = false;
    572                         if (r.app != null) {
    573                             mAm.updateLruProcessLocked(r.app, false, null);
    574                             updateServiceForegroundLocked(r.app, true);
    575                         }
    576                     }
    577                     if (removeNotification) {
    578                         r.cancelNotification();
    579                         r.foregroundId = 0;
    580                         r.foregroundNoti = null;
    581                     }
    582                 }
    583             }
    584         } finally {
    585             Binder.restoreCallingIdentity(origId);
    586         }
    587     }
    588 
    589     private void updateServiceForegroundLocked(ProcessRecord proc, boolean oomAdj) {
    590         boolean anyForeground = false;
    591         for (int i=proc.services.size()-1; i>=0; i--) {
    592             ServiceRecord sr = proc.services.valueAt(i);
    593             if (sr.isForeground) {
    594                 anyForeground = true;
    595                 break;
    596             }
    597         }
    598         if (anyForeground != proc.foregroundServices) {
    599             proc.foregroundServices = anyForeground;
    600             if (oomAdj) {
    601                 mAm.updateOomAdjLocked();
    602             }
    603         }
    604     }
    605 
    606     private boolean updateServiceClientActivitiesLocked(ProcessRecord proc,
    607             ConnectionRecord modCr) {
    608         if (modCr != null && modCr.binding.client != null) {
    609             if (modCr.binding.client.activities.size() <= 0) {
    610                 // This connection is from a client without activities, so adding
    611                 // and removing is not interesting.
    612                 return false;
    613             }
    614         }
    615 
    616         boolean anyClientActivities = false;
    617         for (int i=proc.services.size()-1; i>=0 && !anyClientActivities; i--) {
    618             ServiceRecord sr = proc.services.valueAt(i);
    619             for (int conni=sr.connections.size()-1; conni>=0 && !anyClientActivities; conni--) {
    620                 ArrayList<ConnectionRecord> clist = sr.connections.valueAt(conni);
    621                 for (int cri=clist.size()-1; cri>=0; cri--) {
    622                     ConnectionRecord cr = clist.get(cri);
    623                     if (cr.binding.client == null || cr.binding.client == proc) {
    624                         // Binding to ourself is not interesting.
    625                         continue;
    626                     }
    627                     if (cr.binding.client.activities.size() > 0) {
    628                         anyClientActivities = true;
    629                         break;
    630                     }
    631                 }
    632             }
    633         }
    634         if (anyClientActivities != proc.hasClientActivities) {
    635             proc.hasClientActivities = anyClientActivities;
    636             mAm.updateLruProcessLocked(proc, anyClientActivities, null);
    637             return true;
    638         }
    639         return false;
    640     }
    641 
    642     int bindServiceLocked(IApplicationThread caller, IBinder token,
    643             Intent service, String resolvedType,
    644             IServiceConnection connection, int flags, int userId) {
    645         if (DEBUG_SERVICE) Slog.v(TAG, "bindService: " + service
    646                 + " type=" + resolvedType + " conn=" + connection.asBinder()
    647                 + " flags=0x" + Integer.toHexString(flags));
    648         final ProcessRecord callerApp = mAm.getRecordForAppLocked(caller);
    649         if (callerApp == null) {
    650             throw new SecurityException(
    651                     "Unable to find app for caller " + caller
    652                     + " (pid=" + Binder.getCallingPid()
    653                     + ") when binding service " + service);
    654         }
    655 
    656         ActivityRecord activity = null;
    657         if (token != null) {
    658             activity = ActivityRecord.isInStackLocked(token);
    659             if (activity == null) {
    660                 Slog.w(TAG, "Binding with unknown activity: " + token);
    661                 return 0;
    662             }
    663         }
    664 
    665         int clientLabel = 0;
    666         PendingIntent clientIntent = null;
    667 
    668         if (callerApp.info.uid == Process.SYSTEM_UID) {
    669             // Hacky kind of thing -- allow system stuff to tell us
    670             // what they are, so we can report this elsewhere for
    671             // others to know why certain services are running.
    672             try {
    673                 clientIntent = (PendingIntent)service.getParcelableExtra(
    674                         Intent.EXTRA_CLIENT_INTENT);
    675             } catch (RuntimeException e) {
    676             }
    677             if (clientIntent != null) {
    678                 clientLabel = service.getIntExtra(Intent.EXTRA_CLIENT_LABEL, 0);
    679                 if (clientLabel != 0) {
    680                     // There are no useful extras in the intent, trash them.
    681                     // System code calling with this stuff just needs to know
    682                     // this will happen.
    683                     service = service.cloneFilter();
    684                 }
    685             }
    686         }
    687 
    688         final boolean callerFg = callerApp.setSchedGroup != Process.THREAD_GROUP_BG_NONINTERACTIVE;
    689 
    690         ServiceLookupResult res =
    691             retrieveServiceLocked(service, resolvedType,
    692                     Binder.getCallingPid(), Binder.getCallingUid(), userId, true, callerFg);
    693         if (res == null) {
    694             return 0;
    695         }
    696         if (res.record == null) {
    697             return -1;
    698         }
    699         ServiceRecord s = res.record;
    700 
    701         final long origId = Binder.clearCallingIdentity();
    702 
    703         try {
    704             if (unscheduleServiceRestartLocked(s, callerApp.info.uid, false)) {
    705                 if (DEBUG_SERVICE) Slog.v(TAG, "BIND SERVICE WHILE RESTART PENDING: "
    706                         + s);
    707             }
    708 
    709             if ((flags&Context.BIND_AUTO_CREATE) != 0) {
    710                 s.lastActivity = SystemClock.uptimeMillis();
    711                 if (!s.hasAutoCreateConnections()) {
    712                     // This is the first binding, let the tracker know.
    713                     ProcessStats.ServiceState stracker = s.getTracker();
    714                     if (stracker != null) {
    715                         stracker.setBound(true, mAm.mProcessStats.getMemFactorLocked(),
    716                                 s.lastActivity);
    717                     }
    718                 }
    719             }
    720 
    721             AppBindRecord b = s.retrieveAppBindingLocked(service, callerApp);
    722             ConnectionRecord c = new ConnectionRecord(b, activity,
    723                     connection, flags, clientLabel, clientIntent);
    724 
    725             IBinder binder = connection.asBinder();
    726             ArrayList<ConnectionRecord> clist = s.connections.get(binder);
    727             if (clist == null) {
    728                 clist = new ArrayList<ConnectionRecord>();
    729                 s.connections.put(binder, clist);
    730             }
    731             clist.add(c);
    732             b.connections.add(c);
    733             if (activity != null) {
    734                 if (activity.connections == null) {
    735                     activity.connections = new HashSet<ConnectionRecord>();
    736                 }
    737                 activity.connections.add(c);
    738             }
    739             b.client.connections.add(c);
    740             if ((c.flags&Context.BIND_ABOVE_CLIENT) != 0) {
    741                 b.client.hasAboveClient = true;
    742             }
    743             if (s.app != null) {
    744                 updateServiceClientActivitiesLocked(s.app, c);
    745             }
    746             clist = mServiceConnections.get(binder);
    747             if (clist == null) {
    748                 clist = new ArrayList<ConnectionRecord>();
    749                 mServiceConnections.put(binder, clist);
    750             }
    751             clist.add(c);
    752 
    753             if ((flags&Context.BIND_AUTO_CREATE) != 0) {
    754                 s.lastActivity = SystemClock.uptimeMillis();
    755                 if (bringUpServiceLocked(s, service.getFlags(), callerFg, false) != null) {
    756                     return 0;
    757                 }
    758             }
    759 
    760             if (s.app != null) {
    761                 // This could have made the service more important.
    762                 mAm.updateLruProcessLocked(s.app, s.app.hasClientActivities, b.client);
    763                 mAm.updateOomAdjLocked(s.app);
    764             }
    765 
    766             if (DEBUG_SERVICE) Slog.v(TAG, "Bind " + s + " with " + b
    767                     + ": received=" + b.intent.received
    768                     + " apps=" + b.intent.apps.size()
    769                     + " doRebind=" + b.intent.doRebind);
    770 
    771             if (s.app != null && b.intent.received) {
    772                 // Service is already running, so we can immediately
    773                 // publish the connection.
    774                 try {
    775                     c.conn.connected(s.name, b.intent.binder);
    776                 } catch (Exception e) {
    777                     Slog.w(TAG, "Failure sending service " + s.shortName
    778                             + " to connection " + c.conn.asBinder()
    779                             + " (in " + c.binding.client.processName + ")", e);
    780                 }
    781 
    782                 // If this is the first app connected back to this binding,
    783                 // and the service had previously asked to be told when
    784                 // rebound, then do so.
    785                 if (b.intent.apps.size() == 1 && b.intent.doRebind) {
    786                     requestServiceBindingLocked(s, b.intent, callerFg, true);
    787                 }
    788             } else if (!b.intent.requested) {
    789                 requestServiceBindingLocked(s, b.intent, callerFg, false);
    790             }
    791 
    792             getServiceMap(s.userId).ensureNotStartingBackground(s);
    793 
    794         } finally {
    795             Binder.restoreCallingIdentity(origId);
    796         }
    797 
    798         return 1;
    799     }
    800 
    801     void publishServiceLocked(ServiceRecord r, Intent intent, IBinder service) {
    802         final long origId = Binder.clearCallingIdentity();
    803         try {
    804             if (DEBUG_SERVICE) Slog.v(TAG, "PUBLISHING " + r
    805                     + " " + intent + ": " + service);
    806             if (r != null) {
    807                 Intent.FilterComparison filter
    808                         = new Intent.FilterComparison(intent);
    809                 IntentBindRecord b = r.bindings.get(filter);
    810                 if (b != null && !b.received) {
    811                     b.binder = service;
    812                     b.requested = true;
    813                     b.received = true;
    814                     for (int conni=r.connections.size()-1; conni>=0; conni--) {
    815                         ArrayList<ConnectionRecord> clist = r.connections.valueAt(conni);
    816                         for (int i=0; i<clist.size(); i++) {
    817                             ConnectionRecord c = clist.get(i);
    818                             if (!filter.equals(c.binding.intent.intent)) {
    819                                 if (DEBUG_SERVICE) Slog.v(
    820                                         TAG, "Not publishing to: " + c);
    821                                 if (DEBUG_SERVICE) Slog.v(
    822                                         TAG, "Bound intent: " + c.binding.intent.intent);
    823                                 if (DEBUG_SERVICE) Slog.v(
    824                                         TAG, "Published intent: " + intent);
    825                                 continue;
    826                             }
    827                             if (DEBUG_SERVICE) Slog.v(TAG, "Publishing to: " + c);
    828                             try {
    829                                 c.conn.connected(r.name, service);
    830                             } catch (Exception e) {
    831                                 Slog.w(TAG, "Failure sending service " + r.name +
    832                                       " to connection " + c.conn.asBinder() +
    833                                       " (in " + c.binding.client.processName + ")", e);
    834                             }
    835                         }
    836                     }
    837                 }
    838 
    839                 serviceDoneExecutingLocked(r, mDestroyingServices.contains(r), false);
    840             }
    841         } finally {
    842             Binder.restoreCallingIdentity(origId);
    843         }
    844     }
    845 
    846     boolean unbindServiceLocked(IServiceConnection connection) {
    847         IBinder binder = connection.asBinder();
    848         if (DEBUG_SERVICE) Slog.v(TAG, "unbindService: conn=" + binder);
    849         ArrayList<ConnectionRecord> clist = mServiceConnections.get(binder);
    850         if (clist == null) {
    851             Slog.w(TAG, "Unbind failed: could not find connection for "
    852                   + connection.asBinder());
    853             return false;
    854         }
    855 
    856         final long origId = Binder.clearCallingIdentity();
    857         try {
    858             while (clist.size() > 0) {
    859                 ConnectionRecord r = clist.get(0);
    860                 removeConnectionLocked(r, null, null);
    861 
    862                 if (r.binding.service.app != null) {
    863                     // This could have made the service less important.
    864                     mAm.updateOomAdjLocked(r.binding.service.app);
    865                 }
    866             }
    867         } finally {
    868             Binder.restoreCallingIdentity(origId);
    869         }
    870 
    871         return true;
    872     }
    873 
    874     void unbindFinishedLocked(ServiceRecord r, Intent intent, boolean doRebind) {
    875         final long origId = Binder.clearCallingIdentity();
    876         try {
    877             if (r != null) {
    878                 Intent.FilterComparison filter
    879                         = new Intent.FilterComparison(intent);
    880                 IntentBindRecord b = r.bindings.get(filter);
    881                 if (DEBUG_SERVICE) Slog.v(TAG, "unbindFinished in " + r
    882                         + " at " + b + ": apps="
    883                         + (b != null ? b.apps.size() : 0));
    884 
    885                 boolean inDestroying = mDestroyingServices.contains(r);
    886                 if (b != null) {
    887                     if (b.apps.size() > 0 && !inDestroying) {
    888                         // Applications have already bound since the last
    889                         // unbind, so just rebind right here.
    890                         boolean inFg = false;
    891                         for (int i=b.apps.size()-1; i>=0; i--) {
    892                             ProcessRecord client = b.apps.valueAt(i).client;
    893                             if (client != null && client.setSchedGroup
    894                                     != Process.THREAD_GROUP_BG_NONINTERACTIVE) {
    895                                 inFg = true;
    896                                 break;
    897                             }
    898                         }
    899                         requestServiceBindingLocked(r, b, inFg, true);
    900                     } else {
    901                         // Note to tell the service the next time there is
    902                         // a new client.
    903                         b.doRebind = true;
    904                     }
    905                 }
    906 
    907                 serviceDoneExecutingLocked(r, inDestroying, false);
    908             }
    909         } finally {
    910             Binder.restoreCallingIdentity(origId);
    911         }
    912     }
    913 
    914     private final ServiceRecord findServiceLocked(ComponentName name,
    915             IBinder token, int userId) {
    916         ServiceRecord r = getServiceByName(name, userId);
    917         return r == token ? r : null;
    918     }
    919 
    920     private final class ServiceLookupResult {
    921         final ServiceRecord record;
    922         final String permission;
    923 
    924         ServiceLookupResult(ServiceRecord _record, String _permission) {
    925             record = _record;
    926             permission = _permission;
    927         }
    928     }
    929 
    930     private class ServiceRestarter implements Runnable {
    931         private ServiceRecord mService;
    932 
    933         void setService(ServiceRecord service) {
    934             mService = service;
    935         }
    936 
    937         public void run() {
    938             synchronized(mAm) {
    939                 performServiceRestartLocked(mService);
    940             }
    941         }
    942     }
    943 
    944     private ServiceLookupResult retrieveServiceLocked(Intent service,
    945             String resolvedType, int callingPid, int callingUid, int userId,
    946             boolean createIfNeeded, boolean callingFromFg) {
    947         ServiceRecord r = null;
    948         if (DEBUG_SERVICE) Slog.v(TAG, "retrieveServiceLocked: " + service
    949                 + " type=" + resolvedType + " callingUid=" + callingUid);
    950 
    951         userId = mAm.handleIncomingUser(callingPid, callingUid, userId,
    952                 false, true, "service", null);
    953 
    954         ServiceMap smap = getServiceMap(userId);
    955         final ComponentName comp = service.getComponent();
    956         if (comp != null) {
    957             r = smap.mServicesByName.get(comp);
    958         }
    959         if (r == null) {
    960             Intent.FilterComparison filter = new Intent.FilterComparison(service);
    961             r = smap.mServicesByIntent.get(filter);
    962         }
    963         if (r == null) {
    964             try {
    965                 ResolveInfo rInfo =
    966                     AppGlobals.getPackageManager().resolveService(
    967                                 service, resolvedType,
    968                                 ActivityManagerService.STOCK_PM_FLAGS, userId);
    969                 ServiceInfo sInfo =
    970                     rInfo != null ? rInfo.serviceInfo : null;
    971                 if (sInfo == null) {
    972                     Slog.w(TAG, "Unable to start service " + service + " U=" + userId +
    973                           ": not found");
    974                     return null;
    975                 }
    976                 ComponentName name = new ComponentName(
    977                         sInfo.applicationInfo.packageName, sInfo.name);
    978                 if (userId > 0) {
    979                     if (mAm.isSingleton(sInfo.processName, sInfo.applicationInfo,
    980                             sInfo.name, sInfo.flags)) {
    981                         userId = 0;
    982                         smap = getServiceMap(0);
    983                     }
    984                     sInfo = new ServiceInfo(sInfo);
    985                     sInfo.applicationInfo = mAm.getAppInfoForUser(sInfo.applicationInfo, userId);
    986                 }
    987                 r = smap.mServicesByName.get(name);
    988                 if (r == null && createIfNeeded) {
    989                     Intent.FilterComparison filter
    990                             = new Intent.FilterComparison(service.cloneFilter());
    991                     ServiceRestarter res = new ServiceRestarter();
    992                     BatteryStatsImpl.Uid.Pkg.Serv ss = null;
    993                     BatteryStatsImpl stats = mAm.mBatteryStatsService.getActiveStatistics();
    994                     synchronized (stats) {
    995                         ss = stats.getServiceStatsLocked(
    996                                 sInfo.applicationInfo.uid, sInfo.packageName,
    997                                 sInfo.name);
    998                     }
    999                     r = new ServiceRecord(mAm, ss, name, filter, sInfo, callingFromFg, res);
   1000                     res.setService(r);
   1001                     smap.mServicesByName.put(name, r);
   1002                     smap.mServicesByIntent.put(filter, r);
   1003 
   1004                     // Make sure this component isn't in the pending list.
   1005                     for (int i=mPendingServices.size()-1; i>=0; i--) {
   1006                         ServiceRecord pr = mPendingServices.get(i);
   1007                         if (pr.serviceInfo.applicationInfo.uid == sInfo.applicationInfo.uid
   1008                                 && pr.name.equals(name)) {
   1009                             mPendingServices.remove(i);
   1010                         }
   1011                     }
   1012                 }
   1013             } catch (RemoteException ex) {
   1014                 // pm is in same process, this will never happen.
   1015             }
   1016         }
   1017         if (r != null) {
   1018             if (mAm.checkComponentPermission(r.permission,
   1019                     callingPid, callingUid, r.appInfo.uid, r.exported)
   1020                     != PackageManager.PERMISSION_GRANTED) {
   1021                 if (!r.exported) {
   1022                     Slog.w(TAG, "Permission Denial: Accessing service " + r.name
   1023                             + " from pid=" + callingPid
   1024                             + ", uid=" + callingUid
   1025                             + " that is not exported from uid " + r.appInfo.uid);
   1026                     return new ServiceLookupResult(null, "not exported from uid "
   1027                             + r.appInfo.uid);
   1028                 }
   1029                 Slog.w(TAG, "Permission Denial: Accessing service " + r.name
   1030                         + " from pid=" + callingPid
   1031                         + ", uid=" + callingUid
   1032                         + " requires " + r.permission);
   1033                 return new ServiceLookupResult(null, r.permission);
   1034             }
   1035             if (!mAm.mIntentFirewall.checkService(r.name, service, callingUid, callingPid,
   1036                     resolvedType, r.appInfo)) {
   1037                 return null;
   1038             }
   1039             return new ServiceLookupResult(r, null);
   1040         }
   1041         return null;
   1042     }
   1043 
   1044     private final void bumpServiceExecutingLocked(ServiceRecord r, boolean fg, String why) {
   1045         if (DEBUG_SERVICE) Slog.v(TAG, ">>> EXECUTING "
   1046                 + why + " of " + r + " in app " + r.app);
   1047         else if (DEBUG_SERVICE_EXECUTING) Slog.v(TAG, ">>> EXECUTING "
   1048                 + why + " of " + r.shortName);
   1049         long now = SystemClock.uptimeMillis();
   1050         if (r.executeNesting == 0) {
   1051             r.executeFg = fg;
   1052             ProcessStats.ServiceState stracker = r.getTracker();
   1053             if (stracker != null) {
   1054                 stracker.setExecuting(true, mAm.mProcessStats.getMemFactorLocked(), now);
   1055             }
   1056             if (r.app != null) {
   1057                 r.app.executingServices.add(r);
   1058                 r.app.execServicesFg |= fg;
   1059                 if (r.app.executingServices.size() == 1) {
   1060                     scheduleServiceTimeoutLocked(r.app);
   1061                 }
   1062             }
   1063         } else if (r.app != null && fg && !r.app.execServicesFg) {
   1064             r.app.execServicesFg = true;
   1065             scheduleServiceTimeoutLocked(r.app);
   1066         }
   1067         r.executeFg |= fg;
   1068         r.executeNesting++;
   1069         r.executingStart = now;
   1070     }
   1071 
   1072     private final boolean requestServiceBindingLocked(ServiceRecord r,
   1073             IntentBindRecord i, boolean execInFg, boolean rebind) {
   1074         if (r.app == null || r.app.thread == null) {
   1075             // If service is not currently running, can't yet bind.
   1076             return false;
   1077         }
   1078         if ((!i.requested || rebind) && i.apps.size() > 0) {
   1079             try {
   1080                 bumpServiceExecutingLocked(r, execInFg, "bind");
   1081                 r.app.forceProcessStateUpTo(ActivityManager.PROCESS_STATE_SERVICE);
   1082                 r.app.thread.scheduleBindService(r, i.intent.getIntent(), rebind,
   1083                         r.app.repProcState);
   1084                 if (!rebind) {
   1085                     i.requested = true;
   1086                 }
   1087                 i.hasBound = true;
   1088                 i.doRebind = false;
   1089             } catch (RemoteException e) {
   1090                 if (DEBUG_SERVICE) Slog.v(TAG, "Crashed while binding " + r);
   1091                 return false;
   1092             }
   1093         }
   1094         return true;
   1095     }
   1096 
   1097     private final boolean scheduleServiceRestartLocked(ServiceRecord r,
   1098             boolean allowCancel) {
   1099         boolean canceled = false;
   1100 
   1101         ServiceMap smap = getServiceMap(r.userId);
   1102         if (smap.mServicesByName.get(r.name) != r) {
   1103             ServiceRecord cur = smap.mServicesByName.get(r.name);
   1104             Slog.wtf(TAG, "Attempting to schedule restart of " + r
   1105                     + " when found in map: " + cur);
   1106             return false;
   1107         }
   1108 
   1109         final long now = SystemClock.uptimeMillis();
   1110 
   1111         if ((r.serviceInfo.applicationInfo.flags
   1112                 &ApplicationInfo.FLAG_PERSISTENT) == 0) {
   1113             long minDuration = SERVICE_RESTART_DURATION;
   1114             long resetTime = SERVICE_RESET_RUN_DURATION;
   1115 
   1116             // Any delivered but not yet finished starts should be put back
   1117             // on the pending list.
   1118             final int N = r.deliveredStarts.size();
   1119             if (N > 0) {
   1120                 for (int i=N-1; i>=0; i--) {
   1121                     ServiceRecord.StartItem si = r.deliveredStarts.get(i);
   1122                     si.removeUriPermissionsLocked();
   1123                     if (si.intent == null) {
   1124                         // We'll generate this again if needed.
   1125                     } else if (!allowCancel || (si.deliveryCount < ServiceRecord.MAX_DELIVERY_COUNT
   1126                             && si.doneExecutingCount < ServiceRecord.MAX_DONE_EXECUTING_COUNT)) {
   1127                         r.pendingStarts.add(0, si);
   1128                         long dur = SystemClock.uptimeMillis() - si.deliveredTime;
   1129                         dur *= 2;
   1130                         if (minDuration < dur) minDuration = dur;
   1131                         if (resetTime < dur) resetTime = dur;
   1132                     } else {
   1133                         Slog.w(TAG, "Canceling start item " + si.intent + " in service "
   1134                                 + r.name);
   1135                         canceled = true;
   1136                     }
   1137                 }
   1138                 r.deliveredStarts.clear();
   1139             }
   1140 
   1141             r.totalRestartCount++;
   1142             if (r.restartDelay == 0) {
   1143                 r.restartCount++;
   1144                 r.restartDelay = minDuration;
   1145             } else {
   1146                 // If it has been a "reasonably long time" since the service
   1147                 // was started, then reset our restart duration back to
   1148                 // the beginning, so we don't infinitely increase the duration
   1149                 // on a service that just occasionally gets killed (which is
   1150                 // a normal case, due to process being killed to reclaim memory).
   1151                 if (now > (r.restartTime+resetTime)) {
   1152                     r.restartCount = 1;
   1153                     r.restartDelay = minDuration;
   1154                 } else {
   1155                     r.restartDelay *= SERVICE_RESTART_DURATION_FACTOR;
   1156                     if (r.restartDelay < minDuration) {
   1157                         r.restartDelay = minDuration;
   1158                     }
   1159                 }
   1160             }
   1161 
   1162             r.nextRestartTime = now + r.restartDelay;
   1163 
   1164             // Make sure that we don't end up restarting a bunch of services
   1165             // all at the same time.
   1166             boolean repeat;
   1167             do {
   1168                 repeat = false;
   1169                 for (int i=mRestartingServices.size()-1; i>=0; i--) {
   1170                     ServiceRecord r2 = mRestartingServices.get(i);
   1171                     if (r2 != r && r.nextRestartTime
   1172                             >= (r2.nextRestartTime-SERVICE_MIN_RESTART_TIME_BETWEEN)
   1173                             && r.nextRestartTime
   1174                             < (r2.nextRestartTime+SERVICE_MIN_RESTART_TIME_BETWEEN)) {
   1175                         r.nextRestartTime = r2.nextRestartTime + SERVICE_MIN_RESTART_TIME_BETWEEN;
   1176                         r.restartDelay = r.nextRestartTime - now;
   1177                         repeat = true;
   1178                         break;
   1179                     }
   1180                 }
   1181             } while (repeat);
   1182 
   1183         } else {
   1184             // Persistent processes are immediately restarted, so there is no
   1185             // reason to hold of on restarting their services.
   1186             r.totalRestartCount++;
   1187             r.restartCount = 0;
   1188             r.restartDelay = 0;
   1189             r.nextRestartTime = now;
   1190         }
   1191 
   1192         if (!mRestartingServices.contains(r)) {
   1193             r.createdFromFg = false;
   1194             mRestartingServices.add(r);
   1195             r.makeRestarting(mAm.mProcessStats.getMemFactorLocked(), now);
   1196         }
   1197 
   1198         r.cancelNotification();
   1199 
   1200         mAm.mHandler.removeCallbacks(r.restarter);
   1201         mAm.mHandler.postAtTime(r.restarter, r.nextRestartTime);
   1202         r.nextRestartTime = SystemClock.uptimeMillis() + r.restartDelay;
   1203         Slog.w(TAG, "Scheduling restart of crashed service "
   1204                 + r.shortName + " in " + r.restartDelay + "ms");
   1205         EventLog.writeEvent(EventLogTags.AM_SCHEDULE_SERVICE_RESTART,
   1206                 r.userId, r.shortName, r.restartDelay);
   1207 
   1208         return canceled;
   1209     }
   1210 
   1211     final void performServiceRestartLocked(ServiceRecord r) {
   1212         if (!mRestartingServices.contains(r)) {
   1213             return;
   1214         }
   1215         bringUpServiceLocked(r, r.intent.getIntent().getFlags(), r.createdFromFg, true);
   1216     }
   1217 
   1218     private final boolean unscheduleServiceRestartLocked(ServiceRecord r, int callingUid,
   1219             boolean force) {
   1220         if (!force && r.restartDelay == 0) {
   1221             return false;
   1222         }
   1223         // Remove from the restarting list; if the service is currently on the
   1224         // restarting list, or the call is coming from another app, then this
   1225         // service has become of much more interest so we reset the restart interval.
   1226         boolean removed = mRestartingServices.remove(r);
   1227         if (removed || callingUid != r.appInfo.uid) {
   1228             r.resetRestartCounter();
   1229         }
   1230         if (removed) {
   1231             clearRestartingIfNeededLocked(r);
   1232         }
   1233         mAm.mHandler.removeCallbacks(r.restarter);
   1234         return true;
   1235     }
   1236 
   1237     private void clearRestartingIfNeededLocked(ServiceRecord r) {
   1238         if (r.restartTracker != null) {
   1239             // If this is the last restarting record with this tracker, then clear
   1240             // the tracker's restarting state.
   1241             boolean stillTracking = false;
   1242             for (int i=mRestartingServices.size()-1; i>=0; i--) {
   1243                 if (mRestartingServices.get(i).restartTracker == r.restartTracker) {
   1244                     stillTracking = true;
   1245                     break;
   1246                 }
   1247             }
   1248             if (!stillTracking) {
   1249                 r.restartTracker.setRestarting(false, mAm.mProcessStats.getMemFactorLocked(),
   1250                         SystemClock.uptimeMillis());
   1251                 r.restartTracker = null;
   1252             }
   1253         }
   1254     }
   1255 
   1256     private final String bringUpServiceLocked(ServiceRecord r,
   1257             int intentFlags, boolean execInFg, boolean whileRestarting) {
   1258         //Slog.i(TAG, "Bring up service:");
   1259         //r.dump("  ");
   1260 
   1261         if (r.app != null && r.app.thread != null) {
   1262             sendServiceArgsLocked(r, execInFg, false);
   1263             return null;
   1264         }
   1265 
   1266         if (!whileRestarting && r.restartDelay > 0) {
   1267             // If waiting for a restart, then do nothing.
   1268             return null;
   1269         }
   1270 
   1271         if (DEBUG_SERVICE) Slog.v(TAG, "Bringing up " + r + " " + r.intent);
   1272 
   1273         // We are now bringing the service up, so no longer in the
   1274         // restarting state.
   1275         if (mRestartingServices.remove(r)) {
   1276             clearRestartingIfNeededLocked(r);
   1277         }
   1278 
   1279         // Make sure this service is no longer considered delayed, we are starting it now.
   1280         if (r.delayed) {
   1281             if (DEBUG_DELAYED_STATS) Slog.v(TAG, "REM FR DELAY LIST (bring up): " + r);
   1282             getServiceMap(r.userId).mDelayedStartList.remove(r);
   1283             r.delayed = false;
   1284         }
   1285 
   1286         // Make sure that the user who owns this service is started.  If not,
   1287         // we don't want to allow it to run.
   1288         if (mAm.mStartedUsers.get(r.userId) == null) {
   1289             String msg = "Unable to launch app "
   1290                     + r.appInfo.packageName + "/"
   1291                     + r.appInfo.uid + " for service "
   1292                     + r.intent.getIntent() + ": user " + r.userId + " is stopped";
   1293             Slog.w(TAG, msg);
   1294             bringDownServiceLocked(r);
   1295             return msg;
   1296         }
   1297 
   1298         // Service is now being launched, its package can't be stopped.
   1299         try {
   1300             AppGlobals.getPackageManager().setPackageStoppedState(
   1301                     r.packageName, false, r.userId);
   1302         } catch (RemoteException e) {
   1303         } catch (IllegalArgumentException e) {
   1304             Slog.w(TAG, "Failed trying to unstop package "
   1305                     + r.packageName + ": " + e);
   1306         }
   1307 
   1308         final boolean isolated = (r.serviceInfo.flags&ServiceInfo.FLAG_ISOLATED_PROCESS) != 0;
   1309         final String procName = r.processName;
   1310         ProcessRecord app;
   1311 
   1312         if (!isolated) {
   1313             app = mAm.getProcessRecordLocked(procName, r.appInfo.uid, false);
   1314             if (DEBUG_MU) Slog.v(TAG_MU, "bringUpServiceLocked: appInfo.uid=" + r.appInfo.uid
   1315                         + " app=" + app);
   1316             if (app != null && app.thread != null) {
   1317                 try {
   1318                     app.addPackage(r.appInfo.packageName, mAm.mProcessStats);
   1319                     realStartServiceLocked(r, app, execInFg);
   1320                     return null;
   1321                 } catch (RemoteException e) {
   1322                     Slog.w(TAG, "Exception when starting service " + r.shortName, e);
   1323                 }
   1324 
   1325                 // If a dead object exception was thrown -- fall through to
   1326                 // restart the application.
   1327             }
   1328         } else {
   1329             // If this service runs in an isolated process, then each time
   1330             // we call startProcessLocked() we will get a new isolated
   1331             // process, starting another process if we are currently waiting
   1332             // for a previous process to come up.  To deal with this, we store
   1333             // in the service any current isolated process it is running in or
   1334             // waiting to have come up.
   1335             app = r.isolatedProc;
   1336         }
   1337 
   1338         // Not running -- get it started, and enqueue this service record
   1339         // to be executed when the app comes up.
   1340         if (app == null) {
   1341             if ((app=mAm.startProcessLocked(procName, r.appInfo, true, intentFlags,
   1342                     "service", r.name, false, isolated, false)) == null) {
   1343                 String msg = "Unable to launch app "
   1344                         + r.appInfo.packageName + "/"
   1345                         + r.appInfo.uid + " for service "
   1346                         + r.intent.getIntent() + ": process is bad";
   1347                 Slog.w(TAG, msg);
   1348                 bringDownServiceLocked(r);
   1349                 return msg;
   1350             }
   1351             if (isolated) {
   1352                 r.isolatedProc = app;
   1353             }
   1354         }
   1355 
   1356         if (!mPendingServices.contains(r)) {
   1357             mPendingServices.add(r);
   1358         }
   1359 
   1360         if (r.delayedStop) {
   1361             // Oh and hey we've already been asked to stop!
   1362             r.delayedStop = false;
   1363             if (r.startRequested) {
   1364                 if (DEBUG_DELAYED_STATS) Slog.v(TAG, "Applying delayed stop (in bring up): " + r);
   1365                 stopServiceLocked(r);
   1366             }
   1367         }
   1368 
   1369         return null;
   1370     }
   1371 
   1372     private final void requestServiceBindingsLocked(ServiceRecord r, boolean execInFg) {
   1373         for (int i=r.bindings.size()-1; i>=0; i--) {
   1374             IntentBindRecord ibr = r.bindings.valueAt(i);
   1375             if (!requestServiceBindingLocked(r, ibr, execInFg, false)) {
   1376                 break;
   1377             }
   1378         }
   1379     }
   1380 
   1381     private final void realStartServiceLocked(ServiceRecord r,
   1382             ProcessRecord app, boolean execInFg) throws RemoteException {
   1383         if (app.thread == null) {
   1384             throw new RemoteException();
   1385         }
   1386         if (DEBUG_MU)
   1387             Slog.v(TAG_MU, "realStartServiceLocked, ServiceRecord.uid = " + r.appInfo.uid
   1388                     + ", ProcessRecord.uid = " + app.uid);
   1389         r.app = app;
   1390         r.restartTime = r.lastActivity = SystemClock.uptimeMillis();
   1391 
   1392         app.services.add(r);
   1393         bumpServiceExecutingLocked(r, execInFg, "create");
   1394         mAm.updateLruProcessLocked(app, false, null);
   1395         mAm.updateOomAdjLocked();
   1396 
   1397         boolean created = false;
   1398         try {
   1399             String nameTerm;
   1400             int lastPeriod = r.shortName.lastIndexOf('.');
   1401             nameTerm = lastPeriod >= 0 ? r.shortName.substring(lastPeriod) : r.shortName;
   1402             EventLogTags.writeAmCreateService(
   1403                     r.userId, System.identityHashCode(r), nameTerm, r.app.uid, r.app.pid);
   1404             synchronized (r.stats.getBatteryStats()) {
   1405                 r.stats.startLaunchedLocked();
   1406             }
   1407             mAm.ensurePackageDexOpt(r.serviceInfo.packageName);
   1408             app.forceProcessStateUpTo(ActivityManager.PROCESS_STATE_SERVICE);
   1409             app.thread.scheduleCreateService(r, r.serviceInfo,
   1410                     mAm.compatibilityInfoForPackageLocked(r.serviceInfo.applicationInfo),
   1411                     app.repProcState);
   1412             r.postNotification();
   1413             created = true;
   1414         } finally {
   1415             if (!created) {
   1416                 app.services.remove(r);
   1417                 r.app = null;
   1418                 scheduleServiceRestartLocked(r, false);
   1419             }
   1420         }
   1421 
   1422         requestServiceBindingsLocked(r, execInFg);
   1423 
   1424         // If the service is in the started state, and there are no
   1425         // pending arguments, then fake up one so its onStartCommand() will
   1426         // be called.
   1427         if (r.startRequested && r.callStart && r.pendingStarts.size() == 0) {
   1428             r.pendingStarts.add(new ServiceRecord.StartItem(r, false, r.makeNextStartId(),
   1429                     null, null));
   1430         }
   1431 
   1432         sendServiceArgsLocked(r, execInFg, true);
   1433 
   1434         if (r.delayed) {
   1435             if (DEBUG_DELAYED_STATS) Slog.v(TAG, "REM FR DELAY LIST (new proc): " + r);
   1436             getServiceMap(r.userId).mDelayedStartList.remove(r);
   1437             r.delayed = false;
   1438         }
   1439 
   1440         if (r.delayedStop) {
   1441             // Oh and hey we've already been asked to stop!
   1442             r.delayedStop = false;
   1443             if (r.startRequested) {
   1444                 if (DEBUG_DELAYED_STATS) Slog.v(TAG, "Applying delayed stop (from start): " + r);
   1445                 stopServiceLocked(r);
   1446             }
   1447         }
   1448     }
   1449 
   1450     private final void sendServiceArgsLocked(ServiceRecord r, boolean execInFg,
   1451             boolean oomAdjusted) {
   1452         final int N = r.pendingStarts.size();
   1453         if (N == 0) {
   1454             return;
   1455         }
   1456 
   1457         while (r.pendingStarts.size() > 0) {
   1458             try {
   1459                 ServiceRecord.StartItem si = r.pendingStarts.remove(0);
   1460                 if (DEBUG_SERVICE) Slog.v(TAG, "Sending arguments to: "
   1461                         + r + " " + r.intent + " args=" + si.intent);
   1462                 if (si.intent == null && N > 1) {
   1463                     // If somehow we got a dummy null intent in the middle,
   1464                     // then skip it.  DO NOT skip a null intent when it is
   1465                     // the only one in the list -- this is to support the
   1466                     // onStartCommand(null) case.
   1467                     continue;
   1468                 }
   1469                 si.deliveredTime = SystemClock.uptimeMillis();
   1470                 r.deliveredStarts.add(si);
   1471                 si.deliveryCount++;
   1472                 if (si.neededGrants != null) {
   1473                     mAm.grantUriPermissionUncheckedFromIntentLocked(si.neededGrants,
   1474                             si.getUriPermissionsLocked());
   1475                 }
   1476                 bumpServiceExecutingLocked(r, execInFg, "start");
   1477                 if (!oomAdjusted) {
   1478                     oomAdjusted = true;
   1479                     mAm.updateOomAdjLocked(r.app);
   1480                 }
   1481                 int flags = 0;
   1482                 if (si.deliveryCount > 1) {
   1483                     flags |= Service.START_FLAG_RETRY;
   1484                 }
   1485                 if (si.doneExecutingCount > 0) {
   1486                     flags |= Service.START_FLAG_REDELIVERY;
   1487                 }
   1488                 r.app.thread.scheduleServiceArgs(r, si.taskRemoved, si.id, flags, si.intent);
   1489             } catch (RemoteException e) {
   1490                 // Remote process gone...  we'll let the normal cleanup take
   1491                 // care of this.
   1492                 if (DEBUG_SERVICE) Slog.v(TAG, "Crashed while scheduling start: " + r);
   1493                 break;
   1494             } catch (Exception e) {
   1495                 Slog.w(TAG, "Unexpected exception", e);
   1496                 break;
   1497             }
   1498         }
   1499     }
   1500 
   1501     private final boolean isServiceNeeded(ServiceRecord r, boolean knowConn, boolean hasConn) {
   1502         // Are we still explicitly being asked to run?
   1503         if (r.startRequested) {
   1504             return true;
   1505         }
   1506 
   1507         // Is someone still bound to us keepign us running?
   1508         if (!knowConn) {
   1509             hasConn = r.hasAutoCreateConnections();
   1510         }
   1511         if (hasConn) {
   1512             return true;
   1513         }
   1514 
   1515         return false;
   1516     }
   1517 
   1518     private final void bringDownServiceIfNeededLocked(ServiceRecord r, boolean knowConn,
   1519             boolean hasConn) {
   1520         //Slog.i(TAG, "Bring down service:");
   1521         //r.dump("  ");
   1522 
   1523         if (isServiceNeeded(r, knowConn, hasConn)) {
   1524             return;
   1525         }
   1526 
   1527         // Are we in the process of launching?
   1528         if (mPendingServices.contains(r)) {
   1529             return;
   1530         }
   1531 
   1532         bringDownServiceLocked(r);
   1533     }
   1534 
   1535     private final void bringDownServiceLocked(ServiceRecord r) {
   1536         //Slog.i(TAG, "Bring down service:");
   1537         //r.dump("  ");
   1538 
   1539         // Report to all of the connections that the service is no longer
   1540         // available.
   1541         for (int conni=r.connections.size()-1; conni>=0; conni--) {
   1542             ArrayList<ConnectionRecord> c = r.connections.valueAt(conni);
   1543             for (int i=0; i<c.size(); i++) {
   1544                 ConnectionRecord cr = c.get(i);
   1545                 // There is still a connection to the service that is
   1546                 // being brought down.  Mark it as dead.
   1547                 cr.serviceDead = true;
   1548                 try {
   1549                     cr.conn.connected(r.name, null);
   1550                 } catch (Exception e) {
   1551                     Slog.w(TAG, "Failure disconnecting service " + r.name +
   1552                           " to connection " + c.get(i).conn.asBinder() +
   1553                           " (in " + c.get(i).binding.client.processName + ")", e);
   1554                 }
   1555             }
   1556         }
   1557 
   1558         // Tell the service that it has been unbound.
   1559         if (r.app != null && r.app.thread != null) {
   1560             for (int i=r.bindings.size()-1; i>=0; i--) {
   1561                 IntentBindRecord ibr = r.bindings.valueAt(i);
   1562                 if (DEBUG_SERVICE) Slog.v(TAG, "Bringing down binding " + ibr
   1563                         + ": hasBound=" + ibr.hasBound);
   1564                 if (ibr.hasBound) {
   1565                     try {
   1566                         bumpServiceExecutingLocked(r, false, "bring down unbind");
   1567                         mAm.updateOomAdjLocked(r.app);
   1568                         ibr.hasBound = false;
   1569                         r.app.thread.scheduleUnbindService(r,
   1570                                 ibr.intent.getIntent());
   1571                     } catch (Exception e) {
   1572                         Slog.w(TAG, "Exception when unbinding service "
   1573                                 + r.shortName, e);
   1574                         serviceProcessGoneLocked(r);
   1575                     }
   1576                 }
   1577             }
   1578         }
   1579 
   1580         if (DEBUG_SERVICE) Slog.v(TAG, "Bringing down " + r + " " + r.intent);
   1581         EventLogTags.writeAmDestroyService(
   1582                 r.userId, System.identityHashCode(r), (r.app != null) ? r.app.pid : -1);
   1583 
   1584         final ServiceMap smap = getServiceMap(r.userId);
   1585         smap.mServicesByName.remove(r.name);
   1586         smap.mServicesByIntent.remove(r.intent);
   1587         r.totalRestartCount = 0;
   1588         unscheduleServiceRestartLocked(r, 0, true);
   1589 
   1590         // Also make sure it is not on the pending list.
   1591         for (int i=mPendingServices.size()-1; i>=0; i--) {
   1592             if (mPendingServices.get(i) == r) {
   1593                 mPendingServices.remove(i);
   1594                 if (DEBUG_SERVICE) Slog.v(TAG, "Removed pending: " + r);
   1595             }
   1596         }
   1597 
   1598         r.cancelNotification();
   1599         r.isForeground = false;
   1600         r.foregroundId = 0;
   1601         r.foregroundNoti = null;
   1602 
   1603         // Clear start entries.
   1604         r.clearDeliveredStartsLocked();
   1605         r.pendingStarts.clear();
   1606 
   1607         if (r.app != null) {
   1608             synchronized (r.stats.getBatteryStats()) {
   1609                 r.stats.stopLaunchedLocked();
   1610             }
   1611             r.app.services.remove(r);
   1612             if (r.app.thread != null) {
   1613                 updateServiceForegroundLocked(r.app, false);
   1614                 try {
   1615                     bumpServiceExecutingLocked(r, false, "destroy");
   1616                     mDestroyingServices.add(r);
   1617                     mAm.updateOomAdjLocked(r.app);
   1618                     r.app.thread.scheduleStopService(r);
   1619                 } catch (Exception e) {
   1620                     Slog.w(TAG, "Exception when destroying service "
   1621                             + r.shortName, e);
   1622                     serviceProcessGoneLocked(r);
   1623                 }
   1624             } else {
   1625                 if (DEBUG_SERVICE) Slog.v(
   1626                     TAG, "Removed service that has no process: " + r);
   1627             }
   1628         } else {
   1629             if (DEBUG_SERVICE) Slog.v(
   1630                 TAG, "Removed service that is not running: " + r);
   1631         }
   1632 
   1633         if (r.bindings.size() > 0) {
   1634             r.bindings.clear();
   1635         }
   1636 
   1637         if (r.restarter instanceof ServiceRestarter) {
   1638            ((ServiceRestarter)r.restarter).setService(null);
   1639         }
   1640 
   1641         int memFactor = mAm.mProcessStats.getMemFactorLocked();
   1642         long now = SystemClock.uptimeMillis();
   1643         if (r.tracker != null) {
   1644             r.tracker.setStarted(false, memFactor, now);
   1645             r.tracker.setBound(false, memFactor, now);
   1646             if (r.executeNesting == 0) {
   1647                 r.tracker.clearCurrentOwner(r, false);
   1648                 r.tracker = null;
   1649             }
   1650         }
   1651 
   1652         smap.ensureNotStartingBackground(r);
   1653     }
   1654 
   1655     void removeConnectionLocked(
   1656         ConnectionRecord c, ProcessRecord skipApp, ActivityRecord skipAct) {
   1657         IBinder binder = c.conn.asBinder();
   1658         AppBindRecord b = c.binding;
   1659         ServiceRecord s = b.service;
   1660         ArrayList<ConnectionRecord> clist = s.connections.get(binder);
   1661         if (clist != null) {
   1662             clist.remove(c);
   1663             if (clist.size() == 0) {
   1664                 s.connections.remove(binder);
   1665             }
   1666         }
   1667         b.connections.remove(c);
   1668         if (c.activity != null && c.activity != skipAct) {
   1669             if (c.activity.connections != null) {
   1670                 c.activity.connections.remove(c);
   1671             }
   1672         }
   1673         if (b.client != skipApp) {
   1674             b.client.connections.remove(c);
   1675             if ((c.flags&Context.BIND_ABOVE_CLIENT) != 0) {
   1676                 b.client.updateHasAboveClientLocked();
   1677             }
   1678             if (s.app != null) {
   1679                 updateServiceClientActivitiesLocked(s.app, c);
   1680             }
   1681         }
   1682         clist = mServiceConnections.get(binder);
   1683         if (clist != null) {
   1684             clist.remove(c);
   1685             if (clist.size() == 0) {
   1686                 mServiceConnections.remove(binder);
   1687             }
   1688         }
   1689 
   1690         if (b.connections.size() == 0) {
   1691             b.intent.apps.remove(b.client);
   1692         }
   1693 
   1694         if (!c.serviceDead) {
   1695             if (DEBUG_SERVICE) Slog.v(TAG, "Disconnecting binding " + b.intent
   1696                     + ": shouldUnbind=" + b.intent.hasBound);
   1697             if (s.app != null && s.app.thread != null && b.intent.apps.size() == 0
   1698                     && b.intent.hasBound) {
   1699                 try {
   1700                     bumpServiceExecutingLocked(s, false, "unbind");
   1701                     if (b.client != s.app && (c.flags&Context.BIND_WAIVE_PRIORITY) == 0
   1702                             && s.app.setProcState <= ActivityManager.PROCESS_STATE_RECEIVER) {
   1703                         // If this service's process is not already in the cached list,
   1704                         // then update it in the LRU list here because this may be causing
   1705                         // it to go down there and we want it to start out near the top.
   1706                         mAm.updateLruProcessLocked(s.app, false, null);
   1707                     }
   1708                     mAm.updateOomAdjLocked(s.app);
   1709                     b.intent.hasBound = false;
   1710                     // Assume the client doesn't want to know about a rebind;
   1711                     // we will deal with that later if it asks for one.
   1712                     b.intent.doRebind = false;
   1713                     s.app.thread.scheduleUnbindService(s, b.intent.intent.getIntent());
   1714                 } catch (Exception e) {
   1715                     Slog.w(TAG, "Exception when unbinding service " + s.shortName, e);
   1716                     serviceProcessGoneLocked(s);
   1717                 }
   1718             }
   1719 
   1720             if ((c.flags&Context.BIND_AUTO_CREATE) != 0) {
   1721                 boolean hasAutoCreate = s.hasAutoCreateConnections();
   1722                 if (!hasAutoCreate) {
   1723                     if (s.tracker != null) {
   1724                         s.tracker.setBound(false, mAm.mProcessStats.getMemFactorLocked(),
   1725                                 SystemClock.uptimeMillis());
   1726                     }
   1727                 }
   1728                 bringDownServiceIfNeededLocked(s, true, hasAutoCreate);
   1729             }
   1730         }
   1731     }
   1732 
   1733     void serviceDoneExecutingLocked(ServiceRecord r, int type, int startId, int res) {
   1734         boolean inDestroying = mDestroyingServices.contains(r);
   1735         if (r != null) {
   1736             if (type == 1) {
   1737                 // This is a call from a service start...  take care of
   1738                 // book-keeping.
   1739                 r.callStart = true;
   1740                 switch (res) {
   1741                     case Service.START_STICKY_COMPATIBILITY:
   1742                     case Service.START_STICKY: {
   1743                         // We are done with the associated start arguments.
   1744                         r.findDeliveredStart(startId, true);
   1745                         // Don't stop if killed.
   1746                         r.stopIfKilled = false;
   1747                         break;
   1748                     }
   1749                     case Service.START_NOT_STICKY: {
   1750                         // We are done with the associated start arguments.
   1751                         r.findDeliveredStart(startId, true);
   1752                         if (r.getLastStartId() == startId) {
   1753                             // There is no more work, and this service
   1754                             // doesn't want to hang around if killed.
   1755                             r.stopIfKilled = true;
   1756                         }
   1757                         break;
   1758                     }
   1759                     case Service.START_REDELIVER_INTENT: {
   1760                         // We'll keep this item until they explicitly
   1761                         // call stop for it, but keep track of the fact
   1762                         // that it was delivered.
   1763                         ServiceRecord.StartItem si = r.findDeliveredStart(startId, false);
   1764                         if (si != null) {
   1765                             si.deliveryCount = 0;
   1766                             si.doneExecutingCount++;
   1767                             // Don't stop if killed.
   1768                             r.stopIfKilled = true;
   1769                         }
   1770                         break;
   1771                     }
   1772                     case Service.START_TASK_REMOVED_COMPLETE: {
   1773                         // Special processing for onTaskRemoved().  Don't
   1774                         // impact normal onStartCommand() processing.
   1775                         r.findDeliveredStart(startId, true);
   1776                         break;
   1777                     }
   1778                     default:
   1779                         throw new IllegalArgumentException(
   1780                                 "Unknown service start result: " + res);
   1781                 }
   1782                 if (res == Service.START_STICKY_COMPATIBILITY) {
   1783                     r.callStart = false;
   1784                 }
   1785             }
   1786             final long origId = Binder.clearCallingIdentity();
   1787             serviceDoneExecutingLocked(r, inDestroying, inDestroying);
   1788             Binder.restoreCallingIdentity(origId);
   1789         } else {
   1790             Slog.w(TAG, "Done executing unknown service from pid "
   1791                     + Binder.getCallingPid());
   1792         }
   1793     }
   1794 
   1795     private void serviceProcessGoneLocked(ServiceRecord r) {
   1796         if (r.tracker != null) {
   1797             int memFactor = mAm.mProcessStats.getMemFactorLocked();
   1798             long now = SystemClock.uptimeMillis();
   1799             r.tracker.setExecuting(false, memFactor, now);
   1800             r.tracker.setBound(false, memFactor, now);
   1801             r.tracker.setStarted(false, memFactor, now);
   1802         }
   1803         serviceDoneExecutingLocked(r, true, true);
   1804     }
   1805 
   1806     private void serviceDoneExecutingLocked(ServiceRecord r, boolean inDestroying,
   1807             boolean finishing) {
   1808         if (DEBUG_SERVICE) Slog.v(TAG, "<<< DONE EXECUTING " + r
   1809                 + ": nesting=" + r.executeNesting
   1810                 + ", inDestroying=" + inDestroying + ", app=" + r.app);
   1811         else if (DEBUG_SERVICE_EXECUTING) Slog.v(TAG, "<<< DONE EXECUTING " + r.shortName);
   1812         r.executeNesting--;
   1813         if (r.executeNesting <= 0) {
   1814             if (r.app != null) {
   1815                 if (DEBUG_SERVICE) Slog.v(TAG,
   1816                         "Nesting at 0 of " + r.shortName);
   1817                 r.app.execServicesFg = false;
   1818                 r.app.executingServices.remove(r);
   1819                 if (r.app.executingServices.size() == 0) {
   1820                     if (DEBUG_SERVICE || DEBUG_SERVICE_EXECUTING) Slog.v(TAG,
   1821                             "No more executingServices of " + r.shortName);
   1822                     mAm.mHandler.removeMessages(ActivityManagerService.SERVICE_TIMEOUT_MSG, r.app);
   1823                 } else if (r.executeFg) {
   1824                     // Need to re-evaluate whether the app still needs to be in the foreground.
   1825                     for (int i=r.app.executingServices.size()-1; i>=0; i--) {
   1826                         if (r.app.executingServices.valueAt(i).executeFg) {
   1827                             r.app.execServicesFg = true;
   1828                             break;
   1829                         }
   1830                     }
   1831                 }
   1832                 if (inDestroying) {
   1833                     if (DEBUG_SERVICE) Slog.v(TAG,
   1834                             "doneExecuting remove destroying " + r);
   1835                     mDestroyingServices.remove(r);
   1836                     r.bindings.clear();
   1837                 }
   1838                 mAm.updateOomAdjLocked(r.app);
   1839             }
   1840             r.executeFg = false;
   1841             if (r.tracker != null) {
   1842                 r.tracker.setExecuting(false, mAm.mProcessStats.getMemFactorLocked(),
   1843                         SystemClock.uptimeMillis());
   1844                 if (finishing) {
   1845                     r.tracker.clearCurrentOwner(r, false);
   1846                     r.tracker = null;
   1847                 }
   1848             }
   1849             if (finishing) {
   1850                 if (r.app != null && !r.app.persistent) {
   1851                     r.app.services.remove(r);
   1852                 }
   1853                 r.app = null;
   1854             }
   1855         }
   1856     }
   1857 
   1858     boolean attachApplicationLocked(ProcessRecord proc, String processName) throws Exception {
   1859         boolean didSomething = false;
   1860         // Collect any services that are waiting for this process to come up.
   1861         if (mPendingServices.size() > 0) {
   1862             ServiceRecord sr = null;
   1863             try {
   1864                 for (int i=0; i<mPendingServices.size(); i++) {
   1865                     sr = mPendingServices.get(i);
   1866                     if (proc != sr.isolatedProc && (proc.uid != sr.appInfo.uid
   1867                             || !processName.equals(sr.processName))) {
   1868                         continue;
   1869                     }
   1870 
   1871                     mPendingServices.remove(i);
   1872                     i--;
   1873                     proc.addPackage(sr.appInfo.packageName, mAm.mProcessStats);
   1874                     realStartServiceLocked(sr, proc, sr.createdFromFg);
   1875                     didSomething = true;
   1876                 }
   1877             } catch (Exception e) {
   1878                 Slog.w(TAG, "Exception in new application when starting service "
   1879                         + sr.shortName, e);
   1880                 throw e;
   1881             }
   1882         }
   1883         // Also, if there are any services that are waiting to restart and
   1884         // would run in this process, now is a good time to start them.  It would
   1885         // be weird to bring up the process but arbitrarily not let the services
   1886         // run at this point just because their restart time hasn't come up.
   1887         if (mRestartingServices.size() > 0) {
   1888             ServiceRecord sr = null;
   1889             for (int i=0; i<mRestartingServices.size(); i++) {
   1890                 sr = mRestartingServices.get(i);
   1891                 if (proc != sr.isolatedProc && (proc.uid != sr.appInfo.uid
   1892                         || !processName.equals(sr.processName))) {
   1893                     continue;
   1894                 }
   1895                 mAm.mHandler.removeCallbacks(sr.restarter);
   1896                 mAm.mHandler.post(sr.restarter);
   1897             }
   1898         }
   1899         return didSomething;
   1900     }
   1901 
   1902     void processStartTimedOutLocked(ProcessRecord proc) {
   1903         for (int i=0; i<mPendingServices.size(); i++) {
   1904             ServiceRecord sr = mPendingServices.get(i);
   1905             if ((proc.uid == sr.appInfo.uid
   1906                     && proc.processName.equals(sr.processName))
   1907                     || sr.isolatedProc == proc) {
   1908                 Slog.w(TAG, "Forcing bringing down service: " + sr);
   1909                 sr.isolatedProc = null;
   1910                 mPendingServices.remove(i);
   1911                 i--;
   1912                 bringDownServiceLocked(sr);
   1913             }
   1914         }
   1915     }
   1916 
   1917     private boolean collectForceStopServicesLocked(String name, int userId,
   1918             boolean evenPersistent, boolean doit,
   1919             ArrayMap<ComponentName, ServiceRecord> services,
   1920             ArrayList<ServiceRecord> result) {
   1921         boolean didSomething = false;
   1922         for (int i=0; i<services.size(); i++) {
   1923             ServiceRecord service = services.valueAt(i);
   1924             if ((name == null || service.packageName.equals(name))
   1925                     && (service.app == null || evenPersistent || !service.app.persistent)) {
   1926                 if (!doit) {
   1927                     return true;
   1928                 }
   1929                 didSomething = true;
   1930                 Slog.i(TAG, "  Force stopping service " + service);
   1931                 if (service.app != null) {
   1932                     service.app.removed = true;
   1933                     if (!service.app.persistent) {
   1934                         service.app.services.remove(service);
   1935                     }
   1936                 }
   1937                 service.app = null;
   1938                 service.isolatedProc = null;
   1939                 result.add(service);
   1940             }
   1941         }
   1942         return didSomething;
   1943     }
   1944 
   1945     boolean forceStopLocked(String name, int userId, boolean evenPersistent, boolean doit) {
   1946         boolean didSomething = false;
   1947         ArrayList<ServiceRecord> services = new ArrayList<ServiceRecord>();
   1948         if (userId == UserHandle.USER_ALL) {
   1949             for (int i=0; i<mServiceMap.size(); i++) {
   1950                 didSomething |= collectForceStopServicesLocked(name, userId, evenPersistent,
   1951                         doit, mServiceMap.valueAt(i).mServicesByName, services);
   1952                 if (!doit && didSomething) {
   1953                     return true;
   1954                 }
   1955             }
   1956         } else {
   1957             ServiceMap smap = mServiceMap.get(userId);
   1958             if (smap != null) {
   1959                 ArrayMap<ComponentName, ServiceRecord> items = smap.mServicesByName;
   1960                 didSomething = collectForceStopServicesLocked(name, userId, evenPersistent,
   1961                         doit, items, services);
   1962             }
   1963         }
   1964 
   1965         int N = services.size();
   1966         for (int i=0; i<N; i++) {
   1967             bringDownServiceLocked(services.get(i));
   1968         }
   1969         return didSomething;
   1970     }
   1971 
   1972     void cleanUpRemovedTaskLocked(TaskRecord tr, ComponentName component, Intent baseIntent) {
   1973         ArrayList<ServiceRecord> services = new ArrayList<ServiceRecord>();
   1974         ArrayMap<ComponentName, ServiceRecord> alls = getServices(tr.userId);
   1975         for (int i=0; i<alls.size(); i++) {
   1976             ServiceRecord sr = alls.valueAt(i);
   1977             if (sr.packageName.equals(component.getPackageName())) {
   1978                 services.add(sr);
   1979             }
   1980         }
   1981 
   1982         // Take care of any running services associated with the app.
   1983         for (int i=0; i<services.size(); i++) {
   1984             ServiceRecord sr = services.get(i);
   1985             if (sr.startRequested) {
   1986                 if ((sr.serviceInfo.flags&ServiceInfo.FLAG_STOP_WITH_TASK) != 0) {
   1987                     Slog.i(TAG, "Stopping service " + sr.shortName + ": remove task");
   1988                     stopServiceLocked(sr);
   1989                 } else {
   1990                     sr.pendingStarts.add(new ServiceRecord.StartItem(sr, true,
   1991                             sr.makeNextStartId(), baseIntent, null));
   1992                     if (sr.app != null && sr.app.thread != null) {
   1993                         // We always run in the foreground, since this is called as
   1994                         // part of the "remove task" UI operation.
   1995                         sendServiceArgsLocked(sr, true, false);
   1996                     }
   1997                 }
   1998             }
   1999         }
   2000     }
   2001 
   2002     final void killServicesLocked(ProcessRecord app, boolean allowRestart) {
   2003         // Report disconnected services.
   2004         if (false) {
   2005             // XXX we are letting the client link to the service for
   2006             // death notifications.
   2007             if (app.services.size() > 0) {
   2008                 Iterator<ServiceRecord> it = app.services.iterator();
   2009                 while (it.hasNext()) {
   2010                     ServiceRecord r = it.next();
   2011                     for (int conni=r.connections.size()-1; conni>=0; conni--) {
   2012                         ArrayList<ConnectionRecord> cl = r.connections.valueAt(conni);
   2013                         for (int i=0; i<cl.size(); i++) {
   2014                             ConnectionRecord c = cl.get(i);
   2015                             if (c.binding.client != app) {
   2016                                 try {
   2017                                     //c.conn.connected(r.className, null);
   2018                                 } catch (Exception e) {
   2019                                     // todo: this should be asynchronous!
   2020                                     Slog.w(TAG, "Exception thrown disconnected servce "
   2021                                           + r.shortName
   2022                                           + " from app " + app.processName, e);
   2023                                 }
   2024                             }
   2025                         }
   2026                     }
   2027                 }
   2028             }
   2029         }
   2030 
   2031         // First clear app state from services.
   2032         for (int i=app.services.size()-1; i>=0; i--) {
   2033             ServiceRecord sr = app.services.valueAt(i);
   2034             synchronized (sr.stats.getBatteryStats()) {
   2035                 sr.stats.stopLaunchedLocked();
   2036             }
   2037             if (sr.app != null && !sr.app.persistent) {
   2038                 sr.app.services.remove(sr);
   2039             }
   2040             sr.app = null;
   2041             sr.isolatedProc = null;
   2042             sr.executeNesting = 0;
   2043             sr.forceClearTracker();
   2044             if (mDestroyingServices.remove(sr)) {
   2045                 if (DEBUG_SERVICE) Slog.v(TAG, "killServices remove destroying " + sr);
   2046             }
   2047 
   2048             final int numClients = sr.bindings.size();
   2049             for (int bindingi=numClients-1; bindingi>=0; bindingi--) {
   2050                 IntentBindRecord b = sr.bindings.valueAt(bindingi);
   2051                 if (DEBUG_SERVICE) Slog.v(TAG, "Killing binding " + b
   2052                         + ": shouldUnbind=" + b.hasBound);
   2053                 b.binder = null;
   2054                 b.requested = b.received = b.hasBound = false;
   2055             }
   2056         }
   2057 
   2058         // Clean up any connections this application has to other services.
   2059         for (int i=app.connections.size()-1; i>=0; i--) {
   2060             ConnectionRecord r = app.connections.valueAt(i);
   2061             removeConnectionLocked(r, app, null);
   2062         }
   2063         app.connections.clear();
   2064 
   2065         ServiceMap smap = getServiceMap(app.userId);
   2066 
   2067         // Now do remaining service cleanup.
   2068         for (int i=app.services.size()-1; i>=0; i--) {
   2069             ServiceRecord sr = app.services.valueAt(i);
   2070             // Sanity check: if the service listed for the app is not one
   2071             // we actually are maintaining, drop it.
   2072             if (smap.mServicesByName.get(sr.name) != sr) {
   2073                 ServiceRecord cur = smap.mServicesByName.get(sr.name);
   2074                 Slog.wtf(TAG, "Service " + sr + " in process " + app
   2075                         + " not same as in map: " + cur);
   2076                 app.services.removeAt(i);
   2077                 continue;
   2078             }
   2079 
   2080             // Any services running in the application may need to be placed
   2081             // back in the pending list.
   2082             if (allowRestart && sr.crashCount >= 2 && (sr.serviceInfo.applicationInfo.flags
   2083                     &ApplicationInfo.FLAG_PERSISTENT) == 0) {
   2084                 Slog.w(TAG, "Service crashed " + sr.crashCount
   2085                         + " times, stopping: " + sr);
   2086                 EventLog.writeEvent(EventLogTags.AM_SERVICE_CRASHED_TOO_MUCH,
   2087                         sr.userId, sr.crashCount, sr.shortName, app.pid);
   2088                 bringDownServiceLocked(sr);
   2089             } else if (!allowRestart) {
   2090                 bringDownServiceLocked(sr);
   2091             } else {
   2092                 boolean canceled = scheduleServiceRestartLocked(sr, true);
   2093 
   2094                 // Should the service remain running?  Note that in the
   2095                 // extreme case of so many attempts to deliver a command
   2096                 // that it failed we also will stop it here.
   2097                 if (sr.startRequested && (sr.stopIfKilled || canceled)) {
   2098                     if (sr.pendingStarts.size() == 0) {
   2099                         sr.startRequested = false;
   2100                         if (sr.tracker != null) {
   2101                             sr.tracker.setStarted(false, mAm.mProcessStats.getMemFactorLocked(),
   2102                                     SystemClock.uptimeMillis());
   2103                         }
   2104                         if (!sr.hasAutoCreateConnections()) {
   2105                             // Whoops, no reason to restart!
   2106                             bringDownServiceLocked(sr);
   2107                         }
   2108                     }
   2109                 }
   2110             }
   2111         }
   2112 
   2113         if (!allowRestart) {
   2114             app.services.clear();
   2115 
   2116             // Make sure there are no more restarting services for this process.
   2117             for (int i=mRestartingServices.size()-1; i>=0; i--) {
   2118                 ServiceRecord r = mRestartingServices.get(i);
   2119                 if (r.processName.equals(app.processName) &&
   2120                         r.serviceInfo.applicationInfo.uid == app.info.uid) {
   2121                     mRestartingServices.remove(i);
   2122                     clearRestartingIfNeededLocked(r);
   2123                 }
   2124             }
   2125             for (int i=mPendingServices.size()-1; i>=0; i--) {
   2126                 ServiceRecord r = mPendingServices.get(i);
   2127                 if (r.processName.equals(app.processName) &&
   2128                         r.serviceInfo.applicationInfo.uid == app.info.uid) {
   2129                     mPendingServices.remove(i);
   2130                 }
   2131             }
   2132         }
   2133 
   2134         // Make sure we have no more records on the stopping list.
   2135         int i = mDestroyingServices.size();
   2136         while (i > 0) {
   2137             i--;
   2138             ServiceRecord sr = mDestroyingServices.get(i);
   2139             if (sr.app == app) {
   2140                 sr.forceClearTracker();
   2141                 mDestroyingServices.remove(i);
   2142                 if (DEBUG_SERVICE) Slog.v(TAG, "killServices remove destroying " + sr);
   2143             }
   2144         }
   2145 
   2146         app.executingServices.clear();
   2147     }
   2148 
   2149     ActivityManager.RunningServiceInfo makeRunningServiceInfoLocked(ServiceRecord r) {
   2150         ActivityManager.RunningServiceInfo info =
   2151             new ActivityManager.RunningServiceInfo();
   2152         info.service = r.name;
   2153         if (r.app != null) {
   2154             info.pid = r.app.pid;
   2155         }
   2156         info.uid = r.appInfo.uid;
   2157         info.process = r.processName;
   2158         info.foreground = r.isForeground;
   2159         info.activeSince = r.createTime;
   2160         info.started = r.startRequested;
   2161         info.clientCount = r.connections.size();
   2162         info.crashCount = r.crashCount;
   2163         info.lastActivityTime = r.lastActivity;
   2164         if (r.isForeground) {
   2165             info.flags |= ActivityManager.RunningServiceInfo.FLAG_FOREGROUND;
   2166         }
   2167         if (r.startRequested) {
   2168             info.flags |= ActivityManager.RunningServiceInfo.FLAG_STARTED;
   2169         }
   2170         if (r.app != null && r.app.pid == ActivityManagerService.MY_PID) {
   2171             info.flags |= ActivityManager.RunningServiceInfo.FLAG_SYSTEM_PROCESS;
   2172         }
   2173         if (r.app != null && r.app.persistent) {
   2174             info.flags |= ActivityManager.RunningServiceInfo.FLAG_PERSISTENT_PROCESS;
   2175         }
   2176 
   2177         for (int conni=r.connections.size()-1; conni>=0; conni--) {
   2178             ArrayList<ConnectionRecord> connl = r.connections.valueAt(conni);
   2179             for (int i=0; i<connl.size(); i++) {
   2180                 ConnectionRecord conn = connl.get(i);
   2181                 if (conn.clientLabel != 0) {
   2182                     info.clientPackage = conn.binding.client.info.packageName;
   2183                     info.clientLabel = conn.clientLabel;
   2184                     return info;
   2185                 }
   2186             }
   2187         }
   2188         return info;
   2189     }
   2190 
   2191     List<ActivityManager.RunningServiceInfo> getRunningServiceInfoLocked(int maxNum,
   2192             int flags) {
   2193         ArrayList<ActivityManager.RunningServiceInfo> res
   2194                 = new ArrayList<ActivityManager.RunningServiceInfo>();
   2195 
   2196         final int uid = Binder.getCallingUid();
   2197         final long ident = Binder.clearCallingIdentity();
   2198         try {
   2199             if (ActivityManager.checkUidPermission(
   2200                     android.Manifest.permission.INTERACT_ACROSS_USERS_FULL,
   2201                     uid) == PackageManager.PERMISSION_GRANTED) {
   2202                 int[] users = mAm.getUsersLocked();
   2203                 for (int ui=0; ui<users.length && res.size() < maxNum; ui++) {
   2204                     ArrayMap<ComponentName, ServiceRecord> alls = getServices(users[ui]);
   2205                     for (int i=0; i<alls.size() && res.size() < maxNum; i++) {
   2206                         ServiceRecord sr = alls.valueAt(i);
   2207                         res.add(makeRunningServiceInfoLocked(sr));
   2208                     }
   2209                 }
   2210 
   2211                 for (int i=0; i<mRestartingServices.size() && res.size() < maxNum; i++) {
   2212                     ServiceRecord r = mRestartingServices.get(i);
   2213                     ActivityManager.RunningServiceInfo info =
   2214                             makeRunningServiceInfoLocked(r);
   2215                     info.restarting = r.nextRestartTime;
   2216                     res.add(info);
   2217                 }
   2218             } else {
   2219                 int userId = UserHandle.getUserId(uid);
   2220                 ArrayMap<ComponentName, ServiceRecord> alls = getServices(userId);
   2221                 for (int i=0; i<alls.size() && res.size() < maxNum; i++) {
   2222                     ServiceRecord sr = alls.valueAt(i);
   2223                     res.add(makeRunningServiceInfoLocked(sr));
   2224                 }
   2225 
   2226                 for (int i=0; i<mRestartingServices.size() && res.size() < maxNum; i++) {
   2227                     ServiceRecord r = mRestartingServices.get(i);
   2228                     if (r.userId == userId) {
   2229                         ActivityManager.RunningServiceInfo info =
   2230                                 makeRunningServiceInfoLocked(r);
   2231                         info.restarting = r.nextRestartTime;
   2232                         res.add(info);
   2233                     }
   2234                 }
   2235             }
   2236         } finally {
   2237             Binder.restoreCallingIdentity(ident);
   2238         }
   2239 
   2240         return res;
   2241     }
   2242 
   2243     public PendingIntent getRunningServiceControlPanelLocked(ComponentName name) {
   2244         int userId = UserHandle.getUserId(Binder.getCallingUid());
   2245         ServiceRecord r = getServiceByName(name, userId);
   2246         if (r != null) {
   2247             for (int conni=r.connections.size()-1; conni>=0; conni--) {
   2248                 ArrayList<ConnectionRecord> conn = r.connections.valueAt(conni);
   2249                 for (int i=0; i<conn.size(); i++) {
   2250                     if (conn.get(i).clientIntent != null) {
   2251                         return conn.get(i).clientIntent;
   2252                     }
   2253                 }
   2254             }
   2255         }
   2256         return null;
   2257     }
   2258 
   2259     void serviceTimeout(ProcessRecord proc) {
   2260         String anrMessage = null;
   2261 
   2262         synchronized(this) {
   2263             if (proc.executingServices.size() == 0 || proc.thread == null) {
   2264                 return;
   2265             }
   2266             long maxTime = SystemClock.uptimeMillis() -
   2267                     (proc.execServicesFg ? SERVICE_TIMEOUT : SERVICE_BACKGROUND_TIMEOUT);
   2268             ServiceRecord timeout = null;
   2269             long nextTime = 0;
   2270             for (int i=proc.executingServices.size()-1; i>=0; i--) {
   2271                 ServiceRecord sr = proc.executingServices.valueAt(i);
   2272                 if (sr.executingStart < maxTime) {
   2273                     timeout = sr;
   2274                     break;
   2275                 }
   2276                 if (sr.executingStart > nextTime) {
   2277                     nextTime = sr.executingStart;
   2278                 }
   2279             }
   2280             if (timeout != null && mAm.mLruProcesses.contains(proc)) {
   2281                 Slog.w(TAG, "Timeout executing service: " + timeout);
   2282                 anrMessage = "Executing service " + timeout.shortName;
   2283             } else {
   2284                 Message msg = mAm.mHandler.obtainMessage(
   2285                         ActivityManagerService.SERVICE_TIMEOUT_MSG);
   2286                 msg.obj = proc;
   2287                 mAm.mHandler.sendMessageAtTime(msg, proc.execServicesFg
   2288                         ? (nextTime+SERVICE_TIMEOUT) : (nextTime + SERVICE_BACKGROUND_TIMEOUT));
   2289             }
   2290         }
   2291 
   2292         if (anrMessage != null) {
   2293             mAm.appNotResponding(proc, null, null, false, anrMessage);
   2294         }
   2295     }
   2296 
   2297     void scheduleServiceTimeoutLocked(ProcessRecord proc) {
   2298         if (proc.executingServices.size() == 0 || proc.thread == null) {
   2299             return;
   2300         }
   2301         long now = SystemClock.uptimeMillis();
   2302         Message msg = mAm.mHandler.obtainMessage(
   2303                 ActivityManagerService.SERVICE_TIMEOUT_MSG);
   2304         msg.obj = proc;
   2305         mAm.mHandler.sendMessageAtTime(msg,
   2306                 proc.execServicesFg ? (now+SERVICE_TIMEOUT) : (now+ SERVICE_BACKGROUND_TIMEOUT));
   2307     }
   2308 
   2309     /**
   2310      * Prints a list of ServiceRecords (dumpsys activity services)
   2311      */
   2312     void dumpServicesLocked(FileDescriptor fd, PrintWriter pw, String[] args,
   2313             int opti, boolean dumpAll, boolean dumpClient, String dumpPackage) {
   2314         boolean needSep = false;
   2315         boolean printedAnything = false;
   2316 
   2317         ItemMatcher matcher = new ItemMatcher();
   2318         matcher.build(args, opti);
   2319 
   2320         pw.println("ACTIVITY MANAGER SERVICES (dumpsys activity services)");
   2321         try {
   2322             int[] users = mAm.getUsersLocked();
   2323             for (int user : users) {
   2324                 ServiceMap smap = getServiceMap(user);
   2325                 boolean printed = false;
   2326                 if (smap.mServicesByName.size() > 0) {
   2327                     long nowReal = SystemClock.elapsedRealtime();
   2328                     needSep = false;
   2329                     for (int si=0; si<smap.mServicesByName.size(); si++) {
   2330                         ServiceRecord r = smap.mServicesByName.valueAt(si);
   2331                         if (!matcher.match(r, r.name)) {
   2332                             continue;
   2333                         }
   2334                         if (dumpPackage != null && !dumpPackage.equals(r.appInfo.packageName)) {
   2335                             continue;
   2336                         }
   2337                         if (!printed) {
   2338                             if (printedAnything) {
   2339                                 pw.println();
   2340                             }
   2341                             pw.println("  User " + user + " active services:");
   2342                             printed = true;
   2343                         }
   2344                         printedAnything = true;
   2345                         if (needSep) {
   2346                             pw.println();
   2347                         }
   2348                         pw.print("  * ");
   2349                         pw.println(r);
   2350                         if (dumpAll) {
   2351                             r.dump(pw, "    ");
   2352                             needSep = true;
   2353                         } else {
   2354                             pw.print("    app=");
   2355                             pw.println(r.app);
   2356                             pw.print("    created=");
   2357                             TimeUtils.formatDuration(r.createTime, nowReal, pw);
   2358                             pw.print(" started=");
   2359                             pw.print(r.startRequested);
   2360                             pw.print(" connections=");
   2361                             pw.println(r.connections.size());
   2362                             if (r.connections.size() > 0) {
   2363                                 pw.println("    Connections:");
   2364                                 for (int conni=0; conni<r.connections.size(); conni++) {
   2365                                     ArrayList<ConnectionRecord> clist = r.connections.valueAt(conni);
   2366                                     for (int i = 0; i < clist.size(); i++) {
   2367                                         ConnectionRecord conn = clist.get(i);
   2368                                         pw.print("      ");
   2369                                         pw.print(conn.binding.intent.intent.getIntent()
   2370                                                 .toShortString(false, false, false, false));
   2371                                         pw.print(" -> ");
   2372                                         ProcessRecord proc = conn.binding.client;
   2373                                         pw.println(proc != null ? proc.toShortString() : "null");
   2374                                     }
   2375                                 }
   2376                             }
   2377                         }
   2378                         if (dumpClient && r.app != null && r.app.thread != null) {
   2379                             pw.println("    Client:");
   2380                             pw.flush();
   2381                             try {
   2382                                 TransferPipe tp = new TransferPipe();
   2383                                 try {
   2384                                     r.app.thread.dumpService(tp.getWriteFd().getFileDescriptor(),
   2385                                             r, args);
   2386                                     tp.setBufferPrefix("      ");
   2387                                     // Short timeout, since blocking here can
   2388                                     // deadlock with the application.
   2389                                     tp.go(fd, 2000);
   2390                                 } finally {
   2391                                     tp.kill();
   2392                                 }
   2393                             } catch (IOException e) {
   2394                                 pw.println("      Failure while dumping the service: " + e);
   2395                             } catch (RemoteException e) {
   2396                                 pw.println("      Got a RemoteException while dumping the service");
   2397                             }
   2398                             needSep = true;
   2399                         }
   2400                     }
   2401                     needSep |= printed;
   2402                 }
   2403                 printed = false;
   2404                 for (int si=0, SN=smap.mDelayedStartList.size(); si<SN; si++) {
   2405                     ServiceRecord r = smap.mDelayedStartList.get(si);
   2406                     if (!matcher.match(r, r.name)) {
   2407                         continue;
   2408                     }
   2409                     if (dumpPackage != null && !dumpPackage.equals(r.appInfo.packageName)) {
   2410                         continue;
   2411                     }
   2412                     if (!printed) {
   2413                         if (printedAnything) {
   2414                             pw.println();
   2415                         }
   2416                         pw.println("  User " + user + " delayed start services:");
   2417                         printed = true;
   2418                     }
   2419                     printedAnything = true;
   2420                     pw.print("  * Delayed start "); pw.println(r);
   2421                 }
   2422                 printed = false;
   2423                 for (int si=0, SN=smap.mStartingBackground.size(); si<SN; si++) {
   2424                     ServiceRecord r = smap.mStartingBackground.get(si);
   2425                     if (!matcher.match(r, r.name)) {
   2426                         continue;
   2427                     }
   2428                     if (dumpPackage != null && !dumpPackage.equals(r.appInfo.packageName)) {
   2429                         continue;
   2430                     }
   2431                     if (!printed) {
   2432                         if (printedAnything) {
   2433                             pw.println();
   2434                         }
   2435                         pw.println("  User " + user + " starting in background:");
   2436                         printed = true;
   2437                     }
   2438                     printedAnything = true;
   2439                     pw.print("  * Starting bg "); pw.println(r);
   2440                 }
   2441             }
   2442         } catch (Exception e) {
   2443             Slog.w(TAG, "Exception in dumpServicesLocked", e);
   2444         }
   2445 
   2446         if (mPendingServices.size() > 0) {
   2447             boolean printed = false;
   2448             for (int i=0; i<mPendingServices.size(); i++) {
   2449                 ServiceRecord r = mPendingServices.get(i);
   2450                 if (!matcher.match(r, r.name)) {
   2451                     continue;
   2452                 }
   2453                 if (dumpPackage != null && !dumpPackage.equals(r.appInfo.packageName)) {
   2454                     continue;
   2455                 }
   2456                 printedAnything = true;
   2457                 if (!printed) {
   2458                     if (needSep) pw.println();
   2459                     needSep = true;
   2460                     pw.println("  Pending services:");
   2461                     printed = true;
   2462                 }
   2463                 pw.print("  * Pending "); pw.println(r);
   2464                 r.dump(pw, "    ");
   2465             }
   2466             needSep = true;
   2467         }
   2468 
   2469         if (mRestartingServices.size() > 0) {
   2470             boolean printed = false;
   2471             for (int i=0; i<mRestartingServices.size(); i++) {
   2472                 ServiceRecord r = mRestartingServices.get(i);
   2473                 if (!matcher.match(r, r.name)) {
   2474                     continue;
   2475                 }
   2476                 if (dumpPackage != null && !dumpPackage.equals(r.appInfo.packageName)) {
   2477                     continue;
   2478                 }
   2479                 printedAnything = true;
   2480                 if (!printed) {
   2481                     if (needSep) pw.println();
   2482                     needSep = true;
   2483                     pw.println("  Restarting services:");
   2484                     printed = true;
   2485                 }
   2486                 pw.print("  * Restarting "); pw.println(r);
   2487                 r.dump(pw, "    ");
   2488             }
   2489             needSep = true;
   2490         }
   2491 
   2492         if (mDestroyingServices.size() > 0) {
   2493             boolean printed = false;
   2494             for (int i=0; i< mDestroyingServices.size(); i++) {
   2495                 ServiceRecord r = mDestroyingServices.get(i);
   2496                 if (!matcher.match(r, r.name)) {
   2497                     continue;
   2498                 }
   2499                 if (dumpPackage != null && !dumpPackage.equals(r.appInfo.packageName)) {
   2500                     continue;
   2501                 }
   2502                 printedAnything = true;
   2503                 if (!printed) {
   2504                     if (needSep) pw.println();
   2505                     needSep = true;
   2506                     pw.println("  Destroying services:");
   2507                     printed = true;
   2508                 }
   2509                 pw.print("  * Destroy "); pw.println(r);
   2510                 r.dump(pw, "    ");
   2511             }
   2512             needSep = true;
   2513         }
   2514 
   2515         if (dumpAll) {
   2516             boolean printed = false;
   2517             for (int ic=0; ic<mServiceConnections.size(); ic++) {
   2518                 ArrayList<ConnectionRecord> r = mServiceConnections.valueAt(ic);
   2519                 for (int i=0; i<r.size(); i++) {
   2520                     ConnectionRecord cr = r.get(i);
   2521                     if (!matcher.match(cr.binding.service, cr.binding.service.name)) {
   2522                         continue;
   2523                     }
   2524                     if (dumpPackage != null && (cr.binding.client == null
   2525                             || !dumpPackage.equals(cr.binding.client.info.packageName))) {
   2526                         continue;
   2527                     }
   2528                     printedAnything = true;
   2529                     if (!printed) {
   2530                         if (needSep) pw.println();
   2531                         needSep = true;
   2532                         pw.println("  Connection bindings to services:");
   2533                         printed = true;
   2534                     }
   2535                     pw.print("  * "); pw.println(cr);
   2536                     cr.dump(pw, "    ");
   2537                 }
   2538             }
   2539         }
   2540 
   2541         if (!printedAnything) {
   2542             pw.println("  (nothing)");
   2543         }
   2544     }
   2545 
   2546     /**
   2547      * There are three ways to call this:
   2548      *  - no service specified: dump all the services
   2549      *  - a flattened component name that matched an existing service was specified as the
   2550      *    first arg: dump that one service
   2551      *  - the first arg isn't the flattened component name of an existing service:
   2552      *    dump all services whose component contains the first arg as a substring
   2553      */
   2554     protected boolean dumpService(FileDescriptor fd, PrintWriter pw, String name, String[] args,
   2555             int opti, boolean dumpAll) {
   2556         ArrayList<ServiceRecord> services = new ArrayList<ServiceRecord>();
   2557 
   2558         synchronized (this) {
   2559             int[] users = mAm.getUsersLocked();
   2560             if ("all".equals(name)) {
   2561                 for (int user : users) {
   2562                     ServiceMap smap = mServiceMap.get(user);
   2563                     if (smap == null) {
   2564                         continue;
   2565                     }
   2566                     ArrayMap<ComponentName, ServiceRecord> alls = smap.mServicesByName;
   2567                     for (int i=0; i<alls.size(); i++) {
   2568                         ServiceRecord r1 = alls.valueAt(i);
   2569                         services.add(r1);
   2570                     }
   2571                 }
   2572             } else {
   2573                 ComponentName componentName = name != null
   2574                         ? ComponentName.unflattenFromString(name) : null;
   2575                 int objectId = 0;
   2576                 if (componentName == null) {
   2577                     // Not a '/' separated full component name; maybe an object ID?
   2578                     try {
   2579                         objectId = Integer.parseInt(name, 16);
   2580                         name = null;
   2581                         componentName = null;
   2582                     } catch (RuntimeException e) {
   2583                     }
   2584                 }
   2585 
   2586                 for (int user : users) {
   2587                     ServiceMap smap = mServiceMap.get(user);
   2588                     if (smap == null) {
   2589                         continue;
   2590                     }
   2591                     ArrayMap<ComponentName, ServiceRecord> alls = smap.mServicesByName;
   2592                     for (int i=0; i<alls.size(); i++) {
   2593                         ServiceRecord r1 = alls.valueAt(i);
   2594                         if (componentName != null) {
   2595                             if (r1.name.equals(componentName)) {
   2596                                 services.add(r1);
   2597                             }
   2598                         } else if (name != null) {
   2599                             if (r1.name.flattenToString().contains(name)) {
   2600                                 services.add(r1);
   2601                             }
   2602                         } else if (System.identityHashCode(r1) == objectId) {
   2603                             services.add(r1);
   2604                         }
   2605                     }
   2606                 }
   2607             }
   2608         }
   2609 
   2610         if (services.size() <= 0) {
   2611             return false;
   2612         }
   2613 
   2614         boolean needSep = false;
   2615         for (int i=0; i<services.size(); i++) {
   2616             if (needSep) {
   2617                 pw.println();
   2618             }
   2619             needSep = true;
   2620             dumpService("", fd, pw, services.get(i), args, dumpAll);
   2621         }
   2622         return true;
   2623     }
   2624 
   2625     /**
   2626      * Invokes IApplicationThread.dumpService() on the thread of the specified service if
   2627      * there is a thread associated with the service.
   2628      */
   2629     private void dumpService(String prefix, FileDescriptor fd, PrintWriter pw,
   2630             final ServiceRecord r, String[] args, boolean dumpAll) {
   2631         String innerPrefix = prefix + "  ";
   2632         synchronized (this) {
   2633             pw.print(prefix); pw.print("SERVICE ");
   2634                     pw.print(r.shortName); pw.print(" ");
   2635                     pw.print(Integer.toHexString(System.identityHashCode(r)));
   2636                     pw.print(" pid=");
   2637                     if (r.app != null) pw.println(r.app.pid);
   2638                     else pw.println("(not running)");
   2639             if (dumpAll) {
   2640                 r.dump(pw, innerPrefix);
   2641             }
   2642         }
   2643         if (r.app != null && r.app.thread != null) {
   2644             pw.print(prefix); pw.println("  Client:");
   2645             pw.flush();
   2646             try {
   2647                 TransferPipe tp = new TransferPipe();
   2648                 try {
   2649                     r.app.thread.dumpService(tp.getWriteFd().getFileDescriptor(), r, args);
   2650                     tp.setBufferPrefix(prefix + "    ");
   2651                     tp.go(fd);
   2652                 } finally {
   2653                     tp.kill();
   2654                 }
   2655             } catch (IOException e) {
   2656                 pw.println(prefix + "    Failure while dumping the service: " + e);
   2657             } catch (RemoteException e) {
   2658                 pw.println(prefix + "    Got a RemoteException while dumping the service");
   2659             }
   2660         }
   2661     }
   2662 
   2663 }
   2664