OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "content/browser/webui/url_data_manager.h" | 5 #include "content/browser/webui/url_data_manager.h" |
6 | 6 |
7 #include <vector> | 7 #include <vector> |
8 | 8 |
9 #include "base/bind.h" | 9 #include "base/bind.h" |
10 #include "base/lazy_instance.h" | 10 #include "base/lazy_instance.h" |
11 #include "base/memory/ref_counted_memory.h" | 11 #include "base/memory/ref_counted_memory.h" |
12 #include "base/memory/scoped_ptr.h" | |
13 #include "base/message_loop.h" | 12 #include "base/message_loop.h" |
14 #include "base/string_util.h" | 13 #include "base/string_util.h" |
15 #include "base/synchronization/lock.h" | 14 #include "base/synchronization/lock.h" |
16 #include "content/browser/webui/url_data_manager_backend.h" | 15 #include "content/browser/webui/url_data_manager_backend.h" |
17 #include "content/browser/webui/web_ui_data_source.h" | 16 #include "content/browser/webui/url_data_source_impl.h" |
| 17 #include "content/browser/webui/web_ui_data_source_impl.h" |
18 #include "content/browser/resource_context_impl.h" | 18 #include "content/browser/resource_context_impl.h" |
19 #include "content/public/browser/browser_context.h" | 19 #include "content/public/browser/browser_context.h" |
20 #include "content/public/browser/browser_thread.h" | 20 #include "content/public/browser/browser_thread.h" |
21 #include "content/public/browser/url_data_source.h" | 21 #include "content/public/browser/url_data_source.h" |
22 | 22 |
23 using content::BrowserContext; | 23 namespace content { |
24 using content::BrowserThread; | |
25 | |
26 namespace { | 24 namespace { |
27 | 25 |
28 const char kURLDataManagerKeyName[] = "url_data_manager"; | 26 const char kURLDataManagerKeyName[] = "url_data_manager"; |
29 | 27 |
30 base::LazyInstance<base::Lock>::Leaky g_delete_lock = LAZY_INSTANCE_INITIALIZER; | 28 base::LazyInstance<base::Lock>::Leaky g_delete_lock = LAZY_INSTANCE_INITIALIZER; |
31 | 29 |
32 ChromeURLDataManager* GetFromBrowserContext(BrowserContext* context) { | 30 URLDataManager* GetFromBrowserContext(BrowserContext* context) { |
33 if (!context->GetUserData(kURLDataManagerKeyName)) { | 31 if (!context->GetUserData(kURLDataManagerKeyName)) { |
34 context->SetUserData(kURLDataManagerKeyName, | 32 context->SetUserData(kURLDataManagerKeyName, new URLDataManager(context)); |
35 new ChromeURLDataManager(context)); | |
36 } | 33 } |
37 return static_cast<ChromeURLDataManager*>( | 34 return static_cast<URLDataManager*>( |
38 context->GetUserData(kURLDataManagerKeyName)); | 35 context->GetUserData(kURLDataManagerKeyName)); |
39 } | 36 } |
40 | 37 |
41 // Invoked on the IO thread to do the actual adding of the DataSource. | 38 // Invoked on the IO thread to do the actual adding of the DataSource. |
42 static void AddDataSourceOnIOThread( | 39 static void AddDataSourceOnIOThread( |
43 content::ResourceContext* resource_context, | 40 ResourceContext* resource_context, |
44 scoped_refptr<URLDataSourceImpl> data_source) { | 41 scoped_refptr<URLDataSourceImpl> data_source) { |
45 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 42 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
46 GetURLDataManagerForResourceContext(resource_context)->AddDataSource( | 43 GetURLDataManagerForResourceContext(resource_context)->AddDataSource( |
47 data_source.get()); | 44 data_source.get()); |
48 } | 45 } |
49 | 46 |
50 } // namespace | 47 } // namespace |
51 | 48 |
| 49 // static |
| 50 URLDataManager::URLDataSources* URLDataManager::data_sources_ = NULL; |
52 | 51 |
53 // static | 52 URLDataManager::URLDataManager(BrowserContext* browser_context) |
54 ChromeURLDataManager::URLDataSources* ChromeURLDataManager::data_sources_ = | |
55 NULL; | |
56 | |
57 ChromeURLDataManager::ChromeURLDataManager( | |
58 content::BrowserContext* browser_context) | |
59 : browser_context_(browser_context) { | 53 : browser_context_(browser_context) { |
60 } | 54 } |
61 | 55 |
62 ChromeURLDataManager::~ChromeURLDataManager() { | 56 URLDataManager::~URLDataManager() { |
63 } | 57 } |
64 | 58 |
65 void ChromeURLDataManager::AddDataSource(URLDataSourceImpl* source) { | 59 void URLDataManager::AddDataSource(URLDataSourceImpl* source) { |
66 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 60 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
67 BrowserThread::PostTask( | 61 BrowserThread::PostTask( |
68 BrowserThread::IO, FROM_HERE, | 62 BrowserThread::IO, FROM_HERE, |
69 base::Bind(&AddDataSourceOnIOThread, | 63 base::Bind(&AddDataSourceOnIOThread, |
70 browser_context_->GetResourceContext(), | 64 browser_context_->GetResourceContext(), |
71 make_scoped_refptr(source))); | 65 make_scoped_refptr(source))); |
72 } | 66 } |
73 | 67 |
74 // static | 68 // static |
75 void ChromeURLDataManager::DeleteDataSources() { | 69 void URLDataManager::DeleteDataSources() { |
76 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 70 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
77 URLDataSources sources; | 71 URLDataSources sources; |
78 { | 72 { |
79 base::AutoLock lock(g_delete_lock.Get()); | 73 base::AutoLock lock(g_delete_lock.Get()); |
80 if (!data_sources_) | 74 if (!data_sources_) |
81 return; | 75 return; |
82 data_sources_->swap(sources); | 76 data_sources_->swap(sources); |
83 } | 77 } |
84 for (size_t i = 0; i < sources.size(); ++i) | 78 for (size_t i = 0; i < sources.size(); ++i) |
85 delete sources[i]; | 79 delete sources[i]; |
86 } | 80 } |
87 | 81 |
88 // static | 82 // static |
89 void ChromeURLDataManager::DeleteDataSource( | 83 void URLDataManager::DeleteDataSource(const URLDataSourceImpl* data_source) { |
90 const URLDataSourceImpl* data_source) { | |
91 // Invoked when a DataSource is no longer referenced and needs to be deleted. | 84 // Invoked when a DataSource is no longer referenced and needs to be deleted. |
92 if (BrowserThread::CurrentlyOn(BrowserThread::UI)) { | 85 if (BrowserThread::CurrentlyOn(BrowserThread::UI)) { |
93 // We're on the UI thread, delete right away. | 86 // We're on the UI thread, delete right away. |
94 delete data_source; | 87 delete data_source; |
95 return; | 88 return; |
96 } | 89 } |
97 | 90 |
98 // We're not on the UI thread, add the DataSource to the list of DataSources | 91 // We're not on the UI thread, add the DataSource to the list of DataSources |
99 // to delete. | 92 // to delete. |
100 bool schedule_delete = false; | 93 bool schedule_delete = false; |
101 { | 94 { |
102 base::AutoLock lock(g_delete_lock.Get()); | 95 base::AutoLock lock(g_delete_lock.Get()); |
103 if (!data_sources_) | 96 if (!data_sources_) |
104 data_sources_ = new URLDataSources(); | 97 data_sources_ = new URLDataSources(); |
105 schedule_delete = data_sources_->empty(); | 98 schedule_delete = data_sources_->empty(); |
106 data_sources_->push_back(data_source); | 99 data_sources_->push_back(data_source); |
107 } | 100 } |
108 if (schedule_delete) { | 101 if (schedule_delete) { |
109 // Schedule a task to delete the DataSource back on the UI thread. | 102 // Schedule a task to delete the DataSource back on the UI thread. |
110 BrowserThread::PostTask( | 103 BrowserThread::PostTask( |
111 BrowserThread::UI, FROM_HERE, | 104 BrowserThread::UI, FROM_HERE, |
112 base::Bind(&ChromeURLDataManager::DeleteDataSources)); | 105 base::Bind(&URLDataManager::DeleteDataSources)); |
113 } | 106 } |
114 } | 107 } |
115 | 108 |
116 // static | 109 // static |
117 void ChromeURLDataManager::AddDataSource( | 110 void URLDataManager::AddDataSource(BrowserContext* browser_context, |
118 content::BrowserContext* browser_context, | 111 URLDataSource* source) { |
119 content::URLDataSource* source) { | |
120 GetFromBrowserContext(browser_context)-> | 112 GetFromBrowserContext(browser_context)-> |
121 AddDataSource(new URLDataSourceImpl(source->GetSource(), source)); | 113 AddDataSource(new URLDataSourceImpl(source->GetSource(), source)); |
122 } | 114 } |
123 | 115 |
124 // static | 116 // static |
125 void ChromeURLDataManager::AddWebUIDataSource( | 117 void URLDataManager::AddWebUIDataSource(BrowserContext* browser_context, |
126 content::BrowserContext* browser_context, | 118 WebUIDataSource* source) { |
127 content::WebUIDataSource* source) { | 119 WebUIDataSourceImpl* impl = static_cast<WebUIDataSourceImpl*>(source); |
128 ChromeWebUIDataSource* impl = static_cast<ChromeWebUIDataSource*>(source); | |
129 GetFromBrowserContext(browser_context)->AddDataSource(impl); | 120 GetFromBrowserContext(browser_context)->AddDataSource(impl); |
130 } | 121 } |
131 | 122 |
132 // static | 123 // static |
133 bool ChromeURLDataManager::IsScheduledForDeletion( | 124 bool URLDataManager::IsScheduledForDeletion( |
134 const URLDataSourceImpl* data_source) { | 125 const URLDataSourceImpl* data_source) { |
135 base::AutoLock lock(g_delete_lock.Get()); | 126 base::AutoLock lock(g_delete_lock.Get()); |
136 if (!data_sources_) | 127 if (!data_sources_) |
137 return false; | 128 return false; |
138 return std::find(data_sources_->begin(), data_sources_->end(), data_source) != | 129 return std::find(data_sources_->begin(), data_sources_->end(), data_source) != |
139 data_sources_->end(); | 130 data_sources_->end(); |
140 } | 131 } |
141 | 132 |
142 URLDataSourceImpl::URLDataSourceImpl(const std::string& source_name, | 133 } // namespace content |
143 content::URLDataSource* source) | |
144 : source_name_(source_name), | |
145 backend_(NULL), | |
146 source_(source) { | |
147 } | |
148 | |
149 URLDataSourceImpl::~URLDataSourceImpl() { | |
150 } | |
151 | |
152 void URLDataSourceImpl::SendResponse( | |
153 int request_id, | |
154 base::RefCountedMemory* bytes) { | |
155 // Take a ref-pointer on entry so byte->Release() will always get called. | |
156 scoped_refptr<base::RefCountedMemory> bytes_ptr(bytes); | |
157 if (ChromeURLDataManager::IsScheduledForDeletion(this)) { | |
158 // We're scheduled for deletion. Servicing the request would result in | |
159 // this->AddRef being invoked, even though the ref count is 0 and 'this' is | |
160 // about to be deleted. If the AddRef were allowed through, when 'this' is | |
161 // released it would be deleted again. | |
162 // | |
163 // This scenario occurs with DataSources that make history requests. Such | |
164 // DataSources do a history query in |StartDataRequest| and the request is | |
165 // live until the object is deleted (history requests don't up the ref | |
166 // count). This means it's entirely possible for the DataSource to invoke | |
167 // |SendResponse| between the time when there are no more refs and the time | |
168 // when the object is deleted. | |
169 return; | |
170 } | |
171 BrowserThread::PostTask( | |
172 BrowserThread::IO, FROM_HERE, | |
173 base::Bind(&URLDataSourceImpl::SendResponseOnIOThread, this, request_id, | |
174 bytes_ptr)); | |
175 } | |
176 | |
177 void URLDataSourceImpl::SendResponseOnIOThread( | |
178 int request_id, | |
179 scoped_refptr<base::RefCountedMemory> bytes) { | |
180 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
181 if (backend_) | |
182 backend_->DataAvailable(request_id, bytes); | |
183 } | |
OLD | NEW |