1 /*
2 * Copyright (c) 2012 Intel Corporation. All rights reserved.
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17
18 #define LOG_NDEBUG 0
19 #define LOG_TAG "OMXVideoDecoderVP9HWR"
20 #include <wrs_omxil_core/log.h>
21 #include "OMXVideoDecoderVP9HWR.h"
22
23 #include <system/window.h>
24 #include <HardwareAPI.h>
25 #include <hardware/hardware.h>
26 #include <hardware/gralloc.h>
27 #include <system/graphics.h>
28
29 static const char* VP9_MIME_TYPE = "video/x-vnd.on2.vp9";
30
GetCPUCoreCount()31 static int GetCPUCoreCount()
32 {
33 int cpuCoreCount = 1;
34 #if defined(_SC_NPROCESSORS_ONLN)
35 cpuCoreCount = sysconf(_SC_NPROCESSORS_ONLN);
36 #else
37 // _SC_NPROC_ONLN must be defined...
38 cpuCoreCount = sysconf(_SC_NPROC_ONLN);
39 #endif
40 if (cpuCoreCount < 1) {
41 LOGW("Get CPU Core Count error.");
42 cpuCoreCount = 1;
43 }
44 LOGV("Number of CPU cores: %d", cpuCoreCount);
45 return cpuCoreCount;
46 }
47
48
OMXVideoDecoderVP9HWR()49 OMXVideoDecoderVP9HWR::OMXVideoDecoderVP9HWR()
50 {
51 LOGV("OMXVideoDecoderVP9HWR is constructed.");
52
53 mNumFrameBuffer = 0;
54 mCtx = NULL;
55
56 mNativeBufferCount = OUTPORT_NATIVE_BUFFER_COUNT;
57 extUtilBufferCount = 0;
58 extMappedNativeBufferCount = 0;
59 BuildHandlerList();
60
61 mDecodedImageWidth = 0;
62 mDecodedImageHeight = 0;
63 mDecodedImageNewWidth = 0;
64 mDecodedImageNewHeight = 0;
65
66 #ifdef DECODE_WITH_GRALLOC_BUFFER
67 // setup va
68 VAStatus vaStatus = VA_STATUS_SUCCESS;
69 mDisplay = new Display;
70 *mDisplay = ANDROID_DISPLAY_HANDLE;
71
72 mVADisplay = vaGetDisplay(mDisplay);
73 if (mVADisplay == NULL) {
74 LOGE("vaGetDisplay failed.");
75 }
76
77 int majorVersion, minorVersion;
78 vaStatus = vaInitialize(mVADisplay, &majorVersion, &minorVersion);
79 if (vaStatus != VA_STATUS_SUCCESS) {
80 LOGE("vaInitialize failed.");
81 } else {
82 LOGV("va majorVersion=%d, minorVersion=%d", majorVersion, minorVersion);
83 }
84 #endif
85 }
86
~OMXVideoDecoderVP9HWR()87 OMXVideoDecoderVP9HWR::~OMXVideoDecoderVP9HWR()
88 {
89 LOGV("OMXVideoDecoderVP9HWR is destructed.");
90
91 if (mVADisplay) {
92 vaTerminate(mVADisplay);
93 mVADisplay = NULL;
94 }
95 }
96
97
98 // Callback func for vpx decoder to get decode buffer
99 // Now we map from the vaSurface to deploy gralloc buffer
100 // as decode buffer
getVP9FrameBuffer(void * user_priv,unsigned int new_size,vpx_codec_frame_buffer_t * fb)101 int getVP9FrameBuffer(void *user_priv,
102 unsigned int new_size,
103 vpx_codec_frame_buffer_t *fb)
104 {
105 OMXVideoDecoderVP9HWR * p = (OMXVideoDecoderVP9HWR *)user_priv;
106 if (fb == NULL) {
107 return -1;
108 }
109
110 // TODO: Adaptive playback case needs to reconsider
111 if (p->extNativeBufferSize < new_size) {
112 LOGE("Provided frame buffer size < requesting min size.");
113 return -1;
114 }
115
116 int i;
117 for (i = 0; i < p->extMappedNativeBufferCount; i++ ) {
118 if ((p->extMIDs[i]->m_render_done == true) &&
119 (p->extMIDs[i]->m_released == true)) {
120 fb->data = p->extMIDs[i]->m_usrAddr;
121 fb->size = p->extNativeBufferSize;
122 fb->fb_stride = p->extActualBufferStride;
123 fb->fb_height_stride = p->extActualBufferHeightStride;
124 fb->fb_index = i;
125 p->extMIDs[i]->m_released = false;
126 break;
127 }
128 }
129
130 if (i == p->extMappedNativeBufferCount) {
131 LOGE("No available frame buffer in pool.");
132 return -1;
133 }
134 return 0;
135 }
136
137 // call back function from libvpx to inform frame buffer
138 // not used anymore.
releaseVP9FrameBuffer(void * user_priv,vpx_codec_frame_buffer_t * fb)139 int releaseVP9FrameBuffer(void *user_priv, vpx_codec_frame_buffer_t *fb)
140 {
141 int i;
142 OMXVideoDecoderVP9HWR * p = (OMXVideoDecoderVP9HWR *)user_priv;
143 if (fb == NULL) {
144 return -1;
145 }
146 for (i = 0; i < p->extMappedNativeBufferCount; i++ ) {
147 if (fb->data == p->extMIDs[i]->m_usrAddr) {
148 p->extMIDs[i]->m_released = true;
149 break;
150 }
151 }
152
153 if (i == p->extMappedNativeBufferCount) {
154 LOGE("Not found matching frame buffer in pool, libvpx's wrong?");
155 return -1;
156 }
157 return 0;
158 }
159
160
initDecoder()161 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::initDecoder()
162 {
163 mCtx = new vpx_codec_ctx_t;
164 vpx_codec_err_t vpx_err;
165 vpx_codec_dec_cfg_t cfg;
166 memset(&cfg, 0, sizeof(vpx_codec_dec_cfg_t));
167 cfg.threads = GetCPUCoreCount();
168 if ((vpx_err = vpx_codec_dec_init(
169 (vpx_codec_ctx_t *)mCtx,
170 &vpx_codec_vp9_dx_algo,
171 &cfg, 0))) {
172 LOGE("on2 decoder failed to initialize. (%d)", vpx_err);
173 return OMX_ErrorNotReady;
174 }
175
176 mNumFrameBuffer = OUTPORT_NATIVE_BUFFER_COUNT;
177
178 if (vpx_codec_set_frame_buffer_functions((vpx_codec_ctx_t *)mCtx,
179 getVP9FrameBuffer,
180 releaseVP9FrameBuffer,
181 this)) {
182 LOGE("Failed to configure external frame buffers");
183 return OMX_ErrorNotReady;
184 }
185
186 return OMX_ErrorNone;
187 }
188
destroyDecoder()189 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::destroyDecoder()
190 {
191 vpx_codec_destroy((vpx_codec_ctx_t *)mCtx);
192 delete (vpx_codec_ctx_t *)mCtx;
193 mCtx = NULL;
194 return OMX_ErrorNone;
195 }
196
197
InitInputPortFormatSpecific(OMX_PARAM_PORTDEFINITIONTYPE * paramPortDefinitionInput)198 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::InitInputPortFormatSpecific(
199 OMX_PARAM_PORTDEFINITIONTYPE *paramPortDefinitionInput)
200 {
201 paramPortDefinitionInput->nBufferCountActual = INPORT_ACTUAL_BUFFER_COUNT;
202 paramPortDefinitionInput->nBufferCountMin = INPORT_MIN_BUFFER_COUNT;
203 paramPortDefinitionInput->nBufferSize = INPORT_BUFFER_SIZE;
204 paramPortDefinitionInput->format.video.cMIMEType = (OMX_STRING)VP9_MIME_TYPE;
205 paramPortDefinitionInput->format.video.eCompressionFormat = OMX_VIDEO_CodingVP9;
206
207 return OMX_ErrorNone;
208 }
209
ProcessorInit(void)210 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::ProcessorInit(void)
211 {
212 unsigned int i = 0;
213
214 for (i = 0; i < MAX_NATIVE_BUFFER_COUNT; i++) {
215 extMIDs[i] = (vaapiMemId*)malloc(sizeof(vaapiMemId));
216 extMIDs[i]->m_usrAddr = NULL;
217 extMIDs[i]->m_surface = new VASurfaceID;
218 }
219
220 initDecoder();
221
222 if (RAWDATA_MODE == mWorkingMode) {
223 OMX_PARAM_PORTDEFINITIONTYPE paramPortDefinitionInput;
224
225 memcpy(¶mPortDefinitionInput,
226 this->ports[INPORT_INDEX]->GetPortDefinition(),
227 sizeof(paramPortDefinitionInput));
228
229 extNativeBufferSize = INTERNAL_MAX_FRAME_WIDTH *
230 INTERNAL_MAX_FRAME_HEIGHT * 1.5;
231 extActualBufferStride = INTERNAL_MAX_FRAME_WIDTH;
232 extActualBufferHeightStride = INTERNAL_MAX_FRAME_HEIGHT;
233
234 for (i = 0; i < OUTPORT_ACTUAL_BUFFER_COUNT; i++ ) {
235 extMIDs[i]->m_usrAddr = (unsigned char*)malloc(sizeof(unsigned char) *
236 extNativeBufferSize);
237 extMIDs[i]->m_render_done = true;
238 extMIDs[i]->m_released = true;
239 }
240 extMappedNativeBufferCount = OUTPORT_ACTUAL_BUFFER_COUNT;
241 return OMX_ErrorNone;
242 }
243
244 #ifdef DECODE_WITH_GRALLOC_BUFFER
245 if (mOMXBufferHeaderTypePtrNum > MAX_NATIVE_BUFFER_COUNT) {
246 LOGE("Actual OMX outport buffer header type num (%d) > MAX_NATIVE_BUFFER_COUNT (%d)",
247 mOMXBufferHeaderTypePtrNum, MAX_NATIVE_BUFFER_COUNT);
248 return OMX_ErrorOverflow;
249 }
250
251 OMX_PARAM_PORTDEFINITIONTYPE paramPortDefinitionInput;
252
253 memcpy(¶mPortDefinitionInput,
254 this->ports[INPORT_INDEX]->GetPortDefinition(),
255 sizeof(paramPortDefinitionInput));
256
257 int surfaceWidth = mGraphicBufferParam.graphicBufferWidth;
258 int surfaceHeight = mGraphicBufferParam.graphicBufferHeight;
259 int surfaceStride = mGraphicBufferParam.graphicBufferStride;
260 extNativeBufferSize = mGraphicBufferParam.graphicBufferStride *
261 mGraphicBufferParam.graphicBufferHeight * 1.5;
262 extActualBufferStride = surfaceStride;
263 extActualBufferHeightStride = surfaceHeight;
264
265 for (i = 0; i < mOMXBufferHeaderTypePtrNum; i++) {
266 OMX_BUFFERHEADERTYPE *buf_hdr = mOMXBufferHeaderTypePtrArray[i];
267
268 extMIDs[i]->m_key = (unsigned int)(buf_hdr->pBuffer);
269 extMIDs[i]->m_render_done = false;
270 extMIDs[i]->m_released = true;
271
272 VAStatus va_res;
273 unsigned int buffer;
274 VASurfaceAttrib attribs[2];
275 VASurfaceAttribExternalBuffers* surfExtBuf = new VASurfaceAttribExternalBuffers;
276 int32_t format = VA_RT_FORMAT_YUV420;
277
278 surfExtBuf->buffers= (unsigned long *)&buffer;
279 surfExtBuf->num_buffers = 1;
280 surfExtBuf->pixel_format = VA_FOURCC_NV12;
281 surfExtBuf->width = surfaceWidth;
282 surfExtBuf->height = surfaceHeight;
283 surfExtBuf->data_size = surfaceStride * surfaceHeight * 1.5;
284 surfExtBuf->num_planes = 2;
285 surfExtBuf->pitches[0] = surfaceStride;
286 surfExtBuf->pitches[1] = surfaceStride;
287 surfExtBuf->pitches[2] = 0;
288 surfExtBuf->pitches[3] = 0;
289 surfExtBuf->offsets[0] = 0;
290 surfExtBuf->offsets[1] = surfaceStride * surfaceHeight;
291 surfExtBuf->offsets[2] = 0;
292 surfExtBuf->offsets[3] = 0;
293 surfExtBuf->flags = VA_SURFACE_ATTRIB_MEM_TYPE_ANDROID_GRALLOC;
294
295 surfExtBuf->buffers[0] = (unsigned int)buf_hdr->pBuffer;
296
297 attribs[0].type = (VASurfaceAttribType)VASurfaceAttribMemoryType;
298 attribs[0].flags = VA_SURFACE_ATTRIB_SETTABLE;
299 attribs[0].value.type = VAGenericValueTypeInteger;
300 attribs[0].value.value.i = VA_SURFACE_ATTRIB_MEM_TYPE_ANDROID_GRALLOC;
301
302 attribs[1].type = (VASurfaceAttribType)VASurfaceAttribExternalBufferDescriptor;
303 attribs[1].flags = VA_SURFACE_ATTRIB_SETTABLE;
304 attribs[1].value.type = VAGenericValueTypePointer;
305 attribs[1].value.value.p = (void *)surfExtBuf;
306
307 va_res = vaCreateSurfaces(mVADisplay,
308 format,
309 surfaceWidth,
310 surfaceHeight,
311 extMIDs[i]->m_surface,
312 1,
313 attribs,
314 2);
315
316 if (va_res != VA_STATUS_SUCCESS) {
317 LOGE("Failed to create vaSurface!");
318 return OMX_ErrorUndefined;
319 }
320
321 delete surfExtBuf;
322
323 VAImage image;
324 unsigned char* usrptr;
325
326 va_res = vaDeriveImage(mVADisplay, *(extMIDs[i]->m_surface), &image);
327 if (VA_STATUS_SUCCESS == va_res) {
328 va_res = vaMapBuffer(mVADisplay, image.buf, (void **) &usrptr);
329 if (VA_STATUS_SUCCESS == va_res) {
330 extMIDs[i]->m_usrAddr = usrptr;
331 vaUnmapBuffer(mVADisplay, image.buf);
332 }
333 vaDestroyImage(mVADisplay, image.image_id);
334 }
335 extMappedNativeBufferCount++;
336 }
337 return OMX_ErrorNone;
338 #endif
339 }
340
ProcessorDeinit(void)341 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::ProcessorDeinit(void)
342 {
343 destroyDecoder();
344 unsigned int i = 0;
345 if (mWorkingMode == GRAPHICBUFFER_MODE) {
346 for (i = 0; i < mOMXBufferHeaderTypePtrNum; i++) {
347 if (extMIDs[i]->m_surface != NULL) {
348 vaDestroySurfaces(mVADisplay, extMIDs[i]->m_surface, 1);
349 }
350 }
351
352 } else if (mWorkingMode == RAWDATA_MODE) {
353 for (i = 0; i < OUTPORT_ACTUAL_BUFFER_COUNT; i++ ) {
354 if (extMIDs[i]->m_usrAddr != NULL) {
355 free(extMIDs[i]->m_usrAddr);
356 extMIDs[i]->m_usrAddr = NULL;
357 }
358 }
359 }
360 mOMXBufferHeaderTypePtrNum = 0;
361 memset(&mGraphicBufferParam, 0, sizeof(mGraphicBufferParam));
362 for (i = 0; i < MAX_NATIVE_BUFFER_COUNT; i++) {
363 delete extMIDs[i]->m_surface;
364 free(extMIDs[i]);
365 }
366 return OMXComponentCodecBase::ProcessorDeinit();
367 }
368
ProcessorStop(void)369 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::ProcessorStop(void)
370 {
371 return OMXComponentCodecBase::ProcessorStop();
372 }
373
ProcessorFlush(OMX_U32)374 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::ProcessorFlush(OMX_U32)
375 {
376 return OMX_ErrorNone;
377 }
378
ProcessorPreFillBuffer(OMX_BUFFERHEADERTYPE * buffer)379 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::ProcessorPreFillBuffer(OMX_BUFFERHEADERTYPE* buffer)
380 {
381 unsigned int handle = (unsigned int)buffer->pBuffer;
382 unsigned int i = 0;
383
384 if (buffer->nOutputPortIndex == OUTPORT_INDEX){
385 for (i = 0; i < mOMXBufferHeaderTypePtrNum; i++) {
386 if (handle == extMIDs[i]->m_key) {
387 extMIDs[i]->m_render_done = true;
388 break;
389 }
390 }
391 }
392 return OMX_ErrorNone;
393 }
394
ProcessorProcess(OMX_BUFFERHEADERTYPE *** pBuffers,buffer_retain_t * retains,OMX_U32)395 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::ProcessorProcess(
396 OMX_BUFFERHEADERTYPE ***pBuffers,
397 buffer_retain_t *retains,
398 OMX_U32)
399 {
400 OMX_ERRORTYPE ret;
401 OMX_BUFFERHEADERTYPE *inBuffer = *pBuffers[INPORT_INDEX];
402 OMX_BOOL isResolutionChange = OMX_FALSE;
403
404 if (inBuffer->pBuffer == NULL) {
405 LOGE("Buffer to decode is empty.");
406 return OMX_ErrorBadParameter;
407 }
408
409 if (inBuffer->nFlags & OMX_BUFFERFLAG_CODECCONFIG) {
410 LOGI("Buffer has OMX_BUFFERFLAG_CODECCONFIG flag.");
411 }
412
413 if (inBuffer->nFlags & OMX_BUFFERFLAG_DECODEONLY) {
414 LOGW("Buffer has OMX_BUFFERFLAG_DECODEONLY flag.");
415 }
416
417 if (inBuffer->nFlags & OMX_BUFFERFLAG_EOS) {
418 if (inBuffer->nFilledLen == 0) {
419 (*pBuffers[OUTPORT_INDEX])->nFilledLen = 0;
420 (*pBuffers[OUTPORT_INDEX])->nFlags = OMX_BUFFERFLAG_EOS;
421 return OMX_ErrorNone;
422 }
423 }
424
425 if (vpx_codec_decode((vpx_codec_ctx_t *)mCtx,
426 inBuffer->pBuffer + inBuffer->nOffset,
427 inBuffer->nFilledLen,
428 NULL,
429 0)) {
430 LOGE("on2 decoder failed to decode frame.");
431 return OMX_ErrorBadParameter;
432 }
433
434 ret = FillRenderBuffer(pBuffers[OUTPORT_INDEX],
435 &retains[OUTPORT_INDEX],
436 ((*pBuffers[INPORT_INDEX]))->nFlags,
437 &isResolutionChange);
438
439 if (ret == OMX_ErrorNone) {
440 (*pBuffers[OUTPORT_INDEX])->nTimeStamp = inBuffer->nTimeStamp;
441 }
442
443 if (isResolutionChange) {
444 HandleFormatChange();
445 }
446
447 bool inputEoS = ((*pBuffers[INPORT_INDEX])->nFlags & OMX_BUFFERFLAG_EOS);
448 bool outputEoS = ((*pBuffers[OUTPORT_INDEX])->nFlags & OMX_BUFFERFLAG_EOS);
449 // if output port is not eos, retain the input buffer
450 // until all the output buffers are drained.
451 if (inputEoS && !outputEoS) {
452 retains[INPORT_INDEX] = BUFFER_RETAIN_GETAGAIN;
453 // the input buffer is retained for draining purpose.
454 // Set nFilledLen to 0 so buffer will not be decoded again.
455 (*pBuffers[INPORT_INDEX])->nFilledLen = 0;
456 }
457
458 if (ret == OMX_ErrorNotReady) {
459 retains[OUTPORT_INDEX] = BUFFER_RETAIN_GETAGAIN;
460 ret = OMX_ErrorNone;
461 }
462
463 return ret;
464 }
465
ProcessorReset(void)466 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::ProcessorReset(void)
467 {
468 return OMX_ErrorNone;
469 }
470
ALIGN(int x,int y)471 static int ALIGN(int x, int y)
472 {
473 // y must be a power of 2.
474 return (x + y - 1) & ~(y - 1);
475 }
476
HandleFormatChange(void)477 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::HandleFormatChange(void)
478 {
479 mDecodedImageWidth = mDecodedImageNewWidth;
480 mDecodedImageHeight = mDecodedImageNewHeight;
481
482 // Sync port definition as it may change.
483 OMX_PARAM_PORTDEFINITIONTYPE paramPortDefinitionInput, paramPortDefinitionOutput;
484
485 memcpy(¶mPortDefinitionInput,
486 this->ports[INPORT_INDEX]->GetPortDefinition(),
487 sizeof(paramPortDefinitionInput));
488
489 memcpy(¶mPortDefinitionOutput,
490 this->ports[OUTPORT_INDEX]->GetPortDefinition(),
491 sizeof(paramPortDefinitionOutput));
492
493 uint32_t width = mDecodedImageWidth;
494 uint32_t height = mDecodedImageHeight;
495 uint32_t stride = mDecodedImageWidth;
496 uint32_t sliceHeight = mDecodedImageHeight;
497
498 uint32_t widthCropped = mDecodedImageWidth;
499 uint32_t heightCropped = mDecodedImageHeight;
500 uint32_t strideCropped = widthCropped;
501 uint32_t sliceHeightCropped = heightCropped;
502
503 if (widthCropped == paramPortDefinitionOutput.format.video.nFrameWidth &&
504 heightCropped == paramPortDefinitionOutput.format.video.nFrameHeight) {
505 if (mWorkingMode == RAWDATA_MODE) {
506 LOGW("Change of portsetting is not reported as size is not changed.");
507 return OMX_ErrorNone;
508 }
509 }
510
511 paramPortDefinitionInput.format.video.nFrameWidth = width;
512 paramPortDefinitionInput.format.video.nFrameHeight = height;
513 paramPortDefinitionInput.format.video.nStride = stride;
514 paramPortDefinitionInput.format.video.nSliceHeight = sliceHeight;
515
516 if (mWorkingMode == RAWDATA_MODE) {
517 paramPortDefinitionOutput.format.video.nFrameWidth = widthCropped;
518 paramPortDefinitionOutput.format.video.nFrameHeight = heightCropped;
519 paramPortDefinitionOutput.format.video.nStride = strideCropped;
520 paramPortDefinitionOutput.format.video.nSliceHeight = sliceHeightCropped;
521 } else if (mWorkingMode == GRAPHICBUFFER_MODE) {
522 // when the width and height ES parse are not larger than allocated graphic buffer in outport,
523 // there is no need to reallocate graphic buffer,just report the crop info to omx client
524 if (width <= mGraphicBufferParam.graphicBufferWidth &&
525 height <= mGraphicBufferParam.graphicBufferHeight) {
526 this->ports[INPORT_INDEX]->SetPortDefinition(¶mPortDefinitionInput, true);
527 this->ports[OUTPORT_INDEX]->ReportOutputCrop();
528 return OMX_ErrorNone;
529 }
530
531 if (width > mGraphicBufferParam.graphicBufferWidth ||
532 height > mGraphicBufferParam.graphicBufferHeight) {
533 // update the real decoded resolution to outport instead of display resolution
534 // for graphic buffer reallocation
535 // when the width and height parsed from ES are larger than allocated graphic buffer in outport,
536 paramPortDefinitionOutput.format.video.nFrameWidth = width;
537 paramPortDefinitionOutput.format.video.nFrameHeight = (height + 0x1f) & ~0x1f;
538 paramPortDefinitionOutput.format.video.eColorFormat = GetOutputColorFormat(
539 paramPortDefinitionOutput.format.video.nFrameWidth);
540 paramPortDefinitionOutput.format.video.nStride = stride;
541 paramPortDefinitionOutput.format.video.nSliceHeight = sliceHeight;
542 }
543 }
544
545 paramPortDefinitionOutput.bEnabled = (OMX_BOOL)false;
546 mOMXBufferHeaderTypePtrNum = 0;
547 memset(&mGraphicBufferParam, 0, sizeof(mGraphicBufferParam));
548
549 this->ports[INPORT_INDEX]->SetPortDefinition(¶mPortDefinitionInput, true);
550 this->ports[OUTPORT_INDEX]->SetPortDefinition(¶mPortDefinitionOutput, true);
551
552 this->ports[OUTPORT_INDEX]->ReportPortSettingsChanged();
553 return OMX_ErrorNone;
554 }
555
FillRenderBuffer(OMX_BUFFERHEADERTYPE ** pBuffer,buffer_retain_t * retain,OMX_U32 inportBufferFlags,OMX_BOOL * isResolutionChange)556 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::FillRenderBuffer(OMX_BUFFERHEADERTYPE **pBuffer,
557 buffer_retain_t *retain,
558 OMX_U32 inportBufferFlags,
559 OMX_BOOL *isResolutionChange)
560 {
561 OMX_BUFFERHEADERTYPE *buffer = *pBuffer;
562 OMX_BUFFERHEADERTYPE *buffer_orign = buffer;
563
564 OMX_ERRORTYPE ret = OMX_ErrorNone;
565
566 vpx_codec_iter_t iter = NULL;
567 vpx_image_t *img = NULL;
568 img = vpx_codec_get_frame((vpx_codec_ctx_t *)mCtx, &iter);
569
570 if (img != NULL) {
571 if ((mDecodedImageWidth == 0) && (mDecodedImageHeight == 0)) { // init value
572 mDecodedImageWidth = img->d_w;
573 mDecodedImageHeight = img->d_h;
574 }
575 if ((mDecodedImageWidth != img->d_w) && (mDecodedImageHeight != img->d_h)) {
576 mDecodedImageNewWidth = img->d_w;
577 mDecodedImageNewHeight = img->d_h;
578 *isResolutionChange = OMX_TRUE;
579 }
580 }
581
582 if (mWorkingMode == RAWDATA_MODE) {
583 if (img == NULL) {
584 LOGE("vpx_codec_get_frame return NULL.");
585 return OMX_ErrorNotReady;
586 }
587
588 // in Raw data mode, this flag should be always true
589 extMIDs[img->fb_index]->m_render_done = true;
590
591 void *dst = buffer->pBuffer;
592 uint8_t *dst_y = (uint8_t *)dst;
593
594 const OMX_PARAM_PORTDEFINITIONTYPE *paramPortDefinitionOutput
595 = this->ports[OUTPORT_INDEX]->GetPortDefinition();
596
597 size_t dst_y_size = paramPortDefinitionOutput->format.video.nStride *
598 paramPortDefinitionOutput->format.video.nFrameHeight;
599 size_t dst_c_stride = ALIGN(paramPortDefinitionOutput->format.video.nStride / 2, 16);
600 size_t dst_c_size = dst_c_stride * paramPortDefinitionOutput->format.video.nFrameHeight / 2;
601 uint8_t *dst_v = dst_y + dst_y_size;
602 uint8_t *dst_u = dst_v + dst_c_size;
603
604 //test border
605 dst_y += VPX_DECODE_BORDER * paramPortDefinitionOutput->format.video.nStride + VPX_DECODE_BORDER;
606 dst_v += (VPX_DECODE_BORDER/2) * dst_c_stride + (VPX_DECODE_BORDER/2);
607 dst_u += (VPX_DECODE_BORDER/2) * dst_c_stride + (VPX_DECODE_BORDER/2);
608
609 const uint8_t *srcLine = (const uint8_t *)img->planes[PLANE_Y];
610
611 for (size_t i = 0; i < img->d_h; ++i) {
612 memcpy(dst_y, srcLine, img->d_w);
613
614 srcLine += img->stride[PLANE_Y];
615 dst_y += paramPortDefinitionOutput->format.video.nStride;
616 }
617
618 srcLine = (const uint8_t *)img->planes[PLANE_U];
619 for (size_t i = 0; i < img->d_h / 2; ++i) {
620 memcpy(dst_u, srcLine, img->d_w / 2);
621
622 srcLine += img->stride[PLANE_U];
623 dst_u += dst_c_stride;
624 }
625
626 srcLine = (const uint8_t *)img->planes[PLANE_V];
627 for (size_t i = 0; i < img->d_h / 2; ++i) {
628 memcpy(dst_v, srcLine, img->d_w / 2);
629
630 srcLine += img->stride[PLANE_V];
631 dst_v += dst_c_stride;
632 }
633
634 buffer->nOffset = 0;
635 buffer->nFilledLen = dst_y_size + dst_c_size * 2;
636 if (inportBufferFlags & OMX_BUFFERFLAG_EOS) {
637 buffer->nFlags = OMX_BUFFERFLAG_EOS;
638 }
639
640 if (buffer_orign != buffer) {
641 *retain = BUFFER_RETAIN_OVERRIDDEN;
642 }
643 ret = OMX_ErrorNone;
644
645 return ret;
646
647 }
648
649 #ifdef DECODE_WITH_GRALLOC_BUFFER
650 if (NULL != img) {
651 buffer = *pBuffer = mOMXBufferHeaderTypePtrArray[img->fb_index];
652
653 if ((unsigned int)(buffer->pBuffer) != extMIDs[img->fb_index]->m_key) {
654 LOGE("There is gralloc handle mismatching between pool\
655 and mOMXBufferHeaderTypePtrArray.");
656 return OMX_ErrorNotReady;
657 }
658
659 extMIDs[img->fb_index]->m_render_done = false;
660
661 buffer->nOffset = 0;
662
663 size_t dst_y_size = img->d_w * img->d_h;
664
665 buffer->nFilledLen = dst_y_size * 1.5; // suport only 4:2:0 for now
666
667 if (inportBufferFlags & OMX_BUFFERFLAG_EOS) {
668 buffer->nFlags = OMX_BUFFERFLAG_EOS;
669 }
670
671 if (buffer_orign != buffer) {
672 *retain = BUFFER_RETAIN_OVERRIDDEN;
673 }
674
675 return OMX_ErrorNone;
676 } else {
677 LOGE("vpx_codec_get_frame return NULL.");
678 return OMX_ErrorNotReady;
679 }
680 #endif
681 }
682
PrepareConfigBuffer(VideoConfigBuffer *)683 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::PrepareConfigBuffer(VideoConfigBuffer *)
684 {
685 return OMX_ErrorNone;
686 }
687
PrepareDecodeBuffer(OMX_BUFFERHEADERTYPE *,buffer_retain_t *,VideoDecodeBuffer *)688 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::PrepareDecodeBuffer(OMX_BUFFERHEADERTYPE *,
689 buffer_retain_t *,
690 VideoDecodeBuffer *)
691 {
692 return OMX_ErrorNone;
693 }
694
BuildHandlerList(void)695 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::BuildHandlerList(void)
696 {
697 OMXVideoDecoderBase::BuildHandlerList();
698 return OMX_ErrorNone;
699 }
700
GetParamVideoVp9(OMX_PTR)701 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::GetParamVideoVp9(OMX_PTR)
702 {
703 return OMX_ErrorNone;
704 }
705
SetParamVideoVp9(OMX_PTR)706 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::SetParamVideoVp9(OMX_PTR)
707 {
708 return OMX_ErrorNone;
709 }
710
GetOutputColorFormat(int)711 OMX_COLOR_FORMATTYPE OMXVideoDecoderVP9HWR::GetOutputColorFormat(int)
712 {
713 LOGV("Output color format is OMX_INTEL_COLOR_FormatHalYV12.");
714 return (OMX_COLOR_FORMATTYPE)HAL_PIXEL_FORMAT_YV12;
715 }
716
GetDecoderOutputCropSpecific(OMX_PTR pStructure)717 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::GetDecoderOutputCropSpecific(OMX_PTR pStructure)
718 {
719 OMX_ERRORTYPE ret = OMX_ErrorNone;
720 OMX_CONFIG_RECTTYPE *rectParams = (OMX_CONFIG_RECTTYPE *)pStructure;
721
722 CHECK_TYPE_HEADER(rectParams);
723
724 if (rectParams->nPortIndex != OUTPORT_INDEX) {
725 return OMX_ErrorUndefined;
726 }
727
728 const OMX_PARAM_PORTDEFINITIONTYPE *paramPortDefinitionInput
729 = this->ports[INPORT_INDEX]->GetPortDefinition();
730
731 rectParams->nLeft = VPX_DECODE_BORDER;
732 rectParams->nTop = VPX_DECODE_BORDER;
733 rectParams->nWidth = paramPortDefinitionInput->format.video.nFrameWidth;
734 rectParams->nHeight = paramPortDefinitionInput->format.video.nFrameHeight;
735
736 return ret;
737 }
738
GetNativeBufferUsageSpecific(OMX_PTR pStructure)739 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::GetNativeBufferUsageSpecific(OMX_PTR pStructure)
740 {
741 OMX_ERRORTYPE ret;
742 android::GetAndroidNativeBufferUsageParams *param =
743 (android::GetAndroidNativeBufferUsageParams*)pStructure;
744 CHECK_TYPE_HEADER(param);
745
746 param->nUsage |= (GRALLOC_USAGE_HW_TEXTURE | GRALLOC_USAGE_SW_READ_NEVER \
747 | GRALLOC_USAGE_SW_WRITE_OFTEN | GRALLOC_USAGE_EXTERNAL_DISP);
748 return OMX_ErrorNone;
749
750 }
751
SetNativeBufferModeSpecific(OMX_PTR pStructure)752 OMX_ERRORTYPE OMXVideoDecoderVP9HWR::SetNativeBufferModeSpecific(OMX_PTR pStructure)
753 {
754 OMX_ERRORTYPE ret;
755 EnableAndroidNativeBuffersParams *param =
756 (EnableAndroidNativeBuffersParams*)pStructure;
757
758 CHECK_TYPE_HEADER(param);
759 CHECK_PORT_INDEX_RANGE(param);
760 CHECK_SET_PARAM_STATE();
761
762 if (!param->enable) {
763 mWorkingMode = RAWDATA_MODE;
764 return OMX_ErrorNone;
765 }
766 mWorkingMode = GRAPHICBUFFER_MODE;
767 PortVideo *port = NULL;
768 port = static_cast<PortVideo *>(this->ports[OUTPORT_INDEX]);
769
770
771 OMX_PARAM_PORTDEFINITIONTYPE port_def;
772 memcpy(&port_def,port->GetPortDefinition(),sizeof(port_def));
773 port_def.nBufferCountMin = mNativeBufferCount;
774 port_def.nBufferCountActual = mNativeBufferCount;
775 port_def.format.video.cMIMEType = (OMX_STRING)VA_VED_RAW_MIME_TYPE;
776 port_def.format.video.eColorFormat = OMX_INTEL_COLOR_FormatYUV420PackedSemiPlanar;
777
778 // add borders for libvpx decode need.
779 port_def.format.video.nFrameHeight += VPX_DECODE_BORDER * 2;
780 port_def.format.video.nFrameWidth += VPX_DECODE_BORDER * 2;
781 // make heigth 32bit align
782 port_def.format.video.nFrameHeight = (port_def.format.video.nFrameHeight + 0x1f) & ~0x1f;
783 port_def.format.video.eColorFormat = GetOutputColorFormat(
784 port_def.format.video.nFrameWidth);
785 port->SetPortDefinition(&port_def,true);
786
787 return OMX_ErrorNone;
788 }
789
790
IsAllBufferAvailable(void)791 bool OMXVideoDecoderVP9HWR::IsAllBufferAvailable(void)
792 {
793 bool b = ComponentBase::IsAllBufferAvailable();
794 if (b == false) {
795 return false;
796 }
797
798 PortVideo *port = NULL;
799 port = static_cast<PortVideo *>(this->ports[OUTPORT_INDEX]);
800 const OMX_PARAM_PORTDEFINITIONTYPE* port_def = port->GetPortDefinition();
801 // if output port is disabled, retain the input buffer
802 if (!port_def->bEnabled) {
803 return false;
804 }
805
806 unsigned int i = 0;
807 int found = 0;
808
809 if (RAWDATA_MODE == mWorkingMode) {
810 for (i = 0; i < OUTPORT_ACTUAL_BUFFER_COUNT; i++) {
811 if (extMIDs[i]->m_released == true) {
812 found ++;
813 if (found > 1) { //libvpx sometimes needs 2 buffer when calling decode once.
814 return true;
815 }
816 }
817 }
818 } else { // graphic buffer mode
819 for (i = 0; i < mOMXBufferHeaderTypePtrNum; i++) {
820 if ((extMIDs[i]->m_render_done == true) && (extMIDs[i]->m_released == true)) {
821 found ++;
822 if (found > 1) { //libvpx sometimes needs 2 buffer when calling decode once.
823 return true;
824 }
825 }
826 }
827 }
828
829 b = false;
830
831 return b;
832 }
833
834 DECLARE_OMX_COMPONENT("OMX.Intel.VideoDecoder.VP9.hwr", "video_decoder.vp9", OMXVideoDecoderVP9HWR);
835
836