switch SIM refactor to support MEP
1. Using new telephony API and doing the code refactor 2. To support multi esim profiles case Bug: 199902896 Test: local build pass. Change-Id: I8580022793e5c3fc14159f14b406f864353477f8
This commit is contained in:
@@ -19,8 +19,10 @@ package com.android.settings.network;
|
||||
import android.annotation.IntDef;
|
||||
import android.content.Context;
|
||||
import android.provider.Settings;
|
||||
import android.telephony.SubscriptionInfo;
|
||||
import android.telephony.TelephonyManager;
|
||||
import android.telephony.UiccSlotInfo;
|
||||
import android.telephony.UiccSlotMapping;
|
||||
import android.util.Log;
|
||||
|
||||
import com.android.settingslib.utils.ThreadUtils;
|
||||
@@ -29,17 +31,21 @@ import com.google.common.collect.ImmutableList;
|
||||
|
||||
import java.lang.annotation.Retention;
|
||||
import java.lang.annotation.RetentionPolicy;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collection;
|
||||
import java.util.concurrent.CountDownLatch;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
// ToDo: to do the refactor for renaming
|
||||
public class UiccSlotUtil {
|
||||
|
||||
private static final String TAG = "UiccSlotUtil";
|
||||
|
||||
private static final long DEFAULT_WAIT_AFTER_SWITCH_TIMEOUT_MILLIS = 25 * 1000L;
|
||||
;
|
||||
|
||||
public static final int INVALID_PHYSICAL_SLOT_ID = -1;
|
||||
public static final int INVALID_PORT_ID = -1;
|
||||
|
||||
/**
|
||||
* Mode for switching to eSIM slot which decides whether there is cleanup process, e.g.
|
||||
@@ -47,9 +53,9 @@ public class UiccSlotUtil {
|
||||
*/
|
||||
@Retention(RetentionPolicy.SOURCE)
|
||||
@IntDef({
|
||||
SwitchingEsimMode.NO_CLEANUP,
|
||||
SwitchingEsimMode.ASYNC_CLEANUP,
|
||||
SwitchingEsimMode.SYNC_CLEANUP
|
||||
SwitchingEsimMode.NO_CLEANUP,
|
||||
SwitchingEsimMode.ASYNC_CLEANUP,
|
||||
SwitchingEsimMode.SYNC_CLEANUP
|
||||
})
|
||||
public @interface SwitchingEsimMode {
|
||||
/** No cleanup process after switching to eSIM slot */
|
||||
@@ -76,44 +82,108 @@ public class UiccSlotUtil {
|
||||
* Switches to the removable slot. It waits for SIM_STATE_LOADED after switch. If slotId is
|
||||
* INVALID_PHYSICAL_SLOT_ID, the method will use the first detected inactive removable slot.
|
||||
*
|
||||
* @param slotId the physical removable slot id.
|
||||
* @param slotId the physical removable slot id.
|
||||
* @param context the application context.
|
||||
* @throws UiccSlotsException if there is an error.
|
||||
*/
|
||||
//ToDo: delete this api and refactor the related code.
|
||||
public static synchronized void switchToRemovableSlot(int slotId, Context context)
|
||||
throws UiccSlotsException {
|
||||
switchToRemovableSlot(context, slotId, null);
|
||||
}
|
||||
|
||||
/**
|
||||
* Switches to the removable slot. It waits for SIM_STATE_LOADED after switch. If slotId is
|
||||
* INVALID_PHYSICAL_SLOT_ID, the method will use the first detected inactive removable slot.
|
||||
*
|
||||
* @param slotId the physical removable slot id.
|
||||
* @param context the application context.
|
||||
* @param removedSubInfo In the DSDS+MEP mode, if the all of slots have sims, it should
|
||||
* remove the one of active sim.
|
||||
* If the removedSubInfo is null, then use the default value.
|
||||
* The default value is the esim slot and portId 0.
|
||||
* @throws UiccSlotsException if there is an error.
|
||||
*/
|
||||
public static synchronized void switchToRemovableSlot(Context context, int slotId,
|
||||
SubscriptionInfo removedSubInfo) throws UiccSlotsException {
|
||||
if (ThreadUtils.isMainThread()) {
|
||||
throw new IllegalThreadStateException(
|
||||
"Do not call switchToRemovableSlot on the main thread.");
|
||||
}
|
||||
TelephonyManager telMgr = context.getSystemService(TelephonyManager.class);
|
||||
if (telMgr.isMultiSimEnabled()) {
|
||||
// If this device supports multiple active slots, don't mess with TelephonyManager.
|
||||
Log.i(TAG, "Multiple active slots supported. Not calling switchSlots.");
|
||||
return;
|
||||
}
|
||||
UiccSlotInfo[] slots = telMgr.getUiccSlotsInfo();
|
||||
if (slotId == INVALID_PHYSICAL_SLOT_ID) {
|
||||
for (int i = 0; i < slots.length; i++) {
|
||||
if (slots[i].isRemovable()
|
||||
&& !slots[i].getPorts().stream().findFirst().get().isActive()
|
||||
&& slots[i].getCardStateInfo() != UiccSlotInfo.CARD_STATE_INFO_ERROR
|
||||
&& slots[i].getCardStateInfo() != UiccSlotInfo.CARD_STATE_INFO_RESTRICTED) {
|
||||
performSwitchToRemovableSlot(i, context);
|
||||
return;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (slotId >= slots.length || !slots[slotId].isRemovable()) {
|
||||
throw new UiccSlotsException("The given slotId is not a removable slot: " + slotId);
|
||||
}
|
||||
if (!slots[slotId].getPorts().stream().findFirst().get().isActive()) {
|
||||
performSwitchToRemovableSlot(slotId, context);
|
||||
}
|
||||
}
|
||||
int inactiveRemovableSlot = getInactiveRemovableSlot(telMgr.getUiccSlotsInfo(), slotId);
|
||||
performSwitchToSlot(telMgr,
|
||||
prepareUiccSlotMappingsForRemovableSlot(telMgr.getSimSlotMapping(),
|
||||
inactiveRemovableSlot, removedSubInfo, telMgr.isMultiSimEnabled()),
|
||||
context);
|
||||
}
|
||||
|
||||
private static void performSwitchToRemovableSlot(int slotId, Context context)
|
||||
/**
|
||||
* Switches to the Euicc slot. It waits for SIM_STATE_LOADED after switch.
|
||||
*
|
||||
* @param context the application context.
|
||||
* @param slotId the Euicc slot id.
|
||||
* @param port the Euicc slot port id.
|
||||
* @param removedSubInfo In the DSDS+MEP mode, if the all of slots have sims, it should
|
||||
* remove the one of active sim.
|
||||
* If the removedSubInfo is null, then it uses the default value.
|
||||
* The default value is the esim slot and portId 0.
|
||||
* @throws UiccSlotsException if there is an error.
|
||||
*/
|
||||
public static synchronized void switchToEuiccSlot(Context context, int slotId, int port,
|
||||
SubscriptionInfo removedSubInfo) throws UiccSlotsException {
|
||||
if (ThreadUtils.isMainThread()) {
|
||||
throw new IllegalThreadStateException(
|
||||
"Do not call switchToRemovableSlot on the main thread.");
|
||||
}
|
||||
TelephonyManager telMgr = context.getSystemService(TelephonyManager.class);
|
||||
Collection<UiccSlotMapping> uiccSlotMappings = telMgr.getSimSlotMapping();
|
||||
Log.i(TAG, "The SimSlotMapping: " + uiccSlotMappings);
|
||||
|
||||
if (isTargetSlotActive(uiccSlotMappings, slotId, port)) {
|
||||
Log.i(TAG, "The slot is active, then the sim can enable directly.");
|
||||
return;
|
||||
}
|
||||
|
||||
Collection<UiccSlotMapping> newUiccSlotMappings = new ArrayList<>();
|
||||
if (!telMgr.isMultiSimEnabled()) {
|
||||
// In the 'SS mode', the port is 0.
|
||||
newUiccSlotMappings.add(new UiccSlotMapping(port, slotId, 0));
|
||||
} else {
|
||||
// DSDS+MEP
|
||||
// The target slot+port is not active, but the all of logical slots are full. It
|
||||
// needs to replace one of logical slots.
|
||||
int removedSlot =
|
||||
(removedSubInfo != null) ? removedSubInfo.getSimSlotIndex() : slotId;
|
||||
int removedPort = (removedSubInfo != null) ? removedSubInfo.getPortIndex() : 0;
|
||||
Log.i(TAG,
|
||||
String.format("Start to set SimSlotMapping from slot%d-port%d to slot%d-port%d",
|
||||
slotId, port, removedSlot, removedPort));
|
||||
newUiccSlotMappings =
|
||||
uiccSlotMappings.stream().map(uiccSlotMapping -> {
|
||||
if (uiccSlotMapping.getPhysicalSlotIndex() == removedSlot
|
||||
&& uiccSlotMapping.getPortIndex() == removedPort) {
|
||||
return new UiccSlotMapping(port, slotId,
|
||||
uiccSlotMapping.getLogicalSlotIndex());
|
||||
}
|
||||
return uiccSlotMapping;
|
||||
}).collect(Collectors.toList());
|
||||
}
|
||||
|
||||
Log.i(TAG, "The SimSlotMapping: " + newUiccSlotMappings);
|
||||
performSwitchToSlot(telMgr, newUiccSlotMappings, context);
|
||||
}
|
||||
|
||||
private static boolean isTargetSlotActive(Collection<UiccSlotMapping> uiccSlotMappings,
|
||||
int slotId, int port) {
|
||||
return uiccSlotMappings.stream()
|
||||
.anyMatch(
|
||||
uiccSlotMapping -> uiccSlotMapping.getPhysicalSlotIndex() == slotId
|
||||
&& uiccSlotMapping.getPortIndex() == port);
|
||||
}
|
||||
|
||||
private static void performSwitchToSlot(TelephonyManager telMgr,
|
||||
Collection<UiccSlotMapping> uiccSlotMappings, Context context)
|
||||
throws UiccSlotsException {
|
||||
CarrierConfigChangedReceiver receiver = null;
|
||||
long waitingTimeMillis =
|
||||
@@ -125,7 +195,7 @@ public class UiccSlotUtil {
|
||||
CountDownLatch latch = new CountDownLatch(1);
|
||||
receiver = new CarrierConfigChangedReceiver(latch);
|
||||
receiver.registerOn(context);
|
||||
switchSlots(context, slotId);
|
||||
telMgr.setSimSlotMapping(uiccSlotMappings);
|
||||
latch.await(waitingTimeMillis, TimeUnit.MILLISECONDS);
|
||||
} catch (InterruptedException e) {
|
||||
Thread.currentThread().interrupt();
|
||||
@@ -138,22 +208,80 @@ public class UiccSlotUtil {
|
||||
}
|
||||
|
||||
/**
|
||||
* Changes the logical slot to physical slot mapping. OEM should override this to provide
|
||||
* device-specific implementation if the device supports switching slots.
|
||||
*
|
||||
* @param context the application context.
|
||||
* @param physicalSlots List of physical slot ids in the order of logical slots.
|
||||
* @param slots The UiccSlotInfo list.
|
||||
* @param slotId The physical removable slot id.
|
||||
* @return The inactive physical removable slot id. If the physical removable slot id is
|
||||
* active, then return -1.
|
||||
* @throws UiccSlotsException if there is an error.
|
||||
*/
|
||||
private static void switchSlots(Context context, int... physicalSlots)
|
||||
private static int getInactiveRemovableSlot(UiccSlotInfo[] slots, int slotId)
|
||||
throws UiccSlotsException {
|
||||
TelephonyManager telMgr = context.getSystemService(TelephonyManager.class);
|
||||
if (telMgr.isMultiSimEnabled()) {
|
||||
// If this device supports multiple active slots, don't mess with TelephonyManager.
|
||||
Log.i(TAG, "Multiple active slots supported. Not calling switchSlots.");
|
||||
return;
|
||||
if (slots == null) {
|
||||
throw new UiccSlotsException("UiccSlotInfo is null");
|
||||
}
|
||||
if (!telMgr.switchSlots(physicalSlots)) {
|
||||
throw new UiccSlotsException("Failed to switch slots");
|
||||
if (slotId == INVALID_PHYSICAL_SLOT_ID) {
|
||||
for (int i = 0; i < slots.length; i++) {
|
||||
if (slots[i].isRemovable()
|
||||
&& !slots[i].getPorts().stream().findFirst().get().isActive()
|
||||
&& slots[i].getCardStateInfo() != UiccSlotInfo.CARD_STATE_INFO_ERROR
|
||||
&& slots[i].getCardStateInfo() != UiccSlotInfo.CARD_STATE_INFO_RESTRICTED) {
|
||||
return i;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (slotId >= slots.length || !slots[slotId].isRemovable()) {
|
||||
throw new UiccSlotsException("The given slotId is not a removable slot: " + slotId);
|
||||
}
|
||||
if (!slots[slotId].getPorts().stream().findFirst().get().isActive()) {
|
||||
return slotId;
|
||||
}
|
||||
}
|
||||
return INVALID_PHYSICAL_SLOT_ID;
|
||||
}
|
||||
|
||||
private static Collection<UiccSlotMapping> prepareUiccSlotMappingsForRemovableSlot(
|
||||
Collection<UiccSlotMapping> uiccSlotMappings, int slotId,
|
||||
SubscriptionInfo removedSubInfo, boolean isMultiSimEnabled) {
|
||||
if (slotId == INVALID_PHYSICAL_SLOT_ID
|
||||
|| uiccSlotMappings.stream().anyMatch(uiccSlotMapping ->
|
||||
uiccSlotMapping.getPhysicalSlotIndex() == slotId
|
||||
&& uiccSlotMapping.getPortIndex() == 0)) {
|
||||
// The slot is invalid slot id, then to skip this.
|
||||
// The slot is active, then the sim can enable directly.
|
||||
return uiccSlotMappings;
|
||||
}
|
||||
|
||||
Collection<UiccSlotMapping> newUiccSlotMappings = new ArrayList<>();
|
||||
if (!isMultiSimEnabled) {
|
||||
// In the 'SS mode', the port is 0.
|
||||
newUiccSlotMappings.add(new UiccSlotMapping(0, slotId, 0));
|
||||
} else if (removedSubInfo != null) {
|
||||
// DSDS+MEP
|
||||
// The target slot+port is not active, but the all of logical slots are full. It
|
||||
// needs to replace one of logical slots.
|
||||
Log.i(TAG,
|
||||
String.format("Start to set SimSlotMapping from slot%d-port%d to slot%d-port%d",
|
||||
slotId, 0, removedSubInfo.getSimSlotIndex(),
|
||||
removedSubInfo.getPortIndex()));
|
||||
newUiccSlotMappings =
|
||||
uiccSlotMappings.stream().map(uiccSlotMapping -> {
|
||||
if (uiccSlotMapping.getPhysicalSlotIndex()
|
||||
== removedSubInfo.getSimSlotIndex()
|
||||
&& uiccSlotMapping.getPortIndex()
|
||||
== removedSubInfo.getPortIndex()) {
|
||||
return new UiccSlotMapping(0, slotId,
|
||||
uiccSlotMapping.getLogicalSlotIndex());
|
||||
}
|
||||
return uiccSlotMapping;
|
||||
}).collect(Collectors.toList());
|
||||
} else {
|
||||
// DSDS+no MEP
|
||||
// The removable slot should be in UiccSlotMapping.
|
||||
newUiccSlotMappings = uiccSlotMappings;
|
||||
Log.i(TAG, "The removedSubInfo is null");
|
||||
}
|
||||
|
||||
Log.i(TAG, "The SimSlotMapping: " + newUiccSlotMappings);
|
||||
return newUiccSlotMappings;
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user