• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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(&paramPortDefinitionInput,
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(&paramPortDefinitionInput,
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(&paramPortDefinitionInput,
486         this->ports[INPORT_INDEX]->GetPortDefinition(),
487         sizeof(paramPortDefinitionInput));
488 
489     memcpy(&paramPortDefinitionOutput,
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(&paramPortDefinitionInput, 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(&paramPortDefinitionInput, true);
550     this->ports[OUTPORT_INDEX]->SetPortDefinition(&paramPortDefinitionOutput, 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