From e509fed8b1601be26300f86a96c5b1882bb35bf0 Mon Sep 17 00:00:00 2001 From: Tobias Ollmann Date: Mon, 15 Jul 2024 08:57:44 +0200 Subject: [PATCH 1/2] fix: correctly map selected layers to simulcast layers this partly fixes #133. Quick summary: - on lower resolutions, simulcast only sends 2 (or even 1) layer. - The layers are ordered from low to high resolution (downscale factors 4, 2, 1) - previously, when only sending 2 layers (factors 2, 1), the layer with factor 2 was mapped to the layer with factor 4, i.e. scaled down again) - with this patch, the layers are mapped to the correct factor and not scaled down again --- video/config/encoder_stream_factory.cc | 65 ++++++++++++++------------ 1 file changed, 34 insertions(+), 31 deletions(-) diff --git a/video/config/encoder_stream_factory.cc b/video/config/encoder_stream_factory.cc index 499fdb5b1c..682649e226 100644 --- a/video/config/encoder_stream_factory.cc +++ b/video/config/encoder_stream_factory.cc @@ -354,71 +354,74 @@ EncoderStreamFactory::CreateSimulcastOrConferenceModeScreenshareStreams( ? NormalizeSimulcastSize(trials_, height, encoder_config.number_of_streams) : height; + for (size_t i = 0; i < layers.size(); ++i) { - layers[i].active = encoder_config.simulcast_layers[i].active; - layers[i].scalability_mode = - encoder_config.simulcast_layers[i].scalability_mode; - layers[i].requested_resolution = - encoder_config.simulcast_layers[i].requested_resolution; + // account for layers removed in `GetSimulcastConfig`. Layers are ordered + // from lowest to highest quality, including their + // `scale_down_resolution_by` factor. So we need to map the last + // simulcast_layer to the last layer, not the other way round. Otherwise the + // scaling factor is applied multiple times + + size_t simulcast_layer_index = + i + std::max(1, encoder_config.simulcast_layers.size() - layers.size()); + auto& simulcast_layer = encoder_config.simulcast_layers[simulcast_layer_i]; + layers[i].active = simulcast_layer.active; + layers[i].scalability_mode = simulcast_layer.scalability_mode; + layers[i].requested_resolution = simulcast_layer.requested_resolution; // Update with configured num temporal layers if supported by codec. - if (encoder_config.simulcast_layers[i].num_temporal_layers && + if (simulcast_layer.num_temporal_layers && IsTemporalLayersSupported(codec_name_)) { - layers[i].num_temporal_layers = - *encoder_config.simulcast_layers[i].num_temporal_layers; + layers[i].num_temporal_layers = *simulcast_layer.num_temporal_layers; } - if (encoder_config.simulcast_layers[i].max_framerate > 0) { - layers[i].max_framerate = - encoder_config.simulcast_layers[i].max_framerate; + if (simulcast_layer.max_framerate > 0) { + layers[i].max_framerate = simulcast_layer.max_framerate; } - if (encoder_config.simulcast_layers[i].requested_resolution.has_value()) { + if (simulcast_layer.requested_resolution.has_value()) { auto res = GetLayerResolutionFromRequestedResolution( normalized_width, normalized_height, - *encoder_config.simulcast_layers[i].requested_resolution); + *simulcast_layer.requested_resolution); layers[i].width = res.width; layers[i].height = res.height; } else if (has_scale_resolution_down_by) { - const double scale_resolution_down_by = std::max( - encoder_config.simulcast_layers[i].scale_resolution_down_by, 1.0); + const double scale_resolution_down_by = + std::max(simulcast_layer.scale_resolution_down_by, 1.0); layers[i].width = ScaleDownResolution( normalized_width, scale_resolution_down_by, kMinLayerSize); layers[i].height = ScaleDownResolution( normalized_height, scale_resolution_down_by, kMinLayerSize); } // Update simulcast bitrates with configured min and max bitrate. - if (encoder_config.simulcast_layers[i].min_bitrate_bps > 0) { - layers[i].min_bitrate_bps = - encoder_config.simulcast_layers[i].min_bitrate_bps; + if (simulcast_layer.min_bitrate_bps > 0) { + layers[i].min_bitrate_bps = simulcast_layer.min_bitrate_bps; } - if (encoder_config.simulcast_layers[i].max_bitrate_bps > 0) { - layers[i].max_bitrate_bps = - encoder_config.simulcast_layers[i].max_bitrate_bps; + if (simulcast_layer.max_bitrate_bps > 0) { + layers[i].max_bitrate_bps = simulcast_layer.max_bitrate_bps; } - if (encoder_config.simulcast_layers[i].target_bitrate_bps > 0) { - layers[i].target_bitrate_bps = - encoder_config.simulcast_layers[i].target_bitrate_bps; + if (simulcast_layer.target_bitrate_bps > 0) { + layers[i].target_bitrate_bps = simulcast_layer.target_bitrate_bps; } - if (encoder_config.simulcast_layers[i].min_bitrate_bps > 0 && - encoder_config.simulcast_layers[i].max_bitrate_bps > 0) { + if (simulcast_layer.min_bitrate_bps > 0 && + simulcast_layer.max_bitrate_bps > 0) { // Min and max bitrate are configured. // Set target to 3/4 of the max bitrate (or to max if below min). - if (encoder_config.simulcast_layers[i].target_bitrate_bps <= 0) + if (simulcast_layer.target_bitrate_bps <= 0) layers[i].target_bitrate_bps = layers[i].max_bitrate_bps * 3 / 4; if (layers[i].target_bitrate_bps < layers[i].min_bitrate_bps) layers[i].target_bitrate_bps = layers[i].max_bitrate_bps; - } else if (encoder_config.simulcast_layers[i].min_bitrate_bps > 0) { + } else if (simulcast_layer.min_bitrate_bps > 0) { // Only min bitrate is configured, make sure target/max are above min. layers[i].target_bitrate_bps = std::max(layers[i].target_bitrate_bps, layers[i].min_bitrate_bps); layers[i].max_bitrate_bps = std::max(layers[i].max_bitrate_bps, layers[i].min_bitrate_bps); - } else if (encoder_config.simulcast_layers[i].max_bitrate_bps > 0) { + } else if (simulcast_layer.max_bitrate_bps > 0) { // Only max bitrate is configured, make sure min/target are below max. // Keep target bitrate if it is set explicitly in encoding config. // Otherwise set target bitrate to 3/4 of the max bitrate // or the one calculated from GetSimulcastConfig() which is larger. layers[i].min_bitrate_bps = std::min(layers[i].min_bitrate_bps, layers[i].max_bitrate_bps); - if (encoder_config.simulcast_layers[i].target_bitrate_bps <= 0) { + if (simulcast_layer.target_bitrate_bps <= 0) { layers[i].target_bitrate_bps = std::max( layers[i].target_bitrate_bps, layers[i].max_bitrate_bps * 3 / 4); } @@ -428,7 +431,7 @@ EncoderStreamFactory::CreateSimulcastOrConferenceModeScreenshareStreams( } if (i == layers.size() - 1) { is_highest_layer_max_bitrate_configured = - encoder_config.simulcast_layers[i].max_bitrate_bps > 0; + simulcast_layer.max_bitrate_bps > 0; } } if (!is_screenshare_ && !is_highest_layer_max_bitrate_configured && From 021da3c144386cfdf6ca289cd71730c3df69d801 Mon Sep 17 00:00:00 2001 From: Tobias Ollmann Date: Mon, 15 Jul 2024 09:06:07 +0200 Subject: [PATCH 2/2] fix: properly set mapping in case of 3 layers encoder_config.simulcast_layers.size() is always >= layers.size(), otherwise it not have worked before, so get rid of the (erronous) max --- video/config/encoder_stream_factory.cc | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/video/config/encoder_stream_factory.cc b/video/config/encoder_stream_factory.cc index 682649e226..81d6256f72 100644 --- a/video/config/encoder_stream_factory.cc +++ b/video/config/encoder_stream_factory.cc @@ -363,7 +363,7 @@ EncoderStreamFactory::CreateSimulcastOrConferenceModeScreenshareStreams( // scaling factor is applied multiple times size_t simulcast_layer_index = - i + std::max(1, encoder_config.simulcast_layers.size() - layers.size()); + i + encoder_config.simulcast_layers.size() - layers.size(); auto& simulcast_layer = encoder_config.simulcast_layers[simulcast_layer_i]; layers[i].active = simulcast_layer.active; layers[i].scalability_mode = simulcast_layer.scalability_mode;