This adds a new API, which allows the API user to query the required AVHWFramesContext parameters. This also reduces code duplication across the hwaccels by introducing ff_decode_get_hw_frames_ctx(), which uses the new API function. It takes care of initializing the hw_frames_ctx if needed, and does additional error handling and API usage checking. Support for VDA and Cuvid missing. Signed-off-by: Anton Khirnov <anton@khirnov.net>tags/n4.0
| @@ -13,6 +13,9 @@ libavutil: 2017-03-23 | |||
| API changes, most recent first: | |||
| 2017-xx-xx - xxxxxxx - lavc 58.5.0 - avcodec.h | |||
| Add avcodec_get_hw_frames_parameters(). | |||
| 2017-xx-xx - xxxxxxx - lavu 56.6.0 - pixdesc.h | |||
| Add av_color_range_from_name(), av_color_primaries_from_name(), | |||
| av_color_transfer_from_name(), av_color_space_from_name(), and | |||
| @@ -2990,6 +2990,16 @@ typedef struct AVHWAccel { | |||
| * Internal hwaccel capabilities. | |||
| */ | |||
| int caps_internal; | |||
| /** | |||
| * Fill the given hw_frames context with current codec parameters. Called | |||
| * from get_format. Refer to avcodec_get_hw_frames_parameters() for | |||
| * details. | |||
| * | |||
| * This CAN be called before AVHWAccel.init is called, and you must assume | |||
| * that avctx->hwaccel_priv_data is invalid. | |||
| */ | |||
| int (*frame_params)(AVCodecContext *avctx, AVBufferRef *hw_frames_ctx); | |||
| } AVHWAccel; | |||
| /** | |||
| @@ -3984,6 +3994,109 @@ int avcodec_send_frame(AVCodecContext *avctx, const AVFrame *frame); | |||
| */ | |||
| int avcodec_receive_packet(AVCodecContext *avctx, AVPacket *avpkt); | |||
| /** | |||
| * Create and return a AVHWFramesContext with values adequate for hardware | |||
| * decoding. This is meant to get called from the get_format callback, and is | |||
| * a helper for preparing a AVHWFramesContext for AVCodecContext.hw_frames_ctx. | |||
| * This API is for decoding with certain hardware acceleration modes/APIs only. | |||
| * | |||
| * The returned AVHWFramesContext is not initialized. The caller must do this | |||
| * with av_hwframe_ctx_init(). | |||
| * | |||
| * Calling this function is not a requirement, but makes it simpler to avoid | |||
| * codec or hardware API specific details when manually allocating frames. | |||
| * | |||
| * Alternatively to this, an API user can set AVCodecContext.hw_device_ctx, | |||
| * which sets up AVCodecContext.hw_frames_ctx fully automatically, and makes | |||
| * it unnecessary to call this function or having to care about | |||
| * AVHWFramesContext initialization at all. | |||
| * | |||
| * There are a number of requirements for calling this function: | |||
| * | |||
| * - It must be called from get_format with the same avctx parameter that was | |||
| * passed to get_format. Calling it outside of get_format is not allowed, and | |||
| * can trigger undefined behavior. | |||
| * - The function is not always supported (see description of return values). | |||
| * Even if this function returns successfully, hwaccel initialization could | |||
| * fail later. (The degree to which implementations check whether the stream | |||
| * is actually supported varies. Some do this check only after the user's | |||
| * get_format callback returns.) | |||
| * - The hw_pix_fmt must be one of the choices suggested by get_format. If the | |||
| * user decides to use a AVHWFramesContext prepared with this API function, | |||
| * the user must return the same hw_pix_fmt from get_format. | |||
| * - The device_ref passed to this function must support the given hw_pix_fmt. | |||
| * - After calling this API function, it is the user's responsibility to | |||
| * initialize the AVHWFramesContext (returned by the out_frames_ref parameter), | |||
| * and to set AVCodecContext.hw_frames_ctx to it. If done, this must be done | |||
| * before returning from get_format (this is implied by the normal | |||
| * AVCodecContext.hw_frames_ctx API rules). | |||
| * - The AVHWFramesContext parameters may change every time time get_format is | |||
| * called. Also, AVCodecContext.hw_frames_ctx is reset before get_format. So | |||
| * you are inherently required to go through this process again on every | |||
| * get_format call. | |||
| * - It is perfectly possible to call this function without actually using | |||
| * the resulting AVHWFramesContext. One use-case might be trying to reuse a | |||
| * previously initialized AVHWFramesContext, and calling this API function | |||
| * only to test whether the required frame parameters have changed. | |||
| * - Fields that use dynamically allocated values of any kind must not be set | |||
| * by the user unless setting them is explicitly allowed by the documentation. | |||
| * If the user sets AVHWFramesContext.free and AVHWFramesContext.user_opaque, | |||
| * the new free callback must call the potentially set previous free callback. | |||
| * This API call may set any dynamically allocated fields, including the free | |||
| * callback. | |||
| * | |||
| * The function will set at least the following fields on AVHWFramesContext | |||
| * (potentially more, depending on hwaccel API): | |||
| * | |||
| * - All fields set by av_hwframe_ctx_alloc(). | |||
| * - Set the format field to hw_pix_fmt. | |||
| * - Set the sw_format field to the most suited and most versatile format. (An | |||
| * implication is that this will prefer generic formats over opaque formats | |||
| * with arbitrary restrictions, if possible.) | |||
| * - Set the width/height fields to the coded frame size, rounded up to the | |||
| * API-specific minimum alignment. | |||
| * - Only _if_ the hwaccel requires a pre-allocated pool: set the initial_pool_size | |||
| * field to the number of maximum reference surfaces possible with the codec, | |||
| * plus 1 surface for the user to work (meaning the user can safely reference | |||
| * at most 1 decoded surface at a time), plus additional buffering introduced | |||
| * by frame threading. If the hwaccel does not require pre-allocation, the | |||
| * field is left to 0, and the decoder will allocate new surfaces on demand | |||
| * during decoding. | |||
| * - Possibly AVHWFramesContext.hwctx fields, depending on the underlying | |||
| * hardware API. | |||
| * | |||
| * Essentially, out_frames_ref returns the same as av_hwframe_ctx_alloc(), but | |||
| * with basic frame parameters set. | |||
| * | |||
| * The function is stateless, and does not change the AVCodecContext or the | |||
| * device_ref AVHWDeviceContext. | |||
| * | |||
| * @param avctx The context which is currently calling get_format, and which | |||
| * implicitly contains all state needed for filling the returned | |||
| * AVHWFramesContext properly. | |||
| * @param device_ref A reference to the AVHWDeviceContext describing the device | |||
| * which will be used by the hardware decoder. | |||
| * @param hw_pix_fmt The hwaccel format you are going to return from get_format. | |||
| * @param out_frames_ref On success, set to a reference to an _uninitialized_ | |||
| * AVHWFramesContext, created from the given device_ref. | |||
| * Fields will be set to values required for decoding. | |||
| * Not changed if an error is returned. | |||
| * @return zero on success, a negative value on error. The following error codes | |||
| * have special semantics: | |||
| * AVERROR(ENOENT): the decoder does not support this functionality. Setup | |||
| * is always manual, or it is a decoder which does not | |||
| * support setting AVCodecContext.hw_frames_ctx at all, | |||
| * or it is a software format. | |||
| * AVERROR(EINVAL): it is known that hardware decoding is not supported for | |||
| * this configuration, or the device_ref is not supported | |||
| * for the hwaccel referenced by hw_pix_fmt. | |||
| */ | |||
| int avcodec_get_hw_frames_parameters(AVCodecContext *avctx, | |||
| AVBufferRef *device_ref, | |||
| enum AVPixelFormat hw_pix_fmt, | |||
| AVBufferRef **out_frames_ref); | |||
| /** | |||
| * @defgroup lavc_parsing Frame parsing | |||
| @@ -669,6 +669,88 @@ static AVHWAccel *find_hwaccel(enum AVCodecID codec_id, | |||
| return NULL; | |||
| } | |||
| int ff_decode_get_hw_frames_ctx(AVCodecContext *avctx, | |||
| enum AVHWDeviceType dev_type) | |||
| { | |||
| AVHWDeviceContext *device_ctx; | |||
| AVHWFramesContext *frames_ctx; | |||
| int ret; | |||
| if (!avctx->hwaccel) | |||
| return AVERROR(ENOSYS); | |||
| if (avctx->hw_frames_ctx) | |||
| return 0; | |||
| if (!avctx->hw_device_ctx) { | |||
| av_log(avctx, AV_LOG_ERROR, "A hardware frames or device context is " | |||
| "required for hardware accelerated decoding.\n"); | |||
| return AVERROR(EINVAL); | |||
| } | |||
| device_ctx = (AVHWDeviceContext *)avctx->hw_device_ctx->data; | |||
| if (device_ctx->type != dev_type) { | |||
| av_log(avctx, AV_LOG_ERROR, "Device type %s expected for hardware " | |||
| "decoding, but got %s.\n", av_hwdevice_get_type_name(dev_type), | |||
| av_hwdevice_get_type_name(device_ctx->type)); | |||
| return AVERROR(EINVAL); | |||
| } | |||
| ret = avcodec_get_hw_frames_parameters(avctx, | |||
| avctx->hw_device_ctx, | |||
| avctx->hwaccel->pix_fmt, | |||
| avctx->hw_frames_ctx); | |||
| if (ret < 0) { | |||
| av_buffer_unref(&avctx->hw_frames_ctx); | |||
| return ret; | |||
| } | |||
| frames_ctx = (AVHWFramesContext*)avctx->hw_frames_ctx->data; | |||
| if (frames_ctx->initial_pool_size) { | |||
| // We guarantee 4 base work surfaces. The function above guarantees 1 | |||
| // (the absolute minimum), so add the missing count. | |||
| frames_ctx->initial_pool_size += 3; | |||
| // Add an additional surface per thread is frame threading is enabled. | |||
| if (avctx->active_thread_type & FF_THREAD_FRAME) | |||
| frames_ctx->initial_pool_size += avctx->thread_count; | |||
| } | |||
| ret = av_hwframe_ctx_init(avctx->hw_frames_ctx); | |||
| if (ret < 0) { | |||
| av_buffer_unref(&avctx->hw_frames_ctx); | |||
| return ret; | |||
| } | |||
| return 0; | |||
| } | |||
| int avcodec_get_hw_frames_parameters(AVCodecContext *avctx, | |||
| AVBufferRef *device_ref, | |||
| enum AVPixelFormat hw_pix_fmt, | |||
| AVBufferRef **out_frames_ref) | |||
| { | |||
| AVBufferRef *frames_ref = NULL; | |||
| AVHWAccel *hwa = find_hwaccel(avctx->codec_id, hw_pix_fmt); | |||
| int ret; | |||
| if (!hwa || !hwa->frame_params) | |||
| return AVERROR(ENOENT); | |||
| frames_ref = av_hwframe_ctx_alloc(device_ref); | |||
| if (!frames_ref) | |||
| return AVERROR(ENOMEM); | |||
| ret = hwa->frame_params(avctx, frames_ref); | |||
| if (ret >= 0) { | |||
| *out_frames_ref = frames_ref; | |||
| } else { | |||
| av_buffer_unref(&frames_ref); | |||
| } | |||
| return ret; | |||
| } | |||
| static int setup_hwaccel(AVCodecContext *avctx, | |||
| const enum AVPixelFormat fmt, | |||
| const char *name) | |||
| @@ -23,6 +23,7 @@ | |||
| #include "libavutil/buffer.h" | |||
| #include "libavutil/frame.h" | |||
| #include "libavutil/hwcontext.h" | |||
| #include "avcodec.h" | |||
| @@ -70,4 +71,12 @@ int ff_decode_get_packet(AVCodecContext *avctx, AVPacket *pkt); | |||
| void ff_decode_bsfs_uninit(AVCodecContext *avctx); | |||
| /** | |||
| * Make sure avctx.hw_frames_ctx is set. If it's not set, the function will | |||
| * try to allocate it from hw_device_ctx. If that is not possible, an error | |||
| * message is printed, and an error code is returned. | |||
| */ | |||
| int ff_decode_get_hw_frames_ctx(AVCodecContext *avctx, | |||
| enum AVHWDeviceType dev_type); | |||
| #endif /* AVCODEC_DECODE_H */ | |||
| @@ -29,6 +29,7 @@ | |||
| #include "libavutil/time.h" | |||
| #include "avcodec.h" | |||
| #include "decode.h" | |||
| #include "dxva2_internal.h" | |||
| /* define all the GUIDs used directly here, | |||
| @@ -572,14 +573,20 @@ static void ff_dxva2_unlock(AVCodecContext *avctx) | |||
| #endif | |||
| } | |||
| // This must work before the decoder is created. | |||
| // This somehow needs to be exported to the user. | |||
| static void dxva_adjust_hwframes(AVCodecContext *avctx, AVHWFramesContext *frames_ctx) | |||
| int ff_dxva2_common_frame_params(AVCodecContext *avctx, | |||
| AVBufferRef *hw_frames_ctx) | |||
| { | |||
| FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx); | |||
| AVHWFramesContext *frames_ctx = (AVHWFramesContext *)hw_frames_ctx->data; | |||
| AVHWDeviceContext *device_ctx = frames_ctx->device_ctx; | |||
| int surface_alignment, num_surfaces; | |||
| frames_ctx->format = sctx->pix_fmt; | |||
| if (device_ctx->type == AV_HWDEVICE_TYPE_DXVA2) { | |||
| frames_ctx->format = AV_PIX_FMT_DXVA2_VLD; | |||
| } else if (device_ctx->type == AV_HWDEVICE_TYPE_D3D11VA) { | |||
| frames_ctx->format = AV_PIX_FMT_D3D11; | |||
| } else { | |||
| return AVERROR(EINVAL); | |||
| } | |||
| /* decoding MPEG-2 requires additional alignment on some Intel GPUs, | |||
| but it causes issues for H.264 on certain AMD GPUs..... */ | |||
| @@ -592,8 +599,8 @@ static void dxva_adjust_hwframes(AVCodecContext *avctx, AVHWFramesContext *frame | |||
| else | |||
| surface_alignment = 16; | |||
| /* 4 base work surfaces */ | |||
| num_surfaces = 4; | |||
| /* 1 base work surface */ | |||
| num_surfaces = 1; | |||
| /* add surfaces based on number of possible refs */ | |||
| if (avctx->codec_id == AV_CODEC_ID_H264 || avctx->codec_id == AV_CODEC_ID_HEVC) | |||
| @@ -627,12 +634,16 @@ static void dxva_adjust_hwframes(AVCodecContext *avctx, AVHWFramesContext *frame | |||
| frames_hwctx->BindFlags |= D3D11_BIND_DECODER; | |||
| } | |||
| #endif | |||
| return 0; | |||
| } | |||
| int ff_dxva2_decode_init(AVCodecContext *avctx) | |||
| { | |||
| FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx); | |||
| AVHWFramesContext *frames_ctx = NULL; | |||
| AVHWFramesContext *frames_ctx; | |||
| enum AVHWDeviceType dev_type = avctx->hwaccel->pix_fmt == AV_PIX_FMT_DXVA2_VLD | |||
| ? AV_HWDEVICE_TYPE_DXVA2 : AV_HWDEVICE_TYPE_D3D11VA; | |||
| int ret = 0; | |||
| // Old API. | |||
| @@ -642,32 +653,14 @@ int ff_dxva2_decode_init(AVCodecContext *avctx) | |||
| // (avctx->pix_fmt is not updated yet at this point) | |||
| sctx->pix_fmt = avctx->hwaccel->pix_fmt; | |||
| if (!avctx->hw_frames_ctx && !avctx->hw_device_ctx) { | |||
| av_log(avctx, AV_LOG_ERROR, "Either a hw_frames_ctx or a hw_device_ctx needs to be set for hardware decoding.\n"); | |||
| return AVERROR(EINVAL); | |||
| } | |||
| if (avctx->hw_frames_ctx) { | |||
| frames_ctx = (AVHWFramesContext*)avctx->hw_frames_ctx->data; | |||
| } else { | |||
| avctx->hw_frames_ctx = av_hwframe_ctx_alloc(avctx->hw_device_ctx); | |||
| if (!avctx->hw_frames_ctx) | |||
| return AVERROR(ENOMEM); | |||
| frames_ctx = (AVHWFramesContext*)avctx->hw_frames_ctx->data; | |||
| dxva_adjust_hwframes(avctx, frames_ctx); | |||
| ret = av_hwframe_ctx_init(avctx->hw_frames_ctx); | |||
| if (ret < 0) | |||
| goto fail; | |||
| } | |||
| ret = ff_decode_get_hw_frames_ctx(avctx, dev_type); | |||
| if (ret < 0) | |||
| return ret; | |||
| frames_ctx = (AVHWFramesContext*)avctx->hw_frames_ctx->data; | |||
| sctx->device_ctx = frames_ctx->device_ctx; | |||
| if (frames_ctx->format != sctx->pix_fmt || | |||
| !((sctx->pix_fmt == AV_PIX_FMT_D3D11 && CONFIG_D3D11VA) || | |||
| (sctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD && CONFIG_DXVA2))) { | |||
| if (frames_ctx->format != sctx->pix_fmt) { | |||
| av_log(avctx, AV_LOG_ERROR, "Invalid pixfmt for hwaccel!\n"); | |||
| ret = AVERROR(EINVAL); | |||
| goto fail; | |||
| @@ -523,6 +523,7 @@ AVHWAccel ff_h264_dxva2_hwaccel = { | |||
| .start_frame = dxva2_h264_start_frame, | |||
| .decode_slice = dxva2_h264_decode_slice, | |||
| .end_frame = dxva2_h264_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -539,6 +540,7 @@ AVHWAccel ff_h264_d3d11va_hwaccel = { | |||
| .start_frame = dxva2_h264_start_frame, | |||
| .decode_slice = dxva2_h264_decode_slice, | |||
| .end_frame = dxva2_h264_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -555,6 +557,7 @@ AVHWAccel ff_h264_d3d11va2_hwaccel = { | |||
| .start_frame = dxva2_h264_start_frame, | |||
| .decode_slice = dxva2_h264_decode_slice, | |||
| .end_frame = dxva2_h264_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -432,6 +432,7 @@ AVHWAccel ff_hevc_dxva2_hwaccel = { | |||
| .start_frame = dxva2_hevc_start_frame, | |||
| .decode_slice = dxva2_hevc_decode_slice, | |||
| .end_frame = dxva2_hevc_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct hevc_dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -448,6 +449,7 @@ AVHWAccel ff_hevc_d3d11va_hwaccel = { | |||
| .start_frame = dxva2_hevc_start_frame, | |||
| .decode_slice = dxva2_hevc_decode_slice, | |||
| .end_frame = dxva2_hevc_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct hevc_dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -464,6 +466,7 @@ AVHWAccel ff_hevc_d3d11va2_hwaccel = { | |||
| .start_frame = dxva2_hevc_start_frame, | |||
| .decode_slice = dxva2_hevc_decode_slice, | |||
| .end_frame = dxva2_hevc_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct hevc_dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -156,6 +156,9 @@ int ff_dxva2_decode_init(AVCodecContext *avctx); | |||
| int ff_dxva2_decode_uninit(AVCodecContext *avctx); | |||
| int ff_dxva2_common_frame_params(AVCodecContext *avctx, | |||
| AVBufferRef *hw_frames_ctx); | |||
| int ff_dxva2_is_d3d11(const AVCodecContext *avctx); | |||
| #endif /* AVCODEC_DXVA2_INTERNAL_H */ | |||
| @@ -328,6 +328,7 @@ AVHWAccel ff_mpeg2_dxva2_hwaccel = { | |||
| .start_frame = dxva2_mpeg2_start_frame, | |||
| .decode_slice = dxva2_mpeg2_decode_slice, | |||
| .end_frame = dxva2_mpeg2_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -344,6 +345,7 @@ AVHWAccel ff_mpeg2_d3d11va_hwaccel = { | |||
| .start_frame = dxva2_mpeg2_start_frame, | |||
| .decode_slice = dxva2_mpeg2_decode_slice, | |||
| .end_frame = dxva2_mpeg2_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -360,6 +362,7 @@ AVHWAccel ff_mpeg2_d3d11va2_hwaccel = { | |||
| .start_frame = dxva2_mpeg2_start_frame, | |||
| .decode_slice = dxva2_mpeg2_decode_slice, | |||
| .end_frame = dxva2_mpeg2_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -328,6 +328,7 @@ AVHWAccel ff_wmv3_dxva2_hwaccel = { | |||
| .start_frame = dxva2_vc1_start_frame, | |||
| .decode_slice = dxva2_vc1_decode_slice, | |||
| .end_frame = dxva2_vc1_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -344,6 +345,7 @@ AVHWAccel ff_vc1_dxva2_hwaccel = { | |||
| .start_frame = dxva2_vc1_start_frame, | |||
| .decode_slice = dxva2_vc1_decode_slice, | |||
| .end_frame = dxva2_vc1_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -360,6 +362,7 @@ AVHWAccel ff_wmv3_d3d11va_hwaccel = { | |||
| .start_frame = dxva2_vc1_start_frame, | |||
| .decode_slice = dxva2_vc1_decode_slice, | |||
| .end_frame = dxva2_vc1_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -376,6 +379,7 @@ AVHWAccel ff_wmv3_d3d11va2_hwaccel = { | |||
| .start_frame = dxva2_vc1_start_frame, | |||
| .decode_slice = dxva2_vc1_decode_slice, | |||
| .end_frame = dxva2_vc1_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -392,6 +396,7 @@ AVHWAccel ff_vc1_d3d11va_hwaccel = { | |||
| .start_frame = dxva2_vc1_start_frame, | |||
| .decode_slice = dxva2_vc1_decode_slice, | |||
| .end_frame = dxva2_vc1_end_frame, | |||
| .frame_params = ff_dxva2_common_frame_params, | |||
| .frame_priv_data_size = sizeof(struct dxva2_picture_context), | |||
| .priv_data_size = sizeof(FFDXVASharedContext), | |||
| }; | |||
| @@ -21,6 +21,7 @@ | |||
| #include "libavutil/pixdesc.h" | |||
| #include "avcodec.h" | |||
| #include "decode.h" | |||
| #include "internal.h" | |||
| #include "vaapi_decode.h" | |||
| @@ -270,10 +271,15 @@ static const struct { | |||
| #undef MAP | |||
| }; | |||
| static int vaapi_decode_make_config(AVCodecContext *avctx) | |||
| /* | |||
| * Set *va_config and the frames_ref fields from the current codec parameters | |||
| * in avctx. | |||
| */ | |||
| static int vaapi_decode_make_config(AVCodecContext *avctx, | |||
| AVBufferRef *device_ref, | |||
| VAConfigID *va_config, | |||
| AVBufferRef *frames_ref) | |||
| { | |||
| VAAPIDecodeContext *ctx = avctx->internal->hwaccel_priv_data; | |||
| AVVAAPIHWConfig *hwconfig = NULL; | |||
| AVHWFramesConstraints *constraints = NULL; | |||
| VAStatus vas; | |||
| @@ -283,13 +289,16 @@ static int vaapi_decode_make_config(AVCodecContext *avctx) | |||
| int profile_count, exact_match, alt_profile; | |||
| const AVPixFmtDescriptor *sw_desc, *desc; | |||
| AVHWDeviceContext *device = (AVHWDeviceContext*)device_ref->data; | |||
| AVVAAPIDeviceContext *hwctx = device->hwctx; | |||
| codec_desc = avcodec_descriptor_get(avctx->codec_id); | |||
| if (!codec_desc) { | |||
| err = AVERROR(EINVAL); | |||
| goto fail; | |||
| } | |||
| profile_count = vaMaxNumProfiles(ctx->hwctx->display); | |||
| profile_count = vaMaxNumProfiles(hwctx->display); | |||
| profile_list = av_malloc_array(profile_count, | |||
| sizeof(VAProfile)); | |||
| if (!profile_list) { | |||
| @@ -297,7 +306,7 @@ static int vaapi_decode_make_config(AVCodecContext *avctx) | |||
| goto fail; | |||
| } | |||
| vas = vaQueryConfigProfiles(ctx->hwctx->display, | |||
| vas = vaQueryConfigProfiles(hwctx->display, | |||
| profile_list, &profile_count); | |||
| if (vas != VA_STATUS_SUCCESS) { | |||
| av_log(avctx, AV_LOG_ERROR, "Failed to query profiles: " | |||
| @@ -355,12 +364,9 @@ static int vaapi_decode_make_config(AVCodecContext *avctx) | |||
| } | |||
| } | |||
| ctx->va_profile = profile; | |||
| ctx->va_entrypoint = VAEntrypointVLD; | |||
| vas = vaCreateConfig(ctx->hwctx->display, ctx->va_profile, | |||
| ctx->va_entrypoint, NULL, 0, | |||
| &ctx->va_config); | |||
| vas = vaCreateConfig(hwctx->display, profile, | |||
| VAEntrypointVLD, NULL, 0, | |||
| va_config); | |||
| if (vas != VA_STATUS_SUCCESS) { | |||
| av_log(avctx, AV_LOG_ERROR, "Failed to create decode " | |||
| "configuration: %d (%s).\n", vas, vaErrorStr(vas)); | |||
| @@ -368,20 +374,15 @@ static int vaapi_decode_make_config(AVCodecContext *avctx) | |||
| goto fail; | |||
| } | |||
| hwconfig = av_hwdevice_hwconfig_alloc(avctx->hw_device_ctx ? | |||
| avctx->hw_device_ctx : | |||
| ctx->frames->device_ref); | |||
| hwconfig = av_hwdevice_hwconfig_alloc(device_ref); | |||
| if (!hwconfig) { | |||
| err = AVERROR(ENOMEM); | |||
| goto fail; | |||
| } | |||
| hwconfig->config_id = ctx->va_config; | |||
| hwconfig->config_id = *va_config; | |||
| constraints = | |||
| av_hwdevice_get_hwframe_constraints(avctx->hw_device_ctx ? | |||
| avctx->hw_device_ctx : | |||
| ctx->frames->device_ref, | |||
| hwconfig); | |||
| av_hwdevice_get_hwframe_constraints(device_ref, hwconfig); | |||
| if (!constraints) { | |||
| err = AVERROR(ENOMEM); | |||
| goto fail; | |||
| @@ -407,48 +408,52 @@ static int vaapi_decode_make_config(AVCodecContext *avctx) | |||
| goto fail; | |||
| } | |||
| // Find the first format in the list which matches the expected | |||
| // bit depth and subsampling. If none are found (this can happen | |||
| // when 10-bit streams are decoded to 8-bit surfaces, for example) | |||
| // then just take the first format on the list. | |||
| ctx->surface_format = constraints->valid_sw_formats[0]; | |||
| sw_desc = av_pix_fmt_desc_get(avctx->sw_pix_fmt); | |||
| for (i = 0; constraints->valid_sw_formats[i] != AV_PIX_FMT_NONE; i++) { | |||
| desc = av_pix_fmt_desc_get(constraints->valid_sw_formats[i]); | |||
| if (desc->nb_components != sw_desc->nb_components || | |||
| desc->log2_chroma_w != sw_desc->log2_chroma_w || | |||
| desc->log2_chroma_h != sw_desc->log2_chroma_h) | |||
| continue; | |||
| for (j = 0; j < desc->nb_components; j++) { | |||
| if (desc->comp[j].depth != sw_desc->comp[j].depth) | |||
| break; | |||
| if (frames_ref) { | |||
| AVHWFramesContext *frames = (AVHWFramesContext *)frames_ref->data; | |||
| frames->format = AV_PIX_FMT_VAAPI; | |||
| frames->width = avctx->coded_width; | |||
| frames->height = avctx->coded_height; | |||
| // Find the first format in the list which matches the expected | |||
| // bit depth and subsampling. If none are found (this can happen | |||
| // when 10-bit streams are decoded to 8-bit surfaces, for example) | |||
| // then just take the first format on the list. | |||
| frames->sw_format = constraints->valid_sw_formats[0]; | |||
| sw_desc = av_pix_fmt_desc_get(avctx->sw_pix_fmt); | |||
| for (i = 0; constraints->valid_sw_formats[i] != AV_PIX_FMT_NONE; i++) { | |||
| desc = av_pix_fmt_desc_get(constraints->valid_sw_formats[i]); | |||
| if (desc->nb_components != sw_desc->nb_components || | |||
| desc->log2_chroma_w != sw_desc->log2_chroma_w || | |||
| desc->log2_chroma_h != sw_desc->log2_chroma_h) | |||
| continue; | |||
| for (j = 0; j < desc->nb_components; j++) { | |||
| if (desc->comp[j].depth != sw_desc->comp[j].depth) | |||
| break; | |||
| } | |||
| if (j < desc->nb_components) | |||
| continue; | |||
| frames->sw_format = constraints->valid_sw_formats[i]; | |||
| break; | |||
| } | |||
| if (j < desc->nb_components) | |||
| continue; | |||
| ctx->surface_format = constraints->valid_sw_formats[i]; | |||
| break; | |||
| } | |||
| // Start with at least four surfaces. | |||
| ctx->surface_count = 4; | |||
| // Add per-codec number of surfaces used for storing reference frames. | |||
| switch (avctx->codec_id) { | |||
| case AV_CODEC_ID_H264: | |||
| case AV_CODEC_ID_HEVC: | |||
| ctx->surface_count += 16; | |||
| break; | |||
| case AV_CODEC_ID_VP9: | |||
| ctx->surface_count += 8; | |||
| break; | |||
| case AV_CODEC_ID_VP8: | |||
| ctx->surface_count += 3; | |||
| break; | |||
| default: | |||
| ctx->surface_count += 2; | |||
| frames->initial_pool_size = 1; | |||
| // Add per-codec number of surfaces used for storing reference frames. | |||
| switch (avctx->codec_id) { | |||
| case AV_CODEC_ID_H264: | |||
| case AV_CODEC_ID_HEVC: | |||
| frames->initial_pool_size += 16; | |||
| break; | |||
| case AV_CODEC_ID_VP9: | |||
| frames->initial_pool_size += 8; | |||
| break; | |||
| case AV_CODEC_ID_VP8: | |||
| frames->initial_pool_size += 3; | |||
| break; | |||
| default: | |||
| frames->initial_pool_size += 2; | |||
| } | |||
| } | |||
| // Add an additional surface per thread is frame threading is enabled. | |||
| if (avctx->active_thread_type & FF_THREAD_FRAME) | |||
| ctx->surface_count += avctx->thread_count; | |||
| av_hwframe_constraints_free(&constraints); | |||
| av_freep(&hwconfig); | |||
| @@ -458,14 +463,38 @@ static int vaapi_decode_make_config(AVCodecContext *avctx) | |||
| fail: | |||
| av_hwframe_constraints_free(&constraints); | |||
| av_freep(&hwconfig); | |||
| if (ctx->va_config != VA_INVALID_ID) { | |||
| vaDestroyConfig(ctx->hwctx->display, ctx->va_config); | |||
| ctx->va_config = VA_INVALID_ID; | |||
| if (*va_config != VA_INVALID_ID) { | |||
| vaDestroyConfig(hwctx->display, *va_config); | |||
| *va_config = VA_INVALID_ID; | |||
| } | |||
| av_freep(&profile_list); | |||
| return err; | |||
| } | |||
| int ff_vaapi_common_frame_params(AVCodecContext *avctx, | |||
| AVBufferRef *hw_frames_ctx) | |||
| { | |||
| AVHWFramesContext *hw_frames = (AVHWFramesContext *)hw_frames_ctx->data; | |||
| AVHWDeviceContext *device_ctx = hw_frames->device_ctx; | |||
| AVVAAPIDeviceContext *hwctx; | |||
| VAConfigID va_config = VA_INVALID_ID; | |||
| int err; | |||
| if (device_ctx->type != AV_HWDEVICE_TYPE_VAAPI) | |||
| return AVERROR(EINVAL); | |||
| hwctx = device_ctx->hwctx; | |||
| err = vaapi_decode_make_config(avctx, hw_frames->device_ref, &va_config, | |||
| hw_frames_ctx); | |||
| if (err) | |||
| return err; | |||
| if (va_config != VA_INVALID_ID) | |||
| vaDestroyConfig(hwctx->display, va_config); | |||
| return 0; | |||
| } | |||
| int ff_vaapi_decode_init(AVCodecContext *avctx) | |||
| { | |||
| VAAPIDecodeContext *ctx = avctx->internal->hwaccel_priv_data; | |||
| @@ -502,36 +531,8 @@ int ff_vaapi_decode_init(AVCodecContext *avctx) | |||
| ctx->hwctx->driver_quirks = | |||
| AV_VAAPI_DRIVER_QUIRK_RENDER_PARAM_BUFFERS; | |||
| } else | |||
| #endif | |||
| if (avctx->hw_frames_ctx) { | |||
| // This structure has a shorter lifetime than the enclosing | |||
| // AVCodecContext, so we inherit the references from there | |||
| // and do not need to make separate ones. | |||
| ctx->frames = (AVHWFramesContext*)avctx->hw_frames_ctx->data; | |||
| ctx->hwfc = ctx->frames->hwctx; | |||
| ctx->device = ctx->frames->device_ctx; | |||
| ctx->hwctx = ctx->device->hwctx; | |||
| } else if (avctx->hw_device_ctx) { | |||
| ctx->device = (AVHWDeviceContext*)avctx->hw_device_ctx->data; | |||
| ctx->hwctx = ctx->device->hwctx; | |||
| if (ctx->device->type != AV_HWDEVICE_TYPE_VAAPI) { | |||
| av_log(avctx, AV_LOG_ERROR, "Device supplied for VAAPI " | |||
| "decoding must be a VAAPI device (not %d).\n", | |||
| ctx->device->type); | |||
| err = AVERROR(EINVAL); | |||
| goto fail; | |||
| } | |||
| } else { | |||
| av_log(avctx, AV_LOG_ERROR, "A hardware device or frames context " | |||
| "is required for VAAPI decoding.\n"); | |||
| err = AVERROR(EINVAL); | |||
| goto fail; | |||
| } | |||
| #endif | |||
| #if FF_API_VAAPI_CONTEXT | |||
| if (ctx->have_old_context) { | |||
| @@ -543,34 +544,19 @@ int ff_vaapi_decode_init(AVCodecContext *avctx) | |||
| } else { | |||
| #endif | |||
| err = vaapi_decode_make_config(avctx); | |||
| if (err) | |||
| err = ff_decode_get_hw_frames_ctx(avctx, AV_HWDEVICE_TYPE_VAAPI); | |||
| if (err < 0) | |||
| goto fail; | |||
| if (!avctx->hw_frames_ctx) { | |||
| avctx->hw_frames_ctx = av_hwframe_ctx_alloc(avctx->hw_device_ctx); | |||
| if (!avctx->hw_frames_ctx) { | |||
| err = AVERROR(ENOMEM); | |||
| goto fail; | |||
| } | |||
| ctx->frames = (AVHWFramesContext*)avctx->hw_frames_ctx->data; | |||
| ctx->frames->format = AV_PIX_FMT_VAAPI; | |||
| ctx->frames->width = avctx->coded_width; | |||
| ctx->frames->height = avctx->coded_height; | |||
| ctx->frames->sw_format = ctx->surface_format; | |||
| ctx->frames->initial_pool_size = ctx->surface_count; | |||
| ctx->frames = (AVHWFramesContext*)avctx->hw_frames_ctx->data; | |||
| ctx->hwfc = ctx->frames->hwctx; | |||
| ctx->device = ctx->frames->device_ctx; | |||
| ctx->hwctx = ctx->device->hwctx; | |||
| err = av_hwframe_ctx_init(avctx->hw_frames_ctx); | |||
| if (err < 0) { | |||
| av_log(avctx, AV_LOG_ERROR, "Failed to initialise internal " | |||
| "frames context: %d.\n", err); | |||
| goto fail; | |||
| } | |||
| ctx->hwfc = ctx->frames->hwctx; | |||
| } | |||
| err = vaapi_decode_make_config(avctx, ctx->frames->device_ref, | |||
| &ctx->va_config, avctx->hw_frames_ctx); | |||
| if (err) | |||
| goto fail; | |||
| vas = vaCreateContext(ctx->hwctx->display, ctx->va_config, | |||
| avctx->coded_width, avctx->coded_height, | |||
| @@ -53,8 +53,6 @@ typedef struct VAAPIDecodePicture { | |||
| } VAAPIDecodePicture; | |||
| typedef struct VAAPIDecodeContext { | |||
| VAProfile va_profile; | |||
| VAEntrypoint va_entrypoint; | |||
| VAConfigID va_config; | |||
| VAContextID va_context; | |||
| @@ -96,4 +94,7 @@ int ff_vaapi_decode_cancel(AVCodecContext *avctx, | |||
| int ff_vaapi_decode_init(AVCodecContext *avctx); | |||
| int ff_vaapi_decode_uninit(AVCodecContext *avctx); | |||
| int ff_vaapi_common_frame_params(AVCodecContext *avctx, | |||
| AVBufferRef *hw_frames_ctx); | |||
| #endif /* AVCODEC_VAAPI_DECODE_H */ | |||
| @@ -399,6 +399,7 @@ AVHWAccel ff_h264_vaapi_hwaccel = { | |||
| .frame_priv_data_size = sizeof(VAAPIDecodePicture), | |||
| .init = &ff_vaapi_decode_init, | |||
| .uninit = &ff_vaapi_decode_uninit, | |||
| .frame_params = &ff_vaapi_common_frame_params, | |||
| .priv_data_size = sizeof(VAAPIDecodeContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -434,6 +434,7 @@ AVHWAccel ff_hevc_vaapi_hwaccel = { | |||
| .frame_priv_data_size = sizeof(VAAPIDecodePictureHEVC), | |||
| .init = ff_vaapi_decode_init, | |||
| .uninit = ff_vaapi_decode_uninit, | |||
| .frame_params = ff_vaapi_common_frame_params, | |||
| .priv_data_size = sizeof(VAAPIDecodeContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -184,6 +184,7 @@ AVHWAccel ff_mpeg2_vaapi_hwaccel = { | |||
| .frame_priv_data_size = sizeof(VAAPIDecodePicture), | |||
| .init = &ff_vaapi_decode_init, | |||
| .uninit = &ff_vaapi_decode_uninit, | |||
| .frame_params = &ff_vaapi_common_frame_params, | |||
| .priv_data_size = sizeof(VAAPIDecodeContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -200,6 +200,7 @@ AVHWAccel ff_mpeg4_vaapi_hwaccel = { | |||
| .frame_priv_data_size = sizeof(VAAPIDecodePicture), | |||
| .init = &ff_vaapi_decode_init, | |||
| .uninit = &ff_vaapi_decode_uninit, | |||
| .frame_params = &ff_vaapi_common_frame_params, | |||
| .priv_data_size = sizeof(VAAPIDecodeContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -217,6 +218,7 @@ AVHWAccel ff_h263_vaapi_hwaccel = { | |||
| .frame_priv_data_size = sizeof(VAAPIDecodePicture), | |||
| .init = &ff_vaapi_decode_init, | |||
| .uninit = &ff_vaapi_decode_uninit, | |||
| .frame_params = &ff_vaapi_common_frame_params, | |||
| .priv_data_size = sizeof(VAAPIDecodeContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -399,6 +399,7 @@ AVHWAccel ff_wmv3_vaapi_hwaccel = { | |||
| .frame_priv_data_size = sizeof(VAAPIDecodePicture), | |||
| .init = &ff_vaapi_decode_init, | |||
| .uninit = &ff_vaapi_decode_uninit, | |||
| .frame_params = &ff_vaapi_common_frame_params, | |||
| .priv_data_size = sizeof(VAAPIDecodeContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -415,6 +416,7 @@ AVHWAccel ff_vc1_vaapi_hwaccel = { | |||
| .frame_priv_data_size = sizeof(VAAPIDecodePicture), | |||
| .init = &ff_vaapi_decode_init, | |||
| .uninit = &ff_vaapi_decode_uninit, | |||
| .frame_params = &ff_vaapi_common_frame_params, | |||
| .priv_data_size = sizeof(VAAPIDecodeContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -232,5 +232,6 @@ AVHWAccel ff_vp8_vaapi_hwaccel = { | |||
| .init = &ff_vaapi_decode_init, | |||
| .uninit = &ff_vaapi_decode_uninit, | |||
| .priv_data_size = sizeof(VAAPIDecodeContext), | |||
| .frame_params = &ff_vaapi_common_frame_params, | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -24,6 +24,7 @@ | |||
| #include <limits.h> | |||
| #include "avcodec.h" | |||
| #include "decode.h" | |||
| #include "internal.h" | |||
| #include "h264dec.h" | |||
| #include "vc1.h" | |||
| @@ -100,6 +101,25 @@ int av_vdpau_get_surface_parameters(AVCodecContext *avctx, | |||
| return 0; | |||
| } | |||
| int ff_vdpau_common_frame_params(AVCodecContext *avctx, | |||
| AVBufferRef *hw_frames_ctx) | |||
| { | |||
| AVHWFramesContext *hw_frames = (AVHWFramesContext*)hw_frames_ctx->data; | |||
| VdpChromaType type; | |||
| uint32_t width; | |||
| uint32_t height; | |||
| if (av_vdpau_get_surface_parameters(avctx, &type, &width, &height)) | |||
| return AVERROR(EINVAL); | |||
| hw_frames->format = AV_PIX_FMT_VDPAU; | |||
| hw_frames->sw_format = avctx->sw_pix_fmt; | |||
| hw_frames->width = width; | |||
| hw_frames->height = height; | |||
| return 0; | |||
| } | |||
| int ff_vdpau_common_init(AVCodecContext *avctx, VdpDecoderProfile profile, | |||
| int level) | |||
| { | |||
| @@ -115,6 +135,7 @@ int ff_vdpau_common_init(AVCodecContext *avctx, VdpDecoderProfile profile, | |||
| VdpChromaType type; | |||
| uint32_t width; | |||
| uint32_t height; | |||
| int ret; | |||
| vdctx->width = UINT32_MAX; | |||
| vdctx->height = UINT32_MAX; | |||
| @@ -142,41 +163,14 @@ int ff_vdpau_common_init(AVCodecContext *avctx, VdpDecoderProfile profile, | |||
| type != VDP_CHROMA_TYPE_420) | |||
| return AVERROR(ENOSYS); | |||
| } else { | |||
| AVHWFramesContext *frames_ctx = NULL; | |||
| AVHWFramesContext *frames_ctx; | |||
| AVVDPAUDeviceContext *dev_ctx; | |||
| // We assume the hw_frames_ctx always survives until ff_vdpau_common_uninit | |||
| // is called. This holds true as the user is not allowed to touch | |||
| // hw_device_ctx, or hw_frames_ctx after get_format (and ff_get_format | |||
| // itself also uninits before unreffing hw_frames_ctx). | |||
| if (avctx->hw_frames_ctx) { | |||
| frames_ctx = (AVHWFramesContext*)avctx->hw_frames_ctx->data; | |||
| } else if (avctx->hw_device_ctx) { | |||
| int ret; | |||
| avctx->hw_frames_ctx = av_hwframe_ctx_alloc(avctx->hw_device_ctx); | |||
| if (!avctx->hw_frames_ctx) | |||
| return AVERROR(ENOMEM); | |||
| frames_ctx = (AVHWFramesContext*)avctx->hw_frames_ctx->data; | |||
| frames_ctx->format = AV_PIX_FMT_VDPAU; | |||
| frames_ctx->sw_format = avctx->sw_pix_fmt; | |||
| frames_ctx->width = avctx->coded_width; | |||
| frames_ctx->height = avctx->coded_height; | |||
| ret = av_hwframe_ctx_init(avctx->hw_frames_ctx); | |||
| if (ret < 0) { | |||
| av_buffer_unref(&avctx->hw_frames_ctx); | |||
| return ret; | |||
| } | |||
| } | |||
| if (!frames_ctx) { | |||
| av_log(avctx, AV_LOG_ERROR, "A hardware frames context is " | |||
| "required for VDPAU decoding.\n"); | |||
| return AVERROR(EINVAL); | |||
| } | |||
| ret = ff_decode_get_hw_frames_ctx(avctx, AV_HWDEVICE_TYPE_VDPAU); | |||
| if (ret < 0) | |||
| return ret; | |||
| frames_ctx = (AVHWFramesContext*)avctx->hw_frames_ctx->data; | |||
| dev_ctx = frames_ctx->device_ctx->hwctx; | |||
| vdctx->device = dev_ctx->device; | |||
| @@ -273,6 +273,7 @@ AVHWAccel ff_h264_vdpau_hwaccel = { | |||
| .frame_priv_data_size = sizeof(struct vdpau_picture_context), | |||
| .init = vdpau_h264_init, | |||
| .uninit = ff_vdpau_common_uninit, | |||
| .frame_params = ff_vdpau_common_frame_params, | |||
| .priv_data_size = sizeof(VDPAUContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -424,6 +424,7 @@ AVHWAccel ff_hevc_vdpau_hwaccel = { | |||
| .frame_priv_data_size = sizeof(struct vdpau_picture_context), | |||
| .init = vdpau_hevc_init, | |||
| .uninit = ff_vdpau_common_uninit, | |||
| .frame_params = ff_vdpau_common_frame_params, | |||
| .priv_data_size = sizeof(VDPAUContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -119,5 +119,7 @@ int ff_vdpau_common_end_frame(AVCodecContext *avctx, AVFrame *frame, | |||
| int ff_vdpau_mpeg_end_frame(AVCodecContext *avctx); | |||
| int ff_vdpau_add_buffer(struct vdpau_picture_context *pic, const uint8_t *buf, | |||
| uint32_t buf_size); | |||
| int ff_vdpau_common_frame_params(AVCodecContext *avctx, | |||
| AVBufferRef *hw_frames_ctx); | |||
| #endif /* AVCODEC_VDPAU_INTERNAL_H */ | |||
| @@ -149,6 +149,7 @@ AVHWAccel ff_mpeg2_vdpau_hwaccel = { | |||
| .frame_priv_data_size = sizeof(struct vdpau_picture_context), | |||
| .init = vdpau_mpeg2_init, | |||
| .uninit = ff_vdpau_common_uninit, | |||
| .frame_params = ff_vdpau_common_frame_params, | |||
| .priv_data_size = sizeof(VDPAUContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -118,6 +118,7 @@ AVHWAccel ff_mpeg4_vdpau_hwaccel = { | |||
| .frame_priv_data_size = sizeof(struct vdpau_picture_context), | |||
| .init = vdpau_mpeg4_init, | |||
| .uninit = ff_vdpau_common_uninit, | |||
| .frame_params = ff_vdpau_common_frame_params, | |||
| .priv_data_size = sizeof(VDPAUContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -143,6 +143,7 @@ AVHWAccel ff_wmv3_vdpau_hwaccel = { | |||
| .frame_priv_data_size = sizeof(struct vdpau_picture_context), | |||
| .init = vdpau_vc1_init, | |||
| .uninit = ff_vdpau_common_uninit, | |||
| .frame_params = ff_vdpau_common_frame_params, | |||
| .priv_data_size = sizeof(VDPAUContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -159,6 +160,7 @@ AVHWAccel ff_vc1_vdpau_hwaccel = { | |||
| .frame_priv_data_size = sizeof(struct vdpau_picture_context), | |||
| .init = vdpau_vc1_init, | |||
| .uninit = ff_vdpau_common_uninit, | |||
| .frame_params = ff_vdpau_common_frame_params, | |||
| .priv_data_size = sizeof(VDPAUContext), | |||
| .caps_internal = HWACCEL_CAP_ASYNC_SAFE, | |||
| }; | |||
| @@ -28,7 +28,7 @@ | |||
| #include "libavutil/version.h" | |||
| #define LIBAVCODEC_VERSION_MAJOR 58 | |||
| #define LIBAVCODEC_VERSION_MINOR 4 | |||
| #define LIBAVCODEC_VERSION_MINOR 5 | |||
| #define LIBAVCODEC_VERSION_MICRO 0 | |||
| #define LIBAVCODEC_VERSION_INT AV_VERSION_INT(LIBAVCODEC_VERSION_MAJOR, \ | |||