| // 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 "extensions/browser/api/serial/serial_port_manager.h" |
| |
| #include <utility> |
| |
| #include "base/bind.h" |
| #include "base/lazy_instance.h" |
| #include "base/task/post_task.h" |
| #include "content/public/browser/browser_task_traits.h" |
| #include "content/public/browser/browser_thread.h" |
| #include "content/public/browser/system_connector.h" |
| #include "extensions/browser/api/serial/serial_connection.h" |
| #include "extensions/browser/event_router.h" |
| #include "extensions/browser/extensions_browser_client.h" |
| #include "services/device/public/mojom/constants.mojom.h" |
| #include "services/service_manager/public/cpp/connector.h" |
| |
| namespace extensions { |
| |
| namespace api { |
| |
| namespace { |
| |
| bool ShouldPauseOnReceiveError(serial::ReceiveError error) { |
| return error == serial::RECEIVE_ERROR_DEVICE_LOST || |
| error == serial::RECEIVE_ERROR_SYSTEM_ERROR || |
| error == serial::RECEIVE_ERROR_DISCONNECTED || |
| error == serial::RECEIVE_ERROR_BREAK || |
| error == serial::RECEIVE_ERROR_FRAME_ERROR || |
| error == serial::RECEIVE_ERROR_OVERRUN || |
| error == serial::RECEIVE_ERROR_BUFFER_OVERFLOW || |
| error == serial::RECEIVE_ERROR_PARITY_ERROR; |
| } |
| |
| } // namespace |
| |
| static base::LazyInstance<BrowserContextKeyedAPIFactory<SerialPortManager>>:: |
| DestructorAtExit g_factory = LAZY_INSTANCE_INITIALIZER; |
| |
| // static |
| BrowserContextKeyedAPIFactory<SerialPortManager>* |
| SerialPortManager::GetFactoryInstance() { |
| return g_factory.Pointer(); |
| } |
| |
| // static |
| SerialPortManager* SerialPortManager::Get(content::BrowserContext* context) { |
| return BrowserContextKeyedAPIFactory<SerialPortManager>::Get(context); |
| } |
| |
| SerialPortManager::SerialPortManager(content::BrowserContext* context) |
| : thread_id_(SerialConnection::kThreadId), context_(context) { |
| ApiResourceManager<SerialConnection>* manager = |
| ApiResourceManager<SerialConnection>::Get(context_); |
| DCHECK(manager) << "No serial connection manager."; |
| connections_ = manager->data_; |
| } |
| |
| SerialPortManager::~SerialPortManager() {} |
| |
| SerialPortManager::ReceiveParams::ReceiveParams() {} |
| |
| SerialPortManager::ReceiveParams::ReceiveParams(const ReceiveParams& other) = |
| default; |
| |
| SerialPortManager::ReceiveParams::~ReceiveParams() {} |
| |
| void SerialPortManager::GetDevices( |
| device::mojom::SerialPortManager::GetDevicesCallback callback) { |
| EnsureConnection(); |
| port_manager_->GetDevices(std::move(callback)); |
| } |
| |
| void SerialPortManager::GetPort(const std::string& path, |
| device::mojom::SerialPortRequest request) { |
| EnsureConnection(); |
| port_manager_->GetDevices( |
| base::BindOnce(&SerialPortManager::OnGotDevicesToGetPort, |
| weak_factory_.GetWeakPtr(), path, std::move(request))); |
| } |
| |
| void SerialPortManager::StartConnectionPolling(const std::string& extension_id, |
| int connection_id) { |
| DCHECK_CURRENTLY_ON(thread_id_); |
| auto* connection = connections_->Get(extension_id, connection_id); |
| if (!connection) |
| return; |
| |
| DCHECK_EQ(extension_id, connection->owner_extension_id()); |
| |
| ReceiveParams params; |
| params.thread_id = thread_id_; |
| params.browser_context_id = context_; |
| params.extension_id = extension_id; |
| params.connections = connections_; |
| params.connection_id = connection_id; |
| |
| connection->StartPolling(base::BindRepeating(&DispatchReceiveEvent, params)); |
| } |
| |
| // static |
| void SerialPortManager::DispatchReceiveEvent(const ReceiveParams& params, |
| std::vector<uint8_t> data, |
| serial::ReceiveError error) { |
| DCHECK_CURRENTLY_ON(params.thread_id); |
| |
| // Note that an error (e.g. timeout) does not necessarily mean that no data |
| // was read, so we may fire an onReceive regardless of any error code. |
| if (data.size() > 0) { |
| serial::ReceiveInfo receive_info; |
| receive_info.connection_id = params.connection_id; |
| receive_info.data = std::move(data); |
| std::unique_ptr<base::ListValue> args = |
| serial::OnReceive::Create(receive_info); |
| std::unique_ptr<extensions::Event> event( |
| new extensions::Event(extensions::events::SERIAL_ON_RECEIVE, |
| serial::OnReceive::kEventName, std::move(args))); |
| PostEvent(params, std::move(event)); |
| } |
| |
| if (error != serial::RECEIVE_ERROR_NONE) { |
| serial::ReceiveErrorInfo error_info; |
| error_info.connection_id = params.connection_id; |
| error_info.error = error; |
| std::unique_ptr<base::ListValue> args = |
| serial::OnReceiveError::Create(error_info); |
| std::unique_ptr<extensions::Event> event(new extensions::Event( |
| extensions::events::SERIAL_ON_RECEIVE_ERROR, |
| serial::OnReceiveError::kEventName, std::move(args))); |
| PostEvent(params, std::move(event)); |
| if (ShouldPauseOnReceiveError(error)) { |
| SerialConnection* connection = |
| params.connections->Get(params.extension_id, params.connection_id); |
| if (connection) |
| connection->SetPaused(true); |
| } |
| } |
| } |
| |
| // static |
| void SerialPortManager::PostEvent(const ReceiveParams& params, |
| std::unique_ptr<extensions::Event> event) { |
| DCHECK_CURRENTLY_ON(params.thread_id); |
| |
| base::PostTaskWithTraits( |
| FROM_HERE, {BrowserThread::UI}, |
| base::BindOnce(&DispatchEvent, params.browser_context_id, |
| params.extension_id, std::move(event))); |
| } |
| |
| // static |
| void SerialPortManager::DispatchEvent( |
| void* browser_context_id, |
| const std::string& extension_id, |
| std::unique_ptr<extensions::Event> event) { |
| DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| |
| content::BrowserContext* context = |
| reinterpret_cast<content::BrowserContext*>(browser_context_id); |
| if (!extensions::ExtensionsBrowserClient::Get()->IsValidContext(context)) |
| return; |
| |
| EventRouter* router = EventRouter::Get(context); |
| if (router) |
| router->DispatchEventToExtension(extension_id, std::move(event)); |
| } |
| |
| void SerialPortManager::EnsureConnection() { |
| DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); |
| if (port_manager_) |
| return; |
| |
| DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| DCHECK(content::GetSystemConnector()); |
| content::GetSystemConnector()->BindInterface( |
| device::mojom::kServiceName, mojo::MakeRequest(&port_manager_)); |
| port_manager_.set_connection_error_handler( |
| base::BindOnce(&SerialPortManager::OnPortManagerConnectionError, |
| weak_factory_.GetWeakPtr())); |
| } |
| |
| void SerialPortManager::OnGotDevicesToGetPort( |
| const std::string& path, |
| device::mojom::SerialPortRequest request, |
| std::vector<device::mojom::SerialPortInfoPtr> devices) { |
| DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); |
| |
| for (auto& device : devices) { |
| if (device->path.AsUTF8Unsafe() == path) { |
| port_manager_->GetPort(device->token, std::move(request), |
| /*watcher=*/nullptr); |
| return; |
| } |
| } |
| } |
| |
| void SerialPortManager::OnPortManagerConnectionError() { |
| DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); |
| port_manager_.reset(); |
| } |
| |
| } // namespace api |
| |
| } // namespace extensions |