| /* |
| * Copyright (C) 2023 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.media.tv.ad; |
| |
| import android.annotation.NonNull; |
| import android.annotation.Nullable; |
| import android.content.Context; |
| import android.graphics.Rect; |
| import android.media.tv.TvTrackInfo; |
| import android.net.Uri; |
| import android.os.Bundle; |
| import android.os.IBinder; |
| import android.os.Looper; |
| import android.os.Message; |
| import android.os.RemoteException; |
| import android.util.Log; |
| import android.view.InputChannel; |
| import android.view.InputEvent; |
| import android.view.InputEventReceiver; |
| import android.view.Surface; |
| |
| import com.android.internal.os.HandlerCaller; |
| import com.android.internal.os.SomeArgs; |
| |
| import java.util.List; |
| |
| /** |
| * Implements the internal ITvAdSession interface. |
| * @hide |
| */ |
| public class ITvAdSessionWrapper |
| extends ITvAdSession.Stub implements HandlerCaller.Callback { |
| |
| private static final String TAG = "ITvAdSessionWrapper"; |
| |
| private static final int EXECUTE_MESSAGE_TIMEOUT_SHORT_MILLIS = 1000; |
| private static final int EXECUTE_MESSAGE_TIMEOUT_LONG_MILLIS = 5 * 1000; |
| private static final int DO_RELEASE = 1; |
| private static final int DO_SET_SURFACE = 2; |
| private static final int DO_DISPATCH_SURFACE_CHANGED = 3; |
| private static final int DO_CREATE_MEDIA_VIEW = 4; |
| private static final int DO_RELAYOUT_MEDIA_VIEW = 5; |
| private static final int DO_REMOVE_MEDIA_VIEW = 6; |
| private static final int DO_START_AD_SERVICE = 7; |
| private static final int DO_STOP_AD_SERVICE = 8; |
| private static final int DO_RESET_AD_SERVICE = 9; |
| private static final int DO_SEND_CURRENT_VIDEO_BOUNDS = 10; |
| private static final int DO_SEND_CURRENT_CHANNEL_URI = 11; |
| private static final int DO_SEND_TRACK_INFO_LIST = 12; |
| private static final int DO_SEND_CURRENT_TV_INPUT_ID = 13; |
| private static final int DO_SEND_SIGNING_RESULT = 14; |
| private static final int DO_NOTIFY_ERROR = 15; |
| private static final int DO_NOTIFY_TV_MESSAGE = 16; |
| private static final int DO_NOTIFY_INPUT_SESSION_DATA = 17; |
| |
| private final HandlerCaller mCaller; |
| private TvAdService.Session mSessionImpl; |
| private InputChannel mChannel; |
| private TvAdEventReceiver mReceiver; |
| |
| public ITvAdSessionWrapper( |
| Context context, TvAdService.Session mSessionImpl, InputChannel channel) { |
| this.mSessionImpl = mSessionImpl; |
| mCaller = new HandlerCaller(context, null, this, true /* asyncHandler */); |
| mChannel = channel; |
| if (channel != null) { |
| mReceiver = new TvAdEventReceiver(channel, context.getMainLooper()); |
| } |
| } |
| |
| @Override |
| public void release() { |
| mSessionImpl.scheduleMediaViewCleanup(); |
| mCaller.executeOrSendMessage(mCaller.obtainMessage(DO_RELEASE)); |
| } |
| |
| |
| @Override |
| public void executeMessage(Message msg) { |
| if (mSessionImpl == null) { |
| return; |
| } |
| |
| long startTime = System.nanoTime(); |
| switch (msg.what) { |
| case DO_RELEASE: { |
| mSessionImpl.release(); |
| mSessionImpl = null; |
| if (mReceiver != null) { |
| mReceiver.dispose(); |
| mReceiver = null; |
| } |
| if (mChannel != null) { |
| mChannel.dispose(); |
| mChannel = null; |
| } |
| break; |
| } |
| case DO_SET_SURFACE: { |
| mSessionImpl.setSurface((Surface) msg.obj); |
| break; |
| } |
| case DO_DISPATCH_SURFACE_CHANGED: { |
| SomeArgs args = (SomeArgs) msg.obj; |
| mSessionImpl.dispatchSurfaceChanged( |
| (Integer) args.argi1, (Integer) args.argi2, (Integer) args.argi3); |
| args.recycle(); |
| break; |
| } |
| case DO_CREATE_MEDIA_VIEW: { |
| SomeArgs args = (SomeArgs) msg.obj; |
| mSessionImpl.createMediaView((IBinder) args.arg1, (Rect) args.arg2); |
| args.recycle(); |
| break; |
| } |
| case DO_RELAYOUT_MEDIA_VIEW: { |
| mSessionImpl.relayoutMediaView((Rect) msg.obj); |
| break; |
| } |
| case DO_REMOVE_MEDIA_VIEW: { |
| mSessionImpl.removeMediaView(true); |
| break; |
| } |
| case DO_START_AD_SERVICE: { |
| mSessionImpl.startAdService(); |
| break; |
| } |
| case DO_STOP_AD_SERVICE: { |
| mSessionImpl.stopAdService(); |
| break; |
| } |
| case DO_RESET_AD_SERVICE: { |
| mSessionImpl.resetAdService(); |
| break; |
| } |
| case DO_SEND_CURRENT_VIDEO_BOUNDS: { |
| mSessionImpl.sendCurrentVideoBounds((Rect) msg.obj); |
| break; |
| } |
| case DO_SEND_CURRENT_CHANNEL_URI: { |
| mSessionImpl.sendCurrentChannelUri((Uri) msg.obj); |
| break; |
| } |
| case DO_SEND_TRACK_INFO_LIST: { |
| mSessionImpl.sendTrackInfoList((List<TvTrackInfo>) msg.obj); |
| break; |
| } |
| case DO_SEND_CURRENT_TV_INPUT_ID: { |
| mSessionImpl.sendCurrentTvInputId((String) msg.obj); |
| break; |
| } |
| case DO_SEND_SIGNING_RESULT: { |
| SomeArgs args = (SomeArgs) msg.obj; |
| mSessionImpl.sendSigningResult((String) args.arg1, (byte[]) args.arg2); |
| args.recycle(); |
| break; |
| } |
| case DO_NOTIFY_ERROR: { |
| SomeArgs args = (SomeArgs) msg.obj; |
| mSessionImpl.notifyError((String) args.arg1, (Bundle) args.arg2); |
| args.recycle(); |
| break; |
| } |
| case DO_NOTIFY_TV_MESSAGE: { |
| SomeArgs args = (SomeArgs) msg.obj; |
| mSessionImpl.notifyTvMessage((Integer) args.arg1, (Bundle) args.arg2); |
| args.recycle(); |
| break; |
| } |
| case DO_NOTIFY_INPUT_SESSION_DATA: { |
| SomeArgs args = (SomeArgs) msg.obj; |
| mSessionImpl.notifyTvInputSessionData((String) args.arg1, (Bundle) args.arg2); |
| args.recycle(); |
| break; |
| } |
| default: { |
| Log.w(TAG, "Unhandled message code: " + msg.what); |
| break; |
| } |
| } |
| long durationMs = (System.nanoTime() - startTime) / (1000 * 1000); |
| if (durationMs > EXECUTE_MESSAGE_TIMEOUT_SHORT_MILLIS) { |
| Log.w(TAG, "Handling message (" + msg.what + ") took too long time (duration=" |
| + durationMs + "ms)"); |
| if (durationMs > EXECUTE_MESSAGE_TIMEOUT_LONG_MILLIS) { |
| // TODO: handle timeout |
| } |
| } |
| |
| } |
| |
| @Override |
| public void startAdService() throws RemoteException { |
| mCaller.executeOrSendMessage(mCaller.obtainMessage(DO_START_AD_SERVICE)); |
| } |
| |
| @Override |
| public void stopAdService() { |
| mCaller.executeOrSendMessage(mCaller.obtainMessage(DO_STOP_AD_SERVICE)); |
| } |
| |
| @Override |
| public void resetAdService() { |
| mCaller.executeOrSendMessage(mCaller.obtainMessage(DO_RESET_AD_SERVICE)); |
| } |
| |
| @Override |
| public void setSurface(Surface surface) { |
| mCaller.executeOrSendMessage(mCaller.obtainMessageO(DO_SET_SURFACE, surface)); |
| } |
| |
| @Override |
| public void dispatchSurfaceChanged(int format, int width, int height) { |
| mCaller.executeOrSendMessage( |
| mCaller.obtainMessageIIII(DO_DISPATCH_SURFACE_CHANGED, format, width, height, 0)); |
| } |
| |
| @Override |
| public void sendCurrentVideoBounds(@Nullable Rect bounds) { |
| mCaller.executeOrSendMessage( |
| mCaller.obtainMessageO(DO_SEND_CURRENT_VIDEO_BOUNDS, bounds)); |
| } |
| |
| @Override |
| public void sendCurrentChannelUri(@Nullable Uri channelUri) { |
| mCaller.executeOrSendMessage( |
| mCaller.obtainMessageO(DO_SEND_CURRENT_CHANNEL_URI, channelUri)); |
| } |
| |
| @Override |
| public void sendTrackInfoList(@NonNull List<TvTrackInfo> tracks) { |
| mCaller.executeOrSendMessage( |
| mCaller.obtainMessageO(DO_SEND_TRACK_INFO_LIST, tracks)); |
| } |
| |
| @Override |
| public void sendCurrentTvInputId(@Nullable String inputId) { |
| mCaller.executeOrSendMessage( |
| mCaller.obtainMessageO(DO_SEND_CURRENT_TV_INPUT_ID, inputId)); |
| } |
| |
| @Override |
| public void sendSigningResult(@NonNull String signingId, @NonNull byte[] result) { |
| mCaller.executeOrSendMessage( |
| mCaller.obtainMessageOO(DO_SEND_SIGNING_RESULT, signingId, result)); |
| } |
| |
| @Override |
| public void notifyError(@NonNull String errMsg, @NonNull Bundle params) { |
| mCaller.executeOrSendMessage( |
| mCaller.obtainMessageOO(DO_NOTIFY_ERROR, errMsg, params)); |
| } |
| |
| @Override |
| public void notifyTvMessage(int type, Bundle data) { |
| mCaller.executeOrSendMessage( |
| mCaller.obtainMessageOO(DO_NOTIFY_TV_MESSAGE, type, data)); |
| } |
| |
| @Override |
| public void createMediaView(IBinder windowToken, Rect frame) { |
| mCaller.executeOrSendMessage( |
| mCaller.obtainMessageOO(DO_CREATE_MEDIA_VIEW, windowToken, frame)); |
| } |
| |
| @Override |
| public void relayoutMediaView(Rect frame) { |
| mCaller.executeOrSendMessage(mCaller.obtainMessageO(DO_RELAYOUT_MEDIA_VIEW, frame)); |
| } |
| |
| @Override |
| public void removeMediaView() { |
| mCaller.executeOrSendMessage(mCaller.obtainMessage(DO_REMOVE_MEDIA_VIEW)); |
| } |
| |
| @Override |
| public void notifyTvInputSessionData(String type, Bundle data) { |
| mCaller.executeOrSendMessage( |
| mCaller.obtainMessageOO(DO_NOTIFY_INPUT_SESSION_DATA, type, data)); |
| } |
| |
| private final class TvAdEventReceiver extends InputEventReceiver { |
| TvAdEventReceiver(InputChannel inputChannel, Looper looper) { |
| super(inputChannel, looper); |
| } |
| |
| @Override |
| public void onInputEvent(InputEvent event) { |
| if (mSessionImpl == null) { |
| // The session has been finished. |
| finishInputEvent(event, false); |
| return; |
| } |
| |
| int handled = mSessionImpl.dispatchInputEvent(event, this); |
| if (handled != TvAdManager.Session.DISPATCH_IN_PROGRESS) { |
| finishInputEvent( |
| event, handled == TvAdManager.Session.DISPATCH_HANDLED); |
| } |
| } |
| } |
| } |