lavc/vaapi_encode: fix propagating durations and opaques

input_image is freed by the time the output packet is constructed, so we
need to store copies in VAAPIEncodePicture.
release/6.0
Anton Khirnov 2 years ago committed by Haihao Xiang
parent f4b4e16641
commit 7d49fef8b4
  1. 22
      libavcodec/vaapi_encode.c
  2. 4
      libavcodec/vaapi_encode.h

@ -695,7 +695,7 @@ static int vaapi_encode_output(AVCodecContext *avctx,
pkt->flags |= AV_PKT_FLAG_KEY; pkt->flags |= AV_PKT_FLAG_KEY;
pkt->pts = pic->pts; pkt->pts = pic->pts;
pkt->duration = pic->input_image->duration; pkt->duration = pic->duration;
vas = vaUnmapBuffer(ctx->hwctx->display, pic->output_buffer); vas = vaUnmapBuffer(ctx->hwctx->display, pic->output_buffer);
if (vas != VA_STATUS_SUCCESS) { if (vas != VA_STATUS_SUCCESS) {
@ -706,10 +706,11 @@ static int vaapi_encode_output(AVCodecContext *avctx,
} }
// for no-delay encoders this is handled in generic codec // for no-delay encoders this is handled in generic codec
if (avctx->codec->capabilities & AV_CODEC_CAP_DELAY) { if (avctx->codec->capabilities & AV_CODEC_CAP_DELAY &&
err = ff_encode_reordered_opaque(avctx, pkt, pic->input_image); avctx->flags & AV_CODEC_FLAG_COPY_OPAQUE) {
if (err < 0) pkt->opaque = pic->opaque;
goto fail; pkt->opaque_ref = pic->opaque_ref;
pic->opaque_ref = NULL;
} }
av_buffer_unref(&pic->output_buffer_ref); av_buffer_unref(&pic->output_buffer_ref);
@ -785,6 +786,8 @@ static int vaapi_encode_free(AVCodecContext *avctx,
av_frame_free(&pic->input_image); av_frame_free(&pic->input_image);
av_frame_free(&pic->recon_image); av_frame_free(&pic->recon_image);
av_buffer_unref(&pic->opaque_ref);
av_freep(&pic->param_buffers); av_freep(&pic->param_buffers);
av_freep(&pic->slices); av_freep(&pic->slices);
// Output buffer should already be destroyed. // Output buffer should already be destroyed.
@ -1152,6 +1155,15 @@ static int vaapi_encode_send_frame(AVCodecContext *avctx, AVFrame *frame)
pic->input_surface = (VASurfaceID)(uintptr_t)frame->data[3]; pic->input_surface = (VASurfaceID)(uintptr_t)frame->data[3];
pic->pts = frame->pts; pic->pts = frame->pts;
pic->duration = frame->duration;
if (avctx->flags & AV_CODEC_FLAG_COPY_OPAQUE) {
err = av_buffer_replace(&pic->opaque_ref, frame->opaque_ref);
if (err < 0)
goto fail;
pic->opaque = frame->opaque;
}
av_frame_move_ref(pic->input_image, frame); av_frame_move_ref(pic->input_image, frame);

@ -75,8 +75,12 @@ typedef struct VAAPIEncodePicture {
int64_t display_order; int64_t display_order;
int64_t encode_order; int64_t encode_order;
int64_t pts; int64_t pts;
int64_t duration;
int force_idr; int force_idr;
void *opaque;
AVBufferRef *opaque_ref;
#if VA_CHECK_VERSION(1, 0, 0) #if VA_CHECK_VERSION(1, 0, 0)
// ROI regions. // ROI regions.
VAEncROI *roi; VAEncROI *roi;

Loading…
Cancel
Save