Hello - I'm not sure if my first message was properly received, so I have included it below. Please read the earlier message first. Here is my simple usbhid application. All I want to do is send 32 bytes to Report ID #0.
========== hqcthid.c ========== static void showReports(int fd, unsigned report_type) { struct hiddev_report_info rinfo; struct hiddev_field_info finfo; struct hiddev_usage_ref uref; int i, j, ret; unsigned char go_am[] = "\x00\x18\x30\x40\x0f\x9f\x01\x00\x30\x50\x50\x74\x6f\x6a\x00\x03\x03\x02\x0A\x0c\x01\x01\x00\xa0\x00\x00\x00\x00\x00\x00\x00\x00"; rinfo.report_type = report_type; rinfo.report_id = HID_REPORT_ID_FIRST; ret = ioctl(fd, HIDIOCGREPORTINFO, &rinfo); // Get Reports while (ret >= 0) { printf("HIDIOCGREPORTINFO: report_id=0x%X (%u fields)\n", rinfo.report_id, rinfo.num_fields); // Get Fields for (i = 0; i < rinfo.num_fields; i++) { finfo.report_type = rinfo.report_type; finfo.report_id = rinfo.report_id; finfo.field_index = i; ioctl(fd, HIDIOCGFIELDINFO, &finfo); printf("HIDIOCGFIELDINFO: field_index=%u maxusage=%u flags=0x%X\n" "\tphysical=0x%X logical=0x%X application=0x%X\n" "\tlogical_minimum=%d,maximum=%d physical_minimum=%d,maximum=%d\n", finfo.field_index, finfo.maxusage, finfo.flags, finfo.physical, finfo.logical, finfo.application, finfo.logical_minimum, finfo.logical_maximum, finfo.physical_minimum, finfo.physical_maximum); // Get usages for (j = 0; j < finfo.maxusage; j++) { uref.report_type = finfo.report_type; uref.report_id = finfo.report_id; uref.field_index = i; uref.usage_index = j; ioctl(fd, HIDIOCGUCODE, &uref); ioctl(fd, HIDIOCGUSAGE, &uref); printf(" >> usage_index=%u usage_code=0x%X () value=0x%X\n", uref.usage_index, uref.usage_code, uref.value); // Input usage: set a byte. if(uref.report_type == HID_REPORT_TYPE_INPUT) { uref.value = go_am[j]; printf(" S> Usage_Index=%u Usage_Code=0x%X () Value=0x%X\n", uref.usage_index, uref.usage_code, uref.value); ioctl(fd,HIDIOCSUSAGE,&uref); } } // Input report: send report. if( report_type == HID_REPORT_TYPE_INPUT ) { printf(" S> report sent"); ioctl(fd,HIDIOCSREPORT,&rinfo); } } printf("\n"); rinfo.report_id |= HID_REPORT_ID_NEXT; ret = ioctl(fd, HIDIOCGREPORTINFO, &rinfo); } } int main (int argc, char **argv) { int fd = -1; if (argc != 2) { fprintf(stderr, "usage: %s hiddevice - probably /dev/usb/hiddev0\n", argv[0]); exit(1); } if ((fd = open(argv[1], O_RDONLY)) < 0) { perror("hiddev open"); exit(1); } // Should display the 32 input usages and send a report printf("\n*** INPUT:\n"); showReports(fd, HID_REPORT_TYPE_INPUT); // Should display 32 output usages. printf("\n*** OUTPUT:\n"); showReports(fd, HID_REPORT_TYPE_OUTPUT); // Should display 2 features. printf("\n*** FEATURE:\n"); showReports(fd, HID_REPORT_TYPE_FEATURE); close(fd); return 0; } ========== end hqcthid.c ========== This program basically lists all the usages of the hid device. Except, if the report is an input report then it will additionally set the usages and send a report. I know a _real_ program would be written more efficiently, but this gets the point across. Look at "// Input usage: set a byte" and "// Input report: send report" for the bulk of the work. This is what is displayed for the input report after two consecutive executions of the program; First issue: ===================== *** INPUT: HIDIOCGREPORTINFO: report_id=0x0 (1 fields) HIDIOCGFIELDINFO: field_index=31 maxusage=32 flags=0x2 physical=0x0 logical=0x0 application=0xFFA00001 logical_minimum=0,maximum=65280 physical_minimum=0,maximum=0 >> usage_index=0 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=0 Usage_Code=0xFFA00003 () Value=0x18 >> usage_index=1 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=1 Usage_Code=0xFFA00003 () Value=0x30 >> usage_index=2 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=2 Usage_Code=0xFFA00003 () Value=0x40 >> usage_index=3 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=3 Usage_Code=0xFFA00003 () Value=0xF >> usage_index=4 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=4 Usage_Code=0xFFA00003 () Value=0x9F >> usage_index=5 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=5 Usage_Code=0xFFA00003 () Value=0x1 >> usage_index=6 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=6 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=7 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=7 Usage_Code=0xFFA00003 () Value=0x30 >> usage_index=8 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=8 Usage_Code=0xFFA00003 () Value=0x50 >> usage_index=9 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=9 Usage_Code=0xFFA00003 () Value=0x50 >> usage_index=10 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=10 Usage_Code=0xFFA00003 () Value=0x74 >> usage_index=11 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=11 Usage_Code=0xFFA00003 () Value=0x6F >> usage_index=12 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=12 Usage_Code=0xFFA00003 () Value=0x6A >> usage_index=13 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=13 Usage_Code=0xFFA00003 () Value=0xD0 >> usage_index=14 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=14 Usage_Code=0xFFA00003 () Value=0x3 >> usage_index=15 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=15 Usage_Code=0xFFA00003 () Value=0x3 >> usage_index=16 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=16 Usage_Code=0xFFA00003 () Value=0x20 >> usage_index=17 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=17 Usage_Code=0xFFA00003 () Value=0xA >> usage_index=18 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=18 Usage_Code=0xFFA00003 () Value=0xC >> usage_index=19 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=19 Usage_Code=0xFFA00003 () Value=0x1 >> usage_index=20 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=20 Usage_Code=0xFFA00003 () Value=0x1 >> usage_index=21 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=21 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=22 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=22 Usage_Code=0xFFA00003 () Value=0xA0 >> usage_index=23 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=23 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=24 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=24 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=25 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=25 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=26 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=26 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=27 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=27 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=28 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=28 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=29 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=29 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=30 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=30 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=31 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=31 Usage_Code=0xFFA00003 () Value=0x0 S> sending report S> report sent ====================== So, the values are all 0x00 initially and we supposedly set the values. Unfortunately, the radio does not seem to change state. (>> is existing value and S> is the sent value). Second issue: ====================== *** INPUT: HIDIOCGREPORTINFO: report_id=0x0 (1 fields) HIDIOCGFIELDINFO: field_index=31 maxusage=32 flags=0x2 physical=0x0 logical=0x0 application=0xFFA00001 logical_minimum=0,maximum=65280 physical_minimum=0,maximum=0 >> usage_index=0 usage_code=0xFFA00003 () value=0x3E S> Usage_Index=0 Usage_Code=0xFFA00003 () Value=0x18 >> usage_index=1 usage_code=0xFFA00003 () value=0x2 S> Usage_Index=1 Usage_Code=0xFFA00003 () Value=0x30 >> usage_index=2 usage_code=0xFFA00003 () value=0x7 S> Usage_Index=2 Usage_Code=0xFFA00003 () Value=0x40 >> usage_index=3 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=3 Usage_Code=0xFFA00003 () Value=0xF >> usage_index=4 usage_code=0xFFA00003 () value=0x1C S> Usage_Index=4 Usage_Code=0xFFA00003 () Value=0x9F >> usage_index=5 usage_code=0xFFA00003 () value=0x57 S> Usage_Index=5 Usage_Code=0xFFA00003 () Value=0x1 >> usage_index=6 usage_code=0xFFA00003 () value=0x98 S> Usage_Index=6 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=7 usage_code=0xFFA00003 () value=0xA3 S> Usage_Index=7 Usage_Code=0xFFA00003 () Value=0x30 >> usage_index=8 usage_code=0xFFA00003 () value=0x9 S> Usage_Index=8 Usage_Code=0xFFA00003 () Value=0x50 >> usage_index=9 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=9 Usage_Code=0xFFA00003 () Value=0x50 >> usage_index=10 usage_code=0xFFA00003 () value=0x8 S> Usage_Index=10 Usage_Code=0xFFA00003 () Value=0x74 >> usage_index=11 usage_code=0xFFA00003 () value=0x74 S> Usage_Index=11 Usage_Code=0xFFA00003 () Value=0x6F >> usage_index=12 usage_code=0xFFA00003 () value=0x3A S> Usage_Index=12 Usage_Code=0xFFA00003 () Value=0x6A >> usage_index=13 usage_code=0xFFA00003 () value=0x5 S> Usage_Index=13 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=14 usage_code=0xFFA00003 () value=0x25 S> Usage_Index=14 Usage_Code=0xFFA00003 () Value=0x3 >> usage_index=15 usage_code=0xFFA00003 () value=0x24 S> Usage_Index=15 Usage_Code=0xFFA00003 () Value=0x3 >> usage_index=16 usage_code=0xFFA00003 () value=0xCA S> Usage_Index=16 Usage_Code=0xFFA00003 () Value=0x20 >> usage_index=17 usage_code=0xFFA00003 () value=0x35 S> Usage_Index=17 Usage_Code=0xFFA00003 () Value=0xA >> usage_index=18 usage_code=0xFFA00003 () value=0x14 S> Usage_Index=18 Usage_Code=0xFFA00003 () Value=0xC >> usage_index=19 usage_code=0xFFA00003 () value=0x10 S> Usage_Index=19 Usage_Code=0xFFA00003 () Value=0x1 >> usage_index=20 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=20 Usage_Code=0xFFA00003 () Value=0x1 >> usage_index=21 usage_code=0xFFA00003 () value=0x15 S> Usage_Index=21 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=22 usage_code=0xFFA00003 () value=0x24 S> Usage_Index=22 Usage_Code=0xFFA00003 () Value=0xA0 >> usage_index=23 usage_code=0xFFA00003 () value=0x1 S> Usage_Index=23 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=24 usage_code=0xFFA00003 () value=0x0 S> Usage_Index=24 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=25 usage_code=0xFFA00003 () value=0xE8 S> Usage_Index=25 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=26 usage_code=0xFFA00003 () value=0xC1 S> Usage_Index=26 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=27 usage_code=0xFFA00003 () value=0x1A S> Usage_Index=27 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=28 usage_code=0xFFA00003 () value=0x23 S> Usage_Index=28 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=29 usage_code=0xFFA00003 () value=0xA5 S> Usage_Index=29 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=30 usage_code=0xFFA00003 () value=0xC0 S> Usage_Index=30 Usage_Code=0xFFA00003 () Value=0x0 >> usage_index=31 usage_code=0xFFA00003 () value=0x8 S> Usage_Index=31 Usage_Code=0xFFA00003 () Value=0x0 S> sending report S> report sent ====================== This time the values are not equal to 0x00. instead they are some seemingly random numbers. Can anyone provide me with some hints or maybe a decent example driver to reference? I appreciate your assistance, thanks for supporting a tool as vital as USB. --Paul On 5/9/06, Paul Giblock <[EMAIL PROTECTED]> wrote:
Hello - This is my first post to this newsletter. I have been trying to write host software to communicate with a low-speed USB PIC Microcontroller. The device is configured to send and receive 32 byte reports. The designer of the usb project has code writen in Windows Pascal (Delphi?) to control the device. The code is a rework from the code generated by EasyHID. Here is my output of lsusb: ------------------------------------------------------------ llamabox hqct # lsusb -d 04d8:000a -vvvv Bus 001 Device 002: ID 04d8:000a Microchip Technology, Inc. Device Descriptor: bLength 18 bDescriptorType 1 bcdUSB 2.00 bDeviceClass 0 (Defined at Interface level) bDeviceSubClass 0 bDeviceProtocol 0 bMaxPacketSize0 8 idVendor 0x04d8 Microchip Technology, Inc. idProduct 0x000a bcdDevice 0.01 iManufacturer 1 Datalex iProduct 2 Car-Radio iSerial 0 bNumConfigurations 1 Configuration Descriptor: bLength 9 bDescriptorType 2 wTotalLength 41 bNumInterfaces 1 bConfigurationValue 1 iConfiguration 0 bmAttributes 0xa0 Remote Wakeup MaxPower 100mA Interface Descriptor: bLength 9 bDescriptorType 4 bInterfaceNumber 0 bAlternateSetting 0 bNumEndpoints 2 bInterfaceClass 3 Human Interface Devices bInterfaceSubClass 0 No Subclass bInterfaceProtocol 0 None iInterface 0 HID Device Descriptor: bLength 9 bDescriptorType 33 bcdHID 1.01 bCountryCode 0 Not supported bNumDescriptors 1 bDescriptorType 34 Report wDescriptorLength 47 Report Descriptor: (length is 47) Item(Global): Usage Page, data= [ 0xa0 0xff ] 65440 (null) Item(Local ): Usage, data= [ 0x01 ] 1 (null) Item(Main ): Collection, data= [ 0x01 ] 1 Application Item(Local ): Usage, data= [ 0x03 ] 3 (null) Item(Global): Logical Minimum, data= [ 0x00 ] 0 Item(Global): Logical Maximum, data= [ 0x00 0xff ] 65280 Item(Global): Report Size, data= [ 0x08 ] 8 Item(Global): Report Count, data= [ 0x20 ] 32 Item(Main ): Input, data= [ 0x02 ] 2 Data Variable Absolute No_Wrap Linear Preferred_State No_Null_Position Non_Volatile Bitfield Item(Local ): Usage, data= [ 0x04 ] 4 (null) Item(Global): Logical Minimum, data= [ 0x00 ] 0 Item(Global): Logical Maximum, data= [ 0x00 0xff ] 65280 Item(Global): Report Size, data= [ 0x08 ] 8 Item(Global): Report Count, data= [ 0x20 ] 32 Item(Main ): Output, data= [ 0x02 ] 2 Data Variable Absolute No_Wrap Linear Preferred_State No_Null_Position Non_Volatile Bitfield Item(Local ): Usage, data= [ 0x05 ] 5 (null) Item(Global): Logical Minimum, data= [ 0x00 ] 0 Item(Global): Logical Maximum, data= [ 0x00 0xff ] 65280 Item(Global): Report Size, data= [ 0x08 ] 8 Item(Global): Report Count, data= [ 0x02 ] 2 Item(Main ): Feature, data= [ 0x02 ] 2 Data Variable Absolute No_Wrap Linear Preferred_State No_Null_Position Non_Volatile Bitfield Item(Main ): End Collection, data=none Endpoint Descriptor: bLength 7 bDescriptorType 5 bEndpointAddress 0x81 EP 1 IN bmAttributes 3 Transfer Type Interrupt Synch Type None Usage Type Data wMaxPacketSize 0x0040 1x 64 bytes bInterval 1 Endpoint Descriptor: bLength 7 bDescriptorType 5 bEndpointAddress 0x01 EP 1 OUT bmAttributes 3 Transfer Type Interrupt Synch Type None Usage Type Data wMaxPacketSize 0x0040 1x 64 bytes bInterval 1 ------------------------------------------------------------ I originally tried using a HIDDEV device. I could not figure out how to write to the device properly. Looking at the source shows that write() is implemented to return an error. The documentation is too ambiguous for me to figure out how to write properly... I couldn't get HIDIOCSREPORT to work properly at all. I have read a good deal of the USB HID papers even. I tried writing a kernel module based around usb-skel.c. I was able to change the state of the device noticibly, but I haven't been able to recreate it. So, this has only happened once. I change the skeleton to write to the interrupt endpoints instead of using bulk transfer since that is not supported. I have even tried using libhid and that hasn't worked. The path I have tried for writing is: { 0xffa00001, 0xffa00003 } This has not worked, from what I could tell, at all. I have even tried using three different functions for writing while using the libhid approach: hid_set_output_report(), usb_interrupt_write(), and hid_interrupt_write(). I have tried restarting the computer and device between almost every test. Has anyone been able to communicate with a PIC or a comparable HID device? All I want is to send a 32 byte report. I will post sources if requested, I'm not sure how much they would help though. Thank you, Paul ------------------------------------------------------------ llamabox hqct # cat /proc/bus/usb/devices [...] T: Bus=01 Lev=01 Prnt=01 Port=01 Cnt=01 Dev#= 2 Spd=12 MxCh= 0 D: Ver= 2.00 Cls=00(>ifc ) Sub=00 Prot=00 MxPS= 8 #Cfgs= 1 P: Vendor=04d8 ProdID=000a Rev= 0.01 S: Manufacturer=Datalex S: Product=Car-Radio C:* #Ifs= 1 Cfg#= 1 Atr=a0 MxPwr=100mA I: If#= 0 Alt= 0 #EPs= 2 Cls=03(HID ) Sub=00 Prot=00 Driver=(none) E: Ad=81(I) Atr=03(Int.) MxPS= 64 Ivl=1ms E: Ad=01(O) Atr=03(Int.) MxPS= 64 Ivl=1ms
------------------------------------------------------- Using Tomcat but need to do more? Need to support web services, security? Get stuff done quickly with pre-integrated technology to make your job easier Download IBM WebSphere Application Server v.1.0.1 based on Apache Geronimo http://sel.as-us.falkag.net/sel?cmd=lnk&kid0709&bid&3057&dat1642 _______________________________________________ Linux-usb-users@lists.sourceforge.net To unsubscribe, use the last form field at: https://lists.sourceforge.net/lists/listinfo/linux-usb-users