#include #include #include #include #include #include //uncomment to DEBUG this file alone //#define DEBUG //"make debug" to get DEBUG msgs on entire program #include "dbg.h" #include "usb_operations.h" libusb_device_handle * open_usb_device( libusb_context *context ) { int rv = 0; //context set to NULL since only acting as single user of libusb //libusb_context *context = NULL; //passed in from main debug("Initalizing libusb"); //initialize libusb must be called prior to any other libusb function //returns 0 on success LIBUSB_ERROR code on failure //int libusb_init ( libusb_context ** context) int usb_init = libusb_init(&context); check( usb_init == LIBUSB_SUCCESS, "Failed to initialize libusb: %s", libusb_strerror(usb_init)); //void libusb_set_debug ( libusb_context * ctx, int level ) libusb_set_debug(context, LIBUSB_LOG_LEVEL_ERROR); //LIBUSB_LOG_LEVEL_NONE (0) : no messages ever printed by the library (default) //LIBUSB_LOG_LEVEL_ERROR (1) : error messages are printed to stderr //LIBUSB_LOG_LEVEL_WARNING (2) : warning and error messages are printed to stderr //LIBUSB_LOG_LEVEL_INFO (3) : informational messages are printed to stdout, // warning and error messages are printed to stderr //LIBUSB_LOG_LEVEL_DEBUG (4) : debug and informational messages are printed to stdout, // warnings and errors to stderr //discover all usb devices libusb_device **device_list = NULL; //ssize_t libusb_get_device_list (libusb_context *ctx, libusb_device ***list) // Returns a list of USB devices currently attached to the system. // return value is number of devices plus one as list is null terminated, or LIBUSB_ERROR if negative. // Must free device list after done with it ssize_t dev_count = libusb_get_device_list( context, &device_list); check( dev_count >= 0, "libusb unable to find any devices: %s", libusb_strerror(dev_count)); ssize_t i = 0; libusb_device *retroprog = NULL; libusb_device *device = NULL; struct libusb_device_descriptor desc; libusb_device_handle *handle = NULL; unsigned char str[256]; debug("searching %d total devices", dev_count-1); for( i=0; i 0) { debug("manf_ascii: %s\n",str); } } if (desc.iProduct) { if ( libusb_get_string_descriptor_ascii( handle, desc.iProduct, str, sizeof(str) ) > 0) { debug("prod_ascii: %s\n",str); } } if (desc.bcdDevice) { debug("bcd Device: %x\n",desc.bcdDevice); //old firmware returns 256, new returns 512 //old firmware returns 0x100, new returns 0x200 //USB_CFG_DEVICE_VERSION 0x00, 0x01 for v1.0, 0x00, 0x02 for v2.0 (minor then major) //v2.0 released late 2016 } //TODO verify it's INL Retro Prog by InfiniteNesLives.com, then break //else close handle and keep searching as there might be other devices with matching prod/devIDs //also verify firmware version is compatible // //TODO verify have permission to interface with device, else give notice to user break; } } check( retroprog != NULL, "Could not find INL retro-prog USB device"); //free device list now that INL retro-prog was found and opened //void libusb_free_device_list ( libusb_device ** list, int unref_devices ) libusb_free_device_list( device_list, 1); //don't completely understand the unref_devices = 1... device_list = NULL; //denote that device list is free if end up in error //Guess this is what you're supposed to do.. // the process of opening a device can be viewed as follows: // // Discover devices using libusb_get_device_list(). // Choose the device that you want to operate, and call libusb_open(). // Unref all devices in the discovered device list. // Free the discovered device list. // // The order is important - you must not unreference the device before attempting to open it, because unreferencing it may destroy the device. // // For convenience, the libusb_free_device_list() function includes a parameter to optionally unreference all the devices in the list before freeing the list itself. This combines steps 3 and 4 above. // // As an implementation detail, libusb_open() actually adds a reference to the device in question. This is because the device remains available through the handle via libusb_get_device(). The reference is deleted during libusb_close(). //free device list if it was left open if (device_list) { printf("Whoops! freeing device list\n"); libusb_free_device_list( device_list, 1); } return handle; error: printf("open_usb_device went to error\n"); if (device_list) { printf("freeing device list\n"); libusb_free_device_list( device_list, 1); } if (handle) { printf("closing usb device\n"); libusb_close(handle); } if (usb_init == LIBUSB_SUCCESS) { printf("deinitializing libusb\n"); libusb_exit(context); } if (rv == LIBUSB_ERROR_ACCESS) { printf("-------------------------------------------------------\n"); printf("Denied Permission is expected for initial use on Linux.\n"); printf("See udev-rule-help/Readme.txt in host dir for help gaining permission.\n"); printf("Try command with sudo for a cheap temporary solution.\n"); } return NULL; } void close_usb(libusb_context *context, libusb_device_handle *handle) { //must close device before exiting libusb_close(handle); handle = NULL; //delete handle reference so error won't retry closing //deinitialize libusb to be called after closing all devices and before teminating application libusb_exit(context); return; }