You've already forked FFmpeg
mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-08-10 06:10:52 +02:00
lavc/nvenc: handle frame durations and AV_CODEC_CAP_ENCODER_REORDERED_OPAQUE
This commit is contained in:
@@ -28,6 +28,7 @@
|
|||||||
#include "av1.h"
|
#include "av1.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include "libavutil/buffer.h"
|
||||||
#include "libavutil/hwcontext_cuda.h"
|
#include "libavutil/hwcontext_cuda.h"
|
||||||
#include "libavutil/hwcontext.h"
|
#include "libavutil/hwcontext.h"
|
||||||
#include "libavutil/cuda_check.h"
|
#include "libavutil/cuda_check.h"
|
||||||
@@ -162,6 +163,23 @@ static int nvenc_print_error(AVCodecContext *avctx, NVENCSTATUS err,
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typedef struct FrameData {
|
||||||
|
int64_t pts;
|
||||||
|
int64_t duration;
|
||||||
|
int64_t reordered_opaque;
|
||||||
|
|
||||||
|
void *frame_opaque;
|
||||||
|
AVBufferRef *frame_opaque_ref;
|
||||||
|
} FrameData;
|
||||||
|
|
||||||
|
static void reorder_queue_flush(AVFifo *queue)
|
||||||
|
{
|
||||||
|
FrameData fd;
|
||||||
|
|
||||||
|
while (av_fifo_read(queue, &fd, 1) >= 0)
|
||||||
|
av_buffer_unref(&fd.frame_opaque_ref);
|
||||||
|
}
|
||||||
|
|
||||||
typedef struct GUIDTuple {
|
typedef struct GUIDTuple {
|
||||||
const GUID guid;
|
const GUID guid;
|
||||||
int flags;
|
int flags;
|
||||||
@@ -1748,8 +1766,8 @@ static av_cold int nvenc_setup_surfaces(AVCodecContext *avctx)
|
|||||||
if (!ctx->surfaces)
|
if (!ctx->surfaces)
|
||||||
return AVERROR(ENOMEM);
|
return AVERROR(ENOMEM);
|
||||||
|
|
||||||
ctx->timestamp_list = av_fifo_alloc2(ctx->nb_surfaces, sizeof(int64_t), 0);
|
ctx->reorder_queue = av_fifo_alloc2(ctx->nb_surfaces, sizeof(FrameData), 0);
|
||||||
if (!ctx->timestamp_list)
|
if (!ctx->reorder_queue)
|
||||||
return AVERROR(ENOMEM);
|
return AVERROR(ENOMEM);
|
||||||
|
|
||||||
ctx->unused_surface_queue = av_fifo_alloc2(ctx->nb_surfaces, sizeof(NvencSurface*), 0);
|
ctx->unused_surface_queue = av_fifo_alloc2(ctx->nb_surfaces, sizeof(NvencSurface*), 0);
|
||||||
@@ -1833,7 +1851,8 @@ av_cold int ff_nvenc_encode_close(AVCodecContext *avctx)
|
|||||||
p_nvenc->nvEncEncodePicture(ctx->nvencoder, ¶ms);
|
p_nvenc->nvEncEncodePicture(ctx->nvencoder, ¶ms);
|
||||||
}
|
}
|
||||||
|
|
||||||
av_fifo_freep2(&ctx->timestamp_list);
|
reorder_queue_flush(ctx->reorder_queue);
|
||||||
|
av_fifo_freep2(&ctx->reorder_queue);
|
||||||
av_fifo_freep2(&ctx->output_surface_ready_queue);
|
av_fifo_freep2(&ctx->output_surface_ready_queue);
|
||||||
av_fifo_freep2(&ctx->output_surface_queue);
|
av_fifo_freep2(&ctx->output_surface_queue);
|
||||||
av_fifo_freep2(&ctx->unused_surface_queue);
|
av_fifo_freep2(&ctx->unused_surface_queue);
|
||||||
@@ -2177,18 +2196,45 @@ static void nvenc_codec_specific_pic_params(AVCodecContext *avctx,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void timestamp_queue_enqueue(AVFifo *queue, int64_t timestamp)
|
static void reorder_queue_enqueue(AVFifo *queue, const AVCodecContext *avctx,
|
||||||
|
const AVFrame *frame, AVBufferRef **opaque_ref)
|
||||||
{
|
{
|
||||||
av_fifo_write(queue, ×tamp, 1);
|
FrameData fd;
|
||||||
|
|
||||||
|
fd.pts = frame->pts;
|
||||||
|
fd.duration = frame->duration;
|
||||||
|
fd.reordered_opaque = frame->reordered_opaque;
|
||||||
|
fd.frame_opaque = frame->opaque;
|
||||||
|
fd.frame_opaque_ref = *opaque_ref;
|
||||||
|
|
||||||
|
*opaque_ref = NULL;
|
||||||
|
|
||||||
|
av_fifo_write(queue, &fd, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int64_t timestamp_queue_dequeue(AVFifo *queue)
|
static int64_t reorder_queue_dequeue(AVFifo *queue, AVCodecContext *avctx,
|
||||||
|
AVPacket *pkt)
|
||||||
{
|
{
|
||||||
int64_t timestamp = AV_NOPTS_VALUE;
|
FrameData fd;
|
||||||
// The following call might fail if the queue is empty.
|
|
||||||
av_fifo_read(queue, ×tamp, 1);
|
|
||||||
|
|
||||||
return timestamp;
|
// The following call might fail if the queue is empty.
|
||||||
|
if (av_fifo_read(queue, &fd, 1) < 0)
|
||||||
|
return AV_NOPTS_VALUE;
|
||||||
|
|
||||||
|
if (pkt) {
|
||||||
|
avctx->reordered_opaque = fd.reordered_opaque;
|
||||||
|
pkt->duration = fd.duration;
|
||||||
|
|
||||||
|
if (avctx->flags & AV_CODEC_FLAG_COPY_OPAQUE) {
|
||||||
|
pkt->opaque = fd.frame_opaque;
|
||||||
|
pkt->opaque_ref = fd.frame_opaque_ref;
|
||||||
|
fd.frame_opaque_ref = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
av_buffer_unref(&fd.frame_opaque_ref);
|
||||||
|
|
||||||
|
return fd.pts;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int nvenc_set_timestamp(AVCodecContext *avctx,
|
static int nvenc_set_timestamp(AVCodecContext *avctx,
|
||||||
@@ -2196,12 +2242,14 @@ static int nvenc_set_timestamp(AVCodecContext *avctx,
|
|||||||
AVPacket *pkt)
|
AVPacket *pkt)
|
||||||
{
|
{
|
||||||
NvencContext *ctx = avctx->priv_data;
|
NvencContext *ctx = avctx->priv_data;
|
||||||
|
int64_t dts;
|
||||||
|
|
||||||
pkt->pts = params->outputTimeStamp;
|
pkt->pts = params->outputTimeStamp;
|
||||||
|
|
||||||
|
dts = reorder_queue_dequeue(ctx->reorder_queue, avctx, pkt);
|
||||||
|
|
||||||
if (avctx->codec_descriptor->props & AV_CODEC_PROP_REORDER) {
|
if (avctx->codec_descriptor->props & AV_CODEC_PROP_REORDER) {
|
||||||
pkt->dts = timestamp_queue_dequeue(ctx->timestamp_list);
|
pkt->dts = dts - FFMAX(ctx->encode_config.frameIntervalP - 1, 0) * FFMAX(avctx->ticks_per_frame, 1);
|
||||||
pkt->dts -= FFMAX(ctx->encode_config.frameIntervalP - 1, 0) * FFMAX(avctx->ticks_per_frame, 1);
|
|
||||||
} else {
|
} else {
|
||||||
pkt->dts = pkt->pts;
|
pkt->dts = pkt->pts;
|
||||||
}
|
}
|
||||||
@@ -2298,7 +2346,7 @@ static int process_output_surface(AVCodecContext *avctx, AVPacket *pkt, NvencSur
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
error:
|
error:
|
||||||
timestamp_queue_dequeue(ctx->timestamp_list);
|
reorder_queue_dequeue(ctx->reorder_queue, avctx, NULL);
|
||||||
|
|
||||||
error2:
|
error2:
|
||||||
return res;
|
return res;
|
||||||
@@ -2528,6 +2576,8 @@ static int nvenc_send_frame(AVCodecContext *avctx, const AVFrame *frame)
|
|||||||
int sei_count = 0;
|
int sei_count = 0;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
AVBufferRef *opaque_ref = NULL;
|
||||||
|
|
||||||
NvencContext *ctx = avctx->priv_data;
|
NvencContext *ctx = avctx->priv_data;
|
||||||
NvencDynLoadFunctions *dl_fn = &ctx->nvenc_dload_funcs;
|
NvencDynLoadFunctions *dl_fn = &ctx->nvenc_dload_funcs;
|
||||||
NV_ENCODE_API_FUNCTION_LIST *p_nvenc = &dl_fn->nvenc_funcs;
|
NV_ENCODE_API_FUNCTION_LIST *p_nvenc = &dl_fn->nvenc_funcs;
|
||||||
@@ -2595,9 +2645,17 @@ static int nvenc_send_frame(AVCodecContext *avctx, const AVFrame *frame)
|
|||||||
pic_params.encodePicFlags = NV_ENC_PIC_FLAG_EOS;
|
pic_params.encodePicFlags = NV_ENC_PIC_FLAG_EOS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// make a reference for enqueing in the reorder queue here,
|
||||||
|
// so that reorder_queue_enqueue() cannot fail
|
||||||
|
if (frame && frame->opaque_ref && avctx->flags & AV_CODEC_FLAG_COPY_OPAQUE) {
|
||||||
|
opaque_ref = av_buffer_ref(frame->opaque_ref);
|
||||||
|
if (!opaque_ref)
|
||||||
|
return AVERROR(ENOMEM);
|
||||||
|
}
|
||||||
|
|
||||||
res = nvenc_push_context(avctx);
|
res = nvenc_push_context(avctx);
|
||||||
if (res < 0)
|
if (res < 0)
|
||||||
return res;
|
goto opaque_ref_fail;
|
||||||
|
|
||||||
nv_status = p_nvenc->nvEncEncodePicture(ctx->nvencoder, &pic_params);
|
nv_status = p_nvenc->nvEncEncodePicture(ctx->nvencoder, &pic_params);
|
||||||
|
|
||||||
@@ -2606,17 +2664,17 @@ static int nvenc_send_frame(AVCodecContext *avctx, const AVFrame *frame)
|
|||||||
|
|
||||||
res = nvenc_pop_context(avctx);
|
res = nvenc_pop_context(avctx);
|
||||||
if (res < 0)
|
if (res < 0)
|
||||||
return res;
|
goto opaque_ref_fail;
|
||||||
|
|
||||||
if (nv_status != NV_ENC_SUCCESS &&
|
if (nv_status != NV_ENC_SUCCESS &&
|
||||||
nv_status != NV_ENC_ERR_NEED_MORE_INPUT)
|
nv_status != NV_ENC_ERR_NEED_MORE_INPUT) {
|
||||||
return nvenc_print_error(avctx, nv_status, "EncodePicture failed!");
|
res = nvenc_print_error(avctx, nv_status, "EncodePicture failed!");
|
||||||
|
goto opaque_ref_fail;
|
||||||
|
}
|
||||||
|
|
||||||
if (frame && frame->buf[0]) {
|
if (frame && frame->buf[0]) {
|
||||||
av_fifo_write(ctx->output_surface_queue, &in_surf, 1);
|
av_fifo_write(ctx->output_surface_queue, &in_surf, 1);
|
||||||
|
reorder_queue_enqueue(ctx->reorder_queue, avctx, frame, &opaque_ref);
|
||||||
if (avctx->codec_descriptor->props & AV_CODEC_PROP_REORDER)
|
|
||||||
timestamp_queue_enqueue(ctx->timestamp_list, frame->pts);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* all the pending buffers are now ready for output */
|
/* all the pending buffers are now ready for output */
|
||||||
@@ -2626,6 +2684,10 @@ static int nvenc_send_frame(AVCodecContext *avctx, const AVFrame *frame)
|
|||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
opaque_ref_fail:
|
||||||
|
av_buffer_unref(&opaque_ref);
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ff_nvenc_receive_packet(AVCodecContext *avctx, AVPacket *pkt)
|
int ff_nvenc_receive_packet(AVCodecContext *avctx, AVPacket *pkt)
|
||||||
@@ -2684,5 +2746,5 @@ av_cold void ff_nvenc_encode_flush(AVCodecContext *avctx)
|
|||||||
NvencContext *ctx = avctx->priv_data;
|
NvencContext *ctx = avctx->priv_data;
|
||||||
|
|
||||||
nvenc_send_frame(avctx, NULL);
|
nvenc_send_frame(avctx, NULL);
|
||||||
av_fifo_reset2(ctx->timestamp_list);
|
reorder_queue_flush(ctx->reorder_queue);
|
||||||
}
|
}
|
||||||
|
@@ -171,7 +171,7 @@ typedef struct NvencContext
|
|||||||
AVFifo *unused_surface_queue;
|
AVFifo *unused_surface_queue;
|
||||||
AVFifo *output_surface_queue;
|
AVFifo *output_surface_queue;
|
||||||
AVFifo *output_surface_ready_queue;
|
AVFifo *output_surface_ready_queue;
|
||||||
AVFifo *timestamp_list;
|
AVFifo *reorder_queue;
|
||||||
|
|
||||||
NV_ENC_SEI_PAYLOAD *sei_data;
|
NV_ENC_SEI_PAYLOAD *sei_data;
|
||||||
int sei_data_size;
|
int sei_data_size;
|
||||||
|
@@ -181,7 +181,8 @@ const FFCodec ff_av1_nvenc_encoder = {
|
|||||||
.defaults = defaults,
|
.defaults = defaults,
|
||||||
.p.pix_fmts = ff_nvenc_pix_fmts,
|
.p.pix_fmts = ff_nvenc_pix_fmts,
|
||||||
.p.capabilities = AV_CODEC_CAP_DELAY | AV_CODEC_CAP_HARDWARE |
|
.p.capabilities = AV_CODEC_CAP_DELAY | AV_CODEC_CAP_HARDWARE |
|
||||||
AV_CODEC_CAP_ENCODER_FLUSH | AV_CODEC_CAP_DR1,
|
AV_CODEC_CAP_ENCODER_FLUSH | AV_CODEC_CAP_DR1 |
|
||||||
|
AV_CODEC_CAP_ENCODER_REORDERED_OPAQUE,
|
||||||
.caps_internal = FF_CODEC_CAP_NOT_INIT_THREADSAFE |
|
.caps_internal = FF_CODEC_CAP_NOT_INIT_THREADSAFE |
|
||||||
FF_CODEC_CAP_INIT_CLEANUP,
|
FF_CODEC_CAP_INIT_CLEANUP,
|
||||||
.p.wrapper_name = "nvenc",
|
.p.wrapper_name = "nvenc",
|
||||||
|
@@ -232,7 +232,8 @@ const FFCodec ff_h264_nvenc_encoder = {
|
|||||||
.p.priv_class = &h264_nvenc_class,
|
.p.priv_class = &h264_nvenc_class,
|
||||||
.defaults = defaults,
|
.defaults = defaults,
|
||||||
.p.capabilities = AV_CODEC_CAP_DELAY | AV_CODEC_CAP_HARDWARE |
|
.p.capabilities = AV_CODEC_CAP_DELAY | AV_CODEC_CAP_HARDWARE |
|
||||||
AV_CODEC_CAP_ENCODER_FLUSH | AV_CODEC_CAP_DR1,
|
AV_CODEC_CAP_ENCODER_FLUSH | AV_CODEC_CAP_DR1 |
|
||||||
|
AV_CODEC_CAP_ENCODER_REORDERED_OPAQUE,
|
||||||
.caps_internal = FF_CODEC_CAP_NOT_INIT_THREADSAFE |
|
.caps_internal = FF_CODEC_CAP_NOT_INIT_THREADSAFE |
|
||||||
FF_CODEC_CAP_INIT_CLEANUP,
|
FF_CODEC_CAP_INIT_CLEANUP,
|
||||||
.p.pix_fmts = ff_nvenc_pix_fmts,
|
.p.pix_fmts = ff_nvenc_pix_fmts,
|
||||||
|
@@ -214,7 +214,8 @@ const FFCodec ff_hevc_nvenc_encoder = {
|
|||||||
.defaults = defaults,
|
.defaults = defaults,
|
||||||
.p.pix_fmts = ff_nvenc_pix_fmts,
|
.p.pix_fmts = ff_nvenc_pix_fmts,
|
||||||
.p.capabilities = AV_CODEC_CAP_DELAY | AV_CODEC_CAP_HARDWARE |
|
.p.capabilities = AV_CODEC_CAP_DELAY | AV_CODEC_CAP_HARDWARE |
|
||||||
AV_CODEC_CAP_ENCODER_FLUSH | AV_CODEC_CAP_DR1,
|
AV_CODEC_CAP_ENCODER_FLUSH | AV_CODEC_CAP_DR1 |
|
||||||
|
AV_CODEC_CAP_ENCODER_REORDERED_OPAQUE,
|
||||||
.caps_internal = FF_CODEC_CAP_NOT_INIT_THREADSAFE |
|
.caps_internal = FF_CODEC_CAP_NOT_INIT_THREADSAFE |
|
||||||
FF_CODEC_CAP_INIT_CLEANUP,
|
FF_CODEC_CAP_INIT_CLEANUP,
|
||||||
.p.wrapper_name = "nvenc",
|
.p.wrapper_name = "nvenc",
|
||||||
|
Reference in New Issue
Block a user