/* * 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 #include #include #include #include #include #include #include #include #include #include #include #include #include namespace android { MediaMuxer::MediaMuxer(const char *path, OutputFormat format) : mState(UNINITIALIZED) { if (format == OUTPUT_FORMAT_MPEG_4) { mWriter = new MPEG4Writer(path); mFileMeta = new MetaData; mState = INITIALIZED; } } MediaMuxer::MediaMuxer(int fd, OutputFormat format) : mState(UNINITIALIZED) { if (format == OUTPUT_FORMAT_MPEG_4) { mWriter = new MPEG4Writer(fd); mFileMeta = new MetaData; mState = INITIALIZED; } } MediaMuxer::~MediaMuxer() { Mutex::Autolock autoLock(mMuxerLock); // Clean up all the internal resources. mFileMeta.clear(); mWriter.clear(); mTrackList.clear(); } ssize_t MediaMuxer::addTrack(const sp &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 trackMeta = new MetaData; convertMessageToMetaData(format, trackMeta); sp newTrack = new MediaAdapter(trackMeta); status_t result = mWriter->addSource(newTrack); if (result == OK) { return mTrackList.add(newTrack); } return -1; } 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; } ALOGV("Setting location: latitude = %d, longitude = %d", latitude, longitude); return mWriter->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; } } return mWriter->stop(); } else { ALOGE("stop() is called in invalid state %d", mState); return INVALID_OPERATION; } } status_t MediaMuxer::writeSampleData(const sp &buffer, size_t trackIndex, int64_t timeUs, uint32_t flags) { Mutex::Autolock autoLock(mMuxerLock); if (buffer.get() == NULL) { ALOGE("WriteSampleData() get an NULL buffer."); return -EINVAL; } 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 %d", trackIndex); return -EINVAL; } MediaBuffer* mediaBuffer = new MediaBuffer(buffer); mediaBuffer->add_ref(); // Released in MediaAdapter::signalBufferReturned(). mediaBuffer->set_range(buffer->offset(), buffer->size()); sp 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); } sp currentTrack = mTrackList[trackIndex]; // This pushBuffer will wait until the mediaBuffer is consumed. return currentTrack->pushBuffer(mediaBuffer); } } // namespace android