SoftMPEG4.cpp revision 3a3c3f7fc658ef874f82e46857ad9df3616aac95
1bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber/*
2bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber * Copyright (C) 2011 The Android Open Source Project
3bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber *
4bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber * Licensed under the Apache License, Version 2.0 (the "License");
5bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber * you may not use this file except in compliance with the License.
6bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber * You may obtain a copy of the License at
7bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber *
8bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber *      http://www.apache.org/licenses/LICENSE-2.0
9bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber *
10bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber * Unless required by applicable law or agreed to in writing, software
11bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber * distributed under the License is distributed on an "AS IS" BASIS,
12bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber * See the License for the specific language governing permissions and
14bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber * limitations under the License.
15bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber */
16bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
17bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber//#define LOG_NDEBUG 0
18bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber#define LOG_TAG "SoftMPEG4"
19bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber#include <utils/Log.h>
20bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
21bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber#include "SoftMPEG4.h"
22bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
23bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber#include <media/stagefright/foundation/ADebug.h>
24bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber#include <media/stagefright/MediaDefs.h>
25bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber#include <media/stagefright/MediaErrors.h>
26457116d3a01618acf9a875020ca5860551ba03a6James Dong#include <media/IOMX.h>
27bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
28bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber#include "mp4dec_api.h"
29bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
30bbba88cb1bdc34705d1477208990a06904c022e7Andreas Hubernamespace android {
31bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
32457116d3a01618acf9a875020ca5860551ba03a6James Dongstatic const CodecProfileLevel kM4VProfileLevels[] = {
33457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_MPEG4ProfileSimple, OMX_VIDEO_MPEG4Level0 },
34457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_MPEG4ProfileSimple, OMX_VIDEO_MPEG4Level0b },
35457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_MPEG4ProfileSimple, OMX_VIDEO_MPEG4Level1 },
36457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_MPEG4ProfileSimple, OMX_VIDEO_MPEG4Level2 },
37457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_MPEG4ProfileSimple, OMX_VIDEO_MPEG4Level3 },
38457116d3a01618acf9a875020ca5860551ba03a6James Dong};
39457116d3a01618acf9a875020ca5860551ba03a6James Dong
40457116d3a01618acf9a875020ca5860551ba03a6James Dongstatic const CodecProfileLevel kH263ProfileLevels[] = {
41457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_H263ProfileBaseline, OMX_VIDEO_H263Level10 },
42457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_H263ProfileBaseline, OMX_VIDEO_H263Level20 },
43457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_H263ProfileBaseline, OMX_VIDEO_H263Level30 },
44457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_H263ProfileBaseline, OMX_VIDEO_H263Level45 },
45457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_H263ProfileISWV2,    OMX_VIDEO_H263Level10 },
46457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_H263ProfileISWV2,    OMX_VIDEO_H263Level20 },
47457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_H263ProfileISWV2,    OMX_VIDEO_H263Level30 },
48457116d3a01618acf9a875020ca5860551ba03a6James Dong    { OMX_VIDEO_H263ProfileISWV2,    OMX_VIDEO_H263Level45 },
49457116d3a01618acf9a875020ca5860551ba03a6James Dong};
50457116d3a01618acf9a875020ca5860551ba03a6James Dong
51bbba88cb1bdc34705d1477208990a06904c022e7Andreas HuberSoftMPEG4::SoftMPEG4(
52bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        const char *name,
537f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar        const char *componentRole,
547f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar        OMX_VIDEO_CODINGTYPE codingType,
557f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar        const CodecProfileLevel *profileLevels,
567f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar        size_t numProfileLevels,
57bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        const OMX_CALLBACKTYPE *callbacks,
58bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        OMX_PTR appData,
59bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        OMX_COMPONENTTYPE **component)
607f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    : SoftVideoDecoderOMXComponent(
617f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar            name, componentRole, codingType, profileLevels, numProfileLevels,
627f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar            352 /* width */, 288 /* height */, callbacks, appData, component),
637f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar      mMode(codingType == OMX_VIDEO_CodingH263 ? MODE_H263 : MODE_MPEG4),
64bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber      mHandle(new tagvideoDecControls),
65bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber      mInputBufferCount(0),
66bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber      mSignalledError(false),
67bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber      mInitialized(false),
68bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber      mFramesConfigured(false),
69bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber      mNumSamplesOutput(0),
707f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar      mPvTime(0) {
717f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    initPorts(
727f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar            kNumInputBuffers,
737f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar            8192 /* inputBufferSize */,
747f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar            kNumOutputBuffers,
757f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar            (mMode == MODE_MPEG4)
767f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar            ? MEDIA_MIMETYPE_VIDEO_MPEG4 : MEDIA_MIMETYPE_VIDEO_H263);
77bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    CHECK_EQ(initDecoder(), (status_t)OK);
78bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
79bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
80bbba88cb1bdc34705d1477208990a06904c022e7Andreas HuberSoftMPEG4::~SoftMPEG4() {
81bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    if (mInitialized) {
82bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        PVCleanUpVideoDecoder(mHandle);
83bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
84bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
85bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    delete mHandle;
86bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    mHandle = NULL;
87bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
88bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
89bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huberstatus_t SoftMPEG4::initDecoder() {
90bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    memset(mHandle, 0, sizeof(tagvideoDecControls));
91bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    return OK;
92bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
93bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
9484333e0475bc911adc16417f4ca327c975cf6c36Andreas Hubervoid SoftMPEG4::onQueueFilled(OMX_U32 /* portIndex */) {
95bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    if (mSignalledError || mOutputPortSettingsChange != NONE) {
96bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        return;
97bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
98bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
99bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    List<BufferInfo *> &inQueue = getPortQueue(0);
100bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    List<BufferInfo *> &outQueue = getPortQueue(1);
101bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
102bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    while (!inQueue.empty() && outQueue.size() == kNumOutputBuffers) {
103bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        BufferInfo *inInfo = *inQueue.begin();
104bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        OMX_BUFFERHEADERTYPE *inHeader = inInfo->mHeader;
105bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
106bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        PortInfo *port = editPortInfo(1);
107bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
108bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        OMX_BUFFERHEADERTYPE *outHeader =
109bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            port->mBuffers.editItemAt(mNumSamplesOutput & 1).mHeader;
110bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
1114c44e9fed87ff6363393f2559b150291242da247Marco Nelissen        if ((inHeader->nFlags & OMX_BUFFERFLAG_EOS) && inHeader->nFilledLen == 0) {
112bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            inQueue.erase(inQueue.begin());
113bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            inInfo->mOwnedByUs = false;
114bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            notifyEmptyBufferDone(inHeader);
115bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
116bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            ++mInputBufferCount;
117bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
118bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            outHeader->nFilledLen = 0;
119bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            outHeader->nFlags = OMX_BUFFERFLAG_EOS;
120bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
121bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            List<BufferInfo *>::iterator it = outQueue.begin();
122bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            while ((*it)->mHeader != outHeader) {
123bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                ++it;
124bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            }
125bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
126bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            BufferInfo *outInfo = *it;
127bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            outInfo->mOwnedByUs = false;
128bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            outQueue.erase(it);
129bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            outInfo = NULL;
130bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
131bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            notifyFillBufferDone(outHeader);
132bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            outHeader = NULL;
133bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            return;
134bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
135bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
136bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        uint8_t *bitstream = inHeader->pBuffer + inHeader->nOffset;
137a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu        uint32_t *start_code = (uint32_t *)bitstream;
138a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu        bool volHeader = *start_code == 0xB0010000;
139a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu        if (volHeader) {
140a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu            PVCleanUpVideoDecoder(mHandle);
141a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu            mInitialized = false;
142a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu        }
143bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
144bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        if (!mInitialized) {
145bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            uint8_t *vol_data[1];
146bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            int32_t vol_size = 0;
147bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
148bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            vol_data[0] = NULL;
149bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
150a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu            if ((inHeader->nFlags & OMX_BUFFERFLAG_CODECCONFIG) || volHeader) {
151bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                vol_data[0] = bitstream;
152bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                vol_size = inHeader->nFilledLen;
153bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            }
154bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
155bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            MP4DecodingMode mode =
156bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                (mMode == MODE_MPEG4) ? MPEG4_MODE : H263_MODE;
157bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
158bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            Bool success = PVInitVideoDecoder(
1590f694a12f92a01f95807242320bd65e88c699708Ronghua Wu                    mHandle, vol_data, &vol_size, 1,
1600f694a12f92a01f95807242320bd65e88c699708Ronghua Wu                    outputBufferWidth(), outputBufferHeight(), mode);
161bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
162bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            if (!success) {
1635ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block                ALOGW("PVInitVideoDecoder failed. Unsupported content?");
164bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
165bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                notify(OMX_EventError, OMX_ErrorUndefined, 0, NULL);
166bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                mSignalledError = true;
167bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                return;
168bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            }
169bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
170bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            MP4DecodingMode actualMode = PVGetDecBitstreamMode(mHandle);
171bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            if (mode != actualMode) {
172bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                notify(OMX_EventError, OMX_ErrorUndefined, 0, NULL);
173bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                mSignalledError = true;
174bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                return;
175bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            }
176bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
177bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            PVSetPostProcType((VideoDecControls *) mHandle, 0);
178bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
179a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu            bool hasFrameData = false;
180bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            if (inHeader->nFlags & OMX_BUFFERFLAG_CODECCONFIG) {
181bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                inInfo->mOwnedByUs = false;
182bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                inQueue.erase(inQueue.begin());
183bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                inInfo = NULL;
184bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                notifyEmptyBufferDone(inHeader);
185bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                inHeader = NULL;
186a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu            } else if (volHeader) {
187a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu                hasFrameData = true;
188bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            }
189bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
190bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            mInitialized = true;
191bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
192a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu            if (mode == MPEG4_MODE && handlePortSettingsChange()) {
193bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                return;
194bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            }
195bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
196a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu            if (!hasFrameData) {
197a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu                continue;
198a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu            }
199bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
200bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
201bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        if (!mFramesConfigured) {
202bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            PortInfo *port = editPortInfo(1);
203bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            OMX_BUFFERHEADERTYPE *outHeader = port->mBuffers.editItemAt(1).mHeader;
204bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
205bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            PVSetReferenceYUV(mHandle, outHeader->pBuffer);
206bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
207bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            mFramesConfigured = true;
208bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
209bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
210f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber        uint32_t useExtTimestamp = (inHeader->nOffset == 0);
211f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber
212269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        // decoder deals in ms (int32_t), OMX in us (int64_t)
213269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        // so use fake timestamp instead
214269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        uint32_t timestamp = 0xFFFFFFFF;
215269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        if (useExtTimestamp) {
216269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar            mPvToOmxTimeMap.add(mPvTime, inHeader->nTimeStamp);
217269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar            timestamp = mPvTime;
218269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar            mPvTime++;
219269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        }
220f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber
221bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        int32_t bufferSize = inHeader->nFilledLen;
22202accddf8d69da7b2b5e05631ad222cd842ff547Andreas Huber        int32_t tmp = bufferSize;
223bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
2243a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim        OMX_U32 frameSize = (mWidth * mHeight * 3) / 2;
2253a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim        if (outHeader->nAllocLen < frameSize) {
2263a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim            android_errorWriteLog(0x534e4554, "27833616");
2273a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim            ALOGE("Insufficient output buffer size");
2283a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim            notify(OMX_EventError, OMX_ErrorUndefined, 0, NULL);
2293a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim            mSignalledError = true;
2303a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim            return;
2313a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim        }
2320029faf8f4efbca3844c4174b75d46c432f8f272Andreas Huber        // The PV decoder is lying to us, sometimes it'll claim to only have
2330029faf8f4efbca3844c4174b75d46c432f8f272Andreas Huber        // consumed a subset of the buffer when it clearly consumed all of it.
2340029faf8f4efbca3844c4174b75d46c432f8f272Andreas Huber        // ignore whatever it says...
235bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        if (PVDecodeVideoFrame(
2360029faf8f4efbca3844c4174b75d46c432f8f272Andreas Huber                    mHandle, &bitstream, &timestamp, &tmp,
237bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                    &useExtTimestamp,
238bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                    outHeader->pBuffer) != PV_TRUE) {
23929357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("failed to decode video frame.");
240bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
241bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            notify(OMX_EventError, OMX_ErrorUndefined, 0, NULL);
242bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            mSignalledError = true;
243bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            return;
244bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
245bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
246a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu        // H263 doesn't have VOL header, the frame size information is in short header, i.e. the
247a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu        // decoder may detect size change after PVDecodeVideoFrame.
248a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu        if (handlePortSettingsChange()) {
249bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            return;
250bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
251bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
252f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber        // decoder deals in ms, OMX in us.
253269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        outHeader->nTimeStamp = mPvToOmxTimeMap.valueFor(timestamp);
254269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        mPvToOmxTimeMap.removeItem(timestamp);
255bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
25602accddf8d69da7b2b5e05631ad222cd842ff547Andreas Huber        inHeader->nOffset += bufferSize;
25702accddf8d69da7b2b5e05631ad222cd842ff547Andreas Huber        inHeader->nFilledLen = 0;
2584c44e9fed87ff6363393f2559b150291242da247Marco Nelissen        if (inHeader->nFlags & OMX_BUFFERFLAG_EOS) {
2594c44e9fed87ff6363393f2559b150291242da247Marco Nelissen            outHeader->nFlags = OMX_BUFFERFLAG_EOS;
2604c44e9fed87ff6363393f2559b150291242da247Marco Nelissen        } else {
2614c44e9fed87ff6363393f2559b150291242da247Marco Nelissen            outHeader->nFlags = 0;
2624c44e9fed87ff6363393f2559b150291242da247Marco Nelissen        }
263f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber
264f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber        if (inHeader->nFilledLen == 0) {
265f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber            inInfo->mOwnedByUs = false;
266f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber            inQueue.erase(inQueue.begin());
267f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber            inInfo = NULL;
268f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber            notifyEmptyBufferDone(inHeader);
269f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber            inHeader = NULL;
270f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber        }
271bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
272bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        ++mInputBufferCount;
273bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
274bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        outHeader->nOffset = 0;
2753a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim        outHeader->nFilledLen = frameSize;
276bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
277bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        List<BufferInfo *>::iterator it = outQueue.begin();
278bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        while ((*it)->mHeader != outHeader) {
279bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            ++it;
280bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
281bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
282bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        BufferInfo *outInfo = *it;
283bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        outInfo->mOwnedByUs = false;
284bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        outQueue.erase(it);
285bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        outInfo = NULL;
286bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
287bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        notifyFillBufferDone(outHeader);
288bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        outHeader = NULL;
289bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
290bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        ++mNumSamplesOutput;
291bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
292bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
293bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
294a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wubool SoftMPEG4::handlePortSettingsChange() {
2957f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    uint32_t disp_width, disp_height;
2967f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    PVGetVideoDimensions(mHandle, (int32 *)&disp_width, (int32 *)&disp_height);
297bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
2987f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    uint32_t buf_width, buf_height;
2997f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    PVGetBufferDimensions(mHandle, (int32 *)&buf_width, (int32 *)&buf_height);
300bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
301bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    CHECK_LE(disp_width, buf_width);
302bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    CHECK_LE(disp_height, buf_height);
303bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
3043856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("disp_width = %d, disp_height = %d, buf_width = %d, buf_height = %d",
305bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            disp_width, disp_height, buf_width, buf_height);
306bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
307d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu    CropSettingsMode cropSettingsMode = kCropUnSet;
308d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu    if (disp_width != buf_width || disp_height != buf_height) {
309d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu        cropSettingsMode = kCropSet;
310d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu
311d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu        if (mCropWidth != disp_width || mCropHeight != disp_height) {
312d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            mCropLeft = 0;
313d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            mCropTop = 0;
314d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            mCropWidth = disp_width;
315d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            mCropHeight = disp_height;
316d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            cropSettingsMode = kCropChanged;
317d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu        }
318bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
319bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
320a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu    bool portWillReset = false;
321a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu    const bool fakeStride = true;
322a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu    SoftVideoDecoderOMXComponent::handlePortSettingsChange(
323d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            &portWillReset, buf_width, buf_height, cropSettingsMode, fakeStride);
324a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu    if (portWillReset) {
325bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        if (mMode == MODE_H263) {
326bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            PVCleanUpVideoDecoder(mHandle);
327bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
328bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            uint8_t *vol_data[1];
329bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            int32_t vol_size = 0;
330bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
331bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            vol_data[0] = NULL;
332bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            if (!PVInitVideoDecoder(
3330f694a12f92a01f95807242320bd65e88c699708Ronghua Wu                    mHandle, vol_data, &vol_size, 1, outputBufferWidth(), outputBufferHeight(),
334bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                    H263_MODE)) {
335bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                notify(OMX_EventError, OMX_ErrorUndefined, 0, NULL);
336bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                mSignalledError = true;
337bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                return true;
338bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            }
339bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
340bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
341bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        mFramesConfigured = false;
342bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
343bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
344a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu    return portWillReset;
345bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
346bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
347bbba88cb1bdc34705d1477208990a06904c022e7Andreas Hubervoid SoftMPEG4::onPortFlushCompleted(OMX_U32 portIndex) {
348bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    if (portIndex == 0 && mInitialized) {
349bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        CHECK_EQ((int)PVResetVideoDecoder(mHandle), (int)PV_TRUE);
350bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
351bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
352bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
353d94e716af0e49d775f0c0c4f36dd2c136ba5f2b2Andreas Hubervoid SoftMPEG4::onReset() {
3547f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    SoftVideoDecoderOMXComponent::onReset();
355269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar    mPvToOmxTimeMap.clear();
356d94e716af0e49d775f0c0c4f36dd2c136ba5f2b2Andreas Huber    mSignalledError = false;
35753b0a2b1f9cb6b99b3f0d1a639921d1b24bc30b7Lajos Molnar    mFramesConfigured = false;
35853b0a2b1f9cb6b99b3f0d1a639921d1b24bc30b7Lajos Molnar    if (mInitialized) {
35953b0a2b1f9cb6b99b3f0d1a639921d1b24bc30b7Lajos Molnar        PVCleanUpVideoDecoder(mHandle);
36053b0a2b1f9cb6b99b3f0d1a639921d1b24bc30b7Lajos Molnar        mInitialized = false;
36153b0a2b1f9cb6b99b3f0d1a639921d1b24bc30b7Lajos Molnar    }
362d94e716af0e49d775f0c0c4f36dd2c136ba5f2b2Andreas Huber}
363d94e716af0e49d775f0c0c4f36dd2c136ba5f2b2Andreas Huber
364bbba88cb1bdc34705d1477208990a06904c022e7Andreas Hubervoid SoftMPEG4::updatePortDefinitions() {
3657f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    SoftVideoDecoderOMXComponent::updatePortDefinitions();
366bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
3677f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    /* We have to align our width and height - this should affect stride! */
3687f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    OMX_PARAM_PORTDEFINITIONTYPE *def = &editPortInfo(kOutputPortIndex)->mDef;
369bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    def->nBufferSize =
370bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        (((def->format.video.nFrameWidth + 15) & -16)
371bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            * ((def->format.video.nFrameHeight + 15) & -16) * 3) / 2;
372bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
373bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
374bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}  // namespace android
375bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
376bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huberandroid::SoftOMXComponent *createSoftOMXComponent(
377bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        const char *name, const OMX_CALLBACKTYPE *callbacks,
378bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        OMX_PTR appData, OMX_COMPONENTTYPE **component) {
3797f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    using namespace android;
3807f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    if (!strcmp(name, "OMX.google.h263.decoder")) {
3817f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar        return new android::SoftMPEG4(
3827f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                name, "video_decoder.h263", OMX_VIDEO_CodingH263,
3837f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                kH263ProfileLevels, ARRAY_SIZE(kH263ProfileLevels),
3847f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                callbacks, appData, component);
3857f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    } else if (!strcmp(name, "OMX.google.mpeg4.decoder")) {
3867f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar        return new android::SoftMPEG4(
3877f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                name, "video_decoder.mpeg4", OMX_VIDEO_CodingMPEG4,
3887f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                kM4VProfileLevels, ARRAY_SIZE(kM4VProfileLevels),
3897f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                callbacks, appData, component);
3907f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    } else {
3917f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar        CHECK(!"Unknown component");
3927f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    }
393bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
394bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
395