Chromium Code Reviews| Index: components/usb_service/usb_device_handle_impl.cc |
| diff --git a/components/usb_service/usb_device_handle_impl.cc b/components/usb_service/usb_device_handle_impl.cc |
| index 73b04b95eb7f260121ce1255951c6998d4a9fcf6..a34f0b4abcf69fdd6c80a2134f9240a1ed0a3412 100644 |
| --- a/components/usb_service/usb_device_handle_impl.cc |
| +++ b/components/usb_service/usb_device_handle_impl.cc |
| @@ -7,20 +7,20 @@ |
| #include <algorithm> |
| #include <vector> |
| -#include "base/message_loop/message_loop.h" |
| +#include "base/bind.h" |
| +#include "base/location.h" |
| +#include "base/single_thread_task_runner.h" |
| #include "base/stl_util.h" |
| #include "base/strings/string16.h" |
| #include "base/synchronization/lock.h" |
| +#include "base/thread_task_runner_handle.h" |
| #include "components/usb_service/usb_context.h" |
| #include "components/usb_service/usb_device_impl.h" |
| #include "components/usb_service/usb_error.h" |
| #include "components/usb_service/usb_interface.h" |
| #include "components/usb_service/usb_service.h" |
| -#include "content/public/browser/browser_thread.h" |
| #include "third_party/libusb/src/libusb/libusb.h" |
| -using content::BrowserThread; |
| - |
| namespace usb_service { |
| typedef libusb_device* PlatformUsbDevice; |
| @@ -103,24 +103,8 @@ static UsbTransferStatus ConvertTransferStatus( |
| } |
| } |
| -static void LIBUSB_CALL |
| -PlatformTransferCompletionCallback(PlatformUsbTransferHandle transfer) { |
| - BrowserThread::PostTask(BrowserThread::FILE, |
| - FROM_HERE, |
| - base::Bind(HandleTransferCompletion, transfer)); |
| -} |
| - |
| } // namespace |
| -void HandleTransferCompletion(PlatformUsbTransferHandle transfer) { |
| - DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); |
| - UsbDeviceHandleImpl* const device_handle = |
| - reinterpret_cast<UsbDeviceHandleImpl*>(transfer->user_data); |
| - CHECK(device_handle) << "Device handle is closed before transfer finishes."; |
| - device_handle->TransferComplete(transfer); |
| - libusb_free_transfer(transfer); |
| -} |
| - |
| class UsbDeviceHandleImpl::InterfaceClaimer |
| : public base::RefCountedThreadSafe<UsbDeviceHandleImpl::InterfaceClaimer> { |
| public: |
| @@ -170,10 +154,12 @@ struct UsbDeviceHandleImpl::Transfer { |
| Transfer(); |
| ~Transfer(); |
| + void Complete(UsbTransferStatus status, size_t bytes_transferred); |
| + |
| UsbTransferType transfer_type; |
| scoped_refptr<net::IOBuffer> buffer; |
| scoped_refptr<UsbDeviceHandleImpl::InterfaceClaimer> claimed_interface; |
| - scoped_refptr<base::MessageLoopProxy> message_loop_proxy; |
| + scoped_refptr<base::SingleThreadTaskRunner> task_runner; |
| size_t length; |
| UsbTransferCallback callback; |
| }; |
| @@ -185,6 +171,16 @@ UsbDeviceHandleImpl::Transfer::Transfer() |
| UsbDeviceHandleImpl::Transfer::~Transfer() { |
| } |
| +void UsbDeviceHandleImpl::Transfer::Complete(UsbTransferStatus status, |
| + size_t bytes_transferred) { |
| + if (task_runner->RunsTasksOnCurrentThread()) { |
| + callback.Run(status, buffer, bytes_transferred); |
| + } else { |
| + task_runner->PostTask( |
| + FROM_HERE, base::Bind(callback, status, buffer, bytes_transferred)); |
| + } |
| +} |
| + |
| UsbDeviceHandleImpl::UsbDeviceHandleImpl( |
| scoped_refptr<UsbContext> context, |
| UsbDeviceImpl* device, |
| @@ -193,8 +189,8 @@ UsbDeviceHandleImpl::UsbDeviceHandleImpl( |
| : device_(device), |
| handle_(handle), |
| interfaces_(interfaces), |
| - context_(context) { |
| - DCHECK(thread_checker_.CalledOnValidThread()); |
| + context_(context), |
| + task_runner_(base::ThreadTaskRunnerHandle::Get()) { |
| DCHECK(handle) << "Cannot create device with NULL handle."; |
| DCHECK(interfaces_.get()) << "Unable to list interfaces"; |
| } |
| @@ -216,7 +212,17 @@ void UsbDeviceHandleImpl::Close() { |
| device_->Close(this); |
| } |
| -void UsbDeviceHandleImpl::TransferComplete(PlatformUsbTransferHandle handle) { |
| +/* static */ void LIBUSB_CALL UsbDeviceHandleImpl::PlatformTransferCallback( |
|
Ken Rockot(use gerrit already)
2014/08/28 21:47:03
// static
on its own line
Reilly Grant (use Gerrit)
2014/08/28 22:16:12
Done.
|
| + PlatformUsbTransferHandle transfer) { |
| + UsbDeviceHandleImpl* device_handle = |
| + reinterpret_cast<UsbDeviceHandleImpl*>(transfer->user_data); |
| + device_handle->task_runner_->PostTask( |
| + FROM_HERE, |
| + base::Bind( |
| + &UsbDeviceHandleImpl::CompleteTransfer, device_handle, transfer)); |
| +} |
| + |
| +void UsbDeviceHandleImpl::CompleteTransfer(PlatformUsbTransferHandle handle) { |
| DCHECK(ContainsKey(transfers_, handle)) << "Missing transfer completed"; |
| Transfer transfer = transfers_[handle]; |
| @@ -229,7 +235,6 @@ void UsbDeviceHandleImpl::TransferComplete(PlatformUsbTransferHandle handle) { |
| 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 |
| @@ -246,9 +251,9 @@ void UsbDeviceHandleImpl::TransferComplete(PlatformUsbTransferHandle handle) { |
| new net::IOBuffer(static_cast<int>( |
| std::max(actual_length, static_cast<size_t>(1)))); |
| memcpy(resized_buffer->data(), |
| - buffer->data() + LIBUSB_CONTROL_SETUP_SIZE, |
| + transfer.buffer->data() + LIBUSB_CONTROL_SETUP_SIZE, |
| actual_length); |
| - buffer = resized_buffer; |
| + transfer.buffer = resized_buffer; |
| } |
| } |
| break; |
| @@ -268,8 +273,8 @@ void UsbDeviceHandleImpl::TransferComplete(PlatformUsbTransferHandle handle) { |
| 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, |
| + memmove(transfer.buffer->data() + actual_length, |
| + transfer.buffer->data() + packet_buffer_start, |
| packet->actual_length); |
| } |
| actual_length += packet->actual_length; |
| @@ -289,12 +294,8 @@ void UsbDeviceHandleImpl::TransferComplete(PlatformUsbTransferHandle handle) { |
| break; |
| } |
| - transfer.message_loop_proxy->PostTask( |
| - FROM_HERE, |
| - base::Bind(transfer.callback, |
| - ConvertTransferStatus(handle->status), |
| - buffer, |
| - actual_length)); |
| + transfer.Complete(ConvertTransferStatus(handle->status), actual_length); |
| + libusb_free_transfer(handle); |
| // Must release interface first before actually delete this. |
| transfer.claimed_interface = NULL; |
| @@ -552,20 +553,12 @@ void UsbDeviceHandleImpl::ControlTransfer( |
| libusb_fill_control_transfer(transfer, |
| handle_, |
| reinterpret_cast<uint8*>(resized_buffer->data()), |
| - PlatformTransferCompletionCallback, |
| + &UsbDeviceHandleImpl::PlatformTransferCallback, |
| this, |
| timeout); |
| - BrowserThread::PostTask(BrowserThread::FILE, |
| - FROM_HERE, |
| - base::Bind(&UsbDeviceHandleImpl::SubmitTransfer, |
| - this, |
| - transfer, |
| - USB_TRANSFER_CONTROL, |
| - resized_buffer, |
| - resized_length, |
| - base::MessageLoopProxy::current(), |
| - callback)); |
| + PostOrSubmitTransfer( |
| + transfer, USB_TRANSFER_CONTROL, resized_buffer, resized_length, callback); |
| } |
| void UsbDeviceHandleImpl::BulkTransfer(const UsbEndpointDirection direction, |
| @@ -586,20 +579,11 @@ void UsbDeviceHandleImpl::BulkTransfer(const UsbEndpointDirection direction, |
| new_endpoint, |
| reinterpret_cast<uint8*>(buffer->data()), |
| static_cast<int>(length), |
| - PlatformTransferCompletionCallback, |
| + &UsbDeviceHandleImpl::PlatformTransferCallback, |
| this, |
| timeout); |
| - BrowserThread::PostTask(BrowserThread::FILE, |
| - FROM_HERE, |
| - base::Bind(&UsbDeviceHandleImpl::SubmitTransfer, |
| - this, |
| - transfer, |
| - USB_TRANSFER_BULK, |
| - make_scoped_refptr(buffer), |
| - length, |
| - base::MessageLoopProxy::current(), |
| - callback)); |
| + PostOrSubmitTransfer(transfer, USB_TRANSFER_BULK, buffer, length, callback); |
| } |
| void UsbDeviceHandleImpl::InterruptTransfer( |
| @@ -621,19 +605,12 @@ void UsbDeviceHandleImpl::InterruptTransfer( |
| new_endpoint, |
| reinterpret_cast<uint8*>(buffer->data()), |
| static_cast<int>(length), |
| - PlatformTransferCompletionCallback, |
| + &UsbDeviceHandleImpl::PlatformTransferCallback, |
| this, |
| timeout); |
| - BrowserThread::PostTask(BrowserThread::FILE, |
| - FROM_HERE, |
| - base::Bind(&UsbDeviceHandleImpl::SubmitTransfer, |
| - this, |
| - transfer, |
| - USB_TRANSFER_INTERRUPT, |
| - make_scoped_refptr(buffer), |
| - length, |
| - base::MessageLoopProxy::current(), |
| - callback)); |
| + |
| + PostOrSubmitTransfer( |
| + transfer, USB_TRANSFER_INTERRUPT, buffer, length, callback); |
| } |
| void UsbDeviceHandleImpl::IsochronousTransfer( |
| @@ -662,21 +639,13 @@ void UsbDeviceHandleImpl::IsochronousTransfer( |
| reinterpret_cast<uint8*>(buffer->data()), |
| static_cast<int>(length), |
| packets, |
| - PlatformTransferCompletionCallback, |
| + &UsbDeviceHandleImpl::PlatformTransferCallback, |
| this, |
| timeout); |
| libusb_set_iso_packet_lengths(transfer, packet_length); |
| - BrowserThread::PostTask(BrowserThread::FILE, |
| - FROM_HERE, |
| - base::Bind(&UsbDeviceHandleImpl::SubmitTransfer, |
| - this, |
| - transfer, |
| - USB_TRANSFER_ISOCHRONOUS, |
| - make_scoped_refptr(buffer), |
| - length, |
| - base::MessageLoopProxy::current(), |
| - callback)); |
| + PostOrSubmitTransfer( |
| + transfer, USB_TRANSFER_ISOCHRONOUS, buffer, length, callback); |
| } |
| void UsbDeviceHandleImpl::RefreshEndpointMap() { |
| @@ -704,27 +673,52 @@ UsbDeviceHandleImpl::GetClaimedInterfaceForEndpoint(unsigned char endpoint) { |
| return NULL; |
| } |
| +void UsbDeviceHandleImpl::PostOrSubmitTransfer( |
| + PlatformUsbTransferHandle transfer, |
| + UsbTransferType transfer_type, |
| + net::IOBuffer* buffer, |
| + size_t length, |
| + const UsbTransferCallback& callback) { |
| + if (task_runner_->RunsTasksOnCurrentThread()) { |
| + SubmitTransfer(transfer, |
| + transfer_type, |
| + buffer, |
| + length, |
| + base::ThreadTaskRunnerHandle::Get(), |
| + callback); |
| + } else { |
| + task_runner_->PostTask(FROM_HERE, |
| + base::Bind(&UsbDeviceHandleImpl::SubmitTransfer, |
| + this, |
| + transfer, |
| + transfer_type, |
| + make_scoped_refptr(buffer), |
| + length, |
| + base::ThreadTaskRunnerHandle::Get(), |
| + callback)); |
| + } |
| +} |
| + |
| void UsbDeviceHandleImpl::SubmitTransfer( |
| PlatformUsbTransferHandle handle, |
| UsbTransferType transfer_type, |
| net::IOBuffer* buffer, |
| const size_t length, |
| - scoped_refptr<base::MessageLoopProxy> message_loop_proxy, |
| + scoped_refptr<base::SingleThreadTaskRunner> task_runner, |
| const UsbTransferCallback& callback) { |
| DCHECK(thread_checker_.CalledOnValidThread()); |
| - if (!device_) { |
| - message_loop_proxy->PostTask( |
| - FROM_HERE, |
| - base::Bind( |
| - callback, USB_TRANSFER_DISCONNECT, make_scoped_refptr(buffer), 0)); |
| - } |
| Transfer transfer; |
| transfer.transfer_type = transfer_type; |
| transfer.buffer = buffer; |
| transfer.length = length; |
| transfer.callback = callback; |
| - transfer.message_loop_proxy = message_loop_proxy; |
| + transfer.task_runner = task_runner; |
| + |
| + if (!device_) { |
| + transfer.Complete(USB_TRANSFER_DISCONNECT, 0); |
| + return; |
| + } |
| // It's OK for this method to return NULL. libusb_submit_transfer will fail if |
| // it requires an interface we didn't claim. |
| @@ -735,10 +729,7 @@ void UsbDeviceHandleImpl::SubmitTransfer( |
| transfers_[handle] = transfer; |
| } else { |
| VLOG(1) << "Failed to submit transfer: " << ConvertErrorToString(rv); |
| - message_loop_proxy->PostTask( |
| - FROM_HERE, |
| - base::Bind( |
| - callback, USB_TRANSFER_ERROR, make_scoped_refptr(buffer), 0)); |
| + transfer.Complete(USB_TRANSFER_ERROR, 0); |
| } |
| } |