/* * Copyright (C) 2024 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 android.nfc; import android.annotation.CallbackExecutor; import android.annotation.FlaggedApi; import android.annotation.NonNull; import android.annotation.RequiresPermission; import android.annotation.SystemApi; import android.content.Context; import android.os.Binder; import android.os.RemoteException; import android.util.Log; import java.util.concurrent.Executor; /** * Used for OEM extension APIs. * This class holds all the APIs and callbacks defined for OEMs/vendors to extend the NFC stack * for their proprietary features. * * @hide */ @FlaggedApi(Flags.FLAG_NFC_OEM_EXTENSION) @SystemApi public final class NfcOemExtension { private static final String TAG = "NfcOemExtension"; private static final int OEM_EXTENSION_RESPONSE_THRESHOLD_MS = 2000; private final NfcAdapter mAdapter; private final NfcOemExtensionCallback mOemNfcExtensionCallback; private final Context mContext; private Executor mExecutor = null; private Callback mCallback = null; private final Object mLock = new Object(); /** * Interface for Oem extensions for NFC. */ public interface Callback { /** * Notify Oem to tag is connected or not * ex - if tag is connected notify cover and Nfctest app if app is in testing mode * * @param connected status of the tag true if tag is connected otherwise false * @param tag Tag details */ void onTagConnected(boolean connected, @NonNull Tag tag); } /** * Constructor to be used only by {@link NfcAdapter}. * @hide */ public NfcOemExtension(@NonNull Context context, @NonNull NfcAdapter adapter) { mContext = context; mAdapter = adapter; mOemNfcExtensionCallback = new NfcOemExtensionCallback(); } /** * Register an {@link Callback} to listen for UWB oem extension callbacks *

The provided callback will be invoked by the given {@link Executor}. * * @param executor an {@link Executor} to execute given callback * @param callback oem implementation of {@link Callback} */ @FlaggedApi(Flags.FLAG_NFC_OEM_EXTENSION) @RequiresPermission(android.Manifest.permission.WRITE_SECURE_SETTINGS) public void registerCallback(@NonNull @CallbackExecutor Executor executor, @NonNull Callback callback) { synchronized (mLock) { if (mCallback != null) { Log.e(TAG, "Callback already registered. Unregister existing callback before" + "registering"); throw new IllegalArgumentException(); } try { NfcAdapter.sService.registerOemExtensionCallback(mOemNfcExtensionCallback); mCallback = callback; mExecutor = executor; } catch (RemoteException e) { mAdapter.attemptDeadServiceRecovery(e); } } } /** * Unregister the specified {@link Callback} * *

The same {@link Callback} object used when calling * {@link #registerCallback(Executor, Callback)} must be used. * *

Callbacks are automatically unregistered when an application process goes away * * @param callback oem implementation of {@link Callback} */ @FlaggedApi(Flags.FLAG_NFC_OEM_EXTENSION) @RequiresPermission(android.Manifest.permission.WRITE_SECURE_SETTINGS) public void unregisterCallback(@NonNull Callback callback) { synchronized (mLock) { if (mCallback == null || mCallback != callback) { Log.e(TAG, "Callback not registered"); throw new IllegalArgumentException(); } try { NfcAdapter.sService.unregisterOemExtensionCallback(mOemNfcExtensionCallback); mCallback = null; mExecutor = null; } catch (RemoteException e) { mAdapter.attemptDeadServiceRecovery(e); } } } /** * Clear NfcService preference, interface method to clear NFC preference values on OEM specific * events. For ex: on soft reset, Nfc default values needs to be overridden by OEM defaults. */ @FlaggedApi(Flags.FLAG_NFC_OEM_EXTENSION) @RequiresPermission(android.Manifest.permission.WRITE_SECURE_SETTINGS) public void clearPreference() { try { NfcAdapter.sService.clearPreference(); } catch (RemoteException e) { mAdapter.attemptDeadServiceRecovery(e); } } private final class NfcOemExtensionCallback extends INfcOemExtensionCallback.Stub { @Override public void onTagConnected(boolean connected, Tag tag) throws RemoteException { synchronized (mLock) { if (mCallback == null || mExecutor == null) { return; } final long identity = Binder.clearCallingIdentity(); try { mExecutor.execute(() -> mCallback.onTagConnected(connected, tag)); } finally { Binder.restoreCallingIdentity(identity); } } } } }