[SIM Dialog Migration] Migrate SIM enable dialog from LPA to Settings
Migrates SIM enabling and DSDS dialog from LPA to Settings. Design: https://docs.google.com/document/d/1wb5_hoBkZVbkXGNWHbx4Jf61swjfxsJzkytiTzJosYo/edit?usp=sharing Bug: 160819390 Test: Manually tested eSIM profile enabling. Change-Id: I9ce690a1594adfc90b62840819237acd54418469
This commit is contained in:
190
src/com/android/settings/network/EnableMultiSimSidecar.java
Normal file
190
src/com/android/settings/network/EnableMultiSimSidecar.java
Normal file
@@ -0,0 +1,190 @@
|
||||
/*
|
||||
* Copyright (C) 2020 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;
|
||||
|
||||
import android.app.FragmentManager;
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.IntentFilter;
|
||||
import android.telephony.CarrierConfigManager;
|
||||
import android.telephony.TelephonyManager;
|
||||
import android.telephony.UiccSlotInfo;
|
||||
import android.util.ArraySet;
|
||||
import android.util.Log;
|
||||
|
||||
import com.android.settings.AsyncTaskSidecar;
|
||||
import com.android.settings.SidecarFragment;
|
||||
|
||||
import java.util.Collections;
|
||||
import java.util.Set;
|
||||
import java.util.concurrent.CountDownLatch;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
/**
|
||||
* {@code EnableMultiSimSidecar} enables multi SIM on the device. It should only be called for
|
||||
* Android R+. After {@code run} is called, it sets the configuration on modem side to enable
|
||||
* multiple SIMs. Once the configuration is set successfully, it will listen to UICC card changes
|
||||
* until {@code TelMan.EXTRA_ACTIVE_SIM_SUPPORTED_COUNT} matches {@code mNumOfActiveSim} or timeout.
|
||||
*/
|
||||
public class EnableMultiSimSidecar extends AsyncTaskSidecar<Void, Boolean> {
|
||||
|
||||
// Tags
|
||||
private static final String TAG = "EnableMultiSimSidecar";
|
||||
|
||||
// TODO(b/171846124): Pass timeout value from LPA to Settings
|
||||
private static final long ENABLE_MULTI_SIM_TIMEOUT_MILLS = 40 * 1000L;
|
||||
|
||||
public static EnableMultiSimSidecar get(FragmentManager fm) {
|
||||
return SidecarFragment.get(fm, TAG, EnableMultiSimSidecar.class, null /* args */);
|
||||
}
|
||||
|
||||
final CountDownLatch mSimCardStateChangedLatch = new CountDownLatch(1);
|
||||
private TelephonyManager mTelephonyManager;
|
||||
private int mNumOfActiveSim = 0;
|
||||
|
||||
private final BroadcastReceiver mCarrierConfigChangeReceiver =
|
||||
new BroadcastReceiver() {
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
int readySimsCount = getReadySimsCount();
|
||||
int activeSlotsCount = getActiveSlotsCount();
|
||||
// If the number of ready SIM count and active slots equal to the number of SIMs
|
||||
// need to be activated, the device is successfully switched to multiple active
|
||||
// SIM mode.
|
||||
if (readySimsCount == mNumOfActiveSim && activeSlotsCount == mNumOfActiveSim) {
|
||||
Log.i(
|
||||
TAG,
|
||||
String.format("%d slots are active and ready.", mNumOfActiveSim));
|
||||
mSimCardStateChangedLatch.countDown();
|
||||
return;
|
||||
}
|
||||
Log.i(
|
||||
TAG,
|
||||
String.format(
|
||||
"%d slots are active and %d SIMs are ready. Keep waiting until"
|
||||
+ " timeout.",
|
||||
activeSlotsCount, readySimsCount));
|
||||
}
|
||||
};
|
||||
|
||||
@Override
|
||||
protected Boolean doInBackground(Void aVoid) {
|
||||
return updateMultiSimConfig();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Boolean isDsdsEnabled) {
|
||||
if (isDsdsEnabled) {
|
||||
setState(State.SUCCESS, Substate.UNUSED);
|
||||
} else {
|
||||
setState(State.ERROR, Substate.UNUSED);
|
||||
}
|
||||
}
|
||||
|
||||
public void run(int numberOfSimToActivate) {
|
||||
mTelephonyManager = getContext().getSystemService(TelephonyManager.class);
|
||||
mNumOfActiveSim = numberOfSimToActivate;
|
||||
|
||||
if (mNumOfActiveSim > mTelephonyManager.getSupportedModemCount()) {
|
||||
Log.e(TAG, "Requested number of active SIM is greater than supported modem count.");
|
||||
setState(State.ERROR, Substate.UNUSED);
|
||||
return;
|
||||
}
|
||||
if (mTelephonyManager.doesSwitchMultiSimConfigTriggerReboot()) {
|
||||
Log.e(TAG, "The device does not support reboot free DSDS.");
|
||||
setState(State.ERROR, Substate.UNUSED);
|
||||
return;
|
||||
}
|
||||
super.run(null /* param */);
|
||||
}
|
||||
|
||||
// This method registers a ACTION_SIM_CARD_STATE_CHANGED broadcast receiver and wait for slot
|
||||
// changes. If multi SIMs have been successfully enabled, it returns true. Otherwise, returns
|
||||
// false.
|
||||
private boolean updateMultiSimConfig() {
|
||||
try {
|
||||
getContext()
|
||||
.registerReceiver(
|
||||
mCarrierConfigChangeReceiver,
|
||||
new IntentFilter(CarrierConfigManager.ACTION_CARRIER_CONFIG_CHANGED));
|
||||
mTelephonyManager.switchMultiSimConfig(mNumOfActiveSim);
|
||||
if (mSimCardStateChangedLatch.await(
|
||||
ENABLE_MULTI_SIM_TIMEOUT_MILLS, TimeUnit.MILLISECONDS)) {
|
||||
Log.i(TAG, "Multi SIM were successfully enabled.");
|
||||
return true;
|
||||
} else {
|
||||
Log.e(TAG, "Timeout for waiting SIM status.");
|
||||
return false;
|
||||
}
|
||||
} catch (InterruptedException e) {
|
||||
Log.e(TAG, "Failed to enable multiple SIM due to InterruptedException", e);
|
||||
return false;
|
||||
} finally {
|
||||
getContext().unregisterReceiver(mCarrierConfigChangeReceiver);
|
||||
}
|
||||
}
|
||||
|
||||
// Returns how many SIMs have SIM ready state, not ready state, or removable slot with absent
|
||||
// SIM state.
|
||||
private int getReadySimsCount() {
|
||||
int readyCardsCount = 0;
|
||||
int activeSlotCount = mTelephonyManager.getActiveModemCount();
|
||||
Set<Integer> activeRemovableLogicalSlots = getActiveRemovableLogicalSlotIds();
|
||||
for (int logicalSlotId = 0; logicalSlotId < activeSlotCount; logicalSlotId++) {
|
||||
int simState = mTelephonyManager.getSimState(logicalSlotId);
|
||||
if (simState == TelephonyManager.SIM_STATE_READY
|
||||
|| simState == TelephonyManager.SIM_STATE_NOT_READY
|
||||
|| simState == TelephonyManager.SIM_STATE_LOADED
|
||||
|| (simState == TelephonyManager.SIM_STATE_ABSENT
|
||||
&& activeRemovableLogicalSlots.contains(logicalSlotId))) {
|
||||
readyCardsCount++;
|
||||
}
|
||||
}
|
||||
return readyCardsCount;
|
||||
}
|
||||
|
||||
// Get active slots count from {@code TelephonyManager#getUiccSlotsInfo}.
|
||||
private int getActiveSlotsCount() {
|
||||
UiccSlotInfo[] slotsInfo = mTelephonyManager.getUiccSlotsInfo();
|
||||
if (slotsInfo == null) {
|
||||
return 0;
|
||||
}
|
||||
int activeSlots = 0;
|
||||
for (UiccSlotInfo slotInfo : slotsInfo) {
|
||||
if (slotInfo != null && slotInfo.getIsActive()) {
|
||||
activeSlots++;
|
||||
}
|
||||
}
|
||||
return activeSlots;
|
||||
}
|
||||
|
||||
/** Returns a list of active removable logical slot ids. */
|
||||
public Set<Integer> getActiveRemovableLogicalSlotIds() {
|
||||
UiccSlotInfo[] infos = mTelephonyManager.getUiccSlotsInfo();
|
||||
if (infos == null) {
|
||||
return Collections.emptySet();
|
||||
}
|
||||
Set<Integer> activeRemovableLogicalSlotIds = new ArraySet<>();
|
||||
for (UiccSlotInfo info : infos) {
|
||||
if (info != null && info.getIsActive() && info.isRemovable()) {
|
||||
activeRemovableLogicalSlotIds.add(info.getLogicalSlotIdx());
|
||||
}
|
||||
}
|
||||
return activeRemovableLogicalSlotIds;
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user