我们有一个特定于供应商的血压设备,我们希望通过Android便捷设备进行配置 这是USB端口。 此外,我还为Windows(与WinUSB对应)编写了与 设备,据我所见,Android和WinUSB api(当然)有很多 相似之处,并且应该可以很好地工作。我也可以在Android上打开设备 并提取端点(批量端点)。该设备支持中断和批量读/写端点。
例如,我还可以通过controlTransfer成功地将命令写入设备。
private void tryGetDescriptor() {
byte[] buffer = new byte[18];
// https://www.beyondlogic.org/usbnutshell/usb6.shtml#SetupPacket standard device requests
int retVal = m_devCon.controlTransfer(0x80, 0x06, 0x0100, 0, buffer, 0x12, TIMEOUT);
if(retVal < 0) {
Debug("something failed");
} else {
Debug( "check buffer " + String.valueOf( retVal ) + " arr: " + Arrays.toString(buffer) );
}
}
或我在Delphi中的参考代码所做的一个controlTransfer:
private void startPeripheral() {
int retVal = m_devCon.controlTransfer(0x40, 2, 0, m_readPkgSize, null, 0, 0);
if(retVal < 0) {
Debug("StartPeripheral failed!");
} else {
Debug("Control Transfer returned " + String.valueOf(retVal));
}
}
我现在面临的问题是,根据写入端点,我可能会向设备发送命令 但我没有得到回报(设备至少应回显命令)
在这里处理该线程(我知道同步的东西实际上不是Java的,所以请 对我耐心...)
private UsbDeviceConnection m_devCon;
private UsbEndpoint m_epRead;
private UsbEndpoint m_epWrite;
private String m_command = "";
private boolean terminated = false;
public void sigTerminate(){
terminated = true;
m_command = "";
}
public void init( UsbDeviceConnection devCon, UsbEndpoint readEP, UsbEndpoint writeEP ) {
this.m_devCon = devCon;
this.m_epRead = readEP;
this.m_epWrite = writeEP;
}
public void asyncSendCmd( String cmd ) {
Debug("AsyncSendCmd: " + cmd);
m_command = new String(cmd);
}
private void sendRec() {
Debug("Run thread");
byte[] buffer = new byte[m_readPkgSize];
try {
//while( req.queue( rBuf ) ) {
while (true) {
try {
while (m_command == "" && !terminated)
sleep(10);
Debug("Something found" + m_command);
} catch (InterruptedException f) {
Thread.currentThread().interrupt();
Debug("Thread interruped: " + f.getMessage());
}
String cpyCmd = new String(BR102_RESET );
m_command = "";
if (!terminated && isConnected()) {
// not threadsave but what shels
Debug("Writing command: " + cpyCmd);
byte[] writeBuf = cpyCmd.getBytes();
//int lenWrite = m_devCon.bulkTransfer(m_epWrite, writeBuf, writeBuf.length, TIMEOUT);
//if (lenWrite != writeBuf.length)
// throw new IOException("Write failed with len: " + String.valueOf(lenWrite));
// sleep(48);
UsbRequest wreq = new UsbRequest();
wreq.initialize( m_devCon, m_epWrite);
ByteBuffer wBuf = ByteBuffer.wrap(writeBuf);
wreq.queue( wBuf );
final UsbRequest responsew = m_devCon.requestWait( TIMEOUT );
if(responsew == null) {
Debug("no write response");
break;
}
int lenWrite = wBuf.position();
Debug("Wrote: " + String.valueOf(lenWrite));
wreq.close();
UsbRequest req = new UsbRequest();
req.initialize(m_devCon, m_epRead);
ByteBuffer rBuf = ByteBuffer.wrap( buffer );
req.queue( rBuf );
//int lenRead = m_devCon.bulkTransfer(m_epRead, buffer, m_readPkgSize, TIMEOUT);
final UsbRequest response = m_devCon.requestWait(TIMEOUT);
if(response == null) {
Debug("No Resonse");
break;
}
int lenRead = rBuf.position();
if (lenRead > 0) Debug("lenRead: " + String.valueOf(lenRead));
else Debug("lenRead: " + String.valueOf(lenRead));
req.close();
}
//rBuf.rewind();
//req.close();
}
} catch (Exception e) {
Debug("Error in usb read thread: " + e.toString());// e.getMessage());
}
}
@Override
public void run() {
sendRec();
}
}
可以看到我尝试了很多事情。那是我提出的最复杂的问题。基本上 我的第一次尝试是或多或少具有相同行为的bulkTransfers ...
请注意,“ Reset \ n”命令可以发送2次,然后即使写入失败...
为了完整起见,这里是usb初始化例程:
private void startUSB(UsbDevice device) throws IllegalArgumentException {
UsbManager usbManager = (UsbManager) getSystemService(Context.USB_SERVICE);
if(!usbManager.hasPermission( device )) {
usbManager.requestPermission( device, m_permissionIntent );
return;
}
try {
if( usbReader != null) {
Debug("WTF usbReader already started!!!!");
usbReader.sigTerminate();
usbReader.join();
usbReader = null;
}
// ########################################################
// #### we the permission -> get the interfaces and check for the endpoints
m_device = device;
int numInterfaces = m_device.getInterfaceCount();
if(numInterfaces > 0) {
Debug("Number of interfaces: " + String.valueOf(numInterfaces));
m_devIntf = m_device.getInterface(0);
Debug(String.valueOf(m_devIntf.getInterfaceProtocol()));
Debug("Number of endpoints: " + m_devIntf.getEndpointCount());
m_epRead = null;
m_epWrite = null;
for (int i = 0; i < m_devIntf.getEndpointCount(); i++) {
UsbEndpoint endPoint = m_devIntf.getEndpoint(i);
Debug(String.valueOf(i + 1) + ": Endpoint " + endPoint.toString());
if (endPoint.getDirection() == UsbConstants.USB_DIR_IN && endPoint.getType() == UsbConstants.USB_ENDPOINT_XFER_BULK) {
m_epRead = endPoint;
}
if (endPoint.getDirection() == UsbConstants.USB_DIR_OUT && endPoint.getType() == UsbConstants.USB_ENDPOINT_XFER_BULK) {
m_epWrite = endPoint;
}
}
if (m_epRead == null || m_epWrite == null)
throw new IllegalArgumentException("No bulk read or write endpoint found");
Debug("Got endpoints");
// ############################################################################
// okidoki we have endpoints and we have an interface -> open the device
m_devCon = usbManager.openDevice(m_device);
Debug("Got device");
if (m_devCon == null)
throw new IllegalArgumentException("No Deviceconnection?!");
if (!m_devCon.claimInterface(m_devIntf, false))
throw new IllegalArgumentException("Claim failed");
Debug("All fine - now we can start the conversation");
m_readPkgSize = m_epRead.getMaxPacketSize();
m_writePkgSize = m_epWrite.getMaxPacketSize();
tryGetDescriptor();
// ############################################################################
// ### init reading and wrinting
usbReader = new UsbReadThread();
usbReader.init( m_devCon, m_epRead, m_epWrite );
usbReader.start();
startPeripheral();
} else {
Debug("No interfaces");
}
} catch(Exception e)
{
Debug( "Device connection failed with " + e.getMessage() );
// release
closeUSB();
return;
}
}
答案 0 :(得分:0)
好的-在我有机会看一下固件之后,我终于意识到-尽管发布了4个USB端点-我只能在Interrupt Pipe端点上写,并且只能从bulktransfer端点读取。