NFC Card Reader ACR 122 incompatible with Android

2019-03-26 07:42发布

问题:

I used the ACR 122 before and it worked flawlessly with an Android < 4.1 phone. I used it to exchange P2P SNEP messsages... But since the phone got 4.1, the reader starts beeping and flashing when you place the phone above it. Does anyone else have found this incompatibility? Any solutions?

回答1:

Since Android 4.1 Jelly Bean, NFC peer-to-peer communication is configured to use a higher bitrate (212 kbps), while earlier it was using 106 kbps (in Android devices that have Google Wallet installed it was always using 212 kbps).

I don't know whether the ACR122U can support the higher bitrate through javax.smartcardio. The NFC chip inside (NXP's PN532) can certainly do it. The low-level ISO18092/ECMA-340 protocol for 212 and 424 kbps is different from 106 kbps. With 106 kbps it uses the same modulation as ISO14443, while the higher bitrates use the same modulation as FeliCa (see page 7 and 8 of the standard). So I suspect this cannot be handled through javax.smartcardio.



回答2:

great to see someone else working on this. You can connect to the Reader via "Direct". credits go to Peter Kmet: javax.smartcardio transmit to NFC USB reader without card

This sample will just toggle the Lights

        TerminalFactory factory;
        List<CardTerminal> terminals;
        factory = TerminalFactory.getDefault();
        terminals = factory.terminals().list();
        terminal = terminals.get(0);

        byte[] response = null;
        byte[] command = new byte[] { (byte) 0xff, (byte) 0x00, (byte) 0x40, (byte) 0xd0, (byte) 0x04,
            (byte) 0x05, (byte) 0x05, (byte) 0x02, (byte) 0x01 };
        int controlCode = 0x310000 + 3500 * 4;
        Card card = null;
        card = terminal.connect("DIRECT");
        response = card.transmitControlCommand(controlCode, command);

Please update if you make any progress with Jelly Bean, I will checkout your code as well.



回答3:

This can be helpful to make a connection to terminal without a card in range:

terminal.connect("DIRECT");


回答4:

I solved the Problem with a small "hack". I am using the ACR122U with a SAM Module in it, do not know which Versions have this it's the TouchATag Reader. I found out that the reader has to pass this line, also when the phone is not in the NFC Field:

cardTerminal.connect("*").getBasicChannel();

This is because you have to put the reader into inializer mode before the phone comes into the nfc field, and this only works if you can send APDUs!

If somebody finds a solution to send APDUs without calling connect() on the cardTerminal.

This Problem is not related to the higher bitrates, I managed to get a communication with 424 kbps.

You can find a NFC API I have written with this knowledge in googel code, I will also provide some documentation there:

http://code.google.com/p/java-android-beam-api/

Comments are very welcome!



回答5:

In Java language, you can use:

1) nfctools: https://github.com/grundid/nfctools . It works on Android 4.0, 4.1, 4.2, 4.3, 4.4.

2) http://code.google.com/p/ismb-snep-java. It works on Android 4.0 and Android 4.4 (but not in 4.1,4.2,4.3). In order to support Samsung devices in 4.4 it requires some changes in the protocol (which can be taken from the nfctools, but it is better to use the nfctools to increase compatibility).