From e259531ce59ab1f31de5a23124b22536f6a5a767 Mon Sep 17 00:00:00 2001 From: James Dong Date: Mon, 2 Aug 2010 19:13:40 -0700 Subject: Use audio clock as the reference media clock o Only do this for realtime applications o Adjust other track clock based on audio clock o Assume other track uses wall clock as the media clock o Use some heuristics to reduce the size of stts box by 2/3. - also o Remove one unused key from MetaData.h Change-Id: Ib9432842627b61795b533508158c25258a527332 --- media/libstagefright/MPEG4Writer.cpp | 65 ++++++++++++++++++++++++++++++++++++ 1 file changed, 65 insertions(+) (limited to 'media/libstagefright/MPEG4Writer.cpp') diff --git a/media/libstagefright/MPEG4Writer.cpp b/media/libstagefright/MPEG4Writer.cpp index 1460f37..9fe3864 100644 --- a/media/libstagefright/MPEG4Writer.cpp +++ b/media/libstagefright/MPEG4Writer.cpp @@ -72,6 +72,11 @@ private: bool mIsAudio; bool mIsMPEG4; int64_t mTrackDurationUs; + + // For realtime applications, we need to adjust the media clock + // for video track based on the audio media clock + bool mIsRealTimeRecording; + int64_t mMaxTimeStampUs; int64_t mEstimatedTrackSizeBytes; int64_t mMaxWriteTimeUs; int32_t mTimeScale; @@ -940,6 +945,7 @@ status_t MPEG4Writer::startWriterThread() { mDone = false; mIsFirstChunk = true; + mDriftTimeUs = 0; for (List::iterator it = mTracks.begin(); it != mTracks.end(); ++it) { ChunkInfo info; @@ -967,6 +973,14 @@ status_t MPEG4Writer::Track::start(MetaData *params) { startTimeUs = 0; } + mIsRealTimeRecording = true; + { + int32_t isNotRealTime; + if (params && params->findInt32(kKeyNotRealTime, &isNotRealTime)) { + mIsRealTimeRecording = (isNotRealTime == 0); + } + } + initTrackingProgressStatus(params); sp meta = new MetaData; @@ -1326,6 +1340,10 @@ void MPEG4Writer::Track::threadEntry() { uint32_t previousSampleSize = 0; // Size of the previous sample int64_t previousPausedDurationUs = 0; int64_t timestampUs; + + int64_t wallClockTimeUs = 0; + int64_t lastWallClockTimeUs = 0; + sp meta_data; bool collectStats = collectStatisticalData(); @@ -1429,6 +1447,33 @@ void MPEG4Writer::Track::threadEntry() { } timestampUs -= previousPausedDurationUs; + if (mIsRealTimeRecording && !mIsAudio) { + // The minor adjustment on the timestamp is heuristic/experimental + // We are adjusting the timestamp to reduce the fluctuation of the duration + // of neighboring samples. This in turn helps reduce the track header size, + // especially, the number of entries in the "stts" box. + if (mNumSamples > 1) { + int64_t durationUs = timestampUs + mOwner->getDriftTimeUs() - lastTimestampUs; + int64_t diffUs = (durationUs > lastDurationUs) + ? durationUs - lastDurationUs + : lastDurationUs - durationUs; + if (diffUs <= 5000) { // XXX: Magic number 5ms + timestampUs = lastTimestampUs + lastDurationUs; + } else { + timestampUs += mOwner->getDriftTimeUs(); + } + } + } + CHECK(timestampUs >= 0); + if (mNumSamples > 1) { + if (timestampUs <= lastTimestampUs) { + LOGW("Drop a frame, since it arrives too late!"); + copy->release(); + copy = NULL; + continue; + } + } + LOGV("time stamp: %lld and previous paused duration %lld", timestampUs, previousPausedDurationUs); if (timestampUs > mTrackDurationUs) { @@ -1454,6 +1499,14 @@ void MPEG4Writer::Track::threadEntry() { } lastDurationUs = timestampUs - lastTimestampUs; lastTimestampUs = timestampUs; + if (mIsRealTimeRecording && mIsAudio) { + wallClockTimeUs = systemTime() / 1000; + int64_t wallClockDurationUs = wallClockTimeUs - lastWallClockTimeUs; + if (mNumSamples > 2) { + mOwner->addDriftTimeUs(lastDurationUs - wallClockDurationUs); + } + lastWallClockTimeUs = wallClockTimeUs; + } if (isSync != 0) { mStssTableEntries.push_back(mNumSamples); @@ -1679,6 +1732,18 @@ void MPEG4Writer::Track::logStatisticalData(bool isAudio) { } } +void MPEG4Writer::addDriftTimeUs(int64_t driftTimeUs) { + LOGV("addDriftTimeUs: %lld us", driftTimeUs); + Mutex::Autolock autolock(mLock); + mDriftTimeUs += driftTimeUs; +} + +int64_t MPEG4Writer::getDriftTimeUs() { + LOGV("getDriftTimeUs: %lld us", mDriftTimeUs); + Mutex::Autolock autolock(mLock); + return mDriftTimeUs; +} + void MPEG4Writer::Track::bufferChunk(int64_t timestampUs) { LOGV("bufferChunk"); -- cgit v1.1