blob: 5000940b81c6b19ec472b14963ca80f3d136b0a4 [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
19import com.android.internal.os.BatteryStatsImpl;
Daniel Sandlerd0a2f862010-08-03 15:29:31 -040020import com.android.server.NotificationManagerService;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080021
Dianne Hackbornd8a43f62009-08-17 23:33:56 -070022import android.app.INotificationManager;
23import android.app.Notification;
24import android.app.NotificationManager;
Dianne Hackborn390517b2013-05-30 15:03:32 -070025import android.app.PendingIntent;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080026import android.content.ComponentName;
Dianne Hackborn0c380492012-08-20 17:23:30 -070027import android.content.Context;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080028import android.content.Intent;
29import android.content.pm.ApplicationInfo;
Amith Yamasani742a6712011-05-04 14:49:28 -070030import android.content.pm.PackageManager;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080031import android.content.pm.ServiceInfo;
Dianne Hackborn390517b2013-05-30 15:03:32 -070032import android.net.Uri;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080033import android.os.Binder;
34import android.os.IBinder;
Dianne Hackbornd8a43f62009-08-17 23:33:56 -070035import android.os.RemoteException;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080036import android.os.SystemClock;
Dianne Hackbornf02b60a2012-08-16 10:48:27 -070037import android.os.UserHandle;
Dianne Hackborn390517b2013-05-30 15:03:32 -070038import android.provider.Settings;
39import android.util.ArrayMap;
Joe Onorato8a9b2202010-02-26 18:56:32 -080040import android.util.Slog;
Dianne Hackborn1ebccf52010-08-15 13:04:34 -070041import android.util.TimeUtils;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080042
43import java.io.PrintWriter;
44import java.util.ArrayList;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080045import java.util.List;
46
47/**
48 * A running application service.
49 */
Dianne Hackbornbe4e6aa2013-06-07 13:25:29 -070050final class ServiceRecord extends Binder {
Dianne Hackborn39792d22010-08-19 18:01:52 -070051 // Maximum number of delivery attempts before giving up.
52 static final int MAX_DELIVERY_COUNT = 3;
53
54 // Maximum number of times it can fail during execution before giving up.
55 static final int MAX_DONE_EXECUTING_COUNT = 6;
56
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -080057 final ActivityManagerService ams;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080058 final BatteryStatsImpl.Uid.Pkg.Serv stats;
59 final ComponentName name; // service component.
60 final String shortName; // name.flattenToShortString().
61 final Intent.FilterComparison intent;
62 // original intent used to find service.
63 final ServiceInfo serviceInfo;
64 // all information about the service.
65 final ApplicationInfo appInfo;
66 // information about service's app.
Amith Yamasani742a6712011-05-04 14:49:28 -070067 final int userId; // user that this service is running as
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080068 final String packageName; // the package implementing intent's component
69 final String processName; // process where this component wants to run
70 final String permission;// permission needed to access service
71 final String baseDir; // where activity source (resources etc) located
72 final String resDir; // where public activity source (public resources etc) located
73 final String dataDir; // where activity data should go
74 final boolean exported; // from ServiceInfo.exported
75 final Runnable restarter; // used to schedule retries of starting the service
Dianne Hackborn91268cf2013-06-13 19:06:50 -070076 final ProcessTracker.ServiceState tracker; // tracking service execution, may be null
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080077 final long createTime; // when this service was created
Dianne Hackborn390517b2013-05-30 15:03:32 -070078 final ArrayMap<Intent.FilterComparison, IntentBindRecord> bindings
79 = new ArrayMap<Intent.FilterComparison, IntentBindRecord>();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080080 // All active bindings to the service.
Dianne Hackborn390517b2013-05-30 15:03:32 -070081 final ArrayMap<IBinder, ArrayList<ConnectionRecord>> connections
82 = new ArrayMap<IBinder, ArrayList<ConnectionRecord>>();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080083 // IBinder -> ConnectionRecord of all bound clients
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080084
Dianne Hackbornd8a43f62009-08-17 23:33:56 -070085 ProcessRecord app; // where this service is running or null.
Dianne Hackborna0c283e2012-02-09 10:47:01 -080086 ProcessRecord isolatedProc; // keep track of isolated process, if requested
Dianne Hackbornd8a43f62009-08-17 23:33:56 -070087 boolean isForeground; // is service currently in foreground mode?
88 int foregroundId; // Notification ID of last foreground req.
89 Notification foregroundNoti; // Notification record of foreground state.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080090 long lastActivity; // last time there was some activity on the service.
91 boolean startRequested; // someone explicitly called start?
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -070092 boolean stopIfKilled; // last onStart() said to stop if service killed?
93 boolean callStart; // last onStart() has asked to alway be called on restart.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080094 int executeNesting; // number of outstanding operations keeping foreground.
95 long executingStart; // start time of last execute request.
96 int crashCount; // number of times proc has crashed with service running
97 int totalRestartCount; // number of times we have had to restart.
98 int restartCount; // number of restarts performed in a row.
99 long restartDelay; // delay until next restart attempt.
100 long restartTime; // time of last restart.
101 long nextRestartTime; // time when restartDelay will expire.
102
Dianne Hackbornf210d6b2009-04-13 18:42:49 -0700103 String stringName; // caching of toString
104
Dianne Hackborn0c5001d2011-04-12 18:16:08 -0700105 private int lastStartId; // identifier of most recent start request.
106
Dianne Hackborn7e269642010-08-25 19:50:20 -0700107 static class StartItem {
Dianne Hackborn39792d22010-08-19 18:01:52 -0700108 final ServiceRecord sr;
Dianne Hackborn0c5001d2011-04-12 18:16:08 -0700109 final boolean taskRemoved;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700110 final int id;
111 final Intent intent;
Dianne Hackborn21c241e2012-03-08 13:57:23 -0800112 final ActivityManagerService.NeededUriGrants neededGrants;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700113 long deliveredTime;
114 int deliveryCount;
115 int doneExecutingCount;
Dianne Hackborn7e269642010-08-25 19:50:20 -0700116 UriPermissionOwner uriPermissions;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700117
118 String stringName; // caching of toString
119
Dianne Hackborn0c5001d2011-04-12 18:16:08 -0700120 StartItem(ServiceRecord _sr, boolean _taskRemoved, int _id, Intent _intent,
Dianne Hackborn21c241e2012-03-08 13:57:23 -0800121 ActivityManagerService.NeededUriGrants _neededGrants) {
Dianne Hackborn39792d22010-08-19 18:01:52 -0700122 sr = _sr;
Dianne Hackborn0c5001d2011-04-12 18:16:08 -0700123 taskRemoved = _taskRemoved;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700124 id = _id;
125 intent = _intent;
Dianne Hackborn21c241e2012-03-08 13:57:23 -0800126 neededGrants = _neededGrants;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700127 }
128
Dianne Hackborn7e269642010-08-25 19:50:20 -0700129 UriPermissionOwner getUriPermissionsLocked() {
130 if (uriPermissions == null) {
131 uriPermissions = new UriPermissionOwner(sr.ams, this);
132 }
133 return uriPermissions;
134 }
135
Dianne Hackborn39792d22010-08-19 18:01:52 -0700136 void removeUriPermissionsLocked() {
Dianne Hackborn7e269642010-08-25 19:50:20 -0700137 if (uriPermissions != null) {
138 uriPermissions.removeUriPermissionsLocked();
139 uriPermissions = null;
Dianne Hackborn39792d22010-08-19 18:01:52 -0700140 }
141 }
142
143 public String toString() {
144 if (stringName != null) {
145 return stringName;
146 }
147 StringBuilder sb = new StringBuilder(128);
148 sb.append("ServiceRecord{")
149 .append(Integer.toHexString(System.identityHashCode(sr)))
150 .append(' ').append(sr.shortName)
151 .append(" StartItem ")
152 .append(Integer.toHexString(System.identityHashCode(this)))
153 .append(" id=").append(id).append('}');
154 return stringName = sb.toString();
155 }
156 }
157
158 final ArrayList<StartItem> deliveredStarts = new ArrayList<StartItem>();
159 // start() arguments which been delivered.
160 final ArrayList<StartItem> pendingStarts = new ArrayList<StartItem>();
161 // start() arguments that haven't yet been delivered.
162
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700163 void dumpStartList(PrintWriter pw, String prefix, List<StartItem> list, long now) {
164 final int N = list.size();
165 for (int i=0; i<N; i++) {
166 StartItem si = list.get(i);
167 pw.print(prefix); pw.print("#"); pw.print(i);
168 pw.print(" id="); pw.print(si.id);
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700169 if (now != 0) {
170 pw.print(" dur=");
171 TimeUtils.formatDuration(si.deliveredTime, now, pw);
172 }
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700173 if (si.deliveryCount != 0) {
174 pw.print(" dc="); pw.print(si.deliveryCount);
175 }
176 if (si.doneExecutingCount != 0) {
177 pw.print(" dxc="); pw.print(si.doneExecutingCount);
178 }
Dianne Hackborn39792d22010-08-19 18:01:52 -0700179 pw.println("");
180 pw.print(prefix); pw.print(" intent=");
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700181 if (si.intent != null) pw.println(si.intent.toString());
182 else pw.println("null");
Dianne Hackborn21c241e2012-03-08 13:57:23 -0800183 if (si.neededGrants != null) {
184 pw.print(prefix); pw.print(" neededGrants=");
185 pw.println(si.neededGrants);
Dianne Hackborn39792d22010-08-19 18:01:52 -0700186 }
Dianne Hackborn7e269642010-08-25 19:50:20 -0700187 if (si.uriPermissions != null) {
188 if (si.uriPermissions.readUriPermissions != null) {
189 pw.print(prefix); pw.print(" readUriPermissions=");
190 pw.println(si.uriPermissions.readUriPermissions);
191 }
192 if (si.uriPermissions.writeUriPermissions != null) {
193 pw.print(prefix); pw.print(" writeUriPermissions=");
194 pw.println(si.uriPermissions.writeUriPermissions);
195 }
Dianne Hackborn39792d22010-08-19 18:01:52 -0700196 }
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700197 }
198 }
199
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800200 void dump(PrintWriter pw, String prefix) {
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700201 pw.print(prefix); pw.print("intent={");
Dianne Hackborn21c241e2012-03-08 13:57:23 -0800202 pw.print(intent.getIntent().toShortString(false, true, false, true));
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700203 pw.println('}');
204 pw.print(prefix); pw.print("packageName="); pw.println(packageName);
205 pw.print(prefix); pw.print("processName="); pw.println(processName);
206 if (permission != null) {
207 pw.print(prefix); pw.print("permission="); pw.println(permission);
208 }
Dianne Hackbornfd12af42009-08-27 00:44:33 -0700209 long now = SystemClock.uptimeMillis();
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700210 long nowReal = SystemClock.elapsedRealtime();
211 pw.print(prefix); pw.print("baseDir="); pw.println(baseDir);
Marco Nelissena301fe62011-05-04 08:44:59 -0700212 if (!resDir.equals(baseDir)) {
213 pw.print(prefix); pw.print("resDir="); pw.println(resDir);
214 }
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700215 pw.print(prefix); pw.print("dataDir="); pw.println(dataDir);
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700216 pw.print(prefix); pw.print("app="); pw.println(app);
Dianne Hackborna0c283e2012-02-09 10:47:01 -0800217 if (isolatedProc != null) {
218 pw.print(prefix); pw.print("isolatedProc="); pw.println(isolatedProc);
219 }
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700220 if (isForeground || foregroundId != 0) {
221 pw.print(prefix); pw.print("isForeground="); pw.print(isForeground);
222 pw.print(" foregroundId="); pw.print(foregroundId);
223 pw.print(" foregroundNoti="); pw.println(foregroundNoti);
224 }
Dianne Hackborn9adb9c32010-08-13 14:09:56 -0700225 pw.print(prefix); pw.print("createTime=");
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700226 TimeUtils.formatDuration(createTime, nowReal, pw);
227 pw.print(" lastActivity=");
228 TimeUtils.formatDuration(lastActivity, now, pw);
229 pw.println("");
Dianne Hackborne17aeb32011-04-07 15:11:57 -0700230 pw.print(prefix); pw.print("executingStart=");
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700231 TimeUtils.formatDuration(executingStart, now, pw);
232 pw.print(" restartTime=");
233 TimeUtils.formatDuration(restartTime, now, pw);
234 pw.println("");
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700235 if (startRequested || lastStartId != 0) {
236 pw.print(prefix); pw.print("startRequested="); pw.print(startRequested);
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700237 pw.print(" stopIfKilled="); pw.print(stopIfKilled);
238 pw.print(" callStart="); pw.print(callStart);
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700239 pw.print(" lastStartId="); pw.println(lastStartId);
240 }
241 if (executeNesting != 0 || crashCount != 0 || restartCount != 0
242 || restartDelay != 0 || nextRestartTime != 0) {
243 pw.print(prefix); pw.print("executeNesting="); pw.print(executeNesting);
244 pw.print(" restartCount="); pw.print(restartCount);
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700245 pw.print(" restartDelay=");
246 TimeUtils.formatDuration(restartDelay, now, pw);
247 pw.print(" nextRestartTime=");
248 TimeUtils.formatDuration(nextRestartTime, now, pw);
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700249 pw.print(" crashCount="); pw.println(crashCount);
250 }
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700251 if (deliveredStarts.size() > 0) {
252 pw.print(prefix); pw.println("Delivered Starts:");
Dianne Hackborn1ebccf52010-08-15 13:04:34 -0700253 dumpStartList(pw, prefix, deliveredStarts, now);
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700254 }
255 if (pendingStarts.size() > 0) {
256 pw.print(prefix); pw.println("Pending Starts:");
257 dumpStartList(pw, prefix, pendingStarts, 0);
258 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800259 if (bindings.size() > 0) {
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700260 pw.print(prefix); pw.println("Bindings:");
Dianne Hackborn390517b2013-05-30 15:03:32 -0700261 for (int i=0; i<bindings.size(); i++) {
262 IntentBindRecord b = bindings.valueAt(i);
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700263 pw.print(prefix); pw.print("* IntentBindRecord{");
264 pw.print(Integer.toHexString(System.identityHashCode(b)));
Dianne Hackborn0c380492012-08-20 17:23:30 -0700265 if ((b.collectFlags()&Context.BIND_AUTO_CREATE) != 0) {
266 pw.append(" CREATE");
267 }
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700268 pw.println("}:");
269 b.dumpInService(pw, prefix + " ");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800270 }
271 }
272 if (connections.size() > 0) {
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700273 pw.print(prefix); pw.println("All Connections:");
Dianne Hackborn390517b2013-05-30 15:03:32 -0700274 for (int conni=0; conni<connections.size(); conni++) {
275 ArrayList<ConnectionRecord> c = connections.valueAt(conni);
Dianne Hackborn43d9ac82010-08-25 15:06:25 -0700276 for (int i=0; i<c.size(); i++) {
277 pw.print(prefix); pw.print(" "); pw.println(c.get(i));
278 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800279 }
280 }
281 }
282
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800283 ServiceRecord(ActivityManagerService ams,
284 BatteryStatsImpl.Uid.Pkg.Serv servStats, ComponentName name,
Dianne Hackborn91268cf2013-06-13 19:06:50 -0700285 Intent.FilterComparison intent, ServiceInfo sInfo, Runnable restarter,
286 ProcessTracker.ServiceState tracker) {
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800287 this.ams = ams;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800288 this.stats = servStats;
289 this.name = name;
290 shortName = name.flattenToShortString();
291 this.intent = intent;
292 serviceInfo = sInfo;
293 appInfo = sInfo.applicationInfo;
294 packageName = sInfo.applicationInfo.packageName;
295 processName = sInfo.processName;
296 permission = sInfo.permission;
297 baseDir = sInfo.applicationInfo.sourceDir;
298 resDir = sInfo.applicationInfo.publicSourceDir;
299 dataDir = sInfo.applicationInfo.dataDir;
300 exported = sInfo.exported;
301 this.restarter = restarter;
Dianne Hackborn91268cf2013-06-13 19:06:50 -0700302 this.tracker = tracker;
Dianne Hackborn9adb9c32010-08-13 14:09:56 -0700303 createTime = SystemClock.elapsedRealtime();
304 lastActivity = SystemClock.uptimeMillis();
Dianne Hackbornf02b60a2012-08-16 10:48:27 -0700305 userId = UserHandle.getUserId(appInfo.uid);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800306 }
307
308 public AppBindRecord retrieveAppBindingLocked(Intent intent,
309 ProcessRecord app) {
310 Intent.FilterComparison filter = new Intent.FilterComparison(intent);
311 IntentBindRecord i = bindings.get(filter);
312 if (i == null) {
313 i = new IntentBindRecord(this, filter);
314 bindings.put(filter, i);
315 }
316 AppBindRecord a = i.apps.get(app);
317 if (a != null) {
318 return a;
319 }
320 a = new AppBindRecord(this, i, app);
321 i.apps.put(app, a);
322 return a;
323 }
324
Dianne Hackborn91268cf2013-06-13 19:06:50 -0700325 public boolean hasAutoCreateConnections() {
326 // XXX should probably keep a count of the number of auto-create
327 // connections directly in the service.
328 for (int conni=connections.size()-1; conni>=0; conni--) {
329 ArrayList<ConnectionRecord> cr = connections.valueAt(conni);
330 for (int i=0; i<cr.size(); i++) {
331 if ((cr.get(i).flags&Context.BIND_AUTO_CREATE) != 0) {
332 return true;
333 }
334 }
335 }
336 return false;
337 }
338
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800339 public void resetRestartCounter() {
340 restartCount = 0;
341 restartDelay = 0;
342 restartTime = 0;
343 }
344
Dianne Hackbornf6f9f2d2009-08-21 16:26:03 -0700345 public StartItem findDeliveredStart(int id, boolean remove) {
346 final int N = deliveredStarts.size();
347 for (int i=0; i<N; i++) {
348 StartItem si = deliveredStarts.get(i);
349 if (si.id == id) {
350 if (remove) deliveredStarts.remove(i);
351 return si;
352 }
353 }
354
355 return null;
356 }
357
Dianne Hackborn0c5001d2011-04-12 18:16:08 -0700358 public int getLastStartId() {
359 return lastStartId;
360 }
361
362 public int makeNextStartId() {
363 lastStartId++;
364 if (lastStartId < 1) {
365 lastStartId = 1;
366 }
367 return lastStartId;
368 }
369
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700370 public void postNotification() {
Daniel Sandlerd0a2f862010-08-03 15:29:31 -0400371 final int appUid = appInfo.uid;
372 final int appPid = app.pid;
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700373 if (foregroundId != 0 && foregroundNoti != null) {
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800374 // Do asynchronous communication with notification manager to
375 // avoid deadlocks.
376 final String localPackageName = packageName;
377 final int localForegroundId = foregroundId;
378 final Notification localForegroundNoti = foregroundNoti;
379 ams.mHandler.post(new Runnable() {
380 public void run() {
Daniel Sandlerd0a2f862010-08-03 15:29:31 -0400381 NotificationManagerService nm =
382 (NotificationManagerService) NotificationManager.getService();
383 if (nm == null) {
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800384 return;
385 }
386 try {
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700387 if (localForegroundNoti.icon == 0) {
Dianne Hackborn2ced96c2013-03-21 14:23:46 -0700388 // It is not correct for the caller to supply a notification
389 // icon, but this used to be able to slip through, so for
390 // those dirty apps give it the app's icon.
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700391 localForegroundNoti.icon = appInfo.icon;
Daniel Sandler91fe8452013-04-08 12:23:27 -0400392
393 // Do not allow apps to present a sneaky invisible content view either.
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700394 localForegroundNoti.contentView = null;
395 localForegroundNoti.bigContentView = null;
Daniel Sandler91fe8452013-04-08 12:23:27 -0400396 CharSequence appName = appInfo.loadLabel(
397 ams.mContext.getPackageManager());
398 if (appName == null) {
399 appName = appInfo.packageName;
400 }
401 Context ctx = null;
402 try {
403 ctx = ams.mContext.createPackageContext(
404 appInfo.packageName, 0);
405 Intent runningIntent = new Intent(
406 Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
407 runningIntent.setData(Uri.fromParts("package",
408 appInfo.packageName, null));
409 PendingIntent pi = PendingIntent.getActivity(ams.mContext, 0,
410 runningIntent, PendingIntent.FLAG_UPDATE_CURRENT);
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700411 localForegroundNoti.setLatestEventInfo(ctx,
Daniel Sandler91fe8452013-04-08 12:23:27 -0400412 ams.mContext.getString(
413 com.android.internal.R.string
414 .app_running_notification_title,
415 appName),
416 ams.mContext.getString(
417 com.android.internal.R.string
418 .app_running_notification_text,
419 appName),
420 pi);
421 } catch (PackageManager.NameNotFoundException e) {
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700422 localForegroundNoti.icon = 0;
Dianne Hackborn2ced96c2013-03-21 14:23:46 -0700423 }
424 }
Dianne Hackbornca92a4c2013-05-03 12:07:51 -0700425 if (localForegroundNoti.icon == 0) {
Dianne Hackborn282add72013-03-15 18:48:04 -0700426 // Notifications whose icon is 0 are defined to not show
427 // a notification, silently ignoring it. We don't want to
428 // just ignore it, we want to prevent the service from
429 // being foreground.
430 throw new RuntimeException("icon must be non-zero");
431 }
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800432 int[] outId = new int[1];
Dianne Hackbornf265ea92013-01-31 15:00:51 -0800433 nm.enqueueNotificationInternal(localPackageName, localPackageName,
434 appUid, appPid, null, localForegroundId, localForegroundNoti,
435 outId, userId);
Joe Onorato34fcf972010-02-18 07:45:17 -0500436 } catch (RuntimeException e) {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800437 Slog.w(ActivityManagerService.TAG,
Dianne Hackborn9e0f5d92010-02-22 15:05:42 -0800438 "Error showing notification for service", e);
439 // If it gave us a garbage notification, it doesn't
440 // get to be foreground.
441 ams.setServiceForeground(name, ServiceRecord.this,
Joe Onoratoeaa07182010-09-23 16:49:20 -0700442 0, null, true);
443 ams.crashApplication(appUid, appPid, localPackageName,
444 "Bad notification for startForeground: " + e);
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800445 }
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700446 }
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800447 });
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700448 }
449 }
450
451 public void cancelNotification() {
452 if (foregroundId != 0) {
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800453 // Do asynchronous communication with notification manager to
454 // avoid deadlocks.
455 final String localPackageName = packageName;
456 final int localForegroundId = foregroundId;
457 ams.mHandler.post(new Runnable() {
458 public void run() {
459 INotificationManager inm = NotificationManager.getService();
460 if (inm == null) {
461 return;
462 }
463 try {
Dianne Hackborn41203752012-08-31 14:05:51 -0700464 inm.cancelNotificationWithTag(localPackageName, null,
465 localForegroundId, userId);
Joe Onorato34fcf972010-02-18 07:45:17 -0500466 } catch (RuntimeException e) {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800467 Slog.w(ActivityManagerService.TAG,
Dianne Hackborn9e0f5d92010-02-22 15:05:42 -0800468 "Error canceling notification for service", e);
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800469 } catch (RemoteException e) {
470 }
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700471 }
Dianne Hackbornb1c4a2a2010-01-19 15:36:42 -0800472 });
Dianne Hackbornd8a43f62009-08-17 23:33:56 -0700473 }
474 }
475
Dianne Hackborn39792d22010-08-19 18:01:52 -0700476 public void clearDeliveredStartsLocked() {
477 for (int i=deliveredStarts.size()-1; i>=0; i--) {
478 deliveredStarts.get(i).removeUriPermissionsLocked();
479 }
480 deliveredStarts.clear();
481 }
482
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800483 public String toString() {
Dianne Hackbornf210d6b2009-04-13 18:42:49 -0700484 if (stringName != null) {
485 return stringName;
486 }
487 StringBuilder sb = new StringBuilder(128);
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700488 sb.append("ServiceRecord{")
489 .append(Integer.toHexString(System.identityHashCode(this)))
Dianne Hackbornb12e1352012-09-26 11:39:20 -0700490 .append(" u").append(userId)
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700491 .append(' ').append(shortName).append('}');
Dianne Hackbornf210d6b2009-04-13 18:42:49 -0700492 return stringName = sb.toString();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800493 }
494}