blob: ec998f66ce36eb0eccd4c357379e2d9bc881c020 [file] [log] [blame]
/*
* Copyright (c) 2010, Texas Instruments Incorporated
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* * Neither the name of Texas Instruments Incorporated nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
/**
* @file omx_proxy_h264enc.c
* This file contains methods that provides the functionality for
* the OpenMAX1.1 DOMX Framework Proxy component.
*********************************************************************************************
This is the proxy specific wrapper that passes the component name to the generic proxy init()
The proxy wrapper also does some runtime/static time onfig on per proxy basis
This is a thin wrapper that is called when componentiit() of the proxy is called
static OMX_ERRORTYPE PROXY_Wrapper_init(OMX_HANDLETYPE hComponent, OMX_PTR pAppData);
this layer gets called first whenever a proxy's get handle is called
************************************************************************************************
* @path WTSD_DucatiMMSW\omx\omx_il_1_x\omx_proxy_component\src
*
* @rev 1.0
*/
/*==============================================================
*! Revision History
*! ============================
* 26-August-2011 Abhishek Ranka : Support for color conv at encoder
* input port
*
*! 20-August-2010 Sarthak Aggarwal sarthak@ti.com: Initial Version
*================================================================*/
/******************************************************************
* INCLUDE FILES
******************************************************************/
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include "omx_proxy_common.h"
#include <timm_osal_interfaces.h>
#include "OMX_TI_IVCommon.h"
#include "OMX_TI_Video.h"
#include "OMX_TI_Index.h"
#include <MetadataBufferType.h>
#ifdef ENABLE_GRALLOC_BUFFER
#include "native_handle.h"
#include <hal_public.h>
#include <VideoMetadata.h>
#endif
#define COMPONENT_NAME "OMX.TI.DUCATI1.VIDEO.H264E"
/* needs to be specific for every configuration wrapper */
#define OMX_H264E_INPUT_PORT 0
#define LINUX_PAGE_SIZE 4096
#ifdef ANDROID_QUIRK_CHANGE_PORT_VALUES
OMX_ERRORTYPE LOCAL_PROXY_H264E_GetParameter(OMX_IN OMX_HANDLETYPE hComponent,
OMX_IN OMX_INDEXTYPE nParamIndex, OMX_INOUT OMX_PTR pParamStruct);
OMX_ERRORTYPE LOCAL_PROXY_H264E_SetParameter(OMX_IN OMX_HANDLETYPE hComponent,
OMX_IN OMX_INDEXTYPE nParamIndex, OMX_INOUT OMX_PTR pParamStruct);
#endif
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
/* Opaque color format requires below quirks to be enabled
* ENABLE_GRALLOC_BUFFER
* ANDROID_QUIRCK_CHANGE_PORT_VALUES
*/
#define OMX_H264VE_NUM_INTERNAL_BUF (8)
#define HAL_PIXEL_FORMAT_TI_NV12 (0x100)
#define COLORCONVERT_MAX_SUB_BUFFERS (3)
#define COLORCONVERT_BUFTYPE_VIRTUAL (0x0)
#define COLORCONVERT_BUFTYPE_ION (0x1)
#define COLORCONVERT_BUFTYPE_GRALLOCOPAQUE (0x2)
int COLORCONVERT_open(void **hCC, PROXY_COMPONENT_PRIVATE *pCompPrv);
int COLORCONVERT_PlatformOpaqueToNV12(void *hCC, void *pSrc[],
void *pDst[], int nWidth,
int nHeight, int nStride,
int nSrcBufType, int nDstBufType);
int COLORCONVERT_close(void *hCC,PROXY_COMPONENT_PRIVATE *pCompPrv);
static OMX_ERRORTYPE LOCAL_PROXY_H264E_AllocateBuffer(OMX_IN OMX_HANDLETYPE hComponent,
OMX_INOUT OMX_BUFFERHEADERTYPE ** ppBufferHdr, OMX_IN OMX_U32 nPortIndex,
OMX_IN OMX_PTR pAppPrivate, OMX_IN OMX_U32 nSizeBytes);
static OMX_ERRORTYPE LOCAL_PROXY_H264E_FreeBuffer(OMX_IN OMX_HANDLETYPE hComponent,
OMX_IN OMX_U32 nPortIndex, OMX_IN OMX_BUFFERHEADERTYPE * pBufferHdr);
static OMX_ERRORTYPE LOCAL_PROXY_H264E_ComponentDeInit(OMX_HANDLETYPE hComponent);
typedef struct _OMX_PROXY_H264E_PRIVATE
{
OMX_PTR hBufPipe;
OMX_BOOL bAndroidOpaqueFormat;
OMX_PTR hCC;
IMG_native_handle_t* gralloc_handle[OMX_H264VE_NUM_INTERNAL_BUF];
OMX_S32 nCurBufIndex;
alloc_device_t* mAllocDev;
}OMX_PROXY_H264E_PRIVATE;
#endif
OMX_ERRORTYPE LOCAL_PROXY_H264E_GetExtensionIndex(OMX_IN OMX_HANDLETYPE hComponent,
OMX_IN OMX_STRING cParameterName, OMX_OUT OMX_INDEXTYPE * pIndexType);
OMX_ERRORTYPE LOCAL_PROXY_H264E_EmptyThisBuffer(OMX_HANDLETYPE hComponent,
OMX_BUFFERHEADERTYPE * pBufferHdr);
OMX_ERRORTYPE OMX_ComponentInit(OMX_HANDLETYPE hComponent)
{
OMX_ERRORTYPE eError = OMX_ErrorNone;
OMX_COMPONENTTYPE *pHandle = NULL;
PROXY_COMPONENT_PRIVATE *pComponentPrivate = NULL;
pHandle = (OMX_COMPONENTTYPE *) hComponent;
OMX_TI_PARAM_ENHANCEDPORTRECONFIG tParamStruct;
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
TIMM_OSAL_ERRORTYPE eOSALStatus = TIMM_OSAL_ERR_NONE;
OMX_PROXY_H264E_PRIVATE *pProxy = NULL;
#endif
DOMX_ENTER("");
DOMX_DEBUG("Component name provided is %s", COMPONENT_NAME);
pHandle->pComponentPrivate =
(PROXY_COMPONENT_PRIVATE *)
TIMM_OSAL_Malloc(sizeof(PROXY_COMPONENT_PRIVATE), TIMM_OSAL_TRUE,
0, TIMMOSAL_MEM_SEGMENT_INT);
PROXY_assert(pHandle->pComponentPrivate != NULL,
OMX_ErrorInsufficientResources,
"ERROR IN ALLOCATING PROXY COMPONENT PRIVATE STRUCTURE");
pComponentPrivate =
(PROXY_COMPONENT_PRIVATE *) pHandle->pComponentPrivate;
TIMM_OSAL_Memset(pComponentPrivate, 0,
sizeof(PROXY_COMPONENT_PRIVATE));
pComponentPrivate->cCompName =
TIMM_OSAL_Malloc(MAX_COMPONENT_NAME_LENGTH * sizeof(OMX_U8),
TIMM_OSAL_TRUE, 0, TIMMOSAL_MEM_SEGMENT_INT);
PROXY_assert(pComponentPrivate->cCompName != NULL,
OMX_ErrorInsufficientResources,
" Error in Allocating space for proxy component table");
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
pComponentPrivate->pCompProxyPrv =
(OMX_PROXY_H264E_PRIVATE *)
TIMM_OSAL_Malloc(sizeof(OMX_PROXY_H264E_PRIVATE), TIMM_OSAL_TRUE,
0, TIMMOSAL_MEM_SEGMENT_INT);
PROXY_assert(pComponentPrivate->pCompProxyPrv != NULL,
OMX_ErrorInsufficientResources,
" Could not allocate proxy component private");
TIMM_OSAL_Memset(pComponentPrivate->pCompProxyPrv, 0,
sizeof(OMX_PROXY_H264E_PRIVATE));
pProxy = (OMX_PROXY_H264E_PRIVATE *) pComponentPrivate->pCompProxyPrv;
/* Create Pipe of for encoder input buffers */
eOSALStatus = TIMM_OSAL_CreatePipe(&pProxy->hBufPipe, sizeof(OMX_U32),
OMX_H264VE_NUM_INTERNAL_BUF, 1);
PROXY_assert(eOSALStatus == TIMM_OSAL_ERR_NONE,
OMX_ErrorInsufficientResources,
"Pipe creation failed");
#endif
// Copying component Name - this will be picked up in the proxy common
PROXY_assert(strlen(COMPONENT_NAME) + 1 < MAX_COMPONENT_NAME_LENGTH,
OMX_ErrorInvalidComponentName,
"Length of component name is longer than the max allowed");
TIMM_OSAL_Memcpy(pComponentPrivate->cCompName, COMPONENT_NAME,
strlen(COMPONENT_NAME) + 1);
eError = OMX_ProxyCommonInit(hComponent); // Calling Proxy Common Init()
#ifdef ANDROID_QUIRK_CHANGE_PORT_VALUES
pHandle->SetParameter = LOCAL_PROXY_H264E_SetParameter;
pHandle->GetParameter = LOCAL_PROXY_H264E_GetParameter;
#endif
pComponentPrivate->IsLoadedState = OMX_TRUE;
pHandle->EmptyThisBuffer = LOCAL_PROXY_H264E_EmptyThisBuffer;
pHandle->GetExtensionIndex = LOCAL_PROXY_H264E_GetExtensionIndex;
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
pHandle->ComponentDeInit = LOCAL_PROXY_H264E_ComponentDeInit;
pHandle->FreeBuffer = LOCAL_PROXY_H264E_FreeBuffer;
pHandle->AllocateBuffer = LOCAL_PROXY_H264E_AllocateBuffer;
#endif
EXIT:
if (eError != OMX_ErrorNone)
{
DOMX_DEBUG("Error in Initializing Proxy");
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
if(pProxy->hBufPipe != NULL)
{
TIMM_OSAL_DeletePipe(pProxy->hBufPipe);
pProxy->hBufPipe = NULL;
}
if(pComponentPrivate->pCompProxyPrv != NULL)
{
TIMM_OSAL_Free(pComponentPrivate->pCompProxyPrv);
pComponentPrivate->pCompProxyPrv = NULL;
pProxy = NULL;
}
#endif
if (pComponentPrivate->cCompName != NULL)
{
TIMM_OSAL_Free(pComponentPrivate->cCompName);
pComponentPrivate->cCompName = NULL;
}
if (pComponentPrivate != NULL)
{
TIMM_OSAL_Free(pComponentPrivate);
pComponentPrivate = NULL;
}
}
return eError;
}
#ifdef ANDROID_QUIRK_CHANGE_PORT_VALUES
/* ===========================================================================*/
/**
* @name PROXY_H264E_GetParameter()
* @brief
* @param void
* @return OMX_ErrorNone = Successful
* @sa TBD
*
*/
/* ===========================================================================*/
OMX_ERRORTYPE LOCAL_PROXY_H264E_GetParameter(OMX_IN OMX_HANDLETYPE hComponent,
OMX_IN OMX_INDEXTYPE nParamIndex, OMX_INOUT OMX_PTR pParamStruct)
{
OMX_ERRORTYPE eError = OMX_ErrorNone;
PROXY_COMPONENT_PRIVATE *pCompPrv = NULL;
OMX_COMPONENTTYPE *hComp = (OMX_COMPONENTTYPE *) hComponent;
OMX_PARAM_PORTDEFINITIONTYPE* pPortDef = NULL;
OMX_VIDEO_PARAM_PORTFORMATTYPE* pPortParam = NULL;
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
OMX_PROXY_H264E_PRIVATE *pProxy = NULL;
#endif
PROXY_require((pParamStruct != NULL), OMX_ErrorBadParameter, NULL);
PROXY_assert((hComp->pComponentPrivate != NULL),
OMX_ErrorBadParameter, NULL);
pCompPrv = (PROXY_COMPONENT_PRIVATE *) hComp->pComponentPrivate;
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
pProxy = (OMX_PROXY_H264E_PRIVATE *) pCompPrv->pCompProxyPrv;
#endif
DOMX_ENTER
("hComponent = %p, pCompPrv = %p, nParamIndex = %d, pParamStruct = %p",
hComponent, pCompPrv, nParamIndex, pParamStruct);
eError = PROXY_GetParameter(hComponent,nParamIndex, pParamStruct);
if(nParamIndex == OMX_IndexParamPortDefinition)
{
pPortDef = (OMX_PARAM_PORTDEFINITIONTYPE *)pParamStruct;
if(pPortDef->format.video.eColorFormat == OMX_COLOR_FormatYUV420PackedSemiPlanar)
{
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
if(pProxy->bAndroidOpaqueFormat == OMX_TRUE)
{
pPortDef->format.video.eColorFormat = OMX_COLOR_FormatAndroidOpaque;
}
else
#endif
{
pPortDef->format.video.eColorFormat = OMX_TI_COLOR_FormatYUV420PackedSemiPlanar;
}
}
if(pPortDef->nPortIndex == OMX_H264E_INPUT_PORT)
{
if(pCompPrv->proxyPortBuffers[OMX_H264E_INPUT_PORT].proxyBufferType == EncoderMetadataPointers)
{
pPortDef->nBufferSize = sizeof(video_metadata_t);
}
}
}
else if (nParamIndex == OMX_IndexParamVideoPortFormat)
{
pPortParam = (OMX_VIDEO_PARAM_PORTFORMATTYPE *)pParamStruct;
if((eError == OMX_ErrorNone) &&
(pPortParam->eColorFormat == OMX_COLOR_FormatYUV420PackedSemiPlanar))
{
pPortParam->eColorFormat = OMX_TI_COLOR_FormatYUV420PackedSemiPlanar;
}
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
else if ((eError == OMX_ErrorNoMore) &&
(pPortParam->nIndex == 1))
{
/* HACK:Remote OMX-H264E supports only 1 color format (index 0). The
* OMX_COLOR_FormatAndroidOpaque is supported only at the proxy.
* Call GetParameter() to fill in defaults for parameters and
* override color format and index for the additional
* OMX_COLOR_FormatAndroidOpaque support*/
pPortParam->nIndex = 0;
eError = PROXY_GetParameter(hComponent, nParamIndex, pParamStruct);
pPortParam->nIndex = 1;
pPortParam->eColorFormat = OMX_COLOR_FormatAndroidOpaque;
eError = OMX_ErrorNone;
}
#endif
}
PROXY_assert((eError == OMX_ErrorNone) || (eError == OMX_ErrorNoMore),
eError," Error in Proxy GetParameter");
EXIT:
DOMX_EXIT("eError: %d", eError);
return eError;
}
/* ===========================================================================*/
/**
* @name PROXY_H264E_SetParameter()
* @brief
* @param void
* @return OMX_ErrorNone = Successful
* @sa TBD
*
*/
/* ===========================================================================*/
OMX_ERRORTYPE LOCAL_PROXY_H264E_SetParameter(OMX_IN OMX_HANDLETYPE hComponent,
OMX_IN OMX_INDEXTYPE nParamIndex, OMX_IN OMX_PTR pParamStruct)
{
OMX_ERRORTYPE eError = OMX_ErrorNone;
PROXY_COMPONENT_PRIVATE *pCompPrv = NULL;
OMX_COMPONENTTYPE *hComp = (OMX_COMPONENTTYPE *) hComponent;
OMX_PARAM_PORTDEFINITIONTYPE* pPortDef = NULL;
OMX_VIDEO_PARAM_PORTFORMATTYPE* pPortParams = NULL;
OMX_VIDEO_STOREMETADATAINBUFFERSPARAMS* pStoreMetaData = NULL;
OMX_TI_PARAM_BUFFERPREANNOUNCE tParamSetNPA;
OMX_PARAM_PORTDEFINITIONTYPE sPortDef;
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
OMX_PROXY_H264E_PRIVATE *pProxy = NULL;
#endif
DOMX_ENTER
("hComponent = %p, pCompPrv = %p, nParamIndex = %d, pParamStruct = %p",
hComponent, pCompPrv, nParamIndex, pParamStruct);
PROXY_require((pParamStruct != NULL), OMX_ErrorBadParameter, NULL);
PROXY_require((hComp->pComponentPrivate != NULL),
OMX_ErrorBadParameter, NULL);
pCompPrv = (PROXY_COMPONENT_PRIVATE *) hComp->pComponentPrivate;
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
pProxy = (OMX_PROXY_H264E_PRIVATE *) pCompPrv->pCompProxyPrv;
#endif
if(nParamIndex == OMX_IndexParamPortDefinition)
{
pPortDef = (OMX_PARAM_PORTDEFINITIONTYPE *)pParamStruct;
if(pPortDef->format.video.eColorFormat == OMX_TI_COLOR_FormatYUV420PackedSemiPlanar)
{
pPortDef->format.video.eColorFormat = OMX_COLOR_FormatYUV420PackedSemiPlanar;
}
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
else if(pPortDef->format.video.eColorFormat == OMX_COLOR_FormatAndroidOpaque)
{
if(COLORCONVERT_open(&pProxy->hCC,pCompPrv) != 0)
{
PROXY_assert(0, OMX_ErrorInsufficientResources,
"Failed to open Color converting service");
}
pProxy->bAndroidOpaqueFormat = OMX_TRUE;
pPortDef->format.video.eColorFormat = OMX_COLOR_FormatYUV420PackedSemiPlanar;
}
#endif
}
else if(nParamIndex == OMX_IndexParamVideoPortFormat)
{
pPortParams = (OMX_VIDEO_PARAM_PORTFORMATTYPE *)pParamStruct;
if(pPortParams->eColorFormat == OMX_TI_COLOR_FormatYUV420PackedSemiPlanar)
{
pPortParams->eColorFormat = OMX_COLOR_FormatYUV420PackedSemiPlanar;
}
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
else if(pPortParams->eColorFormat == OMX_COLOR_FormatAndroidOpaque)
{
if(COLORCONVERT_open(&pProxy->hCC,pCompPrv) != 0)
{
PROXY_assert(0, OMX_ErrorInsufficientResources,
"Failed to open Color converting service");
}
pProxy->bAndroidOpaqueFormat = OMX_TRUE;
pPortParams->eColorFormat = OMX_COLOR_FormatYUV420PackedSemiPlanar;
}
#endif
}
else if(nParamIndex == (OMX_INDEXTYPE) OMX_TI_IndexEncoderStoreMetadatInBuffers)
{
pStoreMetaData = (OMX_VIDEO_STOREMETADATAINBUFFERSPARAMS *) pParamStruct;
DOMX_DEBUG("Moving to Metadatamode");
if (pStoreMetaData->nPortIndex == OMX_H264E_INPUT_PORT && pStoreMetaData->bStoreMetaData == OMX_TRUE)
{
tParamSetNPA.nSize = sizeof(OMX_TI_PARAM_BUFFERPREANNOUNCE);
tParamSetNPA.nVersion.s.nVersionMajor = OMX_VER_MAJOR;
tParamSetNPA.nVersion.s.nVersionMinor = OMX_VER_MINOR;
tParamSetNPA.nVersion.s.nRevision = 0x0;
tParamSetNPA.nVersion.s.nStep = 0x0;
tParamSetNPA.nPortIndex = OMX_H264E_INPUT_PORT;
tParamSetNPA.bEnabled = OMX_FALSE;
//Call NPA on OMX encoder on ducati.
PROXY_SetParameter(hComponent,OMX_TI_IndexParamBufferPreAnnouncement, &tParamSetNPA);
pCompPrv->proxyPortBuffers[pStoreMetaData->nPortIndex].proxyBufferType = EncoderMetadataPointers;
DOMX_DEBUG("Moving to Metadatamode done");
/*Initializing Structure */
sPortDef.nSize = sizeof(OMX_PARAM_PORTDEFINITIONTYPE);
sPortDef.nVersion.s.nVersionMajor = OMX_VER_MAJOR;
sPortDef.nVersion.s.nVersionMinor = OMX_VER_MINOR;
sPortDef.nVersion.s.nRevision = 0x0;
sPortDef.nVersion.s.nStep = 0x0;
sPortDef.nPortIndex = OMX_H264E_INPUT_PORT;
eError = PROXY_GetParameter(hComponent,OMX_IndexParamPortDefinition, &sPortDef);
PROXY_assert(eError == OMX_ErrorNone, eError," Error in Proxy GetParameter for Port Def");
sPortDef.format.video.nStride = LINUX_PAGE_SIZE;
eError = PROXY_SetParameter(hComponent,OMX_IndexParamPortDefinition, &sPortDef);
PROXY_assert(eError == OMX_ErrorNone, eError," Error in Proxy SetParameter for Port Def");
}
goto EXIT;
}
eError = PROXY_SetParameter(hComponent, nParamIndex, pParamStruct);
PROXY_assert(eError == OMX_ErrorNone,
eError," Error in Proxy SetParameter");
EXIT:
DOMX_EXIT("eError: %d", eError);
return eError;
}
#endif
/* ===========================================================================*/
/**
* @name PROXY_GetExtensionIndex()
* @brief
* @param void
* @return OMX_ErrorNone = Successful
* @sa TBD
*
*/
/* ===========================================================================*/
OMX_ERRORTYPE LOCAL_PROXY_H264E_GetExtensionIndex(OMX_IN OMX_HANDLETYPE hComponent,
OMX_IN OMX_STRING cParameterName, OMX_OUT OMX_INDEXTYPE * pIndexType)
{
OMX_ERRORTYPE eError = OMX_ErrorNone;
PROXY_COMPONENT_PRIVATE *pCompPrv = NULL;
OMX_COMPONENTTYPE *hComp = hComponent;
PROXY_require((hComp->pComponentPrivate != NULL),
OMX_ErrorBadParameter, NULL);
PROXY_require(cParameterName != NULL, OMX_ErrorBadParameter, NULL);
PROXY_require(pIndexType != NULL, OMX_ErrorBadParameter, NULL);
pCompPrv = (PROXY_COMPONENT_PRIVATE *) hComp->pComponentPrivate;
DOMX_ENTER("%s hComponent = %p, pCompPrv = %p, cParameterName = %s",
__FUNCTION__,hComponent, pCompPrv, cParameterName);
// Check for NULL Parameters
PROXY_require((cParameterName != NULL && pIndexType != NULL),
OMX_ErrorBadParameter, NULL);
// Ensure that String length is not greater than Max allowed length
PROXY_require(strlen(cParameterName) <= 127, OMX_ErrorBadParameter, NULL);
if(strcmp(cParameterName, "OMX.google.android.index.storeMetaDataInBuffers") == 0)
{
// If Index type is 2D Buffer Allocated Dimension
*pIndexType = (OMX_INDEXTYPE) OMX_TI_IndexEncoderStoreMetadatInBuffers;
goto EXIT;
}
PROXY_GetExtensionIndex(hComponent, cParameterName, pIndexType);
EXIT:
DOMX_EXIT("%s eError: %d",__FUNCTION__, eError);
return eError;
}
/* ===========================================================================*/
/**
* @name PROXY_H264E_EmptyThisBuffer()
* @brief
* @param void
* @return OMX_ErrorNone = Successful
* @sa TBD
*
*/
/* ===========================================================================*/
OMX_ERRORTYPE LOCAL_PROXY_H264E_EmptyThisBuffer(OMX_HANDLETYPE hComponent,
OMX_BUFFERHEADERTYPE * pBufferHdr)
{
OMX_ERRORTYPE eError = OMX_ErrorNone;
PROXY_COMPONENT_PRIVATE *pCompPrv;
OMX_COMPONENTTYPE *hComp = (OMX_COMPONENTTYPE *) hComponent;
OMX_PTR pBufferOrig = pBufferHdr->pBuffer;
OMX_U32 nStride = 0, nNumLines = 0;
OMX_PARAM_PORTDEFINITIONTYPE tParamStruct;
OMX_U32 nFilledLen, nAllocLen;
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
OMX_PROXY_H264E_PRIVATE *pProxy = NULL;
TIMM_OSAL_ERRORTYPE eOSALStatus = TIMM_OSAL_ERR_NONE;
OMX_U32 nBufIndex = 0, nSize=0, nRet=0;
#endif
PROXY_require(pBufferHdr != NULL, OMX_ErrorBadParameter, NULL);
PROXY_require(hComp->pComponentPrivate != NULL, OMX_ErrorBadParameter,
NULL);
PROXY_CHK_VERSION(pBufferHdr, OMX_BUFFERHEADERTYPE);
pCompPrv = (PROXY_COMPONENT_PRIVATE *) hComp->pComponentPrivate;
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
pProxy = (OMX_PROXY_H264E_PRIVATE *) pCompPrv->pCompProxyPrv;
#endif
tParamStruct.nSize = sizeof(OMX_PARAM_PORTDEFINITIONTYPE);
tParamStruct.nVersion.s.nVersionMajor = OMX_VER_MAJOR;
tParamStruct.nVersion.s.nVersionMinor = OMX_VER_MINOR;
tParamStruct.nVersion.s.nRevision = 0x0;
tParamStruct.nVersion.s.nStep = 0x0;
tParamStruct.nPortIndex = OMX_H264E_INPUT_PORT;
eError = PROXY_GetParameter(hComponent, OMX_IndexParamPortDefinition, &tParamStruct);
PROXY_require(eError == OMX_ErrorNone, OMX_ErrorBadParameter, "Error is Get Parameter for port def");
nFilledLen = pBufferHdr->nFilledLen;
nAllocLen = pBufferHdr->nAllocLen;
if(nFilledLen != 0)
{
pBufferHdr->nFilledLen = tParamStruct.nBufferSize;
}
pBufferHdr->nAllocLen = tParamStruct.nBufferSize;
DOMX_DEBUG
("%s hComponent=%p, pCompPrv=%p, nFilledLen=%d, nOffset=%d, nFlags=%08x",
__FUNCTION__,hComponent, pCompPrv, pBufferHdr->nFilledLen,
pBufferHdr->nOffset, pBufferHdr->nFlags);
if( pCompPrv->proxyPortBuffers[OMX_H264E_INPUT_PORT].proxyBufferType == EncoderMetadataPointers && nFilledLen != 0 )
{
OMX_U32 *pTempBuffer;
OMX_U32 nMetadataBufferType;
DOMX_DEBUG("Passing meta data to encoder");
pBufferOrig = pBufferHdr->pBuffer;
pTempBuffer = (OMX_U32 *) (pBufferHdr->pBuffer);
nMetadataBufferType = *pTempBuffer;
if(nMetadataBufferType == kMetadataBufferTypeCameraSource)
{
#ifdef ENABLE_GRALLOC_BUFFER
IMG_native_handle_t* pGrallocHandle;
video_metadata_t* pVideoMetadataBuffer;
DOMX_DEBUG("MetadataBufferType is kMetadataBufferTypeCameraSource");
pVideoMetadataBuffer = (video_metadata_t*) ((OMX_U32 *)(pBufferHdr->pBuffer));
pGrallocHandle = (IMG_native_handle_t*) (pVideoMetadataBuffer->handle);
DOMX_DEBUG("Grallloc buffer recieved in metadata buffer 0x%x",pGrallocHandle );
pBufferHdr->pBuffer = (OMX_U8 *)(pGrallocHandle->fd[0]);
((OMX_TI_PLATFORMPRIVATE *) pBufferHdr->pPlatformPrivate)->
pAuxBuf1 = (OMX_PTR) pGrallocHandle->fd[1];
DOMX_DEBUG("%s Gralloc=0x%x, Y-fd=%d, UV-fd=%d", __FUNCTION__, pGrallocHandle,
pGrallocHandle->fd[0], pGrallocHandle->fd[1]);
pBufferHdr->nOffset = pVideoMetadataBuffer->offset;
#endif
}
else if(nMetadataBufferType == kMetadataBufferTypeGrallocSource)
{
#ifdef ENABLE_GRALLOC_BUFFER
IMG_native_handle_t* pGrallocHandle;
buffer_handle_t tBufHandle;
DOMX_DEBUG("MetadataBufferType is kMetadataBufferTypeGrallocSource");
pTempBuffer++;
tBufHandle = *((buffer_handle_t *)pTempBuffer);
pGrallocHandle = (IMG_native_handle_t*) tBufHandle;
DOMX_DEBUG("Grallloc buffer recieved in metadata buffer 0x%x",pGrallocHandle );
pBufferHdr->pBuffer = (OMX_U8 *)(pGrallocHandle->fd[0]);
((OMX_TI_PLATFORMPRIVATE *) pBufferHdr->pPlatformPrivate)->
pAuxBuf1 = (OMX_PTR) pGrallocHandle->fd[1];
DOMX_DEBUG("%s Gralloc=0x%x, Y-fd=%d, UV-fd=%d", __FUNCTION__, pGrallocHandle,
pGrallocHandle->fd[0], pGrallocHandle->fd[1]);
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
if (pProxy->bAndroidOpaqueFormat)
{
/* Dequeue NV12 buffer for encoder */
eOSALStatus = TIMM_OSAL_ReadFromPipe(pProxy->hBufPipe, &nBufIndex,
sizeof(OMX_PTR), (TIMM_OSAL_U32 *)(&nSize),
TIMM_OSAL_SUSPEND);
PROXY_assert(eOSALStatus == TIMM_OSAL_ERR_NONE, OMX_ErrorBadParameter, NULL);
/* Get NV12 data after colorconv*/
nRet = COLORCONVERT_PlatformOpaqueToNV12(pProxy->hCC, (void **) &pGrallocHandle, (void **) &pProxy->gralloc_handle[nBufIndex],
pGrallocHandle->iWidth,
pGrallocHandle->iHeight,
4096, COLORCONVERT_BUFTYPE_GRALLOCOPAQUE,
COLORCONVERT_BUFTYPE_GRALLOCOPAQUE );
if(nRet != 0)
{
eOSALStatus = TIMM_OSAL_WriteToPipe(pProxy->hBufPipe, (void *) &nBufIndex,
sizeof(OMX_U32), TIMM_OSAL_SUSPEND);
PROXY_assert(0, OMX_ErrorBadParameter, "Color conversion routine failed");
}
/* Update pBufferHdr with NV12 buffers for OMX component */
pBufferHdr->pBuffer= pProxy->gralloc_handle[nBufIndex]->fd[0];
((OMX_TI_PLATFORMPRIVATE *) pBufferHdr->pPlatformPrivate)->pAuxBuf1 = pProxy->gralloc_handle[nBufIndex]->fd[1];
}
#endif
#endif
}
else
{
return OMX_ErrorBadParameter;
}
}
PROXY_EmptyThisBuffer(hComponent, pBufferHdr);
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
if (pProxy->bAndroidOpaqueFormat)
{
/*Write buffer to end of pipe for re-circulation for future ETB()*/
eOSALStatus = TIMM_OSAL_WriteToPipe(pProxy->hBufPipe, (void *) &nBufIndex,
sizeof(OMX_U32), TIMM_OSAL_SUSPEND);
PROXY_assert(eOSALStatus == TIMM_OSAL_ERR_NONE, OMX_ErrorBadParameter, "Pipe write failed");
}
#endif
if( pCompPrv->proxyPortBuffers[pBufferHdr->nInputPortIndex].proxyBufferType == EncoderMetadataPointers)
{
pBufferHdr->pBuffer = pBufferOrig;
pBufferHdr->nFilledLen = nFilledLen;
pBufferHdr->nAllocLen = nAllocLen;
}
EXIT:
return eError;
}
#ifdef ANDROID_CUSTOM_OPAQUECOLORFORMAT
static OMX_ERRORTYPE LOCAL_PROXY_H264E_AllocateBuffer(OMX_HANDLETYPE hComponent,
OMX_BUFFERHEADERTYPE ** ppBufferHdr, OMX_U32 nPortIndex,
OMX_PTR pAppPrivate, OMX_U32 nSizeBytes)
{
OMX_ERRORTYPE eError = OMX_ErrorNone;
PROXY_COMPONENT_PRIVATE *pCompPrv = NULL;
OMX_COMPONENTTYPE *hComp = (OMX_COMPONENTTYPE *) hComponent;
OMX_CONFIG_RECTTYPE tParamRect;
OMX_PROXY_H264E_PRIVATE *pProxy = NULL;
TIMM_OSAL_ERRORTYPE eOSALStatus = TIMM_OSAL_ERR_NONE;
int err, nStride;
PROXY_require(hComp->pComponentPrivate != NULL, OMX_ErrorBadParameter,
NULL);
pCompPrv = (PROXY_COMPONENT_PRIVATE *) hComp->pComponentPrivate;
pProxy = (OMX_PROXY_H264E_PRIVATE *) pCompPrv->pCompProxyPrv;
if((nPortIndex == OMX_H264E_INPUT_PORT) &&
(pProxy->bAndroidOpaqueFormat))
{
tParamRect.nSize = sizeof(OMX_CONFIG_RECTTYPE);
tParamRect.nVersion.s.nVersionMajor = 1;
tParamRect.nVersion.s.nVersionMinor = 1;
tParamRect.nVersion.s.nRevision = 0;
tParamRect.nVersion.s.nStep = 0;
tParamRect.nPortIndex = nPortIndex;
eError = PROXY_GetParameter(hComponent, (OMX_INDEXTYPE)OMX_TI_IndexParam2DBufferAllocDimension, &tParamRect);
PROXY_assert(eError == OMX_ErrorNone, eError," Error in Proxy GetParameter");
err = pProxy->mAllocDev->alloc(pProxy->mAllocDev,(int) tParamRect.nWidth,(int) tParamRect.nHeight,
(int) HAL_PIXEL_FORMAT_TI_NV12,(int) GRALLOC_USAGE_HW_RENDER, &(pProxy->gralloc_handle[pProxy->nCurBufIndex]), &nStride);
}
eError = PROXY_AllocateBuffer(hComponent, ppBufferHdr, nPortIndex,
pAppPrivate, nSizeBytes);
EXIT:
if((nPortIndex == OMX_H264E_INPUT_PORT) &&
(pProxy->bAndroidOpaqueFormat))
{
if(eError != OMX_ErrorNone)
{
err = pProxy->mAllocDev->free(pProxy->mAllocDev, pProxy->gralloc_handle[pProxy->nCurBufIndex]);
}
else
{
/*Populate buffer to pipe*/
eOSALStatus = TIMM_OSAL_WriteToPipe(pProxy->hBufPipe, (void *) &pProxy->nCurBufIndex,
sizeof(OMX_U32), TIMM_OSAL_SUSPEND);
pProxy->nCurBufIndex++;
}
}
return eError;
}
static OMX_ERRORTYPE LOCAL_PROXY_H264E_FreeBuffer(OMX_IN OMX_HANDLETYPE hComponent,
OMX_IN OMX_U32 nPortIndex, OMX_IN OMX_BUFFERHEADERTYPE * pBufferHdr)
{
OMX_ERRORTYPE eError = OMX_ErrorNone;
OMX_COMPONENTTYPE *hComp = (OMX_COMPONENTTYPE *) hComponent;
PROXY_COMPONENT_PRIVATE *pCompPrv = NULL;
OMX_U32 nBufIndex, nSize, nCount=0;
OMX_PROXY_H264E_PRIVATE *pProxy = NULL;
PROXY_require(hComp->pComponentPrivate != NULL, OMX_ErrorBadParameter,
NULL);
pCompPrv = (PROXY_COMPONENT_PRIVATE *) hComp->pComponentPrivate;
pProxy = (OMX_PROXY_H264E_PRIVATE *) pCompPrv->pCompProxyPrv;
if((nPortIndex == OMX_H264E_INPUT_PORT) &&
(pProxy->bAndroidOpaqueFormat))
{
pProxy->nCurBufIndex--;
PROXY_require(pProxy->nCurBufIndex >=0,
OMX_ErrorBadParameter, "Buffer index underflow");
if(pProxy->gralloc_handle[pProxy->nCurBufIndex])
{
pProxy->mAllocDev->free(pProxy->mAllocDev, pProxy->gralloc_handle[pProxy->nCurBufIndex]);
pProxy->gralloc_handle[pProxy->nCurBufIndex] = NULL;
}
/*Clear the Bufindex pipe by dummy reads*/
TIMM_OSAL_GetPipeReadyMessageCount(pProxy->hBufPipe, (TIMM_OSAL_U32 *)&nCount);
if(nCount)
{
TIMM_OSAL_ReadFromPipe(pProxy->hBufPipe, &nBufIndex,
sizeof(OMX_PTR), (TIMM_OSAL_U32 *)&nSize, TIMM_OSAL_NO_SUSPEND);
}
}
eError = PROXY_FreeBuffer(hComponent, nPortIndex, pBufferHdr);
EXIT:
return eError;
}
OMX_ERRORTYPE LOCAL_PROXY_H264E_ComponentDeInit(OMX_HANDLETYPE hComponent)
{
OMX_ERRORTYPE eError = OMX_ErrorNone;
PROXY_COMPONENT_PRIVATE *pCompPrv;
OMX_COMPONENTTYPE *hComp = (OMX_COMPONENTTYPE *) hComponent;
OMX_PROXY_H264E_PRIVATE *pProxy = NULL;
TIMM_OSAL_ERRORTYPE eOSALStatus = TIMM_OSAL_ERR_NONE;
OMX_U32 i;
PROXY_require(hComp->pComponentPrivate != NULL, OMX_ErrorBadParameter,
NULL);
pCompPrv = (PROXY_COMPONENT_PRIVATE *) hComp->pComponentPrivate;
pProxy = (OMX_PROXY_H264E_PRIVATE *) pCompPrv->pCompProxyPrv;
if(pProxy->hBufPipe != NULL)
{
eOSALStatus = TIMM_OSAL_DeletePipe(pProxy->hBufPipe);
pProxy->hBufPipe = NULL;
if(eOSALStatus != TIMM_OSAL_ERR_NONE)
{
DOMX_ERROR("Pipe deletion failed");
}
}
if(pProxy->bAndroidOpaqueFormat == OMX_TRUE)
{
/* Cleanup internal buffers in pipe if not freed on FreeBuffer */
for(i=0; i<OMX_H264VE_NUM_INTERNAL_BUF; i++)
{
if(pProxy->gralloc_handle[i])
{
pProxy->mAllocDev->free(pProxy->mAllocDev, pProxy->gralloc_handle[i]);
pProxy->gralloc_handle[i] = NULL;
}
}
COLORCONVERT_close(pProxy->hCC,pCompPrv);
pProxy->bAndroidOpaqueFormat = OMX_FALSE;
if(pCompPrv->pCompProxyPrv != NULL)
{
TIMM_OSAL_Free(pCompPrv->pCompProxyPrv);
pCompPrv->pCompProxyPrv = NULL;
}
}
eError = PROXY_ComponentDeInit(hComponent);
EXIT:
DOMX_EXIT("eError: %d", eError);
return eError;
}
int COLORCONVERT_open(void **hCC, PROXY_COMPONENT_PRIVATE *pCompPrv)
{
int nErr = -1;
hw_module_t const* module = NULL;
OMX_PROXY_H264E_PRIVATE *pProxy = NULL;
pProxy = (OMX_PROXY_H264E_PRIVATE *) pCompPrv->pCompProxyPrv;
nErr = hw_get_module(GRALLOC_HARDWARE_MODULE_ID, &module);
if (nErr == 0)
{
*hCC = (void *) ((IMG_gralloc_module_public_t const *)module);
}
else
{
DOMX_ERROR("FATAL: gralloc api hw_get_module() returned error: Can't find \
%s module err = %x", GRALLOC_HARDWARE_MODULE_ID, nErr);
}
gralloc_open(module, &(pProxy->mAllocDev));
return nErr;
}
int COLORCONVERT_PlatformOpaqueToNV12(void *hCC,
void *pSrc[COLORCONVERT_MAX_SUB_BUFFERS],
void *pDst[COLORCONVERT_MAX_SUB_BUFFERS],
int nWidth, int nHeight, int nStride,
int nSrcBufType,int nDstBufType)
{
IMG_gralloc_module_public_t const* module = hCC;
int nErr = -1;
if((nSrcBufType == COLORCONVERT_BUFTYPE_GRALLOCOPAQUE) && (nDstBufType == COLORCONVERT_BUFTYPE_VIRTUAL))
{
nErr = module->Blit(module, pSrc[0], pDst, HAL_PIXEL_FORMAT_TI_NV12);
}
else if((nSrcBufType == COLORCONVERT_BUFTYPE_GRALLOCOPAQUE) && (nDstBufType == COLORCONVERT_BUFTYPE_GRALLOCOPAQUE ))
{
nErr = module->Blit2(module, pSrc[0], pDst[0], nWidth, nHeight, 0, 0);
}
return nErr;
}
int COLORCONVERT_close(void *hCC,PROXY_COMPONENT_PRIVATE *pCompPrv)
{
OMX_PROXY_H264E_PRIVATE *pProxy = NULL;
pProxy = (OMX_PROXY_H264E_PRIVATE *) pCompPrv->pCompProxyPrv;
if(pProxy && pProxy->mAllocDev)
{
gralloc_close(pProxy->mAllocDev);
}
return 0;
}
#endif