From bf947e7f571eb9274f230d2d114b4ae4a245a1b7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?J=C3=A9r=C3=A9my=20Zurcher?= Date: Wed, 2 Mar 2011 12:53:47 +0100 Subject: use libusb-1.0 API --- CMakeLists.txt | 23 ++++++- libk8055/CMakeLists.txt | 7 ++- libk8055/k8055.h | 3 +- libk8055/libk8055.c | 159 ++++++++++++++++++++++++++---------------------- pyk8055/CMakeLists.txt | 7 ++- 5 files changed, 118 insertions(+), 81 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 451ffb4..c311b02 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -40,12 +40,29 @@ endfunction() # INCLUDE(CheckIncludeFiles) INCLUDE(CheckLibraryExists) -CHECK_INCLUDE_FILES(usb.h HAVE_USB_H) +if(NOT WIN32) + FIND_PACKAGE(PkgConfig) +endif() +if(PKG_CONFIG_FOUND) + pkg_check_modules(LIBUSB libusb) +else() + MESSAGE(STATUS "Checking for module 'libusb'") +endif() +find_path(LIBUSB_INCLUDE_DIR + NAMES libusb.h + PATHS /usr/include/libusb-1.0 /usr/local/include/libusb-1.0 +) +find_library(LIBUSB_LDFLAGS + NAMES libusb +) +include(FindPackageHandleStandardArgs) +FIND_PACKAGE_HANDLE_STANDARD_ARGS(LIBUSB DEFAULT_MSG LIBUSB_LIBRARIES LIBUSB_INCLUDE_DIR) +mark_as_advanced(LIBUSB_INCLUDE_DIR LIBUSB_LIBRARIES) CHECK_INCLUDE_FILES(string.h HAVE_STRING_H) CHECK_INCLUDE_FILES(stdio.h HAVE_STDIO_H) CHECK_INCLUDE_FILES(assert.h HAVE_ASSERT_H) CHECK_INCLUDE_FILES(math.h HAVE_MATH_H) -IF (HAVE_USB_H) +IF (LIBUSB_FOUND) ADD_SUBDIRECTORY(libk8055) ENDIF() # @@ -65,7 +82,7 @@ ENDIF() # BUILD SUMMARY # COLOR_MESSAGE("\n${_escape}[1;36mBuild Summary${_escape}[0m\n--------------------") -IF (HAVE_USB_H) +IF (LIBUSB_FOUND) COLOR_MESSAGE("* ${_escape}[1;35mlibusb${_escape}[0m : ${_escape}[1;32mfound${_escape}[0m build libk8055") ELSE() COLOR_MESSAGE("* ${_escape}[1;35mlibusb${_escape}[0m : ${_escape}[1;31mmissing${_escape}[0m unable to build libk8055") diff --git a/libk8055/CMakeLists.txt b/libk8055/CMakeLists.txt index 75466ff..cb7df5a 100644 --- a/libk8055/CMakeLists.txt +++ b/libk8055/CMakeLists.txt @@ -1,9 +1,12 @@ ADD_LIBRARY( k8055-${VERSION} SHARED libk8055.c ) ADD_LIBRARY( k8055-static-${VERSION} STATIC libk8055.c ) -INCLUDE_DIRECTORIES( ${CMAKE_SOURCE_DIR} ) +INCLUDE_DIRECTORIES( + ${CMAKE_SOURCE_DIR} + ${LIBUSB_INCLUDE_DIR} +) TARGET_LINK_LIBRARIES( k8055-${VERSION} - usb + ${LIBUSB_LDFLAGS} m ) SET_TARGET_PROPERTIES(k8055-${VERSION} PROPERTIES PUBLIC_HEADER k8055.h ) diff --git a/libk8055/k8055.h b/libk8055/k8055.h index b63f51f..69defbd 100644 --- a/libk8055/k8055.h +++ b/libk8055/k8055.h @@ -37,7 +37,8 @@ extern "C" { struct k8055_dev { int dev_no; - struct usb_dev_handle* device_handle; + struct libusb_context *usb_ctx; + struct libusb_device_handle *usb_handle; unsigned char data_in[PACKET_LEN+1]; unsigned char data_out[PACKET_LEN+1]; }; diff --git a/libk8055/libk8055.c b/libk8055/libk8055.c index ad97891..40047d7 100644 --- a/libk8055/libk8055.c +++ b/libk8055/libk8055.c @@ -81,9 +81,9 @@ #include #include -#include #include #include +#include #include "k8055.h" #define STR_BUFF 256 @@ -113,9 +113,9 @@ static int debug = 0; /* Actual read of data from the device endpoint, retry READ_RETRY times if not responding ok */ int k8055_read( struct k8055_dev* dev ) { if( dev->dev_no==0 ) return K8055_ERROR; + int length; for( int i=0; idevice_handle, USB_INP_EP, ( char* )dev->data_in, PACKET_LEN, USB_TIMEOUT ); - if( ( read_status==PACKET_LEN ) && ( dev->data_in[1]==dev->dev_no ) ) return 0; + if( libusb_interrupt_transfer( dev->usb_handle, USB_INP_EP, dev->data_in, PACKET_LEN, &length, USB_TIMEOUT )==0 && ( length==PACKET_LEN ) && ( dev->data_in[1]==dev->dev_no ) ) return 0; if( debug ) fprintf( stderr, "k8055 read retry\n" ); } return K8055_ERROR; @@ -124,9 +124,9 @@ int k8055_read( struct k8055_dev* dev ) { /* Actual write of data to the device endpont, retry WRITE_RETRY times if not reponding correctly */ int k8055_write( struct k8055_dev* dev ) { if( dev->dev_no == 0 ) return K8055_ERROR; + int length; for( int i=0; idevice_handle, USB_OUT_EP, ( char* )dev->data_out, PACKET_LEN, USB_TIMEOUT ); - if( write_status==PACKET_LEN ) return 0; + if( libusb_interrupt_transfer( dev->usb_handle, USB_OUT_EP, dev->data_out, PACKET_LEN, &length, USB_TIMEOUT )==0 && length==PACKET_LEN ) return 0; if( debug ) fprintf( stderr, "k8055 write retry\n" ); } return K8055_ERROR; @@ -169,28 +169,19 @@ int k8055_counter_2( struct k8055_dev* dev ) { } /* If device is owned by some kernel driver, try to disconnect it and claim the device*/ -static int k8055_takeover_device( usb_dev_handle* udev, int interface ) { +static int k8055_takeover_device( libusb_device_handle* handle, int interface ) { char driver_name[STR_BUFF]; memset( driver_name, 0, STR_BUFF ); int ret = K8055_ERROR; - assert( udev != NULL ); - if( usb_get_driver_np( udev, interface, driver_name, sizeof( driver_name ) )==0 ) { - if( debug ) fprintf( stderr, "usb_get_driver_np success: %s\n", driver_name ); - if( usb_detach_kernel_driver_np( udev, interface )==0 ) { - if( debug ) fprintf( stderr, "usb_detach_kernel_driver_np success" ); - } else { - if( debug ) fprintf( stderr, "usb_detach_kernel_driver_np failure : %s\n", usb_strerror() ); - } - } else { - if( debug ) fprintf( stderr, "usb_get_driver_np failure : %s\n", usb_strerror() ); + assert( handle != NULL ); + if( libusb_detach_kernel_driver( handle, interface )!=0 ) { + if( debug ) fprintf( stderr, "usb_detach_kernel_driver failure\n" ); } - if ( usb_claim_interface( udev, interface )==0 ) { - usb_set_altinterface( udev, interface ); - } else { - if( debug ) fprintf( stderr, "usb_claim_interface failure: %s\n", usb_strerror() ); + if ( libusb_claim_interface( handle, interface )!=0 ) { + if( debug ) fprintf( stderr, "usb_claim_interface failure\n" ); return K8055_ERROR; } - usb_set_configuration( udev, 1 ); + libusb_set_configuration( handle, 1 ); if ( debug ) fprintf( stderr, "Found interface %d, took over the device\n", interface ); return 0; } @@ -208,43 +199,62 @@ char* k8055_version( void ) { } int k8055_open_device( struct k8055_dev* dev, int board_address ) { - usb_init(); - usb_find_busses(); - usb_find_devices(); int ipid = K8055_IPID + board_address; - struct usb_bus* busses = usb_get_busses(); - for( struct usb_bus* bus=busses; bus; bus=bus->next ) { - for( struct usb_device* usb_dev=bus->devices; usb_dev; usb_dev=usb_dev->next ) { - if( ( usb_dev->descriptor.idVendor==VELLEMAN_VENDOR_ID ) && ( usb_dev->descriptor.idProduct==ipid ) ) { - dev->dev_no = 0; - dev->device_handle = usb_open( usb_dev ); - if( dev->device_handle==NULL ) { - if( debug ) fprintf( stderr,"usb_open failure : %s\n", usb_strerror() ); - return K8055_ERROR; - } - if( debug ) fprintf( stderr, "Velleman Device Found @ Address %s Vendor 0x0%x Product ID 0x0%x\n", usb_dev->filename, usb_dev->descriptor.idVendor, usb_dev->descriptor.idProduct ); - if( k8055_takeover_device( dev->device_handle, 0 )<0 ) { - if( debug ) fprintf( stderr, "Can not take over the device from the OS driver\n" ); - usb_close( dev->device_handle ); - dev->device_handle = NULL; - return K8055_ERROR; - } else { - memset( dev->data_out,0,PACKET_LEN ); - dev->dev_no = board_address + 1; - dev->data_out[CMD_OFFSET] = CMD_RESET; - k8055_write( dev ); - if ( k8055_read( dev )==0 ) { - if( debug ) fprintf( stderr, "Device %d ready\n", board_address ); - return board_address; - } else { - if( debug ) fprintf( stderr, "Device %d not ready\n", board_address ); - dev->dev_no = 0; - usb_close( dev->device_handle ); - dev->device_handle = NULL; - return K8055_ERROR; - } - } + libusb_device** list; + libusb_device* found = NULL; + libusb_init( &dev->usb_ctx ); + ssize_t cnt = libusb_get_device_list( NULL, &list ); + if( cnt<0 ) { + if( debug ) fprintf( stderr, "Unable to list usb devices\n" ); + return K8055_ERROR; + } + for ( ssize_t i=0; ifilename + if( debug ) fprintf( stderr, "Velleman Device Found @ Address %s Vendor 0x0%x Product ID 0x0%x\n", "002", usb_descr.idVendor, usb_descr.idProduct ); + found = usb_dev; + break; } + } else { + if( debug ) fprintf( stderr, "USB device descriptor unaccessible.\n" ); + } + } + if( found==NULL ) { + if( debug ) fprintf( stderr, "No Velleman device found.\n" ); + return K8055_ERROR; + } + dev->dev_no = 0; + dev->usb_handle = NULL; + if( libusb_open( found , &dev->usb_handle )!=0 ) { + if( debug ) fprintf( stderr,"usb_open failure\n" ); + return K8055_ERROR; + } + if( k8055_takeover_device( dev->usb_handle, 0 )!=0 ) { + if( debug ) fprintf( stderr, "Can not take over the device from the OS driver\n" ); + libusb_release_interface( dev->usb_handle, 0 ); + libusb_close( dev->usb_handle ); + libusb_exit( dev->usb_ctx ); + dev->usb_handle = NULL; + return K8055_ERROR; + } else { + memset( dev->data_out,0,PACKET_LEN ); + dev->dev_no = board_address + 1; + dev->data_out[CMD_OFFSET] = CMD_RESET; + k8055_write( dev ); + if ( k8055_read( dev )==0 ) { + if( debug ) fprintf( stderr, "Device %d ready\n", board_address ); + return board_address; + } else { + if( debug ) fprintf( stderr, "Device %d not ready\n", board_address ); + libusb_release_interface( dev->usb_handle, 0 ); + libusb_close( dev->usb_handle ); + libusb_exit( dev->usb_ctx ); + dev->dev_no = 0; + dev->usb_handle = NULL; + return K8055_ERROR; } } if( debug ) fprintf( stderr, "Could not find Velleman k8055 with address %d\n", board_address ); @@ -254,39 +264,42 @@ int k8055_open_device( struct k8055_dev* dev, int board_address ) { int k8055_close_device( struct k8055_dev* dev ) { if ( dev->dev_no == 0 ) { if ( debug ) fprintf( stderr, "Current device is not open\n" ); - return 0; - } - if( dev->device_handle==NULL ) { + } else if( dev->usb_handle==NULL ) { if ( debug ) fprintf( stderr, "Current device is marked as open, but device hanlde is NULL\n" ); dev->dev_no = 0; - return 0; - } - int rc = usb_close( dev->device_handle ); - if ( rc >= 0 ) { + } else { + if( libusb_release_interface( dev->usb_handle, 0 )!= 0 ) { + if( debug ) fprintf( stderr,"libusb_realese_interface failure.\n" ); + } + libusb_close( dev->usb_handle ); dev->dev_no = 0; - dev->device_handle = NULL; + dev->usb_handle = NULL; } - return rc; + return 0; } int k8055_search_devices( void ) { int ret = 0; - usb_init(); - usb_find_busses(); - usb_find_devices(); - struct usb_bus* busses = usb_get_busses(); - for ( struct usb_bus* bus = busses; bus; bus = bus->next ) { - for( struct usb_device* usb_dev=bus->devices; usb_dev; usb_dev=usb_dev->next ) { - if ( usb_dev->descriptor.idVendor == VELLEMAN_VENDOR_ID ) { + struct libusb_context* usb_ctx; + libusb_device** list; + struct libusb_device_descriptor usb_descr; + libusb_init( &usb_ctx ); + ssize_t cnt = libusb_get_device_list( NULL, &list ); + for ( ssize_t i=0; idescriptor.idProduct == K8055_IPID + 0 ) ret |= 0x01; if( usb_dev->descriptor.idProduct == K8055_IPID + 1 ) ret |= 0x02; if( usb_dev->descriptor.idProduct == K8055_IPID + 2 ) ret |= 0x04; if( usb_dev->descriptor.idProduct == K8055_IPID + 3 ) ret |= 0x08; /* else some other kind of Velleman board */ } + } else { + if( debug ) fprintf( stderr, "USB device descriptor unaccessible.\n" ); } } - if( debug ) fprintf( stderr,"found devices : %X\n",ret ); + if( debug ) fprintf( stderr,"found devices : %X\n", ret ); + libusb_exit( usb_ctx ); return ret; } diff --git a/pyk8055/CMakeLists.txt b/pyk8055/CMakeLists.txt index 33aa210..ebc8d6b 100644 --- a/pyk8055/CMakeLists.txt +++ b/pyk8055/CMakeLists.txt @@ -1,7 +1,10 @@ INCLUDE(${SWIG_USE_FILE}) -INCLUDE_DIRECTORIES(${PYTHON_INCLUDE_PATH}) -INCLUDE_DIRECTORIES(${CMAKE_CURRENT_SOURCE_DIR}) +INCLUDE_DIRECTORIES( + ${PYTHON_INCLUDE_PATH} + ${CMAKE_CURRENT_SOURCE_DIR} + ${LIBUSB_INCLUDE_DIR} + ) SET(CMAKE_SWIG_FLAGS "") -- cgit v1.1-2-g2b99