В настоящее время я работаю над реализацией HCE для моей Galaxy s4. У меня есть считыватель Omnikey 5321-cl, который поддерживает 7816-4. Мой класс Android выглядит так:
public class NfcHceService extends HostApduService{
private int counter = 0;
@Override
public void onStart(android.content.Intent intent, int startId)
{
new Thread(new Runnable() {
@Override
public void run()
{
while (true)
{
Log.d("HCE", String.valueOf(counter));
counter++;
try
{
Thread.sleep(1000);
} catch (InterruptedException e)
{
e.printStackTrace();
}
}
}
}).run();
}
@Override
public byte[] processCommandApdu(byte[] apdu, Bundle extras)
{
byte[] response = null;
if (apdu[2] == 0xA4)
{
// return selecting applet
response = new byte[] { (byte) 0x90, 0x00 };
}
return response;
}
@Override
public void onDeactivated(int reason)
{
android.os.Debug.waitForDebugger();
Log.d("HCE", "onDeactivated");
}
}
И мой класс читателя выглядит так:
private static final byte[] CLA_INS_P1_P2 = { 0x00, (byte) 0xA4, 0x04, 0x00 };
private static final byte[] AID_ANDROID = { (byte) 0xF0, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06 };
private static byte[] createSelectAidApdu(byte[] aid)
{
byte[] result = new byte[6 + aid.length];
System.arraycopy(CLA_INS_P1_P2, 0, result, 0, CLA_INS_P1_P2.length);
result[4] = (byte) (aid.length);
System.arraycopy(aid, 0, result, 5, aid.length);
result[result.length - 1] = 10;
return result;
}
public static void Create_MF() throws CardException
{
// --Variable declaration
Card card = null;
ResponseAPDU answer = null;
// ---------------------------------------------
// --1--Establish connection with the smart card
TerminalFactory factory = TerminalFactory.getDefault();
List<CardTerminal> terminals = factory.terminals().list();
// Use the first terminal
CardTerminal terminal = terminals.get(1);
// Connect with the card
card = terminal.connect("*");
CardChannel channel = card.getBasicChannel();
// ---------------------------------------------
byte[] selectAidApdu = createSelectAidApdu(AID_ANDROID);
answer = channel.transmit(new CommandAPDU(selectAidApdu));
Мой AID для службы hce — F0010203040506.
Теперь моя проблема заключается в том, что я получаю правильный выбор APDU на своем устройстве Android, который выглядит следующим образом:
> RX: Type 4 Tag Command (13 bytes)
CLA:0x00
INS:0xA4(Select)
P1:0x04(Name)
P2:0x00(First or Only)
LC:0x07(7)
Data(7 bytes)
00: f0 01 02 03 04 05 06
Le:0x0A(10)
Но получаю код ответа 6A82. Я так понимаю, это означает, что устройство не находит сервис. Но я не понимаю, почему. Кто-нибудь может помочь?
Манифест:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.hce"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="19"
android:targetSdkVersion="19" />
<uses-permission android:name="android.permission.NFC" />
<uses-feature android:name="android.hardware.nfc.hce" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<service
android:name=".NfcHceService"
android:exported="false"
android:permission="android.permission.BIND_NFC_SERVICE" >
<intent-filter>
<action android:name="android.nfc.cardemulation.HOST_APDU_SERVICE" />
</intent-filter>
<meta-data
android:name="android.nfc.cardemulation.host_apdu_service"
android:resource="@xml/hce_service" />
</service>
</application>
And the xml file:
<host-apdu-service xmlns:android="http://schemas.android.com/apk/res/android"
android:description="@string/my_app_hce_service" >
<aid-group
android:category="other"
android:description="@string/my_app_aid_group" >
<aid-filter android:name="F0010203040506" />
</aid-group></host-apdu-service>
Обновить
public static void Create_MF() throws CardException
{
// --Variable declaration
Card card = null;
ResponseAPDU answer = null;
// ---------------------------------------------
// --1--Establish connection with the smart card
TerminalFactory factory = TerminalFactory.getDefault();
List<CardTerminal> terminals = factory.terminals().list();
// Use the first terminal
CardTerminal terminal = terminals.get(1);
// Connect with the card
card = terminal.connect("*");
CardChannel channel = card.getBasicChannel();
// ---------------------------------------------
byte[] selectAidApdu = { 0x00, (byte) 0xA4, 0x04, 0x00, 0x07, (byte) 0xF0, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06 }; // createSelectAidApdu(AID_ANDROID);
answer = channel.transmit(new CommandAPDU(selectAidApdu));}
ОБНОВЛЕНИЕ 2
Я попытался реализовать isDefaultServiceForAid(). Но я не уверен, правильно ли я его использую и делает ли он то, что я хочу.
CardEmulation card = CardEmulation.getInstance(NfcAdapter.getDefaultAdapter(this));
ComponentName comp = new ComponentName(getApplicationContext(), NfcHceService.class);
boolean tmp = card.isDefaultServiceForAid(comp, "F0010203040506");
Действительно ли он проверяет AID моего сервиса или делает что-то еще? Я хочу проверить, доступна ли моя служба с помощью AID, который я указал в своем hce_service.xml, который объявлен в моем манифесте.
processCommandApdu
никогда не вызывается, добавив строку типаLog.d("HCE", "processCommandApdu");
в качестве первой строки кода этого метода? - person Michael Roland   schedule 14.07.2014if (apdu[2] == 0xA4)
, который должен бытьif (apdu[1] == (byte)0xA4)
(обратите внимание как на неправильный индекс, так и на неправильное целочисленное сравнение), и что вы возвращаетеnull
вместо слова состояния в других случаях (чего вам не следует делать), я не вижу ничего плохого в вашем коде (хотя это ничего не меняет, еслиprocessCommandApdu
все равно не вызывается. - person Michael Roland   schedule 14.07.2014