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

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

Powered by Google App Engine
This is Rietveld 408576698