// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "content/browser/field_trial_synchronizer.h" #include "base/bind.h" #include "base/check_op.h" #include "base/threading/thread.h" #include "components/metrics/persistent_system_profile.h" #include "components/variations/variations_client.h" #include "content/common/renderer_variations_configuration.mojom.h" #include "content/public/browser/browser_context.h" #include "content/public/browser/browser_task_traits.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/render_process_host.h" #include "mojo/public/cpp/bindings/associated_remote.h" namespace content { namespace { FieldTrialSynchronizer* g_instance = nullptr; // Notifies all renderer processes about the |group_name| that is finalized for // the given field trail (|field_trial_name|). This is called on UI thread. void NotifyAllRenderersOfFieldTrial(const std::string& field_trial_name, const std::string& group_name) { // To iterate over RenderProcessHosts, or to send messages to the hosts, we // need to be on the UI thread. DCHECK_CURRENTLY_ON(BrowserThread::UI); // Note this in the persistent profile as it will take a while for a new // "complete" profile to be generated. metrics::GlobalPersistentSystemProfile::GetInstance()->AddFieldTrial( field_trial_name, group_name); for (RenderProcessHost::iterator it(RenderProcessHost::AllHostsIterator()); !it.IsAtEnd(); it.Advance()) { auto* host = it.GetCurrentValue(); IPC::ChannelProxy* channel = host->GetChannel(); // channel might be null in tests. if (host->IsInitializedAndNotDead() && channel) { mojo::AssociatedRemote renderer_variations_configuration; channel->GetRemoteAssociatedInterface(&renderer_variations_configuration); renderer_variations_configuration->SetFieldTrialGroup(field_trial_name, group_name); } } } } // namespace // static void FieldTrialSynchronizer::CreateInstance() { // Only 1 instance is allowed per process. DCHECK(!g_instance); g_instance = new FieldTrialSynchronizer(); } FieldTrialSynchronizer::FieldTrialSynchronizer() { bool success = base::FieldTrialList::AddObserver(this); // Ensure the observer was actually registered. DCHECK(success); variations::VariationsIdsProvider::GetInstance()->AddObserver(this); NotifyAllRenderersOfVariationsHeader(); } void FieldTrialSynchronizer::OnFieldTrialGroupFinalized( const std::string& field_trial_name, const std::string& group_name) { RunOrPostTaskOnThread(FROM_HERE, BrowserThread::UI, base::BindOnce(&NotifyAllRenderersOfFieldTrial, field_trial_name, group_name)); } // static void FieldTrialSynchronizer::NotifyAllRenderersOfVariationsHeader() { // To iterate over RenderProcessHosts, or to send messages to the hosts, we // need to be on the UI thread. DCHECK_CURRENTLY_ON(BrowserThread::UI); for (RenderProcessHost::iterator it(RenderProcessHost::AllHostsIterator()); !it.IsAtEnd(); it.Advance()) { UpdateRendererVariationsHeader(it.GetCurrentValue()); } } // static void FieldTrialSynchronizer::UpdateRendererVariationsHeader( RenderProcessHost* host) { if (!host->IsInitializedAndNotDead()) return; IPC::ChannelProxy* channel = host->GetChannel(); // |channel| might be null in tests. if (!channel) return; variations::VariationsClient* client = host->GetBrowserContext()->GetVariationsClient(); // |client| might be null in tests. if (!client || client->IsOffTheRecord()) return; mojo::AssociatedRemote renderer_variations_configuration; channel->GetRemoteAssociatedInterface(&renderer_variations_configuration); renderer_variations_configuration->SetVariationsHeaders( client->GetVariationsHeaders()); } void FieldTrialSynchronizer::VariationIdsHeaderUpdated() { // PostTask to avoid recursive lock. GetUIThreadTaskRunner({})->PostTask( FROM_HERE, base::BindOnce( &FieldTrialSynchronizer::NotifyAllRenderersOfVariationsHeader)); } FieldTrialSynchronizer::~FieldTrialSynchronizer() { NOTREACHED(); } } // namespace content