/* * Copyright (C) 2007 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.server; import android.app.PendingIntent; import android.app.StatusBarManager; import android.content.BroadcastReceiver; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.pm.PackageManager; import android.content.res.Resources; import android.net.Uri; import android.os.IBinder; import android.os.RemoteException; import android.os.Binder; import android.os.Handler; import android.os.SystemClock; import android.util.Slog; import com.android.internal.statusbar.IStatusBar; import com.android.internal.statusbar.IStatusBarService; import com.android.internal.statusbar.StatusBarIcon; import com.android.internal.statusbar.StatusBarIconList; import com.android.internal.statusbar.StatusBarNotification; import java.io.FileDescriptor; import java.io.PrintWriter; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; /** * A note on locking: We rely on the fact that calls onto mBar are oneway or * if they are local, that they just enqueue messages to not deadlock. */ public class StatusBarManagerService extends IStatusBarService.Stub { static final String TAG = "StatusBarManagerService"; static final boolean SPEW = false; final Context mContext; Handler mHandler = new Handler(); NotificationCallbacks mNotificationCallbacks; volatile IStatusBar mBar; StatusBarIconList mIcons = new StatusBarIconList(); HashMap mNotifications = new HashMap(); // for disabling the status bar ArrayList mDisableRecords = new ArrayList(); int mDisabled = 0; private class DisableRecord implements IBinder.DeathRecipient { String pkg; int what; IBinder token; public void binderDied() { Slog.i(TAG, "binder died for pkg=" + pkg); disable(0, token, pkg); token.unlinkToDeath(this, 0); } } public interface NotificationCallbacks { void onSetDisabled(int status); void onClearAll(); void onNotificationClick(String pkg, String tag, int id); void onPanelRevealed(); void onNotificationError(String pkg, String tag, int id, int uid, int initialPid, String message); } /** * Construct the service, add the status bar view to the window manager */ public StatusBarManagerService(Context context) { mContext = context; final Resources res = context.getResources(); mIcons.defineSlots(res.getStringArray(com.android.internal.R.array.config_statusBarIcons)); } public void setNotificationCallbacks(NotificationCallbacks listener) { mNotificationCallbacks = listener; } // ================================================================================ // Constructing the view // ================================================================================ public void systemReady() { } public void systemReady2() { ComponentName cn = ComponentName.unflattenFromString( mContext.getString(com.android.internal.R.string.config_statusBarComponent)); Intent intent = new Intent(); intent.setComponent(cn); Slog.i(TAG, "Starting service: " + cn); mContext.startService(intent); } // ================================================================================ // From IStatusBarService // ================================================================================ public void expand() { enforceExpandStatusBar(); if (mBar != null) { try { mBar.animateExpand(); } catch (RemoteException ex) { } } } public void collapse() { enforceExpandStatusBar(); if (mBar != null) { try { mBar.animateCollapse(); } catch (RemoteException ex) { } } } public void disable(int what, IBinder token, String pkg) { enforceStatusBar(); // It's important that the the callback and the call to mBar get done // in the same order when multiple threads are calling this function // so they are paired correctly. The messages on the handler will be // handled in the order they were enqueued, but will be outside the lock. synchronized (mDisableRecords) { manageDisableListLocked(what, token, pkg); final int net = gatherDisableActionsLocked(); if (net != mDisabled) { mDisabled = net; mHandler.post(new Runnable() { public void run() { mNotificationCallbacks.onSetDisabled(net); } }); if (mBar != null) { try { mBar.disable(net); } catch (RemoteException ex) { } } } } } public void setIcon(String slot, String iconPackage, int iconId, int iconLevel) { enforceStatusBar(); synchronized (mIcons) { int index = mIcons.getSlotIndex(slot); if (index < 0) { throw new SecurityException("invalid status bar icon slot: " + slot); } StatusBarIcon icon = new StatusBarIcon(iconPackage, iconId, iconLevel); //Slog.d(TAG, "setIcon slot=" + slot + " index=" + index + " icon=" + icon); mIcons.setIcon(index, icon); if (mBar != null) { try { mBar.setIcon(index, icon); } catch (RemoteException ex) { } } } } public void setIconVisibility(String slot, boolean visible) { enforceStatusBar(); synchronized (mIcons) { int index = mIcons.getSlotIndex(slot); if (index < 0) { throw new SecurityException("invalid status bar icon slot: " + slot); } StatusBarIcon icon = mIcons.getIcon(index); if (icon == null) { return; } if (icon.visible != visible) { icon.visible = visible; if (mBar != null) { try { mBar.setIcon(index, icon); } catch (RemoteException ex) { } } } } } public void removeIcon(String slot) { enforceStatusBar(); synchronized (mIcons) { int index = mIcons.getSlotIndex(slot); if (index < 0) { throw new SecurityException("invalid status bar icon slot: " + slot); } mIcons.removeIcon(index); if (mBar != null) { try { mBar.removeIcon(index); } catch (RemoteException ex) { } } } } private void enforceStatusBar() { mContext.enforceCallingOrSelfPermission(android.Manifest.permission.STATUS_BAR, "StatusBarManagerService"); } private void enforceExpandStatusBar() { mContext.enforceCallingOrSelfPermission(android.Manifest.permission.EXPAND_STATUS_BAR, "StatusBarManagerService"); } private void enforceStatusBarService() { mContext.enforceCallingOrSelfPermission(android.Manifest.permission.STATUS_BAR_SERVICE, "StatusBarManagerService"); } // ================================================================================ // Callbacks from the status bar service. // ================================================================================ public void registerStatusBar(IStatusBar bar, StatusBarIconList iconList, List notificationKeys, List notifications) { enforceStatusBarService(); Slog.i(TAG, "registerStatusBar bar=" + bar); mBar = bar; synchronized (mIcons) { iconList.copyFrom(mIcons); } synchronized (mNotifications) { for (Map.Entry e: mNotifications.entrySet()) { notificationKeys.add(e.getKey()); notifications.add(e.getValue()); } } } /** * The status bar service should call this each time the user brings the panel from * invisible to visible in order to clear the notification light. */ public void onPanelRevealed() { enforceStatusBarService(); // tell the notification manager to turn off the lights. mNotificationCallbacks.onPanelRevealed(); } public void onNotificationClick(String pkg, String tag, int id) { enforceStatusBarService(); mNotificationCallbacks.onNotificationClick(pkg, tag, id); } public void onNotificationError(String pkg, String tag, int id, int uid, int initialPid, String message) { enforceStatusBarService(); // WARNING: this will call back into us to do the remove. Don't hold any locks. mNotificationCallbacks.onNotificationError(pkg, tag, id, uid, initialPid, message); } public void onClearAllNotifications() { enforceStatusBarService(); mNotificationCallbacks.onClearAll(); } // ================================================================================ // Callbacks for NotificationManagerService. // ================================================================================ public IBinder addNotification(StatusBarNotification notification) { synchronized (mNotifications) { IBinder key = new Binder(); mNotifications.put(key, notification); if (mBar != null) { try { mBar.addNotification(key, notification); } catch (RemoteException ex) { } } return key; } } public void updateNotification(IBinder key, StatusBarNotification notification) { synchronized (mNotifications) { if (!mNotifications.containsKey(key)) { throw new IllegalArgumentException("updateNotification key not found: " + key); } mNotifications.put(key, notification); if (mBar != null) { try { mBar.updateNotification(key, notification); } catch (RemoteException ex) { } } } } public void removeNotification(IBinder key) { synchronized (mNotifications) { final StatusBarNotification n = mNotifications.remove(key); if (n == null) { throw new IllegalArgumentException("removeNotification key not found: " + key); } if (mBar != null) { try { mBar.removeNotification(key); } catch (RemoteException ex) { } } } } // ================================================================================ // Can be called from any thread // ================================================================================ // lock on mDisableRecords void manageDisableListLocked(int what, IBinder token, String pkg) { if (SPEW) { Slog.d(TAG, "manageDisableList what=0x" + Integer.toHexString(what) + " pkg=" + pkg); } // update the list synchronized (mDisableRecords) { final int N = mDisableRecords.size(); DisableRecord tok = null; int i; for (i=0; i e: mNotifications.entrySet()) { pw.printf(" %2d: %s\n", i, e.getValue().toString()); i++; } } synchronized (mDisableRecords) { final int N = mDisableRecords.size(); pw.println(" mDisableRecords.size=" + N + " mDisabled=0x" + Integer.toHexString(mDisabled)); for (int i=0; i