1 // Copyright 2014 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 "components/keyed_service/content/browser_context_dependency_manager.h"
11 #include "base/bind.h"
12 #include "base/debug/trace_event.h"
13 #include "components/keyed_service/content/browser_context_keyed_base_factory.h"
14 #include "content/public/browser/browser_context.h"
17 #include "base/command_line.h"
18 #include "base/file_util.h"
20 // Dumps dependency information about our browser context keyed services
21 // into a dot file in the browser context directory.
22 const char kDumpBrowserContextDependencyGraphFlag
[] =
23 "dump-browser-context-graph";
26 void BrowserContextDependencyManager::AddComponent(
27 BrowserContextKeyedBaseFactory
* component
) {
28 dependency_graph_
.AddNode(component
);
31 void BrowserContextDependencyManager::RemoveComponent(
32 BrowserContextKeyedBaseFactory
* component
) {
33 dependency_graph_
.RemoveNode(component
);
36 void BrowserContextDependencyManager::AddEdge(
37 BrowserContextKeyedBaseFactory
* depended
,
38 BrowserContextKeyedBaseFactory
* dependee
) {
39 dependency_graph_
.AddEdge(depended
, dependee
);
42 void BrowserContextDependencyManager::RegisterProfilePrefsForServices(
43 const content::BrowserContext
* context
,
44 user_prefs::PrefRegistrySyncable
* pref_registry
) {
45 std::vector
<DependencyNode
*> construction_order
;
46 if (!dependency_graph_
.GetConstructionOrder(&construction_order
)) {
50 for (std::vector
<DependencyNode
*>::const_iterator it
=
51 construction_order
.begin();
52 it
!= construction_order
.end();
54 BrowserContextKeyedBaseFactory
* factory
=
55 static_cast<BrowserContextKeyedBaseFactory
*>(*it
);
56 factory
->RegisterProfilePrefsIfNecessaryForContext(context
, pref_registry
);
60 void BrowserContextDependencyManager::CreateBrowserContextServices(
61 content::BrowserContext
* context
) {
62 DoCreateBrowserContextServices(context
, false);
65 void BrowserContextDependencyManager::CreateBrowserContextServicesForTest(
66 content::BrowserContext
* context
) {
67 DoCreateBrowserContextServices(context
, true);
70 void BrowserContextDependencyManager::DoCreateBrowserContextServices(
71 content::BrowserContext
* context
,
72 bool is_testing_context
) {
75 "BrowserContextDependencyManager::DoCreateBrowserContextServices")
77 MarkBrowserContextLiveForTesting(context
);
80 will_create_browser_context_services_callbacks_
.Notify(context
);
82 std::vector
<DependencyNode
*> construction_order
;
83 if (!dependency_graph_
.GetConstructionOrder(&construction_order
)) {
88 DumpBrowserContextDependencies(context
);
91 for (size_t i
= 0; i
< construction_order
.size(); i
++) {
92 BrowserContextKeyedBaseFactory
* factory
=
93 static_cast<BrowserContextKeyedBaseFactory
*>(construction_order
[i
]);
94 if (is_testing_context
&& factory
->ServiceIsNULLWhileTesting() &&
95 !factory
->HasTestingFactory(context
)) {
96 factory
->SetEmptyTestingFactory(context
);
97 } else if (factory
->ServiceIsCreatedWithBrowserContext()) {
98 // Create the service.
99 factory
->CreateServiceNow(context
);
104 void BrowserContextDependencyManager::DestroyBrowserContextServices(
105 content::BrowserContext
* context
) {
106 std::vector
<DependencyNode
*> destruction_order
;
107 if (!dependency_graph_
.GetDestructionOrder(&destruction_order
)) {
112 DumpBrowserContextDependencies(context
);
115 for (size_t i
= 0; i
< destruction_order
.size(); i
++) {
116 BrowserContextKeyedBaseFactory
* factory
=
117 static_cast<BrowserContextKeyedBaseFactory
*>(destruction_order
[i
]);
118 factory
->BrowserContextShutdown(context
);
122 // The context is now dead to the rest of the program.
123 dead_context_pointers_
.insert(context
);
126 for (size_t i
= 0; i
< destruction_order
.size(); i
++) {
127 BrowserContextKeyedBaseFactory
* factory
=
128 static_cast<BrowserContextKeyedBaseFactory
*>(destruction_order
[i
]);
129 factory
->BrowserContextDestroyed(context
);
133 scoped_ptr
<base::CallbackList
<void(content::BrowserContext
*)>::Subscription
>
134 BrowserContextDependencyManager::
135 RegisterWillCreateBrowserContextServicesCallbackForTesting(
136 const base::Callback
<void(content::BrowserContext
*)>& callback
) {
137 return will_create_browser_context_services_callbacks_
.Add(callback
);
141 void BrowserContextDependencyManager::AssertBrowserContextWasntDestroyed(
142 content::BrowserContext
* context
) {
143 if (dead_context_pointers_
.find(context
) != dead_context_pointers_
.end()) {
144 NOTREACHED() << "Attempted to access a BrowserContext that was ShutDown(). "
145 << "This is most likely a heap smasher in progress. After "
146 << "KeyedService::Shutdown() completes, your "
147 << "service MUST NOT refer to depended BrowserContext "
148 << "services again.";
152 void BrowserContextDependencyManager::MarkBrowserContextLiveForTesting(
153 content::BrowserContext
* context
) {
154 dead_context_pointers_
.erase(context
);
159 BrowserContextDependencyManager
*
160 BrowserContextDependencyManager::GetInstance() {
161 return Singleton
<BrowserContextDependencyManager
>::get();
164 BrowserContextDependencyManager::BrowserContextDependencyManager() {}
166 BrowserContextDependencyManager::~BrowserContextDependencyManager() {}
171 std::string
BrowserContextKeyedBaseFactoryGetNodeName(DependencyNode
* node
) {
172 return static_cast<BrowserContextKeyedBaseFactory
*>(node
)->name();
177 void BrowserContextDependencyManager::DumpBrowserContextDependencies(
178 content::BrowserContext
* context
) {
179 // Whenever we try to build a destruction ordering, we should also dump a
180 // dependency graph to "/path/to/context/context-dependencies.dot".
181 if (CommandLine::ForCurrentProcess()->HasSwitch(
182 kDumpBrowserContextDependencyGraphFlag
)) {
183 base::FilePath dot_file
=
184 context
->GetPath().AppendASCII("browser-context-dependencies.dot");
185 std::string contents
= dependency_graph_
.DumpAsGraphviz(
187 base::Bind(&BrowserContextKeyedBaseFactoryGetNodeName
));
188 base::WriteFile(dot_file
, contents
.c_str(), contents
.size());