/* * Copyright (c) 2020 The WebRTC project authors. All Rights Reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #include "modules/video_coding/svc/create_scalability_structure.h" #include #include "api/video_codecs/scalability_mode.h" #include "modules/video_coding/svc/scalability_structure_full_svc.h" #include "modules/video_coding/svc/scalability_structure_key_svc.h" #include "modules/video_coding/svc/scalability_structure_l2t2_key_shift.h" #include "modules/video_coding/svc/scalability_structure_simulcast.h" #include "modules/video_coding/svc/scalable_video_controller.h" #include "modules/video_coding/svc/scalable_video_controller_no_layering.h" #include "rtc_base/checks.h" namespace webrtc { namespace { struct NamedStructureFactory { ScalabilityMode name; // Use function pointer to make NamedStructureFactory trivally destructable. std::unique_ptr (*factory)(); ScalableVideoController::StreamLayersConfig config; }; // Wrap std::make_unique function to have correct return type. template std::unique_ptr Create() { return std::make_unique(); } template std::unique_ptr CreateH() { // 1.5:1 scaling, see https://w3c.github.io/webrtc-svc/#scalabilitymodes* typename T::ScalingFactor factor; factor.num = 2; factor.den = 3; return std::make_unique(factor); } constexpr ScalableVideoController::StreamLayersConfig kConfigL1T1 = { /*num_spatial_layers=*/1, /*num_temporal_layers=*/1, /*uses_reference_scaling=*/false}; constexpr ScalableVideoController::StreamLayersConfig kConfigL1T2 = { /*num_spatial_layers=*/1, /*num_temporal_layers=*/2, /*uses_reference_scaling=*/false}; constexpr ScalableVideoController::StreamLayersConfig kConfigL1T3 = { /*num_spatial_layers=*/1, /*num_temporal_layers=*/3, /*uses_reference_scaling=*/false}; constexpr ScalableVideoController::StreamLayersConfig kConfigL2T1 = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/1, /*uses_reference_scaling=*/true, {1, 1}, {2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL2T1h = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/1, /*uses_reference_scaling=*/true, {2, 1}, {3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL2T2 = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/2, /*uses_reference_scaling=*/true, {1, 1}, {2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL2T2h = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/2, /*uses_reference_scaling=*/true, {2, 1}, {3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL2T3 = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/3, /*uses_reference_scaling=*/true, {1, 1}, {2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL2T3h = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/3, /*uses_reference_scaling=*/true, {2, 1}, {3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL3T1 = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/1, /*uses_reference_scaling=*/true, {1, 1, 1}, {4, 2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL3T1h = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/1, /*uses_reference_scaling=*/true, {4, 2, 1}, {9, 3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL3T2 = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/2, /*uses_reference_scaling=*/true, {1, 1, 1}, {4, 2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL3T2h = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/2, /*uses_reference_scaling=*/true, {4, 2, 1}, {9, 3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL3T3 = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/3, /*uses_reference_scaling=*/true, {1, 1, 1}, {4, 2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigL3T3h = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/3, /*uses_reference_scaling=*/true, {4, 2, 1}, {9, 3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS2T1 = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/1, /*uses_reference_scaling=*/false, {1, 1}, {2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS2T1h = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/1, /*uses_reference_scaling=*/false, {2, 1}, {3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS2T2 = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/2, /*uses_reference_scaling=*/false, {1, 1}, {2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS2T2h = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/2, /*uses_reference_scaling=*/false, {2, 1}, {3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS2T3 = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/3, /*uses_reference_scaling=*/false, {1, 1}, {2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS2T3h = { /*num_spatial_layers=*/2, /*num_temporal_layers=*/3, /*uses_reference_scaling=*/false, {2, 1}, {3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS3T1 = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/1, /*uses_reference_scaling=*/false, {1, 1, 1}, {4, 2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS3T1h = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/1, /*uses_reference_scaling=*/false, {4, 2, 1}, {9, 3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS3T2 = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/2, /*uses_reference_scaling=*/false, {1, 1, 1}, {4, 2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS3T2h = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/2, /*uses_reference_scaling=*/false, {4, 2, 1}, {9, 3, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS3T3 = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/3, /*uses_reference_scaling=*/false, {1, 1, 1}, {4, 2, 1}}; constexpr ScalableVideoController::StreamLayersConfig kConfigS3T3h = { /*num_spatial_layers=*/3, /*num_temporal_layers=*/3, /*uses_reference_scaling=*/false, {4, 2, 1}, {9, 3, 1}}; constexpr NamedStructureFactory kFactories[] = { {ScalabilityMode::kL1T1, Create, kConfigL1T1}, {ScalabilityMode::kL1T2, Create, kConfigL1T2}, {ScalabilityMode::kL1T3, Create, kConfigL1T3}, {ScalabilityMode::kL2T1, Create, kConfigL2T1}, {ScalabilityMode::kL2T1h, CreateH, kConfigL2T1h}, {ScalabilityMode::kL2T1_KEY, Create, kConfigL2T1}, {ScalabilityMode::kL2T2, Create, kConfigL2T2}, {ScalabilityMode::kL2T2h, CreateH, kConfigL2T2h}, {ScalabilityMode::kL2T2_KEY, Create, kConfigL2T2}, {ScalabilityMode::kL2T2_KEY_SHIFT, Create, kConfigL2T2}, {ScalabilityMode::kL2T3, Create, kConfigL2T3}, {ScalabilityMode::kL2T3h, CreateH, kConfigL2T3h}, {ScalabilityMode::kL2T3_KEY, Create, kConfigL2T3}, {ScalabilityMode::kL3T1, Create, kConfigL3T1}, {ScalabilityMode::kL3T1h, CreateH, kConfigL3T1h}, {ScalabilityMode::kL3T1_KEY, Create, kConfigL3T1}, {ScalabilityMode::kL3T2, Create, kConfigL3T2}, {ScalabilityMode::kL3T2h, CreateH, kConfigL3T2h}, {ScalabilityMode::kL3T2_KEY, Create, kConfigL3T2}, {ScalabilityMode::kL3T3, Create, kConfigL3T3}, {ScalabilityMode::kL3T3h, CreateH, kConfigL3T3h}, {ScalabilityMode::kL3T3_KEY, Create, kConfigL3T3}, {ScalabilityMode::kS2T1, Create, kConfigS2T1}, {ScalabilityMode::kS2T1h, CreateH, kConfigS2T1h}, {ScalabilityMode::kS2T2, Create, kConfigS2T2}, {ScalabilityMode::kS2T2h, CreateH, kConfigS2T2h}, {ScalabilityMode::kS2T3, Create, kConfigS2T3}, {ScalabilityMode::kS2T3h, CreateH, kConfigS2T3h}, {ScalabilityMode::kS3T1, Create, kConfigS3T1}, {ScalabilityMode::kS3T1h, CreateH, kConfigS3T1h}, {ScalabilityMode::kS3T2, Create, kConfigS3T2}, {ScalabilityMode::kS3T2h, CreateH, kConfigS3T2h}, {ScalabilityMode::kS3T3, Create, kConfigS3T3}, {ScalabilityMode::kS3T3h, CreateH, kConfigS3T3h}, }; } // namespace std::unique_ptr CreateScalabilityStructure( ScalabilityMode name) { for (const auto& entry : kFactories) { if (entry.name == name) { return entry.factory(); } } return nullptr; } absl::optional ScalabilityStructureConfig(ScalabilityMode name) { for (const auto& entry : kFactories) { if (entry.name == name) { return entry.config; } } return absl::nullopt; } } // namespace webrtc