/* * 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 android.media.tv; import android.annotation.Nullable; import android.content.Context; import android.graphics.Rect; import android.media.PlaybackParams; import android.net.Uri; import android.os.Bundle; import android.os.IBinder; import android.os.Looper; import android.os.Message; 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; /** * Implements the internal ITvInputSession interface to convert incoming calls on to it back to * calls on the public TvInputSession interface, scheduling them on the main thread of the process. * * @hide */ public class ITvInputSessionWrapper extends ITvInputSession.Stub implements HandlerCaller.Callback { private static final String TAG = "TvInputSessionWrapper"; private static final int EXECUTE_MESSAGE_TIMEOUT_SHORT_MILLIS = 50; private static final int EXECUTE_MESSAGE_TUNE_TIMEOUT_MILLIS = 2000; private static final int EXECUTE_MESSAGE_TIMEOUT_LONG_MILLIS = 5 * 1000; private static final int DO_RELEASE = 1; private static final int DO_SET_MAIN = 2; private static final int DO_SET_SURFACE = 3; private static final int DO_DISPATCH_SURFACE_CHANGED = 4; private static final int DO_SET_STREAM_VOLUME = 5; private static final int DO_TUNE = 6; private static final int DO_SET_CAPTION_ENABLED = 7; private static final int DO_SELECT_TRACK = 8; private static final int DO_APP_PRIVATE_COMMAND = 9; private static final int DO_CREATE_OVERLAY_VIEW = 10; private static final int DO_RELAYOUT_OVERLAY_VIEW = 11; private static final int DO_REMOVE_OVERLAY_VIEW = 12; private static final int DO_UNBLOCK_CONTENT = 13; private static final int DO_TIME_SHIFT_PLAY = 14; private static final int DO_TIME_SHIFT_PAUSE = 15; private static final int DO_TIME_SHIFT_RESUME = 16; private static final int DO_TIME_SHIFT_SEEK_TO = 17; private static final int DO_TIME_SHIFT_SET_PLAYBACK_PARAMS = 18; private static final int DO_TIME_SHIFT_ENABLE_POSITION_TRACKING = 19; private static final int DO_START_RECORDING = 20; private static final int DO_STOP_RECORDING = 21; private final boolean mIsRecordingSession; private final HandlerCaller mCaller; private TvInputService.Session mTvInputSessionImpl; private TvInputService.RecordingSession mTvInputRecordingSessionImpl; private InputChannel mChannel; private TvInputEventReceiver mReceiver; public ITvInputSessionWrapper(Context context, TvInputService.Session sessionImpl, InputChannel channel) { mIsRecordingSession = false; mCaller = new HandlerCaller(context, null, this, true /* asyncHandler */); mTvInputSessionImpl = sessionImpl; mChannel = channel; if (channel != null) { mReceiver = new TvInputEventReceiver(channel, context.getMainLooper()); } } // For the recording session public ITvInputSessionWrapper(Context context, TvInputService.RecordingSession recordingSessionImpl) { mIsRecordingSession = true; mCaller = new HandlerCaller(context, null, this, true /* asyncHandler */); mTvInputRecordingSessionImpl = recordingSessionImpl; } @Override public void executeMessage(Message msg) { if ((mIsRecordingSession && mTvInputRecordingSessionImpl == null) || (!mIsRecordingSession && mTvInputSessionImpl == null)) { return; } long startTime = System.nanoTime(); switch (msg.what) { case DO_RELEASE: { if (mIsRecordingSession) { mTvInputRecordingSessionImpl.release(); mTvInputRecordingSessionImpl = null; } else { mTvInputSessionImpl.release(); mTvInputSessionImpl = null; if (mReceiver != null) { mReceiver.dispose(); mReceiver = null; } if (mChannel != null) { mChannel.dispose(); mChannel = null; } } break; } case DO_SET_MAIN: { mTvInputSessionImpl.setMain((Boolean) msg.obj); break; } case DO_SET_SURFACE: { mTvInputSessionImpl.setSurface((Surface) msg.obj); break; } case DO_DISPATCH_SURFACE_CHANGED: { SomeArgs args = (SomeArgs) msg.obj; mTvInputSessionImpl.dispatchSurfaceChanged(args.argi1, args.argi2, args.argi3); args.recycle(); break; } case DO_SET_STREAM_VOLUME: { mTvInputSessionImpl.setStreamVolume((Float) msg.obj); break; } case DO_TUNE: { SomeArgs args = (SomeArgs) msg.obj; if (mIsRecordingSession) { mTvInputRecordingSessionImpl.tune((Uri) args.arg1, (Bundle) args.arg2); } else { mTvInputSessionImpl.tune((Uri) args.arg1, (Bundle) args.arg2); } args.recycle(); break; } case DO_SET_CAPTION_ENABLED: { mTvInputSessionImpl.setCaptionEnabled((Boolean) msg.obj); break; } case DO_SELECT_TRACK: { SomeArgs args = (SomeArgs) msg.obj; mTvInputSessionImpl.selectTrack((Integer) args.arg1, (String) args.arg2); args.recycle(); break; } case DO_APP_PRIVATE_COMMAND: { SomeArgs args = (SomeArgs) msg.obj; if (mIsRecordingSession) { mTvInputRecordingSessionImpl.appPrivateCommand( (String) args.arg1, (Bundle) args.arg2); } else { mTvInputSessionImpl.appPrivateCommand((String) args.arg1, (Bundle) args.arg2); } args.recycle(); break; } case DO_CREATE_OVERLAY_VIEW: { SomeArgs args = (SomeArgs) msg.obj; mTvInputSessionImpl.createOverlayView((IBinder) args.arg1, (Rect) args.arg2); args.recycle(); break; } case DO_RELAYOUT_OVERLAY_VIEW: { mTvInputSessionImpl.relayoutOverlayView((Rect) msg.obj); break; } case DO_REMOVE_OVERLAY_VIEW: { mTvInputSessionImpl.removeOverlayView(true); break; } case DO_UNBLOCK_CONTENT: { mTvInputSessionImpl.unblockContent((String) msg.obj); break; } case DO_TIME_SHIFT_PLAY: { mTvInputSessionImpl.timeShiftPlay((Uri) msg.obj); break; } case DO_TIME_SHIFT_PAUSE: { mTvInputSessionImpl.timeShiftPause(); break; } case DO_TIME_SHIFT_RESUME: { mTvInputSessionImpl.timeShiftResume(); break; } case DO_TIME_SHIFT_SEEK_TO: { mTvInputSessionImpl.timeShiftSeekTo((Long) msg.obj); break; } case DO_TIME_SHIFT_SET_PLAYBACK_PARAMS: { mTvInputSessionImpl.timeShiftSetPlaybackParams((PlaybackParams) msg.obj); break; } case DO_TIME_SHIFT_ENABLE_POSITION_TRACKING: { mTvInputSessionImpl.timeShiftEnablePositionTracking((Boolean) msg.obj); break; } case DO_START_RECORDING: { mTvInputRecordingSessionImpl.startRecording((Uri) msg.obj); break; } case DO_STOP_RECORDING: { mTvInputRecordingSessionImpl.stopRecording(); 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 (msg.what == DO_TUNE && durationMs > EXECUTE_MESSAGE_TUNE_TIMEOUT_MILLIS) { throw new RuntimeException("Too much time to handle tune request. (" + durationMs + "ms > " + EXECUTE_MESSAGE_TUNE_TIMEOUT_MILLIS + "ms) " + "Consider handling the tune request in a separate thread."); } if (durationMs > EXECUTE_MESSAGE_TIMEOUT_LONG_MILLIS) { throw new RuntimeException("Too much time to handle a request. (type=" + msg.what + ", " + durationMs + "ms > " + EXECUTE_MESSAGE_TIMEOUT_LONG_MILLIS + "ms)."); } } } @Override public void release() { if (!mIsRecordingSession) { mTvInputSessionImpl.scheduleOverlayViewCleanup(); } mCaller.executeOrSendMessage(mCaller.obtainMessage(DO_RELEASE)); } @Override public void setMain(boolean isMain) { mCaller.executeOrSendMessage(mCaller.obtainMessageO(DO_SET_MAIN, isMain)); } @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 final void setVolume(float volume) { mCaller.executeOrSendMessage(mCaller.obtainMessageO(DO_SET_STREAM_VOLUME, volume)); } @Override public void tune(Uri channelUri, Bundle params) { // Clear the pending tune requests. mCaller.removeMessages(DO_TUNE); mCaller.executeOrSendMessage(mCaller.obtainMessageOO(DO_TUNE, channelUri, params)); } @Override public void setCaptionEnabled(boolean enabled) { mCaller.executeOrSendMessage(mCaller.obtainMessageO(DO_SET_CAPTION_ENABLED, enabled)); } @Override public void selectTrack(int type, String trackId) { mCaller.executeOrSendMessage(mCaller.obtainMessageOO(DO_SELECT_TRACK, type, trackId)); } @Override public void appPrivateCommand(String action, Bundle data) { mCaller.executeOrSendMessage(mCaller.obtainMessageOO(DO_APP_PRIVATE_COMMAND, action, data)); } @Override public void createOverlayView(IBinder windowToken, Rect frame) { mCaller.executeOrSendMessage(mCaller.obtainMessageOO(DO_CREATE_OVERLAY_VIEW, windowToken, frame)); } @Override public void relayoutOverlayView(Rect frame) { mCaller.executeOrSendMessage(mCaller.obtainMessageO(DO_RELAYOUT_OVERLAY_VIEW, frame)); } @Override public void removeOverlayView() { mCaller.executeOrSendMessage(mCaller.obtainMessage(DO_REMOVE_OVERLAY_VIEW)); } @Override public void unblockContent(String unblockedRating) { mCaller.executeOrSendMessage(mCaller.obtainMessageO( DO_UNBLOCK_CONTENT, unblockedRating)); } @Override public void timeShiftPlay(Uri recordedProgramUri) { mCaller.executeOrSendMessage(mCaller.obtainMessageO( DO_TIME_SHIFT_PLAY, recordedProgramUri)); } @Override public void timeShiftPause() { mCaller.executeOrSendMessage(mCaller.obtainMessage(DO_TIME_SHIFT_PAUSE)); } @Override public void timeShiftResume() { mCaller.executeOrSendMessage(mCaller.obtainMessage(DO_TIME_SHIFT_RESUME)); } @Override public void timeShiftSeekTo(long timeMs) { mCaller.executeOrSendMessage(mCaller.obtainMessageO(DO_TIME_SHIFT_SEEK_TO, timeMs)); } @Override public void timeShiftSetPlaybackParams(PlaybackParams params) { mCaller.executeOrSendMessage(mCaller.obtainMessageO(DO_TIME_SHIFT_SET_PLAYBACK_PARAMS, params)); } @Override public void timeShiftEnablePositionTracking(boolean enable) { mCaller.executeOrSendMessage(mCaller.obtainMessageO( DO_TIME_SHIFT_ENABLE_POSITION_TRACKING, enable)); } @Override public void startRecording(@Nullable Uri programUri) { mCaller.executeOrSendMessage(mCaller.obtainMessageO(DO_START_RECORDING, programUri)); } @Override public void stopRecording() { mCaller.executeOrSendMessage(mCaller.obtainMessage(DO_STOP_RECORDING)); } private final class TvInputEventReceiver extends InputEventReceiver { public TvInputEventReceiver(InputChannel inputChannel, Looper looper) { super(inputChannel, looper); } @Override public void onInputEvent(InputEvent event) { if (mTvInputSessionImpl == null) { // The session has been finished. finishInputEvent(event, false); return; } int handled = mTvInputSessionImpl.dispatchInputEvent(event, this); if (handled != TvInputManager.Session.DISPATCH_IN_PROGRESS) { finishInputEvent(event, handled == TvInputManager.Session.DISPATCH_HANDLED); } } } }