關於OrangePI 串口通訊解決方案
阿新 • • 發佈:2018-09-18
port kref hose entryset 取數 else hubs har erro
前段時間因為需要做一個物聯網項目,需要使用到OrangePi, 定下裏的方案是使用gpio通訊,但是發現香橙派沒有通用的gpio驅動,所以只得這種選擇串口通信。網上找了一波,使用RS232+USB通信是最便宜的解決方案,驅動的話使用開源庫UsbSerial。也有之前Google 維護的庫,但我覺著不好用。所以在這裏獻醜寫上一篇博客。
導入
添加jitpack倉庫到你的build.gradle文件
allprojects { repositories { jcenter() maven { url "https://jitpack.io" } } }
添加依賴到你的app中
compile ‘com.github.felHR85:UsbSerial:4.5.2‘
設備支持
CP210X devices Default: 9600,8,1,None,flow off
CDC devices Default 115200,8,1,None,flow off
FTDI devices Default: 9600,8,1,None,flow off
PL2303 devices Default 9600,8,1,None,flow off
CH34x devices Default 9600,8,1,None,flow off
CP2130 SPI-USB
使用
- 新建Service
public class UsbService extends Service { public static final String ACTION_USB_READY = "com.felhr.connectivityservices.USB_READY"; public static final String ACTION_USB_ATTACHED = "android.hardware.usb.action.USB_DEVICE_ATTACHED"; public static final String ACTION_USB_DETACHED = "android.hardware.usb.action.USB_DEVICE_DETACHED"; public static final String ACTION_USB_NOT_SUPPORTED = "com.felhr.usbservice.USB_NOT_SUPPORTED"; public static final String ACTION_NO_USB = "com.felhr.usbservice.NO_USB"; public static final String ACTION_USB_PERMISSION_GRANTED = "com.felhr.usbservice.USB_PERMISSION_GRANTED"; public static final String ACTION_USB_PERMISSION_NOT_GRANTED = "com.felhr.usbservice.USB_PERMISSION_NOT_GRANTED"; public static final String ACTION_USB_DISCONNECTED = "com.felhr.usbservice.USB_DISCONNECTED"; public static final String ACTION_CDC_DRIVER_NOT_WORKING = "com.felhr.connectivityservices.ACTION_CDC_DRIVER_NOT_WORKING"; public static final String ACTION_USB_DEVICE_NOT_WORKING = "com.felhr.connectivityservices.ACTION_USB_DEVICE_NOT_WORKING"; public static final int MESSAGE_FROM_SERIAL_PORT = 0; public static final int CTS_CHANGE = 1; public static final int DSR_CHANGE = 2; public static final int SYNC_READ = 3; public static final String ACTION_USB_PERMISSION = "com.android.example.USB_PERMISSION"; private static final int BAUD_RATE = 9600; // BaudRate. Change this value if you need public static boolean SERVICE_CONNECTED = false; private IBinder binder = new UsbBinder(); private Context context; private Handler mHandler; private UsbManager usbManager; private UsbDevice device; private UsbDeviceConnection connection; private UsbSerialDevice serialPort; private boolean serialPortConnected; /* * Data received from serial port will be received here. * byte stream is converted to String and send to UI thread to * be treated there. */ private UsbSerialInterface.UsbReadCallback mCallback = new UsbSerialInterface.UsbReadCallback() { @Override public void onReceivedData(byte[] arg0) { Timber.e(arg0 + "-------接收數據"); try { String data = new String(arg0, "UTF-8"); // 上傳數據到服務器 if (mHandler != null) mHandler.obtainMessage(MESSAGE_FROM_SERIAL_PORT, data).sendToTarget(); } catch (UnsupportedEncodingException e) { e.printStackTrace(); } } }; /* * State changes in the CTS line will be received here * 電梯主板發送數據請求 */ private UsbSerialInterface.UsbCTSCallback ctsCallback = new UsbSerialInterface.UsbCTSCallback() { @Override public void onCTSChanged(boolean state) { if (mHandler != null) mHandler.obtainMessage(CTS_CHANGE).sendToTarget(); } }; /* * State changes in the DSR line will be received here * 發送請求命令到香橙派 */ private UsbSerialInterface.UsbDSRCallback dsrCallback = new UsbSerialInterface.UsbDSRCallback() { @Override public void onDSRChanged(boolean state) { if (mHandler != null) mHandler.obtainMessage(DSR_CHANGE).sendToTarget(); } }; /* * Different notifications from OS will be received here (USB attached, detached, permission responses...) * About BroadcastReceiver: http://developer.android.com/reference/android/content/BroadcastReceiver.html */ private final BroadcastReceiver usbReceiver = new BroadcastReceiver() { @Override public void onReceive(Context arg0, Intent arg1) { switch (Objects.requireNonNull(arg1.getAction())) { case ACTION_USB_PERMISSION: boolean granted = Objects.requireNonNull(arg1.getExtras()).getBoolean(UsbManager.EXTRA_PERMISSION_GRANTED); if (granted) // User accepted our USB connection. Try to open the device as a serial port { Intent intent = new Intent(ACTION_USB_PERMISSION_GRANTED); arg0.sendBroadcast(intent); connection = usbManager.openDevice(device); new ConnectionThread().start(); } else // User not accepted our USB connection. Send an Intent to the Main Activity { Intent intent = new Intent(ACTION_USB_PERMISSION_NOT_GRANTED); arg0.sendBroadcast(intent); } break; case ACTION_USB_ATTACHED: if (!serialPortConnected) findSerialPortDevice(); // A USB device has been attached. Try to open it as a Serial port break; case ACTION_USB_DETACHED: // Usb device was disconnected. send an intent to the Main Activity Intent intent = new Intent(ACTION_USB_DISCONNECTED); arg0.sendBroadcast(intent); if (serialPortConnected) { serialPort.syncClose(); } serialPortConnected = false; break; } } }; /* * onCreate will be executed when service is started. It configures an IntentFilter to listen for * incoming Intents (USB ATTACHED, USB DETACHED...) and it tries to open a serial port. */ @Override public void onCreate() { this.context = this; serialPortConnected = false; UsbService.SERVICE_CONNECTED = true; setFilter(); usbManager = (UsbManager) getSystemService(Context.USB_SERVICE); findSerialPortDevice(); EventBus.getDefault().register(this); } /* MUST READ about services * http://developer.android.com/guide/components/services.html * http://developer.android.com/guide/components/bound-services.html */ @Override public IBinder onBind(Intent intent) { return binder; } @Override public int onStartCommand(Intent intent, int flags, int startId) { return Service.START_NOT_STICKY; } @Override public void onDestroy() { super.onDestroy(); UsbService.SERVICE_CONNECTED = false; EventBus.getDefault().unregister(this); } /* * 寫數據到電梯主板 */ public void write(byte[] data, int timeout) { if (serialPort != null) { serialPort.syncWrite(data,timeout); } } /* * 改變波特率 */ public void changeBaudRate(int baudRate) { if (serialPort != null) serialPort.setBaudRate(baudRate); } public void setHandler(Handler mHandler) { this.mHandler = mHandler; } private void findSerialPortDevice() { // This snippet will try to open the first encountered usb device connected, excluding usb root hubs HashMap<String, UsbDevice> usbDevices = usbManager.getDeviceList(); if (!usbDevices.isEmpty()) { boolean keep = true; for (Map.Entry<String, UsbDevice> entry : usbDevices.entrySet()) { device = entry.getValue(); int deviceVID = device.getVendorId(); int devicePID = device.getProductId(); Timber.e("VendorId: " + deviceVID + "-----devicePID: " + devicePID); if (deviceVID != 0x1d6b && (devicePID != 0x0001 && devicePID != 0x0002 && devicePID != 0x0003)) { // There is a device connected to our Android device. Try to open it as a Serial Port. Timber.e("請求權限..."); requestUserPermission(); keep = false; } else { Timber.e("取消掛載設備..."); connection = null; device = null; } if (!keep) break; } if (!keep) { // There is no USB devices connected (but usb host were listed). Send an intent to MainActivity. Intent intent = new Intent(ACTION_NO_USB); sendBroadcast(intent); } } else { // There is no USB devices connected. Send an intent to MainActivity Intent intent = new Intent(ACTION_NO_USB); sendBroadcast(intent); } } private void setFilter() { IntentFilter filter = new IntentFilter(); filter.addAction(ACTION_USB_PERMISSION); filter.addAction(ACTION_USB_DETACHED); filter.addAction(ACTION_USB_ATTACHED); registerReceiver(usbReceiver, filter); } /* * 請求權限 */ private void requestUserPermission() { PendingIntent mPendingIntent = PendingIntent.getBroadcast(this, 0, new Intent(ACTION_USB_PERMISSION), 0); usbManager.requestPermission(device, mPendingIntent); } public class UsbBinder extends Binder { public UsbService getService() { return UsbService.this; } } /* * 單線程打開usb串口 * Although it should be a fast operation. moving usb operations away from UI thread is a good thing. */ private class ConnectionThread extends Thread { @Override public void run() { Timber.e("準備打開端口..."); serialPort = UsbSerialDevice.createUsbSerialDevice(device, connection); if (serialPort != null) { Timber.e("serialPort獲取成功..."); if (serialPort.syncOpen()) { Timber.e("serialPort打開成功..."); serialPortConnected = true; serialPort.setBaudRate(BAUD_RATE); serialPort.setDataBits(UsbSerialInterface.DATA_BITS_8); serialPort.setStopBits(UsbSerialInterface.STOP_BITS_1); serialPort.setParity(UsbSerialInterface.PARITY_NONE); /** * Current flow control Options: * UsbSerialInterface.FLOW_CONTROL_OFF * UsbSerialInterface.FLOW_CONTROL_RTS_CTS only for CP2102 and FT232 * UsbSerialInterface.FLOW_CONTROL_DSR_DTR only for CP2102 and FT232 */ serialPort.setFlowControl(UsbSerialInterface.FLOW_CONTROL_OFF); serialPort.read(mCallback); serialPort.getCTS(ctsCallback); serialPort.getDSR(dsrCallback); // new ReadThread().start(); Intent intent = new Intent(ACTION_USB_READY); context.sendBroadcast(intent); } else { // Serial port could not be opened, maybe an I/O error or if CDC driver was chosen, it does not really fit // Send an Intent to Main Activity if (serialPort instanceof CDCSerialDevice) { Intent intent = new Intent(ACTION_CDC_DRIVER_NOT_WORKING); context.sendBroadcast(intent); } else { Intent intent = new Intent(ACTION_USB_DEVICE_NOT_WORKING); context.sendBroadcast(intent); } } } else { // No driver for given device, even generic CDC driver could not be loaded Intent intent = new Intent(ACTION_USB_NOT_SUPPORTED); context.sendBroadcast(intent); } } } /** * 開線程讀取數據 */ private class ReadThread extends Thread { @Override public void run() { while (true) { byte[] buffer = new byte[100]; int n = serialPort.syncRead(buffer, 0); if (n > 0) { byte[] received = new byte[n]; System.arraycopy(buffer, 0, received, 0, n); Timber.e("媽的智障--------" + Arrays.toString(received)); String receivedStr = new String(received); Bundle bundle = new Bundle(); bundle.putByteArray("bytes", received); if (mHandler != null) { Message message = mHandler.obtainMessage(SYNC_READ, receivedStr); message.setData(bundle); message.sendToTarget(); } } } } } }
- 開啟服務並進行通信
private UsbService usbService;
private MyHandler mHandler;
private String mStatus = Command.STATUS_INIT;
private Disposable mLoginDispose;
private Disposable m111Dispose;
private final ServiceConnection usbConnection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName arg0, IBinder arg1) {
usbService = ((UsbService.UsbBinder) arg1).getService();
usbService.setHandler(mHandler);
usbService.changeBaudRate(9600);
}
@Override
public void onServiceDisconnected(ComponentName arg0) {
usbService = null;
}
};
/*
* 當前USB狀態...
*/
private final BroadcastReceiver mUsbReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
Timber.d("USB連接過程數據: " + intent.getDataString());
switch (Objects.requireNonNull(intent.getAction())) {
// USB PERMISSION GRANTED
case UsbService.ACTION_USB_PERMISSION_GRANTED:
Timber.d("USB準備完成....");
Toast.makeText(context, "USB Ready", Toast.LENGTH_SHORT).show();
//登錄
mStatus = Command.STATUS_INIT;
mLoginDispose = null;
m111Dispose = null;
EventBus.getDefault().post(new CycleReadEvent());
break;
case UsbService.ACTION_USB_PERMISSION_NOT_GRANTED:
Timber.d("USB沒有授權....");
Toast.makeText(context, "USB Permission not granted", Toast.LENGTH_SHORT).show();
break;
case UsbService.ACTION_NO_USB:
Timber.d("沒有USB連接....");
Toast.makeText(context, "No USB connected", Toast.LENGTH_SHORT).show();
break;
case UsbService.ACTION_USB_DISCONNECTED:
Timber.d("USB斷開連接.....");
Toast.makeText(context, "USB disconnected", Toast.LENGTH_SHORT).show();
break;
case UsbService.ACTION_USB_NOT_SUPPORTED:
Timber.d("USB設備不支持.....");
Toast.makeText(context, "USB device not supported", Toast.LENGTH_SHORT).show();
break;
}
}
};
/*
* 從USB中拿到的數據
* 在此處讀數據
*/
public class MyHandler extends Handler {
private final WeakReference<CallActivity> mActivity;
MyHandler(CallActivity activity) {
mActivity = new WeakReference<>(activity);
}
@Override
public void handleMessage(android.os.Message msg) {
switch (msg.what) {
case UsbService.MESSAGE_FROM_SERIAL_PORT:
byte[] data = (byte[]) msg.obj;
Timber.e("MESSAGE_FROM_SERIAL_PORT-接收到數據:" + Arrays.toString(data));
break;
case UsbService.CTS_CHANGE:
Toast.makeText(mActivity.get(), "CTS_CHANGE", Toast.LENGTH_LONG).show();
break;
case UsbService.DSR_CHANGE:
Toast.makeText(mActivity.get(), "DSR_CHANGE", Toast.LENGTH_LONG).show();
break;
case UsbService.SYNC_READ:
byte[] buffer1 = msg.getData().getByteArray("bytes");
Timber.e("SYNC_READ-接收到數據:" + Arrays.toString(buffer1));
Timber.e("SYNC_READ-接收到數據:" + msg.obj);
dealReadData(buffer1);
break;
}
}
}
};
private void startService(Class<?> service, ServiceConnection serviceConnection, Bundle extras) {
if (!UsbService.SERVICE_CONNECTED) {
Intent startService = new Intent(this, service);
if (extras != null && !extras.isEmpty()) {
Set<String> keys = extras.keySet();
for (String key : keys) {
String extra = extras.getString(key);
startService.putExtra(key, extra);
}
}
startService(startService);
}
Intent bindingIntent = new Intent(this, service);
bindService(bindingIntent, serviceConnection, Context.BIND_AUTO_CREATE);
}
private void setFilters() {
IntentFilter filter = new IntentFilter();
filter.addAction(UsbService.ACTION_USB_PERMISSION_GRANTED);
filter.addAction(UsbService.ACTION_NO_USB);
filter.addAction(UsbService.ACTION_USB_DISCONNECTED);
filter.addAction(UsbService.ACTION_USB_NOT_SUPPORTED);
filter.addAction(UsbService.ACTION_USB_PERMISSION_NOT_GRANTED);
registerReceiver(mUsbReceiver, filter);
}
@Override
protected void onResume() {
super.onResume();
setFilters();
startService(UsbService.class, usbConnection, null);
}
總結
大概使用教程就是這樣,今天來的有點晚。如果有不懂的可以關註我的公眾號留言咨詢。後面還是折騰gpio通信。如果有一起玩的可以聯系我一起交流。
歡迎長按下圖 -> 識別圖中二維碼或者微信掃一掃關註我的公眾號
關於OrangePI 串口通訊解決方案