Home | History | Annotate | Download | only in src
      1 /*--------------------------------------------------------------------------
      2 Copyright (c) 2012, Code Aurora Forum. All rights reserved.
      3 
      4 Redistribution and use in source and binary forms, with or without
      5 modification, are permitted provided that the following conditions are met:
      6     * Redistributions of source code must retain the above copyright
      7       notice, this list of conditions and the following disclaimer.
      8     * Redistributions in binary form must reproduce the above copyright
      9       notice, this list of conditions and the following disclaimer in the
     10       documentation and/or other materials provided with the distribution.
     11     * Neither the name of Code Aurora nor
     12       the names of its contributors may be used to endorse or promote
     13       products derived from this software without specific prior written
     14       permission.
     15 
     16 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
     17 AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     18 IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
     19 NON-INFRINGEMENT ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR
     20 CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
     21 EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
     22 PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
     23 OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
     24 WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
     25 OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
     26 ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     27 --------------------------------------------------------------------------*/
     28 
     29 /*============================================================================
     30                             O p e n M A X   w r a p p e r s
     31                              O p e n  M A X   C o r e
     32 
     33 *//** @file omx_vdec.cpp
     34   This module contains the implementation of the OpenMAX core & component.
     35 
     36 *//*========================================================================*/
     37 
     38 //////////////////////////////////////////////////////////////////////////////
     39 //                             Include Files
     40 //////////////////////////////////////////////////////////////////////////////
     41 
     42 #include <string.h>
     43 #include <pthread.h>
     44 #include <sys/prctl.h>
     45 #include <stdlib.h>
     46 #include <unistd.h>
     47 #include <errno.h>
     48 #include "omx_vdec.h"
     49 #include <fcntl.h>
     50 #include <limits.h>
     51 
     52 #ifndef _ANDROID_
     53 #include <sys/ioctl.h>
     54 #include <sys/mman.h>
     55 #endif //_ANDROID_
     56 
     57 #ifdef _ANDROID_
     58 #include <cutils/properties.h>
     59 #undef USE_EGL_IMAGE_GPU
     60 #endif
     61 
     62 #if  defined (_ANDROID_HONEYCOMB_) || defined (_ANDROID_ICS_)
     63 #include <gralloc_priv.h>
     64 #endif
     65 
     66 #ifdef _ANDROID_
     67 #include "DivXDrmDecrypt.h"
     68 #endif //_ANDROID_
     69 
     70 #ifdef USE_EGL_IMAGE_GPU
     71 #include <EGL/egl.h>
     72 #include <EGL/eglQCOM.h>
     73 #define EGL_BUFFER_HANDLE_QCOM 0x4F00
     74 #define EGL_BUFFER_OFFSET_QCOM 0x4F01
     75 #endif
     76 
     77 #ifdef INPUT_BUFFER_LOG
     78 #define INPUT_BUFFER_FILE_NAME "/data/input-bitstream.\0\0\0\0"
     79 #define INPUT_BUFFER_FILE_NAME_LEN 30
     80 FILE *inputBufferFile1;
     81 char inputfilename [INPUT_BUFFER_FILE_NAME_LEN] = "\0";
     82 #endif
     83 #ifdef OUTPUT_BUFFER_LOG
     84 FILE *outputBufferFile1;
     85 char outputfilename [] = "/data/output.yuv";
     86 #endif
     87 #ifdef OUTPUT_EXTRADATA_LOG
     88 FILE *outputExtradataFile;
     89 char ouputextradatafilename [] = "/data/extradata";
     90 #endif
     91 
     92 #define DEFAULT_FPS 30
     93 #define MAX_INPUT_ERROR DEFAULT_FPS
     94 #define MAX_SUPPORTED_FPS 120
     95 
     96 #define VC1_SP_MP_START_CODE        0xC5000000
     97 #define VC1_SP_MP_START_CODE_MASK   0xFF000000
     98 #define VC1_AP_SEQ_START_CODE       0x0F010000
     99 #define VC1_STRUCT_C_PROFILE_MASK   0xF0
    100 #define VC1_STRUCT_B_LEVEL_MASK     0xE0000000
    101 #define VC1_SIMPLE_PROFILE          0
    102 #define VC1_MAIN_PROFILE            1
    103 #define VC1_ADVANCE_PROFILE         3
    104 #define VC1_SIMPLE_PROFILE_LOW_LEVEL  0
    105 #define VC1_SIMPLE_PROFILE_MED_LEVEL  2
    106 #define VC1_STRUCT_C_LEN            4
    107 #define VC1_STRUCT_C_POS            8
    108 #define VC1_STRUCT_A_POS            12
    109 #define VC1_STRUCT_B_POS            24
    110 #define VC1_SEQ_LAYER_SIZE          36
    111 
    112 #define MEM_DEVICE "/dev/ion"
    113 #define MEM_HEAP_ID ION_CP_MM_HEAP_ID
    114 
    115 #ifdef _ANDROID_
    116     extern "C"{
    117         #include<utils/Log.h>
    118     }
    119 #endif//_ANDROID_
    120 
    121 #define Log2(number, power)  { OMX_U32 temp = number; power = 0; while( (0 == (temp & 0x1)) &&  power < 16) { temp >>=0x1; power++; } }
    122 #define Q16ToFraction(q,num,den) { OMX_U32 power; Log2(q,power);  num = q >> power; den = 0x1 << (16 - power); }
    123 
    124 void* async_message_thread (void *input)
    125 {
    126   struct vdec_ioctl_msg ioctl_msg;
    127   struct vdec_msginfo vdec_msg;
    128   OMX_BUFFERHEADERTYPE *buffer;
    129   struct v4l2_plane plane;
    130   struct pollfd pfd;
    131   struct v4l2_buffer v4l2_buf ={0};
    132    struct v4l2_event dqevent;
    133   pfd.events = POLLIN | POLLRDNORM | POLLOUT | POLLWRNORM | POLLRDBAND | POLLPRI;
    134   omx_vdec *omx = reinterpret_cast<omx_vdec*>(input);
    135   pfd.fd = omx->drv_ctx.video_driver_fd;
    136   int error_code = 0,rc=0;
    137   DEBUG_PRINT_HIGH("omx_vdec: Async thread start\n");
    138   prctl(PR_SET_NAME, (unsigned long)"VideoDecCallBackThread", 0, 0, 0);
    139   while (1)
    140   {
    141 		rc = poll(&pfd, 1, TIMEOUT);
    142 		if (!rc) {
    143 			printf("Poll timedout\n");
    144 			break;
    145 		} else if (rc < 0) {
    146 			printf("Error while polling: %d\n", rc);
    147 			break;
    148 		}
    149 		if ((pfd.revents & POLLIN) || (pfd.revents & POLLRDNORM)) {
    150 			v4l2_buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
    151 			v4l2_buf.memory = V4L2_MEMORY_USERPTR;
    152 			v4l2_buf.length = 1;
    153 			v4l2_buf.m.planes = &plane;
    154 			rc = ioctl(pfd.fd, VIDIOC_DQBUF, &v4l2_buf);
    155 			if (rc) {
    156 				/*TODO: How to handle this case */
    157 				printf("Failed to dequeue buf: %d from capture capability\n", rc);
    158 				break;
    159 			}
    160 			vdec_msg.msgcode=VDEC_MSG_RESP_OUTPUT_BUFFER_DONE;
    161 			vdec_msg.status_code=VDEC_S_SUCCESS;
    162 			vdec_msg.msgdata.output_frame.client_data=(void*)&v4l2_buf;
    163 			vdec_msg.msgdata.output_frame.len=plane.bytesused;
    164 			vdec_msg.msgdata.output_frame.bufferaddr=(void*)plane.m.userptr;
    165 		}
    166 		else if((pfd.revents & POLLOUT) || (pfd.revents & POLLWRNORM)) {
    167 			v4l2_buf.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
    168 			v4l2_buf.memory = V4L2_MEMORY_USERPTR;
    169 			v4l2_buf.m.planes = &plane;
    170 			rc = ioctl(pfd.fd, VIDIOC_DQBUF, &v4l2_buf);
    171 			if (rc) {
    172 				/*TODO: How to handle this case */
    173 				printf("Failed to dequeue buf: %d from output capability\n", rc);
    174 				break;
    175 			}
    176             vdec_msg.msgcode=VDEC_MSG_RESP_INPUT_BUFFER_DONE;
    177 			vdec_msg.status_code=VDEC_S_SUCCESS;
    178 			vdec_msg.msgdata.input_frame_clientdata=(void*)&v4l2_buf;
    179 		} else if (pfd.revents & POLLPRI){
    180 			rc = ioctl(pfd.fd, VIDIOC_DQEVENT, &dqevent);
    181 			if(dqevent.u.data[0] == MSM_VIDC_DECODER_EVENT_CHANGE){
    182 				vdec_msg.msgcode=VDEC_MSG_EVT_CONFIG_CHANGED;
    183 				vdec_msg.status_code=VDEC_S_SUCCESS;
    184 				printf("\n VIDC Port Reconfig recieved \n");
    185 			} else if (dqevent.u.data[0] == MSM_VIDC_DECODER_FLUSH_DONE){
    186 				vdec_msg.msgcode=VDEC_MSG_RESP_FLUSH_OUTPUT_DONE;
    187 				vdec_msg.status_code=VDEC_S_SUCCESS;
    188 				printf("\n VIDC Flush Done Recieved \n");
    189 			} else
    190 				printf("\n VIDC Some Event recieved \n");
    191 		} else if (pfd.revents & POLLERR){
    192 			printf("\n async_message_thread Exited \n");
    193 			break;
    194 		} else{
    195 			/*TODO: How to handle this case */
    196 			continue;
    197 		}
    198 
    199 		if (omx->async_message_process(input,&vdec_msg) < 0) {
    200 			printf("\n async_message_thread Exited  \n");
    201 				break;
    202 		}
    203   }
    204     DEBUG_PRINT_HIGH("omx_vdec: Async thread stop\n");
    205   return NULL;
    206 }
    207 
    208 void* message_thread(void *input)
    209 {
    210   omx_vdec* omx = reinterpret_cast<omx_vdec*>(input);
    211   unsigned char id;
    212   int n;
    213 
    214   DEBUG_PRINT_HIGH("omx_vdec: message thread start\n");
    215   prctl(PR_SET_NAME, (unsigned long)"VideoDecMsgThread", 0, 0, 0);
    216   while (1)
    217   {
    218 
    219     n = read(omx->m_pipe_in, &id, 1);
    220 
    221     if(0 == n)
    222     {
    223       break;
    224     }
    225 
    226     if (1 == n)
    227     {
    228         omx->process_event_cb(omx, id);
    229     }
    230     if ((n < 0) && (errno != EINTR))
    231     {
    232       DEBUG_PRINT_ERROR("\nERROR: read from pipe failed, ret %d errno %d", n, errno);
    233       break;
    234     }
    235   }
    236   DEBUG_PRINT_HIGH("omx_vdec: message thread stop\n");
    237   return 0;
    238 }
    239 
    240 void post_message(omx_vdec *omx, unsigned char id)
    241 {
    242       int ret_value;
    243       DEBUG_PRINT_LOW("omx_vdec: post_message %d pipe out%d\n", id,omx->m_pipe_out);
    244       ret_value = write(omx->m_pipe_out, &id, 1);
    245       DEBUG_PRINT_LOW("post_message to pipe done %d\n",ret_value);
    246 }
    247 
    248 // omx_cmd_queue destructor
    249 omx_vdec::omx_cmd_queue::~omx_cmd_queue()
    250 {
    251   // Nothing to do
    252 }
    253 
    254 // omx cmd queue constructor
    255 omx_vdec::omx_cmd_queue::omx_cmd_queue(): m_read(0),m_write(0),m_size(0)
    256 {
    257     memset(m_q,0,sizeof(omx_event)*OMX_CORE_CONTROL_CMDQ_SIZE);
    258 }
    259 
    260 // omx cmd queue insert
    261 bool omx_vdec::omx_cmd_queue::insert_entry(unsigned p1, unsigned p2, unsigned id)
    262 {
    263   bool ret = true;
    264   if(m_size < OMX_CORE_CONTROL_CMDQ_SIZE)
    265   {
    266     m_q[m_write].id       = id;
    267     m_q[m_write].param1   = p1;
    268     m_q[m_write].param2   = p2;
    269     m_write++;
    270     m_size ++;
    271     if(m_write >= OMX_CORE_CONTROL_CMDQ_SIZE)
    272     {
    273       m_write = 0;
    274     }
    275   }
    276   else
    277   {
    278     ret = false;
    279     DEBUG_PRINT_ERROR("ERROR: %s()::Command Queue Full\n", __func__);
    280   }
    281   return ret;
    282 }
    283 
    284 // omx cmd queue pop
    285 bool omx_vdec::omx_cmd_queue::pop_entry(unsigned *p1, unsigned *p2, unsigned *id)
    286 {
    287   bool ret = true;
    288   if (m_size > 0)
    289   {
    290     *id = m_q[m_read].id;
    291     *p1 = m_q[m_read].param1;
    292     *p2 = m_q[m_read].param2;
    293     // Move the read pointer ahead
    294     ++m_read;
    295     --m_size;
    296     if(m_read >= OMX_CORE_CONTROL_CMDQ_SIZE)
    297     {
    298       m_read = 0;
    299     }
    300   }
    301   else
    302   {
    303     ret = false;
    304   }
    305   return ret;
    306 }
    307 
    308 // Retrieve the first mesg type in the queue
    309 unsigned omx_vdec::omx_cmd_queue::get_q_msg_type()
    310 {
    311     return m_q[m_read].id;
    312 }
    313 
    314 #ifdef _ANDROID_
    315 omx_vdec::ts_arr_list::ts_arr_list()
    316 {
    317   //initialize timestamps array
    318   memset(m_ts_arr_list, 0, ( sizeof(ts_entry) * MAX_NUM_INPUT_OUTPUT_BUFFERS) );
    319 }
    320 omx_vdec::ts_arr_list::~ts_arr_list()
    321 {
    322   //free m_ts_arr_list?
    323 }
    324 
    325 bool omx_vdec::ts_arr_list::insert_ts(OMX_TICKS ts)
    326 {
    327   bool ret = true;
    328   bool duplicate_ts = false;
    329   int idx = 0;
    330 
    331   //insert at the first available empty location
    332   for ( ; idx < MAX_NUM_INPUT_OUTPUT_BUFFERS; idx++)
    333   {
    334     if (!m_ts_arr_list[idx].valid)
    335     {
    336       //found invalid or empty entry, save timestamp
    337       m_ts_arr_list[idx].valid = true;
    338       m_ts_arr_list[idx].timestamp = ts;
    339       DEBUG_PRINT_LOW("Insert_ts(): Inserting TIMESTAMP (%lld) at idx (%d)",
    340                        ts, idx);
    341       break;
    342     }
    343   }
    344 
    345   if (idx == MAX_NUM_INPUT_OUTPUT_BUFFERS)
    346   {
    347     DEBUG_PRINT_LOW("Timestamp array list is FULL. Unsuccessful insert");
    348     ret = false;
    349   }
    350   return ret;
    351 }
    352 
    353 bool omx_vdec::ts_arr_list::pop_min_ts(OMX_TICKS &ts)
    354 {
    355   bool ret = true;
    356   int min_idx = -1;
    357   OMX_TICKS min_ts = 0;
    358   int idx = 0;
    359 
    360   for ( ; idx < MAX_NUM_INPUT_OUTPUT_BUFFERS; idx++)
    361   {
    362 
    363     if (m_ts_arr_list[idx].valid)
    364     {
    365       //found valid entry, save index
    366       if (min_idx < 0)
    367       {
    368         //first valid entry
    369         min_ts = m_ts_arr_list[idx].timestamp;
    370         min_idx = idx;
    371       }
    372       else if (m_ts_arr_list[idx].timestamp < min_ts)
    373       {
    374         min_ts = m_ts_arr_list[idx].timestamp;
    375         min_idx = idx;
    376       }
    377     }
    378 
    379   }
    380 
    381   if (min_idx < 0)
    382   {
    383     //no valid entries found
    384     DEBUG_PRINT_LOW("Timestamp array list is empty. Unsuccessful pop");
    385     ts = 0;
    386     ret = false;
    387   }
    388   else
    389   {
    390     ts = m_ts_arr_list[min_idx].timestamp;
    391     m_ts_arr_list[min_idx].valid = false;
    392     DEBUG_PRINT_LOW("Pop_min_ts:Timestamp (%lld), index(%d)",
    393                      ts, min_idx);
    394   }
    395 
    396   return ret;
    397 
    398 }
    399 
    400 
    401 bool omx_vdec::ts_arr_list::reset_ts_list()
    402 {
    403   bool ret = true;
    404   int idx = 0;
    405 
    406   DEBUG_PRINT_LOW("reset_ts_list(): Resetting timestamp array list");
    407   for ( ; idx < MAX_NUM_INPUT_OUTPUT_BUFFERS; idx++)
    408   {
    409     m_ts_arr_list[idx].valid = false;
    410   }
    411   return ret;
    412 }
    413 #endif
    414 
    415 // factory function executed by the core to create instances
    416 void *get_omx_component_factory_fn(void)
    417 {
    418   return (new omx_vdec);
    419 }
    420 
    421 #ifdef _ANDROID_
    422 #ifdef USE_ION
    423 VideoHeap::VideoHeap(int devicefd, size_t size, void* base,
    424                      struct ion_handle *handle, int ionMapfd)
    425 {
    426     ionInit(devicefd, base, size, 0 , MEM_DEVICE,handle,ionMapfd);
    427 }
    428 #else
    429 VideoHeap::VideoHeap(int fd, size_t size, void* base)
    430 {
    431     // dup file descriptor, map once, use pmem
    432     init(dup(fd), base, size, 0 , MEM_DEVICE);
    433 }
    434 #endif
    435 #endif // _ANDROID_
    436 /* ======================================================================
    437 FUNCTION
    438   omx_vdec::omx_vdec
    439 
    440 DESCRIPTION
    441   Constructor
    442 
    443 PARAMETERS
    444   None
    445 
    446 RETURN VALUE
    447   None.
    448 ========================================================================== */
    449 omx_vdec::omx_vdec(): m_state(OMX_StateInvalid),
    450                       m_app_data(NULL),
    451                       m_inp_mem_ptr(NULL),
    452                       m_out_mem_ptr(NULL),
    453                       m_phdr_pmem_ptr(NULL),
    454                       pending_input_buffers(0),
    455                       pending_output_buffers(0),
    456                       m_out_bm_count(0),
    457                       m_inp_bm_count(0),
    458                       m_inp_bPopulated(OMX_FALSE),
    459                       m_out_bPopulated(OMX_FALSE),
    460                       m_flags(0),
    461                       m_inp_bEnabled(OMX_TRUE),
    462                       m_out_bEnabled(OMX_TRUE),
    463                       m_platform_list(NULL),
    464                       m_platform_entry(NULL),
    465                       m_pmem_info(NULL),
    466                       output_flush_progress (false),
    467                       input_flush_progress (false),
    468                       input_use_buffer (false),
    469                       output_use_buffer (false),
    470                       arbitrary_bytes (true),
    471                       psource_frame (NULL),
    472                       pdest_frame (NULL),
    473                       m_inp_heap_ptr (NULL),
    474                       m_heap_inp_bm_count (0),
    475                       codec_type_parse ((codec_type)0),
    476                       first_frame_meta (true),
    477                       frame_count (0),
    478                       nal_length(0),
    479                       nal_count (0),
    480                       look_ahead_nal (false),
    481                       first_frame(0),
    482                       first_buffer(NULL),
    483                       first_frame_size (0),
    484                       m_error_propogated(false),
    485                       m_device_file_ptr(NULL),
    486                       m_vc1_profile((vc1_profile_type)0),
    487                       prev_ts(LLONG_MAX),
    488                       rst_prev_ts(true),
    489                       frm_int(0),
    490                       m_in_alloc_cnt(0),
    491                       m_display_id(NULL),
    492                       ouput_egl_buffers(false),
    493                       h264_parser(NULL),
    494                       client_extradata(0),
    495                       h264_last_au_ts(LLONG_MAX),
    496                       h264_last_au_flags(0),
    497                       m_inp_err_count(0),
    498 #ifdef _ANDROID_
    499                       m_heap_ptr(NULL),
    500                       m_enable_android_native_buffers(OMX_FALSE),
    501                       m_use_android_native_buffers(OMX_FALSE),
    502 #endif
    503                       in_reconfig(false),
    504                       m_use_output_pmem(OMX_FALSE),
    505                       m_out_mem_region_smi(OMX_FALSE),
    506                       m_out_pvt_entry_pmem(OMX_FALSE),
    507                       secure_mode(false)
    508 #ifdef _ANDROID_
    509                     ,iDivXDrmDecrypt(NULL)
    510 #endif
    511                     ,m_desc_buffer_ptr(NULL)
    512                     ,streaming({false, false})
    513 {
    514   /* Assumption is that , to begin with , we have all the frames with decoder */
    515   DEBUG_PRINT_HIGH("In OMX vdec Constructor");
    516 #ifdef _ANDROID_
    517   char property_value[PROPERTY_VALUE_MAX] = {0};
    518   property_get("vidc.dec.debug.perf", property_value, "0");
    519   perf_flag = atoi(property_value);
    520   if (perf_flag)
    521   {
    522     DEBUG_PRINT_HIGH("vidc.dec.debug.perf is %d", perf_flag);
    523     dec_time.start();
    524     proc_frms = latency = 0;
    525   }
    526   property_value[0] = NULL;
    527   property_get("vidc.dec.debug.ts", property_value, "0");
    528   m_debug_timestamp = atoi(property_value);
    529   DEBUG_PRINT_HIGH("vidc.dec.debug.ts value is %d",m_debug_timestamp);
    530   if (m_debug_timestamp)
    531   {
    532     time_stamp_dts.set_timestamp_reorder_mode(true);
    533   }
    534 
    535   property_value[0] = NULL;
    536   property_get("vidc.dec.debug.concealedmb", property_value, "0");
    537   m_debug_concealedmb = atoi(property_value);
    538   DEBUG_PRINT_HIGH("vidc.dec.debug.concealedmb value is %d",m_debug_concealedmb);
    539 
    540 #endif
    541   memset(&m_cmp,0,sizeof(m_cmp));
    542   memset(&m_cb,0,sizeof(m_cb));
    543   memset (&drv_ctx,0,sizeof(drv_ctx));
    544   memset (&h264_scratch,0,sizeof (OMX_BUFFERHEADERTYPE));
    545   memset (m_hwdevice_name,0,sizeof(m_hwdevice_name));
    546   memset(&op_buf_rcnfg, 0 ,sizeof(vdec_allocatorproperty));
    547   memset(m_demux_offsets, 0, ( sizeof(OMX_U32) * 8192) );
    548   m_demux_entries = 0;
    549   drv_ctx.timestamp_adjust = false;
    550   drv_ctx.video_driver_fd = -1;
    551   m_vendor_config.pData = NULL;
    552   pthread_mutex_init(&m_lock, NULL);
    553   sem_init(&m_cmd_lock,0,0);
    554 #ifdef _ANDROID_
    555   char extradata_value[PROPERTY_VALUE_MAX] = {0};
    556   property_get("vidc.dec.debug.extradata", extradata_value, "0");
    557   m_debug_extradata = atoi(extradata_value);
    558   DEBUG_PRINT_HIGH("vidc.dec.debug.extradata value is %d",m_debug_extradata);
    559 #endif
    560 
    561 }
    562 
    563 
    564 /* ======================================================================
    565 FUNCTION
    566   omx_vdec::~omx_vdec
    567 
    568 DESCRIPTION
    569   Destructor
    570 
    571 PARAMETERS
    572   None
    573 
    574 RETURN VALUE
    575   None.
    576 ========================================================================== */
    577 omx_vdec::~omx_vdec()
    578 {
    579   m_pmem_info = NULL;
    580   DEBUG_PRINT_HIGH("In OMX vdec Destructor");
    581   if(m_pipe_in) close(m_pipe_in);
    582   if(m_pipe_out) close(m_pipe_out);
    583   m_pipe_in = -1;
    584   m_pipe_out = -1;
    585   DEBUG_PRINT_HIGH("Waiting on OMX Msg Thread exit");
    586   pthread_join(msg_thread_id,NULL);
    587   DEBUG_PRINT_HIGH("Waiting on OMX Async Thread exit");
    588   pthread_join(async_thread_id,NULL);
    589   close(drv_ctx.video_driver_fd);
    590   pthread_mutex_destroy(&m_lock);
    591   sem_destroy(&m_cmd_lock);
    592   if (perf_flag)
    593   {
    594     DEBUG_PRINT_HIGH("--> TOTAL PROCESSING TIME");
    595     dec_time.end();
    596   }
    597   DEBUG_PRINT_HIGH("Exit OMX vdec Destructor");
    598 }
    599 
    600 /* ======================================================================
    601 FUNCTION
    602   omx_vdec::OMXCntrlProcessMsgCb
    603 
    604 DESCRIPTION
    605   IL Client callbacks are generated through this routine. The decoder
    606   provides the thread context for this routine.
    607 
    608 PARAMETERS
    609   ctxt -- Context information related to the self.
    610   id   -- Event identifier. This could be any of the following:
    611           1. Command completion event
    612           2. Buffer done callback event
    613           3. Frame done callback event
    614 
    615 RETURN VALUE
    616   None.
    617 
    618 ========================================================================== */
    619 void omx_vdec::process_event_cb(void *ctxt, unsigned char id)
    620 {
    621   unsigned p1; // Parameter - 1
    622   unsigned p2; // Parameter - 2
    623   unsigned ident;
    624   unsigned qsize=0; // qsize
    625   omx_vdec *pThis = (omx_vdec *) ctxt;
    626 
    627   if(!pThis)
    628   {
    629     DEBUG_PRINT_ERROR("ERROR: %s()::Context is incorrect, bailing out\n",
    630         __func__);
    631     return;
    632   }
    633 
    634   // Protect the shared queue data structure
    635   do
    636   {
    637     /*Read the message id's from the queue*/
    638     pthread_mutex_lock(&pThis->m_lock);
    639     qsize = pThis->m_cmd_q.m_size;
    640     if(qsize)
    641     {
    642       pThis->m_cmd_q.pop_entry(&p1,&p2,&ident);
    643     }
    644 
    645     if (qsize == 0 && pThis->m_state != OMX_StatePause)
    646     {
    647       qsize = pThis->m_ftb_q.m_size;
    648       if (qsize)
    649       {
    650         pThis->m_ftb_q.pop_entry(&p1,&p2,&ident);
    651       }
    652     }
    653 
    654     if (qsize == 0 && pThis->m_state != OMX_StatePause)
    655     {
    656       qsize = pThis->m_etb_q.m_size;
    657       if (qsize)
    658       {
    659         pThis->m_etb_q.pop_entry(&p1,&p2,&ident);
    660       }
    661     }
    662     pthread_mutex_unlock(&pThis->m_lock);
    663 
    664     /*process message if we have one*/
    665     if(qsize > 0)
    666     {
    667       id = ident;
    668       switch (id)
    669       {
    670         case OMX_COMPONENT_GENERATE_EVENT:
    671           if (pThis->m_cb.EventHandler)
    672           {
    673             switch (p1)
    674             {
    675               case OMX_CommandStateSet:
    676                 pThis->m_state = (OMX_STATETYPE) p2;
    677                 DEBUG_PRINT_HIGH("\n OMX_CommandStateSet complete, m_state = %d",
    678                     pThis->m_state);
    679                 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
    680                                       OMX_EventCmdComplete, p1, p2, NULL);
    681                 break;
    682 
    683               case OMX_EventError:
    684                 if(p2 == OMX_StateInvalid)
    685                 {
    686                     DEBUG_PRINT_ERROR("\n OMX_EventError: p2 is OMX_StateInvalid");
    687                     pThis->m_state = (OMX_STATETYPE) p2;
    688                     pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
    689                                OMX_EventError, OMX_ErrorInvalidState, p2, NULL);
    690                 }
    691                 else if (p2 == OMX_ErrorHardware)
    692                 {
    693                    pThis->omx_report_error();
    694                 }
    695                 else
    696 		  {
    697                     pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
    698                                       OMX_EventError, p2, NULL, NULL );
    699                 }
    700                 break;
    701 
    702               case OMX_CommandPortDisable:
    703                 DEBUG_PRINT_HIGH("\n OMX_CommandPortDisable complete for port [%d]", p2);
    704                 if (BITMASK_PRESENT(&pThis->m_flags,
    705                     OMX_COMPONENT_OUTPUT_FLUSH_IN_DISABLE_PENDING))
    706                 {
    707                   BITMASK_SET(&pThis->m_flags, OMX_COMPONENT_DISABLE_OUTPUT_DEFERRED);
    708                   break;
    709                 }
    710                 if (p2 == OMX_CORE_OUTPUT_PORT_INDEX && pThis->in_reconfig)
    711                 {
    712                   pThis->op_buf_rcnfg.buffer_type = VDEC_BUFFER_TYPE_OUTPUT;
    713 				  OMX_ERRORTYPE eRet = OMX_ErrorNone;
    714 				  pThis->stream_off();
    715 				  OMX_ERRORTYPE eRet1 = pThis->get_buffer_req(&pThis->op_buf_rcnfg);
    716 				  pThis->in_reconfig = false;
    717                   if(eRet !=  OMX_ErrorNone)
    718                   {
    719                       DEBUG_PRINT_ERROR("set_buffer_req failed eRet = %d",eRet);
    720                       pThis->omx_report_error();
    721                       break;
    722                   }
    723                 }
    724                 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
    725                                       OMX_EventCmdComplete, p1, p2, NULL );
    726                 break;
    727               case OMX_CommandPortEnable:
    728                 DEBUG_PRINT_HIGH("\n OMX_CommandPortEnable complete for port [%d]", p2);
    729                 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,\
    730                                       OMX_EventCmdComplete, p1, p2, NULL );
    731                 break;
    732 
    733               default:
    734                 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
    735                                          OMX_EventCmdComplete, p1, p2, NULL );
    736                 break;
    737 
    738             }
    739           }
    740           else
    741           {
    742             DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL\n", __func__);
    743           }
    744           break;
    745         case OMX_COMPONENT_GENERATE_ETB_ARBITRARY:
    746           if (pThis->empty_this_buffer_proxy_arbitrary((OMX_HANDLETYPE)p1,\
    747               (OMX_BUFFERHEADERTYPE *)p2) != OMX_ErrorNone)
    748           {
    749             DEBUG_PRINT_ERROR("\n empty_this_buffer_proxy_arbitrary failure");
    750             pThis->omx_report_error ();
    751           }
    752       break;
    753         case OMX_COMPONENT_GENERATE_ETB:
    754           if (pThis->empty_this_buffer_proxy((OMX_HANDLETYPE)p1,\
    755               (OMX_BUFFERHEADERTYPE *)p2) != OMX_ErrorNone)
    756           {
    757             DEBUG_PRINT_ERROR("\n empty_this_buffer_proxy failure");
    758             pThis->omx_report_error ();
    759           }
    760          break;
    761 
    762         case OMX_COMPONENT_GENERATE_FTB:
    763           if ( pThis->fill_this_buffer_proxy((OMX_HANDLETYPE)p1,\
    764                (OMX_BUFFERHEADERTYPE *)p2) != OMX_ErrorNone)
    765           {
    766              DEBUG_PRINT_ERROR("\n fill_this_buffer_proxy failure");
    767              pThis->omx_report_error ();
    768           }
    769         break;
    770 
    771         case OMX_COMPONENT_GENERATE_COMMAND:
    772           pThis->send_command_proxy(&pThis->m_cmp,(OMX_COMMANDTYPE)p1,\
    773                                     (OMX_U32)p2,(OMX_PTR)NULL);
    774           break;
    775 
    776         case OMX_COMPONENT_GENERATE_EBD:
    777 
    778           if (p2 != VDEC_S_SUCCESS && p2 != VDEC_S_INPUT_BITSTREAM_ERR)
    779           {
    780             DEBUG_PRINT_ERROR("\n OMX_COMPONENT_GENERATE_EBD failure");
    781             pThis->omx_report_error ();
    782           }
    783           else
    784           {
    785             if (p2 == VDEC_S_INPUT_BITSTREAM_ERR && p1)
    786             {
    787               pThis->m_inp_err_count++;
    788               pThis->time_stamp_dts.remove_time_stamp(
    789               ((OMX_BUFFERHEADERTYPE *)p1)->nTimeStamp,
    790               (pThis->drv_ctx.interlace != VDEC_InterlaceFrameProgressive)
    791                 ?true:false);
    792             }
    793             else
    794             {
    795               pThis->m_inp_err_count = 0;
    796             }
    797             if ( pThis->empty_buffer_done(&pThis->m_cmp,
    798                  (OMX_BUFFERHEADERTYPE *)p1) != OMX_ErrorNone)
    799             {
    800                DEBUG_PRINT_ERROR("\n empty_buffer_done failure");
    801                pThis->omx_report_error ();
    802             }
    803             if(pThis->m_inp_err_count >= MAX_INPUT_ERROR)
    804             {
    805                DEBUG_PRINT_ERROR("\n Input bitstream error for consecutive %d frames.", MAX_INPUT_ERROR);
    806                pThis->omx_report_error ();
    807             }
    808           }
    809           break;
    810         case OMX_COMPONENT_GENERATE_INFO_FIELD_DROPPED:
    811           {
    812             int64_t *timestamp = (int64_t *)p1;
    813             if (p1)
    814             {
    815               pThis->time_stamp_dts.remove_time_stamp(*timestamp,
    816               (pThis->drv_ctx.interlace != VDEC_InterlaceFrameProgressive)
    817               ?true:false);
    818               free(timestamp);
    819             }
    820           }
    821           break;
    822         case OMX_COMPONENT_GENERATE_FBD:
    823           if (p2 != VDEC_S_SUCCESS)
    824           {
    825             DEBUG_PRINT_ERROR("\n OMX_COMPONENT_GENERATE_FBD failure");
    826             pThis->omx_report_error ();
    827           }
    828           else if ( pThis->fill_buffer_done(&pThis->m_cmp,
    829                   (OMX_BUFFERHEADERTYPE *)p1) != OMX_ErrorNone )
    830           {
    831             DEBUG_PRINT_ERROR("\n fill_buffer_done failure");
    832             pThis->omx_report_error ();
    833           }
    834           break;
    835 
    836         case OMX_COMPONENT_GENERATE_EVENT_INPUT_FLUSH:
    837           DEBUG_PRINT_HIGH("\n Driver flush i/p Port complete");
    838           if (!pThis->input_flush_progress)
    839           {
    840             DEBUG_PRINT_ERROR("\n WARNING: Unexpected flush from driver");
    841           }
    842           else
    843           {
    844             pThis->execute_input_flush();
    845             if (pThis->m_cb.EventHandler)
    846             {
    847               if (p2 != VDEC_S_SUCCESS)
    848               {
    849                 DEBUG_PRINT_ERROR("\nOMX_COMPONENT_GENERATE_EVENT_INPUT_FLUSH failure");
    850                 pThis->omx_report_error ();
    851               }
    852               else
    853               {
    854                 /*Check if we need generate event for Flush done*/
    855                 if(BITMASK_PRESENT(&pThis->m_flags,
    856                                    OMX_COMPONENT_INPUT_FLUSH_PENDING))
    857                 {
    858                   BITMASK_CLEAR (&pThis->m_flags,OMX_COMPONENT_INPUT_FLUSH_PENDING);
    859                   DEBUG_PRINT_LOW("\n Input Flush completed - Notify Client");
    860                   pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
    861                                            OMX_EventCmdComplete,OMX_CommandFlush,
    862                                            OMX_CORE_INPUT_PORT_INDEX,NULL );
    863                 }
    864                 if (BITMASK_PRESENT(&pThis->m_flags,
    865                                          OMX_COMPONENT_IDLE_PENDING))
    866                 {
    867                   if (!pThis->output_flush_progress)
    868                   {
    869                      DEBUG_PRINT_LOW("\n Output flush done hence issue stop");
    870                      if (/*ioctl (pThis->drv_ctx.video_driver_fd,
    871                                 VDEC_IOCTL_CMD_STOP,NULL ) < 0*/0)
    872                      {
    873                        DEBUG_PRINT_ERROR("\n VDEC_IOCTL_CMD_STOP failed");
    874                        pThis->omx_report_error ();
    875                      }
    876                   }
    877                 }
    878               }
    879             }
    880             else
    881             {
    882               DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
    883             }
    884           }
    885           break;
    886 
    887         case OMX_COMPONENT_GENERATE_EVENT_OUTPUT_FLUSH:
    888           DEBUG_PRINT_HIGH("\n Driver flush o/p Port complete");
    889           if (!pThis->output_flush_progress)
    890           {
    891             DEBUG_PRINT_ERROR("\n WARNING: Unexpected flush from driver");
    892           }
    893           else
    894           {
    895             pThis->execute_output_flush();
    896             if (pThis->m_cb.EventHandler)
    897             {
    898               if (p2 != VDEC_S_SUCCESS)
    899               {
    900                 DEBUG_PRINT_ERROR("\n OMX_COMPONENT_GENERATE_EVENT_OUTPUT_FLUSH failed");
    901                 pThis->omx_report_error ();
    902               }
    903               else
    904               {
    905                 /*Check if we need generate event for Flush done*/
    906                 if(BITMASK_PRESENT(&pThis->m_flags,
    907                                    OMX_COMPONENT_OUTPUT_FLUSH_PENDING))
    908                 {
    909                   DEBUG_PRINT_LOW("\n Notify Output Flush done");
    910                   BITMASK_CLEAR (&pThis->m_flags,OMX_COMPONENT_OUTPUT_FLUSH_PENDING);
    911                   pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
    912                                            OMX_EventCmdComplete,OMX_CommandFlush,
    913                                            OMX_CORE_OUTPUT_PORT_INDEX,NULL );
    914                 }
    915                 if(BITMASK_PRESENT(&pThis->m_flags,
    916                        OMX_COMPONENT_OUTPUT_FLUSH_IN_DISABLE_PENDING))
    917                 {
    918                   DEBUG_PRINT_LOW("\n Internal flush complete");
    919                   BITMASK_CLEAR (&pThis->m_flags,
    920                                  OMX_COMPONENT_OUTPUT_FLUSH_IN_DISABLE_PENDING);
    921                   if (BITMASK_PRESENT(&pThis->m_flags,
    922                           OMX_COMPONENT_DISABLE_OUTPUT_DEFERRED))
    923                   {
    924                     pThis->post_event(OMX_CommandPortDisable,
    925                                OMX_CORE_OUTPUT_PORT_INDEX,
    926                                OMX_COMPONENT_GENERATE_EVENT);
    927                     BITMASK_CLEAR (&pThis->m_flags,
    928                                    OMX_COMPONENT_DISABLE_OUTPUT_DEFERRED);
    929 
    930                   }
    931                 }
    932 
    933                 if (BITMASK_PRESENT(&pThis->m_flags ,OMX_COMPONENT_IDLE_PENDING))
    934                 {
    935                   if (!pThis->input_flush_progress)
    936                   {
    937                     DEBUG_PRINT_LOW("\n Input flush done hence issue stop");
    938                     if (/*ioctl (pThis->drv_ctx.video_driver_fd,
    939                                VDEC_IOCTL_CMD_STOP,NULL ) < */0)
    940                     {
    941                       DEBUG_PRINT_ERROR("\n VDEC_IOCTL_CMD_STOP failed");
    942                       pThis->omx_report_error ();
    943                     }
    944                   }
    945                 }
    946               }
    947             }
    948             else
    949             {
    950               DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
    951             }
    952           }
    953           break;
    954 
    955         case OMX_COMPONENT_GENERATE_START_DONE:
    956           DEBUG_PRINT_HIGH("\n Rxd OMX_COMPONENT_GENERATE_START_DONE");
    957 
    958           if (pThis->m_cb.EventHandler)
    959           {
    960             if (p2 != VDEC_S_SUCCESS)
    961             {
    962               DEBUG_PRINT_ERROR("\n OMX_COMPONENT_GENERATE_START_DONE Failure");
    963               pThis->omx_report_error ();
    964             }
    965             else
    966             {
    967               DEBUG_PRINT_LOW("\n OMX_COMPONENT_GENERATE_START_DONE Success");
    968               if(BITMASK_PRESENT(&pThis->m_flags,OMX_COMPONENT_EXECUTE_PENDING))
    969               {
    970                 DEBUG_PRINT_LOW("\n Move to executing");
    971                 // Send the callback now
    972                 BITMASK_CLEAR((&pThis->m_flags),OMX_COMPONENT_EXECUTE_PENDING);
    973                 pThis->m_state = OMX_StateExecuting;
    974                 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
    975                                        OMX_EventCmdComplete,OMX_CommandStateSet,
    976                                        OMX_StateExecuting, NULL);
    977               }
    978               else if (BITMASK_PRESENT(&pThis->m_flags,
    979                                        OMX_COMPONENT_PAUSE_PENDING))
    980               {
    981                 if (/*ioctl (pThis->drv_ctx.video_driver_fd,
    982                            VDEC_IOCTL_CMD_PAUSE,NULL ) < */0)
    983                 {
    984                   DEBUG_PRINT_ERROR("\n VDEC_IOCTL_CMD_PAUSE failed");
    985                   pThis->omx_report_error ();
    986                 }
    987               }
    988             }
    989           }
    990           else
    991           {
    992             DEBUG_PRINT_LOW("\n Event Handler callback is NULL");
    993           }
    994           break;
    995 
    996         case OMX_COMPONENT_GENERATE_PAUSE_DONE:
    997           DEBUG_PRINT_HIGH("\n Rxd OMX_COMPONENT_GENERATE_PAUSE_DONE");
    998           if (pThis->m_cb.EventHandler)
    999           {
   1000             if (p2 != VDEC_S_SUCCESS)
   1001             {
   1002               DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_PAUSE_DONE ret failed");
   1003               pThis->omx_report_error ();
   1004             }
   1005             else
   1006             {
   1007               pThis->complete_pending_buffer_done_cbs();
   1008               if(BITMASK_PRESENT(&pThis->m_flags,OMX_COMPONENT_PAUSE_PENDING))
   1009               {
   1010                 DEBUG_PRINT_LOW("\n OMX_COMPONENT_GENERATE_PAUSE_DONE nofity");
   1011                 //Send the callback now
   1012                 BITMASK_CLEAR((&pThis->m_flags),OMX_COMPONENT_PAUSE_PENDING);
   1013                 pThis->m_state = OMX_StatePause;
   1014                 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
   1015                                        OMX_EventCmdComplete,OMX_CommandStateSet,
   1016                                        OMX_StatePause, NULL);
   1017               }
   1018             }
   1019           }
   1020           else
   1021           {
   1022             DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
   1023           }
   1024 
   1025           break;
   1026 
   1027         case OMX_COMPONENT_GENERATE_RESUME_DONE:
   1028           DEBUG_PRINT_HIGH("\n Rxd OMX_COMPONENT_GENERATE_RESUME_DONE");
   1029           if (pThis->m_cb.EventHandler)
   1030           {
   1031             if (p2 != VDEC_S_SUCCESS)
   1032             {
   1033               DEBUG_PRINT_ERROR("\n OMX_COMPONENT_GENERATE_RESUME_DONE failed");
   1034               pThis->omx_report_error ();
   1035             }
   1036             else
   1037             {
   1038               if(BITMASK_PRESENT(&pThis->m_flags,OMX_COMPONENT_EXECUTE_PENDING))
   1039               {
   1040                 DEBUG_PRINT_LOW("\n Moving the decoder to execute state");
   1041                 // Send the callback now
   1042                 BITMASK_CLEAR((&pThis->m_flags),OMX_COMPONENT_EXECUTE_PENDING);
   1043                 pThis->m_state = OMX_StateExecuting;
   1044                 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
   1045                                        OMX_EventCmdComplete,OMX_CommandStateSet,
   1046                                        OMX_StateExecuting,NULL);
   1047               }
   1048             }
   1049           }
   1050           else
   1051           {
   1052             DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
   1053           }
   1054 
   1055           break;
   1056 
   1057         case OMX_COMPONENT_GENERATE_STOP_DONE:
   1058           DEBUG_PRINT_HIGH("\n Rxd OMX_COMPONENT_GENERATE_STOP_DONE");
   1059           if (pThis->m_cb.EventHandler)
   1060           {
   1061             if (p2 != VDEC_S_SUCCESS)
   1062             {
   1063               DEBUG_PRINT_ERROR("\n OMX_COMPONENT_GENERATE_STOP_DONE ret failed");
   1064               pThis->omx_report_error ();
   1065             }
   1066             else
   1067             {
   1068               pThis->complete_pending_buffer_done_cbs();
   1069               if(BITMASK_PRESENT(&pThis->m_flags,OMX_COMPONENT_IDLE_PENDING))
   1070               {
   1071                 DEBUG_PRINT_LOW("\n OMX_COMPONENT_GENERATE_STOP_DONE Success");
   1072                 // Send the callback now
   1073                 BITMASK_CLEAR((&pThis->m_flags),OMX_COMPONENT_IDLE_PENDING);
   1074                 pThis->m_state = OMX_StateIdle;
   1075                 DEBUG_PRINT_LOW("\n Move to Idle State");
   1076                 pThis->m_cb.EventHandler(&pThis->m_cmp,pThis->m_app_data,
   1077                                          OMX_EventCmdComplete,OMX_CommandStateSet,
   1078                                          OMX_StateIdle,NULL);
   1079               }
   1080             }
   1081           }
   1082           else
   1083           {
   1084             DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
   1085           }
   1086 
   1087           break;
   1088 
   1089         case OMX_COMPONENT_GENERATE_PORT_RECONFIG:
   1090           DEBUG_PRINT_HIGH("\n Rxd OMX_COMPONENT_GENERATE_PORT_RECONFIG");
   1091           if (pThis->start_port_reconfig() != OMX_ErrorNone)
   1092               pThis->omx_report_error();
   1093           else
   1094           {
   1095             if (pThis->in_reconfig)
   1096             {
   1097               if (pThis->m_cb.EventHandler) {
   1098                 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
   1099                     OMX_EventPortSettingsChanged, OMX_CORE_OUTPUT_PORT_INDEX, 0, NULL );
   1100               } else {
   1101                 DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
   1102               }
   1103             }
   1104             if (pThis->drv_ctx.interlace != VDEC_InterlaceFrameProgressive)
   1105             {
   1106               OMX_INTERLACETYPE format = (OMX_INTERLACETYPE)-1;
   1107               OMX_EVENTTYPE event = (OMX_EVENTTYPE)OMX_EventIndexsettingChanged;
   1108               if (pThis->drv_ctx.interlace == VDEC_InterlaceInterleaveFrameTopFieldFirst)
   1109                   format = OMX_InterlaceInterleaveFrameTopFieldFirst;
   1110               else if (pThis->drv_ctx.interlace == VDEC_InterlaceInterleaveFrameBottomFieldFirst)
   1111                   format = OMX_InterlaceInterleaveFrameBottomFieldFirst;
   1112               else //unsupported interlace format; raise a error
   1113                   event = OMX_EventError;
   1114               if (pThis->m_cb.EventHandler) {
   1115                 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
   1116                     event, format, 0, NULL );
   1117               } else {
   1118                 DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
   1119               }
   1120             }
   1121           }
   1122         break;
   1123 
   1124         case OMX_COMPONENT_GENERATE_EOS_DONE:
   1125           DEBUG_PRINT_HIGH("\n Rxd OMX_COMPONENT_GENERATE_EOS_DONE");
   1126           if (pThis->m_cb.EventHandler) {
   1127             pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data, OMX_EventBufferFlag,
   1128                             OMX_CORE_OUTPUT_PORT_INDEX, OMX_BUFFERFLAG_EOS, NULL );
   1129           } else {
   1130             DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
   1131           }
   1132           pThis->prev_ts = LLONG_MAX;
   1133           pThis->rst_prev_ts = true;
   1134           break;
   1135 
   1136         case OMX_COMPONENT_GENERATE_HARDWARE_ERROR:
   1137           DEBUG_PRINT_ERROR("\n OMX_COMPONENT_GENERATE_HARDWARE_ERROR");
   1138           pThis->omx_report_error ();
   1139           break;
   1140         case OMX_COMPONENT_GENERATE_INFO_PORT_RECONFIG:
   1141         {
   1142           DEBUG_PRINT_HIGH("\n Rxd OMX_COMPONENT_GENERATE_INFO_PORT_RECONFIG");
   1143           if (pThis->m_cb.EventHandler) {
   1144             pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
   1145                 (OMX_EVENTTYPE)OMX_EventIndexsettingChanged, OMX_CORE_OUTPUT_PORT_INDEX, 0, NULL );
   1146           } else {
   1147             DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
   1148           }
   1149         }
   1150         default:
   1151           break;
   1152         }
   1153       }
   1154     pthread_mutex_lock(&pThis->m_lock);
   1155     qsize = pThis->m_cmd_q.m_size;
   1156     if (pThis->m_state != OMX_StatePause)
   1157         qsize += (pThis->m_ftb_q.m_size + pThis->m_etb_q.m_size);
   1158     pthread_mutex_unlock(&pThis->m_lock);
   1159   }
   1160   while(qsize>0);
   1161 
   1162 }
   1163 
   1164 
   1165 
   1166 /* ======================================================================
   1167 FUNCTION
   1168   omx_vdec::ComponentInit
   1169 
   1170 DESCRIPTION
   1171   Initialize the component.
   1172 
   1173 PARAMETERS
   1174   ctxt -- Context information related to the self.
   1175   id   -- Event identifier. This could be any of the following:
   1176           1. Command completion event
   1177           2. Buffer done callback event
   1178           3. Frame done callback event
   1179 
   1180 RETURN VALUE
   1181   None.
   1182 
   1183 ========================================================================== */
   1184 OMX_ERRORTYPE omx_vdec::component_init(OMX_STRING role)
   1185 {
   1186 
   1187 	OMX_ERRORTYPE eRet = OMX_ErrorNone;
   1188 	struct vdec_ioctl_msg ioctl_msg = {NULL,NULL};
   1189 	struct v4l2_fmtdesc fdesc;
   1190 	struct v4l2_format fmt;
   1191 	struct v4l2_requestbuffers bufreq;
   1192 	unsigned int   alignment = 0,buffer_size = 0;
   1193 	int fds[2];
   1194 	int r,ret=0;
   1195 	bool codec_ambiguous = false;
   1196 	OMX_STRING device_name = "/dev/video32";
   1197 
   1198 	drv_ctx.video_driver_fd = open("/dev/video32", O_RDWR);
   1199 
   1200 	DEBUG_PRINT_HIGH("\n omx_vdec::component_init(): Open returned fd %d, errno %d",
   1201 			drv_ctx.video_driver_fd, errno);
   1202 
   1203 	if(drv_ctx.video_driver_fd == 0){
   1204 		drv_ctx.video_driver_fd = open(device_name, O_RDWR);
   1205 	}
   1206 
   1207 	if(drv_ctx.video_driver_fd < 0)
   1208 	{
   1209 		DEBUG_PRINT_ERROR("Omx_vdec::Comp Init Returning failure, errno %d\n", errno);
   1210 		return OMX_ErrorInsufficientResources;
   1211 	}
   1212 	drv_ctx.frame_rate.fps_numerator = DEFAULT_FPS;
   1213 	drv_ctx.frame_rate.fps_denominator = 1;
   1214 
   1215 
   1216 #ifdef INPUT_BUFFER_LOG
   1217 	strcpy(inputfilename, INPUT_BUFFER_FILE_NAME);
   1218 #endif
   1219 #ifdef OUTPUT_BUFFER_LOG
   1220 	outputBufferFile1 = fopen (outputfilename, "ab");
   1221 #endif
   1222 #ifdef OUTPUT_EXTRADATA_LOG
   1223 	outputExtradataFile = fopen (ouputextradatafilename, "ab");
   1224 #endif
   1225 
   1226 	// Copy the role information which provides the decoder kind
   1227 	strlcpy(drv_ctx.kind,role,128);
   1228 	if(!strncmp(drv_ctx.kind,"OMX.qcom.video.decoder.mpeg4",\
   1229 				OMX_MAX_STRINGNAME_SIZE))
   1230 	{
   1231 		strlcpy((char *)m_cRole, "video_decoder.mpeg4",\
   1232 				OMX_MAX_STRINGNAME_SIZE);
   1233 		drv_ctx.timestamp_adjust = true;
   1234 		drv_ctx.decoder_format = VDEC_CODECTYPE_MPEG4;
   1235 		eCompressionFormat = OMX_VIDEO_CodingMPEG4;
   1236 		/*Initialize Start Code for MPEG4*/
   1237 		codec_type_parse = CODEC_TYPE_MPEG4;
   1238 		m_frame_parser.init_start_codes (codec_type_parse);
   1239 #ifdef INPUT_BUFFER_LOG
   1240 		strcat(inputfilename, "m4v");
   1241 #endif
   1242 	}
   1243 	else if(!strncmp(drv_ctx.kind,"OMX.qcom.video.decoder.mpeg2",\
   1244 				OMX_MAX_STRINGNAME_SIZE))
   1245 	{
   1246 		strlcpy((char *)m_cRole, "video_decoder.mpeg2",\
   1247 				OMX_MAX_STRINGNAME_SIZE);
   1248 		drv_ctx.decoder_format = VDEC_CODECTYPE_MPEG2;
   1249 		eCompressionFormat = OMX_VIDEO_CodingMPEG2;
   1250 		/*Initialize Start Code for MPEG2*/
   1251 		codec_type_parse = CODEC_TYPE_MPEG2;
   1252 		m_frame_parser.init_start_codes (codec_type_parse);
   1253 #ifdef INPUT_BUFFER_LOG
   1254 		strcat(inputfilename, "mpg");
   1255 #endif
   1256 	}
   1257 	else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.h263",\
   1258 				OMX_MAX_STRINGNAME_SIZE))
   1259 	{
   1260 		strlcpy((char *)m_cRole, "video_decoder.h263",OMX_MAX_STRINGNAME_SIZE);
   1261 		DEBUG_PRINT_LOW("\n H263 Decoder selected");
   1262 		drv_ctx.decoder_format = VDEC_CODECTYPE_H263;
   1263 		eCompressionFormat = OMX_VIDEO_CodingH263;
   1264 		codec_type_parse = CODEC_TYPE_H263;
   1265 		m_frame_parser.init_start_codes (codec_type_parse);
   1266 #ifdef INPUT_BUFFER_LOG
   1267 		strcat(inputfilename, "263");
   1268 #endif
   1269 	}
   1270 	else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx311",\
   1271 				OMX_MAX_STRINGNAME_SIZE))
   1272 	{
   1273 		strlcpy((char *)m_cRole, "video_decoder.divx",OMX_MAX_STRINGNAME_SIZE);
   1274 		DEBUG_PRINT_LOW ("\n DIVX 311 Decoder selected");
   1275 		drv_ctx.decoder_format = VDEC_CODECTYPE_DIVX_3;
   1276 		output_capability = V4L2_PIX_FMT_DIVX_311;
   1277 		eCompressionFormat = (OMX_VIDEO_CODINGTYPE)QOMX_VIDEO_CodingDivx;
   1278 		codec_type_parse = CODEC_TYPE_DIVX;
   1279 		m_frame_parser.init_start_codes (codec_type_parse);
   1280 
   1281 	}
   1282 	else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx4",\
   1283 				OMX_MAX_STRINGNAME_SIZE))
   1284 	{
   1285 		strlcpy((char *)m_cRole, "video_decoder.divx",OMX_MAX_STRINGNAME_SIZE);
   1286 		DEBUG_PRINT_ERROR ("\n DIVX 4 Decoder selected");
   1287 		drv_ctx.decoder_format = VDEC_CODECTYPE_DIVX_4;
   1288 		output_capability = V4L2_PIX_FMT_DIVX;
   1289 		eCompressionFormat = (OMX_VIDEO_CODINGTYPE)QOMX_VIDEO_CodingDivx;
   1290 		codec_type_parse = CODEC_TYPE_DIVX;
   1291 		codec_ambiguous = true;
   1292 		m_frame_parser.init_start_codes (codec_type_parse);
   1293 
   1294 	}
   1295 	else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx",\
   1296 				OMX_MAX_STRINGNAME_SIZE))
   1297 	{
   1298 		strlcpy((char *)m_cRole, "video_decoder.divx",OMX_MAX_STRINGNAME_SIZE);
   1299 		DEBUG_PRINT_ERROR ("\n DIVX 5/6 Decoder selected");
   1300 		drv_ctx.decoder_format = VDEC_CODECTYPE_DIVX_6;
   1301 		output_capability = V4L2_PIX_FMT_DIVX;
   1302 		eCompressionFormat = (OMX_VIDEO_CODINGTYPE)QOMX_VIDEO_CodingDivx;
   1303 		codec_type_parse = CODEC_TYPE_DIVX;
   1304 		codec_ambiguous = true;
   1305 		m_frame_parser.init_start_codes (codec_type_parse);
   1306 
   1307 	}
   1308 	else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc",\
   1309 				OMX_MAX_STRINGNAME_SIZE))
   1310 	{
   1311 		strlcpy((char *)m_cRole, "video_decoder.avc",OMX_MAX_STRINGNAME_SIZE);
   1312 		drv_ctx.decoder_format = VDEC_CODECTYPE_H264;
   1313 		output_capability=V4L2_PIX_FMT_H264;
   1314 		eCompressionFormat = OMX_VIDEO_CodingAVC;
   1315 		codec_type_parse = CODEC_TYPE_H264;
   1316 		m_frame_parser.init_start_codes (codec_type_parse);
   1317 		m_frame_parser.init_nal_length(nal_length);
   1318 #ifdef INPUT_BUFFER_LOG
   1319 		strcat(inputfilename, "264");
   1320 #endif
   1321 	}
   1322 	else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vc1",\
   1323 				OMX_MAX_STRINGNAME_SIZE))
   1324 	{
   1325 		strlcpy((char *)m_cRole, "video_decoder.vc1",OMX_MAX_STRINGNAME_SIZE);
   1326 		drv_ctx.decoder_format = VDEC_CODECTYPE_VC1;
   1327 		eCompressionFormat = OMX_VIDEO_CodingWMV;
   1328 		codec_type_parse = CODEC_TYPE_VC1;
   1329 		m_frame_parser.init_start_codes (codec_type_parse);
   1330 #ifdef INPUT_BUFFER_LOG
   1331 		strcat(inputfilename, "vc1");
   1332 #endif
   1333 	}
   1334 	else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.wmv",\
   1335 				OMX_MAX_STRINGNAME_SIZE))
   1336 	{
   1337 		strlcpy((char *)m_cRole, "video_decoder.vc1",OMX_MAX_STRINGNAME_SIZE);
   1338 		drv_ctx.decoder_format = VDEC_CODECTYPE_VC1_RCV;
   1339 		eCompressionFormat = OMX_VIDEO_CodingWMV;
   1340 		codec_type_parse = CODEC_TYPE_VC1;
   1341 		m_frame_parser.init_start_codes (codec_type_parse);
   1342 #ifdef INPUT_BUFFER_LOG
   1343 		strcat(inputfilename, "vc1");
   1344 #endif
   1345 	}
   1346 	else
   1347 	{
   1348 		DEBUG_PRINT_ERROR("\nERROR:Unknown Component\n");
   1349 		eRet = OMX_ErrorInvalidComponentName;
   1350 	}
   1351 #ifdef INPUT_BUFFER_LOG
   1352 	inputBufferFile1 = fopen (inputfilename, "ab");
   1353 #endif
   1354 	if (eRet == OMX_ErrorNone)
   1355 	{
   1356 
   1357 		drv_ctx.output_format = VDEC_YUV_FORMAT_TILE_4x2;
   1358 		capture_capability= V4L2_PIX_FMT_NV12;
   1359 
   1360 		struct v4l2_event_subscription sub;
   1361 		sub.type=V4L2_EVENT_ALL;
   1362 		ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_SUBSCRIBE_EVENT, &sub);
   1363 		if (ret) {
   1364 		printf("\n Subscribe Event Failed \n");
   1365 		return OMX_ErrorInsufficientResources;
   1366 		}
   1367 
   1368 		struct v4l2_capability cap;
   1369 		ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_QUERYCAP, &cap);
   1370 		if (ret) {
   1371 		  printf("Failed to query capabilities\n");
   1372 		  /*TODO: How to handle this case */
   1373 		} else {
   1374 		  printf("Capabilities: driver_name = %s, card = %s, bus_info = %s,"
   1375 				" version = %d, capabilities = %x\n", cap.driver, cap.card,
   1376 				cap.bus_info, cap.version, cap.capabilities);
   1377 		}
   1378 		ret=0;
   1379 		fdesc.type=V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   1380 		fdesc.index=0;
   1381 		while (ioctl(drv_ctx.video_driver_fd, VIDIOC_ENUM_FMT, &fdesc) == 0) {
   1382 			printf("fmt: description: %s, fmt: %x, flags = %x\n", fdesc.description,
   1383 					fdesc.pixelformat, fdesc.flags);
   1384 			fdesc.index++;
   1385 		}
   1386 		fdesc.type=V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
   1387 		fdesc.index=0;
   1388 		while (ioctl(drv_ctx.video_driver_fd, VIDIOC_ENUM_FMT, &fdesc) == 0) {
   1389 
   1390 			printf("fmt: description: %s, fmt: %x, flags = %x\n", fdesc.description,
   1391 					fdesc.pixelformat, fdesc.flags);
   1392 			fdesc.index++;
   1393 		}
   1394 
   1395 		drv_ctx.video_resolution.frame_height=drv_ctx.video_resolution.scan_lines=240;
   1396 		drv_ctx.video_resolution.frame_width=drv_ctx.video_resolution.stride=320;
   1397 		fmt.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
   1398 		fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
   1399 		fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
   1400 		fmt.fmt.pix_mp.pixelformat = output_capability;
   1401 		ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
   1402 		if (ret) {
   1403 			/*TODO: How to handle this case */
   1404 			printf("Failed to set format on capture port\n");
   1405 				}
   1406 		printf("\n Set Format was successful \n ");
   1407 		if (codec_ambiguous) {
   1408 			if (output_capability == V4L2_PIX_FMT_DIVX) {
   1409 				struct v4l2_control divx_ctrl;
   1410 
   1411 				if (drv_ctx.decoder_format == VDEC_CODECTYPE_DIVX_4) {
   1412 					divx_ctrl.id = V4L2_MPEG_VIDC_VIDEO_DIVX_FORMAT_4;
   1413 				} else if (drv_ctx.decoder_format == VDEC_CODECTYPE_DIVX_5) {
   1414 					divx_ctrl.id = V4L2_MPEG_VIDC_VIDEO_DIVX_FORMAT_5;
   1415 				} else {
   1416 					divx_ctrl.id = V4L2_MPEG_VIDC_VIDEO_DIVX_FORMAT_5;
   1417 				}
   1418 
   1419 				divx_ctrl.value = V4L2_CID_MPEG_VIDC_VIDEO_DIVX_FORMAT;
   1420 				ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &fmt);
   1421 				if (ret) {
   1422 					DEBUG_PRINT_ERROR("Failed to set divx version\n");
   1423 				}
   1424 			} else {
   1425 				DEBUG_PRINT_ERROR("Codec should not be ambiguous");
   1426 			}
   1427 		}
   1428 
   1429 		fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   1430 		fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
   1431 		fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
   1432 		fmt.fmt.pix_mp.pixelformat = output_capability;
   1433 		ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
   1434 		if (ret) {
   1435 			/*TODO: How to handle this case */
   1436 			printf("Failed to set format on capture port\n");
   1437 				}
   1438 		printf("\n Set Format was successful \n ");
   1439 
   1440 		/*Get the Buffer requirements for input and output ports*/
   1441 		drv_ctx.ip_buf.buffer_type = VDEC_BUFFER_TYPE_INPUT;
   1442 		drv_ctx.op_buf.buffer_type = VDEC_BUFFER_TYPE_OUTPUT;
   1443 		drv_ctx.op_buf.alignment=4096;
   1444 		drv_ctx.ip_buf.alignment=4096;
   1445 		drv_ctx.interlace = VDEC_InterlaceFrameProgressive;
   1446 		drv_ctx.extradata = 0;
   1447 		drv_ctx.picture_order = VDEC_ORDER_DECODE;
   1448 		drv_ctx.idr_only_decoding = 0;
   1449 
   1450 		m_state = OMX_StateLoaded;
   1451 		eRet=get_buffer_req(&drv_ctx.ip_buf);
   1452 		printf("Input Buffer Size =%d \n ",drv_ctx.ip_buf.buffer_size);
   1453 
   1454 #ifdef DEFAULT_EXTRADATA
   1455 		if (eRet == OMX_ErrorNone && !secure_mode)
   1456 			eRet = enable_extradata(DEFAULT_EXTRADATA);
   1457 #endif
   1458 		if (drv_ctx.decoder_format == VDEC_CODECTYPE_H264)
   1459 		{
   1460 			if (m_frame_parser.mutils == NULL)
   1461 			{
   1462 				m_frame_parser.mutils = new H264_Utils();
   1463 
   1464 				if (m_frame_parser.mutils == NULL)
   1465 				{
   1466 					DEBUG_PRINT_ERROR("\n parser utils Allocation failed ");
   1467 					eRet = OMX_ErrorInsufficientResources;
   1468 				}
   1469 				else
   1470 				{
   1471 					h264_scratch.nAllocLen = drv_ctx.ip_buf.buffer_size;
   1472 					h264_scratch.pBuffer = (OMX_U8 *)malloc (drv_ctx.ip_buf.buffer_size);
   1473 					h264_scratch.nFilledLen = 0;
   1474 					h264_scratch.nOffset = 0;
   1475 
   1476 					if (h264_scratch.pBuffer == NULL)
   1477 					{
   1478 						DEBUG_PRINT_ERROR("\n h264_scratch.pBuffer Allocation failed ");
   1479 						return OMX_ErrorInsufficientResources;
   1480 					}
   1481 					m_frame_parser.mutils->initialize_frame_checking_environment();
   1482 					m_frame_parser.mutils->allocate_rbsp_buffer (drv_ctx.ip_buf.buffer_size);
   1483 				}
   1484 			}
   1485 
   1486 			h264_parser = new h264_stream_parser();
   1487 			if (!h264_parser)
   1488 			{
   1489 				DEBUG_PRINT_ERROR("ERROR: H264 parser allocation failed!");
   1490 				eRet = OMX_ErrorInsufficientResources;
   1491 			}
   1492 		}
   1493 
   1494 		if(pipe(fds))
   1495 		{
   1496 			DEBUG_PRINT_ERROR("pipe creation failed\n");
   1497 			eRet = OMX_ErrorInsufficientResources;
   1498 		}
   1499 		else
   1500 		{
   1501 			int temp1[2];
   1502 			if(fds[0] == 0 || fds[1] == 0)
   1503 			{
   1504 				if (pipe (temp1))
   1505 				{
   1506 					DEBUG_PRINT_ERROR("pipe creation failed\n");
   1507 					return OMX_ErrorInsufficientResources;
   1508 				}
   1509 				//close (fds[0]);
   1510 				//close (fds[1]);
   1511 				fds[0] = temp1 [0];
   1512 				fds[1] = temp1 [1];
   1513 			}
   1514 			m_pipe_in = fds[0];
   1515 			m_pipe_out = fds[1];
   1516 			r = pthread_create(&msg_thread_id,0,message_thread,this);
   1517 
   1518 			if(r < 0)
   1519 			{
   1520 				DEBUG_PRINT_ERROR("\n component_init(): message_thread creation failed");
   1521 				eRet = OMX_ErrorInsufficientResources;
   1522 			}
   1523 		}
   1524 	}
   1525 
   1526 	if (eRet != OMX_ErrorNone)
   1527 	{
   1528 		DEBUG_PRINT_ERROR("\n Component Init Failed");
   1529 		DEBUG_PRINT_HIGH("\n Calling VDEC_IOCTL_STOP_NEXT_MSG");
   1530 		(void)ioctl(drv_ctx.video_driver_fd, VDEC_IOCTL_STOP_NEXT_MSG,
   1531 				NULL);
   1532 		DEBUG_PRINT_HIGH("\n Calling close() on Video Driver");
   1533 		close (drv_ctx.video_driver_fd);
   1534 		drv_ctx.video_driver_fd = -1;
   1535 	}
   1536 	else
   1537 	{
   1538 		DEBUG_PRINT_HIGH("\n omx_vdec::component_init() success");
   1539 	}
   1540 
   1541 	//memset(&h264_mv_buff,0,sizeof(struct h264_mv_buffer));
   1542 	return eRet;
   1543 }
   1544 
   1545 /* ======================================================================
   1546 FUNCTION
   1547   omx_vdec::GetComponentVersion
   1548 
   1549 DESCRIPTION
   1550   Returns the component version.
   1551 
   1552 PARAMETERS
   1553   TBD.
   1554 
   1555 RETURN VALUE
   1556   OMX_ErrorNone.
   1557 
   1558 ========================================================================== */
   1559 OMX_ERRORTYPE  omx_vdec::get_component_version
   1560                                      (
   1561                                       OMX_IN OMX_HANDLETYPE hComp,
   1562                                       OMX_OUT OMX_STRING componentName,
   1563                                       OMX_OUT OMX_VERSIONTYPE* componentVersion,
   1564                                       OMX_OUT OMX_VERSIONTYPE* specVersion,
   1565                                       OMX_OUT OMX_UUIDTYPE* componentUUID
   1566                                       )
   1567 {
   1568     if(m_state == OMX_StateInvalid)
   1569     {
   1570         DEBUG_PRINT_ERROR("Get Comp Version in Invalid State\n");
   1571         return OMX_ErrorInvalidState;
   1572     }
   1573   /* TBD -- Return the proper version */
   1574   if (specVersion)
   1575   {
   1576     specVersion->nVersion = OMX_SPEC_VERSION;
   1577   }
   1578   return OMX_ErrorNone;
   1579 }
   1580 /* ======================================================================
   1581 FUNCTION
   1582   omx_vdec::SendCommand
   1583 
   1584 DESCRIPTION
   1585   Returns zero if all the buffers released..
   1586 
   1587 PARAMETERS
   1588   None.
   1589 
   1590 RETURN VALUE
   1591   true/false
   1592 
   1593 ========================================================================== */
   1594 OMX_ERRORTYPE  omx_vdec::send_command(OMX_IN OMX_HANDLETYPE hComp,
   1595                                       OMX_IN OMX_COMMANDTYPE cmd,
   1596                                       OMX_IN OMX_U32 param1,
   1597                                       OMX_IN OMX_PTR cmdData
   1598                                       )
   1599 {
   1600     DEBUG_PRINT_LOW("\n send_command: Recieved a Command from Client");
   1601     if(m_state == OMX_StateInvalid)
   1602     {
   1603         DEBUG_PRINT_ERROR("ERROR: Send Command in Invalid State\n");
   1604         return OMX_ErrorInvalidState;
   1605     }
   1606     if (cmd == OMX_CommandFlush && param1 != OMX_CORE_INPUT_PORT_INDEX
   1607       && param1 != OMX_CORE_OUTPUT_PORT_INDEX && param1 != OMX_ALL)
   1608     {
   1609       DEBUG_PRINT_ERROR("\n send_command(): ERROR OMX_CommandFlush "
   1610         "to invalid port: %d", param1);
   1611       return OMX_ErrorBadPortIndex;
   1612     }
   1613     post_event((unsigned)cmd,(unsigned)param1,OMX_COMPONENT_GENERATE_COMMAND);
   1614     sem_wait(&m_cmd_lock);
   1615     DEBUG_PRINT_LOW("\n send_command: Command Processed\n");
   1616     return OMX_ErrorNone;
   1617 }
   1618 
   1619 /* ======================================================================
   1620 FUNCTION
   1621   omx_vdec::SendCommand
   1622 
   1623 DESCRIPTION
   1624   Returns zero if all the buffers released..
   1625 
   1626 PARAMETERS
   1627   None.
   1628 
   1629 RETURN VALUE
   1630   true/false
   1631 
   1632 ========================================================================== */
   1633 OMX_ERRORTYPE  omx_vdec::send_command_proxy(OMX_IN OMX_HANDLETYPE hComp,
   1634                                             OMX_IN OMX_COMMANDTYPE cmd,
   1635                                             OMX_IN OMX_U32 param1,
   1636                                             OMX_IN OMX_PTR cmdData
   1637                                             )
   1638 {
   1639   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   1640   OMX_STATETYPE eState = (OMX_STATETYPE) param1;
   1641   int bFlag = 1,sem_posted = 0,ret=0;
   1642 
   1643   DEBUG_PRINT_LOW("\n send_command_proxy(): cmd = %d", cmd);
   1644   DEBUG_PRINT_HIGH("\n send_command_proxy(): Current State %d, Expected State %d",
   1645     m_state, eState);
   1646 
   1647   if(cmd == OMX_CommandStateSet)
   1648   {
   1649     DEBUG_PRINT_HIGH("\n send_command_proxy(): OMX_CommandStateSet issued");
   1650     DEBUG_PRINT_HIGH("\n Current State %d, Expected State %d", m_state, eState);
   1651     /***************************/
   1652     /* Current State is Loaded */
   1653     /***************************/
   1654     if(m_state == OMX_StateLoaded)
   1655     {
   1656       if(eState == OMX_StateIdle)
   1657       {
   1658         //if all buffers are allocated or all ports disabled
   1659         if(allocate_done() ||
   1660           (m_inp_bEnabled == OMX_FALSE && m_out_bEnabled == OMX_FALSE))
   1661         {
   1662           DEBUG_PRINT_LOW("send_command_proxy(): Loaded-->Idle\n");
   1663         }
   1664         else
   1665         {
   1666           DEBUG_PRINT_LOW("send_command_proxy(): Loaded-->Idle-Pending\n");
   1667           BITMASK_SET(&m_flags, OMX_COMPONENT_IDLE_PENDING);
   1668           // Skip the event notification
   1669           bFlag = 0;
   1670         }
   1671       }
   1672       /* Requesting transition from Loaded to Loaded */
   1673       else if(eState == OMX_StateLoaded)
   1674       {
   1675         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Loaded-->Loaded\n");
   1676         post_event(OMX_EventError,OMX_ErrorSameState,\
   1677                    OMX_COMPONENT_GENERATE_EVENT);
   1678         eRet = OMX_ErrorSameState;
   1679       }
   1680       /* Requesting transition from Loaded to WaitForResources */
   1681       else if(eState == OMX_StateWaitForResources)
   1682       {
   1683         /* Since error is None , we will post an event
   1684            at the end of this function definition */
   1685         DEBUG_PRINT_LOW("send_command_proxy(): Loaded-->WaitForResources\n");
   1686       }
   1687       /* Requesting transition from Loaded to Executing */
   1688       else if(eState == OMX_StateExecuting)
   1689       {
   1690         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Loaded-->Executing\n");
   1691         post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
   1692                    OMX_COMPONENT_GENERATE_EVENT);
   1693         eRet = OMX_ErrorIncorrectStateTransition;
   1694       }
   1695       /* Requesting transition from Loaded to Pause */
   1696       else if(eState == OMX_StatePause)
   1697       {
   1698         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Loaded-->Pause\n");
   1699         post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
   1700                    OMX_COMPONENT_GENERATE_EVENT);
   1701         eRet = OMX_ErrorIncorrectStateTransition;
   1702       }
   1703       /* Requesting transition from Loaded to Invalid */
   1704       else if(eState == OMX_StateInvalid)
   1705       {
   1706         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Loaded-->Invalid\n");
   1707         post_event(OMX_EventError,eState,OMX_COMPONENT_GENERATE_EVENT);
   1708         eRet = OMX_ErrorInvalidState;
   1709       }
   1710       else
   1711       {
   1712         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Loaded-->Invalid(%d Not Handled)\n",\
   1713                           eState);
   1714         eRet = OMX_ErrorBadParameter;
   1715       }
   1716     }
   1717 
   1718     /***************************/
   1719     /* Current State is IDLE */
   1720     /***************************/
   1721     else if(m_state == OMX_StateIdle)
   1722     {
   1723       if(eState == OMX_StateLoaded)
   1724       {
   1725         if(release_done())
   1726         {
   1727           /*
   1728              Since error is None , we will post an event at the end
   1729              of this function definition
   1730           */
   1731           DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Loaded\n");
   1732         }
   1733         else
   1734         {
   1735           DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Loaded-Pending\n");
   1736           BITMASK_SET(&m_flags, OMX_COMPONENT_LOADING_PENDING);
   1737           // Skip the event notification
   1738           bFlag = 0;
   1739         }
   1740       }
   1741       /* Requesting transition from Idle to Executing */
   1742       else if(eState == OMX_StateExecuting)
   1743       {
   1744 	    DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Executing\n");
   1745         //BITMASK_SET(&m_flags, OMX_COMPONENT_EXECUTE_PENDING);
   1746         bFlag = 1;
   1747 	    DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Executing\n");
   1748 	    m_state=OMX_StateExecuting;
   1749 	    printf("Stream On CAPTURE Was successful\n");
   1750       }
   1751       /* Requesting transition from Idle to Idle */
   1752       else if(eState == OMX_StateIdle)
   1753       {
   1754         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Idle-->Idle\n");
   1755         post_event(OMX_EventError,OMX_ErrorSameState,\
   1756                    OMX_COMPONENT_GENERATE_EVENT);
   1757         eRet = OMX_ErrorSameState;
   1758       }
   1759       /* Requesting transition from Idle to WaitForResources */
   1760       else if(eState == OMX_StateWaitForResources)
   1761       {
   1762         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Idle-->WaitForResources\n");
   1763         post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
   1764                    OMX_COMPONENT_GENERATE_EVENT);
   1765         eRet = OMX_ErrorIncorrectStateTransition;
   1766       }
   1767        /* Requesting transition from Idle to Pause */
   1768        else if(eState == OMX_StatePause)
   1769       {
   1770          /*To pause the Video core we need to start the driver*/
   1771          if (/*ioctl (drv_ctx.video_driver_fd,VDEC_IOCTL_CMD_START,
   1772                     NULL) < */0)
   1773          {
   1774            DEBUG_PRINT_ERROR("\n VDEC_IOCTL_CMD_START FAILED");
   1775            omx_report_error ();
   1776            eRet = OMX_ErrorHardware;
   1777          }
   1778          else
   1779          {
   1780            BITMASK_SET(&m_flags,OMX_COMPONENT_PAUSE_PENDING);
   1781            DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Pause\n");
   1782            bFlag = 0;
   1783          }
   1784       }
   1785       /* Requesting transition from Idle to Invalid */
   1786        else if(eState == OMX_StateInvalid)
   1787       {
   1788         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Idle-->Invalid\n");
   1789         post_event(OMX_EventError,eState,OMX_COMPONENT_GENERATE_EVENT);
   1790         eRet = OMX_ErrorInvalidState;
   1791       }
   1792       else
   1793       {
   1794         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Idle --> %d Not Handled\n",eState);
   1795         eRet = OMX_ErrorBadParameter;
   1796       }
   1797     }
   1798 
   1799     /******************************/
   1800     /* Current State is Executing */
   1801     /******************************/
   1802     else if(m_state == OMX_StateExecuting)
   1803     {
   1804        DEBUG_PRINT_LOW("\n Command Recieved in OMX_StateExecuting");
   1805        /* Requesting transition from Executing to Idle */
   1806        if(eState == OMX_StateIdle)
   1807        {
   1808          /* Since error is None , we will post an event
   1809          at the end of this function definition
   1810          */
   1811          DEBUG_PRINT_LOW("\n send_command_proxy(): Executing --> Idle \n");
   1812          //BITMASK_SET(&m_flags,OMX_COMPONENT_IDLE_PENDING);
   1813          if(!sem_posted)
   1814          {
   1815            sem_posted = 1;
   1816            sem_post (&m_cmd_lock);
   1817            execute_omx_flush(OMX_ALL);
   1818          }
   1819          bFlag = 1;
   1820 	 int rc=0;
   1821 	 enum v4l2_buf_type btype;
   1822 	 btype = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
   1823 	 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_STREAMOFF, &btype);
   1824 	 if (rc) {
   1825 		 /*TODO: How to handle this case */
   1826 		 printf("\n Failed to call streamoff on OUTPUT Port \n");
   1827 	 } else {
   1828 		 streaming[OUTPUT_PORT] = false;
   1829 	 }
   1830 	 btype = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   1831 	 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_STREAMOFF, &btype);
   1832 	 if (rc) {
   1833 		 /*TODO: How to handle this case */
   1834 		 printf("\n Failed to call streamoff on CAPTURE Port \n");
   1835 	 } else {
   1836 		 streaming[CAPTURE_PORT] = false;
   1837 	 }
   1838 		struct v4l2_event_subscription sub;
   1839 		sub.type=V4L2_EVENT_ALL;
   1840 		ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_UNSUBSCRIBE_EVENT, &sub);
   1841 		if (ret) {
   1842 		printf("\n Subscribe Event Failed \n");
   1843 		eRet = OMX_ErrorHardware;
   1844 		}
   1845 	 m_state == OMX_StateIdle;
   1846        }
   1847        /* Requesting transition from Executing to Paused */
   1848        else if(eState == OMX_StatePause)
   1849        {
   1850          DEBUG_PRINT_LOW("\n PAUSE Command Issued");
   1851          if (/*ioctl (drv_ctx.video_driver_fd,VDEC_IOCTL_CMD_PAUSE,
   1852                     NULL) < */0)
   1853          {
   1854            DEBUG_PRINT_ERROR("\n Error In Pause State");
   1855            post_event(OMX_EventError,OMX_ErrorHardware,\
   1856                       OMX_COMPONENT_GENERATE_EVENT);
   1857            eRet = OMX_ErrorHardware;
   1858          }
   1859          else
   1860          {
   1861            BITMASK_SET(&m_flags,OMX_COMPONENT_PAUSE_PENDING);
   1862            DEBUG_PRINT_LOW("send_command_proxy(): Executing-->Pause\n");
   1863            bFlag = 0;
   1864          }
   1865        }
   1866        /* Requesting transition from Executing to Loaded */
   1867        else if(eState == OMX_StateLoaded)
   1868        {
   1869          DEBUG_PRINT_ERROR("\n send_command_proxy(): Executing --> Loaded \n");
   1870          post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
   1871                     OMX_COMPONENT_GENERATE_EVENT);
   1872          eRet = OMX_ErrorIncorrectStateTransition;
   1873        }
   1874        /* Requesting transition from Executing to WaitForResources */
   1875        else if(eState == OMX_StateWaitForResources)
   1876        {
   1877          DEBUG_PRINT_ERROR("\n send_command_proxy(): Executing --> WaitForResources \n");
   1878          post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
   1879                     OMX_COMPONENT_GENERATE_EVENT);
   1880          eRet = OMX_ErrorIncorrectStateTransition;
   1881        }
   1882        /* Requesting transition from Executing to Executing */
   1883        else if(eState == OMX_StateExecuting)
   1884        {
   1885          DEBUG_PRINT_ERROR("\n send_command_proxy(): Executing --> Executing \n");
   1886          post_event(OMX_EventError,OMX_ErrorSameState,\
   1887                     OMX_COMPONENT_GENERATE_EVENT);
   1888          eRet = OMX_ErrorSameState;
   1889        }
   1890        /* Requesting transition from Executing to Invalid */
   1891        else if(eState == OMX_StateInvalid)
   1892        {
   1893          DEBUG_PRINT_ERROR("\n send_command_proxy(): Executing --> Invalid \n");
   1894          post_event(OMX_EventError,eState,OMX_COMPONENT_GENERATE_EVENT);
   1895          eRet = OMX_ErrorInvalidState;
   1896        }
   1897        else
   1898        {
   1899          DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Executing --> %d Not Handled\n",eState);
   1900          eRet = OMX_ErrorBadParameter;
   1901        }
   1902     }
   1903     /***************************/
   1904     /* Current State is Pause  */
   1905     /***************************/
   1906     else if(m_state == OMX_StatePause)
   1907     {
   1908       /* Requesting transition from Pause to Executing */
   1909       if(eState == OMX_StateExecuting)
   1910       {
   1911         DEBUG_PRINT_LOW("\n Pause --> Executing \n");
   1912         if (/*ioctl (drv_ctx.video_driver_fd,VDEC_IOCTL_CMD_RESUME,
   1913                    NULL) < */0)
   1914         {
   1915           DEBUG_PRINT_ERROR("\n VDEC_IOCTL_CMD_RESUME failed");
   1916           post_event(OMX_EventError,OMX_ErrorHardware,\
   1917                      OMX_COMPONENT_GENERATE_EVENT);
   1918           eRet = OMX_ErrorHardware;
   1919         }
   1920         else
   1921         {
   1922           BITMASK_SET(&m_flags,OMX_COMPONENT_EXECUTE_PENDING);
   1923           DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Executing\n");
   1924           post_event (NULL,VDEC_S_SUCCESS,\
   1925                       OMX_COMPONENT_GENERATE_RESUME_DONE);
   1926           bFlag = 0;
   1927         }
   1928       }
   1929       /* Requesting transition from Pause to Idle */
   1930       else if(eState == OMX_StateIdle)
   1931       {
   1932         /* Since error is None , we will post an event
   1933         at the end of this function definition */
   1934         DEBUG_PRINT_LOW("\n Pause --> Idle \n");
   1935          BITMASK_SET(&m_flags,OMX_COMPONENT_IDLE_PENDING);
   1936          if(!sem_posted)
   1937          {
   1938            sem_posted = 1;
   1939            sem_post (&m_cmd_lock);
   1940            execute_omx_flush(OMX_ALL);
   1941          }
   1942          bFlag = 0;
   1943       }
   1944       /* Requesting transition from Pause to loaded */
   1945       else if(eState == OMX_StateLoaded)
   1946       {
   1947         DEBUG_PRINT_ERROR("\n Pause --> loaded \n");
   1948         post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
   1949                    OMX_COMPONENT_GENERATE_EVENT);
   1950         eRet = OMX_ErrorIncorrectStateTransition;
   1951       }
   1952       /* Requesting transition from Pause to WaitForResources */
   1953       else if(eState == OMX_StateWaitForResources)
   1954       {
   1955         DEBUG_PRINT_ERROR("\n Pause --> WaitForResources \n");
   1956         post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
   1957                    OMX_COMPONENT_GENERATE_EVENT);
   1958         eRet = OMX_ErrorIncorrectStateTransition;
   1959       }
   1960       /* Requesting transition from Pause to Pause */
   1961       else if(eState == OMX_StatePause)
   1962       {
   1963         DEBUG_PRINT_ERROR("\n Pause --> Pause \n");
   1964         post_event(OMX_EventError,OMX_ErrorSameState,\
   1965                    OMX_COMPONENT_GENERATE_EVENT);
   1966         eRet = OMX_ErrorSameState;
   1967       }
   1968        /* Requesting transition from Pause to Invalid */
   1969       else if(eState == OMX_StateInvalid)
   1970       {
   1971         DEBUG_PRINT_ERROR("\n Pause --> Invalid \n");
   1972         post_event(OMX_EventError,eState,OMX_COMPONENT_GENERATE_EVENT);
   1973         eRet = OMX_ErrorInvalidState;
   1974       }
   1975       else
   1976       {
   1977         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Paused --> %d Not Handled\n",eState);
   1978         eRet = OMX_ErrorBadParameter;
   1979       }
   1980     }
   1981      /***************************/
   1982     /* Current State is WaitForResources  */
   1983     /***************************/
   1984     else if(m_state == OMX_StateWaitForResources)
   1985     {
   1986       /* Requesting transition from WaitForResources to Loaded */
   1987       if(eState == OMX_StateLoaded)
   1988       {
   1989         /* Since error is None , we will post an event
   1990         at the end of this function definition */
   1991         DEBUG_PRINT_LOW("send_command_proxy(): WaitForResources-->Loaded\n");
   1992       }
   1993       /* Requesting transition from WaitForResources to WaitForResources */
   1994       else if (eState == OMX_StateWaitForResources)
   1995       {
   1996         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): WaitForResources-->WaitForResources\n");
   1997         post_event(OMX_EventError,OMX_ErrorSameState,
   1998                    OMX_COMPONENT_GENERATE_EVENT);
   1999         eRet = OMX_ErrorSameState;
   2000       }
   2001       /* Requesting transition from WaitForResources to Executing */
   2002       else if(eState == OMX_StateExecuting)
   2003       {
   2004         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): WaitForResources-->Executing\n");
   2005         post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
   2006                    OMX_COMPONENT_GENERATE_EVENT);
   2007         eRet = OMX_ErrorIncorrectStateTransition;
   2008       }
   2009       /* Requesting transition from WaitForResources to Pause */
   2010       else if(eState == OMX_StatePause)
   2011       {
   2012         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): WaitForResources-->Pause\n");
   2013         post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
   2014                    OMX_COMPONENT_GENERATE_EVENT);
   2015         eRet = OMX_ErrorIncorrectStateTransition;
   2016       }
   2017       /* Requesting transition from WaitForResources to Invalid */
   2018       else if(eState == OMX_StateInvalid)
   2019       {
   2020         DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): WaitForResources-->Invalid\n");
   2021         post_event(OMX_EventError,eState,OMX_COMPONENT_GENERATE_EVENT);
   2022         eRet = OMX_ErrorInvalidState;
   2023       }
   2024       /* Requesting transition from WaitForResources to Loaded -
   2025       is NOT tested by Khronos TS */
   2026 
   2027     }
   2028     else
   2029     {
   2030       DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): %d --> %d(Not Handled)\n",m_state,eState);
   2031       eRet = OMX_ErrorBadParameter;
   2032     }
   2033   }
   2034   /********************************/
   2035   /* Current State is Invalid */
   2036   /*******************************/
   2037   else if(m_state == OMX_StateInvalid)
   2038   {
   2039     /* State Transition from Inavlid to any state */
   2040     if(eState == (OMX_StateLoaded || OMX_StateWaitForResources
   2041                   || OMX_StateIdle || OMX_StateExecuting
   2042                   || OMX_StatePause || OMX_StateInvalid))
   2043     {
   2044       DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Invalid -->Loaded\n");
   2045       post_event(OMX_EventError,OMX_ErrorInvalidState,\
   2046                  OMX_COMPONENT_GENERATE_EVENT);
   2047       eRet = OMX_ErrorInvalidState;
   2048     }
   2049   }
   2050   else if (cmd == OMX_CommandFlush)
   2051   {
   2052     DEBUG_PRINT_HIGH("\n send_command_proxy(): OMX_CommandFlush issued"
   2053         "with param1: %d", param1);
   2054     if(OMX_CORE_INPUT_PORT_INDEX == param1 || OMX_ALL == param1)
   2055     {
   2056       BITMASK_SET(&m_flags, OMX_COMPONENT_INPUT_FLUSH_PENDING);
   2057     }
   2058     if(OMX_CORE_OUTPUT_PORT_INDEX == param1 || OMX_ALL == param1)
   2059     {
   2060       BITMASK_SET(&m_flags, OMX_COMPONENT_OUTPUT_FLUSH_PENDING);
   2061     }
   2062     if (!sem_posted){
   2063       sem_posted = 1;
   2064       DEBUG_PRINT_LOW("\n Set the Semaphore");
   2065       sem_post (&m_cmd_lock);
   2066       execute_omx_flush(param1);
   2067     }
   2068     bFlag = 0;
   2069   }
   2070   else if ( cmd == OMX_CommandPortEnable)
   2071   {
   2072     DEBUG_PRINT_HIGH("\n send_command_proxy(): OMX_CommandPortEnable issued"
   2073         "with param1: %d", param1);
   2074     if(param1 == OMX_CORE_INPUT_PORT_INDEX || param1 == OMX_ALL)
   2075       {
   2076         m_inp_bEnabled = OMX_TRUE;
   2077 
   2078         if( (m_state == OMX_StateLoaded &&
   2079              !BITMASK_PRESENT(&m_flags,OMX_COMPONENT_IDLE_PENDING))
   2080             || allocate_input_done())
   2081         {
   2082           post_event(OMX_CommandPortEnable,OMX_CORE_INPUT_PORT_INDEX,
   2083                      OMX_COMPONENT_GENERATE_EVENT);
   2084         }
   2085         else
   2086         {
   2087           DEBUG_PRINT_LOW("send_command_proxy(): Disabled-->Enabled Pending\n");
   2088           BITMASK_SET(&m_flags, OMX_COMPONENT_INPUT_ENABLE_PENDING);
   2089           // Skip the event notification
   2090           bFlag = 0;
   2091         }
   2092       }
   2093       if(param1 == OMX_CORE_OUTPUT_PORT_INDEX || param1 == OMX_ALL)
   2094       {
   2095           DEBUG_PRINT_LOW("\n Enable output Port command recieved");
   2096           m_out_bEnabled = OMX_TRUE;
   2097 
   2098           if( (m_state == OMX_StateLoaded &&
   2099               !BITMASK_PRESENT(&m_flags,OMX_COMPONENT_IDLE_PENDING))
   2100               || (allocate_output_done()))
   2101           {
   2102              post_event(OMX_CommandPortEnable,OMX_CORE_OUTPUT_PORT_INDEX,
   2103                         OMX_COMPONENT_GENERATE_EVENT);
   2104 
   2105           }
   2106           else
   2107           {
   2108               DEBUG_PRINT_LOW("send_command_proxy(): Disabled-->Enabled Pending\n");
   2109               BITMASK_SET(&m_flags, OMX_COMPONENT_OUTPUT_ENABLE_PENDING);
   2110               // Skip the event notification
   2111               bFlag = 0;
   2112           }
   2113       }
   2114   }
   2115   else if (cmd == OMX_CommandPortDisable)
   2116   {
   2117       DEBUG_PRINT_HIGH("\n send_command_proxy(): OMX_CommandPortDisable issued"
   2118           "with param1: %d", param1);
   2119       if(param1 == OMX_CORE_INPUT_PORT_INDEX || param1 == OMX_ALL)
   2120       {
   2121           m_inp_bEnabled = OMX_FALSE;
   2122           if((m_state == OMX_StateLoaded || m_state == OMX_StateIdle)
   2123               && release_input_done())
   2124           {
   2125              post_event(OMX_CommandPortDisable,OMX_CORE_INPUT_PORT_INDEX,
   2126                         OMX_COMPONENT_GENERATE_EVENT);
   2127           }
   2128           else
   2129           {
   2130              BITMASK_SET(&m_flags, OMX_COMPONENT_INPUT_DISABLE_PENDING);
   2131              if(m_state == OMX_StatePause ||m_state == OMX_StateExecuting)
   2132              {
   2133                if(!sem_posted)
   2134                {
   2135                  sem_posted = 1;
   2136                  sem_post (&m_cmd_lock);
   2137                }
   2138                execute_omx_flush(OMX_CORE_INPUT_PORT_INDEX);
   2139              }
   2140 
   2141              // Skip the event notification
   2142              bFlag = 0;
   2143           }
   2144       }
   2145       if(param1 == OMX_CORE_OUTPUT_PORT_INDEX || param1 == OMX_ALL)
   2146       {
   2147           m_out_bEnabled = OMX_FALSE;
   2148           DEBUG_PRINT_LOW("\n Disable output Port command recieved");
   2149           if((m_state == OMX_StateLoaded || m_state == OMX_StateIdle)
   2150               && release_output_done())
   2151           {
   2152              post_event(OMX_CommandPortDisable,OMX_CORE_OUTPUT_PORT_INDEX,\
   2153                         OMX_COMPONENT_GENERATE_EVENT);
   2154           }
   2155           else
   2156          {
   2157             BITMASK_SET(&m_flags, OMX_COMPONENT_OUTPUT_DISABLE_PENDING);
   2158             if(m_state == OMX_StatePause ||m_state == OMX_StateExecuting)
   2159             {
   2160               if (!sem_posted)
   2161               {
   2162                 sem_posted = 1;
   2163                 sem_post (&m_cmd_lock);
   2164               }
   2165                 BITMASK_SET(&m_flags, OMX_COMPONENT_OUTPUT_FLUSH_IN_DISABLE_PENDING);
   2166                 execute_omx_flush(OMX_CORE_OUTPUT_PORT_INDEX);
   2167             }
   2168             // Skip the event notification
   2169             bFlag = 0;
   2170 
   2171          }
   2172       }
   2173   }
   2174   else
   2175   {
   2176     DEBUG_PRINT_ERROR("Error: Invalid Command other than StateSet (%d)\n",cmd);
   2177     eRet = OMX_ErrorNotImplemented;
   2178   }
   2179   if(eRet == OMX_ErrorNone && bFlag)
   2180   {
   2181     post_event(cmd,eState,OMX_COMPONENT_GENERATE_EVENT);
   2182   }
   2183   if(!sem_posted)
   2184   {
   2185     sem_post(&m_cmd_lock);
   2186   }
   2187 
   2188   return eRet;
   2189 }
   2190 
   2191 /* ======================================================================
   2192 FUNCTION
   2193   omx_vdec::ExecuteOmxFlush
   2194 
   2195 DESCRIPTION
   2196   Executes the OMX flush.
   2197 
   2198 PARAMETERS
   2199   flushtype - input flush(1)/output flush(0)/ both.
   2200 
   2201 RETURN VALUE
   2202   true/false
   2203 
   2204 ========================================================================== */
   2205 bool omx_vdec::execute_omx_flush(OMX_U32 flushType)
   2206 {
   2207   struct vdec_ioctl_msg ioctl_msg = {NULL, NULL};
   2208   enum vdec_bufferflush flush_dir;
   2209   bool bRet = false;
   2210   switch (flushType)
   2211   {
   2212     case OMX_CORE_INPUT_PORT_INDEX:
   2213       input_flush_progress = true;
   2214       flush_dir = VDEC_FLUSH_TYPE_INPUT;
   2215     break;
   2216     case OMX_CORE_OUTPUT_PORT_INDEX:
   2217       output_flush_progress = true;
   2218       flush_dir = VDEC_FLUSH_TYPE_OUTPUT;
   2219     break;
   2220     default:
   2221       input_flush_progress = true;
   2222       output_flush_progress = true;
   2223       flush_dir = VDEC_FLUSH_TYPE_ALL;
   2224   }
   2225   ioctl_msg.in = &flush_dir;
   2226   ioctl_msg.out = NULL;
   2227   if (/*ioctl(drv_ctx.video_driver_fd, VDEC_IOCTL_CMD_FLUSH, &ioctl_msg) < */0)
   2228   {
   2229     DEBUG_PRINT_ERROR("\n Flush Port (%d) Failed ", (int)flush_dir);
   2230     bRet = false;
   2231   }
   2232   return bRet;
   2233 }
   2234 /*=========================================================================
   2235 FUNCTION : execute_output_flush
   2236 
   2237 DESCRIPTION
   2238   Executes the OMX flush at OUTPUT PORT.
   2239 
   2240 PARAMETERS
   2241   None.
   2242 
   2243 RETURN VALUE
   2244   true/false
   2245 ==========================================================================*/
   2246 bool omx_vdec::execute_output_flush()
   2247 {
   2248   unsigned      p1 = 0; // Parameter - 1
   2249   unsigned      p2 = 0; // Parameter - 2
   2250   unsigned      ident = 0;
   2251   bool bRet = true;
   2252 
   2253   /*Generate FBD for all Buffers in the FTBq*/
   2254   pthread_mutex_lock(&m_lock);
   2255   DEBUG_PRINT_LOW("\n Initiate Output Flush");
   2256   while (m_ftb_q.m_size)
   2257   {
   2258     DEBUG_PRINT_LOW("\n Buffer queue size %d pending buf cnt %d",
   2259                        m_ftb_q.m_size,pending_output_buffers);
   2260     m_ftb_q.pop_entry(&p1,&p2,&ident);
   2261     DEBUG_PRINT_LOW("\n ID(%x) P1(%x) P2(%x)", ident, p1, p2);
   2262     if(ident == OMX_COMPONENT_GENERATE_FTB )
   2263     {
   2264       pending_output_buffers++;
   2265       fill_buffer_done(&m_cmp,(OMX_BUFFERHEADERTYPE *)p2);
   2266     }
   2267     else if (ident == OMX_COMPONENT_GENERATE_FBD)
   2268     {
   2269       fill_buffer_done(&m_cmp,(OMX_BUFFERHEADERTYPE *)p1);
   2270     }
   2271   }
   2272   pthread_mutex_unlock(&m_lock);
   2273   output_flush_progress = false;
   2274 
   2275   if (arbitrary_bytes)
   2276   {
   2277     prev_ts = LLONG_MAX;
   2278     rst_prev_ts = true;
   2279   }
   2280   DEBUG_PRINT_HIGH("\n OMX flush o/p Port complete PenBuf(%d)", pending_output_buffers);
   2281   return bRet;
   2282 }
   2283 /*=========================================================================
   2284 FUNCTION : execute_input_flush
   2285 
   2286 DESCRIPTION
   2287   Executes the OMX flush at INPUT PORT.
   2288 
   2289 PARAMETERS
   2290   None.
   2291 
   2292 RETURN VALUE
   2293   true/false
   2294 ==========================================================================*/
   2295 bool omx_vdec::execute_input_flush()
   2296 {
   2297   unsigned       i =0;
   2298   unsigned      p1 = 0; // Parameter - 1
   2299   unsigned      p2 = 0; // Parameter - 2
   2300   unsigned      ident = 0;
   2301   bool bRet = true;
   2302 
   2303   /*Generate EBD for all Buffers in the ETBq*/
   2304   DEBUG_PRINT_LOW("\n Initiate Input Flush \n");
   2305 
   2306   pthread_mutex_lock(&m_lock);
   2307   DEBUG_PRINT_LOW("\n Check if the Queue is empty \n");
   2308   while (m_etb_q.m_size)
   2309   {
   2310     m_etb_q.pop_entry(&p1,&p2,&ident);
   2311 
   2312     if (ident == OMX_COMPONENT_GENERATE_ETB_ARBITRARY)
   2313     {
   2314       DEBUG_PRINT_LOW("\n Flush Input Heap Buffer %p",(OMX_BUFFERHEADERTYPE *)p2);
   2315       m_cb.EmptyBufferDone(&m_cmp ,m_app_data, (OMX_BUFFERHEADERTYPE *)p2);
   2316     }
   2317     else if(ident == OMX_COMPONENT_GENERATE_ETB)
   2318     {
   2319       pending_input_buffers++;
   2320       DEBUG_PRINT_LOW("\n Flush Input OMX_COMPONENT_GENERATE_ETB %p, pending_input_buffers %d",
   2321         (OMX_BUFFERHEADERTYPE *)p2, pending_input_buffers);
   2322       empty_buffer_done(&m_cmp,(OMX_BUFFERHEADERTYPE *)p2);
   2323     }
   2324     else if (ident == OMX_COMPONENT_GENERATE_EBD)
   2325     {
   2326       DEBUG_PRINT_LOW("\n Flush Input OMX_COMPONENT_GENERATE_EBD %p",
   2327         (OMX_BUFFERHEADERTYPE *)p1);
   2328       empty_buffer_done(&m_cmp,(OMX_BUFFERHEADERTYPE *)p1);
   2329     }
   2330   }
   2331   time_stamp_dts.flush_timestamp();
   2332   /*Check if Heap Buffers are to be flushed*/
   2333   if (arbitrary_bytes)
   2334   {
   2335     DEBUG_PRINT_LOW("\n Reset all the variables before flusing");
   2336     h264_scratch.nFilledLen = 0;
   2337     nal_count = 0;
   2338     look_ahead_nal = false;
   2339     frame_count = 0;
   2340     h264_last_au_ts = LLONG_MAX;
   2341     h264_last_au_flags = 0;
   2342     memset(m_demux_offsets, 0, ( sizeof(OMX_U32) * 8192) );
   2343     m_demux_entries = 0;
   2344     DEBUG_PRINT_LOW("\n Initialize parser");
   2345     if (m_frame_parser.mutils)
   2346     {
   2347       m_frame_parser.mutils->initialize_frame_checking_environment();
   2348     }
   2349 
   2350     while (m_input_pending_q.m_size)
   2351     {
   2352       m_input_pending_q.pop_entry(&p1,&p2,&ident);
   2353       m_cb.EmptyBufferDone(&m_cmp ,m_app_data, (OMX_BUFFERHEADERTYPE *)p1);
   2354     }
   2355 
   2356     if (psource_frame)
   2357     {
   2358       m_cb.EmptyBufferDone(&m_cmp ,m_app_data,psource_frame);
   2359       psource_frame = NULL;
   2360     }
   2361 
   2362     if (pdest_frame)
   2363     {
   2364       pdest_frame->nFilledLen = 0;
   2365       m_input_free_q.insert_entry((unsigned) pdest_frame,NULL,NULL);
   2366       pdest_frame = NULL;
   2367     }
   2368     m_frame_parser.flush();
   2369   }
   2370   pthread_mutex_unlock(&m_lock);
   2371   input_flush_progress = false;
   2372   if (!arbitrary_bytes)
   2373   {
   2374     prev_ts = LLONG_MAX;
   2375     rst_prev_ts = true;
   2376   }
   2377 #ifdef _ANDROID_
   2378   if (m_debug_timestamp)
   2379   {
   2380     m_timestamp_list.reset_ts_list();
   2381   }
   2382 #endif
   2383   DEBUG_PRINT_HIGH("\n OMX flush i/p Port complete PenBuf(%d)", pending_input_buffers);
   2384   return bRet;
   2385 }
   2386 
   2387 
   2388 /* ======================================================================
   2389 FUNCTION
   2390   omx_vdec::SendCommandEvent
   2391 
   2392 DESCRIPTION
   2393   Send the event to decoder pipe.  This is needed to generate the callbacks
   2394   in decoder thread context.
   2395 
   2396 PARAMETERS
   2397   None.
   2398 
   2399 RETURN VALUE
   2400   true/false
   2401 
   2402 ========================================================================== */
   2403 bool omx_vdec::post_event(unsigned int p1,
   2404                           unsigned int p2,
   2405                           unsigned int id)
   2406 {
   2407   bool bRet      =                      false;
   2408 
   2409 
   2410   pthread_mutex_lock(&m_lock);
   2411 
   2412   if (id == OMX_COMPONENT_GENERATE_FTB ||
   2413       id == OMX_COMPONENT_GENERATE_FBD)
   2414   {
   2415     m_ftb_q.insert_entry(p1,p2,id);
   2416   }
   2417   else if (id == OMX_COMPONENT_GENERATE_ETB ||
   2418            id == OMX_COMPONENT_GENERATE_EBD ||
   2419            id == OMX_COMPONENT_GENERATE_ETB_ARBITRARY)
   2420   {
   2421 	  m_etb_q.insert_entry(p1,p2,id);
   2422   }
   2423   else
   2424   {
   2425     m_cmd_q.insert_entry(p1,p2,id);
   2426   }
   2427 
   2428   bRet = true;
   2429   DEBUG_PRINT_LOW("\n Value of this pointer in post_event %p",this);
   2430   post_message(this, id);
   2431 
   2432   pthread_mutex_unlock(&m_lock);
   2433 
   2434   return bRet;
   2435 }
   2436 
   2437 OMX_ERRORTYPE omx_vdec::get_supported_profile_level_for_1080p(OMX_VIDEO_PARAM_PROFILELEVELTYPE *profileLevelType)
   2438 {
   2439   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   2440   if(!profileLevelType)
   2441     return OMX_ErrorBadParameter;
   2442 
   2443   if(profileLevelType->nPortIndex == 0) {
   2444     if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc",OMX_MAX_STRINGNAME_SIZE))
   2445     {
   2446       if (profileLevelType->nProfileIndex == 0)
   2447       {
   2448         profileLevelType->eProfile = OMX_VIDEO_AVCProfileBaseline;
   2449         profileLevelType->eLevel   = OMX_VIDEO_AVCLevel4;
   2450 
   2451       }
   2452       else if (profileLevelType->nProfileIndex == 1)
   2453       {
   2454         profileLevelType->eProfile = OMX_VIDEO_AVCProfileMain;
   2455         profileLevelType->eLevel   = OMX_VIDEO_AVCLevel4;
   2456       }
   2457       else if(profileLevelType->nProfileIndex == 2)
   2458       {
   2459         profileLevelType->eProfile = OMX_VIDEO_AVCProfileHigh;
   2460         profileLevelType->eLevel   = OMX_VIDEO_AVCLevel4;
   2461       }
   2462       else
   2463       {
   2464         DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported nProfileIndex ret NoMore %d\n",
   2465             profileLevelType->nProfileIndex);
   2466         eRet = OMX_ErrorNoMore;
   2467       }
   2468     }
   2469     else if((!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.h263",OMX_MAX_STRINGNAME_SIZE)))
   2470     {
   2471       if (profileLevelType->nProfileIndex == 0)
   2472       {
   2473         profileLevelType->eProfile = OMX_VIDEO_H263ProfileBaseline;
   2474         profileLevelType->eLevel   = OMX_VIDEO_H263Level70;
   2475       }
   2476       else
   2477       {
   2478         DEBUG_PRINT_ERROR("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported nProfileIndex ret NoMore %d\n", profileLevelType->nProfileIndex);
   2479         eRet = OMX_ErrorNoMore;
   2480       }
   2481     }
   2482     else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg4",OMX_MAX_STRINGNAME_SIZE))
   2483     {
   2484       if (profileLevelType->nProfileIndex == 0)
   2485       {
   2486         profileLevelType->eProfile = OMX_VIDEO_MPEG4ProfileSimple;
   2487         profileLevelType->eLevel   = OMX_VIDEO_MPEG4Level5;
   2488       }
   2489       else if(profileLevelType->nProfileIndex == 1)
   2490       {
   2491         profileLevelType->eProfile = OMX_VIDEO_MPEG4ProfileAdvancedSimple;
   2492         profileLevelType->eLevel   = OMX_VIDEO_MPEG4Level5;
   2493       }
   2494       else
   2495       {
   2496         DEBUG_PRINT_ERROR("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported nProfileIndex ret NoMore %d\n", profileLevelType->nProfileIndex);
   2497         eRet = OMX_ErrorNoMore;
   2498       }
   2499     }
   2500     else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg2",OMX_MAX_STRINGNAME_SIZE))
   2501     {
   2502       if (profileLevelType->nProfileIndex == 0)
   2503       {
   2504         profileLevelType->eProfile = OMX_VIDEO_MPEG2ProfileSimple;
   2505         profileLevelType->eLevel   = OMX_VIDEO_MPEG2LevelHL;
   2506       }
   2507       else if(profileLevelType->nProfileIndex == 1)
   2508       {
   2509         profileLevelType->eProfile = OMX_VIDEO_MPEG2ProfileMain;
   2510         profileLevelType->eLevel   = OMX_VIDEO_MPEG2LevelHL;
   2511       }
   2512       else
   2513       {
   2514         DEBUG_PRINT_ERROR("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported nProfileIndex ret NoMore %d\n", profileLevelType->nProfileIndex);
   2515         eRet = OMX_ErrorNoMore;
   2516       }
   2517     }
   2518   }
   2519   else
   2520   {
   2521     DEBUG_PRINT_ERROR("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported should be queries on Input port only %d\n", profileLevelType->nPortIndex);
   2522     eRet = OMX_ErrorBadPortIndex;
   2523   }
   2524   return eRet;
   2525 }
   2526 
   2527 /* ======================================================================
   2528 FUNCTION
   2529   omx_vdec::GetParameter
   2530 
   2531 DESCRIPTION
   2532   OMX Get Parameter method implementation
   2533 
   2534 PARAMETERS
   2535   <TBD>.
   2536 
   2537 RETURN VALUE
   2538   Error None if successful.
   2539 
   2540 ========================================================================== */
   2541 OMX_ERRORTYPE  omx_vdec::get_parameter(OMX_IN OMX_HANDLETYPE     hComp,
   2542                                            OMX_IN OMX_INDEXTYPE paramIndex,
   2543                                            OMX_INOUT OMX_PTR     paramData)
   2544 {
   2545     OMX_ERRORTYPE eRet = OMX_ErrorNone;
   2546 
   2547     DEBUG_PRINT_LOW("get_parameter: \n");
   2548     if(m_state == OMX_StateInvalid)
   2549     {
   2550         DEBUG_PRINT_ERROR("Get Param in Invalid State\n");
   2551         return OMX_ErrorInvalidState;
   2552     }
   2553     if(paramData == NULL)
   2554     {
   2555         DEBUG_PRINT_LOW("Get Param in Invalid paramData \n");
   2556         return OMX_ErrorBadParameter;
   2557     }
   2558   switch(paramIndex)
   2559   {
   2560     case OMX_IndexParamPortDefinition:
   2561     {
   2562       OMX_PARAM_PORTDEFINITIONTYPE *portDefn =
   2563                             (OMX_PARAM_PORTDEFINITIONTYPE *) paramData;
   2564       DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamPortDefinition\n");
   2565       eRet = update_portdef(portDefn);
   2566       if (eRet == OMX_ErrorNone)
   2567           m_port_def = *portDefn;
   2568       break;
   2569     }
   2570     case OMX_IndexParamVideoInit:
   2571     {
   2572       OMX_PORT_PARAM_TYPE *portParamType =
   2573                               (OMX_PORT_PARAM_TYPE *) paramData;
   2574       DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoInit\n");
   2575 
   2576       portParamType->nVersion.nVersion = OMX_SPEC_VERSION;
   2577       portParamType->nSize = sizeof(portParamType);
   2578       portParamType->nPorts           = 2;
   2579       portParamType->nStartPortNumber = 0;
   2580       break;
   2581     }
   2582     case OMX_IndexParamVideoPortFormat:
   2583     {
   2584       OMX_VIDEO_PARAM_PORTFORMATTYPE *portFmt =
   2585                      (OMX_VIDEO_PARAM_PORTFORMATTYPE *)paramData;
   2586       DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoPortFormat\n");
   2587 
   2588       portFmt->nVersion.nVersion = OMX_SPEC_VERSION;
   2589       portFmt->nSize             = sizeof(portFmt);
   2590 
   2591       if (0 == portFmt->nPortIndex)
   2592       {
   2593         if (0 == portFmt->nIndex)
   2594         {
   2595               portFmt->eColorFormat =  OMX_COLOR_FormatUnused;
   2596               portFmt->eCompressionFormat = eCompressionFormat;
   2597         }
   2598         else
   2599         {
   2600           DEBUG_PRINT_ERROR("get_parameter: OMX_IndexParamVideoPortFormat:"\
   2601               " NoMore compression formats\n");
   2602           eRet =  OMX_ErrorNoMore;
   2603         }
   2604       }
   2605       else if (1 == portFmt->nPortIndex)
   2606       {
   2607         portFmt->eCompressionFormat =  OMX_VIDEO_CodingUnused;
   2608 
   2609         if(0 == portFmt->nIndex)
   2610           portFmt->eColorFormat = (OMX_COLOR_FORMATTYPE)
   2611             QOMX_COLOR_FormatYUV420PackedSemiPlanar64x32Tile2m8ka;
   2612         else
   2613         {
   2614            DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoPortFormat:"\
   2615                   " NoMore Color formats\n");
   2616            eRet =  OMX_ErrorNoMore;
   2617         }
   2618 	portFmt->eColorFormat = OMX_COLOR_FormatYUV420SemiPlanar;
   2619       }
   2620       else
   2621       {
   2622         DEBUG_PRINT_ERROR("get_parameter: Bad port index %d\n",
   2623                           (int)portFmt->nPortIndex);
   2624         eRet = OMX_ErrorBadPortIndex;
   2625       }
   2626       break;
   2627     }
   2628     /*Component should support this port definition*/
   2629     case OMX_IndexParamAudioInit:
   2630     {
   2631         OMX_PORT_PARAM_TYPE *audioPortParamType =
   2632                                               (OMX_PORT_PARAM_TYPE *) paramData;
   2633         DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamAudioInit\n");
   2634         audioPortParamType->nVersion.nVersion = OMX_SPEC_VERSION;
   2635         audioPortParamType->nSize = sizeof(audioPortParamType);
   2636         audioPortParamType->nPorts           = 0;
   2637         audioPortParamType->nStartPortNumber = 0;
   2638         break;
   2639     }
   2640     /*Component should support this port definition*/
   2641     case OMX_IndexParamImageInit:
   2642     {
   2643         OMX_PORT_PARAM_TYPE *imagePortParamType =
   2644                                               (OMX_PORT_PARAM_TYPE *) paramData;
   2645         DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamImageInit\n");
   2646         imagePortParamType->nVersion.nVersion = OMX_SPEC_VERSION;
   2647         imagePortParamType->nSize = sizeof(imagePortParamType);
   2648         imagePortParamType->nPorts           = 0;
   2649         imagePortParamType->nStartPortNumber = 0;
   2650         break;
   2651 
   2652     }
   2653     /*Component should support this port definition*/
   2654     case OMX_IndexParamOtherInit:
   2655     {
   2656         DEBUG_PRINT_ERROR("get_parameter: OMX_IndexParamOtherInit %08x\n",
   2657                           paramIndex);
   2658         eRet =OMX_ErrorUnsupportedIndex;
   2659         break;
   2660     }
   2661     case OMX_IndexParamStandardComponentRole:
   2662     {
   2663         OMX_PARAM_COMPONENTROLETYPE *comp_role;
   2664         comp_role = (OMX_PARAM_COMPONENTROLETYPE *) paramData;
   2665         comp_role->nVersion.nVersion = OMX_SPEC_VERSION;
   2666         comp_role->nSize = sizeof(*comp_role);
   2667 
   2668         DEBUG_PRINT_LOW("Getparameter: OMX_IndexParamStandardComponentRole %d\n",
   2669                     paramIndex);
   2670         strlcpy((char*)comp_role->cRole,(const char*)m_cRole,
   2671                     OMX_MAX_STRINGNAME_SIZE);
   2672         break;
   2673     }
   2674     /* Added for parameter test */
   2675     case OMX_IndexParamPriorityMgmt:
   2676         {
   2677 
   2678             OMX_PRIORITYMGMTTYPE *priorityMgmType =
   2679                                              (OMX_PRIORITYMGMTTYPE *) paramData;
   2680             DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamPriorityMgmt\n");
   2681             priorityMgmType->nVersion.nVersion = OMX_SPEC_VERSION;
   2682             priorityMgmType->nSize = sizeof(priorityMgmType);
   2683 
   2684             break;
   2685         }
   2686     /* Added for parameter test */
   2687     case OMX_IndexParamCompBufferSupplier:
   2688         {
   2689             OMX_PARAM_BUFFERSUPPLIERTYPE *bufferSupplierType =
   2690                                      (OMX_PARAM_BUFFERSUPPLIERTYPE*) paramData;
   2691             DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamCompBufferSupplier\n");
   2692 
   2693             bufferSupplierType->nSize = sizeof(bufferSupplierType);
   2694             bufferSupplierType->nVersion.nVersion = OMX_SPEC_VERSION;
   2695             if(0 == bufferSupplierType->nPortIndex)
   2696                 bufferSupplierType->nPortIndex = OMX_BufferSupplyUnspecified;
   2697             else if (1 == bufferSupplierType->nPortIndex)
   2698                 bufferSupplierType->nPortIndex = OMX_BufferSupplyUnspecified;
   2699             else
   2700                 eRet = OMX_ErrorBadPortIndex;
   2701 
   2702 
   2703             break;
   2704         }
   2705     case OMX_IndexParamVideoAvc:
   2706         {
   2707             DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoAvc %08x\n",
   2708                         paramIndex);
   2709             break;
   2710         }
   2711     case OMX_IndexParamVideoH263:
   2712         {
   2713             DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoH263 %08x\n",
   2714                         paramIndex);
   2715             break;
   2716         }
   2717     case OMX_IndexParamVideoMpeg4:
   2718         {
   2719             DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoMpeg4 %08x\n",
   2720                         paramIndex);
   2721             break;
   2722         }
   2723     case OMX_IndexParamVideoMpeg2:
   2724         {
   2725           DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoMpeg2 %08x\n",
   2726               paramIndex);
   2727           break;
   2728         }
   2729     case OMX_IndexParamVideoProfileLevelQuerySupported:
   2730         {
   2731           DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported %08x\n", paramIndex);
   2732           OMX_VIDEO_PARAM_PROFILELEVELTYPE *profileLevelType =
   2733             (OMX_VIDEO_PARAM_PROFILELEVELTYPE *)paramData;
   2734     eRet = get_supported_profile_level_for_1080p(profileLevelType);
   2735           break;
   2736         }
   2737 #if defined (_ANDROID_HONEYCOMB_) || defined (_ANDROID_ICS_)
   2738     case OMX_GoogleAndroidIndexGetAndroidNativeBufferUsage:
   2739         {
   2740             DEBUG_PRINT_LOW("get_parameter: OMX_GoogleAndroidIndexGetAndroidNativeBufferUsage\n");
   2741             GetAndroidNativeBufferUsageParams* nativeBuffersUsage = (GetAndroidNativeBufferUsageParams *) paramData;
   2742             if(nativeBuffersUsage->nPortIndex == OMX_CORE_OUTPUT_PORT_INDEX) {
   2743 
   2744                 if(secure_mode) {
   2745                         nativeBuffersUsage->nUsage = (GRALLOC_USAGE_PRIVATE_MM_HEAP | GRALLOC_USAGE_PROTECTED |
   2746                                                       GRALLOC_USAGE_PRIVATE_UNCACHED);
   2747                 } else {
   2748                         nativeBuffersUsage->nUsage = (GRALLOC_USAGE_PRIVATE_MM_HEAP | GRALLOC_USAGE_PRIVATE_UNCACHED);
   2749                 }
   2750             } else {
   2751                 DEBUG_PRINT_HIGH("get_parameter: OMX_GoogleAndroidIndexGetAndroidNativeBufferUsage failed!\n");
   2752                 eRet = OMX_ErrorBadParameter;
   2753             }
   2754         }
   2755         break;
   2756 #endif
   2757 
   2758     default:
   2759     {
   2760       DEBUG_PRINT_ERROR("get_parameter: unknown param %08x\n", paramIndex);
   2761       eRet =OMX_ErrorUnsupportedIndex;
   2762     }
   2763 
   2764   }
   2765 
   2766   DEBUG_PRINT_LOW("\n get_parameter returning WxH(%d x %d) SxSH(%d x %d)\n",
   2767       drv_ctx.video_resolution.frame_width,
   2768       drv_ctx.video_resolution.frame_height,
   2769       drv_ctx.video_resolution.stride,
   2770       drv_ctx.video_resolution.scan_lines);
   2771 
   2772   return eRet;
   2773 }
   2774 
   2775 #if defined (_ANDROID_HONEYCOMB_) || defined (_ANDROID_ICS_)
   2776 OMX_ERRORTYPE omx_vdec::use_android_native_buffer(OMX_IN OMX_HANDLETYPE hComp, OMX_PTR data)
   2777 {
   2778     DEBUG_PRINT_LOW("Inside use_android_native_buffer");
   2779     OMX_ERRORTYPE eRet = OMX_ErrorNone;
   2780     UseAndroidNativeBufferParams *params = (UseAndroidNativeBufferParams *)data;
   2781 
   2782     if((params == NULL) ||
   2783       (params->nativeBuffer == NULL) ||
   2784       (params->nativeBuffer->handle == NULL) ||
   2785       !m_enable_android_native_buffers)
   2786         return OMX_ErrorBadParameter;
   2787     m_use_android_native_buffers = OMX_TRUE;
   2788     sp<android_native_buffer_t> nBuf = params->nativeBuffer;
   2789     private_handle_t *handle = (private_handle_t *)nBuf->handle;
   2790     if(OMX_CORE_OUTPUT_PORT_INDEX == params->nPortIndex) {  //android native buffers can be used only on Output port
   2791         OMX_U8 *buffer = NULL;
   2792         if(!secure_mode) {
   2793                 buffer = (OMX_U8*)mmap(0, handle->size,
   2794                     PROT_READ|PROT_WRITE, MAP_SHARED, handle->fd, 0);
   2795                 if(buffer == MAP_FAILED) {
   2796                     DEBUG_PRINT_ERROR("Failed to mmap pmem with fd = %d, size = %d", handle->fd, handle->size);
   2797                     return OMX_ErrorInsufficientResources;
   2798             }
   2799         }
   2800         eRet = use_buffer(hComp,params->bufferHeader,params->nPortIndex,data,handle->size,buffer);
   2801     } else {
   2802         eRet = OMX_ErrorBadParameter;
   2803     }
   2804     return eRet;
   2805 }
   2806 #endif
   2807 /* ======================================================================
   2808 FUNCTION
   2809   omx_vdec::Setparameter
   2810 
   2811 DESCRIPTION
   2812   OMX Set Parameter method implementation.
   2813 
   2814 PARAMETERS
   2815   <TBD>.
   2816 
   2817 RETURN VALUE
   2818   OMX Error None if successful.
   2819 
   2820 ========================================================================== */
   2821 OMX_ERRORTYPE  omx_vdec::set_parameter(OMX_IN OMX_HANDLETYPE     hComp,
   2822                                            OMX_IN OMX_INDEXTYPE paramIndex,
   2823                                            OMX_IN OMX_PTR        paramData)
   2824 {
   2825     OMX_ERRORTYPE eRet = OMX_ErrorNone;
   2826     struct vdec_ioctl_msg ioctl_msg = {NULL,NULL};
   2827 
   2828     if(m_state == OMX_StateInvalid)
   2829     {
   2830         DEBUG_PRINT_ERROR("Set Param in Invalid State\n");
   2831         return OMX_ErrorInvalidState;
   2832     }
   2833     if(paramData == NULL)
   2834     {
   2835          DEBUG_PRINT_ERROR("Get Param in Invalid paramData \n");
   2836          return OMX_ErrorBadParameter;
   2837     }
   2838     if((m_state != OMX_StateLoaded) &&
   2839           BITMASK_ABSENT(&m_flags,OMX_COMPONENT_OUTPUT_ENABLE_PENDING) &&
   2840           (m_out_bEnabled == OMX_TRUE) &&
   2841           BITMASK_ABSENT(&m_flags, OMX_COMPONENT_INPUT_ENABLE_PENDING) &&
   2842           (m_inp_bEnabled == OMX_TRUE)) {
   2843         DEBUG_PRINT_ERROR("Set Param in Invalid State \n");
   2844         return OMX_ErrorIncorrectStateOperation;
   2845     }
   2846   switch(paramIndex)
   2847   {
   2848     case OMX_IndexParamPortDefinition:
   2849     {
   2850       OMX_PARAM_PORTDEFINITIONTYPE *portDefn;
   2851       portDefn = (OMX_PARAM_PORTDEFINITIONTYPE *) paramData;
   2852       //TODO: Check if any allocate buffer/use buffer/useNativeBuffer has
   2853       //been called.
   2854       DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamPortDefinition H= %d, W = %d\n",
   2855              (int)portDefn->format.video.nFrameHeight,
   2856              (int)portDefn->format.video.nFrameWidth);
   2857       if(OMX_DirOutput == portDefn->eDir)
   2858       {
   2859           DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamPortDefinition OP port\n");
   2860           m_display_id = portDefn->format.video.pNativeWindow;
   2861           if ( portDefn->nBufferCountActual >= drv_ctx.op_buf.mincount &&
   2862                portDefn->nBufferSize >=  drv_ctx.op_buf.buffer_size )
   2863             {
   2864               drv_ctx.op_buf.actualcount = portDefn->nBufferCountActual;
   2865               drv_ctx.op_buf.buffer_size = portDefn->nBufferSize;
   2866               eRet = set_buffer_req(&drv_ctx.op_buf);
   2867               if (eRet == OMX_ErrorNone)
   2868                   m_port_def = *portDefn;
   2869           }
   2870           else
   2871           {
   2872               DEBUG_PRINT_ERROR("ERROR: OP Requirements(#%d: %u) Requested(#%d: %u)\n",
   2873                 drv_ctx.op_buf.mincount, drv_ctx.op_buf.buffer_size,
   2874                 portDefn->nBufferCountActual, portDefn->nBufferSize);
   2875               eRet = OMX_ErrorBadParameter;
   2876           }
   2877       }
   2878       else if(OMX_DirInput == portDefn->eDir)
   2879       {
   2880         if((portDefn->format.video.xFramerate >> 16) > 0 &&
   2881            (portDefn->format.video.xFramerate >> 16) <= MAX_SUPPORTED_FPS)
   2882         {
   2883             // Frame rate only should be set if this is a "known value" or to
   2884             // activate ts prediction logic (arbitrary mode only) sending input
   2885             // timestamps with max value (LLONG_MAX).
   2886             DEBUG_PRINT_HIGH("set_parameter: frame rate set by omx client : %d",
   2887                              portDefn->format.video.xFramerate >> 16);
   2888             Q16ToFraction(portDefn->format.video.xFramerate, drv_ctx.frame_rate.fps_numerator,
   2889                           drv_ctx.frame_rate.fps_denominator);
   2890             if(!drv_ctx.frame_rate.fps_numerator)
   2891             {
   2892               DEBUG_PRINT_ERROR("Numerator is zero setting to 30");
   2893               drv_ctx.frame_rate.fps_numerator = 30;
   2894             }
   2895             if(drv_ctx.frame_rate.fps_denominator)
   2896               drv_ctx.frame_rate.fps_numerator = (int)
   2897                   drv_ctx.frame_rate.fps_numerator / drv_ctx.frame_rate.fps_denominator;
   2898               drv_ctx.frame_rate.fps_denominator = 1;
   2899             frm_int = drv_ctx.frame_rate.fps_denominator * 1e6 /
   2900                       drv_ctx.frame_rate.fps_numerator;
   2901             ioctl_msg.in = &drv_ctx.frame_rate;
   2902             if (/*ioctl (drv_ctx.video_driver_fd, VDEC_IOCTL_SET_FRAME_RATE,
   2903                        (void*)&ioctl_msg) < */0)
   2904             {
   2905               DEBUG_PRINT_ERROR("Setting frame rate to driver failed");
   2906             }
   2907             DEBUG_PRINT_LOW("set_parameter: frm_int(%u) fps(%.2f)",
   2908                              frm_int, drv_ctx.frame_rate.fps_numerator /
   2909                              (float)drv_ctx.frame_rate.fps_denominator);
   2910         }
   2911          DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamPortDefinition IP port\n");
   2912          if(drv_ctx.video_resolution.frame_height !=
   2913                portDefn->format.video.nFrameHeight ||
   2914              drv_ctx.video_resolution.frame_width  !=
   2915                portDefn->format.video.nFrameWidth)
   2916          {
   2917              DEBUG_PRINT_LOW("\n SetParam IP: WxH(%d x %d)\n",
   2918                            portDefn->format.video.nFrameWidth,
   2919                            portDefn->format.video.nFrameHeight);
   2920              if (portDefn->format.video.nFrameHeight != 0x0 &&
   2921                  portDefn->format.video.nFrameWidth != 0x0)
   2922              {
   2923                drv_ctx.video_resolution.frame_height =
   2924                  drv_ctx.video_resolution.scan_lines =
   2925                  portDefn->format.video.nFrameHeight;
   2926                drv_ctx.video_resolution.frame_width =
   2927                  drv_ctx.video_resolution.stride =
   2928                  portDefn->format.video.nFrameWidth;
   2929                ioctl_msg.in = &drv_ctx.video_resolution;
   2930                ioctl_msg.out = NULL;
   2931                if (/*ioctl (drv_ctx.video_driver_fd, VDEC_IOCTL_SET_PICRES,
   2932                             (void*)&ioctl_msg) < */0)
   2933                {
   2934                    DEBUG_PRINT_ERROR("\n Set Resolution failed");
   2935                    eRet = OMX_ErrorUnsupportedSetting;
   2936                }
   2937                else
   2938                    eRet = get_buffer_req(&drv_ctx.op_buf);
   2939              }
   2940          }
   2941          else if (portDefn->nBufferCountActual >= drv_ctx.ip_buf.mincount
   2942                   && portDefn->nBufferSize == drv_ctx.ip_buf.buffer_size)
   2943          {
   2944              drv_ctx.ip_buf.actualcount = portDefn->nBufferCountActual;
   2945              drv_ctx.ip_buf.buffer_size = portDefn->nBufferSize;
   2946              eRet = set_buffer_req(&drv_ctx.ip_buf);
   2947          }
   2948          else
   2949          {
   2950              DEBUG_PRINT_ERROR("ERROR: IP Requirements(#%d: %u) Requested(#%d: %u)\n",
   2951                drv_ctx.ip_buf.mincount, drv_ctx.ip_buf.buffer_size,
   2952                portDefn->nBufferCountActual, portDefn->nBufferSize);
   2953              eRet = OMX_ErrorBadParameter;
   2954          }
   2955       }
   2956       else if (portDefn->eDir ==  OMX_DirMax)
   2957       {
   2958           DEBUG_PRINT_ERROR(" Set_parameter: Bad Port idx %d",
   2959                       (int)portDefn->nPortIndex);
   2960           eRet = OMX_ErrorBadPortIndex;
   2961       }
   2962     }
   2963     break;
   2964     case OMX_IndexParamVideoPortFormat:
   2965     {
   2966       OMX_VIDEO_PARAM_PORTFORMATTYPE *portFmt =
   2967                      (OMX_VIDEO_PARAM_PORTFORMATTYPE *)paramData;
   2968       int ret=0;
   2969       struct v4l2_format fmt;
   2970       DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamVideoPortFormat %d\n",
   2971               portFmt->eColorFormat);
   2972 
   2973       if(1 == portFmt->nPortIndex)
   2974       {
   2975 	fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   2976 	fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
   2977 	fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
   2978 	fmt.fmt.pix_mp.pixelformat = capture_capability;
   2979 	enum vdec_output_fromat op_format;
   2980 	if(portFmt->eColorFormat == OMX_COLOR_FormatYUV420SemiPlanar)
   2981 	  op_format = VDEC_YUV_FORMAT_NV12;
   2982 	else if(portFmt->eColorFormat ==
   2983 		QOMX_COLOR_FormatYUV420PackedSemiPlanar64x32Tile2m8ka)
   2984 	  op_format = VDEC_YUV_FORMAT_TILE_4x2;
   2985          else
   2986 	   eRet = OMX_ErrorBadParameter;
   2987 
   2988          if(eRet == OMX_ErrorNone)
   2989          {
   2990 	   drv_ctx.output_format = op_format;
   2991 	   ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
   2992 	   if(ret)
   2993            {
   2994              DEBUG_PRINT_ERROR("\n Set output format failed");
   2995              eRet = OMX_ErrorUnsupportedSetting;
   2996 			/*TODO: How to handle this case */
   2997            }
   2998            else
   2999 	     {
   3000 	       eRet = get_buffer_req(&drv_ctx.op_buf);
   3001 	     }
   3002 	 }
   3003       }
   3004     }
   3005     break;
   3006 
   3007     case OMX_QcomIndexPortDefn:
   3008     {
   3009         OMX_QCOM_PARAM_PORTDEFINITIONTYPE *portFmt =
   3010             (OMX_QCOM_PARAM_PORTDEFINITIONTYPE *) paramData;
   3011         DEBUG_PRINT_LOW("set_parameter: OMX_IndexQcomParamPortDefinitionType %d\n",
   3012             portFmt->nFramePackingFormat);
   3013 
   3014         /* Input port */
   3015         if (portFmt->nPortIndex == 0)
   3016         {
   3017             if (portFmt->nFramePackingFormat == OMX_QCOM_FramePacking_Arbitrary)
   3018             {
   3019               if(secure_mode) {
   3020                 arbitrary_bytes = false;
   3021                 DEBUG_PRINT_ERROR("setparameter: cannot set to arbitary bytes mode in secure session");
   3022                 eRet = OMX_ErrorUnsupportedSetting;
   3023               } else {
   3024                arbitrary_bytes = true;
   3025               }
   3026             }
   3027             else if (portFmt->nFramePackingFormat ==
   3028                 OMX_QCOM_FramePacking_OnlyOneCompleteFrame)
   3029             {
   3030                arbitrary_bytes = false;
   3031             }
   3032             else
   3033             {
   3034                 DEBUG_PRINT_ERROR("Setparameter: unknown FramePacking format %d\n",
   3035                     portFmt->nFramePackingFormat);
   3036                 eRet = OMX_ErrorUnsupportedSetting;
   3037             }
   3038         }
   3039         else if (portFmt->nPortIndex == OMX_CORE_OUTPUT_PORT_INDEX)
   3040         {
   3041           DEBUG_PRINT_HIGH("set_parameter: OMX_IndexQcomParamPortDefinitionType OP Port\n");
   3042           if( (portFmt->nMemRegion > OMX_QCOM_MemRegionInvalid &&
   3043                portFmt->nMemRegion < OMX_QCOM_MemRegionMax) &&
   3044               portFmt->nCacheAttr == OMX_QCOM_CacheAttrNone)
   3045           {
   3046             m_out_mem_region_smi = OMX_TRUE;
   3047             if ((m_out_mem_region_smi && m_out_pvt_entry_pmem))
   3048             {
   3049               DEBUG_PRINT_HIGH("set_parameter: OMX_IndexQcomParamPortDefinitionType OP Port: out pmem set\n");
   3050               m_use_output_pmem = OMX_TRUE;
   3051             }
   3052           }
   3053         }
   3054     }
   3055     break;
   3056 
   3057      case OMX_IndexParamStandardComponentRole:
   3058      {
   3059           OMX_PARAM_COMPONENTROLETYPE *comp_role;
   3060           comp_role = (OMX_PARAM_COMPONENTROLETYPE *) paramData;
   3061           DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamStandardComponentRole %s\n",
   3062                        comp_role->cRole);
   3063 
   3064           if((m_state == OMX_StateLoaded)&&
   3065               !BITMASK_PRESENT(&m_flags,OMX_COMPONENT_IDLE_PENDING))
   3066           {
   3067            DEBUG_PRINT_LOW("Set Parameter called in valid state");
   3068           }
   3069           else
   3070           {
   3071              DEBUG_PRINT_ERROR("Set Parameter called in Invalid State\n");
   3072              return OMX_ErrorIncorrectStateOperation;
   3073           }
   3074 
   3075           if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc",OMX_MAX_STRINGNAME_SIZE))
   3076           {
   3077               if(!strncmp((char*)comp_role->cRole,"video_decoder.avc",OMX_MAX_STRINGNAME_SIZE))
   3078               {
   3079                   strlcpy((char*)m_cRole,"video_decoder.avc",OMX_MAX_STRINGNAME_SIZE);
   3080               }
   3081               else
   3082               {
   3083                   DEBUG_PRINT_ERROR("Setparameter: unknown Index %s\n", comp_role->cRole);
   3084                   eRet =OMX_ErrorUnsupportedSetting;
   3085               }
   3086           }
   3087           else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg4",OMX_MAX_STRINGNAME_SIZE))
   3088           {
   3089               if(!strncmp((const char*)comp_role->cRole,"video_decoder.mpeg4",OMX_MAX_STRINGNAME_SIZE))
   3090               {
   3091                   strlcpy((char*)m_cRole,"video_decoder.mpeg4",OMX_MAX_STRINGNAME_SIZE);
   3092               }
   3093               else
   3094               {
   3095                   DEBUG_PRINT_ERROR("Setparameter: unknown Index %s\n", comp_role->cRole);
   3096                   eRet = OMX_ErrorUnsupportedSetting;
   3097               }
   3098           }
   3099           else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.h263",OMX_MAX_STRINGNAME_SIZE))
   3100           {
   3101               if(!strncmp((const char*)comp_role->cRole,"video_decoder.h263",OMX_MAX_STRINGNAME_SIZE))
   3102               {
   3103                   strlcpy((char*)m_cRole,"video_decoder.h263",OMX_MAX_STRINGNAME_SIZE);
   3104               }
   3105               else
   3106               {
   3107                   DEBUG_PRINT_ERROR("Setparameter: unknown Index %s\n", comp_role->cRole);
   3108                   eRet =OMX_ErrorUnsupportedSetting;
   3109               }
   3110           }
   3111           else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg2",OMX_MAX_STRINGNAME_SIZE))
   3112           {
   3113             if(!strncmp((const char*)comp_role->cRole,"video_decoder.mpeg2",OMX_MAX_STRINGNAME_SIZE))
   3114             {
   3115               strlcpy((char*)m_cRole,"video_decoder.mpeg2",OMX_MAX_STRINGNAME_SIZE);
   3116             }
   3117             else
   3118             {
   3119               DEBUG_PRINT_ERROR("Setparameter: unknown Index %s\n", comp_role->cRole);
   3120               eRet = OMX_ErrorUnsupportedSetting;
   3121             }
   3122           }
   3123           else if((!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx",OMX_MAX_STRINGNAME_SIZE)) ||
   3124                   (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx311",OMX_MAX_STRINGNAME_SIZE))
   3125                   )
   3126           {
   3127               if(!strncmp((const char*)comp_role->cRole,"video_decoder.divx",OMX_MAX_STRINGNAME_SIZE))
   3128               {
   3129                   strlcpy((char*)m_cRole,"video_decoder.divx",OMX_MAX_STRINGNAME_SIZE);
   3130               }
   3131               else
   3132               {
   3133                   DEBUG_PRINT_ERROR("Setparameter: unknown Index %s\n", comp_role->cRole);
   3134                   eRet =OMX_ErrorUnsupportedSetting;
   3135               }
   3136           }
   3137           else if ( (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vc1",OMX_MAX_STRINGNAME_SIZE)) ||
   3138                     (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.wmv",OMX_MAX_STRINGNAME_SIZE))
   3139                     )
   3140           {
   3141               if(!strncmp((const char*)comp_role->cRole,"video_decoder.vc1",OMX_MAX_STRINGNAME_SIZE))
   3142               {
   3143                   strlcpy((char*)m_cRole,"video_decoder.vc1",OMX_MAX_STRINGNAME_SIZE);
   3144               }
   3145               else
   3146               {
   3147                   DEBUG_PRINT_ERROR("Setparameter: unknown Index %s\n", comp_role->cRole);
   3148                   eRet =OMX_ErrorUnsupportedSetting;
   3149               }
   3150           }
   3151           else
   3152           {
   3153                DEBUG_PRINT_ERROR("Setparameter: unknown param %s\n", drv_ctx.kind);
   3154                eRet = OMX_ErrorInvalidComponentName;
   3155           }
   3156           break;
   3157      }
   3158 
   3159     case OMX_IndexParamPriorityMgmt:
   3160         {
   3161             if(m_state != OMX_StateLoaded)
   3162             {
   3163                DEBUG_PRINT_ERROR("Set Parameter called in Invalid State\n");
   3164                return OMX_ErrorIncorrectStateOperation;
   3165             }
   3166             OMX_PRIORITYMGMTTYPE *priorityMgmtype = (OMX_PRIORITYMGMTTYPE*) paramData;
   3167             DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamPriorityMgmt %d\n",
   3168               priorityMgmtype->nGroupID);
   3169 
   3170             DEBUG_PRINT_LOW("set_parameter: priorityMgmtype %d\n",
   3171              priorityMgmtype->nGroupPriority);
   3172 
   3173             m_priority_mgm.nGroupID = priorityMgmtype->nGroupID;
   3174             m_priority_mgm.nGroupPriority = priorityMgmtype->nGroupPriority;
   3175 
   3176             break;
   3177         }
   3178 
   3179       case OMX_IndexParamCompBufferSupplier:
   3180       {
   3181           OMX_PARAM_BUFFERSUPPLIERTYPE *bufferSupplierType = (OMX_PARAM_BUFFERSUPPLIERTYPE*) paramData;
   3182             DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamCompBufferSupplier %d\n",
   3183                 bufferSupplierType->eBufferSupplier);
   3184              if(bufferSupplierType->nPortIndex == 0 || bufferSupplierType->nPortIndex ==1)
   3185                 m_buffer_supplier.eBufferSupplier = bufferSupplierType->eBufferSupplier;
   3186 
   3187              else
   3188 
   3189              eRet = OMX_ErrorBadPortIndex;
   3190 
   3191           break;
   3192 
   3193       }
   3194       case OMX_IndexParamVideoAvc:
   3195           {
   3196               DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamVideoAvc %d\n",
   3197                     paramIndex);
   3198               break;
   3199           }
   3200       case OMX_IndexParamVideoH263:
   3201           {
   3202               DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamVideoH263 %d\n",
   3203                     paramIndex);
   3204               break;
   3205           }
   3206       case OMX_IndexParamVideoMpeg4:
   3207           {
   3208               DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamVideoMpeg4 %d\n",
   3209                     paramIndex);
   3210               break;
   3211           }
   3212       case OMX_IndexParamVideoMpeg2:
   3213           {
   3214               DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamVideoMpeg2 %d\n",
   3215                     paramIndex);
   3216               break;
   3217           }
   3218        case OMX_QcomIndexParamVideoDecoderPictureOrder:
   3219           {
   3220               QOMX_VIDEO_DECODER_PICTURE_ORDER *pictureOrder =
   3221                   (QOMX_VIDEO_DECODER_PICTURE_ORDER *)paramData;
   3222               enum vdec_output_order pic_order = VDEC_ORDER_DISPLAY;
   3223               DEBUG_PRINT_HIGH("set_parameter: OMX_QcomIndexParamVideoDecoderPictureOrder %d\n",
   3224                     pictureOrder->eOutputPictureOrder);
   3225               if (pictureOrder->eOutputPictureOrder == QOMX_VIDEO_DISPLAY_ORDER)
   3226                   pic_order = VDEC_ORDER_DISPLAY;
   3227               else if (pictureOrder->eOutputPictureOrder == QOMX_VIDEO_DECODE_ORDER){
   3228                   pic_order = VDEC_ORDER_DECODE;
   3229                   time_stamp_dts.set_timestamp_reorder_mode(false);
   3230               }
   3231               else
   3232                   eRet = OMX_ErrorBadParameter;
   3233 
   3234               if (eRet == OMX_ErrorNone && pic_order != drv_ctx.picture_order)
   3235               {
   3236                   drv_ctx.picture_order = pic_order;
   3237 		  // ioctl_msg.in = &drv_ctx.picture_order;
   3238                   //ioctl_msg.out = NULL;
   3239                   if (/*ioctl(drv_ctx.video_driver_fd, VDEC_IOCTL_SET_PICTURE_ORDER,
   3240                       (void*)&ioctl_msg) < */0)
   3241                   {
   3242                       DEBUG_PRINT_ERROR("\n Set picture order failed");
   3243                       eRet = OMX_ErrorUnsupportedSetting;
   3244                   }
   3245               }
   3246               break;
   3247           }
   3248     case OMX_QcomIndexParamConcealMBMapExtraData:
   3249       if(!secure_mode)
   3250           eRet = enable_extradata(VDEC_EXTRADATA_MB_ERROR_MAP,
   3251                                   ((QOMX_ENABLETYPE *)paramData)->bEnable);
   3252       else {
   3253           DEBUG_PRINT_ERROR("\n secure mode setting not supported");
   3254           eRet = OMX_ErrorUnsupportedSetting;
   3255       }
   3256       break;
   3257     case OMX_QcomIndexParamFrameInfoExtraData:
   3258       {
   3259         if(!secure_mode)
   3260             eRet = enable_extradata(OMX_FRAMEINFO_EXTRADATA,
   3261                                 ((QOMX_ENABLETYPE *)paramData)->bEnable);
   3262         else {
   3263             DEBUG_PRINT_ERROR("\n secure mode setting not supported");
   3264             eRet = OMX_ErrorUnsupportedSetting;
   3265         }
   3266        break;
   3267       }
   3268     case OMX_QcomIndexParamInterlaceExtraData:
   3269       if(!secure_mode)
   3270           eRet = enable_extradata(OMX_INTERLACE_EXTRADATA,
   3271                               ((QOMX_ENABLETYPE *)paramData)->bEnable);
   3272       else {
   3273           DEBUG_PRINT_ERROR("\n secure mode setting not supported");
   3274           eRet = OMX_ErrorUnsupportedSetting;
   3275       }
   3276       break;
   3277     case OMX_QcomIndexParamH264TimeInfo:
   3278       if(!secure_mode)
   3279           eRet = enable_extradata(OMX_TIMEINFO_EXTRADATA,
   3280                               ((QOMX_ENABLETYPE *)paramData)->bEnable);
   3281       else {
   3282           DEBUG_PRINT_ERROR("\n secure mode setting not supported");
   3283           eRet = OMX_ErrorUnsupportedSetting;
   3284       }
   3285       break;
   3286     case OMX_QcomIndexParamVideoDivx:
   3287       {
   3288         QOMX_VIDEO_PARAM_DIVXTYPE* divXType = (QOMX_VIDEO_PARAM_DIVXTYPE *) paramData;
   3289 
   3290 #if 0
   3291          createDivxDrmContext( divXType->pDrmHandle );
   3292 #endif
   3293       }
   3294       break;
   3295     case OMX_QcomIndexPlatformPvt:
   3296       {
   3297         DEBUG_PRINT_HIGH("set_parameter: OMX_QcomIndexPlatformPvt OP Port\n");
   3298         OMX_QCOM_PLATFORMPRIVATE_EXTN* entryType = (OMX_QCOM_PLATFORMPRIVATE_EXTN *) paramData;
   3299         if (entryType->type != OMX_QCOM_PLATFORM_PRIVATE_PMEM)
   3300         {
   3301           DEBUG_PRINT_HIGH("set_parameter: Platform Private entry type (%d) not supported.", entryType->type);
   3302           eRet = OMX_ErrorUnsupportedSetting;
   3303         }
   3304         else
   3305         {
   3306           m_out_pvt_entry_pmem = OMX_TRUE;
   3307           if ((m_out_mem_region_smi && m_out_pvt_entry_pmem))
   3308           {
   3309             DEBUG_PRINT_HIGH("set_parameter: OMX_QcomIndexPlatformPvt OP Port: out pmem set\n");
   3310             m_use_output_pmem = OMX_TRUE;
   3311           }
   3312         }
   3313 
   3314       }
   3315       break;
   3316     case OMX_QcomIndexParamVideoSyncFrameDecodingMode:
   3317       {
   3318           DEBUG_PRINT_HIGH("set_parameter: OMX_QcomIndexParamVideoSyncFrameDecodingMode");
   3319           DEBUG_PRINT_HIGH("set idr only decoding for thumbnail mode");
   3320           drv_ctx.idr_only_decoding = 1;
   3321           int rc; //= ioctl(drv_ctx.video_driver_fd,
   3322                //       VDEC_IOCTL_SET_IDR_ONLY_DECODING);
   3323           if(rc < 0) {
   3324               DEBUG_PRINT_ERROR("Failed to set IDR only decoding on driver.");
   3325               eRet = OMX_ErrorHardware;
   3326           }
   3327       }
   3328       break;
   3329 
   3330     case OMX_QcomIndexParamIndexExtraDataType:
   3331       {
   3332         if(!secure_mode) {
   3333             QOMX_INDEXEXTRADATATYPE *extradataIndexType = (QOMX_INDEXEXTRADATATYPE *) paramData;
   3334             if ((extradataIndexType->nIndex == OMX_IndexParamPortDefinition) &&
   3335                    (extradataIndexType->bEnabled == OMX_TRUE) &&
   3336                    (extradataIndexType->nPortIndex == 1))
   3337             {
   3338               DEBUG_PRINT_HIGH("set_parameter:  OMX_QcomIndexParamIndexExtraDataType SmoothStreaming\n");
   3339               eRet = enable_extradata(OMX_PORTDEF_EXTRADATA, extradataIndexType->bEnabled);
   3340               // Set smooth streaming parameter
   3341               int rc;// = ioctl(drv_ctx.video_driver_fd,
   3342                    //         VDEC_IOCTL_SET_CONT_ON_RECONFIG);
   3343               if(rc < 0) {
   3344                   DEBUG_PRINT_ERROR("Failed to enable Smooth Streaming on driver.");
   3345                   eRet = OMX_ErrorHardware;
   3346               }
   3347             }
   3348          }
   3349        }
   3350       break;
   3351 
   3352 #if defined (_ANDROID_HONEYCOMB_) || defined (_ANDROID_ICS_)
   3353       /* Need to allow following two set_parameters even in Idle
   3354        * state. This is ANDROID architecture which is not in sync
   3355        * with openmax standard. */
   3356     case OMX_GoogleAndroidIndexEnableAndroidNativeBuffers:
   3357       {
   3358           EnableAndroidNativeBuffersParams* enableNativeBuffers = (EnableAndroidNativeBuffersParams *) paramData;
   3359           if(enableNativeBuffers) {
   3360               m_enable_android_native_buffers = enableNativeBuffers->enable;
   3361           }
   3362       }
   3363       break;
   3364     case OMX_GoogleAndroidIndexUseAndroidNativeBuffer:
   3365       {
   3366           eRet = use_android_native_buffer(hComp, paramData);
   3367       }
   3368       break;
   3369 #endif
   3370     case OMX_QcomIndexParamEnableTimeStampReorder:
   3371       {
   3372         QOMX_INDEXTIMESTAMPREORDER *reorder = (QOMX_INDEXTIMESTAMPREORDER *)paramData;
   3373         if (drv_ctx.picture_order == QOMX_VIDEO_DISPLAY_ORDER) {
   3374           if (reorder->bEnable == OMX_TRUE) {
   3375               frm_int =0;
   3376               time_stamp_dts.set_timestamp_reorder_mode(true);
   3377           }
   3378           else
   3379             time_stamp_dts.set_timestamp_reorder_mode(false);
   3380         } else {
   3381           time_stamp_dts.set_timestamp_reorder_mode(false);
   3382           if (reorder->bEnable == OMX_TRUE)
   3383           {
   3384             eRet = OMX_ErrorUnsupportedSetting;
   3385           }
   3386         }
   3387       }
   3388       break;
   3389     default:
   3390     {
   3391       DEBUG_PRINT_ERROR("Setparameter: unknown param %d\n", paramIndex);
   3392       eRet = OMX_ErrorUnsupportedIndex;
   3393     }
   3394   }
   3395   return eRet;
   3396 }
   3397 
   3398 /* ======================================================================
   3399 FUNCTION
   3400   omx_vdec::GetConfig
   3401 
   3402 DESCRIPTION
   3403   OMX Get Config Method implementation.
   3404 
   3405 PARAMETERS
   3406   <TBD>.
   3407 
   3408 RETURN VALUE
   3409   OMX Error None if successful.
   3410 
   3411 ========================================================================== */
   3412 OMX_ERRORTYPE  omx_vdec::get_config(OMX_IN OMX_HANDLETYPE      hComp,
   3413                                         OMX_IN OMX_INDEXTYPE configIndex,
   3414                                         OMX_INOUT OMX_PTR     configData)
   3415 {
   3416   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   3417 
   3418   if (m_state == OMX_StateInvalid)
   3419   {
   3420      DEBUG_PRINT_ERROR("Get Config in Invalid State\n");
   3421      return OMX_ErrorInvalidState;
   3422   }
   3423 
   3424   switch (configIndex)
   3425   {
   3426     case OMX_QcomIndexConfigInterlaced:
   3427     {
   3428       OMX_QCOM_CONFIG_INTERLACETYPE *configFmt =
   3429                                    (OMX_QCOM_CONFIG_INTERLACETYPE *) configData;
   3430       if (configFmt->nPortIndex == 1)
   3431       {
   3432         if (configFmt->nIndex == 0)
   3433         {
   3434           configFmt->eInterlaceType = OMX_QCOM_InterlaceFrameProgressive;
   3435         }
   3436         else if (configFmt->nIndex == 1)
   3437         {
   3438           configFmt->eInterlaceType =
   3439                                   OMX_QCOM_InterlaceInterleaveFrameTopFieldFirst;
   3440         }
   3441         else if (configFmt->nIndex == 2)
   3442         {
   3443           configFmt->eInterlaceType =
   3444           OMX_QCOM_InterlaceInterleaveFrameBottomFieldFirst;
   3445         }
   3446         else
   3447         {
   3448           DEBUG_PRINT_ERROR("get_config: OMX_QcomIndexConfigInterlaced:"
   3449                             " NoMore Interlaced formats\n");
   3450           eRet = OMX_ErrorNoMore;
   3451         }
   3452 
   3453       }
   3454       else
   3455       {
   3456         DEBUG_PRINT_ERROR("get_config: Bad port index %d queried on only o/p port\n",
   3457         (int)configFmt->nPortIndex);
   3458         eRet = OMX_ErrorBadPortIndex;
   3459       }
   3460     break;
   3461     }
   3462     case OMX_QcomIndexQueryNumberOfVideoDecInstance:
   3463     {
   3464         struct vdec_ioctl_msg ioctl_msg = {NULL,NULL};
   3465         QOMX_VIDEO_QUERY_DECODER_INSTANCES *decoderinstances =
   3466           (QOMX_VIDEO_QUERY_DECODER_INSTANCES*)configData;
   3467         //ioctl_msg.out = (void*)&decoderinstances->nNumOfInstances;
   3468         //(void)(ioctl(drv_ctx.video_driver_fd,
   3469                //VDEC_IOCTL_GET_NUMBER_INSTANCES,&ioctl_msg));
   3470 
   3471 	decoderinstances->nNumOfInstances = 16;
   3472 	/*TODO: How to handle this case */
   3473     break;
   3474     }
   3475   case OMX_QcomIndexConfigVideoFramePackingArrangement:
   3476     {
   3477       if (drv_ctx.decoder_format == VDEC_CODECTYPE_H264)
   3478       {
   3479         OMX_QCOM_FRAME_PACK_ARRANGEMENT *configFmt =
   3480           (OMX_QCOM_FRAME_PACK_ARRANGEMENT *) configData;
   3481         h264_parser->get_frame_pack_data(configFmt);
   3482       }
   3483       else
   3484       {
   3485         DEBUG_PRINT_ERROR("get_config: Framepack data not supported for non H264 codecs");
   3486       }
   3487       break;
   3488     }
   3489     default:
   3490     {
   3491       DEBUG_PRINT_ERROR("get_config: unknown param %d\n",configIndex);
   3492       eRet = OMX_ErrorBadParameter;
   3493     }
   3494 
   3495   }
   3496 
   3497   return eRet;
   3498 }
   3499 
   3500 /* ======================================================================
   3501 FUNCTION
   3502   omx_vdec::SetConfig
   3503 
   3504 DESCRIPTION
   3505   OMX Set Config method implementation
   3506 
   3507 PARAMETERS
   3508   <TBD>.
   3509 
   3510 RETURN VALUE
   3511   OMX Error None if successful.
   3512 ========================================================================== */
   3513 OMX_ERRORTYPE  omx_vdec::set_config(OMX_IN OMX_HANDLETYPE      hComp,
   3514                                         OMX_IN OMX_INDEXTYPE configIndex,
   3515                                         OMX_IN OMX_PTR        configData)
   3516 {
   3517   if(m_state == OMX_StateInvalid)
   3518   {
   3519       DEBUG_PRINT_ERROR("Get Config in Invalid State\n");
   3520       return OMX_ErrorInvalidState;
   3521   }
   3522 
   3523   OMX_ERRORTYPE ret = OMX_ErrorNone;
   3524   OMX_VIDEO_CONFIG_NALSIZE *pNal;
   3525 
   3526   DEBUG_PRINT_LOW("\n Set Config Called");
   3527 
   3528   if (m_state == OMX_StateExecuting)
   3529   {
   3530      DEBUG_PRINT_ERROR("set_config:Ignore in Exe state\n");
   3531      return ret;
   3532   }
   3533 
   3534   if (configIndex == OMX_IndexVendorVideoExtraData)
   3535   {
   3536     OMX_VENDOR_EXTRADATATYPE *config = (OMX_VENDOR_EXTRADATATYPE *) configData;
   3537     DEBUG_PRINT_LOW("\n Index OMX_IndexVendorVideoExtraData called");
   3538     if (!strcmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc"))
   3539     {
   3540       DEBUG_PRINT_LOW("\n Index OMX_IndexVendorVideoExtraData AVC");
   3541       OMX_U32 extra_size;
   3542       // Parsing done here for the AVC atom is definitely not generic
   3543       // Currently this piece of code is working, but certainly
   3544       // not tested with all .mp4 files.
   3545       // Incase of failure, we might need to revisit this
   3546       // for a generic piece of code.
   3547 
   3548       // Retrieve size of NAL length field
   3549       // byte #4 contains the size of NAL lenght field
   3550       nal_length = (config->pData[4] & 0x03) + 1;
   3551 
   3552       extra_size = 0;
   3553       if (nal_length > 2)
   3554       {
   3555         /* Presently we assume that only one SPS and one PPS in AvC1 Atom */
   3556         extra_size = (nal_length - 2) * 2;
   3557       }
   3558 
   3559       // SPS starts from byte #6
   3560       OMX_U8 *pSrcBuf = (OMX_U8 *) (&config->pData[6]);
   3561       OMX_U8 *pDestBuf;
   3562       m_vendor_config.nPortIndex = config->nPortIndex;
   3563 
   3564       // minus 6 --> SPS starts from byte #6
   3565       // minus 1 --> picture param set byte to be ignored from avcatom
   3566       m_vendor_config.nDataSize = config->nDataSize - 6 - 1 + extra_size;
   3567       m_vendor_config.pData = (OMX_U8 *) malloc(m_vendor_config.nDataSize);
   3568       OMX_U32 len;
   3569       OMX_U8 index = 0;
   3570       // case where SPS+PPS is sent as part of set_config
   3571       pDestBuf = m_vendor_config.pData;
   3572 
   3573       DEBUG_PRINT_LOW("Rxd SPS+PPS nPortIndex[%d] len[%d] data[0x%x]\n",
   3574            m_vendor_config.nPortIndex,
   3575            m_vendor_config.nDataSize,
   3576            m_vendor_config.pData);
   3577       while (index < 2)
   3578       {
   3579         uint8 *psize;
   3580         len = *pSrcBuf;
   3581         len = len << 8;
   3582         len |= *(pSrcBuf + 1);
   3583         psize = (uint8 *) & len;
   3584         memcpy(pDestBuf + nal_length, pSrcBuf + 2,len);
   3585         for (int i = 0; i < nal_length; i++)
   3586         {
   3587           pDestBuf[i] = psize[nal_length - 1 - i];
   3588         }
   3589         //memcpy(pDestBuf,pSrcBuf,(len+2));
   3590         pDestBuf += len + nal_length;
   3591         pSrcBuf += len + 2;
   3592         index++;
   3593         pSrcBuf++;   // skip picture param set
   3594         len = 0;
   3595       }
   3596     }
   3597     else if (!strcmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg4") ||
   3598              !strcmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg2"))
   3599     {
   3600       m_vendor_config.nPortIndex = config->nPortIndex;
   3601       m_vendor_config.nDataSize = config->nDataSize;
   3602       m_vendor_config.pData = (OMX_U8 *) malloc((config->nDataSize));
   3603       memcpy(m_vendor_config.pData, config->pData,config->nDataSize);
   3604     }
   3605     else if (!strcmp(drv_ctx.kind, "OMX.qcom.video.decoder.vc1"))
   3606     {
   3607         if(m_vendor_config.pData)
   3608         {
   3609             free(m_vendor_config.pData);
   3610             m_vendor_config.pData = NULL;
   3611             m_vendor_config.nDataSize = 0;
   3612         }
   3613 
   3614         if (((*((OMX_U32 *) config->pData)) &
   3615              VC1_SP_MP_START_CODE_MASK) ==
   3616              VC1_SP_MP_START_CODE)
   3617         {
   3618             DEBUG_PRINT_LOW("set_config - VC1 simple/main profile\n");
   3619             m_vendor_config.nPortIndex = config->nPortIndex;
   3620             m_vendor_config.nDataSize = config->nDataSize;
   3621             m_vendor_config.pData =
   3622                 (OMX_U8 *) malloc(config->nDataSize);
   3623             memcpy(m_vendor_config.pData, config->pData,
   3624                    config->nDataSize);
   3625             m_vc1_profile = VC1_SP_MP_RCV;
   3626         }
   3627         else if (*((OMX_U32 *) config->pData) == VC1_AP_SEQ_START_CODE)
   3628         {
   3629             DEBUG_PRINT_LOW("set_config - VC1 Advance profile\n");
   3630             m_vendor_config.nPortIndex = config->nPortIndex;
   3631             m_vendor_config.nDataSize = config->nDataSize;
   3632             m_vendor_config.pData =
   3633                 (OMX_U8 *) malloc((config->nDataSize));
   3634             memcpy(m_vendor_config.pData, config->pData,
   3635                    config->nDataSize);
   3636             m_vc1_profile = VC1_AP;
   3637         }
   3638         else if ((config->nDataSize == VC1_STRUCT_C_LEN))
   3639         {
   3640             DEBUG_PRINT_LOW("set_config - VC1 Simple/Main profile struct C only\n");
   3641             m_vendor_config.nPortIndex = config->nPortIndex;
   3642             m_vendor_config.nDataSize  = config->nDataSize;
   3643             m_vendor_config.pData = (OMX_U8*)malloc(config->nDataSize);
   3644             memcpy(m_vendor_config.pData,config->pData,config->nDataSize);
   3645             m_vc1_profile = VC1_SP_MP_RCV;
   3646         }
   3647         else
   3648         {
   3649             DEBUG_PRINT_LOW("set_config - Error: Unknown VC1 profile\n");
   3650         }
   3651     }
   3652     return ret;
   3653   }
   3654   else if (configIndex == OMX_IndexConfigVideoNalSize)
   3655   {
   3656 
   3657     pNal = reinterpret_cast < OMX_VIDEO_CONFIG_NALSIZE * >(configData);
   3658     nal_length = pNal->nNaluBytes;
   3659     m_frame_parser.init_nal_length(nal_length);
   3660     DEBUG_PRINT_LOW("\n OMX_IndexConfigVideoNalSize called with Size %d",nal_length);
   3661     return ret;
   3662   }
   3663 
   3664   return OMX_ErrorNotImplemented;
   3665 }
   3666 
   3667 /* ======================================================================
   3668 FUNCTION
   3669   omx_vdec::GetExtensionIndex
   3670 
   3671 DESCRIPTION
   3672   OMX GetExtensionIndex method implementaion.  <TBD>
   3673 
   3674 PARAMETERS
   3675   <TBD>.
   3676 
   3677 RETURN VALUE
   3678   OMX Error None if everything successful.
   3679 
   3680 ========================================================================== */
   3681 OMX_ERRORTYPE  omx_vdec::get_extension_index(OMX_IN OMX_HANDLETYPE      hComp,
   3682                                                 OMX_IN OMX_STRING      paramName,
   3683                                                 OMX_OUT OMX_INDEXTYPE* indexType)
   3684 {
   3685     if(m_state == OMX_StateInvalid)
   3686     {
   3687         DEBUG_PRINT_ERROR("Get Extension Index in Invalid State\n");
   3688         return OMX_ErrorInvalidState;
   3689     }
   3690     else if (!strncmp(paramName, "OMX.QCOM.index.param.video.SyncFrameDecodingMode",sizeof("OMX.QCOM.index.param.video.SyncFrameDecodingMode") - 1)) {
   3691         *indexType = (OMX_INDEXTYPE)OMX_QcomIndexParamVideoSyncFrameDecodingMode;
   3692     }
   3693     else if (!strncmp(paramName, "OMX.QCOM.index.param.IndexExtraData",sizeof("OMX.QCOM.index.param.IndexExtraData") - 1))
   3694     {
   3695         *indexType = (OMX_INDEXTYPE)OMX_QcomIndexParamIndexExtraDataType;
   3696     }
   3697 #if defined (_ANDROID_HONEYCOMB_) || defined (_ANDROID_ICS_)
   3698     else if(!strncmp(paramName,"OMX.google.android.index.enableAndroidNativeBuffers", sizeof("OMX.google.android.index.enableAndroidNativeBuffers") - 1)) {
   3699         *indexType = (OMX_INDEXTYPE)OMX_GoogleAndroidIndexEnableAndroidNativeBuffers;
   3700     }
   3701     else if(!strncmp(paramName,"OMX.google.android.index.useAndroidNativeBuffer2", sizeof("OMX.google.android.index.enableAndroidNativeBuffer2") - 1)) {
   3702         *indexType = (OMX_INDEXTYPE)OMX_GoogleAndroidIndexUseAndroidNativeBuffer2;
   3703     }
   3704     else if(!strncmp(paramName,"OMX.google.android.index.useAndroidNativeBuffer", sizeof("OMX.google.android.index.enableAndroidNativeBuffer") - 1)) {
   3705         DEBUG_PRINT_ERROR("Extension: %s is supported\n", paramName);
   3706         *indexType = (OMX_INDEXTYPE)OMX_GoogleAndroidIndexUseAndroidNativeBuffer;
   3707     }
   3708     else if(!strncmp(paramName,"OMX.google.android.index.getAndroidNativeBufferUsage", sizeof("OMX.google.android.index.getAndroidNativeBufferUsage") - 1)) {
   3709         *indexType = (OMX_INDEXTYPE)OMX_GoogleAndroidIndexGetAndroidNativeBufferUsage;
   3710     }
   3711 #endif
   3712 	else {
   3713         DEBUG_PRINT_ERROR("Extension: %s not implemented\n", paramName);
   3714         return OMX_ErrorNotImplemented;
   3715     }
   3716     return OMX_ErrorNone;
   3717 }
   3718 
   3719 /* ======================================================================
   3720 FUNCTION
   3721   omx_vdec::GetState
   3722 
   3723 DESCRIPTION
   3724   Returns the state information back to the caller.<TBD>
   3725 
   3726 PARAMETERS
   3727   <TBD>.
   3728 
   3729 RETURN VALUE
   3730   Error None if everything is successful.
   3731 ========================================================================== */
   3732 OMX_ERRORTYPE  omx_vdec::get_state(OMX_IN OMX_HANDLETYPE  hComp,
   3733                                        OMX_OUT OMX_STATETYPE* state)
   3734 {
   3735   *state = m_state;
   3736   DEBUG_PRINT_LOW("get_state: Returning the state %d\n",*state);
   3737   return OMX_ErrorNone;
   3738 }
   3739 
   3740 /* ======================================================================
   3741 FUNCTION
   3742   omx_vdec::ComponentTunnelRequest
   3743 
   3744 DESCRIPTION
   3745   OMX Component Tunnel Request method implementation. <TBD>
   3746 
   3747 PARAMETERS
   3748   None.
   3749 
   3750 RETURN VALUE
   3751   OMX Error None if everything successful.
   3752 
   3753 ========================================================================== */
   3754 OMX_ERRORTYPE  omx_vdec::component_tunnel_request(OMX_IN OMX_HANDLETYPE                hComp,
   3755                                                      OMX_IN OMX_U32                        port,
   3756                                                      OMX_IN OMX_HANDLETYPE        peerComponent,
   3757                                                      OMX_IN OMX_U32                    peerPort,
   3758                                                      OMX_INOUT OMX_TUNNELSETUPTYPE* tunnelSetup)
   3759 {
   3760   DEBUG_PRINT_ERROR("Error: component_tunnel_request Not Implemented\n");
   3761   return OMX_ErrorNotImplemented;
   3762 }
   3763 
   3764 /* ======================================================================
   3765 FUNCTION
   3766   omx_vdec::UseOutputBuffer
   3767 
   3768 DESCRIPTION
   3769   Helper function for Use buffer in the input pin
   3770 
   3771 PARAMETERS
   3772   None.
   3773 
   3774 RETURN VALUE
   3775   true/false
   3776 
   3777 ========================================================================== */
   3778 OMX_ERRORTYPE  omx_vdec::use_output_buffer(
   3779                          OMX_IN OMX_HANDLETYPE            hComp,
   3780                          OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
   3781                          OMX_IN OMX_U32                   port,
   3782                          OMX_IN OMX_PTR                   appData,
   3783                          OMX_IN OMX_U32                   bytes,
   3784                          OMX_IN OMX_U8*                   buffer)
   3785 {
   3786   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   3787   OMX_BUFFERHEADERTYPE       *bufHdr= NULL; // buffer header
   3788   unsigned                         i= 0; // Temporary counter
   3789   struct vdec_ioctl_msg ioctl_msg = {NULL,NULL};
   3790   struct vdec_setbuffer_cmd setbuffers;
   3791   OMX_PTR privateAppData = NULL;
   3792   private_handle_t *handle = NULL;
   3793   OMX_U8 *buff = buffer;
   3794   if (!m_out_mem_ptr) {
   3795     DEBUG_PRINT_HIGH("Use_op_buf:Allocating output headers");
   3796     eRet = allocate_output_headers();
   3797   }
   3798 
   3799   if (eRet == OMX_ErrorNone) {
   3800     for(i=0; i< drv_ctx.op_buf.actualcount; i++) {
   3801       if(BITMASK_ABSENT(&m_out_bm_count,i))
   3802       {
   3803         break;
   3804       }
   3805     }
   3806   }
   3807 
   3808   if(i >= drv_ctx.op_buf.actualcount) {
   3809     eRet = OMX_ErrorInsufficientResources;
   3810   }
   3811 
   3812   if (eRet == OMX_ErrorNone) {
   3813 #if defined(_ANDROID_HONEYCOMB_) || defined(_ANDROID_ICS_)
   3814     if(m_enable_android_native_buffers) {
   3815         if(m_use_android_native_buffers) {
   3816            UseAndroidNativeBufferParams *params = (UseAndroidNativeBufferParams *)appData;
   3817            sp<android_native_buffer_t> nBuf = params->nativeBuffer;
   3818            handle = (private_handle_t *)nBuf->handle;
   3819            privateAppData = params->pAppPrivate;
   3820         }
   3821         else {
   3822            handle = (private_handle_t *)buff;
   3823            if(!secure_mode) {
   3824 	       buff =  (OMX_U8*)mmap(0, handle->size,
   3825                              PROT_READ|PROT_WRITE, MAP_SHARED, handle->fd, 0);
   3826                if (buff == MAP_FAILED) {
   3827                    DEBUG_PRINT_ERROR("Failed to mmap pmem with fd = %d, size = %d", handle->fd, handle->size);
   3828                    return OMX_ErrorInsufficientResources;
   3829                }
   3830 	    }
   3831            privateAppData = appData;
   3832         }
   3833         if(!handle) {
   3834             DEBUG_PRINT_ERROR("Native Buffer handle is NULL");
   3835             return OMX_ErrorBadParameter;
   3836         }
   3837         drv_ctx.ptr_outputbuffer[i].pmem_fd = handle->fd;
   3838         drv_ctx.ptr_outputbuffer[i].offset = 0;
   3839         drv_ctx.ptr_outputbuffer[i].bufferaddr = buff;
   3840         drv_ctx.ptr_outputbuffer[i].mmaped_size =
   3841             drv_ctx.ptr_outputbuffer[i].buffer_len = drv_ctx.op_buf.buffer_size;
   3842     } else
   3843 #endif
   3844 
   3845     if (!ouput_egl_buffers && !m_use_output_pmem) {
   3846 #ifdef USE_ION
   3847         drv_ctx.op_buf_ion_info[i].ion_device_fd = alloc_map_ion_memory(
   3848                 drv_ctx.op_buf.buffer_size,drv_ctx.op_buf.alignment,
   3849                 &drv_ctx.op_buf_ion_info[i].ion_alloc_data,
   3850                 &drv_ctx.op_buf_ion_info[i].fd_ion_data, 0);
   3851         if(drv_ctx.op_buf_ion_info[i].ion_device_fd < 0) {
   3852           return OMX_ErrorInsufficientResources;
   3853         }
   3854         drv_ctx.ptr_outputbuffer[i].pmem_fd = \
   3855           drv_ctx.op_buf_ion_info[i].fd_ion_data.fd;
   3856 #else
   3857         drv_ctx.ptr_outputbuffer[i].pmem_fd = \
   3858           open (MEM_DEVICE,O_RDWR);
   3859 
   3860         if (drv_ctx.ptr_outputbuffer[i].pmem_fd < 0) {
   3861           return OMX_ErrorInsufficientResources;
   3862         }
   3863 
   3864         if(drv_ctx.ptr_outputbuffer[i].pmem_fd == 0)
   3865         {
   3866           drv_ctx.ptr_outputbuffer[i].pmem_fd = \
   3867             open (MEM_DEVICE,O_RDWR);
   3868           if (drv_ctx.ptr_outputbuffer[i].pmem_fd < 0) {
   3869             return OMX_ErrorInsufficientResources;
   3870           }
   3871         }
   3872 
   3873         if(!align_pmem_buffers(drv_ctx.ptr_outputbuffer[i].pmem_fd,
   3874           drv_ctx.op_buf.buffer_size,
   3875           drv_ctx.op_buf.alignment))
   3876         {
   3877           DEBUG_PRINT_ERROR("\n align_pmem_buffers() failed");
   3878           close(drv_ctx.ptr_outputbuffer[i].pmem_fd);
   3879           return OMX_ErrorInsufficientResources;
   3880         }
   3881 #endif
   3882         if(!secure_mode) {
   3883             drv_ctx.ptr_outputbuffer[i].bufferaddr =
   3884               (unsigned char *)mmap(NULL, drv_ctx.op_buf.buffer_size,
   3885               PROT_READ|PROT_WRITE, MAP_SHARED,
   3886               drv_ctx.ptr_outputbuffer[i].pmem_fd,0);
   3887             if (drv_ctx.ptr_outputbuffer[i].bufferaddr == MAP_FAILED) {
   3888                 close(drv_ctx.ptr_outputbuffer[i].pmem_fd);
   3889 #ifdef USE_ION
   3890                 free_ion_memory(&drv_ctx.op_buf_ion_info[i]);
   3891 #endif
   3892               return OMX_ErrorInsufficientResources;
   3893             }
   3894         }
   3895         drv_ctx.ptr_outputbuffer[i].offset = 0;
   3896         privateAppData = appData;
   3897      }
   3898      else {
   3899 
   3900        DEBUG_PRINT_LOW("Use_op_buf: out_pmem=%d",m_use_output_pmem);
   3901         if (!appData || !bytes ) {
   3902           if(!secure_mode && !buffer) {
   3903               DEBUG_PRINT_ERROR("\n Bad parameters for use buffer in EGL image case");
   3904               return OMX_ErrorBadParameter;
   3905           }
   3906         }
   3907 
   3908         OMX_QCOM_PLATFORM_PRIVATE_LIST *pmem_list;
   3909         OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *pmem_info;
   3910         pmem_list = (OMX_QCOM_PLATFORM_PRIVATE_LIST*) appData;
   3911         if (!pmem_list->entryList || !pmem_list->entryList->entry ||
   3912             !pmem_list->nEntries ||
   3913             pmem_list->entryList->type != OMX_QCOM_PLATFORM_PRIVATE_PMEM) {
   3914           DEBUG_PRINT_ERROR("\n Pmem info not valid in use buffer");
   3915           return OMX_ErrorBadParameter;
   3916         }
   3917         pmem_info = (OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *)
   3918                     pmem_list->entryList->entry;
   3919         DEBUG_PRINT_LOW("vdec: use buf: pmem_fd=0x%x",
   3920                           pmem_info->pmem_fd);
   3921         drv_ctx.ptr_outputbuffer[i].pmem_fd = pmem_info->pmem_fd;
   3922         drv_ctx.ptr_outputbuffer[i].offset = pmem_info->offset;
   3923         drv_ctx.ptr_outputbuffer[i].bufferaddr = buff;
   3924         drv_ctx.ptr_outputbuffer[i].mmaped_size =
   3925         drv_ctx.ptr_outputbuffer[i].buffer_len = drv_ctx.op_buf.buffer_size;
   3926         privateAppData = appData;
   3927      }
   3928      m_pmem_info[i].offset = drv_ctx.ptr_outputbuffer[i].offset;
   3929      m_pmem_info[i].pmem_fd = drv_ctx.ptr_outputbuffer[i].pmem_fd;
   3930 
   3931      *bufferHdr = (m_out_mem_ptr + i );
   3932      if(secure_mode)
   3933           drv_ctx.ptr_outputbuffer[i].bufferaddr = *bufferHdr;
   3934      //setbuffers.buffer_type = VDEC_BUFFER_TYPE_OUTPUT;
   3935      memcpy (&setbuffers.buffer,&drv_ctx.ptr_outputbuffer[i],
   3936              sizeof (vdec_bufferpayload));
   3937 
   3938      //     ioctl_msg.in  = &setbuffers;
   3939      //     ioctl_msg.out = NULL;
   3940 
   3941      DEBUG_PRINT_HIGH("\n Set the Output Buffer Idx: %d Addr: %x, pmem_fd=%0x%x", i,
   3942                        drv_ctx.ptr_outputbuffer[i],drv_ctx.ptr_outputbuffer[i].pmem_fd );
   3943      // if (ioctl (drv_ctx.video_driver_fd,VDEC_IOCTL_SET_BUFFER,
   3944      //      &ioctl_msg) < 0)
   3945      //  {
   3946      //  DEBUG_PRINT_ERROR("\n Set output buffer failed");
   3947      //   return OMX_ErrorInsufficientResources;
   3948      //  }
   3949      // found an empty buffer at i
   3950      (*bufferHdr)->nAllocLen = drv_ctx.op_buf.buffer_size;
   3951      (*bufferHdr)->pBuffer = buff;
   3952      (*bufferHdr)->pAppPrivate = privateAppData;
   3953      BITMASK_SET(&m_out_bm_count,i);
   3954   }
   3955   return eRet;
   3956 }
   3957 
   3958 /* ======================================================================
   3959 FUNCTION
   3960   omx_vdec::use_input_heap_buffers
   3961 
   3962 DESCRIPTION
   3963   OMX Use Buffer Heap allocation method implementation.
   3964 
   3965 PARAMETERS
   3966   <TBD>.
   3967 
   3968 RETURN VALUE
   3969   OMX Error None , if everything successful.
   3970 
   3971 ========================================================================== */
   3972 OMX_ERRORTYPE  omx_vdec::use_input_heap_buffers(
   3973                          OMX_IN OMX_HANDLETYPE            hComp,
   3974                          OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
   3975                          OMX_IN OMX_U32                   port,
   3976                          OMX_IN OMX_PTR                   appData,
   3977                          OMX_IN OMX_U32                   bytes,
   3978                          OMX_IN OMX_U8*                   buffer)
   3979 {
   3980   DEBUG_PRINT_LOW("Inside %s, %p\n", __FUNCTION__, buffer);
   3981   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   3982   if(!m_inp_heap_ptr)
   3983     m_inp_heap_ptr = (OMX_BUFFERHEADERTYPE*)
   3984                calloc( (sizeof(OMX_BUFFERHEADERTYPE)),
   3985                drv_ctx.ip_buf.actualcount);
   3986   if(!m_phdr_pmem_ptr)
   3987     m_phdr_pmem_ptr = (OMX_BUFFERHEADERTYPE**)
   3988                calloc( (sizeof(OMX_BUFFERHEADERTYPE*)),
   3989                drv_ctx.ip_buf.actualcount);
   3990   if(!m_inp_heap_ptr || !m_phdr_pmem_ptr)
   3991   {
   3992     DEBUG_PRINT_ERROR("Insufficent memory");
   3993     eRet = OMX_ErrorInsufficientResources;
   3994   }
   3995   else if (m_in_alloc_cnt < drv_ctx.ip_buf.actualcount)
   3996   {
   3997     input_use_buffer = true;
   3998     memset(&m_inp_heap_ptr[m_in_alloc_cnt], 0, sizeof(OMX_BUFFERHEADERTYPE));
   3999     m_inp_heap_ptr[m_in_alloc_cnt].pBuffer = buffer;
   4000     m_inp_heap_ptr[m_in_alloc_cnt].nAllocLen = bytes;
   4001     m_inp_heap_ptr[m_in_alloc_cnt].pAppPrivate = appData;
   4002     m_inp_heap_ptr[m_in_alloc_cnt].nInputPortIndex = (OMX_U32) OMX_DirInput;
   4003     m_inp_heap_ptr[m_in_alloc_cnt].nOutputPortIndex = (OMX_U32) OMX_DirMax;
   4004     *bufferHdr = &m_inp_heap_ptr[m_in_alloc_cnt];
   4005     eRet = allocate_input_buffer(hComp, &m_phdr_pmem_ptr[m_in_alloc_cnt], port, appData, bytes);
   4006     DEBUG_PRINT_HIGH("\n Heap buffer(%p) Pmem buffer(%p)", *bufferHdr, m_phdr_pmem_ptr[m_in_alloc_cnt]);
   4007     if (!m_input_free_q.insert_entry((unsigned)m_phdr_pmem_ptr[m_in_alloc_cnt], NULL, NULL))
   4008     {
   4009       DEBUG_PRINT_ERROR("\nERROR:Free_q is full");
   4010       return OMX_ErrorInsufficientResources;
   4011     }
   4012     m_in_alloc_cnt++;
   4013   }
   4014   else
   4015   {
   4016     DEBUG_PRINT_ERROR("All i/p buffers have been set!");
   4017     eRet = OMX_ErrorInsufficientResources;
   4018   }
   4019   return eRet;
   4020 }
   4021 
   4022 /* ======================================================================
   4023 FUNCTION
   4024   omx_vdec::UseBuffer
   4025 
   4026 DESCRIPTION
   4027   OMX Use Buffer method implementation.
   4028 
   4029 PARAMETERS
   4030   <TBD>.
   4031 
   4032 RETURN VALUE
   4033   OMX Error None , if everything successful.
   4034 
   4035 ========================================================================== */
   4036 OMX_ERRORTYPE  omx_vdec::use_buffer(
   4037                          OMX_IN OMX_HANDLETYPE            hComp,
   4038                          OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
   4039                          OMX_IN OMX_U32                   port,
   4040                          OMX_IN OMX_PTR                   appData,
   4041                          OMX_IN OMX_U32                   bytes,
   4042                          OMX_IN OMX_U8*                   buffer)
   4043 {
   4044   OMX_ERRORTYPE error = OMX_ErrorNone;
   4045   struct vdec_setbuffer_cmd setbuffers;
   4046   struct vdec_ioctl_msg ioctl_msg = {NULL,NULL};
   4047 
   4048   if (bufferHdr == NULL || bytes == 0)
   4049   {
   4050       if(!secure_mode && buffer == NULL) {
   4051           DEBUG_PRINT_ERROR("bad param 0x%p %ld 0x%p",bufferHdr, bytes, buffer);
   4052           return OMX_ErrorBadParameter;
   4053       }
   4054   }
   4055   if(m_state == OMX_StateInvalid)
   4056   {
   4057     DEBUG_PRINT_ERROR("Use Buffer in Invalid State\n");
   4058     return OMX_ErrorInvalidState;
   4059   }
   4060   if(port == OMX_CORE_INPUT_PORT_INDEX)
   4061     error = use_input_heap_buffers(hComp, bufferHdr, port, appData, bytes, buffer);
   4062   else if(port == OMX_CORE_OUTPUT_PORT_INDEX)
   4063     error = use_output_buffer(hComp,bufferHdr,port,appData,bytes,buffer); //not tested
   4064   else
   4065   {
   4066     DEBUG_PRINT_ERROR("Error: Invalid Port Index received %d\n",(int)port);
   4067     error = OMX_ErrorBadPortIndex;
   4068   }
   4069   DEBUG_PRINT_LOW("Use Buffer: port %u, buffer %p, eRet %d", port, *bufferHdr, error);
   4070   if(error == OMX_ErrorNone)
   4071   {
   4072     if(allocate_done() && BITMASK_PRESENT(&m_flags,OMX_COMPONENT_IDLE_PENDING))
   4073     {
   4074       // Send the callback now
   4075       BITMASK_CLEAR((&m_flags),OMX_COMPONENT_IDLE_PENDING);
   4076       post_event(OMX_CommandStateSet,OMX_StateIdle,
   4077                          OMX_COMPONENT_GENERATE_EVENT);
   4078     }
   4079     if(port == OMX_CORE_INPUT_PORT_INDEX && m_inp_bPopulated &&
   4080        BITMASK_PRESENT(&m_flags,OMX_COMPONENT_INPUT_ENABLE_PENDING))
   4081     {
   4082       BITMASK_CLEAR((&m_flags),OMX_COMPONENT_INPUT_ENABLE_PENDING);
   4083       post_event(OMX_CommandPortEnable,
   4084           OMX_CORE_INPUT_PORT_INDEX,
   4085           OMX_COMPONENT_GENERATE_EVENT);
   4086     }
   4087     else if(port == OMX_CORE_OUTPUT_PORT_INDEX && m_out_bPopulated &&
   4088             BITMASK_PRESENT(&m_flags,OMX_COMPONENT_OUTPUT_ENABLE_PENDING))
   4089     {
   4090       BITMASK_CLEAR((&m_flags),OMX_COMPONENT_OUTPUT_ENABLE_PENDING);
   4091       post_event(OMX_CommandPortEnable,
   4092                  OMX_CORE_OUTPUT_PORT_INDEX,
   4093                  OMX_COMPONENT_GENERATE_EVENT);
   4094     }
   4095   }
   4096   return error;
   4097 }
   4098 
   4099 OMX_ERRORTYPE omx_vdec::free_input_buffer(unsigned int bufferindex,
   4100                                 OMX_BUFFERHEADERTYPE *pmem_bufferHdr)
   4101 {
   4102   if (m_inp_heap_ptr && !input_use_buffer && arbitrary_bytes)
   4103   {
   4104     if(m_inp_heap_ptr[bufferindex].pBuffer)
   4105       free(m_inp_heap_ptr[bufferindex].pBuffer);
   4106     m_inp_heap_ptr[bufferindex].pBuffer = NULL;
   4107   }
   4108   if (pmem_bufferHdr)
   4109     free_input_buffer(pmem_bufferHdr);
   4110   return OMX_ErrorNone;
   4111 }
   4112 
   4113 OMX_ERRORTYPE omx_vdec::free_input_buffer(OMX_BUFFERHEADERTYPE *bufferHdr)
   4114 {
   4115   unsigned int index = 0;
   4116   if (bufferHdr == NULL || m_inp_mem_ptr == NULL)
   4117   {
   4118     return OMX_ErrorBadParameter;
   4119   }
   4120 
   4121   index = bufferHdr - m_inp_mem_ptr;
   4122   DEBUG_PRINT_LOW("\n Free Input Buffer index = %d",index);
   4123 
   4124   if (index < drv_ctx.ip_buf.actualcount && drv_ctx.ptr_inputbuffer)
   4125   {
   4126     DEBUG_PRINT_LOW("\n Free Input Buffer index = %d",index);
   4127     if (drv_ctx.ptr_inputbuffer[index].pmem_fd > 0)
   4128     {
   4129        struct vdec_ioctl_msg ioctl_msg = {NULL,NULL};
   4130        struct vdec_setbuffer_cmd setbuffers;
   4131        setbuffers.buffer_type = VDEC_BUFFER_TYPE_INPUT;
   4132        memcpy (&setbuffers.buffer,&drv_ctx.ptr_inputbuffer[index],
   4133           sizeof (vdec_bufferpayload));
   4134        ioctl_msg.in  = &setbuffers;
   4135        ioctl_msg.out = NULL;
   4136        int ioctl_r; //= ioctl (drv_ctx.video_driver_fd,
   4137            //                 VDEC_IOCTL_FREE_BUFFER, &ioctl_msg);
   4138        if (ioctl_r < 0)
   4139        {
   4140           DEBUG_PRINT_ERROR("\nVDEC_IOCTL_FREE_BUFFER returned error %d", ioctl_r);
   4141        }
   4142 
   4143        DEBUG_PRINT_LOW("\n unmap the input buffer fd=%d",
   4144                     drv_ctx.ptr_inputbuffer[index].pmem_fd);
   4145        DEBUG_PRINT_LOW("\n unmap the input buffer size=%d  address = %d",
   4146                     drv_ctx.ptr_inputbuffer[index].mmaped_size,
   4147                     drv_ctx.ptr_inputbuffer[index].bufferaddr);
   4148        munmap (drv_ctx.ptr_inputbuffer[index].bufferaddr,
   4149                drv_ctx.ptr_inputbuffer[index].mmaped_size);
   4150        close (drv_ctx.ptr_inputbuffer[index].pmem_fd);
   4151        drv_ctx.ptr_inputbuffer[index].pmem_fd = -1;
   4152        if (m_desc_buffer_ptr && m_desc_buffer_ptr[index].buf_addr)
   4153        {
   4154          free(m_desc_buffer_ptr[index].buf_addr);
   4155          m_desc_buffer_ptr[index].buf_addr = NULL;
   4156          m_desc_buffer_ptr[index].desc_data_size = 0;
   4157        }
   4158 #ifdef USE_ION
   4159        free_ion_memory(&drv_ctx.ip_buf_ion_info[index]);
   4160 #endif
   4161     }
   4162   }
   4163 
   4164   return OMX_ErrorNone;
   4165 }
   4166 
   4167 OMX_ERRORTYPE omx_vdec::free_output_buffer(OMX_BUFFERHEADERTYPE *bufferHdr)
   4168 {
   4169   unsigned int index = 0;
   4170 
   4171   if (bufferHdr == NULL || m_out_mem_ptr == NULL)
   4172   {
   4173     return OMX_ErrorBadParameter;
   4174   }
   4175 
   4176   index = bufferHdr - m_out_mem_ptr;
   4177   DEBUG_PRINT_LOW("\n Free ouput Buffer index = %d",index);
   4178 
   4179   if (index < drv_ctx.op_buf.actualcount
   4180       && drv_ctx.ptr_outputbuffer)
   4181   {
   4182     DEBUG_PRINT_LOW("\n Free ouput Buffer index = %d addr = %x", index,
   4183                     drv_ctx.ptr_outputbuffer[index].bufferaddr);
   4184 
   4185     struct vdec_ioctl_msg ioctl_msg = {NULL,NULL};
   4186     struct vdec_setbuffer_cmd setbuffers;
   4187     setbuffers.buffer_type = VDEC_BUFFER_TYPE_OUTPUT;
   4188     memcpy (&setbuffers.buffer,&drv_ctx.ptr_outputbuffer[index],
   4189         sizeof (vdec_bufferpayload));
   4190     ioctl_msg.in  = &setbuffers;
   4191     ioctl_msg.out = NULL;
   4192     DEBUG_PRINT_LOW("\nRelease the Output Buffer");
   4193     if (/*ioctl (drv_ctx.video_driver_fd, VDEC_IOCTL_FREE_BUFFER,
   4194           &ioctl_msg) < */0)
   4195       DEBUG_PRINT_ERROR("\nRelease output buffer failed in VCD");
   4196 
   4197 #ifdef _ANDROID_
   4198     if(m_enable_android_native_buffers) {
   4199         if(drv_ctx.ptr_outputbuffer[index].pmem_fd > 0) {
   4200             munmap(drv_ctx.ptr_outputbuffer[index].bufferaddr,
   4201                     drv_ctx.ptr_outputbuffer[index].mmaped_size);
   4202         }
   4203         drv_ctx.ptr_outputbuffer[index].pmem_fd = -1;
   4204     } else {
   4205 #endif
   4206         if (drv_ctx.ptr_outputbuffer[0].pmem_fd > 0 && !ouput_egl_buffers && !m_use_output_pmem)
   4207         {
   4208             DEBUG_PRINT_LOW("\n unmap the output buffer fd = %d",
   4209                     drv_ctx.ptr_outputbuffer[0].pmem_fd);
   4210             DEBUG_PRINT_LOW("\n unmap the ouput buffer size=%d  address = %d",
   4211                     drv_ctx.ptr_outputbuffer[0].mmaped_size,
   4212                     drv_ctx.ptr_outputbuffer[0].bufferaddr);
   4213             munmap (drv_ctx.ptr_outputbuffer[0].bufferaddr,
   4214                     drv_ctx.ptr_outputbuffer[0].mmaped_size);
   4215           close (drv_ctx.ptr_outputbuffer[0].pmem_fd);
   4216           drv_ctx.ptr_outputbuffer[0].pmem_fd = -1;
   4217 #ifdef USE_ION
   4218        free_ion_memory(&drv_ctx.op_buf_ion_info[0]);
   4219 #endif
   4220         }
   4221 #ifdef _ANDROID_
   4222     }
   4223 #endif
   4224   }
   4225 
   4226   return OMX_ErrorNone;
   4227 
   4228 }
   4229 
   4230 OMX_ERRORTYPE omx_vdec::allocate_input_heap_buffer(OMX_HANDLETYPE       hComp,
   4231                                          OMX_BUFFERHEADERTYPE **bufferHdr,
   4232                                          OMX_U32              port,
   4233                                          OMX_PTR              appData,
   4234                                          OMX_U32              bytes)
   4235 {
   4236   OMX_BUFFERHEADERTYPE *input = NULL;
   4237   unsigned char *buf_addr = NULL;
   4238   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   4239   unsigned   i = 0;
   4240 
   4241   /* Sanity Check*/
   4242   if (bufferHdr == NULL)
   4243   {
   4244     return OMX_ErrorBadParameter;
   4245   }
   4246 
   4247   if (m_inp_heap_ptr == NULL)
   4248   {
   4249     m_inp_heap_ptr = (OMX_BUFFERHEADERTYPE*) \
   4250                      calloc( (sizeof(OMX_BUFFERHEADERTYPE)),
   4251                      drv_ctx.ip_buf.actualcount);
   4252     m_phdr_pmem_ptr = (OMX_BUFFERHEADERTYPE**) \
   4253                      calloc( (sizeof(OMX_BUFFERHEADERTYPE*)),
   4254                      drv_ctx.ip_buf.actualcount);
   4255 
   4256     if (m_inp_heap_ptr == NULL)
   4257     {
   4258       DEBUG_PRINT_ERROR("\n m_inp_heap_ptr Allocation failed ");
   4259       return OMX_ErrorInsufficientResources;
   4260     }
   4261   }
   4262 
   4263   /*Find a Free index*/
   4264   for(i=0; i< drv_ctx.ip_buf.actualcount; i++)
   4265   {
   4266     if(BITMASK_ABSENT(&m_heap_inp_bm_count,i))
   4267     {
   4268       DEBUG_PRINT_LOW("\n Free Input Buffer Index %d",i);
   4269       break;
   4270     }
   4271   }
   4272 
   4273   if (i < drv_ctx.ip_buf.actualcount)
   4274   {
   4275     buf_addr = (unsigned char *)malloc (drv_ctx.ip_buf.buffer_size);
   4276 
   4277     if (buf_addr == NULL)
   4278     {
   4279       return OMX_ErrorInsufficientResources;
   4280     }
   4281 
   4282     *bufferHdr = (m_inp_heap_ptr + i);
   4283     input = *bufferHdr;
   4284     BITMASK_SET(&m_heap_inp_bm_count,i);
   4285 
   4286     input->pBuffer           = (OMX_U8 *)buf_addr;
   4287     input->nSize             = sizeof(OMX_BUFFERHEADERTYPE);
   4288     input->nVersion.nVersion = OMX_SPEC_VERSION;
   4289     input->nAllocLen         = drv_ctx.ip_buf.buffer_size;
   4290     input->pAppPrivate       = appData;
   4291     input->nInputPortIndex   = OMX_CORE_INPUT_PORT_INDEX;
   4292     DEBUG_PRINT_LOW("\n Address of Heap Buffer %p",*bufferHdr );
   4293     eRet = allocate_input_buffer(hComp,&m_phdr_pmem_ptr [i],port,appData,bytes);
   4294     DEBUG_PRINT_LOW("\n Address of Pmem Buffer %p",m_phdr_pmem_ptr [i] );
   4295     /*Add the Buffers to freeq*/
   4296     if (!m_input_free_q.insert_entry((unsigned)m_phdr_pmem_ptr [i],NULL,NULL))
   4297     {
   4298       DEBUG_PRINT_ERROR("\nERROR:Free_q is full");
   4299       return OMX_ErrorInsufficientResources;
   4300     }
   4301   }
   4302   else
   4303   {
   4304     return OMX_ErrorBadParameter;
   4305   }
   4306 
   4307   return eRet;
   4308 
   4309 }
   4310 
   4311 
   4312 /* ======================================================================
   4313 FUNCTION
   4314   omx_vdec::AllocateInputBuffer
   4315 
   4316 DESCRIPTION
   4317   Helper function for allocate buffer in the input pin
   4318 
   4319 PARAMETERS
   4320   None.
   4321 
   4322 RETURN VALUE
   4323   true/false
   4324 
   4325 ========================================================================== */
   4326 OMX_ERRORTYPE  omx_vdec::allocate_input_buffer(
   4327                          OMX_IN OMX_HANDLETYPE            hComp,
   4328                          OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
   4329                          OMX_IN OMX_U32                   port,
   4330                          OMX_IN OMX_PTR                   appData,
   4331                          OMX_IN OMX_U32                   bytes)
   4332 {
   4333 
   4334   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   4335   struct vdec_setbuffer_cmd setbuffers;
   4336   OMX_BUFFERHEADERTYPE *input = NULL;
   4337   struct vdec_ioctl_msg ioctl_msg = {NULL,NULL};
   4338   unsigned   i = 0;
   4339   unsigned char *buf_addr = NULL;
   4340   int pmem_fd = -1;
   4341 
   4342   if(bytes != drv_ctx.ip_buf.buffer_size)
   4343   {
   4344     DEBUG_PRINT_LOW("\n Requested Size is wrong %d epected is %d",
   4345       bytes, drv_ctx.ip_buf.buffer_size);
   4346     return OMX_ErrorBadParameter;
   4347   }
   4348 
   4349   if(!m_inp_mem_ptr)
   4350   {
   4351     DEBUG_PRINT_HIGH("\n Allocate i/p buffer Header: Cnt(%d) Sz(%d)",
   4352       drv_ctx.ip_buf.actualcount,
   4353       drv_ctx.ip_buf.buffer_size);
   4354 
   4355     m_inp_mem_ptr = (OMX_BUFFERHEADERTYPE*) \
   4356     calloc( (sizeof(OMX_BUFFERHEADERTYPE)), drv_ctx.ip_buf.actualcount);
   4357 
   4358     if (m_inp_mem_ptr == NULL)
   4359     {
   4360       return OMX_ErrorInsufficientResources;
   4361     }
   4362 
   4363     drv_ctx.ptr_inputbuffer = (struct vdec_bufferpayload *) \
   4364     calloc ((sizeof (struct vdec_bufferpayload)),drv_ctx.ip_buf.actualcount);
   4365 
   4366     if (drv_ctx.ptr_inputbuffer == NULL)
   4367     {
   4368       return OMX_ErrorInsufficientResources;
   4369     }
   4370 #ifdef USE_ION
   4371     drv_ctx.ip_buf_ion_info = (struct vdec_ion *) \
   4372     calloc ((sizeof (struct vdec_ion)),drv_ctx.ip_buf.actualcount);
   4373 
   4374     if (drv_ctx.ip_buf_ion_info == NULL)
   4375     {
   4376       return OMX_ErrorInsufficientResources;
   4377     }
   4378 #endif
   4379 
   4380     for (i=0; i < drv_ctx.ip_buf.actualcount; i++)
   4381     {
   4382       drv_ctx.ptr_inputbuffer [i].pmem_fd = -1;
   4383 #ifdef USE_ION
   4384       drv_ctx.ip_buf_ion_info[i].ion_device_fd = -1;
   4385 #endif
   4386     }
   4387   }
   4388 
   4389   for(i=0; i< drv_ctx.ip_buf.actualcount; i++)
   4390   {
   4391     if(BITMASK_ABSENT(&m_inp_bm_count,i))
   4392     {
   4393       DEBUG_PRINT_LOW("\n Free Input Buffer Index %d",i);
   4394       break;
   4395     }
   4396   }
   4397 
   4398   if(i < drv_ctx.ip_buf.actualcount)
   4399   {
   4400     struct v4l2_buffer buf;
   4401     struct v4l2_plane plane;
   4402     int rc;
   4403     DEBUG_PRINT_LOW("\n Allocate input Buffer");
   4404 #ifdef USE_ION
   4405  drv_ctx.ip_buf_ion_info[i].ion_device_fd = alloc_map_ion_memory(
   4406                     drv_ctx.ip_buf.buffer_size,drv_ctx.op_buf.alignment,
   4407                     &drv_ctx.ip_buf_ion_info[i].ion_alloc_data,
   4408 		    &drv_ctx.ip_buf_ion_info[i].fd_ion_data, 0);
   4409     if(drv_ctx.ip_buf_ion_info[i].ion_device_fd < 0) {
   4410         return OMX_ErrorInsufficientResources;
   4411      }
   4412     pmem_fd = drv_ctx.ip_buf_ion_info[i].fd_ion_data.fd;
   4413 #else
   4414     pmem_fd = open (MEM_DEVICE,O_RDWR);
   4415 
   4416     if (pmem_fd < 0)
   4417     {
   4418       DEBUG_PRINT_ERROR("\n open failed for pmem/adsp for input buffer");
   4419       return OMX_ErrorInsufficientResources;
   4420     }
   4421 
   4422     if (pmem_fd == 0)
   4423     {
   4424       pmem_fd = open (MEM_DEVICE,O_RDWR);
   4425 
   4426       if (pmem_fd < 0)
   4427       {
   4428         DEBUG_PRINT_ERROR("\n open failed for pmem/adsp for input buffer");
   4429         return OMX_ErrorInsufficientResources;
   4430       }
   4431     }
   4432 
   4433     if(!align_pmem_buffers(pmem_fd, drv_ctx.ip_buf.buffer_size,
   4434       drv_ctx.ip_buf.alignment))
   4435     {
   4436       DEBUG_PRINT_ERROR("\n align_pmem_buffers() failed");
   4437       close(pmem_fd);
   4438       return OMX_ErrorInsufficientResources;
   4439     }
   4440 #endif
   4441     if (!secure_mode) {
   4442         buf_addr = (unsigned char *)mmap(NULL,
   4443           drv_ctx.ip_buf.buffer_size,
   4444           PROT_READ|PROT_WRITE, MAP_SHARED, pmem_fd, 0);
   4445 
   4446         if (buf_addr == MAP_FAILED)
   4447         {
   4448             close(pmem_fd);
   4449 #ifdef USE_ION
   4450             free_ion_memory(&drv_ctx.ip_buf_ion_info[i]);
   4451 #endif
   4452           DEBUG_PRINT_ERROR("\n Map Failed to allocate input buffer");
   4453           return OMX_ErrorInsufficientResources;
   4454         }
   4455     }
   4456     *bufferHdr = (m_inp_mem_ptr + i);
   4457     if (secure_mode)
   4458         drv_ctx.ptr_inputbuffer [i].bufferaddr = *bufferHdr;
   4459     else
   4460         drv_ctx.ptr_inputbuffer [i].bufferaddr = buf_addr;
   4461     drv_ctx.ptr_inputbuffer [i].pmem_fd = pmem_fd;
   4462     drv_ctx.ptr_inputbuffer [i].buffer_len = drv_ctx.ip_buf.buffer_size;
   4463     drv_ctx.ptr_inputbuffer [i].mmaped_size = drv_ctx.ip_buf.buffer_size;
   4464     drv_ctx.ptr_inputbuffer [i].offset = 0;
   4465 
   4466 
   4467     buf.index = i;
   4468     buf.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
   4469     buf.memory = V4L2_MEMORY_USERPTR;
   4470     plane.bytesused = 0;
   4471     plane.length = drv_ctx.ptr_inputbuffer [i].mmaped_size;
   4472     plane.m.userptr = (unsigned long)drv_ctx.ptr_inputbuffer[i].bufferaddr;
   4473     plane.reserved[0] =drv_ctx.ptr_inputbuffer [i].pmem_fd;
   4474     plane.reserved[1] = 0;
   4475     plane.data_offset = drv_ctx.ptr_inputbuffer[i].offset;
   4476     buf.m.planes = &plane;
   4477     buf.length = 1;
   4478 
   4479      DEBUG_PRINT_LOW("\n Set the Output Buffer Idx: %d Addr: %x", i, drv_ctx.ptr_inputbuffer[i]);
   4480 
   4481      rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_PREPARE_BUF, &buf);
   4482 
   4483      if (rc) {
   4484        printf("Failed to prepare bufs\n");
   4485 	   /*TODO: How to handle this case */
   4486        return OMX_ErrorInsufficientResources;
   4487      }
   4488 
   4489     input = *bufferHdr;
   4490     BITMASK_SET(&m_inp_bm_count,i);
   4491     DEBUG_PRINT_LOW("\n Buffer address %p of pmem",*bufferHdr);
   4492     if (secure_mode)
   4493          input->pBuffer = (OMX_U8 *)drv_ctx.ptr_inputbuffer [i].pmem_fd;
   4494     else
   4495          input->pBuffer           = (OMX_U8 *)buf_addr;
   4496     input->nSize             = sizeof(OMX_BUFFERHEADERTYPE);
   4497     input->nVersion.nVersion = OMX_SPEC_VERSION;
   4498     input->nAllocLen         = drv_ctx.ip_buf.buffer_size;
   4499     input->pAppPrivate       = appData;
   4500     input->nInputPortIndex   = OMX_CORE_INPUT_PORT_INDEX;
   4501     input->pInputPortPrivate = (void *)&drv_ctx.ptr_inputbuffer [i];
   4502 
   4503     if (drv_ctx.disable_dmx)
   4504     {
   4505       eRet = allocate_desc_buffer(i);
   4506     }
   4507   }
   4508   else
   4509   {
   4510     DEBUG_PRINT_ERROR("\nERROR:Input Buffer Index not found");
   4511     eRet = OMX_ErrorInsufficientResources;
   4512   }
   4513   return eRet;
   4514 }
   4515 
   4516 
   4517 /* ======================================================================
   4518 FUNCTION
   4519   omx_vdec::AllocateOutputBuffer
   4520 
   4521 DESCRIPTION
   4522   Helper fn for AllocateBuffer in the output pin
   4523 
   4524 PARAMETERS
   4525   <TBD>.
   4526 
   4527 RETURN VALUE
   4528   OMX Error None if everything went well.
   4529 
   4530 ========================================================================== */
   4531 OMX_ERRORTYPE  omx_vdec::allocate_output_buffer(
   4532                          OMX_IN OMX_HANDLETYPE            hComp,
   4533                          OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
   4534                          OMX_IN OMX_U32                   port,
   4535                          OMX_IN OMX_PTR                   appData,
   4536                          OMX_IN OMX_U32                   bytes)
   4537 {
   4538   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   4539   OMX_BUFFERHEADERTYPE       *bufHdr= NULL; // buffer header
   4540   unsigned                         i= 0; // Temporary counter
   4541   struct vdec_ioctl_msg ioctl_msg = {NULL,NULL};
   4542   struct vdec_setbuffer_cmd setbuffers;
   4543 #ifdef USE_ION
   4544   int ion_device_fd =-1;
   4545   struct ion_allocation_data ion_alloc_data;
   4546   struct ion_fd_data fd_ion_data;
   4547 #endif
   4548   if(!m_out_mem_ptr)
   4549   {
   4550     DEBUG_PRINT_HIGH("\n Allocate o/p buffer Header: Cnt(%d) Sz(%d)",
   4551       drv_ctx.op_buf.actualcount,
   4552       drv_ctx.op_buf.buffer_size);
   4553     int nBufHdrSize        = 0;
   4554     int nPlatformEntrySize = 0;
   4555     int nPlatformListSize  = 0;
   4556     int nPMEMInfoSize = 0;
   4557     int pmem_fd = -1;
   4558     unsigned char *pmem_baseaddress = NULL;
   4559 
   4560     OMX_QCOM_PLATFORM_PRIVATE_LIST      *pPlatformList;
   4561     OMX_QCOM_PLATFORM_PRIVATE_ENTRY     *pPlatformEntry;
   4562     OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *pPMEMInfo;
   4563 
   4564     DEBUG_PRINT_LOW("Allocating First Output Buffer(%d)\n",
   4565       drv_ctx.op_buf.actualcount);
   4566     nBufHdrSize        = drv_ctx.op_buf.actualcount *
   4567                          sizeof(OMX_BUFFERHEADERTYPE);
   4568 
   4569     nPMEMInfoSize      = drv_ctx.op_buf.actualcount *
   4570                          sizeof(OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO);
   4571     nPlatformListSize  = drv_ctx.op_buf.actualcount *
   4572                          sizeof(OMX_QCOM_PLATFORM_PRIVATE_LIST);
   4573     nPlatformEntrySize = drv_ctx.op_buf.actualcount *
   4574                          sizeof(OMX_QCOM_PLATFORM_PRIVATE_ENTRY);
   4575 
   4576     DEBUG_PRINT_LOW("TotalBufHdr %d BufHdrSize %d PMEM %d PL %d\n",nBufHdrSize,
   4577                          sizeof(OMX_BUFFERHEADERTYPE),
   4578                          nPMEMInfoSize,
   4579                          nPlatformListSize);
   4580     DEBUG_PRINT_LOW("PE %d OutputBuffer Count %d \n",nPlatformEntrySize,
   4581                          drv_ctx.op_buf.actualcount);
   4582 #ifdef USE_ION
   4583  ion_device_fd = alloc_map_ion_memory(
   4584                     drv_ctx.op_buf.buffer_size * drv_ctx.op_buf.actualcount,
   4585                     drv_ctx.op_buf.alignment,
   4586                     &ion_alloc_data, &fd_ion_data, 0);
   4587     if (ion_device_fd < 0) {
   4588         return OMX_ErrorInsufficientResources;
   4589     }
   4590     pmem_fd = fd_ion_data.fd;
   4591 #else
   4592     pmem_fd = open (MEM_DEVICE,O_RDWR);
   4593 
   4594     if (pmem_fd < 0)
   4595     {
   4596       DEBUG_PRINT_ERROR("\nERROR:pmem fd for output buffer %d",
   4597         drv_ctx.op_buf.buffer_size);
   4598       return OMX_ErrorInsufficientResources;
   4599     }
   4600 
   4601     if(pmem_fd == 0)
   4602     {
   4603       pmem_fd = open (MEM_DEVICE,O_RDWR);
   4604 
   4605       if (pmem_fd < 0)
   4606       {
   4607         DEBUG_PRINT_ERROR("\nERROR:pmem fd for output buffer %d",
   4608           drv_ctx.op_buf.buffer_size);
   4609         return OMX_ErrorInsufficientResources;
   4610       }
   4611     }
   4612 
   4613     if(!align_pmem_buffers(pmem_fd, drv_ctx.op_buf.buffer_size *
   4614       drv_ctx.op_buf.actualcount,
   4615       drv_ctx.op_buf.alignment))
   4616     {
   4617       DEBUG_PRINT_ERROR("\n align_pmem_buffers() failed");
   4618       close(pmem_fd);
   4619       return OMX_ErrorInsufficientResources;
   4620     }
   4621 #endif
   4622    if (!secure_mode) {
   4623         pmem_baseaddress = (unsigned char *)mmap(NULL,
   4624                            (drv_ctx.op_buf.buffer_size *
   4625                             drv_ctx.op_buf.actualcount),
   4626                             PROT_READ|PROT_WRITE,MAP_SHARED,pmem_fd,0);
   4627         if (pmem_baseaddress == MAP_FAILED)
   4628         {
   4629           DEBUG_PRINT_ERROR("\n MMAP failed for Size %d",
   4630           drv_ctx.op_buf.buffer_size);
   4631           close(pmem_fd);
   4632 #ifdef USE_ION
   4633           free_ion_memory(&drv_ctx.op_buf_ion_info[i]);
   4634 #endif
   4635           return OMX_ErrorInsufficientResources;
   4636         }
   4637     }
   4638     m_out_mem_ptr = (OMX_BUFFERHEADERTYPE  *)calloc(nBufHdrSize,1);
   4639     // Alloc mem for platform specific info
   4640     char *pPtr=NULL;
   4641     pPtr = (char*) calloc(nPlatformListSize + nPlatformEntrySize +
   4642                                      nPMEMInfoSize,1);
   4643     drv_ctx.ptr_outputbuffer = (struct vdec_bufferpayload *)\
   4644       calloc (sizeof(struct vdec_bufferpayload),
   4645       drv_ctx.op_buf.actualcount);
   4646     drv_ctx.ptr_respbuffer = (struct vdec_output_frameinfo  *)\
   4647       calloc (sizeof (struct vdec_output_frameinfo),
   4648       drv_ctx.op_buf.actualcount);
   4649 #ifdef USE_ION
   4650     drv_ctx.op_buf_ion_info = (struct vdec_ion *)\
   4651       calloc (sizeof(struct vdec_ion),
   4652       drv_ctx.op_buf.actualcount);
   4653 #endif
   4654 
   4655     if(m_out_mem_ptr && pPtr && drv_ctx.ptr_outputbuffer
   4656        && drv_ctx.ptr_respbuffer)
   4657     {
   4658       drv_ctx.ptr_outputbuffer[0].mmaped_size =
   4659         (drv_ctx.op_buf.buffer_size *
   4660          drv_ctx.op_buf.actualcount);
   4661       bufHdr          =  m_out_mem_ptr;
   4662       m_platform_list = (OMX_QCOM_PLATFORM_PRIVATE_LIST *)(pPtr);
   4663       m_platform_entry= (OMX_QCOM_PLATFORM_PRIVATE_ENTRY *)
   4664                         (((char *) m_platform_list)  + nPlatformListSize);
   4665       m_pmem_info     = (OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *)
   4666                         (((char *) m_platform_entry) + nPlatformEntrySize);
   4667       pPlatformList   = m_platform_list;
   4668       pPlatformEntry  = m_platform_entry;
   4669       pPMEMInfo       = m_pmem_info;
   4670 
   4671       DEBUG_PRINT_LOW("Memory Allocation Succeeded for OUT port%p\n",m_out_mem_ptr);
   4672 
   4673       // Settting the entire storage nicely
   4674       DEBUG_PRINT_LOW("bHdr %p OutMem %p PE %p\n",bufHdr, m_out_mem_ptr,pPlatformEntry);
   4675       DEBUG_PRINT_LOW(" Pmem Info = %p \n",pPMEMInfo);
   4676       for(i=0; i < drv_ctx.op_buf.actualcount ; i++)
   4677       {
   4678         bufHdr->nSize              = sizeof(OMX_BUFFERHEADERTYPE);
   4679         bufHdr->nVersion.nVersion  = OMX_SPEC_VERSION;
   4680         // Set the values when we determine the right HxW param
   4681         bufHdr->nAllocLen          = bytes;
   4682         bufHdr->nFilledLen         = 0;
   4683         bufHdr->pAppPrivate        = appData;
   4684         bufHdr->nOutputPortIndex   = OMX_CORE_OUTPUT_PORT_INDEX;
   4685         // Platform specific PMEM Information
   4686         // Initialize the Platform Entry
   4687         //DEBUG_PRINT_LOW("Initializing the Platform Entry for %d\n",i);
   4688         pPlatformEntry->type       = OMX_QCOM_PLATFORM_PRIVATE_PMEM;
   4689         pPlatformEntry->entry      = pPMEMInfo;
   4690         // Initialize the Platform List
   4691         pPlatformList->nEntries    = 1;
   4692         pPlatformList->entryList   = pPlatformEntry;
   4693         // Keep pBuffer NULL till vdec is opened
   4694         bufHdr->pBuffer            = NULL;
   4695         bufHdr->nOffset            = 0;
   4696 
   4697         pPMEMInfo->offset          =  drv_ctx.op_buf.buffer_size*i;
   4698         pPMEMInfo->pmem_fd = 0;
   4699         bufHdr->pPlatformPrivate = pPlatformList;
   4700 
   4701         drv_ctx.ptr_outputbuffer[i].pmem_fd = pmem_fd;
   4702 #ifdef USE_ION
   4703         drv_ctx.op_buf_ion_info[i].ion_device_fd = ion_device_fd;
   4704         drv_ctx.op_buf_ion_info[i].ion_alloc_data = ion_alloc_data;
   4705         drv_ctx.op_buf_ion_info[i].fd_ion_data = fd_ion_data;
   4706 #endif
   4707 
   4708         /*Create a mapping between buffers*/
   4709         bufHdr->pOutputPortPrivate = &drv_ctx.ptr_respbuffer[i];
   4710         drv_ctx.ptr_respbuffer[i].client_data = (void *)\
   4711                                             &drv_ctx.ptr_outputbuffer[i];
   4712         drv_ctx.ptr_outputbuffer[i].offset = drv_ctx.op_buf.buffer_size*i;
   4713         drv_ctx.ptr_outputbuffer[i].bufferaddr =
   4714           pmem_baseaddress + (drv_ctx.op_buf.buffer_size*i);
   4715 
   4716         DEBUG_PRINT_LOW("\n pmem_fd = %d offset = %d address = %p",
   4717           pmem_fd, drv_ctx.ptr_outputbuffer[i].offset,
   4718           drv_ctx.ptr_outputbuffer[i].bufferaddr);
   4719         // Move the buffer and buffer header pointers
   4720         bufHdr++;
   4721         pPMEMInfo++;
   4722         pPlatformEntry++;
   4723         pPlatformList++;
   4724       }
   4725     }
   4726     else
   4727     {
   4728       DEBUG_PRINT_ERROR("Output buf mem alloc failed[0x%x][0x%x]\n",\
   4729                                         m_out_mem_ptr, pPtr);
   4730       if(m_out_mem_ptr)
   4731       {
   4732         free(m_out_mem_ptr);
   4733         m_out_mem_ptr = NULL;
   4734       }
   4735       if(pPtr)
   4736       {
   4737         free(pPtr);
   4738         pPtr = NULL;
   4739       }
   4740       if(drv_ctx.ptr_outputbuffer)
   4741       {
   4742         free(drv_ctx.ptr_outputbuffer);
   4743         drv_ctx.ptr_outputbuffer = NULL;
   4744       }
   4745       if(drv_ctx.ptr_respbuffer)
   4746       {
   4747         free(drv_ctx.ptr_respbuffer);
   4748         drv_ctx.ptr_respbuffer = NULL;
   4749       }
   4750 #ifdef USE_ION
   4751     if (drv_ctx.op_buf_ion_info) {
   4752         DEBUG_PRINT_LOW("\n Free o/p ion context");
   4753 	free(drv_ctx.op_buf_ion_info);
   4754         drv_ctx.op_buf_ion_info = NULL;
   4755     }
   4756 #endif
   4757       eRet =  OMX_ErrorInsufficientResources;
   4758     }
   4759   }
   4760 
   4761   for(i=0; i< drv_ctx.op_buf.actualcount; i++)
   4762   {
   4763     if(BITMASK_ABSENT(&m_out_bm_count,i))
   4764     {
   4765       DEBUG_PRINT_LOW("\n Found a Free Output Buffer %d",i);
   4766       break;
   4767     }
   4768   }
   4769 
   4770   if (eRet == OMX_ErrorNone)
   4771   {
   4772     if(i < drv_ctx.op_buf.actualcount)
   4773     {
   4774       struct v4l2_buffer buf;
   4775       struct v4l2_plane plane;
   4776       int rc;
   4777       m_pmem_info[i].offset = drv_ctx.ptr_outputbuffer[i].offset;
   4778 
   4779       drv_ctx.ptr_outputbuffer[i].buffer_len =
   4780         drv_ctx.op_buf.buffer_size;
   4781 
   4782     *bufferHdr = (m_out_mem_ptr + i );
   4783     if (secure_mode) {
   4784        drv_ctx.ptr_outputbuffer[i].bufferaddr = *bufferHdr;
   4785     }
   4786    drv_ctx.ptr_outputbuffer[i].mmaped_size = drv_ctx.op_buf.buffer_size;
   4787 
   4788      buf.index = i;
   4789      buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   4790      buf.memory = V4L2_MEMORY_USERPTR;
   4791      plane.length = drv_ctx.op_buf.buffer_size;
   4792      plane.m.userptr = (unsigned long)(drv_ctx.ptr_outputbuffer[i].bufferaddr-drv_ctx.ptr_outputbuffer[i].offset);
   4793 #ifdef USE_ION
   4794      plane.reserved[0] = drv_ctx.op_buf_ion_info[i].fd_ion_data.fd;
   4795 #endif
   4796      plane.reserved[1] = drv_ctx.ptr_outputbuffer[i].offset;
   4797      plane.data_offset = 0;
   4798      buf.m.planes = &plane;
   4799      buf.length = 1;
   4800 
   4801 	 DEBUG_PRINT_LOW("\n Set the Output Buffer Idx: %d Addr: %x", i, drv_ctx.ptr_outputbuffer[i]);
   4802       rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_PREPARE_BUF, &buf);
   4803       if (rc) {
   4804 		/*TODO: How to handle this case */
   4805        return OMX_ErrorInsufficientResources;
   4806      }
   4807 
   4808 	  if (i == (drv_ctx.op_buf.actualcount -1 ) && !streaming[CAPTURE_PORT]) {
   4809 		enum v4l2_buf_type buf_type;
   4810 		buf_type=V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   4811 		rc=ioctl(drv_ctx.video_driver_fd, VIDIOC_STREAMON,&buf_type);
   4812 		if (rc) {
   4813 			return OMX_ErrorInsufficientResources;
   4814 		} else {
   4815 			streaming[CAPTURE_PORT] = true;
   4816 			DEBUG_PRINT_LOW("\n STREAMON Successful \n ");
   4817 		}
   4818 	  }
   4819 
   4820       (*bufferHdr)->pBuffer = (OMX_U8*)drv_ctx.ptr_outputbuffer[i].bufferaddr;
   4821       (*bufferHdr)->pAppPrivate = appData;
   4822       BITMASK_SET(&m_out_bm_count,i);
   4823     }
   4824     else
   4825     {
   4826       DEBUG_PRINT_ERROR("All the Output Buffers have been Allocated ; Returning Insufficient \n");
   4827       eRet = OMX_ErrorInsufficientResources;
   4828     }
   4829   }
   4830 
   4831   return eRet;
   4832 }
   4833 
   4834 
   4835 // AllocateBuffer  -- API Call
   4836 /* ======================================================================
   4837 FUNCTION
   4838   omx_vdec::AllocateBuffer
   4839 
   4840 DESCRIPTION
   4841   Returns zero if all the buffers released..
   4842 
   4843 PARAMETERS
   4844   None.
   4845 
   4846 RETURN VALUE
   4847   true/false
   4848 
   4849 ========================================================================== */
   4850 OMX_ERRORTYPE  omx_vdec::allocate_buffer(OMX_IN OMX_HANDLETYPE                hComp,
   4851                                      OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
   4852                                      OMX_IN OMX_U32                        port,
   4853                                      OMX_IN OMX_PTR                     appData,
   4854                                      OMX_IN OMX_U32                       bytes)
   4855 {
   4856     unsigned i = 0;
   4857     OMX_ERRORTYPE eRet = OMX_ErrorNone; // OMX return type
   4858 
   4859     DEBUG_PRINT_LOW("\n Allocate buffer on port %d \n", (int)port);
   4860     if(m_state == OMX_StateInvalid)
   4861     {
   4862         DEBUG_PRINT_ERROR("Allocate Buf in Invalid State\n");
   4863         return OMX_ErrorInvalidState;
   4864     }
   4865 
   4866     if(port == OMX_CORE_INPUT_PORT_INDEX)
   4867     {
   4868       if (arbitrary_bytes)
   4869       {
   4870           eRet = allocate_input_heap_buffer (hComp,bufferHdr,port,appData,bytes);
   4871       }
   4872       else
   4873       {
   4874         eRet = allocate_input_buffer(hComp,bufferHdr,port,appData,bytes);
   4875       }
   4876     }
   4877     else if(port == OMX_CORE_OUTPUT_PORT_INDEX)
   4878     {
   4879       eRet = allocate_output_buffer(hComp,bufferHdr,port,appData,bytes);
   4880     }
   4881     else
   4882     {
   4883       DEBUG_PRINT_ERROR("Error: Invalid Port Index received %d\n",(int)port);
   4884       eRet = OMX_ErrorBadPortIndex;
   4885     }
   4886     DEBUG_PRINT_LOW("Checking for Output Allocate buffer Done");
   4887     if(eRet == OMX_ErrorNone)
   4888     {
   4889         if(allocate_done()){
   4890             if(BITMASK_PRESENT(&m_flags,OMX_COMPONENT_IDLE_PENDING))
   4891             {
   4892                 // Send the callback now
   4893                 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_IDLE_PENDING);
   4894                 post_event(OMX_CommandStateSet,OMX_StateIdle,
   4895                                    OMX_COMPONENT_GENERATE_EVENT);
   4896             }
   4897         }
   4898         if(port == OMX_CORE_INPUT_PORT_INDEX && m_inp_bPopulated)
   4899         {
   4900           if(BITMASK_PRESENT(&m_flags,OMX_COMPONENT_INPUT_ENABLE_PENDING))
   4901           {
   4902              BITMASK_CLEAR((&m_flags),OMX_COMPONENT_INPUT_ENABLE_PENDING);
   4903              post_event(OMX_CommandPortEnable,
   4904                         OMX_CORE_INPUT_PORT_INDEX,
   4905                         OMX_COMPONENT_GENERATE_EVENT);
   4906           }
   4907         }
   4908         if(port == OMX_CORE_OUTPUT_PORT_INDEX && m_out_bPopulated)
   4909             {
   4910           if(BITMASK_PRESENT(&m_flags,OMX_COMPONENT_OUTPUT_ENABLE_PENDING))
   4911           {
   4912              BITMASK_CLEAR((&m_flags),OMX_COMPONENT_OUTPUT_ENABLE_PENDING);
   4913                 post_event(OMX_CommandPortEnable,
   4914                            OMX_CORE_OUTPUT_PORT_INDEX,
   4915                            OMX_COMPONENT_GENERATE_EVENT);
   4916             }
   4917         }
   4918     }
   4919     DEBUG_PRINT_LOW("Allocate Buffer exit with ret Code %d\n",eRet);
   4920     return eRet;
   4921 }
   4922 
   4923 // Free Buffer - API call
   4924 /* ======================================================================
   4925 FUNCTION
   4926   omx_vdec::FreeBuffer
   4927 
   4928 DESCRIPTION
   4929 
   4930 PARAMETERS
   4931   None.
   4932 
   4933 RETURN VALUE
   4934   true/false
   4935 
   4936 ========================================================================== */
   4937 OMX_ERRORTYPE  omx_vdec::free_buffer(OMX_IN OMX_HANDLETYPE         hComp,
   4938                                       OMX_IN OMX_U32                 port,
   4939                                       OMX_IN OMX_BUFFERHEADERTYPE* buffer)
   4940 {
   4941     OMX_ERRORTYPE eRet = OMX_ErrorNone;
   4942     unsigned int nPortIndex;
   4943     DEBUG_PRINT_LOW("In for decoder free_buffer \n");
   4944 
   4945     if(m_state == OMX_StateIdle &&
   4946        (BITMASK_PRESENT(&m_flags ,OMX_COMPONENT_LOADING_PENDING)))
   4947     {
   4948         DEBUG_PRINT_LOW(" free buffer while Component in Loading pending\n");
   4949     }
   4950     else if((m_inp_bEnabled == OMX_FALSE && port == OMX_CORE_INPUT_PORT_INDEX)||
   4951             (m_out_bEnabled == OMX_FALSE && port == OMX_CORE_OUTPUT_PORT_INDEX))
   4952     {
   4953         DEBUG_PRINT_LOW("Free Buffer while port %d disabled\n", port);
   4954     }
   4955     else if(m_state == OMX_StateExecuting || m_state == OMX_StatePause)
   4956     {
   4957         DEBUG_PRINT_ERROR("Invalid state to free buffer,ports need to be disabled\n");
   4958         post_event(OMX_EventError,
   4959                    OMX_ErrorPortUnpopulated,
   4960                    OMX_COMPONENT_GENERATE_EVENT);
   4961 
   4962         return OMX_ErrorIncorrectStateOperation;
   4963     }
   4964     else if (m_state != OMX_StateInvalid)
   4965     {
   4966         DEBUG_PRINT_ERROR("Invalid state to free buffer,port lost Buffers\n");
   4967         post_event(OMX_EventError,
   4968                    OMX_ErrorPortUnpopulated,
   4969                    OMX_COMPONENT_GENERATE_EVENT);
   4970     }
   4971 
   4972     if(port == OMX_CORE_INPUT_PORT_INDEX)
   4973     {
   4974       /*Check if arbitrary bytes*/
   4975       if(!arbitrary_bytes && !input_use_buffer)
   4976         nPortIndex = buffer - m_inp_mem_ptr;
   4977       else
   4978         nPortIndex = buffer - m_inp_heap_ptr;
   4979 
   4980         DEBUG_PRINT_LOW("free_buffer on i/p port - Port idx %d \n", nPortIndex);
   4981         if(nPortIndex < drv_ctx.ip_buf.actualcount)
   4982         {
   4983          // Clear the bit associated with it.
   4984          BITMASK_CLEAR(&m_inp_bm_count,nPortIndex);
   4985          BITMASK_CLEAR(&m_heap_inp_bm_count,nPortIndex);
   4986          if (input_use_buffer == true)
   4987          {
   4988 
   4989             DEBUG_PRINT_LOW("\n Free pmem Buffer index %d",nPortIndex);
   4990             if(m_phdr_pmem_ptr)
   4991               free_input_buffer(m_phdr_pmem_ptr[nPortIndex]);
   4992          }
   4993          else
   4994          {
   4995             if (arbitrary_bytes)
   4996             {
   4997               if(m_phdr_pmem_ptr)
   4998                 free_input_buffer(nPortIndex,m_phdr_pmem_ptr[nPortIndex]);
   4999               else
   5000                 free_input_buffer(nPortIndex,NULL);
   5001             }
   5002             else
   5003               free_input_buffer(buffer);
   5004          }
   5005          m_inp_bPopulated = OMX_FALSE;
   5006          /*Free the Buffer Header*/
   5007           if (release_input_done())
   5008           {
   5009             DEBUG_PRINT_HIGH("\n ALL input buffers are freed/released");
   5010             free_input_buffer_header();
   5011           }
   5012         }
   5013         else
   5014         {
   5015             DEBUG_PRINT_ERROR("Error: free_buffer ,Port Index Invalid\n");
   5016             eRet = OMX_ErrorBadPortIndex;
   5017         }
   5018 
   5019         if(BITMASK_PRESENT((&m_flags),OMX_COMPONENT_INPUT_DISABLE_PENDING)
   5020            && release_input_done())
   5021         {
   5022             DEBUG_PRINT_LOW("MOVING TO DISABLED STATE \n");
   5023             BITMASK_CLEAR((&m_flags),OMX_COMPONENT_INPUT_DISABLE_PENDING);
   5024             post_event(OMX_CommandPortDisable,
   5025                        OMX_CORE_INPUT_PORT_INDEX,
   5026                        OMX_COMPONENT_GENERATE_EVENT);
   5027         }
   5028     }
   5029     else if(port == OMX_CORE_OUTPUT_PORT_INDEX)
   5030     {
   5031         // check if the buffer is valid
   5032         nPortIndex = buffer - (OMX_BUFFERHEADERTYPE*)m_out_mem_ptr;
   5033         if(nPortIndex < drv_ctx.op_buf.actualcount)
   5034         {
   5035             DEBUG_PRINT_LOW("free_buffer on o/p port - Port idx %d \n", nPortIndex);
   5036             // Clear the bit associated with it.
   5037             BITMASK_CLEAR(&m_out_bm_count,nPortIndex);
   5038             m_out_bPopulated = OMX_FALSE;
   5039             free_output_buffer (buffer);
   5040 
   5041             if (release_output_done())
   5042             {
   5043               free_output_buffer_header();
   5044             }
   5045         }
   5046         else
   5047         {
   5048             DEBUG_PRINT_ERROR("Error: free_buffer , Port Index Invalid\n");
   5049             eRet = OMX_ErrorBadPortIndex;
   5050         }
   5051         if(BITMASK_PRESENT((&m_flags),OMX_COMPONENT_OUTPUT_DISABLE_PENDING)
   5052            && release_output_done())
   5053         {
   5054             DEBUG_PRINT_LOW("FreeBuffer : If any Disable event pending,post it\n");
   5055 
   5056                 DEBUG_PRINT_LOW("MOVING TO DISABLED STATE \n");
   5057                 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_OUTPUT_DISABLE_PENDING);
   5058 
   5059                 post_event(OMX_CommandPortDisable,
   5060                            OMX_CORE_OUTPUT_PORT_INDEX,
   5061                            OMX_COMPONENT_GENERATE_EVENT);
   5062         }
   5063     }
   5064     else
   5065     {
   5066         eRet = OMX_ErrorBadPortIndex;
   5067     }
   5068     if((eRet == OMX_ErrorNone) &&
   5069        (BITMASK_PRESENT(&m_flags ,OMX_COMPONENT_LOADING_PENDING)))
   5070     {
   5071         if(release_done())
   5072         {
   5073             // Send the callback now
   5074             BITMASK_CLEAR((&m_flags),OMX_COMPONENT_LOADING_PENDING);
   5075             post_event(OMX_CommandStateSet, OMX_StateLoaded,
   5076                                       OMX_COMPONENT_GENERATE_EVENT);
   5077         }
   5078     }
   5079     return eRet;
   5080 }
   5081 
   5082 
   5083 /* ======================================================================
   5084 FUNCTION
   5085   omx_vdec::EmptyThisBuffer
   5086 
   5087 DESCRIPTION
   5088   This routine is used to push the encoded video frames to
   5089   the video decoder.
   5090 
   5091 PARAMETERS
   5092   None.
   5093 
   5094 RETURN VALUE
   5095   OMX Error None if everything went successful.
   5096 
   5097 ========================================================================== */
   5098 OMX_ERRORTYPE  omx_vdec::empty_this_buffer(OMX_IN OMX_HANDLETYPE         hComp,
   5099                                            OMX_IN OMX_BUFFERHEADERTYPE* buffer)
   5100 {
   5101   OMX_ERRORTYPE ret1 = OMX_ErrorNone;
   5102   unsigned int nBufferIndex = drv_ctx.ip_buf.actualcount;
   5103 
   5104   if(m_state == OMX_StateInvalid)
   5105   {
   5106       DEBUG_PRINT_ERROR("Empty this buffer in Invalid State\n");
   5107       return OMX_ErrorInvalidState;
   5108   }
   5109 
   5110   if (buffer == NULL)
   5111   {
   5112     DEBUG_PRINT_ERROR("\nERROR:ETB Buffer is NULL");
   5113     return OMX_ErrorBadParameter;
   5114   }
   5115 
   5116   if (!m_inp_bEnabled)
   5117   {
   5118     DEBUG_PRINT_ERROR("\nERROR:ETB incorrect state operation, input port is disabled.");
   5119     return OMX_ErrorIncorrectStateOperation;
   5120   }
   5121 
   5122   if (buffer->nInputPortIndex != OMX_CORE_INPUT_PORT_INDEX)
   5123   {
   5124     DEBUG_PRINT_ERROR("\nERROR:ETB invalid port in header %d", buffer->nInputPortIndex);
   5125     return OMX_ErrorBadPortIndex;
   5126   }
   5127 
   5128 #ifdef _ANDROID_
   5129   if(iDivXDrmDecrypt)
   5130   {
   5131     OMX_ERRORTYPE drmErr = iDivXDrmDecrypt->Decrypt(buffer);
   5132     if(drmErr != OMX_ErrorNone) {
   5133         // this error can be ignored
   5134         DEBUG_PRINT_LOW("\nERROR:iDivXDrmDecrypt->Decrypt %d", drmErr);
   5135     }
   5136   }
   5137 #endif //_ANDROID_
   5138   if (perf_flag)
   5139   {
   5140     if (!latency)
   5141     {
   5142       dec_time.stop();
   5143       latency = dec_time.processing_time_us();
   5144       dec_time.start();
   5145     }
   5146   }
   5147 
   5148   if (arbitrary_bytes)
   5149   {
   5150     nBufferIndex = buffer - m_inp_heap_ptr;
   5151   }
   5152   else
   5153   {
   5154      if (input_use_buffer == true)
   5155      {
   5156        nBufferIndex = buffer - m_inp_heap_ptr;
   5157        m_inp_mem_ptr[nBufferIndex].nFilledLen = m_inp_heap_ptr[nBufferIndex].nFilledLen;
   5158        m_inp_mem_ptr[nBufferIndex].nTimeStamp = m_inp_heap_ptr[nBufferIndex].nTimeStamp;
   5159        m_inp_mem_ptr[nBufferIndex].nFlags = m_inp_heap_ptr[nBufferIndex].nFlags;
   5160        buffer = &m_inp_mem_ptr[nBufferIndex];
   5161        DEBUG_PRINT_LOW("Non-Arbitrary mode - buffer address is: malloc %p, pmem%p in Index %d, buffer %p of size %d",
   5162                          &m_inp_heap_ptr[nBufferIndex], &m_inp_mem_ptr[nBufferIndex],nBufferIndex, buffer, buffer->nFilledLen);
   5163      }
   5164      else{
   5165        nBufferIndex = buffer - m_inp_mem_ptr;
   5166      }
   5167   }
   5168 
   5169   if (nBufferIndex > drv_ctx.ip_buf.actualcount )
   5170   {
   5171     DEBUG_PRINT_ERROR("\nERROR:ETB nBufferIndex is invalid");
   5172     return OMX_ErrorBadParameter;
   5173   }
   5174 
   5175   DEBUG_PRINT_LOW("[ETB] BHdr(%p) pBuf(%p) nTS(%lld) nFL(%lu)",
   5176     buffer, buffer->pBuffer, buffer->nTimeStamp, buffer->nFilledLen);
   5177   if (arbitrary_bytes)
   5178   {
   5179     post_event ((unsigned)hComp,(unsigned)buffer,
   5180                 OMX_COMPONENT_GENERATE_ETB_ARBITRARY);
   5181   }
   5182   else
   5183   {
   5184     if (!(client_extradata & OMX_TIMEINFO_EXTRADATA))
   5185       set_frame_rate(buffer->nTimeStamp);
   5186     post_event ((unsigned)hComp,(unsigned)buffer,OMX_COMPONENT_GENERATE_ETB);
   5187   }
   5188   return OMX_ErrorNone;
   5189 }
   5190 
   5191 /* ======================================================================
   5192 FUNCTION
   5193   omx_vdec::empty_this_buffer_proxy
   5194 
   5195 DESCRIPTION
   5196   This routine is used to push the encoded video frames to
   5197   the video decoder.
   5198 
   5199 PARAMETERS
   5200   None.
   5201 
   5202 RETURN VALUE
   5203   OMX Error None if everything went successful.
   5204 
   5205 ========================================================================== */
   5206 OMX_ERRORTYPE  omx_vdec::empty_this_buffer_proxy(OMX_IN OMX_HANDLETYPE         hComp,
   5207                                                  OMX_IN OMX_BUFFERHEADERTYPE* buffer)
   5208 {
   5209   int push_cnt = 0,i=0;
   5210   unsigned nPortIndex = 0;
   5211   OMX_ERRORTYPE ret = OMX_ErrorNone;
   5212   struct vdec_input_frameinfo frameinfo;
   5213   struct vdec_bufferpayload *temp_buffer;
   5214   struct vdec_ioctl_msg ioctl_msg;
   5215   struct vdec_seqheader seq_header;
   5216   bool port_setting_changed = true;
   5217   bool not_coded_vop = false;
   5218 
   5219   /*Should we generate a Aync error event*/
   5220   if (buffer == NULL || buffer->pInputPortPrivate == NULL)
   5221   {
   5222     DEBUG_PRINT_ERROR("\nERROR:empty_this_buffer_proxy is invalid");
   5223     return OMX_ErrorBadParameter;
   5224   }
   5225 
   5226   nPortIndex = buffer-((OMX_BUFFERHEADERTYPE *)m_inp_mem_ptr);
   5227 
   5228   if (nPortIndex > drv_ctx.ip_buf.actualcount)
   5229   {
   5230     DEBUG_PRINT_ERROR("\nERROR:empty_this_buffer_proxy invalid nPortIndex[%u]",
   5231         nPortIndex);
   5232     return OMX_ErrorBadParameter;
   5233   }
   5234 
   5235   pending_input_buffers++;
   5236 
   5237   /* return zero length and not an EOS buffer */
   5238   if (!arbitrary_bytes && (buffer->nFilledLen == 0) &&
   5239      ((buffer->nFlags & OMX_BUFFERFLAG_EOS) == 0))
   5240   {
   5241     DEBUG_PRINT_HIGH("\n return zero legth buffer");
   5242     post_event ((unsigned int)buffer,VDEC_S_SUCCESS,
   5243                      OMX_COMPONENT_GENERATE_EBD);
   5244     return OMX_ErrorNone;
   5245   }
   5246 
   5247 
   5248   if(codec_type_parse == CODEC_TYPE_MPEG4 || codec_type_parse == CODEC_TYPE_DIVX){
   5249     mp4StreamType psBits;
   5250     psBits.data = (unsigned char *)(buffer->pBuffer + buffer->nOffset);
   5251     psBits.numBytes = buffer->nFilledLen;
   5252     mp4_headerparser.parseHeader(&psBits);
   5253     not_coded_vop = mp4_headerparser.is_notcodec_vop(
   5254             (buffer->pBuffer + buffer->nOffset),buffer->nFilledLen);
   5255     if(not_coded_vop) {
   5256         DEBUG_PRINT_HIGH("\n Found Not coded vop len %d frame number %d",
   5257              buffer->nFilledLen,frame_count);
   5258         if(buffer->nFlags & OMX_BUFFERFLAG_EOS){
   5259           DEBUG_PRINT_HIGH("\n Eos and Not coded Vop set len to zero");
   5260           not_coded_vop = false;
   5261           buffer->nFilledLen = 0;
   5262         }
   5263     }
   5264   }
   5265 
   5266   if(input_flush_progress == true
   5267 
   5268      || not_coded_vop
   5269 
   5270      )
   5271   {
   5272     DEBUG_PRINT_LOW("\n Flush in progress return buffer ");
   5273     post_event ((unsigned int)buffer,VDEC_S_SUCCESS,
   5274                      OMX_COMPONENT_GENERATE_EBD);
   5275     return OMX_ErrorNone;
   5276   }
   5277 
   5278   temp_buffer = (struct vdec_bufferpayload *)buffer->pInputPortPrivate;
   5279 
   5280   if ((temp_buffer -  drv_ctx.ptr_inputbuffer) > drv_ctx.ip_buf.actualcount)
   5281   {
   5282     return OMX_ErrorBadParameter;
   5283   }
   5284 
   5285   DEBUG_PRINT_LOW("\n ETBProxy: bufhdr = %p, bufhdr->pBuffer = %p", buffer, buffer->pBuffer);
   5286   /*for use buffer we need to memcpy the data*/
   5287   temp_buffer->buffer_len = buffer->nFilledLen;
   5288 
   5289   if (input_use_buffer)
   5290   {
   5291     if (buffer->nFilledLen <= temp_buffer->buffer_len)
   5292     {
   5293       if(arbitrary_bytes)
   5294       {
   5295         memcpy (temp_buffer->bufferaddr, (buffer->pBuffer + buffer->nOffset),buffer->nFilledLen);
   5296       }
   5297       else
   5298       {
   5299         memcpy (temp_buffer->bufferaddr, (m_inp_heap_ptr[nPortIndex].pBuffer + m_inp_heap_ptr[nPortIndex].nOffset),
   5300                 buffer->nFilledLen);
   5301       }
   5302     }
   5303     else
   5304     {
   5305       return OMX_ErrorBadParameter;
   5306     }
   5307 
   5308   }
   5309 
   5310   frameinfo.bufferaddr = temp_buffer->bufferaddr;
   5311   frameinfo.client_data = (void *) buffer;
   5312   frameinfo.datalen = temp_buffer->buffer_len;
   5313   frameinfo.flags = 0;
   5314   frameinfo.offset = buffer->nOffset;
   5315   frameinfo.pmem_fd = temp_buffer->pmem_fd;
   5316   frameinfo.pmem_offset = temp_buffer->offset;
   5317   frameinfo.timestamp = buffer->nTimeStamp;
   5318   if (drv_ctx.disable_dmx && m_desc_buffer_ptr && m_desc_buffer_ptr[nPortIndex].buf_addr)
   5319   {
   5320     DEBUG_PRINT_LOW("ETB: dmx enabled");
   5321     if (m_demux_entries == 0)
   5322     {
   5323       extract_demux_addr_offsets(buffer);
   5324     }
   5325 
   5326     DEBUG_PRINT_LOW("ETB: handle_demux_data - entries=%d",m_demux_entries);
   5327     handle_demux_data(buffer);
   5328     frameinfo.desc_addr = (OMX_U8 *)m_desc_buffer_ptr[nPortIndex].buf_addr;
   5329     frameinfo.desc_size = m_desc_buffer_ptr[nPortIndex].desc_data_size;
   5330   }
   5331   else
   5332   {
   5333     frameinfo.desc_addr = NULL;
   5334     frameinfo.desc_size = 0;
   5335   }
   5336   if(!arbitrary_bytes)
   5337   {
   5338       frameinfo.flags |= buffer->nFlags;
   5339   }
   5340 
   5341 #ifdef _ANDROID_
   5342   if (m_debug_timestamp)
   5343   {
   5344     if(arbitrary_bytes)
   5345     {
   5346       DEBUG_PRINT_LOW("\n Inserting TIMESTAMP (%lld) into queue", buffer->nTimeStamp);
   5347       m_timestamp_list.insert_ts(buffer->nTimeStamp);
   5348     }
   5349     else if(!arbitrary_bytes && !(buffer->nFlags & OMX_BUFFERFLAG_CODECCONFIG))
   5350     {
   5351       DEBUG_PRINT_LOW("\n Inserting TIMESTAMP (%lld) into queue", buffer->nTimeStamp);
   5352       m_timestamp_list.insert_ts(buffer->nTimeStamp);
   5353     }
   5354   }
   5355 #endif
   5356 
   5357 #ifdef INPUT_BUFFER_LOG
   5358   if (inputBufferFile1)
   5359   {
   5360     fwrite((const char *)temp_buffer->bufferaddr,
   5361       temp_buffer->buffer_len,1,inputBufferFile1);
   5362   }
   5363 #endif
   5364 
   5365   if(buffer->nFlags & QOMX_VIDEO_BUFFERFLAG_EOSEQ)
   5366   {
   5367     frameinfo.flags |= QOMX_VIDEO_BUFFERFLAG_EOSEQ;
   5368     buffer->nFlags &= ~QOMX_VIDEO_BUFFERFLAG_EOSEQ;
   5369   }
   5370 
   5371   if (temp_buffer->buffer_len == 0 || (buffer->nFlags & OMX_BUFFERFLAG_EOS))
   5372   {
   5373     DEBUG_PRINT_HIGH("\n Rxd i/p EOS, Notify Driver that EOS has been reached");
   5374     frameinfo.flags |= VDEC_BUFFERFLAG_EOS;
   5375     h264_scratch.nFilledLen = 0;
   5376     nal_count = 0;
   5377     look_ahead_nal = false;
   5378     frame_count = 0;
   5379     if (m_frame_parser.mutils)
   5380       m_frame_parser.mutils->initialize_frame_checking_environment();
   5381     m_frame_parser.flush();
   5382     h264_last_au_ts = LLONG_MAX;
   5383     h264_last_au_flags = 0;
   5384     memset(m_demux_offsets, 0, ( sizeof(OMX_U32) * 8192) );
   5385     m_demux_entries = 0;
   5386   }
   5387     struct v4l2_buffer buf = {0};
   5388 	struct v4l2_plane plane;
   5389 	int rc;
   5390 	unsigned long  print_count;
   5391   if (temp_buffer->buffer_len == 0 || (buffer->nFlags & OMX_BUFFERFLAG_EOS))
   5392   {  buf.flags = V4L2_BUF_FLAG_EOS;
   5393   printf("\n  INPUT EOS reached \n") ;
   5394   }
   5395 	OMX_ERRORTYPE eRet = OMX_ErrorNone;
   5396 	buf.index = nPortIndex;
   5397 	buf.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
   5398 	buf.memory = V4L2_MEMORY_USERPTR;
   5399 	plane.bytesused = temp_buffer->buffer_len;
   5400 	plane.length = drv_ctx.ip_buf.buffer_size;
   5401 	plane.m.userptr = (unsigned long)(temp_buffer->bufferaddr-temp_buffer->offset);
   5402 	plane.reserved[0] = temp_buffer->pmem_fd;
   5403 	plane.reserved[1] = temp_buffer->offset;
   5404 	plane.data_offset = 0;
   5405 	buf.m.planes = &plane;
   5406 	buf.length = 1;
   5407 	rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_QBUF, &buf);
   5408   if(!streaming[OUTPUT_PORT])
   5409   {
   5410 	enum v4l2_buf_type buf_type;
   5411 	int ret,r;
   5412 	buf_type=V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
   5413         DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Executing\n");
   5414 	ret=ioctl(drv_ctx.video_driver_fd, VIDIOC_STREAMON,&buf_type);
   5415 	if(!ret) {
   5416 		printf("Streamon on OUTPUT Plane was successful \n");
   5417 		streaming[OUTPUT_PORT] = true;
   5418 		ret = pthread_create(&async_thread_id,0,async_message_thread,this);
   5419 		if(ret < 0)
   5420 			printf("\n Failed to create async_message_thread \n");
   5421 	} else{
   5422 		/*TODO: How to handle this case */
   5423 		printf(" \n Failed to call streamon on OUTPUT \n");
   5424 	}
   5425 }
   5426   DEBUG_PRINT_LOW("[ETBP] pBuf(%p) nTS(%lld) Sz(%d)",
   5427     frameinfo.bufferaddr, frameinfo.timestamp, frameinfo.datalen);
   5428       time_stamp_dts.insert_timestamp(buffer);
   5429 
   5430   return ret;
   5431 }
   5432 
   5433 /* ======================================================================
   5434 FUNCTION
   5435   omx_vdec::FillThisBuffer
   5436 
   5437 DESCRIPTION
   5438   IL client uses this method to release the frame buffer
   5439   after displaying them.
   5440 
   5441 PARAMETERS
   5442   None.
   5443 
   5444 RETURN VALUE
   5445   true/false
   5446 
   5447 ========================================================================== */
   5448 OMX_ERRORTYPE  omx_vdec::fill_this_buffer(OMX_IN OMX_HANDLETYPE  hComp,
   5449                                           OMX_IN OMX_BUFFERHEADERTYPE* buffer)
   5450 {
   5451 
   5452   if(m_state == OMX_StateInvalid)
   5453   {
   5454       DEBUG_PRINT_ERROR("FTB in Invalid State\n");
   5455       return OMX_ErrorInvalidState;
   5456   }
   5457 
   5458   if (!m_out_bEnabled)
   5459   {
   5460     DEBUG_PRINT_ERROR("\nERROR:FTB incorrect state operation, output port is disabled.");
   5461     return OMX_ErrorIncorrectStateOperation;
   5462   }
   5463 
   5464   if (buffer == NULL || ((buffer - m_out_mem_ptr) >= drv_ctx.op_buf.actualcount))
   5465   {
   5466     return OMX_ErrorBadParameter;
   5467   }
   5468 
   5469   if (buffer->nOutputPortIndex != OMX_CORE_OUTPUT_PORT_INDEX)
   5470   {
   5471     DEBUG_PRINT_ERROR("\nERROR:FTB invalid port in header %d", buffer->nOutputPortIndex);
   5472     return OMX_ErrorBadPortIndex;
   5473   }
   5474 
   5475   DEBUG_PRINT_LOW("[FTB] bufhdr = %p, bufhdr->pBuffer = %p", buffer, buffer->pBuffer);
   5476   post_event((unsigned) hComp, (unsigned)buffer,OMX_COMPONENT_GENERATE_FTB);
   5477   return OMX_ErrorNone;
   5478 }
   5479 /* ======================================================================
   5480 FUNCTION
   5481   omx_vdec::fill_this_buffer_proxy
   5482 
   5483 DESCRIPTION
   5484   IL client uses this method to release the frame buffer
   5485   after displaying them.
   5486 
   5487 PARAMETERS
   5488   None.
   5489 
   5490 RETURN VALUE
   5491   true/false
   5492 
   5493 ========================================================================== */
   5494 OMX_ERRORTYPE  omx_vdec::fill_this_buffer_proxy(
   5495                          OMX_IN OMX_HANDLETYPE        hComp,
   5496                          OMX_IN OMX_BUFFERHEADERTYPE* bufferAdd)
   5497 {
   5498   OMX_ERRORTYPE nRet = OMX_ErrorNone;
   5499   struct vdec_ioctl_msg ioctl_msg = {NULL,NULL};
   5500   OMX_BUFFERHEADERTYPE *buffer = bufferAdd;
   5501   unsigned nPortIndex = 0;
   5502   struct vdec_fillbuffer_cmd fillbuffer;
   5503   struct vdec_bufferpayload     *ptr_outputbuffer = NULL;
   5504   struct vdec_output_frameinfo  *ptr_respbuffer = NULL;
   5505 
   5506   nPortIndex = buffer-((OMX_BUFFERHEADERTYPE *)m_out_mem_ptr);
   5507 
   5508   if (bufferAdd == NULL || ((buffer - m_out_mem_ptr) >
   5509       drv_ctx.op_buf.actualcount) )
   5510     return OMX_ErrorBadParameter;
   5511 
   5512   DEBUG_PRINT_LOW("\n FTBProxy: bufhdr = %p, bufhdr->pBuffer = %p",
   5513       bufferAdd, bufferAdd->pBuffer);
   5514   /*Return back the output buffer to client*/
   5515   if(m_out_bEnabled != OMX_TRUE || output_flush_progress == true)
   5516   {
   5517     DEBUG_PRINT_LOW("\n Output Buffers return flush/disable condition");
   5518     buffer->nFilledLen = 0;
   5519     m_cb.FillBufferDone (hComp,m_app_data,buffer);
   5520     return OMX_ErrorNone;
   5521   }
   5522   pending_output_buffers++;
   5523   ptr_respbuffer = (struct vdec_output_frameinfo*)buffer->pOutputPortPrivate;
   5524   if (ptr_respbuffer)
   5525   {
   5526     ptr_outputbuffer =  (struct vdec_bufferpayload*)ptr_respbuffer->client_data;
   5527   }
   5528 
   5529   if (ptr_respbuffer == NULL || ptr_outputbuffer == NULL)
   5530   {
   5531       DEBUG_PRINT_ERROR("resp buffer or outputbuffer is NULL");
   5532       buffer->nFilledLen = 0;
   5533       m_cb.FillBufferDone (hComp,m_app_data,buffer);
   5534       pending_output_buffers--;
   5535       return OMX_ErrorBadParameter;
   5536   }
   5537 
   5538  // memcpy (&fillbuffer.buffer,ptr_outputbuffer,\
   5539           sizeof(struct vdec_bufferpayload));
   5540  // fillbuffer.client_data = bufferAdd;
   5541 
   5542 	int rc = 0;
   5543 	struct v4l2_buffer buf={0};
   5544 	struct v4l2_plane plane;
   5545 
   5546 	buf.index = nPortIndex;
   5547 	buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   5548 	buf.memory = V4L2_MEMORY_USERPTR;
   5549 	plane.bytesused = buffer->nFilledLen;
   5550 	plane.length = drv_ctx.op_buf.buffer_size;
   5551 	plane.m.userptr = (unsigned long)(drv_ctx.ptr_outputbuffer[nPortIndex].bufferaddr-drv_ctx.ptr_outputbuffer[nPortIndex].offset);
   5552 	plane.reserved[0] = drv_ctx.ptr_outputbuffer[nPortIndex].pmem_fd;
   5553 	plane.reserved[1] = drv_ctx.ptr_outputbuffer[nPortIndex].offset;
   5554 	plane.data_offset = 0;
   5555 	buf.m.planes = &plane;
   5556 	buf.length = 1;
   5557 
   5558 	rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_QBUF, &buf);
   5559 	if (rc) {
   5560 		/*TODO: How to handle this case */
   5561 		printf("Failed to qbuf to driver");
   5562 	}
   5563     //m_cb.FillBufferDone (hComp,m_app_data,buffer);
   5564    // pending_output_buffers--;
   5565    // return OMX_ErrorBadParameter;
   5566   //}
   5567   return OMX_ErrorNone;
   5568 }
   5569 
   5570 /* ======================================================================
   5571 FUNCTION
   5572   omx_vdec::SetCallbacks
   5573 
   5574 DESCRIPTION
   5575   Set the callbacks.
   5576 
   5577 PARAMETERS
   5578   None.
   5579 
   5580 RETURN VALUE
   5581   OMX Error None if everything successful.
   5582 
   5583 ========================================================================== */
   5584 OMX_ERRORTYPE  omx_vdec::set_callbacks(OMX_IN OMX_HANDLETYPE        hComp,
   5585                                            OMX_IN OMX_CALLBACKTYPE* callbacks,
   5586                                            OMX_IN OMX_PTR             appData)
   5587 {
   5588 
   5589   m_cb       = *callbacks;
   5590   DEBUG_PRINT_LOW("\n Callbacks Set %p %p %p",m_cb.EmptyBufferDone,\
   5591                m_cb.EventHandler,m_cb.FillBufferDone);
   5592   m_app_data =    appData;
   5593   return OMX_ErrorNotImplemented;
   5594 }
   5595 
   5596 /* ======================================================================
   5597 FUNCTION
   5598   omx_vdec::ComponentDeInit
   5599 
   5600 DESCRIPTION
   5601   Destroys the component and release memory allocated to the heap.
   5602 
   5603 PARAMETERS
   5604   <TBD>.
   5605 
   5606 RETURN VALUE
   5607   OMX Error None if everything successful.
   5608 
   5609 ========================================================================== */
   5610 OMX_ERRORTYPE  omx_vdec::component_deinit(OMX_IN OMX_HANDLETYPE hComp)
   5611 {
   5612 #ifdef _ANDROID_
   5613     if(iDivXDrmDecrypt)
   5614     {
   5615         delete iDivXDrmDecrypt;
   5616         iDivXDrmDecrypt=NULL;
   5617     }
   5618 #endif //_ANDROID_
   5619 
   5620     int i = 0;
   5621     if (OMX_StateLoaded != m_state)
   5622     {
   5623         DEBUG_PRINT_ERROR("WARNING:Rxd DeInit,OMX not in LOADED state %d\n",\
   5624                           m_state);
   5625         DEBUG_PRINT_ERROR("\nPlayback Ended - FAILED");
   5626     }
   5627     else
   5628     {
   5629       DEBUG_PRINT_HIGH("\n Playback Ended - PASSED");
   5630     }
   5631 
   5632     /*Check if the output buffers have to be cleaned up*/
   5633     if(m_out_mem_ptr)
   5634     {
   5635         DEBUG_PRINT_LOW("Freeing the Output Memory\n");
   5636         for (i=0; i < drv_ctx.op_buf.actualcount; i++ )
   5637         {
   5638           free_output_buffer (&m_out_mem_ptr[i]);
   5639         }
   5640     }
   5641 
   5642     /*Check if the input buffers have to be cleaned up*/
   5643     if(m_inp_mem_ptr || m_inp_heap_ptr)
   5644     {
   5645         DEBUG_PRINT_LOW("Freeing the Input Memory\n");
   5646         for (i=0; i<drv_ctx.ip_buf.actualcount; i++ )
   5647         {
   5648           if (m_inp_mem_ptr)
   5649             free_input_buffer (i,&m_inp_mem_ptr[i]);
   5650           else
   5651             free_input_buffer (i,NULL);
   5652         }
   5653     }
   5654     free_input_buffer_header();
   5655     free_output_buffer_header();
   5656     if(h264_scratch.pBuffer)
   5657     {
   5658         free(h264_scratch.pBuffer);
   5659         h264_scratch.pBuffer = NULL;
   5660     }
   5661 
   5662     if (h264_parser)
   5663     {
   5664         delete h264_parser;
   5665 	h264_parser = NULL;
   5666     }
   5667 
   5668     if(m_platform_list)
   5669     {
   5670         free(m_platform_list);
   5671         m_platform_list = NULL;
   5672     }
   5673     if(m_vendor_config.pData)
   5674     {
   5675         free(m_vendor_config.pData);
   5676         m_vendor_config.pData = NULL;
   5677     }
   5678 
   5679     // Reset counters in mesg queues
   5680     m_ftb_q.m_size=0;
   5681     m_cmd_q.m_size=0;
   5682     m_etb_q.m_size=0;
   5683     m_ftb_q.m_read = m_ftb_q.m_write =0;
   5684     m_cmd_q.m_read = m_cmd_q.m_write =0;
   5685     m_etb_q.m_read = m_etb_q.m_write =0;
   5686 #ifdef _ANDROID_
   5687     if (m_debug_timestamp)
   5688     {
   5689       m_timestamp_list.reset_ts_list();
   5690     }
   5691 #endif
   5692 
   5693     DEBUG_PRINT_LOW("\n Calling VDEC_IOCTL_STOP_NEXT_MSG");
   5694     //(void)ioctl(drv_ctx.video_driver_fd, VDEC_IOCTL_STOP_NEXT_MSG,
   5695        // NULL);
   5696     DEBUG_PRINT_HIGH("\n Close the driver instance");
   5697 
   5698 #ifdef INPUT_BUFFER_LOG
   5699     fclose (inputBufferFile1);
   5700 #endif
   5701 #ifdef OUTPUT_BUFFER_LOG
   5702     fclose (outputBufferFile1);
   5703 #endif
   5704 #ifdef OUTPUT_EXTRADATA_LOG
   5705     fclose (outputExtradataFile);
   5706 #endif
   5707   DEBUG_PRINT_HIGH("\n omx_vdec::component_deinit() complete");
   5708   return OMX_ErrorNone;
   5709 }
   5710 
   5711 /* ======================================================================
   5712 FUNCTION
   5713   omx_vdec::UseEGLImage
   5714 
   5715 DESCRIPTION
   5716   OMX Use EGL Image method implementation <TBD>.
   5717 
   5718 PARAMETERS
   5719   <TBD>.
   5720 
   5721 RETURN VALUE
   5722   Not Implemented error.
   5723 
   5724 ========================================================================== */
   5725 OMX_ERRORTYPE  omx_vdec::use_EGL_image(OMX_IN OMX_HANDLETYPE                hComp,
   5726                                           OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
   5727                                           OMX_IN OMX_U32                        port,
   5728                                           OMX_IN OMX_PTR                     appData,
   5729                                           OMX_IN void*                      eglImage)
   5730 {
   5731   OMX_QCOM_PLATFORM_PRIVATE_LIST pmem_list;
   5732   OMX_QCOM_PLATFORM_PRIVATE_ENTRY pmem_entry;
   5733   OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO pmem_info;
   5734 
   5735 #ifdef USE_EGL_IMAGE_GPU
   5736    PFNEGLQUERYIMAGEQUALCOMMPROC egl_queryfunc;
   5737    EGLint fd = -1, offset = 0,pmemPtr = 0;
   5738 #else
   5739    int fd = -1, offset = 0;
   5740 #endif
   5741    DEBUG_PRINT_HIGH("\nuse EGL image support for decoder");
   5742    if (!bufferHdr || !eglImage|| port != OMX_CORE_OUTPUT_PORT_INDEX) {
   5743      DEBUG_PRINT_ERROR("\n ");
   5744    }
   5745 #ifdef USE_EGL_IMAGE_GPU
   5746    if(m_display_id == NULL) {
   5747         DEBUG_PRINT_ERROR("Display ID is not set by IL client \n");
   5748         return OMX_ErrorInsufficientResources;
   5749    }
   5750    egl_queryfunc = (PFNEGLQUERYIMAGEQUALCOMMPROC)
   5751                     eglGetProcAddress("eglQueryImageKHR");
   5752    egl_queryfunc(m_display_id, eglImage, EGL_BUFFER_HANDLE_QCOM,&fd);
   5753    egl_queryfunc(m_display_id, eglImage, EGL_BUFFER_OFFSET_QCOM,&offset);
   5754    egl_queryfunc(m_display_id, eglImage, EGL_BITMAP_POINTER_KHR,&pmemPtr);
   5755 #else //with OMX test app
   5756     struct temp_egl {
   5757         int pmem_fd;
   5758         int offset;
   5759     };
   5760     struct temp_egl *temp_egl_id = NULL;
   5761     void * pmemPtr = (void *) eglImage;
   5762     temp_egl_id = (struct temp_egl *)eglImage;
   5763     if (temp_egl_id != NULL)
   5764     {
   5765         fd = temp_egl_id->pmem_fd;
   5766         offset = temp_egl_id->offset;
   5767     }
   5768 #endif
   5769     if (fd < 0) {
   5770         DEBUG_PRINT_ERROR("Improper pmem fd by EGL client %d  \n",fd);
   5771         return OMX_ErrorInsufficientResources;
   5772    }
   5773    pmem_info.pmem_fd = (OMX_U32) fd;
   5774    pmem_info.offset = (OMX_U32) offset;
   5775    pmem_entry.entry = (void *) &pmem_info;
   5776    pmem_entry.type = OMX_QCOM_PLATFORM_PRIVATE_PMEM;
   5777    pmem_list.entryList = &pmem_entry;
   5778    pmem_list.nEntries = 1;
   5779    ouput_egl_buffers = true;
   5780    if (OMX_ErrorNone != use_buffer(hComp,bufferHdr, port,
   5781        (void *)&pmem_list, drv_ctx.op_buf.buffer_size,
   5782         (OMX_U8 *)pmemPtr)) {
   5783      DEBUG_PRINT_ERROR("use buffer call failed for egl image\n");
   5784      return OMX_ErrorInsufficientResources;
   5785    }
   5786    return OMX_ErrorNone;
   5787 }
   5788 
   5789 /* ======================================================================
   5790 FUNCTION
   5791   omx_vdec::ComponentRoleEnum
   5792 
   5793 DESCRIPTION
   5794   OMX Component Role Enum method implementation.
   5795 
   5796 PARAMETERS
   5797   <TBD>.
   5798 
   5799 RETURN VALUE
   5800   OMX Error None if everything is successful.
   5801 ========================================================================== */
   5802 OMX_ERRORTYPE  omx_vdec::component_role_enum(OMX_IN OMX_HANDLETYPE hComp,
   5803                                                 OMX_OUT OMX_U8*        role,
   5804                                                 OMX_IN OMX_U32        index)
   5805 {
   5806   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   5807 
   5808   if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg4",OMX_MAX_STRINGNAME_SIZE))
   5809   {
   5810     if((0 == index) && role)
   5811     {
   5812       strlcpy((char *)role, "video_decoder.mpeg4",OMX_MAX_STRINGNAME_SIZE);
   5813       DEBUG_PRINT_LOW("component_role_enum: role %s\n",role);
   5814     }
   5815     else
   5816     {
   5817       eRet = OMX_ErrorNoMore;
   5818     }
   5819   }
   5820   if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg2",OMX_MAX_STRINGNAME_SIZE))
   5821   {
   5822     if((0 == index) && role)
   5823     {
   5824       strlcpy((char *)role, "video_decoder.mpeg2",OMX_MAX_STRINGNAME_SIZE);
   5825       DEBUG_PRINT_LOW("component_role_enum: role %s\n",role);
   5826     }
   5827     else
   5828     {
   5829       eRet = OMX_ErrorNoMore;
   5830     }
   5831   }
   5832   else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.h263",OMX_MAX_STRINGNAME_SIZE))
   5833   {
   5834     if((0 == index) && role)
   5835     {
   5836       strlcpy((char *)role, "video_decoder.h263",OMX_MAX_STRINGNAME_SIZE);
   5837       DEBUG_PRINT_LOW("component_role_enum: role %s\n",role);
   5838     }
   5839     else
   5840     {
   5841       DEBUG_PRINT_LOW("\n No more roles \n");
   5842       eRet = OMX_ErrorNoMore;
   5843     }
   5844   }
   5845 
   5846   else if((!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx",OMX_MAX_STRINGNAME_SIZE)) ||
   5847           (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx311",OMX_MAX_STRINGNAME_SIZE))
   5848           )
   5849 
   5850   {
   5851     if((0 == index) && role)
   5852     {
   5853       strlcpy((char *)role, "video_decoder.divx",OMX_MAX_STRINGNAME_SIZE);
   5854       DEBUG_PRINT_LOW("component_role_enum: role %s\n",role);
   5855     }
   5856     else
   5857     {
   5858       DEBUG_PRINT_LOW("\n No more roles \n");
   5859       eRet = OMX_ErrorNoMore;
   5860     }
   5861   }
   5862   else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc",OMX_MAX_STRINGNAME_SIZE))
   5863   {
   5864     if((0 == index) && role)
   5865     {
   5866       strlcpy((char *)role, "video_decoder.avc",OMX_MAX_STRINGNAME_SIZE);
   5867       DEBUG_PRINT_LOW("component_role_enum: role %s\n",role);
   5868     }
   5869     else
   5870     {
   5871       DEBUG_PRINT_LOW("\n No more roles \n");
   5872       eRet = OMX_ErrorNoMore;
   5873     }
   5874   }
   5875   else if( (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vc1",OMX_MAX_STRINGNAME_SIZE)) ||
   5876            (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.wmv",OMX_MAX_STRINGNAME_SIZE))
   5877            )
   5878   {
   5879     if((0 == index) && role)
   5880     {
   5881       strlcpy((char *)role, "video_decoder.vc1",OMX_MAX_STRINGNAME_SIZE);
   5882       DEBUG_PRINT_LOW("component_role_enum: role %s\n",role);
   5883     }
   5884     else
   5885     {
   5886       DEBUG_PRINT_LOW("\n No more roles \n");
   5887       eRet = OMX_ErrorNoMore;
   5888     }
   5889   }
   5890   else
   5891   {
   5892     DEBUG_PRINT_ERROR("\nERROR:Querying Role on Unknown Component\n");
   5893     eRet = OMX_ErrorInvalidComponentName;
   5894   }
   5895   return eRet;
   5896 }
   5897 
   5898 
   5899 
   5900 
   5901 /* ======================================================================
   5902 FUNCTION
   5903   omx_vdec::AllocateDone
   5904 
   5905 DESCRIPTION
   5906   Checks if entire buffer pool is allocated by IL Client or not.
   5907   Need this to move to IDLE state.
   5908 
   5909 PARAMETERS
   5910   None.
   5911 
   5912 RETURN VALUE
   5913   true/false.
   5914 
   5915 ========================================================================== */
   5916 bool omx_vdec::allocate_done(void)
   5917 {
   5918   bool bRet = false;
   5919   bool bRet_In = false;
   5920   bool bRet_Out = false;
   5921 
   5922   bRet_In = allocate_input_done();
   5923   bRet_Out = allocate_output_done();
   5924 
   5925   if(bRet_In && bRet_Out)
   5926   {
   5927       bRet = true;
   5928   }
   5929 
   5930   return bRet;
   5931 }
   5932 /* ======================================================================
   5933 FUNCTION
   5934   omx_vdec::AllocateInputDone
   5935 
   5936 DESCRIPTION
   5937   Checks if I/P buffer pool is allocated by IL Client or not.
   5938 
   5939 PARAMETERS
   5940   None.
   5941 
   5942 RETURN VALUE
   5943   true/false.
   5944 
   5945 ========================================================================== */
   5946 bool omx_vdec::allocate_input_done(void)
   5947 {
   5948   bool bRet = false;
   5949   unsigned i=0;
   5950 
   5951   if (m_inp_mem_ptr == NULL)
   5952   {
   5953       return bRet;
   5954   }
   5955   if(m_inp_mem_ptr )
   5956   {
   5957     for(;i<drv_ctx.ip_buf.actualcount;i++)
   5958     {
   5959       if(BITMASK_ABSENT(&m_inp_bm_count,i))
   5960       {
   5961         break;
   5962       }
   5963     }
   5964   }
   5965   if(i == drv_ctx.ip_buf.actualcount)
   5966   {
   5967     bRet = true;
   5968     DEBUG_PRINT_HIGH("Allocate done for all i/p buffers");
   5969   }
   5970   if(i==drv_ctx.ip_buf.actualcount && m_inp_bEnabled)
   5971   {
   5972      m_inp_bPopulated = OMX_TRUE;
   5973   }
   5974   return bRet;
   5975 }
   5976 /* ======================================================================
   5977 FUNCTION
   5978   omx_vdec::AllocateOutputDone
   5979 
   5980 DESCRIPTION
   5981   Checks if entire O/P buffer pool is allocated by IL Client or not.
   5982 
   5983 PARAMETERS
   5984   None.
   5985 
   5986 RETURN VALUE
   5987   true/false.
   5988 
   5989 ========================================================================== */
   5990 bool omx_vdec::allocate_output_done(void)
   5991 {
   5992   bool bRet = false;
   5993   unsigned j=0;
   5994 
   5995   if (m_out_mem_ptr == NULL)
   5996   {
   5997       return bRet;
   5998   }
   5999 
   6000   if (m_out_mem_ptr)
   6001   {
   6002     for(;j < drv_ctx.op_buf.actualcount;j++)
   6003     {
   6004       if(BITMASK_ABSENT(&m_out_bm_count,j))
   6005       {
   6006         break;
   6007       }
   6008     }
   6009   }
   6010 
   6011   if(j == drv_ctx.op_buf.actualcount)
   6012   {
   6013     bRet = true;
   6014     DEBUG_PRINT_HIGH("Allocate done for all o/p buffers");
   6015     if(m_out_bEnabled)
   6016        m_out_bPopulated = OMX_TRUE;
   6017   }
   6018 
   6019   return bRet;
   6020 }
   6021 
   6022 /* ======================================================================
   6023 FUNCTION
   6024   omx_vdec::ReleaseDone
   6025 
   6026 DESCRIPTION
   6027   Checks if IL client has released all the buffers.
   6028 
   6029 PARAMETERS
   6030   None.
   6031 
   6032 RETURN VALUE
   6033   true/false
   6034 
   6035 ========================================================================== */
   6036 bool omx_vdec::release_done(void)
   6037 {
   6038   bool bRet = false;
   6039 
   6040   if(release_input_done())
   6041   {
   6042     if(release_output_done())
   6043     {
   6044         bRet = true;
   6045     }
   6046   }
   6047   return bRet;
   6048 }
   6049 
   6050 
   6051 /* ======================================================================
   6052 FUNCTION
   6053   omx_vdec::ReleaseOutputDone
   6054 
   6055 DESCRIPTION
   6056   Checks if IL client has released all the buffers.
   6057 
   6058 PARAMETERS
   6059   None.
   6060 
   6061 RETURN VALUE
   6062   true/false
   6063 
   6064 ========================================================================== */
   6065 bool omx_vdec::release_output_done(void)
   6066 {
   6067   bool bRet = false;
   6068   unsigned i=0,j=0;
   6069 
   6070   DEBUG_PRINT_LOW("\n Value of m_out_mem_ptr %p",m_inp_mem_ptr);
   6071   if(m_out_mem_ptr)
   6072   {
   6073       for(;j < drv_ctx.op_buf.actualcount ; j++)
   6074       {
   6075         if(BITMASK_PRESENT(&m_out_bm_count,j))
   6076         {
   6077           break;
   6078         }
   6079       }
   6080     if(j == drv_ctx.op_buf.actualcount)
   6081     {
   6082       m_out_bm_count = 0;
   6083       bRet = true;
   6084     }
   6085   }
   6086   else
   6087   {
   6088     m_out_bm_count = 0;
   6089     bRet = true;
   6090   }
   6091   return bRet;
   6092 }
   6093 /* ======================================================================
   6094 FUNCTION
   6095   omx_vdec::ReleaseInputDone
   6096 
   6097 DESCRIPTION
   6098   Checks if IL client has released all the buffers.
   6099 
   6100 PARAMETERS
   6101   None.
   6102 
   6103 RETURN VALUE
   6104   true/false
   6105 
   6106 ========================================================================== */
   6107 bool omx_vdec::release_input_done(void)
   6108 {
   6109   bool bRet = false;
   6110   unsigned i=0,j=0;
   6111 
   6112   DEBUG_PRINT_LOW("\n Value of m_inp_mem_ptr %p",m_inp_mem_ptr);
   6113   if(m_inp_mem_ptr)
   6114   {
   6115       for(;j<drv_ctx.ip_buf.actualcount;j++)
   6116       {
   6117         if( BITMASK_PRESENT(&m_inp_bm_count,j))
   6118         {
   6119           break;
   6120         }
   6121       }
   6122     if(j==drv_ctx.ip_buf.actualcount)
   6123     {
   6124       bRet = true;
   6125     }
   6126   }
   6127   else
   6128   {
   6129     bRet = true;
   6130   }
   6131   return bRet;
   6132 }
   6133 
   6134 OMX_ERRORTYPE omx_vdec::fill_buffer_done(OMX_HANDLETYPE hComp,
   6135                                OMX_BUFFERHEADERTYPE * buffer)
   6136 {
   6137   OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *pPMEMInfo = NULL;
   6138   if (!buffer || (buffer - m_out_mem_ptr) >= drv_ctx.op_buf.actualcount)
   6139   {
   6140     DEBUG_PRINT_ERROR("\n [FBD] ERROR in ptr(%p)", buffer);
   6141     return OMX_ErrorBadParameter;
   6142   }
   6143   else if (output_flush_progress)
   6144   {
   6145     DEBUG_PRINT_LOW("FBD: Buffer (%p) flushed", buffer);
   6146     buffer->nFilledLen = 0;
   6147     buffer->nTimeStamp = 0;
   6148     buffer->nFlags &= ~OMX_BUFFERFLAG_EXTRADATA;
   6149     buffer->nFlags &= ~QOMX_VIDEO_BUFFERFLAG_EOSEQ;
   6150     buffer->nFlags &= ~OMX_BUFFERFLAG_DATACORRUPT;
   6151   }
   6152 
   6153   DEBUG_PRINT_LOW("\n fill_buffer_done: bufhdr = %p, bufhdr->pBuffer = %p",
   6154       buffer, buffer->pBuffer);
   6155   pending_output_buffers --;
   6156 
   6157   if (buffer->nFlags & OMX_BUFFERFLAG_EOS)
   6158   {
   6159     DEBUG_PRINT_HIGH("\n Output EOS has been reached");
   6160     if (!output_flush_progress)
   6161       post_event(NULL,NULL,OMX_COMPONENT_GENERATE_EOS_DONE);
   6162 
   6163     if (psource_frame)
   6164     {
   6165       m_cb.EmptyBufferDone(&m_cmp, m_app_data, psource_frame);
   6166       psource_frame = NULL;
   6167     }
   6168     if (pdest_frame)
   6169     {
   6170       pdest_frame->nFilledLen = 0;
   6171       m_input_free_q.insert_entry((unsigned) pdest_frame,NULL,NULL);
   6172       pdest_frame = NULL;
   6173     }
   6174   }
   6175 
   6176   DEBUG_PRINT_LOW("\n In fill Buffer done call address %p ",buffer);
   6177 #ifdef OUTPUT_BUFFER_LOG
   6178   if (outputBufferFile1)
   6179   {
   6180     fwrite (buffer->pBuffer,1,buffer->nFilledLen,
   6181                   outputBufferFile1);
   6182   }
   6183 #endif
   6184 
   6185   /* For use buffer we need to copy the data */
   6186   if (!output_flush_progress)
   6187   {
   6188     time_stamp_dts.get_next_timestamp(buffer,
   6189     (drv_ctx.interlace != VDEC_InterlaceFrameProgressive)
   6190      ?true:false);
   6191   }
   6192   if (m_cb.FillBufferDone)
   6193   {
   6194     if (buffer->nFilledLen > 0)
   6195     {
   6196       if (client_extradata)
   6197         handle_extradata(buffer);
   6198       if (client_extradata & OMX_TIMEINFO_EXTRADATA)
   6199         // Keep min timestamp interval to handle corrupted bit stream scenario
   6200         set_frame_rate(buffer->nTimeStamp);
   6201       else if (arbitrary_bytes)
   6202         adjust_timestamp(buffer->nTimeStamp);
   6203       if (perf_flag)
   6204       {
   6205         if (!proc_frms)
   6206         {
   6207           dec_time.stop();
   6208           latency = dec_time.processing_time_us() - latency;
   6209           DEBUG_PRINT_HIGH(">>> FBD Metrics: Latency(%.2f)mS", latency / 1e3);
   6210           dec_time.start();
   6211           fps_metrics.start();
   6212         }
   6213         proc_frms++;
   6214         if (buffer->nFlags & OMX_BUFFERFLAG_EOS)
   6215         {
   6216           OMX_U64 proc_time = 0;
   6217           fps_metrics.stop();
   6218           proc_time = fps_metrics.processing_time_us();
   6219           DEBUG_PRINT_HIGH(">>> FBD Metrics: proc_frms(%lu) proc_time(%.2f)S fps(%.2f)",
   6220                             proc_frms, (float)proc_time / 1e6,
   6221                             (float)(1e6 * proc_frms) / proc_time);
   6222           proc_frms = 0;
   6223         }
   6224       }
   6225 
   6226 #ifdef OUTPUT_EXTRADATA_LOG
   6227   if (outputExtradataFile)
   6228   {
   6229 
   6230     OMX_OTHER_EXTRADATATYPE *p_extra = NULL;
   6231     p_extra = (OMX_OTHER_EXTRADATATYPE *)
   6232            ((unsigned)(buffer->pBuffer + buffer->nOffset +
   6233             buffer->nFilledLen + 3)&(~3));
   6234     while(p_extra &&
   6235           (OMX_U8*)p_extra < (buffer->pBuffer + buffer->nAllocLen) )
   6236     {
   6237       DEBUG_PRINT_LOW("\nWRITING extradata, size=%d,type=%d",p_extra->nSize, p_extra->eType);
   6238       fwrite (p_extra,1,p_extra->nSize,outputExtradataFile);
   6239       if (p_extra->eType == OMX_ExtraDataNone)
   6240       {
   6241         break;
   6242       }
   6243       p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + p_extra->nSize);
   6244     }
   6245   }
   6246 #endif
   6247     }
   6248     if (buffer->nFlags & OMX_BUFFERFLAG_EOS){
   6249       prev_ts = LLONG_MAX;
   6250       rst_prev_ts = true;
   6251       }
   6252 
   6253     pPMEMInfo = (OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *)
   6254                 ((OMX_QCOM_PLATFORM_PRIVATE_LIST *)
   6255                 buffer->pPlatformPrivate)->entryList->entry;
   6256     DEBUG_PRINT_LOW("\n Before FBD callback Accessed Pmeminfo %d",pPMEMInfo->pmem_fd);
   6257     m_cb.FillBufferDone (hComp,m_app_data,buffer);
   6258     DEBUG_PRINT_LOW("\n After Fill Buffer Done callback %d",pPMEMInfo->pmem_fd);
   6259   }
   6260   else
   6261   {
   6262     return OMX_ErrorBadParameter;
   6263   }
   6264 
   6265   return OMX_ErrorNone;
   6266 }
   6267 
   6268 OMX_ERRORTYPE omx_vdec::empty_buffer_done(OMX_HANDLETYPE         hComp,
   6269                                           OMX_BUFFERHEADERTYPE* buffer)
   6270 {
   6271 
   6272     if (buffer == NULL || ((buffer - m_inp_mem_ptr) > drv_ctx.ip_buf.actualcount))
   6273     {
   6274         DEBUG_PRINT_ERROR("\n empty_buffer_done: ERROR bufhdr = %p", buffer);
   6275        return OMX_ErrorBadParameter;
   6276     }
   6277 
   6278     DEBUG_PRINT_LOW("\n empty_buffer_done: bufhdr = %p, bufhdr->pBuffer = %p",
   6279         buffer, buffer->pBuffer);
   6280     pending_input_buffers--;
   6281 
   6282     if (arbitrary_bytes)
   6283     {
   6284       if (pdest_frame == NULL && input_flush_progress == false)
   6285       {
   6286         DEBUG_PRINT_LOW("\n Push input from buffer done address of Buffer %p",buffer);
   6287         pdest_frame = buffer;
   6288         buffer->nFilledLen = 0;
   6289         buffer->nTimeStamp = LLONG_MAX;
   6290         push_input_buffer (hComp);
   6291       }
   6292       else
   6293       {
   6294         DEBUG_PRINT_LOW("\n Push buffer into freeq address of Buffer %p",buffer);
   6295         buffer->nFilledLen = 0;
   6296         if (!m_input_free_q.insert_entry((unsigned)buffer,NULL,NULL))
   6297         {
   6298           DEBUG_PRINT_ERROR("\nERROR:i/p free Queue is FULL Error");
   6299         }
   6300       }
   6301     }
   6302     else if(m_cb.EmptyBufferDone)
   6303     {
   6304         buffer->nFilledLen = 0;
   6305         if (input_use_buffer == true){
   6306             buffer = &m_inp_heap_ptr[buffer-m_inp_mem_ptr];
   6307         }
   6308         m_cb.EmptyBufferDone(hComp ,m_app_data, buffer);
   6309     }
   6310     return OMX_ErrorNone;
   6311 }
   6312 
   6313 
   6314 int omx_vdec::async_message_process (void *context, void* message)
   6315 {
   6316   omx_vdec* omx = NULL;
   6317   struct vdec_msginfo *vdec_msg = NULL;
   6318   OMX_BUFFERHEADERTYPE* omxhdr = NULL;
   6319   struct v4l2_buffer *v4l2_buf_ptr=NULL;
   6320   struct vdec_output_frameinfo *output_respbuf = NULL;
   6321 	int rc=1;
   6322   if (context == NULL || message == NULL)
   6323   {
   6324     DEBUG_PRINT_ERROR("\n FATAL ERROR in omx_vdec::async_message_process NULL Check");
   6325     return -1;
   6326   }
   6327   vdec_msg = (struct vdec_msginfo *)message;
   6328 
   6329   omx = reinterpret_cast<omx_vdec*>(context);
   6330 
   6331 #ifdef _ANDROID_
   6332   if (omx->m_debug_timestamp)
   6333   {
   6334     if ( (vdec_msg->msgcode == VDEC_MSG_RESP_OUTPUT_BUFFER_DONE) &&
   6335          !(omx->output_flush_progress) )
   6336     {
   6337       OMX_TICKS expected_ts = 0;
   6338       omx->m_timestamp_list.pop_min_ts(expected_ts);
   6339       DEBUG_PRINT_LOW("\n Current timestamp (%lld),Popped TIMESTAMP (%lld) from list",
   6340                        vdec_msg->msgdata.output_frame.time_stamp, expected_ts);
   6341 
   6342       if (vdec_msg->msgdata.output_frame.time_stamp != expected_ts)
   6343       {
   6344         DEBUG_PRINT_ERROR("\n ERROR in omx_vdec::async_message_process timestamp Check");
   6345       }
   6346     }
   6347   }
   6348 #endif
   6349 
   6350   switch (vdec_msg->msgcode)
   6351   {
   6352 
   6353   case VDEC_MSG_EVT_HW_ERROR:
   6354     omx->post_event (NULL,vdec_msg->status_code,\
   6355                      OMX_COMPONENT_GENERATE_HARDWARE_ERROR);
   6356   break;
   6357 
   6358   case VDEC_MSG_RESP_START_DONE:
   6359     omx->post_event (NULL,vdec_msg->status_code,\
   6360                      OMX_COMPONENT_GENERATE_START_DONE);
   6361   break;
   6362 
   6363   case VDEC_MSG_RESP_STOP_DONE:
   6364     omx->post_event (NULL,vdec_msg->status_code,\
   6365                      OMX_COMPONENT_GENERATE_STOP_DONE);
   6366   break;
   6367 
   6368   case VDEC_MSG_RESP_RESUME_DONE:
   6369     omx->post_event (NULL,vdec_msg->status_code,\
   6370                      OMX_COMPONENT_GENERATE_RESUME_DONE);
   6371   break;
   6372 
   6373   case VDEC_MSG_RESP_PAUSE_DONE:
   6374     omx->post_event (NULL,vdec_msg->status_code,\
   6375                      OMX_COMPONENT_GENERATE_PAUSE_DONE);
   6376   break;
   6377 
   6378   case VDEC_MSG_RESP_FLUSH_INPUT_DONE:
   6379     omx->post_event (NULL,vdec_msg->status_code,\
   6380                      OMX_COMPONENT_GENERATE_EVENT_INPUT_FLUSH);
   6381     break;
   6382   case VDEC_MSG_RESP_FLUSH_OUTPUT_DONE:
   6383     omx->post_event (NULL,vdec_msg->status_code,\
   6384                      OMX_COMPONENT_GENERATE_EVENT_OUTPUT_FLUSH);
   6385     break;
   6386   case VDEC_MSG_RESP_INPUT_FLUSHED:
   6387   case VDEC_MSG_RESP_INPUT_BUFFER_DONE:
   6388 
   6389     // omxhdr = (OMX_BUFFERHEADERTYPE* )		\
   6390       //        vdec_msg->msgdata.input_frame_clientdata;
   6391 
   6392     v4l2_buf_ptr = (v4l2_buffer*)vdec_msg->msgdata.input_frame_clientdata;
   6393     omxhdr=omx->m_inp_mem_ptr+v4l2_buf_ptr->index;
   6394     if (omxhdr == NULL ||
   6395        ((omxhdr - omx->m_inp_mem_ptr) > omx->drv_ctx.ip_buf.actualcount) )
   6396     {
   6397        omxhdr = NULL;
   6398        vdec_msg->status_code = VDEC_S_EFATAL;
   6399     }
   6400 
   6401     omx->post_event ((unsigned int)omxhdr,vdec_msg->status_code,
   6402                      OMX_COMPONENT_GENERATE_EBD);
   6403     break;
   6404     case VDEC_MSG_EVT_INFO_FIELD_DROPPED:
   6405       int64_t *timestamp;
   6406       timestamp = (int64_t *) malloc(sizeof(int64_t));
   6407       if (timestamp) {
   6408         *timestamp = vdec_msg->msgdata.output_frame.time_stamp;
   6409         omx->post_event ((unsigned int)timestamp, vdec_msg->status_code,
   6410                          OMX_COMPONENT_GENERATE_INFO_FIELD_DROPPED);
   6411         DEBUG_PRINT_HIGH("\nField dropped time stamp is %lld",
   6412              vdec_msg->msgdata.output_frame.time_stamp);
   6413       }
   6414       break;
   6415   case VDEC_MSG_RESP_OUTPUT_FLUSHED:
   6416     case VDEC_MSG_RESP_OUTPUT_BUFFER_DONE:
   6417 
   6418       v4l2_buf_ptr = (v4l2_buffer*)vdec_msg->msgdata.output_frame.client_data;
   6419       omxhdr=omx->m_out_mem_ptr+v4l2_buf_ptr->index;
   6420       DEBUG_PRINT_LOW("[RespBufDone] Buf(%p) Ts(%lld) Pic_type(%u)",
   6421 		      omxhdr, vdec_msg->msgdata.output_frame.time_stamp,
   6422 		      vdec_msg->msgdata.output_frame.pic_type);
   6423 
   6424     if (omxhdr && omxhdr->pOutputPortPrivate &&
   6425         ((omxhdr - omx->m_out_mem_ptr) < omx->drv_ctx.op_buf.actualcount) &&
   6426          (((struct vdec_output_frameinfo *)omxhdr->pOutputPortPrivate
   6427             - omx->drv_ctx.ptr_respbuffer) < omx->drv_ctx.op_buf.actualcount))
   6428     {
   6429       if ( vdec_msg->msgdata.output_frame.len <=  omxhdr->nAllocLen)
   6430       {
   6431 	omxhdr->nFilledLen = vdec_msg->msgdata.output_frame.len;
   6432 	omxhdr->nOffset = vdec_msg->msgdata.output_frame.offset;
   6433         omxhdr->nTimeStamp = omx->m_out_mem_ptr[v4l2_buf_ptr->index].nTimeStamp;
   6434         omxhdr->nFlags = omx->m_out_mem_ptr[v4l2_buf_ptr->index].nFlags;
   6435 
   6436 	if (v4l2_buf_ptr->flags & V4L2_BUF_FLAG_EOS)
   6437 	{
   6438 	  omxhdr->nFlags |= OMX_BUFFERFLAG_EOS;
   6439 	  //rc = -1;
   6440 	}
   6441 	vdec_msg->msgdata.output_frame.bufferaddr=omx->drv_ctx.ptr_outputbuffer[v4l2_buf_ptr->index].bufferaddr;
   6442         output_respbuf = (struct vdec_output_frameinfo *)\
   6443                           omxhdr->pOutputPortPrivate;
   6444 	// output_respbuf->framesize.bottom =
   6445         //  vdec_msg->msgdata.output_frame.framesize.bottom;
   6446 	// output_respbuf->framesize.left =
   6447 	//   vdec_msg->msgdata.output_frame.framesize.left;
   6448 	// output_respbuf->framesize.right =
   6449 	//   vdec_msg->msgdata.output_frame.framesize.right;
   6450 	// output_respbuf->framesize.top =
   6451 	//   vdec_msg->msgdata.output_frame.framesize.top;
   6452         output_respbuf->len = vdec_msg->msgdata.output_frame.len;
   6453         output_respbuf->offset = vdec_msg->msgdata.output_frame.offset;
   6454 	// output_respbuf->time_stamp = vdec_msg->msgdata.output_frame.time_stamp;
   6455 	// output_respbuf->flags = vdec_msg->msgdata.output_frame.flags;
   6456 	// output_respbuf->pic_type = vdec_msg->msgdata.output_frame.pic_type;
   6457 	// output_respbuf->interlaced_format = vdec_msg->msgdata.output_frame.interlaced_format;
   6458 
   6459         if (omx->output_use_buffer)
   6460           memcpy ( omxhdr->pBuffer,
   6461                    (vdec_msg->msgdata.output_frame.bufferaddr +
   6462                     vdec_msg->msgdata.output_frame.offset),
   6463                     vdec_msg->msgdata.output_frame.len );
   6464       }
   6465       else
   6466         omxhdr->nFilledLen = 0;
   6467       omx->post_event ((unsigned int)omxhdr, vdec_msg->status_code,
   6468                        OMX_COMPONENT_GENERATE_FBD);
   6469     }
   6470     else if (vdec_msg->msgdata.output_frame.flags & OMX_BUFFERFLAG_EOS)
   6471       omx->post_event (NULL, vdec_msg->status_code,
   6472                        OMX_COMPONENT_GENERATE_EOS_DONE);
   6473     else
   6474       omx->post_event (NULL, vdec_msg->status_code,
   6475                        OMX_COMPONENT_GENERATE_HARDWARE_ERROR);
   6476     break;
   6477   case VDEC_MSG_EVT_CONFIG_CHANGED:
   6478     DEBUG_PRINT_HIGH("\n Port settings changed");
   6479 	omx->post_event ((unsigned int)omxhdr,vdec_msg->status_code,
   6480                      OMX_COMPONENT_GENERATE_PORT_RECONFIG);
   6481     break;
   6482   case VDEC_MSG_EVT_INFO_CONFIG_CHANGED:
   6483   {
   6484     DEBUG_PRINT_HIGH("\n Port settings changed info");
   6485     // get_buffer_req and populate port defn structure
   6486     OMX_ERRORTYPE eRet = OMX_ErrorNone;
   6487     omx->m_port_def.nPortIndex = 1;
   6488     eRet = omx->update_portdef(&(omx->m_port_def));
   6489     omx->post_event ((unsigned int)omxhdr,vdec_msg->status_code,
   6490                      OMX_COMPONENT_GENERATE_INFO_PORT_RECONFIG);
   6491     break;
   6492   }
   6493   default:
   6494     break;
   6495   }
   6496   return rc;
   6497 }
   6498 
   6499 OMX_ERRORTYPE omx_vdec::empty_this_buffer_proxy_arbitrary (
   6500                                                    OMX_HANDLETYPE hComp,
   6501                                                    OMX_BUFFERHEADERTYPE *buffer
   6502                                                            )
   6503 {
   6504   unsigned address,p2,id;
   6505   DEBUG_PRINT_LOW("\n Empty this arbitrary");
   6506 
   6507   if (buffer == NULL)
   6508   {
   6509     return OMX_ErrorBadParameter;
   6510   }
   6511   DEBUG_PRINT_LOW("\n ETBProxyArb: bufhdr = %p, bufhdr->pBuffer = %p", buffer, buffer->pBuffer);
   6512   DEBUG_PRINT_LOW("\n ETBProxyArb: nFilledLen %u, flags %d, timestamp %u",
   6513         buffer->nFilledLen, buffer->nFlags, (unsigned)buffer->nTimeStamp);
   6514 
   6515   /* return zero length and not an EOS buffer */
   6516   /* return buffer if input flush in progress */
   6517   if ((input_flush_progress == true) || ((buffer->nFilledLen == 0) &&
   6518      ((buffer->nFlags & OMX_BUFFERFLAG_EOS) == 0)))
   6519   {
   6520     DEBUG_PRINT_HIGH("\n return zero legth buffer or flush in progress");
   6521     m_cb.EmptyBufferDone (hComp,m_app_data,buffer);
   6522     return OMX_ErrorNone;
   6523   }
   6524 
   6525   if (psource_frame == NULL)
   6526   {
   6527     DEBUG_PRINT_LOW("\n Set Buffer as source Buffer %p time stamp %d",buffer,buffer->nTimeStamp);
   6528     psource_frame = buffer;
   6529     DEBUG_PRINT_LOW("\n Try to Push One Input Buffer ");
   6530     push_input_buffer (hComp);
   6531   }
   6532   else
   6533   {
   6534     DEBUG_PRINT_LOW("\n Push the source buffer into pendingq %p",buffer);
   6535     if (!m_input_pending_q.insert_entry((unsigned)buffer,NULL,NULL))
   6536     {
   6537       return OMX_ErrorBadParameter;
   6538     }
   6539   }
   6540 
   6541 
   6542   return OMX_ErrorNone;
   6543 }
   6544 
   6545 OMX_ERRORTYPE omx_vdec::push_input_buffer (OMX_HANDLETYPE hComp)
   6546 {
   6547   unsigned address,p2,id;
   6548   OMX_ERRORTYPE ret = OMX_ErrorNone;
   6549 
   6550   if (pdest_frame == NULL || psource_frame == NULL)
   6551   {
   6552     /*Check if we have a destination buffer*/
   6553     if (pdest_frame == NULL)
   6554     {
   6555       DEBUG_PRINT_LOW("\n Get a Destination buffer from the queue");
   6556       if (m_input_free_q.m_size)
   6557       {
   6558         m_input_free_q.pop_entry(&address,&p2,&id);
   6559         pdest_frame = (OMX_BUFFERHEADERTYPE *)address;
   6560         pdest_frame->nFilledLen = 0;
   6561         pdest_frame->nTimeStamp = LLONG_MAX;
   6562         DEBUG_PRINT_LOW("\n Address of Pmem Buffer %p",pdest_frame);
   6563       }
   6564     }
   6565 
   6566     /*Check if we have a destination buffer*/
   6567     if (psource_frame == NULL)
   6568     {
   6569       DEBUG_PRINT_LOW("\n Get a source buffer from the queue");
   6570       if (m_input_pending_q.m_size)
   6571       {
   6572         m_input_pending_q.pop_entry(&address,&p2,&id);
   6573         psource_frame = (OMX_BUFFERHEADERTYPE *)address;
   6574         DEBUG_PRINT_LOW("\n Next source Buffer %p time stamp %d",psource_frame,
   6575                 psource_frame->nTimeStamp);
   6576         DEBUG_PRINT_LOW("\n Next source Buffer flag %d length %d",
   6577         psource_frame->nFlags,psource_frame->nFilledLen);
   6578 
   6579       }
   6580     }
   6581 
   6582   }
   6583 
   6584   while ((pdest_frame != NULL) && (psource_frame != NULL))
   6585   {
   6586     switch (codec_type_parse)
   6587     {
   6588       case CODEC_TYPE_MPEG4:
   6589       case CODEC_TYPE_H263:
   6590       case CODEC_TYPE_MPEG2:
   6591         ret =  push_input_sc_codec(hComp);
   6592       break;
   6593       case CODEC_TYPE_H264:
   6594         ret = push_input_h264(hComp);
   6595       break;
   6596       case CODEC_TYPE_VC1:
   6597         ret = push_input_vc1(hComp);
   6598       break;
   6599     }
   6600     if (ret != OMX_ErrorNone)
   6601     {
   6602       DEBUG_PRINT_ERROR("\n Pushing input Buffer Failed");
   6603       omx_report_error ();
   6604       break;
   6605     }
   6606   }
   6607 
   6608   return ret;
   6609 }
   6610 
   6611 OMX_ERRORTYPE omx_vdec::push_input_sc_codec(OMX_HANDLETYPE hComp)
   6612 {
   6613   OMX_U32 partial_frame = 1;
   6614   OMX_BOOL generate_ebd = OMX_TRUE;
   6615   unsigned address,p2,id;
   6616 
   6617   DEBUG_PRINT_LOW("\n Start Parsing the bit stream address %p TimeStamp %d",
   6618         psource_frame,psource_frame->nTimeStamp);
   6619   if (m_frame_parser.parse_sc_frame(psource_frame,
   6620                                        pdest_frame,&partial_frame) == -1)
   6621   {
   6622     DEBUG_PRINT_ERROR("\n Error In Parsing Return Error");
   6623     return OMX_ErrorBadParameter;
   6624   }
   6625 
   6626   if (partial_frame == 0)
   6627   {
   6628     DEBUG_PRINT_LOW("\n Frame size %d source %p frame count %d",
   6629           pdest_frame->nFilledLen,psource_frame,frame_count);
   6630 
   6631 
   6632     DEBUG_PRINT_LOW("\n TimeStamp updated %d",pdest_frame->nTimeStamp);
   6633     /*First Parsed buffer will have only header Hence skip*/
   6634     if (frame_count == 0)
   6635     {
   6636       DEBUG_PRINT_LOW("\n H263/MPEG4 Codec First Frame ");
   6637 
   6638       if(codec_type_parse == CODEC_TYPE_MPEG4 ||
   6639          codec_type_parse == CODEC_TYPE_DIVX) {
   6640         mp4StreamType psBits;
   6641         psBits.data = pdest_frame->pBuffer + pdest_frame->nOffset;
   6642         psBits.numBytes = pdest_frame->nFilledLen;
   6643         mp4_headerparser.parseHeader(&psBits);
   6644       }
   6645 
   6646       frame_count++;
   6647     }
   6648     else
   6649     {
   6650       pdest_frame->nFlags &= ~OMX_BUFFERFLAG_EOS;
   6651       if(pdest_frame->nFilledLen)
   6652       {
   6653         /*Push the frame to the Decoder*/
   6654         if (empty_this_buffer_proxy(hComp,pdest_frame) != OMX_ErrorNone)
   6655         {
   6656           return OMX_ErrorBadParameter;
   6657         }
   6658         frame_count++;
   6659         pdest_frame = NULL;
   6660 
   6661         if (m_input_free_q.m_size)
   6662         {
   6663           m_input_free_q.pop_entry(&address,&p2,&id);
   6664           pdest_frame = (OMX_BUFFERHEADERTYPE *) address;
   6665           pdest_frame->nFilledLen = 0;
   6666         }
   6667       }
   6668       else if(!(psource_frame->nFlags & OMX_BUFFERFLAG_EOS))
   6669       {
   6670         DEBUG_PRINT_ERROR("\nZero len buffer return back to POOL");
   6671         m_input_free_q.insert_entry((unsigned) pdest_frame,NULL,NULL);
   6672         pdest_frame = NULL;
   6673       }
   6674     }
   6675   }
   6676   else
   6677   {
   6678     DEBUG_PRINT_LOW("\n Not a Complete Frame %d",pdest_frame->nFilledLen);
   6679     /*Check if Destination Buffer is full*/
   6680     if (pdest_frame->nAllocLen ==
   6681         pdest_frame->nFilledLen + pdest_frame->nOffset)
   6682     {
   6683       DEBUG_PRINT_ERROR("\nERROR:Frame Not found though Destination Filled");
   6684       return OMX_ErrorStreamCorrupt;
   6685     }
   6686   }
   6687 
   6688   if (psource_frame->nFilledLen == 0)
   6689   {
   6690     if (psource_frame->nFlags & OMX_BUFFERFLAG_EOS)
   6691     {
   6692       if (pdest_frame)
   6693       {
   6694         pdest_frame->nFlags |= psource_frame->nFlags;
   6695         DEBUG_PRINT_LOW("\n Frame Found start Decoding Size =%d TimeStamp = %x",
   6696                      pdest_frame->nFilledLen,pdest_frame->nTimeStamp);
   6697         DEBUG_PRINT_LOW("\n Found a frame size = %d number = %d",
   6698                      pdest_frame->nFilledLen,frame_count++);
   6699         /*Push the frame to the Decoder*/
   6700         if (empty_this_buffer_proxy(hComp,pdest_frame) != OMX_ErrorNone)
   6701         {
   6702           return OMX_ErrorBadParameter;
   6703         }
   6704         frame_count++;
   6705         pdest_frame = NULL;
   6706       }
   6707       else
   6708       {
   6709         DEBUG_PRINT_LOW("\n Last frame in else dest addr") ;
   6710         generate_ebd = OMX_FALSE;
   6711       }
   6712    }
   6713     if(generate_ebd)
   6714     {
   6715       DEBUG_PRINT_LOW("\n Buffer Consumed return back to client %p",psource_frame);
   6716       m_cb.EmptyBufferDone (hComp,m_app_data,psource_frame);
   6717       psource_frame = NULL;
   6718 
   6719       if (m_input_pending_q.m_size)
   6720       {
   6721         DEBUG_PRINT_LOW("\n Pull Next source Buffer %p",psource_frame);
   6722         m_input_pending_q.pop_entry(&address,&p2,&id);
   6723         psource_frame = (OMX_BUFFERHEADERTYPE *) address;
   6724         DEBUG_PRINT_LOW("\n Next source Buffer %p time stamp %d",psource_frame,
   6725                 psource_frame->nTimeStamp);
   6726         DEBUG_PRINT_LOW("\n Next source Buffer flag %d length %d",
   6727         psource_frame->nFlags,psource_frame->nFilledLen);
   6728       }
   6729     }
   6730    }
   6731   return OMX_ErrorNone;
   6732 }
   6733 
   6734 OMX_ERRORTYPE omx_vdec::push_input_h264 (OMX_HANDLETYPE hComp)
   6735 {
   6736   OMX_U32 partial_frame = 1;
   6737   unsigned address,p2,id;
   6738   OMX_BOOL isNewFrame = OMX_FALSE;
   6739   OMX_BOOL generate_ebd = OMX_TRUE;
   6740 
   6741   if (h264_scratch.pBuffer == NULL)
   6742   {
   6743     DEBUG_PRINT_ERROR("\nERROR:H.264 Scratch Buffer not allocated");
   6744     return OMX_ErrorBadParameter;
   6745   }
   6746   DEBUG_PRINT_LOW("\n Pending h264_scratch.nFilledLen %d "
   6747       "look_ahead_nal %d", h264_scratch.nFilledLen, look_ahead_nal);
   6748   DEBUG_PRINT_LOW("\n Pending pdest_frame->nFilledLen %d",pdest_frame->nFilledLen);
   6749   if (h264_scratch.nFilledLen && look_ahead_nal)
   6750   {
   6751     look_ahead_nal = false;
   6752     if ((pdest_frame->nAllocLen - pdest_frame->nFilledLen) >=
   6753          h264_scratch.nFilledLen)
   6754     {
   6755       memcpy ((pdest_frame->pBuffer + pdest_frame->nFilledLen),
   6756               h264_scratch.pBuffer,h264_scratch.nFilledLen);
   6757       pdest_frame->nFilledLen += h264_scratch.nFilledLen;
   6758       DEBUG_PRINT_LOW("\n Copy the previous NAL (h264 scratch) into Dest frame");
   6759       h264_scratch.nFilledLen = 0;
   6760     }
   6761     else
   6762     {
   6763       DEBUG_PRINT_ERROR("\n Error:1: Destination buffer overflow for H264");
   6764       return OMX_ErrorBadParameter;
   6765     }
   6766   }
   6767   if (nal_length == 0)
   6768   {
   6769     DEBUG_PRINT_LOW("\n Zero NAL, hence parse using start code");
   6770     if (m_frame_parser.parse_sc_frame(psource_frame,
   6771         &h264_scratch,&partial_frame) == -1)
   6772     {
   6773       DEBUG_PRINT_ERROR("\n Error In Parsing Return Error");
   6774       return OMX_ErrorBadParameter;
   6775     }
   6776   }
   6777   else
   6778   {
   6779     DEBUG_PRINT_LOW("\n Non-zero NAL length clip, hence parse with NAL size %d ",nal_length);
   6780     if (m_frame_parser.parse_h264_nallength(psource_frame,
   6781         &h264_scratch,&partial_frame) == -1)
   6782     {
   6783       DEBUG_PRINT_ERROR("\n Error In Parsing NAL size, Return Error");
   6784       return OMX_ErrorBadParameter;
   6785     }
   6786   }
   6787 
   6788   if (partial_frame == 0)
   6789   {
   6790     if (nal_count == 0 && h264_scratch.nFilledLen == 0)
   6791     {
   6792       DEBUG_PRINT_LOW("\n First NAL with Zero Length, hence Skip");
   6793       nal_count++;
   6794       h264_scratch.nTimeStamp = psource_frame->nTimeStamp;
   6795       h264_scratch.nFlags = psource_frame->nFlags;
   6796     }
   6797     else
   6798     {
   6799       DEBUG_PRINT_LOW("\n Parsed New NAL Length = %d",h264_scratch.nFilledLen);
   6800       if(h264_scratch.nFilledLen)
   6801       {
   6802           h264_parser->parse_nal((OMX_U8*)h264_scratch.pBuffer, h264_scratch.nFilledLen,
   6803                                  NALU_TYPE_SPS);
   6804 #ifndef PROCESS_EXTRADATA_IN_OUTPUT_PORT
   6805         if (client_extradata & OMX_TIMEINFO_EXTRADATA)
   6806           h264_parser->parse_nal((OMX_U8*)h264_scratch.pBuffer,
   6807                                   h264_scratch.nFilledLen, NALU_TYPE_SEI);
   6808         else if (client_extradata & OMX_FRAMEINFO_EXTRADATA)
   6809           // If timeinfo is present frame info from SEI is already processed
   6810           h264_parser->parse_nal((OMX_U8*)h264_scratch.pBuffer,
   6811                                   h264_scratch.nFilledLen, NALU_TYPE_SEI);
   6812 #endif
   6813         m_frame_parser.mutils->isNewFrame(&h264_scratch, 0, isNewFrame);
   6814         nal_count++;
   6815         if (VALID_TS(h264_last_au_ts) && !VALID_TS(pdest_frame->nTimeStamp)) {
   6816           pdest_frame->nTimeStamp = h264_last_au_ts;
   6817           pdest_frame->nFlags = h264_last_au_flags;
   6818 #ifdef PANSCAN_HDLR
   6819           if (client_extradata & OMX_FRAMEINFO_EXTRADATA)
   6820             h264_parser->update_panscan_data(h264_last_au_ts);
   6821 #endif
   6822         }
   6823         if(m_frame_parser.mutils->nalu_type == NALU_TYPE_NON_IDR ||
   6824            m_frame_parser.mutils->nalu_type == NALU_TYPE_IDR) {
   6825           h264_last_au_ts = h264_scratch.nTimeStamp;
   6826           h264_last_au_flags = h264_scratch.nFlags;
   6827 #ifndef PROCESS_EXTRADATA_IN_OUTPUT_PORT
   6828           if (client_extradata & OMX_TIMEINFO_EXTRADATA)
   6829           {
   6830             OMX_S64 ts_in_sei = h264_parser->process_ts_with_sei_vui(h264_last_au_ts);
   6831             if (!VALID_TS(h264_last_au_ts))
   6832               h264_last_au_ts = ts_in_sei;
   6833           }
   6834 #endif
   6835         } else
   6836           h264_last_au_ts = LLONG_MAX;
   6837       }
   6838 
   6839       if (!isNewFrame)
   6840       {
   6841         if ( (pdest_frame->nAllocLen - pdest_frame->nFilledLen) >=
   6842             h264_scratch.nFilledLen)
   6843         {
   6844           DEBUG_PRINT_LOW("\n Not a NewFrame Copy into Dest len %d",
   6845               h264_scratch.nFilledLen);
   6846           memcpy ((pdest_frame->pBuffer + pdest_frame->nFilledLen),
   6847               h264_scratch.pBuffer,h264_scratch.nFilledLen);
   6848           pdest_frame->nFilledLen += h264_scratch.nFilledLen;
   6849           if(m_frame_parser.mutils->nalu_type == NALU_TYPE_EOSEQ)
   6850             pdest_frame->nFlags |= QOMX_VIDEO_BUFFERFLAG_EOSEQ;
   6851           h264_scratch.nFilledLen = 0;
   6852         }
   6853         else
   6854         {
   6855           DEBUG_PRINT_LOW("\n Error:2: Destination buffer overflow for H264");
   6856           return OMX_ErrorBadParameter;
   6857         }
   6858       }
   6859       else
   6860       {
   6861         look_ahead_nal = true;
   6862         DEBUG_PRINT_LOW("\n Frame Found start Decoding Size =%d TimeStamp = %x",
   6863                      pdest_frame->nFilledLen,pdest_frame->nTimeStamp);
   6864         DEBUG_PRINT_LOW("\n Found a frame size = %d number = %d",
   6865                      pdest_frame->nFilledLen,frame_count++);
   6866 
   6867         if (pdest_frame->nFilledLen == 0)
   6868         {
   6869           DEBUG_PRINT_LOW("\n Copy the Current Frame since and push it");
   6870           look_ahead_nal = false;
   6871           if ( (pdest_frame->nAllocLen - pdest_frame->nFilledLen) >=
   6872                h264_scratch.nFilledLen)
   6873           {
   6874             memcpy ((pdest_frame->pBuffer + pdest_frame->nFilledLen),
   6875                     h264_scratch.pBuffer,h264_scratch.nFilledLen);
   6876             pdest_frame->nFilledLen += h264_scratch.nFilledLen;
   6877             h264_scratch.nFilledLen = 0;
   6878           }
   6879           else
   6880           {
   6881             DEBUG_PRINT_ERROR("\n Error:3: Destination buffer overflow for H264");
   6882             return OMX_ErrorBadParameter;
   6883           }
   6884         }
   6885         else
   6886         {
   6887           if(psource_frame->nFilledLen || h264_scratch.nFilledLen)
   6888           {
   6889             DEBUG_PRINT_LOW("\n Reset the EOS Flag");
   6890             pdest_frame->nFlags &= ~OMX_BUFFERFLAG_EOS;
   6891           }
   6892           /*Push the frame to the Decoder*/
   6893           if (empty_this_buffer_proxy(hComp,pdest_frame) != OMX_ErrorNone)
   6894           {
   6895             return OMX_ErrorBadParameter;
   6896           }
   6897           //frame_count++;
   6898           pdest_frame = NULL;
   6899           if (m_input_free_q.m_size)
   6900           {
   6901             m_input_free_q.pop_entry(&address,&p2,&id);
   6902             pdest_frame = (OMX_BUFFERHEADERTYPE *) address;
   6903             DEBUG_PRINT_LOW("\n Pop the next pdest_buffer %p",pdest_frame);
   6904             pdest_frame->nFilledLen = 0;
   6905             pdest_frame->nFlags = 0;
   6906             pdest_frame->nTimeStamp = LLONG_MAX;
   6907           }
   6908         }
   6909       }
   6910     }
   6911   }
   6912   else
   6913   {
   6914     DEBUG_PRINT_LOW("\n Not a Complete Frame, pdest_frame->nFilledLen %d",pdest_frame->nFilledLen);
   6915     /*Check if Destination Buffer is full*/
   6916     if (h264_scratch.nAllocLen ==
   6917         h264_scratch.nFilledLen + h264_scratch.nOffset)
   6918     {
   6919       DEBUG_PRINT_ERROR("\nERROR: Frame Not found though Destination Filled");
   6920       return OMX_ErrorStreamCorrupt;
   6921     }
   6922   }
   6923 
   6924   if (!psource_frame->nFilledLen)
   6925   {
   6926     DEBUG_PRINT_LOW("\n Buffer Consumed return source %p back to client",psource_frame);
   6927 
   6928     if (psource_frame->nFlags & OMX_BUFFERFLAG_EOS)
   6929     {
   6930       if (pdest_frame)
   6931       {
   6932         DEBUG_PRINT_LOW("\n EOS Reached Pass Last Buffer");
   6933         if ( (pdest_frame->nAllocLen - pdest_frame->nFilledLen) >=
   6934              h264_scratch.nFilledLen)
   6935         {
   6936           memcpy ((pdest_frame->pBuffer + pdest_frame->nFilledLen),
   6937                   h264_scratch.pBuffer,h264_scratch.nFilledLen);
   6938           pdest_frame->nFilledLen += h264_scratch.nFilledLen;
   6939           h264_scratch.nFilledLen = 0;
   6940         }
   6941         else
   6942         {
   6943           DEBUG_PRINT_ERROR("\nERROR:4: Destination buffer overflow for H264");
   6944           return OMX_ErrorBadParameter;
   6945         }
   6946         pdest_frame->nTimeStamp = h264_scratch.nTimeStamp;
   6947         pdest_frame->nFlags = h264_scratch.nFlags | psource_frame->nFlags;
   6948 
   6949         DEBUG_PRINT_LOW("\n pdest_frame->nFilledLen =%d TimeStamp = %x",
   6950                      pdest_frame->nFilledLen,pdest_frame->nTimeStamp);
   6951         DEBUG_PRINT_LOW("\n Push AU frame number %d to driver", frame_count++);
   6952 #ifndef PROCESS_EXTRADATA_IN_OUTPUT_PORT
   6953         if (client_extradata & OMX_TIMEINFO_EXTRADATA)
   6954         {
   6955           OMX_S64 ts_in_sei = h264_parser->process_ts_with_sei_vui(pdest_frame->nTimeStamp);
   6956           if (!VALID_TS(pdest_frame->nTimeStamp))
   6957             pdest_frame->nTimeStamp = ts_in_sei;
   6958         }
   6959 #endif
   6960         /*Push the frame to the Decoder*/
   6961         if (empty_this_buffer_proxy(hComp,pdest_frame) != OMX_ErrorNone)
   6962         {
   6963           return OMX_ErrorBadParameter;
   6964         }
   6965         frame_count++;
   6966         pdest_frame = NULL;
   6967       }
   6968       else
   6969       {
   6970         DEBUG_PRINT_LOW("\n Last frame in else dest addr %p size %d",
   6971                      pdest_frame,h264_scratch.nFilledLen);
   6972         generate_ebd = OMX_FALSE;
   6973       }
   6974     }
   6975   }
   6976   if(generate_ebd && !psource_frame->nFilledLen)
   6977   {
   6978     m_cb.EmptyBufferDone (hComp,m_app_data,psource_frame);
   6979     psource_frame = NULL;
   6980     if (m_input_pending_q.m_size)
   6981     {
   6982       DEBUG_PRINT_LOW("\n Pull Next source Buffer %p",psource_frame);
   6983       m_input_pending_q.pop_entry(&address,&p2,&id);
   6984       psource_frame = (OMX_BUFFERHEADERTYPE *) address;
   6985       DEBUG_PRINT_LOW("\nNext source Buffer flag %d src length %d",
   6986       psource_frame->nFlags,psource_frame->nFilledLen);
   6987     }
   6988   }
   6989   return OMX_ErrorNone;
   6990 }
   6991 
   6992 OMX_ERRORTYPE omx_vdec::push_input_vc1 (OMX_HANDLETYPE hComp)
   6993 {
   6994     OMX_U8 *buf, *pdest;
   6995     OMX_U32 partial_frame = 1;
   6996     OMX_U32 buf_len, dest_len;
   6997 
   6998     if(first_frame == 0)
   6999     {
   7000         first_frame = 1;
   7001         DEBUG_PRINT_LOW("\nFirst i/p buffer for VC1 arbitrary bytes\n");
   7002         if(!m_vendor_config.pData)
   7003         {
   7004             DEBUG_PRINT_LOW("\nCheck profile type in 1st source buffer\n");
   7005             buf = psource_frame->pBuffer;
   7006             buf_len = psource_frame->nFilledLen;
   7007 
   7008             if ((*((OMX_U32 *) buf) & VC1_SP_MP_START_CODE_MASK) ==
   7009                 VC1_SP_MP_START_CODE)
   7010             {
   7011                 m_vc1_profile = VC1_SP_MP_RCV;
   7012             }
   7013             else if(*((OMX_U32 *) buf) & VC1_AP_SEQ_START_CODE)
   7014             {
   7015                 m_vc1_profile = VC1_AP;
   7016             }
   7017             else
   7018             {
   7019                 DEBUG_PRINT_ERROR("\nInvalid sequence layer in first buffer\n");
   7020                 return OMX_ErrorStreamCorrupt;
   7021             }
   7022         }
   7023         else
   7024         {
   7025             pdest = pdest_frame->pBuffer + pdest_frame->nFilledLen +
   7026                 pdest_frame->nOffset;
   7027             dest_len = pdest_frame->nAllocLen - (pdest_frame->nFilledLen +
   7028                 pdest_frame->nOffset);
   7029 
   7030             if(dest_len < m_vendor_config.nDataSize)
   7031             {
   7032                 DEBUG_PRINT_ERROR("\nDestination buffer full\n");
   7033                 return OMX_ErrorBadParameter;
   7034             }
   7035             else
   7036             {
   7037                 memcpy(pdest, m_vendor_config.pData, m_vendor_config.nDataSize);
   7038                 pdest_frame->nFilledLen += m_vendor_config.nDataSize;
   7039             }
   7040         }
   7041     }
   7042 
   7043     switch(m_vc1_profile)
   7044     {
   7045         case VC1_AP:
   7046             DEBUG_PRINT_LOW("\n VC1 AP, hence parse using frame start code");
   7047             if (push_input_sc_codec(hComp) != OMX_ErrorNone)
   7048             {
   7049                 DEBUG_PRINT_ERROR("\n Error In Parsing VC1 AP start code");
   7050                 return OMX_ErrorBadParameter;
   7051             }
   7052         break;
   7053 
   7054         case VC1_SP_MP_RCV:
   7055         default:
   7056             DEBUG_PRINT_ERROR("\n Unsupported VC1 profile in ArbitraryBytes Mode\n");
   7057             return OMX_ErrorBadParameter;
   7058     }
   7059     return OMX_ErrorNone;
   7060 }
   7061 
   7062 #ifdef USE_ION
   7063 int omx_vdec::alloc_map_ion_memory(OMX_U32 buffer_size,
   7064               OMX_U32 alignment, struct ion_allocation_data *alloc_data,
   7065 	      struct ion_fd_data *fd_data, int flag)
   7066 {
   7067   int fd = -EINVAL;
   7068   int rc = -EINVAL;
   7069   int ion_dev_flag;
   7070   struct vdec_ion ion_buf_info;
   7071   if (!alloc_data || buffer_size <= 0 || !fd_data) {
   7072      DEBUG_PRINT_ERROR("Invalid arguments to alloc_map_ion_memory\n");
   7073      return -EINVAL;
   7074   }
   7075   ion_dev_flag = O_RDONLY;
   7076   fd = open (MEM_DEVICE, ion_dev_flag);
   7077   if (fd < 0) {
   7078     DEBUG_PRINT_ERROR("opening ion device failed with fd = %d\n", fd);
   7079     return fd;
   7080   }
   7081   alloc_data->flags = 0;
   7082   if(!secure_mode && (flag & ION_FLAG_CACHED))
   7083   {
   7084     alloc_data->flags |= ION_FLAG_CACHED;
   7085   }
   7086   alloc_data->len = buffer_size;
   7087   alloc_data->align = clip2(alignment);
   7088   if (alloc_data->align < 4096)
   7089   {
   7090     alloc_data->align = 4096;
   7091   }
   7092   if(secure_mode) {
   7093     alloc_data->heap_mask = ION_HEAP(MEM_HEAP_ID);
   7094     alloc_data->flags |= ION_SECURE;
   7095   } else {
   7096     alloc_data->heap_mask = ION_HEAP(MEM_HEAP_ID);
   7097   }
   7098   rc = ioctl(fd,ION_IOC_ALLOC,alloc_data);
   7099   if (rc || !alloc_data->handle) {
   7100     DEBUG_PRINT_ERROR("\n ION ALLOC memory failed ");
   7101     alloc_data->handle = NULL;
   7102     close(fd);
   7103     fd = -ENOMEM;
   7104     return fd;
   7105   }
   7106   fd_data->handle = alloc_data->handle;
   7107   rc = ioctl(fd,ION_IOC_MAP,fd_data);
   7108   if (rc) {
   7109     DEBUG_PRINT_ERROR("\n ION MAP failed ");
   7110     ion_buf_info.ion_alloc_data = *alloc_data;
   7111     ion_buf_info.ion_device_fd = fd;
   7112     ion_buf_info.fd_ion_data = *fd_data;
   7113     free_ion_memory(&ion_buf_info);
   7114     fd_data->fd =-1;
   7115     close(fd);
   7116     fd = -ENOMEM;
   7117   }
   7118 
   7119   return fd;
   7120 }
   7121 
   7122 void omx_vdec::free_ion_memory(struct vdec_ion *buf_ion_info) {
   7123 
   7124      if(!buf_ion_info) {
   7125        DEBUG_PRINT_ERROR("\n ION: free called with invalid fd/allocdata");
   7126        return;
   7127      }
   7128      if(ioctl(buf_ion_info->ion_device_fd,ION_IOC_FREE,
   7129              &buf_ion_info->ion_alloc_data.handle)) {
   7130        DEBUG_PRINT_ERROR("\n ION: free failed" );
   7131      }
   7132      close(buf_ion_info->ion_device_fd);
   7133      buf_ion_info->ion_device_fd = -1;
   7134      buf_ion_info->ion_alloc_data.handle = NULL;
   7135      buf_ion_info->fd_ion_data.fd = -1;
   7136 }
   7137 #else
   7138 bool omx_vdec::align_pmem_buffers(int pmem_fd, OMX_U32 buffer_size,
   7139                                   OMX_U32 alignment)
   7140 {
   7141   struct pmem_allocation allocation;
   7142   allocation.size = buffer_size;
   7143   allocation.align = clip2(alignment);
   7144   if (allocation.align < 4096)
   7145   {
   7146     allocation.align = 4096;
   7147   }
   7148   if (ioctl(pmem_fd, PMEM_ALLOCATE_ALIGNED, &allocation) < 0)
   7149   {
   7150     DEBUG_PRINT_ERROR("\n Aligment(%u) failed with pmem driver Sz(%lu)",
   7151       allocation.align, allocation.size);
   7152     return false;
   7153   }
   7154   return true;
   7155 }
   7156 #endif
   7157 void omx_vdec::free_output_buffer_header()
   7158 {
   7159   DEBUG_PRINT_HIGH("\n ALL output buffers are freed/released");
   7160   output_use_buffer = false;
   7161   ouput_egl_buffers = false;
   7162 
   7163   if (m_out_mem_ptr)
   7164   {
   7165     free (m_out_mem_ptr);
   7166     m_out_mem_ptr = NULL;
   7167   }
   7168 
   7169   if(m_platform_list)
   7170   {
   7171     free(m_platform_list);
   7172     m_platform_list = NULL;
   7173   }
   7174 
   7175   if (drv_ctx.ptr_respbuffer)
   7176   {
   7177     free (drv_ctx.ptr_respbuffer);
   7178     drv_ctx.ptr_respbuffer = NULL;
   7179   }
   7180   if (drv_ctx.ptr_outputbuffer)
   7181   {
   7182     free (drv_ctx.ptr_outputbuffer);
   7183     drv_ctx.ptr_outputbuffer = NULL;
   7184   }
   7185 #ifdef USE_ION
   7186     if (drv_ctx.op_buf_ion_info) {
   7187         DEBUG_PRINT_LOW("\n Free o/p ion context");
   7188 	free(drv_ctx.op_buf_ion_info);
   7189         drv_ctx.op_buf_ion_info = NULL;
   7190     }
   7191 #endif
   7192 }
   7193 
   7194 void omx_vdec::free_input_buffer_header()
   7195 {
   7196     input_use_buffer = false;
   7197     if (arbitrary_bytes)
   7198     {
   7199       if (m_frame_parser.mutils)
   7200       {
   7201         DEBUG_PRINT_LOW("\n Free utils parser");
   7202         delete (m_frame_parser.mutils);
   7203         m_frame_parser.mutils = NULL;
   7204       }
   7205 
   7206       if (m_inp_heap_ptr)
   7207       {
   7208         DEBUG_PRINT_LOW("\n Free input Heap Pointer");
   7209         free (m_inp_heap_ptr);
   7210         m_inp_heap_ptr = NULL;
   7211       }
   7212 
   7213       if (m_phdr_pmem_ptr)
   7214       {
   7215         DEBUG_PRINT_LOW("\n Free input pmem header Pointer");
   7216         free (m_phdr_pmem_ptr);
   7217         m_phdr_pmem_ptr = NULL;
   7218       }
   7219     }
   7220     if (m_inp_mem_ptr)
   7221     {
   7222       DEBUG_PRINT_LOW("\n Free input pmem Pointer area");
   7223       free (m_inp_mem_ptr);
   7224       m_inp_mem_ptr = NULL;
   7225     }
   7226     if (drv_ctx.ptr_inputbuffer)
   7227     {
   7228       DEBUG_PRINT_LOW("\n Free Driver Context pointer");
   7229       free (drv_ctx.ptr_inputbuffer);
   7230       drv_ctx.ptr_inputbuffer = NULL;
   7231     }
   7232 #ifdef USE_ION
   7233     if (drv_ctx.ip_buf_ion_info) {
   7234         DEBUG_PRINT_LOW("\n Free ion context");
   7235 	free(drv_ctx.ip_buf_ion_info);
   7236         drv_ctx.ip_buf_ion_info = NULL;
   7237     }
   7238 #endif
   7239 }
   7240 void omx_vdec::stream_off()
   7241 {
   7242 	int rc=0;
   7243 	enum v4l2_buf_type btype;
   7244 	btype = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   7245 	rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_STREAMOFF, &btype);
   7246 	if (rc) {
   7247 		/*TODO: How to handle this case */
   7248 		printf("\n Failed to call streamoff on OUTPUT Port \n");
   7249 	} else {
   7250 		streaming[CAPTURE_PORT] = false;
   7251 	}
   7252 }
   7253 
   7254 OMX_ERRORTYPE omx_vdec::get_buffer_req(vdec_allocatorproperty *buffer_prop)
   7255 {
   7256   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   7257   struct v4l2_requestbuffers bufreq;
   7258   unsigned int buf_size = 0, extra_data_size = 0;
   7259   struct v4l2_format fmt;
   7260   int ret;
   7261     DEBUG_PRINT_LOW("GetBufReq IN: ActCnt(%d) Size(%d)",
   7262     buffer_prop->actualcount, buffer_prop->buffer_size);
   7263 	bufreq.memory = V4L2_MEMORY_USERPTR;
   7264 	if(in_reconfig == true)
   7265 	bufreq.count = 0;
   7266 	else
   7267 	bufreq.count = 2;
   7268    if(buffer_prop->buffer_type == VDEC_BUFFER_TYPE_INPUT){
   7269     bufreq.type=V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
   7270 	fmt.type =V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
   7271     fmt.fmt.pix_mp.pixelformat = output_capability;
   7272   }else if (buffer_prop->buffer_type == VDEC_BUFFER_TYPE_OUTPUT){
   7273     bufreq.type=V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   7274 	fmt.type =V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   7275     fmt.fmt.pix_mp.pixelformat = capture_capability;
   7276   }else {eRet = OMX_ErrorBadParameter;}
   7277   if(eRet==OMX_ErrorNone){
   7278   ret = ioctl(drv_ctx.video_driver_fd,VIDIOC_REQBUFS, &bufreq);
   7279   }
   7280   if(ret)
   7281   {
   7282     DEBUG_PRINT_ERROR("Requesting buffer requirements failed");
   7283 	/*TODO: How to handle this case */
   7284     eRet = OMX_ErrorInsufficientResources;
   7285 	return eRet;
   7286   }
   7287   else
   7288   {
   7289   buffer_prop->actualcount = bufreq.count;
   7290   buffer_prop->mincount = bufreq.count;
   7291   printf("Count = %d \n ",bufreq.count);
   7292   }
   7293   DEBUG_PRINT_LOW("GetBufReq IN: ActCnt(%d) Size(%d)",
   7294     buffer_prop->actualcount, buffer_prop->buffer_size);
   7295 
   7296   fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
   7297   fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
   7298 
   7299   ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
   7300 
   7301   drv_ctx.video_resolution.frame_height = fmt.fmt.pix_mp.height;
   7302   drv_ctx.video_resolution.frame_width = fmt.fmt.pix_mp.width;
   7303 
   7304   printf("Buffer Size = %d \n ",fmt.fmt.pix_mp.plane_fmt[0].sizeimage);
   7305 
   7306   if(ret)
   7307   {
   7308 	/*TODO: How to handle this case */
   7309     DEBUG_PRINT_ERROR("Requesting buffer requirements failed");
   7310     eRet = OMX_ErrorInsufficientResources;
   7311   }
   7312   else
   7313   {
   7314     buffer_prop->buffer_size = fmt.fmt.pix_mp.plane_fmt[0].sizeimage;
   7315     buf_size = buffer_prop->buffer_size;
   7316     if (client_extradata & OMX_FRAMEINFO_EXTRADATA)
   7317     {
   7318       DEBUG_PRINT_HIGH("Frame info extra data enabled!");
   7319       extra_data_size += OMX_FRAMEINFO_EXTRADATA_SIZE;
   7320     }
   7321     if (client_extradata & OMX_INTERLACE_EXTRADATA)
   7322     {
   7323       DEBUG_PRINT_HIGH("Interlace extra data enabled!");
   7324       extra_data_size += OMX_INTERLACE_EXTRADATA_SIZE;
   7325     }
   7326     if (client_extradata & OMX_PORTDEF_EXTRADATA)
   7327     {
   7328        extra_data_size += OMX_PORTDEF_EXTRADATA_SIZE;
   7329        DEBUG_PRINT_HIGH("Smooth streaming enabled extra_data_size=%d\n",
   7330          extra_data_size);
   7331     }
   7332     if (extra_data_size)
   7333     {
   7334       extra_data_size += sizeof(OMX_OTHER_EXTRADATATYPE); //Space for terminator
   7335       buf_size = ((buf_size + 3)&(~3)); //Align extradata start address to 64Bit
   7336     }
   7337     buf_size += extra_data_size;
   7338     buf_size = (buf_size + buffer_prop->alignment - 1)&(~(buffer_prop->alignment - 1));
   7339     DEBUG_PRINT_LOW("GetBufReq UPDATE: ActCnt(%d) Size(%d) BufSize(%d)",
   7340       buffer_prop->actualcount, buffer_prop->buffer_size, buf_size);
   7341     if (in_reconfig) // BufReq will be set to driver when port is disabled
   7342       buffer_prop->buffer_size = buf_size;
   7343     else if (buf_size != buffer_prop->buffer_size)
   7344     {
   7345       buffer_prop->buffer_size = buf_size;
   7346       eRet = set_buffer_req(buffer_prop);
   7347     }
   7348   }
   7349   DEBUG_PRINT_LOW("GetBufReq OUT: ActCnt(%d) Size(%d)",
   7350     buffer_prop->actualcount, buffer_prop->buffer_size);
   7351   return eRet;
   7352 }
   7353 
   7354 OMX_ERRORTYPE omx_vdec::set_buffer_req(vdec_allocatorproperty *buffer_prop)
   7355 {
   7356   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   7357   unsigned buf_size = 0;
   7358   struct v4l2_format fmt;
   7359   int ret;
   7360   DEBUG_PRINT_LOW("SetBufReq IN: ActCnt(%d) Size(%d)",
   7361     buffer_prop->actualcount, buffer_prop->buffer_size);
   7362   buf_size = (buffer_prop->buffer_size + buffer_prop->alignment - 1)&(~(buffer_prop->alignment - 1));
   7363   if (buf_size != buffer_prop->buffer_size)
   7364   {
   7365     DEBUG_PRINT_ERROR("Buffer size alignment error: Requested(%d) Required(%d)",
   7366       buffer_prop->buffer_size, buf_size);
   7367     eRet = OMX_ErrorBadParameter;
   7368   }
   7369   else
   7370   {
   7371     fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
   7372 	fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
   7373 	if(buffer_prop->buffer_type == VDEC_BUFFER_TYPE_INPUT){
   7374 	fmt.type =V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
   7375 	fmt.fmt.pix_mp.pixelformat = output_capability;
   7376 	}else if (buffer_prop->buffer_type == VDEC_BUFFER_TYPE_OUTPUT){
   7377 fmt.type =V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   7378 	fmt.fmt.pix_mp.pixelformat = capture_capability;
   7379 } else {eRet = OMX_ErrorBadParameter;}
   7380     ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
   7381     if(ret)
   7382     {
   7383 	  /*TODO: How to handle this case */
   7384       DEBUG_PRINT_ERROR("Setting buffer requirements failed");
   7385       eRet = OMX_ErrorInsufficientResources;
   7386     }
   7387   }
   7388   return eRet;
   7389 }
   7390 
   7391 OMX_ERRORTYPE omx_vdec::start_port_reconfig()
   7392 {
   7393   struct vdec_ioctl_msg ioctl_msg = {NULL, NULL};
   7394   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   7395   enum v4l2_buf_type btype;
   7396   int rc = 0,i;
   7397   struct v4l2_plane plane;
   7398   struct v4l2_buffer v4l2_buf ={0};
   7399   struct v4l2_decoder_cmd dec;
   7400   dec.cmd = V4L2_DEC_CMD_STOP;
   7401   rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_DECODER_CMD, &dec);
   7402   btype = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
   7403   in_reconfig = true;
   7404   return eRet;
   7405 }
   7406 
   7407 OMX_ERRORTYPE omx_vdec::update_picture_resolution()
   7408 {
   7409   struct vdec_ioctl_msg ioctl_msg = {NULL, NULL};
   7410   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   7411   ioctl_msg.in = NULL;
   7412   ioctl_msg.out = &drv_ctx.video_resolution;
   7413   if (/*ioctl(drv_ctx.video_driver_fd, VDEC_IOCTL_GET_PICRES, &ioctl_msg)*/0)
   7414   {
   7415     DEBUG_PRINT_ERROR("Error VDEC_IOCTL_GET_PICRES");
   7416     eRet = OMX_ErrorHardware;
   7417   }
   7418   return eRet;
   7419 }
   7420 
   7421 OMX_ERRORTYPE omx_vdec::update_portdef(OMX_PARAM_PORTDEFINITIONTYPE *portDefn)
   7422 {
   7423   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   7424   if (!portDefn)
   7425   {
   7426     return OMX_ErrorBadParameter;
   7427   }
   7428   DEBUG_PRINT_LOW("omx_vdec::update_portdef\n");
   7429   portDefn->nVersion.nVersion = OMX_SPEC_VERSION;
   7430   portDefn->nSize = sizeof(portDefn);
   7431   portDefn->eDomain    = OMX_PortDomainVideo;
   7432   if (drv_ctx.frame_rate.fps_denominator > 0)
   7433     portDefn->format.video.xFramerate = drv_ctx.frame_rate.fps_numerator /
   7434                                         drv_ctx.frame_rate.fps_denominator;
   7435   else {
   7436     DEBUG_PRINT_ERROR("Error: Divide by zero \n");
   7437     return OMX_ErrorBadParameter;
   7438   }
   7439   if (0 == portDefn->nPortIndex)
   7440   {
   7441     portDefn->eDir =  OMX_DirInput;
   7442     portDefn->nBufferCountActual = drv_ctx.ip_buf.actualcount;
   7443     portDefn->nBufferCountMin    = drv_ctx.ip_buf.mincount;
   7444     portDefn->nBufferSize        = drv_ctx.ip_buf.buffer_size;
   7445     portDefn->format.video.eColorFormat = OMX_COLOR_FormatUnused;
   7446     portDefn->format.video.eCompressionFormat = eCompressionFormat;
   7447     portDefn->bEnabled   = m_inp_bEnabled;
   7448     portDefn->bPopulated = m_inp_bPopulated;
   7449   }
   7450   else if (1 == portDefn->nPortIndex)
   7451   {
   7452     portDefn->eDir =  OMX_DirOutput;
   7453     eRet=get_buffer_req(&drv_ctx.op_buf);
   7454     if (in_reconfig)
   7455     {
   7456       portDefn->nBufferCountActual = op_buf_rcnfg.actualcount;
   7457       portDefn->nBufferCountMin    = op_buf_rcnfg.mincount;
   7458       portDefn->nBufferSize        = op_buf_rcnfg.buffer_size;
   7459     }
   7460     else
   7461     {
   7462       portDefn->nBufferCountActual = drv_ctx.op_buf.actualcount;
   7463       portDefn->nBufferCountMin    = drv_ctx.op_buf.mincount;
   7464       portDefn->nBufferSize        = drv_ctx.op_buf.buffer_size;
   7465     }
   7466     portDefn->format.video.eCompressionFormat = OMX_VIDEO_CodingUnused;
   7467     portDefn->bEnabled   = m_out_bEnabled;
   7468     portDefn->bPopulated = m_out_bPopulated;
   7469     if (drv_ctx.output_format == VDEC_YUV_FORMAT_NV12)
   7470       portDefn->format.video.eColorFormat = OMX_COLOR_FormatYUV420SemiPlanar;
   7471     else if (drv_ctx.output_format == VDEC_YUV_FORMAT_TILE_4x2)
   7472       portDefn->format.video.eColorFormat = (OMX_COLOR_FORMATTYPE)
   7473         QOMX_COLOR_FormatYUV420PackedSemiPlanar64x32Tile2m8ka;
   7474     else
   7475     {
   7476       DEBUG_PRINT_ERROR("ERROR: Color format unknown: %x\n", drv_ctx.output_format);
   7477     }
   7478   }
   7479   else
   7480   {
   7481     portDefn->eDir = OMX_DirMax;
   7482     DEBUG_PRINT_LOW(" get_parameter: Bad Port idx %d",
   7483              (int)portDefn->nPortIndex);
   7484     eRet = OMX_ErrorBadPortIndex;
   7485   }
   7486   portDefn->format.video.nFrameHeight =  drv_ctx.video_resolution.frame_height;
   7487   portDefn->format.video.nFrameWidth  =  drv_ctx.video_resolution.frame_width;
   7488   portDefn->format.video.nStride = drv_ctx.video_resolution.stride;
   7489   portDefn->format.video.nSliceHeight = drv_ctx.video_resolution.scan_lines;
   7490   DEBUG_PRINT_LOW("update_portdef Width = %d Height = %d Stride = %u"
   7491     "SliceHeight = %u \n", portDefn->format.video.nFrameHeight,
   7492     portDefn->format.video.nFrameWidth,
   7493     portDefn->format.video.nStride,
   7494     portDefn->format.video.nSliceHeight);
   7495   return eRet;
   7496 
   7497 }
   7498 
   7499 OMX_ERRORTYPE omx_vdec::allocate_output_headers()
   7500 {
   7501   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   7502   OMX_BUFFERHEADERTYPE *bufHdr = NULL;
   7503   unsigned i= 0;
   7504 
   7505   if(!m_out_mem_ptr) {
   7506     DEBUG_PRINT_HIGH("\n Use o/p buffer case - Header List allocation");
   7507     int nBufHdrSize        = 0;
   7508     int nPlatformEntrySize = 0;
   7509     int nPlatformListSize  = 0;
   7510     int nPMEMInfoSize = 0;
   7511     OMX_QCOM_PLATFORM_PRIVATE_LIST      *pPlatformList;
   7512     OMX_QCOM_PLATFORM_PRIVATE_ENTRY     *pPlatformEntry;
   7513     OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *pPMEMInfo;
   7514 
   7515     DEBUG_PRINT_LOW("Setting First Output Buffer(%d)\n",
   7516       drv_ctx.op_buf.actualcount);
   7517     nBufHdrSize        = drv_ctx.op_buf.actualcount *
   7518                          sizeof(OMX_BUFFERHEADERTYPE);
   7519 
   7520     nPMEMInfoSize      = drv_ctx.op_buf.actualcount *
   7521                          sizeof(OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO);
   7522     nPlatformListSize  = drv_ctx.op_buf.actualcount *
   7523                          sizeof(OMX_QCOM_PLATFORM_PRIVATE_LIST);
   7524     nPlatformEntrySize = drv_ctx.op_buf.actualcount *
   7525                          sizeof(OMX_QCOM_PLATFORM_PRIVATE_ENTRY);
   7526 
   7527     DEBUG_PRINT_LOW("TotalBufHdr %d BufHdrSize %d PMEM %d PL %d\n",nBufHdrSize,
   7528                          sizeof(OMX_BUFFERHEADERTYPE),
   7529                          nPMEMInfoSize,
   7530                          nPlatformListSize);
   7531     DEBUG_PRINT_LOW("PE %d bmSize %d \n",nPlatformEntrySize,
   7532                          m_out_bm_count);
   7533     m_out_mem_ptr = (OMX_BUFFERHEADERTYPE  *)calloc(nBufHdrSize,1);
   7534     // Alloc mem for platform specific info
   7535     char *pPtr=NULL;
   7536     pPtr = (char*) calloc(nPlatformListSize + nPlatformEntrySize +
   7537                                      nPMEMInfoSize,1);
   7538     drv_ctx.ptr_outputbuffer = (struct vdec_bufferpayload *) \
   7539       calloc (sizeof(struct vdec_bufferpayload),
   7540       drv_ctx.op_buf.actualcount);
   7541     drv_ctx.ptr_respbuffer = (struct vdec_output_frameinfo  *)\
   7542       calloc (sizeof (struct vdec_output_frameinfo),
   7543       drv_ctx.op_buf.actualcount);
   7544 #ifdef USE_ION
   7545     drv_ctx.op_buf_ion_info = (struct vdec_ion * ) \
   7546       calloc (sizeof(struct vdec_ion),drv_ctx.op_buf.actualcount);
   7547 #endif
   7548 
   7549     if(m_out_mem_ptr && pPtr && drv_ctx.ptr_outputbuffer
   7550        && drv_ctx.ptr_respbuffer)
   7551     {
   7552       bufHdr          =  m_out_mem_ptr;
   7553       m_platform_list = (OMX_QCOM_PLATFORM_PRIVATE_LIST *)(pPtr);
   7554       m_platform_entry= (OMX_QCOM_PLATFORM_PRIVATE_ENTRY *)
   7555                         (((char *) m_platform_list)  + nPlatformListSize);
   7556       m_pmem_info     = (OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *)
   7557                         (((char *) m_platform_entry) + nPlatformEntrySize);
   7558       pPlatformList   = m_platform_list;
   7559       pPlatformEntry  = m_platform_entry;
   7560       pPMEMInfo       = m_pmem_info;
   7561 
   7562       DEBUG_PRINT_LOW("Memory Allocation Succeeded for OUT port%p\n",m_out_mem_ptr);
   7563 
   7564       // Settting the entire storage nicely
   7565       DEBUG_PRINT_LOW("bHdr %p OutMem %p PE %p\n",bufHdr,
   7566                       m_out_mem_ptr,pPlatformEntry);
   7567       DEBUG_PRINT_LOW(" Pmem Info = %p \n",pPMEMInfo);
   7568       for(i=0; i < drv_ctx.op_buf.actualcount ; i++)
   7569       {
   7570         bufHdr->nSize              = sizeof(OMX_BUFFERHEADERTYPE);
   7571         bufHdr->nVersion.nVersion  = OMX_SPEC_VERSION;
   7572         // Set the values when we determine the right HxW param
   7573         bufHdr->nAllocLen          = 0;
   7574         bufHdr->nFilledLen         = 0;
   7575         bufHdr->pAppPrivate        = NULL;
   7576         bufHdr->nOutputPortIndex   = OMX_CORE_OUTPUT_PORT_INDEX;
   7577         pPlatformEntry->type       = OMX_QCOM_PLATFORM_PRIVATE_PMEM;
   7578         pPlatformEntry->entry      = pPMEMInfo;
   7579         // Initialize the Platform List
   7580         pPlatformList->nEntries    = 1;
   7581         pPlatformList->entryList   = pPlatformEntry;
   7582         // Keep pBuffer NULL till vdec is opened
   7583         bufHdr->pBuffer            = NULL;
   7584         pPMEMInfo->offset          =  0;
   7585         pPMEMInfo->pmem_fd = 0;
   7586         bufHdr->pPlatformPrivate = pPlatformList;
   7587         drv_ctx.ptr_outputbuffer[i].pmem_fd = -1;
   7588 #ifdef USE_ION
   7589         drv_ctx.op_buf_ion_info[i].ion_device_fd =-1;
   7590 #endif
   7591         /*Create a mapping between buffers*/
   7592         bufHdr->pOutputPortPrivate = &drv_ctx.ptr_respbuffer[i];
   7593         drv_ctx.ptr_respbuffer[i].client_data = (void *) \
   7594                                             &drv_ctx.ptr_outputbuffer[i];
   7595         // Move the buffer and buffer header pointers
   7596         bufHdr++;
   7597         pPMEMInfo++;
   7598         pPlatformEntry++;
   7599         pPlatformList++;
   7600       }
   7601     }
   7602     else
   7603     {
   7604       DEBUG_PRINT_ERROR("Output buf mem alloc failed[0x%x][0x%x]\n",\
   7605                                         m_out_mem_ptr, pPtr);
   7606       if(m_out_mem_ptr)
   7607       {
   7608         free(m_out_mem_ptr);
   7609         m_out_mem_ptr = NULL;
   7610       }
   7611       if(pPtr)
   7612       {
   7613         free(pPtr);
   7614         pPtr = NULL;
   7615       }
   7616       if(drv_ctx.ptr_outputbuffer)
   7617       {
   7618         free(drv_ctx.ptr_outputbuffer);
   7619         drv_ctx.ptr_outputbuffer = NULL;
   7620       }
   7621       if(drv_ctx.ptr_respbuffer)
   7622       {
   7623         free(drv_ctx.ptr_respbuffer);
   7624         drv_ctx.ptr_respbuffer = NULL;
   7625       }
   7626 #ifdef USE_ION
   7627     if (drv_ctx.op_buf_ion_info) {
   7628         DEBUG_PRINT_LOW("\n Free o/p ion context");
   7629 	free(drv_ctx.op_buf_ion_info);
   7630         drv_ctx.op_buf_ion_info = NULL;
   7631     }
   7632 #endif
   7633       eRet =  OMX_ErrorInsufficientResources;
   7634     }
   7635   } else {
   7636     eRet =  OMX_ErrorInsufficientResources;
   7637   }
   7638   return eRet;
   7639 }
   7640 
   7641 void omx_vdec::complete_pending_buffer_done_cbs()
   7642 {
   7643   unsigned p1;
   7644   unsigned p2;
   7645   unsigned ident;
   7646   omx_cmd_queue tmp_q, pending_bd_q;
   7647   pthread_mutex_lock(&m_lock);
   7648   // pop all pending GENERATE FDB from ftb queue
   7649   while (m_ftb_q.m_size)
   7650   {
   7651     m_ftb_q.pop_entry(&p1,&p2,&ident);
   7652     if(ident == OMX_COMPONENT_GENERATE_FBD)
   7653     {
   7654       pending_bd_q.insert_entry(p1,p2,ident);
   7655     }
   7656     else
   7657     {
   7658       tmp_q.insert_entry(p1,p2,ident);
   7659     }
   7660   }
   7661   //return all non GENERATE FDB to ftb queue
   7662   while(tmp_q.m_size)
   7663   {
   7664     tmp_q.pop_entry(&p1,&p2,&ident);
   7665     m_ftb_q.insert_entry(p1,p2,ident);
   7666   }
   7667   // pop all pending GENERATE EDB from etb queue
   7668   while (m_etb_q.m_size)
   7669   {
   7670     m_etb_q.pop_entry(&p1,&p2,&ident);
   7671     if(ident == OMX_COMPONENT_GENERATE_EBD)
   7672     {
   7673       pending_bd_q.insert_entry(p1,p2,ident);
   7674     }
   7675     else
   7676     {
   7677       tmp_q.insert_entry(p1,p2,ident);
   7678     }
   7679   }
   7680   //return all non GENERATE FDB to etb queue
   7681   while(tmp_q.m_size)
   7682   {
   7683     tmp_q.pop_entry(&p1,&p2,&ident);
   7684     m_etb_q.insert_entry(p1,p2,ident);
   7685   }
   7686   pthread_mutex_unlock(&m_lock);
   7687   // process all pending buffer dones
   7688   while(pending_bd_q.m_size)
   7689   {
   7690     pending_bd_q.pop_entry(&p1,&p2,&ident);
   7691     switch(ident)
   7692     {
   7693       case OMX_COMPONENT_GENERATE_EBD:
   7694         if(empty_buffer_done(&m_cmp, (OMX_BUFFERHEADERTYPE *)p1) != OMX_ErrorNone)
   7695         {
   7696           DEBUG_PRINT_ERROR("\nERROR: empty_buffer_done() failed!\n");
   7697           omx_report_error ();
   7698         }
   7699         break;
   7700 
   7701       case OMX_COMPONENT_GENERATE_FBD:
   7702         if(fill_buffer_done(&m_cmp, (OMX_BUFFERHEADERTYPE *)p1) != OMX_ErrorNone )
   7703         {
   7704           DEBUG_PRINT_ERROR("\nERROR: fill_buffer_done() failed!\n");
   7705           omx_report_error ();
   7706         }
   7707         break;
   7708     }
   7709   }
   7710 }
   7711 
   7712 void omx_vdec::set_frame_rate(OMX_S64 act_timestamp)
   7713 {
   7714   OMX_U32 new_frame_interval = 0;
   7715   struct vdec_ioctl_msg ioctl_msg = {NULL, NULL};
   7716   if (VALID_TS(act_timestamp) && VALID_TS(prev_ts) && act_timestamp != prev_ts
   7717      && (((act_timestamp > prev_ts )? act_timestamp - prev_ts: prev_ts-act_timestamp)>2000))
   7718   {
   7719     new_frame_interval = (act_timestamp > prev_ts)?
   7720                           act_timestamp - prev_ts :
   7721                           prev_ts - act_timestamp;
   7722     if (new_frame_interval < frm_int || frm_int == 0)
   7723     {
   7724       frm_int = new_frame_interval;
   7725       if(frm_int)
   7726       {
   7727         drv_ctx.frame_rate.fps_numerator = 1e6;
   7728         drv_ctx.frame_rate.fps_denominator = frm_int;
   7729         DEBUG_PRINT_LOW("set_frame_rate: frm_int(%u) fps(%f)",
   7730                          frm_int, drv_ctx.frame_rate.fps_numerator /
   7731                          (float)drv_ctx.frame_rate.fps_denominator);
   7732         ioctl_msg.in = &drv_ctx.frame_rate;
   7733         if (/*ioctl (drv_ctx.video_driver_fd, VDEC_IOCTL_SET_FRAME_RATE,
   7734                   (void*)&ioctl_msg) < */0)
   7735         {
   7736           DEBUG_PRINT_ERROR("Setting frame rate failed");
   7737         }
   7738       }
   7739     }
   7740   }
   7741   prev_ts = act_timestamp;
   7742 }
   7743 
   7744 void omx_vdec::adjust_timestamp(OMX_S64 &act_timestamp)
   7745 {
   7746   if (rst_prev_ts && VALID_TS(act_timestamp))
   7747   {
   7748     prev_ts = act_timestamp;
   7749     rst_prev_ts = false;
   7750   }
   7751   else if (VALID_TS(prev_ts))
   7752   {
   7753     bool codec_cond = (drv_ctx.timestamp_adjust)?
   7754                       (!VALID_TS(act_timestamp) || (((act_timestamp > prev_ts)?
   7755                       (act_timestamp - prev_ts):(prev_ts - act_timestamp)) <= 2000)):
   7756                       (!VALID_TS(act_timestamp) || act_timestamp == prev_ts);
   7757     if(frm_int > 0 && codec_cond)
   7758     {
   7759       DEBUG_PRINT_LOW("adjust_timestamp: original ts[%lld]", act_timestamp);
   7760       act_timestamp = prev_ts + frm_int;
   7761       DEBUG_PRINT_LOW("adjust_timestamp: predicted ts[%lld]", act_timestamp);
   7762       prev_ts = act_timestamp;
   7763     }
   7764     else
   7765       set_frame_rate(act_timestamp);
   7766   }
   7767   else if (frm_int > 0)           // In this case the frame rate was set along
   7768   {                               // with the port definition, start ts with 0
   7769     act_timestamp = prev_ts = 0;  // and correct if a valid ts is received.
   7770     rst_prev_ts = true;
   7771   }
   7772 }
   7773 
   7774 void omx_vdec::handle_extradata(OMX_BUFFERHEADERTYPE *p_buf_hdr)
   7775 {
   7776   OMX_OTHER_EXTRADATATYPE *p_extra = NULL, *p_sei = NULL, *p_vui = NULL;
   7777   OMX_U32 num_conceal_MB = 0;
   7778   OMX_S64 ts_in_sei = 0;
   7779   OMX_U32 frame_rate = 0;
   7780   p_extra = (OMX_OTHER_EXTRADATATYPE *)
   7781            ((unsigned)(p_buf_hdr->pBuffer + p_buf_hdr->nOffset +
   7782             p_buf_hdr->nFilledLen + 3)&(~3));
   7783   if ((OMX_U8*)p_extra > (p_buf_hdr->pBuffer + p_buf_hdr->nAllocLen))
   7784     p_extra = NULL;
   7785   if (drv_ctx.extradata && (p_buf_hdr->nFlags & OMX_BUFFERFLAG_EXTRADATA))
   7786   {
   7787     // Process driver extradata
   7788     while(p_extra && p_extra->eType != VDEC_EXTRADATA_NONE)
   7789     {
   7790       DEBUG_PRINT_LOW("handle_extradata : pBuf(%p) BufTS(%lld) Type(%x) DataSz(%u)",
   7791            p_buf_hdr, p_buf_hdr->nTimeStamp, p_extra->eType, p_extra->nDataSize);
   7792       if (p_extra->nSize < p_extra->nDataSize)
   7793       {
   7794         DEBUG_PRINT_ERROR(" \n Corrupt metadata Buffer size %d payload size %d",
   7795                           p_extra->nSize, p_extra->nDataSize);
   7796         p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + p_extra->nSize);
   7797         if ((OMX_U8*)p_extra > (p_buf_hdr->pBuffer + p_buf_hdr->nAllocLen) ||
   7798             p_extra->nDataSize == 0)
   7799           p_extra = NULL;
   7800           continue;
   7801       }
   7802       if (p_extra->eType == VDEC_EXTRADATA_MB_ERROR_MAP)
   7803       {
   7804         if (client_extradata & OMX_FRAMEINFO_EXTRADATA)
   7805           num_conceal_MB = count_MB_in_extradata(p_extra);
   7806         if (client_extradata & VDEC_EXTRADATA_MB_ERROR_MAP)
   7807           // Map driver extradata to corresponding OMX type
   7808           p_extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataConcealMB;
   7809         else
   7810           p_extra->eType = OMX_ExtraDataMax; // Invalid type to avoid expose this extradata to OMX client
   7811         if (m_debug_concealedmb) {
   7812             DEBUG_PRINT_HIGH("Concealed MB percentage is %u", num_conceal_MB);
   7813         }
   7814       }
   7815       else if (p_extra->eType == VDEC_EXTRADATA_SEI)
   7816       {
   7817         p_sei = p_extra;
   7818 
   7819         h264_parser->parse_nal((OMX_U8*)p_sei->data, p_sei->nDataSize, NALU_TYPE_SEI);
   7820 
   7821         p_extra->eType = OMX_ExtraDataMax; // Invalid type to avoid expose this extradata to OMX client
   7822       }
   7823       else if (p_extra->eType == VDEC_EXTRADATA_VUI)
   7824       {
   7825         p_vui = p_extra;
   7826 
   7827         h264_parser->parse_nal((OMX_U8*)p_vui->data, p_vui->nDataSize, NALU_TYPE_VUI, false);
   7828 
   7829         p_extra->eType = OMX_ExtraDataMax; // Invalid type to avoid expose this extradata to OMX client
   7830       }
   7831       print_debug_extradata(p_extra);
   7832       p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + p_extra->nSize);
   7833       if ((OMX_U8*)p_extra > (p_buf_hdr->pBuffer + p_buf_hdr->nAllocLen) ||
   7834           p_extra->nDataSize == 0)
   7835         p_extra = NULL;
   7836     }
   7837     if (!(client_extradata & VDEC_EXTRADATA_MB_ERROR_MAP))
   7838     {
   7839       // Driver extradata is only exposed if MB map is requested by client,
   7840       // otherwise can be overwritten by omx extradata.
   7841       p_extra = (OMX_OTHER_EXTRADATATYPE *)
   7842                ((unsigned)(p_buf_hdr->pBuffer + p_buf_hdr->nOffset +
   7843                 p_buf_hdr->nFilledLen + 3)&(~3));
   7844       p_buf_hdr->nFlags &= ~OMX_BUFFERFLAG_EXTRADATA;
   7845     }
   7846   }
   7847 
   7848 #ifdef PROCESS_EXTRADATA_IN_OUTPUT_PORT
   7849   if (drv_ctx.decoder_format == VDEC_CODECTYPE_H264)
   7850   {
   7851     if (client_extradata & OMX_TIMEINFO_EXTRADATA)
   7852     {
   7853       if (p_vui)
   7854         h264_parser->parse_nal((OMX_U8*)p_vui->data, p_vui->nDataSize, NALU_TYPE_VUI, false);
   7855       if (p_sei)
   7856         h264_parser->parse_nal((OMX_U8*)p_sei->data, p_sei->nDataSize, NALU_TYPE_SEI);
   7857       ts_in_sei = h264_parser->process_ts_with_sei_vui(p_buf_hdr->nTimeStamp);
   7858       if (!VALID_TS(p_buf_hdr->nTimeStamp))
   7859         p_buf_hdr->nTimeStamp = ts_in_sei;
   7860     }
   7861     else if ((client_extradata & OMX_FRAMEINFO_EXTRADATA) && p_sei)
   7862       // If timeinfo is present frame info from SEI is already processed
   7863       h264_parser->parse_nal((OMX_U8*)p_sei->data, p_sei->nDataSize, NALU_TYPE_SEI);
   7864   }
   7865 #endif
   7866    if ((client_extradata & OMX_INTERLACE_EXTRADATA) && p_extra &&
   7867       ((OMX_U8*)p_extra + OMX_INTERLACE_EXTRADATA_SIZE) <
   7868        (p_buf_hdr->pBuffer + p_buf_hdr->nAllocLen))
   7869   {
   7870     p_buf_hdr->nFlags |= OMX_BUFFERFLAG_EXTRADATA;
   7871     append_interlace_extradata(p_extra,
   7872          ((struct vdec_output_frameinfo *)p_buf_hdr->pOutputPortPrivate)->interlaced_format);
   7873     p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + p_extra->nSize);
   7874   }
   7875   if (client_extradata & OMX_FRAMEINFO_EXTRADATA && p_extra &&
   7876       ((OMX_U8*)p_extra + OMX_FRAMEINFO_EXTRADATA_SIZE) <
   7877        (p_buf_hdr->pBuffer + p_buf_hdr->nAllocLen))
   7878   {
   7879     p_buf_hdr->nFlags |= OMX_BUFFERFLAG_EXTRADATA;
   7880     /* vui extra data (frame_rate) information */
   7881     if (h264_parser)
   7882         h264_parser->get_frame_rate(&frame_rate);
   7883     append_frame_info_extradata(p_extra, num_conceal_MB,
   7884         ((struct vdec_output_frameinfo *)p_buf_hdr->pOutputPortPrivate)->pic_type,
   7885         p_buf_hdr->nTimeStamp, frame_rate, NULL);
   7886     p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + p_extra->nSize);
   7887   }
   7888   if ((client_extradata & OMX_PORTDEF_EXTRADATA) &&
   7889        p_extra != NULL &&
   7890       ((OMX_U8*)p_extra + OMX_PORTDEF_EXTRADATA_SIZE) <
   7891        (p_buf_hdr->pBuffer + p_buf_hdr->nAllocLen))
   7892   {
   7893     p_buf_hdr->nFlags |= OMX_BUFFERFLAG_EXTRADATA;
   7894     append_portdef_extradata(p_extra);
   7895     p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + p_extra->nSize);
   7896   }
   7897   if (p_buf_hdr->nFlags & OMX_BUFFERFLAG_EXTRADATA)
   7898     if (p_extra &&
   7899       ((OMX_U8*)p_extra + OMX_FRAMEINFO_EXTRADATA_SIZE) <
   7900        (p_buf_hdr->pBuffer + p_buf_hdr->nAllocLen))
   7901       append_terminator_extradata(p_extra);
   7902     else
   7903     {
   7904       DEBUG_PRINT_ERROR("ERROR: Terminator extradata cannot be added");
   7905       p_buf_hdr->nFlags &= ~OMX_BUFFERFLAG_EXTRADATA;
   7906     }
   7907 }
   7908 
   7909 OMX_ERRORTYPE omx_vdec::enable_extradata(OMX_U32 requested_extradata, bool enable)
   7910 {
   7911   OMX_ERRORTYPE ret = OMX_ErrorNone;
   7912   OMX_U32 driver_extradata = 0, extradata_size = 0;
   7913   struct vdec_ioctl_msg ioctl_msg = {NULL, NULL};
   7914   if(m_state != OMX_StateLoaded)
   7915   {
   7916      DEBUG_PRINT_ERROR("ERROR: enable extradata allowed in Loaded state only");
   7917      return OMX_ErrorIncorrectStateOperation;
   7918   }
   7919   if (requested_extradata & OMX_FRAMEINFO_EXTRADATA)
   7920     extradata_size += OMX_FRAMEINFO_EXTRADATA_SIZE;
   7921   if (requested_extradata & OMX_INTERLACE_EXTRADATA)
   7922     extradata_size += OMX_INTERLACE_EXTRADATA_SIZE;
   7923   if (requested_extradata & OMX_PORTDEF_EXTRADATA)
   7924   {
   7925     extradata_size += OMX_PORTDEF_EXTRADATA_SIZE;
   7926   }
   7927   DEBUG_PRINT_ERROR("enable_extradata: actual[%x] requested[%x] enable[%d]",
   7928     client_extradata, requested_extradata, enable);
   7929 
   7930   if (enable)
   7931     requested_extradata |= client_extradata;
   7932   else
   7933   {
   7934     requested_extradata = client_extradata & ~requested_extradata;
   7935     extradata_size *= -1;
   7936   }
   7937 
   7938   driver_extradata = requested_extradata & DRIVER_EXTRADATA_MASK;
   7939   if (requested_extradata & OMX_FRAMEINFO_EXTRADATA)
   7940     driver_extradata |= VDEC_EXTRADATA_MB_ERROR_MAP; // Required for conceal MB frame info
   7941 #ifdef PROCESS_EXTRADATA_IN_OUTPUT_PORT
   7942   if (drv_ctx.decoder_format == VDEC_CODECTYPE_H264)
   7943   {
   7944     driver_extradata |= ((requested_extradata & OMX_FRAMEINFO_EXTRADATA)?
   7945                           VDEC_EXTRADATA_SEI : 0); // Required for pan scan frame info
   7946     driver_extradata |= ((requested_extradata & OMX_TIMEINFO_EXTRADATA)?
   7947                           VDEC_EXTRADATA_VUI | VDEC_EXTRADATA_SEI : 0); //Required for time info
   7948   }
   7949 
   7950 #endif
   7951   if (driver_extradata != drv_ctx.extradata)
   7952   {
   7953     client_extradata = requested_extradata;
   7954     drv_ctx.extradata = driver_extradata;
   7955     //ioctl_msg.in = &drv_ctx.extradata;
   7956     //ioctl_msg.out = NULL;
   7957     //if (ioctl(drv_ctx.video_driver_fd, VDEC_IOCTL_SET_EXTRADATA,
   7958     //    (void*)&ioctl_msg) < 0)
   7959     //{
   7960       //  DEBUG_PRINT_ERROR("\nSet extradata failed");
   7961       //  ret = OMX_ErrorUnsupportedSetting;
   7962     //}   // else
   7963     // ret = get_buffer_req(&drv_ctx.op_buf);
   7964   }
   7965   else if ((client_extradata & ~DRIVER_EXTRADATA_MASK) != (requested_extradata & ~DRIVER_EXTRADATA_MASK))
   7966   {
   7967     client_extradata = requested_extradata;
   7968     drv_ctx.op_buf.buffer_size += extradata_size;
   7969     // align the buffer size
   7970     drv_ctx.op_buf.buffer_size = (drv_ctx.op_buf.buffer_size + drv_ctx.op_buf.alignment - 1)&(~(drv_ctx.op_buf.alignment - 1));
   7971     DEBUG_PRINT_LOW("Aligned buffer size with exreadata = %d\n", drv_ctx.op_buf.buffer_size);
   7972     if (!(client_extradata & ~DRIVER_EXTRADATA_MASK)) // If no omx extradata is required remove space for terminator
   7973       drv_ctx.op_buf.buffer_size -= sizeof(OMX_OTHER_EXTRADATATYPE);
   7974 	  ret = set_buffer_req(&drv_ctx.op_buf);
   7975   }
   7976   return ret;
   7977 }
   7978 
   7979 OMX_U32 omx_vdec::count_MB_in_extradata(OMX_OTHER_EXTRADATATYPE *extra)
   7980 {
   7981   OMX_U32 num_MB = 0, byte_count = 0, num_MB_in_frame = 0;
   7982   OMX_U8 *data_ptr = extra->data, data = 0;
   7983   while (byte_count < extra->nDataSize)
   7984   {
   7985     data = *data_ptr;
   7986     while (data)
   7987     {
   7988       num_MB += (data&0x01);
   7989       data >>= 1;
   7990     }
   7991     data_ptr++;
   7992     byte_count++;
   7993   }
   7994   num_MB_in_frame = ((drv_ctx.video_resolution.frame_width + 15) *
   7995                      (drv_ctx.video_resolution.frame_height + 15)) >> 8;
   7996   return ((num_MB_in_frame > 0)?(num_MB * 100 / num_MB_in_frame) : 0);
   7997 }
   7998 
   7999 void omx_vdec::print_debug_extradata(OMX_OTHER_EXTRADATATYPE *extra)
   8000 {
   8001   if (!m_debug_extradata)
   8002      return;
   8003 
   8004   DEBUG_PRINT_HIGH(
   8005     "============== Extra Data ==============\n"
   8006     "           Size: %u \n"
   8007     "        Version: %u \n"
   8008     "      PortIndex: %u \n"
   8009     "           Type: %x \n"
   8010     "       DataSize: %u \n",
   8011     extra->nSize, extra->nVersion.nVersion,
   8012     extra->nPortIndex, extra->eType, extra->nDataSize);
   8013 
   8014   if (extra->eType == OMX_ExtraDataInterlaceFormat)
   8015   {
   8016     OMX_STREAMINTERLACEFORMAT *intfmt = (OMX_STREAMINTERLACEFORMAT *)extra->data;
   8017     DEBUG_PRINT_HIGH(
   8018       "------ Interlace Format ------\n"
   8019       "                Size: %u \n"
   8020       "             Version: %u \n"
   8021       "           PortIndex: %u \n"
   8022       " Is Interlace Format: %u \n"
   8023       "   Interlace Formats: %u \n"
   8024       "=========== End of Interlace ===========\n",
   8025       intfmt->nSize, intfmt->nVersion.nVersion, intfmt->nPortIndex,
   8026       intfmt->bInterlaceFormat, intfmt->nInterlaceFormats);
   8027   }
   8028   else if (extra->eType == OMX_ExtraDataFrameInfo)
   8029   {
   8030     OMX_QCOM_EXTRADATA_FRAMEINFO *fminfo = (OMX_QCOM_EXTRADATA_FRAMEINFO *)extra->data;
   8031 
   8032     DEBUG_PRINT_HIGH(
   8033       "-------- Frame Format --------\n"
   8034       "             Picture Type: %u \n"
   8035       "           Interlace Type: %u \n"
   8036       " Pan Scan Total Frame Num: %u \n"
   8037       "   Concealed Macro Blocks: %u \n"
   8038       "               frame rate: %u \n"
   8039       "           Aspect Ratio X: %u \n"
   8040       "           Aspect Ratio Y: %u \n",
   8041       fminfo->ePicType,
   8042       fminfo->interlaceType,
   8043       fminfo->panScan.numWindows,
   8044       fminfo->nConcealedMacroblocks,
   8045       fminfo->nFrameRate,
   8046       fminfo->aspectRatio.aspectRatioX,
   8047       fminfo->aspectRatio.aspectRatioY);
   8048 
   8049     for (int i = 0; i < fminfo->panScan.numWindows; i++)
   8050     {
   8051       DEBUG_PRINT_HIGH(
   8052         "------------------------------\n"
   8053         "     Pan Scan Frame Num: %d \n"
   8054         "            Rectangle x: %d \n"
   8055         "            Rectangle y: %d \n"
   8056         "           Rectangle dx: %d \n"
   8057         "           Rectangle dy: %d \n",
   8058         i, fminfo->panScan.window[i].x, fminfo->panScan.window[i].y,
   8059         fminfo->panScan.window[i].dx, fminfo->panScan.window[i].dy);
   8060     }
   8061 
   8062     DEBUG_PRINT_HIGH("========= End of Frame Format ==========");
   8063   }
   8064   else if (extra->eType == OMX_ExtraDataNone)
   8065   {
   8066     DEBUG_PRINT_HIGH("========== End of Terminator ===========");
   8067   }
   8068   else
   8069   {
   8070     DEBUG_PRINT_HIGH("======= End of Driver Extradata ========");
   8071   }
   8072 }
   8073 
   8074 void omx_vdec::append_interlace_extradata(OMX_OTHER_EXTRADATATYPE *extra,
   8075                                           OMX_U32 interlaced_format_type)
   8076 {
   8077   OMX_STREAMINTERLACEFORMAT *interlace_format;
   8078   OMX_U32 mbaff = 0;
   8079   extra->nSize = OMX_INTERLACE_EXTRADATA_SIZE;
   8080   extra->nVersion.nVersion = OMX_SPEC_VERSION;
   8081   extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
   8082   extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataInterlaceFormat;
   8083   extra->nDataSize = sizeof(OMX_STREAMINTERLACEFORMAT);
   8084   interlace_format = (OMX_STREAMINTERLACEFORMAT *)extra->data;
   8085   interlace_format->nSize = sizeof(OMX_STREAMINTERLACEFORMAT);
   8086   interlace_format->nVersion.nVersion = OMX_SPEC_VERSION;
   8087   interlace_format->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
   8088   mbaff = (h264_parser)? (h264_parser->is_mbaff()): false;
   8089   if ((interlaced_format_type == VDEC_InterlaceFrameProgressive)  && !mbaff)
   8090   {
   8091     interlace_format->bInterlaceFormat = OMX_FALSE;
   8092     interlace_format->nInterlaceFormats = OMX_InterlaceFrameProgressive;
   8093     drv_ctx.interlace = VDEC_InterlaceFrameProgressive;
   8094   }
   8095   else
   8096   {
   8097     interlace_format->bInterlaceFormat = OMX_TRUE;
   8098     interlace_format->nInterlaceFormats = OMX_InterlaceInterleaveFrameTopFieldFirst;
   8099     drv_ctx.interlace = VDEC_InterlaceInterleaveFrameTopFieldFirst;
   8100   }
   8101   print_debug_extradata(extra);
   8102 }
   8103 
   8104 
   8105 void omx_vdec::append_frame_info_extradata(OMX_OTHER_EXTRADATATYPE *extra,
   8106     OMX_U32 num_conceal_mb, OMX_U32 picture_type, OMX_S64 timestamp, OMX_U32 frame_rate,
   8107 	vdec_aspectratioinfo *aspect_ratio_info)
   8108 {
   8109   OMX_QCOM_EXTRADATA_FRAMEINFO *frame_info = NULL;
   8110   extra->nSize = OMX_FRAMEINFO_EXTRADATA_SIZE;
   8111   extra->nVersion.nVersion = OMX_SPEC_VERSION;
   8112   extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
   8113   extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataFrameInfo;
   8114   extra->nDataSize = sizeof(OMX_QCOM_EXTRADATA_FRAMEINFO);
   8115   frame_info = (OMX_QCOM_EXTRADATA_FRAMEINFO *)extra->data;
   8116   switch (picture_type)
   8117   {
   8118     case PICTURE_TYPE_I:
   8119       frame_info->ePicType = OMX_VIDEO_PictureTypeI;
   8120     break;
   8121     case PICTURE_TYPE_P:
   8122       frame_info->ePicType = OMX_VIDEO_PictureTypeP;
   8123     break;
   8124     case PICTURE_TYPE_B:
   8125       frame_info->ePicType = OMX_VIDEO_PictureTypeB;
   8126     break;
   8127     default:
   8128        frame_info->ePicType = (OMX_VIDEO_PICTURETYPE)0;
   8129   }
   8130   if (drv_ctx.interlace == VDEC_InterlaceInterleaveFrameTopFieldFirst)
   8131     frame_info->interlaceType = OMX_QCOM_InterlaceInterleaveFrameTopFieldFirst;
   8132   else if (drv_ctx.interlace == VDEC_InterlaceInterleaveFrameBottomFieldFirst)
   8133     frame_info->interlaceType = OMX_QCOM_InterlaceInterleaveFrameBottomFieldFirst;
   8134   else
   8135     frame_info->interlaceType = OMX_QCOM_InterlaceFrameProgressive;
   8136   memset(&frame_info->panScan,0,sizeof(frame_info->panScan));
   8137   memset(&frame_info->aspectRatio, 0, sizeof(frame_info->aspectRatio));
   8138   if (drv_ctx.decoder_format == VDEC_CODECTYPE_H264)
   8139   {
   8140     h264_parser->fill_pan_scan_data(&frame_info->panScan, timestamp);
   8141     h264_parser->fill_aspect_ratio_info(&frame_info->aspectRatio);
   8142   }
   8143   frame_info->nConcealedMacroblocks = num_conceal_mb;
   8144   frame_info->nFrameRate = frame_rate;
   8145   print_debug_extradata(extra);
   8146 }
   8147 
   8148 void omx_vdec::append_portdef_extradata(OMX_OTHER_EXTRADATATYPE *extra)
   8149 {
   8150   OMX_PARAM_PORTDEFINITIONTYPE *portDefn = NULL;
   8151   extra->nSize = OMX_PORTDEF_EXTRADATA_SIZE;
   8152   extra->nVersion.nVersion = OMX_SPEC_VERSION;
   8153   extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
   8154   extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataPortDef;
   8155   extra->nDataSize = sizeof(OMX_PARAM_PORTDEFINITIONTYPE);
   8156   portDefn = (OMX_PARAM_PORTDEFINITIONTYPE *)extra->data;
   8157   *portDefn = m_port_def;
   8158   DEBUG_PRINT_LOW("append_portdef_extradata height = %u width = %u stride = %u"
   8159      "sliceheight = %u \n",portDefn->format.video.nFrameHeight,
   8160      portDefn->format.video.nFrameWidth,
   8161      portDefn->format.video.nStride,
   8162      portDefn->format.video.nSliceHeight);
   8163 }
   8164 
   8165 void omx_vdec::append_terminator_extradata(OMX_OTHER_EXTRADATATYPE *extra)
   8166 {
   8167   extra->nSize = sizeof(OMX_OTHER_EXTRADATATYPE);
   8168   extra->nVersion.nVersion = OMX_SPEC_VERSION;
   8169   extra->eType = OMX_ExtraDataNone;
   8170   extra->nDataSize = 0;
   8171   extra->data[0] = 0;
   8172 
   8173   print_debug_extradata(extra);
   8174 }
   8175 
   8176 OMX_ERRORTYPE  omx_vdec::allocate_desc_buffer(OMX_U32 index)
   8177 {
   8178   OMX_ERRORTYPE eRet = OMX_ErrorNone;
   8179   if (index >= drv_ctx.ip_buf.actualcount)
   8180   {
   8181     DEBUG_PRINT_ERROR("\nERROR:Desc Buffer Index not found");
   8182     return OMX_ErrorInsufficientResources;
   8183   }
   8184   if (m_desc_buffer_ptr == NULL)
   8185   {
   8186     m_desc_buffer_ptr = (desc_buffer_hdr*) \
   8187                      calloc( (sizeof(desc_buffer_hdr)),
   8188                      drv_ctx.ip_buf.actualcount);
   8189     if (m_desc_buffer_ptr == NULL)
   8190     {
   8191       DEBUG_PRINT_ERROR("\n m_desc_buffer_ptr Allocation failed ");
   8192       return OMX_ErrorInsufficientResources;
   8193     }
   8194   }
   8195 
   8196   m_desc_buffer_ptr[index].buf_addr = (unsigned char *)malloc (DESC_BUFFER_SIZE * sizeof(OMX_U8));
   8197   if (m_desc_buffer_ptr[index].buf_addr == NULL)
   8198   {
   8199     DEBUG_PRINT_ERROR("\ndesc buffer Allocation failed ");
   8200     return OMX_ErrorInsufficientResources;
   8201   }
   8202 
   8203   return eRet;
   8204 }
   8205 
   8206 void omx_vdec::insert_demux_addr_offset(OMX_U32 address_offset)
   8207 {
   8208   DEBUG_PRINT_LOW("Inserting address offset (%d) at idx (%d)", address_offset,m_demux_entries);
   8209   if (m_demux_entries < 8192)
   8210   {
   8211     m_demux_offsets[m_demux_entries++] = address_offset;
   8212   }
   8213   return;
   8214 }
   8215 
   8216 void omx_vdec::extract_demux_addr_offsets(OMX_BUFFERHEADERTYPE *buf_hdr)
   8217 {
   8218   OMX_U32 bytes_to_parse = buf_hdr->nFilledLen;
   8219   OMX_U8 *buf = buf_hdr->pBuffer + buf_hdr->nOffset;
   8220   OMX_U32 index = 0;
   8221 
   8222   m_demux_entries = 0;
   8223 
   8224   while (index < bytes_to_parse)
   8225   {
   8226     if ( ((buf[index] == 0x00) && (buf[index+1] == 0x00) &&
   8227           (buf[index+2] == 0x00) && (buf[index+3] == 0x01)) ||
   8228          ((buf[index] == 0x00) && (buf[index+1] == 0x00) &&
   8229           (buf[index+2] == 0x01)) )
   8230     {
   8231       //Found start code, insert address offset
   8232       insert_demux_addr_offset(index);
   8233       if (buf[index+2] == 0x01) // 3 byte start code
   8234         index += 3;
   8235       else                      //4 byte start code
   8236         index += 4;
   8237     }
   8238     else
   8239       index++;
   8240   }
   8241   DEBUG_PRINT_LOW("Extracted (%d) demux entry offsets",m_demux_entries);
   8242   return;
   8243 }
   8244 
   8245 OMX_ERRORTYPE omx_vdec::handle_demux_data(OMX_BUFFERHEADERTYPE *p_buf_hdr)
   8246 {
   8247   //fix this, handle 3 byte start code, vc1 terminator entry
   8248   OMX_U8 *p_demux_data = NULL;
   8249   OMX_U32 desc_data = 0;
   8250   OMX_U32 start_addr = 0;
   8251   OMX_U32 nal_size = 0;
   8252   OMX_U32 suffix_byte = 0;
   8253   OMX_U32 demux_index = 0;
   8254   OMX_U32 buffer_index = 0;
   8255 
   8256   if (m_desc_buffer_ptr == NULL)
   8257   {
   8258     DEBUG_PRINT_ERROR("m_desc_buffer_ptr is NULL. Cannot append demux entries.");
   8259     return OMX_ErrorBadParameter;
   8260   }
   8261 
   8262   buffer_index = p_buf_hdr - ((OMX_BUFFERHEADERTYPE *)m_inp_mem_ptr);
   8263   if (buffer_index > drv_ctx.ip_buf.actualcount)
   8264   {
   8265     DEBUG_PRINT_ERROR("handle_demux_data:Buffer index is incorrect (%d)", buffer_index);
   8266     return OMX_ErrorBadParameter;
   8267   }
   8268 
   8269   p_demux_data = (OMX_U8 *) m_desc_buffer_ptr[buffer_index].buf_addr;
   8270 
   8271   if ( ((OMX_U8*)p_demux_data == NULL) ||
   8272       ((m_demux_entries * 16) + 1) > DESC_BUFFER_SIZE)
   8273   {
   8274     DEBUG_PRINT_ERROR("Insufficient buffer. Cannot append demux entries.");
   8275     return OMX_ErrorBadParameter;
   8276   }
   8277   else
   8278   {
   8279     for (; demux_index < m_demux_entries; demux_index++)
   8280     {
   8281       desc_data = 0;
   8282       start_addr = m_demux_offsets[demux_index];
   8283       if (p_buf_hdr->pBuffer[m_demux_offsets[demux_index] + 2] == 0x01)
   8284       {
   8285         suffix_byte = p_buf_hdr->pBuffer[m_demux_offsets[demux_index] + 3];
   8286       }
   8287       else
   8288       {
   8289         suffix_byte = p_buf_hdr->pBuffer[m_demux_offsets[demux_index] + 4];
   8290       }
   8291       if (demux_index < (m_demux_entries - 1))
   8292       {
   8293         nal_size = m_demux_offsets[demux_index + 1] - m_demux_offsets[demux_index] - 2;
   8294       }
   8295       else
   8296       {
   8297         nal_size = p_buf_hdr->nFilledLen - m_demux_offsets[demux_index] - 2;
   8298       }
   8299       DEBUG_PRINT_LOW("Start_addr(%p), suffix_byte(0x%x),nal_size(%d),demux_index(%d)",
   8300                         start_addr,
   8301                         suffix_byte,
   8302                         nal_size,
   8303                         demux_index);
   8304       desc_data = (start_addr >> 3) << 1;
   8305       desc_data |= (start_addr & 7) << 21;
   8306       desc_data |= suffix_byte << 24;
   8307 
   8308       memcpy(p_demux_data, &desc_data, sizeof(OMX_U32));
   8309       memcpy(p_demux_data + 4, &nal_size, sizeof(OMX_U32));
   8310       memset(p_demux_data + 8, 0, sizeof(OMX_U32));
   8311       memset(p_demux_data + 12, 0, sizeof(OMX_U32));
   8312 
   8313       p_demux_data += 16;
   8314     }
   8315     if (codec_type_parse == CODEC_TYPE_VC1)
   8316     {
   8317       DEBUG_PRINT_LOW("VC1 terminator entry");
   8318       desc_data = 0;
   8319       desc_data = 0x82 << 24;
   8320       memcpy(p_demux_data, &desc_data, sizeof(OMX_U32));
   8321       memset(p_demux_data + 4, 0, sizeof(OMX_U32));
   8322       memset(p_demux_data + 8, 0, sizeof(OMX_U32));
   8323       memset(p_demux_data + 12, 0, sizeof(OMX_U32));
   8324       p_demux_data += 16;
   8325       m_demux_entries++;
   8326     }
   8327     //Add zero word to indicate end of descriptors
   8328     memset(p_demux_data, 0, sizeof(OMX_U32));
   8329 
   8330     m_desc_buffer_ptr[buffer_index].desc_data_size = (m_demux_entries * 16) + sizeof(OMX_U32);
   8331     DEBUG_PRINT_LOW("desc table data size=%d", m_desc_buffer_ptr[buffer_index].desc_data_size);
   8332   }
   8333   memset(m_demux_offsets, 0, ( sizeof(OMX_U32) * 8192) );
   8334   m_demux_entries = 0;
   8335   DEBUG_PRINT_LOW("Demux table complete!");
   8336   return OMX_ErrorNone;
   8337 }
   8338 
   8339 #if 0
   8340 OMX_ERRORTYPE omx_vdec::createDivxDrmContext( OMX_PTR drmHandle )
   8341 {
   8342      OMX_ERRORTYPE err = OMX_ErrorNone;
   8343      if( drmHandle == NULL ) {
   8344         DEBUG_PRINT_HIGH("\n This clip is not DRM encrypted");
   8345         iDivXDrmDecrypt = NULL;
   8346         return err;
   8347      }
   8348 
   8349      iDivXDrmDecrypt = DivXDrmDecrypt::Create( drmHandle );
   8350      if (iDivXDrmDecrypt) {
   8351           DEBUG_PRINT_LOW("\nCreated DIVX DRM, now calling Init");
   8352           OMX_ERRORTYPE err = iDivXDrmDecrypt->Init();
   8353           if(err!=OMX_ErrorNone) {
   8354             DEBUG_PRINT_ERROR("\nERROR:iDivXDrmDecrypt->Init %d", err);
   8355             delete iDivXDrmDecrypt;
   8356             iDivXDrmDecrypt = NULL;
   8357           }
   8358      }
   8359      else {
   8360           DEBUG_PRINT_ERROR("\nUnable to Create DIVX DRM");
   8361           return OMX_ErrorUndefined;
   8362      }
   8363      return err;
   8364 }
   8365 #endif
   8366 
   8367