diff options
author | Haihao Xiang <haihao.xiang@intel.com> | 2023-07-26 15:15:21 +0800 |
---|---|---|
committer | Haihao Xiang <haihao.xiang@intel.com> | 2023-08-04 10:27:55 +0800 |
commit | b66aecf1b4961bb37312f4da1986980ff0cc509e (patch) | |
tree | 35de9b9f4da4b787cb221ee49bbd44e662ffafeb | |
parent | b42ea03f8baec226b8e7d1765f04cec5b18d1284 (diff) | |
download | ffmpeg-b66aecf1b4961bb37312f4da1986980ff0cc509e.tar.gz |
lavfi/vf_vpp_qsv: set color properties for output
User may set color range / matrix coefficient set / primaries / transfer
characteristics for output.
Signed-off-by: Haihao Xiang <haihao.xiang@intel.com>
-rw-r--r-- | libavfilter/vf_vpp_qsv.c | 90 |
1 files changed, 87 insertions, 3 deletions
diff --git a/libavfilter/vf_vpp_qsv.c b/libavfilter/vf_vpp_qsv.c index a79dc61364..de2725f031 100644 --- a/libavfilter/vf_vpp_qsv.c +++ b/libavfilter/vf_vpp_qsv.c @@ -61,6 +61,8 @@ typedef struct VPPContext{ #if QSV_ONEVPL /** Video signal info attached on the input frame */ mfxExtVideoSignalInfo invsi_conf; + /** Video signal info attached on the output frame */ + mfxExtVideoSignalInfo outvsi_conf; #endif /** @@ -104,6 +106,16 @@ typedef struct VPPContext{ char *ow, *oh; char *output_format_str; + /** The color properties for output */ + char *color_primaries_str; + char *color_transfer_str; + char *color_matrix_str; + + int color_range; + enum AVColorPrimaries color_primaries; + enum AVColorTransferCharacteristic color_transfer; + enum AVColorSpace color_matrix; + int has_passthrough; /* apply pass through mode if possible */ int field_rate; /* Generate output at frame rate or field rate for deinterlace mode, 0: frame, 1: field */ } VPPContext; @@ -231,6 +243,11 @@ static av_cold int vpp_preinit(AVFilterContext *ctx) vpp->contrast = 1.0; vpp->transpose = -1; + vpp->color_range = AVCOL_RANGE_UNSPECIFIED; + vpp->color_primaries = AVCOL_PRI_UNSPECIFIED; + vpp->color_transfer = AVCOL_TRC_UNSPECIFIED; + vpp->color_matrix = AVCOL_SPC_UNSPECIFIED; + vpp->has_passthrough = 1; return 0; @@ -250,6 +267,24 @@ static av_cold int vpp_init(AVFilterContext *ctx) } } +#define STRING_OPTION(var_name, func_name, default_value) do { \ + if (vpp->var_name ## _str) { \ + int var = av_ ## func_name ## _from_name(vpp->var_name ## _str); \ + if (var < 0) { \ + av_log(ctx, AV_LOG_ERROR, "Invalid %s.\n", #var_name); \ + return AVERROR(EINVAL); \ + } \ + vpp->var_name = var; \ + } else { \ + vpp->var_name = default_value; \ + } \ + } while (0) + + STRING_OPTION(color_primaries, color_primaries, AVCOL_PRI_UNSPECIFIED); + STRING_OPTION(color_transfer, color_transfer, AVCOL_TRC_UNSPECIFIED); + STRING_OPTION(color_matrix, color_space, AVCOL_SPC_UNSPECIFIED); + +#undef STRING_OPTION return 0; } @@ -353,11 +388,11 @@ static int vpp_set_frame_ext_params(AVFilterContext *ctx, const AVFrame *in, AVF #if QSV_ONEVPL VPPContext *vpp = ctx->priv; QSVVPPContext *qsvvpp = &vpp->qsv; - mfxExtVideoSignalInfo invsi_conf; + mfxExtVideoSignalInfo invsi_conf, outvsi_conf; fp->num_ext_buf = 0; - if (!in || + if (!in || !out || !QSV_RUNTIME_VERSION_ATLEAST(qsvvpp->ver, 2, 0)) return 0; @@ -370,9 +405,32 @@ static int vpp_set_frame_ext_params(AVFilterContext *ctx, const AVFrame *in, AVF invsi_conf.MatrixCoefficients = (in->colorspace == AVCOL_SPC_UNSPECIFIED) ? AVCOL_SPC_BT709 : in->colorspace; invsi_conf.ColourDescriptionPresent = 1; - if (memcmp(&vpp->invsi_conf, &invsi_conf, sizeof(mfxExtVideoSignalInfo))) { + if (vpp->color_range != AVCOL_RANGE_UNSPECIFIED) + out->color_range = vpp->color_range; + if (vpp->color_primaries != AVCOL_PRI_UNSPECIFIED) + out->color_primaries = vpp->color_primaries; + if (vpp->color_transfer != AVCOL_TRC_UNSPECIFIED) + out->color_trc = vpp->color_transfer; + if (vpp->color_matrix != AVCOL_SPC_UNSPECIFIED) + out->colorspace = vpp->color_matrix; + + memset(&outvsi_conf, 0, sizeof(mfxExtVideoSignalInfo)); + outvsi_conf.Header.BufferId = MFX_EXTBUFF_VIDEO_SIGNAL_INFO_OUT; + outvsi_conf.Header.BufferSz = sizeof(mfxExtVideoSignalInfo); + outvsi_conf.VideoFullRange = (out->color_range == AVCOL_RANGE_JPEG); + outvsi_conf.ColourPrimaries = (out->color_primaries == AVCOL_PRI_UNSPECIFIED) ? AVCOL_PRI_BT709 : out->color_primaries; + outvsi_conf.TransferCharacteristics = (out->color_trc == AVCOL_TRC_UNSPECIFIED) ? AVCOL_TRC_BT709 : out->color_trc; + outvsi_conf.MatrixCoefficients = (out->colorspace == AVCOL_SPC_UNSPECIFIED) ? AVCOL_SPC_BT709 : out->colorspace; + outvsi_conf.ColourDescriptionPresent = 1; + + if (memcmp(&vpp->invsi_conf, &invsi_conf, sizeof(mfxExtVideoSignalInfo)) || + memcmp(&vpp->outvsi_conf, &outvsi_conf, sizeof(mfxExtVideoSignalInfo))) { vpp->invsi_conf = invsi_conf; fp->ext_buf[fp->num_ext_buf++] = (mfxExtBuffer*)&vpp->invsi_conf; + + vpp->outvsi_conf = outvsi_conf; + fp->ext_buf[fp->num_ext_buf++] = (mfxExtBuffer*)&vpp->outvsi_conf; + } #endif @@ -560,6 +618,10 @@ static int config_output(AVFilterLink *outlink) if (vpp->use_frc || vpp->use_crop || vpp->deinterlace || vpp->denoise || vpp->detail || vpp->procamp || vpp->rotate || vpp->hflip || inlink->w != outlink->w || inlink->h != outlink->h || in_format != vpp->out_format || + vpp->color_range != AVCOL_RANGE_UNSPECIFIED || + vpp->color_primaries != AVCOL_PRI_UNSPECIFIED || + vpp->color_transfer != AVCOL_TRC_UNSPECIFIED || + vpp->color_matrix != AVCOL_SPC_UNSPECIFIED || !vpp->has_passthrough) return ff_qsvvpp_init(ctx, ¶m); else { @@ -750,6 +812,28 @@ static const AVOption vpp_options[] = { { "field", "Output at field rate (one frame of output for each field)", 0, AV_OPT_TYPE_CONST, { .i64 = 1 }, 0, 0, FLAGS, "rate" }, + { "out_range", "Output color range", + OFFSET(color_range), AV_OPT_TYPE_INT, { .i64 = AVCOL_RANGE_UNSPECIFIED }, + AVCOL_RANGE_UNSPECIFIED, AVCOL_RANGE_JPEG, FLAGS, "range" }, + { "full", "Full range", + 0, AV_OPT_TYPE_CONST, { .i64 = AVCOL_RANGE_JPEG }, 0, 0, FLAGS, "range" }, + { "limited", "Limited range", + 0, AV_OPT_TYPE_CONST, { .i64 = AVCOL_RANGE_MPEG }, 0, 0, FLAGS, "range" }, + { "jpeg", "Full range", + 0, AV_OPT_TYPE_CONST, { .i64 = AVCOL_RANGE_JPEG }, 0, 0, FLAGS, "range" }, + { "mpeg", "Limited range", + 0, AV_OPT_TYPE_CONST, { .i64 = AVCOL_RANGE_MPEG }, 0, 0, FLAGS, "range" }, + { "tv", "Limited range", + 0, AV_OPT_TYPE_CONST, { .i64 = AVCOL_RANGE_MPEG }, 0, 0, FLAGS, "range" }, + { "pc", "Full range", + 0, AV_OPT_TYPE_CONST, { .i64 = AVCOL_RANGE_JPEG }, 0, 0, FLAGS, "range" }, + { "out_color_matrix", "Output color matrix coefficient set", + OFFSET(color_matrix_str), AV_OPT_TYPE_STRING, { .str = NULL }, .flags = FLAGS }, + { "out_color_primaries", "Output color primaries", + OFFSET(color_primaries_str), AV_OPT_TYPE_STRING, { .str = NULL }, .flags = FLAGS }, + { "out_color_transfer", "Output color transfer characteristics", + OFFSET(color_transfer_str), AV_OPT_TYPE_STRING, { .str = NULL }, .flags = FLAGS }, + { NULL } }; |