blob: 3d2b0334b52fecadc66fe9b3f53b8084c0297b45 [file] [log] [blame]
Andreas Huberf9334412010-12-15 15:17:42 -08001/*
2 * Copyright (C) 2010 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef NUPLAYER_RENDERER_H_
18
19#define NUPLAYER_RENDERER_H_
20
Wonsik Kimee8116b2019-07-19 15:55:44 -070021#include <atomic>
22
Lajos Molnar3a474aa2015-04-24 17:10:07 -070023#include <media/AudioResamplerPublic.h>
24#include <media/AVSyncSettings.h>
25
Andreas Huberf9334412010-12-15 15:17:42 -080026#include "NuPlayer.h"
27
28namespace android {
29
Weiyin Jiang35d5af12015-01-28 16:14:02 +080030class AWakeLock;
Wei Jia7b15cb32015-02-03 17:46:06 -080031struct MediaClock;
Wonsik Kim7e34bf52016-08-23 00:09:18 +090032class MediaCodecBuffer;
Dichen Zhangc37b1902018-12-18 11:36:13 -080033struct VideoFrameSchedulerBase;
Andreas Huber5bc087c2010-12-23 10:27:40 -080034
Andreas Huberf9334412010-12-15 15:17:42 -080035struct NuPlayer::Renderer : public AHandler {
Andreas Huberd5e56232013-03-12 11:01:43 -070036 enum Flags {
37 FLAG_REAL_TIME = 1,
Wei Jiabc2fb722014-07-08 16:37:57 -070038 FLAG_OFFLOAD_AUDIO = 2,
Andreas Huberd5e56232013-03-12 11:01:43 -070039 };
Andreas Huberf9334412010-12-15 15:17:42 -080040 Renderer(const sp<MediaPlayerBase::AudioSink> &sink,
Wei Jia0a68f662017-08-30 18:01:26 -070041 const sp<MediaClock> &mediaClock,
Andreas Huberd5e56232013-03-12 11:01:43 -070042 const sp<AMessage> &notify,
43 uint32_t flags = 0);
Andreas Huberf9334412010-12-15 15:17:42 -080044
Wei Jiabc2fb722014-07-08 16:37:57 -070045 static size_t AudioSinkCallback(
46 MediaPlayerBase::AudioSink *audioSink,
47 void *data, size_t size, void *me,
48 MediaPlayerBase::AudioSink::cb_event_t event);
49
Andreas Huberf9334412010-12-15 15:17:42 -080050 void queueBuffer(
51 bool audio,
Wonsik Kim7e34bf52016-08-23 00:09:18 +090052 const sp<MediaCodecBuffer> &buffer,
Andreas Huberf9334412010-12-15 15:17:42 -080053 const sp<AMessage> &notifyConsumed);
54
55 void queueEOS(bool audio, status_t finalResult);
56
Lajos Molnar3a474aa2015-04-24 17:10:07 -070057 status_t setPlaybackSettings(const AudioPlaybackRate &rate /* sanitized */);
58 status_t getPlaybackSettings(AudioPlaybackRate *rate /* nonnull */);
59 status_t setSyncSettings(const AVSyncSettings &sync, float videoFpsHint);
60 status_t getSyncSettings(AVSyncSettings *sync /* nonnull */, float *videoFps /* nonnull */);
Wei Jia98160162015-02-04 17:01:11 -080061
Chong Zhang7137ec72014-11-12 16:41:05 -080062 void flush(bool audio, bool notifyComplete);
Andreas Huberf9334412010-12-15 15:17:42 -080063
64 void signalTimeDiscontinuity();
65
Wei Jiabc2fb722014-07-08 16:37:57 -070066 void signalDisableOffloadAudio();
Ronghua Wua10fd232014-11-06 16:15:20 -080067 void signalEnableOffloadAudio();
Wei Jiabc2fb722014-07-08 16:37:57 -070068
Andreas Huberb4082222011-01-20 15:23:04 -080069 void pause();
70 void resume();
71
Lajos Molnarc851b5d2014-09-18 14:14:29 -070072 void setVideoFrameRate(float fps);
73
Ronghua Wua73d9e02014-10-08 15:13:29 -070074 status_t getCurrentPosition(int64_t *mediaUs);
Ronghua Wua73d9e02014-10-08 15:13:29 -070075 int64_t getVideoLateByUs();
Ronghua Wua73d9e02014-10-08 15:13:29 -070076
Andy Hung202bce12014-12-03 11:47:36 -080077 status_t openAudioSink(
Chong Zhang3b9eb1f2014-10-15 17:05:08 -070078 const sp<AMessage> &format,
79 bool offloadOnly,
80 bool hasVideo,
Andy Hung202bce12014-12-03 11:47:36 -080081 uint32_t flags,
Dhananjay Kumarc387f2b2015-08-06 10:43:16 +053082 bool *isOffloaded,
83 bool isStreaming);
Chong Zhang3b9eb1f2014-10-15 17:05:08 -070084 void closeAudioSink();
85
Wei Jia9a3101b2016-11-08 14:34:24 -080086 // re-open audio sink after all pending audio buffers played.
87 void changeAudioFormat(
88 const sp<AMessage> &format,
89 bool offloadOnly,
90 bool hasVideo,
91 uint32_t flags,
Dhananjay Kumarc387f2b2015-08-06 10:43:16 +053092 bool isStreaming,
Wei Jia9a3101b2016-11-08 14:34:24 -080093 const sp<AMessage> &notify);
94
Andreas Huberf9334412010-12-15 15:17:42 -080095 enum {
Ronghua Wufaeb0f22015-05-21 12:20:21 -070096 kWhatEOS = 'eos ',
97 kWhatFlushComplete = 'fluC',
98 kWhatPosition = 'posi',
99 kWhatVideoRenderingStart = 'vdrd',
100 kWhatMediaRenderingStart = 'mdrd',
101 kWhatAudioTearDown = 'adTD',
Ronghua Wuf5b1db12014-09-09 10:11:08 -0700102 kWhatAudioOffloadPauseTimeout = 'aOPT',
Andreas Huberf9334412010-12-15 15:17:42 -0800103 };
104
Ronghua Wufaeb0f22015-05-21 12:20:21 -0700105 enum AudioTearDownReason {
Wei Jiaa05f1e32016-03-25 16:31:22 -0700106 kDueToError = 0, // Could restart with either offload or non-offload.
Ronghua Wu08529172014-10-02 16:55:52 -0700107 kDueToTimeout,
Wei Jiaa05f1e32016-03-25 16:31:22 -0700108 kForceNonOffload, // Restart only with non-offload.
Ronghua Wu08529172014-10-02 16:55:52 -0700109 };
110
Andreas Huberf9334412010-12-15 15:17:42 -0800111protected:
112 virtual ~Renderer();
113
114 virtual void onMessageReceived(const sp<AMessage> &msg);
115
116private:
117 enum {
Wei Jiabc2fb722014-07-08 16:37:57 -0700118 kWhatDrainAudioQueue = 'draA',
119 kWhatDrainVideoQueue = 'draV',
Lajos Molnard5923402014-10-20 17:47:54 -0700120 kWhatPostDrainVideoQueue = 'pDVQ',
Wei Jiabc2fb722014-07-08 16:37:57 -0700121 kWhatQueueBuffer = 'queB',
122 kWhatQueueEOS = 'qEOS',
Lajos Molnar3a474aa2015-04-24 17:10:07 -0700123 kWhatConfigPlayback = 'cfPB',
124 kWhatConfigSync = 'cfSy',
125 kWhatGetPlaybackSettings = 'gPbS',
126 kWhatGetSyncSettings = 'gSyS',
Wei Jiabc2fb722014-07-08 16:37:57 -0700127 kWhatFlush = 'flus',
Wei Jiabc2fb722014-07-08 16:37:57 -0700128 kWhatPause = 'paus',
129 kWhatResume = 'resm',
Chong Zhang3b9eb1f2014-10-15 17:05:08 -0700130 kWhatOpenAudioSink = 'opnA',
131 kWhatCloseAudioSink = 'clsA',
Wei Jia9a3101b2016-11-08 14:34:24 -0800132 kWhatChangeAudioFormat = 'chgA',
Wei Jiabc2fb722014-07-08 16:37:57 -0700133 kWhatStopAudioSink = 'stpA',
134 kWhatDisableOffloadAudio = 'noOA',
Ronghua Wua10fd232014-11-06 16:15:20 -0800135 kWhatEnableOffloadAudio = 'enOA',
Lajos Molnarc851b5d2014-09-18 14:14:29 -0700136 kWhatSetVideoFrameRate = 'sVFR',
Andreas Huberf9334412010-12-15 15:17:42 -0800137 };
138
Wei Jia9a3101b2016-11-08 14:34:24 -0800139 // if mBuffer != nullptr, it's a buffer containing real data.
140 // else if mNotifyConsumed == nullptr, it's EOS.
141 // else it's a tag for re-opening audio sink in different format.
Andreas Huberf9334412010-12-15 15:17:42 -0800142 struct QueueEntry {
Wonsik Kim7e34bf52016-08-23 00:09:18 +0900143 sp<MediaCodecBuffer> mBuffer;
Wei Jia9a3101b2016-11-08 14:34:24 -0800144 sp<AMessage> mMeta;
Andreas Huberf9334412010-12-15 15:17:42 -0800145 sp<AMessage> mNotifyConsumed;
146 size_t mOffset;
147 status_t mFinalResult;
Lajos Molnard5923402014-10-20 17:47:54 -0700148 int32_t mBufferOrdinal;
Andreas Huberf9334412010-12-15 15:17:42 -0800149 };
150
Andreas Huber714aa7b2011-09-13 08:28:38 -0700151 static const int64_t kMinPositionUpdateDelayUs;
152
Andreas Huberf9334412010-12-15 15:17:42 -0800153 sp<MediaPlayerBase::AudioSink> mAudioSink;
Wei Jiad2f35de2016-02-17 16:48:06 -0800154 bool mUseVirtualAudioSink;
Andreas Huberf9334412010-12-15 15:17:42 -0800155 sp<AMessage> mNotify;
Wei Jiabc2fb722014-07-08 16:37:57 -0700156 Mutex mLock;
Andreas Huberd5e56232013-03-12 11:01:43 -0700157 uint32_t mFlags;
Andreas Huberf9334412010-12-15 15:17:42 -0800158 List<QueueEntry> mAudioQueue;
159 List<QueueEntry> mVideoQueue;
160 uint32_t mNumFramesWritten;
Dichen Zhangc37b1902018-12-18 11:36:13 -0800161 sp<VideoFrameSchedulerBase> mVideoScheduler;
Andreas Huberf9334412010-12-15 15:17:42 -0800162
163 bool mDrainAudioQueuePending;
164 bool mDrainVideoQueuePending;
165 int32_t mAudioQueueGeneration;
166 int32_t mVideoQueueGeneration;
Wei Jia7b15cb32015-02-03 17:46:06 -0800167 int32_t mAudioDrainGeneration;
168 int32_t mVideoDrainGeneration;
Wei Jia7c8d0e02015-08-27 17:40:21 -0700169 int32_t mAudioEOSGeneration;
Andreas Huberf9334412010-12-15 15:17:42 -0800170
Wei Jia0a68f662017-08-30 18:01:26 -0700171 const sp<MediaClock> mMediaClock;
Lajos Molnar3a474aa2015-04-24 17:10:07 -0700172 float mPlaybackRate; // audio track rate
173
174 AudioPlaybackRate mPlaybackSettings;
175 AVSyncSettings mSyncSettings;
176 float mVideoFpsHint;
177
Ronghua Wua73d9e02014-10-08 15:13:29 -0700178 int64_t mAudioFirstAnchorTimeMediaUs;
Ronghua Wueecb7802014-10-19 23:12:50 -0700179 int64_t mAnchorTimeMediaUs;
Lajos Molnarf5926712014-10-21 09:36:43 -0700180 int64_t mAnchorNumFramesWritten;
Ronghua Wua73d9e02014-10-08 15:13:29 -0700181 int64_t mVideoLateByUs;
Wei Jiadc293e32017-10-16 18:29:37 -0700182 int64_t mNextVideoTimeMediaUs;
Ronghua Wua73d9e02014-10-08 15:13:29 -0700183 bool mHasAudio;
184 bool mHasVideo;
Andreas Huberf9334412010-12-15 15:17:42 -0800185
Chong Zhang7137ec72014-11-12 16:41:05 -0800186 bool mNotifyCompleteAudio;
187 bool mNotifyCompleteVideo;
Andreas Huberf9334412010-12-15 15:17:42 -0800188
189 bool mSyncQueues;
190
Wei Jiad4cdba12014-12-18 10:44:17 -0800191 // modified on only renderer's thread.
Andreas Huberb4082222011-01-20 15:23:04 -0800192 bool mPaused;
Andy Hungb03dcb32015-08-27 16:18:59 -0700193 int64_t mPauseDrainAudioAllowedUs; // time when we can drain/deliver audio in pause mode.
Wei Jiad4cdba12014-12-18 10:44:17 -0800194
Chong Zhangfbe8bef2014-08-29 18:34:17 -0700195 bool mVideoSampleReceived;
James Dongf57b4ea2012-07-20 13:38:36 -0700196 bool mVideoRenderingStarted;
Lajos Molnarcbaffcf2013-08-14 18:30:38 -0700197 int32_t mVideoRenderingStartGeneration;
198 int32_t mAudioRenderingStartGeneration;
Andy Hungb12ea0b2015-09-03 15:13:01 -0700199 bool mRenderingDataDelivered;
Andreas Huberb4082222011-01-20 15:23:04 -0800200
Andy Hung528c8402016-01-12 12:39:34 -0800201 int64_t mNextAudioClockUpdateTimeUs;
Wei Jia2995dc72015-07-24 16:00:33 -0700202 // the media timestamp of last audio sample right before EOS.
203 int64_t mLastAudioMediaTimeUs;
Andreas Huber714aa7b2011-09-13 08:28:38 -0700204
Ronghua Wuf5b1db12014-09-09 10:11:08 -0700205 int32_t mAudioOffloadPauseTimeoutGeneration;
Ronghua Wufaeb0f22015-05-21 12:20:21 -0700206 bool mAudioTornDown;
Chong Zhang3b9eb1f2014-10-15 17:05:08 -0700207 audio_offload_info_t mCurrentOffloadInfo;
Ronghua Wuf5b1db12014-09-09 10:11:08 -0700208
Andy Hungf0e83642014-12-19 17:55:56 -0800209 struct PcmInfo {
210 audio_channel_mask_t mChannelMask;
211 audio_output_flags_t mFlags;
212 audio_format_t mFormat;
213 int32_t mNumChannels;
214 int32_t mSampleRate;
215 };
216 PcmInfo mCurrentPcmInfo;
217 static const PcmInfo AUDIO_PCMINFO_INITIALIZER;
218
Lajos Molnard5923402014-10-20 17:47:54 -0700219 int32_t mTotalBuffersQueued;
220 int32_t mLastAudioBufferDrained;
Andy Hunga0b39712015-05-31 22:40:49 -0700221 bool mUseAudioCallback;
Lajos Molnard5923402014-10-20 17:47:54 -0700222
Weiyin Jiang35d5af12015-01-28 16:14:02 +0800223 sp<AWakeLock> mWakeLock;
224
Wonsik Kimee8116b2019-07-19 15:55:44 -0700225 std::atomic_flag mSyncFlag = ATOMIC_FLAG_INIT;
226 Mutex mSyncLock;
227 Condition mSyncCondition;
228 int64_t mSyncCount{0};
229
Wei Jiad4cdba12014-12-18 10:44:17 -0800230 status_t getCurrentPositionOnLooper(int64_t *mediaUs);
231 status_t getCurrentPositionOnLooper(
232 int64_t *mediaUs, int64_t nowUs, bool allowPastQueuedVideo = false);
233 bool getCurrentPositionIfPaused_l(int64_t *mediaUs);
234 status_t getCurrentPositionFromAnchor(
235 int64_t *mediaUs, int64_t nowUs, bool allowPastQueuedVideo = false);
Lajos Molnarf5926712014-10-21 09:36:43 -0700236
Wei Jia4c74fde2016-08-26 16:00:52 -0700237 void notifyEOSCallback();
Wei Jiabc2fb722014-07-08 16:37:57 -0700238 size_t fillAudioBuffer(void *buffer, size_t size);
239
Andreas Huber078cfcf2011-09-15 12:25:04 -0700240 bool onDrainAudioQueue();
Chong Zhang9da0ce42015-06-11 17:18:58 -0700241 void drainAudioQueueUntilLastEOS();
Lajos Molnar06ad1522014-08-28 07:27:44 -0700242 int64_t getPendingAudioPlayoutDurationUs(int64_t nowUs);
Wei Jiabc2fb722014-07-08 16:37:57 -0700243 void postDrainAudioQueue_l(int64_t delayUs = 0);
Andreas Huberf9334412010-12-15 15:17:42 -0800244
Wei Jia9a3101b2016-11-08 14:34:24 -0800245 void clearAnchorTime();
Wei Jia7b15cb32015-02-03 17:46:06 -0800246 void clearAudioFirstAnchorTime_l();
247 void setAudioFirstAnchorTimeIfNeeded_l(int64_t mediaUs);
248 void setVideoLateByUs(int64_t lateUs);
249
Ronghua Wueecb7802014-10-19 23:12:50 -0700250 void onNewAudioMediaTime(int64_t mediaTimeUs);
Ronghua Wua73d9e02014-10-08 15:13:29 -0700251 int64_t getRealTimeUs(int64_t mediaTimeUs, int64_t nowUs);
252
Andreas Huberf9334412010-12-15 15:17:42 -0800253 void onDrainVideoQueue();
Wei Jia7b15cb32015-02-03 17:46:06 -0800254 void postDrainVideoQueue();
Andreas Huberf9334412010-12-15 15:17:42 -0800255
Wei Jia7b15cb32015-02-03 17:46:06 -0800256 void prepareForMediaRenderingStart_l();
257 void notifyIfMediaRenderingStarted_l();
Lajos Molnarcbaffcf2013-08-14 18:30:38 -0700258
Andreas Huberf9334412010-12-15 15:17:42 -0800259 void onQueueBuffer(const sp<AMessage> &msg);
260 void onQueueEOS(const sp<AMessage> &msg);
261 void onFlush(const sp<AMessage> &msg);
Andreas Huber3831a062010-12-21 10:22:33 -0800262 void onAudioSinkChanged();
Wei Jiabc2fb722014-07-08 16:37:57 -0700263 void onDisableOffloadAudio();
Ronghua Wua10fd232014-11-06 16:15:20 -0800264 void onEnableOffloadAudio();
Lajos Molnar3a474aa2015-04-24 17:10:07 -0700265 status_t onConfigPlayback(const AudioPlaybackRate &rate /* sanitized */);
266 status_t onGetPlaybackSettings(AudioPlaybackRate *rate /* nonnull */);
267 status_t onConfigSync(const AVSyncSettings &sync, float videoFpsHint);
268 status_t onGetSyncSettings(AVSyncSettings *sync /* nonnull */, float *videoFps /* nonnull */);
269
Andreas Huberb4082222011-01-20 15:23:04 -0800270 void onPause();
271 void onResume();
Lajos Molnarc851b5d2014-09-18 14:14:29 -0700272 void onSetVideoFrameRate(float fps);
Wei Jia7b15cb32015-02-03 17:46:06 -0800273 int32_t getQueueGeneration(bool audio);
274 int32_t getDrainGeneration(bool audio);
275 bool getSyncQueues();
Ronghua Wufaeb0f22015-05-21 12:20:21 -0700276 void onAudioTearDown(AudioTearDownReason reason);
Andy Hung202bce12014-12-03 11:47:36 -0800277 status_t onOpenAudioSink(
Chong Zhang3b9eb1f2014-10-15 17:05:08 -0700278 const sp<AMessage> &format,
279 bool offloadOnly,
280 bool hasVideo,
Dhananjay Kumarc387f2b2015-08-06 10:43:16 +0530281 uint32_t flags,
282 bool isStreaming);
Chong Zhang3b9eb1f2014-10-15 17:05:08 -0700283 void onCloseAudioSink();
Wei Jia9a3101b2016-11-08 14:34:24 -0800284 void onChangeAudioFormat(const sp<AMessage> &meta, const sp<AMessage> &notify);
Andreas Huberf9334412010-12-15 15:17:42 -0800285
Ronghua Wu5095d702014-08-27 12:05:48 -0700286 void notifyEOS(bool audio, status_t finalResult, int64_t delayUs = 0);
Wei Jiad214a562017-07-10 13:20:55 -0700287 void notifyEOS_l(bool audio, status_t finalResult, int64_t delayUs = 0);
Andreas Huberf9334412010-12-15 15:17:42 -0800288 void notifyFlushComplete(bool audio);
Andreas Huber43c3e6c2011-01-05 12:17:08 -0800289 void notifyPosition();
Andreas Huber3fe62152011-09-16 15:09:22 -0700290 void notifyVideoLateBy(int64_t lateByUs);
James Dongf57b4ea2012-07-20 13:38:36 -0700291 void notifyVideoRenderingStart();
Wei Jiaa05f1e32016-03-25 16:31:22 -0700292 void notifyAudioTearDown(AudioTearDownReason reason);
Andreas Huberf9334412010-12-15 15:17:42 -0800293
294 void flushQueue(List<QueueEntry> *queue);
Wei Jia7b15cb32015-02-03 17:46:06 -0800295 bool dropBufferIfStale(bool audio, const sp<AMessage> &msg);
Wei Jiabc2fb722014-07-08 16:37:57 -0700296 void syncQueuesDone_l();
297
298 bool offloadingAudio() const { return (mFlags & FLAG_OFFLOAD_AUDIO) != 0; }
Andreas Huberf9334412010-12-15 15:17:42 -0800299
Ronghua Wuf5b1db12014-09-09 10:11:08 -0700300 void startAudioOffloadPauseTimeout();
301 void cancelAudioOffloadPauseTimeout();
302
Wei Jia98160162015-02-04 17:01:11 -0800303 int64_t getDurationUsIfPlayedAtSampleRate(uint32_t numFrames);
304
Andreas Huberf9334412010-12-15 15:17:42 -0800305 DISALLOW_EVIL_CONSTRUCTORS(Renderer);
306};
307
Lajos Molnar3a474aa2015-04-24 17:10:07 -0700308} // namespace android
Andreas Huberf9334412010-12-15 15:17:42 -0800309
310#endif // NUPLAYER_RENDERER_H_