com.openxc.interfaces.usb.UsbVehicleInterface Maven / Gradle / Ivy
package com.openxc.interfaces.usb;
import java.io.IOException;
import java.net.URI;
import android.annotation.TargetApi;
import android.app.PendingIntent;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.hardware.usb.UsbConstants;
import android.hardware.usb.UsbDevice;
import android.hardware.usb.UsbDeviceConnection;
import android.hardware.usb.UsbEndpoint;
import android.hardware.usb.UsbInterface;
import android.hardware.usb.UsbManager;
import android.util.Log;
import com.google.common.base.Objects;
import com.openxc.interfaces.UriBasedVehicleInterfaceMixin;
import com.openxc.interfaces.VehicleInterface;
import com.openxc.remote.RawMeasurement;
import com.openxc.sources.BytestreamDataSource;
import com.openxc.sources.DataSourceException;
import com.openxc.sources.DataSourceResourceException;
import com.openxc.sources.SourceCallback;
/**
* A vehicle data source reading measurements from an OpenXC USB device.
*
* This class looks for a USB device and expects to read OpenXC-compatible,
* newline separated JSON messages in USB bulk transfer packets.
*
* The device used (if different from the default) can be specified by passing
* an custom URI to the constructor. The expected format of this URI is defined
* in {@link UsbDeviceUtilities}.
*
* According to Android's USB device usage requirements, this class requests
* permission for the USB device from the user before accessing it. This may
* cause a pop-up dialog that the user must dismiss before the data source will
* become active.
*/
@TargetApi(12)
public class UsbVehicleInterface extends BytestreamDataSource
implements VehicleInterface {
private static final String TAG = "UsbVehicleInterface";
private static final int ENDPOINT_COUNT = 2;
public static final String ACTION_USB_PERMISSION =
"com.ford.openxc.USB_PERMISSION";
public static final String ACTION_USB_DEVICE_ATTACHED =
"com.ford.openxc.USB_DEVICE_ATTACHED";
private UsbManager mManager;
private UsbDeviceConnection mConnection;
private UsbInterface mInterface;
private UsbEndpoint mInEndpoint;
private UsbEndpoint mOutEndpoint;
private PendingIntent mPermissionIntent;
private URI mDeviceUri;
/**
* Construct an instance of UsbVehicleInterface with a receiver callback
* and custom device URI.
*
* If the device cannot be found at initialization, the object will block
* waiting for a signal to check again.
*
* @param context The Activity or Service context, used to get access to the
* Android UsbManager.
* @param callback An object implementing the
* SourceCallback that should receive data as it is
* received and parsed.
* @param deviceUri a USB device URI (see {@link UsbDeviceUtilities} for the
* format) to look for.
* @throws DataSourceException If the URI doesn't have the correct
* format
*/
public UsbVehicleInterface(SourceCallback callback, Context context,
URI deviceUri) throws DataSourceException {
super(callback, context);
mDeviceUri = createUri(deviceUri);
try {
mManager = (UsbManager) getContext().getSystemService(
Context.USB_SERVICE);
} catch(NoClassDefFoundError e) {
String message = "No USB service found on this device -- " +
"can't use USB vehicle interface";
Log.w(TAG, message);
throw new DataSourceException(message);
}
mPermissionIntent = PendingIntent.getBroadcast(getContext(), 0,
new Intent(ACTION_USB_PERMISSION), 0);
start();
}
/**
* Construct an instance of UsbVehicleInterface with a receiver callback
* and the default device URI.
*
* The default device URI is specified in {@link UsbDeviceUtilities}.
*
* @param context The Activity or Service context, used to get access to the
* Android UsbManager.
* @param callback An object implementing the
* SourceCallback that should receive data as it is
* received and parsed.
* @throws DataSourceException in exceptional circumstances, i.e.
* only if the default device URI is malformed.
*/
public UsbVehicleInterface(SourceCallback callback, Context context)
throws DataSourceException {
this(callback, context, null);
}
public UsbVehicleInterface(Context context) throws DataSourceException {
this(null, context);
}
public UsbVehicleInterface(Context context, String uriString)
throws DataSourceException {
this(null, context, createUri(uriString));
}
public synchronized void start() {
super.start();
IntentFilter filter = new IntentFilter(ACTION_USB_PERMISSION);
getContext().registerReceiver(mBroadcastReceiver, filter);
filter = new IntentFilter();
filter.addAction(ACTION_USB_DEVICE_ATTACHED);
getContext().registerReceiver(mBroadcastReceiver, filter);
filter = new IntentFilter();
filter.addAction(UsbManager.ACTION_USB_DEVICE_DETACHED);
getContext().registerReceiver(mBroadcastReceiver, filter);
initializeDevice();
}
@Override
public boolean isConnected() {
return mConnection != null && super.isConnected();
}
/**
* Unregister USB device intent broadcast receivers and stop waiting for a
* connection.
*
* This should be called before the object is given up to the garbage
* collector to avoid leaking a receiver in the Android framework.
*/
@Override
public void stop() {
super.stop();
try {
getContext().unregisterReceiver(mBroadcastReceiver);
} catch(IllegalArgumentException e) {
Log.d(TAG, "Unable to unregster receiver when stopping, probably not registered");
}
}
public boolean receive(RawMeasurement command) {
if(isConnected()) {
String message = command.serialize() + "\u0000";
Log.d(TAG, "Writing string to USB: " + message);
byte[] bytes = message.getBytes();
return write(bytes);
}
return false;
}
public boolean setResource(String otherUri) throws DataSourceException {
if(mDeviceUri == UsbDeviceUtilities.DEFAULT_USB_DEVICE_URI
&& otherUri != null &&
!UriBasedVehicleInterfaceMixin.sameResource(mDeviceUri,
otherUri)) {
mDeviceUri = createUri(otherUri);
stop();
start();
return true;
}
return false;
}
@Override
public String toString() {
return Objects.toStringHelper(this)
.add("device", mDeviceUri)
.add("connection", mConnection)
.add("in_endpoint", mInEndpoint)
.add("out_endpoint", mOutEndpoint)
.toString();
}
protected int read(byte[] bytes) throws IOException {
mConnectionLock.readLock().lock();
int bytesRead = 0;
try {
if(isConnected()) {
bytesRead = mConnection.bulkTransfer(mInEndpoint, bytes, bytes.length, 0);
}
} finally {
mConnectionLock.readLock().unlock();
}
return bytesRead;
}
protected String getTag() {
return TAG;
}
private void initializeDevice() {
try {
connectToDevice(mManager, mDeviceUri);
} catch(DataSourceException e) {
Log.i(TAG, "Unable to load USB device -- " +
"waiting for it to appear", e);
}
}
private boolean write(byte[] bytes) {
if(mConnection != null) {
if(mOutEndpoint != null) {
Log.d(TAG, "Writing bytes to USB: " + bytes);
int transferred = mConnection.bulkTransfer(
mOutEndpoint, bytes, bytes.length, 0);
if(transferred < 0) {
Log.w(TAG, "Unable to write CAN message to USB endpoint, error "
+ transferred);
return false;
}
} else {
Log.w(TAG, "No OUT endpoint available on USB device, " +
"can't send write command");
return false;
}
} else {
return false;
}
return true;
}
private void connectToDevice(UsbManager manager, URI deviceUri)
throws DataSourceResourceException {
connectToDevice(manager,
UsbDeviceUtilities.vendorFromUri(mDeviceUri),
UsbDeviceUtilities.productFromUri(mDeviceUri));
}
private void connectToDevice(UsbManager manager, int vendorId,
int productId) throws DataSourceResourceException {
UsbDevice device = findDevice(manager, vendorId, productId);
if(manager.hasPermission(device)) {
Log.d(TAG, "Already have permission to use " + device);
openConnection(device);
} else {
Log.d(TAG, "Requesting permission for " + device);
manager.requestPermission(device, mPermissionIntent);
}
}
private UsbDeviceConnection setupDevice(UsbManager manager,
UsbDevice device) throws UsbDeviceException {
if(device.getInterfaceCount() != 1) {
throw new UsbDeviceException("USB device didn't have an " +
"interface for us to open");
}
UsbInterface iface = null;
for(int i = 0; i < device.getInterfaceCount(); i++) {
iface = device.getInterface(i);
if(iface.getEndpointCount() == ENDPOINT_COUNT) {
break;
}
}
if(iface == null) {
Log.w(TAG, "Unable to find a USB device interface with the " +
"expected number of endpoints (" + ENDPOINT_COUNT + ")");
return null;
}
for(int i = 0; i < iface.getEndpointCount(); i++) {
UsbEndpoint endpoint = iface.getEndpoint(i);
if(endpoint.getType() ==
UsbConstants.USB_ENDPOINT_XFER_BULK) {
if(endpoint.getDirection() == UsbConstants.USB_DIR_IN) {
Log.d(TAG, "Found IN endpoint " + endpoint);
mInEndpoint = endpoint;
} else {
Log.d(TAG, "Found OUT endpoint " + endpoint);
mOutEndpoint = endpoint;
}
}
if(mInEndpoint != null && mOutEndpoint != null) {
break;
}
}
return openInterface(manager, device, iface);
}
private UsbDevice findDevice(UsbManager manager, int vendorId,
int productId) throws DataSourceResourceException {
Log.d(TAG, "Looking for USB device with vendor ID " + vendorId +
" and product ID " + productId);
for(UsbDevice candidateDevice : manager.getDeviceList().values()) {
if(candidateDevice.getVendorId() == vendorId
&& candidateDevice.getProductId() == productId) {
Log.d(TAG, "Found USB device " + candidateDevice);
return candidateDevice;
}
}
throw new DataSourceResourceException("USB device with vendor " +
"ID " + vendorId + " and product ID " + productId +
" not found");
}
private UsbDeviceConnection openInterface(UsbManager manager,
UsbDevice device, UsbInterface iface)
throws UsbDeviceException {
UsbDeviceConnection connection = manager.openDevice(device);
if(connection == null) {
throw new UsbDeviceException("Couldn't open a connection to " +
"device -- user may not have given permission");
}
mInterface = iface;
connection.claimInterface(mInterface, true);
return connection;
}
/**
* TODO we oddly need to "prime" this endpoint from Android because the
* first message we send, if it's over 1 packet in size, we only get the
* last packet.
*/
private void primeOutput() {
Log.d(TAG, "Priming output endpoint");
write(new String("prime\u0000").getBytes());
}
private void openConnection(UsbDevice device) {
if (device != null) {
mConnectionLock.writeLock().lock();
try {
mConnection = setupDevice(mManager, device);
connected();
Log.i(TAG, "Connected to USB device with " +
mConnection);
} catch(UsbDeviceException e) {
Log.w("Couldn't open USB device", e);
} finally {
mConnectionLock.writeLock().unlock();
}
} else {
Log.d(TAG, "Permission denied for device " + device);
}
}
protected void connected() {
super.connected();
primeOutput();
}
protected void connect() throws DataSourceException { }
protected void disconnect() {
if(!isConnected()) {
return;
}
Log.d(TAG, "Closing connection " + mConnection +
" with USB device");
mConnectionLock.writeLock().lock();
try {
mConnection.close();
mConnection = null;
mInEndpoint = null;
mOutEndpoint = null;
mInterface = null;
disconnected();
} finally {
mConnectionLock.writeLock().unlock();
}
}
private BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
if (ACTION_USB_PERMISSION.equals(action)) {
UsbDevice device = (UsbDevice) intent.getParcelableExtra(
UsbManager.EXTRA_DEVICE);
if(intent.getBooleanExtra(
UsbManager.EXTRA_PERMISSION_GRANTED, false)) {
openConnection(device);
} else {
Log.i(TAG, "User declined permission for device " +
device);
}
} else if(ACTION_USB_DEVICE_ATTACHED.equals(action)) {
Log.d(TAG, "Device attached");
try {
connectToDevice(mManager, mDeviceUri);
} catch(DataSourceException e) {
Log.i(TAG, "Unable to load USB device -- waiting for it " +
"to appear", e);
}
} else if(UsbManager.ACTION_USB_DEVICE_DETACHED.equals(action)) {
Log.d(TAG, "Device detached");
disconnect();
}
}
};
private static URI createUri(String uriString) throws DataSourceException {
URI uri;
if(uriString == null) {
uri = null;
} else {
uri = UriBasedVehicleInterfaceMixin.createUri(uriString);
}
return createUri(uri);
}
private static URI createUri(URI uri) throws DataSourceResourceException {
if(uri == null) {
uri = UsbDeviceUtilities.DEFAULT_USB_DEVICE_URI;
Log.i(TAG, "No USB device specified -- using default " +
uri);
}
if(!validateResource(uri)) {
throw new DataSourceResourceException(
"USB device URI must have the usb:// scheme");
}
// will throw an exception if not in the correct format
UsbDeviceUtilities.vendorFromUri(uri);
UsbDeviceUtilities.productFromUri(uri);
return uri;
}
private static boolean validateResource(URI uri) {
return uri.getScheme() != null && uri.getScheme().equals("usb");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy