blob: ce64af11873c11ef70186815acd7690834b6bce0 [file] [log] [blame]
[email protected]55eb70e762012-02-20 17:38:391// 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.
4
5#include "content/public/browser/browser_context.h"
6
avib7348942015-12-25 20:57:107#include <stddef.h>
8#include <stdint.h>
avi1ed4a4372017-04-25 05:39:419
erge69130f52016-03-02 00:13:2810#include <algorithm>
11#include <limits>
rockot963ad3e82016-04-13 16:33:2312#include <memory>
dcheng36b6aec92015-12-26 06:16:3613#include <utility>
rockot963ad3e82016-04-13 16:33:2314#include <vector>
avib7348942015-12-25 20:57:1015
mmenkec0b2b8b12017-04-21 16:27:5216#include "base/base64.h"
Takashi Toyoshimaa12ecf42018-09-25 07:46:0117#include "base/bind.h"
rockot963ad3e82016-04-13 16:33:2318#include "base/command_line.h"
Takashi Toyoshimaa12ecf42018-09-25 07:46:0119#include "base/feature_list.h"
Chris Cunninghama5b38012017-10-28 07:27:1820#include "base/files/file_path.h"
bencccfe2a2016-03-05 16:54:1421#include "base/guid.h"
erge69130f52016-03-02 00:13:2822#include "base/lazy_instance.h"
mmenkec0b2b8b12017-04-21 16:27:5223#include "base/logging.h"
rockot963ad3e82016-04-13 16:33:2324#include "base/macros.h"
avi1ed4a4372017-04-25 05:39:4125#include "base/memory/ptr_util.h"
Ken Rockotda7edc62018-11-10 01:01:4526#include "base/no_destructor.h"
erge69130f52016-03-02 00:13:2827#include "base/rand_util.h"
Takashi Toyoshima621e2bf2018-09-18 08:05:1628#include "base/supports_user_data.h"
Gabriel Charette44db1422018-08-06 11:19:3329#include "base/task/post_task.h"
Ken Rockot84f58fd2018-06-22 21:54:1430#include "base/threading/sequenced_task_runner_handle.h"
ben21a34c252016-06-29 22:24:3731#include "base/threading/thread_task_runner_handle.h"
Andrey Lushnikovd39d9062018-04-24 17:33:3532#include "base/unguessable_token.h"
avib7348942015-12-25 20:57:1033#include "build/build_config.h"
dmurph7ac019a2016-05-13 00:13:1734#include "content/browser/blob_storage/chrome_blob_storage_context.h"
msrameke169ccb2017-04-26 05:21:4135#include "content/browser/browsing_data/browsing_data_remover_impl.h"
Ken Rockot84f58fd2018-06-22 21:54:1436#include "content/browser/content_service_delegate_impl.h"
[email protected]b441a8492012-06-06 14:55:5737#include "content/browser/download/download_manager_impl.h"
[email protected]c4d281662013-03-31 00:35:0838#include "content/browser/indexed_db/indexed_db_context_impl.h"
[email protected]678c0362012-12-05 08:02:4439#include "content/browser/loader/resource_dispatcher_host_impl.h"
Takashi Toyoshima621e2bf2018-09-18 08:05:1640#include "content/browser/loader/shared_cors_origin_access_list_impl.h"
Andrey Lushnikovebff0442018-07-12 20:02:5841#include "content/browser/permissions/permission_controller_impl.h"
mvanouwerkerk17205ea2014-11-07 17:30:1542#include "content/browser/push_messaging/push_messaging_router.h"
Ken Rockotc7a279c2017-05-04 23:51:4543#include "content/browser/service_manager/common_browser_interfaces.h"
[email protected]4c3a23582012-08-18 08:54:3444#include "content/browser/storage_partition_impl_map.h"
[email protected]d7c7c98a2012-07-12 21:27:4445#include "content/common/child_process_host_impl.h"
[email protected]393b6cb2014-05-15 00:55:1246#include "content/public/browser/blob_handle.h"
Eric Seckler8652dcd52018-09-20 10:42:2847#include "content/public/browser/browser_task_traits.h"
[email protected]55eb70e762012-02-20 17:38:3948#include "content/public/browser/browser_thread.h"
[email protected]b441a8492012-06-06 14:55:5749#include "content/public/browser/content_browser_client.h"
falken04a6912a2016-09-23 21:06:2950#include "content/public/browser/render_process_host.h"
[email protected]536fd0b2013-03-14 17:41:5751#include "content/public/browser/site_instance.h"
rockot963ad3e82016-04-13 16:33:2352#include "content/public/common/content_switches.h"
bend32292b2016-10-07 00:21:5853#include "content/public/common/service_manager_connection.h"
ben51bb6c62016-11-17 20:15:5754#include "content/public/common/service_names.mojom.h"
Chris Cunningham9e669472017-11-15 21:03:1155#include "media/capabilities/video_decode_stats_db_impl.h"
Chris Cunninghama5b38012017-10-28 07:27:1856#include "media/mojo/services/video_decode_perf_history.h"
[email protected]4d7c4ef2012-03-16 01:47:1257#include "net/cookies/cookie_store.h"
[email protected]6b8a3c742014-07-25 00:25:3558#include "net/ssl/channel_id_service.h"
59#include "net/ssl/channel_id_store.h"
[email protected]6e2d3d22012-02-24 18:10:3660#include "net/url_request/url_request_context.h"
[email protected]6939075a2012-08-28 08:35:5361#include "net/url_request/url_request_context_getter.h"
Ken Rockot84f58fd2018-06-22 21:54:1462#include "services/content/public/mojom/constants.mojom.h"
63#include "services/content/service.h"
ben768c8dc2016-08-12 00:26:5064#include "services/file/file_service.h"
Ken Rockot21142de2018-02-10 01:45:3065#include "services/file/public/mojom/constants.mojom.h"
ben768c8dc2016-08-12 00:26:5066#include "services/file/user_id_map.h"
Takashi Toyoshimaa12ecf42018-09-25 07:46:0167#include "services/network/public/cpp/features.h"
rockot734fb662016-10-15 16:41:3068#include "services/service_manager/public/cpp/connector.h"
Ken Rockot543f5e32018-02-04 02:13:5069#include "services/service_manager/public/mojom/service.mojom.h"
Xing Liu89860472018-02-09 20:07:0270#include "storage/browser/blob/blob_storage_context.h"
pilgrime92c5fcd2014-09-10 23:31:2371#include "storage/browser/database/database_tracker.h"
72#include "storage/browser/fileapi/external_mount_points.h"
[email protected]55eb70e762012-02-20 17:38:3973
[email protected]314c3e22012-02-21 03:57:4274using base::UserDataAdapter;
[email protected]55eb70e762012-02-20 17:38:3975
[email protected]55eb70e762012-02-20 17:38:3976namespace content {
77
[email protected]735e20c2012-03-20 01:16:5978namespace {
79
Ken Rockotda7edc62018-11-10 01:01:4580using TokenToContextMap = std::map<base::Token, BrowserContext*>;
81TokenToContextMap& GetTokenToContextMap() {
82 static base::NoDestructor<TokenToContextMap> map;
83 return *map;
84}
ben6c85c4492016-06-16 20:40:5185
Ken Rockotda7edc62018-11-10 01:01:4586class ServiceInstanceGroupHolder : public base::SupportsUserData::Data {
ben6c85c4492016-06-16 20:40:5187 public:
Ken Rockotda7edc62018-11-10 01:01:4588 explicit ServiceInstanceGroupHolder(const base::Token& instance_group)
89 : instance_group_(instance_group) {}
90 ~ServiceInstanceGroupHolder() override {}
ben6c85c4492016-06-16 20:40:5191
Ken Rockotda7edc62018-11-10 01:01:4592 const base::Token& instance_group() const { return instance_group_; }
ben6c85c4492016-06-16 20:40:5193
94 private:
Ken Rockotda7edc62018-11-10 01:01:4595 base::Token instance_group_;
ben6c85c4492016-06-16 20:40:5196
Ken Rockotda7edc62018-11-10 01:01:4597 DISALLOW_COPY_AND_ASSIGN(ServiceInstanceGroupHolder);
ben6c85c4492016-06-16 20:40:5198};
erge69130f52016-03-02 00:13:2899
Ken Rockot84f58fd2018-06-22 21:54:14100class ContentServiceDelegateHolder : public base::SupportsUserData::Data {
101 public:
102 explicit ContentServiceDelegateHolder(BrowserContext* browser_context)
103 : delegate_(browser_context) {}
104 ~ContentServiceDelegateHolder() override = default;
105
106 ContentServiceDelegateImpl* delegate() { return &delegate_; }
107
108 private:
109 ContentServiceDelegateImpl delegate_;
110
111 DISALLOW_COPY_AND_ASSIGN(ContentServiceDelegateHolder);
112};
113
Takashi Toyoshimaa12ecf42018-09-25 07:46:01114// A class used to make an asynchronous Mojo call with cloned patterns for each
115// StoragePartition iteration. |this| instance will be destructed when all
116// existing asynchronous Mojo calls made in SetLists() are done, and |closure|
117// will be invoked on destructing |this|.
118class CorsOriginPatternSetter
119 : public base::RefCounted<CorsOriginPatternSetter> {
120 public:
121 CorsOriginPatternSetter(
122 const url::Origin& source_origin,
123 std::vector<network::mojom::CorsOriginPatternPtr> allow_patterns,
124 std::vector<network::mojom::CorsOriginPatternPtr> block_patterns,
125 base::OnceClosure closure)
126 : source_origin_(source_origin),
127 allow_patterns_(std::move(allow_patterns)),
128 block_patterns_(std::move(block_patterns)),
129 closure_(std::move(closure)) {}
130
131 void SetLists(StoragePartition* partition) {
132 partition->GetNetworkContext()->SetCorsOriginAccessListsForOrigin(
133 source_origin_, ClonePatterns(allow_patterns_),
134 ClonePatterns(block_patterns_),
135 base::BindOnce([](scoped_refptr<CorsOriginPatternSetter> setter) {},
136 base::RetainedRef(this)));
137 }
138
139 private:
140 friend class base::RefCounted<CorsOriginPatternSetter>;
141
142 static std::vector<network::mojom::CorsOriginPatternPtr> ClonePatterns(
143 const std::vector<network::mojom::CorsOriginPatternPtr>& patterns) {
144 std::vector<network::mojom::CorsOriginPatternPtr> cloned_patterns;
145 cloned_patterns.reserve(patterns.size());
146 for (const auto& item : patterns)
147 cloned_patterns.push_back(item.Clone());
148 return cloned_patterns;
149 }
150
151 ~CorsOriginPatternSetter() { std::move(closure_).Run(); }
152
153 const url::Origin source_origin_;
154 const std::vector<network::mojom::CorsOriginPatternPtr> allow_patterns_;
155 const std::vector<network::mojom::CorsOriginPatternPtr> block_patterns_;
156
157 base::OnceClosure closure_;
158};
159
[email protected]e0ce8a1e2012-09-18 10:26:36160// Key names on BrowserContext.
msrameke169ccb2017-04-26 05:21:41161const char kBrowsingDataRemoverKey[] = "browsing-data-remover";
Ken Rockot84f58fd2018-06-22 21:54:14162const char kContentServiceDelegateKey[] = "content-service-delegate";
[email protected]6ef0c3912013-01-25 22:46:34163const char kDownloadManagerKeyName[] = "download_manager";
Takashi Toyoshima621e2bf2018-09-18 08:05:16164const char kPermissionControllerKey[] = "permission-controller";
bend32292b2016-10-07 00:21:58165const char kServiceManagerConnection[] = "service-manager-connection";
Ken Rockotda7edc62018-11-10 01:01:45166const char kServiceInstanceGroup[] = "service-instance-group";
Takashi Toyoshima621e2bf2018-09-18 08:05:16167const char kSharedCorsOriginAccessListKey[] = "shared-cors-origin-access-list";
rockot963ad3e82016-04-13 16:33:23168const char kStoragePartitionMapKeyName[] = "content_storage_partition_map";
Chris Cunninghama5b38012017-10-28 07:27:18169const char kVideoDecodePerfHistoryId[] = "video-decode-perf-history";
erge69130f52016-03-02 00:13:28170
[email protected]9afc14e22013-09-25 22:34:14171#if defined(OS_CHROMEOS)
172const char kMountPointsKey[] = "mount_points";
173#endif // defined(OS_CHROMEOS)
174
Ken Rockotda7edc62018-11-10 01:01:45175void RemoveBrowserContextFromInstanceGroupMap(BrowserContext* browser_context) {
176 ServiceInstanceGroupHolder* holder = static_cast<ServiceInstanceGroupHolder*>(
177 browser_context->GetUserData(kServiceInstanceGroup));
ben6c85c4492016-06-16 20:40:51178 if (holder) {
Ken Rockotda7edc62018-11-10 01:01:45179 auto it = GetTokenToContextMap().find(holder->instance_group());
180 if (it != GetTokenToContextMap().end())
181 GetTokenToContextMap().erase(it);
ben6c85c4492016-06-16 20:40:51182 }
183}
184
[email protected]14acc642012-11-17 12:20:10185StoragePartitionImplMap* GetStoragePartitionMap(
186 BrowserContext* browser_context) {
[email protected]4c3a23582012-08-18 08:54:34187 StoragePartitionImplMap* partition_map =
188 static_cast<StoragePartitionImplMap*>(
a.cavalcantiffab73762015-08-15 02:55:48189 browser_context->GetUserData(kStoragePartitionMapKeyName));
[email protected]d7c7c98a2012-07-12 21:27:44190 if (!partition_map) {
avi1ed4a4372017-04-25 05:39:41191 auto partition_map_owned =
Jeremy Roman04f27c372017-10-27 15:20:55192 std::make_unique<StoragePartitionImplMap>(browser_context);
avi1ed4a4372017-04-25 05:39:41193 partition_map = partition_map_owned.get();
194 browser_context->SetUserData(kStoragePartitionMapKeyName,
195 std::move(partition_map_owned));
[email protected]d7c7c98a2012-07-12 21:27:44196 }
[email protected]14acc642012-11-17 12:20:10197 return partition_map;
198}
199
200StoragePartition* GetStoragePartitionFromConfig(
201 BrowserContext* browser_context,
202 const std::string& partition_domain,
203 const std::string& partition_name,
Dan Elphick6c0d8492017-09-12 09:35:38204 bool in_memory,
205 bool can_create) {
[email protected]14acc642012-11-17 12:20:10206 StoragePartitionImplMap* partition_map =
207 GetStoragePartitionMap(browser_context);
[email protected]d7c7c98a2012-07-12 21:27:44208
[email protected]1bc28312012-11-08 08:31:53209 if (browser_context->IsOffTheRecord())
210 in_memory = true;
211
Dan Elphick6c0d8492017-09-12 09:35:38212 return partition_map->Get(partition_domain, partition_name, in_memory,
213 can_create);
[email protected]d1198fd2012-08-13 22:50:19214}
215
[email protected]6939075a2012-08-28 08:35:53216void SaveSessionStateOnIOThread(
217 const scoped_refptr<net::URLRequestContextGetter>& context_getter,
[email protected]98d6d4562014-06-25 20:57:55218 AppCacheServiceImpl* appcache_service) {
[email protected]6939075a2012-08-28 08:35:53219 net::URLRequestContext* context = context_getter->GetURLRequestContext();
mmenkeded79da2016-02-06 08:28:51220 context->cookie_store()->SetForceKeepSessionState();
Reid Klecknerc1148882018-05-11 00:04:07221 context->channel_id_service()->GetChannelIDStore()->
222 SetForceKeepSessionState();
[email protected]6939075a2012-08-28 08:35:53223 appcache_service->set_force_keep_session_state();
[email protected]6e2d3d22012-02-24 18:10:36224}
225
[email protected]89acda82013-06-25 20:52:50226void SaveSessionStateOnIndexedDBThread(
[email protected]6e2d3d22012-02-24 18:10:36227 scoped_refptr<IndexedDBContextImpl> indexed_db_context) {
[email protected]bf510ed2012-06-05 08:31:43228 indexed_db_context->SetForceKeepSessionState();
[email protected]6e2d3d22012-02-24 18:10:36229}
230
falken41f4175162014-10-29 07:03:41231void ShutdownServiceWorkerContext(StoragePartition* partition) {
232 ServiceWorkerContextWrapper* wrapper =
233 static_cast<ServiceWorkerContextWrapper*>(
234 partition->GetServiceWorkerContext());
235 wrapper->process_manager()->Shutdown();
236}
237
avi1ed4a4372017-04-25 05:39:41238void SetDownloadManager(
239 BrowserContext* context,
240 std::unique_ptr<content::DownloadManager> download_manager) {
ttr31481dc54b2015-08-06 20:11:26241 DCHECK_CURRENTLY_ON(BrowserThread::UI);
242 DCHECK(download_manager);
avi1ed4a4372017-04-25 05:39:41243 context->SetUserData(kDownloadManagerKeyName, std::move(download_manager));
ttr31481dc54b2015-08-06 20:11:26244}
245
bend32292b2016-10-07 00:21:58246class BrowserContextServiceManagerConnectionHolder
rockot963ad3e82016-04-13 16:33:23247 : public base::SupportsUserData::Data {
248 public:
benaad37ecd2017-04-12 22:08:20249 explicit BrowserContextServiceManagerConnectionHolder(
rockot400ea35b2016-10-15 19:15:32250 service_manager::mojom::ServiceRequest request)
benaad37ecd2017-04-12 22:08:20251 : service_manager_connection_(ServiceManagerConnection::Create(
rockotcef38272016-07-15 22:47:47252 std::move(request),
Eric Seckler8652dcd52018-09-20 10:42:28253 base::CreateSingleThreadTaskRunnerWithTraits(
254 {BrowserThread::IO}))) {}
bend32292b2016-10-07 00:21:58255 ~BrowserContextServiceManagerConnectionHolder() override {}
rockot963ad3e82016-04-13 16:33:23256
bend32292b2016-10-07 00:21:58257 ServiceManagerConnection* service_manager_connection() {
258 return service_manager_connection_.get();
259 }
rockot963ad3e82016-04-13 16:33:23260
261 private:
bend32292b2016-10-07 00:21:58262 std::unique_ptr<ServiceManagerConnection> service_manager_connection_;
rockot963ad3e82016-04-13 16:33:23263
bend32292b2016-10-07 00:21:58264 DISALLOW_COPY_AND_ASSIGN(BrowserContextServiceManagerConnectionHolder);
rockot963ad3e82016-04-13 16:33:23265};
266
Jens Widell7ca8fb42018-02-21 14:11:54267base::WeakPtr<storage::BlobStorageContext> BlobStorageContextGetterForBrowser(
Xing Liu89860472018-02-09 20:07:02268 scoped_refptr<ChromeBlobStorageContext> blob_context) {
269 DCHECK_CURRENTLY_ON(BrowserThread::IO);
270 return blob_context->context()->AsWeakPtr();
271}
272
[email protected]735e20c2012-03-20 01:16:59273} // namespace
274
[email protected]14acc642012-11-17 12:20:10275// static
276void BrowserContext::AsyncObliterateStoragePartition(
277 BrowserContext* browser_context,
[email protected]399583b2012-12-11 09:33:42278 const GURL& site,
279 const base::Closure& on_gc_required) {
280 GetStoragePartitionMap(browser_context)->AsyncObliterate(site,
281 on_gc_required);
282}
283
284// static
285void BrowserContext::GarbageCollectStoragePartitions(
dcheng59716272016-04-09 05:19:08286 BrowserContext* browser_context,
287 std::unique_ptr<base::hash_set<base::FilePath>> active_paths,
288 const base::Closure& done) {
dcheng36b6aec92015-12-26 06:16:36289 GetStoragePartitionMap(browser_context)
290 ->GarbageCollect(std::move(active_paths), done);
[email protected]14acc642012-11-17 12:20:10291}
292
[email protected]b441a8492012-06-06 14:55:57293DownloadManager* BrowserContext::GetDownloadManager(
294 BrowserContext* context) {
mostynbfbcdc27a2015-03-13 17:58:52295 DCHECK_CURRENTLY_ON(BrowserThread::UI);
[email protected]b441a8492012-06-06 14:55:57296 if (!context->GetUserData(kDownloadManagerKeyName)) {
Min Qinddb40ce2017-10-28 03:38:26297 DownloadManager* download_manager = new DownloadManagerImpl(context);
[email protected]d25fda12012-06-12 17:05:03298
avi1ed4a4372017-04-25 05:39:41299 SetDownloadManager(context, base::WrapUnique(download_manager));
[email protected]b441a8492012-06-06 14:55:57300 download_manager->SetDelegate(context->GetDownloadManagerDelegate());
[email protected]b441a8492012-06-06 14:55:57301 }
302
[email protected]eba4a4d2013-05-29 02:18:06303 return static_cast<DownloadManager*>(
304 context->GetUserData(kDownloadManagerKeyName));
[email protected]b441a8492012-06-06 14:55:57305}
306
[email protected]6ef0c3912013-01-25 22:46:34307// static
[email protected]cd501a72014-08-22 19:58:31308storage::ExternalMountPoints* BrowserContext::GetMountPoints(
[email protected]6ef0c3912013-01-25 22:46:34309 BrowserContext* context) {
310 // Ensure that these methods are called on the UI thread, except for
311 // unittests where a UI thread might not have been created.
312 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI) ||
Gabriel Charette2983181c2018-03-28 17:01:09313 !BrowserThread::IsThreadInitialized(BrowserThread::UI));
[email protected]6ef0c3912013-01-25 22:46:34314
315#if defined(OS_CHROMEOS)
316 if (!context->GetUserData(kMountPointsKey)) {
[email protected]cd501a72014-08-22 19:58:31317 scoped_refptr<storage::ExternalMountPoints> mount_points =
318 storage::ExternalMountPoints::CreateRefCounted();
[email protected]6ef0c3912013-01-25 22:46:34319 context->SetUserData(
320 kMountPointsKey,
Jeremy Roman04f27c372017-10-27 15:20:55321 std::make_unique<UserDataAdapter<storage::ExternalMountPoints>>(
avicb129c02017-05-03 06:49:29322 mount_points.get()));
[email protected]6ef0c3912013-01-25 22:46:34323 }
324
[email protected]cd501a72014-08-22 19:58:31325 return UserDataAdapter<storage::ExternalMountPoints>::Get(context,
326 kMountPointsKey);
[email protected]6ef0c3912013-01-25 22:46:34327#else
Ivan Kotenkov2c0d2bb32017-11-01 15:41:28328 return nullptr;
[email protected]6ef0c3912013-01-25 22:46:34329#endif
330}
331
msrameke169ccb2017-04-26 05:21:41332// static
333content::BrowsingDataRemover* content::BrowserContext::GetBrowsingDataRemover(
334 BrowserContext* context) {
335 DCHECK_CURRENTLY_ON(BrowserThread::UI);
336
337 if (!context->GetUserData(kBrowsingDataRemoverKey)) {
338 std::unique_ptr<BrowsingDataRemoverImpl> remover =
Jeremy Roman04f27c372017-10-27 15:20:55339 std::make_unique<BrowsingDataRemoverImpl>(context);
msrameke169ccb2017-04-26 05:21:41340 remover->SetEmbedderDelegate(context->GetBrowsingDataRemoverDelegate());
341 context->SetUserData(kBrowsingDataRemoverKey, std::move(remover));
342 }
343
344 return static_cast<BrowsingDataRemoverImpl*>(
345 context->GetUserData(kBrowsingDataRemoverKey));
346}
347
Andrey Lushnikovebff0442018-07-12 20:02:58348// static
349content::PermissionController* content::BrowserContext::GetPermissionController(
350 BrowserContext* context) {
351 DCHECK_CURRENTLY_ON(BrowserThread::UI);
352
353 if (!context->GetUserData(kPermissionControllerKey)) {
354 context->SetUserData(kPermissionControllerKey,
355 std::make_unique<PermissionControllerImpl>(context));
356 }
357
358 return static_cast<PermissionControllerImpl*>(
359 context->GetUserData(kPermissionControllerKey));
360}
361
[email protected]4c3a23582012-08-18 08:54:34362StoragePartition* BrowserContext::GetStoragePartition(
363 BrowserContext* browser_context,
Dan Elphick6c0d8492017-09-12 09:35:38364 SiteInstance* site_instance,
365 bool can_create) {
[email protected]1bc28312012-11-08 08:31:53366 std::string partition_domain;
367 std::string partition_name;
368 bool in_memory = false;
[email protected]4c3a23582012-08-18 08:54:34369
[email protected]4c3a23582012-08-18 08:54:34370 if (site_instance) {
[email protected]1bc28312012-11-08 08:31:53371 GetContentClient()->browser()->GetStoragePartitionConfigForSite(
[email protected]14acc642012-11-17 12:20:10372 browser_context, site_instance->GetSiteURL(), true,
[email protected]1bc28312012-11-08 08:31:53373 &partition_domain, &partition_name, &in_memory);
[email protected]4c3a23582012-08-18 08:54:34374 }
375
Dan Elphick6c0d8492017-09-12 09:35:38376 return GetStoragePartitionFromConfig(browser_context, partition_domain,
377 partition_name, in_memory, can_create);
[email protected]4c3a23582012-08-18 08:54:34378}
379
[email protected]e94bbcb2012-09-07 05:33:57380StoragePartition* BrowserContext::GetStoragePartitionForSite(
381 BrowserContext* browser_context,
Dan Elphick6c0d8492017-09-12 09:35:38382 const GURL& site,
383 bool can_create) {
[email protected]1bc28312012-11-08 08:31:53384 std::string partition_domain;
385 std::string partition_name;
386 bool in_memory;
[email protected]e94bbcb2012-09-07 05:33:57387
[email protected]1bc28312012-11-08 08:31:53388 GetContentClient()->browser()->GetStoragePartitionConfigForSite(
[email protected]14acc642012-11-17 12:20:10389 browser_context, site, true, &partition_domain, &partition_name,
390 &in_memory);
[email protected]1bc28312012-11-08 08:31:53391
Dan Elphick6c0d8492017-09-12 09:35:38392 return GetStoragePartitionFromConfig(browser_context, partition_domain,
393 partition_name, in_memory, can_create);
[email protected]e94bbcb2012-09-07 05:33:57394}
395
[email protected]4c3a23582012-08-18 08:54:34396void BrowserContext::ForEachStoragePartition(
397 BrowserContext* browser_context,
398 const StoragePartitionCallback& callback) {
399 StoragePartitionImplMap* partition_map =
400 static_cast<StoragePartitionImplMap*>(
a.cavalcantiffab73762015-08-15 02:55:48401 browser_context->GetUserData(kStoragePartitionMapKeyName));
[email protected]4c3a23582012-08-18 08:54:34402 if (!partition_map)
403 return;
404
405 partition_map->ForEach(callback);
406}
407
408StoragePartition* BrowserContext::GetDefaultStoragePartition(
409 BrowserContext* browser_context) {
Ivan Kotenkov2c0d2bb32017-11-01 15:41:28410 return GetStoragePartition(browser_context, nullptr);
[email protected]55eb70e762012-02-20 17:38:39411}
412
tbarzicdb712682015-03-06 06:05:41413// static
[email protected]393b6cb2014-05-15 00:55:12414void BrowserContext::CreateMemoryBackedBlob(BrowserContext* browser_context,
Marijn Kruisselbrink604fd7e72017-10-26 16:31:05415 const char* data,
416 size_t length,
417 const std::string& content_type,
418 BlobCallback callback) {
mostynbfbcdc27a2015-03-13 17:58:52419 DCHECK_CURRENTLY_ON(BrowserThread::UI);
[email protected]393b6cb2014-05-15 00:55:12420
421 ChromeBlobStorageContext* blob_context =
422 ChromeBlobStorageContext::GetFor(browser_context);
Eric Seckler8652dcd52018-09-20 10:42:28423 base::PostTaskWithTraitsAndReplyWithResult(
424 FROM_HERE, {BrowserThread::IO},
Marijn Kruisselbrink604fd7e72017-10-26 16:31:05425 base::BindOnce(&ChromeBlobStorageContext::CreateMemoryBackedBlob,
426 base::WrapRefCounted(blob_context), data, length,
427 content_type),
428 std::move(callback));
[email protected]393b6cb2014-05-15 00:55:12429}
430
[email protected]66e53d0282014-08-07 10:04:35431// static
Xing Liu89860472018-02-09 20:07:02432BrowserContext::BlobContextGetter BrowserContext::GetBlobStorageContext(
433 BrowserContext* browser_context) {
434 DCHECK_CURRENTLY_ON(BrowserThread::UI);
435 scoped_refptr<ChromeBlobStorageContext> chrome_blob_context =
436 ChromeBlobStorageContext::GetFor(browser_context);
Jens Widell7ca8fb42018-02-21 14:11:54437 return base::BindRepeating(&BlobStorageContextGetterForBrowser,
438 chrome_blob_context);
Xing Liu89860472018-02-09 20:07:02439}
440
441// static
Marijn Kruisselbrink9e073a82018-06-18 17:48:58442blink::mojom::BlobPtr BrowserContext::GetBlobPtr(
443 BrowserContext* browser_context,
444 const std::string& uuid) {
445 DCHECK_CURRENTLY_ON(BrowserThread::UI);
446 return ChromeBlobStorageContext::GetBlobPtr(browser_context, uuid);
447}
448
449// static
[email protected]66e53d0282014-08-07 10:04:35450void BrowserContext::DeliverPushMessage(
451 BrowserContext* browser_context,
452 const GURL& origin,
avib7348942015-12-25 20:57:10453 int64_t service_worker_registration_id,
Han Leonc1deace2018-08-03 03:52:53454 base::Optional<std::string> payload,
Peter Beverloo7815db1e02017-07-12 19:03:21455 const base::Callback<void(mojom::PushDeliveryStatus)>& callback) {
mostynbfbcdc27a2015-03-13 17:58:52456 DCHECK_CURRENTLY_ON(BrowserThread::UI);
harknessdd4d2b22016-01-27 19:26:43457 PushMessagingRouter::DeliverMessage(browser_context, origin,
Han Leonc1deace2018-08-03 03:52:53458 service_worker_registration_id,
459 std::move(payload), callback);
[email protected]66e53d0282014-08-07 10:04:35460}
461
falken41f4175162014-10-29 07:03:41462// static
463void BrowserContext::NotifyWillBeDestroyed(BrowserContext* browser_context) {
Lukasz Anforowicz58d0dac2018-03-23 15:48:10464 // Make sure NotifyWillBeDestroyed is idempotent. This helps facilitate the
465 // pattern where NotifyWillBeDestroyed is called from *both*
466 // ShellBrowserContext and its derived classes (e.g.
467 // LayoutTestBrowserContext).
468 if (browser_context->was_notify_will_be_destroyed_called_)
469 return;
470 browser_context->was_notify_will_be_destroyed_called_ = true;
471
Ken Rockot99c5bc742018-07-12 15:36:56472 // Subclasses of BrowserContext may expect there to be no more
473 // RenderProcessHosts using them by the time this function returns. We
474 // therefore explicitly tear down embedded Content Service instances now to
475 // ensure that all their WebContents (and therefore RPHs) are torn down too.
476 browser_context->RemoveUserData(kContentServiceDelegateKey);
477
falken41f4175162014-10-29 07:03:41478 // Service Workers must shutdown before the browser context is destroyed,
479 // since they keep render process hosts alive and the codebase assumes that
480 // render process hosts die before their profile (browser context) dies.
481 ForEachStoragePartition(browser_context,
482 base::Bind(ShutdownServiceWorkerContext));
falken04a6912a2016-09-23 21:06:29483
484 // Shared workers also keep render process hosts alive, and are expected to
Darin Fisherd3768b2f62017-10-07 01:00:16485 // return ref counts to 0 after documents close. However, to ensure that
486 // hosts are destructed now, forcibly release their ref counts here.
falken04a6912a2016-09-23 21:06:29487 for (RenderProcessHost::iterator host_iterator =
488 RenderProcessHost::AllHostsIterator();
489 !host_iterator.IsAtEnd(); host_iterator.Advance()) {
490 RenderProcessHost* host = host_iterator.GetCurrentValue();
Benoit Lizeab08362c2017-07-19 14:50:54491 if (host->GetBrowserContext() == browser_context) {
492 // This will also clean up spare RPH references.
Yutaka Hirano09a9afb92017-08-16 09:11:45493 host->DisableKeepAliveRefCount();
Benoit Lizeab08362c2017-07-19 14:50:54494 }
falken04a6912a2016-09-23 21:06:29495 }
falken41f4175162014-10-29 07:03:41496}
497
[email protected]314c3e22012-02-21 03:57:42498void BrowserContext::EnsureResourceContextInitialized(BrowserContext* context) {
[email protected]7e26ac92012-02-27 20:15:05499 // This will be enough to tickle initialization of BrowserContext if
500 // necessary, which initializes ResourceContext. The reason we don't call
[email protected]4c3a23582012-08-18 08:54:34501 // ResourceContext::InitializeResourceContext() directly here is that
502 // ResourceContext initialization may call back into BrowserContext
503 // and when that call returns it'll end rewriting its UserData map. It will
504 // end up rewriting the same value but this still causes a race condition.
505 //
506 // See http://crbug.com/115678.
507 GetDefaultStoragePartition(context);
[email protected]55eb70e762012-02-20 17:38:39508}
509
[email protected]6e2d3d22012-02-24 18:10:36510void BrowserContext::SaveSessionState(BrowserContext* browser_context) {
[email protected]b1b502e2012-09-16 07:31:43511 StoragePartition* storage_partition =
512 BrowserContext::GetDefaultStoragePartition(browser_context);
[email protected]6e2d3d22012-02-24 18:10:36513
Joshua Bell607cb142017-07-24 19:17:16514 storage::DatabaseTracker* database_tracker =
515 storage_partition->GetDatabaseTracker();
516 database_tracker->task_runner()->PostTask(
517 FROM_HERE,
518 base::BindOnce(&storage::DatabaseTracker::SetForceKeepSessionState,
kylecharda69d882017-10-04 05:49:52519 base::WrapRefCounted(database_tracker)));
Joshua Bell607cb142017-07-24 19:17:16520
Gabriel Charette2983181c2018-03-28 17:01:09521 if (BrowserThread::IsThreadInitialized(BrowserThread::IO)) {
Eric Seckler8652dcd52018-09-20 10:42:28522 base::PostTaskWithTraits(
523 FROM_HERE, {BrowserThread::IO},
tzik4fea24af2017-08-23 11:41:47524 base::BindOnce(
[email protected]6939075a2012-08-28 08:35:53525 &SaveSessionStateOnIOThread,
Clark DuVall385b5a52018-06-14 21:33:32526 base::WrapRefCounted(storage_partition->GetURLRequestContext()),
[email protected]98d6d4562014-06-25 20:57:55527 static_cast<AppCacheServiceImpl*>(
[email protected]63ef85512014-06-05 14:21:26528 storage_partition->GetAppCacheService())));
[email protected]6e2d3d22012-02-24 18:10:36529 }
530
Clark DuVall385b5a52018-06-14 21:33:32531 storage_partition->GetCookieManagerForBrowserProcess()
532 ->SetForceKeepSessionState();
533
[email protected]5f2aa722013-08-07 16:59:41534 DOMStorageContextWrapper* dom_storage_context_proxy =
535 static_cast<DOMStorageContextWrapper*>(
[email protected]b1b502e2012-09-16 07:31:43536 storage_partition->GetDOMStorageContext());
[email protected]5f2aa722013-08-07 16:59:41537 dom_storage_context_proxy->SetForceKeepSessionState();
[email protected]735e20c2012-03-20 01:16:59538
[email protected]89acda82013-06-25 20:52:50539 IndexedDBContextImpl* indexed_db_context_impl =
540 static_cast<IndexedDBContextImpl*>(
[email protected]b1b502e2012-09-16 07:31:43541 storage_partition->GetIndexedDBContext());
[email protected]89acda82013-06-25 20:52:50542 // No task runner in unit tests.
543 if (indexed_db_context_impl->TaskRunner()) {
544 indexed_db_context_impl->TaskRunner()->PostTask(
kylecharda69d882017-10-04 05:49:52545 FROM_HERE,
546 base::BindOnce(&SaveSessionStateOnIndexedDBThread,
547 base::WrapRefCounted(indexed_db_context_impl)));
[email protected]6e2d3d22012-02-24 18:10:36548 }
549}
550
ttr31481dc54b2015-08-06 20:11:26551void BrowserContext::SetDownloadManagerForTesting(
552 BrowserContext* browser_context,
avi1ed4a4372017-04-25 05:39:41553 std::unique_ptr<content::DownloadManager> download_manager) {
554 SetDownloadManager(browser_context, std::move(download_manager));
ttr31481dc54b2015-08-06 20:11:26555}
556
rockot963ad3e82016-04-13 16:33:23557// static
erge69130f52016-03-02 00:13:28558void BrowserContext::Initialize(
559 BrowserContext* browser_context,
560 const base::FilePath& path) {
Ken Rockotda7edc62018-11-10 01:01:45561 const base::Token new_group = base::Token::CreateRandom();
562 ServiceInstanceGroupHolder* holder = static_cast<ServiceInstanceGroupHolder*>(
563 browser_context->GetUserData(kServiceInstanceGroup));
564 if (holder) {
565 file::ForgetServiceInstanceGroupUserDirAssociation(
566 holder->instance_group());
ben6c85c4492016-06-16 20:40:51567 }
Ken Rockotda7edc62018-11-10 01:01:45568 file::AssociateServiceInstanceGroupWithUserDir(new_group, path);
569 RemoveBrowserContextFromInstanceGroupMap(browser_context);
570 GetTokenToContextMap()[new_group] = browser_context;
avi1ed4a4372017-04-25 05:39:41571 browser_context->SetUserData(
Ken Rockotda7edc62018-11-10 01:01:45572 kServiceInstanceGroup,
573 std::make_unique<ServiceInstanceGroupHolder>(new_group));
rockot963ad3e82016-04-13 16:33:23574
bend32292b2016-10-07 00:21:58575 ServiceManagerConnection* service_manager_connection =
576 ServiceManagerConnection::GetForProcess();
fdoray4f155f02016-10-12 11:28:50577 if (service_manager_connection && base::ThreadTaskRunnerHandle::IsSet()) {
rockot963ad3e82016-04-13 16:33:23578 // NOTE: Many unit tests create a TestBrowserContext without initializing
bend32292b2016-10-07 00:21:58579 // Mojo or the global service manager connection.
rockot963ad3e82016-04-13 16:33:23580
rockot400ea35b2016-10-15 19:15:32581 service_manager::mojom::ServicePtr service;
Ken Rockotf4d8a942017-05-13 00:10:37582 auto service_request = mojo::MakeRequest(&service);
rockot963ad3e82016-04-13 16:33:23583
rockot400ea35b2016-10-15 19:15:32584 service_manager::mojom::PIDReceiverPtr pid_receiver;
Ken Rockotda7edc62018-11-10 01:01:45585 service_manager::Identity identity(mojom::kBrowserServiceName, new_group);
benbd3c2482017-01-07 05:48:21586 service_manager_connection->GetConnector()->StartService(
bene6a9f012017-01-07 00:43:43587 identity, std::move(service), mojo::MakeRequest(&pid_receiver));
rockot963ad3e82016-04-13 16:33:23588 pid_receiver->SetPID(base::GetCurrentProcId());
589
benaad37ecd2017-04-12 22:08:20590 service_manager_connection->GetConnector()->StartService(identity);
bend32292b2016-10-07 00:21:58591 BrowserContextServiceManagerConnectionHolder* connection_holder =
592 new BrowserContextServiceManagerConnectionHolder(
bend32292b2016-10-07 00:21:58593 std::move(service_request));
avi1ed4a4372017-04-25 05:39:41594 browser_context->SetUserData(kServiceManagerConnection,
595 base::WrapUnique(connection_holder));
rockot963ad3e82016-04-13 16:33:23596
bend32292b2016-10-07 00:21:58597 ServiceManagerConnection* connection =
598 connection_holder->service_manager_connection();
rockot963ad3e82016-04-13 16:33:23599
ben146248de2016-06-14 15:24:59600 // New embedded service factories should be added to |connection| here.
rockot963ad3e82016-04-13 16:33:23601
Ken Rockot84f58fd2018-06-22 21:54:14602 {
603 service_manager::EmbeddedServiceInfo info;
604 info.factory = base::BindRepeating(&file::CreateFileService);
605 connection->AddEmbeddedService(file::mojom::kServiceName, info);
606 }
607
608 browser_context->SetUserData(
609 kContentServiceDelegateKey,
610 std::make_unique<ContentServiceDelegateHolder>(browser_context));
611
612 {
613 service_manager::EmbeddedServiceInfo info;
614 info.task_runner = base::SequencedTaskRunnerHandle::Get();
615 info.factory = base::BindRepeating(
616 [](BrowserContext* context)
617 -> std::unique_ptr<service_manager::Service> {
618 auto* holder = static_cast<ContentServiceDelegateHolder*>(
619 context->GetUserData(kContentServiceDelegateKey));
620 auto* delegate = holder->delegate();
621 auto service = std::make_unique<content::Service>(delegate);
622 delegate->AddService(service.get());
623 return service;
624 },
625 browser_context);
626 connection->AddEmbeddedService(content::mojom::kServiceName, info);
627 }
tibellab3d36b2017-03-10 02:57:25628
629 ContentBrowserClient::StaticServiceMap services;
630 browser_context->RegisterInProcessServices(&services);
631 for (const auto& entry : services) {
632 connection->AddEmbeddedService(entry.first, entry.second);
633 }
Ken Rockotc7a279c2017-05-04 23:51:45634
635 RegisterCommonBrowserInterfaces(connection);
sammc030f5012017-03-13 02:28:45636 connection->Start();
rockot963ad3e82016-04-13 16:33:23637 }
erge69130f52016-03-02 00:13:28638}
639
rockot963ad3e82016-04-13 16:33:23640// static
Ken Rockotda7edc62018-11-10 01:01:45641const base::Token& BrowserContext::GetServiceInstanceGroupFor(
bencccfe2a2016-03-05 16:54:14642 BrowserContext* browser_context) {
Ken Rockotda7edc62018-11-10 01:01:45643 ServiceInstanceGroupHolder* holder = static_cast<ServiceInstanceGroupHolder*>(
644 browser_context->GetUserData(kServiceInstanceGroup));
645 CHECK(holder) << "Attempting to get the instance group for a BrowserContext "
646 << "that was never Initialized().";
647 return holder->instance_group();
erge69130f52016-03-02 00:13:28648}
649
rockot963ad3e82016-04-13 16:33:23650// static
Ken Rockotda7edc62018-11-10 01:01:45651BrowserContext* BrowserContext::GetBrowserContextForServiceInstanceGroup(
652 const base::Token& instance_group) {
653 auto it = GetTokenToContextMap().find(instance_group);
654 return it != GetTokenToContextMap().end() ? it->second : nullptr;
ben6c85c4492016-06-16 20:40:51655}
656
657// static
rockot400ea35b2016-10-15 19:15:32658service_manager::Connector* BrowserContext::GetConnectorFor(
rockot963ad3e82016-04-13 16:33:23659 BrowserContext* browser_context) {
bend32292b2016-10-07 00:21:58660 ServiceManagerConnection* connection =
661 GetServiceManagerConnectionFor(browser_context);
ben5be0b9132016-08-03 00:17:18662 return connection ? connection->GetConnector() : nullptr;
663}
664
665// static
bend32292b2016-10-07 00:21:58666ServiceManagerConnection* BrowserContext::GetServiceManagerConnectionFor(
ben5be0b9132016-08-03 00:17:18667 BrowserContext* browser_context) {
bend32292b2016-10-07 00:21:58668 BrowserContextServiceManagerConnectionHolder* connection_holder =
669 static_cast<BrowserContextServiceManagerConnectionHolder*>(
670 browser_context->GetUserData(kServiceManagerConnection));
671 return connection_holder ? connection_holder->service_manager_connection()
672 : nullptr;
rockot963ad3e82016-04-13 16:33:23673}
674
Takashi Toyoshima621e2bf2018-09-18 08:05:16675// static
Takashi Toyoshimaa12ecf42018-09-25 07:46:01676const SharedCorsOriginAccessList* BrowserContext::GetSharedCorsOriginAccessList(
Takashi Toyoshima621e2bf2018-09-18 08:05:16677 BrowserContext* browser_context) {
678 return UserDataAdapter<SharedCorsOriginAccessList>::Get(
679 browser_context, kSharedCorsOriginAccessListKey);
680}
681
Takashi Toyoshimaa12ecf42018-09-25 07:46:01682// static
683void BrowserContext::SetCorsOriginAccessListsForOrigin(
684 BrowserContext* browser_context,
685 const url::Origin& source_origin,
686 std::vector<network::mojom::CorsOriginPatternPtr> allow_patterns,
687 std::vector<network::mojom::CorsOriginPatternPtr> block_patterns,
688 base::OnceClosure closure) {
689 if (!base::FeatureList::IsEnabled(network::features::kNetworkService)) {
690 UserDataAdapter<SharedCorsOriginAccessList>::Get(
691 browser_context, kSharedCorsOriginAccessListKey)
692 ->SetForOrigin(source_origin, std::move(allow_patterns),
693 std::move(block_patterns), std::move(closure));
694 } else {
695 auto setter = base::MakeRefCounted<CorsOriginPatternSetter>(
696 source_origin, std::move(allow_patterns), std::move(block_patterns),
697 std::move(closure));
698 ForEachStoragePartition(
699 browser_context, base::BindRepeating(&CorsOriginPatternSetter::SetLists,
700 base::RetainedRef(setter.get())));
701 }
702}
703
mmenkec0b2b8b12017-04-21 16:27:52704BrowserContext::BrowserContext()
Takashi Toyoshima621e2bf2018-09-18 08:05:16705 : unique_id_(base::UnguessableToken::Create().ToString()) {
706 SetUserData(kSharedCorsOriginAccessListKey,
707 std::make_unique<UserDataAdapter<SharedCorsOriginAccessList>>(
708 new SharedCorsOriginAccessListImpl()));
709}
mmenkec0b2b8b12017-04-21 16:27:52710
[email protected]55eb70e762012-02-20 17:38:39711BrowserContext::~BrowserContext() {
Ken Rockotda7edc62018-11-10 01:01:45712 CHECK(GetUserData(kServiceInstanceGroup))
erge69130f52016-03-02 00:13:28713 << "Attempting to destroy a BrowserContext that never called "
714 << "Initialize()";
715
kinukof6ed359c2016-07-26 13:27:21716 DCHECK(!GetUserData(kStoragePartitionMapKeyName))
717 << "StoragePartitionMap is not shut down properly";
718
Lukasz Anforowicz58d0dac2018-03-23 15:48:10719 DCHECK(was_notify_will_be_destroyed_called_);
720
Ken Rockotda7edc62018-11-10 01:01:45721 RemoveBrowserContextFromInstanceGroupMap(this);
ben6c85c4492016-06-16 20:40:51722
[email protected]b441a8492012-06-06 14:55:57723 if (GetUserData(kDownloadManagerKeyName))
724 GetDownloadManager(this)->Shutdown();
[email protected]55eb70e762012-02-20 17:38:39725}
726
kinukof6ed359c2016-07-26 13:27:21727void BrowserContext::ShutdownStoragePartitions() {
728 if (GetUserData(kStoragePartitionMapKeyName))
729 RemoveUserData(kStoragePartitionMapKeyName);
730}
731
mmenkec0b2b8b12017-04-21 16:27:52732std::string BrowserContext::GetMediaDeviceIDSalt() {
Andrey Lushnikovd39d9062018-04-24 17:33:35733 return unique_id_;
mmenkec0b2b8b12017-04-21 16:27:52734}
735
736// static
737std::string BrowserContext::CreateRandomMediaDeviceIDSalt() {
Andrey Lushnikovd39d9062018-04-24 17:33:35738 return base::UnguessableToken::Create().ToString();
739}
740
741const std::string& BrowserContext::UniqueId() const {
742 return unique_id_;
mmenkec0b2b8b12017-04-21 16:27:52743}
744
Chris Cunninghama5b38012017-10-28 07:27:18745media::VideoDecodePerfHistory* BrowserContext::GetVideoDecodePerfHistory() {
746 media::VideoDecodePerfHistory* decode_history =
747 static_cast<media::VideoDecodePerfHistory*>(
748 GetUserData(kVideoDecodePerfHistoryId));
749
750 // Lazily created. Note, this does not trigger loading the DB from disk. That
751 // occurs later upon first VideoDecodePerfHistory API request that requires DB
752 // access. DB operations will not block the UI thread.
753 if (!decode_history) {
chcunningham06b81092018-09-24 20:20:51754 std::unique_ptr<media::VideoDecodeStatsDBImpl> stats_db =
755 media::VideoDecodeStatsDBImpl::Create(
756 GetPath().Append(FILE_PATH_LITERAL("VideoDecodeStats")));
757 auto new_decode_history =
758 std::make_unique<media::VideoDecodePerfHistory>(std::move(stats_db));
759 decode_history = new_decode_history.get();
760
761 SetUserData(kVideoDecodePerfHistoryId, std::move(new_decode_history));
Chris Cunninghama5b38012017-10-28 07:27:18762 }
763
764 return decode_history;
765}
766
Min Qind3ff2ed62018-07-21 06:46:59767download::InProgressDownloadManager*
768BrowserContext::RetriveInProgressDownloadManager() {
769 return nullptr;
770}
771
[email protected]55eb70e762012-02-20 17:38:39772} // namespace content