1 /*
2 * Copyright (C) 2010 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 //#define LOG_NDEBUG 0
18 #define LOG_TAG "NuPlayer"
19 #include <utils/Log.h>
20
21 #include "NuPlayer.h"
22
23 #include "HTTPLiveSource.h"
24 #include "NuPlayerDecoder.h"
25 #include "NuPlayerDriver.h"
26 #include "NuPlayerRenderer.h"
27 #include "NuPlayerSource.h"
28 #include "RTSPSource.h"
29 #include "StreamingSource.h"
30 #include "GenericSource.h"
31 #include "mp4/MP4Source.h"
32
33 #include "ATSParser.h"
34
35 #include <cutils/properties.h> // for property_get
36 #include <media/stagefright/foundation/hexdump.h>
37 #include <media/stagefright/foundation/ABuffer.h>
38 #include <media/stagefright/foundation/ADebug.h>
39 #include <media/stagefright/foundation/AMessage.h>
40 #include <media/stagefright/ACodec.h>
41 #include <media/stagefright/MediaDefs.h>
42 #include <media/stagefright/MediaErrors.h>
43 #include <media/stagefright/MetaData.h>
44 #include <gui/IGraphicBufferProducer.h>
45
46 #include "avc_utils.h"
47
48 #include "ESDS.h"
49 #include <media/stagefright/Utils.h>
50
51 namespace android {
52
53 struct NuPlayer::Action : public RefBase {
Actionandroid::NuPlayer::Action54 Action() {}
55
56 virtual void execute(NuPlayer *player) = 0;
57
58 private:
59 DISALLOW_EVIL_CONSTRUCTORS(Action);
60 };
61
62 struct NuPlayer::SeekAction : public Action {
SeekActionandroid::NuPlayer::SeekAction63 SeekAction(int64_t seekTimeUs)
64 : mSeekTimeUs(seekTimeUs) {
65 }
66
executeandroid::NuPlayer::SeekAction67 virtual void execute(NuPlayer *player) {
68 player->performSeek(mSeekTimeUs);
69 }
70
71 private:
72 int64_t mSeekTimeUs;
73
74 DISALLOW_EVIL_CONSTRUCTORS(SeekAction);
75 };
76
77 struct NuPlayer::SetSurfaceAction : public Action {
SetSurfaceActionandroid::NuPlayer::SetSurfaceAction78 SetSurfaceAction(const sp<NativeWindowWrapper> &wrapper)
79 : mWrapper(wrapper) {
80 }
81
executeandroid::NuPlayer::SetSurfaceAction82 virtual void execute(NuPlayer *player) {
83 player->performSetSurface(mWrapper);
84 }
85
86 private:
87 sp<NativeWindowWrapper> mWrapper;
88
89 DISALLOW_EVIL_CONSTRUCTORS(SetSurfaceAction);
90 };
91
92 struct NuPlayer::ShutdownDecoderAction : public Action {
ShutdownDecoderActionandroid::NuPlayer::ShutdownDecoderAction93 ShutdownDecoderAction(bool audio, bool video)
94 : mAudio(audio),
95 mVideo(video) {
96 }
97
executeandroid::NuPlayer::ShutdownDecoderAction98 virtual void execute(NuPlayer *player) {
99 player->performDecoderShutdown(mAudio, mVideo);
100 }
101
102 private:
103 bool mAudio;
104 bool mVideo;
105
106 DISALLOW_EVIL_CONSTRUCTORS(ShutdownDecoderAction);
107 };
108
109 struct NuPlayer::PostMessageAction : public Action {
PostMessageActionandroid::NuPlayer::PostMessageAction110 PostMessageAction(const sp<AMessage> &msg)
111 : mMessage(msg) {
112 }
113
executeandroid::NuPlayer::PostMessageAction114 virtual void execute(NuPlayer *) {
115 mMessage->post();
116 }
117
118 private:
119 sp<AMessage> mMessage;
120
121 DISALLOW_EVIL_CONSTRUCTORS(PostMessageAction);
122 };
123
124 // Use this if there's no state necessary to save in order to execute
125 // the action.
126 struct NuPlayer::SimpleAction : public Action {
127 typedef void (NuPlayer::*ActionFunc)();
128
SimpleActionandroid::NuPlayer::SimpleAction129 SimpleAction(ActionFunc func)
130 : mFunc(func) {
131 }
132
executeandroid::NuPlayer::SimpleAction133 virtual void execute(NuPlayer *player) {
134 (player->*mFunc)();
135 }
136
137 private:
138 ActionFunc mFunc;
139
140 DISALLOW_EVIL_CONSTRUCTORS(SimpleAction);
141 };
142
143 ////////////////////////////////////////////////////////////////////////////////
144
NuPlayer()145 NuPlayer::NuPlayer()
146 : mUIDValid(false),
147 mSourceFlags(0),
148 mVideoIsAVC(false),
149 mAudioEOS(false),
150 mVideoEOS(false),
151 mScanSourcesPending(false),
152 mScanSourcesGeneration(0),
153 mPollDurationGeneration(0),
154 mTimeDiscontinuityPending(false),
155 mFlushingAudio(NONE),
156 mFlushingVideo(NONE),
157 mSkipRenderingAudioUntilMediaTimeUs(-1ll),
158 mSkipRenderingVideoUntilMediaTimeUs(-1ll),
159 mVideoLateByUs(0ll),
160 mNumFramesTotal(0ll),
161 mNumFramesDropped(0ll),
162 mVideoScalingMode(NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW),
163 mStarted(false) {
164 }
165
~NuPlayer()166 NuPlayer::~NuPlayer() {
167 }
168
setUID(uid_t uid)169 void NuPlayer::setUID(uid_t uid) {
170 mUIDValid = true;
171 mUID = uid;
172 }
173
setDriver(const wp<NuPlayerDriver> & driver)174 void NuPlayer::setDriver(const wp<NuPlayerDriver> &driver) {
175 mDriver = driver;
176 }
177
setDataSourceAsync(const sp<IStreamSource> & source)178 void NuPlayer::setDataSourceAsync(const sp<IStreamSource> &source) {
179 sp<AMessage> msg = new AMessage(kWhatSetDataSource, id());
180
181 sp<AMessage> notify = new AMessage(kWhatSourceNotify, id());
182
183 char prop[PROPERTY_VALUE_MAX];
184 if (property_get("media.stagefright.use-mp4source", prop, NULL)
185 && (!strcmp(prop, "1") || !strcasecmp(prop, "true"))) {
186 msg->setObject("source", new MP4Source(notify, source));
187 } else {
188 msg->setObject("source", new StreamingSource(notify, source));
189 }
190
191 msg->post();
192 }
193
IsHTTPLiveURL(const char * url)194 static bool IsHTTPLiveURL(const char *url) {
195 if (!strncasecmp("http://", url, 7)
196 || !strncasecmp("https://", url, 8)
197 || !strncasecmp("file://", url, 7)) {
198 size_t len = strlen(url);
199 if (len >= 5 && !strcasecmp(".m3u8", &url[len - 5])) {
200 return true;
201 }
202
203 if (strstr(url,"m3u8")) {
204 return true;
205 }
206 }
207
208 return false;
209 }
210
setDataSourceAsync(const char * url,const KeyedVector<String8,String8> * headers)211 void NuPlayer::setDataSourceAsync(
212 const char *url, const KeyedVector<String8, String8> *headers) {
213 sp<AMessage> msg = new AMessage(kWhatSetDataSource, id());
214 size_t len = strlen(url);
215
216 sp<AMessage> notify = new AMessage(kWhatSourceNotify, id());
217
218 sp<Source> source;
219 if (IsHTTPLiveURL(url)) {
220 source = new HTTPLiveSource(notify, url, headers, mUIDValid, mUID);
221 } else if (!strncasecmp(url, "rtsp://", 7)) {
222 source = new RTSPSource(notify, url, headers, mUIDValid, mUID);
223 } else if ((!strncasecmp(url, "http://", 7)
224 || !strncasecmp(url, "https://", 8))
225 && ((len >= 4 && !strcasecmp(".sdp", &url[len - 4]))
226 || strstr(url, ".sdp?"))) {
227 source = new RTSPSource(notify, url, headers, mUIDValid, mUID, true);
228 } else {
229 source = new GenericSource(notify, url, headers, mUIDValid, mUID);
230 }
231
232 msg->setObject("source", source);
233 msg->post();
234 }
235
setDataSourceAsync(int fd,int64_t offset,int64_t length)236 void NuPlayer::setDataSourceAsync(int fd, int64_t offset, int64_t length) {
237 sp<AMessage> msg = new AMessage(kWhatSetDataSource, id());
238
239 sp<AMessage> notify = new AMessage(kWhatSourceNotify, id());
240
241 sp<Source> source = new GenericSource(notify, fd, offset, length);
242 msg->setObject("source", source);
243 msg->post();
244 }
245
prepareAsync()246 void NuPlayer::prepareAsync() {
247 (new AMessage(kWhatPrepare, id()))->post();
248 }
249
setVideoSurfaceTextureAsync(const sp<IGraphicBufferProducer> & bufferProducer)250 void NuPlayer::setVideoSurfaceTextureAsync(
251 const sp<IGraphicBufferProducer> &bufferProducer) {
252 sp<AMessage> msg = new AMessage(kWhatSetVideoNativeWindow, id());
253
254 if (bufferProducer == NULL) {
255 msg->setObject("native-window", NULL);
256 } else {
257 msg->setObject(
258 "native-window",
259 new NativeWindowWrapper(
260 new Surface(bufferProducer)));
261 }
262
263 msg->post();
264 }
265
setAudioSink(const sp<MediaPlayerBase::AudioSink> & sink)266 void NuPlayer::setAudioSink(const sp<MediaPlayerBase::AudioSink> &sink) {
267 sp<AMessage> msg = new AMessage(kWhatSetAudioSink, id());
268 msg->setObject("sink", sink);
269 msg->post();
270 }
271
start()272 void NuPlayer::start() {
273 (new AMessage(kWhatStart, id()))->post();
274 }
275
pause()276 void NuPlayer::pause() {
277 (new AMessage(kWhatPause, id()))->post();
278 }
279
resume()280 void NuPlayer::resume() {
281 (new AMessage(kWhatResume, id()))->post();
282 }
283
resetAsync()284 void NuPlayer::resetAsync() {
285 (new AMessage(kWhatReset, id()))->post();
286 }
287
seekToAsync(int64_t seekTimeUs)288 void NuPlayer::seekToAsync(int64_t seekTimeUs) {
289 sp<AMessage> msg = new AMessage(kWhatSeek, id());
290 msg->setInt64("seekTimeUs", seekTimeUs);
291 msg->post();
292 }
293
294 // static
IsFlushingState(FlushStatus state,bool * needShutdown)295 bool NuPlayer::IsFlushingState(FlushStatus state, bool *needShutdown) {
296 switch (state) {
297 case FLUSHING_DECODER:
298 if (needShutdown != NULL) {
299 *needShutdown = false;
300 }
301 return true;
302
303 case FLUSHING_DECODER_SHUTDOWN:
304 if (needShutdown != NULL) {
305 *needShutdown = true;
306 }
307 return true;
308
309 default:
310 return false;
311 }
312 }
313
onMessageReceived(const sp<AMessage> & msg)314 void NuPlayer::onMessageReceived(const sp<AMessage> &msg) {
315 switch (msg->what()) {
316 case kWhatSetDataSource:
317 {
318 ALOGV("kWhatSetDataSource");
319
320 CHECK(mSource == NULL);
321
322 sp<RefBase> obj;
323 CHECK(msg->findObject("source", &obj));
324
325 mSource = static_cast<Source *>(obj.get());
326
327 looper()->registerHandler(mSource);
328
329 CHECK(mDriver != NULL);
330 sp<NuPlayerDriver> driver = mDriver.promote();
331 if (driver != NULL) {
332 driver->notifySetDataSourceCompleted(OK);
333 }
334 break;
335 }
336
337 case kWhatPrepare:
338 {
339 mSource->prepareAsync();
340 break;
341 }
342
343 case kWhatGetTrackInfo:
344 {
345 uint32_t replyID;
346 CHECK(msg->senderAwaitsResponse(&replyID));
347
348 status_t err = INVALID_OPERATION;
349 if (mSource != NULL) {
350 Parcel* reply;
351 CHECK(msg->findPointer("reply", (void**)&reply));
352 err = mSource->getTrackInfo(reply);
353 }
354
355 sp<AMessage> response = new AMessage;
356 response->setInt32("err", err);
357
358 response->postReply(replyID);
359 break;
360 }
361
362 case kWhatSelectTrack:
363 {
364 uint32_t replyID;
365 CHECK(msg->senderAwaitsResponse(&replyID));
366
367 status_t err = INVALID_OPERATION;
368 if (mSource != NULL) {
369 size_t trackIndex;
370 int32_t select;
371 CHECK(msg->findSize("trackIndex", &trackIndex));
372 CHECK(msg->findInt32("select", &select));
373 err = mSource->selectTrack(trackIndex, select);
374 }
375
376 sp<AMessage> response = new AMessage;
377 response->setInt32("err", err);
378
379 response->postReply(replyID);
380 break;
381 }
382
383 case kWhatPollDuration:
384 {
385 int32_t generation;
386 CHECK(msg->findInt32("generation", &generation));
387
388 if (generation != mPollDurationGeneration) {
389 // stale
390 break;
391 }
392
393 int64_t durationUs;
394 if (mDriver != NULL && mSource->getDuration(&durationUs) == OK) {
395 sp<NuPlayerDriver> driver = mDriver.promote();
396 if (driver != NULL) {
397 driver->notifyDuration(durationUs);
398 }
399 }
400
401 msg->post(1000000ll); // poll again in a second.
402 break;
403 }
404
405 case kWhatSetVideoNativeWindow:
406 {
407 ALOGV("kWhatSetVideoNativeWindow");
408
409 mDeferredActions.push_back(
410 new ShutdownDecoderAction(
411 false /* audio */, true /* video */));
412
413 sp<RefBase> obj;
414 CHECK(msg->findObject("native-window", &obj));
415
416 mDeferredActions.push_back(
417 new SetSurfaceAction(
418 static_cast<NativeWindowWrapper *>(obj.get())));
419
420 if (obj != NULL) {
421 // If there is a new surface texture, instantiate decoders
422 // again if possible.
423 mDeferredActions.push_back(
424 new SimpleAction(&NuPlayer::performScanSources));
425 }
426
427 processDeferredActions();
428 break;
429 }
430
431 case kWhatSetAudioSink:
432 {
433 ALOGV("kWhatSetAudioSink");
434
435 sp<RefBase> obj;
436 CHECK(msg->findObject("sink", &obj));
437
438 mAudioSink = static_cast<MediaPlayerBase::AudioSink *>(obj.get());
439 break;
440 }
441
442 case kWhatStart:
443 {
444 ALOGV("kWhatStart");
445
446 mVideoIsAVC = false;
447 mAudioEOS = false;
448 mVideoEOS = false;
449 mSkipRenderingAudioUntilMediaTimeUs = -1;
450 mSkipRenderingVideoUntilMediaTimeUs = -1;
451 mVideoLateByUs = 0;
452 mNumFramesTotal = 0;
453 mNumFramesDropped = 0;
454 mStarted = true;
455
456 mSource->start();
457
458 uint32_t flags = 0;
459
460 if (mSource->isRealTime()) {
461 flags |= Renderer::FLAG_REAL_TIME;
462 }
463
464 mRenderer = new Renderer(
465 mAudioSink,
466 new AMessage(kWhatRendererNotify, id()),
467 flags);
468
469 looper()->registerHandler(mRenderer);
470
471 postScanSources();
472 break;
473 }
474
475 case kWhatScanSources:
476 {
477 int32_t generation;
478 CHECK(msg->findInt32("generation", &generation));
479 if (generation != mScanSourcesGeneration) {
480 // Drop obsolete msg.
481 break;
482 }
483
484 mScanSourcesPending = false;
485
486 ALOGV("scanning sources haveAudio=%d, haveVideo=%d",
487 mAudioDecoder != NULL, mVideoDecoder != NULL);
488
489 bool mHadAnySourcesBefore =
490 (mAudioDecoder != NULL) || (mVideoDecoder != NULL);
491
492 if (mNativeWindow != NULL) {
493 instantiateDecoder(false, &mVideoDecoder);
494 }
495
496 if (mAudioSink != NULL) {
497 instantiateDecoder(true, &mAudioDecoder);
498 }
499
500 if (!mHadAnySourcesBefore
501 && (mAudioDecoder != NULL || mVideoDecoder != NULL)) {
502 // This is the first time we've found anything playable.
503
504 if (mSourceFlags & Source::FLAG_DYNAMIC_DURATION) {
505 schedulePollDuration();
506 }
507 }
508
509 status_t err;
510 if ((err = mSource->feedMoreTSData()) != OK) {
511 if (mAudioDecoder == NULL && mVideoDecoder == NULL) {
512 // We're not currently decoding anything (no audio or
513 // video tracks found) and we just ran out of input data.
514
515 if (err == ERROR_END_OF_STREAM) {
516 notifyListener(MEDIA_PLAYBACK_COMPLETE, 0, 0);
517 } else {
518 notifyListener(MEDIA_ERROR, MEDIA_ERROR_UNKNOWN, err);
519 }
520 }
521 break;
522 }
523
524 if ((mAudioDecoder == NULL && mAudioSink != NULL)
525 || (mVideoDecoder == NULL && mNativeWindow != NULL)) {
526 msg->post(100000ll);
527 mScanSourcesPending = true;
528 }
529 break;
530 }
531
532 case kWhatVideoNotify:
533 case kWhatAudioNotify:
534 {
535 bool audio = msg->what() == kWhatAudioNotify;
536
537 sp<AMessage> codecRequest;
538 CHECK(msg->findMessage("codec-request", &codecRequest));
539
540 int32_t what;
541 CHECK(codecRequest->findInt32("what", &what));
542
543 if (what == ACodec::kWhatFillThisBuffer) {
544 status_t err = feedDecoderInputData(
545 audio, codecRequest);
546
547 if (err == -EWOULDBLOCK) {
548 if (mSource->feedMoreTSData() == OK) {
549 msg->post(10000ll);
550 }
551 }
552 } else if (what == ACodec::kWhatEOS) {
553 int32_t err;
554 CHECK(codecRequest->findInt32("err", &err));
555
556 if (err == ERROR_END_OF_STREAM) {
557 ALOGV("got %s decoder EOS", audio ? "audio" : "video");
558 } else {
559 ALOGV("got %s decoder EOS w/ error %d",
560 audio ? "audio" : "video",
561 err);
562 }
563
564 mRenderer->queueEOS(audio, err);
565 } else if (what == ACodec::kWhatFlushCompleted) {
566 bool needShutdown;
567
568 if (audio) {
569 CHECK(IsFlushingState(mFlushingAudio, &needShutdown));
570 mFlushingAudio = FLUSHED;
571 } else {
572 CHECK(IsFlushingState(mFlushingVideo, &needShutdown));
573 mFlushingVideo = FLUSHED;
574
575 mVideoLateByUs = 0;
576 }
577
578 ALOGV("decoder %s flush completed", audio ? "audio" : "video");
579
580 if (needShutdown) {
581 ALOGV("initiating %s decoder shutdown",
582 audio ? "audio" : "video");
583
584 (audio ? mAudioDecoder : mVideoDecoder)->initiateShutdown();
585
586 if (audio) {
587 mFlushingAudio = SHUTTING_DOWN_DECODER;
588 } else {
589 mFlushingVideo = SHUTTING_DOWN_DECODER;
590 }
591 }
592
593 finishFlushIfPossible();
594 } else if (what == ACodec::kWhatOutputFormatChanged) {
595 if (audio) {
596 int32_t numChannels;
597 CHECK(codecRequest->findInt32(
598 "channel-count", &numChannels));
599
600 int32_t sampleRate;
601 CHECK(codecRequest->findInt32("sample-rate", &sampleRate));
602
603 ALOGV("Audio output format changed to %d Hz, %d channels",
604 sampleRate, numChannels);
605
606 mAudioSink->close();
607
608 audio_output_flags_t flags;
609 int64_t durationUs;
610 // FIXME: we should handle the case where the video decoder
611 // is created after we receive the format change indication.
612 // Current code will just make that we select deep buffer
613 // with video which should not be a problem as it should
614 // not prevent from keeping A/V sync.
615 if (mVideoDecoder == NULL &&
616 mSource->getDuration(&durationUs) == OK &&
617 durationUs
618 > AUDIO_SINK_MIN_DEEP_BUFFER_DURATION_US) {
619 flags = AUDIO_OUTPUT_FLAG_DEEP_BUFFER;
620 } else {
621 flags = AUDIO_OUTPUT_FLAG_NONE;
622 }
623
624 int32_t channelMask;
625 if (!codecRequest->findInt32("channel-mask", &channelMask)) {
626 channelMask = CHANNEL_MASK_USE_CHANNEL_ORDER;
627 }
628
629 CHECK_EQ(mAudioSink->open(
630 sampleRate,
631 numChannels,
632 (audio_channel_mask_t)channelMask,
633 AUDIO_FORMAT_PCM_16_BIT,
634 8 /* bufferCount */,
635 NULL,
636 NULL,
637 flags),
638 (status_t)OK);
639 mAudioSink->start();
640
641 mRenderer->signalAudioSinkChanged();
642 } else {
643 // video
644
645 int32_t width, height;
646 CHECK(codecRequest->findInt32("width", &width));
647 CHECK(codecRequest->findInt32("height", &height));
648
649 int32_t cropLeft, cropTop, cropRight, cropBottom;
650 CHECK(codecRequest->findRect(
651 "crop",
652 &cropLeft, &cropTop, &cropRight, &cropBottom));
653
654 int32_t displayWidth = cropRight - cropLeft + 1;
655 int32_t displayHeight = cropBottom - cropTop + 1;
656
657 ALOGV("Video output format changed to %d x %d "
658 "(crop: %d x %d @ (%d, %d))",
659 width, height,
660 displayWidth,
661 displayHeight,
662 cropLeft, cropTop);
663
664 sp<AMessage> videoInputFormat =
665 mSource->getFormat(false /* audio */);
666
667 // Take into account sample aspect ratio if necessary:
668 int32_t sarWidth, sarHeight;
669 if (videoInputFormat->findInt32("sar-width", &sarWidth)
670 && videoInputFormat->findInt32(
671 "sar-height", &sarHeight)) {
672 ALOGV("Sample aspect ratio %d : %d",
673 sarWidth, sarHeight);
674
675 displayWidth = (displayWidth * sarWidth) / sarHeight;
676
677 ALOGV("display dimensions %d x %d",
678 displayWidth, displayHeight);
679 }
680
681 notifyListener(
682 MEDIA_SET_VIDEO_SIZE, displayWidth, displayHeight);
683 }
684 } else if (what == ACodec::kWhatShutdownCompleted) {
685 ALOGV("%s shutdown completed", audio ? "audio" : "video");
686 if (audio) {
687 mAudioDecoder.clear();
688
689 CHECK_EQ((int)mFlushingAudio, (int)SHUTTING_DOWN_DECODER);
690 mFlushingAudio = SHUT_DOWN;
691 } else {
692 mVideoDecoder.clear();
693
694 CHECK_EQ((int)mFlushingVideo, (int)SHUTTING_DOWN_DECODER);
695 mFlushingVideo = SHUT_DOWN;
696 }
697
698 finishFlushIfPossible();
699 } else if (what == ACodec::kWhatError) {
700 ALOGE("Received error from %s decoder, aborting playback.",
701 audio ? "audio" : "video");
702
703 mRenderer->queueEOS(audio, UNKNOWN_ERROR);
704 } else if (what == ACodec::kWhatDrainThisBuffer) {
705 renderBuffer(audio, codecRequest);
706 } else if (what != ACodec::kWhatComponentAllocated
707 && what != ACodec::kWhatComponentConfigured
708 && what != ACodec::kWhatBuffersAllocated) {
709 ALOGV("Unhandled codec notification %d '%c%c%c%c'.",
710 what,
711 what >> 24,
712 (what >> 16) & 0xff,
713 (what >> 8) & 0xff,
714 what & 0xff);
715 }
716
717 break;
718 }
719
720 case kWhatRendererNotify:
721 {
722 int32_t what;
723 CHECK(msg->findInt32("what", &what));
724
725 if (what == Renderer::kWhatEOS) {
726 int32_t audio;
727 CHECK(msg->findInt32("audio", &audio));
728
729 int32_t finalResult;
730 CHECK(msg->findInt32("finalResult", &finalResult));
731
732 if (audio) {
733 mAudioEOS = true;
734 } else {
735 mVideoEOS = true;
736 }
737
738 if (finalResult == ERROR_END_OF_STREAM) {
739 ALOGV("reached %s EOS", audio ? "audio" : "video");
740 } else {
741 ALOGE("%s track encountered an error (%d)",
742 audio ? "audio" : "video", finalResult);
743
744 notifyListener(
745 MEDIA_ERROR, MEDIA_ERROR_UNKNOWN, finalResult);
746 }
747
748 if ((mAudioEOS || mAudioDecoder == NULL)
749 && (mVideoEOS || mVideoDecoder == NULL)) {
750 notifyListener(MEDIA_PLAYBACK_COMPLETE, 0, 0);
751 }
752 } else if (what == Renderer::kWhatPosition) {
753 int64_t positionUs;
754 CHECK(msg->findInt64("positionUs", &positionUs));
755
756 CHECK(msg->findInt64("videoLateByUs", &mVideoLateByUs));
757
758 if (mDriver != NULL) {
759 sp<NuPlayerDriver> driver = mDriver.promote();
760 if (driver != NULL) {
761 driver->notifyPosition(positionUs);
762
763 driver->notifyFrameStats(
764 mNumFramesTotal, mNumFramesDropped);
765 }
766 }
767 } else if (what == Renderer::kWhatFlushComplete) {
768 int32_t audio;
769 CHECK(msg->findInt32("audio", &audio));
770
771 ALOGV("renderer %s flush completed.", audio ? "audio" : "video");
772 } else if (what == Renderer::kWhatVideoRenderingStart) {
773 notifyListener(MEDIA_INFO, MEDIA_INFO_RENDERING_START, 0);
774 } else if (what == Renderer::kWhatMediaRenderingStart) {
775 ALOGV("media rendering started");
776 notifyListener(MEDIA_STARTED, 0, 0);
777 }
778 break;
779 }
780
781 case kWhatMoreDataQueued:
782 {
783 break;
784 }
785
786 case kWhatReset:
787 {
788 ALOGV("kWhatReset");
789
790 mDeferredActions.push_back(
791 new ShutdownDecoderAction(
792 true /* audio */, true /* video */));
793
794 mDeferredActions.push_back(
795 new SimpleAction(&NuPlayer::performReset));
796
797 processDeferredActions();
798 break;
799 }
800
801 case kWhatSeek:
802 {
803 int64_t seekTimeUs;
804 CHECK(msg->findInt64("seekTimeUs", &seekTimeUs));
805
806 ALOGV("kWhatSeek seekTimeUs=%lld us", seekTimeUs);
807
808 mDeferredActions.push_back(
809 new SimpleAction(&NuPlayer::performDecoderFlush));
810
811 mDeferredActions.push_back(new SeekAction(seekTimeUs));
812
813 processDeferredActions();
814 break;
815 }
816
817 case kWhatPause:
818 {
819 CHECK(mRenderer != NULL);
820 mSource->pause();
821 mRenderer->pause();
822 break;
823 }
824
825 case kWhatResume:
826 {
827 CHECK(mRenderer != NULL);
828 mSource->resume();
829 mRenderer->resume();
830 break;
831 }
832
833 case kWhatSourceNotify:
834 {
835 onSourceNotify(msg);
836 break;
837 }
838
839 default:
840 TRESPASS();
841 break;
842 }
843 }
844
finishFlushIfPossible()845 void NuPlayer::finishFlushIfPossible() {
846 if (mFlushingAudio != FLUSHED && mFlushingAudio != SHUT_DOWN) {
847 return;
848 }
849
850 if (mFlushingVideo != FLUSHED && mFlushingVideo != SHUT_DOWN) {
851 return;
852 }
853
854 ALOGV("both audio and video are flushed now.");
855
856 if (mTimeDiscontinuityPending) {
857 mRenderer->signalTimeDiscontinuity();
858 mTimeDiscontinuityPending = false;
859 }
860
861 if (mAudioDecoder != NULL) {
862 mAudioDecoder->signalResume();
863 }
864
865 if (mVideoDecoder != NULL) {
866 mVideoDecoder->signalResume();
867 }
868
869 mFlushingAudio = NONE;
870 mFlushingVideo = NONE;
871
872 processDeferredActions();
873 }
874
postScanSources()875 void NuPlayer::postScanSources() {
876 if (mScanSourcesPending) {
877 return;
878 }
879
880 sp<AMessage> msg = new AMessage(kWhatScanSources, id());
881 msg->setInt32("generation", mScanSourcesGeneration);
882 msg->post();
883
884 mScanSourcesPending = true;
885 }
886
instantiateDecoder(bool audio,sp<Decoder> * decoder)887 status_t NuPlayer::instantiateDecoder(bool audio, sp<Decoder> *decoder) {
888 if (*decoder != NULL) {
889 return OK;
890 }
891
892 sp<AMessage> format = mSource->getFormat(audio);
893
894 if (format == NULL) {
895 return -EWOULDBLOCK;
896 }
897
898 if (!audio) {
899 AString mime;
900 CHECK(format->findString("mime", &mime));
901 mVideoIsAVC = !strcasecmp(MEDIA_MIMETYPE_VIDEO_AVC, mime.c_str());
902 }
903
904 sp<AMessage> notify =
905 new AMessage(audio ? kWhatAudioNotify : kWhatVideoNotify,
906 id());
907
908 *decoder = audio ? new Decoder(notify) :
909 new Decoder(notify, mNativeWindow);
910 looper()->registerHandler(*decoder);
911
912 (*decoder)->configure(format);
913
914 return OK;
915 }
916
feedDecoderInputData(bool audio,const sp<AMessage> & msg)917 status_t NuPlayer::feedDecoderInputData(bool audio, const sp<AMessage> &msg) {
918 sp<AMessage> reply;
919 CHECK(msg->findMessage("reply", &reply));
920
921 if ((audio && IsFlushingState(mFlushingAudio))
922 || (!audio && IsFlushingState(mFlushingVideo))) {
923 reply->setInt32("err", INFO_DISCONTINUITY);
924 reply->post();
925 return OK;
926 }
927
928 sp<ABuffer> accessUnit;
929
930 bool dropAccessUnit;
931 do {
932 status_t err = mSource->dequeueAccessUnit(audio, &accessUnit);
933
934 if (err == -EWOULDBLOCK) {
935 return err;
936 } else if (err != OK) {
937 if (err == INFO_DISCONTINUITY) {
938 int32_t type;
939 CHECK(accessUnit->meta()->findInt32("discontinuity", &type));
940
941 bool formatChange =
942 (audio &&
943 (type & ATSParser::DISCONTINUITY_AUDIO_FORMAT))
944 || (!audio &&
945 (type & ATSParser::DISCONTINUITY_VIDEO_FORMAT));
946
947 bool timeChange = (type & ATSParser::DISCONTINUITY_TIME) != 0;
948
949 ALOGI("%s discontinuity (formatChange=%d, time=%d)",
950 audio ? "audio" : "video", formatChange, timeChange);
951
952 if (audio) {
953 mSkipRenderingAudioUntilMediaTimeUs = -1;
954 } else {
955 mSkipRenderingVideoUntilMediaTimeUs = -1;
956 }
957
958 if (timeChange) {
959 sp<AMessage> extra;
960 if (accessUnit->meta()->findMessage("extra", &extra)
961 && extra != NULL) {
962 int64_t resumeAtMediaTimeUs;
963 if (extra->findInt64(
964 "resume-at-mediatimeUs", &resumeAtMediaTimeUs)) {
965 ALOGI("suppressing rendering of %s until %lld us",
966 audio ? "audio" : "video", resumeAtMediaTimeUs);
967
968 if (audio) {
969 mSkipRenderingAudioUntilMediaTimeUs =
970 resumeAtMediaTimeUs;
971 } else {
972 mSkipRenderingVideoUntilMediaTimeUs =
973 resumeAtMediaTimeUs;
974 }
975 }
976 }
977 }
978
979 mTimeDiscontinuityPending =
980 mTimeDiscontinuityPending || timeChange;
981
982 if (formatChange || timeChange) {
983 if (mFlushingAudio == NONE && mFlushingVideo == NONE) {
984 // And we'll resume scanning sources once we're done
985 // flushing.
986 mDeferredActions.push_front(
987 new SimpleAction(
988 &NuPlayer::performScanSources));
989 }
990
991 sp<AMessage> newFormat = mSource->getFormat(audio);
992 sp<Decoder> &decoder = audio ? mAudioDecoder : mVideoDecoder;
993 if (formatChange && !decoder->supportsSeamlessFormatChange(newFormat)) {
994 flushDecoder(audio, /* needShutdown = */ true);
995 } else {
996 flushDecoder(audio, /* needShutdown = */ false);
997 err = OK;
998 }
999 } else {
1000 // This stream is unaffected by the discontinuity
1001
1002 if (audio) {
1003 mFlushingAudio = FLUSHED;
1004 } else {
1005 mFlushingVideo = FLUSHED;
1006 }
1007
1008 finishFlushIfPossible();
1009
1010 return -EWOULDBLOCK;
1011 }
1012 }
1013
1014 reply->setInt32("err", err);
1015 reply->post();
1016 return OK;
1017 }
1018
1019 if (!audio) {
1020 ++mNumFramesTotal;
1021 }
1022
1023 dropAccessUnit = false;
1024 if (!audio
1025 && mVideoLateByUs > 100000ll
1026 && mVideoIsAVC
1027 && !IsAVCReferenceFrame(accessUnit)) {
1028 dropAccessUnit = true;
1029 ++mNumFramesDropped;
1030 }
1031 } while (dropAccessUnit);
1032
1033 // ALOGV("returned a valid buffer of %s data", audio ? "audio" : "video");
1034
1035 #if 0
1036 int64_t mediaTimeUs;
1037 CHECK(accessUnit->meta()->findInt64("timeUs", &mediaTimeUs));
1038 ALOGV("feeding %s input buffer at media time %.2f secs",
1039 audio ? "audio" : "video",
1040 mediaTimeUs / 1E6);
1041 #endif
1042
1043 reply->setBuffer("buffer", accessUnit);
1044 reply->post();
1045
1046 return OK;
1047 }
1048
renderBuffer(bool audio,const sp<AMessage> & msg)1049 void NuPlayer::renderBuffer(bool audio, const sp<AMessage> &msg) {
1050 // ALOGV("renderBuffer %s", audio ? "audio" : "video");
1051
1052 sp<AMessage> reply;
1053 CHECK(msg->findMessage("reply", &reply));
1054
1055 if (IsFlushingState(audio ? mFlushingAudio : mFlushingVideo)) {
1056 // We're currently attempting to flush the decoder, in order
1057 // to complete this, the decoder wants all its buffers back,
1058 // so we don't want any output buffers it sent us (from before
1059 // we initiated the flush) to be stuck in the renderer's queue.
1060
1061 ALOGV("we're still flushing the %s decoder, sending its output buffer"
1062 " right back.", audio ? "audio" : "video");
1063
1064 reply->post();
1065 return;
1066 }
1067
1068 sp<ABuffer> buffer;
1069 CHECK(msg->findBuffer("buffer", &buffer));
1070
1071 int64_t &skipUntilMediaTimeUs =
1072 audio
1073 ? mSkipRenderingAudioUntilMediaTimeUs
1074 : mSkipRenderingVideoUntilMediaTimeUs;
1075
1076 if (skipUntilMediaTimeUs >= 0) {
1077 int64_t mediaTimeUs;
1078 CHECK(buffer->meta()->findInt64("timeUs", &mediaTimeUs));
1079
1080 if (mediaTimeUs < skipUntilMediaTimeUs) {
1081 ALOGV("dropping %s buffer at time %lld as requested.",
1082 audio ? "audio" : "video",
1083 mediaTimeUs);
1084
1085 reply->post();
1086 return;
1087 }
1088
1089 skipUntilMediaTimeUs = -1;
1090 }
1091
1092 mRenderer->queueBuffer(audio, buffer, reply);
1093 }
1094
notifyListener(int msg,int ext1,int ext2,const Parcel * in)1095 void NuPlayer::notifyListener(int msg, int ext1, int ext2, const Parcel *in) {
1096 if (mDriver == NULL) {
1097 return;
1098 }
1099
1100 sp<NuPlayerDriver> driver = mDriver.promote();
1101
1102 if (driver == NULL) {
1103 return;
1104 }
1105
1106 driver->notifyListener(msg, ext1, ext2, in);
1107 }
1108
flushDecoder(bool audio,bool needShutdown)1109 void NuPlayer::flushDecoder(bool audio, bool needShutdown) {
1110 ALOGV("[%s] flushDecoder needShutdown=%d",
1111 audio ? "audio" : "video", needShutdown);
1112
1113 if ((audio && mAudioDecoder == NULL) || (!audio && mVideoDecoder == NULL)) {
1114 ALOGI("flushDecoder %s without decoder present",
1115 audio ? "audio" : "video");
1116 }
1117
1118 // Make sure we don't continue to scan sources until we finish flushing.
1119 ++mScanSourcesGeneration;
1120 mScanSourcesPending = false;
1121
1122 (audio ? mAudioDecoder : mVideoDecoder)->signalFlush();
1123 mRenderer->flush(audio);
1124
1125 FlushStatus newStatus =
1126 needShutdown ? FLUSHING_DECODER_SHUTDOWN : FLUSHING_DECODER;
1127
1128 if (audio) {
1129 CHECK(mFlushingAudio == NONE
1130 || mFlushingAudio == AWAITING_DISCONTINUITY);
1131
1132 mFlushingAudio = newStatus;
1133
1134 if (mFlushingVideo == NONE) {
1135 mFlushingVideo = (mVideoDecoder != NULL)
1136 ? AWAITING_DISCONTINUITY
1137 : FLUSHED;
1138 }
1139 } else {
1140 CHECK(mFlushingVideo == NONE
1141 || mFlushingVideo == AWAITING_DISCONTINUITY);
1142
1143 mFlushingVideo = newStatus;
1144
1145 if (mFlushingAudio == NONE) {
1146 mFlushingAudio = (mAudioDecoder != NULL)
1147 ? AWAITING_DISCONTINUITY
1148 : FLUSHED;
1149 }
1150 }
1151 }
1152
getFormat(bool audio)1153 sp<AMessage> NuPlayer::Source::getFormat(bool audio) {
1154 sp<MetaData> meta = getFormatMeta(audio);
1155
1156 if (meta == NULL) {
1157 return NULL;
1158 }
1159
1160 sp<AMessage> msg = new AMessage;
1161
1162 if(convertMetaDataToMessage(meta, &msg) == OK) {
1163 return msg;
1164 }
1165 return NULL;
1166 }
1167
setVideoScalingMode(int32_t mode)1168 status_t NuPlayer::setVideoScalingMode(int32_t mode) {
1169 mVideoScalingMode = mode;
1170 if (mNativeWindow != NULL) {
1171 status_t ret = native_window_set_scaling_mode(
1172 mNativeWindow->getNativeWindow().get(), mVideoScalingMode);
1173 if (ret != OK) {
1174 ALOGE("Failed to set scaling mode (%d): %s",
1175 -ret, strerror(-ret));
1176 return ret;
1177 }
1178 }
1179 return OK;
1180 }
1181
getTrackInfo(Parcel * reply) const1182 status_t NuPlayer::getTrackInfo(Parcel* reply) const {
1183 sp<AMessage> msg = new AMessage(kWhatGetTrackInfo, id());
1184 msg->setPointer("reply", reply);
1185
1186 sp<AMessage> response;
1187 status_t err = msg->postAndAwaitResponse(&response);
1188 return err;
1189 }
1190
selectTrack(size_t trackIndex,bool select)1191 status_t NuPlayer::selectTrack(size_t trackIndex, bool select) {
1192 sp<AMessage> msg = new AMessage(kWhatSelectTrack, id());
1193 msg->setSize("trackIndex", trackIndex);
1194 msg->setInt32("select", select);
1195
1196 sp<AMessage> response;
1197 status_t err = msg->postAndAwaitResponse(&response);
1198
1199 return err;
1200 }
1201
schedulePollDuration()1202 void NuPlayer::schedulePollDuration() {
1203 sp<AMessage> msg = new AMessage(kWhatPollDuration, id());
1204 msg->setInt32("generation", mPollDurationGeneration);
1205 msg->post();
1206 }
1207
cancelPollDuration()1208 void NuPlayer::cancelPollDuration() {
1209 ++mPollDurationGeneration;
1210 }
1211
processDeferredActions()1212 void NuPlayer::processDeferredActions() {
1213 while (!mDeferredActions.empty()) {
1214 // We won't execute any deferred actions until we're no longer in
1215 // an intermediate state, i.e. one more more decoders are currently
1216 // flushing or shutting down.
1217
1218 if (mRenderer != NULL) {
1219 // There's an edge case where the renderer owns all output
1220 // buffers and is paused, therefore the decoder will not read
1221 // more input data and will never encounter the matching
1222 // discontinuity. To avoid this, we resume the renderer.
1223
1224 if (mFlushingAudio == AWAITING_DISCONTINUITY
1225 || mFlushingVideo == AWAITING_DISCONTINUITY) {
1226 mRenderer->resume();
1227 }
1228 }
1229
1230 if (mFlushingAudio != NONE || mFlushingVideo != NONE) {
1231 // We're currently flushing, postpone the reset until that's
1232 // completed.
1233
1234 ALOGV("postponing action mFlushingAudio=%d, mFlushingVideo=%d",
1235 mFlushingAudio, mFlushingVideo);
1236
1237 break;
1238 }
1239
1240 sp<Action> action = *mDeferredActions.begin();
1241 mDeferredActions.erase(mDeferredActions.begin());
1242
1243 action->execute(this);
1244 }
1245 }
1246
performSeek(int64_t seekTimeUs)1247 void NuPlayer::performSeek(int64_t seekTimeUs) {
1248 ALOGV("performSeek seekTimeUs=%lld us (%.2f secs)",
1249 seekTimeUs,
1250 seekTimeUs / 1E6);
1251
1252 mSource->seekTo(seekTimeUs);
1253
1254 if (mDriver != NULL) {
1255 sp<NuPlayerDriver> driver = mDriver.promote();
1256 if (driver != NULL) {
1257 driver->notifyPosition(seekTimeUs);
1258 driver->notifySeekComplete();
1259 }
1260 }
1261
1262 // everything's flushed, continue playback.
1263 }
1264
performDecoderFlush()1265 void NuPlayer::performDecoderFlush() {
1266 ALOGV("performDecoderFlush");
1267
1268 if (mAudioDecoder == NULL && mVideoDecoder == NULL) {
1269 return;
1270 }
1271
1272 mTimeDiscontinuityPending = true;
1273
1274 if (mAudioDecoder != NULL) {
1275 flushDecoder(true /* audio */, false /* needShutdown */);
1276 }
1277
1278 if (mVideoDecoder != NULL) {
1279 flushDecoder(false /* audio */, false /* needShutdown */);
1280 }
1281 }
1282
performDecoderShutdown(bool audio,bool video)1283 void NuPlayer::performDecoderShutdown(bool audio, bool video) {
1284 ALOGV("performDecoderShutdown audio=%d, video=%d", audio, video);
1285
1286 if ((!audio || mAudioDecoder == NULL)
1287 && (!video || mVideoDecoder == NULL)) {
1288 return;
1289 }
1290
1291 mTimeDiscontinuityPending = true;
1292
1293 if (mFlushingAudio == NONE && (!audio || mAudioDecoder == NULL)) {
1294 mFlushingAudio = FLUSHED;
1295 }
1296
1297 if (mFlushingVideo == NONE && (!video || mVideoDecoder == NULL)) {
1298 mFlushingVideo = FLUSHED;
1299 }
1300
1301 if (audio && mAudioDecoder != NULL) {
1302 flushDecoder(true /* audio */, true /* needShutdown */);
1303 }
1304
1305 if (video && mVideoDecoder != NULL) {
1306 flushDecoder(false /* audio */, true /* needShutdown */);
1307 }
1308 }
1309
performReset()1310 void NuPlayer::performReset() {
1311 ALOGV("performReset");
1312
1313 CHECK(mAudioDecoder == NULL);
1314 CHECK(mVideoDecoder == NULL);
1315
1316 cancelPollDuration();
1317
1318 ++mScanSourcesGeneration;
1319 mScanSourcesPending = false;
1320
1321 mRenderer.clear();
1322
1323 if (mSource != NULL) {
1324 mSource->stop();
1325
1326 looper()->unregisterHandler(mSource->id());
1327
1328 mSource.clear();
1329 }
1330
1331 if (mDriver != NULL) {
1332 sp<NuPlayerDriver> driver = mDriver.promote();
1333 if (driver != NULL) {
1334 driver->notifyResetComplete();
1335 }
1336 }
1337
1338 mStarted = false;
1339 }
1340
performScanSources()1341 void NuPlayer::performScanSources() {
1342 ALOGV("performScanSources");
1343
1344 if (!mStarted) {
1345 return;
1346 }
1347
1348 if (mAudioDecoder == NULL || mVideoDecoder == NULL) {
1349 postScanSources();
1350 }
1351 }
1352
performSetSurface(const sp<NativeWindowWrapper> & wrapper)1353 void NuPlayer::performSetSurface(const sp<NativeWindowWrapper> &wrapper) {
1354 ALOGV("performSetSurface");
1355
1356 mNativeWindow = wrapper;
1357
1358 // XXX - ignore error from setVideoScalingMode for now
1359 setVideoScalingMode(mVideoScalingMode);
1360
1361 if (mDriver != NULL) {
1362 sp<NuPlayerDriver> driver = mDriver.promote();
1363 if (driver != NULL) {
1364 driver->notifySetSurfaceComplete();
1365 }
1366 }
1367 }
1368
onSourceNotify(const sp<AMessage> & msg)1369 void NuPlayer::onSourceNotify(const sp<AMessage> &msg) {
1370 int32_t what;
1371 CHECK(msg->findInt32("what", &what));
1372
1373 switch (what) {
1374 case Source::kWhatPrepared:
1375 {
1376 if (mSource == NULL) {
1377 // This is a stale notification from a source that was
1378 // asynchronously preparing when the client called reset().
1379 // We handled the reset, the source is gone.
1380 break;
1381 }
1382
1383 int32_t err;
1384 CHECK(msg->findInt32("err", &err));
1385
1386 sp<NuPlayerDriver> driver = mDriver.promote();
1387 if (driver != NULL) {
1388 driver->notifyPrepareCompleted(err);
1389 }
1390
1391 int64_t durationUs;
1392 if (mDriver != NULL && mSource->getDuration(&durationUs) == OK) {
1393 sp<NuPlayerDriver> driver = mDriver.promote();
1394 if (driver != NULL) {
1395 driver->notifyDuration(durationUs);
1396 }
1397 }
1398 break;
1399 }
1400
1401 case Source::kWhatFlagsChanged:
1402 {
1403 uint32_t flags;
1404 CHECK(msg->findInt32("flags", (int32_t *)&flags));
1405
1406 sp<NuPlayerDriver> driver = mDriver.promote();
1407 if (driver != NULL) {
1408 driver->notifyFlagsChanged(flags);
1409 }
1410
1411 if ((mSourceFlags & Source::FLAG_DYNAMIC_DURATION)
1412 && (!(flags & Source::FLAG_DYNAMIC_DURATION))) {
1413 cancelPollDuration();
1414 } else if (!(mSourceFlags & Source::FLAG_DYNAMIC_DURATION)
1415 && (flags & Source::FLAG_DYNAMIC_DURATION)
1416 && (mAudioDecoder != NULL || mVideoDecoder != NULL)) {
1417 schedulePollDuration();
1418 }
1419
1420 mSourceFlags = flags;
1421 break;
1422 }
1423
1424 case Source::kWhatVideoSizeChanged:
1425 {
1426 int32_t width, height;
1427 CHECK(msg->findInt32("width", &width));
1428 CHECK(msg->findInt32("height", &height));
1429
1430 notifyListener(MEDIA_SET_VIDEO_SIZE, width, height);
1431 break;
1432 }
1433
1434 case Source::kWhatBufferingStart:
1435 {
1436 notifyListener(MEDIA_INFO, MEDIA_INFO_BUFFERING_START, 0);
1437 break;
1438 }
1439
1440 case Source::kWhatBufferingEnd:
1441 {
1442 notifyListener(MEDIA_INFO, MEDIA_INFO_BUFFERING_END, 0);
1443 break;
1444 }
1445
1446 case Source::kWhatSubtitleData:
1447 {
1448 sp<ABuffer> buffer;
1449 CHECK(msg->findBuffer("buffer", &buffer));
1450
1451 int32_t trackIndex;
1452 int64_t timeUs, durationUs;
1453 CHECK(buffer->meta()->findInt32("trackIndex", &trackIndex));
1454 CHECK(buffer->meta()->findInt64("timeUs", &timeUs));
1455 CHECK(buffer->meta()->findInt64("durationUs", &durationUs));
1456
1457 Parcel in;
1458 in.writeInt32(trackIndex);
1459 in.writeInt64(timeUs);
1460 in.writeInt64(durationUs);
1461 in.writeInt32(buffer->size());
1462 in.writeInt32(buffer->size());
1463 in.write(buffer->data(), buffer->size());
1464
1465 notifyListener(MEDIA_SUBTITLE_DATA, 0, 0, &in);
1466 break;
1467 }
1468
1469 case Source::kWhatQueueDecoderShutdown:
1470 {
1471 int32_t audio, video;
1472 CHECK(msg->findInt32("audio", &audio));
1473 CHECK(msg->findInt32("video", &video));
1474
1475 sp<AMessage> reply;
1476 CHECK(msg->findMessage("reply", &reply));
1477
1478 queueDecoderShutdown(audio, video, reply);
1479 break;
1480 }
1481
1482 default:
1483 TRESPASS();
1484 }
1485 }
1486
1487 ////////////////////////////////////////////////////////////////////////////////
1488
notifyFlagsChanged(uint32_t flags)1489 void NuPlayer::Source::notifyFlagsChanged(uint32_t flags) {
1490 sp<AMessage> notify = dupNotify();
1491 notify->setInt32("what", kWhatFlagsChanged);
1492 notify->setInt32("flags", flags);
1493 notify->post();
1494 }
1495
notifyVideoSizeChanged(int32_t width,int32_t height)1496 void NuPlayer::Source::notifyVideoSizeChanged(int32_t width, int32_t height) {
1497 sp<AMessage> notify = dupNotify();
1498 notify->setInt32("what", kWhatVideoSizeChanged);
1499 notify->setInt32("width", width);
1500 notify->setInt32("height", height);
1501 notify->post();
1502 }
1503
notifyPrepared(status_t err)1504 void NuPlayer::Source::notifyPrepared(status_t err) {
1505 sp<AMessage> notify = dupNotify();
1506 notify->setInt32("what", kWhatPrepared);
1507 notify->setInt32("err", err);
1508 notify->post();
1509 }
1510
onMessageReceived(const sp<AMessage> & msg)1511 void NuPlayer::Source::onMessageReceived(const sp<AMessage> &msg) {
1512 TRESPASS();
1513 }
1514
queueDecoderShutdown(bool audio,bool video,const sp<AMessage> & reply)1515 void NuPlayer::queueDecoderShutdown(
1516 bool audio, bool video, const sp<AMessage> &reply) {
1517 ALOGI("queueDecoderShutdown audio=%d, video=%d", audio, video);
1518
1519 mDeferredActions.push_back(
1520 new ShutdownDecoderAction(audio, video));
1521
1522 mDeferredActions.push_back(
1523 new SimpleAction(&NuPlayer::performScanSources));
1524
1525 mDeferredActions.push_back(new PostMessageAction(reply));
1526
1527 processDeferredActions();
1528 }
1529
1530 } // namespace android
1531