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

Unified Diff: chrome/browser/usb/usb_device_handle.cc

Issue 16316004: Separate usb device handle from usb device. (deprecate) (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Fix the threading mess Created 7 years, 6 months 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 side-by-side diff with in-line comments
Download patch
« no previous file with comments | « chrome/browser/usb/usb_device_handle.h ('k') | chrome/browser/usb/usb_interface.h » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: chrome/browser/usb/usb_device_handle.cc
diff --git a/chrome/browser/usb/usb_device_handle.cc b/chrome/browser/usb/usb_device_handle.cc
new file mode 100644
index 0000000000000000000000000000000000000000..9cabbfef55cdf11e5d7df3d1516973089e2b972c
--- /dev/null
+++ b/chrome/browser/usb/usb_device_handle.cc
@@ -0,0 +1,439 @@
+// Copyright (c) 2012 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/usb_device_handle.h"
+
+#include <vector>
+
+#include "base/stl_util.h"
+#include "base/synchronization/lock.h"
+#include "chrome/browser/usb/usb_interface.h"
+#include "chrome/browser/usb/usb_service.h"
+#include "content/public/browser/browser_thread.h"
+
+using content::BrowserThread;
+
+#define CHECK_DEVICE(callback, ...) \
+ do { \
+ if (handle_ == NULL) { \
+ DVLOG(1) << "Device is disconnected: "; \
+ callback.Run(__VA_ARGS__); \
+ return; \
+ } \
+ } while (0)
+
+#define CHECK_DEVICE_OR_RETURN \
+ do { \
+ if (handle_ == NULL) { \
+ DVLOG(1) << "Device is disconnected: "; \
+ return; \
+ } \
+ } while (0)
+
+namespace {
+
+uint8 ConvertTransferDirection(const UsbEndpointDirection direction) {
+ switch (direction) {
+ case USB_DIRECTION_INBOUND:
+ return LIBUSB_ENDPOINT_IN;
+ case USB_DIRECTION_OUTBOUND:
+ return LIBUSB_ENDPOINT_OUT;
+ default:
+ NOTREACHED();
+ return LIBUSB_ENDPOINT_IN;
+ }
+}
+
+uint8 CreateRequestType(
+ const UsbEndpointDirection direction,
+ const UsbDeviceHandle::TransferRequestType request_type,
+ const UsbDeviceHandle::TransferRecipient recipient) {
+ uint8 result = ConvertTransferDirection(direction);
+
+ switch (request_type) {
+ case UsbDeviceHandle::STANDARD:
+ result |= LIBUSB_REQUEST_TYPE_STANDARD;
+ break;
+ case UsbDeviceHandle::CLASS:
+ result |= LIBUSB_REQUEST_TYPE_CLASS;
+ break;
+ case UsbDeviceHandle::VENDOR:
+ result |= LIBUSB_REQUEST_TYPE_VENDOR;
+ break;
+ case UsbDeviceHandle::RESERVED:
+ result |= LIBUSB_REQUEST_TYPE_RESERVED;
+ break;
+ }
+
+ switch (recipient) {
+ case UsbDeviceHandle::DEVICE:
+ result |= LIBUSB_RECIPIENT_DEVICE;
+ break;
+ case UsbDeviceHandle::INTERFACE:
+ result |= LIBUSB_RECIPIENT_INTERFACE;
+ break;
+ case UsbDeviceHandle::ENDPOINT:
+ result |= LIBUSB_RECIPIENT_ENDPOINT;
+ break;
+ case UsbDeviceHandle::OTHER:
+ result |= LIBUSB_RECIPIENT_OTHER;
+ break;
+ }
+
+ return result;
+}
+
+UsbTransferStatus ConvertTransferStatus(const libusb_transfer_status status) {
+ switch (status) {
+ case LIBUSB_TRANSFER_COMPLETED:
+ return USB_TRANSFER_COMPLETED;
+ case LIBUSB_TRANSFER_ERROR:
+ return USB_TRANSFER_ERROR;
+ case LIBUSB_TRANSFER_TIMED_OUT:
+ return USB_TRANSFER_TIMEOUT;
+ case LIBUSB_TRANSFER_STALL:
+ return USB_TRANSFER_STALLED;
+ case LIBUSB_TRANSFER_NO_DEVICE:
+ return USB_TRANSFER_DISCONNECT;
+ case LIBUSB_TRANSFER_OVERFLOW:
+ return USB_TRANSFER_OVERFLOW;
+ case LIBUSB_TRANSFER_CANCELLED:
+ return USB_TRANSFER_CANCELLED;
+ default:
+ NOTREACHED();
+ return USB_TRANSFER_ERROR;
+ }
+}
+
+} // namespace
+
+UsbDeviceHandle::Transfer::Transfer()
+ : transfer_type(USB_TRANSFER_CONTROL), length(0) {}
+
+UsbDeviceHandle::Transfer::~Transfer() {}
+
+UsbDeviceHandle::UsbDeviceHandle(UsbService* service, const int device,
+ const uint16 vendor_id,
+ const uint16 product_id,
+ PlatformUsbDeviceHandle handle)
+ : service_(service),
+ device_(device),
+ vendor_id_(vendor_id),
+ product_id_(product_id),
+ handle_(handle) {
+ DCHECK(handle) << "Cannot create device with NULL handle.";
+}
+
+UsbDeviceHandle::UsbDeviceHandle()
+ : service_(NULL), device_(0), vendor_id_(0), product_id_(0), handle_(NULL) {
+ DCHECK(thread_checker_.CalledOnValidThread());
+}
+
+UsbDeviceHandle::~UsbDeviceHandle() {
+ DCHECK(thread_checker_.CalledOnValidThread());
+ InternalClose();
+}
+
+void UsbDeviceHandle::Close(const base::Callback<void()>& callback) {
+ DCHECK(thread_checker_.CalledOnValidThread());
+ if (handle_ == NULL) return;
+ service_->CloseDeviceHandle(this);
+ callback.Run();
+}
+
+void UsbDeviceHandle::HandleTransferCompletionFileThread(
+ PlatformUsbTransferHandle transfer) {
+ UsbDeviceHandle* const device =
+ reinterpret_cast<UsbDeviceHandle*>(transfer->user_data);
+ if (device) device->TransferComplete(transfer);
+ // We should free the transfer even if the device is removed.
+ libusb_free_transfer(transfer);
+}
+
+// This function dispatches a completed transfer to its handle.
+// It is called from UsbEventDispatcher using libusb_handle_events_timeout.
+void LIBUSB_CALL UsbDeviceHandle::HandleTransferCompletion(
+ PlatformUsbTransferHandle transfer) {
+ BrowserThread::PostTask(
+ BrowserThread::FILE, FROM_HERE,
+ base::Bind(&HandleTransferCompletionFileThread, transfer));
+}
+
+void UsbDeviceHandle::TransferComplete(PlatformUsbTransferHandle handle) {
+ DCHECK(thread_checker_.CalledOnValidThread());
+ DCHECK(handle_ != NULL) << "handle_ can only be reset after "
+ "transfers are unregistered";
+
+ Transfer transfer = transfers_[handle];
+ transfers_.erase(handle);
+
+ if (handle->status != LIBUSB_TRANSFER_COMPLETED &&
+ handle->status != LIBUSB_TRANSFER_CANCELLED) {
+ service_->ScheduleEnumerateDevice();
+ }
+
+ DCHECK_GE(handle->actual_length, 0) << "Negative actual length received";
+ size_t actual_length =
+ static_cast<size_t>(std::max(handle->actual_length, 0));
+
+ DCHECK(transfer.length >= actual_length)
+ << "data too big for our buffer (libusb failure?)";
+
+ scoped_refptr<net::IOBuffer> buffer = transfer.buffer;
+ switch (transfer.transfer_type) {
+ case USB_TRANSFER_CONTROL:
+ // If the transfer is a control transfer we do not expose the control
+ // setup header to the caller. This logic strips off the header if
+ // present before invoking the callback provided with the transfer.
+ if (actual_length > 0) {
+ CHECK(transfer.length >= LIBUSB_CONTROL_SETUP_SIZE)
+ << "buffer was not correctly set: too small for the control header";
+
+ if (transfer.length >= actual_length &&
+ actual_length >= LIBUSB_CONTROL_SETUP_SIZE) {
+ // If the payload is zero bytes long, pad out the allocated buffer
+ // size to one byte so that an IOBuffer of that size can be allocated.
+ scoped_refptr<net::IOBuffer> resized_buffer = new net::IOBuffer(
+ std::max(actual_length, static_cast<size_t>(1)));
+ memcpy(resized_buffer->data(),
+ buffer->data() + LIBUSB_CONTROL_SETUP_SIZE, actual_length);
+ buffer = resized_buffer;
+ }
+ }
+ break;
+
+ case USB_TRANSFER_ISOCHRONOUS:
+ // Isochronous replies might carry data in the different isoc packets even
+ // if the transfer actual_data value is zero. Furthermore, not all of the
+ // received packets might contain data, so we need to calculate how many
+ // data bytes we are effectively providing and pack the results.
+ if (actual_length == 0) {
+ size_t packet_buffer_start = 0;
+ for (int i = 0; i < handle->num_iso_packets; ++i) {
+ PlatformUsbIsoPacketDescriptor packet = &handle->iso_packet_desc[i];
+ if (packet->actual_length > 0) {
+ // We don't need to copy as long as all packets until now provide
+ // all the data the packet can hold.
+ if (actual_length < packet_buffer_start) {
+ CHECK(packet_buffer_start + packet->actual_length <=
+ transfer.length);
+ memmove(buffer->data() + actual_length,
+ buffer->data() + packet_buffer_start,
+ packet->actual_length);
+ }
+ actual_length += packet->actual_length;
+ }
+
+ packet_buffer_start += packet->length;
+ }
+ }
+ break;
+
+ case USB_TRANSFER_BULK:
+ case USB_TRANSFER_INTERRUPT:
+ break;
+
+ default:
+ NOTREACHED() << "Invalid usb transfer type";
+ break;
+ }
+
+ transfer.callback
+ .Run(ConvertTransferStatus(handle->status), buffer, actual_length);
+}
+
+void UsbDeviceHandle::ListInterfaces(UsbConfigDescriptor* config,
+ const UsbInterfaceCallback& callback) {
+ DCHECK(thread_checker_.CalledOnValidThread());
+ CHECK_DEVICE(callback, false);
+
+ PlatformUsbDevice device = libusb_get_device(handle_);
+
+ PlatformUsbConfigDescriptor platform_config;
+ const int list_result =
+ libusb_get_active_config_descriptor(device, &platform_config);
+ if (list_result == 0) {
+ config->Reset(platform_config);
+ }
+ callback.Run(list_result == 0);
+}
+
+void UsbDeviceHandle::ClaimInterface(const int interface_number,
+ const UsbInterfaceCallback& callback) {
+ DCHECK(thread_checker_.CalledOnValidThread());
+ CHECK_DEVICE(callback, false);
+
+ const int claim_result = libusb_claim_interface(handle_, interface_number);
+ callback.Run(claim_result == 0);
+}
+
+void UsbDeviceHandle::ReleaseInterface(const int interface_number,
+ const UsbInterfaceCallback& callback) {
+ DCHECK(thread_checker_.CalledOnValidThread());
+ CHECK_DEVICE(callback, false);
+
+ const int release_result =
+ libusb_release_interface(handle_, interface_number);
+ callback.Run(release_result == 0);
+}
+
+void UsbDeviceHandle::SetInterfaceAlternateSetting(
+ const int interface_number, const int alternate_setting,
+ const UsbInterfaceCallback& callback) {
+ DCHECK(thread_checker_.CalledOnValidThread());
+ CHECK_DEVICE(callback, false);
+
+ const int setting_result = libusb_set_interface_alt_setting(
+ handle_, interface_number, alternate_setting);
+
+ callback.Run(setting_result == 0);
+}
+
+void UsbDeviceHandle::ControlTransfer(const UsbEndpointDirection direction,
+ const TransferRequestType request_type,
+ const TransferRecipient recipient,
+ const uint8 request, const uint16 value,
+ const uint16 index, net::IOBuffer* buffer,
+ const size_t length,
+ const unsigned int timeout,
+ const UsbTransferCallback& callback) {
+ CHECK_DEVICE(callback, USB_TRANSFER_DISCONNECT,
+ scoped_refptr<net::IOBuffer>(), 0);
+
+ const size_t resized_length = LIBUSB_CONTROL_SETUP_SIZE + length;
+ scoped_refptr<net::IOBuffer> resized_buffer(
+ new net::IOBufferWithSize(resized_length));
+ memcpy(resized_buffer->data() + LIBUSB_CONTROL_SETUP_SIZE, buffer->data(),
+ length);
+
+ struct libusb_transfer* const transfer = libusb_alloc_transfer(0);
+ const uint8 converted_type =
+ CreateRequestType(direction, request_type, recipient);
+ libusb_fill_control_setup(reinterpret_cast<uint8*>(resized_buffer->data()),
+ converted_type, request, value, index, length);
+ libusb_fill_control_transfer(transfer, handle_,
+ reinterpret_cast<uint8*>(resized_buffer->data()),
+ HandleTransferCompletion, this, timeout);
+ BrowserThread::PostTask(BrowserThread::FILE, FROM_HERE,
+ base::Bind(&UsbDeviceHandle::SubmitTransfer, this,
+ transfer, USB_TRANSFER_CONTROL,
+ resized_buffer, resized_length, callback));
+}
+
+void UsbDeviceHandle::BulkTransfer(const UsbEndpointDirection direction,
+ const uint8 endpoint, net::IOBuffer* buffer,
+ const size_t length,
+ const unsigned int timeout,
+ const UsbTransferCallback& callback) {
+ CHECK_DEVICE(callback, USB_TRANSFER_DISCONNECT,
+ scoped_refptr<net::IOBuffer>(), 0);
+
+ struct libusb_transfer* const transfer = libusb_alloc_transfer(0);
+ const uint8 new_endpoint = ConvertTransferDirection(direction) | endpoint;
+ libusb_fill_bulk_transfer(transfer, handle_, new_endpoint,
+ reinterpret_cast<uint8*>(buffer->data()), length,
+ HandleTransferCompletion, this, timeout);
+ BrowserThread::PostTask(
+ BrowserThread::FILE, FROM_HERE,
+ base::Bind(&UsbDeviceHandle::SubmitTransfer, this, transfer,
+ USB_TRANSFER_BULK, make_scoped_refptr(buffer), length,
+ callback));
+}
+
+void UsbDeviceHandle::InterruptTransfer(const UsbEndpointDirection direction,
+ const uint8 endpoint,
+ net::IOBuffer* buffer,
+ const size_t length,
+ const unsigned int timeout,
+ const UsbTransferCallback& callback) {
+ CHECK_DEVICE(callback, USB_TRANSFER_DISCONNECT,
+ scoped_refptr<net::IOBuffer>(), 0);
+
+ struct libusb_transfer* const transfer = libusb_alloc_transfer(0);
+ const uint8 new_endpoint = ConvertTransferDirection(direction) | endpoint;
+ libusb_fill_interrupt_transfer(
+ transfer, handle_, new_endpoint, reinterpret_cast<uint8*>(buffer->data()),
+ length, HandleTransferCompletion, this, timeout);
+ BrowserThread::PostTask(
+ BrowserThread::FILE, FROM_HERE,
+ base::Bind(&UsbDeviceHandle::SubmitTransfer, this, transfer,
+ USB_TRANSFER_INTERRUPT, make_scoped_refptr(buffer), length,
+ callback));
+}
+
+void UsbDeviceHandle::IsochronousTransfer(
+ const UsbEndpointDirection direction, const uint8 endpoint,
+ net::IOBuffer* buffer, const size_t length, const unsigned int packets,
+ const unsigned int packet_length, const unsigned int timeout,
+ const UsbTransferCallback& callback) {
+ CHECK_DEVICE(callback, USB_TRANSFER_DISCONNECT,
+ scoped_refptr<net::IOBuffer>(), 0);
+
+ const uint64 total_length = packets * packet_length;
+ CHECK(packets <= length && total_length <= length)
+ << "transfer length is too small";
+
+ struct libusb_transfer* const transfer = libusb_alloc_transfer(packets);
+ const uint8 new_endpoint = ConvertTransferDirection(direction) | endpoint;
+ libusb_fill_iso_transfer(transfer, handle_, new_endpoint,
+ reinterpret_cast<uint8*>(buffer->data()), length,
+ packets, HandleTransferCompletion, this, timeout);
+ libusb_set_iso_packet_lengths(transfer, packet_length);
+
+ BrowserThread::PostTask(
+ BrowserThread::FILE, FROM_HERE,
+ base::Bind(&UsbDeviceHandle::SubmitTransfer, this, transfer,
+ USB_TRANSFER_ISOCHRONOUS, make_scoped_refptr(buffer), length,
+ callback));
+}
+
+void UsbDeviceHandle::ResetDevice(const base::Callback<void(bool)>& callback) {
+ // Blocking operation. Run it on the FILE thread.
+ DCHECK(thread_checker_.CalledOnValidThread());
+ CHECK_DEVICE(callback, false);
+ callback.Run(libusb_reset_device(handle_) == 0);
+}
+
+void UsbDeviceHandle::InternalClose() {
+ DCHECK(thread_checker_.CalledOnValidThread());
+ if (handle_ == NULL) return;
+
+ // The following four lines makes this function re-enterable in case the
+ // callbacks call InternalClose again by, e.g., removing the UsbDevice from
+ // UsbService.
+ PlatformUsbDeviceHandle handle = handle_;
+ handle_ = NULL;
+ std::map<PlatformUsbTransferHandle, Transfer> transfers;
+ std::swap(transfers, transfers_);
+
+ // Cancel all the transfers before libusb_close.
+ // Otherwise the callback will not be invoked.
+ for (std::map<PlatformUsbTransferHandle, Transfer>::iterator it =
+ transfers.begin();
+ it != transfers.end(); it++) {
+ it->first->user_data = NULL;
+ it->second.callback
+ .Run(USB_TRANSFER_DISCONNECT, scoped_refptr<net::IOBuffer>(), 0);
+ }
+ transfers_.clear();
+ libusb_close(handle);
+}
+
+void UsbDeviceHandle::SubmitTransfer(PlatformUsbTransferHandle handle,
+ UsbTransferType transfer_type,
+ net::IOBuffer* buffer, const size_t length,
+ const UsbTransferCallback& callback) {
+ DCHECK(thread_checker_.CalledOnValidThread());
+ // This check must be done after the lock.
+ if (!handle_) return;
+
+ Transfer transfer;
+ transfer.transfer_type = transfer_type;
+ transfer.buffer = buffer;
+ transfer.length = length;
+ transfer.callback = callback;
+
+ transfers_[handle] = transfer;
+ libusb_submit_transfer(handle);
+}
« no previous file with comments | « chrome/browser/usb/usb_device_handle.h ('k') | chrome/browser/usb/usb_interface.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698