IOException с эмуляцией карты на основе хоста

Я использую HCE и столкнулся с IOException на

isoDep.connect();

на определенном устройстве чтения Android cr100 simcent.

HCE отлично работает, когда я включаю режим чтения в NFC с указанными ниже флагами.

public static int READER_FLAGS =
        NfcAdapter.FLAG_READER_NFC_A | NfcAdapter.FLAG_READER_SKIP_NDEF_CHECK;

Но при этом я не могу читать теги NDEF. Хотя тот же код отлично работает на планшете Nexus 7 (2012).

Полный код прилагается

Фрагмент CardReader

public class CardReaderFragment extends Fragment implements LoyaltyCardReader.AccountCallback {

    public static final String TAG = "CardReaderFragment";
    // Recommend NfcAdapter flags for reading from other Android devices. Indicates that this
    // activity is interested in NFC-A devices (including other Android devices), and that the
    // system should not check for the presence of NDEF-formatted data (e.g. Android Beam).
    public static int READER_FLAGS =
            NfcAdapter.FLAG_READER_NFC_A | NfcAdapter.FLAG_READER_SKIP_NDEF_CHECK;
    public LoyaltyCardReader mLoyaltyCardReader;
    private TextView mAccountField;

    /** Called when sample is created. Displays generic UI with welcome text. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // Inflate the layout for this fragment
        View v = inflater.inflate(R.layout.main_fragment, container, false);
        if (v != null) {
            mAccountField = (TextView) v.findViewById(R.id.card_account_field);
            mAccountField.setText("Waiting...");

            mLoyaltyCardReader = new LoyaltyCardReader(this);

            // Disable Android Beam and register our card reader callback
            enableReaderMode();
        }

        return v;
    }

    @Override
    public void onPause() {
        super.onPause();
        disableReaderMode();
    }

    @Override
    public void onResume() {
        super.onResume();
        enableReaderMode();
    }

    private void enableReaderMode() {
        Log.i(TAG, "Enabling reader mode");
        Activity activity = getActivity();
        NfcAdapter nfc = NfcAdapter.getDefaultAdapter(activity);
        if (nfc != null) {
            nfc.enableReaderMode(activity, mLoyaltyCardReader, READER_FLAGS, null);
        }
    }

    private void disableReaderMode() {
        Log.i(TAG, "Disabling reader mode");
        Activity activity = getActivity();
        NfcAdapter nfc = NfcAdapter.getDefaultAdapter(activity);
        if (nfc != null) {
            nfc.disableReaderMode(activity);
        }
    }

    @Override
    public void onAccountReceived(final String account) {
        // This callback is run on a background thread, but updates to UI elements must be performed
        // on the UI thread.
        getActivity().runOnUiThread(new Runnable() {
            @Override
            public void run() {
                mAccountField.setText(account);
            }
        });
    }
}

Считыватель карт лояльности

public class LoyaltyCardReader implements NfcAdapter.ReaderCallback {
    private static final String TAG = "LoyaltyCardReader";
    // AID for our loyalty card service.
    private static final String SAMPLE_LOYALTY_CARD_AID = "F222222222";
    // ISO-DEP command HEADER for selecting an AID.
    // Format: [Class | Instruction | Parameter 1 | Parameter 2]
    private static final String SELECT_APDU_HEADER = "00A40400";
    // "OK" status word sent in response to SELECT AID command (0x9000)
    private static final byte[] SELECT_OK_SW = {(byte) 0x90, (byte) 0x00};

    // Weak reference to prevent retain loop. mAccountCallback is responsible for exiting
    // foreground mode before it becomes invalid (e.g. during onPause() or onStop()).
    private WeakReference<AccountCallback> mAccountCallback;

    public interface AccountCallback {
        public void onAccountReceived(String account);
    }

    public LoyaltyCardReader(AccountCallback accountCallback) {
        mAccountCallback = new WeakReference<AccountCallback>(accountCallback);
    }

    /**
     * Callback when a new tag is discovered by the system.
     * <p>
     * <p>Communication with the card should take place here.
     *
     * @param tag Discovered tag
     */
    @Override
    public void onTagDiscovered(Tag tag) {
        Log.i(TAG, "New tag discovered");
        // Android's Host-based Card Emulation (HCE) feature implements the ISO-DEP (ISO 14443-4)
        // protocol.
        //
        // In order to communicate with a device using HCE, the discovered tag should be processed
        // using the IsoDep class.
        IsoDep isoDep = IsoDep.get(tag);
        if (isoDep != null) {
            try {
                // Connect to the remote NFC device
                isoDep.connect();
                // Build SELECT AID command for our loyalty card service.
                // This command tells the remote device which service we wish to communicate with.
                Log.i(TAG, "Requesting remote AID: " + SAMPLE_LOYALTY_CARD_AID);
                byte[] command = BuildSelectApdu(SAMPLE_LOYALTY_CARD_AID);
                // Send command to remote device
                Log.i(TAG, "Sending: " + ByteArrayToHexString(command));
                byte[] result = isoDep.transceive(command);
                // If AID is successfully selected, 0x9000 is returned as the status word (last 2
                // bytes of the result) by convention. Everything before the status word is
                // optional payload, which is used here to hold the account number.
                int resultLength = result.length;
                byte[] statusWord = {result[resultLength - 2], result[resultLength - 1]};
                byte[] payload = Arrays.copyOf(result, resultLength - 2);
                if (Arrays.equals(SELECT_OK_SW, statusWord)) {
                    // The remote NFC device will immediately respond with its stored account number
                    String accountNumber = new String(payload, "UTF-8");
                    Log.i(TAG, "Received: " + accountNumber);
                    // Inform CardReaderFragment of received account number
                    mAccountCallback.get().onAccountReceived(accountNumber);
                }
            } catch (IOException e) {
                Log.e(TAG, "Error communicating with card: " + e.toString());
            }
        } else {
            Ndef ndef = Ndef.get(tag);
            if (ndef == null) {
                // NDEF is not supported by this Tag.
                Log.d("NFCCardTagNDEF", "even this is null");
//                return;
            }

            NdefMessage ndefMessage = ndef.getCachedNdefMessage();

            if (ndefMessage == null) {
                Log.d("NFCCardTagNDEF", "ndef message is null");
//                return;
            }

            NdefRecord[] records = ndefMessage.getRecords();
            String text = ndefRecordToString(records[0]);
            Log.d("NFCCardTagNFC", "old" + text);
            mAccountCallback.get().onAccountReceived(text);
        }
    }


    public String ndefRecordToString(NdefRecord record) {
        byte[] payload = record.getPayload();
        return new String(payload);
    }

    /**
     * Build APDU for SELECT AID command. This command indicates which service a reader is
     * interested in communicating with. See ISO 7816-4.
     *
     * @param aid Application ID (AID) to select
     * @return APDU for SELECT AID command
     */
    public static byte[] BuildSelectApdu(String aid) {
        // Format: [CLASS | INSTRUCTION | PARAMETER 1 | PARAMETER 2 | LENGTH | DATA]
        return HexStringToByteArray(SELECT_APDU_HEADER + String.format("%02X", aid.length() / 2) + aid);
    }

    /**
     * Utility class to convert a byte array to a hexadecimal string.
     *
     * @param bytes Bytes to convert
     * @return String, containing hexadecimal representation.
     */
    public static String ByteArrayToHexString(byte[] bytes) {
        final char[] hexArray = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F'};
        char[] hexChars = new char[bytes.length * 2];
        int v;
        for (int j = 0; j < bytes.length; j++) {
            v = bytes[j] & 0xFF;
            hexChars[j * 2] = hexArray[v >>> 4];
            hexChars[j * 2 + 1] = hexArray[v & 0x0F];
        }
        return new String(hexChars);
    }

    /**
     * Utility class to convert a hexadecimal string to a byte string.
     * <p>
     * <p>Behavior with input strings containing non-hexadecimal characters is undefined.
     *
     * @param s String containing hexadecimal characters to convert
     * @return Byte array generated from input
     */
    public static byte[] HexStringToByteArray(String s) {
        int len = s.length();
        byte[] data = new byte[len / 2];
        for (int i = 0; i < len; i += 2) {
            data[i / 2] = (byte) ((Character.digit(s.charAt(i), 16) << 4)
                    + Character.digit(s.charAt(i + 1), 16));
        }
        return data;
    }

}

Любая помощь будет оценена


person chossen-addict    schedule 07.07.2017    source источник
comment
Вы используете cr100 как считыватель или как устройство HCE?   -  person Michael Roland    schedule 16.07.2017
comment
Я использую cr100 в качестве ридера   -  person chossen-addict    schedule 16.07.2017
comment
добавьте свой логкэт   -  person Oussema Aroua    schedule 20.07.2017


Ответы (1)


Как правило, с этой ошибкой мало что можно сделать. IOException (или более конкретный TagLostException) после IsoDep.connect() указывает, что считывающий телефон не смог инициировать связь с устройством HCE. Это часто происходит из-за проблем с подключением, вызванных плохой связью (например, несоответствие размеров антенны, плохая конструкция антенны, антенны, частично закрытые батареей устройства без надлежащей конструкции антенны и т. д.) или длительным временем включения (обычно только с пассивными картами и не с HCE). Таким образом, обычно эта ошибка вызвана не логическими проблемами с протоколом связи (которые вы могли бы исправить с помощью программного обеспечения), а проблемами с физическими характеристиками устройств (которые обычно требуют модификаций оборудования).

К сожалению, вы мало что можете с этим поделать. Возможные обходные пути могут быть:

  1. Постарайтесь расположить два телефона вместе (так, чтобы антенна считывающего устройства и антенна устройства HCE) были совмещены друг с другом, а металлические части корпуса устройства или батарея устройства не закрывали антенну другого устройства. При значительном различии размеров антенн постарайтесь совместить остальные границы антенн друг с другом так, чтобы меньшая оказалась внутри большей.

  2. На некоторых(!) устройствах может помочь увеличение тайм-аута передачи перед вызовом connect():

    isoDep.setTimeout(10000);
    

    Однако кажется, что этот тайм-аут не имеет никакого эффекта до вызова соединения на большинстве устройств.

  3. Существуют антенные «усилители» для NFC, которые либо расстраивают резонансную частоту ВЧ для лучшего согласования, либо уменьшают проблемы с соединением, адаптируя форму антенны к батареям, либо полностью заменяют исходную антенну. У меня нет большого опыта в этом, и я не могу процитировать какой-либо источник таких бустеров.

РЕДАКТИРОВАТЬ

После повторного прочтения вашего вопроса кажется, что связь между cr100 и устройством HCE работает, если вы установите флаг NfcAdapter.FLAG_READER_SKIP_NDEF_CHECK на устройстве чтения. Если это так, то IOException действительно может быть вызвано логической ошибкой. В этом случае может случиться так, что cr100 имеет неработающую реализацию обнаружения NFC, которая ломается, если карта ISO-DEP (например, устройство HCE) не реализует спецификацию тега NDEF. Хотя я считаю это крайне маловероятным, вы можете легко проверить это, внедрив приложение тега типа 4 в свое приложение HCE. См. мой ответ здесь и исходный код здесь для примера того, как это сделать.

person Michael Roland    schedule 21.07.2017