1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "base/metrics/field_trial.h"
9 #include "base/build_time.h"
10 #include "base/logging.h"
11 #include "base/rand_util.h"
12 #include "base/sha1.h"
13 #include "base/strings/string_util.h"
14 #include "base/strings/stringprintf.h"
15 #include "base/strings/utf_string_conversions.h"
16 #include "base/sys_byteorder.h"
22 // Created a time value based on |year|, |month| and |day_of_month| parameters.
23 Time
CreateTimeFromParams(int year
, int month
, int day_of_month
) {
24 DCHECK_GT(year
, 1970);
27 DCHECK_GT(day_of_month
, 0);
28 DCHECK_LT(day_of_month
, 32);
30 Time::Exploded exploded
;
32 exploded
.month
= month
;
33 exploded
.day_of_week
= 0; // Should be unused.
34 exploded
.day_of_month
= day_of_month
;
38 exploded
.millisecond
= 0;
40 return Time::FromLocalExploded(exploded
);
43 // Returns the boundary value for comparing against the FieldTrial's added
44 // groups for a given |divisor| (total probability) and |entropy_value|.
45 FieldTrial::Probability
GetGroupBoundaryValue(
46 FieldTrial::Probability divisor
,
47 double entropy_value
) {
48 // Add a tiny epsilon value to get consistent results when converting floating
49 // points to int. Without it, boundary values have inconsistent results, e.g.:
51 // static_cast<FieldTrial::Probability>(100 * 0.56) == 56
52 // static_cast<FieldTrial::Probability>(100 * 0.57) == 56
53 // static_cast<FieldTrial::Probability>(100 * 0.58) == 57
54 // static_cast<FieldTrial::Probability>(100 * 0.59) == 59
55 const double kEpsilon
= 1e-8;
56 const FieldTrial::Probability result
=
57 static_cast<FieldTrial::Probability
>(divisor
* entropy_value
+ kEpsilon
);
58 // Ensure that adding the epsilon still results in a value < |divisor|.
59 return std::min(result
, divisor
- 1);
65 const int FieldTrial::kNotFinalized
= -1;
66 const int FieldTrial::kDefaultGroupNumber
= 0;
67 bool FieldTrial::enable_benchmarking_
= false;
69 const char FieldTrialList::kPersistentStringSeparator('/');
70 const char FieldTrialList::kActivationMarker('*');
71 int FieldTrialList::kNoExpirationYear
= 0;
73 //------------------------------------------------------------------------------
74 // FieldTrial methods and members.
76 FieldTrial::EntropyProvider::~EntropyProvider() {
79 void FieldTrial::Disable() {
80 DCHECK(!group_reported_
);
81 enable_field_trial_
= false;
83 // In case we are disabled after initialization, we need to switch
84 // the trial to the default group.
85 if (group_
!= kNotFinalized
) {
86 // Only reset when not already the default group, because in case we were
87 // forced to the default group, the group number may not be
88 // kDefaultGroupNumber, so we should keep it as is.
89 if (group_name_
!= default_group_name_
)
90 SetGroupChoice(default_group_name_
, kDefaultGroupNumber
);
94 int FieldTrial::AppendGroup(const std::string
& name
,
95 Probability group_probability
) {
96 // When the group choice was previously forced, we only need to return the
97 // the id of the chosen group, and anything can be returned for the others.
99 DCHECK(!group_name_
.empty());
100 if (name
== group_name_
) {
101 // Note that while |group_| may be equal to |kDefaultGroupNumber| on the
102 // forced trial, it will not have the same value as the default group
103 // number returned from the non-forced |FactoryGetFieldTrial()| call,
104 // which takes care to ensure that this does not happen.
107 DCHECK_NE(next_group_number_
, group_
);
108 // We still return different numbers each time, in case some caller need
109 // them to be different.
110 return next_group_number_
++;
113 DCHECK_LE(group_probability
, divisor_
);
114 DCHECK_GE(group_probability
, 0);
116 if (enable_benchmarking_
|| !enable_field_trial_
)
117 group_probability
= 0;
119 accumulated_group_probability_
+= group_probability
;
121 DCHECK_LE(accumulated_group_probability_
, divisor_
);
122 if (group_
== kNotFinalized
&& accumulated_group_probability_
> random_
) {
123 // This is the group that crossed the random line, so we do the assignment.
124 SetGroupChoice(name
, next_group_number_
);
126 return next_group_number_
++;
129 int FieldTrial::group() {
130 FinalizeGroupChoice();
131 if (trial_registered_
)
132 FieldTrialList::NotifyFieldTrialGroupSelection(this);
136 const std::string
& FieldTrial::group_name() {
137 // Call |group()| to ensure group gets assigned and observers are notified.
139 DCHECK(!group_name_
.empty());
143 void FieldTrial::SetForced() {
144 // We might have been forced before (e.g., by CreateFieldTrial) and it's
145 // first come first served, e.g., command line switch has precedence.
149 // And we must finalize the group choice before we mark ourselves as forced.
150 FinalizeGroupChoice();
155 void FieldTrial::EnableBenchmarking() {
156 DCHECK_EQ(0u, FieldTrialList::GetFieldTrialCount());
157 enable_benchmarking_
= true;
161 FieldTrial
* FieldTrial::CreateSimulatedFieldTrial(
162 const std::string
& trial_name
,
163 Probability total_probability
,
164 const std::string
& default_group_name
,
165 double entropy_value
) {
166 return new FieldTrial(trial_name
, total_probability
, default_group_name
,
170 FieldTrial::FieldTrial(const std::string
& trial_name
,
171 const Probability total_probability
,
172 const std::string
& default_group_name
,
173 double entropy_value
)
174 : trial_name_(trial_name
),
175 divisor_(total_probability
),
176 default_group_name_(default_group_name
),
177 random_(GetGroupBoundaryValue(total_probability
, entropy_value
)),
178 accumulated_group_probability_(0),
179 next_group_number_(kDefaultGroupNumber
+ 1),
180 group_(kNotFinalized
),
181 enable_field_trial_(true),
183 group_reported_(false),
184 trial_registered_(false) {
185 DCHECK_GT(total_probability
, 0);
186 DCHECK(!trial_name_
.empty());
187 DCHECK(!default_group_name_
.empty());
190 FieldTrial::~FieldTrial() {}
192 void FieldTrial::SetTrialRegistered() {
193 DCHECK_EQ(kNotFinalized
, group_
);
194 DCHECK(!trial_registered_
);
195 trial_registered_
= true;
198 void FieldTrial::SetGroupChoice(const std::string
& group_name
, int number
) {
200 if (group_name
.empty())
201 StringAppendF(&group_name_
, "%d", group_
);
203 group_name_
= group_name
;
204 DVLOG(1) << "Field trial: " << trial_name_
<< " Group choice:" << group_name_
;
207 void FieldTrial::FinalizeGroupChoice() {
208 if (group_
!= kNotFinalized
)
210 accumulated_group_probability_
= divisor_
;
211 // Here it's OK to use |kDefaultGroupNumber| since we can't be forced and not
214 SetGroupChoice(default_group_name_
, kDefaultGroupNumber
);
217 bool FieldTrial::GetActiveGroup(ActiveGroup
* active_group
) const {
218 if (!group_reported_
|| !enable_field_trial_
)
220 DCHECK_NE(group_
, kNotFinalized
);
221 active_group
->trial_name
= trial_name_
;
222 active_group
->group_name
= group_name_
;
226 bool FieldTrial::GetState(FieldTrialState
* field_trial_state
) const {
227 if (!enable_field_trial_
)
229 field_trial_state
->trial_name
= trial_name_
;
230 // If the group name is empty (hasn't been finalized yet), use the default
231 // group name instead.
232 if (!group_name_
.empty())
233 field_trial_state
->group_name
= group_name_
;
235 field_trial_state
->group_name
= default_group_name_
;
236 field_trial_state
->activated
= group_reported_
;
240 //------------------------------------------------------------------------------
241 // FieldTrialList methods and members.
244 FieldTrialList
* FieldTrialList::global_
= NULL
;
247 bool FieldTrialList::used_without_global_
= false;
249 FieldTrialList::Observer::~Observer() {
252 FieldTrialList::FieldTrialList(
253 const FieldTrial::EntropyProvider
* entropy_provider
)
254 : entropy_provider_(entropy_provider
),
255 observer_list_(new ObserverListThreadSafe
<FieldTrialList::Observer
>(
256 ObserverListBase
<FieldTrialList::Observer
>::NOTIFY_EXISTING_ONLY
)) {
258 DCHECK(!used_without_global_
);
261 Time two_years_from_build_time
= GetBuildTime() + TimeDelta::FromDays(730);
262 Time::Exploded exploded
;
263 two_years_from_build_time
.LocalExplode(&exploded
);
264 kNoExpirationYear
= exploded
.year
;
267 FieldTrialList::~FieldTrialList() {
268 AutoLock
auto_lock(lock_
);
269 while (!registered_
.empty()) {
270 RegistrationMap::iterator it
= registered_
.begin();
271 it
->second
->Release();
272 registered_
.erase(it
->first
);
274 DCHECK_EQ(this, global_
);
279 FieldTrial
* FieldTrialList::FactoryGetFieldTrial(
280 const std::string
& trial_name
,
281 FieldTrial::Probability total_probability
,
282 const std::string
& default_group_name
,
285 const int day_of_month
,
286 FieldTrial::RandomizationType randomization_type
,
287 int* default_group_number
) {
288 return FactoryGetFieldTrialWithRandomizationSeed(
289 trial_name
, total_probability
, default_group_name
,
290 year
, month
, day_of_month
, randomization_type
, 0, default_group_number
);
294 FieldTrial
* FieldTrialList::FactoryGetFieldTrialWithRandomizationSeed(
295 const std::string
& trial_name
,
296 FieldTrial::Probability total_probability
,
297 const std::string
& default_group_name
,
300 const int day_of_month
,
301 FieldTrial::RandomizationType randomization_type
,
302 uint32 randomization_seed
,
303 int* default_group_number
) {
304 if (default_group_number
)
305 *default_group_number
= FieldTrial::kDefaultGroupNumber
;
306 // Check if the field trial has already been created in some other way.
307 FieldTrial
* existing_trial
= Find(trial_name
);
308 if (existing_trial
) {
309 CHECK(existing_trial
->forced_
);
310 // If the default group name differs between the existing forced trial
311 // and this trial, then use a different value for the default group number.
312 if (default_group_number
&&
313 default_group_name
!= existing_trial
->default_group_name()) {
314 // If the new default group number corresponds to the group that was
315 // chosen for the forced trial (which has been finalized when it was
316 // forced), then set the default group number to that.
317 if (default_group_name
== existing_trial
->group_name_internal()) {
318 *default_group_number
= existing_trial
->group_
;
320 // Otherwise, use |kNonConflictingGroupNumber| (-2) for the default
321 // group number, so that it does not conflict with the |AppendGroup()|
322 // result for the chosen group.
323 const int kNonConflictingGroupNumber
= -2;
325 kNonConflictingGroupNumber
!= FieldTrial::kDefaultGroupNumber
,
326 conflicting_default_group_number
);
328 kNonConflictingGroupNumber
!= FieldTrial::kNotFinalized
,
329 conflicting_default_group_number
);
330 *default_group_number
= kNonConflictingGroupNumber
;
333 return existing_trial
;
336 double entropy_value
;
337 if (randomization_type
== FieldTrial::ONE_TIME_RANDOMIZED
) {
338 const FieldTrial::EntropyProvider
* entropy_provider
=
339 GetEntropyProviderForOneTimeRandomization();
340 CHECK(entropy_provider
);
341 entropy_value
= entropy_provider
->GetEntropyForTrial(trial_name
,
344 DCHECK_EQ(FieldTrial::SESSION_RANDOMIZED
, randomization_type
);
345 DCHECK_EQ(0U, randomization_seed
);
346 entropy_value
= RandDouble();
349 FieldTrial
* field_trial
= new FieldTrial(trial_name
, total_probability
,
350 default_group_name
, entropy_value
);
351 if (GetBuildTime() > CreateTimeFromParams(year
, month
, day_of_month
))
352 field_trial
->Disable();
353 FieldTrialList::Register(field_trial
);
358 FieldTrial
* FieldTrialList::Find(const std::string
& name
) {
361 AutoLock
auto_lock(global_
->lock_
);
362 return global_
->PreLockedFind(name
);
366 int FieldTrialList::FindValue(const std::string
& name
) {
367 FieldTrial
* field_trial
= Find(name
);
369 return field_trial
->group();
370 return FieldTrial::kNotFinalized
;
374 std::string
FieldTrialList::FindFullName(const std::string
& name
) {
375 FieldTrial
* field_trial
= Find(name
);
377 return field_trial
->group_name();
378 return std::string();
382 bool FieldTrialList::TrialExists(const std::string
& name
) {
383 return Find(name
) != NULL
;
387 void FieldTrialList::StatesToString(std::string
* output
) {
388 FieldTrial::ActiveGroups active_groups
;
389 GetActiveFieldTrialGroups(&active_groups
);
390 for (FieldTrial::ActiveGroups::const_iterator it
= active_groups
.begin();
391 it
!= active_groups
.end(); ++it
) {
392 DCHECK_EQ(std::string::npos
,
393 it
->trial_name
.find(kPersistentStringSeparator
));
394 DCHECK_EQ(std::string::npos
,
395 it
->group_name
.find(kPersistentStringSeparator
));
396 output
->append(it
->trial_name
);
397 output
->append(1, kPersistentStringSeparator
);
398 output
->append(it
->group_name
);
399 output
->append(1, kPersistentStringSeparator
);
404 void FieldTrialList::AllStatesToString(std::string
* output
) {
407 AutoLock
auto_lock(global_
->lock_
);
409 for (const auto& registered
: global_
->registered_
) {
410 FieldTrial::FieldTrialState trial
;
411 if (!registered
.second
->GetState(&trial
))
413 DCHECK_EQ(std::string::npos
,
414 trial
.trial_name
.find(kPersistentStringSeparator
));
415 DCHECK_EQ(std::string::npos
,
416 trial
.group_name
.find(kPersistentStringSeparator
));
418 output
->append(1, kActivationMarker
);
419 output
->append(trial
.trial_name
);
420 output
->append(1, kPersistentStringSeparator
);
421 output
->append(trial
.group_name
);
422 output
->append(1, kPersistentStringSeparator
);
427 void FieldTrialList::GetActiveFieldTrialGroups(
428 FieldTrial::ActiveGroups
* active_groups
) {
429 DCHECK(active_groups
->empty());
432 AutoLock
auto_lock(global_
->lock_
);
434 for (RegistrationMap::iterator it
= global_
->registered_
.begin();
435 it
!= global_
->registered_
.end(); ++it
) {
436 FieldTrial::ActiveGroup active_group
;
437 if (it
->second
->GetActiveGroup(&active_group
))
438 active_groups
->push_back(active_group
);
443 bool FieldTrialList::CreateTrialsFromString(
444 const std::string
& trials_string
,
445 FieldTrialActivationMode mode
,
446 const std::set
<std::string
>& ignored_trial_names
) {
448 if (trials_string
.empty() || !global_
)
451 size_t next_item
= 0;
452 while (next_item
< trials_string
.length()) {
453 size_t name_end
= trials_string
.find(kPersistentStringSeparator
, next_item
);
454 if (name_end
== trials_string
.npos
|| next_item
== name_end
)
456 size_t group_name_end
= trials_string
.find(kPersistentStringSeparator
,
458 if (name_end
+ 1 == group_name_end
)
460 if (group_name_end
== trials_string
.npos
)
461 group_name_end
= trials_string
.length();
463 // Verify if the trial should be activated or not.
465 bool force_activation
= false;
466 if (trials_string
[next_item
] == kActivationMarker
) {
467 // Name cannot be only the indicator.
468 if (name_end
- next_item
== 1)
471 force_activation
= true;
473 name
.append(trials_string
, next_item
, name_end
- next_item
);
474 std::string
group_name(trials_string
, name_end
+ 1,
475 group_name_end
- name_end
- 1);
476 next_item
= group_name_end
+ 1;
478 if (ignored_trial_names
.find(name
) != ignored_trial_names
.end())
481 FieldTrial
* trial
= CreateFieldTrial(name
, group_name
);
484 if (mode
== ACTIVATE_TRIALS
|| force_activation
) {
485 // Call |group()| to mark the trial as "used" and notify observers, if
486 // any. This is useful to ensure that field trials created in child
487 // processes are properly reported in crash reports.
495 FieldTrial
* FieldTrialList::CreateFieldTrial(
496 const std::string
& name
,
497 const std::string
& group_name
) {
499 DCHECK_GE(name
.size(), 0u);
500 DCHECK_GE(group_name
.size(), 0u);
501 if (name
.empty() || group_name
.empty() || !global_
)
504 FieldTrial
* field_trial
= FieldTrialList::Find(name
);
506 // In single process mode, or when we force them from the command line,
507 // we may have already created the field trial.
508 if (field_trial
->group_name_internal() != group_name
)
512 const int kTotalProbability
= 100;
513 field_trial
= new FieldTrial(name
, kTotalProbability
, group_name
, 0);
514 FieldTrialList::Register(field_trial
);
515 // Force the trial, which will also finalize the group choice.
516 field_trial
->SetForced();
521 void FieldTrialList::AddObserver(Observer
* observer
) {
524 global_
->observer_list_
->AddObserver(observer
);
528 void FieldTrialList::RemoveObserver(Observer
* observer
) {
531 global_
->observer_list_
->RemoveObserver(observer
);
535 void FieldTrialList::NotifyFieldTrialGroupSelection(FieldTrial
* field_trial
) {
540 AutoLock
auto_lock(global_
->lock_
);
541 if (field_trial
->group_reported_
)
543 field_trial
->group_reported_
= true;
546 if (!field_trial
->enable_field_trial_
)
549 global_
->observer_list_
->Notify(
550 FROM_HERE
, &FieldTrialList::Observer::OnFieldTrialGroupFinalized
,
551 field_trial
->trial_name(), field_trial
->group_name_internal());
555 size_t FieldTrialList::GetFieldTrialCount() {
558 AutoLock
auto_lock(global_
->lock_
);
559 return global_
->registered_
.size();
563 const FieldTrial::EntropyProvider
*
564 FieldTrialList::GetEntropyProviderForOneTimeRandomization() {
566 used_without_global_
= true;
570 return global_
->entropy_provider_
.get();
573 FieldTrial
* FieldTrialList::PreLockedFind(const std::string
& name
) {
574 RegistrationMap::iterator it
= registered_
.find(name
);
575 if (registered_
.end() == it
)
581 void FieldTrialList::Register(FieldTrial
* trial
) {
583 used_without_global_
= true;
586 AutoLock
auto_lock(global_
->lock_
);
587 DCHECK(!global_
->PreLockedFind(trial
->trial_name()));
589 trial
->SetTrialRegistered();
590 global_
->registered_
[trial
->trial_name()] = trial
;