使用NFC_EXTRAS在Android ICS上启用CardEmulation

时间:2012-10-18 13:32:09

标签: android payment nfc

我需要NFC和Android的帮助。

经过大量研究以在Android上模拟Mifare 4K后,我发现存在唯一的补丁是针对2.3.4。在StackOverFlow中,NFCGuy告诉我们自Android API 14以来它不需要修补ROM,所以我们可以使用隐藏的Nfc_extras包打开卡仿真。

我使用带有反射的NFC_EXTRAS编译了一个APK,并将我的签名和包添加到了nfcee_access.xml。

在我将cardEmulationRoute设置为ON_WHEN_SCREEN_ON之后,我在logcat中得到一个输出,告诉我NFCEE是ON并且NFC_CC是ON,但是当我让我的Nexus S接近ACR122时它没有检测到仿真Mifare 4K那个2.3.4补丁制造商可以得到。我可以得到一张无法识别的智能卡(我想这是SE就像一张智能卡),但我需要使用模拟的Mifare。

我是否需要修改lib-nfc,因为它在2.3.4补丁中被修改以使该applet正常工作(Mifare Manager)?或者我应用程序访问该程序包应该是什么?

我正在下载android源码以将2.3.4补丁移植到4.1,但是看看他们发布的差异,在lib-nfc库上只有4.1的区别。 (定义评论,理论上用于卡片仿真)

也许没有必要重新编译修改过的ROM,而且我错过了一小步来获得模拟的Mifare 4k。

感谢您在StackOverFlow中为所有人提供的帮助

此致

    private void getSecureElement(){
    try{
        //Obtenemos la clase NFCAdapterExtras
        Class nfcExtrasClazz = Class.forName("com.android.nfc_extras.NfcAdapterExtras");

        if (nfcExtrasClazz == null){
            Log.w("EnableCardEmu", "No existe la clase Extras");
            return;
        }

        Log.w("EnableCardEmu", "Existe la clase");

        //Obtenemos el método "get" de dicha clase
        Method getMethod = nfcExtrasClazz.getMethod("get", Class.forName("android.nfc.NfcAdapter"));

        if (getMethod == null) {
            Log.w("EnableCardEmu", "No existe el método");
        } else {
            Log.w("EnableCardEmu", "Existe el método");
            //Obtenemos el manager del componente NFC del dispositivo
            NfcAdapter adapter = NfcAdapter.getDefaultAdapter(this);

            if (adapter == null)
                Log.w("EnableCardEmu", "Adapter es null");
            else {
                //Instancia del SecureElement
                Log.w("EnableCardEmu", "Adapter NO es null");
                nfcExtras = getMethod.invoke(null, adapter);

                Method getEEMethod = nfcExtras.getClass().getMethod("getEmbeddedExecutionEnvironment", 
                        (Class[]) null);
                embebbed = getEEMethod.invoke(nfcExtras , (Object[]) null);
            }
        }
    } catch (InvocationTargetException ee){
        Log.w("EnableCardEmu", ee.getTargetException());
    }
    catch (Exception e){
        Log.w("EnableCardEmu", e.getClass().getName() + " / " + e.getMessage());
        StackTraceElement[] a = e.getStackTrace();
        for (StackTraceElement aa : a){
            Log.w("EnableCardEmu", aa.toString());
        }
    } 
}

    private void deactivateCardEmulation(){
    try{
        Class clss = Class.forName("com.android.nfc_extras.NfcAdapterExtras");
        Class[] cs = clss.getDeclaredClasses();
        /*          
        for (Class cc : cs){
            Log.w("EnableCardEmu", cc.getName();)
        }*/

        //Class route = Class.forName("com.android.nfc_extras.NfcAdapterExtras$CardEmulationRoute");
        Constructor c = cs[0].getConstructor(Integer.TYPE, Class.forName("com.android.nfc_extras.NfcExecutionEnvironment"));
        Object routeOn = c.newInstance(1, null);

        Class cls = nfcExtras.getClass();           
        Method mtd = cls.getMethod("setCardEmulationRoute", cs[0]);
        mtd.invoke(nfcExtras, routeOn);
    } catch (InvocationTargetException ee){
        Log.w("EnableCardEmu", ee.getTargetException());
    } catch (Exception e){
        Log.w("EnableCardEmu", e.getClass().getName() + " / " + e.getMessage());
    }
}

    private void activateCardEmulation(){
    try{

        Class clss = Class.forName("com.android.nfc_extras.NfcAdapterExtras");
        Class[] cs = clss.getDeclaredClasses();
        /*          
        for (Class cc : cs){
            Log.w("EnableCardEmu", cc.getName();)
        }*/

        //Class route = Class.forName("com.android.nfc_extras.NfcAdapterExtras$CardEmulationRoute");
        Constructor c = cs[0].getConstructor(Integer.TYPE, Class.forName("com.android.nfc_extras.NfcExecutionEnvironment"));
        Object routeOn = c.newInstance(2, embebbed);

        Class cls = nfcExtras.getClass();           
        Method mtd = cls.getMethod("setCardEmulationRoute", cs[0]);
        mtd.invoke(nfcExtras, routeOn);
    } catch (InvocationTargetException ee){
        Log.w("EnableCardEmu", ee.getTargetException());
    } catch (Exception e){
        Log.w("EnableCardEmu", e.getClass().getName() + " / " + e.getMessage());
    }
}

2 个答案:

答案 0 :(得分:3)

我认为问题可能出在Object routeOn = c.newInstance(2, embebbed)。看起来这个对象的类型不对(我不是阅读反射代码的专家)。

我就是这样做的,没有反射,而且工作正常(ISO 14443-4A卡和MIFARE Classic卡同时由设备模拟):

Context mContext; // initialize this
NfcAdapterExtras mAdapterExtras =
  NfcAdapterExtras.get(NfcAdapter.getDefaultAdapter(mContext));
NfcExecutionEnvironment mEe = mAdapterExtras.getEmbeddedExecutionEnvironment();
mAdapterExtras.setCardEmulationRoute(
  new CardEmulationRoute(CardEmulationRoute.ROUTE_ON_WHEN_SCREEN_ON, mEe));

无需为此修改libnfc-nxp。

答案 1 :(得分:3)

您的代码似乎应该正确启用卡仿真模式。我也尝试了同样的事情并遇到了类似的问题,我用MIFARE兼容的RFID阅读器无法检测到手机。

原因是我正在使用的手机有NFC支持,但没有安装嵌入式安全元件。安全元件位于SIM卡上,而ICS 4.0.3中的内置NFC支持无法定位和使用该安全元件。

同一品牌/型号的手机在通过某些手机供应商销售时具有嵌入式安全元件,但未通过我的提供商销售。

我得到了第一个提示,当我尝试在我的NfcExecutionEnvironment实例上执行“打开”并且以下输出显示在我的logcat中时,这就是原因:

D/NFC JNI (  911): phLibNfc_SE_GetSecureElementList()
D/NFC JNI (  911): 
D/NFC JNI (  911): > Number of Secure Element(s) : 0
E/NFC JNI (  911): phLibNfc_SE_GetSecureElementList(): No SMX detected