// FFMPEG Video Encoder Integration for OBS Studio // Copyright (c) 2019 Michael Fabian Dirks // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in all // copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE // SOFTWARE. #include "nvenc_shared.hpp" #include #include "codecs/hevc.hpp" #include "encoder.hpp" #include "ffmpeg/tools.hpp" #include "plugin.hpp" #include "strings.hpp" #include "utility.hpp" extern "C" { #include #pragma warning(push) #pragma warning(disable : 4244) #include #pragma warning(pop) } #define ST_PRESET "NVENC.Preset" #define ST_PRESET_(x) ST_PRESET "." D_VSTR(x) #define ST_RATECONTROL "NVENC.RateControl" #define ST_RATECONTROL_MODE ST_RATECONTROL ".Mode" #define ST_RATECONTROL_MODE_(x) ST_RATECONTROL_MODE "." D_VSTR(x) #define ST_RATECONTROL_TWOPASS ST_RATECONTROL ".TwoPass" #define ST_RATECONTROL_LOOKAHEAD ST_RATECONTROL ".LookAhead" #define ST_RATECONTROL_ADAPTIVEI ST_RATECONTROL ".AdaptiveI" #define ST_RATECONTROL_ADAPTIVEB ST_RATECONTROL ".AdaptiveB" #define ST_RATECONTROL_BITRATE ST_RATECONTROL ".Bitrate" #define ST_RATECONTROL_BITRATE_TARGET ST_RATECONTROL_BITRATE ".Target" #define ST_RATECONTROL_BITRATE_MAXIMUM ST_RATECONTROL_BITRATE ".Maximum" #define ST_RATECONTROL_QUALITY ST_RATECONTROL ".Quality" #define ST_RATECONTROL_QUALITY_MINIMUM ST_RATECONTROL_QUALITY ".Minimum" #define ST_RATECONTROL_QUALITY_MAXIMUM ST_RATECONTROL_QUALITY ".Maximum" #define ST_RATECONTROL_QUALITY_TARGET ST_RATECONTROL_QUALITY ".Target" #define ST_RATECONTROL_QP ST_RATECONTROL ".QP" #define ST_RATECONTROL_QP_I ST_RATECONTROL_QP ".I" #define ST_RATECONTROL_QP_I_INITIAL ST_RATECONTROL_QP_I ".Initial" #define ST_RATECONTROL_QP_P ST_RATECONTROL_QP ".P" #define ST_RATECONTROL_QP_P_INITIAL ST_RATECONTROL_QP_P ".Initial" #define ST_RATECONTROL_QP_B ST_RATECONTROL_QP ".B" #define ST_RATECONTROL_QP_B_INITIAL ST_RATECONTROL_QP_B ".Initial" #define ST_AQ "NVENC.AQ" #define ST_AQ_SPATIAL ST_AQ ".Spatial" #define ST_AQ_TEMPORAL ST_AQ ".Temporal" #define ST_AQ_STRENGTH ST_AQ ".Strength" #define ST_OTHER "NVENC.Other" #define ST_OTHER_BFRAMES ST_OTHER ".BFrames" #define ST_OTHER_BFRAME_REFERENCEMODE ST_OTHER ".BFrameReferenceMode" #define ST_OTHER_ZEROLATENCY ST_OTHER ".ZeroLatency" #define ST_OTHER_WEIGHTED_PREDICTION ST_OTHER ".WeightedPrediction" #define ST_OTHER_NONREFERENCE_PFRAMES ST_OTHER ".NonReferencePFrames" using namespace obsffmpeg::nvenc; std::map obsffmpeg::nvenc::presets{ {preset::DEFAULT, ST_PRESET_(Default)}, {preset::SLOW, ST_PRESET_(Slow)}, {preset::MEDIUM, ST_PRESET_(Medium)}, {preset::FAST, ST_PRESET_(Fast)}, {preset::HIGH_PERFORMANCE, ST_PRESET_(HighPerformance)}, {preset::HIGH_QUALITY, ST_PRESET_(HighQuality)}, {preset::BLURAYDISC, ST_PRESET_(BluRayDisc)}, {preset::LOW_LATENCY, ST_PRESET_(LowLatency)}, {preset::LOW_LATENCY_HIGH_PERFORMANCE, ST_PRESET_(LowLatencyHighPerformance)}, {preset::LOW_LATENCY_HIGH_QUALITY, ST_PRESET_(LowLatencyHighQuality)}, {preset::LOSSLESS, ST_PRESET_(Lossless)}, {preset::LOSSLESS_HIGH_PERFORMANCE, ST_PRESET_(LosslessHighPerformance)}, }; std::map obsffmpeg::nvenc::preset_to_opt{ {preset::DEFAULT, "default"}, {preset::SLOW, "slow"}, {preset::MEDIUM, "medium"}, {preset::FAST, "fast"}, {preset::HIGH_PERFORMANCE, "hp"}, {preset::HIGH_QUALITY, "hq"}, {preset::BLURAYDISC, "bd"}, {preset::LOW_LATENCY, "ll"}, {preset::LOW_LATENCY_HIGH_PERFORMANCE, "llhp"}, {preset::LOW_LATENCY_HIGH_QUALITY, "llhq"}, {preset::LOSSLESS, "lossless"}, {preset::LOSSLESS_HIGH_PERFORMANCE, "losslesshp"}, }; std::map obsffmpeg::nvenc::ratecontrolmodes{ {ratecontrolmode::CQP, ST_RATECONTROL_MODE_(CQP)}, {ratecontrolmode::VBR, ST_RATECONTROL_MODE_(VBR)}, {ratecontrolmode::VBR_HQ, ST_RATECONTROL_MODE_(VBR_HQ)}, {ratecontrolmode::CBR, ST_RATECONTROL_MODE_(CBR)}, {ratecontrolmode::CBR_HQ, ST_RATECONTROL_MODE_(CBR_HQ)}, {ratecontrolmode::CBR_LD_HQ, ST_RATECONTROL_MODE_(CBR_LD_HQ)}, }; std::map obsffmpeg::nvenc::ratecontrolmode_to_opt{ {ratecontrolmode::CQP, "constqp"}, {ratecontrolmode::VBR, "vbr"}, {ratecontrolmode::VBR_HQ, "vbr_hq"}, {ratecontrolmode::CBR, "cbr"}, {ratecontrolmode::CBR_HQ, "cbr_hq"}, {ratecontrolmode::CBR_LD_HQ, "cbr_ld_hq"}, }; std::map obsffmpeg::nvenc::b_ref_modes{ {b_ref_mode::DISABLED, S_STATE_DISABLED}, {b_ref_mode::EACH, ST_OTHER_BFRAME_REFERENCEMODE ".Each"}, {b_ref_mode::MIDDLE, ST_OTHER_BFRAME_REFERENCEMODE ".Middle"}, }; std::map obsffmpeg::nvenc::b_ref_mode_to_opt{ {b_ref_mode::DISABLED, "disabled"}, {b_ref_mode::EACH, "each"}, {b_ref_mode::MIDDLE, "middle"}, }; void obsffmpeg::nvenc::override_update(obsffmpeg::encoder* instance, obs_data_t*) { AVCodecContext* context = const_cast(instance->get_avcodeccontext()); int64_t rclookahead = 0; int64_t surfaces = 0; int64_t async_depth = 0; av_opt_get_int(context, "rc-lookahead", AV_OPT_SEARCH_CHILDREN, &rclookahead); av_opt_get_int(context, "surfaces", AV_OPT_SEARCH_CHILDREN, &surfaces); av_opt_get_int(context, "async_depth", AV_OPT_SEARCH_CHILDREN, &async_depth); // Calculate and set the number of surfaces to allocate (if not user overridden). if (surfaces == 0) { surfaces = std::max(4ll, (context->max_b_frames + 1ll) * 4ll); if (rclookahead > 0) { surfaces = std::max(1ll, std::max(surfaces, rclookahead + (context->max_b_frames + 5ll))); } else if (context->max_b_frames > 0) { surfaces = std::max(4ll, (context->max_b_frames + 1ll) * 4ll); } else { surfaces = 4; } av_opt_set_int(context, "surfaces", surfaces, AV_OPT_SEARCH_CHILDREN); } // Set delay context->delay = static_cast(std::min(std::max(async_depth, 3ll), surfaces - 1)); } void obsffmpeg::nvenc::get_defaults(obs_data_t* settings, const AVCodec*, AVCodecContext*) { obs_data_set_default_int(settings, ST_PRESET, static_cast(preset::DEFAULT)); obs_data_set_default_int(settings, ST_RATECONTROL_MODE, static_cast(ratecontrolmode::CBR_HQ)); obs_data_set_default_int(settings, ST_RATECONTROL_TWOPASS, -1); obs_data_set_default_int(settings, ST_RATECONTROL_LOOKAHEAD, 0); obs_data_set_default_int(settings, ST_RATECONTROL_ADAPTIVEI, -1); obs_data_set_default_int(settings, ST_RATECONTROL_ADAPTIVEB, -1); obs_data_set_default_int(settings, ST_RATECONTROL_BITRATE_TARGET, 6000); obs_data_set_default_int(settings, ST_RATECONTROL_BITRATE_MAXIMUM, 6000); obs_data_set_default_int(settings, S_RATECONTROL_BUFFERSIZE, 12000); obs_data_set_default_int(settings, ST_RATECONTROL_QUALITY_MINIMUM, 51); obs_data_set_default_int(settings, ST_RATECONTROL_QUALITY_MAXIMUM, -1); obs_data_set_default_int(settings, ST_RATECONTROL_QUALITY_TARGET, 0); obs_data_set_default_int(settings, ST_RATECONTROL_QP_I, 21); obs_data_set_default_int(settings, ST_RATECONTROL_QP_I_INITIAL, -1); obs_data_set_default_int(settings, ST_RATECONTROL_QP_P, 21); obs_data_set_default_int(settings, ST_RATECONTROL_QP_P_INITIAL, -1); obs_data_set_default_int(settings, ST_RATECONTROL_QP_B, 21); obs_data_set_default_int(settings, ST_RATECONTROL_QP_B_INITIAL, -1); obs_data_set_default_int(settings, ST_AQ_SPATIAL, -1); obs_data_set_default_int(settings, ST_AQ_STRENGTH, 8); obs_data_set_default_int(settings, ST_AQ_TEMPORAL, -1); obs_data_set_default_int(settings, ST_OTHER_BFRAMES, 2); obs_data_set_default_int(settings, ST_OTHER_BFRAME_REFERENCEMODE, static_cast(b_ref_mode::DISABLED)); obs_data_set_default_int(settings, ST_OTHER_ZEROLATENCY, -1); obs_data_set_default_int(settings, ST_OTHER_WEIGHTED_PREDICTION, -1); obs_data_set_default_int(settings, ST_OTHER_NONREFERENCE_PFRAMES, -1); // Replay Buffer obs_data_set_default_int(settings, "bitrate", 0); } static bool modified_ratecontrol(obs_properties_t* props, obs_property_t*, obs_data_t* settings) { using namespace obsffmpeg::nvenc; bool have_bitrate = false; bool have_bitrate_max = false; bool have_quality = false; bool have_qp = false; bool have_qp_init = false; ratecontrolmode rc = static_cast(obs_data_get_int(settings, ST_RATECONTROL_MODE)); switch (rc) { case ratecontrolmode::CQP: have_qp = true; break; case ratecontrolmode::CBR: case ratecontrolmode::CBR_HQ: case ratecontrolmode::CBR_LD_HQ: have_bitrate = true; break; case ratecontrolmode::VBR: case ratecontrolmode::VBR_HQ: have_bitrate = true; have_bitrate_max = true; have_quality = true; have_qp_init = true; break; } obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_BITRATE), have_bitrate || have_bitrate_max); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_BITRATE_TARGET), have_bitrate); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_BITRATE_MAXIMUM), have_bitrate_max); obs_property_set_visible(obs_properties_get(props, S_RATECONTROL_BUFFERSIZE), have_bitrate || have_bitrate_max); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QUALITY), have_quality); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QUALITY_MINIMUM), have_quality); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QUALITY_MAXIMUM), have_quality); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QUALITY_TARGET), have_quality); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QP), have_qp || have_qp_init); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QP_I), have_qp); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QP_P), have_qp); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QP_B), have_qp); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QP_I_INITIAL), have_qp_init); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QP_P_INITIAL), have_qp_init); obs_property_set_visible(obs_properties_get(props, ST_RATECONTROL_QP_B_INITIAL), have_qp_init); return true; } static bool modified_quality(obs_properties_t* props, obs_property_t*, obs_data_t* settings) { bool enabled = obs_data_get_bool(settings, ST_RATECONTROL_QUALITY); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QUALITY_MINIMUM), enabled); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QUALITY_MAXIMUM), enabled); return true; } static bool modified_aq(obs_properties_t* props, obs_property_t*, obs_data_t* settings) { bool spatial_aq = obs_data_get_int(settings, ST_AQ_SPATIAL) == 1; obs_property_set_visible(obs_properties_get(props, ST_AQ_STRENGTH), spatial_aq); return true; } void obsffmpeg::nvenc::get_properties_pre(obs_properties_t* props, const AVCodec*) { { auto p = obs_properties_add_list(props, ST_PRESET, TRANSLATE(ST_PRESET), OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT); obs_property_set_long_description(p, TRANSLATE(DESC(ST_PRESET))); for (auto kv : presets) { obs_property_list_add_int(p, TRANSLATE(kv.second.c_str()), static_cast(kv.first)); } } } void obsffmpeg::nvenc::get_properties_post(obs_properties_t* props, const AVCodec* codec) { { // Rate Control obs_properties_t* grp = props; if (!obsffmpeg::are_property_groups_broken()) { grp = obs_properties_create(); obs_properties_add_group(props, ST_RATECONTROL, TRANSLATE(ST_RATECONTROL), OBS_GROUP_NORMAL, grp); } { auto p = obs_properties_add_list(grp, ST_RATECONTROL_MODE, TRANSLATE(ST_RATECONTROL_MODE), OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_MODE))); obs_property_set_modified_callback(p, modified_ratecontrol); for (auto kv : ratecontrolmodes) { obs_property_list_add_int(p, TRANSLATE(kv.second.c_str()), static_cast(kv.first)); } } { auto p = obsffmpeg::obs_properties_add_tristate(grp, ST_RATECONTROL_TWOPASS, TRANSLATE(ST_RATECONTROL_TWOPASS)); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_TWOPASS))); } { auto p = obs_properties_add_int_slider(grp, ST_RATECONTROL_LOOKAHEAD, TRANSLATE(ST_RATECONTROL_LOOKAHEAD), 0, 32, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_LOOKAHEAD))); obs_property_int_set_suffix(p, " frames"); } { auto p = obsffmpeg::obs_properties_add_tristate(grp, ST_RATECONTROL_ADAPTIVEI, TRANSLATE(ST_RATECONTROL_ADAPTIVEI)); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_ADAPTIVEI))); } if (strcmp(codec->name, "h264_nvenc") == 0) { auto p = obsffmpeg::obs_properties_add_tristate(grp, ST_RATECONTROL_ADAPTIVEB, TRANSLATE(ST_RATECONTROL_ADAPTIVEB)); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_ADAPTIVEB))); } } { obs_properties_t* grp = props; if (!obsffmpeg::are_property_groups_broken()) { grp = obs_properties_create(); obs_properties_add_group(props, ST_RATECONTROL_BITRATE, TRANSLATE(ST_RATECONTROL_BITRATE), OBS_GROUP_NORMAL, grp); } { auto p = obs_properties_add_int(grp, ST_RATECONTROL_BITRATE_TARGET, TRANSLATE(ST_RATECONTROL_BITRATE_TARGET), 1, std::numeric_limits::max(), 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_BITRATE_TARGET))); obs_property_int_set_suffix(p, " kbit/s"); } { auto p = obs_properties_add_int(grp, ST_RATECONTROL_BITRATE_MAXIMUM, TRANSLATE(ST_RATECONTROL_BITRATE_MAXIMUM), 0, std::numeric_limits::max(), 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_BITRATE_MAXIMUM))); obs_property_int_set_suffix(p, " kbit/s"); } { auto p = obs_properties_add_int(grp, S_RATECONTROL_BUFFERSIZE, TRANSLATE(S_RATECONTROL_BUFFERSIZE), 0, std::numeric_limits::max(), 1); obs_property_set_long_description(p, TRANSLATE(DESC(S_RATECONTROL_BUFFERSIZE))); obs_property_int_set_suffix(p, " kbit"); } } { obs_properties_t* grp = props; if (!obsffmpeg::are_property_groups_broken()) { grp = obs_properties_create(); auto p = obs_properties_add_group(props, ST_RATECONTROL_QUALITY, TRANSLATE(ST_RATECONTROL_QUALITY), OBS_GROUP_CHECKABLE, grp); obs_property_set_modified_callback(p, modified_quality); } else { auto p = obs_properties_add_bool(props, ST_RATECONTROL_QUALITY, TRANSLATE(ST_RATECONTROL_QUALITY)); obs_property_set_modified_callback(p, modified_quality); } { auto p = obs_properties_add_int_slider(grp, ST_RATECONTROL_QUALITY_MINIMUM, TRANSLATE(ST_RATECONTROL_QUALITY_MINIMUM), 0, 51, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_QUALITY_MINIMUM))); } { auto p = obs_properties_add_int_slider(grp, ST_RATECONTROL_QUALITY_MAXIMUM, TRANSLATE(ST_RATECONTROL_QUALITY_MAXIMUM), -1, 51, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_QUALITY_MAXIMUM))); } } { auto p = obs_properties_add_float_slider(props, ST_RATECONTROL_QUALITY_TARGET, TRANSLATE(ST_RATECONTROL_QUALITY_TARGET), 0, 100, 0.01); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_QUALITY_TARGET))); } { obs_properties_t* grp = props; if (!obsffmpeg::are_property_groups_broken()) { grp = obs_properties_create(); auto p = obs_properties_add_group(props, ST_RATECONTROL_QP, TRANSLATE(ST_RATECONTROL_QP), OBS_GROUP_CHECKABLE, grp); obs_property_set_modified_callback(p, modified_quality); } { auto p = obs_properties_add_int_slider(grp, ST_RATECONTROL_QP_I, TRANSLATE(ST_RATECONTROL_QP_I), 0, 51, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_QP_I))); } { auto p = obs_properties_add_int_slider(grp, ST_RATECONTROL_QP_I_INITIAL, TRANSLATE(ST_RATECONTROL_QP_I_INITIAL), -1, 51, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_QP_I_INITIAL))); } { auto p = obs_properties_add_int_slider(grp, ST_RATECONTROL_QP_P, TRANSLATE(ST_RATECONTROL_QP_P), 0, 51, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_QP_P))); } { auto p = obs_properties_add_int_slider(grp, ST_RATECONTROL_QP_P_INITIAL, TRANSLATE(ST_RATECONTROL_QP_P_INITIAL), -1, 51, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_QP_P_INITIAL))); } { auto p = obs_properties_add_int_slider(grp, ST_RATECONTROL_QP_B, TRANSLATE(ST_RATECONTROL_QP_B), 0, 51, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_QP_B))); } { auto p = obs_properties_add_int_slider(grp, ST_RATECONTROL_QP_B_INITIAL, TRANSLATE(ST_RATECONTROL_QP_B_INITIAL), -1, 51, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_RATECONTROL_QP_B_INITIAL))); } } { obs_properties_t* grp = props; if (!obsffmpeg::are_property_groups_broken()) { grp = obs_properties_create(); obs_properties_add_group(props, ST_AQ, TRANSLATE(ST_AQ), OBS_GROUP_NORMAL, grp); } { auto p = obsffmpeg::obs_properties_add_tristate(grp, ST_AQ_SPATIAL, TRANSLATE(ST_AQ_SPATIAL)); obs_property_set_long_description(p, TRANSLATE(DESC(ST_AQ_SPATIAL))); obs_property_set_modified_callback(p, modified_aq); } { auto p = obs_properties_add_int_slider(grp, ST_AQ_STRENGTH, TRANSLATE(ST_AQ_STRENGTH), 1, 15, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_AQ_STRENGTH))); } { auto p = obsffmpeg::obs_properties_add_tristate(grp, ST_AQ_TEMPORAL, TRANSLATE(ST_AQ_TEMPORAL)); obs_property_set_long_description(p, TRANSLATE(DESC(ST_AQ_TEMPORAL))); } } { obs_properties_t* grp = props; if (!obsffmpeg::are_property_groups_broken()) { grp = obs_properties_create(); obs_properties_add_group(props, ST_OTHER, TRANSLATE(ST_OTHER), OBS_GROUP_NORMAL, grp); } { auto p = obs_properties_add_int_slider(grp, ST_OTHER_BFRAMES, TRANSLATE(ST_OTHER_BFRAMES), 0, 4, 1); obs_property_set_long_description(p, TRANSLATE(DESC(ST_OTHER_BFRAMES))); obs_property_int_set_suffix(p, " frames"); } { auto p = obs_properties_add_list(grp, ST_OTHER_BFRAME_REFERENCEMODE, TRANSLATE(ST_OTHER_BFRAME_REFERENCEMODE), OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT); obs_property_set_long_description(p, TRANSLATE(DESC(ST_OTHER_BFRAME_REFERENCEMODE))); for (auto kv : b_ref_modes) { obs_property_list_add_int(p, TRANSLATE(kv.second.c_str()), static_cast(kv.first)); } } { auto p = obsffmpeg::obs_properties_add_tristate(grp, ST_OTHER_ZEROLATENCY, TRANSLATE(ST_OTHER_ZEROLATENCY)); obs_property_set_long_description(p, TRANSLATE(DESC(ST_OTHER_ZEROLATENCY))); } { auto p = obsffmpeg::obs_properties_add_tristate(grp, ST_OTHER_WEIGHTED_PREDICTION, TRANSLATE(ST_OTHER_WEIGHTED_PREDICTION)); obs_property_set_long_description(p, TRANSLATE(DESC(ST_OTHER_WEIGHTED_PREDICTION))); } { auto p = obsffmpeg::obs_properties_add_tristate(grp, ST_OTHER_NONREFERENCE_PFRAMES, TRANSLATE(ST_OTHER_NONREFERENCE_PFRAMES)); obs_property_set_long_description(p, TRANSLATE(DESC(ST_OTHER_NONREFERENCE_PFRAMES))); } } } void obsffmpeg::nvenc::get_runtime_properties(obs_properties_t* props, const AVCodec*, AVCodecContext*) { obs_property_set_enabled(obs_properties_get(props, ST_PRESET), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_MODE), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_TWOPASS), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_LOOKAHEAD), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_ADAPTIVEI), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_ADAPTIVEB), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_BITRATE), true); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_BITRATE_TARGET), true); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_BITRATE_MAXIMUM), true); obs_property_set_enabled(obs_properties_get(props, S_RATECONTROL_BUFFERSIZE), true); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QUALITY), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QUALITY_MINIMUM), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QUALITY_MAXIMUM), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QUALITY_TARGET), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QP), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QP_I), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QP_I_INITIAL), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QP_P), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QP_P_INITIAL), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QP_B), false); obs_property_set_enabled(obs_properties_get(props, ST_RATECONTROL_QP_B_INITIAL), false); obs_property_set_enabled(obs_properties_get(props, ST_AQ), false); obs_property_set_enabled(obs_properties_get(props, ST_AQ_SPATIAL), false); obs_property_set_enabled(obs_properties_get(props, ST_AQ_STRENGTH), false); obs_property_set_enabled(obs_properties_get(props, ST_AQ_TEMPORAL), false); obs_property_set_enabled(obs_properties_get(props, ST_OTHER), false); obs_property_set_enabled(obs_properties_get(props, ST_OTHER_BFRAMES), false); obs_property_set_enabled(obs_properties_get(props, ST_OTHER_BFRAME_REFERENCEMODE), false); obs_property_set_enabled(obs_properties_get(props, ST_OTHER_ZEROLATENCY), false); obs_property_set_enabled(obs_properties_get(props, ST_OTHER_WEIGHTED_PREDICTION), false); obs_property_set_enabled(obs_properties_get(props, ST_OTHER_NONREFERENCE_PFRAMES), false); } void obsffmpeg::nvenc::update(obs_data_t* settings, const AVCodec* codec, AVCodecContext* context) { { preset c_preset = static_cast(obs_data_get_int(settings, ST_PRESET)); auto found = preset_to_opt.find(c_preset); if (found != preset_to_opt.end()) { av_opt_set(context->priv_data, "preset", found->second.c_str(), 0); } else { av_opt_set(context->priv_data, "preset", nullptr, 0); } } { // Rate Control bool have_bitrate = false; bool have_bitrate_max = false; bool have_quality = false; bool have_qp = false; bool have_qp_init = false; ratecontrolmode rc = static_cast(obs_data_get_int(settings, ST_RATECONTROL_MODE)); auto rcopt = nvenc::ratecontrolmode_to_opt.find(rc); if (rcopt != nvenc::ratecontrolmode_to_opt.end()) { av_opt_set(context->priv_data, "rc", rcopt->second.c_str(), 0); } av_opt_set_int(context->priv_data, "cbr", 0, 0); switch (rc) { case ratecontrolmode::CQP: have_qp = true; break; case ratecontrolmode::CBR: case ratecontrolmode::CBR_HQ: case ratecontrolmode::CBR_LD_HQ: have_bitrate = true; av_opt_set_int(context->priv_data, "cbr", 1, 0); break; case ratecontrolmode::VBR: case ratecontrolmode::VBR_HQ: have_bitrate_max = true; have_bitrate = true; have_quality = true; have_qp_init = true; break; } int tp = static_cast(obs_data_get_int(settings, ST_RATECONTROL_TWOPASS)); if (tp >= 0) { av_opt_set_int(context->priv_data, "2pass", tp ? 1 : 0, 0); } int la = static_cast(obs_data_get_int(settings, ST_RATECONTROL_LOOKAHEAD)); av_opt_set_int(context->priv_data, "rc-lookahead", la, 0); if (la > 0) { int64_t adapt_i = obs_data_get_int(settings, ST_RATECONTROL_ADAPTIVEI); if (!is_tristate_default(adapt_i)) { av_opt_set_int(context->priv_data, "no-scenecut", adapt_i, AV_OPT_SEARCH_CHILDREN); } if (strcmp(codec->name, "h264_nvenc")) { int64_t adapt_b = obs_data_get_int(settings, ST_RATECONTROL_ADAPTIVEB); if (!is_tristate_default(adapt_b)) { av_opt_set_int(context->priv_data, "b_adapt", adapt_b, AV_OPT_SEARCH_CHILDREN); } } } if (have_bitrate) { context->bit_rate = static_cast(obs_data_get_int(settings, ST_RATECONTROL_BITRATE_TARGET) * 1000); // Support for Replay Buffer obs_data_set_int(settings, "bitrate", obs_data_get_int(settings, ST_RATECONTROL_BITRATE_TARGET)); } if (have_bitrate_max) context->rc_max_rate = static_cast(obs_data_get_int(settings, ST_RATECONTROL_BITRATE_MAXIMUM) * 1000); if (have_bitrate || have_bitrate_max) context->rc_buffer_size = static_cast(obs_data_get_int(settings, S_RATECONTROL_BUFFERSIZE) * 1000); if (have_quality && obs_data_get_bool(settings, ST_RATECONTROL_QUALITY)) { int qmin = static_cast(obs_data_get_int(settings, ST_RATECONTROL_QUALITY_MINIMUM)); context->qmin = qmin; if (qmin >= 0) { context->qmax = static_cast(obs_data_get_int(settings, ST_RATECONTROL_QUALITY_MAXIMUM)); } } { double_t v = obs_data_get_double(settings, ST_RATECONTROL_QUALITY_TARGET) / 100.0 * 51.0; if (v > 0) { av_opt_set_double(context->priv_data, "cq", v, 0); } } if (have_qp) { av_opt_set_int(context->priv_data, "init_qpI", static_cast(obs_data_get_int(settings, ST_RATECONTROL_QP_I)), 0); av_opt_set_int(context->priv_data, "init_qpP", static_cast(obs_data_get_int(settings, ST_RATECONTROL_QP_P)), 0); av_opt_set_int(context->priv_data, "init_qpB", static_cast(obs_data_get_int(settings, ST_RATECONTROL_QP_B)), 0); } if (have_qp_init) { av_opt_set_int(context->priv_data, "init_qpI", obs_data_get_int(settings, ST_RATECONTROL_QP_I_INITIAL), 0); av_opt_set_int(context->priv_data, "init_qpP", obs_data_get_int(settings, ST_RATECONTROL_QP_P_INITIAL), 0); av_opt_set_int(context->priv_data, "init_qpB", obs_data_get_int(settings, ST_RATECONTROL_QP_B_INITIAL), 0); } } { // AQ int64_t saq = obs_data_get_int(settings, ST_AQ_SPATIAL); int64_t taq = obs_data_get_int(settings, ST_AQ_TEMPORAL); if (strcmp(codec->name, "h264_nvenc") == 0) { if (!is_tristate_default(saq)) av_opt_set_int(context->priv_data, "spatial-aq", saq, 0); if (!is_tristate_default(taq)) av_opt_set_int(context->priv_data, "temporal-aq", taq, 0); } else { if (!is_tristate_default(saq)) av_opt_set_int(context->priv_data, "spatial_aq", saq, 0); if (!is_tristate_default(taq)) av_opt_set_int(context->priv_data, "temporal_aq", taq, 0); } if (is_tristate_enabled(saq)) av_opt_set_int(context->priv_data, "aq-strength", static_cast(obs_data_get_int(settings, ST_AQ_STRENGTH)), 0); } { // Other int64_t zl = obs_data_get_int(settings, ST_OTHER_ZEROLATENCY); int64_t wp = obs_data_get_int(settings, ST_OTHER_WEIGHTED_PREDICTION); int64_t nrp = obs_data_get_int(settings, ST_OTHER_NONREFERENCE_PFRAMES); context->max_b_frames = static_cast(obs_data_get_int(settings, ST_OTHER_BFRAMES)); if (!is_tristate_default(zl)) av_opt_set_int(context->priv_data, "zerolatency", zl, 0); if (!is_tristate_default(nrp)) av_opt_set_int(context->priv_data, "nonref_p", nrp, 0); if ((context->max_b_frames != 0) && is_tristate_enabled(wp)) { PLOG_WARNING("[%s] Weighted Prediction disabled because of B-Frames being used.", codec->name); av_opt_set_int(context->priv_data, "weighted_pred", 0, 0); } else if (!is_tristate_default(wp)) { av_opt_set_int(context->priv_data, "weighted_pred", wp, 0); } { auto found = b_ref_mode_to_opt.find( static_cast(obs_data_get_int(settings, ST_OTHER_BFRAME_REFERENCEMODE))); if (found != b_ref_mode_to_opt.end()) { av_opt_set(context->priv_data, "b_ref_mode", found->second.c_str(), 0); } } } } void obsffmpeg::nvenc::log_options(obs_data_t*, const AVCodec* codec, AVCodecContext* context) { PLOG_INFO("[%s] Nvidia NVENC:", codec->name); ffmpeg::tools::print_av_option_string(context, "preset", " Preset", [](int64_t v) { preset val = static_cast(v); std::string name = ""; auto index = preset_to_opt.find(val); if (index != preset_to_opt.end()) name = index->second; return name; }); ffmpeg::tools::print_av_option_string(context, "rc", " Rate Control", [](int64_t v) { ratecontrolmode val = static_cast(v); std::string name = ""; auto index = ratecontrolmode_to_opt.find(val); if (index != ratecontrolmode_to_opt.end()) name = index->second; return name; }); ffmpeg::tools::print_av_option_bool(context, "2pass", " Two Pass"); ffmpeg::tools::print_av_option_int(context, "rc-lookahead", " Look-Ahead", "Frames"); ffmpeg::tools::print_av_option_bool(context, "no-scenecut", " Adaptive I-Frames"); if (strcmp(codec->name, "h264_nvenc") == 0) ffmpeg::tools::print_av_option_bool(context, "b_adapt", " Adaptive B-Frames"); PLOG_INFO("[%s] Bitrate:", codec->name); ffmpeg::tools::print_av_option_int(context, "bitrate", " Target", "bits/sec"); ffmpeg::tools::print_av_option_int(context, "rc_max_rate", " Maximum", "bits/sec"); ffmpeg::tools::print_av_option_int(context, "rc_buffer_size", " Buffer", "bits"); PLOG_INFO("[%s] Quality:", codec->name); ffmpeg::tools::print_av_option_int(context, "qmin", " Minimum", ""); ffmpeg::tools::print_av_option_int(context, "cq", " Target", ""); ffmpeg::tools::print_av_option_int(context, "qmax", " Maximum", ""); PLOG_INFO("[%s] Quantization Parameters:", codec->name); ffmpeg::tools::print_av_option_int(context, "init_qpI", " I-Frame", ""); ffmpeg::tools::print_av_option_int(context, "init_qpP", " P-Frame", ""); ffmpeg::tools::print_av_option_int(context, "init_qpB", " B-Frame", ""); ffmpeg::tools::print_av_option_int(context, "max_b_frames", " B-Frames", "Frames"); ffmpeg::tools::print_av_option_string(context, "b_ref_mode", " Reference Mode", [](int64_t v) { b_ref_mode val = static_cast(v); std::string name = ""; auto index = b_ref_mode_to_opt.find(val); if (index != b_ref_mode_to_opt.end()) name = index->second; return name; }); PLOG_INFO("[%s] Adaptive Quantization:", codec->name); if (strcmp(codec->name, "h264_nvenc") == 0) { ffmpeg::tools::print_av_option_bool(context, "spatial-aq", " Spatial AQ"); ffmpeg::tools::print_av_option_int(context, "aq-strength", " Strength", ""); ffmpeg::tools::print_av_option_bool(context, "temporal-aq", " Temporal AQ"); } else { ffmpeg::tools::print_av_option_bool(context, "spatial_aq", " Spatial AQ"); ffmpeg::tools::print_av_option_int(context, "aq-strength", " Strength", ""); ffmpeg::tools::print_av_option_bool(context, "temporal_aq", " Temporal AQ"); } PLOG_INFO("[%s] Other:", codec->name); ffmpeg::tools::print_av_option_bool(context, "zerolatency", " Zero Latency"); ffmpeg::tools::print_av_option_bool(context, "weighted_pred", " Weighted Prediction"); ffmpeg::tools::print_av_option_bool(context, "nonref_p", " Non-reference P-Frames"); ffmpeg::tools::print_av_option_bool(context, "strict_gop", " Strict GOP"); ffmpeg::tools::print_av_option_bool(context, "aud", " Access Unit Delimiters"); ffmpeg::tools::print_av_option_bool(context, "bluray-compat", " Bluray Compatibility"); if (strcmp(codec->name, "h264_nvenc") == 0) ffmpeg::tools::print_av_option_bool(context, "a53cc", " A53 Closed Captions"); ffmpeg::tools::print_av_option_int(context, "dpb_size", " DPB Size", ""); }