blob: cfc45b668c08487978f05fc3254befdae95fd457 [file] [log] [blame]
// 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 "device/devices_app/usb/device_manager_impl.h"
#include "base/bind.h"
#include "base/location.h"
#include "base/memory/scoped_ptr.h"
#include "base/message_loop/message_loop.h"
#include "base/scoped_observer.h"
#include "base/sequenced_task_runner.h"
#include "base/stl_util.h"
#include "base/thread_task_runner_handle.h"
#include "device/core/device_client.h"
#include "device/devices_app/usb/device_impl.h"
#include "device/devices_app/usb/public/interfaces/device.mojom.h"
#include "device/devices_app/usb/type_converters.h"
#include "device/usb/usb_device.h"
#include "device/usb/usb_device_filter.h"
#include "device/usb/usb_service.h"
#include "third_party/mojo/src/mojo/public/cpp/bindings/array.h"
#include "third_party/mojo/src/mojo/public/cpp/bindings/interface_request.h"
namespace device {
namespace usb {
namespace {
void OnGetDevicesOnServiceThread(
const std::vector<UsbDeviceFilter>& filters,
const base::Callback<void(mojo::Array<DeviceInfoPtr>)>& callback,
scoped_refptr<base::TaskRunner> callback_task_runner,
const std::vector<scoped_refptr<UsbDevice>>& devices) {
mojo::Array<DeviceInfoPtr> mojo_devices(0);
for (size_t i = 0; i < devices.size(); ++i) {
if (UsbDeviceFilter::MatchesAny(devices[i], filters) || filters.empty())
mojo_devices.push_back(DeviceInfo::From(*devices[i]));
}
callback_task_runner->PostTask(
FROM_HERE, base::Bind(callback, base::Passed(&mojo_devices)));
}
void GetDevicesOnServiceThread(
const std::vector<UsbDeviceFilter>& filters,
const base::Callback<void(mojo::Array<DeviceInfoPtr>)>& callback,
scoped_refptr<base::TaskRunner> callback_task_runner) {
DCHECK(DeviceClient::Get());
UsbService* usb_service = DeviceClient::Get()->GetUsbService();
if (!usb_service) {
mojo::Array<DeviceInfoPtr> no_devices(0);
callback_task_runner->PostTask(
FROM_HERE, base::Bind(callback, base::Passed(&no_devices)));
return;
}
usb_service->GetDevices(base::Bind(&OnGetDevicesOnServiceThread, filters,
callback, callback_task_runner));
}
void RunOpenDeviceCallback(const DeviceManager::OpenDeviceCallback& callback,
OpenDeviceError error) {
callback.Run(error);
}
void OnOpenDeviceOnServiceThread(
mojo::InterfaceRequest<Device> device_request,
const DeviceManager::OpenDeviceCallback& callback,
scoped_refptr<base::TaskRunner> callback_task_runner,
scoped_refptr<UsbDeviceHandle> device_handle) {
if (!device_handle) {
callback_task_runner->PostTask(FROM_HERE,
base::Bind(&RunOpenDeviceCallback, callback,
OPEN_DEVICE_ERROR_ACCESS_DENIED));
return;
}
// Owned by its MessagePipe.
new DeviceImpl(device_handle, device_request.Pass());
callback_task_runner->PostTask(
FROM_HERE,
base::Bind(&RunOpenDeviceCallback, callback, OPEN_DEVICE_ERROR_OK));
}
void OpenDeviceOnServiceThread(
const std::string& guid,
mojo::InterfaceRequest<Device> device_request,
const DeviceManager::OpenDeviceCallback& callback,
scoped_refptr<base::TaskRunner> callback_task_runner) {
DCHECK(DeviceClient::Get());
UsbService* usb_service = DeviceClient::Get()->GetUsbService();
if (!usb_service) {
callback_task_runner->PostTask(FROM_HERE,
base::Bind(&RunOpenDeviceCallback, callback,
OPEN_DEVICE_ERROR_NOT_FOUND));
return;
}
scoped_refptr<UsbDevice> device = usb_service->GetDevice(guid);
if (!device) {
callback_task_runner->PostTask(FROM_HERE,
base::Bind(&RunOpenDeviceCallback, callback,
OPEN_DEVICE_ERROR_NOT_FOUND));
return;
}
device->Open(base::Bind(&OnOpenDeviceOnServiceThread,
base::Passed(&device_request), callback,
callback_task_runner));
}
void FilterDeviceListAndThen(
const DeviceManagerImpl::GetDevicesCallback& callback,
mojo::Array<DeviceInfoPtr> devices,
mojo::Array<mojo::String> allowed_guids) {
std::set<std::string> allowed_guid_set;
for (size_t i = 0; i < allowed_guids.size(); ++i)
allowed_guid_set.insert(allowed_guids[i]);
mojo::Array<DeviceInfoPtr> allowed_devices(0);
for (size_t i = 0; i < devices.size(); ++i) {
if (ContainsKey(allowed_guid_set, devices[i]->guid))
allowed_devices.push_back(devices[i].Pass());
}
callback.Run(allowed_devices.Pass());
}
} // namespace
class DeviceManagerImpl::ServiceThreadHelper
: public UsbService::Observer,
public base::MessageLoop::DestructionObserver {
public:
ServiceThreadHelper(base::WeakPtr<DeviceManagerImpl> manager,
scoped_refptr<base::TaskRunner> task_runner)
: observer_(this), manager_(manager), task_runner_(task_runner) {}
~ServiceThreadHelper() override {
base::MessageLoop::current()->RemoveDestructionObserver(this);
}
static void Start(scoped_ptr<ServiceThreadHelper> self) {
UsbService* usb_service = DeviceClient::Get()->GetUsbService();
if (usb_service)
self->observer_.Add(usb_service);
// |self| now owned by the current message loop.
base::MessageLoop::current()->AddDestructionObserver(self.release());
}
private:
// UsbService::Observer
void OnDeviceAdded(scoped_refptr<UsbDevice> device) override {
DeviceInfoPtr mojo_device(DeviceInfo::From(*device));
task_runner_->PostTask(
FROM_HERE, base::Bind(&DeviceManagerImpl::OnDeviceAdded, manager_,
base::Passed(&mojo_device)));
}
void OnDeviceRemoved(scoped_refptr<UsbDevice> device) override {
task_runner_->PostTask(
FROM_HERE, base::Bind(&DeviceManagerImpl::OnDeviceRemoved, manager_,
device->guid()));
}
void WillDestroyUsbService() override { observer_.RemoveAll(); }
// base::MessageLoop::DestructionObserver
void WillDestroyCurrentMessageLoop() override { delete this; }
ScopedObserver<UsbService, UsbService::Observer> observer_;
base::WeakPtr<DeviceManagerImpl> manager_;
scoped_refptr<base::TaskRunner> task_runner_;
};
DeviceManagerImpl::DeviceManagerImpl(
mojo::InterfaceRequest<DeviceManager> request,
PermissionProviderPtr permission_provider,
scoped_refptr<base::SequencedTaskRunner> service_task_runner)
: permission_provider_(permission_provider.Pass()),
service_task_runner_(service_task_runner),
binding_(this, request.Pass()),
weak_factory_(this) {
// This object owns itself and will be destroyed if either the message pipe
// it is bound to is closed or the PermissionProvider it depends on is
// unavailable.
binding_.set_connection_error_handler([this]() { delete this; });
permission_provider_.set_connection_error_handler([this]() { delete this; });
scoped_ptr<ServiceThreadHelper> helper(new ServiceThreadHelper(
weak_factory_.GetWeakPtr(), base::ThreadTaskRunnerHandle::Get()));
helper_ = helper.get();
service_task_runner_->PostTask(
FROM_HERE,
base::Bind(&ServiceThreadHelper::Start, base::Passed(&helper)));
}
DeviceManagerImpl::~DeviceManagerImpl() {
// It is safe to call this if |helper_| was already destroyed when
// |service_task_runner_| exited as the task will never execute.
service_task_runner_->DeleteSoon(FROM_HERE, helper_);
connection_error_handler_.Run();
}
void DeviceManagerImpl::GetDevices(EnumerationOptionsPtr options,
const GetDevicesCallback& callback) {
std::vector<UsbDeviceFilter> filters;
if (options)
filters = options->filters.To<std::vector<UsbDeviceFilter>>();
auto get_devices_callback = base::Bind(&DeviceManagerImpl::OnGetDevices,
weak_factory_.GetWeakPtr(), callback);
service_task_runner_->PostTask(
FROM_HERE,
base::Bind(&GetDevicesOnServiceThread, filters, get_devices_callback,
base::ThreadTaskRunnerHandle::Get()));
}
void DeviceManagerImpl::GetDeviceChanges(
const GetDeviceChangesCallback& callback) {
device_change_callbacks_.push(callback);
MaybeRunDeviceChangesCallback();
}
void DeviceManagerImpl::OpenDevice(
const mojo::String& guid,
mojo::InterfaceRequest<Device> device_request,
const OpenDeviceCallback& callback) {
mojo::Array<mojo::String> requested_guids(1);
requested_guids[0] = guid;
permission_provider_->HasDevicePermission(
requested_guids.Pass(),
base::Bind(&DeviceManagerImpl::OnOpenDevicePermissionCheckComplete,
base::Unretained(this), base::Passed(&device_request),
callback));
}
void DeviceManagerImpl::OnOpenDevicePermissionCheckComplete(
mojo::InterfaceRequest<Device> device_request,
const OpenDeviceCallback& callback,
mojo::Array<mojo::String> allowed_guids) {
if (allowed_guids.size() == 0) {
callback.Run(OPEN_DEVICE_ERROR_ACCESS_DENIED);
return;
}
DCHECK(allowed_guids.size() == 1);
service_task_runner_->PostTask(
FROM_HERE, base::Bind(&OpenDeviceOnServiceThread, allowed_guids[0],
base::Passed(&device_request), callback,
base::ThreadTaskRunnerHandle::Get()));
}
void DeviceManagerImpl::OnGetDevices(const GetDevicesCallback& callback,
mojo::Array<DeviceInfoPtr> devices) {
mojo::Array<mojo::String> requested_guids(devices.size());
for (size_t i = 0; i < devices.size(); ++i)
requested_guids[i] = devices[i]->guid;
permission_provider_->HasDevicePermission(
requested_guids.Pass(),
base::Bind(&FilterDeviceListAndThen, callback, base::Passed(&devices)));
}
void DeviceManagerImpl::OnDeviceAdded(DeviceInfoPtr device) {
DCHECK(!ContainsKey(devices_removed_, device->guid));
devices_added_.push_back(device.Pass());
MaybeRunDeviceChangesCallback();
}
void DeviceManagerImpl::OnDeviceRemoved(std::string device_guid) {
bool found = false;
mojo::Array<DeviceInfoPtr> devices_added;
for (size_t i = 0; i < devices_added_.size(); ++i) {
if (devices_added_[i]->guid == device_guid)
found = true;
else
devices_added.push_back(devices_added_[i].Pass());
}
devices_added.Swap(&devices_added_);
if (!found)
devices_removed_.insert(device_guid);
MaybeRunDeviceChangesCallback();
}
void DeviceManagerImpl::MaybeRunDeviceChangesCallback() {
if (!permission_request_pending_ && !device_change_callbacks_.empty()) {
mojo::Array<DeviceInfoPtr> devices_added;
devices_added.Swap(&devices_added_);
std::set<std::string> devices_removed;
devices_removed.swap(devices_removed_);
mojo::Array<mojo::String> requested_guids(devices_added.size() +
devices_removed.size());
{
size_t i;
for (i = 0; i < devices_added.size(); ++i)
requested_guids[i] = devices_added[i]->guid;
for (const std::string& guid : devices_removed)
requested_guids[i++] = guid;
}
permission_request_pending_ = true;
permission_provider_->HasDevicePermission(
requested_guids.Pass(),
base::Bind(&DeviceManagerImpl::OnEnumerationPermissionCheckComplete,
base::Unretained(this), base::Passed(&devices_added),
devices_removed));
}
}
void DeviceManagerImpl::OnEnumerationPermissionCheckComplete(
mojo::Array<DeviceInfoPtr> devices_added,
const std::set<std::string>& devices_removed,
mojo::Array<mojo::String> allowed_guids) {
permission_request_pending_ = false;
if (allowed_guids.size() > 0) {
std::set<std::string> allowed_guid_set;
for (size_t i = 0; i < allowed_guids.size(); ++i)
allowed_guid_set.insert(allowed_guids[i]);
DeviceChangeNotificationPtr notification = DeviceChangeNotification::New();
notification->devices_added.resize(0);
for (size_t i = 0; i < devices_added.size(); ++i) {
if (ContainsKey(allowed_guid_set, devices_added[i]->guid))
notification->devices_added.push_back(devices_added[i].Pass());
}
notification->devices_removed.resize(0);
for (const std::string& guid : devices_removed) {
if (ContainsKey(allowed_guid_set, guid))
notification->devices_removed.push_back(guid);
}
DCHECK(!device_change_callbacks_.empty());
const GetDeviceChangesCallback& callback = device_change_callbacks_.front();
callback.Run(notification.Pass());
device_change_callbacks_.pop();
}
if (devices_added_.size() > 0 || !devices_removed_.empty())
MaybeRunDeviceChangesCallback();
}
} // namespace usb
} // namespace device