SoftMPEG4.cpp revision baa9146401e28c5acf54dea21ddd197f0d3a8fcd
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
205180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa            OMX_U32 yFrameSize = sizeof(uint8) * mHandle->size;
206180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa            if ((outHeader->nAllocLen < yFrameSize) ||
207180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa                    (outHeader->nAllocLen - yFrameSize < yFrameSize / 2)) {
208baa9146401e28c5acf54dea21ddd197f0d3a8fcdPawin Vongmasa                ALOGE("Too small output buffer for reference frame: %lu bytes",
209baa9146401e28c5acf54dea21ddd197f0d3a8fcdPawin Vongmasa                        (unsigned long)outHeader->nAllocLen);
210180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa                android_errorWriteLog(0x534e4554, "30033990");
211180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa                notify(OMX_EventError, OMX_ErrorUndefined, 0, NULL);
212180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa                mSignalledError = true;
213180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa                return;
214180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa            }
215bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            PVSetReferenceYUV(mHandle, outHeader->pBuffer);
216bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            mFramesConfigured = true;
217bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
218bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
219f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber        uint32_t useExtTimestamp = (inHeader->nOffset == 0);
220f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber
221269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        // decoder deals in ms (int32_t), OMX in us (int64_t)
222269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        // so use fake timestamp instead
223269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        uint32_t timestamp = 0xFFFFFFFF;
224269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        if (useExtTimestamp) {
225269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar            mPvToOmxTimeMap.add(mPvTime, inHeader->nTimeStamp);
226269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar            timestamp = mPvTime;
227269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar            mPvTime++;
228269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        }
229f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber
230bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        int32_t bufferSize = inHeader->nFilledLen;
23102accddf8d69da7b2b5e05631ad222cd842ff547Andreas Huber        int32_t tmp = bufferSize;
232bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
233180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa        OMX_U32 frameSize;
234180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa        OMX_U64 yFrameSize = (OMX_U64)mWidth * (OMX_U64)mHeight;
235180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa        if (yFrameSize > ((OMX_U64)UINT32_MAX / 3) * 2) {
236180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa            ALOGE("Frame size too large");
237180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa            notify(OMX_EventError, OMX_ErrorUndefined, 0, NULL);
238180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa            mSignalledError = true;
239180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa            return;
240180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa        }
241180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa        frameSize = (OMX_U32)(yFrameSize + (yFrameSize / 2));
242180185da195c42d366a16d9c087413dd8c07d58bPawin Vongmasa
2433a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim        if (outHeader->nAllocLen < frameSize) {
2443a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim            android_errorWriteLog(0x534e4554, "27833616");
2453a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim            ALOGE("Insufficient output buffer size");
2463a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim            notify(OMX_EventError, OMX_ErrorUndefined, 0, NULL);
2473a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim            mSignalledError = true;
2483a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim            return;
2493a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim        }
2500029faf8f4efbca3844c4174b75d46c432f8f272Andreas Huber        // The PV decoder is lying to us, sometimes it'll claim to only have
2510029faf8f4efbca3844c4174b75d46c432f8f272Andreas Huber        // consumed a subset of the buffer when it clearly consumed all of it.
2520029faf8f4efbca3844c4174b75d46c432f8f272Andreas Huber        // ignore whatever it says...
253bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        if (PVDecodeVideoFrame(
2540029faf8f4efbca3844c4174b75d46c432f8f272Andreas Huber                    mHandle, &bitstream, &timestamp, &tmp,
255bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                    &useExtTimestamp,
256bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                    outHeader->pBuffer) != PV_TRUE) {
25729357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("failed to decode video frame.");
258bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
259bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            notify(OMX_EventError, OMX_ErrorUndefined, 0, NULL);
260bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            mSignalledError = true;
261bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            return;
262bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
263bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
264a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu        // H263 doesn't have VOL header, the frame size information is in short header, i.e. the
265a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu        // decoder may detect size change after PVDecodeVideoFrame.
266a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu        if (handlePortSettingsChange()) {
267bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            return;
268bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
269bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
270f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber        // decoder deals in ms, OMX in us.
271269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        outHeader->nTimeStamp = mPvToOmxTimeMap.valueFor(timestamp);
272269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar        mPvToOmxTimeMap.removeItem(timestamp);
273bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
27402accddf8d69da7b2b5e05631ad222cd842ff547Andreas Huber        inHeader->nOffset += bufferSize;
27502accddf8d69da7b2b5e05631ad222cd842ff547Andreas Huber        inHeader->nFilledLen = 0;
2764c44e9fed87ff6363393f2559b150291242da247Marco Nelissen        if (inHeader->nFlags & OMX_BUFFERFLAG_EOS) {
2774c44e9fed87ff6363393f2559b150291242da247Marco Nelissen            outHeader->nFlags = OMX_BUFFERFLAG_EOS;
2784c44e9fed87ff6363393f2559b150291242da247Marco Nelissen        } else {
2794c44e9fed87ff6363393f2559b150291242da247Marco Nelissen            outHeader->nFlags = 0;
2804c44e9fed87ff6363393f2559b150291242da247Marco Nelissen        }
281f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber
282f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber        if (inHeader->nFilledLen == 0) {
283f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber            inInfo->mOwnedByUs = false;
284f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber            inQueue.erase(inQueue.begin());
285f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber            inInfo = NULL;
286f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber            notifyEmptyBufferDone(inHeader);
287f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber            inHeader = NULL;
288f2af5a2c607e71ff4cd39da28b077c0a68b206feAndreas Huber        }
289bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
290bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        ++mInputBufferCount;
291bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
292bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        outHeader->nOffset = 0;
2933a3c3f7fc658ef874f82e46857ad9df3616aac95Wonsik Kim        outHeader->nFilledLen = frameSize;
294bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
295bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        List<BufferInfo *>::iterator it = outQueue.begin();
296bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        while ((*it)->mHeader != outHeader) {
297bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            ++it;
298bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
299bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
300bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        BufferInfo *outInfo = *it;
301bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        outInfo->mOwnedByUs = false;
302bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        outQueue.erase(it);
303bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        outInfo = NULL;
304bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
305bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        notifyFillBufferDone(outHeader);
306bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        outHeader = NULL;
307bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
308bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        ++mNumSamplesOutput;
309bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
310bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
311bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
312a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wubool SoftMPEG4::handlePortSettingsChange() {
3137f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    uint32_t disp_width, disp_height;
3147f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    PVGetVideoDimensions(mHandle, (int32 *)&disp_width, (int32 *)&disp_height);
315bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
3167f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    uint32_t buf_width, buf_height;
3177f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    PVGetBufferDimensions(mHandle, (int32 *)&buf_width, (int32 *)&buf_height);
318bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
319bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    CHECK_LE(disp_width, buf_width);
320bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    CHECK_LE(disp_height, buf_height);
321bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
3223856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("disp_width = %d, disp_height = %d, buf_width = %d, buf_height = %d",
323bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            disp_width, disp_height, buf_width, buf_height);
324bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
325d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu    CropSettingsMode cropSettingsMode = kCropUnSet;
326d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu    if (disp_width != buf_width || disp_height != buf_height) {
327d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu        cropSettingsMode = kCropSet;
328d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu
329d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu        if (mCropWidth != disp_width || mCropHeight != disp_height) {
330d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            mCropLeft = 0;
331d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            mCropTop = 0;
332d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            mCropWidth = disp_width;
333d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            mCropHeight = disp_height;
334d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            cropSettingsMode = kCropChanged;
335d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu        }
336bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
337bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
338a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu    bool portWillReset = false;
339a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu    const bool fakeStride = true;
340a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu    SoftVideoDecoderOMXComponent::handlePortSettingsChange(
341d5a2f55034022f2d0425fa0701894d0c4787b726Ronghua Wu            &portWillReset, buf_width, buf_height, cropSettingsMode, fakeStride);
342a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu    if (portWillReset) {
343bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        if (mMode == MODE_H263) {
344bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            PVCleanUpVideoDecoder(mHandle);
345bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
346bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            uint8_t *vol_data[1];
347bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            int32_t vol_size = 0;
348bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
349bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            vol_data[0] = NULL;
350bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            if (!PVInitVideoDecoder(
3510f694a12f92a01f95807242320bd65e88c699708Ronghua Wu                    mHandle, vol_data, &vol_size, 1, outputBufferWidth(), outputBufferHeight(),
352bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                    H263_MODE)) {
353bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                notify(OMX_EventError, OMX_ErrorUndefined, 0, NULL);
354bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                mSignalledError = true;
355bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber                return true;
356bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            }
357bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        }
358bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
359bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        mFramesConfigured = false;
360bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
361bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
362a694dd0ce2caaf921f7bc894df87a5d52594b4ebRonghua Wu    return portWillReset;
363bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
364bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
365bbba88cb1bdc34705d1477208990a06904c022e7Andreas Hubervoid SoftMPEG4::onPortFlushCompleted(OMX_U32 portIndex) {
366bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    if (portIndex == 0 && mInitialized) {
367bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        CHECK_EQ((int)PVResetVideoDecoder(mHandle), (int)PV_TRUE);
368bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    }
369bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
370bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
371d94e716af0e49d775f0c0c4f36dd2c136ba5f2b2Andreas Hubervoid SoftMPEG4::onReset() {
3727f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    SoftVideoDecoderOMXComponent::onReset();
373269a355679fce6a71523faeefc2ff575abbd1a8eLajos Molnar    mPvToOmxTimeMap.clear();
374d94e716af0e49d775f0c0c4f36dd2c136ba5f2b2Andreas Huber    mSignalledError = false;
37553b0a2b1f9cb6b99b3f0d1a639921d1b24bc30b7Lajos Molnar    mFramesConfigured = false;
37653b0a2b1f9cb6b99b3f0d1a639921d1b24bc30b7Lajos Molnar    if (mInitialized) {
37753b0a2b1f9cb6b99b3f0d1a639921d1b24bc30b7Lajos Molnar        PVCleanUpVideoDecoder(mHandle);
37853b0a2b1f9cb6b99b3f0d1a639921d1b24bc30b7Lajos Molnar        mInitialized = false;
37953b0a2b1f9cb6b99b3f0d1a639921d1b24bc30b7Lajos Molnar    }
380d94e716af0e49d775f0c0c4f36dd2c136ba5f2b2Andreas Huber}
381d94e716af0e49d775f0c0c4f36dd2c136ba5f2b2Andreas Huber
382bbba88cb1bdc34705d1477208990a06904c022e7Andreas Hubervoid SoftMPEG4::updatePortDefinitions() {
3837f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    SoftVideoDecoderOMXComponent::updatePortDefinitions();
384bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
3857f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    /* We have to align our width and height - this should affect stride! */
3867f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    OMX_PARAM_PORTDEFINITIONTYPE *def = &editPortInfo(kOutputPortIndex)->mDef;
387bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber    def->nBufferSize =
388bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        (((def->format.video.nFrameWidth + 15) & -16)
389bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber            * ((def->format.video.nFrameHeight + 15) & -16) * 3) / 2;
390bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
391bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
392bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}  // namespace android
393bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
394bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huberandroid::SoftOMXComponent *createSoftOMXComponent(
395bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        const char *name, const OMX_CALLBACKTYPE *callbacks,
396bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber        OMX_PTR appData, OMX_COMPONENTTYPE **component) {
3977f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    using namespace android;
3987f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    if (!strcmp(name, "OMX.google.h263.decoder")) {
3997f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar        return new android::SoftMPEG4(
4007f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                name, "video_decoder.h263", OMX_VIDEO_CodingH263,
4017f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                kH263ProfileLevels, ARRAY_SIZE(kH263ProfileLevels),
4027f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                callbacks, appData, component);
4037f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    } else if (!strcmp(name, "OMX.google.mpeg4.decoder")) {
4047f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar        return new android::SoftMPEG4(
4057f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                name, "video_decoder.mpeg4", OMX_VIDEO_CodingMPEG4,
4067f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                kM4VProfileLevels, ARRAY_SIZE(kM4VProfileLevels),
4077f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar                callbacks, appData, component);
4087f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    } else {
4097f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar        CHECK(!"Unknown component");
4107f616d3cc5366a4b8af20d3d0c768e3de1df0666Lajos Molnar    }
411bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber}
412bbba88cb1bdc34705d1477208990a06904c022e7Andreas Huber
413