/* Copyright (c) 2013, The Linux Foundation. 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 The Linux Foundation 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 "AS IS" AND ANY EXPRESS OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
* 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.
*
*/
#include <pthread.h>
#include <errno.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <poll.h>
#include "mm_jpeg_dbg.h"
#include "mm_jpeg_interface.h"
#include "mm_jpeg.h"
#include "mm_jpeg_inlines.h"
OMX_ERRORTYPE mm_jpegdec_ebd(OMX_HANDLETYPE hComponent,
OMX_PTR pAppData,
OMX_BUFFERHEADERTYPE *pBuffer);
OMX_ERRORTYPE mm_jpegdec_fbd(OMX_HANDLETYPE hComponent,
OMX_PTR pAppData,
OMX_BUFFERHEADERTYPE* pBuffer);
OMX_ERRORTYPE mm_jpegdec_event_handler(OMX_HANDLETYPE hComponent,
OMX_PTR pAppData,
OMX_EVENTTYPE eEvent,
OMX_U32 nData1,
OMX_U32 nData2,
OMX_PTR pEventData);
/** mm_jpegdec_destroy_job
*
* Arguments:
* @p_session: Session obj
*
* Return:
* 0 for success else failure
*
* Description:
* Destroy the job based paramenters
*
**/
static int32_t mm_jpegdec_destroy_job(mm_jpeg_job_session_t *p_session)
{
mm_jpeg_decode_job_t *p_jobparams = &p_session->decode_job;
int32_t rc = 0;
return rc;
}
/** mm_jpeg_job_done:
*
* Arguments:
* @p_session: decode session
*
* Return:
* OMX_ERRORTYPE
*
* Description:
* Finalize the job
*
**/
static void mm_jpegdec_job_done(mm_jpeg_job_session_t *p_session)
{
mm_jpeg_obj *my_obj = (mm_jpeg_obj *)p_session->jpeg_obj;
mm_jpeg_job_q_node_t *node = NULL;
/*Destroy job related params*/
mm_jpegdec_destroy_job(p_session);
/*remove the job*/
node = mm_jpeg_queue_remove_job_by_job_id(&my_obj->ongoing_job_q,
p_session->jobId);
if (node) {
free(node);
}
p_session->encoding = OMX_FALSE;
/* wake up jobMgr thread to work on new job if there is any */
cam_sem_post(&my_obj->job_mgr.job_sem);
}
/** mm_jpegdec_session_send_buffers:
*
* Arguments:
* @data: job session
*
* Return:
* OMX error values
*
* Description:
* Send the buffers to OMX layer
*
**/
OMX_ERRORTYPE mm_jpegdec_session_send_buffers(void *data)
{
uint32_t i = 0;
mm_jpeg_job_session_t* p_session = (mm_jpeg_job_session_t *)data;
OMX_ERRORTYPE ret = OMX_ErrorNone;
QOMX_BUFFER_INFO lbuffer_info;
mm_jpeg_decode_params_t *p_params = &p_session->dec_params;
mm_jpeg_decode_job_t *p_jobparams = &p_session->decode_job;
memset(&lbuffer_info, 0x0, sizeof(QOMX_BUFFER_INFO));
for (i = 0; i < p_params->num_src_bufs; i++) {
CDBG("%s:%d] Source buffer %d", __func__, __LINE__, i);
lbuffer_info.fd = p_params->src_main_buf[i].fd;
ret = OMX_UseBuffer(p_session->omx_handle, &(p_session->p_in_omx_buf[i]), 0,
&lbuffer_info, p_params->src_main_buf[i].buf_size,
p_params->src_main_buf[i].buf_vaddr);
if (ret) {
CDBG_ERROR("%s:%d] Error %d", __func__, __LINE__, ret);
return ret;
}
}
CDBG("%s:%d]", __func__, __LINE__);
return ret;
}
/** mm_jpeg_session_free_buffers:
*
* Arguments:
* @data: job session
*
* Return:
* OMX error values
*
* Description:
* Free the buffers from OMX layer
*
**/
OMX_ERRORTYPE mm_jpegdec_session_free_buffers(void *data)
{
OMX_ERRORTYPE ret = OMX_ErrorNone;
uint32_t i = 0;
mm_jpeg_job_session_t* p_session = (mm_jpeg_job_session_t *)data;
mm_jpeg_decode_params_t *p_params = &p_session->dec_params;
mm_jpeg_decode_job_t *p_jobparams = &p_session->decode_job;
for (i = 0; i < p_params->num_src_bufs; i++) {
CDBG("%s:%d] Source buffer %d", __func__, __LINE__, i);
ret = OMX_FreeBuffer(p_session->omx_handle, 0, p_session->p_in_omx_buf[i]);
if (ret) {
CDBG_ERROR("%s:%d] Error %d", __func__, __LINE__, ret);
return ret;
}
}
for (i = 0; i < p_params->num_dst_bufs; i++) {
CDBG("%s:%d] Dest buffer %d", __func__, __LINE__, i);
ret = OMX_FreeBuffer(p_session->omx_handle, 1, p_session->p_out_omx_buf[i]);
if (ret) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
return ret;
}
}
CDBG("%s:%d]", __func__, __LINE__);
return ret;
}
/** mm_jpegdec_session_create:
*
* Arguments:
* @p_session: job session
*
* Return:
* OMX error types
*
* Description:
* Create a jpeg encode session
*
**/
OMX_ERRORTYPE mm_jpegdec_session_create(mm_jpeg_job_session_t* p_session)
{
OMX_ERRORTYPE rc = OMX_ErrorNone;
mm_jpeg_cirq_t *p_cirq = NULL;
pthread_mutex_init(&p_session->lock, NULL);
pthread_cond_init(&p_session->cond, NULL);
cirq_reset(&p_session->cb_q);
p_session->state_change_pending = OMX_FALSE;
p_session->abort_state = MM_JPEG_ABORT_NONE;
p_session->error_flag = OMX_ErrorNone;
p_session->ebd_count = 0;
p_session->fbd_count = 0;
p_session->encode_pid = -1;
p_session->config = OMX_FALSE;
p_session->omx_callbacks.EmptyBufferDone = mm_jpegdec_ebd;
p_session->omx_callbacks.FillBufferDone = mm_jpegdec_fbd;
p_session->omx_callbacks.EventHandler = mm_jpegdec_event_handler;
p_session->exif_count_local = 0;
rc = OMX_GetHandle(&p_session->omx_handle,
"OMX.qcom.image.jpeg.decoder",
(void *)p_session,
&p_session->omx_callbacks);
if (OMX_ErrorNone != rc) {
CDBG_ERROR("%s:%d] OMX_GetHandle failed (%d)", __func__, __LINE__, rc);
return rc;
}
return rc;
}
/** mm_jpegdec_session_destroy:
*
* Arguments:
* @p_session: job session
*
* Return:
* none
*
* Description:
* Destroy a jpeg encode session
*
**/
void mm_jpegdec_session_destroy(mm_jpeg_job_session_t* p_session)
{
OMX_ERRORTYPE rc = OMX_ErrorNone;
CDBG("%s:%d] E", __func__, __LINE__);
if (NULL == p_session->omx_handle) {
CDBG_ERROR("%s:%d] invalid handle", __func__, __LINE__);
return;
}
rc = mm_jpeg_session_change_state(p_session, OMX_StateIdle, NULL);
if (rc) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
}
rc = mm_jpeg_session_change_state(p_session, OMX_StateLoaded,
mm_jpegdec_session_free_buffers);
if (rc) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
}
rc = OMX_FreeHandle(p_session->omx_handle);
if (0 != rc) {
CDBG_ERROR("%s:%d] OMX_FreeHandle failed (%d)", __func__, __LINE__, rc);
}
p_session->omx_handle = NULL;
pthread_mutex_destroy(&p_session->lock);
pthread_cond_destroy(&p_session->cond);
CDBG("%s:%d] X", __func__, __LINE__);
}
/** mm_jpeg_session_config_port:
*
* Arguments:
* @p_session: job session
*
* Return:
* OMX error values
*
* Description:
* Configure OMX ports
*
**/
OMX_ERRORTYPE mm_jpegdec_session_config_ports(mm_jpeg_job_session_t* p_session)
{
OMX_ERRORTYPE ret = OMX_ErrorNone;
mm_jpeg_decode_params_t *p_params = &p_session->dec_params;
mm_jpeg_decode_job_t *p_jobparams = &p_session->decode_job;
mm_jpeg_buf_t *p_src_buf =
&p_params->src_main_buf[p_jobparams->src_index];
p_session->inputPort.nPortIndex = 0;
p_session->outputPort.nPortIndex = 1;
ret = OMX_GetParameter(p_session->omx_handle, OMX_IndexParamPortDefinition,
&p_session->inputPort);
if (ret) {
CDBG_ERROR("%s:%d] failed", __func__, __LINE__);
return ret;
}
ret = OMX_GetParameter(p_session->omx_handle, OMX_IndexParamPortDefinition,
&p_session->outputPort);
if (ret) {
CDBG_ERROR("%s:%d] failed", __func__, __LINE__);
return ret;
}
p_session->inputPort.format.image.nFrameWidth =
p_jobparams->main_dim.src_dim.width;
p_session->inputPort.format.image.nFrameHeight =
p_jobparams->main_dim.src_dim.height;
p_session->inputPort.format.image.nStride =
p_src_buf->offset.mp[0].stride;
p_session->inputPort.format.image.nSliceHeight =
p_src_buf->offset.mp[0].scanline;
p_session->inputPort.format.image.eColorFormat =
map_jpeg_format(p_params->color_format);
p_session->inputPort.nBufferSize =
p_params->src_main_buf[p_jobparams->src_index].buf_size;
p_session->inputPort.nBufferCountActual = p_params->num_src_bufs;
ret = OMX_SetParameter(p_session->omx_handle, OMX_IndexParamPortDefinition,
&p_session->inputPort);
if (ret) {
CDBG_ERROR("%s:%d] failed", __func__, __LINE__);
return ret;
}
return ret;
}
/** mm_jpegdec_session_config_main:
*
* Arguments:
* @p_session: job session
*
* Return:
* OMX error values
*
* Description:
* Configure main image
*
**/
OMX_ERRORTYPE mm_jpegdec_session_config_main(mm_jpeg_job_session_t *p_session)
{
OMX_ERRORTYPE rc = OMX_ErrorNone;
OMX_IMAGE_PARAM_QFACTORTYPE q_factor;
mm_jpeg_decode_params_t *p_params = &p_session->dec_params;
mm_jpeg_decode_job_t *p_jobparams = &p_session->decode_job;
/* config port */
CDBG("%s:%d] config port", __func__, __LINE__);
rc = mm_jpegdec_session_config_ports(p_session);
if (OMX_ErrorNone != rc) {
CDBG_ERROR("%s: config port failed", __func__);
return rc;
}
/* TODO: config crop */
return rc;
}
/** mm_jpeg_session_configure:
*
* Arguments:
* @data: encode session
*
* Return:
* none
*
* Description:
* Configure the session
*
**/
static OMX_ERRORTYPE mm_jpegdec_session_configure(mm_jpeg_job_session_t *p_session)
{
OMX_ERRORTYPE ret = OMX_ErrorNone;
mm_jpeg_decode_params_t *p_params = &p_session->dec_params;
mm_jpeg_decode_job_t *p_jobparams = &p_session->decode_job;
mm_jpeg_obj *my_obj = (mm_jpeg_obj *)p_session->jpeg_obj;
CDBG("%s:%d] E ", __func__, __LINE__);
MM_JPEG_CHK_ABORT(p_session, ret, error);
/* config main img */
ret = mm_jpegdec_session_config_main(p_session);
if (OMX_ErrorNone != ret) {
CDBG_ERROR("%s:%d] config main img failed", __func__, __LINE__);
goto error;
}
/* TODO: common config (if needed) */
ret = mm_jpeg_session_change_state(p_session, OMX_StateIdle,
mm_jpegdec_session_send_buffers);
if (ret) {
CDBG_ERROR("%s:%d] change state to idle failed %d",
__func__, __LINE__, ret);
goto error;
}
ret = mm_jpeg_session_change_state(p_session, OMX_StateExecuting,
NULL);
if (ret) {
CDBG_ERROR("%s:%d] change state to executing failed %d",
__func__, __LINE__, ret);
goto error;
}
error:
CDBG("%s:%d] X ret %d", __func__, __LINE__, ret);
return ret;
}
static OMX_ERRORTYPE mm_jpeg_session_port_enable(
mm_jpeg_job_session_t *p_session,
OMX_U32 nPortIndex,
OMX_BOOL wait)
{
OMX_ERRORTYPE ret = OMX_ErrorNone;
OMX_EVENTTYPE lEvent;
pthread_mutex_lock(&p_session->lock);
p_session->event_pending = OMX_TRUE;
pthread_mutex_unlock(&p_session->lock);
ret = OMX_SendCommand(p_session->omx_handle, OMX_CommandPortEnable,
nPortIndex, NULL);
if (ret) {
CDBG_ERROR("%s:%d] failed", __func__, __LINE__);
return ret;
}
if (wait == OMX_TRUE) {
// Wait for cmd complete
pthread_mutex_lock(&p_session->lock);
if (p_session->event_pending == OMX_TRUE) {
CDBG("%s:%d] before wait", __func__, __LINE__);
pthread_cond_wait(&p_session->cond, &p_session->lock);
lEvent = p_session->omxEvent;
CDBG("%s:%d] after wait", __func__, __LINE__);
}
lEvent = p_session->omxEvent;
pthread_mutex_unlock(&p_session->lock);
if (lEvent != OMX_EventCmdComplete) {
CDBG("%s:%d] Unexpected event %d", __func__, __LINE__,lEvent);
return OMX_ErrorUndefined;
}
}
return OMX_ErrorNone;
}
static OMX_ERRORTYPE mm_jpeg_session_port_disable(
mm_jpeg_job_session_t *p_session,
OMX_U32 nPortIndex,
OMX_BOOL wait)
{
OMX_ERRORTYPE ret = OMX_ErrorNone;
OMX_EVENTTYPE lEvent;
pthread_mutex_lock(&p_session->lock);
p_session->event_pending = OMX_TRUE;
pthread_mutex_unlock(&p_session->lock);
ret = OMX_SendCommand(p_session->omx_handle, OMX_CommandPortDisable,
nPortIndex, NULL);
if (ret) {
CDBG_ERROR("%s:%d] failed", __func__, __LINE__);
return ret;
}
if (wait == OMX_TRUE) {
// Wait for cmd complete
pthread_mutex_lock(&p_session->lock);
if (p_session->event_pending == OMX_TRUE) {
CDBG("%s:%d] before wait", __func__, __LINE__);
pthread_cond_wait(&p_session->cond, &p_session->lock);
CDBG("%s:%d] after wait", __func__, __LINE__);
}
lEvent = p_session->omxEvent;
pthread_mutex_unlock(&p_session->lock);
if (lEvent != OMX_EventCmdComplete) {
CDBG("%s:%d] Unexpected event %d", __func__, __LINE__,lEvent);
return OMX_ErrorUndefined;
}
}
return OMX_ErrorNone;
}
/** mm_jpegdec_session_decode:
*
* Arguments:
* @p_session: encode session
*
* Return:
* OMX_ERRORTYPE
*
* Description:
* Start the encoding
*
**/
static OMX_ERRORTYPE mm_jpegdec_session_decode(mm_jpeg_job_session_t *p_session)
{
OMX_ERRORTYPE ret = OMX_ErrorNone;
mm_jpeg_decode_params_t *p_params = &p_session->dec_params;
mm_jpeg_decode_job_t *p_jobparams = &p_session->decode_job;
int dest_idx = 0;
mm_jpeg_obj *my_obj = (mm_jpeg_obj *)p_session->jpeg_obj;
OMX_EVENTTYPE lEvent;
OMX_U32 i;
QOMX_BUFFER_INFO lbuffer_info;
pthread_mutex_lock(&p_session->lock);
p_session->abort_state = MM_JPEG_ABORT_NONE;
p_session->encoding = OMX_FALSE;
pthread_mutex_unlock(&p_session->lock);
if (OMX_FALSE == p_session->config) {
ret = mm_jpegdec_session_configure(p_session);
if (ret) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
goto error;
}
p_session->config = OMX_TRUE;
}
pthread_mutex_lock(&p_session->lock);
p_session->encoding = OMX_TRUE;
pthread_mutex_unlock(&p_session->lock);
MM_JPEG_CHK_ABORT(p_session, ret, error);
p_session->event_pending = OMX_TRUE;
ret = OMX_EmptyThisBuffer(p_session->omx_handle,
p_session->p_in_omx_buf[p_jobparams->src_index]);
if (ret) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
goto error;
}
// Wait for port settings changed
pthread_mutex_lock(&p_session->lock);
if (p_session->event_pending == OMX_TRUE) {
CDBG("%s:%d] before wait", __func__, __LINE__);
pthread_cond_wait(&p_session->cond, &p_session->lock);
}
lEvent = p_session->omxEvent;
CDBG("%s:%d] after wait", __func__, __LINE__);
pthread_mutex_unlock(&p_session->lock);
if (lEvent != OMX_EventPortSettingsChanged) {
CDBG("%s:%d] Unexpected event %d", __func__, __LINE__,lEvent);
goto error;
}
// Disable output port (wait)
mm_jpeg_session_port_disable(p_session,
p_session->outputPort.nPortIndex,
OMX_TRUE);
// Get port definition
ret = OMX_GetParameter(p_session->omx_handle, OMX_IndexParamPortDefinition,
&p_session->outputPort);
if (ret) {
CDBG_ERROR("%s:%d] failed", __func__, __LINE__);
return ret;
}
// Set port definition
p_session->outputPort.format.image.nFrameWidth =
p_jobparams->main_dim.dst_dim.width;
p_session->outputPort.format.image.nFrameHeight =
p_jobparams->main_dim.dst_dim.height;
p_session->outputPort.format.image.eColorFormat =
map_jpeg_format(p_params->color_format);
p_session->outputPort.nBufferSize =
p_params->dest_buf[p_jobparams->dst_index].buf_size;
p_session->outputPort.nBufferCountActual = p_params->num_dst_bufs;
p_session->outputPort.format.image.nSliceHeight =
p_params->dest_buf[p_jobparams->dst_index].offset.mp[0].scanline;
p_session->outputPort.format.image.nStride =
p_params->dest_buf[p_jobparams->dst_index].offset.mp[0].stride;
ret = OMX_SetParameter(p_session->omx_handle, OMX_IndexParamPortDefinition,
&p_session->outputPort);
if (ret) {
CDBG_ERROR("%s:%d] failed", __func__, __LINE__);
return ret;
}
// Enable port (no wait)
mm_jpeg_session_port_enable(p_session,
p_session->outputPort.nPortIndex,
OMX_FALSE);
memset(&lbuffer_info, 0x0, sizeof(QOMX_BUFFER_INFO));
// Use buffers
for (i = 0; i < p_params->num_dst_bufs; i++) {
lbuffer_info.fd = p_params->dest_buf[i].fd;
CDBG("%s:%d] Dest buffer %d", __func__, __LINE__, i);
ret = OMX_UseBuffer(p_session->omx_handle, &(p_session->p_out_omx_buf[i]),
1, &lbuffer_info, p_params->dest_buf[i].buf_size,
p_params->dest_buf[i].buf_vaddr);
if (ret) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
return ret;
}
}
// Wait for port enable completion
pthread_mutex_lock(&p_session->lock);
if (p_session->event_pending == OMX_TRUE) {
CDBG("%s:%d] before wait", __func__, __LINE__);
pthread_cond_wait(&p_session->cond, &p_session->lock);
lEvent = p_session->omxEvent;
CDBG("%s:%d] after wait", __func__, __LINE__);
}
lEvent = p_session->omxEvent;
pthread_mutex_unlock(&p_session->lock);
if (lEvent != OMX_EventCmdComplete) {
CDBG("%s:%d] Unexpected event %d", __func__, __LINE__,lEvent);
goto error;
}
ret = OMX_FillThisBuffer(p_session->omx_handle,
p_session->p_out_omx_buf[p_jobparams->dst_index]);
if (ret) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
goto error;
}
MM_JPEG_CHK_ABORT(p_session, ret, error);
error:
CDBG("%s:%d] X ", __func__, __LINE__);
return ret;
}
/** mm_jpegdec_process_decoding_job:
*
* Arguments:
* @my_obj: jpeg client
* @job_node: job node
*
* Return:
* 0 for success -1 otherwise
*
* Description:
* Start the encoding job
*
**/
int32_t mm_jpegdec_process_decoding_job(mm_jpeg_obj *my_obj, mm_jpeg_job_q_node_t* job_node)
{
int32_t rc = 0;
OMX_ERRORTYPE ret = OMX_ErrorNone;
mm_jpeg_job_session_t *p_session = NULL;
mm_jpeg_job_q_node_t *node = NULL;
/* check if valid session */
p_session = mm_jpeg_get_session(my_obj, job_node->dec_info.job_id);
if (NULL == p_session) {
CDBG_ERROR("%s:%d] invalid job id %x", __func__, __LINE__,
job_node->dec_info.job_id);
return -1;
}
/* sent encode cmd to OMX, queue job into ongoing queue */
rc = mm_jpeg_queue_enq(&my_obj->ongoing_job_q, job_node);
if (rc) {
CDBG_ERROR("%s:%d] jpeg enqueue failed %d",
__func__, __LINE__, ret);
goto error;
}
p_session->decode_job = job_node->dec_info.decode_job;
p_session->jobId = job_node->dec_info.job_id;
ret = mm_jpegdec_session_decode(p_session);
if (ret) {
CDBG_ERROR("%s:%d] encode session failed", __func__, __LINE__);
goto error;
}
CDBG("%s:%d] Success X ", __func__, __LINE__);
return rc;
error:
if ((OMX_ErrorNone != ret) &&
(NULL != p_session->dec_params.jpeg_cb)) {
p_session->job_status = JPEG_JOB_STATUS_ERROR;
CDBG("%s:%d] send jpeg error callback %d", __func__, __LINE__,
p_session->job_status);
p_session->dec_params.jpeg_cb(p_session->job_status,
p_session->client_hdl,
p_session->jobId,
NULL,
p_session->dec_params.userdata);
}
/*remove the job*/
mm_jpegdec_job_done(p_session);
CDBG("%s:%d] Error X ", __func__, __LINE__);
return rc;
}
/** mm_jpeg_start_decode_job:
*
* Arguments:
* @my_obj: jpeg object
* @client_hdl: client handle
* @job: pointer to encode job
* @jobId: job id
*
* Return:
* 0 for success else failure
*
* Description:
* Start the encoding job
*
**/
int32_t mm_jpegdec_start_decode_job(mm_jpeg_obj *my_obj,
mm_jpeg_job_t *job,
uint32_t *job_id)
{
int32_t rc = -1;
uint8_t session_idx = 0;
uint8_t client_idx = 0;
mm_jpeg_job_q_node_t* node = NULL;
mm_jpeg_job_session_t *p_session = NULL;
mm_jpeg_decode_job_t *p_jobparams = &job->decode_job;
*job_id = 0;
/* check if valid session */
session_idx = GET_SESSION_IDX(p_jobparams->session_id);
client_idx = GET_CLIENT_IDX(p_jobparams->session_id);
CDBG("%s:%d] session_idx %d client idx %d", __func__, __LINE__,
session_idx, client_idx);
if ((session_idx >= MM_JPEG_MAX_SESSION) ||
(client_idx >= MAX_JPEG_CLIENT_NUM)) {
CDBG_ERROR("%s:%d] invalid session id %x", __func__, __LINE__,
job->decode_job.session_id);
return rc;
}
p_session = &my_obj->clnt_mgr[client_idx].session[session_idx];
if (OMX_FALSE == p_session->active) {
CDBG_ERROR("%s:%d] session not active %x", __func__, __LINE__,
job->decode_job.session_id);
return rc;
}
if ((p_jobparams->src_index >= p_session->dec_params.num_src_bufs) ||
(p_jobparams->dst_index >= p_session->dec_params.num_dst_bufs)) {
CDBG_ERROR("%s:%d] invalid buffer indices", __func__, __LINE__);
return rc;
}
/* enqueue new job into todo job queue */
node = (mm_jpeg_job_q_node_t *)malloc(sizeof(mm_jpeg_job_q_node_t));
if (NULL == node) {
CDBG_ERROR("%s: No memory for mm_jpeg_job_q_node_t", __func__);
return -1;
}
*job_id = job->decode_job.session_id |
((p_session->job_hist++ % JOB_HIST_MAX) << 16);
memset(node, 0, sizeof(mm_jpeg_job_q_node_t));
node->dec_info.decode_job = job->decode_job;
node->dec_info.job_id = *job_id;
node->dec_info.client_handle = p_session->client_hdl;
node->type = MM_JPEG_CMD_TYPE_DECODE_JOB;
rc = mm_jpeg_queue_enq(&my_obj->job_mgr.job_queue, node);
if (0 == rc) {
cam_sem_post(&my_obj->job_mgr.job_sem);
}
return rc;
}
/** mm_jpegdec_create_session:
*
* Arguments:
* @my_obj: jpeg object
* @client_hdl: client handle
* @p_params: pointer to encode params
* @p_session_id: session id
*
* Return:
* 0 for success else failure
*
* Description:
* Start the encoding session
*
**/
int32_t mm_jpegdec_create_session(mm_jpeg_obj *my_obj,
uint32_t client_hdl,
mm_jpeg_decode_params_t *p_params,
uint32_t* p_session_id)
{
int32_t rc = 0;
OMX_ERRORTYPE ret = OMX_ErrorNone;
uint8_t clnt_idx = 0;
int session_idx = -1;
mm_jpeg_job_session_t *p_session = NULL;
*p_session_id = 0;
/* validate the parameters */
if ((p_params->num_src_bufs > MM_JPEG_MAX_BUF)
|| (p_params->num_dst_bufs > MM_JPEG_MAX_BUF)) {
CDBG_ERROR("%s:%d] invalid num buffers", __func__, __LINE__);
return rc;
}
/* check if valid client */
clnt_idx = mm_jpeg_util_get_index_by_handler(client_hdl);
if (clnt_idx >= MAX_JPEG_CLIENT_NUM) {
CDBG_ERROR("%s: invalid client with handler (%d)", __func__, client_hdl);
return rc;
}
session_idx = mm_jpeg_get_new_session_idx(my_obj, clnt_idx, &p_session);
if (session_idx < 0) {
CDBG_ERROR("%s:%d] invalid session id (%d)", __func__, __LINE__, session_idx);
return rc;
}
ret = mm_jpegdec_session_create(p_session);
if (OMX_ErrorNone != ret) {
p_session->active = OMX_FALSE;
CDBG_ERROR("%s:%d] jpeg session create failed", __func__, __LINE__);
return rc;
}
*p_session_id = (JOB_ID_MAGICVAL << 24) | (session_idx << 8) | clnt_idx;
/*copy the params*/
p_session->dec_params = *p_params;
p_session->client_hdl = client_hdl;
p_session->sessionId = *p_session_id;
p_session->jpeg_obj = (void*)my_obj; /* save a ptr to jpeg_obj */
CDBG("%s:%d] session id %x", __func__, __LINE__, *p_session_id);
return rc;
}
/** mm_jpegdec_destroy_session:
*
* Arguments:
* @my_obj: jpeg object
* @session_id: session index
*
* Return:
* 0 for success else failure
*
* Description:
* Destroy the encoding session
*
**/
int32_t mm_jpegdec_destroy_session(mm_jpeg_obj *my_obj,
mm_jpeg_job_session_t *p_session)
{
int32_t rc = 0;
uint8_t clnt_idx = 0;
mm_jpeg_job_q_node_t *node = NULL;
OMX_BOOL ret = OMX_FALSE;
if (NULL == p_session) {
CDBG_ERROR("%s:%d] invalid session", __func__, __LINE__);
return rc;
}
uint32_t session_id = p_session->sessionId;
pthread_mutex_lock(&my_obj->job_lock);
/* abort job if in todo queue */
CDBG("%s:%d] abort todo jobs", __func__, __LINE__);
node = mm_jpeg_queue_remove_job_by_session_id(&my_obj->job_mgr.job_queue, session_id);
while (NULL != node) {
free(node);
node = mm_jpeg_queue_remove_job_by_session_id(&my_obj->job_mgr.job_queue, session_id);
}
/* abort job if in ongoing queue */
CDBG("%s:%d] abort ongoing jobs", __func__, __LINE__);
node = mm_jpeg_queue_remove_job_by_session_id(&my_obj->ongoing_job_q, session_id);
while (NULL != node) {
free(node);
node = mm_jpeg_queue_remove_job_by_session_id(&my_obj->ongoing_job_q, session_id);
}
/* abort the current session */
mm_jpeg_session_abort(p_session);
mm_jpegdec_session_destroy(p_session);
mm_jpeg_remove_session_idx(my_obj, session_id);
pthread_mutex_unlock(&my_obj->job_lock);
/* wake up jobMgr thread to work on new job if there is any */
cam_sem_post(&my_obj->job_mgr.job_sem);
CDBG("%s:%d] X", __func__, __LINE__);
return rc;
}
/** mm_jpegdec_destroy_session_by_id:
*
* Arguments:
* @my_obj: jpeg object
* @session_id: session index
*
* Return:
* 0 for success else failure
*
* Description:
* Destroy the encoding session
*
**/
int32_t mm_jpegdec_destroy_session_by_id(mm_jpeg_obj *my_obj, uint32_t session_id)
{
int32_t rc = 0;
mm_jpeg_job_session_t *p_session = mm_jpeg_get_session(my_obj, session_id);
if (NULL == p_session) {
CDBG_ERROR("%s:%d] session is not valid", __func__, __LINE__);
return rc;
}
return mm_jpegdec_destroy_session(my_obj, p_session);
}
OMX_ERRORTYPE mm_jpegdec_ebd(OMX_HANDLETYPE hComponent,
OMX_PTR pAppData,
OMX_BUFFERHEADERTYPE *pBuffer)
{
OMX_ERRORTYPE ret = OMX_ErrorNone;
mm_jpeg_job_session_t *p_session = (mm_jpeg_job_session_t *) pAppData;
CDBG("%s:%d] count %d ", __func__, __LINE__, p_session->ebd_count);
pthread_mutex_lock(&p_session->lock);
p_session->ebd_count++;
pthread_mutex_unlock(&p_session->lock);
return 0;
}
OMX_ERRORTYPE mm_jpegdec_fbd(OMX_HANDLETYPE hComponent,
OMX_PTR pAppData,
OMX_BUFFERHEADERTYPE *pBuffer)
{
OMX_ERRORTYPE ret = OMX_ErrorNone;
mm_jpeg_job_session_t *p_session = (mm_jpeg_job_session_t *) pAppData;
uint32_t i = 0;
int rc = 0;
mm_jpeg_output_t output_buf;
CDBG("%s:%d] count %d ", __func__, __LINE__, p_session->fbd_count);
pthread_mutex_lock(&p_session->lock);
if (MM_JPEG_ABORT_NONE != p_session->abort_state) {
pthread_mutex_unlock(&p_session->lock);
return ret;
}
p_session->fbd_count++;
if (NULL != p_session->dec_params.jpeg_cb) {
p_session->job_status = JPEG_JOB_STATUS_DONE;
output_buf.buf_filled_len = (uint32_t)pBuffer->nFilledLen;
output_buf.buf_vaddr = pBuffer->pBuffer;
output_buf.fd = 0;
CDBG("%s:%d] send jpeg callback %d", __func__, __LINE__,
p_session->job_status);
p_session->dec_params.jpeg_cb(p_session->job_status,
p_session->client_hdl,
p_session->jobId,
&output_buf,
p_session->dec_params.userdata);
/* remove from ready queue */
mm_jpegdec_job_done(p_session);
}
pthread_mutex_unlock(&p_session->lock);
CDBG("%s:%d] ", __func__, __LINE__);
return ret;
}
OMX_ERRORTYPE mm_jpegdec_event_handler(OMX_HANDLETYPE hComponent,
OMX_PTR pAppData,
OMX_EVENTTYPE eEvent,
OMX_U32 nData1,
OMX_U32 nData2,
OMX_PTR pEventData)
{
mm_jpeg_job_session_t *p_session = (mm_jpeg_job_session_t *) pAppData;
CDBG("%s:%d] %d %d %d state %d", __func__, __LINE__, eEvent, (int)nData1,
(int)nData2, p_session->abort_state);
CDBG("%s:%d] AppData=%p ", __func__, __LINE__, pAppData);
pthread_mutex_lock(&p_session->lock);
p_session->omxEvent = eEvent;
if (MM_JPEG_ABORT_INIT == p_session->abort_state) {
p_session->abort_state = MM_JPEG_ABORT_DONE;
pthread_cond_signal(&p_session->cond);
pthread_mutex_unlock(&p_session->lock);
return OMX_ErrorNone;
}
if (eEvent == OMX_EventError) {
if (p_session->encoding == OMX_TRUE) {
CDBG("%s:%d] Error during encoding", __func__, __LINE__);
/* send jpeg callback */
if (NULL != p_session->dec_params.jpeg_cb) {
p_session->job_status = JPEG_JOB_STATUS_ERROR;
CDBG("%s:%d] send jpeg error callback %d", __func__, __LINE__,
p_session->job_status);
p_session->dec_params.jpeg_cb(p_session->job_status,
p_session->client_hdl,
p_session->jobId,
NULL,
p_session->dec_params.userdata);
}
/* remove from ready queue */
mm_jpegdec_job_done(p_session);
}
pthread_cond_signal(&p_session->cond);
} else if (eEvent == OMX_EventCmdComplete) {
p_session->state_change_pending = OMX_FALSE;
p_session->event_pending = OMX_FALSE;
pthread_cond_signal(&p_session->cond);
} else if (eEvent == OMX_EventPortSettingsChanged) {
p_session->event_pending = OMX_FALSE;
pthread_cond_signal(&p_session->cond);
}
pthread_mutex_unlock(&p_session->lock);
CDBG("%s:%d]", __func__, __LINE__);
return OMX_ErrorNone;
}
/** mm_jpegdec_abort_job:
*
* Arguments:
* @my_obj: jpeg object
* @client_hdl: client handle
* @jobId: job id
*
* Return:
* 0 for success else failure
*
* Description:
* Abort the encoding session
*
**/
int32_t mm_jpegdec_abort_job(mm_jpeg_obj *my_obj,
uint32_t jobId)
{
int32_t rc = -1;
uint8_t clnt_idx = 0;
mm_jpeg_job_q_node_t *node = NULL;
OMX_BOOL ret = OMX_FALSE;
mm_jpeg_job_session_t *p_session = NULL;
CDBG("%s:%d] ", __func__, __LINE__);
pthread_mutex_lock(&my_obj->job_lock);
/* abort job if in todo queue */
node = mm_jpeg_queue_remove_job_by_job_id(&my_obj->job_mgr.job_queue, jobId);
if (NULL != node) {
free(node);
goto abort_done;
}
/* abort job if in ongoing queue */
node = mm_jpeg_queue_remove_job_by_job_id(&my_obj->ongoing_job_q, jobId);
if (NULL != node) {
/* find job that is OMX ongoing, ask OMX to abort the job */
p_session = mm_jpeg_get_session(my_obj, node->dec_info.job_id);
if (p_session) {
mm_jpeg_session_abort(p_session);
} else {
CDBG_ERROR("%s:%d] Invalid job id 0x%x", __func__, __LINE__,
node->dec_info.job_id);
}
free(node);
goto abort_done;
}
abort_done:
pthread_mutex_unlock(&my_obj->job_lock);
return rc;
}
/** mm_jpegdec_init:
*
* Arguments:
* @my_obj: jpeg object
*
* Return:
* 0 for success else failure
*
* Description:
* Initializes the jpeg client
*
**/
int32_t mm_jpegdec_init(mm_jpeg_obj *my_obj)
{
int32_t rc = 0;
/* init locks */
pthread_mutex_init(&my_obj->job_lock, NULL);
/* init ongoing job queue */
rc = mm_jpeg_queue_init(&my_obj->ongoing_job_q);
if (0 != rc) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
return -1;
}
/* init job semaphore and launch jobmgr thread */
CDBG("%s:%d] Launch jobmgr thread rc %d", __func__, __LINE__, rc);
rc = mm_jpeg_jobmgr_thread_launch(my_obj);
if (0 != rc) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
return -1;
}
/* load OMX */
if (OMX_ErrorNone != OMX_Init()) {
/* roll back in error case */
CDBG_ERROR("%s:%d] OMX_Init failed (%d)", __func__, __LINE__, rc);
mm_jpeg_jobmgr_thread_release(my_obj);
mm_jpeg_queue_deinit(&my_obj->ongoing_job_q);
pthread_mutex_destroy(&my_obj->job_lock);
}
return rc;
}
/** mm_jpegdec_deinit:
*
* Arguments:
* @my_obj: jpeg object
*
* Return:
* 0 for success else failure
*
* Description:
* Deinits the jpeg client
*
**/
int32_t mm_jpegdec_deinit(mm_jpeg_obj *my_obj)
{
int32_t rc = 0;
/* release jobmgr thread */
rc = mm_jpeg_jobmgr_thread_release(my_obj);
if (0 != rc) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
}
/* unload OMX engine */
OMX_Deinit();
/* deinit ongoing job and cb queue */
rc = mm_jpeg_queue_deinit(&my_obj->ongoing_job_q);
if (0 != rc) {
CDBG_ERROR("%s:%d] Error", __func__, __LINE__);
}
/* destroy locks */
pthread_mutex_destroy(&my_obj->job_lock);
return rc;
}