1*90c8c64dSAndroid Build Coastguard Worker /* 2*90c8c64dSAndroid Build Coastguard Worker * Copyright (C) 2013 The Android Open Source Project 3*90c8c64dSAndroid Build Coastguard Worker * 4*90c8c64dSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*90c8c64dSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*90c8c64dSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*90c8c64dSAndroid Build Coastguard Worker * 8*90c8c64dSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*90c8c64dSAndroid Build Coastguard Worker * 10*90c8c64dSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*90c8c64dSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*90c8c64dSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*90c8c64dSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*90c8c64dSAndroid Build Coastguard Worker * limitations under the License. 15*90c8c64dSAndroid Build Coastguard Worker */ 16*90c8c64dSAndroid Build Coastguard Worker package com.example.android.cardreader; 17*90c8c64dSAndroid Build Coastguard Worker 18*90c8c64dSAndroid Build Coastguard Worker import android.nfc.NfcAdapter; 19*90c8c64dSAndroid Build Coastguard Worker import android.nfc.Tag; 20*90c8c64dSAndroid Build Coastguard Worker import android.nfc.tech.IsoDep; 21*90c8c64dSAndroid Build Coastguard Worker 22*90c8c64dSAndroid Build Coastguard Worker import com.example.android.common.logger.Log; 23*90c8c64dSAndroid Build Coastguard Worker 24*90c8c64dSAndroid Build Coastguard Worker import java.io.IOException; 25*90c8c64dSAndroid Build Coastguard Worker import java.lang.ref.WeakReference; 26*90c8c64dSAndroid Build Coastguard Worker import java.util.Arrays; 27*90c8c64dSAndroid Build Coastguard Worker 28*90c8c64dSAndroid Build Coastguard Worker /** 29*90c8c64dSAndroid Build Coastguard Worker * Callback class, invoked when an NFC card is scanned while the device is running in reader mode. 30*90c8c64dSAndroid Build Coastguard Worker * 31*90c8c64dSAndroid Build Coastguard Worker * Reader mode can be invoked by calling NfcAdapter 32*90c8c64dSAndroid Build Coastguard Worker */ 33*90c8c64dSAndroid Build Coastguard Worker public class LoyaltyCardReader implements NfcAdapter.ReaderCallback { 34*90c8c64dSAndroid Build Coastguard Worker private static final String TAG = "LoyaltyCardReader"; 35*90c8c64dSAndroid Build Coastguard Worker // AID for our loyalty card service. 36*90c8c64dSAndroid Build Coastguard Worker private static final String SAMPLE_LOYALTY_CARD_AID = "F222222222"; 37*90c8c64dSAndroid Build Coastguard Worker // ISO-DEP command HEADER for selecting an AID. 38*90c8c64dSAndroid Build Coastguard Worker // Format: [Class | Instruction | Parameter 1 | Parameter 2] 39*90c8c64dSAndroid Build Coastguard Worker private static final String SELECT_APDU_HEADER = "00A40400"; 40*90c8c64dSAndroid Build Coastguard Worker // "OK" status word sent in response to SELECT AID command (0x9000) 41*90c8c64dSAndroid Build Coastguard Worker private static final byte[] SELECT_OK_SW = {(byte) 0x90, (byte) 0x00}; 42*90c8c64dSAndroid Build Coastguard Worker 43*90c8c64dSAndroid Build Coastguard Worker // Weak reference to prevent retain loop. mAccountCallback is responsible for exiting 44*90c8c64dSAndroid Build Coastguard Worker // foreground mode before it becomes invalid (e.g. during onPause() or onStop()). 45*90c8c64dSAndroid Build Coastguard Worker private WeakReference<AccountCallback> mAccountCallback; 46*90c8c64dSAndroid Build Coastguard Worker 47*90c8c64dSAndroid Build Coastguard Worker public interface AccountCallback { onAccountReceived(String account)48*90c8c64dSAndroid Build Coastguard Worker public void onAccountReceived(String account); 49*90c8c64dSAndroid Build Coastguard Worker } 50*90c8c64dSAndroid Build Coastguard Worker LoyaltyCardReader(AccountCallback accountCallback)51*90c8c64dSAndroid Build Coastguard Worker public LoyaltyCardReader(AccountCallback accountCallback) { 52*90c8c64dSAndroid Build Coastguard Worker mAccountCallback = new WeakReference<AccountCallback>(accountCallback); 53*90c8c64dSAndroid Build Coastguard Worker } 54*90c8c64dSAndroid Build Coastguard Worker 55*90c8c64dSAndroid Build Coastguard Worker /** 56*90c8c64dSAndroid Build Coastguard Worker * Callback when a new tag is discovered by the system. 57*90c8c64dSAndroid Build Coastguard Worker * 58*90c8c64dSAndroid Build Coastguard Worker * <p>Communication with the card should take place here. 59*90c8c64dSAndroid Build Coastguard Worker * 60*90c8c64dSAndroid Build Coastguard Worker * @param tag Discovered tag 61*90c8c64dSAndroid Build Coastguard Worker */ 62*90c8c64dSAndroid Build Coastguard Worker @Override onTagDiscovered(Tag tag)63*90c8c64dSAndroid Build Coastguard Worker public void onTagDiscovered(Tag tag) { 64*90c8c64dSAndroid Build Coastguard Worker Log.i(TAG, "New tag discovered"); 65*90c8c64dSAndroid Build Coastguard Worker // Android's Host-based Card Emulation (HCE) feature implements the ISO-DEP (ISO 14443-4) 66*90c8c64dSAndroid Build Coastguard Worker // protocol. 67*90c8c64dSAndroid Build Coastguard Worker // 68*90c8c64dSAndroid Build Coastguard Worker // In order to communicate with a device using HCE, the discovered tag should be processed 69*90c8c64dSAndroid Build Coastguard Worker // using the IsoDep class. 70*90c8c64dSAndroid Build Coastguard Worker IsoDep isoDep = IsoDep.get(tag); 71*90c8c64dSAndroid Build Coastguard Worker if (isoDep != null) { 72*90c8c64dSAndroid Build Coastguard Worker try { 73*90c8c64dSAndroid Build Coastguard Worker // Connect to the remote NFC device 74*90c8c64dSAndroid Build Coastguard Worker isoDep.connect(); 75*90c8c64dSAndroid Build Coastguard Worker // Build SELECT AID command for our loyalty card service. 76*90c8c64dSAndroid Build Coastguard Worker // This command tells the remote device which service we wish to communicate with. 77*90c8c64dSAndroid Build Coastguard Worker Log.i(TAG, "Requesting remote AID: " + SAMPLE_LOYALTY_CARD_AID); 78*90c8c64dSAndroid Build Coastguard Worker byte[] command = BuildSelectApdu(SAMPLE_LOYALTY_CARD_AID); 79*90c8c64dSAndroid Build Coastguard Worker // Send command to remote device 80*90c8c64dSAndroid Build Coastguard Worker Log.i(TAG, "Sending: " + ByteArrayToHexString(command)); 81*90c8c64dSAndroid Build Coastguard Worker byte[] result = isoDep.transceive(command); 82*90c8c64dSAndroid Build Coastguard Worker // If AID is successfully selected, 0x9000 is returned as the status word (last 2 83*90c8c64dSAndroid Build Coastguard Worker // bytes of the result) by convention. Everything before the status word is 84*90c8c64dSAndroid Build Coastguard Worker // optional payload, which is used here to hold the account number. 85*90c8c64dSAndroid Build Coastguard Worker int resultLength = result.length; 86*90c8c64dSAndroid Build Coastguard Worker byte[] statusWord = {result[resultLength-2], result[resultLength-1]}; 87*90c8c64dSAndroid Build Coastguard Worker byte[] payload = Arrays.copyOf(result, resultLength-2); 88*90c8c64dSAndroid Build Coastguard Worker if (Arrays.equals(SELECT_OK_SW, statusWord)) { 89*90c8c64dSAndroid Build Coastguard Worker // The remote NFC device will immediately respond with its stored account number 90*90c8c64dSAndroid Build Coastguard Worker String accountNumber = new String(payload, "UTF-8"); 91*90c8c64dSAndroid Build Coastguard Worker Log.i(TAG, "Received: " + accountNumber); 92*90c8c64dSAndroid Build Coastguard Worker // Inform CardReaderFragment of received account number 93*90c8c64dSAndroid Build Coastguard Worker mAccountCallback.get().onAccountReceived(accountNumber); 94*90c8c64dSAndroid Build Coastguard Worker } 95*90c8c64dSAndroid Build Coastguard Worker } catch (IOException e) { 96*90c8c64dSAndroid Build Coastguard Worker Log.e(TAG, "Error communicating with card: " + e.toString()); 97*90c8c64dSAndroid Build Coastguard Worker } 98*90c8c64dSAndroid Build Coastguard Worker } 99*90c8c64dSAndroid Build Coastguard Worker } 100*90c8c64dSAndroid Build Coastguard Worker 101*90c8c64dSAndroid Build Coastguard Worker /** 102*90c8c64dSAndroid Build Coastguard Worker * Build APDU for SELECT AID command. This command indicates which service a reader is 103*90c8c64dSAndroid Build Coastguard Worker * interested in communicating with. See ISO 7816-4. 104*90c8c64dSAndroid Build Coastguard Worker * 105*90c8c64dSAndroid Build Coastguard Worker * @param aid Application ID (AID) to select 106*90c8c64dSAndroid Build Coastguard Worker * @return APDU for SELECT AID command 107*90c8c64dSAndroid Build Coastguard Worker */ BuildSelectApdu(String aid)108*90c8c64dSAndroid Build Coastguard Worker public static byte[] BuildSelectApdu(String aid) { 109*90c8c64dSAndroid Build Coastguard Worker // Format: [CLASS | INSTRUCTION | PARAMETER 1 | PARAMETER 2 | LENGTH | DATA] 110*90c8c64dSAndroid Build Coastguard Worker return HexStringToByteArray(SELECT_APDU_HEADER + String.format("%02X", aid.length() / 2) + aid); 111*90c8c64dSAndroid Build Coastguard Worker } 112*90c8c64dSAndroid Build Coastguard Worker 113*90c8c64dSAndroid Build Coastguard Worker /** 114*90c8c64dSAndroid Build Coastguard Worker * Utility class to convert a byte array to a hexadecimal string. 115*90c8c64dSAndroid Build Coastguard Worker * 116*90c8c64dSAndroid Build Coastguard Worker * @param bytes Bytes to convert 117*90c8c64dSAndroid Build Coastguard Worker * @return String, containing hexadecimal representation. 118*90c8c64dSAndroid Build Coastguard Worker */ ByteArrayToHexString(byte[] bytes)119*90c8c64dSAndroid Build Coastguard Worker public static String ByteArrayToHexString(byte[] bytes) { 120*90c8c64dSAndroid Build Coastguard Worker final char[] hexArray = {'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'}; 121*90c8c64dSAndroid Build Coastguard Worker char[] hexChars = new char[bytes.length * 2]; 122*90c8c64dSAndroid Build Coastguard Worker int v; 123*90c8c64dSAndroid Build Coastguard Worker for ( int j = 0; j < bytes.length; j++ ) { 124*90c8c64dSAndroid Build Coastguard Worker v = bytes[j] & 0xFF; 125*90c8c64dSAndroid Build Coastguard Worker hexChars[j * 2] = hexArray[v >>> 4]; 126*90c8c64dSAndroid Build Coastguard Worker hexChars[j * 2 + 1] = hexArray[v & 0x0F]; 127*90c8c64dSAndroid Build Coastguard Worker } 128*90c8c64dSAndroid Build Coastguard Worker return new String(hexChars); 129*90c8c64dSAndroid Build Coastguard Worker } 130*90c8c64dSAndroid Build Coastguard Worker 131*90c8c64dSAndroid Build Coastguard Worker /** 132*90c8c64dSAndroid Build Coastguard Worker * Utility class to convert a hexadecimal string to a byte string. 133*90c8c64dSAndroid Build Coastguard Worker * 134*90c8c64dSAndroid Build Coastguard Worker * <p>Behavior with input strings containing non-hexadecimal characters is undefined. 135*90c8c64dSAndroid Build Coastguard Worker * 136*90c8c64dSAndroid Build Coastguard Worker * @param s String containing hexadecimal characters to convert 137*90c8c64dSAndroid Build Coastguard Worker * @return Byte array generated from input 138*90c8c64dSAndroid Build Coastguard Worker */ HexStringToByteArray(String s)139*90c8c64dSAndroid Build Coastguard Worker public static byte[] HexStringToByteArray(String s) { 140*90c8c64dSAndroid Build Coastguard Worker int len = s.length(); 141*90c8c64dSAndroid Build Coastguard Worker byte[] data = new byte[len / 2]; 142*90c8c64dSAndroid Build Coastguard Worker for (int i = 0; i < len; i += 2) { 143*90c8c64dSAndroid Build Coastguard Worker data[i / 2] = (byte) ((Character.digit(s.charAt(i), 16) << 4) 144*90c8c64dSAndroid Build Coastguard Worker + Character.digit(s.charAt(i+1), 16)); 145*90c8c64dSAndroid Build Coastguard Worker } 146*90c8c64dSAndroid Build Coastguard Worker return data; 147*90c8c64dSAndroid Build Coastguard Worker } 148*90c8c64dSAndroid Build Coastguard Worker 149*90c8c64dSAndroid Build Coastguard Worker } 150