OMXVideoEncoderH263.cpp revision 28516617d7c679a9d1c4d1a5a29af157eb25cc29
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
43e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    mParamH263.eLevel = OMX_VIDEO_H263Level70; //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]);
11328516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying    }
11428516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying
1154ce57b82fa719f2aa9247c7f4a44853715a29987Guoliang Ji    inBuf.data = buffers[INPORT_INDEX]->pBuffer + buffers[INPORT_INDEX]->nOffset;
1164ce57b82fa719f2aa9247c7f4a44853715a29987Guoliang Ji    inBuf.size = buffers[INPORT_INDEX]->nFilledLen;
11719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
11819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV("buffer_in.data=%x, data_size=%d",
11919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang         (unsigned)inBuf.data, inBuf.size);
12019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
12119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outBuf.data =	buffers[OUTPORT_INDEX]->pBuffer + buffers[OUTPORT_INDEX]->nOffset;
12219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outBuf.bufferSize = buffers[OUTPORT_INDEX]->nAllocLen - buffers[OUTPORT_INDEX]->nOffset;
12319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outBuf.dataSize = 0;
12419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
12519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if(mFrameRetrieved) {
12619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        // encode and setConfig need to be thread safe
12719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        pthread_mutex_unlock(&mSerializationLock);
12819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        ret = mVideoEncoder->encode(&inBuf);
12919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        pthread_mutex_unlock(&mSerializationLock);
13019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
13119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        CHECK_ENCODE_STATUS("encode");
13219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        mFrameRetrieved = OMX_FALSE;
13319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
13419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        // This is for buffer contention, we won't release current buffer
13519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        // but the last input buffer
13619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        ports[INPORT_INDEX]->ReturnAllRetainedBuffers();
13719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
13819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
13919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outBuf.format = OUTPUT_EVERYTHING;
14019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    ret = mVideoEncoder->getOutput(&outBuf);
14119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    // CHECK_ENCODE_STATUS("encode");
14219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if(ret == ENCODE_NO_REQUEST_DATA) {
14319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        mFrameRetrieved = OMX_TRUE;
14419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[OUTPORT_INDEX] = BUFFER_RETAIN_GETAGAIN;
14519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[INPORT_INDEX] = BUFFER_RETAIN_ACCUMULATE;
14619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        goto out;
14719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
14819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
14919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV("output data size = %d", outBuf.dataSize);
15019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outfilledlen = outBuf.dataSize;
15119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    outtimestamp = buffers[INPORT_INDEX]->nTimeStamp;
15219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
15319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
15419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (outBuf.flag & ENCODE_BUFFERFLAG_SYNCFRAME) {
15519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        outflags |= OMX_BUFFERFLAG_SYNCFRAME;
15619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
15719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
15819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if(outBuf.flag & ENCODE_BUFFERFLAG_ENDOFFRAME) {
15919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        outflags |= OMX_BUFFERFLAG_ENDOFFRAME;
16019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        mFrameRetrieved = OMX_TRUE;
16119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[INPORT_INDEX] = BUFFER_RETAIN_ACCUMULATE;
16219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
16319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    } else {
16419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[INPORT_INDEX] = BUFFER_RETAIN_GETAGAIN;  //get again
16519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
16619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
16719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
16819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (outfilledlen > 0) {
16919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[OUTPORT_INDEX] = BUFFER_RETAIN_NOT_RETAIN;
17019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    } else {
17119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        retains[OUTPORT_INDEX] = BUFFER_RETAIN_GETAGAIN;
17219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
17319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
17419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
17519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if(ret == ENCODE_SLICESIZE_OVERFLOW) {
17619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        LOGV("%s(), mix_video_encode returns MIX_RESULT_VIDEO_ENC_SLICESIZE_OVERFLOW"
17719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang             , __func__);
17819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        oret = (OMX_ERRORTYPE)OMX_ErrorIntelExtSliceSizeOverflow;
17919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
18019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang#if SHOW_FPS
18119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    {
18219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        struct timeval t;
18319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        OMX_TICKS current_ts, interval_ts;
18419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        float current_fps, average_fps;
18519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
18619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        t.tv_sec = t.tv_usec = 0;
18719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        gettimeofday(&t, NULL);
18819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
18919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        current_ts =(nsecs_t)t.tv_sec * 1000000000 + (nsecs_t)t.tv_usec * 1000;
19019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        interval_ts = current_ts - lastTs;
19119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        lastTs = current_ts;
19219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
19319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        current_fps = (float)1000000000 / (float)interval_ts;
19419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        average_fps = (current_fps + lastFps) / 2;
19519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        lastFps = current_fps;
19619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
19719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        LOGD("FPS = %2.1f\n", average_fps);
19819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
19919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang#endif
20019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
20119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sangout:
20219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
20319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if(retains[OUTPORT_INDEX] != BUFFER_RETAIN_GETAGAIN) {
20419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        buffers[OUTPORT_INDEX]->nFilledLen = outfilledlen;
20519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        buffers[OUTPORT_INDEX]->nTimeStamp = outtimestamp;
20619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        buffers[OUTPORT_INDEX]->nFlags = outflags;
20719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
20819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        LOGV("********** output buffer: len=%d, ts=%ld, flags=%x",
20919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang             outfilledlen,
21019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang             outtimestamp,
21119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang             outflags);
21219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
21319b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
21419b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (retains[INPORT_INDEX] == BUFFER_RETAIN_NOT_RETAIN ||
21519b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang            retains[INPORT_INDEX] == BUFFER_RETAIN_ACCUMULATE ) {
21619b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        mFrameInputCount ++;
21719b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    }
21819b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
21919b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    if (retains[OUTPORT_INDEX] == BUFFER_RETAIN_NOT_RETAIN)
22019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang        mFrameOutputCount ++;
22119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
22228516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying    if (bAndroidOpaqueFormat && buffers[INPORT_INDEX]->nFilledLen != 0) {
22328516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying        // Restore input buffer's content
22428516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying        buffers[INPORT_INDEX]->nFilledLen = 4 + sizeof(buffer_handle_t);
22528516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying        memcpy(buffers[INPORT_INDEX]->pBuffer, mBufferHandleMaps[mCurHandle].backBuffer,
22628516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying                buffers[INPORT_INDEX]->nFilledLen);
22728516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying
22828516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying    }
22928516617d7c679a9d1c4d1a5a29af157eb25cc29Chang Ying
23019b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    LOGV_IF(oret == OMX_ErrorNone, "%s(),%d: exit, encode is done\n", __func__, __LINE__);
23119b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang
23219b2ab9f325bdbf3afe530e943fa5a0c0020b308Shuduo Sang    return oret;
233e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
234e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
235e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
236e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMX_ERRORTYPE OMXVideoEncoderH263::BuildHandlerList(void) {
237e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    OMXVideoEncoderBase::BuildHandlerList();
238e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    AddHandler(OMX_IndexParamVideoH263, GetParamVideoH263, SetParamVideoH263);
239e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    return OMX_ErrorNone;
240e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
241e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
242e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMX_ERRORTYPE OMXVideoEncoderH263::GetParamVideoH263(OMX_PTR pStructure) {
243e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    OMX_ERRORTYPE ret;
244e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    OMX_VIDEO_PARAM_H263TYPE *p = (OMX_VIDEO_PARAM_H263TYPE *)pStructure;
245e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    CHECK_TYPE_HEADER(p);
246e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    CHECK_PORT_INDEX(p, OUTPORT_INDEX);
247e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
248e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    memcpy(p, &mParamH263, sizeof(*p));
249e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    return OMX_ErrorNone;
250e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
251e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
252e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuOMX_ERRORTYPE OMXVideoEncoderH263::SetParamVideoH263(OMX_PTR pStructure) {
253e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    OMX_ERRORTYPE ret;
254e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    OMX_VIDEO_PARAM_H263TYPE *p = (OMX_VIDEO_PARAM_H263TYPE *)pStructure;
255e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    CHECK_TYPE_HEADER(p);
256e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    CHECK_PORT_INDEX(p, OUTPORT_INDEX);
257e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    CHECK_SET_PARAM_STATE();
258e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
259e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    // TODO: do we need to check if port is enabled?
260e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    // TODO: see SetPortH263Param implementation - Can we make simple copy????
261e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    memcpy(&mParamH263, p, sizeof(mParamH263));
262e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu    return OMX_ErrorNone;
263e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu}
264e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
265e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
266e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy QiuDECLARE_OMX_COMPONENT("OMX.Intel.VideoEncoder.H263", "video_encoder.h263", OMXVideoEncoderH263);
267e7ace334fb7c64f7b32aa3746e5a11bcefa60bfbAndy Qiu
268