blob: 3b4c0a757cf7dc77c2aa061bb4b7fc5ead24fd83 [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>
28#include <media/stagefright/foundation/ABuffer.h>
29#include <media/stagefright/foundation/ADebug.h>
30#include <media/stagefright/foundation/AMessage.h>
Wei Jiabc2fb722014-07-08 16:37:57 -070031#include <media/stagefright/MediaErrors.h>
32
Chong Zhang7137ec72014-11-12 16:41:05 -080033#include "ATSParser.h"
34
Wei Jiabc2fb722014-07-08 16:37:57 -070035namespace android {
36
Chong Zhang7137ec72014-11-12 16:41:05 -080037// TODO optimize buffer size for power consumption
38// The offload read buffer size is 32 KB but 24 KB uses less power.
39static const size_t kAggregateBufferSizeBytes = 24 * 1024;
Phil Burkc5cc2e22014-09-09 20:08:39 -070040static const size_t kMaxCachedBytes = 200000;
Wei Jiabc2fb722014-07-08 16:37:57 -070041
42NuPlayer::DecoderPassThrough::DecoderPassThrough(
Wei Jiac6cfd702014-11-11 16:33:20 -080043 const sp<AMessage> &notify,
44 const sp<Source> &source,
45 const sp<Renderer> &renderer)
Chong Zhang7137ec72014-11-12 16:41:05 -080046 : mNotify(notify),
Wei Jiac6cfd702014-11-11 16:33:20 -080047 mSource(source),
48 mRenderer(renderer),
49 mSkipRenderingUntilMediaTimeUs(-1ll),
Wei Jiabc2fb722014-07-08 16:37:57 -070050 mBufferGeneration(0),
51 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
Chong Zhang7137ec72014-11-12 16:41:05 -080062void NuPlayer::DecoderPassThrough::getStats(
63 int64_t *numFramesTotal, int64_t *numFramesDropped) const {
64 *numFramesTotal = 0;
65 *numFramesDropped = 0;
Wei Jiabc2fb722014-07-08 16:37:57 -070066}
67
68void NuPlayer::DecoderPassThrough::onConfigure(const sp<AMessage> &format) {
69 ALOGV("[%s] onConfigure", mComponentName.c_str());
Chong Zhangde01afb2014-08-13 13:48:10 -070070 mCachedBytes = 0;
Phil Burkc5cc2e22014-09-09 20:08:39 -070071 mPendingBuffersToDrain = 0;
Wei Jiabc2fb722014-07-08 16:37:57 -070072 mReachedEOS = false;
73 ++mBufferGeneration;
74
Chong Zhang7137ec72014-11-12 16:41:05 -080075 onRequestInputBuffers();
Wei Jiabc2fb722014-07-08 16:37:57 -070076
Wei Jiac6cfd702014-11-11 16:33:20 -080077 uint32_t flags;
78 int64_t durationUs;
79 if (mSource->getDuration(&durationUs) == OK &&
80 durationUs > AUDIO_SINK_MIN_DEEP_BUFFER_DURATION_US) {
81 flags = AUDIO_OUTPUT_FLAG_DEEP_BUFFER;
82 } else {
83 flags = AUDIO_OUTPUT_FLAG_NONE;
84 }
85
86 mRenderer->openAudioSink(
87 format, true /* offloadOnly */, false /* hasVideo */, flags);
Wei Jiabc2fb722014-07-08 16:37:57 -070088}
89
Chong Zhang7137ec72014-11-12 16:41:05 -080090void NuPlayer::DecoderPassThrough::onSetRenderer(
91 const sp<Renderer> &renderer) {
92 // renderer can't be changed during offloading
93 ALOGW_IF(renderer != mRenderer,
94 "ignoring request to change renderer");
95}
96
97void NuPlayer::DecoderPassThrough::onGetInputBuffers(
98 Vector<sp<ABuffer> > * /* dstBuffers */) {
99 ALOGE("onGetInputBuffers() called unexpectedly");
100}
101
Wei Jiabc2fb722014-07-08 16:37:57 -0700102bool NuPlayer::DecoderPassThrough::isStaleReply(const sp<AMessage> &msg) {
103 int32_t generation;
104 CHECK(msg->findInt32("generation", &generation));
105 return generation != mBufferGeneration;
106}
107
Chong Zhang7137ec72014-11-12 16:41:05 -0800108bool NuPlayer::DecoderPassThrough::isCacheFullOrEOS() const {
109 ALOGV("[%s] mCachedBytes = %zu, mReachedEOS = %d",
110 mComponentName.c_str(), mCachedBytes, mReachedEOS);
Wei Jiabc2fb722014-07-08 16:37:57 -0700111
Chong Zhang7137ec72014-11-12 16:41:05 -0800112 return mCachedBytes >= kMaxCachedBytes || mReachedEOS;
Wei Jiabc2fb722014-07-08 16:37:57 -0700113}
114
Chong Zhang7137ec72014-11-12 16:41:05 -0800115void NuPlayer::DecoderPassThrough::doRequestBuffers() {
116 status_t err = OK;
117 while (!isCacheFullOrEOS()) {
118 sp<AMessage> msg = new AMessage();
119
120 err = fetchInputData(msg);
121 if (err != OK) {
122 break;
123 }
124
125 onInputBufferFetched(msg);
126 }
127
128 if (err == -EWOULDBLOCK
129 && mSource->feedMoreTSData() == OK) {
130 scheduleRequestBuffers();
131 }
132}
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) {
227 return err;
228 } else if (err != OK) {
229 if (err == INFO_DISCONTINUITY) {
230 int32_t type;
231 CHECK(accessUnit->meta()->findInt32("discontinuity", &type));
232
233 bool formatChange =
234 (type & ATSParser::DISCONTINUITY_AUDIO_FORMAT) != 0;
235
236 bool timeChange =
237 (type & ATSParser::DISCONTINUITY_TIME) != 0;
238
239 ALOGI("audio discontinuity (formatChange=%d, time=%d)",
240 formatChange, timeChange);
241
242 if (formatChange || timeChange) {
243 sp<AMessage> msg = mNotify->dup();
244 msg->setInt32("what", kWhatInputDiscontinuity);
245 // will perform seamless format change,
246 // only notify NuPlayer to scan sources
247 msg->setInt32("formatChange", false);
248 msg->post();
249 }
250
251 if (timeChange) {
252 onFlush(false /* notifyComplete */);
253 err = OK;
254 } else if (formatChange) {
255 // do seamless format change
256 err = OK;
257 } else {
258 // This stream is unaffected by the discontinuity
259 return -EWOULDBLOCK;
260 }
261 }
262
263 reply->setInt32("err", err);
264 return OK;
265 }
266
267 accessUnit = aggregateBuffer(accessUnit);
268 } while (accessUnit == NULL);
269
270#if 0
271 int64_t mediaTimeUs;
272 CHECK(accessUnit->meta()->findInt64("timeUs", &mediaTimeUs));
273 ALOGV("feeding audio input buffer at media time %.2f secs",
274 mediaTimeUs / 1E6);
275#endif
276
277 reply->setBuffer("buffer", accessUnit);
278
279 return OK;
280}
281
282void NuPlayer::DecoderPassThrough::onInputBufferFetched(
Wei Jiabc2fb722014-07-08 16:37:57 -0700283 const sp<AMessage> &msg) {
284 if (mReachedEOS) {
285 return;
286 }
287
288 sp<ABuffer> buffer;
Chong Zhang7137ec72014-11-12 16:41:05 -0800289 bool hasBuffer = msg->findBuffer("buffer", &buffer);
Wei Jiabc2fb722014-07-08 16:37:57 -0700290 if (buffer == NULL) {
Chong Zhang7137ec72014-11-12 16:41:05 -0800291 int32_t streamErr = ERROR_END_OF_STREAM;
292 CHECK(msg->findInt32("err", &streamErr) || !hasBuffer);
293 if (streamErr == OK) {
294 return;
295 }
296
Wei Jiabc2fb722014-07-08 16:37:57 -0700297 mReachedEOS = true;
Wei Jiac6cfd702014-11-11 16:33:20 -0800298 if (mRenderer != NULL) {
299 mRenderer->queueEOS(true /* audio */, ERROR_END_OF_STREAM);
300 }
Wei Jiabc2fb722014-07-08 16:37:57 -0700301 return;
302 }
303
Wei Jiac6cfd702014-11-11 16:33:20 -0800304 sp<AMessage> extra;
305 if (buffer->meta()->findMessage("extra", &extra) && extra != NULL) {
306 int64_t resumeAtMediaTimeUs;
307 if (extra->findInt64(
308 "resume-at-mediatimeUs", &resumeAtMediaTimeUs)) {
309 ALOGI("[%s] suppressing rendering until %lld us",
310 mComponentName.c_str(), (long long)resumeAtMediaTimeUs);
311 mSkipRenderingUntilMediaTimeUs = resumeAtMediaTimeUs;
312 }
313 }
314
315 int32_t bufferSize = buffer->size();
316 mCachedBytes += bufferSize;
317
318 if (mSkipRenderingUntilMediaTimeUs >= 0) {
319 int64_t timeUs = 0;
320 CHECK(buffer->meta()->findInt64("timeUs", &timeUs));
321
322 if (timeUs < mSkipRenderingUntilMediaTimeUs) {
323 ALOGV("[%s] dropping buffer at time %lld as requested.",
324 mComponentName.c_str(), (long long)timeUs);
325
326 onBufferConsumed(bufferSize);
327 return;
328 }
329
330 mSkipRenderingUntilMediaTimeUs = -1;
331 }
332
333 if (mRenderer == NULL) {
334 onBufferConsumed(bufferSize);
335 return;
336 }
Chong Zhangde01afb2014-08-13 13:48:10 -0700337
Wei Jiabc2fb722014-07-08 16:37:57 -0700338 sp<AMessage> reply = new AMessage(kWhatBufferConsumed, id());
339 reply->setInt32("generation", mBufferGeneration);
Wei Jiac6cfd702014-11-11 16:33:20 -0800340 reply->setInt32("size", bufferSize);
Wei Jiabc2fb722014-07-08 16:37:57 -0700341
Wei Jiac6cfd702014-11-11 16:33:20 -0800342 mRenderer->queueBuffer(true /* audio */, buffer, reply);
343
Phil Burkc5cc2e22014-09-09 20:08:39 -0700344 ++mPendingBuffersToDrain;
Chong Zhang7137ec72014-11-12 16:41:05 -0800345 ALOGV("onInputBufferFilled: #ToDrain = %zu, cachedBytes = %zu",
346 mPendingBuffersToDrain, mCachedBytes);
Wei Jiabc2fb722014-07-08 16:37:57 -0700347}
348
Chong Zhangde01afb2014-08-13 13:48:10 -0700349void NuPlayer::DecoderPassThrough::onBufferConsumed(int32_t size) {
Phil Burkc5cc2e22014-09-09 20:08:39 -0700350 --mPendingBuffersToDrain;
Chong Zhangde01afb2014-08-13 13:48:10 -0700351 mCachedBytes -= size;
Chong Zhang7137ec72014-11-12 16:41:05 -0800352 ALOGV("onBufferConsumed: #ToDrain = %zu, cachedBytes = %zu",
353 mPendingBuffersToDrain, mCachedBytes);
354 onRequestInputBuffers();
Wei Jiabc2fb722014-07-08 16:37:57 -0700355}
356
Chong Zhang7137ec72014-11-12 16:41:05 -0800357void NuPlayer::DecoderPassThrough::onResume() {
358 onRequestInputBuffers();
359}
360
361void NuPlayer::DecoderPassThrough::onFlush(bool notifyComplete) {
Wei Jiabc2fb722014-07-08 16:37:57 -0700362 ++mBufferGeneration;
Wei Jiac6cfd702014-11-11 16:33:20 -0800363 mSkipRenderingUntilMediaTimeUs = -1;
364
365 if (mRenderer != NULL) {
Chong Zhang7137ec72014-11-12 16:41:05 -0800366 mRenderer->flush(true /* audio */, notifyComplete);
367 mRenderer->signalTimeDiscontinuity();
Wei Jiac6cfd702014-11-11 16:33:20 -0800368 }
Wei Jiabc2fb722014-07-08 16:37:57 -0700369
Chong Zhang7137ec72014-11-12 16:41:05 -0800370 if (notifyComplete) {
371 sp<AMessage> notify = mNotify->dup();
372 notify->setInt32("what", kWhatFlushCompleted);
373 notify->post();
374 }
375
Phil Burkc5cc2e22014-09-09 20:08:39 -0700376 mPendingBuffersToDrain = 0;
Chong Zhangde01afb2014-08-13 13:48:10 -0700377 mCachedBytes = 0;
Wei Jiabc2fb722014-07-08 16:37:57 -0700378 mReachedEOS = false;
379}
380
Chong Zhang7137ec72014-11-12 16:41:05 -0800381void NuPlayer::DecoderPassThrough::onShutdown(bool notifyComplete) {
Wei Jiabc2fb722014-07-08 16:37:57 -0700382 ++mBufferGeneration;
Wei Jiac6cfd702014-11-11 16:33:20 -0800383 mSkipRenderingUntilMediaTimeUs = -1;
Wei Jiabc2fb722014-07-08 16:37:57 -0700384
Chong Zhang7137ec72014-11-12 16:41:05 -0800385 if (notifyComplete) {
386 sp<AMessage> notify = mNotify->dup();
387 notify->setInt32("what", kWhatShutdownCompleted);
388 notify->post();
389 }
390
Wei Jiabc2fb722014-07-08 16:37:57 -0700391 mReachedEOS = true;
392}
393
394void NuPlayer::DecoderPassThrough::onMessageReceived(const sp<AMessage> &msg) {
395 ALOGV("[%s] onMessage: %s", mComponentName.c_str(),
396 msg->debugString().c_str());
397
398 switch (msg->what()) {
Wei Jiabc2fb722014-07-08 16:37:57 -0700399 case kWhatBufferConsumed:
400 {
401 if (!isStaleReply(msg)) {
Chong Zhangde01afb2014-08-13 13:48:10 -0700402 int32_t size;
403 CHECK(msg->findInt32("size", &size));
404 onBufferConsumed(size);
Wei Jiabc2fb722014-07-08 16:37:57 -0700405 }
406 break;
407 }
408
Wei Jiabc2fb722014-07-08 16:37:57 -0700409 default:
Chong Zhang7137ec72014-11-12 16:41:05 -0800410 DecoderBase::onMessageReceived(msg);
Wei Jiabc2fb722014-07-08 16:37:57 -0700411 break;
412 }
413}
414
415} // namespace android