/* * Copyright (C) 2015 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.mtp; import android.annotation.NonNull; import android.annotation.Nullable; import android.app.Notification; import android.app.Service; import android.app.NotificationManager; import android.content.Intent; import android.os.IBinder; import android.os.Parcelable; import android.service.notification.StatusBarNotification; import android.util.Log; import com.android.internal.util.Preconditions; import java.util.HashSet; import java.util.Set; /** * Service to manage lifetime of DocumentsProvider's process. * The service prevents the system from killing the process that holds USB connections. The service * starts to run when the first MTP device is opened, and stops when the last MTP device is closed. */ public class MtpDocumentsService extends Service { static final String ACTION_UPDATE_NOTIFICATION = "com.android.mtp.UPDATE_NOTIFICATION"; static final String EXTRA_DEVICE_IDS = "deviceIds"; static final String EXTRA_DEVICE_NOTIFICATIONS = "deviceNotifications"; private NotificationManager mNotificationManager; @Override public IBinder onBind(Intent intent) { // The service is used via intents. return null; } @Override public void onCreate() { super.onCreate(); mNotificationManager = getSystemService(NotificationManager.class); } @Override public int onStartCommand(Intent intent, int flags, int startId) { // If intent is null, the service was restarted. if (intent == null || ACTION_UPDATE_NOTIFICATION.equals(intent.getAction())) { final int[] ids = intent.hasExtra(EXTRA_DEVICE_IDS) ? intent.getExtras().getIntArray(EXTRA_DEVICE_IDS) : null; final Notification[] notifications = intent.hasExtra(EXTRA_DEVICE_NOTIFICATIONS) ? castToNotifications(intent.getExtras().getParcelableArray( EXTRA_DEVICE_NOTIFICATIONS)) : null; return updateForegroundState(ids, notifications) ? START_STICKY : START_NOT_STICKY; } return START_NOT_STICKY; } /** * Updates the foreground state of the service. * @return Whether the service is foreground or not. */ private boolean updateForegroundState( @Nullable int[] ids, @Nullable Notification[] notifications) { final Set openedNotification = new HashSet<>(); final int size = ids != null ? ids.length : 0; if (size != 0) { Preconditions.checkArgument(ids != null); Preconditions.checkArgument(notifications != null); Preconditions.checkArgument(ids.length == notifications.length); } for (int i = 0; i < size; i++) { if (i == 0) { // Mark this service as foreground with the notification so that the process is // not killed by the system while a MTP device is opened. startForeground(ids[i], notifications[i]); } else { // Only one notification can be shown as a foreground notification. We need to // show the rest as normal notification. mNotificationManager.notify(ids[i], notifications[i]); } openedNotification.add(ids[i]); } final StatusBarNotification[] activeNotifications = mNotificationManager.getActiveNotifications(); for (final StatusBarNotification notification : activeNotifications) { if (!openedNotification.contains(notification.getId())) { mNotificationManager.cancel(notification.getId()); } } if (size == 0) { // There is no opened device. stopForeground(true /* removeNotification */); stopSelf(); return false; } return true; } private static @NonNull Notification[] castToNotifications(@NonNull Parcelable[] src) { Preconditions.checkNotNull(src); final Notification[] notifications = new Notification[src.length]; for (int i = 0; i < src.length; i++) { notifications[i] = (Notification) src[i]; } return notifications; } }