|  | // Copyright 2018 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 "content/browser/serial/serial_service.h" | 
|  |  | 
|  | #include <utility> | 
|  |  | 
|  | #include "base/bind.h" | 
|  | #include "base/callback.h" | 
|  | #include "content/browser/renderer_host/back_forward_cache_disable.h" | 
|  | #include "content/browser/web_contents/web_contents_impl.h" | 
|  | #include "content/public/browser/back_forward_cache.h" | 
|  | #include "content/public/browser/content_browser_client.h" | 
|  | #include "content/public/browser/render_frame_host.h" | 
|  | #include "content/public/browser/serial_chooser.h" | 
|  | #include "content/public/browser/serial_delegate.h" | 
|  | #include "content/public/browser/web_contents.h" | 
|  | #include "content/public/common/content_client.h" | 
|  | #include "mojo/public/cpp/bindings/pending_remote.h" | 
|  | #include "third_party/blink/public/mojom/permissions_policy/permissions_policy.mojom.h" | 
|  |  | 
|  | namespace content { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | blink::mojom::SerialPortInfoPtr ToBlinkType( | 
|  | const device::mojom::SerialPortInfo& port) { | 
|  | auto info = blink::mojom::SerialPortInfo::New(); | 
|  | info->token = port.token; | 
|  | info->has_usb_vendor_id = port.has_vendor_id; | 
|  | if (port.has_vendor_id) | 
|  | info->usb_vendor_id = port.vendor_id; | 
|  | info->has_usb_product_id = port.has_product_id; | 
|  | if (port.has_product_id) | 
|  | info->usb_product_id = port.product_id; | 
|  | return info; | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | SerialService::SerialService(RenderFrameHost* render_frame_host) | 
|  | : render_frame_host_(render_frame_host) { | 
|  | DCHECK(render_frame_host_->IsFeatureEnabled( | 
|  | blink::mojom::PermissionsPolicyFeature::kSerial)); | 
|  | // Serial API is not supported for back-forward cache for now because we | 
|  | // don't have support for closing/freezing ports when the frame is added to | 
|  | // the back-forward cache, so we mark frames that use this API as disabled | 
|  | // for back-forward cache. | 
|  | BackForwardCache::DisableForRenderFrameHost( | 
|  | render_frame_host, | 
|  | BackForwardCacheDisable::DisabledReason( | 
|  | BackForwardCacheDisable::DisabledReasonId::kSerial)); | 
|  |  | 
|  | watchers_.set_disconnect_handler(base::BindRepeating( | 
|  | &SerialService::OnWatcherConnectionError, base::Unretained(this))); | 
|  |  | 
|  | SerialDelegate* delegate = GetContentClient()->browser()->GetSerialDelegate(); | 
|  | if (delegate) | 
|  | delegate->AddObserver(render_frame_host_, this); | 
|  | } | 
|  |  | 
|  | SerialService::~SerialService() { | 
|  | SerialDelegate* delegate = GetContentClient()->browser()->GetSerialDelegate(); | 
|  | if (delegate) | 
|  | delegate->RemoveObserver(render_frame_host_, this); | 
|  |  | 
|  | // The remaining watchers will be closed from this end. | 
|  | if (!watchers_.empty()) | 
|  | DecrementActiveFrameCount(); | 
|  | } | 
|  |  | 
|  | void SerialService::Bind( | 
|  | mojo::PendingReceiver<blink::mojom::SerialService> receiver) { | 
|  | receivers_.Add(this, std::move(receiver)); | 
|  | } | 
|  |  | 
|  | void SerialService::SetClient( | 
|  | mojo::PendingRemote<blink::mojom::SerialServiceClient> client) { | 
|  | clients_.Add(std::move(client)); | 
|  | } | 
|  |  | 
|  | void SerialService::GetPorts(GetPortsCallback callback) { | 
|  | SerialDelegate* delegate = GetContentClient()->browser()->GetSerialDelegate(); | 
|  | if (!delegate) { | 
|  | std::move(callback).Run(std::vector<blink::mojom::SerialPortInfoPtr>()); | 
|  | return; | 
|  | } | 
|  |  | 
|  | delegate->GetPortManager(render_frame_host_) | 
|  | ->GetDevices(base::BindOnce(&SerialService::FinishGetPorts, | 
|  | weak_factory_.GetWeakPtr(), | 
|  | std::move(callback))); | 
|  | } | 
|  |  | 
|  | void SerialService::RequestPort( | 
|  | std::vector<blink::mojom::SerialPortFilterPtr> filters, | 
|  | RequestPortCallback callback) { | 
|  | SerialDelegate* delegate = GetContentClient()->browser()->GetSerialDelegate(); | 
|  | if (!delegate) { | 
|  | std::move(callback).Run(nullptr); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (!delegate->CanRequestPortPermission(render_frame_host_)) { | 
|  | std::move(callback).Run(nullptr); | 
|  | return; | 
|  | } | 
|  |  | 
|  | chooser_ = delegate->RunChooser( | 
|  | render_frame_host_, std::move(filters), | 
|  | base::BindOnce(&SerialService::FinishRequestPort, | 
|  | weak_factory_.GetWeakPtr(), std::move(callback))); | 
|  | } | 
|  |  | 
|  | void SerialService::OpenPort( | 
|  | const base::UnguessableToken& token, | 
|  | device::mojom::SerialConnectionOptionsPtr options, | 
|  | mojo::PendingRemote<device::mojom::SerialPortClient> client, | 
|  | OpenPortCallback callback) { | 
|  | SerialDelegate* delegate = GetContentClient()->browser()->GetSerialDelegate(); | 
|  | if (!delegate) { | 
|  | std::move(callback).Run(mojo::NullRemote()); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (watchers_.empty()) { | 
|  | auto* web_contents_impl = static_cast<WebContentsImpl*>( | 
|  | WebContents::FromRenderFrameHost(render_frame_host_)); | 
|  | web_contents_impl->IncrementSerialActiveFrameCount(); | 
|  | } | 
|  |  | 
|  | mojo::PendingRemote<device::mojom::SerialPortConnectionWatcher> watcher; | 
|  | watchers_.Add(this, watcher.InitWithNewPipeAndPassReceiver()); | 
|  | delegate->GetPortManager(render_frame_host_) | 
|  | ->OpenPort(token, /*use_alternate_path=*/false, std::move(options), | 
|  | std::move(client), std::move(watcher), std::move(callback)); | 
|  | } | 
|  |  | 
|  | void SerialService::OnPortAdded(const device::mojom::SerialPortInfo& port) { | 
|  | SerialDelegate* delegate = GetContentClient()->browser()->GetSerialDelegate(); | 
|  | if (!delegate || !delegate->HasPortPermission(render_frame_host_, port)) | 
|  | return; | 
|  |  | 
|  | for (const auto& client : clients_) | 
|  | client->OnPortAdded(ToBlinkType(port)); | 
|  | } | 
|  |  | 
|  | void SerialService::OnPortRemoved(const device::mojom::SerialPortInfo& port) { | 
|  | SerialDelegate* delegate = GetContentClient()->browser()->GetSerialDelegate(); | 
|  | if (!delegate || !delegate->HasPortPermission(render_frame_host_, port)) | 
|  | return; | 
|  |  | 
|  | for (const auto& client : clients_) | 
|  | client->OnPortRemoved(ToBlinkType(port)); | 
|  | } | 
|  |  | 
|  | void SerialService::OnPortManagerConnectionError() { | 
|  | // Reflect the loss of the SerialPortManager connection into the renderer | 
|  | // in order to force caches to be cleared and connections to be | 
|  | // re-established. | 
|  | receivers_.Clear(); | 
|  | clients_.Clear(); | 
|  | chooser_.reset(); | 
|  |  | 
|  | if (!watchers_.empty()) { | 
|  | watchers_.Clear(); | 
|  | DecrementActiveFrameCount(); | 
|  | } | 
|  | } | 
|  |  | 
|  | void SerialService::FinishGetPorts( | 
|  | GetPortsCallback callback, | 
|  | std::vector<device::mojom::SerialPortInfoPtr> ports) { | 
|  | std::vector<blink::mojom::SerialPortInfoPtr> result; | 
|  | SerialDelegate* delegate = GetContentClient()->browser()->GetSerialDelegate(); | 
|  | if (!delegate) { | 
|  | std::move(callback).Run(std::move(result)); | 
|  | return; | 
|  | } | 
|  |  | 
|  | for (const auto& port : ports) { | 
|  | if (delegate->HasPortPermission(render_frame_host_, *port)) | 
|  | result.push_back(ToBlinkType(*port)); | 
|  | } | 
|  |  | 
|  | std::move(callback).Run(std::move(result)); | 
|  | } | 
|  |  | 
|  | void SerialService::FinishRequestPort(RequestPortCallback callback, | 
|  | device::mojom::SerialPortInfoPtr port) { | 
|  | SerialDelegate* delegate = GetContentClient()->browser()->GetSerialDelegate(); | 
|  | if (!delegate || !port) { | 
|  | std::move(callback).Run(nullptr); | 
|  | return; | 
|  | } | 
|  |  | 
|  | std::move(callback).Run(ToBlinkType(*port)); | 
|  | } | 
|  |  | 
|  | void SerialService::OnWatcherConnectionError() { | 
|  | if (watchers_.empty()) | 
|  | DecrementActiveFrameCount(); | 
|  | } | 
|  |  | 
|  | void SerialService::DecrementActiveFrameCount() { | 
|  | auto* web_contents_impl = static_cast<WebContentsImpl*>( | 
|  | WebContents::FromRenderFrameHost(render_frame_host_)); | 
|  | web_contents_impl->DecrementSerialActiveFrameCount(); | 
|  | } | 
|  |  | 
|  | RENDER_DOCUMENT_HOST_USER_DATA_KEY_IMPL(SerialService) | 
|  |  | 
|  | }  // namespace content |