OMXVideoEncoderH263.cpp revision dbb4b89361b47ddf1988c654c5cc8e07818dd5ec
1e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu/*
2e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu* Copyright (c) 2009-2011 Intel Corporation.  All rights reserved.
3e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu*
4e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu* Licensed under the Apache License, Version 2.0 (the "License");
5e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu* you may not use this file except in compliance with the License.
6e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu* You may obtain a copy of the License at
7e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu*
8e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu* http://www.apache.org/licenses/LICENSE-2.0
9e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu*
10e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu* Unless required by applicable law or agreed to in writing, software
11e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu* distributed under the License is distributed on an "AS IS" BASIS,
12e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu* See the License for the specific language governing permissions and
14e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu* limitations under the License.
15e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu*/
16e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
17e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
1819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang//#define LOG_NDEBUG 0
19e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu#define LOG_TAG "OMXVideoEncoderH263"
20e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu#include <utils/Log.h>
21e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu#include "OMXVideoEncoderH263.h"
22e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
2319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sangstatic const char *H263_MIME_TYPE = "video/h263";
24e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
25e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMXVideoEncoderH263::OMXVideoEncoderH263() {
2619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV("Constructer for OMXVideoEncoderH263.");
27e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    BuildHandlerList();
2819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    mVideoEncoder = createVideoEncoder(H263_MIME_TYPE);
2919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (!mVideoEncoder) LOGE("OMX_ErrorInsufficientResources");
30e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
31e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
32e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMXVideoEncoderH263::~OMXVideoEncoderH263() {
3319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV("Destructer for OMXVideoEncoderH263.");
34e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
35e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
36e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMX_ERRORTYPE OMXVideoEncoderH263::InitOutputPortFormatSpecific(OMX_PARAM_PORTDEFINITIONTYPE *paramPortDefinitionOutput) {
37e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    // OMX_VIDEO_PARAM_H263TYPE
38e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    memset(&mParamH263, 0, sizeof(mParamH263));
39e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    SetTypeHeader(&mParamH263, sizeof(mParamH263));
40e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    mParamH263.nPortIndex = OUTPORT_INDEX;
41e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    mParamH263.eProfile = OMX_VIDEO_H263ProfileBaseline;
4219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    // TODO: check eLevel, 10
43cae415b84125ffc14bfdb121394a60b74d506f48Yanli    mParamH263.eLevel = OMX_VIDEO_H263Level45; //OMX_VIDEO_H263Level10;
44e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
45e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    // override OMX_PARAM_PORTDEFINITIONTYPE
46e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    paramPortDefinitionOutput->nBufferCountActual = OUTPORT_ACTUAL_BUFFER_COUNT;
47e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    paramPortDefinitionOutput->nBufferCountMin = OUTPORT_MIN_BUFFER_COUNT;
48e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    paramPortDefinitionOutput->nBufferSize = OUTPORT_BUFFER_SIZE;
49e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    paramPortDefinitionOutput->format.video.cMIMEType = (OMX_STRING)H263_MIME_TYPE;
50e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    paramPortDefinitionOutput->format.video.eCompressionFormat = OMX_VIDEO_CodingH263;
51e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
52e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    // override OMX_VIDEO_PARAM_PROFILELEVELTYPE
53e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    // TODO: check if profile/level supported is correct
54e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    mParamProfileLevel.eProfile = mParamH263.eProfile;
55e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    mParamProfileLevel.eLevel = mParamH263.eLevel; //OMX_VIDEO_H263Level70
56e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
57e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    // override OMX_VIDEO_PARAM_BITRATETYPE
58e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    mParamBitrate.nTargetBitrate = 64000;
59e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
6019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    // override OMX_VIDEO_CONFIG_INTEL_BITRATETYPE
6119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    mConfigIntelBitrate.nInitialQP = 15;  // Initial QP for I frames
62e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    return OMX_ErrorNone;
63e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
64e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
6519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo SangOMX_ERRORTYPE OMXVideoEncoderH263::SetVideoEncoderParam(void) {
6619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
6719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (!mEncoderParams) {
6819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        LOGE("NULL pointer: mEncoderParams");
6919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        return OMX_ErrorBadParameter;
7019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
7119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
7219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    mVideoEncoder->getParameters(mEncoderParams);
7319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    mEncoderParams->profile = (VAProfile)PROFILE_H263BASELINE;
7419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    return OMXVideoEncoderBase::SetVideoEncoderParam();
7519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang}
7619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
77e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMX_ERRORTYPE OMXVideoEncoderH263::ProcessorInit(void) {
7819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV("OMXVideoEncoderH263::ProcessorInit\n");
79e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    return OMXVideoEncoderBase::ProcessorInit();
80e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
81e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
82e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMX_ERRORTYPE OMXVideoEncoderH263::ProcessorDeinit(void) {
83e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    return OMXVideoEncoderBase::ProcessorDeinit();
84e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
85e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
86e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMX_ERRORTYPE OMXVideoEncoderH263::ProcessorProcess(
8719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    OMX_BUFFERHEADERTYPE **buffers,
8819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    buffer_retain_t *retains,
8919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    OMX_U32 numberBuffers) {
9019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV("OMXVideoEncoderH263::ProcessorProcess \n");
9119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
9219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    VideoEncOutputBuffer outBuf;
9319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    VideoEncRawBuffer inBuf;
9419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    OMX_U32 outfilledlen = 0;
9519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    OMX_S64 outtimestamp = 0;
9619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    OMX_U32 outflags = 0;
9719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
9819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    OMX_ERRORTYPE oret = OMX_ErrorNone;
9919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    Encode_Status ret = ENCODE_SUCCESS;
10019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
10119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV("%s(): enter encode\n", __func__);
10219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
10319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV_IF(buffers[INPORT_INDEX]->nFlags & OMX_BUFFERFLAG_EOS,
10419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang            "%s(),%d: got OMX_BUFFERFLAG_EOS\n", __func__, __LINE__);
10519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
10619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (!buffers[INPORT_INDEX]->nFilledLen) {
10719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        LOGE("%s(),%d: input buffer's nFilledLen is zero\n", __func__, __LINE__);
10819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        goto out;
10919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
11019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
11128516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying    if (bAndroidOpaqueFormat) {
11228516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying        mCurHandle = rgba2nv12conversion(buffers[INPORT_INDEX]);
113dbb4b89361b47ddf1988c654c5cc8e07818dd5ecZhao Liang        if (mCurHandle < 0)
114dbb4b89361b47ddf1988c654c5cc8e07818dd5ecZhao Liang            return OMX_ErrorUndefined;
11528516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying    }
11628516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying
1174ce57b82fa719f2aa9247c7f4a44853715a29987Guoliang Ji    inBuf.data = buffers[INPORT_INDEX]->pBuffer + buffers[INPORT_INDEX]->nOffset;
1184ce57b82fa719f2aa9247c7f4a44853715a29987Guoliang Ji    inBuf.size = buffers[INPORT_INDEX]->nFilledLen;
119dbb4b89361b47ddf1988c654c5cc8e07818dd5ecZhao Liang    inBuf.type = FTYPE_UNKNOWN;
120dbb4b89361b47ddf1988c654c5cc8e07818dd5ecZhao Liang    inBuf.flag = 0;
121dbb4b89361b47ddf1988c654c5cc8e07818dd5ecZhao Liang    inBuf.timeStamp = buffers[INPORT_INDEX]->nTimeStamp;
12219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
12319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV("buffer_in.data=%x, data_size=%d",
12419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang         (unsigned)inBuf.data, inBuf.size);
12519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
12619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outBuf.data =	buffers[OUTPORT_INDEX]->pBuffer + buffers[OUTPORT_INDEX]->nOffset;
12719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outBuf.bufferSize = buffers[OUTPORT_INDEX]->nAllocLen - buffers[OUTPORT_INDEX]->nOffset;
12819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outBuf.dataSize = 0;
12919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
13019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if(mFrameRetrieved) {
13119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        // encode and setConfig need to be thread safe
13219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        pthread_mutex_unlock(&mSerializationLock);
13319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        ret = mVideoEncoder->encode(&inBuf);
13419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        pthread_mutex_unlock(&mSerializationLock);
13519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
13619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        CHECK_ENCODE_STATUS("encode");
13719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        mFrameRetrieved = OMX_FALSE;
13819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
13919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        // This is for buffer contention, we won't release current buffer
14019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        // but the last input buffer
14119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        ports[INPORT_INDEX]->ReturnAllRetainedBuffers();
14219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
14319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
14419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outBuf.format = OUTPUT_EVERYTHING;
14519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    ret = mVideoEncoder->getOutput(&outBuf);
14619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    // CHECK_ENCODE_STATUS("encode");
14719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if(ret == ENCODE_NO_REQUEST_DATA) {
14819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        mFrameRetrieved = OMX_TRUE;
14919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[OUTPORT_INDEX] = BUFFER_RETAIN_GETAGAIN;
15019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[INPORT_INDEX] = BUFFER_RETAIN_ACCUMULATE;
15119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        goto out;
15219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
15319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
15419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV("output data size = %d", outBuf.dataSize);
15519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outfilledlen = outBuf.dataSize;
156dbb4b89361b47ddf1988c654c5cc8e07818dd5ecZhao Liang    outtimestamp = outBuf.timeStamp;
15719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
15819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
15919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (outBuf.flag & ENCODE_BUFFERFLAG_SYNCFRAME) {
16019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        outflags |= OMX_BUFFERFLAG_SYNCFRAME;
16119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
16219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
16319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if(outBuf.flag & ENCODE_BUFFERFLAG_ENDOFFRAME) {
16419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        outflags |= OMX_BUFFERFLAG_ENDOFFRAME;
16519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        mFrameRetrieved = OMX_TRUE;
16619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[INPORT_INDEX] = BUFFER_RETAIN_ACCUMULATE;
16719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
16819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    } else {
16919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[INPORT_INDEX] = BUFFER_RETAIN_GETAGAIN;  //get again
17019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
17119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
17219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
17319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (outfilledlen > 0) {
17419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[OUTPORT_INDEX] = BUFFER_RETAIN_NOT_RETAIN;
17519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    } else {
17619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[OUTPORT_INDEX] = BUFFER_RETAIN_GETAGAIN;
17719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
17819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
17919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
18019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if(ret == ENCODE_SLICESIZE_OVERFLOW) {
18119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        LOGV("%s(), mix_video_encode returns MIX_RESULT_VIDEO_ENC_SLICESIZE_OVERFLOW"
18219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang             , __func__);
18319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        oret = (OMX_ERRORTYPE)OMX_ErrorIntelExtSliceSizeOverflow;
18419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
18519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang#if SHOW_FPS
18619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    {
18719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        struct timeval t;
18819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        OMX_TICKS current_ts, interval_ts;
18919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        float current_fps, average_fps;
19019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
19119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        t.tv_sec = t.tv_usec = 0;
19219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        gettimeofday(&t, NULL);
19319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
19419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        current_ts =(nsecs_t)t.tv_sec * 1000000000 + (nsecs_t)t.tv_usec * 1000;
19519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        interval_ts = current_ts - lastTs;
19619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        lastTs = current_ts;
19719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
19819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        current_fps = (float)1000000000 / (float)interval_ts;
19919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        average_fps = (current_fps + lastFps) / 2;
20019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        lastFps = current_fps;
20119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
20219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        LOGD("FPS = %2.1f\n", average_fps);
20319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
20419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang#endif
20519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
20619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sangout:
20719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
20819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if(retains[OUTPORT_INDEX] != BUFFER_RETAIN_GETAGAIN) {
20919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        buffers[OUTPORT_INDEX]->nFilledLen = outfilledlen;
21019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        buffers[OUTPORT_INDEX]->nTimeStamp = outtimestamp;
21119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        buffers[OUTPORT_INDEX]->nFlags = outflags;
21219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
21319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        LOGV("********** output buffer: len=%d, ts=%ld, flags=%x",
21419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang             outfilledlen,
21519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang             outtimestamp,
21619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang             outflags);
21719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
21819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
21919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (retains[INPORT_INDEX] == BUFFER_RETAIN_NOT_RETAIN ||
22019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang            retains[INPORT_INDEX] == BUFFER_RETAIN_ACCUMULATE ) {
22119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        mFrameInputCount ++;
22219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
22319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
22419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (retains[OUTPORT_INDEX] == BUFFER_RETAIN_NOT_RETAIN)
22519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        mFrameOutputCount ++;
22619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
22728516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying    if (bAndroidOpaqueFormat && buffers[INPORT_INDEX]->nFilledLen != 0) {
22828516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying        // Restore input buffer's content
22928516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying        buffers[INPORT_INDEX]->nFilledLen = 4 + sizeof(buffer_handle_t);
23028516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying        memcpy(buffers[INPORT_INDEX]->pBuffer, mBufferHandleMaps[mCurHandle].backBuffer,
23128516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying                buffers[INPORT_INDEX]->nFilledLen);
23228516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying
23328516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying    }
23428516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying
23519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV_IF(oret == OMX_ErrorNone, "%s(),%d: exit, encode is done\n", __func__, __LINE__);
23619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
23719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    return oret;
238e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
239e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
240e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
241e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMX_ERRORTYPE OMXVideoEncoderH263::BuildHandlerList(void) {
242e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    OMXVideoEncoderBase::BuildHandlerList();
243e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    AddHandler(OMX_IndexParamVideoH263, GetParamVideoH263, SetParamVideoH263);
244cae415b84125ffc14bfdb121394a60b74d506f48Yanli    AddHandler(OMX_IndexParamVideoProfileLevelQuerySupported, GetParamVideoProfileLevelQuerySupported, SetParamVideoProfileLevelQuerySupported);
245e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    return OMX_ErrorNone;
246e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
247e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
248cae415b84125ffc14bfdb121394a60b74d506f48YanliOMX_ERRORTYPE OMXVideoEncoderH263::GetParamVideoProfileLevelQuerySupported(OMX_PTR pStructure) {
249cae415b84125ffc14bfdb121394a60b74d506f48Yanli    OMX_ERRORTYPE ret;
250cae415b84125ffc14bfdb121394a60b74d506f48Yanli    OMX_VIDEO_PARAM_PROFILELEVELTYPE *p = (OMX_VIDEO_PARAM_PROFILELEVELTYPE *)pStructure;
251cae415b84125ffc14bfdb121394a60b74d506f48Yanli    CHECK_TYPE_HEADER(p);
252cae415b84125ffc14bfdb121394a60b74d506f48Yanli    CHECK_PORT_INDEX(p, OUTPORT_INDEX);
253cae415b84125ffc14bfdb121394a60b74d506f48Yanli
254cae415b84125ffc14bfdb121394a60b74d506f48Yanli    struct ProfileLevelTable {
255cae415b84125ffc14bfdb121394a60b74d506f48Yanli        OMX_U32 profile;
256cae415b84125ffc14bfdb121394a60b74d506f48Yanli        OMX_U32 level;
257cae415b84125ffc14bfdb121394a60b74d506f48Yanli    } plTable[] = {
258cae415b84125ffc14bfdb121394a60b74d506f48Yanli        {OMX_VIDEO_H263ProfileBaseline, OMX_VIDEO_H263Level45}
259cae415b84125ffc14bfdb121394a60b74d506f48Yanli    };
260cae415b84125ffc14bfdb121394a60b74d506f48Yanli
261cae415b84125ffc14bfdb121394a60b74d506f48Yanli    OMX_U32 count = sizeof(plTable)/sizeof(ProfileLevelTable);
262cae415b84125ffc14bfdb121394a60b74d506f48Yanli    CHECK_ENUMERATION_RANGE(p->nProfileIndex,count);
263cae415b84125ffc14bfdb121394a60b74d506f48Yanli
264cae415b84125ffc14bfdb121394a60b74d506f48Yanli    p->eProfile = plTable[p->nProfileIndex].profile;
265cae415b84125ffc14bfdb121394a60b74d506f48Yanli    p->eLevel = plTable[p->nProfileIndex].level;
266cae415b84125ffc14bfdb121394a60b74d506f48Yanli
267cae415b84125ffc14bfdb121394a60b74d506f48Yanli    return OMX_ErrorNone;
268cae415b84125ffc14bfdb121394a60b74d506f48Yanli}
269cae415b84125ffc14bfdb121394a60b74d506f48Yanli
270cae415b84125ffc14bfdb121394a60b74d506f48YanliOMX_ERRORTYPE OMXVideoEncoderH263::SetParamVideoProfileLevelQuerySupported(OMX_PTR pStructure) {
271cae415b84125ffc14bfdb121394a60b74d506f48Yanli    LOGW("SetParamVideoH263ProfileLevel is not supported.");
272cae415b84125ffc14bfdb121394a60b74d506f48Yanli    return OMX_ErrorUnsupportedSetting;
273cae415b84125ffc14bfdb121394a60b74d506f48Yanli}
274cae415b84125ffc14bfdb121394a60b74d506f48Yanli
275e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMX_ERRORTYPE OMXVideoEncoderH263::GetParamVideoH263(OMX_PTR pStructure) {
276e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    OMX_ERRORTYPE ret;
277e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    OMX_VIDEO_PARAM_H263TYPE *p = (OMX_VIDEO_PARAM_H263TYPE *)pStructure;
278e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    CHECK_TYPE_HEADER(p);
279e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    CHECK_PORT_INDEX(p, OUTPORT_INDEX);
280e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
281e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    memcpy(p, &mParamH263, sizeof(*p));
282e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    return OMX_ErrorNone;
283e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
284e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
285e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMX_ERRORTYPE OMXVideoEncoderH263::SetParamVideoH263(OMX_PTR pStructure) {
286e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    OMX_ERRORTYPE ret;
287e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    OMX_VIDEO_PARAM_H263TYPE *p = (OMX_VIDEO_PARAM_H263TYPE *)pStructure;
288e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    CHECK_TYPE_HEADER(p);
289e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    CHECK_PORT_INDEX(p, OUTPORT_INDEX);
290e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    CHECK_SET_PARAM_STATE();
291e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
292e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    // TODO: do we need to check if port is enabled?
293e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    // TODO: see SetPortH263Param implementation - Can we make simple copy????
294e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    memcpy(&mParamH263, p, sizeof(mParamH263));
295e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    return OMX_ErrorNone;
296e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
297e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
298e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
299e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuDECLARE_OMX_COMPONENT("OMX.Intel.VideoEncoder.H263", "video_encoder.h263", OMXVideoEncoderH263);
300e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
301