2008/10/14 からのアクセス回数 11808
avr/USB接続でlibusbを使って自作のUSBデバイスにアクセスすることができるようになりました。 次は、javaからこのUSBデバイスを操作してみます。
まず、libusbのjavaラッパーが必要です。 世の中には、自分がやりたいと思うことをすでにやっている人がいますから、自分で最初から作る前に
libusb java
でグーグル検索すると
を見つけることができました。
Windowsの人は、
私はMacOSXなので、SVNからソースをダウンロードしてコンパイルすることにしました。
Eclipseを使って、
SVNから
をチェックアウトします。
次に、LibusbJavaを使ってMacOSX用のダイナミックライブラリを作成します。
LibusbJavaプロジェクトの build.xmlの22行に以下の1行を追加してください。
<property name="version.mac" value="${version.major}.${version.minor}.${version.micro}" />
次にbuild.xmlを選択し、右クリックでRun As->Ant build...を選択します。 ターゲットして、Macを選択するとLibusbJava.jnilibが作成されます。
同様に、ch.ntb.usbプロジェクトでbuild.xmlを使ってjarファイルを作成します。
Eclipseで新規プロジェクトで、
テストプログラムUSBTiny45.javaを作成します。
static long usbOpenDevice(short idvendor, short idproduct) throws USBException {
Usb_Device dev = null;
Usb_Bus bus = USB.getBus();
long usbHandle = 0L;
while (bus != null) {
dev = bus.getDevices();
while (dev != null) {
usbHandle = LibusbJava.usb_open(dev);
if (usbHandle != 0) {
Usb_Device_Descriptor devDesc = dev.getDescriptor();
String retp = LibusbJava.usb_get_string_simple(usbHandle,
(int)devDesc.getIProduct());
String retm = LibusbJava.usb_get_string_simple(usbHandle,
(int)devDesc.getIManufacturer());
if (retp != null && retm != null) {
if (devDesc.getIdVendor() == idvendor
&& devDesc.getIdProduct() == idproduct) {
return usbHandle;
}
}
}
dev = dev.getNext();
}
bus = bus.getNext();
}
LibusbJava.usb_close(usbHandle);
return 0L;
}
static private void testUsbOpenDevice() throws Exception {
byte[] buffer = new byte[2312];
int i=3, j=4, k=5, l=6, m=7, n=8, o=9;
int ret;
long devHandle = usbOpenDevice(vendorId, productId);
if (devHandle == 0) {
System.err.println("usbOpenDevice failed\n");
return;
}
ret = LibusbJava.usb_control_msg(devHandle,
USB.REQ_TYPE_TYPE_VENDOR|USB.REQ_TYPE_RECIP_DEVICE
|USB_ENDPOINT_IN,
i, j+256*k, l+256*m, buffer, n+256*o, 5000);
System.out.format("ret=%d \n", ret);
for (int p=0; p < ret; p++) {
System.out.format("buffer[%d]=%d \n", p, buffer[p]);
}
LibusbJava.usb_close(devHandle);
}
とします。これでよしとmain関数を実行したのですが、うまく動きません。
ことが分かりました。
USBデバイスが認識されているかは、ch.ntb.usb-0.5.7.jarのUsbViewのmainメソッドを実行すると分かります。
めでたく、実行結果がCと同じく
ret=7 buffer[0]=9 buffer[1]=3 buffer[2]=4 buffer[3]=5 buffer[4]=6 buffer[5]=7 buffer[6]=8
になりました。
Device.java#initDeviceの114行目の
updateMaxPacketSize(device);
をdevDesc.getAltinterface()が-1の時には、コールしないように修正すれば、
テストプログラムは、以下のように簡単になります。
static private void testDeviceAndUSB() throws Exception {
byte[] buffer = new byte[2312];
int i=3, j=4, k=5, l=6, m=7, n=8, o=9;
int ret;
Device device = USB.getDevice(vendorId, productId);
if (device == null) {
System.err.println("usbOpenDevice failed\n");
return;
}
device.open(1, 0, -1);
ret = device.controlMsg(
USB.REQ_TYPE_TYPE_VENDOR|USB.REQ_TYPE_RECIP_DEVICE
|USB_ENDPOINT_IN,
i, j+256*k, l+256*m, buffer, n+256*o, 5000, false);
System.out.format("ret=%d \n", ret);
for (int p=0; p < ret; p++) {
System.out.format("buffer[%d]=%d \n", p, buffer[p]);
}
device.close();
}
作成したUSBTiny45.javaは以下からダウンロードできます。
2代目のMacMiniのOSをSonw Leopardにアップグレードしたら、Tigerで作ったLibusbJava.jnilibが動作しなくなってしまいました。
仕方なく、もう一度antを実行したのですが、ダメです。ここでは、試行錯誤の結果どうにかLibusbJava.jnilibができたので、その方法をメモしておきます。
どうせなら、最新のソースにアップすることにしました。
チェックアウトは、以下のように行います。
$ svn co https://libusbjava.svn.sourceforge.net/svnroot/libusbjava/trunk/LibusbJava/
OSの変更でMacPortのlibusbを作り直しました。
LibusbJavaのページには、0.1の古いバージョンを使うとあるので、llibusb-compatもインストールします。 また、libusb-compatには、libusbppが含まれないのでlibusb-legacyもインストールします。
$ sudo port install libusb $ sudo port install libusb-compat $ sudo port install libusb-legacy
次に、libusbJavaが/usr/local/lib/libusb.dylib、/usr/local/lib/libusbpp.dylibを参照しているので、 シンボリックリンクを作ります。
$ cd /usr/local/lib $ sudo ln -s /opt/local/lib/libusb-0.1.4.dylib ./libusb.dylib $ sudo ln -s /opt/local/lib/libusb-legacy/libusbpp-legacy.dylib ./libusbpp.dylib
ソースファイルをチェックアウトした場所に戻って、コンパイルをします。
$ ant mac
途中省略
[exec] /usr/libexec/gcc/i686-apple-darwin10/4.2.1/collect2 -dynamic
-dylib -dylib_current_version 0.2.4.0 -arch x86_64 -macosx_version_min 10.6.6
-weak_reference_mismatches non-weak -o libusbJava.jnilib
-L/usr/lib/gcc/i686-apple-darwin10/4.2.1/x86_64
-L/usr/lib/gcc/i686-apple-darwin10/4.2.1/x86_64
-L/usr/lib/i686-apple-darwin10/4.2.1
-L/usr/lib/gcc/i686-apple-darwin10/4.2.1
-L/usr/lib/gcc/i686-apple-darwin10/4.2.1
-L/usr/lib/gcc/i686-apple-darwin10/4.2.1/../../../i686-apple-darwin10/4.2.1
-L/usr/lib/gcc/i686-apple-darwin10/4.2.1/../../..
/var/folders/Xo/XoxQsiGT2RWIck+BYuxhKU+++TI/-Tmp-//cc725VBz.o
-lstdc++ /usr/local/lib/libusb.dylib /usr/local/lib/libusbpp.dylib -lSystem -lgcc -lSystem
[exec] ld: malformed version number: 0.2.4.0
[exec] collect2: ld returned 1 exit status
[exec] Result: 1
とcollect2の実行でldがmalformed versionエラーを出力します。
原因は、collect2の引数の
-dylib_current_version 0.2.4.0
オプションのバージョンにあるようで、これを除くとエラーはでません。
そこで、build.xmlの90行目の
<arg line="-current_version ${version}" />
を削除して、
$ ant mac
を実行すると、無事Snow Leopardで動くLibusbJava.jnilibができます。
念のため、作成したLibusbJava.jnilibを以下に置いておきます。
この記事は、
皆様のご意見、ご希望をお待ちしております。