MPEG4Writer.h revision 25adc33c1ba77b95162ebce8a4c2a7a07baeb98f
120111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber/*
220111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber * Copyright (C) 2009 The Android Open Source Project
320111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber *
420111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber * Licensed under the Apache License, Version 2.0 (the "License");
520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber * you may not use this file except in compliance with the License.
620111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber * You may obtain a copy of the License at
720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber *
820111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber *      http://www.apache.org/licenses/LICENSE-2.0
920111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber *
1020111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber * Unless required by applicable law or agreed to in writing, software
1120111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber * distributed under the License is distributed on an "AS IS" BASIS,
1220111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1320111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber * See the License for the specific language governing permissions and
1420111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber * limitations under the License.
1520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber */
1620111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
1720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber#ifndef MPEG4_WRITER_H_
1820111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
1920111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber#define MPEG4_WRITER_H_
2020111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
2120111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber#include <stdio.h>
2220111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
23b2487f03f12dcafdb801fc0007c8df8412397f44Marco Nelissen#include <media/IMediaSource.h>
242dce41ad26cb3e9e15c9e456a84bcf5309548ca0Andreas Huber#include <media/stagefright/MediaWriter.h>
2520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber#include <utils/List.h>
2620111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber#include <utils/threads.h>
2720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
2820111aa043c5f404472bc63b90bc5aad906b1101Andreas Hubernamespace android {
2920111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
30e76dba7af9589d9ed7b116eec3a74168a8352925Chong Zhangclass AMessage;
3120111aa043c5f404472bc63b90bc5aad906b1101Andreas Huberclass MediaBuffer;
3220111aa043c5f404472bc63b90bc5aad906b1101Andreas Huberclass MetaData;
3320111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
342dce41ad26cb3e9e15c9e456a84bcf5309548ca0Andreas Huberclass MPEG4Writer : public MediaWriter {
3520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huberpublic:
3630ab66297501757d745b9ae10da61adcd891f497Andreas Huber    MPEG4Writer(int fd);
3720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
38acc47642e0f5d962f6289e6ba687fabf68f8312bJames Dong    // Limitations
3925adc33c1ba77b95162ebce8a4c2a7a07baeb98fHangyu Kuang    // No more than one video and/or one audio source can be added, but
4025adc33c1ba77b95162ebce8a4c2a7a07baeb98fHangyu Kuang    // multiple metadata sources can be added.
41b2487f03f12dcafdb801fc0007c8df8412397f44Marco Nelissen    virtual status_t addSource(const sp<IMediaSource> &source);
42acc47642e0f5d962f6289e6ba687fabf68f8312bJames Dong
43acc47642e0f5d962f6289e6ba687fabf68f8312bJames Dong    // Returns INVALID_OPERATION if there is no source or track.
442dec2b5be2056c6d9428897dc672185872d30d17James Dong    virtual status_t start(MetaData *param = NULL);
458bcc65c753085fe3328592cceda0cf0e8f8b0a45James Dong    virtual status_t stop() { return reset(); }
4637187916a486504acaf83bea30147eb5fbf46ae5James Dong    virtual status_t pause();
472dce41ad26cb3e9e15c9e456a84bcf5309548ca0Andreas Huber    virtual bool reachedEOS();
48dedf414d3fe2e79ee0aad0f1c82ca16ebd886ff6James Dong    virtual status_t dump(int fd, const Vector<String16>& args);
4920111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
5020111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    void beginBox(const char *fourcc);
51e76dba7af9589d9ed7b116eec3a74168a8352925Chong Zhang    void beginBox(uint32_t id);
5220111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    void writeInt8(int8_t x);
5320111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    void writeInt16(int16_t x);
5420111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    void writeInt32(int32_t x);
5520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    void writeInt64(int64_t x);
5620111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    void writeCString(const char *s);
5720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    void writeFourcc(const char *fourcc);
5820111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    void write(const void *data, size_t size);
598c460498c028888c533ab442be12b6d4b669b965James Dong    inline size_t write(const void *ptr, size_t size, size_t nmemb);
6020111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    void endBox();
6113aec890216948b0c364f8f92792129d0335f506James Dong    uint32_t interleaveDuration() const { return mInterleaveDurationUs; }
6213aec890216948b0c364f8f92792129d0335f506James Dong    status_t setInterleaveDuration(uint32_t duration);
638f5f2fcee5c12d08df71d17017410c50951fc2e3James Dong    int32_t getTimeScale() const { return mTimeScale; }
6420111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
6507b1bb529a1ae76c46a71b01338c166f9490629dJames Dong    status_t setGeoData(int latitudex10000, int longitudex10000);
66e76dba7af9589d9ed7b116eec3a74168a8352925Chong Zhang    status_t setCaptureRate(float captureFps);
674dbff11975e737482537e1636051690188f3fbc4Praveen Chavan    status_t setTemporalLayerCount(uint32_t layerCount);
68d8cf55d878edddfc36bb821a95b88dfb2453c2c3Robert Shih    virtual void setStartTimeOffsetMs(int ms) { mStartTimeOffsetMs = ms; }
69d8cf55d878edddfc36bb821a95b88dfb2453c2c3Robert Shih    virtual int32_t getStartTimeOffsetMs() const { return mStartTimeOffsetMs; }
7007b1bb529a1ae76c46a71b01338c166f9490629dJames Dong
71693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huberprotected:
72693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber    virtual ~MPEG4Writer();
73693d271e62a3726689ff68f4505ba49228eb94b2Andreas Huber
7420111aa043c5f404472bc63b90bc5aad906b1101Andreas Huberprivate:
7520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    class Track;
7620111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
77c7fc37a3dab9bd1f96713649f351b5990e6316ffJames Dong    int  mFd;
78674ebd0b4e1143e38392a4e3bb38b4679a4577bcJames Dong    status_t mInitCheck;
79de05c8eab188e98798f2b9c3dfac53dbc18ef584ztenghui    bool mIsRealTimeRecording;
80b4d5320bda29cd9694461c9b61d0211f801ff0afJames Dong    bool mUse4ByteNalLength;
811acfe8649f8169caf2ff098c2dc2de880d9a3760James Dong    bool mUse32BitOffset;
82a007e8229fb2be4866c483f9cd6c4af238a2da5eJames Dong    bool mIsFileSizeLimitExplicitlyRequested;
83a7d1a2dd776bf356c228785a94ba8e0ff6a2ec7fJames Dong    bool mPaused;
84411ba422e3635d534928ffd81abf54f4f291c739James Dong    bool mStarted;  // Writer thread + track threads started successfully
85411ba422e3635d534928ffd81abf54f4f291c739James Dong    bool mWriterThreadStarted;  // Only writer thread started successfully
86c7fc37a3dab9bd1f96713649f351b5990e6316ffJames Dong    off64_t mOffset;
8720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    off_t mMdatOffset;
887837c17063a4c50bc856ba59418516fdab731de7James Dong    uint8_t *mMoovBoxBuffer;
89c7fc37a3dab9bd1f96713649f351b5990e6316ffJames Dong    off64_t mMoovBoxBufferOffset;
907837c17063a4c50bc856ba59418516fdab731de7James Dong    bool  mWriteMoovBoxToMemory;
91c7fc37a3dab9bd1f96713649f351b5990e6316ffJames Dong    off64_t mFreeBoxOffset;
927837c17063a4c50bc856ba59418516fdab731de7James Dong    bool mStreamableFile;
93c7fc37a3dab9bd1f96713649f351b5990e6316ffJames Dong    off64_t mEstimatedMoovBoxSize;
947c4820d23b68f748b8dfff3d4bb5b13e9d4811a8Chong Zhang    off64_t mMoovExtraSize;
9513aec890216948b0c364f8f92792129d0335f506James Dong    uint32_t mInterleaveDurationUs;
968f5f2fcee5c12d08df71d17017410c50951fc2e3James Dong    int32_t mTimeScale;
973c0131f02b6f008321608044c53bccce2ac5f6ddJames Dong    int64_t mStartTimestampUs;
9807b1bb529a1ae76c46a71b01338c166f9490629dJames Dong    int mLatitudex10000;
9907b1bb529a1ae76c46a71b01338c166f9490629dJames Dong    int mLongitudex10000;
10025adc33c1ba77b95162ebce8a4c2a7a07baeb98fHangyu Kuang    bool mHasAudioTrack;
10125adc33c1ba77b95162ebce8a4c2a7a07baeb98fHangyu Kuang    bool mHasVideoTrack;
10207b1bb529a1ae76c46a71b01338c166f9490629dJames Dong    bool mAreGeoTagsAvailable;
10386b7f47aa7482424cf8fd248f1315311919be3b0James Dong    int32_t mStartTimeOffsetMs;
1041c9747a4653aec1395c2bd6896c9b87cb5447837James Dong
10520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    Mutex mLock;
10620111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
10720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    List<Track *> mTracks;
10820111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
109c7fc37a3dab9bd1f96713649f351b5990e6316ffJames Dong    List<off64_t> mBoxes;
11020111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
111e76dba7af9589d9ed7b116eec3a74168a8352925Chong Zhang    sp<AMessage> mMetaKeys;
112e76dba7af9589d9ed7b116eec3a74168a8352925Chong Zhang
113f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dong    void setStartTimestampUs(int64_t timeUs);
114f60cafe0e6aad8f9ce54660fa88b651ae4e749e6James Dong    int64_t getStartTimestampUs();  // Not const
11593d6b102a13afa23bfa80d74c399d93d542e6ad6James Dong    status_t startTracks(MetaData *params);
11658ae9c530247668f8af36e30d228c716c226b3d4James Dong    size_t numTracks();
1172dec2b5be2056c6d9428897dc672185872d30d17James Dong    int64_t estimateMoovBoxSize(int32_t bitRate);
1183c0131f02b6f008321608044c53bccce2ac5f6ddJames Dong
1191c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    struct Chunk {
1201c9747a4653aec1395c2bd6896c9b87cb5447837James Dong        Track               *mTrack;        // Owner
1211c9747a4653aec1395c2bd6896c9b87cb5447837James Dong        int64_t             mTimeStampUs;   // Timestamp of the 1st sample
1221c9747a4653aec1395c2bd6896c9b87cb5447837James Dong        List<MediaBuffer *> mSamples;       // Sample data
1231c9747a4653aec1395c2bd6896c9b87cb5447837James Dong
1241c9747a4653aec1395c2bd6896c9b87cb5447837James Dong        // Convenient constructor
125fcac8fa9421f442f024018628a6042d7a14fbfb0James Dong        Chunk(): mTrack(NULL), mTimeStampUs(0) {}
126fcac8fa9421f442f024018628a6042d7a14fbfb0James Dong
1271c9747a4653aec1395c2bd6896c9b87cb5447837James Dong        Chunk(Track *track, int64_t timeUs, List<MediaBuffer *> samples)
1281c9747a4653aec1395c2bd6896c9b87cb5447837James Dong            : mTrack(track), mTimeStampUs(timeUs), mSamples(samples) {
1291c9747a4653aec1395c2bd6896c9b87cb5447837James Dong        }
1301c9747a4653aec1395c2bd6896c9b87cb5447837James Dong
1311c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    };
1321c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    struct ChunkInfo {
1331c9747a4653aec1395c2bd6896c9b87cb5447837James Dong        Track               *mTrack;        // Owner
1341c9747a4653aec1395c2bd6896c9b87cb5447837James Dong        List<Chunk>         mChunks;        // Remaining chunks to be written
13570ccfd44c4bfe95ddecc2df6c3695efc48229d72James Dong
13670ccfd44c4bfe95ddecc2df6c3695efc48229d72James Dong        // Previous chunk timestamp that has been written
13770ccfd44c4bfe95ddecc2df6c3695efc48229d72James Dong        int64_t mPrevChunkTimestampUs;
13870ccfd44c4bfe95ddecc2df6c3695efc48229d72James Dong
13970ccfd44c4bfe95ddecc2df6c3695efc48229d72James Dong        // Max time interval between neighboring chunks
14070ccfd44c4bfe95ddecc2df6c3695efc48229d72James Dong        int64_t mMaxInterChunkDurUs;
14170ccfd44c4bfe95ddecc2df6c3695efc48229d72James Dong
1421c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    };
1431c9747a4653aec1395c2bd6896c9b87cb5447837James Dong
1441c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    bool            mIsFirstChunk;
1451c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    volatile bool   mDone;                  // Writer thread is done?
1461c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    pthread_t       mThread;                // Thread id for the writer
1471c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    List<ChunkInfo> mChunkInfos;            // Chunk infos
1481c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    Condition       mChunkReadyCondition;   // Signal that chunks are available
1491c9747a4653aec1395c2bd6896c9b87cb5447837James Dong
1501c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    // Writer thread handling
1511c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    status_t startWriterThread();
1521c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    void stopWriterThread();
1531c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    static void *ThreadWrapper(void *me);
1541c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    void threadFunc();
1551c9747a4653aec1395c2bd6896c9b87cb5447837James Dong
1561c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    // Buffer a single chunk to be written out later.
1571c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    void bufferChunk(const Chunk& chunk);
1581c9747a4653aec1395c2bd6896c9b87cb5447837James Dong
1591c9747a4653aec1395c2bd6896c9b87cb5447837James Dong    // Write all buffered chunks from all tracks
160fcac8fa9421f442f024018628a6042d7a14fbfb0James Dong    void writeAllChunks();
1611c9747a4653aec1395c2bd6896c9b87cb5447837James Dong
162fcac8fa9421f442f024018628a6042d7a14fbfb0James Dong    // Retrieve the proper chunk to write if there is one
163fcac8fa9421f442f024018628a6042d7a14fbfb0James Dong    // Return true if a chunk is found; otherwise, return false.
164fcac8fa9421f442f024018628a6042d7a14fbfb0James Dong    bool findChunkToWrite(Chunk *chunk);
1651c9747a4653aec1395c2bd6896c9b87cb5447837James Dong
166fcac8fa9421f442f024018628a6042d7a14fbfb0James Dong    // Actually write the given chunk to the file.
167fcac8fa9421f442f024018628a6042d7a14fbfb0James Dong    void writeChunkToFile(Chunk* chunk);
1681c9747a4653aec1395c2bd6896c9b87cb5447837James Dong
169e259531ce59ab1f31de5a23124b22536f6a5a767James Dong    // Adjust other track media clock (presumably wall clock)
170e259531ce59ab1f31de5a23124b22536f6a5a767James Dong    // based on audio track media clock with the drift time.
171e259531ce59ab1f31de5a23124b22536f6a5a767James Dong    int64_t mDriftTimeUs;
172d707fcb3e29707ca4a5935c294ef0b38eb5aba5fJames Dong    void setDriftTimeUs(int64_t driftTimeUs);
173e259531ce59ab1f31de5a23124b22536f6a5a767James Dong    int64_t getDriftTimeUs();
174e259531ce59ab1f31de5a23124b22536f6a5a767James Dong
175b4d5320bda29cd9694461c9b61d0211f801ff0afJames Dong    // Return whether the nal length is 4 bytes or 2 bytes
176b4d5320bda29cd9694461c9b61d0211f801ff0afJames Dong    // Only makes sense for H.264/AVC
177b4d5320bda29cd9694461c9b61d0211f801ff0afJames Dong    bool useNalLengthFour();
178b4d5320bda29cd9694461c9b61d0211f801ff0afJames Dong
179de05c8eab188e98798f2b9c3dfac53dbc18ef584ztenghui    // Return whether the writer is used for real time recording.
180de05c8eab188e98798f2b9c3dfac53dbc18ef584ztenghui    // In real time recording mode, new samples will be allowed to buffered into
181de05c8eab188e98798f2b9c3dfac53dbc18ef584ztenghui    // chunks in higher priority thread, even though the file writer has not
182de05c8eab188e98798f2b9c3dfac53dbc18ef584ztenghui    // drained the chunks yet.
183de05c8eab188e98798f2b9c3dfac53dbc18ef584ztenghui    // By default, real time recording is on.
184de05c8eab188e98798f2b9c3dfac53dbc18ef584ztenghui    bool isRealTimeRecording() const;
185de05c8eab188e98798f2b9c3dfac53dbc18ef584ztenghui
18613aec890216948b0c364f8f92792129d0335f506James Dong    void lock();
18713aec890216948b0c364f8f92792129d0335f506James Dong    void unlock();
18813aec890216948b0c364f8f92792129d0335f506James Dong
18913aec890216948b0c364f8f92792129d0335f506James Dong    // Acquire lock before calling these methods
190c7fc37a3dab9bd1f96713649f351b5990e6316ffJames Dong    off64_t addSample_l(MediaBuffer *buffer);
191c7fc37a3dab9bd1f96713649f351b5990e6316ffJames Dong    off64_t addLengthPrefixedSample_l(MediaBuffer *buffer);
19280f78b773f92048944a850efb7b60629643370cdPraveen Chavan    off64_t addMultipleLengthPrefixedSamples_l(MediaBuffer *buffer);
19320111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
194d599cd4573b5a2d5914c5040e0565ef866749b77James Dong    bool exceedsFileSizeLimit();
1951f90c4b3fda9dbd2824d67fd679f850572f114dcJames Dong    bool use32BitFileOffset() const;
196d599cd4573b5a2d5914c5040e0565ef866749b77James Dong    bool exceedsFileDurationLimit();
19778a1a286f736888ae7af8860b2c424af0d978848James Dong    bool isFileStreamable() const;
198bc07bcc65e91c7aea9713631ae67760dcf1b0286James Dong    void trackProgressStatus(size_t trackId, int64_t timeUs, status_t err = OK);
19913f6284305e4b27395a23db7882d670bdb1bcae1James Dong    void writeCompositionMatrix(int32_t degrees);
200b21c564ce47041f9dd3ab65e36fed57c4937a42dJames Dong    void writeMvhdBox(int64_t durationUs);
201b21c564ce47041f9dd3ab65e36fed57c4937a42dJames Dong    void writeMoovBox(int64_t durationUs);
2022cf9c5073ca3342ee52673ad68763fadd2c2be79James Dong    void writeFtypBox(MetaData *param);
20307b1bb529a1ae76c46a71b01338c166f9490629dJames Dong    void writeUdtaBox();
20407b1bb529a1ae76c46a71b01338c166f9490629dJames Dong    void writeGeoDataBox();
20507b1bb529a1ae76c46a71b01338c166f9490629dJames Dong    void writeLatitude(int degreex10000);
20607b1bb529a1ae76c46a71b01338c166f9490629dJames Dong    void writeLongitude(int degreex10000);
2077c4820d23b68f748b8dfff3d4bb5b13e9d4811a8Chong Zhang
2087c4820d23b68f748b8dfff3d4bb5b13e9d4811a8Chong Zhang    void addDeviceMeta();
209e76dba7af9589d9ed7b116eec3a74168a8352925Chong Zhang    void writeHdlr();
210e76dba7af9589d9ed7b116eec3a74168a8352925Chong Zhang    void writeKeys();
211e76dba7af9589d9ed7b116eec3a74168a8352925Chong Zhang    void writeIlst();
212e76dba7af9589d9ed7b116eec3a74168a8352925Chong Zhang    void writeMetaBox();
21370ccfd44c4bfe95ddecc2df6c3695efc48229d72James Dong    void sendSessionSummary();
214411ba422e3635d534928ffd81abf54f4f291c739James Dong    void release();
2158bcc65c753085fe3328592cceda0cf0e8f8b0a45James Dong    status_t reset();
2167837c17063a4c50bc856ba59418516fdab731de7James Dong
217efcdf187baab9ec29e12adcf85e63fe3e6d5aa4dJohannes Carlsson    static uint32_t getMpeg4Time();
218efcdf187baab9ec29e12adcf85e63fe3e6d5aa4dJohannes Carlsson
21920111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    MPEG4Writer(const MPEG4Writer &);
22020111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    MPEG4Writer &operator=(const MPEG4Writer &);
22120111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber};
22220111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
22320111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber}  // namespace android
22420111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
22520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber#endif  // MPEG4_WRITER_H_
226