Android
Karta inteligentna
Szukaj…
Wysyłanie i odbieranie kart inteligentnych
W przypadku połączenia znajduje się fragment, który pomoże Ci zrozumieć:
//Allows you to enumerate and communicate with connected USB devices.
UsbManager mUsbManager = (UsbManager) getSystemService(Context.USB_SERVICE);
//Explicitly asking for permission
final String ACTION_USB_PERMISSION = "com.android.example.USB_PERMISSION";
PendingIntent mPermissionIntent = PendingIntent.getBroadcast(this, 0, new Intent(ACTION_USB_PERMISSION), 0);
HashMap<String, UsbDevice> deviceList = mUsbManager.getDeviceList();
UsbDevice device = deviceList.get("//the device you want to work with");
if (device != null) {
mUsbManager.requestPermission(device, mPermissionIntent);
}
Teraz musisz zrozumieć, że w java komunikacja odbywa się za pomocą pakietu javax.smarcard, który nie jest dostępny dla Androida, więc zajrzyj tutaj, aby dowiedzieć się, jak możesz komunikować się lub wysyłać / odbierać APDU (polecenie smartcard).
Teraz, jak powiedziano w wyżej wspomnianej odpowiedzi
Nie można po prostu wysłać APDU (polecenie karty inteligentnej) przez punkt końcowy luzowania i oczekiwać, że otrzyma odpowiedź APDU przez punkt końcowy luzowania. Aby uzyskać punkty końcowe, zobacz poniższy fragment kodu:
UsbEndpoint epOut = null, epIn = null;
UsbInterface usbInterface;
UsbDeviceConnection connection = mUsbManager.openDevice(device);
for (int i = 0; i < device.getInterfaceCount(); i++) {
usbInterface = device.getInterface(i);
connection.claimInterface(usbInterface, true);
for (int j = 0; j < usbInterface.getEndpointCount(); j++) {
UsbEndpoint ep = usbInterface.getEndpoint(j);
if (ep.getType() == UsbConstants.USB_ENDPOINT_XFER_BULK) {
if (ep.getDirection() == UsbConstants.USB_DIR_OUT) {
// from host to device
epOut = ep;
} else if (ep.getDirection() == UsbConstants.USB_DIR_IN) {
// from device to host
epIn = ep;
}
}
}
}
Teraz masz masowe punkty końcowe i masowe do wysyłania i odbierania poleceń APDU i bloków odpowiedzi APDU:
Aby wysyłać polecenia, zobacz fragment kodu poniżej:
public void write(UsbDeviceConnection connection, UsbEndpoint epOut, byte[] command) {
result = new StringBuilder();
connection.bulkTransfer(epOut, command, command.length, TIMEOUT);
//For Printing logs you can use result variable
for (byte bb : command) {
result.append(String.format(" %02X ", bb));
}
}
Aby otrzymać / odczytać odpowiedź, przeczytaj poniższy fragment kodu:
public int read(UsbDeviceConnection connection, UsbEndpoint epIn) {
result = new StringBuilder();
final byte[] buffer = new byte[epIn.getMaxPacketSize()];
int byteCount = 0;
byteCount = connection.bulkTransfer(epIn, buffer, buffer.length, TIMEOUT);
//For Printing logs you can use result variable
if (byteCount >= 0) {
for (byte bb : buffer) {
result.append(String.format(" %02X ", bb));
}
//Buffer received was : result.toString()
} else {
//Something went wrong as count was : " + byteCount
}
return byteCount;
}
Teraz, jeśli zobaczysz tę odpowiedź tutaj, pierwszym poleceniem do wysłania jest:
PC_to_RDR_IccPowerOn, aby aktywować kartę.
który możesz utworzyć, czytając sekcję 6.1.1 dokumentu specyfikacji urządzeń USB tutaj.
Teraz weźmy przykład tego polecenia, takiego jak tutaj: 62000000000000000000
Jak możesz to wysłać:
write(connection, epOut, "62000000000000000000");
Teraz po pomyślnym wysłaniu polecenia APDU możesz odczytać odpowiedź, używając:
read(connection, epIn);
I otrzymaj coś takiego
80 18000000 00 00 00 00 00 3BBF11008131FE45455041000000000000000000000000F1
Teraz odpowiedź otrzymana w kodzie tutaj będzie w result
zmiennej read()
metody z kodu