blob: 3cc08b27994d3e17ca568e6cbfa58dce1c20c3c1 [file] [log] [blame]
Niels Möllerdac03d92019-02-13 08:52:27 +01001/*
2 * Copyright 2013 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11#include "sdk/media_constraints.h"
12
13#include "absl/types/optional.h"
14#include "api/peer_connection_interface.h"
15
16namespace webrtc {
17namespace {
18
19// Find the highest-priority instance of the T-valued constraint named by
Artem Titovd7ac5812021-07-27 12:23:39 +020020// `key` and return its value as `value`. `constraints` can be null.
21// If `mandatory_constraints` is non-null, it is incremented if the key appears
Niels Möllerdac03d92019-02-13 08:52:27 +010022// among the mandatory constraints.
23// Returns true if the key was found and has a valid value for type T.
24// If the key appears multiple times as an optional constraint, appearances
25// after the first are ignored.
26// Note: Because this uses FindFirst, repeated optional constraints whose
27// first instance has an unrecognized value are not handled precisely in
28// accordance with the specification.
29template <typename T>
30bool FindConstraint(const MediaConstraints* constraints,
31 const std::string& key,
32 T* value,
33 size_t* mandatory_constraints) {
34 std::string string_value;
35 if (!FindConstraint(constraints, key, &string_value, mandatory_constraints)) {
36 return false;
37 }
38 return rtc::FromString(string_value, value);
39}
40
41// Specialization for std::string, since a string doesn't need conversion.
42template <>
43bool FindConstraint(const MediaConstraints* constraints,
44 const std::string& key,
45 std::string* value,
46 size_t* mandatory_constraints) {
47 if (!constraints) {
48 return false;
49 }
50 if (constraints->GetMandatory().FindFirst(key, value)) {
51 if (mandatory_constraints) {
52 ++*mandatory_constraints;
53 }
54 return true;
55 }
56 if (constraints->GetOptional().FindFirst(key, value)) {
57 return true;
58 }
59 return false;
60}
61
62bool FindConstraint(const MediaConstraints* constraints,
63 const std::string& key,
64 bool* value,
65 size_t* mandatory_constraints) {
66 return FindConstraint<bool>(constraints, key, value, mandatory_constraints);
67}
68
69bool FindConstraint(const MediaConstraints* constraints,
70 const std::string& key,
71 int* value,
72 size_t* mandatory_constraints) {
73 return FindConstraint<int>(constraints, key, value, mandatory_constraints);
74}
75
76// Converts a constraint (mandatory takes precedence over optional) to an
77// absl::optional.
78template <typename T>
79void ConstraintToOptional(const MediaConstraints* constraints,
80 const std::string& key,
81 absl::optional<T>* value_out) {
82 T value;
83 bool present = FindConstraint<T>(constraints, key, &value, nullptr);
84 if (present) {
85 *value_out = value;
86 }
87}
88} // namespace
89
90const char MediaConstraints::kValueTrue[] = "true";
91const char MediaConstraints::kValueFalse[] = "false";
92
93// Constraints declared as static members in mediastreaminterface.h
94
95// Audio constraints.
96const char MediaConstraints::kGoogEchoCancellation[] = "googEchoCancellation";
Niels Möllerdac03d92019-02-13 08:52:27 +010097const char MediaConstraints::kAutoGainControl[] = "googAutoGainControl";
98const char MediaConstraints::kExperimentalAutoGainControl[] =
99 "googAutoGainControl2";
100const char MediaConstraints::kNoiseSuppression[] = "googNoiseSuppression";
101const char MediaConstraints::kExperimentalNoiseSuppression[] =
102 "googNoiseSuppression2";
103const char MediaConstraints::kHighpassFilter[] = "googHighpassFilter";
104const char MediaConstraints::kTypingNoiseDetection[] =
105 "googTypingNoiseDetection";
106const char MediaConstraints::kAudioMirroring[] = "googAudioMirroring";
107const char MediaConstraints::kAudioNetworkAdaptorConfig[] =
108 "googAudioNetworkAdaptorConfig";
Xavier Lepaul1e12f2a2022-01-13 17:06:26 +0100109const char MediaConstraints::kInitAudioRecordingOnSend[] =
110 "InitAudioRecordingOnSend";
Niels Möllerdac03d92019-02-13 08:52:27 +0100111
112// Constraint keys for CreateOffer / CreateAnswer defined in W3C specification.
113const char MediaConstraints::kOfferToReceiveAudio[] = "OfferToReceiveAudio";
114const char MediaConstraints::kOfferToReceiveVideo[] = "OfferToReceiveVideo";
115const char MediaConstraints::kVoiceActivityDetection[] =
116 "VoiceActivityDetection";
117const char MediaConstraints::kIceRestart[] = "IceRestart";
118// Google specific constraint for BUNDLE enable/disable.
119const char MediaConstraints::kUseRtpMux[] = "googUseRtpMUX";
120
121// Below constraints should be used during PeerConnection construction.
Niels Möllerdac03d92019-02-13 08:52:27 +0100122// Google-specific constraint keys.
123const char MediaConstraints::kEnableDscp[] = "googDscp";
124const char MediaConstraints::kEnableIPv6[] = "googIPv6";
125const char MediaConstraints::kEnableVideoSuspendBelowMinBitrate[] =
126 "googSuspendBelowMinBitrate";
127const char MediaConstraints::kCombinedAudioVideoBwe[] =
128 "googCombinedAudioVideoBwe";
129const char MediaConstraints::kScreencastMinBitrate[] =
130 "googScreencastMinBitrate";
131// TODO(ronghuawu): Remove once cpu overuse detection is stable.
132const char MediaConstraints::kCpuOveruseDetection[] = "googCpuOveruseDetection";
133
Mirta Dvornicic479a3c02019-06-04 15:38:50 +0200134const char MediaConstraints::kRawPacketizationForVideoEnabled[] =
135 "googRawPacketizationForVideoEnabled";
136
Niels Möllerdac03d92019-02-13 08:52:27 +0100137const char MediaConstraints::kNumSimulcastLayers[] = "googNumSimulcastLayers";
138
Artem Titovd7ac5812021-07-27 12:23:39 +0200139// Set `value` to the value associated with the first appearance of `key`, or
140// return false if `key` is not found.
Niels Möllerdac03d92019-02-13 08:52:27 +0100141bool MediaConstraints::Constraints::FindFirst(const std::string& key,
142 std::string* value) const {
143 for (Constraints::const_iterator iter = begin(); iter != end(); ++iter) {
144 if (iter->key == key) {
145 *value = iter->value;
146 return true;
147 }
148 }
149 return false;
150}
151
152void CopyConstraintsIntoRtcConfiguration(
153 const MediaConstraints* constraints,
154 PeerConnectionInterface::RTCConfiguration* configuration) {
155 // Copy info from constraints into configuration, if present.
156 if (!constraints) {
157 return;
158 }
159
160 bool enable_ipv6;
161 if (FindConstraint(constraints, MediaConstraints::kEnableIPv6, &enable_ipv6,
162 nullptr)) {
163 configuration->disable_ipv6 = !enable_ipv6;
164 }
165 FindConstraint(constraints, MediaConstraints::kEnableDscp,
166 &configuration->media_config.enable_dscp, nullptr);
167 FindConstraint(constraints, MediaConstraints::kCpuOveruseDetection,
168 &configuration->media_config.video.enable_cpu_adaptation,
169 nullptr);
Niels Möllerdac03d92019-02-13 08:52:27 +0100170 // Find Suspend Below Min Bitrate constraint.
171 FindConstraint(
172 constraints, MediaConstraints::kEnableVideoSuspendBelowMinBitrate,
173 &configuration->media_config.video.suspend_below_min_bitrate, nullptr);
174 ConstraintToOptional<int>(constraints,
175 MediaConstraints::kScreencastMinBitrate,
176 &configuration->screencast_min_bitrate);
177 ConstraintToOptional<bool>(constraints,
178 MediaConstraints::kCombinedAudioVideoBwe,
179 &configuration->combined_audio_video_bwe);
Niels Möllerdac03d92019-02-13 08:52:27 +0100180}
181
182void CopyConstraintsIntoAudioOptions(const MediaConstraints* constraints,
183 cricket::AudioOptions* options) {
184 if (!constraints) {
185 return;
186 }
187
188 ConstraintToOptional<bool>(constraints,
189 MediaConstraints::kGoogEchoCancellation,
190 &options->echo_cancellation);
Niels Möllerdac03d92019-02-13 08:52:27 +0100191 ConstraintToOptional<bool>(constraints, MediaConstraints::kAutoGainControl,
192 &options->auto_gain_control);
193 ConstraintToOptional<bool>(constraints,
194 MediaConstraints::kExperimentalAutoGainControl,
195 &options->experimental_agc);
196 ConstraintToOptional<bool>(constraints, MediaConstraints::kNoiseSuppression,
197 &options->noise_suppression);
198 ConstraintToOptional<bool>(constraints,
199 MediaConstraints::kExperimentalNoiseSuppression,
200 &options->experimental_ns);
201 ConstraintToOptional<bool>(constraints, MediaConstraints::kHighpassFilter,
202 &options->highpass_filter);
203 ConstraintToOptional<bool>(constraints,
204 MediaConstraints::kTypingNoiseDetection,
205 &options->typing_detection);
206 ConstraintToOptional<bool>(constraints, MediaConstraints::kAudioMirroring,
207 &options->stereo_swapping);
208 ConstraintToOptional<std::string>(
209 constraints, MediaConstraints::kAudioNetworkAdaptorConfig,
210 &options->audio_network_adaptor_config);
Artem Titovd7ac5812021-07-27 12:23:39 +0200211 // When `kAudioNetworkAdaptorConfig` is defined, it both means that audio
Niels Möllerdac03d92019-02-13 08:52:27 +0100212 // network adaptor is desired, and provides the config string.
213 if (options->audio_network_adaptor_config) {
214 options->audio_network_adaptor = true;
215 }
Xavier Lepaul1e12f2a2022-01-13 17:06:26 +0100216 ConstraintToOptional<bool>(constraints,
217 MediaConstraints::kInitAudioRecordingOnSend,
218 &options->init_recording_on_send);
Niels Möllerdac03d92019-02-13 08:52:27 +0100219}
220
221bool CopyConstraintsIntoOfferAnswerOptions(
222 const MediaConstraints* constraints,
223 PeerConnectionInterface::RTCOfferAnswerOptions* offer_answer_options) {
224 if (!constraints) {
225 return true;
226 }
227
228 bool value = false;
229 size_t mandatory_constraints_satisfied = 0;
230
231 if (FindConstraint(constraints, MediaConstraints::kOfferToReceiveAudio,
232 &value, &mandatory_constraints_satisfied)) {
233 offer_answer_options->offer_to_receive_audio =
234 value ? PeerConnectionInterface::RTCOfferAnswerOptions::
235 kOfferToReceiveMediaTrue
236 : 0;
237 }
238
239 if (FindConstraint(constraints, MediaConstraints::kOfferToReceiveVideo,
240 &value, &mandatory_constraints_satisfied)) {
241 offer_answer_options->offer_to_receive_video =
242 value ? PeerConnectionInterface::RTCOfferAnswerOptions::
243 kOfferToReceiveMediaTrue
244 : 0;
245 }
246 if (FindConstraint(constraints, MediaConstraints::kVoiceActivityDetection,
247 &value, &mandatory_constraints_satisfied)) {
248 offer_answer_options->voice_activity_detection = value;
249 }
250 if (FindConstraint(constraints, MediaConstraints::kUseRtpMux, &value,
251 &mandatory_constraints_satisfied)) {
252 offer_answer_options->use_rtp_mux = value;
253 }
254 if (FindConstraint(constraints, MediaConstraints::kIceRestart, &value,
255 &mandatory_constraints_satisfied)) {
256 offer_answer_options->ice_restart = value;
257 }
258
Mirta Dvornicic479a3c02019-06-04 15:38:50 +0200259 if (FindConstraint(constraints,
260 MediaConstraints::kRawPacketizationForVideoEnabled, &value,
261 &mandatory_constraints_satisfied)) {
262 offer_answer_options->raw_packetization_for_video = value;
263 }
264
Niels Möllerdac03d92019-02-13 08:52:27 +0100265 int layers;
266 if (FindConstraint(constraints, MediaConstraints::kNumSimulcastLayers,
267 &layers, &mandatory_constraints_satisfied)) {
268 offer_answer_options->num_simulcast_layers = layers;
269 }
270
271 return mandatory_constraints_satisfied == constraints->GetMandatory().size();
272}
273
274} // namespace webrtc