From 28259c13db784d4b4175ca323dc1eeffec7f919b Mon Sep 17 00:00:00 2001 From: Anton Khirnov Date: Sat, 30 Jan 2016 21:33:48 +0100 Subject: [PATCH 1/3] nvenc: factor out the pixel format list --- libavcodec/nvenc.c | 7 +++++++ libavcodec/nvenc.h | 2 ++ libavcodec/nvenc_h264.c | 10 ++-------- libavcodec/nvenc_hevc.c | 10 ++-------- 4 files changed, 13 insertions(+), 16 deletions(-) diff --git a/libavcodec/nvenc.c b/libavcodec/nvenc.c index 2dfea9431e..53f0b13e16 100644 --- a/libavcodec/nvenc.c +++ b/libavcodec/nvenc.c @@ -76,6 +76,13 @@ } \ } while (0) +const enum AVPixelFormat ff_nvenc_pix_fmts[] = { + AV_PIX_FMT_NV12, + AV_PIX_FMT_YUV420P, + AV_PIX_FMT_YUV444P, + AV_PIX_FMT_NONE +}; + static const struct { NVENCSTATUS nverr; int averr; diff --git a/libavcodec/nvenc.h b/libavcodec/nvenc.h index 32d3345414..7b1e8a5596 100644 --- a/libavcodec/nvenc.h +++ b/libavcodec/nvenc.h @@ -135,4 +135,6 @@ int ff_nvenc_encode_close(AVCodecContext *avctx); int ff_nvenc_encode_frame(AVCodecContext *avctx, AVPacket *pkt, const AVFrame *frame, int *got_packet); +extern const enum AVPixelFormat ff_nvenc_pix_fmts[]; + #endif /* AVCODEC_NVENC_H */ diff --git a/libavcodec/nvenc_h264.c b/libavcodec/nvenc_h264.c index 92eb5196b3..7374b13f68 100644 --- a/libavcodec/nvenc_h264.c +++ b/libavcodec/nvenc_h264.c @@ -101,12 +101,9 @@ AVCodec ff_h264_nvenc_encoder = { .priv_data_size = sizeof(NVENCContext), .priv_class = &nvenc_hevc_class, .defaults = defaults, - .pix_fmts = (const enum AVPixelFormat[]){ AV_PIX_FMT_NV12, - AV_PIX_FMT_YUV420P, - AV_PIX_FMT_YUV444P, - AV_PIX_FMT_NONE }, .capabilities = AV_CODEC_CAP_DELAY, .caps_internal = FF_CODEC_CAP_INIT_CLEANUP, + .pix_fmts = ff_nvenc_pix_fmts, }; #if FF_API_NVENC_OLD_NAME @@ -135,11 +132,8 @@ AVCodec ff_nvenc_h264_encoder = { .priv_data_size = sizeof(NVENCContext), .priv_class = &nvenc_h264_old_class, .defaults = defaults, - .pix_fmts = (const enum AVPixelFormat[]){ AV_PIX_FMT_NV12, - AV_PIX_FMT_YUV420P, - AV_PIX_FMT_YUV444P, - AV_PIX_FMT_NONE }, .capabilities = AV_CODEC_CAP_DELAY, .caps_internal = FF_CODEC_CAP_INIT_CLEANUP, + .pix_fmts = ff_nvenc_pix_fmts, }; #endif diff --git a/libavcodec/nvenc_hevc.c b/libavcodec/nvenc_hevc.c index 2a6919797e..f0c6d41b83 100644 --- a/libavcodec/nvenc_hevc.c +++ b/libavcodec/nvenc_hevc.c @@ -99,10 +99,7 @@ AVCodec ff_hevc_nvenc_encoder = { .priv_data_size = sizeof(NVENCContext), .priv_class = &nvenc_hevc_class, .defaults = defaults, - .pix_fmts = (const enum AVPixelFormat[]){ AV_PIX_FMT_NV12, - AV_PIX_FMT_YUV420P, - AV_PIX_FMT_YUV444P, - AV_PIX_FMT_NONE }, + .pix_fmts = ff_nvenc_pix_fmts, .capabilities = AV_CODEC_CAP_DELAY, .caps_internal = FF_CODEC_CAP_INIT_CLEANUP, }; @@ -133,10 +130,7 @@ AVCodec ff_nvenc_hevc_encoder = { .priv_data_size = sizeof(NVENCContext), .priv_class = &nvenc_hevc_old_class, .defaults = defaults, - .pix_fmts = (const enum AVPixelFormat[]){ AV_PIX_FMT_NV12, - AV_PIX_FMT_YUV420P, - AV_PIX_FMT_YUV444P, - AV_PIX_FMT_NONE }, + .pix_fmts = ff_nvenc_pix_fmts, .capabilities = AV_CODEC_CAP_DELAY, .caps_internal = FF_CODEC_CAP_INIT_CLEANUP, }; From 118beda355076acf03611e5c8bc147a9c634e54f Mon Sep 17 00:00:00 2001 From: Anton Khirnov Date: Sat, 30 Jan 2016 22:06:47 +0100 Subject: [PATCH 2/3] nvenc: merge input and output surface structs An input frame always corresponds to exactly one output packet, so there is no point in complicating the situation by managing them separately. --- libavcodec/nvenc.c | 118 +++++++++++++++------------------------------ libavcodec/nvenc.h | 16 ++---- 2 files changed, 45 insertions(+), 89 deletions(-) diff --git a/libavcodec/nvenc.c b/libavcodec/nvenc.c index 53f0b13e16..caf683e13c 100644 --- a/libavcodec/nvenc.c +++ b/libavcodec/nvenc.c @@ -750,8 +750,8 @@ static int nvenc_alloc_surface(AVCodecContext *avctx, int idx) if (ret != NV_ENC_SUCCESS) return nvenc_print_error(avctx, ret, "CreateInputBuffer failed"); - ctx->in[idx].in = in_buffer.inputBuffer; - ctx->in[idx].format = in_buffer.bufferFmt; + ctx->frames[idx].in = in_buffer.inputBuffer; + ctx->frames[idx].format = in_buffer.bufferFmt; /* 1MB is large enough to hold most output frames. * NVENC increases this automaticaly if it's not enough. */ @@ -763,8 +763,7 @@ static int nvenc_alloc_surface(AVCodecContext *avctx, int idx) if (ret != NV_ENC_SUCCESS) return nvenc_print_error(avctx, ret, "CreateBitstreamBuffer failed"); - ctx->out[idx].out = out_buffer.bitstreamBuffer; - ctx->out[idx].busy = 0; + ctx->frames[idx].out = out_buffer.bitstreamBuffer; return 0; } @@ -777,21 +776,17 @@ static int nvenc_setup_surfaces(AVCodecContext *avctx) ctx->nb_surfaces = FFMAX(4 + avctx->max_b_frames, ctx->nb_surfaces); - ctx->in = av_mallocz(ctx->nb_surfaces * sizeof(*ctx->in)); - if (!ctx->in) - return AVERROR(ENOMEM); - - ctx->out = av_mallocz(ctx->nb_surfaces * sizeof(*ctx->out)); - if (!ctx->out) + ctx->frames = av_mallocz_array(ctx->nb_surfaces, sizeof(*ctx->frames)); + if (!ctx->frames) return AVERROR(ENOMEM); ctx->timestamps = av_fifo_alloc(ctx->nb_surfaces * sizeof(int64_t)); if (!ctx->timestamps) return AVERROR(ENOMEM); - ctx->pending = av_fifo_alloc(ctx->nb_surfaces * sizeof(ctx->out)); + ctx->pending = av_fifo_alloc(ctx->nb_surfaces * sizeof(*ctx->frames)); if (!ctx->pending) return AVERROR(ENOMEM); - ctx->ready = av_fifo_alloc(ctx->nb_surfaces * sizeof(ctx->out)); + ctx->ready = av_fifo_alloc(ctx->nb_surfaces * sizeof(*ctx->frames)); if (!ctx->ready) return AVERROR(ENOMEM); @@ -846,15 +841,14 @@ av_cold int ff_nvenc_encode_close(AVCodecContext *avctx) av_fifo_free(ctx->pending); av_fifo_free(ctx->ready); - if (ctx->in) { + if (ctx->frames) { for (i = 0; i < ctx->nb_surfaces; ++i) { - nv->nvEncDestroyInputBuffer(ctx->nvenc_ctx, ctx->in[i].in); - nv->nvEncDestroyBitstreamBuffer(ctx->nvenc_ctx, ctx->out[i].out); + nv->nvEncDestroyInputBuffer(ctx->nvenc_ctx, ctx->frames[i].in); + nv->nvEncDestroyBitstreamBuffer(ctx->nvenc_ctx, ctx->frames[i].out); } } - av_freep(&ctx->in); - av_freep(&ctx->out); + av_freep(&ctx->frames); if (ctx->nvenc_ctx) nv->nvEncDestroyEncoder(ctx->nvenc_ctx); @@ -895,27 +889,14 @@ av_cold int ff_nvenc_encode_init(AVCodecContext *avctx) return 0; } -static NVENCInputSurface *get_input_surface(NVENCContext *ctx) +static NVENCFrame *get_free_frame(NVENCContext *ctx) { int i; for (i = 0; i < ctx->nb_surfaces; i++) { - if (!ctx->in[i].locked) { - ctx->in[i].locked = 1; - return &ctx->in[i]; - } - } - - return NULL; -} - -static NVENCOutputSurface *get_output_surface(NVENCContext *ctx) -{ - int i; - - for (i = 0; i < ctx->nb_surfaces; i++) { - if (!ctx->out[i].busy) { - return &ctx->out[i]; + if (!ctx->frames[i].locked) { + ctx->frames[i].locked = 1; + return &ctx->frames[i]; } } @@ -976,20 +957,16 @@ static int nvenc_copy_frame(NV_ENC_LOCK_INPUT_BUFFER *in, const AVFrame *frame) return 0; } -static int nvenc_enqueue_frame(AVCodecContext *avctx, const AVFrame *frame, - NVENCInputSurface **in_surf) +static int nvenc_upload_frame(AVCodecContext *avctx, const AVFrame *frame, + NVENCFrame *nvenc_frame) { NVENCContext *ctx = avctx->priv_data; NV_ENCODE_API_FUNCTION_LIST *nv = &ctx->nvel.nvenc_funcs; NV_ENC_LOCK_INPUT_BUFFER params = { 0 }; - NVENCInputSurface *in = get_input_surface(ctx); int ret; - if (!in) - return AVERROR_BUG; - params.version = NV_ENC_LOCK_INPUT_BUFFER_VER; - params.inputBuffer = in->in; + params.inputBuffer = nvenc_frame->in; ret = nv->nvEncLockInputBuffer(ctx->nvenc_ctx, ¶ms); @@ -1000,16 +977,14 @@ static int nvenc_enqueue_frame(AVCodecContext *avctx, const AVFrame *frame, if (ret < 0) goto fail; - ret = nv->nvEncUnlockInputBuffer(ctx->nvenc_ctx, in->in); + ret = nv->nvEncUnlockInputBuffer(ctx->nvenc_ctx, nvenc_frame->in); if (ret != NV_ENC_SUCCESS) return nvenc_print_error(avctx, ret, "Cannot unlock the buffer"); - *in_surf = in; - return 0; fail: - nv->nvEncUnlockInputBuffer(ctx->nvenc_ctx, in->in); + nv->nvEncUnlockInputBuffer(ctx->nvenc_ctx, nvenc_frame->in); return ret; } @@ -1045,19 +1020,6 @@ static inline int nvenc_dequeue_timestamp(AVFifoBuffer *f, int64_t *pts) return av_fifo_generic_read(f, pts, sizeof(*pts), NULL); } -static inline int nvenc_enqueue_surface(AVFifoBuffer *f, - NVENCOutputSurface *surf) -{ - surf->busy = 1; - return av_fifo_generic_write(f, &surf, sizeof(surf), NULL); -} - -static inline int nvenc_dequeue_surface(AVFifoBuffer *f, - NVENCOutputSurface **surf) -{ - return av_fifo_generic_read(f, surf, sizeof(*surf), NULL); -} - static int nvenc_set_timestamp(AVCodecContext *avctx, NV_ENC_LOCK_BITSTREAM *params, AVPacket *pkt) @@ -1095,15 +1057,15 @@ static int nvenc_get_frame(AVCodecContext *avctx, AVPacket *pkt) NVENCContext *ctx = avctx->priv_data; NV_ENCODE_API_FUNCTION_LIST *nv = &ctx->nvel.nvenc_funcs; NV_ENC_LOCK_BITSTREAM params = { 0 }; - NVENCOutputSurface *out = NULL; + NVENCFrame *frame; int ret; - ret = nvenc_dequeue_surface(ctx->ready, &out); + ret = av_fifo_generic_read(ctx->ready, &frame, sizeof(frame), NULL); if (ret) return ret; params.version = NV_ENC_LOCK_BITSTREAM_VER; - params.outputBitstream = out->out; + params.outputBitstream = frame->out; ret = nv->nvEncLockBitstream(ctx->nvenc_ctx, ¶ms); if (ret < 0) @@ -1115,11 +1077,11 @@ static int nvenc_get_frame(AVCodecContext *avctx, AVPacket *pkt) memcpy(pkt->data, params.bitstreamBufferPtr, pkt->size); - ret = nv->nvEncUnlockBitstream(ctx->nvenc_ctx, out->out); + ret = nv->nvEncUnlockBitstream(ctx->nvenc_ctx, frame->out); if (ret < 0) return nvenc_print_error(avctx, ret, "Cannot unlock the bitstream"); - out->busy = out->in->locked = 0; + frame->locked = 0; ret = nvenc_set_timestamp(avctx, ¶ms, pkt); if (ret < 0) @@ -1168,27 +1130,27 @@ int ff_nvenc_encode_frame(AVCodecContext *avctx, AVPacket *pkt, NVENCContext *ctx = avctx->priv_data; NV_ENCODE_API_FUNCTION_LIST *nv = &ctx->nvel.nvenc_funcs; NV_ENC_PIC_PARAMS params = { 0 }; - NVENCInputSurface *in = NULL; - NVENCOutputSurface *out = NULL; + NVENCFrame *nvenc_frame = NULL; int enc_ret, ret; params.version = NV_ENC_PIC_PARAMS_VER; if (frame) { - ret = nvenc_enqueue_frame(avctx, frame, &in); + nvenc_frame = get_free_frame(ctx); + if (!nvenc_frame) { + av_log(avctx, AV_LOG_ERROR, "No free surfaces\n"); + return AVERROR_BUG; + } + + ret = nvenc_upload_frame(avctx, frame, nvenc_frame); if (ret < 0) return ret; - out = get_output_surface(ctx); - if (!out) - return AVERROR_BUG; - out->in = in; - - params.inputBuffer = in->in; - params.bufferFmt = in->format; + params.inputBuffer = nvenc_frame->in; + params.bufferFmt = nvenc_frame->format; params.inputWidth = frame->width; params.inputHeight = frame->height; - params.outputBitstream = out->out; + params.outputBitstream = nvenc_frame->out; params.inputTimeStamp = frame->pts; if (avctx->flags & AV_CODEC_FLAG_INTERLACED_DCT) { @@ -1219,8 +1181,8 @@ int ff_nvenc_encode_frame(AVCodecContext *avctx, AVPacket *pkt, enc_ret != NV_ENC_ERR_NEED_MORE_INPUT) return nvenc_print_error(avctx, enc_ret, "Error encoding the frame"); - if (out) { - ret = nvenc_enqueue_surface(ctx->pending, out); + if (nvenc_frame) { + ret = av_fifo_generic_write(ctx->pending, &nvenc_frame, sizeof(nvenc_frame), NULL); if (ret < 0) return ret; } @@ -1228,8 +1190,8 @@ int ff_nvenc_encode_frame(AVCodecContext *avctx, AVPacket *pkt, /* all the pending buffers are now ready for output */ if (enc_ret == NV_ENC_SUCCESS) { while (av_fifo_size(ctx->pending) > 0) { - av_fifo_generic_read(ctx->pending, &out, sizeof(out), NULL); - av_fifo_generic_write(ctx->ready, &out, sizeof(out), NULL); + av_fifo_generic_read(ctx->pending, &nvenc_frame, sizeof(nvenc_frame), NULL); + av_fifo_generic_write(ctx->ready, &nvenc_frame, sizeof(nvenc_frame), NULL); } } diff --git a/libavcodec/nvenc.h b/libavcodec/nvenc.h index 7b1e8a5596..6edca75195 100644 --- a/libavcodec/nvenc.h +++ b/libavcodec/nvenc.h @@ -27,17 +27,12 @@ #include "avcodec.h" -typedef struct NVENCInputSurface { - NV_ENC_INPUT_PTR in; +typedef struct NVENCFrame { + NV_ENC_INPUT_PTR in; + NV_ENC_OUTPUT_PTR out; NV_ENC_BUFFER_FORMAT format; int locked; -} NVENCInputSurface; - -typedef struct NVENCOutputSurface { - NV_ENC_OUTPUT_PTR out; - NVENCInputSurface *in; - int busy; -} NVENCOutputSurface; +} NVENCFrame; typedef CUresult(CUDAAPI *PCUINIT)(unsigned int Flags); typedef CUresult(CUDAAPI *PCUDEVICEGETCOUNT)(int *count); @@ -107,8 +102,7 @@ typedef struct NVENCContext { CUcontext cu_context; int nb_surfaces; - NVENCInputSurface *in; - NVENCOutputSurface *out; + NVENCFrame *frames; AVFifoBuffer *timestamps; AVFifoBuffer *pending, *ready; From d005ccc630e42daab8ec2afecf972d1551a9401a Mon Sep 17 00:00:00 2001 From: Anton Khirnov Date: Sat, 30 Jan 2016 22:13:04 +0100 Subject: [PATCH 3/3] nvenc: rename a misnamed function This function copies the encoded bistream into the caller's packet, calling it 'get_frame' is misleading. --- libavcodec/nvenc.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/libavcodec/nvenc.c b/libavcodec/nvenc.c index caf683e13c..0783c355c8 100644 --- a/libavcodec/nvenc.c +++ b/libavcodec/nvenc.c @@ -1052,7 +1052,7 @@ static int nvenc_set_timestamp(AVCodecContext *avctx, return nvenc_dequeue_timestamp(ctx->timestamps, &pkt->dts); } -static int nvenc_get_frame(AVCodecContext *avctx, AVPacket *pkt) +static int nvenc_get_output(AVCodecContext *avctx, AVPacket *pkt) { NVENCContext *ctx = avctx->priv_data; NV_ENCODE_API_FUNCTION_LIST *nv = &ctx->nvel.nvenc_funcs; @@ -1196,7 +1196,7 @@ int ff_nvenc_encode_frame(AVCodecContext *avctx, AVPacket *pkt, } if (output_ready(avctx, !frame)) { - ret = nvenc_get_frame(avctx, pkt); + ret = nvenc_get_output(avctx, pkt); if (ret < 0) return ret; *got_packet = 1;