blob: f795478ac8058c73f4db5d0e78d0977192dfcad6 [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"
Wei Jia53692fa2017-12-11 10:33:46 -080024#include <cutils/properties.h>
25#include <media/DataSource.h>
Dongwon Kangbc8f53b2018-01-25 17:01:44 -080026#include <media/MediaBufferHolder.h>
Wei Jia28288fb2017-12-15 13:45:29 -080027#include <media/NdkWrapper.h>
Wei Jia53692fa2017-12-11 10:33:46 -080028#include <media/stagefright/foundation/ABuffer.h>
29#include <media/stagefright/foundation/ADebug.h>
30#include <media/stagefright/foundation/AMessage.h>
Wei Jia53692fa2017-12-11 10:33:46 -080031#include <media/stagefright/MediaBuffer.h>
32#include <media/stagefright/MediaClock.h>
33#include <media/stagefright/MediaDefs.h>
Wei Jia53692fa2017-12-11 10:33:46 -080034#include <media/stagefright/MetaData.h>
Robert Shih0cd95062018-01-21 17:41:16 -080035#include <media/stagefright/NdkUtils.h>
Wei Jia53692fa2017-12-11 10:33:46 -080036#include <media/stagefright/Utils.h>
Robert Shih49fb89d2018-01-31 17:53:19 -080037#include "NdkMediaDataSourceCallbacksPriv.h"
Wei Jia53692fa2017-12-11 10:33:46 -080038
39namespace android {
40
41static const int kInitialMarkMs = 5000; // 5secs
42
43//static const int kPausePlaybackMarkMs = 2000; // 2secs
44static const int kResumePlaybackMarkMs = 15000; // 15secs
45
Wei Jia2409c872018-02-02 10:34:33 -080046NuPlayer2::GenericSource2::GenericSource2(
Wei Jia53692fa2017-12-11 10:33:46 -080047 const sp<AMessage> &notify,
Wei Jia53692fa2017-12-11 10:33:46 -080048 uid_t uid,
49 const sp<MediaClock> &mediaClock)
50 : Source(notify),
51 mAudioTimeUs(0),
52 mAudioLastDequeueTimeUs(0),
53 mVideoTimeUs(0),
54 mVideoLastDequeueTimeUs(0),
55 mPrevBufferPercentage(-1),
56 mPollBufferingGeneration(0),
57 mSentPauseOnBuffering(false),
58 mAudioDataGeneration(0),
59 mVideoDataGeneration(0),
60 mFetchSubtitleDataGeneration(0),
61 mFetchTimedTextDataGeneration(0),
62 mDurationUs(-1ll),
63 mAudioIsVorbis(false),
64 mIsSecure(false),
65 mIsStreaming(false),
Wei Jia53692fa2017-12-11 10:33:46 -080066 mUID(uid),
67 mMediaClock(mediaClock),
68 mFd(-1),
69 mBitrate(-1ll),
70 mPendingReadBufferTypes(0) {
Wei Jia2409c872018-02-02 10:34:33 -080071 ALOGV("GenericSource2");
Wei Jia53692fa2017-12-11 10:33:46 -080072 CHECK(mediaClock != NULL);
73
74 mBufferingSettings.mInitialMarkMs = kInitialMarkMs;
75 mBufferingSettings.mResumePlaybackMarkMs = kResumePlaybackMarkMs;
76 resetDataSource();
77}
78
Wei Jia2409c872018-02-02 10:34:33 -080079void NuPlayer2::GenericSource2::resetDataSource() {
Wei Jia53692fa2017-12-11 10:33:46 -080080 ALOGV("resetDataSource");
81
Robert Shih3d2b89b2018-10-24 17:53:23 -070082 mDisconnected = false;
Wei Jia53692fa2017-12-11 10:33:46 -080083 mUri.clear();
84 mUriHeaders.clear();
85 if (mFd >= 0) {
86 close(mFd);
87 mFd = -1;
88 }
89 mOffset = 0;
90 mLength = 0;
91 mStarted = false;
92 mPreparing = false;
93
94 mIsDrmProtected = false;
95 mIsDrmReleased = false;
96 mIsSecure = false;
97 mMimes.clear();
98}
99
Wei Jia2409c872018-02-02 10:34:33 -0800100status_t NuPlayer2::GenericSource2::setDataSource(
Wei Jia53692fa2017-12-11 10:33:46 -0800101 const char *url,
102 const KeyedVector<String8, String8> *headers) {
103 Mutex::Autolock _l(mLock);
104 ALOGV("setDataSource url: %s", url);
105
106 resetDataSource();
107
Wei Jia53692fa2017-12-11 10:33:46 -0800108 mUri = url;
109
110 if (headers) {
111 mUriHeaders = *headers;
112 }
113
114 // delay data source creation to prepareAsync() to avoid blocking
115 // the calling thread in setDataSource for any significant time.
116 return OK;
117}
118
Wei Jia2409c872018-02-02 10:34:33 -0800119status_t NuPlayer2::GenericSource2::setDataSource(
Wei Jia53692fa2017-12-11 10:33:46 -0800120 int fd, int64_t offset, int64_t length) {
121 Mutex::Autolock _l(mLock);
122 ALOGV("setDataSource %d/%lld/%lld", fd, (long long)offset, (long long)length);
123
124 resetDataSource();
125
126 mFd = dup(fd);
127 mOffset = offset;
128 mLength = length;
129
130 // delay data source creation to prepareAsync() to avoid blocking
131 // the calling thread in setDataSource for any significant time.
132 return OK;
133}
134
Wei Jia2409c872018-02-02 10:34:33 -0800135status_t NuPlayer2::GenericSource2::setDataSource(const sp<DataSource>& source) {
Wei Jia53692fa2017-12-11 10:33:46 -0800136 Mutex::Autolock _l(mLock);
137 ALOGV("setDataSource (source: %p)", source.get());
138
139 resetDataSource();
Robert Shih49fb89d2018-01-31 17:53:19 -0800140 AMediaDataSource *aSource = convertDataSourceToAMediaDataSource(source);
141 mDataSourceWrapper = new AMediaDataSourceWrapper(aSource);
Wei Jia53692fa2017-12-11 10:33:46 -0800142 return OK;
143}
144
Wei Jia2409c872018-02-02 10:34:33 -0800145sp<MetaData> NuPlayer2::GenericSource2::getFileFormatMeta() const {
Wei Jia53692fa2017-12-11 10:33:46 -0800146 Mutex::Autolock _l(mLock);
147 return mFileMeta;
148}
149
Wei Jia2409c872018-02-02 10:34:33 -0800150status_t NuPlayer2::GenericSource2::initFromDataSource() {
Robert Shih0cd95062018-01-21 17:41:16 -0800151 mExtractor = new AMediaExtractorWrapper(AMediaExtractor_new());
Robert Shih49fb89d2018-01-31 17:53:19 -0800152 CHECK(mFd >=0 || mDataSourceWrapper != NULL);
153 sp<AMediaDataSourceWrapper> aSourceWrapper = mDataSourceWrapper;
Robert Shih0cd95062018-01-21 17:41:16 -0800154 const int fd = mFd;
Wei Jia53692fa2017-12-11 10:33:46 -0800155
156 mLock.unlock();
157 // This might take long time if data source is not reliable.
Robert Shih0cd95062018-01-21 17:41:16 -0800158 status_t err;
Robert Shih49fb89d2018-01-31 17:53:19 -0800159 if (aSourceWrapper != NULL) {
160 err = mExtractor->setDataSource(aSourceWrapper->getAMediaDataSource());
Robert Shih0cd95062018-01-21 17:41:16 -0800161 } else {
Robert Shih49fb89d2018-01-31 17:53:19 -0800162 err = mExtractor->setDataSource(fd, mOffset, mLength);
Robert Shih0cd95062018-01-21 17:41:16 -0800163 }
Wei Jia53692fa2017-12-11 10:33:46 -0800164
Robert Shih0cd95062018-01-21 17:41:16 -0800165 if (err != OK) {
Robert Shih49fb89d2018-01-31 17:53:19 -0800166 ALOGE("initFromDataSource, failed to set extractor data source!");
Robert Shih0cd95062018-01-21 17:41:16 -0800167 mLock.lock();
Wei Jia53692fa2017-12-11 10:33:46 -0800168 return UNKNOWN_ERROR;
169 }
170
Robert Shih0cd95062018-01-21 17:41:16 -0800171 size_t numtracks = mExtractor->getTrackCount();
Wei Jia53692fa2017-12-11 10:33:46 -0800172 if (numtracks == 0) {
173 ALOGE("initFromDataSource, source has no track!");
Robert Shih0cd95062018-01-21 17:41:16 -0800174 mLock.lock();
Wei Jia53692fa2017-12-11 10:33:46 -0800175 return UNKNOWN_ERROR;
176 }
177
Robert Shih0cd95062018-01-21 17:41:16 -0800178 mFileMeta = convertMediaFormatWrapperToMetaData(mExtractor->getFormat());
Robert Shih49fb89d2018-01-31 17:53:19 -0800179 mLock.lock();
Wei Jia53692fa2017-12-11 10:33:46 -0800180 if (mFileMeta != NULL) {
181 int64_t duration;
182 if (mFileMeta->findInt64(kKeyDuration, &duration)) {
183 mDurationUs = duration;
184 }
185 }
186
187 int32_t totalBitrate = 0;
188
189 mMimes.clear();
190
191 for (size_t i = 0; i < numtracks; ++i) {
Wei Jia53692fa2017-12-11 10:33:46 -0800192
Robert Shih0cd95062018-01-21 17:41:16 -0800193 sp<AMediaFormatWrapper> trackFormat = mExtractor->getTrackFormat(i);
194 if (trackFormat == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800195 ALOGE("no metadata for track %zu", i);
196 return UNKNOWN_ERROR;
197 }
198
Robert Shih0cd95062018-01-21 17:41:16 -0800199 sp<AMediaExtractorWrapper> trackExtractor = new AMediaExtractorWrapper(AMediaExtractor_new());
Robert Shih3d2b89b2018-10-24 17:53:23 -0700200 if (aSourceWrapper != NULL) {
201 trackExtractor->setDataSource(aSourceWrapper->getAMediaDataSource());
202 } else {
203 trackExtractor->setDataSource(fd, mOffset, mLength);
204 }
Robert Shih0cd95062018-01-21 17:41:16 -0800205
Wei Jia53692fa2017-12-11 10:33:46 -0800206 const char *mime;
Robert Shih0cd95062018-01-21 17:41:16 -0800207 sp<MetaData> meta = convertMediaFormatWrapperToMetaData(trackFormat);
Wei Jia53692fa2017-12-11 10:33:46 -0800208 CHECK(meta->findCString(kKeyMIMEType, &mime));
209
210 ALOGV("initFromDataSource track[%zu]: %s", i, mime);
211
212 // Do the string compare immediately with "mime",
213 // we can't assume "mime" would stay valid after another
214 // extractor operation, some extractors might modify meta
215 // during getTrack() and make it invalid.
216 if (!strncasecmp(mime, "audio/", 6)) {
Robert Shih0cd95062018-01-21 17:41:16 -0800217 if (mAudioTrack.mExtractor == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800218 mAudioTrack.mIndex = i;
Robert Shih0cd95062018-01-21 17:41:16 -0800219 mAudioTrack.mExtractor = trackExtractor;
220 mAudioTrack.mExtractor->selectTrack(i);
221 mAudioTrack.mPackets = new AnotherPacketSource(meta);
Wei Jia53692fa2017-12-11 10:33:46 -0800222
223 if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_VORBIS)) {
224 mAudioIsVorbis = true;
225 } else {
226 mAudioIsVorbis = false;
227 }
228
229 mMimes.add(String8(mime));
230 }
231 } else if (!strncasecmp(mime, "video/", 6)) {
Robert Shih0cd95062018-01-21 17:41:16 -0800232 if (mVideoTrack.mExtractor == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800233 mVideoTrack.mIndex = i;
Robert Shih0cd95062018-01-21 17:41:16 -0800234 mVideoTrack.mExtractor = trackExtractor;
235 mVideoTrack.mExtractor->selectTrack(i);
236 mVideoTrack.mPackets = new AnotherPacketSource(meta);
Wei Jia53692fa2017-12-11 10:33:46 -0800237
238 // video always at the beginning
239 mMimes.insertAt(String8(mime), 0);
240 }
241 }
242
Robert Shih0cd95062018-01-21 17:41:16 -0800243 mExtractors.push(trackExtractor);
Wei Jia53692fa2017-12-11 10:33:46 -0800244 int64_t durationUs;
245 if (meta->findInt64(kKeyDuration, &durationUs)) {
246 if (durationUs > mDurationUs) {
247 mDurationUs = durationUs;
248 }
249 }
250
251 int32_t bitrate;
252 if (totalBitrate >= 0 && meta->findInt32(kKeyBitRate, &bitrate)) {
253 totalBitrate += bitrate;
254 } else {
255 totalBitrate = -1;
256 }
257 }
258
Robert Shih0cd95062018-01-21 17:41:16 -0800259 ALOGV("initFromDataSource mExtractors.size(): %zu mIsSecure: %d mime[0]: %s", mExtractors.size(),
Wei Jia53692fa2017-12-11 10:33:46 -0800260 mIsSecure, (mMimes.isEmpty() ? "NONE" : mMimes[0].string()));
261
Robert Shih0cd95062018-01-21 17:41:16 -0800262 if (mExtractors.size() == 0) {
Wei Jia53692fa2017-12-11 10:33:46 -0800263 ALOGE("b/23705695");
264 return UNKNOWN_ERROR;
265 }
266
267 // Modular DRM: The return value doesn't affect source initialization.
268 (void)checkDrmInfo();
269
270 mBitrate = totalBitrate;
271
272 return OK;
273}
274
Wei Jia2409c872018-02-02 10:34:33 -0800275status_t NuPlayer2::GenericSource2::getBufferingSettings(
Wei Jia53692fa2017-12-11 10:33:46 -0800276 BufferingSettings* buffering /* nonnull */) {
277 {
278 Mutex::Autolock _l(mLock);
279 *buffering = mBufferingSettings;
280 }
281
282 ALOGV("getBufferingSettings{%s}", buffering->toString().string());
283 return OK;
284}
285
Wei Jia2409c872018-02-02 10:34:33 -0800286status_t NuPlayer2::GenericSource2::setBufferingSettings(const BufferingSettings& buffering) {
Wei Jia53692fa2017-12-11 10:33:46 -0800287 ALOGV("setBufferingSettings{%s}", buffering.toString().string());
288
289 Mutex::Autolock _l(mLock);
290 mBufferingSettings = buffering;
291 return OK;
292}
293
Wei Jia2409c872018-02-02 10:34:33 -0800294int64_t NuPlayer2::GenericSource2::getLastReadPosition() {
Robert Shih0cd95062018-01-21 17:41:16 -0800295 if (mAudioTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800296 return mAudioTimeUs;
Robert Shih0cd95062018-01-21 17:41:16 -0800297 } else if (mVideoTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800298 return mVideoTimeUs;
299 } else {
300 return 0;
301 }
302}
303
Wei Jia2409c872018-02-02 10:34:33 -0800304bool NuPlayer2::GenericSource2::isStreaming() const {
Wei Jia53692fa2017-12-11 10:33:46 -0800305 Mutex::Autolock _l(mLock);
306 return mIsStreaming;
307}
308
Wei Jia2409c872018-02-02 10:34:33 -0800309NuPlayer2::GenericSource2::~GenericSource2() {
310 ALOGV("~GenericSource2");
Wei Jia53692fa2017-12-11 10:33:46 -0800311 if (mLooper != NULL) {
312 mLooper->unregisterHandler(id());
313 mLooper->stop();
314 }
Robert Shih49fb89d2018-01-31 17:53:19 -0800315 if (mDataSourceWrapper != NULL) {
316 mDataSourceWrapper->close();
Wei Jia17459332018-01-09 14:21:23 -0800317 }
Wei Jia53692fa2017-12-11 10:33:46 -0800318 resetDataSource();
319}
320
Wei Jiaf01e3122018-10-18 11:49:44 -0700321void NuPlayer2::GenericSource2::prepareAsync(int64_t startTimeUs) {
Wei Jia53692fa2017-12-11 10:33:46 -0800322 Mutex::Autolock _l(mLock);
323 ALOGV("prepareAsync: (looper: %d)", (mLooper != NULL));
324
325 if (mLooper == NULL) {
326 mLooper = new ALooper;
327 mLooper->setName("generic");
Wei Jiac5c79da2017-12-21 18:03:05 -0800328 mLooper->start(false, /* runOnCallingThread */
329 true, /* canCallJava */
330 PRIORITY_DEFAULT);
Wei Jia53692fa2017-12-11 10:33:46 -0800331
332 mLooper->registerHandler(this);
333 }
334
335 sp<AMessage> msg = new AMessage(kWhatPrepareAsync, this);
Wei Jiaf01e3122018-10-18 11:49:44 -0700336 msg->setInt64("startTimeUs", startTimeUs);
337
Wei Jia53692fa2017-12-11 10:33:46 -0800338 msg->post();
339}
340
Wei Jiaf01e3122018-10-18 11:49:44 -0700341void NuPlayer2::GenericSource2::onPrepareAsync(int64_t startTimeUs) {
Robert Shih49fb89d2018-01-31 17:53:19 -0800342 ALOGV("onPrepareAsync: mFd %d mUri %s mDataSourceWrapper: %p",
343 mFd, mUri.c_str(), mDataSourceWrapper.get());
Wei Jia53692fa2017-12-11 10:33:46 -0800344
Robert Shih49fb89d2018-01-31 17:53:19 -0800345 if (!mUri.empty()) {
346 const char* uri = mUri.c_str();
347 size_t numheaders = mUriHeaders.size();
348 const char **key_values = numheaders ? new const char *[numheaders * 2] : NULL;
349 for (size_t i = 0; i < numheaders; ++i) {
350 key_values[i * 2] = mUriHeaders.keyAt(i).c_str();
351 key_values[i * 2 + 1] = mUriHeaders.valueAt(i).c_str();
Wei Jia53692fa2017-12-11 10:33:46 -0800352 }
Robert Shih49fb89d2018-01-31 17:53:19 -0800353 mLock.unlock();
354 AMediaDataSource *aSource = AMediaDataSource_newUri(uri, numheaders, key_values);
355 mLock.lock();
356 mDataSourceWrapper = aSource ? new AMediaDataSourceWrapper(aSource) : NULL;
357 delete[] key_values;
358 // For cached streaming cases, we need to wait for enough
359 // buffering before reporting prepared.
360 mIsStreaming = !strncasecmp("http://", uri, 7) || !strncasecmp("https://", uri, 8);
Wei Jia53692fa2017-12-11 10:33:46 -0800361 }
362
Robert Shih49fb89d2018-01-31 17:53:19 -0800363 if (mDisconnected || (mFd < 0 && mDataSourceWrapper == NULL)) {
364 ALOGE("mDisconnected(%d) or Failed to create data source!", mDisconnected);
365 notifyPreparedAndCleanup(UNKNOWN_ERROR);
366 return;
Wei Jia53692fa2017-12-11 10:33:46 -0800367 }
368
Wei Jia53692fa2017-12-11 10:33:46 -0800369 // init extractor from data source
370 status_t err = initFromDataSource();
Robert Shih49fb89d2018-01-31 17:53:19 -0800371 if (mFd >= 0) {
372 close(mFd);
373 mFd = -1;
374 }
Wei Jia53692fa2017-12-11 10:33:46 -0800375
376 if (err != OK) {
377 ALOGE("Failed to init from data source!");
378 notifyPreparedAndCleanup(err);
379 return;
380 }
381
Robert Shih0cd95062018-01-21 17:41:16 -0800382 if (mVideoTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800383 sp<MetaData> meta = getFormatMeta_l(false /* audio */);
384 sp<AMessage> msg = new AMessage;
385 err = convertMetaDataToMessage(meta, &msg);
386 if(err != OK) {
387 notifyPreparedAndCleanup(err);
388 return;
389 }
390 notifyVideoSizeChanged(msg);
391 }
392
393 notifyFlagsChanged(
394 // FLAG_SECURE will be known if/when prepareDrm is called by the app
395 // FLAG_PROTECTED will be known if/when prepareDrm is called by the app
396 FLAG_CAN_PAUSE |
397 FLAG_CAN_SEEK_BACKWARD |
398 FLAG_CAN_SEEK_FORWARD |
399 FLAG_CAN_SEEK);
400
Wei Jiaf01e3122018-10-18 11:49:44 -0700401 doSeek(startTimeUs, MediaPlayer2SeekMode::SEEK_CLOSEST);
Wei Jia53692fa2017-12-11 10:33:46 -0800402 finishPrepareAsync();
403
404 ALOGV("onPrepareAsync: Done");
405}
406
Wei Jia2409c872018-02-02 10:34:33 -0800407void NuPlayer2::GenericSource2::finishPrepareAsync() {
Wei Jia53692fa2017-12-11 10:33:46 -0800408 ALOGV("finishPrepareAsync");
409
Wei Jia53692fa2017-12-11 10:33:46 -0800410 if (mIsStreaming) {
Wei Jia53692fa2017-12-11 10:33:46 -0800411 mPreparing = true;
Wei Jiaf01e3122018-10-18 11:49:44 -0700412 ++mPollBufferingGeneration;
Wei Jia53692fa2017-12-11 10:33:46 -0800413 schedulePollBuffering();
414 } else {
415 notifyPrepared();
416 }
417
Robert Shih0cd95062018-01-21 17:41:16 -0800418 if (mAudioTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800419 postReadBuffer(MEDIA_TRACK_TYPE_AUDIO);
420 }
421
Robert Shih0cd95062018-01-21 17:41:16 -0800422 if (mVideoTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800423 postReadBuffer(MEDIA_TRACK_TYPE_VIDEO);
424 }
425}
426
Wei Jia2409c872018-02-02 10:34:33 -0800427void NuPlayer2::GenericSource2::notifyPreparedAndCleanup(status_t err) {
Wei Jia53692fa2017-12-11 10:33:46 -0800428 if (err != OK) {
Robert Shih49fb89d2018-01-31 17:53:19 -0800429 mDataSourceWrapper.clear();
Wei Jia53692fa2017-12-11 10:33:46 -0800430
431 mBitrate = -1;
432 mPrevBufferPercentage = -1;
433 ++mPollBufferingGeneration;
434 }
435 notifyPrepared(err);
436}
437
Wei Jia2409c872018-02-02 10:34:33 -0800438void NuPlayer2::GenericSource2::start() {
Wei Jia53692fa2017-12-11 10:33:46 -0800439 Mutex::Autolock _l(mLock);
440 ALOGI("start");
441
Robert Shih0cd95062018-01-21 17:41:16 -0800442 if (mAudioTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800443 postReadBuffer(MEDIA_TRACK_TYPE_AUDIO);
444 }
445
Robert Shih0cd95062018-01-21 17:41:16 -0800446 if (mVideoTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800447 postReadBuffer(MEDIA_TRACK_TYPE_VIDEO);
448 }
449
450 mStarted = true;
451}
452
Wei Jia2409c872018-02-02 10:34:33 -0800453void NuPlayer2::GenericSource2::stop() {
Wei Jia53692fa2017-12-11 10:33:46 -0800454 Mutex::Autolock _l(mLock);
455 mStarted = false;
456}
457
Wei Jia2409c872018-02-02 10:34:33 -0800458void NuPlayer2::GenericSource2::pause() {
Wei Jia53692fa2017-12-11 10:33:46 -0800459 Mutex::Autolock _l(mLock);
460 mStarted = false;
461}
462
Wei Jia2409c872018-02-02 10:34:33 -0800463void NuPlayer2::GenericSource2::resume() {
Wei Jia53692fa2017-12-11 10:33:46 -0800464 Mutex::Autolock _l(mLock);
465 mStarted = true;
466}
467
Wei Jia2409c872018-02-02 10:34:33 -0800468void NuPlayer2::GenericSource2::disconnect() {
Wei Jia53692fa2017-12-11 10:33:46 -0800469 {
470 Mutex::Autolock _l(mLock);
Wei Jia53692fa2017-12-11 10:33:46 -0800471 mDisconnected = true;
472 }
Robert Shih49fb89d2018-01-31 17:53:19 -0800473 if (mDataSourceWrapper != NULL) {
474 mDataSourceWrapper->close();
Wei Jia53692fa2017-12-11 10:33:46 -0800475 }
476}
477
Wei Jia2409c872018-02-02 10:34:33 -0800478status_t NuPlayer2::GenericSource2::feedMoreTSData() {
Wei Jia53692fa2017-12-11 10:33:46 -0800479 return OK;
480}
481
Wei Jia2409c872018-02-02 10:34:33 -0800482void NuPlayer2::GenericSource2::onMessageReceived(const sp<AMessage> &msg) {
Wei Jia53692fa2017-12-11 10:33:46 -0800483 Mutex::Autolock _l(mLock);
484 switch (msg->what()) {
485 case kWhatPrepareAsync:
486 {
Wei Jiaf01e3122018-10-18 11:49:44 -0700487 int64_t startTimeUs;
488 CHECK(msg->findInt64("startTimeUs", &startTimeUs));
489 onPrepareAsync(startTimeUs);
Wei Jia53692fa2017-12-11 10:33:46 -0800490 break;
491 }
492 case kWhatFetchSubtitleData:
493 {
494 fetchTextData(kWhatSendSubtitleData, MEDIA_TRACK_TYPE_SUBTITLE,
495 mFetchSubtitleDataGeneration, mSubtitleTrack.mPackets, msg);
496 break;
497 }
498
499 case kWhatFetchTimedTextData:
500 {
501 fetchTextData(kWhatSendTimedTextData, MEDIA_TRACK_TYPE_TIMEDTEXT,
502 mFetchTimedTextDataGeneration, mTimedTextTrack.mPackets, msg);
503 break;
504 }
505
506 case kWhatSendSubtitleData:
507 {
508 sendTextData(kWhatSubtitleData, MEDIA_TRACK_TYPE_SUBTITLE,
509 mFetchSubtitleDataGeneration, mSubtitleTrack.mPackets, msg);
510 break;
511 }
512
513 case kWhatSendGlobalTimedTextData:
514 {
515 sendGlobalTextData(kWhatTimedTextData, mFetchTimedTextDataGeneration, msg);
516 break;
517 }
518 case kWhatSendTimedTextData:
519 {
520 sendTextData(kWhatTimedTextData, MEDIA_TRACK_TYPE_TIMEDTEXT,
521 mFetchTimedTextDataGeneration, mTimedTextTrack.mPackets, msg);
522 break;
523 }
524
525 case kWhatChangeAVSource:
526 {
527 int32_t trackIndex;
528 CHECK(msg->findInt32("trackIndex", &trackIndex));
Robert Shih0cd95062018-01-21 17:41:16 -0800529 const sp<AMediaExtractorWrapper> extractor = mExtractors.itemAt(trackIndex);
Wei Jia53692fa2017-12-11 10:33:46 -0800530
531 Track* track;
Robert Shih0cd95062018-01-21 17:41:16 -0800532 AString mime;
Wei Jia53692fa2017-12-11 10:33:46 -0800533 media_track_type trackType, counterpartType;
Robert Shih0cd95062018-01-21 17:41:16 -0800534 sp<AMediaFormatWrapper> format = extractor->getTrackFormat(trackIndex);
535 format->getString(AMEDIAFORMAT_KEY_MIME, &mime);
536 if (!strncasecmp(mime.c_str(), "audio/", 6)) {
Wei Jia53692fa2017-12-11 10:33:46 -0800537 track = &mAudioTrack;
538 trackType = MEDIA_TRACK_TYPE_AUDIO;
539 counterpartType = MEDIA_TRACK_TYPE_VIDEO;;
540 } else {
Robert Shih0cd95062018-01-21 17:41:16 -0800541 CHECK(!strncasecmp(mime.c_str(), "video/", 6));
Wei Jia53692fa2017-12-11 10:33:46 -0800542 track = &mVideoTrack;
543 trackType = MEDIA_TRACK_TYPE_VIDEO;
544 counterpartType = MEDIA_TRACK_TYPE_AUDIO;;
545 }
546
547
Robert Shih0cd95062018-01-21 17:41:16 -0800548 track->mExtractor = extractor;
549 track->mExtractor->selectSingleTrack(trackIndex);
Wei Jia53692fa2017-12-11 10:33:46 -0800550 track->mIndex = trackIndex;
551 ++mAudioDataGeneration;
552 ++mVideoDataGeneration;
553
554 int64_t timeUs, actualTimeUs;
555 const bool formatChange = true;
556 if (trackType == MEDIA_TRACK_TYPE_AUDIO) {
557 timeUs = mAudioLastDequeueTimeUs;
558 } else {
559 timeUs = mVideoLastDequeueTimeUs;
560 }
561 readBuffer(trackType, timeUs, MediaPlayer2SeekMode::SEEK_PREVIOUS_SYNC /* mode */,
562 &actualTimeUs, formatChange);
563 readBuffer(counterpartType, -1, MediaPlayer2SeekMode::SEEK_PREVIOUS_SYNC /* mode */,
564 NULL, !formatChange);
565 ALOGV("timeUs %lld actualTimeUs %lld", (long long)timeUs, (long long)actualTimeUs);
566
567 break;
568 }
569
570 case kWhatSeek:
571 {
572 onSeek(msg);
573 break;
574 }
575
576 case kWhatReadBuffer:
577 {
578 onReadBuffer(msg);
579 break;
580 }
581
582 case kWhatPollBuffering:
583 {
584 int32_t generation;
585 CHECK(msg->findInt32("generation", &generation));
586 if (generation == mPollBufferingGeneration) {
587 onPollBuffering();
588 }
589 break;
590 }
591
592 default:
593 Source::onMessageReceived(msg);
594 break;
595 }
596}
597
Wei Jia2409c872018-02-02 10:34:33 -0800598void NuPlayer2::GenericSource2::fetchTextData(
Wei Jia53692fa2017-12-11 10:33:46 -0800599 uint32_t sendWhat,
600 media_track_type type,
601 int32_t curGen,
602 const sp<AnotherPacketSource>& packets,
603 const sp<AMessage>& msg) {
604 int32_t msgGeneration;
605 CHECK(msg->findInt32("generation", &msgGeneration));
606 if (msgGeneration != curGen) {
607 // stale
608 return;
609 }
610
611 int32_t avail;
612 if (packets->hasBufferAvailable(&avail)) {
613 return;
614 }
615
616 int64_t timeUs;
617 CHECK(msg->findInt64("timeUs", &timeUs));
618
619 int64_t subTimeUs = 0;
620 readBuffer(type, timeUs, MediaPlayer2SeekMode::SEEK_PREVIOUS_SYNC /* mode */, &subTimeUs);
621
622 status_t eosResult;
623 if (!packets->hasBufferAvailable(&eosResult)) {
624 return;
625 }
626
627 if (msg->what() == kWhatFetchSubtitleData) {
628 subTimeUs -= 1000000ll; // send subtile data one second earlier
629 }
630 sp<AMessage> msg2 = new AMessage(sendWhat, this);
631 msg2->setInt32("generation", msgGeneration);
632 mMediaClock->addTimer(msg2, subTimeUs);
633}
634
Wei Jia2409c872018-02-02 10:34:33 -0800635void NuPlayer2::GenericSource2::sendTextData(
Wei Jia53692fa2017-12-11 10:33:46 -0800636 uint32_t what,
637 media_track_type type,
638 int32_t curGen,
639 const sp<AnotherPacketSource>& packets,
640 const sp<AMessage>& msg) {
641 int32_t msgGeneration;
642 CHECK(msg->findInt32("generation", &msgGeneration));
643 if (msgGeneration != curGen) {
644 // stale
645 return;
646 }
647
648 int64_t subTimeUs;
649 if (packets->nextBufferTime(&subTimeUs) != OK) {
650 return;
651 }
652
653 int64_t nextSubTimeUs;
654 readBuffer(type, -1, MediaPlayer2SeekMode::SEEK_PREVIOUS_SYNC /* mode */, &nextSubTimeUs);
655
656 sp<ABuffer> buffer;
657 status_t dequeueStatus = packets->dequeueAccessUnit(&buffer);
658 if (dequeueStatus == OK) {
659 sp<AMessage> notify = dupNotify();
660 notify->setInt32("what", what);
661 notify->setBuffer("buffer", buffer);
662 notify->post();
663
664 if (msg->what() == kWhatSendSubtitleData) {
665 nextSubTimeUs -= 1000000ll; // send subtile data one second earlier
666 }
667 mMediaClock->addTimer(msg, nextSubTimeUs);
668 }
669}
670
Wei Jia2409c872018-02-02 10:34:33 -0800671void NuPlayer2::GenericSource2::sendGlobalTextData(
Wei Jia53692fa2017-12-11 10:33:46 -0800672 uint32_t what,
673 int32_t curGen,
674 sp<AMessage> msg) {
675 int32_t msgGeneration;
676 CHECK(msg->findInt32("generation", &msgGeneration));
677 if (msgGeneration != curGen) {
678 // stale
679 return;
680 }
681
Robert Shih0cd95062018-01-21 17:41:16 -0800682 void *data = NULL;
Wei Jia53692fa2017-12-11 10:33:46 -0800683 size_t size = 0;
Robert Shih0cd95062018-01-21 17:41:16 -0800684 if (mTimedTextTrack.mExtractor->getTrackFormat(mTimedTextTrack.mIndex)->getBuffer(
685 "text", &data, &size)) {
Wei Jia53692fa2017-12-11 10:33:46 -0800686 mGlobalTimedText = new ABuffer(size);
687 if (mGlobalTimedText->data()) {
688 memcpy(mGlobalTimedText->data(), data, size);
689 sp<AMessage> globalMeta = mGlobalTimedText->meta();
690 globalMeta->setInt64("timeUs", 0);
691 globalMeta->setString("mime", MEDIA_MIMETYPE_TEXT_3GPP);
692 globalMeta->setInt32("global", 1);
693 sp<AMessage> notify = dupNotify();
694 notify->setInt32("what", what);
695 notify->setBuffer("buffer", mGlobalTimedText);
696 notify->post();
697 }
698 }
699}
700
Robert Shih0cd95062018-01-21 17:41:16 -0800701sp<AMessage> NuPlayer2::GenericSource2::getFormat(bool audio) {
702 Mutex::Autolock _l(mLock);
703 return getFormat_l(audio);
704}
705
Wei Jia2409c872018-02-02 10:34:33 -0800706sp<MetaData> NuPlayer2::GenericSource2::getFormatMeta(bool audio) {
Wei Jia53692fa2017-12-11 10:33:46 -0800707 Mutex::Autolock _l(mLock);
708 return getFormatMeta_l(audio);
709}
710
Robert Shih0cd95062018-01-21 17:41:16 -0800711sp<AMessage> NuPlayer2::GenericSource2::getFormat_l(bool audio) {
712 sp<AMediaExtractorWrapper> extractor = audio ? mAudioTrack.mExtractor : mVideoTrack.mExtractor;
713 size_t trackIndex = audio ? mAudioTrack.mIndex : mVideoTrack.mIndex;
Wei Jia53692fa2017-12-11 10:33:46 -0800714
Robert Shih0cd95062018-01-21 17:41:16 -0800715 if (extractor == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800716 return NULL;
717 }
718
Robert Shih0cd95062018-01-21 17:41:16 -0800719 return extractor->getTrackFormat(trackIndex)->toAMessage();
720}
721
722sp<MetaData> NuPlayer2::GenericSource2::getFormatMeta_l(bool audio) {
723 sp<AMediaExtractorWrapper> extractor = audio ? mAudioTrack.mExtractor : mVideoTrack.mExtractor;
724 size_t trackIndex = audio ? mAudioTrack.mIndex : mVideoTrack.mIndex;
725
726 if (extractor == NULL) {
727 return NULL;
728 }
729
730 return convertMediaFormatWrapperToMetaData(extractor->getTrackFormat(trackIndex));
Wei Jia53692fa2017-12-11 10:33:46 -0800731}
732
Wei Jia2409c872018-02-02 10:34:33 -0800733status_t NuPlayer2::GenericSource2::dequeueAccessUnit(
Wei Jia53692fa2017-12-11 10:33:46 -0800734 bool audio, sp<ABuffer> *accessUnit) {
735 Mutex::Autolock _l(mLock);
736 // If has gone through stop/releaseDrm sequence, we no longer send down any buffer b/c
737 // the codec's crypto object has gone away (b/37960096).
738 // Note: This will be unnecessary when stop() changes behavior and releases codec (b/35248283).
739 if (!mStarted && mIsDrmReleased) {
740 return -EWOULDBLOCK;
741 }
742
743 Track *track = audio ? &mAudioTrack : &mVideoTrack;
744
Robert Shih0cd95062018-01-21 17:41:16 -0800745 if (track->mExtractor == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800746 return -EWOULDBLOCK;
747 }
748
749 status_t finalResult;
750 if (!track->mPackets->hasBufferAvailable(&finalResult)) {
751 if (finalResult == OK) {
752 postReadBuffer(
753 audio ? MEDIA_TRACK_TYPE_AUDIO : MEDIA_TRACK_TYPE_VIDEO);
754 return -EWOULDBLOCK;
755 }
756 return finalResult;
757 }
758
759 status_t result = track->mPackets->dequeueAccessUnit(accessUnit);
760
761 // start pulling in more buffers if cache is running low
762 // so that decoder has less chance of being starved
763 if (!mIsStreaming) {
764 if (track->mPackets->getAvailableBufferCount(&finalResult) < 2) {
765 postReadBuffer(audio? MEDIA_TRACK_TYPE_AUDIO : MEDIA_TRACK_TYPE_VIDEO);
766 }
767 } else {
768 int64_t durationUs = track->mPackets->getBufferedDurationUs(&finalResult);
769 // TODO: maxRebufferingMarkMs could be larger than
770 // mBufferingSettings.mResumePlaybackMarkMs
771 int64_t restartBufferingMarkUs =
772 mBufferingSettings.mResumePlaybackMarkMs * 1000ll / 2;
773 if (finalResult == OK) {
774 if (durationUs < restartBufferingMarkUs) {
775 postReadBuffer(audio? MEDIA_TRACK_TYPE_AUDIO : MEDIA_TRACK_TYPE_VIDEO);
776 }
777 if (track->mPackets->getAvailableBufferCount(&finalResult) < 2
778 && !mSentPauseOnBuffering && !mPreparing) {
Wei Jia53692fa2017-12-11 10:33:46 -0800779 mSentPauseOnBuffering = true;
780 sp<AMessage> notify = dupNotify();
781 notify->setInt32("what", kWhatPauseOnBufferingStart);
782 notify->post();
783 }
784 }
785 }
786
787 if (result != OK) {
Robert Shih0cd95062018-01-21 17:41:16 -0800788 if (mSubtitleTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800789 mSubtitleTrack.mPackets->clear();
790 mFetchSubtitleDataGeneration++;
791 }
Robert Shih0cd95062018-01-21 17:41:16 -0800792 if (mTimedTextTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800793 mTimedTextTrack.mPackets->clear();
794 mFetchTimedTextDataGeneration++;
795 }
796 return result;
797 }
798
799 int64_t timeUs;
800 status_t eosResult; // ignored
801 CHECK((*accessUnit)->meta()->findInt64("timeUs", &timeUs));
802 if (audio) {
803 mAudioLastDequeueTimeUs = timeUs;
804 } else {
805 mVideoLastDequeueTimeUs = timeUs;
806 }
807
Robert Shih0cd95062018-01-21 17:41:16 -0800808 if (mSubtitleTrack.mExtractor != NULL
Wei Jia53692fa2017-12-11 10:33:46 -0800809 && !mSubtitleTrack.mPackets->hasBufferAvailable(&eosResult)) {
810 sp<AMessage> msg = new AMessage(kWhatFetchSubtitleData, this);
811 msg->setInt64("timeUs", timeUs);
812 msg->setInt32("generation", mFetchSubtitleDataGeneration);
813 msg->post();
814 }
815
Robert Shih0cd95062018-01-21 17:41:16 -0800816 if (mTimedTextTrack.mExtractor != NULL
Wei Jia53692fa2017-12-11 10:33:46 -0800817 && !mTimedTextTrack.mPackets->hasBufferAvailable(&eosResult)) {
818 sp<AMessage> msg = new AMessage(kWhatFetchTimedTextData, this);
819 msg->setInt64("timeUs", timeUs);
820 msg->setInt32("generation", mFetchTimedTextDataGeneration);
821 msg->post();
822 }
823
824 return result;
825}
826
Wei Jia2409c872018-02-02 10:34:33 -0800827status_t NuPlayer2::GenericSource2::getDuration(int64_t *durationUs) {
Wei Jia53692fa2017-12-11 10:33:46 -0800828 Mutex::Autolock _l(mLock);
829 *durationUs = mDurationUs;
830 return OK;
831}
832
Wei Jia2409c872018-02-02 10:34:33 -0800833size_t NuPlayer2::GenericSource2::getTrackCount() const {
Wei Jia53692fa2017-12-11 10:33:46 -0800834 Mutex::Autolock _l(mLock);
Robert Shih0cd95062018-01-21 17:41:16 -0800835 return mExtractors.size();
Wei Jia53692fa2017-12-11 10:33:46 -0800836}
837
Wei Jia2409c872018-02-02 10:34:33 -0800838sp<AMessage> NuPlayer2::GenericSource2::getTrackInfo(size_t trackIndex) const {
Wei Jia53692fa2017-12-11 10:33:46 -0800839 Mutex::Autolock _l(mLock);
Robert Shih0cd95062018-01-21 17:41:16 -0800840 size_t trackCount = mExtractors.size();
Wei Jia53692fa2017-12-11 10:33:46 -0800841 if (trackIndex >= trackCount) {
842 return NULL;
843 }
844
Robert Shih0cd95062018-01-21 17:41:16 -0800845 sp<AMessage> format = mExtractors.itemAt(trackIndex)->getTrackFormat(trackIndex)->toAMessage();
846 if (format == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800847 ALOGE("no metadata for track %zu", trackIndex);
848 return NULL;
849 }
850
Robert Shih0cd95062018-01-21 17:41:16 -0800851 AString mime;
852 CHECK(format->findString(AMEDIAFORMAT_KEY_MIME, &mime));
Wei Jia53692fa2017-12-11 10:33:46 -0800853
854 int32_t trackType;
Robert Shih0cd95062018-01-21 17:41:16 -0800855 if (!strncasecmp(mime.c_str(), "video/", 6)) {
Wei Jia53692fa2017-12-11 10:33:46 -0800856 trackType = MEDIA_TRACK_TYPE_VIDEO;
Robert Shih0cd95062018-01-21 17:41:16 -0800857 } else if (!strncasecmp(mime.c_str(), "audio/", 6)) {
Wei Jia53692fa2017-12-11 10:33:46 -0800858 trackType = MEDIA_TRACK_TYPE_AUDIO;
Robert Shih0cd95062018-01-21 17:41:16 -0800859 } else if (!strcasecmp(mime.c_str(), MEDIA_MIMETYPE_TEXT_3GPP)) {
Wei Jia53692fa2017-12-11 10:33:46 -0800860 trackType = MEDIA_TRACK_TYPE_TIMEDTEXT;
861 } else {
862 trackType = MEDIA_TRACK_TYPE_UNKNOWN;
863 }
864 format->setInt32("type", trackType);
865
Robert Shih0cd95062018-01-21 17:41:16 -0800866 AString lang;
867 if (!format->findString("language", &lang)) {
868 format->setString("language", "und");
Wei Jia53692fa2017-12-11 10:33:46 -0800869 }
Wei Jia53692fa2017-12-11 10:33:46 -0800870
871 if (trackType == MEDIA_TRACK_TYPE_SUBTITLE) {
872 int32_t isAutoselect = 1, isDefault = 0, isForced = 0;
Robert Shih0cd95062018-01-21 17:41:16 -0800873 format->findInt32(AMEDIAFORMAT_KEY_IS_AUTOSELECT, &isAutoselect);
874 format->findInt32(AMEDIAFORMAT_KEY_IS_DEFAULT, &isDefault);
875 format->findInt32(AMEDIAFORMAT_KEY_IS_FORCED_SUBTITLE, &isForced);
Wei Jia53692fa2017-12-11 10:33:46 -0800876
877 format->setInt32("auto", !!isAutoselect);
878 format->setInt32("default", !!isDefault);
879 format->setInt32("forced", !!isForced);
880 }
881
882 return format;
883}
884
Wei Jia2409c872018-02-02 10:34:33 -0800885ssize_t NuPlayer2::GenericSource2::getSelectedTrack(media_track_type type) const {
Wei Jia53692fa2017-12-11 10:33:46 -0800886 Mutex::Autolock _l(mLock);
887 const Track *track = NULL;
888 switch (type) {
889 case MEDIA_TRACK_TYPE_VIDEO:
890 track = &mVideoTrack;
891 break;
892 case MEDIA_TRACK_TYPE_AUDIO:
893 track = &mAudioTrack;
894 break;
895 case MEDIA_TRACK_TYPE_TIMEDTEXT:
896 track = &mTimedTextTrack;
897 break;
898 case MEDIA_TRACK_TYPE_SUBTITLE:
899 track = &mSubtitleTrack;
900 break;
901 default:
902 break;
903 }
904
Robert Shih0cd95062018-01-21 17:41:16 -0800905 if (track != NULL && track->mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -0800906 return track->mIndex;
907 }
908
909 return -1;
910}
911
Wei Jia2409c872018-02-02 10:34:33 -0800912status_t NuPlayer2::GenericSource2::selectTrack(size_t trackIndex, bool select, int64_t timeUs) {
Wei Jia53692fa2017-12-11 10:33:46 -0800913 Mutex::Autolock _l(mLock);
914 ALOGV("%s track: %zu", select ? "select" : "deselect", trackIndex);
915
Robert Shih0cd95062018-01-21 17:41:16 -0800916 if (trackIndex >= mExtractors.size()) {
Wei Jia53692fa2017-12-11 10:33:46 -0800917 return BAD_INDEX;
918 }
919
920 if (!select) {
921 Track* track = NULL;
Robert Shih0cd95062018-01-21 17:41:16 -0800922 if (mSubtitleTrack.mExtractor != NULL && trackIndex == mSubtitleTrack.mIndex) {
Wei Jia53692fa2017-12-11 10:33:46 -0800923 track = &mSubtitleTrack;
924 mFetchSubtitleDataGeneration++;
Robert Shih0cd95062018-01-21 17:41:16 -0800925 } else if (mTimedTextTrack.mExtractor != NULL && trackIndex == mTimedTextTrack.mIndex) {
Wei Jia53692fa2017-12-11 10:33:46 -0800926 track = &mTimedTextTrack;
927 mFetchTimedTextDataGeneration++;
928 }
929 if (track == NULL) {
930 return INVALID_OPERATION;
931 }
Robert Shih0cd95062018-01-21 17:41:16 -0800932 track->mExtractor = NULL;
Wei Jia53692fa2017-12-11 10:33:46 -0800933 track->mPackets->clear();
934 return OK;
935 }
936
Robert Shih0cd95062018-01-21 17:41:16 -0800937 const sp<AMediaExtractorWrapper> extractor = mExtractors.itemAt(trackIndex);
938 sp<MetaData> meta = convertMediaFormatWrapperToMetaData(extractor->getTrackFormat(trackIndex));
Wei Jia53692fa2017-12-11 10:33:46 -0800939 const char *mime;
940 CHECK(meta->findCString(kKeyMIMEType, &mime));
941 if (!strncasecmp(mime, "text/", 5)) {
942 bool isSubtitle = strcasecmp(mime, MEDIA_MIMETYPE_TEXT_3GPP);
943 Track *track = isSubtitle ? &mSubtitleTrack : &mTimedTextTrack;
Robert Shih0cd95062018-01-21 17:41:16 -0800944 if (track->mExtractor != NULL && track->mIndex == trackIndex) {
Wei Jia53692fa2017-12-11 10:33:46 -0800945 return OK;
946 }
947 track->mIndex = trackIndex;
Robert Shih0cd95062018-01-21 17:41:16 -0800948 track->mExtractor = mExtractors.itemAt(trackIndex);
949 track->mExtractor->selectSingleTrack(trackIndex);
Wei Jia53692fa2017-12-11 10:33:46 -0800950 if (track->mPackets == NULL) {
Robert Shih0cd95062018-01-21 17:41:16 -0800951 track->mPackets = new AnotherPacketSource(meta);
Wei Jia53692fa2017-12-11 10:33:46 -0800952 } else {
953 track->mPackets->clear();
Robert Shih0cd95062018-01-21 17:41:16 -0800954 track->mPackets->setFormat(meta);
Wei Jia53692fa2017-12-11 10:33:46 -0800955
956 }
957
958 if (isSubtitle) {
959 mFetchSubtitleDataGeneration++;
960 } else {
961 mFetchTimedTextDataGeneration++;
962 }
963
964 status_t eosResult; // ignored
Robert Shih0cd95062018-01-21 17:41:16 -0800965 if (mSubtitleTrack.mExtractor != NULL
Wei Jia53692fa2017-12-11 10:33:46 -0800966 && !mSubtitleTrack.mPackets->hasBufferAvailable(&eosResult)) {
967 sp<AMessage> msg = new AMessage(kWhatFetchSubtitleData, this);
968 msg->setInt64("timeUs", timeUs);
969 msg->setInt32("generation", mFetchSubtitleDataGeneration);
970 msg->post();
971 }
972
973 sp<AMessage> msg2 = new AMessage(kWhatSendGlobalTimedTextData, this);
974 msg2->setInt32("generation", mFetchTimedTextDataGeneration);
975 msg2->post();
976
Robert Shih0cd95062018-01-21 17:41:16 -0800977 if (mTimedTextTrack.mExtractor != NULL
Wei Jia53692fa2017-12-11 10:33:46 -0800978 && !mTimedTextTrack.mPackets->hasBufferAvailable(&eosResult)) {
979 sp<AMessage> msg = new AMessage(kWhatFetchTimedTextData, this);
980 msg->setInt64("timeUs", timeUs);
981 msg->setInt32("generation", mFetchTimedTextDataGeneration);
982 msg->post();
983 }
984
985 return OK;
986 } else if (!strncasecmp(mime, "audio/", 6) || !strncasecmp(mime, "video/", 6)) {
987 bool audio = !strncasecmp(mime, "audio/", 6);
988 Track *track = audio ? &mAudioTrack : &mVideoTrack;
Robert Shih0cd95062018-01-21 17:41:16 -0800989 if (track->mExtractor != NULL && track->mIndex == trackIndex) {
Wei Jia53692fa2017-12-11 10:33:46 -0800990 return OK;
991 }
992
993 sp<AMessage> msg = new AMessage(kWhatChangeAVSource, this);
994 msg->setInt32("trackIndex", trackIndex);
995 msg->post();
996 return OK;
997 }
998
999 return INVALID_OPERATION;
1000}
1001
Wei Jia2409c872018-02-02 10:34:33 -08001002status_t NuPlayer2::GenericSource2::seekTo(int64_t seekTimeUs, MediaPlayer2SeekMode mode) {
Wei Jia53692fa2017-12-11 10:33:46 -08001003 ALOGV("seekTo: %lld, %d", (long long)seekTimeUs, mode);
1004 sp<AMessage> msg = new AMessage(kWhatSeek, this);
1005 msg->setInt64("seekTimeUs", seekTimeUs);
1006 msg->setInt32("mode", mode);
1007
1008 // Need to call readBuffer on |mLooper| to ensure the calls to
1009 // IMediaSource::read* are serialized. Note that IMediaSource::read*
1010 // is called without |mLock| acquired and MediaSource is not thread safe.
1011 sp<AMessage> response;
1012 status_t err = msg->postAndAwaitResponse(&response);
1013 if (err == OK && response != NULL) {
1014 CHECK(response->findInt32("err", &err));
1015 }
1016
1017 return err;
1018}
1019
Wei Jia2409c872018-02-02 10:34:33 -08001020void NuPlayer2::GenericSource2::onSeek(const sp<AMessage>& msg) {
Wei Jia53692fa2017-12-11 10:33:46 -08001021 int64_t seekTimeUs;
1022 int32_t mode;
1023 CHECK(msg->findInt64("seekTimeUs", &seekTimeUs));
1024 CHECK(msg->findInt32("mode", &mode));
1025
1026 sp<AMessage> response = new AMessage;
1027 status_t err = doSeek(seekTimeUs, (MediaPlayer2SeekMode)mode);
1028 response->setInt32("err", err);
1029
1030 sp<AReplyToken> replyID;
1031 CHECK(msg->senderAwaitsResponse(&replyID));
1032 response->postReply(replyID);
1033}
1034
Wei Jia2409c872018-02-02 10:34:33 -08001035status_t NuPlayer2::GenericSource2::doSeek(int64_t seekTimeUs, MediaPlayer2SeekMode mode) {
Robert Shih0cd95062018-01-21 17:41:16 -08001036 if (mVideoTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001037 ++mVideoDataGeneration;
1038
1039 int64_t actualTimeUs;
1040 readBuffer(MEDIA_TRACK_TYPE_VIDEO, seekTimeUs, mode, &actualTimeUs);
1041
1042 if (mode != MediaPlayer2SeekMode::SEEK_CLOSEST) {
1043 seekTimeUs = actualTimeUs;
1044 }
1045 mVideoLastDequeueTimeUs = actualTimeUs;
1046 }
1047
Robert Shih0cd95062018-01-21 17:41:16 -08001048 if (mAudioTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001049 ++mAudioDataGeneration;
1050 readBuffer(MEDIA_TRACK_TYPE_AUDIO, seekTimeUs, MediaPlayer2SeekMode::SEEK_CLOSEST);
1051 mAudioLastDequeueTimeUs = seekTimeUs;
1052 }
1053
Robert Shih0cd95062018-01-21 17:41:16 -08001054 if (mSubtitleTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001055 mSubtitleTrack.mPackets->clear();
1056 mFetchSubtitleDataGeneration++;
1057 }
1058
Robert Shih0cd95062018-01-21 17:41:16 -08001059 if (mTimedTextTrack.mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001060 mTimedTextTrack.mPackets->clear();
1061 mFetchTimedTextDataGeneration++;
1062 }
1063
1064 ++mPollBufferingGeneration;
1065 schedulePollBuffering();
1066 return OK;
1067}
1068
Wei Jia2409c872018-02-02 10:34:33 -08001069sp<ABuffer> NuPlayer2::GenericSource2::mediaBufferToABuffer(
Dongwon Kang1889c3e2018-02-01 13:44:57 -08001070 MediaBufferBase* mb,
Wei Jia53692fa2017-12-11 10:33:46 -08001071 media_track_type trackType) {
1072 bool audio = trackType == MEDIA_TRACK_TYPE_AUDIO;
1073 size_t outLength = mb->range_length();
1074
1075 if (audio && mAudioIsVorbis) {
1076 outLength += sizeof(int32_t);
1077 }
1078
1079 sp<ABuffer> ab;
1080
1081 if (mIsDrmProtected) {
1082 // Modular DRM
1083 // Enabled for both video/audio so 1) media buffer is reused without extra copying
1084 // 2) meta data can be retrieved in onInputBufferFetched for calling queueSecureInputBuffer.
1085
1086 // data is already provided in the buffer
1087 ab = new ABuffer(NULL, mb->range_length());
Dongwon Kangbc8f53b2018-01-25 17:01:44 -08001088 ab->meta()->setObject("mediaBufferHolder", new MediaBufferHolder(mb));
Wei Jia53692fa2017-12-11 10:33:46 -08001089
1090 // Modular DRM: Required b/c of the above add_ref.
1091 // If ref>0, there must be an observer, or it'll crash at release().
1092 // TODO: MediaBuffer might need to be revised to ease such need.
1093 mb->setObserver(this);
1094 // setMediaBufferBase() interestingly doesn't increment the ref count on its own.
1095 // Extra increment (since we want to keep mb alive and attached to ab beyond this function
1096 // call. This is to counter the effect of mb->release() towards the end.
1097 mb->add_ref();
1098
1099 } else {
1100 ab = new ABuffer(outLength);
1101 memcpy(ab->data(),
1102 (const uint8_t *)mb->data() + mb->range_offset(),
1103 mb->range_length());
1104 }
1105
1106 if (audio && mAudioIsVorbis) {
1107 int32_t numPageSamples;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001108 if (!mb->meta_data().findInt32(kKeyValidSamples, &numPageSamples)) {
Wei Jia53692fa2017-12-11 10:33:46 -08001109 numPageSamples = -1;
1110 }
1111
1112 uint8_t* abEnd = ab->data() + mb->range_length();
1113 memcpy(abEnd, &numPageSamples, sizeof(numPageSamples));
1114 }
1115
1116 sp<AMessage> meta = ab->meta();
1117
1118 int64_t timeUs;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001119 CHECK(mb->meta_data().findInt64(kKeyTime, &timeUs));
Wei Jia53692fa2017-12-11 10:33:46 -08001120 meta->setInt64("timeUs", timeUs);
1121
1122 if (trackType == MEDIA_TRACK_TYPE_VIDEO) {
1123 int32_t layerId;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001124 if (mb->meta_data().findInt32(kKeyTemporalLayerId, &layerId)) {
Wei Jia53692fa2017-12-11 10:33:46 -08001125 meta->setInt32("temporal-layer-id", layerId);
1126 }
1127 }
1128
1129 if (trackType == MEDIA_TRACK_TYPE_TIMEDTEXT) {
Robert Shih0cd95062018-01-21 17:41:16 -08001130 AString mime;
1131 sp<AMediaExtractorWrapper> extractor = mTimedTextTrack.mExtractor;
1132 size_t trackIndex = mTimedTextTrack.mIndex;
1133 CHECK(extractor != NULL
1134 && extractor->getTrackFormat(trackIndex)->getString(AMEDIAFORMAT_KEY_MIME, &mime));
1135 meta->setString("mime", mime.c_str());
Wei Jia53692fa2017-12-11 10:33:46 -08001136 }
1137
1138 int64_t durationUs;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001139 if (mb->meta_data().findInt64(kKeyDuration, &durationUs)) {
Wei Jia53692fa2017-12-11 10:33:46 -08001140 meta->setInt64("durationUs", durationUs);
1141 }
1142
1143 if (trackType == MEDIA_TRACK_TYPE_SUBTITLE) {
Robert Shihd83d4f42018-02-24 19:02:46 -08001144 meta->setInt32(AMEDIAFORMAT_KEY_TRACK_INDEX, mSubtitleTrack.mIndex);
Wei Jia53692fa2017-12-11 10:33:46 -08001145 }
1146
1147 uint32_t dataType; // unused
1148 const void *seiData;
1149 size_t seiLength;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001150 if (mb->meta_data().findData(kKeySEI, &dataType, &seiData, &seiLength)) {
Wei Jia53692fa2017-12-11 10:33:46 -08001151 sp<ABuffer> sei = ABuffer::CreateAsCopy(seiData, seiLength);;
1152 meta->setBuffer("sei", sei);
1153 }
1154
1155 const void *mpegUserDataPointer;
1156 size_t mpegUserDataLength;
Marco Nelissen3d21ae32018-02-16 08:24:08 -08001157 if (mb->meta_data().findData(
Wei Jia53692fa2017-12-11 10:33:46 -08001158 kKeyMpegUserData, &dataType, &mpegUserDataPointer, &mpegUserDataLength)) {
1159 sp<ABuffer> mpegUserData = ABuffer::CreateAsCopy(mpegUserDataPointer, mpegUserDataLength);
Robert Shihd83d4f42018-02-24 19:02:46 -08001160 meta->setBuffer(AMEDIAFORMAT_KEY_MPEG_USER_DATA, mpegUserData);
Wei Jia53692fa2017-12-11 10:33:46 -08001161 }
1162
1163 mb->release();
1164 mb = NULL;
1165
1166 return ab;
1167}
1168
Wei Jia2409c872018-02-02 10:34:33 -08001169int32_t NuPlayer2::GenericSource2::getDataGeneration(media_track_type type) const {
Wei Jia53692fa2017-12-11 10:33:46 -08001170 int32_t generation = -1;
1171 switch (type) {
1172 case MEDIA_TRACK_TYPE_VIDEO:
1173 generation = mVideoDataGeneration;
1174 break;
1175 case MEDIA_TRACK_TYPE_AUDIO:
1176 generation = mAudioDataGeneration;
1177 break;
1178 case MEDIA_TRACK_TYPE_TIMEDTEXT:
1179 generation = mFetchTimedTextDataGeneration;
1180 break;
1181 case MEDIA_TRACK_TYPE_SUBTITLE:
1182 generation = mFetchSubtitleDataGeneration;
1183 break;
1184 default:
1185 break;
1186 }
1187
1188 return generation;
1189}
1190
Wei Jia2409c872018-02-02 10:34:33 -08001191void NuPlayer2::GenericSource2::postReadBuffer(media_track_type trackType) {
Wei Jia53692fa2017-12-11 10:33:46 -08001192 if ((mPendingReadBufferTypes & (1 << trackType)) == 0) {
1193 mPendingReadBufferTypes |= (1 << trackType);
1194 sp<AMessage> msg = new AMessage(kWhatReadBuffer, this);
1195 msg->setInt32("trackType", trackType);
1196 msg->post();
1197 }
1198}
1199
Wei Jia2409c872018-02-02 10:34:33 -08001200void NuPlayer2::GenericSource2::onReadBuffer(const sp<AMessage>& msg) {
Wei Jia53692fa2017-12-11 10:33:46 -08001201 int32_t tmpType;
1202 CHECK(msg->findInt32("trackType", &tmpType));
1203 media_track_type trackType = (media_track_type)tmpType;
1204 mPendingReadBufferTypes &= ~(1 << trackType);
1205 readBuffer(trackType);
1206}
1207
Wei Jia2409c872018-02-02 10:34:33 -08001208void NuPlayer2::GenericSource2::readBuffer(
Wei Jia53692fa2017-12-11 10:33:46 -08001209 media_track_type trackType, int64_t seekTimeUs, MediaPlayer2SeekMode mode,
1210 int64_t *actualTimeUs, bool formatChange) {
1211 Track *track;
1212 size_t maxBuffers = 1;
1213 switch (trackType) {
1214 case MEDIA_TRACK_TYPE_VIDEO:
1215 track = &mVideoTrack;
1216 maxBuffers = 8; // too large of a number may influence seeks
1217 break;
1218 case MEDIA_TRACK_TYPE_AUDIO:
1219 track = &mAudioTrack;
1220 maxBuffers = 64;
1221 break;
1222 case MEDIA_TRACK_TYPE_SUBTITLE:
1223 track = &mSubtitleTrack;
1224 break;
1225 case MEDIA_TRACK_TYPE_TIMEDTEXT:
1226 track = &mTimedTextTrack;
1227 break;
1228 default:
1229 TRESPASS();
1230 }
1231
Robert Shih0cd95062018-01-21 17:41:16 -08001232 if (track->mExtractor == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001233 return;
1234 }
1235
1236 if (actualTimeUs) {
1237 *actualTimeUs = seekTimeUs;
1238 }
1239
Wei Jia53692fa2017-12-11 10:33:46 -08001240
1241 bool seeking = false;
Robert Shih0cd95062018-01-21 17:41:16 -08001242 sp<AMediaExtractorWrapper> extractor = track->mExtractor;
Wei Jia53692fa2017-12-11 10:33:46 -08001243 if (seekTimeUs >= 0) {
Robert Shih0cd95062018-01-21 17:41:16 -08001244 extractor->seekTo(seekTimeUs, mode);
Wei Jia53692fa2017-12-11 10:33:46 -08001245 seeking = true;
1246 }
1247
Wei Jia53692fa2017-12-11 10:33:46 -08001248 int32_t generation = getDataGeneration(trackType);
1249 for (size_t numBuffers = 0; numBuffers < maxBuffers; ) {
Robert Shih0cd95062018-01-21 17:41:16 -08001250 Vector<sp<ABuffer> > aBuffers;
Wei Jia53692fa2017-12-11 10:33:46 -08001251
Wei Jia53692fa2017-12-11 10:33:46 -08001252 mLock.unlock();
Robert Shih0cd95062018-01-21 17:41:16 -08001253
1254 sp<AMediaFormatWrapper> format;
1255 ssize_t sampleSize = -1;
1256 status_t err = extractor->getSampleFormat(format);
1257 if (err == OK) {
1258 sampleSize = extractor->getSampleSize();
Wei Jia53692fa2017-12-11 10:33:46 -08001259 }
Robert Shih0cd95062018-01-21 17:41:16 -08001260
1261 if (err != OK || sampleSize < 0) {
1262 mLock.lock();
1263 track->mPackets->signalEOS(err != OK ? err : ERROR_END_OF_STREAM);
1264 break;
1265 }
1266
1267 sp<ABuffer> abuf = new ABuffer(sampleSize);
1268 sampleSize = extractor->readSampleData(abuf);
Wei Jia53692fa2017-12-11 10:33:46 -08001269 mLock.lock();
1270
Wei Jia53692fa2017-12-11 10:33:46 -08001271 // in case track has been changed since we don't have lock for some time.
1272 if (generation != getDataGeneration(trackType)) {
Wei Jia53692fa2017-12-11 10:33:46 -08001273 break;
1274 }
1275
Robert Shih0cd95062018-01-21 17:41:16 -08001276 int64_t timeUs = extractor->getSampleTime();
1277 if (timeUs < 0) {
1278 track->mPackets->signalEOS(ERROR_MALFORMED);
Wei Jia53692fa2017-12-11 10:33:46 -08001279 break;
1280 }
1281
Robert Shih0cd95062018-01-21 17:41:16 -08001282 sp<AMessage> meta = abuf->meta();
1283 format->writeToAMessage(meta);
1284 meta->setInt64("timeUs", timeUs);
1285 if (trackType == MEDIA_TRACK_TYPE_AUDIO) {
1286 mAudioTimeUs = timeUs;
1287 } else if (trackType == MEDIA_TRACK_TYPE_VIDEO) {
1288 mVideoTimeUs = timeUs;
Wei Jia53692fa2017-12-11 10:33:46 -08001289 }
Robert Shih0cd95062018-01-21 17:41:16 -08001290
1291 queueDiscontinuityIfNeeded(seeking, formatChange, trackType, track);
1292
1293 if (numBuffers == 0 && actualTimeUs != nullptr) {
1294 *actualTimeUs = timeUs;
1295 }
1296 if (seeking) {
1297 if (meta != nullptr && mode == MediaPlayer2SeekMode::SEEK_CLOSEST
1298 && seekTimeUs > timeUs) {
1299 sp<AMessage> extra = new AMessage;
1300 extra->setInt64("resume-at-mediaTimeUs", seekTimeUs);
1301 meta->setMessage("extra", extra);
1302 }
1303 }
1304
1305 track->mPackets->queueAccessUnit(abuf);
1306 formatChange = false;
1307 seeking = false;
1308 ++numBuffers;
1309 extractor->advance();
1310
Wei Jia53692fa2017-12-11 10:33:46 -08001311 }
1312
1313 if (mIsStreaming
1314 && (trackType == MEDIA_TRACK_TYPE_VIDEO || trackType == MEDIA_TRACK_TYPE_AUDIO)) {
1315 status_t finalResult;
1316 int64_t durationUs = track->mPackets->getBufferedDurationUs(&finalResult);
1317
1318 // TODO: maxRebufferingMarkMs could be larger than
1319 // mBufferingSettings.mResumePlaybackMarkMs
1320 int64_t markUs = (mPreparing ? mBufferingSettings.mInitialMarkMs
1321 : mBufferingSettings.mResumePlaybackMarkMs) * 1000ll;
1322 if (finalResult == ERROR_END_OF_STREAM || durationUs >= markUs) {
1323 if (mPreparing || mSentPauseOnBuffering) {
1324 Track *counterTrack =
1325 (trackType == MEDIA_TRACK_TYPE_VIDEO ? &mAudioTrack : &mVideoTrack);
Robert Shih0cd95062018-01-21 17:41:16 -08001326 if (counterTrack->mExtractor != NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001327 durationUs = counterTrack->mPackets->getBufferedDurationUs(&finalResult);
1328 }
1329 if (finalResult == ERROR_END_OF_STREAM || durationUs >= markUs) {
1330 if (mPreparing) {
1331 notifyPrepared();
1332 mPreparing = false;
1333 } else {
Wei Jia53692fa2017-12-11 10:33:46 -08001334 mSentPauseOnBuffering = false;
1335 sp<AMessage> notify = dupNotify();
1336 notify->setInt32("what", kWhatResumeOnBufferingEnd);
1337 notify->post();
1338 }
1339 }
1340 }
1341 return;
1342 }
1343
1344 postReadBuffer(trackType);
1345 }
1346}
1347
Wei Jia2409c872018-02-02 10:34:33 -08001348void NuPlayer2::GenericSource2::queueDiscontinuityIfNeeded(
Wei Jia53692fa2017-12-11 10:33:46 -08001349 bool seeking, bool formatChange, media_track_type trackType, Track *track) {
1350 // formatChange && seeking: track whose source is changed during selection
1351 // formatChange && !seeking: track whose source is not changed during selection
1352 // !formatChange: normal seek
1353 if ((seeking || formatChange)
1354 && (trackType == MEDIA_TRACK_TYPE_AUDIO
1355 || trackType == MEDIA_TRACK_TYPE_VIDEO)) {
1356 ATSParser::DiscontinuityType type = (formatChange && seeking)
1357 ? ATSParser::DISCONTINUITY_FORMATCHANGE
1358 : ATSParser::DISCONTINUITY_NONE;
1359 track->mPackets->queueDiscontinuity(type, NULL /* extra */, true /* discard */);
1360 }
1361}
1362
Wei Jia2409c872018-02-02 10:34:33 -08001363void NuPlayer2::GenericSource2::notifyBufferingUpdate(int32_t percentage) {
Wei Jia53692fa2017-12-11 10:33:46 -08001364 // Buffering percent could go backward as it's estimated from remaining
1365 // data and last access time. This could cause the buffering position
1366 // drawn on media control to jitter slightly. Remember previously reported
1367 // percentage and don't allow it to go backward.
1368 if (percentage < mPrevBufferPercentage) {
1369 percentage = mPrevBufferPercentage;
1370 } else if (percentage > 100) {
1371 percentage = 100;
1372 }
1373
1374 mPrevBufferPercentage = percentage;
1375
1376 ALOGV("notifyBufferingUpdate: buffering %d%%", percentage);
1377
1378 sp<AMessage> notify = dupNotify();
1379 notify->setInt32("what", kWhatBufferingUpdate);
1380 notify->setInt32("percentage", percentage);
1381 notify->post();
1382}
1383
Wei Jia2409c872018-02-02 10:34:33 -08001384void NuPlayer2::GenericSource2::schedulePollBuffering() {
Wei Jiaf01e3122018-10-18 11:49:44 -07001385 if (mIsStreaming) {
1386 sp<AMessage> msg = new AMessage(kWhatPollBuffering, this);
1387 msg->setInt32("generation", mPollBufferingGeneration);
1388 // Enquires buffering status every second.
1389 msg->post(1000000ll);
1390 }
Wei Jia53692fa2017-12-11 10:33:46 -08001391}
1392
Wei Jia2409c872018-02-02 10:34:33 -08001393void NuPlayer2::GenericSource2::onPollBuffering() {
Wei Jia53692fa2017-12-11 10:33:46 -08001394 int64_t cachedDurationUs = -1ll;
Wei Jia53692fa2017-12-11 10:33:46 -08001395
Robert Shih49fb89d2018-01-31 17:53:19 -08001396 sp<AMediaExtractorWrapper> extractor;
1397 if (mVideoTrack.mExtractor != NULL) {
1398 extractor = mVideoTrack.mExtractor;
1399 } else if (mAudioTrack.mExtractor != NULL) {
1400 extractor = mAudioTrack.mExtractor;
Wei Jia53692fa2017-12-11 10:33:46 -08001401 }
1402
Robert Shih49fb89d2018-01-31 17:53:19 -08001403 if (extractor != NULL) {
1404 cachedDurationUs = extractor->getCachedDuration();
Wei Jia53692fa2017-12-11 10:33:46 -08001405 }
1406
1407 if (cachedDurationUs >= 0ll) {
Robert Shih49fb89d2018-01-31 17:53:19 -08001408 ssize_t sampleSize = extractor->getSampleSize();
1409 if (sampleSize >= 0ll) {
Wei Jia53692fa2017-12-11 10:33:46 -08001410 int64_t cachedPosUs = getLastReadPosition() + cachedDurationUs;
1411 int percentage = 100.0 * cachedPosUs / mDurationUs;
1412 if (percentage > 100) {
1413 percentage = 100;
1414 }
1415
1416 notifyBufferingUpdate(percentage);
Robert Shih49fb89d2018-01-31 17:53:19 -08001417 ALOGV("onPollBuffering: cachedDurationUs %.1f sec", cachedDurationUs / 1000000.0f);
1418 } else {
1419 notifyBufferingUpdate(100);
1420 ALOGV("onPollBuffering: EOS");
Wei Jia53692fa2017-12-11 10:33:46 -08001421 }
Wei Jia53692fa2017-12-11 10:33:46 -08001422 }
1423
1424 schedulePollBuffering();
1425}
1426
1427// Modular DRM
Wei Jia2409c872018-02-02 10:34:33 -08001428status_t NuPlayer2::GenericSource2::prepareDrm(
Wei Jia53692fa2017-12-11 10:33:46 -08001429 const uint8_t uuid[16],
1430 const Vector<uint8_t> &drmSessionId,
1431 sp<AMediaCryptoWrapper> *outCrypto) {
1432 Mutex::Autolock _l(mLock);
1433 ALOGV("prepareDrm");
1434
1435 mIsDrmProtected = false;
1436 mIsDrmReleased = false;
1437 mIsSecure = false;
1438
1439 status_t status = OK;
1440 sp<AMediaCryptoWrapper> crypto =
1441 new AMediaCryptoWrapper(uuid, drmSessionId.array(), drmSessionId.size());
1442 if (crypto == NULL) {
1443 ALOGE("prepareDrm: failed to create crypto.");
1444 return UNKNOWN_ERROR;
1445 }
1446 ALOGV("prepareDrm: crypto created for uuid: %s",
1447 DrmUUID::toHexString(uuid).string());
1448
1449 *outCrypto = crypto;
1450 // as long a there is an active crypto
1451 mIsDrmProtected = true;
1452
1453 if (mMimes.size() == 0) {
1454 status = UNKNOWN_ERROR;
1455 ALOGE("prepareDrm: Unexpected. Must have at least one track. status: %d", status);
1456 return status;
1457 }
1458
1459 // first mime in this list is either the video track, or the first audio track
1460 const char *mime = mMimes[0].string();
1461 mIsSecure = crypto->requiresSecureDecoderComponent(mime);
1462 ALOGV("prepareDrm: requiresSecureDecoderComponent mime: %s isSecure: %d",
1463 mime, mIsSecure);
1464
1465 // Checking the member flags while in the looper to send out the notification.
1466 // The legacy mDecryptHandle!=NULL check (for FLAG_PROTECTED) is equivalent to mIsDrmProtected.
1467 notifyFlagsChanged(
1468 (mIsSecure ? FLAG_SECURE : 0) |
1469 // Setting "protected screen" only for L1: b/38390836
1470 (mIsSecure ? FLAG_PROTECTED : 0) |
1471 FLAG_CAN_PAUSE |
1472 FLAG_CAN_SEEK_BACKWARD |
1473 FLAG_CAN_SEEK_FORWARD |
1474 FLAG_CAN_SEEK);
1475
1476 if (status == OK) {
1477 ALOGV("prepareDrm: mCrypto: %p", outCrypto->get());
1478 ALOGD("prepareDrm ret: %d ", status);
1479 } else {
1480 ALOGE("prepareDrm err: %d", status);
1481 }
1482 return status;
1483}
1484
Wei Jia2409c872018-02-02 10:34:33 -08001485status_t NuPlayer2::GenericSource2::releaseDrm() {
Wei Jia53692fa2017-12-11 10:33:46 -08001486 Mutex::Autolock _l(mLock);
1487 ALOGV("releaseDrm");
1488
1489 if (mIsDrmProtected) {
1490 mIsDrmProtected = false;
1491 // to prevent returning any more buffer after stop/releaseDrm (b/37960096)
1492 mIsDrmReleased = true;
1493 ALOGV("releaseDrm: mIsDrmProtected is reset.");
1494 } else {
1495 ALOGE("releaseDrm: mIsDrmProtected is already false.");
1496 }
1497
1498 return OK;
1499}
1500
Wei Jia2409c872018-02-02 10:34:33 -08001501status_t NuPlayer2::GenericSource2::checkDrmInfo()
Wei Jia53692fa2017-12-11 10:33:46 -08001502{
1503 // clearing the flag at prepare in case the player is reused after stop/releaseDrm with the
1504 // same source without being reset (called by prepareAsync/initFromDataSource)
1505 mIsDrmReleased = false;
1506
Robert Shih0cd95062018-01-21 17:41:16 -08001507 if (mExtractor == NULL) {
1508 ALOGV("checkDrmInfo: No extractor");
Wei Jia53692fa2017-12-11 10:33:46 -08001509 return OK; // letting the caller responds accordingly
1510 }
1511
Robert Shih0cd95062018-01-21 17:41:16 -08001512 PsshInfo *psshInfo = mExtractor->getPsshInfo();
1513 if (psshInfo == NULL) {
Wei Jia53692fa2017-12-11 10:33:46 -08001514 ALOGV("checkDrmInfo: No PSSH");
1515 return OK; // source without DRM info
1516 }
1517
Dongwon Kang41929fb2018-09-09 08:29:56 -07001518 PlayerMessage playerMsg;
1519 status_t ret = NuPlayer2Drm::retrieveDrmInfo(psshInfo, &playerMsg);
1520 ALOGV("checkDrmInfo: MEDIA_DRM_INFO PSSH drm info size: %d", (int)playerMsg.ByteSize());
Wei Jia53692fa2017-12-11 10:33:46 -08001521
Dongwon Kang41929fb2018-09-09 08:29:56 -07001522 if (ret != OK) {
1523 ALOGE("checkDrmInfo: failed to retrive DrmInfo %d", ret);
Wei Jia53692fa2017-12-11 10:33:46 -08001524 return UNKNOWN_ERROR;
1525 }
1526
Dongwon Kang41929fb2018-09-09 08:29:56 -07001527 int size = playerMsg.ByteSize();
1528 sp<ABuffer> drmInfoBuf = new ABuffer(size);
1529 playerMsg.SerializeToArray(drmInfoBuf->data(), size);
1530 drmInfoBuf->setRange(0, size);
1531 notifyDrmInfo(drmInfoBuf);
Wei Jia53692fa2017-12-11 10:33:46 -08001532
1533 return OK;
1534}
1535
Dongwon Kang1889c3e2018-02-01 13:44:57 -08001536void NuPlayer2::GenericSource2::signalBufferReturned(MediaBufferBase *buffer)
Wei Jia53692fa2017-12-11 10:33:46 -08001537{
1538 //ALOGV("signalBufferReturned %p refCount: %d", buffer, buffer->localRefcount());
1539
1540 buffer->setObserver(NULL);
1541 buffer->release(); // this leads to delete since that there is no observor
1542}
1543
1544} // namespace android