1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "device/usb/usb_device_handle_impl.h"
10 #include "base/bind.h"
11 #include "base/location.h"
12 #include "base/single_thread_task_runner.h"
13 #include "base/stl_util.h"
14 #include "base/strings/string16.h"
15 #include "base/synchronization/lock.h"
16 #include "base/thread_task_runner_handle.h"
17 #include "components/device_event_log/device_event_log.h"
18 #include "device/usb/usb_context.h"
19 #include "device/usb/usb_descriptors.h"
20 #include "device/usb/usb_device_impl.h"
21 #include "device/usb/usb_error.h"
22 #include "device/usb/usb_service.h"
23 #include "third_party/libusb/src/libusb/libusb.h"
27 typedef libusb_device
* PlatformUsbDevice
;
29 void HandleTransferCompletion(PlatformUsbTransferHandle transfer
);
33 static uint8
ConvertTransferDirection(const UsbEndpointDirection direction
) {
35 case USB_DIRECTION_INBOUND
:
36 return LIBUSB_ENDPOINT_IN
;
37 case USB_DIRECTION_OUTBOUND
:
38 return LIBUSB_ENDPOINT_OUT
;
41 return LIBUSB_ENDPOINT_IN
;
45 static uint8
CreateRequestType(
46 const UsbEndpointDirection direction
,
47 const UsbDeviceHandle::TransferRequestType request_type
,
48 const UsbDeviceHandle::TransferRecipient recipient
) {
49 uint8 result
= ConvertTransferDirection(direction
);
51 switch (request_type
) {
52 case UsbDeviceHandle::STANDARD
:
53 result
|= LIBUSB_REQUEST_TYPE_STANDARD
;
55 case UsbDeviceHandle::CLASS
:
56 result
|= LIBUSB_REQUEST_TYPE_CLASS
;
58 case UsbDeviceHandle::VENDOR
:
59 result
|= LIBUSB_REQUEST_TYPE_VENDOR
;
61 case UsbDeviceHandle::RESERVED
:
62 result
|= LIBUSB_REQUEST_TYPE_RESERVED
;
67 case UsbDeviceHandle::DEVICE
:
68 result
|= LIBUSB_RECIPIENT_DEVICE
;
70 case UsbDeviceHandle::INTERFACE
:
71 result
|= LIBUSB_RECIPIENT_INTERFACE
;
73 case UsbDeviceHandle::ENDPOINT
:
74 result
|= LIBUSB_RECIPIENT_ENDPOINT
;
76 case UsbDeviceHandle::OTHER
:
77 result
|= LIBUSB_RECIPIENT_OTHER
;
84 static UsbTransferStatus
ConvertTransferStatus(
85 const libusb_transfer_status status
) {
87 case LIBUSB_TRANSFER_COMPLETED
:
88 return USB_TRANSFER_COMPLETED
;
89 case LIBUSB_TRANSFER_ERROR
:
90 return USB_TRANSFER_ERROR
;
91 case LIBUSB_TRANSFER_TIMED_OUT
:
92 return USB_TRANSFER_TIMEOUT
;
93 case LIBUSB_TRANSFER_STALL
:
94 return USB_TRANSFER_STALLED
;
95 case LIBUSB_TRANSFER_NO_DEVICE
:
96 return USB_TRANSFER_DISCONNECT
;
97 case LIBUSB_TRANSFER_OVERFLOW
:
98 return USB_TRANSFER_OVERFLOW
;
99 case LIBUSB_TRANSFER_CANCELLED
:
100 return USB_TRANSFER_CANCELLED
;
103 return USB_TRANSFER_ERROR
;
109 class UsbDeviceHandleImpl::InterfaceClaimer
110 : public base::RefCountedThreadSafe
<UsbDeviceHandleImpl::InterfaceClaimer
> {
112 InterfaceClaimer(const scoped_refptr
<UsbDeviceHandleImpl
> handle
,
113 const int interface_number
);
117 int alternate_setting() const { return alternate_setting_
; }
118 void set_alternate_setting(const int alternate_setting
) {
119 alternate_setting_
= alternate_setting
;
123 friend class UsbDevice
;
124 friend class base::RefCountedThreadSafe
<InterfaceClaimer
>;
127 const scoped_refptr
<UsbDeviceHandleImpl
> handle_
;
128 const int interface_number_
;
129 int alternate_setting_
;
131 DISALLOW_COPY_AND_ASSIGN(InterfaceClaimer
);
134 UsbDeviceHandleImpl::InterfaceClaimer::InterfaceClaimer(
135 const scoped_refptr
<UsbDeviceHandleImpl
> handle
,
136 const int interface_number
)
138 interface_number_(interface_number
),
139 alternate_setting_(0) {
142 UsbDeviceHandleImpl::InterfaceClaimer::~InterfaceClaimer() {
143 libusb_release_interface(handle_
->handle(), interface_number_
);
146 bool UsbDeviceHandleImpl::InterfaceClaimer::Claim() const {
147 const int rv
= libusb_claim_interface(handle_
->handle(), interface_number_
);
148 if (rv
!= LIBUSB_SUCCESS
) {
149 USB_LOG(EVENT
) << "Failed to claim interface " << interface_number_
<< ": "
150 << ConvertPlatformUsbErrorToString(rv
);
152 return rv
== LIBUSB_SUCCESS
;
155 // This inner class owns the underlying libusb_transfer and may outlast
156 // the UsbDeviceHandle that created it.
157 class UsbDeviceHandleImpl::Transfer
{
159 static scoped_ptr
<Transfer
> CreateControlTransfer(
160 scoped_refptr
<UsbDeviceHandleImpl
> device_handle
,
166 scoped_refptr
<net::IOBuffer
> buffer
,
167 unsigned int timeout
,
168 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
169 const TransferCallback
& callback
);
170 static scoped_ptr
<Transfer
> CreateBulkTransfer(
171 scoped_refptr
<UsbDeviceHandleImpl
> device_handle
,
173 scoped_refptr
<net::IOBuffer
> buffer
,
175 unsigned int timeout
,
176 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
177 const TransferCallback
& callback
);
178 static scoped_ptr
<Transfer
> CreateInterruptTransfer(
179 scoped_refptr
<UsbDeviceHandleImpl
> device_handle
,
181 scoped_refptr
<net::IOBuffer
> buffer
,
183 unsigned int timeout
,
184 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
185 const TransferCallback
& callback
);
186 static scoped_ptr
<Transfer
> CreateIsochronousTransfer(
187 scoped_refptr
<UsbDeviceHandleImpl
> device_handle
,
189 scoped_refptr
<net::IOBuffer
> buffer
,
191 unsigned int packets
,
192 unsigned int packet_length
,
193 unsigned int timeout
,
194 scoped_refptr
<base::TaskRunner
> task_runner
,
195 const TransferCallback
& callback
);
201 void ProcessCompletion();
202 void TransferComplete(UsbTransferStatus status
, size_t bytes_transferred
);
204 const UsbDeviceHandleImpl::InterfaceClaimer
* claimed_interface() const {
205 return claimed_interface_
.get();
208 scoped_refptr
<base::TaskRunner
> callback_task_runner() const {
209 return callback_task_runner_
;
213 Transfer(scoped_refptr
<UsbDeviceHandleImpl
> device_handle
,
214 scoped_refptr
<InterfaceClaimer
> claimed_interface
,
215 UsbTransferType transfer_type
,
216 scoped_refptr
<net::IOBuffer
> buffer
,
218 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
219 const TransferCallback
& callback
);
221 static void LIBUSB_CALL
PlatformCallback(PlatformUsbTransferHandle handle
);
223 UsbTransferType transfer_type_
;
224 scoped_refptr
<UsbDeviceHandleImpl
> device_handle_
;
225 PlatformUsbTransferHandle platform_transfer_
= nullptr;
226 scoped_refptr
<net::IOBuffer
> buffer_
;
227 scoped_refptr
<UsbDeviceHandleImpl::InterfaceClaimer
> claimed_interface_
;
229 bool cancelled_
= false;
230 scoped_refptr
<base::SequencedTaskRunner
> task_runner_
;
231 scoped_refptr
<base::TaskRunner
> callback_task_runner_
;
232 TransferCallback callback_
;
236 scoped_ptr
<UsbDeviceHandleImpl::Transfer
>
237 UsbDeviceHandleImpl::Transfer::CreateControlTransfer(
238 scoped_refptr
<UsbDeviceHandleImpl
> device_handle
,
244 scoped_refptr
<net::IOBuffer
> buffer
,
245 unsigned int timeout
,
246 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
247 const TransferCallback
& callback
) {
248 scoped_ptr
<Transfer
> transfer(new Transfer(
249 device_handle
, nullptr, USB_TRANSFER_CONTROL
, buffer
,
250 length
+ LIBUSB_CONTROL_SETUP_SIZE
, callback_task_runner
, callback
));
252 transfer
->platform_transfer_
= libusb_alloc_transfer(0);
253 if (!transfer
->platform_transfer_
) {
254 USB_LOG(ERROR
) << "Failed to allocate control transfer.";
258 libusb_fill_control_setup(reinterpret_cast<uint8
*>(buffer
->data()), type
,
259 request
, value
, index
, length
);
260 libusb_fill_control_transfer(transfer
->platform_transfer_
,
261 device_handle
->handle_
,
262 reinterpret_cast<uint8
*>(buffer
->data()),
263 &UsbDeviceHandleImpl::Transfer::PlatformCallback
,
264 transfer
.get(), timeout
);
266 return transfer
.Pass();
270 scoped_ptr
<UsbDeviceHandleImpl::Transfer
>
271 UsbDeviceHandleImpl::Transfer::CreateBulkTransfer(
272 scoped_refptr
<UsbDeviceHandleImpl
> device_handle
,
274 scoped_refptr
<net::IOBuffer
> buffer
,
276 unsigned int timeout
,
277 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
278 const TransferCallback
& callback
) {
279 scoped_ptr
<Transfer
> transfer(new Transfer(
280 device_handle
, device_handle
->GetClaimedInterfaceForEndpoint(endpoint
),
281 USB_TRANSFER_BULK
, buffer
, length
, callback_task_runner
, callback
));
283 transfer
->platform_transfer_
= libusb_alloc_transfer(0);
284 if (!transfer
->platform_transfer_
) {
285 USB_LOG(ERROR
) << "Failed to allocate bulk transfer.";
289 libusb_fill_bulk_transfer(
290 transfer
->platform_transfer_
, device_handle
->handle_
, endpoint
,
291 reinterpret_cast<uint8
*>(buffer
->data()), static_cast<int>(length
),
292 &UsbDeviceHandleImpl::Transfer::PlatformCallback
, transfer
.get(),
295 return transfer
.Pass();
299 scoped_ptr
<UsbDeviceHandleImpl::Transfer
>
300 UsbDeviceHandleImpl::Transfer::CreateInterruptTransfer(
301 scoped_refptr
<UsbDeviceHandleImpl
> device_handle
,
303 scoped_refptr
<net::IOBuffer
> buffer
,
305 unsigned int timeout
,
306 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
307 const TransferCallback
& callback
) {
308 scoped_ptr
<Transfer
> transfer(new Transfer(
309 device_handle
, device_handle
->GetClaimedInterfaceForEndpoint(endpoint
),
310 USB_TRANSFER_INTERRUPT
, buffer
, length
, callback_task_runner
, callback
));
312 transfer
->platform_transfer_
= libusb_alloc_transfer(0);
313 if (!transfer
->platform_transfer_
) {
314 USB_LOG(ERROR
) << "Failed to allocate interrupt transfer.";
318 libusb_fill_interrupt_transfer(
319 transfer
->platform_transfer_
, device_handle
->handle_
, endpoint
,
320 reinterpret_cast<uint8
*>(buffer
->data()), static_cast<int>(length
),
321 &UsbDeviceHandleImpl::Transfer::PlatformCallback
, transfer
.get(),
324 return transfer
.Pass();
328 scoped_ptr
<UsbDeviceHandleImpl::Transfer
>
329 UsbDeviceHandleImpl::Transfer::CreateIsochronousTransfer(
330 scoped_refptr
<UsbDeviceHandleImpl
> device_handle
,
332 scoped_refptr
<net::IOBuffer
> buffer
,
334 unsigned int packets
,
335 unsigned int packet_length
,
336 unsigned int timeout
,
337 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
338 const TransferCallback
& callback
) {
339 DCHECK(packets
<= length
&& (packets
* packet_length
) <= length
)
340 << "transfer length is too small";
342 scoped_ptr
<Transfer
> transfer(new Transfer(
343 device_handle
, device_handle
->GetClaimedInterfaceForEndpoint(endpoint
),
344 USB_TRANSFER_ISOCHRONOUS
, buffer
, length
, callback_task_runner
,
347 transfer
->platform_transfer_
= libusb_alloc_transfer(packets
);
348 if (!transfer
->platform_transfer_
) {
349 USB_LOG(ERROR
) << "Failed to allocate isochronous transfer.";
353 libusb_fill_iso_transfer(
354 transfer
->platform_transfer_
, device_handle
->handle_
, endpoint
,
355 reinterpret_cast<uint8
*>(buffer
->data()), static_cast<int>(length
),
356 packets
, &Transfer::PlatformCallback
, transfer
.get(), timeout
);
357 libusb_set_iso_packet_lengths(transfer
->platform_transfer_
, packet_length
);
359 return transfer
.Pass();
362 UsbDeviceHandleImpl::Transfer::Transfer(
363 scoped_refptr
<UsbDeviceHandleImpl
> device_handle
,
364 scoped_refptr
<InterfaceClaimer
> claimed_interface
,
365 UsbTransferType transfer_type
,
366 scoped_refptr
<net::IOBuffer
> buffer
,
368 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
369 const TransferCallback
& callback
)
370 : transfer_type_(transfer_type
),
371 device_handle_(device_handle
),
373 claimed_interface_(claimed_interface
),
375 callback_task_runner_(callback_task_runner
),
376 callback_(callback
) {
377 task_runner_
= base::ThreadTaskRunnerHandle::Get();
380 UsbDeviceHandleImpl::Transfer::~Transfer() {
381 if (platform_transfer_
) {
382 libusb_free_transfer(platform_transfer_
);
386 void UsbDeviceHandleImpl::Transfer::Submit() {
387 const int rv
= libusb_submit_transfer(platform_transfer_
);
388 if (rv
!= LIBUSB_SUCCESS
) {
389 USB_LOG(EVENT
) << "Failed to submit transfer: "
390 << ConvertPlatformUsbErrorToString(rv
);
391 TransferComplete(USB_TRANSFER_ERROR
, 0);
395 void UsbDeviceHandleImpl::Transfer::Cancel() {
397 libusb_cancel_transfer(platform_transfer_
);
398 claimed_interface_
= nullptr;
403 void UsbDeviceHandleImpl::Transfer::ProcessCompletion() {
404 DCHECK_GE(platform_transfer_
->actual_length
, 0)
405 << "Negative actual length received";
406 size_t actual_length
=
407 static_cast<size_t>(std::max(platform_transfer_
->actual_length
, 0));
409 DCHECK(length_
>= actual_length
)
410 << "data too big for our buffer (libusb failure?)";
412 switch (transfer_type_
) {
413 case USB_TRANSFER_CONTROL
:
414 // If the transfer is a control transfer we do not expose the control
415 // setup header to the caller. This logic strips off the header if
416 // present before invoking the callback provided with the transfer.
417 if (actual_length
> 0) {
418 CHECK(length_
>= LIBUSB_CONTROL_SETUP_SIZE
)
419 << "buffer was not correctly set: too small for the control header";
421 if (length_
>= (LIBUSB_CONTROL_SETUP_SIZE
+ actual_length
)) {
422 // If the payload is zero bytes long, pad out the allocated buffer
423 // size to one byte so that an IOBuffer of that size can be allocated.
424 scoped_refptr
<net::IOBuffer
> resized_buffer
=
425 new net::IOBuffer(static_cast<int>(
426 std::max(actual_length
, static_cast<size_t>(1))));
427 memcpy(resized_buffer
->data(),
428 buffer_
->data() + LIBUSB_CONTROL_SETUP_SIZE
, actual_length
);
429 buffer_
= resized_buffer
;
434 case USB_TRANSFER_ISOCHRONOUS
:
435 // Isochronous replies might carry data in the different isoc packets even
436 // if the transfer actual_data value is zero. Furthermore, not all of the
437 // received packets might contain data, so we need to calculate how many
438 // data bytes we are effectively providing and pack the results.
439 if (actual_length
== 0) {
440 size_t packet_buffer_start
= 0;
441 for (int i
= 0; i
< platform_transfer_
->num_iso_packets
; ++i
) {
442 PlatformUsbIsoPacketDescriptor packet
=
443 &platform_transfer_
->iso_packet_desc
[i
];
444 if (packet
->actual_length
> 0) {
445 // We don't need to copy as long as all packets until now provide
446 // all the data the packet can hold.
447 if (actual_length
< packet_buffer_start
) {
448 CHECK(packet_buffer_start
+ packet
->actual_length
<= length_
);
449 memmove(buffer_
->data() + actual_length
,
450 buffer_
->data() + packet_buffer_start
,
451 packet
->actual_length
);
453 actual_length
+= packet
->actual_length
;
456 packet_buffer_start
+= packet
->length
;
461 case USB_TRANSFER_BULK
:
462 case USB_TRANSFER_INTERRUPT
:
466 NOTREACHED() << "Invalid usb transfer type";
470 TransferComplete(ConvertTransferStatus(platform_transfer_
->status
),
475 void LIBUSB_CALL
UsbDeviceHandleImpl::Transfer::PlatformCallback(
476 PlatformUsbTransferHandle platform_transfer
) {
478 reinterpret_cast<Transfer
*>(platform_transfer
->user_data
);
479 DCHECK(transfer
->platform_transfer_
== platform_transfer
);
480 transfer
->ProcessCompletion();
483 void UsbDeviceHandleImpl::Transfer::TransferComplete(UsbTransferStatus status
,
484 size_t bytes_transferred
) {
485 task_runner_
->PostTask(
487 base::Bind(&UsbDeviceHandleImpl::TransferComplete
, device_handle_
,
488 base::Unretained(this),
489 base::Bind(callback_
, status
, buffer_
, bytes_transferred
)));
492 scoped_refptr
<UsbDevice
> UsbDeviceHandleImpl::GetDevice() const {
496 void UsbDeviceHandleImpl::Close() {
497 DCHECK(thread_checker_
.CalledOnValidThread());
499 device_
->Close(this);
502 void UsbDeviceHandleImpl::SetConfiguration(int configuration_value
,
503 const ResultCallback
& callback
) {
504 DCHECK(thread_checker_
.CalledOnValidThread());
510 for (Transfer
* transfer
: transfers_
) {
513 claimed_interfaces_
.clear();
515 blocking_task_runner_
->PostTask(
517 base::Bind(&UsbDeviceHandleImpl::SetConfigurationOnBlockingThread
, this,
518 configuration_value
, callback
));
521 void UsbDeviceHandleImpl::ClaimInterface(int interface_number
,
522 const ResultCallback
& callback
) {
523 DCHECK(thread_checker_
.CalledOnValidThread());
528 if (ContainsKey(claimed_interfaces_
, interface_number
)) {
533 blocking_task_runner_
->PostTask(
535 base::Bind(&UsbDeviceHandleImpl::ClaimInterfaceOnBlockingThread
, this,
536 interface_number
, callback
));
539 bool UsbDeviceHandleImpl::ReleaseInterface(int interface_number
) {
540 DCHECK(thread_checker_
.CalledOnValidThread());
543 if (!ContainsKey(claimed_interfaces_
, interface_number
))
546 // Cancel all the transfers on that interface.
547 InterfaceClaimer
* interface_claimer
=
548 claimed_interfaces_
[interface_number
].get();
549 for (Transfer
* transfer
: transfers_
) {
550 if (transfer
->claimed_interface() == interface_claimer
) {
554 claimed_interfaces_
.erase(interface_number
);
556 RefreshEndpointMap();
560 void UsbDeviceHandleImpl::SetInterfaceAlternateSetting(
561 int interface_number
,
562 int alternate_setting
,
563 const ResultCallback
& callback
) {
564 DCHECK(thread_checker_
.CalledOnValidThread());
565 if (!device_
|| !ContainsKey(claimed_interfaces_
, interface_number
)) {
570 blocking_task_runner_
->PostTask(
573 &UsbDeviceHandleImpl::SetInterfaceAlternateSettingOnBlockingThread
,
574 this, interface_number
, alternate_setting
, callback
));
577 void UsbDeviceHandleImpl::ResetDevice(const ResultCallback
& callback
) {
578 DCHECK(thread_checker_
.CalledOnValidThread());
584 blocking_task_runner_
->PostTask(
585 FROM_HERE
, base::Bind(&UsbDeviceHandleImpl::ResetDeviceOnBlockingThread
,
589 void UsbDeviceHandleImpl::ClearHalt(uint8 endpoint
,
590 const ResultCallback
& callback
) {
591 DCHECK(thread_checker_
.CalledOnValidThread());
597 InterfaceClaimer
* interface_claimer
=
598 GetClaimedInterfaceForEndpoint(endpoint
).get();
599 for (Transfer
* transfer
: transfers_
) {
600 if (transfer
->claimed_interface() == interface_claimer
) {
605 blocking_task_runner_
->PostTask(
606 FROM_HERE
, base::Bind(&UsbDeviceHandleImpl::ClearHaltOnBlockingThread
,
607 this, endpoint
, callback
));
610 void UsbDeviceHandleImpl::ControlTransfer(UsbEndpointDirection direction
,
611 TransferRequestType request_type
,
612 TransferRecipient recipient
,
616 scoped_refptr
<net::IOBuffer
> buffer
,
618 unsigned int timeout
,
619 const TransferCallback
& callback
) {
620 if (task_runner_
->BelongsToCurrentThread()) {
621 ControlTransferInternal(direction
, request_type
, recipient
, request
, value
,
622 index
, buffer
, length
, timeout
, task_runner_
,
625 task_runner_
->PostTask(
626 FROM_HERE
, base::Bind(&UsbDeviceHandleImpl::ControlTransferInternal
,
627 this, direction
, request_type
, recipient
, request
,
628 value
, index
, buffer
, length
, timeout
,
629 base::ThreadTaskRunnerHandle::Get(), callback
));
633 void UsbDeviceHandleImpl::IsochronousTransfer(
634 UsbEndpointDirection direction
,
636 scoped_refptr
<net::IOBuffer
> buffer
,
638 unsigned int packets
,
639 unsigned int packet_length
,
640 unsigned int timeout
,
641 const TransferCallback
& callback
) {
642 if (task_runner_
->BelongsToCurrentThread()) {
643 IsochronousTransferInternal(direction
, endpoint
, buffer
, length
, packets
,
644 packet_length
, timeout
, task_runner_
, callback
);
646 task_runner_
->PostTask(
648 base::Bind(&UsbDeviceHandleImpl::IsochronousTransferInternal
, this,
649 direction
, endpoint
, buffer
, length
, packets
, packet_length
,
650 timeout
, base::ThreadTaskRunnerHandle::Get(), callback
));
654 void UsbDeviceHandleImpl::GenericTransfer(UsbEndpointDirection direction
,
656 scoped_refptr
<net::IOBuffer
> buffer
,
658 unsigned int timeout
,
659 const TransferCallback
& callback
) {
660 if (task_runner_
->BelongsToCurrentThread()) {
661 GenericTransferInternal(direction
, endpoint
, buffer
, length
, timeout
,
662 task_runner_
, callback
);
664 task_runner_
->PostTask(
666 base::Bind(&UsbDeviceHandleImpl::GenericTransferInternal
, this,
667 direction
, endpoint
, buffer
, length
, timeout
,
668 base::ThreadTaskRunnerHandle::Get(), callback
));
672 UsbDeviceHandleImpl::UsbDeviceHandleImpl(
673 scoped_refptr
<UsbContext
> context
,
674 scoped_refptr
<UsbDeviceImpl
> device
,
675 PlatformUsbDeviceHandle handle
,
676 scoped_refptr
<base::SequencedTaskRunner
> blocking_task_runner
)
680 task_runner_(base::ThreadTaskRunnerHandle::Get()),
681 blocking_task_runner_(blocking_task_runner
) {
682 DCHECK(handle
) << "Cannot create device with NULL handle.";
685 UsbDeviceHandleImpl::~UsbDeviceHandleImpl() {
686 // This class is RefCountedThreadSafe and so the destructor may be called on
688 libusb_close(handle_
);
691 void UsbDeviceHandleImpl::SetConfigurationOnBlockingThread(
692 int configuration_value
,
693 const ResultCallback
& callback
) {
694 int rv
= libusb_set_configuration(handle_
, configuration_value
);
695 if (rv
!= LIBUSB_SUCCESS
) {
696 USB_LOG(EVENT
) << "Failed to set configuration " << configuration_value
697 << ": " << ConvertPlatformUsbErrorToString(rv
);
699 task_runner_
->PostTask(
700 FROM_HERE
, base::Bind(&UsbDeviceHandleImpl::SetConfigurationComplete
,
701 this, rv
== LIBUSB_SUCCESS
, callback
));
704 void UsbDeviceHandleImpl::SetConfigurationComplete(
706 const ResultCallback
& callback
) {
708 device_
->RefreshActiveConfiguration();
709 RefreshEndpointMap();
711 callback
.Run(success
);
714 void UsbDeviceHandleImpl::ClaimInterfaceOnBlockingThread(
715 int interface_number
,
716 const ResultCallback
& callback
) {
717 int rv
= libusb_claim_interface(handle_
, interface_number
);
718 if (rv
!= LIBUSB_SUCCESS
) {
719 VLOG(1) << "Failed to claim interface: "
720 << ConvertPlatformUsbErrorToString(rv
);
722 task_runner_
->PostTask(
723 FROM_HERE
, base::Bind(&UsbDeviceHandleImpl::ClaimInterfaceComplete
, this,
724 interface_number
, rv
== LIBUSB_SUCCESS
, callback
));
727 void UsbDeviceHandleImpl::ClaimInterfaceComplete(
728 int interface_number
,
730 const ResultCallback
& callback
) {
732 claimed_interfaces_
[interface_number
] =
733 new InterfaceClaimer(this, interface_number
);
734 RefreshEndpointMap();
736 callback
.Run(success
);
739 void UsbDeviceHandleImpl::SetInterfaceAlternateSettingOnBlockingThread(
740 int interface_number
,
741 int alternate_setting
,
742 const ResultCallback
& callback
) {
743 int rv
= libusb_set_interface_alt_setting(handle_
, interface_number
,
745 if (rv
!= LIBUSB_SUCCESS
) {
746 USB_LOG(EVENT
) << "Failed to set interface " << interface_number
747 << " to alternate setting " << alternate_setting
<< ": "
748 << ConvertPlatformUsbErrorToString(rv
);
750 task_runner_
->PostTask(
752 base::Bind(&UsbDeviceHandleImpl::SetInterfaceAlternateSettingComplete
,
753 this, interface_number
, alternate_setting
,
754 rv
== LIBUSB_SUCCESS
, callback
));
757 void UsbDeviceHandleImpl::SetInterfaceAlternateSettingComplete(
758 int interface_number
,
759 int alternate_setting
,
761 const ResultCallback
& callback
) {
763 claimed_interfaces_
[interface_number
]->set_alternate_setting(
765 RefreshEndpointMap();
767 callback
.Run(success
);
770 void UsbDeviceHandleImpl::ResetDeviceOnBlockingThread(
771 const ResultCallback
& callback
) {
772 int rv
= libusb_reset_device(handle_
);
773 if (rv
!= LIBUSB_SUCCESS
) {
774 USB_LOG(EVENT
) << "Failed to reset device: "
775 << ConvertPlatformUsbErrorToString(rv
);
777 task_runner_
->PostTask(FROM_HERE
, base::Bind(callback
, rv
== LIBUSB_SUCCESS
));
780 void UsbDeviceHandleImpl::ClearHaltOnBlockingThread(
782 const ResultCallback
& callback
) {
783 int rv
= libusb_clear_halt(handle_
, endpoint
);
784 if (rv
!= LIBUSB_SUCCESS
) {
785 USB_LOG(EVENT
) << "Failed to clear halt: "
786 << ConvertPlatformUsbErrorToString(rv
);
788 task_runner_
->PostTask(FROM_HERE
, base::Bind(callback
, rv
== LIBUSB_SUCCESS
));
791 void UsbDeviceHandleImpl::RefreshEndpointMap() {
792 DCHECK(thread_checker_
.CalledOnValidThread());
793 endpoint_map_
.clear();
794 const UsbConfigDescriptor
* config
= device_
->GetActiveConfiguration();
796 for (const auto& map_entry
: claimed_interfaces_
) {
797 int interface_number
= map_entry
.first
;
798 const scoped_refptr
<InterfaceClaimer
>& claimed_iface
= map_entry
.second
;
800 for (const UsbInterfaceDescriptor
& iface
: config
->interfaces
) {
801 if (iface
.interface_number
== interface_number
&&
802 iface
.alternate_setting
== claimed_iface
->alternate_setting()) {
803 for (const UsbEndpointDescriptor
& endpoint
: iface
.endpoints
) {
804 endpoint_map_
[endpoint
.address
] = {interface_number
,
805 endpoint
.transfer_type
};
814 scoped_refptr
<UsbDeviceHandleImpl::InterfaceClaimer
>
815 UsbDeviceHandleImpl::GetClaimedInterfaceForEndpoint(uint8 endpoint
) {
816 if (ContainsKey(endpoint_map_
, endpoint
))
817 return claimed_interfaces_
[endpoint_map_
[endpoint
].interface_number
];
821 void UsbDeviceHandleImpl::ControlTransferInternal(
822 UsbEndpointDirection direction
,
823 TransferRequestType request_type
,
824 TransferRecipient recipient
,
828 scoped_refptr
<net::IOBuffer
> buffer
,
830 unsigned int timeout
,
831 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
832 const TransferCallback
& callback
) {
833 DCHECK(thread_checker_
.CalledOnValidThread());
836 callback
.Run(USB_TRANSFER_DISCONNECT
, buffer
, 0);
840 if (length
> UINT16_MAX
) {
841 USB_LOG(USER
) << "Transfer too long.";
842 callback
.Run(USB_TRANSFER_ERROR
, buffer
, 0);
846 const size_t resized_length
= LIBUSB_CONTROL_SETUP_SIZE
+ length
;
847 scoped_refptr
<net::IOBuffer
> resized_buffer(
848 new net::IOBufferWithSize(static_cast<int>(resized_length
)));
849 if (!resized_buffer
.get()) {
850 callback
.Run(USB_TRANSFER_ERROR
, buffer
, 0);
853 memcpy(resized_buffer
->data() + LIBUSB_CONTROL_SETUP_SIZE
, buffer
->data(),
856 scoped_ptr
<Transfer
> transfer
= Transfer::CreateControlTransfer(
857 this, CreateRequestType(direction
, request_type
, recipient
), request
,
858 value
, index
, static_cast<uint16
>(length
), resized_buffer
, timeout
,
859 callback_task_runner
, callback
);
861 callback
.Run(USB_TRANSFER_ERROR
, buffer
, 0);
865 SubmitTransfer(transfer
.Pass());
868 void UsbDeviceHandleImpl::BulkTransferInternal(
869 const UsbEndpointDirection direction
,
870 const uint8 endpoint
,
871 scoped_refptr
<net::IOBuffer
> buffer
,
873 const unsigned int timeout
,
874 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
875 const TransferCallback
& callback
) {
876 DCHECK(thread_checker_
.CalledOnValidThread());
879 callback
.Run(USB_TRANSFER_DISCONNECT
, buffer
, 0);
883 if (length
> INT_MAX
) {
884 USB_LOG(USER
) << "Transfer too long.";
885 callback
.Run(USB_TRANSFER_ERROR
, buffer
, 0);
889 scoped_ptr
<Transfer
> transfer
= Transfer::CreateBulkTransfer(
890 this, ConvertTransferDirection(direction
) | endpoint
, buffer
,
891 static_cast<int>(length
), timeout
, callback_task_runner
, callback
);
893 SubmitTransfer(transfer
.Pass());
896 void UsbDeviceHandleImpl::InterruptTransferInternal(
897 UsbEndpointDirection direction
,
899 scoped_refptr
<net::IOBuffer
> buffer
,
901 unsigned int timeout
,
902 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
903 const TransferCallback
& callback
) {
904 DCHECK(thread_checker_
.CalledOnValidThread());
907 callback
.Run(USB_TRANSFER_DISCONNECT
, buffer
, 0);
911 if (length
> INT_MAX
) {
912 USB_LOG(USER
) << "Transfer too long.";
913 callback
.Run(USB_TRANSFER_ERROR
, buffer
, 0);
917 scoped_ptr
<Transfer
> transfer
= Transfer::CreateInterruptTransfer(
918 this, ConvertTransferDirection(direction
) | endpoint
, buffer
,
919 static_cast<int>(length
), timeout
, callback_task_runner
, callback
);
921 SubmitTransfer(transfer
.Pass());
924 void UsbDeviceHandleImpl::IsochronousTransferInternal(
925 const UsbEndpointDirection direction
,
927 scoped_refptr
<net::IOBuffer
> buffer
,
929 unsigned int packets
,
930 unsigned int packet_length
,
931 unsigned int timeout
,
932 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
933 const TransferCallback
& callback
) {
934 DCHECK(thread_checker_
.CalledOnValidThread());
937 callback
.Run(USB_TRANSFER_DISCONNECT
, buffer
, 0);
941 if (length
> INT_MAX
) {
942 USB_LOG(USER
) << "Transfer too long.";
943 callback
.Run(USB_TRANSFER_ERROR
, buffer
, 0);
947 scoped_ptr
<Transfer
> transfer
= Transfer::CreateIsochronousTransfer(
948 this, ConvertTransferDirection(direction
) | endpoint
, buffer
,
949 static_cast<int>(length
), packets
, packet_length
, timeout
,
950 callback_task_runner
, callback
);
952 SubmitTransfer(transfer
.Pass());
955 void UsbDeviceHandleImpl::GenericTransferInternal(
956 UsbEndpointDirection direction
,
958 scoped_refptr
<net::IOBuffer
> buffer
,
960 unsigned int timeout
,
961 scoped_refptr
<base::TaskRunner
> callback_task_runner
,
962 const TransferCallback
& callback
) {
963 if (!ContainsKey(endpoint_map_
, endpoint
)) {
965 << "Failed to do generic transfer since endpoint not in endpoint_map_.";
966 callback
.Run(USB_TRANSFER_ERROR
, buffer
, 0);
970 UsbTransferType transfer_type
= endpoint_map_
[endpoint
].transfer_type
;
971 if (transfer_type
== USB_TRANSFER_BULK
) {
972 BulkTransferInternal(direction
, endpoint
, buffer
, length
, timeout
,
973 task_runner_
, callback
);
974 } else if (transfer_type
== USB_TRANSFER_INTERRUPT
) {
975 InterruptTransferInternal(direction
, endpoint
, buffer
, length
, timeout
,
976 task_runner_
, callback
);
979 << "Failed to do generic transfer since transfer_type is not "
980 "bulk or interrupt.";
981 callback
.Run(USB_TRANSFER_ERROR
, buffer
, 0);
985 void UsbDeviceHandleImpl::SubmitTransfer(scoped_ptr
<Transfer
> transfer
) {
986 DCHECK(thread_checker_
.CalledOnValidThread());
988 // Transfer is owned by libusb until its completion callback is run. This
989 // object holds a weak reference.
990 transfers_
.insert(transfer
.get());
991 blocking_task_runner_
->PostTask(
993 base::Bind(&Transfer::Submit
, base::Unretained(transfer
.release())));
996 void UsbDeviceHandleImpl::TransferComplete(Transfer
* transfer
,
997 const base::Closure
& callback
) {
998 DCHECK(thread_checker_
.CalledOnValidThread());
999 DCHECK(ContainsKey(transfers_
, transfer
)) << "Missing transfer completed";
1000 transfers_
.erase(transfer
);
1002 if (transfer
->callback_task_runner()->RunsTasksOnCurrentThread()) {
1005 transfer
->callback_task_runner()->PostTask(FROM_HERE
, callback
);
1008 // libusb_free_transfer races with libusb_submit_transfer and only work-
1009 // around is to make sure to call them on the same thread.
1010 blocking_task_runner_
->DeleteSoon(FROM_HERE
, transfer
);
1013 void UsbDeviceHandleImpl::InternalClose() {
1014 DCHECK(thread_checker_
.CalledOnValidThread());
1018 // Cancel all the transfers.
1019 for (Transfer
* transfer
: transfers_
) {
1020 // The callback will be called some time later.
1024 // Attempt-release all the interfaces.
1025 // It will be retained until the transfer cancellation is finished.
1026 claimed_interfaces_
.clear();
1028 // Cannot close device handle here. Need to wait for libusb_cancel_transfer to
1033 } // namespace device