android app在系統重啟繞過USB授權對話方塊,自動獲取USB許可權
最近由於專案需要參考了很多關於不修改android原始碼情況下實現遮蔽USB授權對話方塊的博文,結合自身實踐闡述給出細節的實現過程。
當前實現的策略是開發一個android內部服務,用於響應需要使用USB授權的app的申請,app需要在使用USB之前主動申請我們自行開發的授權,走在預設授權對話方塊觸發之前提出申請。
一、建立一個android內部服務apk
1)具體使用的AndroidManifest.xm
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.harvbot.usbpermissionissuer"> <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:supportsRtl="true" android:theme="@style/AppTheme"> <receiver android:name=".LaunchReceiver"> <intent-filter> <action android:name="ACTION_USB_PERMISSION_ISSUER" /> </intent-filter> </receiver> <service android:enabled="true" android:name=".StubService" /> </application> <uses-permission android:name="android.permission.MANAGE_USB" /> <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> </manifest>
2)建立android.hardware.usb.IUsbManager.java替換系統原始碼
/* * This file is auto-generated. DO NOT MODIFY. * Original file: frameworks/base/core/java/android/hardware/usb/IUsbManager.aidl */ package android.hardware.usb; /** @hide */ public interface IUsbManager extends android.os.IInterface { /** Local-side IPC implementation stub class. */ public static abstract class Stub extends android.os.Binder implements android.hardware.usb.IUsbManager { private static final java.lang.String DESCRIPTOR = "android.hardware.usb.IUsbManager"; /** Construct the stub at attach it to the interface. */ public Stub() { this.attachInterface(this, DESCRIPTOR); } /** * Cast an IBinder object into an android.hardware.usb.IUsbManager interface, * generating a proxy if needed. */ public static android.hardware.usb.IUsbManager asInterface(android.os.IBinder obj) { if ((obj==null)) { return null; } android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR); if (((iin!=null)&&(iin instanceof android.hardware.usb.IUsbManager))) { return ((android.hardware.usb.IUsbManager)iin); } return new android.hardware.usb.IUsbManager.Stub.Proxy(obj); } @Override public android.os.IBinder asBinder() { return this; } @Override public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException { switch (code) { case INTERFACE_TRANSACTION: { reply.writeString(DESCRIPTOR); return true; } case TRANSACTION_getDeviceList: { data.enforceInterface(DESCRIPTOR); android.os.Bundle _arg0; _arg0 = new android.os.Bundle(); this.getDeviceList(_arg0); reply.writeNoException(); if ((_arg0!=null)) { reply.writeInt(1); _arg0.writeToParcel(reply, android.os.Parcelable.PARCELABLE_WRITE_RETURN_VALUE); } else { reply.writeInt(0); } return true; } case TRANSACTION_openDevice: { data.enforceInterface(DESCRIPTOR); java.lang.String _arg0; _arg0 = data.readString(); android.os.ParcelFileDescriptor _result = this.openDevice(_arg0); reply.writeNoException(); if ((_result!=null)) { reply.writeInt(1); _result.writeToParcel(reply, android.os.Parcelable.PARCELABLE_WRITE_RETURN_VALUE); } else { reply.writeInt(0); } return true; } case TRANSACTION_getCurrentAccessory: { data.enforceInterface(DESCRIPTOR); android.hardware.usb.UsbAccessory _result = this.getCurrentAccessory(); reply.writeNoException(); if ((_result!=null)) { reply.writeInt(1); _result.writeToParcel(reply, android.os.Parcelable.PARCELABLE_WRITE_RETURN_VALUE); } else { reply.writeInt(0); } return true; } case TRANSACTION_openAccessory: { data.enforceInterface(DESCRIPTOR); android.hardware.usb.UsbAccessory _arg0; if ((0!=data.readInt())) { _arg0 = android.hardware.usb.UsbAccessory.CREATOR.createFromParcel(data); } else { _arg0 = null; } android.os.ParcelFileDescriptor _result = this.openAccessory(_arg0); reply.writeNoException(); if ((_result!=null)) { reply.writeInt(1); _result.writeToParcel(reply, android.os.Parcelable.PARCELABLE_WRITE_RETURN_VALUE); } else { reply.writeInt(0); } return true; } case TRANSACTION_setDevicePackage: { data.enforceInterface(DESCRIPTOR); android.hardware.usb.UsbDevice _arg0; if ((0!=data.readInt())) { _arg0 = android.hardware.usb.UsbDevice.CREATOR.createFromParcel(data); } else { _arg0 = null; } java.lang.String _arg1; _arg1 = data.readString(); int _arg2; _arg2 = data.readInt(); this.setDevicePackage(_arg0, _arg1, _arg2); reply.writeNoException(); return true; } case TRANSACTION_setAccessoryPackage: { data.enforceInterface(DESCRIPTOR); android.hardware.usb.UsbAccessory _arg0; if ((0!=data.readInt())) { _arg0 = android.hardware.usb.UsbAccessory.CREATOR.createFromParcel(data); } else { _arg0 = null; } java.lang.String _arg1; _arg1 = data.readString(); int _arg2; _arg2 = data.readInt(); this.setAccessoryPackage(_arg0, _arg1, _arg2); reply.writeNoException(); return true; } case TRANSACTION_hasDevicePermission: { data.enforceInterface(DESCRIPTOR); android.hardware.usb.UsbDevice _arg0; if ((0!=data.readInt())) { _arg0 = android.hardware.usb.UsbDevice.CREATOR.createFromParcel(data); } else { _arg0 = null; } boolean _result = this.hasDevicePermission(_arg0); reply.writeNoException(); reply.writeInt(((_result)?(1):(0))); return true; } case TRANSACTION_hasAccessoryPermission: { data.enforceInterface(DESCRIPTOR); android.hardware.usb.UsbAccessory _arg0; if ((0!=data.readInt())) { _arg0 = android.hardware.usb.UsbAccessory.CREATOR.createFromParcel(data); } else { _arg0 = null; } boolean _result = this.hasAccessoryPermission(_arg0); reply.writeNoException(); reply.writeInt(((_result)?(1):(0))); return true; } case TRANSACTION_requestDevicePermission: { data.enforceInterface(DESCRIPTOR); android.hardware.usb.UsbDevice _arg0; if ((0!=data.readInt())) { _arg0 = android.hardware.usb.UsbDevice.CREATOR.createFromParcel(data); } else { _arg0 = null; } java.lang.String _arg1; _arg1 = data.readString(); android.app.PendingIntent _arg2; if ((0!=data.readInt())) { _arg2 = android.app.PendingIntent.CREATOR.createFromParcel(data); } else { _arg2 = null; } this.requestDevicePermission(_arg0, _arg1, _arg2); reply.writeNoException(); return true; } case TRANSACTION_requestAccessoryPermission: { data.enforceInterface(DESCRIPTOR); android.hardware.usb.UsbAccessory _arg0; if ((0!=data.readInt())) { _arg0 = android.hardware.usb.UsbAccessory.CREATOR.createFromParcel(data); } else { _arg0 = null; } java.lang.String _arg1; _arg1 = data.readString(); android.app.PendingIntent _arg2; if ((0!=data.readInt())) { _arg2 = android.app.PendingIntent.CREATOR.createFromParcel(data); } else { _arg2 = null; } this.requestAccessoryPermission(_arg0, _arg1, _arg2); reply.writeNoException(); return true; } case TRANSACTION_grantDevicePermission: { data.enforceInterface(DESCRIPTOR); android.hardware.usb.UsbDevice _arg0; if ((0!=data.readInt())) { _arg0 = android.hardware.usb.UsbDevice.CREATOR.createFromParcel(data); } else { _arg0 = null; } int _arg1; _arg1 = data.readInt(); this.grantDevicePermission(_arg0, _arg1); reply.writeNoException(); return true; } case TRANSACTION_grantAccessoryPermission: { data.enforceInterface(DESCRIPTOR); android.hardware.usb.UsbAccessory _arg0; if ((0!=data.readInt())) { _arg0 = android.hardware.usb.UsbAccessory.CREATOR.createFromParcel(data); } else { _arg0 = null; } int _arg1; _arg1 = data.readInt(); this.grantAccessoryPermission(_arg0, _arg1); reply.writeNoException(); return true; } case TRANSACTION_hasDefaults: { data.enforceInterface(DESCRIPTOR); java.lang.String _arg0; _arg0 = data.readString(); int _arg1; _arg1 = data.readInt(); boolean _result = this.hasDefaults(_arg0, _arg1); reply.writeNoException(); reply.writeInt(((_result)?(1):(0))); return true; } case TRANSACTION_clearDefaults: { data.enforceInterface(DESCRIPTOR); java.lang.String _arg0; _arg0 = data.readString(); int _arg1; _arg1 = data.readInt(); this.clearDefaults(_arg0, _arg1); reply.writeNoException(); return true; } case TRANSACTION_setCurrentFunction: { data.enforceInterface(DESCRIPTOR); java.lang.String _arg0; _arg0 = data.readString(); boolean _arg1; _arg1 = (0!=data.readInt()); this.setCurrentFunction(_arg0, _arg1); reply.writeNoException(); return true; } case TRANSACTION_setMassStorageBackingFile: { data.enforceInterface(DESCRIPTOR); java.lang.String _arg0; _arg0 = data.readString(); this.setMassStorageBackingFile(_arg0); reply.writeNoException(); return true; } case TRANSACTION_allowUsbDebugging: { data.enforceInterface(DESCRIPTOR); boolean _arg0; _arg0 = (0!=data.readInt()); java.lang.String _arg1; _arg1 = data.readString(); this.allowUsbDebugging(_arg0, _arg1); reply.writeNoException(); return true; } case TRANSACTION_denyUsbDebugging: { data.enforceInterface(DESCRIPTOR); this.denyUsbDebugging(); reply.writeNoException(); return true; } case TRANSACTION_clearUsbDebuggingKeys: { data.enforceInterface(DESCRIPTOR); this.clearUsbDebuggingKeys(); reply.writeNoException(); return true; } } return super.onTransact(code, data, reply, flags); } private static class Proxy implements android.hardware.usb.IUsbManager { private android.os.IBinder mRemote; Proxy(android.os.IBinder remote) { mRemote = remote; } @Override public android.os.IBinder asBinder() { return mRemote; } public java.lang.String getInterfaceDescriptor() { return DESCRIPTOR; } /* Returns a list of all currently attached USB devices */ @Override public void getDeviceList(android.os.Bundle devices) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); mRemote.transact(Stub.TRANSACTION_getDeviceList, _data, _reply, 0); _reply.readException(); if ((0!=_reply.readInt())) { devices.readFromParcel(_reply); } } finally { _reply.recycle(); _data.recycle(); } } /* Returns a file descriptor for communicating with the USB device. * The native fd can be passed to usb_device_new() in libusbhost. */ @Override public android.os.ParcelFileDescriptor openDevice(java.lang.String deviceName) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); android.os.ParcelFileDescriptor _result; try { _data.writeInterfaceToken(DESCRIPTOR); _data.writeString(deviceName); mRemote.transact(Stub.TRANSACTION_openDevice, _data, _reply, 0); _reply.readException(); if ((0!=_reply.readInt())) { _result = android.os.ParcelFileDescriptor.CREATOR.createFromParcel(_reply); } else { _result = null; } } finally { _reply.recycle(); _data.recycle(); } return _result; } /* Returns the currently attached USB accessory */ @Override public android.hardware.usb.UsbAccessory getCurrentAccessory() throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); android.hardware.usb.UsbAccessory _result; try { _data.writeInterfaceToken(DESCRIPTOR); mRemote.transact(Stub.TRANSACTION_getCurrentAccessory, _data, _reply, 0); _reply.readException(); if ((0!=_reply.readInt())) { _result = android.hardware.usb.UsbAccessory.CREATOR.createFromParcel(_reply); } else { _result = null; } } finally { _reply.recycle(); _data.recycle(); } return _result; } /* Returns a file descriptor for communicating with the USB accessory. * This file descriptor can be used with standard Java file operations. */ @Override public android.os.ParcelFileDescriptor openAccessory(android.hardware.usb.UsbAccessory accessory) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); android.os.ParcelFileDescriptor _result; try { _data.writeInterfaceToken(DESCRIPTOR); if ((accessory!=null)) { _data.writeInt(1); accessory.writeToParcel(_data, 0); } else { _data.writeInt(0); } mRemote.transact(Stub.TRANSACTION_openAccessory, _data, _reply, 0); _reply.readException(); if ((0!=_reply.readInt())) { _result = android.os.ParcelFileDescriptor.CREATOR.createFromParcel(_reply); } else { _result = null; } } finally { _reply.recycle(); _data.recycle(); } return _result; } /* Sets the default package for a USB device * (or clears it if the package name is null) */ @Override public void setDevicePackage(android.hardware.usb.UsbDevice device, java.lang.String packageName, int userId) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); if ((device!=null)) { _data.writeInt(1); device.writeToParcel(_data, 0); } else { _data.writeInt(0); } _data.writeString(packageName); _data.writeInt(userId); mRemote.transact(Stub.TRANSACTION_setDevicePackage, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Sets the default package for a USB accessory * (or clears it if the package name is null) */ @Override public void setAccessoryPackage(android.hardware.usb.UsbAccessory accessory, java.lang.String packageName, int userId) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); if ((accessory!=null)) { _data.writeInt(1); accessory.writeToParcel(_data, 0); } else { _data.writeInt(0); } _data.writeString(packageName); _data.writeInt(userId); mRemote.transact(Stub.TRANSACTION_setAccessoryPackage, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Returns true if the caller has permission to access the device. */ @Override public boolean hasDevicePermission(android.hardware.usb.UsbDevice device) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); boolean _result; try { _data.writeInterfaceToken(DESCRIPTOR); if ((device!=null)) { _data.writeInt(1); device.writeToParcel(_data, 0); } else { _data.writeInt(0); } mRemote.transact(Stub.TRANSACTION_hasDevicePermission, _data, _reply, 0); _reply.readException(); _result = (0!=_reply.readInt()); } finally { _reply.recycle(); _data.recycle(); } return _result; } /* Returns true if the caller has permission to access the accessory. */ @Override public boolean hasAccessoryPermission(android.hardware.usb.UsbAccessory accessory) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); boolean _result; try { _data.writeInterfaceToken(DESCRIPTOR); if ((accessory!=null)) { _data.writeInt(1); accessory.writeToParcel(_data, 0); } else { _data.writeInt(0); } mRemote.transact(Stub.TRANSACTION_hasAccessoryPermission, _data, _reply, 0); _reply.readException(); _result = (0!=_reply.readInt()); } finally { _reply.recycle(); _data.recycle(); } return _result; } /* Requests permission for the given package to access the device. * Will display a system dialog to query the user if permission * had not already been given. */ @Override public void requestDevicePermission(android.hardware.usb.UsbDevice device, java.lang.String packageName, android.app.PendingIntent pi) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); if ((device!=null)) { _data.writeInt(1); device.writeToParcel(_data, 0); } else { _data.writeInt(0); } _data.writeString(packageName); if ((pi!=null)) { _data.writeInt(1); pi.writeToParcel(_data, 0); } else { _data.writeInt(0); } mRemote.transact(Stub.TRANSACTION_requestDevicePermission, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Requests permission for the given package to access the accessory. * Will display a system dialog to query the user if permission * had not already been given. Result is returned via pi. */ @Override public void requestAccessoryPermission(android.hardware.usb.UsbAccessory accessory, java.lang.String packageName, android.app.PendingIntent pi) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); if ((accessory!=null)) { _data.writeInt(1); accessory.writeToParcel(_data, 0); } else { _data.writeInt(0); } _data.writeString(packageName); if ((pi!=null)) { _data.writeInt(1); pi.writeToParcel(_data, 0); } else { _data.writeInt(0); } mRemote.transact(Stub.TRANSACTION_requestAccessoryPermission, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Grants permission for the given UID to access the device */ @Override public void grantDevicePermission(android.hardware.usb.UsbDevice device, int uid) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); if ((device!=null)) { _data.writeInt(1); device.writeToParcel(_data, 0); } else { _data.writeInt(0); } _data.writeInt(uid); mRemote.transact(Stub.TRANSACTION_grantDevicePermission, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Grants permission for the given UID to access the accessory */ @Override public void grantAccessoryPermission(android.hardware.usb.UsbAccessory accessory, int uid) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); if ((accessory!=null)) { _data.writeInt(1); accessory.writeToParcel(_data, 0); } else { _data.writeInt(0); } _data.writeInt(uid); mRemote.transact(Stub.TRANSACTION_grantAccessoryPermission, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Returns true if the USB manager has default preferences or permissions for the package */ @Override public boolean hasDefaults(java.lang.String packageName, int userId) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); boolean _result; try { _data.writeInterfaceToken(DESCRIPTOR); _data.writeString(packageName); _data.writeInt(userId); mRemote.transact(Stub.TRANSACTION_hasDefaults, _data, _reply, 0); _reply.readException(); _result = (0!=_reply.readInt()); } finally { _reply.recycle(); _data.recycle(); } return _result; } /* Clears default preferences and permissions for the package */ @Override public void clearDefaults(java.lang.String packageName, int userId) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); _data.writeString(packageName); _data.writeInt(userId); mRemote.transact(Stub.TRANSACTION_clearDefaults, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Sets the current USB function. */ @Override public void setCurrentFunction(java.lang.String function, boolean makeDefault) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); _data.writeString(function); _data.writeInt(((makeDefault)?(1):(0))); mRemote.transact(Stub.TRANSACTION_setCurrentFunction, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Sets the file path for USB mass storage backing file. */ @Override public void setMassStorageBackingFile(java.lang.String path) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); _data.writeString(path); mRemote.transact(Stub.TRANSACTION_setMassStorageBackingFile, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Allow USB debugging from the attached host. If alwaysAllow is true, add the * the public key to list of host keys that the user has approved. */ @Override public void allowUsbDebugging(boolean alwaysAllow, java.lang.String publicKey) throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); _data.writeInt(((alwaysAllow)?(1):(0))); _data.writeString(publicKey); mRemote.transact(Stub.TRANSACTION_allowUsbDebugging, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Deny USB debugging from the attached host */ @Override public void denyUsbDebugging() throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); mRemote.transact(Stub.TRANSACTION_denyUsbDebugging, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } /* Clear public keys installed for secure USB debugging */ @Override public void clearUsbDebuggingKeys() throws android.os.RemoteException { android.os.Parcel _data = android.os.Parcel.obtain(); android.os.Parcel _reply = android.os.Parcel.obtain(); try { _data.writeInterfaceToken(DESCRIPTOR); mRemote.transact(Stub.TRANSACTION_clearUsbDebuggingKeys, _data, _reply, 0); _reply.readException(); } finally { _reply.recycle(); _data.recycle(); } } } static final int TRANSACTION_getDeviceList = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0); static final int TRANSACTION_openDevice = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1); static final int TRANSACTION_getCurrentAccessory = (android.os.IBinder.FIRST_CALL_TRANSACTION + 2); static final int TRANSACTION_openAccessory = (android.os.IBinder.FIRST_CALL_TRANSACTION + 3); static final int TRANSACTION_setDevicePackage = (android.os.IBinder.FIRST_CALL_TRANSACTION + 4); static final int TRANSACTION_setAccessoryPackage = (android.os.IBinder.FIRST_CALL_TRANSACTION + 5); static final int TRANSACTION_hasDevicePermission = (android.os.IBinder.FIRST_CALL_TRANSACTION + 6); static final int TRANSACTION_hasAccessoryPermission = (android.os.IBinder.FIRST_CALL_TRANSACTION + 7); static final int TRANSACTION_requestDevicePermission = (android.os.IBinder.FIRST_CALL_TRANSACTION + 8); static final int TRANSACTION_requestAccessoryPermission = (android.os.IBinder.FIRST_CALL_TRANSACTION + 9); static final int TRANSACTION_grantDevicePermission = (android.os.IBinder.FIRST_CALL_TRANSACTION + 10); static final int TRANSACTION_grantAccessoryPermission = (android.os.IBinder.FIRST_CALL_TRANSACTION + 11); static final int TRANSACTION_hasDefaults = (android.os.IBinder.FIRST_CALL_TRANSACTION + 12); static final int TRANSACTION_clearDefaults = (android.os.IBinder.FIRST_CALL_TRANSACTION + 13); static final int TRANSACTION_setCurrentFunction = (android.os.IBinder.FIRST_CALL_TRANSACTION + 14); static final int TRANSACTION_setMassStorageBackingFile = (android.os.IBinder.FIRST_CALL_TRANSACTION + 15); static final int TRANSACTION_allowUsbDebugging = (android.os.IBinder.FIRST_CALL_TRANSACTION + 16); static final int TRANSACTION_denyUsbDebugging = (android.os.IBinder.FIRST_CALL_TRANSACTION + 17); static final int TRANSACTION_clearUsbDebuggingKeys = (android.os.IBinder.FIRST_CALL_TRANSACTION + 18); } /* Returns a list of all currently attached USB devices */ public void getDeviceList(android.os.Bundle devices) throws android.os.RemoteException; /* Returns a file descriptor for communicating with the USB device. * The native fd can be passed to usb_device_new() in libusbhost. */ public android.os.ParcelFileDescriptor openDevice(java.lang.String deviceName) throws android.os.RemoteException; /* Returns the currently attached USB accessory */ public android.hardware.usb.UsbAccessory getCurrentAccessory() throws android.os.RemoteException; /* Returns a file descriptor for communicating with the USB accessory. * This file descriptor can be used with standard Java file operations. */ public android.os.ParcelFileDescriptor openAccessory(android.hardware.usb.UsbAccessory accessory) throws android.os.RemoteException; /* Sets the default package for a USB device * (or clears it if the package name is null) */ public void setDevicePackage(android.hardware.usb.UsbDevice device, java.lang.String packageName, int userId) throws android.os.RemoteException; /* Sets the default package for a USB accessory * (or clears it if the package name is null) */ public void setAccessoryPackage(android.hardware.usb.UsbAccessory accessory, java.lang.String packageName, int userId) throws android.os.RemoteException; /* Returns true if the caller has permission to access the device. */ public boolean hasDevicePermission(android.hardware.usb.UsbDevice device) throws android.os.RemoteException; /* Returns true if the caller has permission to access the accessory. */ public boolean hasAccessoryPermission(android.hardware.usb.UsbAccessory accessory) throws android.os.RemoteException; /* Requests permission for the given package to access the device. * Will display a system dialog to query the user if permission * had not already been given. */ public void requestDevicePermission(android.hardware.usb.UsbDevice device, java.lang.String packageName, android.app.PendingIntent pi) throws android.os.RemoteException; /* Requests permission for the given package to access the accessory. * Will display a system dialog to query the user if permission * had not already been given. Result is returned via pi. */ public void requestAccessoryPermission(android.hardware.usb.UsbAccessory accessory, java.lang.String packageName, android.app.PendingIntent pi) throws android.os.RemoteException; /* Grants permission for the given UID to access the device */ public void grantDevicePermission(android.hardware.usb.UsbDevice device, int uid) throws android.os.RemoteException; /* Grants permission for the given UID to access the accessory */ public void grantAccessoryPermission(android.hardware.usb.UsbAccessory accessory, int uid) throws android.os.RemoteException; /* Returns true if the USB manager has default preferences or permissions for the package */ public boolean hasDefaults(java.lang.String packageName, int userId) throws android.os.RemoteException; /* Clears default preferences and permissions for the package */ public void clearDefaults(java.lang.String packageName, int userId) throws android.os.RemoteException; /* Sets the current USB function. */ public void setCurrentFunction(java.lang.String function, boolean makeDefault) throws android.os.RemoteException; /* Sets the file path for USB mass storage backing file. */ public void setMassStorageBackingFile(java.lang.String path) throws android.os.RemoteException; /* Allow USB debugging from the attached host. If alwaysAllow is true, add the * the public key to list of host keys that the user has approved. */ public void allowUsbDebugging(boolean alwaysAllow, java.lang.String publicKey) throws android.os.RemoteException; /* Deny USB debugging from the attached host */ public void denyUsbDebugging() throws android.os.RemoteException; /* Clear public keys installed for secure USB debugging */ public void clearUsbDebuggingKeys() throws android.os.RemoteException; }
3)建立android.os.ServiceManager.java替換原始碼
package android.os; import java.util.Map; public final class ServiceManager { public static IBinder getService( String name ) { throw new RuntimeException( "Stub!" ); } /** * Place a new @a service called @a name into the service * manager. * * @param name the name of the new service * @param service the service object */ public static void addService( String name, IBinder service ) { throw new RuntimeException( "Stub!" ); } /** * Retrieve an existing service called @a name from the * service manager. Non-blocking. */ public static IBinder checkService( String name ) { throw new RuntimeException( "Stub!" ); } public static String[] listServices() throws RemoteException { throw new RuntimeException( "Stub!" ); } /** * This is only intended to be called when the process is first being brought * up and bound by the activity manager. There is only one thread in the process * at that time, so no locking is done. * * @param cache the cache of service references * @hide */ public static void initServiceCache( Map<String, IBinder> cache ) { throw new RuntimeException( "Stub!" ); } }
4)建立響應app申請的BroadcastReceiver類,com.harvbot.usbpermissionissuer.LaunchReceiver.java
package com.harvbot.usbpermissionissuer;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.content.res.XmlResourceParser;
import android.hardware.usb.IUsbManager;
import android.hardware.usb.UsbDevice;
import android.hardware.usb.UsbManager;
import android.os.IBinder;
import android.os.ServiceManager;
import android.text.TextUtils;
import android.util.Log;
import org.json.JSONObject;
import org.xmlpull.v1.XmlPullParserException;
import com.harvbot.usbpermissionissuer.logging.LogManager;
public class LaunchReceiver extends BroadcastReceiver
{
private final String TAG = "com.harvbot.usb";
private final String ACTION_USB_PERMISSION_ISSUER = "ACTION_USB_PERMISSION_ISSUER";
private LogManager logger;
public void onReceive( Context context, Intent intent )
{
logger = new LogManager(context);
String action = intent.getAction();
if( action != null && action.equals( ACTION_USB_PERMISSION_ISSUER ) )
{
logger.log("Receiver event processing.");
try {
UsbDeviceDescriptor deviceFilter = new UsbDeviceDescriptor();
deviceFilter.packageName = intent.getStringExtra("packageName");
deviceFilter.vendorId = intent.getIntExtra("vendorId", -1);
deviceFilter.productId = intent.getIntExtra("productId", -1);
deviceFilter.deviceClass = intent.getIntExtra("deviceClass", -1);
deviceFilter.deviceSubclass = intent.getIntExtra("deviceSubclass", -1);
if(TextUtils.isEmpty(deviceFilter.packageName)){
logger.log("PackageName is Null");
return;
}
PackageManager pm = context.getPackageManager();
ApplicationInfo ai = pm.getApplicationInfo(deviceFilter.packageName, 0);
UsbManager manager = (UsbManager) context.getSystemService(Context.USB_SERVICE);
IBinder b = ServiceManager.getService(Context.USB_SERVICE);
IUsbManager service = IUsbManager.Stub.asInterface(b);
HashMap<String, UsbDevice> deviceList = manager.getDeviceList();
logger.log("List of usb devices was loaded. Number of attached devices: " +
new Integer(deviceList.size()).toString());
if((deviceFilter.deviceClass != -1 && deviceFilter.deviceSubclass != -1) ||
(deviceFilter.productId != -1 && deviceFilter.vendorId != -1)) {
logger.log("The usb permission will be granted for application " + ai.packageName);
Iterator<UsbDevice> deviceIterator = deviceList.values().iterator();
while (deviceIterator.hasNext()) {
UsbDevice device = deviceIterator.next();
if ((device.getDeviceClass() == deviceFilter.deviceClass &&
device.getDeviceSubclass() == deviceFilter.deviceSubclass) ||
(device.getVendorId() == deviceFilter.vendorId &&
device.getProductId() == deviceFilter.productId)) {
try {
service.grantDevicePermission(device, ai.uid);
service.setDevicePackage(device, ai.packageName, ai.uid);
logger.log("Usb permission is granted for application " +
ai.packageName + " for device " + device.toString() +
" and user " + ai.uid + " is granted");
} catch (SecurityException se) {
logger.log(se.toString());
Log.e(TAG, se.toString());
}
}
}
} else {
logger.log("Device params is Error "+deviceFilter.toString());
}
}
catch(Exception e)
{
logger.log(e.toString());
Log.e(TAG, e.toString());
}
}
}
}
額外需要的UsbDeviceDescriptor.java和StubService.java一併給出,日誌實現自行替換
package com.harvbot.usbpermissionissuer;
//二選一,確定USB裝置
public class UsbDeviceDescriptor {
public String packageName;
//第一組,確定USB裝置
public int deviceClass;
public int deviceSubclass;
//第二級,確定USB裝置
public int vendorId;
public int productId;
@Override
public String toString() {
return "packageName:" + packageName
+ ", deviceClass:" + deviceClass
+ ", deviceSubclass: " + deviceSubclass
+ ", vendorId: " + vendorId
+ ", productId: " + productId;
}
}
package com.harvbot.usbpermissionissuer;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.util.Log;
import android.widget.Toast;
public class StubService extends Service
{
private static final String TAG = "StubService";
@Override
public IBinder onBind(Intent intent) {
return null;
}
public void onDestroy() {
Log.d(TAG, "onDestroy");
}
@Override
public void onStart(Intent intent, int startid)
{
Log.d(TAG, "onStart");
}
}
5)以上關鍵檔案工程化後生成usb-permission-issuer.apk,需要將其拷貝到android系統下的/system/prv-app目錄下(該目錄是系統預裝apk路徑),並修改其檔案許可權,例如777
備註:我的做法是直接下載了“kingroot”和“RE檔案管理”,安裝後,使用“kingroot”給“RE檔案管理”開啟root許可權,然後通過“RE檔案管理”將“usb-permission-issuer.apk”拷貝到/system/prv-app目錄下,並修改其檔案許可權(選擇檔案長按,RE檔案管理會出現更多功能選單,裡面有許可權修改)
6)在需要的app進行動態廣播申請,例項如下:
private void UsbInit()
{
UsbInitPermission(1);
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
UsbInitPermission(2);
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
UsbInitPermission(3);
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
UsbInitPermission(4);
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
static final String YOUR_APP_PACKAGE_NAMESPACE = "你的app-package";
static final String ACTION_USB_PERMISSION_APP = "ACTION_USB_PERMISSION_ISSUER";
private void UsbInitPermission(int id)
{
Intent intent_usb = new Intent();
intent_usb.setAction(ACTION_USB_PERMISSION_APP);
intent_usb.putExtra("packageName", YOUR_APP_PACKAGE_NAMESPACE);
//device_filter.xml裡的欄位
intent_usb.putExtra("vendorId", 1659);
switch(id)
{
case 1:
intent_usb.putExtra("productId", 8963);
break;
case 2:
intent_usb.putExtra("productId", 9553);
break;
case 3:
intent_usb.putExtra("productId", 8964);
break;
case 4:
intent_usb.putExtra("productId", 9475);
break;
}
//傳送廣播
sendBroadcast(intent_usb);
}
然後再合適的地方呼叫UsbInit(),等待一段時間後會獲得授權,再使用你的USB,這樣就不會再有USB授權對話方塊彈出
7)如果懶得編譯授權的apk,可以直接採用https://download.csdn.net/download/py8105/10313534,再完成5、6步驟。