blob: b948fe293890af05b1193549907af0e4afb8738a [file] [log] [blame]
ztenghui6269d532013-02-04 15:59:38 -08001/*
2 * Copyright 2013, 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 "MediaMuxer"
19#include <utils/Log.h>
20
21#include <media/stagefright/MediaMuxer.h>
22
23#include <media/stagefright/foundation/ABuffer.h>
24#include <media/stagefright/foundation/ADebug.h>
25#include <media/stagefright/foundation/AMessage.h>
26#include <media/stagefright/MediaAdapter.h>
27#include <media/stagefright/MediaBuffer.h>
28#include <media/stagefright/MediaCodec.h>
29#include <media/stagefright/MediaDefs.h>
30#include <media/stagefright/MediaErrors.h>
31#include <media/stagefright/MediaSource.h>
32#include <media/stagefright/MetaData.h>
33#include <media/stagefright/MPEG4Writer.h>
34#include <media/stagefright/Utils.h>
35
36namespace android {
37
ztenghuiafde4e52013-02-22 14:32:59 -080038MediaMuxer::MediaMuxer(const char *path, OutputFormat format)
ztenghui11287472013-03-12 15:43:56 -070039 : mState(UNINITIALIZED) {
ztenghuiafde4e52013-02-22 14:32:59 -080040 if (format == OUTPUT_FORMAT_MPEG_4) {
41 mWriter = new MPEG4Writer(path);
ztenghui11287472013-03-12 15:43:56 -070042 mFileMeta = new MetaData;
43 mState = INITIALIZED;
ztenghuiafde4e52013-02-22 14:32:59 -080044 }
ztenghui11287472013-03-12 15:43:56 -070045
ztenghui6269d532013-02-04 15:59:38 -080046}
47
ztenghuiafde4e52013-02-22 14:32:59 -080048MediaMuxer::MediaMuxer(int fd, OutputFormat format)
ztenghui11287472013-03-12 15:43:56 -070049 : mState(UNINITIALIZED) {
ztenghuiafde4e52013-02-22 14:32:59 -080050 if (format == OUTPUT_FORMAT_MPEG_4) {
51 mWriter = new MPEG4Writer(fd);
ztenghui11287472013-03-12 15:43:56 -070052 mFileMeta = new MetaData;
53 mState = INITIALIZED;
ztenghuiafde4e52013-02-22 14:32:59 -080054 }
ztenghui6269d532013-02-04 15:59:38 -080055}
56
57MediaMuxer::~MediaMuxer() {
58 Mutex::Autolock autoLock(mMuxerLock);
59
60 // Clean up all the internal resources.
ztenghui11287472013-03-12 15:43:56 -070061 mFileMeta.clear();
ztenghui6269d532013-02-04 15:59:38 -080062 mWriter.clear();
63 mTrackList.clear();
64}
65
66ssize_t MediaMuxer::addTrack(const sp<AMessage> &format) {
67 Mutex::Autolock autoLock(mMuxerLock);
68
69 if (format.get() == NULL) {
70 ALOGE("addTrack() get a null format");
71 return -EINVAL;
72 }
73
ztenghui11287472013-03-12 15:43:56 -070074 if (mState != INITIALIZED) {
ztenghui6269d532013-02-04 15:59:38 -080075 ALOGE("addTrack() must be called after constructor and before start().");
76 return INVALID_OPERATION;
77 }
78
ztenghui11287472013-03-12 15:43:56 -070079 sp<MetaData> trackMeta = new MetaData;
80 convertMessageToMetaData(format, trackMeta);
ztenghui6269d532013-02-04 15:59:38 -080081
ztenghui11287472013-03-12 15:43:56 -070082 sp<MediaAdapter> newTrack = new MediaAdapter(trackMeta);
James Dong5883d532013-03-12 10:40:20 -070083 status_t result = mWriter->addSource(newTrack);
84 if (result == OK) {
85 return mTrackList.add(newTrack);
86 }
87 return -1;
ztenghui6269d532013-02-04 15:59:38 -080088}
89
ztenghui11287472013-03-12 15:43:56 -070090status_t MediaMuxer::setOrientationHint(int degrees) {
91 Mutex::Autolock autoLock(mMuxerLock);
92 if (mState != INITIALIZED) {
93 ALOGE("setOrientationHint() must be called before start().");
94 return INVALID_OPERATION;
95 }
96
97 if (degrees != 0 && degrees != 90 && degrees != 180 && degrees != 270) {
98 ALOGE("setOrientationHint() get invalid degrees");
99 return -EINVAL;
100 }
101
102 mFileMeta->setInt32(kKeyRotation, degrees);
103 return OK;
104}
105
ztenghui6269d532013-02-04 15:59:38 -0800106status_t MediaMuxer::start() {
107 Mutex::Autolock autoLock(mMuxerLock);
ztenghui11287472013-03-12 15:43:56 -0700108 if (mState == INITIALIZED) {
ztenghui6269d532013-02-04 15:59:38 -0800109 mState = STARTED;
ztenghui11287472013-03-12 15:43:56 -0700110 return mWriter->start(mFileMeta.get());
ztenghui6269d532013-02-04 15:59:38 -0800111 } else {
112 ALOGE("start() is called in invalid state %d", mState);
113 return INVALID_OPERATION;
114 }
115}
116
117status_t MediaMuxer::stop() {
118 Mutex::Autolock autoLock(mMuxerLock);
119
120 if (mState == STARTED) {
121 mState = STOPPED;
122 for (size_t i = 0; i < mTrackList.size(); i++) {
James Dong5883d532013-03-12 10:40:20 -0700123 if (mTrackList[i]->stop() != OK) {
124 return INVALID_OPERATION;
125 }
ztenghui6269d532013-02-04 15:59:38 -0800126 }
127 return mWriter->stop();
128 } else {
129 ALOGE("stop() is called in invalid state %d", mState);
130 return INVALID_OPERATION;
131 }
132}
133
134status_t MediaMuxer::writeSampleData(const sp<ABuffer> &buffer, size_t trackIndex,
135 int64_t timeUs, uint32_t flags) {
136 Mutex::Autolock autoLock(mMuxerLock);
137
ztenghui6269d532013-02-04 15:59:38 -0800138 if (buffer.get() == NULL) {
139 ALOGE("WriteSampleData() get an NULL buffer.");
140 return -EINVAL;
141 }
142
143 if (mState != STARTED) {
144 ALOGE("WriteSampleData() is called in invalid state %d", mState);
145 return INVALID_OPERATION;
146 }
147
148 if (trackIndex >= mTrackList.size()) {
149 ALOGE("WriteSampleData() get an invalid index %d", trackIndex);
150 return -EINVAL;
151 }
152
153 MediaBuffer* mediaBuffer = new MediaBuffer(buffer);
154
155 mediaBuffer->add_ref(); // Released in MediaAdapter::signalBufferReturned().
156 mediaBuffer->set_range(buffer->offset(), buffer->size());
157
ztenghui11287472013-03-12 15:43:56 -0700158 sp<MetaData> sampleMetaData = mediaBuffer->meta_data();
159 sampleMetaData->setInt64(kKeyTime, timeUs);
ztenghui6269d532013-02-04 15:59:38 -0800160 // Just set the kKeyDecodingTime as the presentation time for now.
ztenghui11287472013-03-12 15:43:56 -0700161 sampleMetaData->setInt64(kKeyDecodingTime, timeUs);
ztenghui6269d532013-02-04 15:59:38 -0800162
ztenghuiafde4e52013-02-22 14:32:59 -0800163 if (flags & SAMPLE_FLAG_SYNC) {
ztenghui11287472013-03-12 15:43:56 -0700164 sampleMetaData->setInt32(kKeyIsSyncFrame, true);
ztenghui6269d532013-02-04 15:59:38 -0800165 }
166
ztenghuiafde4e52013-02-22 14:32:59 -0800167 sp<MediaAdapter> currentTrack = mTrackList[trackIndex];
ztenghui6269d532013-02-04 15:59:38 -0800168 // This pushBuffer will wait until the mediaBuffer is consumed.
169 return currentTrack->pushBuffer(mediaBuffer);
170}
171
172} // namespace android