| // Copyright 2015 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "chrome/browser/usb/web_usb_permission_provider.h" |
| |
| #include <stddef.h> |
| #include <utility> |
| |
| #include "base/stl_util.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/browser/usb/usb_blocklist.h" |
| #include "chrome/browser/usb/usb_chooser_context.h" |
| #include "chrome/browser/usb/usb_chooser_context_factory.h" |
| #include "chrome/browser/usb/usb_tab_helper.h" |
| #include "content/public/browser/browser_thread.h" |
| #include "content/public/browser/render_frame_host.h" |
| #include "content/public/browser/web_contents.h" |
| #include "device/usb/usb_device.h" |
| #include "device/usb/webusb_descriptors.h" |
| |
| using content::RenderFrameHost; |
| using content::WebContents; |
| |
| namespace { |
| |
| bool FindOriginInDescriptorSet(const device::WebUsbAllowedOrigins* set, |
| const GURL& origin, |
| const uint8_t* configuration_value, |
| const uint8_t* first_interface) { |
| if (!set) |
| return false; |
| if (base::ContainsValue(set->origins, origin)) |
| return true; |
| for (const auto& configuration : set->configurations) { |
| if (configuration_value && |
| *configuration_value != configuration.configuration_value) |
| continue; |
| if (base::ContainsValue(configuration.origins, origin)) |
| return true; |
| for (const auto& function : configuration.functions) { |
| if (first_interface && *first_interface != function.first_interface) |
| continue; |
| if (base::ContainsValue(function.origins, origin)) |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| } // namespace |
| |
| // static |
| bool WebUSBPermissionProvider::HasDevicePermission( |
| RenderFrameHost* render_frame_host, |
| scoped_refptr<const device::UsbDevice> device) { |
| DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| |
| if (UsbBlocklist::Get().IsExcluded(device)) |
| return false; |
| |
| WebContents* web_contents = |
| WebContents::FromRenderFrameHost(render_frame_host); |
| RenderFrameHost* main_frame = web_contents->GetMainFrame(); |
| GURL embedding_origin = main_frame->GetLastCommittedURL().GetOrigin(); |
| GURL requesting_origin = render_frame_host->GetLastCommittedURL().GetOrigin(); |
| Profile* profile = |
| Profile::FromBrowserContext(web_contents->GetBrowserContext()); |
| UsbChooserContext* chooser_context = |
| UsbChooserContextFactory::GetForProfile(profile); |
| |
| if (!chooser_context->HasDevicePermission(requesting_origin, embedding_origin, |
| device)) { |
| return false; |
| } |
| |
| // On Android it is not possible to read the WebUSB descriptors until Chrome |
| // has been granted permission to open it. Instead we grant provisional access |
| // to the device and perform the allowed origins check when the client tries |
| // to open it. |
| if (!device->permission_granted()) |
| return true; |
| |
| // Embedded frames must have their origin in the list provided by the device. |
| if (render_frame_host != main_frame) { |
| return FindOriginInDescriptorSet(device->webusb_allowed_origins(), |
| requesting_origin, nullptr, nullptr); |
| } |
| |
| return true; |
| } |
| |
| WebUSBPermissionProvider::WebUSBPermissionProvider( |
| RenderFrameHost* render_frame_host) |
| : render_frame_host_(render_frame_host), weak_factory_(this) { |
| DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| DCHECK(render_frame_host_); |
| } |
| |
| WebUSBPermissionProvider::~WebUSBPermissionProvider() {} |
| |
| base::WeakPtr<device::usb::PermissionProvider> |
| WebUSBPermissionProvider::GetWeakPtr() { |
| return weak_factory_.GetWeakPtr(); |
| } |
| |
| bool WebUSBPermissionProvider::HasDevicePermission( |
| scoped_refptr<const device::UsbDevice> device) const { |
| return HasDevicePermission(render_frame_host_, device); |
| } |
| |
| bool WebUSBPermissionProvider::HasConfigurationPermission( |
| uint8_t requested_configuration_value, |
| scoped_refptr<const device::UsbDevice> device) const { |
| DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| |
| // Embedded frames may only access configurations if their origin in the list |
| // provided by the device. |
| RenderFrameHost* main_frame = |
| WebContents::FromRenderFrameHost(render_frame_host_)->GetMainFrame(); |
| if (render_frame_host_ != main_frame) { |
| return FindOriginInDescriptorSet( |
| device->webusb_allowed_origins(), |
| render_frame_host_->GetLastCommittedURL().GetOrigin(), |
| &requested_configuration_value, nullptr); |
| } |
| |
| return true; |
| } |
| |
| bool WebUSBPermissionProvider::HasFunctionPermission( |
| uint8_t requested_function, |
| uint8_t configuration_value, |
| scoped_refptr<const device::UsbDevice> device) const { |
| DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| |
| // Embedded frames may only access configurations if their origin in the list |
| // provided by the device. |
| RenderFrameHost* main_frame = |
| WebContents::FromRenderFrameHost(render_frame_host_)->GetMainFrame(); |
| if (render_frame_host_ != main_frame) { |
| return FindOriginInDescriptorSet( |
| device->webusb_allowed_origins(), |
| render_frame_host_->GetLastCommittedURL().GetOrigin(), |
| &configuration_value, &requested_function); |
| } |
| |
| return true; |
| } |
| |
| void WebUSBPermissionProvider::IncrementConnectionCount() { |
| DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| WebContents* web_contents = |
| WebContents::FromRenderFrameHost(render_frame_host_); |
| UsbTabHelper* tab_helper = UsbTabHelper::FromWebContents(web_contents); |
| tab_helper->IncrementConnectionCount(render_frame_host_); |
| } |
| |
| void WebUSBPermissionProvider::DecrementConnectionCount() { |
| DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| WebContents* web_contents = |
| WebContents::FromRenderFrameHost(render_frame_host_); |
| UsbTabHelper* tab_helper = UsbTabHelper::FromWebContents(web_contents); |
| tab_helper->DecrementConnectionCount(render_frame_host_); |
| } |