1 /*--------------------------------------------------------------------------
2 Copyright (c) 2010 - 2017, The Linux Foundation. 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 The Linux Foundation 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 This module contains the implementation of the OpenMAX core & component.
34
35 *//*========================================================================*/
36
37 //////////////////////////////////////////////////////////////////////////////
38 // Include Files
39 //////////////////////////////////////////////////////////////////////////////
40
41 #define __STDC_FORMAT_MACROS
42 #include <inttypes.h>
43
44 #include <string.h>
45 #include <pthread.h>
46 #include <sys/prctl.h>
47 #include <stdlib.h>
48 #include <unistd.h>
49 #include <errno.h>
50 #include "omx_vdec.h"
51 #include <fcntl.h>
52 #include <limits.h>
53 #include <stdlib.h>
54 #include <media/hardware/HardwareAPI.h>
55 #include <sys/eventfd.h>
56 #include <nativebase/nativebase.h>
57
58 #if !defined(_ANDROID_) || defined(SYS_IOCTL)
59 #include <sys/ioctl.h>
60 #include <sys/mman.h>
61 #endif
62
63 #ifdef _ANDROID_
64 #include <cutils/properties.h>
65 #undef USE_EGL_IMAGE_GPU
66
67 #ifdef _QUERY_DISP_RES_
68 #include "display_config.h"
69 #endif
70 #endif
71
72 #ifdef _USE_GLIB_
73 #include <glib.h>
74 #define strlcpy g_strlcpy
75 #endif
76
77 #include <qdMetaData.h>
78 #include <gralloc_priv.h>
79
80 #ifdef ANDROID_JELLYBEAN_MR2
81 #include "QComOMXMetadata.h"
82 #endif
83
84 #ifdef USE_EGL_IMAGE_GPU
85 #include <EGL/egl.h>
86 #include <EGL/eglQCOM.h>
87 #define EGL_BUFFER_HANDLE 0x4F00
88 #define EGL_BUFFER_OFFSET 0x4F01
89 #endif
90
91 #define BUFFER_LOG_LOC "/data/vendor/media"
92
93 #ifdef OUTPUT_EXTRADATA_LOG
94 FILE *outputExtradataFile;
95 char output_extradata_filename [] = "/data/vendor/media/extradata";
96 #endif
97
98 #define DEFAULT_FPS 30
99 #define MAX_SUPPORTED_FPS 240
100 #define DEFAULT_WIDTH_ALIGNMENT 128
101 #define DEFAULT_HEIGHT_ALIGNMENT 32
102
103 #define VC1_SP_MP_START_CODE 0xC5000000
104 #define VC1_SP_MP_START_CODE_MASK 0xFF000000
105 #define VC1_AP_SEQ_START_CODE 0x0F010000
106 #define VC1_STRUCT_C_PROFILE_MASK 0xF0
107 #define VC1_STRUCT_B_LEVEL_MASK 0xE0000000
108 #define VC1_SIMPLE_PROFILE 0
109 #define VC1_MAIN_PROFILE 1
110 #define VC1_ADVANCE_PROFILE 3
111 #define VC1_SIMPLE_PROFILE_LOW_LEVEL 0
112 #define VC1_SIMPLE_PROFILE_MED_LEVEL 2
113 #define VC1_STRUCT_C_LEN 4
114 #define VC1_STRUCT_C_POS 8
115 #define VC1_STRUCT_A_POS 12
116 #define VC1_STRUCT_B_POS 24
117 #define VC1_SEQ_LAYER_SIZE 36
118 #define POLL_TIMEOUT 0x7fffffff
119
120 #define MEM_DEVICE "/dev/ion"
121
122 #ifdef _ANDROID_
123 extern "C" {
124 #include<utils/Log.h>
125 }
126 #endif//_ANDROID_
127
128 #define SZ_4K 0x1000
129 #define SZ_1M 0x100000
130
131 #define Log2(number, power) { OMX_U32 temp = number; power = 0; while( (0 == (temp & 0x1)) && power < 16) { temp >>=0x1; power++; } }
132 #define Q16ToFraction(q,num,den) { OMX_U32 power; Log2(q,power); num = q >> power; den = 0x1 << (16 - power); }
133 #define EXTRADATA_IDX(__num_planes) ((__num_planes) ? (__num_planes) - 1 : 0)
134 #define ALIGN(x, to_align) ((((unsigned) x) + (to_align - 1)) & ~(to_align - 1))
135
136 #define DEFAULT_EXTRADATA (OMX_INTERLACE_EXTRADATA | OMX_FRAMEPACK_EXTRADATA | OMX_OUTPUTCROP_EXTRADATA \
137 | OMX_DISPLAY_INFO_EXTRADATA | OMX_HDR_COLOR_INFO_EXTRADATA)
138 #define DEFAULT_CONCEAL_COLOR "32784" //0x8010, black by default
139
140 #ifndef ION_FLAG_CP_BITSTREAM
141 #define ION_FLAG_CP_BITSTREAM 0
142 #endif
143
144 #ifndef ION_FLAG_CP_PIXEL
145 #define ION_FLAG_CP_PIXEL 0
146 #endif
147
148 #ifdef MASTER_SIDE_CP
149 #define MEM_HEAP_ID ION_SECURE_HEAP_ID
150 #define SECURE_ALIGN SZ_4K
151 #define SECURE_FLAGS_INPUT_BUFFER (ION_SECURE | ION_FLAG_CP_BITSTREAM)
152 #define SECURE_FLAGS_OUTPUT_BUFFER (ION_SECURE | ION_FLAG_CP_PIXEL)
153 #else //SLAVE_SIDE_CP
154 #define MEM_HEAP_ID ION_CP_MM_HEAP_ID
155 #define SECURE_ALIGN SZ_1M
156 #define SECURE_FLAGS_INPUT_BUFFER ION_SECURE
157 #define SECURE_FLAGS_OUTPUT_BUFFER ION_SECURE
158 #endif
159
160 #define LUMINANCE_DIV_FACTOR 10000.0
161
162 #define MIN(x,y) (((x) < (y)) ? (x) : (y))
163 #define MAX(x,y) (((x) > (y)) ? (x) : (y))
164
165 static OMX_U32 maxSmoothStreamingWidth = 1920;
166 static OMX_U32 maxSmoothStreamingHeight = 1088;
167
async_message_thread(void * input)168 void* async_message_thread (void *input)
169 {
170 OMX_BUFFERHEADERTYPE *buffer;
171 struct v4l2_plane plane[VIDEO_MAX_PLANES];
172 struct pollfd pfds[2];
173 struct v4l2_buffer v4l2_buf;
174 memset((void *)&v4l2_buf,0,sizeof(v4l2_buf));
175 struct v4l2_event dqevent;
176 omx_vdec *omx = reinterpret_cast<omx_vdec*>(input);
177 pfds[0].events = POLLIN | POLLRDNORM | POLLOUT | POLLWRNORM | POLLRDBAND | POLLPRI;
178 pfds[1].events = POLLIN | POLLERR;
179 pfds[0].fd = omx->drv_ctx.video_driver_fd;
180 pfds[1].fd = omx->m_poll_efd;
181 int error_code = 0,rc=0,bytes_read = 0,bytes_written = 0;
182 DEBUG_PRINT_HIGH("omx_vdec: Async thread start");
183 prctl(PR_SET_NAME, (unsigned long)"VideoDecCallBackThread", 0, 0, 0);
184 while (!omx->async_thread_force_stop) {
185 rc = poll(pfds, 2, POLL_TIMEOUT);
186 if (!rc) {
187 DEBUG_PRINT_ERROR("Poll timedout");
188 break;
189 } else if (rc < 0 && errno != EINTR && errno != EAGAIN) {
190 DEBUG_PRINT_ERROR("Error while polling: %d, errno = %d", rc, errno);
191 break;
192 }
193 if ((pfds[1].revents & POLLIN) || (pfds[1].revents & POLLERR)) {
194 DEBUG_PRINT_HIGH("async_message_thread interrupted to be exited");
195 break;
196 }
197 if ((pfds[0].revents & POLLIN) || (pfds[0].revents & POLLRDNORM)) {
198 struct vdec_msginfo vdec_msg;
199 memset(&vdec_msg, 0, sizeof(vdec_msg));
200 v4l2_buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
201 v4l2_buf.memory = V4L2_MEMORY_USERPTR;
202 v4l2_buf.length = omx->drv_ctx.num_planes;
203 v4l2_buf.m.planes = plane;
204 while (!ioctl(pfds[0].fd, VIDIOC_DQBUF, &v4l2_buf)) {
205 vdec_msg.msgcode=VDEC_MSG_RESP_OUTPUT_BUFFER_DONE;
206 vdec_msg.status_code=VDEC_S_SUCCESS;
207 vdec_msg.msgdata.output_frame.client_data=(void*)&v4l2_buf;
208 vdec_msg.msgdata.output_frame.len=plane[0].bytesused;
209 vdec_msg.msgdata.output_frame.bufferaddr=(void*)plane[0].m.userptr;
210 vdec_msg.msgdata.output_frame.time_stamp= ((uint64_t)v4l2_buf.timestamp.tv_sec * (uint64_t)1000000) +
211 (uint64_t)v4l2_buf.timestamp.tv_usec;
212
213 if (omx->async_message_process(input,&vdec_msg) < 0) {
214 DEBUG_PRINT_HIGH("async_message_thread Exited");
215 break;
216 }
217 }
218 }
219 if ((pfds[0].revents & POLLOUT) || (pfds[0].revents & POLLWRNORM)) {
220 struct vdec_msginfo vdec_msg;
221 v4l2_buf.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
222 v4l2_buf.memory = V4L2_MEMORY_USERPTR;
223 v4l2_buf.length = 1;
224 v4l2_buf.m.planes = plane;
225 while (!ioctl(pfds[0].fd, VIDIOC_DQBUF, &v4l2_buf)) {
226 vdec_msg.msgcode=VDEC_MSG_RESP_INPUT_BUFFER_DONE;
227 vdec_msg.status_code=VDEC_S_SUCCESS;
228 vdec_msg.msgdata.input_frame_clientdata=(void*)&v4l2_buf;
229 if (omx->async_message_process(input,&vdec_msg) < 0) {
230 DEBUG_PRINT_HIGH("async_message_thread Exited");
231 break;
232 }
233 }
234 }
235 if (pfds[0].revents & POLLPRI) {
236 rc = ioctl(pfds[0].fd, VIDIOC_DQEVENT, &dqevent);
237 if (dqevent.type == V4L2_EVENT_MSM_VIDC_PORT_SETTINGS_CHANGED_INSUFFICIENT ) {
238 struct vdec_msginfo vdec_msg;
239 unsigned int *ptr = (unsigned int *)(void *)dqevent.u.data;
240
241 vdec_msg.msgcode=VDEC_MSG_EVT_CONFIG_CHANGED;
242 vdec_msg.status_code=VDEC_S_SUCCESS;
243 vdec_msg.msgdata.output_frame.picsize.frame_height = ptr[0];
244 vdec_msg.msgdata.output_frame.picsize.frame_width = ptr[1];
245 DEBUG_PRINT_HIGH("VIDC Port Reconfig received insufficient");
246 if(ptr[2] & V4L2_EVENT_BITDEPTH_FLAG) {
247 omx->dpb_bit_depth = ptr[3];
248 DEBUG_PRINT_HIGH("VIDC Port Reconfig Bitdepth change - %d", ptr[3]);
249 }
250 if(ptr[2] & V4L2_EVENT_PICSTRUCT_FLAG) {
251 omx->m_progressive = ptr[4];
252 DEBUG_PRINT_HIGH("VIDC Port Reconfig PicStruct change - %d", ptr[4]);
253 }
254 if(ptr[2] & V4L2_EVENT_COLOUR_SPACE_FLAG) {
255 if (ptr[5] == MSM_VIDC_BT2020) {
256 omx->m_color_space = omx_vdec::BT2020;
257 } else {
258 omx->m_color_space = omx_vdec::EXCEPT_BT2020;
259 }
260 DEBUG_PRINT_HIGH("VIDC Port Reconfig ColorSpace change - %d", omx->m_color_space);
261 }
262 if (omx->async_message_process(input,&vdec_msg) < 0) {
263 DEBUG_PRINT_HIGH("async_message_thread Exited");
264 break;
265 }
266 } else if (dqevent.type == V4L2_EVENT_MSM_VIDC_FLUSH_DONE) {
267 struct vdec_msginfo vdec_msg;
268 uint32_t flush_type = *(uint32_t *)dqevent.u.data;
269 // Old driver doesn't send flushType information.
270 // To make this backward compatible fallback to old approach
271 // if the flush_type is not present.
272 vdec_msg.status_code=VDEC_S_SUCCESS;
273 if (!flush_type || (flush_type & V4L2_QCOM_CMD_FLUSH_OUTPUT)) {
274 vdec_msg.msgcode=VDEC_MSG_RESP_FLUSH_INPUT_DONE;
275 DEBUG_PRINT_HIGH("VIDC Input Flush Done Recieved");
276 if (omx->async_message_process(input,&vdec_msg) < 0) {
277 DEBUG_PRINT_HIGH("async_message_thread Exited");
278 break;
279 }
280 }
281
282 if (!flush_type || (flush_type & V4L2_QCOM_CMD_FLUSH_CAPTURE)) {
283 vdec_msg.msgcode=VDEC_MSG_RESP_FLUSH_OUTPUT_DONE;
284 DEBUG_PRINT_HIGH("VIDC Output Flush Done Recieved");
285 if (omx->async_message_process(input,&vdec_msg) < 0) {
286 DEBUG_PRINT_HIGH("async_message_thread Exited");
287 break;
288 }
289 }
290 } else if (dqevent.type == V4L2_EVENT_MSM_VIDC_HW_OVERLOAD) {
291 struct vdec_msginfo vdec_msg;
292 vdec_msg.msgcode=VDEC_MSG_EVT_HW_OVERLOAD;
293 vdec_msg.status_code=VDEC_S_SUCCESS;
294 DEBUG_PRINT_ERROR("HW Overload received");
295 if (omx->async_message_process(input,&vdec_msg) < 0) {
296 DEBUG_PRINT_HIGH("async_message_thread Exited");
297 break;
298 }
299 } else if (dqevent.type == V4L2_EVENT_MSM_VIDC_HW_UNSUPPORTED) {
300 struct vdec_msginfo vdec_msg;
301 vdec_msg.msgcode=VDEC_MSG_EVT_HW_UNSUPPORTED;
302 vdec_msg.status_code=VDEC_S_SUCCESS;
303 DEBUG_PRINT_ERROR("HW Unsupported received");
304 if (omx->async_message_process(input,&vdec_msg) < 0) {
305 DEBUG_PRINT_HIGH("async_message_thread Exited");
306 break;
307 }
308 } else if (dqevent.type == V4L2_EVENT_MSM_VIDC_SYS_ERROR) {
309 struct vdec_msginfo vdec_msg;
310 vdec_msg.msgcode = VDEC_MSG_EVT_HW_ERROR;
311 vdec_msg.status_code = VDEC_S_SUCCESS;
312 DEBUG_PRINT_HIGH("SYS Error Recieved");
313 if (omx->async_message_process(input,&vdec_msg) < 0) {
314 DEBUG_PRINT_HIGH("async_message_thread Exited");
315 break;
316 }
317 } else if (dqevent.type == V4L2_EVENT_MSM_VIDC_RELEASE_BUFFER_REFERENCE) {
318 unsigned int *ptr = (unsigned int *)(void *)dqevent.u.data;
319
320 DEBUG_PRINT_LOW("REFERENCE RELEASE EVENT RECVD fd = %d offset = %d", ptr[0], ptr[1]);
321 } else if (dqevent.type == V4L2_EVENT_MSM_VIDC_RELEASE_UNQUEUED_BUFFER) {
322 unsigned int *ptr = (unsigned int *)(void *)dqevent.u.data;
323 struct vdec_msginfo vdec_msg;
324
325 DEBUG_PRINT_LOW("Release unqueued buffer event recvd fd = %d offset = %d", ptr[0], ptr[1]);
326
327 v4l2_buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
328 v4l2_buf.memory = V4L2_MEMORY_USERPTR;
329 v4l2_buf.length = omx->drv_ctx.num_planes;
330 v4l2_buf.m.planes = plane;
331 v4l2_buf.index = ptr[5];
332 v4l2_buf.flags = 0;
333
334 vdec_msg.msgcode = VDEC_MSG_RESP_OUTPUT_BUFFER_DONE;
335 vdec_msg.status_code = VDEC_S_SUCCESS;
336 vdec_msg.msgdata.output_frame.client_data = (void*)&v4l2_buf;
337 vdec_msg.msgdata.output_frame.len = 0;
338 vdec_msg.msgdata.output_frame.bufferaddr = (void*)(intptr_t)ptr[2];
339 vdec_msg.msgdata.output_frame.time_stamp = ((uint64_t)ptr[3] * (uint64_t)1000000) +
340 (uint64_t)ptr[4];
341 if (omx->async_message_process(input,&vdec_msg) < 0) {
342 DEBUG_PRINT_HIGH("async_message_thread Exitedn");
343 break;
344 }
345 } else {
346 DEBUG_PRINT_HIGH("VIDC Some Event recieved");
347 continue;
348 }
349 }
350 }
351 DEBUG_PRINT_HIGH("omx_vdec: Async thread stop");
352 return NULL;
353 }
354
message_thread_dec(void * input)355 void* message_thread_dec(void *input)
356 {
357 omx_vdec* omx = reinterpret_cast<omx_vdec*>(input);
358 unsigned char id;
359 int n;
360
361 fd_set readFds;
362 int res = 0;
363 struct timeval tv;
364
365 DEBUG_PRINT_HIGH("omx_vdec: message thread start");
366 prctl(PR_SET_NAME, (unsigned long)"VideoDecMsgThread", 0, 0, 0);
367 while (!omx->message_thread_stop) {
368
369 tv.tv_sec = 2;
370 tv.tv_usec = 0;
371
372 FD_ZERO(&readFds);
373 FD_SET(omx->m_pipe_in, &readFds);
374
375 res = select(omx->m_pipe_in + 1, &readFds, NULL, NULL, &tv);
376 if (res < 0) {
377 DEBUG_PRINT_ERROR("select() ERROR: %s", strerror(errno));
378 continue;
379 } else if (res == 0 /*timeout*/ || omx->message_thread_stop) {
380 continue;
381 }
382
383 n = read(omx->m_pipe_in, &id, 1);
384
385 if (0 == n) {
386 break;
387 }
388
389 if (1 == n) {
390 omx->process_event_cb(omx, id);
391 }
392
393 if ((n < 0) && (errno != EINTR)) {
394 DEBUG_PRINT_LOW("ERROR: read from pipe failed, ret %d errno %d", n, errno);
395 break;
396 }
397 }
398 DEBUG_PRINT_HIGH("omx_vdec: message thread stop");
399 return 0;
400 }
401
post_message(omx_vdec * omx,unsigned char id)402 void post_message(omx_vdec *omx, unsigned char id)
403 {
404 int ret_value;
405 DEBUG_PRINT_LOW("omx_vdec: post_message %d pipe out%d", id,omx->m_pipe_out);
406 ret_value = write(omx->m_pipe_out, &id, 1);
407 if (ret_value <= 0) {
408 DEBUG_PRINT_ERROR("post_message to pipe failed : %s", strerror(errno));
409 } else {
410 DEBUG_PRINT_LOW("post_message to pipe done %d",ret_value);
411 }
412 }
413
414 // omx_cmd_queue destructor
~omx_cmd_queue()415 omx_vdec::omx_cmd_queue::~omx_cmd_queue()
416 {
417 // Nothing to do
418 }
419
420 // omx cmd queue constructor
omx_cmd_queue()421 omx_vdec::omx_cmd_queue::omx_cmd_queue(): m_read(0),m_write(0),m_size(0)
422 {
423 memset(m_q,0,sizeof(omx_event)*OMX_CORE_CONTROL_CMDQ_SIZE);
424 }
425
426 // omx cmd queue insert
insert_entry(unsigned long p1,unsigned long p2,unsigned long id)427 bool omx_vdec::omx_cmd_queue::insert_entry(unsigned long p1, unsigned long p2, unsigned long id)
428 {
429 bool ret = true;
430 if (m_size < OMX_CORE_CONTROL_CMDQ_SIZE) {
431 m_q[m_write].id = id;
432 m_q[m_write].param1 = p1;
433 m_q[m_write].param2 = p2;
434 m_write++;
435 m_size ++;
436 if (m_write >= OMX_CORE_CONTROL_CMDQ_SIZE) {
437 m_write = 0;
438 }
439 } else {
440 ret = false;
441 DEBUG_PRINT_ERROR("ERROR: %s()::Command Queue Full", __func__);
442 }
443 return ret;
444 }
445
446 // omx cmd queue pop
pop_entry(unsigned long * p1,unsigned long * p2,unsigned long * id)447 bool omx_vdec::omx_cmd_queue::pop_entry(unsigned long *p1, unsigned long *p2, unsigned long *id)
448 {
449 bool ret = true;
450 if (m_size > 0) {
451 *id = m_q[m_read].id;
452 *p1 = m_q[m_read].param1;
453 *p2 = m_q[m_read].param2;
454 // Move the read pointer ahead
455 ++m_read;
456 --m_size;
457 if (m_read >= OMX_CORE_CONTROL_CMDQ_SIZE) {
458 m_read = 0;
459 }
460 } else {
461 ret = false;
462 }
463 return ret;
464 }
465
466 // Retrieve the first mesg type in the queue
get_q_msg_type()467 unsigned omx_vdec::omx_cmd_queue::get_q_msg_type()
468 {
469 return m_q[m_read].id;
470 }
471
472 #ifdef _ANDROID_
ts_arr_list()473 omx_vdec::ts_arr_list::ts_arr_list()
474 {
475 //initialize timestamps array
476 memset(m_ts_arr_list, 0, ( sizeof(ts_entry) * MAX_NUM_INPUT_OUTPUT_BUFFERS) );
477 }
~ts_arr_list()478 omx_vdec::ts_arr_list::~ts_arr_list()
479 {
480 //free m_ts_arr_list?
481 }
482
insert_ts(OMX_TICKS ts)483 bool omx_vdec::ts_arr_list::insert_ts(OMX_TICKS ts)
484 {
485 bool ret = true;
486 bool duplicate_ts = false;
487 int idx = 0;
488
489 //insert at the first available empty location
490 for ( ; idx < MAX_NUM_INPUT_OUTPUT_BUFFERS; idx++) {
491 if (!m_ts_arr_list[idx].valid) {
492 //found invalid or empty entry, save timestamp
493 m_ts_arr_list[idx].valid = true;
494 m_ts_arr_list[idx].timestamp = ts;
495 DEBUG_PRINT_LOW("Insert_ts(): Inserting TIMESTAMP (%lld) at idx (%d)",
496 ts, idx);
497 break;
498 }
499 }
500
501 if (idx == MAX_NUM_INPUT_OUTPUT_BUFFERS) {
502 DEBUG_PRINT_LOW("Timestamp array list is FULL. Unsuccessful insert");
503 ret = false;
504 }
505 return ret;
506 }
507
pop_min_ts(OMX_TICKS & ts)508 bool omx_vdec::ts_arr_list::pop_min_ts(OMX_TICKS &ts)
509 {
510 bool ret = true;
511 int min_idx = -1;
512 OMX_TICKS min_ts = 0;
513 int idx = 0;
514
515 for ( ; idx < MAX_NUM_INPUT_OUTPUT_BUFFERS; idx++) {
516
517 if (m_ts_arr_list[idx].valid) {
518 //found valid entry, save index
519 if (min_idx < 0) {
520 //first valid entry
521 min_ts = m_ts_arr_list[idx].timestamp;
522 min_idx = idx;
523 } else if (m_ts_arr_list[idx].timestamp < min_ts) {
524 min_ts = m_ts_arr_list[idx].timestamp;
525 min_idx = idx;
526 }
527 }
528
529 }
530
531 if (min_idx < 0) {
532 //no valid entries found
533 DEBUG_PRINT_LOW("Timestamp array list is empty. Unsuccessful pop");
534 ts = 0;
535 ret = false;
536 } else {
537 ts = m_ts_arr_list[min_idx].timestamp;
538 m_ts_arr_list[min_idx].valid = false;
539 DEBUG_PRINT_LOW("Pop_min_ts:Timestamp (%lld), index(%d)",
540 ts, min_idx);
541 }
542
543 return ret;
544
545 }
546
547
reset_ts_list()548 bool omx_vdec::ts_arr_list::reset_ts_list()
549 {
550 bool ret = true;
551 int idx = 0;
552
553 DEBUG_PRINT_LOW("reset_ts_list(): Resetting timestamp array list");
554 for ( ; idx < MAX_NUM_INPUT_OUTPUT_BUFFERS; idx++) {
555 m_ts_arr_list[idx].valid = false;
556 }
557 return ret;
558 }
559 #endif
560
561 // factory function executed by the core to create instances
get_omx_component_factory_fn(void)562 void *get_omx_component_factory_fn(void)
563 {
564 return (new omx_vdec);
565 }
566
is_platform_tp10capture_supported()567 bool is_platform_tp10capture_supported()
568 {
569 char platform_name[PROPERTY_VALUE_MAX] = {0};
570 property_get("ro.board.platform", platform_name, "0");
571 if (!strncmp(platform_name, "msm8998", 7)) {
572 DEBUG_PRINT_HIGH("TP10 on capture port is supported");
573 return true;
574 }
575 DEBUG_PRINT_HIGH("TP10 on capture port is not supported");
576 return false;
577 }
578
579 /* ======================================================================
580 FUNCTION
581 omx_vdec::omx_vdec
582
583 DESCRIPTION
584 Constructor
585
586 PARAMETERS
587 None
588
589 RETURN VALUE
590 None.
591 ========================================================================== */
omx_vdec()592 omx_vdec::omx_vdec(): m_error_propogated(false),
593 m_state(OMX_StateInvalid),
594 m_app_data(NULL),
595 m_inp_mem_ptr(NULL),
596 m_out_mem_ptr(NULL),
597 m_client_output_extradata_mem_ptr(NULL),
598 input_flush_progress (false),
599 output_flush_progress (false),
600 input_use_buffer (false),
601 output_use_buffer (false),
602 ouput_egl_buffers(false),
603 m_use_output_pmem(OMX_FALSE),
604 m_out_mem_region_smi(OMX_FALSE),
605 m_out_pvt_entry_pmem(OMX_FALSE),
606 pending_input_buffers(0),
607 pending_output_buffers(0),
608 m_out_bm_count(0),
609 m_inp_bm_count(0),
610 m_out_extradata_bm_count(0),
611 m_inp_bPopulated(OMX_FALSE),
612 m_out_bPopulated(OMX_FALSE),
613 m_flags(0),
614 m_inp_bEnabled(OMX_TRUE),
615 m_out_bEnabled(OMX_TRUE),
616 m_in_alloc_cnt(0),
617 m_platform_list(NULL),
618 m_platform_entry(NULL),
619 m_pmem_info(NULL),
620 h264_parser(NULL),
621 arbitrary_bytes (true),
622 psource_frame (NULL),
623 pdest_frame (NULL),
624 m_inp_heap_ptr (NULL),
625 m_phdr_pmem_ptr(NULL),
626 m_heap_inp_bm_count (0),
627 codec_type_parse ((codec_type)0),
628 first_frame_meta (true),
629 frame_count (0),
630 nal_count (0),
631 nal_length(0),
632 look_ahead_nal (false),
633 first_frame(0),
634 first_buffer(NULL),
635 first_frame_size (0),
636 m_device_file_ptr(NULL),
637 m_vc1_profile((vc1_profile_type)0),
638 h264_last_au_ts(LLONG_MAX),
639 h264_last_au_flags(0),
640 m_disp_hor_size(0),
641 m_disp_vert_size(0),
642 prev_ts(LLONG_MAX),
643 prev_ts_actual(LLONG_MAX),
644 rst_prev_ts(true),
645 frm_int(0),
646 m_fps_received(0),
647 m_fps_prev(0),
648 m_drc_enable(0),
649 in_reconfig(false),
650 m_display_id(NULL),
651 client_extradata(0),
652 m_reject_avc_1080p_mp (0),
653 #ifdef _ANDROID_
654 m_enable_android_native_buffers(OMX_FALSE),
655 m_use_android_native_buffers(OMX_FALSE),
656 #endif
657 m_desc_buffer_ptr(NULL),
658 secure_mode(false),
659 allocate_native_handle(false),
660 m_other_extradata(NULL),
661 m_profile(0),
662 m_need_turbo(0),
663 client_set_fps(false),
664 stereo_output_mode(HAL_NO_3D),
665 m_last_rendered_TS(-1),
666 m_queued_codec_config_count(0),
667 current_perf_level(V4L2_CID_MPEG_VIDC_PERF_LEVEL_NOMINAL),
668 secure_scaling_to_non_secure_opb(false),
669 m_force_compressed_for_dpb(true),
670 m_is_display_session(false)
671 {
672 m_pipe_in = -1;
673 m_pipe_out = -1;
674 m_poll_efd = -1;
675 drv_ctx.video_driver_fd = -1;
676 drv_ctx.extradata_info.ion.fd_ion_data.fd = -1;
677 /* Assumption is that , to begin with , we have all the frames with decoder */
678 DEBUG_PRINT_HIGH("In %u bit OMX vdec Constructor", (unsigned int)sizeof(long) * 8);
679 memset(&m_debug,0,sizeof(m_debug));
680 #ifdef _ANDROID_
681 char property_value[PROPERTY_VALUE_MAX] = {0};
682 property_get("vendor.vidc.debug.level", property_value, "1");
683 debug_level = strtoul(property_value, NULL, 16);
684 property_value[0] = '\0';
685
686 DEBUG_PRINT_HIGH("In OMX vdec Constructor");
687
688 property_get("vendor.vidc.dec.debug.perf", property_value, "0");
689 perf_flag = atoi(property_value);
690 if (perf_flag) {
691 DEBUG_PRINT_HIGH("vidc.dec.debug.perf is %d", perf_flag);
692 dec_time.start();
693 }
694 proc_frms = latency = 0;
695 prev_n_filled_len = 0;
696 property_value[0] = '\0';
697 property_get("vendor.vidc.dec.debug.ts", property_value, "0");
698 m_debug_timestamp = atoi(property_value);
699 DEBUG_PRINT_HIGH("vidc.dec.debug.ts value is %d",m_debug_timestamp);
700 if (m_debug_timestamp) {
701 time_stamp_dts.set_timestamp_reorder_mode(true);
702 time_stamp_dts.enable_debug_print(true);
703 }
704
705 property_value[0] = '\0';
706 property_get("vendor.vidc.dec.debug.concealedmb", property_value, "0");
707 m_debug_concealedmb = atoi(property_value);
708 DEBUG_PRINT_HIGH("vidc.dec.debug.concealedmb value is %d",m_debug_concealedmb);
709
710 property_value[0] = '\0';
711 property_get("vendor.vidc.dec.profile.check", property_value, "0");
712 m_reject_avc_1080p_mp = atoi(property_value);
713 DEBUG_PRINT_HIGH("vendor.vidc.dec.profile.check value is %d",m_reject_avc_1080p_mp);
714
715 property_value[0] = '\0';
716 property_get("vendor.vidc.dec.log.in", property_value, "0");
717 m_debug.in_buffer_log = atoi(property_value);
718
719 property_value[0] = '\0';
720 property_get("vendor.vidc.dec.log.out", property_value, "0");
721 m_debug.out_buffer_log = atoi(property_value);
722 snprintf(m_debug.log_loc, PROPERTY_VALUE_MAX, "%s", BUFFER_LOG_LOC);
723
724 property_value[0] = '\0';
725 property_get("vendor.vidc.dec.meta.log.out", property_value, "0");
726 m_debug.out_meta_buffer_log = atoi(property_value);
727 snprintf(m_debug.log_loc, PROPERTY_VALUE_MAX, "%s", BUFFER_LOG_LOC);
728
729 property_value[0] = '\0';
730 property_get("vendor.vidc.log.loc", property_value, "");
731 if (*property_value)
732 strlcpy(m_debug.log_loc, property_value, PROPERTY_VALUE_MAX);
733
734 property_value[0] = '\0';
735 property_get("vendor.vidc.dec.120fps.enabled", property_value, "0");
736
737 //if this feature is not enabled then reset this value -ve
738 if(atoi(property_value)) {
739 DEBUG_PRINT_LOW("feature 120 FPS decode enabled");
740 m_last_rendered_TS = 0;
741 }
742
743 property_value[0] = '\0';
744 property_get("vendor.vidc.dec.debug.dyn.disabled", property_value, "0");
745 m_disable_dynamic_buf_mode = atoi(property_value);
746 DEBUG_PRINT_HIGH("vidc.dec.debug.dyn.disabled value is %d",m_disable_dynamic_buf_mode);
747
748 property_value[0] = '\0';
749 property_get("vendor.vidc.dec.drc.enable", property_value, "0");
750 if (atoi(property_value)) {
751 m_drc_enable = true;
752 DEBUG_PRINT_HIGH("DRC enabled");
753 }
754
755 #ifdef _UBWC_
756 property_value[0] = '\0';
757 property_get("debug.gralloc.gfx_ubwc_disable", property_value, "0");
758 m_disable_ubwc_mode = atoi(property_value);
759 DEBUG_PRINT_HIGH("UBWC mode is %s", m_disable_ubwc_mode ? "disabled" : "enabled");
760 #else
761 m_disable_ubwc_mode = true;
762 #endif
763 #endif
764 memset(&m_cmp,0,sizeof(m_cmp));
765 memset(&m_cb,0,sizeof(m_cb));
766 memset (&drv_ctx,0,sizeof(drv_ctx));
767 memset (&h264_scratch,0,sizeof (OMX_BUFFERHEADERTYPE));
768 memset (m_hwdevice_name,0,sizeof(m_hwdevice_name));
769 memset(m_demux_offsets, 0, ( sizeof(OMX_U32) * 8192) );
770 memset(&m_custom_buffersize, 0, sizeof(m_custom_buffersize));
771 memset(&m_client_color_space, 0, sizeof(DescribeColorAspectsParams));
772 memset(&m_internal_color_space, 0, sizeof(DescribeColorAspectsParams));
773 memset(&m_client_hdr_info, 0, sizeof(DescribeHDRStaticInfoParams));
774 memset(&m_internal_hdr_info, 0, sizeof(DescribeHDRStaticInfoParams));
775 memset(&m_color_mdata, 0, sizeof(ColorMetaData));
776 m_demux_entries = 0;
777 msg_thread_id = 0;
778 async_thread_id = 0;
779 msg_thread_created = false;
780 async_thread_created = false;
781 async_thread_force_stop = false;
782 message_thread_stop = false;
783 #ifdef _ANDROID_ICS_
784 memset(&native_buffer, 0 ,(sizeof(struct nativebuffer) * MAX_NUM_INPUT_OUTPUT_BUFFERS));
785 #endif
786 memset(&drv_ctx.extradata_info, 0, sizeof(drv_ctx.extradata_info));
787
788 /* invalidate m_frame_pack_arrangement */
789 memset(&m_frame_pack_arrangement, 0, sizeof(OMX_QCOM_FRAME_PACK_ARRANGEMENT));
790 m_frame_pack_arrangement.cancel_flag = 1;
791
792 drv_ctx.timestamp_adjust = false;
793 m_vendor_config.pData = NULL;
794 pthread_mutex_init(&m_lock, NULL);
795 pthread_mutex_init(&c_lock, NULL);
796 pthread_mutex_init(&buf_lock, NULL);
797 sem_init(&m_cmd_lock,0,0);
798 sem_init(&m_safe_flush, 0, 0);
799 streaming[CAPTURE_PORT] =
800 streaming[OUTPUT_PORT] = false;
801 #ifdef _ANDROID_
802 char extradata_value[PROPERTY_VALUE_MAX] = {0};
803 property_get("vendor.vidc.dec.debug.extradata", extradata_value, "0");
804 m_debug_extradata = atoi(extradata_value);
805 DEBUG_PRINT_HIGH("vendor.vidc.dec.debug.extradata value is %d",m_debug_extradata);
806 #endif
807 m_fill_output_msg = OMX_COMPONENT_GENERATE_FTB;
808 client_buffers.set_vdec_client(this);
809 dynamic_buf_mode = false;
810 out_dynamic_list = NULL;
811 is_down_scalar_enabled = false;
812 m_enable_downscalar = 0;
813 m_downscalar_width = 0;
814 m_downscalar_height = 0;
815 m_force_down_scalar = 0;
816 m_reconfig_height = 0;
817 m_reconfig_width = 0;
818 m_smoothstreaming_mode = false;
819 m_smoothstreaming_width = 0;
820 m_smoothstreaming_height = 0;
821 m_decode_order_mode = false;
822 is_q6_platform = false;
823 m_perf_control.send_hint_to_mpctl(true);
824 m_input_pass_buffer_fd = false;
825 memset(&m_extradata_info, 0, sizeof(m_extradata_info));
826 m_client_color_space.nPortIndex = (OMX_U32)OMX_CORE_INPUT_PORT_INDEX;
827 m_client_color_space.sAspects.mRange = ColorAspects::RangeUnspecified;
828 m_client_color_space.sAspects.mPrimaries = ColorAspects::PrimariesUnspecified;
829 m_client_color_space.sAspects.mMatrixCoeffs = ColorAspects::MatrixUnspecified;
830 m_client_color_space.sAspects.mTransfer = ColorAspects::TransferUnspecified;
831
832 m_internal_color_space.nPortIndex = (OMX_U32)OMX_CORE_OUTPUT_PORT_INDEX;
833 m_internal_color_space.sAspects.mRange = ColorAspects::RangeUnspecified;
834 m_internal_color_space.sAspects.mPrimaries = ColorAspects::PrimariesUnspecified;
835 m_internal_color_space.sAspects.mMatrixCoeffs = ColorAspects::MatrixUnspecified;
836 m_internal_color_space.sAspects.mTransfer = ColorAspects::TransferUnspecified;
837 m_internal_color_space.nSize = sizeof(DescribeColorAspectsParams);
838
839 m_client_hdr_info.nPortIndex = (OMX_U32)OMX_CORE_INPUT_PORT_INDEX;
840 m_internal_hdr_info.nPortIndex = (OMX_U32)OMX_CORE_OUTPUT_PORT_INDEX;
841 m_change_client_hdr_info = false;
842 pthread_mutex_init(&m_hdr_info_client_lock, NULL);
843
844 char dither_value[PROPERTY_VALUE_MAX] = {0};
845 property_get("vendor.vidc.dec.dither", dither_value, "0");
846 if ((atoi(dither_value) > DITHER_ALL_COLORSPACE) ||
847 (atoi(dither_value) < DITHER_DISABLE)) {
848 m_dither_config = DITHER_ALL_COLORSPACE;
849 } else {
850 m_dither_config = is_platform_tp10capture_supported() ? (dither_type)atoi(dither_value) : DITHER_ALL_COLORSPACE;
851 }
852
853 DEBUG_PRINT_HIGH("Dither config is %d", m_dither_config);
854 m_color_space = EXCEPT_BT2020;
855 }
856
857 static const int event_type[] = {
858 V4L2_EVENT_MSM_VIDC_FLUSH_DONE,
859 V4L2_EVENT_MSM_VIDC_PORT_SETTINGS_CHANGED_SUFFICIENT,
860 V4L2_EVENT_MSM_VIDC_PORT_SETTINGS_CHANGED_INSUFFICIENT,
861 V4L2_EVENT_MSM_VIDC_PORT_SETTINGS_BITDEPTH_CHANGED_INSUFFICIENT,
862 V4L2_EVENT_MSM_VIDC_RELEASE_BUFFER_REFERENCE,
863 V4L2_EVENT_MSM_VIDC_RELEASE_UNQUEUED_BUFFER,
864 V4L2_EVENT_MSM_VIDC_SYS_ERROR,
865 V4L2_EVENT_MSM_VIDC_HW_OVERLOAD,
866 V4L2_EVENT_MSM_VIDC_HW_UNSUPPORTED
867 };
868
subscribe_to_events(int fd)869 static OMX_ERRORTYPE subscribe_to_events(int fd)
870 {
871 OMX_ERRORTYPE eRet = OMX_ErrorNone;
872 struct v4l2_event_subscription sub;
873 int array_sz = sizeof(event_type)/sizeof(int);
874 int i,rc;
875 if (fd < 0) {
876 DEBUG_PRINT_ERROR("Invalid input: %d", fd);
877 return OMX_ErrorBadParameter;
878 }
879
880 for (i = 0; i < array_sz; ++i) {
881 memset(&sub, 0, sizeof(sub));
882 sub.type = event_type[i];
883 rc = ioctl(fd, VIDIOC_SUBSCRIBE_EVENT, &sub);
884 if (rc) {
885 DEBUG_PRINT_ERROR("Failed to subscribe event: 0x%x", sub.type);
886 break;
887 }
888 }
889 if (i < array_sz) {
890 for (--i; i >=0 ; i--) {
891 memset(&sub, 0, sizeof(sub));
892 sub.type = event_type[i];
893 rc = ioctl(fd, VIDIOC_UNSUBSCRIBE_EVENT, &sub);
894 if (rc)
895 DEBUG_PRINT_ERROR("Failed to unsubscribe event: 0x%x", sub.type);
896 }
897 eRet = OMX_ErrorNotImplemented;
898 }
899 return eRet;
900 }
901
902
unsubscribe_to_events(int fd)903 static OMX_ERRORTYPE unsubscribe_to_events(int fd)
904 {
905 OMX_ERRORTYPE eRet = OMX_ErrorNone;
906 struct v4l2_event_subscription sub;
907 int array_sz = sizeof(event_type)/sizeof(int);
908 int i,rc;
909 if (fd < 0) {
910 DEBUG_PRINT_ERROR("Invalid input: %d", fd);
911 return OMX_ErrorBadParameter;
912 }
913
914 for (i = 0; i < array_sz; ++i) {
915 memset(&sub, 0, sizeof(sub));
916 sub.type = event_type[i];
917 rc = ioctl(fd, VIDIOC_UNSUBSCRIBE_EVENT, &sub);
918 if (rc) {
919 DEBUG_PRINT_ERROR("Failed to unsubscribe event: 0x%x", sub.type);
920 break;
921 }
922 }
923 return eRet;
924 }
925
926 /* ======================================================================
927 FUNCTION
928 omx_vdec::~omx_vdec
929
930 DESCRIPTION
931 Destructor
932
933 PARAMETERS
934 None
935
936 RETURN VALUE
937 None.
938 ========================================================================== */
~omx_vdec()939 omx_vdec::~omx_vdec()
940 {
941 m_pmem_info = NULL;
942 DEBUG_PRINT_HIGH("In OMX vdec Destructor");
943 if (msg_thread_created) {
944 DEBUG_PRINT_HIGH("Signalling close to OMX Msg Thread");
945 message_thread_stop = true;
946 post_message(this, OMX_COMPONENT_CLOSE_MSG);
947 DEBUG_PRINT_HIGH("Waiting on OMX Msg Thread exit");
948 pthread_join(msg_thread_id,NULL);
949 }
950 close(m_pipe_in);
951 close(m_pipe_out);
952 m_pipe_in = -1;
953 m_pipe_out = -1;
954 DEBUG_PRINT_HIGH("Waiting on OMX Async Thread exit");
955 if(eventfd_write(m_poll_efd, 1)) {
956 DEBUG_PRINT_ERROR("eventfd_write failed for fd: %d, errno = %d, force stop async_thread", m_poll_efd, errno);
957 async_thread_force_stop = true;
958 }
959
960 if (async_thread_created)
961 pthread_join(async_thread_id,NULL);
962 unsubscribe_to_events(drv_ctx.video_driver_fd);
963 close(m_poll_efd);
964 close(drv_ctx.video_driver_fd);
965 pthread_mutex_destroy(&m_lock);
966 pthread_mutex_destroy(&c_lock);
967 pthread_mutex_destroy(&buf_lock);
968 sem_destroy(&m_cmd_lock);
969 pthread_mutex_destroy(&m_hdr_info_client_lock);
970 if (perf_flag) {
971 DEBUG_PRINT_HIGH("--> TOTAL PROCESSING TIME");
972 dec_time.end();
973 }
974 DEBUG_PRINT_INFO("Exit OMX vdec Destructor: fd=%d",drv_ctx.video_driver_fd);
975 m_perf_control.send_hint_to_mpctl(false);
976 }
977
release_buffers(omx_vdec * obj,enum vdec_buffer buffer_type)978 int release_buffers(omx_vdec* obj, enum vdec_buffer buffer_type)
979 {
980 struct v4l2_requestbuffers bufreq;
981 int rc = 0;
982 if (buffer_type == VDEC_BUFFER_TYPE_OUTPUT) {
983 bufreq.memory = V4L2_MEMORY_USERPTR;
984 bufreq.count = 0;
985 bufreq.type=V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
986 rc = ioctl(obj->drv_ctx.video_driver_fd,VIDIOC_REQBUFS, &bufreq);
987 } else if(buffer_type == VDEC_BUFFER_TYPE_INPUT) {
988 bufreq.memory = V4L2_MEMORY_USERPTR;
989 bufreq.count = 0;
990 bufreq.type=V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
991 rc = ioctl(obj->drv_ctx.video_driver_fd,VIDIOC_REQBUFS, &bufreq);
992 }
993 return rc;
994 }
995
set_dpb(bool is_split_mode,int dpb_color_format)996 OMX_ERRORTYPE omx_vdec::set_dpb(bool is_split_mode, int dpb_color_format)
997 {
998 int rc = 0;
999 struct v4l2_ext_control ctrl[2];
1000 struct v4l2_ext_controls controls;
1001
1002 DEBUG_PRINT_HIGH("DPB mode: %s DPB color format: %s OPB color format: %s",
1003 is_split_mode ? "split" : "combined",
1004 dpb_color_format == V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_UBWC ? "nv12_ubwc":
1005 dpb_color_format == V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_TP10_UBWC ? "nv12_10bit_ubwc":
1006 dpb_color_format == V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_NONE ? "same as opb":
1007 "unknown",
1008 capture_capability == V4L2_PIX_FMT_NV12 ? "nv12":
1009 capture_capability == V4L2_PIX_FMT_NV12_UBWC ? "nv12_ubwc":
1010 capture_capability == V4L2_PIX_FMT_NV12_TP10_UBWC ? "nv12_10bit_ubwc":
1011 "unknown");
1012
1013 ctrl[0].id = V4L2_CID_MPEG_VIDC_VIDEO_STREAM_OUTPUT_MODE;
1014 if (is_split_mode) {
1015 ctrl[0].value = V4L2_CID_MPEG_VIDC_VIDEO_STREAM_OUTPUT_SECONDARY;
1016 } else {
1017 ctrl[0].value = V4L2_CID_MPEG_VIDC_VIDEO_STREAM_OUTPUT_PRIMARY;
1018 }
1019
1020 ctrl[1].id = V4L2_CID_MPEG_VIDC_VIDEO_DPB_COLOR_FORMAT;
1021 ctrl[1].value = dpb_color_format;
1022
1023 controls.count = 2;
1024 controls.ctrl_class = V4L2_CTRL_CLASS_MPEG;
1025 controls.controls = ctrl;
1026
1027 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_EXT_CTRLS, &controls);
1028 if (rc) {
1029 DEBUG_PRINT_ERROR("Failed to set ext ctrls for opb_dpb: %d\n", rc);
1030 return OMX_ErrorUnsupportedSetting;
1031 }
1032 return OMX_ErrorNone;
1033 }
1034
1035
decide_dpb_buffer_mode(bool split_opb_dpb_with_same_color_fmt)1036 OMX_ERRORTYPE omx_vdec::decide_dpb_buffer_mode(bool split_opb_dpb_with_same_color_fmt)
1037 {
1038 OMX_ERRORTYPE eRet = OMX_ErrorNone;
1039 struct v4l2_format fmt;
1040 int rc = 0;
1041 bool cpu_access = (capture_capability != V4L2_PIX_FMT_NV12_UBWC) &&
1042 capture_capability != V4L2_PIX_FMT_NV12_TP10_UBWC;
1043 bool tp10_enable = !drv_ctx.idr_only_decoding &&
1044 !client_buffers.is_color_conversion_enabled() &&
1045 dpb_bit_depth == MSM_VIDC_BIT_DEPTH_10;
1046 bool dither_enable = true;
1047
1048 switch (m_dither_config) {
1049 case DITHER_DISABLE:
1050 dither_enable = false;
1051 break;
1052 case DITHER_COLORSPACE_EXCEPTBT2020:
1053 dither_enable = (m_color_space == EXCEPT_BT2020);
1054 break;
1055 case DITHER_ALL_COLORSPACE:
1056 dither_enable = true;
1057 break;
1058 default:
1059 DEBUG_PRINT_ERROR("Unsupported dither configuration:%d", m_dither_config);
1060 }
1061
1062 if (tp10_enable && !dither_enable) {
1063 drv_ctx.output_format = VDEC_YUV_FORMAT_NV12_TP10_UBWC;
1064 capture_capability = V4L2_PIX_FMT_NV12_TP10_UBWC;
1065 cpu_access = false;
1066
1067 memset(&fmt, 0x0, sizeof(struct v4l2_format));
1068 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
1069 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
1070 if (rc) {
1071 DEBUG_PRINT_ERROR("%s: Failed get format on capture mplane", __func__);
1072 return OMX_ErrorUnsupportedSetting;
1073 }
1074 fmt.fmt.pix_mp.pixelformat = capture_capability;
1075 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
1076 if (rc) {
1077 DEBUG_PRINT_ERROR("%s: Failed set format on capture mplane", __func__);
1078 return OMX_ErrorUnsupportedSetting;
1079 }
1080 }
1081
1082
1083 if (!BITMASK_PRESENT(&m_flags ,OMX_COMPONENT_IDLE_PENDING) &&
1084 !BITMASK_PRESENT(&m_flags, OMX_COMPONENT_OUTPUT_ENABLE_PENDING)) {
1085 DEBUG_PRINT_LOW("Invalid state to decide on dpb-opb split");
1086 return eRet;
1087 }
1088
1089
1090 if (cpu_access) {
1091 if (dpb_bit_depth == MSM_VIDC_BIT_DEPTH_8) {
1092 /* Disabled split mode for VP9. In split mode the DPB buffers are part of the internal
1093 * scratch buffers and the driver does not does the reference buffer management for
1094 * scratch buffers. In case of VP9 with spatial scalability, when a sequence changed
1095 * event is received with the new resolution, and when a flush is sent by the driver, it
1096 * releases all the references of internal scratch buffers. However as per the VP9
1097 * spatial scalability, even after the flush, the buffers which have not yet received
1098 * release reference event should not be unmapped and freed. Currently in driver,
1099 * reference buffer management of the internal scratch buffer is not implemented
1100 * and hence the DPB buffers get unmapped. For other codecs it does not matter
1101 * as with the new SPS/PPS, the DPB is flushed.
1102 */
1103 bool is_not_vp9 = eCompressionFormat != OMX_VIDEO_CodingVP9;
1104 bool eligible_for_split_dpb_ubwc =
1105 m_progressive == MSM_VIDC_PIC_STRUCT_PROGRESSIVE && //@ Due to Venus limitation for Interlaced, Split mode enabled only for Progressive.
1106 is_not_vp9 && //@ Split mode disabled for VP9.
1107 !drv_ctx.idr_only_decoding && //@ Split mode disabled for Thumbnail usecase.
1108 !m_disable_split_mode; //@ Set prop to disable split mode
1109
1110 //Since opb is linear, dpb should also be linear.
1111 if (split_opb_dpb_with_same_color_fmt) {
1112 eligible_for_split_dpb_ubwc = false;
1113 }
1114
1115 if (eligible_for_split_dpb_ubwc) {
1116 //split DPB-OPB
1117 //DPB -> UBWC , OPB -> Linear
1118 eRet = set_dpb(true, V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_UBWC);
1119 } else if (split_opb_dpb_with_same_color_fmt) {
1120 //DPB -> Linear, OPB -> Linear
1121 eRet = set_dpb(true, V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_NONE);
1122 } else {
1123 //DPB-OPB combined linear
1124 eRet = set_dpb(false, V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_NONE);
1125 }
1126 } else if (dpb_bit_depth == MSM_VIDC_BIT_DEPTH_10) {
1127 //split DPB-OPB
1128 //DPB -> UBWC, OPB -> Linear
1129 eRet = set_dpb(true, V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_TP10_UBWC);
1130 }
1131 } else { //no cpu access
1132 if (dpb_bit_depth == MSM_VIDC_BIT_DEPTH_8) {
1133 if (split_opb_dpb_with_same_color_fmt) {
1134 //split DPB-OPB
1135 //DPB -> UBWC, OPB -> UBWC
1136 eRet = set_dpb(true, V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_UBWC);
1137 } else {
1138 //DPB-OPB combined UBWC
1139 eRet = set_dpb(false, V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_NONE);
1140 }
1141 } else if (dpb_bit_depth == MSM_VIDC_BIT_DEPTH_10) {
1142 if (dither_enable) {
1143 //split DPB-OPB
1144 //DPB -> TP10UBWC, OPB -> UBWC
1145 eRet = set_dpb(true, V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_TP10_UBWC);
1146 } else {
1147 //combined DPB-OPB
1148 //DPB -> TP10UBWC, OPB -> TP10UBWC
1149 eRet = set_dpb(false, V4L2_MPEG_VIDC_VIDEO_DPB_COLOR_FMT_TP10_UBWC);
1150 }
1151 }
1152 }
1153 if (eRet) {
1154 DEBUG_PRINT_HIGH("Failed to set DPB buffer mode: %d", eRet);
1155 }
1156
1157
1158
1159 return eRet;
1160 }
1161
enable_downscalar()1162 int omx_vdec::enable_downscalar()
1163 {
1164 int rc = 0;
1165 struct v4l2_control control;
1166 struct v4l2_format fmt;
1167
1168 if (is_down_scalar_enabled) {
1169 DEBUG_PRINT_LOW("%s: already enabled", __func__);
1170 return 0;
1171 }
1172
1173 DEBUG_PRINT_LOW("omx_vdec::enable_downscalar");
1174 rc = decide_dpb_buffer_mode(true);
1175 if (rc) {
1176 DEBUG_PRINT_ERROR("%s: decide_dpb_buffer_mode Failed ", __func__);
1177 return rc;
1178 }
1179 is_down_scalar_enabled = true;
1180
1181 memset(&control, 0x0, sizeof(struct v4l2_control));
1182 control.id = V4L2_CID_MPEG_VIDC_VIDEO_KEEP_ASPECT_RATIO;
1183 control.value = 1;
1184 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control);
1185 if (rc) {
1186 DEBUG_PRINT_ERROR("%s: Failed to set VIDEO_KEEP_ASPECT_RATIO", __func__);
1187 return rc;
1188 }
1189
1190 return 0;
1191 }
1192
disable_downscalar()1193 int omx_vdec::disable_downscalar()
1194 {
1195 int rc = 0;
1196 struct v4l2_control control;
1197
1198 if (!is_down_scalar_enabled) {
1199 DEBUG_PRINT_LOW("omx_vdec::disable_downscalar: already disabled");
1200 return 0;
1201 }
1202
1203 rc = decide_dpb_buffer_mode(false);
1204 if (rc < 0) {
1205 DEBUG_PRINT_ERROR("%s:decide_dpb_buffer_mode failed\n", __func__);
1206 return rc;
1207 }
1208 is_down_scalar_enabled = false;
1209
1210 return rc;
1211 }
1212
decide_downscalar()1213 int omx_vdec::decide_downscalar()
1214 {
1215 int rc = 0;
1216 struct v4l2_format fmt;
1217 enum color_fmts color_format;
1218 OMX_U32 width, height;
1219 OMX_BOOL isPortraitVideo = OMX_FALSE;
1220
1221 if (capture_capability == V4L2_PIX_FMT_NV12_TP10_UBWC) {
1222 rc = disable_downscalar();
1223 if (rc) {
1224 DEBUG_PRINT_ERROR("Disable downscalar failed!");
1225 return rc;
1226 }
1227 return 0;
1228 }
1229
1230 if (!m_enable_downscalar) {
1231 DEBUG_PRINT_LOW("%s: downscalar not supported", __func__);
1232 return 0;
1233 }
1234
1235 #ifdef _QUERY_DISP_RES_
1236 memset(&fmt, 0x0, sizeof(struct v4l2_format));
1237 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
1238 fmt.fmt.pix_mp.pixelformat = capture_capability;
1239 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
1240 if (rc < 0) {
1241 DEBUG_PRINT_ERROR("%s: Failed to get format on capture mplane", __func__);
1242 return rc;
1243 }
1244 isPortraitVideo = fmt.fmt.pix_mp.width < fmt.fmt.pix_mp.height ? OMX_TRUE : OMX_FALSE;
1245 if (!m_downscalar_width || !m_downscalar_height) {
1246 qdutils::DisplayAttributes dpa = {}, dsa = {}, dva = {};
1247 int prim_config, ext_config, virt_config;
1248
1249 prim_config = qdutils::getActiveConfig(qdutils::DISPLAY_PRIMARY);
1250 dpa = qdutils::getDisplayAttributes(prim_config, qdutils::DISPLAY_PRIMARY);
1251 DEBUG_PRINT_HIGH("%s: Primary dpa.xres = %d dpa.yres=%d dpa.xdpi = %f dpa.ydpi = %f ",
1252 __func__, dpa.xres, dpa.yres, dpa.xdpi, dpa.ydpi);
1253
1254 ext_config = qdutils::getActiveConfig(qdutils::DISPLAY_EXTERNAL);
1255 dsa = qdutils::getDisplayAttributes(ext_config, qdutils::DISPLAY_EXTERNAL);
1256 DEBUG_PRINT_HIGH("%s: HDMI dsa.xres = %d dsa.yres = %d dsa.xdpi = %f dsa.ydpi = %f ",
1257 __func__, dsa.xres, dsa.yres, dsa.xdpi, dsa.ydpi);
1258
1259 virt_config = qdutils::getActiveConfig(qdutils::DISPLAY_VIRTUAL);
1260 dva = qdutils::getDisplayAttributes(virt_config, qdutils::DISPLAY_VIRTUAL);
1261 DEBUG_PRINT_HIGH("%s: Virtual dva.xres = %d dva.yres = %d dva.xdpi = %f dva.ydpi = %f ",
1262 __func__, dva.xres, dva.yres, dva.xdpi, dva.ydpi);
1263
1264 /* Below logic takes care of following conditions:
1265 * 1. Choose display resolution as maximum resolution of all the connected
1266 * displays (secondary, primary, virtual), so that we do not downscale
1267 * unnecessarily which might be supported on one of the display losing quality.
1268 * 2. Displays connected might be in landscape or portrait mode, so the xres might
1269 * be smaller or greater than the yres. So we first take the max of the two
1270 * in width and min of two in height and then rotate it if below point is true.
1271 * 3. Video might also be in portrait mode, so invert the downscalar width and
1272 * height for such cases.
1273 */
1274 if (dsa.xres * dsa.yres > dpa.xres * dpa.yres) {
1275 m_downscalar_width = MAX(dsa.xres, dsa.yres);
1276 m_downscalar_height = MIN(dsa.xres, dsa.yres);
1277 } else if (dva.xres * dva.yres > dpa.xres * dpa.yres) {
1278 m_downscalar_width = MAX(dva.xres, dva.yres);
1279 m_downscalar_height = MIN(dva.xres, dva.yres);
1280
1281 } else {
1282 m_downscalar_width = MAX(dpa.xres, dpa.yres);
1283 m_downscalar_height = MIN(dpa.xres, dpa.yres);
1284 }
1285 if (isPortraitVideo) {
1286 // Swap width and height
1287 m_downscalar_width = m_downscalar_width ^ m_downscalar_height;
1288 m_downscalar_height = m_downscalar_width ^ m_downscalar_height;
1289 m_downscalar_width = m_downscalar_width ^ m_downscalar_height;
1290 }
1291 }
1292 m_downscalar_width = ALIGN(m_downscalar_width, 128);
1293 m_downscalar_height = ALIGN(m_downscalar_height, 32);
1294 #endif
1295
1296 if (!m_downscalar_width || !m_downscalar_height) {
1297 DEBUG_PRINT_LOW("%s: Invalid downscalar configuration", __func__);
1298 return 0;
1299 }
1300
1301 if (m_force_down_scalar) {
1302 DEBUG_PRINT_LOW("%s: m_force_down_scalar %d ", __func__, m_force_down_scalar);
1303 return 0;
1304 }
1305
1306 memset(&fmt, 0x0, sizeof(struct v4l2_format));
1307 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
1308 fmt.fmt.pix_mp.pixelformat = capture_capability;
1309 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
1310 if (rc < 0) {
1311 DEBUG_PRINT_ERROR("%s: Failed to get format on capture mplane", __func__);
1312 return rc;
1313 }
1314
1315 height = fmt.fmt.pix_mp.height;
1316 width = fmt.fmt.pix_mp.width;
1317
1318 DEBUG_PRINT_HIGH("%s: driver wxh = %dx%d, downscalar wxh = %dx%d m_is_display_session = %d", __func__,
1319 fmt.fmt.pix_mp.width, fmt.fmt.pix_mp.height, m_downscalar_width, m_downscalar_height, m_is_display_session);
1320
1321 if ((fmt.fmt.pix_mp.width * fmt.fmt.pix_mp.height > m_downscalar_width * m_downscalar_height) &&
1322 m_is_display_session) {
1323 rc = enable_downscalar();
1324 if (rc < 0) {
1325 DEBUG_PRINT_ERROR("%s: enable_downscalar failed\n", __func__);
1326 return rc;
1327 }
1328
1329 width = m_downscalar_width > fmt.fmt.pix_mp.width ?
1330 fmt.fmt.pix_mp.width : m_downscalar_width;
1331 height = m_downscalar_height > fmt.fmt.pix_mp.height ?
1332 fmt.fmt.pix_mp.height : m_downscalar_height;
1333 switch (capture_capability) {
1334 case V4L2_PIX_FMT_NV12:
1335 color_format = COLOR_FMT_NV12;
1336 break;
1337 case V4L2_PIX_FMT_NV12_UBWC:
1338 color_format = COLOR_FMT_NV12_UBWC;
1339 break;
1340 case V4L2_PIX_FMT_NV12_TP10_UBWC:
1341 color_format = COLOR_FMT_NV12_BPP10_UBWC;
1342 break;
1343 default:
1344 DEBUG_PRINT_ERROR("Color format not recognized\n");
1345 rc = OMX_ErrorUndefined;
1346 return rc;
1347 }
1348 } else {
1349
1350 rc = disable_downscalar();
1351 if (rc < 0) {
1352 DEBUG_PRINT_ERROR("%s: disable_downscalar failed\n", __func__);
1353 return rc;
1354 }
1355 }
1356
1357 memset(&fmt, 0x0, sizeof(struct v4l2_format));
1358 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
1359 fmt.fmt.pix_mp.height = height;
1360 fmt.fmt.pix_mp.width = width;
1361 fmt.fmt.pix_mp.pixelformat = capture_capability;
1362 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
1363 if (rc) {
1364 DEBUG_PRINT_ERROR("%s: Failed set format on capture mplane", __func__);
1365 return rc;
1366 }
1367
1368 rc = get_buffer_req(&drv_ctx.op_buf);
1369 if (rc) {
1370 DEBUG_PRINT_ERROR("%s: Failed to get output buffer requirements", __func__);
1371 return rc;
1372 }
1373
1374 return rc;
1375 }
1376
1377 /* ======================================================================
1378 FUNCTION
1379 omx_vdec::OMXCntrlProcessMsgCb
1380
1381 DESCRIPTION
1382 IL Client callbacks are generated through this routine. The decoder
1383 provides the thread context for this routine.
1384
1385 PARAMETERS
1386 ctxt -- Context information related to the self.
1387 id -- Event identifier. This could be any of the following:
1388 1. Command completion event
1389 2. Buffer done callback event
1390 3. Frame done callback event
1391
1392 RETURN VALUE
1393 None.
1394
1395 ========================================================================== */
process_event_cb(void * ctxt,unsigned char id)1396 void omx_vdec::process_event_cb(void *ctxt, unsigned char id)
1397 {
1398 unsigned long p1; // Parameter - 1
1399 unsigned long p2; // Parameter - 2
1400 unsigned long ident;
1401 unsigned qsize=0; // qsize
1402 omx_vdec *pThis = (omx_vdec *) ctxt;
1403
1404 if (!pThis) {
1405 DEBUG_PRINT_ERROR("ERROR: %s()::Context is incorrect, bailing out",
1406 __func__);
1407 return;
1408 }
1409
1410 // Protect the shared queue data structure
1411 do {
1412 /*Read the message id's from the queue*/
1413 pthread_mutex_lock(&pThis->m_lock);
1414 qsize = pThis->m_cmd_q.m_size;
1415 if (qsize) {
1416 pThis->m_cmd_q.pop_entry(&p1, &p2, &ident);
1417 }
1418
1419 if (qsize == 0 && pThis->m_state != OMX_StatePause) {
1420 qsize = pThis->m_ftb_q.m_size;
1421 if (qsize) {
1422 pThis->m_ftb_q.pop_entry(&p1, &p2, &ident);
1423 }
1424 }
1425
1426 if (qsize == 0 && pThis->m_state != OMX_StatePause) {
1427 qsize = pThis->m_etb_q.m_size;
1428 if (qsize) {
1429 pThis->m_etb_q.pop_entry(&p1, &p2, &ident);
1430 }
1431 }
1432 pthread_mutex_unlock(&pThis->m_lock);
1433
1434 /*process message if we have one*/
1435 if (qsize > 0) {
1436 id = ident;
1437 switch (id) {
1438 case OMX_COMPONENT_GENERATE_EVENT:
1439 if (pThis->m_cb.EventHandler) {
1440 switch (p1) {
1441 case OMX_CommandStateSet:
1442 pThis->m_state = (OMX_STATETYPE) p2;
1443 DEBUG_PRINT_HIGH("OMX_CommandStateSet complete, m_state = %d",
1444 pThis->m_state);
1445 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
1446 OMX_EventCmdComplete, p1, p2, NULL);
1447 break;
1448
1449 case OMX_EventError:
1450 if (p2 == OMX_StateInvalid) {
1451 DEBUG_PRINT_ERROR("OMX_EventError: p2 is OMX_StateInvalid");
1452 pThis->m_state = (OMX_STATETYPE) p2;
1453 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
1454 OMX_EventError, OMX_ErrorInvalidState, p2, NULL);
1455 } else if (p2 == (unsigned long)OMX_ErrorHardware) {
1456 pThis->omx_report_error();
1457 } else {
1458 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
1459 OMX_EventError, p2, (OMX_U32)NULL, NULL );
1460 }
1461 break;
1462
1463 case OMX_CommandPortDisable:
1464 DEBUG_PRINT_HIGH("OMX_CommandPortDisable complete for port [%lu]", p2);
1465 if (BITMASK_PRESENT(&pThis->m_flags,
1466 OMX_COMPONENT_OUTPUT_FLUSH_IN_DISABLE_PENDING)) {
1467 BITMASK_SET(&pThis->m_flags, OMX_COMPONENT_DISABLE_OUTPUT_DEFERRED);
1468 break;
1469 }
1470 if (p2 == OMX_CORE_OUTPUT_PORT_INDEX) {
1471 OMX_ERRORTYPE eRet = OMX_ErrorNone;
1472 pThis->stream_off(OMX_CORE_OUTPUT_PORT_INDEX);
1473 if (release_buffers(pThis, VDEC_BUFFER_TYPE_OUTPUT))
1474 DEBUG_PRINT_HIGH("Failed to release output buffers");
1475 OMX_ERRORTYPE eRet1 = pThis->get_buffer_req(&pThis->drv_ctx.op_buf);
1476 if (eRet != OMX_ErrorNone) {
1477 DEBUG_PRINT_ERROR("set_buffer_req failed eRet = %d",eRet);
1478 pThis->omx_report_error();
1479 break;
1480 }
1481 }
1482 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
1483 OMX_EventCmdComplete, p1, p2, NULL );
1484 break;
1485 case OMX_CommandPortEnable:
1486 DEBUG_PRINT_HIGH("OMX_CommandPortEnable complete for port [%lu]", p2);
1487 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,\
1488 OMX_EventCmdComplete, p1, p2, NULL );
1489 pThis->in_reconfig = false;
1490 break;
1491
1492 default:
1493 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
1494 OMX_EventCmdComplete, p1, p2, NULL );
1495 break;
1496
1497 }
1498 } else {
1499 DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
1500 }
1501 break;
1502 case OMX_COMPONENT_GENERATE_ETB_ARBITRARY:
1503 if (pThis->empty_this_buffer_proxy_arbitrary((OMX_HANDLETYPE)p1,\
1504 (OMX_BUFFERHEADERTYPE *)(intptr_t)p2) != OMX_ErrorNone) {
1505 DEBUG_PRINT_ERROR("empty_this_buffer_proxy_arbitrary failure");
1506 pThis->omx_report_error ();
1507 }
1508 break;
1509 case OMX_COMPONENT_GENERATE_ETB: {
1510 OMX_ERRORTYPE iret;
1511 iret = pThis->empty_this_buffer_proxy((OMX_HANDLETYPE)p1, (OMX_BUFFERHEADERTYPE *)p2);
1512 if (iret == OMX_ErrorInsufficientResources) {
1513 DEBUG_PRINT_ERROR("empty_this_buffer_proxy failure due to HW overload");
1514 pThis->omx_report_hw_overload ();
1515 } else if (iret != OMX_ErrorNone) {
1516 DEBUG_PRINT_ERROR("empty_this_buffer_proxy failure");
1517 pThis->omx_report_error ();
1518 }
1519 }
1520 break;
1521
1522 case OMX_COMPONENT_GENERATE_FTB:
1523 if ( pThis->fill_this_buffer_proxy((OMX_HANDLETYPE)(intptr_t)p1,\
1524 (OMX_BUFFERHEADERTYPE *)(intptr_t)p2) != OMX_ErrorNone) {
1525 DEBUG_PRINT_ERROR("fill_this_buffer_proxy failure");
1526 pThis->omx_report_error ();
1527 }
1528 break;
1529
1530 case OMX_COMPONENT_GENERATE_COMMAND:
1531 pThis->send_command_proxy(&pThis->m_cmp,(OMX_COMMANDTYPE)p1,\
1532 (OMX_U32)p2,(OMX_PTR)NULL);
1533 break;
1534
1535 case OMX_COMPONENT_GENERATE_EBD:
1536
1537 if (p2 != VDEC_S_SUCCESS && p2 != VDEC_S_INPUT_BITSTREAM_ERR) {
1538 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_EBD failure");
1539 pThis->omx_report_error ();
1540 } else {
1541 if (p2 == VDEC_S_INPUT_BITSTREAM_ERR && p1) {
1542 pThis->time_stamp_dts.remove_time_stamp(
1543 ((OMX_BUFFERHEADERTYPE *)(intptr_t)p1)->nTimeStamp,
1544 (pThis->drv_ctx.interlace != VDEC_InterlaceFrameProgressive)
1545 ?true:false);
1546 }
1547
1548 if ( pThis->empty_buffer_done(&pThis->m_cmp,
1549 (OMX_BUFFERHEADERTYPE *)(intptr_t)p1) != OMX_ErrorNone) {
1550 DEBUG_PRINT_ERROR("empty_buffer_done failure");
1551 pThis->omx_report_error ();
1552 }
1553 }
1554 break;
1555 case OMX_COMPONENT_GENERATE_INFO_FIELD_DROPPED: {
1556 int64_t *timestamp = (int64_t *)(intptr_t)p1;
1557 if (p1) {
1558 pThis->time_stamp_dts.remove_time_stamp(*timestamp,
1559 (pThis->drv_ctx.interlace != VDEC_InterlaceFrameProgressive)
1560 ?true:false);
1561 free(timestamp);
1562 }
1563 }
1564 break;
1565 case OMX_COMPONENT_GENERATE_FBD:
1566 if (p2 != VDEC_S_SUCCESS) {
1567 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_FBD failure");
1568 pThis->omx_report_error ();
1569 } else if ( pThis->fill_buffer_done(&pThis->m_cmp,
1570 (OMX_BUFFERHEADERTYPE *)(intptr_t)p1) != OMX_ErrorNone ) {
1571 DEBUG_PRINT_ERROR("fill_buffer_done failure");
1572 pThis->omx_report_error ();
1573 }
1574 break;
1575
1576 case OMX_COMPONENT_GENERATE_EVENT_INPUT_FLUSH:
1577 DEBUG_PRINT_HIGH("Driver flush i/p Port complete");
1578 if (!pThis->input_flush_progress) {
1579 DEBUG_PRINT_HIGH("WARNING: Unexpected flush from driver");
1580 } else {
1581 pThis->execute_input_flush();
1582 if (pThis->m_cb.EventHandler) {
1583 if (p2 != VDEC_S_SUCCESS) {
1584 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_EVENT_INPUT_FLUSH failure");
1585 pThis->omx_report_error ();
1586 } else {
1587 /*Check if we need generate event for Flush done*/
1588 pThis->notify_flush_done(ctxt);
1589
1590 if (BITMASK_PRESENT(&pThis->m_flags,
1591 OMX_COMPONENT_IDLE_PENDING)) {
1592 if (pThis->stream_off(OMX_CORE_INPUT_PORT_INDEX)) {
1593 DEBUG_PRINT_ERROR("Failed to call streamoff on OUTPUT Port");
1594 pThis->omx_report_error ();
1595 } else {
1596 pThis->streaming[OUTPUT_PORT] = false;
1597 }
1598 if (!pThis->output_flush_progress) {
1599 DEBUG_PRINT_LOW("Input flush done hence issue stop");
1600 pThis->post_event ((unsigned int)NULL, VDEC_S_SUCCESS,\
1601 OMX_COMPONENT_GENERATE_STOP_DONE);
1602 }
1603 }
1604 }
1605 } else {
1606 DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
1607 }
1608 }
1609 break;
1610
1611 case OMX_COMPONENT_GENERATE_EVENT_OUTPUT_FLUSH:
1612 DEBUG_PRINT_HIGH("Driver flush o/p Port complete");
1613 if (!pThis->output_flush_progress) {
1614 DEBUG_PRINT_HIGH("WARNING: Unexpected flush from driver");
1615 } else {
1616 pThis->execute_output_flush();
1617 if (pThis->m_cb.EventHandler) {
1618 if (p2 != VDEC_S_SUCCESS) {
1619 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_EVENT_OUTPUT_FLUSH failed");
1620 pThis->omx_report_error ();
1621 } else {
1622 /*Check if we need generate event for Flush done*/
1623 pThis->notify_flush_done(ctxt);
1624
1625 if (BITMASK_PRESENT(&pThis->m_flags,
1626 OMX_COMPONENT_OUTPUT_FLUSH_IN_DISABLE_PENDING)) {
1627 DEBUG_PRINT_LOW("Internal flush complete");
1628 BITMASK_CLEAR (&pThis->m_flags,
1629 OMX_COMPONENT_OUTPUT_FLUSH_IN_DISABLE_PENDING);
1630 if (BITMASK_PRESENT(&pThis->m_flags,
1631 OMX_COMPONENT_DISABLE_OUTPUT_DEFERRED)) {
1632 pThis->post_event(OMX_CommandPortDisable,
1633 OMX_CORE_OUTPUT_PORT_INDEX,
1634 OMX_COMPONENT_GENERATE_EVENT);
1635 BITMASK_CLEAR (&pThis->m_flags,
1636 OMX_COMPONENT_DISABLE_OUTPUT_DEFERRED);
1637 BITMASK_CLEAR (&pThis->m_flags,
1638 OMX_COMPONENT_OUTPUT_DISABLE_PENDING);
1639
1640 }
1641 }
1642
1643 if (BITMASK_PRESENT(&pThis->m_flags ,OMX_COMPONENT_IDLE_PENDING)) {
1644 if (pThis->stream_off(OMX_CORE_OUTPUT_PORT_INDEX)) {
1645 DEBUG_PRINT_ERROR("Failed to call streamoff on CAPTURE Port");
1646 pThis->omx_report_error ();
1647 break;
1648 }
1649 pThis->streaming[CAPTURE_PORT] = false;
1650 if (!pThis->input_flush_progress) {
1651 DEBUG_PRINT_LOW("Output flush done hence issue stop");
1652 pThis->post_event ((unsigned int)NULL, VDEC_S_SUCCESS,\
1653 OMX_COMPONENT_GENERATE_STOP_DONE);
1654 }
1655 }
1656 }
1657 } else {
1658 DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
1659 }
1660 }
1661 break;
1662
1663 case OMX_COMPONENT_GENERATE_START_DONE:
1664 DEBUG_PRINT_HIGH("Rxd OMX_COMPONENT_GENERATE_START_DONE");
1665
1666 if (pThis->m_cb.EventHandler) {
1667 if (p2 != VDEC_S_SUCCESS) {
1668 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_START_DONE Failure");
1669 pThis->omx_report_error ();
1670 } else {
1671 DEBUG_PRINT_LOW("OMX_COMPONENT_GENERATE_START_DONE Success");
1672 if (BITMASK_PRESENT(&pThis->m_flags,OMX_COMPONENT_EXECUTE_PENDING)) {
1673 DEBUG_PRINT_LOW("Move to executing");
1674 // Send the callback now
1675 BITMASK_CLEAR((&pThis->m_flags),OMX_COMPONENT_EXECUTE_PENDING);
1676 pThis->m_state = OMX_StateExecuting;
1677 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
1678 OMX_EventCmdComplete,OMX_CommandStateSet,
1679 OMX_StateExecuting, NULL);
1680 } else if (BITMASK_PRESENT(&pThis->m_flags,
1681 OMX_COMPONENT_PAUSE_PENDING)) {
1682 if (/*ioctl (pThis->drv_ctx.video_driver_fd,
1683 VDEC_IOCTL_CMD_PAUSE,NULL ) < */0) {
1684 DEBUG_PRINT_ERROR("VDEC_IOCTL_CMD_PAUSE failed");
1685 pThis->omx_report_error ();
1686 }
1687 }
1688 }
1689 } else {
1690 DEBUG_PRINT_LOW("Event Handler callback is NULL");
1691 }
1692 break;
1693
1694 case OMX_COMPONENT_GENERATE_PAUSE_DONE:
1695 DEBUG_PRINT_HIGH("Rxd OMX_COMPONENT_GENERATE_PAUSE_DONE");
1696 if (pThis->m_cb.EventHandler) {
1697 if (p2 != VDEC_S_SUCCESS) {
1698 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_PAUSE_DONE ret failed");
1699 pThis->omx_report_error ();
1700 } else {
1701 pThis->complete_pending_buffer_done_cbs();
1702 if (BITMASK_PRESENT(&pThis->m_flags,OMX_COMPONENT_PAUSE_PENDING)) {
1703 DEBUG_PRINT_LOW("OMX_COMPONENT_GENERATE_PAUSE_DONE nofity");
1704 //Send the callback now
1705 BITMASK_CLEAR((&pThis->m_flags),OMX_COMPONENT_PAUSE_PENDING);
1706 pThis->m_state = OMX_StatePause;
1707 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
1708 OMX_EventCmdComplete,OMX_CommandStateSet,
1709 OMX_StatePause, NULL);
1710 }
1711 }
1712 } else {
1713 DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
1714 }
1715
1716 break;
1717
1718 case OMX_COMPONENT_GENERATE_RESUME_DONE:
1719 DEBUG_PRINT_HIGH("Rxd OMX_COMPONENT_GENERATE_RESUME_DONE");
1720 if (pThis->m_cb.EventHandler) {
1721 if (p2 != VDEC_S_SUCCESS) {
1722 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_RESUME_DONE failed");
1723 pThis->omx_report_error ();
1724 } else {
1725 if (BITMASK_PRESENT(&pThis->m_flags,OMX_COMPONENT_EXECUTE_PENDING)) {
1726 DEBUG_PRINT_LOW("Moving the decoder to execute state");
1727 // Send the callback now
1728 BITMASK_CLEAR((&pThis->m_flags),OMX_COMPONENT_EXECUTE_PENDING);
1729 pThis->m_state = OMX_StateExecuting;
1730 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
1731 OMX_EventCmdComplete,OMX_CommandStateSet,
1732 OMX_StateExecuting,NULL);
1733 }
1734 }
1735 } else {
1736 DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
1737 }
1738
1739 break;
1740
1741 case OMX_COMPONENT_GENERATE_STOP_DONE:
1742 DEBUG_PRINT_HIGH("Rxd OMX_COMPONENT_GENERATE_STOP_DONE");
1743 if (pThis->m_cb.EventHandler) {
1744 if (p2 != VDEC_S_SUCCESS) {
1745 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_STOP_DONE ret failed");
1746 pThis->omx_report_error ();
1747 } else {
1748 pThis->complete_pending_buffer_done_cbs();
1749 if (BITMASK_PRESENT(&pThis->m_flags,OMX_COMPONENT_IDLE_PENDING)) {
1750 DEBUG_PRINT_LOW("OMX_COMPONENT_GENERATE_STOP_DONE Success");
1751 // Send the callback now
1752 BITMASK_CLEAR((&pThis->m_flags),OMX_COMPONENT_IDLE_PENDING);
1753 pThis->m_state = OMX_StateIdle;
1754 DEBUG_PRINT_LOW("Move to Idle State");
1755 pThis->m_cb.EventHandler(&pThis->m_cmp,pThis->m_app_data,
1756 OMX_EventCmdComplete,OMX_CommandStateSet,
1757 OMX_StateIdle,NULL);
1758 }
1759 }
1760 } else {
1761 DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
1762 }
1763
1764 break;
1765
1766 case OMX_COMPONENT_GENERATE_PORT_RECONFIG:
1767 if (p2 == OMX_IndexParamPortDefinition) {
1768 DEBUG_PRINT_HIGH("Rxd PORT_RECONFIG: OMX_IndexParamPortDefinition");
1769 pThis->in_reconfig = true;
1770 pThis->m_need_turbo &= ~TURBO_MODE_HIGH_FPS;
1771 } else if (p2 == OMX_IndexConfigCommonOutputCrop) {
1772 DEBUG_PRINT_HIGH("Rxd PORT_RECONFIG: OMX_IndexConfigCommonOutputCrop");
1773
1774 /* Check if resolution is changed in smooth streaming mode */
1775 if (pThis->m_smoothstreaming_mode &&
1776 (pThis->framesize.nWidth !=
1777 pThis->drv_ctx.video_resolution.frame_width) ||
1778 (pThis->framesize.nHeight !=
1779 pThis->drv_ctx.video_resolution.frame_height)) {
1780
1781 DEBUG_PRINT_HIGH("Resolution changed from: wxh = %dx%d to: wxh = %dx%d",
1782 pThis->framesize.nWidth,
1783 pThis->framesize.nHeight,
1784 pThis->drv_ctx.video_resolution.frame_width,
1785 pThis->drv_ctx.video_resolution.frame_height);
1786
1787 /* Update new resolution */
1788 pThis->framesize.nWidth =
1789 pThis->drv_ctx.video_resolution.frame_width;
1790 pThis->framesize.nHeight =
1791 pThis->drv_ctx.video_resolution.frame_height;
1792
1793 /* Update C2D with new resolution */
1794 if (!pThis->client_buffers.update_buffer_req()) {
1795 DEBUG_PRINT_ERROR("Setting C2D buffer requirements failed");
1796 }
1797 }
1798
1799 /* Update new crop information */
1800 pThis->rectangle.nLeft = pThis->drv_ctx.frame_size.left;
1801 pThis->rectangle.nTop = pThis->drv_ctx.frame_size.top;
1802 pThis->rectangle.nWidth = pThis->drv_ctx.frame_size.right;
1803 pThis->rectangle.nHeight = pThis->drv_ctx.frame_size.bottom;
1804
1805 /* Validate the new crop information */
1806 if (pThis->rectangle.nLeft + pThis->rectangle.nWidth >
1807 pThis->drv_ctx.video_resolution.frame_width) {
1808
1809 DEBUG_PRINT_HIGH("Crop L[%u] + R[%u] > W[%u]",
1810 pThis->rectangle.nLeft, pThis->rectangle.nWidth,
1811 pThis->drv_ctx.video_resolution.frame_width);
1812 pThis->rectangle.nLeft = 0;
1813
1814 if (pThis->rectangle.nWidth >
1815 pThis->drv_ctx.video_resolution.frame_width) {
1816
1817 DEBUG_PRINT_HIGH("Crop R[%u] > W[%u]",
1818 pThis->rectangle.nWidth,
1819 pThis->drv_ctx.video_resolution.frame_width);
1820 pThis->rectangle.nWidth =
1821 pThis->drv_ctx.video_resolution.frame_width;
1822 }
1823 }
1824 if (pThis->rectangle.nTop + pThis->rectangle.nHeight >
1825 pThis->drv_ctx.video_resolution.frame_height) {
1826
1827 DEBUG_PRINT_HIGH("Crop T[%u] + B[%u] > H[%u]",
1828 pThis->rectangle.nTop, pThis->rectangle.nHeight,
1829 pThis->drv_ctx.video_resolution.frame_height);
1830 pThis->rectangle.nTop = 0;
1831
1832 if (pThis->rectangle.nHeight >
1833 pThis->drv_ctx.video_resolution.frame_height) {
1834
1835 DEBUG_PRINT_HIGH("Crop B[%u] > H[%u]",
1836 pThis->rectangle.nHeight,
1837 pThis->drv_ctx.video_resolution.frame_height);
1838 pThis->rectangle.nHeight =
1839 pThis->drv_ctx.video_resolution.frame_height;
1840 }
1841 }
1842 DEBUG_PRINT_HIGH("Updated Crop Info: L: %u, T: %u, R: %u, B: %u",
1843 pThis->rectangle.nLeft, pThis->rectangle.nTop,
1844 pThis->rectangle.nWidth, pThis->rectangle.nHeight);
1845 } else if (p2 == OMX_QTIIndexConfigDescribeColorAspects) {
1846 DEBUG_PRINT_HIGH("Rxd PORT_RECONFIG: OMX_QTIIndexConfigDescribeColorAspects");
1847 } else if (p2 == OMX_QTIIndexConfigDescribeHDRColorInfo) {
1848 DEBUG_PRINT_HIGH("Rxd PORT_RECONFIG: OMX_QTIIndexConfigDescribeHDRcolorinfo");
1849 } else {
1850 DEBUG_PRINT_ERROR("Rxd Invalid PORT_RECONFIG event (%lu)", p2);
1851 break;
1852 }
1853 if (pThis->m_debug.outfile) {
1854 fclose(pThis->m_debug.outfile);
1855 pThis->m_debug.outfile = NULL;
1856 }
1857 if (pThis->m_debug.out_ymeta_file) {
1858 fclose(pThis->m_debug.out_ymeta_file);
1859 pThis->m_debug.out_ymeta_file = NULL;
1860 }
1861 if (pThis->m_debug.out_uvmeta_file) {
1862 fclose(pThis->m_debug.out_uvmeta_file);
1863 pThis->m_debug.out_uvmeta_file = NULL;
1864 }
1865
1866 if (pThis->secure_mode && pThis->m_cb.EventHandler && pThis->in_reconfig) {
1867 pThis->prefetchNewBuffers();
1868 }
1869
1870 if (pThis->m_cb.EventHandler) {
1871 uint32_t frame_data[7];
1872 frame_data[0] = (p2 == OMX_IndexParamPortDefinition) ?
1873 pThis->m_reconfig_height : pThis->rectangle.nHeight;
1874 frame_data[1] = (p2 == OMX_IndexParamPortDefinition) ?
1875 pThis->m_reconfig_width : pThis->rectangle.nWidth;
1876
1877 frame_data[2] = (p2 == OMX_IndexParamPortDefinition) ?
1878 frame_data[0] : pThis->drv_ctx.video_resolution.frame_height;
1879
1880 frame_data[3] = (p2 == OMX_IndexParamPortDefinition) ?
1881 frame_data[1] : pThis->drv_ctx.video_resolution.frame_width;
1882 frame_data[4] = pThis->dpb_bit_depth;
1883 frame_data[5] = pThis->m_color_space;
1884 frame_data[6] = pThis->m_dither_config;
1885
1886 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
1887 OMX_EventPortSettingsChanged, p1, p2, (void*) frame_data );
1888 } else {
1889 DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
1890 }
1891 break;
1892
1893 case OMX_COMPONENT_GENERATE_EOS_DONE:
1894 DEBUG_PRINT_HIGH("Rxd OMX_COMPONENT_GENERATE_EOS_DONE");
1895 if (pThis->m_cb.EventHandler) {
1896 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data, OMX_EventBufferFlag,
1897 OMX_CORE_OUTPUT_PORT_INDEX, OMX_BUFFERFLAG_EOS, NULL );
1898 } else {
1899 DEBUG_PRINT_ERROR("ERROR: %s()::EventHandler is NULL", __func__);
1900 }
1901 pThis->prev_ts = LLONG_MAX;
1902 pThis->rst_prev_ts = true;
1903 break;
1904
1905 case OMX_COMPONENT_GENERATE_HARDWARE_ERROR:
1906 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_HARDWARE_ERROR");
1907 pThis->omx_report_error();
1908 break;
1909
1910 case OMX_COMPONENT_GENERATE_UNSUPPORTED_SETTING:
1911 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_UNSUPPORTED_SETTING");
1912 pThis->omx_report_unsupported_setting();
1913 break;
1914
1915 case OMX_COMPONENT_GENERATE_HARDWARE_OVERLOAD:
1916 DEBUG_PRINT_ERROR("OMX_COMPONENT_GENERATE_HARDWARE_OVERLOAD");
1917 pThis->omx_report_hw_overload();
1918 break;
1919
1920 default:
1921 break;
1922 }
1923 }
1924 pthread_mutex_lock(&pThis->m_lock);
1925 qsize = pThis->m_cmd_q.m_size;
1926 if (pThis->m_state != OMX_StatePause)
1927 qsize += (pThis->m_ftb_q.m_size + pThis->m_etb_q.m_size);
1928 pthread_mutex_unlock(&pThis->m_lock);
1929 } while (qsize>0);
1930
1931 }
1932
update_resolution(int width,int height,int stride,int scan_lines)1933 int omx_vdec::update_resolution(int width, int height, int stride, int scan_lines)
1934 {
1935 int format_changed = 0;
1936 if ((height != (int)drv_ctx.video_resolution.frame_height) ||
1937 (width != (int)drv_ctx.video_resolution.frame_width)) {
1938 DEBUG_PRINT_HIGH("NOTE_CIF: W/H %d (%d), %d (%d)",
1939 width, drv_ctx.video_resolution.frame_width,
1940 height,drv_ctx.video_resolution.frame_height);
1941 format_changed = 1;
1942 }
1943 drv_ctx.video_resolution.frame_height = height;
1944 drv_ctx.video_resolution.frame_width = width;
1945 drv_ctx.video_resolution.scan_lines = scan_lines;
1946 drv_ctx.video_resolution.stride = stride;
1947
1948 if (!is_down_scalar_enabled) {
1949 rectangle.nLeft = m_extradata_info.output_crop_rect.nLeft;
1950 rectangle.nTop = m_extradata_info.output_crop_rect.nTop;
1951 rectangle.nWidth = m_extradata_info.output_crop_rect.nWidth;
1952 rectangle.nHeight = m_extradata_info.output_crop_rect.nHeight;
1953 }
1954 return format_changed;
1955 }
1956
is_video_session_supported()1957 OMX_ERRORTYPE omx_vdec::is_video_session_supported()
1958 {
1959 if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc",
1960 OMX_MAX_STRINGNAME_SIZE) &&
1961 (m_profile == HIGH_PROFILE || m_profile == MAIN_PROFILE)) {
1962 m_decoder_capability.max_width = 1280;
1963 m_decoder_capability.max_height = 720;
1964 DEBUG_PRINT_HIGH("Set max_width=1280 & max_height=720 for H264 HP/MP");
1965 }
1966
1967 if ((drv_ctx.video_resolution.frame_width *
1968 drv_ctx.video_resolution.frame_height >
1969 m_decoder_capability.max_width *
1970 m_decoder_capability.max_height) ||
1971 (drv_ctx.video_resolution.frame_width*
1972 drv_ctx.video_resolution.frame_height <
1973 m_decoder_capability.min_width *
1974 m_decoder_capability.min_height)) {
1975 DEBUG_PRINT_ERROR(
1976 "Unsupported WxH = (%u)x(%u) supported range is min(%u)x(%u) - max(%u)x(%u)",
1977 drv_ctx.video_resolution.frame_width,
1978 drv_ctx.video_resolution.frame_height,
1979 m_decoder_capability.min_width,
1980 m_decoder_capability.min_height,
1981 m_decoder_capability.max_width,
1982 m_decoder_capability.max_height);
1983 return OMX_ErrorUnsupportedSetting;
1984 }
1985 DEBUG_PRINT_HIGH("video session supported");
1986 return OMX_ErrorNone;
1987 }
1988
log_input_buffers(const char * buffer_addr,int buffer_len)1989 int omx_vdec::log_input_buffers(const char *buffer_addr, int buffer_len)
1990 {
1991 if (m_debug.in_buffer_log && !m_debug.infile) {
1992 if(!strncmp(drv_ctx.kind,"OMX.qcom.video.decoder.mpeg4", OMX_MAX_STRINGNAME_SIZE)) {
1993 snprintf(m_debug.infile_name, OMX_MAX_STRINGNAME_SIZE, "%s/input_dec_%d_%d_%p.m4v",
1994 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
1995 } else if(!strncmp(drv_ctx.kind,"OMX.qcom.video.decoder.mpeg2", OMX_MAX_STRINGNAME_SIZE)) {
1996 snprintf(m_debug.infile_name, OMX_MAX_STRINGNAME_SIZE, "%s/input_dec_%d_%d_%p.mpg", m_debug.log_loc,
1997 drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
1998 } else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.h263", OMX_MAX_STRINGNAME_SIZE)) {
1999 snprintf(m_debug.infile_name, OMX_MAX_STRINGNAME_SIZE, "%s/input_dec_%d_%d_%p.263",
2000 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2001 } else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc", OMX_MAX_STRINGNAME_SIZE) ||
2002 !strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mvc", OMX_MAX_STRINGNAME_SIZE)) {
2003 snprintf(m_debug.infile_name, OMX_MAX_STRINGNAME_SIZE, "%s/input_dec_%d_%d_%p.264",
2004 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2005 } else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.hevc", OMX_MAX_STRINGNAME_SIZE)) {
2006 snprintf(m_debug.infile_name, OMX_MAX_STRINGNAME_SIZE, "%s/input_dec_%d_%d_%p.265",
2007 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2008 } else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vc1", OMX_MAX_STRINGNAME_SIZE)) {
2009 snprintf(m_debug.infile_name, OMX_MAX_STRINGNAME_SIZE, "%s/input_dec_%d_%d_%p.vc1",
2010 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2011 } else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.wmv", OMX_MAX_STRINGNAME_SIZE)) {
2012 snprintf(m_debug.infile_name, OMX_MAX_STRINGNAME_SIZE, "%s/input_dec_%d_%d_%p.vc1",
2013 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2014 } else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp8", OMX_MAX_STRINGNAME_SIZE)) {
2015 snprintf(m_debug.infile_name, OMX_MAX_STRINGNAME_SIZE, "%s/input_dec_%d_%d_%p.ivf",
2016 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2017 } else if(!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp9", OMX_MAX_STRINGNAME_SIZE)) {
2018 snprintf(m_debug.infile_name, OMX_MAX_STRINGNAME_SIZE, "%s/input_dec_%d_%d_%p.ivf",
2019 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2020 } else {
2021 snprintf(m_debug.infile_name, OMX_MAX_STRINGNAME_SIZE, "%s/input_dec_%d_%d_%p.divx",
2022 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2023 }
2024 m_debug.infile = fopen (m_debug.infile_name, "ab");
2025 if (!m_debug.infile) {
2026 DEBUG_PRINT_HIGH("Failed to open input file: %s for logging", m_debug.infile_name);
2027 m_debug.infile_name[0] = '\0';
2028 return -1;
2029 }
2030 if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp8", OMX_MAX_STRINGNAME_SIZE) ||
2031 !strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp9", OMX_MAX_STRINGNAME_SIZE)) {
2032 struct ivf_file_header {
2033 OMX_U8 signature[4]; //='DKIF';
2034 OMX_U8 version ; //= 0;
2035 OMX_U8 headersize ; //= 32;
2036 OMX_U32 FourCC;
2037 OMX_U8 width;
2038 OMX_U8 height;
2039 OMX_U32 rate;
2040 OMX_U32 scale;
2041 OMX_U32 length;
2042 OMX_U8 unused[4];
2043 } file_header;
2044
2045 memset((void *)&file_header,0,sizeof(file_header));
2046 file_header.signature[0] = 'D';
2047 file_header.signature[1] = 'K';
2048 file_header.signature[2] = 'I';
2049 file_header.signature[3] = 'F';
2050 file_header.version = 0;
2051 file_header.headersize = 32;
2052 switch (drv_ctx.decoder_format) {
2053 case VDEC_CODECTYPE_VP8:
2054 file_header.FourCC = 0x30385056;
2055 break;
2056 case VDEC_CODECTYPE_VP9:
2057 file_header.FourCC = 0x30395056;
2058 break;
2059 default:
2060 DEBUG_PRINT_ERROR("unsupported format for VP8/VP9");
2061 break;
2062 }
2063 fwrite((const char *)&file_header,
2064 sizeof(file_header),1,m_debug.infile);
2065 }
2066 }
2067 if (m_debug.infile && buffer_addr && buffer_len) {
2068 if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp8", OMX_MAX_STRINGNAME_SIZE) ||
2069 !strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp9", OMX_MAX_STRINGNAME_SIZE)) {
2070 struct vpx_ivf_frame_header {
2071 OMX_U32 framesize;
2072 OMX_U32 timestamp_lo;
2073 OMX_U32 timestamp_hi;
2074 } vpx_frame_header;
2075 vpx_frame_header.framesize = buffer_len;
2076 /* Currently FW doesn't use timestamp values */
2077 vpx_frame_header.timestamp_lo = 0;
2078 vpx_frame_header.timestamp_hi = 0;
2079 fwrite((const char *)&vpx_frame_header,
2080 sizeof(vpx_frame_header),1,m_debug.infile);
2081 }
2082 fwrite(buffer_addr, buffer_len, 1, m_debug.infile);
2083 }
2084 return 0;
2085 }
2086
log_output_buffers(OMX_BUFFERHEADERTYPE * buffer)2087 int omx_vdec::log_output_buffers(OMX_BUFFERHEADERTYPE *buffer) {
2088 int buf_index = 0;
2089 char *temp = NULL;
2090
2091 if (!(m_debug.out_buffer_log || m_debug.out_meta_buffer_log) || !buffer || !buffer->nFilledLen)
2092 return 0;
2093
2094 if (m_debug.out_buffer_log && !m_debug.outfile) {
2095 snprintf(m_debug.outfile_name, OMX_MAX_STRINGNAME_SIZE, "%s/output_%d_%d_%p.yuv",
2096 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2097 m_debug.outfile = fopen (m_debug.outfile_name, "ab");
2098 if (!m_debug.outfile) {
2099 DEBUG_PRINT_HIGH("Failed to open output file: %s for logging", m_debug.log_loc);
2100 m_debug.outfile_name[0] = '\0';
2101 return -1;
2102 }
2103 }
2104
2105 if (m_debug.out_meta_buffer_log && !m_debug.out_ymeta_file && !m_debug.out_uvmeta_file) {
2106 snprintf(m_debug.out_ymetafile_name, OMX_MAX_STRINGNAME_SIZE, "%s/output_%d_%d_%p.ymeta",
2107 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2108 snprintf(m_debug.out_uvmetafile_name, OMX_MAX_STRINGNAME_SIZE, "%s/output_%d_%d_%p.uvmeta",
2109 m_debug.log_loc, drv_ctx.video_resolution.frame_width, drv_ctx.video_resolution.frame_height, this);
2110 m_debug.out_ymeta_file = fopen (m_debug.out_ymetafile_name, "ab");
2111 m_debug.out_uvmeta_file = fopen (m_debug.out_uvmetafile_name, "ab");
2112 if (!m_debug.out_ymeta_file || !m_debug.out_uvmeta_file) {
2113 DEBUG_PRINT_HIGH("Failed to open output y/uv meta file: %s for logging", m_debug.log_loc);
2114 m_debug.out_ymetafile_name[0] = '\0';
2115 m_debug.out_uvmetafile_name[0] = '\0';
2116 return -1;
2117 }
2118 }
2119
2120 buf_index = buffer - m_out_mem_ptr;
2121 temp = (char *)drv_ctx.ptr_outputbuffer[buf_index].bufferaddr;
2122
2123 if (drv_ctx.output_format == VDEC_YUV_FORMAT_NV12_UBWC ||
2124 drv_ctx.output_format == VDEC_YUV_FORMAT_NV12_TP10_UBWC) {
2125 DEBUG_PRINT_HIGH("Logging UBWC yuv width/height(%u/%u)",
2126 drv_ctx.video_resolution.frame_width,
2127 drv_ctx.video_resolution.frame_height);
2128
2129 if (m_debug.outfile)
2130 fwrite(temp, buffer->nFilledLen, 1, m_debug.outfile);
2131
2132 if (m_debug.out_ymeta_file && m_debug.out_uvmeta_file) {
2133 unsigned int width = 0, height = 0;
2134 unsigned int y_plane, y_meta_plane;
2135 int y_stride = 0, y_sclines = 0;
2136 int y_meta_stride = 0, y_meta_scanlines = 0, uv_meta_stride = 0, uv_meta_scanlines = 0;
2137 int color_fmt = (drv_ctx.output_format== VDEC_YUV_FORMAT_NV12_UBWC)? COLOR_FMT_NV12_UBWC: COLOR_FMT_NV12_BPP10_UBWC;
2138 int i;
2139 int bytes_written = 0;
2140
2141 width = drv_ctx.video_resolution.frame_width;
2142 height = drv_ctx.video_resolution.frame_height;
2143 y_meta_stride = VENUS_Y_META_STRIDE(color_fmt, width);
2144 y_meta_scanlines = VENUS_Y_META_SCANLINES(color_fmt, height);
2145 y_stride = VENUS_Y_STRIDE(color_fmt, width);
2146 y_sclines = VENUS_Y_SCANLINES(color_fmt, height);
2147 uv_meta_stride = VENUS_UV_META_STRIDE(color_fmt, width);
2148 uv_meta_scanlines = VENUS_UV_META_SCANLINES(color_fmt, height);
2149
2150 y_meta_plane = MSM_MEDIA_ALIGN(y_meta_stride * y_meta_scanlines, 4096);
2151 y_plane = MSM_MEDIA_ALIGN(y_stride * y_sclines, 4096);
2152
2153 temp = (char *)drv_ctx.ptr_outputbuffer[buf_index].bufferaddr;
2154 for (i = 0; i < y_meta_scanlines; i++) {
2155 bytes_written = fwrite(temp, y_meta_stride, 1, m_debug.out_ymeta_file);
2156 temp += y_meta_stride;
2157 }
2158
2159 temp = (char *)drv_ctx.ptr_outputbuffer[buf_index].bufferaddr + y_meta_plane + y_plane;
2160 for(i = 0; i < uv_meta_scanlines; i++) {
2161 bytes_written += fwrite(temp, uv_meta_stride, 1, m_debug.out_uvmeta_file);
2162 temp += uv_meta_stride;
2163 }
2164 }
2165 } else if (m_debug.outfile && drv_ctx.output_format == VDEC_YUV_FORMAT_NV12) {
2166 int stride = drv_ctx.video_resolution.stride;
2167 int scanlines = drv_ctx.video_resolution.scan_lines;
2168 if (m_smoothstreaming_mode) {
2169 stride = drv_ctx.video_resolution.frame_width;
2170 scanlines = drv_ctx.video_resolution.frame_height;
2171 stride = (stride + DEFAULT_WIDTH_ALIGNMENT - 1) & (~(DEFAULT_WIDTH_ALIGNMENT - 1));
2172 scanlines = (scanlines + DEFAULT_HEIGHT_ALIGNMENT - 1) & (~(DEFAULT_HEIGHT_ALIGNMENT - 1));
2173 }
2174 unsigned i;
2175 DEBUG_PRINT_HIGH("Logging width/height(%u/%u) stride/scanlines(%u/%u)",
2176 drv_ctx.video_resolution.frame_width,
2177 drv_ctx.video_resolution.frame_height, stride, scanlines);
2178 int bytes_written = 0;
2179 for (i = 0; i < drv_ctx.video_resolution.frame_height; i++) {
2180 bytes_written = fwrite(temp, drv_ctx.video_resolution.frame_width, 1, m_debug.outfile);
2181 temp += stride;
2182 }
2183 temp = (char *)drv_ctx.ptr_outputbuffer[buf_index].bufferaddr + stride * scanlines;
2184 int stride_c = stride;
2185 for(i = 0; i < drv_ctx.video_resolution.frame_height/2; i++) {
2186 bytes_written += fwrite(temp, drv_ctx.video_resolution.frame_width, 1, m_debug.outfile);
2187 temp += stride_c;
2188 }
2189 }
2190 return 0;
2191 }
2192
2193 /* ======================================================================
2194 FUNCTION
2195 omx_vdec::ComponentInit
2196
2197 DESCRIPTION
2198 Initialize the component.
2199
2200 PARAMETERS
2201 ctxt -- Context information related to the self.
2202 id -- Event identifier. This could be any of the following:
2203 1. Command completion event
2204 2. Buffer done callback event
2205 3. Frame done callback event
2206
2207 RETURN VALUE
2208 None.
2209
2210 ========================================================================== */
component_init(OMX_STRING role)2211 OMX_ERRORTYPE omx_vdec::component_init(OMX_STRING role)
2212 {
2213
2214 OMX_ERRORTYPE eRet = OMX_ErrorNone;
2215 struct v4l2_fmtdesc fdesc;
2216 struct v4l2_format fmt;
2217 struct v4l2_requestbuffers bufreq;
2218 struct v4l2_control control;
2219 struct v4l2_frmsizeenum frmsize;
2220 unsigned int alignment = 0,buffer_size = 0;
2221 int fds[2];
2222 int r,ret=0;
2223 bool codec_ambiguous = false;
2224 OMX_STRING device_name = (OMX_STRING)"/dev/video32";
2225 char property_value[PROPERTY_VALUE_MAX] = {0};
2226 FILE *soc_file = NULL;
2227 char buffer[10];
2228
2229 #ifdef _ANDROID_
2230 char platform_name[PROPERTY_VALUE_MAX];
2231 property_get("ro.board.platform", platform_name, "0");
2232 if (!strncmp(platform_name, "msm8610", 7)) {
2233 device_name = (OMX_STRING)"/dev/video/q6_dec";
2234 is_q6_platform = true;
2235 maxSmoothStreamingWidth = 1280;
2236 maxSmoothStreamingHeight = 720;
2237 }
2238 #endif
2239
2240 is_thulium_v1 = false;
2241 soc_file = fopen("/sys/devices/soc0/soc_id", "r");
2242 if (soc_file) {
2243 fread(buffer, 1, 4, soc_file);
2244 fclose(soc_file);
2245 if (atoi(buffer) == 246) {
2246 soc_file = fopen("/sys/devices/soc0/revision", "r");
2247 if (soc_file) {
2248 fread(buffer, 1, 4, soc_file);
2249 fclose(soc_file);
2250 if (atoi(buffer) == 1) {
2251 is_thulium_v1 = true;
2252 DEBUG_PRINT_HIGH("is_thulium_v1 = TRUE");
2253 }
2254 }
2255 }
2256 }
2257
2258 #ifdef _ANDROID_
2259 /*
2260 * turn off frame parsing for Android by default.
2261 * Clients may configure OMX_QCOM_FramePacking_Arbitrary to enable this mode
2262 */
2263 arbitrary_bytes = false;
2264 property_get("vendor.vidc.dec.debug.arbitrarybytes.mode", property_value, "0");
2265 if (atoi(property_value)) {
2266 DEBUG_PRINT_HIGH("arbitrary_bytes mode enabled via property command");
2267 arbitrary_bytes = true;
2268 }
2269 #endif
2270
2271 if (!strncmp(role, "OMX.qcom.video.decoder.avc.secure",
2272 OMX_MAX_STRINGNAME_SIZE)) {
2273 secure_mode = true;
2274 arbitrary_bytes = false;
2275 role = (OMX_STRING)"OMX.qcom.video.decoder.avc";
2276 } else if (!strncmp(role, "OMX.qcom.video.decoder.mpeg2.secure",
2277 OMX_MAX_STRINGNAME_SIZE)) {
2278 secure_mode = true;
2279 arbitrary_bytes = false;
2280 role = (OMX_STRING)"OMX.qcom.video.decoder.mpeg2";
2281 } else if (!strncmp(role, "OMX.qcom.video.decoder.hevc.secure",
2282 OMX_MAX_STRINGNAME_SIZE)) {
2283 secure_mode = true;
2284 arbitrary_bytes = false;
2285 role = (OMX_STRING)"OMX.qcom.video.decoder.hevc";
2286 } else if (!strncmp(role, "OMX.qcom.video.decoder.vc1.secure",
2287 OMX_MAX_STRINGNAME_SIZE)) {
2288 secure_mode = true;
2289 arbitrary_bytes = false;
2290 role = (OMX_STRING)"OMX.qcom.video.decoder.vc1";
2291 } else if (!strncmp(role, "OMX.qcom.video.decoder.wmv.secure",
2292 OMX_MAX_STRINGNAME_SIZE)) {
2293 secure_mode = true;
2294 arbitrary_bytes = false;
2295 role = (OMX_STRING)"OMX.qcom.video.decoder.wmv";
2296 } else if (!strncmp(role, "OMX.qcom.video.decoder.mpeg4.secure",
2297 OMX_MAX_STRINGNAME_SIZE)) {
2298 secure_mode = true;
2299 arbitrary_bytes = false;
2300 role = (OMX_STRING)"OMX.qcom.video.decoder.mpeg4";
2301 } else if (!strncmp(role, "OMX.qcom.video.decoder.vp9.secure",
2302 OMX_MAX_STRINGNAME_SIZE)) {
2303 secure_mode = true;
2304 arbitrary_bytes = false;
2305 role = (OMX_STRING)"OMX.qcom.video.decoder.vp9";
2306 }
2307 else if (!strncmp(role, "OMX.qcom.video.decoder.vp8.secure",
2308 OMX_MAX_STRINGNAME_SIZE)) {
2309 secure_mode = true;
2310 arbitrary_bytes = false;
2311 role = (OMX_STRING)"OMX.qcom.video.decoder.vp8";
2312 }
2313
2314 drv_ctx.video_driver_fd = open(device_name, O_RDWR);
2315
2316 DEBUG_PRINT_INFO("component_init: %s : fd=%d", role, drv_ctx.video_driver_fd);
2317
2318 if (drv_ctx.video_driver_fd < 0) {
2319 DEBUG_PRINT_ERROR("Omx_vdec::Comp Init Returning failure, errno %d", errno);
2320 return OMX_ErrorInsufficientResources;
2321 }
2322 drv_ctx.frame_rate.fps_numerator = DEFAULT_FPS;
2323 drv_ctx.frame_rate.fps_denominator = 1;
2324 operating_frame_rate = DEFAULT_FPS;
2325 m_poll_efd = eventfd(0, 0);
2326 if (m_poll_efd < 0) {
2327 DEBUG_PRINT_ERROR("Failed to create event fd(%s)", strerror(errno));
2328 return OMX_ErrorInsufficientResources;
2329 }
2330 ret = subscribe_to_events(drv_ctx.video_driver_fd);
2331 if (!ret) {
2332 async_thread_created = true;
2333 ret = pthread_create(&async_thread_id,0,async_message_thread,this);
2334 }
2335 if (ret) {
2336 DEBUG_PRINT_ERROR("Failed to create async_message_thread");
2337 async_thread_created = false;
2338 return OMX_ErrorInsufficientResources;
2339 }
2340
2341 #ifdef OUTPUT_EXTRADATA_LOG
2342 outputExtradataFile = fopen (output_extradata_filename, "ab");
2343 #endif
2344
2345 // Copy the role information which provides the decoder kind
2346 strlcpy(drv_ctx.kind,role,128);
2347
2348 if (!strncmp(drv_ctx.kind,"OMX.qcom.video.decoder.mpeg4",\
2349 OMX_MAX_STRINGNAME_SIZE)) {
2350 strlcpy((char *)m_cRole, "video_decoder.mpeg4",\
2351 OMX_MAX_STRINGNAME_SIZE);
2352 drv_ctx.timestamp_adjust = true;
2353 drv_ctx.decoder_format = VDEC_CODECTYPE_MPEG4;
2354 eCompressionFormat = OMX_VIDEO_CodingMPEG4;
2355 output_capability=V4L2_PIX_FMT_MPEG4;
2356 /*Initialize Start Code for MPEG4*/
2357 codec_type_parse = CODEC_TYPE_MPEG4;
2358 m_frame_parser.init_start_codes(codec_type_parse);
2359 } else if (!strncmp(drv_ctx.kind,"OMX.qcom.video.decoder.mpeg2",\
2360 OMX_MAX_STRINGNAME_SIZE)) {
2361 strlcpy((char *)m_cRole, "video_decoder.mpeg2",\
2362 OMX_MAX_STRINGNAME_SIZE);
2363 drv_ctx.decoder_format = VDEC_CODECTYPE_MPEG2;
2364 output_capability = V4L2_PIX_FMT_MPEG2;
2365 eCompressionFormat = OMX_VIDEO_CodingMPEG2;
2366 /*Initialize Start Code for MPEG2*/
2367 codec_type_parse = CODEC_TYPE_MPEG2;
2368 m_frame_parser.init_start_codes(codec_type_parse);
2369 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.h263",\
2370 OMX_MAX_STRINGNAME_SIZE)) {
2371 strlcpy((char *)m_cRole, "video_decoder.h263",OMX_MAX_STRINGNAME_SIZE);
2372 DEBUG_PRINT_LOW("H263 Decoder selected");
2373 drv_ctx.decoder_format = VDEC_CODECTYPE_H263;
2374 eCompressionFormat = OMX_VIDEO_CodingH263;
2375 output_capability = V4L2_PIX_FMT_H263;
2376 codec_type_parse = CODEC_TYPE_H263;
2377 m_frame_parser.init_start_codes(codec_type_parse);
2378 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx311",\
2379 OMX_MAX_STRINGNAME_SIZE)) {
2380 strlcpy((char *)m_cRole, "video_decoder.divx",OMX_MAX_STRINGNAME_SIZE);
2381 DEBUG_PRINT_LOW ("DIVX 311 Decoder selected");
2382 drv_ctx.decoder_format = VDEC_CODECTYPE_DIVX_3;
2383 output_capability = V4L2_PIX_FMT_DIVX_311;
2384 eCompressionFormat = (OMX_VIDEO_CODINGTYPE)QOMX_VIDEO_CodingDivx;
2385 codec_type_parse = CODEC_TYPE_DIVX;
2386 m_frame_parser.init_start_codes(codec_type_parse);
2387
2388 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx4",\
2389 OMX_MAX_STRINGNAME_SIZE)) {
2390 strlcpy((char *)m_cRole, "video_decoder.divx",OMX_MAX_STRINGNAME_SIZE);
2391 DEBUG_PRINT_ERROR ("DIVX 4 Decoder selected");
2392 drv_ctx.decoder_format = VDEC_CODECTYPE_DIVX_4;
2393 output_capability = V4L2_PIX_FMT_DIVX;
2394 eCompressionFormat = (OMX_VIDEO_CODINGTYPE)QOMX_VIDEO_CodingDivx;
2395 codec_type_parse = CODEC_TYPE_DIVX;
2396 codec_ambiguous = true;
2397 m_frame_parser.init_start_codes(codec_type_parse);
2398
2399 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx",\
2400 OMX_MAX_STRINGNAME_SIZE)) {
2401 strlcpy((char *)m_cRole, "video_decoder.divx",OMX_MAX_STRINGNAME_SIZE);
2402 DEBUG_PRINT_ERROR ("DIVX 5/6 Decoder selected");
2403 drv_ctx.decoder_format = VDEC_CODECTYPE_DIVX_6;
2404 output_capability = V4L2_PIX_FMT_DIVX;
2405 eCompressionFormat = (OMX_VIDEO_CODINGTYPE)QOMX_VIDEO_CodingDivx;
2406 codec_type_parse = CODEC_TYPE_DIVX;
2407 codec_ambiguous = true;
2408 m_frame_parser.init_start_codes(codec_type_parse);
2409
2410 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc",\
2411 OMX_MAX_STRINGNAME_SIZE)) {
2412 strlcpy((char *)m_cRole, "video_decoder.avc",OMX_MAX_STRINGNAME_SIZE);
2413 drv_ctx.decoder_format = VDEC_CODECTYPE_H264;
2414 output_capability=V4L2_PIX_FMT_H264;
2415 eCompressionFormat = OMX_VIDEO_CodingAVC;
2416 codec_type_parse = CODEC_TYPE_H264;
2417 m_frame_parser.init_start_codes(codec_type_parse);
2418 m_frame_parser.init_nal_length(nal_length);
2419 if (is_thulium_v1) {
2420 arbitrary_bytes = true;
2421 DEBUG_PRINT_HIGH("Enable arbitrary_bytes for h264");
2422 }
2423 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mvc",\
2424 OMX_MAX_STRINGNAME_SIZE)) {
2425 strlcpy((char *)m_cRole, "video_decoder.mvc", OMX_MAX_STRINGNAME_SIZE);
2426 drv_ctx.decoder_format = VDEC_CODECTYPE_MVC;
2427 output_capability = V4L2_PIX_FMT_H264_MVC;
2428 eCompressionFormat = (OMX_VIDEO_CODINGTYPE)QOMX_VIDEO_CodingMVC;
2429 codec_type_parse = CODEC_TYPE_H264;
2430 m_frame_parser.init_start_codes(codec_type_parse);
2431 m_frame_parser.init_nal_length(nal_length);
2432 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.hevc",\
2433 OMX_MAX_STRINGNAME_SIZE)) {
2434 strlcpy((char *)m_cRole, "video_decoder.hevc",OMX_MAX_STRINGNAME_SIZE);
2435 drv_ctx.decoder_format = VDEC_CODECTYPE_HEVC;
2436 output_capability = V4L2_PIX_FMT_HEVC;
2437 eCompressionFormat = (OMX_VIDEO_CODINGTYPE)QOMX_VIDEO_CodingHevc;
2438 codec_type_parse = CODEC_TYPE_HEVC;
2439 m_frame_parser.init_start_codes(codec_type_parse);
2440 m_frame_parser.init_nal_length(nal_length);
2441 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vc1",\
2442 OMX_MAX_STRINGNAME_SIZE)) {
2443 strlcpy((char *)m_cRole, "video_decoder.vc1",OMX_MAX_STRINGNAME_SIZE);
2444 drv_ctx.decoder_format = VDEC_CODECTYPE_VC1;
2445 eCompressionFormat = OMX_VIDEO_CodingWMV;
2446 codec_type_parse = CODEC_TYPE_VC1;
2447 output_capability = V4L2_PIX_FMT_VC1_ANNEX_G;
2448 m_frame_parser.init_start_codes(codec_type_parse);
2449 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.wmv",\
2450 OMX_MAX_STRINGNAME_SIZE)) {
2451 strlcpy((char *)m_cRole, "video_decoder.vc1",OMX_MAX_STRINGNAME_SIZE);
2452 drv_ctx.decoder_format = VDEC_CODECTYPE_VC1_RCV;
2453 eCompressionFormat = OMX_VIDEO_CodingWMV;
2454 codec_type_parse = CODEC_TYPE_VC1;
2455 output_capability = V4L2_PIX_FMT_VC1_ANNEX_L;
2456 m_frame_parser.init_start_codes(codec_type_parse);
2457 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp8", \
2458 OMX_MAX_STRINGNAME_SIZE)) {
2459 strlcpy((char *)m_cRole, "video_decoder.vp8",OMX_MAX_STRINGNAME_SIZE);
2460 drv_ctx.decoder_format = VDEC_CODECTYPE_VP8;
2461 output_capability = V4L2_PIX_FMT_VP8;
2462 eCompressionFormat = OMX_VIDEO_CodingVP8;
2463 codec_type_parse = CODEC_TYPE_VP8;
2464 arbitrary_bytes = false;
2465 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp9", \
2466 OMX_MAX_STRINGNAME_SIZE)) {
2467 strlcpy((char *)m_cRole, "video_decoder.vp9",OMX_MAX_STRINGNAME_SIZE);
2468 drv_ctx.decoder_format = VDEC_CODECTYPE_VP9;
2469 output_capability = V4L2_PIX_FMT_VP9;
2470 eCompressionFormat = OMX_VIDEO_CodingVP9;
2471 codec_type_parse = CODEC_TYPE_VP9;
2472 arbitrary_bytes = false;
2473 } else {
2474 DEBUG_PRINT_ERROR("ERROR:Unknown Component");
2475 eRet = OMX_ErrorInvalidComponentName;
2476 }
2477
2478 if (eRet == OMX_ErrorNone) {
2479 OMX_COLOR_FORMATTYPE dest_color_format;
2480 if (m_disable_ubwc_mode) {
2481 drv_ctx.output_format = VDEC_YUV_FORMAT_NV12;
2482 } else {
2483 drv_ctx.output_format = VDEC_YUV_FORMAT_NV12_UBWC;
2484 }
2485 if (eCompressionFormat == (OMX_VIDEO_CODINGTYPE)QOMX_VIDEO_CodingMVC)
2486 dest_color_format = (OMX_COLOR_FORMATTYPE)
2487 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mMultiView;
2488 else
2489 dest_color_format = (OMX_COLOR_FORMATTYPE)
2490 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m;
2491 if (!client_buffers.set_color_format(dest_color_format)) {
2492 DEBUG_PRINT_ERROR("Setting color format failed");
2493 eRet = OMX_ErrorInsufficientResources;
2494 }
2495
2496 dpb_bit_depth = MSM_VIDC_BIT_DEPTH_8;
2497 m_progressive = MSM_VIDC_PIC_STRUCT_PROGRESSIVE;
2498
2499 if (m_disable_ubwc_mode) {
2500 capture_capability = V4L2_PIX_FMT_NV12;
2501 } else {
2502 capture_capability = V4L2_PIX_FMT_NV12_UBWC;
2503 }
2504
2505 struct v4l2_capability cap;
2506 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_QUERYCAP, &cap);
2507 if (ret) {
2508 DEBUG_PRINT_ERROR("Failed to query capabilities");
2509 /*TODO: How to handle this case */
2510 } else {
2511 DEBUG_PRINT_LOW("Capabilities: driver_name = %s, card = %s, bus_info = %s,"
2512 " version = %d, capabilities = %x", cap.driver, cap.card,
2513 cap.bus_info, cap.version, cap.capabilities);
2514 }
2515 ret=0;
2516 fdesc.type=V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
2517 fdesc.index=0;
2518 while (ioctl(drv_ctx.video_driver_fd, VIDIOC_ENUM_FMT, &fdesc) == 0) {
2519 DEBUG_PRINT_HIGH("fmt: description: %s, fmt: %x, flags = %x", fdesc.description,
2520 fdesc.pixelformat, fdesc.flags);
2521 fdesc.index++;
2522 }
2523 fdesc.type=V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
2524 fdesc.index=0;
2525 while (ioctl(drv_ctx.video_driver_fd, VIDIOC_ENUM_FMT, &fdesc) == 0) {
2526
2527 DEBUG_PRINT_HIGH("fmt: description: %s, fmt: %x, flags = %x", fdesc.description,
2528 fdesc.pixelformat, fdesc.flags);
2529 fdesc.index++;
2530 }
2531 m_extradata_info.output_crop_rect.nLeft = 0;
2532 m_extradata_info.output_crop_rect.nTop = 0;
2533 m_extradata_info.output_crop_rect.nWidth = 320;
2534 m_extradata_info.output_crop_rect.nHeight = 240;
2535 update_resolution(320, 240, 320, 240);
2536
2537 fmt.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
2538 fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
2539 fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
2540 fmt.fmt.pix_mp.pixelformat = output_capability;
2541 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
2542 if (ret) {
2543 /*TODO: How to handle this case */
2544 DEBUG_PRINT_ERROR("Failed to set format on output port");
2545 return OMX_ErrorInsufficientResources;
2546 }
2547 DEBUG_PRINT_HIGH("Set Format was successful");
2548 if (codec_ambiguous) {
2549 if (output_capability == V4L2_PIX_FMT_DIVX) {
2550 struct v4l2_control divx_ctrl;
2551
2552 if (drv_ctx.decoder_format == VDEC_CODECTYPE_DIVX_4) {
2553 divx_ctrl.value = V4L2_MPEG_VIDC_VIDEO_DIVX_FORMAT_4;
2554 } else if (drv_ctx.decoder_format == VDEC_CODECTYPE_DIVX_5) {
2555 divx_ctrl.value = V4L2_MPEG_VIDC_VIDEO_DIVX_FORMAT_5;
2556 } else {
2557 divx_ctrl.value = V4L2_MPEG_VIDC_VIDEO_DIVX_FORMAT_6;
2558 }
2559
2560 divx_ctrl.id = V4L2_CID_MPEG_VIDC_VIDEO_DIVX_FORMAT;
2561 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &divx_ctrl);
2562 if (ret) {
2563 DEBUG_PRINT_ERROR("Failed to set divx version");
2564 }
2565 } else {
2566 DEBUG_PRINT_ERROR("Codec should not be ambiguous");
2567 }
2568 }
2569
2570 property_get("persist.vendor.vidc.dec.conceal_color", property_value, DEFAULT_CONCEAL_COLOR);
2571 m_conceal_color= atoi(property_value);
2572 DEBUG_PRINT_HIGH("trying to set 0x%u as conceal color\n", (unsigned int)m_conceal_color);
2573 control.id = V4L2_CID_MPEG_VIDC_VIDEO_CONCEAL_COLOR;
2574 control.value = m_conceal_color;
2575 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control);
2576 if (ret) {
2577 DEBUG_PRINT_ERROR("Failed to set conceal color %d\n", ret);
2578 }
2579
2580 //Get the hardware capabilities
2581 memset((void *)&frmsize,0,sizeof(frmsize));
2582 frmsize.index = 0;
2583 frmsize.pixel_format = output_capability;
2584 ret = ioctl(drv_ctx.video_driver_fd,
2585 VIDIOC_ENUM_FRAMESIZES, &frmsize);
2586 if (ret || frmsize.type != V4L2_FRMSIZE_TYPE_STEPWISE) {
2587 DEBUG_PRINT_ERROR("Failed to get framesizes");
2588 return OMX_ErrorHardware;
2589 }
2590
2591 if (frmsize.type == V4L2_FRMSIZE_TYPE_STEPWISE) {
2592 m_decoder_capability.min_width = frmsize.stepwise.min_width;
2593 m_decoder_capability.max_width = frmsize.stepwise.max_width;
2594 m_decoder_capability.min_height = frmsize.stepwise.min_height;
2595 m_decoder_capability.max_height = frmsize.stepwise.max_height;
2596 }
2597
2598 memset(&fmt, 0x0, sizeof(struct v4l2_format));
2599 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
2600 fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
2601 fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
2602 fmt.fmt.pix_mp.pixelformat = capture_capability;
2603 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
2604 if (ret) {
2605 /*TODO: How to handle this case */
2606 DEBUG_PRINT_ERROR("Failed to set format on capture port");
2607 }
2608 memset(&framesize, 0, sizeof(OMX_FRAMESIZETYPE));
2609 framesize.nWidth = drv_ctx.video_resolution.frame_width;
2610 framesize.nHeight = drv_ctx.video_resolution.frame_height;
2611
2612 memset(&rectangle, 0, sizeof(OMX_CONFIG_RECTTYPE));
2613 rectangle.nWidth = drv_ctx.video_resolution.frame_width;
2614 rectangle.nHeight = drv_ctx.video_resolution.frame_height;
2615
2616 DEBUG_PRINT_HIGH("Set Format was successful");
2617 if (secure_mode) {
2618 control.id = V4L2_CID_MPEG_VIDC_VIDEO_SECURE;
2619 control.value = 1;
2620 DEBUG_PRINT_LOW("Omx_vdec:: calling to open secure device %d", ret);
2621 ret=ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL,&control);
2622 if (ret) {
2623 DEBUG_PRINT_ERROR("Omx_vdec:: Unable to open secure device %d", ret);
2624 return OMX_ErrorInsufficientResources;
2625 }
2626 }
2627 if (output_capability == V4L2_PIX_FMT_H264_MVC) {
2628 control.id = V4L2_CID_MPEG_VIDC_VIDEO_MVC_BUFFER_LAYOUT;
2629 control.value = V4L2_MPEG_VIDC_VIDEO_MVC_TOP_BOTTOM;
2630 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control);
2631 if (ret) {
2632 DEBUG_PRINT_ERROR("Failed to set MVC buffer layout");
2633 return OMX_ErrorInsufficientResources;
2634 }
2635 }
2636
2637 if (is_thulium_v1) {
2638 eRet = enable_smoothstreaming();
2639 if (eRet != OMX_ErrorNone) {
2640 DEBUG_PRINT_ERROR("Failed to enable smooth streaming on driver");
2641 return eRet;
2642 }
2643 }
2644
2645 /*Get the Buffer requirements for input and output ports*/
2646 drv_ctx.ip_buf.buffer_type = VDEC_BUFFER_TYPE_INPUT;
2647 drv_ctx.op_buf.buffer_type = VDEC_BUFFER_TYPE_OUTPUT;
2648
2649 if (secure_mode) {
2650 drv_ctx.op_buf.alignment = SECURE_ALIGN;
2651 drv_ctx.ip_buf.alignment = SECURE_ALIGN;
2652 } else {
2653 drv_ctx.op_buf.alignment = SZ_4K;
2654 drv_ctx.ip_buf.alignment = SZ_4K;
2655 }
2656
2657 drv_ctx.interlace = VDEC_InterlaceFrameProgressive;
2658 drv_ctx.extradata = 0;
2659 drv_ctx.picture_order = VDEC_ORDER_DISPLAY;
2660 control.id = V4L2_CID_MPEG_VIDC_VIDEO_OUTPUT_ORDER;
2661 control.value = V4L2_MPEG_VIDC_VIDEO_OUTPUT_ORDER_DISPLAY;
2662 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control);
2663 drv_ctx.idr_only_decoding = 0;
2664
2665 #ifdef _ANDROID_
2666 property_get("vendor.vidc.dec.enable.downscalar",property_value,"0");
2667 if (atoi(property_value)) {
2668 m_enable_downscalar = atoi(property_value);
2669 property_get("vendor.vidc.dec.downscalar_width",property_value,"0");
2670 if (atoi(property_value)) {
2671 m_downscalar_width = atoi(property_value);
2672 }
2673 property_get("vendor.vidc.dec.downscalar_height",property_value,"0");
2674 if (atoi(property_value)) {
2675 m_downscalar_height = atoi(property_value);
2676 }
2677
2678 if (m_downscalar_width < m_decoder_capability.min_width ||
2679 m_downscalar_height < m_decoder_capability.min_height) {
2680 m_downscalar_width = 0;
2681 m_downscalar_height = 0;
2682 }
2683
2684 DEBUG_PRINT_LOW("Downscaler configured WxH %dx%d\n",
2685 m_downscalar_width, m_downscalar_height);
2686 }
2687 property_get("vendor.vidc.disable.split.mode",property_value,"0");
2688 m_disable_split_mode = atoi(property_value);
2689 DEBUG_PRINT_HIGH("split mode is %s", m_disable_split_mode ? "disabled" : "enabled");
2690 #endif
2691 m_state = OMX_StateLoaded;
2692 #ifdef DEFAULT_EXTRADATA
2693 enable_extradata(DEFAULT_EXTRADATA, true, true);
2694 #endif
2695 eRet = get_buffer_req(&drv_ctx.ip_buf);
2696 DEBUG_PRINT_HIGH("Input Buffer Size =%u",(unsigned int)drv_ctx.ip_buf.buffer_size);
2697 get_buffer_req(&drv_ctx.op_buf);
2698 if (drv_ctx.decoder_format == VDEC_CODECTYPE_H264 ||
2699 drv_ctx.decoder_format == VDEC_CODECTYPE_HEVC ||
2700 drv_ctx.decoder_format == VDEC_CODECTYPE_MVC) {
2701 h264_scratch.nAllocLen = drv_ctx.ip_buf.buffer_size;
2702 h264_scratch.pBuffer = (OMX_U8 *)malloc (drv_ctx.ip_buf.buffer_size);
2703 h264_scratch.nFilledLen = 0;
2704 h264_scratch.nOffset = 0;
2705
2706 if (h264_scratch.pBuffer == NULL) {
2707 DEBUG_PRINT_ERROR("h264_scratch.pBuffer Allocation failed ");
2708 return OMX_ErrorInsufficientResources;
2709 }
2710 }
2711 if (drv_ctx.decoder_format == VDEC_CODECTYPE_H264 ||
2712 drv_ctx.decoder_format == VDEC_CODECTYPE_MVC) {
2713 if (m_frame_parser.mutils == NULL) {
2714 m_frame_parser.mutils = new H264_Utils();
2715 if (m_frame_parser.mutils == NULL) {
2716 DEBUG_PRINT_ERROR("parser utils Allocation failed ");
2717 eRet = OMX_ErrorInsufficientResources;
2718 } else {
2719 m_frame_parser.mutils->initialize_frame_checking_environment();
2720 m_frame_parser.mutils->allocate_rbsp_buffer (drv_ctx.ip_buf.buffer_size);
2721 }
2722 }
2723
2724 h264_parser = new h264_stream_parser();
2725 if (!h264_parser) {
2726 DEBUG_PRINT_ERROR("ERROR: H264 parser allocation failed!");
2727 eRet = OMX_ErrorInsufficientResources;
2728 }
2729 }
2730
2731 if (pipe(fds)) {
2732 DEBUG_PRINT_ERROR("pipe creation failed");
2733 eRet = OMX_ErrorInsufficientResources;
2734 } else {
2735 m_pipe_in = fds[0];
2736 m_pipe_out = fds[1];
2737 msg_thread_created = true;
2738 r = pthread_create(&msg_thread_id,0,message_thread_dec,this);
2739
2740 if (r < 0) {
2741 DEBUG_PRINT_ERROR("component_init(): message_thread_dec creation failed");
2742 msg_thread_created = false;
2743 eRet = OMX_ErrorInsufficientResources;
2744 }
2745 }
2746 }
2747
2748 {
2749 VendorExtensionStore *extStore = const_cast<VendorExtensionStore *>(&mVendorExtensionStore);
2750 init_vendor_extensions(*extStore);
2751 mVendorExtensionStore.dumpExtensions((const char *)role);
2752 }
2753
2754 if (eRet != OMX_ErrorNone) {
2755 DEBUG_PRINT_ERROR("Component Init Failed");
2756 } else {
2757 DEBUG_PRINT_INFO("omx_vdec::component_init() success : fd=%d",
2758 drv_ctx.video_driver_fd);
2759 }
2760 //memset(&h264_mv_buff,0,sizeof(struct h264_mv_buffer));
2761 return eRet;
2762 }
2763
2764 /* ======================================================================
2765 FUNCTION
2766 omx_vdec::GetComponentVersion
2767
2768 DESCRIPTION
2769 Returns the component version.
2770
2771 PARAMETERS
2772 TBD.
2773
2774 RETURN VALUE
2775 OMX_ErrorNone.
2776
2777 ========================================================================== */
get_component_version(OMX_IN OMX_HANDLETYPE hComp,OMX_OUT OMX_STRING componentName,OMX_OUT OMX_VERSIONTYPE * componentVersion,OMX_OUT OMX_VERSIONTYPE * specVersion,OMX_OUT OMX_UUIDTYPE * componentUUID)2778 OMX_ERRORTYPE omx_vdec::get_component_version
2779 (
2780 OMX_IN OMX_HANDLETYPE hComp,
2781 OMX_OUT OMX_STRING componentName,
2782 OMX_OUT OMX_VERSIONTYPE* componentVersion,
2783 OMX_OUT OMX_VERSIONTYPE* specVersion,
2784 OMX_OUT OMX_UUIDTYPE* componentUUID
2785 )
2786 {
2787 (void) hComp;
2788 (void) componentName;
2789 (void) componentVersion;
2790 (void) componentUUID;
2791 if (m_state == OMX_StateInvalid) {
2792 DEBUG_PRINT_ERROR("Get Comp Version in Invalid State");
2793 return OMX_ErrorInvalidState;
2794 }
2795 /* TBD -- Return the proper version */
2796 if (specVersion) {
2797 specVersion->nVersion = OMX_SPEC_VERSION;
2798 }
2799 return OMX_ErrorNone;
2800 }
2801 /* ======================================================================
2802 FUNCTION
2803 omx_vdec::SendCommand
2804
2805 DESCRIPTION
2806 Returns zero if all the buffers released..
2807
2808 PARAMETERS
2809 None.
2810
2811 RETURN VALUE
2812 true/false
2813
2814 ========================================================================== */
send_command(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_COMMANDTYPE cmd,OMX_IN OMX_U32 param1,OMX_IN OMX_PTR cmdData)2815 OMX_ERRORTYPE omx_vdec::send_command(OMX_IN OMX_HANDLETYPE hComp,
2816 OMX_IN OMX_COMMANDTYPE cmd,
2817 OMX_IN OMX_U32 param1,
2818 OMX_IN OMX_PTR cmdData
2819 )
2820 {
2821 (void) hComp;
2822 (void) cmdData;
2823 DEBUG_PRINT_LOW("send_command: Recieved a Command from Client");
2824 if (m_state == OMX_StateInvalid) {
2825 DEBUG_PRINT_ERROR("ERROR: Send Command in Invalid State");
2826 return OMX_ErrorInvalidState;
2827 }
2828 if (cmd == OMX_CommandFlush && param1 != OMX_CORE_INPUT_PORT_INDEX
2829 && param1 != OMX_CORE_OUTPUT_PORT_INDEX && param1 != OMX_ALL) {
2830 DEBUG_PRINT_ERROR("send_command(): ERROR OMX_CommandFlush "
2831 "to invalid port: %u", (unsigned int)param1);
2832 return OMX_ErrorBadPortIndex;
2833 }
2834
2835 post_event((unsigned)cmd,(unsigned)param1,OMX_COMPONENT_GENERATE_COMMAND);
2836 sem_wait(&m_cmd_lock);
2837 DEBUG_PRINT_LOW("send_command: Command Processed");
2838 return OMX_ErrorNone;
2839 }
2840
2841 /* ======================================================================
2842 FUNCTION
2843 omx_vdec::SendCommand
2844
2845 DESCRIPTION
2846 Returns zero if all the buffers released..
2847
2848 PARAMETERS
2849 None.
2850
2851 RETURN VALUE
2852 true/false
2853
2854 ========================================================================== */
send_command_proxy(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_COMMANDTYPE cmd,OMX_IN OMX_U32 param1,OMX_IN OMX_PTR cmdData)2855 OMX_ERRORTYPE omx_vdec::send_command_proxy(OMX_IN OMX_HANDLETYPE hComp,
2856 OMX_IN OMX_COMMANDTYPE cmd,
2857 OMX_IN OMX_U32 param1,
2858 OMX_IN OMX_PTR cmdData
2859 )
2860 {
2861 (void) hComp;
2862 (void) cmdData;
2863 OMX_ERRORTYPE eRet = OMX_ErrorNone;
2864 OMX_STATETYPE eState = (OMX_STATETYPE) param1;
2865 int bFlag = 1,sem_posted = 0,ret=0;
2866
2867 DEBUG_PRINT_LOW("send_command_proxy(): cmd = %d", cmd);
2868 DEBUG_PRINT_HIGH("send_command_proxy(): Current State %d, Expected State %d",
2869 m_state, eState);
2870
2871 if (cmd == OMX_CommandStateSet) {
2872 DEBUG_PRINT_HIGH("send_command_proxy(): OMX_CommandStateSet issued");
2873 DEBUG_PRINT_HIGH("Current State %d, Expected State %d", m_state, eState);
2874 /***************************/
2875 /* Current State is Loaded */
2876 /***************************/
2877 if (m_state == OMX_StateLoaded) {
2878 if (eState == OMX_StateIdle) {
2879 //if all buffers are allocated or all ports disabled
2880 if (allocate_done() ||
2881 (m_inp_bEnabled == OMX_FALSE && m_out_bEnabled == OMX_FALSE)) {
2882 DEBUG_PRINT_LOW("send_command_proxy(): Loaded-->Idle");
2883 } else {
2884 DEBUG_PRINT_LOW("send_command_proxy(): Loaded-->Idle-Pending");
2885 BITMASK_SET(&m_flags, OMX_COMPONENT_IDLE_PENDING);
2886 // Skip the event notification
2887 bFlag = 0;
2888 }
2889 }
2890 /* Requesting transition from Loaded to Loaded */
2891 else if (eState == OMX_StateLoaded) {
2892 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Loaded-->Loaded");
2893 post_event(OMX_EventError,OMX_ErrorSameState,\
2894 OMX_COMPONENT_GENERATE_EVENT);
2895 eRet = OMX_ErrorSameState;
2896 }
2897 /* Requesting transition from Loaded to WaitForResources */
2898 else if (eState == OMX_StateWaitForResources) {
2899 /* Since error is None , we will post an event
2900 at the end of this function definition */
2901 DEBUG_PRINT_LOW("send_command_proxy(): Loaded-->WaitForResources");
2902 }
2903 /* Requesting transition from Loaded to Executing */
2904 else if (eState == OMX_StateExecuting) {
2905 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Loaded-->Executing");
2906 post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
2907 OMX_COMPONENT_GENERATE_EVENT);
2908 eRet = OMX_ErrorIncorrectStateTransition;
2909 }
2910 /* Requesting transition from Loaded to Pause */
2911 else if (eState == OMX_StatePause) {
2912 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Loaded-->Pause");
2913 post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
2914 OMX_COMPONENT_GENERATE_EVENT);
2915 eRet = OMX_ErrorIncorrectStateTransition;
2916 }
2917 /* Requesting transition from Loaded to Invalid */
2918 else if (eState == OMX_StateInvalid) {
2919 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Loaded-->Invalid");
2920 post_event(OMX_EventError,eState,OMX_COMPONENT_GENERATE_EVENT);
2921 eRet = OMX_ErrorInvalidState;
2922 } else {
2923 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Loaded-->Invalid(%d Not Handled)",\
2924 eState);
2925 eRet = OMX_ErrorBadParameter;
2926 }
2927 }
2928
2929 /***************************/
2930 /* Current State is IDLE */
2931 /***************************/
2932 else if (m_state == OMX_StateIdle) {
2933 if (eState == OMX_StateLoaded) {
2934 if (release_done()) {
2935 /*
2936 * Since error is None , we will post an event at the end
2937 * of this function definition
2938 * Reset buffer requirements here to ensure setting buffer requirement
2939 * when component move to executing state from loaded state via Idle.
2940 */
2941 drv_ctx.op_buf.buffer_size = 0;
2942 drv_ctx.op_buf.actualcount = 0;
2943 DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Loaded");
2944 } else {
2945 DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Loaded-Pending");
2946 BITMASK_SET(&m_flags, OMX_COMPONENT_LOADING_PENDING);
2947 // Skip the event notification
2948 bFlag = 0;
2949 }
2950 }
2951 /* Requesting transition from Idle to Executing */
2952 else if (eState == OMX_StateExecuting) {
2953 DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Executing");
2954 //BITMASK_SET(&m_flags, OMX_COMPONENT_EXECUTE_PENDING);
2955 bFlag = 1;
2956 DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Executing");
2957 m_state=OMX_StateExecuting;
2958 DEBUG_PRINT_HIGH("Stream On CAPTURE Was successful");
2959 }
2960 /* Requesting transition from Idle to Idle */
2961 else if (eState == OMX_StateIdle) {
2962 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Idle-->Idle");
2963 post_event(OMX_EventError,OMX_ErrorSameState,\
2964 OMX_COMPONENT_GENERATE_EVENT);
2965 eRet = OMX_ErrorSameState;
2966 }
2967 /* Requesting transition from Idle to WaitForResources */
2968 else if (eState == OMX_StateWaitForResources) {
2969 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Idle-->WaitForResources");
2970 post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
2971 OMX_COMPONENT_GENERATE_EVENT);
2972 eRet = OMX_ErrorIncorrectStateTransition;
2973 }
2974 /* Requesting transition from Idle to Pause */
2975 else if (eState == OMX_StatePause) {
2976 /*To pause the Video core we need to start the driver*/
2977 if (/*ioctl (drv_ctx.video_driver_fd,VDEC_IOCTL_CMD_START,
2978 NULL) < */0) {
2979 DEBUG_PRINT_ERROR("VDEC_IOCTL_CMD_START FAILED");
2980 omx_report_error ();
2981 eRet = OMX_ErrorHardware;
2982 } else {
2983 BITMASK_SET(&m_flags,OMX_COMPONENT_PAUSE_PENDING);
2984 DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Pause");
2985 bFlag = 0;
2986 }
2987 }
2988 /* Requesting transition from Idle to Invalid */
2989 else if (eState == OMX_StateInvalid) {
2990 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Idle-->Invalid");
2991 post_event(OMX_EventError,eState,OMX_COMPONENT_GENERATE_EVENT);
2992 eRet = OMX_ErrorInvalidState;
2993 } else {
2994 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Idle --> %d Not Handled",eState);
2995 eRet = OMX_ErrorBadParameter;
2996 }
2997 }
2998
2999 /******************************/
3000 /* Current State is Executing */
3001 /******************************/
3002 else if (m_state == OMX_StateExecuting) {
3003 DEBUG_PRINT_LOW("Command Recieved in OMX_StateExecuting");
3004 /* Requesting transition from Executing to Idle */
3005 if (eState == OMX_StateIdle) {
3006 /* Since error is None , we will post an event
3007 at the end of this function definition
3008 */
3009 DEBUG_PRINT_LOW("send_command_proxy(): Executing --> Idle");
3010 BITMASK_SET(&m_flags,OMX_COMPONENT_IDLE_PENDING);
3011 if (!sem_posted) {
3012 sem_posted = 1;
3013 sem_post (&m_cmd_lock);
3014 execute_omx_flush(OMX_ALL);
3015 }
3016 bFlag = 0;
3017 }
3018 /* Requesting transition from Executing to Paused */
3019 else if (eState == OMX_StatePause) {
3020 DEBUG_PRINT_LOW("PAUSE Command Issued");
3021 m_state = OMX_StatePause;
3022 bFlag = 1;
3023 }
3024 /* Requesting transition from Executing to Loaded */
3025 else if (eState == OMX_StateLoaded) {
3026 DEBUG_PRINT_ERROR("send_command_proxy(): Executing --> Loaded");
3027 post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
3028 OMX_COMPONENT_GENERATE_EVENT);
3029 eRet = OMX_ErrorIncorrectStateTransition;
3030 }
3031 /* Requesting transition from Executing to WaitForResources */
3032 else if (eState == OMX_StateWaitForResources) {
3033 DEBUG_PRINT_ERROR("send_command_proxy(): Executing --> WaitForResources");
3034 post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
3035 OMX_COMPONENT_GENERATE_EVENT);
3036 eRet = OMX_ErrorIncorrectStateTransition;
3037 }
3038 /* Requesting transition from Executing to Executing */
3039 else if (eState == OMX_StateExecuting) {
3040 DEBUG_PRINT_ERROR("send_command_proxy(): Executing --> Executing");
3041 post_event(OMX_EventError,OMX_ErrorSameState,\
3042 OMX_COMPONENT_GENERATE_EVENT);
3043 eRet = OMX_ErrorSameState;
3044 }
3045 /* Requesting transition from Executing to Invalid */
3046 else if (eState == OMX_StateInvalid) {
3047 DEBUG_PRINT_ERROR("send_command_proxy(): Executing --> Invalid");
3048 post_event(OMX_EventError,eState,OMX_COMPONENT_GENERATE_EVENT);
3049 eRet = OMX_ErrorInvalidState;
3050 } else {
3051 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Executing --> %d Not Handled",eState);
3052 eRet = OMX_ErrorBadParameter;
3053 }
3054 }
3055 /***************************/
3056 /* Current State is Pause */
3057 /***************************/
3058 else if (m_state == OMX_StatePause) {
3059 /* Requesting transition from Pause to Executing */
3060 if (eState == OMX_StateExecuting) {
3061 DEBUG_PRINT_LOW("Pause --> Executing");
3062 m_state = OMX_StateExecuting;
3063 bFlag = 1;
3064 }
3065 /* Requesting transition from Pause to Idle */
3066 else if (eState == OMX_StateIdle) {
3067 /* Since error is None , we will post an event
3068 at the end of this function definition */
3069 DEBUG_PRINT_LOW("Pause --> Idle");
3070 BITMASK_SET(&m_flags,OMX_COMPONENT_IDLE_PENDING);
3071 if (!sem_posted) {
3072 sem_posted = 1;
3073 sem_post (&m_cmd_lock);
3074 execute_omx_flush(OMX_ALL);
3075 }
3076 bFlag = 0;
3077 }
3078 /* Requesting transition from Pause to loaded */
3079 else if (eState == OMX_StateLoaded) {
3080 DEBUG_PRINT_ERROR("Pause --> loaded");
3081 post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
3082 OMX_COMPONENT_GENERATE_EVENT);
3083 eRet = OMX_ErrorIncorrectStateTransition;
3084 }
3085 /* Requesting transition from Pause to WaitForResources */
3086 else if (eState == OMX_StateWaitForResources) {
3087 DEBUG_PRINT_ERROR("Pause --> WaitForResources");
3088 post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
3089 OMX_COMPONENT_GENERATE_EVENT);
3090 eRet = OMX_ErrorIncorrectStateTransition;
3091 }
3092 /* Requesting transition from Pause to Pause */
3093 else if (eState == OMX_StatePause) {
3094 DEBUG_PRINT_ERROR("Pause --> Pause");
3095 post_event(OMX_EventError,OMX_ErrorSameState,\
3096 OMX_COMPONENT_GENERATE_EVENT);
3097 eRet = OMX_ErrorSameState;
3098 }
3099 /* Requesting transition from Pause to Invalid */
3100 else if (eState == OMX_StateInvalid) {
3101 DEBUG_PRINT_ERROR("Pause --> Invalid");
3102 post_event(OMX_EventError,eState,OMX_COMPONENT_GENERATE_EVENT);
3103 eRet = OMX_ErrorInvalidState;
3104 } else {
3105 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Paused --> %d Not Handled",eState);
3106 eRet = OMX_ErrorBadParameter;
3107 }
3108 }
3109 /***************************/
3110 /* Current State is WaitForResources */
3111 /***************************/
3112 else if (m_state == OMX_StateWaitForResources) {
3113 /* Requesting transition from WaitForResources to Loaded */
3114 if (eState == OMX_StateLoaded) {
3115 /* Since error is None , we will post an event
3116 at the end of this function definition */
3117 DEBUG_PRINT_LOW("send_command_proxy(): WaitForResources-->Loaded");
3118 }
3119 /* Requesting transition from WaitForResources to WaitForResources */
3120 else if (eState == OMX_StateWaitForResources) {
3121 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): WaitForResources-->WaitForResources");
3122 post_event(OMX_EventError,OMX_ErrorSameState,
3123 OMX_COMPONENT_GENERATE_EVENT);
3124 eRet = OMX_ErrorSameState;
3125 }
3126 /* Requesting transition from WaitForResources to Executing */
3127 else if (eState == OMX_StateExecuting) {
3128 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): WaitForResources-->Executing");
3129 post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
3130 OMX_COMPONENT_GENERATE_EVENT);
3131 eRet = OMX_ErrorIncorrectStateTransition;
3132 }
3133 /* Requesting transition from WaitForResources to Pause */
3134 else if (eState == OMX_StatePause) {
3135 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): WaitForResources-->Pause");
3136 post_event(OMX_EventError,OMX_ErrorIncorrectStateTransition,\
3137 OMX_COMPONENT_GENERATE_EVENT);
3138 eRet = OMX_ErrorIncorrectStateTransition;
3139 }
3140 /* Requesting transition from WaitForResources to Invalid */
3141 else if (eState == OMX_StateInvalid) {
3142 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): WaitForResources-->Invalid");
3143 post_event(OMX_EventError,eState,OMX_COMPONENT_GENERATE_EVENT);
3144 eRet = OMX_ErrorInvalidState;
3145 }
3146 /* Requesting transition from WaitForResources to Loaded -
3147 is NOT tested by Khronos TS */
3148
3149 } else {
3150 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): %d --> %d(Not Handled)",m_state,eState);
3151 eRet = OMX_ErrorBadParameter;
3152 }
3153 }
3154 /********************************/
3155 /* Current State is Invalid */
3156 /*******************************/
3157 else if (m_state == OMX_StateInvalid) {
3158 /* State Transition from Inavlid to any state */
3159 if (eState == (OMX_StateLoaded || OMX_StateWaitForResources
3160 || OMX_StateIdle || OMX_StateExecuting
3161 || OMX_StatePause || OMX_StateInvalid)) {
3162 DEBUG_PRINT_ERROR("ERROR::send_command_proxy(): Invalid -->Loaded");
3163 post_event(OMX_EventError,OMX_ErrorInvalidState,\
3164 OMX_COMPONENT_GENERATE_EVENT);
3165 eRet = OMX_ErrorInvalidState;
3166 }
3167 } else if (cmd == OMX_CommandFlush) {
3168 DEBUG_PRINT_HIGH("send_command_proxy(): OMX_CommandFlush issued"
3169 "with param1: %u", (unsigned int)param1);
3170 #ifdef _MSM8974_
3171 send_codec_config();
3172 #endif
3173 if (cmd == OMX_CommandFlush && (param1 == OMX_CORE_INPUT_PORT_INDEX ||
3174 param1 == OMX_ALL)) {
3175 if (android_atomic_add(0, &m_queued_codec_config_count) > 0) {
3176 struct timespec ts;
3177
3178 clock_gettime(CLOCK_REALTIME, &ts);
3179 ts.tv_sec += 2;
3180 DEBUG_PRINT_LOW("waiting for %d EBDs of CODEC CONFIG buffers ",
3181 m_queued_codec_config_count);
3182 BITMASK_SET(&m_flags, OMX_COMPONENT_FLUSH_DEFERRED);
3183 if (sem_timedwait(&m_safe_flush, &ts)) {
3184 DEBUG_PRINT_ERROR("Failed to wait for EBDs of CODEC CONFIG buffers");
3185 }
3186 BITMASK_CLEAR (&m_flags,OMX_COMPONENT_FLUSH_DEFERRED);
3187 }
3188 }
3189
3190 if (OMX_CORE_INPUT_PORT_INDEX == param1 || OMX_ALL == param1) {
3191 BITMASK_SET(&m_flags, OMX_COMPONENT_INPUT_FLUSH_PENDING);
3192 }
3193 if (OMX_CORE_OUTPUT_PORT_INDEX == param1 || OMX_ALL == param1) {
3194 BITMASK_SET(&m_flags, OMX_COMPONENT_OUTPUT_FLUSH_PENDING);
3195 }
3196 if (!sem_posted) {
3197 sem_posted = 1;
3198 DEBUG_PRINT_LOW("Set the Semaphore");
3199 sem_post (&m_cmd_lock);
3200 execute_omx_flush(param1);
3201 }
3202 bFlag = 0;
3203 } else if ( cmd == OMX_CommandPortEnable) {
3204 DEBUG_PRINT_HIGH("send_command_proxy(): OMX_CommandPortEnable issued"
3205 "with param1: %u", (unsigned int)param1);
3206 if (param1 == OMX_CORE_INPUT_PORT_INDEX || param1 == OMX_ALL) {
3207 m_inp_bEnabled = OMX_TRUE;
3208
3209 if ( (m_state == OMX_StateLoaded &&
3210 !BITMASK_PRESENT(&m_flags,OMX_COMPONENT_IDLE_PENDING))
3211 || allocate_input_done()) {
3212 post_event(OMX_CommandPortEnable,OMX_CORE_INPUT_PORT_INDEX,
3213 OMX_COMPONENT_GENERATE_EVENT);
3214 } else {
3215 DEBUG_PRINT_LOW("send_command_proxy(): Disabled-->Enabled Pending");
3216 BITMASK_SET(&m_flags, OMX_COMPONENT_INPUT_ENABLE_PENDING);
3217 // Skip the event notification
3218 bFlag = 0;
3219 }
3220 }
3221 if (param1 == OMX_CORE_OUTPUT_PORT_INDEX || param1 == OMX_ALL) {
3222 DEBUG_PRINT_LOW("Enable output Port command recieved");
3223 m_out_bEnabled = OMX_TRUE;
3224
3225 if ( (m_state == OMX_StateLoaded &&
3226 !BITMASK_PRESENT(&m_flags,OMX_COMPONENT_IDLE_PENDING))
3227 || (allocate_output_done())) {
3228 post_event(OMX_CommandPortEnable,OMX_CORE_OUTPUT_PORT_INDEX,
3229 OMX_COMPONENT_GENERATE_EVENT);
3230
3231 } else {
3232 DEBUG_PRINT_LOW("send_command_proxy(): Disabled-->Enabled Pending");
3233 BITMASK_SET(&m_flags, OMX_COMPONENT_OUTPUT_ENABLE_PENDING);
3234 // Skip the event notification
3235 bFlag = 0;
3236 /* enable/disable downscaling if required */
3237 ret = decide_downscalar();
3238 if (ret) {
3239 DEBUG_PRINT_LOW("decide_downscalar failed\n");
3240 }
3241 }
3242 }
3243 } else if (cmd == OMX_CommandPortDisable) {
3244 DEBUG_PRINT_HIGH("send_command_proxy(): OMX_CommandPortDisable issued"
3245 "with param1: %u", (unsigned int)param1);
3246 if (param1 == OMX_CORE_INPUT_PORT_INDEX || param1 == OMX_ALL) {
3247 codec_config_flag = false;
3248 m_inp_bEnabled = OMX_FALSE;
3249 if ((m_state == OMX_StateLoaded || m_state == OMX_StateIdle)
3250 && release_input_done()) {
3251 post_event(OMX_CommandPortDisable,OMX_CORE_INPUT_PORT_INDEX,
3252 OMX_COMPONENT_GENERATE_EVENT);
3253 } else {
3254 BITMASK_SET(&m_flags, OMX_COMPONENT_INPUT_DISABLE_PENDING);
3255 if (m_state == OMX_StatePause ||m_state == OMX_StateExecuting) {
3256 if (!sem_posted) {
3257 sem_posted = 1;
3258 sem_post (&m_cmd_lock);
3259 }
3260 execute_omx_flush(OMX_CORE_INPUT_PORT_INDEX);
3261 }
3262
3263 // Skip the event notification
3264 bFlag = 0;
3265 }
3266 }
3267 if (param1 == OMX_CORE_OUTPUT_PORT_INDEX || param1 == OMX_ALL) {
3268 m_out_bEnabled = OMX_FALSE;
3269 DEBUG_PRINT_LOW("Disable output Port command recieved");
3270 if ((m_state == OMX_StateLoaded || m_state == OMX_StateIdle)
3271 && release_output_done()) {
3272 post_event(OMX_CommandPortDisable,OMX_CORE_OUTPUT_PORT_INDEX,\
3273 OMX_COMPONENT_GENERATE_EVENT);
3274 } else {
3275 BITMASK_SET(&m_flags, OMX_COMPONENT_OUTPUT_DISABLE_PENDING);
3276 if (m_state == OMX_StatePause ||m_state == OMX_StateExecuting) {
3277 if (!sem_posted) {
3278 sem_posted = 1;
3279 sem_post (&m_cmd_lock);
3280 }
3281 BITMASK_SET(&m_flags, OMX_COMPONENT_OUTPUT_FLUSH_IN_DISABLE_PENDING);
3282 execute_omx_flush(OMX_CORE_OUTPUT_PORT_INDEX);
3283 }
3284 // Skip the event notification
3285 bFlag = 0;
3286
3287 }
3288 }
3289 } else {
3290 DEBUG_PRINT_ERROR("Error: Invalid Command other than StateSet (%d)",cmd);
3291 eRet = OMX_ErrorNotImplemented;
3292 }
3293 if (eRet == OMX_ErrorNone && bFlag) {
3294 post_event(cmd,eState,OMX_COMPONENT_GENERATE_EVENT);
3295 }
3296 if (!sem_posted) {
3297 sem_post(&m_cmd_lock);
3298 }
3299
3300 return eRet;
3301 }
3302
3303 /* ======================================================================
3304 FUNCTION
3305 omx_vdec::ExecuteOmxFlush
3306
3307 DESCRIPTION
3308 Executes the OMX flush.
3309
3310 PARAMETERS
3311 flushtype - input flush(1)/output flush(0)/ both.
3312
3313 RETURN VALUE
3314 true/false
3315
3316 ========================================================================== */
execute_omx_flush(OMX_U32 flushType)3317 bool omx_vdec::execute_omx_flush(OMX_U32 flushType)
3318 {
3319 bool bRet = false;
3320 struct v4l2_plane plane;
3321 struct v4l2_buffer v4l2_buf;
3322 struct v4l2_decoder_cmd dec;
3323 DEBUG_PRINT_LOW("in %s, flushing %u", __func__, (unsigned int)flushType);
3324 memset((void *)&v4l2_buf,0,sizeof(v4l2_buf));
3325 dec.cmd = V4L2_DEC_QCOM_CMD_FLUSH;
3326
3327 DEBUG_PRINT_HIGH("in %s: reconfig? %d", __func__, in_reconfig);
3328
3329 if (in_reconfig && flushType == OMX_CORE_OUTPUT_PORT_INDEX) {
3330 output_flush_progress = true;
3331 dec.flags = V4L2_DEC_QCOM_CMD_FLUSH_CAPTURE;
3332 } else {
3333 /* XXX: The driver/hardware does not support flushing of individual ports
3334 * in all states. So we pretty much need to flush both ports internally,
3335 * but client should only get the FLUSH_(INPUT|OUTPUT)_DONE for the one it
3336 * requested. Since OMX_COMPONENT_(OUTPUT|INPUT)_FLUSH_PENDING isn't set,
3337 * we automatically omit sending the FLUSH done for the "opposite" port. */
3338 input_flush_progress = true;
3339 output_flush_progress = true;
3340 dec.flags = V4L2_DEC_QCOM_CMD_FLUSH_OUTPUT | V4L2_DEC_QCOM_CMD_FLUSH_CAPTURE;
3341 request_perf_level(VIDC_TURBO);
3342 }
3343
3344 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_DECODER_CMD, &dec)) {
3345 DEBUG_PRINT_ERROR("Flush Port (%u) Failed ", (unsigned int)flushType);
3346 bRet = false;
3347 }
3348
3349 return bRet;
3350 }
3351 /*=========================================================================
3352 FUNCTION : execute_output_flush
3353
3354 DESCRIPTION
3355 Executes the OMX flush at OUTPUT PORT.
3356
3357 PARAMETERS
3358 None.
3359
3360 RETURN VALUE
3361 true/false
3362 ==========================================================================*/
execute_output_flush()3363 bool omx_vdec::execute_output_flush()
3364 {
3365 unsigned long p1 = 0; // Parameter - 1
3366 unsigned long p2 = 0; // Parameter - 2
3367 unsigned long ident = 0;
3368 bool bRet = true;
3369
3370 /*Generate FBD for all Buffers in the FTBq*/
3371 pthread_mutex_lock(&m_lock);
3372 DEBUG_PRINT_LOW("Initiate Output Flush");
3373
3374 //reset last render TS
3375 if(m_last_rendered_TS > 0) {
3376 m_last_rendered_TS = 0;
3377 }
3378
3379 while (m_ftb_q.m_size) {
3380 DEBUG_PRINT_LOW("Buffer queue size %lu pending buf cnt %d",
3381 m_ftb_q.m_size,pending_output_buffers);
3382 m_ftb_q.pop_entry(&p1,&p2,&ident);
3383 DEBUG_PRINT_LOW("ID(%lx) P1(%lx) P2(%lx)", ident, p1, p2);
3384 if (ident == m_fill_output_msg ) {
3385 m_cb.FillBufferDone(&m_cmp, m_app_data, (OMX_BUFFERHEADERTYPE *)(intptr_t)p2);
3386 } else if (ident == OMX_COMPONENT_GENERATE_FBD) {
3387 fill_buffer_done(&m_cmp,(OMX_BUFFERHEADERTYPE *)(intptr_t)p1);
3388 }
3389 }
3390 pthread_mutex_unlock(&m_lock);
3391 output_flush_progress = false;
3392
3393 if (arbitrary_bytes) {
3394 prev_ts = LLONG_MAX;
3395 rst_prev_ts = true;
3396 }
3397 DEBUG_PRINT_HIGH("OMX flush o/p Port complete PenBuf(%d)", pending_output_buffers);
3398 return bRet;
3399 }
3400 /*=========================================================================
3401 FUNCTION : execute_input_flush
3402
3403 DESCRIPTION
3404 Executes the OMX flush at INPUT PORT.
3405
3406 PARAMETERS
3407 None.
3408
3409 RETURN VALUE
3410 true/false
3411 ==========================================================================*/
execute_input_flush()3412 bool omx_vdec::execute_input_flush()
3413 {
3414 unsigned i =0;
3415 unsigned long p1 = 0; // Parameter - 1
3416 unsigned long p2 = 0; // Parameter - 2
3417 unsigned long ident = 0;
3418 bool bRet = true;
3419
3420 /*Generate EBD for all Buffers in the ETBq*/
3421 DEBUG_PRINT_LOW("Initiate Input Flush");
3422
3423 pthread_mutex_lock(&m_lock);
3424 DEBUG_PRINT_LOW("Check if the Queue is empty");
3425 while (m_etb_q.m_size) {
3426 m_etb_q.pop_entry(&p1,&p2,&ident);
3427
3428 if (ident == OMX_COMPONENT_GENERATE_ETB_ARBITRARY) {
3429 DEBUG_PRINT_LOW("Flush Input Heap Buffer %p",(OMX_BUFFERHEADERTYPE *)p2);
3430 m_cb.EmptyBufferDone(&m_cmp ,m_app_data, (OMX_BUFFERHEADERTYPE *)p2);
3431 } else if (ident == OMX_COMPONENT_GENERATE_ETB) {
3432 pending_input_buffers++;
3433 VIDC_TRACE_INT_LOW("ETB-pending", pending_input_buffers);
3434 DEBUG_PRINT_LOW("Flush Input OMX_COMPONENT_GENERATE_ETB %p, pending_input_buffers %d",
3435 (OMX_BUFFERHEADERTYPE *)p2, pending_input_buffers);
3436 empty_buffer_done(&m_cmp,(OMX_BUFFERHEADERTYPE *)p2);
3437 } else if (ident == OMX_COMPONENT_GENERATE_EBD) {
3438 DEBUG_PRINT_LOW("Flush Input OMX_COMPONENT_GENERATE_EBD %p",
3439 (OMX_BUFFERHEADERTYPE *)p1);
3440 empty_buffer_done(&m_cmp,(OMX_BUFFERHEADERTYPE *)p1);
3441 }
3442 }
3443 time_stamp_dts.flush_timestamp();
3444 /*Check if Heap Buffers are to be flushed*/
3445 if (arbitrary_bytes && !(codec_config_flag)) {
3446 DEBUG_PRINT_LOW("Reset all the variables before flusing");
3447 h264_scratch.nFilledLen = 0;
3448 nal_count = 0;
3449 look_ahead_nal = false;
3450 frame_count = 0;
3451 h264_last_au_ts = LLONG_MAX;
3452 h264_last_au_flags = 0;
3453 memset(m_demux_offsets, 0, ( sizeof(OMX_U32) * 8192) );
3454 m_demux_entries = 0;
3455 DEBUG_PRINT_LOW("Initialize parser");
3456 if (m_frame_parser.mutils) {
3457 m_frame_parser.mutils->initialize_frame_checking_environment();
3458 }
3459
3460 while (m_input_pending_q.m_size) {
3461 m_input_pending_q.pop_entry(&p1,&p2,&ident);
3462 m_cb.EmptyBufferDone(&m_cmp ,m_app_data, (OMX_BUFFERHEADERTYPE *)p1);
3463 }
3464
3465 if (psource_frame) {
3466 m_cb.EmptyBufferDone(&m_cmp ,m_app_data,psource_frame);
3467 psource_frame = NULL;
3468 }
3469
3470 if (pdest_frame) {
3471 pdest_frame->nFilledLen = 0;
3472 m_input_free_q.insert_entry((unsigned long) pdest_frame, (unsigned int)NULL,
3473 (unsigned int)NULL);
3474 pdest_frame = NULL;
3475 }
3476 m_frame_parser.flush();
3477 } else if (codec_config_flag) {
3478 DEBUG_PRINT_HIGH("frame_parser flushing skipped due to codec config buffer "
3479 "is not sent to the driver yet");
3480 }
3481 pthread_mutex_unlock(&m_lock);
3482 input_flush_progress = false;
3483 if (!arbitrary_bytes) {
3484 prev_ts = LLONG_MAX;
3485 rst_prev_ts = true;
3486 }
3487 #ifdef _ANDROID_
3488 if (m_debug_timestamp) {
3489 m_timestamp_list.reset_ts_list();
3490 }
3491 #endif
3492 DEBUG_PRINT_HIGH("OMX flush i/p Port complete PenBuf(%d)", pending_input_buffers);
3493 return bRet;
3494 }
3495
3496 /*=========================================================================
3497 FUNCTION : notify_flush_done
3498
3499 DESCRIPTION
3500 Notifies flush done to the OMX Client.
3501
3502 PARAMETERS
3503 ctxt -- Context information related to the self..
3504
3505 RETURN VALUE
3506 NONE
3507 ==========================================================================*/
notify_flush_done(void * ctxt)3508 void omx_vdec::notify_flush_done(void *ctxt) {
3509
3510 omx_vdec *pThis = (omx_vdec *) ctxt;
3511
3512 if (!pThis->input_flush_progress && !pThis->output_flush_progress) {
3513 if (BITMASK_PRESENT(&pThis->m_flags,
3514 OMX_COMPONENT_OUTPUT_FLUSH_PENDING)) {
3515 DEBUG_PRINT_LOW("Notify Output Flush done");
3516 BITMASK_CLEAR (&pThis->m_flags,OMX_COMPONENT_OUTPUT_FLUSH_PENDING);
3517 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
3518 OMX_EventCmdComplete,OMX_CommandFlush,
3519 OMX_CORE_OUTPUT_PORT_INDEX,NULL );
3520 }
3521
3522 if (BITMASK_PRESENT(&pThis->m_flags,
3523 OMX_COMPONENT_INPUT_FLUSH_PENDING)) {
3524 BITMASK_CLEAR (&pThis->m_flags,OMX_COMPONENT_INPUT_FLUSH_PENDING);
3525 DEBUG_PRINT_LOW("Input Flush completed - Notify Client");
3526 pThis->m_cb.EventHandler(&pThis->m_cmp, pThis->m_app_data,
3527 OMX_EventCmdComplete,OMX_CommandFlush,
3528 OMX_CORE_INPUT_PORT_INDEX,NULL );
3529 }
3530 }
3531 }
3532
3533 /* ======================================================================
3534 FUNCTION
3535 omx_vdec::SendCommandEvent
3536
3537 DESCRIPTION
3538 Send the event to decoder pipe. This is needed to generate the callbacks
3539 in decoder thread context.
3540
3541 PARAMETERS
3542 None.
3543
3544 RETURN VALUE
3545 true/false
3546
3547 ========================================================================== */
post_event(unsigned long p1,unsigned long p2,unsigned long id)3548 bool omx_vdec::post_event(unsigned long p1,
3549 unsigned long p2,
3550 unsigned long id)
3551 {
3552 bool bRet = false;
3553
3554 /* Just drop messages typically generated by hardware (w/o client request),
3555 * if we've reported an error to client. */
3556 if (m_error_propogated) {
3557 switch (id) {
3558 case OMX_COMPONENT_GENERATE_PORT_RECONFIG:
3559 case OMX_COMPONENT_GENERATE_HARDWARE_ERROR:
3560 DEBUG_PRINT_ERROR("Dropping message %lx "
3561 "since client expected to be in error state", id);
3562 return false;
3563 default:
3564 /* whatever */
3565 break;
3566 }
3567 }
3568
3569 pthread_mutex_lock(&m_lock);
3570
3571 if (id == m_fill_output_msg ||
3572 id == OMX_COMPONENT_GENERATE_FBD ||
3573 id == OMX_COMPONENT_GENERATE_PORT_RECONFIG ||
3574 id == OMX_COMPONENT_GENERATE_EVENT_OUTPUT_FLUSH) {
3575 m_ftb_q.insert_entry(p1,p2,id);
3576 } else if (id == OMX_COMPONENT_GENERATE_ETB ||
3577 id == OMX_COMPONENT_GENERATE_EBD ||
3578 id == OMX_COMPONENT_GENERATE_ETB_ARBITRARY ||
3579 id == OMX_COMPONENT_GENERATE_EVENT_INPUT_FLUSH) {
3580 m_etb_q.insert_entry(p1,p2,id);
3581 } else {
3582 m_cmd_q.insert_entry(p1,p2,id);
3583 }
3584
3585 bRet = true;
3586 DEBUG_PRINT_LOW("Value of this pointer in post_event %p",this);
3587 post_message(this, id);
3588
3589 pthread_mutex_unlock(&m_lock);
3590
3591 return bRet;
3592 }
3593
get_supported_profile_level(OMX_VIDEO_PARAM_PROFILELEVELTYPE * profileLevelType)3594 OMX_ERRORTYPE omx_vdec::get_supported_profile_level(OMX_VIDEO_PARAM_PROFILELEVELTYPE *profileLevelType)
3595 {
3596 OMX_ERRORTYPE eRet = OMX_ErrorNone;
3597 if (!profileLevelType)
3598 return OMX_ErrorBadParameter;
3599
3600 if (profileLevelType->nPortIndex == 0) {
3601 if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc",OMX_MAX_STRINGNAME_SIZE)) {
3602 profileLevelType->eLevel = OMX_VIDEO_AVCLevel51;
3603 if (profileLevelType->nProfileIndex == 0) {
3604 profileLevelType->eProfile = OMX_VIDEO_AVCProfileBaseline;
3605 } else if (profileLevelType->nProfileIndex == 1) {
3606 profileLevelType->eProfile = OMX_VIDEO_AVCProfileMain;
3607 } else if (profileLevelType->nProfileIndex == 2) {
3608 profileLevelType->eProfile = OMX_VIDEO_AVCProfileHigh;
3609 } else if (profileLevelType->nProfileIndex == 3) {
3610 profileLevelType->eProfile = OMX_VIDEO_AVCProfileConstrainedBaseline;
3611 } else if (profileLevelType->nProfileIndex == 4) {
3612 profileLevelType->eProfile = QOMX_VIDEO_AVCProfileConstrainedBaseline;
3613 } else if (profileLevelType->nProfileIndex == 5) {
3614 profileLevelType->eProfile = OMX_VIDEO_AVCProfileConstrainedHigh;
3615 } else if (profileLevelType->nProfileIndex == 6) {
3616 profileLevelType->eProfile = QOMX_VIDEO_AVCProfileConstrainedHigh;
3617 } else {
3618 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported nProfileIndex ret NoMore %u",
3619 (unsigned int)profileLevelType->nProfileIndex);
3620 eRet = OMX_ErrorNoMore;
3621 }
3622 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mvc", OMX_MAX_STRINGNAME_SIZE)) {
3623 if (profileLevelType->nProfileIndex == 0) {
3624 profileLevelType->eProfile = QOMX_VIDEO_MVCProfileStereoHigh;
3625 profileLevelType->eLevel = QOMX_VIDEO_MVCLevel51;
3626 } else {
3627 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported nProfileIndex ret NoMore %u",
3628 (unsigned int)profileLevelType->nProfileIndex);
3629 eRet = OMX_ErrorNoMore;
3630 }
3631 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.hevc", OMX_MAX_STRINGNAME_SIZE)) {
3632 if (profileLevelType->nProfileIndex == 0) {
3633 profileLevelType->eProfile = OMX_VIDEO_HEVCProfileMain;
3634 profileLevelType->eLevel = OMX_VIDEO_HEVCMainTierLevel51;
3635 } else if (profileLevelType->nProfileIndex == 1) {
3636 profileLevelType->eProfile = OMX_VIDEO_HEVCProfileMain10;
3637 profileLevelType->eLevel = OMX_VIDEO_HEVCMainTierLevel51;
3638 } else if (profileLevelType->nProfileIndex == 2) {
3639 profileLevelType->eProfile = OMX_VIDEO_HEVCProfileMain10HDR10;
3640 profileLevelType->eLevel = OMX_VIDEO_HEVCMainTierLevel51;
3641 } else {
3642 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported nProfileIndex ret NoMore %u",
3643 (unsigned int)profileLevelType->nProfileIndex);
3644 eRet = OMX_ErrorNoMore;
3645 }
3646 } else if ((!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.h263",OMX_MAX_STRINGNAME_SIZE))) {
3647 if (profileLevelType->nProfileIndex == 0) {
3648 profileLevelType->eProfile = OMX_VIDEO_H263ProfileBaseline;
3649 profileLevelType->eLevel = OMX_VIDEO_H263Level70;
3650 } else {
3651 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported nProfileIndex ret NoMore %u",
3652 (unsigned int)profileLevelType->nProfileIndex);
3653 eRet = OMX_ErrorNoMore;
3654 }
3655 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg4",OMX_MAX_STRINGNAME_SIZE)) {
3656 if (profileLevelType->nProfileIndex == 0) {
3657 profileLevelType->eProfile = OMX_VIDEO_MPEG4ProfileSimple;
3658 profileLevelType->eLevel = OMX_VIDEO_MPEG4Level5;
3659 } else if (profileLevelType->nProfileIndex == 1) {
3660 profileLevelType->eProfile = OMX_VIDEO_MPEG4ProfileAdvancedSimple;
3661 profileLevelType->eLevel = OMX_VIDEO_MPEG4Level5;
3662 } else {
3663 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported nProfileIndex ret NoMore %u",
3664 (unsigned int)profileLevelType->nProfileIndex);
3665 eRet = OMX_ErrorNoMore;
3666 }
3667 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp8",OMX_MAX_STRINGNAME_SIZE) ||
3668 !strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp9",OMX_MAX_STRINGNAME_SIZE)) {
3669 eRet = OMX_ErrorNoMore;
3670 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg2",OMX_MAX_STRINGNAME_SIZE)) {
3671 if (profileLevelType->nProfileIndex == 0) {
3672 profileLevelType->eProfile = OMX_VIDEO_MPEG2ProfileSimple;
3673 profileLevelType->eLevel = OMX_VIDEO_MPEG2LevelHL;
3674 } else if (profileLevelType->nProfileIndex == 1) {
3675 profileLevelType->eProfile = OMX_VIDEO_MPEG2ProfileMain;
3676 profileLevelType->eLevel = OMX_VIDEO_MPEG2LevelHL;
3677 } else {
3678 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported nProfileIndex ret NoMore %u",
3679 (unsigned int)profileLevelType->nProfileIndex);
3680 eRet = OMX_ErrorNoMore;
3681 }
3682 } else {
3683 DEBUG_PRINT_ERROR("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported ret NoMore for codec: %s", drv_ctx.kind);
3684 eRet = OMX_ErrorNoMore;
3685 }
3686 } else {
3687 DEBUG_PRINT_ERROR("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported should be queries on Input port only %u",
3688 (unsigned int)profileLevelType->nPortIndex);
3689 eRet = OMX_ErrorBadPortIndex;
3690 }
3691 return eRet;
3692 }
3693
3694 /* ======================================================================
3695 FUNCTION
3696 omx_vdec::GetParameter
3697
3698 DESCRIPTION
3699 OMX Get Parameter method implementation
3700
3701 PARAMETERS
3702 <TBD>.
3703
3704 RETURN VALUE
3705 Error None if successful.
3706
3707 ========================================================================== */
get_parameter(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_INDEXTYPE paramIndex,OMX_INOUT OMX_PTR paramData)3708 OMX_ERRORTYPE omx_vdec::get_parameter(OMX_IN OMX_HANDLETYPE hComp,
3709 OMX_IN OMX_INDEXTYPE paramIndex,
3710 OMX_INOUT OMX_PTR paramData)
3711 {
3712 (void) hComp;
3713 OMX_ERRORTYPE eRet = OMX_ErrorNone;
3714
3715 DEBUG_PRINT_LOW("get_parameter:");
3716 if (m_state == OMX_StateInvalid) {
3717 DEBUG_PRINT_ERROR("Get Param in Invalid State");
3718 return OMX_ErrorInvalidState;
3719 }
3720 if (paramData == NULL) {
3721 DEBUG_PRINT_LOW("Get Param in Invalid paramData");
3722 return OMX_ErrorBadParameter;
3723 }
3724 switch ((unsigned long)paramIndex) {
3725 case OMX_IndexParamPortDefinition: {
3726 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PARAM_PORTDEFINITIONTYPE);
3727 OMX_PARAM_PORTDEFINITIONTYPE *portDefn =
3728 (OMX_PARAM_PORTDEFINITIONTYPE *) paramData;
3729 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamPortDefinition");
3730 decide_dpb_buffer_mode(is_down_scalar_enabled);
3731 eRet = update_portdef(portDefn);
3732 if (eRet == OMX_ErrorNone)
3733 m_port_def = *portDefn;
3734 break;
3735 }
3736 case OMX_IndexParamVideoInit: {
3737 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PORT_PARAM_TYPE);
3738 OMX_PORT_PARAM_TYPE *portParamType =
3739 (OMX_PORT_PARAM_TYPE *) paramData;
3740 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoInit");
3741
3742 portParamType->nVersion.nVersion = OMX_SPEC_VERSION;
3743 portParamType->nSize = sizeof(OMX_PORT_PARAM_TYPE);
3744 portParamType->nPorts = 2;
3745 portParamType->nStartPortNumber = 0;
3746 break;
3747 }
3748 case OMX_IndexParamVideoPortFormat: {
3749 VALIDATE_OMX_PARAM_DATA(paramData, OMX_VIDEO_PARAM_PORTFORMATTYPE);
3750 OMX_VIDEO_PARAM_PORTFORMATTYPE *portFmt =
3751 (OMX_VIDEO_PARAM_PORTFORMATTYPE *)paramData;
3752 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoPortFormat");
3753
3754 portFmt->nVersion.nVersion = OMX_SPEC_VERSION;
3755 portFmt->nSize = sizeof(OMX_VIDEO_PARAM_PORTFORMATTYPE);
3756
3757 if (0 == portFmt->nPortIndex) {
3758 if (0 == portFmt->nIndex) {
3759 portFmt->eColorFormat = OMX_COLOR_FormatUnused;
3760 portFmt->eCompressionFormat = eCompressionFormat;
3761 } else {
3762 DEBUG_PRINT_ERROR("get_parameter: OMX_IndexParamVideoPortFormat:"\
3763 " NoMore compression formats");
3764 eRet = OMX_ErrorNoMore;
3765 }
3766 } else if (1 == portFmt->nPortIndex) {
3767 portFmt->eCompressionFormat = OMX_VIDEO_CodingUnused;
3768
3769 // Distinguish non-surface mode from normal playback use-case based on
3770 // usage hinted via "OMX.google.android.index.useAndroidNativeBuffer2"
3771 // For non-android, use the default list
3772 // Also use default format-list if FLEXIBLE YUV is supported,
3773 // as the client negotiates the standard color-format if it needs to
3774 bool useNonSurfaceMode = false;
3775 #if defined(_ANDROID_) && !defined(FLEXYUV_SUPPORTED)
3776 useNonSurfaceMode = (m_enable_android_native_buffers == OMX_FALSE);
3777 #endif
3778 if (is_thulium_v1) {
3779 portFmt->eColorFormat = getPreferredColorFormatDefaultMode(portFmt->nIndex);
3780 } else {
3781 portFmt->eColorFormat = useNonSurfaceMode ?
3782 getPreferredColorFormatNonSurfaceMode(portFmt->nIndex) :
3783 getPreferredColorFormatDefaultMode(portFmt->nIndex);
3784 }
3785
3786 if (portFmt->eColorFormat == OMX_COLOR_FormatMax ) {
3787 eRet = OMX_ErrorNoMore;
3788 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoPortFormat:"\
3789 " NoMore Color formats");
3790 }
3791 DEBUG_PRINT_HIGH("returning color-format: 0x%x", portFmt->eColorFormat);
3792 } else {
3793 DEBUG_PRINT_ERROR("get_parameter: Bad port index %d",
3794 (int)portFmt->nPortIndex);
3795 eRet = OMX_ErrorBadPortIndex;
3796 }
3797 break;
3798 }
3799 /*Component should support this port definition*/
3800 case OMX_IndexParamAudioInit: {
3801 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PORT_PARAM_TYPE);
3802 OMX_PORT_PARAM_TYPE *audioPortParamType =
3803 (OMX_PORT_PARAM_TYPE *) paramData;
3804 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamAudioInit");
3805 audioPortParamType->nVersion.nVersion = OMX_SPEC_VERSION;
3806 audioPortParamType->nSize = sizeof(OMX_PORT_PARAM_TYPE);
3807 audioPortParamType->nPorts = 0;
3808 audioPortParamType->nStartPortNumber = 0;
3809 break;
3810 }
3811 /*Component should support this port definition*/
3812 case OMX_IndexParamImageInit: {
3813 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PORT_PARAM_TYPE);
3814 OMX_PORT_PARAM_TYPE *imagePortParamType =
3815 (OMX_PORT_PARAM_TYPE *) paramData;
3816 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamImageInit");
3817 imagePortParamType->nVersion.nVersion = OMX_SPEC_VERSION;
3818 imagePortParamType->nSize = sizeof(OMX_PORT_PARAM_TYPE);
3819 imagePortParamType->nPorts = 0;
3820 imagePortParamType->nStartPortNumber = 0;
3821 break;
3822
3823 }
3824 /*Component should support this port definition*/
3825 case OMX_IndexParamOtherInit: {
3826 DEBUG_PRINT_ERROR("get_parameter: OMX_IndexParamOtherInit %08x",
3827 paramIndex);
3828 eRet =OMX_ErrorUnsupportedIndex;
3829 break;
3830 }
3831 case OMX_IndexParamStandardComponentRole: {
3832 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PARAM_COMPONENTROLETYPE);
3833 OMX_PARAM_COMPONENTROLETYPE *comp_role;
3834 comp_role = (OMX_PARAM_COMPONENTROLETYPE *) paramData;
3835 comp_role->nVersion.nVersion = OMX_SPEC_VERSION;
3836 comp_role->nSize = sizeof(*comp_role);
3837
3838 DEBUG_PRINT_LOW("Getparameter: OMX_IndexParamStandardComponentRole %d",
3839 paramIndex);
3840 strlcpy((char*)comp_role->cRole,(const char*)m_cRole,
3841 OMX_MAX_STRINGNAME_SIZE);
3842 break;
3843 }
3844 /* Added for parameter test */
3845 case OMX_IndexParamPriorityMgmt: {
3846 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PRIORITYMGMTTYPE);
3847 OMX_PRIORITYMGMTTYPE *priorityMgmType =
3848 (OMX_PRIORITYMGMTTYPE *) paramData;
3849 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamPriorityMgmt");
3850 priorityMgmType->nVersion.nVersion = OMX_SPEC_VERSION;
3851 priorityMgmType->nSize = sizeof(OMX_PRIORITYMGMTTYPE);
3852
3853 break;
3854 }
3855 /* Added for parameter test */
3856 case OMX_IndexParamCompBufferSupplier: {
3857 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PARAM_BUFFERSUPPLIERTYPE);
3858 OMX_PARAM_BUFFERSUPPLIERTYPE *bufferSupplierType =
3859 (OMX_PARAM_BUFFERSUPPLIERTYPE*) paramData;
3860 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamCompBufferSupplier");
3861
3862 bufferSupplierType->nSize = sizeof(OMX_PARAM_BUFFERSUPPLIERTYPE);
3863 bufferSupplierType->nVersion.nVersion = OMX_SPEC_VERSION;
3864 if (0 == bufferSupplierType->nPortIndex)
3865 bufferSupplierType->nPortIndex = OMX_BufferSupplyUnspecified;
3866 else if (1 == bufferSupplierType->nPortIndex)
3867 bufferSupplierType->nPortIndex = OMX_BufferSupplyUnspecified;
3868 else
3869 eRet = OMX_ErrorBadPortIndex;
3870
3871
3872 break;
3873 }
3874 case OMX_IndexParamVideoAvc: {
3875 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoAvc %08x",
3876 paramIndex);
3877 break;
3878 }
3879 case (OMX_INDEXTYPE)QOMX_IndexParamVideoMvc: {
3880 DEBUG_PRINT_LOW("get_parameter: QOMX_IndexParamVideoMvc %08x",
3881 paramIndex);
3882 break;
3883 }
3884 case OMX_IndexParamVideoH263: {
3885 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoH263 %08x",
3886 paramIndex);
3887 break;
3888 }
3889 case OMX_IndexParamVideoMpeg4: {
3890 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoMpeg4 %08x",
3891 paramIndex);
3892 break;
3893 }
3894 case OMX_IndexParamVideoMpeg2: {
3895 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoMpeg2 %08x",
3896 paramIndex);
3897 break;
3898 }
3899 case OMX_IndexParamVideoProfileLevelQuerySupported: {
3900 VALIDATE_OMX_PARAM_DATA(paramData, OMX_VIDEO_PARAM_PROFILELEVELTYPE);
3901 DEBUG_PRINT_LOW("get_parameter: OMX_IndexParamVideoProfileLevelQuerySupported %08x", paramIndex);
3902 OMX_VIDEO_PARAM_PROFILELEVELTYPE *profileLevelType =
3903 (OMX_VIDEO_PARAM_PROFILELEVELTYPE *)paramData;
3904 eRet = get_supported_profile_level(profileLevelType);
3905 break;
3906 }
3907 #if defined (_ANDROID_HONEYCOMB_) || defined (_ANDROID_ICS_)
3908 case OMX_GoogleAndroidIndexGetAndroidNativeBufferUsage: {
3909 VALIDATE_OMX_PARAM_DATA(paramData, GetAndroidNativeBufferUsageParams);
3910 DEBUG_PRINT_LOW("get_parameter: OMX_GoogleAndroidIndexGetAndroidNativeBufferUsage");
3911 GetAndroidNativeBufferUsageParams* nativeBuffersUsage = (GetAndroidNativeBufferUsageParams *) paramData;
3912 if (nativeBuffersUsage->nPortIndex == OMX_CORE_OUTPUT_PORT_INDEX) {
3913
3914 if (secure_mode && !secure_scaling_to_non_secure_opb) {
3915 nativeBuffersUsage->nUsage = (GRALLOC_USAGE_PRIVATE_MM_HEAP | GRALLOC_USAGE_PROTECTED |
3916 GRALLOC_USAGE_PRIVATE_UNCACHED);
3917 } else {
3918 nativeBuffersUsage->nUsage = GRALLOC_USAGE_PRIVATE_UNCACHED;
3919 }
3920 } else {
3921 DEBUG_PRINT_HIGH("get_parameter: OMX_GoogleAndroidIndexGetAndroidNativeBufferUsage failed!");
3922 eRet = OMX_ErrorBadParameter;
3923 }
3924 }
3925 break;
3926 #endif
3927
3928 #ifdef FLEXYUV_SUPPORTED
3929 case OMX_QcomIndexFlexibleYUVDescription: {
3930 DEBUG_PRINT_LOW("get_parameter: describeColorFormat");
3931 VALIDATE_OMX_PARAM_DATA(paramData, DescribeColorFormatParams);
3932 eRet = describeColorFormat(paramData);
3933 break;
3934 }
3935 #endif
3936 case OMX_IndexParamVideoProfileLevelCurrent: {
3937 VALIDATE_OMX_PARAM_DATA(paramData, OMX_VIDEO_PARAM_PROFILELEVELTYPE);
3938 OMX_VIDEO_PARAM_PROFILELEVELTYPE* pParam = (OMX_VIDEO_PARAM_PROFILELEVELTYPE*)paramData;
3939 struct v4l2_control profile_control, level_control;
3940
3941 switch (drv_ctx.decoder_format) {
3942 case VDEC_CODECTYPE_H264:
3943 profile_control.id = V4L2_CID_MPEG_VIDEO_H264_PROFILE;
3944 level_control.id = V4L2_CID_MPEG_VIDEO_H264_LEVEL;
3945 break;
3946 default:
3947 DEBUG_PRINT_ERROR("get_param of OMX_IndexParamVideoProfileLevelCurrent only available for H264");
3948 eRet = OMX_ErrorNotImplemented;
3949 break;
3950 }
3951
3952 if (!eRet && !ioctl(drv_ctx.video_driver_fd, VIDIOC_G_CTRL, &profile_control)) {
3953 switch ((enum v4l2_mpeg_video_h264_profile)profile_control.value) {
3954 case V4L2_MPEG_VIDEO_H264_PROFILE_BASELINE:
3955 case V4L2_MPEG_VIDEO_H264_PROFILE_CONSTRAINED_BASELINE:
3956 pParam->eProfile = OMX_VIDEO_AVCProfileBaseline;
3957 break;
3958 case V4L2_MPEG_VIDEO_H264_PROFILE_MAIN:
3959 pParam->eProfile = OMX_VIDEO_AVCProfileMain;
3960 break;
3961 case V4L2_MPEG_VIDEO_H264_PROFILE_EXTENDED:
3962 pParam->eProfile = OMX_VIDEO_AVCProfileExtended;
3963 break;
3964 case V4L2_MPEG_VIDEO_H264_PROFILE_HIGH:
3965 pParam->eProfile = OMX_VIDEO_AVCProfileHigh;
3966 break;
3967 case V4L2_MPEG_VIDEO_H264_PROFILE_HIGH_10:
3968 pParam->eProfile = OMX_VIDEO_AVCProfileHigh10;
3969 break;
3970 case V4L2_MPEG_VIDEO_H264_PROFILE_HIGH_422:
3971 pParam->eProfile = OMX_VIDEO_AVCProfileHigh422;
3972 break;
3973 case V4L2_MPEG_VIDEO_H264_PROFILE_HIGH_444_PREDICTIVE:
3974 case V4L2_MPEG_VIDEO_H264_PROFILE_HIGH_10_INTRA:
3975 case V4L2_MPEG_VIDEO_H264_PROFILE_HIGH_422_INTRA:
3976 case V4L2_MPEG_VIDEO_H264_PROFILE_HIGH_444_INTRA:
3977 case V4L2_MPEG_VIDEO_H264_PROFILE_CAVLC_444_INTRA:
3978 case V4L2_MPEG_VIDEO_H264_PROFILE_SCALABLE_BASELINE:
3979 case V4L2_MPEG_VIDEO_H264_PROFILE_SCALABLE_HIGH:
3980 case V4L2_MPEG_VIDEO_H264_PROFILE_SCALABLE_HIGH_INTRA:
3981 case V4L2_MPEG_VIDEO_H264_PROFILE_STEREO_HIGH:
3982 case V4L2_MPEG_VIDEO_H264_PROFILE_MULTIVIEW_HIGH:
3983 case V4L2_MPEG_VIDEO_H264_PROFILE_CONSTRAINED_HIGH:
3984 eRet = OMX_ErrorUnsupportedIndex;
3985 break;
3986 }
3987 } else {
3988 eRet = OMX_ErrorUnsupportedIndex;
3989 }
3990
3991
3992 if (!eRet && !ioctl(drv_ctx.video_driver_fd, VIDIOC_G_CTRL, &level_control)) {
3993 switch ((enum v4l2_mpeg_video_h264_level)level_control.value) {
3994 case V4L2_MPEG_VIDEO_H264_LEVEL_1_0:
3995 pParam->eLevel = OMX_VIDEO_AVCLevel1;
3996 break;
3997 case V4L2_MPEG_VIDEO_H264_LEVEL_1B:
3998 pParam->eLevel = OMX_VIDEO_AVCLevel1b;
3999 break;
4000 case V4L2_MPEG_VIDEO_H264_LEVEL_1_1:
4001 pParam->eLevel = OMX_VIDEO_AVCLevel11;
4002 break;
4003 case V4L2_MPEG_VIDEO_H264_LEVEL_1_2:
4004 pParam->eLevel = OMX_VIDEO_AVCLevel12;
4005 break;
4006 case V4L2_MPEG_VIDEO_H264_LEVEL_1_3:
4007 pParam->eLevel = OMX_VIDEO_AVCLevel13;
4008 break;
4009 case V4L2_MPEG_VIDEO_H264_LEVEL_2_0:
4010 pParam->eLevel = OMX_VIDEO_AVCLevel2;
4011 break;
4012 case V4L2_MPEG_VIDEO_H264_LEVEL_2_1:
4013 pParam->eLevel = OMX_VIDEO_AVCLevel21;
4014 break;
4015 case V4L2_MPEG_VIDEO_H264_LEVEL_2_2:
4016 pParam->eLevel = OMX_VIDEO_AVCLevel22;
4017 break;
4018 case V4L2_MPEG_VIDEO_H264_LEVEL_3_0:
4019 pParam->eLevel = OMX_VIDEO_AVCLevel3;
4020 break;
4021 case V4L2_MPEG_VIDEO_H264_LEVEL_3_1:
4022 pParam->eLevel = OMX_VIDEO_AVCLevel31;
4023 break;
4024 case V4L2_MPEG_VIDEO_H264_LEVEL_3_2:
4025 pParam->eLevel = OMX_VIDEO_AVCLevel32;
4026 break;
4027 case V4L2_MPEG_VIDEO_H264_LEVEL_4_0:
4028 pParam->eLevel = OMX_VIDEO_AVCLevel4;
4029 break;
4030 case V4L2_MPEG_VIDEO_H264_LEVEL_4_1:
4031 pParam->eLevel = OMX_VIDEO_AVCLevel41;
4032 break;
4033 case V4L2_MPEG_VIDEO_H264_LEVEL_4_2:
4034 pParam->eLevel = OMX_VIDEO_AVCLevel42;
4035 break;
4036 case V4L2_MPEG_VIDEO_H264_LEVEL_5_0:
4037 pParam->eLevel = OMX_VIDEO_AVCLevel5;
4038 break;
4039 case V4L2_MPEG_VIDEO_H264_LEVEL_5_1:
4040 pParam->eLevel = OMX_VIDEO_AVCLevel51;
4041 break;
4042 case V4L2_MPEG_VIDEO_H264_LEVEL_5_2:
4043 pParam->eLevel = OMX_VIDEO_AVCLevel52;
4044 break;
4045 }
4046 } else {
4047 eRet = OMX_ErrorUnsupportedIndex;
4048 }
4049
4050 break;
4051
4052 }
4053 case OMX_QTIIndexParamVideoClientExtradata:
4054 {
4055 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_EXTRADATA_ENABLE);
4056 DEBUG_PRINT_LOW("get_parameter: OMX_QTIIndexParamVideoClientExtradata");
4057 QOMX_EXTRADATA_ENABLE *pParam =
4058 (QOMX_EXTRADATA_ENABLE *)paramData;
4059 if (pParam->nPortIndex == OMX_CORE_OUTPUT_EXTRADATA_INDEX) {
4060 pParam->bEnable = client_extradata ? OMX_TRUE : OMX_FALSE;
4061 eRet = OMX_ErrorNone;
4062 } else {
4063 eRet = OMX_ErrorUnsupportedIndex;
4064 }
4065 break;
4066 }
4067 case OMX_QTIIndexParamDitherControl:
4068 {
4069 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_VIDEO_DITHER_CONTROL);
4070 DEBUG_PRINT_LOW("get_parameter: QOMX_VIDEO_DITHER_CONTROL");
4071 QOMX_VIDEO_DITHER_CONTROL *pParam =
4072 (QOMX_VIDEO_DITHER_CONTROL *) paramData;
4073 pParam->eDitherType = (QOMX_VIDEO_DITHERTYPE) m_dither_config;
4074 eRet = OMX_ErrorNone;
4075 break;
4076 }
4077 default: {
4078 DEBUG_PRINT_ERROR("get_parameter: unknown param %08x", paramIndex);
4079 eRet =OMX_ErrorUnsupportedIndex;
4080 }
4081
4082 }
4083
4084 DEBUG_PRINT_LOW("get_parameter returning WxH(%d x %d) SxSH(%d x %d)",
4085 drv_ctx.video_resolution.frame_width,
4086 drv_ctx.video_resolution.frame_height,
4087 drv_ctx.video_resolution.stride,
4088 drv_ctx.video_resolution.scan_lines);
4089
4090 return eRet;
4091 }
4092
4093 #if defined (_ANDROID_HONEYCOMB_) || defined (_ANDROID_ICS_)
use_android_native_buffer(OMX_IN OMX_HANDLETYPE hComp,OMX_PTR data)4094 OMX_ERRORTYPE omx_vdec::use_android_native_buffer(OMX_IN OMX_HANDLETYPE hComp, OMX_PTR data)
4095 {
4096 DEBUG_PRINT_LOW("Inside use_android_native_buffer");
4097 OMX_ERRORTYPE eRet = OMX_ErrorNone;
4098 UseAndroidNativeBufferParams *params = (UseAndroidNativeBufferParams *)data;
4099
4100 if ((params == NULL) ||
4101 (params->nativeBuffer == NULL) ||
4102 (params->nativeBuffer->handle == NULL) ||
4103 !m_enable_android_native_buffers)
4104 return OMX_ErrorBadParameter;
4105 m_use_android_native_buffers = OMX_TRUE;
4106 sp<android_native_buffer_t> nBuf = params->nativeBuffer;
4107 private_handle_t *handle = (private_handle_t *)nBuf->handle;
4108 if (OMX_CORE_OUTPUT_PORT_INDEX == params->nPortIndex) { //android native buffers can be used only on Output port
4109 OMX_U8 *buffer = NULL;
4110 if (!secure_mode) {
4111 buffer = (OMX_U8*)mmap(0, handle->size,
4112 PROT_READ|PROT_WRITE, MAP_SHARED, handle->fd, 0);
4113 if (buffer == MAP_FAILED) {
4114 DEBUG_PRINT_ERROR("Failed to mmap pmem with fd = %d, size = %d", handle->fd, handle->size);
4115 return OMX_ErrorInsufficientResources;
4116 }
4117 }
4118 eRet = use_buffer(hComp,params->bufferHeader,params->nPortIndex,data,handle->size,buffer);
4119 } else {
4120 eRet = OMX_ErrorBadParameter;
4121 }
4122 return eRet;
4123 }
4124 #endif
4125
enable_smoothstreaming()4126 OMX_ERRORTYPE omx_vdec::enable_smoothstreaming() {
4127 struct v4l2_control control;
4128 struct v4l2_format fmt;
4129 control.id = V4L2_CID_MPEG_VIDC_VIDEO_CONTINUE_DATA_TRANSFER;
4130 control.value = 1;
4131 int rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL,&control);
4132 if (rc < 0) {
4133 DEBUG_PRINT_ERROR("Failed to enable Smooth Streaming on driver.");
4134 return OMX_ErrorHardware;
4135 }
4136 m_smoothstreaming_mode = true;
4137 return OMX_ErrorNone;
4138 }
4139
4140 /* ======================================================================
4141 FUNCTION
4142 omx_vdec::Setparameter
4143
4144 DESCRIPTION
4145 OMX Set Parameter method implementation.
4146
4147 PARAMETERS
4148 <TBD>.
4149
4150 RETURN VALUE
4151 OMX Error None if successful.
4152
4153 ========================================================================== */
set_parameter(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_INDEXTYPE paramIndex,OMX_IN OMX_PTR paramData)4154 OMX_ERRORTYPE omx_vdec::set_parameter(OMX_IN OMX_HANDLETYPE hComp,
4155 OMX_IN OMX_INDEXTYPE paramIndex,
4156 OMX_IN OMX_PTR paramData)
4157 {
4158 OMX_ERRORTYPE eRet = OMX_ErrorNone;
4159 int ret=0;
4160 struct v4l2_format fmt;
4161 #ifdef _ANDROID_
4162 char property_value[PROPERTY_VALUE_MAX] = {0};
4163 #endif
4164 if (m_state == OMX_StateInvalid) {
4165 DEBUG_PRINT_ERROR("Set Param in Invalid State");
4166 return OMX_ErrorInvalidState;
4167 }
4168 if (paramData == NULL) {
4169 DEBUG_PRINT_ERROR("Get Param in Invalid paramData");
4170 return OMX_ErrorBadParameter;
4171 }
4172 if ((m_state != OMX_StateLoaded) &&
4173 BITMASK_ABSENT(&m_flags,OMX_COMPONENT_OUTPUT_ENABLE_PENDING) &&
4174 (m_out_bEnabled == OMX_TRUE) &&
4175 BITMASK_ABSENT(&m_flags, OMX_COMPONENT_INPUT_ENABLE_PENDING) &&
4176 (m_inp_bEnabled == OMX_TRUE)) {
4177 DEBUG_PRINT_ERROR("Set Param in Invalid State");
4178 return OMX_ErrorIncorrectStateOperation;
4179 }
4180 switch ((unsigned long)paramIndex) {
4181 case OMX_IndexParamPortDefinition: {
4182 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PARAM_PORTDEFINITIONTYPE);
4183 OMX_PARAM_PORTDEFINITIONTYPE *portDefn;
4184 portDefn = (OMX_PARAM_PORTDEFINITIONTYPE *) paramData;
4185 //TODO: Check if any allocate buffer/use buffer/useNativeBuffer has
4186 //been called.
4187 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamPortDefinition H= %d, W = %d",
4188 (int)portDefn->format.video.nFrameHeight,
4189 (int)portDefn->format.video.nFrameWidth);
4190
4191 if (portDefn->nBufferCountActual > MAX_NUM_INPUT_OUTPUT_BUFFERS) {
4192 DEBUG_PRINT_ERROR("ERROR: Buffers requested exceeds max limit %d",
4193 portDefn->nBufferCountActual);
4194 eRet = OMX_ErrorBadParameter;
4195 break;
4196 }
4197 if (OMX_CORE_OUTPUT_EXTRADATA_INDEX == portDefn->nPortIndex) {
4198 if (portDefn->nBufferCountActual < MIN_NUM_INPUT_OUTPUT_EXTRADATA_BUFFERS ||
4199 portDefn->nBufferSize != m_client_out_extradata_info.getSize()) {
4200 DEBUG_PRINT_ERROR("ERROR: Bad parameeters request for extradata limit %d size - %d",
4201 portDefn->nBufferCountActual, portDefn->nBufferSize);
4202 eRet = OMX_ErrorBadParameter;
4203 break;
4204 }
4205 m_client_out_extradata_info.set_extradata_info(portDefn->nBufferSize,
4206 portDefn->nBufferCountActual);
4207 break;
4208 }
4209
4210 if (OMX_DirOutput == portDefn->eDir) {
4211 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamPortDefinition OP port");
4212 bool port_format_changed = false;
4213 m_display_id = portDefn->format.video.pNativeWindow;
4214 unsigned int buffer_size;
4215 /* update output port resolution with client supplied dimensions
4216 in case scaling is enabled, else it follows input resolution set
4217 */
4218 decide_dpb_buffer_mode(is_down_scalar_enabled);
4219 if (is_down_scalar_enabled) {
4220 DEBUG_PRINT_LOW("SetParam OP: WxH(%u x %u)",
4221 (unsigned int)portDefn->format.video.nFrameWidth,
4222 (unsigned int)portDefn->format.video.nFrameHeight);
4223 if (portDefn->format.video.nFrameHeight != 0x0 &&
4224 portDefn->format.video.nFrameWidth != 0x0) {
4225 memset(&fmt, 0x0, sizeof(struct v4l2_format));
4226 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
4227 fmt.fmt.pix_mp.pixelformat = capture_capability;
4228 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
4229 if (ret) {
4230 DEBUG_PRINT_ERROR("Get Resolution failed");
4231 eRet = OMX_ErrorHardware;
4232 break;
4233 }
4234 if ((portDefn->format.video.nFrameHeight != (unsigned int)fmt.fmt.pix_mp.height) ||
4235 (portDefn->format.video.nFrameWidth != (unsigned int)fmt.fmt.pix_mp.width)) {
4236 port_format_changed = true;
4237 }
4238
4239 /* set crop info */
4240 rectangle.nLeft = 0;
4241 rectangle.nTop = 0;
4242 rectangle.nWidth = portDefn->format.video.nFrameWidth;
4243 rectangle.nHeight = portDefn->format.video.nFrameHeight;
4244
4245 m_extradata_info.output_crop_rect.nLeft = 0;
4246 m_extradata_info.output_crop_rect.nTop = 0;
4247 m_extradata_info.output_crop_rect.nWidth = rectangle.nWidth;
4248 m_extradata_info.output_crop_rect.nHeight = rectangle.nHeight;
4249
4250 eRet = is_video_session_supported();
4251 if (eRet)
4252 break;
4253 memset(&fmt, 0x0, sizeof(struct v4l2_format));
4254 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
4255 fmt.fmt.pix_mp.height = (unsigned int)portDefn->format.video.nFrameHeight;
4256 fmt.fmt.pix_mp.width = (unsigned int)portDefn->format.video.nFrameWidth;
4257 fmt.fmt.pix_mp.pixelformat = capture_capability;
4258 DEBUG_PRINT_LOW("fmt.fmt.pix_mp.height = %d , fmt.fmt.pix_mp.width = %d",
4259 fmt.fmt.pix_mp.height, fmt.fmt.pix_mp.width);
4260 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
4261 if (ret) {
4262 DEBUG_PRINT_ERROR("Set Resolution failed");
4263 eRet = OMX_ErrorUnsupportedSetting;
4264 } else
4265 eRet = get_buffer_req(&drv_ctx.op_buf);
4266 }
4267
4268 if (eRet) {
4269 break;
4270 }
4271
4272 if (secure_mode) {
4273 struct v4l2_control control;
4274 control.id = V4L2_CID_MPEG_VIDC_VIDEO_SECURE_SCALING_THRESHOLD;
4275 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_G_CTRL, &control) < 0) {
4276 DEBUG_PRINT_ERROR("Failed getting secure scaling threshold : %d, id was : %x", errno, control.id);
4277 eRet = OMX_ErrorHardware;
4278 } else {
4279 /* This is a workaround for a bug in fw which uses stride
4280 * and slice instead of width and height to check against
4281 * the threshold.
4282 */
4283 OMX_U32 stride, slice;
4284 if (drv_ctx.output_format == VDEC_YUV_FORMAT_NV12) {
4285 stride = VENUS_Y_STRIDE(COLOR_FMT_NV12, portDefn->format.video.nFrameWidth);
4286 slice = VENUS_Y_SCANLINES(COLOR_FMT_NV12, portDefn->format.video.nFrameHeight);
4287 } else if (drv_ctx.output_format == VDEC_YUV_FORMAT_NV12_UBWC) {
4288 stride = VENUS_Y_STRIDE(COLOR_FMT_NV12_UBWC, portDefn->format.video.nFrameWidth);
4289 slice = VENUS_Y_SCANLINES(COLOR_FMT_NV12_UBWC, portDefn->format.video.nFrameHeight);
4290 } else if (drv_ctx.output_format == VDEC_YUV_FORMAT_NV12_TP10_UBWC) {
4291 stride = VENUS_Y_STRIDE(COLOR_FMT_NV12_BPP10_UBWC, portDefn->format.video.nFrameWidth);
4292 slice = VENUS_Y_SCANLINES(COLOR_FMT_NV12_BPP10_UBWC, portDefn->format.video.nFrameHeight);
4293 } else {
4294 stride = portDefn->format.video.nFrameWidth;
4295 slice = portDefn->format.video.nFrameHeight;
4296 }
4297
4298 DEBUG_PRINT_LOW("Stride is %d, slice is %d, sxs is %d\n", stride, slice, stride * slice);
4299 DEBUG_PRINT_LOW("Threshold value is %d\n", control.value);
4300
4301 if (stride * slice <= (OMX_U32)control.value) {
4302 secure_scaling_to_non_secure_opb = true;
4303 DEBUG_PRINT_HIGH("Enabling secure scalar out of CPZ");
4304 control.id = V4L2_CID_MPEG_VIDC_VIDEO_NON_SECURE_OUTPUT2;
4305 control.value = 1;
4306 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control) < 0) {
4307 DEBUG_PRINT_ERROR("Enabling non-secure output2 failed");
4308 eRet = OMX_ErrorUnsupportedSetting;
4309 }
4310 }
4311 }
4312 }
4313 }
4314
4315 if (eRet) {
4316 break;
4317 }
4318
4319 if (portDefn->nBufferCountActual > MAX_NUM_INPUT_OUTPUT_BUFFERS) {
4320 DEBUG_PRINT_ERROR("Requested o/p buf count (%u) exceeds limit (%u)",
4321 portDefn->nBufferCountActual, MAX_NUM_INPUT_OUTPUT_BUFFERS);
4322 eRet = OMX_ErrorBadParameter;
4323 } else if (!client_buffers.get_buffer_req(buffer_size)) {
4324 DEBUG_PRINT_ERROR("Error in getting buffer requirements");
4325 eRet = OMX_ErrorBadParameter;
4326 } else if (!port_format_changed) {
4327
4328 // Buffer count can change only when port is unallocated
4329 if (m_out_mem_ptr &&
4330 (portDefn->nBufferCountActual != drv_ctx.op_buf.actualcount ||
4331 portDefn->nBufferSize != drv_ctx.op_buf.buffer_size)) {
4332
4333 DEBUG_PRINT_ERROR("Cannot change o/p buffer count since all buffers are not freed yet !");
4334 eRet = OMX_ErrorInvalidState;
4335 break;
4336 }
4337
4338 // route updating of buffer requirements via c2d proxy.
4339 // Based on whether c2d is enabled, requirements will be handed
4340 // to the vidc driver appropriately
4341 eRet = client_buffers.set_buffer_req(portDefn->nBufferSize,
4342 portDefn->nBufferCountActual);
4343 if (eRet == OMX_ErrorNone) {
4344 m_port_def = *portDefn;
4345 } else {
4346 DEBUG_PRINT_ERROR("ERROR: OP Requirements(#%d: %u) Requested(#%u: %u)",
4347 drv_ctx.op_buf.mincount, (unsigned int)buffer_size,
4348 (unsigned int)portDefn->nBufferCountActual, (unsigned int)portDefn->nBufferSize);
4349 eRet = OMX_ErrorBadParameter;
4350 }
4351 }
4352 } else if (OMX_DirInput == portDefn->eDir) {
4353 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamPortDefinition IP port");
4354 bool port_format_changed = false;
4355 if ((portDefn->format.video.xFramerate >> 16) > 0 &&
4356 (portDefn->format.video.xFramerate >> 16) <= MAX_SUPPORTED_FPS) {
4357 // Frame rate only should be set if this is a "known value" or to
4358 // activate ts prediction logic (arbitrary mode only) sending input
4359 // timestamps with max value (LLONG_MAX).
4360 m_fps_received = portDefn->format.video.xFramerate;
4361 DEBUG_PRINT_HIGH("set_parameter: frame rate set by omx client : %u",
4362 (unsigned int)portDefn->format.video.xFramerate >> 16);
4363 Q16ToFraction(portDefn->format.video.xFramerate, drv_ctx.frame_rate.fps_numerator,
4364 drv_ctx.frame_rate.fps_denominator);
4365 if (!drv_ctx.frame_rate.fps_numerator) {
4366 DEBUG_PRINT_ERROR("Numerator is zero setting to 30");
4367 drv_ctx.frame_rate.fps_numerator = 30;
4368 }
4369 if (drv_ctx.frame_rate.fps_denominator)
4370 drv_ctx.frame_rate.fps_numerator = (int)
4371 drv_ctx.frame_rate.fps_numerator / drv_ctx.frame_rate.fps_denominator;
4372 drv_ctx.frame_rate.fps_denominator = 1;
4373 frm_int = drv_ctx.frame_rate.fps_denominator * 1e6 /
4374 drv_ctx.frame_rate.fps_numerator;
4375 DEBUG_PRINT_LOW("set_parameter: frm_int(%u) fps(%.2f)",
4376 (unsigned int)frm_int, drv_ctx.frame_rate.fps_numerator /
4377 (float)drv_ctx.frame_rate.fps_denominator);
4378
4379 struct v4l2_outputparm oparm;
4380 /*XXX: we're providing timing info as seconds per frame rather than frames
4381 * per second.*/
4382 oparm.timeperframe.numerator = drv_ctx.frame_rate.fps_denominator;
4383 oparm.timeperframe.denominator = drv_ctx.frame_rate.fps_numerator;
4384
4385 struct v4l2_streamparm sparm;
4386 sparm.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
4387 sparm.parm.output = oparm;
4388 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_PARM, &sparm)) {
4389 DEBUG_PRINT_ERROR("Unable to convey fps info to driver, performance might be affected");
4390 eRet = OMX_ErrorHardware;
4391 break;
4392 }
4393 m_perf_control.request_cores(frm_int);
4394 }
4395
4396 if (drv_ctx.video_resolution.frame_height !=
4397 portDefn->format.video.nFrameHeight ||
4398 drv_ctx.video_resolution.frame_width !=
4399 portDefn->format.video.nFrameWidth) {
4400 DEBUG_PRINT_LOW("SetParam IP: WxH(%u x %u)",
4401 (unsigned int)portDefn->format.video.nFrameWidth,
4402 (unsigned int)portDefn->format.video.nFrameHeight);
4403 port_format_changed = true;
4404 OMX_U32 frameWidth = portDefn->format.video.nFrameWidth;
4405 OMX_U32 frameHeight = portDefn->format.video.nFrameHeight;
4406 if (frameHeight != 0x0 && frameWidth != 0x0) {
4407 if (m_smoothstreaming_mode &&
4408 ((frameWidth * frameHeight) <
4409 (m_smoothstreaming_width * m_smoothstreaming_height))) {
4410 frameWidth = m_smoothstreaming_width;
4411 frameHeight = m_smoothstreaming_height;
4412 DEBUG_PRINT_LOW("NOTE: Setting resolution %u x %u "
4413 "for adaptive-playback/smooth-streaming",
4414 (unsigned int)frameWidth, (unsigned int)frameHeight);
4415 }
4416
4417 m_extradata_info.output_crop_rect.nLeft = 0;
4418 m_extradata_info.output_crop_rect.nTop = 0;
4419 m_extradata_info.output_crop_rect.nWidth = frameWidth;
4420 m_extradata_info.output_crop_rect.nHeight = frameHeight;
4421
4422 update_resolution(frameWidth, frameHeight,
4423 frameWidth, frameHeight);
4424 eRet = is_video_session_supported();
4425 if (eRet)
4426 break;
4427 if (is_down_scalar_enabled) {
4428 memset(&fmt, 0x0, sizeof(struct v4l2_format));
4429 fmt.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
4430 fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
4431 fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
4432 fmt.fmt.pix_mp.pixelformat = output_capability;
4433 DEBUG_PRINT_LOW("DS Enabled : height = %d , width = %d",
4434 fmt.fmt.pix_mp.height,fmt.fmt.pix_mp.width);
4435 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
4436 } else {
4437 memset(&fmt, 0x0, sizeof(struct v4l2_format));
4438 fmt.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
4439 fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
4440 fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
4441 fmt.fmt.pix_mp.pixelformat = output_capability;
4442 DEBUG_PRINT_LOW("DS Disabled : height = %d , width = %d",
4443 fmt.fmt.pix_mp.height,fmt.fmt.pix_mp.width);
4444 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
4445 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
4446 fmt.fmt.pix_mp.pixelformat = capture_capability;
4447 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
4448 }
4449 if (ret) {
4450 DEBUG_PRINT_ERROR("Set Resolution failed");
4451 eRet = OMX_ErrorUnsupportedSetting;
4452 } else {
4453 if (!is_down_scalar_enabled)
4454 eRet = get_buffer_req(&drv_ctx.op_buf);
4455 }
4456 }
4457 }
4458 if (m_custom_buffersize.input_buffersize
4459 && (portDefn->nBufferSize > m_custom_buffersize.input_buffersize)) {
4460 DEBUG_PRINT_ERROR("ERROR: Custom buffer size set by client: %d, trying to set: %d",
4461 m_custom_buffersize.input_buffersize, portDefn->nBufferSize);
4462 eRet = OMX_ErrorBadParameter;
4463 break;
4464 }
4465 if (portDefn->nBufferCountActual > MAX_NUM_INPUT_OUTPUT_BUFFERS) {
4466 DEBUG_PRINT_ERROR("Requested i/p buf count (%u) exceeds limit (%u)",
4467 portDefn->nBufferCountActual, MAX_NUM_INPUT_OUTPUT_BUFFERS);
4468 eRet = OMX_ErrorBadParameter;
4469 break;
4470 }
4471 // Buffer count can change only when port is unallocated
4472 if (m_inp_mem_ptr &&
4473 (portDefn->nBufferCountActual != drv_ctx.ip_buf.actualcount ||
4474 portDefn->nBufferSize != drv_ctx.ip_buf.buffer_size)) {
4475 DEBUG_PRINT_ERROR("Cannot change i/p buffer count since all buffers are not freed yet !");
4476 eRet = OMX_ErrorInvalidState;
4477 break;
4478 }
4479
4480 if (portDefn->nBufferCountActual >= drv_ctx.ip_buf.mincount
4481 || portDefn->nBufferSize != drv_ctx.ip_buf.buffer_size) {
4482 port_format_changed = true;
4483 vdec_allocatorproperty *buffer_prop = &drv_ctx.ip_buf;
4484 drv_ctx.ip_buf.actualcount = portDefn->nBufferCountActual;
4485 drv_ctx.ip_buf.buffer_size = (portDefn->nBufferSize + buffer_prop->alignment - 1) &
4486 (~(buffer_prop->alignment - 1));
4487 eRet = set_buffer_req(buffer_prop);
4488 }
4489 if (false == port_format_changed) {
4490 DEBUG_PRINT_ERROR("ERROR: IP Requirements(#%d: %u) Requested(#%u: %u)",
4491 drv_ctx.ip_buf.mincount, (unsigned int)drv_ctx.ip_buf.buffer_size,
4492 (unsigned int)portDefn->nBufferCountActual, (unsigned int)portDefn->nBufferSize);
4493 eRet = OMX_ErrorBadParameter;
4494 }
4495 } else if (portDefn->eDir == OMX_DirMax) {
4496 DEBUG_PRINT_ERROR(" Set_parameter: Bad Port idx %d",
4497 (int)portDefn->nPortIndex);
4498 eRet = OMX_ErrorBadPortIndex;
4499 }
4500 }
4501 break;
4502 case OMX_IndexParamVideoPortFormat: {
4503 VALIDATE_OMX_PARAM_DATA(paramData, OMX_VIDEO_PARAM_PORTFORMATTYPE);
4504 OMX_VIDEO_PARAM_PORTFORMATTYPE *portFmt =
4505 (OMX_VIDEO_PARAM_PORTFORMATTYPE *)paramData;
4506 int ret=0;
4507 struct v4l2_format fmt;
4508 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamVideoPortFormat 0x%x, port: %u",
4509 portFmt->eColorFormat, (unsigned int)portFmt->nPortIndex);
4510
4511 memset(&fmt, 0x0, sizeof(struct v4l2_format));
4512 if (1 == portFmt->nPortIndex) {
4513 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
4514 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
4515 if (ret < 0) {
4516 DEBUG_PRINT_ERROR("%s: Failed to get format on capture mplane", __func__);
4517 return OMX_ErrorBadParameter;
4518 }
4519 enum vdec_output_fromat op_format;
4520 if (portFmt->eColorFormat == (OMX_COLOR_FORMATTYPE)
4521 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m ||
4522 portFmt->eColorFormat == (OMX_COLOR_FORMATTYPE)
4523 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mMultiView ||
4524 portFmt->eColorFormat == OMX_COLOR_FormatYUV420Planar ||
4525 portFmt->eColorFormat == OMX_COLOR_FormatYUV420SemiPlanar) {
4526 op_format = (enum vdec_output_fromat)VDEC_YUV_FORMAT_NV12;
4527 } else if (portFmt->eColorFormat == (OMX_COLOR_FORMATTYPE)
4528 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mCompressed) {
4529 op_format = (enum vdec_output_fromat)VDEC_YUV_FORMAT_NV12_UBWC;
4530 } else
4531 eRet = OMX_ErrorBadParameter;
4532
4533 if (portFmt->eColorFormat == (OMX_COLOR_FORMATTYPE)
4534 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mCompressed) {
4535 fmt.fmt.pix_mp.pixelformat = capture_capability = V4L2_PIX_FMT_NV12_UBWC;
4536 } else {
4537 fmt.fmt.pix_mp.pixelformat = capture_capability = V4L2_PIX_FMT_NV12;
4538 }
4539
4540 if (eRet == OMX_ErrorNone) {
4541 drv_ctx.output_format = op_format;
4542 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
4543 if (ret) {
4544 DEBUG_PRINT_ERROR("Set output format failed");
4545 eRet = OMX_ErrorUnsupportedSetting;
4546 /*TODO: How to handle this case */
4547 } else {
4548 eRet = get_buffer_req(&drv_ctx.op_buf);
4549 }
4550 }
4551 if (eRet == OMX_ErrorNone) {
4552 if (!client_buffers.set_color_format(portFmt->eColorFormat)) {
4553 DEBUG_PRINT_ERROR("Set color format failed");
4554 eRet = OMX_ErrorBadParameter;
4555 }
4556 }
4557 }
4558 }
4559 break;
4560
4561 case OMX_QcomIndexPortDefn: {
4562 VALIDATE_OMX_PARAM_DATA(paramData, OMX_QCOM_PARAM_PORTDEFINITIONTYPE);
4563 OMX_QCOM_PARAM_PORTDEFINITIONTYPE *portFmt =
4564 (OMX_QCOM_PARAM_PORTDEFINITIONTYPE *) paramData;
4565 DEBUG_PRINT_LOW("set_parameter: OMX_IndexQcomParamPortDefinitionType %u",
4566 (unsigned int)portFmt->nFramePackingFormat);
4567
4568 /* Input port */
4569 if (portFmt->nPortIndex == 0) {
4570 // arbitrary_bytes mode cannot be changed arbitrarily since this controls how:
4571 // - headers are allocated and
4572 // - headers-indices are derived
4573 // Avoid changing arbitrary_bytes when the port is already allocated
4574 if (m_inp_mem_ptr) {
4575 DEBUG_PRINT_ERROR("Cannot change arbitrary-bytes-mode since input port is not free!");
4576 return OMX_ErrorUnsupportedSetting;
4577 }
4578 if (portFmt->nFramePackingFormat == OMX_QCOM_FramePacking_Arbitrary) {
4579 if (secure_mode || m_input_pass_buffer_fd) {
4580 arbitrary_bytes = false;
4581 DEBUG_PRINT_ERROR("setparameter: cannot set to arbitary bytes mode");
4582 eRet = OMX_ErrorUnsupportedSetting;
4583 } else {
4584 arbitrary_bytes = true;
4585 }
4586 } else if (portFmt->nFramePackingFormat ==
4587 OMX_QCOM_FramePacking_OnlyOneCompleteFrame) {
4588 arbitrary_bytes = false;
4589 #ifdef _ANDROID_
4590 property_get("vendor.vidc.dec.debug.arbitrarybytes.mode", property_value, "0");
4591 if (atoi(property_value)) {
4592 DEBUG_PRINT_HIGH("arbitrary_bytes enabled via property command");
4593 arbitrary_bytes = true;
4594 }
4595 #endif
4596 } else {
4597 DEBUG_PRINT_ERROR("Setparameter: unknown FramePacking format %u",
4598 (unsigned int)portFmt->nFramePackingFormat);
4599 eRet = OMX_ErrorUnsupportedSetting;
4600 }
4601 } else if (portFmt->nPortIndex == OMX_CORE_OUTPUT_PORT_INDEX) {
4602 DEBUG_PRINT_HIGH("set_parameter: OMX_IndexQcomParamPortDefinitionType OP Port");
4603 if ( (portFmt->nMemRegion > OMX_QCOM_MemRegionInvalid &&
4604 portFmt->nMemRegion < OMX_QCOM_MemRegionMax) &&
4605 portFmt->nCacheAttr == OMX_QCOM_CacheAttrNone) {
4606 m_out_mem_region_smi = OMX_TRUE;
4607 if ((m_out_mem_region_smi && m_out_pvt_entry_pmem)) {
4608 DEBUG_PRINT_HIGH("set_parameter: OMX_IndexQcomParamPortDefinitionType OP Port: out pmem set");
4609 m_use_output_pmem = OMX_TRUE;
4610 }
4611 }
4612 }
4613 }
4614 if (is_thulium_v1 && !strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc",
4615 OMX_MAX_STRINGNAME_SIZE)) {
4616 arbitrary_bytes = true;
4617 DEBUG_PRINT_HIGH("Force arbitrary_bytes to true for h264");
4618 }
4619 break;
4620
4621 case OMX_QTIIndexParamVideoClientExtradata: {
4622 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_EXTRADATA_ENABLE);
4623 DEBUG_PRINT_LOW("set_parameter: OMX_QTIIndexParamVideoClientExtradata");
4624 QOMX_EXTRADATA_ENABLE *pParam =
4625 (QOMX_EXTRADATA_ENABLE *)paramData;
4626
4627 if (m_state != OMX_StateLoaded) {
4628 DEBUG_PRINT_ERROR("Set Parameter called in Invalid state");
4629 return OMX_ErrorIncorrectStateOperation;
4630 }
4631
4632 if (pParam->nPortIndex == OMX_CORE_OUTPUT_EXTRADATA_INDEX) {
4633 m_client_out_extradata_info.enable_client_extradata(pParam->bEnable);
4634 } else {
4635 DEBUG_PRINT_ERROR("Incorrect portIndex - %d", pParam->nPortIndex);
4636 eRet = OMX_ErrorUnsupportedIndex;
4637 }
4638 break;
4639 }
4640 case OMX_IndexParamStandardComponentRole: {
4641 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PARAM_COMPONENTROLETYPE);
4642 OMX_PARAM_COMPONENTROLETYPE *comp_role;
4643 comp_role = (OMX_PARAM_COMPONENTROLETYPE *) paramData;
4644 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamStandardComponentRole %s",
4645 comp_role->cRole);
4646
4647 if ((m_state == OMX_StateLoaded)&&
4648 !BITMASK_PRESENT(&m_flags, OMX_COMPONENT_IDLE_PENDING)) {
4649 DEBUG_PRINT_LOW("Set Parameter called in valid state");
4650 } else {
4651 DEBUG_PRINT_ERROR("Set Parameter called in Invalid State");
4652 return OMX_ErrorIncorrectStateOperation;
4653 }
4654
4655 if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc", OMX_MAX_STRINGNAME_SIZE)) {
4656 if (!strncmp((char*)comp_role->cRole, "video_decoder.avc", OMX_MAX_STRINGNAME_SIZE)) {
4657 strlcpy((char*)m_cRole, "video_decoder.avc", OMX_MAX_STRINGNAME_SIZE);
4658 } else {
4659 DEBUG_PRINT_ERROR("Setparameter: unknown Index %s", comp_role->cRole);
4660 eRet =OMX_ErrorUnsupportedSetting;
4661 }
4662 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mvc", OMX_MAX_STRINGNAME_SIZE)) {
4663 if (!strncmp((char*)comp_role->cRole, "video_decoder.mvc", OMX_MAX_STRINGNAME_SIZE)) {
4664 strlcpy((char*)m_cRole, "video_decoder.mvc", OMX_MAX_STRINGNAME_SIZE);
4665 } else {
4666 DEBUG_PRINT_ERROR("Setparameter: unknown Index %s", comp_role->cRole);
4667 eRet = OMX_ErrorUnsupportedSetting;
4668 }
4669 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg4", OMX_MAX_STRINGNAME_SIZE)) {
4670 if (!strncmp((const char*)comp_role->cRole, "video_decoder.mpeg4", OMX_MAX_STRINGNAME_SIZE)) {
4671 strlcpy((char*)m_cRole, "video_decoder.mpeg4", OMX_MAX_STRINGNAME_SIZE);
4672 } else {
4673 DEBUG_PRINT_ERROR("Setparameter: unknown Index %s", comp_role->cRole);
4674 eRet = OMX_ErrorUnsupportedSetting;
4675 }
4676 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.h263", OMX_MAX_STRINGNAME_SIZE)) {
4677 if (!strncmp((const char*)comp_role->cRole, "video_decoder.h263", OMX_MAX_STRINGNAME_SIZE)) {
4678 strlcpy((char*)m_cRole, "video_decoder.h263", OMX_MAX_STRINGNAME_SIZE);
4679 } else {
4680 DEBUG_PRINT_ERROR("Setparameter: unknown Index %s", comp_role->cRole);
4681 eRet =OMX_ErrorUnsupportedSetting;
4682 }
4683 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg2", OMX_MAX_STRINGNAME_SIZE)) {
4684 if (!strncmp((const char*)comp_role->cRole, "video_decoder.mpeg2", OMX_MAX_STRINGNAME_SIZE)) {
4685 strlcpy((char*)m_cRole, "video_decoder.mpeg2", OMX_MAX_STRINGNAME_SIZE);
4686 } else {
4687 DEBUG_PRINT_ERROR("Setparameter: unknown Index %s", comp_role->cRole);
4688 eRet = OMX_ErrorUnsupportedSetting;
4689 }
4690 } else if ((!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx", OMX_MAX_STRINGNAME_SIZE)) ||
4691 (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx311", OMX_MAX_STRINGNAME_SIZE)) ||
4692 (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx4", OMX_MAX_STRINGNAME_SIZE))
4693 ) {
4694 if (!strncmp((const char*)comp_role->cRole, "video_decoder.divx", OMX_MAX_STRINGNAME_SIZE)) {
4695 strlcpy((char*)m_cRole, "video_decoder.divx", OMX_MAX_STRINGNAME_SIZE);
4696 } else {
4697 DEBUG_PRINT_ERROR("Setparameter: unknown Index %s", comp_role->cRole);
4698 eRet =OMX_ErrorUnsupportedSetting;
4699 }
4700 } else if ( (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vc1", OMX_MAX_STRINGNAME_SIZE)) ||
4701 (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.wmv", OMX_MAX_STRINGNAME_SIZE))
4702 ) {
4703 if (!strncmp((const char*)comp_role->cRole, "video_decoder.vc1", OMX_MAX_STRINGNAME_SIZE)) {
4704 strlcpy((char*)m_cRole, "video_decoder.vc1", OMX_MAX_STRINGNAME_SIZE);
4705 } else {
4706 DEBUG_PRINT_ERROR("Setparameter: unknown Index %s", comp_role->cRole);
4707 eRet =OMX_ErrorUnsupportedSetting;
4708 }
4709 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp8", OMX_MAX_STRINGNAME_SIZE)) {
4710 if (!strncmp((const char*)comp_role->cRole, "video_decoder.vp8", OMX_MAX_STRINGNAME_SIZE) ||
4711 !strncmp((const char*)comp_role->cRole, "video_decoder.vpx", OMX_MAX_STRINGNAME_SIZE)) {
4712 strlcpy((char*)m_cRole, "video_decoder.vp8", OMX_MAX_STRINGNAME_SIZE);
4713 } else {
4714 DEBUG_PRINT_ERROR("Setparameter: unknown Index %s", comp_role->cRole);
4715 eRet = OMX_ErrorUnsupportedSetting;
4716 }
4717 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp9", OMX_MAX_STRINGNAME_SIZE)) {
4718 if (!strncmp((const char*)comp_role->cRole, "video_decoder.vp9", OMX_MAX_STRINGNAME_SIZE) ||
4719 !strncmp((const char*)comp_role->cRole, "video_decoder.vpx", OMX_MAX_STRINGNAME_SIZE)) {
4720 strlcpy((char*)m_cRole, "video_decoder.vp9", OMX_MAX_STRINGNAME_SIZE);
4721 } else {
4722 DEBUG_PRINT_ERROR("Setparameter: unknown Index %s", comp_role->cRole);
4723 eRet = OMX_ErrorUnsupportedSetting;
4724 }
4725 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.hevc", OMX_MAX_STRINGNAME_SIZE)) {
4726 if (!strncmp((const char*)comp_role->cRole, "video_decoder.hevc", OMX_MAX_STRINGNAME_SIZE)) {
4727 strlcpy((char*)m_cRole, "video_decoder.hevc", OMX_MAX_STRINGNAME_SIZE);
4728 } else {
4729 DEBUG_PRINT_ERROR("Setparameter: unknown Index %s", comp_role->cRole);
4730 eRet = OMX_ErrorUnsupportedSetting;
4731 }
4732 } else {
4733 DEBUG_PRINT_ERROR("Setparameter: unknown param %s", drv_ctx.kind);
4734 eRet = OMX_ErrorInvalidComponentName;
4735 }
4736 break;
4737 }
4738
4739 case OMX_IndexParamPriorityMgmt: {
4740 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PRIORITYMGMTTYPE);
4741 if (m_state != OMX_StateLoaded) {
4742 DEBUG_PRINT_ERROR("Set Parameter called in Invalid State");
4743 return OMX_ErrorIncorrectStateOperation;
4744 }
4745 OMX_PRIORITYMGMTTYPE *priorityMgmtype = (OMX_PRIORITYMGMTTYPE*) paramData;
4746 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamPriorityMgmt %u",
4747 (unsigned int)priorityMgmtype->nGroupID);
4748
4749 DEBUG_PRINT_LOW("set_parameter: priorityMgmtype %u",
4750 (unsigned int)priorityMgmtype->nGroupPriority);
4751
4752 m_priority_mgm.nGroupID = priorityMgmtype->nGroupID;
4753 m_priority_mgm.nGroupPriority = priorityMgmtype->nGroupPriority;
4754
4755 break;
4756 }
4757
4758 case OMX_IndexParamCompBufferSupplier: {
4759 VALIDATE_OMX_PARAM_DATA(paramData, OMX_PARAM_BUFFERSUPPLIERTYPE);
4760 OMX_PARAM_BUFFERSUPPLIERTYPE *bufferSupplierType = (OMX_PARAM_BUFFERSUPPLIERTYPE*) paramData;
4761 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamCompBufferSupplier %d",
4762 bufferSupplierType->eBufferSupplier);
4763 if (bufferSupplierType->nPortIndex == 0 || bufferSupplierType->nPortIndex ==1)
4764 m_buffer_supplier.eBufferSupplier = bufferSupplierType->eBufferSupplier;
4765
4766 else
4767
4768 eRet = OMX_ErrorBadPortIndex;
4769
4770 break;
4771
4772 }
4773 case OMX_IndexParamVideoAvc: {
4774 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamVideoAvc %d",
4775 paramIndex);
4776 break;
4777 }
4778 case (OMX_INDEXTYPE)QOMX_IndexParamVideoMvc: {
4779 DEBUG_PRINT_LOW("set_parameter: QOMX_IndexParamVideoMvc %d",
4780 paramIndex);
4781 break;
4782 }
4783 case OMX_IndexParamVideoH263: {
4784 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamVideoH263 %d",
4785 paramIndex);
4786 break;
4787 }
4788 case OMX_IndexParamVideoMpeg4: {
4789 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamVideoMpeg4 %d",
4790 paramIndex);
4791 break;
4792 }
4793 case OMX_IndexParamVideoMpeg2: {
4794 DEBUG_PRINT_LOW("set_parameter: OMX_IndexParamVideoMpeg2 %d",
4795 paramIndex);
4796 break;
4797 }
4798 case OMX_QTIIndexParamLowLatencyMode: {
4799 struct v4l2_control control;
4800 int rc = 0;
4801 QOMX_EXTNINDEX_VIDEO_VENC_LOW_LATENCY_MODE* pParam =
4802 (QOMX_EXTNINDEX_VIDEO_VENC_LOW_LATENCY_MODE*)paramData;
4803 if (pParam->bLowLatencyMode) {
4804 DEBUG_PRINT_HIGH("Enabling DECODE order");
4805 time_stamp_dts.set_timestamp_reorder_mode(false);
4806 control.id = V4L2_CID_MPEG_VIDC_VIDEO_OUTPUT_ORDER;
4807 control.value = V4L2_MPEG_VIDC_VIDEO_OUTPUT_ORDER_DECODE;
4808 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control);
4809 if (rc) {
4810 DEBUG_PRINT_ERROR("Set picture order failed");
4811 eRet = OMX_ErrorUnsupportedSetting;
4812 }
4813 }
4814 break;
4815 }
4816 case OMX_QcomIndexParamVideoDecoderPictureOrder: {
4817 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_VIDEO_DECODER_PICTURE_ORDER);
4818 QOMX_VIDEO_DECODER_PICTURE_ORDER *pictureOrder =
4819 (QOMX_VIDEO_DECODER_PICTURE_ORDER *)paramData;
4820 struct v4l2_control control;
4821 int pic_order,rc=0;
4822 DEBUG_PRINT_HIGH("set_parameter: OMX_QcomIndexParamVideoDecoderPictureOrder %d",
4823 pictureOrder->eOutputPictureOrder);
4824 if (pictureOrder->eOutputPictureOrder == QOMX_VIDEO_DISPLAY_ORDER) {
4825 pic_order = V4L2_MPEG_VIDC_VIDEO_OUTPUT_ORDER_DISPLAY;
4826 } else if (pictureOrder->eOutputPictureOrder == QOMX_VIDEO_DECODE_ORDER) {
4827 pic_order = V4L2_MPEG_VIDC_VIDEO_OUTPUT_ORDER_DECODE;
4828 time_stamp_dts.set_timestamp_reorder_mode(false);
4829 } else
4830 eRet = OMX_ErrorBadParameter;
4831 if (eRet == OMX_ErrorNone) {
4832 control.id = V4L2_CID_MPEG_VIDC_VIDEO_OUTPUT_ORDER;
4833 control.value = pic_order;
4834 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control);
4835 if (rc) {
4836 DEBUG_PRINT_ERROR("Set picture order failed");
4837 eRet = OMX_ErrorUnsupportedSetting;
4838 }
4839 }
4840 m_decode_order_mode =
4841 pictureOrder->eOutputPictureOrder == QOMX_VIDEO_DECODE_ORDER;
4842 break;
4843 }
4844 case OMX_QcomIndexParamConcealMBMapExtraData:
4845 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
4846 eRet = enable_extradata(VDEC_EXTRADATA_MB_ERROR_MAP, false,
4847 ((QOMX_ENABLETYPE *)paramData)->bEnable);
4848 break;
4849 case OMX_QcomIndexParamFrameInfoExtraData:
4850 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
4851 eRet = enable_extradata(OMX_FRAMEINFO_EXTRADATA, false,
4852 ((QOMX_ENABLETYPE *)paramData)->bEnable);
4853 break;
4854 case OMX_ExtraDataFrameDimension:
4855 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
4856 eRet = enable_extradata(OMX_FRAMEDIMENSION_EXTRADATA, false,
4857 ((QOMX_ENABLETYPE *)paramData)->bEnable);
4858 break;
4859 case OMX_QcomIndexParamInterlaceExtraData:
4860 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
4861 eRet = enable_extradata(OMX_INTERLACE_EXTRADATA, false,
4862 ((QOMX_ENABLETYPE *)paramData)->bEnable);
4863 break;
4864 case OMX_QcomIndexParamH264TimeInfo:
4865 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
4866 eRet = enable_extradata(OMX_TIMEINFO_EXTRADATA, false,
4867 ((QOMX_ENABLETYPE *)paramData)->bEnable);
4868 break;
4869 case OMX_QcomIndexParamVideoFramePackingExtradata:
4870 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
4871 eRet = enable_extradata(OMX_FRAMEPACK_EXTRADATA, false,
4872 ((QOMX_ENABLETYPE *)paramData)->bEnable);
4873 break;
4874 case OMX_QcomIndexParamVideoQPExtraData:
4875 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
4876 eRet = enable_extradata(OMX_QP_EXTRADATA, false,
4877 ((QOMX_ENABLETYPE *)paramData)->bEnable);
4878 break;
4879 case OMX_QcomIndexParamVideoInputBitsInfoExtraData:
4880 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
4881 eRet = enable_extradata(OMX_BITSINFO_EXTRADATA, false,
4882 ((QOMX_ENABLETYPE *)paramData)->bEnable);
4883 break;
4884 case OMX_QcomIndexEnableExtnUserData:
4885 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
4886 eRet = enable_extradata(OMX_EXTNUSER_EXTRADATA, false,
4887 ((QOMX_ENABLETYPE *)paramData)->bEnable);
4888 break;
4889 case OMX_QTIIndexParamVQZipSEIExtraData:
4890 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
4891 eRet = enable_extradata(OMX_VQZIPSEI_EXTRADATA, false,
4892 ((QOMX_ENABLETYPE *)paramData)->bEnable);
4893 break;
4894 case OMX_QcomIndexParamVideoDivx: {
4895 QOMX_VIDEO_PARAM_DIVXTYPE* divXType = (QOMX_VIDEO_PARAM_DIVXTYPE *) paramData;
4896 }
4897 break;
4898 case OMX_QcomIndexPlatformPvt: {
4899 VALIDATE_OMX_PARAM_DATA(paramData, OMX_QCOM_PLATFORMPRIVATE_EXTN);
4900 DEBUG_PRINT_HIGH("set_parameter: OMX_QcomIndexPlatformPvt OP Port");
4901 OMX_QCOM_PLATFORMPRIVATE_EXTN* entryType = (OMX_QCOM_PLATFORMPRIVATE_EXTN *) paramData;
4902 if (entryType->type != OMX_QCOM_PLATFORM_PRIVATE_PMEM) {
4903 DEBUG_PRINT_HIGH("set_parameter: Platform Private entry type (%d) not supported.", entryType->type);
4904 eRet = OMX_ErrorUnsupportedSetting;
4905 } else {
4906 m_out_pvt_entry_pmem = OMX_TRUE;
4907 if ((m_out_mem_region_smi && m_out_pvt_entry_pmem)) {
4908 DEBUG_PRINT_HIGH("set_parameter: OMX_QcomIndexPlatformPvt OP Port: out pmem set");
4909 m_use_output_pmem = OMX_TRUE;
4910 }
4911 }
4912
4913 }
4914 break;
4915 case OMX_QcomIndexParamVideoSyncFrameDecodingMode: {
4916 DEBUG_PRINT_HIGH("set_parameter: OMX_QcomIndexParamVideoSyncFrameDecodingMode");
4917 DEBUG_PRINT_HIGH("set idr only decoding for thumbnail mode");
4918 struct v4l2_control control;
4919 int rc;
4920 drv_ctx.idr_only_decoding = 1;
4921 control.id = V4L2_CID_MPEG_VIDC_VIDEO_OUTPUT_ORDER;
4922 control.value = V4L2_MPEG_VIDC_VIDEO_OUTPUT_ORDER_DECODE;
4923 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control);
4924 if (rc) {
4925 DEBUG_PRINT_ERROR("Set picture order failed");
4926 eRet = OMX_ErrorUnsupportedSetting;
4927 } else {
4928 control.id = V4L2_CID_MPEG_VIDC_VIDEO_SYNC_FRAME_DECODE;
4929 control.value = V4L2_MPEG_VIDC_VIDEO_SYNC_FRAME_DECODE_ENABLE;
4930 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control);
4931 if (rc) {
4932 DEBUG_PRINT_ERROR("Sync frame setting failed");
4933 eRet = OMX_ErrorUnsupportedSetting;
4934 }
4935 /*Setting sync frame decoding on driver might change buffer
4936 * requirements so update them here*/
4937 if (get_buffer_req(&drv_ctx.ip_buf)) {
4938 DEBUG_PRINT_ERROR("Sync frame setting failed: falied to get buffer i/p requirements");
4939 eRet = OMX_ErrorUnsupportedSetting;
4940 }
4941 if (get_buffer_req(&drv_ctx.op_buf)) {
4942 DEBUG_PRINT_ERROR("Sync frame setting failed: falied to get buffer o/p requirements");
4943 eRet = OMX_ErrorUnsupportedSetting;
4944 }
4945 }
4946 }
4947 break;
4948
4949 case OMX_QcomIndexParamIndexExtraDataType: {
4950 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_INDEXEXTRADATATYPE);
4951 QOMX_INDEXEXTRADATATYPE *extradataIndexType = (QOMX_INDEXEXTRADATATYPE *) paramData;
4952 if ((extradataIndexType->nIndex == OMX_IndexParamPortDefinition) &&
4953 (extradataIndexType->bEnabled == OMX_TRUE) &&
4954 (extradataIndexType->nPortIndex == 1)) {
4955 DEBUG_PRINT_HIGH("set_parameter: OMX_QcomIndexParamIndexExtraDataType SmoothStreaming");
4956 eRet = enable_extradata(OMX_PORTDEF_EXTRADATA, false, extradataIndexType->bEnabled);
4957
4958 }
4959 }
4960 break;
4961 case OMX_QcomIndexParamEnableSmoothStreaming: {
4962 #ifndef SMOOTH_STREAMING_DISABLED
4963 eRet = enable_smoothstreaming();
4964 #else
4965 eRet = OMX_ErrorUnsupportedSetting;
4966 #endif
4967 }
4968 break;
4969 #if defined (_ANDROID_HONEYCOMB_) || defined (_ANDROID_ICS_)
4970 /* Need to allow following two set_parameters even in Idle
4971 * state. This is ANDROID architecture which is not in sync
4972 * with openmax standard. */
4973 case OMX_GoogleAndroidIndexEnableAndroidNativeBuffers: {
4974 VALIDATE_OMX_PARAM_DATA(paramData, EnableAndroidNativeBuffersParams);
4975 EnableAndroidNativeBuffersParams* enableNativeBuffers = (EnableAndroidNativeBuffersParams *) paramData;
4976 if (enableNativeBuffers->nPortIndex != OMX_CORE_OUTPUT_PORT_INDEX) {
4977 DEBUG_PRINT_ERROR("Enable/Disable android-native-buffers allowed only on output port!");
4978 eRet = OMX_ErrorUnsupportedSetting;
4979 break;
4980 } else if (m_out_mem_ptr) {
4981 DEBUG_PRINT_ERROR("Enable/Disable android-native-buffers is not allowed since Output port is not free !");
4982 eRet = OMX_ErrorInvalidState;
4983 break;
4984 }
4985 if (enableNativeBuffers) {
4986 m_enable_android_native_buffers = enableNativeBuffers->enable;
4987 }
4988 #if !defined(FLEXYUV_SUPPORTED)
4989 if (m_enable_android_native_buffers) {
4990 // Use the most-preferred-native-color-format as surface-mode is hinted here
4991 if(!client_buffers.set_color_format(getPreferredColorFormatDefaultMode(0))) {
4992 DEBUG_PRINT_ERROR("Failed to set native color format!");
4993 eRet = OMX_ErrorUnsupportedSetting;
4994 }
4995 }
4996 #endif
4997 }
4998 break;
4999 case OMX_GoogleAndroidIndexUseAndroidNativeBuffer: {
5000 VALIDATE_OMX_PARAM_DATA(paramData, UseAndroidNativeBufferParams);
5001 eRet = use_android_native_buffer(hComp, paramData);
5002 }
5003 break;
5004 #if ALLOCATE_OUTPUT_NATIVEHANDLE
5005 case OMX_GoogleAndroidIndexAllocateNativeHandle: {
5006
5007 AllocateNativeHandleParams* allocateNativeHandleParams = (AllocateNativeHandleParams *) paramData;
5008 VALIDATE_OMX_PARAM_DATA(paramData, AllocateNativeHandleParams);
5009
5010 if (allocateNativeHandleParams->nPortIndex != OMX_CORE_INPUT_PORT_INDEX) {
5011 DEBUG_PRINT_ERROR("Enable/Disable allocate-native-handle allowed only on input port!");
5012 eRet = OMX_ErrorUnsupportedSetting;
5013 break;
5014 } else if (m_inp_mem_ptr) {
5015 DEBUG_PRINT_ERROR("Enable/Disable allocate-native-handle is not allowed since Input port is not free !");
5016 eRet = OMX_ErrorInvalidState;
5017 break;
5018 }
5019
5020 if (allocateNativeHandleParams != NULL) {
5021 allocate_native_handle = allocateNativeHandleParams->enable;
5022 }
5023 }
5024 break;
5025 #endif //ALLOCATE_OUTPUT_NATIVEHANDLE
5026 #endif
5027 case OMX_QcomIndexParamEnableTimeStampReorder: {
5028 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_INDEXTIMESTAMPREORDER);
5029 QOMX_INDEXTIMESTAMPREORDER *reorder = (QOMX_INDEXTIMESTAMPREORDER *)paramData;
5030 if (drv_ctx.picture_order == (vdec_output_order)QOMX_VIDEO_DISPLAY_ORDER) {
5031 if (reorder->bEnable == OMX_TRUE) {
5032 frm_int =0;
5033 time_stamp_dts.set_timestamp_reorder_mode(true);
5034 } else
5035 time_stamp_dts.set_timestamp_reorder_mode(false);
5036 } else {
5037 time_stamp_dts.set_timestamp_reorder_mode(false);
5038 if (reorder->bEnable == OMX_TRUE) {
5039 eRet = OMX_ErrorUnsupportedSetting;
5040 }
5041 }
5042 }
5043 break;
5044 case OMX_IndexParamVideoProfileLevelCurrent: {
5045 VALIDATE_OMX_PARAM_DATA(paramData, OMX_VIDEO_PARAM_PROFILELEVELTYPE);
5046 OMX_VIDEO_PARAM_PROFILELEVELTYPE* pParam =
5047 (OMX_VIDEO_PARAM_PROFILELEVELTYPE*)paramData;
5048 if (pParam) {
5049 m_profile_lvl.eProfile = pParam->eProfile;
5050 m_profile_lvl.eLevel = pParam->eLevel;
5051 }
5052 break;
5053
5054 }
5055 case OMX_QcomIndexParamVideoMetaBufferMode:
5056 {
5057 VALIDATE_OMX_PARAM_DATA(paramData, StoreMetaDataInBuffersParams);
5058 StoreMetaDataInBuffersParams *metabuffer =
5059 (StoreMetaDataInBuffersParams *)paramData;
5060 if (!metabuffer) {
5061 DEBUG_PRINT_ERROR("Invalid param: %p", metabuffer);
5062 eRet = OMX_ErrorBadParameter;
5063 break;
5064 }
5065 if (m_disable_dynamic_buf_mode) {
5066 DEBUG_PRINT_HIGH("Dynamic buffer mode is disabled");
5067 eRet = OMX_ErrorUnsupportedSetting;
5068 break;
5069 }
5070 if (metabuffer->nPortIndex == OMX_CORE_OUTPUT_PORT_INDEX) {
5071
5072 if (m_out_mem_ptr) {
5073 DEBUG_PRINT_ERROR("Enable/Disable dynamic-buffer-mode is not allowed since Output port is not free !");
5074 eRet = OMX_ErrorInvalidState;
5075 break;
5076 }
5077 //set property dynamic buffer mode to driver.
5078 struct v4l2_control control;
5079 struct v4l2_format fmt;
5080 control.id = V4L2_CID_MPEG_VIDC_VIDEO_ALLOC_MODE_OUTPUT;
5081 if (metabuffer->bStoreMetaData == true) {
5082 control.value = V4L2_MPEG_VIDC_VIDEO_DYNAMIC;
5083 } else {
5084 control.value = V4L2_MPEG_VIDC_VIDEO_STATIC;
5085 }
5086 int rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL,&control);
5087 if (!rc) {
5088 DEBUG_PRINT_HIGH("%s buffer mode",
5089 (metabuffer->bStoreMetaData == true)? "Enabled dynamic" : "Disabled dynamic");
5090 } else {
5091 DEBUG_PRINT_ERROR("Failed to %s buffer mode",
5092 (metabuffer->bStoreMetaData == true)? "enable dynamic" : "disable dynamic");
5093 }
5094 dynamic_buf_mode = metabuffer->bStoreMetaData;
5095 } else {
5096 DEBUG_PRINT_ERROR(
5097 "OMX_QcomIndexParamVideoMetaBufferMode not supported for port: %u",
5098 (unsigned int)metabuffer->nPortIndex);
5099 eRet = OMX_ErrorUnsupportedSetting;
5100 }
5101 break;
5102 }
5103 case OMX_QcomIndexParamVideoDownScalar:
5104 {
5105 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_INDEXDOWNSCALAR);
5106 QOMX_INDEXDOWNSCALAR* pParam = (QOMX_INDEXDOWNSCALAR*)paramData;
5107 struct v4l2_control control;
5108 int rc;
5109 DEBUG_PRINT_LOW("set_parameter: OMX_QcomIndexParamVideoDownScalar %d\n", pParam->bEnable);
5110
5111 if (pParam && pParam->bEnable) {
5112 rc = enable_downscalar();
5113 if (rc < 0) {
5114 DEBUG_PRINT_ERROR("%s: enable_downscalar failed\n", __func__);
5115 return OMX_ErrorUnsupportedSetting;
5116 }
5117 m_force_down_scalar = pParam->bEnable;
5118 } else {
5119 rc = disable_downscalar();
5120 if (rc < 0) {
5121 DEBUG_PRINT_ERROR("%s: disable_downscalar failed\n", __func__);
5122 return OMX_ErrorUnsupportedSetting;
5123 }
5124 m_force_down_scalar = pParam->bEnable;
5125 }
5126 break;
5127 }
5128 #ifdef ADAPTIVE_PLAYBACK_SUPPORTED
5129 case OMX_QcomIndexParamVideoAdaptivePlaybackMode:
5130 {
5131 VALIDATE_OMX_PARAM_DATA(paramData, PrepareForAdaptivePlaybackParams);
5132 DEBUG_PRINT_LOW("set_parameter: OMX_GoogleAndroidIndexPrepareForAdaptivePlayback");
5133 PrepareForAdaptivePlaybackParams* pParams =
5134 (PrepareForAdaptivePlaybackParams *) paramData;
5135 if (pParams->nPortIndex == OMX_CORE_OUTPUT_PORT_INDEX) {
5136 if (!pParams->bEnable) {
5137 return OMX_ErrorNone;
5138 }
5139 if (pParams->nMaxFrameWidth > maxSmoothStreamingWidth
5140 || pParams->nMaxFrameHeight > maxSmoothStreamingHeight) {
5141 DEBUG_PRINT_ERROR(
5142 "Adaptive playback request exceeds max supported resolution : [%u x %u] vs [%u x %u]",
5143 (unsigned int)pParams->nMaxFrameWidth, (unsigned int)pParams->nMaxFrameHeight,
5144 (unsigned int)maxSmoothStreamingWidth, (unsigned int)maxSmoothStreamingHeight);
5145 eRet = OMX_ErrorBadParameter;
5146 } else {
5147 eRet = enable_adaptive_playback(pParams->nMaxFrameWidth, pParams->nMaxFrameHeight);
5148 }
5149 } else {
5150 DEBUG_PRINT_ERROR(
5151 "Prepare for adaptive playback supported only on output port");
5152 eRet = OMX_ErrorBadParameter;
5153 }
5154 break;
5155 }
5156
5157 case OMX_QTIIndexParamVideoPreferAdaptivePlayback:
5158 {
5159 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
5160 DEBUG_PRINT_LOW("set_parameter: OMX_QTIIndexParamVideoPreferAdaptivePlayback");
5161 m_disable_dynamic_buf_mode = ((QOMX_ENABLETYPE *)paramData)->bEnable;
5162 if (m_disable_dynamic_buf_mode) {
5163 DEBUG_PRINT_HIGH("Prefer Adaptive Playback is set");
5164 }
5165 break;
5166 }
5167 #endif
5168 case OMX_QcomIndexParamVideoCustomBufferSize:
5169 {
5170 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_VIDEO_CUSTOM_BUFFERSIZE);
5171 DEBUG_PRINT_LOW("set_parameter: OMX_QcomIndexParamVideoCustomBufferSize");
5172 QOMX_VIDEO_CUSTOM_BUFFERSIZE* pParam = (QOMX_VIDEO_CUSTOM_BUFFERSIZE*)paramData;
5173 if (pParam->nPortIndex == OMX_CORE_INPUT_PORT_INDEX) {
5174 struct v4l2_control control;
5175 control.id = V4L2_CID_MPEG_VIDC_VIDEO_BUFFER_SIZE_LIMIT;
5176 control.value = pParam->nBufferSize;
5177 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
5178 DEBUG_PRINT_ERROR("Failed to set input buffer size");
5179 eRet = OMX_ErrorUnsupportedSetting;
5180 } else {
5181 eRet = get_buffer_req(&drv_ctx.ip_buf);
5182 if (eRet == OMX_ErrorNone) {
5183 m_custom_buffersize.input_buffersize = drv_ctx.ip_buf.buffer_size;
5184 DEBUG_PRINT_HIGH("Successfully set custom input buffer size = %d",
5185 m_custom_buffersize.input_buffersize);
5186 } else {
5187 DEBUG_PRINT_ERROR("Failed to get buffer requirement");
5188 }
5189 }
5190 } else {
5191 DEBUG_PRINT_ERROR("ERROR: Custom buffer size in not supported on output port");
5192 eRet = OMX_ErrorBadParameter;
5193 }
5194 break;
5195 }
5196 case OMX_QTIIndexParamVQZIPSEIType:
5197 {
5198 VALIDATE_OMX_PARAM_DATA(paramData, OMX_QTI_VIDEO_PARAM_VQZIP_SEI_TYPE);
5199 DEBUG_PRINT_LOW("set_parameter: OMX_QTIIndexParamVQZIPSEIType");
5200 OMX_QTI_VIDEO_PARAM_VQZIP_SEI_TYPE *pParam =
5201 (OMX_QTI_VIDEO_PARAM_VQZIP_SEI_TYPE *)paramData;
5202 DEBUG_PRINT_LOW("Enable VQZIP SEI: %d", pParam->bEnable);
5203
5204 eRet = enable_extradata(OMX_VQZIPSEI_EXTRADATA, false,
5205 ((QOMX_ENABLETYPE *)paramData)->bEnable);
5206 if (eRet != OMX_ErrorNone) {
5207 DEBUG_PRINT_ERROR("ERROR: Failed to set SEI Extradata");
5208 eRet = OMX_ErrorBadParameter;
5209 client_extradata = client_extradata & ~OMX_VQZIPSEI_EXTRADATA;
5210 break;
5211 }
5212 eRet = enable_extradata(OMX_QP_EXTRADATA, false,
5213 ((QOMX_ENABLETYPE *)paramData)->bEnable);
5214 if (eRet != OMX_ErrorNone) {
5215 DEBUG_PRINT_ERROR("ERROR: Failed to set QP Extradata");
5216 eRet = OMX_ErrorBadParameter;
5217 client_extradata = client_extradata & ~OMX_VQZIPSEI_EXTRADATA;
5218 client_extradata = client_extradata & ~OMX_QP_EXTRADATA;
5219 break;
5220 }
5221 eRet = enable_extradata(OMX_FRAMEINFO_EXTRADATA, false,
5222 ((QOMX_ENABLETYPE *)paramData)->bEnable);
5223 if (eRet != OMX_ErrorNone) {
5224 DEBUG_PRINT_ERROR("ERROR: Failed to set FrameInfo Extradata");
5225 eRet = OMX_ErrorBadParameter;
5226 client_extradata = client_extradata & ~OMX_VQZIPSEI_EXTRADATA;
5227 client_extradata = client_extradata & ~OMX_QP_EXTRADATA;
5228 client_extradata = client_extradata & ~OMX_FRAMEINFO_EXTRADATA;
5229 }
5230 break;
5231 }
5232 case OMX_QTIIndexParamPassInputBufferFd:
5233 {
5234 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_ENABLETYPE);
5235 if (arbitrary_bytes) {
5236 DEBUG_PRINT_ERROR("OMX_QTIIndexParamPassInputBufferFd not supported in arbitrary buffer mode");
5237 eRet = OMX_ErrorUnsupportedSetting;
5238 break;
5239 }
5240
5241 m_input_pass_buffer_fd = ((QOMX_ENABLETYPE *)paramData)->bEnable;
5242 if (m_input_pass_buffer_fd)
5243 DEBUG_PRINT_LOW("Enable passing input buffer FD");
5244 break;
5245 }
5246 case OMX_QTIIndexParamForceCompressedForDPB:
5247 {
5248 VALIDATE_OMX_PARAM_DATA(paramData, OMX_QTI_VIDEO_PARAM_FORCE_COMPRESSED_FOR_DPB_TYPE);
5249 DEBUG_PRINT_LOW("set_parameter: OMX_QTIIndexParamForceCompressedForDPB");
5250 OMX_QTI_VIDEO_PARAM_FORCE_COMPRESSED_FOR_DPB_TYPE *pParam =
5251 (OMX_QTI_VIDEO_PARAM_FORCE_COMPRESSED_FOR_DPB_TYPE *)paramData;
5252 if (m_disable_ubwc_mode) {
5253 DEBUG_PRINT_ERROR("OMX_QTIIndexParamForceCompressedForDPB not supported when ubwc disabled");
5254 eRet = OMX_ErrorUnsupportedSetting;
5255 break;
5256 }
5257 if (!paramData) {
5258 DEBUG_PRINT_ERROR("set_parameter: OMX_QTIIndexParamForceCompressedForDPB paramData NULL");
5259 eRet = OMX_ErrorBadParameter;
5260 break;
5261 }
5262
5263 m_force_compressed_for_dpb = pParam->bEnable;
5264 break;
5265 }
5266 case OMX_QTIIndexParamForceUnCompressedForOPB:
5267 {
5268 DEBUG_PRINT_LOW("set_parameter: OMX_QTIIndexParamForceUnCompressedForOPB");
5269 OMX_QTI_VIDEO_PARAM_FORCE_UNCOMPRESSED_FOR_OPB_TYPE *pParam =
5270 (OMX_QTI_VIDEO_PARAM_FORCE_UNCOMPRESSED_FOR_OPB_TYPE *)paramData;
5271 if (!paramData) {
5272 DEBUG_PRINT_ERROR("set_parameter: OMX_QTIIndexParamForceUnCompressedForOPB paramData is NULL");
5273 eRet = OMX_ErrorBadParameter;
5274 break;
5275 }
5276 m_disable_ubwc_mode = pParam->bEnable;
5277 DEBUG_PRINT_LOW("set_parameter: UBWC %s for OPB", pParam->bEnable ? "disabled" : "enabled");
5278 break;
5279 }
5280 case OMX_QTIIndexParamDitherControl:
5281 {
5282 VALIDATE_OMX_PARAM_DATA(paramData, QOMX_VIDEO_DITHER_CONTROL);
5283 DEBUG_PRINT_LOW("set_parameter: OMX_QTIIndexParamDitherControl");
5284 QOMX_VIDEO_DITHER_CONTROL *pParam = (QOMX_VIDEO_DITHER_CONTROL *)paramData;
5285 DEBUG_PRINT_LOW("set_parameter: Dither Config from client is: %d", pParam->eDitherType);
5286 if (( pParam->eDitherType < QOMX_DITHER_DISABLE ) ||
5287 ( pParam->eDitherType > QOMX_DITHER_ALL_COLORSPACE)) {
5288 DEBUG_PRINT_ERROR("set_parameter: DitherType outside the range");
5289 eRet = OMX_ErrorBadParameter;
5290 break;
5291 }
5292 m_dither_config = is_platform_tp10capture_supported() ? (dither_type)pParam->eDitherType : DITHER_ALL_COLORSPACE;
5293 DEBUG_PRINT_LOW("set_parameter: Final Dither Config is: %d", m_dither_config);
5294 break;
5295 }
5296 default: {
5297 DEBUG_PRINT_ERROR("Setparameter: unknown param %d", paramIndex);
5298 eRet = OMX_ErrorUnsupportedIndex;
5299 }
5300 }
5301 if (eRet != OMX_ErrorNone)
5302 DEBUG_PRINT_ERROR("set_parameter: Error: 0x%x, setting param 0x%x", eRet, paramIndex);
5303 return eRet;
5304 }
5305
5306 /* ======================================================================
5307 FUNCTION
5308 omx_vdec::GetConfig
5309
5310 DESCRIPTION
5311 OMX Get Config Method implementation.
5312
5313 PARAMETERS
5314 <TBD>.
5315
5316 RETURN VALUE
5317 OMX Error None if successful.
5318
5319 ========================================================================== */
get_config(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_INDEXTYPE configIndex,OMX_INOUT OMX_PTR configData)5320 OMX_ERRORTYPE omx_vdec::get_config(OMX_IN OMX_HANDLETYPE hComp,
5321 OMX_IN OMX_INDEXTYPE configIndex,
5322 OMX_INOUT OMX_PTR configData)
5323 {
5324 (void) hComp;
5325 OMX_ERRORTYPE eRet = OMX_ErrorNone;
5326
5327 if (m_state == OMX_StateInvalid) {
5328 DEBUG_PRINT_ERROR("Get Config in Invalid State");
5329 return OMX_ErrorInvalidState;
5330 }
5331
5332 switch ((unsigned long)configIndex) {
5333 case OMX_QcomIndexConfigInterlaced: {
5334 VALIDATE_OMX_PARAM_DATA(configData, OMX_QCOM_CONFIG_INTERLACETYPE);
5335 OMX_QCOM_CONFIG_INTERLACETYPE *configFmt =
5336 (OMX_QCOM_CONFIG_INTERLACETYPE *) configData;
5337 if (configFmt->nPortIndex == 1) {
5338 if (configFmt->nIndex == 0) {
5339 configFmt->eInterlaceType = OMX_QCOM_InterlaceFrameProgressive;
5340 } else if (configFmt->nIndex == 1) {
5341 configFmt->eInterlaceType =
5342 OMX_QCOM_InterlaceInterleaveFrameTopFieldFirst;
5343 } else if (configFmt->nIndex == 2) {
5344 configFmt->eInterlaceType =
5345 OMX_QCOM_InterlaceInterleaveFrameBottomFieldFirst;
5346 } else {
5347 DEBUG_PRINT_ERROR("get_config: OMX_QcomIndexConfigInterlaced:"
5348 " NoMore Interlaced formats");
5349 eRet = OMX_ErrorNoMore;
5350 }
5351
5352 } else {
5353 DEBUG_PRINT_ERROR("get_config: Bad port index %d queried on only o/p port",
5354 (int)configFmt->nPortIndex);
5355 eRet = OMX_ErrorBadPortIndex;
5356 }
5357 break;
5358 }
5359 case OMX_QcomIndexQueryNumberOfVideoDecInstance: {
5360 VALIDATE_OMX_PARAM_DATA(configData, QOMX_VIDEO_QUERY_DECODER_INSTANCES);
5361 QOMX_VIDEO_QUERY_DECODER_INSTANCES *decoderinstances =
5362 (QOMX_VIDEO_QUERY_DECODER_INSTANCES*)configData;
5363 decoderinstances->nNumOfInstances = 16;
5364 /*TODO: How to handle this case */
5365 break;
5366 }
5367 case OMX_QcomIndexConfigVideoFramePackingArrangement: {
5368 if (drv_ctx.decoder_format == VDEC_CODECTYPE_H264) {
5369 VALIDATE_OMX_PARAM_DATA(configData, OMX_QCOM_FRAME_PACK_ARRANGEMENT);
5370 OMX_QCOM_FRAME_PACK_ARRANGEMENT *configFmt =
5371 (OMX_QCOM_FRAME_PACK_ARRANGEMENT *) configData;
5372 memcpy(configFmt, &m_frame_pack_arrangement,
5373 sizeof(OMX_QCOM_FRAME_PACK_ARRANGEMENT));
5374 } else {
5375 DEBUG_PRINT_ERROR("get_config: Framepack data not supported for non H264 codecs");
5376 }
5377 break;
5378 }
5379 case OMX_IndexConfigCommonOutputCrop: {
5380 VALIDATE_OMX_PARAM_DATA(configData, OMX_CONFIG_RECTTYPE);
5381 OMX_CONFIG_RECTTYPE *rect = (OMX_CONFIG_RECTTYPE *) configData;
5382 memcpy(rect, &rectangle, sizeof(OMX_CONFIG_RECTTYPE));
5383 DEBUG_PRINT_HIGH("get_config: crop info: L: %u, T: %u, R: %u, B: %u",
5384 rectangle.nLeft, rectangle.nTop,
5385 rectangle.nWidth, rectangle.nHeight);
5386 break;
5387 }
5388 case OMX_QcomIndexConfigPerfLevel: {
5389 VALIDATE_OMX_PARAM_DATA(configData, OMX_QCOM_VIDEO_CONFIG_PERF_LEVEL);
5390 struct v4l2_control control;
5391 OMX_QCOM_VIDEO_CONFIG_PERF_LEVEL *perf =
5392 (OMX_QCOM_VIDEO_CONFIG_PERF_LEVEL *)configData;
5393
5394 control.id = V4L2_CID_MPEG_VIDC_SET_PERF_LEVEL;
5395 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_G_CTRL, &control) < 0) {
5396 DEBUG_PRINT_ERROR("Failed getting performance level: %d", errno);
5397 eRet = OMX_ErrorHardware;
5398 }
5399
5400 if (eRet == OMX_ErrorNone) {
5401 switch (control.value) {
5402 case V4L2_CID_MPEG_VIDC_PERF_LEVEL_TURBO:
5403 perf->ePerfLevel = OMX_QCOM_PerfLevelTurbo;
5404 break;
5405 default:
5406 DEBUG_PRINT_HIGH("Unknown perf level %d, reporting Nominal instead", control.value);
5407 /* Fall through */
5408 case V4L2_CID_MPEG_VIDC_PERF_LEVEL_NOMINAL:
5409 perf->ePerfLevel = OMX_QCOM_PerfLevelNominal;
5410 break;
5411 }
5412 }
5413
5414 break;
5415 }
5416 case OMX_QcomIndexConfigH264EntropyCodingCabac: {
5417 VALIDATE_OMX_PARAM_DATA(configData, QOMX_VIDEO_H264ENTROPYCODINGTYPE);
5418 QOMX_VIDEO_H264ENTROPYCODINGTYPE *coding = (QOMX_VIDEO_H264ENTROPYCODINGTYPE *)configData;
5419 struct v4l2_control control;
5420
5421 if (drv_ctx.decoder_format != VDEC_CODECTYPE_H264) {
5422 DEBUG_PRINT_ERROR("get_config of OMX_QcomIndexConfigH264EntropyCodingCabac only available for H264");
5423 eRet = OMX_ErrorNotImplemented;
5424 break;
5425 }
5426
5427 control.id = V4L2_CID_MPEG_VIDEO_H264_ENTROPY_MODE;
5428 if (!ioctl(drv_ctx.video_driver_fd, VIDIOC_G_CTRL, &control)) {
5429 coding->bCabac = (OMX_BOOL)
5430 (control.value == V4L2_MPEG_VIDEO_H264_ENTROPY_MODE_CABAC);
5431 /* We can't query driver at the moment for the cabac mode, so
5432 * just use 0xff...f as a place holder for future improvement */
5433 coding->nCabacInitIdc = ~0;
5434 } else {
5435 eRet = OMX_ErrorUnsupportedIndex;
5436 }
5437
5438 break;
5439 }
5440 case OMX_QTIIndexConfigDescribeColorAspects:
5441 {
5442 VALIDATE_OMX_PARAM_DATA(configData, DescribeColorAspectsParams);
5443 DescribeColorAspectsParams *params = (DescribeColorAspectsParams *)configData;
5444
5445 if (params->bRequestingDataSpace) {
5446 DEBUG_PRINT_HIGH("Does not handle dataspace request");
5447 return OMX_ErrorUnsupportedSetting;
5448 }
5449
5450 print_debug_color_aspects(&(m_client_color_space.sAspects), "GetConfig Client");
5451 print_debug_color_aspects(&(m_internal_color_space.sAspects), "GetConfig Internal");
5452
5453 // For VPX, use client-color if specified.
5454 // For the rest, try to use the stream-color if present
5455 bool preferClientColor = (output_capability == V4L2_PIX_FMT_VP8 ||
5456 output_capability == V4L2_PIX_FMT_VP9);
5457
5458 const ColorAspects &preferredColor = preferClientColor ?
5459 m_client_color_space.sAspects : m_internal_color_space.sAspects;
5460 const ColorAspects &defaultColor = preferClientColor ?
5461 m_internal_color_space.sAspects : m_client_color_space.sAspects;
5462
5463 params->sAspects.mPrimaries = preferredColor.mPrimaries != ColorAspects::PrimariesUnspecified ?
5464 preferredColor.mPrimaries : defaultColor.mPrimaries;
5465 params->sAspects.mTransfer = preferredColor.mTransfer != ColorAspects::TransferUnspecified ?
5466 preferredColor.mTransfer : defaultColor.mTransfer;
5467 params->sAspects.mMatrixCoeffs = preferredColor.mMatrixCoeffs != ColorAspects::MatrixUnspecified ?
5468 preferredColor.mMatrixCoeffs : defaultColor.mMatrixCoeffs;
5469 params->sAspects.mRange = preferredColor.mRange != ColorAspects::RangeUnspecified ?
5470 preferredColor.mRange : defaultColor.mRange;
5471
5472 print_debug_color_aspects(&(params->sAspects), "GetConfig");
5473
5474 break;
5475 }
5476 case OMX_QTIIndexConfigDescribeHDRColorInfo:
5477 {
5478 VALIDATE_OMX_PARAM_DATA(configData, DescribeHDRStaticInfoParams);
5479 DescribeHDRStaticInfoParams *params = (DescribeHDRStaticInfoParams *)configData;
5480 auto_lock lock(m_hdr_info_client_lock);
5481
5482 print_debug_hdr_color_info(&(m_client_hdr_info.sInfo), "GetConfig Client HDR");
5483 print_debug_hdr_color_info(&(m_internal_hdr_info.sInfo), "GetConfig Internal HDR");
5484
5485 if (m_change_client_hdr_info) {
5486 DEBUG_PRINT_LOW("Updating Client's HDR Info with internal");
5487 memcpy(&m_client_hdr_info.sInfo,
5488 &m_internal_hdr_info.sInfo, sizeof(HDRStaticInfo));
5489 m_change_client_hdr_info = false;
5490 }
5491
5492 memcpy(&(params->sInfo), &(m_client_hdr_info.sInfo), sizeof(HDRStaticInfo));
5493
5494 break;
5495 }
5496 case OMX_IndexConfigAndroidVendorExtension:
5497 {
5498 VALIDATE_OMX_PARAM_DATA(configData, OMX_CONFIG_ANDROID_VENDOR_EXTENSIONTYPE);
5499
5500 OMX_CONFIG_ANDROID_VENDOR_EXTENSIONTYPE *ext =
5501 reinterpret_cast<OMX_CONFIG_ANDROID_VENDOR_EXTENSIONTYPE *>(configData);
5502 VALIDATE_OMX_VENDOR_EXTENSION_PARAM_DATA(ext);
5503 return get_vendor_extension_config(ext);
5504 }
5505 default:
5506 {
5507 DEBUG_PRINT_ERROR("get_config: unknown param %d",configIndex);
5508 eRet = OMX_ErrorBadParameter;
5509 }
5510
5511 }
5512
5513 return eRet;
5514 }
5515
5516 /* ======================================================================
5517 FUNCTION
5518 omx_vdec::SetConfig
5519
5520 DESCRIPTION
5521 OMX Set Config method implementation
5522
5523 PARAMETERS
5524 <TBD>.
5525
5526 RETURN VALUE
5527 OMX Error None if successful.
5528 ========================================================================== */
set_config(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_INDEXTYPE configIndex,OMX_IN OMX_PTR configData)5529 OMX_ERRORTYPE omx_vdec::set_config(OMX_IN OMX_HANDLETYPE hComp,
5530 OMX_IN OMX_INDEXTYPE configIndex,
5531 OMX_IN OMX_PTR configData)
5532 {
5533 (void) hComp;
5534 if (m_state == OMX_StateInvalid) {
5535 DEBUG_PRINT_ERROR("Get Config in Invalid State");
5536 return OMX_ErrorInvalidState;
5537 }
5538
5539 OMX_ERRORTYPE ret = OMX_ErrorNone;
5540 OMX_VIDEO_CONFIG_NALSIZE *pNal;
5541
5542 DEBUG_PRINT_LOW("Set Config Called");
5543
5544 if (configIndex == OMX_IndexConfigVideoNalSize) {
5545 struct v4l2_control temp;
5546 temp.id = V4L2_CID_MPEG_VIDC_VIDEO_STREAM_FORMAT;
5547
5548 VALIDATE_OMX_PARAM_DATA(configData, OMX_VIDEO_CONFIG_NALSIZE);
5549 pNal = reinterpret_cast < OMX_VIDEO_CONFIG_NALSIZE * >(configData);
5550 switch (pNal->nNaluBytes) {
5551 case 0:
5552 temp.value = V4L2_MPEG_VIDC_VIDEO_NAL_FORMAT_STARTCODES;
5553 break;
5554 case 2:
5555 temp.value = V4L2_MPEG_VIDC_VIDEO_NAL_FORMAT_TWO_BYTE_LENGTH;
5556 break;
5557 case 4:
5558 temp.value = V4L2_MPEG_VIDC_VIDEO_NAL_FORMAT_FOUR_BYTE_LENGTH;
5559 break;
5560 default:
5561 return OMX_ErrorUnsupportedSetting;
5562 }
5563
5564 if (!arbitrary_bytes) {
5565 /* In arbitrary bytes mode, the assembler strips out nal size and replaces
5566 * with start code, so only need to notify driver in frame by frame mode */
5567 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &temp)) {
5568 DEBUG_PRINT_ERROR("Failed to set V4L2_CID_MPEG_VIDC_VIDEO_STREAM_FORMAT");
5569 return OMX_ErrorHardware;
5570 }
5571 }
5572
5573 nal_length = pNal->nNaluBytes;
5574 m_frame_parser.init_nal_length(nal_length);
5575
5576 DEBUG_PRINT_LOW("OMX_IndexConfigVideoNalSize called with Size %d", nal_length);
5577 return ret;
5578 } else if ((int)configIndex == (int)OMX_IndexVendorVideoFrameRate) {
5579 OMX_VENDOR_VIDEOFRAMERATE *config = (OMX_VENDOR_VIDEOFRAMERATE *) configData;
5580 DEBUG_PRINT_HIGH("Index OMX_IndexVendorVideoFrameRate %u", (unsigned int)config->nFps);
5581
5582 if (config->nPortIndex == OMX_CORE_INPUT_PORT_INDEX) {
5583 if (config->bEnabled) {
5584 if ((config->nFps >> 16) > 0 &&
5585 (config->nFps >> 16) <= MAX_SUPPORTED_FPS) {
5586 m_fps_received = config->nFps;
5587 DEBUG_PRINT_HIGH("set_config: frame rate set by omx client : %u",
5588 (unsigned int)config->nFps >> 16);
5589 Q16ToFraction(config->nFps, drv_ctx.frame_rate.fps_numerator,
5590 drv_ctx.frame_rate.fps_denominator);
5591
5592 if (!drv_ctx.frame_rate.fps_numerator) {
5593 DEBUG_PRINT_ERROR("Numerator is zero setting to 30");
5594 drv_ctx.frame_rate.fps_numerator = 30;
5595 }
5596
5597 if (drv_ctx.frame_rate.fps_denominator) {
5598 drv_ctx.frame_rate.fps_numerator = (int)
5599 drv_ctx.frame_rate.fps_numerator / drv_ctx.frame_rate.fps_denominator;
5600 }
5601
5602 drv_ctx.frame_rate.fps_denominator = 1;
5603 frm_int = drv_ctx.frame_rate.fps_denominator * 1e6 /
5604 drv_ctx.frame_rate.fps_numerator;
5605
5606 struct v4l2_outputparm oparm;
5607 /*XXX: we're providing timing info as seconds per frame rather than frames
5608 * per second.*/
5609 oparm.timeperframe.numerator = drv_ctx.frame_rate.fps_denominator;
5610 oparm.timeperframe.denominator = drv_ctx.frame_rate.fps_numerator;
5611
5612 struct v4l2_streamparm sparm;
5613 sparm.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
5614 sparm.parm.output = oparm;
5615 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_PARM, &sparm)) {
5616 DEBUG_PRINT_ERROR("Unable to convey fps info to driver, \
5617 performance might be affected");
5618 ret = OMX_ErrorHardware;
5619 }
5620 client_set_fps = true;
5621 } else {
5622 DEBUG_PRINT_ERROR("Frame rate not supported.");
5623 ret = OMX_ErrorUnsupportedSetting;
5624 }
5625 } else {
5626 DEBUG_PRINT_HIGH("set_config: Disabled client's frame rate");
5627 client_set_fps = false;
5628 }
5629 } else {
5630 DEBUG_PRINT_ERROR(" Set_config: Bad Port idx %d",
5631 (int)config->nPortIndex);
5632 ret = OMX_ErrorBadPortIndex;
5633 }
5634
5635 return ret;
5636 } else if ((int)configIndex == (int)OMX_QcomIndexConfigPerfLevel) {
5637 OMX_QCOM_VIDEO_CONFIG_PERF_LEVEL *perf =
5638 (OMX_QCOM_VIDEO_CONFIG_PERF_LEVEL *)configData;
5639 struct v4l2_control control;
5640
5641 DEBUG_PRINT_LOW("Set perf level: %d", perf->ePerfLevel);
5642 control.id = V4L2_CID_MPEG_VIDC_SET_PERF_LEVEL;
5643
5644 switch (perf->ePerfLevel) {
5645 case OMX_QCOM_PerfLevelNominal:
5646 control.value = V4L2_CID_MPEG_VIDC_PERF_LEVEL_NOMINAL;
5647 m_need_turbo &= ~TURBO_MODE_CLIENT_REQUESTED;
5648 break;
5649 case OMX_QCOM_PerfLevelTurbo:
5650 control.value = V4L2_CID_MPEG_VIDC_PERF_LEVEL_TURBO;
5651 m_need_turbo |= TURBO_MODE_CLIENT_REQUESTED;
5652 break;
5653 default:
5654 ret = OMX_ErrorUnsupportedSetting;
5655 break;
5656 }
5657
5658 if (ret == OMX_ErrorNone) {
5659 ret = (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control) < 0) ?
5660 OMX_ErrorUnsupportedSetting : OMX_ErrorNone;
5661 }
5662
5663 return ret;
5664 } else if ((int)configIndex == (int)OMX_QcomIndexConfigPictureTypeDecode) {
5665 OMX_QCOM_VIDEO_CONFIG_PICTURE_TYPE_DECODE *config =
5666 (OMX_QCOM_VIDEO_CONFIG_PICTURE_TYPE_DECODE *)configData;
5667 struct v4l2_control control;
5668 DEBUG_PRINT_LOW("Set picture type decode: %d", config->eDecodeType);
5669 control.id = V4L2_CID_MPEG_VIDC_VIDEO_PICTYPE_DEC_MODE;
5670
5671 switch (config->eDecodeType) {
5672 case OMX_QCOM_PictypeDecode_I:
5673 control.value = V4L2_MPEG_VIDC_VIDEO_PICTYPE_DECODE_ON;
5674 break;
5675 case OMX_QCOM_PictypeDecode_IPB:
5676 default:
5677 control.value = V4L2_MPEG_VIDC_VIDEO_PICTYPE_DECODE_OFF;
5678 break;
5679 }
5680
5681 ret = (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control) < 0) ?
5682 OMX_ErrorUnsupportedSetting : OMX_ErrorNone;
5683 if (ret)
5684 DEBUG_PRINT_ERROR("Failed to set picture type decode");
5685
5686 return ret;
5687 } else if ((int)configIndex == (int)OMX_IndexConfigPriority) {
5688 OMX_PARAM_U32TYPE *priority = (OMX_PARAM_U32TYPE *)configData;
5689 DEBUG_PRINT_LOW("Set_config: priority %d",priority->nU32);
5690
5691 struct v4l2_control control;
5692
5693 control.id = V4L2_CID_MPEG_VIDC_VIDEO_PRIORITY;
5694 if (priority->nU32 == 0)
5695 control.value = V4L2_MPEG_VIDC_VIDEO_PRIORITY_REALTIME_ENABLE;
5696 else
5697 control.value = V4L2_MPEG_VIDC_VIDEO_PRIORITY_REALTIME_DISABLE;
5698
5699 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
5700 DEBUG_PRINT_ERROR("Failed to set Priority");
5701 ret = OMX_ErrorUnsupportedSetting;
5702 }
5703 return ret;
5704 } else if ((int)configIndex == (int)OMX_IndexConfigOperatingRate) {
5705 OMX_PARAM_U32TYPE *rate = (OMX_PARAM_U32TYPE *)configData;
5706 DEBUG_PRINT_LOW("Set_config: operating-rate %u fps", rate->nU32 >> 16);
5707
5708 struct v4l2_control control;
5709
5710 control.id = V4L2_CID_MPEG_VIDC_VIDEO_OPERATING_RATE;
5711 control.value = rate->nU32;
5712
5713 operating_frame_rate = rate->nU32 >> 16;
5714
5715 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
5716 ret = errno == -EBUSY ? OMX_ErrorInsufficientResources :
5717 OMX_ErrorUnsupportedSetting;
5718 DEBUG_PRINT_ERROR("Failed to set operating rate %u fps (%s)",
5719 rate->nU32 >> 16, errno == -EBUSY ? "HW Overload" : strerror(errno));
5720 }
5721 return ret;
5722
5723 } else if ((int)configIndex == (int)OMX_QTIIndexConfigDescribeColorAspects) {
5724 VALIDATE_OMX_PARAM_DATA(configData, DescribeColorAspectsParams);
5725 DescribeColorAspectsParams *params = (DescribeColorAspectsParams *)configData;
5726 if (!DEFAULT_EXTRADATA & OMX_DISPLAY_INFO_EXTRADATA) {
5727 enable_extradata(OMX_DISPLAY_INFO_EXTRADATA, true, true);
5728 }
5729
5730 print_debug_color_aspects(&(params->sAspects), "Set Config");
5731 memcpy(&m_client_color_space, params, sizeof(DescribeColorAspectsParams));
5732 return ret;
5733 } else if ((int)configIndex == (int)OMX_QTIIndexConfigDescribeHDRColorInfo) {
5734 VALIDATE_OMX_PARAM_DATA(configData, DescribeHDRStaticInfoParams);
5735 DescribeHDRStaticInfoParams *params = (DescribeHDRStaticInfoParams *)configData;
5736 if (!DEFAULT_EXTRADATA & OMX_HDR_COLOR_INFO_EXTRADATA) {
5737 ret = enable_extradata(OMX_HDR_COLOR_INFO_EXTRADATA, true, true);
5738 if (ret != OMX_ErrorNone) {
5739 DEBUG_PRINT_ERROR("Failed to enable OMX_HDR_COLOR_INFO_EXTRADATA");
5740 return ret;
5741 }
5742 }
5743
5744 print_debug_hdr_color_info(&(params->sInfo), "Set Config HDR");
5745 memcpy(&m_client_hdr_info, params, sizeof(DescribeHDRStaticInfoParams));
5746 return ret;
5747
5748 } else if ((int)configIndex == (int)OMX_IndexConfigAndroidVendorExtension) {
5749 VALIDATE_OMX_PARAM_DATA(configData, OMX_CONFIG_ANDROID_VENDOR_EXTENSIONTYPE);
5750
5751 OMX_CONFIG_ANDROID_VENDOR_EXTENSIONTYPE *ext =
5752 reinterpret_cast<OMX_CONFIG_ANDROID_VENDOR_EXTENSIONTYPE *>(configData);
5753 VALIDATE_OMX_VENDOR_EXTENSION_PARAM_DATA(ext);
5754
5755 return set_vendor_extension_config(ext);
5756 }
5757
5758 return OMX_ErrorNotImplemented;
5759 }
5760
5761 #define extn_equals(param, extn) (!strcmp(param, extn))
5762
5763 /* ======================================================================
5764 FUNCTION
5765 omx_vdec::GetExtensionIndex
5766
5767 DESCRIPTION
5768 OMX GetExtensionIndex method implementaion. <TBD>
5769
5770 PARAMETERS
5771 <TBD>.
5772
5773 RETURN VALUE
5774 OMX Error None if everything successful.
5775
5776 ========================================================================== */
get_extension_index(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_STRING paramName,OMX_OUT OMX_INDEXTYPE * indexType)5777 OMX_ERRORTYPE omx_vdec::get_extension_index(OMX_IN OMX_HANDLETYPE hComp,
5778 OMX_IN OMX_STRING paramName,
5779 OMX_OUT OMX_INDEXTYPE* indexType)
5780 {
5781 (void) hComp;
5782 if (m_state == OMX_StateInvalid) {
5783 DEBUG_PRINT_ERROR("Get Extension Index in Invalid State");
5784 return OMX_ErrorInvalidState;
5785 } else if (extn_equals(paramName, "OMX.QCOM.index.param.video.SyncFrameDecodingMode")) {
5786 *indexType = (OMX_INDEXTYPE)OMX_QcomIndexParamVideoSyncFrameDecodingMode;
5787 } else if (extn_equals(paramName, "OMX.QCOM.index.param.IndexExtraData")) {
5788 *indexType = (OMX_INDEXTYPE)OMX_QcomIndexParamIndexExtraDataType;
5789 } else if (extn_equals(paramName, OMX_QCOM_INDEX_PARAM_VIDEO_FRAMEPACKING_EXTRADATA)) {
5790 *indexType = (OMX_INDEXTYPE)OMX_QcomIndexParamVideoFramePackingExtradata;
5791 } else if (extn_equals(paramName, OMX_QCOM_INDEX_CONFIG_VIDEO_FRAMEPACKING_INFO)) {
5792 *indexType = (OMX_INDEXTYPE)OMX_QcomIndexConfigVideoFramePackingArrangement;
5793 } else if (extn_equals(paramName, OMX_QCOM_INDEX_PARAM_VIDEO_QP_EXTRADATA)) {
5794 *indexType = (OMX_INDEXTYPE)OMX_QcomIndexParamVideoQPExtraData;
5795 } else if (extn_equals(paramName, OMX_QCOM_INDEX_PARAM_VIDEO_INPUTBITSINFO_EXTRADATA)) {
5796 *indexType = (OMX_INDEXTYPE)OMX_QcomIndexParamVideoInputBitsInfoExtraData;
5797 } else if (extn_equals(paramName, OMX_QCOM_INDEX_PARAM_VIDEO_EXTNUSER_EXTRADATA)) {
5798 *indexType = (OMX_INDEXTYPE)OMX_QcomIndexEnableExtnUserData;
5799 }
5800 #if defined (_ANDROID_HONEYCOMB_) || defined (_ANDROID_ICS_)
5801 else if (extn_equals(paramName, "OMX.google.android.index.enableAndroidNativeBuffers")) {
5802 *indexType = (OMX_INDEXTYPE)OMX_GoogleAndroidIndexEnableAndroidNativeBuffers;
5803 } else if (extn_equals(paramName, "OMX.google.android.index.useAndroidNativeBuffer2")) {
5804 *indexType = (OMX_INDEXTYPE)OMX_GoogleAndroidIndexUseAndroidNativeBuffer2;
5805 } else if (extn_equals(paramName, "OMX.google.android.index.useAndroidNativeBuffer")) {
5806 *indexType = (OMX_INDEXTYPE)OMX_GoogleAndroidIndexUseAndroidNativeBuffer;
5807 } else if (extn_equals(paramName, "OMX.google.android.index.getAndroidNativeBufferUsage")) {
5808 *indexType = (OMX_INDEXTYPE)OMX_GoogleAndroidIndexGetAndroidNativeBufferUsage;
5809 }
5810 #if ALLOCATE_OUTPUT_NATIVEHANDLE
5811 else if (extn_equals(paramName, "OMX.google.android.index.allocateNativeHandle")) {
5812 *indexType = (OMX_INDEXTYPE)OMX_GoogleAndroidIndexAllocateNativeHandle;
5813 }
5814 #endif //ALLOCATE_OUTPUT_NATIVEHANDLE
5815 #endif
5816 else if (extn_equals(paramName, "OMX.google.android.index.storeMetaDataInBuffers")) {
5817 *indexType = (OMX_INDEXTYPE)OMX_QcomIndexParamVideoMetaBufferMode;
5818 }
5819 #ifdef ADAPTIVE_PLAYBACK_SUPPORTED
5820 else if (extn_equals(paramName, "OMX.google.android.index.prepareForAdaptivePlayback")) {
5821 *indexType = (OMX_INDEXTYPE)OMX_QcomIndexParamVideoAdaptivePlaybackMode;
5822 } else if (extn_equals(paramName, OMX_QTI_INDEX_PARAM_VIDEO_PREFER_ADAPTIVE_PLAYBACK)) {
5823 *indexType = (OMX_INDEXTYPE)OMX_QTIIndexParamVideoPreferAdaptivePlayback;
5824 }
5825 #endif
5826 #ifdef FLEXYUV_SUPPORTED
5827 else if (extn_equals(paramName,"OMX.google.android.index.describeColorFormat")) {
5828 *indexType = (OMX_INDEXTYPE)OMX_QcomIndexFlexibleYUVDescription;
5829 }
5830 #endif
5831 else if (extn_equals(paramName, "OMX.QCOM.index.param.video.PassInputBufferFd")) {
5832 *indexType = (OMX_INDEXTYPE)OMX_QTIIndexParamPassInputBufferFd;
5833 } else if (extn_equals(paramName, "OMX.QTI.index.param.video.ForceCompressedForDPB")) {
5834 *indexType = (OMX_INDEXTYPE)OMX_QTIIndexParamForceCompressedForDPB;
5835 } else if (extn_equals(paramName, "OMX.QTI.index.param.video.ForceUnCompressedForOPB")) {
5836 *indexType = (OMX_INDEXTYPE)OMX_QTIIndexParamForceUnCompressedForOPB;
5837 } else if (extn_equals(paramName, "OMX.QTI.index.param.video.LowLatency")) {
5838 *indexType = (OMX_INDEXTYPE)OMX_QTIIndexParamLowLatencyMode;
5839 } else if (extn_equals(paramName, OMX_QTI_INDEX_PARAM_VIDEO_CLIENT_EXTRADATA)) {
5840 *indexType = (OMX_INDEXTYPE)OMX_QTIIndexParamVideoClientExtradata;
5841 } else if (extn_equals(paramName, "OMX.google.android.index.describeColorAspects")) {
5842 *indexType = (OMX_INDEXTYPE)OMX_QTIIndexConfigDescribeColorAspects;
5843 } else if (extn_equals(paramName, "OMX.google.android.index.describeHDRStaticInfo")) {
5844 *indexType = (OMX_INDEXTYPE)OMX_QTIIndexConfigDescribeHDRColorInfo;
5845 } else {
5846 DEBUG_PRINT_HIGH("Extension: %s not implemented", paramName);
5847 return OMX_ErrorNotImplemented;
5848 }
5849 return OMX_ErrorNone;
5850 }
5851
5852 /* ======================================================================
5853 FUNCTION
5854 omx_vdec::GetState
5855
5856 DESCRIPTION
5857 Returns the state information back to the caller.<TBD>
5858
5859 PARAMETERS
5860 <TBD>.
5861
5862 RETURN VALUE
5863 Error None if everything is successful.
5864 ========================================================================== */
get_state(OMX_IN OMX_HANDLETYPE hComp,OMX_OUT OMX_STATETYPE * state)5865 OMX_ERRORTYPE omx_vdec::get_state(OMX_IN OMX_HANDLETYPE hComp,
5866 OMX_OUT OMX_STATETYPE* state)
5867 {
5868 (void) hComp;
5869 *state = m_state;
5870 DEBUG_PRINT_LOW("get_state: Returning the state %d",*state);
5871 return OMX_ErrorNone;
5872 }
5873
5874 /* ======================================================================
5875 FUNCTION
5876 omx_vdec::ComponentTunnelRequest
5877
5878 DESCRIPTION
5879 OMX Component Tunnel Request method implementation. <TBD>
5880
5881 PARAMETERS
5882 None.
5883
5884 RETURN VALUE
5885 OMX Error None if everything successful.
5886
5887 ========================================================================== */
component_tunnel_request(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_U32 port,OMX_IN OMX_HANDLETYPE peerComponent,OMX_IN OMX_U32 peerPort,OMX_INOUT OMX_TUNNELSETUPTYPE * tunnelSetup)5888 OMX_ERRORTYPE omx_vdec::component_tunnel_request(OMX_IN OMX_HANDLETYPE hComp,
5889 OMX_IN OMX_U32 port,
5890 OMX_IN OMX_HANDLETYPE peerComponent,
5891 OMX_IN OMX_U32 peerPort,
5892 OMX_INOUT OMX_TUNNELSETUPTYPE* tunnelSetup)
5893 {
5894 (void) hComp;
5895 (void) port;
5896 (void) peerComponent;
5897 (void) peerPort;
5898 (void) tunnelSetup;
5899 DEBUG_PRINT_ERROR("Error: component_tunnel_request Not Implemented");
5900 return OMX_ErrorNotImplemented;
5901 }
5902
5903 /* ======================================================================
5904 FUNCTION
5905 omx_vdec::UseOutputBuffer
5906
5907 DESCRIPTION
5908 Helper function for Use buffer in the input pin
5909
5910 PARAMETERS
5911 None.
5912
5913 RETURN VALUE
5914 true/false
5915
5916 ========================================================================== */
allocate_extradata()5917 OMX_ERRORTYPE omx_vdec::allocate_extradata()
5918 {
5919 #ifdef USE_ION
5920 if (drv_ctx.extradata_info.buffer_size) {
5921 if (drv_ctx.extradata_info.ion.ion_alloc_data.handle) {
5922 munmap((void *)drv_ctx.extradata_info.uaddr, drv_ctx.extradata_info.size);
5923 close(drv_ctx.extradata_info.ion.fd_ion_data.fd);
5924 free_ion_memory(&drv_ctx.extradata_info.ion);
5925 }
5926 drv_ctx.extradata_info.size = (drv_ctx.extradata_info.size + 4095) & (~4095);
5927 drv_ctx.extradata_info.ion.ion_device_fd = alloc_map_ion_memory(
5928 drv_ctx.extradata_info.size, 4096,
5929 &drv_ctx.extradata_info.ion.ion_alloc_data,
5930 &drv_ctx.extradata_info.ion.fd_ion_data, 0);
5931 if (drv_ctx.extradata_info.ion.ion_device_fd < 0) {
5932 DEBUG_PRINT_ERROR("Failed to alloc extradata memory");
5933 return OMX_ErrorInsufficientResources;
5934 }
5935 drv_ctx.extradata_info.uaddr = (char *)mmap(NULL,
5936 drv_ctx.extradata_info.size,
5937 PROT_READ|PROT_WRITE, MAP_SHARED,
5938 drv_ctx.extradata_info.ion.fd_ion_data.fd , 0);
5939 if (drv_ctx.extradata_info.uaddr == MAP_FAILED) {
5940 DEBUG_PRINT_ERROR("Failed to map extradata memory");
5941 close(drv_ctx.extradata_info.ion.fd_ion_data.fd);
5942 free_ion_memory(&drv_ctx.extradata_info.ion);
5943 return OMX_ErrorInsufficientResources;
5944 }
5945 }
5946 #endif
5947 if (!m_other_extradata) {
5948 m_other_extradata = (OMX_OTHER_EXTRADATATYPE *)malloc(drv_ctx.extradata_info.buffer_size);
5949 if (!m_other_extradata) {
5950 DEBUG_PRINT_ERROR("Failed to alloc memory\n");
5951 return OMX_ErrorInsufficientResources;
5952 }
5953 }
5954 return OMX_ErrorNone;
5955 }
5956
free_extradata()5957 void omx_vdec::free_extradata()
5958 {
5959 #ifdef USE_ION
5960 if (drv_ctx.extradata_info.uaddr) {
5961 munmap((void *)drv_ctx.extradata_info.uaddr, drv_ctx.extradata_info.size);
5962 close(drv_ctx.extradata_info.ion.fd_ion_data.fd);
5963 free_ion_memory(&drv_ctx.extradata_info.ion);
5964 }
5965 #endif
5966 if (m_other_extradata) {
5967 free(m_other_extradata);
5968 m_other_extradata = NULL;
5969 }
5970 }
5971
use_output_buffer(OMX_IN OMX_HANDLETYPE hComp,OMX_INOUT OMX_BUFFERHEADERTYPE ** bufferHdr,OMX_IN OMX_U32 port,OMX_IN OMX_PTR appData,OMX_IN OMX_U32 bytes,OMX_IN OMX_U8 * buffer)5972 OMX_ERRORTYPE omx_vdec::use_output_buffer(
5973 OMX_IN OMX_HANDLETYPE hComp,
5974 OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
5975 OMX_IN OMX_U32 port,
5976 OMX_IN OMX_PTR appData,
5977 OMX_IN OMX_U32 bytes,
5978 OMX_IN OMX_U8* buffer)
5979 {
5980 OMX_ERRORTYPE eRet = OMX_ErrorNone;
5981 OMX_BUFFERHEADERTYPE *bufHdr= NULL; // buffer header
5982 unsigned i= 0; // Temporary counter
5983 struct vdec_setbuffer_cmd setbuffers;
5984 OMX_PTR privateAppData = NULL;
5985 private_handle_t *handle = NULL;
5986 OMX_U8 *buff = buffer;
5987 struct v4l2_buffer buf;
5988 struct v4l2_plane plane[VIDEO_MAX_PLANES];
5989 int extra_idx = 0;
5990 (void) hComp;
5991 (void) port;
5992
5993 if (!m_out_mem_ptr) {
5994 DEBUG_PRINT_HIGH("Use_op_buf:Allocating output headers");
5995 eRet = allocate_output_headers();
5996 if (eRet == OMX_ErrorNone)
5997 eRet = allocate_extradata();
5998 }
5999
6000 if (eRet == OMX_ErrorNone) {
6001 for (i=0; i< drv_ctx.op_buf.actualcount; i++) {
6002 if (BITMASK_ABSENT(&m_out_bm_count,i)) {
6003 break;
6004 }
6005 }
6006 }
6007
6008 if (i >= drv_ctx.op_buf.actualcount) {
6009 DEBUG_PRINT_ERROR("Already using %d o/p buffers", drv_ctx.op_buf.actualcount);
6010 eRet = OMX_ErrorInsufficientResources;
6011 }
6012
6013 if (eRet != OMX_ErrorNone)
6014 return eRet;
6015
6016 if (dynamic_buf_mode) {
6017 *bufferHdr = (m_out_mem_ptr + i );
6018 (*bufferHdr)->pBuffer = NULL;
6019 if (i == (drv_ctx.op_buf.actualcount - 1) && !streaming[CAPTURE_PORT]) {
6020 enum v4l2_buf_type buf_type;
6021 int rr = 0;
6022 buf_type=V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
6023 if (rr = ioctl(drv_ctx.video_driver_fd, VIDIOC_STREAMON, &buf_type)) {
6024 DEBUG_PRINT_ERROR("STREAMON FAILED : %d", rr);
6025 return OMX_ErrorInsufficientResources;
6026 } else {
6027 streaming[CAPTURE_PORT] = true;
6028 DEBUG_PRINT_LOW("STREAMON Successful");
6029 }
6030
6031 DEBUG_PRINT_HIGH("Enabling Turbo mode");
6032 request_perf_level(VIDC_TURBO);
6033 }
6034 BITMASK_SET(&m_out_bm_count,i);
6035 (*bufferHdr)->pAppPrivate = appData;
6036 (*bufferHdr)->pBuffer = buffer;
6037 (*bufferHdr)->nAllocLen = sizeof(struct VideoDecoderOutputMetaData);
6038 return eRet;
6039 }
6040
6041 if (eRet == OMX_ErrorNone) {
6042 #if defined(_ANDROID_HONEYCOMB_) || defined(_ANDROID_ICS_)
6043 if (m_enable_android_native_buffers) {
6044 if (m_use_android_native_buffers) {
6045 UseAndroidNativeBufferParams *params = (UseAndroidNativeBufferParams *)appData;
6046 sp<android_native_buffer_t> nBuf = params->nativeBuffer;
6047 handle = (private_handle_t *)nBuf->handle;
6048 privateAppData = params->pAppPrivate;
6049 } else {
6050 handle = (private_handle_t *)buff;
6051 privateAppData = appData;
6052 }
6053 if (!handle) {
6054 DEBUG_PRINT_ERROR("handle is invalid");
6055 return OMX_ErrorBadParameter;
6056 }
6057
6058 if ((OMX_U32)handle->size < drv_ctx.op_buf.buffer_size) {
6059 if (secure_mode && secure_scaling_to_non_secure_opb) {
6060 DEBUG_PRINT_HIGH("Buffer size expected %u, got %u, but it's ok since we will never map it",
6061 (unsigned int)drv_ctx.op_buf.buffer_size, (unsigned int)handle->size);
6062 } else {
6063 DEBUG_PRINT_ERROR("Insufficient sized buffer given for playback,"
6064 " expected %u, got %u",
6065 (unsigned int)drv_ctx.op_buf.buffer_size, (unsigned int)handle->size);
6066 return OMX_ErrorBadParameter;
6067 }
6068 }
6069
6070 drv_ctx.op_buf.buffer_size = handle->size;
6071
6072 if (!m_use_android_native_buffers) {
6073 if (!secure_mode) {
6074 buff = (OMX_U8*)mmap(0, handle->size,
6075 PROT_READ|PROT_WRITE, MAP_SHARED, handle->fd, 0);
6076 if (buff == MAP_FAILED) {
6077 DEBUG_PRINT_ERROR("Failed to mmap pmem with fd = %d, size = %d", handle->fd, handle->size);
6078 return OMX_ErrorInsufficientResources;
6079 }
6080 }
6081 }
6082 #if defined(_ANDROID_ICS_)
6083 native_buffer[i].nativehandle = handle;
6084 native_buffer[i].privatehandle = handle;
6085 #endif
6086 if (!handle) {
6087 DEBUG_PRINT_ERROR("Native Buffer handle is NULL");
6088 return OMX_ErrorBadParameter;
6089 }
6090 drv_ctx.ptr_outputbuffer[i].pmem_fd = handle->fd;
6091 drv_ctx.ptr_outputbuffer[i].offset = 0;
6092 drv_ctx.ptr_outputbuffer[i].bufferaddr = buff;
6093 drv_ctx.ptr_outputbuffer[i].buffer_len = drv_ctx.op_buf.buffer_size;
6094 drv_ctx.ptr_outputbuffer[i].mmaped_size = handle->size;
6095 } else
6096 #endif
6097
6098 if (!ouput_egl_buffers && !m_use_output_pmem) {
6099 #ifdef USE_ION
6100 drv_ctx.op_buf_ion_info[i].ion_device_fd = alloc_map_ion_memory(
6101 drv_ctx.op_buf.buffer_size,drv_ctx.op_buf.alignment,
6102 &drv_ctx.op_buf_ion_info[i].ion_alloc_data,
6103 &drv_ctx.op_buf_ion_info[i].fd_ion_data,
6104 secure_mode ? SECURE_FLAGS_OUTPUT_BUFFER : 0);
6105 if (drv_ctx.op_buf_ion_info[i].ion_device_fd < 0) {
6106 DEBUG_PRINT_ERROR("ION device fd is bad %d", drv_ctx.op_buf_ion_info[i].ion_device_fd);
6107 return OMX_ErrorInsufficientResources;
6108 }
6109 drv_ctx.ptr_outputbuffer[i].pmem_fd = \
6110 drv_ctx.op_buf_ion_info[i].fd_ion_data.fd;
6111 #else
6112 drv_ctx.ptr_outputbuffer[i].pmem_fd = \
6113 open (MEM_DEVICE,O_RDWR);
6114
6115 if (drv_ctx.ptr_outputbuffer[i].pmem_fd < 0) {
6116 DEBUG_PRINT_ERROR("ION/pmem buffer fd is bad %d", drv_ctx.ptr_outputbuffer[i].pmem_fd);
6117 return OMX_ErrorInsufficientResources;
6118 }
6119
6120 /* FIXME: why is this code even here? We already open MEM_DEVICE a few lines above */
6121 if (drv_ctx.ptr_outputbuffer[i].pmem_fd == 0) {
6122 drv_ctx.ptr_outputbuffer[i].pmem_fd = \
6123 open (MEM_DEVICE,O_RDWR);
6124 if (drv_ctx.ptr_outputbuffer[i].pmem_fd < 0) {
6125 DEBUG_PRINT_ERROR("ION/pmem buffer fd is bad %d", drv_ctx.ptr_outputbuffer[i].pmem_fd);
6126 return OMX_ErrorInsufficientResources;
6127 }
6128 }
6129
6130 if (!align_pmem_buffers(drv_ctx.ptr_outputbuffer[i].pmem_fd,
6131 drv_ctx.op_buf.buffer_size,
6132 drv_ctx.op_buf.alignment)) {
6133 DEBUG_PRINT_ERROR("align_pmem_buffers() failed");
6134 close(drv_ctx.ptr_outputbuffer[i].pmem_fd);
6135 return OMX_ErrorInsufficientResources;
6136 }
6137 #endif
6138 if (!secure_mode) {
6139 drv_ctx.ptr_outputbuffer[i].bufferaddr =
6140 (unsigned char *)mmap(NULL, drv_ctx.op_buf.buffer_size,
6141 PROT_READ|PROT_WRITE, MAP_SHARED,
6142 drv_ctx.ptr_outputbuffer[i].pmem_fd,0);
6143 if (drv_ctx.ptr_outputbuffer[i].bufferaddr == MAP_FAILED) {
6144 close(drv_ctx.ptr_outputbuffer[i].pmem_fd);
6145 #ifdef USE_ION
6146 free_ion_memory(&drv_ctx.op_buf_ion_info[i]);
6147 #endif
6148 DEBUG_PRINT_ERROR("Unable to mmap output buffer");
6149 return OMX_ErrorInsufficientResources;
6150 }
6151 }
6152 drv_ctx.ptr_outputbuffer[i].offset = 0;
6153 privateAppData = appData;
6154 } else {
6155
6156 DEBUG_PRINT_LOW("Use_op_buf: out_pmem=%d",m_use_output_pmem);
6157 if (!appData || !bytes ) {
6158 if (!secure_mode && !buffer) {
6159 DEBUG_PRINT_ERROR("Bad parameters for use buffer in EGL image case");
6160 return OMX_ErrorBadParameter;
6161 }
6162 }
6163
6164 OMX_QCOM_PLATFORM_PRIVATE_LIST *pmem_list;
6165 OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *pmem_info;
6166 pmem_list = (OMX_QCOM_PLATFORM_PRIVATE_LIST*) appData;
6167 if (!pmem_list || !pmem_list->entryList || !pmem_list->entryList->entry ||
6168 !pmem_list->nEntries ||
6169 pmem_list->entryList->type != OMX_QCOM_PLATFORM_PRIVATE_PMEM) {
6170 DEBUG_PRINT_ERROR("Pmem info not valid in use buffer");
6171 return OMX_ErrorBadParameter;
6172 }
6173 pmem_info = (OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *)
6174 pmem_list->entryList->entry;
6175 DEBUG_PRINT_LOW("vdec: use buf: pmem_fd=0x%lx",
6176 pmem_info->pmem_fd);
6177 drv_ctx.ptr_outputbuffer[i].pmem_fd = pmem_info->pmem_fd;
6178 drv_ctx.ptr_outputbuffer[i].offset = pmem_info->offset;
6179 drv_ctx.ptr_outputbuffer[i].bufferaddr = buff;
6180 drv_ctx.ptr_outputbuffer[i].mmaped_size =
6181 drv_ctx.ptr_outputbuffer[i].buffer_len = drv_ctx.op_buf.buffer_size;
6182 privateAppData = appData;
6183 }
6184 m_pmem_info[i].offset = drv_ctx.ptr_outputbuffer[i].offset;
6185 m_pmem_info[i].pmem_fd = drv_ctx.ptr_outputbuffer[i].pmem_fd;
6186 m_pmem_info[i].size = drv_ctx.ptr_outputbuffer[i].buffer_len;
6187 m_pmem_info[i].mapped_size = drv_ctx.ptr_outputbuffer[i].mmaped_size;
6188 m_pmem_info[i].buffer = drv_ctx.ptr_outputbuffer[i].bufferaddr;
6189
6190 *bufferHdr = (m_out_mem_ptr + i );
6191 if (secure_mode)
6192 drv_ctx.ptr_outputbuffer[i].bufferaddr = *bufferHdr;
6193 //setbuffers.buffer_type = VDEC_BUFFER_TYPE_OUTPUT;
6194 memcpy (&setbuffers.buffer,&drv_ctx.ptr_outputbuffer[i],
6195 sizeof (vdec_bufferpayload));
6196
6197 DEBUG_PRINT_HIGH("Set the Output Buffer Idx: %d Addr: %p, pmem_fd=0x%x", i,
6198 drv_ctx.ptr_outputbuffer[i].bufferaddr,
6199 drv_ctx.ptr_outputbuffer[i].pmem_fd );
6200
6201 buf.index = i;
6202 buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
6203 buf.memory = V4L2_MEMORY_USERPTR;
6204 plane[0].length = drv_ctx.op_buf.buffer_size;
6205 plane[0].m.userptr = (unsigned long)drv_ctx.ptr_outputbuffer[i].bufferaddr -
6206 (unsigned long)drv_ctx.ptr_outputbuffer[i].offset;
6207 plane[0].reserved[0] = drv_ctx.ptr_outputbuffer[i].pmem_fd;
6208 plane[0].reserved[1] = drv_ctx.ptr_outputbuffer[i].offset;
6209 plane[0].data_offset = 0;
6210 extra_idx = EXTRADATA_IDX(drv_ctx.num_planes);
6211 if (extra_idx && (extra_idx < VIDEO_MAX_PLANES)) {
6212 plane[extra_idx].length = drv_ctx.extradata_info.buffer_size;
6213 plane[extra_idx].m.userptr = (long unsigned int) (drv_ctx.extradata_info.uaddr + i * drv_ctx.extradata_info.buffer_size);
6214 #ifdef USE_ION
6215 plane[extra_idx].reserved[0] = drv_ctx.extradata_info.ion.fd_ion_data.fd;
6216 #endif
6217 plane[extra_idx].reserved[1] = i * drv_ctx.extradata_info.buffer_size;
6218 plane[extra_idx].data_offset = 0;
6219 } else if (extra_idx >= VIDEO_MAX_PLANES) {
6220 DEBUG_PRINT_ERROR("Extradata index is more than allowed: %d", extra_idx);
6221 return OMX_ErrorBadParameter;
6222 }
6223 buf.m.planes = plane;
6224 buf.length = drv_ctx.num_planes;
6225
6226 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_PREPARE_BUF, &buf)) {
6227 DEBUG_PRINT_ERROR("Failed to prepare bufs");
6228 /*TODO: How to handle this case */
6229 return OMX_ErrorInsufficientResources;
6230 }
6231
6232 if (i == (drv_ctx.op_buf.actualcount -1) && !streaming[CAPTURE_PORT]) {
6233 enum v4l2_buf_type buf_type;
6234 buf_type=V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
6235 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_STREAMON,&buf_type)) {
6236 return OMX_ErrorInsufficientResources;
6237 } else {
6238 streaming[CAPTURE_PORT] = true;
6239 DEBUG_PRINT_LOW("STREAMON Successful");
6240 }
6241
6242 DEBUG_PRINT_HIGH("Enabling Turbo mode");
6243 request_perf_level(VIDC_TURBO);
6244 }
6245
6246 (*bufferHdr)->nAllocLen = drv_ctx.op_buf.buffer_size;
6247 if (m_enable_android_native_buffers) {
6248 DEBUG_PRINT_LOW("setting pBuffer to private_handle_t %p", handle);
6249 (*bufferHdr)->pBuffer = (OMX_U8 *)handle;
6250 } else {
6251 (*bufferHdr)->pBuffer = buff;
6252 }
6253 (*bufferHdr)->pAppPrivate = privateAppData;
6254 BITMASK_SET(&m_out_bm_count,i);
6255 }
6256 return eRet;
6257 }
6258
allocate_client_output_extradata_headers()6259 OMX_ERRORTYPE omx_vdec::allocate_client_output_extradata_headers() {
6260 OMX_ERRORTYPE eRet = OMX_ErrorNone;
6261 OMX_BUFFERHEADERTYPE *bufHdr = NULL;
6262 int i = 0;
6263
6264 if (!m_client_output_extradata_mem_ptr) {
6265 int nBufferCount = 0;
6266
6267 nBufferCount = m_client_out_extradata_info.getBufferCount();
6268 DEBUG_PRINT_HIGH("allocate_client_output_extradata_headers buffer_count - %d", nBufferCount);
6269
6270 m_client_output_extradata_mem_ptr = (OMX_BUFFERHEADERTYPE *)calloc(nBufferCount, sizeof(OMX_BUFFERHEADERTYPE));
6271
6272 if (m_client_output_extradata_mem_ptr) {
6273 bufHdr = m_client_output_extradata_mem_ptr;
6274 for (i=0; i < nBufferCount; i++) {
6275 bufHdr->nSize = sizeof(OMX_BUFFERHEADERTYPE);
6276 bufHdr->nVersion.nVersion = OMX_SPEC_VERSION;
6277 // Set the values when we determine the right HxW param
6278 bufHdr->nAllocLen = 0;
6279 bufHdr->nFilledLen = 0;
6280 bufHdr->pAppPrivate = NULL;
6281 bufHdr->nOutputPortIndex = OMX_CORE_OUTPUT_EXTRADATA_INDEX;
6282 bufHdr->pBuffer = NULL;
6283 bufHdr->pOutputPortPrivate = NULL;
6284 bufHdr++;
6285 }
6286 } else {
6287 DEBUG_PRINT_ERROR("Extradata header buf mem alloc failed[0x%p]",\
6288 m_client_output_extradata_mem_ptr);
6289 eRet = OMX_ErrorInsufficientResources;
6290 }
6291 }
6292 return eRet;
6293 }
use_client_output_extradata_buffer(OMX_IN OMX_HANDLETYPE hComp,OMX_INOUT OMX_BUFFERHEADERTYPE ** bufferHdr,OMX_IN OMX_U32 port,OMX_IN OMX_PTR appData,OMX_IN OMX_U32 bytes,OMX_IN OMX_U8 * buffer)6294 OMX_ERRORTYPE omx_vdec::use_client_output_extradata_buffer(
6295 OMX_IN OMX_HANDLETYPE hComp,
6296 OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
6297 OMX_IN OMX_U32 port,
6298 OMX_IN OMX_PTR appData,
6299 OMX_IN OMX_U32 bytes,
6300 OMX_IN OMX_U8* buffer)
6301 {
6302 OMX_ERRORTYPE eRet = OMX_ErrorNone;
6303 unsigned i = 0; // Temporary counter
6304 unsigned buffer_count = m_client_out_extradata_info.getBufferCount();;
6305 OMX_U32 buffer_size = m_client_out_extradata_info.getSize();
6306 (void) hComp;
6307
6308 if (port != OMX_CORE_OUTPUT_EXTRADATA_INDEX ||
6309 !client_extradata || bytes != buffer_size|| bufferHdr == NULL) {
6310 DEBUG_PRINT_ERROR("Bad Parameters PortIndex is - %d expected is- %d,"
6311 "client_extradata - %d, bytes = %d expected is %d bufferHdr - %p", port,
6312 OMX_CORE_OUTPUT_EXTRADATA_INDEX, client_extradata, bytes, buffer_size, bufferHdr);
6313 eRet = OMX_ErrorBadParameter;
6314 return eRet;
6315 }
6316
6317 if (!m_client_output_extradata_mem_ptr) {
6318 eRet = allocate_client_output_extradata_headers();
6319 }
6320
6321 if (eRet == OMX_ErrorNone) {
6322 for (i = 0; i < buffer_count; i++) {
6323 if (BITMASK_ABSENT(&m_out_extradata_bm_count,i)) {
6324 break;
6325 }
6326 }
6327 }
6328
6329 if (i >= buffer_count) {
6330 DEBUG_PRINT_ERROR("Already using %d Extradata o/p buffers", buffer_count);
6331 eRet = OMX_ErrorInsufficientResources;
6332 }
6333
6334 if (eRet == OMX_ErrorNone) {
6335 BITMASK_SET(&m_out_extradata_bm_count,i);
6336 *bufferHdr = (m_client_output_extradata_mem_ptr + i );
6337 (*bufferHdr)->pAppPrivate = appData;
6338 (*bufferHdr)->pBuffer = buffer;
6339 (*bufferHdr)->nAllocLen = bytes;
6340 }
6341
6342 return eRet;
6343 }
6344 /* ======================================================================
6345 FUNCTION
6346 omx_vdec::use_input_heap_buffers
6347
6348 DESCRIPTION
6349 OMX Use Buffer Heap allocation method implementation.
6350
6351 PARAMETERS
6352 <TBD>.
6353
6354 RETURN VALUE
6355 OMX Error None , if everything successful.
6356
6357 ========================================================================== */
use_input_heap_buffers(OMX_IN OMX_HANDLETYPE hComp,OMX_INOUT OMX_BUFFERHEADERTYPE ** bufferHdr,OMX_IN OMX_U32 port,OMX_IN OMX_PTR appData,OMX_IN OMX_U32 bytes,OMX_IN OMX_U8 * buffer)6358 OMX_ERRORTYPE omx_vdec::use_input_heap_buffers(
6359 OMX_IN OMX_HANDLETYPE hComp,
6360 OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
6361 OMX_IN OMX_U32 port,
6362 OMX_IN OMX_PTR appData,
6363 OMX_IN OMX_U32 bytes,
6364 OMX_IN OMX_U8* buffer)
6365 {
6366 DEBUG_PRINT_LOW("Inside %s, %p", __FUNCTION__, buffer);
6367 OMX_ERRORTYPE eRet = OMX_ErrorNone;
6368
6369 if (secure_mode) {
6370 DEBUG_PRINT_ERROR("use_input_heap_buffers is not allowed in secure mode");
6371 return OMX_ErrorUndefined;
6372 }
6373
6374 if (!m_inp_heap_ptr)
6375 m_inp_heap_ptr = (OMX_BUFFERHEADERTYPE*)
6376 calloc( (sizeof(OMX_BUFFERHEADERTYPE)),
6377 drv_ctx.ip_buf.actualcount);
6378 if (!m_phdr_pmem_ptr)
6379 m_phdr_pmem_ptr = (OMX_BUFFERHEADERTYPE**)
6380 calloc( (sizeof(OMX_BUFFERHEADERTYPE*)),
6381 drv_ctx.ip_buf.actualcount);
6382 if (!m_inp_heap_ptr || !m_phdr_pmem_ptr) {
6383 DEBUG_PRINT_ERROR("Insufficent memory");
6384 eRet = OMX_ErrorInsufficientResources;
6385 } else if (m_in_alloc_cnt < drv_ctx.ip_buf.actualcount) {
6386 input_use_buffer = true;
6387 memset(&m_inp_heap_ptr[m_in_alloc_cnt], 0, sizeof(OMX_BUFFERHEADERTYPE));
6388 m_inp_heap_ptr[m_in_alloc_cnt].pBuffer = buffer;
6389 m_inp_heap_ptr[m_in_alloc_cnt].nAllocLen = bytes;
6390 m_inp_heap_ptr[m_in_alloc_cnt].pAppPrivate = appData;
6391 m_inp_heap_ptr[m_in_alloc_cnt].nInputPortIndex = (OMX_U32) OMX_DirInput;
6392 m_inp_heap_ptr[m_in_alloc_cnt].nOutputPortIndex = (OMX_U32) OMX_DirMax;
6393 *bufferHdr = &m_inp_heap_ptr[m_in_alloc_cnt];
6394 eRet = allocate_input_buffer(hComp, &m_phdr_pmem_ptr[m_in_alloc_cnt], port, appData, bytes);
6395 DEBUG_PRINT_HIGH("Heap buffer(%p) Pmem buffer(%p)", *bufferHdr, m_phdr_pmem_ptr[m_in_alloc_cnt]);
6396 if (!m_input_free_q.insert_entry((unsigned long)m_phdr_pmem_ptr[m_in_alloc_cnt],
6397 (unsigned)NULL, (unsigned)NULL)) {
6398 DEBUG_PRINT_ERROR("ERROR:Free_q is full");
6399 return OMX_ErrorInsufficientResources;
6400 }
6401 m_in_alloc_cnt++;
6402 } else {
6403 DEBUG_PRINT_ERROR("All i/p buffers have been set!");
6404 eRet = OMX_ErrorInsufficientResources;
6405 }
6406 return eRet;
6407 }
6408
6409 /* ======================================================================
6410 FUNCTION
6411 omx_vdec::UseBuffer
6412
6413 DESCRIPTION
6414 OMX Use Buffer method implementation.
6415
6416 PARAMETERS
6417 <TBD>.
6418
6419 RETURN VALUE
6420 OMX Error None , if everything successful.
6421
6422 ========================================================================== */
use_buffer(OMX_IN OMX_HANDLETYPE hComp,OMX_INOUT OMX_BUFFERHEADERTYPE ** bufferHdr,OMX_IN OMX_U32 port,OMX_IN OMX_PTR appData,OMX_IN OMX_U32 bytes,OMX_IN OMX_U8 * buffer)6423 OMX_ERRORTYPE omx_vdec::use_buffer(
6424 OMX_IN OMX_HANDLETYPE hComp,
6425 OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
6426 OMX_IN OMX_U32 port,
6427 OMX_IN OMX_PTR appData,
6428 OMX_IN OMX_U32 bytes,
6429 OMX_IN OMX_U8* buffer)
6430 {
6431 OMX_ERRORTYPE error = OMX_ErrorNone;
6432 struct vdec_setbuffer_cmd setbuffers;
6433
6434 if (bufferHdr == NULL || bytes == 0 || (!secure_mode && buffer == NULL)) {
6435 DEBUG_PRINT_ERROR("bad param 0x%p %u 0x%p",bufferHdr, (unsigned int)bytes, buffer);
6436 return OMX_ErrorBadParameter;
6437 }
6438 if (m_state == OMX_StateInvalid) {
6439 DEBUG_PRINT_ERROR("Use Buffer in Invalid State");
6440 return OMX_ErrorInvalidState;
6441 }
6442 if (port == OMX_CORE_INPUT_PORT_INDEX) {
6443 // If this is not the first allocation (i.e m_inp_mem_ptr is allocated),
6444 // ensure that use-buffer was called for previous allocation.
6445 // Mix-and-match of useBuffer and allocateBuffer is not allowed
6446 if (m_inp_mem_ptr && !input_use_buffer) {
6447 DEBUG_PRINT_ERROR("'Use' Input buffer called after 'Allocate' Input buffer !");
6448 return OMX_ErrorUndefined;
6449 }
6450 error = use_input_heap_buffers(hComp, bufferHdr, port, appData, bytes, buffer);
6451 } else if (port == OMX_CORE_OUTPUT_PORT_INDEX) {
6452 error = use_output_buffer(hComp,bufferHdr,port,appData,bytes,buffer); //not tested
6453 } else if (port == OMX_CORE_OUTPUT_EXTRADATA_INDEX) {
6454 error = use_client_output_extradata_buffer(hComp,bufferHdr,port,appData,bytes,buffer);
6455 } else {
6456 DEBUG_PRINT_ERROR("Error: Invalid Port Index received %d",(int)port);
6457 error = OMX_ErrorBadPortIndex;
6458 }
6459 DEBUG_PRINT_LOW("Use Buffer: port %u, buffer %p, eRet %d", (unsigned int)port, *bufferHdr, error);
6460 if (error == OMX_ErrorNone) {
6461 if (allocate_done() && BITMASK_PRESENT(&m_flags,OMX_COMPONENT_IDLE_PENDING)) {
6462 // Send the callback now
6463 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_IDLE_PENDING);
6464 post_event(OMX_CommandStateSet,OMX_StateIdle,
6465 OMX_COMPONENT_GENERATE_EVENT);
6466 }
6467 if (port == OMX_CORE_INPUT_PORT_INDEX && m_inp_bPopulated &&
6468 BITMASK_PRESENT(&m_flags,OMX_COMPONENT_INPUT_ENABLE_PENDING)) {
6469 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_INPUT_ENABLE_PENDING);
6470 post_event(OMX_CommandPortEnable,
6471 OMX_CORE_INPUT_PORT_INDEX,
6472 OMX_COMPONENT_GENERATE_EVENT);
6473 } else if (port == OMX_CORE_OUTPUT_PORT_INDEX && m_out_bPopulated &&
6474 BITMASK_PRESENT(&m_flags,OMX_COMPONENT_OUTPUT_ENABLE_PENDING)) {
6475 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_OUTPUT_ENABLE_PENDING);
6476 post_event(OMX_CommandPortEnable,
6477 OMX_CORE_OUTPUT_PORT_INDEX,
6478 OMX_COMPONENT_GENERATE_EVENT);
6479 }
6480 }
6481 return error;
6482 }
6483
free_input_buffer(unsigned int bufferindex,OMX_BUFFERHEADERTYPE * pmem_bufferHdr)6484 OMX_ERRORTYPE omx_vdec::free_input_buffer(unsigned int bufferindex,
6485 OMX_BUFFERHEADERTYPE *pmem_bufferHdr)
6486 {
6487 if (m_inp_heap_ptr && !input_use_buffer && arbitrary_bytes) {
6488 if (m_inp_heap_ptr[bufferindex].pBuffer)
6489 free(m_inp_heap_ptr[bufferindex].pBuffer);
6490 m_inp_heap_ptr[bufferindex].pBuffer = NULL;
6491 }
6492 if (pmem_bufferHdr)
6493 free_input_buffer(pmem_bufferHdr);
6494 return OMX_ErrorNone;
6495 }
6496
free_input_buffer(OMX_BUFFERHEADERTYPE * bufferHdr)6497 OMX_ERRORTYPE omx_vdec::free_input_buffer(OMX_BUFFERHEADERTYPE *bufferHdr)
6498 {
6499 unsigned int index = 0;
6500 if (bufferHdr == NULL || m_inp_mem_ptr == NULL) {
6501 return OMX_ErrorBadParameter;
6502 }
6503
6504 index = bufferHdr - m_inp_mem_ptr;
6505 DEBUG_PRINT_LOW("Free Input Buffer index = %d",index);
6506
6507 auto_lock l(buf_lock);
6508 bufferHdr->pInputPortPrivate = NULL;
6509
6510 if (index < drv_ctx.ip_buf.actualcount && drv_ctx.ptr_inputbuffer) {
6511 DEBUG_PRINT_LOW("Free Input Buffer index = %d",index);
6512 if (drv_ctx.ptr_inputbuffer[index].pmem_fd >= 0) {
6513 struct vdec_setbuffer_cmd setbuffers;
6514 setbuffers.buffer_type = VDEC_BUFFER_TYPE_INPUT;
6515 memcpy (&setbuffers.buffer,&drv_ctx.ptr_inputbuffer[index],
6516 sizeof (vdec_bufferpayload));
6517 if (!secure_mode) {
6518 DEBUG_PRINT_LOW("unmap the input buffer fd=%d",
6519 drv_ctx.ptr_inputbuffer[index].pmem_fd);
6520 DEBUG_PRINT_LOW("unmap the input buffer size=%u address = %p",
6521 (unsigned int)drv_ctx.ptr_inputbuffer[index].mmaped_size,
6522 drv_ctx.ptr_inputbuffer[index].bufferaddr);
6523 munmap (drv_ctx.ptr_inputbuffer[index].bufferaddr,
6524 drv_ctx.ptr_inputbuffer[index].mmaped_size);
6525 }
6526
6527 if (allocate_native_handle){
6528 native_handle_t *nh = (native_handle_t *)bufferHdr->pBuffer;
6529 native_handle_close(nh);
6530 native_handle_delete(nh);
6531 } else {
6532 // Close fd for non-secure and secure non-native-handle case
6533 close(drv_ctx.ptr_inputbuffer[index].pmem_fd);
6534 }
6535 drv_ctx.ptr_inputbuffer[index].pmem_fd = -1;
6536
6537 if (m_desc_buffer_ptr && m_desc_buffer_ptr[index].buf_addr) {
6538 free(m_desc_buffer_ptr[index].buf_addr);
6539 m_desc_buffer_ptr[index].buf_addr = NULL;
6540 m_desc_buffer_ptr[index].desc_data_size = 0;
6541 }
6542 #ifdef USE_ION
6543 free_ion_memory(&drv_ctx.ip_buf_ion_info[index]);
6544 #endif
6545 m_in_alloc_cnt--;
6546 }
6547 }
6548
6549 return OMX_ErrorNone;
6550 }
6551
free_output_buffer(OMX_BUFFERHEADERTYPE * bufferHdr)6552 OMX_ERRORTYPE omx_vdec::free_output_buffer(OMX_BUFFERHEADERTYPE *bufferHdr)
6553 {
6554 unsigned int index = 0;
6555
6556 if (bufferHdr == NULL || m_out_mem_ptr == NULL) {
6557 return OMX_ErrorBadParameter;
6558 }
6559
6560 index = bufferHdr - m_out_mem_ptr;
6561 DEBUG_PRINT_LOW("Free ouput Buffer index = %d",index);
6562
6563 if (index < drv_ctx.op_buf.actualcount
6564 && drv_ctx.ptr_outputbuffer) {
6565 DEBUG_PRINT_LOW("Free ouput Buffer index = %d addr = %p", index,
6566 drv_ctx.ptr_outputbuffer[index].bufferaddr);
6567
6568 struct vdec_setbuffer_cmd setbuffers;
6569 setbuffers.buffer_type = VDEC_BUFFER_TYPE_OUTPUT;
6570 memcpy (&setbuffers.buffer,&drv_ctx.ptr_outputbuffer[index],
6571 sizeof (vdec_bufferpayload));
6572
6573 if (!dynamic_buf_mode) {
6574 if (streaming[CAPTURE_PORT] &&
6575 !(in_reconfig || BITMASK_PRESENT(&m_flags,OMX_COMPONENT_OUTPUT_FLUSH_PENDING))) {
6576 if (stream_off(OMX_CORE_OUTPUT_PORT_INDEX)) {
6577 DEBUG_PRINT_ERROR("STREAMOFF Failed");
6578 } else {
6579 DEBUG_PRINT_LOW("STREAMOFF Successful");
6580 }
6581 }
6582 #ifdef _ANDROID_
6583 if (m_enable_android_native_buffers) {
6584 if (!secure_mode) {
6585 if (drv_ctx.ptr_outputbuffer[index].pmem_fd > 0) {
6586 munmap(drv_ctx.ptr_outputbuffer[index].bufferaddr,
6587 drv_ctx.ptr_outputbuffer[index].mmaped_size);
6588 }
6589 }
6590 drv_ctx.ptr_outputbuffer[index].pmem_fd = -1;
6591 } else {
6592 #endif
6593 if (drv_ctx.ptr_outputbuffer[index].pmem_fd > 0
6594 && !ouput_egl_buffers && !m_use_output_pmem) {
6595 if (drv_ctx.op_buf_map_info[index].free_buffer) {
6596 if (!secure_mode) {
6597 DEBUG_PRINT_LOW("unmap the output buffer fd = %d",
6598 drv_ctx.ptr_outputbuffer[0].pmem_fd);
6599 DEBUG_PRINT_LOW("unmap the ouput buffer size=%u address = %p",
6600 (unsigned int)drv_ctx.op_buf_map_info[index].map_size,
6601 drv_ctx.op_buf_map_info[index].base_address);
6602 munmap (drv_ctx.op_buf_map_info[index].base_address,
6603 drv_ctx.op_buf_map_info[index].map_size);
6604 }
6605 close (drv_ctx.ptr_outputbuffer[index].pmem_fd);
6606 drv_ctx.ptr_outputbuffer[index].pmem_fd = -1;
6607 #ifdef USE_ION
6608 free_ion_memory(&drv_ctx.op_buf_ion_info[index]);
6609 #endif
6610 } else {
6611 drv_ctx.op_buf_ion_info[index].ion_device_fd = -1;
6612 drv_ctx.op_buf_ion_info[index].ion_alloc_data.handle = 0;
6613 drv_ctx.op_buf_ion_info[index].fd_ion_data.fd = -1;
6614 }
6615 drv_ctx.op_buf_map_info[index].free_buffer = false;
6616 drv_ctx.op_buf_map_info[index].base_address = NULL;
6617 drv_ctx.op_buf_map_info[index].map_size = 0;
6618 drv_ctx.op_buf_map_info[index].offset = 0;
6619 }
6620 #ifdef _ANDROID_
6621 }
6622 #endif
6623 } //!dynamic_buf_mode
6624 if (release_output_done()) {
6625 free_extradata();
6626 }
6627 }
6628
6629 return OMX_ErrorNone;
6630
6631 }
6632
allocate_input_heap_buffer(OMX_HANDLETYPE hComp,OMX_BUFFERHEADERTYPE ** bufferHdr,OMX_U32 port,OMX_PTR appData,OMX_U32 bytes)6633 OMX_ERRORTYPE omx_vdec::allocate_input_heap_buffer(OMX_HANDLETYPE hComp,
6634 OMX_BUFFERHEADERTYPE **bufferHdr,
6635 OMX_U32 port,
6636 OMX_PTR appData,
6637 OMX_U32 bytes)
6638 {
6639 OMX_BUFFERHEADERTYPE *input = NULL;
6640 unsigned char *buf_addr = NULL;
6641 OMX_ERRORTYPE eRet = OMX_ErrorNone;
6642 unsigned i = 0;
6643
6644 /* Sanity Check*/
6645 if (bufferHdr == NULL) {
6646 return OMX_ErrorBadParameter;
6647 }
6648
6649 if (m_inp_heap_ptr == NULL) {
6650 m_inp_heap_ptr = (OMX_BUFFERHEADERTYPE*) \
6651 calloc( (sizeof(OMX_BUFFERHEADERTYPE)),
6652 drv_ctx.ip_buf.actualcount);
6653 m_phdr_pmem_ptr = (OMX_BUFFERHEADERTYPE**) \
6654 calloc( (sizeof(OMX_BUFFERHEADERTYPE*)),
6655 drv_ctx.ip_buf.actualcount);
6656
6657 if (m_inp_heap_ptr == NULL || m_phdr_pmem_ptr == NULL) {
6658 DEBUG_PRINT_ERROR("m_inp_heap_ptr or m_phdr_pmem_ptr Allocation failed ");
6659 return OMX_ErrorInsufficientResources;
6660 }
6661 }
6662
6663 /*Find a Free index*/
6664 for (i=0; i< drv_ctx.ip_buf.actualcount; i++) {
6665 if (BITMASK_ABSENT(&m_heap_inp_bm_count,i)) {
6666 DEBUG_PRINT_LOW("Free Input Buffer Index %d",i);
6667 break;
6668 }
6669 }
6670
6671 if (i < drv_ctx.ip_buf.actualcount) {
6672 buf_addr = (unsigned char *)malloc (drv_ctx.ip_buf.buffer_size);
6673
6674 if (buf_addr == NULL) {
6675 return OMX_ErrorInsufficientResources;
6676 }
6677
6678 *bufferHdr = (m_inp_heap_ptr + i);
6679 input = *bufferHdr;
6680 BITMASK_SET(&m_heap_inp_bm_count,i);
6681
6682 input->pBuffer = (OMX_U8 *)buf_addr;
6683 input->nSize = sizeof(OMX_BUFFERHEADERTYPE);
6684 input->nVersion.nVersion = OMX_SPEC_VERSION;
6685 input->nAllocLen = drv_ctx.ip_buf.buffer_size;
6686 input->pAppPrivate = appData;
6687 input->nInputPortIndex = OMX_CORE_INPUT_PORT_INDEX;
6688 DEBUG_PRINT_LOW("Address of Heap Buffer %p",*bufferHdr );
6689 eRet = allocate_input_buffer(hComp,&m_phdr_pmem_ptr [i],port,appData,bytes);
6690 DEBUG_PRINT_LOW("Address of Pmem Buffer %p",m_phdr_pmem_ptr[i]);
6691 /*Add the Buffers to freeq*/
6692 if (!m_input_free_q.insert_entry((unsigned long)m_phdr_pmem_ptr[i],
6693 (unsigned)NULL, (unsigned)NULL)) {
6694 DEBUG_PRINT_ERROR("ERROR:Free_q is full");
6695 return OMX_ErrorInsufficientResources;
6696 }
6697 } else {
6698 return OMX_ErrorBadParameter;
6699 }
6700
6701 return eRet;
6702
6703 }
6704
6705
6706 /* ======================================================================
6707 FUNCTION
6708 omx_vdec::AllocateInputBuffer
6709
6710 DESCRIPTION
6711 Helper function for allocate buffer in the input pin
6712
6713 PARAMETERS
6714 None.
6715
6716 RETURN VALUE
6717 true/false
6718
6719 ========================================================================== */
allocate_input_buffer(OMX_IN OMX_HANDLETYPE hComp,OMX_INOUT OMX_BUFFERHEADERTYPE ** bufferHdr,OMX_IN OMX_U32 port,OMX_IN OMX_PTR appData,OMX_IN OMX_U32 bytes)6720 OMX_ERRORTYPE omx_vdec::allocate_input_buffer(
6721 OMX_IN OMX_HANDLETYPE hComp,
6722 OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
6723 OMX_IN OMX_U32 port,
6724 OMX_IN OMX_PTR appData,
6725 OMX_IN OMX_U32 bytes)
6726 {
6727 OMX_ERRORTYPE eRet = OMX_ErrorNone;
6728 struct vdec_setbuffer_cmd setbuffers;
6729 OMX_BUFFERHEADERTYPE *input = NULL;
6730 unsigned i = 0;
6731 unsigned char *buf_addr = NULL;
6732 int pmem_fd = -1, ret = 0;
6733
6734 (void) hComp;
6735 (void) port;
6736
6737
6738 if (bytes != drv_ctx.ip_buf.buffer_size) {
6739 DEBUG_PRINT_LOW("Requested Size is wrong %u epected is %u",
6740 (unsigned int)bytes, (unsigned int)drv_ctx.ip_buf.buffer_size);
6741 return OMX_ErrorBadParameter;
6742 }
6743
6744 if (!m_inp_mem_ptr) {
6745 struct v4l2_requestbuffers bufreq;
6746 bufreq.memory = V4L2_MEMORY_USERPTR;
6747 bufreq.type=V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
6748 bufreq.count = drv_ctx.ip_buf.actualcount;
6749 ret = ioctl(drv_ctx.video_driver_fd,VIDIOC_REQBUFS, &bufreq);
6750 if (ret) {
6751 DEBUG_PRINT_ERROR("Setting buffer requirements (reqbufs) failed %s", strerror(errno));
6752 /*TODO: How to handle this case */
6753 eRet = OMX_ErrorInsufficientResources;
6754 } else if (bufreq.count != drv_ctx.ip_buf.actualcount) {
6755 DEBUG_PRINT_ERROR("%s Count(%d) is not expected to change to %d",
6756 __FUNCTION__, drv_ctx.ip_buf.actualcount, bufreq.count);
6757 eRet = OMX_ErrorInsufficientResources;
6758 }
6759
6760 DEBUG_PRINT_HIGH("Allocate i/p buffer Header: Cnt(%d) Sz(%u)",
6761 drv_ctx.ip_buf.actualcount,
6762 (unsigned int)drv_ctx.ip_buf.buffer_size);
6763
6764 m_inp_mem_ptr = (OMX_BUFFERHEADERTYPE*) \
6765 calloc( (sizeof(OMX_BUFFERHEADERTYPE)), drv_ctx.ip_buf.actualcount);
6766
6767 if (m_inp_mem_ptr == NULL) {
6768 return OMX_ErrorInsufficientResources;
6769 }
6770
6771 drv_ctx.ptr_inputbuffer = (struct vdec_bufferpayload *) \
6772 calloc ((sizeof (struct vdec_bufferpayload)),drv_ctx.ip_buf.actualcount);
6773
6774 if (drv_ctx.ptr_inputbuffer == NULL) {
6775 return OMX_ErrorInsufficientResources;
6776 }
6777 #ifdef USE_ION
6778 drv_ctx.ip_buf_ion_info = (struct vdec_ion *) \
6779 calloc ((sizeof (struct vdec_ion)),drv_ctx.ip_buf.actualcount);
6780
6781 if (drv_ctx.ip_buf_ion_info == NULL) {
6782 return OMX_ErrorInsufficientResources;
6783 }
6784 #endif
6785
6786 for (i=0; i < drv_ctx.ip_buf.actualcount; i++) {
6787 drv_ctx.ptr_inputbuffer [i].pmem_fd = -1;
6788 #ifdef USE_ION
6789 drv_ctx.ip_buf_ion_info[i].ion_device_fd = -1;
6790 #endif
6791 }
6792 }
6793
6794 for (i=0; i< drv_ctx.ip_buf.actualcount; i++) {
6795 if (BITMASK_ABSENT(&m_inp_bm_count,i)) {
6796 DEBUG_PRINT_LOW("Free Input Buffer Index %d",i);
6797 break;
6798 }
6799 }
6800
6801 if (i < drv_ctx.ip_buf.actualcount) {
6802 struct v4l2_buffer buf;
6803 struct v4l2_plane plane;
6804 int rc;
6805 DEBUG_PRINT_LOW("Allocate input Buffer");
6806 #ifdef USE_ION
6807 drv_ctx.ip_buf_ion_info[i].ion_device_fd = alloc_map_ion_memory(
6808 drv_ctx.ip_buf.buffer_size,drv_ctx.op_buf.alignment,
6809 &drv_ctx.ip_buf_ion_info[i].ion_alloc_data,
6810 &drv_ctx.ip_buf_ion_info[i].fd_ion_data, secure_mode ?
6811 SECURE_FLAGS_INPUT_BUFFER : 0);
6812 if (drv_ctx.ip_buf_ion_info[i].ion_device_fd < 0) {
6813 return OMX_ErrorInsufficientResources;
6814 }
6815 pmem_fd = drv_ctx.ip_buf_ion_info[i].fd_ion_data.fd;
6816 #else
6817 pmem_fd = open (MEM_DEVICE,O_RDWR);
6818
6819 if (pmem_fd < 0) {
6820 DEBUG_PRINT_ERROR("open failed for pmem/adsp for input buffer");
6821 return OMX_ErrorInsufficientResources;
6822 }
6823
6824 if (pmem_fd == 0) {
6825 pmem_fd = open (MEM_DEVICE,O_RDWR);
6826
6827 if (pmem_fd < 0) {
6828 DEBUG_PRINT_ERROR("open failed for pmem/adsp for input buffer");
6829 return OMX_ErrorInsufficientResources;
6830 }
6831 }
6832
6833 if (!align_pmem_buffers(pmem_fd, drv_ctx.ip_buf.buffer_size,
6834 drv_ctx.ip_buf.alignment)) {
6835 DEBUG_PRINT_ERROR("align_pmem_buffers() failed");
6836 close(pmem_fd);
6837 return OMX_ErrorInsufficientResources;
6838 }
6839 #endif
6840 if (!secure_mode) {
6841 buf_addr = (unsigned char *)mmap(NULL,
6842 drv_ctx.ip_buf.buffer_size,
6843 PROT_READ|PROT_WRITE, MAP_SHARED, pmem_fd, 0);
6844
6845 if (buf_addr == MAP_FAILED) {
6846 close(pmem_fd);
6847 #ifdef USE_ION
6848 free_ion_memory(&drv_ctx.ip_buf_ion_info[i]);
6849 #endif
6850 DEBUG_PRINT_ERROR("Map Failed to allocate input buffer");
6851 return OMX_ErrorInsufficientResources;
6852 }
6853 }
6854 *bufferHdr = (m_inp_mem_ptr + i);
6855 if (secure_mode)
6856 drv_ctx.ptr_inputbuffer [i].bufferaddr = *bufferHdr;
6857 else
6858 drv_ctx.ptr_inputbuffer [i].bufferaddr = buf_addr;
6859 drv_ctx.ptr_inputbuffer [i].pmem_fd = pmem_fd;
6860 drv_ctx.ptr_inputbuffer [i].buffer_len = drv_ctx.ip_buf.buffer_size;
6861 drv_ctx.ptr_inputbuffer [i].mmaped_size = drv_ctx.ip_buf.buffer_size;
6862 drv_ctx.ptr_inputbuffer [i].offset = 0;
6863
6864
6865 buf.index = i;
6866 buf.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
6867 buf.memory = V4L2_MEMORY_USERPTR;
6868 plane.bytesused = 0;
6869 plane.length = drv_ctx.ptr_inputbuffer [i].mmaped_size;
6870 plane.m.userptr = (unsigned long)drv_ctx.ptr_inputbuffer[i].bufferaddr;
6871 plane.reserved[0] =drv_ctx.ptr_inputbuffer [i].pmem_fd;
6872 plane.reserved[1] = 0;
6873 plane.data_offset = drv_ctx.ptr_inputbuffer[i].offset;
6874 buf.m.planes = &plane;
6875 buf.length = 1;
6876
6877 DEBUG_PRINT_LOW("Set the input Buffer Idx: %d Addr: %p", i,
6878 drv_ctx.ptr_inputbuffer[i].bufferaddr);
6879
6880 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_PREPARE_BUF, &buf);
6881
6882 if (rc) {
6883 DEBUG_PRINT_ERROR("Failed to prepare bufs");
6884 /*TODO: How to handle this case */
6885 return OMX_ErrorInsufficientResources;
6886 }
6887
6888 input = *bufferHdr;
6889 BITMASK_SET(&m_inp_bm_count,i);
6890 DEBUG_PRINT_LOW("Buffer address %p of pmem",*bufferHdr);
6891 if (allocate_native_handle) {
6892 native_handle_t *nh = native_handle_create(1 /*numFds*/, 0 /*numInts*/);
6893 if (!nh) {
6894 DEBUG_PRINT_ERROR("Native handle create failed");
6895 return OMX_ErrorInsufficientResources;
6896 }
6897 nh->data[0] = drv_ctx.ptr_inputbuffer[i].pmem_fd;
6898 input->pBuffer = (OMX_U8 *)nh;
6899 } else if (secure_mode || m_input_pass_buffer_fd) {
6900 /*Legacy method, pass ion fd stashed directly in pBuffer*/
6901 input->pBuffer = (OMX_U8 *)(intptr_t)drv_ctx.ptr_inputbuffer[i].pmem_fd;
6902 } else {
6903 input->pBuffer = (OMX_U8 *)buf_addr;
6904 }
6905 input->nSize = sizeof(OMX_BUFFERHEADERTYPE);
6906 input->nVersion.nVersion = OMX_SPEC_VERSION;
6907 input->nAllocLen = drv_ctx.ip_buf.buffer_size;
6908 input->pAppPrivate = appData;
6909 input->nInputPortIndex = OMX_CORE_INPUT_PORT_INDEX;
6910 input->pInputPortPrivate = (void *)&drv_ctx.ptr_inputbuffer [i];
6911
6912 if (drv_ctx.disable_dmx) {
6913 eRet = allocate_desc_buffer(i);
6914 }
6915 } else {
6916 DEBUG_PRINT_ERROR("ERROR:Input Buffer Index not found");
6917 eRet = OMX_ErrorInsufficientResources;
6918 }
6919 return eRet;
6920 }
6921
6922
6923 /* ======================================================================
6924 FUNCTION
6925 omx_vdec::AllocateOutputBuffer
6926
6927 DESCRIPTION
6928 Helper fn for AllocateBuffer in the output pin
6929
6930 PARAMETERS
6931 <TBD>.
6932
6933 RETURN VALUE
6934 OMX Error None if everything went well.
6935
6936 ========================================================================== */
allocate_output_buffer(OMX_IN OMX_HANDLETYPE hComp,OMX_INOUT OMX_BUFFERHEADERTYPE ** bufferHdr,OMX_IN OMX_U32 port,OMX_IN OMX_PTR appData,OMX_IN OMX_U32 bytes)6937 OMX_ERRORTYPE omx_vdec::allocate_output_buffer(
6938 OMX_IN OMX_HANDLETYPE hComp,
6939 OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
6940 OMX_IN OMX_U32 port,
6941 OMX_IN OMX_PTR appData,
6942 OMX_IN OMX_U32 bytes)
6943 {
6944 (void)hComp;
6945 (void)port;
6946 OMX_ERRORTYPE eRet = OMX_ErrorNone;
6947 OMX_BUFFERHEADERTYPE *bufHdr= NULL; // buffer header
6948 unsigned i= 0; // Temporary counter
6949 struct vdec_setbuffer_cmd setbuffers;
6950 int extra_idx = 0;
6951 #ifdef USE_ION
6952 int ion_device_fd =-1;
6953 struct ion_allocation_data ion_alloc_data;
6954 struct ion_fd_data fd_ion_data;
6955 #endif
6956 if (!m_out_mem_ptr) {
6957 DEBUG_PRINT_HIGH("Allocate o/p buffer Header: Cnt(%d) Sz(%u)",
6958 drv_ctx.op_buf.actualcount,
6959 (unsigned int)drv_ctx.op_buf.buffer_size);
6960 int nBufHdrSize = 0;
6961 int nPlatformEntrySize = 0;
6962 int nPlatformListSize = 0;
6963 int nPMEMInfoSize = 0;
6964
6965 OMX_QCOM_PLATFORM_PRIVATE_LIST *pPlatformList;
6966 OMX_QCOM_PLATFORM_PRIVATE_ENTRY *pPlatformEntry;
6967 OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *pPMEMInfo;
6968
6969 DEBUG_PRINT_LOW("Allocating First Output Buffer(%d)",
6970 drv_ctx.op_buf.actualcount);
6971 nBufHdrSize = drv_ctx.op_buf.actualcount *
6972 sizeof(OMX_BUFFERHEADERTYPE);
6973
6974 nPMEMInfoSize = drv_ctx.op_buf.actualcount *
6975 sizeof(OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO);
6976 nPlatformListSize = drv_ctx.op_buf.actualcount *
6977 sizeof(OMX_QCOM_PLATFORM_PRIVATE_LIST);
6978 nPlatformEntrySize = drv_ctx.op_buf.actualcount *
6979 sizeof(OMX_QCOM_PLATFORM_PRIVATE_ENTRY);
6980
6981 DEBUG_PRINT_LOW("TotalBufHdr %d BufHdrSize %u PMEM %d PL %d",nBufHdrSize,
6982 (unsigned int)sizeof(OMX_BUFFERHEADERTYPE),
6983 nPMEMInfoSize,
6984 nPlatformListSize);
6985 DEBUG_PRINT_LOW("PE %d OutputBuffer Count %d",nPlatformEntrySize,
6986 drv_ctx.op_buf.actualcount);
6987 unsigned char *pmem_baseaddress_temp[MAX_NUM_INPUT_OUTPUT_BUFFERS] = {NULL};
6988 int pmem_fd[MAX_NUM_INPUT_OUTPUT_BUFFERS] = {-1};
6989 int alloc_size[MAX_NUM_INPUT_OUTPUT_BUFFERS] = {0};
6990 static const int block_max_size = 128 * 1024 * 1024;
6991 unsigned int block_buf_count = block_max_size / drv_ctx.op_buf.buffer_size;
6992 unsigned int last_block_buf_count = 0, block_count = 1;
6993 if (drv_ctx.op_buf.actualcount <= block_buf_count) {
6994 block_buf_count = drv_ctx.op_buf.actualcount;
6995 } else {
6996 last_block_buf_count = drv_ctx.op_buf.actualcount % block_buf_count;
6997 block_count = (drv_ctx.op_buf.actualcount + block_buf_count - 1) / block_buf_count;
6998 }
6999 unsigned int idx = 0;
7000 DEBUG_PRINT_LOW("block_count = %d block_buf_count = %d last_block_buf_count = %d",
7001 block_count, block_buf_count, last_block_buf_count);
7002 #ifdef USE_ION
7003 struct vdec_ion op_buf_ion_info_temp[MAX_NUM_INPUT_OUTPUT_BUFFERS];
7004 int cache_flag = client_buffers.is_color_conversion_enabled() ? 0 : ION_FLAG_CACHED;
7005 for (; idx < block_count; idx++) {
7006 int cnt = ((idx == block_count - 1) && last_block_buf_count) ?
7007 last_block_buf_count : block_buf_count;
7008 alloc_size[idx] = drv_ctx.op_buf.buffer_size * cnt;
7009 // Allocate output buffers as cached to improve performance of software-reading
7010 // of the YUVs. Output buffers are cache-invalidated in driver.
7011 // If color-conversion is involved, Only the C2D output buffers are cached, no
7012 // need to cache the decoder's output buffers
7013 op_buf_ion_info_temp[idx].ion_device_fd = alloc_map_ion_memory(
7014 alloc_size[idx],
7015 secure_scaling_to_non_secure_opb ? SZ_4K : drv_ctx.op_buf.alignment,
7016 &op_buf_ion_info_temp[idx].ion_alloc_data, &op_buf_ion_info_temp[idx].fd_ion_data,
7017 (secure_mode && !secure_scaling_to_non_secure_opb) ?
7018 SECURE_FLAGS_OUTPUT_BUFFER : cache_flag);
7019 if (op_buf_ion_info_temp[idx].ion_device_fd < 0) {
7020 DEBUG_PRINT_LOW("Failed to allocate chunk %ul size = %d", idx, alloc_size[idx]);
7021 break;
7022 }
7023 pmem_fd[idx] = op_buf_ion_info_temp[idx].fd_ion_data.fd;
7024 DEBUG_PRINT_LOW("Allocated chunk %ul fd = %ul size = %d", idx, pmem_fd[idx], alloc_size[idx]);
7025 }
7026 if (idx != block_count) {
7027 for (OMX_U32 i = 0; i < idx; i++) {
7028 close(pmem_fd[i]);
7029 free_ion_memory(&op_buf_ion_info_temp[i]);
7030 }
7031 return OMX_ErrorInsufficientResources;
7032 }
7033
7034 #else
7035 for (idx = 0; idx < block_count; idx++) {
7036 pmem_fd[idx] = open (MEM_DEVICE,O_RDWR);
7037
7038 if (pmem_fd[idx] < 0) {
7039 DEBUG_PRINT_ERROR("ERROR:pmem fd for output buffer %d",
7040 drv_ctx.op_buf.buffer_size);
7041 break;
7042 }
7043 int cnt = ((idx == block_count - 1) && last_block_buf_count) ?
7044 last_block_buf_count : block_buf_count;
7045 if (!align_pmem_buffers(pmem_fd[idx], drv_ctx.op_buf.buffer_size * cnt,
7046 drv_ctx.op_buf.alignment)) {
7047 DEBUG_PRINT_ERROR("align_pmem_buffers() failed");
7048 close(pmem_fd[idx]);
7049 break;
7050 }
7051 }
7052 if (idx != block_count) {
7053 for (i = 0; i < idx; i++) {
7054 close(pmem_fd[i]);
7055 }
7056 return OMX_ErrorInsufficientResources;
7057 }
7058 #endif
7059 if (!secure_mode) {
7060 idx = 0;
7061 for (; idx < block_count; idx++) {
7062 pmem_baseaddress_temp[idx] = (unsigned char *)mmap(NULL,
7063 alloc_size[idx],
7064 PROT_READ|PROT_WRITE, MAP_SHARED, pmem_fd[idx], 0);
7065
7066 if (pmem_baseaddress_temp[idx] == MAP_FAILED) {
7067 DEBUG_PRINT_ERROR("MMAP failed for Size %u for fd = %d",
7068 (unsigned int)alloc_size[idx], pmem_fd[idx]);
7069 close(pmem_fd[idx]);
7070 #ifdef USE_ION
7071 free_ion_memory(&op_buf_ion_info_temp[idx]);
7072 #endif
7073 break;
7074 }
7075 }
7076 if (idx != block_count) {
7077 for (i = 0; i < idx; i++) {
7078 munmap(pmem_baseaddress_temp[idx], alloc_size[idx]);
7079 close(pmem_fd[i]);
7080 #ifdef USE_ION
7081 free_ion_memory(&op_buf_ion_info_temp[i]);
7082 #endif
7083 }
7084 return OMX_ErrorInsufficientResources;
7085 }
7086 }
7087 m_out_mem_ptr = (OMX_BUFFERHEADERTYPE *)calloc(nBufHdrSize,1);
7088 // Alloc mem for platform specific info
7089 char *pPtr=NULL;
7090 pPtr = (char*) calloc(nPlatformListSize + nPlatformEntrySize +
7091 nPMEMInfoSize,1);
7092 drv_ctx.ptr_outputbuffer = (struct vdec_bufferpayload *)\
7093 calloc (sizeof(struct vdec_bufferpayload),
7094 drv_ctx.op_buf.actualcount);
7095 drv_ctx.ptr_respbuffer = (struct vdec_output_frameinfo *)\
7096 calloc (sizeof (struct vdec_output_frameinfo),
7097 drv_ctx.op_buf.actualcount);
7098 if (!drv_ctx.ptr_outputbuffer || !drv_ctx.ptr_respbuffer) {
7099 DEBUG_PRINT_ERROR("Failed to alloc drv_ctx.ptr_outputbuffer or drv_ctx.ptr_respbuffer ");
7100 return OMX_ErrorInsufficientResources;
7101 }
7102
7103 #ifdef USE_ION
7104 drv_ctx.op_buf_ion_info = (struct vdec_ion *)\
7105 calloc (sizeof(struct vdec_ion),
7106 drv_ctx.op_buf.actualcount);
7107 if (!drv_ctx.op_buf_ion_info) {
7108 DEBUG_PRINT_ERROR("Failed to alloc drv_ctx.op_buf_ion_info");
7109 return OMX_ErrorInsufficientResources;
7110 }
7111 #endif
7112 drv_ctx.op_buf_map_info = (struct vdec_ion_map_info *)\
7113 calloc (sizeof(struct vdec_ion_map_info),
7114 drv_ctx.op_buf.actualcount);
7115 if (!drv_ctx.op_buf_map_info) {
7116 DEBUG_PRINT_ERROR("Failed to alloc drv_ctx.op_buf_map_info");
7117 return OMX_ErrorInsufficientResources;
7118 }
7119
7120 if (m_out_mem_ptr && pPtr && drv_ctx.ptr_outputbuffer
7121 && drv_ctx.ptr_respbuffer) {
7122 drv_ctx.ptr_outputbuffer[0].mmaped_size =
7123 (drv_ctx.op_buf.buffer_size *
7124 drv_ctx.op_buf.actualcount);
7125 bufHdr = m_out_mem_ptr;
7126 m_platform_list = (OMX_QCOM_PLATFORM_PRIVATE_LIST *)(pPtr);
7127 m_platform_entry= (OMX_QCOM_PLATFORM_PRIVATE_ENTRY *)
7128 (((char *) m_platform_list) + nPlatformListSize);
7129 m_pmem_info = (OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *)
7130 (((char *) m_platform_entry) + nPlatformEntrySize);
7131 pPlatformList = m_platform_list;
7132 pPlatformEntry = m_platform_entry;
7133 pPMEMInfo = m_pmem_info;
7134
7135 DEBUG_PRINT_LOW("Memory Allocation Succeeded for OUT port%p",m_out_mem_ptr);
7136
7137 // Settting the entire storage nicely
7138 DEBUG_PRINT_LOW("bHdr %p OutMem %p PE %p",bufHdr, m_out_mem_ptr,pPlatformEntry);
7139 DEBUG_PRINT_LOW(" Pmem Info = %p",pPMEMInfo);
7140 for (i=0; i < drv_ctx.op_buf.actualcount ; i++) {
7141 bufHdr->nSize = sizeof(OMX_BUFFERHEADERTYPE);
7142 bufHdr->nVersion.nVersion = OMX_SPEC_VERSION;
7143 // Set the values when we determine the right HxW param
7144 bufHdr->nAllocLen = bytes;
7145 bufHdr->nFilledLen = 0;
7146 bufHdr->pAppPrivate = appData;
7147 bufHdr->nOutputPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
7148 // Platform specific PMEM Information
7149 // Initialize the Platform Entry
7150 //DEBUG_PRINT_LOW("Initializing the Platform Entry for %d",i);
7151 pPlatformEntry->type = OMX_QCOM_PLATFORM_PRIVATE_PMEM;
7152 pPlatformEntry->entry = pPMEMInfo;
7153 // Initialize the Platform List
7154 pPlatformList->nEntries = 1;
7155 pPlatformList->entryList = pPlatformEntry;
7156 // Keep pBuffer NULL till vdec is opened
7157 bufHdr->pBuffer = NULL;
7158 bufHdr->nOffset = 0;
7159
7160 pPMEMInfo->offset = drv_ctx.op_buf.buffer_size*i;
7161 pPMEMInfo->pmem_fd = -1;
7162 bufHdr->pPlatformPrivate = pPlatformList;
7163
7164 int block_idx = i / block_buf_count;
7165 drv_ctx.ptr_outputbuffer[i].pmem_fd = pmem_fd[block_idx];
7166 m_pmem_info[i].pmem_fd = pmem_fd[block_idx];
7167 #ifdef USE_ION
7168 drv_ctx.op_buf_ion_info[i].ion_device_fd = op_buf_ion_info_temp[block_idx].ion_device_fd;
7169 drv_ctx.op_buf_ion_info[i].ion_alloc_data = op_buf_ion_info_temp[block_idx].ion_alloc_data;
7170 drv_ctx.op_buf_ion_info[i].fd_ion_data = op_buf_ion_info_temp[block_idx].fd_ion_data;
7171 #endif
7172 drv_ctx.op_buf_map_info[i].free_buffer = !(i % block_buf_count);
7173 drv_ctx.op_buf_map_info[i].base_address = pmem_baseaddress_temp[block_idx];
7174 drv_ctx.op_buf_map_info[i].map_size = alloc_size[block_idx];
7175 drv_ctx.op_buf_map_info[i].offset = drv_ctx.op_buf.buffer_size * (i % block_buf_count);
7176
7177 /*Create a mapping between buffers*/
7178 bufHdr->pOutputPortPrivate = &drv_ctx.ptr_respbuffer[i];
7179 drv_ctx.ptr_respbuffer[i].client_data = (void *)\
7180 &drv_ctx.ptr_outputbuffer[i];
7181 drv_ctx.ptr_outputbuffer[i].offset = drv_ctx.op_buf.buffer_size * (i % block_buf_count);
7182 drv_ctx.ptr_outputbuffer[i].bufferaddr =
7183 pmem_baseaddress_temp[block_idx] + (drv_ctx.op_buf.buffer_size * (i % block_buf_count));
7184 m_pmem_info[i].size = drv_ctx.ptr_outputbuffer[i].buffer_len;
7185 m_pmem_info[i].mapped_size = drv_ctx.ptr_outputbuffer[i].mmaped_size;
7186 m_pmem_info[i].buffer = drv_ctx.ptr_outputbuffer[i].bufferaddr;
7187
7188 DEBUG_PRINT_LOW("pmem_fd = %d offset = %u address = %p",
7189 pmem_fd[block_idx], (unsigned int)drv_ctx.ptr_outputbuffer[i].offset,
7190 drv_ctx.ptr_outputbuffer[i].bufferaddr);
7191 // Move the buffer and buffer header pointers
7192 bufHdr++;
7193 pPMEMInfo++;
7194 pPlatformEntry++;
7195 pPlatformList++;
7196 }
7197 } else {
7198 DEBUG_PRINT_ERROR("Output buf mem alloc failed[0x%p][0x%p]",\
7199 m_out_mem_ptr, pPtr);
7200 if (m_out_mem_ptr) {
7201 free(m_out_mem_ptr);
7202 m_out_mem_ptr = NULL;
7203 }
7204 if (pPtr) {
7205 free(pPtr);
7206 pPtr = NULL;
7207 }
7208 if (drv_ctx.ptr_outputbuffer) {
7209 free(drv_ctx.ptr_outputbuffer);
7210 drv_ctx.ptr_outputbuffer = NULL;
7211 }
7212 if (drv_ctx.ptr_respbuffer) {
7213 free(drv_ctx.ptr_respbuffer);
7214 drv_ctx.ptr_respbuffer = NULL;
7215 }
7216 #ifdef USE_ION
7217 if (drv_ctx.op_buf_ion_info) {
7218 DEBUG_PRINT_LOW("Free o/p ion context");
7219 free(drv_ctx.op_buf_ion_info);
7220 drv_ctx.op_buf_ion_info = NULL;
7221 }
7222 #endif
7223 free(drv_ctx.op_buf_map_info);
7224 drv_ctx.op_buf_map_info = NULL;
7225 eRet = OMX_ErrorInsufficientResources;
7226 }
7227 if (eRet == OMX_ErrorNone)
7228 eRet = allocate_extradata();
7229 }
7230
7231 for (i=0; i< drv_ctx.op_buf.actualcount; i++) {
7232 if (BITMASK_ABSENT(&m_out_bm_count,i)) {
7233 DEBUG_PRINT_LOW("Found a Free Output Buffer %d",i);
7234 break;
7235 }
7236 }
7237
7238 if (eRet == OMX_ErrorNone) {
7239 if (i < drv_ctx.op_buf.actualcount) {
7240 struct v4l2_buffer buf;
7241 struct v4l2_plane plane[VIDEO_MAX_PLANES];
7242 int rc;
7243 m_pmem_info[i].offset = drv_ctx.ptr_outputbuffer[i].offset;
7244
7245 drv_ctx.ptr_outputbuffer[i].buffer_len =
7246 drv_ctx.op_buf.buffer_size;
7247
7248 *bufferHdr = (m_out_mem_ptr + i );
7249 if (secure_mode) {
7250 #ifdef USE_ION
7251 drv_ctx.ptr_outputbuffer[i].bufferaddr =
7252 (OMX_U8 *)(intptr_t)drv_ctx.op_buf_ion_info[i].fd_ion_data.fd;
7253 #else
7254 drv_ctx.ptr_outputbuffer[i].bufferaddr = *bufferHdr;
7255 #endif
7256 }
7257 drv_ctx.ptr_outputbuffer[i].mmaped_size = drv_ctx.op_buf.buffer_size;
7258
7259 buf.index = i;
7260 buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
7261 buf.memory = V4L2_MEMORY_USERPTR;
7262 plane[0].length = drv_ctx.op_buf.buffer_size;
7263 plane[0].m.userptr = (unsigned long)drv_ctx.ptr_outputbuffer[i].bufferaddr -
7264 (unsigned long)drv_ctx.ptr_outputbuffer[i].offset;
7265 #ifdef USE_ION
7266 plane[0].reserved[0] = drv_ctx.op_buf_ion_info[i].fd_ion_data.fd;
7267 #endif
7268 plane[0].reserved[1] = drv_ctx.op_buf_map_info[i].offset;
7269 plane[0].data_offset = 0;
7270 extra_idx = EXTRADATA_IDX(drv_ctx.num_planes);
7271 if (extra_idx && (extra_idx < VIDEO_MAX_PLANES)) {
7272 plane[extra_idx].length = drv_ctx.extradata_info.buffer_size;
7273 plane[extra_idx].m.userptr = (long unsigned int) (drv_ctx.extradata_info.uaddr + i * drv_ctx.extradata_info.buffer_size);
7274 #ifdef USE_ION
7275 plane[extra_idx].reserved[0] = drv_ctx.extradata_info.ion.fd_ion_data.fd;
7276 #endif
7277 plane[extra_idx].reserved[1] = i * drv_ctx.extradata_info.buffer_size;
7278 plane[extra_idx].data_offset = 0;
7279 } else if (extra_idx >= VIDEO_MAX_PLANES) {
7280 DEBUG_PRINT_ERROR("Extradata index higher than allowed: %d", extra_idx);
7281 return OMX_ErrorBadParameter;
7282 }
7283 buf.m.planes = plane;
7284 buf.length = drv_ctx.num_planes;
7285 DEBUG_PRINT_LOW("Set the Output Buffer Idx: %d Addr: %p", i, drv_ctx.ptr_outputbuffer[i].bufferaddr);
7286 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_PREPARE_BUF, &buf);
7287 if (rc) {
7288 /*TODO: How to handle this case */
7289 return OMX_ErrorInsufficientResources;
7290 }
7291
7292 if (i == (drv_ctx.op_buf.actualcount -1 ) && !streaming[CAPTURE_PORT]) {
7293 enum v4l2_buf_type buf_type;
7294 buf_type=V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
7295 rc=ioctl(drv_ctx.video_driver_fd, VIDIOC_STREAMON,&buf_type);
7296 if (rc) {
7297 return OMX_ErrorInsufficientResources;
7298 } else {
7299 streaming[CAPTURE_PORT] = true;
7300 DEBUG_PRINT_LOW("STREAMON Successful");
7301 }
7302
7303 DEBUG_PRINT_HIGH("Enabling Turbo mode");
7304 request_perf_level(VIDC_TURBO);
7305 }
7306
7307 (*bufferHdr)->pBuffer = (OMX_U8*)drv_ctx.ptr_outputbuffer[i].bufferaddr;
7308 (*bufferHdr)->pAppPrivate = appData;
7309 BITMASK_SET(&m_out_bm_count,i);
7310 } else {
7311 DEBUG_PRINT_ERROR("All the Output Buffers have been Allocated ; Returning Insufficient");
7312 eRet = OMX_ErrorInsufficientResources;
7313 }
7314 }
7315
7316 return eRet;
7317 }
7318
7319
7320 // AllocateBuffer -- API Call
7321 /* ======================================================================
7322 FUNCTION
7323 omx_vdec::AllocateBuffer
7324
7325 DESCRIPTION
7326 Returns zero if all the buffers released..
7327
7328 PARAMETERS
7329 None.
7330
7331 RETURN VALUE
7332 true/false
7333
7334 ========================================================================== */
allocate_buffer(OMX_IN OMX_HANDLETYPE hComp,OMX_INOUT OMX_BUFFERHEADERTYPE ** bufferHdr,OMX_IN OMX_U32 port,OMX_IN OMX_PTR appData,OMX_IN OMX_U32 bytes)7335 OMX_ERRORTYPE omx_vdec::allocate_buffer(OMX_IN OMX_HANDLETYPE hComp,
7336 OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
7337 OMX_IN OMX_U32 port,
7338 OMX_IN OMX_PTR appData,
7339 OMX_IN OMX_U32 bytes)
7340 {
7341 unsigned i = 0;
7342 OMX_ERRORTYPE eRet = OMX_ErrorNone; // OMX return type
7343
7344 DEBUG_PRINT_LOW("Allocate buffer on port %d", (int)port);
7345 if (m_state == OMX_StateInvalid) {
7346 DEBUG_PRINT_ERROR("Allocate Buf in Invalid State");
7347 return OMX_ErrorInvalidState;
7348 }
7349
7350 if (port == OMX_CORE_INPUT_PORT_INDEX) {
7351 // If this is not the first allocation (i.e m_inp_mem_ptr is allocated),
7352 // ensure that use-buffer was never called.
7353 // Mix-and-match of useBuffer and allocateBuffer is not allowed
7354 if (m_inp_mem_ptr && input_use_buffer) {
7355 DEBUG_PRINT_ERROR("'Allocate' Input buffer called after 'Use' Input buffer !");
7356 return OMX_ErrorUndefined;
7357 }
7358 if (arbitrary_bytes) {
7359 eRet = allocate_input_heap_buffer (hComp,bufferHdr,port,appData,bytes);
7360 } else {
7361 eRet = allocate_input_buffer(hComp,bufferHdr,port,appData,bytes);
7362 }
7363 } else if (port == OMX_CORE_OUTPUT_PORT_INDEX) {
7364 eRet = client_buffers.allocate_buffers_color_convert(hComp,bufferHdr,port,
7365 appData,bytes);
7366 } else {
7367 DEBUG_PRINT_ERROR("Error: Invalid Port Index received %d",(int)port);
7368 eRet = OMX_ErrorBadPortIndex;
7369 }
7370 DEBUG_PRINT_LOW("Checking for Output Allocate buffer Done");
7371 if (eRet == OMX_ErrorNone) {
7372 if (allocate_done()) {
7373 if (BITMASK_PRESENT(&m_flags,OMX_COMPONENT_IDLE_PENDING)) {
7374 // Send the callback now
7375 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_IDLE_PENDING);
7376 post_event(OMX_CommandStateSet,OMX_StateIdle,
7377 OMX_COMPONENT_GENERATE_EVENT);
7378 }
7379 }
7380 if (port == OMX_CORE_INPUT_PORT_INDEX && m_inp_bPopulated) {
7381 if (BITMASK_PRESENT(&m_flags,OMX_COMPONENT_INPUT_ENABLE_PENDING)) {
7382 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_INPUT_ENABLE_PENDING);
7383 post_event(OMX_CommandPortEnable,
7384 OMX_CORE_INPUT_PORT_INDEX,
7385 OMX_COMPONENT_GENERATE_EVENT);
7386 }
7387 }
7388 if (port == OMX_CORE_OUTPUT_PORT_INDEX && m_out_bPopulated) {
7389 if (BITMASK_PRESENT(&m_flags,OMX_COMPONENT_OUTPUT_ENABLE_PENDING)) {
7390 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_OUTPUT_ENABLE_PENDING);
7391 post_event(OMX_CommandPortEnable,
7392 OMX_CORE_OUTPUT_PORT_INDEX,
7393 OMX_COMPONENT_GENERATE_EVENT);
7394 }
7395 }
7396 }
7397 DEBUG_PRINT_LOW("Allocate Buffer exit with ret Code %d",eRet);
7398 return eRet;
7399 }
7400
7401 // Free Buffer - API call
7402 /* ======================================================================
7403 FUNCTION
7404 omx_vdec::FreeBuffer
7405
7406 DESCRIPTION
7407
7408 PARAMETERS
7409 None.
7410
7411 RETURN VALUE
7412 true/false
7413
7414 ========================================================================== */
free_buffer(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_U32 port,OMX_IN OMX_BUFFERHEADERTYPE * buffer)7415 OMX_ERRORTYPE omx_vdec::free_buffer(OMX_IN OMX_HANDLETYPE hComp,
7416 OMX_IN OMX_U32 port,
7417 OMX_IN OMX_BUFFERHEADERTYPE* buffer)
7418 {
7419 OMX_ERRORTYPE eRet = OMX_ErrorNone;
7420 unsigned int nPortIndex;
7421 (void) hComp;
7422 DEBUG_PRINT_LOW("In for decoder free_buffer");
7423
7424 if (m_state == OMX_StateIdle &&
7425 (BITMASK_PRESENT(&m_flags ,OMX_COMPONENT_LOADING_PENDING))) {
7426 DEBUG_PRINT_LOW(" free buffer while Component in Loading pending");
7427 } else if ((m_inp_bEnabled == OMX_FALSE && port == OMX_CORE_INPUT_PORT_INDEX)||
7428 (m_out_bEnabled == OMX_FALSE && port == OMX_CORE_OUTPUT_PORT_INDEX)) {
7429 DEBUG_PRINT_LOW("Free Buffer while port %u disabled", (unsigned int)port);
7430 } else if ((port == OMX_CORE_INPUT_PORT_INDEX &&
7431 BITMASK_PRESENT(&m_flags, OMX_COMPONENT_INPUT_ENABLE_PENDING)) ||
7432 (port == OMX_CORE_OUTPUT_PORT_INDEX &&
7433 BITMASK_PRESENT(&m_flags, OMX_COMPONENT_OUTPUT_ENABLE_PENDING))) {
7434 DEBUG_PRINT_LOW("Free Buffer while port %u enable pending", (unsigned int)port);
7435 } else if (m_state == OMX_StateExecuting || m_state == OMX_StatePause) {
7436 DEBUG_PRINT_ERROR("Invalid state to free buffer,ports need to be disabled");
7437 post_event(OMX_EventError,
7438 OMX_ErrorPortUnpopulated,
7439 OMX_COMPONENT_GENERATE_EVENT);
7440
7441 return OMX_ErrorIncorrectStateOperation;
7442 } else if (m_state != OMX_StateInvalid) {
7443 DEBUG_PRINT_ERROR("Invalid state to free buffer,port lost Buffers");
7444 post_event(OMX_EventError,
7445 OMX_ErrorPortUnpopulated,
7446 OMX_COMPONENT_GENERATE_EVENT);
7447 }
7448
7449 if (port == OMX_CORE_INPUT_PORT_INDEX) {
7450 /*Check if arbitrary bytes*/
7451 if (!arbitrary_bytes && !input_use_buffer)
7452 nPortIndex = buffer - m_inp_mem_ptr;
7453 else
7454 nPortIndex = buffer - m_inp_heap_ptr;
7455
7456 DEBUG_PRINT_LOW("free_buffer on i/p port - Port idx %d", nPortIndex);
7457 if (nPortIndex < drv_ctx.ip_buf.actualcount &&
7458 BITMASK_PRESENT(&m_inp_bm_count, nPortIndex)) {
7459 // Clear the bit associated with it.
7460 BITMASK_CLEAR(&m_inp_bm_count,nPortIndex);
7461 BITMASK_CLEAR(&m_heap_inp_bm_count,nPortIndex);
7462 if (input_use_buffer == true) {
7463
7464 DEBUG_PRINT_LOW("Free pmem Buffer index %d",nPortIndex);
7465 if (m_phdr_pmem_ptr)
7466 free_input_buffer(m_phdr_pmem_ptr[nPortIndex]);
7467 } else {
7468 if (arbitrary_bytes) {
7469 if (m_phdr_pmem_ptr)
7470 free_input_buffer(nPortIndex,m_phdr_pmem_ptr[nPortIndex]);
7471 else
7472 free_input_buffer(nPortIndex,NULL);
7473 } else
7474 free_input_buffer(buffer);
7475 }
7476 m_inp_bPopulated = OMX_FALSE;
7477 if(release_input_done())
7478 release_buffers(this, VDEC_BUFFER_TYPE_INPUT);
7479 /*Free the Buffer Header*/
7480 if (release_input_done()) {
7481 DEBUG_PRINT_HIGH("ALL input buffers are freed/released");
7482 free_input_buffer_header();
7483 }
7484 } else {
7485 DEBUG_PRINT_ERROR("Error: free_buffer ,Port Index Invalid");
7486 eRet = OMX_ErrorBadPortIndex;
7487 }
7488
7489 if (BITMASK_PRESENT((&m_flags),OMX_COMPONENT_INPUT_DISABLE_PENDING)
7490 && release_input_done()) {
7491 DEBUG_PRINT_LOW("MOVING TO DISABLED STATE");
7492 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_INPUT_DISABLE_PENDING);
7493 post_event(OMX_CommandPortDisable,
7494 OMX_CORE_INPUT_PORT_INDEX,
7495 OMX_COMPONENT_GENERATE_EVENT);
7496 }
7497 } else if (port == OMX_CORE_OUTPUT_PORT_INDEX) {
7498 // check if the buffer is valid
7499 nPortIndex = buffer - client_buffers.get_il_buf_hdr();
7500 if (nPortIndex < drv_ctx.op_buf.actualcount &&
7501 BITMASK_PRESENT(&m_out_bm_count, nPortIndex)) {
7502 DEBUG_PRINT_LOW("free_buffer on o/p port - Port idx %d", nPortIndex);
7503 // Clear the bit associated with it.
7504 BITMASK_CLEAR(&m_out_bm_count,nPortIndex);
7505 m_out_bPopulated = OMX_FALSE;
7506 client_buffers.free_output_buffer (buffer);
7507
7508 if(release_output_done()) {
7509 release_buffers(this, VDEC_BUFFER_TYPE_OUTPUT);
7510 }
7511 if (release_output_done()) {
7512 free_output_buffer_header();
7513 }
7514 } else {
7515 DEBUG_PRINT_ERROR("Error: free_buffer , Port Index Invalid");
7516 eRet = OMX_ErrorBadPortIndex;
7517 }
7518 if (BITMASK_PRESENT((&m_flags),OMX_COMPONENT_OUTPUT_DISABLE_PENDING)
7519 && release_output_done()) {
7520 DEBUG_PRINT_LOW("FreeBuffer : If any Disable event pending,post it");
7521
7522 DEBUG_PRINT_LOW("MOVING TO DISABLED STATE");
7523 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_OUTPUT_DISABLE_PENDING);
7524 #ifdef _ANDROID_ICS_
7525 if (m_enable_android_native_buffers) {
7526 DEBUG_PRINT_LOW("FreeBuffer - outport disabled: reset native buffers");
7527 memset(&native_buffer, 0 ,(sizeof(struct nativebuffer) * MAX_NUM_INPUT_OUTPUT_BUFFERS));
7528 }
7529 #endif
7530
7531 post_event(OMX_CommandPortDisable,
7532 OMX_CORE_OUTPUT_PORT_INDEX,
7533 OMX_COMPONENT_GENERATE_EVENT);
7534 }
7535 } else if (port == OMX_CORE_OUTPUT_EXTRADATA_INDEX) {
7536 nPortIndex = buffer - m_client_output_extradata_mem_ptr;
7537 DEBUG_PRINT_LOW("free_buffer on extradata output port - Port idx %d", nPortIndex);
7538
7539 BITMASK_CLEAR(&m_out_extradata_bm_count,nPortIndex);
7540
7541 if (release_output_extradata_done()) {
7542 free_output_extradata_buffer_header();
7543 }
7544 } else {
7545 eRet = OMX_ErrorBadPortIndex;
7546 }
7547 if ((eRet == OMX_ErrorNone) &&
7548 (BITMASK_PRESENT(&m_flags ,OMX_COMPONENT_LOADING_PENDING))) {
7549 if (release_done()) {
7550 /*
7551 * Reset buffer requirements here to ensure setting buffer requirement
7552 * when component move to executing state from loaded state via idle.
7553 */
7554 drv_ctx.op_buf.buffer_size = 0;
7555 drv_ctx.op_buf.actualcount = 0;
7556
7557 // Send the callback now
7558 BITMASK_CLEAR((&m_flags),OMX_COMPONENT_LOADING_PENDING);
7559 post_event(OMX_CommandStateSet, OMX_StateLoaded,
7560 OMX_COMPONENT_GENERATE_EVENT);
7561 }
7562 }
7563 return eRet;
7564 }
7565
7566
7567 /* ======================================================================
7568 FUNCTION
7569 omx_vdec::EmptyThisBuffer
7570
7571 DESCRIPTION
7572 This routine is used to push the encoded video frames to
7573 the video decoder.
7574
7575 PARAMETERS
7576 None.
7577
7578 RETURN VALUE
7579 OMX Error None if everything went successful.
7580
7581 ========================================================================== */
empty_this_buffer(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_BUFFERHEADERTYPE * buffer)7582 OMX_ERRORTYPE omx_vdec::empty_this_buffer(OMX_IN OMX_HANDLETYPE hComp,
7583 OMX_IN OMX_BUFFERHEADERTYPE* buffer)
7584 {
7585 OMX_ERRORTYPE ret1 = OMX_ErrorNone;
7586 unsigned int nBufferIndex = drv_ctx.ip_buf.actualcount;
7587
7588 if (m_state != OMX_StateExecuting &&
7589 m_state != OMX_StatePause &&
7590 m_state != OMX_StateIdle) {
7591 DEBUG_PRINT_ERROR("Empty this buffer in Invalid State");
7592 return OMX_ErrorInvalidState;
7593 }
7594
7595 if (buffer == NULL) {
7596 DEBUG_PRINT_ERROR("ERROR:ETB Buffer is NULL");
7597 return OMX_ErrorBadParameter;
7598 }
7599
7600 if (!m_inp_bEnabled) {
7601 DEBUG_PRINT_ERROR("ERROR:ETB incorrect state operation, input port is disabled.");
7602 return OMX_ErrorIncorrectStateOperation;
7603 }
7604
7605 if (buffer->nInputPortIndex != OMX_CORE_INPUT_PORT_INDEX) {
7606 DEBUG_PRINT_ERROR("ERROR:ETB invalid port in header %u", (unsigned int)buffer->nInputPortIndex);
7607 return OMX_ErrorBadPortIndex;
7608 }
7609
7610 if (perf_flag) {
7611 if (!latency) {
7612 dec_time.stop();
7613 latency = dec_time.processing_time_us();
7614 dec_time.start();
7615 }
7616 }
7617
7618 if (arbitrary_bytes) {
7619 nBufferIndex = buffer - m_inp_heap_ptr;
7620 } else {
7621 if (input_use_buffer == true) {
7622 nBufferIndex = buffer - m_inp_heap_ptr;
7623 if (nBufferIndex >= drv_ctx.ip_buf.actualcount ) {
7624 DEBUG_PRINT_ERROR("ERROR: ETB nBufferIndex is invalid in use-buffer mode");
7625 return OMX_ErrorBadParameter;
7626 }
7627 m_inp_mem_ptr[nBufferIndex].nFilledLen = m_inp_heap_ptr[nBufferIndex].nFilledLen;
7628 m_inp_mem_ptr[nBufferIndex].nTimeStamp = m_inp_heap_ptr[nBufferIndex].nTimeStamp;
7629 m_inp_mem_ptr[nBufferIndex].nFlags = m_inp_heap_ptr[nBufferIndex].nFlags;
7630 buffer = &m_inp_mem_ptr[nBufferIndex];
7631 DEBUG_PRINT_LOW("Non-Arbitrary mode - buffer address is: malloc %p, pmem%p in Index %d, buffer %p of size %u",
7632 &m_inp_heap_ptr[nBufferIndex], &m_inp_mem_ptr[nBufferIndex],nBufferIndex, buffer, (unsigned int)buffer->nFilledLen);
7633 } else {
7634 nBufferIndex = buffer - m_inp_mem_ptr;
7635 }
7636 }
7637
7638 if (nBufferIndex >= drv_ctx.ip_buf.actualcount ) {
7639 DEBUG_PRINT_ERROR("ERROR:ETB nBufferIndex is invalid");
7640 return OMX_ErrorBadParameter;
7641 }
7642
7643 if (buffer->nFlags & OMX_BUFFERFLAG_CODECCONFIG) {
7644 codec_config_flag = true;
7645 DEBUG_PRINT_LOW("%s: codec_config buffer", __FUNCTION__);
7646 }
7647
7648 /* The client should not set this when codec is in arbitrary bytes mode */
7649 if (m_input_pass_buffer_fd) {
7650 buffer->pBuffer = (OMX_U8*)drv_ctx.ptr_inputbuffer[nBufferIndex].bufferaddr;
7651 }
7652
7653 DEBUG_PRINT_LOW("[ETB] BHdr(%p) pBuf(%p) nTS(%lld) nFL(%u)",
7654 buffer, buffer->pBuffer, buffer->nTimeStamp, (unsigned int)buffer->nFilledLen);
7655 if (arbitrary_bytes) {
7656 post_event ((unsigned long)hComp,(unsigned long)buffer,
7657 OMX_COMPONENT_GENERATE_ETB_ARBITRARY);
7658 } else {
7659 post_event ((unsigned long)hComp,(unsigned long)buffer,OMX_COMPONENT_GENERATE_ETB);
7660 }
7661 time_stamp_dts.insert_timestamp(buffer);
7662 return OMX_ErrorNone;
7663 }
7664
7665 /* ======================================================================
7666 FUNCTION
7667 omx_vdec::empty_this_buffer_proxy
7668
7669 DESCRIPTION
7670 This routine is used to push the encoded video frames to
7671 the video decoder.
7672
7673 PARAMETERS
7674 None.
7675
7676 RETURN VALUE
7677 OMX Error None if everything went successful.
7678
7679 ========================================================================== */
empty_this_buffer_proxy(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_BUFFERHEADERTYPE * buffer)7680 OMX_ERRORTYPE omx_vdec::empty_this_buffer_proxy(OMX_IN OMX_HANDLETYPE hComp,
7681 OMX_IN OMX_BUFFERHEADERTYPE* buffer)
7682 {
7683 VIDC_TRACE_NAME_HIGH("ETB");
7684 (void) hComp;
7685 int push_cnt = 0,i=0;
7686 unsigned nPortIndex = 0;
7687 OMX_ERRORTYPE ret = OMX_ErrorNone;
7688 struct vdec_input_frameinfo frameinfo;
7689 struct vdec_bufferpayload *temp_buffer;
7690 struct vdec_seqheader seq_header;
7691 bool port_setting_changed = true;
7692
7693 /*Should we generate a Aync error event*/
7694 if (buffer == NULL || buffer->pInputPortPrivate == NULL) {
7695 DEBUG_PRINT_ERROR("ERROR:empty_this_buffer_proxy is invalid");
7696 return OMX_ErrorBadParameter;
7697 }
7698
7699 nPortIndex = buffer-((OMX_BUFFERHEADERTYPE *)m_inp_mem_ptr);
7700
7701 if (nPortIndex >= drv_ctx.ip_buf.actualcount) {
7702 DEBUG_PRINT_ERROR("ERROR:empty_this_buffer_proxy invalid nPortIndex[%u]",
7703 nPortIndex);
7704 return OMX_ErrorBadParameter;
7705 }
7706
7707 pending_input_buffers++;
7708 VIDC_TRACE_INT_LOW("ETB-pending", pending_input_buffers);
7709
7710 /* return zero length and not an EOS buffer */
7711 if (!arbitrary_bytes && (buffer->nFilledLen == 0) &&
7712 ((buffer->nFlags & OMX_BUFFERFLAG_EOS) == 0)) {
7713 DEBUG_PRINT_HIGH("return zero legth buffer");
7714 post_event ((unsigned long)buffer,VDEC_S_SUCCESS,
7715 OMX_COMPONENT_GENERATE_EBD);
7716 return OMX_ErrorNone;
7717 }
7718
7719 if (input_flush_progress == true) {
7720 DEBUG_PRINT_LOW("Flush in progress return buffer ");
7721 post_event ((unsigned long)buffer,VDEC_S_SUCCESS,
7722 OMX_COMPONENT_GENERATE_EBD);
7723 return OMX_ErrorNone;
7724 }
7725
7726 auto_lock l(buf_lock);
7727 temp_buffer = (struct vdec_bufferpayload *)buffer->pInputPortPrivate;
7728
7729 if (!temp_buffer || (temp_buffer - drv_ctx.ptr_inputbuffer) > (int)drv_ctx.ip_buf.actualcount) {
7730 return OMX_ErrorBadParameter;
7731 }
7732 /* If its first frame, H264 codec and reject is true, then parse the nal
7733 and get the profile. Based on this, reject the clip playback */
7734 if (first_frame == 0 && codec_type_parse == CODEC_TYPE_H264 &&
7735 m_reject_avc_1080p_mp) {
7736 first_frame = 1;
7737 DEBUG_PRINT_ERROR("Parse nal to get the profile");
7738 h264_parser->parse_nal((OMX_U8*)buffer->pBuffer, buffer->nFilledLen,
7739 NALU_TYPE_SPS);
7740 m_profile = h264_parser->get_profile();
7741 ret = is_video_session_supported();
7742 if (ret) {
7743 post_event ((unsigned long)buffer,VDEC_S_SUCCESS,OMX_COMPONENT_GENERATE_EBD);
7744 post_event(OMX_EventError, OMX_ErrorInvalidState,OMX_COMPONENT_GENERATE_EVENT);
7745 /* Move the state to Invalid to avoid queueing of pending ETB to the driver */
7746 m_state = OMX_StateInvalid;
7747 return OMX_ErrorNone;
7748 }
7749 }
7750
7751 VIDC_TRACE_INT_LOW("ETB-TS", buffer->nTimeStamp / 1000);
7752 VIDC_TRACE_INT_LOW("ETB-size", buffer->nFilledLen);
7753 DEBUG_PRINT_LOW("ETBProxy: bufhdr = %p, bufhdr->pBuffer = %p", buffer, buffer->pBuffer);
7754 /*for use buffer we need to memcpy the data*/
7755 temp_buffer->buffer_len = buffer->nFilledLen;
7756
7757 if (input_use_buffer && temp_buffer->bufferaddr && !secure_mode) {
7758 if (buffer->nFilledLen <= temp_buffer->buffer_len) {
7759 if (arbitrary_bytes) {
7760 memcpy (temp_buffer->bufferaddr, (buffer->pBuffer + buffer->nOffset),buffer->nFilledLen);
7761 } else {
7762 memcpy (temp_buffer->bufferaddr, (m_inp_heap_ptr[nPortIndex].pBuffer + m_inp_heap_ptr[nPortIndex].nOffset),
7763 buffer->nFilledLen);
7764 }
7765 } else {
7766 return OMX_ErrorBadParameter;
7767 }
7768
7769 }
7770
7771 frameinfo.bufferaddr = temp_buffer->bufferaddr;
7772 frameinfo.client_data = (void *) buffer;
7773 frameinfo.datalen = temp_buffer->buffer_len;
7774 frameinfo.flags = 0;
7775 frameinfo.offset = buffer->nOffset;
7776 frameinfo.pmem_fd = temp_buffer->pmem_fd;
7777 frameinfo.pmem_offset = temp_buffer->offset;
7778 frameinfo.timestamp = buffer->nTimeStamp;
7779 if (drv_ctx.disable_dmx && m_desc_buffer_ptr && m_desc_buffer_ptr[nPortIndex].buf_addr) {
7780 DEBUG_PRINT_LOW("ETB: dmx enabled");
7781 if (m_demux_entries == 0) {
7782 extract_demux_addr_offsets(buffer);
7783 }
7784
7785 DEBUG_PRINT_LOW("ETB: handle_demux_data - entries=%u",(unsigned int)m_demux_entries);
7786 handle_demux_data(buffer);
7787 frameinfo.desc_addr = (OMX_U8 *)m_desc_buffer_ptr[nPortIndex].buf_addr;
7788 frameinfo.desc_size = m_desc_buffer_ptr[nPortIndex].desc_data_size;
7789 } else {
7790 frameinfo.desc_addr = NULL;
7791 frameinfo.desc_size = 0;
7792 }
7793 if (!arbitrary_bytes) {
7794 frameinfo.flags |= buffer->nFlags;
7795 }
7796
7797 #ifdef _ANDROID_
7798 if (m_debug_timestamp) {
7799 if (arbitrary_bytes) {
7800 DEBUG_PRINT_LOW("Inserting TIMESTAMP (%lld) into queue", buffer->nTimeStamp);
7801 m_timestamp_list.insert_ts(buffer->nTimeStamp);
7802 } else if (!arbitrary_bytes && !(buffer->nFlags & OMX_BUFFERFLAG_CODECCONFIG)) {
7803 DEBUG_PRINT_LOW("Inserting TIMESTAMP (%lld) into queue", buffer->nTimeStamp);
7804 m_timestamp_list.insert_ts(buffer->nTimeStamp);
7805 }
7806 }
7807 #endif
7808
7809 log_input_buffers((const char *)temp_buffer->bufferaddr, temp_buffer->buffer_len);
7810
7811 if (buffer->nFlags & QOMX_VIDEO_BUFFERFLAG_EOSEQ) {
7812 frameinfo.flags |= QOMX_VIDEO_BUFFERFLAG_EOSEQ;
7813 buffer->nFlags &= ~QOMX_VIDEO_BUFFERFLAG_EOSEQ;
7814 }
7815
7816 if (temp_buffer->buffer_len == 0 || (buffer->nFlags & OMX_BUFFERFLAG_EOS)) {
7817 DEBUG_PRINT_HIGH("Rxd i/p EOS, Notify Driver that EOS has been reached");
7818 frameinfo.flags |= VDEC_BUFFERFLAG_EOS;
7819 h264_scratch.nFilledLen = 0;
7820 nal_count = 0;
7821 look_ahead_nal = false;
7822 frame_count = 0;
7823 if (m_frame_parser.mutils)
7824 m_frame_parser.mutils->initialize_frame_checking_environment();
7825 m_frame_parser.flush();
7826 h264_last_au_ts = LLONG_MAX;
7827 h264_last_au_flags = 0;
7828 memset(m_demux_offsets, 0, ( sizeof(OMX_U32) * 8192) );
7829 m_demux_entries = 0;
7830 }
7831 struct v4l2_buffer buf;
7832 struct v4l2_plane plane;
7833 memset( (void *)&buf, 0, sizeof(buf));
7834 memset( (void *)&plane, 0, sizeof(plane));
7835 int rc;
7836 unsigned long print_count;
7837 if (temp_buffer->buffer_len == 0 || (buffer->nFlags & OMX_BUFFERFLAG_EOS)) {
7838 buf.flags = V4L2_QCOM_BUF_FLAG_EOS;
7839 DEBUG_PRINT_HIGH("INPUT EOS reached") ;
7840 }
7841 OMX_ERRORTYPE eRet = OMX_ErrorNone;
7842 buf.index = nPortIndex;
7843 buf.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
7844 buf.memory = V4L2_MEMORY_USERPTR;
7845 plane.bytesused = temp_buffer->buffer_len;
7846 plane.length = drv_ctx.ip_buf.buffer_size;
7847 plane.m.userptr = (unsigned long)temp_buffer->bufferaddr -
7848 (unsigned long)temp_buffer->offset;
7849 plane.reserved[0] = temp_buffer->pmem_fd;
7850 plane.reserved[1] = temp_buffer->offset;
7851 plane.data_offset = 0;
7852 buf.m.planes = &plane;
7853 buf.length = 1;
7854 if (frameinfo.timestamp >= LLONG_MAX) {
7855 buf.flags |= V4L2_QCOM_BUF_TIMESTAMP_INVALID;
7856 }
7857 //assumption is that timestamp is in milliseconds
7858 buf.timestamp.tv_sec = frameinfo.timestamp / 1000000;
7859 buf.timestamp.tv_usec = (frameinfo.timestamp % 1000000);
7860 buf.flags |= (buffer->nFlags & OMX_BUFFERFLAG_CODECCONFIG) ? V4L2_QCOM_BUF_FLAG_CODECCONFIG: 0;
7861 buf.flags |= (buffer->nFlags & OMX_BUFFERFLAG_DECODEONLY) ? V4L2_QCOM_BUF_FLAG_DECODEONLY: 0;
7862
7863 if (buffer->nFlags & OMX_BUFFERFLAG_CODECCONFIG) {
7864 DEBUG_PRINT_LOW("Increment codec_config buffer counter");
7865 android_atomic_inc(&m_queued_codec_config_count);
7866 }
7867
7868 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_QBUF, &buf);
7869 if (rc) {
7870 DEBUG_PRINT_ERROR("Failed to qbuf Input buffer to driver, send ETB back to client");
7871 m_cb.EmptyBufferDone(hComp, m_app_data, buffer);
7872 return OMX_ErrorHardware;
7873 }
7874
7875 if (codec_config_flag && !(buffer->nFlags & OMX_BUFFERFLAG_CODECCONFIG)) {
7876 codec_config_flag = false;
7877 }
7878 if (!streaming[OUTPUT_PORT]) {
7879 enum v4l2_buf_type buf_type;
7880 int ret,r;
7881
7882 buf_type=V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
7883 DEBUG_PRINT_LOW("send_command_proxy(): Idle-->Executing");
7884 ret=ioctl(drv_ctx.video_driver_fd, VIDIOC_STREAMON,&buf_type);
7885 if (!ret) {
7886 DEBUG_PRINT_HIGH("Streamon on OUTPUT Plane was successful");
7887 streaming[OUTPUT_PORT] = true;
7888 } else if (errno == EBUSY) {
7889 DEBUG_PRINT_ERROR("Failed to call stream on OUTPUT due to HW_OVERLOAD");
7890 post_event ((unsigned long)buffer, VDEC_S_SUCCESS,
7891 OMX_COMPONENT_GENERATE_EBD);
7892 return OMX_ErrorInsufficientResources;
7893 } else {
7894 DEBUG_PRINT_ERROR("Failed to call streamon on OUTPUT");
7895 DEBUG_PRINT_LOW("If Stream on failed no buffer should be queued");
7896 post_event ((unsigned long)buffer, VDEC_S_SUCCESS,
7897 OMX_COMPONENT_GENERATE_EBD);
7898 return OMX_ErrorBadParameter;
7899 }
7900 }
7901 DEBUG_PRINT_LOW("[ETBP] pBuf(%p) nTS(%lld) Sz(%u)",
7902 frameinfo.bufferaddr, (long long)frameinfo.timestamp,
7903 (unsigned int)frameinfo.datalen);
7904
7905 return ret;
7906 }
7907
7908 /* ======================================================================
7909 FUNCTION
7910 omx_vdec::FillThisBuffer
7911
7912 DESCRIPTION
7913 IL client uses this method to release the frame buffer
7914 after displaying them.
7915
7916 PARAMETERS
7917 None.
7918
7919 RETURN VALUE
7920 true/false
7921
7922 ========================================================================== */
fill_this_buffer(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_BUFFERHEADERTYPE * buffer)7923 OMX_ERRORTYPE omx_vdec::fill_this_buffer(OMX_IN OMX_HANDLETYPE hComp,
7924 OMX_IN OMX_BUFFERHEADERTYPE* buffer)
7925 {
7926 if (m_state != OMX_StateExecuting &&
7927 m_state != OMX_StatePause &&
7928 m_state != OMX_StateIdle) {
7929 DEBUG_PRINT_ERROR("FTB in Invalid State");
7930 return OMX_ErrorInvalidState;
7931 }
7932
7933 if (buffer->nOutputPortIndex != OMX_CORE_OUTPUT_PORT_INDEX) {
7934 DEBUG_PRINT_ERROR("ERROR:FTB invalid port in header %u", (unsigned int)buffer->nOutputPortIndex);
7935 return OMX_ErrorBadPortIndex;
7936 }
7937
7938 if (!m_out_bEnabled) {
7939 DEBUG_PRINT_ERROR("ERROR:FTB incorrect state operation, output port is disabled.");
7940 return OMX_ErrorIncorrectStateOperation;
7941 }
7942
7943 unsigned nPortIndex = 0;
7944 if (dynamic_buf_mode) {
7945 private_handle_t *handle = NULL;
7946 struct VideoDecoderOutputMetaData *meta;
7947 unsigned int nPortIndex = 0;
7948
7949 if (!buffer || !buffer->pBuffer) {
7950 DEBUG_PRINT_ERROR("%s: invalid params: %p", __FUNCTION__, buffer);
7951 return OMX_ErrorBadParameter;
7952 }
7953
7954 //get the buffer type and fd info
7955 meta = (struct VideoDecoderOutputMetaData *)buffer->pBuffer;
7956 handle = (private_handle_t *)meta->pHandle;
7957 DEBUG_PRINT_LOW("FTB: metabuf: %p buftype: %d bufhndl: %p ", meta, meta->eType, meta->pHandle);
7958
7959 if (!handle) {
7960 DEBUG_PRINT_ERROR("FTB: Error: IL client passed an invalid buf handle - %p", handle);
7961 return OMX_ErrorBadParameter;
7962 }
7963 //Fill outputbuffer with buffer details, this will be sent to f/w during VIDIOC_QBUF
7964 nPortIndex = buffer-((OMX_BUFFERHEADERTYPE *)client_buffers.get_il_buf_hdr());
7965 if (nPortIndex < drv_ctx.op_buf.actualcount &&
7966 nPortIndex < MAX_NUM_INPUT_OUTPUT_BUFFERS) {
7967 drv_ctx.ptr_outputbuffer[nPortIndex].pmem_fd = handle->fd;
7968 drv_ctx.ptr_outputbuffer[nPortIndex].bufferaddr = (OMX_U8*) buffer;
7969
7970 //Store private handle from GraphicBuffer
7971 native_buffer[nPortIndex].privatehandle = handle;
7972 native_buffer[nPortIndex].nativehandle = handle;
7973 } else {
7974 DEBUG_PRINT_ERROR("[FTB]Invalid native_buffer index: %d", nPortIndex);
7975 return OMX_ErrorBadParameter;
7976 }
7977
7978 //buffer->nAllocLen will be sizeof(struct VideoDecoderOutputMetaData). Overwrite
7979 //this with a more sane size so that we don't compensate in rest of code
7980 //We'll restore this size later on, so that it's transparent to client
7981 buffer->nFilledLen = 0;
7982 buffer->nAllocLen = handle->size;
7983
7984 if (handle->flags & private_handle_t::PRIV_FLAGS_DISP_CONSUMER) {
7985 m_is_display_session = true;
7986 } else {
7987 m_is_display_session = false;
7988 }
7989 DEBUG_PRINT_LOW("%s: m_is_display_session = %d", __func__, m_is_display_session);
7990
7991 drv_ctx.op_buf.buffer_size = handle->size;
7992 }
7993
7994 nPortIndex = buffer - client_buffers.get_il_buf_hdr();
7995 if (buffer == NULL ||
7996 (nPortIndex >= drv_ctx.op_buf.actualcount)) {
7997 DEBUG_PRINT_ERROR("FTB: ERROR: invalid buffer index, nPortIndex %u bufCount %u",
7998 nPortIndex, drv_ctx.op_buf.actualcount);
7999 return OMX_ErrorBadParameter;
8000 }
8001
8002 DEBUG_PRINT_LOW("[FTB] bufhdr = %p, bufhdr->pBuffer = %p", buffer, buffer->pBuffer);
8003 post_event((unsigned long) hComp, (unsigned long)buffer, m_fill_output_msg);
8004 return OMX_ErrorNone;
8005 }
8006 /* ======================================================================
8007 FUNCTION
8008 omx_vdec::fill_this_buffer_proxy
8009
8010 DESCRIPTION
8011 IL client uses this method to release the frame buffer
8012 after displaying them.
8013
8014 PARAMETERS
8015 None.
8016
8017 RETURN VALUE
8018 true/false
8019
8020 ========================================================================== */
fill_this_buffer_proxy(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_BUFFERHEADERTYPE * bufferAdd)8021 OMX_ERRORTYPE omx_vdec::fill_this_buffer_proxy(
8022 OMX_IN OMX_HANDLETYPE hComp,
8023 OMX_IN OMX_BUFFERHEADERTYPE* bufferAdd)
8024 {
8025 VIDC_TRACE_NAME_HIGH("FTB");
8026 OMX_ERRORTYPE nRet = OMX_ErrorNone;
8027 OMX_BUFFERHEADERTYPE *buffer = bufferAdd;
8028 unsigned nPortIndex = 0;
8029 struct vdec_fillbuffer_cmd fillbuffer;
8030 struct vdec_bufferpayload *ptr_outputbuffer = NULL;
8031 struct vdec_output_frameinfo *ptr_respbuffer = NULL;
8032
8033 nPortIndex = buffer-((OMX_BUFFERHEADERTYPE *)client_buffers.get_il_buf_hdr());
8034
8035 if (bufferAdd == NULL || nPortIndex >= drv_ctx.op_buf.actualcount) {
8036 DEBUG_PRINT_ERROR("FTBProxy: ERROR: invalid buffer index, nPortIndex %u bufCount %u",
8037 nPortIndex, drv_ctx.op_buf.actualcount);
8038 return OMX_ErrorBadParameter;
8039 }
8040
8041 DEBUG_PRINT_LOW("FTBProxy: bufhdr = %p, bufhdr->pBuffer = %p",
8042 bufferAdd, bufferAdd->pBuffer);
8043 /*Return back the output buffer to client*/
8044 if (m_out_bEnabled != OMX_TRUE || output_flush_progress == true || in_reconfig) {
8045 DEBUG_PRINT_LOW("Output Buffers return flush/disable condition");
8046 buffer->nFilledLen = 0;
8047 m_cb.FillBufferDone (hComp,m_app_data,buffer);
8048 return OMX_ErrorNone;
8049 }
8050
8051 if (dynamic_buf_mode) {
8052 drv_ctx.ptr_outputbuffer[nPortIndex].offset = 0;
8053 drv_ctx.ptr_outputbuffer[nPortIndex].buffer_len = buffer->nAllocLen;
8054 buf_ref_add(nPortIndex);
8055 drv_ctx.ptr_outputbuffer[nPortIndex].mmaped_size = buffer->nAllocLen;
8056 }
8057
8058 pending_output_buffers++;
8059 VIDC_TRACE_INT_LOW("FTB-pending", pending_output_buffers);
8060 buffer = client_buffers.get_dr_buf_hdr(bufferAdd);
8061 if (!buffer) {
8062 DEBUG_PRINT_ERROR("err: client_buffer ptr invalid");
8063 return OMX_ErrorBadParameter;
8064 }
8065 ptr_respbuffer = (struct vdec_output_frameinfo*)buffer->pOutputPortPrivate;
8066 if (ptr_respbuffer) {
8067 ptr_outputbuffer = (struct vdec_bufferpayload*)ptr_respbuffer->client_data;
8068 }
8069
8070 if (ptr_respbuffer == NULL || ptr_outputbuffer == NULL) {
8071 DEBUG_PRINT_ERROR("resp buffer or outputbuffer is NULL");
8072 buffer->nFilledLen = 0;
8073 m_cb.FillBufferDone (hComp,m_app_data,buffer);
8074 pending_output_buffers--;
8075 VIDC_TRACE_INT_LOW("FTB-pending", pending_output_buffers);
8076 return OMX_ErrorBadParameter;
8077 }
8078
8079 int rc = 0;
8080 struct v4l2_buffer buf;
8081 struct v4l2_plane plane[VIDEO_MAX_PLANES];
8082 memset( (void *)&buf, 0, sizeof(buf));
8083 memset( (void *)plane, 0, (sizeof(struct v4l2_plane)*VIDEO_MAX_PLANES));
8084 unsigned int extra_idx = 0;
8085
8086 buf.index = nPortIndex;
8087 buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
8088 buf.memory = V4L2_MEMORY_USERPTR;
8089 plane[0].bytesused = buffer->nFilledLen;
8090 plane[0].length = buffer->nAllocLen;
8091 plane[0].m.userptr =
8092 (unsigned long)drv_ctx.ptr_outputbuffer[nPortIndex].bufferaddr -
8093 (unsigned long)drv_ctx.ptr_outputbuffer[nPortIndex].offset;
8094 plane[0].reserved[0] = drv_ctx.ptr_outputbuffer[nPortIndex].pmem_fd;
8095 plane[0].reserved[1] = drv_ctx.ptr_outputbuffer[nPortIndex].offset;
8096 plane[0].data_offset = 0;
8097 extra_idx = EXTRADATA_IDX(drv_ctx.num_planes);
8098 if (extra_idx && (extra_idx < VIDEO_MAX_PLANES)) {
8099 plane[extra_idx].bytesused = 0;
8100 plane[extra_idx].length = drv_ctx.extradata_info.buffer_size;
8101 plane[extra_idx].m.userptr = (long unsigned int) (drv_ctx.extradata_info.uaddr + nPortIndex * drv_ctx.extradata_info.buffer_size);
8102 #ifdef USE_ION
8103 plane[extra_idx].reserved[0] = drv_ctx.extradata_info.ion.fd_ion_data.fd;
8104 #endif
8105 plane[extra_idx].reserved[1] = nPortIndex * drv_ctx.extradata_info.buffer_size;
8106 plane[extra_idx].data_offset = 0;
8107 } else if (extra_idx >= VIDEO_MAX_PLANES) {
8108 DEBUG_PRINT_ERROR("Extradata index higher than expected: %u", extra_idx);
8109 return OMX_ErrorBadParameter;
8110 }
8111 buf.m.planes = plane;
8112 buf.length = drv_ctx.num_planes;
8113 DEBUG_PRINT_LOW("SENDING FTB TO F/W - fd[0] = %d fd[1] = %d offset[1] = %d in_flush = %d",
8114 plane[0].reserved[0],plane[extra_idx].reserved[0], plane[extra_idx].reserved[1], output_flush_progress);
8115
8116 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_QBUF, &buf);
8117 if (rc) {
8118 buffer->nFilledLen = 0;
8119 DEBUG_PRINT_ERROR("Failed to qbuf to driver, error %s", strerror(errno));
8120 m_cb.FillBufferDone(hComp, m_app_data, buffer);
8121 return OMX_ErrorHardware;
8122 }
8123
8124 return OMX_ErrorNone;
8125 }
8126
8127 /* ======================================================================
8128 FUNCTION
8129 omx_vdec::SetCallbacks
8130
8131 DESCRIPTION
8132 Set the callbacks.
8133
8134 PARAMETERS
8135 None.
8136
8137 RETURN VALUE
8138 OMX Error None if everything successful.
8139
8140 ========================================================================== */
set_callbacks(OMX_IN OMX_HANDLETYPE hComp,OMX_IN OMX_CALLBACKTYPE * callbacks,OMX_IN OMX_PTR appData)8141 OMX_ERRORTYPE omx_vdec::set_callbacks(OMX_IN OMX_HANDLETYPE hComp,
8142 OMX_IN OMX_CALLBACKTYPE* callbacks,
8143 OMX_IN OMX_PTR appData)
8144 {
8145 (void) hComp;
8146 m_cb = *callbacks;
8147 DEBUG_PRINT_LOW("Callbacks Set %p %p %p",m_cb.EmptyBufferDone,\
8148 m_cb.EventHandler,m_cb.FillBufferDone);
8149 m_app_data = appData;
8150 return OMX_ErrorNotImplemented;
8151 }
8152
8153 /* ======================================================================
8154 FUNCTION
8155 omx_vdec::ComponentDeInit
8156
8157 DESCRIPTION
8158 Destroys the component and release memory allocated to the heap.
8159
8160 PARAMETERS
8161 <TBD>.
8162
8163 RETURN VALUE
8164 OMX Error None if everything successful.
8165
8166 ========================================================================== */
component_deinit(OMX_IN OMX_HANDLETYPE hComp)8167 OMX_ERRORTYPE omx_vdec::component_deinit(OMX_IN OMX_HANDLETYPE hComp)
8168 {
8169 (void) hComp;
8170
8171 unsigned i = 0;
8172 if (OMX_StateLoaded != m_state) {
8173 DEBUG_PRINT_ERROR("WARNING:Rxd DeInit,OMX not in LOADED state %d",\
8174 m_state);
8175 DEBUG_PRINT_ERROR("Playback Ended - FAILED");
8176 } else {
8177 DEBUG_PRINT_HIGH("Playback Ended - PASSED");
8178 }
8179
8180 /*Check if the output buffers have to be cleaned up*/
8181 if (m_out_mem_ptr) {
8182 DEBUG_PRINT_LOW("Freeing the Output Memory");
8183 for (i = 0; i < drv_ctx.op_buf.actualcount; i++ ) {
8184 if (BITMASK_PRESENT(&m_out_bm_count, i)) {
8185 BITMASK_CLEAR(&m_out_bm_count, i);
8186 client_buffers.free_output_buffer (&m_out_mem_ptr[i]);
8187 }
8188
8189 if (release_output_done()) {
8190 break;
8191 }
8192 }
8193 #ifdef _ANDROID_ICS_
8194 memset(&native_buffer, 0, (sizeof(nativebuffer) * MAX_NUM_INPUT_OUTPUT_BUFFERS));
8195 #endif
8196 }
8197
8198 /*Check if the input buffers have to be cleaned up*/
8199 if (m_inp_mem_ptr || m_inp_heap_ptr) {
8200 DEBUG_PRINT_LOW("Freeing the Input Memory");
8201 for (i = 0; i<drv_ctx.ip_buf.actualcount; i++ ) {
8202
8203 if (BITMASK_PRESENT(&m_inp_bm_count, i)) {
8204 BITMASK_CLEAR(&m_inp_bm_count, i);
8205 if (m_inp_mem_ptr)
8206 free_input_buffer (i,&m_inp_mem_ptr[i]);
8207 else
8208 free_input_buffer (i,NULL);
8209 }
8210
8211 if (release_input_done()) {
8212 break;
8213 }
8214 }
8215 }
8216 free_input_buffer_header();
8217 free_output_buffer_header();
8218 if (h264_scratch.pBuffer) {
8219 free(h264_scratch.pBuffer);
8220 h264_scratch.pBuffer = NULL;
8221 }
8222
8223 if (h264_parser) {
8224 delete h264_parser;
8225 h264_parser = NULL;
8226 }
8227
8228 if (m_frame_parser.mutils) {
8229 DEBUG_PRINT_LOW("Free utils parser");
8230 delete (m_frame_parser.mutils);
8231 m_frame_parser.mutils = NULL;
8232 }
8233
8234 if (m_platform_list) {
8235 free(m_platform_list);
8236 m_platform_list = NULL;
8237 }
8238 if (m_vendor_config.pData) {
8239 free(m_vendor_config.pData);
8240 m_vendor_config.pData = NULL;
8241 }
8242
8243 // Reset counters in mesg queues
8244 m_ftb_q.m_size=0;
8245 m_cmd_q.m_size=0;
8246 m_etb_q.m_size=0;
8247 m_ftb_q.m_read = m_ftb_q.m_write =0;
8248 m_cmd_q.m_read = m_cmd_q.m_write =0;
8249 m_etb_q.m_read = m_etb_q.m_write =0;
8250 #ifdef _ANDROID_
8251 if (m_debug_timestamp) {
8252 m_timestamp_list.reset_ts_list();
8253 }
8254 #endif
8255
8256 DEBUG_PRINT_LOW("Calling VDEC_IOCTL_STOP_NEXT_MSG");
8257 //(void)ioctl(drv_ctx.video_driver_fd, VDEC_IOCTL_STOP_NEXT_MSG,
8258 // NULL);
8259 DEBUG_PRINT_HIGH("Close the driver instance");
8260
8261 if (m_debug.infile) {
8262 fclose(m_debug.infile);
8263 m_debug.infile = NULL;
8264 }
8265 if (m_debug.outfile) {
8266 fclose(m_debug.outfile);
8267 m_debug.outfile = NULL;
8268 }
8269 if (m_debug.out_ymeta_file) {
8270 fclose(m_debug.out_ymeta_file);
8271 m_debug.out_ymeta_file = NULL;
8272 }
8273 if (m_debug.out_uvmeta_file) {
8274 fclose(m_debug.out_uvmeta_file);
8275 m_debug.out_uvmeta_file = NULL;
8276 }
8277 #ifdef OUTPUT_EXTRADATA_LOG
8278 if (outputExtradataFile)
8279 fclose (outputExtradataFile);
8280 #endif
8281 DEBUG_PRINT_INFO("omx_vdec::component_deinit() complete");
8282 return OMX_ErrorNone;
8283 }
8284
8285 /* ======================================================================
8286 FUNCTION
8287 omx_vdec::UseEGLImage
8288
8289 DESCRIPTION
8290 OMX Use EGL Image method implementation <TBD>.
8291
8292 PARAMETERS
8293 <TBD>.
8294
8295 RETURN VALUE
8296 Not Implemented error.
8297
8298 ========================================================================== */
use_EGL_image(OMX_IN OMX_HANDLETYPE hComp,OMX_INOUT OMX_BUFFERHEADERTYPE ** bufferHdr,OMX_IN OMX_U32 port,OMX_IN OMX_PTR appData,OMX_IN void * eglImage)8299 OMX_ERRORTYPE omx_vdec::use_EGL_image(OMX_IN OMX_HANDLETYPE hComp,
8300 OMX_INOUT OMX_BUFFERHEADERTYPE** bufferHdr,
8301 OMX_IN OMX_U32 port,
8302 OMX_IN OMX_PTR appData,
8303 OMX_IN void* eglImage)
8304 {
8305 (void) appData;
8306 OMX_QCOM_PLATFORM_PRIVATE_LIST pmem_list;
8307 OMX_QCOM_PLATFORM_PRIVATE_ENTRY pmem_entry;
8308 OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO pmem_info;
8309
8310 #ifdef USE_EGL_IMAGE_GPU
8311 PFNEGLQUERYIMAGEQUALCOMMPROC egl_queryfunc;
8312 EGLint fd = -1, offset = 0,pmemPtr = 0;
8313 #else
8314 int fd = -1, offset = 0;
8315 #endif
8316 DEBUG_PRINT_HIGH("use EGL image support for decoder");
8317 if (!bufferHdr || !eglImage|| port != OMX_CORE_OUTPUT_PORT_INDEX) {
8318 DEBUG_PRINT_ERROR("Invalid EGL image");
8319 }
8320 #ifdef USE_EGL_IMAGE_GPU
8321 if (m_display_id == NULL) {
8322 DEBUG_PRINT_ERROR("Display ID is not set by IL client");
8323 return OMX_ErrorInsufficientResources;
8324 }
8325 egl_queryfunc = (PFNEGLQUERYIMAGEQUALCOMMPROC)
8326 eglGetProcAddress("eglQueryImageKHR");
8327 egl_queryfunc(m_display_id, eglImage, EGL_BUFFER_HANDLE, &fd);
8328 egl_queryfunc(m_display_id, eglImage, EGL_BUFFER_OFFSET, &offset);
8329 egl_queryfunc(m_display_id, eglImage, EGL_BITMAP_POINTER_KHR, &pmemPtr);
8330 #else //with OMX test app
8331 struct temp_egl {
8332 int pmem_fd;
8333 int offset;
8334 };
8335 struct temp_egl *temp_egl_id = NULL;
8336 void * pmemPtr = (void *) eglImage;
8337 temp_egl_id = (struct temp_egl *)eglImage;
8338 if (temp_egl_id != NULL) {
8339 fd = temp_egl_id->pmem_fd;
8340 offset = temp_egl_id->offset;
8341 }
8342 #endif
8343 if (fd < 0) {
8344 DEBUG_PRINT_ERROR("Improper pmem fd by EGL client %d",fd);
8345 return OMX_ErrorInsufficientResources;
8346 }
8347 pmem_info.pmem_fd = (OMX_U32) fd;
8348 pmem_info.offset = (OMX_U32) offset;
8349 pmem_entry.entry = (void *) &pmem_info;
8350 pmem_entry.type = OMX_QCOM_PLATFORM_PRIVATE_PMEM;
8351 pmem_list.entryList = &pmem_entry;
8352 pmem_list.nEntries = 1;
8353 ouput_egl_buffers = true;
8354 if (OMX_ErrorNone != use_buffer(hComp,bufferHdr, port,
8355 (void *)&pmem_list, drv_ctx.op_buf.buffer_size,
8356 (OMX_U8 *)pmemPtr)) {
8357 DEBUG_PRINT_ERROR("use buffer call failed for egl image");
8358 return OMX_ErrorInsufficientResources;
8359 }
8360 return OMX_ErrorNone;
8361 }
8362
8363 /* ======================================================================
8364 FUNCTION
8365 omx_vdec::ComponentRoleEnum
8366
8367 DESCRIPTION
8368 OMX Component Role Enum method implementation.
8369
8370 PARAMETERS
8371 <TBD>.
8372
8373 RETURN VALUE
8374 OMX Error None if everything is successful.
8375 ========================================================================== */
component_role_enum(OMX_IN OMX_HANDLETYPE hComp,OMX_OUT OMX_U8 * role,OMX_IN OMX_U32 index)8376 OMX_ERRORTYPE omx_vdec::component_role_enum(OMX_IN OMX_HANDLETYPE hComp,
8377 OMX_OUT OMX_U8* role,
8378 OMX_IN OMX_U32 index)
8379 {
8380 (void) hComp;
8381 OMX_ERRORTYPE eRet = OMX_ErrorNone;
8382
8383 if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg4",OMX_MAX_STRINGNAME_SIZE)) {
8384 if ((0 == index) && role) {
8385 strlcpy((char *)role, "video_decoder.mpeg4",OMX_MAX_STRINGNAME_SIZE);
8386 DEBUG_PRINT_LOW("component_role_enum: role %s",role);
8387 } else {
8388 eRet = OMX_ErrorNoMore;
8389 }
8390 }
8391 if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mpeg2",OMX_MAX_STRINGNAME_SIZE)) {
8392 if ((0 == index) && role) {
8393 strlcpy((char *)role, "video_decoder.mpeg2",OMX_MAX_STRINGNAME_SIZE);
8394 DEBUG_PRINT_LOW("component_role_enum: role %s",role);
8395 } else {
8396 eRet = OMX_ErrorNoMore;
8397 }
8398 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.h263",OMX_MAX_STRINGNAME_SIZE)) {
8399 if ((0 == index) && role) {
8400 strlcpy((char *)role, "video_decoder.h263",OMX_MAX_STRINGNAME_SIZE);
8401 DEBUG_PRINT_LOW("component_role_enum: role %s",role);
8402 } else {
8403 DEBUG_PRINT_LOW("No more roles");
8404 eRet = OMX_ErrorNoMore;
8405 }
8406 }
8407
8408 else if ((!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx",OMX_MAX_STRINGNAME_SIZE)) ||
8409 (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.divx311",OMX_MAX_STRINGNAME_SIZE))) {
8410 if ((0 == index) && role) {
8411 strlcpy((char *)role, "video_decoder.divx",OMX_MAX_STRINGNAME_SIZE);
8412 DEBUG_PRINT_LOW("component_role_enum: role %s",role);
8413 } else {
8414 DEBUG_PRINT_LOW("No more roles");
8415 eRet = OMX_ErrorNoMore;
8416 }
8417 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.avc",OMX_MAX_STRINGNAME_SIZE)) {
8418 if ((0 == index) && role) {
8419 strlcpy((char *)role, "video_decoder.avc",OMX_MAX_STRINGNAME_SIZE);
8420 DEBUG_PRINT_LOW("component_role_enum: role %s",role);
8421 } else {
8422 DEBUG_PRINT_LOW("No more roles");
8423 eRet = OMX_ErrorNoMore;
8424 }
8425 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.mvc", OMX_MAX_STRINGNAME_SIZE)) {
8426 if ((0 == index) && role) {
8427 strlcpy((char *)role, "video_decoder.mvc", OMX_MAX_STRINGNAME_SIZE);
8428 DEBUG_PRINT_LOW("component_role_enum: role %s",role);
8429 } else {
8430 DEBUG_PRINT_LOW("No more roles");
8431 eRet = OMX_ErrorNoMore;
8432 }
8433 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.hevc", OMX_MAX_STRINGNAME_SIZE)) {
8434 if ((0 == index) && role) {
8435 strlcpy((char *)role, "video_decoder.hevc", OMX_MAX_STRINGNAME_SIZE);
8436 DEBUG_PRINT_LOW("component_role_enum: role %s", role);
8437 } else {
8438 DEBUG_PRINT_LOW("No more roles");
8439 eRet = OMX_ErrorNoMore;
8440 }
8441 } else if ( (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vc1",OMX_MAX_STRINGNAME_SIZE)) ||
8442 (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.wmv",OMX_MAX_STRINGNAME_SIZE))
8443 ) {
8444 if ((0 == index) && role) {
8445 strlcpy((char *)role, "video_decoder.vc1",OMX_MAX_STRINGNAME_SIZE);
8446 DEBUG_PRINT_LOW("component_role_enum: role %s",role);
8447 } else {
8448 DEBUG_PRINT_LOW("No more roles");
8449 eRet = OMX_ErrorNoMore;
8450 }
8451 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp8",OMX_MAX_STRINGNAME_SIZE)) {
8452 if ((0 == index) && role) {
8453 strlcpy((char *)role, "video_decoder.vp8",OMX_MAX_STRINGNAME_SIZE);
8454 DEBUG_PRINT_LOW("component_role_enum: role %s",role);
8455 } else {
8456 DEBUG_PRINT_LOW("No more roles");
8457 eRet = OMX_ErrorNoMore;
8458 }
8459 } else if (!strncmp(drv_ctx.kind, "OMX.qcom.video.decoder.vp9",OMX_MAX_STRINGNAME_SIZE)) {
8460 if ((0 == index) && role) {
8461 strlcpy((char *)role, "video_decoder.vp9",OMX_MAX_STRINGNAME_SIZE);
8462 DEBUG_PRINT_LOW("component_role_enum: role %s",role);
8463 } else {
8464 DEBUG_PRINT_LOW("No more roles");
8465 eRet = OMX_ErrorNoMore;
8466 }
8467 } else {
8468 DEBUG_PRINT_ERROR("ERROR:Querying Role on Unknown Component");
8469 eRet = OMX_ErrorInvalidComponentName;
8470 }
8471 return eRet;
8472 }
8473
8474
8475
8476
8477 /* ======================================================================
8478 FUNCTION
8479 omx_vdec::AllocateDone
8480
8481 DESCRIPTION
8482 Checks if entire buffer pool is allocated by IL Client or not.
8483 Need this to move to IDLE state.
8484
8485 PARAMETERS
8486 None.
8487
8488 RETURN VALUE
8489 true/false.
8490
8491 ========================================================================== */
allocate_done(void)8492 bool omx_vdec::allocate_done(void)
8493 {
8494 bool bRet = false;
8495 bool bRet_In = false;
8496 bool bRet_Out = false;
8497 bool bRet_Out_Extra = false;
8498
8499 bRet_In = allocate_input_done();
8500 bRet_Out = allocate_output_done();
8501 bRet_Out_Extra = allocate_output_extradata_done();
8502
8503 if (bRet_In && bRet_Out && bRet_Out_Extra) {
8504 bRet = true;
8505 }
8506
8507 return bRet;
8508 }
8509 /* ======================================================================
8510 FUNCTION
8511 omx_vdec::AllocateInputDone
8512
8513 DESCRIPTION
8514 Checks if I/P buffer pool is allocated by IL Client or not.
8515
8516 PARAMETERS
8517 None.
8518
8519 RETURN VALUE
8520 true/false.
8521
8522 ========================================================================== */
allocate_input_done(void)8523 bool omx_vdec::allocate_input_done(void)
8524 {
8525 bool bRet = false;
8526 unsigned i=0;
8527
8528 if (m_inp_mem_ptr == NULL) {
8529 return bRet;
8530 }
8531 if (m_inp_mem_ptr ) {
8532 for (; i<drv_ctx.ip_buf.actualcount; i++) {
8533 if (BITMASK_ABSENT(&m_inp_bm_count,i)) {
8534 break;
8535 }
8536 }
8537 }
8538 if (i == drv_ctx.ip_buf.actualcount) {
8539 bRet = true;
8540 DEBUG_PRINT_HIGH("Allocate done for all i/p buffers");
8541 }
8542 if (i==drv_ctx.ip_buf.actualcount && m_inp_bEnabled) {
8543 m_inp_bPopulated = OMX_TRUE;
8544 }
8545 return bRet;
8546 }
8547 /* ======================================================================
8548 FUNCTION
8549 omx_vdec::AllocateOutputDone
8550
8551 DESCRIPTION
8552 Checks if entire O/P buffer pool is allocated by IL Client or not.
8553
8554 PARAMETERS
8555 None.
8556
8557 RETURN VALUE
8558 true/false.
8559
8560 ========================================================================== */
allocate_output_done(void)8561 bool omx_vdec::allocate_output_done(void)
8562 {
8563 bool bRet = false;
8564 unsigned j=0;
8565
8566 if (m_out_mem_ptr == NULL) {
8567 return bRet;
8568 }
8569
8570 if (m_out_mem_ptr) {
8571 for (; j < drv_ctx.op_buf.actualcount; j++) {
8572 if (BITMASK_ABSENT(&m_out_bm_count,j)) {
8573 break;
8574 }
8575 }
8576 }
8577
8578 if (j == drv_ctx.op_buf.actualcount) {
8579 bRet = true;
8580 DEBUG_PRINT_HIGH("Allocate done for all o/p buffers");
8581 if (m_out_bEnabled)
8582 m_out_bPopulated = OMX_TRUE;
8583 }
8584
8585 return bRet;
8586 }
8587
allocate_output_extradata_done(void)8588 bool omx_vdec::allocate_output_extradata_done(void) {
8589 bool bRet = false;
8590 unsigned j=0;
8591 unsigned nBufferCount = 0;
8592
8593 nBufferCount = m_client_out_extradata_info.getBufferCount();
8594
8595 if (!m_client_out_extradata_info.is_client_extradata_enabled()) {
8596 return true;
8597 }
8598
8599 if (m_client_output_extradata_mem_ptr) {
8600 for (; j < nBufferCount; j++) {
8601 if (BITMASK_ABSENT(&m_out_extradata_bm_count,j)) {
8602 break;
8603 }
8604 }
8605
8606 if (j == nBufferCount) {
8607 bRet = true;
8608 DEBUG_PRINT_HIGH("Allocate done for all extradata o/p buffers");
8609 }
8610 }
8611
8612 return bRet;
8613 }
8614 /* ======================================================================
8615 FUNCTION
8616 omx_vdec::ReleaseDone
8617
8618 DESCRIPTION
8619 Checks if IL client has released all the buffers.
8620
8621 PARAMETERS
8622 None.
8623
8624 RETURN VALUE
8625 true/false
8626
8627 ========================================================================== */
release_done(void)8628 bool omx_vdec::release_done(void)
8629 {
8630 bool bRet = false;
8631
8632 if (release_input_done()) {
8633 if (release_output_done()) {
8634 if (release_output_extradata_done()) {
8635 bRet = true;
8636 }
8637 }
8638 }
8639 return bRet;
8640 }
8641
8642
8643 /* ======================================================================
8644 FUNCTION
8645 omx_vdec::ReleaseOutputDone
8646
8647 DESCRIPTION
8648 Checks if IL client has released all the buffers.
8649
8650 PARAMETERS
8651 None.
8652
8653 RETURN VALUE
8654 true/false
8655
8656 ========================================================================== */
release_output_done(void)8657 bool omx_vdec::release_output_done(void)
8658 {
8659 bool bRet = false;
8660 unsigned i=0,j=0;
8661
8662 DEBUG_PRINT_LOW("Value of m_out_mem_ptr %p", m_out_mem_ptr);
8663 if (m_out_mem_ptr) {
8664 for (; j < drv_ctx.op_buf.actualcount ; j++) {
8665 if (BITMASK_PRESENT(&m_out_bm_count,j)) {
8666 break;
8667 }
8668 }
8669 if (j == drv_ctx.op_buf.actualcount) {
8670 m_out_bm_count = 0;
8671 bRet = true;
8672 }
8673 } else {
8674 m_out_bm_count = 0;
8675 bRet = true;
8676 }
8677 return bRet;
8678 }
8679 /* ======================================================================
8680 FUNCTION
8681 omx_vdec::ReleaseInputDone
8682
8683 DESCRIPTION
8684 Checks if IL client has released all the buffers.
8685
8686 PARAMETERS
8687 None.
8688
8689 RETURN VALUE
8690 true/false
8691
8692 ========================================================================== */
release_input_done(void)8693 bool omx_vdec::release_input_done(void)
8694 {
8695 bool bRet = false;
8696 unsigned i=0,j=0;
8697
8698 DEBUG_PRINT_LOW("Value of m_inp_mem_ptr %p",m_inp_mem_ptr);
8699 if (m_inp_mem_ptr) {
8700 for (; j<drv_ctx.ip_buf.actualcount; j++) {
8701 if ( BITMASK_PRESENT(&m_inp_bm_count,j)) {
8702 break;
8703 }
8704 }
8705 if (j==drv_ctx.ip_buf.actualcount) {
8706 bRet = true;
8707 }
8708 } else {
8709 bRet = true;
8710 }
8711 return bRet;
8712 }
8713
release_output_extradata_done(void)8714 bool omx_vdec::release_output_extradata_done(void) {
8715 bool bRet = false;
8716 unsigned i=0,j=0, buffer_count=0;
8717
8718 buffer_count = m_client_out_extradata_info.getBufferCount();
8719 DEBUG_PRINT_LOW("Value of m_client_output_extradata_mem_ptr %p buffer_count - %d",
8720 m_client_output_extradata_mem_ptr, buffer_count);
8721
8722 if (m_client_output_extradata_mem_ptr) {
8723 for (; j<buffer_count; j++) {
8724 if ( BITMASK_PRESENT(&m_out_extradata_bm_count,j)) {
8725 break;
8726 }
8727 }
8728 if (j == buffer_count) {
8729 bRet = true;
8730 }
8731 } else {
8732 bRet = true;
8733 }
8734 return bRet;
8735 }
8736
fill_buffer_done(OMX_HANDLETYPE hComp,OMX_BUFFERHEADERTYPE * buffer)8737 OMX_ERRORTYPE omx_vdec::fill_buffer_done(OMX_HANDLETYPE hComp,
8738 OMX_BUFFERHEADERTYPE * buffer)
8739 {
8740 VIDC_TRACE_NAME_HIGH("FBD");
8741 OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *pPMEMInfo = NULL;
8742 if (!buffer || (buffer - m_out_mem_ptr) >= (int)drv_ctx.op_buf.actualcount) {
8743 DEBUG_PRINT_ERROR("[FBD] ERROR in ptr(%p)", buffer);
8744 return OMX_ErrorBadParameter;
8745 } else if (output_flush_progress) {
8746 DEBUG_PRINT_LOW("FBD: Buffer (%p) flushed", buffer);
8747 buffer->nFilledLen = 0;
8748 buffer->nTimeStamp = 0;
8749 buffer->nFlags &= ~OMX_BUFFERFLAG_EXTRADATA;
8750 buffer->nFlags &= ~QOMX_VIDEO_BUFFERFLAG_EOSEQ;
8751 buffer->nFlags &= ~OMX_BUFFERFLAG_DATACORRUPT;
8752 }
8753
8754 if (m_debug_extradata) {
8755 if (buffer->nFlags & QOMX_VIDEO_BUFFERFLAG_EOSEQ) {
8756 DEBUG_PRINT_HIGH("***************************************************");
8757 DEBUG_PRINT_HIGH("FillBufferDone: End Of Sequence Received");
8758 DEBUG_PRINT_HIGH("***************************************************");
8759 }
8760
8761 if (buffer->nFlags & OMX_BUFFERFLAG_DATACORRUPT) {
8762 DEBUG_PRINT_HIGH("***************************************************");
8763 DEBUG_PRINT_HIGH("FillBufferDone: OMX_BUFFERFLAG_DATACORRUPT Received");
8764 DEBUG_PRINT_HIGH("***************************************************");
8765 }
8766 }
8767
8768
8769 DEBUG_PRINT_LOW("fill_buffer_done: bufhdr = %p, bufhdr->pBuffer = %p, flags: 0x%x, timestamp: %lld",
8770 buffer, buffer->pBuffer, buffer->nFlags, buffer->nTimeStamp);
8771 pending_output_buffers --;
8772 VIDC_TRACE_INT_LOW("FTB-pending", pending_output_buffers);
8773
8774 if (buffer->nFlags & OMX_BUFFERFLAG_EOS) {
8775 DEBUG_PRINT_HIGH("Output EOS has been reached");
8776 if (!output_flush_progress)
8777 post_event((unsigned)NULL, (unsigned)NULL,
8778 OMX_COMPONENT_GENERATE_EOS_DONE);
8779
8780 if (psource_frame) {
8781 m_cb.EmptyBufferDone(&m_cmp, m_app_data, psource_frame);
8782 psource_frame = NULL;
8783 }
8784 if (pdest_frame) {
8785 pdest_frame->nFilledLen = 0;
8786 m_input_free_q.insert_entry((unsigned long) pdest_frame,(unsigned)NULL,
8787 (unsigned)NULL);
8788 pdest_frame = NULL;
8789 }
8790 }
8791
8792 #ifdef OUTPUT_EXTRADATA_LOG
8793 if (outputExtradataFile) {
8794 int buf_index = buffer - m_out_mem_ptr;
8795 OMX_U8 *pBuffer = (OMX_U8 *)(drv_ctx.ptr_outputbuffer[buf_index].bufferaddr);
8796
8797 OMX_OTHER_EXTRADATATYPE *p_extra = NULL;
8798 p_extra = (OMX_OTHER_EXTRADATATYPE *)
8799 ((unsigned long)(pBuffer + buffer->nOffset + buffer->nFilledLen + 3)&(~3));
8800
8801 while (p_extra && (OMX_U8*)p_extra < (pBuffer + buffer->nAllocLen) ) {
8802 DEBUG_PRINT_LOW("WRITING extradata, size=%d,type=%x",
8803 p_extra->nSize, p_extra->eType);
8804 fwrite (p_extra,1,p_extra->nSize,outputExtradataFile);
8805
8806 if (p_extra->eType == OMX_ExtraDataNone) {
8807 break;
8808 }
8809 p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + p_extra->nSize);
8810 }
8811 }
8812 #endif
8813
8814 /* For use buffer we need to copy the data */
8815 if (!output_flush_progress) {
8816 /* This is the error check for non-recoverable errros */
8817 bool is_duplicate_ts_valid = true;
8818 bool is_interlaced = (drv_ctx.interlace != VDEC_InterlaceFrameProgressive);
8819
8820 if (output_capability == V4L2_PIX_FMT_MPEG4 ||
8821 output_capability == V4L2_PIX_FMT_MPEG2 ||
8822 output_capability == V4L2_PIX_FMT_DIVX ||
8823 output_capability == V4L2_PIX_FMT_DIVX_311)
8824 is_duplicate_ts_valid = false;
8825
8826 if ((output_capability == V4L2_PIX_FMT_H264 ||
8827 output_capability == V4L2_PIX_FMT_H264_MVC) &&
8828 is_interlaced) {
8829 if (buffer->nFlags & QOMX_VIDEO_BUFFERFLAG_MBAFF) {
8830 is_interlaced = false;
8831 }
8832 }
8833
8834 if (buffer->nFilledLen > 0) {
8835 time_stamp_dts.get_next_timestamp(buffer,
8836 is_interlaced && is_duplicate_ts_valid);
8837 if (m_debug_timestamp) {
8838 {
8839 OMX_TICKS expected_ts = 0;
8840 m_timestamp_list.pop_min_ts(expected_ts);
8841 if (is_interlaced && is_duplicate_ts_valid) {
8842 m_timestamp_list.pop_min_ts(expected_ts);
8843 }
8844 DEBUG_PRINT_LOW("Current timestamp (%lld),Popped TIMESTAMP (%lld) from list",
8845 buffer->nTimeStamp, expected_ts);
8846
8847 if (buffer->nTimeStamp != expected_ts) {
8848 DEBUG_PRINT_ERROR("ERROR in omx_vdec::async_message_process timestamp Check");
8849 }
8850 }
8851 }
8852 }
8853 }
8854 VIDC_TRACE_INT_LOW("FBD-TS", buffer->nTimeStamp / 1000);
8855
8856 if (m_cb.FillBufferDone) {
8857 if (buffer->nFilledLen > 0) {
8858 if (arbitrary_bytes)
8859 adjust_timestamp(buffer->nTimeStamp);
8860 else
8861 set_frame_rate(buffer->nTimeStamp);
8862
8863 proc_frms++;
8864 if (perf_flag) {
8865 if (1 == proc_frms) {
8866 dec_time.stop();
8867 latency = dec_time.processing_time_us() - latency;
8868 DEBUG_PRINT_HIGH(">>> FBD Metrics: Latency(%.2f)mS", latency / 1e3);
8869 dec_time.start();
8870 fps_metrics.start();
8871 }
8872 if (buffer->nFlags & OMX_BUFFERFLAG_EOS) {
8873 OMX_U64 proc_time = 0;
8874 fps_metrics.stop();
8875 proc_time = fps_metrics.processing_time_us();
8876 DEBUG_PRINT_HIGH(">>> FBD Metrics: proc_frms(%u) proc_time(%.2f)S fps(%.2f)",
8877 (unsigned int)proc_frms, (float)proc_time / 1e6,
8878 (float)(1e6 * proc_frms) / proc_time);
8879 }
8880 }
8881 }
8882 if (buffer->nFlags & OMX_BUFFERFLAG_EOS) {
8883 prev_ts = LLONG_MAX;
8884 rst_prev_ts = true;
8885 proc_frms = 0;
8886 }
8887
8888 pPMEMInfo = (OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *)
8889 ((OMX_QCOM_PLATFORM_PRIVATE_LIST *)
8890 buffer->pPlatformPrivate)->entryList->entry;
8891 DEBUG_PRINT_LOW("Before FBD callback Accessed Pmeminfo %lu",pPMEMInfo->pmem_fd);
8892 OMX_BUFFERHEADERTYPE *il_buffer;
8893 il_buffer = client_buffers.get_il_buf_hdr(buffer);
8894 OMX_U32 current_framerate = (int)(drv_ctx.frame_rate.fps_numerator / drv_ctx.frame_rate.fps_denominator);
8895
8896 if (il_buffer && m_last_rendered_TS >= 0) {
8897 OMX_TICKS ts_delta = (OMX_TICKS)llabs(il_buffer->nTimeStamp - m_last_rendered_TS);
8898
8899 // Current frame can be send for rendering if
8900 // (a) current FPS is <= 60
8901 // (b) is the next frame after the frame with TS 0
8902 // (c) is the first frame after seek
8903 // (d) the delta TS b\w two consecutive frames is > 16 ms
8904 // (e) its TS is equal to previous frame TS
8905 // (f) if marked EOS
8906
8907 if(current_framerate <= 60 || m_last_rendered_TS == 0 ||
8908 il_buffer->nTimeStamp == 0 || ts_delta >= 16000 ||
8909 ts_delta == 0 || (il_buffer->nFlags & OMX_BUFFERFLAG_EOS)) {
8910 m_last_rendered_TS = il_buffer->nTimeStamp;
8911 } else {
8912 //mark for droping
8913 buffer->nFilledLen = 0;
8914 }
8915
8916 DEBUG_PRINT_LOW(" -- %s Frame -- info:: fps(%d) lastRenderTime(%lld) bufferTs(%lld) ts_delta(%lld)",
8917 buffer->nFilledLen? "Rendering":"Dropping",current_framerate,m_last_rendered_TS,
8918 il_buffer->nTimeStamp,ts_delta);
8919
8920 //above code makes sure that delta b\w two consecutive frames is not
8921 //greater than 16ms, slow-mo feature, so cap fps to max 60
8922 if (current_framerate > 60 ) {
8923 current_framerate = 60;
8924 }
8925 }
8926
8927 // add current framerate to gralloc meta data
8928 if ((buffer->nFilledLen > 0) && m_enable_android_native_buffers && m_out_mem_ptr) {
8929 // If valid fps was received, directly send it to display for the 1st fbd.
8930 // Otherwise, calculate fps using fbd timestamps
8931 float refresh_rate = m_fps_prev;
8932 if (m_fps_received) {
8933 if (1 == proc_frms) {
8934 refresh_rate = m_fps_received / (float)(1<<16);
8935 }
8936 } else {
8937 // check if dynamic refresh rate change feature enabled or not
8938 if (m_drc_enable) {
8939 // set coarse fps when 2 fbds received and
8940 // set fps again when 30 fbds received as it should be
8941 // more accurate than the one set when only 2 fbds received.
8942 if (2 == proc_frms || 30 == proc_frms) {
8943 if (drv_ctx.frame_rate.fps_denominator) {
8944 refresh_rate = drv_ctx.frame_rate.fps_numerator /
8945 (float) drv_ctx.frame_rate.fps_denominator;
8946 }
8947 }
8948 } else {
8949 // calculate and set refresh rate for every frame from second frame onwards
8950 // display will assume the default refresh rate for first frame (which is 60 fps)
8951 if (m_fps_prev) {
8952 if (drv_ctx.frame_rate.fps_denominator) {
8953 refresh_rate = drv_ctx.frame_rate.fps_numerator /
8954 (float) drv_ctx.frame_rate.fps_denominator;
8955 }
8956 }
8957 }
8958 }
8959 if (refresh_rate > 60) {
8960 refresh_rate = 60;
8961 }
8962 DEBUG_PRINT_LOW("frc set refresh_rate %f, frame %d", refresh_rate, proc_frms);
8963 OMX_U32 buf_index = buffer - m_out_mem_ptr;
8964 setMetaData((private_handle_t *)native_buffer[buf_index].privatehandle,
8965 UPDATE_REFRESH_RATE, (void*)&refresh_rate);
8966 m_fps_prev = refresh_rate;
8967 }
8968
8969 if (buffer->nFilledLen && m_enable_android_native_buffers && m_out_mem_ptr) {
8970 OMX_U32 buf_index = buffer - m_out_mem_ptr;
8971 DEBUG_PRINT_LOW("stereo_output_mode = %d",stereo_output_mode);
8972 setMetaData((private_handle_t *)native_buffer[buf_index].privatehandle,
8973 S3D_FORMAT, (void*)&stereo_output_mode);
8974 }
8975
8976 if (il_buffer) {
8977 log_output_buffers(il_buffer);
8978 if (dynamic_buf_mode) {
8979 unsigned int nPortIndex = 0;
8980 nPortIndex = buffer-((OMX_BUFFERHEADERTYPE *)client_buffers.get_il_buf_hdr());
8981
8982 // Since we're passing around handles, adjust nFilledLen and nAllocLen
8983 // to size of the handle. Do it _after_ log_output_buffers which
8984 // requires the respective sizes to be accurate.
8985
8986 buffer->nAllocLen = sizeof(struct VideoDecoderOutputMetaData);
8987 buffer->nFilledLen = buffer->nFilledLen ?
8988 sizeof(struct VideoDecoderOutputMetaData) : 0;
8989
8990 //Clear graphic buffer handles in dynamic mode
8991 if (nPortIndex < drv_ctx.op_buf.actualcount &&
8992 nPortIndex < MAX_NUM_INPUT_OUTPUT_BUFFERS) {
8993 native_buffer[nPortIndex].privatehandle = NULL;
8994 native_buffer[nPortIndex].nativehandle = NULL;
8995 } else {
8996 DEBUG_PRINT_ERROR("[FBD]Invalid native_buffer index: %d", nPortIndex);
8997 return OMX_ErrorBadParameter;
8998 }
8999 }
9000 m_cb.FillBufferDone (hComp,m_app_data,il_buffer);
9001 } else {
9002 DEBUG_PRINT_ERROR("Invalid buffer address from get_il_buf_hdr");
9003 return OMX_ErrorBadParameter;
9004 }
9005 DEBUG_PRINT_LOW("After Fill Buffer Done callback %lu",pPMEMInfo->pmem_fd);
9006 } else {
9007 return OMX_ErrorBadParameter;
9008 }
9009
9010 #ifdef ADAPTIVE_PLAYBACK_SUPPORTED
9011 if (m_smoothstreaming_mode && m_out_mem_ptr) {
9012 OMX_U32 buf_index = buffer - m_out_mem_ptr;
9013 BufferDim_t dim;
9014 private_handle_t *private_handle = NULL;
9015 dim.sliceWidth = framesize.nWidth;
9016 dim.sliceHeight = framesize.nHeight;
9017 if (buf_index < drv_ctx.op_buf.actualcount &&
9018 buf_index < MAX_NUM_INPUT_OUTPUT_BUFFERS &&
9019 native_buffer[buf_index].privatehandle)
9020 private_handle = native_buffer[buf_index].privatehandle;
9021 if (private_handle) {
9022 DEBUG_PRINT_LOW("set metadata: update buf-geometry with stride %d slice %d",
9023 dim.sliceWidth, dim.sliceHeight);
9024 setMetaData(private_handle, UPDATE_BUFFER_GEOMETRY, (void*)&dim);
9025 }
9026 }
9027 #endif
9028
9029 return OMX_ErrorNone;
9030 }
9031
empty_buffer_done(OMX_HANDLETYPE hComp,OMX_BUFFERHEADERTYPE * buffer)9032 OMX_ERRORTYPE omx_vdec::empty_buffer_done(OMX_HANDLETYPE hComp,
9033 OMX_BUFFERHEADERTYPE* buffer)
9034 {
9035 VIDC_TRACE_NAME_HIGH("EBD");
9036 int nBufferIndex = buffer - m_inp_mem_ptr;
9037
9038 if (buffer == NULL || (nBufferIndex >= (int)drv_ctx.ip_buf.actualcount)) {
9039 DEBUG_PRINT_ERROR("empty_buffer_done: ERROR bufhdr = %p", buffer);
9040 return OMX_ErrorBadParameter;
9041 }
9042
9043 DEBUG_PRINT_LOW("empty_buffer_done: bufhdr = %p, bufhdr->pBuffer = %p, bufhdr->nFlags = 0x%x",
9044 buffer, buffer->pBuffer, buffer->nFlags);
9045 pending_input_buffers--;
9046 VIDC_TRACE_INT_LOW("ETB-pending", pending_input_buffers);
9047
9048 if (arbitrary_bytes) {
9049 if (pdest_frame == NULL && input_flush_progress == false) {
9050 DEBUG_PRINT_LOW("Push input from buffer done address of Buffer %p",buffer);
9051 pdest_frame = buffer;
9052 buffer->nFilledLen = 0;
9053 buffer->nTimeStamp = LLONG_MAX;
9054 push_input_buffer (hComp);
9055 } else {
9056 DEBUG_PRINT_LOW("Push buffer into freeq address of Buffer %p",buffer);
9057 buffer->nFilledLen = 0;
9058 if (!m_input_free_q.insert_entry((unsigned long)buffer,
9059 (unsigned)NULL, (unsigned)NULL)) {
9060 DEBUG_PRINT_ERROR("ERROR:i/p free Queue is FULL Error");
9061 }
9062 }
9063 } else if (m_cb.EmptyBufferDone) {
9064 buffer->nFilledLen = 0;
9065 if (input_use_buffer == true) {
9066 buffer = &m_inp_heap_ptr[buffer-m_inp_mem_ptr];
9067 }
9068
9069 /* Restore the FD that we over-wrote in ETB */
9070 if (m_input_pass_buffer_fd) {
9071 buffer->pBuffer = (OMX_U8*)(uintptr_t)drv_ctx.ptr_inputbuffer[nBufferIndex].pmem_fd;
9072 }
9073
9074 m_cb.EmptyBufferDone(hComp ,m_app_data, buffer);
9075 }
9076 return OMX_ErrorNone;
9077 }
9078
async_message_process(void * context,void * message)9079 int omx_vdec::async_message_process (void *context, void* message)
9080 {
9081 omx_vdec* omx = NULL;
9082 struct vdec_msginfo *vdec_msg = NULL;
9083 OMX_BUFFERHEADERTYPE* omxhdr = NULL;
9084 struct v4l2_buffer *v4l2_buf_ptr = NULL;
9085 struct v4l2_plane *plane = NULL;
9086 struct vdec_output_frameinfo *output_respbuf = NULL;
9087 int rc=1;
9088 if (context == NULL || message == NULL) {
9089 DEBUG_PRINT_ERROR("FATAL ERROR in omx_vdec::async_message_process NULL Check");
9090 return -1;
9091 }
9092 vdec_msg = (struct vdec_msginfo *)message;
9093
9094 omx = reinterpret_cast<omx_vdec*>(context);
9095
9096 switch (vdec_msg->msgcode) {
9097
9098 case VDEC_MSG_EVT_HW_ERROR:
9099 omx->post_event ((unsigned)NULL, vdec_msg->status_code,\
9100 OMX_COMPONENT_GENERATE_HARDWARE_ERROR);
9101 break;
9102
9103 case VDEC_MSG_EVT_HW_OVERLOAD:
9104 omx->post_event ((unsigned)NULL, vdec_msg->status_code,\
9105 OMX_COMPONENT_GENERATE_HARDWARE_OVERLOAD);
9106 break;
9107
9108 case VDEC_MSG_EVT_HW_UNSUPPORTED:
9109 omx->post_event ((unsigned)NULL, vdec_msg->status_code,\
9110 OMX_COMPONENT_GENERATE_UNSUPPORTED_SETTING);
9111 break;
9112
9113 case VDEC_MSG_RESP_START_DONE:
9114 omx->post_event ((unsigned)NULL, vdec_msg->status_code,\
9115 OMX_COMPONENT_GENERATE_START_DONE);
9116 break;
9117
9118 case VDEC_MSG_RESP_STOP_DONE:
9119 omx->post_event ((unsigned)NULL, vdec_msg->status_code,\
9120 OMX_COMPONENT_GENERATE_STOP_DONE);
9121 break;
9122
9123 case VDEC_MSG_RESP_RESUME_DONE:
9124 omx->post_event ((unsigned)NULL, vdec_msg->status_code,\
9125 OMX_COMPONENT_GENERATE_RESUME_DONE);
9126 break;
9127
9128 case VDEC_MSG_RESP_PAUSE_DONE:
9129 omx->post_event ((unsigned)NULL, vdec_msg->status_code,\
9130 OMX_COMPONENT_GENERATE_PAUSE_DONE);
9131 break;
9132
9133 case VDEC_MSG_RESP_FLUSH_INPUT_DONE:
9134 omx->post_event ((unsigned)NULL, vdec_msg->status_code,\
9135 OMX_COMPONENT_GENERATE_EVENT_INPUT_FLUSH);
9136 break;
9137 case VDEC_MSG_RESP_FLUSH_OUTPUT_DONE:
9138 omx->post_event ((unsigned)NULL, vdec_msg->status_code,\
9139 OMX_COMPONENT_GENERATE_EVENT_OUTPUT_FLUSH);
9140 break;
9141 case VDEC_MSG_RESP_INPUT_FLUSHED:
9142 case VDEC_MSG_RESP_INPUT_BUFFER_DONE:
9143
9144 /* omxhdr = (OMX_BUFFERHEADERTYPE* )
9145 vdec_msg->msgdata.input_frame_clientdata; */
9146
9147 v4l2_buf_ptr = (v4l2_buffer*)vdec_msg->msgdata.input_frame_clientdata;
9148 if (omx->m_inp_mem_ptr == NULL || v4l2_buf_ptr == NULL ||
9149 v4l2_buf_ptr->index >= omx->drv_ctx.ip_buf.actualcount) {
9150 omxhdr = NULL;
9151 vdec_msg->status_code = VDEC_S_EFATAL;
9152 break;
9153
9154 }
9155 omxhdr = omx->m_inp_mem_ptr + v4l2_buf_ptr->index;
9156
9157 if (v4l2_buf_ptr->flags & V4L2_QCOM_BUF_INPUT_UNSUPPORTED) {
9158 DEBUG_PRINT_HIGH("Unsupported input");
9159 omx->post_event ((unsigned)NULL, vdec_msg->status_code,\
9160 OMX_COMPONENT_GENERATE_HARDWARE_ERROR);
9161 }
9162 if (v4l2_buf_ptr->flags & V4L2_QCOM_BUF_DATA_CORRUPT) {
9163 omxhdr->nFlags |= OMX_BUFFERFLAG_DATACORRUPT;
9164 vdec_msg->status_code = VDEC_S_INPUT_BITSTREAM_ERR;
9165 }
9166 if (omxhdr->nFlags & OMX_BUFFERFLAG_CODECCONFIG) {
9167
9168 DEBUG_PRINT_LOW("Decrement codec_config buffer counter");
9169 android_atomic_dec(&omx->m_queued_codec_config_count);
9170 if ((android_atomic_add(0, &omx->m_queued_codec_config_count) == 0) &&
9171 BITMASK_PRESENT(&omx->m_flags, OMX_COMPONENT_FLUSH_DEFERRED)) {
9172 DEBUG_PRINT_LOW("sem post for CODEC CONFIG buffer");
9173 sem_post(&omx->m_safe_flush);
9174 }
9175 }
9176 if (v4l2_buf_ptr->flags & V4L2_BUF_FLAG_KEYFRAME ||
9177 v4l2_buf_ptr->flags & V4L2_QCOM_BUF_FLAG_IDRFRAME) {
9178 omxhdr->nFlags |= OMX_BUFFERFLAG_SYNCFRAME;
9179 }
9180 omx->post_event ((unsigned long)omxhdr,vdec_msg->status_code,
9181 OMX_COMPONENT_GENERATE_EBD);
9182 break;
9183 case VDEC_MSG_EVT_INFO_FIELD_DROPPED:
9184 int64_t *timestamp;
9185 timestamp = (int64_t *) malloc(sizeof(int64_t));
9186 if (timestamp) {
9187 *timestamp = vdec_msg->msgdata.output_frame.time_stamp;
9188 omx->post_event ((unsigned long)timestamp, vdec_msg->status_code,
9189 OMX_COMPONENT_GENERATE_INFO_FIELD_DROPPED);
9190 DEBUG_PRINT_HIGH("Field dropped time stamp is %lld",
9191 (long long)vdec_msg->msgdata.output_frame.time_stamp);
9192 }
9193 break;
9194 case VDEC_MSG_RESP_OUTPUT_FLUSHED:
9195 case VDEC_MSG_RESP_OUTPUT_BUFFER_DONE:
9196
9197 v4l2_buf_ptr = (v4l2_buffer*)vdec_msg->msgdata.output_frame.client_data;
9198 if (v4l2_buf_ptr == NULL || omx->m_out_mem_ptr == NULL ||
9199 v4l2_buf_ptr->index >= omx->drv_ctx.op_buf.actualcount) {
9200 omxhdr = NULL;
9201 vdec_msg->status_code = VDEC_S_EFATAL;
9202 break;
9203 }
9204 plane = v4l2_buf_ptr->m.planes;
9205 omxhdr = omx->m_out_mem_ptr + v4l2_buf_ptr->index;
9206
9207 if (omxhdr && omxhdr->pOutputPortPrivate &&
9208 ((omxhdr - omx->m_out_mem_ptr) < (int)omx->drv_ctx.op_buf.actualcount) &&
9209 (((struct vdec_output_frameinfo *)omxhdr->pOutputPortPrivate
9210 - omx->drv_ctx.ptr_respbuffer) < (int)omx->drv_ctx.op_buf.actualcount)) {
9211
9212 if (vdec_msg->msgdata.output_frame.len <= omxhdr->nAllocLen) {
9213 omxhdr->nFilledLen = vdec_msg->msgdata.output_frame.len;
9214 omxhdr->nOffset = vdec_msg->msgdata.output_frame.offset;
9215 omxhdr->nTimeStamp = vdec_msg->msgdata.output_frame.time_stamp;
9216 omxhdr->nFlags = 0;
9217
9218 if (v4l2_buf_ptr->flags & V4L2_QCOM_BUF_FLAG_EOS) {
9219 omxhdr->nFlags |= OMX_BUFFERFLAG_EOS;
9220 //rc = -1;
9221 }
9222 if (omxhdr->nFilledLen) {
9223 omxhdr->nFlags |= OMX_BUFFERFLAG_ENDOFFRAME;
9224 }
9225 if (v4l2_buf_ptr->flags & V4L2_BUF_FLAG_KEYFRAME || v4l2_buf_ptr->flags & V4L2_QCOM_BUF_FLAG_IDRFRAME) {
9226 omxhdr->nFlags |= OMX_BUFFERFLAG_SYNCFRAME;
9227 } else {
9228 omxhdr->nFlags &= ~OMX_BUFFERFLAG_SYNCFRAME;
9229 }
9230 if (v4l2_buf_ptr->flags & V4L2_QCOM_BUF_FLAG_EOSEQ) {
9231 omxhdr->nFlags |= QOMX_VIDEO_BUFFERFLAG_EOSEQ;
9232 }
9233 if (v4l2_buf_ptr->flags & V4L2_QCOM_BUF_FLAG_DECODEONLY) {
9234 omxhdr->nFlags |= OMX_BUFFERFLAG_DECODEONLY;
9235 }
9236 if (v4l2_buf_ptr->flags & V4L2_MSM_BUF_FLAG_MBAFF) {
9237 omxhdr->nFlags |= QOMX_VIDEO_BUFFERFLAG_MBAFF;
9238 }
9239 if (v4l2_buf_ptr->flags & V4L2_QCOM_BUF_FLAG_READONLY) {
9240 omxhdr->nFlags |= OMX_BUFFERFLAG_READONLY;
9241 DEBUG_PRINT_LOW("F_B_D: READONLY BUFFER - REFERENCE WITH F/W fd = %d",
9242 omx->drv_ctx.ptr_outputbuffer[v4l2_buf_ptr->index].pmem_fd);
9243 }
9244
9245 if (omxhdr && (v4l2_buf_ptr->flags & V4L2_QCOM_BUF_DROP_FRAME) &&
9246 !omx->output_flush_progress &&
9247 !(v4l2_buf_ptr->flags & V4L2_QCOM_BUF_FLAG_DECODEONLY) &&
9248 !(v4l2_buf_ptr->flags & V4L2_QCOM_BUF_FLAG_EOS)) {
9249 omx->time_stamp_dts.remove_time_stamp(
9250 omxhdr->nTimeStamp,
9251 (omx->drv_ctx.interlace != VDEC_InterlaceFrameProgressive)
9252 ?true:false);
9253
9254 omxhdr->nFilledLen = 0;
9255 omx->pending_output_buffers--;
9256 omx->post_event ((unsigned long)NULL,(unsigned long)omx->client_buffers.get_il_buf_hdr(omxhdr),
9257 OMX_COMPONENT_GENERATE_FTB);
9258 break;
9259 }
9260 if (v4l2_buf_ptr->flags & V4L2_QCOM_BUF_DATA_CORRUPT) {
9261 omxhdr->nFlags |= OMX_BUFFERFLAG_DATACORRUPT;
9262 }
9263
9264 output_respbuf = (struct vdec_output_frameinfo *)\
9265 omxhdr->pOutputPortPrivate;
9266 if (!output_respbuf) {
9267 DEBUG_PRINT_ERROR("async_message_process: invalid output buf received");
9268 return -1;
9269 }
9270 output_respbuf->len = vdec_msg->msgdata.output_frame.len;
9271 output_respbuf->offset = vdec_msg->msgdata.output_frame.offset;
9272
9273 if (v4l2_buf_ptr->flags & V4L2_BUF_FLAG_KEYFRAME) {
9274 output_respbuf->pic_type = PICTURE_TYPE_I;
9275 }
9276 if (v4l2_buf_ptr->flags & V4L2_BUF_FLAG_PFRAME) {
9277 output_respbuf->pic_type = PICTURE_TYPE_P;
9278 }
9279 if (v4l2_buf_ptr->flags & V4L2_BUF_FLAG_BFRAME) {
9280 output_respbuf->pic_type = PICTURE_TYPE_B;
9281 }
9282
9283 if (vdec_msg->msgdata.output_frame.len) {
9284 if (!omx->output_flush_progress && (omxhdr->nFilledLen > 0)) {
9285 // set the default colorspace advised by client, since the bitstream may be
9286 // devoid of colorspace-info.
9287 if (omx->m_enable_android_native_buffers) {
9288 ColorSpace_t color_space = ITU_R_601;
9289
9290 // Disabled ?
9291 // WA for VP8. Vp8 encoder does not embed color-info (yet!).
9292 // Encoding RGBA results in 601-LR for all resolutions.
9293 // This conflicts with the client't defaults which are based on resolution.
9294 // Eg: 720p will be encoded as 601-LR. Client will say 709.
9295 // Re-enable this code once vp8 encoder generates color-info and hence the
9296 // decoder will be able to override with the correct source color.
9297 #if 0
9298 switch (omx->m_client_color_space.sAspects.mPrimaries) {
9299 case ColorAspects::PrimariesBT601_6_625:
9300 case ColorAspects::PrimariesBT601_6_525:
9301 {
9302 color_space = omx->m_client_color_space.sAspects.mRange == ColorAspects::RangeFull ?
9303 ITU_R_601_FR : ITU_R_601;
9304 break;
9305 }
9306 case ColorAspects::PrimariesBT709_5:
9307 {
9308 color_space = ITU_R_709;
9309 break;
9310 }
9311 default:
9312 {
9313 break;
9314 }
9315 }
9316 #endif
9317 DEBUG_PRINT_LOW("setMetaData for Color Space (client) = 0x%x (601=%u FR=%u 709=%u)",
9318 color_space, ITU_R_601, ITU_R_601_FR, ITU_R_709);
9319 omx->set_colorspace_in_handle(color_space, omxhdr - omx->m_out_mem_ptr);
9320 }
9321 }
9322
9323 DEBUG_PRINT_LOW("Processing extradata");
9324 omx->handle_extradata(omxhdr);
9325
9326 if (omx->m_extradata_info.output_crop_updated) {
9327 DEBUG_PRINT_LOW("Read FBD crop from output extra data");
9328 vdec_msg->msgdata.output_frame.framesize.left = omx->m_extradata_info.output_crop_rect.nLeft;
9329 vdec_msg->msgdata.output_frame.framesize.top = omx->m_extradata_info.output_crop_rect.nTop;
9330 vdec_msg->msgdata.output_frame.framesize.right = omx->m_extradata_info.output_crop_rect.nWidth;
9331 vdec_msg->msgdata.output_frame.framesize.bottom = omx->m_extradata_info.output_crop_rect.nHeight;
9332 vdec_msg->msgdata.output_frame.picsize.frame_width = omx->m_extradata_info.output_width;
9333 vdec_msg->msgdata.output_frame.picsize.frame_height = omx->m_extradata_info.output_height;
9334 } else {
9335 DEBUG_PRINT_LOW("Read FBD crop from v4l2 reserved fields");
9336 vdec_msg->msgdata.output_frame.framesize.left = plane[0].reserved[2];
9337 vdec_msg->msgdata.output_frame.framesize.top = plane[0].reserved[3];
9338 vdec_msg->msgdata.output_frame.framesize.right = plane[0].reserved[2] + plane[0].reserved[4];
9339 vdec_msg->msgdata.output_frame.framesize.bottom = plane[0].reserved[3] + plane[0].reserved[5];
9340 vdec_msg->msgdata.output_frame.picsize.frame_width = plane[0].reserved[6];
9341 vdec_msg->msgdata.output_frame.picsize.frame_height = plane[0].reserved[7];
9342
9343 /* Copy these values back to OMX internal variables to make both handlign same*/
9344
9345 omx->m_extradata_info.output_crop_rect.nLeft = vdec_msg->msgdata.output_frame.framesize.left;
9346 omx->m_extradata_info.output_crop_rect.nTop = vdec_msg->msgdata.output_frame.framesize.top;
9347 omx->m_extradata_info.output_crop_rect.nWidth = vdec_msg->msgdata.output_frame.framesize.right;
9348 omx->m_extradata_info.output_crop_rect.nHeight = vdec_msg->msgdata.output_frame.framesize.bottom;
9349 omx->m_extradata_info.output_width = vdec_msg->msgdata.output_frame.picsize.frame_width;
9350 omx->m_extradata_info.output_height = vdec_msg->msgdata.output_frame.picsize.frame_height;
9351 }
9352 }
9353
9354 vdec_msg->msgdata.output_frame.bufferaddr =
9355 omx->drv_ctx.ptr_outputbuffer[v4l2_buf_ptr->index].bufferaddr;
9356
9357 if (vdec_msg->msgdata.output_frame.len)
9358 memcpy(&omx->drv_ctx.frame_size,
9359 &vdec_msg->msgdata.output_frame.framesize,
9360 sizeof(struct vdec_framesize));
9361
9362 DEBUG_PRINT_LOW("[RespBufDone] Fd(%d) Buf(%p) Ts(%lld) PicType(%u) Flags (0x%x)"
9363 " FillLen(%u) Crop: L(%u) T(%u) R(%u) B(%u)",
9364 omx->drv_ctx.ptr_outputbuffer[v4l2_buf_ptr->index].pmem_fd,
9365 omxhdr, (long long)vdec_msg->msgdata.output_frame.time_stamp,
9366 vdec_msg->msgdata.output_frame.pic_type, v4l2_buf_ptr->flags,
9367 (unsigned int)vdec_msg->msgdata.output_frame.len,
9368 vdec_msg->msgdata.output_frame.framesize.left,
9369 vdec_msg->msgdata.output_frame.framesize.top,
9370 vdec_msg->msgdata.output_frame.framesize.right,
9371 vdec_msg->msgdata.output_frame.framesize.bottom);
9372
9373 /* Post event if resolution OR crop changed */
9374 /* filled length will be changed if resolution changed */
9375 /* Crop parameters can be changed even without resolution change */
9376 if (omxhdr->nFilledLen
9377 && ((omx->prev_n_filled_len != omxhdr->nFilledLen)
9378 || (omx->drv_ctx.frame_size.left != vdec_msg->msgdata.output_frame.framesize.left)
9379 || (omx->drv_ctx.frame_size.top != vdec_msg->msgdata.output_frame.framesize.top)
9380 || (omx->drv_ctx.frame_size.right != vdec_msg->msgdata.output_frame.framesize.right)
9381 || (omx->drv_ctx.frame_size.bottom != vdec_msg->msgdata.output_frame.framesize.bottom)
9382 || (omx->drv_ctx.video_resolution.frame_width != vdec_msg->msgdata.output_frame.picsize.frame_width)
9383 || (omx->drv_ctx.video_resolution.frame_height != vdec_msg->msgdata.output_frame.picsize.frame_height) )) {
9384
9385 DEBUG_PRINT_HIGH("Parameters Changed From: Len: %u, WxH: %dx%d, L: %u, T: %u, R: %u, B: %u --> Len: %u, WxH: %dx%d, L: %u, T: %u, R: %u, B: %u",
9386 omx->prev_n_filled_len,
9387 omx->drv_ctx.video_resolution.frame_width,
9388 omx->drv_ctx.video_resolution.frame_height,
9389 omx->drv_ctx.frame_size.left, omx->drv_ctx.frame_size.top,
9390 omx->drv_ctx.frame_size.right, omx->drv_ctx.frame_size.bottom,
9391 omxhdr->nFilledLen, vdec_msg->msgdata.output_frame.picsize.frame_width,
9392 vdec_msg->msgdata.output_frame.picsize.frame_height,
9393 vdec_msg->msgdata.output_frame.framesize.left,
9394 vdec_msg->msgdata.output_frame.framesize.top,
9395 vdec_msg->msgdata.output_frame.framesize.right,
9396 vdec_msg->msgdata.output_frame.framesize.bottom);
9397
9398 omx->drv_ctx.video_resolution.frame_width =
9399 vdec_msg->msgdata.output_frame.picsize.frame_width;
9400 omx->drv_ctx.video_resolution.frame_height =
9401 vdec_msg->msgdata.output_frame.picsize.frame_height;
9402 if (omx->drv_ctx.output_format == VDEC_YUV_FORMAT_NV12) {
9403 omx->drv_ctx.video_resolution.stride =
9404 VENUS_Y_STRIDE(COLOR_FMT_NV12, omx->drv_ctx.video_resolution.frame_width);
9405 omx->drv_ctx.video_resolution.scan_lines =
9406 VENUS_Y_SCANLINES(COLOR_FMT_NV12, omx->drv_ctx.video_resolution.frame_height);
9407 } else if (omx->drv_ctx.output_format == VDEC_YUV_FORMAT_NV12_UBWC) {
9408 omx->drv_ctx.video_resolution.stride =
9409 VENUS_Y_STRIDE(COLOR_FMT_NV12_UBWC, omx->drv_ctx.video_resolution.frame_width);
9410 omx->drv_ctx.video_resolution.scan_lines =
9411 VENUS_Y_SCANLINES(COLOR_FMT_NV12_UBWC, omx->drv_ctx.video_resolution.frame_height);
9412 } else if (omx->drv_ctx.output_format == VDEC_YUV_FORMAT_NV12_TP10_UBWC) {
9413 omx->drv_ctx.video_resolution.stride =
9414 VENUS_Y_STRIDE(COLOR_FMT_NV12_BPP10_UBWC, omx->drv_ctx.video_resolution.frame_width);
9415 omx->drv_ctx.video_resolution.scan_lines =
9416 VENUS_Y_SCANLINES(COLOR_FMT_NV12_BPP10_UBWC, omx->drv_ctx.video_resolution.frame_height);
9417 }
9418
9419 omx->post_event(OMX_CORE_OUTPUT_PORT_INDEX,
9420 OMX_IndexConfigCommonOutputCrop,
9421 OMX_COMPONENT_GENERATE_PORT_RECONFIG);
9422 }
9423
9424 if (omxhdr->nFilledLen)
9425 omx->prev_n_filled_len = omxhdr->nFilledLen;
9426
9427 if (omxhdr && omxhdr->nFilledLen && !omx->m_need_turbo) {
9428 omx->request_perf_level(VIDC_NOMINAL);
9429 }
9430 if (omx->output_use_buffer && omxhdr->pBuffer &&
9431 vdec_msg->msgdata.output_frame.bufferaddr)
9432 memcpy ( omxhdr->pBuffer, (void *)
9433 ((unsigned long)vdec_msg->msgdata.output_frame.bufferaddr +
9434 (unsigned long)vdec_msg->msgdata.output_frame.offset),
9435 vdec_msg->msgdata.output_frame.len);
9436 } else {
9437 DEBUG_PRINT_ERROR("Invalid filled length = %u, buffer size = %u, prev_length = %u",
9438 (unsigned int)vdec_msg->msgdata.output_frame.len,
9439 omxhdr->nAllocLen, omx->prev_n_filled_len);
9440 omxhdr->nFilledLen = 0;
9441 }
9442
9443 omx->post_event ((unsigned long)omxhdr, vdec_msg->status_code,
9444 OMX_COMPONENT_GENERATE_FBD);
9445
9446 } else if (vdec_msg->msgdata.output_frame.flags & OMX_BUFFERFLAG_EOS) {
9447 omx->post_event ((unsigned long)NULL, vdec_msg->status_code,
9448 OMX_COMPONENT_GENERATE_EOS_DONE);
9449 } else {
9450 omx->post_event ((unsigned int)NULL, vdec_msg->status_code,
9451 OMX_COMPONENT_GENERATE_HARDWARE_ERROR);
9452 }
9453 break;
9454 case VDEC_MSG_EVT_CONFIG_CHANGED:
9455 DEBUG_PRINT_HIGH("Port settings changed");
9456 omx->m_reconfig_width = vdec_msg->msgdata.output_frame.picsize.frame_width;
9457 omx->m_reconfig_height = vdec_msg->msgdata.output_frame.picsize.frame_height;
9458 omx->post_event (OMX_CORE_OUTPUT_PORT_INDEX, OMX_IndexParamPortDefinition,
9459 OMX_COMPONENT_GENERATE_PORT_RECONFIG);
9460 if (!omx->m_need_turbo) {
9461 omx->request_perf_level(VIDC_NOMINAL);
9462 }
9463 break;
9464 default:
9465 break;
9466 }
9467 return rc;
9468 }
9469
empty_this_buffer_proxy_arbitrary(OMX_HANDLETYPE hComp,OMX_BUFFERHEADERTYPE * buffer)9470 OMX_ERRORTYPE omx_vdec::empty_this_buffer_proxy_arbitrary (
9471 OMX_HANDLETYPE hComp,
9472 OMX_BUFFERHEADERTYPE *buffer
9473 )
9474 {
9475 unsigned address,p2,id;
9476 DEBUG_PRINT_LOW("Empty this arbitrary");
9477
9478 if (buffer == NULL) {
9479 return OMX_ErrorBadParameter;
9480 }
9481 DEBUG_PRINT_LOW("ETBProxyArb: bufhdr = %p, bufhdr->pBuffer = %p", buffer, buffer->pBuffer);
9482 DEBUG_PRINT_LOW("ETBProxyArb: nFilledLen %u, flags %u, timestamp %lld",
9483 (unsigned int)buffer->nFilledLen, (unsigned int)buffer->nFlags, buffer->nTimeStamp);
9484
9485 /* return zero length and not an EOS buffer */
9486 /* return buffer if input flush in progress */
9487 if ((input_flush_progress == true) || ((buffer->nFilledLen == 0) &&
9488 ((buffer->nFlags & OMX_BUFFERFLAG_EOS) == 0))) {
9489 DEBUG_PRINT_HIGH("return zero legth buffer or flush in progress");
9490 m_cb.EmptyBufferDone (hComp,m_app_data,buffer);
9491 return OMX_ErrorNone;
9492 }
9493
9494 if (psource_frame == NULL) {
9495 DEBUG_PRINT_LOW("Set Buffer as source Buffer %p time stamp %lld",buffer,buffer->nTimeStamp);
9496 psource_frame = buffer;
9497 DEBUG_PRINT_LOW("Try to Push One Input Buffer ");
9498 push_input_buffer (hComp);
9499 } else {
9500 DEBUG_PRINT_LOW("Push the source buffer into pendingq %p",buffer);
9501 if (!m_input_pending_q.insert_entry((unsigned long)buffer, (unsigned)NULL,
9502 (unsigned)NULL)) {
9503 return OMX_ErrorBadParameter;
9504 }
9505 }
9506
9507 if (codec_config_flag && !(buffer->nFlags & OMX_BUFFERFLAG_CODECCONFIG)) {
9508 codec_config_flag = false;
9509 }
9510 return OMX_ErrorNone;
9511 }
9512
push_input_buffer(OMX_HANDLETYPE hComp)9513 OMX_ERRORTYPE omx_vdec::push_input_buffer (OMX_HANDLETYPE hComp)
9514 {
9515 unsigned long address,p2,id;
9516 OMX_ERRORTYPE ret = OMX_ErrorNone;
9517
9518 if (pdest_frame == NULL || psource_frame == NULL) {
9519 /*Check if we have a destination buffer*/
9520 if (pdest_frame == NULL) {
9521 DEBUG_PRINT_LOW("Get a Destination buffer from the queue");
9522 if (m_input_free_q.m_size) {
9523 m_input_free_q.pop_entry(&address,&p2,&id);
9524 pdest_frame = (OMX_BUFFERHEADERTYPE *)address;
9525 pdest_frame->nFilledLen = 0;
9526 pdest_frame->nTimeStamp = LLONG_MAX;
9527 DEBUG_PRINT_LOW("Address of Pmem Buffer %p",pdest_frame);
9528 }
9529 }
9530
9531 /*Check if we have a destination buffer*/
9532 if (psource_frame == NULL) {
9533 DEBUG_PRINT_LOW("Get a source buffer from the queue");
9534 if (m_input_pending_q.m_size) {
9535 m_input_pending_q.pop_entry(&address,&p2,&id);
9536 psource_frame = (OMX_BUFFERHEADERTYPE *)address;
9537 DEBUG_PRINT_LOW("Next source Buffer %p time stamp %lld",psource_frame,
9538 psource_frame->nTimeStamp);
9539 DEBUG_PRINT_LOW("Next source Buffer flag %u length %u",
9540 (unsigned int)psource_frame->nFlags, (unsigned int)psource_frame->nFilledLen);
9541
9542 }
9543 }
9544
9545 }
9546
9547 while ((pdest_frame != NULL) && (psource_frame != NULL)) {
9548 switch (codec_type_parse) {
9549 case CODEC_TYPE_MPEG4:
9550 case CODEC_TYPE_H263:
9551 case CODEC_TYPE_MPEG2:
9552 ret = push_input_sc_codec(hComp);
9553 break;
9554 case CODEC_TYPE_H264:
9555 ret = push_input_h264(hComp);
9556 break;
9557 case CODEC_TYPE_HEVC:
9558 ret = push_input_hevc(hComp);
9559 break;
9560 case CODEC_TYPE_VC1:
9561 ret = push_input_vc1(hComp);
9562 break;
9563 default:
9564 break;
9565 }
9566 if (ret != OMX_ErrorNone) {
9567 DEBUG_PRINT_ERROR("Pushing input Buffer Failed");
9568 omx_report_error ();
9569 break;
9570 }
9571 }
9572
9573 return ret;
9574 }
9575
push_input_sc_codec(OMX_HANDLETYPE hComp)9576 OMX_ERRORTYPE omx_vdec::push_input_sc_codec(OMX_HANDLETYPE hComp)
9577 {
9578 OMX_U32 partial_frame = 1;
9579 OMX_BOOL generate_ebd = OMX_TRUE;
9580 unsigned long address = 0, p2 = 0, id = 0;
9581
9582 DEBUG_PRINT_LOW("Start Parsing the bit stream address %p TimeStamp %lld",
9583 psource_frame,psource_frame->nTimeStamp);
9584 if (m_frame_parser.parse_sc_frame(psource_frame,
9585 pdest_frame,&partial_frame) == -1) {
9586 DEBUG_PRINT_ERROR("Error In Parsing Return Error");
9587 return OMX_ErrorBadParameter;
9588 }
9589
9590 if (partial_frame == 0) {
9591 DEBUG_PRINT_LOW("Frame size %u source %p frame count %d",
9592 (unsigned int)pdest_frame->nFilledLen,psource_frame,frame_count);
9593
9594
9595 DEBUG_PRINT_LOW("TimeStamp updated %lld", pdest_frame->nTimeStamp);
9596 /*First Parsed buffer will have only header Hence skip*/
9597 if (frame_count == 0) {
9598 DEBUG_PRINT_LOW("H263/MPEG4 Codec First Frame ");
9599
9600 if (codec_type_parse == CODEC_TYPE_MPEG4 ||
9601 codec_type_parse == CODEC_TYPE_DIVX) {
9602 mp4StreamType psBits;
9603 psBits.data = pdest_frame->pBuffer + pdest_frame->nOffset;
9604 psBits.numBytes = pdest_frame->nFilledLen;
9605 mp4_headerparser.parseHeader(&psBits);
9606 }
9607
9608 frame_count++;
9609 } else {
9610 pdest_frame->nFlags &= ~OMX_BUFFERFLAG_EOS;
9611 if (pdest_frame->nFilledLen) {
9612 /*Push the frame to the Decoder*/
9613 if (empty_this_buffer_proxy(hComp,pdest_frame) != OMX_ErrorNone) {
9614 return OMX_ErrorBadParameter;
9615 }
9616 frame_count++;
9617 pdest_frame = NULL;
9618
9619 if (m_input_free_q.m_size) {
9620 m_input_free_q.pop_entry(&address,&p2,&id);
9621 pdest_frame = (OMX_BUFFERHEADERTYPE *) address;
9622 pdest_frame->nFilledLen = 0;
9623 }
9624 } else if (!(psource_frame->nFlags & OMX_BUFFERFLAG_EOS)) {
9625 DEBUG_PRINT_ERROR("Zero len buffer return back to POOL");
9626 m_input_free_q.insert_entry((unsigned long) pdest_frame, (unsigned)NULL,
9627 (unsigned)NULL);
9628 pdest_frame = NULL;
9629 }
9630 }
9631 } else {
9632 DEBUG_PRINT_LOW("Not a Complete Frame %u", (unsigned int)pdest_frame->nFilledLen);
9633 /*Check if Destination Buffer is full*/
9634 if (pdest_frame->nAllocLen ==
9635 pdest_frame->nFilledLen + pdest_frame->nOffset) {
9636 DEBUG_PRINT_ERROR("ERROR:Frame Not found though Destination Filled");
9637 return OMX_ErrorStreamCorrupt;
9638 }
9639 }
9640
9641 if (psource_frame->nFilledLen == 0) {
9642 if (psource_frame->nFlags & OMX_BUFFERFLAG_EOS) {
9643 if (pdest_frame) {
9644 pdest_frame->nFlags |= psource_frame->nFlags;
9645 pdest_frame->nTimeStamp = psource_frame->nTimeStamp;
9646 DEBUG_PRINT_LOW("Frame Found start Decoding Size =%u TimeStamp = %lld",
9647 (unsigned int)pdest_frame->nFilledLen,pdest_frame->nTimeStamp);
9648 DEBUG_PRINT_LOW("Found a frame size = %u number = %d",
9649 (unsigned int)pdest_frame->nFilledLen,frame_count++);
9650 /*Push the frame to the Decoder*/
9651 if (empty_this_buffer_proxy(hComp,pdest_frame) != OMX_ErrorNone) {
9652 return OMX_ErrorBadParameter;
9653 }
9654 frame_count++;
9655 pdest_frame = NULL;
9656 } else {
9657 DEBUG_PRINT_LOW("Last frame in else dest addr") ;
9658 generate_ebd = OMX_FALSE;
9659 }
9660 }
9661 if (generate_ebd) {
9662 DEBUG_PRINT_LOW("Buffer Consumed return back to client %p",psource_frame);
9663 m_cb.EmptyBufferDone (hComp,m_app_data,psource_frame);
9664 psource_frame = NULL;
9665
9666 if (m_input_pending_q.m_size) {
9667 DEBUG_PRINT_LOW("Pull Next source Buffer %p",psource_frame);
9668 m_input_pending_q.pop_entry(&address,&p2,&id);
9669 psource_frame = (OMX_BUFFERHEADERTYPE *) address;
9670 DEBUG_PRINT_LOW("Next source Buffer %p time stamp %lld",psource_frame,
9671 psource_frame->nTimeStamp);
9672 DEBUG_PRINT_LOW("Next source Buffer flag %u length %u",
9673 (unsigned int)psource_frame->nFlags, (unsigned int)psource_frame->nFilledLen);
9674 }
9675 }
9676 }
9677 return OMX_ErrorNone;
9678 }
9679
push_input_h264(OMX_HANDLETYPE hComp)9680 OMX_ERRORTYPE omx_vdec::push_input_h264 (OMX_HANDLETYPE hComp)
9681 {
9682 OMX_U32 partial_frame = 1;
9683 unsigned long address = 0, p2 = 0, id = 0;
9684 OMX_BOOL isNewFrame = OMX_FALSE;
9685 OMX_BOOL generate_ebd = OMX_TRUE;
9686
9687 if (h264_scratch.pBuffer == NULL) {
9688 DEBUG_PRINT_ERROR("ERROR:H.264 Scratch Buffer not allocated");
9689 return OMX_ErrorBadParameter;
9690 }
9691 DEBUG_PRINT_LOW("Pending h264_scratch.nFilledLen %u "
9692 "look_ahead_nal %d", (unsigned int)h264_scratch.nFilledLen, look_ahead_nal);
9693 DEBUG_PRINT_LOW("Pending pdest_frame->nFilledLen %u",(unsigned int)pdest_frame->nFilledLen);
9694 if (h264_scratch.nFilledLen && look_ahead_nal) {
9695 look_ahead_nal = false;
9696 if ((pdest_frame->nAllocLen - pdest_frame->nFilledLen) >=
9697 h264_scratch.nFilledLen) {
9698 memcpy ((pdest_frame->pBuffer + pdest_frame->nFilledLen),
9699 h264_scratch.pBuffer,h264_scratch.nFilledLen);
9700 pdest_frame->nFilledLen += h264_scratch.nFilledLen;
9701 DEBUG_PRINT_LOW("Copy the previous NAL (h264 scratch) into Dest frame");
9702 h264_scratch.nFilledLen = 0;
9703 } else {
9704 DEBUG_PRINT_ERROR("Error:1: Destination buffer overflow for H264");
9705 return OMX_ErrorBadParameter;
9706 }
9707 }
9708
9709 /* If an empty input is queued with EOS, do not coalesce with the destination-frame yet, as this may result
9710 in EOS flag getting associated with the destination
9711 */
9712 if (!psource_frame->nFilledLen && (psource_frame->nFlags & OMX_BUFFERFLAG_EOS) &&
9713 pdest_frame->nFilledLen) {
9714 DEBUG_PRINT_HIGH("delay ETB for 'empty buffer with EOS'");
9715 generate_ebd = OMX_FALSE;
9716 }
9717
9718 if (nal_length == 0) {
9719 DEBUG_PRINT_LOW("Zero NAL, hence parse using start code");
9720 if (m_frame_parser.parse_sc_frame(psource_frame,
9721 &h264_scratch,&partial_frame) == -1) {
9722 DEBUG_PRINT_ERROR("Error In Parsing Return Error");
9723 return OMX_ErrorBadParameter;
9724 }
9725 } else {
9726 DEBUG_PRINT_LOW("Non-zero NAL length clip, hence parse with NAL size %d ",nal_length);
9727 if (m_frame_parser.parse_h264_nallength(psource_frame,
9728 &h264_scratch,&partial_frame) == -1) {
9729 DEBUG_PRINT_ERROR("Error In Parsing NAL size, Return Error");
9730 return OMX_ErrorBadParameter;
9731 }
9732 }
9733
9734 if (partial_frame == 0) {
9735 if (nal_count == 0 && h264_scratch.nFilledLen == 0) {
9736 DEBUG_PRINT_LOW("First NAL with Zero Length, hence Skip");
9737 nal_count++;
9738 h264_scratch.nTimeStamp = psource_frame->nTimeStamp;
9739 h264_scratch.nFlags = psource_frame->nFlags;
9740 } else {
9741 DEBUG_PRINT_LOW("Parsed New NAL Length = %u",(unsigned int)h264_scratch.nFilledLen);
9742 if (h264_scratch.nFilledLen) {
9743 h264_parser->parse_nal((OMX_U8*)h264_scratch.pBuffer, h264_scratch.nFilledLen,
9744 NALU_TYPE_SPS);
9745 #ifndef PROCESS_EXTRADATA_IN_OUTPUT_PORT
9746 if (client_extradata & OMX_TIMEINFO_EXTRADATA)
9747 h264_parser->parse_nal((OMX_U8*)h264_scratch.pBuffer,
9748 h264_scratch.nFilledLen, NALU_TYPE_SEI);
9749 else if (client_extradata & OMX_FRAMEINFO_EXTRADATA)
9750 // If timeinfo is present frame info from SEI is already processed
9751 h264_parser->parse_nal((OMX_U8*)h264_scratch.pBuffer,
9752 h264_scratch.nFilledLen, NALU_TYPE_SEI);
9753 #endif
9754 m_frame_parser.mutils->isNewFrame(&h264_scratch, 0, isNewFrame);
9755 nal_count++;
9756 if (VALID_TS(h264_last_au_ts) && !VALID_TS(pdest_frame->nTimeStamp)) {
9757 pdest_frame->nTimeStamp = h264_last_au_ts;
9758 pdest_frame->nFlags = h264_last_au_flags;
9759 #ifdef PANSCAN_HDLR
9760 if (client_extradata & OMX_FRAMEINFO_EXTRADATA)
9761 h264_parser->update_panscan_data(h264_last_au_ts);
9762 #endif
9763 }
9764 if (m_frame_parser.mutils->nalu_type == NALU_TYPE_NON_IDR ||
9765 m_frame_parser.mutils->nalu_type == NALU_TYPE_IDR) {
9766 h264_last_au_ts = h264_scratch.nTimeStamp;
9767 h264_last_au_flags = h264_scratch.nFlags;
9768 #ifndef PROCESS_EXTRADATA_IN_OUTPUT_PORT
9769 if (client_extradata & OMX_TIMEINFO_EXTRADATA) {
9770 OMX_S64 ts_in_sei = h264_parser->process_ts_with_sei_vui(h264_last_au_ts);
9771 if (!VALID_TS(h264_last_au_ts))
9772 h264_last_au_ts = ts_in_sei;
9773 }
9774 #endif
9775 } else
9776 h264_last_au_ts = LLONG_MAX;
9777 }
9778
9779 if (!isNewFrame) {
9780 if ( (pdest_frame->nAllocLen - pdest_frame->nFilledLen) >=
9781 h264_scratch.nFilledLen) {
9782 DEBUG_PRINT_LOW("Not a NewFrame Copy into Dest len %u",
9783 (unsigned int)h264_scratch.nFilledLen);
9784 memcpy ((pdest_frame->pBuffer + pdest_frame->nFilledLen),
9785 h264_scratch.pBuffer,h264_scratch.nFilledLen);
9786 pdest_frame->nFilledLen += h264_scratch.nFilledLen;
9787 if (m_frame_parser.mutils->nalu_type == NALU_TYPE_EOSEQ)
9788 pdest_frame->nFlags |= QOMX_VIDEO_BUFFERFLAG_EOSEQ;
9789 h264_scratch.nFilledLen = 0;
9790 } else {
9791 DEBUG_PRINT_LOW("Error:2: Destination buffer overflow for H264");
9792 return OMX_ErrorBadParameter;
9793 }
9794 } else if(h264_scratch.nFilledLen) {
9795 look_ahead_nal = true;
9796 DEBUG_PRINT_LOW("Frame Found start Decoding Size =%u TimeStamp = %llu",
9797 (unsigned int)pdest_frame->nFilledLen,pdest_frame->nTimeStamp);
9798 DEBUG_PRINT_LOW("Found a frame size = %u number = %d",
9799 (unsigned int)pdest_frame->nFilledLen,frame_count++);
9800
9801 if (pdest_frame->nFilledLen == 0) {
9802 DEBUG_PRINT_LOW("Copy the Current Frame since and push it");
9803 look_ahead_nal = false;
9804 if ( (pdest_frame->nAllocLen - pdest_frame->nFilledLen) >=
9805 h264_scratch.nFilledLen) {
9806 memcpy ((pdest_frame->pBuffer + pdest_frame->nFilledLen),
9807 h264_scratch.pBuffer,h264_scratch.nFilledLen);
9808 pdest_frame->nFilledLen += h264_scratch.nFilledLen;
9809 h264_scratch.nFilledLen = 0;
9810 } else {
9811 DEBUG_PRINT_ERROR("Error:3: Destination buffer overflow for H264");
9812 return OMX_ErrorBadParameter;
9813 }
9814 } else {
9815 if (psource_frame->nFilledLen || h264_scratch.nFilledLen) {
9816 DEBUG_PRINT_LOW("Reset the EOS Flag");
9817 pdest_frame->nFlags &= ~OMX_BUFFERFLAG_EOS;
9818 }
9819 /*Push the frame to the Decoder*/
9820 if (empty_this_buffer_proxy(hComp,pdest_frame) != OMX_ErrorNone) {
9821 return OMX_ErrorBadParameter;
9822 }
9823 //frame_count++;
9824 pdest_frame = NULL;
9825 if (m_input_free_q.m_size) {
9826 m_input_free_q.pop_entry(&address,&p2,&id);
9827 pdest_frame = (OMX_BUFFERHEADERTYPE *) address;
9828 DEBUG_PRINT_LOW("Pop the next pdest_buffer %p",pdest_frame);
9829 pdest_frame->nFilledLen = 0;
9830 pdest_frame->nFlags = 0;
9831 pdest_frame->nTimeStamp = LLONG_MAX;
9832 }
9833 }
9834 }
9835 }
9836 } else {
9837 DEBUG_PRINT_LOW("Not a Complete Frame, pdest_frame->nFilledLen %u", (unsigned int)pdest_frame->nFilledLen);
9838 /*Check if Destination Buffer is full*/
9839 if (h264_scratch.nAllocLen ==
9840 h264_scratch.nFilledLen + h264_scratch.nOffset) {
9841 DEBUG_PRINT_ERROR("ERROR: Frame Not found though Destination Filled");
9842 return OMX_ErrorStreamCorrupt;
9843 }
9844 }
9845
9846 if (!psource_frame->nFilledLen) {
9847 DEBUG_PRINT_LOW("Buffer Consumed return source %p back to client",psource_frame);
9848
9849 if (psource_frame->nFlags & OMX_BUFFERFLAG_EOS) {
9850 if (pdest_frame) {
9851 DEBUG_PRINT_LOW("EOS Reached Pass Last Buffer");
9852 if ( (pdest_frame->nAllocLen - pdest_frame->nFilledLen) >=
9853 h264_scratch.nFilledLen) {
9854 if(pdest_frame->nFilledLen == 0) {
9855 /* No residual frame from before, send whatever
9856 * we have left */
9857 memcpy((pdest_frame->pBuffer + pdest_frame->nFilledLen),
9858 h264_scratch.pBuffer, h264_scratch.nFilledLen);
9859 pdest_frame->nFilledLen += h264_scratch.nFilledLen;
9860 h264_scratch.nFilledLen = 0;
9861 pdest_frame->nTimeStamp = h264_scratch.nTimeStamp;
9862 } else {
9863 m_frame_parser.mutils->isNewFrame(&h264_scratch, 0, isNewFrame);
9864 if(!isNewFrame) {
9865 /* Have a residual frame, but we know that the
9866 * AU in this frame is belonging to whatever
9867 * frame we had left over. So append it */
9868 memcpy ((pdest_frame->pBuffer + pdest_frame->nFilledLen),
9869 h264_scratch.pBuffer,h264_scratch.nFilledLen);
9870 pdest_frame->nFilledLen += h264_scratch.nFilledLen;
9871 h264_scratch.nFilledLen = 0;
9872 if (h264_last_au_ts != LLONG_MAX)
9873 pdest_frame->nTimeStamp = h264_last_au_ts;
9874 } else {
9875 /* Completely new frame, let's just push what
9876 * we have now. The resulting EBD would trigger
9877 * another push */
9878 generate_ebd = OMX_FALSE;
9879 pdest_frame->nTimeStamp = h264_last_au_ts;
9880 h264_last_au_ts = h264_scratch.nTimeStamp;
9881 }
9882 }
9883 } else {
9884 DEBUG_PRINT_ERROR("ERROR:4: Destination buffer overflow for H264");
9885 return OMX_ErrorBadParameter;
9886 }
9887
9888 /* Iff we coalesced two buffers, inherit the flags of both bufs */
9889 if(generate_ebd == OMX_TRUE) {
9890 pdest_frame->nFlags = h264_scratch.nFlags | psource_frame->nFlags;
9891 }
9892
9893 DEBUG_PRINT_LOW("pdest_frame->nFilledLen =%u TimeStamp = %llu",
9894 (unsigned int)pdest_frame->nFilledLen,pdest_frame->nTimeStamp);
9895 DEBUG_PRINT_LOW("Push AU frame number %d to driver", frame_count++);
9896 #ifndef PROCESS_EXTRADATA_IN_OUTPUT_PORT
9897 if (client_extradata & OMX_TIMEINFO_EXTRADATA) {
9898 OMX_S64 ts_in_sei = h264_parser->process_ts_with_sei_vui(pdest_frame->nTimeStamp);
9899 if (!VALID_TS(pdest_frame->nTimeStamp))
9900 pdest_frame->nTimeStamp = ts_in_sei;
9901 }
9902 #endif
9903 /*Push the frame to the Decoder*/
9904 if (empty_this_buffer_proxy(hComp,pdest_frame) != OMX_ErrorNone) {
9905 return OMX_ErrorBadParameter;
9906 }
9907 frame_count++;
9908 pdest_frame = NULL;
9909 } else {
9910 DEBUG_PRINT_LOW("Last frame in else dest addr %p size %u",
9911 pdest_frame, (unsigned int)h264_scratch.nFilledLen);
9912 generate_ebd = OMX_FALSE;
9913 }
9914 }
9915 }
9916 if (generate_ebd && !psource_frame->nFilledLen) {
9917 m_cb.EmptyBufferDone (hComp,m_app_data,psource_frame);
9918 psource_frame = NULL;
9919 if (m_input_pending_q.m_size) {
9920 DEBUG_PRINT_LOW("Pull Next source Buffer %p",psource_frame);
9921 m_input_pending_q.pop_entry(&address,&p2,&id);
9922 psource_frame = (OMX_BUFFERHEADERTYPE *) address;
9923 DEBUG_PRINT_LOW("Next source Buffer flag %u src length %u",
9924 (unsigned int)psource_frame->nFlags, (unsigned int)psource_frame->nFilledLen);
9925 }
9926 }
9927 return OMX_ErrorNone;
9928 }
9929
copy_buffer(OMX_BUFFERHEADERTYPE * pDst,OMX_BUFFERHEADERTYPE * pSrc)9930 OMX_ERRORTYPE copy_buffer(OMX_BUFFERHEADERTYPE* pDst, OMX_BUFFERHEADERTYPE* pSrc)
9931 {
9932 OMX_ERRORTYPE rc = OMX_ErrorNone;
9933 if ((pDst->nAllocLen - pDst->nFilledLen) >= pSrc->nFilledLen) {
9934 memcpy((pDst->pBuffer + pDst->nFilledLen), pSrc->pBuffer, pSrc->nFilledLen);
9935 if (pDst->nTimeStamp == LLONG_MAX) {
9936 pDst->nTimeStamp = pSrc->nTimeStamp;
9937 DEBUG_PRINT_LOW("Assign Dst nTimeStamp = %lld", pDst->nTimeStamp);
9938 }
9939 pDst->nFilledLen += pSrc->nFilledLen;
9940 pSrc->nFilledLen = 0;
9941 } else {
9942 DEBUG_PRINT_ERROR("Error: Destination buffer overflow");
9943 rc = OMX_ErrorBadParameter;
9944 }
9945 return rc;
9946 }
9947
push_input_hevc(OMX_HANDLETYPE hComp)9948 OMX_ERRORTYPE omx_vdec::push_input_hevc(OMX_HANDLETYPE hComp)
9949 {
9950 OMX_U32 partial_frame = 1;
9951 unsigned long address,p2,id;
9952 OMX_BOOL isNewFrame = OMX_FALSE;
9953 OMX_BOOL generate_ebd = OMX_TRUE;
9954 OMX_ERRORTYPE rc = OMX_ErrorNone;
9955 if (h264_scratch.pBuffer == NULL) {
9956 DEBUG_PRINT_ERROR("ERROR:Hevc Scratch Buffer not allocated");
9957 return OMX_ErrorBadParameter;
9958 }
9959
9960 DEBUG_PRINT_LOW("h264_scratch.nFilledLen %u has look_ahead_nal %d \
9961 pdest_frame nFilledLen %u nTimeStamp %lld",
9962 (unsigned int)h264_scratch.nFilledLen, look_ahead_nal, (unsigned int)pdest_frame->nFilledLen, pdest_frame->nTimeStamp);
9963
9964 if (h264_scratch.nFilledLen && look_ahead_nal) {
9965 look_ahead_nal = false;
9966 rc = copy_buffer(pdest_frame, &h264_scratch);
9967 if (rc != OMX_ErrorNone) {
9968 return rc;
9969 }
9970 }
9971
9972 if (nal_length == 0) {
9973 if (m_frame_parser.parse_sc_frame(psource_frame,
9974 &h264_scratch,&partial_frame) == -1) {
9975 DEBUG_PRINT_ERROR("Error In Parsing Return Error");
9976 return OMX_ErrorBadParameter;
9977 }
9978 } else {
9979 DEBUG_PRINT_LOW("Non-zero NAL length clip, hence parse with NAL size %d",nal_length);
9980 if (m_frame_parser.parse_h264_nallength(psource_frame,
9981 &h264_scratch,&partial_frame) == -1) {
9982 DEBUG_PRINT_ERROR("Error In Parsing NAL size, Return Error");
9983 return OMX_ErrorBadParameter;
9984 }
9985 }
9986
9987 if (partial_frame == 0) {
9988 if (nal_count == 0 && h264_scratch.nFilledLen == 0) {
9989 DEBUG_PRINT_LOW("First NAL with Zero Length, hence Skip");
9990 nal_count++;
9991 h264_scratch.nTimeStamp = psource_frame->nTimeStamp;
9992 h264_scratch.nFlags = psource_frame->nFlags;
9993 } else {
9994 DEBUG_PRINT_LOW("Parsed New NAL Length = %u", (unsigned int)h264_scratch.nFilledLen);
9995 if (h264_scratch.nFilledLen) {
9996 m_hevc_utils.isNewFrame(&h264_scratch, 0, isNewFrame);
9997 nal_count++;
9998 }
9999
10000 if (!isNewFrame) {
10001 DEBUG_PRINT_LOW("Not a new frame, copy h264_scratch nFilledLen %u \
10002 nTimestamp %lld, pdest_frame nFilledLen %u nTimestamp %lld",
10003 (unsigned int)h264_scratch.nFilledLen, h264_scratch.nTimeStamp,
10004 (unsigned int)pdest_frame->nFilledLen, pdest_frame->nTimeStamp);
10005 rc = copy_buffer(pdest_frame, &h264_scratch);
10006 if (rc != OMX_ErrorNone) {
10007 return rc;
10008 }
10009 } else {
10010 look_ahead_nal = true;
10011 if (pdest_frame->nFilledLen == 0) {
10012 look_ahead_nal = false;
10013 DEBUG_PRINT_LOW("dest nation buffer empty, copy scratch buffer");
10014 rc = copy_buffer(pdest_frame, &h264_scratch);
10015 if (rc != OMX_ErrorNone) {
10016 return OMX_ErrorBadParameter;
10017 }
10018 } else {
10019 if (psource_frame->nFilledLen || h264_scratch.nFilledLen) {
10020 pdest_frame->nFlags &= ~OMX_BUFFERFLAG_EOS;
10021 }
10022 DEBUG_PRINT_LOW("FrameDetected # %d pdest_frame nFilledLen %u \
10023 nTimeStamp %lld, look_ahead_nal in h264_scratch \
10024 nFilledLen %u nTimeStamp %lld",
10025 frame_count++, (unsigned int)pdest_frame->nFilledLen,
10026 pdest_frame->nTimeStamp, (unsigned int)h264_scratch.nFilledLen,
10027 h264_scratch.nTimeStamp);
10028 if (empty_this_buffer_proxy(hComp, pdest_frame) != OMX_ErrorNone) {
10029 return OMX_ErrorBadParameter;
10030 }
10031 pdest_frame = NULL;
10032 if (m_input_free_q.m_size) {
10033 m_input_free_q.pop_entry(&address, &p2, &id);
10034 pdest_frame = (OMX_BUFFERHEADERTYPE *) address;
10035 DEBUG_PRINT_LOW("pop the next pdest_buffer %p", pdest_frame);
10036 pdest_frame->nFilledLen = 0;
10037 pdest_frame->nFlags = 0;
10038 pdest_frame->nTimeStamp = LLONG_MAX;
10039 }
10040 }
10041 }
10042 }
10043 } else {
10044 DEBUG_PRINT_LOW("psource_frame is partial nFilledLen %u nTimeStamp %lld, \
10045 pdest_frame nFilledLen %u nTimeStamp %lld, h264_scratch \
10046 nFilledLen %u nTimeStamp %lld",
10047 (unsigned int)psource_frame->nFilledLen, psource_frame->nTimeStamp,
10048 (unsigned int)pdest_frame->nFilledLen, pdest_frame->nTimeStamp,
10049 (unsigned int)h264_scratch.nFilledLen, h264_scratch.nTimeStamp);
10050
10051 if (h264_scratch.nAllocLen ==
10052 h264_scratch.nFilledLen + h264_scratch.nOffset) {
10053 DEBUG_PRINT_ERROR("ERROR: Frame Not found though Destination Filled");
10054 return OMX_ErrorStreamCorrupt;
10055 }
10056 }
10057
10058 if (!psource_frame->nFilledLen) {
10059 DEBUG_PRINT_LOW("Buffer Consumed return source %p back to client", psource_frame);
10060 if (psource_frame->nFlags & OMX_BUFFERFLAG_EOS) {
10061 if (pdest_frame) {
10062 DEBUG_PRINT_LOW("EOS Reached Pass Last Buffer");
10063 rc = copy_buffer(pdest_frame, &h264_scratch);
10064 if ( rc != OMX_ErrorNone ) {
10065 return rc;
10066 }
10067 pdest_frame->nTimeStamp = h264_scratch.nTimeStamp;
10068 pdest_frame->nFlags = h264_scratch.nFlags | psource_frame->nFlags;
10069 DEBUG_PRINT_LOW("Push EOS frame number:%d nFilledLen =%u TimeStamp = %lld",
10070 frame_count, (unsigned int)pdest_frame->nFilledLen, pdest_frame->nTimeStamp);
10071 if (empty_this_buffer_proxy(hComp, pdest_frame) != OMX_ErrorNone) {
10072 return OMX_ErrorBadParameter;
10073 }
10074 frame_count++;
10075 pdest_frame = NULL;
10076 } else {
10077 DEBUG_PRINT_LOW("Last frame in else dest addr %p size %u",
10078 pdest_frame, (unsigned int)h264_scratch.nFilledLen);
10079 generate_ebd = OMX_FALSE;
10080 }
10081 }
10082 }
10083
10084 if (generate_ebd && !psource_frame->nFilledLen) {
10085 m_cb.EmptyBufferDone (hComp, m_app_data, psource_frame);
10086 psource_frame = NULL;
10087 if (m_input_pending_q.m_size) {
10088 m_input_pending_q.pop_entry(&address, &p2, &id);
10089 psource_frame = (OMX_BUFFERHEADERTYPE *)address;
10090 DEBUG_PRINT_LOW("Next source Buffer flag %u nFilledLen %u, nTimeStamp %lld",
10091 (unsigned int)psource_frame->nFlags, (unsigned int)psource_frame->nFilledLen, psource_frame->nTimeStamp);
10092 }
10093 }
10094 return OMX_ErrorNone;
10095 }
10096
push_input_vc1(OMX_HANDLETYPE hComp)10097 OMX_ERRORTYPE omx_vdec::push_input_vc1(OMX_HANDLETYPE hComp)
10098 {
10099 OMX_U8 *buf, *pdest;
10100 OMX_U32 partial_frame = 1;
10101 OMX_U32 buf_len, dest_len;
10102
10103 if (first_frame == 0) {
10104 first_frame = 1;
10105 DEBUG_PRINT_LOW("First i/p buffer for VC1 arbitrary bytes");
10106 if (!m_vendor_config.pData) {
10107 DEBUG_PRINT_LOW("Check profile type in 1st source buffer");
10108 buf = psource_frame->pBuffer;
10109 buf_len = psource_frame->nFilledLen;
10110
10111 if ((*((OMX_U32 *) buf) & VC1_SP_MP_START_CODE_MASK) ==
10112 VC1_SP_MP_START_CODE) {
10113 m_vc1_profile = VC1_SP_MP_RCV;
10114 } else if (*((OMX_U32 *) buf) & VC1_AP_SEQ_START_CODE) {
10115 m_vc1_profile = VC1_AP;
10116 } else {
10117 DEBUG_PRINT_ERROR("Invalid sequence layer in first buffer");
10118 return OMX_ErrorStreamCorrupt;
10119 }
10120 } else {
10121 pdest = pdest_frame->pBuffer + pdest_frame->nFilledLen +
10122 pdest_frame->nOffset;
10123 dest_len = pdest_frame->nAllocLen - (pdest_frame->nFilledLen +
10124 pdest_frame->nOffset);
10125
10126 if (dest_len < m_vendor_config.nDataSize) {
10127 DEBUG_PRINT_ERROR("Destination buffer full");
10128 return OMX_ErrorBadParameter;
10129 } else {
10130 memcpy(pdest, m_vendor_config.pData, m_vendor_config.nDataSize);
10131 pdest_frame->nFilledLen += m_vendor_config.nDataSize;
10132 }
10133 }
10134 }
10135
10136 switch (m_vc1_profile) {
10137 case VC1_AP:
10138 DEBUG_PRINT_LOW("VC1 AP, hence parse using frame start code");
10139 if (push_input_sc_codec(hComp) != OMX_ErrorNone) {
10140 DEBUG_PRINT_ERROR("Error In Parsing VC1 AP start code");
10141 return OMX_ErrorBadParameter;
10142 }
10143 break;
10144
10145 case VC1_SP_MP_RCV:
10146 default:
10147 DEBUG_PRINT_ERROR("Unsupported VC1 profile in ArbitraryBytes Mode");
10148 return OMX_ErrorBadParameter;
10149 }
10150 return OMX_ErrorNone;
10151 }
10152
10153 #ifndef USE_ION
align_pmem_buffers(int pmem_fd,OMX_U32 buffer_size,OMX_U32 alignment)10154 bool omx_vdec::align_pmem_buffers(int pmem_fd, OMX_U32 buffer_size,
10155 OMX_U32 alignment)
10156 {
10157 struct pmem_allocation allocation;
10158 allocation.size = buffer_size;
10159 allocation.align = clip2(alignment);
10160 if (allocation.align < 4096) {
10161 allocation.align = 4096;
10162 }
10163 if (ioctl(pmem_fd, PMEM_ALLOCATE_ALIGNED, &allocation) < 0) {
10164 DEBUG_PRINT_ERROR("Aligment(%u) failed with pmem driver Sz(%lu)",
10165 allocation.align, allocation.size);
10166 return false;
10167 }
10168 return true;
10169 }
10170 #endif
10171 #ifdef USE_ION
alloc_map_ion_memory(OMX_U32 buffer_size,OMX_U32 alignment,struct ion_allocation_data * alloc_data,struct ion_fd_data * fd_data,int flag)10172 int omx_vdec::alloc_map_ion_memory(OMX_U32 buffer_size,
10173 OMX_U32 alignment, struct ion_allocation_data *alloc_data,
10174 struct ion_fd_data *fd_data, int flag)
10175 {
10176 int fd = -EINVAL;
10177 int rc = -EINVAL;
10178 int ion_dev_flag;
10179 struct vdec_ion ion_buf_info;
10180 if (!alloc_data || buffer_size <= 0 || !fd_data) {
10181 DEBUG_PRINT_ERROR("Invalid arguments to alloc_map_ion_memory");
10182 return -EINVAL;
10183 }
10184 ion_dev_flag = O_RDONLY;
10185 fd = open (MEM_DEVICE, ion_dev_flag);
10186 if (fd < 0) {
10187 DEBUG_PRINT_ERROR("opening ion device failed with fd = %d", fd);
10188 return fd;
10189 }
10190
10191 alloc_data->flags = flag;
10192 alloc_data->len = buffer_size;
10193 alloc_data->align = clip2(alignment);
10194 if (alloc_data->align < 4096) {
10195 alloc_data->align = 4096;
10196 }
10197
10198 alloc_data->heap_id_mask = ION_HEAP(ION_IOMMU_HEAP_ID);
10199 if (secure_mode && (alloc_data->flags & ION_SECURE)) {
10200 alloc_data->heap_id_mask = ION_HEAP(MEM_HEAP_ID);
10201 }
10202
10203 /* Use secure display cma heap for obvious reasons. */
10204 if (alloc_data->flags & ION_FLAG_CP_BITSTREAM) {
10205 alloc_data->heap_id_mask |= ION_HEAP(ION_SECURE_DISPLAY_HEAP_ID);
10206 }
10207
10208 rc = ioctl(fd,ION_IOC_ALLOC,alloc_data);
10209 if (rc || !alloc_data->handle) {
10210 DEBUG_PRINT_ERROR("ION ALLOC memory failed");
10211 alloc_data->handle = 0;
10212 close(fd);
10213 fd = -ENOMEM;
10214 return fd;
10215 }
10216 fd_data->handle = alloc_data->handle;
10217 rc = ioctl(fd,ION_IOC_MAP,fd_data);
10218 if (rc) {
10219 DEBUG_PRINT_ERROR("ION MAP failed ");
10220 ion_buf_info.ion_alloc_data = *alloc_data;
10221 ion_buf_info.ion_device_fd = fd;
10222 ion_buf_info.fd_ion_data = *fd_data;
10223 free_ion_memory(&ion_buf_info);
10224 fd_data->fd =-1;
10225 fd = -ENOMEM;
10226 }
10227
10228 return fd;
10229 }
10230
free_ion_memory(struct vdec_ion * buf_ion_info)10231 void omx_vdec::free_ion_memory(struct vdec_ion *buf_ion_info)
10232 {
10233
10234 if (!buf_ion_info) {
10235 DEBUG_PRINT_ERROR("ION: free called with invalid fd/allocdata");
10236 return;
10237 }
10238 if (ioctl(buf_ion_info->ion_device_fd,ION_IOC_FREE,
10239 &buf_ion_info->ion_alloc_data.handle)) {
10240 DEBUG_PRINT_ERROR("ION: free failed" );
10241 }
10242 close(buf_ion_info->ion_device_fd);
10243 buf_ion_info->ion_device_fd = -1;
10244 buf_ion_info->ion_alloc_data.handle = 0;
10245 buf_ion_info->fd_ion_data.fd = -1;
10246 }
10247 #endif
free_output_buffer_header()10248 void omx_vdec::free_output_buffer_header()
10249 {
10250 DEBUG_PRINT_HIGH("ALL output buffers are freed/released");
10251 output_use_buffer = false;
10252 ouput_egl_buffers = false;
10253
10254 if (m_out_mem_ptr) {
10255 free (m_out_mem_ptr);
10256 m_out_mem_ptr = NULL;
10257 }
10258
10259 if (m_platform_list) {
10260 free(m_platform_list);
10261 m_platform_list = NULL;
10262 }
10263
10264 if (drv_ctx.ptr_respbuffer) {
10265 free (drv_ctx.ptr_respbuffer);
10266 drv_ctx.ptr_respbuffer = NULL;
10267 }
10268 if (drv_ctx.ptr_outputbuffer) {
10269 free (drv_ctx.ptr_outputbuffer);
10270 drv_ctx.ptr_outputbuffer = NULL;
10271 }
10272 #ifdef USE_ION
10273 if (drv_ctx.op_buf_ion_info) {
10274 DEBUG_PRINT_LOW("Free o/p ion context");
10275 free(drv_ctx.op_buf_ion_info);
10276 drv_ctx.op_buf_ion_info = NULL;
10277 }
10278 #endif
10279 free(drv_ctx.op_buf_map_info);
10280 drv_ctx.op_buf_map_info = NULL;
10281 buf_ref_remove();
10282 }
10283
free_input_buffer_header()10284 void omx_vdec::free_input_buffer_header()
10285 {
10286 input_use_buffer = false;
10287 if (arbitrary_bytes) {
10288 if (m_inp_heap_ptr) {
10289 DEBUG_PRINT_LOW("Free input Heap Pointer");
10290 free (m_inp_heap_ptr);
10291 m_inp_heap_ptr = NULL;
10292 }
10293
10294 if (m_phdr_pmem_ptr) {
10295 DEBUG_PRINT_LOW("Free input pmem header Pointer");
10296 free (m_phdr_pmem_ptr);
10297 m_phdr_pmem_ptr = NULL;
10298 }
10299 }
10300 if (m_inp_mem_ptr) {
10301 DEBUG_PRINT_LOW("Free input pmem Pointer area");
10302 free (m_inp_mem_ptr);
10303 m_inp_mem_ptr = NULL;
10304 }
10305 /* We just freed all the buffer headers, every thing in m_input_free_q,
10306 * m_input_pending_q, pdest_frame, and psource_frame is now invalid */
10307 while (m_input_free_q.m_size) {
10308 unsigned long address, p2, id;
10309 m_input_free_q.pop_entry(&address, &p2, &id);
10310 }
10311 while (m_input_pending_q.m_size) {
10312 unsigned long address, p2, id;
10313 m_input_pending_q.pop_entry(&address, &p2, &id);
10314 }
10315 pdest_frame = NULL;
10316 psource_frame = NULL;
10317 if (drv_ctx.ptr_inputbuffer) {
10318 DEBUG_PRINT_LOW("Free Driver Context pointer");
10319 free (drv_ctx.ptr_inputbuffer);
10320 drv_ctx.ptr_inputbuffer = NULL;
10321 }
10322 #ifdef USE_ION
10323 if (drv_ctx.ip_buf_ion_info) {
10324 DEBUG_PRINT_LOW("Free ion context");
10325 free(drv_ctx.ip_buf_ion_info);
10326 drv_ctx.ip_buf_ion_info = NULL;
10327 }
10328 #endif
10329 }
10330
free_output_extradata_buffer_header()10331 void omx_vdec::free_output_extradata_buffer_header() {
10332 client_extradata = false;
10333 if (m_client_output_extradata_mem_ptr) {
10334 DEBUG_PRINT_LOW("Free extradata pmem Pointer area");
10335 free(m_client_output_extradata_mem_ptr);
10336 m_client_output_extradata_mem_ptr = NULL;
10337 }
10338 }
10339
stream_off(OMX_U32 port)10340 int omx_vdec::stream_off(OMX_U32 port)
10341 {
10342 enum v4l2_buf_type btype;
10343 int rc = 0;
10344 enum v4l2_ports v4l2_port = OUTPUT_PORT;
10345
10346 if (port == OMX_CORE_INPUT_PORT_INDEX) {
10347 btype = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
10348 v4l2_port = OUTPUT_PORT;
10349 } else if (port == OMX_CORE_OUTPUT_PORT_INDEX) {
10350 btype = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
10351 v4l2_port = CAPTURE_PORT;
10352 } else if (port == OMX_ALL) {
10353 int rc_input = stream_off(OMX_CORE_INPUT_PORT_INDEX);
10354 int rc_output = stream_off(OMX_CORE_OUTPUT_PORT_INDEX);
10355
10356 if (!rc_input)
10357 return rc_input;
10358 else
10359 return rc_output;
10360 }
10361
10362 if (!streaming[v4l2_port]) {
10363 // already streamed off, warn and move on
10364 DEBUG_PRINT_HIGH("Warning: Attempting to stream off on %d port,"
10365 " which is already streamed off", v4l2_port);
10366 return 0;
10367 }
10368
10369 DEBUG_PRINT_HIGH("Streaming off %d port", v4l2_port);
10370
10371 rc = ioctl(drv_ctx.video_driver_fd, VIDIOC_STREAMOFF, &btype);
10372 if (rc) {
10373 /*TODO: How to handle this case */
10374 DEBUG_PRINT_ERROR("Failed to call streamoff on %d Port", v4l2_port);
10375 } else {
10376 streaming[v4l2_port] = false;
10377 }
10378
10379 return rc;
10380 }
10381
get_buffer_req(vdec_allocatorproperty * buffer_prop)10382 OMX_ERRORTYPE omx_vdec::get_buffer_req(vdec_allocatorproperty *buffer_prop)
10383 {
10384 OMX_ERRORTYPE eRet = OMX_ErrorNone;
10385 struct v4l2_requestbuffers bufreq;
10386 unsigned int buf_size = 0, extra_data_size = 0, default_extra_data_size = 0;
10387 unsigned int final_extra_data_size = 0;
10388 struct v4l2_format fmt;
10389 int ret = 0;
10390 DEBUG_PRINT_LOW("GetBufReq IN: ActCnt(%d) Size(%u)",
10391 buffer_prop->actualcount, (unsigned int)buffer_prop->buffer_size);
10392 bufreq.memory = V4L2_MEMORY_USERPTR;
10393 bufreq.count = 1;
10394 if (buffer_prop->buffer_type == VDEC_BUFFER_TYPE_INPUT) {
10395 bufreq.type=V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
10396 fmt.type =V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
10397 fmt.fmt.pix_mp.pixelformat = output_capability;
10398 } else if (buffer_prop->buffer_type == VDEC_BUFFER_TYPE_OUTPUT) {
10399 bufreq.type=V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
10400 fmt.type =V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
10401 fmt.fmt.pix_mp.pixelformat = capture_capability;
10402 } else {
10403 eRet = OMX_ErrorBadParameter;
10404 }
10405 if (eRet==OMX_ErrorNone) {
10406 ret = ioctl(drv_ctx.video_driver_fd,VIDIOC_REQBUFS, &bufreq);
10407 }
10408 if (ret) {
10409 DEBUG_PRINT_ERROR("Requesting buffer requirements failed");
10410 /*TODO: How to handle this case */
10411 eRet = OMX_ErrorInsufficientResources;
10412 return eRet;
10413 } else {
10414 bool is_res_1080p_or_below = (drv_ctx.video_resolution.frame_width <= 1920 &&
10415 drv_ctx.video_resolution.frame_height <= 1088) ||
10416 (drv_ctx.video_resolution.frame_height <= 1088 &&
10417 drv_ctx.video_resolution.frame_width <= 1920);
10418
10419 int fps = drv_ctx.frame_rate.fps_numerator / (float)drv_ctx.frame_rate.fps_denominator;
10420 bool fps_above_180 = (fps >= 180 || operating_frame_rate >= 180) ? true : false;
10421 bool increase_output = (buffer_prop->buffer_type == VDEC_BUFFER_TYPE_OUTPUT) && (bufreq.count >= 16);
10422
10423 if (increase_output && fps_above_180 &&
10424 output_capability == V4L2_PIX_FMT_H264 &&
10425 is_res_1080p_or_below) {
10426 m_need_turbo |= TURBO_MODE_HIGH_FPS;
10427 DEBUG_PRINT_LOW("High fps - fps = %d operating_rate = %d", fps, operating_frame_rate);
10428 DEBUG_PRINT_LOW("getbufreq[output]: Increase buffer count (%d) to (%d) to support high fps",
10429 bufreq.count, bufreq.count + 10);
10430 bufreq.count += 10;
10431 ret = ioctl(drv_ctx.video_driver_fd,VIDIOC_REQBUFS, &bufreq);
10432 if (ret) {
10433 DEBUG_PRINT_ERROR("(Failed to set updated buffer count to driver");
10434 eRet = OMX_ErrorInsufficientResources;
10435 return eRet;
10436 }
10437 DEBUG_PRINT_LOW("new buf count = %d set to driver", bufreq.count);
10438 request_perf_level(VIDC_TURBO);
10439 }
10440
10441 buffer_prop->actualcount = bufreq.count;
10442 buffer_prop->mincount = bufreq.count;
10443 DEBUG_PRINT_HIGH("Count = %d",bufreq.count);
10444 }
10445 DEBUG_PRINT_LOW("GetBufReq IN: ActCnt(%d) Size(%u)",
10446 buffer_prop->actualcount, (unsigned int)buffer_prop->buffer_size);
10447
10448 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
10449
10450 if (fmt.type == V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE)
10451 drv_ctx.num_planes = fmt.fmt.pix_mp.num_planes;
10452 DEBUG_PRINT_HIGH("Buffer Size = %d",fmt.fmt.pix_mp.plane_fmt[0].sizeimage);
10453
10454 if (ret) {
10455 /*TODO: How to handle this case */
10456 DEBUG_PRINT_ERROR("Requesting buffer requirements failed");
10457 eRet = OMX_ErrorInsufficientResources;
10458 } else {
10459 int extra_idx = 0;
10460
10461 eRet = is_video_session_supported();
10462 if (eRet)
10463 return eRet;
10464
10465 buffer_prop->buffer_size = fmt.fmt.pix_mp.plane_fmt[0].sizeimage;
10466 buf_size = buffer_prop->buffer_size;
10467 extra_idx = EXTRADATA_IDX(drv_ctx.num_planes);
10468 if (extra_idx && (extra_idx < VIDEO_MAX_PLANES)) {
10469 extra_data_size = fmt.fmt.pix_mp.plane_fmt[extra_idx].sizeimage;
10470 } else if (extra_idx >= VIDEO_MAX_PLANES) {
10471 DEBUG_PRINT_ERROR("Extradata index is more than allowed: %d", extra_idx);
10472 return OMX_ErrorBadParameter;
10473 }
10474
10475 default_extra_data_size = VENUS_EXTRADATA_SIZE(
10476 drv_ctx.video_resolution.frame_height,
10477 drv_ctx.video_resolution.frame_width);
10478 final_extra_data_size = extra_data_size > default_extra_data_size ?
10479 extra_data_size : default_extra_data_size;
10480
10481 final_extra_data_size = (final_extra_data_size + buffer_prop->alignment - 1) &
10482 (~(buffer_prop->alignment - 1));
10483
10484 drv_ctx.extradata_info.size = buffer_prop->actualcount * final_extra_data_size;
10485 drv_ctx.extradata_info.count = buffer_prop->actualcount;
10486 drv_ctx.extradata_info.buffer_size = final_extra_data_size;
10487 buf_size = (buf_size + buffer_prop->alignment - 1)&(~(buffer_prop->alignment - 1));
10488 DEBUG_PRINT_LOW("GetBufReq UPDATE: ActCnt(%d) Size(%u) BufSize(%d)",
10489 buffer_prop->actualcount, (unsigned int)buffer_prop->buffer_size, buf_size);
10490 if (extra_data_size)
10491 DEBUG_PRINT_LOW("GetBufReq UPDATE: extradata: TotalSize(%d) BufferSize(%lu)",
10492 drv_ctx.extradata_info.size, drv_ctx.extradata_info.buffer_size);
10493
10494 if (in_reconfig) // BufReq will be set to driver when port is disabled
10495 buffer_prop->buffer_size = buf_size;
10496 else if (buf_size != buffer_prop->buffer_size) {
10497 buffer_prop->buffer_size = buf_size;
10498 eRet = set_buffer_req(buffer_prop);
10499 }
10500 }
10501 DEBUG_PRINT_LOW("GetBufReq OUT: ActCnt(%d) Size(%u)",
10502 buffer_prop->actualcount, (unsigned int)buffer_prop->buffer_size);
10503 return eRet;
10504 }
10505
set_buffer_req(vdec_allocatorproperty * buffer_prop)10506 OMX_ERRORTYPE omx_vdec::set_buffer_req(vdec_allocatorproperty *buffer_prop)
10507 {
10508 OMX_ERRORTYPE eRet = OMX_ErrorNone;
10509 unsigned buf_size = 0;
10510 struct v4l2_format fmt, c_fmt;
10511 struct v4l2_requestbuffers bufreq;
10512 int ret = 0;
10513 DEBUG_PRINT_LOW("SetBufReq IN: ActCnt(%d) Size(%u)",
10514 buffer_prop->actualcount, (unsigned int)buffer_prop->buffer_size);
10515 buf_size = (buffer_prop->buffer_size + buffer_prop->alignment - 1)&(~(buffer_prop->alignment - 1));
10516 if (buf_size != buffer_prop->buffer_size) {
10517 DEBUG_PRINT_ERROR("Buffer size alignment error: Requested(%u) Required(%d)",
10518 (unsigned int)buffer_prop->buffer_size, buf_size);
10519 eRet = OMX_ErrorBadParameter;
10520 } else {
10521 memset(&fmt, 0x0, sizeof(struct v4l2_format));
10522 memset(&c_fmt, 0x0, sizeof(struct v4l2_format));
10523 fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
10524 fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
10525 fmt.fmt.pix_mp.plane_fmt[0].sizeimage = buf_size;
10526
10527 if (buffer_prop->buffer_type == VDEC_BUFFER_TYPE_INPUT) {
10528 fmt.type =V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
10529 fmt.fmt.pix_mp.pixelformat = output_capability;
10530 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
10531 } else if (buffer_prop->buffer_type == VDEC_BUFFER_TYPE_OUTPUT) {
10532 c_fmt.type =V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
10533 c_fmt.fmt.pix_mp.pixelformat = capture_capability;
10534 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &c_fmt);
10535 c_fmt.fmt.pix_mp.plane_fmt[0].sizeimage = buf_size;
10536 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &c_fmt);
10537 } else {
10538 eRet = OMX_ErrorBadParameter;
10539 }
10540
10541 if (ret) {
10542 /*TODO: How to handle this case */
10543 DEBUG_PRINT_ERROR("Setting buffer requirements (format) failed %d", ret);
10544 eRet = OMX_ErrorInsufficientResources;
10545 }
10546
10547 bufreq.memory = V4L2_MEMORY_USERPTR;
10548 bufreq.count = buffer_prop->actualcount;
10549 if (buffer_prop->buffer_type == VDEC_BUFFER_TYPE_INPUT) {
10550 bufreq.type=V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
10551 } else if (buffer_prop->buffer_type == VDEC_BUFFER_TYPE_OUTPUT) {
10552 bufreq.type=V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
10553 } else {
10554 eRet = OMX_ErrorBadParameter;
10555 }
10556
10557 if (eRet==OMX_ErrorNone) {
10558 ret = ioctl(drv_ctx.video_driver_fd,VIDIOC_REQBUFS, &bufreq);
10559 }
10560
10561 if (ret) {
10562 DEBUG_PRINT_ERROR("Setting buffer requirements (reqbufs) failed %d", ret);
10563 /*TODO: How to handle this case */
10564 eRet = OMX_ErrorInsufficientResources;
10565 } else if (bufreq.count < buffer_prop->actualcount) {
10566 DEBUG_PRINT_ERROR("Driver refused to change the number of buffers"
10567 " on v4l2 port %d to %d (prefers %d)", bufreq.type,
10568 buffer_prop->actualcount, bufreq.count);
10569 eRet = OMX_ErrorInsufficientResources;
10570 } else {
10571 if (!client_buffers.update_buffer_req()) {
10572 DEBUG_PRINT_ERROR("Setting c2D buffer requirements failed");
10573 eRet = OMX_ErrorInsufficientResources;
10574 }
10575 }
10576 }
10577 return eRet;
10578 }
10579
update_picture_resolution()10580 OMX_ERRORTYPE omx_vdec::update_picture_resolution()
10581 {
10582 OMX_ERRORTYPE eRet = OMX_ErrorNone;
10583 return eRet;
10584 }
10585
update_portdef(OMX_PARAM_PORTDEFINITIONTYPE * portDefn)10586 OMX_ERRORTYPE omx_vdec::update_portdef(OMX_PARAM_PORTDEFINITIONTYPE *portDefn)
10587 {
10588 OMX_ERRORTYPE eRet = OMX_ErrorNone;
10589 struct v4l2_format fmt;
10590 if (!portDefn) {
10591 return OMX_ErrorBadParameter;
10592 }
10593 DEBUG_PRINT_LOW("omx_vdec::update_portdef");
10594 portDefn->nVersion.nVersion = OMX_SPEC_VERSION;
10595 portDefn->nSize = sizeof(OMX_PARAM_PORTDEFINITIONTYPE);
10596 portDefn->eDomain = OMX_PortDomainVideo;
10597 memset(&fmt, 0x0, sizeof(struct v4l2_format));
10598 if (0 == portDefn->nPortIndex) {
10599 portDefn->eDir = OMX_DirInput;
10600 portDefn->nBufferCountActual = drv_ctx.ip_buf.actualcount;
10601 portDefn->nBufferCountMin = drv_ctx.ip_buf.mincount;
10602 portDefn->nBufferSize = drv_ctx.ip_buf.buffer_size;
10603 portDefn->format.video.eColorFormat = OMX_COLOR_FormatUnused;
10604 portDefn->format.video.eCompressionFormat = eCompressionFormat;
10605 //for input port, always report the fps value set by client,
10606 //to distinguish whether client got valid fps from parser.
10607 portDefn->format.video.xFramerate = m_fps_received;
10608 portDefn->bEnabled = m_inp_bEnabled;
10609 portDefn->bPopulated = m_inp_bPopulated;
10610
10611 fmt.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
10612 fmt.fmt.pix_mp.pixelformat = output_capability;
10613 ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
10614 } else if (1 == portDefn->nPortIndex) {
10615 unsigned int buf_size = 0;
10616 int ret = 0;
10617 if (in_reconfig && !is_down_scalar_enabled) {
10618 fmt.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
10619 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
10620 fmt.fmt.pix_mp.pixelformat = capture_capability;
10621 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
10622 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
10623 }
10624
10625 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
10626 fmt.fmt.pix_mp.pixelformat = capture_capability;
10627 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
10628 if (ret) {
10629 DEBUG_PRINT_ERROR("Get Resolution failed");
10630 return OMX_ErrorHardware;
10631 }
10632 drv_ctx.op_buf.buffer_size = fmt.fmt.pix_mp.plane_fmt[0].sizeimage;
10633 if (!client_buffers.update_buffer_req()) {
10634 DEBUG_PRINT_ERROR("client_buffers.update_buffer_req Failed");
10635 return OMX_ErrorHardware;
10636 }
10637
10638 if (!client_buffers.get_buffer_req(buf_size)) {
10639 DEBUG_PRINT_ERROR("update buffer requirements");
10640 return OMX_ErrorHardware;
10641 }
10642 portDefn->nBufferSize = buf_size;
10643 portDefn->eDir = OMX_DirOutput;
10644 portDefn->nBufferCountActual = drv_ctx.op_buf.actualcount;
10645 portDefn->nBufferCountMin = drv_ctx.op_buf.mincount;
10646 portDefn->format.video.eCompressionFormat = OMX_VIDEO_CodingUnused;
10647 if (drv_ctx.frame_rate.fps_denominator > 0)
10648 portDefn->format.video.xFramerate = (drv_ctx.frame_rate.fps_numerator /
10649 drv_ctx.frame_rate.fps_denominator) << 16; //Q16 format
10650 else {
10651 DEBUG_PRINT_ERROR("Error: Divide by zero");
10652 return OMX_ErrorBadParameter;
10653 }
10654 portDefn->bEnabled = m_out_bEnabled;
10655 portDefn->bPopulated = m_out_bPopulated;
10656 if (!client_buffers.get_color_format(portDefn->format.video.eColorFormat)) {
10657 DEBUG_PRINT_ERROR("Error in getting color format");
10658 return OMX_ErrorHardware;
10659 }
10660 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
10661 fmt.fmt.pix_mp.pixelformat = capture_capability;
10662 } else if (OMX_CORE_OUTPUT_EXTRADATA_INDEX == portDefn->nPortIndex) {
10663 portDefn->nBufferSize = m_client_out_extradata_info.getSize();
10664 portDefn->nBufferCountMin = MIN_NUM_INPUT_OUTPUT_EXTRADATA_BUFFERS;
10665 portDefn->nBufferCountActual = MIN_NUM_INPUT_OUTPUT_EXTRADATA_BUFFERS;
10666 portDefn->eDir = OMX_DirOutput;
10667 } else {
10668 portDefn->eDir = OMX_DirMax;
10669 DEBUG_PRINT_LOW(" get_parameter: Bad Port idx %d",
10670 (int)portDefn->nPortIndex);
10671 eRet = OMX_ErrorBadPortIndex;
10672 }
10673 update_resolution(fmt.fmt.pix_mp.width, fmt.fmt.pix_mp.height,
10674 fmt.fmt.pix_mp.plane_fmt[0].bytesperline, fmt.fmt.pix_mp.plane_fmt[0].reserved[0]);
10675
10676 portDefn->format.video.nFrameHeight = drv_ctx.video_resolution.frame_height;
10677 portDefn->format.video.nFrameWidth = drv_ctx.video_resolution.frame_width;
10678 portDefn->format.video.nStride = drv_ctx.video_resolution.stride;
10679 portDefn->format.video.nSliceHeight = drv_ctx.video_resolution.scan_lines;
10680
10681 if ((portDefn->format.video.eColorFormat == OMX_COLOR_FormatYUV420Planar) ||
10682 (portDefn->format.video.eColorFormat == OMX_COLOR_FormatYUV420SemiPlanar)) {
10683 portDefn->format.video.nStride = ALIGN(drv_ctx.video_resolution.frame_width, 16);
10684 portDefn->format.video.nSliceHeight = drv_ctx.video_resolution.frame_height;
10685 }
10686 DEBUG_PRINT_HIGH("update_portdef(%u): Width = %u Height = %u Stride = %d "
10687 "SliceHeight = %u eColorFormat = %d nBufSize %u nBufCnt %u",
10688 (unsigned int)portDefn->nPortIndex,
10689 (unsigned int)portDefn->format.video.nFrameWidth,
10690 (unsigned int)portDefn->format.video.nFrameHeight,
10691 (int)portDefn->format.video.nStride,
10692 (unsigned int)portDefn->format.video.nSliceHeight,
10693 (unsigned int)portDefn->format.video.eColorFormat,
10694 (unsigned int)portDefn->nBufferSize,
10695 (unsigned int)portDefn->nBufferCountActual);
10696
10697 return eRet;
10698 }
10699
allocate_output_headers()10700 OMX_ERRORTYPE omx_vdec::allocate_output_headers()
10701 {
10702 OMX_ERRORTYPE eRet = OMX_ErrorNone;
10703 OMX_BUFFERHEADERTYPE *bufHdr = NULL;
10704 unsigned i = 0;
10705
10706 if (!m_out_mem_ptr) {
10707 DEBUG_PRINT_HIGH("Use o/p buffer case - Header List allocation");
10708 int nBufHdrSize = 0;
10709 int nPlatformEntrySize = 0;
10710 int nPlatformListSize = 0;
10711 int nPMEMInfoSize = 0;
10712 OMX_QCOM_PLATFORM_PRIVATE_LIST *pPlatformList;
10713 OMX_QCOM_PLATFORM_PRIVATE_ENTRY *pPlatformEntry;
10714 OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *pPMEMInfo;
10715
10716 DEBUG_PRINT_LOW("Setting First Output Buffer(%d)",
10717 drv_ctx.op_buf.actualcount);
10718 nBufHdrSize = drv_ctx.op_buf.actualcount *
10719 sizeof(OMX_BUFFERHEADERTYPE);
10720
10721 nPMEMInfoSize = drv_ctx.op_buf.actualcount *
10722 sizeof(OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO);
10723 nPlatformListSize = drv_ctx.op_buf.actualcount *
10724 sizeof(OMX_QCOM_PLATFORM_PRIVATE_LIST);
10725 nPlatformEntrySize = drv_ctx.op_buf.actualcount *
10726 sizeof(OMX_QCOM_PLATFORM_PRIVATE_ENTRY);
10727
10728 DEBUG_PRINT_LOW("TotalBufHdr %d BufHdrSize %u PMEM %d PL %d",nBufHdrSize,
10729 (unsigned int)sizeof(OMX_BUFFERHEADERTYPE),
10730 nPMEMInfoSize,
10731 nPlatformListSize);
10732 DEBUG_PRINT_LOW("PE %d bmSize % " PRId64 , nPlatformEntrySize,
10733 m_out_bm_count);
10734 m_out_mem_ptr = (OMX_BUFFERHEADERTYPE *)calloc(nBufHdrSize,1);
10735 // Alloc mem for platform specific info
10736 char *pPtr=NULL;
10737 pPtr = (char*) calloc(nPlatformListSize + nPlatformEntrySize +
10738 nPMEMInfoSize,1);
10739 drv_ctx.ptr_outputbuffer = (struct vdec_bufferpayload *) \
10740 calloc (sizeof(struct vdec_bufferpayload),
10741 drv_ctx.op_buf.actualcount);
10742 drv_ctx.ptr_respbuffer = (struct vdec_output_frameinfo *)\
10743 calloc (sizeof (struct vdec_output_frameinfo),
10744 drv_ctx.op_buf.actualcount);
10745 if (!drv_ctx.ptr_outputbuffer || !drv_ctx.ptr_respbuffer) {
10746 DEBUG_PRINT_ERROR("Failed to alloc drv_ctx.ptr_outputbuffer or drv_ctx.ptr_respbuffer");
10747 return OMX_ErrorInsufficientResources;
10748 }
10749
10750 #ifdef USE_ION
10751 drv_ctx.op_buf_ion_info = (struct vdec_ion * ) \
10752 calloc (sizeof(struct vdec_ion),drv_ctx.op_buf.actualcount);
10753 if (!drv_ctx.op_buf_ion_info) {
10754 DEBUG_PRINT_ERROR("Failed to alloc drv_ctx.op_buf_ion_info");
10755 return OMX_ErrorInsufficientResources;
10756 }
10757 #endif
10758 if (dynamic_buf_mode) {
10759 out_dynamic_list = (struct dynamic_buf_list *) \
10760 calloc (sizeof(struct dynamic_buf_list), drv_ctx.op_buf.actualcount);
10761 if (out_dynamic_list) {
10762 for (unsigned int i = 0; i < drv_ctx.op_buf.actualcount; i++)
10763 out_dynamic_list[i].dup_fd = -1;
10764 }
10765 }
10766
10767 if (m_out_mem_ptr && pPtr && drv_ctx.ptr_outputbuffer
10768 && drv_ctx.ptr_respbuffer) {
10769 bufHdr = m_out_mem_ptr;
10770 m_platform_list = (OMX_QCOM_PLATFORM_PRIVATE_LIST *)(pPtr);
10771 m_platform_entry= (OMX_QCOM_PLATFORM_PRIVATE_ENTRY *)
10772 (((char *) m_platform_list) + nPlatformListSize);
10773 m_pmem_info = (OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO *)
10774 (((char *) m_platform_entry) + nPlatformEntrySize);
10775 pPlatformList = m_platform_list;
10776 pPlatformEntry = m_platform_entry;
10777 pPMEMInfo = m_pmem_info;
10778
10779 DEBUG_PRINT_LOW("Memory Allocation Succeeded for OUT port%p",m_out_mem_ptr);
10780
10781 // Settting the entire storage nicely
10782 DEBUG_PRINT_LOW("bHdr %p OutMem %p PE %p",bufHdr,
10783 m_out_mem_ptr,pPlatformEntry);
10784 DEBUG_PRINT_LOW(" Pmem Info = %p",pPMEMInfo);
10785 for (i=0; i < drv_ctx.op_buf.actualcount ; i++) {
10786 bufHdr->nSize = sizeof(OMX_BUFFERHEADERTYPE);
10787 bufHdr->nVersion.nVersion = OMX_SPEC_VERSION;
10788 // Set the values when we determine the right HxW param
10789 bufHdr->nAllocLen = 0;
10790 bufHdr->nFilledLen = 0;
10791 bufHdr->pAppPrivate = NULL;
10792 bufHdr->nOutputPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
10793 pPlatformEntry->type = OMX_QCOM_PLATFORM_PRIVATE_PMEM;
10794 pPlatformEntry->entry = pPMEMInfo;
10795 // Initialize the Platform List
10796 pPlatformList->nEntries = 1;
10797 pPlatformList->entryList = pPlatformEntry;
10798 // Keep pBuffer NULL till vdec is opened
10799 bufHdr->pBuffer = NULL;
10800 pPMEMInfo->offset = 0;
10801 pPMEMInfo->pmem_fd = -1;
10802 bufHdr->pPlatformPrivate = pPlatformList;
10803 drv_ctx.ptr_outputbuffer[i].pmem_fd = -1;
10804 #ifdef USE_ION
10805 drv_ctx.op_buf_ion_info[i].ion_device_fd =-1;
10806 #endif
10807 /*Create a mapping between buffers*/
10808 bufHdr->pOutputPortPrivate = &drv_ctx.ptr_respbuffer[i];
10809 drv_ctx.ptr_respbuffer[i].client_data = (void *) \
10810 &drv_ctx.ptr_outputbuffer[i];
10811 // Move the buffer and buffer header pointers
10812 bufHdr++;
10813 pPMEMInfo++;
10814 pPlatformEntry++;
10815 pPlatformList++;
10816 }
10817 } else {
10818 DEBUG_PRINT_ERROR("Output buf mem alloc failed[0x%p][0x%p]",\
10819 m_out_mem_ptr, pPtr);
10820 if (m_out_mem_ptr) {
10821 free(m_out_mem_ptr);
10822 m_out_mem_ptr = NULL;
10823 }
10824 if (pPtr) {
10825 free(pPtr);
10826 pPtr = NULL;
10827 }
10828 if (drv_ctx.ptr_outputbuffer) {
10829 free(drv_ctx.ptr_outputbuffer);
10830 drv_ctx.ptr_outputbuffer = NULL;
10831 }
10832 if (drv_ctx.ptr_respbuffer) {
10833 free(drv_ctx.ptr_respbuffer);
10834 drv_ctx.ptr_respbuffer = NULL;
10835 }
10836 #ifdef USE_ION
10837 if (drv_ctx.op_buf_ion_info) {
10838 DEBUG_PRINT_LOW("Free o/p ion context");
10839 free(drv_ctx.op_buf_ion_info);
10840 drv_ctx.op_buf_ion_info = NULL;
10841 }
10842 #endif
10843 free(drv_ctx.op_buf_map_info);
10844 drv_ctx.op_buf_map_info = NULL;
10845 eRet = OMX_ErrorInsufficientResources;
10846 }
10847 } else {
10848 eRet = OMX_ErrorInsufficientResources;
10849 }
10850 return eRet;
10851 }
10852
complete_pending_buffer_done_cbs()10853 void omx_vdec::complete_pending_buffer_done_cbs()
10854 {
10855 unsigned long p1, p2, ident;
10856 omx_cmd_queue tmp_q, pending_bd_q;
10857 pthread_mutex_lock(&m_lock);
10858 // pop all pending GENERATE FDB from ftb queue
10859 while (m_ftb_q.m_size) {
10860 m_ftb_q.pop_entry(&p1,&p2,&ident);
10861 if (ident == OMX_COMPONENT_GENERATE_FBD) {
10862 pending_bd_q.insert_entry(p1,p2,ident);
10863 } else {
10864 tmp_q.insert_entry(p1,p2,ident);
10865 }
10866 }
10867 //return all non GENERATE FDB to ftb queue
10868 while (tmp_q.m_size) {
10869 tmp_q.pop_entry(&p1,&p2,&ident);
10870 m_ftb_q.insert_entry(p1,p2,ident);
10871 }
10872 // pop all pending GENERATE EDB from etb queue
10873 while (m_etb_q.m_size) {
10874 m_etb_q.pop_entry(&p1,&p2,&ident);
10875 if (ident == OMX_COMPONENT_GENERATE_EBD) {
10876 pending_bd_q.insert_entry(p1,p2,ident);
10877 } else {
10878 tmp_q.insert_entry(p1,p2,ident);
10879 }
10880 }
10881 //return all non GENERATE FDB to etb queue
10882 while (tmp_q.m_size) {
10883 tmp_q.pop_entry(&p1,&p2,&ident);
10884 m_etb_q.insert_entry(p1,p2,ident);
10885 }
10886 pthread_mutex_unlock(&m_lock);
10887 // process all pending buffer dones
10888 while (pending_bd_q.m_size) {
10889 pending_bd_q.pop_entry(&p1,&p2,&ident);
10890 switch (ident) {
10891 case OMX_COMPONENT_GENERATE_EBD:
10892 if (empty_buffer_done(&m_cmp, (OMX_BUFFERHEADERTYPE *)p1) != OMX_ErrorNone) {
10893 DEBUG_PRINT_ERROR("ERROR: empty_buffer_done() failed!");
10894 omx_report_error ();
10895 }
10896 break;
10897
10898 case OMX_COMPONENT_GENERATE_FBD:
10899 if (fill_buffer_done(&m_cmp, (OMX_BUFFERHEADERTYPE *)p1) != OMX_ErrorNone ) {
10900 DEBUG_PRINT_ERROR("ERROR: fill_buffer_done() failed!");
10901 omx_report_error ();
10902 }
10903 break;
10904 }
10905 }
10906 }
10907
set_frame_rate(OMX_S64 act_timestamp)10908 void omx_vdec::set_frame_rate(OMX_S64 act_timestamp)
10909 {
10910 OMX_U32 new_frame_interval = 0;
10911 if (VALID_TS(act_timestamp) && VALID_TS(prev_ts) && act_timestamp != prev_ts
10912 && llabs(act_timestamp - prev_ts) > 2000) {
10913 new_frame_interval = client_set_fps ? frm_int : (act_timestamp - prev_ts) > 0 ?
10914 llabs(act_timestamp - prev_ts) : llabs(act_timestamp - prev_ts_actual);
10915 if (new_frame_interval != frm_int || frm_int == 0) {
10916 frm_int = new_frame_interval;
10917 if (frm_int) {
10918 drv_ctx.frame_rate.fps_numerator = 1e6;
10919 drv_ctx.frame_rate.fps_denominator = frm_int;
10920 DEBUG_PRINT_LOW("set_frame_rate: frm_int(%u) fps(%f)",
10921 (unsigned int)frm_int, drv_ctx.frame_rate.fps_numerator /
10922 (float)drv_ctx.frame_rate.fps_denominator);
10923 m_perf_control.request_cores(frm_int);
10924 /* We need to report the difference between this FBD and the previous FBD
10925 * back to the driver for clock scaling purposes. */
10926 struct v4l2_outputparm oparm;
10927 /*XXX: we're providing timing info as seconds per frame rather than frames
10928 * per second.*/
10929 oparm.timeperframe.numerator = drv_ctx.frame_rate.fps_denominator;
10930 oparm.timeperframe.denominator = drv_ctx.frame_rate.fps_numerator;
10931
10932 struct v4l2_streamparm sparm;
10933 sparm.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
10934 sparm.parm.output = oparm;
10935 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_PARM, &sparm)) {
10936 DEBUG_PRINT_ERROR("Unable to convey fps info to driver, \
10937 performance might be affected");
10938 }
10939
10940 }
10941 }
10942 }
10943 prev_ts = act_timestamp;
10944 }
10945
adjust_timestamp(OMX_S64 & act_timestamp)10946 void omx_vdec::adjust_timestamp(OMX_S64 &act_timestamp)
10947 {
10948 if (rst_prev_ts && VALID_TS(act_timestamp)) {
10949 prev_ts = act_timestamp;
10950 prev_ts_actual = act_timestamp;
10951 rst_prev_ts = false;
10952 } else if (VALID_TS(prev_ts)) {
10953 bool codec_cond = (drv_ctx.timestamp_adjust)?
10954 (!VALID_TS(act_timestamp) || act_timestamp < prev_ts_actual || llabs(act_timestamp - prev_ts_actual) <= 2000) :
10955 (!VALID_TS(act_timestamp) || act_timestamp <= prev_ts_actual);
10956 prev_ts_actual = act_timestamp; //unadjusted previous timestamp
10957 if (frm_int > 0 && codec_cond) {
10958 DEBUG_PRINT_LOW("adjust_timestamp: original ts[%lld]", act_timestamp);
10959 act_timestamp = prev_ts + frm_int;
10960 DEBUG_PRINT_LOW("adjust_timestamp: predicted ts[%lld]", act_timestamp);
10961 prev_ts = act_timestamp;
10962 } else {
10963 if (drv_ctx.picture_order == VDEC_ORDER_DISPLAY && act_timestamp < prev_ts) {
10964 // ensure that timestamps can never step backwards when in display order
10965 act_timestamp = prev_ts;
10966 }
10967 set_frame_rate(act_timestamp);
10968 }
10969 } else if (frm_int > 0) // In this case the frame rate was set along
10970 { // with the port definition, start ts with 0
10971 act_timestamp = prev_ts = 0; // and correct if a valid ts is received.
10972 rst_prev_ts = true;
10973 }
10974 }
10975
get_omx_output_buffer_header(int index)10976 OMX_BUFFERHEADERTYPE* omx_vdec::get_omx_output_buffer_header(int index)
10977 {
10978 return m_out_mem_ptr + index;
10979 }
10980
convert_color_space_info(OMX_U32 primaries,OMX_U32 range,OMX_U32 transfer,OMX_U32 matrix,ColorSpace_t * color_space,ColorAspects * aspects)10981 void omx_vdec::convert_color_space_info(OMX_U32 primaries, OMX_U32 range,
10982 OMX_U32 transfer, OMX_U32 matrix, ColorSpace_t *color_space, ColorAspects *aspects)
10983 {
10984 switch (primaries) {
10985 case MSM_VIDC_BT709_5:
10986 *color_space = ITU_R_709;
10987 aspects->mPrimaries = ColorAspects::PrimariesBT709_5;
10988 break;
10989 case MSM_VIDC_BT470_6_M:
10990 aspects->mPrimaries = ColorAspects::PrimariesBT470_6M;
10991 break;
10992 case MSM_VIDC_BT601_6_625:
10993 aspects->mPrimaries = ColorAspects::PrimariesBT601_6_625;
10994 break;
10995 case MSM_VIDC_BT601_6_525:
10996 *color_space = range ? ITU_R_601_FR : ITU_R_601;
10997 aspects->mPrimaries = ColorAspects::PrimariesBT601_6_525;
10998 break;
10999 case MSM_VIDC_GENERIC_FILM:
11000 aspects->mPrimaries = ColorAspects::PrimariesGenericFilm;
11001 break;
11002 case MSM_VIDC_BT2020:
11003 aspects->mPrimaries = ColorAspects::PrimariesBT2020;
11004 break;
11005 case MSM_VIDC_UNSPECIFIED:
11006 //Client does not expect ColorAspects::PrimariesUnspecified, but rather the supplied default
11007 default:
11008 //aspects->mPrimaries = ColorAspects::PrimariesOther;
11009 aspects->mPrimaries = m_client_color_space.sAspects.mPrimaries;
11010 break;
11011 }
11012
11013 aspects->mRange = range ? ColorAspects::RangeFull : ColorAspects::RangeLimited;
11014
11015 switch (transfer) {
11016 case MSM_VIDC_TRANSFER_BT709_5:
11017 case MSM_VIDC_TRANSFER_601_6_525: // case MSM_VIDC_TRANSFER_601_6_625:
11018 aspects->mTransfer = ColorAspects::TransferSMPTE170M;
11019 break;
11020 case MSM_VIDC_TRANSFER_BT_470_6_M:
11021 aspects->mTransfer = ColorAspects::TransferGamma22;
11022 break;
11023 case MSM_VIDC_TRANSFER_BT_470_6_BG:
11024 aspects->mTransfer = ColorAspects::TransferGamma28;
11025 break;
11026 case MSM_VIDC_TRANSFER_SMPTE_240M:
11027 aspects->mTransfer = ColorAspects::TransferSMPTE240M;
11028 break;
11029 case MSM_VIDC_TRANSFER_LINEAR:
11030 aspects->mTransfer = ColorAspects::TransferLinear;
11031 break;
11032 case MSM_VIDC_TRANSFER_IEC_61966:
11033 aspects->mTransfer = ColorAspects::TransferXvYCC;
11034 break;
11035 case MSM_VIDC_TRANSFER_BT_1361:
11036 aspects->mTransfer = ColorAspects::TransferBT1361;
11037 break;
11038 case MSM_VIDC_TRANSFER_SRGB:
11039 aspects->mTransfer = ColorAspects::TransferSRGB;
11040 break;
11041 case MSM_VIDC_TRANSFER_SMPTE_ST2084:
11042 aspects->mTransfer = ColorAspects::TransferST2084;
11043 break;
11044 case MSM_VIDC_TRANSFER_HLG:
11045 aspects->mTransfer = ColorAspects::TransferHLG;
11046 break;
11047 default:
11048 //aspects->mTransfer = ColorAspects::TransferOther;
11049 aspects->mTransfer = m_client_color_space.sAspects.mTransfer;
11050 break;
11051 }
11052
11053 switch (matrix) {
11054 case MSM_VIDC_MATRIX_BT_709_5:
11055 aspects->mMatrixCoeffs = ColorAspects::MatrixBT709_5;
11056 break;
11057 case MSM_VIDC_MATRIX_FCC_47:
11058 aspects->mMatrixCoeffs = ColorAspects::MatrixBT470_6M;
11059 break;
11060 case MSM_VIDC_MATRIX_601_6_625:
11061 case MSM_VIDC_MATRIX_601_6_525:
11062 aspects->mMatrixCoeffs = ColorAspects::MatrixBT601_6;
11063 break;
11064 case MSM_VIDC_MATRIX_SMPTE_240M:
11065 aspects->mMatrixCoeffs = ColorAspects::MatrixSMPTE240M;
11066 break;
11067 case MSM_VIDC_MATRIX_BT_2020:
11068 aspects->mMatrixCoeffs = ColorAspects::MatrixBT2020;
11069 break;
11070 case MSM_VIDC_MATRIX_BT_2020_CONST:
11071 aspects->mMatrixCoeffs = ColorAspects::MatrixBT2020Constant;
11072 break;
11073 default:
11074 //aspects->mMatrixCoeffs = ColorAspects::MatrixOther;
11075 aspects->mMatrixCoeffs = m_client_color_space.sAspects.mMatrixCoeffs;
11076 break;
11077 }
11078 }
11079
print_debug_color_aspects(ColorAspects * a,const char * prefix)11080 void omx_vdec::print_debug_color_aspects(ColorAspects *a, const char *prefix) {
11081 DEBUG_PRINT_HIGH("%s : Color aspects : Primaries = %d(%s) Range = %d(%s) Tx = %d(%s) Matrix = %d(%s)",
11082 prefix, a->mPrimaries, asString(a->mPrimaries), a->mRange, asString(a->mRange),
11083 a->mTransfer, asString(a->mTransfer), a->mMatrixCoeffs, asString(a->mMatrixCoeffs));
11084 }
11085
prepare_color_aspects_metadata(OMX_U32 primaries,OMX_U32 range,OMX_U32 transfer,OMX_U32 matrix,ColorMetaData * color_mdata)11086 void omx_vdec::prepare_color_aspects_metadata(OMX_U32 primaries, OMX_U32 range,
11087 OMX_U32 transfer, OMX_U32 matrix,
11088 ColorMetaData *color_mdata)
11089 {
11090
11091 /* ColorAspects in qdMetaData */
11092 color_mdata->colorPrimaries = (enum ColorPrimaries) primaries;
11093 color_mdata->range = (enum ColorRange)range;
11094 color_mdata->transfer = (enum GammaTransfer)transfer;
11095 color_mdata->matrixCoefficients = (enum MatrixCoEfficients)matrix;
11096 }
11097
handle_color_space_info(void * data,ColorSpace_t * color_space,ColorMetaData * color_mdata,bool & set_color_aspects_only)11098 bool omx_vdec::handle_color_space_info(void *data,
11099 ColorSpace_t *color_space,
11100 ColorMetaData *color_mdata,
11101 bool& set_color_aspects_only)
11102 {
11103 ColorAspects tempAspects;
11104 memset(&tempAspects, 0x0, sizeof(ColorAspects));
11105 ColorAspects *aspects = &tempAspects;
11106
11107 /* Set default ColorAspects */
11108 prepare_color_aspects_metadata(ColorPrimaries_BT601_6_625, Range_Full,
11109 Transfer_SMPTE_170M, MatrixCoEff_BT601_6_625,
11110 color_mdata);
11111
11112 switch(output_capability) {
11113 case V4L2_PIX_FMT_MPEG2:
11114 {
11115 struct msm_vidc_mpeg2_seqdisp_payload *seqdisp_payload;
11116 seqdisp_payload = (struct msm_vidc_mpeg2_seqdisp_payload *)data;
11117
11118 /* Refer MPEG2 Spec @ Rec. ISO/IEC 13818-2, ITU-T Draft Rec. H.262 to
11119 * understand this code */
11120
11121 if (seqdisp_payload && seqdisp_payload->color_descp) {
11122
11123 convert_color_space_info(seqdisp_payload->color_primaries, 1,
11124 seqdisp_payload->transfer_char, seqdisp_payload->matrix_coeffs,
11125 color_space,aspects);
11126 m_disp_hor_size = seqdisp_payload->disp_width;
11127 m_disp_vert_size = seqdisp_payload->disp_height;
11128 set_color_aspects_only = true;
11129 prepare_color_aspects_metadata(seqdisp_payload->color_primaries, 1,
11130 seqdisp_payload->transfer_char, seqdisp_payload->matrix_coeffs,
11131 color_mdata);
11132 }
11133 }
11134 break;
11135 case V4L2_PIX_FMT_H264:
11136 case V4L2_PIX_FMT_HEVC:
11137 {
11138 struct msm_vidc_vui_display_info_payload *display_info_payload;
11139 display_info_payload = (struct msm_vidc_vui_display_info_payload*)data;
11140
11141 /* Refer H264 Spec @ Rec. ITU-T H.264 (02/2014) to understand this code */
11142
11143 if (display_info_payload->video_signal_present_flag &&
11144 display_info_payload->color_description_present_flag) {
11145 convert_color_space_info(display_info_payload->color_primaries,
11146 display_info_payload->video_full_range_flag,
11147 display_info_payload->transfer_characteristics,
11148 display_info_payload->matrix_coefficients,
11149 color_space,aspects);
11150 set_color_aspects_only = true;
11151 prepare_color_aspects_metadata(display_info_payload->color_primaries,
11152 display_info_payload->video_full_range_flag,
11153 display_info_payload->transfer_characteristics,
11154 display_info_payload->matrix_coefficients,
11155 color_mdata);
11156 }
11157 }
11158 break;
11159 case V4L2_PIX_FMT_VC1_ANNEX_G:
11160 case V4L2_PIX_FMT_VC1_ANNEX_L:
11161 {
11162 struct msm_vidc_vc1_seqdisp_payload *vc1_seq_disp_payload;
11163 vc1_seq_disp_payload = (struct msm_vidc_vc1_seqdisp_payload*)data;
11164
11165 /* Refer VC-1 Spec @ SMPTE Draft Standard for Television Date: 2005-08-23
11166 * SMPTE 421M to understand this code */
11167
11168 if (m_enable_android_native_buffers &&
11169 vc1_seq_disp_payload->color_primaries) {
11170
11171 convert_color_space_info(vc1_seq_disp_payload->color_primaries,
11172 1,
11173 vc1_seq_disp_payload->transfer_char,
11174 vc1_seq_disp_payload->matrix_coeffs,
11175 color_space,aspects);
11176 set_color_aspects_only = true;
11177 prepare_color_aspects_metadata(vc1_seq_disp_payload->color_primaries,
11178 1,
11179 vc1_seq_disp_payload->transfer_char,
11180 vc1_seq_disp_payload->matrix_coeffs,
11181 color_mdata);
11182 }
11183 }
11184 break;
11185 case V4L2_PIX_FMT_VP8:
11186 {
11187 struct msm_vidc_vpx_colorspace_payload *vpx_color_space_payload;
11188 vpx_color_space_payload = (struct msm_vidc_vpx_colorspace_payload*)data;
11189 set_color_aspects_only = false;
11190 /* Refer VP8 Data Format in latest VP8 spec and Decoding Guide November 2011
11191 * to understand this code */
11192
11193 if (vpx_color_space_payload->color_space == 0) {
11194 *color_space = ITU_R_601;
11195 aspects->mPrimaries = ColorAspects::PrimariesBT601_6_525;
11196 aspects->mRange = ColorAspects::RangeLimited;
11197 } else {
11198 DEBUG_PRINT_ERROR("Unsupported Color space for VP8");
11199 break;
11200 }
11201 }
11202 break;
11203 case V4L2_PIX_FMT_VP9:
11204 {
11205 struct msm_vidc_vpx_colorspace_payload *vpx_color_space_payload;
11206 vpx_color_space_payload = (struct msm_vidc_vpx_colorspace_payload*)data;
11207 set_color_aspects_only = false;
11208 /* Refer VP9 Spec @ VP9 Bitstream & Decoding Process Specification - v0.6 31st March 2016
11209 * to understand this code */
11210
11211 switch(vpx_color_space_payload->color_space) {
11212 case MSM_VIDC_CS_BT_601:
11213 aspects->mMatrixCoeffs = ColorAspects::MatrixBT601_6;
11214 aspects->mTransfer = ColorAspects::TransferSMPTE170M;
11215 aspects->mPrimaries = ColorAspects::PrimariesBT601_6_625;
11216 aspects->mRange = m_client_color_space.sAspects.mRange;
11217 break;
11218 case MSM_VIDC_CS_BT_709:
11219 *color_space = ITU_R_709;
11220 aspects->mMatrixCoeffs = ColorAspects::MatrixBT709_5;
11221 aspects->mTransfer = ColorAspects::TransferSMPTE170M;
11222 aspects->mPrimaries = ColorAspects::PrimariesBT709_5;
11223 aspects->mRange = m_client_color_space.sAspects.mRange;
11224 break;
11225 case MSM_VIDC_CS_SMPTE_170:
11226 aspects->mMatrixCoeffs = ColorAspects::MatrixBT709_5;
11227 aspects->mTransfer = ColorAspects::TransferSMPTE170M;
11228 aspects->mPrimaries = m_client_color_space.sAspects.mPrimaries;
11229 aspects->mRange = m_client_color_space.sAspects.mRange;
11230 break;
11231 case MSM_VIDC_CS_SMPTE_240:
11232 aspects->mMatrixCoeffs = m_client_color_space.sAspects.mMatrixCoeffs;
11233 aspects->mTransfer = ColorAspects::TransferSMPTE240M;
11234 aspects->mPrimaries = m_client_color_space.sAspects.mPrimaries;
11235 aspects->mRange = m_client_color_space.sAspects.mRange;
11236 break;
11237 case MSM_VIDC_CS_BT_2020:
11238 aspects->mMatrixCoeffs = ColorAspects::MatrixBT2020;
11239 aspects->mTransfer = ColorAspects:: TransferSMPTE170M;
11240 aspects->mPrimaries = ColorAspects::PrimariesBT2020;
11241 aspects->mRange = m_client_color_space.sAspects.mRange;
11242 break;
11243 case MSM_VIDC_CS_RESERVED:
11244 aspects->mMatrixCoeffs = ColorAspects::MatrixOther;
11245 aspects->mTransfer = ColorAspects::TransferOther;
11246 aspects->mPrimaries = ColorAspects::PrimariesOther;
11247 aspects->mRange = m_client_color_space.sAspects.mRange;
11248 break;
11249 case MSM_VIDC_CS_RGB:
11250 aspects->mMatrixCoeffs = ColorAspects::MatrixBT709_5;
11251 aspects->mTransfer = ColorAspects::TransferSMPTE170M;
11252 aspects->mPrimaries = ColorAspects::PrimariesOther;
11253 aspects->mRange = m_client_color_space.sAspects.mRange;
11254 break;
11255 default:
11256 break;
11257 }
11258 }
11259 break;
11260 default:
11261 break;
11262 }
11263
11264 print_debug_color_aspects(aspects, "Bitstream");
11265
11266 if (m_internal_color_space.sAspects.mPrimaries != aspects->mPrimaries ||
11267 m_internal_color_space.sAspects.mTransfer != aspects->mTransfer ||
11268 m_internal_color_space.sAspects.mMatrixCoeffs != aspects->mMatrixCoeffs ||
11269 m_internal_color_space.sAspects.mRange != aspects->mRange) {
11270 memcpy(&(m_internal_color_space.sAspects), aspects, sizeof(ColorAspects));
11271
11272 m_color_mdata.colorPrimaries = color_mdata->colorPrimaries;
11273 m_color_mdata.range = color_mdata->range;
11274 m_color_mdata.transfer = color_mdata->transfer;
11275 m_color_mdata.matrixCoefficients = color_mdata->matrixCoefficients;
11276
11277 DEBUG_PRINT_HIGH("Initiating PORT Reconfig due to Color Aspects Change");
11278 print_debug_color_aspects(&(m_internal_color_space.sAspects), "Internal");
11279 print_debug_color_aspects(&(m_client_color_space.sAspects), "Client");
11280
11281 post_event(OMX_CORE_OUTPUT_PORT_INDEX,
11282 OMX_QTIIndexConfigDescribeColorAspects,
11283 OMX_COMPONENT_GENERATE_PORT_RECONFIG);
11284 return true;
11285 }
11286 return false;
11287 }
11288
set_colorspace_in_handle(ColorSpace_t color_space,unsigned int buf_index)11289 void omx_vdec::set_colorspace_in_handle(ColorSpace_t color_space, unsigned int buf_index) {
11290 private_handle_t *private_handle = NULL;
11291 if (buf_index < drv_ctx.op_buf.actualcount &&
11292 buf_index < MAX_NUM_INPUT_OUTPUT_BUFFERS &&
11293 native_buffer[buf_index].privatehandle) {
11294 private_handle = native_buffer[buf_index].privatehandle;
11295 }
11296 if (private_handle) {
11297 setMetaData(private_handle, UPDATE_COLOR_SPACE, (void*)&color_space);
11298 }
11299 }
11300
print_debug_hdr_color_info(HDRStaticInfo * hdr_info,const char * prefix)11301 void omx_vdec::print_debug_hdr_color_info(HDRStaticInfo *hdr_info, const char *prefix)
11302 {
11303 if (!hdr_info->mID) {
11304 DEBUG_PRINT_LOW("%s : HDRstaticinfo MDC: mR.x = %d mR.y = %d", prefix,
11305 hdr_info->sType1.mR.x, hdr_info->sType1.mR.y);
11306 DEBUG_PRINT_LOW("%s : HDRstaticinfo MDC: mG.x = %d mG.y = %d", prefix,
11307 hdr_info->sType1.mG.x, hdr_info->sType1.mG.y);
11308 DEBUG_PRINT_LOW("%s : HDRstaticinfo MDC: mB.x = %d mB.y = %d", prefix,
11309 hdr_info->sType1.mB.x, hdr_info->sType1.mB.y);
11310 DEBUG_PRINT_LOW("%s : HDRstaticinfo MDC: mW.x = %d mW.y = %d", prefix,
11311 hdr_info->sType1.mW.x, hdr_info->sType1.mW.y);
11312 DEBUG_PRINT_LOW("%s : HDRstaticinfo MDC: maxDispLum = %d minDispLum = %d", prefix,
11313 hdr_info->sType1.mMaxDisplayLuminance, hdr_info->sType1.mMinDisplayLuminance);
11314 DEBUG_PRINT_LOW("%s : HDRstaticinfo CLL: CLL = %d FLL = %d", prefix,
11315 hdr_info->sType1.mMaxContentLightLevel, hdr_info->sType1.mMaxFrameAverageLightLevel);
11316 }
11317
11318 }
11319
print_debug_hdr_color_info_mdata(ColorMetaData * color_mdata)11320 void omx_vdec::print_debug_hdr_color_info_mdata(ColorMetaData* color_mdata)
11321 {
11322 DEBUG_PRINT_LOW("setMetaData COLOR_METADATA : color_primaries = %u, range = %u, transfer = %u, matrix = %u",
11323 color_mdata->colorPrimaries, color_mdata->range,
11324 color_mdata->transfer, color_mdata->matrixCoefficients);
11325
11326 for(uint8_t i = 0; i < 3; i++) {
11327 for(uint8_t j = 0; j < 2; j++) {
11328 DEBUG_PRINT_LOW("setMetadata COLOR_METADATA : rgbPrimaries[%d][%d] = %d", i, j, color_mdata->masteringDisplayInfo.primaries.rgbPrimaries[i][j]);
11329 }
11330 }
11331
11332 DEBUG_PRINT_LOW("setMetadata COLOR_METADATA : whitepoint[0] = %d whitepoint[1] = %d",
11333 color_mdata->masteringDisplayInfo.primaries.whitePoint[0],
11334 color_mdata->masteringDisplayInfo.primaries.whitePoint[1]);
11335
11336 DEBUG_PRINT_LOW("setMetadata COLOR_METADATA : maxDispLum = %d minDispLum = %d",
11337 color_mdata->masteringDisplayInfo.maxDisplayLuminance,
11338 color_mdata->masteringDisplayInfo.minDisplayLuminance);
11339
11340 DEBUG_PRINT_LOW("setMetadata COLOR_METADATA : maxCLL = %d maxFLL = %d",
11341 color_mdata->contentLightLevel.maxContentLightLevel,
11342 color_mdata->contentLightLevel.minPicAverageLightLevel);
11343
11344
11345 }
11346
handle_content_light_level_info(void * data,ContentLightLevel * light_level_mdata)11347 bool omx_vdec::handle_content_light_level_info(void* data, ContentLightLevel* light_level_mdata)
11348 {
11349 struct msm_vidc_content_light_level_sei_payload *light_level_payload =
11350 (msm_vidc_content_light_level_sei_payload*)(data);
11351
11352 light_level_mdata->lightLevelSEIEnabled = true;
11353 light_level_mdata->maxContentLightLevel = light_level_payload->nMaxContentLight;
11354 light_level_mdata->minPicAverageLightLevel = light_level_payload->nMaxPicAverageLight;
11355
11356 if ((m_internal_hdr_info.sInfo.sType1.mMaxContentLightLevel != light_level_payload->nMaxContentLight) ||
11357 (m_internal_hdr_info.sInfo.sType1.mMaxFrameAverageLightLevel != light_level_payload->nMaxPicAverageLight)) {
11358 m_internal_hdr_info.sInfo.sType1.mMaxContentLightLevel = light_level_payload->nMaxContentLight;
11359 m_internal_hdr_info.sInfo.sType1.mMaxFrameAverageLightLevel = light_level_payload->nMaxPicAverageLight;
11360 return true;
11361 }
11362 return false;
11363 }
11364
handle_mastering_display_color_info(void * data,MasteringDisplay * mastering_display_mdata)11365 bool omx_vdec::handle_mastering_display_color_info(void* data, MasteringDisplay* mastering_display_mdata)
11366 {
11367 struct msm_vidc_mastering_display_colour_sei_payload *mastering_display_payload =
11368 (msm_vidc_mastering_display_colour_sei_payload*)(data);
11369 HDRStaticInfo* hdr_info = &m_internal_hdr_info.sInfo;
11370 bool internal_disp_changed_flag = false;
11371
11372 mastering_display_mdata->colorVolumeSEIEnabled = true;
11373 for (uint8_t i = 0; i < 3; i++) {
11374 mastering_display_mdata->primaries.rgbPrimaries[i][0] = mastering_display_payload->nDisplayPrimariesX[i];
11375 mastering_display_mdata->primaries.rgbPrimaries[i][1] = mastering_display_payload->nDisplayPrimariesY[i];
11376 }
11377 mastering_display_mdata->primaries.whitePoint[0] = mastering_display_payload->nWhitePointX;
11378 mastering_display_mdata->primaries.whitePoint[1] = mastering_display_payload->nWhitePointY;
11379 mastering_display_mdata->maxDisplayLuminance = mastering_display_payload->nMaxDisplayMasteringLuminance;
11380 mastering_display_mdata->minDisplayLuminance = mastering_display_payload->nMinDisplayMasteringLuminance;
11381
11382 internal_disp_changed_flag |= (hdr_info->sType1.mR.x != mastering_display_payload->nDisplayPrimariesX[0]) ||
11383 (hdr_info->sType1.mR.y != mastering_display_payload->nDisplayPrimariesY[0]);
11384 internal_disp_changed_flag |= (hdr_info->sType1.mG.x != mastering_display_payload->nDisplayPrimariesX[1]) ||
11385 (hdr_info->sType1.mG.y != mastering_display_payload->nDisplayPrimariesY[1]);
11386 internal_disp_changed_flag |= (hdr_info->sType1.mB.x != mastering_display_payload->nDisplayPrimariesX[2]) ||
11387 (hdr_info->sType1.mB.y != mastering_display_payload->nDisplayPrimariesY[2]);
11388
11389 internal_disp_changed_flag |= (hdr_info->sType1.mW.x != mastering_display_payload->nWhitePointX) ||
11390 (hdr_info->sType1.mW.y != mastering_display_payload->nWhitePointY);
11391
11392 /* Maximum Display Luminance from the bitstream is in 0.0001 cd/m2 while the HDRStaticInfo extension
11393 requires it in cd/m2, so dividing by 10000 and rounding the value after division
11394 */
11395 uint16_t max_display_luminance_cd_m2 =
11396 static_cast<int>((mastering_display_payload->nMaxDisplayMasteringLuminance / LUMINANCE_DIV_FACTOR) + 0.5);
11397 internal_disp_changed_flag |= (hdr_info->sType1.mMaxDisplayLuminance != max_display_luminance_cd_m2) ||
11398 (hdr_info->sType1.mMinDisplayLuminance != mastering_display_payload->nMinDisplayMasteringLuminance);
11399
11400 if (internal_disp_changed_flag) {
11401 hdr_info->sType1.mR.x = mastering_display_payload->nDisplayPrimariesX[0];
11402 hdr_info->sType1.mR.y = mastering_display_payload->nDisplayPrimariesY[0];
11403 hdr_info->sType1.mG.x = mastering_display_payload->nDisplayPrimariesX[1];
11404 hdr_info->sType1.mG.y = mastering_display_payload->nDisplayPrimariesY[1];
11405 hdr_info->sType1.mB.x = mastering_display_payload->nDisplayPrimariesX[2];
11406 hdr_info->sType1.mB.y = mastering_display_payload->nDisplayPrimariesY[2];
11407 hdr_info->sType1.mW.x = mastering_display_payload->nWhitePointX;
11408 hdr_info->sType1.mW.y = mastering_display_payload->nWhitePointY;
11409
11410 hdr_info->sType1.mMaxDisplayLuminance = max_display_luminance_cd_m2;
11411 hdr_info->sType1.mMinDisplayLuminance = mastering_display_payload->nMinDisplayMasteringLuminance;
11412 }
11413
11414 return internal_disp_changed_flag;
11415 }
11416
set_colormetadata_in_handle(ColorMetaData * color_mdata,unsigned int buf_index)11417 void omx_vdec::set_colormetadata_in_handle(ColorMetaData *color_mdata, unsigned int buf_index)
11418 {
11419 private_handle_t *private_handle = NULL;
11420 if (buf_index < drv_ctx.op_buf.actualcount &&
11421 buf_index < MAX_NUM_INPUT_OUTPUT_BUFFERS &&
11422 native_buffer[buf_index].privatehandle) {
11423 private_handle = native_buffer[buf_index].privatehandle;
11424 }
11425 if (private_handle) {
11426 setMetaData(private_handle, COLOR_METADATA, (void*)color_mdata);
11427 }
11428 }
11429
handle_extradata(OMX_BUFFERHEADERTYPE * p_buf_hdr)11430 void omx_vdec::handle_extradata(OMX_BUFFERHEADERTYPE *p_buf_hdr)
11431 {
11432 OMX_OTHER_EXTRADATATYPE *p_extra = NULL, *p_sei = NULL, *p_vui = NULL, *p_client_extra = NULL;
11433 OMX_U8 *pBuffer = NULL;
11434 OMX_U32 num_conceal_MB = 0;
11435 OMX_TICKS time_stamp = 0;
11436 OMX_U32 frame_rate = 0;
11437 unsigned long consumed_len = 0;
11438 OMX_U32 num_MB_in_frame;
11439 OMX_U32 recovery_sei_flags = 1;
11440 int enable = OMX_InterlaceFrameProgressive;
11441 bool internal_hdr_info_changed_flag = false;
11442 bool color_event = false;
11443 ColorMetaData color_mdata;
11444 memset(&color_mdata, 0x0, sizeof(ColorMetaData));
11445 bool set_disp_color_aspects_only = false;
11446 ColorSpace_t color_space = ITU_R_601;
11447
11448 int buf_index = p_buf_hdr - m_out_mem_ptr;
11449 if (buf_index >= drv_ctx.extradata_info.count) {
11450 DEBUG_PRINT_ERROR("handle_extradata: invalid index(%d) max(%d)",
11451 buf_index, drv_ctx.extradata_info.count);
11452 return;
11453 }
11454 struct msm_vidc_panscan_window_payload *panscan_payload = NULL;
11455
11456 if (drv_ctx.ptr_outputbuffer[buf_index].bufferaddr == NULL) {
11457 DEBUG_PRINT_ERROR("handle_extradata: Error: Mapped output buffer address is NULL");
11458 return;
11459 }
11460
11461 if (!drv_ctx.extradata_info.uaddr) {
11462 DEBUG_PRINT_HIGH("NULL drv_ctx.extradata_info.uaddr");
11463 return;
11464 }
11465 if (!secure_mode && (drv_ctx.extradata_info.buffer_size > (p_buf_hdr->nAllocLen - p_buf_hdr->nFilledLen)) ) {
11466 DEBUG_PRINT_ERROR("Error: Insufficient size allocated for extra-data");
11467 p_extra = NULL;
11468 return;
11469 }
11470 if (!secure_mode) {
11471 pBuffer = (OMX_U8*)mmap(0, drv_ctx.ptr_outputbuffer[buf_index].buffer_len,
11472 PROT_READ|PROT_WRITE, MAP_SHARED, drv_ctx.ptr_outputbuffer[buf_index].pmem_fd, 0);
11473 if (pBuffer == MAP_FAILED) {
11474 DEBUG_PRINT_ERROR("handle_extradata output buffer mmap failed - errno: %d", errno);
11475 return;
11476 }
11477 p_extra = (OMX_OTHER_EXTRADATATYPE *)
11478 ((unsigned long)(pBuffer + p_buf_hdr->nOffset + p_buf_hdr->nFilledLen + 3)&(~3));
11479 } else
11480 p_extra = m_other_extradata;
11481
11482 AutoUnmap autounmap(pBuffer, drv_ctx.ptr_outputbuffer[buf_index].buffer_len);
11483 if (m_client_output_extradata_mem_ptr &&
11484 m_client_out_extradata_info.getSize() >= drv_ctx.extradata_info.buffer_size) {
11485 p_client_extra = (OMX_OTHER_EXTRADATATYPE *)((m_client_output_extradata_mem_ptr + buf_index)->pBuffer);
11486 }
11487
11488 char *p_extradata = drv_ctx.extradata_info.uaddr + buf_index * drv_ctx.extradata_info.buffer_size;
11489
11490 if (!secure_mode && ((OMX_U8*)p_extra > (pBuffer + p_buf_hdr->nAllocLen))) {
11491 p_extra = NULL;
11492 DEBUG_PRINT_ERROR("Error: out of bound memory access by p_extra");
11493 return;
11494 }
11495 m_extradata_info.output_crop_updated = OMX_FALSE;
11496 OMX_OTHER_EXTRADATATYPE *data = (struct OMX_OTHER_EXTRADATATYPE *)p_extradata;
11497 if (data && p_extra) {
11498 while ((consumed_len < drv_ctx.extradata_info.buffer_size)
11499 && (data->eType != (OMX_EXTRADATATYPE)MSM_VIDC_EXTRADATA_NONE)) {
11500 if ((consumed_len + data->nSize) > (unsigned)drv_ctx.extradata_info.buffer_size) {
11501 DEBUG_PRINT_LOW("Invalid extra data size");
11502 break;
11503 }
11504
11505 if (!secure_mode && ((OMX_U8*)p_extra > (pBuffer + p_buf_hdr->nAllocLen))) {
11506 p_extra = NULL;
11507 DEBUG_PRINT_ERROR("Error: out of bound memory access by p_extra");
11508 return;
11509 }
11510
11511 DEBUG_PRINT_LOW("handle_extradata: eType = 0x%x", data->eType);
11512 switch ((unsigned long)data->eType) {
11513 case MSM_VIDC_EXTRADATA_INTERLACE_VIDEO:
11514 struct msm_vidc_interlace_payload *payload;
11515 OMX_U32 interlace_color_format;
11516 payload = (struct msm_vidc_interlace_payload *)(void *)data->data;
11517 if (payload) {
11518 enable = OMX_InterlaceFrameProgressive;
11519 switch (payload->format) {
11520 case MSM_VIDC_INTERLACE_FRAME_PROGRESSIVE:
11521 drv_ctx.interlace = VDEC_InterlaceFrameProgressive;
11522 break;
11523 case MSM_VIDC_INTERLACE_INTERLEAVE_FRAME_TOPFIELDFIRST:
11524 drv_ctx.interlace = VDEC_InterlaceInterleaveFrameTopFieldFirst;
11525 enable = OMX_InterlaceInterleaveFrameTopFieldFirst;
11526 break;
11527 case MSM_VIDC_INTERLACE_INTERLEAVE_FRAME_BOTTOMFIELDFIRST:
11528 drv_ctx.interlace = VDEC_InterlaceInterleaveFrameBottomFieldFirst;
11529 enable = OMX_InterlaceInterleaveFrameBottomFieldFirst;
11530 break;
11531 default:
11532 DEBUG_PRINT_LOW("default case - set to progressive");
11533 drv_ctx.interlace = VDEC_InterlaceFrameProgressive;
11534 }
11535 switch (payload->color_format) {
11536 case MSM_VIDC_HAL_INTERLACE_COLOR_FORMAT_NV12:
11537 interlace_color_format = (int)QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m;
11538 break;
11539 case MSM_VIDC_HAL_INTERLACE_COLOR_FORMAT_NV12_UBWC:
11540 interlace_color_format = (int)QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mCompressed;
11541 break;
11542 default:
11543 interlace_color_format = (int)QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m;
11544 DEBUG_PRINT_ERROR("Error - Unknown color format hint for interlaced frame");
11545 }
11546 }
11547
11548 if (m_enable_android_native_buffers) {
11549 DEBUG_PRINT_LOW("setMetaData INTERLACED format:%d color_format: %x enable:%d mbaff:%d",
11550 payload->format, interlace_color_format ,enable,
11551 (p_buf_hdr->nFlags & QOMX_VIDEO_BUFFERFLAG_MBAFF)?true:false);
11552
11553 setMetaData((private_handle_t *)native_buffer[buf_index].privatehandle,
11554 PP_PARAM_INTERLACED, (void*)&enable);
11555
11556 if (interlace_color_format == QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m) {
11557 setMetaData((private_handle_t *)native_buffer[buf_index].privatehandle,
11558 LINEAR_FORMAT, (void*)&interlace_color_format);
11559 } else if (interlace_color_format == QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mCompressed) {
11560 setMetaData((private_handle_t *)native_buffer[buf_index].privatehandle,
11561 LINEAR_FORMAT, NULL);
11562 }
11563 }
11564 if (client_extradata & OMX_INTERLACE_EXTRADATA) {
11565 append_interlace_extradata(p_extra, payload->format);
11566 p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + ALIGN(p_extra->nSize, 4));
11567 if (p_client_extra) {
11568 append_interlace_extradata(p_client_extra, payload->format);
11569 p_client_extra = (OMX_OTHER_EXTRADATATYPE *)
11570 (((OMX_U8 *)p_client_extra) + ALIGN(p_client_extra->nSize, 4));
11571 }
11572 }
11573 break;
11574 case MSM_VIDC_EXTRADATA_FRAME_RATE:
11575 struct msm_vidc_framerate_payload *frame_rate_payload;
11576 frame_rate_payload = (struct msm_vidc_framerate_payload *)(void *)data->data;
11577 frame_rate = frame_rate_payload->frame_rate;
11578 break;
11579 case MSM_VIDC_EXTRADATA_TIMESTAMP:
11580 struct msm_vidc_ts_payload *time_stamp_payload;
11581 time_stamp_payload = (struct msm_vidc_ts_payload *)(void *)data->data;
11582 time_stamp = time_stamp_payload->timestamp_lo;
11583 time_stamp |= ((unsigned long long)time_stamp_payload->timestamp_hi << 32);
11584 p_buf_hdr->nTimeStamp = time_stamp;
11585 break;
11586 case MSM_VIDC_EXTRADATA_NUM_CONCEALED_MB:
11587 struct msm_vidc_concealmb_payload *conceal_mb_payload;
11588 conceal_mb_payload = (struct msm_vidc_concealmb_payload *)(void *)data->data;
11589 num_MB_in_frame = ((drv_ctx.video_resolution.frame_width + 15) *
11590 (drv_ctx.video_resolution.frame_height + 15)) >> 8;
11591 num_conceal_MB = ((num_MB_in_frame > 0)?(conceal_mb_payload->num_mbs * 100 / num_MB_in_frame) : 0);
11592 break;
11593 case MSM_VIDC_EXTRADATA_INDEX:
11594 int *etype;
11595 etype = (int *)(void *)data->data;
11596 if (etype && *etype == MSM_VIDC_EXTRADATA_ASPECT_RATIO) {
11597 struct msm_vidc_aspect_ratio_payload *aspect_ratio_payload;
11598 aspect_ratio_payload = (struct msm_vidc_aspect_ratio_payload *)(++etype);
11599 if (aspect_ratio_payload) {
11600 ((struct vdec_output_frameinfo *)
11601 p_buf_hdr->pOutputPortPrivate)->aspect_ratio_info.par_width = aspect_ratio_payload->aspect_width;
11602 ((struct vdec_output_frameinfo *)
11603 p_buf_hdr->pOutputPortPrivate)->aspect_ratio_info.par_height = aspect_ratio_payload->aspect_height;
11604 }
11605 } else if (etype && *etype == MSM_VIDC_EXTRADATA_OUTPUT_CROP) {
11606 struct msm_vidc_output_crop_payload *output_crop_payload;
11607 output_crop_payload = (struct msm_vidc_output_crop_payload *)(++etype);
11608 if (output_crop_payload) {
11609 m_extradata_info.output_crop_rect.nLeft = output_crop_payload->left;
11610 m_extradata_info.output_crop_rect.nTop = output_crop_payload->top;
11611 m_extradata_info.output_crop_rect.nWidth = output_crop_payload->left + output_crop_payload->display_width;
11612 m_extradata_info.output_crop_rect.nHeight = output_crop_payload->top + output_crop_payload->display_height;
11613 m_extradata_info.output_width = output_crop_payload->width;
11614 m_extradata_info.output_height = output_crop_payload->height;
11615 m_extradata_info.output_crop_updated = OMX_TRUE;
11616 }
11617 }
11618 break;
11619 case MSM_VIDC_EXTRADATA_RECOVERY_POINT_SEI:
11620 struct msm_vidc_recoverysei_payload *recovery_sei_payload;
11621 recovery_sei_payload = (struct msm_vidc_recoverysei_payload *)(void *)data->data;
11622 recovery_sei_flags = recovery_sei_payload->flags;
11623 if (recovery_sei_flags != MSM_VIDC_FRAME_RECONSTRUCTION_CORRECT) {
11624 p_buf_hdr->nFlags |= OMX_BUFFERFLAG_DATACORRUPT;
11625 DEBUG_PRINT_HIGH("***************************************************");
11626 DEBUG_PRINT_HIGH("FillBufferDone: OMX_BUFFERFLAG_DATACORRUPT Received");
11627 DEBUG_PRINT_HIGH("***************************************************");
11628 }
11629 break;
11630 case MSM_VIDC_EXTRADATA_PANSCAN_WINDOW:
11631 panscan_payload = (struct msm_vidc_panscan_window_payload *)(void *)data->data;
11632 if (panscan_payload->num_panscan_windows > MAX_PAN_SCAN_WINDOWS) {
11633 DEBUG_PRINT_ERROR("Panscan windows are more than supported\n");
11634 DEBUG_PRINT_ERROR("Max supported = %d FW returned = %d\n",
11635 MAX_PAN_SCAN_WINDOWS, panscan_payload->num_panscan_windows);
11636 return;
11637 }
11638 break;
11639 case MSM_VIDC_EXTRADATA_MPEG2_SEQDISP:
11640 case MSM_VIDC_EXTRADATA_VUI_DISPLAY_INFO:
11641 case MSM_VIDC_EXTRADATA_VC1_SEQDISP:
11642 case MSM_VIDC_EXTRADATA_VPX_COLORSPACE_INFO:
11643 color_event = handle_color_space_info((void *)data->data, &color_space, &color_mdata, set_disp_color_aspects_only);
11644 break;
11645 case MSM_VIDC_EXTRADATA_S3D_FRAME_PACKING:
11646 struct msm_vidc_s3d_frame_packing_payload *s3d_frame_packing_payload;
11647 s3d_frame_packing_payload = (struct msm_vidc_s3d_frame_packing_payload *)(void *)data->data;
11648 switch (s3d_frame_packing_payload->fpa_type) {
11649 case MSM_VIDC_FRAMEPACK_SIDE_BY_SIDE:
11650 if (s3d_frame_packing_payload->content_interprtation_type == 1)
11651 stereo_output_mode = HAL_3D_SIDE_BY_SIDE_L_R;
11652 else if (s3d_frame_packing_payload->content_interprtation_type == 2)
11653 stereo_output_mode = HAL_3D_SIDE_BY_SIDE_R_L;
11654 else {
11655 DEBUG_PRINT_ERROR("Unsupported side-by-side framepacking type");
11656 stereo_output_mode = HAL_NO_3D;
11657 }
11658 break;
11659 case MSM_VIDC_FRAMEPACK_TOP_BOTTOM:
11660 stereo_output_mode = HAL_3D_TOP_BOTTOM;
11661 break;
11662 default:
11663 DEBUG_PRINT_ERROR("Unsupported framepacking type");
11664 stereo_output_mode = HAL_NO_3D;
11665 }
11666 DEBUG_PRINT_LOW("setMetaData FRAMEPACKING : fpa_type = %u, content_interprtation_type = %u, stereo_output_mode= %d",
11667 s3d_frame_packing_payload->fpa_type, s3d_frame_packing_payload->content_interprtation_type, stereo_output_mode);
11668 if (client_extradata & OMX_FRAMEPACK_EXTRADATA) {
11669 append_framepack_extradata(p_extra, s3d_frame_packing_payload);
11670 p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + ALIGN(p_extra->nSize, 4));
11671 if (p_client_extra) {
11672 append_framepack_extradata(p_client_extra, s3d_frame_packing_payload);
11673 p_client_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_client_extra) + ALIGN(p_client_extra->nSize, 4));
11674 }
11675 }
11676 break;
11677 case MSM_VIDC_EXTRADATA_FRAME_QP:
11678 struct msm_vidc_frame_qp_payload *qp_payload;
11679 qp_payload = (struct msm_vidc_frame_qp_payload*)(void *)data->data;
11680 if (client_extradata & OMX_QP_EXTRADATA) {
11681 append_qp_extradata(p_extra, qp_payload);
11682 p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + ALIGN(p_extra->nSize, 4));
11683 if (p_client_extra) {
11684 append_qp_extradata(p_client_extra, qp_payload);
11685 p_client_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_client_extra) + ALIGN(p_client_extra->nSize, 4));
11686 }
11687 }
11688 break;
11689 case MSM_VIDC_EXTRADATA_FRAME_BITS_INFO:
11690 struct msm_vidc_frame_bits_info_payload *bits_info_payload;
11691 bits_info_payload = (struct msm_vidc_frame_bits_info_payload*)(void *)data->data;
11692 if (client_extradata & OMX_BITSINFO_EXTRADATA) {
11693 append_bitsinfo_extradata(p_extra, bits_info_payload);
11694 p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + ALIGN(p_extra->nSize, 4));
11695 if (p_client_extra) {
11696 append_bitsinfo_extradata(p_client_extra, bits_info_payload);
11697 p_client_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_client_extra) + ALIGN(p_client_extra->nSize, 4));
11698 }
11699 }
11700 break;
11701 case MSM_VIDC_EXTRADATA_STREAM_USERDATA:
11702 if (client_extradata & OMX_EXTNUSER_EXTRADATA) {
11703 append_user_extradata(p_extra, data);
11704 p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + ALIGN(p_extra->nSize, 4));
11705 if (p_client_extra) {
11706 append_user_extradata(p_client_extra, data);
11707 p_client_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_client_extra) + ALIGN(p_client_extra->nSize, 4));
11708 }
11709 }
11710 break;
11711 case MSM_VIDC_EXTRADATA_VQZIP_SEI:
11712 struct msm_vidc_vqzip_sei_payload *vqzip_payload;
11713 vqzip_payload = (struct msm_vidc_vqzip_sei_payload*)(void *)data->data;
11714 if (client_extradata & OMX_VQZIPSEI_EXTRADATA) {
11715 p_buf_hdr->nFlags |= OMX_BUFFERFLAG_EXTRADATA;
11716 append_vqzip_extradata(p_extra, vqzip_payload);
11717 p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + ALIGN(p_extra->nSize, 4));
11718 if (p_client_extra) {
11719 append_vqzip_extradata(p_client_extra, vqzip_payload);
11720 p_client_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_client_extra) + ALIGN(p_client_extra->nSize, 4));
11721 }
11722 }
11723 break;
11724 case MSM_VIDC_EXTRADATA_CONTENT_LIGHT_LEVEL_SEI:
11725
11726 internal_hdr_info_changed_flag |= handle_content_light_level_info((void*)data->data,
11727 &(color_mdata.contentLightLevel));
11728 break;
11729 case MSM_VIDC_EXTRADATA_MASTERING_DISPLAY_COLOUR_SEI:
11730 internal_hdr_info_changed_flag |= handle_mastering_display_color_info((void*)data->data,
11731 &(color_mdata.masteringDisplayInfo));
11732 break;
11733 default:
11734 DEBUG_PRINT_LOW("Unrecognized extradata");
11735 goto unrecognized_extradata;
11736 }
11737 consumed_len += data->nSize;
11738 data = (OMX_OTHER_EXTRADATATYPE *)((char *)data + data->nSize);
11739 }
11740 if (client_extradata & OMX_FRAMEINFO_EXTRADATA) {
11741 p_buf_hdr->nFlags |= OMX_BUFFERFLAG_EXTRADATA;
11742 append_frame_info_extradata(p_extra,
11743 num_conceal_MB, recovery_sei_flags, ((struct vdec_output_frameinfo *)p_buf_hdr->pOutputPortPrivate)->pic_type, frame_rate,
11744 time_stamp, panscan_payload,&((struct vdec_output_frameinfo *)
11745 p_buf_hdr->pOutputPortPrivate)->aspect_ratio_info);
11746 p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + ALIGN(p_extra->nSize, 4));
11747 if (p_client_extra) {
11748 append_frame_info_extradata(p_client_extra,
11749 num_conceal_MB, recovery_sei_flags, ((struct vdec_output_frameinfo *)p_buf_hdr->pOutputPortPrivate)->pic_type, frame_rate,
11750 time_stamp, panscan_payload,&((struct vdec_output_frameinfo *)
11751 p_buf_hdr->pOutputPortPrivate)->aspect_ratio_info);
11752 p_client_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_client_extra) + ALIGN(p_client_extra->nSize, 4));
11753 }
11754 }
11755 if (client_extradata & OMX_FRAMEDIMENSION_EXTRADATA) {
11756 append_frame_dimension_extradata(p_extra);
11757 p_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_extra) + ALIGN(p_extra->nSize, 4));
11758 if (p_client_extra) {
11759 append_frame_dimension_extradata(p_client_extra);
11760 p_client_extra = (OMX_OTHER_EXTRADATATYPE *) (((OMX_U8 *) p_client_extra) + ALIGN(p_client_extra->nSize, 4));
11761 }
11762 }
11763
11764 if(internal_hdr_info_changed_flag) {
11765 print_debug_hdr_color_info(&(m_internal_hdr_info.sInfo), "Internal");
11766 print_debug_hdr_color_info(&(m_client_hdr_info.sInfo), "Client");
11767 memcpy(&m_color_mdata, &color_mdata, sizeof(ColorMetaData));
11768 auto_lock lock(m_hdr_info_client_lock);
11769 m_change_client_hdr_info = true;
11770 if(!color_event) {
11771 DEBUG_PRINT_HIGH("Initiating PORT Reconfig due to HDR Info Change");
11772 post_event(OMX_CORE_OUTPUT_PORT_INDEX,
11773 OMX_QTIIndexConfigDescribeHDRColorInfo,
11774 OMX_COMPONENT_GENERATE_PORT_RECONFIG);
11775 }
11776 }
11777
11778 if (m_enable_android_native_buffers) {
11779 if (set_disp_color_aspects_only) {
11780 print_debug_hdr_color_info_mdata(&m_color_mdata);
11781 set_colormetadata_in_handle(&m_color_mdata, buf_index);
11782 } else {
11783 DEBUG_PRINT_HIGH("setMetaData for Color Space = 0x%x (601=%u FR=%u 709=%u)", color_space, ITU_R_601, ITU_R_601_FR, ITU_R_709);
11784 set_colorspace_in_handle(color_space, buf_index);
11785 }
11786 }
11787
11788 }
11789 unrecognized_extradata:
11790 if (client_extradata && p_extra) {
11791 p_buf_hdr->nFlags |= OMX_BUFFERFLAG_EXTRADATA;
11792 append_terminator_extradata(p_extra);
11793 if (p_client_extra) {
11794 append_terminator_extradata(p_client_extra);
11795 }
11796 }
11797 if (secure_mode && p_extradata && m_other_extradata) {
11798 struct vdec_output_frameinfo *ptr_extradatabuff = NULL;
11799 memcpy(p_extradata, m_other_extradata, drv_ctx.extradata_info.buffer_size);
11800 ptr_extradatabuff = (struct vdec_output_frameinfo *)p_buf_hdr->pOutputPortPrivate;
11801 ptr_extradatabuff->metadata_info.metabufaddr = (void *)p_extradata;
11802 ptr_extradatabuff->metadata_info.size = drv_ctx.extradata_info.buffer_size;
11803 ptr_extradatabuff->metadata_info.fd = drv_ctx.extradata_info.ion.fd_ion_data.fd;
11804 ptr_extradatabuff->metadata_info.offset = buf_index * drv_ctx.extradata_info.buffer_size;
11805 ptr_extradatabuff->metadata_info.buffer_size = drv_ctx.extradata_info.size;
11806 }
11807 return;
11808 }
11809
enable_extradata(OMX_U64 requested_extradata,bool is_internal,bool enable)11810 OMX_ERRORTYPE omx_vdec::enable_extradata(OMX_U64 requested_extradata,
11811 bool is_internal, bool enable)
11812 {
11813 OMX_ERRORTYPE ret = OMX_ErrorNone;
11814 struct v4l2_control control;
11815 if (m_state != OMX_StateLoaded) {
11816 DEBUG_PRINT_ERROR("ERROR: enable extradata allowed in Loaded state only");
11817 return OMX_ErrorIncorrectStateOperation;
11818 }
11819 DEBUG_PRINT_HIGH("NOTE: enable_extradata: actual[%u] requested[%u] enable[%d], is_internal: %d",
11820 (unsigned int)client_extradata, (unsigned int)requested_extradata, enable, is_internal);
11821
11822 if (!is_internal) {
11823 if (enable)
11824 client_extradata |= requested_extradata;
11825 else
11826 client_extradata = client_extradata & ~requested_extradata;
11827 }
11828
11829 if (enable) {
11830 if (requested_extradata & OMX_INTERLACE_EXTRADATA) {
11831 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11832 control.value = V4L2_MPEG_VIDC_EXTRADATA_INTERLACE_VIDEO;
11833 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11834 DEBUG_PRINT_HIGH("Failed to set interlaced extradata."
11835 " Quality of interlaced clips might be impacted.");
11836 }
11837 }
11838 if (requested_extradata & OMX_FRAMEINFO_EXTRADATA) {
11839 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11840 control.value = V4L2_MPEG_VIDC_EXTRADATA_FRAME_RATE;
11841 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11842 DEBUG_PRINT_HIGH("Failed to set framerate extradata");
11843 }
11844 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11845 control.value = V4L2_MPEG_VIDC_EXTRADATA_NUM_CONCEALED_MB;
11846 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11847 DEBUG_PRINT_HIGH("Failed to set concealed MB extradata");
11848 }
11849 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11850 control.value = V4L2_MPEG_VIDC_EXTRADATA_RECOVERY_POINT_SEI;
11851 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11852 DEBUG_PRINT_HIGH("Failed to set recovery point SEI extradata");
11853 }
11854 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11855 control.value = V4L2_MPEG_VIDC_EXTRADATA_PANSCAN_WINDOW;
11856 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11857 DEBUG_PRINT_HIGH("Failed to set panscan extradata");
11858 }
11859 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11860 control.value = V4L2_MPEG_VIDC_EXTRADATA_ASPECT_RATIO;
11861 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11862 DEBUG_PRINT_HIGH("Failed to set panscan extradata");
11863 }
11864 if (output_capability == V4L2_PIX_FMT_MPEG2) {
11865 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11866 control.value = V4L2_MPEG_VIDC_EXTRADATA_MPEG2_SEQDISP;
11867 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11868 DEBUG_PRINT_HIGH("Failed to set panscan extradata");
11869 }
11870 }
11871 }
11872 if (requested_extradata & OMX_TIMEINFO_EXTRADATA) {
11873 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11874 control.value = V4L2_MPEG_VIDC_EXTRADATA_TIMESTAMP;
11875 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11876 DEBUG_PRINT_HIGH("Failed to set timeinfo extradata");
11877 }
11878 }
11879 if (!secure_mode && (requested_extradata & OMX_FRAMEPACK_EXTRADATA)) {
11880 if (output_capability == V4L2_PIX_FMT_H264) {
11881 DEBUG_PRINT_HIGH("enable OMX_FRAMEPACK_EXTRADATA");
11882 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11883 control.value = V4L2_MPEG_VIDC_EXTRADATA_S3D_FRAME_PACKING;
11884 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11885 DEBUG_PRINT_HIGH("Failed to set S3D_FRAME_PACKING extradata");
11886 }
11887 } else {
11888 DEBUG_PRINT_HIGH("OMX_FRAMEPACK_EXTRADATA supported for H264 only");
11889 }
11890 }
11891 if (requested_extradata & OMX_QP_EXTRADATA) {
11892 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11893 control.value = V4L2_MPEG_VIDC_EXTRADATA_FRAME_QP;
11894 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11895 DEBUG_PRINT_HIGH("Failed to set QP extradata");
11896 }
11897 }
11898 if (requested_extradata & OMX_BITSINFO_EXTRADATA) {
11899 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11900 control.value = V4L2_MPEG_VIDC_EXTRADATA_FRAME_BITS_INFO;
11901 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11902 DEBUG_PRINT_HIGH("Failed to set frame bits info extradata");
11903 }
11904 }
11905 if (!secure_mode && (requested_extradata & OMX_EXTNUSER_EXTRADATA)) {
11906 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11907 control.value = V4L2_MPEG_VIDC_EXTRADATA_STREAM_USERDATA;
11908 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11909 DEBUG_PRINT_HIGH("Failed to set stream userdata extradata");
11910 }
11911 }
11912 if (requested_extradata & OMX_VQZIPSEI_EXTRADATA) {
11913 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11914 control.value = V4L2_MPEG_VIDC_EXTRADATA_VQZIP_SEI;
11915 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11916 DEBUG_PRINT_HIGH("Failed to set VQZip SEI extradata");
11917 }
11918 client_extradata |= OMX_VQZIPSEI_EXTRADATA;
11919
11920 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11921 control.value = V4L2_MPEG_VIDC_EXTRADATA_FRAME_QP;
11922 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11923 DEBUG_PRINT_HIGH("Failed to set QP extradata");
11924 }
11925 client_extradata |= OMX_QP_EXTRADATA;
11926 }
11927 if (requested_extradata & OMX_OUTPUTCROP_EXTRADATA) {
11928 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11929 control.value = V4L2_MPEG_VIDC_EXTRADATA_OUTPUT_CROP;
11930 DEBUG_PRINT_LOW("Enable output crop extra data");
11931 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11932 DEBUG_PRINT_HIGH("Failed to set output crop extradata");
11933 }
11934 }
11935 if (requested_extradata & OMX_DISPLAY_INFO_EXTRADATA) {
11936 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11937 switch(output_capability) {
11938 case V4L2_PIX_FMT_H264:
11939 case V4L2_PIX_FMT_HEVC:
11940 control.value = V4L2_MPEG_VIDC_EXTRADATA_VUI_DISPLAY;
11941 break;
11942 case CODEC_TYPE_MPEG2:
11943 control.value = V4L2_MPEG_VIDC_EXTRADATA_MPEG2_SEQDISP;
11944 break;
11945 case V4L2_PIX_FMT_VP8:
11946 case V4L2_PIX_FMT_VP9:
11947 control.value = V4L2_MPEG_VIDC_EXTRADATA_VPX_COLORSPACE;
11948 break;
11949 case V4L2_PIX_FMT_VC1_ANNEX_G:
11950 case V4L2_PIX_FMT_VC1_ANNEX_L:
11951 control.value = V4L2_MPEG_VIDC_EXTRADATA_VC1_SEQDISP;
11952 break;
11953 default:
11954 DEBUG_PRINT_HIGH("Don't support Disp info for this codec : %s", drv_ctx.kind);
11955 return ret;
11956 }
11957
11958 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11959 DEBUG_PRINT_HIGH("Failed to set Display info extradata");
11960 }
11961 }
11962 if (requested_extradata & OMX_HDR_COLOR_INFO_EXTRADATA) {
11963 control.id = V4L2_CID_MPEG_VIDC_VIDEO_EXTRADATA;
11964 if (output_capability == V4L2_PIX_FMT_H264 ||
11965 output_capability == V4L2_PIX_FMT_HEVC) {
11966 control.value = V4L2_MPEG_VIDC_EXTRADATA_DISPLAY_COLOUR_SEI;
11967 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11968 DEBUG_PRINT_HIGH("Failed to set Display Colour SEI extradata");
11969 }
11970 control.value = V4L2_MPEG_VIDC_EXTRADATA_CONTENT_LIGHT_LEVEL_SEI;
11971 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
11972 DEBUG_PRINT_HIGH("Failed to set Content Light Level SEI extradata");
11973 }
11974 }
11975 }
11976 }
11977 ret = get_buffer_req(&drv_ctx.op_buf);
11978 return ret;
11979 }
11980
count_MB_in_extradata(OMX_OTHER_EXTRADATATYPE * extra)11981 OMX_U32 omx_vdec::count_MB_in_extradata(OMX_OTHER_EXTRADATATYPE *extra)
11982 {
11983 OMX_U32 num_MB = 0, byte_count = 0, num_MB_in_frame = 0;
11984 OMX_U8 *data_ptr = extra->data, data = 0;
11985 while (byte_count < extra->nDataSize) {
11986 data = *data_ptr;
11987 while (data) {
11988 num_MB += (data&0x01);
11989 data >>= 1;
11990 }
11991 data_ptr++;
11992 byte_count++;
11993 }
11994 num_MB_in_frame = ((drv_ctx.video_resolution.frame_width + 15) *
11995 (drv_ctx.video_resolution.frame_height + 15)) >> 8;
11996 return ((num_MB_in_frame > 0)?(num_MB * 100 / num_MB_in_frame) : 0);
11997 }
11998
print_debug_extradata(OMX_OTHER_EXTRADATATYPE * extra)11999 void omx_vdec::print_debug_extradata(OMX_OTHER_EXTRADATATYPE *extra)
12000 {
12001 if (!m_debug_extradata || !extra)
12002 return;
12003
12004
12005 DEBUG_PRINT_HIGH(
12006 "============== Extra Data ==============\n"
12007 " Size: %u\n"
12008 " Version: %u\n"
12009 " PortIndex: %u\n"
12010 " Type: %x\n"
12011 " DataSize: %u",
12012 (unsigned int)extra->nSize, (unsigned int)extra->nVersion.nVersion,
12013 (unsigned int)extra->nPortIndex, extra->eType, (unsigned int)extra->nDataSize);
12014
12015 if (extra->eType == (OMX_EXTRADATATYPE)OMX_ExtraDataInterlaceFormat) {
12016 OMX_STREAMINTERLACEFORMAT *intfmt = (OMX_STREAMINTERLACEFORMAT *)(void *)extra->data;
12017 DEBUG_PRINT_HIGH(
12018 "------ Interlace Format ------\n"
12019 " Size: %u\n"
12020 " Version: %u\n"
12021 " PortIndex: %u\n"
12022 " Is Interlace Format: %d\n"
12023 " Interlace Formats: %u\n"
12024 "=========== End of Interlace ===========",
12025 (unsigned int)intfmt->nSize, (unsigned int)intfmt->nVersion.nVersion, (unsigned int)intfmt->nPortIndex,
12026 intfmt->bInterlaceFormat, (unsigned int)intfmt->nInterlaceFormats);
12027 } else if (extra->eType == (OMX_EXTRADATATYPE)OMX_ExtraDataFrameInfo) {
12028 OMX_QCOM_EXTRADATA_FRAMEINFO *fminfo = (OMX_QCOM_EXTRADATA_FRAMEINFO *)(void *)extra->data;
12029
12030 DEBUG_PRINT_HIGH(
12031 "-------- Frame Format --------\n"
12032 " Picture Type: %d\n"
12033 " Interlace Type: %d\n"
12034 " Pan Scan Total Frame Num: %u\n"
12035 " Concealed Macro Blocks: %u\n"
12036 " Recovery SEI Flag: %u\n"
12037 " frame rate: %u\n"
12038 " Time Stamp: %llu\n"
12039 " Aspect Ratio X: %u\n"
12040 " Aspect Ratio Y: %u",
12041 fminfo->ePicType,
12042 fminfo->interlaceType,
12043 (unsigned int)fminfo->panScan.numWindows,
12044 (unsigned int)fminfo->nConcealedMacroblocks,
12045 (unsigned int)fminfo->nRecoverySeiFlag,
12046 (unsigned int)fminfo->nFrameRate,
12047 fminfo->nTimeStamp,
12048 (unsigned int)fminfo->aspectRatio.aspectRatioX,
12049 (unsigned int)fminfo->aspectRatio.aspectRatioY);
12050
12051 for (OMX_U32 i = 0; i < fminfo->panScan.numWindows; i++) {
12052 DEBUG_PRINT_HIGH(
12053 "------------------------------"
12054 " Pan Scan Frame Num: %u\n"
12055 " Rectangle x: %d\n"
12056 " Rectangle y: %d\n"
12057 " Rectangle dx: %d\n"
12058 " Rectangle dy: %d",
12059 (unsigned int)i, (unsigned int)fminfo->panScan.window[i].x, (unsigned int)fminfo->panScan.window[i].y,
12060 (unsigned int)fminfo->panScan.window[i].dx, (unsigned int)fminfo->panScan.window[i].dy);
12061 }
12062
12063 DEBUG_PRINT_HIGH("========= End of Frame Format ==========");
12064 } else if (extra->eType == (OMX_EXTRADATATYPE)OMX_ExtraDataFramePackingArrangement) {
12065 OMX_QCOM_FRAME_PACK_ARRANGEMENT *framepack = (OMX_QCOM_FRAME_PACK_ARRANGEMENT *)(void *)extra->data;
12066 DEBUG_PRINT_HIGH(
12067 "------------------ Framepack Format ----------\n"
12068 " id: %u \n"
12069 " cancel_flag: %u \n"
12070 " type: %u \n"
12071 " quincunx_sampling_flagFormat: %u \n"
12072 " content_interpretation_type: %u \n"
12073 " spatial_flipping_flag: %u \n"
12074 " frame0_flipped_flag: %u \n"
12075 " field_views_flag: %u \n"
12076 " current_frame_is_frame0_flag: %u \n"
12077 " frame0_self_contained_flag: %u \n"
12078 " frame1_self_contained_flag: %u \n"
12079 " frame0_grid_position_x: %u \n"
12080 " frame0_grid_position_y: %u \n"
12081 " frame1_grid_position_x: %u \n"
12082 " frame1_grid_position_y: %u \n"
12083 " reserved_byte: %u \n"
12084 " repetition_period: %u \n"
12085 " extension_flag: %u \n"
12086 "================== End of Framepack ===========",
12087 (unsigned int)framepack->id,
12088 (unsigned int)framepack->cancel_flag,
12089 (unsigned int)framepack->type,
12090 (unsigned int)framepack->quincunx_sampling_flag,
12091 (unsigned int)framepack->content_interpretation_type,
12092 (unsigned int)framepack->spatial_flipping_flag,
12093 (unsigned int)framepack->frame0_flipped_flag,
12094 (unsigned int)framepack->field_views_flag,
12095 (unsigned int)framepack->current_frame_is_frame0_flag,
12096 (unsigned int)framepack->frame0_self_contained_flag,
12097 (unsigned int)framepack->frame1_self_contained_flag,
12098 (unsigned int)framepack->frame0_grid_position_x,
12099 (unsigned int)framepack->frame0_grid_position_y,
12100 (unsigned int)framepack->frame1_grid_position_x,
12101 (unsigned int)framepack->frame1_grid_position_y,
12102 (unsigned int)framepack->reserved_byte,
12103 (unsigned int)framepack->repetition_period,
12104 (unsigned int)framepack->extension_flag);
12105 } else if (extra->eType == (OMX_EXTRADATATYPE)OMX_ExtraDataQP) {
12106 OMX_QCOM_EXTRADATA_QP * qp = (OMX_QCOM_EXTRADATA_QP *)(void *)extra->data;
12107 DEBUG_PRINT_HIGH(
12108 "---- QP (Frame quantization parameter) ----\n"
12109 " Frame QP: %u \n"
12110 "================ End of QP ================\n",
12111 (unsigned int)qp->nQP);
12112 } else if (extra->eType == (OMX_EXTRADATATYPE)OMX_ExtraDataInputBitsInfo) {
12113 OMX_QCOM_EXTRADATA_BITS_INFO * bits = (OMX_QCOM_EXTRADATA_BITS_INFO *)(void *)extra->data;
12114 DEBUG_PRINT_HIGH(
12115 "--------- Input bits information --------\n"
12116 " Header bits: %u \n"
12117 " Frame bits: %u \n"
12118 "===== End of Input bits information =====\n",
12119 (unsigned int)bits->header_bits, (unsigned int)bits->frame_bits);
12120 } else if (extra->eType == (OMX_EXTRADATATYPE)OMX_ExtraDataMP2UserData) {
12121 OMX_QCOM_EXTRADATA_USERDATA *userdata = (OMX_QCOM_EXTRADATA_USERDATA *)(void *)extra->data;
12122 OMX_U8 *data_ptr = (OMX_U8 *)userdata->data;
12123 OMX_U32 userdata_size = extra->nDataSize - sizeof(userdata->type);
12124 OMX_U32 i = 0;
12125 DEBUG_PRINT_HIGH(
12126 "-------------- Userdata -------------\n"
12127 " Stream userdata type: %u\n"
12128 " userdata size: %u\n"
12129 " STREAM_USERDATA:",
12130 (unsigned int)userdata->type, (unsigned int)userdata_size);
12131 for (i = 0; i < userdata_size; i+=4) {
12132 DEBUG_PRINT_HIGH(" %x %x %x %x",
12133 data_ptr[i], data_ptr[i+1],
12134 data_ptr[i+2], data_ptr[i+3]);
12135 }
12136 DEBUG_PRINT_HIGH(
12137 "=========== End of Userdata ===========");
12138 } else if (extra->eType == (OMX_EXTRADATATYPE)OMX_ExtraDataVQZipSEI) {
12139 OMX_QCOM_EXTRADATA_VQZIPSEI *vq = (OMX_QCOM_EXTRADATA_VQZIPSEI *)(void *)extra->data;
12140 DEBUG_PRINT_HIGH(
12141 "-------------- VQZip -------------\n"
12142 " Size: %u\n",
12143 (unsigned int)vq->nSize);
12144 DEBUG_PRINT_HIGH( "=========== End of VQZip ===========");
12145 } else if (extra->eType == OMX_ExtraDataNone) {
12146 DEBUG_PRINT_HIGH("========== End of Terminator ===========");
12147 } else {
12148 DEBUG_PRINT_HIGH("======= End of Driver Extradata ========");
12149 }
12150 }
12151
append_interlace_extradata(OMX_OTHER_EXTRADATATYPE * extra,OMX_U32 interlaced_format_type)12152 void omx_vdec::append_interlace_extradata(OMX_OTHER_EXTRADATATYPE *extra,
12153 OMX_U32 interlaced_format_type)
12154 {
12155 OMX_STREAMINTERLACEFORMAT *interlace_format;
12156
12157 if (!(client_extradata & OMX_INTERLACE_EXTRADATA)) {
12158 return;
12159 }
12160 if (!extra) {
12161 DEBUG_PRINT_ERROR("Error: append_interlace_extradata - invalid input");
12162 return;
12163 }
12164 extra->nSize = OMX_INTERLACE_EXTRADATA_SIZE;
12165 extra->nVersion.nVersion = OMX_SPEC_VERSION;
12166 extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12167 extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataInterlaceFormat;
12168 extra->nDataSize = sizeof(OMX_STREAMINTERLACEFORMAT);
12169 interlace_format = (OMX_STREAMINTERLACEFORMAT *)(void *)extra->data;
12170 interlace_format->nSize = sizeof(OMX_STREAMINTERLACEFORMAT);
12171 interlace_format->nVersion.nVersion = OMX_SPEC_VERSION;
12172 interlace_format->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12173
12174 if (interlaced_format_type == MSM_VIDC_INTERLACE_FRAME_PROGRESSIVE) {
12175 interlace_format->bInterlaceFormat = OMX_FALSE;
12176 interlace_format->nInterlaceFormats = OMX_InterlaceFrameProgressive;
12177 drv_ctx.interlace = VDEC_InterlaceFrameProgressive;
12178 } else if (interlaced_format_type == MSM_VIDC_INTERLACE_INTERLEAVE_FRAME_TOPFIELDFIRST) {
12179 interlace_format->bInterlaceFormat = OMX_TRUE;
12180 interlace_format->nInterlaceFormats = OMX_InterlaceInterleaveFrameTopFieldFirst;
12181 drv_ctx.interlace = VDEC_InterlaceInterleaveFrameTopFieldFirst;
12182 } else if (interlaced_format_type == MSM_VIDC_INTERLACE_INTERLEAVE_FRAME_BOTTOMFIELDFIRST) {
12183 interlace_format->bInterlaceFormat = OMX_TRUE;
12184 interlace_format->nInterlaceFormats = OMX_InterlaceInterleaveFrameBottomFieldFirst;
12185 drv_ctx.interlace = VDEC_InterlaceInterleaveFrameBottomFieldFirst;
12186 } else {
12187 //default case - set to progressive
12188 interlace_format->bInterlaceFormat = OMX_FALSE;
12189 interlace_format->nInterlaceFormats = OMX_InterlaceFrameProgressive;
12190 drv_ctx.interlace = VDEC_InterlaceFrameProgressive;
12191 }
12192 print_debug_extradata(extra);
12193 }
12194
append_frame_dimension_extradata(OMX_OTHER_EXTRADATATYPE * extra)12195 void omx_vdec::append_frame_dimension_extradata(OMX_OTHER_EXTRADATATYPE *extra)
12196 {
12197 OMX_QCOM_EXTRADATA_FRAMEDIMENSION *frame_dimension;
12198 if (!(client_extradata & OMX_FRAMEDIMENSION_EXTRADATA)) {
12199 return;
12200 }
12201 extra->nSize = OMX_FRAMEDIMENSION_EXTRADATA_SIZE;
12202 extra->nVersion.nVersion = OMX_SPEC_VERSION;
12203 extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12204 extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataFrameDimension;
12205 extra->nDataSize = sizeof(OMX_QCOM_EXTRADATA_FRAMEDIMENSION);
12206 frame_dimension = (OMX_QCOM_EXTRADATA_FRAMEDIMENSION *)(void *)extra->data;
12207 frame_dimension->nDecWidth = rectangle.nLeft;
12208 frame_dimension->nDecHeight = rectangle.nTop;
12209 frame_dimension->nActualWidth = rectangle.nWidth;
12210 frame_dimension->nActualHeight = rectangle.nHeight;
12211 }
12212
fill_aspect_ratio_info(struct vdec_aspectratioinfo * aspect_ratio_info,OMX_QCOM_EXTRADATA_FRAMEINFO * frame_info)12213 void omx_vdec::fill_aspect_ratio_info(
12214 struct vdec_aspectratioinfo *aspect_ratio_info,
12215 OMX_QCOM_EXTRADATA_FRAMEINFO *frame_info)
12216 {
12217 m_extradata = frame_info;
12218 m_extradata->aspectRatio.aspectRatioX = aspect_ratio_info->par_width;
12219 m_extradata->aspectRatio.aspectRatioY = aspect_ratio_info->par_height;
12220 DEBUG_PRINT_LOW("aspectRatioX %u aspectRatioY %u", (unsigned int)m_extradata->aspectRatio.aspectRatioX,
12221 (unsigned int)m_extradata->aspectRatio.aspectRatioY);
12222 }
12223
append_frame_info_extradata(OMX_OTHER_EXTRADATATYPE * extra,OMX_U32 num_conceal_mb,OMX_U32 recovery_sei_flag,OMX_U32 picture_type,OMX_U32 frame_rate,OMX_TICKS time_stamp,struct msm_vidc_panscan_window_payload * panscan_payload,struct vdec_aspectratioinfo * aspect_ratio_info)12224 void omx_vdec::append_frame_info_extradata(OMX_OTHER_EXTRADATATYPE *extra,
12225 OMX_U32 num_conceal_mb, OMX_U32 recovery_sei_flag, OMX_U32 picture_type, OMX_U32 frame_rate,
12226 OMX_TICKS time_stamp, struct msm_vidc_panscan_window_payload *panscan_payload,
12227 struct vdec_aspectratioinfo *aspect_ratio_info)
12228 {
12229 OMX_QCOM_EXTRADATA_FRAMEINFO *frame_info = NULL;
12230 struct msm_vidc_panscan_window *panscan_window;
12231 if (!(client_extradata & OMX_FRAMEINFO_EXTRADATA)) {
12232 return;
12233 }
12234 extra->nSize = OMX_FRAMEINFO_EXTRADATA_SIZE;
12235 extra->nVersion.nVersion = OMX_SPEC_VERSION;
12236 extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12237 extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataFrameInfo;
12238 extra->nDataSize = sizeof(OMX_QCOM_EXTRADATA_FRAMEINFO);
12239 frame_info = (OMX_QCOM_EXTRADATA_FRAMEINFO *)(void *)extra->data;
12240 switch (picture_type) {
12241 case PICTURE_TYPE_I:
12242 frame_info->ePicType = OMX_VIDEO_PictureTypeI;
12243 break;
12244 case PICTURE_TYPE_P:
12245 frame_info->ePicType = OMX_VIDEO_PictureTypeP;
12246 break;
12247 case PICTURE_TYPE_B:
12248 frame_info->ePicType = OMX_VIDEO_PictureTypeB;
12249 break;
12250 default:
12251 frame_info->ePicType = (OMX_VIDEO_PICTURETYPE)0;
12252 }
12253 if (drv_ctx.interlace == VDEC_InterlaceInterleaveFrameTopFieldFirst)
12254 frame_info->interlaceType = OMX_QCOM_InterlaceInterleaveFrameTopFieldFirst;
12255 else if (drv_ctx.interlace == VDEC_InterlaceInterleaveFrameBottomFieldFirst)
12256 frame_info->interlaceType = OMX_QCOM_InterlaceInterleaveFrameBottomFieldFirst;
12257 else
12258 frame_info->interlaceType = OMX_QCOM_InterlaceFrameProgressive;
12259 memset(&frame_info->aspectRatio, 0, sizeof(frame_info->aspectRatio));
12260 frame_info->nConcealedMacroblocks = num_conceal_mb;
12261 frame_info->nRecoverySeiFlag = recovery_sei_flag;
12262 frame_info->nFrameRate = frame_rate;
12263 frame_info->nTimeStamp = time_stamp;
12264 frame_info->panScan.numWindows = 0;
12265 if (output_capability == V4L2_PIX_FMT_MPEG2) {
12266 if (m_disp_hor_size && m_disp_vert_size) {
12267 frame_info->displayAspectRatio.displayHorizontalSize = m_disp_hor_size;
12268 frame_info->displayAspectRatio.displayVerticalSize = m_disp_vert_size;
12269 } else {
12270 frame_info->displayAspectRatio.displayHorizontalSize = 0;
12271 frame_info->displayAspectRatio.displayVerticalSize = 0;
12272 }
12273 }
12274
12275 if (panscan_payload) {
12276 frame_info->panScan.numWindows = panscan_payload->num_panscan_windows;
12277 panscan_window = &panscan_payload->wnd[0];
12278 for (OMX_U32 i = 0; i < frame_info->panScan.numWindows; i++) {
12279 frame_info->panScan.window[i].x = panscan_window->panscan_window_width;
12280 frame_info->panScan.window[i].y = panscan_window->panscan_window_height;
12281 frame_info->panScan.window[i].dx = panscan_window->panscan_width_offset;
12282 frame_info->panScan.window[i].dy = panscan_window->panscan_height_offset;
12283 panscan_window++;
12284 }
12285 }
12286 fill_aspect_ratio_info(aspect_ratio_info, frame_info);
12287 print_debug_extradata(extra);
12288 }
12289
append_portdef_extradata(OMX_OTHER_EXTRADATATYPE * extra)12290 void omx_vdec::append_portdef_extradata(OMX_OTHER_EXTRADATATYPE *extra)
12291 {
12292 OMX_PARAM_PORTDEFINITIONTYPE *portDefn = NULL;
12293 extra->nSize = OMX_PORTDEF_EXTRADATA_SIZE;
12294 extra->nVersion.nVersion = OMX_SPEC_VERSION;
12295 extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12296 extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataPortDef;
12297 extra->nDataSize = sizeof(OMX_PARAM_PORTDEFINITIONTYPE);
12298 portDefn = (OMX_PARAM_PORTDEFINITIONTYPE *)(void *)extra->data;
12299 *portDefn = m_port_def;
12300 DEBUG_PRINT_LOW("append_portdef_extradata height = %u width = %u "
12301 "stride = %u sliceheight = %u",(unsigned int)portDefn->format.video.nFrameHeight,
12302 (unsigned int)portDefn->format.video.nFrameWidth,
12303 (unsigned int)portDefn->format.video.nStride,
12304 (unsigned int)portDefn->format.video.nSliceHeight);
12305 }
12306
append_framepack_extradata(OMX_OTHER_EXTRADATATYPE * extra,struct msm_vidc_s3d_frame_packing_payload * s3d_frame_packing_payload)12307 void omx_vdec::append_framepack_extradata(OMX_OTHER_EXTRADATATYPE *extra,
12308 struct msm_vidc_s3d_frame_packing_payload *s3d_frame_packing_payload)
12309 {
12310 OMX_QCOM_FRAME_PACK_ARRANGEMENT *framepack;
12311 if (FRAME_PACK_SIZE*sizeof(OMX_U32) != sizeof(struct msm_vidc_s3d_frame_packing_payload)) {
12312 DEBUG_PRINT_ERROR("frame packing size mismatch");
12313 return;
12314 }
12315 extra->nSize = OMX_FRAMEPACK_EXTRADATA_SIZE;
12316 extra->nVersion.nVersion = OMX_SPEC_VERSION;
12317 extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12318 extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataFramePackingArrangement;
12319 extra->nDataSize = sizeof(OMX_QCOM_FRAME_PACK_ARRANGEMENT);
12320 framepack = (OMX_QCOM_FRAME_PACK_ARRANGEMENT *)(void *)extra->data;
12321 framepack->nSize = sizeof(OMX_QCOM_FRAME_PACK_ARRANGEMENT);
12322 framepack->nVersion.nVersion = OMX_SPEC_VERSION;
12323 framepack->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12324 memcpy(&framepack->id, s3d_frame_packing_payload,
12325 sizeof(struct msm_vidc_s3d_frame_packing_payload));
12326 memcpy(&m_frame_pack_arrangement, framepack,
12327 sizeof(OMX_QCOM_FRAME_PACK_ARRANGEMENT));
12328 print_debug_extradata(extra);
12329 }
12330
append_qp_extradata(OMX_OTHER_EXTRADATATYPE * extra,struct msm_vidc_frame_qp_payload * qp_payload)12331 void omx_vdec::append_qp_extradata(OMX_OTHER_EXTRADATATYPE *extra,
12332 struct msm_vidc_frame_qp_payload *qp_payload)
12333 {
12334 OMX_QCOM_EXTRADATA_QP * qp = NULL;
12335 if (!qp_payload) {
12336 DEBUG_PRINT_ERROR("QP payload is NULL");
12337 return;
12338 }
12339 extra->nSize = OMX_QP_EXTRADATA_SIZE;
12340 extra->nVersion.nVersion = OMX_SPEC_VERSION;
12341 extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12342 extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataQP;
12343 extra->nDataSize = sizeof(OMX_QCOM_EXTRADATA_QP);
12344 qp = (OMX_QCOM_EXTRADATA_QP *)(void *)extra->data;
12345 qp->nQP = qp_payload->frame_qp;
12346 print_debug_extradata(extra);
12347 }
12348
append_bitsinfo_extradata(OMX_OTHER_EXTRADATATYPE * extra,struct msm_vidc_frame_bits_info_payload * bits_payload)12349 void omx_vdec::append_bitsinfo_extradata(OMX_OTHER_EXTRADATATYPE *extra,
12350 struct msm_vidc_frame_bits_info_payload *bits_payload)
12351 {
12352 OMX_QCOM_EXTRADATA_BITS_INFO * bits = NULL;
12353 if (!bits_payload) {
12354 DEBUG_PRINT_ERROR("bits info payload is NULL");
12355 return;
12356 }
12357 extra->nSize = OMX_BITSINFO_EXTRADATA_SIZE;
12358 extra->nVersion.nVersion = OMX_SPEC_VERSION;
12359 extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12360 extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataInputBitsInfo;
12361 extra->nDataSize = sizeof(OMX_QCOM_EXTRADATA_BITS_INFO);
12362 bits = (OMX_QCOM_EXTRADATA_BITS_INFO*)(void *)extra->data;
12363 bits->frame_bits = bits_payload->frame_bits;
12364 bits->header_bits = bits_payload->header_bits;
12365 print_debug_extradata(extra);
12366 }
12367
append_user_extradata(OMX_OTHER_EXTRADATATYPE * extra,OMX_OTHER_EXTRADATATYPE * p_user)12368 void omx_vdec::append_user_extradata(OMX_OTHER_EXTRADATATYPE *extra,
12369 OMX_OTHER_EXTRADATATYPE *p_user)
12370 {
12371 int userdata_size = 0;
12372 struct msm_vidc_stream_userdata_payload *userdata_payload = NULL;
12373 userdata_payload =
12374 (struct msm_vidc_stream_userdata_payload *)(void *)p_user->data;
12375 userdata_size = p_user->nDataSize;
12376 extra->nSize = OMX_USERDATA_EXTRADATA_SIZE + userdata_size;
12377 extra->nVersion.nVersion = OMX_SPEC_VERSION;
12378 extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12379 extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataMP2UserData;
12380 extra->nDataSize = userdata_size;
12381 if (extra->nDataSize && (p_user->nDataSize >= extra->nDataSize))
12382 memcpy(extra->data, p_user->data, extra->nDataSize);
12383 print_debug_extradata(extra);
12384 }
12385
append_terminator_extradata(OMX_OTHER_EXTRADATATYPE * extra)12386 void omx_vdec::append_terminator_extradata(OMX_OTHER_EXTRADATATYPE *extra)
12387 {
12388 if (!client_extradata) {
12389 return;
12390 }
12391 extra->nSize = sizeof(OMX_OTHER_EXTRADATATYPE);
12392 extra->nVersion.nVersion = OMX_SPEC_VERSION;
12393 extra->eType = OMX_ExtraDataNone;
12394 extra->nDataSize = 0;
12395 extra->data[0] = 0;
12396
12397 print_debug_extradata(extra);
12398 }
12399
append_vqzip_extradata(OMX_OTHER_EXTRADATATYPE * extra,struct msm_vidc_vqzip_sei_payload * vqzip_payload)12400 void omx_vdec::append_vqzip_extradata(OMX_OTHER_EXTRADATATYPE *extra,
12401 struct msm_vidc_vqzip_sei_payload *vqzip_payload)
12402 {
12403 OMX_QCOM_EXTRADATA_VQZIPSEI *vq = NULL;
12404
12405 extra->nSize = OMX_VQZIPSEI_EXTRADATA_SIZE + vqzip_payload->size;
12406 extra->nVersion.nVersion = OMX_SPEC_VERSION;
12407 extra->nPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
12408 extra->eType = (OMX_EXTRADATATYPE)OMX_ExtraDataVQZipSEI;
12409 extra->nDataSize = sizeof(OMX_QCOM_EXTRADATA_VQZIPSEI) + vqzip_payload->size;
12410
12411 vq = (OMX_QCOM_EXTRADATA_VQZIPSEI *)(void *)extra->data;
12412 vq->nSize = vqzip_payload->size;
12413 memcpy(vq->data, vqzip_payload->data, vqzip_payload->size);
12414
12415 print_debug_extradata(extra);
12416 }
12417
allocate_desc_buffer(OMX_U32 index)12418 OMX_ERRORTYPE omx_vdec::allocate_desc_buffer(OMX_U32 index)
12419 {
12420 OMX_ERRORTYPE eRet = OMX_ErrorNone;
12421 if (index >= drv_ctx.ip_buf.actualcount) {
12422 DEBUG_PRINT_ERROR("ERROR:Desc Buffer Index not found");
12423 return OMX_ErrorInsufficientResources;
12424 }
12425 if (m_desc_buffer_ptr == NULL) {
12426 m_desc_buffer_ptr = (desc_buffer_hdr*) \
12427 calloc( (sizeof(desc_buffer_hdr)),
12428 drv_ctx.ip_buf.actualcount);
12429 if (m_desc_buffer_ptr == NULL) {
12430 DEBUG_PRINT_ERROR("m_desc_buffer_ptr Allocation failed ");
12431 return OMX_ErrorInsufficientResources;
12432 }
12433 }
12434
12435 m_desc_buffer_ptr[index].buf_addr = (unsigned char *)malloc (DESC_BUFFER_SIZE * sizeof(OMX_U8));
12436 if (m_desc_buffer_ptr[index].buf_addr == NULL) {
12437 DEBUG_PRINT_ERROR("desc buffer Allocation failed ");
12438 return OMX_ErrorInsufficientResources;
12439 }
12440
12441 return eRet;
12442 }
12443
insert_demux_addr_offset(OMX_U32 address_offset)12444 void omx_vdec::insert_demux_addr_offset(OMX_U32 address_offset)
12445 {
12446 DEBUG_PRINT_LOW("Inserting address offset (%u) at idx (%u)", (unsigned int)address_offset,(unsigned int)m_demux_entries);
12447 if (m_demux_entries < 8192) {
12448 m_demux_offsets[m_demux_entries++] = address_offset;
12449 }
12450 return;
12451 }
12452
extract_demux_addr_offsets(OMX_BUFFERHEADERTYPE * buf_hdr)12453 void omx_vdec::extract_demux_addr_offsets(OMX_BUFFERHEADERTYPE *buf_hdr)
12454 {
12455 OMX_U32 bytes_to_parse = buf_hdr->nFilledLen;
12456 OMX_U8 *buf = buf_hdr->pBuffer + buf_hdr->nOffset;
12457 OMX_U32 index = 0;
12458
12459 m_demux_entries = 0;
12460
12461 while (index < bytes_to_parse) {
12462 if ( ((buf[index] == 0x00) && (buf[index+1] == 0x00) &&
12463 (buf[index+2] == 0x00) && (buf[index+3] == 0x01)) ||
12464 ((buf[index] == 0x00) && (buf[index+1] == 0x00) &&
12465 (buf[index+2] == 0x01)) ) {
12466 //Found start code, insert address offset
12467 insert_demux_addr_offset(index);
12468 if (buf[index+2] == 0x01) // 3 byte start code
12469 index += 3;
12470 else //4 byte start code
12471 index += 4;
12472 } else
12473 index++;
12474 }
12475 DEBUG_PRINT_LOW("Extracted (%u) demux entry offsets", (unsigned int)m_demux_entries);
12476 return;
12477 }
12478
handle_demux_data(OMX_BUFFERHEADERTYPE * p_buf_hdr)12479 OMX_ERRORTYPE omx_vdec::handle_demux_data(OMX_BUFFERHEADERTYPE *p_buf_hdr)
12480 {
12481 //fix this, handle 3 byte start code, vc1 terminator entry
12482 OMX_U8 *p_demux_data = NULL;
12483 OMX_U32 desc_data = 0;
12484 OMX_U32 start_addr = 0;
12485 OMX_U32 nal_size = 0;
12486 OMX_U32 suffix_byte = 0;
12487 OMX_U32 demux_index = 0;
12488 OMX_U32 buffer_index = 0;
12489
12490 if (m_desc_buffer_ptr == NULL) {
12491 DEBUG_PRINT_ERROR("m_desc_buffer_ptr is NULL. Cannot append demux entries.");
12492 return OMX_ErrorBadParameter;
12493 }
12494
12495 buffer_index = p_buf_hdr - ((OMX_BUFFERHEADERTYPE *)m_inp_mem_ptr);
12496 if (buffer_index > drv_ctx.ip_buf.actualcount) {
12497 DEBUG_PRINT_ERROR("handle_demux_data:Buffer index is incorrect (%u)", (unsigned int)buffer_index);
12498 return OMX_ErrorBadParameter;
12499 }
12500
12501 p_demux_data = (OMX_U8 *) m_desc_buffer_ptr[buffer_index].buf_addr;
12502
12503 if ( ((OMX_U8*)p_demux_data == NULL) ||
12504 ((m_demux_entries * 16) + 1) > DESC_BUFFER_SIZE) {
12505 DEBUG_PRINT_ERROR("Insufficient buffer. Cannot append demux entries.");
12506 return OMX_ErrorBadParameter;
12507 } else {
12508 for (; demux_index < m_demux_entries; demux_index++) {
12509 desc_data = 0;
12510 start_addr = m_demux_offsets[demux_index];
12511 if (p_buf_hdr->pBuffer[m_demux_offsets[demux_index] + 2] == 0x01) {
12512 suffix_byte = p_buf_hdr->pBuffer[m_demux_offsets[demux_index] + 3];
12513 } else {
12514 suffix_byte = p_buf_hdr->pBuffer[m_demux_offsets[demux_index] + 4];
12515 }
12516 if (demux_index < (m_demux_entries - 1)) {
12517 nal_size = m_demux_offsets[demux_index + 1] - m_demux_offsets[demux_index] - 2;
12518 } else {
12519 nal_size = p_buf_hdr->nFilledLen - m_demux_offsets[demux_index] - 2;
12520 }
12521 DEBUG_PRINT_LOW("Start_addr(0x%x), suffix_byte(0x%x),nal_size(%u),demux_index(%u)",
12522 (unsigned int)start_addr,
12523 (unsigned int)suffix_byte,
12524 (unsigned int)nal_size,
12525 (unsigned int)demux_index);
12526 desc_data = (start_addr >> 3) << 1;
12527 desc_data |= (start_addr & 7) << 21;
12528 desc_data |= suffix_byte << 24;
12529
12530 memcpy(p_demux_data, &desc_data, sizeof(OMX_U32));
12531 memcpy(p_demux_data + 4, &nal_size, sizeof(OMX_U32));
12532 memset(p_demux_data + 8, 0, sizeof(OMX_U32));
12533 memset(p_demux_data + 12, 0, sizeof(OMX_U32));
12534
12535 p_demux_data += 16;
12536 }
12537 if (codec_type_parse == CODEC_TYPE_VC1) {
12538 DEBUG_PRINT_LOW("VC1 terminator entry");
12539 desc_data = 0;
12540 desc_data = 0x82 << 24;
12541 memcpy(p_demux_data, &desc_data, sizeof(OMX_U32));
12542 memset(p_demux_data + 4, 0, sizeof(OMX_U32));
12543 memset(p_demux_data + 8, 0, sizeof(OMX_U32));
12544 memset(p_demux_data + 12, 0, sizeof(OMX_U32));
12545 p_demux_data += 16;
12546 m_demux_entries++;
12547 }
12548 //Add zero word to indicate end of descriptors
12549 memset(p_demux_data, 0, sizeof(OMX_U32));
12550
12551 m_desc_buffer_ptr[buffer_index].desc_data_size = (m_demux_entries * 16) + sizeof(OMX_U32);
12552 DEBUG_PRINT_LOW("desc table data size=%u", (unsigned int)m_desc_buffer_ptr[buffer_index].desc_data_size);
12553 }
12554 memset(m_demux_offsets, 0, ( sizeof(OMX_U32) * 8192) );
12555 m_demux_entries = 0;
12556 DEBUG_PRINT_LOW("Demux table complete!");
12557 return OMX_ErrorNone;
12558 }
12559
request_perf_level(enum vidc_perf_level perf_level)12560 void omx_vdec::request_perf_level(enum vidc_perf_level perf_level)
12561 {
12562 struct v4l2_control control;
12563 char property_value[PROPERTY_VALUE_MAX] = {0};
12564
12565 property_get("vendor.vidc.debug.turbo", property_value, "0");
12566 memset(&control, 0, sizeof(v4l2_control));
12567 control.id = V4L2_CID_MPEG_VIDC_SET_PERF_LEVEL;
12568 switch (perf_level) {
12569 case VIDC_NOMINAL:
12570 if (atoi(property_value))
12571 control.value = V4L2_CID_MPEG_VIDC_PERF_LEVEL_TURBO;
12572 else
12573 control.value = V4L2_CID_MPEG_VIDC_PERF_LEVEL_NOMINAL;
12574 break;
12575 case VIDC_TURBO:
12576 control.value = V4L2_CID_MPEG_VIDC_PERF_LEVEL_TURBO;
12577 break;
12578 default:
12579 DEBUG_PRINT_ERROR("Requested PERF level not supported");
12580 break;
12581 }
12582 if ((current_perf_level == (OMX_U32)control.value) && !in_reconfig)
12583 return;
12584
12585 DEBUG_PRINT_HIGH("changing performance level to %d", control.value);
12586 if (!ioctl(drv_ctx.video_driver_fd, VIDIOC_S_CTRL, &control)) {
12587 current_perf_level = control.value;
12588 } else {
12589 DEBUG_PRINT_ERROR("Failed to set PERF level");
12590 }
12591 }
12592
allocate_color_convert_buf()12593 omx_vdec::allocate_color_convert_buf::allocate_color_convert_buf()
12594 {
12595 enabled = false;
12596 omx = NULL;
12597 init_members();
12598 ColorFormat = OMX_COLOR_FormatMax;
12599 dest_format = YCbCr420P;
12600 m_c2d_width = 0;
12601 m_c2d_height = 0;
12602 }
12603
set_vdec_client(void * client)12604 void omx_vdec::allocate_color_convert_buf::set_vdec_client(void *client)
12605 {
12606 omx = reinterpret_cast<omx_vdec*>(client);
12607 }
12608
init_members()12609 void omx_vdec::allocate_color_convert_buf::init_members()
12610 {
12611 allocated_count = 0;
12612 buffer_size_req = 0;
12613 buffer_alignment_req = 0;
12614 m_c2d_width = m_c2d_height = 0;
12615 memset(m_platform_list_client,0,sizeof(m_platform_list_client));
12616 memset(m_platform_entry_client,0,sizeof(m_platform_entry_client));
12617 memset(m_pmem_info_client,0,sizeof(m_pmem_info_client));
12618 memset(m_out_mem_ptr_client,0,sizeof(m_out_mem_ptr_client));
12619 #ifdef USE_ION
12620 memset(op_buf_ion_info,0,sizeof(m_platform_entry_client));
12621 #endif
12622 for (int i = 0; i < MAX_COUNT; i++)
12623 pmem_fd[i] = -1;
12624 }
12625
~allocate_color_convert_buf()12626 omx_vdec::allocate_color_convert_buf::~allocate_color_convert_buf()
12627 {
12628 c2d.destroy();
12629 }
12630
update_buffer_req()12631 bool omx_vdec::allocate_color_convert_buf::update_buffer_req()
12632 {
12633 bool status = true;
12634 unsigned int src_size = 0, destination_size = 0;
12635 unsigned int height, width;
12636 struct v4l2_format fmt;
12637 OMX_COLOR_FORMATTYPE drv_color_format;
12638
12639 if (!omx) {
12640 DEBUG_PRINT_ERROR("Invalid client in color convert");
12641 return false;
12642 }
12643 if (!enabled) {
12644 DEBUG_PRINT_HIGH("No color conversion required");
12645 return status;
12646 }
12647 pthread_mutex_lock(&omx->c_lock);
12648
12649 memset(&fmt, 0x0, sizeof(struct v4l2_format));
12650 fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
12651 fmt.fmt.pix_mp.pixelformat = omx->capture_capability;
12652 ioctl(omx->drv_ctx.video_driver_fd, VIDIOC_G_FMT, &fmt);
12653 width = fmt.fmt.pix_mp.width;
12654 height = fmt.fmt.pix_mp.height;
12655
12656 bool resolution_upgrade = (height > m_c2d_height ||
12657 width > m_c2d_width);
12658 if (resolution_upgrade) {
12659 // resolution upgraded ? ensure we are yet to allocate;
12660 // failing which, c2d buffers will never be reallocated and bad things will happen
12661 if (allocated_count > 0) {
12662 DEBUG_PRINT_ERROR("Cannot change C2D buffer requirements with %d active allocations",
12663 allocated_count);
12664 status = false;
12665 goto fail_update_buf_req;
12666 }
12667 }
12668
12669 if (omx->drv_ctx.output_format != VDEC_YUV_FORMAT_NV12 &&
12670 ColorFormat != OMX_COLOR_FormatYUV420Planar) {
12671 DEBUG_PRINT_ERROR("update_buffer_req: Unsupported color conversion");
12672 status = false;
12673 goto fail_update_buf_req;
12674 }
12675 c2d.close();
12676 status = c2d.open(height,
12677 width,
12678 NV12_128m,dest_format);
12679 if (status) {
12680 status = c2d.get_buffer_size(C2D_INPUT,src_size);
12681 if (status)
12682 status = c2d.get_buffer_size(C2D_OUTPUT,destination_size);
12683 }
12684 if (status) {
12685 if (!src_size || src_size > omx->drv_ctx.op_buf.buffer_size ||
12686 !destination_size) {
12687 DEBUG_PRINT_ERROR("ERROR: Size mismatch in C2D src_size %d"
12688 "driver size %u destination size %d",
12689 src_size, (unsigned int)omx->drv_ctx.op_buf.buffer_size,
12690 destination_size);
12691 status = false;
12692 c2d.close();
12693 buffer_size_req = 0;
12694 // TODO: make this fatal. Driver is not supposed to quote size
12695 // smaller than what C2D needs !!
12696 } else {
12697 buffer_size_req = destination_size;
12698 m_c2d_height = height;
12699 m_c2d_width = width;
12700 }
12701 }
12702 fail_update_buf_req:
12703 pthread_mutex_unlock(&omx->c_lock);
12704 return status;
12705 }
12706
set_color_format(OMX_COLOR_FORMATTYPE dest_color_format)12707 bool omx_vdec::allocate_color_convert_buf::set_color_format(
12708 OMX_COLOR_FORMATTYPE dest_color_format)
12709 {
12710 bool status = true, drv_colorformat_c2d_enable = false;
12711 bool dest_color_format_c2d_enable = false;
12712 OMX_COLOR_FORMATTYPE drv_color_format = OMX_COLOR_FormatUnused;
12713 if (!omx) {
12714 DEBUG_PRINT_ERROR("Invalid client in color convert");
12715 return false;
12716 }
12717 pthread_mutex_lock(&omx->c_lock);
12718 if (omx->drv_ctx.output_format == VDEC_YUV_FORMAT_NV12)
12719 if (omx->drv_ctx.decoder_format == VDEC_CODECTYPE_MVC)
12720 drv_color_format = (OMX_COLOR_FORMATTYPE)
12721 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mMultiView;
12722 else
12723 drv_color_format = (OMX_COLOR_FORMATTYPE)
12724 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m;
12725 else if (omx->drv_ctx.output_format == VDEC_YUV_FORMAT_NV12_UBWC) {
12726 drv_color_format = (OMX_COLOR_FORMATTYPE)
12727 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mCompressed;
12728 } else if (omx->drv_ctx.output_format == VDEC_YUV_FORMAT_NV12_TP10_UBWC) {
12729 drv_color_format = (OMX_COLOR_FORMATTYPE)
12730 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m10bitCompressed;
12731 } else {
12732 DEBUG_PRINT_ERROR("Incorrect color format");
12733 status = false;
12734 }
12735 drv_colorformat_c2d_enable = (drv_color_format != dest_color_format) &&
12736 (drv_color_format != (OMX_COLOR_FORMATTYPE)
12737 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mMultiView) &&
12738 (drv_color_format != (OMX_COLOR_FORMATTYPE)
12739 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mCompressed) &&
12740 (drv_color_format != (OMX_COLOR_FORMATTYPE)
12741 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m10bitCompressed);
12742
12743 dest_color_format_c2d_enable = (dest_color_format != (OMX_COLOR_FORMATTYPE)
12744 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mCompressed) &&
12745 (dest_color_format != (OMX_COLOR_FORMATTYPE)
12746 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m10bitCompressed);
12747
12748 if (status && drv_colorformat_c2d_enable && dest_color_format_c2d_enable) {
12749 DEBUG_PRINT_LOW("Enabling C2D");
12750 if ((dest_color_format != OMX_COLOR_FormatYUV420Planar) &&
12751 (dest_color_format != OMX_COLOR_FormatYUV420SemiPlanar)) {
12752 DEBUG_PRINT_ERROR("Unsupported color format for c2d");
12753 status = false;
12754 } else {
12755 ColorFormat = dest_color_format;
12756 dest_format = (dest_color_format == OMX_COLOR_FormatYUV420Planar) ?
12757 YCbCr420P : YCbCr420SP;
12758 if (enabled)
12759 c2d.destroy();
12760 enabled = false;
12761 if (!c2d.init()) {
12762 DEBUG_PRINT_ERROR("open failed for c2d");
12763 status = false;
12764 } else
12765 enabled = true;
12766 }
12767 } else {
12768 if (enabled)
12769 c2d.destroy();
12770 enabled = false;
12771 }
12772 pthread_mutex_unlock(&omx->c_lock);
12773 return status;
12774 }
12775
get_il_buf_hdr()12776 OMX_BUFFERHEADERTYPE* omx_vdec::allocate_color_convert_buf::get_il_buf_hdr()
12777 {
12778 if (!omx) {
12779 DEBUG_PRINT_ERROR("Invalid param get_buf_hdr");
12780 return NULL;
12781 }
12782 if (!enabled)
12783 return omx->m_out_mem_ptr;
12784 return m_out_mem_ptr_client;
12785 }
12786
get_il_buf_hdr(OMX_BUFFERHEADERTYPE * bufadd)12787 OMX_BUFFERHEADERTYPE* omx_vdec::allocate_color_convert_buf::get_il_buf_hdr
12788 (OMX_BUFFERHEADERTYPE *bufadd)
12789 {
12790 if (!omx) {
12791 DEBUG_PRINT_ERROR("Invalid param get_buf_hdr");
12792 return NULL;
12793 }
12794 if (!enabled)
12795 return bufadd;
12796
12797 unsigned index = 0;
12798 index = bufadd - omx->m_out_mem_ptr;
12799 if (index < omx->drv_ctx.op_buf.actualcount) {
12800 m_out_mem_ptr_client[index].nFlags = (bufadd->nFlags & OMX_BUFFERFLAG_EOS);
12801 m_out_mem_ptr_client[index].nTimeStamp = bufadd->nTimeStamp;
12802 bool status;
12803 if (!omx->in_reconfig && !omx->output_flush_progress && bufadd->nFilledLen) {
12804 pthread_mutex_lock(&omx->c_lock);
12805 cache_clean_buffer(index);
12806 status = c2d.convert(omx->drv_ctx.ptr_outputbuffer[index].pmem_fd,
12807 omx->drv_ctx.op_buf_map_info[index].base_address, bufadd->pBuffer, pmem_fd[index],
12808 pmem_baseaddress[index], pmem_baseaddress[index]);
12809 if (!status) {
12810 DEBUG_PRINT_ERROR("Failed color conversion %d", status);
12811 m_out_mem_ptr_client[index].nFilledLen = 0;
12812 pthread_mutex_unlock(&omx->c_lock);
12813 return &m_out_mem_ptr_client[index];
12814 } else {
12815 unsigned int filledLen = 0;
12816 c2d.get_output_filled_length(filledLen);
12817 m_out_mem_ptr_client[index].nFilledLen = filledLen;
12818 cache_clean_invalidate_buffer(index);
12819 }
12820 pthread_mutex_unlock(&omx->c_lock);
12821 } else
12822 m_out_mem_ptr_client[index].nFilledLen = 0;
12823 return &m_out_mem_ptr_client[index];
12824 }
12825 DEBUG_PRINT_ERROR("Index messed up in the get_il_buf_hdr");
12826 return NULL;
12827 }
12828
get_dr_buf_hdr(OMX_BUFFERHEADERTYPE * bufadd)12829 OMX_BUFFERHEADERTYPE* omx_vdec::allocate_color_convert_buf::get_dr_buf_hdr
12830 (OMX_BUFFERHEADERTYPE *bufadd)
12831 {
12832 if (!omx) {
12833 DEBUG_PRINT_ERROR("Invalid param get_buf_hdr");
12834 return NULL;
12835 }
12836 if (!enabled)
12837 return bufadd;
12838 unsigned index = 0;
12839 index = bufadd - m_out_mem_ptr_client;
12840 if (index < omx->drv_ctx.op_buf.actualcount) {
12841 return &omx->m_out_mem_ptr[index];
12842 }
12843 DEBUG_PRINT_ERROR("Index messed up in the get_dr_buf_hdr");
12844 return NULL;
12845 }
get_buffer_req(unsigned int & buffer_size)12846 bool omx_vdec::allocate_color_convert_buf::get_buffer_req
12847 (unsigned int &buffer_size)
12848 {
12849 bool status = true;
12850 pthread_mutex_lock(&omx->c_lock);
12851 if (!enabled)
12852 buffer_size = omx->drv_ctx.op_buf.buffer_size;
12853 else {
12854 if (!c2d.get_buffer_size(C2D_OUTPUT,buffer_size)) {
12855 DEBUG_PRINT_ERROR("Get buffer size failed");
12856 status = false;
12857 goto fail_get_buffer_size;
12858 }
12859 }
12860 fail_get_buffer_size:
12861 pthread_mutex_unlock(&omx->c_lock);
12862 return status;
12863 }
12864
set_buffer_req(OMX_U32 buffer_size,OMX_U32 actual_count)12865 OMX_ERRORTYPE omx_vdec::allocate_color_convert_buf::set_buffer_req(
12866 OMX_U32 buffer_size, OMX_U32 actual_count) {
12867 OMX_U32 expectedSize = enabled ? buffer_size_req : omx->drv_ctx.op_buf.buffer_size;
12868
12869 if (buffer_size < expectedSize) {
12870 DEBUG_PRINT_ERROR("OP Requirements: Client size(%u) insufficient v/s requested(%u)",
12871 buffer_size, expectedSize);
12872 return OMX_ErrorBadParameter;
12873 }
12874 if (actual_count < omx->drv_ctx.op_buf.actualcount) {
12875 DEBUG_PRINT_ERROR("OP Requirements: Client count(%u) insufficient v/s requested(%u)",
12876 actual_count, omx->drv_ctx.op_buf.actualcount);
12877 return OMX_ErrorBadParameter;
12878 }
12879
12880 bool reqs_updated = false;
12881 if (enabled) {
12882 // disallow changing buffer size/count while we have active allocated buffers
12883 if (allocated_count > 0) {
12884 DEBUG_PRINT_ERROR("Cannot change C2D buffer size from %u to %u with %d active allocations",
12885 buffer_size_req, buffer_size, allocated_count);
12886 return OMX_ErrorInvalidState;
12887 }
12888
12889 buffer_size_req = buffer_size;
12890 } else {
12891 if (buffer_size > omx->drv_ctx.op_buf.buffer_size) {
12892 omx->drv_ctx.op_buf.buffer_size = buffer_size;
12893 reqs_updated = true;
12894 }
12895 }
12896
12897 if (actual_count > omx->drv_ctx.op_buf.actualcount) {
12898 omx->drv_ctx.op_buf.actualcount = actual_count;
12899 reqs_updated = true;
12900 }
12901
12902 if (reqs_updated) {
12903 omx->drv_ctx.extradata_info.count = omx->drv_ctx.op_buf.actualcount;
12904 omx->drv_ctx.extradata_info.size = omx->drv_ctx.extradata_info.count *
12905 omx->drv_ctx.extradata_info.buffer_size;
12906 return omx->set_buffer_req(&(omx->drv_ctx.op_buf));
12907 }
12908 return OMX_ErrorNone;
12909 }
12910
free_output_buffer(OMX_BUFFERHEADERTYPE * bufhdr)12911 OMX_ERRORTYPE omx_vdec::allocate_color_convert_buf::free_output_buffer(
12912 OMX_BUFFERHEADERTYPE *bufhdr)
12913 {
12914 unsigned int index = 0;
12915
12916 if (!enabled)
12917 return omx->free_output_buffer(bufhdr);
12918 if (enabled && omx->is_component_secure())
12919 return OMX_ErrorNone;
12920 if (!allocated_count || !bufhdr) {
12921 DEBUG_PRINT_ERROR("Color convert no buffer to be freed %p",bufhdr);
12922 return OMX_ErrorBadParameter;
12923 }
12924 index = bufhdr - m_out_mem_ptr_client;
12925 if (index >= omx->drv_ctx.op_buf.actualcount) {
12926 DEBUG_PRINT_ERROR("Incorrect index color convert free_output_buffer");
12927 return OMX_ErrorBadParameter;
12928 }
12929 if (pmem_fd[index] >= 0) {
12930 munmap(pmem_baseaddress[index], buffer_size_req);
12931 close(pmem_fd[index]);
12932 }
12933 pmem_fd[index] = -1;
12934 #ifdef USE_ION
12935 omx->free_ion_memory(&op_buf_ion_info[index]);
12936 #endif
12937 if (allocated_count > 0)
12938 allocated_count--;
12939 else
12940 allocated_count = 0;
12941 if (!allocated_count) {
12942 pthread_mutex_lock(&omx->c_lock);
12943 c2d.close();
12944 init_members();
12945 pthread_mutex_unlock(&omx->c_lock);
12946 }
12947 return omx->free_output_buffer(&omx->m_out_mem_ptr[index]);
12948 }
12949
allocate_buffers_color_convert(OMX_HANDLETYPE hComp,OMX_BUFFERHEADERTYPE ** bufferHdr,OMX_U32 port,OMX_PTR appData,OMX_U32 bytes)12950 OMX_ERRORTYPE omx_vdec::allocate_color_convert_buf::allocate_buffers_color_convert(OMX_HANDLETYPE hComp,
12951 OMX_BUFFERHEADERTYPE **bufferHdr,OMX_U32 port,OMX_PTR appData,OMX_U32 bytes)
12952 {
12953 OMX_ERRORTYPE eRet = OMX_ErrorNone;
12954 if (!enabled) {
12955 eRet = omx->allocate_output_buffer(hComp,bufferHdr,port,appData,bytes);
12956 return eRet;
12957 }
12958 if (enabled && omx->is_component_secure()) {
12959 DEBUG_PRINT_ERROR("Notin color convert mode secure_mode %d",
12960 omx->is_component_secure());
12961 return OMX_ErrorUnsupportedSetting;
12962 }
12963 if (!bufferHdr || bytes > buffer_size_req) {
12964 DEBUG_PRINT_ERROR("Invalid params allocate_buffers_color_convert %p", bufferHdr);
12965 DEBUG_PRINT_ERROR("color_convert buffer_size_req %u bytes %u",
12966 (unsigned int)buffer_size_req, (unsigned int)bytes);
12967 return OMX_ErrorBadParameter;
12968 }
12969 if (allocated_count >= omx->drv_ctx.op_buf.actualcount) {
12970 DEBUG_PRINT_ERROR("Actual count err in allocate_buffers_color_convert");
12971 return OMX_ErrorInsufficientResources;
12972 }
12973 OMX_BUFFERHEADERTYPE *temp_bufferHdr = NULL;
12974 eRet = omx->allocate_output_buffer(hComp,&temp_bufferHdr,
12975 port,appData,omx->drv_ctx.op_buf.buffer_size);
12976 if (eRet != OMX_ErrorNone || !temp_bufferHdr) {
12977 DEBUG_PRINT_ERROR("Buffer allocation failed color_convert");
12978 return eRet;
12979 }
12980 if ((temp_bufferHdr - omx->m_out_mem_ptr) >=
12981 (int)omx->drv_ctx.op_buf.actualcount) {
12982 DEBUG_PRINT_ERROR("Invalid header index %ld",
12983 (long int)(temp_bufferHdr - omx->m_out_mem_ptr));
12984 return OMX_ErrorUndefined;
12985 }
12986 unsigned int i = allocated_count;
12987 #ifdef USE_ION
12988 // Allocate color-conversion buffers as cached to improve software-reading
12989 // performance of YUV (thumbnails). NOTE: These buffers will need an explicit
12990 // cache invalidation.
12991 op_buf_ion_info[i].ion_device_fd = omx->alloc_map_ion_memory(
12992 buffer_size_req,buffer_alignment_req,
12993 &op_buf_ion_info[i].ion_alloc_data,&op_buf_ion_info[i].fd_ion_data,
12994 ION_FLAG_CACHED);
12995 pmem_fd[i] = op_buf_ion_info[i].fd_ion_data.fd;
12996 if (op_buf_ion_info[i].ion_device_fd < 0) {
12997 DEBUG_PRINT_ERROR("alloc_map_ion failed in color_convert");
12998 return OMX_ErrorInsufficientResources;
12999 }
13000 pmem_baseaddress[i] = (unsigned char *)mmap(NULL,buffer_size_req,
13001 PROT_READ|PROT_WRITE,MAP_SHARED,pmem_fd[i],0);
13002
13003 if (pmem_baseaddress[i] == MAP_FAILED) {
13004 DEBUG_PRINT_ERROR("MMAP failed for Size %d",buffer_size_req);
13005 close(pmem_fd[i]);
13006 omx->free_ion_memory(&op_buf_ion_info[i]);
13007 return OMX_ErrorInsufficientResources;
13008 }
13009 #endif
13010 m_pmem_info_client[i].offset = 0;
13011 m_platform_entry_client[i].entry = (void *)&m_pmem_info_client[i];
13012 m_platform_entry_client[i].type = OMX_QCOM_PLATFORM_PRIVATE_PMEM;
13013 m_platform_list_client[i].nEntries = 1;
13014 m_platform_list_client[i].entryList = &m_platform_entry_client[i];
13015 m_out_mem_ptr_client[i].pOutputPortPrivate = NULL;
13016 m_out_mem_ptr_client[i].nAllocLen = buffer_size_req;
13017 m_out_mem_ptr_client[i].nFilledLen = 0;
13018 m_out_mem_ptr_client[i].nFlags = 0;
13019 m_out_mem_ptr_client[i].nOutputPortIndex = OMX_CORE_OUTPUT_PORT_INDEX;
13020 m_out_mem_ptr_client[i].nSize = sizeof(OMX_BUFFERHEADERTYPE);
13021 m_out_mem_ptr_client[i].nVersion.nVersion = OMX_SPEC_VERSION;
13022 m_out_mem_ptr_client[i].pPlatformPrivate = &m_platform_list_client[i];
13023 m_out_mem_ptr_client[i].pBuffer = pmem_baseaddress[i];
13024 m_out_mem_ptr_client[i].pAppPrivate = appData;
13025 *bufferHdr = &m_out_mem_ptr_client[i];
13026 DEBUG_PRINT_HIGH("IL client buffer header %p", *bufferHdr);
13027 allocated_count++;
13028 return eRet;
13029 }
13030
is_component_secure()13031 bool omx_vdec::is_component_secure()
13032 {
13033 return secure_mode;
13034 }
13035
get_color_format(OMX_COLOR_FORMATTYPE & dest_color_format)13036 bool omx_vdec::allocate_color_convert_buf::get_color_format(OMX_COLOR_FORMATTYPE &dest_color_format)
13037 {
13038 bool status = true;
13039 if (!enabled) {
13040 if (omx->drv_ctx.output_format == VDEC_YUV_FORMAT_NV12) {
13041 if (omx->drv_ctx.decoder_format == VDEC_CODECTYPE_MVC)
13042 dest_color_format = (OMX_COLOR_FORMATTYPE)
13043 QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mMultiView;
13044 else
13045 dest_color_format = (OMX_COLOR_FORMATTYPE)QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m;
13046 } else if (omx->drv_ctx.output_format == VDEC_YUV_FORMAT_NV12_UBWC){
13047 dest_color_format = (OMX_COLOR_FORMATTYPE)QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mCompressed;
13048 } else if (omx->drv_ctx.output_format == VDEC_YUV_FORMAT_NV12_TP10_UBWC){
13049 dest_color_format = (OMX_COLOR_FORMATTYPE)QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m10bitCompressed;
13050 } else
13051 status = false;
13052 } else {
13053 if (ColorFormat == OMX_COLOR_FormatYUV420Planar ||
13054 ColorFormat == OMX_COLOR_FormatYUV420SemiPlanar) {
13055 dest_color_format = ColorFormat;
13056 } else
13057 status = false;
13058 }
13059 return status;
13060 }
13061
cache_ops(unsigned int index,unsigned int cmd)13062 OMX_ERRORTYPE omx_vdec::allocate_color_convert_buf::cache_ops(
13063 unsigned int index, unsigned int cmd)
13064 {
13065 if (!enabled) {
13066 return OMX_ErrorNone;
13067 }
13068
13069 if (!omx || index >= omx->drv_ctx.op_buf.actualcount) {
13070 DEBUG_PRINT_ERROR("%s: Invalid param", __func__);
13071 return OMX_ErrorBadParameter;
13072 }
13073
13074 struct ion_flush_data flush_data;
13075 struct ion_custom_data custom_data;
13076
13077 memset(&flush_data, 0x0, sizeof(flush_data));
13078 memset(&custom_data, 0x0, sizeof(custom_data));
13079
13080 flush_data.vaddr = pmem_baseaddress[index];
13081 flush_data.fd = op_buf_ion_info[index].fd_ion_data.fd;
13082 flush_data.handle = op_buf_ion_info[index].fd_ion_data.handle;
13083 flush_data.length = buffer_size_req;
13084 custom_data.cmd = cmd;
13085 custom_data.arg = (unsigned long)&flush_data;
13086
13087 DEBUG_PRINT_LOW("Cache %s: fd=%d handle=%d va=%p size=%d",
13088 (cmd == ION_IOC_CLEAN_CACHES) ? "Clean" : "Invalidate",
13089 flush_data.fd, flush_data.handle, flush_data.vaddr,
13090 flush_data.length);
13091 int ret = ioctl(op_buf_ion_info[index].ion_device_fd, ION_IOC_CUSTOM, &custom_data);
13092 if (ret < 0) {
13093 DEBUG_PRINT_ERROR("Cache %s failed: %s\n",
13094 (cmd == ION_IOC_CLEAN_CACHES) ? "Clean" : "Invalidate",
13095 strerror(errno));
13096 return OMX_ErrorUndefined;
13097 }
13098 return OMX_ErrorNone;
13099 }
13100
buf_ref_add(int nPortIndex)13101 void omx_vdec::buf_ref_add(int nPortIndex)
13102 {
13103 unsigned long i = 0;
13104 bool buf_present = false;
13105 long fd = drv_ctx.ptr_outputbuffer[nPortIndex].pmem_fd;
13106 OMX_U32 offset = drv_ctx.ptr_outputbuffer[nPortIndex].offset;
13107
13108 if (!dynamic_buf_mode || !out_dynamic_list) {
13109 return;
13110 }
13111
13112 pthread_mutex_lock(&m_lock);
13113 for (i = 0; i < drv_ctx.op_buf.actualcount; i++) {
13114 //check the buffer fd, offset, uv addr with list contents
13115 //If present increment reference.
13116 if ((out_dynamic_list[i].fd == fd) &&
13117 (out_dynamic_list[i].offset == offset)) {
13118 DEBUG_PRINT_LOW("buf_ref_add: [ALREADY PRESENT] fd = %u ref_count = %u",
13119 (unsigned int)out_dynamic_list[i].fd, (unsigned int)out_dynamic_list[i].ref_count);
13120 if (!secure_mode) {
13121 drv_ctx.ptr_outputbuffer[nPortIndex].bufferaddr = out_dynamic_list[i].buffaddr;
13122 }
13123 buf_present = true;
13124 break;
13125 }
13126 }
13127 if (!buf_present) {
13128 for (i = 0; i < drv_ctx.op_buf.actualcount; i++) {
13129 //search for a entry to insert details of the new buffer
13130 if (out_dynamic_list[i].dup_fd < 0) {
13131 out_dynamic_list[i].fd = fd;
13132 out_dynamic_list[i].offset = offset;
13133 out_dynamic_list[i].dup_fd = dup(fd);
13134 out_dynamic_list[i].ref_count++;
13135 DEBUG_PRINT_LOW("buf_ref_add: [ADDED] fd = %u ref_count = %u",
13136 (unsigned int)out_dynamic_list[i].fd, (unsigned int)out_dynamic_list[i].ref_count);
13137
13138 if (!secure_mode) {
13139 drv_ctx.ptr_outputbuffer[nPortIndex].bufferaddr =
13140 (OMX_U8*)mmap(0, drv_ctx.ptr_outputbuffer[nPortIndex].buffer_len,
13141 PROT_READ|PROT_WRITE, MAP_SHARED,
13142 drv_ctx.ptr_outputbuffer[nPortIndex].pmem_fd, 0);
13143 //mmap returns (void *)-1 on failure and sets error code in errno.
13144 if (drv_ctx.ptr_outputbuffer[nPortIndex].bufferaddr == MAP_FAILED) {
13145 DEBUG_PRINT_ERROR("buf_ref_add: mmap failed - errno: %d", errno);
13146 drv_ctx.ptr_outputbuffer[nPortIndex].bufferaddr = NULL;
13147 break;
13148 }
13149 out_dynamic_list[i].buffaddr = drv_ctx.ptr_outputbuffer[nPortIndex].bufferaddr;
13150 out_dynamic_list[i].mapped_size = drv_ctx.ptr_outputbuffer[nPortIndex].buffer_len;
13151 DEBUG_PRINT_LOW("mmap: %p %ld", out_dynamic_list[i].buffaddr, out_dynamic_list[i].mapped_size);
13152 }
13153 break;
13154 }
13155 }
13156 }
13157 pthread_mutex_unlock(&m_lock);
13158 }
13159
buf_ref_remove()13160 void omx_vdec::buf_ref_remove()
13161 {
13162 unsigned long i = 0;
13163
13164 if (!dynamic_buf_mode || !out_dynamic_list) {
13165 return;
13166 }
13167
13168 pthread_mutex_lock(&m_lock);
13169 for (i = 0; i < drv_ctx.op_buf.actualcount; i++) {
13170 if (!secure_mode && out_dynamic_list[i].buffaddr && out_dynamic_list[i].mapped_size) {
13171 DEBUG_PRINT_LOW("munmap: %p %ld", out_dynamic_list[i].buffaddr, out_dynamic_list[i].mapped_size);
13172 munmap(out_dynamic_list[i].buffaddr,
13173 out_dynamic_list[i].mapped_size);
13174 }
13175
13176 DEBUG_PRINT_LOW("buf_ref_remove: [REMOVED] fd = %u ref_count = %u",
13177 (unsigned int)out_dynamic_list[i].fd, (unsigned int)out_dynamic_list[i].ref_count);
13178 close(out_dynamic_list[i].dup_fd);
13179 out_dynamic_list[i].dup_fd = -1;
13180 }
13181 pthread_mutex_unlock(&m_lock);
13182
13183 if (out_dynamic_list) {
13184 free(out_dynamic_list);
13185 out_dynamic_list = NULL;
13186 }
13187 }
13188
13189 #ifdef _MSM8974_
send_codec_config()13190 void omx_vdec::send_codec_config() {
13191 if (codec_config_flag) {
13192 unsigned long p1 = 0; // Parameter - 1
13193 unsigned long p2 = 0; // Parameter - 2
13194 unsigned long ident = 0;
13195 pthread_mutex_lock(&m_lock);
13196 DEBUG_PRINT_LOW("\n Check Queue for codec_config buffer \n");
13197 while (m_etb_q.m_size) {
13198 m_etb_q.pop_entry(&p1,&p2,&ident);
13199 if (ident == OMX_COMPONENT_GENERATE_ETB_ARBITRARY) {
13200 if (((OMX_BUFFERHEADERTYPE *)p2)->nFlags & OMX_BUFFERFLAG_CODECCONFIG) {
13201 if (empty_this_buffer_proxy_arbitrary((OMX_HANDLETYPE)p1,\
13202 (OMX_BUFFERHEADERTYPE *)p2) != OMX_ErrorNone) {
13203 DEBUG_PRINT_ERROR("\n empty_this_buffer_proxy_arbitrary failure");
13204 omx_report_error();
13205 }
13206 } else {
13207 DEBUG_PRINT_LOW("\n Flush Input Heap Buffer %p",(OMX_BUFFERHEADERTYPE *)p2);
13208 m_cb.EmptyBufferDone(&m_cmp ,m_app_data, (OMX_BUFFERHEADERTYPE *)p2);
13209 }
13210 } else if (ident == OMX_COMPONENT_GENERATE_ETB) {
13211 if (((OMX_BUFFERHEADERTYPE *)p2)->nFlags & OMX_BUFFERFLAG_CODECCONFIG) {
13212 if (empty_this_buffer_proxy((OMX_HANDLETYPE)p1,\
13213 (OMX_BUFFERHEADERTYPE *)p2) != OMX_ErrorNone) {
13214 DEBUG_PRINT_ERROR("\n empty_this_buffer_proxy failure");
13215 omx_report_error ();
13216 }
13217 } else {
13218 pending_input_buffers++;
13219 VIDC_TRACE_INT_LOW("ETB-pending", pending_input_buffers);
13220 DEBUG_PRINT_LOW("\n Flush Input OMX_COMPONENT_GENERATE_ETB %p, pending_input_buffers %d",
13221 (OMX_BUFFERHEADERTYPE *)p2, pending_input_buffers);
13222 empty_buffer_done(&m_cmp,(OMX_BUFFERHEADERTYPE *)p2);
13223 }
13224 } else if (ident == OMX_COMPONENT_GENERATE_EBD) {
13225 DEBUG_PRINT_LOW("\n Flush Input OMX_COMPONENT_GENERATE_EBD %p",
13226 (OMX_BUFFERHEADERTYPE *)p1);
13227 empty_buffer_done(&m_cmp,(OMX_BUFFERHEADERTYPE *)p1);
13228 }
13229 }
13230 pthread_mutex_unlock(&m_lock);
13231 }
13232 }
13233 #endif
13234
perf_control()13235 omx_vdec::perf_control::perf_control()
13236 {
13237 m_perf_lib = NULL;
13238 m_perf_handle = 0;
13239 m_perf_lock_acquire = NULL;
13240 m_perf_lock_release = NULL;
13241 }
13242
~perf_control()13243 omx_vdec::perf_control::~perf_control()
13244 {
13245 if (m_perf_handle != 0 && m_perf_lock_release) {
13246 DEBUG_PRINT_LOW("NOTE2: release perf lock");
13247 m_perf_lock_release(m_perf_handle);
13248 }
13249 if (m_perf_lib) {
13250 dlclose(m_perf_lib);
13251 }
13252 }
13253
13254 struct omx_vdec::perf_control::mpctl_stats omx_vdec::perf_control::mpctl_obj = {0, 0, 0};
13255
13256 omx_vdec::perf_lock omx_vdec::perf_control::m_perf_lock;
13257
send_hint_to_mpctl(bool state)13258 void omx_vdec::perf_control::send_hint_to_mpctl(bool state)
13259 {
13260 if (load_lib() == false) {
13261 return;
13262 }
13263 m_perf_lock.lock();
13264 /* 0x4401 maps to video decode playback hint
13265 * in perflock, enum number is 44 and state
13266 * being sent on perflock acquire is 01 (true)
13267 */
13268 int arg = 0x4401;
13269
13270 if (state == true) {
13271 mpctl_obj.vid_inst_count++;
13272 } else if (state == false) {
13273 mpctl_obj.vid_inst_count--;
13274 }
13275
13276 if (m_perf_lock_acquire && mpctl_obj.vid_inst_count == 1 && mpctl_obj.vid_acquired == false) {
13277 mpctl_obj.vid_disp_handle = m_perf_lock_acquire(0, 0, &arg, sizeof(arg) / sizeof(int));
13278 mpctl_obj.vid_acquired = true;
13279 DEBUG_PRINT_INFO("Video slvp perflock acquired");
13280 } else if (m_perf_lock_release && (mpctl_obj.vid_inst_count == 0 || mpctl_obj.vid_inst_count > 1) && mpctl_obj.vid_acquired == true) {
13281 m_perf_lock_release(mpctl_obj.vid_disp_handle);
13282 mpctl_obj.vid_acquired = false;
13283 DEBUG_PRINT_INFO("Video slvp perflock released");
13284 }
13285 m_perf_lock.unlock();
13286 }
13287
request_cores(int frame_duration_us)13288 void omx_vdec::perf_control::request_cores(int frame_duration_us)
13289 {
13290 if (frame_duration_us > MIN_FRAME_DURATION_FOR_PERF_REQUEST_US) {
13291 return;
13292 }
13293 bool retVal = load_lib();
13294 if (retVal && m_perf_lock_acquire && m_perf_handle == 0) {
13295 int arg = 0x700 /*base value*/ + 2 /*cores*/;
13296 m_perf_handle = m_perf_lock_acquire(m_perf_handle, 0, &arg, sizeof(arg)/sizeof(int));
13297 if (m_perf_handle) {
13298 DEBUG_PRINT_HIGH("perf lock acquired");
13299 }
13300 }
13301 }
13302
load_lib()13303 bool omx_vdec::perf_control::load_lib()
13304 {
13305 char perf_lib_path[PROPERTY_VALUE_MAX] = {0};
13306 if (m_perf_lib)
13307 return true;
13308
13309 if((property_get("ro.vendor.extension_library", perf_lib_path, NULL) <= 0)) {
13310 DEBUG_PRINT_ERROR("vendor library not set in ro.vendor.extension_library");
13311 goto handle_err;
13312 }
13313
13314 if ((m_perf_lib = dlopen(perf_lib_path, RTLD_NOW)) == NULL) {
13315 DEBUG_PRINT_ERROR("Failed to open %s : %s",perf_lib_path, dlerror());
13316 goto handle_err;
13317 } else {
13318 m_perf_lock_acquire = (perf_lock_acquire_t)dlsym(m_perf_lib, "perf_lock_acq");
13319 if (m_perf_lock_acquire == NULL) {
13320 DEBUG_PRINT_ERROR("Failed to load symbol: perf_lock_acq");
13321 goto handle_err;
13322 }
13323 m_perf_lock_release = (perf_lock_release_t)dlsym(m_perf_lib, "perf_lock_rel");
13324 if (m_perf_lock_release == NULL) {
13325 DEBUG_PRINT_ERROR("Failed to load symbol: perf_lock_rel");
13326 goto handle_err;
13327 }
13328 }
13329 return true;
13330
13331 handle_err:
13332 if (m_perf_lib) {
13333 dlclose(m_perf_lib);
13334 }
13335 m_perf_lib = NULL;
13336 return false;
13337 }
13338
enable_adaptive_playback(unsigned long nMaxFrameWidth,unsigned long nMaxFrameHeight)13339 OMX_ERRORTYPE omx_vdec::enable_adaptive_playback(unsigned long nMaxFrameWidth,
13340 unsigned long nMaxFrameHeight)
13341 {
13342
13343 OMX_ERRORTYPE eRet = OMX_ErrorNone;
13344 int ret = 0;
13345 unsigned long min_res_buf_count = 0;
13346
13347 eRet = enable_smoothstreaming();
13348 if (eRet != OMX_ErrorNone) {
13349 DEBUG_PRINT_ERROR("Failed to enable Adaptive Playback on driver");
13350 return eRet;
13351 }
13352
13353 DEBUG_PRINT_HIGH("Enabling Adaptive playback for %lu x %lu",
13354 nMaxFrameWidth,
13355 nMaxFrameHeight);
13356 m_smoothstreaming_mode = true;
13357 m_smoothstreaming_width = nMaxFrameWidth;
13358 m_smoothstreaming_height = nMaxFrameHeight;
13359
13360 //Get upper limit buffer count for min supported resolution
13361 struct v4l2_format fmt;
13362 fmt.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
13363 fmt.fmt.pix_mp.height = m_decoder_capability.min_height;
13364 fmt.fmt.pix_mp.width = m_decoder_capability.min_width;
13365 fmt.fmt.pix_mp.pixelformat = output_capability;
13366
13367 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
13368 if (ret) {
13369 DEBUG_PRINT_ERROR("Set Resolution failed for HxW = %ux%u",
13370 m_decoder_capability.min_height,
13371 m_decoder_capability.min_width);
13372 return OMX_ErrorUnsupportedSetting;
13373 }
13374
13375 eRet = get_buffer_req(&drv_ctx.op_buf);
13376 if (eRet != OMX_ErrorNone) {
13377 DEBUG_PRINT_ERROR("failed to get_buffer_req");
13378 return eRet;
13379 }
13380
13381 min_res_buf_count = drv_ctx.op_buf.mincount;
13382 DEBUG_PRINT_LOW("enable adaptive - upper limit buffer count = %lu for HxW %ux%u",
13383 min_res_buf_count, m_decoder_capability.min_height, m_decoder_capability.min_width);
13384
13385 m_extradata_info.output_crop_rect.nLeft = 0;
13386 m_extradata_info.output_crop_rect.nTop = 0;
13387 m_extradata_info.output_crop_rect.nWidth = m_smoothstreaming_width;
13388 m_extradata_info.output_crop_rect.nHeight = m_smoothstreaming_height;
13389
13390 update_resolution(m_smoothstreaming_width, m_smoothstreaming_height,
13391 m_smoothstreaming_width, m_smoothstreaming_height);
13392 eRet = is_video_session_supported();
13393 if (eRet != OMX_ErrorNone) {
13394 DEBUG_PRINT_ERROR("video session is not supported");
13395 return eRet;
13396 }
13397
13398 //Get upper limit buffer size for max smooth streaming resolution set
13399 fmt.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
13400 fmt.fmt.pix_mp.height = drv_ctx.video_resolution.frame_height;
13401 fmt.fmt.pix_mp.width = drv_ctx.video_resolution.frame_width;
13402 fmt.fmt.pix_mp.pixelformat = output_capability;
13403 ret = ioctl(drv_ctx.video_driver_fd, VIDIOC_S_FMT, &fmt);
13404 if (ret) {
13405 DEBUG_PRINT_ERROR("Set Resolution failed for adaptive playback");
13406 return OMX_ErrorUnsupportedSetting;
13407 }
13408
13409 eRet = get_buffer_req(&drv_ctx.op_buf);
13410 if (eRet != OMX_ErrorNone) {
13411 DEBUG_PRINT_ERROR("failed to get_buffer_req!!");
13412 return eRet;
13413 }
13414 DEBUG_PRINT_LOW("enable adaptive - upper limit buffer size = %u",
13415 (unsigned int)drv_ctx.op_buf.buffer_size);
13416
13417 drv_ctx.op_buf.mincount = min_res_buf_count;
13418 drv_ctx.op_buf.actualcount = min_res_buf_count;
13419 drv_ctx.op_buf.buffer_size = drv_ctx.op_buf.buffer_size;
13420 eRet = set_buffer_req(&drv_ctx.op_buf);
13421 if (eRet != OMX_ErrorNone) {
13422 DEBUG_PRINT_ERROR("failed to set_buffer_req");
13423 return eRet;
13424 }
13425
13426 eRet = get_buffer_req(&drv_ctx.op_buf);
13427 if (eRet != OMX_ErrorNone) {
13428 DEBUG_PRINT_ERROR("failed to get_buffer_req!!!");
13429 return eRet;
13430 }
13431 DEBUG_PRINT_HIGH("adaptive playback enabled, buf count = %u bufsize = %u",
13432 drv_ctx.op_buf.mincount, (unsigned int)drv_ctx.op_buf.buffer_size);
13433 return eRet;
13434 }
13435
13436 //static
describeColorFormat(OMX_PTR pParam)13437 OMX_ERRORTYPE omx_vdec::describeColorFormat(OMX_PTR pParam) {
13438
13439 #ifndef FLEXYUV_SUPPORTED
13440 return OMX_ErrorUndefined;
13441 #else
13442
13443 if (pParam == NULL) {
13444 DEBUG_PRINT_ERROR("describeColorFormat: invalid params");
13445 return OMX_ErrorBadParameter;
13446 }
13447
13448 DescribeColorFormatParams *params = (DescribeColorFormatParams*)pParam;
13449
13450 MediaImage *img = &(params->sMediaImage);
13451 switch(params->eColorFormat) {
13452 case QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m:
13453 {
13454 img->mType = MediaImage::MEDIA_IMAGE_TYPE_YUV;
13455 img->mNumPlanes = 3;
13456 // mWidth and mHeight represent the W x H of the largest plane
13457 // In our case, this happens to be the Stride x Scanlines of Y plane
13458 img->mWidth = params->nFrameWidth;
13459 img->mHeight = params->nFrameHeight;
13460 size_t planeWidth = VENUS_Y_STRIDE(COLOR_FMT_NV12, params->nFrameWidth);
13461 size_t planeHeight = VENUS_Y_SCANLINES(COLOR_FMT_NV12, params->nFrameHeight);
13462 img->mBitDepth = 8;
13463 //Plane 0 (Y)
13464 img->mPlane[MediaImage::Y].mOffset = 0;
13465 img->mPlane[MediaImage::Y].mColInc = 1;
13466 img->mPlane[MediaImage::Y].mRowInc = planeWidth; //same as stride
13467 img->mPlane[MediaImage::Y].mHorizSubsampling = 1;
13468 img->mPlane[MediaImage::Y].mVertSubsampling = 1;
13469 //Plane 1 (U)
13470 img->mPlane[MediaImage::U].mOffset = planeWidth * planeHeight;
13471 img->mPlane[MediaImage::U].mColInc = 2; //interleaved UV
13472 img->mPlane[MediaImage::U].mRowInc =
13473 VENUS_UV_STRIDE(COLOR_FMT_NV12, params->nFrameWidth);
13474 img->mPlane[MediaImage::U].mHorizSubsampling = 2;
13475 img->mPlane[MediaImage::U].mVertSubsampling = 2;
13476 //Plane 2 (V)
13477 img->mPlane[MediaImage::V].mOffset = planeWidth * planeHeight + 1;
13478 img->mPlane[MediaImage::V].mColInc = 2; //interleaved UV
13479 img->mPlane[MediaImage::V].mRowInc =
13480 VENUS_UV_STRIDE(COLOR_FMT_NV12, params->nFrameWidth);
13481 img->mPlane[MediaImage::V].mHorizSubsampling = 2;
13482 img->mPlane[MediaImage::V].mVertSubsampling = 2;
13483 break;
13484 }
13485
13486 case OMX_COLOR_FormatYUV420Planar:
13487 case OMX_COLOR_FormatYUV420SemiPlanar:
13488 // We need not describe the standard OMX linear formats as these are
13489 // understood by client. Fail this deliberately to let client fill-in
13490 return OMX_ErrorUnsupportedSetting;
13491
13492 default:
13493 // Rest all formats which are non-linear cannot be described
13494 DEBUG_PRINT_LOW("color-format %x is not flexible", params->eColorFormat);
13495 img->mType = MediaImage::MEDIA_IMAGE_TYPE_UNKNOWN;
13496 return OMX_ErrorNone;
13497 };
13498
13499 DEBUG_PRINT_LOW("NOTE: Describe color format : %x", params->eColorFormat);
13500 DEBUG_PRINT_LOW(" FrameWidth x FrameHeight : %d x %d", params->nFrameWidth, params->nFrameHeight);
13501 DEBUG_PRINT_LOW(" YWidth x YHeight : %d x %d", img->mWidth, img->mHeight);
13502 for (size_t i = 0; i < img->mNumPlanes; ++i) {
13503 DEBUG_PRINT_LOW(" Plane[%zu] : offset=%d / xStep=%d / yStep = %d",
13504 i, img->mPlane[i].mOffset, img->mPlane[i].mColInc, img->mPlane[i].mRowInc);
13505 }
13506 return OMX_ErrorNone;
13507 #endif //FLEXYUV_SUPPORTED
13508 }
13509
prefetchNewBuffers()13510 void omx_vdec::prefetchNewBuffers() {
13511
13512 struct v4l2_decoder_cmd dec;
13513 uint32_t prefetch_count;
13514 uint32_t prefetch_size;
13515 uint32_t want_size;
13516 uint32_t have_size;
13517 int color_fmt, rc;
13518 uint32_t new_calculated_size;
13519 uint32_t new_buffer_size;
13520 uint32_t new_buffer_count;
13521 uint32_t old_buffer_size;
13522 uint32_t old_buffer_count;
13523
13524 memset((void *)&dec, 0 , sizeof(dec));
13525 DEBUG_PRINT_LOW("Old size : %zu, count : %d, width : %u, height : %u\n",
13526 drv_ctx.op_buf.buffer_size, drv_ctx.op_buf.actualcount,
13527 drv_ctx.video_resolution.frame_width,
13528 drv_ctx.video_resolution.frame_height);
13529 dec.cmd = V4L2_DEC_QCOM_CMD_RECONFIG_HINT;
13530 if (ioctl(drv_ctx.video_driver_fd, VIDIOC_DECODER_CMD, &dec)) {
13531 DEBUG_PRINT_ERROR("Buffer info cmd failed : %d\n", errno);
13532 } else {
13533 DEBUG_PRINT_LOW("From driver, new size is %d, count is %d\n",
13534 dec.raw.data[0], dec.raw.data[1]);
13535 }
13536
13537 switch ((int)drv_ctx.output_format) {
13538 case VDEC_YUV_FORMAT_NV12:
13539 color_fmt = COLOR_FMT_NV12;
13540 break;
13541 case VDEC_YUV_FORMAT_NV12_UBWC:
13542 color_fmt = COLOR_FMT_NV12_UBWC;
13543 break;
13544 case VDEC_YUV_FORMAT_NV12_TP10_UBWC:
13545 color_fmt = COLOR_FMT_NV12_BPP10_UBWC;
13546 break;
13547 default:
13548 color_fmt = -1;
13549 DEBUG_PRINT_HIGH("Color format : %x not supported for secure memory prefetching\n", drv_ctx.output_format);
13550 return;
13551 }
13552
13553 new_calculated_size = VENUS_BUFFER_SIZE(color_fmt, m_reconfig_width, m_reconfig_height);
13554 DEBUG_PRINT_LOW("New calculated size for width : %d, height : %d, is %d\n",
13555 m_reconfig_width, m_reconfig_height, new_calculated_size);
13556 new_buffer_size = (dec.raw.data[0] > new_calculated_size) ? dec.raw.data[0] : new_calculated_size;
13557 new_buffer_count = dec.raw.data[1];
13558 old_buffer_size = drv_ctx.op_buf.buffer_size;
13559 old_buffer_count = drv_ctx.op_buf.actualcount;
13560
13561 new_buffer_count = old_buffer_count > new_buffer_count ? old_buffer_count : new_buffer_count;
13562
13563 prefetch_count = new_buffer_count;
13564 prefetch_size = new_buffer_size - old_buffer_size;
13565 want_size = new_buffer_size * new_buffer_count;
13566 have_size = old_buffer_size * old_buffer_count;
13567
13568 if (want_size > have_size) {
13569 DEBUG_PRINT_LOW("Want: %d, have : %d\n", want_size, have_size);
13570 DEBUG_PRINT_LOW("prefetch_count: %d, prefetch_size : %d\n", prefetch_count, prefetch_size);
13571
13572 int ion_fd = open(MEM_DEVICE, O_RDONLY);
13573 if (ion_fd < 0) {
13574 DEBUG_PRINT_ERROR("Ion fd open failed : %d\n", ion_fd);
13575 return;
13576 }
13577
13578 struct ion_custom_data *custom_data = (struct ion_custom_data*) malloc(sizeof(*custom_data));
13579 struct ion_prefetch_data *prefetch_data = (struct ion_prefetch_data*) malloc(sizeof(*prefetch_data));
13580 struct ion_prefetch_regions *regions = (struct ion_prefetch_regions*) malloc(sizeof(*regions));
13581 size_t *sizes = (size_t*) malloc(sizeof(size_t) * prefetch_count);
13582
13583 if (custom_data == NULL || prefetch_data == NULL || regions == NULL || sizes == NULL) {
13584 DEBUG_PRINT_ERROR("prefetch data allocation failed");
13585 goto prefetch_exit;
13586 }
13587
13588 for (uint32_t i = 0; i < prefetch_count; i++) {
13589 sizes[i] = prefetch_size;
13590 }
13591
13592 regions[0].nr_sizes = prefetch_count;
13593 regions[0].sizes = sizes;
13594 regions[0].vmid = ION_FLAG_CP_PIXEL;
13595
13596 prefetch_data->nr_regions = 1;
13597 prefetch_data->regions = regions;
13598 prefetch_data->heap_id = ION_HEAP(ION_SECURE_HEAP_ID);
13599
13600 custom_data->cmd = ION_IOC_PREFETCH;
13601 custom_data->arg = (unsigned long )prefetch_data;
13602
13603 rc = ioctl(ion_fd, ION_IOC_CUSTOM, custom_data);
13604 if (rc) {
13605 DEBUG_PRINT_ERROR("Custom prefetch ioctl failed rc : %d, errno : %d\n", rc, errno);
13606 }
13607
13608 prefetch_exit:
13609 close(ion_fd);
13610 free(sizes);
13611 free(regions);
13612 free(prefetch_data);
13613 free(custom_data);
13614 }
13615 }
13616
13617
13618 // No code beyond this !
13619
13620 // inline import of vendor-extensions implementation
13621 #include "omx_vdec_extensions.hpp"
13622