blob: ffeea86df818cd5a6fc7cf9863212897c1cb21a5 [file] [log] [blame]
Wei Jiabc2fb722014-07-08 16:37:57 -07001/*
2 * Copyright (C) 2014 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17//#define LOG_NDEBUG 0
18#define LOG_TAG "NuPlayerDecoderPassThrough"
19#include <utils/Log.h>
20#include <inttypes.h>
21
22#include "NuPlayerDecoderPassThrough.h"
23
Wei Jiac6cfd702014-11-11 16:33:20 -080024#include "NuPlayerRenderer.h"
25#include "NuPlayerSource.h"
26
Wei Jiabc2fb722014-07-08 16:37:57 -070027#include <media/ICrypto.h>
Wonsik Kim7e34bf52016-08-23 00:09:18 +090028#include <media/MediaCodecBuffer.h>
Wei Jiabc2fb722014-07-08 16:37:57 -070029#include <media/stagefright/foundation/ABuffer.h>
30#include <media/stagefright/foundation/ADebug.h>
31#include <media/stagefright/foundation/AMessage.h>
Wei Jiabc2fb722014-07-08 16:37:57 -070032#include <media/stagefright/MediaErrors.h>
33
Chong Zhang7137ec72014-11-12 16:41:05 -080034#include "ATSParser.h"
35
Wei Jiabc2fb722014-07-08 16:37:57 -070036namespace android {
37
Chong Zhang7137ec72014-11-12 16:41:05 -080038// TODO optimize buffer size for power consumption
39// The offload read buffer size is 32 KB but 24 KB uses less power.
40static const size_t kAggregateBufferSizeBytes = 24 * 1024;
Phil Burkc5cc2e22014-09-09 20:08:39 -070041static const size_t kMaxCachedBytes = 200000;
Wei Jiabc2fb722014-07-08 16:37:57 -070042
43NuPlayer::DecoderPassThrough::DecoderPassThrough(
Wei Jiac6cfd702014-11-11 16:33:20 -080044 const sp<AMessage> &notify,
45 const sp<Source> &source,
46 const sp<Renderer> &renderer)
Andy Hung202bce12014-12-03 11:47:36 -080047 : DecoderBase(notify),
Wei Jiac6cfd702014-11-11 16:33:20 -080048 mSource(source),
49 mRenderer(renderer),
50 mSkipRenderingUntilMediaTimeUs(-1ll),
Wei Jiabc2fb722014-07-08 16:37:57 -070051 mReachedEOS(true),
Chong Zhang7137ec72014-11-12 16:41:05 -080052 mPendingAudioErr(OK),
Phil Burkc5cc2e22014-09-09 20:08:39 -070053 mPendingBuffersToDrain(0),
Chong Zhangde01afb2014-08-13 13:48:10 -070054 mCachedBytes(0),
Wei Jiabc2fb722014-07-08 16:37:57 -070055 mComponentName("pass through decoder") {
Wei Jiac6cfd702014-11-11 16:33:20 -080056 ALOGW_IF(renderer == NULL, "expect a non-NULL renderer");
Wei Jiabc2fb722014-07-08 16:37:57 -070057}
58
59NuPlayer::DecoderPassThrough::~DecoderPassThrough() {
60}
61
Wei Jiabc2fb722014-07-08 16:37:57 -070062void NuPlayer::DecoderPassThrough::onConfigure(const sp<AMessage> &format) {
63 ALOGV("[%s] onConfigure", mComponentName.c_str());
Chong Zhangde01afb2014-08-13 13:48:10 -070064 mCachedBytes = 0;
Phil Burkc5cc2e22014-09-09 20:08:39 -070065 mPendingBuffersToDrain = 0;
Wei Jiabc2fb722014-07-08 16:37:57 -070066 mReachedEOS = false;
67 ++mBufferGeneration;
68
Chong Zhang7137ec72014-11-12 16:41:05 -080069 onRequestInputBuffers();
Wei Jiabc2fb722014-07-08 16:37:57 -070070
Haynes Mathew George8b635332015-03-30 17:59:47 -070071 int32_t hasVideo = 0;
72 format->findInt32("has-video", &hasVideo);
73
Andy Hung202bce12014-12-03 11:47:36 -080074 // The audio sink is already opened before the PassThrough decoder is created.
75 // Opening again might be relevant if decoder is instantiated after shutdown and
76 // format is different.
77 status_t err = mRenderer->openAudioSink(
Haynes Mathew George8b635332015-03-30 17:59:47 -070078 format, true /* offloadOnly */, hasVideo,
Andy Hung202bce12014-12-03 11:47:36 -080079 AUDIO_OUTPUT_FLAG_NONE /* flags */, NULL /* isOffloaded */);
80 if (err != OK) {
81 handleError(err);
Wei Jiac6cfd702014-11-11 16:33:20 -080082 }
Wei Jiabc2fb722014-07-08 16:37:57 -070083}
84
Ronghua Wu8db88132015-04-22 13:51:35 -070085void NuPlayer::DecoderPassThrough::onSetParameters(const sp<AMessage> &/*params*/) {
86 ALOGW("onSetParameters() called unexpectedly");
87}
88
Chong Zhang7137ec72014-11-12 16:41:05 -080089void NuPlayer::DecoderPassThrough::onSetRenderer(
90 const sp<Renderer> &renderer) {
91 // renderer can't be changed during offloading
92 ALOGW_IF(renderer != mRenderer,
93 "ignoring request to change renderer");
94}
95
96void NuPlayer::DecoderPassThrough::onGetInputBuffers(
Wonsik Kim7e34bf52016-08-23 00:09:18 +090097 Vector<sp<MediaCodecBuffer> > * /* dstBuffers */) {
Chong Zhang7137ec72014-11-12 16:41:05 -080098 ALOGE("onGetInputBuffers() called unexpectedly");
99}
100
Wei Jiabc2fb722014-07-08 16:37:57 -0700101bool NuPlayer::DecoderPassThrough::isStaleReply(const sp<AMessage> &msg) {
102 int32_t generation;
103 CHECK(msg->findInt32("generation", &generation));
104 return generation != mBufferGeneration;
105}
106
Ronghua Wuf1828912014-12-01 15:33:36 -0800107bool NuPlayer::DecoderPassThrough::isDoneFetching() const {
108 ALOGV("[%s] mCachedBytes = %zu, mReachedEOS = %d mPaused = %d",
109 mComponentName.c_str(), mCachedBytes, mReachedEOS, mPaused);
Wei Jiabc2fb722014-07-08 16:37:57 -0700110
Ronghua Wuf1828912014-12-01 15:33:36 -0800111 return mCachedBytes >= kMaxCachedBytes || mReachedEOS || mPaused;
Wei Jiabc2fb722014-07-08 16:37:57 -0700112}
113
Chong Zhang3b032b32015-04-17 15:49:06 -0700114/*
115 * returns true if we should request more data
116 */
117bool NuPlayer::DecoderPassThrough::doRequestBuffers() {
Chong Zhang7137ec72014-11-12 16:41:05 -0800118 status_t err = OK;
Ronghua Wuf1828912014-12-01 15:33:36 -0800119 while (!isDoneFetching()) {
Chong Zhang7137ec72014-11-12 16:41:05 -0800120 sp<AMessage> msg = new AMessage();
121
122 err = fetchInputData(msg);
123 if (err != OK) {
124 break;
125 }
126
127 onInputBufferFetched(msg);
128 }
129
Chong Zhang3b032b32015-04-17 15:49:06 -0700130 return err == -EWOULDBLOCK
131 && mSource->feedMoreTSData() == OK;
Chong Zhang7137ec72014-11-12 16:41:05 -0800132}
133
134status_t NuPlayer::DecoderPassThrough::dequeueAccessUnit(sp<ABuffer> *accessUnit) {
135 status_t err;
136
137 // Did we save an accessUnit earlier because of a discontinuity?
138 if (mPendingAudioAccessUnit != NULL) {
139 *accessUnit = mPendingAudioAccessUnit;
140 mPendingAudioAccessUnit.clear();
141 err = mPendingAudioErr;
142 ALOGV("feedDecoderInputData() use mPendingAudioAccessUnit");
143 } else {
144 err = mSource->dequeueAccessUnit(true /* audio */, accessUnit);
145 }
146
147 if (err == INFO_DISCONTINUITY || err == ERROR_END_OF_STREAM) {
148 if (mAggregateBuffer != NULL) {
149 // We already have some data so save this for later.
150 mPendingAudioErr = err;
151 mPendingAudioAccessUnit = *accessUnit;
152 (*accessUnit).clear();
153 ALOGD("return aggregated buffer and save err(=%d) for later", err);
154 err = OK;
155 }
156 }
157
158 return err;
159}
160
161sp<ABuffer> NuPlayer::DecoderPassThrough::aggregateBuffer(
162 const sp<ABuffer> &accessUnit) {
163 sp<ABuffer> aggregate;
164
165 if (accessUnit == NULL) {
166 // accessUnit is saved to mPendingAudioAccessUnit
167 // return current mAggregateBuffer
168 aggregate = mAggregateBuffer;
169 mAggregateBuffer.clear();
170 return aggregate;
171 }
172
173 size_t smallSize = accessUnit->size();
174 if ((mAggregateBuffer == NULL)
175 // Don't bother if only room for a few small buffers.
176 && (smallSize < (kAggregateBufferSizeBytes / 3))) {
177 // Create a larger buffer for combining smaller buffers from the extractor.
178 mAggregateBuffer = new ABuffer(kAggregateBufferSizeBytes);
179 mAggregateBuffer->setRange(0, 0); // start empty
180 }
181
182 if (mAggregateBuffer != NULL) {
183 int64_t timeUs;
184 int64_t dummy;
185 bool smallTimestampValid = accessUnit->meta()->findInt64("timeUs", &timeUs);
186 bool bigTimestampValid = mAggregateBuffer->meta()->findInt64("timeUs", &dummy);
187 // Will the smaller buffer fit?
188 size_t bigSize = mAggregateBuffer->size();
189 size_t roomLeft = mAggregateBuffer->capacity() - bigSize;
190 // Should we save this small buffer for the next big buffer?
191 // If the first small buffer did not have a timestamp then save
192 // any buffer that does have a timestamp until the next big buffer.
193 if ((smallSize > roomLeft)
194 || (!bigTimestampValid && (bigSize > 0) && smallTimestampValid)) {
195 mPendingAudioErr = OK;
196 mPendingAudioAccessUnit = accessUnit;
197 aggregate = mAggregateBuffer;
198 mAggregateBuffer.clear();
199 } else {
200 // Grab time from first small buffer if available.
201 if ((bigSize == 0) && smallTimestampValid) {
202 mAggregateBuffer->meta()->setInt64("timeUs", timeUs);
203 }
204 // Append small buffer to the bigger buffer.
205 memcpy(mAggregateBuffer->base() + bigSize, accessUnit->data(), smallSize);
206 bigSize += smallSize;
207 mAggregateBuffer->setRange(0, bigSize);
208
209 ALOGV("feedDecoderInputData() smallSize = %zu, bigSize = %zu, capacity = %zu",
210 smallSize, bigSize, mAggregateBuffer->capacity());
211 }
212 } else {
213 // decided not to aggregate
214 aggregate = accessUnit;
215 }
216
217 return aggregate;
218}
219
220status_t NuPlayer::DecoderPassThrough::fetchInputData(sp<AMessage> &reply) {
221 sp<ABuffer> accessUnit;
222
223 do {
224 status_t err = dequeueAccessUnit(&accessUnit);
225
226 if (err == -EWOULDBLOCK) {
Wei Jia14532f22015-12-29 11:28:15 -0800227 // Flush out the aggregate buffer to try to avoid underrun.
228 accessUnit = aggregateBuffer(NULL /* accessUnit */);
229 if (accessUnit != NULL) {
230 break;
231 }
Chong Zhang7137ec72014-11-12 16:41:05 -0800232 return err;
233 } else if (err != OK) {
234 if (err == INFO_DISCONTINUITY) {
235 int32_t type;
236 CHECK(accessUnit->meta()->findInt32("discontinuity", &type));
237
238 bool formatChange =
239 (type & ATSParser::DISCONTINUITY_AUDIO_FORMAT) != 0;
240
241 bool timeChange =
242 (type & ATSParser::DISCONTINUITY_TIME) != 0;
243
244 ALOGI("audio discontinuity (formatChange=%d, time=%d)",
245 formatChange, timeChange);
246
247 if (formatChange || timeChange) {
248 sp<AMessage> msg = mNotify->dup();
249 msg->setInt32("what", kWhatInputDiscontinuity);
250 // will perform seamless format change,
251 // only notify NuPlayer to scan sources
252 msg->setInt32("formatChange", false);
253 msg->post();
254 }
255
256 if (timeChange) {
Chong Zhang66704af2015-03-03 19:32:35 -0800257 doFlush(false /* notifyComplete */);
Chong Zhang7137ec72014-11-12 16:41:05 -0800258 err = OK;
259 } else if (formatChange) {
260 // do seamless format change
261 err = OK;
262 } else {
263 // This stream is unaffected by the discontinuity
264 return -EWOULDBLOCK;
265 }
266 }
267
268 reply->setInt32("err", err);
269 return OK;
270 }
271
272 accessUnit = aggregateBuffer(accessUnit);
273 } while (accessUnit == NULL);
274
275#if 0
276 int64_t mediaTimeUs;
277 CHECK(accessUnit->meta()->findInt64("timeUs", &mediaTimeUs));
278 ALOGV("feeding audio input buffer at media time %.2f secs",
279 mediaTimeUs / 1E6);
280#endif
281
282 reply->setBuffer("buffer", accessUnit);
283
284 return OK;
285}
286
287void NuPlayer::DecoderPassThrough::onInputBufferFetched(
Wei Jiabc2fb722014-07-08 16:37:57 -0700288 const sp<AMessage> &msg) {
289 if (mReachedEOS) {
290 return;
291 }
292
Wonsik Kim7e34bf52016-08-23 00:09:18 +0900293 sp<RefBase> obj;
294 bool hasBuffer = msg->findObject("buffer", &obj);
295 sp<MediaCodecBuffer> buffer;
296 if (hasBuffer) {
297 buffer = static_cast<MediaCodecBuffer *>(obj.get());
298 }
Wei Jiabc2fb722014-07-08 16:37:57 -0700299 if (buffer == NULL) {
Chong Zhang7137ec72014-11-12 16:41:05 -0800300 int32_t streamErr = ERROR_END_OF_STREAM;
301 CHECK(msg->findInt32("err", &streamErr) || !hasBuffer);
302 if (streamErr == OK) {
303 return;
304 }
305
Wei Jiabc2fb722014-07-08 16:37:57 -0700306 mReachedEOS = true;
Wei Jiac6cfd702014-11-11 16:33:20 -0800307 if (mRenderer != NULL) {
308 mRenderer->queueEOS(true /* audio */, ERROR_END_OF_STREAM);
309 }
Wei Jiabc2fb722014-07-08 16:37:57 -0700310 return;
311 }
312
Wei Jiac6cfd702014-11-11 16:33:20 -0800313 sp<AMessage> extra;
314 if (buffer->meta()->findMessage("extra", &extra) && extra != NULL) {
315 int64_t resumeAtMediaTimeUs;
316 if (extra->findInt64(
317 "resume-at-mediatimeUs", &resumeAtMediaTimeUs)) {
318 ALOGI("[%s] suppressing rendering until %lld us",
319 mComponentName.c_str(), (long long)resumeAtMediaTimeUs);
320 mSkipRenderingUntilMediaTimeUs = resumeAtMediaTimeUs;
321 }
322 }
323
324 int32_t bufferSize = buffer->size();
325 mCachedBytes += bufferSize;
326
327 if (mSkipRenderingUntilMediaTimeUs >= 0) {
328 int64_t timeUs = 0;
329 CHECK(buffer->meta()->findInt64("timeUs", &timeUs));
330
331 if (timeUs < mSkipRenderingUntilMediaTimeUs) {
332 ALOGV("[%s] dropping buffer at time %lld as requested.",
333 mComponentName.c_str(), (long long)timeUs);
334
335 onBufferConsumed(bufferSize);
336 return;
337 }
338
339 mSkipRenderingUntilMediaTimeUs = -1;
340 }
341
342 if (mRenderer == NULL) {
343 onBufferConsumed(bufferSize);
344 return;
345 }
Chong Zhangde01afb2014-08-13 13:48:10 -0700346
Lajos Molnar1d15ab52015-03-04 16:46:34 -0800347 sp<AMessage> reply = new AMessage(kWhatBufferConsumed, this);
Wei Jiabc2fb722014-07-08 16:37:57 -0700348 reply->setInt32("generation", mBufferGeneration);
Wei Jiac6cfd702014-11-11 16:33:20 -0800349 reply->setInt32("size", bufferSize);
Wei Jiabc2fb722014-07-08 16:37:57 -0700350
Wei Jiac6cfd702014-11-11 16:33:20 -0800351 mRenderer->queueBuffer(true /* audio */, buffer, reply);
352
Phil Burkc5cc2e22014-09-09 20:08:39 -0700353 ++mPendingBuffersToDrain;
Chong Zhang7137ec72014-11-12 16:41:05 -0800354 ALOGV("onInputBufferFilled: #ToDrain = %zu, cachedBytes = %zu",
355 mPendingBuffersToDrain, mCachedBytes);
Wei Jiabc2fb722014-07-08 16:37:57 -0700356}
357
Chong Zhangde01afb2014-08-13 13:48:10 -0700358void NuPlayer::DecoderPassThrough::onBufferConsumed(int32_t size) {
Phil Burkc5cc2e22014-09-09 20:08:39 -0700359 --mPendingBuffersToDrain;
Chong Zhangde01afb2014-08-13 13:48:10 -0700360 mCachedBytes -= size;
Chong Zhang7137ec72014-11-12 16:41:05 -0800361 ALOGV("onBufferConsumed: #ToDrain = %zu, cachedBytes = %zu",
362 mPendingBuffersToDrain, mCachedBytes);
363 onRequestInputBuffers();
Wei Jiabc2fb722014-07-08 16:37:57 -0700364}
365
Chong Zhangf8d71772014-11-26 15:08:34 -0800366void NuPlayer::DecoderPassThrough::onResume(bool notifyComplete) {
Ronghua Wuf1828912014-12-01 15:33:36 -0800367 mPaused = false;
368
Chong Zhang7137ec72014-11-12 16:41:05 -0800369 onRequestInputBuffers();
Chong Zhangf8d71772014-11-26 15:08:34 -0800370
371 if (notifyComplete) {
372 sp<AMessage> notify = mNotify->dup();
373 notify->setInt32("what", kWhatResumeCompleted);
374 notify->post();
375 }
Chong Zhang7137ec72014-11-12 16:41:05 -0800376}
377
Chong Zhang66704af2015-03-03 19:32:35 -0800378void NuPlayer::DecoderPassThrough::doFlush(bool notifyComplete) {
Wei Jiabc2fb722014-07-08 16:37:57 -0700379 ++mBufferGeneration;
Wei Jiac6cfd702014-11-11 16:33:20 -0800380 mSkipRenderingUntilMediaTimeUs = -1;
Ronghua Wuf1828912014-12-01 15:33:36 -0800381 mPendingAudioAccessUnit.clear();
382 mPendingAudioErr = OK;
383 mAggregateBuffer.clear();
Wei Jiac6cfd702014-11-11 16:33:20 -0800384
385 if (mRenderer != NULL) {
Chong Zhang7137ec72014-11-12 16:41:05 -0800386 mRenderer->flush(true /* audio */, notifyComplete);
387 mRenderer->signalTimeDiscontinuity();
Wei Jiac6cfd702014-11-11 16:33:20 -0800388 }
Wei Jiabc2fb722014-07-08 16:37:57 -0700389
Phil Burkc5cc2e22014-09-09 20:08:39 -0700390 mPendingBuffersToDrain = 0;
Chong Zhangde01afb2014-08-13 13:48:10 -0700391 mCachedBytes = 0;
Wei Jiabc2fb722014-07-08 16:37:57 -0700392 mReachedEOS = false;
393}
394
Chong Zhang66704af2015-03-03 19:32:35 -0800395void NuPlayer::DecoderPassThrough::onFlush() {
396 doFlush(true /* notifyComplete */);
397
398 mPaused = true;
399 sp<AMessage> notify = mNotify->dup();
400 notify->setInt32("what", kWhatFlushCompleted);
401 notify->post();
402
403}
404
Chong Zhang7137ec72014-11-12 16:41:05 -0800405void NuPlayer::DecoderPassThrough::onShutdown(bool notifyComplete) {
Wei Jiabc2fb722014-07-08 16:37:57 -0700406 ++mBufferGeneration;
Wei Jiac6cfd702014-11-11 16:33:20 -0800407 mSkipRenderingUntilMediaTimeUs = -1;
Wei Jiabc2fb722014-07-08 16:37:57 -0700408
Chong Zhang7137ec72014-11-12 16:41:05 -0800409 if (notifyComplete) {
410 sp<AMessage> notify = mNotify->dup();
411 notify->setInt32("what", kWhatShutdownCompleted);
412 notify->post();
413 }
414
Wei Jiabc2fb722014-07-08 16:37:57 -0700415 mReachedEOS = true;
416}
417
418void NuPlayer::DecoderPassThrough::onMessageReceived(const sp<AMessage> &msg) {
419 ALOGV("[%s] onMessage: %s", mComponentName.c_str(),
420 msg->debugString().c_str());
421
422 switch (msg->what()) {
Wei Jiabc2fb722014-07-08 16:37:57 -0700423 case kWhatBufferConsumed:
424 {
425 if (!isStaleReply(msg)) {
Chong Zhangde01afb2014-08-13 13:48:10 -0700426 int32_t size;
427 CHECK(msg->findInt32("size", &size));
428 onBufferConsumed(size);
Wei Jiabc2fb722014-07-08 16:37:57 -0700429 }
430 break;
431 }
432
Wei Jiabc2fb722014-07-08 16:37:57 -0700433 default:
Chong Zhang7137ec72014-11-12 16:41:05 -0800434 DecoderBase::onMessageReceived(msg);
Wei Jiabc2fb722014-07-08 16:37:57 -0700435 break;
436 }
437}
438
439} // namespace android