forked from ob-f/OpenBot
-
Notifications
You must be signed in to change notification settings - Fork 3
Expand file tree
/
Copy pathUsbConnection.java
More file actions
231 lines (205 loc) · 7.49 KB
/
UsbConnection.java
File metadata and controls
231 lines (205 loc) · 7.49 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
// Created by Matthias Mueller - Intel Intelligent Systems Lab - 2020
package org.openbot.env;
import static java.nio.charset.StandardCharsets.UTF_8;
import android.app.PendingIntent;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.hardware.usb.UsbDevice;
import android.hardware.usb.UsbDeviceConnection;
import android.hardware.usb.UsbManager;
import android.os.AsyncTask;
import android.widget.Toast;
import androidx.localbroadcastmanager.content.LocalBroadcastManager;
import com.felhr.usbserial.UsbSerialDevice;
import com.felhr.usbserial.UsbSerialInterface;
import java.io.UnsupportedEncodingException;
import java.util.Map;
import org.openbot.utils.Constants;
public class UsbConnection {
private static final int USB_VENDOR_ID = 6790; // 0x2341; // 9025
private static final int USB_PRODUCT_ID = 29987; // 0x0001;
private static final Logger LOGGER = new Logger();
private final UsbManager usbManager;
// private UsbDevice usbDevice;
PendingIntent usbPermissionIntent;
public static final String ACTION_USB_PERMISSION = "UsbConnection.USB_PERMISSION";
private UsbDeviceConnection connection;
private UsbSerialDevice serialDevice;
private final LocalBroadcastManager localBroadcastManager;
private String buffer = "";
private final Context context;
private final int baudRate;
private boolean busy;
private int vendorId;
private int productId;
private String productName;
private String deviceName;
private String manufacturerName;
public UsbConnection(Context context, int baudRate) {
this.context = context;
this.baudRate = baudRate;
localBroadcastManager = LocalBroadcastManager.getInstance(this.context);
usbManager = (UsbManager) context.getSystemService(Context.USB_SERVICE);
usbPermissionIntent =
PendingIntent.getBroadcast(this.context, 0, new Intent(ACTION_USB_PERMISSION), 0);
}
private final UsbSerialInterface.UsbReadCallback callback =
data -> {
try {
String dataUtf8 = new String(data, "UTF-8");
buffer += dataUtf8;
int index;
while ((index = buffer.indexOf('\n')) != -1) {
final String dataStr = buffer.substring(0, index).trim();
buffer = buffer.length() == index ? "" : buffer.substring(index + 1);
AsyncTask.execute(() -> onSerialDataReceived(dataStr));
}
} catch (UnsupportedEncodingException e) {
LOGGER.e("Error receiving USB data");
}
};
private final BroadcastReceiver usbReceiver =
new BroadcastReceiver() {
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
if (ACTION_USB_PERMISSION.equals(action)) {
synchronized (this) {
UsbDevice usbDevice = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
if (intent.getBooleanExtra(UsbManager.EXTRA_PERMISSION_GRANTED, false)) {
if (usbDevice != null) {
// call method to set up device communication
startSerialConnection(usbDevice);
}
} else {
LOGGER.d("Permission denied for device " + usbDevice);
Toast.makeText(
UsbConnection.this.context,
"USB Host permission is required!",
Toast.LENGTH_LONG)
.show();
}
}
} else if (UsbManager.ACTION_USB_DEVICE_DETACHED.equals(action)) {
LOGGER.i("USB device detached");
UsbDevice device = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
if (device != null) {
stopUsbConnection();
}
}
}
};
public boolean startUsbConnection() {
IntentFilter localIntentFilter = new IntentFilter();
localIntentFilter.addAction(UsbManager.ACTION_USB_DEVICE_DETACHED);
localIntentFilter.addAction(ACTION_USB_PERMISSION);
localBroadcastManager.registerReceiver(usbReceiver, localIntentFilter);
context.registerReceiver(usbReceiver, localIntentFilter);
Map<String, UsbDevice> connectedDevices = usbManager.getDeviceList();
if (!connectedDevices.isEmpty()) {
for (UsbDevice usbDevice : connectedDevices.values()) {
// if (usbDevice.getVendorId() == USB_VENDOR_ID && usbDevice.getProductId() ==
// USB_PRODUCT_ID) {
LOGGER.i("Device found: " + usbDevice.getDeviceName());
if (usbManager.hasPermission(usbDevice)) {
return startSerialConnection(usbDevice);
} else {
usbManager.requestPermission(usbDevice, usbPermissionIntent);
Toast.makeText(context, "Please allow USB Host connection.", Toast.LENGTH_SHORT).show();
return false;
}
// }
}
}
LOGGER.w("Could not start USB connection - No devices found");
return false;
}
private boolean startSerialConnection(UsbDevice device) {
LOGGER.i("Ready to open USB device connection");
connection = usbManager.openDevice(device);
serialDevice = UsbSerialDevice.createUsbSerialDevice(device, connection);
boolean success = false;
if (serialDevice != null) {
if (serialDevice.open()) {
vendorId = device.getVendorId();
productId = device.getProductId();
productName = device.getProductName();
deviceName = device.getDeviceName();
manufacturerName = device.getManufacturerName();
serialDevice.setBaudRate(baudRate);
serialDevice.setDataBits(UsbSerialInterface.DATA_BITS_8);
serialDevice.setStopBits(UsbSerialInterface.STOP_BITS_1);
serialDevice.setParity(UsbSerialInterface.PARITY_NONE);
serialDevice.setFlowControl(UsbSerialInterface.FLOW_CONTROL_OFF);
serialDevice.read(callback);
LOGGER.i("Serial connection opened");
success = true;
} else {
LOGGER.w("Cannot open serial connection");
}
} else {
LOGGER.w("Could not create Usb Serial Device");
}
return success;
}
private void onSerialDataReceived(String data) {
// Add whatever you want here
LOGGER.i("Serial data received: " + data);
localBroadcastManager.sendBroadcast(
new Intent(Constants.USB_ACTION_DATA_RECEIVED)
.putExtra("from", "usb")
.putExtra("data", data));
}
public void stopUsbConnection() {
try {
if (serialDevice != null) {
serialDevice.close();
}
if (connection != null) {
connection.close();
}
} finally {
serialDevice = null;
connection = null;
}
localBroadcastManager.unregisterReceiver(usbReceiver);
try {
// Register or UnRegister your broadcast receiver here
context.unregisterReceiver(usbReceiver);
} catch (IllegalArgumentException e) {
e.printStackTrace();
}
}
public void send(String msg) {
if (isOpen() && !isBusy()) {
busy = true;
serialDevice.write(msg.getBytes(UTF_8));
busy = false;
}
}
public boolean isOpen() {
return connection != null;
}
public boolean isBusy() {
return busy;
}
public int getBaudRate() {
return baudRate;
}
public int getVendorId() {
return vendorId;
}
public int getProductId() {
return productId;
}
public String getProductName() {
return productName;
}
public String getDeviceName() {
return deviceName;
}
public String getManufacturerName() {
return manufacturerName;
}
}