1 /* 2 * Copyright (C) 2014 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.cts.verifier.tv; 18 19 import android.annotation.SuppressLint; 20 import android.content.BroadcastReceiver; 21 import android.content.ComponentName; 22 import android.content.Context; 23 import android.content.Intent; 24 import android.content.IntentFilter; 25 import android.graphics.Bitmap; 26 import android.graphics.BitmapFactory; 27 import android.graphics.Canvas; 28 import android.graphics.Color; 29 import android.graphics.Rect; 30 import android.media.PlaybackParams; 31 import android.media.tv.TvContentRating; 32 import android.media.tv.TvContract; 33 import android.media.tv.TvInputManager; 34 import android.media.tv.TvInputService; 35 import android.media.tv.TvTrackInfo; 36 import android.net.Uri; 37 import android.os.Bundle; 38 import android.os.Handler; 39 import android.os.Looper; 40 import android.os.Message; 41 import android.util.Log; 42 import android.view.LayoutInflater; 43 import android.view.Surface; 44 import android.view.View; 45 import android.widget.TextView; 46 47 import com.android.cts.verifier.R; 48 49 import java.util.ArrayList; 50 import java.util.List; 51 52 @SuppressLint("NewApi") 53 public class MockTvInputService extends TvInputService { 54 private static final String TAG = "MockTvInputService"; 55 56 private static final String BROADCAST_ACTION = "action"; 57 private static final String SELECT_TRACK_TYPE = "type"; 58 private static final String SELECT_TRACK_ID = "id"; 59 private static final String CAPTION_ENABLED = "enabled"; 60 private static final String PAUSE_CALLED = "pause_called"; 61 private static final float DISPLAY_RATIO_EPSILON = 0.01f; 62 63 private static final Object sLock = new Object(); 64 private static Callback sTuneCallback = null; 65 private static Callback sOverlayViewCallback = null; 66 private static Callback sBroadcastCallback = null; 67 private static Callback sUnblockContentCallback = null; 68 private static Callback sSelectTrackCallback = null; 69 private static Callback sSetCaptionEnabledCallback = null; 70 // Callbacks for time shift. 71 private static Callback sResumeAfterPauseCallback = null; 72 private static Callback sPositionTrackingCallback = null; 73 private static Callback sRewindCallback = null; 74 private static Callback sFastForwardCallback = null; 75 private static Callback sSeekToPreviousCallback = null; 76 private static Callback sSeekToNextCallback = null; 77 private static Callback sOverlayViewSizeChangedCallback = null; 78 79 private static TvContentRating sRating = null; 80 81 static final TvTrackInfo sEngAudioTrack = 82 new TvTrackInfo.Builder(TvTrackInfo.TYPE_AUDIO, "audio_eng") 83 .setAudioChannelCount(2) 84 .setAudioSampleRate(48000) 85 .setLanguage("eng") 86 .build(); 87 static final TvTrackInfo sSpaAudioTrack = 88 new TvTrackInfo.Builder(TvTrackInfo.TYPE_AUDIO, "audio_spa") 89 .setAudioChannelCount(2) 90 .setAudioSampleRate(48000) 91 .setLanguage("spa") 92 .build(); 93 static final TvTrackInfo sEngSubtitleTrack = 94 new TvTrackInfo.Builder(TvTrackInfo.TYPE_SUBTITLE, "subtitle_eng") 95 .setLanguage("eng") 96 .build(); 97 static final TvTrackInfo sKorSubtitleTrack = 98 new TvTrackInfo.Builder(TvTrackInfo.TYPE_SUBTITLE, "subtitle_kor") 99 .setLanguage("kor") 100 .build(); 101 // These parameters make the display aspect ratio of sDummyVideoTrack be 4:3, 102 // which is one of common standards. 103 static final TvTrackInfo sDummyVideoTrack = 104 new TvTrackInfo.Builder(TvTrackInfo.TYPE_VIDEO, "video_dummy") 105 .setVideoWidth(704) 106 .setVideoHeight(480) 107 .setVideoPixelAspectRatio(0.909f) 108 .setVideoFrameRate(60) 109 .build(); 110 111 private final BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() { 112 @Override 113 public void onReceive(Context context, Intent intent) { 114 synchronized (sLock) { 115 if (sBroadcastCallback != null) { 116 String expectedAction = 117 sBroadcastCallback.getBundle().getString(BROADCAST_ACTION); 118 if (intent.getAction().equals(expectedAction)) { 119 sBroadcastCallback.post(); 120 sBroadcastCallback = null; 121 } 122 } 123 } 124 } 125 }; 126 expectTune(View postTarget, Runnable successCallback)127 static void expectTune(View postTarget, Runnable successCallback) { 128 synchronized (sLock) { 129 sTuneCallback = new Callback(postTarget, successCallback); 130 } 131 } 132 expectBroadcast(View postTarget, String action, Runnable successCallback)133 static void expectBroadcast(View postTarget, String action, Runnable successCallback) { 134 synchronized (sLock) { 135 sBroadcastCallback = new Callback(postTarget, successCallback); 136 sBroadcastCallback.getBundle().putString(BROADCAST_ACTION, action); 137 } 138 } 139 expectUnblockContent(View postTarget, Runnable successCallback)140 static void expectUnblockContent(View postTarget, Runnable successCallback) { 141 synchronized (sLock) { 142 sUnblockContentCallback = new Callback(postTarget, successCallback); 143 } 144 } 145 setBlockRating(TvContentRating rating)146 static void setBlockRating(TvContentRating rating) { 147 synchronized (sLock) { 148 sRating = rating; 149 } 150 } 151 expectOverlayView(View postTarget, Runnable successCallback)152 static void expectOverlayView(View postTarget, Runnable successCallback) { 153 synchronized (sLock) { 154 sOverlayViewCallback = new Callback(postTarget, successCallback); 155 } 156 } 157 expectSelectTrack(int type, String id, View postTarget, Runnable successCallback)158 static void expectSelectTrack(int type, String id, View postTarget, Runnable successCallback) { 159 synchronized (sLock) { 160 sSelectTrackCallback = new Callback(postTarget, successCallback); 161 sSelectTrackCallback.getBundle().putInt(SELECT_TRACK_TYPE, type); 162 sSelectTrackCallback.getBundle().putString(SELECT_TRACK_ID, id); 163 } 164 } 165 expectSetCaptionEnabled(boolean enabled, View postTarget, Runnable successCallback)166 static void expectSetCaptionEnabled(boolean enabled, View postTarget, 167 Runnable successCallback) { 168 synchronized (sLock) { 169 sSetCaptionEnabledCallback = new Callback(postTarget, successCallback); 170 sSetCaptionEnabledCallback.getBundle().putBoolean(CAPTION_ENABLED, enabled); 171 } 172 } 173 expectResumeAfterPause(View postTarget, Runnable successCallback)174 static void expectResumeAfterPause(View postTarget, Runnable successCallback) { 175 synchronized (sLock) { 176 sResumeAfterPauseCallback = new Callback(postTarget, successCallback); 177 } 178 } 179 expectPositionTracking(View postTarget, Runnable successCallback)180 static void expectPositionTracking(View postTarget, Runnable successCallback) { 181 synchronized (sLock) { 182 sPositionTrackingCallback = new Callback(postTarget, successCallback); 183 } 184 } 185 expectRewind(View postTarget, Runnable successCallback)186 static void expectRewind(View postTarget, Runnable successCallback) { 187 synchronized (sLock) { 188 sRewindCallback = new Callback(postTarget, successCallback); 189 } 190 } 191 expectFastForward(View postTarget, Runnable successCallback)192 static void expectFastForward(View postTarget, Runnable successCallback) { 193 synchronized (sLock) { 194 sFastForwardCallback = new Callback(postTarget, successCallback); 195 } 196 } 197 expectSeekToPrevious(View postTarget, Runnable successCallback)198 static void expectSeekToPrevious(View postTarget, Runnable successCallback) { 199 synchronized (sLock) { 200 sSeekToPreviousCallback = new Callback(postTarget, successCallback); 201 } 202 } 203 expectSeekToNext(View postTarget, Runnable successCallback)204 static void expectSeekToNext(View postTarget, Runnable successCallback) { 205 synchronized (sLock) { 206 sSeekToNextCallback = new Callback(postTarget, successCallback); 207 } 208 } 209 expectedVideoAspectRatio(View postTarget, Runnable successCallback)210 static void expectedVideoAspectRatio(View postTarget, Runnable successCallback) { 211 synchronized (sLock) { 212 sOverlayViewSizeChangedCallback = new Callback(postTarget, successCallback); 213 } 214 } 215 getInputId(Context context)216 static String getInputId(Context context) { 217 return TvContract.buildInputId(new ComponentName(context, 218 MockTvInputService.class.getName())); 219 } 220 221 @Override onCreate()222 public void onCreate() { 223 super.onCreate(); 224 IntentFilter intentFilter = new IntentFilter(); 225 intentFilter.addAction(TvInputManager.ACTION_BLOCKED_RATINGS_CHANGED); 226 intentFilter.addAction(TvInputManager.ACTION_PARENTAL_CONTROLS_ENABLED_CHANGED); 227 registerReceiver(mBroadcastReceiver, intentFilter, Context.RECEIVER_EXPORTED); 228 } 229 230 @Override onDestroy()231 public void onDestroy() { 232 unregisterReceiver(mBroadcastReceiver); 233 super.onDestroy(); 234 } 235 236 @Override onCreateSession(String inputId)237 public Session onCreateSession(String inputId) { 238 Session session = new MockSessionImpl(this); 239 session.setOverlayViewEnabled(true); 240 return session; 241 } 242 243 private static class MockSessionImpl extends Session { 244 private static final int MSG_SEEK = 1000; 245 private static final int SEEK_DELAY_MS = 300; 246 247 private final Context mContext; 248 private Surface mSurface = null; 249 private List<TvTrackInfo> mTracks = new ArrayList<>(); 250 251 private long mRecordStartTimeMs; 252 private long mPausedTimeMs; 253 // The time in milliseconds when the current position is lastly updated. 254 private long mLastCurrentPositionUpdateTimeMs; 255 // The current playback position. 256 private long mCurrentPositionMs; 257 // The current playback speed rate. 258 private float mSpeed; 259 260 private final Handler mHandler = new Handler(Looper.getMainLooper()) { 261 @Override 262 public void handleMessage(Message msg) { 263 if (msg.what == MSG_SEEK) { 264 // Actually, this input doesn't play any videos, it just shows the image. 265 // So we should simulate the playback here by changing the current playback 266 // position periodically in order to test the time shift. 267 // If the playback is paused, the current playback position doesn't need to be 268 // changed. 269 if (mPausedTimeMs == 0) { 270 long currentTimeMs = System.currentTimeMillis(); 271 mCurrentPositionMs += (long) ((currentTimeMs 272 - mLastCurrentPositionUpdateTimeMs) * mSpeed); 273 mCurrentPositionMs = Math.max(mRecordStartTimeMs, 274 Math.min(mCurrentPositionMs, currentTimeMs)); 275 mLastCurrentPositionUpdateTimeMs = currentTimeMs; 276 } 277 sendEmptyMessageDelayed(MSG_SEEK, SEEK_DELAY_MS); 278 } 279 super.handleMessage(msg); 280 } 281 }; 282 MockSessionImpl(Context context)283 private MockSessionImpl(Context context) { 284 super(context); 285 mContext = context; 286 mTracks.add(sEngAudioTrack); 287 mTracks.add(sSpaAudioTrack); 288 mTracks.add(sEngSubtitleTrack); 289 mTracks.add(sKorSubtitleTrack); 290 mTracks.add(sDummyVideoTrack); 291 } 292 293 @Override onRelease()294 public void onRelease() { 295 } 296 draw()297 private void draw() { 298 Surface surface = mSurface; 299 if (surface == null) return; 300 if (!surface.isValid()) return; 301 302 Canvas c = surface.lockCanvas(null); 303 if (c == null) return; 304 try { 305 Bitmap b = BitmapFactory.decodeResource( 306 mContext.getResources(), R.drawable.icon); 307 int srcWidth = b.getWidth(); 308 int srcHeight = b.getHeight(); 309 int dstWidth = c.getWidth(); 310 int dstHeight = c.getHeight(); 311 c.drawColor(Color.BLACK); 312 c.drawBitmap(b, new Rect(0, 0, srcWidth, srcHeight), 313 new Rect(10, 10, dstWidth - 10, dstHeight - 10), null); 314 } finally { 315 surface.unlockCanvasAndPost(c); 316 } 317 } 318 319 @Override onCreateOverlayView()320 public View onCreateOverlayView() { 321 LayoutInflater inflater = (LayoutInflater) mContext.getSystemService( 322 LAYOUT_INFLATER_SERVICE); 323 View view = inflater.inflate(R.layout.tv_overlay, null); 324 TextView textView = (TextView) view.findViewById(R.id.overlay_view_text); 325 textView.addOnLayoutChangeListener(new View.OnLayoutChangeListener() { 326 @Override 327 public void onLayoutChange(View v, int left, int top, int right, int bottom, 328 int oldLeft, int oldTop, int oldRight, int oldBottom) { 329 Callback overlayViewCallback = null; 330 synchronized (sLock) { 331 overlayViewCallback = sOverlayViewCallback; 332 sOverlayViewCallback = null; 333 } 334 if (overlayViewCallback != null) { 335 overlayViewCallback.post(); 336 } 337 } 338 }); 339 return view; 340 } 341 342 @Override onSetSurface(Surface surface)343 public boolean onSetSurface(Surface surface) { 344 mSurface = surface; 345 if (surface != null) { 346 draw(); 347 } 348 return true; 349 } 350 351 @Override onSetStreamVolume(float volume)352 public void onSetStreamVolume(float volume) { 353 } 354 355 @Override onTune(Uri channelUri)356 public boolean onTune(Uri channelUri) { 357 synchronized (sLock) { 358 if (sRating != null) { 359 notifyContentBlocked(sRating); 360 } 361 if (sTuneCallback != null) { 362 sTuneCallback.post(); 363 sTuneCallback = null; 364 } 365 if (sRating == null) { 366 notifyContentAllowed(); 367 } 368 } 369 notifyVideoAvailable(); 370 notifyTracksChanged(mTracks); 371 notifyTrackSelected(TvTrackInfo.TYPE_AUDIO, sEngAudioTrack.getId()); 372 notifyTrackSelected(TvTrackInfo.TYPE_SUBTITLE, null); 373 notifyTrackSelected(TvTrackInfo.TYPE_VIDEO, sDummyVideoTrack.getId()); 374 375 notifyTimeShiftStatusChanged(TvInputManager.TIME_SHIFT_STATUS_AVAILABLE); 376 mRecordStartTimeMs = mCurrentPositionMs = mLastCurrentPositionUpdateTimeMs 377 = System.currentTimeMillis(); 378 mPausedTimeMs = 0; 379 mHandler.sendEmptyMessageDelayed(MSG_SEEK, SEEK_DELAY_MS); 380 mSpeed = 1; 381 return true; 382 } 383 384 @Override onSelectTrack(int type, String trackId)385 public boolean onSelectTrack(int type, String trackId) { 386 synchronized (sLock) { 387 if (sSelectTrackCallback != null) { 388 Bundle bundle = sSelectTrackCallback.getBundle(); 389 if (bundle.getInt(SELECT_TRACK_TYPE) == type 390 && bundle.getString(SELECT_TRACK_ID).equals(trackId)) { 391 sSelectTrackCallback.post(); 392 sSelectTrackCallback = null; 393 } 394 } 395 } 396 notifyTrackSelected(type, trackId); 397 return true; 398 } 399 400 @Override onSetCaptionEnabled(boolean enabled)401 public void onSetCaptionEnabled(boolean enabled) { 402 synchronized (sLock) { 403 if (sSetCaptionEnabledCallback != null) { 404 Bundle bundle = sSetCaptionEnabledCallback.getBundle(); 405 if (bundle.getBoolean(CAPTION_ENABLED) == enabled) { 406 sSetCaptionEnabledCallback.post(); 407 sSetCaptionEnabledCallback = null; 408 } 409 } 410 } 411 } 412 413 @Override onUnblockContent(TvContentRating unblockedRating)414 public void onUnblockContent(TvContentRating unblockedRating) { 415 synchronized (sLock) { 416 if (sRating != null && sRating.equals(unblockedRating)) { 417 sUnblockContentCallback.post(); 418 sRating = null; 419 notifyContentAllowed(); 420 } 421 } 422 } 423 424 @Override onTimeShiftGetCurrentPosition()425 public long onTimeShiftGetCurrentPosition() { 426 synchronized (sLock) { 427 if (sPositionTrackingCallback != null) { 428 sPositionTrackingCallback.post(); 429 sPositionTrackingCallback = null; 430 } 431 } 432 Log.d(TAG, "currentPositionMs=" + mCurrentPositionMs); 433 return mCurrentPositionMs; 434 } 435 436 @Override onTimeShiftGetStartPosition()437 public long onTimeShiftGetStartPosition() { 438 return mRecordStartTimeMs; 439 } 440 441 @Override onTimeShiftPause()442 public void onTimeShiftPause() { 443 synchronized (sLock) { 444 if (sResumeAfterPauseCallback != null) { 445 sResumeAfterPauseCallback.mBundle.putBoolean(PAUSE_CALLED, true); 446 } 447 } 448 mCurrentPositionMs = mPausedTimeMs = mLastCurrentPositionUpdateTimeMs 449 = System.currentTimeMillis(); 450 } 451 452 @Override onTimeShiftResume()453 public void onTimeShiftResume() { 454 synchronized (sLock) { 455 if (sResumeAfterPauseCallback != null 456 && sResumeAfterPauseCallback.mBundle.getBoolean(PAUSE_CALLED)) { 457 sResumeAfterPauseCallback.post(); 458 sResumeAfterPauseCallback = null; 459 } 460 } 461 mSpeed = 1; 462 mPausedTimeMs = 0; 463 mLastCurrentPositionUpdateTimeMs = System.currentTimeMillis(); 464 } 465 466 @Override onTimeShiftSeekTo(long timeMs)467 public void onTimeShiftSeekTo(long timeMs) { 468 synchronized (sLock) { 469 if (mCurrentPositionMs > timeMs) { 470 if (sSeekToPreviousCallback != null) { 471 sSeekToPreviousCallback.post(); 472 sSeekToPreviousCallback = null; 473 } 474 } else if (mCurrentPositionMs < timeMs) { 475 if (sSeekToNextCallback != null) { 476 sSeekToNextCallback.post(); 477 sSeekToNextCallback = null; 478 } 479 } 480 } 481 mLastCurrentPositionUpdateTimeMs = System.currentTimeMillis(); 482 mCurrentPositionMs = Math.max(mRecordStartTimeMs, 483 Math.min(timeMs, mLastCurrentPositionUpdateTimeMs)); 484 } 485 486 @Override onTimeShiftSetPlaybackParams(PlaybackParams params)487 public void onTimeShiftSetPlaybackParams(PlaybackParams params) { 488 synchronized(sLock) { 489 if (params != null) { 490 if (params.getSpeed() > 1) { 491 if (sFastForwardCallback != null) { 492 sFastForwardCallback.post(); 493 sFastForwardCallback = null; 494 } 495 } else if (params.getSpeed() < 1) { 496 if (sRewindCallback != null) { 497 sRewindCallback.post(); 498 sRewindCallback = null; 499 } 500 } 501 } 502 } 503 mSpeed = params.getSpeed(); 504 } 505 506 @Override onOverlayViewSizeChanged(int width, int height)507 public void onOverlayViewSizeChanged(int width, int height) { 508 synchronized(sLock) { 509 draw(); 510 if (sOverlayViewSizeChangedCallback != null) { 511 if (sDummyVideoTrack.getVideoHeight() <= 0 512 || sDummyVideoTrack.getVideoWidth() <= 0) { 513 Log.w(TAG, 514 "The width or height of the selected video track is invalid."); 515 } else if (height <= 0 || width <= 0) { 516 Log.w(TAG, "The width or height of the OverlayView is incorrect."); 517 } else if (Math.abs((float)width / height 518 - (float)sDummyVideoTrack.getVideoWidth() 519 * sDummyVideoTrack.getVideoPixelAspectRatio() 520 / sDummyVideoTrack.getVideoHeight()) < DISPLAY_RATIO_EPSILON) { 521 // Verify the video display aspect ratio is correct 522 // and setVideoPixelAspectRatio() works for the view size. 523 sOverlayViewSizeChangedCallback.post(); 524 sOverlayViewSizeChangedCallback = null; 525 } 526 } 527 } 528 } 529 } 530 531 private static class Callback { 532 private final View mPostTarget; 533 private final Runnable mAction; 534 private final Bundle mBundle = new Bundle(); 535 Callback(View postTarget, Runnable action)536 Callback(View postTarget, Runnable action) { 537 mPostTarget = postTarget; 538 mAction = action; 539 } 540 post()541 public void post() { 542 mPostTarget.post(mAction); 543 } 544 getBundle()545 public Bundle getBundle() { 546 return mBundle; 547 } 548 } 549 } 550