/* * Copyright (C) 2013 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 android.support.v7.media; import android.app.Service; import android.content.Intent; import android.os.Handler; import android.os.IBinder; import android.os.IBinder.DeathRecipient; import android.os.Bundle; import android.os.DeadObjectException; import android.os.Message; import android.os.Messenger; import android.os.RemoteException; import android.util.Log; import android.util.SparseArray; import java.lang.ref.WeakReference; import java.util.ArrayList; /** * Base class for media route provider services. *

* To implement your own media route provider service, extend this class and * override the {@link #onCreateMediaRouteProvider} method to return an * instance of your {@link MediaRouteProvider}. *

* Declare your media route provider service in your application manifest * like this: *

*
 *   <service android:name=".MyMediaRouteProviderService"
 *           android:label="@string/my_media_route_provider_service">
 *       <intent-filter>
 *           <action android:name="android.media.MediaRouteProviderService" />
 *       </intent-filter>
 *   </service>
 * 
*/ public abstract class MediaRouteProviderService extends Service { private static final String TAG = "MediaRouteProviderSrv"; // max. 23 chars private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG); private final ArrayList mClients = new ArrayList(); private final ReceiveHandler mReceiveHandler; private final Messenger mReceiveMessenger; private final PrivateHandler mPrivateHandler; private final ProviderCallback mProviderCallback; private MediaRouteProvider mProvider; private MediaRouteDiscoveryRequest mCompositeDiscoveryRequest; /** * The {@link Intent} that must be declared as handled by the service. * Put this in your manifest. */ public static final String SERVICE_INTERFACE = "android.media.MediaRouteProviderService"; /* * Messages sent from the client to the service. * DO NOT RENUMBER THESE! */ /** (client v1) * Register client. * - replyTo : client messenger * - arg1 : request id * - arg2 : client version */ static final int CLIENT_MSG_REGISTER = 1; /** (client v1) * Unregister client. * - replyTo : client messenger * - arg1 : request id */ static final int CLIENT_MSG_UNREGISTER = 2; /** (client v1) * Create route controller. * - replyTo : client messenger * - arg1 : request id * - arg2 : route controller id * - CLIENT_DATA_ROUTE_ID : route id string */ static final int CLIENT_MSG_CREATE_ROUTE_CONTROLLER = 3; /** (client v1) * Release route controller. * - replyTo : client messenger * - arg1 : request id * - arg2 : route controller id */ static final int CLIENT_MSG_RELEASE_ROUTE_CONTROLLER = 4; /** (client v1) * Select route. * - replyTo : client messenger * - arg1 : request id * - arg2 : route controller id */ static final int CLIENT_MSG_SELECT_ROUTE = 5; /** (client v1) * Unselect route. * - replyTo : client messenger * - arg1 : request id * - arg2 : route controller id */ static final int CLIENT_MSG_UNSELECT_ROUTE = 6; /** (client v1) * Set route volume. * - replyTo : client messenger * - arg1 : request id * - arg2 : route controller id * - CLIENT_DATA_VOLUME : volume integer */ static final int CLIENT_MSG_SET_ROUTE_VOLUME = 7; /** (client v1) * Update route volume. * - replyTo : client messenger * - arg1 : request id * - arg2 : route controller id * - CLIENT_DATA_VOLUME : volume delta integer */ static final int CLIENT_MSG_UPDATE_ROUTE_VOLUME = 8; /** (client v1) * Route control request. * - replyTo : client messenger * - arg1 : request id * - arg2 : route controller id * - obj : media control intent */ static final int CLIENT_MSG_ROUTE_CONTROL_REQUEST = 9; /** (client v1) * Sets the discovery request. * - replyTo : client messenger * - arg1 : request id * - obj : discovery request bundle, or null if none */ static final int CLIENT_MSG_SET_DISCOVERY_REQUEST = 10; static final String CLIENT_DATA_ROUTE_ID = "routeId"; static final String CLIENT_DATA_VOLUME = "volume"; /* * Messages sent from the service to the client. * DO NOT RENUMBER THESE! */ /** (service v1) * Generic failure sent in response to any unrecognized or malformed request. * - arg1 : request id */ static final int SERVICE_MSG_GENERIC_FAILURE = 0; /** (service v1) * Generic failure sent in response to a successful message. * - arg1 : request id */ static final int SERVICE_MSG_GENERIC_SUCCESS = 1; /** (service v1) * Registration succeeded. * - arg1 : request id * - arg2 : server version * - obj : route provider descriptor bundle, or null */ static final int SERVICE_MSG_REGISTERED = 2; /** (service v1) * Route control request success result. * - arg1 : request id * - obj : result data bundle, or null */ static final int SERVICE_MSG_CONTROL_REQUEST_SUCCEEDED = 3; /** (service v1) * Route control request failure result. * - arg1 : request id * - obj : result data bundle, or null * - SERVICE_DATA_ERROR: error message */ static final int SERVICE_MSG_CONTROL_REQUEST_FAILED = 4; /** (service v1) * Route provider descriptor changed. (unsolicited event) * - arg1 : reserved (0) * - obj : route provider descriptor bundle, or null */ static final int SERVICE_MSG_DESCRIPTOR_CHANGED = 5; static final String SERVICE_DATA_ERROR = "error"; /* * Recognized client version numbers. (Reserved for future use.) * DO NOT RENUMBER THESE! */ static final int CLIENT_VERSION_1 = 1; static final int CLIENT_VERSION_CURRENT = CLIENT_VERSION_1; /* * Recognized server version numbers. (Reserved for future use.) * DO NOT RENUMBER THESE! */ static final int SERVICE_VERSION_1 = 1; static final int SERVICE_VERSION_CURRENT = SERVICE_VERSION_1; /* * Private messages used internally. (Yes, you can renumber these.) */ private static final int PRIVATE_MSG_CLIENT_DIED = 1; /** * Creates a media route provider service. */ public MediaRouteProviderService() { mReceiveHandler = new ReceiveHandler(this); mReceiveMessenger = new Messenger(mReceiveHandler); mPrivateHandler = new PrivateHandler(); mProviderCallback = new ProviderCallback(); } /** * Called by the system when it is time to create the media route provider. * * @return The media route provider offered by this service, or null if * this service has decided not to offer a media route provider. */ public abstract MediaRouteProvider onCreateMediaRouteProvider(); /** * Gets the media route provider offered by this service. * * @return The media route provider offered by this service, or null if * it has not yet been created. * * @see #onCreateMediaRouteProvider() */ public MediaRouteProvider getMediaRouteProvider() { return mProvider; } @Override public IBinder onBind(Intent intent) { if (intent.getAction().equals(SERVICE_INTERFACE)) { if (mProvider == null) { MediaRouteProvider provider = onCreateMediaRouteProvider(); if (provider != null) { String providerPackage = provider.getMetadata().getPackageName(); if (!providerPackage.equals(getPackageName())) { throw new IllegalStateException("onCreateMediaRouteProvider() returned " + "a provider whose package name does not match the package " + "name of the service. A media route provider service can " + "only export its own media route providers. " + "Provider package name: " + providerPackage + ". Service package name: " + getPackageName() + "."); } mProvider = provider; mProvider.setCallback(mProviderCallback); } } if (mProvider != null) { return mReceiveMessenger.getBinder(); } } return null; } private boolean onRegisterClient(Messenger messenger, int requestId, int version) { if (version >= CLIENT_VERSION_1) { int index = findClient(messenger); if (index < 0) { ClientRecord client = new ClientRecord(messenger, version); if (client.register()) { mClients.add(client); if (DEBUG) { Log.d(TAG, client + ": Registered, version=" + version); } if (requestId != 0) { MediaRouteProviderDescriptor descriptor = mProvider.getDescriptor(); sendReply(messenger, SERVICE_MSG_REGISTERED, requestId, SERVICE_VERSION_CURRENT, descriptor != null ? descriptor.asBundle() : null, null); } return true; } } } return false; } private boolean onUnregisterClient(Messenger messenger, int requestId) { int index = findClient(messenger); if (index >= 0) { ClientRecord client = mClients.remove(index); if (DEBUG) { Log.d(TAG, client + ": Unregistered"); } client.dispose(); sendGenericSuccess(messenger, requestId); return true; } return false; } private void onBinderDied(Messenger messenger) { int index = findClient(messenger); if (index >= 0) { ClientRecord client = mClients.remove(index); if (DEBUG) { Log.d(TAG, client + ": Binder died"); } client.dispose(); } } private boolean onCreateRouteController(Messenger messenger, int requestId, int controllerId, String routeId) { ClientRecord client = getClient(messenger); if (client != null) { if (client.createRouteController(routeId, controllerId)) { if (DEBUG) { Log.d(TAG, client + ": Route controller created" + ", controllerId=" + controllerId + ", routeId=" + routeId); } sendGenericSuccess(messenger, requestId); return true; } } return false; } private boolean onReleaseRouteController(Messenger messenger, int requestId, int controllerId) { ClientRecord client = getClient(messenger); if (client != null) { if (client.releaseRouteController(controllerId)) { if (DEBUG) { Log.d(TAG, client + ": Route controller released" + ", controllerId=" + controllerId); } sendGenericSuccess(messenger, requestId); return true; } } return false; } private boolean onSelectRoute(Messenger messenger, int requestId, int controllerId) { ClientRecord client = getClient(messenger); if (client != null) { MediaRouteProvider.RouteController controller = client.getRouteController(controllerId); if (controller != null) { controller.onSelect(); if (DEBUG) { Log.d(TAG, client + ": Route selected" + ", controllerId=" + controllerId); } sendGenericSuccess(messenger, requestId); return true; } } return false; } private boolean onUnselectRoute(Messenger messenger, int requestId, int controllerId) { ClientRecord client = getClient(messenger); if (client != null) { MediaRouteProvider.RouteController controller = client.getRouteController(controllerId); if (controller != null) { controller.onUnselect(); if (DEBUG) { Log.d(TAG, client + ": Route unselected" + ", controllerId=" + controllerId); } sendGenericSuccess(messenger, requestId); return true; } } return false; } private boolean onSetRouteVolume(Messenger messenger, int requestId, int controllerId, int volume) { ClientRecord client = getClient(messenger); if (client != null) { MediaRouteProvider.RouteController controller = client.getRouteController(controllerId); if (controller != null) { controller.onSetVolume(volume); if (DEBUG) { Log.d(TAG, client + ": Route volume changed" + ", controllerId=" + controllerId + ", volume=" + volume); } sendGenericSuccess(messenger, requestId); return true; } } return false; } private boolean onUpdateRouteVolume(Messenger messenger, int requestId, int controllerId, int delta) { ClientRecord client = getClient(messenger); if (client != null) { MediaRouteProvider.RouteController controller = client.getRouteController(controllerId); if (controller != null) { controller.onUpdateVolume(delta); if (DEBUG) { Log.d(TAG, client + ": Route volume updated" + ", controllerId=" + controllerId + ", delta=" + delta); } sendGenericSuccess(messenger, requestId); return true; } } return false; } private boolean onRouteControlRequest(final Messenger messenger, final int requestId, final int controllerId, final Intent intent) { final ClientRecord client = getClient(messenger); if (client != null) { MediaRouteProvider.RouteController controller = client.getRouteController(controllerId); if (controller != null) { MediaRouter.ControlRequestCallback callback = null; if (requestId != 0) { callback = new MediaRouter.ControlRequestCallback() { @Override public void onResult(Bundle data) { if (DEBUG) { Log.d(TAG, client + ": Route control request succeeded" + ", controllerId=" + controllerId + ", intent=" + intent + ", data=" + data); } if (findClient(messenger) >= 0) { sendReply(messenger, SERVICE_MSG_CONTROL_REQUEST_SUCCEEDED, requestId, 0, data, null); } } @Override public void onError(String error, Bundle data) { if (DEBUG) { Log.d(TAG, client + ": Route control request failed" + ", controllerId=" + controllerId + ", intent=" + intent + ", error=" + error + ", data=" + data); } if (findClient(messenger) >= 0) { if (error != null) { Bundle bundle = new Bundle(); bundle.putString(SERVICE_DATA_ERROR, error); sendReply(messenger, SERVICE_MSG_CONTROL_REQUEST_FAILED, requestId, 0, data, bundle); } else { sendReply(messenger, SERVICE_MSG_CONTROL_REQUEST_FAILED, requestId, 0, data, null); } } } }; } if (controller.onControlRequest(intent, callback)) { if (DEBUG) { Log.d(TAG, client + ": Route control request delivered" + ", controllerId=" + controllerId + ", intent=" + intent); } return true; } } } return false; } private boolean onSetDiscoveryRequest(Messenger messenger, int requestId, MediaRouteDiscoveryRequest request) { ClientRecord client = getClient(messenger); if (client != null) { boolean actuallyChanged = client.setDiscoveryRequest(request); if (DEBUG) { Log.d(TAG, client + ": Set discovery request, request=" + request + ", actuallyChanged=" + actuallyChanged + ", compositeDiscoveryRequest=" + mCompositeDiscoveryRequest); } sendGenericSuccess(messenger, requestId); return true; } return false; } private void sendDescriptorChanged(MediaRouteProviderDescriptor descriptor) { Bundle descriptorBundle = descriptor != null ? descriptor.asBundle() : null; final int count = mClients.size(); for (int i = 0; i < count; i++) { ClientRecord client = mClients.get(i); sendReply(client.mMessenger, SERVICE_MSG_DESCRIPTOR_CHANGED, 0, 0, descriptorBundle, null); if (DEBUG) { Log.d(TAG, client + ": Sent descriptor change event, descriptor=" + descriptor); } } } private boolean updateCompositeDiscoveryRequest() { MediaRouteDiscoveryRequest composite = null; MediaRouteSelector.Builder selectorBuilder = null; boolean activeScan = false; final int count = mClients.size(); for (int i = 0; i < count; i++) { MediaRouteDiscoveryRequest request = mClients.get(i).mDiscoveryRequest; if (request != null && (!request.getSelector().isEmpty() || request.isActiveScan())) { activeScan |= request.isActiveScan(); if (composite == null) { composite = request; } else { if (selectorBuilder == null) { selectorBuilder = new MediaRouteSelector.Builder(composite.getSelector()); } selectorBuilder.addSelector(request.getSelector()); } } } if (selectorBuilder != null) { composite = new MediaRouteDiscoveryRequest(selectorBuilder.build(), activeScan); } if (mCompositeDiscoveryRequest != composite && (mCompositeDiscoveryRequest == null || !mCompositeDiscoveryRequest.equals(composite))) { mCompositeDiscoveryRequest = composite; mProvider.setDiscoveryRequest(composite); return true; } return false; } private ClientRecord getClient(Messenger messenger) { int index = findClient(messenger); return index >= 0 ? mClients.get(index) : null; } private int findClient(Messenger messenger) { final int count = mClients.size(); for (int i = 0; i < count; i++) { ClientRecord client = mClients.get(i); if (client.hasMessenger(messenger)) { return i; } } return -1; } private static void sendGenericFailure(Messenger messenger, int requestId) { if (requestId != 0) { sendReply(messenger, SERVICE_MSG_GENERIC_FAILURE, requestId, 0, null, null); } } private static void sendGenericSuccess(Messenger messenger, int requestId) { if (requestId != 0) { sendReply(messenger, SERVICE_MSG_GENERIC_SUCCESS, requestId, 0, null, null); } } private static void sendReply(Messenger messenger, int what, int requestId, int arg, Object obj, Bundle data) { Message msg = Message.obtain(); msg.what = what; msg.arg1 = requestId; msg.arg2 = arg; msg.obj = obj; msg.setData(data); try { messenger.send(msg); } catch (DeadObjectException ex) { // The client died. } catch (RemoteException ex) { Log.e(TAG, "Could not send message to " + getClientId(messenger), ex); } } private static String getClientId(Messenger messenger) { return "Client connection " + messenger.getBinder().toString(); } /** * Returns true if the messenger object is valid. *

* The messenger constructor and unparceling code does not check whether the * provided IBinder is a valid IMessenger object. As a result, it's possible * for a peer to send an invalid IBinder that will result in crashes downstream. * This method checks that the messenger is in a valid state. *

*/ static boolean isValidRemoteMessenger(Messenger messenger) { try { return messenger != null && messenger.getBinder() != null; } catch (NullPointerException ex) { // If the messenger was constructed with a binder interface other than // IMessenger then the call to getBinder() will crash with an NPE. return false; } } private final class PrivateHandler extends Handler { @Override public void handleMessage(Message msg) { switch (msg.what) { case PRIVATE_MSG_CLIENT_DIED: onBinderDied((Messenger)msg.obj); break; } } } private final class ProviderCallback extends MediaRouteProvider.Callback { @Override public void onDescriptorChanged(MediaRouteProvider provider, MediaRouteProviderDescriptor descriptor) { sendDescriptorChanged(descriptor); } } private final class ClientRecord implements DeathRecipient { public final Messenger mMessenger; public final int mVersion; public MediaRouteDiscoveryRequest mDiscoveryRequest; private final SparseArray mControllers = new SparseArray(); public ClientRecord(Messenger messenger, int version) { mMessenger = messenger; mVersion = version; } public boolean register() { try { mMessenger.getBinder().linkToDeath(this, 0); return true; } catch (RemoteException ex) { binderDied(); } return false; } public void dispose() { int count = mControllers.size(); for (int i = 0; i < count; i++) { mControllers.valueAt(i).onRelease(); } mControllers.clear(); mMessenger.getBinder().unlinkToDeath(this, 0); setDiscoveryRequest(null); } public boolean hasMessenger(Messenger other) { return mMessenger.getBinder() == other.getBinder(); } public boolean createRouteController(String routeId, int controllerId) { if (mControllers.indexOfKey(controllerId) < 0) { MediaRouteProvider.RouteController controller = mProvider.onCreateRouteController(routeId); mControllers.put(controllerId, controller); return true; } return false; } public boolean releaseRouteController(int controllerId) { MediaRouteProvider.RouteController controller = mControllers.get(controllerId); if (controller != null) { mControllers.remove(controllerId); controller.onRelease(); return true; } return false; } public MediaRouteProvider.RouteController getRouteController(int controllerId) { return mControllers.get(controllerId); } public boolean setDiscoveryRequest(MediaRouteDiscoveryRequest request) { if (mDiscoveryRequest != request && (mDiscoveryRequest == null || !mDiscoveryRequest.equals(request))) { mDiscoveryRequest = request; return updateCompositeDiscoveryRequest(); } return false; } // Runs on a binder thread. @Override public void binderDied() { mPrivateHandler.obtainMessage(PRIVATE_MSG_CLIENT_DIED, mMessenger).sendToTarget(); } @Override public String toString() { return getClientId(mMessenger); } } /** * Handler that receives messages from clients. *

* This inner class is static and only retains a weak reference to the service * to prevent the service from being leaked in case one of the clients is holding an * active reference to the server's messenger. *

* This handler should not be used to handle any messages other than those * that come from the client. *

*/ private static final class ReceiveHandler extends Handler { private final WeakReference mServiceRef; public ReceiveHandler(MediaRouteProviderService service) { mServiceRef = new WeakReference(service); } @Override public void handleMessage(Message msg) { final Messenger messenger = msg.replyTo; if (isValidRemoteMessenger(messenger)) { final int what = msg.what; final int requestId = msg.arg1; final int arg = msg.arg2; final Object obj = msg.obj; final Bundle data = msg.peekData(); if (!processMessage(what, messenger, requestId, arg, obj, data)) { if (DEBUG) { Log.d(TAG, getClientId(messenger) + ": Message failed, what=" + what + ", requestId=" + requestId + ", arg=" + arg + ", obj=" + obj + ", data=" + data); } sendGenericFailure(messenger, requestId); } } else { if (DEBUG) { Log.d(TAG, "Ignoring message without valid reply messenger."); } } } private boolean processMessage(int what, Messenger messenger, int requestId, int arg, Object obj, Bundle data) { MediaRouteProviderService service = mServiceRef.get(); if (service != null) { switch (what) { case CLIENT_MSG_REGISTER: return service.onRegisterClient(messenger, requestId, arg); case CLIENT_MSG_UNREGISTER: return service.onUnregisterClient(messenger, requestId); case CLIENT_MSG_CREATE_ROUTE_CONTROLLER: { String routeId = data.getString(CLIENT_DATA_ROUTE_ID); if (routeId != null) { return service.onCreateRouteController( messenger, requestId, arg, routeId); } break; } case CLIENT_MSG_RELEASE_ROUTE_CONTROLLER: return service.onReleaseRouteController(messenger, requestId, arg); case CLIENT_MSG_SELECT_ROUTE: return service.onSelectRoute(messenger, requestId, arg); case CLIENT_MSG_UNSELECT_ROUTE: return service.onUnselectRoute(messenger, requestId, arg); case CLIENT_MSG_SET_ROUTE_VOLUME: { int volume = data.getInt(CLIENT_DATA_VOLUME, -1); if (volume >= 0) { return service.onSetRouteVolume( messenger, requestId, arg, volume); } break; } case CLIENT_MSG_UPDATE_ROUTE_VOLUME: { int delta = data.getInt(CLIENT_DATA_VOLUME, 0); if (delta != 0) { return service.onUpdateRouteVolume( messenger, requestId, arg, delta); } break; } case CLIENT_MSG_ROUTE_CONTROL_REQUEST: if (obj instanceof Intent) { return service.onRouteControlRequest( messenger, requestId, arg, (Intent)obj); } break; case CLIENT_MSG_SET_DISCOVERY_REQUEST: { if (obj == null || obj instanceof Bundle) { MediaRouteDiscoveryRequest request = MediaRouteDiscoveryRequest.fromBundle((Bundle)obj); return service.onSetDiscoveryRequest( messenger, requestId, request != null && request.isValid() ? request : null); } } } } return false; } } }