| /* |
| * Copyright (C) 2018 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.contentsuggestions; |
| |
| import static android.Manifest.permission.MANAGE_CONTENT_SUGGESTIONS; |
| import static android.content.pm.PackageManager.PERMISSION_GRANTED; |
| |
| import android.annotation.NonNull; |
| import android.annotation.Nullable; |
| import android.annotation.UserIdInt; |
| import android.app.contentsuggestions.ClassificationsRequest; |
| import android.app.contentsuggestions.ContentSuggestionsManager; |
| import android.app.contentsuggestions.IClassificationsCallback; |
| import android.app.contentsuggestions.IContentSuggestionsManager; |
| import android.app.contentsuggestions.ISelectionsCallback; |
| import android.app.contentsuggestions.SelectionsRequest; |
| import android.content.Context; |
| import android.graphics.Bitmap; |
| import android.graphics.ColorSpace; |
| import android.hardware.HardwareBuffer; |
| import android.os.Binder; |
| import android.os.Bundle; |
| import android.os.RemoteException; |
| import android.os.ResultReceiver; |
| import android.os.ShellCallback; |
| import android.os.UserHandle; |
| import android.os.UserManager; |
| import android.util.Slog; |
| import android.window.TaskSnapshot; |
| |
| import com.android.internal.os.IResultReceiver; |
| import com.android.server.LocalServices; |
| import com.android.server.infra.AbstractMasterSystemService; |
| import com.android.server.infra.FrameworkResourcesServiceNameResolver; |
| import com.android.server.wm.ActivityTaskManagerInternal; |
| |
| import java.io.FileDescriptor; |
| |
| /** |
| * The system service for providing recents / overview with content suggestion selections and |
| * classifications. |
| * |
| * <p>Calls are received here from |
| * {@link android.app.contentsuggestions.ContentSuggestionsManager} then delegated to |
| * a per user version of the service. From there they are routed to the remote actual implementation |
| * that provides the suggestion selections and classifications. |
| */ |
| public class ContentSuggestionsManagerService extends |
| AbstractMasterSystemService< |
| ContentSuggestionsManagerService, ContentSuggestionsPerUserService> { |
| |
| private static final String TAG = ContentSuggestionsManagerService.class.getSimpleName(); |
| private static final boolean VERBOSE = false; // TODO: make dynamic |
| |
| private static final int MAX_TEMP_SERVICE_DURATION_MS = 1_000 * 60 * 2; // 2 minutes |
| /** |
| * Key into the extras Bundle passed to {@link #provideContextImage(int, Bundle)}. |
| */ |
| private static final String EXTRA_BITMAP = "android.contentsuggestions.extra.BITMAP"; |
| |
| private ActivityTaskManagerInternal mActivityTaskManagerInternal; |
| |
| public ContentSuggestionsManagerService(Context context) { |
| super(context, new FrameworkResourcesServiceNameResolver(context, |
| com.android.internal.R.string.config_defaultContentSuggestionsService), |
| UserManager.DISALLOW_CONTENT_SUGGESTIONS); |
| mActivityTaskManagerInternal = LocalServices.getService(ActivityTaskManagerInternal.class); |
| } |
| |
| @Override |
| protected ContentSuggestionsPerUserService newServiceLocked(int resolvedUserId, |
| boolean disabled) { |
| return new ContentSuggestionsPerUserService(this, mLock, resolvedUserId); |
| } |
| |
| @Override |
| public void onStart() { |
| publishBinderService( |
| Context.CONTENT_SUGGESTIONS_SERVICE, new ContentSuggestionsManagerStub()); |
| } |
| |
| @Override |
| protected void enforceCallingPermissionForManagement() { |
| getContext().enforceCallingPermission(MANAGE_CONTENT_SUGGESTIONS, TAG); |
| } |
| |
| @Override |
| protected int getMaximumTemporaryServiceDurationMs() { |
| return MAX_TEMP_SERVICE_DURATION_MS; |
| } |
| |
| private void enforceCaller(int userId, String func) { |
| Context ctx = getContext(); |
| if (ctx.checkCallingPermission(MANAGE_CONTENT_SUGGESTIONS) == PERMISSION_GRANTED |
| || mServiceNameResolver.isTemporary(userId) |
| || mActivityTaskManagerInternal.isCallerRecents(Binder.getCallingUid())) { |
| return; |
| } |
| |
| String msg = "Permission Denial: " + func + " from pid=" |
| + Binder.getCallingPid() |
| + ", uid=" + Binder.getCallingUid() |
| + " expected caller is recents"; |
| Slog.w(TAG, msg); |
| throw new SecurityException(msg); |
| } |
| |
| private class ContentSuggestionsManagerStub extends IContentSuggestionsManager.Stub { |
| @Override |
| public void provideContextBitmap( |
| int userId, |
| @NonNull Bitmap bitmap, |
| @NonNull Bundle imageContextRequestExtras) { |
| if (bitmap == null) { |
| throw new IllegalArgumentException("Expected non-null bitmap"); |
| } |
| if (imageContextRequestExtras == null) { |
| throw new IllegalArgumentException("Expected non-null imageContextRequestExtras"); |
| } |
| enforceCaller(UserHandle.getCallingUserId(), "provideContextBitmap"); |
| |
| synchronized (mLock) { |
| final ContentSuggestionsPerUserService service = getServiceForUserLocked(userId); |
| if (service != null) { |
| // TODO(b/147324195): Temporarily pass bitmap until we change the service API. |
| imageContextRequestExtras.putParcelable(EXTRA_BITMAP, bitmap); |
| service.provideContextImageFromBitmapLocked(imageContextRequestExtras); |
| } else { |
| if (VERBOSE) { |
| Slog.v(TAG, "provideContextImageLocked: no service for " + userId); |
| } |
| } |
| } |
| } |
| |
| @Override |
| public void provideContextImage( |
| int userId, |
| int taskId, |
| @NonNull Bundle imageContextRequestExtras) { |
| if (imageContextRequestExtras == null) { |
| throw new IllegalArgumentException("Expected non-null imageContextRequestExtras"); |
| } |
| enforceCaller(UserHandle.getCallingUserId(), "provideContextImage"); |
| |
| HardwareBuffer snapshotBuffer = null; |
| int colorSpaceId = 0; |
| |
| // Skip taking TaskSnapshot when bitmap is provided. |
| if (!imageContextRequestExtras.containsKey(ContentSuggestionsManager.EXTRA_BITMAP)) { |
| // Can block, so call before acquiring the lock. |
| TaskSnapshot snapshot = |
| mActivityTaskManagerInternal.getTaskSnapshotBlocking(taskId, false); |
| if (snapshot != null) { |
| snapshotBuffer = snapshot.getHardwareBuffer(); |
| ColorSpace colorSpace = snapshot.getColorSpace(); |
| if (colorSpace != null) { |
| colorSpaceId = colorSpace.getId(); |
| } |
| } |
| } |
| |
| synchronized (mLock) { |
| final ContentSuggestionsPerUserService service = getServiceForUserLocked(userId); |
| if (service != null) { |
| service.provideContextImageLocked(taskId, snapshotBuffer, colorSpaceId, |
| imageContextRequestExtras); |
| } else { |
| if (VERBOSE) { |
| Slog.v(TAG, "provideContextImageLocked: no service for " + userId); |
| } |
| } |
| } |
| } |
| |
| @Override |
| public void suggestContentSelections( |
| int userId, |
| @NonNull SelectionsRequest selectionsRequest, |
| @NonNull ISelectionsCallback selectionsCallback) { |
| enforceCaller(UserHandle.getCallingUserId(), "suggestContentSelections"); |
| |
| synchronized (mLock) { |
| final ContentSuggestionsPerUserService service = getServiceForUserLocked(userId); |
| if (service != null) { |
| service.suggestContentSelectionsLocked(selectionsRequest, selectionsCallback); |
| } else { |
| if (VERBOSE) { |
| Slog.v(TAG, "suggestContentSelectionsLocked: no service for " + userId); |
| } |
| } |
| } |
| } |
| |
| @Override |
| public void classifyContentSelections( |
| int userId, |
| @NonNull ClassificationsRequest classificationsRequest, |
| @NonNull IClassificationsCallback callback) { |
| enforceCaller(UserHandle.getCallingUserId(), "classifyContentSelections"); |
| |
| synchronized (mLock) { |
| final ContentSuggestionsPerUserService service = getServiceForUserLocked(userId); |
| if (service != null) { |
| service.classifyContentSelectionsLocked(classificationsRequest, callback); |
| } else { |
| if (VERBOSE) { |
| Slog.v(TAG, "classifyContentSelectionsLocked: no service for " + userId); |
| } |
| } |
| } |
| } |
| |
| @Override |
| public void notifyInteraction( |
| int userId, @NonNull String requestId, @NonNull Bundle bundle) { |
| enforceCaller(UserHandle.getCallingUserId(), "notifyInteraction"); |
| |
| synchronized (mLock) { |
| final ContentSuggestionsPerUserService service = getServiceForUserLocked(userId); |
| if (service != null) { |
| service.notifyInteractionLocked(requestId, bundle); |
| } else { |
| if (VERBOSE) { |
| Slog.v(TAG, "reportInteractionLocked: no service for " + userId); |
| } |
| } |
| } |
| } |
| |
| @Override |
| public void isEnabled(int userId, @NonNull IResultReceiver receiver) |
| throws RemoteException { |
| enforceCaller(UserHandle.getCallingUserId(), "isEnabled"); |
| |
| boolean isDisabled; |
| synchronized (mLock) { |
| isDisabled = isDisabledLocked(userId); |
| } |
| receiver.send(isDisabled ? 0 : 1, null); |
| } |
| |
| @Override |
| public void resetTemporaryService(@UserIdInt int userId) { |
| ContentSuggestionsManagerService.this.resetTemporaryService(userId); |
| } |
| |
| @Override |
| public void setTemporaryService( |
| @UserIdInt int userId, @NonNull String serviceName, int duration) { |
| ContentSuggestionsManagerService.this.setTemporaryService( |
| userId, serviceName, duration); |
| } |
| |
| @Override |
| public void setDefaultServiceEnabled(@UserIdInt int userId, boolean enabled) { |
| ContentSuggestionsManagerService.this.setDefaultServiceEnabled(userId, enabled); |
| } |
| |
| public void onShellCommand(@Nullable FileDescriptor in, @Nullable FileDescriptor out, |
| @Nullable FileDescriptor err, |
| @NonNull String[] args, @Nullable ShellCallback callback, |
| @NonNull ResultReceiver resultReceiver) throws RemoteException { |
| // Ensure that the caller is the shell process |
| final int callingUid = Binder.getCallingUid(); |
| if (callingUid != android.os.Process.SHELL_UID |
| && callingUid != android.os.Process.ROOT_UID) { |
| Slog.e(TAG, "Expected shell caller"); |
| return; |
| } |
| new ContentSuggestionsManagerServiceShellCommand(ContentSuggestionsManagerService.this) |
| .exec(this, in, out, err, args, callback, resultReceiver); |
| } |
| } |
| } |