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