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 "chrome/browser/speech/chrome_speech_recognition_manager_delegate.h" | 5 #include "chrome/browser/speech/chrome_speech_recognition_manager_delegate.h" |
6 | 6 |
7 #include <set> | 7 #include <set> |
8 #include <string> | 8 #include <string> |
9 | 9 |
10 #include "base/bind.h" | 10 #include "base/bind.h" |
11 #include "base/prefs/pref_service.h" | 11 #include "base/prefs/pref_service.h" |
12 #include "base/strings/utf_string_conversions.h" | 12 #include "base/strings/utf_string_conversions.h" |
13 #include "base/synchronization/lock.h" | 13 #include "base/synchronization/lock.h" |
14 #include "base/threading/thread_restrictions.h" | 14 #include "base/threading/thread_restrictions.h" |
15 #include "chrome/browser/browser_process.h" | 15 #include "chrome/browser/browser_process.h" |
16 #include "chrome/browser/profiles/profile_manager.h" | 16 #include "chrome/browser/profiles/profile_manager.h" |
17 #include "chrome/browser/tab_contents/tab_util.h" | 17 #include "chrome/browser/tab_contents/tab_util.h" |
18 #include "chrome/common/pref_names.h" | 18 #include "chrome/common/pref_names.h" |
19 #include "chrome/common/url_constants.h" | 19 #include "chrome/common/url_constants.h" |
20 #include "content/public/browser/browser_thread.h" | 20 #include "content/public/browser/browser_thread.h" |
21 #include "content/public/browser/notification_registrar.h" | 21 #include "content/public/browser/notification_registrar.h" |
22 #include "content/public/browser/notification_source.h" | 22 #include "content/public/browser/notification_source.h" |
23 #include "content/public/browser/notification_types.h" | 23 #include "content/public/browser/notification_types.h" |
24 #include "content/public/browser/render_frame_host.h" | |
24 #include "content/public/browser/render_process_host.h" | 25 #include "content/public/browser/render_process_host.h" |
25 #include "content/public/browser/render_view_host.h" | 26 #include "content/public/browser/render_view_host.h" |
26 #include "content/public/browser/resource_context.h" | 27 #include "content/public/browser/resource_context.h" |
27 #include "content/public/browser/speech_recognition_manager.h" | 28 #include "content/public/browser/speech_recognition_manager.h" |
28 #include "content/public/browser/speech_recognition_session_config.h" | 29 #include "content/public/browser/speech_recognition_session_config.h" |
29 #include "content/public/browser/speech_recognition_session_context.h" | 30 #include "content/public/browser/speech_recognition_session_context.h" |
30 #include "content/public/browser/web_contents.h" | 31 #include "content/public/browser/web_contents.h" |
31 #include "content/public/common/speech_recognition_error.h" | 32 #include "content/public/common/speech_recognition_error.h" |
32 #include "content/public/common/speech_recognition_result.h" | 33 #include "content/public/common/speech_recognition_result.h" |
33 #include "net/url_request/url_request_context_getter.h" | 34 #include "net/url_request/url_request_context_getter.h" |
34 | 35 |
35 #if defined(OS_WIN) | 36 #if defined(OS_WIN) |
36 #include "chrome/installer/util/wmi.h" | 37 #include "chrome/installer/util/wmi.h" |
37 #endif | 38 #endif |
38 | 39 |
39 #if defined(ENABLE_EXTENSIONS) | 40 #if defined(ENABLE_EXTENSIONS) |
40 #include "chrome/browser/extensions/extension_service.h" | 41 #include "chrome/browser/extensions/extension_service.h" |
41 #include "extensions/browser/view_type_utils.h" | 42 #include "extensions/browser/view_type_utils.h" |
42 #endif | 43 #endif |
43 | 44 |
44 using content::BrowserThread; | 45 using content::BrowserThread; |
45 using content::SpeechRecognitionManager; | 46 using content::SpeechRecognitionManager; |
46 using content::WebContents; | 47 using content::WebContents; |
47 | 48 |
48 namespace speech { | 49 namespace speech { |
49 | 50 |
50 namespace { | 51 namespace { |
51 | 52 |
52 void TabClosedCallbackOnIOThread(int render_process_id, int render_view_id) { | 53 void TabClosedCallbackOnIOThread(WebContents* web_contents) { |
53 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 54 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
54 | 55 |
55 SpeechRecognitionManager* manager = SpeechRecognitionManager::GetInstance(); | 56 SpeechRecognitionManager* manager = SpeechRecognitionManager::GetInstance(); |
56 // |manager| becomes NULL if a browser shutdown happens between the post of | 57 // |manager| becomes NULL if a browser shutdown happens between the post of |
57 // this task (from the UI thread) and this call (on the IO thread). In this | 58 // this task (from the UI thread) and this call (on the IO thread). In this |
58 // case we just return. | 59 // case we just return. |
59 if (!manager) | 60 if (!manager) |
60 return; | 61 return; |
61 | 62 |
62 manager->AbortAllSessionsForRenderView(render_process_id, render_view_id); | 63 |
64 web_contents->ForEachFrame( | |
Charlie Reis
2014/11/11 04:37:31
This is illegal. We can't use a WebContents on th
| |
65 base::Bind(&SpeechRecognitionManager::AbortAllSessionsForRenderFrame, | |
66 base::Unretained(manager))); | |
63 } | 67 } |
64 | 68 |
65 } // namespace | 69 } // namespace |
66 | 70 |
67 | 71 |
68 // Asynchronously fetches the PC and audio hardware/driver info if | 72 // Asynchronously fetches the PC and audio hardware/driver info if |
69 // the user has opted into UMA. This information is sent with speech input | 73 // the user has opted into UMA. This information is sent with speech input |
70 // requests to the server for identifying and improving quality issues with | 74 // requests to the server for identifying and improving quality issues with |
71 // specific device configurations. | 75 // specific device configurations. |
72 class ChromeSpeechRecognitionManagerDelegate::OptionalRequestInfo | 76 class ChromeSpeechRecognitionManagerDelegate::OptionalRequestInfo |
(...skipping 60 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
133 | 137 |
134 // Simple utility to get notified when a WebContent (a tab or an extension's | 138 // Simple utility to get notified when a WebContent (a tab or an extension's |
135 // background page) is closed or crashes. The callback will always be called on | 139 // background page) is closed or crashes. The callback will always be called on |
136 // the UI thread. | 140 // the UI thread. |
137 // There is no restriction on the constructor, however this class must be | 141 // There is no restriction on the constructor, however this class must be |
138 // destroyed on the UI thread, due to the NotificationRegistrar dependency. | 142 // destroyed on the UI thread, due to the NotificationRegistrar dependency. |
139 class ChromeSpeechRecognitionManagerDelegate::TabWatcher | 143 class ChromeSpeechRecognitionManagerDelegate::TabWatcher |
140 : public base::RefCountedThreadSafe<TabWatcher>, | 144 : public base::RefCountedThreadSafe<TabWatcher>, |
141 public content::NotificationObserver { | 145 public content::NotificationObserver { |
142 public: | 146 public: |
143 typedef base::Callback<void(int render_process_id, int render_view_id)> | 147 typedef base::Callback<void(WebContents* web_contents)> |
144 TabClosedCallback; | 148 TabClosedCallback; |
145 | 149 |
146 explicit TabWatcher(TabClosedCallback tab_closed_callback) | 150 explicit TabWatcher(TabClosedCallback tab_closed_callback) |
147 : tab_closed_callback_(tab_closed_callback) { | 151 : tab_closed_callback_(tab_closed_callback) { |
148 } | 152 } |
149 | 153 |
150 // Starts monitoring the WebContents corresponding to the given | 154 // Starts monitoring the WebContents corresponding to the given |
151 // |render_process_id|, |render_view_id| pair, invoking |tab_closed_callback_| | 155 // |render_process_id|, |render_view_id| pair, invoking |tab_closed_callback_| |
152 // if closed/unloaded. | 156 // if closed/unloaded. |
153 void Watch(int render_process_id, int render_view_id) { | 157 void Watch(content::RenderFrameHost* render_frame_host) { |
154 if (!BrowserThread::CurrentlyOn(BrowserThread::UI)) { | 158 if (!BrowserThread::CurrentlyOn(BrowserThread::UI)) { |
155 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, base::Bind( | 159 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, base::Bind( |
156 &TabWatcher::Watch, this, render_process_id, render_view_id)); | 160 &TabWatcher::Watch, this, base::Unretained(render_frame_host))); |
157 return; | 161 return; |
158 } | 162 } |
159 | 163 |
160 WebContents* web_contents = tab_util::GetWebContentsByID(render_process_id, | 164 WebContents* web_contents = |
161 render_view_id); | 165 WebContents::FromRenderFrameHost(render_frame_host); |
166 | |
162 // Sessions initiated by speech input extension APIs will end up in a NULL | 167 // Sessions initiated by speech input extension APIs will end up in a NULL |
163 // WebContent here, but they are properly managed by the | 168 // WebContent here, but they are properly managed by the |
164 // chrome::SpeechInputExtensionManager. However, sessions initiated within a | 169 // chrome::SpeechInputExtensionManager. However, sessions initiated within a |
165 // extension using the (new) speech JS APIs, will be properly handled here. | 170 // extension using the (new) speech JS APIs, will be properly handled here. |
166 // TODO(primiano) turn this line into a DCHECK once speech input extension | 171 // TODO(primiano) turn this line into a DCHECK once speech input extension |
167 // API is deprecated. | 172 // API is deprecated. |
168 if (!web_contents) | 173 if (!web_contents) |
169 return; | 174 return; |
170 | 175 |
171 // Avoid multiple registrations on |registrar_| for the same |web_contents|. | 176 // Avoid multiple registrations on |registrar_| for the same |web_contents|. |
172 if (FindWebContents(web_contents) != registered_web_contents_.end()) { | 177 if (registered_web_contents_.find(web_contents) != |
178 registered_web_contents_.end()) { | |
173 return; | 179 return; |
174 } | 180 } |
175 registered_web_contents_.push_back( | 181 registered_web_contents_.insert(web_contents); |
176 WebContentsInfo(web_contents, render_process_id, render_view_id)); | |
177 | 182 |
178 // Lazy initialize the registrar. | 183 // Lazy initialize the registrar. |
179 if (!registrar_.get()) | 184 if (!registrar_.get()) |
180 registrar_.reset(new content::NotificationRegistrar()); | 185 registrar_.reset(new content::NotificationRegistrar()); |
181 | 186 |
182 registrar_->Add(this, | 187 registrar_->Add(this, |
183 content::NOTIFICATION_RENDER_VIEW_HOST_CHANGED, | 188 content::NOTIFICATION_RENDER_VIEW_HOST_CHANGED, |
184 content::Source<WebContents>(web_contents)); | 189 content::Source<WebContents>(web_contents)); |
185 registrar_->Add(this, | 190 registrar_->Add(this, |
186 content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED, | 191 content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED, |
187 content::Source<WebContents>(web_contents)); | 192 content::Source<WebContents>(web_contents)); |
188 } | 193 } |
189 | 194 |
190 // content::NotificationObserver implementation. | 195 // content::NotificationObserver implementation. |
191 void Observe(int type, | 196 void Observe(int type, |
192 const content::NotificationSource& source, | 197 const content::NotificationSource& source, |
193 const content::NotificationDetails& details) override { | 198 const content::NotificationDetails& details) override { |
194 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 199 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
195 DCHECK(type == content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED || | 200 DCHECK(type == content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED || |
196 type == content::NOTIFICATION_RENDER_VIEW_HOST_CHANGED); | 201 type == content::NOTIFICATION_RENDER_VIEW_HOST_CHANGED); |
197 | 202 |
198 WebContents* web_contents = content::Source<WebContents>(source).ptr(); | 203 WebContents* web_contents = content::Source<WebContents>(source).ptr(); |
199 std::vector<WebContentsInfo>::iterator iter = FindWebContents(web_contents); | 204 DCHECK(registered_web_contents_.find(web_contents) != |
200 DCHECK(iter != registered_web_contents_.end()); | 205 registered_web_contents_.end()); |
201 int render_process_id = iter->render_process_id; | 206 registered_web_contents_.erase(web_contents); |
202 int render_view_id = iter->render_view_id; | |
203 registered_web_contents_.erase(iter); | |
204 | 207 |
205 registrar_->Remove(this, | 208 registrar_->Remove(this, |
206 content::NOTIFICATION_RENDER_VIEW_HOST_CHANGED, | 209 content::NOTIFICATION_RENDER_VIEW_HOST_CHANGED, |
207 content::Source<WebContents>(web_contents)); | 210 content::Source<WebContents>(web_contents)); |
208 registrar_->Remove(this, | 211 registrar_->Remove(this, |
209 content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED, | 212 content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED, |
210 content::Source<WebContents>(web_contents)); | 213 content::Source<WebContents>(web_contents)); |
211 | 214 |
212 tab_closed_callback_.Run(render_process_id, render_view_id); | 215 tab_closed_callback_.Run(web_contents); |
213 } | 216 } |
214 | 217 |
215 private: | 218 private: |
216 struct WebContentsInfo { | |
217 WebContentsInfo(content::WebContents* web_contents, | |
218 int render_process_id, | |
219 int render_view_id) | |
220 : web_contents(web_contents), | |
221 render_process_id(render_process_id), | |
222 render_view_id(render_view_id) {} | |
223 | |
224 ~WebContentsInfo() {} | |
225 | |
226 content::WebContents* web_contents; | |
227 int render_process_id; | |
228 int render_view_id; | |
229 }; | |
230 | |
231 friend class base::RefCountedThreadSafe<TabWatcher>; | 219 friend class base::RefCountedThreadSafe<TabWatcher>; |
232 | 220 |
233 ~TabWatcher() override { | 221 ~TabWatcher() override { |
234 // Must be destroyed on the UI thread due to |registrar_| non thread-safety. | 222 // Must be destroyed on the UI thread due to |registrar_| non thread-safety. |
235 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 223 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
236 } | 224 } |
237 | 225 |
238 // Helper function to find the iterator in |registered_web_contents_| which | |
239 // contains |web_contents|. | |
240 std::vector<WebContentsInfo>::iterator FindWebContents( | |
241 content::WebContents* web_contents) { | |
242 for (std::vector<WebContentsInfo>::iterator i( | |
243 registered_web_contents_.begin()); | |
244 i != registered_web_contents_.end(); ++i) { | |
245 if (i->web_contents == web_contents) | |
246 return i; | |
247 } | |
248 | |
249 return registered_web_contents_.end(); | |
250 } | |
251 | |
252 // Lazy-initialized and used on the UI thread to handle web contents | 226 // Lazy-initialized and used on the UI thread to handle web contents |
253 // notifications (tab closing). | 227 // notifications (tab closing). |
254 scoped_ptr<content::NotificationRegistrar> registrar_; | 228 scoped_ptr<content::NotificationRegistrar> registrar_; |
255 | 229 |
256 // Keeps track of which WebContent(s) have been registered, in order to avoid | 230 // Keeps track of which WebContent(s) have been registered, in order to avoid |
257 // double registrations on |registrar_| and to pass the correct render | 231 // double registrations on |registrar_| and to pass the correct render |
258 // process id and render view id to |tab_closed_callback_| after the process | 232 // process id and render view id to |tab_closed_callback_| after the process |
259 // has gone away. | 233 // has gone away. |
260 std::vector<WebContentsInfo> registered_web_contents_; | 234 std::set<WebContents*> registered_web_contents_; |
261 | 235 |
262 // Callback used to notify, on the thread specified by |callback_thread_| the | 236 // Callback used to notify, on the thread specified by |callback_thread_| the |
263 // closure of a registered tab. | 237 // closure of a registered tab. |
264 TabClosedCallback tab_closed_callback_; | 238 TabClosedCallback tab_closed_callback_; |
265 | 239 |
266 DISALLOW_COPY_AND_ASSIGN(TabWatcher); | 240 DISALLOW_COPY_AND_ASSIGN(TabWatcher); |
267 }; | 241 }; |
268 | 242 |
269 ChromeSpeechRecognitionManagerDelegate | 243 ChromeSpeechRecognitionManagerDelegate |
270 ::ChromeSpeechRecognitionManagerDelegate() { | 244 ::ChromeSpeechRecognitionManagerDelegate() { |
271 } | 245 } |
272 | 246 |
273 ChromeSpeechRecognitionManagerDelegate | 247 ChromeSpeechRecognitionManagerDelegate |
274 ::~ChromeSpeechRecognitionManagerDelegate() { | 248 ::~ChromeSpeechRecognitionManagerDelegate() { |
275 } | 249 } |
276 | 250 |
277 void ChromeSpeechRecognitionManagerDelegate::TabClosedCallback( | 251 void ChromeSpeechRecognitionManagerDelegate::TabClosedCallback( |
278 int render_process_id, int render_view_id) { | 252 WebContents* web_contents) { |
279 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 253 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
280 | 254 |
281 // Tell the S.R. Manager (which lives on the IO thread) to abort all the | 255 // Tell the S.R. Manager (which lives on the IO thread) to abort all the |
282 // sessions for the given renderer view. | 256 // sessions for the given WebContents. |
283 BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, base::Bind( | 257 BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, base::Bind( |
284 &TabClosedCallbackOnIOThread, render_process_id, render_view_id)); | 258 &TabClosedCallbackOnIOThread, base::Unretained(web_contents))); |
285 } | 259 } |
286 | 260 |
287 void ChromeSpeechRecognitionManagerDelegate::OnRecognitionStart( | 261 void ChromeSpeechRecognitionManagerDelegate::OnRecognitionStart( |
288 int session_id) { | 262 int session_id) { |
289 const content::SpeechRecognitionSessionContext& context = | 263 const content::SpeechRecognitionSessionContext& context = |
290 SpeechRecognitionManager::GetInstance()->GetSessionContext(session_id); | 264 SpeechRecognitionManager::GetInstance()->GetSessionContext(session_id); |
291 | 265 |
292 // Register callback to auto abort session on tab closure. | 266 // Register callback to auto abort session on tab closure. |
293 // |tab_watcher_| is lazyly istantiated on the first call. | 267 // |tab_watcher_| is lazyly istantiated on the first call. |
294 if (!tab_watcher_.get()) { | 268 if (!tab_watcher_.get()) { |
295 tab_watcher_ = new TabWatcher( | 269 tab_watcher_ = new TabWatcher( |
296 base::Bind(&ChromeSpeechRecognitionManagerDelegate::TabClosedCallback, | 270 base::Bind(&ChromeSpeechRecognitionManagerDelegate::TabClosedCallback, |
297 base::Unretained(this))); | 271 base::Unretained(this))); |
298 } | 272 } |
299 tab_watcher_->Watch(context.render_process_id, context.render_view_id); | 273 tab_watcher_->Watch(context.render_frame_host); |
300 } | 274 } |
301 | 275 |
302 void ChromeSpeechRecognitionManagerDelegate::OnAudioStart(int session_id) { | 276 void ChromeSpeechRecognitionManagerDelegate::OnAudioStart(int session_id) { |
303 } | 277 } |
304 | 278 |
305 void ChromeSpeechRecognitionManagerDelegate::OnEnvironmentEstimationComplete( | 279 void ChromeSpeechRecognitionManagerDelegate::OnEnvironmentEstimationComplete( |
306 int session_id) { | 280 int session_id) { |
307 } | 281 } |
308 | 282 |
309 void ChromeSpeechRecognitionManagerDelegate::OnSoundStart(int session_id) { | 283 void ChromeSpeechRecognitionManagerDelegate::OnSoundStart(int session_id) { |
(...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
350 } | 324 } |
351 | 325 |
352 void ChromeSpeechRecognitionManagerDelegate::CheckRecognitionIsAllowed( | 326 void ChromeSpeechRecognitionManagerDelegate::CheckRecognitionIsAllowed( |
353 int session_id, | 327 int session_id, |
354 base::Callback<void(bool ask_user, bool is_allowed)> callback) { | 328 base::Callback<void(bool ask_user, bool is_allowed)> callback) { |
355 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 329 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
356 | 330 |
357 const content::SpeechRecognitionSessionContext& context = | 331 const content::SpeechRecognitionSessionContext& context = |
358 SpeechRecognitionManager::GetInstance()->GetSessionContext(session_id); | 332 SpeechRecognitionManager::GetInstance()->GetSessionContext(session_id); |
359 | 333 |
360 // Make sure that initiators (extensions/web pages) properly set the | 334 // Check that the context is appropriate, and whether or not we need to |
361 // |render_process_id| field, which is needed later to retrieve the profile. | 335 // request permission from the user. |
362 DCHECK_NE(context.render_process_id, 0); | |
363 | |
364 int render_process_id = context.render_process_id; | |
365 int render_view_id = context.render_view_id; | |
366 if (context.embedder_render_process_id) { | |
367 // If this is a request originated from a guest, we need to re-route the | |
368 // permission check through the embedder (app). | |
369 render_process_id = context.embedder_render_process_id; | |
370 render_view_id = context.embedder_render_view_id; | |
371 } | |
372 | |
373 // Check that the render view type is appropriate, and whether or not we | |
374 // need to request permission from the user. | |
375 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, | 336 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, |
376 base::Bind(&CheckRenderViewType, | 337 base::Bind(&CheckRenderViewType, callback, context)); |
377 callback, | |
378 render_process_id, | |
379 render_view_id)); | |
380 } | 338 } |
381 | 339 |
382 content::SpeechRecognitionEventListener* | 340 content::SpeechRecognitionEventListener* |
383 ChromeSpeechRecognitionManagerDelegate::GetEventListener() { | 341 ChromeSpeechRecognitionManagerDelegate::GetEventListener() { |
384 return this; | 342 return this; |
385 } | 343 } |
386 | 344 |
387 bool ChromeSpeechRecognitionManagerDelegate::FilterProfanities( | 345 bool ChromeSpeechRecognitionManagerDelegate::FilterProfanities( |
388 int render_process_id) { | 346 int render_process_id) { |
389 content::RenderProcessHost* rph = | 347 content::RenderProcessHost* rph = |
390 content::RenderProcessHost::FromID(render_process_id); | 348 content::RenderProcessHost::FromID(render_process_id); |
391 if (!rph) // Guard against race conditions on RPH lifetime. | 349 if (!rph) // Guard against race conditions on RPH lifetime. |
392 return true; | 350 return true; |
393 | 351 |
394 return Profile::FromBrowserContext(rph->GetBrowserContext())->GetPrefs()-> | 352 return Profile::FromBrowserContext(rph->GetBrowserContext())->GetPrefs()-> |
395 GetBoolean(prefs::kSpeechRecognitionFilterProfanities); | 353 GetBoolean(prefs::kSpeechRecognitionFilterProfanities); |
396 } | 354 } |
397 | 355 |
398 // static. | 356 // static. |
399 void ChromeSpeechRecognitionManagerDelegate::CheckRenderViewType( | 357 void ChromeSpeechRecognitionManagerDelegate::CheckRenderViewType( |
400 base::Callback<void(bool ask_user, bool is_allowed)> callback, | 358 base::Callback<void(bool ask_user, bool is_allowed)> callback, |
401 int render_process_id, | 359 const content::SpeechRecognitionSessionContext& context) { |
402 int render_view_id) { | |
403 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 360 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
404 const content::RenderViewHost* render_view_host = | |
405 content::RenderViewHost::FromID(render_process_id, render_view_id); | |
406 | |
407 bool allowed = false; | 361 bool allowed = false; |
408 bool check_permission = false; | 362 bool check_permission = false; |
409 | 363 |
410 if (!render_view_host) { | 364 content::RenderFrameHost* render_frame_host = context.render_frame_host; |
365 | |
366 if (context.embedder_render_process_id) { | |
367 // If this is a request originated from a guest, we need to re-route the | |
368 // permission check through the embedder (app). | |
369 render_frame_host = content::RenderFrameHost::FromID( | |
370 context.embedder_render_process_id, context.embedder_render_frame_id); | |
371 } | |
372 | |
373 if (!render_frame_host) { | |
411 // This happens for extensions. Manifest should be checked for permission. | 374 // This happens for extensions. Manifest should be checked for permission. |
412 allowed = true; | 375 allowed = true; |
413 check_permission = false; | 376 check_permission = false; |
414 BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, | 377 BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, |
415 base::Bind(callback, check_permission, allowed)); | 378 base::Bind(callback, check_permission, allowed)); |
416 return; | 379 return; |
417 } | 380 } |
418 | 381 |
419 WebContents* web_contents = WebContents::FromRenderViewHost(render_view_host); | 382 WebContents* web_contents = |
383 WebContents::FromRenderFrameHost(render_frame_host); | |
420 | 384 |
421 // chrome://app-list/ uses speech recognition. | 385 // chrome://app-list/ uses speech recognition. |
422 if (web_contents->GetCommittedWebUI() && | 386 if (web_contents->GetCommittedWebUI() && |
423 web_contents->GetLastCommittedURL().spec() == | 387 web_contents->GetLastCommittedURL().spec() == |
424 chrome::kChromeUIAppListStartPageURL) { | 388 chrome::kChromeUIAppListStartPageURL) { |
425 allowed = true; | 389 allowed = true; |
426 check_permission = false; | 390 check_permission = false; |
427 } | 391 } |
428 | 392 |
429 #if defined(ENABLE_EXTENSIONS) | 393 #if defined(ENABLE_EXTENSIONS) |
(...skipping 13 matching lines...) Expand all Loading... | |
443 // Otherwise this should be a regular tab contents. | 407 // Otherwise this should be a regular tab contents. |
444 allowed = true; | 408 allowed = true; |
445 check_permission = true; | 409 check_permission = true; |
446 #endif | 410 #endif |
447 | 411 |
448 BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, | 412 BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, |
449 base::Bind(callback, check_permission, allowed)); | 413 base::Bind(callback, check_permission, allowed)); |
450 } | 414 } |
451 | 415 |
452 } // namespace speech | 416 } // namespace speech |
OLD | NEW |