Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(88)

Side by Side Diff: chrome/browser/speech/chrome_speech_recognition_manager_delegate.cc

Issue 636863003: Make SpeechRecognition per RenderFrame instead of per RenderView. Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: cleanups Created 6 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
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
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
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
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
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698