| /* |
| * Copyright 2013, The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| //#define LOG_NDEBUG 0 |
| #define LOG_TAG "MediaMuxer" |
| |
| #include "webm/WebmWriter.h" |
| |
| #include <utils/Log.h> |
| |
| #include <media/stagefright/MediaMuxer.h> |
| |
| #include <media/mediarecorder.h> |
| #include <media/stagefright/MediaSource.h> |
| #include <media/stagefright/foundation/ABuffer.h> |
| #include <media/stagefright/foundation/ADebug.h> |
| #include <media/stagefright/foundation/AMessage.h> |
| #include <media/stagefright/MediaAdapter.h> |
| #include <media/stagefright/MediaBuffer.h> |
| #include <media/stagefright/MediaCodec.h> |
| #include <media/stagefright/MediaDefs.h> |
| #include <media/stagefright/MediaErrors.h> |
| #include <media/stagefright/MetaData.h> |
| #include <media/stagefright/MPEG4Writer.h> |
| #include <media/stagefright/OggWriter.h> |
| #include <media/stagefright/Utils.h> |
| |
| namespace android { |
| |
| static bool isMp4Format(MediaMuxer::OutputFormat format) { |
| return format == MediaMuxer::OUTPUT_FORMAT_MPEG_4 || |
| format == MediaMuxer::OUTPUT_FORMAT_THREE_GPP || |
| format == MediaMuxer::OUTPUT_FORMAT_HEIF; |
| } |
| |
| MediaMuxer* MediaMuxer::create(int fd, OutputFormat format) { |
| bool isInputValid = true; |
| if (isMp4Format(format)) { |
| isInputValid = MPEG4Writer::isFdOpenModeValid(fd); |
| } else if (format == OUTPUT_FORMAT_WEBM) { |
| isInputValid = WebmWriter::isFdOpenModeValid(fd); |
| } else if (format == OUTPUT_FORMAT_OGG) { |
| isInputValid = OggWriter::isFdOpenModeValid(fd); |
| } else { |
| ALOGE("MediaMuxer does not support output format %d", format); |
| return nullptr; |
| } |
| if (!isInputValid) { |
| ALOGE("File descriptor is not suitable for format %d", format); |
| return nullptr; |
| } |
| |
| MediaMuxer *muxer = new (std::nothrow) MediaMuxer(fd, (MediaMuxer::OutputFormat)format); |
| if (muxer == nullptr) { |
| ALOGE("Failed to create writer object"); |
| } |
| return muxer; |
| } |
| |
| MediaMuxer::MediaMuxer(int fd, OutputFormat format) |
| : mFormat(format), |
| mState(UNINITIALIZED) { |
| if (isMp4Format(format)) { |
| mWriter = new MPEG4Writer(fd); |
| } else if (format == OUTPUT_FORMAT_WEBM) { |
| mWriter = new WebmWriter(fd); |
| } else if (format == OUTPUT_FORMAT_OGG) { |
| mWriter = new OggWriter(fd); |
| } |
| |
| if (mWriter != NULL) { |
| mFileMeta = new MetaData; |
| if (format == OUTPUT_FORMAT_HEIF) { |
| // Note that the key uses recorder file types. |
| mFileMeta->setInt32(kKeyFileType, output_format::OUTPUT_FORMAT_HEIF); |
| } else if (format == OUTPUT_FORMAT_OGG) { |
| mFileMeta->setInt32(kKeyFileType, output_format::OUTPUT_FORMAT_OGG); |
| } |
| mState = INITIALIZED; |
| } |
| } |
| |
| MediaMuxer::~MediaMuxer() { |
| Mutex::Autolock autoLock(mMuxerLock); |
| |
| // Clean up all the internal resources. |
| mFileMeta.clear(); |
| mWriter.clear(); |
| mTrackList.clear(); |
| mFormatList.clear(); |
| } |
| |
| ssize_t MediaMuxer::addTrack(const sp<AMessage> &format) { |
| Mutex::Autolock autoLock(mMuxerLock); |
| |
| if (format.get() == NULL) { |
| ALOGE("addTrack() get a null format"); |
| return -EINVAL; |
| } |
| |
| if (mState != INITIALIZED) { |
| ALOGE("addTrack() must be called after constructor and before start()."); |
| return INVALID_OPERATION; |
| } |
| |
| sp<MetaData> trackMeta = new MetaData; |
| if (convertMessageToMetaData(format, trackMeta) != OK) { |
| return BAD_VALUE; |
| } |
| |
| sp<MediaAdapter> newTrack = new MediaAdapter(trackMeta); |
| status_t result = mWriter->addSource(newTrack); |
| if (result != OK) { |
| return -1; |
| } |
| float captureFps = -1.0; |
| if (format->findAsFloat("time-lapse-fps", &captureFps)) { |
| ALOGV("addTrack() time-lapse-fps: %f", captureFps); |
| result = mWriter->setCaptureRate(captureFps); |
| if (result != OK) { |
| ALOGW("addTrack() setCaptureRate failed :%d", result); |
| } |
| } |
| |
| mFormatList.add(format); |
| return mTrackList.add(newTrack); |
| } |
| |
| status_t MediaMuxer::setOrientationHint(int degrees) { |
| Mutex::Autolock autoLock(mMuxerLock); |
| if (mState != INITIALIZED) { |
| ALOGE("setOrientationHint() must be called before start()."); |
| return INVALID_OPERATION; |
| } |
| |
| if (degrees != 0 && degrees != 90 && degrees != 180 && degrees != 270) { |
| ALOGE("setOrientationHint() get invalid degrees"); |
| return -EINVAL; |
| } |
| |
| mFileMeta->setInt32(kKeyRotation, degrees); |
| return OK; |
| } |
| |
| status_t MediaMuxer::setLocation(int latitude, int longitude) { |
| Mutex::Autolock autoLock(mMuxerLock); |
| if (mState != INITIALIZED) { |
| ALOGE("setLocation() must be called before start()."); |
| return INVALID_OPERATION; |
| } |
| if (!isMp4Format(mFormat)) { |
| ALOGE("setLocation() is only supported for .mp4, .3gp, .heic or .avif output."); |
| return INVALID_OPERATION; |
| } |
| |
| ALOGV("Setting location: latitude = %d, longitude = %d", latitude, longitude); |
| return static_cast<MPEG4Writer*>(mWriter.get())->setGeoData(latitude, longitude); |
| } |
| |
| status_t MediaMuxer::start() { |
| Mutex::Autolock autoLock(mMuxerLock); |
| if (mState == INITIALIZED) { |
| mState = STARTED; |
| mFileMeta->setInt32(kKeyRealTimeRecording, false); |
| return mWriter->start(mFileMeta.get()); |
| } else { |
| ALOGE("start() is called in invalid state %d", mState); |
| return INVALID_OPERATION; |
| } |
| } |
| |
| status_t MediaMuxer::stop() { |
| Mutex::Autolock autoLock(mMuxerLock); |
| if (mState == STARTED) { |
| mState = STOPPED; |
| for (size_t i = 0; i < mTrackList.size(); i++) { |
| if (mTrackList[i]->stop() != OK) { |
| return INVALID_OPERATION; |
| } |
| } |
| status_t err = mWriter->stop(); |
| if (err != OK) { |
| ALOGE("stop() err: %d", err); |
| } |
| return err; |
| } else { |
| ALOGE("stop() is called in invalid state %d", mState); |
| return INVALID_OPERATION; |
| } |
| } |
| |
| status_t MediaMuxer::writeSampleData(const sp<ABuffer> &buffer, size_t trackIndex, |
| int64_t timeUs, uint32_t flags) { |
| if (buffer.get() == NULL) { |
| ALOGE("WriteSampleData() get an NULL buffer."); |
| return -EINVAL; |
| } |
| if (!mWriter->isSampleMetadataValid(trackIndex, timeUs)) { |
| return -EINVAL; |
| } |
| { |
| /* As MediaMuxer's writeSampleData handles inputs from multiple tracks, |
| * limited the scope of mMuxerLock to this inner block so that the |
| * current track's buffer does not wait until the completion |
| * of processing of previous buffer of the same or another track. |
| * It's the responsibility of individual track - MediaAdapter object |
| * to gate its buffers. |
| */ |
| Mutex::Autolock autoLock(mMuxerLock); |
| if (mState != STARTED) { |
| ALOGE("WriteSampleData() is called in invalid state %d", mState); |
| return INVALID_OPERATION; |
| } |
| } |
| |
| if (trackIndex >= mTrackList.size()) { |
| ALOGE("WriteSampleData() get an invalid index %zu", trackIndex); |
| return -EINVAL; |
| } |
| |
| MediaBuffer* mediaBuffer = new MediaBuffer(buffer); |
| |
| mediaBuffer->add_ref(); // Released in MediaAdapter::signalBufferReturned(). |
| mediaBuffer->set_range(buffer->offset(), buffer->size()); |
| |
| MetaDataBase &sampleMetaData = mediaBuffer->meta_data(); |
| sampleMetaData.setInt64(kKeyTime, timeUs); |
| // Just set the kKeyDecodingTime as the presentation time for now. |
| sampleMetaData.setInt64(kKeyDecodingTime, timeUs); |
| |
| if (flags & MediaCodec::BUFFER_FLAG_SYNCFRAME) { |
| sampleMetaData.setInt32(kKeyIsSyncFrame, true); |
| } |
| |
| if (flags & MediaCodec::BUFFER_FLAG_MUXER_DATA) { |
| sampleMetaData.setInt32(kKeyIsMuxerData, 1); |
| } |
| |
| if (flags & MediaCodec::BUFFER_FLAG_CODECCONFIG) { |
| sampleMetaData.setInt32(kKeyIsCodecConfig, true); |
| ALOGV("BUFFER_FLAG_CODEC_CONFIG"); |
| } |
| |
| if (flags & MediaCodec::BUFFER_FLAG_EOS) { |
| sampleMetaData.setInt32(kKeyIsEndOfStream, 1); |
| ALOGV("BUFFER_FLAG_EOS"); |
| } |
| |
| sp<AMessage> bufMeta = buffer->meta(); |
| int64_t val64; |
| if (bufMeta->findInt64("sample-file-offset", &val64)) { |
| sampleMetaData.setInt64(kKeySampleFileOffset, val64); |
| } |
| if (bufMeta->findInt64( |
| "last-sample-index-in-chunk" /*AMEDIAFORMAT_KEY_LAST_SAMPLE_INDEX_IN_CHUNK*/, |
| &val64)) { |
| sampleMetaData.setInt64(kKeyLastSampleIndexInChunk, val64); |
| } |
| |
| sp<MediaAdapter> currentTrack = mTrackList[trackIndex]; |
| // This pushBuffer will wait until the mediaBuffer is consumed. |
| return currentTrack->pushBuffer(mediaBuffer); |
| } |
| |
| ssize_t MediaMuxer::getTrackCount() { |
| Mutex::Autolock autoLock(mMuxerLock); |
| if (mState != INITIALIZED && mState != STARTED) { |
| ALOGE("getTrackCount() must be called either in INITIALIZED or STARTED state"); |
| return -1; |
| } |
| return mTrackList.size(); |
| } |
| |
| sp<AMessage> MediaMuxer::getTrackFormat([[maybe_unused]] size_t idx) { |
| Mutex::Autolock autoLock(mMuxerLock); |
| if (mState != INITIALIZED && mState != STARTED) { |
| ALOGE("getTrackFormat() must be called either in INITIALIZED or STARTED state"); |
| return nullptr; |
| } |
| if (idx < 0 || idx >= mFormatList.size()) { |
| ALOGE("getTrackFormat() idx is out of range"); |
| return nullptr; |
| } |
| return mFormatList[idx]; |
| } |
| |
| } // namespace android |