NfcService.java revision 7a7f8f8fd82936f0ee005ccfa7ac5c36760ed902
1f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly/* 2f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * Copyright (C) 2010 The Android Open Source Project 3f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * 4f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * Licensed under the Apache License, Version 2.0 (the "License"); 5f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * you may not use this file except in compliance with the License. 6f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * You may obtain a copy of the License at 7f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * 8f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * http://www.apache.org/licenses/LICENSE-2.0 9f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * 10f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * Unless required by applicable law or agreed to in writing, software 11f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * distributed under the License is distributed on an "AS IS" BASIS, 12f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * See the License for the specific language governing permissions and 14f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly * limitations under the License. 15f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly */ 16f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1713d8819d9d716c8f0ba03288d058f0bd462d70a7Nick Pellypackage com.android.nfc; 18f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 19d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamiltonimport com.android.internal.nfc.LlcpServiceSocket; 20d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamiltonimport com.android.internal.nfc.LlcpSocket; 2124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamiltonimport com.android.nfc.RegisteredComponentCache.ComponentInfo; 22ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamiltonimport com.android.nfc.ndefpush.NdefPushClient; 23ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamiltonimport com.android.nfc.ndefpush.NdefPushServer; 248afd14d3b23d3124c48ee275ba2845aede6542a1Jeff Hamiltonimport com.android.nfc3.R; 25d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton 2605973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamiltonimport android.app.Activity; 271a9eca5f84036d7dd3e28000290caa2f641856deNick Pellyimport android.app.ActivityManagerNative; 282f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pellyimport android.app.Application; 2924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamiltonimport android.app.IActivityManager; 3005973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamiltonimport android.app.PendingIntent; 3193d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamiltonimport android.app.PendingIntent.CanceledException; 32bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parksimport android.app.StatusBarManager; 33b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneauimport android.content.ActivityNotFoundException; 3413d8819d9d716c8f0ba03288d058f0bd462d70a7Nick Pellyimport android.content.BroadcastReceiver; 3505973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamiltonimport android.content.ComponentName; 3613d8819d9d716c8f0ba03288d058f0bd462d70a7Nick Pellyimport android.content.Context; 3713d8819d9d716c8f0ba03288d058f0bd462d70a7Nick Pellyimport android.content.Intent; 3813d8819d9d716c8f0ba03288d058f0bd462d70a7Nick Pellyimport android.content.IntentFilter; 390e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pellyimport android.content.SharedPreferences; 4093d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamiltonimport android.content.pm.PackageManager; 4124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamiltonimport android.content.pm.ResolveInfo; 423fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamiltonimport android.net.Uri; 43bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parksimport android.nfc.ApduList; 44f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.ErrorCodes; 45f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.FormatException; 46f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.ILlcpConnectionlessSocket; 47f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.ILlcpServiceSocket; 48f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.ILlcpSocket; 490e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pellyimport android.nfc.INfcAdapter; 5049d53329a0c720a7e430220d77805bc1763545b1Nick Pellyimport android.nfc.INfcAdapterExtras; 51f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.INfcTag; 52f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.IP2pInitiator; 53f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.IP2pTarget; 54f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.LlcpPacket; 55f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.NdefMessage; 563fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamiltonimport android.nfc.NdefRecord; 57f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.nfc.NfcAdapter; 580e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pellyimport android.nfc.Tag; 5924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamiltonimport android.nfc.TechListParcel; 609d5511f2640903a79d24578a12a93e50a96f0c0eMartijn Coenenimport android.nfc.TransceiveResult; 617c034a7fe7d36b1ab039af2c44717812ea02657eNick Pellyimport android.os.AsyncTask; 62b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamiltonimport android.os.Bundle; 63b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneauimport android.os.Handler; 6449d53329a0c720a7e430220d77805bc1763545b1Nick Pellyimport android.os.IBinder; 65b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneauimport android.os.Message; 66533043d1003de2f6a20a29201100d94c3c7bc9caNick Pellyimport android.os.PowerManager; 67f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.os.RemoteException; 6813d8819d9d716c8f0ba03288d058f0bd462d70a7Nick Pellyimport android.os.ServiceManager; 69f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pellyimport android.util.Log; 70f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 7157d376f1ee1a3939977b95759525585abb9601fbJeff Hamiltonimport java.io.ByteArrayOutputStream; 72bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parksimport java.io.DataInputStream; 73bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parksimport java.io.DataOutputStream; 7457d376f1ee1a3939977b95759525585abb9601fbJeff Hamiltonimport java.io.FileInputStream; 7557d376f1ee1a3939977b95759525585abb9601fbJeff Hamiltonimport java.io.FileNotFoundException; 7657d376f1ee1a3939977b95759525585abb9601fbJeff Hamiltonimport java.io.FileOutputStream; 7757d376f1ee1a3939977b95759525585abb9601fbJeff Hamiltonimport java.io.IOException; 783fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamiltonimport java.nio.charset.Charsets; 7924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamiltonimport java.util.ArrayList; 803fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamiltonimport java.util.Arrays; 813ca6ffff72f4599e80f85de5ae8e7f55012f38d6Jeff Hamiltonimport java.util.HashMap; 822436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Feririmport java.util.Iterator; 83bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parksimport java.util.List; 843ca6ffff72f4599e80f85de5ae8e7f55012f38d6Jeff Hamilton 852f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pellypublic class NfcService extends Application { 86bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks private static final String ACTION_MASTER_CLEAR_NOTIFICATION = "android.intent.action.MASTER_CLEAR_NOTIFICATION"; 87bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 88ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton static final boolean DBG = false; 89fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks 9057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton private static final String MY_TAG_FILE_NAME = "mytag"; 91bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks private static final String TEAR_DOWN_SCRIPTS_FILE_NAME = "teardowns"; 9257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 9313d8819d9d716c8f0ba03288d058f0bd462d70a7Nick Pelly static { 9413d8819d9d716c8f0ba03288d058f0bd462d70a7Nick Pelly System.loadLibrary("nfc_jni"); 9513d8819d9d716c8f0ba03288d058f0bd462d70a7Nick Pelly } 9613d8819d9d716c8f0ba03288d058f0bd462d70a7Nick Pelly 973fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton /** 983fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton * NFC Forum "URI Record Type Definition" 993fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton * 1003fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton * This is a mapping of "URI Identifier Codes" to URI string prefixes, 1013fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton * per section 3.2.2 of the NFC Forum URI Record Type Definition document. 1023fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton */ 1033fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton private static final String[] URI_PREFIX_MAP = new String[] { 1043fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "", // 0x00 1053fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "http://www.", // 0x01 1063fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "https://www.", // 0x02 1073fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "http://", // 0x03 1083fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "https://", // 0x04 1093fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "tel:", // 0x05 1103fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "mailto:", // 0x06 1113fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "ftp://anonymous:anonymous@", // 0x07 1123fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "ftp://ftp.", // 0x08 1133fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "ftps://", // 0x09 1143fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "sftp://", // 0x0A 1153fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "smb://", // 0x0B 1163fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "nfs://", // 0x0C 1173fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "ftp://", // 0x0D 1183fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "dav://", // 0x0E 1193fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "news:", // 0x0F 1203fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "telnet://", // 0x10 1213fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "imap:", // 0x11 1223fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "rtsp://", // 0x12 1233fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "urn:", // 0x13 1243fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "pop:", // 0x14 1253fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "sip:", // 0x15 1263fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "sips:", // 0x16 1273fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "tftp:", // 0x17 1283fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "btspp://", // 0x18 1293fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "btl2cap://", // 0x19 1303fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "btgoep://", // 0x1A 1313fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "tcpobex://", // 0x1B 1323fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "irdaobex://", // 0x1C 1333fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "file://", // 0x1D 1343fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "urn:epc:id:", // 0x1E 1353fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "urn:epc:tag:", // 0x1F 1363fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "urn:epc:pat:", // 0x20 1373fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "urn:epc:raw:", // 0x21 1383fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton "urn:epc:", // 0x22 1393fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton }; 1403fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 141d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton public static final String SERVICE_NAME = "nfc"; 142fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks 143f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final String TAG = "NfcService"; 144f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 145bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly private static final String NFC_PERM = android.Manifest.permission.NFC; 146bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly private static final String NFC_PERM_ERROR = "NFC permission required"; 147bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly private static final String ADMIN_PERM = android.Manifest.permission.WRITE_SECURE_SETTINGS; 148bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly private static final String ADMIN_PERM_ERROR = "WRITE_SECURE_SETTINGS permission required"; 14993d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton // STOPSHIP: This needs to be updated to the line below 15093d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton// private static final String NFCEE_ADMIN_PERM = "com.android.nfc.permission.NFCEE_ADMIN"; 15193d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton private static final String NFCEE_ADMIN_PERM = NFC_PERM; 15293d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton private static final String NFCEE_ADMIN_PERM_ERROR = "NFCEE_ADMIN permission required"; 153bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1540e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF = "NfcServicePrefs"; 155f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1560e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF_NFC_ON = "nfc_on"; 1570e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final boolean NFC_ON_DEFAULT = true; 158f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1590e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF_LLCP_LTO = "llcp_lto"; 160e1fac398523a97e3bcf513393a91478d79a8763fSylvain Fonteneau private static final int LLCP_LTO_DEFAULT = 150; 1610e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int LLCP_LTO_MAX = 255; 162f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1630e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /** Maximum Information Unit */ 1640e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF_LLCP_MIU = "llcp_miu"; 1650e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int LLCP_MIU_DEFAULT = 128; 1660e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int LLCP_MIU_MAX = 2176; 167f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1680e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /** Well Known Service List */ 1690e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF_LLCP_WKS = "llcp_wks"; 1700e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int LLCP_WKS_DEFAULT = 1; 1710e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int LLCP_WKS_MAX = 15; 172f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1730e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF_LLCP_OPT = "llcp_opt"; 1740e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int LLCP_OPT_DEFAULT = 0; 1750e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int LLCP_OPT_MAX = 3; 176f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1770e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF_DISCOVERY_A = "discovery_a"; 1780e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final boolean DISCOVERY_A_DEFAULT = true; 179f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1800e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF_DISCOVERY_B = "discovery_b"; 1810e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final boolean DISCOVERY_B_DEFAULT = true; 182f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1830e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF_DISCOVERY_F = "discovery_f"; 1840e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final boolean DISCOVERY_F_DEFAULT = true; 185f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1860e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF_DISCOVERY_15693 = "discovery_15693"; 1870e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final boolean DISCOVERY_15693_DEFAULT = true; 188f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1890e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final String PREF_DISCOVERY_NFCIP = "discovery_nfcip"; 190a989351c7383aa6b3a6086b10f32c39c1d28fa5dJeff Hamilton private static final boolean DISCOVERY_NFCIP_DEFAULT = true; 191f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1920e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /** NFC Reader Discovery mode for enableDiscovery() */ 1930e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int DISCOVERY_MODE_READER = 0; 194f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 195f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final int PROPERTY_LLCP_LTO = 0; 196f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final String PROPERTY_LLCP_LTO_VALUE = "llcp.lto"; 1970e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int PROPERTY_LLCP_MIU = 1; 198f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final String PROPERTY_LLCP_MIU_VALUE = "llcp.miu"; 1990e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int PROPERTY_LLCP_WKS = 2; 200f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final String PROPERTY_LLCP_WKS_VALUE = "llcp.wks"; 2010e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int PROPERTY_LLCP_OPT = 3; 202f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final String PROPERTY_LLCP_OPT_VALUE = "llcp.opt"; 203f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final int PROPERTY_NFC_DISCOVERY_A = 4; 204f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final String PROPERTY_NFC_DISCOVERY_A_VALUE = "discovery.iso14443A"; 2050e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int PROPERTY_NFC_DISCOVERY_B = 5; 206f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final String PROPERTY_NFC_DISCOVERY_B_VALUE = "discovery.iso14443B"; 2070e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int PROPERTY_NFC_DISCOVERY_F = 6; 208f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final String PROPERTY_NFC_DISCOVERY_F_VALUE = "discovery.felica"; 2090e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int PROPERTY_NFC_DISCOVERY_15693 = 7; 210f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final String PROPERTY_NFC_DISCOVERY_15693_VALUE = "discovery.iso15693"; 2110e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private static final int PROPERTY_NFC_DISCOVERY_NFCIP = 8; 212f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private static final String PROPERTY_NFC_DISCOVERY_NFCIP_VALUE = "discovery.nfcip"; 213f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 214b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau static final int MSG_NDEF_TAG = 0; 215b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau static final int MSG_CARD_EMULATION = 1; 216b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau static final int MSG_LLCP_LINK_ACTIVATION = 2; 217b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau static final int MSG_LLCP_LINK_DEACTIVATED = 3; 218b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau static final int MSG_TARGET_DESELECTED = 4; 21957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton static final int MSG_SHOW_MY_TAG_ICON = 5; 22057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton static final int MSG_HIDE_MY_TAG_ICON = 6; 221b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton static final int MSG_MOCK_NDEF = 7; 222c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas static final int MSG_SE_FIELD_ACTIVATED = 8; 223c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas static final int MSG_SE_FIELD_DEACTIVATED = 9; 224b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau 22549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly // Copied from com.android.nfc_extras to avoid library dependency 22649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly // Must keep in sync with com.android.nfc_extras 22749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly static final int ROUTE_OFF = 1; 22849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly static final int ROUTE_ON_WHEN_SCREEN_ON = 2; 22949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public static final String ACTION_RF_FIELD_ON_DETECTED = 23049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly "com.android.nfc_extras.action.RF_FIELD_ON_DETECTED"; 23149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public static final String ACTION_RF_FIELD_OFF_DETECTED = 23249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly "com.android.nfc_extras.action.RF_FIELD_OFF_DETECTED"; 23349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public static final String ACTION_AID_SELECTED = 23449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly "com.android.nfc_extras.action.AID_SELECTED"; 23549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public static final String EXTRA_AID = "com.android.nfc_extras.extra.AID"; 23649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly 23705973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton // Locked on mNfcAdapter 23805973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton PendingIntent mDispatchOverrideIntent; 23924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton IntentFilter[] mDispatchOverrideFilters; 24049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly String[][] mDispatchOverrideTechLists; 24105973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton 24274180bda362a8bc9d2f701d2c17bec0f63c20bbfBrad Fitzpatrick // TODO: none of these appear to be synchronized but are 24374180bda362a8bc9d2f701d2c17bec0f63c20bbfBrad Fitzpatrick // read/written from different threads (notably Binder threads)... 244f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private int mGeneratedSocketHandle = 0; 24574180bda362a8bc9d2f701d2c17bec0f63c20bbfBrad Fitzpatrick private volatile boolean mIsNfcEnabled = false; 24649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private boolean mIsDiscoveryOn = false; 24749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly 24849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly // NFC Execution Environment 24949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly // fields below are protected by this 25049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private static final int SECURE_ELEMENT_ID = 11259375; //TODO: remove hard-coded value 2510bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas private NativeNfcSecureElement mSecureElement; 25249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private OpenSecureElement mOpenEe; // null when EE closed 25349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private int mEeRoutingState; // contactless interface routing 2540bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 2552f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly // fields below are used in multiple threads and protected by synchronized(this) 2562f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly private final HashMap<Integer, Object> mObjectMap = new HashMap<Integer, Object>(); 2572f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly private final HashMap<Integer, Object> mSocketMap = new HashMap<Integer, Object>(); 25865945ad77cadb7a3bdf171497877d2325b23def5Nick Pelly private boolean mScreenOn; 2592f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly 2602f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly // fields below are final after onCreate() 26105973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton Context mContext; 2620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private NativeNfcManager mManager; 2630e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private SharedPreferences mPrefs; 2640e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private SharedPreferences.Editor mPrefsEditor; 265533043d1003de2f6a20a29201100d94c3c7bc9caNick Pelly private PowerManager.WakeLock mWakeLock; 2661a9eca5f84036d7dd3e28000290caa2f641856deNick Pelly private IActivityManager mIActivityManager; 267ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton NdefPushClient mNdefPushClient; 268ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton NdefPushServer mNdefPushServer; 26924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton RegisteredComponentCache mTechListFilters; 270d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton 271d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton private static NfcService sService; 272d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton 273bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks private HashMap<String, ApduList> mTearDownApdus = new HashMap<String, ApduList>(); 274bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 27593d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton public static void enforceAdminPerm(Context context) { 27693d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton int admin = context.checkCallingOrSelfPermission(ADMIN_PERM); 27793d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton int nfcee = context.checkCallingOrSelfPermission(NFCEE_ADMIN_PERM); 27893d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton if (admin != PackageManager.PERMISSION_GRANTED 27993d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton && nfcee != PackageManager.PERMISSION_GRANTED) { 28093d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton throw new SecurityException(ADMIN_PERM_ERROR); 28193d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton } 28293d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton } 28393d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton 28493d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton public static void enforceNfceeAdminPerm(Context context) { 28593d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton context.enforceCallingOrSelfPermission(NFCEE_ADMIN_PERM, NFCEE_ADMIN_PERM_ERROR); 28693d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton } 28793d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton 288d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton public static NfcService getInstance() { 289d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton return sService; 290d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton } 291f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 2920e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly @Override 2930e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public void onCreate() { 2942f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly super.onCreate(); 2952f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly 2962f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly Log.i(TAG, "Starting NFC service"); 2972f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly 298d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton sService = this; 299d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton 3000e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mContext = this; 301b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau mManager = new NativeNfcManager(mContext, this); 3020e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mManager.initializeNativeStructure(); 30374180bda362a8bc9d2f701d2c17bec0f63c20bbfBrad Fitzpatrick 304ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton mNdefPushClient = new NdefPushClient(this); 305ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton mNdefPushServer = new NdefPushServer(); 30657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 30724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton mTechListFilters = new RegisteredComponentCache(this, 3088d69f5205e3c4a7c2eb869bbf4983d3d9fe45ab0Nick Pelly NfcAdapter.ACTION_TECH_DISCOVERED, NfcAdapter.ACTION_TECH_DISCOVERED); 30924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton 3100bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas mSecureElement = new NativeNfcSecureElement(); 3110bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 3120e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefs = mContext.getSharedPreferences(PREF, Context.MODE_PRIVATE); 3130e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor = mPrefs.edit(); 314f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 315d6fdd3fbb880f1503d56616608e6823b51320dc3Nick Pelly mIsNfcEnabled = false; // real preference read later 316f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 317533043d1003de2f6a20a29201100d94c3c7bc9caNick Pelly PowerManager pm = (PowerManager)getSystemService(Context.POWER_SERVICE); 31852028e699e43e6322021098bbefd761fe6596747Nick Pelly mScreenOn = pm.isScreenOn(); 319533043d1003de2f6a20a29201100d94c3c7bc9caNick Pelly mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "NfcService"); 320533043d1003de2f6a20a29201100d94c3c7bc9caNick Pelly 3211a9eca5f84036d7dd3e28000290caa2f641856deNick Pelly mIActivityManager = ActivityManagerNative.getDefault(); 3221a9eca5f84036d7dd3e28000290caa2f641856deNick Pelly 323bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks readTearDownApdus(); 324bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 325d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton ServiceManager.addService(SERVICE_NAME, mNfcAdapter); 326f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 327eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton IntentFilter filter = new IntentFilter(NativeNfcManager.INTERNAL_TARGET_DESELECTED_ACTION); 32865945ad77cadb7a3bdf171497877d2325b23def5Nick Pelly filter.addAction(Intent.ACTION_SCREEN_OFF); 32965945ad77cadb7a3bdf171497877d2325b23def5Nick Pelly filter.addAction(Intent.ACTION_SCREEN_ON); 330bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks filter.addAction(ACTION_MASTER_CLEAR_NOTIFICATION); 331bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks mContext.registerReceiver(mReceiver, filter); 332bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 333bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks filter = new IntentFilter(); 334bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks filter.addAction(Intent.ACTION_PACKAGE_REMOVED); 335bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks filter.addDataScheme("package"); 336bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 3370e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mContext.registerReceiver(mReceiver, filter); 3380e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 3390e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly Thread t = new Thread() { 3400e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly @Override 3410e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public void run() { 3420e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean nfc_on = mPrefs.getBoolean(PREF_NFC_ON, NFC_ON_DEFAULT); 3430e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (nfc_on) { 3440e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly _enable(false); 345f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 346f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 3470e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly }; 3480e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly t.start(); 3490e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 3500e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 3510e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly @Override 3522f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly public void onTerminate() { 3532f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly super.onTerminate(); 3542f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly // NFC application is persistent, it should not be destroyed by framework 3550e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly Log.wtf(TAG, "NFC service is under attack!"); 3560e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 3570e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 3580e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private final INfcAdapter.Stub mNfcAdapter = new INfcAdapter.Stub() { 3593ca6ffff72f4599e80f85de5ae8e7f55012f38d6Jeff Hamilton /** Protected by "this" */ 3603ca6ffff72f4599e80f85de5ae8e7f55012f38d6Jeff Hamilton NdefMessage mLocalMessage = null; 3614acf84356e35861f7df6d07a2b9ff84842a9f221Nick Pelly 362fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 3630e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public boolean enable() throws RemoteException { 36493d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton NfcService.enforceAdminPerm(mContext); 3650e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 3660e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean isSuccess = false; 3670e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean previouslyEnabled = isEnabled(); 3680e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!previouslyEnabled) { 3690e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly reset(); 3700e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly isSuccess = _enable(previouslyEnabled); 371f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 3720e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return isSuccess; 373f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 374f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 375fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 3760e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public boolean disable() throws RemoteException { 3770e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean isSuccess = false; 37893d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton NfcService.enforceAdminPerm(mContext); 3790e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean previouslyEnabled = isEnabled(); 380ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "Disabling NFC. previous=" + previouslyEnabled); 3810e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 3820e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (previouslyEnabled) { 3836f7fd8dbda9092e36e58273e843ae3e4a200e2ffJeff Hamilton /* tear down the my tag server */ 384ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton mNdefPushServer.stop(); 385e260fc52ed2b4e6e1b7394d6f841d01c98888327Arnaud Ferir 386e260fc52ed2b4e6e1b7394d6f841d01c98888327Arnaud Ferir // Stop watchdog if tag present 387e260fc52ed2b4e6e1b7394d6f841d01c98888327Arnaud Ferir // A convenient way to stop the watchdog properly consists of 388e260fc52ed2b4e6e1b7394d6f841d01c98888327Arnaud Ferir // disconnecting the tag. The polling loop shall be stopped before 389e260fc52ed2b4e6e1b7394d6f841d01c98888327Arnaud Ferir // to avoid the tag being discovered again. 39049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mIsDiscoveryOn = false; 39149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly applyRouting(); 392e260fc52ed2b4e6e1b7394d6f841d01c98888327Arnaud Ferir maybeDisconnectTarget(); 393e260fc52ed2b4e6e1b7394d6f841d01c98888327Arnaud Ferir 3940e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly isSuccess = mManager.deinitialize(); 395ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "NFC success of deinitialize = " + isSuccess); 3960e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (isSuccess) { 3970e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mIsNfcEnabled = false; 398a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton // Clear out any old dispatch overrides and NDEF push message 39905973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton synchronized (this) { 40005973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton mDispatchOverrideFilters = null; 40105973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton mDispatchOverrideIntent = null; 40205973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 403a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton mNdefPushClient.setForegroundMessage(null); 4040e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 405f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 4060e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 4070e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly updateNfcOnSetting(previouslyEnabled); 4080e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 4090e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return isSuccess; 410f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 411f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 412fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 41305973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton public void enableForegroundDispatch(ComponentName activity, PendingIntent intent, 41424dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton IntentFilter[] filters, TechListParcel techListsParcel) { 415a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton // Permission check 41605973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 417a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton 418a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton // Argument validation 419a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton if (activity == null || intent == null) { 420ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton throw new IllegalArgumentException(); 421ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton } 422a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton 423a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton // Validate the IntentFilters 424a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton if (filters != null) { 425a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton if (filters.length == 0) { 426a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton filters = null; 427a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton } else { 428a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton for (IntentFilter filter : filters) { 429a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton if (filter == null) { 430a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton throw new IllegalArgumentException("null IntentFilter"); 431a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton } 432a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton } 433a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton } 434a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton } 435a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton 43624dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // Validate the tech lists 43724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton String[][] techLists = null; 43824dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (techListsParcel != null) { 43924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton techLists = techListsParcel.getTechLists(); 44024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 44149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly 44205973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton synchronized (this) { 443a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton if (mDispatchOverrideIntent != null) { 44405973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton Log.e(TAG, "Replacing active dispatch overrides"); 44505973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 44605973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton mDispatchOverrideIntent = intent; 447a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton mDispatchOverrideFilters = filters; 44824dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton mDispatchOverrideTechLists = techLists; 44905973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 45005973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 45105973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton 45205973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton @Override 45305973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton public void disableForegroundDispatch(ComponentName activity) { 45405973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 45505973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton synchronized (this) { 456a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton if (mDispatchOverrideIntent == null) { 45705973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton Log.e(TAG, "No active foreground dispatching"); 45805973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 45905973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton mDispatchOverrideIntent = null; 460a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton mDispatchOverrideFilters = null; 46105973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 46205973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 46305973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton 46405973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton @Override 465ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton public void enableForegroundNdefPush(ComponentName activity, NdefMessage msg) { 466ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 467ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton if (activity == null || msg == null) { 468ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton throw new IllegalArgumentException(); 469ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton } 470ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton if (mNdefPushClient.setForegroundMessage(msg)) { 471ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton Log.e(TAG, "Replacing active NDEF push message"); 472ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton } 473ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton } 474ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton 475ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton @Override 476ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton public void disableForegroundNdefPush(ComponentName activity) { 477ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 478ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton if (!mNdefPushClient.setForegroundMessage(null)) { 479ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton Log.e(TAG, "No active foreground NDEF push message"); 480ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton } 481ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton } 482ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton 483ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton @Override 4840e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int createLlcpConnectionlessSocket(int sap) throws RemoteException { 485d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 486bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 487f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 488f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 489f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 490f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 491f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 4920e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Check SAP is not already used */ 4930e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 4941878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* Store the socket handle */ 4951878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau int sockeHandle = mGeneratedSocketHandle; 4961878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau NativeLlcpConnectionlessSocket socket; 497f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 4981878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau socket = mManager.doCreateLlcpConnectionlessSocket(sap); 4991878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau if (socket != null) { 5001878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau synchronized(NfcService.this) { 5011878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* update socket handle generation */ 5021878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau mGeneratedSocketHandle++; 5030e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 5041878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* Add the socket into the socket map */ 5051878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau mSocketMap.put(mGeneratedSocketHandle, socket); 5061878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return mGeneratedSocketHandle; 507f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 508f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 5091878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* Get Error Status */ 5101878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau int errorStatus = mManager.doGetLastError(); 5111878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau 5121878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau switch (errorStatus) { 5131878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau case ErrorCodes.ERROR_BUFFER_TO_SMALL: 5141878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return ErrorCodes.ERROR_BUFFER_TO_SMALL; 5151878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau case ErrorCodes.ERROR_INSUFFICIENT_RESOURCES: 5161878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return ErrorCodes.ERROR_INSUFFICIENT_RESOURCES; 5171878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau default: 5181878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return ErrorCodes.ERROR_SOCKET_CREATION; 5191878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau } 520f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 521f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 522f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 523fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 5240e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int createLlcpServiceSocket(int sap, String sn, int miu, int rw, int linearBufferLength) 5250e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly throws RemoteException { 526d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 527bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 528f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 529f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 530f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 531f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 532f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 5331878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau NativeLlcpServiceSocket socket; 5340e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 5351878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau socket = mManager.doCreateLlcpServiceSocket(sap, sn, miu, rw, linearBufferLength); 5361878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau if (socket != null) { 5371878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau synchronized(NfcService.this) { 5381878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* update socket handle generation */ 5391878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau mGeneratedSocketHandle++; 5400e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 5411878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* Add the socket into the socket map */ 5421878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau mSocketMap.put(mGeneratedSocketHandle, socket); 5431878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return mGeneratedSocketHandle; 544f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 545f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 5461878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* Get Error Status */ 5471878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau int errorStatus = mManager.doGetLastError(); 5481878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau 5491878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau switch (errorStatus) { 5501878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau case ErrorCodes.ERROR_BUFFER_TO_SMALL: 5511878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return ErrorCodes.ERROR_BUFFER_TO_SMALL; 5521878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau case ErrorCodes.ERROR_INSUFFICIENT_RESOURCES: 5531878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return ErrorCodes.ERROR_INSUFFICIENT_RESOURCES; 5541878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau default: 5551878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return ErrorCodes.ERROR_SOCKET_CREATION; 5561878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau } 557f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 558f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 559f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 560fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 5610e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int createLlcpSocket(int sap, int miu, int rw, int linearBufferLength) 5620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly throws RemoteException { 563d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 564bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 565f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 566f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 567f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 568f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 569f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 5701878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau if (DBG) Log.d(TAG, "creating llcp socket"); 5711878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau NativeLlcpSocket socket; 572bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 5731878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau socket = mManager.doCreateLlcpSocket(sap, miu, rw, linearBufferLength); 574f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 5751878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau if (socket != null) { 5761878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau synchronized(NfcService.this) { 5771878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* update socket handle generation */ 5781878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau mGeneratedSocketHandle++; 5790e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 5801878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* Add the socket into the socket map */ 5811878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau mSocketMap.put(mGeneratedSocketHandle, socket); 5821878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return mGeneratedSocketHandle; 5830e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 584f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 5851878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* Get Error Status */ 5861878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau int errorStatus = mManager.doGetLastError(); 5871878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau 5881878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau Log.d(TAG, "failed to create llcp socket: " + ErrorCodes.asString(errorStatus)); 5891878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau 5901878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau switch (errorStatus) { 5911878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau case ErrorCodes.ERROR_BUFFER_TO_SMALL: 5921878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return ErrorCodes.ERROR_BUFFER_TO_SMALL; 5931878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau case ErrorCodes.ERROR_INSUFFICIENT_RESOURCES: 5941878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return ErrorCodes.ERROR_INSUFFICIENT_RESOURCES; 5951878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau default: 5961878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return ErrorCodes.ERROR_SOCKET_CREATION; 5971878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau } 598f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 599f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 600f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 601fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 6020e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public ILlcpConnectionlessSocket getLlcpConnectionlessInterface() throws RemoteException { 603d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 6040e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return mLlcpConnectionlessSocketService; 6050e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 606bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 607fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 6080e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public ILlcpSocket getLlcpInterface() throws RemoteException { 609d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 6100e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return mLlcpSocket; 6110e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 612f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 613fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 6140e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public ILlcpServiceSocket getLlcpServiceInterface() throws RemoteException { 615d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 6160e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return mLlcpServerSocketService; 6170e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 618f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 619fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 6200e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public INfcTag getNfcTagInterface() throws RemoteException { 621d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 6220e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return mNfcTagService; 6230e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 6240e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 625fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 6260e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public IP2pInitiator getP2pInitiatorInterface() throws RemoteException { 627d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 6280e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return mP2pInitiatorService; 6290e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 6300e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 631fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 6320e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public IP2pTarget getP2pTargetInterface() throws RemoteException { 633d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 6340e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return mP2pTargetService; 6350e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 6360e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 637bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks @Override 63849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public INfcAdapterExtras getNfcAdapterExtrasInterface() { 63993d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton NfcService.enforceNfceeAdminPerm(mContext); 64049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly return mExtrasService; 6410bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 6420bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 643fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 6440e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public String getProperties(String param) throws RemoteException { 645d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 6460e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 6470e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (param == null) { 6480e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 6490e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 6500e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 6510e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (param.equals(PROPERTY_LLCP_LTO_VALUE)) { 6520e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return Integer.toString(mPrefs.getInt(PREF_LLCP_LTO, LLCP_LTO_DEFAULT)); 6530e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_LLCP_MIU_VALUE)) { 6540e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return Integer.toString(mPrefs.getInt(PREF_LLCP_MIU, LLCP_MIU_DEFAULT)); 6550e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_LLCP_WKS_VALUE)) { 6560e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return Integer.toString(mPrefs.getInt(PREF_LLCP_WKS, LLCP_WKS_DEFAULT)); 6570e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_LLCP_OPT_VALUE)) { 6580e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return Integer.toString(mPrefs.getInt(PREF_LLCP_OPT, LLCP_OPT_DEFAULT)); 6590e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_NFC_DISCOVERY_A_VALUE)) { 6600e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return Boolean.toString(mPrefs.getBoolean(PREF_DISCOVERY_A, DISCOVERY_A_DEFAULT)); 6610e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_NFC_DISCOVERY_B_VALUE)) { 6620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return Boolean.toString(mPrefs.getBoolean(PREF_DISCOVERY_B, DISCOVERY_B_DEFAULT)); 6630e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_NFC_DISCOVERY_F_VALUE)) { 6640e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return Boolean.toString(mPrefs.getBoolean(PREF_DISCOVERY_F, DISCOVERY_F_DEFAULT)); 6650e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_NFC_DISCOVERY_NFCIP_VALUE)) { 6660e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return Boolean.toString(mPrefs.getBoolean(PREF_DISCOVERY_NFCIP, DISCOVERY_NFCIP_DEFAULT)); 6670e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_NFC_DISCOVERY_15693_VALUE)) { 6680e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return Boolean.toString(mPrefs.getBoolean(PREF_DISCOVERY_15693, DISCOVERY_15693_DEFAULT)); 669f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 6700e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return "Unknown property"; 671f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 672f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 673f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 674fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 6750e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public boolean isEnabled() throws RemoteException { 6760e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return mIsNfcEnabled; 6770e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 6780e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 679fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 6800e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int setProperties(String param, String value) throws RemoteException { 68193d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton NfcService.enforceAdminPerm(mContext); 6820e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 6830e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (isEnabled()) { 6840e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_NFC_ON; 6850e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 6860e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 6870e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly int val; 6880e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 6890e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Check params validity */ 6900e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (param == null || value == null) { 6910e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_INVALID_PARAM; 6920e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 6930e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 6940e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (param.equals(PROPERTY_LLCP_LTO_VALUE)) { 6950e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly val = Integer.parseInt(value); 6960e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 6970e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Check params */ 6980e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (val > LLCP_LTO_MAX) 6990e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_INVALID_PARAM; 7000e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7010e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Store value */ 7020e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor.putInt(PREF_LLCP_LTO, val); 7032f386c612159a48933c12dbcaf6735bd9ad1c660Brad Fitzpatrick mPrefsEditor.apply(); 7040e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7050e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Update JNI */ 7060e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mManager.doSetProperties(PROPERTY_LLCP_LTO, val); 7070e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7080e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_LLCP_MIU_VALUE)) { 7090e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly val = Integer.parseInt(value); 7100e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7110e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Check params */ 7120e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if ((val < LLCP_MIU_DEFAULT) || (val > LLCP_MIU_MAX)) 7130e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_INVALID_PARAM; 7140e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7150e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Store value */ 7160e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor.putInt(PREF_LLCP_MIU, val); 7172f386c612159a48933c12dbcaf6735bd9ad1c660Brad Fitzpatrick mPrefsEditor.apply(); 7180e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7190e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Update JNI */ 7200e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mManager.doSetProperties(PROPERTY_LLCP_MIU, val); 7210e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7220e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_LLCP_WKS_VALUE)) { 7230e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly val = Integer.parseInt(value); 7240e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7250e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Check params */ 7260e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (val > LLCP_WKS_MAX) 7270e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_INVALID_PARAM; 7280e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7290e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Store value */ 7300e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor.putInt(PREF_LLCP_WKS, val); 7312f386c612159a48933c12dbcaf6735bd9ad1c660Brad Fitzpatrick mPrefsEditor.apply(); 7320e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7330e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Update JNI */ 7340e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mManager.doSetProperties(PROPERTY_LLCP_WKS, val); 7350e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7360e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_LLCP_OPT_VALUE)) { 7370e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly val = Integer.parseInt(value); 7380e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7390e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Check params */ 7400e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (val > LLCP_OPT_MAX) 7410e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_INVALID_PARAM; 7420e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7430e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Store value */ 7440e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor.putInt(PREF_LLCP_OPT, val); 7452f386c612159a48933c12dbcaf6735bd9ad1c660Brad Fitzpatrick mPrefsEditor.apply(); 7460e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7470e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Update JNI */ 7480e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mManager.doSetProperties(PROPERTY_LLCP_OPT, val); 7490e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7500e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_NFC_DISCOVERY_A_VALUE)) { 7510e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean b = Boolean.parseBoolean(value); 7520e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7530e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Store value */ 7540e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor.putBoolean(PREF_DISCOVERY_A, b); 7552f386c612159a48933c12dbcaf6735bd9ad1c660Brad Fitzpatrick mPrefsEditor.apply(); 7560e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7570e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Update JNI */ 7580e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mManager.doSetProperties(PROPERTY_NFC_DISCOVERY_A, b ? 1 : 0); 7590e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7600e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_NFC_DISCOVERY_B_VALUE)) { 7610e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean b = Boolean.parseBoolean(value); 7620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7630e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Store value */ 7640e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor.putBoolean(PREF_DISCOVERY_B, b); 7652f386c612159a48933c12dbcaf6735bd9ad1c660Brad Fitzpatrick mPrefsEditor.apply(); 7660e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7670e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Update JNI */ 7680e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mManager.doSetProperties(PROPERTY_NFC_DISCOVERY_B, b ? 1 : 0); 7690e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7700e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_NFC_DISCOVERY_F_VALUE)) { 7710e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean b = Boolean.parseBoolean(value); 7720e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7730e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Store value */ 7740e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor.putBoolean(PREF_DISCOVERY_F, b); 7752f386c612159a48933c12dbcaf6735bd9ad1c660Brad Fitzpatrick mPrefsEditor.apply(); 7760e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7770e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Update JNI */ 7780e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mManager.doSetProperties(PROPERTY_NFC_DISCOVERY_F, b ? 1 : 0); 7790e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7800e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_NFC_DISCOVERY_15693_VALUE)) { 7810e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean b = Boolean.parseBoolean(value); 7820e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7830e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Store value */ 7840e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor.putBoolean(PREF_DISCOVERY_15693, b); 7852f386c612159a48933c12dbcaf6735bd9ad1c660Brad Fitzpatrick mPrefsEditor.apply(); 7860e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7870e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Update JNI */ 7880e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mManager.doSetProperties(PROPERTY_NFC_DISCOVERY_15693, b ? 1 : 0); 7890e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7900e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else if (param.equals(PROPERTY_NFC_DISCOVERY_NFCIP_VALUE)) { 7910e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean b = Boolean.parseBoolean(value); 7920e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7930e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Store value */ 7940e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor.putBoolean(PREF_DISCOVERY_NFCIP, b); 7952f386c612159a48933c12dbcaf6735bd9ad1c660Brad Fitzpatrick mPrefsEditor.apply(); 7960e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 7970e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Update JNI */ 7980e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mManager.doSetProperties(PROPERTY_NFC_DISCOVERY_NFCIP, b ? 1 : 0); 7990e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 800f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 8010e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_INVALID_PARAM; 802f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 8030e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 8040e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.SUCCESS; 805f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 806d65348455ae09a467f288d65dbf924fc60c8d290Jeff Hamilton 807d65348455ae09a467f288d65dbf924fc60c8d290Jeff Hamilton @Override 808d65348455ae09a467f288d65dbf924fc60c8d290Jeff Hamilton public NdefMessage localGet() throws RemoteException { 80957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 81057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 81157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton synchronized (this) { 81257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton return mLocalMessage; 81357d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 814d65348455ae09a467f288d65dbf924fc60c8d290Jeff Hamilton } 815d65348455ae09a467f288d65dbf924fc60c8d290Jeff Hamilton 816d65348455ae09a467f288d65dbf924fc60c8d290Jeff Hamilton @Override 817d65348455ae09a467f288d65dbf924fc60c8d290Jeff Hamilton public void localSet(NdefMessage message) throws RemoteException { 81893d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton NfcService.enforceAdminPerm(mContext); 81957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 82057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton synchronized (this) { 82157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton mLocalMessage = message; 82257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton Context context = NfcService.this.getApplicationContext(); 82357d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 82457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton // Send a message to the UI thread to show or hide the icon so the requests are 82557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton // serialized and the icon can't get out of sync with reality. 82657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton if (message != null) { 82757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton FileOutputStream out = null; 82857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 82957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton try { 83057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton out = context.openFileOutput(MY_TAG_FILE_NAME, Context.MODE_PRIVATE); 83157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton byte[] bytes = message.toByteArray(); 83257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton if (bytes.length == 0) { 83357d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton Log.w(TAG, "Setting a empty mytag"); 83457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 83557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 83657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton out.write(bytes); 83757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } catch (IOException e) { 83857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton Log.e(TAG, "Could not write mytag file", e); 83957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } finally { 84057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton try { 84157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton if (out != null) { 84257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton out.flush(); 84357d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton out.close(); 84457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 84557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } catch (IOException e) { 84657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton // Ignore 84757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 84857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 84957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 85057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton // Only show the icon if NFC is enabled. 85157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton if (mIsNfcEnabled) { 85257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton sendMessage(MSG_SHOW_MY_TAG_ICON, null); 85357d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 85457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } else { 85557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton context.deleteFile(MY_TAG_FILE_NAME); 85657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton sendMessage(MSG_HIDE_MY_TAG_ICON, null); 85757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 85857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 859d65348455ae09a467f288d65dbf924fc60c8d290Jeff Hamilton } 8600e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly }; 8610e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 8620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private final ILlcpSocket mLlcpSocket = new ILlcpSocket.Stub() { 863f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau 864f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau private NativeLlcpSocket findSocket(int nativeHandle) { 865bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly Object socket = NfcService.this.findSocket(nativeHandle); 866bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly if (!(socket instanceof NativeLlcpSocket)) { 867f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau return null; 868f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau } 869bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly return (NativeLlcpSocket) socket; 870f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau } 871f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau 872fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 8730e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int close(int nativeHandle) throws RemoteException { 874d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 875bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 876f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeLlcpSocket socket = null; 877f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 878f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 879f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 880f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 881f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 882f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 883f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the socket in the hmap */ 884f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 885f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (socket != null) { 8861878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau socket.doClose(); 8871878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* Remove the socket closed from the hmap */ 8881878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau RemoveSocket(nativeHandle); 8891878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return ErrorCodes.SUCCESS; 890f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 8910e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_IO; 892f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 893f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 894f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 895fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 8960e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int connect(int nativeHandle, int sap) throws RemoteException { 897d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 898bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 899f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeLlcpSocket socket = null; 9000e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean isSuccess = false; 901f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 902f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 903f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 904f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 905f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 906f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 907f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the socket in the hmap */ 908f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 909f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (socket != null) { 91093915e1c6fe6d5d16fcebeda610782bf5812c9b4Nick Pelly isSuccess = socket.doConnect(sap); 9110e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (isSuccess) { 9120e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.SUCCESS; 913f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 914f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_IO; 915f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 916f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 917f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_IO; 918f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 9190e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 920f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 921f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 922fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 9230e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int connectByName(int nativeHandle, String sn) throws RemoteException { 924d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 925bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 926f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeLlcpSocket socket = null; 927f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly boolean isSuccess = false; 928f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 929f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 930f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 931f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 932f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 933f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 934f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the socket in the hmap */ 935f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 936f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (socket != null) { 93793915e1c6fe6d5d16fcebeda610782bf5812c9b4Nick Pelly isSuccess = socket.doConnectBy(sn); 938f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (isSuccess) { 939f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.SUCCESS; 940f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 941f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_IO; 942f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 943f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 944f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_IO; 945f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 9460e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 947f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 948f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 949fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 9500e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int getLocalSap(int nativeHandle) throws RemoteException { 951d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 952bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 9530e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeLlcpSocket socket = null; 954f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 955f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 956f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 957f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 958f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 959f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 9600e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the socket in the hmap */ 961f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 9620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (socket != null) { 9630e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return socket.getSap(); 964f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 9650e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return 0; 966f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 967f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 968f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 969fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 9700e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int getLocalSocketMiu(int nativeHandle) throws RemoteException { 971d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 972bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 9730e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeLlcpSocket socket = null; 974f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 975f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 976f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 9770e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 978f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 979f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 980f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the socket in the hmap */ 981f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 982f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (socket != null) { 9830e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return socket.getMiu(); 9840e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 9850e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return 0; 986f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 987f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 988f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 989fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 9900e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int getLocalSocketRw(int nativeHandle) throws RemoteException { 991d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 992bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 9930e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeLlcpSocket socket = null; 994f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 995f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 996f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 997f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 998f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 999f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1000f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the socket in the hmap */ 1001f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 10020e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (socket != null) { 10030e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return socket.getRw(); 10040e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 10050e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return 0; 10060e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10070e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10080e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 1009fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 10100e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int getRemoteSocketMiu(int nativeHandle) throws RemoteException { 1011d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 10120e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10130e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeLlcpSocket socket = null; 10140e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10150e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 10160e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 10170e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 10180e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10190e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10200e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the socket in the hmap */ 1021f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 10220e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (socket != null) { 10230e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (socket.doGetRemoteSocketMiu() != 0) { 10240e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return socket.doGetRemoteSocketMiu(); 10250e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 10260e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_SOCKET_NOT_CONNECTED; 10270e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10280e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 10290e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_SOCKET_NOT_CONNECTED; 10300e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10310e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10320e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 1033fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 10340e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int getRemoteSocketRw(int nativeHandle) throws RemoteException { 1035d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 10360e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10370e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeLlcpSocket socket = null; 10380e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10390e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 10400e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 10410e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 10420e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10430e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10440e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the socket in the hmap */ 1045f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 10460e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (socket != null) { 10470e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (socket.doGetRemoteSocketRw() != 0) { 10480e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return socket.doGetRemoteSocketRw(); 10490e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 10500e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_SOCKET_NOT_CONNECTED; 10510e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10520e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 10530e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_SOCKET_NOT_CONNECTED; 10540e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10550e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10560e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 1057fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 10580e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int receive(int nativeHandle, byte[] receiveBuffer) throws RemoteException { 1059d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 10600e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10610e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeLlcpSocket socket = null; 10620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10630e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 10640e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 10650e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 10660e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10670e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10680e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the socket in the hmap */ 1069f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 10700e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (socket != null) { 107128f72bd9880d0b00c486603489d3862733eb638fSylvain Fonteneau return socket.doReceive(receiveBuffer); 10720e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 107328f72bd9880d0b00c486603489d3862733eb638fSylvain Fonteneau return 0; 10740e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10750e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10760e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 1077fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 10780e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int send(int nativeHandle, byte[] data) throws RemoteException { 1079d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 10800e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10810e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeLlcpSocket socket = null; 10820e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean isSuccess = false; 10830e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10840e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 10850e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 10860e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 10870e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10880e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 10890e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the socket in the hmap */ 1090f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 10910e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (socket != null) { 10920e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly isSuccess = socket.doSend(data); 10930e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (isSuccess) { 10940e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.SUCCESS; 10950e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 10960e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_IO; 10970e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 10980e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 10990e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_IO; 11000e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 11010e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 11020e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly }; 11030e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 11040e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private final ILlcpServiceSocket mLlcpServerSocketService = new ILlcpServiceSocket.Stub() { 11050e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 1106f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau private NativeLlcpServiceSocket findSocket(int nativeHandle) { 1107bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly Object socket = NfcService.this.findSocket(nativeHandle); 1108bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly if (!(socket instanceof NativeLlcpServiceSocket)) { 1109f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau return null; 1110f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau } 1111bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly return (NativeLlcpServiceSocket) socket; 1112f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau } 1113f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau 1114fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 11150e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int accept(int nativeHandle) throws RemoteException { 1116d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 11170e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 11180e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeLlcpServiceSocket socket = null; 11190e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeLlcpSocket clientSocket = null; 11200e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 11210e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 11220e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 11230e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 11240e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 11250e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 11260e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the socket in the hmap */ 1127f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 11280e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (socket != null) { 112993915e1c6fe6d5d16fcebeda610782bf5812c9b4Nick Pelly clientSocket = socket.doAccept(socket.getMiu(), 11300e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly socket.getRw(), socket.getLinearBufferLength()); 11310e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (clientSocket != null) { 11320e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Add the socket into the socket map */ 11332f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly synchronized(this) { 11341878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau mGeneratedSocketHandle++; 11351878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau mSocketMap.put(mGeneratedSocketHandle, clientSocket); 11361878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau return mGeneratedSocketHandle; 11372f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly } 11380e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 11390e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_IO; 11400e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 11410e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } else { 11420e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_IO; 11430e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 11440e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 11450e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 1146fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 11470e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public void close(int nativeHandle) throws RemoteException { 1148d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 11490e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 11500e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeLlcpServiceSocket socket = null; 11510e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 11520e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 11530e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 11540e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return; 11550e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 11560e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 11570e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the socket in the hmap */ 1158f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 11590e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (socket != null) { 11601878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau socket.doClose(); 1161f63da5f0887eba86d877faaee41dc5dfb8fccd50Jeff Hamilton synchronized (this) { 11620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Remove the socket closed from the hmap */ 11630e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly RemoveSocket(nativeHandle); 11640e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 11650e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 11660e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1167f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly }; 1168f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1169f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private final ILlcpConnectionlessSocket mLlcpConnectionlessSocketService = new ILlcpConnectionlessSocket.Stub() { 1170f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1171f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau private NativeLlcpConnectionlessSocket findSocket(int nativeHandle) { 1172bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly Object socket = NfcService.this.findSocket(nativeHandle); 1173bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly if (!(socket instanceof NativeLlcpConnectionlessSocket)) { 1174f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau return null; 1175f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau } 1176bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly return (NativeLlcpConnectionlessSocket) socket; 1177f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau } 1178f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau 1179fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1180f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly public void close(int nativeHandle) throws RemoteException { 1181d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1182bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1183f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeLlcpConnectionlessSocket socket = null; 1184f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1185f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1186f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1187f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return; 1188f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1189f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1190f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the socket in the hmap */ 1191f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 1192f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (socket != null) { 11931878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau socket.doClose(); 11941878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau /* Remove the socket closed from the hmap */ 11951878c7b6e2cef1e52138ee3a5588913cc165faa5Sylvain Fonteneau RemoveSocket(nativeHandle); 1196f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1197f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1198f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1199fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1200f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly public int getSap(int nativeHandle) throws RemoteException { 1201d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1202bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1203f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeLlcpConnectionlessSocket socket = null; 1204f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1205f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1206f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1207f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 1208f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1209f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1210f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the socket in the hmap */ 1211f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 1212f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (socket != null) { 1213f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return socket.getSap(); 1214f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 1215f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return 0; 1216f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1217f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1218f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1219fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1220f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly public LlcpPacket receiveFrom(int nativeHandle) throws RemoteException { 1221d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1222bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1223f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeLlcpConnectionlessSocket socket = null; 1224f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly LlcpPacket packet; 1225f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1226f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1227f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1228f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1229f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1230f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1231f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the socket in the hmap */ 1232f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 1233f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (socket != null) { 1234f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly packet = socket.doReceiveFrom(socket.getLinkMiu()); 1235f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (packet != null) { 1236f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return packet; 1237f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1238f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1239f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 1240f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1241f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1242f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1243f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1244fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1245f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly public int sendTo(int nativeHandle, LlcpPacket packet) throws RemoteException { 1246d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1247bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1248f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeLlcpConnectionlessSocket socket = null; 1249f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly boolean isSuccess = false; 1250f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1251f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1252f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1253f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 1254f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1255f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1256f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the socket in the hmap */ 1257f5d6f3253dca74631ba78c715f224cfa80e61d29Sylvain Fonteneau socket = findSocket(nativeHandle); 1258f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (socket != null) { 1259f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly isSuccess = socket.doSendTo(packet.getRemoteSap(), packet.getDataBuffer()); 1260f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (isSuccess) { 1261f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.SUCCESS; 1262f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 1263f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_IO; 1264f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1265f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 1266f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_IO; 1267f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1268f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1269f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly }; 1270f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1271f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private final INfcTag mNfcTagService = new INfcTag.Stub() { 1272f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1273fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1274f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly public int close(int nativeHandle) throws RemoteException { 1275d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1276bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1277f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeNfcTag tag = null; 1278f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1279f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1280f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1281f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 1282f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1283f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1284f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the tag in the hmap */ 1285f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly tag = (NativeNfcTag) findObject(nativeHandle); 1286f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (tag != null) { 1287b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau /* Remove the device from the hmap */ 1288b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau unregisterObject(nativeHandle); 128921545af22f9b913ec9cb124287aab2fcb0cf2b3bNick Pelly tag.disconnect(); 1290b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau return ErrorCodes.SUCCESS; 1291f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1292f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* Restart polling loop for notification */ 129349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly applyRouting(); 1294f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_DISCONNECT; 1295f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1296f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1297fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1298ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen public int connect(int nativeHandle, int technology) throws RemoteException { 1299d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1300bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1301f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeNfcTag tag = null; 1302f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1303f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1304f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1305f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 1306f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1307f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1308f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the tag in the hmap */ 1309f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly tag = (NativeNfcTag) findObject(nativeHandle); 1310b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau if (tag == null) { 1311b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau return ErrorCodes.ERROR_DISCONNECT; 1312f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1313ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen 1314ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen // Note that on most tags, all technologies are behind a single 1315ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen // handle. This means that the connect at the lower levels 1316ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen // will do nothing, as the tag is already connected to that handle. 1317ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen if (tag.connect(technology)) { 1318ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen return ErrorCodes.SUCCESS; 1319ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen } else { 1320ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen return ErrorCodes.ERROR_DISCONNECT; 1321ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen } 1322f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1323f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1324fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1325aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen public int reconnect(int nativeHandle) throws RemoteException { 1326aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1327aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen 1328aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen NativeNfcTag tag = null; 1329aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen 1330aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen // Check if NFC is enabled 1331aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen if (!mIsNfcEnabled) { 1332aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen return ErrorCodes.ERROR_NOT_INITIALIZED; 1333aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen } 1334aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen 1335aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen /* find the tag in the hmap */ 1336aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen tag = (NativeNfcTag) findObject(nativeHandle); 1337aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen if (tag != null) { 1338aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen if (tag.reconnect()) { 1339aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen return ErrorCodes.SUCCESS; 1340aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen } else { 1341aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen return ErrorCodes.ERROR_DISCONNECT; 1342aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen } 1343aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen } 1344aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen return ErrorCodes.ERROR_DISCONNECT; 1345aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen } 1346aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen 1347aae427142dc22e7e419c146bc7748d9daff518e8Martijn Coenen @Override 1348b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton public int[] getTechList(int nativeHandle) throws RemoteException { 1349d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1350bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1351f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1352f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1353f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1354f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1355f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1356f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the tag in the hmap */ 1357b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton NativeNfcTag tag = (NativeNfcTag) findObject(nativeHandle); 1358f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (tag != null) { 1359b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton return tag.getTechList(); 1360f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1361f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1362f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1363f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1364fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1365f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly public byte[] getUid(int nativeHandle) throws RemoteException { 1366f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeNfcTag tag = null; 1367f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly byte[] uid; 1368f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1369f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1370f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1371f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1372f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1373f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1374f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the tag in the hmap */ 1375f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly tag = (NativeNfcTag) findObject(nativeHandle); 1376f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (tag != null) { 1377f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly uid = tag.getUid(); 1378f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return uid; 1379f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1380f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1381f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1382f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1383fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1384b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau public boolean isPresent(int nativeHandle) throws RemoteException { 1385b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau NativeNfcTag tag = null; 1386b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau 1387b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau // Check if NFC is enabled 1388b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau if (!mIsNfcEnabled) { 1389b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau return false; 1390b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau } 1391b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau 1392b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau /* find the tag in the hmap */ 1393b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau tag = (NativeNfcTag) findObject(nativeHandle); 1394b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau if (tag == null) { 1395b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau return false; 1396b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau } 1397b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau 1398ab2b44b97936d2c5dbf6eda1245ca793e840713fMartijn Coenen return tag.isPresent(); 1399b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau } 1400b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau 1401fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1402f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly public boolean isNdef(int nativeHandle) throws RemoteException { 1403f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeNfcTag tag = null; 1404f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly boolean isSuccess = false; 1405f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1406f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1407f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1408f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return isSuccess; 1409f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1410f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1411f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the tag in the hmap */ 1412f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly tag = (NativeNfcTag) findObject(nativeHandle); 14133ba3b10867c36bff57b72ff99c7b56d63d418f3fMartijn Coenen int[] ndefInfo = new int[2]; 1414f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (tag != null) { 14153ba3b10867c36bff57b72ff99c7b56d63d418f3fMartijn Coenen isSuccess = tag.checkNdef(ndefInfo); 1416f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1417f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return isSuccess; 1418f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1419f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1420fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 14219d5511f2640903a79d24578a12a93e50a96f0c0eMartijn Coenen public TransceiveResult transceive(int nativeHandle, byte[] data, boolean raw) 142297c6942c7c7f9df3bb8dbcc01cf7bb6e2e090005Martijn Coenen throws RemoteException { 1423d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1424bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1425f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeNfcTag tag = null; 1426f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly byte[] response; 1427f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1428f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1429f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1430f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1431f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1432f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1433f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the tag in the hmap */ 1434f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly tag = (NativeNfcTag) findObject(nativeHandle); 1435f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (tag != null) { 14369d5511f2640903a79d24578a12a93e50a96f0c0eMartijn Coenen int[] targetLost = new int[1]; 14379d5511f2640903a79d24578a12a93e50a96f0c0eMartijn Coenen response = tag.transceive(data, raw, targetLost); 14389d5511f2640903a79d24578a12a93e50a96f0c0eMartijn Coenen TransceiveResult transResult = new TransceiveResult( 14399d5511f2640903a79d24578a12a93e50a96f0c0eMartijn Coenen (response != null) ? true : false, 14409d5511f2640903a79d24578a12a93e50a96f0c0eMartijn Coenen (targetLost[0] == 1) ? true : false, 14419d5511f2640903a79d24578a12a93e50a96f0c0eMartijn Coenen response); 14429d5511f2640903a79d24578a12a93e50a96f0c0eMartijn Coenen return transResult; 1443f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1444f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1445f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1446f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1447fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 14483fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton public NdefMessage ndefRead(int nativeHandle) throws RemoteException { 1449d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1450bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1451f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeNfcTag tag; 1452f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1453f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1454f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1455f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1456f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1457f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1458f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the tag in the hmap */ 1459f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly tag = (NativeNfcTag) findObject(nativeHandle); 1460f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (tag != null) { 1461b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau byte[] buf = tag.read(); 1462f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (buf == null) 1463f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1464f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1465f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* Create an NdefMessage */ 1466f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly try { 1467f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return new NdefMessage(buf); 1468f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } catch (FormatException e) { 1469f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1470f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1471f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1472f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return null; 1473f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1474f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1475fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 14763fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton public int ndefWrite(int nativeHandle, NdefMessage msg) throws RemoteException { 1477d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1478bebaa6cc1a1eb2ce0656e17b0e09ed4747878d8eNick Pelly 1479f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeNfcTag tag; 1480f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1481f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Check if NFC is enabled 1482f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (!mIsNfcEnabled) { 1483f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 1484f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1485f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1486f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* find the tag in the hmap */ 1487f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly tag = (NativeNfcTag) findObject(nativeHandle); 1488f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (tag == null) { 1489f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_IO; 1490f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1491f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1492b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau if (tag.write(msg.toByteArray())) { 1493f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.SUCCESS; 1494f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1495f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly else { 1496f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_IO; 1497f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1498f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1499f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1500f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1501fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 1502f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly public int getLastError(int nativeHandle) throws RemoteException { 1503bb78d3749bd0cc3801fdd9d5df06b0f6ee7a400fMartijn Coenen return(mManager.doGetLastError()); 1504f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1505f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1506fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 15073fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton public boolean ndefIsWritable(int nativeHandle) throws RemoteException { 15083fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton throw new UnsupportedOperationException(); 1509f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1510f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1511fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 15123fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton public int ndefMakeReadOnly(int nativeHandle) throws RemoteException { 151303ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 151403ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen 151503ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen NativeNfcTag tag; 151603ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen 151703ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen // Check if NFC is enabled 151803ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen if (!mIsNfcEnabled) { 151903ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen return ErrorCodes.ERROR_NOT_INITIALIZED; 152003ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen } 152103ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen 152203ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen /* find the tag in the hmap */ 152303ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen tag = (NativeNfcTag) findObject(nativeHandle); 152403ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen if (tag == null) { 152503ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen return ErrorCodes.ERROR_IO; 152603ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen } 152703ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen 152803ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen if (tag.makeReadonly()) { 152903ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen return ErrorCodes.SUCCESS; 153003ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen } 153103ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen else { 153203ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen return ErrorCodes.ERROR_IO; 153303ee488afaa982ce934c5de399db9f9fa88c7d1eMartijn Coenen } 1534f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1535f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 15360aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen @Override 15370aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen public int formatNdef(int nativeHandle, byte[] key) throws RemoteException { 15380aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 15390aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen 15400aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen NativeNfcTag tag; 15410aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen 15420aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen // Check if NFC is enabled 15430aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen if (!mIsNfcEnabled) { 15440aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen return ErrorCodes.ERROR_NOT_INITIALIZED; 15450aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen } 15460aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen 15470aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen /* find the tag in the hmap */ 15480aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen tag = (NativeNfcTag) findObject(nativeHandle); 15490aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen if (tag == null) { 15500aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen return ErrorCodes.ERROR_IO; 15510aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen } 15520aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen 15530aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen if (tag.formatNdef(key)) { 15540aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen return ErrorCodes.SUCCESS; 15550aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen } 15560aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen else { 15570aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen return ErrorCodes.ERROR_IO; 15580aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen } 15590aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen } 15600aac9419555e69512a886a6b7fa6ce2d1947c72fMartijn Coenen 15611b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen @Override 15621b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen public void setIsoDepTimeout(int timeout) throws RemoteException { 15631b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 15641b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen 15651b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen mManager.setIsoDepTimeout(timeout); 15661b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen } 1567f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 15681b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen @Override 15691b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen public void resetIsoDepTimeout() throws RemoteException { 15701b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 15711b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen 15721b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen mManager.resetIsoDepTimeout(); 15731b61f1dee91101e249b9be65d95366fa745b3b78Martijn Coenen } 15740e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly }; 1575f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 15760e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private final IP2pInitiator mP2pInitiatorService = new IP2pInitiator.Stub() { 1577f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1578fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 15790e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public byte[] getGeneralBytes(int nativeHandle) throws RemoteException { 1580d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1581f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 15820e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeP2pDevice device; 1583f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 15840e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 15850e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 15860e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 15870e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1588f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 15890e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the device in the hmap */ 15900e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly device = (NativeP2pDevice) findObject(nativeHandle); 15910e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (device != null) { 15920e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly byte[] buff = device.getGeneralBytes(); 15930e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (buff == null) 15940e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 15950e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return buff; 15960e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 15970e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 15980e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1599f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1600fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 16010e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int getMode(int nativeHandle) throws RemoteException { 1602d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1603f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16040e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeP2pDevice device; 1605f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16060e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 16070e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 16080e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 16090e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1610f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16110e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the device in the hmap */ 16120e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly device = (NativeP2pDevice) findObject(nativeHandle); 16130e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (device != null) { 16140e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return device.getMode(); 16150e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1616f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return ErrorCodes.ERROR_INVALID_PARAM; 1617f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1618f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1619fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 16200e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public byte[] receive(int nativeHandle) throws RemoteException { 1621d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1622f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16230e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeP2pDevice device; 1624f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16250e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 16260e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 16270e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 16280e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1629f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16300e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the device in the hmap */ 16310e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly device = (NativeP2pDevice) findObject(nativeHandle); 16320e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (device != null) { 16330e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly byte[] buff = device.doReceive(); 16340e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (buff == null) 16350e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 16360e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return buff; 16370e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 16380e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Restart polling loop for notification */ 163949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly applyRouting(); 16400e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 16410e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1642f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1643fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 16440e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public boolean send(int nativeHandle, byte[] data) throws RemoteException { 1645d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1646f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16470e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeP2pDevice device; 16480e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean isSuccess = false; 1649f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16500e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 16510e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 16520e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return isSuccess; 16530e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1654f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16550e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the device in the hmap */ 16560e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly device = (NativeP2pDevice) findObject(nativeHandle); 16570e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (device != null) { 16580e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly isSuccess = device.doSend(data); 16590e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 16600e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return isSuccess; 16610e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 16620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly }; 1663f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16640e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private final IP2pTarget mP2pTargetService = new IP2pTarget.Stub() { 1665f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1666fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 16670e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int connect(int nativeHandle) throws RemoteException { 1668d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1669f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16700e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeP2pDevice device; 1671f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16720e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 16730e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 16740e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 16750e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1676f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16770e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the device in the hmap */ 16780e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly device = (NativeP2pDevice) findObject(nativeHandle); 16790e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (device != null) { 16800e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (device.doConnect()) { 16810e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.SUCCESS; 16820e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 16830e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 16840e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_CONNECT; 16850e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1686f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1687fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 16880e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public boolean disconnect(int nativeHandle) throws RemoteException { 1689d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1690f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16910e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeP2pDevice device; 16920e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean isSuccess = false; 1693f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16940e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 16950e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 16960e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return isSuccess; 16970e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1698f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 16990e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the device in the hmap */ 17000e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly device = (NativeP2pDevice) findObject(nativeHandle); 17010e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (device != null) { 17020e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (isSuccess = device.doDisconnect()) { 17030e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* remove the device from the hmap */ 1704b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau unregisterObject(nativeHandle); 17050e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Restart polling loop for notification */ 170649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly applyRouting(); 17070e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1708f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 17090e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return isSuccess; 1710f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 17110e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1712f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1713fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 17140e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public byte[] getGeneralBytes(int nativeHandle) throws RemoteException { 1715d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1716f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 17170e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeP2pDevice device; 17180e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 17190e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 17200e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 17210e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 1722f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1723f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 17240e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the device in the hmap */ 17250e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly device = (NativeP2pDevice) findObject(nativeHandle); 17260e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (device != null) { 17270e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly byte[] buff = device.getGeneralBytes(); 17280e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (buff == null) 17290e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 17300e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return buff; 17310e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 17320e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 17330e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1734f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1735fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 17360e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public int getMode(int nativeHandle) throws RemoteException { 1737d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1738f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 17390e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeP2pDevice device; 1740f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 17410e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 17420e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 17430e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_NOT_INITIALIZED; 1744f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1745f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 17460e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the device in the hmap */ 17470e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly device = (NativeP2pDevice) findObject(nativeHandle); 17480e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (device != null) { 17490e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return device.getMode(); 17500e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 17510e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return ErrorCodes.ERROR_INVALID_PARAM; 17520e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 1753f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 1754fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 175597c6942c7c7f9df3bb8dbcc01cf7bb6e2e090005Martijn Coenen public byte[] transceive(int nativeHandle, byte[] data) 175697c6942c7c7f9df3bb8dbcc01cf7bb6e2e090005Martijn Coenen throws RemoteException { 1757d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); 1758f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 17590e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NativeP2pDevice device; 1760f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 17610e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly // Check if NFC is enabled 17620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (!mIsNfcEnabled) { 17630e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 1764f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1765f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 17660e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* find the device in the hmap */ 17670e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly device = (NativeP2pDevice) findObject(nativeHandle); 17680e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (device != null) { 17690e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly byte[] buff = device.doTransceive(data); 17700e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (buff == null) 17710e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 17720e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return buff; 17730e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 17740e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return null; 17750e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 17760e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly }; 1777f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 177849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private INfcAdapterExtras mExtrasService = new INfcAdapterExtras.Stub() { 177949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private Bundle writeNoException() { 178049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Bundle p = new Bundle(); 178149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly p.putInt("e", 0); 178249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly return p; 178349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 178449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private Bundle writeIoException(IOException e) { 178549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Bundle p = new Bundle(); 178649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly p.putInt("e", -1); 178749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly p.putString("m", e.getMessage()); 178849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly return p; 178949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 17900bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 1791bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks @Override 179249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public Bundle open(IBinder b) throws RemoteException { 179393d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton NfcService.enforceNfceeAdminPerm(mContext); 1794bd555ee64250126b60b24814120a2049943920caNick Pelly 179549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Bundle result; 179649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly try { 179749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly _open(b); 179849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly result = writeNoException(); 179949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } catch (IOException e) { 180049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly result = writeIoException(e); 18010bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 180249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly return result; 180349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 18040bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 180549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private void _open(IBinder b) throws IOException, RemoteException { 180649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly synchronized(NfcService.this) { 180749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (!mIsNfcEnabled) { 180849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly throw new IOException("NFC adapter is disabled"); 180949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 181049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (mOpenEe != null) { 181149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly throw new IOException("NFC EE already open"); 181249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 18130bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 181449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly int handle = mSecureElement.doOpenSecureElementConnection(); 181549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (handle == 0) { 181649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly throw new IOException("NFC EE failed to open"); 181749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 1818ba6401757f8017faeb77423f2d08fd51be1d1051Nick Pelly mManager.doSetIsoDepTimeout(10000); 1819ba6401757f8017faeb77423f2d08fd51be1d1051Nick Pelly 182049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mOpenEe = new OpenSecureElement(getCallingPid(), handle); 182149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly try { 182249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly b.linkToDeath(mOpenEe, 0); 182349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } catch (RemoteException e) { 182449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mOpenEe.binderDied(); 182549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 182649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 18270bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 18280bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 1829bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks @Override 183049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public Bundle close() throws RemoteException { 183193d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton NfcService.enforceNfceeAdminPerm(mContext); 18320bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 183349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Bundle result; 183449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly try { 183549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly _close(); 183649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly result = writeNoException(); 183749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } catch (IOException e) { 183849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly result = writeIoException(e); 18390bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 184049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly return result; 184149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 18420bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 184349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly void _close() throws IOException, RemoteException { 184449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly // Blocks until a pending open() or transceive() times out. 184549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly //TODO: This is incorrect behavior - the close should interrupt pending 184649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly // operations. However this is not supported by current hardware. 18470bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 184849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly synchronized(NfcService.this) { 184949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (!mIsNfcEnabled) { 185049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly throw new IOException("NFC adapter is disabled"); 1851221b4d6ee301fbfe19402798f7d3c11e6878c888daniel_tomas } 185249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (mOpenEe == null) { 185349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly throw new IOException("NFC EE closed"); 185449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 185549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (mOpenEe.pid != -1 && getCallingPid() != mOpenEe.pid) { 185649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly throw new SecurityException("Wrong PID"); 1857221b4d6ee301fbfe19402798f7d3c11e6878c888daniel_tomas } 1858221b4d6ee301fbfe19402798f7d3c11e6878c888daniel_tomas 1859ba6401757f8017faeb77423f2d08fd51be1d1051Nick Pelly mManager.doResetIsoDepTimeout(); 186049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mSecureElement.doDisconnect(mOpenEe.handle); 186149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mOpenEe = null; 186249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly 186349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly applyRouting(); 18640bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 18650bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 18660bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 1867bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks @Override 186849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public Bundle transceive(byte[] in) throws RemoteException { 186993d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton NfcService.enforceNfceeAdminPerm(mContext); 1870bd555ee64250126b60b24814120a2049943920caNick Pelly 187149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Bundle result; 187249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly byte[] out; 187349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly try { 187449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly out = _transceive(in); 187549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly result = writeNoException(); 187649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly result.putByteArray("out", out); 187749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } catch (IOException e) { 187849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly result = writeIoException(e); 18790bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 188049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly return result; 188149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 18820bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 188349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private byte[] _transceive(byte[] data) throws IOException, RemoteException { 188449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly synchronized(NfcService.this) { 188549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (!mIsNfcEnabled) { 188649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly throw new IOException("NFC is not enabled"); 188749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 188849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (mOpenEe == null){ 188949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly throw new IOException("NFC EE is not open"); 189049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 189149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (getCallingPid() != mOpenEe.pid) { 189249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly throw new SecurityException("Wrong PID"); 189349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 18940bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 18950bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 189649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly return mSecureElement.doTransceive(mOpenEe.handle, data); 18970bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 18980bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 1899bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks @Override 190049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public int getCardEmulationRoute() throws RemoteException { 190193d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton NfcService.enforceNfceeAdminPerm(mContext); 190249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly return mEeRoutingState; 19030bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 19040bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 1905bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks @Override 190649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public void setCardEmulationRoute(int route) throws RemoteException { 190793d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton NfcService.enforceNfceeAdminPerm(mContext); 190849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mEeRoutingState = route; 190949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly applyRouting(); 19100bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 1911bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 1912bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks @Override 1913bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks public void registerTearDownApdus(String packageName, ApduList apdu) throws RemoteException { 1914bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks NfcService.enforceNfceeAdminPerm(mContext); 1915bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks synchronized(NfcService.this) { 1916bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks mTearDownApdus.put(packageName, apdu); 1917bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks writeTearDownApdusLocked(); 1918bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 1919bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 1920bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 1921bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks @Override 1922bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks public void unregisterTearDownApdus(String packageName) throws RemoteException { 1923bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks NfcService.enforceNfceeAdminPerm(mContext); 1924bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks synchronized(NfcService.this) { 1925bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks mTearDownApdus.remove(packageName); 1926bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks writeTearDownApdusLocked(); 1927bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 1928bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 19290bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas }; 19300bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 193149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly /** resources kept while secure element is open */ 193249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private class OpenSecureElement implements IBinder.DeathRecipient { 193349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public int pid; // pid that opened SE 193449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public int handle; // low-level handle 193549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public OpenSecureElement(int pid, int handle) { 193649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly this.pid = pid; 193749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly this.handle = handle; 193849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } 1939bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks @Override 194049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly public void binderDied() { 194149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly synchronized (NfcService.this) { 194249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (DBG) Log.d(TAG, "Tracked app " + pid + " died"); 194349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly pid = -1; 19440bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas try { 194549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mExtrasService.close(); 194649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } catch (RemoteException e) { /* local call never fails */ } 19470bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 19480bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 19490bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas } 19500bd11735e8a28db1692f28abcc3e065abae0e8ddDaniel Tomas 19510e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private boolean _enable(boolean oldEnabledState) { 19526efda9f3970382554437be037aed863be9889499Sylvain Fonteneau applyProperties(); 19536efda9f3970382554437be037aed863be9889499Sylvain Fonteneau 19540e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly boolean isSuccess = mManager.initialize(); 19550e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly if (isSuccess) { 195665945ad77cadb7a3bdf171497877d2325b23def5Nick Pelly mIsNfcEnabled = true; 195749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mIsDiscoveryOn = true; 195865945ad77cadb7a3bdf171497877d2325b23def5Nick Pelly 19590e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Start polling loop */ 196049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly applyRouting(); 196157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 196257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton /* bring up the my tag server */ 1963ca1a86ecb8edce740a232c3439355e8d5b706e7aJeff Hamilton mNdefPushServer.start(); 196457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 1965f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } else { 19660e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mIsNfcEnabled = false; 1967f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1968f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 19690e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly updateNfcOnSetting(oldEnabledState); 1970f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 19710e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly return isSuccess; 1972f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 1973f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 197449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly /** apply NFC discovery and EE routing */ 197549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly private synchronized void applyRouting() { 197649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (mIsNfcEnabled && mOpenEe == null) { 197749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (mScreenOn) { 197849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (mEeRoutingState == ROUTE_ON_WHEN_SCREEN_ON) { 197949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Log.d(TAG, "NFC-EE routing ON"); 198049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mManager.doSelectSecureElement(SECURE_ELEMENT_ID); 198149d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } else { 198249d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Log.d(TAG, "NFC-EE routing OFF"); 198349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mManager.doDeselectSecureElement(SECURE_ELEMENT_ID); 1984221b4d6ee301fbfe19402798f7d3c11e6878c888daniel_tomas } 198549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (mIsDiscoveryOn) { 198649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Log.d(TAG, "NFC-C discovery ON"); 198749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mManager.enableDiscovery(DISCOVERY_MODE_READER); 198849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly } else { 198949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Log.d(TAG, "NFC-C discovery OFF"); 199049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mManager.disableDiscovery(); 1991221b4d6ee301fbfe19402798f7d3c11e6878c888daniel_tomas } 1992221b4d6ee301fbfe19402798f7d3c11e6878c888daniel_tomas } else { 199349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Log.d(TAG, "NFC-EE routing OFF"); 199449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mManager.doDeselectSecureElement(SECURE_ELEMENT_ID); 199549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Log.d(TAG, "NFC-C discovery OFF"); 199649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly mManager.disableDiscovery(); 1997221b4d6ee301fbfe19402798f7d3c11e6878c888daniel_tomas } 199865945ad77cadb7a3bdf171497877d2325b23def5Nick Pelly } 199965945ad77cadb7a3bdf171497877d2325b23def5Nick Pelly } 200065945ad77cadb7a3bdf171497877d2325b23def5Nick Pelly 20012436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir /** Disconnect any target if present */ 20022436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir private synchronized void maybeDisconnectTarget() { 20032436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir if (mIsNfcEnabled) { 20042436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir Iterator<?> iterator = mObjectMap.values().iterator(); 20052436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir while(iterator.hasNext()) { 20062436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir Object object = iterator.next(); 2007663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau if(object instanceof NativeNfcTag) { 2008663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau // Disconnect from tags 20092436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir NativeNfcTag tag = (NativeNfcTag) object; 20102436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir tag.disconnect(); 20112436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir } 2012663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau else if(object instanceof NativeP2pDevice) { 2013663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau // Disconnect from P2P devices 2014663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau NativeP2pDevice device = (NativeP2pDevice) object; 2015663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau if (device.getMode() == NativeP2pDevice.MODE_P2P_TARGET) { 2016663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau // Remote peer is target, request disconnection 2017663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau device.doDisconnect(); 2018663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau } 2019663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau else { 2020663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau // Remote peer is initiator, we cannot disconnect 2021663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau // Just wait for field removal 2022663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau } 2023663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau } 2024663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau iterator.remove(); 20252436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir } 20262436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir } 20272436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir } 20282436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir 2029bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks private void readTearDownApdus() { 2030bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks FileInputStream input = null; 2031bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2032bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks try { 2033bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks input = openFileInput(TEAR_DOWN_SCRIPTS_FILE_NAME); 2034bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks DataInputStream stream = new DataInputStream(input); 2035bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2036bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks int packagesSize = stream.readInt(); 2037bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2038bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks for (int i = 0 ; i < packagesSize ; i++) { 2039bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks String packageName = stream.readUTF(); 2040bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks ApduList apdu = new ApduList(); 2041bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2042bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks int commandsSize = stream.readInt(); 2043bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2044bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks for (int j = 0 ; j < commandsSize ; j++) { 2045bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks int length = stream.readInt(); 2046bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2047bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks byte[] cmd = new byte[length]; 2048bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2049bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks stream.read(cmd); 2050bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks apdu.add(cmd); 2051bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2052bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2053bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks mTearDownApdus.put(packageName, apdu); 2054bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2055bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } catch (FileNotFoundException e) { 2056bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks // Ignore. 2057bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } catch (IOException e) { 2058bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks Log.e(TAG, "Could not read tear down scripts file: ", e); 2059bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks deleteFile(TEAR_DOWN_SCRIPTS_FILE_NAME); 2060bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } finally { 2061bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks try { 2062bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks if (input != null) { 2063bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks input.close(); 2064bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2065bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } catch (IOException e) { 2066bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks // Ignore 2067bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2068bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2069bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2070bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2071bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks private void writeTearDownApdusLocked() { 2072bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks FileOutputStream output = null; 2073bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks DataOutputStream stream = null; 2074bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2075bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks try { 2076bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks output = openFileOutput(TEAR_DOWN_SCRIPTS_FILE_NAME, Context.MODE_PRIVATE); 2077bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks stream = new DataOutputStream(output); 2078bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2079bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks stream.writeInt(mTearDownApdus.size()); 2080bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2081bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks for (String packageName : mTearDownApdus.keySet()) { 2082bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks stream.writeUTF(packageName); 2083bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2084bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks List<byte[]> commands = mTearDownApdus.get(packageName).get(); 2085bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks stream.writeInt(commands.size()); 2086bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2087bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks for (byte[] cmd : commands) { 2088bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks stream.writeInt(cmd.length); 2089bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks stream.write(cmd, 0, cmd.length); 2090bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2091bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2092bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2093bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } catch (IOException e) { 2094bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } finally { 2095bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks try { 2096bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks if (output != null) { 2097bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks stream.flush(); 2098bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks stream.close(); 2099bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2100bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } catch (IOException e) { 2101bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks // Ignore 2102bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2103bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2104bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2105bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2106f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly private void applyProperties() { 2107f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mManager.doSetProperties(PROPERTY_LLCP_LTO, mPrefs.getInt(PREF_LLCP_LTO, LLCP_LTO_DEFAULT)); 2108f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mManager.doSetProperties(PROPERTY_LLCP_MIU, mPrefs.getInt(PREF_LLCP_MIU, LLCP_MIU_DEFAULT)); 2109f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mManager.doSetProperties(PROPERTY_LLCP_WKS, mPrefs.getInt(PREF_LLCP_WKS, LLCP_WKS_DEFAULT)); 2110f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mManager.doSetProperties(PROPERTY_LLCP_OPT, mPrefs.getInt(PREF_LLCP_OPT, LLCP_OPT_DEFAULT)); 2111f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mManager.doSetProperties(PROPERTY_NFC_DISCOVERY_A, 2112f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mPrefs.getBoolean(PREF_DISCOVERY_A, DISCOVERY_A_DEFAULT) ? 1 : 0); 2113f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mManager.doSetProperties(PROPERTY_NFC_DISCOVERY_B, 2114f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mPrefs.getBoolean(PREF_DISCOVERY_B, DISCOVERY_B_DEFAULT) ? 1 : 0); 2115f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mManager.doSetProperties(PROPERTY_NFC_DISCOVERY_F, 2116f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mPrefs.getBoolean(PREF_DISCOVERY_F, DISCOVERY_F_DEFAULT) ? 1 : 0); 2117f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mManager.doSetProperties(PROPERTY_NFC_DISCOVERY_15693, 2118f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mPrefs.getBoolean(PREF_DISCOVERY_15693, DISCOVERY_15693_DEFAULT) ? 1 : 0); 2119f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mManager.doSetProperties(PROPERTY_NFC_DISCOVERY_NFCIP, 2120f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly mPrefs.getBoolean(PREF_DISCOVERY_NFCIP, DISCOVERY_NFCIP_DEFAULT) ? 1 : 0); 2121f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly } 2122f5280191c0e422a730504eebeced4acbcfc1bd9fNick Pelly 21230e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private void updateNfcOnSetting(boolean oldEnabledState) { 21240e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mPrefsEditor.putBoolean(PREF_NFC_ON, mIsNfcEnabled); 21252f386c612159a48933c12dbcaf6735bd9ad1c660Brad Fitzpatrick mPrefsEditor.apply(); 21260e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 21272f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly synchronized(this) { 21284acf84356e35861f7df6d07a2b9ff84842a9f221Nick Pelly if (oldEnabledState != mIsNfcEnabled) { 21292f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly Intent intent = new Intent(NfcAdapter.ACTION_ADAPTER_STATE_CHANGE); 21304acf84356e35861f7df6d07a2b9ff84842a9f221Nick Pelly intent.setFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT); 21312f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly intent.putExtra(NfcAdapter.EXTRA_NEW_BOOLEAN_STATE, mIsNfcEnabled); 21322f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly mContext.sendBroadcast(intent); 21332f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly } 213457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 213557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton if (mIsNfcEnabled) { 213657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 213757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton Context context = getApplicationContext(); 213857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 213957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton // Set this to null by default. If there isn't a tag on disk 214057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton // or if there was an error reading the tag then this will cause 214157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton // the status bar icon to be removed. 214257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton NdefMessage myTag = null; 214357d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 214457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton FileInputStream input = null; 214557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 214657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton try { 214757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton input = context.openFileInput(MY_TAG_FILE_NAME); 214857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton ByteArrayOutputStream bytes = new ByteArrayOutputStream(); 214957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 215057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton byte[] buffer = new byte[4096]; 215157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton int read = 0; 215257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton while ((read = input.read(buffer)) > 0) { 215357d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton bytes.write(buffer, 0, read); 215457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 215557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 215657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton myTag = new NdefMessage(bytes.toByteArray()); 215757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } catch (FileNotFoundException e) { 215857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton // Ignore. 215957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } catch (IOException e) { 216057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton Log.e(TAG, "Could not read mytag file: ", e); 216157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton context.deleteFile(MY_TAG_FILE_NAME); 216257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } catch (FormatException e) { 216357d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton Log.e(TAG, "Invalid NdefMessage for mytag", e); 216457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton context.deleteFile(MY_TAG_FILE_NAME); 216557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } finally { 216657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton try { 216757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton if (input != null) { 216857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton input.close(); 216957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 217057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } catch (IOException e) { 217157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton // Ignore 217257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 217357d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 217457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 217557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton try { 217657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton mNfcAdapter.localSet(myTag); 217757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } catch (RemoteException e) { 217857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton // Ignore 217957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 218057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } else { 218157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton sendMessage(MSG_HIDE_MY_TAG_ICON, null); 218257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 21830e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 2184f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 2185f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 2186f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Reset all internals 21872f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly private synchronized void reset() { 218874180bda362a8bc9d2f701d2c17bec0f63c20bbfBrad Fitzpatrick // TODO: none of these appear to be synchronized but are 218974180bda362a8bc9d2f701d2c17bec0f63c20bbfBrad Fitzpatrick // read/written from different threads (notably Binder threads)... 2190f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 2191f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Clear tables 2192f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly mObjectMap.clear(); 2193f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly mSocketMap.clear(); 2194f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 2195f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly // Reset variables 2196f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly mIsNfcEnabled = false; 2197f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 2198f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 21992f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly private synchronized Object findObject(int key) { 2200f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly Object device = null; 2201f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 2202f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly device = mObjectMap.get(key); 2203f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly if (device == null) { 2204f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly Log.w(TAG, "Handle not found !"); 2205f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 2206f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 2207f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly return device; 2208f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 2209f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 22102f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly synchronized void registerTagObject(NativeNfcTag nativeTag) { 2211b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau mObjectMap.put(nativeTag.getHandle(), nativeTag); 2212b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau } 2213b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau 22142f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly synchronized void unregisterObject(int handle) { 2215b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau mObjectMap.remove(handle); 2216f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 2217f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 22182f8ac1e6cdeb32569bc6477d53a2d0d5608758b1Nick Pelly private synchronized Object findSocket(int key) { 2219bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly if (mSocketMap == null) { 2220bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly return null; 2221bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly } 2222bfb74ec41fe9d2d1ce72467888432f0b3296425fNick Pelly return mSocketMap.get(key); 2223f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 2224f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 2225f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly private void RemoveSocket(int key) { 2226f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly mSocketMap.remove(key); 2227f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 2228f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 2229d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton /** For use by code in this process */ 2230d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton public LlcpSocket createLlcpSocket(int sap, int miu, int rw, int linearBufferLength) { 2231d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton try { 2232d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton int handle = mNfcAdapter.createLlcpSocket(sap, miu, rw, linearBufferLength); 22331be4eb6d61c4b6e02c5a5cb7ac3b3a68cae7d7a2Jeff Hamilton if (ErrorCodes.isError(handle)) { 22341be4eb6d61c4b6e02c5a5cb7ac3b3a68cae7d7a2Jeff Hamilton Log.e(TAG, "unable to create socket: " + ErrorCodes.asString(handle)); 22351be4eb6d61c4b6e02c5a5cb7ac3b3a68cae7d7a2Jeff Hamilton return null; 22361be4eb6d61c4b6e02c5a5cb7ac3b3a68cae7d7a2Jeff Hamilton } 2237d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton return new LlcpSocket(mLlcpSocket, handle); 2238d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton } catch (RemoteException e) { 2239d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton // This will never happen since the code is calling into it's own process 2240d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton throw new IllegalStateException("unable to talk to myself", e); 2241d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton } 2242d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton } 2243d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton 2244d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton /** For use by code in this process */ 2245d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton public LlcpServiceSocket createLlcpServiceSocket(int sap, String sn, int miu, int rw, 2246d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton int linearBufferLength) { 2247d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton try { 2248d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton int handle = mNfcAdapter.createLlcpServiceSocket(sap, sn, miu, rw, linearBufferLength); 22491be4eb6d61c4b6e02c5a5cb7ac3b3a68cae7d7a2Jeff Hamilton if (ErrorCodes.isError(handle)) { 22501be4eb6d61c4b6e02c5a5cb7ac3b3a68cae7d7a2Jeff Hamilton Log.e(TAG, "unable to create socket: " + ErrorCodes.asString(handle)); 22511be4eb6d61c4b6e02c5a5cb7ac3b3a68cae7d7a2Jeff Hamilton return null; 22521be4eb6d61c4b6e02c5a5cb7ac3b3a68cae7d7a2Jeff Hamilton } 2253d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton return new LlcpServiceSocket(mLlcpServerSocketService, mLlcpSocket, handle); 2254d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton } catch (RemoteException e) { 2255d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton // This will never happen since the code is calling into it's own process 2256d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton throw new IllegalStateException("unable to talk to myself", e); 2257d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton } 2258d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton } 2259d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton 22600e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private void activateLlcpLink() { 22610e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly /* Broadcast Intent Link LLCP activated */ 22620e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly Intent LlcpLinkIntent = new Intent(); 22630e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly LlcpLinkIntent.setAction(NfcAdapter.ACTION_LLCP_LINK_STATE_CHANGED); 2264f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 22650e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly LlcpLinkIntent.putExtra(NfcAdapter.EXTRA_LLCP_LINK_STATE_CHANGED, 22660e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly NfcAdapter.LLCP_LINK_STATE_ACTIVATED); 2267f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 2268ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "Broadcasting LLCP activation"); 22690e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly mContext.sendOrderedBroadcast(LlcpLinkIntent, NFC_PERM); 22700e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly } 2271f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 227257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton public void sendMockNdefTag(NdefMessage msg) { 2273b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton sendMessage(MSG_MOCK_NDEF, msg); 227457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 227557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 2276b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau void sendMessage(int what, Object obj) { 2277b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau Message msg = mHandler.obtainMessage(); 2278b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau msg.what = what; 2279b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau msg.obj = obj; 2280b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau mHandler.sendMessage(msg); 2281b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau } 2282b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau 2283b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton final class NfcServiceHandler extends Handler { 2284232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen 2285232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen public NdefMessage[] findAndReadNdef(NativeNfcTag nativeTag) { 2286232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen // Try to find NDEF on any of the technologies. 2287232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen int[] technologies = nativeTag.getTechList(); 2288232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen int[] handles = nativeTag.getHandleList(); 2289232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen int techIndex = 0; 2290232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen int lastHandleScanned = 0; 2291232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen boolean ndefFoundAndConnected = false; 2292232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen NdefMessage[] ndefMsgs = null; 2293eb488e0b323ec2d4fe03528e2b9f3a73e66ae17dMartijn Coenen boolean foundFormattable = false; 2294e08c8939be916c393a62b995b2dee438cb8cec2fMartijn Coenen int formattableHandle = 0; 2295e08c8939be916c393a62b995b2dee438cb8cec2fMartijn Coenen int formattableTechnology = 0; 2296232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen 2297232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen while ((!ndefFoundAndConnected) && (techIndex < technologies.length)) { 2298232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen if (handles[techIndex] != lastHandleScanned) { 2299232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen // We haven't seen this handle yet, connect and checkndef 2300232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen if (nativeTag.connect(technologies[techIndex])) { 2301eb488e0b323ec2d4fe03528e2b9f3a73e66ae17dMartijn Coenen // Check if this type is NDEF formatable 2302434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen if (!foundFormattable) { 2303434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen if (nativeTag.isNdefFormatable()) { 2304434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen foundFormattable = true; 2305434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen formattableHandle = nativeTag.getConnectedHandle(); 2306434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen formattableTechnology = nativeTag.getConnectedTechnology(); 2307434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen // We'll only add formattable tech if no ndef is 2308434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen // found - this is because libNFC refuses to format 2309434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen // an already NDEF formatted tag. 2310434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen } 2311434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen nativeTag.reconnect(); 2312434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen } // else, already found formattable technology 2313434bb34457422a9cad664b73ed0d425db4a3093dMartijn Coenen 2314232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen int[] ndefinfo = new int[2]; 2315232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen if (nativeTag.checkNdef(ndefinfo)) { 2316232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen ndefFoundAndConnected = true; 2317232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen boolean generateEmptyNdef = false; 2318232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen 2319232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen int supportedNdefLength = ndefinfo[0]; 2320232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen int cardState = ndefinfo[1]; 2321232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen byte[] buff = nativeTag.read(); 2322232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen if (buff != null) { 2323232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen ndefMsgs = new NdefMessage[1]; 2324232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen try { 2325232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen ndefMsgs[0] = new NdefMessage(buff); 2326232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen nativeTag.addNdefTechnology(ndefMsgs[0], 2327ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen nativeTag.getConnectedHandle(), 23280a58617a63e32d2f43e3aabc705fb7b9de464482Martijn Coenen nativeTag.getConnectedLibNfcType(), 2329f34ea55002a5f7fbe8486c6ab6f4a17481a9558fMartijn Coenen nativeTag.getConnectedTechnology(), 2330232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen supportedNdefLength, cardState); 2331232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen nativeTag.reconnect(); 2332232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } catch (FormatException e) { 2333232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen // Create an intent anyway, without NDEF messages 2334232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen generateEmptyNdef = true; 2335232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } 2336232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } else { 2337232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen generateEmptyNdef = true; 2338232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } 2339232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen 2340232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen if (generateEmptyNdef) { 2341232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen ndefMsgs = new NdefMessage[] { }; 2342ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen nativeTag.addNdefTechnology(null, 2343ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen nativeTag.getConnectedHandle(), 23440a58617a63e32d2f43e3aabc705fb7b9de464482Martijn Coenen nativeTag.getConnectedLibNfcType(), 2345f34ea55002a5f7fbe8486c6ab6f4a17481a9558fMartijn Coenen nativeTag.getConnectedTechnology(), 2346ae7d8d800ba73502d21e54d1deef16be0f061866Martijn Coenen supportedNdefLength, cardState); 2347232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen nativeTag.reconnect(); 2348232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } 2349232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } // else, no NDEF on this tech, continue loop 2350232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } else { 2351232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen // Connect failed, tag maybe lost. Try next handle 2352232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen // anyway. 2353232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } 2354232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } 2355232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen lastHandleScanned = handles[techIndex]; 2356232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen techIndex++; 2357232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } 2358e08c8939be916c393a62b995b2dee438cb8cec2fMartijn Coenen if (ndefMsgs == null && foundFormattable) { 2359e08c8939be916c393a62b995b2dee438cb8cec2fMartijn Coenen // Tag is not NDEF yet, and found a formattable target, 2360e08c8939be916c393a62b995b2dee438cb8cec2fMartijn Coenen // so add formattable tech to tech list. 2361e08c8939be916c393a62b995b2dee438cb8cec2fMartijn Coenen nativeTag.addNdefFormatableTechnology( 2362e08c8939be916c393a62b995b2dee438cb8cec2fMartijn Coenen formattableHandle, 2363e08c8939be916c393a62b995b2dee438cb8cec2fMartijn Coenen formattableTechnology); 2364e08c8939be916c393a62b995b2dee438cb8cec2fMartijn Coenen } 2365232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen 2366232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen return ndefMsgs; 2367232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } 2368232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen 2369b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau @Override 2370b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau public void handleMessage(Message msg) { 2371f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly switch (msg.what) { 2372b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton case MSG_MOCK_NDEF: { 2373b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton NdefMessage ndefMsg = (NdefMessage) msg.obj; 2374b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton Tag tag = Tag.createMockTag(new byte[] { 0x00 }, 2375b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton new int[] { }, 2376b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton new Bundle[] { }); 237757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton Log.d(TAG, "mock NDEF tag, starting corresponding activity"); 237857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton Log.d(TAG, tag.toString()); 23793fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton dispatchTag(tag, new NdefMessage[] { ndefMsg }); 238057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton break; 238157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 238257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 2383f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly case MSG_NDEF_TAG: 2384ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "Tag detected, notifying applications"); 2385f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly NativeNfcTag nativeTag = (NativeNfcTag) msg.obj; 2386232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen NdefMessage[] ndefMsgs = findAndReadNdef(nativeTag); 2387232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen 2388232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen if (ndefMsgs != null) { 238998eb4249dcec63009b1839c005f0cf9048e1f7beMartijn Coenen nativeTag.startPresenceChecking(); 2390232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen dispatchNativeTag(nativeTag, ndefMsgs); 2391232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } else { 2392232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen // No ndef found or connect failed, just try to reconnect and dispatch 2393232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen if (nativeTag.reconnect()) { 239498eb4249dcec63009b1839c005f0cf9048e1f7beMartijn Coenen nativeTag.startPresenceChecking(); 23953fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton dispatchNativeTag(nativeTag, null); 2396232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen } else { 2397232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen Log.w(TAG, "Failed to connect to tag"); 2398232eabb7c6f04506dbf272e06b115178311a57b9Martijn Coenen nativeTag.disconnect(); 2399f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly } 2400f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly } 2401f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly break; 24023fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 2403f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly case MSG_CARD_EMULATION: 2404ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "Card Emulation message"); 2405f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly byte[] aid = (byte[]) msg.obj; 240649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly /* Send broadcast */ 240749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly Intent aidIntent = new Intent(); 240849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly aidIntent.setAction(ACTION_AID_SELECTED); 240949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly aidIntent.putExtra(EXTRA_AID, aid); 241049d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (DBG) Log.d(TAG, "Broadcasting ACTION_AID_SELECTED"); 241193d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton mContext.sendBroadcast(aidIntent, NFCEE_ADMIN_PERM); 2412f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly break; 2413f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly 2414f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly case MSG_LLCP_LINK_ACTIVATION: 2415f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly NativeP2pDevice device = (NativeP2pDevice) msg.obj; 2416f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly 2417f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly Log.d(TAG, "LLCP Activation message"); 2418f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly 2419f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly if (device.getMode() == NativeP2pDevice.MODE_P2P_TARGET) { 2420ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "NativeP2pDevice.MODE_P2P_TARGET"); 2421f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly if (device.doConnect()) { 2422f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly /* Check Llcp compliancy */ 2423f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly if (mManager.doCheckLlcp()) { 2424f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly /* Activate Llcp Link */ 2425f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly if (mManager.doActivateLlcp()) { 2426ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "Initiator Activate LLCP OK"); 2427663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau // Register P2P device 2428663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau mObjectMap.put(device.getHandle(), device); 2429eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton activateLlcpLink(); 2430eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton } else { 2431eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton /* should not happen */ 2432eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton Log.w(TAG, "Initiator Activate LLCP NOK. Disconnect."); 2433eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton device.doDisconnect(); 2434b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau } 2435b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau 2436f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly } else { 2437ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "Remote Target does not support LLCP. Disconnect."); 2438f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly device.doDisconnect(); 2439b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau } 2440eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton } else { 24417f5487be0414b1f92b41cc5f7b0b7db846d9b1c9Arnaud Ferir if (DBG) Log.d(TAG, "Cannot connect remote Target. Polling loop restarted..."); 24427f5487be0414b1f92b41cc5f7b0b7db846d9b1c9Arnaud Ferir /* The polling loop should have been restarted in failing doConnect */ 2443f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly } 2444f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly 2445f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly } else if (device.getMode() == NativeP2pDevice.MODE_P2P_INITIATOR) { 2446ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "NativeP2pDevice.MODE_P2P_INITIATOR"); 2447f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly /* Check Llcp compliancy */ 2448f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly if (mManager.doCheckLlcp()) { 2449f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly /* Activate Llcp Link */ 2450f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly if (mManager.doActivateLlcp()) { 2451ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "Target Activate LLCP OK"); 2452663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau // Register P2P device 2453663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau mObjectMap.put(device.getHandle(), device); 2454eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton activateLlcpLink(); 2455eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton } 24566f7fd8dbda9092e36e58273e843ae3e4a200e2ffJeff Hamilton } else { 2457ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton Log.w(TAG, "checkLlcp failed"); 2458f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly } 2459b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau } 2460f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly break; 2461f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly 2462f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly case MSG_LLCP_LINK_DEACTIVATED: 2463eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton device = (NativeP2pDevice) msg.obj; 2464eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton 2465eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton Log.d(TAG, "LLCP Link Deactivated message. Restart polling loop."); 2466663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau synchronized (NfcService.this) { 2467663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau /* Check if the device has been already unregistered */ 2468663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau if (mObjectMap.remove(device.getHandle()) != null) { 2469663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau /* Disconnect if we are initiator */ 2470663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau if (device.getMode() == NativeP2pDevice.MODE_P2P_TARGET) { 2471663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau if (DBG) Log.d(TAG, "disconnecting from target"); 2472663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau /* Restart polling loop */ 2473663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau device.doDisconnect(); 2474663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau } else { 2475663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau if (DBG) Log.d(TAG, "not disconnecting from initiator"); 2476663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau } 2477663c0153d74ddbe8e8ec1f9f55780beafa2543faSylvain Fonteneau } 2478cdab1112486288ee6ccc9eb5c55456f9618f6b9cJeff Hamilton } 2479eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton 2480f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly /* Broadcast Intent Link LLCP activated */ 2481f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly Intent LlcpLinkIntent = new Intent(); 2482f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly LlcpLinkIntent.setAction(NfcAdapter.ACTION_LLCP_LINK_STATE_CHANGED); 2483f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly LlcpLinkIntent.putExtra(NfcAdapter.EXTRA_LLCP_LINK_STATE_CHANGED, 2484f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly NfcAdapter.LLCP_LINK_STATE_DEACTIVATED); 2485ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "Broadcasting LLCP deactivation"); 2486f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly mContext.sendOrderedBroadcast(LlcpLinkIntent, NFC_PERM); 2487f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly break; 2488f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly 2489f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly case MSG_TARGET_DESELECTED: 2490f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly /* Broadcast Intent Target Deselected */ 2491ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "Target Deselected"); 2492f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly Intent TargetDeselectedIntent = new Intent(); 2493f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly TargetDeselectedIntent.setAction(mManager.INTERNAL_TARGET_DESELECTED_ACTION); 2494ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "Broadcasting Intent"); 2495f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly mContext.sendOrderedBroadcast(TargetDeselectedIntent, NFC_PERM); 2496f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly break; 2497f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly 249857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton case MSG_SHOW_MY_TAG_ICON: { 249957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton StatusBarManager sb = (StatusBarManager) getSystemService( 250057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton Context.STATUS_BAR_SERVICE); 250157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton sb.setIcon("nfc", R.drawable.stat_sys_nfc, 0); 250257d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton break; 250357d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 250457d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 250557d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton case MSG_HIDE_MY_TAG_ICON: { 250657d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton StatusBarManager sb = (StatusBarManager) getSystemService( 250757d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton Context.STATUS_BAR_SERVICE); 250857d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton sb.removeIcon("nfc"); 250957d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton break; 251057d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton } 251157d376f1ee1a3939977b95759525585abb9601fbJeff Hamilton 2512c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas case MSG_SE_FIELD_ACTIVATED:{ 2513c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas if (DBG) Log.d(TAG, "SE FIELD ACTIVATED"); 2514c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas Intent eventFieldOnIntent = new Intent(); 251549d53329a0c720a7e430220d77805bc1763545b1Nick Pelly eventFieldOnIntent.setAction(ACTION_RF_FIELD_ON_DETECTED); 2516c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas if (DBG) Log.d(TAG, "Broadcasting Intent"); 251793d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton mContext.sendBroadcast(eventFieldOnIntent, NFCEE_ADMIN_PERM); 2518c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas break; 2519c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas } 2520c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas 2521c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas case MSG_SE_FIELD_DEACTIVATED:{ 2522c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas if (DBG) Log.d(TAG, "SE FIELD DEACTIVATED"); 2523c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas Intent eventFieldOffIntent = new Intent(); 252449d53329a0c720a7e430220d77805bc1763545b1Nick Pelly eventFieldOffIntent.setAction(ACTION_RF_FIELD_OFF_DETECTED); 2525c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas if (DBG) Log.d(TAG, "Broadcasting Intent"); 252693d8a69ccadfa01b0a5ec3d7edeb921a1da4bce8Jeff Hamilton mContext.sendBroadcast(eventFieldOffIntent, NFCEE_ADMIN_PERM); 2527c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas break; 2528c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas } 2529c9a2ae7cb238e4c72818d084cba0b05e76cba1efdaniel_tomas 2530f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly default: 2531f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly Log.e(TAG, "Unknown message received"); 2532f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly break; 2533f348bf5b21c5c39d7b3627327db1e61007d07539Nick Pelly } 2534b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau } 2535d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton 25363fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton private Intent buildTagIntent(Tag tag, NdefMessage[] msgs, String action) { 25373fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton Intent intent = new Intent(action); 2538b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton intent.putExtra(NfcAdapter.EXTRA_TAG, tag); 2539b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton intent.putExtra(NfcAdapter.EXTRA_ID, tag.getId()); 2540b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton intent.putExtra(NfcAdapter.EXTRA_NDEF_MESSAGES, msgs); 2541b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); 2542d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton return intent; 2543d0ec3981792e38afd119fc1c995f111f6182f6c8Jeff Hamilton } 25443fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 25453fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton private void dispatchNativeTag(NativeNfcTag nativeTag, NdefMessage[] msgs) { 25463fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton Tag tag = new Tag(nativeTag.getUid(), nativeTag.getTechList(), 25470c322dc2d942285bfcd9fe5c0bba1a5f36a967f5Jeff Hamilton nativeTag.getTechExtras(), nativeTag.getHandle(), mNfcTagService); 254810389c11abdae4c794082f6c51f11b28b422eddbMartijn Coenen registerTagObject(nativeTag); 254910389c11abdae4c794082f6c51f11b28b422eddbMartijn Coenen if (!dispatchTag(tag, msgs)) { 255010389c11abdae4c794082f6c51f11b28b422eddbMartijn Coenen unregisterObject(nativeTag.getHandle()); 25513fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton nativeTag.disconnect(); 25523fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 25533fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 25543fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 25553fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton public byte[] concat(byte[]... arrays) { 25563fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton int length = 0; 25573fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton for (byte[] array : arrays) { 25583fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton length += array.length; 25593fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 25603fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton byte[] result = new byte[length]; 25613fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton int pos = 0; 25623fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton for (byte[] array : arrays) { 25633fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton System.arraycopy(array, 0, result, pos, array.length); 25643fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton pos += array.length; 25653fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 25663fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton return result; 25673fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 25683fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 25693fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton private Uri parseWellKnownUriRecord(NdefRecord record) { 25703fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton byte[] payload = record.getPayload(); 25713fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 25723fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton /* 25733fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton * payload[0] contains the URI Identifier Code, per the 25743fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton * NFC Forum "URI Record Type Definition" section 3.2.2. 25753fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton * 25763fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton * payload[1]...payload[payload.length - 1] contains the rest of 25773fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton * the URI. 25783fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton */ 25793fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton String prefix = URI_PREFIX_MAP[(payload[0] & 0xff)]; 25803fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton byte[] fullUri = concat(prefix.getBytes(Charsets.UTF_8), 25813fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton Arrays.copyOfRange(payload, 1, payload.length)); 25823fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton return Uri.parse(new String(fullUri, Charsets.UTF_8)); 25833fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 25843fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 25853fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton private boolean setTypeOrDataFromNdef(Intent intent, NdefRecord record) { 25863fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton short tnf = record.getTnf(); 25873fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton byte[] type = record.getType(); 2588adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton try { 2589adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton switch (tnf) { 2590adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton case NdefRecord.TNF_MIME_MEDIA: { 2591adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton intent.setType(new String(type, Charsets.US_ASCII)); 25923fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton return true; 2593adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton } 2594adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton case NdefRecord.TNF_ABSOLUTE_URI: { 2595adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton intent.setData(Uri.parse(new String(record.getPayload(), Charsets.UTF_8))); 2596adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton return true; 2597adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton } 2598adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton case NdefRecord.TNF_WELL_KNOWN: { 2599adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton byte[] payload = record.getPayload(); 2600adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton if (payload == null || payload.length == 0) return false; 2601adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton if (Arrays.equals(type, NdefRecord.RTD_TEXT)) { 2602adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton intent.setType("text/plain"); 2603adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton return true; 2604adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton } else if (Arrays.equals(type, NdefRecord.RTD_SMART_POSTER)) { 2605adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton // Parse the smart poster looking for the URI 2606adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton try { 2607adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton NdefMessage msg = new NdefMessage(record.getPayload()); 2608adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton for (NdefRecord subRecord : msg.getRecords()) { 2609adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton short subTnf = subRecord.getTnf(); 2610adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton if (subTnf == NdefRecord.TNF_WELL_KNOWN 2611adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton && Arrays.equals(subRecord.getType(), 2612adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton NdefRecord.RTD_URI)) { 2613adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton intent.setData(parseWellKnownUriRecord(subRecord)); 2614adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton return true; 2615adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton } else if (subTnf == NdefRecord.TNF_ABSOLUTE_URI) { 2616adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton intent.setData(Uri.parse(new String(subRecord.getPayload(), 2617adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton Charsets.UTF_8))); 2618adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton return true; 2619adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton } 26203fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 2621adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton } catch (FormatException e) { 2622adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton return false; 26233fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 2624adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton } else if (Arrays.equals(type, NdefRecord.RTD_URI)) { 2625adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton intent.setData(parseWellKnownUriRecord(record)); 2626adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton return true; 26273fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 2628adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton return false; 26293fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 26303fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 2631adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton return false; 2632adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton } catch (Exception e) { 2633adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton Log.e(TAG, "failed to parse record", e); 2634adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton return false; 26353fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 26363fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 26373fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 26383fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton /** Returns false if no activities were found to dispatch to */ 26393fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton private boolean dispatchTag(Tag tag, NdefMessage[] msgs) { 26403fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton if (DBG) { 26413fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton Log.d(TAG, "Dispatching tag"); 26423fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton Log.d(TAG, tag.toString()); 26433fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 26443fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 264505973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton IntentFilter[] overrideFilters; 264605973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton PendingIntent overrideIntent; 264724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton String[][] overrideTechLists; 2648a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton boolean foregroundNdefPush = mNdefPushClient.getForegroundMessage() != null; 264905973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton synchronized (mNfcAdapter) { 265005973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton overrideFilters = mDispatchOverrideFilters; 265105973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton overrideIntent = mDispatchOverrideIntent; 265224dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton overrideTechLists = mDispatchOverrideTechLists; 265305973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 265405973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton 265505973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton // First look for dispatch overrides 2656a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton if (overrideIntent != null) { 265705973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton if (DBG) Log.d(TAG, "Attempting to dispatch tag with override"); 265849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly try { 265924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (dispatchTagInternal(tag, msgs, overrideIntent, overrideFilters, 266024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton overrideTechLists)) { 266105973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton if (DBG) Log.d(TAG, "Dispatched to override"); 266205973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton return true; 266305973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 266405973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton Log.w(TAG, "Dispatch override registered, but no filters matched"); 266505973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } catch (CanceledException e) { 266605973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton Log.w(TAG, "Dispatch overrides pending intent was canceled"); 266705973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton synchronized (mNfcAdapter) { 266805973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton mDispatchOverrideFilters = null; 266905973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton mDispatchOverrideIntent = null; 267024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton mDispatchOverrideTechLists = null; 267105973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 267205973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 267305973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 267405973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton 2675a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton // If there is not foreground NDEF push setup try a normal dispatch. 2676a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton // 2677a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton // This is avoided when disabled in the NDEF push case to avoid the situation where each 2678a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton // user has a different app in the foreground, causing each to launch itself on the 2679a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton // remote device and the apps swapping which is in the foreground on each phone. 2680a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton if (!foregroundNdefPush) { 2681a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton try { 268224dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton return dispatchTagInternal(tag, msgs, null, null, null); 2683a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton } catch (CanceledException e) { 2684a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton Log.e(TAG, "CanceledException unexpected here", e); 2685a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton return false; 2686a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton } 268705973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 2688a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton 2689a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton return false; 269005973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 269105973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton 269224dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton /** Returns true if the tech list filter matches the techs on the tag */ 269324dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton private boolean filterMatch(String[] tagTechs, String[] filterTechs) { 269424dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (filterTechs == null || filterTechs.length == 0) return false; 269524dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton 269624dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton for (String tech : filterTechs) { 269724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (Arrays.binarySearch(tagTechs, tech) < 0) { 269824dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton return false; 269924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 270024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 270124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton return true; 270224dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 270324dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton 270405973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton // Dispatch to either an override pending intent or a standard startActivity() 270505973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton private boolean dispatchTagInternal(Tag tag, NdefMessage[] msgs, 270624dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton PendingIntent overrideIntent, IntentFilter[] overrideFilters, 270724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton String[][] overrideTechLists) 270805973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton throws CanceledException{ 27093fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton Intent intent; 271024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton 271124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // 271224dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // Try the NDEF content specific dispatch 271324dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // 27143fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton if (msgs != null && msgs.length > 0) { 27153fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton NdefMessage msg = msgs[0]; 27163fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton NdefRecord[] records = msg.getRecords(); 27173fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton if (records.length > 0) { 27183fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton // Found valid NDEF data, try to dispatch that first 27193fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton NdefRecord record = records[0]; 27203fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 27213fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton intent = buildTagIntent(tag, msgs, NfcAdapter.ACTION_NDEF_DISCOVERED); 2722adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton if (setTypeOrDataFromNdef(intent, record)) { 2723adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton // The record contains filterable data, try to start a matching activity 272424dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (startDispatchActivity(intent, overrideIntent, overrideFilters, 272524dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton overrideTechLists)) { 2726adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton // If an activity is found then skip further dispatching 2727adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton return true; 2728adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton } else { 2729adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton if (DBG) Log.d(TAG, "No activities for NDEF handling of " + intent); 2730adb527a9ce286d019d766a3b207a2251dea89bd1Jeff Hamilton } 27313fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 27323fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 27333fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 27343fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 273524dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // 27363fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton // Try the technology specific dispatch 273724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // 273824dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton String[] tagTechs = tag.getTechList(); 273924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton Arrays.sort(tagTechs); 274024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton 274124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (overrideIntent != null) { 274224dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // There are dispatch overrides in place 274324dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (overrideTechLists != null) { 274424dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton for (String[] filterTechs : overrideTechLists) { 274524dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (filterMatch(tagTechs, filterTechs)) { 274624dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // An override matched, send it to the foreground activity. 274724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton intent = buildTagIntent(tag, msgs, 27488d69f5205e3c4a7c2eb869bbf4983d3d9fe45ab0Nick Pelly NfcAdapter.ACTION_TECH_DISCOVERED); 274924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton overrideIntent.send(mContext, Activity.RESULT_OK, intent); 275024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton return true; 275124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 275224dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 275324dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 275405973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } else { 275524dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // Standard tech dispatch path 275624dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton ArrayList<ResolveInfo> matches = new ArrayList<ResolveInfo>(); 275724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton ArrayList<ComponentInfo> registered = mTechListFilters.getComponents(); 275849d53329a0c720a7e430220d77805bc1763545b1Nick Pelly 275924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // Check each registered activity to see if it matches 276024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton for (ComponentInfo info : registered) { 276124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // Don't allow wild card matching 276224dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (filterMatch(tagTechs, info.techs)) { 27633ca2b3f925d434559300ee2fb10bfcdd365a0fb6Jeff Hamilton // Add the activity as a match if it's not already in the list 27643ca2b3f925d434559300ee2fb10bfcdd365a0fb6Jeff Hamilton if (!matches.contains(info.resolveInfo)) { 27653ca2b3f925d434559300ee2fb10bfcdd365a0fb6Jeff Hamilton matches.add(info.resolveInfo); 27663ca2b3f925d434559300ee2fb10bfcdd365a0fb6Jeff Hamilton } 276724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 276824dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 276949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly 277024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (matches.size() == 1) { 277124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // Single match, launch directly 27728d69f5205e3c4a7c2eb869bbf4983d3d9fe45ab0Nick Pelly intent = buildTagIntent(tag, msgs, NfcAdapter.ACTION_TECH_DISCOVERED); 277324dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton ResolveInfo info = matches.get(0); 277424dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton intent.setClassName(info.activityInfo.packageName, info.activityInfo.name); 277524dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton try { 277624dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton mContext.startActivity(intent); 277724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton return true; 277824dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } catch (ActivityNotFoundException e) { 277924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (DBG) Log.w(TAG, "No activities for technology handling of " + intent); 278024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 278124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } else if (matches.size() > 1) { 278224dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // Multiple matches, show a custom activity chooser dialog 278324dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton intent = new Intent(mContext, TechListChooserActivity.class); 278424dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); 278524dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton intent.putExtra(Intent.EXTRA_INTENT, 27868d69f5205e3c4a7c2eb869bbf4983d3d9fe45ab0Nick Pelly buildTagIntent(tag, msgs, NfcAdapter.ACTION_TECH_DISCOVERED)); 278724dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton intent.putParcelableArrayListExtra(TechListChooserActivity.EXTRA_RESOLVE_INFOS, 278824dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton matches); 278924dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton try { 279024dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton mContext.startActivity(intent); 279124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton return true; 279224dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } catch (ActivityNotFoundException e) { 279324dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (DBG) Log.w(TAG, "No activities for technology handling of " + intent); 279424dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 279524dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } else { 279624dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // No matches, move on 279749d53329a0c720a7e430220d77805bc1763545b1Nick Pelly if (DBG) Log.w(TAG, "No activities for technology handling"); 279824dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } 27993fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 28003fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton 280124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // 28023fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton // Try the generic intent 280324dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton // 28043fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton intent = buildTagIntent(tag, msgs, NfcAdapter.ACTION_TAG_DISCOVERED); 280524dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (startDispatchActivity(intent, overrideIntent, overrideFilters, overrideTechLists)) { 28063fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton return true; 280705973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } else { 28083fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton Log.e(TAG, "No tag fallback activity found for " + intent); 28093fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton return false; 28103fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 28113fb30ae5bf51d9ffe6271a345d55905dade8040dJeff Hamilton } 281205973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton 281305973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton private boolean startDispatchActivity(Intent intent, PendingIntent overrideIntent, 281424dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton IntentFilter[] overrideFilters, String[][] overrideTechLists) 281524dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton throws CanceledException { 281605973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton if (overrideIntent != null) { 2817a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton boolean found = false; 281824dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton if (overrideFilters == null && overrideTechLists == null) { 2819a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton // No filters means to always dispatch regardless of match 2820a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton found = true; 282124dbea55709219e42aa3b6b6578f29ffd447a786Jeff Hamilton } else if (overrideFilters != null) { 2822a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton for (IntentFilter filter : overrideFilters) { 2823a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton if (filter.match(mContext.getContentResolver(), intent, false, TAG) >= 0) { 2824a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton found = true; 2825a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton break; 2826a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton } 282705973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 282805973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 2829a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton 2830a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton if (found) { 2831a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton Log.i(TAG, "Dispatching to override intent " + overrideIntent); 2832a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton overrideIntent.send(mContext, Activity.RESULT_OK, intent); 2833a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton return true; 2834a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton } else { 2835a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton return false; 2836a1935861ea244b45e29b0ec9b6f263de3c08a2d6Jeff Hamilton } 283705973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } else { 283805973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton try { 28391a9eca5f84036d7dd3e28000290caa2f641856deNick Pelly // If the current app called stopAppSwitches() then our startActivity() 28401a9eca5f84036d7dd3e28000290caa2f641856deNick Pelly // can be delayed for several seconds. This happens with the default home 28411a9eca5f84036d7dd3e28000290caa2f641856deNick Pelly // screen. As a system service we can override this behavior with 28421a9eca5f84036d7dd3e28000290caa2f641856deNick Pelly // resumeAppSwitches() 28431a9eca5f84036d7dd3e28000290caa2f641856deNick Pelly mIActivityManager.resumeAppSwitches(); 28441a9eca5f84036d7dd3e28000290caa2f641856deNick Pelly } catch (RemoteException e) { } 28451a9eca5f84036d7dd3e28000290caa2f641856deNick Pelly try { 284605973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton mContext.startActivity(intent); 284705973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton return true; 284805973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } catch (ActivityNotFoundException e) { 284905973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton return false; 285005973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 285105973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 285205973d55daf68a286c932ee4e7ffbd6bb53789e0Jeff Hamilton } 2853b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton } 2854b230214bcb557184cb54174889cbede53b92d54fSylvain Fonteneau 2855b74200f40f9d4f536b8782974d444f1f9178076fJeff Hamilton private NfcServiceHandler mHandler = new NfcServiceHandler(); 285649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly 28577c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly private class EnableDisableDiscoveryTask extends AsyncTask<Boolean, Void, Void> { 2858fa746bcc16d57ff12d373b9139558f5bc7164b30Jason parks @Override 28597c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly protected Void doInBackground(Boolean... enable) { 2860df4e65b5dacb313e7d68bad797a843a175febbedNick Pelly if (enable != null && enable.length > 0 && enable[0]) { 2861161f84b5487ce4c1ebef9fe24ba4de00f6f756eaNick Pelly synchronized (NfcService.this) { 2862161f84b5487ce4c1ebef9fe24ba4de00f6f756eaNick Pelly mScreenOn = true; 286349d53329a0c720a7e430220d77805bc1763545b1Nick Pelly applyRouting(); 2864161f84b5487ce4c1ebef9fe24ba4de00f6f756eaNick Pelly } 28657c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly } else { 2866533043d1003de2f6a20a29201100d94c3c7bc9caNick Pelly mWakeLock.acquire(); 2867161f84b5487ce4c1ebef9fe24ba4de00f6f756eaNick Pelly synchronized (NfcService.this) { 2868161f84b5487ce4c1ebef9fe24ba4de00f6f756eaNick Pelly mScreenOn = false; 286949d53329a0c720a7e430220d77805bc1763545b1Nick Pelly applyRouting(); 28702436ffe91853535fad87b7a8e03d8883bae20f20Arnaud Ferir maybeDisconnectTarget(); 2871161f84b5487ce4c1ebef9fe24ba4de00f6f756eaNick Pelly } 2872533043d1003de2f6a20a29201100d94c3c7bc9caNick Pelly mWakeLock.release(); 28737c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly } 28747c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly return null; 28757c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly } 28767c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly } 28777c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly 28780e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly private final BroadcastReceiver mReceiver = new BroadcastReceiver() { 28790e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly @Override 28800e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly public void onReceive(Context context, Intent intent) { 2881eead88c5e2bdd34eb33fdf2c76717f9edb9e0396Jeff Hamilton if (intent.getAction().equals( 2882f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly NativeNfcManager.INTERNAL_TARGET_DESELECTED_ACTION)) { 2883ef92efa7fd86d75cc35465b84e8740f941ff0f0aJeff Hamilton if (DBG) Log.d(TAG, "INERNAL_TARGET_DESELECTED_ACTION"); 28840e6a0a0f50132e14d5ecad61c46e07c67fbb26fbNick Pelly 2885f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly /* Restart polling loop for notification */ 288649d53329a0c720a7e430220d77805bc1763545b1Nick Pelly applyRouting(); 2887f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly 288865945ad77cadb7a3bdf171497877d2325b23def5Nick Pelly } else if (intent.getAction().equals(Intent.ACTION_SCREEN_ON)) { 28897c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly // Perform discovery enable in thread to protect against ANR when the 28907c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly // NFC stack wedges. This is *not* the correct way to fix this issue - 28917c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly // configuration of the local NFC adapter should be very quick and should 28927c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly // be safe on the main thread, and the NFC stack should not wedge. 2893df4e65b5dacb313e7d68bad797a843a175febbedNick Pelly new EnableDisableDiscoveryTask().execute(new Boolean(true)); 289465945ad77cadb7a3bdf171497877d2325b23def5Nick Pelly } else if (intent.getAction().equals(Intent.ACTION_SCREEN_OFF)) { 28957c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly // Perform discovery disable in thread to protect against ANR when the 28967c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly // NFC stack wedges. This is *not* the correct way to fix this issue - 28977c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly // configuration of the local NFC adapter should be very quick and should 28987c034a7fe7d36b1ab039af2c44717812ea02657eNick Pelly // be safe on the main thread, and the NFC stack should not wedge. 2899df4e65b5dacb313e7d68bad797a843a175febbedNick Pelly new EnableDisableDiscoveryTask().execute(new Boolean(false)); 2900bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } else if (intent.getAction().equals(ACTION_MASTER_CLEAR_NOTIFICATION)) { 2901bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks int handle = mSecureElement.doOpenSecureElementConnection(); 2902bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks if (handle == 0) { 2903bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks Log.e(TAG, "Could not open the secure element!"); 2904bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks return; 2905bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2906bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2907bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks synchronized (NfcService.this) { 2908bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks for (String packageName : mTearDownApdus.keySet()) { 2909bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks for (byte[] cmd : mTearDownApdus.get(packageName).get()) { 2910bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks mSecureElement.doTransceive(handle, cmd); 2911bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2912bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2913bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2914bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 2915bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks mSecureElement.doDisconnect(handle); 2916bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } else if (intent.getAction().equals(Intent.ACTION_PACKAGE_REMOVED)) { 29177a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton boolean dataRemoved = intent.getBooleanExtra(Intent.EXTRA_DATA_REMOVED, false); 29187a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton if (dataRemoved) { 29197a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton Uri data = intent.getData(); 29207a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton if (data == null) return; 29217a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton String packageName = data.getSchemeSpecificPart(); 29227a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton ApduList apdus = null; 29237a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton 29247a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton synchronized (NfcService.this) { 29257a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton apdus = mTearDownApdus.remove(packageName); 29267a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton if (apdus == null) { 29277a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton return; 29287a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton } 29295e7b58d9c95a9074a42c3e4820e2467dfa85af10Jeff Hamilton 29305e7b58d9c95a9074a42c3e4820e2467dfa85af10Jeff Hamilton 29317a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton writeTearDownApdusLocked(); 29327a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton } 2933bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 29347a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton int handle = mSecureElement.doOpenSecureElementConnection(); 29357a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton if (handle == 0) { 29367a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton Log.e(TAG, "Could not open the secure element!"); 29377a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton return; 29387a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton } 2939bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks 29407a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton try { 29417a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton for (byte[] cmd : apdus.get()) { 29427a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton mSecureElement.doTransceive(handle, cmd); 29437a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton } 29447a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton } finally { 29457a7f8f8fd82936f0ee005ccfa7ac5c36760ed902Jeff Hamilton mSecureElement.doDisconnect(handle); 2946bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2947bcd6a9954c5abafc6b14aabcc7768d0f03cc956cJason parks } 2948f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 2949f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly } 2950f067256d9556b1e01143f9ae2fd824fa5dc03138Nick Pelly }; 295174180bda362a8bc9d2f701d2c17bec0f63c20bbfBrad Fitzpatrick} 2952