blob: 3cd86923d16acaea7e8ba34808122facc4abc7ad [file] [log] [blame]
/*
* 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;
}
else
{
if (surface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_KERNEL_DRM ||
surface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_DRM_PRIME ||
surface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_DRM_PRIME_2 ||
surface->pSurfDesc->uiVaMemType == VA_SURFACE_ATTRIB_MEM_TYPE_USER_PTR)
{
return true;
}else
{
return false;
}
}
}
//!
//! \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_RESCREATE_CUSTOM_PARAMS gmmCustomParams;
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 cpTag = 0;
int mem_type = mediaSurface->memType;
#ifdef _MMC_SUPPORTED
bool bMemCompEnable = true;
#else
bool bMemCompEnable = false;
#endif
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;
alignedWidth = MOS_ALIGN_CEIL(width, 2);
alignedHeight = MOS_ALIGN_CEIL(height, 2);
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:
#if VA_CHECK_VERSION(1, 9, 0)
case Media_Format_Y212:
#endif
case Media_Format_Y216:
case Media_Format_AYUV:
case Media_Format_Y410:
#if VA_CHECK_VERSION(1, 9, 0)
case Media_Format_Y412:
#endif
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))
{
// Default set as compression not supported, surface compression import only support from Memory Type VA_SURFACE_ATTRIB_MEM_TYPE_DRM_PRIME_2
bMemCompEnable = false;
bMemCompRC = false;
pitch = mediaSurface->pSurfDesc->uiPitches[0];
DDI_CHK_CONDITION(pitch == 0, "Invalid pich.", VA_STATUS_ERROR_INVALID_PARAMETER);
// 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;
//Overwrite the tile format that matches the exteral 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 )
{
#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);
}
else
{
DDI_ASSERTMESSAGE("Failed to create drm buffer vmap.");
return VA_STATUS_ERROR_ALLOCATION_FAILED;
}
}
else
{
DDI_ASSERTMESSAGE("Unsupported external surface memory type.");
return VA_STATUS_ERROR_ALLOCATION_FAILED;
}
// Set cp flag to indicate the secure surface
if (mediaSurface->pSurfDesc->uiFlags & VA_SURFACE_EXTBUF_DESC_PROTECTED)
{
cpTag = PROTECTED_SURFACE_TAG;
}
// For secure/compressible external surface, call legacy gmm interface due to limitation of new interface
if (cpTag || bMemCompEnable)
{
MOS_ZeroMemory(&gmmParams, sizeof(gmmParams));
gmmParams.BaseWidth = width;
gmmParams.BaseHeight = height;
gmmParams.ArraySize = 1;
gmmParams.Type = RESOURCE_2D;
gmmParams.CpTag = cpTag;
gmmParams.Format = mediaDrvCtx->m_caps->ConvertMediaFmtToGmmFmt(format);
DDI_CHK_CONDITION(gmmParams.Format == GMM_FORMAT_INVALID, "Unsupported format", VA_STATUS_ERROR_UNSUPPORTED_RT_FORMAT);
switch (tileformat)
{
case I915_TILING_Y:
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);
gmmResourceInfo = mediaDrvCtx->pGmmClientContext->CreateResInfoObject(&gmmParams);
}
else
{
int32_t baseHeight = 0;
DDI_CHK_CONDITION(mediaSurface->pSurfDesc->uiPlanes == 0,
"Invalid plane number.",
VA_STATUS_ERROR_INVALID_PARAMETER);
if (mediaSurface->pSurfDesc->uiPlanes == 1)
{
DDI_CHK_CONDITION(mediaSurface->pSurfDesc->uiSize == 0,
"Invalid Size.",
VA_STATUS_ERROR_INVALID_PARAMETER);
baseHeight = mediaSurface->pSurfDesc->uiSize / pitch;
}
else
{
DDI_CHK_CONDITION(mediaSurface->pSurfDesc->uiOffsets[1] == 0,
"Invalid offset.",
VA_STATUS_ERROR_INVALID_PARAMETER);
baseHeight = mediaSurface->pSurfDesc->uiOffsets[1] / pitch;
}
// Create GmmResourceInfo
MOS_ZeroMemory(&gmmCustomParams, sizeof(gmmCustomParams));
gmmCustomParams.Type = RESOURCE_2D;
gmmCustomParams.Format = mediaDrvCtx->m_caps->ConvertMediaFmtToGmmFmt(format);
gmmCustomParams.BaseWidth64 = width;
gmmCustomParams.BaseHeight = baseHeight;
gmmCustomParams.Pitch = pitch;
gmmCustomParams.Size = mediaSurface->pSurfDesc->uiSize;
gmmCustomParams.BaseAlignment = 4096;
gmmCustomParams.NoOfPlanes = mediaSurface->pSurfDesc->uiPlanes;
switch (tileformat)
{
case I915_TILING_Y:
gmmCustomParams.Flags.Info.TiledY = true;
break;
case I915_TILING_X:
gmmCustomParams.Flags.Info.TiledX = true;
break;
case I915_TILING_NONE:
default:
gmmCustomParams.Flags.Info.Linear = true;
}
switch(mediaSurface->pSurfDesc->uiPlanes)
{
case 1:
gmmCustomParams.PlaneOffset.X[GMM_PLANE_Y] = 0;
gmmCustomParams.PlaneOffset.Y[GMM_PLANE_Y] = mediaSurface->pSurfDesc->uiOffsets[0] / pitch;
break;
case 2:
gmmCustomParams.PlaneOffset.X[GMM_PLANE_Y] = 0;
gmmCustomParams.PlaneOffset.Y[GMM_PLANE_Y] = mediaSurface->pSurfDesc->uiOffsets[0] / pitch;
gmmCustomParams.PlaneOffset.X[GMM_PLANE_U] = 0;
gmmCustomParams.PlaneOffset.Y[GMM_PLANE_U] = mediaSurface->pSurfDesc->uiOffsets[1] / pitch;
gmmCustomParams.PlaneOffset.X[GMM_PLANE_V] = 0;
gmmCustomParams.PlaneOffset.Y[GMM_PLANE_V] = mediaSurface->pSurfDesc->uiOffsets[1] / pitch;
break;
case 3:
if (mediaSurface->format == Media_Format_YV12)
{
gmmCustomParams.PlaneOffset.X[GMM_PLANE_Y] = 0;
gmmCustomParams.PlaneOffset.Y[GMM_PLANE_Y] = mediaSurface->pSurfDesc->uiOffsets[0] / pitch;
gmmCustomParams.PlaneOffset.X[GMM_PLANE_U] = 0;
gmmCustomParams.PlaneOffset.Y[GMM_PLANE_U] = mediaSurface->pSurfDesc->uiOffsets[2] / pitch;
gmmCustomParams.PlaneOffset.X[GMM_PLANE_V] = 0;
gmmCustomParams.PlaneOffset.Y[GMM_PLANE_V] = mediaSurface->pSurfDesc->uiOffsets[1] / pitch;
}
else
{
gmmCustomParams.PlaneOffset.X[GMM_PLANE_Y] = 0;
gmmCustomParams.PlaneOffset.Y[GMM_PLANE_Y] = mediaSurface->pSurfDesc->uiOffsets[0] / pitch;
gmmCustomParams.PlaneOffset.X[GMM_PLANE_U] = 0;
gmmCustomParams.PlaneOffset.Y[GMM_PLANE_U] = mediaSurface->pSurfDesc->uiOffsets[1] / pitch;
gmmCustomParams.PlaneOffset.X[GMM_PLANE_V] = 0;
gmmCustomParams.PlaneOffset.Y[GMM_PLANE_V] = mediaSurface->pSurfDesc->uiOffsets[2] / pitch;
}
break;
default:
DDI_ASSERTMESSAGE("Invalid plane number.");
return VA_STATUS_ERROR_ALLOCATION_FAILED;
}
gmmResourceInfo = mediaDrvCtx->pGmmClientContext->CreateCustomResInfoObject(&gmmCustomParams);
}
if(nullptr == gmmResourceInfo)
{
DDI_ASSERTMESSAGE("Gmm Create Resource Failed.");
hRes = VA_STATUS_ERROR_ALLOCATION_FAILED;
goto finish;
}
if (bo)
{
mediaSurface->pGmmResourceInfo = gmmResourceInfo;
mediaSurface->bMapped = false;
mediaSurface->format = format;
mediaSurface->iWidth = width;
mediaSurface->iHeight = gmmResourceInfo->GetBaseHeight();
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("Allocate external surface %7d bytes (%d x %d resource).", mediaSurface->pSurfDesc->uiSize, width, height);
uint32_t event[] = {bo->handle, format, width, height, pitch, bo->size, tileformat, cpTag};
MOS_TraceEventExt(EVENT_VA_SURFACE, EVENT_TYPE_INFO, event, sizeof(event), &gmmResourceInfo->GetResFlags(), sizeof(GMM_RESOURCE_FLAG));
}
else
{
DDI_ASSERTMESSAGE("Fail to allocate external surface");
return VA_STATUS_ERROR_ALLOCATION_FAILED;
}
}
else
{
if (mediaSurface->pSurfDesc)
{
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;
if (format == Media_Format_YV12 ||
format == Media_Format_I420)
{
alignedHeight = MOS_ALIGN_CEIL(height, 2);
}
}
}
// Create GmmResourceInfo
MOS_ZeroMemory(&gmmParams, sizeof(gmmParams));
gmmParams.BaseWidth = alignedWidth;
gmmParams.BaseHeight = alignedHeight;
gmmParams.ArraySize = 1;
gmmParams.Type = RESOURCE_2D;
gmmParams.Format = mediaDrvCtx->m_caps->ConvertMediaFmtToGmmFmt(format);
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)gmmResourceInfo->GetRenderPitch();
uint32_t gmmSize = (uint32_t)gmmResourceInfo->GetSizeSurface();
uint32_t gmmHeight = gmmResourceInfo->GetBaseHeight();
if ( 0 == gmmPitch || 0 == gmmSize || 0 == gmmHeight)
{
DDI_ASSERTMESSAGE("Gmm Create Resource Failed.");
hRes = VA_STATUS_ERROR_ALLOCATION_FAILED;
goto finish;
}
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;
}
MemoryPolicyParameter memPolicyPar;
MOS_ZeroMemory(&memPolicyPar, sizeof(MemoryPolicyParameter));
memPolicyPar.skuTable = &mediaDrvCtx->SkuTable;
memPolicyPar.waTable = &mediaDrvCtx->WaTable;
memPolicyPar.resInfo = mediaSurface->pGmmResourceInfo;
memPolicyPar.resName = "Media Surface";
memPolicyPar.preferredMemType = (MEDIA_IS_WA(&mediaDrvCtx->WaTable, WaForceAllocateLML4)) ? MOS_MEMPOOL_DEVICEMEMORY : mem_type;
mem_type = MemoryPolicyManager::UpdateMemoryPolicy(&memPolicyPar);
if ( tileformat == I915_TILING_NONE )
{
bo = mos_bo_alloc(mediaDrvCtx->pDrmBufMgr, "MEDIA", gmmSize, 4096, mem_type);
pitch = gmmPitch;
}
else
{
unsigned long ulPitch = 0;
bo = mos_bo_alloc_tiled(mediaDrvCtx->pDrmBufMgr, "MEDIA", gmmPitch, (gmmSize + gmmPitch -1)/gmmPitch, 1, &tileformat, (unsigned long *)&ulPitch, 0, mem_type);
pitch = ulPitch;
}
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);
uint32_t event[] = {bo->handle, format, width, height, pitch, bo->size, tileformat, cpTag};
MOS_TraceEventExt(EVENT_VA_SURFACE, EVENT_TYPE_INFO, event, sizeof(event), &gmmResourceInfo->GetResFlags(), sizeof(GMM_RESOURCE_FLAG));
}
else
{
DDI_ASSERTMESSAGE("Fail to Alloc %7d bytes (%d x %d resource).",gmmSize, width, height);
hRes = VA_STATUS_ERROR_ALLOCATION_FAILED;
goto finish;
}
}
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);
MemoryPolicyParameter memPolicyPar;
MOS_ZeroMemory(&memPolicyPar, sizeof(MemoryPolicyParameter));
memPolicyPar.skuTable = &mediaBuffer->pMediaCtx->SkuTable;
memPolicyPar.waTable = &mediaBuffer->pMediaCtx->WaTable;
memPolicyPar.resInfo = mediaBuffer->pGmmResourceInfo;
memPolicyPar.resName = "Media Buffer";
memPolicyPar.uiType = mediaBuffer->uiType;
memPolicyPar.preferredMemType = mediaBuffer->bUseSysGfxMem ? MOS_MEMPOOL_SYSTEMMEMORY : 0;
mem_type = MemoryPolicyManager::UpdateMemoryPolicy(&memPolicyPar);
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);
uint32_t event[] = {bo->handle, format, size, 1, size, bo->size, 0, 0};
MOS_TraceEventExt(EVENT_VA_BUFFER, EVENT_TYPE_INFO, event, sizeof(event), &mediaBuffer->pGmmResourceInfo->GetResFlags(), sizeof(GMM_RESOURCE_FLAG));
}
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();
MemoryPolicyParameter memPolicyPar;
MOS_ZeroMemory(&memPolicyPar, sizeof(MemoryPolicyParameter));
memPolicyPar.skuTable = &mediaBuffer->pMediaCtx->SkuTable;
memPolicyPar.waTable = &mediaBuffer->pMediaCtx->WaTable;
memPolicyPar.resInfo = mediaBuffer->pGmmResourceInfo;
memPolicyPar.resName = "Media 2D Buffer";
mem_type = MemoryPolicyManager::UpdateMemoryPolicy(&memPolicyPar);
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);
uint32_t event[] = {bo->handle, mediaBuffer->format, width, height, gmmPitch, bo->size, tileformat, 0};
MOS_TraceEventExt(EVENT_VA_BUFFER, EVENT_TYPE_INFO, event, sizeof(event), &gmmResourceInfo->GetResFlags(), sizeof(GMM_RESOURCE_FLAG));
}
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);
static VAStatus CreateShadowResource(DDI_MEDIA_SURFACE *surface)
{
VAStatus vaStatus = VA_STATUS_SUCCESS;
DDI_CHK_NULL(surface, "nullptr surface", VA_STATUS_ERROR_INVALID_SURFACE);
if (surface->pGmmResourceInfo->GetSetCpSurfTag(0, 0) != 0)
{
return VA_STATUS_ERROR_INVALID_SURFACE;
}
if (surface->iWidth <= 512 || surface->iRealHeight <= 512 || surface->format == Media_Format_P016)
{
return VA_STATUS_ERROR_UNSUPPORTED_RT_FORMAT;
}
surface->pShadowBuffer = (DDI_MEDIA_BUFFER *)MOS_AllocAndZeroMemory(sizeof(DDI_MEDIA_BUFFER));
DDI_CHK_NULL(surface->pShadowBuffer, "Failed to allocate shadow buffer", VA_STATUS_ERROR_INVALID_BUFFER);
surface->pShadowBuffer->pMediaCtx = surface->pMediaCtx;
surface->pShadowBuffer->bUseSysGfxMem = true;
surface->pShadowBuffer->iSize = surface->pGmmResourceInfo->GetSizeSurface();
vaStatus = DdiMediaUtil_AllocateBuffer(Media_Format_Buffer,
surface->pShadowBuffer->iSize,
surface->pShadowBuffer,
surface->pMediaCtx->pDrmBufMgr);
if (vaStatus != VA_STATUS_SUCCESS)
{
MOS_FreeMemory(surface->pShadowBuffer);
surface->pShadowBuffer = nullptr;
}
return vaStatus;
}
//!
//! \brief Swizzle surface by Hardware, current only support VEBOX
//!
//! \param [in] surface
//! Pointer of surface
//! \param [in] isDeSwizzle
//! Whether it's de-swizzling or not
//! Swizzling - copying from video memory to temporary buffer
//! De-swizzling - copying from temporary buffer to video memory
//!
//! \return VAStatus
//! VA_STATUS_SUCCESS if success, else fail reason
//!
static VAStatus SwizzleSurfaceByHW(DDI_MEDIA_SURFACE *surface, bool isDeSwizzle = false)
{
DDI_CHK_NULL(surface, "nullptr surface", VA_STATUS_ERROR_INVALID_SURFACE);
DDI_CHK_NULL(surface->pMediaCtx, "nullptr media context", VA_STATUS_ERROR_INVALID_CONTEXT);
MOS_CONTEXT mosCtx = { };
PERF_DATA perfData = { };
PDDI_MEDIA_CONTEXT mediaDrvCtx = surface->pMediaCtx;
// Get the buf manager for codechal create
mosCtx.bufmgr = mediaDrvCtx->pDrmBufMgr;
mosCtx.m_gpuContextMgr = mediaDrvCtx->m_gpuContextMgr;
mosCtx.m_cmdBufMgr = mediaDrvCtx->m_cmdBufMgr;
mosCtx.fd = mediaDrvCtx->fd;
mosCtx.iDeviceId = mediaDrvCtx->iDeviceId;
mosCtx.SkuTable = mediaDrvCtx->SkuTable;
mosCtx.WaTable = mediaDrvCtx->WaTable;
mosCtx.gtSystemInfo = *mediaDrvCtx->pGtSystemInfo;
mosCtx.platform = mediaDrvCtx->platform;
mosCtx.ppMediaMemDecompState = &mediaDrvCtx->pMediaMemDecompState;
mosCtx.pfnMemoryDecompress = mediaDrvCtx->pfnMemoryDecompress;
mosCtx.pfnMediaMemoryCopy = mediaDrvCtx->pfnMediaMemoryCopy;
mosCtx.pfnMediaMemoryCopy2D = mediaDrvCtx->pfnMediaMemoryCopy2D;
mosCtx.pPerfData = &perfData;
mosCtx.gtSystemInfo = *mediaDrvCtx->pGtSystemInfo;
mosCtx.m_auxTableMgr = mediaDrvCtx->m_auxTableMgr;
mosCtx.pGmmClientContext = mediaDrvCtx->pGmmClientContext;
mosCtx.m_osDeviceContext = mediaDrvCtx->m_osDeviceContext;
MOS_RESOURCE source = {};
MOS_RESOURCE target = {};
if (isDeSwizzle)
{
DdiMedia_MediaBufferToMosResource(surface->pShadowBuffer, &source);
DdiMedia_MediaSurfaceToMosResource(surface, &target);
}
else
{
DdiMedia_MediaSurfaceToMosResource(surface, &source);
DdiMedia_MediaBufferToMosResource(surface->pShadowBuffer, &target);
}
return mediaDrvCtx->pfnMediaMemoryTileConvert(
&mosCtx,
&source,
&target,
surface->iWidth,
surface->iRealHeight,
0,
0,
!isDeSwizzle,
false);
}
// 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))
{
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 = VA_STATUS_SUCCESS;
if (MEDIA_IS_SKU(&surface->pMediaCtx->SkuTable, FtrLocalMemory))
{
if (surface->pShadowBuffer == nullptr)
{
CreateShadowResource(surface);
}
if (surface->pShadowBuffer != nullptr)
{
vaStatus = SwizzleSurfaceByHW(surface);
int err = 0;
if (vaStatus == VA_STATUS_SUCCESS)
{
err = mos_bo_map(surface->pShadowBuffer->bo, flag & MOS_LOCKFLAG_WRITEONLY);
}
if (vaStatus != VA_STATUS_SUCCESS || err != 0)
{
DdiMediaUtil_FreeBuffer(surface->pShadowBuffer);
MOS_FreeMemory(surface->pShadowBuffer);
surface->pShadowBuffer = nullptr;
}
}
}
mos_bo_map(surface->bo, flag & MOS_LOCKFLAG_WRITEONLY);
if (surface->pShadowBuffer == nullptr)
{
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);
}
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;
if (surface->pShadowBuffer)
{
surface->pData = (uint8_t *)surface->pShadowBuffer->bo->virt;
}
else if (surface->pSystemShadow)
{
surface->pData = surface->pSystemShadow;
}
else
{
surface->pData = (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->pShadowBuffer != nullptr)
{
SwizzleSurfaceByHW(surface, true);
mos_bo_unmap(surface->pShadowBuffer->bo);
DdiMediaUtil_FreeBuffer(surface->pShadowBuffer);
MOS_FreeMemory(surface->pShadowBuffer);
surface->pShadowBuffer = nullptr;
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);
}
if(surface->bMapped)
{
DdiMediaUtil_UnlockSurface(surface);
DDI_VERBOSEMESSAGE("DDI: try to free a locked surface.");
}
mos_bo_unreference(surface->bo);
// For External Buffer, only needs to destory SurfaceDescriptor
if (surface->pSurfDesc)
{
MOS_FreeMemory(surface->pSurfDesc);
surface->pSurfDesc = 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;
}
VAStatus DdiMediaUtil_SetMediaResetEnableFlag(PDDI_MEDIA_CONTEXT mediaCtx)
{
DDI_CHK_NULL(mediaCtx,"nullptr mediaCtx!", VA_STATUS_ERROR_INVALID_CONTEXT);
if(!MEDIA_IS_SKU(&mediaCtx->SkuTable, FtrSWMediaReset))
{
mediaCtx->bMediaResetEnable = false;
return VA_STATUS_SUCCESS;
}
MOS_USER_FEATURE_VALUE_DATA userFeatureData;
MOS_ZeroMemory(&userFeatureData, sizeof(userFeatureData));
MOS_USER_FEATURE_INVALID_KEY_ASSERT(
MOS_UserFeature_ReadValue_ID(nullptr,
__MEDIA_USER_FEATURE_VALUE_MEDIA_RESET_ENABLE_ID,
&userFeatureData,
nullptr));
mediaCtx->bMediaResetEnable = userFeatureData.i32Data ? true : false;
if(mediaCtx->bMediaResetEnable)
{
return VA_STATUS_SUCCESS;
}
char* mediaResetEnv = getenv("INTEL_MEDIA_RESET_WATCHDOG");
if(!mediaResetEnv)
{
mediaCtx->bMediaResetEnable = false;
return VA_STATUS_SUCCESS;
}
mediaCtx->bMediaResetEnable = strcmp(mediaResetEnv, "1") ? false : true;
return VA_STATUS_SUCCESS;
}