2017-11-27 04:36:06 +02:00
/*
* This file is part of FFmpeg .
*
* FFmpeg is free software ; you can redistribute it and / or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation ; either
* version 2.1 of the License , or ( at your option ) any later version .
*
* FFmpeg is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* Lesser General Public License for more details .
*
* You should have received a copy of the GNU Lesser General Public
* License along with FFmpeg ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
2017-11-29 19:24:08 +02:00
# include "config.h"
2022-02-23 14:56:49 +02:00
# include "config_components.h"
2017-11-29 19:24:08 +02:00
2017-11-27 04:36:06 +02:00
# include "libavutil/avassert.h"
# include "libavutil/imgutils.h"
# include "libavutil/hwcontext.h"
# if CONFIG_D3D11VA
# include "libavutil/hwcontext_d3d11va.h"
# endif
2018-04-14 16:46:10 +02:00
# if CONFIG_DXVA2
# define COBJMACROS
# include "libavutil/hwcontext_dxva2.h"
# endif
2017-11-27 04:36:06 +02:00
# include "libavutil/mem.h"
# include "libavutil/pixdesc.h"
# include "libavutil/time.h"
# include "amfenc.h"
2020-06-09 23:31:32 +02:00
# include "encode.h"
2017-11-27 04:36:06 +02:00
# include "internal.h"
2024-07-31 18:01:41 +02:00
# include "libavutil/mastering_display_metadata.h"
static int amf_save_hdr_metadata ( AVCodecContext * avctx , const AVFrame * frame , AMFHDRMetadata * hdrmeta )
{
AVFrameSideData * sd_display ;
AVFrameSideData * sd_light ;
AVMasteringDisplayMetadata * display_meta ;
AVContentLightMetadata * light_meta ;
sd_display = av_frame_get_side_data ( frame , AV_FRAME_DATA_MASTERING_DISPLAY_METADATA ) ;
if ( sd_display ) {
display_meta = ( AVMasteringDisplayMetadata * ) sd_display - > data ;
if ( display_meta - > has_luminance ) {
const unsigned int luma_den = 10000 ;
hdrmeta - > maxMasteringLuminance =
( amf_uint32 ) ( luma_den * av_q2d ( display_meta - > max_luminance ) ) ;
hdrmeta - > minMasteringLuminance =
FFMIN ( ( amf_uint32 ) ( luma_den * av_q2d ( display_meta - > min_luminance ) ) , hdrmeta - > maxMasteringLuminance ) ;
}
if ( display_meta - > has_primaries ) {
const unsigned int chroma_den = 50000 ;
hdrmeta - > redPrimary [ 0 ] =
FFMIN ( ( amf_uint16 ) ( chroma_den * av_q2d ( display_meta - > display_primaries [ 0 ] [ 0 ] ) ) , chroma_den ) ;
hdrmeta - > redPrimary [ 1 ] =
FFMIN ( ( amf_uint16 ) ( chroma_den * av_q2d ( display_meta - > display_primaries [ 0 ] [ 1 ] ) ) , chroma_den ) ;
hdrmeta - > greenPrimary [ 0 ] =
FFMIN ( ( amf_uint16 ) ( chroma_den * av_q2d ( display_meta - > display_primaries [ 1 ] [ 0 ] ) ) , chroma_den ) ;
hdrmeta - > greenPrimary [ 1 ] =
FFMIN ( ( amf_uint16 ) ( chroma_den * av_q2d ( display_meta - > display_primaries [ 1 ] [ 1 ] ) ) , chroma_den ) ;
hdrmeta - > bluePrimary [ 0 ] =
FFMIN ( ( amf_uint16 ) ( chroma_den * av_q2d ( display_meta - > display_primaries [ 2 ] [ 0 ] ) ) , chroma_den ) ;
hdrmeta - > bluePrimary [ 1 ] =
FFMIN ( ( amf_uint16 ) ( chroma_den * av_q2d ( display_meta - > display_primaries [ 2 ] [ 1 ] ) ) , chroma_den ) ;
hdrmeta - > whitePoint [ 0 ] =
FFMIN ( ( amf_uint16 ) ( chroma_den * av_q2d ( display_meta - > white_point [ 0 ] ) ) , chroma_den ) ;
hdrmeta - > whitePoint [ 1 ] =
FFMIN ( ( amf_uint16 ) ( chroma_den * av_q2d ( display_meta - > white_point [ 1 ] ) ) , chroma_den ) ;
}
sd_light = av_frame_get_side_data ( frame , AV_FRAME_DATA_CONTENT_LIGHT_LEVEL ) ;
if ( sd_light ) {
light_meta = ( AVContentLightMetadata * ) sd_light - > data ;
if ( light_meta ) {
hdrmeta - > maxContentLightLevel = ( amf_uint16 ) light_meta - > MaxCLL ;
hdrmeta - > maxFrameAverageLightLevel = ( amf_uint16 ) light_meta - > MaxFALL ;
}
}
return 0 ;
}
return 1 ;
}
2017-11-27 04:36:06 +02:00
# if CONFIG_D3D11VA
# include <d3d11.h>
# endif
# ifdef _WIN32
# include "compat/w32dlfcn.h"
# else
# include <dlfcn.h>
# endif
# define FFMPEG_AMF_WRITER_ID L"ffmpeg_amf"
# define PTS_PROP L"PtsProp"
const enum AVPixelFormat ff_amf_pix_fmts [ ] = {
AV_PIX_FMT_NV12 ,
AV_PIX_FMT_YUV420P ,
# if CONFIG_D3D11VA
AV_PIX_FMT_D3D11 ,
2018-04-14 16:46:10 +02:00
# endif
# if CONFIG_DXVA2
AV_PIX_FMT_DXVA2_VLD ,
2017-11-27 04:36:06 +02:00
# endif
2024-07-31 18:00:57 +02:00
AV_PIX_FMT_P010 ,
2017-11-27 04:36:06 +02:00
AV_PIX_FMT_NONE
} ;
typedef struct FormatMap {
enum AVPixelFormat av_format ;
enum AMF_SURFACE_FORMAT amf_format ;
} FormatMap ;
static const FormatMap format_map [ ] =
{
{ AV_PIX_FMT_NONE , AMF_SURFACE_UNKNOWN } ,
{ AV_PIX_FMT_NV12 , AMF_SURFACE_NV12 } ,
2024-07-31 18:00:57 +02:00
{ AV_PIX_FMT_P010 , AMF_SURFACE_P010 } ,
2017-11-27 04:36:06 +02:00
{ AV_PIX_FMT_BGR0 , AMF_SURFACE_BGRA } ,
{ AV_PIX_FMT_RGB0 , AMF_SURFACE_RGBA } ,
{ AV_PIX_FMT_GRAY8 , AMF_SURFACE_GRAY8 } ,
{ AV_PIX_FMT_YUV420P , AMF_SURFACE_YUV420P } ,
{ AV_PIX_FMT_YUYV422 , AMF_SURFACE_YUY2 } ,
} ;
static enum AMF_SURFACE_FORMAT amf_av_to_amf_format ( enum AVPixelFormat fmt )
{
int i ;
for ( i = 0 ; i < amf_countof ( format_map ) ; i + + ) {
if ( format_map [ i ] . av_format = = fmt ) {
return format_map [ i ] . amf_format ;
}
}
return AMF_SURFACE_UNKNOWN ;
}
static void AMF_CDECL_CALL AMFTraceWriter_Write ( AMFTraceWriter * pThis ,
const wchar_t * scope , const wchar_t * message )
{
AmfTraceWriter * tracer = ( AmfTraceWriter * ) pThis ;
av_log ( tracer - > avctx , AV_LOG_DEBUG , " %ls: %ls " , scope , message ) ; // \n is provided from AMF
}
static void AMF_CDECL_CALL AMFTraceWriter_Flush ( AMFTraceWriter * pThis )
{
}
static AMFTraceWriterVtbl tracer_vtbl =
{
. Write = AMFTraceWriter_Write ,
. Flush = AMFTraceWriter_Flush ,
} ;
static int amf_load_library ( AVCodecContext * avctx )
{
2018-04-14 16:46:15 +02:00
AmfContext * ctx = avctx - > priv_data ;
AMFInit_Fn init_fun ;
AMFQueryVersion_Fn version_fun ;
AMF_RESULT res ;
2017-11-27 04:36:06 +02:00
ctx - > delayed_frame = av_frame_alloc ( ) ;
if ( ! ctx - > delayed_frame ) {
return AVERROR ( ENOMEM ) ;
}
2022-01-10 12:04:36 +02:00
// hardcoded to current HW queue size - will auto-realloc if too small
ctx - > timestamp_list = av_fifo_alloc2 ( avctx - > max_b_frames + 16 , sizeof ( int64_t ) ,
AV_FIFO_FLAG_AUTO_GROW ) ;
2017-11-27 04:36:06 +02:00
if ( ! ctx - > timestamp_list ) {
return AVERROR ( ENOMEM ) ;
}
ctx - > dts_delay = 0 ;
ctx - > library = dlopen ( AMF_DLL_NAMEA , RTLD_NOW | RTLD_LOCAL ) ;
AMF_RETURN_IF_FALSE ( ctx , ctx - > library ! = NULL ,
AVERROR_UNKNOWN , " DLL %s failed to open \n " , AMF_DLL_NAMEA ) ;
init_fun = ( AMFInit_Fn ) dlsym ( ctx - > library , AMF_INIT_FUNCTION_NAME ) ;
AMF_RETURN_IF_FALSE ( ctx , init_fun ! = NULL , AVERROR_UNKNOWN , " DLL %s failed to find function %s \n " , AMF_DLL_NAMEA , AMF_INIT_FUNCTION_NAME ) ;
version_fun = ( AMFQueryVersion_Fn ) dlsym ( ctx - > library , AMF_QUERY_VERSION_FUNCTION_NAME ) ;
AMF_RETURN_IF_FALSE ( ctx , version_fun ! = NULL , AVERROR_UNKNOWN , " DLL %s failed to find function %s \n " , AMF_DLL_NAMEA , AMF_QUERY_VERSION_FUNCTION_NAME ) ;
res = version_fun ( & ctx - > version ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " %s failed with error %d \n " , AMF_QUERY_VERSION_FUNCTION_NAME , res ) ;
res = init_fun ( AMF_FULL_VERSION , & ctx - > factory ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " %s failed with error %d \n " , AMF_INIT_FUNCTION_NAME , res ) ;
res = ctx - > factory - > pVtbl - > GetTrace ( ctx - > factory , & ctx - > trace ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " GetTrace() failed with error %d \n " , res ) ;
res = ctx - > factory - > pVtbl - > GetDebug ( ctx - > factory , & ctx - > debug ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " GetDebug() failed with error %d \n " , res ) ;
return 0 ;
}
2018-04-14 16:45:55 +02:00
# if CONFIG_D3D11VA
static int amf_init_from_d3d11_device ( AVCodecContext * avctx , AVD3D11VADeviceContext * hwctx )
{
AmfContext * ctx = avctx - > priv_data ;
AMF_RESULT res ;
res = ctx - > context - > pVtbl - > InitDX11 ( ctx - > context , hwctx - > device , AMF_DX11_1 ) ;
if ( res ! = AMF_OK ) {
if ( res = = AMF_NOT_SUPPORTED )
av_log ( avctx , AV_LOG_ERROR , " AMF via D3D11 is not supported on the given device. \n " ) ;
else
av_log ( avctx , AV_LOG_ERROR , " AMF failed to initialise on the given D3D11 device: %d. \n " , res ) ;
return AVERROR ( ENODEV ) ;
}
return 0 ;
}
# endif
2018-04-14 16:46:10 +02:00
# if CONFIG_DXVA2
static int amf_init_from_dxva2_device ( AVCodecContext * avctx , AVDXVA2DeviceContext * hwctx )
{
AmfContext * ctx = avctx - > priv_data ;
HANDLE device_handle ;
IDirect3DDevice9 * device ;
HRESULT hr ;
AMF_RESULT res ;
int ret ;
hr = IDirect3DDeviceManager9_OpenDeviceHandle ( hwctx - > devmgr , & device_handle ) ;
if ( FAILED ( hr ) ) {
av_log ( avctx , AV_LOG_ERROR , " Failed to open device handle for Direct3D9 device: %lx. \n " , ( unsigned long ) hr ) ;
return AVERROR_EXTERNAL ;
}
hr = IDirect3DDeviceManager9_LockDevice ( hwctx - > devmgr , device_handle , & device , FALSE ) ;
if ( SUCCEEDED ( hr ) ) {
IDirect3DDeviceManager9_UnlockDevice ( hwctx - > devmgr , device_handle , FALSE ) ;
ret = 0 ;
} else {
av_log ( avctx , AV_LOG_ERROR , " Failed to lock device handle for Direct3D9 device: %lx. \n " , ( unsigned long ) hr ) ;
ret = AVERROR_EXTERNAL ;
}
IDirect3DDeviceManager9_CloseDeviceHandle ( hwctx - > devmgr , device_handle ) ;
if ( ret < 0 )
return ret ;
res = ctx - > context - > pVtbl - > InitDX9 ( ctx - > context , device ) ;
IDirect3DDevice9_Release ( device ) ;
if ( res ! = AMF_OK ) {
if ( res = = AMF_NOT_SUPPORTED )
av_log ( avctx , AV_LOG_ERROR , " AMF via D3D9 is not supported on the given device. \n " ) ;
else
av_log ( avctx , AV_LOG_ERROR , " AMF failed to initialise on given D3D9 device: %d. \n " , res ) ;
return AVERROR ( ENODEV ) ;
}
return 0 ;
}
# endif
2017-11-27 04:36:06 +02:00
static int amf_init_context ( AVCodecContext * avctx )
{
2018-04-14 16:45:55 +02:00
AmfContext * ctx = avctx - > priv_data ;
2019-08-08 20:33:25 +02:00
AMFContext1 * context1 = NULL ;
2018-04-14 16:45:55 +02:00
AMF_RESULT res ;
av_unused int ret ;
2017-11-27 04:36:06 +02:00
2018-04-09 18:48:33 +02:00
ctx - > hwsurfaces_in_queue = 0 ;
ctx - > hwsurfaces_in_queue_max = 16 ;
2018-01-03 19:55:53 +02:00
// configure AMF logger
2017-11-27 04:36:06 +02:00
// the return of these functions indicates old state and do not affect behaviour
ctx - > trace - > pVtbl - > EnableWriter ( ctx - > trace , AMF_TRACE_WRITER_DEBUG_OUTPUT , ctx - > log_to_dbg ! = 0 ) ;
if ( ctx - > log_to_dbg )
ctx - > trace - > pVtbl - > SetWriterLevel ( ctx - > trace , AMF_TRACE_WRITER_DEBUG_OUTPUT , AMF_TRACE_TRACE ) ;
ctx - > trace - > pVtbl - > EnableWriter ( ctx - > trace , AMF_TRACE_WRITER_CONSOLE , 0 ) ;
ctx - > trace - > pVtbl - > SetGlobalLevel ( ctx - > trace , AMF_TRACE_TRACE ) ;
// connect AMF logger to av_log
ctx - > tracer . vtbl = & tracer_vtbl ;
ctx - > tracer . avctx = avctx ;
ctx - > trace - > pVtbl - > RegisterWriter ( ctx - > trace , FFMPEG_AMF_WRITER_ID , ( AMFTraceWriter * ) & ctx - > tracer , 1 ) ;
ctx - > trace - > pVtbl - > SetWriterLevel ( ctx - > trace , FFMPEG_AMF_WRITER_ID , AMF_TRACE_TRACE ) ;
res = ctx - > factory - > pVtbl - > CreateContext ( ctx - > factory , & ctx - > context ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " CreateContext() failed with error %d \n " , res ) ;
2018-04-14 16:45:55 +02:00
// If a device was passed to the encoder, try to initialise from that.
2017-11-27 04:36:06 +02:00
if ( avctx - > hw_frames_ctx ) {
2018-04-14 16:45:55 +02:00
AVHWFramesContext * frames_ctx = ( AVHWFramesContext * ) avctx - > hw_frames_ctx - > data ;
if ( amf_av_to_amf_format ( frames_ctx - > sw_format ) = = AMF_SURFACE_UNKNOWN ) {
av_log ( avctx , AV_LOG_ERROR , " Format of input frames context (%s) is not supported by AMF. \n " ,
av_get_pix_fmt_name ( frames_ctx - > sw_format ) ) ;
return AVERROR ( EINVAL ) ;
2017-11-27 04:36:06 +02:00
}
2018-04-14 16:45:55 +02:00
switch ( frames_ctx - > device_ctx - > type ) {
# if CONFIG_D3D11VA
case AV_HWDEVICE_TYPE_D3D11VA :
ret = amf_init_from_d3d11_device ( avctx , frames_ctx - > device_ctx - > hwctx ) ;
if ( ret < 0 )
return ret ;
break ;
2018-04-14 16:46:10 +02:00
# endif
# if CONFIG_DXVA2
case AV_HWDEVICE_TYPE_DXVA2 :
ret = amf_init_from_dxva2_device ( avctx , frames_ctx - > device_ctx - > hwctx ) ;
if ( ret < 0 )
return ret ;
break ;
2018-04-14 16:45:55 +02:00
# endif
default :
av_log ( avctx , AV_LOG_ERROR , " AMF initialisation from a %s frames context is not supported. \n " ,
av_hwdevice_get_type_name ( frames_ctx - > device_ctx - > type ) ) ;
return AVERROR ( ENOSYS ) ;
2017-11-27 04:36:06 +02:00
}
2018-04-14 16:45:55 +02:00
ctx - > hw_frames_ctx = av_buffer_ref ( avctx - > hw_frames_ctx ) ;
if ( ! ctx - > hw_frames_ctx )
return AVERROR ( ENOMEM ) ;
if ( frames_ctx - > initial_pool_size > 0 )
ctx - > hwsurfaces_in_queue_max = frames_ctx - > initial_pool_size - 1 ;
} else if ( avctx - > hw_device_ctx ) {
AVHWDeviceContext * device_ctx = ( AVHWDeviceContext * ) avctx - > hw_device_ctx - > data ;
switch ( device_ctx - > type ) {
# if CONFIG_D3D11VA
case AV_HWDEVICE_TYPE_D3D11VA :
ret = amf_init_from_d3d11_device ( avctx , device_ctx - > hwctx ) ;
if ( ret < 0 )
return ret ;
break ;
2018-04-14 16:46:10 +02:00
# endif
# if CONFIG_DXVA2
case AV_HWDEVICE_TYPE_DXVA2 :
ret = amf_init_from_dxva2_device ( avctx , device_ctx - > hwctx ) ;
if ( ret < 0 )
return ret ;
break ;
2017-11-27 04:36:06 +02:00
# endif
2018-04-14 16:45:55 +02:00
default :
av_log ( avctx , AV_LOG_ERROR , " AMF initialisation from a %s device is not supported. \n " ,
av_hwdevice_get_type_name ( device_ctx - > type ) ) ;
return AVERROR ( ENOSYS ) ;
}
ctx - > hw_device_ctx = av_buffer_ref ( avctx - > hw_device_ctx ) ;
if ( ! ctx - > hw_device_ctx )
return AVERROR ( ENOMEM ) ;
} else {
2017-11-27 04:36:06 +02:00
res = ctx - > context - > pVtbl - > InitDX11 ( ctx - > context , NULL , AMF_DX11_1 ) ;
2018-04-14 16:45:55 +02:00
if ( res = = AMF_OK ) {
av_log ( avctx , AV_LOG_VERBOSE , " AMF initialisation succeeded via D3D11. \n " ) ;
} else {
2017-11-27 04:36:06 +02:00
res = ctx - > context - > pVtbl - > InitDX9 ( ctx - > context , NULL ) ;
2018-04-14 16:45:55 +02:00
if ( res = = AMF_OK ) {
av_log ( avctx , AV_LOG_VERBOSE , " AMF initialisation succeeded via D3D9. \n " ) ;
} else {
2019-08-08 20:33:25 +02:00
AMFGuid guid = IID_AMFContext1 ( ) ;
res = ctx - > context - > pVtbl - > QueryInterface ( ctx - > context , & guid , ( void * * ) & context1 ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " CreateContext1() failed with error %d \n " , res ) ;
res = context1 - > pVtbl - > InitVulkan ( context1 , NULL ) ;
context1 - > pVtbl - > Release ( context1 ) ;
if ( res ! = AMF_OK ) {
if ( res = = AMF_NOT_SUPPORTED )
av_log ( avctx , AV_LOG_ERROR , " AMF via Vulkan is not supported on the given device. \n " ) ;
else
av_log ( avctx , AV_LOG_ERROR , " AMF failed to initialise on the given Vulkan device: %d. \n " , res ) ;
return AVERROR ( ENOSYS ) ;
}
av_log ( avctx , AV_LOG_VERBOSE , " AMF initialisation succeeded via Vulkan. \n " ) ;
2018-04-14 16:45:55 +02:00
}
2017-11-27 04:36:06 +02:00
}
}
return 0 ;
}
static int amf_init_encoder ( AVCodecContext * avctx )
{
2018-04-14 16:46:15 +02:00
AmfContext * ctx = avctx - > priv_data ;
const wchar_t * codec_id = NULL ;
AMF_RESULT res ;
enum AVPixelFormat pix_fmt ;
2017-11-27 04:36:06 +02:00
switch ( avctx - > codec - > id ) {
case AV_CODEC_ID_H264 :
codec_id = AMFVideoEncoderVCE_AVC ;
break ;
case AV_CODEC_ID_HEVC :
codec_id = AMFVideoEncoder_HEVC ;
break ;
2022-12-20 17:20:38 +02:00
case AV_CODEC_ID_AV1 :
codec_id = AMFVideoEncoder_AV1 ;
break ;
2017-11-27 04:36:06 +02:00
default :
break ;
}
AMF_RETURN_IF_FALSE ( ctx , codec_id ! = NULL , AVERROR ( EINVAL ) , " Codec %d is not supported \n " , avctx - > codec - > id ) ;
2018-04-14 16:46:05 +02:00
if ( ctx - > hw_frames_ctx )
pix_fmt = ( ( AVHWFramesContext * ) ctx - > hw_frames_ctx - > data ) - > sw_format ;
else
pix_fmt = avctx - > pix_fmt ;
2024-08-01 00:00:33 +02:00
if ( pix_fmt = = AV_PIX_FMT_P010 ) {
AMF_RETURN_IF_FALSE ( ctx , ctx - > version > = AMF_MAKE_FULL_VERSION ( 1 , 4 , 32 , 0 ) , AVERROR_UNKNOWN , " 10-bit encoder is not supported by AMD GPU drivers versions lower than 23.30. \n " ) ;
}
2018-04-14 16:46:05 +02:00
ctx - > format = amf_av_to_amf_format ( pix_fmt ) ;
AMF_RETURN_IF_FALSE ( ctx , ctx - > format ! = AMF_SURFACE_UNKNOWN , AVERROR ( EINVAL ) ,
" Format %s is not supported \n " , av_get_pix_fmt_name ( pix_fmt ) ) ;
2017-11-27 04:36:06 +02:00
res = ctx - > factory - > pVtbl - > CreateComponent ( ctx - > factory , ctx - > context , codec_id , & ctx - > encoder ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_ENCODER_NOT_FOUND , " CreateComponent(%ls) failed with error %d \n " , codec_id , res ) ;
2024-10-12 00:28:00 +02:00
ctx - > submitted_frame = 0 ;
2017-11-27 04:36:06 +02:00
return 0 ;
}
int av_cold ff_amf_encode_close ( AVCodecContext * avctx )
{
2018-04-14 16:46:15 +02:00
AmfContext * ctx = avctx - > priv_data ;
if ( ctx - > delayed_surface ) {
2017-11-27 04:36:06 +02:00
ctx - > delayed_surface - > pVtbl - > Release ( ctx - > delayed_surface ) ;
ctx - > delayed_surface = NULL ;
}
if ( ctx - > encoder ) {
ctx - > encoder - > pVtbl - > Terminate ( ctx - > encoder ) ;
ctx - > encoder - > pVtbl - > Release ( ctx - > encoder ) ;
ctx - > encoder = NULL ;
}
if ( ctx - > context ) {
ctx - > context - > pVtbl - > Terminate ( ctx - > context ) ;
ctx - > context - > pVtbl - > Release ( ctx - > context ) ;
ctx - > context = NULL ;
}
av_buffer_unref ( & ctx - > hw_device_ctx ) ;
av_buffer_unref ( & ctx - > hw_frames_ctx ) ;
if ( ctx - > trace ) {
ctx - > trace - > pVtbl - > UnregisterWriter ( ctx - > trace , FFMPEG_AMF_WRITER_ID ) ;
}
if ( ctx - > library ) {
dlclose ( ctx - > library ) ;
ctx - > library = NULL ;
}
ctx - > trace = NULL ;
ctx - > debug = NULL ;
ctx - > factory = NULL ;
ctx - > version = 0 ;
ctx - > delayed_drain = 0 ;
av_frame_free ( & ctx - > delayed_frame ) ;
2022-01-10 12:04:36 +02:00
av_fifo_freep2 ( & ctx - > timestamp_list ) ;
2017-11-27 04:36:06 +02:00
return 0 ;
}
static int amf_copy_surface ( AVCodecContext * avctx , const AVFrame * frame ,
AMFSurface * surface )
{
2018-04-14 16:46:15 +02:00
AMFPlane * plane ;
uint8_t * dst_data [ 4 ] ;
int dst_linesize [ 4 ] ;
int planes ;
int i ;
2017-11-27 04:36:06 +02:00
2018-04-14 16:46:00 +02:00
planes = surface - > pVtbl - > GetPlanesCount ( surface ) ;
av_assert0 ( planes < FF_ARRAY_ELEMS ( dst_data ) ) ;
2017-11-27 04:36:06 +02:00
for ( i = 0 ; i < planes ; i + + ) {
plane = surface - > pVtbl - > GetPlaneAt ( surface , i ) ;
dst_data [ i ] = plane - > pVtbl - > GetNative ( plane ) ;
dst_linesize [ i ] = plane - > pVtbl - > GetHPitch ( plane ) ;
}
2023-09-07 00:09:10 +02:00
av_image_copy2 ( dst_data , dst_linesize ,
frame - > data , frame - > linesize , frame - > format ,
avctx - > width , avctx - > height ) ;
2017-11-27 04:36:06 +02:00
2018-04-14 16:46:00 +02:00
return 0 ;
2017-11-27 04:36:06 +02:00
}
static int amf_copy_buffer ( AVCodecContext * avctx , AVPacket * pkt , AMFBuffer * buffer )
{
2018-04-14 16:46:15 +02:00
AmfContext * ctx = avctx - > priv_data ;
int ret ;
AMFVariantStruct var = { 0 } ;
int64_t timestamp = AV_NOPTS_VALUE ;
int64_t size = buffer - > pVtbl - > GetSize ( buffer ) ;
2017-11-27 04:36:06 +02:00
2021-03-13 01:49:44 +02:00
if ( ( ret = ff_get_encode_buffer ( avctx , pkt , size , 0 ) ) < 0 ) {
2017-11-27 04:36:06 +02:00
return ret ;
}
memcpy ( pkt - > data , buffer - > pVtbl - > GetNative ( buffer ) , size ) ;
switch ( avctx - > codec - > id ) {
case AV_CODEC_ID_H264 :
buffer - > pVtbl - > GetProperty ( buffer , AMF_VIDEO_ENCODER_OUTPUT_DATA_TYPE , & var ) ;
if ( var . int64Value = = AMF_VIDEO_ENCODER_OUTPUT_DATA_TYPE_IDR ) {
pkt - > flags = AV_PKT_FLAG_KEY ;
}
break ;
case AV_CODEC_ID_HEVC :
buffer - > pVtbl - > GetProperty ( buffer , AMF_VIDEO_ENCODER_HEVC_OUTPUT_DATA_TYPE , & var ) ;
if ( var . int64Value = = AMF_VIDEO_ENCODER_HEVC_OUTPUT_DATA_TYPE_IDR ) {
pkt - > flags = AV_PKT_FLAG_KEY ;
}
break ;
2022-12-20 17:20:38 +02:00
case AV_CODEC_ID_AV1 :
buffer - > pVtbl - > GetProperty ( buffer , AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE , & var ) ;
if ( var . int64Value = = AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE_KEY ) {
pkt - > flags = AV_PKT_FLAG_KEY ;
}
2017-11-27 04:36:06 +02:00
default :
break ;
}
buffer - > pVtbl - > GetProperty ( buffer , PTS_PROP , & var ) ;
pkt - > pts = var . int64Value ; // original pts
2022-01-10 12:04:36 +02:00
AMF_RETURN_IF_FALSE ( ctx , av_fifo_read ( ctx - > timestamp_list , & timestamp , 1 ) > = 0 ,
AVERROR_UNKNOWN , " timestamp_list is empty \n " ) ;
2017-11-27 04:36:06 +02:00
// calc dts shift if max_b_frames > 0
2023-05-17 14:59:40 +02:00
if ( ( ctx - > max_b_frames > 0 | | ( ( ctx - > pa_adaptive_mini_gop = = 1 ) ? true : false ) ) & & ctx - > dts_delay = = 0 ) {
2017-11-27 04:36:06 +02:00
int64_t timestamp_last = AV_NOPTS_VALUE ;
2022-01-10 12:04:36 +02:00
size_t can_read = av_fifo_can_read ( ctx - > timestamp_list ) ;
AMF_RETURN_IF_FALSE ( ctx , can_read > 0 , AVERROR_UNKNOWN ,
2017-11-27 04:36:06 +02:00
" timestamp_list is empty while max_b_frames = %d \n " , avctx - > max_b_frames ) ;
2022-01-10 12:04:36 +02:00
av_fifo_peek ( ctx - > timestamp_list , & timestamp_last , 1 , can_read - 1 ) ;
2017-11-27 04:36:06 +02:00
if ( timestamp < 0 | | timestamp_last < AV_NOPTS_VALUE ) {
return AVERROR ( ERANGE ) ;
}
ctx - > dts_delay = timestamp_last - timestamp ;
}
pkt - > dts = timestamp - ctx - > dts_delay ;
return 0 ;
}
2018-01-03 19:55:53 +02:00
// amfenc API implementation
2017-11-27 04:36:06 +02:00
int ff_amf_encode_init ( AVCodecContext * avctx )
{
2018-04-14 16:46:15 +02:00
int ret ;
2017-11-27 04:36:06 +02:00
if ( ( ret = amf_load_library ( avctx ) ) = = 0 ) {
if ( ( ret = amf_init_context ( avctx ) ) = = 0 ) {
if ( ( ret = amf_init_encoder ( avctx ) ) = = 0 ) {
return 0 ;
}
}
}
ff_amf_encode_close ( avctx ) ;
return ret ;
}
2018-04-09 18:48:33 +02:00
static AMF_RESULT amf_set_property_buffer ( AMFSurface * object , const wchar_t * name , AMFBuffer * val )
{
AMF_RESULT res ;
AMFVariantStruct var ;
res = AMFVariantInit ( & var ) ;
if ( res = = AMF_OK ) {
AMFGuid guid_AMFInterface = IID_AMFInterface ( ) ;
AMFInterface * amf_interface ;
res = val - > pVtbl - > QueryInterface ( val , & guid_AMFInterface , ( void * * ) & amf_interface ) ;
if ( res = = AMF_OK ) {
res = AMFVariantAssignInterface ( & var , amf_interface ) ;
amf_interface - > pVtbl - > Release ( amf_interface ) ;
}
if ( res = = AMF_OK ) {
res = object - > pVtbl - > SetProperty ( object , name , var ) ;
}
AMFVariantClear ( & var ) ;
}
return res ;
}
static AMF_RESULT amf_get_property_buffer ( AMFData * object , const wchar_t * name , AMFBuffer * * val )
{
AMF_RESULT res ;
AMFVariantStruct var ;
res = AMFVariantInit ( & var ) ;
if ( res = = AMF_OK ) {
res = object - > pVtbl - > GetProperty ( object , name , & var ) ;
if ( res = = AMF_OK ) {
if ( var . type = = AMF_VARIANT_INTERFACE ) {
AMFGuid guid_AMFBuffer = IID_AMFBuffer ( ) ;
AMFInterface * amf_interface = AMFVariantInterface ( & var ) ;
res = amf_interface - > pVtbl - > QueryInterface ( amf_interface , & guid_AMFBuffer , ( void * * ) val ) ;
} else {
res = AMF_INVALID_DATA_TYPE ;
}
}
AMFVariantClear ( & var ) ;
}
return res ;
}
static AMFBuffer * amf_create_buffer_with_frame_ref ( const AVFrame * frame , AMFContext * context )
{
AVFrame * frame_ref ;
AMFBuffer * frame_ref_storage_buffer = NULL ;
AMF_RESULT res ;
res = context - > pVtbl - > AllocBuffer ( context , AMF_MEMORY_HOST , sizeof ( frame_ref ) , & frame_ref_storage_buffer ) ;
if ( res = = AMF_OK ) {
frame_ref = av_frame_clone ( frame ) ;
if ( frame_ref ) {
memcpy ( frame_ref_storage_buffer - > pVtbl - > GetNative ( frame_ref_storage_buffer ) , & frame_ref , sizeof ( frame_ref ) ) ;
} else {
frame_ref_storage_buffer - > pVtbl - > Release ( frame_ref_storage_buffer ) ;
frame_ref_storage_buffer = NULL ;
}
}
return frame_ref_storage_buffer ;
}
static void amf_release_buffer_with_frame_ref ( AMFBuffer * frame_ref_storage_buffer )
{
2018-04-14 16:46:15 +02:00
AVFrame * frame_ref ;
memcpy ( & frame_ref , frame_ref_storage_buffer - > pVtbl - > GetNative ( frame_ref_storage_buffer ) , sizeof ( frame_ref ) ) ;
av_frame_free ( & frame_ref ) ;
2018-04-09 18:48:33 +02:00
frame_ref_storage_buffer - > pVtbl - > Release ( frame_ref_storage_buffer ) ;
}
2017-11-27 04:36:06 +02:00
2020-06-09 23:31:32 +02:00
int ff_amf_receive_packet ( AVCodecContext * avctx , AVPacket * avpkt )
2017-11-27 04:36:06 +02:00
{
2018-04-14 16:46:15 +02:00
AmfContext * ctx = avctx - > priv_data ;
AMFSurface * surface ;
AMF_RESULT res ;
int ret ;
2020-06-09 23:31:32 +02:00
AMF_RESULT res_query ;
AMFData * data = NULL ;
AVFrame * frame = ctx - > delayed_frame ;
int block_and_wait ;
2023-05-17 14:59:40 +02:00
int query_output_data_flag = 0 ;
AMF_RESULT res_resubmit ;
2017-11-27 04:36:06 +02:00
if ( ! ctx - > encoder )
return AVERROR ( EINVAL ) ;
2020-06-09 23:31:32 +02:00
if ( ! frame - > buf [ 0 ] ) {
ret = ff_encode_get_frame ( avctx , frame ) ;
if ( ret < 0 & & ret ! = AVERROR_EOF )
return ret ;
}
if ( ! frame - > buf [ 0 ] ) { // submit drain
2017-11-27 04:36:06 +02:00
if ( ! ctx - > eof ) { // submit drain one time only
if ( ctx - > delayed_surface ! = NULL ) {
ctx - > delayed_drain = 1 ; // input queue is full: resubmit Drain() in ff_amf_receive_packet
} else if ( ! ctx - > delayed_drain ) {
res = ctx - > encoder - > pVtbl - > Drain ( ctx - > encoder ) ;
if ( res = = AMF_INPUT_FULL ) {
ctx - > delayed_drain = 1 ; // input queue is full: resubmit Drain() in ff_amf_receive_packet
} else {
if ( res = = AMF_OK ) {
ctx - > eof = 1 ; // drain started
}
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " Drain() failed with error %d \n " , res ) ;
}
}
}
2020-06-09 23:31:32 +02:00
} else if ( ! ctx - > delayed_surface ) { // submit frame
2018-04-14 16:46:00 +02:00
int hw_surface = 0 ;
2017-11-27 04:36:06 +02:00
// prepare surface from frame
2018-04-14 16:46:00 +02:00
switch ( frame - > format ) {
2017-11-27 04:36:06 +02:00
# if CONFIG_D3D11VA
2018-04-14 16:46:00 +02:00
case AV_PIX_FMT_D3D11 :
{
static const GUID AMFTextureArrayIndexGUID = { 0x28115527 , 0xe7c3 , 0x4b66 , { 0x99 , 0xd3 , 0x4f , 0x2a , 0xe6 , 0xb4 , 0x7f , 0xaf } } ;
ID3D11Texture2D * texture = ( ID3D11Texture2D * ) frame - > data [ 0 ] ; // actual texture
int index = ( intptr_t ) frame - > data [ 1 ] ; // index is a slice in texture array is - set to tell AMF which slice to use
2017-11-27 04:36:06 +02:00
2018-04-14 16:46:00 +02:00
av_assert0 ( frame - > hw_frames_ctx & & ctx - > hw_frames_ctx & &
frame - > hw_frames_ctx - > data = = ctx - > hw_frames_ctx - > data ) ;
texture - > lpVtbl - > SetPrivateData ( texture , & AMFTextureArrayIndexGUID , sizeof ( index ) , & index ) ;
res = ctx - > context - > pVtbl - > CreateSurfaceFromDX11Native ( ctx - > context , texture , & surface , NULL ) ; // wrap to AMF surface
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR ( ENOMEM ) , " CreateSurfaceFromDX11Native() failed with error %d \n " , res ) ;
hw_surface = 1 ;
}
break ;
2018-04-14 16:46:10 +02:00
# endif
# if CONFIG_DXVA2
case AV_PIX_FMT_DXVA2_VLD :
{
IDirect3DSurface9 * texture = ( IDirect3DSurface9 * ) frame - > data [ 3 ] ; // actual texture
res = ctx - > context - > pVtbl - > CreateSurfaceFromDX9Native ( ctx - > context , texture , & surface , NULL ) ; // wrap to AMF surface
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR ( ENOMEM ) , " CreateSurfaceFromDX9Native() failed with error %d \n " , res ) ;
hw_surface = 1 ;
}
break ;
2018-04-14 16:46:00 +02:00
# endif
default :
{
res = ctx - > context - > pVtbl - > AllocSurface ( ctx - > context , AMF_MEMORY_HOST , ctx - > format , avctx - > width , avctx - > height , & surface ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR ( ENOMEM ) , " AllocSurface() failed with error %d \n " , res ) ;
amf_copy_surface ( avctx , frame , surface ) ;
}
break ;
}
if ( hw_surface ) {
AMFBuffer * frame_ref_storage_buffer ;
2017-11-27 04:36:06 +02:00
// input HW surfaces can be vertically aligned by 16; tell AMF the real size
surface - > pVtbl - > SetCrop ( surface , 0 , 0 , frame - > width , frame - > height ) ;
2018-04-09 18:48:33 +02:00
frame_ref_storage_buffer = amf_create_buffer_with_frame_ref ( frame , ctx - > context ) ;
AMF_RETURN_IF_FALSE ( ctx , frame_ref_storage_buffer ! = NULL , AVERROR ( ENOMEM ) , " create_buffer_with_frame_ref() returned NULL \n " ) ;
res = amf_set_property_buffer ( surface , L " av_frame_ref " , frame_ref_storage_buffer ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " SetProperty failed for \" av_frame_ref \" with error %d \n " , res ) ;
ctx - > hwsurfaces_in_queue + + ;
frame_ref_storage_buffer - > pVtbl - > Release ( frame_ref_storage_buffer ) ;
2017-11-27 04:36:06 +02:00
}
2018-04-14 16:46:00 +02:00
2024-07-31 18:01:41 +02:00
// HDR10 metadata
if ( frame - > color_trc = = AVCOL_TRC_SMPTE2084 ) {
AMFBuffer * hdrmeta_buffer = NULL ;
res = ctx - > context - > pVtbl - > AllocBuffer ( ctx - > context , AMF_MEMORY_HOST , sizeof ( AMFHDRMetadata ) , & hdrmeta_buffer ) ;
if ( res = = AMF_OK ) {
AMFHDRMetadata * hdrmeta = ( AMFHDRMetadata * ) hdrmeta_buffer - > pVtbl - > GetNative ( hdrmeta_buffer ) ;
if ( amf_save_hdr_metadata ( avctx , frame , hdrmeta ) = = 0 ) {
switch ( avctx - > codec - > id ) {
case AV_CODEC_ID_H264 :
AMF_ASSIGN_PROPERTY_INTERFACE ( res , ctx - > encoder , AMF_VIDEO_ENCODER_INPUT_HDR_METADATA , hdrmeta_buffer ) ; break ;
case AV_CODEC_ID_HEVC :
AMF_ASSIGN_PROPERTY_INTERFACE ( res , ctx - > encoder , AMF_VIDEO_ENCODER_HEVC_INPUT_HDR_METADATA , hdrmeta_buffer ) ; break ;
2024-07-31 18:13:07 +02:00
case AV_CODEC_ID_AV1 :
AMF_ASSIGN_PROPERTY_INTERFACE ( res , ctx - > encoder , AMF_VIDEO_ENCODER_AV1_INPUT_HDR_METADATA , hdrmeta_buffer ) ; break ;
2024-07-31 18:01:41 +02:00
}
res = amf_set_property_buffer ( surface , L " av_frame_hdrmeta " , hdrmeta_buffer ) ;
AMF_RETURN_IF_FALSE ( avctx , res = = AMF_OK , AVERROR_UNKNOWN , " SetProperty failed for \" av_frame_hdrmeta \" with error %d \n " , res ) ;
}
hdrmeta_buffer - > pVtbl - > Release ( hdrmeta_buffer ) ;
}
}
2017-11-27 04:36:06 +02:00
surface - > pVtbl - > SetPts ( surface , frame - > pts ) ;
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , PTS_PROP , frame - > pts ) ;
switch ( avctx - > codec - > id ) {
case AV_CODEC_ID_H264 :
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_INSERT_AUD , ! ! ctx - > aud ) ;
2024-09-08 22:07:43 +02:00
switch ( frame - > pict_type ) {
case AV_PICTURE_TYPE_I :
if ( ctx - > forced_idr ) {
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_INSERT_SPS , 1 ) ;
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_INSERT_PPS , 1 ) ;
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_FORCE_PICTURE_TYPE , AMF_VIDEO_ENCODER_PICTURE_TYPE_IDR ) ;
} else {
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_FORCE_PICTURE_TYPE , AMF_VIDEO_ENCODER_PICTURE_TYPE_I ) ;
}
break ;
case AV_PICTURE_TYPE_P :
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_FORCE_PICTURE_TYPE , AMF_VIDEO_ENCODER_PICTURE_TYPE_P ) ;
break ;
case AV_PICTURE_TYPE_B :
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_FORCE_PICTURE_TYPE , AMF_VIDEO_ENCODER_PICTURE_TYPE_B ) ;
break ;
}
2017-11-27 04:36:06 +02:00
break ;
case AV_CODEC_ID_HEVC :
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_HEVC_INSERT_AUD , ! ! ctx - > aud ) ;
2024-09-08 22:07:43 +02:00
switch ( frame - > pict_type ) {
case AV_PICTURE_TYPE_I :
if ( ctx - > forced_idr ) {
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_HEVC_INSERT_HEADER , 1 ) ;
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_HEVC_FORCE_PICTURE_TYPE , AMF_VIDEO_ENCODER_HEVC_PICTURE_TYPE_IDR ) ;
} else {
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_HEVC_FORCE_PICTURE_TYPE , AMF_VIDEO_ENCODER_HEVC_PICTURE_TYPE_I ) ;
}
break ;
case AV_PICTURE_TYPE_P :
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_HEVC_FORCE_PICTURE_TYPE , AMF_VIDEO_ENCODER_HEVC_PICTURE_TYPE_P ) ;
break ;
}
break ;
case AV_CODEC_ID_AV1 :
if ( frame - > pict_type = = AV_PICTURE_TYPE_I ) {
if ( ctx - > forced_idr ) {
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_AV1_FORCE_INSERT_SEQUENCE_HEADER , 1 ) ;
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE , AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE_KEY ) ;
} else {
AMF_ASSIGN_PROPERTY_INT64 ( res , surface , AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE , AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE_INTRA_ONLY ) ;
}
}
2017-11-27 04:36:06 +02:00
break ;
default :
break ;
}
// submit surface
res = ctx - > encoder - > pVtbl - > SubmitInput ( ctx - > encoder , ( AMFData * ) surface ) ;
if ( res = = AMF_INPUT_FULL ) { // handle full queue
//store surface for later submission
ctx - > delayed_surface = surface ;
} else {
2020-06-09 23:31:32 +02:00
int64_t pts = frame - > pts ;
2017-11-27 04:36:06 +02:00
surface - > pVtbl - > Release ( surface ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " SubmitInput() failed with error %d \n " , res ) ;
2020-06-09 23:31:32 +02:00
av_frame_unref ( frame ) ;
2022-01-10 12:04:36 +02:00
ret = av_fifo_write ( ctx - > timestamp_list , & pts , 1 ) ;
2024-10-12 00:28:00 +02:00
if ( ctx - > submitted_frame = = 0 )
{
ctx - > use_b_frame = ( ctx - > max_b_frames > 0 | | ( ( ctx - > pa_adaptive_mini_gop = = 1 ) ? true : false ) ) ;
}
ctx - > submitted_frame + + ;
2022-01-10 12:04:36 +02:00
if ( ret < 0 )
2017-11-27 04:36:06 +02:00
return ret ;
}
}
do {
block_and_wait = 0 ;
// poll data
2024-10-12 00:28:00 +02:00
if ( ! avpkt - > data & & ! avpkt - > buf & & ( ctx - > use_b_frame ? ( ctx - > submitted_frame > = 2 ) : true ) ) {
2023-05-17 14:59:40 +02:00
res_query = ctx - > encoder - > pVtbl - > QueryOutput ( ctx - > encoder , & data ) ;
if ( data ) {
// copy data to packet
AMFBuffer * buffer ;
AMFGuid guid = IID_AMFBuffer ( ) ;
2023-08-06 12:00:12 +02:00
query_output_data_flag = 1 ;
2023-05-17 14:59:40 +02:00
data - > pVtbl - > QueryInterface ( data , & guid , ( void * * ) & buffer ) ; // query for buffer interface
ret = amf_copy_buffer ( avctx , avpkt , buffer ) ;
2024-10-12 00:28:00 +02:00
ctx - > submitted_frame + + ;
2023-05-17 14:59:40 +02:00
buffer - > pVtbl - > Release ( buffer ) ;
if ( data - > pVtbl - > HasProperty ( data , L " av_frame_ref " ) ) {
AMFBuffer * frame_ref_storage_buffer ;
res = amf_get_property_buffer ( data , L " av_frame_ref " , & frame_ref_storage_buffer ) ;
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " GetProperty failed for \" av_frame_ref \" with error %d \n " , res ) ;
amf_release_buffer_with_frame_ref ( frame_ref_storage_buffer ) ;
ctx - > hwsurfaces_in_queue - - ;
}
2017-11-27 04:36:06 +02:00
2023-05-17 14:59:40 +02:00
data - > pVtbl - > Release ( data ) ;
2017-11-27 04:36:06 +02:00
2023-05-17 14:59:40 +02:00
AMF_RETURN_IF_FALSE ( ctx , ret > = 0 , ret , " amf_copy_buffer() failed with error %d \n " , ret ) ;
}
}
res_resubmit = AMF_OK ;
if ( ctx - > delayed_surface ! = NULL ) { // try to resubmit frame
2024-07-31 18:01:41 +02:00
if ( ctx - > delayed_surface - > pVtbl - > HasProperty ( ctx - > delayed_surface , L " av_frame_hdrmeta " ) ) {
AMFBuffer * hdrmeta_buffer = NULL ;
res = amf_get_property_buffer ( ( AMFData * ) ctx - > delayed_surface , L " av_frame_hdrmeta " , & hdrmeta_buffer ) ;
AMF_RETURN_IF_FALSE ( avctx , res = = AMF_OK , AVERROR_UNKNOWN , " GetProperty failed for \" av_frame_hdrmeta \" with error %d \n " , res ) ;
switch ( avctx - > codec - > id ) {
case AV_CODEC_ID_H264 :
AMF_ASSIGN_PROPERTY_INTERFACE ( res , ctx - > encoder , AMF_VIDEO_ENCODER_INPUT_HDR_METADATA , hdrmeta_buffer ) ; break ;
case AV_CODEC_ID_HEVC :
AMF_ASSIGN_PROPERTY_INTERFACE ( res , ctx - > encoder , AMF_VIDEO_ENCODER_HEVC_INPUT_HDR_METADATA , hdrmeta_buffer ) ; break ;
2024-09-10 03:14:15 +02:00
case AV_CODEC_ID_AV1 :
AMF_ASSIGN_PROPERTY_INTERFACE ( res , ctx - > encoder , AMF_VIDEO_ENCODER_AV1_INPUT_HDR_METADATA , hdrmeta_buffer ) ; break ;
2024-07-31 18:01:41 +02:00
}
hdrmeta_buffer - > pVtbl - > Release ( hdrmeta_buffer ) ;
}
2023-05-17 14:59:40 +02:00
res_resubmit = ctx - > encoder - > pVtbl - > SubmitInput ( ctx - > encoder , ( AMFData * ) ctx - > delayed_surface ) ;
if ( res_resubmit ! = AMF_INPUT_FULL ) {
int64_t pts = ctx - > delayed_surface - > pVtbl - > GetPts ( ctx - > delayed_surface ) ;
ctx - > delayed_surface - > pVtbl - > Release ( ctx - > delayed_surface ) ;
ctx - > delayed_surface = NULL ;
av_frame_unref ( ctx - > delayed_frame ) ;
AMF_RETURN_IF_FALSE ( ctx , res_resubmit = = AMF_OK , AVERROR_UNKNOWN , " Repeated SubmitInput() failed with error %d \n " , res_resubmit ) ;
2024-10-12 00:28:00 +02:00
ctx - > submitted_frame + + ;
2023-05-17 14:59:40 +02:00
ret = av_fifo_write ( ctx - > timestamp_list , & pts , 1 ) ;
if ( ret < 0 )
return ret ;
}
} else if ( ctx - > delayed_drain ) { // try to resubmit drain
res = ctx - > encoder - > pVtbl - > Drain ( ctx - > encoder ) ;
if ( res ! = AMF_INPUT_FULL ) {
ctx - > delayed_drain = 0 ;
ctx - > eof = 1 ; // drain started
AMF_RETURN_IF_FALSE ( ctx , res = = AMF_OK , AVERROR_UNKNOWN , " Repeated Drain() failed with error %d \n " , res ) ;
} else {
av_log ( avctx , AV_LOG_WARNING , " Data acquired but delayed drain submission got AMF_INPUT_FULL- should not happen \n " ) ;
}
}
2017-11-27 04:36:06 +02:00
2023-05-17 14:59:40 +02:00
if ( query_output_data_flag = = 0 ) {
if ( res_resubmit = = AMF_INPUT_FULL | | ctx - > delayed_drain | | ( ctx - > eof & & res_query ! = AMF_EOF ) | | ( ctx - > hwsurfaces_in_queue > = ctx - > hwsurfaces_in_queue_max ) ) {
block_and_wait = 1 ;
2024-10-12 00:28:00 +02:00
// Only sleep if the driver doesn't support waiting in QueryOutput()
// or if we already have output data so we will skip calling it.
if ( ! ctx - > query_timeout_supported | | avpkt - > data | | avpkt - > buf ) {
av_usleep ( 1000 ) ;
}
2017-11-27 04:36:06 +02:00
}
}
} while ( block_and_wait ) ;
if ( res_query = = AMF_EOF ) {
ret = AVERROR_EOF ;
} else if ( data = = NULL ) {
ret = AVERROR ( EAGAIN ) ;
} else {
ret = 0 ;
}
return ret ;
}
2020-10-15 02:16:21 +02:00
2024-07-31 18:00:57 +02:00
int ff_amf_get_color_profile ( AVCodecContext * avctx )
{
amf_int64 color_profile = AMF_VIDEO_CONVERTER_COLOR_PROFILE_UNKNOWN ;
if ( avctx - > color_range = = AVCOL_RANGE_JPEG ) {
/// Color Space for Full (JPEG) Range
switch ( avctx - > colorspace ) {
case AVCOL_SPC_SMPTE170M :
color_profile = AMF_VIDEO_CONVERTER_COLOR_PROFILE_FULL_601 ;
break ;
case AVCOL_SPC_BT709 :
color_profile = AMF_VIDEO_CONVERTER_COLOR_PROFILE_FULL_709 ;
break ;
case AVCOL_SPC_BT2020_NCL :
case AVCOL_SPC_BT2020_CL :
color_profile = AMF_VIDEO_CONVERTER_COLOR_PROFILE_FULL_2020 ;
break ;
}
} else {
/// Color Space for Limited (MPEG) range
switch ( avctx - > colorspace ) {
case AVCOL_SPC_SMPTE170M :
color_profile = AMF_VIDEO_CONVERTER_COLOR_PROFILE_601 ;
break ;
case AVCOL_SPC_BT709 :
color_profile = AMF_VIDEO_CONVERTER_COLOR_PROFILE_709 ;
break ;
case AVCOL_SPC_BT2020_NCL :
case AVCOL_SPC_BT2020_CL :
color_profile = AMF_VIDEO_CONVERTER_COLOR_PROFILE_2020 ;
break ;
}
}
return color_profile ;
}
2020-10-15 02:16:21 +02:00
const AVCodecHWConfigInternal * const ff_amfenc_hw_configs [ ] = {
# if CONFIG_D3D11VA
HW_CONFIG_ENCODER_FRAMES ( D3D11 , D3D11VA ) ,
HW_CONFIG_ENCODER_DEVICE ( NONE , D3D11VA ) ,
# endif
# if CONFIG_DXVA2
HW_CONFIG_ENCODER_FRAMES ( DXVA2_VLD , DXVA2 ) ,
HW_CONFIG_ENCODER_DEVICE ( NONE , DXVA2 ) ,
# endif
NULL ,
} ;