blob: d10b5cff3fa45fd942458d35dbb13d5f942bf3a2 [file] [log] [blame]
andresp@webrtc.orga36ad692014-05-14 12:24:04 +00001// Copyright (c) 2014 The WebRTC project authors. All Rights Reserved.
2//
3// Use of this source code is governed by a BSD-style license
4// that can be found in the LICENSE file in the root of the source
5// tree. An additional intellectual property rights grant can be found
6// in the file PATENTS. All contributing project authors may
7// be found in the AUTHORS file in the root of the source tree.
8//
9
Karl Wiberg79eb1d92017-11-08 12:26:07 +010010#include "system_wrappers/include/field_trial.h"
andresp@webrtc.orga36ad692014-05-14 12:24:04 +000011
Yves Gerey988cc082018-10-23 12:03:01 +020012#include <stddef.h>
Jonas Olsson5b2eda42019-06-11 14:29:40 +020013
14#include <map>
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +000015#include <string>
16
Jonas Olsson5b2eda42019-06-11 14:29:40 +020017#include "absl/strings/string_view.h"
18#include "rtc_base/checks.h"
19#include "rtc_base/logging.h"
Konrad Hofbaueree1e0152019-12-05 16:25:40 +010020#include "rtc_base/string_encode.h"
Jonas Olsson5b2eda42019-06-11 14:29:40 +020021
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +000022// Simple field trial implementation, which allows client to
23// specify desired flags in InitFieldTrialsFromString.
andresp@webrtc.orga36ad692014-05-14 12:24:04 +000024namespace webrtc {
25namespace field_trial {
26
Karl Wiberg79eb1d92017-11-08 12:26:07 +010027static const char* trials_init_string = NULL;
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +000028
Mirko Bonadei92e00382018-09-15 10:37:11 +020029#ifndef WEBRTC_EXCLUDE_FIELD_TRIAL_DEFAULT
Jonas Olsson5b2eda42019-06-11 14:29:40 +020030namespace {
31constexpr char kPersistentStringSeparator = '/';
32// Validates the given field trial string.
33// E.g.:
34// "WebRTC-experimentFoo/Enabled/WebRTC-experimentBar/Enabled100kbps/"
35// Assigns the process to group "Enabled" on WebRTCExperimentFoo trial
36// and to group "Enabled100kbps" on WebRTCExperimentBar.
37//
38// E.g. invalid config:
39// "WebRTC-experiment1/Enabled" (note missing / separator at the end).
Konrad Hofbaueree1e0152019-12-05 16:25:40 +010040bool FieldTrialsStringIsValidInternal(const absl::string_view trials) {
Jonas Olsson5b2eda42019-06-11 14:29:40 +020041 if (trials.empty())
42 return true;
43
44 size_t next_item = 0;
45 std::map<absl::string_view, absl::string_view> field_trials;
46 while (next_item < trials.length()) {
47 size_t name_end = trials.find(kPersistentStringSeparator, next_item);
48 if (name_end == trials.npos || next_item == name_end)
49 return false;
50 size_t group_name_end =
51 trials.find(kPersistentStringSeparator, name_end + 1);
52 if (group_name_end == trials.npos || name_end + 1 == group_name_end)
53 return false;
54 absl::string_view name = trials.substr(next_item, name_end - next_item);
55 absl::string_view group_name =
56 trials.substr(name_end + 1, group_name_end - name_end - 1);
57
58 next_item = group_name_end + 1;
59
60 // Fail if duplicate with different group name.
61 if (field_trials.find(name) != field_trials.end() &&
62 field_trials.find(name)->second != group_name) {
63 return false;
64 }
65
66 field_trials[name] = group_name;
67 }
68
69 return true;
70}
71} // namespace
72
Konrad Hofbaueree1e0152019-12-05 16:25:40 +010073bool FieldTrialsStringIsValid(const char* trials_string) {
74 return FieldTrialsStringIsValidInternal(trials_string);
75}
76
77void InsertOrReplaceFieldTrialStringsInMap(
78 std::map<std::string, std::string>* fieldtrial_map,
79 const absl::string_view trials_string) {
80 if (FieldTrialsStringIsValidInternal(trials_string)) {
81 std::vector<std::string> tokens;
82 rtc::split(std::string(trials_string), '/', &tokens);
83 // Skip last token which is empty due to trailing '/'.
84 for (size_t idx = 0; idx < tokens.size() - 1; idx += 2) {
85 (*fieldtrial_map)[tokens[idx]] = tokens[idx + 1];
86 }
87 } else {
Mirko Bonadei25ab3222021-07-08 20:08:20 +020088 RTC_NOTREACHED() << "Invalid field trials string:" << trials_string;
Konrad Hofbaueree1e0152019-12-05 16:25:40 +010089 }
90}
91
92std::string MergeFieldTrialsStrings(const char* first, const char* second) {
93 std::map<std::string, std::string> fieldtrial_map;
94 InsertOrReplaceFieldTrialStringsInMap(&fieldtrial_map, first);
95 InsertOrReplaceFieldTrialStringsInMap(&fieldtrial_map, second);
96
97 // Merge into fieldtrial string.
98 std::string merged = "";
99 for (auto const& fieldtrial : fieldtrial_map) {
100 merged += fieldtrial.first + '/' + fieldtrial.second + '/';
101 }
102 return merged;
103}
104
andresp@webrtc.orga36ad692014-05-14 12:24:04 +0000105std::string FindFullName(const std::string& name) {
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +0000106 if (trials_init_string == NULL)
107 return std::string();
108
109 std::string trials_string(trials_init_string);
110 if (trials_string.empty())
111 return std::string();
112
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +0000113 size_t next_item = 0;
114 while (next_item < trials_string.length()) {
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +0000115 // Find next name/value pair in field trial configuration string.
Karl Wiberg79eb1d92017-11-08 12:26:07 +0100116 size_t field_name_end =
117 trials_string.find(kPersistentStringSeparator, next_item);
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +0000118 if (field_name_end == trials_string.npos || field_name_end == next_item)
119 break;
Karl Wiberg79eb1d92017-11-08 12:26:07 +0100120 size_t field_value_end =
121 trials_string.find(kPersistentStringSeparator, field_name_end + 1);
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +0000122 if (field_value_end == trials_string.npos ||
123 field_value_end == field_name_end + 1)
124 break;
125 std::string field_name(trials_string, next_item,
Karl Wiberg79eb1d92017-11-08 12:26:07 +0100126 field_name_end - next_item);
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +0000127 std::string field_value(trials_string, field_name_end + 1,
Karl Wiberg79eb1d92017-11-08 12:26:07 +0100128 field_value_end - field_name_end - 1);
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +0000129 next_item = field_value_end + 1;
130
131 if (name == field_name)
132 return field_value;
133 }
andresp@webrtc.orga36ad692014-05-14 12:24:04 +0000134 return std::string();
135}
Mirko Bonadei92e00382018-09-15 10:37:11 +0200136#endif // WEBRTC_EXCLUDE_FIELD_TRIAL_DEFAULT
andresp@webrtc.orga36ad692014-05-14 12:24:04 +0000137
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +0000138// Optionally initialize field trial from a string.
139void InitFieldTrialsFromString(const char* trials_string) {
Jonas Olsson5b2eda42019-06-11 14:29:40 +0200140 RTC_LOG(LS_INFO) << "Setting field trial string:" << trials_string;
141#ifndef WEBRTC_EXCLUDE_FIELD_TRIAL_DEFAULT
142 if (trials_string) {
Konrad Hofbaueree1e0152019-12-05 16:25:40 +0100143 RTC_DCHECK(FieldTrialsStringIsValidInternal(trials_string))
Jonas Olsson5b2eda42019-06-11 14:29:40 +0200144 << "Invalid field trials string:" << trials_string;
145 };
146#endif // WEBRTC_EXCLUDE_FIELD_TRIAL_DEFAULT
glaznev@webrtc.org669bc7e2015-02-09 18:17:46 +0000147 trials_init_string = trials_string;
148}
149
phoglund37ebcf02016-01-08 05:04:57 -0800150const char* GetFieldTrialString() {
151 return trials_init_string;
152}
153
andresp@webrtc.orga36ad692014-05-14 12:24:04 +0000154} // namespace field_trial
155} // namespace webrtc