blob: 9fdd293f838230192eba14b4e2e857500cf739c3 [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2006 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package com.android.server.am;
18
Dianne Hackborn2ced96c2013-03-21 14:23:46 -070019import android.app.PendingIntent;
20import android.net.Uri;
21import android.provider.Settings;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080022import com.android.internal.os.BatteryStatsImpl;
Daniel Sandlerd0a2f862010-08-03 15:29:31 -040023import com.android.server.NotificationManagerService;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080024
Dianne Hackbornd8a43f62009-08-17 23:33:56 -070025import android.app.INotificationManager;
26import android.app.Notification;
27import android.app.NotificationManager;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080028import android.content.ComponentName;
Dianne Hackborn0c380492012-08-20 17:23:30 -070029import android.content.Context;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080030import android.content.Intent;
31import android.content.pm.ApplicationInfo;
Amith Yamasani742a6712011-05-04 14:49:28 -070032import android.content.pm.PackageManager;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080033import android.content.pm.ServiceInfo;
34import android.os.Binder;
35import android.os.IBinder;
Dianne Hackbornd8a43f62009-08-17 23:33:56 -070036import android.os.RemoteException;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080037import android.os.SystemClock;
Dianne Hackbornf02b60a2012-08-16 10:48:27 -070038import android.os.UserHandle;
Joe Onorato8a9b2202010-02-26 18:56:32 -080039import android.util.Slog;
Dianne Hackborn1ebccf52010-08-15 13:04:34 -070040import android.util.TimeUtils;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080041
42import java.io.PrintWriter;
43import java.util.ArrayList;
44import java.util.HashMap;
Dianne Hackborn39792d22010-08-19 18:01:52 -070045import java.util.HashSet;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080046import java.util.Iterator;
47import java.util.List;
48
49/**
50 * A running application service.
51 */
52class ServiceRecord extends Binder {
Dianne Hackborn39792d22010-08-19 18:01:52 -070053 // Maximum number of delivery attempts before giving up.
54 static final int MAX_DELIVERY_COUNT = 3;
55
56 // Maximum number of times it can fail during execution before giving up.
57 static final int MAX_DONE_EXECUTING_COUNT = 6;
58
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -080059 final ActivityManagerService ams;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080060 final BatteryStatsImpl.Uid.Pkg.Serv stats;
61 final ComponentName name; // service component.
62 final String shortName; // name.flattenToShortString().
63 final Intent.FilterComparison intent;
64 // original intent used to find service.
65 final ServiceInfo serviceInfo;
66 // all information about the service.
67 final ApplicationInfo appInfo;
68 // information about service's app.
Amith Yamasani742a6712011-05-04 14:49:28 -070069 final int userId; // user that this service is running as
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080070 final String packageName; // the package implementing intent's component
71 final String processName; // process where this component wants to run
72 final String permission;// permission needed to access service
73 final String baseDir; // where activity source (resources etc) located
74 final String resDir; // where public activity source (public resources etc) located
75 final String dataDir; // where activity data should go
76 final boolean exported; // from ServiceInfo.exported
77 final Runnable restarter; // used to schedule retries of starting the service
78 final long createTime; // when this service was created
79 final HashMap<Intent.FilterComparison, IntentBindRecord> bindings
80 = new HashMap<Intent.FilterComparison, IntentBindRecord>();
81 // All active bindings to the service.
Dianne Hackborn43d9ac82010-08-25 15:06:25 -070082 final HashMap<IBinder, ArrayList<ConnectionRecord>> connections
83 = new HashMap<IBinder, ArrayList<ConnectionRecord>>();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080084 // IBinder -> ConnectionRecord of all bound clients
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080085
Dianne Hackbornd8a43f62009-08-17 23:33:56 -070086 ProcessRecord app; // where this service is running or null.
Dianne Hackborna0c283e2012-02-09 10:47:01 -080087 ProcessRecord isolatedProc; // keep track of isolated process, if requested
Dianne Hackbornd8a43f62009-08-17 23:33:56 -070088 boolean isForeground; // is service currently in foreground mode?
89 int foregroundId; // Notification ID of last foreground req.
90 Notification foregroundNoti; // Notification record of foreground state.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080091 long lastActivity; // last time there was some activity on the service.
92 boolean startRequested; // someone explicitly called start?
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -070093 boolean stopIfKilled; // last onStart() said to stop if service killed?
94 boolean callStart; // last onStart() has asked to alway be called on restart.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080095 int executeNesting; // number of outstanding operations keeping foreground.
96 long executingStart; // start time of last execute request.
97 int crashCount; // number of times proc has crashed with service running
98 int totalRestartCount; // number of times we have had to restart.
99 int restartCount; // number of restarts performed in a row.
100 long restartDelay; // delay until next restart attempt.
101 long restartTime; // time of last restart.
102 long nextRestartTime; // time when restartDelay will expire.
103
Dianne Hackbornf210d6b2009-04-13 18:42:49 -0700104 String stringName; // caching of toString
105
Dianne Hackborn0c5001d2011-04-12 18:16:08 -0700106 private int lastStartId; // identifier of most recent start request.
107
Dianne Hackborn7e269642010-08-25 19:50:20 -0700108 static class StartItem {
Dianne Hackborn39792d22010-08-19 18:01:52 -0700109 final ServiceRecord sr;
Dianne Hackborn0c5001d2011-04-12 18:16:08 -0700110 final boolean taskRemoved;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700111 final int id;
112 final Intent intent;
Dianne Hackborn21c241e2012-03-08 13:57:23 -0800113 final ActivityManagerService.NeededUriGrants neededGrants;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700114 long deliveredTime;
115 int deliveryCount;
116 int doneExecutingCount;
Dianne Hackborn7e269642010-08-25 19:50:20 -0700117 UriPermissionOwner uriPermissions;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700118
119 String stringName; // caching of toString
120
Dianne Hackborn0c5001d2011-04-12 18:16:08 -0700121 StartItem(ServiceRecord _sr, boolean _taskRemoved, int _id, Intent _intent,
Dianne Hackborn21c241e2012-03-08 13:57:23 -0800122 ActivityManagerService.NeededUriGrants _neededGrants) {
Dianne Hackborn39792d22010-08-19 18:01:52 -0700123 sr = _sr;
Dianne Hackborn0c5001d2011-04-12 18:16:08 -0700124 taskRemoved = _taskRemoved;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700125 id = _id;
126 intent = _intent;
Dianne Hackborn21c241e2012-03-08 13:57:23 -0800127 neededGrants = _neededGrants;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700128 }
129
Dianne Hackborn7e269642010-08-25 19:50:20 -0700130 UriPermissionOwner getUriPermissionsLocked() {
131 if (uriPermissions == null) {
132 uriPermissions = new UriPermissionOwner(sr.ams, this);
133 }
134 return uriPermissions;
135 }
136
Dianne Hackborn39792d22010-08-19 18:01:52 -0700137 void removeUriPermissionsLocked() {
Dianne Hackborn7e269642010-08-25 19:50:20 -0700138 if (uriPermissions != null) {
139 uriPermissions.removeUriPermissionsLocked();
140 uriPermissions = null;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700141 }
142 }
143
144 public String toString() {
145 if (stringName != null) {
146 return stringName;
147 }
148 StringBuilder sb = new StringBuilder(128);
149 sb.append("ServiceRecord{")
150 .append(Integer.toHexString(System.identityHashCode(sr)))
151 .append(' ').append(sr.shortName)
152 .append(" StartItem ")
153 .append(Integer.toHexString(System.identityHashCode(this)))
154 .append(" id=").append(id).append('}');
155 return stringName = sb.toString();
156 }
157 }
158
159 final ArrayList<StartItem> deliveredStarts = new ArrayList<StartItem>();
160 // start() arguments which been delivered.
161 final ArrayList<StartItem> pendingStarts = new ArrayList<StartItem>();
162 // start() arguments that haven't yet been delivered.
163
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700164 void dumpStartList(PrintWriter pw, String prefix, List<StartItem> list, long now) {
165 final int N = list.size();
166 for (int i=0; i<N; i++) {
167 StartItem si = list.get(i);
168 pw.print(prefix); pw.print("#"); pw.print(i);
169 pw.print(" id="); pw.print(si.id);
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700170 if (now != 0) {
171 pw.print(" dur=");
172 TimeUtils.formatDuration(si.deliveredTime, now, pw);
173 }
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700174 if (si.deliveryCount != 0) {
175 pw.print(" dc="); pw.print(si.deliveryCount);
176 }
177 if (si.doneExecutingCount != 0) {
178 pw.print(" dxc="); pw.print(si.doneExecutingCount);
179 }
Dianne Hackborn39792d22010-08-19 18:01:52 -0700180 pw.println("");
181 pw.print(prefix); pw.print(" intent=");
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700182 if (si.intent != null) pw.println(si.intent.toString());
183 else pw.println("null");
Dianne Hackborn21c241e2012-03-08 13:57:23 -0800184 if (si.neededGrants != null) {
185 pw.print(prefix); pw.print(" neededGrants=");
186 pw.println(si.neededGrants);
Dianne Hackborn39792d22010-08-19 18:01:52 -0700187 }
Dianne Hackborn7e269642010-08-25 19:50:20 -0700188 if (si.uriPermissions != null) {
189 if (si.uriPermissions.readUriPermissions != null) {
190 pw.print(prefix); pw.print(" readUriPermissions=");
191 pw.println(si.uriPermissions.readUriPermissions);
192 }
193 if (si.uriPermissions.writeUriPermissions != null) {
194 pw.print(prefix); pw.print(" writeUriPermissions=");
195 pw.println(si.uriPermissions.writeUriPermissions);
196 }
Dianne Hackborn39792d22010-08-19 18:01:52 -0700197 }
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700198 }
199 }
200
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800201 void dump(PrintWriter pw, String prefix) {
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700202 pw.print(prefix); pw.print("intent={");
Dianne Hackborn21c241e2012-03-08 13:57:23 -0800203 pw.print(intent.getIntent().toShortString(false, true, false, true));
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700204 pw.println('}');
205 pw.print(prefix); pw.print("packageName="); pw.println(packageName);
206 pw.print(prefix); pw.print("processName="); pw.println(processName);
207 if (permission != null) {
208 pw.print(prefix); pw.print("permission="); pw.println(permission);
209 }
Dianne Hackbornfd12af42009-08-27 00:44:33 -0700210 long now = SystemClock.uptimeMillis();
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700211 long nowReal = SystemClock.elapsedRealtime();
212 pw.print(prefix); pw.print("baseDir="); pw.println(baseDir);
Marco Nelissena301fe62011-05-04 08:44:59 -0700213 if (!resDir.equals(baseDir)) {
214 pw.print(prefix); pw.print("resDir="); pw.println(resDir);
215 }
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700216 pw.print(prefix); pw.print("dataDir="); pw.println(dataDir);
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700217 pw.print(prefix); pw.print("app="); pw.println(app);
Dianne Hackborna0c283e2012-02-09 10:47:01 -0800218 if (isolatedProc != null) {
219 pw.print(prefix); pw.print("isolatedProc="); pw.println(isolatedProc);
220 }
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700221 if (isForeground || foregroundId != 0) {
222 pw.print(prefix); pw.print("isForeground="); pw.print(isForeground);
223 pw.print(" foregroundId="); pw.print(foregroundId);
224 pw.print(" foregroundNoti="); pw.println(foregroundNoti);
225 }
Dianne Hackborn9adb9c32010-08-13 14:09:56 -0700226 pw.print(prefix); pw.print("createTime=");
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700227 TimeUtils.formatDuration(createTime, nowReal, pw);
228 pw.print(" lastActivity=");
229 TimeUtils.formatDuration(lastActivity, now, pw);
230 pw.println("");
Dianne Hackborne17aeb32011-04-07 15:11:57 -0700231 pw.print(prefix); pw.print("executingStart=");
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700232 TimeUtils.formatDuration(executingStart, now, pw);
233 pw.print(" restartTime=");
234 TimeUtils.formatDuration(restartTime, now, pw);
235 pw.println("");
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700236 if (startRequested || lastStartId != 0) {
237 pw.print(prefix); pw.print("startRequested="); pw.print(startRequested);
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700238 pw.print(" stopIfKilled="); pw.print(stopIfKilled);
239 pw.print(" callStart="); pw.print(callStart);
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700240 pw.print(" lastStartId="); pw.println(lastStartId);
241 }
242 if (executeNesting != 0 || crashCount != 0 || restartCount != 0
243 || restartDelay != 0 || nextRestartTime != 0) {
244 pw.print(prefix); pw.print("executeNesting="); pw.print(executeNesting);
245 pw.print(" restartCount="); pw.print(restartCount);
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700246 pw.print(" restartDelay=");
247 TimeUtils.formatDuration(restartDelay, now, pw);
248 pw.print(" nextRestartTime=");
249 TimeUtils.formatDuration(nextRestartTime, now, pw);
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700250 pw.print(" crashCount="); pw.println(crashCount);
251 }
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700252 if (deliveredStarts.size() > 0) {
253 pw.print(prefix); pw.println("Delivered Starts:");
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700254 dumpStartList(pw, prefix, deliveredStarts, now);
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700255 }
256 if (pendingStarts.size() > 0) {
257 pw.print(prefix); pw.println("Pending Starts:");
258 dumpStartList(pw, prefix, pendingStarts, 0);
259 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800260 if (bindings.size() > 0) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800261 Iterator<IntentBindRecord> it = bindings.values().iterator();
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700262 pw.print(prefix); pw.println("Bindings:");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800263 while (it.hasNext()) {
264 IntentBindRecord b = it.next();
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700265 pw.print(prefix); pw.print("* IntentBindRecord{");
266 pw.print(Integer.toHexString(System.identityHashCode(b)));
Dianne Hackborn0c380492012-08-20 17:23:30 -0700267 if ((b.collectFlags()&Context.BIND_AUTO_CREATE) != 0) {
268 pw.append(" CREATE");
269 }
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700270 pw.println("}:");
271 b.dumpInService(pw, prefix + " ");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800272 }
273 }
274 if (connections.size() > 0) {
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700275 pw.print(prefix); pw.println("All Connections:");
Dianne Hackborn43d9ac82010-08-25 15:06:25 -0700276 Iterator<ArrayList<ConnectionRecord>> it = connections.values().iterator();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800277 while (it.hasNext()) {
Dianne Hackborn43d9ac82010-08-25 15:06:25 -0700278 ArrayList<ConnectionRecord> c = it.next();
279 for (int i=0; i<c.size(); i++) {
280 pw.print(prefix); pw.print(" "); pw.println(c.get(i));
281 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800282 }
283 }
284 }
285
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800286 ServiceRecord(ActivityManagerService ams,
287 BatteryStatsImpl.Uid.Pkg.Serv servStats, ComponentName name,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800288 Intent.FilterComparison intent, ServiceInfo sInfo, Runnable restarter) {
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800289 this.ams = ams;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800290 this.stats = servStats;
291 this.name = name;
292 shortName = name.flattenToShortString();
293 this.intent = intent;
294 serviceInfo = sInfo;
295 appInfo = sInfo.applicationInfo;
296 packageName = sInfo.applicationInfo.packageName;
297 processName = sInfo.processName;
298 permission = sInfo.permission;
299 baseDir = sInfo.applicationInfo.sourceDir;
300 resDir = sInfo.applicationInfo.publicSourceDir;
301 dataDir = sInfo.applicationInfo.dataDir;
302 exported = sInfo.exported;
303 this.restarter = restarter;
Dianne Hackborn9adb9c32010-08-13 14:09:56 -0700304 createTime = SystemClock.elapsedRealtime();
305 lastActivity = SystemClock.uptimeMillis();
Dianne Hackbornf02b60a2012-08-16 10:48:27 -0700306 userId = UserHandle.getUserId(appInfo.uid);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800307 }
308
309 public AppBindRecord retrieveAppBindingLocked(Intent intent,
310 ProcessRecord app) {
311 Intent.FilterComparison filter = new Intent.FilterComparison(intent);
312 IntentBindRecord i = bindings.get(filter);
313 if (i == null) {
314 i = new IntentBindRecord(this, filter);
315 bindings.put(filter, i);
316 }
317 AppBindRecord a = i.apps.get(app);
318 if (a != null) {
319 return a;
320 }
321 a = new AppBindRecord(this, i, app);
322 i.apps.put(app, a);
323 return a;
324 }
325
326 public void resetRestartCounter() {
327 restartCount = 0;
328 restartDelay = 0;
329 restartTime = 0;
330 }
331
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700332 public StartItem findDeliveredStart(int id, boolean remove) {
333 final int N = deliveredStarts.size();
334 for (int i=0; i<N; i++) {
335 StartItem si = deliveredStarts.get(i);
336 if (si.id == id) {
337 if (remove) deliveredStarts.remove(i);
338 return si;
339 }
340 }
341
342 return null;
343 }
344
Dianne Hackborn0c5001d2011-04-12 18:16:08 -0700345 public int getLastStartId() {
346 return lastStartId;
347 }
348
349 public int makeNextStartId() {
350 lastStartId++;
351 if (lastStartId < 1) {
352 lastStartId = 1;
353 }
354 return lastStartId;
355 }
356
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700357 public void postNotification() {
Daniel Sandlerd0a2f862010-08-03 15:29:31 -0400358 final int appUid = appInfo.uid;
359 final int appPid = app.pid;
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700360 if (foregroundId != 0 && foregroundNoti != null) {
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800361 // Do asynchronous communication with notification manager to
362 // avoid deadlocks.
363 final String localPackageName = packageName;
364 final int localForegroundId = foregroundId;
365 final Notification localForegroundNoti = foregroundNoti;
366 ams.mHandler.post(new Runnable() {
367 public void run() {
Daniel Sandlerd0a2f862010-08-03 15:29:31 -0400368 NotificationManagerService nm =
369 (NotificationManagerService) NotificationManager.getService();
370 if (nm == null) {
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800371 return;
372 }
373 try {
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700374 if (localForegroundNoti.icon == 0) {
Dianne Hackborn2ced96c2013-03-21 14:23:46 -0700375 // It is not correct for the caller to supply a notification
376 // icon, but this used to be able to slip through, so for
377 // those dirty apps give it the app's icon.
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700378 localForegroundNoti.icon = appInfo.icon;
Daniel Sandler91fe8452013-04-08 12:23:27 -0400379
380 // Do not allow apps to present a sneaky invisible content view either.
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700381 localForegroundNoti.contentView = null;
382 localForegroundNoti.bigContentView = null;
Daniel Sandler91fe8452013-04-08 12:23:27 -0400383 CharSequence appName = appInfo.loadLabel(
384 ams.mContext.getPackageManager());
385 if (appName == null) {
386 appName = appInfo.packageName;
387 }
388 Context ctx = null;
389 try {
390 ctx = ams.mContext.createPackageContext(
391 appInfo.packageName, 0);
392 Intent runningIntent = new Intent(
393 Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
394 runningIntent.setData(Uri.fromParts("package",
395 appInfo.packageName, null));
396 PendingIntent pi = PendingIntent.getActivity(ams.mContext, 0,
397 runningIntent, PendingIntent.FLAG_UPDATE_CURRENT);
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700398 localForegroundNoti.setLatestEventInfo(ctx,
Daniel Sandler91fe8452013-04-08 12:23:27 -0400399 ams.mContext.getString(
400 com.android.internal.R.string
401 .app_running_notification_title,
402 appName),
403 ams.mContext.getString(
404 com.android.internal.R.string
405 .app_running_notification_text,
406 appName),
407 pi);
408 } catch (PackageManager.NameNotFoundException e) {
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700409 localForegroundNoti.icon = 0;
Dianne Hackborn2ced96c2013-03-21 14:23:46 -0700410 }
411 }
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700412 if (localForegroundNoti.icon == 0) {
Dianne Hackborn282add72013-03-15 18:48:04 -0700413 // Notifications whose icon is 0 are defined to not show
414 // a notification, silently ignoring it. We don't want to
415 // just ignore it, we want to prevent the service from
416 // being foreground.
417 throw new RuntimeException("icon must be non-zero");
418 }
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800419 int[] outId = new int[1];
Dianne Hackbornf265ea92013-01-31 15:00:51 -0800420 nm.enqueueNotificationInternal(localPackageName, localPackageName,
421 appUid, appPid, null, localForegroundId, localForegroundNoti,
422 outId, userId);
Joe Onorato34fcf972010-02-18 07:45:17 -0500423 } catch (RuntimeException e) {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800424 Slog.w(ActivityManagerService.TAG,
Dianne Hackborn9e0f5d92010-02-22 15:05:42 -0800425 "Error showing notification for service", e);
426 // If it gave us a garbage notification, it doesn't
427 // get to be foreground.
428 ams.setServiceForeground(name, ServiceRecord.this,
Joe Onoratoeaa07182010-09-23 16:49:20 -0700429 0, null, true);
430 ams.crashApplication(appUid, appPid, localPackageName,
431 "Bad notification for startForeground: " + e);
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800432 }
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700433 }
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800434 });
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700435 }
436 }
437
438 public void cancelNotification() {
439 if (foregroundId != 0) {
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800440 // Do asynchronous communication with notification manager to
441 // avoid deadlocks.
442 final String localPackageName = packageName;
443 final int localForegroundId = foregroundId;
444 ams.mHandler.post(new Runnable() {
445 public void run() {
446 INotificationManager inm = NotificationManager.getService();
447 if (inm == null) {
448 return;
449 }
450 try {
Dianne Hackborn41203752012-08-31 14:05:51 -0700451 inm.cancelNotificationWithTag(localPackageName, null,
452 localForegroundId, userId);
Joe Onorato34fcf972010-02-18 07:45:17 -0500453 } catch (RuntimeException e) {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800454 Slog.w(ActivityManagerService.TAG,
Dianne Hackborn9e0f5d92010-02-22 15:05:42 -0800455 "Error canceling notification for service", e);
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800456 } catch (RemoteException e) {
457 }
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700458 }
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800459 });
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700460 }
461 }
462
Dianne Hackborn39792d22010-08-19 18:01:52 -0700463 public void clearDeliveredStartsLocked() {
464 for (int i=deliveredStarts.size()-1; i>=0; i--) {
465 deliveredStarts.get(i).removeUriPermissionsLocked();
466 }
467 deliveredStarts.clear();
468 }
469
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800470 public String toString() {
Dianne Hackbornf210d6b2009-04-13 18:42:49 -0700471 if (stringName != null) {
472 return stringName;
473 }
474 StringBuilder sb = new StringBuilder(128);
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700475 sb.append("ServiceRecord{")
476 .append(Integer.toHexString(System.identityHashCode(this)))
Dianne Hackbornb12e1352012-09-26 11:39:20 -0700477 .append(" u").append(userId)
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700478 .append(' ').append(shortName).append('}');
Dianne Hackbornf210d6b2009-04-13 18:42:49 -0700479 return stringName = sb.toString();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800480 }
481}