| /* |
| * Copyright (C) 2010 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package com.android.nfc; |
| |
| import com.android.nfc.DeviceHost.DeviceHostListener; |
| import com.android.nfc.DeviceHost.LlcpConnectionlessSocket; |
| import com.android.nfc.DeviceHost.LlcpServerSocket; |
| import com.android.nfc.DeviceHost.LlcpSocket; |
| import com.android.nfc.DeviceHost.NfcDepEndpoint; |
| import com.android.nfc.DeviceHost.TagEndpoint; |
| import com.android.nfc.handover.HandoverManager; |
| import com.android.nfc.dhimpl.NativeNfcManager; |
| import com.android.nfc.dhimpl.NativeNfcSecureElement; |
| |
| import android.app.Application; |
| import android.app.KeyguardManager; |
| import android.app.PendingIntent; |
| import android.content.BroadcastReceiver; |
| import android.content.ContentResolver; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.content.IntentFilter; |
| import android.content.SharedPreferences; |
| import android.content.pm.PackageInfo; |
| import android.content.pm.PackageManager; |
| import android.media.AudioManager; |
| import android.media.SoundPool; |
| import android.net.Uri; |
| import android.nfc.ErrorCodes; |
| import android.nfc.FormatException; |
| import android.nfc.INdefPushCallback; |
| import android.nfc.INfcAdapter; |
| import android.nfc.INfcAdapterExtras; |
| import android.nfc.INfcTag; |
| import android.nfc.NdefMessage; |
| import android.nfc.NfcAdapter; |
| import android.nfc.Tag; |
| import android.nfc.TechListParcel; |
| import android.nfc.TransceiveResult; |
| import android.nfc.tech.Ndef; |
| import android.nfc.tech.TagTechnology; |
| import android.os.AsyncTask; |
| import android.os.Binder; |
| import android.os.Bundle; |
| import android.os.Handler; |
| import android.os.IBinder; |
| import android.os.Message; |
| import android.os.PowerManager; |
| import android.os.Process; |
| import android.os.RemoteException; |
| import android.os.ServiceManager; |
| import android.provider.Settings; |
| import android.util.Log; |
| |
| import java.io.FileDescriptor; |
| import java.io.IOException; |
| import java.io.PrintWriter; |
| import java.util.Arrays; |
| import java.util.HashMap; |
| import java.util.HashSet; |
| import java.util.List; |
| import java.util.concurrent.ExecutionException; |
| |
| public class NfcService extends Application implements DeviceHostListener { |
| private static final String ACTION_MASTER_CLEAR_NOTIFICATION = "android.intent.action.MASTER_CLEAR_NOTIFICATION"; |
| |
| static final boolean DBG = false; |
| static final String TAG = "NfcService"; |
| |
| public static final String SERVICE_NAME = "nfc"; |
| |
| /** Regular NFC permission */ |
| private static final String NFC_PERM = android.Manifest.permission.NFC; |
| private static final String NFC_PERM_ERROR = "NFC permission required"; |
| |
| /** NFC ADMIN permission - only for system apps */ |
| private static final String ADMIN_PERM = android.Manifest.permission.WRITE_SECURE_SETTINGS; |
| private static final String ADMIN_PERM_ERROR = "WRITE_SECURE_SETTINGS permission required"; |
| |
| public static final String PREF = "NfcServicePrefs"; |
| |
| static final String PREF_NFC_ON = "nfc_on"; |
| static final boolean NFC_ON_DEFAULT = true; |
| static final String PREF_NDEF_PUSH_ON = "ndef_push_on"; |
| static final boolean NDEF_PUSH_ON_DEFAULT = true; |
| static final String PREF_FIRST_BEAM = "first_beam"; |
| static final String PREF_FIRST_BOOT = "first_boot"; |
| static final String PREF_AIRPLANE_OVERRIDE = "airplane_override"; |
| |
| static final boolean PN544_QUIRK_DISCONNECT_BEFORE_RECONFIGURE = true; |
| |
| static final int MSG_NDEF_TAG = 0; |
| static final int MSG_CARD_EMULATION = 1; |
| static final int MSG_LLCP_LINK_ACTIVATION = 2; |
| static final int MSG_LLCP_LINK_DEACTIVATED = 3; |
| static final int MSG_TARGET_DESELECTED = 4; |
| static final int MSG_MOCK_NDEF = 7; |
| static final int MSG_SE_FIELD_ACTIVATED = 8; |
| static final int MSG_SE_FIELD_DEACTIVATED = 9; |
| static final int MSG_SE_APDU_RECEIVED = 10; |
| static final int MSG_SE_EMV_CARD_REMOVAL = 11; |
| static final int MSG_SE_MIFARE_ACCESS = 12; |
| |
| static final int TASK_ENABLE = 1; |
| static final int TASK_DISABLE = 2; |
| static final int TASK_BOOT = 3; |
| static final int TASK_EE_WIPE = 4; |
| |
| // Screen state, used by mScreenState |
| static final int SCREEN_STATE_UNKNOWN = 0; |
| static final int SCREEN_STATE_OFF = 1; |
| static final int SCREEN_STATE_ON_LOCKED = 2; |
| static final int SCREEN_STATE_ON_UNLOCKED = 3; |
| |
| // Copied from com.android.nfc_extras to avoid library dependency |
| // Must keep in sync with com.android.nfc_extras |
| static final int ROUTE_OFF = 1; |
| static final int ROUTE_ON_WHEN_SCREEN_ON = 2; |
| |
| /** minimum screen state that enables NFC polling (discovery) */ |
| static final int POLLING_MODE = SCREEN_STATE_ON_UNLOCKED; |
| |
| // for use with playSound() |
| public static final int SOUND_START = 0; |
| public static final int SOUND_END = 1; |
| public static final int SOUND_ERROR = 2; |
| |
| public static final String ACTION_RF_FIELD_ON_DETECTED = |
| "com.android.nfc_extras.action.RF_FIELD_ON_DETECTED"; |
| public static final String ACTION_RF_FIELD_OFF_DETECTED = |
| "com.android.nfc_extras.action.RF_FIELD_OFF_DETECTED"; |
| public static final String ACTION_AID_SELECTED = |
| "com.android.nfc_extras.action.AID_SELECTED"; |
| public static final String EXTRA_AID = "com.android.nfc_extras.extra.AID"; |
| |
| public static final String ACTION_APDU_RECEIVED = |
| "com.android.nfc_extras.action.APDU_RECEIVED"; |
| public static final String EXTRA_APDU_BYTES = |
| "com.android.nfc_extras.extra.APDU_BYTES"; |
| |
| public static final String ACTION_EMV_CARD_REMOVAL = |
| "com.android.nfc_extras.action.EMV_CARD_REMOVAL"; |
| |
| public static final String ACTION_MIFARE_ACCESS_DETECTED = |
| "com.android.nfc_extras.action.MIFARE_ACCESS_DETECTED"; |
| public static final String EXTRA_MIFARE_BLOCK = |
| "com.android.nfc_extras.extra.MIFARE_BLOCK"; |
| |
| //TODO: dont hardcode this |
| private static final byte[][] EE_WIPE_APDUS = { |
| {(byte)0x00, (byte)0xa4, (byte)0x04, (byte)0x00, (byte)0x00}, |
| {(byte)0x00, (byte)0xa4, (byte)0x04, (byte)0x00, (byte)0x07, (byte)0xa0, (byte)0x00, |
| (byte)0x00, (byte)0x04, (byte)0x76, (byte)0x20, (byte)0x10, (byte)0x00}, |
| {(byte)0x80, (byte)0xe2, (byte)0x01, (byte)0x03, (byte)0x00}, |
| {(byte)0x00, (byte)0xa4, (byte)0x04, (byte)0x00, (byte)0x00}, |
| {(byte)0x00, (byte)0xa4, (byte)0x04, (byte)0x00, (byte)0x07, (byte)0xa0, (byte)0x00, |
| (byte)0x00, (byte)0x04, (byte)0x76, (byte)0x30, (byte)0x30, (byte)0x00}, |
| {(byte)0x80, (byte)0xb4, (byte)0x00, (byte)0x00, (byte)0x00}, |
| {(byte)0x00, (byte)0xa4, (byte)0x04, (byte)0x00, (byte)0x00}, |
| }; |
| |
| // NFC Execution Environment |
| // fields below are protected by this |
| private NativeNfcSecureElement mSecureElement; |
| private OpenSecureElement mOpenEe; // null when EE closed |
| private int mEeRoutingState; // contactless interface routing |
| |
| // fields below must be used only on the UI thread and therefore aren't synchronized |
| boolean mP2pStarted = false; |
| |
| // fields below are used in multiple threads and protected by synchronized(this) |
| final HashMap<Integer, Object> mObjectMap = new HashMap<Integer, Object>(); |
| HashSet<String> mSePackages = new HashSet<String>(); |
| int mScreenState; |
| boolean mIsNdefPushEnabled; |
| boolean mNfceeRouteEnabled; // current Device Host state of NFC-EE routing |
| boolean mNfcPollingEnabled; // current Device Host state of NFC-C polling |
| List<PackageInfo> mInstalledPackages; // cached version of installed packages |
| |
| // mState is protected by this, however it is only modified in onCreate() |
| // and the default AsyncTask thread so it is read unprotected from that |
| // thread |
| int mState; // one of NfcAdapter.STATE_ON, STATE_TURNING_ON, etc |
| |
| // fields below are final after onCreate() |
| Context mContext; |
| private DeviceHost mDeviceHost; |
| private SharedPreferences mPrefs; |
| private SharedPreferences.Editor mPrefsEditor; |
| private PowerManager.WakeLock mWakeLock; |
| int mStartSound; |
| int mEndSound; |
| int mErrorSound; |
| SoundPool mSoundPool; // playback synchronized on this |
| P2pLinkManager mP2pLinkManager; |
| TagService mNfcTagService; |
| NfcAdapterService mNfcAdapter; |
| NfcAdapterExtrasService mExtrasService; |
| boolean mIsAirplaneSensitive; |
| boolean mIsAirplaneToggleable; |
| NfceeAccessControl mNfceeAccessControl; |
| |
| private NfcDispatcher mNfcDispatcher; |
| private PowerManager mPowerManager; |
| private KeyguardManager mKeyguard; |
| |
| private static NfcService sService; |
| |
| public static void enforceAdminPerm(Context context) { |
| context.enforceCallingOrSelfPermission(ADMIN_PERM, ADMIN_PERM_ERROR); |
| } |
| |
| public void enforceNfceeAdminPerm(String pkg) { |
| if (pkg == null) { |
| throw new SecurityException("caller must pass a package name"); |
| } |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| if (!mNfceeAccessControl.check(Binder.getCallingUid(), pkg)) { |
| throw new SecurityException(NfceeAccessControl.NFCEE_ACCESS_PATH + |
| " denies NFCEE access to " + pkg); |
| } |
| } |
| |
| public static NfcService getInstance() { |
| return sService; |
| } |
| |
| @Override |
| public void onRemoteEndpointDiscovered(TagEndpoint tag) { |
| sendMessage(NfcService.MSG_NDEF_TAG, tag); |
| } |
| |
| /** |
| * Notifies transaction |
| */ |
| @Override |
| public void onCardEmulationDeselected() { |
| sendMessage(NfcService.MSG_TARGET_DESELECTED, null); |
| } |
| |
| /** |
| * Notifies transaction |
| */ |
| @Override |
| public void onCardEmulationAidSelected(byte[] aid) { |
| sendMessage(NfcService.MSG_CARD_EMULATION, aid); |
| } |
| |
| /** |
| * Notifies P2P Device detected, to activate LLCP link |
| */ |
| @Override |
| public void onLlcpLinkActivated(NfcDepEndpoint device) { |
| sendMessage(NfcService.MSG_LLCP_LINK_ACTIVATION, device); |
| } |
| |
| /** |
| * Notifies P2P Device detected, to activate LLCP link |
| */ |
| @Override |
| public void onLlcpLinkDeactivated(NfcDepEndpoint device) { |
| sendMessage(NfcService.MSG_LLCP_LINK_DEACTIVATED, device); |
| } |
| |
| @Override |
| public void onRemoteFieldActivated() { |
| sendMessage(NfcService.MSG_SE_FIELD_ACTIVATED, null); |
| } |
| |
| @Override |
| public void onRemoteFieldDeactivated() { |
| sendMessage(NfcService.MSG_SE_FIELD_DEACTIVATED, null); |
| } |
| |
| @Override |
| public void onSeApduReceived(byte[] apdu) { |
| sendMessage(NfcService.MSG_SE_APDU_RECEIVED, apdu); |
| } |
| |
| @Override |
| public void onSeEmvCardRemoval() { |
| sendMessage(NfcService.MSG_SE_EMV_CARD_REMOVAL, null); |
| } |
| |
| @Override |
| public void onSeMifareAccess(byte[] block) { |
| sendMessage(NfcService.MSG_SE_MIFARE_ACCESS, block); |
| } |
| |
| @Override |
| public void onCreate() { |
| super.onCreate(); |
| |
| mNfcTagService = new TagService(); |
| mNfcAdapter = new NfcAdapterService(); |
| mExtrasService = new NfcAdapterExtrasService(); |
| |
| Log.i(TAG, "Starting NFC service"); |
| |
| sService = this; |
| |
| mContext = this; |
| mDeviceHost = new NativeNfcManager(this, this); |
| |
| HandoverManager handoverManager = new HandoverManager(mContext); |
| mNfcDispatcher = new NfcDispatcher(this, handoverManager); |
| mP2pLinkManager = new P2pLinkManager(mContext, handoverManager); |
| |
| mSecureElement = new NativeNfcSecureElement(mContext); |
| mEeRoutingState = ROUTE_OFF; |
| |
| mNfceeAccessControl = new NfceeAccessControl(this); |
| |
| mPrefs = getSharedPreferences(PREF, Context.MODE_PRIVATE); |
| mPrefsEditor = mPrefs.edit(); |
| |
| mState = NfcAdapter.STATE_OFF; |
| mIsNdefPushEnabled = mPrefs.getBoolean(PREF_NDEF_PUSH_ON, NDEF_PUSH_ON_DEFAULT); |
| |
| mPowerManager = (PowerManager) getSystemService(Context.POWER_SERVICE); |
| |
| mWakeLock = mPowerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "NfcService"); |
| mKeyguard = (KeyguardManager) getSystemService(Context.KEYGUARD_SERVICE); |
| mScreenState = checkScreenState(); |
| |
| ServiceManager.addService(SERVICE_NAME, mNfcAdapter); |
| |
| IntentFilter filter = new IntentFilter(NativeNfcManager.INTERNAL_TARGET_DESELECTED_ACTION); |
| filter.addAction(Intent.ACTION_SCREEN_OFF); |
| filter.addAction(Intent.ACTION_SCREEN_ON); |
| filter.addAction(ACTION_MASTER_CLEAR_NOTIFICATION); |
| filter.addAction(Intent.ACTION_USER_PRESENT); |
| filter.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE); |
| filter.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE); |
| registerForAirplaneMode(filter); |
| registerReceiver(mReceiver, filter); |
| |
| filter = new IntentFilter(); |
| filter.addAction(Intent.ACTION_PACKAGE_ADDED); |
| filter.addAction(Intent.ACTION_PACKAGE_REMOVED); |
| filter.addDataScheme("package"); |
| |
| registerReceiver(mReceiver, filter); |
| |
| updatePackageCache(); |
| |
| new EnableDisableTask().execute(TASK_BOOT); // do blocking boot tasks |
| } |
| |
| void initSoundPool() { |
| synchronized(this) { |
| if (mSoundPool == null) { |
| mSoundPool = new SoundPool(1, AudioManager.STREAM_NOTIFICATION, 0); |
| mStartSound = mSoundPool.load(this, R.raw.start, 1); |
| mEndSound = mSoundPool.load(this, R.raw.end, 1); |
| mErrorSound = mSoundPool.load(this, R.raw.error, 1); |
| } |
| } |
| } |
| |
| void releaseSoundPool() { |
| synchronized(this) { |
| if (mSoundPool != null) { |
| mSoundPool.release(); |
| mSoundPool = null; |
| } |
| } |
| } |
| |
| void registerForAirplaneMode(IntentFilter filter) { |
| final ContentResolver resolver = mContext.getContentResolver(); |
| final String airplaneModeRadios = Settings.System.getString(resolver, |
| Settings.System.AIRPLANE_MODE_RADIOS); |
| final String toggleableRadios = Settings.System.getString(resolver, |
| Settings.System.AIRPLANE_MODE_TOGGLEABLE_RADIOS); |
| |
| mIsAirplaneSensitive = airplaneModeRadios == null ? true : |
| airplaneModeRadios.contains(Settings.System.RADIO_NFC); |
| mIsAirplaneToggleable = toggleableRadios == null ? false : |
| toggleableRadios.contains(Settings.System.RADIO_NFC); |
| |
| if (mIsAirplaneSensitive) { |
| filter.addAction(Intent.ACTION_AIRPLANE_MODE_CHANGED); |
| } |
| } |
| |
| void updatePackageCache() { |
| PackageManager pm = getPackageManager(); |
| List<PackageInfo> packages = pm.getInstalledPackages(0); |
| synchronized (this) { |
| mInstalledPackages = packages; |
| } |
| } |
| |
| int checkScreenState() { |
| if (!mPowerManager.isScreenOn()) { |
| return SCREEN_STATE_OFF; |
| } else if (mKeyguard.isKeyguardLocked()) { |
| return SCREEN_STATE_ON_LOCKED; |
| } else { |
| return SCREEN_STATE_ON_UNLOCKED; |
| } |
| } |
| |
| /** |
| * Manages tasks that involve turning on/off the NFC controller. |
| * |
| * <p>All work that might turn the NFC adapter on or off must be done |
| * through this task, to keep the handling of mState simple. |
| * In other words, mState is only modified in these tasks (and we |
| * don't need a lock to read it in these tasks). |
| * |
| * <p>These tasks are all done on the same AsyncTask background |
| * thread, so they are serialized. Each task may temporarily transition |
| * mState to STATE_TURNING_OFF or STATE_TURNING_ON, but must exit in |
| * either STATE_ON or STATE_OFF. This way each task can be guaranteed |
| * of starting in either STATE_OFF or STATE_ON, without needing to hold |
| * NfcService.this for the entire task. |
| * |
| * <p>AsyncTask's are also implicitly queued. This is useful for corner |
| * cases like turning airplane mode on while TASK_ENABLE is in progress. |
| * The TASK_DISABLE triggered by airplane mode will be correctly executed |
| * immediately after TASK_ENABLE is complete. This seems like the most sane |
| * way to deal with these situations. |
| * |
| * <p>{@link #TASK_ENABLE} enables the NFC adapter, without changing |
| * preferences |
| * <p>{@link #TASK_DISABLE} disables the NFC adapter, without changing |
| * preferences |
| * <p>{@link #TASK_BOOT} does first boot work and may enable NFC |
| * <p>{@link #TASK_EE_WIPE} wipes the Execution Environment, and in the |
| * process may temporarily enable the NFC adapter |
| */ |
| class EnableDisableTask extends AsyncTask<Integer, Void, Void> { |
| @Override |
| protected Void doInBackground(Integer... params) { |
| // Sanity check mState |
| switch (mState) { |
| case NfcAdapter.STATE_TURNING_OFF: |
| case NfcAdapter.STATE_TURNING_ON: |
| Log.e(TAG, "Processing EnableDisable task " + params[0] + " from bad state " + |
| mState); |
| return null; |
| } |
| |
| /* AsyncTask sets this thread to THREAD_PRIORITY_BACKGROUND, |
| * override with the default. THREAD_PRIORITY_BACKGROUND causes |
| * us to service software I2C too slow for firmware download |
| * with the NXP PN544. |
| * TODO: move this to the DAL I2C layer in libnfc-nxp, since this |
| * problem only occurs on I2C platforms using PN544 |
| */ |
| Process.setThreadPriority(Process.THREAD_PRIORITY_DEFAULT); |
| |
| switch (params[0].intValue()) { |
| case TASK_ENABLE: |
| enableInternal(); |
| break; |
| case TASK_DISABLE: |
| disableInternal(); |
| break; |
| case TASK_BOOT: |
| Log.d(TAG,"checking on firmware download"); |
| boolean airplaneOverride = mPrefs.getBoolean(PREF_AIRPLANE_OVERRIDE, false); |
| if (mPrefs.getBoolean(PREF_NFC_ON, NFC_ON_DEFAULT) && |
| (!mIsAirplaneSensitive || !isAirplaneModeOn() || airplaneOverride)) { |
| Log.d(TAG,"NFC is on. Doing normal stuff"); |
| enableInternal(); |
| } else { |
| Log.d(TAG,"NFC is off. Checking firmware version"); |
| mDeviceHost.checkFirmware(); |
| } |
| if (mPrefs.getBoolean(PREF_FIRST_BOOT, true)) { |
| Log.i(TAG, "First Boot"); |
| mPrefsEditor.putBoolean(PREF_FIRST_BOOT, false); |
| mPrefsEditor.apply(); |
| executeEeWipe(); |
| } |
| break; |
| case TASK_EE_WIPE: |
| executeEeWipe(); |
| break; |
| } |
| |
| // Restore default AsyncTask priority |
| Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND); |
| return null; |
| } |
| |
| /** |
| * Enable NFC adapter functions. |
| * Does not toggle preferences. |
| */ |
| boolean enableInternal() { |
| if (mState == NfcAdapter.STATE_ON) { |
| return true; |
| } |
| Log.i(TAG, "Enabling NFC"); |
| updateState(NfcAdapter.STATE_TURNING_ON); |
| |
| |
| if (!mDeviceHost.initialize()) { |
| Log.w(TAG, "Error enabling NFC"); |
| updateState(NfcAdapter.STATE_OFF); |
| return false; |
| } |
| |
| synchronized(NfcService.this) { |
| mObjectMap.clear(); |
| |
| mP2pLinkManager.enableDisable(mIsNdefPushEnabled, true); |
| updateState(NfcAdapter.STATE_ON); |
| } |
| |
| initSoundPool(); |
| |
| /* Start polling loop */ |
| |
| applyRouting(true); |
| return true; |
| } |
| |
| /** |
| * Disable all NFC adapter functions. |
| * Does not toggle preferences. |
| */ |
| boolean disableInternal() { |
| if (mState == NfcAdapter.STATE_OFF) { |
| return true; |
| } |
| Log.i(TAG, "Disabling NFC"); |
| updateState(NfcAdapter.STATE_TURNING_OFF); |
| |
| /* Sometimes mDeviceHost.deinitialize() hangs, use a watch-dog. |
| * Implemented with a new thread (instead of a Handler or AsyncTask), |
| * because the UI Thread and AsyncTask thread-pools can also get hung |
| * when the NFC controller stops responding */ |
| WatchDogThread watchDog = new WatchDogThread(); |
| watchDog.start(); |
| |
| mP2pLinkManager.enableDisable(false, false); |
| |
| synchronized (NfcService.this) { |
| if (mOpenEe != null) { |
| try { |
| _nfcEeClose(-1, mOpenEe.binder); |
| } catch (IOException e) { } |
| } |
| } |
| |
| // Stop watchdog if tag present |
| // A convenient way to stop the watchdog properly consists of |
| // disconnecting the tag. The polling loop shall be stopped before |
| // to avoid the tag being discovered again. |
| applyRouting(true); |
| maybeDisconnectTarget(); |
| |
| mNfcDispatcher.setForegroundDispatch(null, null, null); |
| |
| boolean result = mDeviceHost.deinitialize(); |
| if (DBG) Log.d(TAG, "mDeviceHost.deinitialize() = " + result); |
| |
| watchDog.cancel(); |
| |
| updateState(NfcAdapter.STATE_OFF); |
| |
| releaseSoundPool(); |
| |
| return result; |
| } |
| |
| void executeEeWipe() { |
| // TODO: read SE reset list from /system/etc |
| byte[][]apdus = EE_WIPE_APDUS; |
| |
| boolean tempEnable = mState == NfcAdapter.STATE_OFF; |
| if (tempEnable) { |
| if (!enableInternal()) { |
| Log.w(TAG, "Could not enable NFC to wipe NFC-EE"); |
| return; |
| } |
| } |
| Log.i(TAG, "Executing SE wipe"); |
| int handle = mSecureElement.doOpenSecureElementConnection(); |
| if (handle == 0) { |
| Log.w(TAG, "Could not open the secure element"); |
| if (tempEnable) { |
| disableInternal(); |
| } |
| return; |
| } |
| |
| mDeviceHost.setTimeout(TagTechnology.ISO_DEP, 10000); |
| |
| for (byte[] cmd : apdus) { |
| byte[] resp = mSecureElement.doTransceive(handle, cmd); |
| if (resp == null) { |
| Log.w(TAG, "Transceive failed, could not wipe NFC-EE"); |
| break; |
| } |
| } |
| |
| mDeviceHost.resetTimeouts(); |
| mSecureElement.doDisconnect(handle); |
| |
| if (tempEnable) { |
| disableInternal(); |
| } |
| } |
| |
| void updateState(int newState) { |
| synchronized (NfcService.this) { |
| if (newState == mState) { |
| return; |
| } |
| mState = newState; |
| Intent intent = new Intent(NfcAdapter.ACTION_ADAPTER_STATE_CHANGED); |
| intent.setFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT); |
| intent.putExtra(NfcAdapter.EXTRA_ADAPTER_STATE, mState); |
| mContext.sendBroadcast(intent); |
| } |
| } |
| } |
| |
| void saveNfcOnSetting(boolean on) { |
| synchronized (NfcService.this) { |
| mPrefsEditor.putBoolean(PREF_NFC_ON, on); |
| mPrefsEditor.apply(); |
| } |
| } |
| |
| public void playSound(int sound) { |
| synchronized (this) { |
| if (mSoundPool == null) { |
| Log.w(TAG, "Not playing sound when NFC is disabled"); |
| return; |
| } |
| switch (sound) { |
| case SOUND_START: |
| mSoundPool.play(mStartSound, 1.0f, 1.0f, 0, 0, 1.0f); |
| break; |
| case SOUND_END: |
| mSoundPool.play(mEndSound, 1.0f, 1.0f, 0, 0, 1.0f); |
| break; |
| case SOUND_ERROR: |
| mSoundPool.play(mErrorSound, 1.0f, 1.0f, 0, 0, 1.0f); |
| break; |
| } |
| } |
| } |
| |
| @Override |
| public void onTerminate() { |
| super.onTerminate(); |
| // NFC application is persistent, it should not be destroyed by framework |
| Log.wtf(TAG, "NFC service is under attack!"); |
| } |
| |
| final class NfcAdapterService extends INfcAdapter.Stub { |
| @Override |
| public boolean enable() throws RemoteException { |
| NfcService.enforceAdminPerm(mContext); |
| |
| saveNfcOnSetting(true); |
| |
| if (mIsAirplaneSensitive && isAirplaneModeOn()) { |
| if (!mIsAirplaneToggleable) { |
| Log.i(TAG, "denying enable() request (airplane mode)"); |
| return false; |
| } |
| // Make sure the override survives a reboot |
| mPrefsEditor.putBoolean(PREF_AIRPLANE_OVERRIDE, true); |
| mPrefsEditor.apply(); |
| } |
| new EnableDisableTask().execute(TASK_ENABLE); |
| |
| return true; |
| } |
| |
| @Override |
| public boolean disable(boolean saveState) throws RemoteException { |
| NfcService.enforceAdminPerm(mContext); |
| |
| if (saveState) { |
| saveNfcOnSetting(false); |
| } |
| |
| new EnableDisableTask().execute(TASK_DISABLE); |
| |
| return true; |
| } |
| |
| @Override |
| public boolean isNdefPushEnabled() throws RemoteException { |
| synchronized (NfcService.this) { |
| return mState == NfcAdapter.STATE_ON && mIsNdefPushEnabled; |
| } |
| } |
| |
| @Override |
| public boolean enableNdefPush() throws RemoteException { |
| NfcService.enforceAdminPerm(mContext); |
| synchronized(NfcService.this) { |
| if (mIsNdefPushEnabled) { |
| return true; |
| } |
| Log.i(TAG, "enabling NDEF Push"); |
| mPrefsEditor.putBoolean(PREF_NDEF_PUSH_ON, true); |
| mPrefsEditor.apply(); |
| mIsNdefPushEnabled = true; |
| if (isNfcEnabled()) { |
| mP2pLinkManager.enableDisable(true, true); |
| } |
| } |
| return true; |
| } |
| |
| @Override |
| public boolean disableNdefPush() throws RemoteException { |
| NfcService.enforceAdminPerm(mContext); |
| synchronized(NfcService.this) { |
| if (!mIsNdefPushEnabled) { |
| return true; |
| } |
| Log.i(TAG, "disabling NDEF Push"); |
| mPrefsEditor.putBoolean(PREF_NDEF_PUSH_ON, false); |
| mPrefsEditor.apply(); |
| mIsNdefPushEnabled = false; |
| if (isNfcEnabled()) { |
| mP2pLinkManager.enableDisable(false, true); |
| } |
| } |
| return true; |
| } |
| |
| @Override |
| public void setForegroundDispatch(PendingIntent intent, |
| IntentFilter[] filters, TechListParcel techListsParcel) { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| // Short-cut the disable path |
| if (intent == null && filters == null && techListsParcel == null) { |
| mNfcDispatcher.setForegroundDispatch(null, null, null); |
| return; |
| } |
| |
| // Validate the IntentFilters |
| if (filters != null) { |
| if (filters.length == 0) { |
| filters = null; |
| } else { |
| for (IntentFilter filter : filters) { |
| if (filter == null) { |
| throw new IllegalArgumentException("null IntentFilter"); |
| } |
| } |
| } |
| } |
| |
| // Validate the tech lists |
| String[][] techLists = null; |
| if (techListsParcel != null) { |
| techLists = techListsParcel.getTechLists(); |
| } |
| |
| mNfcDispatcher.setForegroundDispatch(intent, filters, techLists); |
| } |
| |
| @Override |
| public void setNdefPushCallback(INdefPushCallback callback) { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| mP2pLinkManager.setNdefCallback(callback); |
| } |
| |
| @Override |
| public INfcTag getNfcTagInterface() throws RemoteException { |
| return mNfcTagService; |
| } |
| |
| @Override |
| public INfcAdapterExtras getNfcAdapterExtrasInterface(String pkg) { |
| NfcService.this.enforceNfceeAdminPerm(pkg); |
| return mExtrasService; |
| } |
| |
| @Override |
| public int getState() throws RemoteException { |
| synchronized (NfcService.this) { |
| return mState; |
| } |
| } |
| |
| @Override |
| protected void dump(FileDescriptor fd, PrintWriter pw, String[] args) { |
| NfcService.this.dump(fd, pw, args); |
| } |
| |
| @Override |
| public void dispatch(Tag tag) throws RemoteException { |
| enforceAdminPerm(mContext); |
| mNfcDispatcher.dispatchTag(tag); |
| } |
| |
| @Override |
| public void setP2pModes(int initiatorModes, int targetModes) throws RemoteException { |
| enforceAdminPerm(mContext); |
| |
| mDeviceHost.setP2pInitiatorModes(initiatorModes); |
| mDeviceHost.setP2pTargetModes(targetModes); |
| mDeviceHost.disableDiscovery(); |
| mDeviceHost.enableDiscovery(); |
| } |
| } |
| |
| final class TagService extends INfcTag.Stub { |
| @Override |
| public int close(int nativeHandle) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| TagEndpoint tag = null; |
| |
| if (!isNfcEnabled()) { |
| return ErrorCodes.ERROR_NOT_INITIALIZED; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag != null) { |
| /* Remove the device from the hmap */ |
| unregisterObject(nativeHandle); |
| tag.disconnect(); |
| return ErrorCodes.SUCCESS; |
| } |
| /* Restart polling loop for notification */ |
| applyRouting(true); |
| return ErrorCodes.ERROR_DISCONNECT; |
| } |
| |
| @Override |
| public int connect(int nativeHandle, int technology) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| TagEndpoint tag = null; |
| |
| if (!isNfcEnabled()) { |
| return ErrorCodes.ERROR_NOT_INITIALIZED; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag == null) { |
| return ErrorCodes.ERROR_DISCONNECT; |
| } |
| |
| if (!tag.isPresent()) { |
| return ErrorCodes.ERROR_DISCONNECT; |
| } |
| |
| if (technology == TagTechnology.NFC_B) { |
| return ErrorCodes.ERROR_NOT_SUPPORTED; |
| } |
| |
| // Note that on most tags, all technologies are behind a single |
| // handle. This means that the connect at the lower levels |
| // will do nothing, as the tag is already connected to that handle. |
| if (tag.connect(technology)) { |
| return ErrorCodes.SUCCESS; |
| } else { |
| return ErrorCodes.ERROR_DISCONNECT; |
| } |
| } |
| |
| @Override |
| public int reconnect(int nativeHandle) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| TagEndpoint tag = null; |
| |
| // Check if NFC is enabled |
| if (!isNfcEnabled()) { |
| return ErrorCodes.ERROR_NOT_INITIALIZED; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag != null) { |
| if (tag.reconnect()) { |
| return ErrorCodes.SUCCESS; |
| } else { |
| return ErrorCodes.ERROR_DISCONNECT; |
| } |
| } |
| return ErrorCodes.ERROR_DISCONNECT; |
| } |
| |
| @Override |
| public int[] getTechList(int nativeHandle) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| // Check if NFC is enabled |
| if (!isNfcEnabled()) { |
| return null; |
| } |
| |
| /* find the tag in the hmap */ |
| TagEndpoint tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag != null) { |
| return tag.getTechList(); |
| } |
| return null; |
| } |
| |
| @Override |
| public boolean isPresent(int nativeHandle) throws RemoteException { |
| TagEndpoint tag = null; |
| |
| // Check if NFC is enabled |
| if (!isNfcEnabled()) { |
| return false; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag == null) { |
| return false; |
| } |
| |
| return tag.isPresent(); |
| } |
| |
| @Override |
| public boolean isNdef(int nativeHandle) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| TagEndpoint tag = null; |
| |
| // Check if NFC is enabled |
| if (!isNfcEnabled()) { |
| return false; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| int[] ndefInfo = new int[2]; |
| if (tag == null) { |
| return false; |
| } |
| return tag.checkNdef(ndefInfo); |
| } |
| |
| @Override |
| public TransceiveResult transceive(int nativeHandle, byte[] data, boolean raw) |
| throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| TagEndpoint tag = null; |
| byte[] response; |
| |
| // Check if NFC is enabled |
| if (!isNfcEnabled()) { |
| return null; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag != null) { |
| // Check if length is within limits |
| if (data.length > getMaxTransceiveLength(tag.getConnectedTechnology())) { |
| return new TransceiveResult(TransceiveResult.RESULT_EXCEEDED_LENGTH, null); |
| } |
| int[] targetLost = new int[1]; |
| response = tag.transceive(data, raw, targetLost); |
| int result; |
| if (response != null) { |
| result = TransceiveResult.RESULT_SUCCESS; |
| } else if (targetLost[0] == 1) { |
| result = TransceiveResult.RESULT_TAGLOST; |
| } else { |
| result = TransceiveResult.RESULT_FAILURE; |
| } |
| return new TransceiveResult(result, response); |
| } |
| return null; |
| } |
| |
| @Override |
| public NdefMessage ndefRead(int nativeHandle) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| TagEndpoint tag; |
| |
| // Check if NFC is enabled |
| if (!isNfcEnabled()) { |
| return null; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag != null) { |
| byte[] buf = tag.readNdef(); |
| if (buf == null) { |
| return null; |
| } |
| |
| /* Create an NdefMessage */ |
| try { |
| return new NdefMessage(buf); |
| } catch (FormatException e) { |
| return null; |
| } |
| } |
| return null; |
| } |
| |
| @Override |
| public int ndefWrite(int nativeHandle, NdefMessage msg) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| TagEndpoint tag; |
| |
| // Check if NFC is enabled |
| if (!isNfcEnabled()) { |
| return ErrorCodes.ERROR_NOT_INITIALIZED; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag == null) { |
| return ErrorCodes.ERROR_IO; |
| } |
| |
| if (msg == null) return ErrorCodes.ERROR_INVALID_PARAM; |
| |
| if (tag.writeNdef(msg.toByteArray())) { |
| return ErrorCodes.SUCCESS; |
| } else { |
| return ErrorCodes.ERROR_IO; |
| } |
| |
| } |
| |
| @Override |
| public boolean ndefIsWritable(int nativeHandle) throws RemoteException { |
| throw new UnsupportedOperationException(); |
| } |
| |
| @Override |
| public int ndefMakeReadOnly(int nativeHandle) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| TagEndpoint tag; |
| |
| // Check if NFC is enabled |
| if (!isNfcEnabled()) { |
| return ErrorCodes.ERROR_NOT_INITIALIZED; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag == null) { |
| return ErrorCodes.ERROR_IO; |
| } |
| |
| if (tag.makeReadOnly()) { |
| return ErrorCodes.SUCCESS; |
| } else { |
| return ErrorCodes.ERROR_IO; |
| } |
| } |
| |
| @Override |
| public int formatNdef(int nativeHandle, byte[] key) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| TagEndpoint tag; |
| |
| // Check if NFC is enabled |
| if (!isNfcEnabled()) { |
| return ErrorCodes.ERROR_NOT_INITIALIZED; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag == null) { |
| return ErrorCodes.ERROR_IO; |
| } |
| |
| if (tag.formatNdef(key)) { |
| return ErrorCodes.SUCCESS; |
| } else { |
| return ErrorCodes.ERROR_IO; |
| } |
| } |
| |
| @Override |
| public Tag rediscover(int nativeHandle) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| TagEndpoint tag = null; |
| |
| // Check if NFC is enabled |
| if (!isNfcEnabled()) { |
| return null; |
| } |
| |
| /* find the tag in the hmap */ |
| tag = (TagEndpoint) findObject(nativeHandle); |
| if (tag != null) { |
| // For now the prime usecase for rediscover() is to be able |
| // to access the NDEF technology after formatting without |
| // having to remove the tag from the field, or similar |
| // to have access to NdefFormatable in case low-level commands |
| // were used to remove NDEF. So instead of doing a full stack |
| // rediscover (which is poorly supported at the moment anyway), |
| // we simply remove these two technologies and detect them |
| // again. |
| tag.removeTechnology(TagTechnology.NDEF); |
| tag.removeTechnology(TagTechnology.NDEF_FORMATABLE); |
| tag.findAndReadNdef(); |
| // Build a new Tag object to return |
| Tag newTag = new Tag(tag.getUid(), tag.getTechList(), |
| tag.getTechExtras(), tag.getHandle(), this); |
| return newTag; |
| } |
| return null; |
| } |
| |
| @Override |
| public int setTimeout(int tech, int timeout) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| boolean success = mDeviceHost.setTimeout(tech, timeout); |
| if (success) { |
| return ErrorCodes.SUCCESS; |
| } else { |
| return ErrorCodes.ERROR_INVALID_PARAM; |
| } |
| } |
| |
| @Override |
| public int getTimeout(int tech) throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| return mDeviceHost.getTimeout(tech); |
| } |
| |
| @Override |
| public void resetTimeouts() throws RemoteException { |
| mContext.enforceCallingOrSelfPermission(NFC_PERM, NFC_PERM_ERROR); |
| |
| mDeviceHost.resetTimeouts(); |
| } |
| |
| @Override |
| public boolean canMakeReadOnly(int ndefType) throws RemoteException { |
| return mDeviceHost.canMakeReadOnly(ndefType); |
| } |
| |
| @Override |
| public int getMaxTransceiveLength(int tech) throws RemoteException { |
| return mDeviceHost.getMaxTransceiveLength(tech); |
| } |
| |
| @Override |
| public boolean getExtendedLengthApdusSupported() throws RemoteException { |
| return mDeviceHost.getExtendedLengthApdusSupported(); |
| } |
| } |
| |
| void _nfcEeClose(int callingPid, IBinder binder) throws IOException { |
| // Blocks until a pending open() or transceive() times out. |
| //TODO: This is incorrect behavior - the close should interrupt pending |
| // operations. However this is not supported by current hardware. |
| |
| synchronized (NfcService.this) { |
| if (!isNfcEnabledOrShuttingDown()) { |
| throw new IOException("NFC adapter is disabled"); |
| } |
| if (mOpenEe == null) { |
| throw new IOException("NFC EE closed"); |
| } |
| if (callingPid != -1 && callingPid != mOpenEe.pid) { |
| throw new SecurityException("Wrong PID"); |
| } |
| if (mOpenEe.binder != binder) { |
| throw new SecurityException("Wrong binder handle"); |
| } |
| |
| binder.unlinkToDeath(mOpenEe, 0); |
| mDeviceHost.resetTimeouts(); |
| mSecureElement.doDisconnect(mOpenEe.handle); |
| mOpenEe = null; |
| |
| applyRouting(true); |
| } |
| } |
| |
| final class NfcAdapterExtrasService extends INfcAdapterExtras.Stub { |
| private Bundle writeNoException() { |
| Bundle p = new Bundle(); |
| p.putInt("e", 0); |
| return p; |
| } |
| private Bundle writeIoException(IOException e) { |
| Bundle p = new Bundle(); |
| p.putInt("e", -1); |
| p.putString("m", e.getMessage()); |
| return p; |
| } |
| |
| @Override |
| public Bundle open(String pkg, IBinder b) throws RemoteException { |
| NfcService.this.enforceNfceeAdminPerm(pkg); |
| |
| Bundle result; |
| try { |
| _open(b); |
| result = writeNoException(); |
| } catch (IOException e) { |
| result = writeIoException(e); |
| } |
| return result; |
| } |
| |
| private void _open(IBinder b) throws IOException { |
| synchronized(NfcService.this) { |
| if (!isNfcEnabled()) { |
| throw new IOException("NFC adapter is disabled"); |
| } |
| if (mOpenEe != null) { |
| throw new IOException("NFC EE already open"); |
| } |
| |
| int handle = mSecureElement.doOpenSecureElementConnection(); |
| if (handle == 0) { |
| throw new IOException("NFC EE failed to open"); |
| } |
| mDeviceHost.setTimeout(TagTechnology.ISO_DEP, 10000); |
| |
| mOpenEe = new OpenSecureElement(getCallingPid(), handle, b); |
| try { |
| b.linkToDeath(mOpenEe, 0); |
| } catch (RemoteException e) { |
| mOpenEe.binderDied(); |
| } |
| |
| // Add the calling package to the list of packages that have accessed |
| // the secure element. |
| for (String packageName : getPackageManager().getPackagesForUid(getCallingUid())) { |
| mSePackages.add(packageName); |
| } |
| } |
| } |
| |
| @Override |
| public Bundle close(String pkg, IBinder binder) throws RemoteException { |
| NfcService.this.enforceNfceeAdminPerm(pkg); |
| |
| Bundle result; |
| try { |
| _nfcEeClose(getCallingPid(), binder); |
| result = writeNoException(); |
| } catch (IOException e) { |
| result = writeIoException(e); |
| } |
| return result; |
| } |
| |
| @Override |
| public Bundle transceive(String pkg, byte[] in) throws RemoteException { |
| NfcService.this.enforceNfceeAdminPerm(pkg); |
| |
| Bundle result; |
| byte[] out; |
| try { |
| out = _transceive(in); |
| result = writeNoException(); |
| result.putByteArray("out", out); |
| } catch (IOException e) { |
| result = writeIoException(e); |
| } |
| return result; |
| } |
| |
| private byte[] _transceive(byte[] data) throws IOException { |
| synchronized(NfcService.this) { |
| if (!isNfcEnabled()) { |
| throw new IOException("NFC is not enabled"); |
| } |
| if (mOpenEe == null) { |
| throw new IOException("NFC EE is not open"); |
| } |
| if (getCallingPid() != mOpenEe.pid) { |
| throw new SecurityException("Wrong PID"); |
| } |
| } |
| |
| return mSecureElement.doTransceive(mOpenEe.handle, data); |
| } |
| |
| @Override |
| public int getCardEmulationRoute(String pkg) throws RemoteException { |
| NfcService.this.enforceNfceeAdminPerm(pkg); |
| return mEeRoutingState; |
| } |
| |
| @Override |
| public void setCardEmulationRoute(String pkg, int route) throws RemoteException { |
| NfcService.this.enforceNfceeAdminPerm(pkg); |
| mEeRoutingState = route; |
| applyRouting(true); |
| } |
| |
| @Override |
| public void authenticate(String pkg, byte[] token) throws RemoteException { |
| NfcService.this.enforceNfceeAdminPerm(pkg); |
| } |
| } |
| |
| /** resources kept while secure element is open */ |
| private class OpenSecureElement implements IBinder.DeathRecipient { |
| public int pid; // pid that opened SE |
| // binder handle used for DeathReceipient. Must keep |
| // a reference to this, otherwise it can get GC'd and |
| // the binder stub code might create a different BinderProxy |
| // for the same remote IBinder, causing mismatched |
| // link()/unlink() |
| public IBinder binder; |
| public int handle; // low-level handle |
| public OpenSecureElement(int pid, int handle, IBinder binder) { |
| this.pid = pid; |
| this.handle = handle; |
| this.binder = binder; |
| } |
| @Override |
| public void binderDied() { |
| synchronized (NfcService.this) { |
| Log.i(TAG, "Tracked app " + pid + " died"); |
| pid = -1; |
| try { |
| _nfcEeClose(-1, binder); |
| } catch (IOException e) { /* already closed */ } |
| } |
| } |
| @Override |
| public String toString() { |
| return new StringBuilder('@').append(Integer.toHexString(hashCode())).append("[pid=") |
| .append(pid).append(" handle=").append(handle).append("]").toString(); |
| } |
| } |
| |
| boolean isNfcEnabledOrShuttingDown() { |
| synchronized (this) { |
| return (mState == NfcAdapter.STATE_ON || mState == NfcAdapter.STATE_TURNING_OFF); |
| } |
| } |
| |
| boolean isNfcEnabled() { |
| synchronized (this) { |
| return mState == NfcAdapter.STATE_ON; |
| } |
| } |
| |
| class WatchDogThread extends Thread { |
| boolean mWatchDogCanceled = false; |
| @Override |
| public void run() { |
| boolean slept = false; |
| while (!slept) { |
| try { |
| Thread.sleep(10000); |
| slept = true; |
| } catch (InterruptedException e) { } |
| } |
| synchronized (this) { |
| if (!mWatchDogCanceled) { |
| // Trigger watch-dog |
| Log.e(TAG, "Watch dog triggered"); |
| mDeviceHost.doAbort(); |
| } |
| } |
| } |
| public synchronized void cancel() { |
| mWatchDogCanceled = true; |
| } |
| } |
| |
| /** |
| * Read mScreenState and apply NFC-C polling and NFC-EE routing |
| */ |
| void applyRouting(boolean force) { |
| synchronized (this) { |
| if (!isNfcEnabledOrShuttingDown() || mOpenEe != null) { |
| // PN544 cannot be reconfigured while EE is open |
| return; |
| } |
| WatchDogThread watchDog = new WatchDogThread(); |
| |
| try { |
| watchDog.start(); |
| |
| if (PN544_QUIRK_DISCONNECT_BEFORE_RECONFIGURE && mScreenState == SCREEN_STATE_OFF) { |
| /* TODO undo this after the LLCP stack is fixed. |
| * Use a different sequence when turning the screen off to |
| * workaround race conditions in pn544 libnfc. The race occurs |
| * when we change routing while there is a P2P target connect. |
| * The async LLCP callback will crash since the routing code |
| * is overwriting globals it relies on. |
| */ |
| if (POLLING_MODE > SCREEN_STATE_OFF) { |
| if (force || mNfcPollingEnabled) { |
| Log.d(TAG, "NFC-C OFF, disconnect"); |
| mNfcPollingEnabled = false; |
| mDeviceHost.disableDiscovery(); |
| maybeDisconnectTarget(); |
| } |
| } |
| if (mEeRoutingState == ROUTE_ON_WHEN_SCREEN_ON) { |
| if (force || mNfceeRouteEnabled) { |
| Log.d(TAG, "NFC-EE OFF"); |
| mNfceeRouteEnabled = false; |
| mDeviceHost.doDeselectSecureElement(); |
| } |
| } |
| return; |
| } |
| |
| // configure NFC-EE routing |
| if (mScreenState >= SCREEN_STATE_ON_LOCKED && |
| mEeRoutingState == ROUTE_ON_WHEN_SCREEN_ON) { |
| if (force || !mNfceeRouteEnabled) { |
| Log.d(TAG, "NFC-EE ON"); |
| mNfceeRouteEnabled = true; |
| mDeviceHost.doSelectSecureElement(); |
| } |
| } else { |
| if (force || mNfceeRouteEnabled) { |
| Log.d(TAG, "NFC-EE OFF"); |
| mNfceeRouteEnabled = false; |
| mDeviceHost.doDeselectSecureElement(); |
| } |
| } |
| |
| // configure NFC-C polling |
| if (mScreenState >= POLLING_MODE) { |
| if (force || !mNfcPollingEnabled) { |
| Log.d(TAG, "NFC-C ON"); |
| mNfcPollingEnabled = true; |
| mDeviceHost.enableDiscovery(); |
| } |
| } else { |
| if (force || mNfcPollingEnabled) { |
| Log.d(TAG, "NFC-C OFF"); |
| mNfcPollingEnabled = false; |
| mDeviceHost.disableDiscovery(); |
| } |
| } |
| } finally { |
| watchDog.cancel(); |
| } |
| } |
| } |
| |
| /** Disconnect any target if present */ |
| void maybeDisconnectTarget() { |
| if (!isNfcEnabledOrShuttingDown()) { |
| return; |
| } |
| Object[] objectsToDisconnect; |
| synchronized (this) { |
| Object[] objectValues = mObjectMap.values().toArray(); |
| // Copy the array before we clear mObjectMap, |
| // just in case the HashMap values are backed by the same array |
| objectsToDisconnect = Arrays.copyOf(objectValues, objectValues.length); |
| mObjectMap.clear(); |
| } |
| for (Object o : objectsToDisconnect) { |
| if (DBG) Log.d(TAG, "disconnecting " + o.getClass().getName()); |
| if (o instanceof TagEndpoint) { |
| // Disconnect from tags |
| TagEndpoint tag = (TagEndpoint) o; |
| tag.disconnect(); |
| } else if (o instanceof NfcDepEndpoint) { |
| // Disconnect from P2P devices |
| NfcDepEndpoint device = (NfcDepEndpoint) o; |
| if (device.getMode() == NfcDepEndpoint.MODE_P2P_TARGET) { |
| // Remote peer is target, request disconnection |
| device.disconnect(); |
| } else { |
| // Remote peer is initiator, we cannot disconnect |
| // Just wait for field removal |
| } |
| } |
| } |
| } |
| |
| Object findObject(int key) { |
| synchronized (this) { |
| Object device = mObjectMap.get(key); |
| if (device == null) { |
| Log.w(TAG, "Handle not found"); |
| } |
| return device; |
| } |
| } |
| |
| void registerTagObject(TagEndpoint tag) { |
| synchronized (this) { |
| mObjectMap.put(tag.getHandle(), tag); |
| } |
| } |
| |
| void unregisterObject(int handle) { |
| synchronized (this) { |
| mObjectMap.remove(handle); |
| } |
| } |
| |
| /** For use by code in this process */ |
| public LlcpSocket createLlcpSocket(int sap, int miu, int rw, int linearBufferLength) |
| throws LlcpException { |
| return mDeviceHost.createLlcpSocket(sap, miu, rw, linearBufferLength); |
| } |
| |
| /** For use by code in this process */ |
| public LlcpConnectionlessSocket createLlcpConnectionLessSocket(int sap, String sn) |
| throws LlcpException { |
| return mDeviceHost.createLlcpConnectionlessSocket(sap, sn); |
| } |
| |
| /** For use by code in this process */ |
| public LlcpServerSocket createLlcpServerSocket(int sap, String sn, int miu, int rw, |
| int linearBufferLength) throws LlcpException { |
| return mDeviceHost.createLlcpServerSocket(sap, sn, miu, rw, linearBufferLength); |
| } |
| |
| public void sendMockNdefTag(NdefMessage msg) { |
| sendMessage(MSG_MOCK_NDEF, msg); |
| } |
| |
| void sendMessage(int what, Object obj) { |
| Message msg = mHandler.obtainMessage(); |
| msg.what = what; |
| msg.obj = obj; |
| mHandler.sendMessage(msg); |
| } |
| |
| final class NfcServiceHandler extends Handler { |
| @Override |
| public void handleMessage(Message msg) { |
| switch (msg.what) { |
| case MSG_MOCK_NDEF: { |
| NdefMessage ndefMsg = (NdefMessage) msg.obj; |
| Bundle extras = new Bundle(); |
| extras.putParcelable(Ndef.EXTRA_NDEF_MSG, ndefMsg); |
| extras.putInt(Ndef.EXTRA_NDEF_MAXLENGTH, 0); |
| extras.putInt(Ndef.EXTRA_NDEF_CARDSTATE, Ndef.NDEF_MODE_READ_ONLY); |
| extras.putInt(Ndef.EXTRA_NDEF_TYPE, Ndef.TYPE_OTHER); |
| Tag tag = Tag.createMockTag(new byte[] { 0x00 }, |
| new int[] { TagTechnology.NDEF }, |
| new Bundle[] { extras }); |
| Log.d(TAG, "mock NDEF tag, starting corresponding activity"); |
| Log.d(TAG, tag.toString()); |
| boolean delivered = mNfcDispatcher.dispatchTag(tag); |
| if (delivered) { |
| playSound(SOUND_END); |
| } else { |
| playSound(SOUND_ERROR); |
| } |
| break; |
| } |
| |
| case MSG_NDEF_TAG: |
| if (DBG) Log.d(TAG, "Tag detected, notifying applications"); |
| TagEndpoint tag = (TagEndpoint) msg.obj; |
| playSound(SOUND_START); |
| NdefMessage ndefMsg = tag.findAndReadNdef(); |
| |
| if (ndefMsg != null) { |
| tag.startPresenceChecking(); |
| dispatchTagEndpoint(tag); |
| } else { |
| if (tag.reconnect()) { |
| tag.startPresenceChecking(); |
| dispatchTagEndpoint(tag); |
| } else { |
| tag.disconnect(); |
| playSound(SOUND_ERROR); |
| } |
| } |
| break; |
| |
| case MSG_CARD_EMULATION: |
| if (DBG) Log.d(TAG, "Card Emulation message"); |
| byte[] aid = (byte[]) msg.obj; |
| /* Send broadcast */ |
| Intent aidIntent = new Intent(); |
| aidIntent.setAction(ACTION_AID_SELECTED); |
| aidIntent.putExtra(EXTRA_AID, aid); |
| if (DBG) Log.d(TAG, "Broadcasting " + ACTION_AID_SELECTED); |
| sendSeBroadcast(aidIntent); |
| break; |
| |
| case MSG_SE_EMV_CARD_REMOVAL: |
| if (DBG) Log.d(TAG, "Card Removal message"); |
| /* Send broadcast */ |
| Intent cardRemovalIntent = new Intent(); |
| cardRemovalIntent.setAction(ACTION_EMV_CARD_REMOVAL); |
| if (DBG) Log.d(TAG, "Broadcasting " + ACTION_EMV_CARD_REMOVAL); |
| sendSeBroadcast(cardRemovalIntent); |
| break; |
| |
| case MSG_SE_APDU_RECEIVED: |
| if (DBG) Log.d(TAG, "APDU Received message"); |
| byte[] apduBytes = (byte[]) msg.obj; |
| /* Send broadcast */ |
| Intent apduReceivedIntent = new Intent(); |
| apduReceivedIntent.setAction(ACTION_APDU_RECEIVED); |
| if (apduBytes != null && apduBytes.length > 0) { |
| apduReceivedIntent.putExtra(EXTRA_APDU_BYTES, apduBytes); |
| } |
| if (DBG) Log.d(TAG, "Broadcasting " + ACTION_APDU_RECEIVED); |
| sendSeBroadcast(apduReceivedIntent); |
| break; |
| |
| case MSG_SE_MIFARE_ACCESS: |
| if (DBG) Log.d(TAG, "MIFARE access message"); |
| /* Send broadcast */ |
| byte[] mifareCmd = (byte[]) msg.obj; |
| Intent mifareAccessIntent = new Intent(); |
| mifareAccessIntent.setAction(ACTION_MIFARE_ACCESS_DETECTED); |
| if (mifareCmd != null && mifareCmd.length > 1) { |
| int mifareBlock = mifareCmd[1] & 0xff; |
| if (DBG) Log.d(TAG, "Mifare Block=" + mifareBlock); |
| mifareAccessIntent.putExtra(EXTRA_MIFARE_BLOCK, mifareBlock); |
| } |
| if (DBG) Log.d(TAG, "Broadcasting " + ACTION_MIFARE_ACCESS_DETECTED); |
| sendSeBroadcast(mifareAccessIntent); |
| break; |
| |
| case MSG_LLCP_LINK_ACTIVATION: |
| llcpActivated((NfcDepEndpoint) msg.obj); |
| break; |
| |
| case MSG_LLCP_LINK_DEACTIVATED: |
| NfcDepEndpoint device = (NfcDepEndpoint) msg.obj; |
| boolean needsDisconnect = false; |
| |
| Log.d(TAG, "LLCP Link Deactivated message. Restart polling loop."); |
| synchronized (NfcService.this) { |
| /* Check if the device has been already unregistered */ |
| if (mObjectMap.remove(device.getHandle()) != null) { |
| /* Disconnect if we are initiator */ |
| if (device.getMode() == NfcDepEndpoint.MODE_P2P_TARGET) { |
| if (DBG) Log.d(TAG, "disconnecting from target"); |
| needsDisconnect = true; |
| } else { |
| if (DBG) Log.d(TAG, "not disconnecting from initiator"); |
| } |
| } |
| } |
| if (needsDisconnect) { |
| device.disconnect(); // restarts polling loop |
| } |
| |
| mP2pLinkManager.onLlcpDeactivated(); |
| break; |
| |
| case MSG_TARGET_DESELECTED: |
| /* Broadcast Intent Target Deselected */ |
| if (DBG) Log.d(TAG, "Target Deselected"); |
| Intent intent = new Intent(); |
| intent.setAction(NativeNfcManager.INTERNAL_TARGET_DESELECTED_ACTION); |
| if (DBG) Log.d(TAG, "Broadcasting Intent"); |
| mContext.sendOrderedBroadcast(intent, NFC_PERM); |
| break; |
| |
| case MSG_SE_FIELD_ACTIVATED: { |
| if (DBG) Log.d(TAG, "SE FIELD ACTIVATED"); |
| Intent eventFieldOnIntent = new Intent(); |
| eventFieldOnIntent.setAction(ACTION_RF_FIELD_ON_DETECTED); |
| sendSeBroadcast(eventFieldOnIntent); |
| break; |
| } |
| |
| case MSG_SE_FIELD_DEACTIVATED: { |
| if (DBG) Log.d(TAG, "SE FIELD DEACTIVATED"); |
| Intent eventFieldOffIntent = new Intent(); |
| eventFieldOffIntent.setAction(ACTION_RF_FIELD_OFF_DETECTED); |
| sendSeBroadcast(eventFieldOffIntent); |
| break; |
| } |
| |
| default: |
| Log.e(TAG, "Unknown message received"); |
| break; |
| } |
| } |
| |
| private void sendSeBroadcast(Intent intent) { |
| intent.addFlags(Intent.FLAG_INCLUDE_STOPPED_PACKAGES); |
| // Resume app switches so the receivers can start activites without delay |
| mNfcDispatcher.resumeAppSwitches(); |
| |
| synchronized(this) { |
| for (PackageInfo pkg : mInstalledPackages) { |
| if (pkg != null && pkg.applicationInfo != null) { |
| if (mNfceeAccessControl.check(pkg.applicationInfo)) { |
| intent.setPackage(pkg.packageName); |
| mContext.sendBroadcast(intent); |
| } |
| } |
| } |
| } |
| } |
| |
| private boolean llcpActivated(NfcDepEndpoint device) { |
| Log.d(TAG, "LLCP Activation message"); |
| |
| if (device.getMode() == NfcDepEndpoint.MODE_P2P_TARGET) { |
| if (DBG) Log.d(TAG, "NativeP2pDevice.MODE_P2P_TARGET"); |
| if (device.connect()) { |
| /* Check LLCP compliancy */ |
| if (mDeviceHost.doCheckLlcp()) { |
| /* Activate LLCP Link */ |
| if (mDeviceHost.doActivateLlcp()) { |
| if (DBG) Log.d(TAG, "Initiator Activate LLCP OK"); |
| synchronized (NfcService.this) { |
| // Register P2P device |
| mObjectMap.put(device.getHandle(), device); |
| } |
| mP2pLinkManager.onLlcpActivated(); |
| return true; |
| } else { |
| /* should not happen */ |
| Log.w(TAG, "Initiator LLCP activation failed. Disconnect."); |
| device.disconnect(); |
| } |
| } else { |
| if (DBG) Log.d(TAG, "Remote Target does not support LLCP. Disconnect."); |
| device.disconnect(); |
| } |
| } else { |
| if (DBG) Log.d(TAG, "Cannot connect remote Target. Polling loop restarted."); |
| /* |
| * The polling loop should have been restarted in failing |
| * doConnect |
| */ |
| } |
| } else if (device.getMode() == NfcDepEndpoint.MODE_P2P_INITIATOR) { |
| if (DBG) Log.d(TAG, "NativeP2pDevice.MODE_P2P_INITIATOR"); |
| /* Check LLCP compliancy */ |
| if (mDeviceHost.doCheckLlcp()) { |
| /* Activate LLCP Link */ |
| if (mDeviceHost.doActivateLlcp()) { |
| if (DBG) Log.d(TAG, "Target Activate LLCP OK"); |
| synchronized (NfcService.this) { |
| // Register P2P device |
| mObjectMap.put(device.getHandle(), device); |
| } |
| mP2pLinkManager.onLlcpActivated(); |
| return true; |
| } |
| } else { |
| Log.w(TAG, "checkLlcp failed"); |
| } |
| } |
| |
| return false; |
| } |
| |
| private void dispatchTagEndpoint(TagEndpoint tagEndpoint) { |
| Tag tag = new Tag(tagEndpoint.getUid(), tagEndpoint.getTechList(), |
| tagEndpoint.getTechExtras(), tagEndpoint.getHandle(), mNfcTagService); |
| registerTagObject(tagEndpoint); |
| if (!mNfcDispatcher.dispatchTag(tag)) { |
| unregisterObject(tagEndpoint.getHandle()); |
| playSound(SOUND_ERROR); |
| } else { |
| playSound(SOUND_END); |
| } |
| } |
| } |
| |
| private NfcServiceHandler mHandler = new NfcServiceHandler(); |
| |
| class ApplyRoutingTask extends AsyncTask<Integer, Void, Void> { |
| @Override |
| protected Void doInBackground(Integer... params) { |
| synchronized (NfcService.this) { |
| if (params == null || params.length != 1) { |
| // force apply current routing |
| applyRouting(true); |
| return null; |
| } |
| mScreenState = params[0].intValue(); |
| |
| boolean needWakelock = mScreenState == SCREEN_STATE_OFF; |
| if (needWakelock) { |
| mWakeLock.acquire(); |
| } |
| applyRouting(false); |
| if (needWakelock) { |
| mWakeLock.release(); |
| } |
| return null; |
| } |
| } |
| } |
| |
| private final BroadcastReceiver mReceiver = new BroadcastReceiver() { |
| @Override |
| public void onReceive(Context context, Intent intent) { |
| String action = intent.getAction(); |
| if (action.equals( |
| NativeNfcManager.INTERNAL_TARGET_DESELECTED_ACTION)) { |
| // Perform applyRouting() in AsyncTask to serialize blocking calls |
| new ApplyRoutingTask().execute(); |
| } else if (action.equals(Intent.ACTION_SCREEN_ON) |
| || action.equals(Intent.ACTION_SCREEN_OFF) |
| || action.equals(Intent.ACTION_USER_PRESENT)) { |
| // Perform applyRouting() in AsyncTask to serialize blocking calls |
| int screenState = SCREEN_STATE_OFF; |
| if (action.equals(Intent.ACTION_SCREEN_OFF)) { |
| screenState = SCREEN_STATE_OFF; |
| } else if (action.equals(Intent.ACTION_SCREEN_ON)) { |
| screenState = mKeyguard.isKeyguardLocked() ? |
| SCREEN_STATE_ON_LOCKED : SCREEN_STATE_ON_UNLOCKED; |
| } else if (action.equals(Intent.ACTION_USER_PRESENT)) { |
| screenState = SCREEN_STATE_ON_UNLOCKED; |
| } |
| new ApplyRoutingTask().execute(Integer.valueOf(screenState)); |
| } else if (action.equals(ACTION_MASTER_CLEAR_NOTIFICATION)) { |
| EnableDisableTask eeWipeTask = new EnableDisableTask(); |
| eeWipeTask.execute(TASK_EE_WIPE); |
| try { |
| eeWipeTask.get(); // blocks until EE wipe is complete |
| } catch (ExecutionException e) { |
| Log.w(TAG, "failed to wipe NFC-EE"); |
| } catch (InterruptedException e) { |
| Log.w(TAG, "failed to wipe NFC-EE"); |
| } |
| } else if (action.equals(Intent.ACTION_PACKAGE_REMOVED) || |
| action.equals(Intent.ACTION_PACKAGE_ADDED) || |
| action.equals(Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE) || |
| action.equals(Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE)) { |
| updatePackageCache(); |
| |
| if (action.equals(Intent.ACTION_PACKAGE_REMOVED)) { |
| // Clear the NFCEE access cache in case a UID gets recycled |
| mNfceeAccessControl.invalidateCache(); |
| |
| boolean dataRemoved = intent.getBooleanExtra(Intent.EXTRA_DATA_REMOVED, false); |
| if (dataRemoved) { |
| Uri data = intent.getData(); |
| if (data == null) return; |
| String packageName = data.getSchemeSpecificPart(); |
| |
| synchronized (NfcService.this) { |
| if (mSePackages.contains(packageName)) { |
| new EnableDisableTask().execute(TASK_EE_WIPE); |
| mSePackages.remove(packageName); |
| } |
| } |
| } |
| } |
| } else if (action.equals(Intent.ACTION_AIRPLANE_MODE_CHANGED)) { |
| boolean isAirplaneModeOn = intent.getBooleanExtra("state", false); |
| // Query the airplane mode from Settings.System just to make sure that |
| // some random app is not sending this intent |
| if (isAirplaneModeOn != isAirplaneModeOn()) { |
| return; |
| } |
| if (!mIsAirplaneSensitive) { |
| return; |
| } |
| mPrefsEditor.putBoolean(PREF_AIRPLANE_OVERRIDE, false); |
| mPrefsEditor.apply(); |
| if (isAirplaneModeOn) { |
| new EnableDisableTask().execute(TASK_DISABLE); |
| } else if (!isAirplaneModeOn && mPrefs.getBoolean(PREF_NFC_ON, NFC_ON_DEFAULT)) { |
| new EnableDisableTask().execute(TASK_ENABLE); |
| } |
| } |
| } |
| }; |
| |
| /** Returns true if airplane mode is currently on */ |
| boolean isAirplaneModeOn() { |
| return Settings.System.getInt(mContext.getContentResolver(), |
| Settings.System.AIRPLANE_MODE_ON, 0) == 1; |
| } |
| |
| /** for debugging only - no i18n */ |
| static String stateToString(int state) { |
| switch (state) { |
| case NfcAdapter.STATE_OFF: |
| return "off"; |
| case NfcAdapter.STATE_TURNING_ON: |
| return "turning on"; |
| case NfcAdapter.STATE_ON: |
| return "on"; |
| case NfcAdapter.STATE_TURNING_OFF: |
| return "turning off"; |
| default: |
| return "<error>"; |
| } |
| } |
| |
| /** For debugging only - no i18n */ |
| static String screenStateToString(int screenState) { |
| switch (screenState) { |
| case SCREEN_STATE_OFF: |
| return "OFF"; |
| case SCREEN_STATE_ON_LOCKED: |
| return "ON_LOCKED"; |
| case SCREEN_STATE_ON_UNLOCKED: |
| return "ON_UNLOCKED"; |
| default: |
| return "UNKNOWN"; |
| } |
| } |
| |
| void dump(FileDescriptor fd, PrintWriter pw, String[] args) { |
| if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.DUMP) |
| != PackageManager.PERMISSION_GRANTED) { |
| pw.println("Permission Denial: can't dump nfc from from pid=" |
| + Binder.getCallingPid() + ", uid=" + Binder.getCallingUid() |
| + " without permission " + android.Manifest.permission.DUMP); |
| return; |
| } |
| |
| synchronized (this) { |
| pw.println("mState=" + stateToString(mState)); |
| pw.println("mIsZeroClickRequested=" + mIsNdefPushEnabled); |
| pw.println("mScreenState=" + screenStateToString(mScreenState)); |
| pw.println("mNfcPollingEnabled=" + mNfcPollingEnabled); |
| pw.println("mNfceeRouteEnabled=" + mNfceeRouteEnabled); |
| pw.println("mIsAirplaneSensitive=" + mIsAirplaneSensitive); |
| pw.println("mIsAirplaneToggleable=" + mIsAirplaneToggleable); |
| pw.println("mOpenEe=" + mOpenEe); |
| mP2pLinkManager.dump(fd, pw, args); |
| mNfceeAccessControl.dump(fd, pw, args); |
| mNfcDispatcher.dump(fd, pw, args); |
| pw.println(mDeviceHost.dump()); |
| |
| } |
| } |
| } |