| /* |
| * Copyright (c) 2009-2020, Intel Corporation |
| * |
| * Permission is hereby granted, free of charge, to any person obtaining a |
| * copy of this software and associated documentation files (the "Software"), |
| * to deal in the Software without restriction, including without limitation |
| * the rights to use, copy, modify, merge, publish, distribute, sublicense, |
| * and/or sell copies of the Software, and to permit persons to whom the |
| * Software is furnished to do so, subject to the following conditions: |
| * |
| * The above copyright notice and this permission notice shall be included |
| * in all copies or substantial portions of the Software. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS |
| * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
| * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR |
| * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, |
| * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR |
| * OTHER DEALINGS IN THE SOFTWARE. |
| */ |
| //! |
| //! \file media_libva_util.cpp |
| //! \brief libva(and its extension) utility |
| //! |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <pthread.h> |
| #include <sys/time.h> |
| #include <unistd.h> |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| #include <fcntl.h> |
| #include <dlfcn.h> |
| #include <errno.h> |
| |
| #include "media_libva_util.h" |
| #include "mos_utilities.h" |
| #include "mos_os.h" |
| #include "mos_defs.h" |
| #include "hwinfo_linux.h" |
| #include "media_ddi_decode_base.h" |
| #include "media_ddi_encode_base.h" |
| #include "media_libva_decoder.h" |
| #include "media_libva_encoder.h" |
| #include "media_libva_caps.h" |
| #include "memory_policy_manager.h" |
| #include "drm_fourcc.h" |
| |
| // default protected surface tag |
| #define PROTECTED_SURFACE_TAG 0x3000f |
| |
| #ifdef DEBUG |
| static int32_t frameCountFps = -1; |
| static struct timeval tv1; |
| static pthread_mutex_t fpsMutex = PTHREAD_MUTEX_INITIALIZER; |
| static int32_t vaFpsSampleSize = 100; |
| |
| #define LENGTH_OF_FPS_FILE_NAME 128 |
| |
| #ifdef ANDROID |
| #define FPS_FILE_NAME "/mnt/sdcard/fps.txt" |
| #else |
| #define FPS_FILE_NAME "./fps.txt" |
| #endif |
| #endif |
| #ifdef DEBUG |
| void DdiMediaUtil_MediaPrintFps() |
| { |
| struct timeval tv2; |
| |
| if (0 == vaFpsSampleSize) |
| { |
| return; |
| } |
| gettimeofday(&tv2, 0); |
| |
| pthread_mutex_lock(&fpsMutex); |
| if (-1 == frameCountFps) |
| { |
| gettimeofday(&tv1, 0); |
| } |
| |
| if (++frameCountFps >= vaFpsSampleSize) |
| { |
| char fpsFileName[LENGTH_OF_FPS_FILE_NAME]; |
| FILE *fp = nullptr; |
| char temp[LENGTH_OF_FPS_FILE_NAME]; |
| |
| int64_t diff = (tv2.tv_sec - tv1.tv_sec)*1000000 + tv2.tv_usec - tv1.tv_usec; |
| float fps = frameCountFps / (diff / 1000000.0); |
| DDI_NORMALMESSAGE("FPS:%6.4f, Interval:%11lu.", fps,((uint64_t)tv2.tv_sec)*1000 + (tv2.tv_usec/1000)); |
| sprintf(temp,"FPS:%6.4f, Interval:%11lu\n", fps,((uint64_t)tv2.tv_sec)*1000 + (tv2.tv_usec/1000)); |
| |
| MOS_ZeroMemory(fpsFileName,LENGTH_OF_FPS_FILE_NAME); |
| sprintf(fpsFileName, FPS_FILE_NAME); |
| if ((fp = fopen(fpsFileName, "wb")) == nullptr) |
| { |
| pthread_mutex_unlock(&fpsMutex); |
| DDI_ASSERTMESSAGE("Unable to open fps file."); |
| } |
| |
| fwrite(temp, 1, strlen(temp), fp); |
| fclose(fp); |
| frameCountFps = -1; |
| } |
| pthread_mutex_unlock(&fpsMutex); |
| } |
| #else |
| void DdiMediaUtil_MediaPrintFps() |
| { |
| return; |
| } |
| #endif |
| |
| /* |
| * DdiMediaUtil_IsExternalSurface |
| * Descripion: if the bo of media surface was allocated from App, |
| * should return true, otherwise, false. In current implemeation |
| * external buffer passed with pSurfDesc. |
| */ |
| bool DdiMediaUtil_IsExternalSurface(PDDI_MEDIA_SURFACE surface) |
| { |
| if ( nullptr == surface ) |
| { |
| return false; |
| } |
| else if ( surface->pSurfDesc == nullptr ) |
| { |
| return false; |
| } |
| |
| return true; |
| } |
| |
| //! |
| //! \brief Allocate surface |
| //! |
| //! \param [in] format |
| //! Ddi media format |
| //! \param [in] width |
| //! Width of the region |
| //! \param [in] height |
| //! Height of the region |
| //! \param [out] mediaSurface |
| //! Pointer to ddi media surface |
| //! \param [in] mediaDrvCtx |
| //! Pointer to ddi media context |
| //! |
| //! \return VAStatus |
| //! VA_STATUS_SUCCESS if success, else fail reason |
| //! |
| VAStatus DdiMediaUtil_AllocateSurface( |
| DDI_MEDIA_FORMAT format, |
| int32_t width, |
| int32_t height, |
| PDDI_MEDIA_SURFACE mediaSurface, |
| PDDI_MEDIA_CONTEXT mediaDrvCtx) |
| { |
| uint32_t pitch = 0; |
| MOS_LINUX_BO *bo = nullptr; |
| GMM_RESCREATE_PARAMS gmmParams; |
| GMM_RESOURCE_INFO *gmmResourceInfo = nullptr; |
| |
| DDI_CHK_NULL(mediaSurface, "mediaSurface is nullptr", VA_STATUS_ERROR_INVALID_BUFFER); |
| DDI_CHK_NULL(mediaDrvCtx, "mediaDrvCtx is nullptr", VA_STATUS_ERROR_INVALID_BUFFER); |
| DDI_CHK_NULL(mediaDrvCtx->pGmmClientContext, "mediaDrvCtx->pGmmClientContext is nullptr", VA_STATUS_ERROR_INVALID_BUFFER); |
| |
| int32_t size = 0; |
| uint32_t tileformat = I915_TILING_NONE; |
| VAStatus hRes = VA_STATUS_SUCCESS; |
| int32_t alignedWidth = width; |
| int32_t alignedHeight = height; |
| uint32_t tag = 0; |
| int mem_type = MOS_MEMPOOL_VIDEOMEMORY; |
| bool bMemCompEnable = true; |
| bool bMemCompRC = false; |
| |
| switch (format) |
| { |
| case Media_Format_X8R8G8B8: |
| case Media_Format_X8B8G8R8: |
| case Media_Format_A8B8G8R8: |
| case Media_Format_R8G8B8A8: |
| case Media_Format_R5G6B5: |
| case Media_Format_R8G8B8: |
| case Media_Format_R10G10B10A2: |
| case Media_Format_B10G10R10A2: |
| case Media_Format_R10G10B10X2: |
| case Media_Format_B10G10R10X2: |
| case Media_Format_A16R16G16B16: |
| case Media_Format_A16B16G16R16: |
| if (VA_SURFACE_ATTRIB_USAGE_HINT_ENCODER != mediaSurface->surfaceUsageHint && |
| !(mediaSurface->surfaceUsageHint & VA_SURFACE_ATTRIB_USAGE_HINT_VPP_WRITE)) |
| { |
| tileformat = I915_TILING_NONE; |
| break; |
| } |
| case Media_Format_YV12: |
| case Media_Format_I420: |
| case Media_Format_IYUV: |
| if (VA_SURFACE_ATTRIB_USAGE_HINT_ENCODER != mediaSurface->surfaceUsageHint && |
| !(mediaSurface->surfaceUsageHint & VA_SURFACE_ATTRIB_USAGE_HINT_VPP_WRITE)) |
| { |
| tileformat = I915_TILING_NONE; |
| break; |
| } |
| case Media_Format_RGBP: |
| case Media_Format_BGRP: |
| case Media_Format_A8R8G8B8: |
| if (VA_SURFACE_ATTRIB_USAGE_HINT_ENCODER != mediaSurface->surfaceUsageHint && |
| !(mediaSurface->surfaceUsageHint & VA_SURFACE_ATTRIB_USAGE_HINT_DECODER) && |
| !(mediaSurface->surfaceUsageHint & VA_SURFACE_ATTRIB_USAGE_HINT_VPP_WRITE)) |
| { |
| tileformat = I915_TILING_NONE; |
| break; |
| } |
| case Media_Format_NV12: |
| case Media_Format_NV21: |
| case Media_Format_444P: |
| case Media_Format_422H: |
| case Media_Format_411P: |
| case Media_Format_422V: |
| case Media_Format_IMC3: |
| case Media_Format_400P: |
| case Media_Format_P010: |
| case Media_Format_P012: |
| case Media_Format_P016: |
| case Media_Format_YUY2: |
| case Media_Format_Y210: |
| case Media_Format_Y216: |
| case Media_Format_AYUV: |
| case Media_Format_Y410: |
| case Media_Format_Y416: |
| case Media_Format_Y8: |
| case Media_Format_Y16S: |
| case Media_Format_Y16U: |
| case Media_Format_VYUY: |
| case Media_Format_YVYU: |
| case Media_Format_UYVY: |
| if (VA_SURFACE_ATTRIB_USAGE_HINT_ENCODER != mediaSurface->surfaceUsageHint && |
| !(mediaSurface->surfaceUsageHint & VA_SURFACE_ATTRIB_USAGE_HINT_VPP_WRITE)) |
| { |
| #if UFO_GRALLOC_NEW_FORMAT |
| //Planar type surface align 64 to improve performance. |
| alignedHeight = MOS_ALIGN_CEIL(height, 64); |
| #else |
| //Planar type surface align 32 to improve performance. |
| alignedHeight = MOS_ALIGN_CEIL(height, 32); |
| #endif |
| } |
| alignedWidth = MOS_ALIGN_CEIL(width, 8); |
| if (mediaSurface->surfaceUsageHint & VA_SURFACE_ATTRIB_USAGE_HINT_VPP_WRITE) |
| { |
| if ((format == Media_Format_YV12) || |
| (format == Media_Format_I420)) |
| { |
| alignedWidth = MOS_ALIGN_CEIL(width, 128); |
| } |
| |
| if ((format == Media_Format_NV12) || |
| (format == Media_Format_P010) || |
| (format == Media_Format_RGBP) || |
| (format == Media_Format_BGRP)) |
| { |
| #if UFO_GRALLOC_NEW_FORMAT |
| //Planar type surface align 64 to improve performance. |
| alignedHeight = MOS_ALIGN_CEIL(height, 64); |
| #else |
| //Planar type surface align 32 to improve performance. |
| alignedHeight = MOS_ALIGN_CEIL(height, 32); |
| #endif |
| } |
| } |
| tileformat = I915_TILING_Y; |
| break; |
| case Media_Format_Buffer: |
| tileformat = I915_TILING_NONE; |
| break; |
| default: |
| DDI_ASSERTMESSAGE("Unsupported format"); |
| hRes = VA_STATUS_ERROR_UNSUPPORTED_RT_FORMAT; |
| goto finish; |
| } |
| |
| if( DdiMediaUtil_IsExternalSurface(mediaSurface) ) |
| { |
| if (mediaSurface->pSurfDesc->uiFlags & VA_SURFACE_EXTBUF_DESC_PROTECTED) |
| { |
| tag = PROTECTED_SURFACE_TAG; |
| } |
| // DRM buffer allocated by Application, No need to re-allocate new DRM buffer |
| if((mediaSurface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_KERNEL_DRM) || |
| (mediaSurface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_DRM_PRIME)) |
| { |
| if (mediaSurface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_KERNEL_DRM) |
| { |
| bo = mos_bo_gem_create_from_name(mediaDrvCtx->pDrmBufMgr, "MEDIA", mediaSurface->pSurfDesc->ulBuffer); |
| } |
| else |
| { |
| bo = mos_bo_gem_create_from_prime(mediaDrvCtx->pDrmBufMgr, mediaSurface->pSurfDesc->ulBuffer, mediaSurface->pSurfDesc->uiSize); |
| } |
| |
| if( bo != nullptr ) |
| { |
| uint32_t swizzle_mode; |
| |
| pitch = mediaSurface->pSurfDesc->uiPitches[0]; |
| |
| //Overwirte the tileformat matches with the right buffer |
| mos_bo_get_tiling(bo, &tileformat, &swizzle_mode); |
| } |
| else |
| { |
| DDI_ASSERTMESSAGE("Failed to create drm buffer object according to input buffer descriptor."); |
| return VA_STATUS_ERROR_ALLOCATION_FAILED; |
| } |
| } |
| else if (mediaSurface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_DRM_PRIME_2) |
| { |
| bo = mos_bo_gem_create_from_prime(mediaDrvCtx->pDrmBufMgr, mediaSurface->pSurfDesc->ulBuffer, mediaSurface->pSurfDesc->uiSize); |
| if( bo != nullptr ) |
| { |
| pitch = mediaSurface->pSurfDesc->uiPitches[0]; |
| switch (mediaSurface->pSurfDesc->modifier) |
| { |
| case DRM_FORMAT_MOD_LINEAR: |
| tileformat = I915_TILING_NONE; |
| bMemCompEnable = false; |
| break; |
| case I915_FORMAT_MOD_X_TILED: |
| tileformat = I915_TILING_X; |
| bMemCompEnable = false; |
| break; |
| case I915_FORMAT_MOD_Y_TILED: |
| case I915_FORMAT_MOD_Yf_TILED: |
| tileformat = I915_TILING_Y; |
| bMemCompEnable = false; |
| break; |
| case I915_FORMAT_MOD_Y_TILED_GEN12_RC_CCS: |
| tileformat = I915_TILING_Y; |
| bMemCompEnable = true; |
| bMemCompRC = true; |
| break; |
| case I915_FORMAT_MOD_Y_TILED_GEN12_MC_CCS: |
| tileformat = I915_TILING_Y; |
| bMemCompEnable = true; |
| bMemCompRC = false; |
| break; |
| default: |
| DDI_ASSERTMESSAGE("Unsupported modifier."); |
| hRes = VA_STATUS_ERROR_INVALID_PARAMETER; |
| goto finish; |
| } |
| } |
| else |
| { |
| DDI_ASSERTMESSAGE("Failed to create drm buffer object according to input buffer descriptor."); |
| return VA_STATUS_ERROR_ALLOCATION_FAILED; |
| } |
| } |
| else if( mediaSurface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_USER_PTR ) |
| { |
| |
| pitch = mediaSurface->pSurfDesc->uiPitches[0]; |
| |
| #ifdef DRM_IOCTL_I915_GEM_USERPTR |
| bo = mos_bo_alloc_userptr( mediaDrvCtx->pDrmBufMgr, |
| "SysSurface", |
| (void *)mediaSurface->pSurfDesc->ulBuffer, |
| mediaSurface->pSurfDesc->uiTile, |
| pitch, |
| mediaSurface->pSurfDesc->uiBuffserSize, |
| I915_USERPTR_UNSYNCHRONIZED |
| ); |
| #else |
| bo = mos_bo_alloc_vmap( mediaDrvCtx->pDrmBufMgr, |
| "SysSurface", |
| (void *)mediaSurface->pSurfDesc->ulBuffer, |
| mediaSurface->pSurfDesc->uiTile, |
| pitch, |
| mediaSurface->pSurfDesc->uiBuffserSize, |
| 0 |
| ); |
| #endif |
| if( bo != nullptr ) |
| { |
| uint32_t swizzle_mode; |
| |
| //Overwrite the tile format that matches the exteral buffer |
| mos_bo_get_tiling(bo, &tileformat, &swizzle_mode); |
| DDI_VERBOSEMESSAGE("Success to create drm buffer vmap."); |
| } |
| else |
| { |
| DDI_ASSERTMESSAGE("Failed to create drm buffer vmap."); |
| return VA_STATUS_ERROR_ALLOCATION_FAILED; |
| } |
| } |
| else if( mediaSurface->pSurfDesc->uiFlags & VA_SURFACE_EXTBUF_DESC_ENABLE_TILING ) |
| { |
| tileformat = I915_TILING_Y; |
| } |
| else if (mediaSurface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_VA) |
| { |
| tileformat = I915_TILING_NONE; |
| alignedHeight = height; |
| } else { |
| DDI_ASSERTMESSAGE("Input buffer descriptor (%d) is not supported by current driver.", mediaSurface->pSurfDesc->uiFlags); |
| return VA_STATUS_ERROR_ALLOCATION_FAILED; |
| } |
| } |
| |
| // Create GmmResourceInfo |
| MOS_ZeroMemory(&gmmParams, sizeof(gmmParams)); |
| if (DdiMediaUtil_IsExternalSurface(mediaSurface) && |
| mediaSurface->pSurfDesc->uiVaMemType != VA_SURFACE_ATTRIB_MEM_TYPE_VA) |
| { |
| gmmParams.BaseWidth = mediaSurface->iWidth; |
| gmmParams.BaseHeight = mediaSurface->iHeight; |
| if(mediaSurface->pSurfDesc->uiPlanes > 0 && mediaSurface->pSurfDesc->uiPitches[0] > 0) |
| { |
| if((mediaSurface->pSurfDesc->uiPlanes > 1) && (mediaSurface->pSurfDesc->uiOffsets[1] > mediaSurface->pSurfDesc->uiOffsets[0])) |
| { |
| gmmParams.BaseHeight = (mediaSurface->pSurfDesc->uiOffsets[1] - mediaSurface->pSurfDesc->uiOffsets[0]) / mediaSurface->pSurfDesc->uiPitches[0]; |
| } |
| else if(mediaSurface->pSurfDesc->uiSize > 0) |
| { |
| gmmParams.BaseHeight = mediaSurface->pSurfDesc->uiSize / mediaSurface->pSurfDesc->uiPitches[0]; |
| } |
| } |
| if(mediaSurface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_USER_PTR) |
| { |
| gmmParams.ExistingSysMemSize = mediaSurface->pSurfDesc->uiBuffserSize; |
| gmmParams.pExistingSysMem = mediaSurface->pSurfDesc->ulBuffer; |
| gmmParams.Flags.Info.ExistingSysMem = true; |
| } |
| } |
| else |
| { |
| gmmParams.BaseWidth = alignedWidth; |
| gmmParams.BaseHeight = alignedHeight; |
| } |
| |
| gmmParams.ArraySize = 1; |
| gmmParams.Type = RESOURCE_2D; |
| gmmParams.Format = mediaDrvCtx->m_caps->ConvertMediaFmtToGmmFmt(format); |
| gmmParams.CpTag = tag; |
| |
| DDI_CHK_CONDITION(gmmParams.Format == GMM_FORMAT_INVALID, |
| "Unsupported format", |
| VA_STATUS_ERROR_UNSUPPORTED_RT_FORMAT); |
| switch (tileformat) |
| { |
| case I915_TILING_Y: |
| // Disable MMC for application required surfaces, because some cases' output streams have corruption. |
| gmmParams.Flags.Gpu.MMC = false; |
| if (MEDIA_IS_SKU(&mediaDrvCtx->SkuTable, FtrE2ECompression) && |
| (!MEDIA_IS_WA(&mediaDrvCtx->WaTable, WaDisableVPMmc) && |
| !MEDIA_IS_WA(&mediaDrvCtx->WaTable, WaDisableCodecMmc)) && |
| bMemCompEnable) |
| { |
| gmmParams.Flags.Gpu.MMC = true; |
| gmmParams.Flags.Info.MediaCompressed = 1; |
| gmmParams.Flags.Info.RenderCompressed = 0; |
| gmmParams.Flags.Gpu.CCS = 1; |
| gmmParams.Flags.Gpu.RenderTarget = 1; |
| gmmParams.Flags.Gpu.UnifiedAuxSurface = 1; |
| |
| if (bMemCompRC) |
| { |
| gmmParams.Flags.Info.MediaCompressed = 0; |
| gmmParams.Flags.Info.RenderCompressed = 1; |
| } |
| |
| if(MEDIA_IS_SKU(&mediaDrvCtx->SkuTable, FtrFlatPhysCCS)) |
| { |
| gmmParams.Flags.Gpu.UnifiedAuxSurface = 0; |
| } |
| } |
| break; |
| case I915_TILING_X: |
| gmmParams.Flags.Info.TiledX = true; |
| break; |
| default: |
| gmmParams.Flags.Info.Linear = true; |
| } |
| |
| gmmParams.Flags.Gpu.Video = true; |
| gmmParams.Flags.Info.LocalOnly = MEDIA_IS_SKU(&mediaDrvCtx->SkuTable, FtrLocalMemory); |
| |
| mediaSurface->pGmmResourceInfo = gmmResourceInfo = mediaDrvCtx->pGmmClientContext->CreateResInfoObject(&gmmParams); |
| |
| if(nullptr == gmmResourceInfo) |
| { |
| DDI_ASSERTMESSAGE("Gmm Create Resource Failed."); |
| hRes = VA_STATUS_ERROR_ALLOCATION_FAILED; |
| goto finish; |
| } |
| uint32_t gmmPitch; |
| uint32_t gmmSize; |
| uint32_t gmmHeight; |
| |
| gmmPitch = (uint32_t)gmmResourceInfo->GetRenderPitch(); |
| if( DdiMediaUtil_IsExternalSurface(mediaSurface) && mediaSurface->pSurfDesc->uiPitches[0]) |
| { |
| gmmResourceInfo->OverridePitch(mediaSurface->pSurfDesc->uiPitches[0]); |
| } |
| gmmPitch = (uint32_t)gmmResourceInfo->GetRenderPitch(); |
| gmmSize = (uint32_t)gmmResourceInfo->GetSizeSurface(); |
| gmmHeight = gmmResourceInfo->GetBaseHeight(); |
| |
| if ( 0 == gmmPitch || 0 == gmmSize || 0 == gmmHeight) |
| { |
| DDI_ASSERTMESSAGE("Gmm Create Resource Failed."); |
| hRes = VA_STATUS_ERROR_ALLOCATION_FAILED; |
| goto finish; |
| } |
| |
| mem_type = MemoryPolicyManager::UpdateMemoryPolicy(&mediaDrvCtx->SkuTable, mediaSurface->pGmmResourceInfo); |
| |
| if (!DdiMediaUtil_IsExternalSurface(mediaSurface) || |
| mediaSurface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_VA) |
| { |
| switch (gmmResourceInfo->GetTileType()) |
| { |
| case GMM_TILED_Y: |
| tileformat = I915_TILING_Y; |
| break; |
| case GMM_TILED_X: |
| tileformat = I915_TILING_X; |
| break; |
| case GMM_NOT_TILED: |
| tileformat = I915_TILING_NONE; |
| break; |
| default: |
| tileformat = I915_TILING_Y; |
| break; |
| } |
| |
| unsigned long ulPitch = 0; |
| if ( tileformat == I915_TILING_NONE ) |
| { |
| bo = mos_bo_alloc(mediaDrvCtx->pDrmBufMgr, "MEDIA", gmmSize, 4096, mem_type); |
| pitch = gmmPitch; |
| } |
| else |
| { |
| bo = mos_bo_alloc_tiled(mediaDrvCtx->pDrmBufMgr, "MEDIA", gmmPitch, (MOS_ALIGN_CEIL(gmmSize, gmmPitch))/gmmPitch, 1, &tileformat, (unsigned long *)&ulPitch, 0, mem_type); |
| pitch = ulPitch; |
| } |
| } |
| else if(mediaSurface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_USER_PTR) |
| { |
| gmmHeight = height; |
| } |
| else |
| { |
| // Check Pitch |
| if (gmmPitch > pitch) |
| { |
| DDI_ASSERTMESSAGE("External Surface doesn't meet the reqirements of Media driver."); |
| DdiMediaUtil_FreeSurface(mediaSurface); |
| hRes = VA_STATUS_ERROR_ALLOCATION_FAILED; |
| |
| goto finish; |
| } |
| |
| // Check Alignment |
| __GMM_BUFFER_TYPE gmmRestrictions = {0}; |
| gmmResourceInfo->GetRestrictions(gmmRestrictions); |
| unsigned long pitchAlign = gmmRestrictions.RenderPitchAlignment; |
| unsigned long sizeAlign = gmmRestrictions.Alignment; |
| |
| if (!MOS_IS_ALIGNED(pitch, pitchAlign)) |
| { |
| DDI_ASSERTMESSAGE("External Surface doesn't meet the reqirements of Media driver."); |
| DdiMediaUtil_FreeSurface(mediaSurface); |
| hRes = VA_STATUS_ERROR_ALLOCATION_FAILED; |
| |
| goto finish; |
| } |
| } |
| |
| mediaSurface->bMapped = false; |
| if (bo) |
| { |
| mediaSurface->format = format; |
| mediaSurface->iWidth = width; |
| mediaSurface->iHeight = gmmHeight; |
| mediaSurface->iRealHeight = height; |
| mediaSurface->iPitch = pitch; |
| mediaSurface->iRefCount = 0; |
| mediaSurface->bo = bo; |
| mediaSurface->TileType = tileformat; |
| mediaSurface->isTiled = (tileformat != I915_TILING_NONE) ? 1 : 0; |
| mediaSurface->pData = (uint8_t*) bo->virt; |
| DDI_VERBOSEMESSAGE("Alloc %7d bytes (%d x %d resource).",gmmSize, width, height); |
| } |
| else |
| { |
| DDI_ASSERTMESSAGE("Fail to Alloc %7d bytes (%d x %d resource).",gmmSize, width, height); |
| hRes = VA_STATUS_ERROR_ALLOCATION_FAILED; |
| } |
| |
| finish: |
| return hRes; |
| } |
| |
| //! |
| //! \brief Allocate buffer |
| //! |
| //! \param [in] format |
| //! Ddi media format |
| //! \param [in] size |
| //! Size of the region |
| //! \param [out] mediaBuffer |
| //! Pointer to ddi media buffer |
| //! \param [in] bufmgr |
| //! Mos buffer manager |
| //! |
| //! \return VAStatus |
| //! VA_STATUS_SUCCESS if success, else fail reason |
| //! |
| VAStatus DdiMediaUtil_AllocateBuffer( |
| DDI_MEDIA_FORMAT format, |
| int32_t size, |
| PDDI_MEDIA_BUFFER mediaBuffer, |
| MOS_BUFMGR *bufmgr) |
| { |
| |
| DDI_CHK_NULL(mediaBuffer, "mediaBuffer is nullptr", VA_STATUS_ERROR_INVALID_BUFFER); |
| DDI_CHK_NULL(mediaBuffer->pMediaCtx, "mediaBuffer->pMediaCtx is nullptr", VA_STATUS_ERROR_INVALID_BUFFER); |
| DDI_CHK_NULL(mediaBuffer->pMediaCtx->pGmmClientContext, "mediaBuffer->pMediaCtx->pGmmClientContext is nullptr", VA_STATUS_ERROR_INVALID_BUFFER); |
| if(format >= Media_Format_Count) |
| return VA_STATUS_ERROR_INVALID_PARAMETER; |
| |
| VAStatus hRes = VA_STATUS_SUCCESS; |
| int32_t mem_type = MOS_MEMPOOL_VIDEOMEMORY; |
| |
| // create fake GmmResourceInfo |
| GMM_RESCREATE_PARAMS gmmParams; |
| MOS_ZeroMemory(&gmmParams, sizeof(gmmParams)); |
| gmmParams.BaseWidth = 1; |
| gmmParams.BaseHeight = 1; |
| gmmParams.ArraySize = 0; |
| gmmParams.Type = RESOURCE_1D; |
| gmmParams.Format = GMM_FORMAT_GENERIC_8BIT; |
| gmmParams.Flags.Gpu.Video = true; |
| gmmParams.Flags.Info.Linear = true; |
| DDI_CHK_NULL(mediaBuffer->pMediaCtx, "MediaCtx is null", VA_STATUS_ERROR_INVALID_BUFFER); |
| gmmParams.Flags.Info.LocalOnly = MEDIA_IS_SKU(&mediaBuffer->pMediaCtx->SkuTable, FtrLocalMemory); |
| |
| mediaBuffer->pGmmResourceInfo = mediaBuffer->pMediaCtx->pGmmClientContext->CreateResInfoObject(&gmmParams); |
| |
| DDI_CHK_NULL(mediaBuffer->pGmmResourceInfo, "pGmmResourceInfo is nullptr", VA_STATUS_ERROR_INVALID_BUFFER); |
| mediaBuffer->pGmmResourceInfo->OverrideSize(mediaBuffer->iSize); |
| mediaBuffer->pGmmResourceInfo->OverrideBaseWidth(mediaBuffer->iSize); |
| mediaBuffer->pGmmResourceInfo->OverridePitch(mediaBuffer->iSize); |
| |
| mem_type = MemoryPolicyManager::UpdateMemoryPolicy(&mediaBuffer->pMediaCtx->SkuTable, mediaBuffer->pGmmResourceInfo); |
| MOS_LINUX_BO *bo = mos_bo_alloc(bufmgr, "Media Buffer", size, 4096, mem_type); |
| |
| mediaBuffer->bMapped = false; |
| if (bo) |
| { |
| mediaBuffer->format = format; |
| mediaBuffer->iSize = size; |
| mediaBuffer->iRefCount = 0; |
| mediaBuffer->bo = bo; |
| mediaBuffer->pData = (uint8_t*) bo->virt; |
| |
| DDI_VERBOSEMESSAGE("Alloc %8d bytes resource.",size); |
| } |
| else |
| { |
| DDI_ASSERTMESSAGE("Fail to Alloc %8d bytes resource.",size); |
| hRes = VA_STATUS_ERROR_ALLOCATION_FAILED; |
| goto finish; |
| } |
| |
| finish: |
| return hRes; |
| } |
| |
| //! |
| //! \brief Allocate 2D buffer |
| //! |
| //! \param [in] height |
| //! Height of the region |
| //! \param [in] width |
| //! Width of the region |
| //! \param [out] mediaBuffer |
| //! Pointer to ddi media buffer |
| //! \param [in] bufmgr |
| //! Mos buffer manager |
| //! |
| //! \return VAStatus |
| //! VA_STATUS_SUCCESS if success, else fail reason |
| //! |
| VAStatus DdiMediaUtil_Allocate2DBuffer( |
| uint32_t height, |
| uint32_t width, |
| PDDI_MEDIA_BUFFER mediaBuffer, |
| MOS_BUFMGR *bufmgr) |
| { |
| DDI_CHK_NULL(mediaBuffer, "mediaBuffer is nullptr", VA_STATUS_ERROR_INVALID_BUFFER); |
| DDI_CHK_NULL(mediaBuffer->pMediaCtx, "mediaBuffer->pMediaCtx is nullptr", VA_STATUS_ERROR_INVALID_BUFFER); |
| DDI_CHK_NULL(mediaBuffer->pMediaCtx->pGmmClientContext, "mediaBuffer->pMediaCtx->pGmmClientContext is nullptr", VA_STATUS_ERROR_INVALID_BUFFER); |
| |
| int32_t size = 0; |
| uint32_t tileformat = I915_TILING_NONE; |
| VAStatus hRes = VA_STATUS_SUCCESS; |
| int32_t mem_type = MOS_MEMPOOL_VIDEOMEMORY; |
| |
| // Create GmmResourceInfo |
| GMM_RESCREATE_PARAMS gmmParams; |
| MOS_ZeroMemory(&gmmParams, sizeof(gmmParams)); |
| gmmParams.BaseWidth = width; |
| gmmParams.BaseHeight = height; |
| gmmParams.ArraySize = 1; |
| gmmParams.Type = RESOURCE_2D; |
| gmmParams.Format = GMM_FORMAT_GENERIC_8BIT; |
| |
| DDI_CHK_CONDITION(gmmParams.Format == GMM_FORMAT_INVALID, |
| "Unsupported format", |
| VA_STATUS_ERROR_UNSUPPORTED_RT_FORMAT); |
| |
| gmmParams.Flags.Info.Linear = true; |
| gmmParams.Flags.Gpu.Video = true; |
| DDI_CHK_NULL(mediaBuffer->pMediaCtx, "MediaCtx is null", VA_STATUS_ERROR_INVALID_BUFFER); |
| gmmParams.Flags.Info.LocalOnly = MEDIA_IS_SKU(&mediaBuffer->pMediaCtx->SkuTable, FtrLocalMemory); |
| GMM_RESOURCE_INFO *gmmResourceInfo; |
| mediaBuffer->pGmmResourceInfo = gmmResourceInfo = mediaBuffer->pMediaCtx->pGmmClientContext->CreateResInfoObject(&gmmParams); |
| |
| if(nullptr == gmmResourceInfo) |
| { |
| DDI_VERBOSEMESSAGE("Gmm Create Resource Failed."); |
| hRes = VA_STATUS_ERROR_ALLOCATION_FAILED; |
| goto finish; |
| } |
| uint32_t gmmPitch; |
| uint32_t gmmSize; |
| uint32_t gmmHeight; |
| gmmPitch = (uint32_t)gmmResourceInfo->GetRenderPitch(); |
| gmmSize = (uint32_t)gmmResourceInfo->GetSizeSurface(); |
| gmmHeight = gmmResourceInfo->GetBaseHeight(); |
| |
| mem_type = MemoryPolicyManager::UpdateMemoryPolicy(&mediaBuffer->pMediaCtx->SkuTable, mediaBuffer->pGmmResourceInfo); |
| |
| MOS_LINUX_BO *bo; |
| bo = mos_bo_alloc(bufmgr, "Media 2D Buffer", gmmSize, 4096, mem_type); |
| |
| mediaBuffer->bMapped = false; |
| if (bo) |
| { |
| mediaBuffer->format = Media_Format_2DBuffer; |
| mediaBuffer->uiWidth = width; |
| mediaBuffer->uiHeight = gmmHeight; |
| mediaBuffer->uiPitch = gmmPitch; |
| mediaBuffer->iSize = gmmSize; |
| mediaBuffer->iRefCount = 0; |
| mediaBuffer->bo = bo; |
| mediaBuffer->TileType = tileformat; |
| mediaBuffer->pData = (uint8_t*) bo->virt; |
| DDI_VERBOSEMESSAGE("Alloc %7d bytes (%d x %d resource)\n",size, width, height); |
| } |
| else |
| { |
| DDI_VERBOSEMESSAGE("Fail to Alloc %7d bytes (%d x %d resource)\n", size, width, height); |
| hRes = VA_STATUS_ERROR_ALLOCATION_FAILED; |
| } |
| |
| finish: |
| return hRes; |
| } |
| |
| VAStatus DdiMediaUtil_CreateSurface(DDI_MEDIA_SURFACE *surface, PDDI_MEDIA_CONTEXT mediaDrvCtx) |
| { |
| VAStatus hr = VA_STATUS_SUCCESS; |
| |
| DDI_CHK_NULL(surface, "nullptr surface", VA_STATUS_ERROR_INVALID_BUFFER); |
| |
| // better to differentiate 1D and 2D type |
| hr = DdiMediaUtil_AllocateSurface(surface->format, |
| surface->iWidth, |
| surface->iHeight, |
| surface, |
| mediaDrvCtx); |
| if (VA_STATUS_SUCCESS == hr && nullptr != surface->bo) |
| surface->base = surface->name; |
| |
| return hr; |
| } |
| |
| VAStatus DdiMediaUtil_CreateBuffer(DDI_MEDIA_BUFFER *buffer, MOS_BUFMGR *bufmgr) |
| { |
| VAStatus hr = VA_STATUS_SUCCESS; |
| |
| DDI_CHK_NULL(buffer, "nullptr buffer", VA_STATUS_ERROR_INVALID_BUFFER); |
| |
| DDI_CHK_LESS(buffer->format, Media_Format_Count, "Invalid buffer->format", VA_STATUS_ERROR_INVALID_PARAMETER); |
| |
| if (buffer->format == Media_Format_CPU) |
| { |
| buffer->pData= (uint8_t*)MOS_AllocAndZeroMemory(buffer->iSize); |
| if (nullptr == buffer->pData) |
| hr = VA_STATUS_ERROR_ALLOCATION_FAILED; |
| } |
| |
| else |
| { |
| if (Media_Format_2DBuffer == buffer->format) |
| { |
| hr = DdiMediaUtil_Allocate2DBuffer(buffer->uiHeight, |
| buffer->uiWidth, |
| buffer, |
| bufmgr); |
| } |
| else |
| { |
| hr = DdiMediaUtil_AllocateBuffer(buffer->format, |
| buffer->iSize, |
| buffer, |
| bufmgr); |
| } |
| } |
| |
| buffer->uiLockedBufID = VA_INVALID_ID; |
| buffer->uiLockedImageID = VA_INVALID_ID; |
| buffer->iRefCount = 0; |
| |
| return hr; |
| } |
| |
| VAStatus SwizzleSurface(PDDI_MEDIA_CONTEXT mediaCtx, PGMM_RESOURCE_INFO pGmmResInfo, void *pLockedAddr, uint32_t TileType, uint8_t* pResourceBase, bool bUpload); |
| |
| // add thread protection for multiple thread? |
| void* DdiMediaUtil_LockSurface(DDI_MEDIA_SURFACE *surface, uint32_t flag) |
| { |
| DDI_CHK_NULL(surface, "nullptr surface", nullptr); |
| DDI_CHK_NULL(surface->bo, "nullptr surface->bo", nullptr); |
| if((false == surface->bMapped) && (0 == surface->iRefCount)) |
| { |
| if (surface->pMediaCtx->bIsAtomSOC) |
| { |
| mos_gem_bo_map_gtt(surface->bo); |
| } |
| else |
| { |
| if (surface->TileType == I915_TILING_NONE) |
| { |
| mos_bo_map(surface->bo, flag & MOS_LOCKFLAG_WRITEONLY); |
| } |
| else if ((surface->pMediaCtx->m_useSwSwizzling) && !(flag & MOS_LOCKFLAG_NO_SWIZZLE)) |
| { |
| mos_bo_map(surface->bo, flag & MOS_LOCKFLAG_WRITEONLY); |
| if (surface->pSystemShadow == nullptr) |
| { |
| surface->pSystemShadow = (uint8_t*)MOS_AllocMemory(surface->bo->size); |
| DDI_CHK_CONDITION((surface->pSystemShadow == nullptr), "Failed to allocate shadow surface", nullptr); |
| } |
| |
| uint64_t surfSize = surface->pGmmResourceInfo->GetSizeMainSurface(); |
| DDI_CHK_CONDITION((surface->TileType != I915_TILING_Y), "Unsupported tile type", nullptr); |
| DDI_CHK_CONDITION((surfSize <= 0 || surface->iPitch <= 0), "Invalid surface size or pitch", nullptr); |
| |
| VAStatus vaStatus = SwizzleSurface(surface->pMediaCtx, |
| surface->pGmmResourceInfo, |
| surface->bo->virt, |
| (MOS_TILE_TYPE)surface->TileType, |
| (uint8_t *)surface->pSystemShadow, |
| false); |
| DDI_CHK_CONDITION((vaStatus != VA_STATUS_SUCCESS), "SwizzleSurface failed", nullptr); |
| |
| } |
| else if (flag & MOS_LOCKFLAG_NO_SWIZZLE) |
| { |
| mos_bo_map(surface->bo, flag & MOS_LOCKFLAG_READONLY); |
| } |
| else if (flag & MOS_LOCKFLAG_WRITEONLY) |
| { |
| mos_gem_bo_map_gtt(surface->bo); |
| } |
| else |
| { |
| mos_gem_bo_map_unsynchronized(surface->bo); // only call mmap_gtt ioctl |
| mos_gem_bo_start_gtt_access(surface->bo, 0); // set to GTT domain,0 means readonly |
| } |
| } |
| surface->uiMapFlag = flag; |
| surface->pData = surface->pSystemShadow ? surface->pSystemShadow : (uint8_t*) surface->bo->virt; |
| surface->data_size = surface->bo->size; |
| surface->bMapped = true; |
| } |
| else |
| { |
| // do nothing here |
| } |
| surface->iRefCount++; |
| |
| return surface->pData; |
| } |
| |
| void DdiMediaUtil_UnlockSurface(DDI_MEDIA_SURFACE *surface) |
| { |
| DDI_CHK_NULL(surface, "nullptr surface", ); |
| DDI_CHK_NULL(surface->bo, "nullptr surface->bo", ); |
| if (0 == surface->iRefCount) |
| return; |
| |
| if((true == surface->bMapped) && (1 == surface->iRefCount)) |
| { |
| if (surface->pMediaCtx->bIsAtomSOC) |
| { |
| mos_gem_bo_unmap_gtt(surface->bo); |
| } |
| else |
| { |
| if (surface->TileType == I915_TILING_NONE) |
| { |
| mos_bo_unmap(surface->bo); |
| } |
| else if (surface->pSystemShadow) |
| { |
| SwizzleSurface(surface->pMediaCtx, |
| surface->pGmmResourceInfo, |
| surface->bo->virt, |
| (MOS_TILE_TYPE)surface->TileType, |
| (uint8_t *)surface->pSystemShadow, |
| true); |
| |
| MOS_FreeMemory(surface->pSystemShadow); |
| surface->pSystemShadow = nullptr; |
| |
| mos_bo_unmap(surface->bo); |
| } |
| else if(surface->uiMapFlag & MOS_LOCKFLAG_NO_SWIZZLE) |
| { |
| mos_bo_unmap(surface->bo); |
| } |
| else |
| { |
| mos_gem_bo_unmap_gtt(surface->bo); |
| } |
| } |
| surface->pData = nullptr; |
| surface->bo->virt = nullptr; |
| surface->bMapped = false; |
| } |
| else |
| { |
| // do nothing here |
| } |
| |
| surface->iRefCount--; |
| |
| return; |
| } |
| |
| // add thread protection for multiple thread? |
| // MapBuffer? |
| void* DdiMediaUtil_LockBuffer(DDI_MEDIA_BUFFER *buf, uint32_t flag) |
| { |
| DDI_CHK_NULL(buf, "nullptr buf", nullptr); |
| if((Media_Format_CPU != buf->format) && (false == buf->bMapped)) |
| { |
| if (nullptr != buf->pSurface) |
| { |
| DdiMediaUtil_LockSurface(buf->pSurface, flag); |
| buf->pData = buf->pSurface->pData; |
| } |
| else |
| { |
| if (buf->pMediaCtx->bIsAtomSOC) |
| { |
| mos_gem_bo_map_gtt(buf->bo); |
| } |
| else |
| { |
| if (buf->TileType == I915_TILING_NONE) |
| { |
| mos_bo_map(buf->bo, ((MOS_LOCKFLAG_READONLY | MOS_LOCKFLAG_WRITEONLY) & flag)); |
| } |
| else |
| { |
| mos_gem_bo_map_gtt(buf->bo); |
| } |
| } |
| |
| buf->pData = (uint8_t*)(buf->bo->virt); |
| } |
| |
| buf->bMapped = true; |
| buf->iRefCount++; |
| } |
| else if ((Media_Format_CPU == buf->format) && (false == buf->bMapped)) |
| { |
| buf->bMapped = true; |
| buf->iRefCount++; |
| } |
| else |
| { |
| buf->iRefCount++; |
| } |
| |
| return buf->pData; |
| } |
| |
| void DdiMediaUtil_UnlockBuffer(DDI_MEDIA_BUFFER *buf) |
| { |
| DDI_CHK_NULL(buf, "nullptr buf", ); |
| if (0 == buf->iRefCount) |
| return; |
| if((true == buf->bMapped) && (Media_Format_CPU != buf->format) && (1 == buf->iRefCount)) |
| { |
| if (nullptr != buf->pSurface) |
| { |
| DdiMediaUtil_UnlockSurface(buf->pSurface); |
| } |
| else |
| { |
| if (buf->pMediaCtx->bIsAtomSOC) |
| { |
| mos_gem_bo_unmap_gtt(buf->bo); |
| } |
| else |
| { |
| if (buf->TileType == I915_TILING_NONE) |
| { |
| mos_bo_unmap(buf->bo); |
| } |
| else |
| { |
| mos_gem_bo_unmap_gtt(buf->bo); |
| } |
| } |
| buf->bo->virt = nullptr; |
| } |
| |
| buf->pData = nullptr; |
| |
| buf->bMapped = false; |
| } |
| else if ((true == buf->bMapped) && (Media_Format_CPU == buf->format) && (1 == buf->iRefCount)) |
| { |
| buf->bMapped = false; |
| } |
| else |
| { |
| // do nothing here |
| } |
| buf->iRefCount--; |
| return; |
| } |
| |
| // should ref_count added for bo? |
| void DdiMediaUtil_FreeSurface(DDI_MEDIA_SURFACE *surface) |
| { |
| DDI_CHK_NULL(surface, "nullptr surface", ); |
| DDI_CHK_NULL(surface->bo, "nullptr surface->bo", ); |
| DDI_CHK_NULL(surface->pMediaCtx, "nullptr surface->pMediaCtx", ); |
| DDI_CHK_NULL(surface->pMediaCtx->pGmmClientContext, "nullptr surface->pMediaCtx->pGmmClientContext", ); |
| |
| // Unmap Aux mapping if the surface was mapped |
| if (surface->pMediaCtx->m_auxTableMgr) |
| { |
| surface->pMediaCtx->m_auxTableMgr->UnmapResource(surface->pGmmResourceInfo, surface->bo); |
| } |
| |
| // For External Buffer, only needs to destory SurfaceDescriptor |
| if ( DdiMediaUtil_IsExternalSurface(surface) ) |
| { |
| // In DdiMediaUtil_AllocateSurface call, driver will increase the surface reference count by calling drm_intel_bo_gem_create_from_name |
| // Thus, when freeing the surface, the drm_intel_bo_unreference function should be called to avoid memory leak |
| mos_bo_unreference(surface->bo); |
| MOS_FreeMemory(surface->pSurfDesc); |
| surface->pSurfDesc = nullptr; |
| } |
| else |
| { |
| // calling sequence checking |
| if (surface->bMapped) |
| { |
| DdiMediaUtil_UnlockSurface(surface); |
| DDI_VERBOSEMESSAGE("DDI: try to free a locked surface."); |
| } |
| mos_bo_unreference(surface->bo); |
| surface->bo = nullptr; |
| } |
| |
| if (nullptr != surface->pGmmResourceInfo) |
| { |
| surface->pMediaCtx->pGmmClientContext->DestroyResInfoObject(surface->pGmmResourceInfo); |
| surface->pGmmResourceInfo = nullptr; |
| } |
| } |
| |
| |
| // should ref_count added for bo? |
| void DdiMediaUtil_FreeBuffer(DDI_MEDIA_BUFFER *buf) |
| { |
| DDI_CHK_NULL(buf, "nullptr", ); |
| // calling sequence checking |
| if (buf->bMapped) |
| { |
| DdiMediaUtil_UnlockBuffer(buf); |
| DDI_VERBOSEMESSAGE("DDI: try to free a locked buffer."); |
| } |
| if (buf->format == Media_Format_CPU) |
| { |
| MOS_FreeMemory(buf->pData); |
| buf->pData = nullptr; |
| } |
| else |
| { |
| mos_bo_unreference(buf->bo); |
| buf->bo = nullptr; |
| } |
| |
| if (nullptr != buf->pMediaCtx && nullptr != buf->pMediaCtx->pGmmClientContext && nullptr != buf->pGmmResourceInfo) |
| { |
| buf->pMediaCtx->pGmmClientContext->DestroyResInfoObject(buf->pGmmResourceInfo); |
| buf->pGmmResourceInfo = nullptr; |
| } |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////////////////////////////// |
| // purpose: fill a rect structure with the regsion specified by parameters |
| // rect[in]: input pointer to the rect |
| // offset_x: x offset of the region |
| // offset_y: y offset of the region |
| // width: width of the region |
| // hiehgt: height of the regsion |
| //////////////////////////////////////////////////////////////////////////////////////////////////// |
| VAStatus DdiMediaUtil_FillPositionToRect(RECT *rect, int16_t offset_x, int16_t offset_y, int16_t width, int16_t height) |
| { |
| DDI_CHK_NULL(rect, "Invalid Rect.", VA_STATUS_ERROR_INVALID_PARAMETER); |
| |
| rect->left = offset_x; |
| rect->top = offset_y; |
| rect->right = offset_x + width; |
| rect->bottom = offset_y + height; |
| |
| return VA_STATUS_SUCCESS; |
| } |
| |
| void DdiMediaUtil_InitMutex(PMEDIA_MUTEX_T mutex) |
| { |
| pthread_mutex_init(mutex, nullptr); |
| } |
| |
| void DdiMediaUtil_DestroyMutex(PMEDIA_MUTEX_T mutex) |
| { |
| int32_t ret = pthread_mutex_destroy(mutex); |
| if(ret != 0) |
| { |
| DDI_NORMALMESSAGE("can't destroy the mutex!\n"); |
| } |
| } |
| |
| void DdiMediaUtil_LockMutex(PMEDIA_MUTEX_T mutex) |
| { |
| int32_t ret = pthread_mutex_lock(mutex); |
| if(ret != 0) |
| { |
| DDI_NORMALMESSAGE("can't lock the mutex!\n"); |
| } |
| } |
| |
| void DdiMediaUtil_UnLockMutex(PMEDIA_MUTEX_T mutex) |
| { |
| int32_t ret = pthread_mutex_unlock(mutex); |
| if(ret != 0) |
| { |
| DDI_NORMALMESSAGE("can't unlock the mutex!\n"); |
| } |
| } |
| |
| void DdiMediaUtil_DestroySemaphore(PMEDIA_SEM_T sem) |
| { |
| int32_t ret = sem_destroy(sem); |
| if(ret != 0) |
| { |
| DDI_NORMALMESSAGE("can't destroy the semaphore!\n"); |
| } |
| } |
| |
| void DdiMediaUtil_WaitSemaphore(PMEDIA_SEM_T sem) |
| { |
| int32_t ret = sem_wait(sem); |
| if(ret != 0) |
| { |
| DDI_NORMALMESSAGE("wait semaphore error!\n"); |
| } |
| } |
| |
| int32_t DdiMediaUtil_TryWaitSemaphore(PMEDIA_SEM_T sem) |
| { |
| return sem_trywait(sem); |
| } |
| |
| void DdiMediaUtil_PostSemaphore(PMEDIA_SEM_T sem) |
| { |
| int32_t ret = sem_post(sem); |
| if(ret != 0) |
| { |
| DDI_NORMALMESSAGE("post semaphore error!\n"); |
| } |
| } |
| |
| // heap related |
| PDDI_MEDIA_SURFACE_HEAP_ELEMENT DdiMediaUtil_AllocPMediaSurfaceFromHeap(PDDI_MEDIA_HEAP surfaceHeap) |
| { |
| DDI_CHK_NULL(surfaceHeap, "nullptr surfaceHeap", nullptr); |
| |
| PDDI_MEDIA_SURFACE_HEAP_ELEMENT mediaSurfaceHeapElmt = nullptr; |
| |
| if (nullptr == surfaceHeap->pFirstFreeHeapElement) |
| { |
| void *newHeapBase = MOS_ReallocMemory(surfaceHeap->pHeapBase, (surfaceHeap->uiAllocatedHeapElements + DDI_MEDIA_HEAP_INCREMENTAL_SIZE) * sizeof(DDI_MEDIA_SURFACE_HEAP_ELEMENT)); |
| |
| if (nullptr == newHeapBase) |
| { |
| DDI_ASSERTMESSAGE("DDI: realloc failed."); |
| return nullptr; |
| } |
| surfaceHeap->pHeapBase = newHeapBase; |
| PDDI_MEDIA_SURFACE_HEAP_ELEMENT surfaceHeapBase = (PDDI_MEDIA_SURFACE_HEAP_ELEMENT)surfaceHeap->pHeapBase; |
| surfaceHeap->pFirstFreeHeapElement = (void*)(&surfaceHeapBase[surfaceHeap->uiAllocatedHeapElements]); |
| for (int32_t i = 0; i < (DDI_MEDIA_HEAP_INCREMENTAL_SIZE); i++) |
| { |
| mediaSurfaceHeapElmt = &surfaceHeapBase[surfaceHeap->uiAllocatedHeapElements + i]; |
| mediaSurfaceHeapElmt->pNextFree = (i == (DDI_MEDIA_HEAP_INCREMENTAL_SIZE - 1))? nullptr : &surfaceHeapBase[surfaceHeap->uiAllocatedHeapElements + i + 1]; |
| mediaSurfaceHeapElmt->uiVaSurfaceID = surfaceHeap->uiAllocatedHeapElements + i; |
| } |
| surfaceHeap->uiAllocatedHeapElements += DDI_MEDIA_HEAP_INCREMENTAL_SIZE; |
| } |
| |
| mediaSurfaceHeapElmt = (PDDI_MEDIA_SURFACE_HEAP_ELEMENT)surfaceHeap->pFirstFreeHeapElement; |
| surfaceHeap->pFirstFreeHeapElement = mediaSurfaceHeapElmt->pNextFree; |
| |
| return mediaSurfaceHeapElmt; |
| } |
| |
| |
| void DdiMediaUtil_ReleasePMediaSurfaceFromHeap(PDDI_MEDIA_HEAP surfaceHeap, uint32_t vaSurfaceID) |
| { |
| DDI_CHK_NULL(surfaceHeap, "nullptr surfaceHeap", ); |
| |
| DDI_CHK_LESS(vaSurfaceID, surfaceHeap->uiAllocatedHeapElements, "invalid surface id", ); |
| PDDI_MEDIA_SURFACE_HEAP_ELEMENT mediaSurfaceHeapBase = (PDDI_MEDIA_SURFACE_HEAP_ELEMENT)surfaceHeap->pHeapBase; |
| DDI_CHK_NULL(mediaSurfaceHeapBase, "nullptr mediaSurfaceHeapBase", ); |
| |
| PDDI_MEDIA_SURFACE_HEAP_ELEMENT mediaSurfaceHeapElmt = &mediaSurfaceHeapBase[vaSurfaceID]; |
| DDI_CHK_NULL(mediaSurfaceHeapElmt->pSurface, "surface is already released", ); |
| void *firstFree = surfaceHeap->pFirstFreeHeapElement; |
| surfaceHeap->pFirstFreeHeapElement = (void*)mediaSurfaceHeapElmt; |
| mediaSurfaceHeapElmt->pNextFree = (PDDI_MEDIA_SURFACE_HEAP_ELEMENT)firstFree; |
| mediaSurfaceHeapElmt->pSurface = nullptr; |
| } |
| |
| |
| PDDI_MEDIA_BUFFER_HEAP_ELEMENT DdiMediaUtil_AllocPMediaBufferFromHeap(PDDI_MEDIA_HEAP bufferHeap) |
| { |
| DDI_CHK_NULL(bufferHeap, "nullptr bufferHeap", nullptr); |
| |
| PDDI_MEDIA_BUFFER_HEAP_ELEMENT mediaBufferHeapElmt = nullptr; |
| if (nullptr == bufferHeap->pFirstFreeHeapElement) |
| { |
| void *newHeapBase = MOS_ReallocMemory(bufferHeap->pHeapBase, (bufferHeap->uiAllocatedHeapElements + DDI_MEDIA_HEAP_INCREMENTAL_SIZE) * sizeof(DDI_MEDIA_BUFFER_HEAP_ELEMENT)); |
| if (nullptr == newHeapBase) |
| { |
| DDI_ASSERTMESSAGE("DDI: realloc failed."); |
| return nullptr; |
| } |
| bufferHeap->pHeapBase = newHeapBase; |
| PDDI_MEDIA_BUFFER_HEAP_ELEMENT mediaBufferHeapBase = (PDDI_MEDIA_BUFFER_HEAP_ELEMENT)bufferHeap->pHeapBase; |
| bufferHeap->pFirstFreeHeapElement = (void*)(&mediaBufferHeapBase[bufferHeap->uiAllocatedHeapElements]); |
| for (int32_t i = 0; i < (DDI_MEDIA_HEAP_INCREMENTAL_SIZE); i++) |
| { |
| mediaBufferHeapElmt = &mediaBufferHeapBase[bufferHeap->uiAllocatedHeapElements + i]; |
| mediaBufferHeapElmt->pNextFree = (i == (DDI_MEDIA_HEAP_INCREMENTAL_SIZE - 1))? nullptr : &mediaBufferHeapBase[bufferHeap->uiAllocatedHeapElements + i + 1]; |
| mediaBufferHeapElmt->uiVaBufferID = bufferHeap->uiAllocatedHeapElements + i; |
| } |
| bufferHeap->uiAllocatedHeapElements += DDI_MEDIA_HEAP_INCREMENTAL_SIZE; |
| } |
| |
| mediaBufferHeapElmt = (PDDI_MEDIA_BUFFER_HEAP_ELEMENT)bufferHeap->pFirstFreeHeapElement; |
| bufferHeap->pFirstFreeHeapElement = mediaBufferHeapElmt->pNextFree; |
| return mediaBufferHeapElmt; |
| } |
| |
| |
| void DdiMediaUtil_ReleasePMediaBufferFromHeap(PDDI_MEDIA_HEAP bufferHeap, uint32_t vaBufferID) |
| { |
| DDI_CHK_NULL(bufferHeap, "nullptr bufferHeap", ); |
| |
| DDI_CHK_LESS(vaBufferID, bufferHeap->uiAllocatedHeapElements, "invalid buffer id", ); |
| PDDI_MEDIA_BUFFER_HEAP_ELEMENT mediaBufferHeapBase = (PDDI_MEDIA_BUFFER_HEAP_ELEMENT)bufferHeap->pHeapBase; |
| PDDI_MEDIA_BUFFER_HEAP_ELEMENT mediaBufferHeapElmt = &mediaBufferHeapBase[vaBufferID]; |
| DDI_CHK_NULL(mediaBufferHeapElmt->pBuffer, "buffer is already released", ); |
| void *firstFree = bufferHeap->pFirstFreeHeapElement; |
| bufferHeap->pFirstFreeHeapElement = (void*)mediaBufferHeapElmt; |
| mediaBufferHeapElmt->pNextFree = (PDDI_MEDIA_BUFFER_HEAP_ELEMENT)firstFree; |
| mediaBufferHeapElmt->pBuffer = nullptr; |
| } |
| |
| PDDI_MEDIA_IMAGE_HEAP_ELEMENT DdiMediaUtil_AllocPVAImageFromHeap(PDDI_MEDIA_HEAP imageHeap) |
| { |
| PDDI_MEDIA_IMAGE_HEAP_ELEMENT vaimageHeapElmt = nullptr; |
| |
| DDI_CHK_NULL(imageHeap, "nullptr imageHeap", nullptr); |
| |
| if (nullptr == imageHeap->pFirstFreeHeapElement) |
| { |
| void *newHeapBase = MOS_ReallocMemory(imageHeap->pHeapBase, (imageHeap->uiAllocatedHeapElements + DDI_MEDIA_HEAP_INCREMENTAL_SIZE) * sizeof(DDI_MEDIA_IMAGE_HEAP_ELEMENT)); |
| |
| if (nullptr == newHeapBase) |
| { |
| DDI_ASSERTMESSAGE("DDI: realloc failed."); |
| return nullptr; |
| } |
| imageHeap->pHeapBase = newHeapBase; |
| PDDI_MEDIA_IMAGE_HEAP_ELEMENT vaimageHeapBase = (PDDI_MEDIA_IMAGE_HEAP_ELEMENT)imageHeap->pHeapBase; |
| imageHeap->pFirstFreeHeapElement = (void*)(&vaimageHeapBase[imageHeap->uiAllocatedHeapElements]); |
| for (int32_t i = 0; i < (DDI_MEDIA_HEAP_INCREMENTAL_SIZE); i++) |
| { |
| vaimageHeapElmt = &vaimageHeapBase[imageHeap->uiAllocatedHeapElements + i]; |
| vaimageHeapElmt->pNextFree = (i == (DDI_MEDIA_HEAP_INCREMENTAL_SIZE - 1))? nullptr : &vaimageHeapBase[imageHeap->uiAllocatedHeapElements + i + 1]; |
| vaimageHeapElmt->uiVaImageID = imageHeap->uiAllocatedHeapElements + i; |
| } |
| imageHeap->uiAllocatedHeapElements += DDI_MEDIA_HEAP_INCREMENTAL_SIZE; |
| |
| } |
| |
| vaimageHeapElmt = (PDDI_MEDIA_IMAGE_HEAP_ELEMENT)imageHeap->pFirstFreeHeapElement; |
| imageHeap->pFirstFreeHeapElement = vaimageHeapElmt->pNextFree; |
| return vaimageHeapElmt; |
| } |
| |
| |
| void DdiMediaUtil_ReleasePVAImageFromHeap(PDDI_MEDIA_HEAP imageHeap, uint32_t vaImageID) |
| { |
| PDDI_MEDIA_IMAGE_HEAP_ELEMENT vaImageHeapBase = nullptr; |
| PDDI_MEDIA_IMAGE_HEAP_ELEMENT vaImageHeapElmt = nullptr; |
| void *firstFree = nullptr; |
| |
| DDI_CHK_NULL(imageHeap, "nullptr imageHeap", ); |
| |
| DDI_CHK_LESS(vaImageID, imageHeap->uiAllocatedHeapElements, "invalid image id", ); |
| vaImageHeapBase = (PDDI_MEDIA_IMAGE_HEAP_ELEMENT)imageHeap->pHeapBase; |
| vaImageHeapElmt = &vaImageHeapBase[vaImageID]; |
| DDI_CHK_NULL(vaImageHeapElmt->pImage, "image is already released", ); |
| firstFree = imageHeap->pFirstFreeHeapElement; |
| imageHeap->pFirstFreeHeapElement = (void*)vaImageHeapElmt; |
| vaImageHeapElmt->pNextFree = (PDDI_MEDIA_IMAGE_HEAP_ELEMENT)firstFree; |
| vaImageHeapElmt->pImage = nullptr; |
| } |
| |
| PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT DdiMediaUtil_AllocPVAContextFromHeap(PDDI_MEDIA_HEAP vaContextHeap) |
| { |
| PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT vacontextHeapElmt = nullptr; |
| DDI_CHK_NULL(vaContextHeap, "nullptr vaContextHeap", nullptr); |
| |
| if (nullptr == vaContextHeap->pFirstFreeHeapElement) |
| { |
| void *newHeapBase = MOS_ReallocMemory(vaContextHeap->pHeapBase, (vaContextHeap->uiAllocatedHeapElements + DDI_MEDIA_HEAP_INCREMENTAL_SIZE) * sizeof(DDI_MEDIA_VACONTEXT_HEAP_ELEMENT)); |
| |
| if (nullptr == newHeapBase) |
| { |
| DDI_ASSERTMESSAGE("DDI: realloc failed."); |
| return nullptr; |
| } |
| vaContextHeap->pHeapBase = newHeapBase; |
| PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT vacontextHeapBase = (PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT)vaContextHeap->pHeapBase; |
| vaContextHeap->pFirstFreeHeapElement = (void*)(&(vacontextHeapBase[vaContextHeap->uiAllocatedHeapElements])); |
| for (int32_t i = 0; i < (DDI_MEDIA_HEAP_INCREMENTAL_SIZE); i++) |
| { |
| vacontextHeapElmt = &vacontextHeapBase[vaContextHeap->uiAllocatedHeapElements + i]; |
| vacontextHeapElmt->pNextFree = (i == (DDI_MEDIA_HEAP_INCREMENTAL_SIZE - 1))? nullptr : &vacontextHeapBase[vaContextHeap->uiAllocatedHeapElements + i + 1]; |
| vacontextHeapElmt->uiVaContextID = vaContextHeap->uiAllocatedHeapElements + i; |
| vacontextHeapElmt->pVaContext = nullptr; |
| } |
| vaContextHeap->uiAllocatedHeapElements += DDI_MEDIA_HEAP_INCREMENTAL_SIZE; |
| } |
| |
| vacontextHeapElmt = (PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT)vaContextHeap->pFirstFreeHeapElement; |
| vaContextHeap->pFirstFreeHeapElement = vacontextHeapElmt->pNextFree; |
| return vacontextHeapElmt; |
| } |
| |
| |
| void DdiMediaUtil_ReleasePVAContextFromHeap(PDDI_MEDIA_HEAP vaContextHeap, uint32_t vaContextID) |
| { |
| DDI_CHK_NULL(vaContextHeap, "nullptr vaContextHeap", ); |
| DDI_CHK_LESS(vaContextID, vaContextHeap->uiAllocatedHeapElements, "invalid context id", ); |
| PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT vaContextHeapBase = (PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT)vaContextHeap->pHeapBase; |
| PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT vaContextHeapElmt = &vaContextHeapBase[vaContextID]; |
| DDI_CHK_NULL(vaContextHeapElmt->pVaContext, "context is already released", ); |
| void *firstFree = vaContextHeap->pFirstFreeHeapElement; |
| vaContextHeap->pFirstFreeHeapElement = (void*)vaContextHeapElmt; |
| vaContextHeapElmt->pNextFree = (PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT)firstFree; |
| vaContextHeapElmt->pVaContext = nullptr; |
| } |
| |
| void DdiMediaUtil_UnRefBufObjInMediaBuffer(PDDI_MEDIA_BUFFER buf) |
| { |
| mos_bo_unreference(buf->bo); |
| } |
| |
| // Open Intel's Graphics Device to get the file descriptor |
| int32_t DdiMediaUtil_OpenGraphicsAdaptor(char *devName) |
| { |
| struct stat st; |
| int32_t hDevice = -1; |
| if(nullptr == devName) |
| { |
| DDI_ASSERTMESSAGE("Invalid Graphics Node"); |
| return -1; |
| } |
| |
| if (-1 == stat (devName, &st)) |
| { |
| DDI_ASSERTMESSAGE("Cannot identify '%s': %d, %s.", devName, errno, strerror (errno)); |
| return -1; |
| } |
| |
| if (!S_ISCHR (st.st_mode)) |
| { |
| DDI_ASSERTMESSAGE("%s is no device.", devName); |
| return -1; |
| } |
| |
| hDevice = open (devName, O_RDWR); |
| if (-1 == hDevice) |
| { |
| DDI_ASSERTMESSAGE("Cannot open '%s': %d, %s.", devName, errno, strerror (errno)); |
| return -1; |
| } |
| |
| return hDevice; |
| } |
| |
| VAStatus DdiMediaUtil_UnRegisterRTSurfaces( |
| VADriverContextP ctx, |
| PDDI_MEDIA_SURFACE surface) |
| { |
| DDI_CHK_NULL(ctx,"nullptr context!", VA_STATUS_ERROR_INVALID_CONTEXT); |
| PDDI_MEDIA_CONTEXT mediaCtx = DdiMedia_GetMediaContext(ctx); |
| DDI_CHK_NULL(mediaCtx,"nullptr mediaCtx!", VA_STATUS_ERROR_INVALID_CONTEXT); |
| DDI_CHK_NULL(surface, "nullptr surface!", VA_STATUS_ERROR_INVALID_PARAMETER); |
| |
| //Look through all decode contexts to unregister the surface in each decode context's RTtable. |
| if (mediaCtx->pDecoderCtxHeap != nullptr) |
| { |
| PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT decVACtxHeapBase; |
| |
| DdiMediaUtil_LockMutex(&mediaCtx->DecoderMutex); |
| decVACtxHeapBase = (PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT)mediaCtx->pDecoderCtxHeap->pHeapBase; |
| for (uint32_t j = 0; j < mediaCtx->pDecoderCtxHeap->uiAllocatedHeapElements; j++) |
| { |
| if (decVACtxHeapBase[j].pVaContext != nullptr) |
| { |
| PDDI_DECODE_CONTEXT decCtx = (PDDI_DECODE_CONTEXT)decVACtxHeapBase[j].pVaContext; |
| if (decCtx && decCtx->m_ddiDecode) |
| { |
| //not check the return value since the surface may not be registered in the context. pay attention to LOGW. |
| decCtx->m_ddiDecode->UnRegisterRTSurfaces(&decCtx->RTtbl, surface); |
| } |
| } |
| } |
| DdiMediaUtil_UnLockMutex(&mediaCtx->DecoderMutex); |
| } |
| if (mediaCtx->pEncoderCtxHeap != nullptr) |
| { |
| PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT pEncVACtxHeapBase; |
| |
| DdiMediaUtil_LockMutex(&mediaCtx->EncoderMutex); |
| pEncVACtxHeapBase = (PDDI_MEDIA_VACONTEXT_HEAP_ELEMENT)mediaCtx->pEncoderCtxHeap->pHeapBase; |
| for (uint32_t j = 0; j < mediaCtx->pEncoderCtxHeap->uiAllocatedHeapElements; j++) |
| { |
| if (pEncVACtxHeapBase[j].pVaContext != nullptr) |
| { |
| PDDI_ENCODE_CONTEXT pEncCtx = (PDDI_ENCODE_CONTEXT)pEncVACtxHeapBase[j].pVaContext; |
| if (pEncCtx && pEncCtx->m_encode) |
| { |
| //not check the return value since the surface may not be registered in the context. pay attention to LOGW. |
| pEncCtx->m_encode->UnRegisterRTSurfaces(&pEncCtx->RTtbl, surface); |
| } |
| } |
| } |
| DdiMediaUtil_UnLockMutex(&mediaCtx->EncoderMutex); |
| } |
| |
| return VA_STATUS_SUCCESS; |
| } |