blob: 196b10364a3726c8a36804cc5011332b73b8f782 [file] [log] [blame]
Wei Jia53692fa2017-12-11 10:33:46 -08001/*
2 * Copyright 2017 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
Wei Jia2409c872018-02-02 10:34:33 -080018#define LOG_TAG "GenericSource2"
Wei Jia53692fa2017-12-11 10:33:46 -080019
Wei Jia2409c872018-02-02 10:34:33 -080020#include "GenericSource2.h"
Wei Jia53692fa2017-12-11 10:33:46 -080021#include "NuPlayer2Drm.h"
22
23#include "AnotherPacketSource.h"
24#include <binder/IServiceManager.h>
25#include <cutils/properties.h>
26#include <media/DataSource.h>
Dongwon Kangbc8f53b2018-01-25 17:01:44 -080027#include <media/MediaBufferHolder.h>
Wei Jia53692fa2017-12-11 10:33:46 -080028#include <media/IMediaExtractorService.h>
Dongwon Kang49ce6712018-01-24 10:16:25 -080029#include <media/IMediaSource.h>
Wei Jia53692fa2017-12-11 10:33:46 -080030#include <media/MediaHTTPService.h>
31#include <media/MediaExtractor.h>
32#include <media/MediaSource.h>
Wei Jia28288fb2017-12-15 13:45:29 -080033#include <media/NdkWrapper.h>
Wei Jia53692fa2017-12-11 10:33:46 -080034#include <media/stagefright/foundation/ABuffer.h>
35#include <media/stagefright/foundation/ADebug.h>
36#include <media/stagefright/foundation/AMessage.h>
37#include <media/stagefright/DataSourceFactory.h>
Wei Jia53692fa2017-12-11 10:33:46 -080038#include <media/stagefright/InterfaceUtils.h>
39#include <media/stagefright/MediaBuffer.h>
40#include <media/stagefright/MediaClock.h>
41#include <media/stagefright/MediaDefs.h>
42#include <media/stagefright/MediaExtractorFactory.h>
43#include <media/stagefright/MetaData.h>
Robert Shih0cd95062018-01-21 17:41:16 -080044#include <media/stagefright/NdkUtils.h>
Wei Jia53692fa2017-12-11 10:33:46 -080045#include <media/stagefright/Utils.h>
46#include "../../libstagefright/include/NuCachedSource2.h"
47#include "../../libstagefright/include/HTTPBase.h"
48
49namespace android {
50
51static const int kInitialMarkMs = 5000; // 5secs
52
53//static const int kPausePlaybackMarkMs = 2000; // 2secs
54static const int kResumePlaybackMarkMs = 15000; // 15secs
55
Wei Jia2409c872018-02-02 10:34:33 -080056NuPlayer2::GenericSource2::GenericSource2(
Wei Jia53692fa2017-12-11 10:33:46 -080057 const sp<AMessage> &notify,
Wei Jia53692fa2017-12-11 10:33:46 -080058 uid_t uid,
59 const sp<MediaClock> &mediaClock)
60 : Source(notify),
61 mAudioTimeUs(0),
62 mAudioLastDequeueTimeUs(0),
63 mVideoTimeUs(0),
64 mVideoLastDequeueTimeUs(0),
65 mPrevBufferPercentage(-1),
66 mPollBufferingGeneration(0),
67 mSentPauseOnBuffering(false),
68 mAudioDataGeneration(0),
69 mVideoDataGeneration(0),
70 mFetchSubtitleDataGeneration(0),
71 mFetchTimedTextDataGeneration(0),
72 mDurationUs(-1ll),
73 mAudioIsVorbis(false),
74 mIsSecure(false),
75 mIsStreaming(false),
Wei Jia53692fa2017-12-11 10:33:46 -080076 mUID(uid),
77 mMediaClock(mediaClock),
78 mFd(-1),
79 mBitrate(-1ll),
80 mPendingReadBufferTypes(0) {
Wei Jia2409c872018-02-02 10:34:33 -080081 ALOGV("GenericSource2");
Wei Jia53692fa2017-12-11 10:33:46 -080082 CHECK(mediaClock != NULL);
83
84 mBufferingSettings.mInitialMarkMs = kInitialMarkMs;
85 mBufferingSettings.mResumePlaybackMarkMs = kResumePlaybackMarkMs;
86 resetDataSource();
87}
88
Wei Jia2409c872018-02-02 10:34:33 -080089void NuPlayer2::GenericSource2::resetDataSource() {
Wei Jia53692fa2017-12-11 10:33:46 -080090 ALOGV("resetDataSource");
91
92 mHTTPService.clear();
93 mHttpSource.clear();
94 mDisconnected = false;
95 mUri.clear();
96 mUriHeaders.clear();
97 if (mFd >= 0) {
98 close(mFd);
99 mFd = -1;
100 }
101 mOffset = 0;
102 mLength = 0;
103 mStarted = false;
104 mPreparing = false;
105
106 mIsDrmProtected = false;
107 mIsDrmReleased = false;
108 mIsSecure = false;
109 mMimes.clear();
110}
111
Wei Jia2409c872018-02-02 10:34:33 -0800112status_t NuPlayer2::GenericSource2::setDataSource(
Wei Jia53692fa2017-12-11 10:33:46 -0800113 const sp<MediaHTTPService> &httpService,
114 const char *url,
115 const KeyedVector<String8, String8> *headers) {
116 Mutex::Autolock _l(mLock);
117 ALOGV("setDataSource url: %s", url);
118
119 resetDataSource();
120
121 mHTTPService = httpService;
122 mUri = url;
123
124 if (headers) {
125 mUriHeaders = *headers;
126 }
127
128 // delay data source creation to prepareAsync() to avoid blocking
129 // the calling thread in setDataSource for any significant time.
130 return OK;
131}
132
Wei Jia2409c872018-02-02 10:34:33 -0800133status_t NuPlayer2::GenericSource2::setDataSource(
Wei Jia53692fa2017-12-11 10:33:46 -0800134 int fd, int64_t offset, int64_t length) {
135 Mutex::Autolock _l(mLock);
136 ALOGV("setDataSource %d/%lld/%lld", fd, (long long)offset, (long long)length);
137
138 resetDataSource();
139
140 mFd = dup(fd);
141 mOffset = offset;
142 mLength = length;
143
144 // delay data source creation to prepareAsync() to avoid blocking
145 // the calling thread in setDataSource for any significant time.
146 return OK;
147}
148
Wei Jia2409c872018-02-02 10:34:33 -0800149status_t NuPlayer2::GenericSource2::setDataSource(const sp<DataSource>& source) {
Wei Jia53692fa2017-12-11 10:33:46 -0800150 Mutex::Autolock _l(mLock);
151 ALOGV("setDataSource (source: %p)", source.get());
152
153 resetDataSource();
154 mDataSource = source;
155 return OK;
156}
157
Wei Jia2409c872018-02-02 10:34:33 -0800158sp<MetaData> NuPlayer2::GenericSource2::getFileFormatMeta() const {
Wei Jia53692fa2017-12-11 10:33:46 -0800159 Mutex::Autolock _l(mLock);
160 return mFileMeta;
161}
162
Wei Jia2409c872018-02-02 10:34:33 -0800163status_t NuPlayer2::GenericSource2::initFromDataSource() {
Robert Shih0cd95062018-01-21 17:41:16 -0800164 mExtractor = new AMediaExtractorWrapper(AMediaExtractor_new());
165 CHECK(mDataSource != NULL || mFd != -1);
Wei Jia53692fa2017-12-11 10:33:46 -0800166 sp<DataSource> dataSource = mDataSource;
Robert Shih0cd95062018-01-21 17:41:16 -0800167 const int fd = mFd;
168 const int64_t offset = mOffset;
169 const int64_t length = mLength;
Wei Jia53692fa2017-12-11 10:33:46 -0800170
171 mLock.unlock();
172 // This might take long time if data source is not reliable.
Robert Shih0cd95062018-01-21 17:41:16 -0800173 status_t err;
174 if (dataSource != nullptr) {
175 mDataSourceWrapper = new AMediaDataSourceWrapper(dataSource);
176 err = mExtractor->setDataSource(mDataSourceWrapper->getAMediaDataSource());
177 } else {
178 err = mExtractor->setDataSource(fd, offset, length);
179 }
Wei Jia53692fa2017-12-11 10:33:46 -0800180
Robert Shih0cd95062018-01-21 17:41:16 -0800181 if (err != OK) {
182 ALOGE("initFromDataSource, failed to create data source!");
183 mLock.lock();
Wei Jia53692fa2017-12-11 10:33:46 -0800184 return UNKNOWN_ERROR;
185 }
186
Robert Shih0cd95062018-01-21 17:41:16 -0800187 size_t numtracks = mExtractor->getTrackCount();
Wei Jia53692fa2017-12-11 10:33:46 -0800188 if (numtracks == 0) {
189 ALOGE("initFromDataSource, source has no track!");
Robert Shih0cd95062018-01-21 17:41:16 -0800190 mLock.lock();
Wei Jia53692fa2017-12-11 10:33:46 -0800191 return UNKNOWN_ERROR;
192 }
193
194 mLock.lock();
Robert Shih0cd95062018-01-21 17:41:16 -0800195 mFd = -1;
196 mDataSource = dataSource;
197 mFileMeta = convertMediaFormatWrapperToMetaData(mExtractor->getFormat());
Wei Jia53692fa2017-12-11 10:33:46 -0800198 if (mFileMeta != NULL) {
199 int64_t duration;
200 if (mFileMeta->findInt64(kKeyDuration, &duration)) {
201 mDurationUs = duration;
202 }
203 }
204
205 int32_t totalBitrate = 0;
206
207 mMimes.clear();
208
209 for (size_t i = 0; i < numtracks; ++i) {
Wei Jia53692fa2017-12-11 10:33:46 -0800210
Robert Shih0cd95062018-01-21 17:41:16 -0800211 sp<AMediaFormatWrapper> trackFormat = mExtractor->getTrackFormat(i);
212 if (trackFormat == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800213 ALOGE("no metadata for track %zu", i);
214 return UNKNOWN_ERROR;
215 }
216
Robert Shih0cd95062018-01-21 17:41:16 -0800217 sp<AMediaExtractorWrapper> trackExtractor = new AMediaExtractorWrapper(AMediaExtractor_new());
218 if (mDataSourceWrapper != nullptr) {
219 err = trackExtractor->setDataSource(mDataSourceWrapper->getAMediaDataSource());
220 } else {
221 err = trackExtractor->setDataSource(fd, offset, length);
222 }
223
Wei Jia53692fa2017-12-11 10:33:46 -0800224 const char *mime;
Robert Shih0cd95062018-01-21 17:41:16 -0800225 sp<MetaData> meta = convertMediaFormatWrapperToMetaData(trackFormat);
Wei Jia53692fa2017-12-11 10:33:46 -0800226 CHECK(meta->findCString(kKeyMIMEType, &mime));
227
228 ALOGV("initFromDataSource track[%zu]: %s", i, mime);
229
230 // Do the string compare immediately with "mime",
231 // we can't assume "mime" would stay valid after another
232 // extractor operation, some extractors might modify meta
233 // during getTrack() and make it invalid.
234 if (!strncasecmp(mime, "audio/", 6)) {
Robert Shih0cd95062018-01-21 17:41:16 -0800235 if (mAudioTrack.mExtractor == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800236 mAudioTrack.mIndex = i;
Robert Shih0cd95062018-01-21 17:41:16 -0800237 mAudioTrack.mExtractor = trackExtractor;
238 mAudioTrack.mExtractor->selectTrack(i);
239 mAudioTrack.mPackets = new AnotherPacketSource(meta);
Wei Jia53692fa2017-12-11 10:33:46 -0800240
241 if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_VORBIS)) {
242 mAudioIsVorbis = true;
243 } else {
244 mAudioIsVorbis = false;
245 }
246
247 mMimes.add(String8(mime));
248 }
249 } else if (!strncasecmp(mime, "video/", 6)) {
Robert Shih0cd95062018-01-21 17:41:16 -0800250 if (mVideoTrack.mExtractor == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800251 mVideoTrack.mIndex = i;
Robert Shih0cd95062018-01-21 17:41:16 -0800252 mVideoTrack.mExtractor = trackExtractor;
253 mVideoTrack.mExtractor->selectTrack(i);
254 mVideoTrack.mPackets = new AnotherPacketSource(meta);
Wei Jia53692fa2017-12-11 10:33:46 -0800255
256 // video always at the beginning
257 mMimes.insertAt(String8(mime), 0);
258 }
259 }
260
Robert Shih0cd95062018-01-21 17:41:16 -0800261 mExtractors.push(trackExtractor);
Wei Jia53692fa2017-12-11 10:33:46 -0800262 int64_t durationUs;
263 if (meta->findInt64(kKeyDuration, &durationUs)) {
264 if (durationUs > mDurationUs) {
265 mDurationUs = durationUs;
266 }
267 }
268
269 int32_t bitrate;
270 if (totalBitrate >= 0 && meta->findInt32(kKeyBitRate, &bitrate)) {
271 totalBitrate += bitrate;
272 } else {
273 totalBitrate = -1;
274 }
275 }
276
Robert Shih0cd95062018-01-21 17:41:16 -0800277 ALOGV("initFromDataSource mExtractors.size(): %zu mIsSecure: %d mime[0]: %s", mExtractors.size(),
Wei Jia53692fa2017-12-11 10:33:46 -0800278 mIsSecure, (mMimes.isEmpty() ? "NONE" : mMimes[0].string()));
279
Robert Shih0cd95062018-01-21 17:41:16 -0800280 if (mExtractors.size() == 0) {
Wei Jia53692fa2017-12-11 10:33:46 -0800281 ALOGE("b/23705695");
282 return UNKNOWN_ERROR;
283 }
284
285 // Modular DRM: The return value doesn't affect source initialization.
286 (void)checkDrmInfo();
287
288 mBitrate = totalBitrate;
289
290 return OK;
291}
292
Wei Jia2409c872018-02-02 10:34:33 -0800293status_t NuPlayer2::GenericSource2::getBufferingSettings(
Wei Jia53692fa2017-12-11 10:33:46 -0800294 BufferingSettings* buffering /* nonnull */) {
295 {
296 Mutex::Autolock _l(mLock);
297 *buffering = mBufferingSettings;
298 }
299
300 ALOGV("getBufferingSettings{%s}", buffering->toString().string());
301 return OK;
302}
303
Wei Jia2409c872018-02-02 10:34:33 -0800304status_t NuPlayer2::GenericSource2::setBufferingSettings(const BufferingSettings& buffering) {
Wei Jia53692fa2017-12-11 10:33:46 -0800305 ALOGV("setBufferingSettings{%s}", buffering.toString().string());
306
307 Mutex::Autolock _l(mLock);
308 mBufferingSettings = buffering;
309 return OK;
310}
311
Wei Jia2409c872018-02-02 10:34:33 -0800312int64_t NuPlayer2::GenericSource2::getLastReadPosition() {
Robert Shih0cd95062018-01-21 17:41:16 -0800313 if (mAudioTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800314 return mAudioTimeUs;
Robert Shih0cd95062018-01-21 17:41:16 -0800315 } else if (mVideoTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800316 return mVideoTimeUs;
317 } else {
318 return 0;
319 }
320}
321
Wei Jia2409c872018-02-02 10:34:33 -0800322bool NuPlayer2::GenericSource2::isStreaming() const {
Wei Jia53692fa2017-12-11 10:33:46 -0800323 Mutex::Autolock _l(mLock);
324 return mIsStreaming;
325}
326
Wei Jia2409c872018-02-02 10:34:33 -0800327NuPlayer2::GenericSource2::~GenericSource2() {
328 ALOGV("~GenericSource2");
Wei Jia53692fa2017-12-11 10:33:46 -0800329 if (mLooper != NULL) {
330 mLooper->unregisterHandler(id());
331 mLooper->stop();
332 }
Wei Jia17459332018-01-09 14:21:23 -0800333 if (mDataSource != NULL) {
334 mDataSource->close();
335 }
Wei Jia53692fa2017-12-11 10:33:46 -0800336 resetDataSource();
337}
338
Wei Jia2409c872018-02-02 10:34:33 -0800339void NuPlayer2::GenericSource2::prepareAsync() {
Wei Jia53692fa2017-12-11 10:33:46 -0800340 Mutex::Autolock _l(mLock);
341 ALOGV("prepareAsync: (looper: %d)", (mLooper != NULL));
342
343 if (mLooper == NULL) {
344 mLooper = new ALooper;
345 mLooper->setName("generic");
Wei Jiac5c79da2017-12-21 18:03:05 -0800346 mLooper->start(false, /* runOnCallingThread */
347 true, /* canCallJava */
348 PRIORITY_DEFAULT);
Wei Jia53692fa2017-12-11 10:33:46 -0800349
350 mLooper->registerHandler(this);
351 }
352
353 sp<AMessage> msg = new AMessage(kWhatPrepareAsync, this);
354 msg->post();
355}
356
Wei Jia2409c872018-02-02 10:34:33 -0800357void NuPlayer2::GenericSource2::onPrepareAsync() {
Wei Jia53692fa2017-12-11 10:33:46 -0800358 ALOGV("onPrepareAsync: mDataSource: %d", (mDataSource != NULL));
359
360 // delayed data source creation
361 if (mDataSource == NULL) {
362 // set to false first, if the extractor
363 // comes back as secure, set it to true then.
364 mIsSecure = false;
365
366 if (!mUri.empty()) {
367 const char* uri = mUri.c_str();
368 String8 contentType;
369
370 if (!strncasecmp("http://", uri, 7) || !strncasecmp("https://", uri, 8)) {
371 mHttpSource = DataSourceFactory::CreateMediaHTTP(mHTTPService);
372 if (mHttpSource == NULL) {
373 ALOGE("Failed to create http source!");
374 notifyPreparedAndCleanup(UNKNOWN_ERROR);
375 return;
376 }
377 }
378
379 mLock.unlock();
380 // This might take long time if connection has some issue.
381 sp<DataSource> dataSource = DataSourceFactory::CreateFromURI(
382 mHTTPService, uri, &mUriHeaders, &contentType,
383 static_cast<HTTPBase *>(mHttpSource.get()));
384 mLock.lock();
385 if (!mDisconnected) {
386 mDataSource = dataSource;
387 }
Wei Jia53692fa2017-12-11 10:33:46 -0800388 }
389
Robert Shih0cd95062018-01-21 17:41:16 -0800390 if (mFd == -1 && mDataSource == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800391 ALOGE("Failed to create data source!");
392 notifyPreparedAndCleanup(UNKNOWN_ERROR);
393 return;
394 }
395 }
396
Robert Shih0cd95062018-01-21 17:41:16 -0800397 if (mDataSource != nullptr && mDataSource->flags() & DataSource::kIsCachingDataSource) {
Wei Jia53692fa2017-12-11 10:33:46 -0800398 mCachedSource = static_cast<NuCachedSource2 *>(mDataSource.get());
399 }
400
401 // For cached streaming cases, we need to wait for enough
402 // buffering before reporting prepared.
403 mIsStreaming = (mCachedSource != NULL);
404
405 // init extractor from data source
406 status_t err = initFromDataSource();
407
408 if (err != OK) {
409 ALOGE("Failed to init from data source!");
410 notifyPreparedAndCleanup(err);
411 return;
412 }
413
Robert Shih0cd95062018-01-21 17:41:16 -0800414 if (mVideoTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800415 sp<MetaData> meta = getFormatMeta_l(false /* audio */);
416 sp<AMessage> msg = new AMessage;
417 err = convertMetaDataToMessage(meta, &msg);
418 if(err != OK) {
419 notifyPreparedAndCleanup(err);
420 return;
421 }
422 notifyVideoSizeChanged(msg);
423 }
424
425 notifyFlagsChanged(
426 // FLAG_SECURE will be known if/when prepareDrm is called by the app
427 // FLAG_PROTECTED will be known if/when prepareDrm is called by the app
428 FLAG_CAN_PAUSE |
429 FLAG_CAN_SEEK_BACKWARD |
430 FLAG_CAN_SEEK_FORWARD |
431 FLAG_CAN_SEEK);
432
433 finishPrepareAsync();
434
435 ALOGV("onPrepareAsync: Done");
436}
437
Wei Jia2409c872018-02-02 10:34:33 -0800438void NuPlayer2::GenericSource2::finishPrepareAsync() {
Wei Jia53692fa2017-12-11 10:33:46 -0800439 ALOGV("finishPrepareAsync");
440
Wei Jia53692fa2017-12-11 10:33:46 -0800441 if (mIsStreaming) {
442 mCachedSource->resumeFetchingIfNecessary();
443 mPreparing = true;
444 schedulePollBuffering();
445 } else {
446 notifyPrepared();
447 }
448
Robert Shih0cd95062018-01-21 17:41:16 -0800449 if (mAudioTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800450 postReadBuffer(MEDIA_TRACK_TYPE_AUDIO);
451 }
452
Robert Shih0cd95062018-01-21 17:41:16 -0800453 if (mVideoTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800454 postReadBuffer(MEDIA_TRACK_TYPE_VIDEO);
455 }
456}
457
Wei Jia2409c872018-02-02 10:34:33 -0800458void NuPlayer2::GenericSource2::notifyPreparedAndCleanup(status_t err) {
Wei Jia53692fa2017-12-11 10:33:46 -0800459 if (err != OK) {
460 mDataSource.clear();
461 mCachedSource.clear();
462 mHttpSource.clear();
463
464 mBitrate = -1;
465 mPrevBufferPercentage = -1;
466 ++mPollBufferingGeneration;
467 }
468 notifyPrepared(err);
469}
470
Wei Jia2409c872018-02-02 10:34:33 -0800471void NuPlayer2::GenericSource2::start() {
Wei Jia53692fa2017-12-11 10:33:46 -0800472 Mutex::Autolock _l(mLock);
473 ALOGI("start");
474
Robert Shih0cd95062018-01-21 17:41:16 -0800475 if (mAudioTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800476 postReadBuffer(MEDIA_TRACK_TYPE_AUDIO);
477 }
478
Robert Shih0cd95062018-01-21 17:41:16 -0800479 if (mVideoTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800480 postReadBuffer(MEDIA_TRACK_TYPE_VIDEO);
481 }
482
483 mStarted = true;
484}
485
Wei Jia2409c872018-02-02 10:34:33 -0800486void NuPlayer2::GenericSource2::stop() {
Wei Jia53692fa2017-12-11 10:33:46 -0800487 Mutex::Autolock _l(mLock);
488 mStarted = false;
489}
490
Wei Jia2409c872018-02-02 10:34:33 -0800491void NuPlayer2::GenericSource2::pause() {
Wei Jia53692fa2017-12-11 10:33:46 -0800492 Mutex::Autolock _l(mLock);
493 mStarted = false;
494}
495
Wei Jia2409c872018-02-02 10:34:33 -0800496void NuPlayer2::GenericSource2::resume() {
Wei Jia53692fa2017-12-11 10:33:46 -0800497 Mutex::Autolock _l(mLock);
498 mStarted = true;
499}
500
Wei Jia2409c872018-02-02 10:34:33 -0800501void NuPlayer2::GenericSource2::disconnect() {
Wei Jia53692fa2017-12-11 10:33:46 -0800502 sp<DataSource> dataSource, httpSource;
503 {
504 Mutex::Autolock _l(mLock);
505 dataSource = mDataSource;
506 httpSource = mHttpSource;
507 mDisconnected = true;
508 }
509
510 if (dataSource != NULL) {
511 // disconnect data source
512 if (dataSource->flags() & DataSource::kIsCachingDataSource) {
513 static_cast<NuCachedSource2 *>(dataSource.get())->disconnect();
514 }
515 } else if (httpSource != NULL) {
516 static_cast<HTTPBase *>(httpSource.get())->disconnect();
517 }
Robert Shih0cd95062018-01-21 17:41:16 -0800518
519 mDataSourceWrapper = NULL;
520
Wei Jia53692fa2017-12-11 10:33:46 -0800521}
522
Wei Jia2409c872018-02-02 10:34:33 -0800523status_t NuPlayer2::GenericSource2::feedMoreTSData() {
Wei Jia53692fa2017-12-11 10:33:46 -0800524 return OK;
525}
526
Wei Jia2409c872018-02-02 10:34:33 -0800527void NuPlayer2::GenericSource2::sendCacheStats() {
Wei Jia53692fa2017-12-11 10:33:46 -0800528 int32_t kbps = 0;
529 status_t err = UNKNOWN_ERROR;
530
531 if (mCachedSource != NULL) {
532 err = mCachedSource->getEstimatedBandwidthKbps(&kbps);
533 }
534
535 if (err == OK) {
536 sp<AMessage> notify = dupNotify();
537 notify->setInt32("what", kWhatCacheStats);
538 notify->setInt32("bandwidth", kbps);
539 notify->post();
540 }
541}
542
Wei Jia2409c872018-02-02 10:34:33 -0800543void NuPlayer2::GenericSource2::onMessageReceived(const sp<AMessage> &msg) {
Wei Jia53692fa2017-12-11 10:33:46 -0800544 Mutex::Autolock _l(mLock);
545 switch (msg->what()) {
546 case kWhatPrepareAsync:
547 {
548 onPrepareAsync();
549 break;
550 }
551 case kWhatFetchSubtitleData:
552 {
553 fetchTextData(kWhatSendSubtitleData, MEDIA_TRACK_TYPE_SUBTITLE,
554 mFetchSubtitleDataGeneration, mSubtitleTrack.mPackets, msg);
555 break;
556 }
557
558 case kWhatFetchTimedTextData:
559 {
560 fetchTextData(kWhatSendTimedTextData, MEDIA_TRACK_TYPE_TIMEDTEXT,
561 mFetchTimedTextDataGeneration, mTimedTextTrack.mPackets, msg);
562 break;
563 }
564
565 case kWhatSendSubtitleData:
566 {
567 sendTextData(kWhatSubtitleData, MEDIA_TRACK_TYPE_SUBTITLE,
568 mFetchSubtitleDataGeneration, mSubtitleTrack.mPackets, msg);
569 break;
570 }
571
572 case kWhatSendGlobalTimedTextData:
573 {
574 sendGlobalTextData(kWhatTimedTextData, mFetchTimedTextDataGeneration, msg);
575 break;
576 }
577 case kWhatSendTimedTextData:
578 {
579 sendTextData(kWhatTimedTextData, MEDIA_TRACK_TYPE_TIMEDTEXT,
580 mFetchTimedTextDataGeneration, mTimedTextTrack.mPackets, msg);
581 break;
582 }
583
584 case kWhatChangeAVSource:
585 {
586 int32_t trackIndex;
587 CHECK(msg->findInt32("trackIndex", &trackIndex));
Robert Shih0cd95062018-01-21 17:41:16 -0800588 const sp<AMediaExtractorWrapper> extractor = mExtractors.itemAt(trackIndex);
Wei Jia53692fa2017-12-11 10:33:46 -0800589
590 Track* track;
Robert Shih0cd95062018-01-21 17:41:16 -0800591 AString mime;
Wei Jia53692fa2017-12-11 10:33:46 -0800592 media_track_type trackType, counterpartType;
Robert Shih0cd95062018-01-21 17:41:16 -0800593 sp<AMediaFormatWrapper> format = extractor->getTrackFormat(trackIndex);
594 format->getString(AMEDIAFORMAT_KEY_MIME, &mime);
595 if (!strncasecmp(mime.c_str(), "audio/", 6)) {
Wei Jia53692fa2017-12-11 10:33:46 -0800596 track = &mAudioTrack;
597 trackType = MEDIA_TRACK_TYPE_AUDIO;
598 counterpartType = MEDIA_TRACK_TYPE_VIDEO;;
599 } else {
Robert Shih0cd95062018-01-21 17:41:16 -0800600 CHECK(!strncasecmp(mime.c_str(), "video/", 6));
Wei Jia53692fa2017-12-11 10:33:46 -0800601 track = &mVideoTrack;
602 trackType = MEDIA_TRACK_TYPE_VIDEO;
603 counterpartType = MEDIA_TRACK_TYPE_AUDIO;;
604 }
605
606
Robert Shih0cd95062018-01-21 17:41:16 -0800607 track->mExtractor = extractor;
608 track->mExtractor->selectSingleTrack(trackIndex);
Wei Jia53692fa2017-12-11 10:33:46 -0800609 track->mIndex = trackIndex;
610 ++mAudioDataGeneration;
611 ++mVideoDataGeneration;
612
613 int64_t timeUs, actualTimeUs;
614 const bool formatChange = true;
615 if (trackType == MEDIA_TRACK_TYPE_AUDIO) {
616 timeUs = mAudioLastDequeueTimeUs;
617 } else {
618 timeUs = mVideoLastDequeueTimeUs;
619 }
620 readBuffer(trackType, timeUs, MediaPlayer2SeekMode::SEEK_PREVIOUS_SYNC /* mode */,
621 &actualTimeUs, formatChange);
622 readBuffer(counterpartType, -1, MediaPlayer2SeekMode::SEEK_PREVIOUS_SYNC /* mode */,
623 NULL, !formatChange);
624 ALOGV("timeUs %lld actualTimeUs %lld", (long long)timeUs, (long long)actualTimeUs);
625
626 break;
627 }
628
629 case kWhatSeek:
630 {
631 onSeek(msg);
632 break;
633 }
634
635 case kWhatReadBuffer:
636 {
637 onReadBuffer(msg);
638 break;
639 }
640
641 case kWhatPollBuffering:
642 {
643 int32_t generation;
644 CHECK(msg->findInt32("generation", &generation));
645 if (generation == mPollBufferingGeneration) {
646 onPollBuffering();
647 }
648 break;
649 }
650
651 default:
652 Source::onMessageReceived(msg);
653 break;
654 }
655}
656
Wei Jia2409c872018-02-02 10:34:33 -0800657void NuPlayer2::GenericSource2::fetchTextData(
Wei Jia53692fa2017-12-11 10:33:46 -0800658 uint32_t sendWhat,
659 media_track_type type,
660 int32_t curGen,
661 const sp<AnotherPacketSource>& packets,
662 const sp<AMessage>& msg) {
663 int32_t msgGeneration;
664 CHECK(msg->findInt32("generation", &msgGeneration));
665 if (msgGeneration != curGen) {
666 // stale
667 return;
668 }
669
670 int32_t avail;
671 if (packets->hasBufferAvailable(&avail)) {
672 return;
673 }
674
675 int64_t timeUs;
676 CHECK(msg->findInt64("timeUs", &timeUs));
677
678 int64_t subTimeUs = 0;
679 readBuffer(type, timeUs, MediaPlayer2SeekMode::SEEK_PREVIOUS_SYNC /* mode */, &subTimeUs);
680
681 status_t eosResult;
682 if (!packets->hasBufferAvailable(&eosResult)) {
683 return;
684 }
685
686 if (msg->what() == kWhatFetchSubtitleData) {
687 subTimeUs -= 1000000ll; // send subtile data one second earlier
688 }
689 sp<AMessage> msg2 = new AMessage(sendWhat, this);
690 msg2->setInt32("generation", msgGeneration);
691 mMediaClock->addTimer(msg2, subTimeUs);
692}
693
Wei Jia2409c872018-02-02 10:34:33 -0800694void NuPlayer2::GenericSource2::sendTextData(
Wei Jia53692fa2017-12-11 10:33:46 -0800695 uint32_t what,
696 media_track_type type,
697 int32_t curGen,
698 const sp<AnotherPacketSource>& packets,
699 const sp<AMessage>& msg) {
700 int32_t msgGeneration;
701 CHECK(msg->findInt32("generation", &msgGeneration));
702 if (msgGeneration != curGen) {
703 // stale
704 return;
705 }
706
707 int64_t subTimeUs;
708 if (packets->nextBufferTime(&subTimeUs) != OK) {
709 return;
710 }
711
712 int64_t nextSubTimeUs;
713 readBuffer(type, -1, MediaPlayer2SeekMode::SEEK_PREVIOUS_SYNC /* mode */, &nextSubTimeUs);
714
715 sp<ABuffer> buffer;
716 status_t dequeueStatus = packets->dequeueAccessUnit(&buffer);
717 if (dequeueStatus == OK) {
718 sp<AMessage> notify = dupNotify();
719 notify->setInt32("what", what);
720 notify->setBuffer("buffer", buffer);
721 notify->post();
722
723 if (msg->what() == kWhatSendSubtitleData) {
724 nextSubTimeUs -= 1000000ll; // send subtile data one second earlier
725 }
726 mMediaClock->addTimer(msg, nextSubTimeUs);
727 }
728}
729
Wei Jia2409c872018-02-02 10:34:33 -0800730void NuPlayer2::GenericSource2::sendGlobalTextData(
Wei Jia53692fa2017-12-11 10:33:46 -0800731 uint32_t what,
732 int32_t curGen,
733 sp<AMessage> msg) {
734 int32_t msgGeneration;
735 CHECK(msg->findInt32("generation", &msgGeneration));
736 if (msgGeneration != curGen) {
737 // stale
738 return;
739 }
740
Robert Shih0cd95062018-01-21 17:41:16 -0800741 void *data = NULL;
Wei Jia53692fa2017-12-11 10:33:46 -0800742 size_t size = 0;
Robert Shih0cd95062018-01-21 17:41:16 -0800743 if (mTimedTextTrack.mExtractor->getTrackFormat(mTimedTextTrack.mIndex)->getBuffer(
744 "text", &data, &size)) {
Wei Jia53692fa2017-12-11 10:33:46 -0800745 mGlobalTimedText = new ABuffer(size);
746 if (mGlobalTimedText->data()) {
747 memcpy(mGlobalTimedText->data(), data, size);
748 sp<AMessage> globalMeta = mGlobalTimedText->meta();
749 globalMeta->setInt64("timeUs", 0);
750 globalMeta->setString("mime", MEDIA_MIMETYPE_TEXT_3GPP);
751 globalMeta->setInt32("global", 1);
752 sp<AMessage> notify = dupNotify();
753 notify->setInt32("what", what);
754 notify->setBuffer("buffer", mGlobalTimedText);
755 notify->post();
756 }
757 }
758}
759
Robert Shih0cd95062018-01-21 17:41:16 -0800760sp<AMessage> NuPlayer2::GenericSource2::getFormat(bool audio) {
761 Mutex::Autolock _l(mLock);
762 return getFormat_l(audio);
763}
764
Wei Jia2409c872018-02-02 10:34:33 -0800765sp<MetaData> NuPlayer2::GenericSource2::getFormatMeta(bool audio) {
Wei Jia53692fa2017-12-11 10:33:46 -0800766 Mutex::Autolock _l(mLock);
767 return getFormatMeta_l(audio);
768}
769
Robert Shih0cd95062018-01-21 17:41:16 -0800770sp<AMessage> NuPlayer2::GenericSource2::getFormat_l(bool audio) {
771 sp<AMediaExtractorWrapper> extractor = audio ? mAudioTrack.mExtractor : mVideoTrack.mExtractor;
772 size_t trackIndex = audio ? mAudioTrack.mIndex : mVideoTrack.mIndex;
Wei Jia53692fa2017-12-11 10:33:46 -0800773
Robert Shih0cd95062018-01-21 17:41:16 -0800774 if (extractor == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800775 return NULL;
776 }
777
Robert Shih0cd95062018-01-21 17:41:16 -0800778 return extractor->getTrackFormat(trackIndex)->toAMessage();
779}
780
781sp<MetaData> NuPlayer2::GenericSource2::getFormatMeta_l(bool audio) {
782 sp<AMediaExtractorWrapper> extractor = audio ? mAudioTrack.mExtractor : mVideoTrack.mExtractor;
783 size_t trackIndex = audio ? mAudioTrack.mIndex : mVideoTrack.mIndex;
784
785 if (extractor == NULL) {
786 return NULL;
787 }
788
789 return convertMediaFormatWrapperToMetaData(extractor->getTrackFormat(trackIndex));
Wei Jia53692fa2017-12-11 10:33:46 -0800790}
791
Wei Jia2409c872018-02-02 10:34:33 -0800792status_t NuPlayer2::GenericSource2::dequeueAccessUnit(
Wei Jia53692fa2017-12-11 10:33:46 -0800793 bool audio, sp<ABuffer> *accessUnit) {
794 Mutex::Autolock _l(mLock);
795 // If has gone through stop/releaseDrm sequence, we no longer send down any buffer b/c
796 // the codec's crypto object has gone away (b/37960096).
797 // Note: This will be unnecessary when stop() changes behavior and releases codec (b/35248283).
798 if (!mStarted && mIsDrmReleased) {
799 return -EWOULDBLOCK;
800 }
801
802 Track *track = audio ? &mAudioTrack : &mVideoTrack;
803
Robert Shih0cd95062018-01-21 17:41:16 -0800804 if (track->mExtractor == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800805 return -EWOULDBLOCK;
806 }
807
808 status_t finalResult;
809 if (!track->mPackets->hasBufferAvailable(&finalResult)) {
810 if (finalResult == OK) {
811 postReadBuffer(
812 audio ? MEDIA_TRACK_TYPE_AUDIO : MEDIA_TRACK_TYPE_VIDEO);
813 return -EWOULDBLOCK;
814 }
815 return finalResult;
816 }
817
818 status_t result = track->mPackets->dequeueAccessUnit(accessUnit);
819
820 // start pulling in more buffers if cache is running low
821 // so that decoder has less chance of being starved
822 if (!mIsStreaming) {
823 if (track->mPackets->getAvailableBufferCount(&finalResult) < 2) {
824 postReadBuffer(audio? MEDIA_TRACK_TYPE_AUDIO : MEDIA_TRACK_TYPE_VIDEO);
825 }
826 } else {
827 int64_t durationUs = track->mPackets->getBufferedDurationUs(&finalResult);
828 // TODO: maxRebufferingMarkMs could be larger than
829 // mBufferingSettings.mResumePlaybackMarkMs
830 int64_t restartBufferingMarkUs =
831 mBufferingSettings.mResumePlaybackMarkMs * 1000ll / 2;
832 if (finalResult == OK) {
833 if (durationUs < restartBufferingMarkUs) {
834 postReadBuffer(audio? MEDIA_TRACK_TYPE_AUDIO : MEDIA_TRACK_TYPE_VIDEO);
835 }
836 if (track->mPackets->getAvailableBufferCount(&finalResult) < 2
837 && !mSentPauseOnBuffering && !mPreparing) {
838 mCachedSource->resumeFetchingIfNecessary();
839 sendCacheStats();
840 mSentPauseOnBuffering = true;
841 sp<AMessage> notify = dupNotify();
842 notify->setInt32("what", kWhatPauseOnBufferingStart);
843 notify->post();
844 }
845 }
846 }
847
848 if (result != OK) {
Robert Shih0cd95062018-01-21 17:41:16 -0800849 if (mSubtitleTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800850 mSubtitleTrack.mPackets->clear();
851 mFetchSubtitleDataGeneration++;
852 }
Robert Shih0cd95062018-01-21 17:41:16 -0800853 if (mTimedTextTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800854 mTimedTextTrack.mPackets->clear();
855 mFetchTimedTextDataGeneration++;
856 }
857 return result;
858 }
859
860 int64_t timeUs;
861 status_t eosResult; // ignored
862 CHECK((*accessUnit)->meta()->findInt64("timeUs", &timeUs));
863 if (audio) {
864 mAudioLastDequeueTimeUs = timeUs;
865 } else {
866 mVideoLastDequeueTimeUs = timeUs;
867 }
868
Robert Shih0cd95062018-01-21 17:41:16 -0800869 if (mSubtitleTrack.mExtractor != NULL
Wei Jia53692fa2017-12-11 10:33:46 -0800870 && !mSubtitleTrack.mPackets->hasBufferAvailable(&eosResult)) {
871 sp<AMessage> msg = new AMessage(kWhatFetchSubtitleData, this);
872 msg->setInt64("timeUs", timeUs);
873 msg->setInt32("generation", mFetchSubtitleDataGeneration);
874 msg->post();
875 }
876
Robert Shih0cd95062018-01-21 17:41:16 -0800877 if (mTimedTextTrack.mExtractor != NULL
Wei Jia53692fa2017-12-11 10:33:46 -0800878 && !mTimedTextTrack.mPackets->hasBufferAvailable(&eosResult)) {
879 sp<AMessage> msg = new AMessage(kWhatFetchTimedTextData, this);
880 msg->setInt64("timeUs", timeUs);
881 msg->setInt32("generation", mFetchTimedTextDataGeneration);
882 msg->post();
883 }
884
885 return result;
886}
887
Wei Jia2409c872018-02-02 10:34:33 -0800888status_t NuPlayer2::GenericSource2::getDuration(int64_t *durationUs) {
Wei Jia53692fa2017-12-11 10:33:46 -0800889 Mutex::Autolock _l(mLock);
890 *durationUs = mDurationUs;
891 return OK;
892}
893
Wei Jia2409c872018-02-02 10:34:33 -0800894size_t NuPlayer2::GenericSource2::getTrackCount() const {
Wei Jia53692fa2017-12-11 10:33:46 -0800895 Mutex::Autolock _l(mLock);
Robert Shih0cd95062018-01-21 17:41:16 -0800896 return mExtractors.size();
Wei Jia53692fa2017-12-11 10:33:46 -0800897}
898
Wei Jia2409c872018-02-02 10:34:33 -0800899sp<AMessage> NuPlayer2::GenericSource2::getTrackInfo(size_t trackIndex) const {
Wei Jia53692fa2017-12-11 10:33:46 -0800900 Mutex::Autolock _l(mLock);
Robert Shih0cd95062018-01-21 17:41:16 -0800901 size_t trackCount = mExtractors.size();
Wei Jia53692fa2017-12-11 10:33:46 -0800902 if (trackIndex >= trackCount) {
903 return NULL;
904 }
905
Robert Shih0cd95062018-01-21 17:41:16 -0800906 sp<AMessage> format = mExtractors.itemAt(trackIndex)->getTrackFormat(trackIndex)->toAMessage();
907 if (format == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800908 ALOGE("no metadata for track %zu", trackIndex);
909 return NULL;
910 }
911
Robert Shih0cd95062018-01-21 17:41:16 -0800912 AString mime;
913 CHECK(format->findString(AMEDIAFORMAT_KEY_MIME, &mime));
Wei Jia53692fa2017-12-11 10:33:46 -0800914
915 int32_t trackType;
Robert Shih0cd95062018-01-21 17:41:16 -0800916 if (!strncasecmp(mime.c_str(), "video/", 6)) {
Wei Jia53692fa2017-12-11 10:33:46 -0800917 trackType = MEDIA_TRACK_TYPE_VIDEO;
Robert Shih0cd95062018-01-21 17:41:16 -0800918 } else if (!strncasecmp(mime.c_str(), "audio/", 6)) {
Wei Jia53692fa2017-12-11 10:33:46 -0800919 trackType = MEDIA_TRACK_TYPE_AUDIO;
Robert Shih0cd95062018-01-21 17:41:16 -0800920 } else if (!strcasecmp(mime.c_str(), MEDIA_MIMETYPE_TEXT_3GPP)) {
Wei Jia53692fa2017-12-11 10:33:46 -0800921 trackType = MEDIA_TRACK_TYPE_TIMEDTEXT;
922 } else {
923 trackType = MEDIA_TRACK_TYPE_UNKNOWN;
924 }
925 format->setInt32("type", trackType);
926
Robert Shih0cd95062018-01-21 17:41:16 -0800927 AString lang;
928 if (!format->findString("language", &lang)) {
929 format->setString("language", "und");
Wei Jia53692fa2017-12-11 10:33:46 -0800930 }
Wei Jia53692fa2017-12-11 10:33:46 -0800931
932 if (trackType == MEDIA_TRACK_TYPE_SUBTITLE) {
933 int32_t isAutoselect = 1, isDefault = 0, isForced = 0;
Robert Shih0cd95062018-01-21 17:41:16 -0800934 format->findInt32(AMEDIAFORMAT_KEY_IS_AUTOSELECT, &isAutoselect);
935 format->findInt32(AMEDIAFORMAT_KEY_IS_DEFAULT, &isDefault);
936 format->findInt32(AMEDIAFORMAT_KEY_IS_FORCED_SUBTITLE, &isForced);
Wei Jia53692fa2017-12-11 10:33:46 -0800937
938 format->setInt32("auto", !!isAutoselect);
939 format->setInt32("default", !!isDefault);
940 format->setInt32("forced", !!isForced);
941 }
942
943 return format;
944}
945
Wei Jia2409c872018-02-02 10:34:33 -0800946ssize_t NuPlayer2::GenericSource2::getSelectedTrack(media_track_type type) const {
Wei Jia53692fa2017-12-11 10:33:46 -0800947 Mutex::Autolock _l(mLock);
948 const Track *track = NULL;
949 switch (type) {
950 case MEDIA_TRACK_TYPE_VIDEO:
951 track = &mVideoTrack;
952 break;
953 case MEDIA_TRACK_TYPE_AUDIO:
954 track = &mAudioTrack;
955 break;
956 case MEDIA_TRACK_TYPE_TIMEDTEXT:
957 track = &mTimedTextTrack;
958 break;
959 case MEDIA_TRACK_TYPE_SUBTITLE:
960 track = &mSubtitleTrack;
961 break;
962 default:
963 break;
964 }
965
Robert Shih0cd95062018-01-21 17:41:16 -0800966 if (track != NULL && track->mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800967 return track->mIndex;
968 }
969
970 return -1;
971}
972
Wei Jia2409c872018-02-02 10:34:33 -0800973status_t NuPlayer2::GenericSource2::selectTrack(size_t trackIndex, bool select, int64_t timeUs) {
Wei Jia53692fa2017-12-11 10:33:46 -0800974 Mutex::Autolock _l(mLock);
975 ALOGV("%s track: %zu", select ? "select" : "deselect", trackIndex);
976
Robert Shih0cd95062018-01-21 17:41:16 -0800977 if (trackIndex >= mExtractors.size()) {
Wei Jia53692fa2017-12-11 10:33:46 -0800978 return BAD_INDEX;
979 }
980
981 if (!select) {
982 Track* track = NULL;
Robert Shih0cd95062018-01-21 17:41:16 -0800983 if (mSubtitleTrack.mExtractor != NULL && trackIndex == mSubtitleTrack.mIndex) {
Wei Jia53692fa2017-12-11 10:33:46 -0800984 track = &mSubtitleTrack;
985 mFetchSubtitleDataGeneration++;
Robert Shih0cd95062018-01-21 17:41:16 -0800986 } else if (mTimedTextTrack.mExtractor != NULL && trackIndex == mTimedTextTrack.mIndex) {
Wei Jia53692fa2017-12-11 10:33:46 -0800987 track = &mTimedTextTrack;
988 mFetchTimedTextDataGeneration++;
989 }
990 if (track == NULL) {
991 return INVALID_OPERATION;
992 }
Robert Shih0cd95062018-01-21 17:41:16 -0800993 track->mExtractor = NULL;
Wei Jia53692fa2017-12-11 10:33:46 -0800994 track->mPackets->clear();
995 return OK;
996 }
997
Robert Shih0cd95062018-01-21 17:41:16 -0800998 const sp<AMediaExtractorWrapper> extractor = mExtractors.itemAt(trackIndex);
999 sp<MetaData> meta = convertMediaFormatWrapperToMetaData(extractor->getTrackFormat(trackIndex));
Wei Jia53692fa2017-12-11 10:33:46 -08001000 const char *mime;
1001 CHECK(meta->findCString(kKeyMIMEType, &mime));
1002 if (!strncasecmp(mime, "text/", 5)) {
1003 bool isSubtitle = strcasecmp(mime, MEDIA_MIMETYPE_TEXT_3GPP);
1004 Track *track = isSubtitle ? &mSubtitleTrack : &mTimedTextTrack;
Robert Shih0cd95062018-01-21 17:41:16 -08001005 if (track->mExtractor != NULL && track->mIndex == trackIndex) {
Wei Jia53692fa2017-12-11 10:33:46 -08001006 return OK;
1007 }
1008 track->mIndex = trackIndex;
Robert Shih0cd95062018-01-21 17:41:16 -08001009 track->mExtractor = mExtractors.itemAt(trackIndex);
1010 track->mExtractor->selectSingleTrack(trackIndex);
Wei Jia53692fa2017-12-11 10:33:46 -08001011 if (track->mPackets == NULL) {
Robert Shih0cd95062018-01-21 17:41:16 -08001012 track->mPackets = new AnotherPacketSource(meta);
Wei Jia53692fa2017-12-11 10:33:46 -08001013 } else {
1014 track->mPackets->clear();
Robert Shih0cd95062018-01-21 17:41:16 -08001015 track->mPackets->setFormat(meta);
Wei Jia53692fa2017-12-11 10:33:46 -08001016
1017 }
1018
1019 if (isSubtitle) {
1020 mFetchSubtitleDataGeneration++;
1021 } else {
1022 mFetchTimedTextDataGeneration++;
1023 }
1024
1025 status_t eosResult; // ignored
Robert Shih0cd95062018-01-21 17:41:16 -08001026 if (mSubtitleTrack.mExtractor != NULL
Wei Jia53692fa2017-12-11 10:33:46 -08001027 && !mSubtitleTrack.mPackets->hasBufferAvailable(&eosResult)) {
1028 sp<AMessage> msg = new AMessage(kWhatFetchSubtitleData, this);
1029 msg->setInt64("timeUs", timeUs);
1030 msg->setInt32("generation", mFetchSubtitleDataGeneration);
1031 msg->post();
1032 }
1033
1034 sp<AMessage> msg2 = new AMessage(kWhatSendGlobalTimedTextData, this);
1035 msg2->setInt32("generation", mFetchTimedTextDataGeneration);
1036 msg2->post();
1037
Robert Shih0cd95062018-01-21 17:41:16 -08001038 if (mTimedTextTrack.mExtractor != NULL
Wei Jia53692fa2017-12-11 10:33:46 -08001039 && !mTimedTextTrack.mPackets->hasBufferAvailable(&eosResult)) {
1040 sp<AMessage> msg = new AMessage(kWhatFetchTimedTextData, this);
1041 msg->setInt64("timeUs", timeUs);
1042 msg->setInt32("generation", mFetchTimedTextDataGeneration);
1043 msg->post();
1044 }
1045
1046 return OK;
1047 } else if (!strncasecmp(mime, "audio/", 6) || !strncasecmp(mime, "video/", 6)) {
1048 bool audio = !strncasecmp(mime, "audio/", 6);
1049 Track *track = audio ? &mAudioTrack : &mVideoTrack;
Robert Shih0cd95062018-01-21 17:41:16 -08001050 if (track->mExtractor != NULL && track->mIndex == trackIndex) {
Wei Jia53692fa2017-12-11 10:33:46 -08001051 return OK;
1052 }
1053
1054 sp<AMessage> msg = new AMessage(kWhatChangeAVSource, this);
1055 msg->setInt32("trackIndex", trackIndex);
1056 msg->post();
1057 return OK;
1058 }
1059
1060 return INVALID_OPERATION;
1061}
1062
Wei Jia2409c872018-02-02 10:34:33 -08001063status_t NuPlayer2::GenericSource2::seekTo(int64_t seekTimeUs, MediaPlayer2SeekMode mode) {
Wei Jia53692fa2017-12-11 10:33:46 -08001064 ALOGV("seekTo: %lld, %d", (long long)seekTimeUs, mode);
1065 sp<AMessage> msg = new AMessage(kWhatSeek, this);
1066 msg->setInt64("seekTimeUs", seekTimeUs);
1067 msg->setInt32("mode", mode);
1068
1069 // Need to call readBuffer on |mLooper| to ensure the calls to
1070 // IMediaSource::read* are serialized. Note that IMediaSource::read*
1071 // is called without |mLock| acquired and MediaSource is not thread safe.
1072 sp<AMessage> response;
1073 status_t err = msg->postAndAwaitResponse(&response);
1074 if (err == OK && response != NULL) {
1075 CHECK(response->findInt32("err", &err));
1076 }
1077
1078 return err;
1079}
1080
Wei Jia2409c872018-02-02 10:34:33 -08001081void NuPlayer2::GenericSource2::onSeek(const sp<AMessage>& msg) {
Wei Jia53692fa2017-12-11 10:33:46 -08001082 int64_t seekTimeUs;
1083 int32_t mode;
1084 CHECK(msg->findInt64("seekTimeUs", &seekTimeUs));
1085 CHECK(msg->findInt32("mode", &mode));
1086
1087 sp<AMessage> response = new AMessage;
1088 status_t err = doSeek(seekTimeUs, (MediaPlayer2SeekMode)mode);
1089 response->setInt32("err", err);
1090
1091 sp<AReplyToken> replyID;
1092 CHECK(msg->senderAwaitsResponse(&replyID));
1093 response->postReply(replyID);
1094}
1095
Wei Jia2409c872018-02-02 10:34:33 -08001096status_t NuPlayer2::GenericSource2::doSeek(int64_t seekTimeUs, MediaPlayer2SeekMode mode) {
Robert Shih0cd95062018-01-21 17:41:16 -08001097 if (mVideoTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001098 ++mVideoDataGeneration;
1099
1100 int64_t actualTimeUs;
1101 readBuffer(MEDIA_TRACK_TYPE_VIDEO, seekTimeUs, mode, &actualTimeUs);
1102
1103 if (mode != MediaPlayer2SeekMode::SEEK_CLOSEST) {
1104 seekTimeUs = actualTimeUs;
1105 }
1106 mVideoLastDequeueTimeUs = actualTimeUs;
1107 }
1108
Robert Shih0cd95062018-01-21 17:41:16 -08001109 if (mAudioTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001110 ++mAudioDataGeneration;
1111 readBuffer(MEDIA_TRACK_TYPE_AUDIO, seekTimeUs, MediaPlayer2SeekMode::SEEK_CLOSEST);
1112 mAudioLastDequeueTimeUs = seekTimeUs;
1113 }
1114
Robert Shih0cd95062018-01-21 17:41:16 -08001115 if (mSubtitleTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001116 mSubtitleTrack.mPackets->clear();
1117 mFetchSubtitleDataGeneration++;
1118 }
1119
Robert Shih0cd95062018-01-21 17:41:16 -08001120 if (mTimedTextTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001121 mTimedTextTrack.mPackets->clear();
1122 mFetchTimedTextDataGeneration++;
1123 }
1124
1125 ++mPollBufferingGeneration;
1126 schedulePollBuffering();
1127 return OK;
1128}
1129
Wei Jia2409c872018-02-02 10:34:33 -08001130sp<ABuffer> NuPlayer2::GenericSource2::mediaBufferToABuffer(
Dongwon Kang1889c3e2018-02-01 13:44:57 -08001131 MediaBufferBase* mb,
Wei Jia53692fa2017-12-11 10:33:46 -08001132 media_track_type trackType) {
1133 bool audio = trackType == MEDIA_TRACK_TYPE_AUDIO;
1134 size_t outLength = mb->range_length();
1135
1136 if (audio && mAudioIsVorbis) {
1137 outLength += sizeof(int32_t);
1138 }
1139
1140 sp<ABuffer> ab;
1141
1142 if (mIsDrmProtected) {
1143 // Modular DRM
1144 // Enabled for both video/audio so 1) media buffer is reused without extra copying
1145 // 2) meta data can be retrieved in onInputBufferFetched for calling queueSecureInputBuffer.
1146
1147 // data is already provided in the buffer
1148 ab = new ABuffer(NULL, mb->range_length());
Dongwon Kangbc8f53b2018-01-25 17:01:44 -08001149 ab->meta()->setObject("mediaBufferHolder", new MediaBufferHolder(mb));
Wei Jia53692fa2017-12-11 10:33:46 -08001150
1151 // Modular DRM: Required b/c of the above add_ref.
1152 // If ref>0, there must be an observer, or it'll crash at release().
1153 // TODO: MediaBuffer might need to be revised to ease such need.
1154 mb->setObserver(this);
1155 // setMediaBufferBase() interestingly doesn't increment the ref count on its own.
1156 // Extra increment (since we want to keep mb alive and attached to ab beyond this function
1157 // call. This is to counter the effect of mb->release() towards the end.
1158 mb->add_ref();
1159
1160 } else {
1161 ab = new ABuffer(outLength);
1162 memcpy(ab->data(),
1163 (const uint8_t *)mb->data() + mb->range_offset(),
1164 mb->range_length());
1165 }
1166
1167 if (audio && mAudioIsVorbis) {
1168 int32_t numPageSamples;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001169 if (!mb->meta_data().findInt32(kKeyValidSamples, &numPageSamples)) {
Wei Jia53692fa2017-12-11 10:33:46 -08001170 numPageSamples = -1;
1171 }
1172
1173 uint8_t* abEnd = ab->data() + mb->range_length();
1174 memcpy(abEnd, &numPageSamples, sizeof(numPageSamples));
1175 }
1176
1177 sp<AMessage> meta = ab->meta();
1178
1179 int64_t timeUs;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001180 CHECK(mb->meta_data().findInt64(kKeyTime, &timeUs));
Wei Jia53692fa2017-12-11 10:33:46 -08001181 meta->setInt64("timeUs", timeUs);
1182
1183 if (trackType == MEDIA_TRACK_TYPE_VIDEO) {
1184 int32_t layerId;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001185 if (mb->meta_data().findInt32(kKeyTemporalLayerId, &layerId)) {
Wei Jia53692fa2017-12-11 10:33:46 -08001186 meta->setInt32("temporal-layer-id", layerId);
1187 }
1188 }
1189
1190 if (trackType == MEDIA_TRACK_TYPE_TIMEDTEXT) {
Robert Shih0cd95062018-01-21 17:41:16 -08001191 AString mime;
1192 sp<AMediaExtractorWrapper> extractor = mTimedTextTrack.mExtractor;
1193 size_t trackIndex = mTimedTextTrack.mIndex;
1194 CHECK(extractor != NULL
1195 && extractor->getTrackFormat(trackIndex)->getString(AMEDIAFORMAT_KEY_MIME, &mime));
1196 meta->setString("mime", mime.c_str());
Wei Jia53692fa2017-12-11 10:33:46 -08001197 }
1198
1199 int64_t durationUs;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001200 if (mb->meta_data().findInt64(kKeyDuration, &durationUs)) {
Wei Jia53692fa2017-12-11 10:33:46 -08001201 meta->setInt64("durationUs", durationUs);
1202 }
1203
1204 if (trackType == MEDIA_TRACK_TYPE_SUBTITLE) {
Robert Shihd83d4f42018-02-24 19:02:46 -08001205 meta->setInt32(AMEDIAFORMAT_KEY_TRACK_INDEX, mSubtitleTrack.mIndex);
Wei Jia53692fa2017-12-11 10:33:46 -08001206 }
1207
1208 uint32_t dataType; // unused
1209 const void *seiData;
1210 size_t seiLength;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001211 if (mb->meta_data().findData(kKeySEI, &dataType, &seiData, &seiLength)) {
Wei Jia53692fa2017-12-11 10:33:46 -08001212 sp<ABuffer> sei = ABuffer::CreateAsCopy(seiData, seiLength);;
1213 meta->setBuffer("sei", sei);
1214 }
1215
1216 const void *mpegUserDataPointer;
1217 size_t mpegUserDataLength;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001218 if (mb->meta_data().findData(
Wei Jia53692fa2017-12-11 10:33:46 -08001219 kKeyMpegUserData, &dataType, &mpegUserDataPointer, &mpegUserDataLength)) {
1220 sp<ABuffer> mpegUserData = ABuffer::CreateAsCopy(mpegUserDataPointer, mpegUserDataLength);
Robert Shihd83d4f42018-02-24 19:02:46 -08001221 meta->setBuffer(AMEDIAFORMAT_KEY_MPEG_USER_DATA, mpegUserData);
Wei Jia53692fa2017-12-11 10:33:46 -08001222 }
1223
1224 mb->release();
1225 mb = NULL;
1226
1227 return ab;
1228}
1229
Wei Jia2409c872018-02-02 10:34:33 -08001230int32_t NuPlayer2::GenericSource2::getDataGeneration(media_track_type type) const {
Wei Jia53692fa2017-12-11 10:33:46 -08001231 int32_t generation = -1;
1232 switch (type) {
1233 case MEDIA_TRACK_TYPE_VIDEO:
1234 generation = mVideoDataGeneration;
1235 break;
1236 case MEDIA_TRACK_TYPE_AUDIO:
1237 generation = mAudioDataGeneration;
1238 break;
1239 case MEDIA_TRACK_TYPE_TIMEDTEXT:
1240 generation = mFetchTimedTextDataGeneration;
1241 break;
1242 case MEDIA_TRACK_TYPE_SUBTITLE:
1243 generation = mFetchSubtitleDataGeneration;
1244 break;
1245 default:
1246 break;
1247 }
1248
1249 return generation;
1250}
1251
Wei Jia2409c872018-02-02 10:34:33 -08001252void NuPlayer2::GenericSource2::postReadBuffer(media_track_type trackType) {
Wei Jia53692fa2017-12-11 10:33:46 -08001253 if ((mPendingReadBufferTypes & (1 << trackType)) == 0) {
1254 mPendingReadBufferTypes |= (1 << trackType);
1255 sp<AMessage> msg = new AMessage(kWhatReadBuffer, this);
1256 msg->setInt32("trackType", trackType);
1257 msg->post();
1258 }
1259}
1260
Wei Jia2409c872018-02-02 10:34:33 -08001261void NuPlayer2::GenericSource2::onReadBuffer(const sp<AMessage>& msg) {
Wei Jia53692fa2017-12-11 10:33:46 -08001262 int32_t tmpType;
1263 CHECK(msg->findInt32("trackType", &tmpType));
1264 media_track_type trackType = (media_track_type)tmpType;
1265 mPendingReadBufferTypes &= ~(1 << trackType);
1266 readBuffer(trackType);
1267}
1268
Wei Jia2409c872018-02-02 10:34:33 -08001269void NuPlayer2::GenericSource2::readBuffer(
Wei Jia53692fa2017-12-11 10:33:46 -08001270 media_track_type trackType, int64_t seekTimeUs, MediaPlayer2SeekMode mode,
1271 int64_t *actualTimeUs, bool formatChange) {
1272 Track *track;
1273 size_t maxBuffers = 1;
1274 switch (trackType) {
1275 case MEDIA_TRACK_TYPE_VIDEO:
1276 track = &mVideoTrack;
1277 maxBuffers = 8; // too large of a number may influence seeks
1278 break;
1279 case MEDIA_TRACK_TYPE_AUDIO:
1280 track = &mAudioTrack;
1281 maxBuffers = 64;
1282 break;
1283 case MEDIA_TRACK_TYPE_SUBTITLE:
1284 track = &mSubtitleTrack;
1285 break;
1286 case MEDIA_TRACK_TYPE_TIMEDTEXT:
1287 track = &mTimedTextTrack;
1288 break;
1289 default:
1290 TRESPASS();
1291 }
1292
Robert Shih0cd95062018-01-21 17:41:16 -08001293 if (track->mExtractor == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001294 return;
1295 }
1296
1297 if (actualTimeUs) {
1298 *actualTimeUs = seekTimeUs;
1299 }
1300
Wei Jia53692fa2017-12-11 10:33:46 -08001301
1302 bool seeking = false;
Robert Shih0cd95062018-01-21 17:41:16 -08001303 sp<AMediaExtractorWrapper> extractor = track->mExtractor;
Wei Jia53692fa2017-12-11 10:33:46 -08001304 if (seekTimeUs >= 0) {
Robert Shih0cd95062018-01-21 17:41:16 -08001305 extractor->seekTo(seekTimeUs, mode);
Wei Jia53692fa2017-12-11 10:33:46 -08001306 seeking = true;
1307 }
1308
Wei Jia53692fa2017-12-11 10:33:46 -08001309 int32_t generation = getDataGeneration(trackType);
1310 for (size_t numBuffers = 0; numBuffers < maxBuffers; ) {
Robert Shih0cd95062018-01-21 17:41:16 -08001311 Vector<sp<ABuffer> > aBuffers;
Wei Jia53692fa2017-12-11 10:33:46 -08001312
Wei Jia53692fa2017-12-11 10:33:46 -08001313 mLock.unlock();
Robert Shih0cd95062018-01-21 17:41:16 -08001314
1315 sp<AMediaFormatWrapper> format;
1316 ssize_t sampleSize = -1;
1317 status_t err = extractor->getSampleFormat(format);
1318 if (err == OK) {
1319 sampleSize = extractor->getSampleSize();
Wei Jia53692fa2017-12-11 10:33:46 -08001320 }
Robert Shih0cd95062018-01-21 17:41:16 -08001321
1322 if (err != OK || sampleSize < 0) {
1323 mLock.lock();
1324 track->mPackets->signalEOS(err != OK ? err : ERROR_END_OF_STREAM);
1325 break;
1326 }
1327
1328 sp<ABuffer> abuf = new ABuffer(sampleSize);
1329 sampleSize = extractor->readSampleData(abuf);
Wei Jia53692fa2017-12-11 10:33:46 -08001330 mLock.lock();
1331
Wei Jia53692fa2017-12-11 10:33:46 -08001332 // in case track has been changed since we don't have lock for some time.
1333 if (generation != getDataGeneration(trackType)) {
Wei Jia53692fa2017-12-11 10:33:46 -08001334 break;
1335 }
1336
Robert Shih0cd95062018-01-21 17:41:16 -08001337 int64_t timeUs = extractor->getSampleTime();
1338 if (timeUs < 0) {
1339 track->mPackets->signalEOS(ERROR_MALFORMED);
Wei Jia53692fa2017-12-11 10:33:46 -08001340 break;
1341 }
1342
Robert Shih0cd95062018-01-21 17:41:16 -08001343 sp<AMessage> meta = abuf->meta();
1344 format->writeToAMessage(meta);
1345 meta->setInt64("timeUs", timeUs);
1346 if (trackType == MEDIA_TRACK_TYPE_AUDIO) {
1347 mAudioTimeUs = timeUs;
1348 } else if (trackType == MEDIA_TRACK_TYPE_VIDEO) {
1349 mVideoTimeUs = timeUs;
Wei Jia53692fa2017-12-11 10:33:46 -08001350 }
Robert Shih0cd95062018-01-21 17:41:16 -08001351
1352 queueDiscontinuityIfNeeded(seeking, formatChange, trackType, track);
1353
1354 if (numBuffers == 0 && actualTimeUs != nullptr) {
1355 *actualTimeUs = timeUs;
1356 }
1357 if (seeking) {
1358 if (meta != nullptr && mode == MediaPlayer2SeekMode::SEEK_CLOSEST
1359 && seekTimeUs > timeUs) {
1360 sp<AMessage> extra = new AMessage;
1361 extra->setInt64("resume-at-mediaTimeUs", seekTimeUs);
1362 meta->setMessage("extra", extra);
1363 }
1364 }
1365
1366 track->mPackets->queueAccessUnit(abuf);
1367 formatChange = false;
1368 seeking = false;
1369 ++numBuffers;
1370 extractor->advance();
1371
Wei Jia53692fa2017-12-11 10:33:46 -08001372 }
1373
1374 if (mIsStreaming
1375 && (trackType == MEDIA_TRACK_TYPE_VIDEO || trackType == MEDIA_TRACK_TYPE_AUDIO)) {
1376 status_t finalResult;
1377 int64_t durationUs = track->mPackets->getBufferedDurationUs(&finalResult);
1378
1379 // TODO: maxRebufferingMarkMs could be larger than
1380 // mBufferingSettings.mResumePlaybackMarkMs
1381 int64_t markUs = (mPreparing ? mBufferingSettings.mInitialMarkMs
1382 : mBufferingSettings.mResumePlaybackMarkMs) * 1000ll;
1383 if (finalResult == ERROR_END_OF_STREAM || durationUs >= markUs) {
1384 if (mPreparing || mSentPauseOnBuffering) {
1385 Track *counterTrack =
1386 (trackType == MEDIA_TRACK_TYPE_VIDEO ? &mAudioTrack : &mVideoTrack);
Robert Shih0cd95062018-01-21 17:41:16 -08001387 if (counterTrack->mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001388 durationUs = counterTrack->mPackets->getBufferedDurationUs(&finalResult);
1389 }
1390 if (finalResult == ERROR_END_OF_STREAM || durationUs >= markUs) {
1391 if (mPreparing) {
1392 notifyPrepared();
1393 mPreparing = false;
1394 } else {
1395 sendCacheStats();
1396 mSentPauseOnBuffering = false;
1397 sp<AMessage> notify = dupNotify();
1398 notify->setInt32("what", kWhatResumeOnBufferingEnd);
1399 notify->post();
1400 }
1401 }
1402 }
1403 return;
1404 }
1405
1406 postReadBuffer(trackType);
1407 }
1408}
1409
Wei Jia2409c872018-02-02 10:34:33 -08001410void NuPlayer2::GenericSource2::queueDiscontinuityIfNeeded(
Wei Jia53692fa2017-12-11 10:33:46 -08001411 bool seeking, bool formatChange, media_track_type trackType, Track *track) {
1412 // formatChange && seeking: track whose source is changed during selection
1413 // formatChange && !seeking: track whose source is not changed during selection
1414 // !formatChange: normal seek
1415 if ((seeking || formatChange)
1416 && (trackType == MEDIA_TRACK_TYPE_AUDIO
1417 || trackType == MEDIA_TRACK_TYPE_VIDEO)) {
1418 ATSParser::DiscontinuityType type = (formatChange && seeking)
1419 ? ATSParser::DISCONTINUITY_FORMATCHANGE
1420 : ATSParser::DISCONTINUITY_NONE;
1421 track->mPackets->queueDiscontinuity(type, NULL /* extra */, true /* discard */);
1422 }
1423}
1424
Wei Jia2409c872018-02-02 10:34:33 -08001425void NuPlayer2::GenericSource2::notifyBufferingUpdate(int32_t percentage) {
Wei Jia53692fa2017-12-11 10:33:46 -08001426 // Buffering percent could go backward as it's estimated from remaining
1427 // data and last access time. This could cause the buffering position
1428 // drawn on media control to jitter slightly. Remember previously reported
1429 // percentage and don't allow it to go backward.
1430 if (percentage < mPrevBufferPercentage) {
1431 percentage = mPrevBufferPercentage;
1432 } else if (percentage > 100) {
1433 percentage = 100;
1434 }
1435
1436 mPrevBufferPercentage = percentage;
1437
1438 ALOGV("notifyBufferingUpdate: buffering %d%%", percentage);
1439
1440 sp<AMessage> notify = dupNotify();
1441 notify->setInt32("what", kWhatBufferingUpdate);
1442 notify->setInt32("percentage", percentage);
1443 notify->post();
1444}
1445
Wei Jia2409c872018-02-02 10:34:33 -08001446void NuPlayer2::GenericSource2::schedulePollBuffering() {
Wei Jia53692fa2017-12-11 10:33:46 -08001447 sp<AMessage> msg = new AMessage(kWhatPollBuffering, this);
1448 msg->setInt32("generation", mPollBufferingGeneration);
1449 // Enquires buffering status every second.
1450 msg->post(1000000ll);
1451}
1452
Wei Jia2409c872018-02-02 10:34:33 -08001453void NuPlayer2::GenericSource2::onPollBuffering() {
Wei Jia53692fa2017-12-11 10:33:46 -08001454 status_t finalStatus = UNKNOWN_ERROR;
1455 int64_t cachedDurationUs = -1ll;
1456 ssize_t cachedDataRemaining = -1;
1457
1458 if (mCachedSource != NULL) {
1459 cachedDataRemaining = mCachedSource->approxDataRemaining(&finalStatus);
1460
1461 if (finalStatus == OK) {
1462 off64_t size;
1463 int64_t bitrate = 0ll;
1464 if (mDurationUs > 0 && mCachedSource->getSize(&size) == OK) {
1465 // |bitrate| uses bits/second unit, while size is number of bytes.
1466 bitrate = size * 8000000ll / mDurationUs;
1467 } else if (mBitrate > 0) {
1468 bitrate = mBitrate;
1469 }
1470 if (bitrate > 0) {
1471 cachedDurationUs = cachedDataRemaining * 8000000ll / bitrate;
1472 }
1473 }
1474 }
1475
1476 if (finalStatus != OK) {
1477 ALOGV("onPollBuffering: EOS (finalStatus = %d)", finalStatus);
1478
1479 if (finalStatus == ERROR_END_OF_STREAM) {
1480 notifyBufferingUpdate(100);
1481 }
1482
1483 return;
1484 }
1485
1486 if (cachedDurationUs >= 0ll) {
1487 if (mDurationUs > 0ll) {
1488 int64_t cachedPosUs = getLastReadPosition() + cachedDurationUs;
1489 int percentage = 100.0 * cachedPosUs / mDurationUs;
1490 if (percentage > 100) {
1491 percentage = 100;
1492 }
1493
1494 notifyBufferingUpdate(percentage);
1495 }
1496
1497 ALOGV("onPollBuffering: cachedDurationUs %.1f sec", cachedDurationUs / 1000000.0f);
1498 }
1499
1500 schedulePollBuffering();
1501}
1502
1503// Modular DRM
Wei Jia2409c872018-02-02 10:34:33 -08001504status_t NuPlayer2::GenericSource2::prepareDrm(
Wei Jia53692fa2017-12-11 10:33:46 -08001505 const uint8_t uuid[16],
1506 const Vector<uint8_t> &drmSessionId,
1507 sp<AMediaCryptoWrapper> *outCrypto) {
1508 Mutex::Autolock _l(mLock);
1509 ALOGV("prepareDrm");
1510
1511 mIsDrmProtected = false;
1512 mIsDrmReleased = false;
1513 mIsSecure = false;
1514
1515 status_t status = OK;
1516 sp<AMediaCryptoWrapper> crypto =
1517 new AMediaCryptoWrapper(uuid, drmSessionId.array(), drmSessionId.size());
1518 if (crypto == NULL) {
1519 ALOGE("prepareDrm: failed to create crypto.");
1520 return UNKNOWN_ERROR;
1521 }
1522 ALOGV("prepareDrm: crypto created for uuid: %s",
1523 DrmUUID::toHexString(uuid).string());
1524
1525 *outCrypto = crypto;
1526 // as long a there is an active crypto
1527 mIsDrmProtected = true;
1528
1529 if (mMimes.size() == 0) {
1530 status = UNKNOWN_ERROR;
1531 ALOGE("prepareDrm: Unexpected. Must have at least one track. status: %d", status);
1532 return status;
1533 }
1534
1535 // first mime in this list is either the video track, or the first audio track
1536 const char *mime = mMimes[0].string();
1537 mIsSecure = crypto->requiresSecureDecoderComponent(mime);
1538 ALOGV("prepareDrm: requiresSecureDecoderComponent mime: %s isSecure: %d",
1539 mime, mIsSecure);
1540
1541 // Checking the member flags while in the looper to send out the notification.
1542 // The legacy mDecryptHandle!=NULL check (for FLAG_PROTECTED) is equivalent to mIsDrmProtected.
1543 notifyFlagsChanged(
1544 (mIsSecure ? FLAG_SECURE : 0) |
1545 // Setting "protected screen" only for L1: b/38390836
1546 (mIsSecure ? FLAG_PROTECTED : 0) |
1547 FLAG_CAN_PAUSE |
1548 FLAG_CAN_SEEK_BACKWARD |
1549 FLAG_CAN_SEEK_FORWARD |
1550 FLAG_CAN_SEEK);
1551
1552 if (status == OK) {
1553 ALOGV("prepareDrm: mCrypto: %p", outCrypto->get());
1554 ALOGD("prepareDrm ret: %d ", status);
1555 } else {
1556 ALOGE("prepareDrm err: %d", status);
1557 }
1558 return status;
1559}
1560
Wei Jia2409c872018-02-02 10:34:33 -08001561status_t NuPlayer2::GenericSource2::releaseDrm() {
Wei Jia53692fa2017-12-11 10:33:46 -08001562 Mutex::Autolock _l(mLock);
1563 ALOGV("releaseDrm");
1564
1565 if (mIsDrmProtected) {
1566 mIsDrmProtected = false;
1567 // to prevent returning any more buffer after stop/releaseDrm (b/37960096)
1568 mIsDrmReleased = true;
1569 ALOGV("releaseDrm: mIsDrmProtected is reset.");
1570 } else {
1571 ALOGE("releaseDrm: mIsDrmProtected is already false.");
1572 }
1573
1574 return OK;
1575}
1576
Wei Jia2409c872018-02-02 10:34:33 -08001577status_t NuPlayer2::GenericSource2::checkDrmInfo()
Wei Jia53692fa2017-12-11 10:33:46 -08001578{
1579 // clearing the flag at prepare in case the player is reused after stop/releaseDrm with the
1580 // same source without being reset (called by prepareAsync/initFromDataSource)
1581 mIsDrmReleased = false;
1582
Robert Shih0cd95062018-01-21 17:41:16 -08001583 if (mExtractor == NULL) {
1584 ALOGV("checkDrmInfo: No extractor");
Wei Jia53692fa2017-12-11 10:33:46 -08001585 return OK; // letting the caller responds accordingly
1586 }
1587
Robert Shih0cd95062018-01-21 17:41:16 -08001588 PsshInfo *psshInfo = mExtractor->getPsshInfo();
1589 if (psshInfo == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001590 ALOGV("checkDrmInfo: No PSSH");
1591 return OK; // source without DRM info
1592 }
1593
Robert Shih0cd95062018-01-21 17:41:16 -08001594 sp<ABuffer> drmInfoBuffer = NuPlayer2Drm::retrieveDrmInfo(psshInfo);
1595 ALOGV("checkDrmInfo: MEDIA_DRM_INFO PSSH drm info size: %d", (int)drmInfoBuffer->size());
Wei Jia53692fa2017-12-11 10:33:46 -08001596
Robert Shih48843252018-01-09 15:24:07 -08001597 if (drmInfoBuffer->size() == 0) {
Robert Shih0cd95062018-01-21 17:41:16 -08001598 ALOGE("checkDrmInfo: Unexpected parcel size: 0");
Wei Jia53692fa2017-12-11 10:33:46 -08001599 return UNKNOWN_ERROR;
1600 }
1601
Wei Jia53692fa2017-12-11 10:33:46 -08001602 notifyDrmInfo(drmInfoBuffer);
1603
1604 return OK;
1605}
1606
Dongwon Kang1889c3e2018-02-01 13:44:57 -08001607void NuPlayer2::GenericSource2::signalBufferReturned(MediaBufferBase *buffer)
Wei Jia53692fa2017-12-11 10:33:46 -08001608{
1609 //ALOGV("signalBufferReturned %p refCount: %d", buffer, buffer->localRefcount());
1610
1611 buffer->setObserver(NULL);
1612 buffer->release(); // this leads to delete since that there is no observor
1613}
1614
1615} // namespace android