USB HIDデバイスは1つのイベントのみを発生させる


11

ASUS PSR 2000 Webサーフィンリモートコントロールに付属しているeDIO USBマルチリモートコントローラー(赤外線レシーバー)を持っています。

リモートから送信されたキーストロークを受信できるように、リモートコントローラをpiに接続しようとしています。

コントローラがHIDデバイスとして検出されました。lsusb -vコマンドの詳細は次のとおりです

    Bus 001 Device 007: ID 147a:e001 Formosa Industrial Computing, Inc.
    Couldn't open device, some information will be missing
    Device Descriptor:
    bLength                18
    bDescriptorType         1
    bcdUSB               1.10
    bDeviceClass            0 (Defined at Interface level)
   bDeviceSubClass         0
   bDeviceProtocol         0
   bMaxPacketSize0         8
   idVendor           0x147a Formosa Industrial Computing, Inc.
   idProduct          0xe001
   bcdDevice            1.22
   iManufacturer           1
   iProduct                2
   iSerial                 0
   bNumConfigurations      1
  Configuration Descriptor:
  bLength                 9
  bDescriptorType         2
wTotalLength           34
bNumInterfaces          1
bConfigurationValue     1
iConfiguration          4
bmAttributes         0xa0
  (Bus Powered)
  Remote Wakeup
MaxPower              300mA
Interface Descriptor:
  bLength                 9
  bDescriptorType         4
  bInterfaceNumber        0
  bAlternateSetting       0
  bNumEndpoints           1
  bInterfaceClass         3 Human Interface Device
  bInterfaceSubClass      1 Boot Interface Subclass
  bInterfaceProtocol      2 Mouse
  iInterface              0
    HID Device Descriptor:
      bLength                 9
      bDescriptorType        33
      bcdHID               1.10
      bCountryCode            0 Not supported
      bNumDescriptors         1
      bDescriptorType        34 Report
      wDescriptorLength      20
     Report Descriptors:
       ** UNAVAILABLE **
  Endpoint Descriptor:
    bLength                 7
    bDescriptorType         5
    bEndpointAddress     0x81  EP 1 IN
    bmAttributes            3
      Transfer Type            Interrupt
      Synch Type               None
      Usage Type               Data
    wMaxPacketSize     0x0004  1x 4 bytes
    bInterval              10

イベントが作成されたdevフォルダー内のターゲットデバイスも表示できます

    pi@raspberrypi /dev/input/by-id $ dir
    usb-Cypress_Semiconductor_eDio_USB_Multi_Remote_Controlle-event-if00

これに関連付けられているイベントハンドラーは、次のコマンドからわかるように次のとおりです。

pi@raspberrypi /proc/bus/input $ cat devices
I: Bus=0003 Vendor=147a Product=e001 Version=0110
N: Name="Cypress Semiconductor eDio USB Multi Remote Controlle"
P: Phys=usb-bcm2708_usb-1.2/input0
S: Sysfs=/devices/platform/bcm2708_usb/usb1/1-1/1-1.2/1-1.2:1.0/input/input2
U: Uniq=
H: Handlers=event0
B: PROP=0
B: EV=1

問題は、デバイス用に作成されたイベントハンドラーから出力を読み取ろうとするときです。最初のキーストロークは登録されていますが、後続のキーストロークはCATコマンドで表示されません。

 pi@raspberrypi /dev/input $ cat event0 | xxd
 0000000: e007 9450 9476 0900 0000 0000 0000 0000  ...P.v..........

デバイスを機能させるために何ができるか教えてください。最初のキーストローク後にキーを押しても、デバイスが再接続されない限り、何も返されません。

問題を解決するために何をする必要があるかを提案してください。


誰か何か???私はデバイスで何が起こっているのか無知です。おそらく、モデレーターが問題を解決するのに役立ちますか?
SteveIrwin

質問はいいです。しかし、それはかなりローカライズされているので、同じ問題を抱えていた人は多くないでしょう。あなたがそれを見ることができるように、私がクリス・ウォレスの話すボートによって使われる非常に類似した何かを見たと知ってあなたは落ち着くかもしれません。問題を診断するために最初に尋ねるのは、電源の問題が発生する可能性があるため、セルフパワーハブを使用していますか。
Jivings、2012年

なしで試しました|xxdか?出力をバッファリングします。私はリモートから送信されたキーコードを取得するためにirwパッケージlircから使用しました。
macrojames

カスタムドライバーは、Linuxカーネルパッチを意味します。libusbはUSBエンドポイントに直接アクセスできるため、より簡単なオプションはlibusbを使用することです。
LarsPötter12年

回答:


5

問題は不完全なUSB Descrioptorsのようです:

  Couldn't open device, some information will be missing
  Report Descriptors:
  ** UNAVAILABLE **

読むことができる記述子は、これがマウスであると言います。

  bInterfaceProtocol      2 Mouse

そして、データフォーマットを記述する20バイトの記述子があるでしょう:

  bDescriptorType        34 Report
  wDescriptorLength      20

しかし、それは欠けています。

ハードウェアとソフトウェアの特定の組み合わせに奇妙な問題があるか、プログラマーが怠惰で、レポート記述子を実装していなかった可能性があります。独自のドライバーはおそらくそれを必要としないためです。しかし、おそらくそれにより、入力デバイスを作成するドライバが混乱しました。

libusbを使用して、エンドポイントから4バイトを読み取ることができます。おそらくポーリングは機能します。または、元のドライバーでデバイスを使用する場合は、USB通信を確認してください。そして、もしあなたが高価なUSBロガーを置いていない場合、これは非常にトリッキーです。しかし、LinuxカーネルはソフトウェアUSBロギングをサポートしており、Windows用のソフトウェアロガーがいくつか利用できます。


4

最後に、LibusbのラッパーであるPyUSBライブラリを使用して独自の実装を作成する時間を得ました。

私はここにコードを投稿しています。誰かを助けるかもしれません。

ここで使用する構成ファイルを作成する別のコードがあります。すべてのリモートキーが必要ないため、すべてのリモートキーをマップしていません。

import usb.core
import usb.util
import ConfigParser 
import shlex
import subprocess
import logging

# find our device
diction={
  6402315641282315:'1',
  6402415641282415:'2',
  6402515641282515:'3',
  6402615641282615:'4',
  6402715641282715:'5',
  6402815641282815:'6',
  6402915641282915:'7',
  6403015641283015:'8',
  6403115641283115:'9',
  }



def load_config():
    dict={}
    config = ConfigParser.RawConfigParser()
    config.read('/codes/remote/remote.cfg')

    dict['vendor']=config.getint('Settings','idVendor')

    dict['product']=config.getint('Settings','idProduct')

    dict['interface']=config.getint('Settings', 'interface')

    r=config.options('Key Mappings')

    for item in r:
        if config.get('Key Mappings',item)!='': 
            dict[item]=config.get('Key Mappings',item)
            #print config.get('Key Mappings',item)
    return dict

def pyus():

    try:
        load_log()
        dict=load_config()
        join_int = lambda nums: int(''.join(str(i) for i in nums))
        #print dict

        dev = usb.core.find(idVendor=dict['vendor'], idProduct=dict['product'])
        interface=dict['interface']

        if dev is None:
            raise ValueError('Device not found')

        if dev.is_kernel_driver_active(interface) is True:
                #print "but we need to detach kernel driver"
                dev.detach_kernel_driver(interface)
        #dev.detatch_kernel_driver(interface) 
        # set the active configuration. With no arguments, the first
        # configuration will be the active one
        dev.set_configuration()

        # get an endpoint instance
        cfg = dev.get_active_configuration()
        interface_number = cfg[(0,0)].bInterfaceNumber
        alternate_setting = usb.control.get_interface(dev,interface_number)
        intf = usb.util.find_descriptor(
            cfg, bInterfaceNumber = interface_number,
            bAlternateSetting = alternate_setting
        )

        ep = usb.util.find_descriptor(
            intf,
            # match the first IN endpoint
            custom_match = \
            lambda e: \
                usb.util.endpoint_direction(e.bEndpointAddress) == \
                usb.util.ENDPOINT_IN
        )

        assert ep is not None
        #print 'packet details',ep.bEndpointAddress , ep.wMaxPacketSize

        while 1:
            try:
                data = dev.read(ep.bEndpointAddress, ep.wMaxPacketSize*2,interface,1000)
                data=data.tolist()
                key=join_int(data)
                #print "Key is " , key
                if  key in diction:

                    try:
                        args=shlex.split(dict[diction[key]])
                        #print args
                        p=subprocess.Popen(args, stderr=subprocess.STDOUT, stdout=subprocess.PIPE)
                        #print "Pressed key is ",diction[key]
                    except:
                        pass


            except usb.core.USBError as e:
                pass
    except:
        pass

pyus()
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.