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