mirror of
https://git.ffmpeg.org/ffmpeg.git
synced 2025-04-07 09:48:29 +00:00
lavc: vdpau: add support for new hw_frames_ctx and hw_device_ctx API
This supports retrieving the device from a provided hw_frames_ctx, and automatically creating a hw_frames_ctx if hw_device_ctx is set. The old API is not deprecated yet. The user can still use av_vdpau_bind_context() (with or without setting hw_frames_ctx), or use the API before that by allocating and setting hwaccel_context manually.
This commit is contained in:
parent
16a163b55a
commit
1a7ddba576
libavcodec
@ -118,29 +118,76 @@ int ff_vdpau_common_init(AVCodecContext *avctx, VdpDecoderProfile profile,
|
|||||||
|
|
||||||
vdctx->width = UINT32_MAX;
|
vdctx->width = UINT32_MAX;
|
||||||
vdctx->height = UINT32_MAX;
|
vdctx->height = UINT32_MAX;
|
||||||
hwctx->reset = 0;
|
|
||||||
|
|
||||||
if (hwctx->context.decoder != VDP_INVALID_HANDLE) {
|
|
||||||
vdctx->decoder = hwctx->context.decoder;
|
|
||||||
vdctx->render = hwctx->context.render;
|
|
||||||
vdctx->device = VDP_INVALID_HANDLE;
|
|
||||||
return 0; /* Decoder created by user */
|
|
||||||
}
|
|
||||||
|
|
||||||
vdctx->device = hwctx->device;
|
|
||||||
vdctx->get_proc_address = hwctx->get_proc_address;
|
|
||||||
|
|
||||||
if (hwctx->flags & AV_HWACCEL_FLAG_IGNORE_LEVEL)
|
|
||||||
level = 0;
|
|
||||||
else if (level < 0)
|
|
||||||
return AVERROR(ENOTSUP);
|
|
||||||
|
|
||||||
if (av_vdpau_get_surface_parameters(avctx, &type, &width, &height))
|
if (av_vdpau_get_surface_parameters(avctx, &type, &width, &height))
|
||||||
return AVERROR(ENOSYS);
|
return AVERROR(ENOSYS);
|
||||||
|
|
||||||
if (!(hwctx->flags & AV_HWACCEL_FLAG_ALLOW_HIGH_DEPTH) &&
|
if (hwctx) {
|
||||||
type != VDP_CHROMA_TYPE_420)
|
hwctx->reset = 0;
|
||||||
return AVERROR(ENOSYS);
|
|
||||||
|
if (hwctx->context.decoder != VDP_INVALID_HANDLE) {
|
||||||
|
vdctx->decoder = hwctx->context.decoder;
|
||||||
|
vdctx->render = hwctx->context.render;
|
||||||
|
vdctx->device = VDP_INVALID_HANDLE;
|
||||||
|
return 0; /* Decoder created by user */
|
||||||
|
}
|
||||||
|
|
||||||
|
vdctx->device = hwctx->device;
|
||||||
|
vdctx->get_proc_address = hwctx->get_proc_address;
|
||||||
|
|
||||||
|
if (hwctx->flags & AV_HWACCEL_FLAG_IGNORE_LEVEL)
|
||||||
|
level = 0;
|
||||||
|
|
||||||
|
if (!(hwctx->flags & AV_HWACCEL_FLAG_ALLOW_HIGH_DEPTH) &&
|
||||||
|
type != VDP_CHROMA_TYPE_420)
|
||||||
|
return AVERROR(ENOSYS);
|
||||||
|
} else {
|
||||||
|
AVHWFramesContext *frames_ctx = NULL;
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
dev_ctx = frames_ctx->device_ctx->hwctx;
|
||||||
|
|
||||||
|
vdctx->device = dev_ctx->device;
|
||||||
|
vdctx->get_proc_address = dev_ctx->get_proc_address;
|
||||||
|
|
||||||
|
if (avctx->hwaccel_flags & AV_HWACCEL_FLAG_IGNORE_LEVEL)
|
||||||
|
level = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (level < 0)
|
||||||
|
return AVERROR(ENOTSUP);
|
||||||
|
|
||||||
status = vdctx->get_proc_address(vdctx->device,
|
status = vdctx->get_proc_address(vdctx->device,
|
||||||
VDP_FUNC_ID_VIDEO_SURFACE_QUERY_CAPABILITIES,
|
VDP_FUNC_ID_VIDEO_SURFACE_QUERY_CAPABILITIES,
|
||||||
@ -238,7 +285,7 @@ static int ff_vdpau_common_reinit(AVCodecContext *avctx)
|
|||||||
if (vdctx->device == VDP_INVALID_HANDLE)
|
if (vdctx->device == VDP_INVALID_HANDLE)
|
||||||
return 0; /* Decoder created by user */
|
return 0; /* Decoder created by user */
|
||||||
if (avctx->coded_width == vdctx->width &&
|
if (avctx->coded_width == vdctx->width &&
|
||||||
avctx->coded_height == vdctx->height && !hwctx->reset)
|
avctx->coded_height == vdctx->height && (!hwctx || !hwctx->reset))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
avctx->hwaccel->uninit(avctx);
|
avctx->hwaccel->uninit(avctx);
|
||||||
|
@ -28,6 +28,8 @@
|
|||||||
#include <vdpau/vdpau.h>
|
#include <vdpau/vdpau.h>
|
||||||
|
|
||||||
#include "libavutil/frame.h"
|
#include "libavutil/frame.h"
|
||||||
|
#include "libavutil/hwcontext.h"
|
||||||
|
#include "libavutil/hwcontext_vdpau.h"
|
||||||
|
|
||||||
#include "avcodec.h"
|
#include "avcodec.h"
|
||||||
#include "vdpau.h"
|
#include "vdpau.h"
|
||||||
|
@ -29,7 +29,7 @@
|
|||||||
|
|
||||||
#define LIBAVCODEC_VERSION_MAJOR 57
|
#define LIBAVCODEC_VERSION_MAJOR 57
|
||||||
#define LIBAVCODEC_VERSION_MINOR 37
|
#define LIBAVCODEC_VERSION_MINOR 37
|
||||||
#define LIBAVCODEC_VERSION_MICRO 0
|
#define LIBAVCODEC_VERSION_MICRO 1
|
||||||
|
|
||||||
#define LIBAVCODEC_VERSION_INT AV_VERSION_INT(LIBAVCODEC_VERSION_MAJOR, \
|
#define LIBAVCODEC_VERSION_INT AV_VERSION_INT(LIBAVCODEC_VERSION_MAJOR, \
|
||||||
LIBAVCODEC_VERSION_MINOR, \
|
LIBAVCODEC_VERSION_MINOR, \
|
||||||
|
Loading…
Reference in New Issue
Block a user