iso.go 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. // Copyright 2013 Google Inc. All rights reserved.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package usb
  15. /*
  16. #include "libusb.h"
  17. int submit(struct libusb_transfer *xfer);
  18. void print_xfer(struct libusb_transfer *xfer);
  19. int extract_data(struct libusb_transfer *xfer, void *data, int max, unsigned char *status);
  20. */
  21. import "C"
  22. import (
  23. "fmt"
  24. "log"
  25. "time"
  26. "unsafe"
  27. )
  28. //export iso_callback
  29. func iso_callback(cptr unsafe.Pointer) {
  30. ch := *(*chan struct{})(cptr)
  31. close(ch)
  32. }
  33. func (end *endpoint) allocTransfer() *Transfer {
  34. // Use libusb_get_max_iso_packet_size ?
  35. const (
  36. iso_packets = 8 // 128 // 242
  37. packet_size = 2 * 960 // 1760
  38. )
  39. xfer := C.libusb_alloc_transfer(C.int(iso_packets))
  40. if xfer == nil {
  41. log.Printf("usb: transfer allocation failed?!")
  42. return nil
  43. }
  44. buf := make([]byte, iso_packets*packet_size)
  45. done := make(chan struct{}, 1)
  46. xfer.dev_handle = end.Device.handle
  47. xfer.endpoint = C.uchar(end.Address)
  48. xfer._type = C.LIBUSB_TRANSFER_TYPE_ISOCHRONOUS
  49. xfer.buffer = (*C.uchar)((unsafe.Pointer)(&buf[0]))
  50. xfer.length = C.int(len(buf))
  51. xfer.num_iso_packets = iso_packets
  52. C.libusb_set_iso_packet_lengths(xfer, packet_size)
  53. /*
  54. pkts := *(*[]C.struct_libusb_packet_descriptor)(unsafe.Pointer(&reflect.SliceHeader{
  55. Data: uintptr(unsafe.Pointer(&xfer.iso_packet_desc)),
  56. Len: iso_packets,
  57. Cap: iso_packets,
  58. }))
  59. */
  60. t := &Transfer{
  61. xfer: xfer,
  62. done: done,
  63. buf: buf,
  64. }
  65. xfer.user_data = (unsafe.Pointer)(&t.done)
  66. return t
  67. }
  68. type Transfer struct {
  69. xfer *C.struct_libusb_transfer
  70. pkts []*C.struct_libusb_packet_descriptor
  71. done chan struct{}
  72. buf []byte
  73. }
  74. func (t *Transfer) Submit(timeout time.Duration) error {
  75. //log.Printf("iso: submitting %#v", t.xfer)
  76. t.xfer.timeout = C.uint(timeout / time.Millisecond)
  77. if errno := C.submit(t.xfer); errno < 0 {
  78. return usbError(errno)
  79. }
  80. return nil
  81. }
  82. func (t *Transfer) Wait(b []byte) (n int, err error) {
  83. select {
  84. case <-time.After(10 * time.Second):
  85. return 0, fmt.Errorf("wait timed out after 10s")
  86. case <-t.done:
  87. }
  88. // Non-iso transfers:
  89. //n = int(t.xfer.actual_length)
  90. //copy(b, ((*[1 << 16]byte)(unsafe.Pointer(t.xfer.buffer)))[:n])
  91. //C.print_xfer(t.xfer)
  92. /*
  93. buf, offset := ((*[1 << 16]byte)(unsafe.Pointer(t.xfer.buffer))), 0
  94. for i, pkt := range *t.pkts {
  95. log.Printf("Type is %T", t.pkts)
  96. n += copy(b[n:], buf[offset:][:pkt.actual_length])
  97. offset += pkt.Length
  98. if pkt.status != 0 && err == nil {
  99. err = error(TransferStatus(pkt.status))
  100. }
  101. }
  102. */
  103. var status uint8
  104. n = int(C.extract_data(t.xfer, unsafe.Pointer(&b[0]), C.int(len(b)), (*C.uchar)(unsafe.Pointer(&status))))
  105. if status != 0 {
  106. err = TransferStatus(status)
  107. }
  108. return n, err
  109. }
  110. func (t *Transfer) Close() error {
  111. C.libusb_free_transfer(t.xfer)
  112. return nil
  113. }
  114. func isochronous_xfer(e *endpoint, buf []byte, timeout time.Duration) (int, error) {
  115. t := e.allocTransfer()
  116. defer t.Close()
  117. if err := t.Submit(timeout); err != nil {
  118. log.Printf("iso: xfer failed to submit: %s", err)
  119. return 0, err
  120. }
  121. n, err := t.Wait(buf)
  122. if err != nil {
  123. log.Printf("iso: xfer failed: %s", err)
  124. return 0, err
  125. }
  126. return n, err
  127. }