/* --------------------------------------------------------------------------*/ /* */ /* Pd interface to the USB HID API using libhid, which is built on libusb */ /* Written by Hans-Christoph Steiner */ /* */ /* Copyright (c) 2004 Hans-Christoph Steiner */ /* */ /* This program is free software; you can redistribute it and/or */ /* modify it under the terms of the GNU General Public License */ /* as published by the Free Software Foundation; either version 2 */ /* of the License, or (at your option) any later version. */ /* */ /* See file LICENSE for further informations on licensing terms. */ /* */ /* This program is distributed in the hope that it will be useful, */ /* but WITHOUT ANY WARRANTY; without even the implied warranty of */ /* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */ /* GNU General Public License for more details. */ /* */ /* You should have received a copy of the GNU General Public License */ /* along with this program; if not, write to the Free Software Foundation, */ /* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /* */ /* --------------------------------------------------------------------------*/ #include #include #include #include #include "m_pd.h" /*------------------------------------------------------------------------------ * INCLUDE HACK */ /* NOTE: included from libusb/usbi.h. UGLY, i know, but so is libusb! * this struct doesn't seem to be defined in usb.h, only prototyped */ struct usb_dev_handle { int fd; struct usb_bus *bus; struct usb_device *device; int config; int interface; int altsetting; void *impl_info; }; /*------------------------------------------------------------------------------ * GLOBAL VARIABLES */ /* * count the number of instances of this object so that certain free() * functions can be called only after the final instance is detroyed. */ t_int usbhid_instance_count; #define HID_ID_MAX 32 char *hid_id[HID_ID_MAX]; /* FIXME: 32 devices MAX */ t_int hid_id_count; /*------------------------------------------------------------------------------ * CLASS DEF */ typedef struct _usbhid { t_object x_obj; /* usbhid types */ HIDInterface *x_hidinterface; u_int8_t x_iSerialNumber; hid_return x_hid_return; /* internal state */ t_int x_device_number; t_int x_read_element_count; t_int *x_read_elements; t_int x_write_element_count; t_int *x_write_elements; /* clock support */ t_clock *x_clock; t_int x_delay; t_int x_started; /* outlets */ t_atom *output; t_int output_count; t_outlet *x_data_outlet; t_outlet *x_control_outlet; } t_usbhid; /*------------------------------------------------------------------------------ * LOCAL DEFINES */ #define USBHID_MAJOR_VERSION 0 #define USBHID_MINOR_VERSION 0 //#define DEBUG(x) #define DEBUG(x) x static t_class *usbhid_class; /* for USB strings */ #define STRING_BUFFER_LENGTH 256 #define SEND_PACKET_LENGTH 1 #define RECEIVE_PACKET_LENGTH 6 #define PATH_LENGTH 3 /*------------------------------------------------------------------------------ * SUPPORT FUNCTIONS */ /* add one new atom to the list to be outputted */ static void add_atom_to_output(t_usbhid *x, t_atom *new_atom) { t_atom *new_atom_list; new_atom_list = (t_atom *)getbytes((x->output_count + 1) * sizeof(t_atom)); memcpy(new_atom_list, x->output, x->output_count * sizeof(t_atom)); freebytes(x->output, x->output_count * sizeof(t_atom)); x->output = new_atom_list; memcpy(x->output + x->output_count, new_atom, sizeof(t_atom)); ++(x->output_count); } static void reset_output(t_usbhid *x) { if(x->output) { freebytes(x->output, x->output_count * sizeof(t_atom)); x->output = NULL; x->output_count = 0; } } /* * This function is used in a HIDInterfaceMatcher to iterate thru all of the * HID devices on the USB bus */ static bool device_iterator (struct usb_dev_handle const* usbdev, void* custom, unsigned int length) { bool ret = false; t_int i; char current_dev_path[10]; /* only here to prevent the unused warning */ /* TODO remove */ length = *((unsigned long*)custom); /* Obtain the device's full path */ sprintf(current_dev_path, "%s/%s", usbdev->bus->dirname, usbdev->device->filename); /* Check if we already saw this dev */ for ( i = 0 ; ( hid_id[i] != NULL ) ; i++ ) { if (!strcmp(hid_id[i], current_dev_path ) ) break; } /* Append device to the list if needed */ if (hid_id[i] == NULL) { hid_id[i] = (char *) malloc(strlen(usbdev->device->filename) + strlen(usbdev->bus->dirname) ); sprintf(hid_id[i], "%s/%s", usbdev->bus->dirname, usbdev->device->filename); } else /* device already seen */ { return false; } /* Filter non HID device */ if ( (usbdev->device->descriptor.bDeviceClass == 0) /* Class defined at interface level */ && usbdev->device->config && usbdev->device->config->interface->altsetting->bInterfaceClass == USB_CLASS_HID) { post("bus %s device %s: %d %d", usbdev->bus->dirname, usbdev->device->filename, usbdev->device->descriptor.idVendor, usbdev->device->descriptor.idProduct); ret = true; } else { ret = false; } return ret; } /* -------------------------------------------------------------------------- */ /* This function is used in a HIDInterfaceMatcher in order to match devices by * serial number. */ /* static bool match_serial_number(struct usb_dev_handle* usbdev, void* custom, unsigned int length) */ /* { */ /* bool ret; */ /* char* buffer = (char*)malloc(length); */ /* usb_get_string_simple(usbdev, usb_device(usbdev)->descriptor.iSerialNumber, */ /* buffer, length); */ /* ret = strncmp(buffer, (char*)custom, length) == 0; */ /* free(buffer); */ /* return ret; */ /* } */ /* -------------------------------------------------------------------------- */ /* static bool get_device_by_number(struct usb_dev_handle* usbdev, */ /* void* custom, */ /* unsigned int length) */ /* { */ /* bool ret; */ /* return ret; */ /* } */ /* -------------------------------------------------------------------------- */ static t_int* make_hid_path(t_int element_count, t_int argc, t_atom *argv) { DEBUG(post("make_hid_path");); t_int i; t_int *return_array = NULL; post("element_count %d",element_count); return_array = (t_int *) getbytes(sizeof(t_int) * element_count); for(i=0; i < element_count; ++i) { /* * A usbhid path component is 32 bits, the high 16 bits identify the usage page, * and the low 16 bits the item number. */ return_array[i] = (atom_getintarg(i*2,argc,argv) << 16) + atom_getintarg(i*2+1,argc,argv); /* TODO: print error if a symbol is found in the data list */ } return return_array; } static t_int get_device_string(HIDInterface *hidif, char *device_string) { int length; t_int ret = 0; char buffer[STRING_BUFFER_LENGTH]; if ( !hid_is_opened(hidif) ) return(0); if (hidif->device->descriptor.iManufacturer) { length = usb_get_string_simple(hidif->dev_handle, hidif->device->descriptor.iManufacturer, buffer, STRING_BUFFER_LENGTH); if (length > 0) { strncat(device_string, buffer, STRING_BUFFER_LENGTH - strlen(device_string)); strncat(device_string, " ",1); ret = 1; } else { post("(unable to fetch manufacturer string)"); } } if (hidif->device->descriptor.iProduct) { length = usb_get_string_simple(hidif->dev_handle, hidif->device->descriptor.iProduct, buffer, STRING_BUFFER_LENGTH); if (length > 0) { strncat(device_string, buffer, STRING_BUFFER_LENGTH - strlen(device_string)); strncat(device_string, " ",1); ret = 1; } else { post("(unable to fetch product string)"); } } if (hidif->device->descriptor.iSerialNumber) { length = usb_get_string_simple(hidif->dev_handle, hidif->device->descriptor.iSerialNumber, buffer, STRING_BUFFER_LENGTH); if (length > 0) strncat(device_string, buffer, STRING_BUFFER_LENGTH - strlen(device_string)); else post("(unable to fetch product string)"); } return ret; } /*------------------------------------------------------------------------------ * IMPLEMENTATION */ /* -------------------------------------------------------------------------- */ static void usbhid_open(t_usbhid *x, t_float vendor_id, t_float product_id) { DEBUG(post("usbhid_open");); char string_buffer[STRING_BUFFER_LENGTH]; HIDInterfaceMatcher matcher = { (unsigned short)vendor_id, (unsigned short)product_id, NULL, NULL, 0 }; if ( !hid_is_opened(x->x_hidinterface) ) { x->x_hid_return = hid_force_open(x->x_hidinterface, 0, &matcher, 3); if (x->x_hid_return == HID_RET_SUCCESS) { if (get_device_string(x->x_hidinterface,string_buffer)) post("[usbhid]: opened %s",string_buffer); } else { error("[usbhid] hid_force_open failed with return code %d\n", x->x_hid_return); } } } /* -------------------------------------------------------------------------- */ static void usbhid_read(t_usbhid *x,t_float length_arg) { DEBUG(post("usbhid_read");); /* int const PATH_IN[PATH_LENGTH] = { 0xffa00001, 0xffa00002, 0xffa10003 }; */ // int const PATH_OUT[3] = { 0x00010002, 0x00010001, 0x00010030 }; int i; int packet_bytes = (int)length_arg; t_atom *temp_atom = getbytes(sizeof(t_atom)); char packet[packet_bytes]; if ( !hid_is_opened(x->x_hidinterface) ) { // usbhid_open(x); return; } x->x_hid_return = hid_get_input_report(x->x_hidinterface, x->x_read_elements, x->x_read_element_count, packet, length_arg); if (x->x_hid_return != HID_RET_SUCCESS) error("[usbhid] hid_get_input_report failed with return code %d\n", x->x_hid_return); reset_output(x); for(i=0; ix_data_outlet, &s_list, x->output_count, x->output); } /* -------------------------------------------------------------------------- */ /* set the HID packet for which elements to read */ static void usbhid_set_read(t_usbhid *x, int argc, t_atom *argv) { DEBUG(post("usbhid_set_read");); t_int i; x->x_read_element_count = argc / 2; x->x_read_elements = make_hid_path(x->x_read_element_count, argc, argv); post("x_read_element_count %d",x->x_read_element_count); for(i=0;ix_read_element_count;++i) post("x_read_elements %d: %d",i,x->x_read_elements[i]); } /* -------------------------------------------------------------------------- */ /* set the HID packet for which elements to write */ static void usbhid_set_write(t_usbhid *x, int argc, t_atom *argv) { DEBUG(post("usbhid_set_write");); t_int i; x->x_write_element_count = argc / 2; x->x_write_elements = make_hid_path(x->x_write_element_count, argc, argv); post("x_write_element_count %d",x->x_write_element_count); for(i=0;ix_write_element_count;++i) post("x_write_elements %d: %d",i,x->x_write_elements[i]); } /* -------------------------------------------------------------------------- */ /* convert a list to a HID packet and set it */ static void usbhid_set(t_usbhid *x, t_symbol *s, int argc, t_atom *argv) { DEBUG(post("usbhid_set");); t_symbol *subselector; if(argc) { subselector = atom_getsymbol(&argv[0]); if(strcmp(subselector->s_name,"read") == 0) usbhid_set_read(x,argc-1,argv+1); if(strcmp(subselector->s_name,"write") == 0) usbhid_set_write(x,argc-1,argv+1); } } /* -------------------------------------------------------------------------- */ static void usbhid_get(t_usbhid *x, t_symbol *s, int argc, t_atom *argv) { DEBUG(post("usbhid_get");); t_symbol *subselector; unsigned int i = 0; t_atom *temp_atom = getbytes(sizeof(t_atom)); if (!hid_is_opened(x->x_hidinterface)) { error("[usbget] cannot dump tree of unopened HIDinterface."); } else if(argc) { subselector = atom_getsymbol(&argv[0]); if(strcmp(subselector->s_name,"read") == 0) { post("[usbhid] parse tree of HIDInterface %s:\n", x->x_hidinterface->id); reset_output(x); while (HIDParse(x->x_hidinterface->hid_parser, x->x_hidinterface->hid_data)) { switch(x->x_hidinterface->hid_data->Type) { case 0x80: SETSYMBOL(temp_atom, gensym("input")); break; case 0x90: SETSYMBOL(temp_atom, gensym("output")); break; case 0xb0: SETSYMBOL(temp_atom, gensym("feature")); break; default: SETSYMBOL(temp_atom, gensym("UNKNOWN_TYPE")); } add_atom_to_output(x,temp_atom); SETFLOAT(temp_atom, x->x_hidinterface->hid_data->Size); add_atom_to_output(x,temp_atom); SETFLOAT(temp_atom, x->x_hidinterface->hid_data->Offset); add_atom_to_output(x,temp_atom); SETSYMBOL(temp_atom, gensym("path")); add_atom_to_output(x,temp_atom); for (i = 0; i < x->x_hidinterface->hid_data->Path.Size; ++i) { SETSYMBOL(temp_atom, gensym("usage")); add_atom_to_output(x,temp_atom); SETFLOAT(temp_atom, x->x_hidinterface->hid_data->Path.Node[i].UPage); add_atom_to_output(x,temp_atom); SETFLOAT(temp_atom, x->x_hidinterface->hid_data->Path.Node[i].Usage); add_atom_to_output(x,temp_atom); post("page: 0x%04x\t%d\t\tusage: 0x%04x\t%d", x->x_hidinterface->hid_data->Path.Node[i].UPage, x->x_hidinterface->hid_data->Path.Node[i].UPage, x->x_hidinterface->hid_data->Path.Node[i].Usage, x->x_hidinterface->hid_data->Path.Node[i].Usage); } post("type: 0x%02x\n", x->x_hidinterface->hid_data->Type); SETSYMBOL(temp_atom, gensym("logical")); add_atom_to_output(x,temp_atom); SETFLOAT(temp_atom, x->x_hidinterface->hid_data->LogMin); add_atom_to_output(x,temp_atom); SETFLOAT(temp_atom, x->x_hidinterface->hid_data->LogMax); add_atom_to_output(x,temp_atom); } outlet_anything(x->x_control_outlet, gensym("device"), x->output_count, x->output); } if(strcmp(subselector->s_name,"write") == 0) { } } else { error("[usbhid] must specify \"read\" or \"write\" as first element of %s message",s->s_name); } } /* -------------------------------------------------------------------------- */ static void usbhid_close(t_usbhid *x) { DEBUG(post("usbhid_close");); t_int ret; char string_buffer[STRING_BUFFER_LENGTH]; /* just to be safe, stop it first */ // usbhid_stop(x); if ( hid_is_opened(x->x_hidinterface) ) { ret = get_device_string(x->x_hidinterface,string_buffer); x->x_hid_return = hid_close(x->x_hidinterface); if (x->x_hid_return == HID_RET_SUCCESS) { if (ret) post("[usbhid]: closed %s",string_buffer); } else { error("[usbhid] could not close %d, error #%d",x->x_device_number,x->x_hid_return); } } } /* -------------------------------------------------------------------------- */ static void usbhid_print(t_usbhid *x) { DEBUG(post("usbhid_print");); t_int i; char string_buffer[STRING_BUFFER_LENGTH]; /* t_atom event_data[3]; */ for ( i = 0 ; ( hid_id[i] != NULL ) ; i++ ) { if( hid_id[i] != NULL ) post("hid_id[%d]: %s",i,hid_id[i]); } if (get_device_string(x->x_hidinterface,string_buffer)) post("%s is currently open",string_buffer); x->x_hid_return = hid_dump_tree(stdout, x->x_hidinterface); if (x->x_hid_return != HID_RET_SUCCESS) { fprintf(stderr, "hid_dump_tree failed with return code %d\n", x->x_hid_return); return; } /* SETSYMBOL(event_data, gensym(type)); /\* type *\/ */ /* SETSYMBOL(event_data + 1, gensym(code)); /\* code *\/ */ /* SETSYMBOL(event_data + 2, value); /\* value *\/ */ // outlet_list(x->x_control_outlet, &s_list, } /* -------------------------------------------------------------------------- */ static void usbhid_reset(t_usbhid *x) { DEBUG(post("usbhid_reset");); hid_reset_HIDInterface(x->x_hidinterface); x->x_hid_return = hid_init(); } /* -------------------------------------------------------------------------- */ static void usbhid_free(t_usbhid* x) { DEBUG(post("usbhid_free");); usbhid_close(x); clock_free(x->x_clock); freebytes(x->x_read_elements,sizeof(t_int) * x->x_read_element_count); freebytes(x->x_write_elements,sizeof(t_int) * x->x_write_element_count); hid_delete_HIDInterface(&(x->x_hidinterface)); x->x_hid_return = hid_cleanup(); if (x->x_hid_return != HID_RET_SUCCESS) error("[usbhid] hid_cleanup failed with return code %d\n", x->x_hid_return); usbhid_instance_count--; } /* -------------------------------------------------------------------------- */ static void *usbhid_new(t_float f) { t_int i; HIDInterfaceMatcher matcher; t_usbhid *x = (t_usbhid *)pd_new(usbhid_class); DEBUG(post("usbhid_new");); /* only display the version when the first instance is loaded */ if(!usbhid_instance_count) post("[usbhid] %d.%d, written by Hans-Christoph Steiner ", USBHID_MAJOR_VERSION, USBHID_MINOR_VERSION); /* x->x_clock = clock_new(x, (t_method)usbhid_read); */ /* create anything outlet used for HID data */ x->x_data_outlet = outlet_new(&x->x_obj, 0); x->x_control_outlet = outlet_new(&x->x_obj, 0); /* hid_write_library_config(stdout); */ //hid_set_debug(HID_DEBUG_NONE); hid_set_debug(HID_DEBUG_NOTRACES); //hid_set_debug(HID_DEBUG_ALL); hid_set_debug_stream(stdout); hid_set_usb_debug(0); /* data init */ x->output = NULL; x->output_count = 0; for (i = 0 ; i < HID_ID_MAX ; i++) hid_id[i] = NULL; if (! hid_is_initialised() ) x->x_hid_return = hid_init(); x->x_hidinterface = hid_new_HIDInterface(); matcher.vendor_id = HID_ID_MATCH_ANY; matcher.product_id = HID_ID_MATCH_ANY; matcher.matcher_fn = device_iterator; x->x_device_number = f; /* find and report the list of devices */ /* open recursively all HID devices found */ while ( (x->x_hid_return = hid_force_open(x->x_hidinterface, 0, &matcher, 2)) != HID_RET_DEVICE_NOT_FOUND) { /* printf("************************************************************************\n"); */ /* hid_write_identification(stdout, x->x_hidinterface); */ /* Only dump HID tree if asked */ /* hid_dump_tree(stdout, x->x_hidinterface); */ hid_close(x->x_hidinterface); } /* Open the device and save settings. If there is an error, return the object * anyway, so that the inlets and outlets are created, thus not breaking the * patch. */ /* if (usbhid_open(x,f)) */ /* error("[usbhid] device %d did not open",(t_int)f); */ usbhid_instance_count++; return (x); } void usbhid_setup(void) { DEBUG(post("usbhid_setup");); usbhid_class = class_new(gensym("usbhid"), (t_newmethod)usbhid_new, (t_method)usbhid_free, sizeof(t_usbhid), CLASS_DEFAULT, A_DEFFLOAT, NULL); /* add inlet datatype methods */ // class_addbang(usbhid_class,(t_method) usbhid_bang); /* add inlet message methods */ class_addmethod(usbhid_class,(t_method) usbhid_print,gensym("print"),0); class_addmethod(usbhid_class,(t_method) usbhid_reset,gensym("reset"),0); class_addmethod(usbhid_class,(t_method) usbhid_read,gensym("read"),A_DEFFLOAT,0); class_addmethod(usbhid_class,(t_method) usbhid_set,gensym("set"),A_GIMME,0); class_addmethod(usbhid_class,(t_method) usbhid_get,gensym("get"),A_GIMME,0); class_addmethod(usbhid_class,(t_method) usbhid_open,gensym("open"),A_DEFFLOAT,A_DEFFLOAT,0); class_addmethod(usbhid_class,(t_method) usbhid_close,gensym("close"),0); }