/* * 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 static android.Manifest.permission.BIND_NFC_SERVICE; import static android.nfc.OemLogItems.EVENT_DISABLE; import static android.nfc.OemLogItems.EVENT_ENABLE; import static com.android.nfc.ScreenStateHelper.SCREEN_STATE_ON_LOCKED; import static com.android.nfc.ScreenStateHelper.SCREEN_STATE_ON_UNLOCKED; import android.annotation.FlaggedApi; import android.annotation.NonNull; import android.annotation.Nullable; import android.app.ActivityManager; import android.app.AlarmManager; import android.app.Application; import android.app.BroadcastOptions; import android.app.KeyguardManager; import android.app.KeyguardManager.DeviceLockedStateListener; import android.app.KeyguardManager.KeyguardLockedStateListener; import android.app.PendingIntent; import android.app.admin.SecurityLog; import android.app.backup.BackupManager; import android.app.role.RoleManager; 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.ApplicationInfo; import android.content.pm.PackageInfo; import android.content.pm.PackageManager; import android.database.ContentObserver; import android.hardware.display.DisplayManager; import android.hardware.display.DisplayManager.DisplayListener; import android.media.AudioAttributes; import android.media.SoundPool; import android.media.SoundPool.OnLoadCompleteListener; import android.net.Uri; import android.nfc.AvailableNfcAntenna; import android.nfc.Constants; import android.nfc.Entry; import android.nfc.ErrorCodes; import android.nfc.FormatException; import android.nfc.IAppCallback; import android.nfc.INfcAdapter; import android.nfc.INfcAdapterExtras; import android.nfc.INfcCardEmulation; import android.nfc.INfcControllerAlwaysOnListener; import android.nfc.INfcDta; import android.nfc.INfcFCardEmulation; import android.nfc.INfcOemExtensionCallback; import android.nfc.INfcTag; import android.nfc.INfcUnlockHandler; import android.nfc.INfcVendorNciCallback; import android.nfc.INfcWlcStateListener; import android.nfc.IT4tNdefNfcee; import android.nfc.ITagRemovedCallback; import android.nfc.NdefMessage; import android.nfc.NfcAdapter; import android.nfc.NfcAntennaInfo; import android.nfc.NfcOemExtension; import android.nfc.OemLogItems; import android.nfc.T4tNdefNfcee; import android.nfc.T4tNdefNfceeCcFileInfo; import android.nfc.Tag; import android.nfc.TechListParcel; import android.nfc.TransceiveResult; import android.nfc.WlcListenerDeviceInfo; import android.nfc.cardemulation.CardEmulation; import android.nfc.cardemulation.PollingFrame; import android.nfc.tech.Ndef; import android.nfc.tech.TagTechnology; import android.os.AsyncTask; import android.os.Binder; import android.os.Build; import android.os.Bundle; import android.os.Handler; import android.os.IBinder; import android.os.Looper; import android.os.Message; import android.os.ParcelFileDescriptor; import android.os.PowerManager; import android.os.PowerManager.OnThermalStatusChangedListener; import android.os.Process; import android.os.RemoteException; import android.os.ResultReceiver; import android.os.SystemClock; import android.os.Trace; import android.os.UserHandle; import android.os.UserManager; import android.os.VibrationAttributes; import android.os.VibrationEffect; import android.os.Vibrator; import android.provider.Settings; import android.provider.Settings.SettingNotFoundException; import android.se.omapi.ISecureElementService; import android.sysprop.NfcProperties; import android.util.EventLog; import android.util.Log; import android.util.proto.ProtoOutputStream; import android.view.Display; import android.widget.Toast; import androidx.annotation.VisibleForTesting; import com.android.nfc.DeviceHost.DeviceHostListener; import com.android.nfc.DeviceHost.TagEndpoint; import com.android.nfc.cardemulation.CardEmulationManager; import com.android.nfc.cardemulation.util.StatsdUtils; import com.android.nfc.dhimpl.NativeNfcManager; import com.android.nfc.flags.FeatureFlags; import com.android.nfc.flags.Flags; import com.android.nfc.handover.HandoverDataParser; import com.android.nfc.proto.NfcEventProto; import com.android.nfc.wlc.NfcCharging; import com.google.protobuf.ByteString; import org.json.JSONException; import org.json.JSONObject; import java.io.File; import java.io.FileDescriptor; import java.io.FileOutputStream; import java.io.IOException; import java.io.PrintWriter; import java.io.StringWriter; import java.io.UnsupportedEncodingException; import java.nio.ByteBuffer; import java.nio.file.Files; import java.security.SecureRandom; import java.time.Instant; import java.util.ArrayList; import java.util.Arrays; import java.util.Collections; import java.util.HashMap; import java.util.HashSet; import java.util.HexFormat; import java.util.Iterator; import java.util.List; import java.util.Map; import java.util.NoSuchElementException; import java.util.Scanner; import java.util.Set; import java.util.Timer; import java.util.TimerTask; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.FutureTask; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; import java.util.stream.Collectors; public class NfcService implements DeviceHostListener, ForegroundUtils.Callback { static final boolean DBG = NfcProperties.debug_enabled().orElse(true); private static final boolean VDBG = false; // turn on for local testing. static final String TAG = "NfcService"; private static final int APP_INFO_FLAGS_SYSTEM_APP = ApplicationInfo.FLAG_SYSTEM | ApplicationInfo.FLAG_UPDATED_SYSTEM_APP; public static final String SERVICE_NAME = "nfc"; private static final String SYSTEM_UI = "com.android.systemui"; public static final String PREF = "NfcServicePrefs"; public static final String PREF_TAG_APP_LIST = "TagIntentAppPreferenceListPrefs"; static final String PREF_NFC_ON = "nfc_on"; static final String PREF_NFC_READER_OPTION_ON = "nfc_reader_on"; static final String PREF_NFC_CHARGING_ON = "nfc_charging_on"; static final boolean NFC_CHARGING_ON_DEFAULT = true; static final String PREF_MIGRATE_TO_DE_COMPLETE = "migrate_to_de_complete"; static final String PREF_SECURE_NFC_ON = "secure_nfc_on"; static final String PREF_FIRST_BOOT = "first_boot"; static final String PREF_ANTENNA_BLOCKED_MESSAGE_SHOWN = "antenna_blocked_message_shown"; static final boolean ANTENNA_BLOCKED_MESSAGE_SHOWN_DEFAULT = false; static final String NATIVE_LOG_FILE_NAME = "native_crash_logs"; static final String NATIVE_LOG_FILE_PATH = "/data/misc/nfc/logs"; static final int NATIVE_CRASH_FILE_SIZE = 1024 * 1024; private static final String WAIT_FOR_OEM_ALLOW_BOOT_TIMER_TAG = "NfcWaitForSimTag"; static final byte[] T4T_NFCEE_CC_FILE_ID = {(byte) (0xE1), (byte) (0x03)}; public static final int T4T_NFCEE_MAPPING_VERSION_2_0 = 0x20; @VisibleForTesting public static final int WAIT_FOR_OEM_ALLOW_BOOT_TIMEOUT_MS = 5_000; static final int MSG_NDEF_TAG = 0; // Previously used: MSG_LLCP_LINK_ACTIVATION = 1 // Previously used: MSG_LLCP_LINK_DEACTIVATED = 2 static final int MSG_MOCK_NDEF = 3; // Previously used: MSG_LLCP_LINK_FIRST_PACKET = 4 static final int MSG_ROUTE_AID = 5; static final int MSG_UNROUTE_AID = 6; static final int MSG_COMMIT_ROUTING = 7; // Previously used: MSG_INVOKE_BEAM = 8 static final int MSG_RF_FIELD_ACTIVATED = 9; static final int MSG_RF_FIELD_DEACTIVATED = 10; static final int MSG_RESUME_POLLING = 11; static final int MSG_REGISTER_T3T_IDENTIFIER = 12; static final int MSG_DEREGISTER_T3T_IDENTIFIER = 13; static final int MSG_TAG_DEBOUNCE = 14; // Previously used: MSG_UPDATE_STATS = 15 static final int MSG_APPLY_SCREEN_STATE = 16; static final int MSG_TRANSACTION_EVENT = 17; static final int MSG_PREFERRED_PAYMENT_CHANGED = 18; static final int MSG_TOAST_DEBOUNCE_EVENT = 19; static final int MSG_DELAY_POLLING = 20; static final int MSG_CLEAR_ROUTING_TABLE = 21; static final int MSG_UPDATE_ISODEP_PROTOCOL_ROUTE = 22; static final int MSG_UPDATE_TECHNOLOGY_ABF_ROUTE = 23; static final int MSG_WATCHDOG_PING = 24; static final int MSG_SE_SELECTED_EVENT = 25; static final int MSG_UPDATE_SYSTEM_CODE_ROUTE = 26; static final int MSG_PREFERRED_SIM_CHANGED = 27; static final String MSG_ROUTE_AID_PARAM_TAG = "power"; static final int MSG_RESTART_DISCOVERY = 28; // Negative value for NO polling delay static final int NO_POLL_DELAY = -1; static final long MAX_POLLING_PAUSE_TIMEOUT = 40000; static final int MAX_TOAST_DEBOUNCE_TIME = 10000; static final int DISABLE_POLLING_FLAGS = 0x1000; static final int RF_COALESCING_WINDOW = 50; static final int TASK_ENABLE = 1; static final int TASK_DISABLE = 2; static final int TASK_BOOT = 3; static final int TASK_ENABLE_ALWAYS_ON = 4; static final int TASK_DISABLE_ALWAYS_ON = 5; // SE selected types public static final int SE_SELECTED_AID = 0x01; public static final int SE_SELECTED_TECH = 0x02; public static final int SE_SELECTED_PROTOCOL = 0x04; // Polling technology masks static final int NFC_POLL_A = 0x01; static final int NFC_POLL_B = 0x02; static final int NFC_POLL_F = 0x04; static final int NFC_POLL_V = 0x08; static final int NFC_POLL_B_PRIME = 0x10; static final int NFC_POLL_KOVIO = 0x20; // Listen technology masks static final int NFC_LISTEN_A = 0x01; static final int NFC_LISTEN_B = 0x02; static final int NFC_LISTEN_F = 0x04; static final int NFC_LISTEN_V = 0x08; static final String PREF_POLL_TECH = "polling_tech_dfl"; // Default polling tech mask static final int DEFAULT_POLL_TECH = 0x2f; // See: Polling technology masks above static final String PREF_LISTEN_TECH = "listen_tech_dfl"; // Default listening tech mask static final int DEFAULT_LISTEN_TECH = 0xf; // See: Listen technology masks above // minimum screen state that enables NFC polling static final int NFC_POLLING_MODE = ScreenStateHelper.SCREEN_STATE_ON_UNLOCKED; // Time to wait for NFC controller to initialize before watchdog // goes off. This time is chosen large, because firmware download // may be a part of initialization. static final int INIT_WATCHDOG_MS = 90000; // Time to wait for routing to be applied before watchdog // goes off static final int ROUTING_WATCHDOG_MS = 6000; // Default delay used for presence checks static final int DEFAULT_PRESENCE_CHECK_DELAY = 125; // Removal Detection Wait Time Range static final int MIN_RF_REMOVAL_DETECTION_TIMEOUT = 0x00; static final int MAX_RF_REMOVAL_DETECTION_TIMEOUT = 0x13; static final NfcProperties.snoop_log_mode_values NFC_SNOOP_LOG_MODE = NfcProperties.snoop_log_mode().orElse(NfcProperties.snoop_log_mode_values.FILTERED); static final boolean NFC_VENDOR_DEBUG_ENABLED = NfcProperties.vendor_debug_enabled().orElse(false); // RF field events as defined in NFC extras 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 APP_NAME_ENABLING_NFC = "com.android.nfc.PACKAGE_NAME_ENABLING_NFC"; public static boolean sIsShortRecordLayout = false; public static boolean sIsNfcRestore = false; // for use with playSound() public static final int SOUND_END = 1; public static final int SOUND_ERROR = 2; public static final int NCI_VERSION_2_0 = 0x20; public static final int NCI_VERSION_1_0 = 0x10; // Timeout to re-apply routing if a tag was present and we postponed it private static final int APPLY_ROUTING_RETRY_TIMEOUT_MS = 5000; private static final VibrationAttributes HARDWARE_FEEDBACK_VIBRATION_ATTRIBUTES = VibrationAttributes.createForUsage(VibrationAttributes.USAGE_HARDWARE_FEEDBACK); private static final int NCI_STATUS_OK = 0x00; private static final int NCI_STATUS_REJECTED = 0x01; private static final int NCI_STATUS_MESSAGE_CORRUPTED = 0x02; private static final int NCI_STATUS_FAILED = 0x03; private static final int SEND_VENDOR_CMD_TIMEOUT_MS = 3_000; private static final int CHECK_FIRMWARE_TIMEOUT_MS = 60_000; private static final int NCI_GID_PROP = 0x0F; private static final int NCI_MSG_PROP_ANDROID = 0x0C; private static final int NCI_MSG_PROP_ANDROID_POWER_SAVING = 0x01; private static final int NCI_PROP_ANDROID_QUERY_POWER_SAVING_STATUS_CMD = 0x0A; private static final int POWER_STATE_SWITCH_ON = 0x01; public static final int WAIT_FOR_OEM_CALLBACK_TIMEOUT_MS = 3000; public static final int WAIT_FOR_COMMIT_ROUTING_TIMEOUT_MS = 10000; private static final long TIME_TO_MONITOR_AFTER_FIELD_ON_MS = 10000L; private final Looper mLooper; private final UserManager mUserManager; private final ActivityManager mActivityManager; private static int nci_version = NCI_VERSION_1_0; // NFC Execution Environment // fields below are protected by this private final boolean mPollingDisableAllowed; private HashMap mPollingDisableDeathRecipients = new HashMap(); private final ReaderModeDeathRecipient mReaderModeDeathRecipient = new ReaderModeDeathRecipient(); private final SeServiceDeathRecipient mSeServiceDeathRecipient = new SeServiceDeathRecipient(); private final DiscoveryTechDeathRecipient mDiscoveryTechDeathRecipient = new DiscoveryTechDeathRecipient(); private final NfcUnlockManager mNfcUnlockManager; private final BackupManager mBackupManager; private final SecureRandom mCookieGenerator = new SecureRandom(); // Tag app preference list for the target UserId. HashMap> mTagAppPrefList = new HashMap>(); // Tag app blocklist hash static final String PREF_TAG_APP_BLOCK_LIST_HASH = "tag_app_block_list_hash"; // Tag app blocklist hash default static final int PREF_TAG_APP_BLOCK_LIST_HASH_DEFAULT = 0; // Tag app preference blocked list. static final List TAG_APP_BLOCKLIST = new ArrayList(); // cached version of installed packages requesting Android.permission.NFC_TRANSACTION_EVENTS // for current user and profiles. The Integer part is the userId. HashMap> mNfcEventInstalledPackages = new HashMap>(); // cached version of installed packages requesting // Android.permission.NFC_PREFERRED_PAYMENT_INFO for current user and profiles. // The Integer part is the userId. HashMap> mNfcPreferredPaymentChangedInstalledPackages = new HashMap>(); // fields below are used in multiple threads and protected by synchronized(this) final HashMap mObjectMap = new HashMap(); int mScreenState; boolean mInProvisionMode; // whether we're in setup wizard and enabled NFC provisioning boolean mIsSecureNfcEnabled; boolean mSkipNdefRead; NfcDiscoveryParameters mCurrentDiscoveryParameters = NfcDiscoveryParameters.getNfcOffParameters(); ReaderModeParams mReaderModeParams; DiscoveryTechParams mDiscoveryTechParams; private int mUserId; boolean mPollingPaused; // True if nfc notification message already shown boolean mAntennaBlockedMessageShown; private static int mDispatchFailedCount; private static int mDispatchFailedMax; static final int INVALID_NATIVE_HANDLE = -1; static final int MOCK_NATIVE_HANDLE = 0; byte mDebounceTagUid[]; int mDebounceTagDebounceMs; int mDebounceTagNativeHandle = INVALID_NATIVE_HANDLE; ITagRemovedCallback mDebounceTagRemovedCallback; // Only accessed on one thread so doesn't need locking NdefMessage mLastReadNdefMessage; // 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 // mAlwaysOnState 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 mAlwaysOnState; // one of NfcAdapter.STATE_ON, STATE_TURNING_ON, etc int mAlwaysOnMode; // one of NfcOemExtension.ENABLE_DEFAULT, ENABLE_TRANSPARENT, etc private boolean mIsPowerSavingModeEnabled = false; // fields below are final after onCreate() boolean mIsReaderOptionEnabled = true; boolean mReaderOptionCapable; Context mContext; NfcInjector mNfcInjector; NfcEventLog mNfcEventLog; private DeviceHost mDeviceHost; private SharedPreferences mPrefs; private SharedPreferences.Editor mPrefsEditor; private SharedPreferences mTagAppPrefListPrefs; private PowerManager.WakeLock mRoutingWakeLock; private PowerManager.WakeLock mRequireUnlockWakeLock; private long mLastFieldOnTimestamp = 0; int mEndSound; int mErrorSound; SoundPool mSoundPool; // playback synchronized on this TagService mNfcTagService; T4tNdefNfceeService mT4tNdefNfceeService; NfcAdapterService mNfcAdapter; NfcDtaService mNfcDtaService; RoutingTableParser mRoutingTableParser; boolean mIsDebugBuild; boolean mIsHceCapable; boolean mIsHceFCapable; boolean mIsSecureNfcCapable; boolean mIsRequestUnlockShowed; boolean mIsRecovering; boolean mIsNfcUserRestricted; boolean mIsNfcUserChangeRestricted; boolean mIsWatchType; boolean mPendingPowerStateUpdate; boolean mIsWlcCapable; boolean mIsWlcEnabled; boolean mIsRWCapable; boolean mIsRDCapable; WlcListenerDeviceInfo mWlcListenerDeviceInfo; public NfcDiagnostics mNfcDiagnostics; // polling delay control variables private final int mPollDelayTime; private final int mPollDelayTimeLong; private final int mAppInActivityDetectionTime; private final int mTagRemovalDetectionWaitTime; private Timer mAppInActivityDetectionTimer; private final int mPollDelayCountMax; private int mPollDelayCount; private int mReadErrorCount; private int mReadErrorCountMax; private boolean mPollDelayed; boolean mNotifyDispatchFailed; boolean mNotifyReadFailed; // for recording the latest Tag object cookie long mCookieUpToDate = -1; private DeviceConfigFacade mDeviceConfigFacade; private NfcDispatcher mNfcDispatcher; private PowerManager mPowerManager; private KeyguardManager mKeyguard; private HandoverDataParser mHandoverDataParser; private ContentResolver mContentResolver; @VisibleForTesting CardEmulationManager mCardEmulationManager; private NfcCharging mNfcCharging; private Vibrator mVibrator; private VibrationEffect mVibrationEffect; private ISecureElementService mSEService; private final AlarmManager mAlarmManager; private ScreenStateHelper mScreenStateHelper; private ForegroundUtils mForegroundUtils; private final NfcPermissions mNfcPermissions; private static NfcService sService; private static boolean sToast_debounce = false; private static int sToast_debounce_time_ms = 3000; public static boolean sIsDtaMode = false; private final boolean mIsTagAppPrefSupported; private int mTagAppBlockListHash; private final boolean mIsAlwaysOnSupported; private final Set mAlwaysOnListeners = Collections.synchronizedSet(new HashSet<>()); private int mAidMatchingExactOnly = 0x02; public static final int T4TNFCEE_STATUS_FAILED = -1; private Object mT4tNdefNfcEeObj = new Object(); private Bundle mT4tNdefNfceeReturnBundle = new Bundle(); private final FeatureFlags mFeatureFlags; private final Set mWlcStateListener = Collections.synchronizedSet(new HashSet<>()); @Nullable private final StatsdUtils mStatsdUtils; private final boolean mCheckDisplayStateForScreenState; private INfcVendorNciCallback mNfcVendorNciCallBack = null; private INfcOemExtensionCallback mNfcOemExtensionCallback = null; private CountDownLatch mCommitRoutingCountDownLatch = null; private int mCommitRoutingStatus; private final DisplayListener mDisplayListener = new DisplayListener() { @Override public void onDisplayAdded(int displayId) { } @Override public void onDisplayRemoved(int displayId) { } @Override public void onDisplayChanged(int displayId) { if (displayId == Display.DEFAULT_DISPLAY) { handleScreenStateChanged(); } } }; private Object mDiscoveryLock = new Object(); private boolean mCardEmulationActivated = false; private boolean mRfFieldActivated = false; private boolean mRfDiscoveryStarted = false; private boolean mEeListenActivated = false; // Scheduled executor for routing table update private final ScheduledExecutorService mRtUpdateScheduler = Executors.newScheduledThreadPool(1); private ScheduledFuture mRtUpdateScheduledTask = null; private static final int STATUS_OK = NfcOemExtension.STATUS_OK; private static final int STATUS_UNKNOWN_ERROR = NfcOemExtension.STATUS_UNKNOWN_ERROR; private static final int ACTION_ON_ENABLE = 0; private static final int ACTION_ON_DISABLE = 1; private static final int ACTION_ON_TAG_DISPATCH = 2; private static final int ACTION_ON_READ_NDEF = 3; private static final int ACTION_ON_APPLY_ROUTING = 4; private static final int ACTION_ON_ROUTING_CHANGED = 5; public static NfcService getInstance() { return sService; } @Override public void onRemoteEndpointDiscovered(TagEndpoint tag) { Log.d(TAG, "onRemoteEndpointDiscovered"); sendMessage(MSG_NDEF_TAG, tag); } /** * Notifies transaction */ @Override public void onHostCardEmulationActivated(int technology) { mCardEmulationActivated = true; try { if (mNfcOemExtensionCallback != null) { mNfcOemExtensionCallback.onCardEmulationActivated(mCardEmulationActivated); } } catch (RemoteException e) { Log.e(TAG, "onHostCardEmulationActivated: error: ", e); } if (mCardEmulationManager != null) { mCardEmulationManager.onHostCardEmulationActivated(technology); if (android.nfc.Flags.nfcPersistLog()) { mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setHostCardEmulationStateChange( NfcEventProto.NfcHostCardEmulationStateChange.newBuilder() .setTechnology(technology) .setEnable(true) .build()) .build()); } } } @Override public void onHostCardEmulationData(int technology, byte[] data) { if (mCardEmulationManager != null) { mCardEmulationManager.onHostCardEmulationData(technology, data); if (android.nfc.Flags.nfcPersistLog() && NFC_VENDOR_DEBUG_ENABLED) { mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setHostCardEmulationData( NfcEventProto.NfcHostCardEmulationData.newBuilder() .setTechnology(technology) .setData(ByteString.copyFrom(data)) .build()) .build()); } } } @Override public void onHostCardEmulationDeactivated(int technology) { mCardEmulationActivated = false; try { if (mNfcOemExtensionCallback != null) { mNfcOemExtensionCallback.onCardEmulationActivated(mCardEmulationActivated); } } catch (RemoteException e) { Log.e(TAG, "onHostCardEmulationDeactivated: e=", e); } if (mCardEmulationManager != null) { mCardEmulationManager.onHostCardEmulationDeactivated(technology); if (android.nfc.Flags.nfcPersistLog()) { mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setHostCardEmulationStateChange( NfcEventProto.NfcHostCardEmulationStateChange.newBuilder() .setTechnology(technology) .setEnable(false) .build()) .build()); } } } /** Notifies Removal Detection procedure completed, endpoint deactivation reason returned by NFCC */ public void onEndpointRemoved(int reason) { Log.d(TAG, "onEndpointRemoved: Deactivation reason is " + reason); if (mIsWlcEnabled && mNfcCharging.NfcChargingMode) { mNfcCharging.onEndpointRemoved(reason); } /* else send intent to notify other applications */ } public boolean startRemovalDetection(int waiting_time_int) { return mDeviceHost.detectEpRemoval(waiting_time_int); } @Override public void onRemoteFieldActivated() { mRfFieldActivated = true; mLastFieldOnTimestamp = mNfcInjector.getWallClockMillis(); mNfcInjector.ensureWatchdogMonitoring(); try { if (mNfcOemExtensionCallback != null) { mNfcOemExtensionCallback.onRfFieldDetected(mRfFieldActivated); } } catch (RemoteException e) { Log.e(TAG, "onRemoteFieldActivated: e=", e); } if (Flags.coalesceRfEvents() && mHandler.hasMessages(MSG_RF_FIELD_DEACTIVATED)) { mHandler.removeMessages(MSG_RF_FIELD_DEACTIVATED); } else { sendMessage(MSG_RF_FIELD_ACTIVATED, null); } if (mStatsdUtils != null) { mStatsdUtils.logFieldChanged(true, 0); } if (android.nfc.Flags.nfcPersistLog() && NFC_VENDOR_DEBUG_ENABLED) { mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setRemoteFieldStateChange( NfcEventProto.NfcRemoteFieldStateChange.newBuilder() .setFieldOn(true) .build()) .build()); } if (android.nfc.Flags.nfcEventListener() && mCardEmulationManager != null) { mCardEmulationManager.onRemoteFieldChanged(true); } } @Override public void onRemoteFieldDeactivated() { mRfFieldActivated = false; try { if (mNfcOemExtensionCallback != null) { mNfcOemExtensionCallback.onRfFieldDetected(mRfFieldActivated); } } catch (RemoteException e) { Log.e(TAG, "onRemoteFieldDeactivated: e=", e); } if (Flags.coalesceRfEvents()) { mHandler.sendMessageDelayed( mHandler.obtainMessage(MSG_RF_FIELD_DEACTIVATED), RF_COALESCING_WINDOW); } else { sendMessage(MSG_RF_FIELD_DEACTIVATED, null); } if (mStatsdUtils != null) { mStatsdUtils.logFieldChanged(false, 0); } if (android.nfc.Flags.nfcPersistLog() && NFC_VENDOR_DEBUG_ENABLED) { mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setRemoteFieldStateChange( NfcEventProto.NfcRemoteFieldStateChange.newBuilder() .setFieldOn(false) .build()) .build()); } if (android.nfc.Flags.nfcEventListener() && mCardEmulationManager != null) { mCardEmulationManager.onRemoteFieldChanged(false); } } List mPollingFramesToBeSent = new ArrayList<>(); final Runnable mPollingLoopsDetectedRunnable = new Runnable() { public void run() { List frames; synchronized (mPollingLoopsDetectedRunnable) { frames = mPollingFramesToBeSent; mPollingFramesToBeSent = new ArrayList<>(); } if (mCardEmulationManager != null) { mCardEmulationManager.onPollingLoopDetected(new ArrayList<>(frames)); } } }; @Override public void onPollingLoopDetected(List frames) { if (mCardEmulationManager != null) { if (Flags.postCallbacks()) { synchronized (mPollingLoopsDetectedRunnable) { mPollingFramesToBeSent.addAll(frames); if (!mHandler.hasCallbacks(mPollingLoopsDetectedRunnable)) { mHandler.post(mPollingLoopsDetectedRunnable); } } } else { mCardEmulationManager.onPollingLoopDetected((frames)); } } } @Override public void onNfcTransactionEvent(byte[] aid, byte[] data, String seName) { byte[][] dataObj = {aid, data, seName.getBytes()}; sendMessage(MSG_TRANSACTION_EVENT, dataObj); } @Override public void onEeUpdated() { if (mNfcOemExtensionCallback != null) { try { mNfcOemExtensionCallback.onEeUpdated(); } catch (RemoteException e) { Log.e(TAG, "onEeUpdated: e=", e); } } if (mRtUpdateScheduledTask != null && !mRtUpdateScheduledTask.isDone()) { mRtUpdateScheduledTask.cancel(false); } // Delay routing table update to allow remove useless operations when several // ntf are received mRtUpdateScheduledTask = mRtUpdateScheduler.schedule( () -> { if (mIsHceCapable) { if (DBG) Log.d(TAG, "onEeUpdated: trigger routing table update"); mCardEmulationManager.onTriggerRoutingTableUpdate(); } }, 50, TimeUnit.MILLISECONDS); } private void restartStack() { synchronized (NfcService.this) { if (DBG) { Log.d(TAG, "restartStack: mIsRecovering=" + mIsRecovering); } if (!mIsRecovering) { mIsRecovering = true; } else { return; } } if (DBG) { Log.d(TAG, "restartStack: Restarting NFC Service"); } try { mContext.unregisterReceiver(mReceiver); } catch (IllegalArgumentException e) { Log.w(TAG, "restartStack: Failed to unregisterScreenState BroadCastReceiver: " + e); } new EnableDisableTask().execute(TASK_DISABLE); new EnableDisableTask().execute(TASK_ENABLE); } @Override public void onHwErrorReported() { if (android.nfc.Flags.nfcEventListener() && mCardEmulationManager != null) { mCardEmulationManager.onInternalErrorReported( CardEmulation.NFC_INTERNAL_ERROR_NFC_HARDWARE_ERROR); } restartStack(); } @FlaggedApi(android.nfc.Flags.FLAG_NFC_EVENT_LISTENER) @Override public void onCommandTimeout() { if (android.nfc.Flags.nfcEventListener() && mCardEmulationManager != null) { mCardEmulationManager.onInternalErrorReported( CardEmulation.NFC_INTERNAL_ERROR_COMMAND_TIMEOUT); } } @Override public void onVendorSpecificEvent(int gid, int oid, byte[] payload) { mHandler.post(() -> mNfcAdapter.sendVendorNciNotification(gid, oid, payload)); } @Override public void onObserveModeStateChanged(boolean enable) { if (Flags.postCallbacks()) { mHandler.post(() -> { if (mCardEmulationManager != null) { mCardEmulationManager.onObserveModeStateChange(enable); } }); } else { if (mCardEmulationManager != null) { mCardEmulationManager.onObserveModeStateChange(enable); } } } @Override public void onObserveModeDisabledInFirmware(PollingFrame exitFrame) { mCardEmulationManager.onObserveModeDisabledInFirmware(exitFrame); onObserveModeStateChanged(false); } @Override public void onObserveModeEnabledInFirmware() { onObserveModeStateChanged(true); } @Override public void onEeListenActivated(boolean isActivated) { mEeListenActivated = isActivated; mCardEmulationManager.onEeListenActivated(isActivated); try { if (mNfcOemExtensionCallback != null) { mNfcOemExtensionCallback.onEeListenActivated(isActivated); } } catch (RemoteException e) { Log.e(TAG, "onEeListenActivated: e=", e); } } @Override public void onRfDiscoveryEvent(boolean isDiscoveryStarted) { synchronized (mDiscoveryLock) { mRfDiscoveryStarted = isDiscoveryStarted; } try { if (mNfcOemExtensionCallback != null) { mNfcOemExtensionCallback.onRfDiscoveryStarted(mRfDiscoveryStarted); } } catch (RemoteException e) { Log.e(TAG, "onRfDiscoveryEvent: e=", e); } } @Override public void onSeSelected(int type) { sendMessage(MSG_SE_SELECTED_EVENT, type); } @Override public void onRestartRfDiscovery() { sendMessage(NfcService.MSG_RESTART_DISCOVERY, null); } /** * Enable or Disable PowerSaving Mode based on flag */ private boolean setPowerSavingMode(boolean flag) { synchronized (NfcService.this) { if ((flag && mState != NfcAdapter.STATE_ON) || (!flag && mState != NfcAdapter.STATE_OFF)) { Log.d(TAG, "setPowerSavingMode: Enable Power Saving Mode is allowed in " + "Nfc On state or " + "Disable PowerSaving is allowed only if it is enabled"); return false; } } Log.d(TAG, "setPowerSavingMode: " + flag); if (flag) { if (mDeviceHost.setPowerSavingMode(flag)) { mIsPowerSavingModeEnabled = true; new EnableDisableTask().execute(TASK_DISABLE); return true; } } else { new EnableDisableTask().execute(TASK_ENABLE); return true; } Log.d(TAG, "PowerSavingMode: failed"); return false; } public void onWlcData(Map WlcDeviceInfo) { for (String key : WlcDeviceInfo.keySet()) { Log.d(TAG, "onWlcData: " + key + " = " + WlcDeviceInfo.get(key)); } synchronized (mWlcStateListener) { mWlcListenerDeviceInfo = new WlcListenerDeviceInfo( WlcDeviceInfo.get(mNfcCharging.VendorId), WlcDeviceInfo.get(mNfcCharging.TemperatureListener), WlcDeviceInfo.get(mNfcCharging.BatteryLevel), WlcDeviceInfo.get(mNfcCharging.State)); for (INfcWlcStateListener listener : mWlcStateListener) { try { listener.onWlcStateChanged(mWlcListenerDeviceInfo); } catch (RemoteException e) { Log.e(TAG, "onWlcData: error"); } } } } /** Notifies WLC procedure stopped */ @Override public void onWlcStopped(int wpt_end_condition) { Log.d(TAG, "onWlcStopped: End condition is " + wpt_end_condition); mNfcCharging.onWlcStopped(wpt_end_condition); } public void onTagRfDiscovered(boolean discovered) { Log.d(TAG, "onTagRfDiscovered: " + discovered); executeOemOnTagConnectedCallback(discovered); } final class ReaderModeParams { public int flags; public IAppCallback callback; public int presenceCheckDelay; public IBinder binder; public int uid; public byte[] annotation; } final class DiscoveryTechParams { public IBinder binder; public int uid; } void saveNfcOnSetting(boolean on) { synchronized (NfcService.this) { mPrefsEditor.putBoolean(PREF_NFC_ON, on); mPrefsEditor.apply(); mBackupManager.dataChanged(); } } boolean getNfcOnSetting() { synchronized (NfcService.this) { return mPrefs.getBoolean(PREF_NFC_ON, mDeviceConfigFacade.getNfcDefaultState()); } } void saveNfcListenTech(int tech) { synchronized (NfcService.this) { mPrefsEditor.putInt(PREF_LISTEN_TECH, tech); mPrefsEditor.apply(); mBackupManager.dataChanged(); } } int getNfcListenTech() { synchronized (NfcService.this) { return mPrefs.getInt(PREF_LISTEN_TECH, DEFAULT_LISTEN_TECH); } } void saveNfcPollTech(int tech) { synchronized (NfcService.this) { mPrefsEditor.putInt(PREF_POLL_TECH, tech); mPrefsEditor.apply(); mBackupManager.dataChanged(); } } int getNfcPollTech() { synchronized (NfcService.this) { return mPrefs.getInt(PREF_POLL_TECH, DEFAULT_POLL_TECH); } } /** Returns true if NFC has user restriction set. */ private boolean isNfcUserRestricted() { return mUserManager.getUserRestrictions().getBoolean( UserManager.DISALLOW_NEAR_FIELD_COMMUNICATION_RADIO); } /** Returns true if NFC state change by user is restricted. */ private boolean isNfcUserChangeRestricted() { return mUserManager.getUserRestrictions().getBoolean( UserManager.DISALLOW_CHANGE_NEAR_FIELD_COMMUNICATION_RADIO ); } boolean shouldEnableNfc() { return getNfcOnSetting() && !mNfcInjector.isSatelliteModeOn() && !isNfcUserRestricted() && allowOemEnable(); } boolean allowOemEnable() { if (mNfcOemExtensionCallback == null) return true; return receiveOemCallbackResult(ACTION_ON_ENABLE); } boolean allowOemDisable() { if (mNfcOemExtensionCallback == null) return true; return receiveOemCallbackResult(ACTION_ON_DISABLE); } boolean receiveOemCallbackResult(int action) { CountDownLatch latch = new CountDownLatch(1); NfcCallbackResultReceiver.OnReceiveResultListener listener = new NfcCallbackResultReceiver.OnReceiveResultListener(); ResultReceiver receiver = new NfcCallbackResultReceiver(latch, listener); try { switch (action) { case ACTION_ON_ENABLE: mNfcOemExtensionCallback.onEnable(receiver); break; case ACTION_ON_DISABLE: mNfcOemExtensionCallback.onDisable(receiver); break; case ACTION_ON_TAG_DISPATCH: mNfcOemExtensionCallback.onTagDispatch(receiver); break; case ACTION_ON_READ_NDEF: mNfcOemExtensionCallback.onNdefRead(receiver); break; case ACTION_ON_APPLY_ROUTING: mNfcOemExtensionCallback.onApplyRouting(receiver); break; case ACTION_ON_ROUTING_CHANGED: mNfcOemExtensionCallback.onRoutingChanged(receiver); break; } } catch (RemoteException remoteException) { return false; } try { boolean success = latch.await(WAIT_FOR_OEM_CALLBACK_TIMEOUT_MS, TimeUnit.MILLISECONDS); if (!success) { return false; } else { return listener.getResultCode() == 1; } } catch (InterruptedException ie) { return false; } } private void registerGlobalBroadcastsReceiver() { IntentFilter filter = new IntentFilter(Intent.ACTION_SCREEN_OFF); filter.addAction(Intent.ACTION_SCREEN_ON); filter.addAction(Intent.ACTION_USER_PRESENT); filter.addAction(Intent.ACTION_USER_SWITCHED); filter.addAction(Intent.ACTION_USER_ADDED); filter.addAction(Intent.ACTION_BOOT_COMPLETED); if (mFeatureFlags.enableDirectBootAware()) filter.addAction(Intent.ACTION_USER_UNLOCKED); mContext.registerReceiverForAllUsers(mReceiver, filter, null, null); } public NfcService(Application nfcApplication, NfcInjector nfcInjector) { mUserId = ActivityManager.getCurrentUser(); mContext = nfcApplication; mNfcInjector = nfcInjector; mLooper = mNfcInjector.getMainLooper(); mHandler = new NfcServiceHandler(mLooper); mNfcEventLog = mNfcInjector.getNfcEventLog(); mNfcTagService = new TagService(); mNfcAdapter = new NfcAdapterService(); mRoutingTableParser = mNfcInjector.getRoutingTableParser(); mT4tNdefNfceeService = new T4tNdefNfceeService(); Log.i(TAG, "Starting NFC service"); sService = this; mScreenStateHelper = mNfcInjector.getScreenStateHelper(); mContentResolver = mContext.getContentResolver(); mDeviceHost = mNfcInjector.makeDeviceHost(this); mNfcUnlockManager = mNfcInjector.getNfcUnlockManager(); mHandoverDataParser = mNfcInjector.getHandoverDataParser(); mInProvisionMode = mNfcInjector.isInProvisionMode(); mDeviceConfigFacade = mNfcInjector.getDeviceConfigFacade(); mNfcDispatcher = mNfcInjector.getNfcDispatcher(); mPrefs = mContext.getSharedPreferences(PREF, Context.MODE_PRIVATE); mPrefsEditor = mPrefs.edit(); mState = NfcAdapter.STATE_OFF; mAlwaysOnState = NfcAdapter.STATE_OFF; mAlwaysOnMode = NfcOemExtension.ENABLE_DEFAULT; mIsDebugBuild = "userdebug".equals(Build.TYPE) || "eng".equals(Build.TYPE); mPowerManager = mContext.getSystemService(PowerManager.class); mRoutingWakeLock = mPowerManager.newWakeLock( PowerManager.PARTIAL_WAKE_LOCK, "NfcService:mRoutingWakeLock"); mRequireUnlockWakeLock = mPowerManager.newWakeLock(PowerManager.SCREEN_BRIGHT_WAKE_LOCK | PowerManager.ACQUIRE_CAUSES_WAKEUP | PowerManager.ON_AFTER_RELEASE, "NfcService:mRequireUnlockWakeLock"); mKeyguard = mContext.getSystemService(KeyguardManager.class); mUserManager = mContext.getSystemService(UserManager.class); mActivityManager = mContext.getSystemService(ActivityManager.class); mVibrator = mContext.getSystemService(Vibrator.class); mVibrationEffect = mNfcInjector.getVibrationEffect(); PackageManager pm = mContext.getPackageManager(); mIsWatchType = pm.hasSystemFeature(PackageManager.FEATURE_WATCH); mNfcDiagnostics = mNfcInjector.getNfcDiagnostics(); mAlarmManager = mContext.getSystemService(AlarmManager.class); mCheckDisplayStateForScreenState = mDeviceConfigFacade.getCheckDisplayStateForScreenState(); if (mInProvisionMode) { mScreenState = mScreenStateHelper.checkScreenStateProvisionMode(); } else { mScreenState = mScreenStateHelper.checkScreenState(mCheckDisplayStateForScreenState); } if (mCheckDisplayStateForScreenState) { DisplayManager displayManager = mContext.getSystemService(DisplayManager.class); displayManager.registerDisplayListener(mDisplayListener, mHandler); } mBackupManager = mNfcInjector.getBackupManager(); mFeatureFlags = mNfcInjector.getFeatureFlags(); mStatsdUtils = mNfcInjector.getStatsdUtils(); // Intents for all users registerGlobalBroadcastsReceiver(); // Listen for work profile adds or removes. IntentFilter managedProfileFilter = new IntentFilter(); managedProfileFilter.addAction(Intent.ACTION_MANAGED_PROFILE_ADDED); managedProfileFilter.addAction(Intent.ACTION_MANAGED_PROFILE_REMOVED); managedProfileFilter.addAction(Intent.ACTION_MANAGED_PROFILE_AVAILABLE); managedProfileFilter.addAction(Intent.ACTION_MANAGED_PROFILE_UNAVAILABLE); mContext.registerReceiverForAllUsers(mManagedProfileReceiver, managedProfileFilter, null, null); IntentFilter ownerFilter = new IntentFilter(Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE); ownerFilter.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE); ownerFilter.addAction(Intent.ACTION_SHUTDOWN); mContext.registerReceiverForAllUsers(mOwnerReceiver, ownerFilter, null, null); ownerFilter = new IntentFilter(); ownerFilter.addAction(Intent.ACTION_PACKAGE_ADDED); ownerFilter.addAction(Intent.ACTION_PACKAGE_REMOVED); ownerFilter.addDataScheme("package"); mContext.registerReceiverForAllUsers(mOwnerReceiver, ownerFilter, null, null); addDeviceLockedStateListener(); updatePackageCache(); mIsRWCapable = pm.hasSystemFeature(PackageManager.FEATURE_NFC); mIsWlcCapable = android.nfc.Flags.enableNfcCharging() && pm.hasSystemFeature(PackageManager.FEATURE_NFC_CHARGING); if (mIsWlcCapable) { mNfcCharging = mNfcInjector.getNfcCharging(mDeviceHost); mIsWlcEnabled = mPrefs.getBoolean(PREF_NFC_CHARGING_ON, NFC_CHARGING_ON_DEFAULT); // Register ThermalStatusChangedListener addThermalStatusListener(); } mIsRDCapable = mContext.getResources().getBoolean(R.bool.removal_detection_default); mIsHceCapable = pm.hasSystemFeature(PackageManager.FEATURE_NFC_HOST_CARD_EMULATION) || pm.hasSystemFeature(PackageManager.FEATURE_NFC_HOST_CARD_EMULATION_NFCF); mIsHceFCapable = pm.hasSystemFeature(PackageManager.FEATURE_NFC_HOST_CARD_EMULATION_NFCF); if (mIsHceCapable) { mCardEmulationManager = mNfcInjector.getCardEmulationManager(); } mForegroundUtils = mNfcInjector.getForegroundUtils(); mIsSecureNfcCapable = mDeviceConfigFacade.isSecureNfcCapable(); mIsSecureNfcEnabled = mPrefs.getBoolean(PREF_SECURE_NFC_ON, mDeviceConfigFacade.getDefaultSecureNfcState()) && mIsSecureNfcCapable; mDeviceHost.setNfcSecure(mIsSecureNfcEnabled); sToast_debounce_time_ms = mContext.getResources().getInteger(R.integer.toast_debounce_time_ms); if (sToast_debounce_time_ms > MAX_TOAST_DEBOUNCE_TIME) { sToast_debounce_time_ms = MAX_TOAST_DEBOUNCE_TIME; } // Notification message variables mDispatchFailedCount = 0; if (mDeviceConfigFacade.isAntennaBlockedAlertEnabled() && !mPrefs.getBoolean(PREF_ANTENNA_BLOCKED_MESSAGE_SHOWN, ANTENNA_BLOCKED_MESSAGE_SHOWN_DEFAULT)) { mAntennaBlockedMessageShown = false; mDispatchFailedMax = mContext.getResources().getInteger(R.integer.max_antenna_blocked_failure_count); } else { mAntennaBlockedMessageShown = true; } // Polling delay count for switching from stage one to stage two. mPollDelayCountMax = mDeviceConfigFacade.getUnknownTagPollingDelayMax(); // Stage one: polling delay time for the first few unknown tag detections mPollDelayTime = mDeviceConfigFacade.getUnknownTagPollingDelay(); // Stage two: longer polling delay time after max_poll_delay_count mPollDelayTimeLong = mDeviceConfigFacade.getUnknownTagPollingDelayLong(); // Polling delay if read error found more than max count. mReadErrorCountMax = mContext.getResources().getInteger(R.integer.unknown_tag_read_error_count_max); mNotifyDispatchFailed = mContext.getResources().getBoolean(R.bool.enable_notify_dispatch_failed); mNotifyReadFailed = mContext.getResources().getBoolean(R.bool.enable_notify_read_failed); mPollingDisableAllowed = mDeviceConfigFacade.getPollingDisableAllowed(); mAppInActivityDetectionTime = mContext.getResources().getInteger(R.integer.inactive_presence_check_allowed_time); mTagRemovalDetectionWaitTime = mContext.getResources().getInteger(R.integer.removal_detection_waiting_time); // Make sure this is only called when object construction is complete. mNfcInjector.getNfcManagerRegisterer().register(mNfcAdapter); mIsAlwaysOnSupported = mDeviceConfigFacade.getNfccAlwaysOnAllowed(); mIsTagAppPrefSupported = mContext.getResources().getBoolean(R.bool.tag_intent_app_pref_supported); mTagAppBlockListHash = mPrefs.getInt(PREF_TAG_APP_BLOCK_LIST_HASH, PREF_TAG_APP_BLOCK_LIST_HASH_DEFAULT); // Get default blocked package list from resource file TAG_APP_BLOCKLIST.addAll( Arrays.asList(mContext.getResources().getStringArray( R.array.tag_intent_blocked_app_list))); Uri uri = Settings.Global.getUriFor(Constants.SETTINGS_SATELLITE_MODE_ENABLED); if (uri == null) { Log.e(TAG, "NfcService(constructor): satellite mode key does not exist in Settings"); } else { mContext.getContentResolver().registerContentObserver( uri, false, new ContentObserver(null) { @Override public void onChange(boolean selfChange) { if (mNfcInjector.isSatelliteModeSensitive()) { Log.i(TAG, "NfcService(constructor): Satellite mode change detected"); if (isTaskBootCompleted()) { if (shouldEnableNfc()) { new EnableDisableTask().execute(TASK_ENABLE); } else { new EnableDisableTask().execute(TASK_DISABLE); } } else { Log.i(TAG, "NfcService(constructor): Satellite mode change " + "detected, skip NFC init is not completed"); } } } }); } mIsNfcUserRestricted = isNfcUserRestricted(); mIsNfcUserChangeRestricted = isNfcUserChangeRestricted(); mContext.registerReceiver( new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { mIsNfcUserChangeRestricted = isNfcUserChangeRestricted(); if (mIsNfcUserRestricted == isNfcUserRestricted()) { return; } Log.i(TAG, "NfcService(constructor): Disallow NFC user restriction " + "changed from " + mIsNfcUserRestricted + " to " + !mIsNfcUserRestricted + "."); mIsNfcUserRestricted = !mIsNfcUserRestricted; if (isTaskBootCompleted()) { if (shouldEnableNfc()) { new EnableDisableTask().execute(TASK_ENABLE); } else { new EnableDisableTask().execute(TASK_DISABLE); } } else { Log.i(TAG, "restriction change detected - skip NFC init is not completed"); } } }, new IntentFilter(UserManager.ACTION_USER_RESTRICTIONS_CHANGED) ); mNfcPermissions = new NfcPermissions(mContext); mReaderOptionCapable = mDeviceConfigFacade.isReaderOptionCapable(); if (mReaderOptionCapable) { mIsReaderOptionEnabled = mPrefs.getBoolean(PREF_NFC_READER_OPTION_ON, mDeviceConfigFacade.getDefaultReaderOption() || mInProvisionMode); } executeTaskBoot(); // do blocking boot tasks if ((NFC_SNOOP_LOG_MODE.equals(NfcProperties.snoop_log_mode_values.FULL) || NFC_VENDOR_DEBUG_ENABLED) && mDeviceConfigFacade.getEnableDeveloperNotification()) { new NfcDeveloperOptionNotification(mContext).startNotification(); } connectToSeService(); } private static Boolean isTaskBootCompleted() { return NfcProperties.initialized().orElse(Boolean.FALSE); } private void executeTaskBoot() { // If overlay is set, delay the NFC boot up until the OEM extension indicates it is ready to // proceed with NFC bootup. if (mDeviceConfigFacade.getEnableOemExtension()) { // Send intent for OEM extension to initialize. Intent intent = new Intent(NfcOemExtension.ACTION_OEM_EXTENSION_INIT); mContext.sendBroadcastAsUser(intent, UserHandle.CURRENT, BIND_NFC_SERVICE); Log.i(TAG, "executeTaskBoot: Sent intent for OEM extension to initialize"); return; } new EnableDisableTask().execute(TASK_BOOT); } private List getEnabledUserIds() { List userIds = new ArrayList(); UserManager um = mContext.createContextAsUser(UserHandle.of(ActivityManager.getCurrentUser()), 0) .getSystemService(UserManager.class); List luh = um.getEnabledProfiles(); for (UserHandle uh : luh) { userIds.add(uh.getIdentifier()); } return userIds; } private void initTagAppPrefList() { if (!mIsTagAppPrefSupported) return; boolean force = mTagAppBlockListHash != TAG_APP_BLOCKLIST.hashCode(); if (force) { mTagAppBlockListHash = TAG_APP_BLOCKLIST.hashCode(); mPrefsEditor.putInt(PREF_TAG_APP_BLOCK_LIST_HASH, mTagAppBlockListHash); mPrefsEditor.apply(); } mTagAppPrefList.clear(); mTagAppPrefListPrefs = mContext.getSharedPreferences(PREF_TAG_APP_LIST, Context.MODE_PRIVATE); boolean changed = false; if (mTagAppPrefListPrefs == null) { Log.e(TAG, "initTagAppPrefList: Can't get PREF_TAG_APP_LIST"); return; } try { for (Integer userId : getEnabledUserIds()) { HashMap map = new HashMap<>(); String jsonString = mTagAppPrefListPrefs.getString(Integer.toString(userId), (new JSONObject()).toString()); if (jsonString != null) { JSONObject jsonObject = new JSONObject(jsonString); Iterator keysItr = jsonObject.keys(); while (keysItr.hasNext()) { String key = keysItr.next(); Boolean value = jsonObject.getBoolean(key); map.put(key, value); if (DBG) { Log.d(TAG, "initTagAppPrefList: uid:" + userId + "key:" + key + ": " + value); } } } // Put default blocked pkgs for (String pkg : TAG_APP_BLOCKLIST) { if ((force || !map.containsKey(pkg)) && isPackageInstalled(pkg, userId)) { map.put(pkg, false); changed = true; } } mTagAppPrefList.put(userId, map); } } catch (JSONException e) { Log.e(TAG, "initTagAppPrefList: JSONException=" + e); } if (changed) storeTagAppPrefList(); } private void storeTagAppPrefList() { if (!mIsTagAppPrefSupported) return; mTagAppPrefListPrefs = mContext.getSharedPreferences(PREF_TAG_APP_LIST, Context.MODE_PRIVATE); if (mTagAppPrefListPrefs != null) { for (Integer userId : getEnabledUserIds()) { SharedPreferences.Editor editor = mTagAppPrefListPrefs.edit(); HashMap map; synchronized (NfcService.this) { map = mTagAppPrefList.getOrDefault(userId, new HashMap<>()); } if (map.size() > 0) { String userIdStr = Integer.toString(userId); JSONObject jsonObject = new JSONObject(map); String jsonString = jsonObject.toString(); editor.remove(userIdStr).putString(userIdStr, jsonString).apply(); } } } else { Log.e(TAG, "storeTagAppPrefList: Can't get PREF_TAG_APP_LIST"); } } private boolean isPackageInstalled(String pkgName, int userId) { final PackageInfo info; try { info = mContext.createContextAsUser(UserHandle.of(userId), 0) .getPackageManager().getPackageInfo(pkgName, PackageManager.MATCH_ALL); } catch (PackageManager.NameNotFoundException e) { return false; } return info != null; } // Remove obsolete entries private void renewTagAppPrefList(String action) { if (!mIsTagAppPrefSupported) return; if (!action.equals(Intent.ACTION_PACKAGE_ADDED) && !action.equals(Intent.ACTION_PACKAGE_REMOVED)) return; boolean changed = false; for (Integer userId : getEnabledUserIds()) { synchronized (NfcService.this) { if (action.equals(Intent.ACTION_PACKAGE_ADDED)) { HashMap map = mTagAppPrefList.getOrDefault(userId, new HashMap<>()); for (String pkg : TAG_APP_BLOCKLIST) { if (!map.containsKey(pkg) && isPackageInstalled(pkg, userId)) { map.put(pkg, false); changed = true; mTagAppPrefList.put(userId, map); } } } else if (action.equals(Intent.ACTION_PACKAGE_REMOVED)) { changed |= mTagAppPrefList.getOrDefault(userId, new HashMap<>()) .keySet().removeIf(k2 -> !isPackageInstalled(k2, userId)); } } } if (DBG) Log.d(TAG, "renewTagAppPrefList: TagAppPreference changed " + changed); if (changed) storeTagAppPrefList(); } private boolean isSEServiceAvailable() { if (mSEService == null) { connectToSeService(); } return (mSEService != null); } private void connectToSeService() { try { mSEService = mNfcInjector.connectToSeService(); if (mSEService != null) { IBinder seServiceBinder = mSEService.asBinder(); seServiceBinder.linkToDeath(mSeServiceDeathRecipient, 0); } } catch (RemoteException e) { Log.e(TAG, "connectToSeService: e=" + e); } } void initSoundPoolIfNeededAndPlaySound(Runnable playSoundRunnable) { if (mSoundPool == null) { // For the first sound play which triggers the sound pool initialization, play the // sound after sound pool load is complete. OnLoadCompleteListener onLoadCompleteListener = new OnLoadCompleteListener() { private int mNumLoadComplete = 0; @Override public void onLoadComplete(SoundPool soundPool, int sampleId, int status) { // Check that both end/error sounds are loaded before playing the sound. if (++mNumLoadComplete == 2) { Log.d(TAG, "initSoundPoolIfNeededAndPlaySound: playing sound"); playSoundRunnable.run(); } } }; mSoundPool = new SoundPool.Builder() .setMaxStreams(1) .setAudioAttributes( new AudioAttributes.Builder() .setUsage(AudioAttributes.USAGE_ASSISTANCE_SONIFICATION) .setContentType(AudioAttributes.CONTENT_TYPE_SONIFICATION) .build()) .build(); mSoundPool.setOnLoadCompleteListener(onLoadCompleteListener); mEndSound = mSoundPool.load(mContext, R.raw.end, 1); mErrorSound = mSoundPool.load(mContext, R.raw.error, 1); } else { // sound pool already loaded, play the sound. Log.d(TAG, "initSoundPoolIfNeededAndPlaySound: Sound pool is already loaded, " + "playing sound"); playSoundRunnable.run(); } } void releaseSoundPool() { synchronized (this) { if (mSoundPool != null) { mSoundPool.release(); mSoundPool = null; } } } void updatePackageCache() { UserManager um = mContext.createContextAsUser( UserHandle.of(ActivityManager.getCurrentUser()), /*flags=*/0) .getSystemService(UserManager.class); List luh = um.getEnabledProfiles(); synchronized (this) { mNfcEventInstalledPackages.clear(); mNfcPreferredPaymentChangedInstalledPackages.clear(); for (UserHandle uh : luh) { if (um.isQuietModeEnabled(uh)) continue; PackageManager pm; try { pm = mContext.createContextAsUser(uh, /*flags=*/0).getPackageManager(); } catch (IllegalStateException e) { Log.d(TAG, "updatePackageCache: Fail to get PackageManager for user: " + uh); continue; } List packagesNfcEvents = pm.getPackagesHoldingPermissions( new String[] {android.Manifest.permission.NFC_TRANSACTION_EVENT}, PackageManager.GET_ACTIVITIES); List packagesNfcPreferredPaymentChanged = pm.getPackagesHoldingPermissions( new String[] {android.Manifest.permission.NFC_PREFERRED_PAYMENT_INFO}, PackageManager.GET_ACTIVITIES); List packageListNfcEvent = new ArrayList(); for (int i = 0; i < packagesNfcEvents.size(); i++) { packageListNfcEvent.add(packagesNfcEvents.get(i).packageName); } mNfcEventInstalledPackages.put(uh.getIdentifier(), packageListNfcEvent); List packageListNfcPreferredPaymentChanged = new ArrayList(); for (int i = 0; i < packagesNfcPreferredPaymentChanged.size(); i++) { packageListNfcPreferredPaymentChanged.add( packagesNfcPreferredPaymentChanged.get(i).packageName); } mNfcPreferredPaymentChangedInstalledPackages.put( uh.getIdentifier(), packageListNfcPreferredPaymentChanged); } } } /** * Manages tasks that involve turning on/off the NFC controller. *

*

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). *

*

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. *

*

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. *

*

{@link #TASK_ENABLE} enables the NFC adapter, without changing * preferences *

{@link #TASK_DISABLE} disables the NFC adapter, without changing * preferences *

{@link #TASK_BOOT} does first boot work and may enable NFC */ class EnableDisableTask extends AsyncTask { int action; @Override protected Boolean doInBackground(Integer... params) { // Quick 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 false; } action = params[0].intValue(); boolean result = true; /* 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 (action) { case TASK_ENABLE: if (shouldEnableNfc()) { onOemPreExecute(); result = enableInternal(); if (sIsNfcRestore && mIsTagAppPrefSupported) { synchronized (NfcService.this) { initTagAppPrefList(); sIsNfcRestore = false; } } } else { result = false; } break; case TASK_DISABLE: if (allowOemDisable()) { onOemPreExecute(); result = disableInternal(); } else { result = false; } break; case TASK_BOOT: // Initialize the event log cache. boolean initialized; if (mPrefs.getBoolean(PREF_FIRST_BOOT, true)) { Log.i(TAG, "First Boot"); mPrefsEditor.putBoolean(PREF_FIRST_BOOT, false); mPrefsEditor.apply(); mDeviceHost.factoryReset(); setPaymentForegroundPreference(mUserId); } Log.d(TAG, "checking on firmware download"); boolean enableNfc = shouldEnableNfc(); onOemPreExecute(); if (enableNfc) { Log.d(TAG, "NFC is on. Doing normal stuff"); initialized = enableInternal(); } else { Log.d(TAG, "NFC is off. Checking firmware version"); initialized = mDeviceHost.checkFirmware(); } mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setBootupState(NfcEventProto.NfcBootupState.newBuilder() .setEnabled(enableNfc) .build()) .build()); if (initialized) { // TODO(279846422) The system property will be temporary // available for vendors that depend on it. // Remove this code when a replacement API is added. NfcProperties.initialized(true); } synchronized (NfcService.this) { initTagAppPrefList(); } result = initialized; break; case TASK_ENABLE_ALWAYS_ON: /* Get mode from AsyncTask params */ result = enableAlwaysOnInternal(params[1]); break; case TASK_DISABLE_ALWAYS_ON: result = disableAlwaysOnInternal(); break; default: break; } // Restore default AsyncTask priority Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND); return result; } @Override protected void onPostExecute(Boolean result) { Log.d(TAG, "onPostExecute: result - " + result); if (mNfcOemExtensionCallback != null) { try { if (action == TASK_BOOT) mNfcOemExtensionCallback .onBootFinished(result ? STATUS_OK : STATUS_UNKNOWN_ERROR); else if (action == TASK_ENABLE) mNfcOemExtensionCallback .onEnableFinished(result ? STATUS_OK : STATUS_UNKNOWN_ERROR); else if (action == TASK_DISABLE) mNfcOemExtensionCallback .onDisableFinished(result ? STATUS_OK : STATUS_UNKNOWN_ERROR); } catch (RemoteException remoteException) { Log.e(TAG, "onPostExecute: Failed to call remote oem extension callback"); } } } void onOemPreExecute() { if (mNfcOemExtensionCallback != null) { try { if (action == TASK_BOOT) mNfcOemExtensionCallback.onBootStarted(); else if (action == TASK_ENABLE) mNfcOemExtensionCallback.onEnableStarted(); else if (action == TASK_DISABLE) mNfcOemExtensionCallback.onDisableStarted(); } catch (RemoteException remoteException) { Log.e(TAG, "onOemPreExecute: Failed to call remote oem extension callback"); } } } boolean isAlwaysOnInDefaultMode() { return mAlwaysOnMode == NfcOemExtension.ENABLE_DEFAULT; } /** * Enable NFC adapter functions. * Does not toggle preferences. */ boolean enableInternal() { if (DBG) Log.d(TAG, "EnableDisableTask.enableInternal: begin"); if (mState == NfcAdapter.STATE_ON) { return true; } else if (mAlwaysOnState == NfcAdapter.STATE_ON) { if (!isAlwaysOnInDefaultMode()) { Log.i(TAG, "enableInternal: ControllerAlwaysOn Not In DEFAULT_MODE " + "- disableAlwaysOn!"); disableAlwaysOnInternal(); } } Log.i(TAG, "Enabling NFC"); NfcStatsLog.write(NfcStatsLog.NFC_STATE_CHANGED, mIsSecureNfcEnabled ? NfcStatsLog.NFC_STATE_CHANGED__STATE__ON_LOCKED : NfcStatsLog.NFC_STATE_CHANGED__STATE__ON); updateState(NfcAdapter.STATE_TURNING_ON); WatchDogThread watchDog = new WatchDogThread("enableInternal", INIT_WATCHDOG_MS); watchDog.start(); mCardEmulationManager.updateForDefaultSwpToEuicc(); try { mRoutingWakeLock.acquire(); try { if (!mIsAlwaysOnSupported || mIsRecovering || (mAlwaysOnState != NfcAdapter.STATE_ON && mAlwaysOnState != NfcAdapter.STATE_TURNING_OFF)) { if (mIsRecovering) { // Recovering needs the full init. Put default value mAlwaysOnState = NfcAdapter.STATE_OFF; } if (!mDeviceHost.initialize()) { Log.w(TAG, "enableInternal: Error enabling NFC"); updateState(NfcAdapter.STATE_OFF); return false; } } else if (mAlwaysOnState == NfcAdapter.STATE_ON || mAlwaysOnState == NfcAdapter.STATE_TURNING_OFF) { Log.i(TAG, "enableInternal: Already initialized"); } else { Log.e(TAG, "enableInternal: Unexpected bad state " + mAlwaysOnState); updateState(NfcAdapter.STATE_OFF); return false; } } finally { if (mRoutingWakeLock.isHeld()) { mRoutingWakeLock.release(); } } } finally { watchDog.cancel(); } mSkipNdefRead = NfcProperties.skipNdefRead().orElse(false); nci_version = getNciVersion(); Log.d(TAG, "enableInternal: NCI_Version: " + nci_version); mPendingPowerStateUpdate = false; synchronized (NfcService.this) { mObjectMap.clear(); updateState(NfcAdapter.STATE_ON); onPreferredPaymentChanged(NfcAdapter.PREFERRED_PAYMENT_LOADED); } if (mInProvisionMode) { mScreenState = mScreenStateHelper.checkScreenStateProvisionMode(); } else { mScreenState = mScreenStateHelper.checkScreenState(mCheckDisplayStateForScreenState); } int screen_state_mask = (mNfcUnlockManager.isLockscreenPollingEnabled()) ? (ScreenStateHelper.SCREEN_POLLING_TAG_MASK | mScreenState) : mScreenState; if (mNfcUnlockManager.isLockscreenPollingEnabled()) applyRouting(false); mDeviceHost.doSetScreenState(screen_state_mask, mIsWlcEnabled); sToast_debounce = false; restoreSavedTech(); /* Skip applyRouting if always on state is switching */ if (!mIsAlwaysOnSupported || (mAlwaysOnState != NfcAdapter.STATE_TURNING_ON && mAlwaysOnState != NfcAdapter.STATE_TURNING_OFF)) { /* Start polling loop */ applyRouting(true); } if (mIsHceCapable) { // Generate the initial card emulation routing table mCardEmulationManager.onNfcEnabled(); } if (mIsRecovering) { // Intents for all users registerGlobalBroadcastsReceiver(); mIsRecovering = false; } if (mIsPowerSavingModeEnabled) { mDeviceHost.setPowerSavingMode(false); mIsPowerSavingModeEnabled = false; } if (DBG) Log.d(TAG, "EnableDisableTask.enableInternal: end"); return true; } /** * Disable all NFC adapter functions. * Does not toggle preferences. */ boolean disableInternal() { if (DBG) Log.d(TAG, "EnableDisableTask.disableInternal: beging"); if (mState == NfcAdapter.STATE_OFF) { return true; } Log.i(TAG, "Disabling NFC"); NfcStatsLog.write( NfcStatsLog.NFC_STATE_CHANGED, NfcStatsLog.NFC_STATE_CHANGED__STATE__OFF); 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("disableInternal", ROUTING_WATCHDOG_MS); watchDog.start(); if (mIsWlcEnabled) { if (mNfcCharging.NfcChargingOnGoing == true) { mNfcCharging.disconnectNfcCharging(); mNfcCharging.NfcChargingOnGoing = false; } mNfcCharging.resetInternalValues(); } if (mIsHceCapable) { mCardEmulationManager.onNfcDisabled(); } // 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. maybeDisconnectTarget(); synchronized (NfcService.this) { // Disable delay polling when disabling mPollDelayed = false; mPollDelayCount = 0; mReadErrorCount = 0; mHandler.removeMessages(MSG_DELAY_POLLING); mPollingDisableDeathRecipients.clear(); mReaderModeParams = null; mDiscoveryTechParams = null; } mNfcDispatcher.resetForegroundDispatch(); boolean result; if (!mIsAlwaysOnSupported || mIsRecovering || (mAlwaysOnState == NfcAdapter.STATE_OFF) || (mAlwaysOnState == NfcAdapter.STATE_TURNING_OFF)) { result = mDeviceHost.deinitialize(); if (DBG) Log.d(TAG, "disableInternal: mDeviceHost.deinitialize() = " + result); } else { mDeviceHost.disableDiscovery(); result = true; Log.i(TAG, "disableInternal: AlwaysOn set, disableDiscovery()"); } watchDog.cancel(); synchronized (NfcService.this) { mCurrentDiscoveryParameters = NfcDiscoveryParameters.getNfcOffParameters(); updateState(NfcAdapter.STATE_OFF); } releaseSoundPool(); if (DBG) Log.d(TAG, "EnableDisableTask.disableInternal: end"); return result; } /** * Enable always on feature. */ boolean enableAlwaysOnInternal(int mode) { if (mAlwaysOnState == NfcAdapter.STATE_ON) { return true; } else if (mState == NfcAdapter.STATE_TURNING_ON || mAlwaysOnState == NfcAdapter.STATE_TURNING_OFF) { Log.e(TAG, "enableAlwaysOnInternal: Processing from bad state"); return false; } else if (mState == NfcAdapter.STATE_ON) { updateAlwaysOnState(NfcAdapter.STATE_TURNING_ON); mDeviceHost.setNfceePowerAndLinkCtrl(true); updateAlwaysOnState(NfcAdapter.STATE_ON); } else if (mState == NfcAdapter.STATE_OFF) { /* Special case when NFCC is OFF without initialize. * Temporarily enable NfcAdapter but don't applyRouting. * Then disable NfcAdapter without deinitialize to keep the NFCC stays initialized. * mState will switch back to OFF in the end. * And the NFCC stays initialized. */ updateAlwaysOnState(NfcAdapter.STATE_TURNING_ON); if (mode != NfcOemExtension.ENABLE_DEFAULT) { mDeviceHost.setPartialInitMode(mode); mAlwaysOnMode = mode; } if (!enableInternal()) { updateAlwaysOnState(NfcAdapter.STATE_OFF); return false; } disableInternal(); mDeviceHost.setNfceePowerAndLinkCtrl(true); updateAlwaysOnState(NfcAdapter.STATE_ON); } return true; } /** * Disable always on feature. */ boolean disableAlwaysOnInternal() { if (mAlwaysOnState == NfcAdapter.STATE_OFF) { return true; } else if ((mState == NfcAdapter.STATE_TURNING_ON || mAlwaysOnState == NfcAdapter.STATE_TURNING_OFF) && (!(mAlwaysOnState == NfcAdapter.STATE_ON))) { if (!isAlwaysOnInDefaultMode()) { Log.e(TAG, "disableAlwaysOnInternal: Processing from bad state"); return false; } } else if (mState == NfcAdapter.STATE_ON) { updateAlwaysOnState(NfcAdapter.STATE_TURNING_OFF); mDeviceHost.setNfceePowerAndLinkCtrl(false); updateAlwaysOnState(NfcAdapter.STATE_OFF); } else if (mState == NfcAdapter.STATE_OFF || (mAlwaysOnState == NfcAdapter.STATE_ON)) { /* Special case when mState is OFF but NFCC is already initialized. * Deinitialize mDevicehost directly. */ updateAlwaysOnState(NfcAdapter.STATE_TURNING_OFF); mDeviceHost.setNfceePowerAndLinkCtrl(false); boolean result = mDeviceHost.deinitialize(); if (DBG) { Log.d(TAG, "disableAlwaysOnInternal: mDeviceHost.deinitialize() = " + result); } updateAlwaysOnState(NfcAdapter.STATE_OFF); return result; } return true; } void updateState(int newState) { synchronized (NfcService.this) { if (newState == mState) { return; } mState = newState; if (android.nfc.Flags.nfcEventListener() && mCardEmulationManager != null) { mCardEmulationManager.onNfcStateChanged(newState); } if (mState == NfcAdapter.STATE_ON && mCardEmulationManager != null) { mCardEmulationManager.updateForShouldDefaultToObserveMode(getUserId()); mCardEmulationManager.updateFirmwareExitFramesForWalletRole(getUserId()); } if (mAlwaysOnState != NfcAdapter.STATE_TURNING_ON) { 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.sendBroadcastAsUser(intent, UserHandle.CURRENT); if (mNfcOemExtensionCallback != null) { try { mNfcOemExtensionCallback.onStateUpdated(mState); } catch (RemoteException remoteException) { Log.e(TAG, "updateState: Failed to invoke onStateUpdated " + "oem callback"); } } } } } void updateAlwaysOnState(int newState) { synchronized (NfcService.this) { if (newState == mAlwaysOnState) { return; } if (newState == NfcAdapter.STATE_OFF) { mAlwaysOnMode = NfcOemExtension.ENABLE_DEFAULT; mDeviceHost.setPartialInitMode(NfcOemExtension.ENABLE_DEFAULT); } mAlwaysOnState = newState; if (mAlwaysOnState == NfcAdapter.STATE_OFF || mAlwaysOnState == NfcAdapter.STATE_ON) { synchronized (mAlwaysOnListeners) { for (INfcControllerAlwaysOnListener listener : mAlwaysOnListeners) { try { listener.onControllerAlwaysOnChanged( mAlwaysOnState == NfcAdapter.STATE_ON); } catch (RemoteException e) { Log.e(TAG, "updateAlwaysOnState: error"); } } } } } } } private void clearListenTech(boolean keepListenTech) { if (getNfcListenTech() != DEFAULT_LISTEN_TECH) { int listenTech = -1; if (keepListenTech) { Log.d(TAG, "clearListenTech: keep listenTech"); listenTech = NfcAdapter.FLAG_LISTEN_KEEP; } else { Log.d(TAG, "clearListenTech: clear listenTech"); listenTech = (NfcAdapter.FLAG_LISTEN_KEEP | NfcAdapter.FLAG_USE_ALL_TECH | NfcAdapter.FLAG_SET_DEFAULT_TECH); } mDeviceHost.setDiscoveryTech(NfcAdapter.FLAG_READER_KEEP, listenTech); } } private void restoreSavedTech() { Log.i(TAG, "restoreSavedTech"); int pollTech = -1; if (mPrefs.contains(PREF_POLL_TECH)) { pollTech = getNfcPollTech(); } int listenTech = -1; if (mPrefs.contains(PREF_LISTEN_TECH)) { listenTech = getNfcListenTech(); } if (listenTech == -1 || listenTech == DEFAULT_LISTEN_TECH) listenTech = (NfcAdapter.FLAG_LISTEN_KEEP|NfcAdapter.FLAG_USE_ALL_TECH); if (pollTech == -1 || pollTech == DEFAULT_POLL_TECH) pollTech = (NfcAdapter.FLAG_READER_KEEP|NfcAdapter.FLAG_USE_ALL_TECH); mDeviceHost.setDiscoveryTech(pollTech|NfcAdapter.FLAG_SET_DEFAULT_TECH, listenTech|NfcAdapter.FLAG_SET_DEFAULT_TECH); } public void playSound(int sound) { synchronized (this) { switch (sound) { case SOUND_END: // Lazy init sound pool when needed. initSoundPoolIfNeededAndPlaySound(() -> { int playReturn = mSoundPool.play(mEndSound, 1.0f, 1.0f, 0, 0, 1.0f); Log.d(TAG, "playSound: Sound pool play return: " + playReturn); }); break; case SOUND_ERROR: // Lazy init sound pool when needed. initSoundPoolIfNeededAndPlaySound(() -> { int playReturn = mSoundPool.play(mErrorSound, 1.0f, 1.0f, 0, 0, 1.0f); Log.d(TAG, "playSound: Sound pool play return: " + playReturn); }); break; } } } synchronized int getUserId() { return mUserId; } private void resetReaderModeParams() { synchronized (NfcService.this) { if (mPollingDisableDeathRecipients.size() == 0) { Log.d(TAG, "resetReaderModeParams: Disabling reader mode because app died" + " or moved to background"); mReaderModeParams = null; StopPresenceChecking(); // listenTech is different from the default value, the stored listenTech will be included. // When using enableReaderMode, change listenTech to default & restore to the previous value. if (isNfcEnabled()) { restoreSavedTech(); } mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setReaderModeChange(NfcEventProto.NfcReaderModeChange.newBuilder() .setFlags(0) .build()) .build()); if (isNfcEnabled()) { applyRouting(false); } } } } @Override public void onUidToBackground(int uid) { Log.i(TAG, "onUidToBackground: Uid " + uid); synchronized (NfcService.this) { if (mReaderModeParams != null && mReaderModeParams.uid == uid) { mReaderModeParams.binder.unlinkToDeath(mReaderModeDeathRecipient, 0); resetReaderModeParams(); } if (mDiscoveryTechParams != null && mDiscoveryTechParams.uid == uid) { mDiscoveryTechParams.binder.unlinkToDeath(mDiscoveryTechDeathRecipient, 0); mDeviceHost.resetDiscoveryTech(); mDiscoveryTechParams = null; if (isNfcEnabled()) { applyRouting(true); } } } } public void enableNfc() { saveNfcOnSetting(true); new EnableDisableTask().execute(TASK_ENABLE); } private @NonNull CharSequence getAppName(@NonNull String packageName, int uid) { ApplicationInfo applicationInfo = null; try { applicationInfo = mContext.getPackageManager().getApplicationInfoAsUser( packageName, 0, UserHandle.getUserHandleForUid(uid)); } catch (PackageManager.NameNotFoundException e) { Log.e(TAG, "getAppName: Failed for " + packageName); return ""; } return mContext.getPackageManager().getApplicationLabel(applicationInfo); } public boolean isSecureNfcEnabled() { return mIsSecureNfcEnabled; } /** Helper method to check if the entity initiating the binder call is a DO/PO app. */ private boolean isDeviceOrProfileOwner(int uid, String packageName) { return mNfcPermissions.isDeviceOwner(uid, packageName) || mNfcPermissions.isProfileOwner(uid, packageName); } final class NfcAdapterService extends INfcAdapter.Stub { @Override public boolean enable(String pkg) throws RemoteException { if (Flags.checkPassedInPackage()) { mNfcPermissions.checkPackage(Binder.getCallingUid(), pkg); } boolean isDeviceOrProfileOwner = isDeviceOrProfileOwner(Binder.getCallingUid(), pkg); if (!NfcPermissions.checkAdminPermissions(mContext) && !isDeviceOrProfileOwner) { throw new SecurityException( "caller is not a system app, device owner or profile owner!"); } if (!isDeviceOrProfileOwner && mIsNfcUserChangeRestricted) { throw new SecurityException("Change nfc state by system app is not allowed!"); } if (!isTaskBootCompleted()) { Log.e(TAG, "enable: NFC is not initialized yet:" + isTaskBootCompleted()); return false; } notifyOemLogEvent(new OemLogItems.Builder(OemLogItems.LOG_ACTION_NFC_TOGGLE) .setCallingPid(Binder.getCallingPid()) .setCallingEvent(EVENT_ENABLE) .build()); Log.i(TAG, "enable: enabling, package:" + pkg); List allowlist = new ArrayList<>( Arrays.asList(mContext.getResources().getStringArray(R.array.nfc_allow_list))); if (!allowlist.isEmpty() && !allowlist.contains(pkg)) { Intent allowUsingNfcIntent = new Intent() .putExtra(APP_NAME_ENABLING_NFC, getAppName(pkg, getUserId())) .setClass(mContext, NfcEnableAllowlistActivity.class); mContext.startActivityAsUser(allowUsingNfcIntent, UserHandle.CURRENT); return true; } mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setStateChange(NfcEventProto.NfcStateChange.newBuilder() .setAppInfo(NfcEventProto.NfcAppInfo.newBuilder() .setPackageName(pkg) .setUid(Binder.getCallingUid()) .build()) .setEnabled(true) .build()) .build()); if (android.nfc.Flags.nfcStateChangeSecurityLogEventEnabled()) { SecurityLog.writeEvent(SecurityLog.TAG_NFC_ENABLED); } enableNfc(); return true; } @Override public boolean disable(boolean saveState, String pkg) throws RemoteException { if (Flags.checkPassedInPackage()) { mNfcPermissions.checkPackage(Binder.getCallingUid(), pkg); } boolean isDeviceOrProfileOwner = isDeviceOrProfileOwner(Binder.getCallingUid(), pkg); if (!NfcPermissions.checkAdminPermissions(mContext) && !isDeviceOrProfileOwner) { throw new SecurityException( "caller is not a system app, device owner or profile owner!"); } if (!isDeviceOrProfileOwner && mIsNfcUserChangeRestricted) { throw new SecurityException("Change nfc state by system app is not allowed!"); } notifyOemLogEvent(new OemLogItems.Builder(OemLogItems.LOG_ACTION_NFC_TOGGLE) .setCallingPid(Binder.getCallingPid()) .setCallingEvent(EVENT_DISABLE) .build()); Log.i(TAG, "disable: disabling, package:" + pkg); if (saveState) { saveNfcOnSetting(false); } mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setStateChange(NfcEventProto.NfcStateChange.newBuilder() .setAppInfo(NfcEventProto.NfcAppInfo.newBuilder() .setPackageName(pkg) .setUid(Binder.getCallingUid()) .build()) .setEnabled(false) .build()) .build()); if (android.nfc.Flags.nfcStateChangeSecurityLogEventEnabled()) { SecurityLog.writeEvent(SecurityLog.TAG_NFC_DISABLED); } new EnableDisableTask().execute(TASK_DISABLE); return true; } @Override public boolean isReaderModeAnnotationSupported() { return mDeviceHost.isReaderModeAnnotationSupported(); } @Override public boolean isObserveModeSupported() { if (!isNfcEnabled()) { Log.e(TAG, "isObserveModeSupported: NFC must be enabled but is: " + mState); return false; } return mDeviceHost.isObserveModeSupported(); } @Override public boolean isObserveModeEnabled() { synchronized (NfcService.this) { if (!isNfcEnabled()) { Log.e(TAG, "isObserveModeEnabled: NFC must be enabled but is: " + mState); return false; } NfcPermissions.enforceUserPermissions(mContext); return mDeviceHost.isObserveModeEnabled(); } } @Override public boolean setObserveMode(boolean enable, String packageName) { synchronized (NfcService.this) { if (Flags.checkPassedInPackage()) { mNfcPermissions.checkPackage(Binder.getCallingUid(), packageName); } if (!isNfcEnabled()) { Log.e(TAG, "setObserveMode: NFC must be enabled but is: " + mState); return false; } int callingUid = Binder.getCallingUid(); UserHandle callingUser = Binder.getCallingUserHandle(); int triggerSource = StatsdUtils.TRIGGER_SOURCE_UNKNOWN; if (!NfcInjector.isPrivileged(callingUid)) { NfcPermissions.enforceUserPermissions(mContext); if (packageName == null) { Log.e(TAG, "setObserveMode: no package name associated with " + "non-privileged calling UID"); } if (mCardEmulationManager.isPreferredServicePackageNameForUser( packageName, callingUser.getIdentifier())) { if (android.permission.flags.Flags.walletRoleEnabled()) { if (packageName != null) { triggerSource = packageName.equals(getWalletRoleHolder(callingUser)) ? StatsdUtils.TRIGGER_SOURCE_WALLET_ROLE_HOLDER : StatsdUtils.TRIGGER_SOURCE_FOREGROUND_APP; } } else { if (mForegroundUtils.isInForeground(callingUid)) { triggerSource = StatsdUtils.TRIGGER_SOURCE_FOREGROUND_APP; } } } else { Log.e(TAG, "setObserveMode: Caller not preferred NFC service."); return false; } } if (mCardEmulationManager.isHostCardEmulationActivated()) { Log.w(TAG, "setObserveMode: Cannot set observe mode during a transaction."); return false; } Log.d( TAG, "setObserveMode: package " + packageName + " with UID (" + callingUid + ") setting observe mode to " + enable); long start = SystemClock.elapsedRealtime(); Trace.beginSection("setObserveMode: " + enable); boolean result = mDeviceHost.setObserveMode(enable); Trace.endSection(); int latency = Math.toIntExact(SystemClock.elapsedRealtime() - start); if (mStatsdUtils != null) { mStatsdUtils.logObserveModeStateChanged(enable, triggerSource, latency); } mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setObserveModeChange( NfcEventProto.NfcObserveModeChange.newBuilder() .setAppInfo( NfcEventProto.NfcAppInfo.newBuilder() .setPackageName(packageName) .setUid(callingUid) .build()) .setEnable(enable) .setLatencyMs(latency) .setResult(result) .build()) .build()); return result; } } private String getWalletRoleHolder(UserHandle user) { RoleManager roleManager = mContext.createContextAsUser(user, 0) .getSystemService(RoleManager.class); List roleHolders = roleManager.getRoleHolders(RoleManager.ROLE_WALLET); return roleHolders.isEmpty() ? null : roleHolders.get(0); } @Override public int pausePolling(long timeoutInMs) { NfcPermissions.enforceAdminPermissions(mContext); checkAndHandleRemovalDetectionMode(false); synchronized (mDiscoveryLock) { if (!mRfDiscoveryStarted) { if (DBG) Log.d(TAG, "pausePolling: already disabled!"); return NfcOemExtension.POLLING_STATE_CHANGE_ALREADY_IN_REQUESTED_STATE; } } synchronized (NfcService.this) { mPollingPaused = true; mDeviceHost.disableDiscovery(); /* timeoutInMs 0 will stop discovery without any timeout * polling will not auto resume */ if (timeoutInMs == 0) { if (DBG) Log.d(TAG, "pausePolling: without timeout"); return NfcOemExtension.POLLING_STATE_CHANGE_SUCCEEDED; } if (timeoutInMs < 0 || timeoutInMs > this.getMaxPausePollingTimeoutMs()) { throw new IllegalArgumentException( "Invalid timeout " + timeoutInMs + " ms!"); } mHandler.sendMessageDelayed( mHandler.obtainMessage(MSG_RESUME_POLLING), timeoutInMs); return NfcOemExtension.POLLING_STATE_CHANGE_SUCCEEDED; } } @Override public int resumePolling() { NfcPermissions.enforceAdminPermissions(mContext); boolean rfDiscoveryStarted; synchronized (mDiscoveryLock) { rfDiscoveryStarted = mRfDiscoveryStarted; } synchronized (NfcService.this) { if (!mPollingPaused) { if (rfDiscoveryStarted) { if (DBG) Log.d(TAG, "resumePolling: already enabled!"); return NfcOemExtension.POLLING_STATE_CHANGE_ALREADY_IN_REQUESTED_STATE; } else { if (DBG) Log.d(TAG, "resumePolling: Enable explicitly!"); } } mHandler.removeMessages(MSG_RESUME_POLLING); mPollingPaused = false; new ApplyRoutingTask().execute(); if (DBG) Log.d(TAG, "resumePolling: done"); return NfcOemExtension.POLLING_STATE_CHANGE_SUCCEEDED; } } @Override public boolean isNfcSecureEnabled() throws RemoteException { synchronized (NfcService.this) { return mIsSecureNfcEnabled; } } @Override public boolean setNfcSecure(boolean enable) { NfcPermissions.enforceAdminPermissions(mContext); if (mNfcInjector.isDeviceLocked() && !enable) { Log.i(TAG, "setNfcSecure: Device need to be unlocked before setting Secure NFC OFF"); return false; } synchronized (NfcService.this) { if (mIsSecureNfcEnabled == enable) { Log.e(TAG, "setNfcSecure: error, can't apply the same state twice!"); return false; } Log.i(TAG, "setNfcSecure: " + enable); mPrefsEditor.putBoolean(PREF_SECURE_NFC_ON, enable); mPrefsEditor.apply(); mIsSecureNfcEnabled = enable; mBackupManager.dataChanged(); mDeviceHost.setNfcSecure(enable); if (android.nfc.Flags.nfcPersistLog()) { mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setSecureChange( NfcEventProto.NfcSecureChange.newBuilder() .setEnable(enable) .build()) .build()); } if (mIsHceCapable) { // update HCE/HCEF routing and commitRouting if Nfc is enabled mCardEmulationManager.onTriggerRoutingTableUpdate(); } else if (isNfcEnabled()) { // commit only tech/protocol route without HCE support mDeviceHost.commitRouting(); } } NfcStatsLog.write(NfcStatsLog.NFC_STATE_CHANGED, mIsSecureNfcEnabled ? NfcStatsLog.NFC_STATE_CHANGED__STATE__ON_LOCKED : NfcStatsLog.NFC_STATE_CHANGED__STATE__ON); return true; } @Override public void setForegroundDispatch(PendingIntent intent, IntentFilter[] filters, TechListParcel techListsParcel) { NfcPermissions.enforceUserPermissions(mContext); if (!mForegroundUtils.isInForeground(Binder.getCallingUid())) { Log.e(TAG, "setForegroundDispatch: Caller not in foreground."); return; } // Short-cut the disable path if (intent == null && filters == null && techListsParcel == null) { mNfcDispatcher.resetForegroundDispatch(); 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 setAppCallback(IAppCallback callback) { NfcPermissions.enforceUserPermissions(mContext); } @Override public boolean ignore(int nativeHandle, int debounceMs, ITagRemovedCallback callback) throws RemoteException { NfcPermissions.enforceUserPermissions(mContext); if (nativeHandle == MOCK_NATIVE_HANDLE || (debounceMs == 0 && mDebounceTagNativeHandle != INVALID_NATIVE_HANDLE && nativeHandle == mDebounceTagNativeHandle)) { // Remove any previous messages and immediately debounce. mHandler.removeMessages(MSG_TAG_DEBOUNCE); synchronized (NfcService.this) { mDebounceTagRemovedCallback = callback; } mHandler.sendEmptyMessage(MSG_TAG_DEBOUNCE); return true; } TagEndpoint tag = (TagEndpoint) findAndRemoveObject(nativeHandle); if (tag != null) { // Store UID and params int uidLength = tag.getUid().length; synchronized (NfcService.this) { mDebounceTagDebounceMs = debounceMs; mDebounceTagNativeHandle = nativeHandle; mDebounceTagUid = new byte[uidLength]; mDebounceTagRemovedCallback = callback; System.arraycopy(tag.getUid(), 0, mDebounceTagUid, 0, uidLength); } // Disconnect from this tag; this should resume the normal // polling loop (and enter listen mode for a while), before // we pick up any tags again. tag.disconnect(); mHandler.sendEmptyMessageDelayed(MSG_TAG_DEBOUNCE, debounceMs); return true; } else { return false; } } @Override public void verifyNfcPermission() { NfcPermissions.enforceUserPermissions(mContext); } @Override public INfcTag getNfcTagInterface() throws RemoteException { return mNfcTagService; } @Override public INfcCardEmulation getNfcCardEmulationInterface() { if (mIsHceCapable) { return mCardEmulationManager.getNfcCardEmulationInterface(); } else { return null; } } @Override public INfcFCardEmulation getNfcFCardEmulationInterface() { if (mIsHceFCapable) { return mCardEmulationManager.getNfcFCardEmulationInterface(); } else { return null; } } @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 { NfcPermissions.enforceAdminPermissions(mContext); mNfcDispatcher.dispatchTag(tag); } @Override public void updateDiscoveryTechnology( IBinder binder, int pollTech, int listenTech, String packageName) throws RemoteException { if (Flags.checkPassedInPackage()) { mNfcPermissions.checkPackage(Binder.getCallingUid(), packageName); } NfcPermissions.enforceUserPermissions(mContext); int callingUid = Binder.getCallingUid(); boolean privilegedCaller = NfcInjector.isPrivileged(callingUid) || NfcPermissions.checkAdminPermissions(mContext); // Allow non-foreground callers with system uid or systemui privilegedCaller |= packageName.equals(SYSTEM_UI); Log.d(TAG, "updateDiscoveryTechnology: uid=" + callingUid + ", packageName: " + packageName); if (!privilegedCaller) { pollTech &= ~NfcAdapter.FLAG_SET_DEFAULT_TECH; listenTech &= ~NfcAdapter.FLAG_SET_DEFAULT_TECH; if (!mForegroundUtils.registerUidToBackgroundCallback( NfcService.this, callingUid)) { Log.e(TAG, "updateDiscoveryTechnology: Unprivileged caller shall be in foreground"); return; } } else if (((pollTech & NfcAdapter.FLAG_SET_DEFAULT_TECH) != 0 || (listenTech & NfcAdapter.FLAG_SET_DEFAULT_TECH) != 0)) { if (!isNfcEnabled()) { Log.d(TAG, "updateDiscoveryTechnology: NFC is not enabled."); return; } if ((pollTech & NfcAdapter.FLAG_SET_DEFAULT_TECH) != 0) { if ((pollTech & NfcAdapter.FLAG_READER_KEEP) == 0 && (pollTech & NfcAdapter.FLAG_USE_ALL_TECH) != NfcAdapter.FLAG_USE_ALL_TECH) { pollTech = getReaderModeTechMask(pollTech); saveNfcPollTech(pollTech & ~NfcAdapter.FLAG_SET_DEFAULT_TECH); Log.i(TAG, "updateDiscoveryTechnology: Default pollTech is set to 0x" + Integer.toHexString(pollTech)); } else if ((pollTech & (NfcAdapter.FLAG_READER_KEEP | NfcAdapter.FLAG_USE_ALL_TECH)) == (NfcAdapter.FLAG_READER_KEEP | NfcAdapter.FLAG_USE_ALL_TECH)) { saveNfcPollTech(DEFAULT_POLL_TECH); } } if ((listenTech & NfcAdapter.FLAG_SET_DEFAULT_TECH) != 0) { if ((listenTech & NfcAdapter.FLAG_LISTEN_KEEP) == 0 && (listenTech & NfcAdapter.FLAG_USE_ALL_TECH) != NfcAdapter.FLAG_USE_ALL_TECH) { saveNfcListenTech(listenTech & ~NfcAdapter.FLAG_SET_DEFAULT_TECH); Log.i(TAG, "updateDiscoveryTechnology: Default listenTech is set to 0x" + Integer.toHexString(listenTech)); } else if ((listenTech & (NfcAdapter.FLAG_LISTEN_KEEP | NfcAdapter.FLAG_USE_ALL_TECH)) == (NfcAdapter.FLAG_LISTEN_KEEP | NfcAdapter.FLAG_USE_ALL_TECH)) { saveNfcListenTech(DEFAULT_LISTEN_TECH); } } if ((pollTech & NfcAdapter.FLAG_READER_KEEP) != 0) { pollTech = getNfcPollTech(); } if ((listenTech & NfcAdapter.FLAG_LISTEN_KEEP) != 0) { listenTech = getNfcListenTech(); } mDeviceHost.setDiscoveryTech(pollTech, listenTech); applyRouting(true); return; } checkAndHandleRemovalDetectionMode(false); synchronized (NfcService.this) { if (!isNfcEnabled()) { Log.d(TAG, "updateDiscoveryTechnology: NFC is not enabled."); return; } Log.d(TAG, "updateDiscoveryTechnology: pollTech=0x" + Integer.toHexString(pollTech) + ", listenTech=0x" + Integer.toHexString(listenTech)); if (pollTech == NfcAdapter.FLAG_USE_ALL_TECH && listenTech == NfcAdapter.FLAG_USE_ALL_TECH && mDiscoveryTechParams != null) { try { binder.unlinkToDeath(mDiscoveryTechDeathRecipient, 0); mDeviceHost.resetDiscoveryTech(); mDiscoveryTechParams = null; } catch (NoSuchElementException e) { Log.e(TAG, "updateDiscoveryTechnology: Change Tech Binder was never " + "registered"); } } else if (!(pollTech == NfcAdapter.FLAG_USE_ALL_TECH && // Do not call for // resetDiscoveryTech listenTech == NfcAdapter.FLAG_USE_ALL_TECH)) { if ((pollTech & NfcAdapter.FLAG_READER_KEEP) != 0) { pollTech = getNfcPollTech(); } else { pollTech = getReaderModeTechMask(pollTech); } if ((listenTech & NfcAdapter.FLAG_LISTEN_KEEP) != 0) { listenTech = getNfcListenTech(); } try { mDeviceHost.setDiscoveryTech(pollTech, listenTech); mDiscoveryTechParams = new DiscoveryTechParams(); mDiscoveryTechParams.uid = callingUid; mDiscoveryTechParams.binder = binder; binder.linkToDeath(mDiscoveryTechDeathRecipient, 0); if (android.nfc.Flags.nfcPersistLog()) { mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setDiscoveryTechnologyUpdate(NfcEventProto .NfcDiscoveryTechnologyUpdate.newBuilder() .setAppInfo(NfcEventProto.NfcAppInfo .newBuilder() .setPackageName(packageName) .setUid(callingUid) .build()) .setPollTech(pollTech) .setListenTech(listenTech) .build()) .build()); } } catch (RemoteException e) { Log.e(TAG, "updateDiscoveryTechnology: Remote binder has already died"); return; } } else { return; } applyRouting(true); } } @Override public void setReaderMode( IBinder binder, IAppCallback callback, int flags, Bundle extras, String packageName) throws RemoteException { if (Flags.checkPassedInPackage()) { mNfcPermissions.checkPackage(Binder.getCallingUid(), packageName); } int callingUid = Binder.getCallingUid(); int callingPid = Binder.getCallingPid(); boolean privilegedCaller = NfcInjector.isPrivileged(callingUid) || NfcPermissions.checkAdminPermissions(mContext); // Allow non-foreground callers with system uid or systemui privilegedCaller |= packageName.equals(SYSTEM_UI); Log.d(TAG, "setReaderMode: uid=" + callingUid + ", packageName: " + packageName + ", flags: " + flags); if (!privilegedCaller && !mForegroundUtils.registerUidToBackgroundCallback( NfcService.this, callingUid)) { Log.e(TAG, "setReaderMode: Caller is not in foreground and is not system process"); return; } boolean disablePolling = flags != 0 && getReaderModeTechMask(flags) == 0; // Only allow to disable polling for specific callers if (disablePolling && !(privilegedCaller && mPollingDisableAllowed)) { Log.e(TAG, "setReaderMode: called with invalid flag parameter."); return; } if (extras != null && extras.containsKey(NfcAdapter.EXTRA_READER_TECH_A_POLLING_LOOP_ANNOTATION) && !isReaderModeAnnotationSupported()) { Log.e(TAG, "setReaderMode() called with annotation on an unsupported device."); return; } synchronized (NfcService.this) { if (!isNfcEnabled() && !privilegedCaller) { Log.e(TAG, "setReaderMode: called while NFC is not enabled."); return; } if (flags != 0) { try { if (disablePolling) { ReaderModeDeathRecipient pollingDisableDeathRecipient = new ReaderModeDeathRecipient(); binder.linkToDeath(pollingDisableDeathRecipient, 0); mPollingDisableDeathRecipients.put( callingPid, pollingDisableDeathRecipient); } else { if (mPollingDisableDeathRecipients.size() != 0) { Log.e(TAG, "setReaderMode: active polling is forced to disable now"); return; } binder.linkToDeath(mReaderModeDeathRecipient, 0); } if (mPollDelayed) { mHandler.removeMessages(MSG_DELAY_POLLING); mPollDelayCount = 0; mReadErrorCount = 0; mPollDelayed = false; mDeviceHost.startStopPolling(true); if (DBG) Log.d(TAG, "setReaderMode: polling is started"); } // listenTech is different from the default value, the stored listenTech will be included. // When using setReaderMode, change listenTech to default & restore to previous value. if (isNfcEnabled()) { clearListenTech(disablePolling); } updateReaderModeParams(callback, flags, extras, binder, callingUid); } catch (RemoteException e) { Log.e(TAG, "setReaderMode: Remote binder has already died"); return; } } else { try { ReaderModeDeathRecipient pollingDisableDeathRecipient = mPollingDisableDeathRecipients.get(callingPid); mPollingDisableDeathRecipients.remove(callingPid); if (mPollingDisableDeathRecipients.size() == 0) { mReaderModeParams = null; StopPresenceChecking(); } if (pollingDisableDeathRecipient != null) { binder.unlinkToDeath(pollingDisableDeathRecipient, 0); } else { binder.unlinkToDeath(mReaderModeDeathRecipient, 0); } } catch (NoSuchElementException e) { Log.e(TAG, "setReaderMode: Reader mode Binder was never registered"); } finally { // listenTech is different from the default value, the stored listenTech will be included. // When using enableReaderMode, change listenTech to default & restore to the previous value. if (isNfcEnabled()) { restoreSavedTech(); } } } mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setReaderModeChange(NfcEventProto.NfcReaderModeChange.newBuilder() .setAppInfo(NfcEventProto.NfcAppInfo.newBuilder() .setPackageName(packageName) .setUid(callingUid) .build()) .setFlags(flags) .build()) .build()); if (isNfcEnabled()) { applyRouting(false); } } checkAndHandleRemovalDetectionMode(true); } @Override public INfcAdapterExtras getNfcAdapterExtrasInterface(String pkg) throws RemoteException { // nfc-extras implementation is no longer present in AOSP. return null; } @Override public INfcDta getNfcDtaInterface(String pkg) throws RemoteException { NfcPermissions.enforceAdminPermissions(mContext); if (mNfcDtaService == null) { mNfcDtaService = new NfcDtaService(); } return mNfcDtaService; } @Override public IT4tNdefNfcee getT4tNdefNfceeInterface() throws RemoteException { return mT4tNdefNfceeService; } @Override public void addNfcUnlockHandler(INfcUnlockHandler unlockHandler, int[] techList) { NfcPermissions.enforceAdminPermissions(mContext); int lockscreenPollMask = computeLockscreenPollMask(techList); synchronized (NfcService.this) { mNfcUnlockManager.addUnlockHandler(unlockHandler, lockscreenPollMask); } applyRouting(false); } @Override public void removeNfcUnlockHandler(INfcUnlockHandler token) throws RemoteException { synchronized (NfcService.this) { mNfcUnlockManager.removeUnlockHandler(token.asBinder()); } applyRouting(false); } @Override public boolean deviceSupportsNfcSecure() { return mIsSecureNfcCapable; } @Override public NfcAntennaInfo getNfcAntennaInfo() { int positionX[] = mContext.getResources().getIntArray( R.array.antenna_x); int positionY[] = mContext.getResources().getIntArray( R.array.antenna_y); int width = mContext.getResources().getInteger(R.integer.device_width); int height = mContext.getResources().getInteger(R.integer.device_height); boolean isFoldable = mContext.getResources().getBoolean(R.bool.device_foldable); // If overlays are not set, try reading properties. if (positionX.length == 0 || positionY.length == 0) { positionX = NfcProperties.info_antpos_X().stream() .mapToInt(Integer::intValue) .toArray(); positionY = NfcProperties.info_antpos_Y().stream() .mapToInt(Integer::intValue) .toArray(); width = NfcProperties.info_antpos_device_width().orElse(0); height = NfcProperties.info_antpos_device_height().orElse(0); isFoldable = NfcProperties.info_antpos_device_foldable().orElse(false); } if (positionX.length != positionY.length) { return null; } List availableNfcAntennas = new ArrayList<>(); for (int i = 0; i < positionX.length; i++) { if (positionX[i] >= width || positionY[i] >= height) { return null; } availableNfcAntennas.add(new AvailableNfcAntenna(positionX[i], positionY[i])); } return new NfcAntennaInfo( width, height, isFoldable, availableNfcAntennas); } @Override public boolean setWlcEnabled(boolean enable) { if (!mIsWlcCapable) { return false; } NfcPermissions.enforceAdminPermissions(mContext); // enable or disable WLC if (DBG) Log.d(TAG, "setWlcEnabled: " + enable); synchronized (NfcService.this) { // check whether NFC is enabled if (!isNfcEnabled()) { return false; } mPrefsEditor.putBoolean(PREF_NFC_CHARGING_ON, enable); mPrefsEditor.apply(); mIsWlcEnabled = enable; mBackupManager.dataChanged(); } if (android.nfc.Flags.nfcPersistLog()) { mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setWlcStateChange( NfcEventProto.NfcWlcStateChange.newBuilder() .setEnable(enable) .build()) .build()); } return true; } @Override public boolean isWlcEnabled() throws RemoteException { if (!mIsWlcCapable) { return false; } // check whether WLC is enabled or disabled synchronized (NfcService.this) { return mIsWlcEnabled; } } @Override public WlcListenerDeviceInfo getWlcListenerDeviceInfo() { if (!mIsWlcCapable) { return null; } synchronized (NfcService.this) { return mWlcListenerDeviceInfo; } } private int computeLockscreenPollMask(int[] techList) { Map techCodeToMask = new HashMap(); techCodeToMask.put(TagTechnology.NFC_A, NFC_POLL_A); techCodeToMask.put(TagTechnology.NFC_B, NFC_POLL_B); techCodeToMask.put(TagTechnology.NFC_V, NFC_POLL_V); techCodeToMask.put(TagTechnology.NFC_F, NFC_POLL_F); techCodeToMask.put(TagTechnology.NFC_BARCODE, NFC_POLL_KOVIO); int mask = 0; for (int i = 0; i < techList.length; i++) { if (techCodeToMask.containsKey(techList[i])) { mask |= techCodeToMask.get(techList[i]).intValue(); } } return mask; } private int getReaderModeTechMask(int flags) { int techMask = 0; if ((flags & NfcAdapter.FLAG_READER_NFC_A) != 0) { techMask |= NFC_POLL_A; } if ((flags & NfcAdapter.FLAG_READER_NFC_B) != 0) { techMask |= NFC_POLL_B; } if ((flags & NfcAdapter.FLAG_READER_NFC_F) != 0) { techMask |= NFC_POLL_F; } if ((flags & NfcAdapter.FLAG_READER_NFC_V) != 0) { techMask |= NFC_POLL_V; } if ((flags & NfcAdapter.FLAG_READER_NFC_BARCODE) != 0) { techMask |= NFC_POLL_KOVIO; } return techMask; } private void updateReaderModeParams( IAppCallback callback, int flags, Bundle extras, IBinder binder, int uid) { synchronized (NfcService.this) { mReaderModeParams = new ReaderModeParams(); mReaderModeParams.callback = callback; mReaderModeParams.flags = flags; mReaderModeParams.presenceCheckDelay = extras != null ? (extras.getInt(NfcAdapter.EXTRA_READER_PRESENCE_CHECK_DELAY, DEFAULT_PRESENCE_CHECK_DELAY)) : DEFAULT_PRESENCE_CHECK_DELAY; mReaderModeParams.binder = binder; mReaderModeParams.uid = uid; mReaderModeParams.annotation = extras == null ? null : extras.getByteArray( NfcAdapter.EXTRA_READER_TECH_A_POLLING_LOOP_ANNOTATION); } } private int setTagAppPreferenceInternal(int userId, String pkg, boolean allow) { if (!isPackageInstalled(pkg, userId)) { return NfcAdapter.TAG_INTENT_APP_PREF_RESULT_PACKAGE_NOT_FOUND; } if (DBG) { Log.i(TAG, "setTagAppPreferenceInternal: UserId:" + userId + " pkg:" + pkg + ":" + allow); } synchronized (NfcService.this) { mTagAppPrefList.computeIfAbsent(userId, key -> new HashMap()) .put(pkg, allow); } storeTagAppPrefList(); return NfcAdapter.TAG_INTENT_APP_PREF_RESULT_SUCCESS; } @Override public void setControllerAlwaysOn(int mode) throws RemoteException { NfcPermissions.enforceSetControllerAlwaysOnPermissions(mContext); if (!mIsAlwaysOnSupported) { throw new UnsupportedOperationException("isControllerAlwaysOn not supported"); } if (mode != NfcOemExtension.DISABLE) { /* AsyncTask params */ Integer[] paramIntegers = {TASK_ENABLE_ALWAYS_ON, mode}; new EnableDisableTask().execute(paramIntegers); } else { new EnableDisableTask().execute(TASK_DISABLE_ALWAYS_ON); } } @Override public boolean isControllerAlwaysOn() throws RemoteException { NfcPermissions.enforceSetControllerAlwaysOnPermissions(mContext); return mIsAlwaysOnSupported && mAlwaysOnState == NfcAdapter.STATE_ON; } @Override public boolean isControllerAlwaysOnSupported() throws RemoteException { NfcPermissions.enforceSetControllerAlwaysOnPermissions(mContext); return mIsAlwaysOnSupported; } @Override public void registerControllerAlwaysOnListener( INfcControllerAlwaysOnListener listener) throws RemoteException { NfcPermissions.enforceSetControllerAlwaysOnPermissions(mContext); if (!mIsAlwaysOnSupported) return; mAlwaysOnListeners.add(listener); } @Override public void unregisterControllerAlwaysOnListener( INfcControllerAlwaysOnListener listener) throws RemoteException { NfcPermissions.enforceSetControllerAlwaysOnPermissions(mContext); if (!mIsAlwaysOnSupported) return; mAlwaysOnListeners.remove(listener); } @Override public boolean isTagIntentAppPreferenceSupported() throws RemoteException { return mIsTagAppPrefSupported; } @Override public Map getTagIntentAppPreferenceForUser(int userId) throws RemoteException { NfcPermissions.enforceAdminPermissions(mContext); if (!mIsTagAppPrefSupported) throw new UnsupportedOperationException(); synchronized (NfcService.this) { return mTagAppPrefList.getOrDefault(userId, new HashMap<>()); } } @Override public int setTagIntentAppPreferenceForUser(int userId, String pkg, boolean allow) throws RemoteException { NfcPermissions.enforceAdminPermissions(mContext); if (!mIsTagAppPrefSupported) throw new UnsupportedOperationException(); return setTagAppPreferenceInternal(userId, pkg, allow); } @Override public boolean isTagIntentAllowed(String pkg, int userId) throws RemoteException { if (!android.nfc.Flags.nfcCheckTagIntentPreference()) { return true; } if (!mIsTagAppPrefSupported) { return true; } HashMap map; synchronized (NfcService.this) { map = mTagAppPrefList.getOrDefault(userId, new HashMap<>()); } return map.getOrDefault(pkg, true); } public boolean enableReaderOption(boolean enable, String pkg) { Log.d(TAG, "enableReaderOption: enabled = " + enable + " calling uid = " + Binder.getCallingUid()); if (!mReaderOptionCapable) return false; NfcPermissions.enforceAdminPermissions(mContext); synchronized (NfcService.this) { mPrefsEditor.putBoolean(PREF_NFC_READER_OPTION_ON, enable); mPrefsEditor.apply(); mIsReaderOptionEnabled = enable; mBackupManager.dataChanged(); } applyRouting(true); if (mNfcOemExtensionCallback != null) { try { mNfcOemExtensionCallback.onReaderOptionChanged(enable); } catch (RemoteException e) { Log.e(TAG, "enableReaderOption: onReaderOptionChanged failed e = " + e.toString()); } } if (android.nfc.Flags.nfcPersistLog()) { mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setReaderOptionChange( NfcEventProto.NfcReaderOptionChange.newBuilder() .setEnable(enable) .setAppInfo( NfcEventProto.NfcAppInfo.newBuilder() .setPackageName(pkg) .setUid(Binder.getCallingUid()) .build()) .build()) .build()); } return true; } @Override public boolean isReaderOptionSupported() { return mReaderOptionCapable; } @Override public boolean isReaderOptionEnabled() { return mIsReaderOptionEnabled; } @Override public void registerWlcStateListener( INfcWlcStateListener listener) throws RemoteException { if (!mIsWlcCapable) { return; } NfcPermissions.enforceAdminPermissions(mContext); mWlcStateListener.add(listener); } @Override public void unregisterWlcStateListener( INfcWlcStateListener listener) throws RemoteException { if (!mIsWlcCapable) { return; } NfcPermissions.enforceAdminPermissions(mContext); mWlcStateListener.remove(listener); } @Override public void notifyPollingLoop(PollingFrame frame) { try { byte[] data; int type = frame.getType(); int gain = frame.getVendorSpecificGain(); byte[] frame_data = frame.getData(); long timestamp = frame.getTimestamp(); HexFormat format = HexFormat.ofDelimiter(" "); String timestampBytes = format.formatHex(new byte[] { (byte) (timestamp >>> 24), (byte) (timestamp >>> 16), (byte) (timestamp >>> 8), (byte) timestamp }); int frame_data_length = frame_data == null ? 0 : frame_data.length; String frame_data_str = frame_data_length == 0 ? "" : " " + format.formatHex(frame_data); String type_str = "FF"; switch (type) { case PollingFrame.POLLING_LOOP_TYPE_ON: type_str = "00"; data = new byte[] { 0x01 }; break; case PollingFrame.POLLING_LOOP_TYPE_OFF: type_str = "00"; data = new byte[] { 0x00 }; break; case PollingFrame.POLLING_LOOP_TYPE_A: type_str = "01"; break; case PollingFrame.POLLING_LOOP_TYPE_B: type_str = "02"; break; case PollingFrame.POLLING_LOOP_TYPE_F: type_str = "03"; break; case PollingFrame.POLLING_LOOP_TYPE_UNKNOWN: type_str = "07"; break; } data = format.parseHex("6f 0C " + String.format("%02x", 9 + frame_data_length) + " 03 " + type_str + " 00 " + String.format("%02x", 5 + frame_data_length) + " " + timestampBytes + " " + String.format("%02x", gain) + frame_data_str); ((NativeNfcManager) mDeviceHost).injectNtf(data); } catch (Exception ex) { Log.e(TAG, "notifyPollingLoop: error when notifying polling loop", ex); } } @Override public void notifyTestHceData(int technology, byte[] data) { onHostCardEmulationData(technology, data); } @Override public void notifyHceDeactivated() { try { mCardEmulationManager.onHostCardEmulationDeactivated(1); } catch (Exception ex) { Log.e(TAG, "notifyHceDeactivated: e=", ex); } } @Override public int handleShellCommand(@NonNull ParcelFileDescriptor in, @NonNull ParcelFileDescriptor out, @NonNull ParcelFileDescriptor err, @NonNull String[] args) { NfcShellCommand shellCommand = new NfcShellCommand(NfcService.this, mContext); return shellCommand.exec(this, in.getFileDescriptor(), out.getFileDescriptor(), err.getFileDescriptor(), args); } private static boolean isPowerSavingModeCmd(int gid, int oid, byte[] payload) { return gid == NCI_GID_PROP && oid == NCI_MSG_PROP_ANDROID && payload.length > 0 && payload[0] == NCI_MSG_PROP_ANDROID_POWER_SAVING; } private static boolean isQueryPowerSavingStatusCmd(int gid, int oid, byte[] payload) { return gid == NCI_GID_PROP && oid == NCI_MSG_PROP_ANDROID && payload.length > 0 && payload[0] == NCI_PROP_ANDROID_QUERY_POWER_SAVING_STATUS_CMD; } @Override public int sendVendorNciMessage(int mt, int gid, int oid, byte[] payload) throws RemoteException { NfcPermissions.enforceAdminPermissions(mContext); if ((!isNfcEnabled() && !mIsPowerSavingModeEnabled) && !isControllerAlwaysOn()) { Log.e(TAG, "sendVendorNciMessage: Nfc is not enabled"); return NCI_STATUS_FAILED; } FutureTask sendVendorCmdTask = new FutureTask<>( () -> { synchronized (NfcService.this) { if (isPowerSavingModeCmd(gid, oid, payload)) { boolean status = setPowerSavingMode(payload[1] == 0x01); return status ? NCI_STATUS_OK : NCI_STATUS_FAILED; } else if (isQueryPowerSavingStatusCmd(gid, oid, payload)) { NfcVendorNciResponse response = new NfcVendorNciResponse( (byte) NCI_STATUS_OK, NCI_GID_PROP, NCI_MSG_PROP_ANDROID, new byte[] { (byte) NCI_PROP_ANDROID_QUERY_POWER_SAVING_STATUS_CMD, 0x00, mIsPowerSavingModeEnabled ? (byte) 0x01 : (byte) 0x00}); if (response.status == NCI_STATUS_OK) { mHandler.post(() -> mNfcAdapter.sendVendorNciResponse( response.gid, response.oid, response.payload)); } return Integer.valueOf(response.status); } else { NfcVendorNciResponse response = mDeviceHost.sendRawVendorCmd(mt, gid, oid, payload); if (response.status == NCI_STATUS_OK) { mHandler.post(() -> mNfcAdapter.sendVendorNciResponse( response.gid, response.oid, response.payload)); } return Integer.valueOf(response.status); } }}); int status = NCI_STATUS_FAILED; try { status = runTaskOnSingleThreadExecutor(sendVendorCmdTask, SEND_VENDOR_CMD_TIMEOUT_MS); } catch (TimeoutException e) { Log.e(TAG, "sendVendorNciMessage: Failed , status : TIMEOUT", e); } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } return status; } @Override public void registerVendorExtensionCallback(INfcVendorNciCallback callbacks) throws RemoteException { synchronized (NfcService.this) { if (DBG) Log.i(TAG, "registerVendorExtensionCallback"); NfcPermissions.enforceAdminPermissions(mContext); mNfcVendorNciCallBack = callbacks; mDeviceHost.enableVendorNciNotifications(true); } } @Override public void unregisterVendorExtensionCallback(INfcVendorNciCallback callbacks) throws RemoteException { synchronized (NfcService.this) { if (DBG) Log.i(TAG, "unregisterVendorExtensionCallback"); NfcPermissions.enforceAdminPermissions(mContext); mNfcVendorNciCallBack = null; mDeviceHost.enableVendorNciNotifications(false); } } @Override public void registerOemExtensionCallback(INfcOemExtensionCallback callbacks) throws RemoteException { if (DBG) Log.i(TAG, "registerOemExtensionCallback"); NfcPermissions.enforceAdminPermissions(mContext); mNfcOemExtensionCallback = callbacks; updateNfCState(); if (mCardEmulationManager != null) { mCardEmulationManager.setOemExtension(mNfcOemExtensionCallback); } if (mNfcDispatcher != null) { mNfcDispatcher.setOemExtension(mNfcOemExtensionCallback); } } @Override public void unregisterOemExtensionCallback(INfcOemExtensionCallback callbacks) throws RemoteException { if (DBG) Log.i(TAG, "unregisterOemExtensionCallback"); NfcPermissions.enforceAdminPermissions(mContext); mNfcOemExtensionCallback = null; if (mCardEmulationManager != null) { mCardEmulationManager.setOemExtension(mNfcOemExtensionCallback); } if (mNfcDispatcher != null) { mNfcDispatcher.setOemExtension(mNfcOemExtensionCallback); } } @Override public Map fetchActiveNfceeList() throws RemoteException { Map map = new HashMap(); if (isNfcEnabled()) { map = mDeviceHost.dofetchActiveNfceeList(); } return map; } @Override public void clearPreference() throws RemoteException { if (DBG) Log.i(TAG, "clearPreference"); NfcPermissions.enforceAdminPermissions(mContext); if (android.nfc.Flags.nfcPersistLog()) { mNfcEventLog.logEvent(NfcEventProto.EventType.newBuilder() .setClearPreference( NfcEventProto.NfcClearPreference.newBuilder() .build()) .build()); } mPrefsEditor.clear(); if (mIsNfcUserChangeRestricted) { mPrefsEditor.putBoolean(PREF_NFC_ON, getNfcOnSetting()); } mPrefsEditor.putBoolean( PREF_NFC_READER_OPTION_ON, mDeviceConfigFacade.getDefaultReaderOption()); mPrefsEditor.commit(); } @Override public void setScreenState() throws RemoteException { if (DBG) Log.i(TAG, "setScreenState"); NfcPermissions.enforceAdminPermissions(mContext); applyScreenState(mScreenStateHelper.checkScreenState(mCheckDisplayStateForScreenState)); } @Override public void checkFirmware() throws RemoteException { if (DBG) Log.i(TAG, "checkFirmware"); NfcPermissions.enforceAdminPermissions(mContext); if (isNfcEnabled()) { if (DBG) Log.i(TAG, "Check firmware by restarting Nfc stack"); restartStack(); return; } FutureTask checkFirmwareTask = new FutureTask<>(() -> { if (DBG) Log.i(TAG, "Nfc is disabled, checking Firmware"); mDeviceHost.checkFirmware(); return 0; }); try { runTaskOnSingleThreadExecutor( checkFirmwareTask, CHECK_FIRMWARE_TIMEOUT_MS); } catch (TimeoutException e) { Log.e(TAG, "checkFirmware: failed, status : TIMEOUT", e); } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } } @Override public void triggerInitialization() throws RemoteException { if (DBG) Log.i(TAG, "triggerInitialization"); NfcPermissions.enforceAdminPermissions(mContext); new EnableDisableTask().execute(TASK_BOOT); } @Override public boolean getSettingStatus() throws RemoteException { if (DBG) Log.i(TAG, "getSettingStatus"); NfcPermissions.enforceAdminPermissions(mContext); return getNfcOnSetting(); } @Override public boolean isTagPresent() throws RemoteException { if (DBG) Log.i(TAG, "isTagPresent"); NfcPermissions.enforceAdminPermissions(mContext); return NfcService.this.isTagPresent(); } @Override public List getRoutingTableEntryList() throws RemoteException { if (DBG) Log.i(TAG, "getRoutingTableEntry"); NfcPermissions.enforceAdminPermissions(mContext); return mRoutingTableParser.getRoutingTableEntryList(mDeviceHost); } @Override public void indicateDataMigration(boolean inProgress, String pkg) throws RemoteException { if (Flags.checkPassedInPackage()) { mNfcPermissions.checkPackage(Binder.getCallingUid(), pkg); } if (DBG) Log.i(TAG, "indicateDataMigration: inProgress: " + inProgress); NfcPermissions.enforceAdminPermissions(mContext); mNfcEventLog.logEvent( NfcEventProto.EventType.newBuilder() .setDataMigrationInProgress(NfcEventProto.NfcDataMigrationInProgress .newBuilder() .setAppInfo(NfcEventProto.NfcAppInfo.newBuilder() .setPackageName(pkg) .setUid(Binder.getCallingUid()) .build()) .setInProgress(inProgress) .build()) .build()); } @Override public int commitRouting() throws RemoteException { if (isNfcDisabledOrDisabling()) { Log.d(TAG, "Skip commit routing when NFCC is off " + "or turning off"); return STATUS_UNKNOWN_ERROR; } if (DBG) Log.i(TAG, "commitRouting"); NfcPermissions.enforceAdminPermissions(mContext); return mDeviceHost.commitRouting(); } @Override public long getMaxPausePollingTimeoutMs() { if (DBG) Log.i(TAG, "getMaxPausePollingTimeoutMs"); NfcPermissions.enforceAdminPermissions(mContext); int timeoutOverlay = mContext.getResources() .getInteger(R.integer.max_pause_polling_time_out_ms); return timeoutOverlay > 0 ? (long) timeoutOverlay : MAX_POLLING_PAUSE_TIMEOUT; } private void updateNfCState() { if (mNfcOemExtensionCallback != null) { try { if (DBG) Log.i(TAG, "updateNfCState"); mNfcOemExtensionCallback.onCardEmulationActivated(mCardEmulationActivated); mNfcOemExtensionCallback.onRfFieldDetected(mRfFieldActivated); mNfcOemExtensionCallback.onRfDiscoveryStarted(mRfDiscoveryStarted); mNfcOemExtensionCallback.onEeListenActivated(mEeListenActivated); } catch (RemoteException e) { Log.e(TAG, "updateNfCState: Failed to update, e=", e); } } } private synchronized void sendVendorNciResponse(int gid, int oid, byte[] payload) { if (VDBG) Log.i(TAG, "onVendorNciResponseReceived"); if (mNfcVendorNciCallBack != null) { try { mNfcVendorNciCallBack.onVendorResponseReceived(gid, oid, payload); } catch (RemoteException e) { Log.e(TAG, "onVendorNciResponseReceived: failed, e=", e); } } } private synchronized void sendVendorNciNotification(int gid, int oid, byte[] payload) { if (VDBG) Log.i(TAG, "sendVendorNciNotification"); if (mNfcVendorNciCallBack != null) { try { mNfcVendorNciCallBack.onVendorNotificationReceived(gid, oid, payload); } catch (RemoteException e) { Log.e(TAG, "sendVendorNciNotification: failed, e=", e); } } } } final class SeServiceDeathRecipient implements IBinder.DeathRecipient { @Override public void binderDied() { synchronized (NfcService.this) { Log.i(TAG, "binderDied: SE Service died"); mSEService = null; } } } final class ReaderModeDeathRecipient implements IBinder.DeathRecipient { @Override public void binderDied() { synchronized (NfcService.this) { if (mReaderModeParams != null) { mPollingDisableDeathRecipients.values().remove(this); resetReaderModeParams(); } } } } final class DiscoveryTechDeathRecipient implements IBinder.DeathRecipient { @Override public void binderDied() { if (DBG) Log.d(TAG, "binderDied: setDiscoveryTech death recipient"); synchronized (NfcService.this) { if (isNfcEnabled() && mDiscoveryTechParams != null) { mDeviceHost.resetDiscoveryTech(); mDiscoveryTechParams = null; } } applyRouting(true); } } final class TagService extends INfcTag.Stub { @Override public int connect(int nativeHandle, int technology) throws RemoteException { NfcPermissions.enforceUserPermissions(mContext); TagEndpoint tag = null; if (!isNfcEnabled()) { return ErrorCodes.ERROR_NOT_INITIALIZED; } if (!isReaderOptionEnabled()) { return ErrorCodes.ERROR_NOT_INITIALIZED; } if (checkAndHandleRemovalDetectionMode(true)) { 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; } // 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 { NfcPermissions.enforceUserPermissions(mContext); TagEndpoint tag = null; // Check if NFC is enabled if (!isNfcEnabled()) { return ErrorCodes.ERROR_NOT_INITIALIZED; } if (!isReaderOptionEnabled()) { return ErrorCodes.ERROR_NOT_INITIALIZED; } if (checkAndHandleRemovalDetectionMode(true)) { 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 { NfcPermissions.enforceUserPermissions(mContext); // Check if NFC is enabled if (!isNfcEnabled()) { return null; } if (!isReaderOptionEnabled()) { return null; } if (checkAndHandleRemovalDetectionMode(true)) { 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; } if (!isReaderOptionEnabled()) { return false; } if (checkAndHandleRemovalDetectionMode(true)) { 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 { NfcPermissions.enforceUserPermissions(mContext); TagEndpoint tag = null; // Check if NFC is enabled if (!isNfcEnabled()) { return false; } if (!isReaderOptionEnabled()) { return false; } if (checkAndHandleRemovalDetectionMode(true)) { 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 { NfcPermissions.enforceUserPermissions(mContext); TagEndpoint tag = null; byte[] response; // Check if NFC is enabled if (!isNfcEnabled()) { return null; } if (!isReaderOptionEnabled()) { return null; } if (checkAndHandleRemovalDetectionMode(true)) { 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 { NfcPermissions.enforceUserPermissions(mContext); TagEndpoint tag; // Check if NFC is enabled if (!isNfcEnabled()) { return null; } if (!isReaderOptionEnabled()) { return null; } if (checkAndHandleRemovalDetectionMode(true)) { 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 { NfcPermissions.enforceUserPermissions(mContext); TagEndpoint tag; // Check if NFC is enabled if (!isNfcEnabled()) { return ErrorCodes.ERROR_NOT_INITIALIZED; } if (!isReaderOptionEnabled()) { return ErrorCodes.ERROR_NOT_INITIALIZED; } if (checkAndHandleRemovalDetectionMode(true)) { 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 { if (checkAndHandleRemovalDetectionMode(true)) { return false; } throw new UnsupportedOperationException(); } @Override public int ndefMakeReadOnly(int nativeHandle) throws RemoteException { NfcPermissions.enforceUserPermissions(mContext); TagEndpoint tag; // Check if NFC is enabled if (!isNfcEnabled()) { return ErrorCodes.ERROR_NOT_INITIALIZED; } if (!isReaderOptionEnabled()) { return ErrorCodes.ERROR_NOT_INITIALIZED; } if (checkAndHandleRemovalDetectionMode(true)) { 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 { NfcPermissions.enforceUserPermissions(mContext); TagEndpoint tag; // Check if NFC is enabled if (!isNfcEnabled()) { return ErrorCodes.ERROR_NOT_INITIALIZED; } if (!isReaderOptionEnabled()) { return ErrorCodes.ERROR_NOT_INITIALIZED; } if (checkAndHandleRemovalDetectionMode(true)) { 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 { NfcPermissions.enforceUserPermissions(mContext); TagEndpoint tag = null; // Check if NFC is enabled if (!isNfcEnabled()) { return null; } if (!isReaderOptionEnabled()) { return null; } if (checkAndHandleRemovalDetectionMode(true)) { 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 try { /* Avoid setting mCookieUpToDate to negative values */ mCookieUpToDate = mCookieGenerator.nextLong() >>> 1; Tag newTag = new Tag(tag.getUid(), tag.getTechList(), tag.getTechExtras(), tag.getHandle(), mCookieUpToDate, this); return newTag; } catch (Exception e) { Log.e(TAG, "Tag creation exception.", e); return null; } } return null; } @Override public int setTimeout(int tech, int timeout) throws RemoteException { NfcPermissions.enforceUserPermissions(mContext); 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 { NfcPermissions.enforceUserPermissions(mContext); return mDeviceHost.getTimeout(tech); } @Override public void resetTimeouts() throws RemoteException { NfcPermissions.enforceUserPermissions(mContext); 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(); } @Override public boolean isTagUpToDate(long cookie) throws RemoteException { if (mCookieUpToDate != -1 && mCookieUpToDate == cookie) { if (DBG) { Log.d(TAG, "isTagUpToDate: Tag " + Long.toString(cookie) + " is up to date"); } return true; } if (DBG) { Log.d(TAG, "isTagUpToDate: Tag " + Long.toString(cookie) + " is out of date"); } EventLog.writeEvent(0x534e4554, "199291025", -1, "The obsolete tag was attempted to be accessed"); return false; } } final class NfcDtaService extends INfcDta.Stub { public void enableDta() throws RemoteException { NfcPermissions.enforceAdminPermissions(mContext); if (!sIsDtaMode) { mDeviceHost.enableDtaMode(); sIsDtaMode = true; Log.d(TAG, "enableDta: DTA Mode is Enabled"); } else { Log.d(TAG, "enableDta: DTA Mode is already Enabled"); } } public void disableDta() throws RemoteException { NfcPermissions.enforceAdminPermissions(mContext); if (sIsDtaMode) { mDeviceHost.disableDtaMode(); sIsDtaMode = false; Log.d(TAG, "disableDta: DTA Mode is Disabled"); } else { Log.d(TAG, "disableDta: DTA Mode is already Disabled"); } } public boolean enableServer(String serviceName, int serviceSap, int miu, int rwSize,int testCaseId) throws RemoteException { NfcPermissions.enforceAdminPermissions(mContext); return false; } public void disableServer() throws RemoteException { } public boolean enableClient(String serviceName, int miu, int rwSize, int testCaseId) throws RemoteException { NfcPermissions.enforceAdminPermissions(mContext); return false; } public void disableClient() throws RemoteException { return; } public boolean registerMessageService(String msgServiceName) throws RemoteException { NfcPermissions.enforceAdminPermissions(mContext); if (msgServiceName == null) return false; DtaServiceConnector.setMessageService(msgServiceName); return true; } }; class T4tNdefNfceeService extends IT4tNdefNfcee.Stub { @Override public int writeData(final int fileId, byte[] data) { NfcPermissions.enforceAdminPermissions(mContext); if (mDeviceHost.isNdefOperationOngoing()) { return T4tNdefNfcee.WRITE_DATA_ERROR_DEVICE_BUSY; } int status = T4tNdefNfcee.WRITE_DATA_ERROR_INTERNAL; try { ByteBuffer fileIdInBytes = ByteBuffer.allocate(2); fileIdInBytes.putShort((short)fileId); status = mDeviceHost.doWriteData(fileIdInBytes.array(), data); if (status > 0) status = T4tNdefNfcee.WRITE_DATA_SUCCESS; } catch (Exception e) { Log.e(TAG, "writeData: e=", e); } Log.i(TAG, "writeData : " + status); return status; } @Override public byte[] readData(final int fileId) { NfcPermissions.enforceAdminPermissions(mContext); if (mDeviceHost.isNdefOperationOngoing()) { throw new IllegalStateException("Device is busy"); } byte[] readData = {}; ByteBuffer fileIdInBytes = ByteBuffer.allocate(2); fileIdInBytes.putShort((short)fileId); readData = mDeviceHost.doReadData(fileIdInBytes.array()); if (readData == null) { throw new IllegalStateException("Ndef Nfcee read failed"); } return readData; } @Override public T4tNdefNfceeCcFileInfo readCcfile() { NfcPermissions.enforceAdminPermissions(mContext); if (mDeviceHost.isNdefOperationOngoing()) { throw new IllegalStateException("Device is busy"); } T4tNdefNfceeCcFileInfo ccFileInfo = null; byte[] readData = {}; try { readData = mDeviceHost.doReadData(T4T_NFCEE_CC_FILE_ID); } catch (Exception e) { Log.e(TAG, "readCcfile: Exception e=", e); return ccFileInfo; } if (readData.length >= 15) { int cclen = ((Byte.toUnsignedInt(readData[0])) << 8) + (Byte.toUnsignedInt(readData[1])); int version = Byte.toUnsignedInt(readData[2]); if (version == T4T_NFCEE_MAPPING_VERSION_2_0) { int ndefFileId = ((Byte.toUnsignedInt(readData[9])) << 8) + Byte.toUnsignedInt(readData[10]); int ndefMaxFileSize = ((Byte.toUnsignedInt(readData[11])) << 8) + Byte.toUnsignedInt(readData[12]); boolean isReadAllowed = readData[13] == 0; boolean isWriteAllowed = readData[14] == 0; ccFileInfo = new T4tNdefNfceeCcFileInfo( cclen, version, ndefFileId, ndefMaxFileSize, isReadAllowed, isWriteAllowed); } else { Log.e(TAG, "readCcfile: Unsupported NDEF mapping version received. " + "Versions otherthan 2.0 are not supported."); throw new UnsupportedOperationException( "Unsupported NDEF mapping version received"); } } else { Log.e(TAG, "readCcfile: Empty data"); } return ccFileInfo; } @Override public int clearNdefData() { NfcPermissions.enforceAdminPermissions(mContext); if (mDeviceHost.isNdefOperationOngoing()) { return T4tNdefNfcee.CLEAR_DATA_FAILED_DEVICE_BUSY; } boolean isEnabled = (isNfcEnabled() || (((mIsAlwaysOnSupported && mAlwaysOnState == NfcAdapter.STATE_ON)) && (mAlwaysOnMode == NfcOemExtension.ENABLE_EE))); if (!isEnabled) { mDeviceHost.setPartialInitMode(NfcOemExtension.ENABLE_EE); mDeviceHost.initialize(); } boolean status = mDeviceHost.doClearNdefData(); if (!isEnabled) { mDeviceHost.deinitialize(); } Log.i(TAG, "clearNdefData: " + status); return status ? T4tNdefNfcee.CLEAR_DATA_SUCCESS : T4tNdefNfcee.CLEAR_DATA_FAILED_INTERNAL; } @Override public boolean isNdefOperationOngoing() { NfcPermissions.enforceAdminPermissions(mContext); boolean status = mDeviceHost.isNdefOperationOngoing(); Log.i(TAG, "isNdefOperationOngoing: " + status); return status; } @Override public boolean isNdefNfceeEmulationSupported() { NfcPermissions.enforceAdminPermissions(mContext); boolean status = mDeviceHost.isNdefNfceeEmulationSupported(); Log.i(TAG, "isT4tNdefNfceeEmulationSupported: " + status); return status; } } boolean isNfcEnabledOrShuttingDown() { synchronized (this) { return (mState == NfcAdapter.STATE_ON || mState == NfcAdapter.STATE_TURNING_OFF); } } boolean isNfcDisabledOrDisabling() { synchronized (this) { return (mState == NfcAdapter.STATE_OFF || mState == NfcAdapter.STATE_TURNING_OFF); } } boolean isNfcEnabled() { synchronized (this) { return mState == NfcAdapter.STATE_ON; } } boolean isReaderOptionEnabled() { synchronized (this) { return mIsReaderOptionEnabled || mReaderModeParams != null; } } class WatchDogThread extends Thread { final Object mCancelWaiter = new Object(); final int mTimeout; boolean mCanceled = false; public WatchDogThread(String threadName, int timeout) { super(threadName); mTimeout = timeout; } @Override public void run() { try { synchronized (mCancelWaiter) { mCancelWaiter.wait(mTimeout); if (mCanceled) { return; } } } catch (InterruptedException e) { // Should not happen; fall-through to abort. Log.w(TAG, "run: Watchdog thread interrupted"); interrupt(); } if (mRoutingWakeLock.isHeld()) { Log.e(TAG, "run: Watchdog triggered, release lock before aborting"); mRoutingWakeLock.release(); } Log.e(TAG, "run: Watchdog triggered, aborting"); NfcStatsLog.write(NfcStatsLog.NFC_STATE_CHANGED, NfcStatsLog.NFC_STATE_CHANGED__STATE__CRASH_RESTART); if (android.nfc.Flags.nfcEventListener() && mCardEmulationManager != null) { mCardEmulationManager.onInternalErrorReported( CardEmulation.NFC_INTERNAL_ERROR_NFC_CRASH_RESTART); } storeNativeCrashLogs(); mDeviceHost.doAbort(getName()); } public synchronized void cancel() { synchronized (mCancelWaiter) { mCanceled = true; mCancelWaiter.notify(); } } } static byte[] hexStringToBytes(String s) { if (s == null || s.length() == 0) return null; int len = s.length(); if (len % 2 != 0) { s = '0' + s; len++; } byte[] data = new byte[len / 2]; for (int i = 0; i < len; i += 2) { int high = Character.digit(s.charAt(i), 16); int low = Character.digit(s.charAt(i + 1), 16); if (high == -1 || low == -1) { Log.e(TAG, "Invalid hex character found."); return null; } data[i / 2] = (byte) ((high << 4) + low); } return data; } private void addDeviceLockedStateListener() { if (android.app.Flags.deviceUnlockListener() && Flags.useDeviceLockListener()) { try { mKeyguard.addDeviceLockedStateListener( mContext.getMainExecutor(), mDeviceLockedStateListener); } catch (Exception e) { Log.e(TAG, "addDeviceLockedStateListener: e=" + e); } } else { try { mKeyguard.addKeyguardLockedStateListener(mContext.getMainExecutor(), mIKeyguardLockedStateListener); } catch (Exception e) { Log.e(TAG, "addDeviceLockedStateListener: Exception in addKeyguardLockedStateListener " + e); } } } /** * Receives KeyGuard lock state updates */ private KeyguardLockedStateListener mIKeyguardLockedStateListener = new KeyguardLockedStateListener() { @Override public void onKeyguardLockedStateChanged(boolean isKeyguardLocked) { if (!mIsWlcCapable || !mNfcCharging.NfcChargingOnGoing) { applyScreenState(mScreenStateHelper.checkScreenState(mCheckDisplayStateForScreenState)); } } }; /** * Receives Device lock state updates */ private DeviceLockedStateListener mDeviceLockedStateListener = new DeviceLockedStateListener() { @Override public void onDeviceLockedStateChanged(boolean isDeviceLocked) { if (!mIsWlcCapable || !mNfcCharging.NfcChargingOnGoing) { applyScreenState(mScreenStateHelper.checkScreenState( mCheckDisplayStateForScreenState)); } } }; private void addThermalStatusListener() { try { if (mPowerManager != null) { mPowerManager.addThermalStatusListener(mContext.getMainExecutor(), mOnThermalStatusChangedListener); } } catch (Exception e) { Log.e(TAG, "addThermalStatusListener: e=" + e); } } /** * Receives Thermal state updates */ private OnThermalStatusChangedListener mOnThermalStatusChangedListener = new OnThermalStatusChangedListener() { @Override public void onThermalStatusChanged(int status) { switch (status) { case PowerManager.THERMAL_STATUS_MODERATE: Log.d(TAG, "onThermalStatusChanged: THERMAL_STATUS_MODERATE"); break; case PowerManager.THERMAL_STATUS_SEVERE: Log.d(TAG, "onThermalStatusChanged: THERMAL_STATUS_SEVERE"); break; case PowerManager.THERMAL_STATUS_CRITICAL: Log.d(TAG, "onThermalStatusChanged: THERMAL_STATUS_CRITICAL"); break; default: Log.d(TAG, "onThermalStatusChanged: : " + status); break; } } }; /** * Read mScreenState and apply NFC-C polling and NFC-EE routing */ void applyRouting(boolean force) { if (DBG) { Log.d(TAG, "applyRouting"); } synchronized (this) { if (!isNfcEnabledOrShuttingDown()) { return; } if (mNfcOemExtensionCallback != null && receiveOemCallbackResult(ACTION_ON_APPLY_ROUTING)) { Log.d(TAG, "applyRouting: skip due to oem callback"); return; } refreshTagDispatcherInProvisionMode(); if (mPollingPaused) { Log.d(TAG, "applyRouting: Not updating discovery parameters, polling paused"); return; } // Special case: if we're transitioning to unlocked state while // still talking to a tag, postpone re-configuration. if (mScreenState == ScreenStateHelper.SCREEN_STATE_ON_UNLOCKED && isTagPresent()) { Log.d(TAG, "applyRouting: Not updating discovery parameters, tag connected"); mHandler.sendMessageDelayed(mHandler.obtainMessage(MSG_RESUME_POLLING), APPLY_ROUTING_RETRY_TIMEOUT_MS); return; } WatchDogThread watchDog = new WatchDogThread("applyRouting", ROUTING_WATCHDOG_MS); try { watchDog.start(); // Compute new polling parameters NfcDiscoveryParameters newParams = computeDiscoveryParameters(mScreenState); if (force || !newParams.equals(mCurrentDiscoveryParameters)) { if (newParams.shouldEnableDiscovery()) { boolean shouldRestart = mCurrentDiscoveryParameters.shouldEnableDiscovery(); mDeviceHost.enableDiscovery(newParams, shouldRestart); } else { mDeviceHost.disableDiscovery(); } mCurrentDiscoveryParameters = newParams; } else { Log.d(TAG, "applyRouting: Discovery configuration equal, not updating"); } } finally { watchDog.cancel(); } } } private NfcDiscoveryParameters computeDiscoveryParameters(int screenState) { // Recompute discovery parameters based on screen state NfcDiscoveryParameters.Builder paramsBuilder = NfcDiscoveryParameters.newBuilder(); // Polling if (screenState >= NFC_POLLING_MODE && isReaderOptionEnabled()) { // Check if reader-mode is enabled if (mReaderModeParams != null) { int techMask = 0; if ((mReaderModeParams.flags & NfcAdapter.FLAG_READER_NFC_A) != 0) techMask |= NFC_POLL_A; if ((mReaderModeParams.flags & NfcAdapter.FLAG_READER_NFC_B) != 0) techMask |= NFC_POLL_B; if ((mReaderModeParams.flags & NfcAdapter.FLAG_READER_NFC_F) != 0) techMask |= NFC_POLL_F; if ((mReaderModeParams.flags & NfcAdapter.FLAG_READER_NFC_V) != 0) techMask |= NFC_POLL_V; if ((mReaderModeParams.flags & NfcAdapter.FLAG_READER_NFC_BARCODE) != 0) techMask |= NFC_POLL_KOVIO; paramsBuilder.setTechMask(techMask); paramsBuilder.setEnableReaderMode(true); if (mReaderModeParams.flags != 0 && techMask == 0) { paramsBuilder.setEnableHostRouting(true); } } else { paramsBuilder.setTechMask(NfcDiscoveryParameters.NFC_POLL_DEFAULT); } } else if (screenState == SCREEN_STATE_ON_LOCKED && mInProvisionMode) { paramsBuilder.setTechMask(NfcDiscoveryParameters.NFC_POLL_DEFAULT); } else if (screenState == SCREEN_STATE_ON_LOCKED && mNfcUnlockManager.isLockscreenPollingEnabled() && isReaderOptionEnabled()) { int techMask = 0; if (mNfcUnlockManager.isLockscreenPollingEnabled()) techMask |= mNfcUnlockManager.getLockscreenPollMask(); paramsBuilder.setTechMask(techMask); paramsBuilder.setEnableLowPowerDiscovery(false); } if (mReaderModeParams != null && mReaderModeParams.annotation != null) { paramsBuilder.setTechAPollingLoopAnnotation(mReaderModeParams.annotation); } if (mIsHceCapable) { // Host routing is always enabled, provided we aren't in reader mode if (mReaderModeParams == null || mReaderModeParams.flags == DISABLE_POLLING_FLAGS) { paramsBuilder.setEnableHostRouting(true); } } return paramsBuilder.build(); } private boolean isTagPresent() { for (Object object : mObjectMap.values()) { if (object instanceof TagEndpoint) { return ((TagEndpoint) object).isPresent(); } } return false; } private void refreshTagDispatcherInProvisionMode() { if (mInProvisionMode) { mInProvisionMode = mNfcInjector.isInProvisionMode(); if (!mInProvisionMode) { mNfcDispatcher.disableProvisioningMode(); } } } private boolean isPresenceCheckStopped() { boolean isStopped = false; synchronized (this) { Object[] objectValues = mObjectMap.values().toArray(); for (Object object : objectValues) { if (object instanceof TagEndpoint) { if (((TagEndpoint) object).isPresenceCheckStopped()) { isStopped = true; } } } } return isStopped; } /** * Stops the Presence check thread without calling * Disconnect API and onTagDisconnect callback */ private void prepareForRemovalDetectionMode() { synchronized (this) { Object[] objectValues = mObjectMap.values().toArray(); for (Object object : objectValues) { if (object instanceof TagEndpoint) { ((TagEndpoint) object).prepareForRemovalDetectionMode(); } } } } private void StopPresenceChecking() { Object[] objectValues = mObjectMap.values().toArray(); if (!ArrayUtils.isEmpty(objectValues)) { // If there are some tags connected, we need to execute the callback to indicate // the tag is being forcibly disconnected. executeOemOnTagConnectedCallback(false); } for (Object object : objectValues) { if (object instanceof TagEndpoint) { TagEndpoint tag = (TagEndpoint)object; ((TagEndpoint) object).stopPresenceChecking(); } } } /** * Disconnect any target if present */ void maybeDisconnectTarget() { if (!isNfcEnabledOrShuttingDown()) { return; } clearAppInactivityDetectionContext(); 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, "maybeDisconnectTarget: " + o.getClass().getName()); if (o instanceof TagEndpoint) { // Disconnect from tags TagEndpoint tag = (TagEndpoint) o; tag.disconnect(); } } } Object findObject(int key) { synchronized (this) { Object device = mObjectMap.get(key); if (device == null) { Log.w(TAG, "maybeDisconnectTarget: Handle not found"); } return device; } } Object findAndRemoveObject(int handle) { synchronized (this) { Object device = mObjectMap.get(handle); if (device == null) { Log.w(TAG, "findAndRemoveObject: Handle not found"); } else { mObjectMap.remove(handle); } return device; } } void registerTagObject(TagEndpoint tag) { synchronized (this) { mObjectMap.put(tag.getHandle(), tag); } } void unregisterObject(int handle) { synchronized (this) { mObjectMap.remove(handle); } } public int getAidRoutingTableSize () { int aidTableSize = 0x00; aidTableSize = mDeviceHost.getAidTableSize(); return aidTableSize; } public void sendMockNdefTag(NdefMessage msg) { sendMessage(MSG_MOCK_NDEF, msg); } public void routeAids(String aid, int route, int aidInfo, int power) { Message msg = mHandler.obtainMessage(); msg.what = MSG_ROUTE_AID; msg.arg1 = route; msg.obj = aid; msg.arg2 = aidInfo; Bundle aidPowerState = new Bundle(); aidPowerState.putInt(MSG_ROUTE_AID_PARAM_TAG, power); msg.setData(aidPowerState); mHandler.sendMessage(msg); } public void unrouteAids(String aid) { sendMessage(MSG_UNROUTE_AID, aid); } public int getNciVersion() { return mDeviceHost.getNciVersion(); } private byte[] getT3tIdentifierBytes(String systemCode, String nfcId2, String t3tPmm) { ByteBuffer buffer = ByteBuffer.allocate(2 + 8 + 8); /* systemcode + nfcid2 + t3tpmm */ buffer.put(hexStringToBytes(systemCode)); buffer.put(hexStringToBytes(nfcId2)); buffer.put(hexStringToBytes(t3tPmm)); byte[] t3tIdBytes = new byte[buffer.position()]; buffer.position(0); buffer.get(t3tIdBytes); return t3tIdBytes; } public void registerT3tIdentifier(String systemCode, String nfcId2, String t3tPmm) { Log.d(TAG, "registerT3tIdentifier"); byte[] t3tIdentifier = getT3tIdentifierBytes(systemCode, nfcId2, t3tPmm); sendMessage(MSG_REGISTER_T3T_IDENTIFIER, t3tIdentifier); } public void deregisterT3tIdentifier(String systemCode, String nfcId2, String t3tPmm) { Log.d(TAG, "deregisterT3tIdentifier"); byte[] t3tIdentifier = getT3tIdentifierBytes(systemCode, nfcId2, t3tPmm); sendMessage(MSG_DEREGISTER_T3T_IDENTIFIER, t3tIdentifier); } public void clearT3tIdentifiersCache() { Log.d(TAG, "clearT3tIdentifiersCache"); mDeviceHost.clearT3tIdentifiersCache(); } public int getLfT3tMax() { return mDeviceHost.getLfT3tMax(); } public int commitRouting(boolean isOverrideOrRecover) { if (!isOverrideOrRecover) { // Clear the Handler queue, only last commit_msg is relevant mHandler.removeMessages(MSG_COMMIT_ROUTING); mHandler.sendEmptyMessage(MSG_COMMIT_ROUTING); return STATUS_OK; } mCommitRoutingCountDownLatch = new CountDownLatch(1); mHandler.sendEmptyMessage(MSG_COMMIT_ROUTING); try { boolean success = mCommitRoutingCountDownLatch .await(WAIT_FOR_COMMIT_ROUTING_TIMEOUT_MS, TimeUnit.MILLISECONDS); if (!success) { Log.e(TAG, "commitRouting: timed out!"); return STATUS_UNKNOWN_ERROR; } else { Log.i(TAG, "commitRouting: status= " + mCommitRoutingStatus); return mCommitRoutingStatus; } } catch (InterruptedException e) { return STATUS_UNKNOWN_ERROR; } finally { mCommitRoutingCountDownLatch = null; } } public boolean sendScreenMessageAfterNfcCharging() { if (DBG) Log.d(TAG, "sendScreenMessageAfterNfcCharging"); if (mPendingPowerStateUpdate == true) { int screenState = mScreenStateHelper.checkScreenState(mCheckDisplayStateForScreenState); if (DBG) { Log.d(TAG, "sendScreenMessageAfterNfcCharging: applying postponed screen state " + screenState); } NfcService.getInstance().sendMessage(MSG_APPLY_SCREEN_STATE, screenState); mPendingPowerStateUpdate = false; return true; } return false; } /** * get default T4TNfcee power state supported */ private int getT4tNfceePowerState() { int powerState = mDeviceHost.getT4TNfceePowerState(); synchronized (NfcService.this) { if (mIsSecureNfcEnabled) { /* Secure nfc on,Setting power state screen on unlocked */ powerState = POWER_STATE_SWITCH_ON; } } if (DBG) Log.d(TAG, "T4TNfceePowerState: " + powerState); return powerState; } /** * get info on NDEF-NFCEE feature from HAL config file */ public boolean isNdefNfceefeatureEnabled() { return mDeviceHost.isNdefNfceefeatureEnabled(); } public boolean sendData(byte[] data) { notifyOemLogEvent(new OemLogItems.Builder(OemLogItems.LOG_ACTION_HCE_DATA) .setApduResponse(data).build()); return mDeviceHost.sendRawFrame(data); } public void onPreferredPaymentChanged(int reason) { sendMessage(MSG_PREFERRED_PAYMENT_CHANGED, reason); } public void onPreferredSimChanged() { mHandler.sendEmptyMessage(MSG_PREFERRED_SIM_CHANGED); } public void clearRoutingTable(int clearFlags) { //Remove any previously sent messages not yet processed mHandler.removeMessages(MSG_COMMIT_ROUTING); mHandler.removeMessages(MSG_ROUTE_AID); mHandler.removeMessages(MSG_CLEAR_ROUTING_TABLE); mHandler.removeMessages(MSG_UNROUTE_AID); sendMessage(MSG_CLEAR_ROUTING_TABLE, clearFlags); } public void setIsoDepProtocolRoute(int route) { sendMessage(MSG_UPDATE_ISODEP_PROTOCOL_ROUTE, route); } /** * Set NFCC technology routing for ABF listening */ public void setTechnologyABFRoute(int route, int felicaRoute) { Message msg = mHandler.obtainMessage(); msg.what = MSG_UPDATE_TECHNOLOGY_ABF_ROUTE; msg.arg1 = route; msg.arg2 = felicaRoute; mHandler.sendMessage(msg); } public void setSystemCodeRoute(int route) { sendMessage(MSG_UPDATE_SYSTEM_CODE_ROUTE, route); } void sendMessage(int what, Object obj) { Message msg = mHandler.obtainMessage(); msg.what = what; msg.obj = obj; mHandler.sendMessage(msg); } /** * Send require device unlock for NFC intent to system UI. */ public void sendRequireUnlockIntent() { if (!mIsRequestUnlockShowed && mNfcInjector.isDeviceLocked()) { if (DBG) Log.d(TAG, "sendRequireUnlockIntent"); mIsRequestUnlockShowed = true; mRequireUnlockWakeLock.acquire(); Intent requireUnlockIntent = new Intent(NfcAdapter.ACTION_REQUIRE_UNLOCK_FOR_NFC); requireUnlockIntent.setPackage(SYSTEM_UI); mContext.sendBroadcast(requireUnlockIntent); mRequireUnlockWakeLock.release(); } } final class NfcServiceHandler extends Handler { public NfcServiceHandler(Looper looper) { super(looper); } @Override public void handleMessage(Message msg) { switch (msg.what) { case MSG_ROUTE_AID: { Log.d(TAG, "handleMessage: MSG_ROUTE_AID"); int route = msg.arg1; int aidInfo = msg.arg2; String aid = (String) msg.obj; int power = 0x00; Bundle bundle = msg.getData(); if (bundle != null) { power = bundle.getInt(MSG_ROUTE_AID_PARAM_TAG); } mDeviceHost.routeAid(hexStringToBytes(aid), route, aidInfo, power); // Restart polling config break; } case MSG_UNROUTE_AID: { Log.d(TAG, "handleMessage: MSG_UNROUTE_AID"); String aid = (String) msg.obj; mDeviceHost.unrouteAid(hexStringToBytes(aid)); break; } case MSG_REGISTER_T3T_IDENTIFIER: { Log.d(TAG, "handleMessage: MSG_REGISTER_T3T_IDENTIFIER"); mDeviceHost.disableDiscovery(); byte[] t3tIdentifier = (byte[]) msg.obj; mDeviceHost.registerT3tIdentifier(t3tIdentifier); NfcDiscoveryParameters params = computeDiscoveryParameters(mScreenState); boolean shouldRestart = mCurrentDiscoveryParameters.shouldEnableDiscovery(); mDeviceHost.enableDiscovery(params, shouldRestart); break; } case MSG_DEREGISTER_T3T_IDENTIFIER: { Log.d(TAG, "handleMessage: MSG_DEREGISTER_T3T_IDENTIFIER"); mDeviceHost.disableDiscovery(); byte[] t3tIdentifier = (byte[]) msg.obj; mDeviceHost.deregisterT3tIdentifier(t3tIdentifier); NfcDiscoveryParameters params = computeDiscoveryParameters(mScreenState); boolean shouldRestart = mCurrentDiscoveryParameters.shouldEnableDiscovery(); mDeviceHost.enableDiscovery(params, shouldRestart); break; } case MSG_COMMIT_ROUTING: { Log.d(TAG, "handleMessage: MSG_COMMIT_ROUTING"); synchronized (NfcService.this) { if (isNfcDisabledOrDisabling()) { Log.d(TAG, "handleMessage: Skip commit routing when NFCC is off " + "or turning off"); if (mCommitRoutingCountDownLatch != null) { mCommitRoutingStatus = STATUS_UNKNOWN_ERROR; mCommitRoutingCountDownLatch.countDown(); } return; } if (mCurrentDiscoveryParameters.shouldEnableDiscovery()) { if (mNfcOemExtensionCallback != null) { if (receiveOemCallbackResult(ACTION_ON_ROUTING_CHANGED)) { Log.e(TAG, "handleMessage: Oem skip commitRouting"); if (mCommitRoutingCountDownLatch != null) { mCommitRoutingStatus = STATUS_UNKNOWN_ERROR; mCommitRoutingCountDownLatch.countDown(); } return; } } mCommitRoutingStatus = mDeviceHost.commitRouting(); if (mCommitRoutingCountDownLatch != null) { mCommitRoutingCountDownLatch.countDown(); } } else { Log.d(TAG, "handleMessage: Not committing routing because " + "discovery is disabled"); } } break; } case MSG_MOCK_NDEF: { Log.d(TAG, "handleMessage: 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); /* Avoid setting mCookieUpToDate to negative values */ mCookieUpToDate = mCookieGenerator.nextLong() >>> 1; Tag tag = Tag.createMockTag(new byte[]{0x00}, new int[]{TagTechnology.NDEF}, new Bundle[]{extras}, mCookieUpToDate); Log.d(TAG, "handleMessage: mock NDEF tag, starting corresponding activity"); Log.d(TAG, tag.toString()); int dispatchStatus = mNfcDispatcher.dispatchTag(tag); if (dispatchStatus == NfcDispatcher.DISPATCH_SUCCESS) { playSound(SOUND_END); } else if (dispatchStatus == NfcDispatcher.DISPATCH_FAIL) { playSound(SOUND_ERROR); } break; } case MSG_NDEF_TAG: if (!isNfcEnabled()) break; if (DBG) Log.d(TAG, "Tag detected, notifying applications"); clearAppInactivityDetectionContext(); TagEndpoint tag = (TagEndpoint) msg.obj; byte[] debounceTagUid; int debounceTagMs; ITagRemovedCallback debounceTagRemovedCallback; synchronized (NfcService.this) { debounceTagUid = mDebounceTagUid; debounceTagMs = mDebounceTagDebounceMs; debounceTagRemovedCallback = mDebounceTagRemovedCallback; } ReaderModeParams readerParams = null; int presenceCheckDelay = DEFAULT_PRESENCE_CHECK_DELAY; DeviceHost.TagDisconnectedCallback callback = new DeviceHost.TagDisconnectedCallback() { @Override public void onTagDisconnected() { mCookieUpToDate = -1; clearAppInactivityDetectionContext(); executeOemOnTagConnectedCallback(false); applyRouting(false); } }; synchronized (NfcService.this) { readerParams = mReaderModeParams; } executeOemOnTagConnectedCallback(true); if (mNfcOemExtensionCallback != null && receiveOemCallbackResult(ACTION_ON_READ_NDEF)) { Log.d(TAG, "handleMessage: skip due to oem callback"); tag.startPresenceChecking(presenceCheckDelay, callback); break; } if (readerParams != null) { presenceCheckDelay = readerParams.presenceCheckDelay; if ((readerParams.flags & NfcAdapter.FLAG_READER_SKIP_NDEF_CHECK) != 0) { if (DBG) { Log.d(TAG, "handleMessage: Skipping NDEF detection in reader mode"); } tag.startPresenceChecking(presenceCheckDelay, callback); dispatchTagEndpoint(tag, readerParams); break; } if (mIsDebugBuild && mSkipNdefRead) { if (DBG) { Log.d(TAG, "handleMessage: Only NDEF detection in reader mode"); } tag.findNdef(); tag.startPresenceChecking(presenceCheckDelay, callback); dispatchTagEndpoint(tag, readerParams); break; } } if (tag.getConnectedTechnology() == TagTechnology.NFC_BARCODE) { // When these tags start containing NDEF, they will require // the stack to deal with them in a different way, since // they are activated only really shortly. // For now, don't consider NDEF on these. if (DBG) { Log.d(TAG, "handleMessage: Skipping NDEF detection for NFC Barcode"); } tag.startPresenceChecking(presenceCheckDelay, callback); dispatchTagEndpoint(tag, readerParams); if (readerParams == null) { scheduleAppInactivityDetectionTask(); } break; } NdefMessage ndefMsg = tag.findAndReadNdef(); if (ndefMsg == null) { // First try to see if this was a bad tag read if (!tag.reconnect()) { tag.disconnect(); if (DBG) Log.d(TAG, "handleMessage: Read NDEF error"); executeOemOnTagConnectedCallback(false); if (mScreenState == ScreenStateHelper.SCREEN_STATE_ON_UNLOCKED) { if (mReadErrorCount < mReadErrorCountMax) { mReadErrorCount++; } else { pollingDelay(); } if (!sToast_debounce && mNotifyReadFailed) { Toast.makeText(mContext, R.string.tag_read_error, Toast.LENGTH_SHORT).show(); sToast_debounce = true; mHandler.sendEmptyMessageDelayed(MSG_TOAST_DEBOUNCE_EVENT, sToast_debounce_time_ms); } } break; } } if (debounceTagUid != null) { // If we're debouncing and the UID or the NDEF message of the tag match, // don't dispatch but drop it. if (Arrays.equals(debounceTagUid, tag.getUid()) || (ndefMsg != null && ndefMsg.equals(mLastReadNdefMessage))) { mHandler.removeMessages(MSG_TAG_DEBOUNCE); mHandler.sendEmptyMessageDelayed(MSG_TAG_DEBOUNCE, debounceTagMs); tag.disconnect(); return; } else { synchronized (NfcService.this) { mDebounceTagUid = null; mDebounceTagRemovedCallback = null; mDebounceTagNativeHandle = INVALID_NATIVE_HANDLE; } if (debounceTagRemovedCallback != null) { try { debounceTagRemovedCallback.onTagRemoved(); } catch (RemoteException e) { // Ignore } } } } mLastReadNdefMessage = ndefMsg; if (mIsWlcEnabled) { if (DBG) Log.d(TAG, "handleMessage: Wlc enabled, check for WLC_CAP record"); if (!mNfcCharging.NfcChargingMode && (mNfcCharging.checkWlcCapMsg(ndefMsg) == true)) { if (DBG) Log.d(TAG, "handleMessage: checkWlcCapMsg returned true"); if (mNfcCharging.startNfcCharging(tag)) { mNfcCharging.NfcChargingMode = true; if (DBG) { Log.d(TAG, "handleMessage: " + "Nfc charging mode started successfully"); } } else { if (DBG) { Log.d(TAG, "handleMessage: Nfc charging mode not detected"); } } } else if (mIsRWCapable) { tag.startPresenceChecking(presenceCheckDelay, callback); dispatchTagEndpoint(tag, readerParams); } else { tag.startPresenceChecking(presenceCheckDelay, callback); } } else if (mIsRWCapable) { tag.startPresenceChecking(presenceCheckDelay, callback); dispatchTagEndpoint(tag, readerParams); } else { tag.startPresenceChecking(presenceCheckDelay, callback); } if (readerParams == null) { scheduleAppInactivityDetectionTask(); } break; case MSG_RF_FIELD_ACTIVATED: Log.d(TAG, "handleMessage: MSG_RF_FIELD_ACTIVATED"); notifyOemLogEvent(new OemLogItems .Builder(OemLogItems.LOG_ACTION_RF_FIELD_STATE_CHANGED) .setRfFieldOnTime(Instant.now()).build()); if (mCardEmulationManager != null) { mCardEmulationManager.onFieldChangeDetected(true); } Intent fieldOnIntent = new Intent(ACTION_RF_FIELD_ON_DETECTED); sendNfcPermissionProtectedBroadcast(fieldOnIntent); if (mIsSecureNfcEnabled) { sendRequireUnlockIntent(); } break; case MSG_RF_FIELD_DEACTIVATED: Log.d(TAG, "handleMessage: MSG_RF_FIELD_DEACTIVATED"); notifyOemLogEvent(new OemLogItems .Builder(OemLogItems.LOG_ACTION_RF_FIELD_STATE_CHANGED) .setRfFieldOnTime(Instant.now()).build()); if (mCardEmulationManager != null) { mCardEmulationManager.onFieldChangeDetected(false); } Intent fieldOffIntent = new Intent(ACTION_RF_FIELD_OFF_DETECTED); sendNfcPermissionProtectedBroadcast(fieldOffIntent); break; case MSG_RESUME_POLLING: Log.d(TAG, "handleMessage: MSG_RESUME_POLLING"); mNfcAdapter.resumePolling(); break; case MSG_TAG_DEBOUNCE: Log.d(TAG, "handleMessage: MSG_TAG_DEBOUNCE"); // Didn't see the tag again, tag is gone ITagRemovedCallback tagRemovedCallback; synchronized (NfcService.this) { mDebounceTagUid = null; tagRemovedCallback = mDebounceTagRemovedCallback; mDebounceTagRemovedCallback = null; mDebounceTagNativeHandle = INVALID_NATIVE_HANDLE; } if (tagRemovedCallback != null) { try { tagRemovedCallback.onTagRemoved(); } catch (RemoteException e) { // Ignore } } break; case MSG_APPLY_SCREEN_STATE: mScreenState = (Integer)msg.obj; Log.d(TAG, "handleMessage: MSG_APPLY_SCREEN_STATE" + ScreenStateHelper.screenStateToString(mScreenState)); synchronized (NfcService.this) { // Disable delay polling when screen state changed mPollDelayed = false; mHandler.removeMessages(MSG_DELAY_POLLING); // If NFC is turning off, we shouldn't need any changes here if (mState == NfcAdapter.STATE_TURNING_OFF) return; } notifyOemLogEvent( new OemLogItems.Builder(OemLogItems.LOG_ACTION_SCREEN_STATE_CHANGED) .build()); mRoutingWakeLock.acquire(); try { if (mScreenState == ScreenStateHelper.SCREEN_STATE_ON_UNLOCKED) { applyRouting(false); mIsRequestUnlockShowed = false; } int screen_state_mask = (mNfcUnlockManager.isLockscreenPollingEnabled()) ? (ScreenStateHelper.SCREEN_POLLING_TAG_MASK | mScreenState) : mScreenState; if (mNfcUnlockManager.isLockscreenPollingEnabled()) { applyRouting(false); } mDeviceHost.doSetScreenState(screen_state_mask, mIsWlcEnabled); } finally { if (mRoutingWakeLock.isHeld()) { mRoutingWakeLock.release(); } } break; case MSG_TRANSACTION_EVENT: Log.d(TAG, "handleMessage: MSG_TRANSACTION_EVENT"); if (mCardEmulationManager != null) { mCardEmulationManager.onOffHostAidSelected(); } byte[][] data = (byte[][]) msg.obj; synchronized (NfcService.this) { sendOffHostTransactionEvent(data[0], data[1], data[2]); } break; case MSG_SE_SELECTED_EVENT: Log.d(TAG, "handleMessage: MSG_SE_SELECTED_EVENT"); int type = (int) msg.obj; if (mCardEmulationManager != null && type == SE_SELECTED_AID) { mCardEmulationManager.onOffHostAidSelected(); } break; case MSG_PREFERRED_PAYMENT_CHANGED: Log.d(TAG, "handleMessage: MSG_PREFERRED_PAYMENT_CHANGED"); Intent preferredPaymentChangedIntent = new Intent(NfcAdapter.ACTION_PREFERRED_PAYMENT_CHANGED); preferredPaymentChangedIntent.putExtra( NfcAdapter.EXTRA_PREFERRED_PAYMENT_CHANGED_REASON, (int)msg.obj); sendPreferredPaymentChangedEvent(preferredPaymentChangedIntent); break; case MSG_TOAST_DEBOUNCE_EVENT: Log.d(TAG, "handleMessage: MSG_TOAST_DEBOUNCE_EVENT"); sToast_debounce = false; break; case MSG_DELAY_POLLING: Log.d(TAG, "handleMessage: MSG_DELAY_POLLING"); synchronized (NfcService.this) { if (!mPollDelayed) { return; } mPollDelayed = false; mDeviceHost.startStopPolling(true); } if (DBG) Log.d(TAG, "handleMessage: Polling is started"); break; case MSG_CLEAR_ROUTING_TABLE: if (!isNfcEnabled()) break; Log.d(TAG, "handleMessage: MSG_CLEAR_ROUTING_TABLE"); int clearFlags = (Integer)msg.obj; mDeviceHost.clearRoutingEntry(clearFlags); break; case MSG_UPDATE_ISODEP_PROTOCOL_ROUTE: Log.d(TAG, "handleMessage: MSG_UPDATE_ISODEP_PROTOCOL_ROUTE"); if (isNfcEnabled()) { mDeviceHost.setIsoDepProtocolRoute((Integer) msg.obj); } break; case MSG_UPDATE_TECHNOLOGY_ABF_ROUTE: Log.d(TAG, "handleMessage: MSG_UPDATE_TECHNOLOGY_ABF_ROUTE"); int msgRoute = msg.arg1; int felicaRoute = msg.arg2; if (isNfcEnabled()) { mDeviceHost.setTechnologyABFRoute(msgRoute, felicaRoute); } break; case MSG_WATCHDOG_PING: Log.d(TAG, "handleMessage: MSG_WATCHDOG_PING"); NfcWatchdog watchdog = (NfcWatchdog) msg.obj; watchdog.notifyHasReturned(); if (mLastFieldOnTimestamp + TIME_TO_MONITOR_AFTER_FIELD_ON_MS > mNfcInjector.getWallClockMillis()) { watchdog.stopMonitoring(); } break; case MSG_UPDATE_SYSTEM_CODE_ROUTE: Log.d(TAG, "handleMessage: MSG_UPDATE_SYSTEM_CODE_ROUTE"); mDeviceHost.setSystemCodeRoute((Integer) msg.obj); break; case MSG_PREFERRED_SIM_CHANGED: if (!isNfcEnabled()) break; Log.d(TAG, "handleMessage: MSG_PREFERRED_SIM_CHANGED"); new EnableDisableTask().execute(TASK_DISABLE); new EnableDisableTask().execute(TASK_ENABLE); break; case MSG_RESTART_DISCOVERY: Log.d(TAG, "handleMessage: MSG_RESTART_DISCOVERY"); mDeviceHost.restartRfDiscovery(); break; default: Log.e(TAG, "handleMessage: Unknown message received"); break; } } private void sendOffHostTransactionEvent(byte[] aid, byte[] data, byte[] readerByteArray) { String reader = ""; int uid = -1; int offhostCategory = NfcStatsLog.NFC_CARDEMULATION_OCCURRED__CATEGORY__OFFHOST; try { StringBuilder aidString = new StringBuilder(aid.length); for (byte b : aid) { aidString.append(String.format("%02X", b)); } String aidCategory = mCardEmulationManager .getRegisteredAidCategory(aidString.toString()); if (DBG) { Log.d(TAG, "sendOffHostTransactionEvent: aid category=" + aidCategory); } if (mStatsdUtils != null) { mStatsdUtils.setCardEmulationEventCategory(aidCategory); } else { switch (aidCategory) { case CardEmulation.CATEGORY_PAYMENT: offhostCategory = NfcStatsLog .NFC_CARDEMULATION_OCCURRED__CATEGORY__OFFHOST_PAYMENT; break; case CardEmulation.CATEGORY_OTHER: offhostCategory = NfcStatsLog .NFC_CARDEMULATION_OCCURRED__CATEGORY__OFFHOST_OTHER; break; default: offhostCategory = NfcStatsLog .NFC_CARDEMULATION_OCCURRED__CATEGORY__OFFHOST; }; } reader = new String(readerByteArray, "UTF-8"); if (!isSEServiceAvailable() || mNfcEventInstalledPackages.isEmpty()) { return; } for (int userId : mNfcEventInstalledPackages.keySet()) { List packagesOfUser = mNfcEventInstalledPackages.get(userId); String[] installedPackages = new String[packagesOfUser.size()]; boolean[] nfcAccess = mSEService.isNfcEventAllowed(reader, aid, packagesOfUser.toArray(installedPackages), userId); if (nfcAccess == null) { continue; } Intent intent = new Intent(NfcAdapter.ACTION_TRANSACTION_DETECTED); intent.addFlags(Intent.FLAG_INCLUDE_STOPPED_PACKAGES); intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); intent.putExtra(NfcAdapter.EXTRA_AID, aid); intent.putExtra(NfcAdapter.EXTRA_DATA, data); intent.putExtra(NfcAdapter.EXTRA_SECURE_ELEMENT_NAME, reader); String url = new String("nfc://secure:0/" + reader + "/" + aidString.toString()); intent.setData(Uri.parse(url)); final BroadcastOptions options = BroadcastOptions.makeBasic(); options.setBackgroundActivityStartsAllowed(true); Map hasIntentPackages = mContext .getPackageManager() .queryBroadcastReceiversAsUser(intent, 0, UserHandle.of(userId)) .stream() .collect(Collectors.toMap( activity -> activity.activityInfo.applicationInfo.packageName, activity -> activity.activityInfo.applicationInfo.uid, (packageName1, packageName2) -> { if (DBG) { Log.d(TAG, "sendOffHostTransactionEvent: " + "queryBroadcastReceiversAsUser" + " duplicate: " + packageName1 + ", " + packageName2); } return packageName1; })); if (DBG) { String[] packageNames = hasIntentPackages .keySet().toArray(new String[hasIntentPackages.size()]); Log.d(TAG, "sendOffHostTransactionEvent: queryBroadcastReceiversAsUser: " + Arrays.toString(packageNames)); } boolean foundFirstPackage = false; for (int i = 0; i < nfcAccess.length; i++) { if (nfcAccess[i]) { String packageName = packagesOfUser.get(i); if (DBG) { Log.d(TAG, "sendOffHostTransactionEvent: to " + packageName); } if (!foundFirstPackage && hasIntentPackages.containsKey(packageName)) { uid = hasIntentPackages.get(packageName); if (mStatsdUtils != null) { mStatsdUtils.setCardEmulationEventUid(uid); } foundFirstPackage = true; } intent.setPackage(packagesOfUser.get(i)); mContext.sendBroadcastAsUser(intent, UserHandle.of(userId), null, options.toBundle()); } } } } catch (RemoteException e) { Log.e(TAG, "sendOffHostTransactionEvent: Error in isNfcEventAllowed() " + e); } catch (UnsupportedEncodingException e) { Log.e(TAG, "sendOffHostTransactionEvent: Incorrect format for Secure Element name" + e); } catch (IllegalArgumentException e) { Log.e(TAG, "sendOffHostTransactionEvent: Error " + e); } finally { if (mStatsdUtils != null) { mStatsdUtils.logCardEmulationOffhostEvent(reader); } else { NfcStatsLog.write(NfcStatsLog.NFC_CARDEMULATION_OCCURRED, offhostCategory, reader, uid); } } } private void sendNfcPermissionProtectedBroadcast(Intent intent) { if (mNfcEventInstalledPackages.isEmpty()) { return; } intent.addFlags(Intent.FLAG_INCLUDE_STOPPED_PACKAGES); for (int userId : mNfcEventInstalledPackages.keySet()) { for (String packageName : mNfcEventInstalledPackages.get(userId)) { intent.setPackage(packageName); mContext.sendBroadcastAsUser(intent, UserHandle.of(userId)); } } } /* Returns the list of packages request for nfc preferred payment service changed and * have access to NFC Events on any SE */ private ArrayList getNfcPreferredPaymentChangedSEAccessAllowedPackages(int userId) { synchronized (NfcService.this) { if (!isSEServiceAvailable() || mNfcPreferredPaymentChangedInstalledPackages.get(userId).isEmpty()) { return null; } String[] readers = null; try { readers = mSEService.getReaders(); } catch (RemoteException e) { Log.e(TAG, "getNfcPreferredPaymentChangedSEAccessAllowedPackages: " + "Error in getReaders() " + e); return null; } if (readers == null || readers.length == 0) { return null; } boolean[] nfcAccessFinal = null; List packagesOfUser = mNfcPreferredPaymentChangedInstalledPackages.get(userId); String[] installedPackages = new String[packagesOfUser.size()]; for (String reader : readers) { try { boolean[] accessList = mSEService.isNfcEventAllowed(reader, null, packagesOfUser.toArray(installedPackages), userId ); if (accessList == null) { continue; } if (nfcAccessFinal == null) { nfcAccessFinal = accessList; } for (int i = 0; i < accessList.length; i++) { if (accessList[i]) { nfcAccessFinal[i] = true; } } } catch (RemoteException e) { Log.e(TAG, "getNfcPreferredPaymentChangedSEAccessAllowedPackages: " + "Error in isNfcEventAllowed() " + e); } catch (IllegalArgumentException e) { Log.e(TAG, "getNfcPreferredPaymentChangedSEAccessAllowedPackages: Error " + e); } } if (nfcAccessFinal == null) { return null; } ArrayList packages = new ArrayList(); for (int i = 0; i < nfcAccessFinal.length; i++) { if (nfcAccessFinal[i]) { packages.add(packagesOfUser.get(i)); } } return packages; } } private boolean isSystemApp(ApplicationInfo applicationInfo) { return ((applicationInfo.flags & APP_INFO_FLAGS_SYSTEM_APP) != 0); } private void sendPreferredPaymentChangedEvent(Intent intent) { intent.addFlags(Intent.FLAG_INCLUDE_STOPPED_PACKAGES); // Resume app switches so the receivers can start activities without delay mNfcDispatcher.resumeAppSwitches(); synchronized (this) { for (int userId : mNfcPreferredPaymentChangedInstalledPackages.keySet()) { ArrayList SEPackages = getNfcPreferredPaymentChangedSEAccessAllowedPackages(userId); UserHandle userHandle = UserHandle.of(userId); if (SEPackages != null && !SEPackages.isEmpty()) { for (String packageName : SEPackages) { intent.setPackage(packageName); intent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND); mContext.sendBroadcastAsUser(intent, userHandle); } } PackageManager pm; try { pm = mContext.createContextAsUser(userHandle, /*flags=*/0) .getPackageManager(); } catch (IllegalStateException e) { Log.d(TAG, "sendPreferredPaymentChangedEvent: " + "Fail to get PackageManager for user: " + userHandle); continue; } for (String pkgName : mNfcPreferredPaymentChangedInstalledPackages.get(userId)) { try { PackageInfo info = pm.getPackageInfo(pkgName, 0); if (SEPackages != null && SEPackages.contains(pkgName)) { continue; } if (info.applicationInfo != null && (isSystemApp(info.applicationInfo) || info.applicationInfo.isPrivilegedApp())) { intent.setPackage(pkgName); intent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND); mContext.sendBroadcastAsUser(intent, userHandle); } } catch (Exception e) { Log.e(TAG, "sendPreferredPaymentChangedEvent: Exception in getPackageInfo " + e); } } } } } private void pollingDelay() { if (mPollDelayTime <= NO_POLL_DELAY) return; synchronized (NfcService.this) { if (!mPollDelayed) { mPollDelayed = true; mDeviceHost.startStopPolling(false); int delayTime = mPollDelayTime; if (mPollDelayCount < mPollDelayCountMax) { mPollDelayCount++; } else { delayTime = mPollDelayTimeLong; } if (DBG) Log.d(TAG, "pollingDelay: delay " + delayTime); mHandler.sendMessageDelayed( mHandler.obtainMessage(MSG_DELAY_POLLING), delayTime); } else { if (DBG) Log.d(TAG, "pollingDelay: Keep waiting"); } } } private void dispatchTagEndpoint(TagEndpoint tagEndpoint, ReaderModeParams readerParams) { if (mNfcOemExtensionCallback != null && receiveOemCallbackResult(ACTION_ON_TAG_DISPATCH)) { Log.d(TAG, "dispatchTagEndpoint: skip due to oem callback"); return; } try { /* Avoid setting mCookieUpToDate to negative values */ mCookieUpToDate = mCookieGenerator.nextLong() >>> 1; Tag tag = new Tag(tagEndpoint.getUid(), tagEndpoint.getTechList(), tagEndpoint.getTechExtras(), tagEndpoint.getHandle(), mCookieUpToDate, mNfcTagService); registerTagObject(tagEndpoint); if (readerParams != null) { try { if ((readerParams.flags & NfcAdapter.FLAG_READER_NO_PLATFORM_SOUNDS) == 0) { mVibrator.vibrate(mVibrationEffect, HARDWARE_FEEDBACK_VIBRATION_ATTRIBUTES); playSound(SOUND_END); } if (readerParams.callback != null) { if (mScreenState == ScreenStateHelper.SCREEN_STATE_ON_UNLOCKED) { mPowerManager.userActivity(SystemClock.uptimeMillis(), PowerManager.USER_ACTIVITY_EVENT_OTHER, 0); } readerParams.callback.onTagDiscovered(tag); return; } else { // Follow normal dispatch below } } catch (RemoteException e) { Log.e(TAG, "dispatchTagEndpoint: Reader mode remote has died, falling back ", e); // Intentional fall-through } catch (Exception e) { // Catch any other exception Log.e(TAG, "dispatchTagEndpoint: App exception, not dispatching ", e); return; } } refreshTagDispatcherInProvisionMode(); int dispatchResult = mNfcDispatcher.dispatchTag(tag); if (dispatchResult == NfcDispatcher.DISPATCH_FAIL && !isEndPointRemovalDetectionSupported()) { if (DBG) Log.d(TAG, "dispatchTagEndpoint: Tag dispatch failed"); executeOemOnTagConnectedCallback(false); unregisterObject(tagEndpoint.getHandle()); if (mPollDelayTime > NO_POLL_DELAY) { pollingDelay(); tagEndpoint.stopPresenceChecking(); } else { Log.d(TAG, "dispatchTagEndpoint: Keep presence checking"); } if (mScreenState == ScreenStateHelper.SCREEN_STATE_ON_UNLOCKED && mNotifyDispatchFailed && !mInProvisionMode) { if (!sToast_debounce) { Toast.makeText(mContext, R.string.tag_dispatch_failed, Toast.LENGTH_SHORT).show(); sToast_debounce = true; mHandler.sendEmptyMessageDelayed(MSG_TOAST_DEBOUNCE_EVENT, sToast_debounce_time_ms); } playSound(SOUND_ERROR); } if (!mAntennaBlockedMessageShown && mDispatchFailedCount++ > mDispatchFailedMax) { new NfcBlockedNotification(mContext).startNotification(); synchronized (NfcService.this) { mPrefsEditor.putBoolean(PREF_ANTENNA_BLOCKED_MESSAGE_SHOWN, true); mPrefsEditor.apply(); } mBackupManager.dataChanged(); mAntennaBlockedMessageShown = true; mDispatchFailedCount = 0; if (DBG) { Log.d(TAG, "dispatchTagEndpoint: Tag dispatch failed notification"); } } } else if (dispatchResult == NfcDispatcher.DISPATCH_SUCCESS) { synchronized (NfcService.this) { mPollDelayCount = 0; mReadErrorCount = 0; } if (mScreenState == ScreenStateHelper.SCREEN_STATE_ON_UNLOCKED) { mPowerManager.userActivity(SystemClock.uptimeMillis(), PowerManager.USER_ACTIVITY_EVENT_OTHER, 0); } mDispatchFailedCount = 0; mVibrator.vibrate(mVibrationEffect, HARDWARE_FEEDBACK_VIBRATION_ATTRIBUTES); playSound(SOUND_END); notifyOemLogEvent(new OemLogItems.Builder(OemLogItems.LOG_ACTION_TAG_DETECTED) .setTag(tag).build()); } } catch (Exception e) { Log.e(TAG, "dispatchTagEndpoint: Tag creation exception, not dispatching ", e); return; } } } private void executeOemOnTagConnectedCallback(boolean connected) { if (mNfcOemExtensionCallback != null) { try { mNfcOemExtensionCallback.onTagConnected(connected); } catch (RemoteException e) { Log.e(TAG, e.toString()); } } } private NfcServiceHandler mHandler; class ApplyRoutingTask extends AsyncTask { @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(); mRoutingWakeLock.acquire(); try { applyRouting(false); } finally { if (mRoutingWakeLock.isHeld()) { mRoutingWakeLock.release(); } } return null; } } } private void handleScreenStateChanged() { // Perform applyRouting() in AsyncTask to serialize blocking calls if (mIsWlcCapable && mNfcCharging.NfcChargingOnGoing) { Log.d(TAG, "handleScreenStateChanged: MSG_APPLY_SCREEN_STATE postponing due " + "to a charging pier device"); mPendingPowerStateUpdate = true; return; } int screenState = mScreenStateHelper.checkScreenState(mCheckDisplayStateForScreenState); if (screenState == SCREEN_STATE_ON_LOCKED || screenState == SCREEN_STATE_ON_UNLOCKED) { synchronized (NfcService.this) { mPollDelayCount = 0; mReadErrorCount = 0; } } applyScreenState(screenState); } boolean isEndPointRemovalDetectionSupported() { if (!(mIsRDCapable && mDeviceHost.isRemovalDetectionInPollModeSupported())) { Log.d(TAG, "isEndPointRemovalDetectionSupported: not supported"); return false; } if (!(mAppInActivityDetectionTime >= MIN_RF_REMOVAL_DETECTION_TIMEOUT && (mTagRemovalDetectionWaitTime >= MIN_RF_REMOVAL_DETECTION_TIMEOUT && mTagRemovalDetectionWaitTime <= MAX_RF_REMOVAL_DETECTION_TIMEOUT))) { Log.d(TAG, "isEndPointRemovalDetectionSupported: Unexpected wait time"); return false; } return true; } void scheduleAppInactivityDetectionTask() { if (isEndPointRemovalDetectionSupported()) { clearAppInactivityDetectionContext(); mAppInActivityDetectionTimer = new Timer(); AppInActivityHandlerTask task = new AppInActivityHandlerTask(); mAppInActivityDetectionTimer.schedule(task,mAppInActivityDetectionTime); Log.d(TAG, "scheduleAppInactivityDetectionTask: scheduled"); } } boolean checkAndHandleRemovalDetectionMode(boolean isDisconnectNeeded) { if (mAppInActivityDetectionTimer != null) { if (isPresenceCheckStopped()) { Log.d(TAG, "checkAndHandleRemovalDetectionMode: Removal detection state"); if (isDisconnectNeeded) { Log.d(TAG, "checkAndHandleRemovalDetectionMode: Restarting discovery.."); maybeDisconnectTarget(); return true; } } else { Log.d(TAG, "checkAndHandleRemovalDetectionMode: Clearing Removal " + "Detection Timer Context"); clearAppInactivityDetectionContext(); } } return false; } class AppInActivityHandlerTask extends TimerTask { public void run() { Log.d(TAG, "run: App Inactivity detected, Requesting to Start Removal " + "Detection Procedure"); if (isTagPresent()) { prepareForRemovalDetectionMode(); mHandler.post(() -> Toast.makeText(mContext, "No activity over reader mode, RF removal detection procedure started", Toast.LENGTH_LONG).show()); /* Request JNI to start remove detection procedure */ startRemovalDetection(mTagRemovalDetectionWaitTime); } else { clearAppInactivityDetectionContext(); } } } void clearAppInactivityDetectionContext() { if (mAppInActivityDetectionTimer != null) { mAppInActivityDetectionTimer.cancel(); mAppInActivityDetectionTimer = null; } } private final BroadcastReceiver mReceiver = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); if (action.equals(Intent.ACTION_SCREEN_ON) || action.equals(Intent.ACTION_SCREEN_OFF) || action.equals(Intent.ACTION_USER_PRESENT)) { handleScreenStateChanged(); } else if (action.equals(Intent.ACTION_BOOT_COMPLETED) && mIsHceCapable) { if (DBG) Log.d(TAG, action + " received"); mCardEmulationManager.onBootCompleted(); } else if (action.equals(Intent.ACTION_USER_SWITCHED)) { int userId = intent.getIntExtra(Intent.EXTRA_USER_HANDLE, 0); mUserId = userId; updatePackageCache(); if (DBG) Log.d(TAG, action + "mReceiver.onReceive: UserId: " + userId); if (mIsHceCapable) { mCardEmulationManager.onUserSwitched(getUserId()); } applyScreenState(mScreenStateHelper.checkScreenState(mCheckDisplayStateForScreenState)); if ((NFC_SNOOP_LOG_MODE.equals(NfcProperties.snoop_log_mode_values.FULL) || NFC_VENDOR_DEBUG_ENABLED) && mDeviceConfigFacade.getEnableDeveloperNotification()){ new NfcDeveloperOptionNotification(mContext.createContextAsUser( UserHandle.of(ActivityManager.getCurrentUser()), /*flags=*/0)) .startNotification(); } // Reload when another userId activated synchronized (NfcService.this) { initTagAppPrefList(); } } else if (action.equals(Intent.ACTION_USER_ADDED)) { int userId = intent.getIntExtra(Intent.EXTRA_USER_HANDLE, 0); setPaymentForegroundPreference(userId); if ((NFC_SNOOP_LOG_MODE.equals(NfcProperties.snoop_log_mode_values.FULL) || NFC_VENDOR_DEBUG_ENABLED) && mDeviceConfigFacade.getEnableDeveloperNotification()) { new NfcDeveloperOptionNotification(mContext.createContextAsUser( UserHandle.of(ActivityManager.getCurrentUser()), /*flags=*/0)) .startNotification(); } } else if (action.equals(Intent.ACTION_USER_UNLOCKED) && mFeatureFlags.enableDirectBootAware()) { // If this is first unlock after upgrading to NFC stack that is direct boot aware, // migrate over the data from CE directory to DE directory for access before user // unlock in subsequent bootups. if (!mPrefs.getBoolean(PREF_MIGRATE_TO_DE_COMPLETE, false)) { Log.i(TAG, "mReceiver.onReceive: Migrating shared prefs to DE directory " + "from CE directory"); Context ceContext = mContext.createCredentialProtectedStorageContext(); SharedPreferences cePreferences = ceContext.getSharedPreferences(PREF, Context.MODE_PRIVATE); SharedPreferences ceTagPreferences = ceContext.getSharedPreferences(PREF_TAG_APP_LIST, Context.MODE_PRIVATE); Log.d(TAG, "mReceiver.onReceive: CE Shared Pref values: " + cePreferences.getAll() + ", " + ceTagPreferences.getAll()); if (cePreferences.getAll().isEmpty()) { Log.d(TAG, "mReceiver.onReceive: No NFC Shared preferences to " + "migrate from CE data"); } else { if (!mContext.moveSharedPreferencesFrom(ceContext, PREF)) { Log.e(TAG, "mReceiver.onReceive: Failed to migrate NFC Shared preferences " + "to DE directory"); return; } } if (ceTagPreferences.getAll().isEmpty()) { Log.d(TAG, "mReceiver.onReceive: No NFC Shared preferences for tag app " + "to migrate from CE data"); } else { if (!mContext.moveSharedPreferencesFrom(ceContext, PREF_TAG_APP_LIST)) { Log.e(TAG, "mReceiver.onReceive: Failed to migrate NFC Shared " + "preferences for tag app list to DE directory"); return; } initTagAppPrefList(); } if (mIsHceCapable) { mCardEmulationManager.migrateSettingsFilesFromCe(ceContext); } // If the move is completed, refresh our reference to the shared preferences. mPrefs = mContext.getSharedPreferences(PREF, Context.MODE_PRIVATE); mPrefsEditor = mPrefs.edit(); mPrefsEditor.putBoolean(PREF_MIGRATE_TO_DE_COMPLETE, true); mPrefsEditor.apply(); } } } }; private final BroadcastReceiver mManagedProfileReceiver = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); UserHandle user = intent.getParcelableExtra(Intent.EXTRA_USER); // User should be filled for below intents, check the existence. if (user == null) { Log.d(TAG, intent.getAction() + "mManagedProfileReceiver.onReceive: broadcast without EXTRA_USER"); return; } if (mCardEmulationManager == null) { return; } if (action.equals(Intent.ACTION_MANAGED_PROFILE_ADDED) || action.equals(Intent.ACTION_MANAGED_PROFILE_AVAILABLE) || action.equals(Intent.ACTION_MANAGED_PROFILE_REMOVED) || action.equals(Intent.ACTION_MANAGED_PROFILE_UNAVAILABLE)) { if (DBG) { Log.d(TAG, action + "mManagedProfileReceiver.onReceive: UserId: " + user.getIdentifier()); } mCardEmulationManager.onManagedProfileChanged(); setPaymentForegroundPreference(user.getIdentifier()); synchronized (NfcService.this) { initTagAppPrefList(); } } } }; private final BroadcastReceiver mOwnerReceiver = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); 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(); renewTagAppPrefList(action); } else if (action.equals(Intent.ACTION_SHUTDOWN)) { if (DBG) { Log.d(TAG, "mOwnerReceiver.onReceive: Shutdown received with UserId: " + getSendingUser().getIdentifier()); } if (!getSendingUser().equals(UserHandle.ALL)) { return; } if (DBG) Log.d(TAG, "mOwnerReceiver.onReceive: Device is shutting down"); if (mIsAlwaysOnSupported && mAlwaysOnState == NfcAdapter.STATE_ON) { new EnableDisableTask().execute(TASK_DISABLE_ALWAYS_ON); } if (isNfcEnabled()) { mDeviceHost.shutdown(); } } } }; private void applyScreenState(int screenState) { if (mFeatureFlags.reduceStateTransition() && mIsWatchType && !mCardEmulationManager.isRequiresScreenOnServiceExist()) { if (screenState == ScreenStateHelper.SCREEN_STATE_OFF_LOCKED) { screenState = SCREEN_STATE_ON_LOCKED; } else if (screenState == ScreenStateHelper.SCREEN_STATE_OFF_UNLOCKED) { screenState = ScreenStateHelper.SCREEN_STATE_ON_UNLOCKED; } } if (DBG) { Log.d(TAG, "applyScreenState: screenState = " + ScreenStateHelper.screenStateToString(screenState)); } if (mScreenState != screenState) { if (nci_version != NCI_VERSION_2_0) { new ApplyRoutingTask().execute(Integer.valueOf(screenState)); } if (DBG) Log.d(TAG, "applyScreenState: screenState != mScreenState=" + mScreenState); sendMessage(MSG_APPLY_SCREEN_STATE, screenState); } } private void setPaymentForegroundPreference(int user) { Context uc; try { uc = mContext.createContextAsUser(UserHandle.of(user), 0); } catch (IllegalStateException e) { Log.d(TAG, "setPaymentForegroundPreference: Fail to get user context for user: " + user); return; } try { // Check whether the Settings.Secure.NFC_PAYMENT_FOREGROUND exists or not. Settings.Secure.getInt(uc.getContentResolver(), Constants.SETTINGS_SECURE_NFC_PAYMENT_FOREGROUND); } catch (SettingNotFoundException e) { boolean foregroundPreference = mContext.getResources().getBoolean(R.bool.payment_foreground_preference); Settings.Secure.putInt(uc.getContentResolver(), Constants.SETTINGS_SECURE_NFC_PAYMENT_FOREGROUND, foregroundPreference ? 1 : 0); Log.d(TAG, "setPaymentForegroundPreference: NFC_PAYMENT_FOREGROUND preference:" + foregroundPreference); } } /** * 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 ""; } } static int stateToProtoEnum(int state) { switch (state) { case NfcAdapter.STATE_OFF: return NfcServiceDumpProto.STATE_OFF; case NfcAdapter.STATE_TURNING_ON: return NfcServiceDumpProto.STATE_TURNING_ON; case NfcAdapter.STATE_ON: return NfcServiceDumpProto.STATE_ON; case NfcAdapter.STATE_TURNING_OFF: return NfcServiceDumpProto.STATE_TURNING_OFF; default: return NfcServiceDumpProto.STATE_UNKNOWN; } } private void copyNativeCrashLogsIfAny(PrintWriter pw) { try { File file = new File(NATIVE_LOG_FILE_PATH, NATIVE_LOG_FILE_NAME); if (!file.exists()) { return; } pw.println("---BEGIN: NATIVE CRASH LOG----"); Scanner sc = new Scanner(file); while(sc.hasNextLine()) { String s = sc.nextLine(); pw.println(s); } pw.println("---END: NATIVE CRASH LOG----"); sc.close(); } catch (IOException e) { Log.e(TAG, "Exception in copyNativeCrashLogsIfAny " + e); } } public void storeNativeCrashLogs() { FileOutputStream fos = null; try { File file = new File(NATIVE_LOG_FILE_PATH, NATIVE_LOG_FILE_NAME); if (file.length() >= NATIVE_CRASH_FILE_SIZE) { file.createNewFile(); } fos = new FileOutputStream(file, true); mDeviceHost.dump(new PrintWriter(new StringWriter()), fos.getFD()); fos.flush(); } catch (IOException e) { Log.e(TAG, "storeNativeCrashLogs: e=" + e); } finally { if (fos != null) { try { fos.close(); } catch (IOException e) { Log.e(TAG, "storeNativeCrashLogs: file close " + e); } } } } private void dumpTagAppPreference(PrintWriter pw) { pw.println("mIsTagAppPrefSupported =" + mIsTagAppPrefSupported); if (!mIsTagAppPrefSupported) return; pw.println("TagAppPreference:"); for (Integer userId : getEnabledUserIds()) { HashMap map; synchronized (NfcService.this) { map = mTagAppPrefList.getOrDefault(userId, new HashMap<>()); } if (map.size() > 0) pw.println("userId=" + userId); for (Map.Entry entry : map.entrySet()) { pw.println("pkg: " + entry.getKey() + " : " + entry.getValue()); } } } 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 pid=" + Binder.getCallingPid() + ", uid=" + Binder.getCallingUid() + " without permission " + android.Manifest.permission.DUMP); return; } for (String arg : args) { if ("--proto".equals(arg)) { FileOutputStream fos = null; try { fos = new FileOutputStream(fd); ProtoOutputStream proto = new ProtoOutputStream(fos); synchronized (this) { dumpDebug(proto); } proto.flush(); } catch (Exception e) { Log.e(TAG, "dump: exception=" + e); } finally { if (fos != null) { try { fos.close(); } catch (IOException e) { Log.e(TAG, "dump: Exception in storeNativeCrashLogs " + e); } } } return; } } synchronized (this) { pw.println("mState=" + stateToString(mState)); pw.println("mAlwaysOnState=" + stateToString(mAlwaysOnState)); pw.println("mScreenState=" + ScreenStateHelper.screenStateToString(mScreenState)); pw.println("mIsSecureNfcEnabled=" + mIsSecureNfcEnabled); pw.println("mIsReaderOptionEnabled=" + mIsReaderOptionEnabled); pw.println("mIsAlwaysOnSupported=" + mIsAlwaysOnSupported); if (mIsWlcCapable) { pw.println("WlcEnabled=" + mIsWlcEnabled); } pw.println("SnoopLogMode=" + NFC_SNOOP_LOG_MODE); pw.println("VendorDebugEnabled=" + NFC_VENDOR_DEBUG_ENABLED); pw.println("mIsPowerSavingModeEnabled=" + mIsPowerSavingModeEnabled); pw.println("mIsObserveModeSupported=" + mNfcAdapter.isObserveModeSupported()); pw.println("mIsObserveModeEnabled=" + mNfcAdapter.isObserveModeEnabled()); pw.println("listenTech=0x" + Integer.toHexString(getNfcListenTech())); pw.println("pollTech=0x" + Integer.toHexString(getNfcPollTech())); pw.println(mCurrentDiscoveryParameters); if (mIsHceCapable) { mCardEmulationManager.dump(fd, pw, args); } mNfcDispatcher.dump(fd, pw, args); if (mState == NfcAdapter.STATE_ON) { mRoutingTableParser.dump(mDeviceHost, pw); } dumpTagAppPreference(pw); mNfcInjector.getNfcEventLog().dump(fd, pw, args); copyNativeCrashLogsIfAny(pw); pw.flush(); mDeviceHost.dump(pw,fd); } } /** * Dump debugging information as a NfcServiceDumpProto * * Note: * See proto definition in frameworks/base/core/proto/android/nfc/nfc_service.proto * When writing a nested message, must call {@link ProtoOutputStream#start(long)} before and * {@link ProtoOutputStream#end(long)} after. * Never reuse a proto field number. When removing a field, mark it as reserved. */ private void dumpDebug(ProtoOutputStream proto) { proto.write(NfcServiceDumpProto.STATE, stateToProtoEnum(mState)); proto.write(NfcServiceDumpProto.IN_PROVISION_MODE, mInProvisionMode); proto.write(NfcServiceDumpProto.SCREEN_STATE, ScreenStateHelper.screenStateToProtoEnum(mScreenState)); proto.write(NfcServiceDumpProto.SECURE_NFC_ENABLED, mIsSecureNfcEnabled); proto.write(NfcServiceDumpProto.POLLING_PAUSED, mPollingPaused); proto.write(NfcServiceDumpProto.HCE_CAPABLE, mIsHceCapable); proto.write(NfcServiceDumpProto.HCE_F_CAPABLE, mIsHceFCapable); proto.write(NfcServiceDumpProto.SECURE_NFC_CAPABLE, mIsSecureNfcCapable); proto.write(NfcServiceDumpProto.VR_MODE_ENABLED, false); long token = proto.start(NfcServiceDumpProto.DISCOVERY_PARAMS); mCurrentDiscoveryParameters.dumpDebug(proto); proto.end(token); if (mIsHceCapable) { token = proto.start(NfcServiceDumpProto.CARD_EMULATION_MANAGER); mCardEmulationManager.dumpDebug(proto); proto.end(token); } token = proto.start(NfcServiceDumpProto.NFC_DISPATCHER); mNfcDispatcher.dumpDebug(proto); proto.end(token); // Dump native crash logs if any File file = new File(NATIVE_LOG_FILE_PATH, NATIVE_LOG_FILE_NAME); if (!file.exists()) { return; } try { String logs = Files.lines(file.toPath()).collect(Collectors.joining("\n")); proto.write(NfcServiceDumpProto.NATIVE_CRASH_LOGS, logs); } catch (IOException e) { Log.e(TAG, "dumpDebug: IOException=" + e); } } private int runTaskOnSingleThreadExecutor(FutureTask task, int timeoutMs) throws InterruptedException, TimeoutException, ExecutionException { ExecutorService executor = Executors.newSingleThreadExecutor(); executor.submit(task); try { return task.get(timeoutMs, TimeUnit.MILLISECONDS); } catch (TimeoutException e) { executor.shutdownNow(); throw e; } } @VisibleForTesting public Handler getHandler() { return mHandler; } public void notifyOemLogEvent(OemLogItems item) { if (mNfcOemExtensionCallback != null) { try { mNfcOemExtensionCallback.onLogEventNotified(item); } catch (RemoteException e) { Log.e(TAG, "notifyOemLogEvent: failed e = " + e.toString()); } } } public boolean isFirmwareExitFramesSupported() { return mDeviceHost.isFirmwareExitFramesSupported(); } public int getNumberOfFirmwareExitFramesSupported() { return mDeviceHost.getNumberOfFirmwareExitFramesSupported(); } public boolean setFirmwareExitFrameTable(List exitFrames, int timeoutMs) { // Check if NFC is enabled if (!isNfcEnabled()) { Log.w(TAG, "setFirmwareExitFrameTable: NFC is not enabled"); return false; } if (mCardEmulationManager != null && mCardEmulationManager.isHostCardEmulationActivated()) { Log.w(TAG,"setFirmwareExitFrameTable: " + "Cannot set exit rame table in during a transaction."); return false; } byte[] timeoutBytes = new byte[2]; if (timeoutMs > 0xFFFF) { Log.w(TAG, "setFirmwareExitFrameTable: timeout is larger than 16 bits, " + "timeout will be truncated"); timeoutBytes = new byte[] {(byte) 0xFF, (byte) 0xFF}; } else { // Convert to little endian, two byte array timeoutBytes[0] = (byte) timeoutMs; timeoutBytes[1] = (byte) (timeoutMs >> 8); } boolean result = mDeviceHost.setFirmwareExitFrameTable(exitFrames.toArray(ExitFrame[]::new), timeoutBytes); if (result && mStatsdUtils != null) { mStatsdUtils.logExitFrameTableChanged(exitFrames.size(), timeoutMs); } return result; } }