diff options
author | Dianne Hackborn <hackbod@google.com> | 2014-04-04 18:02:06 -0700 |
---|---|---|
committer | Dianne Hackborn <hackbod@google.com> | 2014-04-24 17:48:58 -0700 |
commit | 91097de49b0f683b00e26a75dbc0ac6082344137 (patch) | |
tree | 82c3185634a71233ce2e81a3645b07b1ba55f412 /services/voiceinteraction | |
parent | 23af77a3cd1febc740d885ff03ead09837df269c (diff) | |
download | frameworks_base-91097de49b0f683b00e26a75dbc0ac6082344137.zip frameworks_base-91097de49b0f683b00e26a75dbc0ac6082344137.tar.gz frameworks_base-91097de49b0f683b00e26a75dbc0ac6082344137.tar.bz2 |
Initial implementation of new voice interaction API.
This gives a basic working implementation of a persist
running service that can start a voice interaction when
it wants, with the target activity(s) able to go through
the protocol to interact with it. It may even work when
the screen is off by putting the activity manager in the
correct state to act like the screen is on.
Includes a sample app that is a voice interation service
and also has an activity it can launch.
Now that I have this initial implementation, I think I
want to rework some aspects of the API.
Change-Id: I7646d0af8fb4ac768c63a18fe3de43f8091f60e9
Diffstat (limited to 'services/voiceinteraction')
3 files changed, 346 insertions, 0 deletions
diff --git a/services/voiceinteraction/Android.mk b/services/voiceinteraction/Android.mk new file mode 100644 index 0000000..c9e5dd0 --- /dev/null +++ b/services/voiceinteraction/Android.mk @@ -0,0 +1,12 @@ +LOCAL_PATH := $(call my-dir) + +include $(CLEAR_VARS) + +LOCAL_MODULE := services.voiceinteraction + +LOCAL_SRC_FILES += \ + $(call all-java-files-under,java) + +LOCAL_JAVA_LIBRARIES := services.core + +include $(BUILD_STATIC_JAVA_LIBRARY) diff --git a/services/voiceinteraction/java/com/android/server/voiceinteraction/VoiceInteractionManagerService.java b/services/voiceinteraction/java/com/android/server/voiceinteraction/VoiceInteractionManagerService.java new file mode 100644 index 0000000..9e2bcab --- /dev/null +++ b/services/voiceinteraction/java/com/android/server/voiceinteraction/VoiceInteractionManagerService.java @@ -0,0 +1,209 @@ +/* + * Copyright (C) 2014 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.voiceinteraction; + +import android.app.ActivityManager; +import android.content.ComponentName; +import android.content.ContentResolver; +import android.content.Context; +import android.content.Intent; +import android.database.ContentObserver; +import android.os.Binder; +import android.os.Handler; +import android.os.UserHandle; +import android.provider.Settings; +import android.service.voice.IVoiceInteractionService; +import android.service.voice.IVoiceInteractionSession; +import android.util.Slog; +import com.android.internal.app.IVoiceInteractionManagerService; +import com.android.internal.app.IVoiceInteractor; +import com.android.internal.content.PackageMonitor; +import com.android.internal.os.BackgroundThread; +import com.android.server.SystemService; +import com.android.server.UiThread; + +import java.io.FileDescriptor; +import java.io.PrintWriter; + + +/** + * SystemService that publishes an IVoiceInteractionManagerService. + */ +public class VoiceInteractionManagerService extends SystemService { + + static final String TAG = "VoiceInteractionManagerService"; + + final Context mContext; + final ContentResolver mResolver; + + public VoiceInteractionManagerService(Context context) { + super(context); + mContext = context; + mResolver = context.getContentResolver(); + } + + @Override + public void onStart() { + publishBinderService(Context.VOICE_INTERACTION_MANAGER_SERVICE, mServiceStub); + } + + @Override + public void onBootPhase(int phase) { + if (phase == PHASE_THIRD_PARTY_APPS_CAN_START) { + mServiceStub.systemRunning(isSafeMode()); + } + } + + @Override + public void onSwitchUser(int userHandle) { + mServiceStub.switchUser(userHandle); + } + + // implementation entry point and binder service + private final VoiceInteractionManagerServiceStub mServiceStub + = new VoiceInteractionManagerServiceStub(); + + class VoiceInteractionManagerServiceStub extends IVoiceInteractionManagerService.Stub { + + VoiceInteractionManagerServiceImpl mImpl; + + private boolean mSafeMode; + private int mCurUser; + + public void systemRunning(boolean safeMode) { + mSafeMode = safeMode; + + mPackageMonitor.register(mContext, BackgroundThread.getHandler().getLooper(), + UserHandle.ALL, true); + new SettingsObserver(UiThread.getHandler()); + + synchronized (this) { + mCurUser = ActivityManager.getCurrentUser(); + switchImplementationIfNeededLocked(); + } + } + + public void switchUser(int userHandle) { + synchronized (this) { + mCurUser = userHandle; + switchImplementationIfNeededLocked(); + } + } + + void switchImplementationIfNeededLocked() { + if (!mSafeMode) { + String curService = Settings.Secure.getStringForUser( + mResolver, Settings.Secure.VOICE_INTERACTION_SERVICE, mCurUser); + ComponentName serviceComponent = null; + if (curService != null && !curService.isEmpty()) { + try { + serviceComponent = ComponentName.unflattenFromString(curService); + } catch (RuntimeException e) { + Slog.wtf(TAG, "Bad voice interaction service name " + curService, e); + serviceComponent = null; + } + } + if (mImpl == null || mImpl.mUser != mCurUser + || !mImpl.mComponent.equals(serviceComponent)) { + if (mImpl != null) { + mImpl.shutdownLocked(); + } + if (serviceComponent != null) { + mImpl = new VoiceInteractionManagerServiceImpl(mContext, + UiThread.getHandler(), this, mCurUser, serviceComponent); + mImpl.startLocked(); + } else { + mImpl = null; + } + } + } + } + + @Override + public int startVoiceActivity(Intent intent, String resolvedType, + IVoiceInteractionService service, + IVoiceInteractionSession session, IVoiceInteractor interactor) { + synchronized (this) { + if (mImpl == null || service.asBinder() != mImpl.mService.asBinder()) { + throw new SecurityException( + "Caller is not the current voice interaction service"); + } + final int callingPid = Binder.getCallingPid(); + final int callingUid = Binder.getCallingUid(); + final long caller = Binder.clearCallingIdentity(); + try { + return mImpl.startVoiceActivityLocked(callingPid, callingUid, + intent, resolvedType, session, interactor); + } finally { + Binder.restoreCallingIdentity(caller); + } + } + } + + @Override + public void dump(FileDescriptor fd, PrintWriter pw, String[] args) { + mContext.enforceCallingOrSelfPermission(android.Manifest.permission.DUMP, TAG); + } + + class SettingsObserver extends ContentObserver { + SettingsObserver(Handler handler) { + super(handler); + ContentResolver resolver = mContext.getContentResolver(); + resolver.registerContentObserver(Settings.Secure.getUriFor( + Settings.Secure.VOICE_INTERACTION_SERVICE), false, this); + } + + @Override public void onChange(boolean selfChange) { + synchronized (VoiceInteractionManagerServiceStub.this) { + switchImplementationIfNeededLocked(); + } + } + } + + PackageMonitor mPackageMonitor = new PackageMonitor() { + @Override + public boolean onHandleForceStop(Intent intent, String[] packages, int uid, boolean doit) { + return super.onHandleForceStop(intent, packages, uid, doit); + } + + @Override + public void onHandleUserStop(Intent intent, int userHandle) { + super.onHandleUserStop(intent, userHandle); + } + + @Override + public void onPackageDisappeared(String packageName, int reason) { + super.onPackageDisappeared(packageName, reason); + } + + @Override + public void onPackageAppeared(String packageName, int reason) { + super.onPackageAppeared(packageName, reason); + } + + @Override + public void onPackageModified(String packageName) { + super.onPackageModified(packageName); + } + + @Override + public void onSomePackagesChanged() { + super.onSomePackagesChanged(); + } + }; + } +} diff --git a/services/voiceinteraction/java/com/android/server/voiceinteraction/VoiceInteractionManagerServiceImpl.java b/services/voiceinteraction/java/com/android/server/voiceinteraction/VoiceInteractionManagerServiceImpl.java new file mode 100644 index 0000000..af8ae1e --- /dev/null +++ b/services/voiceinteraction/java/com/android/server/voiceinteraction/VoiceInteractionManagerServiceImpl.java @@ -0,0 +1,125 @@ +/* + * Copyright (C) 2014 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.voiceinteraction; + +import android.app.ActivityManagerNative; +import android.app.IActivityManager; +import android.content.ComponentName; +import android.content.Context; +import android.content.Intent; +import android.content.ServiceConnection; +import android.content.pm.PackageManager; +import android.content.pm.ServiceInfo; +import android.os.Handler; +import android.os.IBinder; +import android.os.RemoteException; +import android.os.UserHandle; +import android.service.voice.IVoiceInteractionService; +import android.service.voice.IVoiceInteractionSession; +import android.service.voice.VoiceInteractionService; +import android.util.Slog; +import com.android.internal.app.IVoiceInteractor; + +class VoiceInteractionManagerServiceImpl { + final static String TAG = "VoiceInteractionServiceManager"; + + final Context mContext; + final Handler mHandler; + final Object mLock; + final int mUser; + final ComponentName mComponent; + final IActivityManager mAm; + boolean mBound = false; + IVoiceInteractionService mService; + IVoiceInteractionSession mActiveSession; + IVoiceInteractor mActiveInteractor; + + final ServiceConnection mConnection = new ServiceConnection() { + @Override + public void onServiceConnected(ComponentName name, IBinder service) { + synchronized (mLock) { + mService = IVoiceInteractionService.Stub.asInterface(service); + } + } + + @Override + public void onServiceDisconnected(ComponentName name) { + mService = null; + } + }; + + VoiceInteractionManagerServiceImpl(Context context, Handler handler, Object lock, + int userHandle, ComponentName service) { + mContext = context; + mHandler = handler; + mLock = lock; + mUser = userHandle; + mComponent = service; + mAm = ActivityManagerNative.getDefault(); + } + + public int startVoiceActivityLocked(int callingPid, int callingUid, Intent intent, + String resolvedType, IVoiceInteractionSession session, IVoiceInteractor interactor) { + if (session == null) { + throw new NullPointerException("session is null"); + } + if (interactor == null) { + throw new NullPointerException("interactor is null"); + } + if (mActiveSession != null) { + // XXX cancel current session. + } + intent.addCategory(Intent.CATEGORY_VOICE); + intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_MULTIPLE_TASK); + mActiveSession = session; + mActiveInteractor = interactor; + try { + return mAm.startVoiceActivity(mComponent.getPackageName(), callingPid, callingUid, + intent, resolvedType, mActiveSession, mActiveInteractor, + 0, null, null, null, mUser); + } catch (RemoteException e) { + throw new IllegalStateException("Unexpected remote error", e); + } + } + + void startLocked() { + Intent intent = new Intent(VoiceInteractionService.SERVICE_INTERFACE); + intent.setComponent(mComponent); + try { + ServiceInfo si = mContext.getPackageManager().getServiceInfo(mComponent, 0); + if (!android.Manifest.permission.BIND_VOICE_INTERACTION.equals(si.permission)) { + Slog.w(TAG, "Not using voice interaction service " + mComponent + + ": does not require permission " + + android.Manifest.permission.BIND_VOICE_INTERACTION); + return; + } + } catch (PackageManager.NameNotFoundException e) { + Slog.w(TAG, "Unable to find voice interaction service: " + mComponent, e); + return; + } + mContext.bindServiceAsUser(intent, mConnection, + Context.BIND_AUTO_CREATE, new UserHandle(mUser)); + mBound = true; + } + + void shutdownLocked() { + if (mBound) { + mContext.unbindService(mConnection); + mBound = false; + } + } +} |