/* * 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 com.android.settings.network.telephony; import android.content.Context; import android.os.Build; import android.safetycenter.SafetyCenterManager; import android.telephony.SubscriptionInfo; import android.telephony.TelephonyManager; import android.util.Log; import androidx.annotation.NonNull; import androidx.annotation.VisibleForTesting; import com.android.settings.network.SubscriptionUtil; import java.util.List; /** * {@link TelephonyTogglePreferenceController} for accessing Cellular Security settings through * Safety Center. */ public class CellularSecurityNotificationsPreferenceController extends TelephonyTogglePreferenceController { private static final String LOG_TAG = "CellularSecurityNotificationsPreferenceController"; private TelephonyManager mTelephonyManager; @VisibleForTesting protected SafetyCenterManager mSafetyCenterManager; /** * Class constructor of "Cellular Security" preference. * * @param context of settings * @param prefKey assigned within UI entry of XML file */ public CellularSecurityNotificationsPreferenceController( @NonNull Context context, @NonNull String prefKey) { super(context, prefKey); mTelephonyManager = mContext.getSystemService(TelephonyManager.class); mSafetyCenterManager = mContext.getSystemService(SafetyCenterManager.class); } /** * Initialization based on a given subscription id. * * @param subId is the subscription id * @return this instance after initialization */ @NonNull public CellularSecurityNotificationsPreferenceController init(@NonNull int subId) { mTelephonyManager = mContext.getSystemService(TelephonyManager.class) .createForSubscriptionId(subId); return this; } @Override public int getAvailabilityStatus(int subId) { if (!isSafetyCenterSupported()) { return UNSUPPORTED_ON_DEVICE; } // Check there are valid SIM cards which can be displayed to the user, otherwise this // setting should not be shown. List availableSubs = SubscriptionUtil.getAvailableSubscriptions(mContext); if (availableSubs.isEmpty()) { return CONDITIONALLY_UNAVAILABLE; } // Checking for hardware support, i.e. IRadio AIDL version must be >= 2.2 try { areNotificationsEnabled(); } catch (UnsupportedOperationException e) { Log.i(LOG_TAG, "Cellular security notifications are unsupported: " + e.getMessage()); return UNSUPPORTED_ON_DEVICE; } return AVAILABLE; } /** * Return {@code true} if cellular security notifications are on * *

NOTE: This method returns the active state of the preference controller and is not * the parameter passed into {@link #setChecked(boolean)}, which is instead the requested future * state. */ @Override public boolean isChecked() { try { // Note: the default behavior for this toggle is disabled (as the underlying // TelephonyManager APIs are disabled by default) return areNotificationsEnabled(); } catch (Exception e) { Log.e(LOG_TAG, "Failed isNullCipherNotificationsEnabled and " + "isCellularIdentifierDisclosureNotificationsEnabled." + "Defaulting toggle to checked = true. Exception: " + e.getMessage()); return false; } } /** * Called when a user preference changes on the toggle. We pass this info on to the Telephony * Framework so that the modem can be updated with the user's preference. * *

See {@link com.android.settings.core.TogglePreferenceController#setChecked(boolean)} for * details. * * @param isChecked The toggle value that we're being requested to enforce. A value of {@code * true} denotes that both (1) null cipher/integrity notifications, and * (2) IMSI disclosure notifications will be enabled by the modem after this * function completes, if they are not already. */ @Override public boolean setChecked(boolean isChecked) { if (isChecked) { Log.i(LOG_TAG, "Enabling cellular security notifications."); } else { Log.i(LOG_TAG, "Disabling cellular security notifications."); } try { setNotifications(isChecked); } catch (Exception e) { Log.e(LOG_TAG, "Failed setCellularIdentifierDisclosureNotificationEnabled or " + " setNullCipherNotificationsEnabled. Setting not updated. Exception: " + e.getMessage()); // Reset to defaults so we don't end up in an inconsistent state setNotifications(!isChecked); return false; } return true; } private void setNotifications(boolean isChecked) { mTelephonyManager.setEnableCellularIdentifierDisclosureNotifications(isChecked); mTelephonyManager.setNullCipherNotificationsEnabled(isChecked); } private boolean areNotificationsEnabled() { if (mTelephonyManager == null) { Log.w(LOG_TAG, "Telephony manager not yet initialized"); mTelephonyManager = mContext.getSystemService(TelephonyManager.class); } return mTelephonyManager.isNullCipherNotificationsEnabled() && mTelephonyManager.isCellularIdentifierDisclosureNotificationsEnabled(); } protected boolean isSafetyCenterSupported() { if (Build.VERSION.SDK_INT < Build.VERSION_CODES.TIRAMISU) { return false; } mSafetyCenterManager = mContext.getSystemService( SafetyCenterManager.class); if (mSafetyCenterManager == null) { return false; } return mSafetyCenterManager.isSafetyCenterEnabled(); } }