Skip site navigation (1)Skip section navigation (2)
Date:      Sun, 17 Feb 2008 19:59:21 -0500
From:      Chuck Robey <chuckr@chuckr.org>
To:        Markus Brueffer <markus@FreeBSD.org>
Cc:        Alexander Leidinger <Alexander@leidinger.net>, freebsd-current@freebsd.org
Subject:   Re: about usb
Message-ID:  <47B8D869.1030009@chuckr.org>
In-Reply-To: <200802171642.04403.markus@FreeBSD.org>
References:  <47B20739.1050602@chuckr.org> <20080216121241.450ff175@deskjail> <47B79A48.1030406@chuckr.org> <200802171642.04403.markus@FreeBSD.org>

next in thread | previous in thread | raw e-mail | index | archive | help
-----BEGIN PGP SIGNED MESSAGE-----
Hash: SHA1

Markus Brueffer wrote:
> Am Sonntag, 17. Februar 2008 03:22:00 schrieb Chuck Robey:
>> I am still having a miserable time trying to figure out the HID spec, and
>> Kai Wang deserves being nominated for Sainthood, for putting up with my
>> horde of stupid questions.  If anyone knows of a HID device which has a
>> published [HID report in hex, Parsed output, and comments on how it got
>> from one to the other] I would give nearly anything for such a thing
>> (anything I happen to have, which doesn't happen to include cash right
>> now).
> 

Boy I am really, really pleased you made this public, and I believe folks
will be googling this for years now.  What a great thing to do!

> This is the raw report descriptor of a very simple device: an Apple Mighty 
> Mouse:
> 
> 0x05 0x01 0x09 0x02 0xa1 0x01 0x05 0x09
> 0x19 0x01 0x29 0x04 0x15 0x00 0x25 0x01
> 0x95 0x04 0x75 0x01 0x81 0x02 0x95 0x01
> 0x75 0x04 0x81 0x01 0x05 0x01 0x09 0x01
> 0xa1 0x00 0x09 0x30 0x09 0x31 0x09 0x32
> 0x09 0x38 0x15 0x81 0x25 0x7f 0x75 0x08
> 0x95 0x04 0x81 0x06 0xc0 0x05 0xff 0x09
> 0xc0 0x75 0x08 0x95 0x01 0x81 0x02 0xc0
> 
> Decoding it works as follows:
> 
> Let's look at the first byte 0x05. The binary representation is 00000101.
> Splitting this up as described in chapter 6.2.2.2 gives the following (in 
> binary representation):
> 
> bTag=0000
> bType=01
> bSize=01
> 
> First, as bTag is not 1111, we know that this is not a long item (see 
> 6.2.2.3). So this must be a short item.
> 
> bType tells us that this is a global item (6.2.2.2). Finally bSize specifies 
> the number of following bytes that belong to this item which is 1 in this 
> case. So, looking back at the raw descriptor we know now that the sequence 
> 0x05 0x01 forms one item. We also know that it is a short item and that it is 
> a global item. What we don't know yet is, what global item this is. For this, 
> we have to look at bTag again. As we know that this is a global item, we take 
> a look at chapter 6.2.2.7 "Global Items". It tells us that 0000 stands for 
> Usage Page. Which usage page is meant is encoded in the data part of the 
> item, which is 0x01. From the HID Usage Tables document Chapter 3 we learn 
> that 0x01 stands for "Generic Desktop Controls". This gives us the following 
> for the first item:
> 
> 0x05 0x01	USAGE_PAGE (Generic Desktop)
> 
> Now the second item beginning with 0x09: Binary representation is 00001001. 
> Splitting this up yields:
> 
> bTag=0000
> bType=10
> bSize=01
> 
> Again this is a short item as bTag != 1111. bType is a decimal 2 and chapter 
> 6.2.2.2 tells us that this is a local item. Advancing to chapter 
> 6.2.2.7 "Local Items" gives us the meaning for bTag, which is "Usage". As 
> bSize is 1, only the next byte in the report descriptor is considered for the 
> data of the item, so the usage is 0x02. Looking at the HID Usage Tables 
> document again, this time chapter 4, as we already learned that this usage 
> belongs to the usage page Generic Desktop, we see that 0x02 is equivalent 
> to "Mouse". Now we have:
> 
> 0x05 0x01       USAGE_PAGE (Generic Desktop)
> 0x09 0x02       USAGE (Mouse)
> 
> Up to the next item which begins with 0xa1, binary 10100001:
> 
> bTag=1010
> bType=00
> bSize=01
> 
> Again this is a short item as bTag != 1111. bType is 0, which stands for Main 
> Item, according to chapter 6.2.2.2. Looking at chapter 6.2.2.4, we learn that 
> the bTag of 1010 actually means "Collection". The data part of 1 byte (see 
> bSize), which is in this case 0x01 tells us that this is an "Application 
> Collection". So, now we have:
> 
> 0x05 0x01       USAGE_PAGE (Generic Desktop)
> 0x09 0x02       USAGE (Mouse)
> 0xa1 0x01       COLLECTION (Application)
> 
> If you continue decoding it this way, you will finally get the following:
> 
> 0x05 0x01       USAGE_PAGE (Generic Desktop)
> 0x09 0x02       USAGE (Mouse)
> 0xa1 0x01       COLLECTION (Application)
> 0x05 0x09       USAGE_PAGE (Button)
> 0x19 0x01       USAGE_MINIMUM (Button 1)
> 0x29 0x04       USAGE_MAXIMUM (Button 4)
> 0x15 0x00       LOGICAL_MINIMUM (0)
> 0x25 0x01       LOGICAL_MAXIMUM (1)
> 0x95 0x04       REPORT_COUNT (4)
> 0x75 0x01       REPORT_SIZE (1)
> 0x81 0x02       INPUT (Data, Var, Abs)
> 0x95 0x01       REPORT_COUNT (1)
> 0x75 0x04       REPORT_SIZE (4)
> 0x81 0x01       INPUT (Const, Var, Abs)
> 0x05 0x01       USAGE_PAGE (Generic Desktop)
> 0x09 0x01       USAGE (Pointer)
> 0xa1 0x00       COLLECTION (Physical)
> 0x09 0x30       USAGE (X)
> 0x09 0x31       USAGE (Y)
> 0x09 0x32       USAGE (Z)
> 0x09 0x38       USAGE (WHEEL)
> 0x15 0x81       LOGICAL_MINIMUM (0x81)
> 0x25 0x7f       LOGICAL_MAXIMUM (0x7f)
> 0x75 0x08       REPORT_SIZE (8)
> 0x95 0x04       REPORT_COUNT (4)
> 0x81 0x06       INPUT (Data, Var, Rel)
> 0xc0            END COLLECTION
> 0x05 0xff       USAGE_PAGE (0xff)
> 0x09 0xc0       USAGE (0xc0)
> 0x75 0x08       REPORT_SIZE (8)
> 0x95 0x01       REPORT_COUNT (1)
> 0x81 0x02       INPUT (Data, Var, Abs)
> 0xc0
> 
> To actually give meaning to the decoded descriptor, you have to learn how 
> local, global and main items relate to each other. Main items are actually 
> the most important ones, as they directly correlate to physical controls of 
> your device (apart from the collection items). Global and local items 
> describe the different main items, whereas local items only describe the next 
> main item. Global items instead keep their state accross main items until 
> they are specified again or are being overrriden by a local item.
> 
> In this example we have several main items. Lets have a look at the first one
> 
> 0x05 0x01       USAGE_PAGE (Generic Desktop)
> 0x09 0x02       USAGE (Mouse)
> 0xa1 0x01       COLLECTION (Application)
> 
> This is an application collection. Usage page and usage describe the context 
> of the items that are within it, so we have:
> 
> Collection page=Generic_Desktop usage=Mouse
> 
> The next item is a bit more interresting. It's an input item:
> 
> 0x05 0x09       USAGE_PAGE (Button)
> 0x19 0x01       USAGE_MINIMUM (Button 1)
> 0x29 0x04       USAGE_MAXIMUM (Button 4)
> 0x15 0x00       LOGICAL_MINIMUM (0)
> 0x25 0x01       LOGICAL_MAXIMUM (1)
> 0x95 0x04       REPORT_COUNT (4)
> 0x75 0x01       REPORT_SIZE (1)
> 0x81 0x02       INPUT (Data, Var, Abs)
> 
> It actually describes 4 controls (report count). The meaning of these 4 
> controls is specified by the usage page and usage min/max. In this case 
> Button 1 - Button 4. Logical min/max are 0 and 1 and mean that the range of 
> the data in the data stream, that is submitted by the device, is between 0 
> and 1, which is actually only 0 and 1 in this case. Report size finally gives 
> the size in the data stream for each button state that is submitted by the 
> device in number of bits. Now we have:
> 
> Collection page=Generic_Desktop usage=Mouse
> Input   size=1 count=1 page=Button usage=Button_1, logical range 0..1
> Input   size=1 count=1 page=Button usage=Button_2, logical range 0..1
> Input   size=1 count=1 page=Button usage=Button_3, logical range 0..1
> Input   size=1 count=1 page=Button usage=Button_4, logical range 0..1
> 
> The next input item is a const item, which is being used for padding in this 
> case. It can be ignored. In the end we get the following:
> 
> Collection page=Generic_Desktop usage=Mouse
> Input   size=1 count=1 page=Button usage=Button_1, logical range 0..1
> Input   size=1 count=1 page=Button usage=Button_2, logical range 0..1
> Input   size=1 count=1 page=Button usage=Button_3, logical range 0..1
> Input   size=1 count=1 page=Button usage=Button_4, logical range 0..1
> Input   size=4 count=1 page=0x0000 usage=0x0000 Const, logical range 0..1
> Collection page=Generic_Desktop usage=Pointer
> Input   size=8 count=1 page=Generic_Desktop usage=X, logical range -127..127
> Input   size=8 count=1 page=Generic_Desktop usage=Y, logical range -127..127
> Input   size=8 count=1 page=Generic_Desktop usage=Z, logical range -127..127
> Input   size=8 count=1 page=Generic_Desktop usage=Wheel, logical 
> range -127..127
> End collection
> Input   size=8 count=1 page=0x00ff usage=0x00c0, logical range -127..127
> End collection
> 
> This actually means, that the mouse reports 6 bytes of data in each 
> transmission. The first 4 bits indicate the button states, the next 4 bits 
> can be ignored. Byte 2 reports the X axis state, byte 3, 4 and 5 data for Y, 
> Z and the wheel. The last byte reports vendor specific data (usage page 
> 0x00ff).
> 
> That's it in short. There is of course much to learn regarding the details, 
> but it should give you a general overview of how this works.
> 
> Markus

-----BEGIN PGP SIGNATURE-----
Version: GnuPG v2.0.4 (FreeBSD)
Comment: Using GnuPG with Mozilla - http://enigmail.mozdev.org

iD8DBQFHuNhoz62J6PPcoOkRAiuYAJ4iMn+/EX+lDnmvotxzOPa9JLJ6jQCdHtDN
HGyri1EPkH97rMTojc/x/MQ=
=YWSy
-----END PGP SIGNATURE-----



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?47B8D869.1030009>