Add Device.Manufacturer(), .Product() and .SerialNumber() (#14)
This commit is contained in:

committed by
zagrodzki

parent
1aaa100bdb
commit
757722bf8e
1
AUTHORS
1
AUTHORS
@@ -13,3 +13,4 @@ Thordur Bjornsson <thorduri@secnorth.net>
|
||||
Vincent Serpoul <vincent@serpoul.com>
|
||||
Josef Filzmaier <josef.filzmaier@gmail.com>
|
||||
Nico MT <nicovell3@gmail.com>
|
||||
Deomid "rojer" Ryabkov <rojer@rojer.me>
|
||||
|
22
device.go
22
device.go
@@ -45,6 +45,10 @@ type DeviceDesc struct {
|
||||
|
||||
// Configuration information
|
||||
Configs map[int]ConfigDesc
|
||||
|
||||
iManufacturer int // The Manufacturer descriptor index
|
||||
iProduct int // The Product descriptor index
|
||||
iSerialNumber int // The SerialNumber descriptor index
|
||||
}
|
||||
|
||||
// String returns a human-readable version of the device descriptor.
|
||||
@@ -211,6 +215,24 @@ func (d *Device) GetStringDescriptor(descIndex int) (string, error) {
|
||||
return libusb.getStringDesc(d.handle, descIndex)
|
||||
}
|
||||
|
||||
// Manufacturer returns the device's manufacturer name.
|
||||
// GetStringDescriptor's string conversion rules apply.
|
||||
func (d *Device) Manufacturer() (string, error) {
|
||||
return d.GetStringDescriptor(d.Desc.iManufacturer)
|
||||
}
|
||||
|
||||
// Product returns the device's product name.
|
||||
// GetStringDescriptor's string conversion rules apply.
|
||||
func (d *Device) Product() (string, error) {
|
||||
return d.GetStringDescriptor(d.Desc.iProduct)
|
||||
}
|
||||
|
||||
// SerialNumber returns the device's serial number.
|
||||
// GetStringDescriptor's string conversion rules apply.
|
||||
func (d *Device) SerialNumber() (string, error) {
|
||||
return d.GetStringDescriptor(d.Desc.iSerialNumber)
|
||||
}
|
||||
|
||||
// SetAutoDetach enables/disables automatic kernel driver detachment.
|
||||
// When autodetach is enabled gousb will automatically detach the kernel driver
|
||||
// on the interface and reattach it when releasing the interface.
|
||||
|
@@ -44,6 +44,28 @@ func TestClaimAndRelease(t *testing.T) {
|
||||
t.Fatalf("OpenDeviceWithVIDPID(0x8888, 0x0002): %v", err)
|
||||
}
|
||||
|
||||
mfg, err := dev.Manufacturer()
|
||||
if err != nil {
|
||||
t.Errorf("%s.Manufacturer(): %v", dev, err)
|
||||
}
|
||||
if mfg != "ACME Industries" {
|
||||
t.Errorf("%s.Manufacturer(): %q", dev, mfg)
|
||||
}
|
||||
prod, err := dev.Product()
|
||||
if err != nil {
|
||||
t.Errorf("%s.Product(): %v", dev, err)
|
||||
}
|
||||
if prod != "Fidgety Gadget" {
|
||||
t.Errorf("%s.Product(): %q", dev, prod)
|
||||
}
|
||||
sn, err := dev.SerialNumber()
|
||||
if err != nil {
|
||||
t.Errorf("%s.SerialNumber(): %v", dev, err)
|
||||
}
|
||||
if sn != "01234567" {
|
||||
t.Errorf("%s.SerialNumber(): %q", dev, sn)
|
||||
}
|
||||
|
||||
if err = dev.SetAutoDetach(true); err != nil {
|
||||
t.Fatalf("%s.SetAutoDetach(true): %v", dev, err)
|
||||
}
|
||||
@@ -67,7 +89,7 @@ func TestClaimAndRelease(t *testing.T) {
|
||||
if err != nil {
|
||||
t.Fatalf("%s.InEndpoint(%d): got error %v, want nil", intf, ep1Addr, err)
|
||||
}
|
||||
if want := fakeDevices[devIdx].Configs[cfgNum].Interfaces[if1Num].AltSettings[alt1Num].Endpoints[ep1Addr]; !reflect.DeepEqual(got.Desc, want) {
|
||||
if want := fakeDevices[devIdx].devDesc.Configs[cfgNum].Interfaces[if1Num].AltSettings[alt1Num].Endpoints[ep1Addr]; !reflect.DeepEqual(got.Desc, want) {
|
||||
t.Errorf("%s.InEndpoint(%d): got %+v, want %+v", intf, ep1Addr, got, want)
|
||||
}
|
||||
|
||||
@@ -117,6 +139,10 @@ func TestClaimAndRelease(t *testing.T) {
|
||||
t.Fatalf("%s.Close(): got error %v, want nil", dev, err)
|
||||
}
|
||||
|
||||
if _, err := dev.Manufacturer(); err == nil {
|
||||
t.Errorf("%s.Manufacturer(): expected an error after device is closed", dev)
|
||||
}
|
||||
|
||||
if _, err := dev.Config(cfgNum); err == nil {
|
||||
t.Fatalf("%s.Config(1): got error nil, want no nil because it is closed", dev)
|
||||
}
|
||||
|
@@ -15,11 +15,18 @@
|
||||
package gousb
|
||||
|
||||
// fake devices connected through the fakeLibusb stack.
|
||||
var fakeDevices = []*DeviceDesc{
|
||||
type fakeDevice struct {
|
||||
devDesc *DeviceDesc
|
||||
strDesc map[int]string
|
||||
alt uint8
|
||||
}
|
||||
|
||||
var fakeDevices = []fakeDevice{
|
||||
// Bus 001 Device 001: ID 9999:0001
|
||||
// One config, one interface, one setup,
|
||||
// two endpoints: 0x01 OUT, 0x82 IN.
|
||||
&DeviceDesc{
|
||||
{
|
||||
devDesc: &DeviceDesc{
|
||||
Bus: 1,
|
||||
Address: 1,
|
||||
Spec: Version(2, 0),
|
||||
@@ -56,11 +63,13 @@ var fakeDevices = []*DeviceDesc{
|
||||
}},
|
||||
}},
|
||||
},
|
||||
},
|
||||
// Bus 001 Device 002: ID 8888:0002
|
||||
// One config, two interfaces. interface #0 with no endpoints,
|
||||
// interface #1 with two alt setups with different packet sizes for
|
||||
// endpoints. Two isochronous endpoints, 0x05 OUT and 0x86 OUT.
|
||||
&DeviceDesc{
|
||||
{
|
||||
devDesc: &DeviceDesc{
|
||||
Bus: 1,
|
||||
Address: 2,
|
||||
Spec: Version(2, 0),
|
||||
@@ -145,11 +154,21 @@ var fakeDevices = []*DeviceDesc{
|
||||
}},
|
||||
}},
|
||||
}},
|
||||
iManufacturer: 1,
|
||||
iProduct: 2,
|
||||
iSerialNumber: 3,
|
||||
},
|
||||
strDesc: map[int]string{
|
||||
1: "ACME Industries",
|
||||
2: "Fidgety Gadget",
|
||||
3: "01234567",
|
||||
},
|
||||
},
|
||||
// Bus 001 Device 003: ID 9999:0002
|
||||
// One config, one interface, one setup,
|
||||
// two endpoints: 0x01 OUT, 0x81 IN.
|
||||
&DeviceDesc{
|
||||
{
|
||||
devDesc: &DeviceDesc{
|
||||
Bus: 1,
|
||||
Address: 3,
|
||||
Spec: Version(2, 0),
|
||||
@@ -186,4 +205,5 @@ var fakeDevices = []*DeviceDesc{
|
||||
}},
|
||||
}},
|
||||
},
|
||||
},
|
||||
}
|
||||
|
@@ -21,11 +21,6 @@ import (
|
||||
"time"
|
||||
)
|
||||
|
||||
type fakeDevice struct {
|
||||
desc *DeviceDesc
|
||||
alt uint8
|
||||
}
|
||||
|
||||
type fakeTransfer struct {
|
||||
// done is the channel that needs to be closed when the transfer has finished.
|
||||
done chan struct{}
|
||||
@@ -75,7 +70,7 @@ func (f *fakeLibusb) setDebug(*libusbContext, int) {}
|
||||
func (f *fakeLibusb) dereference(d *libusbDevice) {}
|
||||
func (f *fakeLibusb) getDeviceDesc(d *libusbDevice) (*DeviceDesc, error) {
|
||||
if dev, ok := f.fakeDevices[d]; ok {
|
||||
return dev.desc, nil
|
||||
return dev.devDesc, nil
|
||||
}
|
||||
return nil, fmt.Errorf("invalid USB device %p", d)
|
||||
}
|
||||
@@ -109,8 +104,16 @@ func (f *fakeLibusb) setConfig(d *libusbDevHandle, cfg uint8) error {
|
||||
}
|
||||
return nil
|
||||
}
|
||||
func (f *fakeLibusb) getStringDesc(*libusbDevHandle, int) (string, error) {
|
||||
return "", errors.New("not implemented")
|
||||
func (f *fakeLibusb) getStringDesc(d *libusbDevHandle, index int) (string, error) {
|
||||
dev, ok := f.fakeDevices[f.handles[d]]
|
||||
if !ok {
|
||||
return "", fmt.Errorf("invalid USB device %p", d)
|
||||
}
|
||||
str, ok := dev.strDesc[index]
|
||||
if !ok {
|
||||
return "", fmt.Errorf("invalid string descriptor index %d", index)
|
||||
}
|
||||
return str, nil
|
||||
}
|
||||
func (f *fakeLibusb) setAutoDetach(*libusbDevHandle, int) error { return nil }
|
||||
|
||||
@@ -202,10 +205,9 @@ func newFakeLibusb() (*fakeLibusb, func() error) {
|
||||
// without using the full USB stack. Since the fake library uses the
|
||||
// libusbDevice only as an identifier, use an arbitrary unique pointer.
|
||||
// The contents of these pointers is never accessed.
|
||||
fl.fakeDevices[newDevicePointer()] = &fakeDevice{
|
||||
desc: d,
|
||||
alt: 0,
|
||||
}
|
||||
fd := new(fakeDevice)
|
||||
*fd = d
|
||||
fl.fakeDevices[newDevicePointer()] = fd
|
||||
}
|
||||
libusb = fl
|
||||
return fl, func() error {
|
||||
|
@@ -238,6 +238,9 @@ func (libusbImpl) getDeviceDesc(d *libusbDevice) (*DeviceDesc, error) {
|
||||
SubClass: Class(desc.bDeviceSubClass),
|
||||
Protocol: Protocol(desc.bDeviceProtocol),
|
||||
MaxControlPacketSize: int(desc.bMaxPacketSize0),
|
||||
iManufacturer: int(desc.iManufacturer),
|
||||
iProduct: int(desc.iProduct),
|
||||
iSerialNumber: int(desc.iSerialNumber),
|
||||
}
|
||||
// Enumerate configurations
|
||||
cfgs := make(map[int]ConfigDesc)
|
||||
|
Reference in New Issue
Block a user