Replace libusb with hidapi on macOS
macOS won't let libusb access the RF2500 HID device, but hidapi does.
This commit is contained in:
parent
587dbf1516
commit
95dd38b52a
14
Makefile
14
Makefile
|
@ -39,6 +39,7 @@ else
|
||||||
endif
|
endif
|
||||||
|
|
||||||
BSLHID_OBJ ?= transport/bslhid.o
|
BSLHID_OBJ ?= transport/bslhid.o
|
||||||
|
RF25000_OBJ ?= transport/rf2500.o
|
||||||
|
|
||||||
ifeq ($(OS),Windows_NT)
|
ifeq ($(OS),Windows_NT)
|
||||||
MSPDEBUG_CC = $(CC)
|
MSPDEBUG_CC = $(CC)
|
||||||
|
@ -66,13 +67,18 @@ else
|
||||||
|
|
||||||
ifeq ($(UNAME_S),Darwin) # Mac OS X/MacPorts stuff
|
ifeq ($(UNAME_S),Darwin) # Mac OS X/MacPorts stuff
|
||||||
ifeq ($(shell fink -V > /dev/null 2>&1 && echo ok),ok)
|
ifeq ($(shell fink -V > /dev/null 2>&1 && echo ok),ok)
|
||||||
PORTS_CFLAGS := $(shell pkg-config --cflags libusb)
|
PORTS_CFLAGS := $(shell pkg-config --cflags hidapi libusb)
|
||||||
PORTS_LDFLAGS := $(shell pkg-config --libs libusb) -ltermcap -pthread
|
PORTS_LDFLAGS := $(shell pkg-config --libs hidapi libusb) -ltermcap -pthread
|
||||||
|
else ifeq ($(shell brew --version > /dev/null 2>&1 && echo ok),ok)
|
||||||
|
PORTS_CFLAGS := $(shell pkg-config --cflags hidapi)
|
||||||
|
PORTS_LDFLAGS := $(shell pkg-config --libs hidapi) -framework IOKit -framework CoreFoundation
|
||||||
else
|
else
|
||||||
PORTS_CFLAGS := -I/opt/local/include
|
PORTS_CFLAGS := -I/opt/local/include
|
||||||
PORTS_LDFLAGS := -L/opt/local/lib -framework IOKit -framework CoreFoundation
|
PORTS_LDFLAGS := -L/opt/local/lib -lhidapi -framework IOKit -framework CoreFoundation
|
||||||
endif
|
endif
|
||||||
BSLHID_OBJ = transport/bslosx.o
|
BSLHID_OBJ = transport/bslosx.o
|
||||||
|
RF25000_OBJ = transport/rf2500hidapi.o
|
||||||
|
LDFLAGS =
|
||||||
else ifneq ($(filter $(UNAME_S),OpenBSD NetBSD DragonFly),)
|
else ifneq ($(filter $(UNAME_S),OpenBSD NetBSD DragonFly),)
|
||||||
PORTS_CFLAGS := $(shell pkg-config --cflags libusb)
|
PORTS_CFLAGS := $(shell pkg-config --cflags libusb)
|
||||||
PORTS_LDFLAGS := $(shell pkg-config --libs libusb) -ltermcap -pthread
|
PORTS_LDFLAGS := $(shell pkg-config --libs libusb) -ltermcap -pthread
|
||||||
|
@ -150,10 +156,10 @@ OBJ=\
|
||||||
transport/cp210x.o \
|
transport/cp210x.o \
|
||||||
transport/cdc_acm.o \
|
transport/cdc_acm.o \
|
||||||
transport/ftdi.o \
|
transport/ftdi.o \
|
||||||
transport/rf2500.o \
|
|
||||||
transport/ti3410.o \
|
transport/ti3410.o \
|
||||||
transport/comport.o \
|
transport/comport.o \
|
||||||
$(BSLHID_OBJ) \
|
$(BSLHID_OBJ) \
|
||||||
|
$(RF25000_OBJ) \
|
||||||
drivers/device.o \
|
drivers/device.o \
|
||||||
drivers/bsl.o \
|
drivers/bsl.o \
|
||||||
drivers/fet.o \
|
drivers/fet.o \
|
||||||
|
|
|
@ -19,11 +19,7 @@
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#ifndef __Windows__
|
#include <hidapi.h>
|
||||||
#include <usb.h>
|
|
||||||
#else
|
|
||||||
#include <lusb0_usb.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "rf2500.h"
|
#include "rf2500.h"
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
|
@ -34,7 +30,7 @@ struct rf2500_transport {
|
||||||
struct transport base;
|
struct transport base;
|
||||||
|
|
||||||
int int_number;
|
int int_number;
|
||||||
struct usb_dev_handle *handle;
|
hid_device *handle;
|
||||||
|
|
||||||
uint8_t buf[64];
|
uint8_t buf[64];
|
||||||
int len;
|
int len;
|
||||||
|
@ -59,60 +55,6 @@ struct rf2500_transport {
|
||||||
#define USB_FET_IN_EP 0x81
|
#define USB_FET_IN_EP 0x81
|
||||||
#define USB_FET_OUT_EP 0x01
|
#define USB_FET_OUT_EP 0x01
|
||||||
|
|
||||||
static int open_interface(struct rf2500_transport *tr,
|
|
||||||
struct usb_device *dev, int ino)
|
|
||||||
{
|
|
||||||
printc_dbg("Trying to open interface %d on %s\n", ino, dev->filename);
|
|
||||||
|
|
||||||
tr->int_number = ino;
|
|
||||||
|
|
||||||
tr->handle = usb_open(dev);
|
|
||||||
if (!tr->handle) {
|
|
||||||
pr_error("rf2500: can't open device");
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if defined(__linux__)
|
|
||||||
if (usb_detach_kernel_driver_np(tr->handle, tr->int_number) < 0)
|
|
||||||
pr_error("rf2500: warning: can't "
|
|
||||||
"detach kernel driver");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef __Windows__
|
|
||||||
if (usb_set_configuration(tr->handle, 1) < 0) {
|
|
||||||
pr_error("rf2500: can't set configuration 1");
|
|
||||||
usb_close(tr->handle);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (usb_claim_interface(tr->handle, tr->int_number) < 0) {
|
|
||||||
pr_error("rf2500: can't claim interface");
|
|
||||||
usb_close(tr->handle);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int open_device(struct rf2500_transport *tr,
|
|
||||||
struct usb_device *dev)
|
|
||||||
{
|
|
||||||
struct usb_config_descriptor *c = &dev->config[0];
|
|
||||||
int i;
|
|
||||||
|
|
||||||
for (i = 0; i < c->bNumInterfaces; i++) {
|
|
||||||
struct usb_interface *intf = &c->interface[i];
|
|
||||||
struct usb_interface_descriptor *desc = &intf->altsetting[0];
|
|
||||||
|
|
||||||
if (desc->bInterfaceClass == USB_FET_INTERFACE_CLASS &&
|
|
||||||
!open_interface(tr, dev, desc->bInterfaceNumber))
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int usbtr_send(transport_t tr_base, const uint8_t *data, int len)
|
static int usbtr_send(transport_t tr_base, const uint8_t *data, int len)
|
||||||
{
|
{
|
||||||
struct rf2500_transport *tr = (struct rf2500_transport *)tr_base;
|
struct rf2500_transport *tr = (struct rf2500_transport *)tr_base;
|
||||||
|
@ -138,8 +80,8 @@ static int usbtr_send(transport_t tr_base, const uint8_t *data, int len)
|
||||||
#ifdef DEBUG_USBTR
|
#ifdef DEBUG_USBTR
|
||||||
debug_hexdump("USB transfer out", pbuf, txlen);
|
debug_hexdump("USB transfer out", pbuf, txlen);
|
||||||
#endif
|
#endif
|
||||||
if (usb_bulk_write(tr->handle, USB_FET_OUT_EP,
|
if (hid_write(tr->handle,
|
||||||
(char *)pbuf, txlen, 10000) < 0) {
|
(const unsigned char *)pbuf, txlen) < 0) {
|
||||||
pr_error("rf2500: can't send data");
|
pr_error("rf2500: can't send data");
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
@ -157,9 +99,8 @@ static int usbtr_recv(transport_t tr_base, uint8_t *databuf, int max_len)
|
||||||
int rlen;
|
int rlen;
|
||||||
|
|
||||||
if (tr->offset >= tr->len) {
|
if (tr->offset >= tr->len) {
|
||||||
if (usb_bulk_read(tr->handle, USB_FET_IN_EP,
|
if (hid_read_timeout(tr->handle, (unsigned char *)tr->buf,
|
||||||
(char *)tr->buf, sizeof(tr->buf),
|
sizeof(tr->buf), 10000) < 0) {
|
||||||
10000) < 0) {
|
|
||||||
pr_error("rf2500: can't receive data");
|
pr_error("rf2500: can't receive data");
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
@ -187,27 +128,23 @@ static void usbtr_destroy(transport_t tr_base)
|
||||||
{
|
{
|
||||||
struct rf2500_transport *tr = (struct rf2500_transport *)tr_base;
|
struct rf2500_transport *tr = (struct rf2500_transport *)tr_base;
|
||||||
|
|
||||||
usb_release_interface(tr->handle, tr->int_number);
|
hid_close(tr->handle);
|
||||||
usb_close(tr->handle);
|
|
||||||
free(tr);
|
free(tr);
|
||||||
|
hid_exit();
|
||||||
}
|
}
|
||||||
|
|
||||||
static int usbtr_flush(transport_t tr_base)
|
static int usbtr_flush(transport_t tr_base)
|
||||||
{
|
{
|
||||||
struct rf2500_transport *tr = (struct rf2500_transport *)tr_base;
|
struct rf2500_transport *tr = (struct rf2500_transport *)tr_base;
|
||||||
|
|
||||||
#if !defined(__APPLE__) && !defined(__sun__)
|
unsigned char buf[64];
|
||||||
char buf[64];
|
|
||||||
|
|
||||||
/* Flush out lingering data.
|
/* Flush out lingering data.
|
||||||
*
|
*
|
||||||
* The timeout apparently doesn't work on OS/X, and this loop
|
* The timeout apparently doesn't work on OS/X, and this loop
|
||||||
* just hangs once the endpoint buffer empties.
|
* just hangs once the endpoint buffer empties.
|
||||||
*/
|
*/
|
||||||
while (usb_bulk_read(tr->handle, USB_FET_IN_EP,
|
while (hid_read_timeout(tr->handle, buf, sizeof(buf), 100) > 0);
|
||||||
buf, sizeof(buf),
|
|
||||||
100) > 0);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
tr->len = 0;
|
tr->len = 0;
|
||||||
tr->offset = 0;
|
tr->offset = 0;
|
||||||
|
@ -231,7 +168,7 @@ static const struct transport_class rf2500_transport = {
|
||||||
transport_t rf2500_open(const char *devpath, const char *requested_serial)
|
transport_t rf2500_open(const char *devpath, const char *requested_serial)
|
||||||
{
|
{
|
||||||
struct rf2500_transport *tr = malloc(sizeof(*tr));
|
struct rf2500_transport *tr = malloc(sizeof(*tr));
|
||||||
struct usb_device *dev;
|
hid_device *handle;
|
||||||
|
|
||||||
if (!tr) {
|
if (!tr) {
|
||||||
pr_error("rf2500: can't allocate memory");
|
pr_error("rf2500: can't allocate memory");
|
||||||
|
@ -241,27 +178,23 @@ transport_t rf2500_open(const char *devpath, const char *requested_serial)
|
||||||
memset(tr, 0, sizeof(*tr));
|
memset(tr, 0, sizeof(*tr));
|
||||||
tr->base.ops = &rf2500_transport;
|
tr->base.ops = &rf2500_transport;
|
||||||
|
|
||||||
usb_init();
|
hid_init();
|
||||||
usb_find_busses();
|
|
||||||
usb_find_devices();
|
|
||||||
|
|
||||||
if (devpath)
|
if (devpath)
|
||||||
dev = usbutil_find_by_loc(devpath);
|
handle = hid_open_path(devpath);
|
||||||
else
|
else
|
||||||
dev = usbutil_find_by_id(USB_FET_VENDOR, USB_FET_PRODUCT,
|
handle = hid_open(USB_FET_VENDOR, USB_FET_PRODUCT,
|
||||||
requested_serial);
|
(wchar_t *)requested_serial);
|
||||||
|
|
||||||
if (!dev) {
|
if (!handle) {
|
||||||
free(tr);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (open_device(tr, dev) < 0) {
|
|
||||||
printc_err("rf2500: failed to open RF2500 device\n");
|
printc_err("rf2500: failed to open RF2500 device\n");
|
||||||
free(tr);
|
free(tr);
|
||||||
|
hid_exit();
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tr->handle = handle;
|
||||||
|
|
||||||
usbtr_flush(&tr->base);
|
usbtr_flush(&tr->base);
|
||||||
|
|
||||||
return (transport_t)tr;
|
return (transport_t)tr;
|
||||||
|
|
Loading…
Reference in New Issue