| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1 | /* | 
 | 2 |  * Copyright (C) 2006 The Android Open Source Project | 
 | 3 |  * | 
 | 4 |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
 | 5 |  * you may not use this file except in compliance with the License. | 
 | 6 |  * You may obtain a copy of the License at | 
 | 7 |  * | 
 | 8 |  *      http://www.apache.org/licenses/LICENSE-2.0 | 
 | 9 |  * | 
 | 10 |  * Unless required by applicable law or agreed to in writing, software | 
 | 11 |  * distributed under the License is distributed on an "AS IS" BASIS, | 
 | 12 |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | 13 |  * See the License for the specific language governing permissions and | 
 | 14 |  * limitations under the License. | 
 | 15 |  */ | 
 | 16 |  | 
 | 17 | package com.android.phone; | 
 | 18 |  | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 19 | import static com.android.internal.telephony.PhoneConstants.SUBSCRIPTION_KEY; | 
 | 20 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 21 | import android.app.ActivityManager; | 
 | 22 | import android.app.AppOpsManager; | 
| Sailesh Nepal | bd76e4e | 2013-10-27 13:59:44 -0700 | [diff] [blame] | 23 | import android.content.ComponentName; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 24 | import android.content.Context; | 
 | 25 | import android.content.Intent; | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 26 | import android.content.SharedPreferences; | 
| Amith Yamasani | 6e11887 | 2016-02-19 12:53:51 -0800 | [diff] [blame] | 27 | import android.content.pm.PackageInfo; | 
| Shishir Agrawal | 60f9c95 | 2014-06-23 12:00:43 -0700 | [diff] [blame] | 28 | import android.content.pm.PackageManager; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 29 | import android.net.Uri; | 
 | 30 | import android.os.AsyncResult; | 
 | 31 | import android.os.Binder; | 
 | 32 | import android.os.Bundle; | 
 | 33 | import android.os.Handler; | 
 | 34 | import android.os.Looper; | 
 | 35 | import android.os.Message; | 
 | 36 | import android.os.Process; | 
| Adam Lesinski | 903a54c | 2016-04-11 14:49:52 -0700 | [diff] [blame] | 37 | import android.os.ResultReceiver; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 38 | import android.os.ServiceManager; | 
 | 39 | import android.os.UserHandle; | 
| Stuart Scott | 981d858 | 2015-04-21 14:09:50 -0700 | [diff] [blame] | 40 | import android.os.UserManager; | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 41 | import android.preference.PreferenceManager; | 
| Ihab Awad | f2177b7 | 2013-11-25 13:33:23 -0800 | [diff] [blame] | 42 | import android.provider.Settings; | 
| Meng Wang | 1a7c35a | 2016-05-05 20:56:15 -0700 | [diff] [blame] | 43 | import android.service.carrier.CarrierIdentifier; | 
| Santos Cordon | 7a1885b | 2015-02-03 11:15:19 -0800 | [diff] [blame] | 44 | import android.telecom.PhoneAccount; | 
| Nancy Chen | 31f9ba1 | 2016-01-06 11:42:12 -0800 | [diff] [blame] | 45 | import android.telecom.PhoneAccountHandle; | 
| Andrew Lee | 9431b83 | 2015-03-09 18:46:45 -0700 | [diff] [blame] | 46 | import android.telecom.TelecomManager; | 
| Junda Liu | 12f7d80 | 2015-05-01 12:06:44 -0700 | [diff] [blame] | 47 | import android.telephony.CarrierConfigManager; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 48 | import android.telephony.CellInfo; | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 49 | import android.telephony.IccOpenLogicalChannelResponse; | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 50 | import android.telephony.ModemActivityInfo; | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 51 | import android.telephony.NeighboringCellInfo; | 
| Wink Saville | 5d475dd | 2014-10-17 15:00:58 -0700 | [diff] [blame] | 52 | import android.telephony.RadioAccessFamily; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 53 | import android.telephony.ServiceState; | 
| Wink Saville | 0f3b5fc | 2014-11-11 08:40:49 -0800 | [diff] [blame] | 54 | import android.telephony.SubscriptionInfo; | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 55 | import android.telephony.SubscriptionManager; | 
| Sanket Padawe | 99ef1e3 | 2016-05-18 16:12:33 -0700 | [diff] [blame] | 56 | import android.telephony.TelephonyHistogram; | 
| Ta-wei Yen | b692960 | 2016-05-24 15:48:27 -0700 | [diff] [blame] | 57 | import android.telephony.TelephonyManager; | 
 | 58 | import android.telephony.VisualVoicemailSmsFilterSettings; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 59 | import android.text.TextUtils; | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 60 | import android.util.ArraySet; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 61 | import android.util.Log; | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 62 | import android.util.Pair; | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 63 | import android.util.Slog; | 
| Andrew Lee | 312e817 | 2014-10-23 17:01:36 -0700 | [diff] [blame] | 64 | import com.android.ims.ImsManager; | 
| Brad Ebinger | 7668100 | 2017-01-23 13:50:20 -0800 | [diff] [blame] | 65 | import com.android.ims.internal.IImsServiceController; | 
 | 66 | import com.android.ims.internal.IImsServiceFeatureListener; | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 67 | import com.android.internal.telephony.CallManager; | 
| Shishir Agrawal | 302c869 | 2015-06-19 13:49:39 -0700 | [diff] [blame] | 68 | import com.android.internal.telephony.CellNetworkScanResult; | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 69 | import com.android.internal.telephony.CommandException; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 70 | import com.android.internal.telephony.DefaultPhoneNotifier; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 71 | import com.android.internal.telephony.ITelephony; | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 72 | import com.android.internal.telephony.IccCard; | 
| Narayan Kamath | 1c496c2 | 2015-04-16 14:40:19 +0100 | [diff] [blame] | 73 | import com.android.internal.telephony.MccTable; | 
| Shishir Agrawal | 302c869 | 2015-06-19 13:49:39 -0700 | [diff] [blame] | 74 | import com.android.internal.telephony.OperatorInfo; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 75 | import com.android.internal.telephony.Phone; | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 76 | import com.android.internal.telephony.PhoneConstants; | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 77 | import com.android.internal.telephony.PhoneFactory; | 
| Wink Saville | 5d475dd | 2014-10-17 15:00:58 -0700 | [diff] [blame] | 78 | import com.android.internal.telephony.ProxyController; | 
| Sanket Padawe | 99ef1e3 | 2016-05-18 16:12:33 -0700 | [diff] [blame] | 79 | import com.android.internal.telephony.RIL; | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 80 | import com.android.internal.telephony.RILConstants; | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 81 | import com.android.internal.telephony.SubscriptionController; | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 82 | import com.android.internal.telephony.uicc.IccIoResult; | 
 | 83 | import com.android.internal.telephony.uicc.IccUtils; | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 84 | import com.android.internal.telephony.uicc.UiccCard; | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 85 | import com.android.internal.telephony.uicc.UiccController; | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 86 | import com.android.internal.util.HexDump; | 
| Ta-wei Yen | b0f695b | 2016-08-08 17:33:11 -0700 | [diff] [blame] | 87 | import com.android.phone.settings.VisualVoicemailSettingsUtil; | 
| Nancy Chen | 31f9ba1 | 2016-01-06 11:42:12 -0800 | [diff] [blame] | 88 | import com.android.phone.settings.VoicemailNotificationSettingsUtil; | 
| Ta-wei Yen | c236d6b | 2016-06-21 13:33:12 -0700 | [diff] [blame] | 89 | import java.io.FileDescriptor; | 
 | 90 | import java.io.PrintWriter; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 91 | import java.util.ArrayList; | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 92 | import java.util.Arrays; | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 93 | import java.util.List; | 
| Narayan Kamath | 1c496c2 | 2015-04-16 14:40:19 +0100 | [diff] [blame] | 94 | import java.util.Locale; | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 95 | import java.util.Map; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 96 |  | 
 | 97 | /** | 
 | 98 |  * Implementation of the ITelephony interface. | 
 | 99 |  */ | 
| Santos Cordon | 117fee7 | 2014-05-16 17:56:12 -0700 | [diff] [blame] | 100 | public class PhoneInterfaceManager extends ITelephony.Stub { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 101 |     private static final String LOG_TAG = "PhoneInterfaceManager"; | 
 | 102 |     private static final boolean DBG = (PhoneGlobals.DBG_LEVEL >= 2); | 
 | 103 |     private static final boolean DBG_LOC = false; | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 104 |     private static final boolean DBG_MERGE = false; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 105 |  | 
 | 106 |     // Message codes used with mMainThreadHandler | 
 | 107 |     private static final int CMD_HANDLE_PIN_MMI = 1; | 
 | 108 |     private static final int CMD_HANDLE_NEIGHBORING_CELL = 2; | 
 | 109 |     private static final int EVENT_NEIGHBORING_CELL_DONE = 3; | 
 | 110 |     private static final int CMD_ANSWER_RINGING_CALL = 4; | 
 | 111 |     private static final int CMD_END_CALL = 5;  // not used yet | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 112 |     private static final int CMD_TRANSMIT_APDU_LOGICAL_CHANNEL = 7; | 
 | 113 |     private static final int EVENT_TRANSMIT_APDU_LOGICAL_CHANNEL_DONE = 8; | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 114 |     private static final int CMD_OPEN_CHANNEL = 9; | 
 | 115 |     private static final int EVENT_OPEN_CHANNEL_DONE = 10; | 
 | 116 |     private static final int CMD_CLOSE_CHANNEL = 11; | 
 | 117 |     private static final int EVENT_CLOSE_CHANNEL_DONE = 12; | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 118 |     private static final int CMD_NV_READ_ITEM = 13; | 
 | 119 |     private static final int EVENT_NV_READ_ITEM_DONE = 14; | 
 | 120 |     private static final int CMD_NV_WRITE_ITEM = 15; | 
 | 121 |     private static final int EVENT_NV_WRITE_ITEM_DONE = 16; | 
 | 122 |     private static final int CMD_NV_WRITE_CDMA_PRL = 17; | 
 | 123 |     private static final int EVENT_NV_WRITE_CDMA_PRL_DONE = 18; | 
 | 124 |     private static final int CMD_NV_RESET_CONFIG = 19; | 
 | 125 |     private static final int EVENT_NV_RESET_CONFIG_DONE = 20; | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 126 |     private static final int CMD_GET_PREFERRED_NETWORK_TYPE = 21; | 
 | 127 |     private static final int EVENT_GET_PREFERRED_NETWORK_TYPE_DONE = 22; | 
 | 128 |     private static final int CMD_SET_PREFERRED_NETWORK_TYPE = 23; | 
 | 129 |     private static final int EVENT_SET_PREFERRED_NETWORK_TYPE_DONE = 24; | 
| Sailesh Nepal | 35b5945 | 2014-03-06 09:26:56 -0800 | [diff] [blame] | 130 |     private static final int CMD_SEND_ENVELOPE = 25; | 
 | 131 |     private static final int EVENT_SEND_ENVELOPE_DONE = 26; | 
| Derek Tan | 6b088ee | 2014-09-05 14:15:18 -0700 | [diff] [blame] | 132 |     private static final int CMD_INVOKE_OEM_RIL_REQUEST_RAW = 27; | 
 | 133 |     private static final int EVENT_INVOKE_OEM_RIL_REQUEST_RAW_DONE = 28; | 
 | 134 |     private static final int CMD_TRANSMIT_APDU_BASIC_CHANNEL = 29; | 
 | 135 |     private static final int EVENT_TRANSMIT_APDU_BASIC_CHANNEL_DONE = 30; | 
 | 136 |     private static final int CMD_EXCHANGE_SIM_IO = 31; | 
 | 137 |     private static final int EVENT_EXCHANGE_SIM_IO_DONE = 32; | 
| Shishir Agrawal | 76d5da9 | 2014-11-09 16:17:25 -0800 | [diff] [blame] | 138 |     private static final int CMD_SET_VOICEMAIL_NUMBER = 33; | 
 | 139 |     private static final int EVENT_SET_VOICEMAIL_NUMBER_DONE = 34; | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 140 |     private static final int CMD_SET_NETWORK_SELECTION_MODE_AUTOMATIC = 35; | 
 | 141 |     private static final int EVENT_SET_NETWORK_SELECTION_MODE_AUTOMATIC_DONE = 36; | 
| Prerepa Viswanadham | 7fcff69 | 2015-06-03 11:20:55 -0700 | [diff] [blame] | 142 |     private static final int CMD_GET_MODEM_ACTIVITY_INFO = 37; | 
 | 143 |     private static final int EVENT_GET_MODEM_ACTIVITY_INFO_DONE = 38; | 
| Shishir Agrawal | 302c869 | 2015-06-19 13:49:39 -0700 | [diff] [blame] | 144 |     private static final int CMD_PERFORM_NETWORK_SCAN = 39; | 
 | 145 |     private static final int EVENT_PERFORM_NETWORK_SCAN_DONE = 40; | 
 | 146 |     private static final int CMD_SET_NETWORK_SELECTION_MODE_MANUAL = 41; | 
 | 147 |     private static final int EVENT_SET_NETWORK_SELECTION_MODE_MANUAL_DONE = 42; | 
| Meng Wang | 1a7c35a | 2016-05-05 20:56:15 -0700 | [diff] [blame] | 148 |     private static final int CMD_SET_ALLOWED_CARRIERS = 43; | 
 | 149 |     private static final int EVENT_SET_ALLOWED_CARRIERS_DONE = 44; | 
 | 150 |     private static final int CMD_GET_ALLOWED_CARRIERS = 45; | 
 | 151 |     private static final int EVENT_GET_ALLOWED_CARRIERS_DONE = 46; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 152 |  | 
 | 153 |     /** The singleton instance. */ | 
 | 154 |     private static PhoneInterfaceManager sInstance; | 
 | 155 |  | 
| Wink Saville | 3ab207e | 2014-11-20 13:07:20 -0800 | [diff] [blame] | 156 |     private PhoneGlobals mApp; | 
 | 157 |     private Phone mPhone; | 
 | 158 |     private CallManager mCM; | 
| Stuart Scott | 981d858 | 2015-04-21 14:09:50 -0700 | [diff] [blame] | 159 |     private UserManager mUserManager; | 
| Wink Saville | 3ab207e | 2014-11-20 13:07:20 -0800 | [diff] [blame] | 160 |     private AppOpsManager mAppOps; | 
 | 161 |     private MainThreadHandler mMainThreadHandler; | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 162 |     private SubscriptionController mSubscriptionController; | 
| Wink Saville | 3ab207e | 2014-11-20 13:07:20 -0800 | [diff] [blame] | 163 |     private SharedPreferences mTelephonySharedPreferences; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 164 |  | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 165 |     private static final String PREF_CARRIERS_ALPHATAG_PREFIX = "carrier_alphtag_"; | 
 | 166 |     private static final String PREF_CARRIERS_NUMBER_PREFIX = "carrier_number_"; | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 167 |     private static final String PREF_CARRIERS_SUBSCRIBER_PREFIX = "carrier_subscriber_"; | 
| Derek Tan | 89e89d4 | 2014-07-08 17:00:10 -0700 | [diff] [blame] | 168 |  | 
 | 169 |     /** | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 170 |      * A request object to use for transmitting data to an ICC. | 
 | 171 |      */ | 
 | 172 |     private static final class IccAPDUArgument { | 
 | 173 |         public int channel, cla, command, p1, p2, p3; | 
 | 174 |         public String data; | 
 | 175 |  | 
 | 176 |         public IccAPDUArgument(int channel, int cla, int command, | 
 | 177 |                 int p1, int p2, int p3, String data) { | 
 | 178 |             this.channel = channel; | 
 | 179 |             this.cla = cla; | 
 | 180 |             this.command = command; | 
 | 181 |             this.p1 = p1; | 
 | 182 |             this.p2 = p2; | 
 | 183 |             this.p3 = p3; | 
 | 184 |             this.data = data; | 
 | 185 |         } | 
 | 186 |     } | 
 | 187 |  | 
 | 188 |     /** | 
| Shishir Agrawal | 77ba317 | 2015-09-10 14:50:19 -0700 | [diff] [blame] | 189 |      * A request object to use for transmitting data to an ICC. | 
 | 190 |      */ | 
 | 191 |     private static final class ManualNetworkSelectionArgument { | 
 | 192 |         public OperatorInfo operatorInfo; | 
 | 193 |         public boolean persistSelection; | 
 | 194 |  | 
 | 195 |         public ManualNetworkSelectionArgument(OperatorInfo operatorInfo, boolean persistSelection) { | 
 | 196 |             this.operatorInfo = operatorInfo; | 
 | 197 |             this.persistSelection = persistSelection; | 
 | 198 |         } | 
 | 199 |     } | 
 | 200 |  | 
 | 201 |     /** | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 202 |      * A request object for use with {@link MainThreadHandler}. Requesters should wait() on the | 
 | 203 |      * request after sending. The main thread will notify the request when it is complete. | 
 | 204 |      */ | 
 | 205 |     private static final class MainThreadRequest { | 
 | 206 |         /** The argument to use for the request */ | 
 | 207 |         public Object argument; | 
 | 208 |         /** The result of the request that is run on the main thread */ | 
 | 209 |         public Object result; | 
| Sanket Padawe | 56e75a3 | 2016-02-08 12:18:19 -0800 | [diff] [blame] | 210 |         // The subscriber id that this request applies to. Defaults to | 
 | 211 |         // SubscriptionManager.INVALID_SUBSCRIPTION_ID | 
 | 212 |         public Integer subId = SubscriptionManager.INVALID_SUBSCRIPTION_ID; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 213 |  | 
 | 214 |         public MainThreadRequest(Object argument) { | 
 | 215 |             this.argument = argument; | 
 | 216 |         } | 
| Shishir Agrawal | 76d5da9 | 2014-11-09 16:17:25 -0800 | [diff] [blame] | 217 |  | 
 | 218 |         public MainThreadRequest(Object argument, Integer subId) { | 
 | 219 |             this.argument = argument; | 
| Sanket Padawe | 56e75a3 | 2016-02-08 12:18:19 -0800 | [diff] [blame] | 220 |             if (subId != null) { | 
 | 221 |                 this.subId = subId; | 
 | 222 |             } | 
| Shishir Agrawal | 76d5da9 | 2014-11-09 16:17:25 -0800 | [diff] [blame] | 223 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 224 |     } | 
 | 225 |  | 
| Sailesh Nepal | cc0375f | 2013-11-13 09:15:18 -0800 | [diff] [blame] | 226 |     private static final class IncomingThirdPartyCallArgs { | 
 | 227 |         public final ComponentName component; | 
 | 228 |         public final String callId; | 
 | 229 |         public final String callerDisplayName; | 
 | 230 |  | 
 | 231 |         public IncomingThirdPartyCallArgs(ComponentName component, String callId, | 
 | 232 |                 String callerDisplayName) { | 
 | 233 |             this.component = component; | 
 | 234 |             this.callId = callId; | 
 | 235 |             this.callerDisplayName = callerDisplayName; | 
 | 236 |         } | 
 | 237 |     } | 
 | 238 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 239 |     /** | 
 | 240 |      * A handler that processes messages on the main thread in the phone process. Since many | 
 | 241 |      * of the Phone calls are not thread safe this is needed to shuttle the requests from the | 
 | 242 |      * inbound binder threads to the main thread in the phone process.  The Binder thread | 
 | 243 |      * may provide a {@link MainThreadRequest} object in the msg.obj field that they are waiting | 
 | 244 |      * on, which will be notified when the operation completes and will contain the result of the | 
 | 245 |      * request. | 
 | 246 |      * | 
 | 247 |      * <p>If a MainThreadRequest object is provided in the msg.obj field, | 
 | 248 |      * note that request.result must be set to something non-null for the calling thread to | 
 | 249 |      * unblock. | 
 | 250 |      */ | 
 | 251 |     private final class MainThreadHandler extends Handler { | 
 | 252 |         @Override | 
 | 253 |         public void handleMessage(Message msg) { | 
 | 254 |             MainThreadRequest request; | 
 | 255 |             Message onCompleted; | 
 | 256 |             AsyncResult ar; | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 257 |             UiccCard uiccCard; | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 258 |             IccAPDUArgument iccArgument; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 259 |  | 
 | 260 |             switch (msg.what) { | 
| Yorke Lee | 716f67e | 2015-06-17 15:39:16 -0700 | [diff] [blame] | 261 |                 case CMD_HANDLE_PIN_MMI: { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 262 |                     request = (MainThreadRequest) msg.obj; | 
| Yorke Lee | 716f67e | 2015-06-17 15:39:16 -0700 | [diff] [blame] | 263 |                     final Phone phone = getPhoneFromRequest(request); | 
 | 264 |                     request.result = phone != null ? | 
 | 265 |                             getPhoneFromRequest(request).handlePinMmi((String) request.argument) | 
 | 266 |                             : false; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 267 |                     // Wake up the requesting thread | 
 | 268 |                     synchronized (request) { | 
 | 269 |                         request.notifyAll(); | 
 | 270 |                     } | 
 | 271 |                     break; | 
| Yorke Lee | 716f67e | 2015-06-17 15:39:16 -0700 | [diff] [blame] | 272 |                 } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 273 |  | 
 | 274 |                 case CMD_HANDLE_NEIGHBORING_CELL: | 
 | 275 |                     request = (MainThreadRequest) msg.obj; | 
 | 276 |                     onCompleted = obtainMessage(EVENT_NEIGHBORING_CELL_DONE, | 
 | 277 |                             request); | 
 | 278 |                     mPhone.getNeighboringCids(onCompleted); | 
 | 279 |                     break; | 
 | 280 |  | 
 | 281 |                 case EVENT_NEIGHBORING_CELL_DONE: | 
 | 282 |                     ar = (AsyncResult) msg.obj; | 
 | 283 |                     request = (MainThreadRequest) ar.userObj; | 
 | 284 |                     if (ar.exception == null && ar.result != null) { | 
 | 285 |                         request.result = ar.result; | 
 | 286 |                     } else { | 
 | 287 |                         // create an empty list to notify the waiting thread | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 288 |                         request.result = new ArrayList<NeighboringCellInfo>(0); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 289 |                     } | 
 | 290 |                     // Wake up the requesting thread | 
 | 291 |                     synchronized (request) { | 
 | 292 |                         request.notifyAll(); | 
 | 293 |                     } | 
 | 294 |                     break; | 
 | 295 |  | 
 | 296 |                 case CMD_ANSWER_RINGING_CALL: | 
| Wink Saville | 0887461 | 2014-08-31 19:19:58 -0700 | [diff] [blame] | 297 |                     request = (MainThreadRequest) msg.obj; | 
| Stuart Scott | 584921c | 2015-01-15 17:10:34 -0800 | [diff] [blame] | 298 |                     int answer_subId = request.subId; | 
| Wink Saville | 0887461 | 2014-08-31 19:19:58 -0700 | [diff] [blame] | 299 |                     answerRingingCallInternal(answer_subId); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 300 |                     break; | 
 | 301 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 302 |                 case CMD_END_CALL: | 
 | 303 |                     request = (MainThreadRequest) msg.obj; | 
| Stuart Scott | 584921c | 2015-01-15 17:10:34 -0800 | [diff] [blame] | 304 |                     int end_subId = request.subId; | 
| Wink Saville | 0887461 | 2014-08-31 19:19:58 -0700 | [diff] [blame] | 305 |                     final boolean hungUp; | 
| Anthony Lee | ae4e36d | 2015-05-21 07:17:46 -0700 | [diff] [blame] | 306 |                     Phone phone = getPhone(end_subId); | 
 | 307 |                     if (phone == null) { | 
 | 308 |                         if (DBG) log("CMD_END_CALL: no phone for id: " + end_subId); | 
 | 309 |                         break; | 
 | 310 |                     } | 
 | 311 |                     int phoneType = phone.getPhoneType(); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 312 |                     if (phoneType == PhoneConstants.PHONE_TYPE_CDMA) { | 
 | 313 |                         // CDMA: If the user presses the Power button we treat it as | 
 | 314 |                         // ending the complete call session | 
| Wink Saville | 0887461 | 2014-08-31 19:19:58 -0700 | [diff] [blame] | 315 |                         hungUp = PhoneUtils.hangupRingingAndActive(getPhone(end_subId)); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 316 |                     } else if (phoneType == PhoneConstants.PHONE_TYPE_GSM) { | 
 | 317 |                         // GSM: End the call as per the Phone state | 
 | 318 |                         hungUp = PhoneUtils.hangup(mCM); | 
 | 319 |                     } else { | 
 | 320 |                         throw new IllegalStateException("Unexpected phone type: " + phoneType); | 
 | 321 |                     } | 
 | 322 |                     if (DBG) log("CMD_END_CALL: " + (hungUp ? "hung up!" : "no call to hang up")); | 
 | 323 |                     request.result = hungUp; | 
 | 324 |                     // Wake up the requesting thread | 
 | 325 |                     synchronized (request) { | 
 | 326 |                         request.notifyAll(); | 
 | 327 |                     } | 
 | 328 |                     break; | 
 | 329 |  | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 330 |                 case CMD_TRANSMIT_APDU_LOGICAL_CHANNEL: | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 331 |                     request = (MainThreadRequest) msg.obj; | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 332 |                     iccArgument = (IccAPDUArgument) request.argument; | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 333 |                     uiccCard = getUiccCardFromRequest(request); | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 334 |                     if (uiccCard == null) { | 
 | 335 |                         loge("iccTransmitApduLogicalChannel: No UICC"); | 
 | 336 |                         request.result = new IccIoResult(0x6F, 0, (byte[])null); | 
 | 337 |                         synchronized (request) { | 
 | 338 |                             request.notifyAll(); | 
 | 339 |                         } | 
 | 340 |                     } else { | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 341 |                         onCompleted = obtainMessage(EVENT_TRANSMIT_APDU_LOGICAL_CHANNEL_DONE, | 
 | 342 |                             request); | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 343 |                         uiccCard.iccTransmitApduLogicalChannel( | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 344 |                             iccArgument.channel, iccArgument.cla, iccArgument.command, | 
 | 345 |                             iccArgument.p1, iccArgument.p2, iccArgument.p3, iccArgument.data, | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 346 |                             onCompleted); | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 347 |                     } | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 348 |                     break; | 
 | 349 |  | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 350 |                 case EVENT_TRANSMIT_APDU_LOGICAL_CHANNEL_DONE: | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 351 |                     ar = (AsyncResult) msg.obj; | 
 | 352 |                     request = (MainThreadRequest) ar.userObj; | 
 | 353 |                     if (ar.exception == null && ar.result != null) { | 
 | 354 |                         request.result = ar.result; | 
 | 355 |                     } else { | 
 | 356 |                         request.result = new IccIoResult(0x6F, 0, (byte[])null); | 
 | 357 |                         if (ar.result == null) { | 
 | 358 |                             loge("iccTransmitApduLogicalChannel: Empty response"); | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 359 |                         } else if (ar.exception instanceof CommandException) { | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 360 |                             loge("iccTransmitApduLogicalChannel: CommandException: " + | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 361 |                                     ar.exception); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 362 |                         } else { | 
 | 363 |                             loge("iccTransmitApduLogicalChannel: Unknown exception"); | 
 | 364 |                         } | 
 | 365 |                     } | 
 | 366 |                     synchronized (request) { | 
 | 367 |                         request.notifyAll(); | 
 | 368 |                     } | 
 | 369 |                     break; | 
 | 370 |  | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 371 |                 case CMD_TRANSMIT_APDU_BASIC_CHANNEL: | 
 | 372 |                     request = (MainThreadRequest) msg.obj; | 
 | 373 |                     iccArgument = (IccAPDUArgument) request.argument; | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 374 |                     uiccCard = getUiccCardFromRequest(request); | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 375 |                     if (uiccCard == null) { | 
 | 376 |                         loge("iccTransmitApduBasicChannel: No UICC"); | 
 | 377 |                         request.result = new IccIoResult(0x6F, 0, (byte[])null); | 
 | 378 |                         synchronized (request) { | 
 | 379 |                             request.notifyAll(); | 
 | 380 |                         } | 
 | 381 |                     } else { | 
 | 382 |                         onCompleted = obtainMessage(EVENT_TRANSMIT_APDU_BASIC_CHANNEL_DONE, | 
 | 383 |                             request); | 
 | 384 |                         uiccCard.iccTransmitApduBasicChannel( | 
 | 385 |                             iccArgument.cla, iccArgument.command, iccArgument.p1, iccArgument.p2, | 
 | 386 |                             iccArgument.p3, iccArgument.data, onCompleted); | 
 | 387 |                     } | 
 | 388 |                     break; | 
 | 389 |  | 
 | 390 |                 case EVENT_TRANSMIT_APDU_BASIC_CHANNEL_DONE: | 
 | 391 |                     ar = (AsyncResult) msg.obj; | 
 | 392 |                     request = (MainThreadRequest) ar.userObj; | 
 | 393 |                     if (ar.exception == null && ar.result != null) { | 
 | 394 |                         request.result = ar.result; | 
 | 395 |                     } else { | 
 | 396 |                         request.result = new IccIoResult(0x6F, 0, (byte[])null); | 
 | 397 |                         if (ar.result == null) { | 
 | 398 |                             loge("iccTransmitApduBasicChannel: Empty response"); | 
 | 399 |                         } else if (ar.exception instanceof CommandException) { | 
 | 400 |                             loge("iccTransmitApduBasicChannel: CommandException: " + | 
 | 401 |                                     ar.exception); | 
 | 402 |                         } else { | 
 | 403 |                             loge("iccTransmitApduBasicChannel: Unknown exception"); | 
 | 404 |                         } | 
 | 405 |                     } | 
 | 406 |                     synchronized (request) { | 
 | 407 |                         request.notifyAll(); | 
 | 408 |                     } | 
 | 409 |                     break; | 
 | 410 |  | 
 | 411 |                 case CMD_EXCHANGE_SIM_IO: | 
 | 412 |                     request = (MainThreadRequest) msg.obj; | 
 | 413 |                     iccArgument = (IccAPDUArgument) request.argument; | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 414 |                     uiccCard = getUiccCardFromRequest(request); | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 415 |                     if (uiccCard == null) { | 
 | 416 |                         loge("iccExchangeSimIO: No UICC"); | 
 | 417 |                         request.result = new IccIoResult(0x6F, 0, (byte[])null); | 
 | 418 |                         synchronized (request) { | 
 | 419 |                             request.notifyAll(); | 
 | 420 |                         } | 
 | 421 |                     } else { | 
 | 422 |                         onCompleted = obtainMessage(EVENT_EXCHANGE_SIM_IO_DONE, | 
 | 423 |                                 request); | 
 | 424 |                         uiccCard.iccExchangeSimIO(iccArgument.cla, /* fileID */ | 
 | 425 |                                 iccArgument.command, iccArgument.p1, iccArgument.p2, iccArgument.p3, | 
 | 426 |                                 iccArgument.data, onCompleted); | 
 | 427 |                     } | 
 | 428 |                     break; | 
 | 429 |  | 
 | 430 |                 case EVENT_EXCHANGE_SIM_IO_DONE: | 
 | 431 |                     ar = (AsyncResult) msg.obj; | 
 | 432 |                     request = (MainThreadRequest) ar.userObj; | 
 | 433 |                     if (ar.exception == null && ar.result != null) { | 
 | 434 |                         request.result = ar.result; | 
 | 435 |                     } else { | 
 | 436 |                         request.result = new IccIoResult(0x6f, 0, (byte[])null); | 
 | 437 |                     } | 
 | 438 |                     synchronized (request) { | 
 | 439 |                         request.notifyAll(); | 
 | 440 |                     } | 
 | 441 |                     break; | 
 | 442 |  | 
| Derek Tan | 4d5e5c1 | 2014-02-04 11:54:58 -0800 | [diff] [blame] | 443 |                 case CMD_SEND_ENVELOPE: | 
 | 444 |                     request = (MainThreadRequest) msg.obj; | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 445 |                     uiccCard = getUiccCardFromRequest(request); | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 446 |                     if (uiccCard == null) { | 
 | 447 |                         loge("sendEnvelopeWithStatus: No UICC"); | 
 | 448 |                         request.result = new IccIoResult(0x6F, 0, (byte[])null); | 
 | 449 |                         synchronized (request) { | 
 | 450 |                             request.notifyAll(); | 
 | 451 |                         } | 
 | 452 |                     } else { | 
 | 453 |                         onCompleted = obtainMessage(EVENT_SEND_ENVELOPE_DONE, request); | 
 | 454 |                         uiccCard.sendEnvelopeWithStatus((String)request.argument, onCompleted); | 
 | 455 |                     } | 
| Derek Tan | 4d5e5c1 | 2014-02-04 11:54:58 -0800 | [diff] [blame] | 456 |                     break; | 
 | 457 |  | 
 | 458 |                 case EVENT_SEND_ENVELOPE_DONE: | 
 | 459 |                     ar = (AsyncResult) msg.obj; | 
 | 460 |                     request = (MainThreadRequest) ar.userObj; | 
| Shishir Agrawal | 9f9877d | 2014-03-14 09:36:27 -0700 | [diff] [blame] | 461 |                     if (ar.exception == null && ar.result != null) { | 
 | 462 |                         request.result = ar.result; | 
| Derek Tan | 4d5e5c1 | 2014-02-04 11:54:58 -0800 | [diff] [blame] | 463 |                     } else { | 
| Shishir Agrawal | 9f9877d | 2014-03-14 09:36:27 -0700 | [diff] [blame] | 464 |                         request.result = new IccIoResult(0x6F, 0, (byte[])null); | 
 | 465 |                         if (ar.result == null) { | 
 | 466 |                             loge("sendEnvelopeWithStatus: Empty response"); | 
 | 467 |                         } else if (ar.exception instanceof CommandException) { | 
 | 468 |                             loge("sendEnvelopeWithStatus: CommandException: " + | 
 | 469 |                                     ar.exception); | 
 | 470 |                         } else { | 
 | 471 |                             loge("sendEnvelopeWithStatus: exception:" + ar.exception); | 
 | 472 |                         } | 
| Derek Tan | 4d5e5c1 | 2014-02-04 11:54:58 -0800 | [diff] [blame] | 473 |                     } | 
 | 474 |                     synchronized (request) { | 
 | 475 |                         request.notifyAll(); | 
 | 476 |                     } | 
 | 477 |                     break; | 
 | 478 |  | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 479 |                 case CMD_OPEN_CHANNEL: | 
 | 480 |                     request = (MainThreadRequest) msg.obj; | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 481 |                     uiccCard = getUiccCardFromRequest(request); | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 482 |                     if (uiccCard == null) { | 
 | 483 |                         loge("iccOpenLogicalChannel: No UICC"); | 
| Shishir Agrawal | fc0492a | 2016-02-17 11:15:33 -0800 | [diff] [blame] | 484 |                         request.result = new IccOpenLogicalChannelResponse(-1, | 
 | 485 |                             IccOpenLogicalChannelResponse.STATUS_MISSING_RESOURCE, null); | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 486 |                         synchronized (request) { | 
 | 487 |                             request.notifyAll(); | 
 | 488 |                         } | 
 | 489 |                     } else { | 
 | 490 |                         onCompleted = obtainMessage(EVENT_OPEN_CHANNEL_DONE, request); | 
 | 491 |                         uiccCard.iccOpenLogicalChannel((String)request.argument, onCompleted); | 
 | 492 |                     } | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 493 |                     break; | 
 | 494 |  | 
 | 495 |                 case EVENT_OPEN_CHANNEL_DONE: | 
 | 496 |                     ar = (AsyncResult) msg.obj; | 
 | 497 |                     request = (MainThreadRequest) ar.userObj; | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 498 |                     IccOpenLogicalChannelResponse openChannelResp; | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 499 |                     if (ar.exception == null && ar.result != null) { | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 500 |                         int[] result = (int[]) ar.result; | 
 | 501 |                         int channelId = result[0]; | 
 | 502 |                         byte[] selectResponse = null; | 
 | 503 |                         if (result.length > 1) { | 
 | 504 |                             selectResponse = new byte[result.length - 1]; | 
 | 505 |                             for (int i = 1; i < result.length; ++i) { | 
 | 506 |                                 selectResponse[i - 1] = (byte) result[i]; | 
 | 507 |                             } | 
 | 508 |                         } | 
 | 509 |                         openChannelResp = new IccOpenLogicalChannelResponse(channelId, | 
| Shishir Agrawal | 527e8bf | 2014-08-25 08:54:56 -0700 | [diff] [blame] | 510 |                             IccOpenLogicalChannelResponse.STATUS_NO_ERROR, selectResponse); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 511 |                     } else { | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 512 |                         if (ar.result == null) { | 
 | 513 |                             loge("iccOpenLogicalChannel: Empty response"); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 514 |                         } | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 515 |                         if (ar.exception != null) { | 
 | 516 |                             loge("iccOpenLogicalChannel: Exception: " + ar.exception); | 
 | 517 |                         } | 
 | 518 |  | 
| Shishir Agrawal | 527e8bf | 2014-08-25 08:54:56 -0700 | [diff] [blame] | 519 |                         int errorCode = IccOpenLogicalChannelResponse.STATUS_UNKNOWN_ERROR; | 
| Junda Liu | a754ba1 | 2015-05-20 01:17:52 -0700 | [diff] [blame] | 520 |                         if (ar.exception instanceof CommandException) { | 
 | 521 |                             CommandException.Error error = | 
 | 522 |                                 ((CommandException) (ar.exception)).getCommandError(); | 
 | 523 |                             if (error == CommandException.Error.MISSING_RESOURCE) { | 
| Shishir Agrawal | 527e8bf | 2014-08-25 08:54:56 -0700 | [diff] [blame] | 524 |                                 errorCode = IccOpenLogicalChannelResponse.STATUS_MISSING_RESOURCE; | 
| Junda Liu | a754ba1 | 2015-05-20 01:17:52 -0700 | [diff] [blame] | 525 |                             } else if (error == CommandException.Error.NO_SUCH_ELEMENT) { | 
| Shishir Agrawal | 527e8bf | 2014-08-25 08:54:56 -0700 | [diff] [blame] | 526 |                                 errorCode = IccOpenLogicalChannelResponse.STATUS_NO_SUCH_ELEMENT; | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 527 |                             } | 
 | 528 |                         } | 
 | 529 |                         openChannelResp = new IccOpenLogicalChannelResponse( | 
 | 530 |                             IccOpenLogicalChannelResponse.INVALID_CHANNEL, errorCode, null); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 531 |                     } | 
| Shishir Agrawal | 82c8a46 | 2014-07-31 18:13:17 -0700 | [diff] [blame] | 532 |                     request.result = openChannelResp; | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 533 |                     synchronized (request) { | 
 | 534 |                         request.notifyAll(); | 
 | 535 |                     } | 
 | 536 |                     break; | 
 | 537 |  | 
 | 538 |                 case CMD_CLOSE_CHANNEL: | 
 | 539 |                     request = (MainThreadRequest) msg.obj; | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 540 |                     uiccCard = getUiccCardFromRequest(request); | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 541 |                     if (uiccCard == null) { | 
 | 542 |                         loge("iccCloseLogicalChannel: No UICC"); | 
 | 543 |                         request.result = new IccIoResult(0x6F, 0, (byte[])null); | 
 | 544 |                         synchronized (request) { | 
 | 545 |                             request.notifyAll(); | 
 | 546 |                         } | 
 | 547 |                     } else { | 
 | 548 |                         onCompleted = obtainMessage(EVENT_CLOSE_CHANNEL_DONE, request); | 
 | 549 |                         uiccCard.iccCloseLogicalChannel((Integer) request.argument, onCompleted); | 
 | 550 |                     } | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 551 |                     break; | 
 | 552 |  | 
 | 553 |                 case EVENT_CLOSE_CHANNEL_DONE: | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 554 |                     handleNullReturnEvent(msg, "iccCloseLogicalChannel"); | 
 | 555 |                     break; | 
 | 556 |  | 
 | 557 |                 case CMD_NV_READ_ITEM: | 
 | 558 |                     request = (MainThreadRequest) msg.obj; | 
 | 559 |                     onCompleted = obtainMessage(EVENT_NV_READ_ITEM_DONE, request); | 
 | 560 |                     mPhone.nvReadItem((Integer) request.argument, onCompleted); | 
 | 561 |                     break; | 
 | 562 |  | 
 | 563 |                 case EVENT_NV_READ_ITEM_DONE: | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 564 |                     ar = (AsyncResult) msg.obj; | 
 | 565 |                     request = (MainThreadRequest) ar.userObj; | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 566 |                     if (ar.exception == null && ar.result != null) { | 
 | 567 |                         request.result = ar.result;     // String | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 568 |                     } else { | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 569 |                         request.result = ""; | 
 | 570 |                         if (ar.result == null) { | 
 | 571 |                             loge("nvReadItem: Empty response"); | 
 | 572 |                         } else if (ar.exception instanceof CommandException) { | 
 | 573 |                             loge("nvReadItem: CommandException: " + | 
 | 574 |                                     ar.exception); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 575 |                         } else { | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 576 |                             loge("nvReadItem: Unknown exception"); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 577 |                         } | 
 | 578 |                     } | 
 | 579 |                     synchronized (request) { | 
 | 580 |                         request.notifyAll(); | 
 | 581 |                     } | 
 | 582 |                     break; | 
 | 583 |  | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 584 |                 case CMD_NV_WRITE_ITEM: | 
 | 585 |                     request = (MainThreadRequest) msg.obj; | 
 | 586 |                     onCompleted = obtainMessage(EVENT_NV_WRITE_ITEM_DONE, request); | 
 | 587 |                     Pair<Integer, String> idValue = (Pair<Integer, String>) request.argument; | 
 | 588 |                     mPhone.nvWriteItem(idValue.first, idValue.second, onCompleted); | 
 | 589 |                     break; | 
 | 590 |  | 
 | 591 |                 case EVENT_NV_WRITE_ITEM_DONE: | 
 | 592 |                     handleNullReturnEvent(msg, "nvWriteItem"); | 
 | 593 |                     break; | 
 | 594 |  | 
 | 595 |                 case CMD_NV_WRITE_CDMA_PRL: | 
 | 596 |                     request = (MainThreadRequest) msg.obj; | 
 | 597 |                     onCompleted = obtainMessage(EVENT_NV_WRITE_CDMA_PRL_DONE, request); | 
 | 598 |                     mPhone.nvWriteCdmaPrl((byte[]) request.argument, onCompleted); | 
 | 599 |                     break; | 
 | 600 |  | 
 | 601 |                 case EVENT_NV_WRITE_CDMA_PRL_DONE: | 
 | 602 |                     handleNullReturnEvent(msg, "nvWriteCdmaPrl"); | 
 | 603 |                     break; | 
 | 604 |  | 
 | 605 |                 case CMD_NV_RESET_CONFIG: | 
 | 606 |                     request = (MainThreadRequest) msg.obj; | 
 | 607 |                     onCompleted = obtainMessage(EVENT_NV_RESET_CONFIG_DONE, request); | 
 | 608 |                     mPhone.nvResetConfig((Integer) request.argument, onCompleted); | 
 | 609 |                     break; | 
 | 610 |  | 
 | 611 |                 case EVENT_NV_RESET_CONFIG_DONE: | 
 | 612 |                     handleNullReturnEvent(msg, "nvResetConfig"); | 
 | 613 |                     break; | 
 | 614 |  | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 615 |                 case CMD_GET_PREFERRED_NETWORK_TYPE: | 
 | 616 |                     request = (MainThreadRequest) msg.obj; | 
 | 617 |                     onCompleted = obtainMessage(EVENT_GET_PREFERRED_NETWORK_TYPE_DONE, request); | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 618 |                     getPhoneFromRequest(request).getPreferredNetworkType(onCompleted); | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 619 |                     break; | 
 | 620 |  | 
 | 621 |                 case EVENT_GET_PREFERRED_NETWORK_TYPE_DONE: | 
 | 622 |                     ar = (AsyncResult) msg.obj; | 
 | 623 |                     request = (MainThreadRequest) ar.userObj; | 
 | 624 |                     if (ar.exception == null && ar.result != null) { | 
 | 625 |                         request.result = ar.result;     // Integer | 
 | 626 |                     } else { | 
| Sanket Padawe | cfc2d35 | 2016-01-05 19:52:14 -0800 | [diff] [blame] | 627 |                         request.result = null; | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 628 |                         if (ar.result == null) { | 
 | 629 |                             loge("getPreferredNetworkType: Empty response"); | 
 | 630 |                         } else if (ar.exception instanceof CommandException) { | 
 | 631 |                             loge("getPreferredNetworkType: CommandException: " + | 
 | 632 |                                     ar.exception); | 
 | 633 |                         } else { | 
 | 634 |                             loge("getPreferredNetworkType: Unknown exception"); | 
 | 635 |                         } | 
 | 636 |                     } | 
 | 637 |                     synchronized (request) { | 
 | 638 |                         request.notifyAll(); | 
 | 639 |                     } | 
 | 640 |                     break; | 
 | 641 |  | 
 | 642 |                 case CMD_SET_PREFERRED_NETWORK_TYPE: | 
 | 643 |                     request = (MainThreadRequest) msg.obj; | 
 | 644 |                     onCompleted = obtainMessage(EVENT_SET_PREFERRED_NETWORK_TYPE_DONE, request); | 
 | 645 |                     int networkType = (Integer) request.argument; | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 646 |                     getPhoneFromRequest(request).setPreferredNetworkType(networkType, onCompleted); | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 647 |                     break; | 
 | 648 |  | 
 | 649 |                 case EVENT_SET_PREFERRED_NETWORK_TYPE_DONE: | 
 | 650 |                     handleNullReturnEvent(msg, "setPreferredNetworkType"); | 
 | 651 |                     break; | 
 | 652 |  | 
| Steven Liu | 4bf01bc | 2014-07-17 11:05:29 -0500 | [diff] [blame] | 653 |                 case CMD_INVOKE_OEM_RIL_REQUEST_RAW: | 
 | 654 |                     request = (MainThreadRequest)msg.obj; | 
 | 655 |                     onCompleted = obtainMessage(EVENT_INVOKE_OEM_RIL_REQUEST_RAW_DONE, request); | 
 | 656 |                     mPhone.invokeOemRilRequestRaw((byte[])request.argument, onCompleted); | 
 | 657 |                     break; | 
 | 658 |  | 
 | 659 |                 case EVENT_INVOKE_OEM_RIL_REQUEST_RAW_DONE: | 
 | 660 |                     ar = (AsyncResult)msg.obj; | 
 | 661 |                     request = (MainThreadRequest)ar.userObj; | 
 | 662 |                     request.result = ar; | 
 | 663 |                     synchronized (request) { | 
 | 664 |                         request.notifyAll(); | 
 | 665 |                     } | 
 | 666 |                     break; | 
 | 667 |  | 
| Shishir Agrawal | 76d5da9 | 2014-11-09 16:17:25 -0800 | [diff] [blame] | 668 |                 case CMD_SET_VOICEMAIL_NUMBER: | 
 | 669 |                     request = (MainThreadRequest) msg.obj; | 
 | 670 |                     onCompleted = obtainMessage(EVENT_SET_VOICEMAIL_NUMBER_DONE, request); | 
 | 671 |                     Pair<String, String> tagNum = (Pair<String, String>) request.argument; | 
| Stuart Scott | 584921c | 2015-01-15 17:10:34 -0800 | [diff] [blame] | 672 |                     getPhoneFromRequest(request).setVoiceMailNumber(tagNum.first, tagNum.second, | 
 | 673 |                             onCompleted); | 
| Shishir Agrawal | 76d5da9 | 2014-11-09 16:17:25 -0800 | [diff] [blame] | 674 |                     break; | 
 | 675 |  | 
 | 676 |                 case EVENT_SET_VOICEMAIL_NUMBER_DONE: | 
 | 677 |                     handleNullReturnEvent(msg, "setVoicemailNumber"); | 
 | 678 |                     break; | 
 | 679 |  | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 680 |                 case CMD_SET_NETWORK_SELECTION_MODE_AUTOMATIC: | 
 | 681 |                     request = (MainThreadRequest) msg.obj; | 
 | 682 |                     onCompleted = obtainMessage(EVENT_SET_NETWORK_SELECTION_MODE_AUTOMATIC_DONE, | 
 | 683 |                             request); | 
 | 684 |                     getPhoneFromRequest(request).setNetworkSelectionModeAutomatic(onCompleted); | 
 | 685 |                     break; | 
 | 686 |  | 
 | 687 |                 case EVENT_SET_NETWORK_SELECTION_MODE_AUTOMATIC_DONE: | 
 | 688 |                     handleNullReturnEvent(msg, "setNetworkSelectionModeAutomatic"); | 
 | 689 |                     break; | 
 | 690 |  | 
| Shishir Agrawal | 302c869 | 2015-06-19 13:49:39 -0700 | [diff] [blame] | 691 |                 case CMD_PERFORM_NETWORK_SCAN: | 
 | 692 |                     request = (MainThreadRequest) msg.obj; | 
 | 693 |                     onCompleted = obtainMessage(EVENT_PERFORM_NETWORK_SCAN_DONE, request); | 
 | 694 |                     getPhoneFromRequest(request).getAvailableNetworks(onCompleted); | 
 | 695 |                     break; | 
 | 696 |  | 
 | 697 |                 case EVENT_PERFORM_NETWORK_SCAN_DONE: | 
 | 698 |                     ar = (AsyncResult) msg.obj; | 
 | 699 |                     request = (MainThreadRequest) ar.userObj; | 
 | 700 |                     CellNetworkScanResult cellScanResult; | 
 | 701 |                     if (ar.exception == null && ar.result != null) { | 
 | 702 |                         cellScanResult = new CellNetworkScanResult( | 
 | 703 |                                 CellNetworkScanResult.STATUS_SUCCESS, | 
 | 704 |                                 (List<OperatorInfo>) ar.result); | 
 | 705 |                     } else { | 
 | 706 |                         if (ar.result == null) { | 
 | 707 |                             loge("getCellNetworkScanResults: Empty response"); | 
 | 708 |                         } | 
 | 709 |                         if (ar.exception != null) { | 
 | 710 |                             loge("getCellNetworkScanResults: Exception: " + ar.exception); | 
 | 711 |                         } | 
 | 712 |                         int errorCode = CellNetworkScanResult.STATUS_UNKNOWN_ERROR; | 
 | 713 |                         if (ar.exception instanceof CommandException) { | 
 | 714 |                             CommandException.Error error = | 
 | 715 |                                 ((CommandException) (ar.exception)).getCommandError(); | 
 | 716 |                             if (error == CommandException.Error.RADIO_NOT_AVAILABLE) { | 
 | 717 |                                 errorCode = CellNetworkScanResult.STATUS_RADIO_NOT_AVAILABLE; | 
 | 718 |                             } else if (error == CommandException.Error.GENERIC_FAILURE) { | 
 | 719 |                                 errorCode = CellNetworkScanResult.STATUS_RADIO_GENERIC_FAILURE; | 
 | 720 |                             } | 
 | 721 |                         } | 
 | 722 |                         cellScanResult = new CellNetworkScanResult(errorCode, null); | 
 | 723 |                     } | 
 | 724 |                     request.result = cellScanResult; | 
 | 725 |                     synchronized (request) { | 
 | 726 |                         request.notifyAll(); | 
 | 727 |                     } | 
 | 728 |                     break; | 
 | 729 |  | 
 | 730 |                 case CMD_SET_NETWORK_SELECTION_MODE_MANUAL: | 
 | 731 |                     request = (MainThreadRequest) msg.obj; | 
| Shishir Agrawal | 77ba317 | 2015-09-10 14:50:19 -0700 | [diff] [blame] | 732 |                     ManualNetworkSelectionArgument selArg = | 
 | 733 |                             (ManualNetworkSelectionArgument) request.argument; | 
| Shishir Agrawal | 302c869 | 2015-06-19 13:49:39 -0700 | [diff] [blame] | 734 |                     onCompleted = obtainMessage(EVENT_SET_NETWORK_SELECTION_MODE_MANUAL_DONE, | 
 | 735 |                             request); | 
| Shishir Agrawal | 77ba317 | 2015-09-10 14:50:19 -0700 | [diff] [blame] | 736 |                     getPhoneFromRequest(request).selectNetworkManually(selArg.operatorInfo, | 
 | 737 |                             selArg.persistSelection, onCompleted); | 
| Shishir Agrawal | 302c869 | 2015-06-19 13:49:39 -0700 | [diff] [blame] | 738 |                     break; | 
 | 739 |  | 
 | 740 |                 case EVENT_SET_NETWORK_SELECTION_MODE_MANUAL_DONE: | 
 | 741 |                     handleNullReturnEvent(msg, "setNetworkSelectionModeManual"); | 
 | 742 |                     break; | 
 | 743 |  | 
| Prerepa Viswanadham | 7fcff69 | 2015-06-03 11:20:55 -0700 | [diff] [blame] | 744 |                 case CMD_GET_MODEM_ACTIVITY_INFO: | 
 | 745 |                     request = (MainThreadRequest) msg.obj; | 
 | 746 |                     onCompleted = obtainMessage(EVENT_GET_MODEM_ACTIVITY_INFO_DONE, request); | 
| Prerepa Viswanadham | 61a60ad | 2015-06-08 18:07:51 -0700 | [diff] [blame] | 747 |                     mPhone.getModemActivityInfo(onCompleted); | 
| Prerepa Viswanadham | 7fcff69 | 2015-06-03 11:20:55 -0700 | [diff] [blame] | 748 |                     break; | 
 | 749 |  | 
 | 750 |                 case EVENT_GET_MODEM_ACTIVITY_INFO_DONE: | 
 | 751 |                     ar = (AsyncResult) msg.obj; | 
 | 752 |                     request = (MainThreadRequest) ar.userObj; | 
 | 753 |                     if (ar.exception == null && ar.result != null) { | 
 | 754 |                         request.result = ar.result; | 
 | 755 |                     } else { | 
 | 756 |                         if (ar.result == null) { | 
 | 757 |                             loge("queryModemActivityInfo: Empty response"); | 
 | 758 |                         } else if (ar.exception instanceof CommandException) { | 
 | 759 |                             loge("queryModemActivityInfo: CommandException: " + | 
 | 760 |                                     ar.exception); | 
 | 761 |                         } else { | 
 | 762 |                             loge("queryModemActivityInfo: Unknown exception"); | 
 | 763 |                         } | 
 | 764 |                     } | 
| Amit Mahajan | d476622 | 2016-01-28 15:28:28 -0800 | [diff] [blame] | 765 |                     // Result cannot be null. Return ModemActivityInfo with all fields set to 0. | 
 | 766 |                     if (request.result == null) { | 
 | 767 |                         request.result = new ModemActivityInfo(0, 0, 0, null, 0, 0); | 
 | 768 |                     } | 
| Prerepa Viswanadham | 7fcff69 | 2015-06-03 11:20:55 -0700 | [diff] [blame] | 769 |                     synchronized (request) { | 
 | 770 |                         request.notifyAll(); | 
 | 771 |                     } | 
 | 772 |                     break; | 
 | 773 |  | 
| Meng Wang | 1a7c35a | 2016-05-05 20:56:15 -0700 | [diff] [blame] | 774 |                 case CMD_SET_ALLOWED_CARRIERS: | 
 | 775 |                     request = (MainThreadRequest) msg.obj; | 
 | 776 |                     onCompleted = obtainMessage(EVENT_SET_ALLOWED_CARRIERS_DONE, request); | 
 | 777 |                     mPhone.setAllowedCarriers( | 
 | 778 |                             (List<CarrierIdentifier>) request.argument, | 
 | 779 |                             onCompleted); | 
 | 780 |                     break; | 
 | 781 |  | 
 | 782 |                 case EVENT_SET_ALLOWED_CARRIERS_DONE: | 
 | 783 |                     ar = (AsyncResult) msg.obj; | 
 | 784 |                     request = (MainThreadRequest) ar.userObj; | 
 | 785 |                     if (ar.exception == null && ar.result != null) { | 
 | 786 |                         request.result = ar.result; | 
 | 787 |                     } else { | 
 | 788 |                         if (ar.result == null) { | 
 | 789 |                             loge("setAllowedCarriers: Empty response"); | 
 | 790 |                         } else if (ar.exception instanceof CommandException) { | 
 | 791 |                             loge("setAllowedCarriers: CommandException: " + | 
 | 792 |                                     ar.exception); | 
 | 793 |                         } else { | 
 | 794 |                             loge("setAllowedCarriers: Unknown exception"); | 
 | 795 |                         } | 
 | 796 |                     } | 
 | 797 |                     // Result cannot be null. Return -1 on error. | 
 | 798 |                     if (request.result == null) { | 
 | 799 |                         request.result = new int[]{-1}; | 
 | 800 |                     } | 
 | 801 |                     synchronized (request) { | 
 | 802 |                         request.notifyAll(); | 
 | 803 |                     } | 
 | 804 |                     break; | 
 | 805 |  | 
 | 806 |                 case CMD_GET_ALLOWED_CARRIERS: | 
 | 807 |                     request = (MainThreadRequest) msg.obj; | 
 | 808 |                     onCompleted = obtainMessage(EVENT_GET_ALLOWED_CARRIERS_DONE, request); | 
 | 809 |                     mPhone.getAllowedCarriers(onCompleted); | 
 | 810 |                     break; | 
 | 811 |  | 
 | 812 |                 case EVENT_GET_ALLOWED_CARRIERS_DONE: | 
 | 813 |                     ar = (AsyncResult) msg.obj; | 
 | 814 |                     request = (MainThreadRequest) ar.userObj; | 
 | 815 |                     if (ar.exception == null && ar.result != null) { | 
 | 816 |                         request.result = ar.result; | 
 | 817 |                     } else { | 
 | 818 |                         if (ar.result == null) { | 
 | 819 |                             loge("getAllowedCarriers: Empty response"); | 
 | 820 |                         } else if (ar.exception instanceof CommandException) { | 
 | 821 |                             loge("getAllowedCarriers: CommandException: " + | 
 | 822 |                                     ar.exception); | 
 | 823 |                         } else { | 
 | 824 |                             loge("getAllowedCarriers: Unknown exception"); | 
 | 825 |                         } | 
 | 826 |                     } | 
 | 827 |                     // Result cannot be null. Return empty list of CarrierIdentifier. | 
 | 828 |                     if (request.result == null) { | 
 | 829 |                         request.result = new ArrayList<CarrierIdentifier>(0); | 
 | 830 |                     } | 
 | 831 |                     synchronized (request) { | 
 | 832 |                         request.notifyAll(); | 
 | 833 |                     } | 
 | 834 |                     break; | 
 | 835 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 836 |                 default: | 
 | 837 |                     Log.w(LOG_TAG, "MainThreadHandler: unexpected message code: " + msg.what); | 
 | 838 |                     break; | 
 | 839 |             } | 
 | 840 |         } | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 841 |  | 
 | 842 |         private void handleNullReturnEvent(Message msg, String command) { | 
 | 843 |             AsyncResult ar = (AsyncResult) msg.obj; | 
 | 844 |             MainThreadRequest request = (MainThreadRequest) ar.userObj; | 
 | 845 |             if (ar.exception == null) { | 
 | 846 |                 request.result = true; | 
 | 847 |             } else { | 
 | 848 |                 request.result = false; | 
 | 849 |                 if (ar.exception instanceof CommandException) { | 
 | 850 |                     loge(command + ": CommandException: " + ar.exception); | 
 | 851 |                 } else { | 
 | 852 |                     loge(command + ": Unknown exception"); | 
 | 853 |                 } | 
 | 854 |             } | 
 | 855 |             synchronized (request) { | 
 | 856 |                 request.notifyAll(); | 
 | 857 |             } | 
 | 858 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 859 |     } | 
 | 860 |  | 
 | 861 |     /** | 
 | 862 |      * Posts the specified command to be executed on the main thread, | 
 | 863 |      * waits for the request to complete, and returns the result. | 
 | 864 |      * @see #sendRequestAsync | 
 | 865 |      */ | 
 | 866 |     private Object sendRequest(int command, Object argument) { | 
| Sanket Padawe | 56e75a3 | 2016-02-08 12:18:19 -0800 | [diff] [blame] | 867 |         return sendRequest(command, argument, SubscriptionManager.INVALID_SUBSCRIPTION_ID); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 868 |     } | 
 | 869 |  | 
 | 870 |     /** | 
 | 871 |      * Posts the specified command to be executed on the main thread, | 
 | 872 |      * waits for the request to complete, and returns the result. | 
 | 873 |      * @see #sendRequestAsync | 
 | 874 |      */ | 
| Shishir Agrawal | 76d5da9 | 2014-11-09 16:17:25 -0800 | [diff] [blame] | 875 |     private Object sendRequest(int command, Object argument, Integer subId) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 876 |         if (Looper.myLooper() == mMainThreadHandler.getLooper()) { | 
 | 877 |             throw new RuntimeException("This method will deadlock if called from the main thread."); | 
 | 878 |         } | 
 | 879 |  | 
| Shishir Agrawal | 76d5da9 | 2014-11-09 16:17:25 -0800 | [diff] [blame] | 880 |         MainThreadRequest request = new MainThreadRequest(argument, subId); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 881 |         Message msg = mMainThreadHandler.obtainMessage(command, request); | 
 | 882 |         msg.sendToTarget(); | 
 | 883 |  | 
 | 884 |         // Wait for the request to complete | 
 | 885 |         synchronized (request) { | 
 | 886 |             while (request.result == null) { | 
 | 887 |                 try { | 
 | 888 |                     request.wait(); | 
 | 889 |                 } catch (InterruptedException e) { | 
 | 890 |                     // Do nothing, go back and wait until the request is complete | 
 | 891 |                 } | 
 | 892 |             } | 
 | 893 |         } | 
 | 894 |         return request.result; | 
 | 895 |     } | 
 | 896 |  | 
 | 897 |     /** | 
 | 898 |      * Asynchronous ("fire and forget") version of sendRequest(): | 
 | 899 |      * Posts the specified command to be executed on the main thread, and | 
 | 900 |      * returns immediately. | 
 | 901 |      * @see #sendRequest | 
 | 902 |      */ | 
 | 903 |     private void sendRequestAsync(int command) { | 
 | 904 |         mMainThreadHandler.sendEmptyMessage(command); | 
 | 905 |     } | 
 | 906 |  | 
 | 907 |     /** | 
| Sailesh Nepal | bd76e4e | 2013-10-27 13:59:44 -0700 | [diff] [blame] | 908 |      * Same as {@link #sendRequestAsync(int)} except it takes an argument. | 
 | 909 |      * @see {@link #sendRequest(int,Object)} | 
 | 910 |      */ | 
 | 911 |     private void sendRequestAsync(int command, Object argument) { | 
 | 912 |         MainThreadRequest request = new MainThreadRequest(argument); | 
 | 913 |         Message msg = mMainThreadHandler.obtainMessage(command, request); | 
 | 914 |         msg.sendToTarget(); | 
 | 915 |     } | 
 | 916 |  | 
 | 917 |     /** | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 918 |      * Initialize the singleton PhoneInterfaceManager instance. | 
 | 919 |      * This is only done once, at startup, from PhoneApp.onCreate(). | 
 | 920 |      */ | 
| Sailesh Nepal | 194161e | 2014-07-03 08:57:44 -0700 | [diff] [blame] | 921 |     /* package */ static PhoneInterfaceManager init(PhoneGlobals app, Phone phone) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 922 |         synchronized (PhoneInterfaceManager.class) { | 
 | 923 |             if (sInstance == null) { | 
| Sailesh Nepal | 194161e | 2014-07-03 08:57:44 -0700 | [diff] [blame] | 924 |                 sInstance = new PhoneInterfaceManager(app, phone); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 925 |             } else { | 
 | 926 |                 Log.wtf(LOG_TAG, "init() called multiple times!  sInstance = " + sInstance); | 
 | 927 |             } | 
 | 928 |             return sInstance; | 
 | 929 |         } | 
 | 930 |     } | 
 | 931 |  | 
 | 932 |     /** Private constructor; @see init() */ | 
| Sailesh Nepal | 194161e | 2014-07-03 08:57:44 -0700 | [diff] [blame] | 933 |     private PhoneInterfaceManager(PhoneGlobals app, Phone phone) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 934 |         mApp = app; | 
 | 935 |         mPhone = phone; | 
 | 936 |         mCM = PhoneGlobals.getInstance().mCM; | 
| Stuart Scott | 981d858 | 2015-04-21 14:09:50 -0700 | [diff] [blame] | 937 |         mUserManager = (UserManager) app.getSystemService(Context.USER_SERVICE); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 938 |         mAppOps = (AppOpsManager)app.getSystemService(Context.APP_OPS_SERVICE); | 
 | 939 |         mMainThreadHandler = new MainThreadHandler(); | 
| Andrew Lee | df14ead | 2014-10-17 14:22:52 -0700 | [diff] [blame] | 940 |         mTelephonySharedPreferences = | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 941 |                 PreferenceManager.getDefaultSharedPreferences(mPhone.getContext()); | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 942 |         mSubscriptionController = SubscriptionController.getInstance(); | 
| Wink Saville | 3ab207e | 2014-11-20 13:07:20 -0800 | [diff] [blame] | 943 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 944 |         publish(); | 
 | 945 |     } | 
 | 946 |  | 
 | 947 |     private void publish() { | 
 | 948 |         if (DBG) log("publish: " + this); | 
 | 949 |  | 
 | 950 |         ServiceManager.addService("phone", this); | 
 | 951 |     } | 
 | 952 |  | 
| Stuart Scott | 584921c | 2015-01-15 17:10:34 -0800 | [diff] [blame] | 953 |     private Phone getPhoneFromRequest(MainThreadRequest request) { | 
| Sanket Padawe | 56e75a3 | 2016-02-08 12:18:19 -0800 | [diff] [blame] | 954 |         return (request.subId == SubscriptionManager.INVALID_SUBSCRIPTION_ID) | 
 | 955 |                 ? mPhone : getPhone(request.subId); | 
| Stuart Scott | 584921c | 2015-01-15 17:10:34 -0800 | [diff] [blame] | 956 |     } | 
 | 957 |  | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 958 |     private UiccCard getUiccCardFromRequest(MainThreadRequest request) { | 
 | 959 |         Phone phone = getPhoneFromRequest(request); | 
 | 960 |         return phone == null ? null : | 
 | 961 |                 UiccController.getInstance().getUiccCard(phone.getPhoneId()); | 
 | 962 |     } | 
 | 963 |  | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 964 |     // returns phone associated with the subId. | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 965 |     private Phone getPhone(int subId) { | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 966 |         return PhoneFactory.getPhone(mSubscriptionController.getPhoneId(subId)); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 967 |     } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 968 |     // | 
 | 969 |     // Implementation of the ITelephony interface. | 
 | 970 |     // | 
 | 971 |  | 
 | 972 |     public void dial(String number) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 973 |         dialForSubscriber(getPreferredVoiceSubscription(), number); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 974 |     } | 
 | 975 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 976 |     public void dialForSubscriber(int subId, String number) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 977 |         if (DBG) log("dial: " + number); | 
 | 978 |         // No permission check needed here: This is just a wrapper around the | 
 | 979 |         // ACTION_DIAL intent, which is available to any app since it puts up | 
 | 980 |         // the UI before it does anything. | 
 | 981 |  | 
 | 982 |         String url = createTelUrl(number); | 
 | 983 |         if (url == null) { | 
 | 984 |             return; | 
 | 985 |         } | 
 | 986 |  | 
 | 987 |         // PENDING: should we just silently fail if phone is offhook or ringing? | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 988 |         PhoneConstants.State state = mCM.getState(subId); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 989 |         if (state != PhoneConstants.State.OFFHOOK && state != PhoneConstants.State.RINGING) { | 
 | 990 |             Intent  intent = new Intent(Intent.ACTION_DIAL, Uri.parse(url)); | 
 | 991 |             intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); | 
 | 992 |             mApp.startActivity(intent); | 
 | 993 |         } | 
 | 994 |     } | 
 | 995 |  | 
 | 996 |     public void call(String callingPackage, String number) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 997 |         callForSubscriber(getPreferredVoiceSubscription(), callingPackage, number); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 998 |     } | 
 | 999 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1000 |     public void callForSubscriber(int subId, String callingPackage, String number) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1001 |         if (DBG) log("call: " + number); | 
 | 1002 |  | 
 | 1003 |         // This is just a wrapper around the ACTION_CALL intent, but we still | 
 | 1004 |         // need to do a permission check since we're calling startActivity() | 
 | 1005 |         // from the context of the phone app. | 
 | 1006 |         enforceCallPermission(); | 
 | 1007 |  | 
 | 1008 |         if (mAppOps.noteOp(AppOpsManager.OP_CALL_PHONE, Binder.getCallingUid(), callingPackage) | 
 | 1009 |                 != AppOpsManager.MODE_ALLOWED) { | 
 | 1010 |             return; | 
 | 1011 |         } | 
 | 1012 |  | 
 | 1013 |         String url = createTelUrl(number); | 
 | 1014 |         if (url == null) { | 
 | 1015 |             return; | 
 | 1016 |         } | 
 | 1017 |  | 
| Wink Saville | 0887461 | 2014-08-31 19:19:58 -0700 | [diff] [blame] | 1018 |         boolean isValid = false; | 
| Narayan Kamath | 1c496c2 | 2015-04-16 14:40:19 +0100 | [diff] [blame] | 1019 |         final List<SubscriptionInfo> slist = getActiveSubscriptionInfoList(); | 
| Wink Saville | 3ab207e | 2014-11-20 13:07:20 -0800 | [diff] [blame] | 1020 |         if (slist != null) { | 
 | 1021 |             for (SubscriptionInfo subInfoRecord : slist) { | 
 | 1022 |                 if (subInfoRecord.getSubscriptionId() == subId) { | 
 | 1023 |                     isValid = true; | 
 | 1024 |                     break; | 
 | 1025 |                 } | 
| Wink Saville | 0887461 | 2014-08-31 19:19:58 -0700 | [diff] [blame] | 1026 |             } | 
 | 1027 |         } | 
 | 1028 |         if (isValid == false) { | 
 | 1029 |             return; | 
 | 1030 |         } | 
 | 1031 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1032 |         Intent intent = new Intent(Intent.ACTION_CALL, Uri.parse(url)); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1033 |         intent.putExtra(SUBSCRIPTION_KEY, subId); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1034 |         intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); | 
 | 1035 |         mApp.startActivity(intent); | 
 | 1036 |     } | 
 | 1037 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1038 |     /** | 
 | 1039 |      * End a call based on call state | 
 | 1040 |      * @return true is a call was ended | 
 | 1041 |      */ | 
 | 1042 |     public boolean endCall() { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1043 |         return endCallForSubscriber(getDefaultSubscription()); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1044 |     } | 
 | 1045 |  | 
 | 1046 |     /** | 
 | 1047 |      * End a call based on the call state of the subId | 
 | 1048 |      * @return true is a call was ended | 
 | 1049 |      */ | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1050 |     public boolean endCallForSubscriber(int subId) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1051 |         enforceCallPermission(); | 
| Stuart Scott | 584921c | 2015-01-15 17:10:34 -0800 | [diff] [blame] | 1052 |         return (Boolean) sendRequest(CMD_END_CALL, null, new Integer(subId)); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1053 |     } | 
 | 1054 |  | 
 | 1055 |     public void answerRingingCall() { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1056 |         answerRingingCallForSubscriber(getDefaultSubscription()); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1057 |     } | 
 | 1058 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1059 |     public void answerRingingCallForSubscriber(int subId) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1060 |         if (DBG) log("answerRingingCall..."); | 
 | 1061 |         // TODO: there should eventually be a separate "ANSWER_PHONE" permission, | 
 | 1062 |         // but that can probably wait till the big TelephonyManager API overhaul. | 
 | 1063 |         // For now, protect this call with the MODIFY_PHONE_STATE permission. | 
 | 1064 |         enforceModifyPermission(); | 
| Stuart Scott | 584921c | 2015-01-15 17:10:34 -0800 | [diff] [blame] | 1065 |         sendRequest(CMD_ANSWER_RINGING_CALL, null, new Integer(subId)); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1066 |     } | 
 | 1067 |  | 
 | 1068 |     /** | 
 | 1069 |      * Make the actual telephony calls to implement answerRingingCall(). | 
 | 1070 |      * This should only be called from the main thread of the Phone app. | 
 | 1071 |      * @see #answerRingingCall | 
 | 1072 |      * | 
 | 1073 |      * TODO: it would be nice to return true if we answered the call, or | 
 | 1074 |      * false if there wasn't actually a ringing incoming call, or some | 
 | 1075 |      * other error occurred.  (In other words, pass back the return value | 
 | 1076 |      * from PhoneUtils.answerCall() or PhoneUtils.answerAndEndActive().) | 
 | 1077 |      * But that would require calling this method via sendRequest() rather | 
 | 1078 |      * than sendRequestAsync(), and right now we don't actually *need* that | 
 | 1079 |      * return value, so let's just return void for now. | 
 | 1080 |      */ | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1081 |     private void answerRingingCallInternal(int subId) { | 
| Wink Saville | 0887461 | 2014-08-31 19:19:58 -0700 | [diff] [blame] | 1082 |         final boolean hasRingingCall = !getPhone(subId).getRingingCall().isIdle(); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1083 |         if (hasRingingCall) { | 
| Wink Saville | 0887461 | 2014-08-31 19:19:58 -0700 | [diff] [blame] | 1084 |             final boolean hasActiveCall = !getPhone(subId).getForegroundCall().isIdle(); | 
 | 1085 |             final boolean hasHoldingCall = !getPhone(subId).getBackgroundCall().isIdle(); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1086 |             if (hasActiveCall && hasHoldingCall) { | 
 | 1087 |                 // Both lines are in use! | 
 | 1088 |                 // TODO: provide a flag to let the caller specify what | 
 | 1089 |                 // policy to use if both lines are in use.  (The current | 
 | 1090 |                 // behavior is hardwired to "answer incoming, end ongoing", | 
 | 1091 |                 // which is how the CALL button is specced to behave.) | 
 | 1092 |                 PhoneUtils.answerAndEndActive(mCM, mCM.getFirstActiveRingingCall()); | 
 | 1093 |                 return; | 
 | 1094 |             } else { | 
 | 1095 |                 // answerCall() will automatically hold the current active | 
 | 1096 |                 // call, if there is one. | 
 | 1097 |                 PhoneUtils.answerCall(mCM.getFirstActiveRingingCall()); | 
 | 1098 |                 return; | 
 | 1099 |             } | 
 | 1100 |         } else { | 
 | 1101 |             // No call was ringing. | 
 | 1102 |             return; | 
 | 1103 |         } | 
 | 1104 |     } | 
 | 1105 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1106 |     /** | 
| Santos Cordon | 5422a8d | 2014-09-12 04:20:56 -0700 | [diff] [blame] | 1107 |      * This method is no longer used and can be removed once TelephonyManager stops referring to it. | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1108 |      */ | 
| Santos Cordon | 5422a8d | 2014-09-12 04:20:56 -0700 | [diff] [blame] | 1109 |     public void silenceRinger() { | 
 | 1110 |         Log.e(LOG_TAG, "silenseRinger not supported"); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1111 |     } | 
 | 1112 |  | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1113 |     @Override | 
 | 1114 |     public boolean isOffhook(String callingPackage) { | 
 | 1115 |         return isOffhookForSubscriber(getDefaultSubscription(), callingPackage); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1116 |     } | 
 | 1117 |  | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1118 |     @Override | 
 | 1119 |     public boolean isOffhookForSubscriber(int subId, String callingPackage) { | 
 | 1120 |         if (!canReadPhoneState(callingPackage, "isOffhookForSubscriber")) { | 
 | 1121 |             return false; | 
 | 1122 |         } | 
 | 1123 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1124 |         final Phone phone = getPhone(subId); | 
 | 1125 |         if (phone != null) { | 
 | 1126 |             return (phone.getState() == PhoneConstants.State.OFFHOOK); | 
 | 1127 |         } else { | 
 | 1128 |             return false; | 
 | 1129 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1130 |     } | 
 | 1131 |  | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1132 |     @Override | 
 | 1133 |     public boolean isRinging(String callingPackage) { | 
 | 1134 |         return (isRingingForSubscriber(getDefaultSubscription(), callingPackage)); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1135 |     } | 
 | 1136 |  | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1137 |     @Override | 
 | 1138 |     public boolean isRingingForSubscriber(int subId, String callingPackage) { | 
 | 1139 |         if (!canReadPhoneState(callingPackage, "isRingingForSubscriber")) { | 
 | 1140 |             return false; | 
 | 1141 |         } | 
 | 1142 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1143 |         final Phone phone = getPhone(subId); | 
 | 1144 |         if (phone != null) { | 
 | 1145 |             return (phone.getState() == PhoneConstants.State.RINGING); | 
 | 1146 |         } else { | 
 | 1147 |             return false; | 
 | 1148 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1149 |     } | 
 | 1150 |  | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1151 |     @Override | 
 | 1152 |     public boolean isIdle(String callingPackage) { | 
 | 1153 |         return isIdleForSubscriber(getDefaultSubscription(), callingPackage); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1154 |     } | 
 | 1155 |  | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1156 |     @Override | 
 | 1157 |     public boolean isIdleForSubscriber(int subId, String callingPackage) { | 
 | 1158 |         if (!canReadPhoneState(callingPackage, "isIdleForSubscriber")) { | 
 | 1159 |             return false; | 
 | 1160 |         } | 
 | 1161 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1162 |         final Phone phone = getPhone(subId); | 
 | 1163 |         if (phone != null) { | 
 | 1164 |             return (phone.getState() == PhoneConstants.State.IDLE); | 
 | 1165 |         } else { | 
 | 1166 |             return false; | 
 | 1167 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1168 |     } | 
 | 1169 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1170 |     public boolean supplyPin(String pin) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1171 |         return supplyPinForSubscriber(getDefaultSubscription(), pin); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1172 |     } | 
 | 1173 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1174 |     public boolean supplyPinForSubscriber(int subId, String pin) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1175 |         int [] resultArray = supplyPinReportResultForSubscriber(subId, pin); | 
| Wink Saville | 9de0f75 | 2013-10-22 19:04:03 -0700 | [diff] [blame] | 1176 |         return (resultArray[0] == PhoneConstants.PIN_RESULT_SUCCESS) ? true : false; | 
 | 1177 |     } | 
 | 1178 |  | 
 | 1179 |     public boolean supplyPuk(String puk, String pin) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1180 |         return supplyPukForSubscriber(getDefaultSubscription(), puk, pin); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1181 |     } | 
 | 1182 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1183 |     public boolean supplyPukForSubscriber(int subId, String puk, String pin) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1184 |         int [] resultArray = supplyPukReportResultForSubscriber(subId, puk, pin); | 
| Wink Saville | 9de0f75 | 2013-10-22 19:04:03 -0700 | [diff] [blame] | 1185 |         return (resultArray[0] == PhoneConstants.PIN_RESULT_SUCCESS) ? true : false; | 
 | 1186 |     } | 
 | 1187 |  | 
 | 1188 |     /** {@hide} */ | 
 | 1189 |     public int[] supplyPinReportResult(String pin) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1190 |         return supplyPinReportResultForSubscriber(getDefaultSubscription(), pin); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1191 |     } | 
 | 1192 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1193 |     public int[] supplyPinReportResultForSubscriber(int subId, String pin) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1194 |         enforceModifyPermission(); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1195 |         final UnlockSim checkSimPin = new UnlockSim(getPhone(subId).getIccCard()); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1196 |         checkSimPin.start(); | 
 | 1197 |         return checkSimPin.unlockSim(null, pin); | 
 | 1198 |     } | 
 | 1199 |  | 
| Wink Saville | 9de0f75 | 2013-10-22 19:04:03 -0700 | [diff] [blame] | 1200 |     /** {@hide} */ | 
 | 1201 |     public int[] supplyPukReportResult(String puk, String pin) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1202 |         return supplyPukReportResultForSubscriber(getDefaultSubscription(), puk, pin); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1203 |     } | 
 | 1204 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1205 |     public int[] supplyPukReportResultForSubscriber(int subId, String puk, String pin) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1206 |         enforceModifyPermission(); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1207 |         final UnlockSim checkSimPuk = new UnlockSim(getPhone(subId).getIccCard()); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1208 |         checkSimPuk.start(); | 
 | 1209 |         return checkSimPuk.unlockSim(puk, pin); | 
 | 1210 |     } | 
 | 1211 |  | 
 | 1212 |     /** | 
| Wink Saville | 9de0f75 | 2013-10-22 19:04:03 -0700 | [diff] [blame] | 1213 |      * Helper thread to turn async call to SimCard#supplyPin into | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1214 |      * a synchronous one. | 
 | 1215 |      */ | 
 | 1216 |     private static class UnlockSim extends Thread { | 
 | 1217 |  | 
 | 1218 |         private final IccCard mSimCard; | 
 | 1219 |  | 
 | 1220 |         private boolean mDone = false; | 
| Wink Saville | 9de0f75 | 2013-10-22 19:04:03 -0700 | [diff] [blame] | 1221 |         private int mResult = PhoneConstants.PIN_GENERAL_FAILURE; | 
 | 1222 |         private int mRetryCount = -1; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1223 |  | 
 | 1224 |         // For replies from SimCard interface | 
 | 1225 |         private Handler mHandler; | 
 | 1226 |  | 
 | 1227 |         // For async handler to identify request type | 
 | 1228 |         private static final int SUPPLY_PIN_COMPLETE = 100; | 
 | 1229 |  | 
 | 1230 |         public UnlockSim(IccCard simCard) { | 
 | 1231 |             mSimCard = simCard; | 
 | 1232 |         } | 
 | 1233 |  | 
 | 1234 |         @Override | 
 | 1235 |         public void run() { | 
 | 1236 |             Looper.prepare(); | 
 | 1237 |             synchronized (UnlockSim.this) { | 
 | 1238 |                 mHandler = new Handler() { | 
 | 1239 |                     @Override | 
 | 1240 |                     public void handleMessage(Message msg) { | 
 | 1241 |                         AsyncResult ar = (AsyncResult) msg.obj; | 
 | 1242 |                         switch (msg.what) { | 
 | 1243 |                             case SUPPLY_PIN_COMPLETE: | 
 | 1244 |                                 Log.d(LOG_TAG, "SUPPLY_PIN_COMPLETE"); | 
 | 1245 |                                 synchronized (UnlockSim.this) { | 
| Wink Saville | 9de0f75 | 2013-10-22 19:04:03 -0700 | [diff] [blame] | 1246 |                                     mRetryCount = msg.arg1; | 
 | 1247 |                                     if (ar.exception != null) { | 
 | 1248 |                                         if (ar.exception instanceof CommandException && | 
 | 1249 |                                                 ((CommandException)(ar.exception)).getCommandError() | 
 | 1250 |                                                 == CommandException.Error.PASSWORD_INCORRECT) { | 
 | 1251 |                                             mResult = PhoneConstants.PIN_PASSWORD_INCORRECT; | 
 | 1252 |                                         } else { | 
 | 1253 |                                             mResult = PhoneConstants.PIN_GENERAL_FAILURE; | 
 | 1254 |                                         } | 
 | 1255 |                                     } else { | 
 | 1256 |                                         mResult = PhoneConstants.PIN_RESULT_SUCCESS; | 
 | 1257 |                                     } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1258 |                                     mDone = true; | 
 | 1259 |                                     UnlockSim.this.notifyAll(); | 
 | 1260 |                                 } | 
 | 1261 |                                 break; | 
 | 1262 |                         } | 
 | 1263 |                     } | 
 | 1264 |                 }; | 
 | 1265 |                 UnlockSim.this.notifyAll(); | 
 | 1266 |             } | 
 | 1267 |             Looper.loop(); | 
 | 1268 |         } | 
 | 1269 |  | 
 | 1270 |         /* | 
 | 1271 |          * Use PIN or PUK to unlock SIM card | 
 | 1272 |          * | 
 | 1273 |          * If PUK is null, unlock SIM card with PIN | 
 | 1274 |          * | 
 | 1275 |          * If PUK is not null, unlock SIM card with PUK and set PIN code | 
 | 1276 |          */ | 
| Wink Saville | 9de0f75 | 2013-10-22 19:04:03 -0700 | [diff] [blame] | 1277 |         synchronized int[] unlockSim(String puk, String pin) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1278 |  | 
 | 1279 |             while (mHandler == null) { | 
 | 1280 |                 try { | 
 | 1281 |                     wait(); | 
 | 1282 |                 } catch (InterruptedException e) { | 
 | 1283 |                     Thread.currentThread().interrupt(); | 
 | 1284 |                 } | 
 | 1285 |             } | 
 | 1286 |             Message callback = Message.obtain(mHandler, SUPPLY_PIN_COMPLETE); | 
 | 1287 |  | 
 | 1288 |             if (puk == null) { | 
 | 1289 |                 mSimCard.supplyPin(pin, callback); | 
 | 1290 |             } else { | 
 | 1291 |                 mSimCard.supplyPuk(puk, pin, callback); | 
 | 1292 |             } | 
 | 1293 |  | 
 | 1294 |             while (!mDone) { | 
 | 1295 |                 try { | 
 | 1296 |                     Log.d(LOG_TAG, "wait for done"); | 
 | 1297 |                     wait(); | 
 | 1298 |                 } catch (InterruptedException e) { | 
 | 1299 |                     // Restore the interrupted status | 
 | 1300 |                     Thread.currentThread().interrupt(); | 
 | 1301 |                 } | 
 | 1302 |             } | 
 | 1303 |             Log.d(LOG_TAG, "done"); | 
| Wink Saville | 9de0f75 | 2013-10-22 19:04:03 -0700 | [diff] [blame] | 1304 |             int[] resultArray = new int[2]; | 
 | 1305 |             resultArray[0] = mResult; | 
 | 1306 |             resultArray[1] = mRetryCount; | 
 | 1307 |             return resultArray; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1308 |         } | 
 | 1309 |     } | 
 | 1310 |  | 
 | 1311 |     public void updateServiceLocation() { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1312 |         updateServiceLocationForSubscriber(getDefaultSubscription()); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1313 |  | 
 | 1314 |     } | 
 | 1315 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1316 |     public void updateServiceLocationForSubscriber(int subId) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1317 |         // No permission check needed here: this call is harmless, and it's | 
 | 1318 |         // needed for the ServiceState.requestStateUpdate() call (which is | 
 | 1319 |         // already intentionally exposed to 3rd parties.) | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1320 |         final Phone phone = getPhone(subId); | 
 | 1321 |         if (phone != null) { | 
 | 1322 |             phone.updateServiceLocation(); | 
 | 1323 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1324 |     } | 
 | 1325 |  | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1326 |     @Override | 
 | 1327 |     public boolean isRadioOn(String callingPackage) { | 
 | 1328 |         return isRadioOnForSubscriber(getDefaultSubscription(), callingPackage); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1329 |     } | 
 | 1330 |  | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1331 |     @Override | 
 | 1332 |     public boolean isRadioOnForSubscriber(int subId, String callingPackage) { | 
 | 1333 |         if (!canReadPhoneState(callingPackage, "isRadioOnForSubscriber")) { | 
 | 1334 |             return false; | 
 | 1335 |         } | 
 | 1336 |         return isRadioOnForSubscriber(subId); | 
 | 1337 |     } | 
 | 1338 |  | 
 | 1339 |     private boolean isRadioOnForSubscriber(int subId) { | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1340 |         final Phone phone = getPhone(subId); | 
 | 1341 |         if (phone != null) { | 
 | 1342 |             return phone.getServiceState().getState() != ServiceState.STATE_POWER_OFF; | 
 | 1343 |         } else { | 
 | 1344 |             return false; | 
 | 1345 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1346 |     } | 
 | 1347 |  | 
 | 1348 |     public void toggleRadioOnOff() { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1349 |         toggleRadioOnOffForSubscriber(getDefaultSubscription()); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1350 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1351 |     } | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1352 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1353 |     public void toggleRadioOnOffForSubscriber(int subId) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1354 |         enforceModifyPermission(); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1355 |         final Phone phone = getPhone(subId); | 
 | 1356 |         if (phone != null) { | 
 | 1357 |             phone.setRadioPower(!isRadioOnForSubscriber(subId)); | 
 | 1358 |         } | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1359 |     } | 
 | 1360 |  | 
 | 1361 |     public boolean setRadio(boolean turnOn) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1362 |         return setRadioForSubscriber(getDefaultSubscription(), turnOn); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1363 |     } | 
 | 1364 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1365 |     public boolean setRadioForSubscriber(int subId, boolean turnOn) { | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1366 |         enforceModifyPermission(); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1367 |         final Phone phone = getPhone(subId); | 
 | 1368 |         if (phone == null) { | 
 | 1369 |             return false; | 
 | 1370 |         } | 
 | 1371 |         if ((phone.getServiceState().getState() != | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1372 |                 ServiceState.STATE_POWER_OFF) != turnOn) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1373 |             toggleRadioOnOffForSubscriber(subId); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1374 |         } | 
 | 1375 |         return true; | 
 | 1376 |     } | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1377 |  | 
| Naveen Kalla | 1fd79bd | 2014-08-08 00:48:59 -0700 | [diff] [blame] | 1378 |     public boolean needMobileRadioShutdown() { | 
 | 1379 |         /* | 
 | 1380 |          * If any of the Radios are available, it will need to be | 
 | 1381 |          * shutdown. So return true if any Radio is available. | 
 | 1382 |          */ | 
 | 1383 |         for (int i = 0; i < TelephonyManager.getDefault().getPhoneCount(); i++) { | 
 | 1384 |             Phone phone = PhoneFactory.getPhone(i); | 
 | 1385 |             if (phone != null && phone.isRadioAvailable()) return true; | 
 | 1386 |         } | 
 | 1387 |         logv(TelephonyManager.getDefault().getPhoneCount() + " Phones are shutdown."); | 
 | 1388 |         return false; | 
 | 1389 |     } | 
 | 1390 |  | 
 | 1391 |     public void shutdownMobileRadios() { | 
 | 1392 |         for (int i = 0; i < TelephonyManager.getDefault().getPhoneCount(); i++) { | 
 | 1393 |             logv("Shutting down Phone " + i); | 
 | 1394 |             shutdownRadioUsingPhoneId(i); | 
 | 1395 |         } | 
 | 1396 |     } | 
 | 1397 |  | 
 | 1398 |     private void shutdownRadioUsingPhoneId(int phoneId) { | 
 | 1399 |         enforceModifyPermission(); | 
 | 1400 |         Phone phone = PhoneFactory.getPhone(phoneId); | 
 | 1401 |         if (phone != null && phone.isRadioAvailable()) { | 
 | 1402 |             phone.shutdownRadio(); | 
 | 1403 |         } | 
 | 1404 |     } | 
 | 1405 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1406 |     public boolean setRadioPower(boolean turnOn) { | 
| Wei Liu | 9ae2a06 | 2016-08-08 11:09:34 -0700 | [diff] [blame] | 1407 |         final Phone defaultPhone = PhoneFactory.getDefaultPhone(); | 
 | 1408 |         if (defaultPhone != null) { | 
 | 1409 |             defaultPhone.setRadioPower(turnOn); | 
 | 1410 |             return true; | 
 | 1411 |         } else { | 
 | 1412 |             loge("There's no default phone."); | 
 | 1413 |             return false; | 
 | 1414 |         } | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1415 |     } | 
 | 1416 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1417 |     public boolean setRadioPowerForSubscriber(int subId, boolean turnOn) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1418 |         enforceModifyPermission(); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1419 |         final Phone phone = getPhone(subId); | 
 | 1420 |         if (phone != null) { | 
 | 1421 |             phone.setRadioPower(turnOn); | 
 | 1422 |             return true; | 
 | 1423 |         } else { | 
 | 1424 |             return false; | 
 | 1425 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1426 |     } | 
 | 1427 |  | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1428 |     // FIXME: subId version needed | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1429 |     @Override | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1430 |     public boolean enableDataConnectivity() { | 
 | 1431 |         enforceModifyPermission(); | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 1432 |         int subId = mSubscriptionController.getDefaultDataSubId(); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1433 |         final Phone phone = getPhone(subId); | 
 | 1434 |         if (phone != null) { | 
 | 1435 |             phone.setDataEnabled(true); | 
 | 1436 |             return true; | 
 | 1437 |         } else { | 
 | 1438 |             return false; | 
 | 1439 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1440 |     } | 
 | 1441 |  | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1442 |     // FIXME: subId version needed | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1443 |     @Override | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1444 |     public boolean disableDataConnectivity() { | 
 | 1445 |         enforceModifyPermission(); | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 1446 |         int subId = mSubscriptionController.getDefaultDataSubId(); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1447 |         final Phone phone = getPhone(subId); | 
 | 1448 |         if (phone != null) { | 
 | 1449 |             phone.setDataEnabled(false); | 
 | 1450 |             return true; | 
 | 1451 |         } else { | 
 | 1452 |             return false; | 
 | 1453 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1454 |     } | 
 | 1455 |  | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1456 |     // FIXME: subId version needed | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1457 |     @Override | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1458 |     public boolean isDataConnectivityPossible() { | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 1459 |         int subId = mSubscriptionController.getDefaultDataSubId(); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1460 |         final Phone phone = getPhone(subId); | 
 | 1461 |         if (phone != null) { | 
 | 1462 |             return phone.isDataConnectivityPossible(); | 
 | 1463 |         } else { | 
 | 1464 |             return false; | 
 | 1465 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1466 |     } | 
 | 1467 |  | 
 | 1468 |     public boolean handlePinMmi(String dialString) { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1469 |         return handlePinMmiForSubscriber(getDefaultSubscription(), dialString); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1470 |     } | 
 | 1471 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1472 |     public boolean handlePinMmiForSubscriber(int subId, String dialString) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1473 |         enforceModifyPermission(); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1474 |         if (!SubscriptionManager.isValidSubscriptionId(subId)) { | 
 | 1475 |             return false; | 
 | 1476 |         } | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1477 |         return (Boolean) sendRequest(CMD_HANDLE_PIN_MMI, dialString, subId); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1478 |     } | 
 | 1479 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1480 |     public int getCallState() { | 
| Shishir Agrawal | a9f3218 | 2016-04-12 12:00:16 -0700 | [diff] [blame] | 1481 |         return getCallStateForSlot(getSlotForDefaultSubscription()); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1482 |     } | 
 | 1483 |  | 
| Shishir Agrawal | a9f3218 | 2016-04-12 12:00:16 -0700 | [diff] [blame] | 1484 |     public int getCallStateForSlot(int slotId) { | 
 | 1485 |         Phone phone = PhoneFactory.getPhone(slotId); | 
 | 1486 |         return phone == null ? TelephonyManager.CALL_STATE_IDLE : | 
 | 1487 |             DefaultPhoneNotifier.convertCallState(phone.getState()); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1488 |     } | 
 | 1489 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1490 |     @Override | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1491 |     public int getDataState() { | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 1492 |         Phone phone = getPhone(mSubscriptionController.getDefaultDataSubId()); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1493 |         if (phone != null) { | 
 | 1494 |             return DefaultPhoneNotifier.convertDataState(phone.getDataConnectionState()); | 
 | 1495 |         } else { | 
 | 1496 |             return DefaultPhoneNotifier.convertDataState(PhoneConstants.DataState.DISCONNECTED); | 
 | 1497 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1498 |     } | 
 | 1499 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1500 |     @Override | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1501 |     public int getDataActivity() { | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 1502 |         Phone phone = getPhone(mSubscriptionController.getDefaultDataSubId()); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1503 |         if (phone != null) { | 
 | 1504 |             return DefaultPhoneNotifier.convertDataActivityState(phone.getDataActivityState()); | 
 | 1505 |         } else { | 
 | 1506 |             return TelephonyManager.DATA_ACTIVITY_NONE; | 
 | 1507 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1508 |     } | 
 | 1509 |  | 
 | 1510 |     @Override | 
| Svetoslav | 64fad26 | 2015-04-14 14:35:21 -0700 | [diff] [blame] | 1511 |     public Bundle getCellLocation(String callingPackage) { | 
 | 1512 |         enforceFineOrCoarseLocationPermission("getCellLocation"); | 
 | 1513 |  | 
 | 1514 |         // OP_COARSE_LOCATION controls both fine and coarse location. | 
 | 1515 |         if (mAppOps.noteOp(AppOpsManager.OP_COARSE_LOCATION, Binder.getCallingUid(), | 
 | 1516 |                 callingPackage) != AppOpsManager.MODE_ALLOWED) { | 
| Amit Mahajan | 9acc70d | 2015-06-02 13:17:33 -0700 | [diff] [blame] | 1517 |             log("getCellLocation: returning null; mode != allowed"); | 
| Svetoslav | 64fad26 | 2015-04-14 14:35:21 -0700 | [diff] [blame] | 1518 |             return null; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1519 |         } | 
 | 1520 |  | 
| Sanket Padawe | ee13a9b | 2016-03-08 17:30:28 -0800 | [diff] [blame] | 1521 |         if (checkIfCallerIsSelfOrForegroundUser() || | 
 | 1522 |                 checkCallerInteractAcrossUsersFull()) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1523 |             if (DBG_LOC) log("getCellLocation: is active user"); | 
 | 1524 |             Bundle data = new Bundle(); | 
| Legler Wu | 2c01cdf | 2014-12-08 19:00:59 +0800 | [diff] [blame] | 1525 |             Phone phone = getPhone(mSubscriptionController.getDefaultDataSubId()); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1526 |             if (phone == null) { | 
 | 1527 |                 return null; | 
 | 1528 |             } | 
| Legler Wu | 2c01cdf | 2014-12-08 19:00:59 +0800 | [diff] [blame] | 1529 |             phone.getCellLocation().fillInNotifierBundle(data); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1530 |             return data; | 
 | 1531 |         } else { | 
| Amit Mahajan | 9acc70d | 2015-06-02 13:17:33 -0700 | [diff] [blame] | 1532 |             log("getCellLocation: suppress non-active user"); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1533 |             return null; | 
 | 1534 |         } | 
 | 1535 |     } | 
 | 1536 |  | 
| Svetoslav | 64fad26 | 2015-04-14 14:35:21 -0700 | [diff] [blame] | 1537 |     private void enforceFineOrCoarseLocationPermission(String message) { | 
 | 1538 |         try { | 
 | 1539 |             mApp.enforceCallingOrSelfPermission( | 
 | 1540 |                     android.Manifest.permission.ACCESS_FINE_LOCATION, null); | 
 | 1541 |         } catch (SecurityException e) { | 
 | 1542 |             // If we have ACCESS_FINE_LOCATION permission, skip the check for ACCESS_COARSE_LOCATION | 
 | 1543 |             // A failure should throw the SecurityException from ACCESS_COARSE_LOCATION since this | 
 | 1544 |             // is the weaker precondition | 
 | 1545 |             mApp.enforceCallingOrSelfPermission( | 
 | 1546 |                     android.Manifest.permission.ACCESS_COARSE_LOCATION, message); | 
 | 1547 |         } | 
 | 1548 |     } | 
 | 1549 |  | 
 | 1550 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1551 |     @Override | 
 | 1552 |     public void enableLocationUpdates() { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1553 |         enableLocationUpdatesForSubscriber(getDefaultSubscription()); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1554 |     } | 
 | 1555 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1556 |     @Override | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1557 |     public void enableLocationUpdatesForSubscriber(int subId) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1558 |         mApp.enforceCallingOrSelfPermission( | 
 | 1559 |                 android.Manifest.permission.CONTROL_LOCATION_UPDATES, null); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1560 |         final Phone phone = getPhone(subId); | 
 | 1561 |         if (phone != null) { | 
 | 1562 |             phone.enableLocationUpdates(); | 
 | 1563 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1564 |     } | 
 | 1565 |  | 
 | 1566 |     @Override | 
 | 1567 |     public void disableLocationUpdates() { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1568 |         disableLocationUpdatesForSubscriber(getDefaultSubscription()); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1569 |     } | 
 | 1570 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1571 |     @Override | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1572 |     public void disableLocationUpdatesForSubscriber(int subId) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1573 |         mApp.enforceCallingOrSelfPermission( | 
 | 1574 |                 android.Manifest.permission.CONTROL_LOCATION_UPDATES, null); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1575 |         final Phone phone = getPhone(subId); | 
 | 1576 |         if (phone != null) { | 
 | 1577 |             phone.disableLocationUpdates(); | 
 | 1578 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1579 |     } | 
 | 1580 |  | 
 | 1581 |     @Override | 
 | 1582 |     @SuppressWarnings("unchecked") | 
 | 1583 |     public List<NeighboringCellInfo> getNeighboringCellInfo(String callingPackage) { | 
| Svetoslav | 64fad26 | 2015-04-14 14:35:21 -0700 | [diff] [blame] | 1584 |         enforceFineOrCoarseLocationPermission("getNeighboringCellInfo"); | 
 | 1585 |  | 
 | 1586 |         // OP_COARSE_LOCATION controls both fine and coarse location. | 
 | 1587 |         if (mAppOps.noteOp(AppOpsManager.OP_COARSE_LOCATION, Binder.getCallingUid(), | 
 | 1588 |                 callingPackage) != AppOpsManager.MODE_ALLOWED) { | 
 | 1589 |             return null; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1590 |         } | 
 | 1591 |  | 
 | 1592 |         if (mAppOps.noteOp(AppOpsManager.OP_NEIGHBORING_CELLS, Binder.getCallingUid(), | 
 | 1593 |                 callingPackage) != AppOpsManager.MODE_ALLOWED) { | 
 | 1594 |             return null; | 
 | 1595 |         } | 
| Svetoslav | 64fad26 | 2015-04-14 14:35:21 -0700 | [diff] [blame] | 1596 |  | 
| Sanket Padawe | ee13a9b | 2016-03-08 17:30:28 -0800 | [diff] [blame] | 1597 |         if (checkIfCallerIsSelfOrForegroundUser() || | 
 | 1598 |                 checkCallerInteractAcrossUsersFull()) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1599 |             if (DBG_LOC) log("getNeighboringCellInfo: is active user"); | 
 | 1600 |  | 
 | 1601 |             ArrayList<NeighboringCellInfo> cells = null; | 
 | 1602 |  | 
 | 1603 |             try { | 
 | 1604 |                 cells = (ArrayList<NeighboringCellInfo>) sendRequest( | 
| Sanket Padawe | 56e75a3 | 2016-02-08 12:18:19 -0800 | [diff] [blame] | 1605 |                         CMD_HANDLE_NEIGHBORING_CELL, null, | 
 | 1606 |                         SubscriptionManager.INVALID_SUBSCRIPTION_ID); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1607 |             } catch (RuntimeException e) { | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1608 |                 Log.e(LOG_TAG, "getNeighboringCellInfo " + e); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1609 |             } | 
 | 1610 |             return cells; | 
 | 1611 |         } else { | 
 | 1612 |             if (DBG_LOC) log("getNeighboringCellInfo: suppress non-active user"); | 
 | 1613 |             return null; | 
 | 1614 |         } | 
 | 1615 |     } | 
 | 1616 |  | 
 | 1617 |  | 
 | 1618 |     @Override | 
| Svetoslav | 64fad26 | 2015-04-14 14:35:21 -0700 | [diff] [blame] | 1619 |     public List<CellInfo> getAllCellInfo(String callingPackage) { | 
 | 1620 |         enforceFineOrCoarseLocationPermission("getAllCellInfo"); | 
 | 1621 |  | 
 | 1622 |         // OP_COARSE_LOCATION controls both fine and coarse location. | 
 | 1623 |         if (mAppOps.noteOp(AppOpsManager.OP_COARSE_LOCATION, Binder.getCallingUid(), | 
 | 1624 |                 callingPackage) != AppOpsManager.MODE_ALLOWED) { | 
 | 1625 |             return null; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1626 |         } | 
 | 1627 |  | 
| Sanket Padawe | ee13a9b | 2016-03-08 17:30:28 -0800 | [diff] [blame] | 1628 |         if (checkIfCallerIsSelfOrForegroundUser() || | 
 | 1629 |                 checkCallerInteractAcrossUsersFull()) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1630 |             if (DBG_LOC) log("getAllCellInfo: is active user"); | 
| Legler Wu | 2c01cdf | 2014-12-08 19:00:59 +0800 | [diff] [blame] | 1631 |             List<CellInfo> cellInfos = new ArrayList<CellInfo>(); | 
 | 1632 |             for (Phone phone : PhoneFactory.getPhones()) { | 
| Robert Greenwalt | d9eb7da | 2015-08-19 12:18:12 -0700 | [diff] [blame] | 1633 |                 final List<CellInfo> info = phone.getAllCellInfo(); | 
 | 1634 |                 if (info != null) cellInfos.addAll(phone.getAllCellInfo()); | 
| Legler Wu | 2c01cdf | 2014-12-08 19:00:59 +0800 | [diff] [blame] | 1635 |             } | 
 | 1636 |             return cellInfos; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1637 |         } else { | 
 | 1638 |             if (DBG_LOC) log("getAllCellInfo: suppress non-active user"); | 
 | 1639 |             return null; | 
 | 1640 |         } | 
 | 1641 |     } | 
 | 1642 |  | 
| Sailesh Nepal | bd76e4e | 2013-10-27 13:59:44 -0700 | [diff] [blame] | 1643 |     @Override | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1644 |     public void setCellInfoListRate(int rateInMillis) { | 
| Jack Yu | 3128d9e | 2017-01-16 10:15:34 -0800 | [diff] [blame] | 1645 |         enforceModifyPermission(); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1646 |         mPhone.setCellInfoListRate(rateInMillis); | 
 | 1647 |     } | 
 | 1648 |  | 
| Shishir Agrawal | a9f3218 | 2016-04-12 12:00:16 -0700 | [diff] [blame] | 1649 |     @Override | 
 | 1650 |     public String getImeiForSlot(int slotId, String callingPackage) { | 
 | 1651 |       if (!canReadPhoneState(callingPackage, "getImeiForSlot")) { | 
 | 1652 |           return null; | 
 | 1653 |       } | 
 | 1654 |       Phone phone = PhoneFactory.getPhone(slotId); | 
 | 1655 |       return phone == null ? null : phone.getImei(); | 
 | 1656 |     } | 
 | 1657 |  | 
 | 1658 |     @Override | 
 | 1659 |     public String getDeviceSoftwareVersionForSlot(int slotId, String callingPackage) { | 
 | 1660 |       if (!canReadPhoneState(callingPackage, "getDeviceSoftwareVersionForSlot")) { | 
 | 1661 |           return null; | 
 | 1662 |       } | 
 | 1663 |       Phone phone = PhoneFactory.getPhone(slotId); | 
 | 1664 |       return phone == null ? null : phone.getDeviceSvn(); | 
 | 1665 |     } | 
 | 1666 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1667 |     // | 
 | 1668 |     // Internal helper methods. | 
 | 1669 |     // | 
 | 1670 |  | 
| Sanket Padawe | ee13a9b | 2016-03-08 17:30:28 -0800 | [diff] [blame] | 1671 |     /** | 
 | 1672 |      * Returns true if the caller holds INTERACT_ACROSS_USERS_FULL. | 
 | 1673 |      */ | 
 | 1674 |     private boolean checkCallerInteractAcrossUsersFull() { | 
 | 1675 |         return mPhone.getContext().checkCallingOrSelfPermission( | 
 | 1676 |                 android.Manifest.permission.INTERACT_ACROSS_USERS_FULL) | 
 | 1677 |                 == PackageManager.PERMISSION_GRANTED; | 
 | 1678 |     } | 
 | 1679 |  | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 1680 |     private static boolean checkIfCallerIsSelfOrForegroundUser() { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1681 |         boolean ok; | 
 | 1682 |  | 
 | 1683 |         boolean self = Binder.getCallingUid() == Process.myUid(); | 
 | 1684 |         if (!self) { | 
 | 1685 |             // Get the caller's user id then clear the calling identity | 
 | 1686 |             // which will be restored in the finally clause. | 
 | 1687 |             int callingUser = UserHandle.getCallingUserId(); | 
 | 1688 |             long ident = Binder.clearCallingIdentity(); | 
 | 1689 |  | 
 | 1690 |             try { | 
 | 1691 |                 // With calling identity cleared the current user is the foreground user. | 
 | 1692 |                 int foregroundUser = ActivityManager.getCurrentUser(); | 
 | 1693 |                 ok = (foregroundUser == callingUser); | 
 | 1694 |                 if (DBG_LOC) { | 
 | 1695 |                     log("checkIfCallerIsSelfOrForegoundUser: foregroundUser=" + foregroundUser | 
 | 1696 |                             + " callingUser=" + callingUser + " ok=" + ok); | 
 | 1697 |                 } | 
 | 1698 |             } catch (Exception ex) { | 
 | 1699 |                 if (DBG_LOC) loge("checkIfCallerIsSelfOrForegoundUser: Exception ex=" + ex); | 
 | 1700 |                 ok = false; | 
 | 1701 |             } finally { | 
 | 1702 |                 Binder.restoreCallingIdentity(ident); | 
 | 1703 |             } | 
 | 1704 |         } else { | 
 | 1705 |             if (DBG_LOC) log("checkIfCallerIsSelfOrForegoundUser: is self"); | 
 | 1706 |             ok = true; | 
 | 1707 |         } | 
 | 1708 |         if (DBG_LOC) log("checkIfCallerIsSelfOrForegoundUser: ret=" + ok); | 
 | 1709 |         return ok; | 
 | 1710 |     } | 
 | 1711 |  | 
 | 1712 |     /** | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1713 |      * Make sure the caller has the MODIFY_PHONE_STATE permission. | 
 | 1714 |      * | 
 | 1715 |      * @throws SecurityException if the caller does not have the required permission | 
 | 1716 |      */ | 
 | 1717 |     private void enforceModifyPermission() { | 
 | 1718 |         mApp.enforceCallingOrSelfPermission(android.Manifest.permission.MODIFY_PHONE_STATE, null); | 
 | 1719 |     } | 
 | 1720 |  | 
 | 1721 |     /** | 
| Junda Liu | a2e3601 | 2014-07-09 18:30:01 -0700 | [diff] [blame] | 1722 |      * Make sure either system app or the caller has carrier privilege. | 
 | 1723 |      * | 
 | 1724 |      * @throws SecurityException if the caller does not have the required permission/privilege | 
 | 1725 |      */ | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 1726 |     private void enforceModifyPermissionOrCarrierPrivilege(int subId) { | 
| Shishir Agrawal | f1ac4c9 | 2014-07-14 13:54:28 -0700 | [diff] [blame] | 1727 |         int permission = mApp.checkCallingOrSelfPermission( | 
 | 1728 |                 android.Manifest.permission.MODIFY_PHONE_STATE); | 
 | 1729 |         if (permission == PackageManager.PERMISSION_GRANTED) { | 
 | 1730 |             return; | 
 | 1731 |         } | 
 | 1732 |  | 
 | 1733 |         log("No modify permission, check carrier privilege next."); | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 1734 |         enforceCarrierPrivilege(subId); | 
| Junda Liu | a2e3601 | 2014-07-09 18:30:01 -0700 | [diff] [blame] | 1735 |     } | 
 | 1736 |  | 
 | 1737 |     /** | 
 | 1738 |      * Make sure the caller has carrier privilege. | 
 | 1739 |      * | 
 | 1740 |      * @throws SecurityException if the caller does not have the required permission | 
 | 1741 |      */ | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 1742 |     private void enforceCarrierPrivilege(int subId) { | 
 | 1743 |         if (getCarrierPrivilegeStatus(subId) != | 
 | 1744 |                     TelephonyManager.CARRIER_PRIVILEGE_STATUS_HAS_ACCESS) { | 
| Shishir Agrawal | f1ac4c9 | 2014-07-14 13:54:28 -0700 | [diff] [blame] | 1745 |             loge("No Carrier Privilege."); | 
 | 1746 |             throw new SecurityException("No Carrier Privilege."); | 
| Junda Liu | a2e3601 | 2014-07-09 18:30:01 -0700 | [diff] [blame] | 1747 |         } | 
 | 1748 |     } | 
 | 1749 |  | 
 | 1750 |     /** | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1751 |      * Make sure the caller has the CALL_PHONE permission. | 
 | 1752 |      * | 
 | 1753 |      * @throws SecurityException if the caller does not have the required permission | 
 | 1754 |      */ | 
 | 1755 |     private void enforceCallPermission() { | 
 | 1756 |         mApp.enforceCallingOrSelfPermission(android.Manifest.permission.CALL_PHONE, null); | 
 | 1757 |     } | 
 | 1758 |  | 
| Stuart Scott | 8eef64f | 2015-04-08 15:13:54 -0700 | [diff] [blame] | 1759 |     private void enforceConnectivityInternalPermission() { | 
 | 1760 |         mApp.enforceCallingOrSelfPermission( | 
 | 1761 |                 android.Manifest.permission.CONNECTIVITY_INTERNAL, | 
 | 1762 |                 "ConnectivityService"); | 
 | 1763 |     } | 
 | 1764 |  | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1765 |     private String createTelUrl(String number) { | 
 | 1766 |         if (TextUtils.isEmpty(number)) { | 
 | 1767 |             return null; | 
 | 1768 |         } | 
 | 1769 |  | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 1770 |         return "tel:" + number; | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1771 |     } | 
 | 1772 |  | 
| Ihab Awad | f9e9273 | 2013-12-05 18:02:52 -0800 | [diff] [blame] | 1773 |     private static void log(String msg) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1774 |         Log.d(LOG_TAG, "[PhoneIntfMgr] " + msg); | 
 | 1775 |     } | 
 | 1776 |  | 
| Naveen Kalla | 1fd79bd | 2014-08-08 00:48:59 -0700 | [diff] [blame] | 1777 |     private static void logv(String msg) { | 
 | 1778 |         Log.v(LOG_TAG, "[PhoneIntfMgr] " + msg); | 
 | 1779 |     } | 
 | 1780 |  | 
| Ihab Awad | f9e9273 | 2013-12-05 18:02:52 -0800 | [diff] [blame] | 1781 |     private static void loge(String msg) { | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1782 |         Log.e(LOG_TAG, "[PhoneIntfMgr] " + msg); | 
 | 1783 |     } | 
 | 1784 |  | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1785 |     @Override | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1786 |     public int getActivePhoneType() { | 
| Shishir Agrawal | a9f3218 | 2016-04-12 12:00:16 -0700 | [diff] [blame] | 1787 |         return getActivePhoneTypeForSlot(getSlotForDefaultSubscription()); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1788 |     } | 
 | 1789 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1790 |     @Override | 
| Shishir Agrawal | a9f3218 | 2016-04-12 12:00:16 -0700 | [diff] [blame] | 1791 |     public int getActivePhoneTypeForSlot(int slotId) { | 
 | 1792 |         final Phone phone = PhoneFactory.getPhone(slotId); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1793 |         if (phone == null) { | 
 | 1794 |             return PhoneConstants.PHONE_TYPE_NONE; | 
 | 1795 |         } else { | 
| Shishir Agrawal | a9f3218 | 2016-04-12 12:00:16 -0700 | [diff] [blame] | 1796 |             return phone.getPhoneType(); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1797 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1798 |     } | 
 | 1799 |  | 
 | 1800 |     /** | 
 | 1801 |      * Returns the CDMA ERI icon index to display | 
 | 1802 |      */ | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1803 |     @Override | 
 | 1804 |     public int getCdmaEriIconIndex(String callingPackage) { | 
 | 1805 |         return getCdmaEriIconIndexForSubscriber(getDefaultSubscription(), callingPackage); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1806 |     } | 
 | 1807 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1808 |     @Override | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1809 |     public int getCdmaEriIconIndexForSubscriber(int subId, String callingPackage) { | 
 | 1810 |         if (!canReadPhoneState(callingPackage, "getCdmaEriIconIndexForSubscriber")) { | 
 | 1811 |             return -1; | 
 | 1812 |         } | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1813 |         final Phone phone = getPhone(subId); | 
 | 1814 |         if (phone != null) { | 
 | 1815 |             return phone.getCdmaEriIconIndex(); | 
 | 1816 |         } else { | 
 | 1817 |             return -1; | 
 | 1818 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1819 |     } | 
 | 1820 |  | 
 | 1821 |     /** | 
 | 1822 |      * Returns the CDMA ERI icon mode, | 
 | 1823 |      * 0 - ON | 
 | 1824 |      * 1 - FLASHING | 
 | 1825 |      */ | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1826 |     @Override | 
 | 1827 |     public int getCdmaEriIconMode(String callingPackage) { | 
 | 1828 |         return getCdmaEriIconModeForSubscriber(getDefaultSubscription(), callingPackage); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1829 |     } | 
 | 1830 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1831 |     @Override | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1832 |     public int getCdmaEriIconModeForSubscriber(int subId, String callingPackage) { | 
 | 1833 |         if (!canReadPhoneState(callingPackage, "getCdmaEriIconModeForSubscriber")) { | 
 | 1834 |             return -1; | 
 | 1835 |         } | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1836 |         final Phone phone = getPhone(subId); | 
 | 1837 |         if (phone != null) { | 
 | 1838 |             return phone.getCdmaEriIconMode(); | 
 | 1839 |         } else { | 
 | 1840 |             return -1; | 
 | 1841 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1842 |     } | 
 | 1843 |  | 
 | 1844 |     /** | 
 | 1845 |      * Returns the CDMA ERI text, | 
 | 1846 |      */ | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1847 |     @Override | 
 | 1848 |     public String getCdmaEriText(String callingPackage) { | 
 | 1849 |         return getCdmaEriTextForSubscriber(getDefaultSubscription(), callingPackage); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1850 |     } | 
 | 1851 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1852 |     @Override | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 1853 |     public String getCdmaEriTextForSubscriber(int subId, String callingPackage) { | 
 | 1854 |         if (!canReadPhoneState(callingPackage, "getCdmaEriIconTextForSubscriber")) { | 
 | 1855 |             return null; | 
 | 1856 |         } | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1857 |         final Phone phone = getPhone(subId); | 
 | 1858 |         if (phone != null) { | 
 | 1859 |             return phone.getCdmaEriText(); | 
 | 1860 |         } else { | 
 | 1861 |             return null; | 
 | 1862 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1863 |     } | 
 | 1864 |  | 
 | 1865 |     /** | 
| Junda Liu | ca05d5d | 2014-08-14 22:36:34 -0700 | [diff] [blame] | 1866 |      * Returns the CDMA MDN. | 
 | 1867 |      */ | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1868 |     @Override | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1869 |     public String getCdmaMdn(int subId) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 1870 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1871 |         final Phone phone = getPhone(subId); | 
 | 1872 |         if (mPhone.getPhoneType() == PhoneConstants.PHONE_TYPE_CDMA && phone != null) { | 
 | 1873 |             return phone.getLine1Number(); | 
| Junda Liu | ca05d5d | 2014-08-14 22:36:34 -0700 | [diff] [blame] | 1874 |         } else { | 
 | 1875 |             return null; | 
 | 1876 |         } | 
 | 1877 |     } | 
 | 1878 |  | 
 | 1879 |     /** | 
 | 1880 |      * Returns the CDMA MIN. | 
 | 1881 |      */ | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1882 |     @Override | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1883 |     public String getCdmaMin(int subId) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 1884 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1885 |         final Phone phone = getPhone(subId); | 
 | 1886 |         if (phone != null && phone.getPhoneType() == PhoneConstants.PHONE_TYPE_CDMA) { | 
 | 1887 |             return phone.getCdmaMin(); | 
| Junda Liu | ca05d5d | 2014-08-14 22:36:34 -0700 | [diff] [blame] | 1888 |         } else { | 
 | 1889 |             return null; | 
 | 1890 |         } | 
 | 1891 |     } | 
 | 1892 |  | 
 | 1893 |     /** | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1894 |      * Returns true if CDMA provisioning needs to run. | 
 | 1895 |      */ | 
 | 1896 |     public boolean needsOtaServiceProvisioning() { | 
 | 1897 |         return mPhone.needsOtaServiceProvisioning(); | 
 | 1898 |     } | 
 | 1899 |  | 
 | 1900 |     /** | 
| Shishir Agrawal | 76d5da9 | 2014-11-09 16:17:25 -0800 | [diff] [blame] | 1901 |      * Sets the voice mail number of a given subId. | 
 | 1902 |      */ | 
 | 1903 |     @Override | 
 | 1904 |     public boolean setVoiceMailNumber(int subId, String alphaTag, String number) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 1905 |         enforceCarrierPrivilege(subId); | 
| Shishir Agrawal | 76d5da9 | 2014-11-09 16:17:25 -0800 | [diff] [blame] | 1906 |         Boolean success = (Boolean) sendRequest(CMD_SET_VOICEMAIL_NUMBER, | 
 | 1907 |                 new Pair<String, String>(alphaTag, number), new Integer(subId)); | 
 | 1908 |         return success; | 
 | 1909 |     } | 
 | 1910 |  | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 1911 |     @Override | 
| Ta-wei Yen | b0f695b | 2016-08-08 17:33:11 -0700 | [diff] [blame] | 1912 |     public void setVisualVoicemailEnabled(String callingPackage, | 
 | 1913 |             PhoneAccountHandle phoneAccountHandle, boolean enabled) { | 
 | 1914 |         mAppOps.checkPackage(Binder.getCallingUid(), callingPackage); | 
 | 1915 |         if (!TextUtils.equals(callingPackage, | 
 | 1916 |                 TelecomManager.from(mPhone.getContext()).getDefaultDialerPackage())) { | 
 | 1917 |             enforceModifyPermissionOrCarrierPrivilege( | 
 | 1918 |                     PhoneUtils.getSubIdForPhoneAccountHandle(phoneAccountHandle)); | 
 | 1919 |         } | 
 | 1920 |         VisualVoicemailSettingsUtil.setEnabled(mPhone.getContext(), phoneAccountHandle, enabled); | 
 | 1921 |     } | 
 | 1922 |  | 
 | 1923 |     @Override | 
 | 1924 |     public boolean isVisualVoicemailEnabled(String callingPackage, | 
 | 1925 |             PhoneAccountHandle phoneAccountHandle) { | 
 | 1926 |         if (!canReadPhoneState(callingPackage, "isVisualVoicemailEnabled")) { | 
 | 1927 |             return false; | 
 | 1928 |         } | 
 | 1929 |         return VisualVoicemailSettingsUtil.isEnabled(mPhone.getContext(), phoneAccountHandle); | 
 | 1930 |     } | 
 | 1931 |  | 
 | 1932 |     @Override | 
| Ta-wei Yen | b692960 | 2016-05-24 15:48:27 -0700 | [diff] [blame] | 1933 |     public void enableVisualVoicemailSmsFilter(String callingPackage, int subId, | 
 | 1934 |             VisualVoicemailSmsFilterSettings settings) { | 
 | 1935 |         mAppOps.checkPackage(Binder.getCallingUid(), callingPackage); | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 1936 |         VisualVoicemailSmsFilterConfig | 
| Ta-wei Yen | b692960 | 2016-05-24 15:48:27 -0700 | [diff] [blame] | 1937 |                 .enableVisualVoicemailSmsFilter(mPhone.getContext(), callingPackage, subId, | 
 | 1938 |                         settings); | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 1939 |     } | 
 | 1940 |  | 
 | 1941 |     @Override | 
| Ta-wei Yen | b692960 | 2016-05-24 15:48:27 -0700 | [diff] [blame] | 1942 |     public void disableVisualVoicemailSmsFilter(String callingPackage, int subId) { | 
 | 1943 |         mAppOps.checkPackage(Binder.getCallingUid(), callingPackage); | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 1944 |         VisualVoicemailSmsFilterConfig | 
| Ta-wei Yen | b692960 | 2016-05-24 15:48:27 -0700 | [diff] [blame] | 1945 |                 .disableVisualVoicemailSmsFilter(mPhone.getContext(), callingPackage, subId); | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 1946 |     } | 
 | 1947 |  | 
 | 1948 |     @Override | 
| Ta-wei Yen | b692960 | 2016-05-24 15:48:27 -0700 | [diff] [blame] | 1949 |     public VisualVoicemailSmsFilterSettings getVisualVoicemailSmsFilterSettings( | 
 | 1950 |             String callingPackage, int subId) { | 
 | 1951 |         mAppOps.checkPackage(Binder.getCallingUid(), callingPackage); | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 1952 |         return VisualVoicemailSmsFilterConfig | 
| Ta-wei Yen | b692960 | 2016-05-24 15:48:27 -0700 | [diff] [blame] | 1953 |                 .getVisualVoicemailSmsFilterSettings(mPhone.getContext(), callingPackage, subId); | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 1954 |     } | 
 | 1955 |  | 
 | 1956 |     @Override | 
| Ta-wei Yen | b692960 | 2016-05-24 15:48:27 -0700 | [diff] [blame] | 1957 |     public VisualVoicemailSmsFilterSettings getSystemVisualVoicemailSmsFilterSettings( | 
 | 1958 |             String packageName, int subId) { | 
 | 1959 |         enforceReadPrivilegedPermission(); | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 1960 |         return VisualVoicemailSmsFilterConfig | 
| Ta-wei Yen | b692960 | 2016-05-24 15:48:27 -0700 | [diff] [blame] | 1961 |                 .getVisualVoicemailSmsFilterSettings(mPhone.getContext(), packageName, subId); | 
| Ta-wei Yen | 87c4984 | 2016-05-13 21:19:52 -0700 | [diff] [blame] | 1962 |     } | 
| Shishir Agrawal | 76d5da9 | 2014-11-09 16:17:25 -0800 | [diff] [blame] | 1963 |     /** | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1964 |      * Returns the unread count of voicemails | 
 | 1965 |      */ | 
 | 1966 |     public int getVoiceMessageCount() { | 
| Wink Saville | add7cc5 | 2014-09-08 14:23:09 -0700 | [diff] [blame] | 1967 |         return getVoiceMessageCountForSubscriber(getDefaultSubscription()); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1968 |     } | 
 | 1969 |  | 
 | 1970 |     /** | 
 | 1971 |      * Returns the unread count of voicemails for a subId | 
 | 1972 |      */ | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1973 |     @Override | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 1974 |     public int getVoiceMessageCountForSubscriber( int subId) { | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 1975 |         final Phone phone = getPhone(subId); | 
 | 1976 |         if (phone != null) { | 
 | 1977 |             return phone.getVoiceMessageCount(); | 
 | 1978 |         } else { | 
 | 1979 |             return 0; | 
 | 1980 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1981 |     } | 
 | 1982 |  | 
 | 1983 |     /** | 
| Robert Greenwalt | a5dcfcb | 2015-07-10 09:06:29 -0700 | [diff] [blame] | 1984 |      * Returns the data network type. | 
 | 1985 |      * Legacy call, permission-free. | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 1986 |      * | 
 | 1987 |      * @Deprecated to be removed Q3 2013 use {@link #getDataNetworkType}. | 
 | 1988 |      */ | 
 | 1989 |     @Override | 
| Robert Greenwalt | a5dcfcb | 2015-07-10 09:06:29 -0700 | [diff] [blame] | 1990 |     public int getNetworkType() { | 
 | 1991 |         final Phone phone = getPhone(getDefaultSubscription()); | 
 | 1992 |         if (phone != null) { | 
 | 1993 |             return phone.getServiceState().getDataNetworkType(); | 
 | 1994 |         } else { | 
 | 1995 |             return TelephonyManager.NETWORK_TYPE_UNKNOWN; | 
 | 1996 |         } | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 1997 |     } | 
 | 1998 |  | 
 | 1999 |     /** | 
 | 2000 |      * Returns the network type for a subId | 
 | 2001 |      */ | 
 | 2002 |     @Override | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 2003 |     public int getNetworkTypeForSubscriber(int subId, String callingPackage) { | 
 | 2004 |         if (!canReadPhoneState(callingPackage, "getNetworkTypeForSubscriber")) { | 
 | 2005 |             return TelephonyManager.NETWORK_TYPE_UNKNOWN; | 
 | 2006 |         } | 
| Robert Greenwalt | a5dcfcb | 2015-07-10 09:06:29 -0700 | [diff] [blame] | 2007 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 2008 |         final Phone phone = getPhone(subId); | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 2009 |         if (phone != null) { | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 2010 |             return phone.getServiceState().getDataNetworkType(); | 
 | 2011 |         } else { | 
 | 2012 |             return TelephonyManager.NETWORK_TYPE_UNKNOWN; | 
 | 2013 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 2014 |     } | 
 | 2015 |  | 
 | 2016 |     /** | 
 | 2017 |      * Returns the data network type | 
 | 2018 |      */ | 
 | 2019 |     @Override | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 2020 |     public int getDataNetworkType(String callingPackage) { | 
 | 2021 |         return getDataNetworkTypeForSubscriber(getDefaultSubscription(), callingPackage); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 2022 |     } | 
 | 2023 |  | 
 | 2024 |     /** | 
 | 2025 |      * Returns the data network type for a subId | 
 | 2026 |      */ | 
 | 2027 |     @Override | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 2028 |     public int getDataNetworkTypeForSubscriber(int subId, String callingPackage) { | 
 | 2029 |         if (!canReadPhoneState(callingPackage, "getDataNetworkTypeForSubscriber")) { | 
 | 2030 |             return TelephonyManager.NETWORK_TYPE_UNKNOWN; | 
 | 2031 |         } | 
 | 2032 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 2033 |         final Phone phone = getPhone(subId); | 
 | 2034 |         if (phone != null) { | 
 | 2035 |             return phone.getServiceState().getDataNetworkType(); | 
 | 2036 |         } else { | 
 | 2037 |             return TelephonyManager.NETWORK_TYPE_UNKNOWN; | 
 | 2038 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 2039 |     } | 
 | 2040 |  | 
 | 2041 |     /** | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 2042 |      * Returns the Voice network type for a subId | 
 | 2043 |      */ | 
 | 2044 |     @Override | 
| Robert Greenwalt | a5dcfcb | 2015-07-10 09:06:29 -0700 | [diff] [blame] | 2045 |     public int getVoiceNetworkTypeForSubscriber(int subId, String callingPackage) { | 
 | 2046 |         if (!canReadPhoneState(callingPackage, "getDataNetworkTypeForSubscriber")) { | 
 | 2047 |             return TelephonyManager.NETWORK_TYPE_UNKNOWN; | 
 | 2048 |         } | 
 | 2049 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 2050 |         final Phone phone = getPhone(subId); | 
 | 2051 |         if (phone != null) { | 
 | 2052 |             return phone.getServiceState().getVoiceNetworkType(); | 
 | 2053 |         } else { | 
 | 2054 |             return TelephonyManager.NETWORK_TYPE_UNKNOWN; | 
 | 2055 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 2056 |     } | 
 | 2057 |  | 
 | 2058 |     /** | 
 | 2059 |      * @return true if a ICC card is present | 
 | 2060 |      */ | 
 | 2061 |     public boolean hasIccCard() { | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 2062 |         // FIXME Make changes to pass defaultSimId of type int | 
| Amit Mahajan | a6fc2a8 | 2015-01-06 11:53:51 -0800 | [diff] [blame] | 2063 |         return hasIccCardUsingSlotId(mSubscriptionController.getSlotId(getDefaultSubscription())); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 2064 |     } | 
 | 2065 |  | 
 | 2066 |     /** | 
 | 2067 |      * @return true if a ICC card is present for a slotId | 
 | 2068 |      */ | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 2069 |     @Override | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 2070 |     public boolean hasIccCardUsingSlotId(int slotId) { | 
| Amit Mahajan | a6fc2a8 | 2015-01-06 11:53:51 -0800 | [diff] [blame] | 2071 |         int subId[] = mSubscriptionController.getSubIdUsingSlotId(slotId); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 2072 |         final Phone phone = getPhone(subId[0]); | 
 | 2073 |         if (subId != null && phone != null) { | 
 | 2074 |             return phone.getIccCard().hasIccCard(); | 
| Amit Mahajan | a6fc2a8 | 2015-01-06 11:53:51 -0800 | [diff] [blame] | 2075 |         } else { | 
 | 2076 |             return false; | 
 | 2077 |         } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 2078 |     } | 
 | 2079 |  | 
 | 2080 |     /** | 
 | 2081 |      * Return if the current radio is LTE on CDMA. This | 
 | 2082 |      * is a tri-state return value as for a period of time | 
 | 2083 |      * the mode may be unknown. | 
 | 2084 |      * | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 2085 |      * @param callingPackage the name of the package making the call. | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 2086 |      * @return {@link Phone#LTE_ON_CDMA_UNKNOWN}, {@link Phone#LTE_ON_CDMA_FALSE} | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 2087 |      * or {@link Phone#LTE_ON_CDMA_TRUE} | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 2088 |      */ | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 2089 |     @Override | 
 | 2090 |     public int getLteOnCdmaMode(String callingPackage) { | 
 | 2091 |         return getLteOnCdmaModeForSubscriber(getDefaultSubscription(), callingPackage); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 2092 |     } | 
 | 2093 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 2094 |     @Override | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 2095 |     public int getLteOnCdmaModeForSubscriber(int subId, String callingPackage) { | 
 | 2096 |         if (!canReadPhoneState(callingPackage, "getLteOnCdmaModeForSubscriber")) { | 
 | 2097 |             return PhoneConstants.LTE_ON_CDMA_UNKNOWN; | 
 | 2098 |         } | 
 | 2099 |  | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 2100 |         final Phone phone = getPhone(subId); | 
 | 2101 |         if (phone == null) { | 
 | 2102 |             return PhoneConstants.LTE_ON_CDMA_UNKNOWN; | 
 | 2103 |         } else { | 
 | 2104 |             return phone.getLteOnCdmaMode(); | 
 | 2105 |         } | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 2106 |     } | 
 | 2107 |  | 
 | 2108 |     public void setPhone(Phone phone) { | 
 | 2109 |         mPhone = phone; | 
 | 2110 |     } | 
 | 2111 |  | 
 | 2112 |     /** | 
 | 2113 |      * {@hide} | 
 | 2114 |      * Returns Default subId, 0 in the case of single standby. | 
 | 2115 |      */ | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 2116 |     private int getDefaultSubscription() { | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 2117 |         return mSubscriptionController.getDefaultSubId(); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 2118 |     } | 
 | 2119 |  | 
| Shishir Agrawal | a9f3218 | 2016-04-12 12:00:16 -0700 | [diff] [blame] | 2120 |     private int getSlotForDefaultSubscription() { | 
 | 2121 |         return mSubscriptionController.getPhoneId(getDefaultSubscription()); | 
 | 2122 |     } | 
 | 2123 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 2124 |     private int getPreferredVoiceSubscription() { | 
| Wink Saville | ac1bdfd | 2014-11-20 23:04:44 -0800 | [diff] [blame] | 2125 |         return mSubscriptionController.getDefaultVoiceSubId(); | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 2126 |     } | 
| Ihab Awad | f2177b7 | 2013-11-25 13:33:23 -0800 | [diff] [blame] | 2127 |  | 
 | 2128 |     /** | 
 | 2129 |      * @see android.telephony.TelephonyManager.WifiCallingChoices | 
 | 2130 |      */ | 
 | 2131 |     public int getWhenToMakeWifiCalls() { | 
| Sailesh Nepal | d1e6815 | 2013-12-12 19:08:02 -0800 | [diff] [blame] | 2132 |         return Settings.System.getInt(mPhone.getContext().getContentResolver(), | 
 | 2133 |                 Settings.System.WHEN_TO_MAKE_WIFI_CALLS, getWhenToMakeWifiCallsDefaultPreference()); | 
| Ihab Awad | f2177b7 | 2013-11-25 13:33:23 -0800 | [diff] [blame] | 2134 |     } | 
 | 2135 |  | 
 | 2136 |     /** | 
 | 2137 |      * @see android.telephony.TelephonyManager.WifiCallingChoices | 
 | 2138 |      */ | 
 | 2139 |     public void setWhenToMakeWifiCalls(int preference) { | 
| Sailesh Nepal | d1e6815 | 2013-12-12 19:08:02 -0800 | [diff] [blame] | 2140 |         if (DBG) log("setWhenToMakeWifiCallsStr, storing setting = " + preference); | 
 | 2141 |         Settings.System.putInt(mPhone.getContext().getContentResolver(), | 
 | 2142 |                 Settings.System.WHEN_TO_MAKE_WIFI_CALLS, preference); | 
| Ihab Awad | f9e9273 | 2013-12-05 18:02:52 -0800 | [diff] [blame] | 2143 |     } | 
 | 2144 |  | 
| Sailesh Nepal | d1e6815 | 2013-12-12 19:08:02 -0800 | [diff] [blame] | 2145 |     private static int getWhenToMakeWifiCallsDefaultPreference() { | 
| Santos Cordon | da120f4 | 2014-08-06 04:44:34 -0700 | [diff] [blame] | 2146 |         // TODO: Use a build property to choose this value. | 
| Evan Charlton | 9829e88 | 2013-12-19 15:30:38 -0800 | [diff] [blame] | 2147 |         return TelephonyManager.WifiCallingChoices.ALWAYS_USE; | 
| Ihab Awad | f2177b7 | 2013-11-25 13:33:23 -0800 | [diff] [blame] | 2148 |     } | 
| Shishir Agrawal | 69f6812 | 2013-12-16 17:25:49 -0800 | [diff] [blame] | 2149 |  | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2150 |     @Override | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2151 |     public IccOpenLogicalChannelResponse iccOpenLogicalChannel(int subId, String AID) { | 
 | 2152 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2153 |  | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2154 |         if (DBG) log("iccOpenLogicalChannel: subId=" + subId + " aid=" + AID); | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2155 |         IccOpenLogicalChannelResponse response = (IccOpenLogicalChannelResponse)sendRequest( | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2156 |             CMD_OPEN_CHANNEL, AID, subId); | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2157 |         if (DBG) log("iccOpenLogicalChannel: " + response); | 
 | 2158 |         return response; | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2159 |     } | 
 | 2160 |  | 
 | 2161 |     @Override | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2162 |     public boolean iccCloseLogicalChannel(int subId, int channel) { | 
 | 2163 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2164 |  | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2165 |         if (DBG) log("iccCloseLogicalChannel: subId=" + subId + " chnl=" + channel); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2166 |         if (channel < 0) { | 
 | 2167 |           return false; | 
 | 2168 |         } | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2169 |         Boolean success = (Boolean)sendRequest(CMD_CLOSE_CHANNEL, channel, subId); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2170 |         if (DBG) log("iccCloseLogicalChannel: " + success); | 
 | 2171 |         return success; | 
 | 2172 |     } | 
 | 2173 |  | 
 | 2174 |     @Override | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2175 |     public String iccTransmitApduLogicalChannel(int subId, int channel, int cla, | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2176 |             int command, int p1, int p2, int p3, String data) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2177 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2178 |  | 
 | 2179 |         if (DBG) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2180 |             log("iccTransmitApduLogicalChannel: subId=" + subId + " chnl=" + channel + | 
 | 2181 |                     " cla=" + cla + " cmd=" + command + " p1=" + p1 + " p2=" + p2 + " p3=" + p3 + | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2182 |                     " data=" + data); | 
 | 2183 |         } | 
 | 2184 |  | 
 | 2185 |         if (channel < 0) { | 
 | 2186 |             return ""; | 
 | 2187 |         } | 
 | 2188 |  | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2189 |         IccIoResult response = (IccIoResult)sendRequest(CMD_TRANSMIT_APDU_LOGICAL_CHANNEL, | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2190 |                 new IccAPDUArgument(channel, cla, command, p1, p2, p3, data), subId); | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2191 |         if (DBG) log("iccTransmitApduLogicalChannel: " + response); | 
 | 2192 |  | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2193 |         // Append the returned status code to the end of the response payload. | 
 | 2194 |         String s = Integer.toHexString( | 
 | 2195 |                 (response.sw1 << 8) + response.sw2 + 0x10000).substring(1); | 
| Shishir Agrawal | 5ec1417 | 2014-08-05 17:05:45 -0700 | [diff] [blame] | 2196 |         if (response.payload != null) { | 
 | 2197 |             s = IccUtils.bytesToHexString(response.payload) + s; | 
 | 2198 |         } | 
| Shishir Agrawal | 566b761 | 2013-10-28 14:41:00 -0700 | [diff] [blame] | 2199 |         return s; | 
 | 2200 |     } | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 2201 |  | 
| Evan Charlton | c66da36 | 2014-05-16 14:06:40 -0700 | [diff] [blame] | 2202 |     @Override | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2203 |     public String iccTransmitApduBasicChannel(int subId, int cla, int command, int p1, int p2, | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2204 |                 int p3, String data) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2205 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2206 |  | 
 | 2207 |         if (DBG) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2208 |             log("iccTransmitApduBasicChannel: subId=" + subId + " cla=" + cla + " cmd=" + command | 
 | 2209 |                     + " p1=" + p1 + " p2=" + p2 + " p3=" + p3 + " data=" + data); | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2210 |         } | 
 | 2211 |  | 
 | 2212 |         IccIoResult response = (IccIoResult)sendRequest(CMD_TRANSMIT_APDU_BASIC_CHANNEL, | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2213 |                 new IccAPDUArgument(0, cla, command, p1, p2, p3, data), subId); | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2214 |         if (DBG) log("iccTransmitApduBasicChannel: " + response); | 
 | 2215 |  | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2216 |         // Append the returned status code to the end of the response payload. | 
 | 2217 |         String s = Integer.toHexString( | 
 | 2218 |                 (response.sw1 << 8) + response.sw2 + 0x10000).substring(1); | 
| Shishir Agrawal | 5ec1417 | 2014-08-05 17:05:45 -0700 | [diff] [blame] | 2219 |         if (response.payload != null) { | 
 | 2220 |             s = IccUtils.bytesToHexString(response.payload) + s; | 
 | 2221 |         } | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2222 |         return s; | 
 | 2223 |     } | 
 | 2224 |  | 
 | 2225 |     @Override | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2226 |     public byte[] iccExchangeSimIO(int subId, int fileID, int command, int p1, int p2, int p3, | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2227 |             String filePath) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2228 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2229 |  | 
 | 2230 |         if (DBG) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2231 |             log("Exchange SIM_IO " + subId + ":" + fileID + ":" + command + " " + | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2232 |                 p1 + " " + p2 + " " + p3 + ":" + filePath); | 
 | 2233 |         } | 
 | 2234 |  | 
 | 2235 |         IccIoResult response = | 
 | 2236 |             (IccIoResult)sendRequest(CMD_EXCHANGE_SIM_IO, | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2237 |                     new IccAPDUArgument(-1, fileID, command, p1, p2, p3, filePath), | 
 | 2238 |                     subId); | 
| Shishir Agrawal | da0bb0d | 2014-07-29 21:18:53 -0700 | [diff] [blame] | 2239 |  | 
 | 2240 |         if (DBG) { | 
 | 2241 |           log("Exchange SIM_IO [R]" + response); | 
 | 2242 |         } | 
 | 2243 |  | 
 | 2244 |         byte[] result = null; | 
 | 2245 |         int length = 2; | 
 | 2246 |         if (response.payload != null) { | 
 | 2247 |             length = 2 + response.payload.length; | 
 | 2248 |             result = new byte[length]; | 
 | 2249 |             System.arraycopy(response.payload, 0, result, 0, response.payload.length); | 
 | 2250 |         } else { | 
 | 2251 |             result = new byte[length]; | 
 | 2252 |         } | 
 | 2253 |  | 
 | 2254 |         result[length - 1] = (byte) response.sw2; | 
 | 2255 |         result[length - 2] = (byte) response.sw1; | 
 | 2256 |         return result; | 
 | 2257 |     } | 
 | 2258 |  | 
 | 2259 |     @Override | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2260 |     public String sendEnvelopeWithStatus(int subId, String content) { | 
 | 2261 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Evan Charlton | c66da36 | 2014-05-16 14:06:40 -0700 | [diff] [blame] | 2262 |  | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2263 |         IccIoResult response = (IccIoResult)sendRequest(CMD_SEND_ENVELOPE, content, subId); | 
| Evan Charlton | c66da36 | 2014-05-16 14:06:40 -0700 | [diff] [blame] | 2264 |         if (response.payload == null) { | 
 | 2265 |           return ""; | 
 | 2266 |         } | 
 | 2267 |  | 
 | 2268 |         // Append the returned status code to the end of the response payload. | 
 | 2269 |         String s = Integer.toHexString( | 
 | 2270 |                 (response.sw1 << 8) + response.sw2 + 0x10000).substring(1); | 
 | 2271 |         s = IccUtils.bytesToHexString(response.payload) + s; | 
 | 2272 |         return s; | 
 | 2273 |     } | 
 | 2274 |  | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 2275 |     /** | 
 | 2276 |      * Read one of the NV items defined in {@link com.android.internal.telephony.RadioNVItems} | 
 | 2277 |      * and {@code ril_nv_items.h}. Used for device configuration by some CDMA operators. | 
 | 2278 |      * | 
 | 2279 |      * @param itemID the ID of the item to read | 
 | 2280 |      * @return the NV item as a String, or null on error. | 
 | 2281 |      */ | 
 | 2282 |     @Override | 
 | 2283 |     public String nvReadItem(int itemID) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2284 |         enforceModifyPermissionOrCarrierPrivilege(getDefaultSubscription()); | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 2285 |         if (DBG) log("nvReadItem: item " + itemID); | 
 | 2286 |         String value = (String) sendRequest(CMD_NV_READ_ITEM, itemID); | 
 | 2287 |         if (DBG) log("nvReadItem: item " + itemID + " is \"" + value + '"'); | 
 | 2288 |         return value; | 
 | 2289 |     } | 
 | 2290 |  | 
 | 2291 |     /** | 
 | 2292 |      * Write one of the NV items defined in {@link com.android.internal.telephony.RadioNVItems} | 
 | 2293 |      * and {@code ril_nv_items.h}. Used for device configuration by some CDMA operators. | 
 | 2294 |      * | 
 | 2295 |      * @param itemID the ID of the item to read | 
 | 2296 |      * @param itemValue the value to write, as a String | 
 | 2297 |      * @return true on success; false on any failure | 
 | 2298 |      */ | 
 | 2299 |     @Override | 
 | 2300 |     public boolean nvWriteItem(int itemID, String itemValue) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2301 |         enforceModifyPermissionOrCarrierPrivilege(getDefaultSubscription()); | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 2302 |         if (DBG) log("nvWriteItem: item " + itemID + " value \"" + itemValue + '"'); | 
 | 2303 |         Boolean success = (Boolean) sendRequest(CMD_NV_WRITE_ITEM, | 
 | 2304 |                 new Pair<Integer, String>(itemID, itemValue)); | 
 | 2305 |         if (DBG) log("nvWriteItem: item " + itemID + ' ' + (success ? "ok" : "fail")); | 
 | 2306 |         return success; | 
 | 2307 |     } | 
 | 2308 |  | 
 | 2309 |     /** | 
 | 2310 |      * Update the CDMA Preferred Roaming List (PRL) in the radio NV storage. | 
 | 2311 |      * Used for device configuration by some CDMA operators. | 
 | 2312 |      * | 
 | 2313 |      * @param preferredRoamingList byte array containing the new PRL | 
 | 2314 |      * @return true on success; false on any failure | 
 | 2315 |      */ | 
 | 2316 |     @Override | 
 | 2317 |     public boolean nvWriteCdmaPrl(byte[] preferredRoamingList) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2318 |         enforceModifyPermissionOrCarrierPrivilege(getDefaultSubscription()); | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 2319 |         if (DBG) log("nvWriteCdmaPrl: value: " + HexDump.toHexString(preferredRoamingList)); | 
 | 2320 |         Boolean success = (Boolean) sendRequest(CMD_NV_WRITE_CDMA_PRL, preferredRoamingList); | 
 | 2321 |         if (DBG) log("nvWriteCdmaPrl: " + (success ? "ok" : "fail")); | 
 | 2322 |         return success; | 
 | 2323 |     } | 
 | 2324 |  | 
 | 2325 |     /** | 
 | 2326 |      * Perform the specified type of NV config reset. | 
 | 2327 |      * Used for device configuration by some CDMA operators. | 
 | 2328 |      * | 
 | 2329 |      * @param resetType the type of reset to perform (1 == factory reset; 2 == NV-only reset) | 
 | 2330 |      * @return true on success; false on any failure | 
 | 2331 |      */ | 
 | 2332 |     @Override | 
 | 2333 |     public boolean nvResetConfig(int resetType) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2334 |         enforceModifyPermissionOrCarrierPrivilege(getDefaultSubscription()); | 
| Jake Hamby | e994d46 | 2014-02-03 13:10:13 -0800 | [diff] [blame] | 2335 |         if (DBG) log("nvResetConfig: type " + resetType); | 
 | 2336 |         Boolean success = (Boolean) sendRequest(CMD_NV_RESET_CONFIG, resetType); | 
 | 2337 |         if (DBG) log("nvResetConfig: type " + resetType + ' ' + (success ? "ok" : "fail")); | 
 | 2338 |         return success; | 
 | 2339 |     } | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 2340 |  | 
 | 2341 |     /** | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 2342 |      * {@hide} | 
 | 2343 |      * Returns Default sim, 0 in the case of single standby. | 
 | 2344 |      */ | 
 | 2345 |     public int getDefaultSim() { | 
 | 2346 |         //TODO Need to get it from Telephony Devcontroller | 
 | 2347 |         return 0; | 
 | 2348 |     } | 
 | 2349 |  | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 2350 |     public String[] getPcscfAddress(String apnType, String callingPackage) { | 
 | 2351 |         if (!canReadPhoneState(callingPackage, "getPcscfAddress")) { | 
 | 2352 |             return new String[0]; | 
 | 2353 |         } | 
 | 2354 |  | 
 | 2355 |  | 
| ram | 87fca6f | 2014-07-18 18:58:44 +0530 | [diff] [blame] | 2356 |         return mPhone.getPcscfAddress(apnType); | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 2357 |     } | 
 | 2358 |  | 
| Brad Ebinger | 7668100 | 2017-01-23 13:50:20 -0800 | [diff] [blame] | 2359 |     /** | 
 | 2360 |      * Returns the {@link IImsServiceController} that corresponds to the given slot Id and IMS | 
 | 2361 |      * feature or {@link null} if the service is not available. If an ImsServiceController is | 
 | 2362 |      * available, the {@link IImsServiceFeatureListener} callback is registered as a listener for | 
 | 2363 |      * feature updates. | 
 | 2364 |      */ | 
 | 2365 |     public IImsServiceController getImsServiceControllerAndListen(int slotId, int feature, | 
 | 2366 |             IImsServiceFeatureListener callback) { | 
 | 2367 |         enforceModifyPermission(); | 
 | 2368 |         return PhoneFactory.getImsResolver().getImsServiceControllerAndListen(slotId, feature, | 
 | 2369 |                 callback); | 
 | 2370 |     } | 
 | 2371 |  | 
| Wink Saville | 36469e7 | 2014-06-11 15:17:00 -0700 | [diff] [blame] | 2372 |     public void setImsRegistrationState(boolean registered) { | 
 | 2373 |         enforceModifyPermission(); | 
 | 2374 |         mPhone.setImsRegistrationState(registered); | 
 | 2375 |     } | 
 | 2376 |  | 
 | 2377 |     /** | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 2378 |      * Set the network selection mode to automatic. | 
 | 2379 |      * | 
 | 2380 |      */ | 
 | 2381 |     @Override | 
 | 2382 |     public void setNetworkSelectionModeAutomatic(int subId) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2383 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 2384 |         if (DBG) log("setNetworkSelectionModeAutomatic: subId " + subId); | 
 | 2385 |         sendRequest(CMD_SET_NETWORK_SELECTION_MODE_AUTOMATIC, null, subId); | 
 | 2386 |     } | 
 | 2387 |  | 
 | 2388 |     /** | 
| Shishir Agrawal | 302c869 | 2015-06-19 13:49:39 -0700 | [diff] [blame] | 2389 |      * Set the network selection mode to manual with the selected carrier. | 
 | 2390 |      */ | 
 | 2391 |     @Override | 
| Shishir Agrawal | 77ba317 | 2015-09-10 14:50:19 -0700 | [diff] [blame] | 2392 |     public boolean setNetworkSelectionModeManual(int subId, OperatorInfo operator, | 
 | 2393 |             boolean persistSelection) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2394 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Shishir Agrawal | 302c869 | 2015-06-19 13:49:39 -0700 | [diff] [blame] | 2395 |         if (DBG) log("setNetworkSelectionModeManual: subId:" + subId + " operator:" + operator); | 
| Shishir Agrawal | 77ba317 | 2015-09-10 14:50:19 -0700 | [diff] [blame] | 2396 |         ManualNetworkSelectionArgument arg = new ManualNetworkSelectionArgument(operator, | 
 | 2397 |                 persistSelection); | 
 | 2398 |         return (Boolean) sendRequest(CMD_SET_NETWORK_SELECTION_MODE_MANUAL, arg, subId); | 
| Shishir Agrawal | 302c869 | 2015-06-19 13:49:39 -0700 | [diff] [blame] | 2399 |     } | 
 | 2400 |  | 
 | 2401 |     /** | 
 | 2402 |      * Scans for available networks. | 
 | 2403 |      */ | 
 | 2404 |     @Override | 
 | 2405 |     public CellNetworkScanResult getCellNetworkScanResults(int subId) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2406 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Shishir Agrawal | 302c869 | 2015-06-19 13:49:39 -0700 | [diff] [blame] | 2407 |         if (DBG) log("getCellNetworkScanResults: subId " + subId); | 
 | 2408 |         CellNetworkScanResult result = (CellNetworkScanResult) sendRequest( | 
 | 2409 |                 CMD_PERFORM_NETWORK_SCAN, null, subId); | 
 | 2410 |         return result; | 
 | 2411 |     } | 
 | 2412 |  | 
 | 2413 |     /** | 
| Junda Liu | 84d15a2 | 2014-07-02 11:21:04 -0700 | [diff] [blame] | 2414 |      * Get the calculated preferred network type. | 
 | 2415 |      * Used for debugging incorrect network type. | 
 | 2416 |      * | 
 | 2417 |      * @return the preferred network type, defined in RILConstants.java. | 
 | 2418 |      */ | 
 | 2419 |     @Override | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 2420 |     public int getCalculatedPreferredNetworkType(String callingPackage) { | 
 | 2421 |         if (!canReadPhoneState(callingPackage, "getCalculatedPreferredNetworkType")) { | 
 | 2422 |             return RILConstants.PREFERRED_NETWORK_MODE; | 
 | 2423 |         } | 
 | 2424 |  | 
| Amit Mahajan | 43330e0 | 2014-11-18 11:54:45 -0800 | [diff] [blame] | 2425 |         return PhoneFactory.calculatePreferredNetworkType(mPhone.getContext(), 0); // wink FIXME: need to get SubId from somewhere. | 
| Junda Liu | 84d15a2 | 2014-07-02 11:21:04 -0700 | [diff] [blame] | 2426 |     } | 
 | 2427 |  | 
 | 2428 |     /** | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 2429 |      * Get the preferred network type. | 
 | 2430 |      * Used for device configuration by some CDMA operators. | 
 | 2431 |      * | 
 | 2432 |      * @return the preferred network type, defined in RILConstants.java. | 
 | 2433 |      */ | 
 | 2434 |     @Override | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 2435 |     public int getPreferredNetworkType(int subId) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2436 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 2437 |         if (DBG) log("getPreferredNetworkType"); | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 2438 |         int[] result = (int[]) sendRequest(CMD_GET_PREFERRED_NETWORK_TYPE, null, subId); | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 2439 |         int networkType = (result != null ? result[0] : -1); | 
 | 2440 |         if (DBG) log("getPreferredNetworkType: " + networkType); | 
 | 2441 |         return networkType; | 
 | 2442 |     } | 
 | 2443 |  | 
 | 2444 |     /** | 
 | 2445 |      * Set the preferred network type. | 
 | 2446 |      * Used for device configuration by some CDMA operators. | 
 | 2447 |      * | 
 | 2448 |      * @param networkType the preferred network type, defined in RILConstants.java. | 
 | 2449 |      * @return true on success; false on any failure. | 
 | 2450 |      */ | 
 | 2451 |     @Override | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 2452 |     public boolean setPreferredNetworkType(int subId, int networkType) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2453 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 2454 |         if (DBG) log("setPreferredNetworkType: subId " + subId + " type " + networkType); | 
 | 2455 |         Boolean success = (Boolean) sendRequest(CMD_SET_PREFERRED_NETWORK_TYPE, networkType, subId); | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 2456 |         if (DBG) log("setPreferredNetworkType: " + (success ? "ok" : "fail")); | 
| Junda Liu | 80bc0d1 | 2014-07-14 16:36:44 -0700 | [diff] [blame] | 2457 |         if (success) { | 
 | 2458 |             Settings.Global.putInt(mPhone.getContext().getContentResolver(), | 
| Stuart Scott | 5478880 | 2015-03-30 13:18:01 -0700 | [diff] [blame] | 2459 |                     Settings.Global.PREFERRED_NETWORK_MODE + subId, networkType); | 
| Junda Liu | 80bc0d1 | 2014-07-14 16:36:44 -0700 | [diff] [blame] | 2460 |         } | 
| Jake Hamby | 7c27be3 | 2014-03-03 13:25:59 -0800 | [diff] [blame] | 2461 |         return success; | 
 | 2462 |     } | 
| Robert Greenwalt | ed86e58 | 2014-05-21 20:03:20 -0700 | [diff] [blame] | 2463 |  | 
 | 2464 |     /** | 
| Junda Liu | 475951f | 2014-11-07 16:45:03 -0800 | [diff] [blame] | 2465 |      * Check TETHER_DUN_REQUIRED and TETHER_DUN_APN settings, net.tethering.noprovisioning | 
 | 2466 |      * SystemProperty, and config_tether_apndata to decide whether DUN APN is required for | 
 | 2467 |      * tethering. | 
 | 2468 |      * | 
 | 2469 |      * @return 0: Not required. 1: required. 2: Not set. | 
 | 2470 |      * @hide | 
 | 2471 |      */ | 
 | 2472 |     @Override | 
 | 2473 |     public int getTetherApnRequired() { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2474 |         enforceModifyPermission(); | 
| Junda Liu | 475951f | 2014-11-07 16:45:03 -0800 | [diff] [blame] | 2475 |         int dunRequired = Settings.Global.getInt(mPhone.getContext().getContentResolver(), | 
 | 2476 |                 Settings.Global.TETHER_DUN_REQUIRED, 2); | 
 | 2477 |         // If not set, check net.tethering.noprovisioning, TETHER_DUN_APN setting and | 
 | 2478 |         // config_tether_apndata. | 
 | 2479 |         if (dunRequired == 2 && mPhone.hasMatchedTetherApnSetting()) { | 
 | 2480 |             dunRequired = 1; | 
 | 2481 |         } | 
 | 2482 |         return dunRequired; | 
 | 2483 |     } | 
 | 2484 |  | 
 | 2485 |     /** | 
| Robert Greenwalt | ed86e58 | 2014-05-21 20:03:20 -0700 | [diff] [blame] | 2486 |      * Set mobile data enabled | 
 | 2487 |      * Used by the user through settings etc to turn on/off mobile data | 
 | 2488 |      * | 
 | 2489 |      * @param enable {@code true} turn turn data on, else {@code false} | 
 | 2490 |      */ | 
 | 2491 |     @Override | 
| Wink Saville | e7353bb | 2014-12-05 14:21:41 -0800 | [diff] [blame] | 2492 |     public void setDataEnabled(int subId, boolean enable) { | 
| Jeff Davidson | f7eecf0 | 2016-11-18 17:05:56 -0800 | [diff] [blame] | 2493 |         enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Wink Saville | e7353bb | 2014-12-05 14:21:41 -0800 | [diff] [blame] | 2494 |         int phoneId = mSubscriptionController.getPhoneId(subId); | 
| Joe Onorato | a601dd2 | 2016-02-23 13:03:53 -0800 | [diff] [blame] | 2495 |         if (DBG) log("getDataEnabled: subId=" + subId + " phoneId=" + phoneId); | 
| Wink Saville | e7353bb | 2014-12-05 14:21:41 -0800 | [diff] [blame] | 2496 |         Phone phone = PhoneFactory.getPhone(phoneId); | 
 | 2497 |         if (phone != null) { | 
| Joe Onorato | a601dd2 | 2016-02-23 13:03:53 -0800 | [diff] [blame] | 2498 |             if (DBG) log("setDataEnabled: subId=" + subId + " enable=" + enable); | 
| Wink Saville | e7353bb | 2014-12-05 14:21:41 -0800 | [diff] [blame] | 2499 |             phone.setDataEnabled(enable); | 
 | 2500 |         } else { | 
 | 2501 |             loge("setDataEnabled: no phone for subId=" + subId); | 
 | 2502 |         } | 
| Robert Greenwalt | ed86e58 | 2014-05-21 20:03:20 -0700 | [diff] [blame] | 2503 |     } | 
 | 2504 |  | 
 | 2505 |     /** | 
| Robert Greenwalt | 646120a | 2014-05-23 11:54:03 -0700 | [diff] [blame] | 2506 |      * Get whether mobile data is enabled. | 
 | 2507 |      * | 
| Jeff Davidson | f7eecf0 | 2016-11-18 17:05:56 -0800 | [diff] [blame] | 2508 |      * Accepts either ACCESS_NETWORK_STATE, MODIFY_PHONE_STATE or carrier privileges. | 
| Robert Greenwalt | ed86e58 | 2014-05-21 20:03:20 -0700 | [diff] [blame] | 2509 |      * | 
 | 2510 |      * @return {@code true} if data is enabled else {@code false} | 
 | 2511 |      */ | 
 | 2512 |     @Override | 
| Wink Saville | e7353bb | 2014-12-05 14:21:41 -0800 | [diff] [blame] | 2513 |     public boolean getDataEnabled(int subId) { | 
| Robert Greenwalt | 646120a | 2014-05-23 11:54:03 -0700 | [diff] [blame] | 2514 |         try { | 
 | 2515 |             mApp.enforceCallingOrSelfPermission(android.Manifest.permission.ACCESS_NETWORK_STATE, | 
 | 2516 |                     null); | 
 | 2517 |         } catch (Exception e) { | 
| Jeff Davidson | f7eecf0 | 2016-11-18 17:05:56 -0800 | [diff] [blame] | 2518 |             enforceModifyPermissionOrCarrierPrivilege(subId); | 
| Robert Greenwalt | 646120a | 2014-05-23 11:54:03 -0700 | [diff] [blame] | 2519 |         } | 
| Wink Saville | e7353bb | 2014-12-05 14:21:41 -0800 | [diff] [blame] | 2520 |         int phoneId = mSubscriptionController.getPhoneId(subId); | 
| Joe Onorato | a601dd2 | 2016-02-23 13:03:53 -0800 | [diff] [blame] | 2521 |         if (DBG) log("getDataEnabled: subId=" + subId + " phoneId=" + phoneId); | 
| Wink Saville | e7353bb | 2014-12-05 14:21:41 -0800 | [diff] [blame] | 2522 |         Phone phone = PhoneFactory.getPhone(phoneId); | 
 | 2523 |         if (phone != null) { | 
 | 2524 |             boolean retVal = phone.getDataEnabled(); | 
| Joe Onorato | a601dd2 | 2016-02-23 13:03:53 -0800 | [diff] [blame] | 2525 |             if (DBG) log("getDataEnabled: subId=" + subId + " retVal=" + retVal); | 
| Wink Saville | e7353bb | 2014-12-05 14:21:41 -0800 | [diff] [blame] | 2526 |             return retVal; | 
 | 2527 |         } else { | 
| Joe Onorato | a601dd2 | 2016-02-23 13:03:53 -0800 | [diff] [blame] | 2528 |             if (DBG) loge("getDataEnabled: no phone subId=" + subId + " retVal=false"); | 
| Wink Saville | e7353bb | 2014-12-05 14:21:41 -0800 | [diff] [blame] | 2529 |             return false; | 
 | 2530 |         } | 
| Robert Greenwalt | ed86e58 | 2014-05-21 20:03:20 -0700 | [diff] [blame] | 2531 |     } | 
| Shishir Agrawal | 60f9c95 | 2014-06-23 12:00:43 -0700 | [diff] [blame] | 2532 |  | 
 | 2533 |     @Override | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2534 |     public int getCarrierPrivilegeStatus(int subId) { | 
 | 2535 |         final Phone phone = getPhone(subId); | 
 | 2536 |         if (phone == null) { | 
 | 2537 |             loge("getCarrierPrivilegeStatus: Invalid subId"); | 
 | 2538 |             return TelephonyManager.CARRIER_PRIVILEGE_STATUS_NO_ACCESS; | 
 | 2539 |         } | 
 | 2540 |         UiccCard card = UiccController.getInstance().getUiccCard(phone.getPhoneId()); | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 2541 |         if (card == null) { | 
| Shishir Agrawal | 5e5becd | 2014-11-18 11:38:23 -0800 | [diff] [blame] | 2542 |             loge("getCarrierPrivilegeStatus: No UICC"); | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 2543 |             return TelephonyManager.CARRIER_PRIVILEGE_STATUS_RULES_NOT_LOADED; | 
 | 2544 |         } | 
 | 2545 |         return card.getCarrierPrivilegeStatusForCurrentTransaction( | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2546 |                 phone.getContext().getPackageManager()); | 
| Shishir Agrawal | 60f9c95 | 2014-06-23 12:00:43 -0700 | [diff] [blame] | 2547 |     } | 
| Junda Liu | 2934034 | 2014-07-10 15:23:27 -0700 | [diff] [blame] | 2548 |  | 
 | 2549 |     @Override | 
| Zach Johnson | 50ecba3 | 2015-05-19 00:24:21 -0700 | [diff] [blame] | 2550 |     public int checkCarrierPrivilegesForPackage(String pkgName) { | 
| Junda Liu | 317d70b | 2016-03-08 09:33:53 -0800 | [diff] [blame] | 2551 |         if (TextUtils.isEmpty(pkgName)) | 
 | 2552 |             return TelephonyManager.CARRIER_PRIVILEGE_STATUS_NO_ACCESS; | 
| Shishir Agrawal | 2140925 | 2015-01-15 23:33:50 -0800 | [diff] [blame] | 2553 |         UiccCard card = UiccController.getInstance().getUiccCard(mPhone.getPhoneId()); | 
| Shishir Agrawal | eb8771e | 2014-07-22 11:24:08 -0700 | [diff] [blame] | 2554 |         if (card == null) { | 
 | 2555 |             loge("checkCarrierPrivilegesForPackage: No UICC"); | 
 | 2556 |             return TelephonyManager.CARRIER_PRIVILEGE_STATUS_RULES_NOT_LOADED; | 
 | 2557 |         } | 
| Zach Johnson | 50ecba3 | 2015-05-19 00:24:21 -0700 | [diff] [blame] | 2558 |         return card.getCarrierPrivilegeStatus(mPhone.getContext().getPackageManager(), pkgName); | 
 | 2559 |     } | 
 | 2560 |  | 
 | 2561 |     @Override | 
 | 2562 |     public int checkCarrierPrivilegesForPackageAnyPhone(String pkgName) { | 
| Junda Liu | 317d70b | 2016-03-08 09:33:53 -0800 | [diff] [blame] | 2563 |         if (TextUtils.isEmpty(pkgName)) | 
 | 2564 |             return TelephonyManager.CARRIER_PRIVILEGE_STATUS_NO_ACCESS; | 
| Zach Johnson | 50ecba3 | 2015-05-19 00:24:21 -0700 | [diff] [blame] | 2565 |         int result = TelephonyManager.CARRIER_PRIVILEGE_STATUS_RULES_NOT_LOADED; | 
 | 2566 |         for (int i = 0; i < TelephonyManager.getDefault().getPhoneCount(); i++) { | 
 | 2567 |             UiccCard card = UiccController.getInstance().getUiccCard(i); | 
 | 2568 |             if (card == null) { | 
| Jonathan Basseri | 7d320df | 2015-06-16 12:17:08 -0700 | [diff] [blame] | 2569 |               // No UICC in that slot. | 
| Zach Johnson | 50ecba3 | 2015-05-19 00:24:21 -0700 | [diff] [blame] | 2570 |               continue; | 
 | 2571 |             } | 
 | 2572 |  | 
 | 2573 |             result = card.getCarrierPrivilegeStatus( | 
 | 2574 |                 mPhone.getContext().getPackageManager(), pkgName); | 
 | 2575 |             if (result == TelephonyManager.CARRIER_PRIVILEGE_STATUS_HAS_ACCESS) { | 
 | 2576 |                 break; | 
 | 2577 |             } | 
 | 2578 |         } | 
 | 2579 |  | 
 | 2580 |         return result; | 
| Junda Liu | 2934034 | 2014-07-10 15:23:27 -0700 | [diff] [blame] | 2581 |     } | 
| Derek Tan | 89e89d4 | 2014-07-08 17:00:10 -0700 | [diff] [blame] | 2582 |  | 
 | 2583 |     @Override | 
| Junda Liu | e64de78 | 2015-04-16 17:19:16 -0700 | [diff] [blame] | 2584 |     public List<String> getCarrierPackageNamesForIntentAndPhone(Intent intent, int phoneId) { | 
 | 2585 |         if (!SubscriptionManager.isValidPhoneId(phoneId)) { | 
 | 2586 |             loge("phoneId " + phoneId + " is not valid."); | 
 | 2587 |             return null; | 
 | 2588 |         } | 
 | 2589 |         UiccCard card = UiccController.getInstance().getUiccCard(phoneId); | 
| Shishir Agrawal | eb6439a | 2014-07-21 13:19:38 -0700 | [diff] [blame] | 2590 |         if (card == null) { | 
| Diego Pontoriero | af74c86 | 2014-08-28 11:51:16 -0700 | [diff] [blame] | 2591 |             loge("getCarrierPackageNamesForIntent: No UICC"); | 
| Shishir Agrawal | eb6439a | 2014-07-21 13:19:38 -0700 | [diff] [blame] | 2592 |             return null ; | 
 | 2593 |         } | 
| Diego Pontoriero | af74c86 | 2014-08-28 11:51:16 -0700 | [diff] [blame] | 2594 |         return card.getCarrierPackageNamesForIntent( | 
| Svetoslav | 483aff7 | 2015-04-21 14:16:07 -0700 | [diff] [blame] | 2595 |                 mPhone.getContext().getPackageManager(), intent); | 
| Shishir Agrawal | eb6439a | 2014-07-21 13:19:38 -0700 | [diff] [blame] | 2596 |     } | 
 | 2597 |  | 
| Amith Yamasani | 6e11887 | 2016-02-19 12:53:51 -0800 | [diff] [blame] | 2598 |     @Override | 
 | 2599 |     public List<String> getPackagesWithCarrierPrivileges() { | 
 | 2600 |         PackageManager pm = mPhone.getContext().getPackageManager(); | 
 | 2601 |         List<String> privilegedPackages = new ArrayList<>(); | 
 | 2602 |         List<PackageInfo> packages = null; | 
 | 2603 |         for (int i = 0; i < TelephonyManager.getDefault().getPhoneCount(); i++) { | 
 | 2604 |             UiccCard card = UiccController.getInstance().getUiccCard(i); | 
 | 2605 |             if (card == null) { | 
 | 2606 |                 // No UICC in that slot. | 
 | 2607 |                 continue; | 
 | 2608 |             } | 
 | 2609 |             if (card.hasCarrierPrivilegeRules()) { | 
 | 2610 |                 if (packages == null) { | 
 | 2611 |                     // Only check packages in user 0 for now | 
 | 2612 |                     packages = pm.getInstalledPackagesAsUser( | 
 | 2613 |                             PackageManager.MATCH_DISABLED_COMPONENTS | 
 | 2614 |                             | PackageManager.MATCH_DISABLED_UNTIL_USED_COMPONENTS | 
 | 2615 |                             | PackageManager.GET_SIGNATURES, UserHandle.USER_SYSTEM); | 
 | 2616 |                 } | 
 | 2617 |                 for (int p = packages.size() - 1; p >= 0; p--) { | 
 | 2618 |                     PackageInfo pkgInfo = packages.get(p); | 
 | 2619 |                     if (pkgInfo != null && pkgInfo.packageName != null | 
 | 2620 |                             && card.getCarrierPrivilegeStatus(pkgInfo) | 
 | 2621 |                                 == TelephonyManager.CARRIER_PRIVILEGE_STATUS_HAS_ACCESS) { | 
 | 2622 |                         privilegedPackages.add(pkgInfo.packageName); | 
 | 2623 |                     } | 
 | 2624 |                 } | 
 | 2625 |             } | 
 | 2626 |         } | 
 | 2627 |         return privilegedPackages; | 
 | 2628 |     } | 
 | 2629 |  | 
| Wink Saville | b564aae | 2014-10-23 10:18:09 -0700 | [diff] [blame] | 2630 |     private String getIccId(int subId) { | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 2631 |         final Phone phone = getPhone(subId); | 
 | 2632 |         UiccCard card = phone == null ? null : phone.getUiccCard(); | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 2633 |         if (card == null) { | 
 | 2634 |             loge("getIccId: No UICC"); | 
 | 2635 |             return null; | 
 | 2636 |         } | 
 | 2637 |         String iccId = card.getIccId(); | 
 | 2638 |         if (TextUtils.isEmpty(iccId)) { | 
 | 2639 |             loge("getIccId: ICC ID is null or empty."); | 
 | 2640 |             return null; | 
 | 2641 |         } | 
 | 2642 |         return iccId; | 
 | 2643 |     } | 
 | 2644 |  | 
| Shishir Agrawal | eb6439a | 2014-07-21 13:19:38 -0700 | [diff] [blame] | 2645 |     @Override | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 2646 |     public boolean setLine1NumberForDisplayForSubscriber(int subId, String alphaTag, | 
 | 2647 |             String number) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2648 |         enforceCarrierPrivilege(subId); | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 2649 |  | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 2650 |         final String iccId = getIccId(subId); | 
| Sanket Padawe | 356d763 | 2015-06-22 14:03:32 -0700 | [diff] [blame] | 2651 |         final Phone phone = getPhone(subId); | 
 | 2652 |         if (phone == null) { | 
 | 2653 |             return false; | 
 | 2654 |         } | 
 | 2655 |         final String subscriberId = phone.getSubscriberId(); | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 2656 |  | 
 | 2657 |         if (DBG_MERGE) { | 
 | 2658 |             Slog.d(LOG_TAG, "Setting line number for ICC=" + iccId + ", subscriberId=" | 
 | 2659 |                     + subscriberId + " to " + number); | 
 | 2660 |         } | 
 | 2661 |  | 
| Shishir Agrawal | 495d7e1 | 2014-12-01 11:50:28 -0800 | [diff] [blame] | 2662 |         if (TextUtils.isEmpty(iccId)) { | 
 | 2663 |             return false; | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 2664 |         } | 
| Shishir Agrawal | 495d7e1 | 2014-12-01 11:50:28 -0800 | [diff] [blame] | 2665 |  | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 2666 |         final SharedPreferences.Editor editor = mTelephonySharedPreferences.edit(); | 
 | 2667 |  | 
 | 2668 |         final String alphaTagPrefKey = PREF_CARRIERS_ALPHATAG_PREFIX + iccId; | 
| Shishir Agrawal | 495d7e1 | 2014-12-01 11:50:28 -0800 | [diff] [blame] | 2669 |         if (alphaTag == null) { | 
 | 2670 |             editor.remove(alphaTagPrefKey); | 
 | 2671 |         } else { | 
 | 2672 |             editor.putString(alphaTagPrefKey, alphaTag); | 
 | 2673 |         } | 
 | 2674 |  | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 2675 |         // Record both the line number and IMSI for this ICCID, since we need to | 
 | 2676 |         // track all merged IMSIs based on line number | 
 | 2677 |         final String numberPrefKey = PREF_CARRIERS_NUMBER_PREFIX + iccId; | 
 | 2678 |         final String subscriberPrefKey = PREF_CARRIERS_SUBSCRIBER_PREFIX + iccId; | 
| Shishir Agrawal | 495d7e1 | 2014-12-01 11:50:28 -0800 | [diff] [blame] | 2679 |         if (number == null) { | 
 | 2680 |             editor.remove(numberPrefKey); | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 2681 |             editor.remove(subscriberPrefKey); | 
| Shishir Agrawal | 495d7e1 | 2014-12-01 11:50:28 -0800 | [diff] [blame] | 2682 |         } else { | 
 | 2683 |             editor.putString(numberPrefKey, number); | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 2684 |             editor.putString(subscriberPrefKey, subscriberId); | 
| Shishir Agrawal | 495d7e1 | 2014-12-01 11:50:28 -0800 | [diff] [blame] | 2685 |         } | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 2686 |  | 
| Shishir Agrawal | 495d7e1 | 2014-12-01 11:50:28 -0800 | [diff] [blame] | 2687 |         editor.commit(); | 
 | 2688 |         return true; | 
| Derek Tan | 7226c84 | 2014-07-02 17:42:23 -0700 | [diff] [blame] | 2689 |     } | 
 | 2690 |  | 
 | 2691 |     @Override | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 2692 |     public String getLine1NumberForDisplay(int subId, String callingPackage) { | 
| Makoto Onuki | fee6934 | 2015-06-29 14:44:50 -0700 | [diff] [blame] | 2693 |         // This is open to apps with WRITE_SMS. | 
 | 2694 |         if (!canReadPhoneNumber(callingPackage, "getLine1NumberForDisplay")) { | 
| Amit Mahajan | 9cf1151 | 2015-11-09 11:40:48 -0800 | [diff] [blame] | 2695 |             if (DBG_MERGE) log("getLine1NumberForDisplay returning null due to permission"); | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 2696 |             return null; | 
 | 2697 |         } | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 2698 |  | 
 | 2699 |         String iccId = getIccId(subId); | 
 | 2700 |         if (iccId != null) { | 
 | 2701 |             String numberPrefKey = PREF_CARRIERS_NUMBER_PREFIX + iccId; | 
| Amit Mahajan | 9cf1151 | 2015-11-09 11:40:48 -0800 | [diff] [blame] | 2702 |             if (DBG_MERGE) { | 
 | 2703 |                 log("getLine1NumberForDisplay returning " + | 
 | 2704 |                         mTelephonySharedPreferences.getString(numberPrefKey, null)); | 
 | 2705 |             } | 
| Andrew Lee | df14ead | 2014-10-17 14:22:52 -0700 | [diff] [blame] | 2706 |             return mTelephonySharedPreferences.getString(numberPrefKey, null); | 
| Derek Tan | 7226c84 | 2014-07-02 17:42:23 -0700 | [diff] [blame] | 2707 |         } | 
| Amit Mahajan | 9cf1151 | 2015-11-09 11:40:48 -0800 | [diff] [blame] | 2708 |         if (DBG_MERGE) log("getLine1NumberForDisplay returning null as iccId is null"); | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 2709 |         return null; | 
| Derek Tan | 7226c84 | 2014-07-02 17:42:23 -0700 | [diff] [blame] | 2710 |     } | 
 | 2711 |  | 
 | 2712 |     @Override | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 2713 |     public String getLine1AlphaTagForDisplay(int subId, String callingPackage) { | 
 | 2714 |         if (!canReadPhoneState(callingPackage, "getLine1AlphaTagForDisplay")) { | 
 | 2715 |             return null; | 
 | 2716 |         } | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 2717 |  | 
 | 2718 |         String iccId = getIccId(subId); | 
 | 2719 |         if (iccId != null) { | 
 | 2720 |             String alphaTagPrefKey = PREF_CARRIERS_ALPHATAG_PREFIX + iccId; | 
| Andrew Lee | df14ead | 2014-10-17 14:22:52 -0700 | [diff] [blame] | 2721 |             return mTelephonySharedPreferences.getString(alphaTagPrefKey, null); | 
| Derek Tan | 7226c84 | 2014-07-02 17:42:23 -0700 | [diff] [blame] | 2722 |         } | 
| Derek Tan | 97ebb42 | 2014-09-05 16:55:38 -0700 | [diff] [blame] | 2723 |         return null; | 
| Derek Tan | 7226c84 | 2014-07-02 17:42:23 -0700 | [diff] [blame] | 2724 |     } | 
| Shishir Agrawal | b1ebf8c | 2014-07-17 16:32:41 -0700 | [diff] [blame] | 2725 |  | 
 | 2726 |     @Override | 
| Fyodor Kupolov | 8e53b0b | 2015-06-17 13:17:50 -0700 | [diff] [blame] | 2727 |     public String[] getMergedSubscriberIds(String callingPackage) { | 
 | 2728 |         if (!canReadPhoneState(callingPackage, "getMergedSubscriberIds")) { | 
 | 2729 |             return null; | 
 | 2730 |         } | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 2731 |         final Context context = mPhone.getContext(); | 
 | 2732 |         final TelephonyManager tele = TelephonyManager.from(context); | 
 | 2733 |         final SubscriptionManager sub = SubscriptionManager.from(context); | 
 | 2734 |  | 
 | 2735 |         // Figure out what subscribers are currently active | 
 | 2736 |         final ArraySet<String> activeSubscriberIds = new ArraySet<>(); | 
| Fyodor Kupolov | 8e53b0b | 2015-06-17 13:17:50 -0700 | [diff] [blame] | 2737 |         // Clear calling identity, when calling TelephonyManager, because callerUid must be | 
 | 2738 |         // the process, where TelephonyManager was instantiated. Otherwise AppOps check will fail. | 
 | 2739 |         final long identity  = Binder.clearCallingIdentity(); | 
 | 2740 |         try { | 
 | 2741 |             final int[] subIds = sub.getActiveSubscriptionIdList(); | 
 | 2742 |             for (int subId : subIds) { | 
 | 2743 |                 activeSubscriberIds.add(tele.getSubscriberId(subId)); | 
 | 2744 |             } | 
 | 2745 |         } finally { | 
 | 2746 |             Binder.restoreCallingIdentity(identity); | 
| Jeff Sharkey | 85190e6 | 2014-12-05 09:40:12 -0800 | [diff] [blame] | 2747 |         } | 
 | 2748 |  | 
 | 2749 |         // First pass, find a number override for an active subscriber | 
 | 2750 |         String mergeNumber = null; | 
 | 2751 |         final Map<String, ?> prefs = mTelephonySharedPreferences.getAll(); | 
 | 2752 |         for (String key : prefs.keySet()) { | 
 | 2753 |             if (key.startsWith(PREF_CARRIERS_SUBSCRIBER_PREFIX)) { | 
 | 2754 |                 final String subscriberId = (String) prefs.get(key); | 
 | 2755 |                 if (activeSubscriberIds.contains(subscriberId)) { | 
 | 2756 |                     final String iccId = key.substring(PREF_CARRIERS_SUBSCRIBER_PREFIX.length()); | 
 | 2757 |                     final String numberKey = PREF_CARRIERS_NUMBER_PREFIX + iccId; | 
 | 2758 |                     mergeNumber = (String) prefs.get(numberKey); | 
 | 2759 |                     if (DBG_MERGE) { | 
 | 2760 |                         Slog.d(LOG_TAG, "Found line number " + mergeNumber | 
 | 2761 |                                 + " for active subscriber " + subscriberId); | 
 | 2762 |                     } | 
 | 2763 |                     if (!TextUtils.isEmpty(mergeNumber)) { | 
 | 2764 |                         break; | 
 | 2765 |                     } | 
 | 2766 |                 } | 
 | 2767 |             } | 
 | 2768 |         } | 
 | 2769 |  | 
 | 2770 |         // Shortcut when no active merged subscribers | 
 | 2771 |         if (TextUtils.isEmpty(mergeNumber)) { | 
 | 2772 |             return null; | 
 | 2773 |         } | 
 | 2774 |  | 
 | 2775 |         // Second pass, find all subscribers under that line override | 
 | 2776 |         final ArraySet<String> result = new ArraySet<>(); | 
 | 2777 |         for (String key : prefs.keySet()) { | 
 | 2778 |             if (key.startsWith(PREF_CARRIERS_NUMBER_PREFIX)) { | 
 | 2779 |                 final String number = (String) prefs.get(key); | 
 | 2780 |                 if (mergeNumber.equals(number)) { | 
 | 2781 |                     final String iccId = key.substring(PREF_CARRIERS_NUMBER_PREFIX.length()); | 
 | 2782 |                     final String subscriberKey = PREF_CARRIERS_SUBSCRIBER_PREFIX + iccId; | 
 | 2783 |                     final String subscriberId = (String) prefs.get(subscriberKey); | 
 | 2784 |                     if (!TextUtils.isEmpty(subscriberId)) { | 
 | 2785 |                         result.add(subscriberId); | 
 | 2786 |                     } | 
 | 2787 |                 } | 
 | 2788 |             } | 
 | 2789 |         } | 
 | 2790 |  | 
 | 2791 |         final String[] resultArray = result.toArray(new String[result.size()]); | 
 | 2792 |         Arrays.sort(resultArray); | 
 | 2793 |         if (DBG_MERGE) { | 
 | 2794 |             Slog.d(LOG_TAG, "Found subscribers " + Arrays.toString(resultArray) + " after merge"); | 
 | 2795 |         } | 
 | 2796 |         return resultArray; | 
 | 2797 |     } | 
 | 2798 |  | 
 | 2799 |     @Override | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2800 |     public boolean setOperatorBrandOverride(int subId, String brand) { | 
 | 2801 |         enforceCarrierPrivilege(subId); | 
 | 2802 |         final Phone phone = getPhone(subId); | 
 | 2803 |         return phone == null ? false : phone.setOperatorBrandOverride(brand); | 
| Shishir Agrawal | b1ebf8c | 2014-07-17 16:32:41 -0700 | [diff] [blame] | 2804 |     } | 
| Steven Liu | 4bf01bc | 2014-07-17 11:05:29 -0500 | [diff] [blame] | 2805 |  | 
 | 2806 |     @Override | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2807 |     public boolean setRoamingOverride(int subId, List<String> gsmRoamingList, | 
| Shishir Agrawal | 621a47c | 2014-12-01 10:25:09 -0800 | [diff] [blame] | 2808 |             List<String> gsmNonRoamingList, List<String> cdmaRoamingList, | 
 | 2809 |             List<String> cdmaNonRoamingList) { | 
| Shishir Agrawal | c04d975 | 2016-02-19 10:41:00 -0800 | [diff] [blame] | 2810 |         enforceCarrierPrivilege(subId); | 
 | 2811 |         final Phone phone = getPhone(subId); | 
 | 2812 |         if (phone == null) { | 
 | 2813 |             return false; | 
 | 2814 |         } | 
 | 2815 |         return phone.setRoamingOverride(gsmRoamingList, gsmNonRoamingList, cdmaRoamingList, | 
| Shishir Agrawal | 621a47c | 2014-12-01 10:25:09 -0800 | [diff] [blame] | 2816 |                 cdmaNonRoamingList); | 
 | 2817 |     } | 
 | 2818 |  | 
 | 2819 |     @Override | 
| Steven Liu | 4bf01bc | 2014-07-17 11:05:29 -0500 | [diff] [blame] | 2820 |     public int invokeOemRilRequestRaw(byte[] oemReq, byte[] oemResp) { | 
 | 2821 |         enforceModifyPermission(); | 
 | 2822 |  | 
 | 2823 |         int returnValue = 0; | 
 | 2824 |         try { | 
 | 2825 |             AsyncResult result = (AsyncResult)sendRequest(CMD_INVOKE_OEM_RIL_REQUEST_RAW, oemReq); | 
 | 2826 |             if(result.exception == null) { | 
 | 2827 |                 if (result.result != null) { | 
 | 2828 |                     byte[] responseData = (byte[])(result.result); | 
 | 2829 |                     if(responseData.length > oemResp.length) { | 
 | 2830 |                         Log.w(LOG_TAG, "Buffer to copy response too small: Response length is " + | 
 | 2831 |                                 responseData.length +  "bytes. Buffer Size is " + | 
 | 2832 |                                 oemResp.length + "bytes."); | 
 | 2833 |                     } | 
 | 2834 |                     System.arraycopy(responseData, 0, oemResp, 0, responseData.length); | 
 | 2835 |                     returnValue = responseData.length; | 
 | 2836 |                 } | 
 | 2837 |             } else { | 
 | 2838 |                 CommandException ex = (CommandException) result.exception; | 
 | 2839 |                 returnValue = ex.getCommandError().ordinal(); | 
 | 2840 |                 if(returnValue > 0) returnValue *= -1; | 
 | 2841 |             } | 
 | 2842 |         } catch (RuntimeException e) { | 
 | 2843 |             Log.w(LOG_TAG, "sendOemRilRequestRaw: Runtime Exception"); | 
 | 2844 |             returnValue = (CommandException.Error.GENERIC_FAILURE.ordinal()); | 
 | 2845 |             if(returnValue > 0) returnValue *= -1; | 
 | 2846 |         } | 
 | 2847 |  | 
 | 2848 |         return returnValue; | 
 | 2849 |     } | 
| Wink Saville | 5d475dd | 2014-10-17 15:00:58 -0700 | [diff] [blame] | 2850 |  | 
 | 2851 |     @Override | 
 | 2852 |     public void setRadioCapability(RadioAccessFamily[] rafs) { | 
 | 2853 |         try { | 
 | 2854 |             ProxyController.getInstance().setRadioCapability(rafs); | 
 | 2855 |         } catch (RuntimeException e) { | 
 | 2856 |             Log.w(LOG_TAG, "setRadioCapability: Runtime Exception"); | 
 | 2857 |         } | 
 | 2858 |     } | 
 | 2859 |  | 
 | 2860 |     @Override | 
| Robert Greenwalt | 36b23af | 2015-07-06 17:59:14 -0700 | [diff] [blame] | 2861 |     public int getRadioAccessFamily(int phoneId, String callingPackage) { | 
 | 2862 |         if (!canReadPhoneState(callingPackage, "getRadioAccessFamily")) { | 
 | 2863 |             return RadioAccessFamily.RAF_UNKNOWN; | 
 | 2864 |         } | 
 | 2865 |  | 
| Wink Saville | 5d475dd | 2014-10-17 15:00:58 -0700 | [diff] [blame] | 2866 |         return ProxyController.getInstance().getRadioAccessFamily(phoneId); | 
 | 2867 |     } | 
| Andrew Lee | df14ead | 2014-10-17 14:22:52 -0700 | [diff] [blame] | 2868 |  | 
 | 2869 |     @Override | 
 | 2870 |     public void enableVideoCalling(boolean enable) { | 
 | 2871 |         enforceModifyPermission(); | 
| Tyler Gunn | fdd69de | 2015-12-04 21:24:38 -0800 | [diff] [blame] | 2872 |         ImsManager.setVtSetting(mPhone.getContext(), enable); | 
| Andrew Lee | df14ead | 2014-10-17 14:22:52 -0700 | [diff] [blame] | 2873 |     } | 
 | 2874 |  | 
 | 2875 |     @Override | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 2876 |     public boolean isVideoCallingEnabled(String callingPackage) { | 
 | 2877 |         if (!canReadPhoneState(callingPackage, "isVideoCallingEnabled")) { | 
 | 2878 |             return false; | 
 | 2879 |         } | 
 | 2880 |  | 
| Andrew Lee | 77527ac | 2014-10-21 16:57:39 -0700 | [diff] [blame] | 2881 |         // Check the user preference and the  system-level IMS setting. Even if the user has | 
 | 2882 |         // enabled video calling, if IMS is disabled we aren't able to support video calling. | 
 | 2883 |         // In the long run, we may instead need to check if there exists a connection service | 
 | 2884 |         // which can support video calling. | 
| Andrew Lee | 312e817 | 2014-10-23 17:01:36 -0700 | [diff] [blame] | 2885 |         return ImsManager.isVtEnabledByPlatform(mPhone.getContext()) | 
 | 2886 |                 && ImsManager.isEnhanced4gLteModeSettingEnabledByUser(mPhone.getContext()) | 
| Tyler Gunn | fdd69de | 2015-12-04 21:24:38 -0800 | [diff] [blame] | 2887 |                 && ImsManager.isVtEnabledByUser(mPhone.getContext()); | 
| Andrew Lee | df14ead | 2014-10-17 14:22:52 -0700 | [diff] [blame] | 2888 |     } | 
| Libin.Tang@motorola.com | afe8264 | 2014-12-18 13:27:53 -0600 | [diff] [blame] | 2889 |  | 
| Andrew Lee | a1239f2 | 2015-03-02 17:44:07 -0800 | [diff] [blame] | 2890 |     @Override | 
 | 2891 |     public boolean canChangeDtmfToneLength() { | 
| Jonathan Basseri | 9504c6b | 2015-06-04 14:23:32 -0700 | [diff] [blame] | 2892 |         return mApp.getCarrierConfig().getBoolean(CarrierConfigManager.KEY_DTMF_TYPE_ENABLED_BOOL); | 
| Andrew Lee | a1239f2 | 2015-03-02 17:44:07 -0800 | [diff] [blame] | 2893 |     } | 
 | 2894 |  | 
 | 2895 |     @Override | 
 | 2896 |     public boolean isWorldPhone() { | 
| Jonathan Basseri | 9504c6b | 2015-06-04 14:23:32 -0700 | [diff] [blame] | 2897 |         return mApp.getCarrierConfig().getBoolean(CarrierConfigManager.KEY_WORLD_PHONE_BOOL); | 
| Andrew Lee | a1239f2 | 2015-03-02 17:44:07 -0800 | [diff] [blame] | 2898 |     } | 
 | 2899 |  | 
| Andrew Lee | 9431b83 | 2015-03-09 18:46:45 -0700 | [diff] [blame] | 2900 |     @Override | 
 | 2901 |     public boolean isTtyModeSupported() { | 
 | 2902 |         TelecomManager telecomManager = TelecomManager.from(mPhone.getContext()); | 
 | 2903 |         TelephonyManager telephonyManager = | 
 | 2904 |                 (TelephonyManager) mPhone.getContext().getSystemService(Context.TELEPHONY_SERVICE); | 
| Wooki Wu | 1f82f7a | 2016-02-15 15:59:58 +0800 | [diff] [blame] | 2905 |         return telecomManager.isTtySupported(); | 
| Andrew Lee | 9431b83 | 2015-03-09 18:46:45 -0700 | [diff] [blame] | 2906 |     } | 
 | 2907 |  | 
 | 2908 |     @Override | 
 | 2909 |     public boolean isHearingAidCompatibilitySupported() { | 
 | 2910 |         return mPhone.getContext().getResources().getBoolean(R.bool.hac_enabled); | 
 | 2911 |     } | 
 | 2912 |  | 
| Sanket Padawe | 7310cc7 | 2015-01-14 09:53:20 -0800 | [diff] [blame] | 2913 |     /** | 
 | 2914 |      * Returns the unique device ID of phone, for example, the IMEI for | 
 | 2915 |      * GSM and the MEID for CDMA phones. Return null if device ID is not available. | 
 | 2916 |      * | 
 | 2917 |      * <p>Requires Permission: | 
 | 2918 |      *   {@link android.Manifest.permission#READ_PHONE_STATE READ_PHONE_STATE} | 
 | 2919 |      */ | 
 | 2920 |     @Override | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 2921 |     public String getDeviceId(String callingPackage) { | 
 | 2922 |         if (!canReadPhoneState(callingPackage, "getDeviceId")) { | 
 | 2923 |             return null; | 
 | 2924 |         } | 
 | 2925 |  | 
| Sanket Padawe | 7310cc7 | 2015-01-14 09:53:20 -0800 | [diff] [blame] | 2926 |         final Phone phone = PhoneFactory.getPhone(0); | 
 | 2927 |         if (phone != null) { | 
 | 2928 |             return phone.getDeviceId(); | 
 | 2929 |         } else { | 
 | 2930 |             return null; | 
 | 2931 |         } | 
 | 2932 |     } | 
 | 2933 |  | 
| Libin.Tang@motorola.com | afe8264 | 2014-12-18 13:27:53 -0600 | [diff] [blame] | 2934 |     /* | 
 | 2935 |      * {@hide} | 
 | 2936 |      * Returns the IMS Registration Status | 
 | 2937 |      */ | 
| Santos Cordon | 7a1885b | 2015-02-03 11:15:19 -0800 | [diff] [blame] | 2938 |     @Override | 
| Libin.Tang@motorola.com | afe8264 | 2014-12-18 13:27:53 -0600 | [diff] [blame] | 2939 |     public boolean isImsRegistered() { | 
 | 2940 |         return mPhone.isImsRegistered(); | 
 | 2941 |     } | 
| Santos Cordon | 7a1885b | 2015-02-03 11:15:19 -0800 | [diff] [blame] | 2942 |  | 
 | 2943 |     @Override | 
 | 2944 |     public int getSubIdForPhoneAccount(PhoneAccount phoneAccount) { | 
 | 2945 |         return PhoneUtils.getSubIdForPhoneAccount(phoneAccount); | 
 | 2946 |     } | 
| Nathan Harold | dcfc793 | 2015-03-18 10:01:20 -0700 | [diff] [blame] | 2947 |  | 
| Nathan Harold | c55097a | 2015-03-11 18:14:50 -0700 | [diff] [blame] | 2948 |     /* | 
 | 2949 |      * {@hide} | 
 | 2950 |      * Returns the IMS Registration Status | 
 | 2951 |      */ | 
| Etan Cohen | 3b7a1bc | 2015-05-28 15:57:13 -0700 | [diff] [blame] | 2952 |     public boolean isWifiCallingAvailable() { | 
| Nathan Harold | c55097a | 2015-03-11 18:14:50 -0700 | [diff] [blame] | 2953 |         return mPhone.isWifiCallingEnabled(); | 
 | 2954 |     } | 
 | 2955 |  | 
 | 2956 |     /* | 
 | 2957 |      * {@hide} | 
 | 2958 |      * Returns the IMS Registration Status | 
 | 2959 |      */ | 
| Etan Cohen | 3b7a1bc | 2015-05-28 15:57:13 -0700 | [diff] [blame] | 2960 |     public boolean isVolteAvailable() { | 
| Nathan Harold | c55097a | 2015-03-11 18:14:50 -0700 | [diff] [blame] | 2961 |         return mPhone.isVolteEnabled(); | 
 | 2962 |     } | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 2963 |  | 
| Etan Cohen | 3b7a1bc | 2015-05-28 15:57:13 -0700 | [diff] [blame] | 2964 |     /* | 
 | 2965 |      * {@hide} Returns the IMS Registration Status | 
 | 2966 |      */ | 
 | 2967 |     public boolean isVideoTelephonyAvailable() { | 
 | 2968 |         return mPhone.isVideoEnabled(); | 
 | 2969 |     } | 
 | 2970 |  | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 2971 |     private boolean canReadPhoneState(String callingPackage, String message) { | 
| Etan Cohen | 921655c | 2015-06-24 13:54:50 -0700 | [diff] [blame] | 2972 |         try { | 
| Amit Mahajan | 83ea23b | 2015-07-30 16:05:11 -0700 | [diff] [blame] | 2973 |             mApp.enforceCallingOrSelfPermission( | 
| Etan Cohen | 921655c | 2015-06-24 13:54:50 -0700 | [diff] [blame] | 2974 |                     android.Manifest.permission.READ_PRIVILEGED_PHONE_STATE, message); | 
 | 2975 |  | 
| Amit Mahajan | 83ea23b | 2015-07-30 16:05:11 -0700 | [diff] [blame] | 2976 |             // SKIP checking for run-time permission since caller or self has PRIVILEDGED permission | 
| Etan Cohen | 921655c | 2015-06-24 13:54:50 -0700 | [diff] [blame] | 2977 |             return true; | 
| Amit Mahajan | 785783f | 2015-06-29 10:36:50 -0700 | [diff] [blame] | 2978 |         } catch (SecurityException e) { | 
 | 2979 |             mApp.enforceCallingOrSelfPermission(android.Manifest.permission.READ_PHONE_STATE, | 
 | 2980 |                     message); | 
| Etan Cohen | 921655c | 2015-06-24 13:54:50 -0700 | [diff] [blame] | 2981 |         } | 
| Svet Ganov | b320e18 | 2015-04-16 12:30:10 -0700 | [diff] [blame] | 2982 |  | 
 | 2983 |         if (mAppOps.noteOp(AppOpsManager.OP_READ_PHONE_STATE, Binder.getCallingUid(), | 
 | 2984 |                 callingPackage) != AppOpsManager.MODE_ALLOWED) { | 
 | 2985 |             return false; | 
 | 2986 |         } | 
 | 2987 |  | 
 | 2988 |         return true; | 
 | 2989 |     } | 
| Stuart Scott | 8eef64f | 2015-04-08 15:13:54 -0700 | [diff] [blame] | 2990 |  | 
| Makoto Onuki | fee6934 | 2015-06-29 14:44:50 -0700 | [diff] [blame] | 2991 |     /** | 
| Amit Mahajan | b9b4978 | 2015-09-15 18:16:32 -0700 | [diff] [blame] | 2992 |      * Besides READ_PHONE_STATE, WRITE_SMS and READ_SMS also allow apps to get phone numbers. | 
| Makoto Onuki | fee6934 | 2015-06-29 14:44:50 -0700 | [diff] [blame] | 2993 |      */ | 
 | 2994 |     private boolean canReadPhoneNumber(String callingPackage, String message) { | 
| Makoto Onuki | e4072d1 | 2015-08-03 15:12:23 -0700 | [diff] [blame] | 2995 |         // Default SMS app can always read it. | 
 | 2996 |         if (mAppOps.noteOp(AppOpsManager.OP_WRITE_SMS, | 
 | 2997 |                 Binder.getCallingUid(), callingPackage) == AppOpsManager.MODE_ALLOWED) { | 
 | 2998 |             return true; | 
 | 2999 |         } | 
 | 3000 |         try { | 
 | 3001 |             return canReadPhoneState(callingPackage, message); | 
| Amit Mahajan | b9b4978 | 2015-09-15 18:16:32 -0700 | [diff] [blame] | 3002 |         } catch (SecurityException readPhoneStateSecurityException) { | 
 | 3003 |             try { | 
 | 3004 |                 // Can be read with READ_SMS too. | 
 | 3005 |                 mApp.enforceCallingOrSelfPermission(android.Manifest.permission.READ_SMS, message); | 
 | 3006 |                 return mAppOps.noteOp(AppOpsManager.OP_READ_SMS, | 
 | 3007 |                         Binder.getCallingUid(), callingPackage) == AppOpsManager.MODE_ALLOWED; | 
 | 3008 |             } catch (SecurityException readSmsSecurityException) { | 
 | 3009 |                 // Throw exception with message including both READ_PHONE_STATE and READ_SMS | 
 | 3010 |                 // permissions | 
 | 3011 |                 throw new SecurityException(message + ": Neither user " + Binder.getCallingUid() + | 
 | 3012 |                         " nor current process has " + android.Manifest.permission.READ_PHONE_STATE + | 
 | 3013 |                         " or " + android.Manifest.permission.READ_SMS + "."); | 
 | 3014 |             } | 
| Makoto Onuki | e4072d1 | 2015-08-03 15:12:23 -0700 | [diff] [blame] | 3015 |         } | 
| Makoto Onuki | fee6934 | 2015-06-29 14:44:50 -0700 | [diff] [blame] | 3016 |     } | 
 | 3017 |  | 
| Stuart Scott | 8eef64f | 2015-04-08 15:13:54 -0700 | [diff] [blame] | 3018 |     @Override | 
 | 3019 |     public void factoryReset(int subId) { | 
 | 3020 |         enforceConnectivityInternalPermission(); | 
| Stuart Scott | 981d858 | 2015-04-21 14:09:50 -0700 | [diff] [blame] | 3021 |         if (mUserManager.hasUserRestriction(UserManager.DISALLOW_NETWORK_RESET)) { | 
 | 3022 |             return; | 
 | 3023 |         } | 
 | 3024 |  | 
| Svet Ganov | cc087f8 | 2015-05-12 20:35:54 -0700 | [diff] [blame] | 3025 |         final long identity = Binder.clearCallingIdentity(); | 
 | 3026 |         try { | 
| Stuart Scott | 981d858 | 2015-04-21 14:09:50 -0700 | [diff] [blame] | 3027 |             if (SubscriptionManager.isUsableSubIdValue(subId) && !mUserManager.hasUserRestriction( | 
 | 3028 |                     UserManager.DISALLOW_CONFIG_MOBILE_NETWORKS)) { | 
| Svet Ganov | cc087f8 | 2015-05-12 20:35:54 -0700 | [diff] [blame] | 3029 |                 // Enable data | 
 | 3030 |                 setDataEnabled(subId, true); | 
 | 3031 |                 // Set network selection mode to automatic | 
 | 3032 |                 setNetworkSelectionModeAutomatic(subId); | 
 | 3033 |                 // Set preferred mobile network type to the best available | 
 | 3034 |                 setPreferredNetworkType(subId, Phone.PREFERRED_NT_MODE); | 
 | 3035 |                 // Turn off roaming | 
 | 3036 |                 SubscriptionManager.from(mApp).setDataRoaming(0, subId); | 
 | 3037 |             } | 
 | 3038 |         } finally { | 
 | 3039 |             Binder.restoreCallingIdentity(identity); | 
| Stuart Scott | 8eef64f | 2015-04-08 15:13:54 -0700 | [diff] [blame] | 3040 |         } | 
 | 3041 |     } | 
| Narayan Kamath | 1c496c2 | 2015-04-16 14:40:19 +0100 | [diff] [blame] | 3042 |  | 
 | 3043 |     @Override | 
 | 3044 |     public String getLocaleFromDefaultSim() { | 
 | 3045 |         // We query all subscriptions instead of just the active ones, because | 
 | 3046 |         // this might be called early on in the provisioning flow when the | 
 | 3047 |         // subscriptions potentially aren't active yet. | 
 | 3048 |         final List<SubscriptionInfo> slist = getAllSubscriptionInfoList(); | 
 | 3049 |         if (slist == null || slist.isEmpty()) { | 
 | 3050 |             return null; | 
 | 3051 |         } | 
 | 3052 |  | 
 | 3053 |         // This function may be called very early, say, from the setup wizard, at | 
 | 3054 |         // which point we won't have a default subscription set. If that's the case | 
 | 3055 |         // we just choose the first, which will be valid in "most cases". | 
 | 3056 |         final int defaultSubId = getDefaultSubscription(); | 
 | 3057 |         SubscriptionInfo info = null; | 
 | 3058 |         if (defaultSubId == SubscriptionManager.INVALID_SUBSCRIPTION_ID) { | 
 | 3059 |             info = slist.get(0); | 
 | 3060 |         } else { | 
 | 3061 |             for (SubscriptionInfo item : slist) { | 
 | 3062 |                 if (item.getSubscriptionId() == defaultSubId) { | 
 | 3063 |                     info = item; | 
 | 3064 |                     break; | 
 | 3065 |                 } | 
 | 3066 |             } | 
 | 3067 |  | 
 | 3068 |             if (info == null) { | 
 | 3069 |                 return null; | 
 | 3070 |             } | 
 | 3071 |         } | 
 | 3072 |  | 
 | 3073 |         // Try and fetch the locale from the carrier properties or from the SIM language | 
 | 3074 |         // preferences (EF-PL and EF-LI)... | 
| Tony Hill | 183b2de | 2015-06-24 14:53:58 +0100 | [diff] [blame] | 3075 |         final int mcc = info.getMcc(); | 
| Narayan Kamath | 1c496c2 | 2015-04-16 14:40:19 +0100 | [diff] [blame] | 3076 |         final Phone defaultPhone = getPhone(info.getSubscriptionId()); | 
| Narayan Kamath | 011676f | 2015-07-29 12:04:08 +0100 | [diff] [blame] | 3077 |         String simLanguage = null; | 
| Narayan Kamath | 1c496c2 | 2015-04-16 14:40:19 +0100 | [diff] [blame] | 3078 |         if (defaultPhone != null) { | 
 | 3079 |             final Locale localeFromDefaultSim = defaultPhone.getLocaleFromSimAndCarrierPrefs(); | 
 | 3080 |             if (localeFromDefaultSim != null) { | 
| Narayan Kamath | 011676f | 2015-07-29 12:04:08 +0100 | [diff] [blame] | 3081 |                 if (!localeFromDefaultSim.getCountry().isEmpty()) { | 
| Tony Hill | 183b2de | 2015-06-24 14:53:58 +0100 | [diff] [blame] | 3082 |                     if (DBG) log("Using locale from default SIM:" + localeFromDefaultSim); | 
 | 3083 |                     return localeFromDefaultSim.toLanguageTag(); | 
| Narayan Kamath | 011676f | 2015-07-29 12:04:08 +0100 | [diff] [blame] | 3084 |                 } else { | 
 | 3085 |                     simLanguage = localeFromDefaultSim.getLanguage(); | 
| Tony Hill | 183b2de | 2015-06-24 14:53:58 +0100 | [diff] [blame] | 3086 |                 } | 
| Narayan Kamath | 1c496c2 | 2015-04-16 14:40:19 +0100 | [diff] [blame] | 3087 |             } | 
 | 3088 |         } | 
 | 3089 |  | 
| Narayan Kamath | 011676f | 2015-07-29 12:04:08 +0100 | [diff] [blame] | 3090 |         // The SIM language preferences only store a language (e.g. fr = French), not an | 
 | 3091 |         // exact locale (e.g. fr_FR = French/France). So, if the locale returned from | 
 | 3092 |         // the SIM and carrier preferences does not include a country we add the country | 
 | 3093 |         // determined from the SIM MCC to provide an exact locale. | 
 | 3094 |         final Locale mccLocale = MccTable.getLocaleFromMcc(mPhone.getContext(), mcc, simLanguage); | 
| Tony Hill | 183b2de | 2015-06-24 14:53:58 +0100 | [diff] [blame] | 3095 |         if (mccLocale != null) { | 
 | 3096 |             if (DBG) log("No locale from default SIM, using mcc locale:" + mccLocale); | 
 | 3097 |             return mccLocale.toLanguageTag(); | 
| Narayan Kamath | 1c496c2 | 2015-04-16 14:40:19 +0100 | [diff] [blame] | 3098 |         } | 
 | 3099 |  | 
| Tony Hill | 183b2de | 2015-06-24 14:53:58 +0100 | [diff] [blame] | 3100 |         if (DBG) log("No locale found - returning null"); | 
| Narayan Kamath | 1c496c2 | 2015-04-16 14:40:19 +0100 | [diff] [blame] | 3101 |         return null; | 
 | 3102 |     } | 
 | 3103 |  | 
 | 3104 |     private List<SubscriptionInfo> getAllSubscriptionInfoList() { | 
 | 3105 |         final long identity = Binder.clearCallingIdentity(); | 
 | 3106 |         try { | 
 | 3107 |             return mSubscriptionController.getAllSubInfoList( | 
 | 3108 |                     mPhone.getContext().getOpPackageName()); | 
 | 3109 |         } finally { | 
 | 3110 |             Binder.restoreCallingIdentity(identity); | 
 | 3111 |         } | 
 | 3112 |     } | 
 | 3113 |  | 
 | 3114 |     private List<SubscriptionInfo> getActiveSubscriptionInfoList() { | 
 | 3115 |         final long identity = Binder.clearCallingIdentity(); | 
 | 3116 |         try { | 
 | 3117 |             return mSubscriptionController.getActiveSubscriptionInfoList( | 
 | 3118 |                     mPhone.getContext().getOpPackageName()); | 
 | 3119 |         } finally { | 
 | 3120 |             Binder.restoreCallingIdentity(identity); | 
 | 3121 |         } | 
 | 3122 |     } | 
| Prerepa Viswanadham | 7fcff69 | 2015-06-03 11:20:55 -0700 | [diff] [blame] | 3123 |  | 
 | 3124 |     /** | 
| Adam Lesinski | 903a54c | 2016-04-11 14:49:52 -0700 | [diff] [blame] | 3125 |      * Responds to the ResultReceiver with the {@link android.telephony.ModemActivityInfo} object | 
 | 3126 |      * representing the state of the modem. | 
 | 3127 |      * | 
 | 3128 |      * NOTE: This clears the modem state, so there should only every be one caller. | 
 | 3129 |      * @hide | 
| Prerepa Viswanadham | 7fcff69 | 2015-06-03 11:20:55 -0700 | [diff] [blame] | 3130 |      */ | 
 | 3131 |     @Override | 
| Adam Lesinski | 903a54c | 2016-04-11 14:49:52 -0700 | [diff] [blame] | 3132 |     public void requestModemActivityInfo(ResultReceiver result) { | 
 | 3133 |         enforceModifyPermission(); | 
 | 3134 |  | 
 | 3135 |         ModemActivityInfo info = (ModemActivityInfo) sendRequest(CMD_GET_MODEM_ACTIVITY_INFO, null); | 
 | 3136 |         Bundle bundle = new Bundle(); | 
 | 3137 |         bundle.putParcelable(TelephonyManager.MODEM_ACTIVITY_RESULT_KEY, info); | 
 | 3138 |         result.send(0, bundle); | 
| Prerepa Viswanadham | 7fcff69 | 2015-06-03 11:20:55 -0700 | [diff] [blame] | 3139 |     } | 
| Jack Yu | 85bd38a | 2015-11-09 11:34:32 -0800 | [diff] [blame] | 3140 |  | 
 | 3141 |     /** | 
 | 3142 |      * {@hide} | 
 | 3143 |      * Returns the service state information on specified subscription. | 
 | 3144 |      */ | 
 | 3145 |     @Override | 
 | 3146 |     public ServiceState getServiceStateForSubscriber(int subId, String callingPackage) { | 
 | 3147 |  | 
 | 3148 |         if (!canReadPhoneState(callingPackage, "getServiceStateForSubscriber")) { | 
 | 3149 |             return null; | 
 | 3150 |         } | 
 | 3151 |  | 
 | 3152 |         final Phone phone = getPhone(subId); | 
 | 3153 |         if (phone == null) { | 
 | 3154 |             return null; | 
 | 3155 |         } | 
 | 3156 |  | 
 | 3157 |         return phone.getServiceState(); | 
 | 3158 |     } | 
| Nancy Chen | 31f9ba1 | 2016-01-06 11:42:12 -0800 | [diff] [blame] | 3159 |  | 
 | 3160 |     /** | 
 | 3161 |      * Returns the URI for the per-account voicemail ringtone set in Phone settings. | 
 | 3162 |      * | 
 | 3163 |      * @param accountHandle The handle for the {@link PhoneAccount} for which to retrieve the | 
 | 3164 |      * voicemail ringtone. | 
 | 3165 |      * @return The URI for the ringtone to play when receiving a voicemail from a specific | 
 | 3166 |      * PhoneAccount. | 
 | 3167 |      */ | 
 | 3168 |     @Override | 
 | 3169 |     public Uri getVoicemailRingtoneUri(PhoneAccountHandle accountHandle) { | 
 | 3170 |         final Phone phone = PhoneUtils.getPhoneForPhoneAccountHandle(accountHandle); | 
 | 3171 |         if (phone == null) { | 
 | 3172 |             return null; | 
 | 3173 |         } | 
 | 3174 |  | 
 | 3175 |         return VoicemailNotificationSettingsUtil.getRingtoneUri(phone); | 
 | 3176 |     } | 
 | 3177 |  | 
 | 3178 |     /** | 
 | 3179 |      * Returns whether vibration is set for voicemail notification in Phone settings. | 
 | 3180 |      * | 
 | 3181 |      * @param accountHandle The handle for the {@link PhoneAccount} for which to retrieve the | 
 | 3182 |      * voicemail vibration setting. | 
 | 3183 |      * @return {@code true} if the vibration is set for this PhoneAccount, {@code false} otherwise. | 
 | 3184 |      */ | 
 | 3185 |     @Override | 
 | 3186 |     public boolean isVoicemailVibrationEnabled(PhoneAccountHandle accountHandle) { | 
 | 3187 |         final Phone phone = PhoneUtils.getPhoneForPhoneAccountHandle(accountHandle); | 
 | 3188 |         if (phone == null) { | 
 | 3189 |             return false; | 
 | 3190 |         } | 
 | 3191 |  | 
 | 3192 |         return VoicemailNotificationSettingsUtil.isVibrationEnabled(phone); | 
 | 3193 |     } | 
 | 3194 |  | 
| Youhan Wang | e64578a | 2016-05-02 15:32:42 -0700 | [diff] [blame] | 3195 |     /** | 
 | 3196 |      * Make sure either called from same process as self (phone) or IPC caller has read privilege. | 
 | 3197 |      * | 
 | 3198 |      * @throws SecurityException if the caller does not have the required permission | 
 | 3199 |      */ | 
 | 3200 |     private void enforceReadPrivilegedPermission() { | 
 | 3201 |         mApp.enforceCallingOrSelfPermission(android.Manifest.permission.READ_PRIVILEGED_PHONE_STATE, | 
 | 3202 |                 null); | 
 | 3203 |     } | 
 | 3204 |  | 
 | 3205 |     /** | 
 | 3206 |      * Return the application ID for the app type. | 
 | 3207 |      * | 
 | 3208 |      * @param subId the subscription ID that this request applies to. | 
 | 3209 |      * @param appType the uicc app type. | 
 | 3210 |      * @return Application ID for specificied app type, or null if no uicc. | 
 | 3211 |      */ | 
 | 3212 |     @Override | 
 | 3213 |     public String getAidForAppType(int subId, int appType) { | 
 | 3214 |         enforceReadPrivilegedPermission(); | 
 | 3215 |         Phone phone = getPhone(subId); | 
 | 3216 |         if (phone == null) { | 
 | 3217 |             return null; | 
 | 3218 |         } | 
 | 3219 |         String aid = null; | 
 | 3220 |         try { | 
 | 3221 |             aid = UiccController.getInstance().getUiccCard(phone.getPhoneId()) | 
 | 3222 |                     .getApplicationByType(appType).getAid(); | 
 | 3223 |         } catch (Exception e) { | 
 | 3224 |             Log.e(LOG_TAG, "Not getting aid. Exception ex=" + e); | 
 | 3225 |         } | 
 | 3226 |         return aid; | 
 | 3227 |     } | 
 | 3228 |  | 
| Youhan Wang | 4001d25 | 2016-05-11 10:29:41 -0700 | [diff] [blame] | 3229 |     /** | 
 | 3230 |      * Return the Electronic Serial Number. | 
 | 3231 |      * | 
 | 3232 |      * @param subId the subscription ID that this request applies to. | 
 | 3233 |      * @return ESN or null if error. | 
 | 3234 |      */ | 
 | 3235 |     @Override | 
 | 3236 |     public String getEsn(int subId) { | 
 | 3237 |         enforceReadPrivilegedPermission(); | 
 | 3238 |         Phone phone = getPhone(subId); | 
 | 3239 |         if (phone == null) { | 
 | 3240 |             return null; | 
 | 3241 |         } | 
 | 3242 |         String esn = null; | 
 | 3243 |         try { | 
 | 3244 |             esn = phone.getEsn(); | 
 | 3245 |         } catch (Exception e) { | 
 | 3246 |             Log.e(LOG_TAG, "Not getting ESN. Exception ex=" + e); | 
 | 3247 |         } | 
 | 3248 |         return esn; | 
 | 3249 |     } | 
 | 3250 |  | 
| Sanket Padawe | 99ef1e3 | 2016-05-18 16:12:33 -0700 | [diff] [blame] | 3251 |     /** | 
| Youhan Wang | 66ad5d7 | 2016-07-18 17:56:58 -0700 | [diff] [blame] | 3252 |      * Return the Preferred Roaming List Version. | 
 | 3253 |      * | 
 | 3254 |      * @param subId the subscription ID that this request applies to. | 
 | 3255 |      * @return PRLVersion or null if error. | 
 | 3256 |      */ | 
 | 3257 |     @Override | 
 | 3258 |     public String getCdmaPrlVersion(int subId) { | 
 | 3259 |         enforceReadPrivilegedPermission(); | 
 | 3260 |         Phone phone = getPhone(subId); | 
 | 3261 |         if (phone == null) { | 
 | 3262 |             return null; | 
 | 3263 |         } | 
 | 3264 |         String cdmaPrlVersion = null; | 
 | 3265 |         try { | 
 | 3266 |             cdmaPrlVersion = phone.getCdmaPrlVersion(); | 
 | 3267 |         } catch (Exception e) { | 
 | 3268 |             Log.e(LOG_TAG, "Not getting PRLVersion", e); | 
 | 3269 |         } | 
 | 3270 |         return cdmaPrlVersion; | 
 | 3271 |     } | 
 | 3272 |  | 
 | 3273 |     /** | 
| Sanket Padawe | 99ef1e3 | 2016-05-18 16:12:33 -0700 | [diff] [blame] | 3274 |      * Get snapshot of Telephony histograms | 
 | 3275 |      * @return List of Telephony histograms | 
 | 3276 |      * @hide | 
 | 3277 |      */ | 
 | 3278 |     @Override | 
 | 3279 |     public List<TelephonyHistogram> getTelephonyHistograms() { | 
 | 3280 |         enforceModifyPermissionOrCarrierPrivilege(getDefaultSubscription()); | 
 | 3281 |         return RIL.getTelephonyRILTimingHistograms(); | 
 | 3282 |     } | 
| Meng Wang | 1a7c35a | 2016-05-05 20:56:15 -0700 | [diff] [blame] | 3283 |  | 
 | 3284 |     /** | 
 | 3285 |      * {@hide} | 
 | 3286 |      * Set the allowed carrier list for slotId | 
 | 3287 |      * Require system privileges. In the future we may add this to carrier APIs. | 
 | 3288 |      * | 
 | 3289 |      * @return The number of carriers set successfully, should match length of carriers | 
 | 3290 |      */ | 
 | 3291 |     @Override | 
 | 3292 |     public int setAllowedCarriers(int slotId, List<CarrierIdentifier> carriers) { | 
 | 3293 |         enforceModifyPermission(); | 
 | 3294 |         int subId = SubscriptionManager.getSubId(slotId)[0]; | 
 | 3295 |         int[] retVal = (int[]) sendRequest(CMD_SET_ALLOWED_CARRIERS, carriers, subId); | 
 | 3296 |         return retVal[0]; | 
 | 3297 |     } | 
 | 3298 |  | 
 | 3299 |     /** | 
 | 3300 |      * {@hide} | 
 | 3301 |      * Get the allowed carrier list for slotId. | 
 | 3302 |      * Require system privileges. In the future we may add this to carrier APIs. | 
 | 3303 |      * | 
 | 3304 |      * @return List of {@link android.service.telephony.CarrierIdentifier}; empty list | 
 | 3305 |      * means all carriers are allowed. | 
 | 3306 |      */ | 
 | 3307 |     @Override | 
 | 3308 |     public List<CarrierIdentifier> getAllowedCarriers(int slotId) { | 
 | 3309 |         enforceReadPrivilegedPermission(); | 
 | 3310 |         int subId = SubscriptionManager.getSubId(slotId)[0]; | 
 | 3311 |         return (List<CarrierIdentifier>) sendRequest(CMD_GET_ALLOWED_CARRIERS, null, subId); | 
 | 3312 |     } | 
 | 3313 |  | 
| fionaxu | 59545b4 | 2016-05-25 15:53:37 -0700 | [diff] [blame] | 3314 |     /** | 
 | 3315 |      * Action set from carrier signalling broadcast receivers to enable/disable metered apns | 
 | 3316 |      * @param subId the subscription ID that this action applies to. | 
 | 3317 |      * @param enabled control enable or disable metered apns. | 
 | 3318 |      * {@hide} | 
 | 3319 |      */ | 
 | 3320 |     @Override | 
 | 3321 |     public void carrierActionSetMeteredApnsEnabled(int subId, boolean enabled) { | 
 | 3322 |         enforceModifyPermission(); | 
 | 3323 |         final Phone phone = getPhone(subId); | 
 | 3324 |         if (phone == null) { | 
 | 3325 |             loge("carrierAction: SetMeteredApnsEnabled fails with invalid subId: " + subId); | 
 | 3326 |             return; | 
 | 3327 |         } | 
 | 3328 |         try { | 
 | 3329 |             phone.carrierActionSetMeteredApnsEnabled(enabled); | 
 | 3330 |         } catch (Exception e) { | 
 | 3331 |             Log.e(LOG_TAG, "carrierAction: SetMeteredApnsEnabled fails. Exception ex=" + e); | 
 | 3332 |         } | 
 | 3333 |     } | 
 | 3334 |  | 
 | 3335 |     /** | 
 | 3336 |      * Action set from carrier signalling broadcast receivers to enable/disable radio | 
 | 3337 |      * @param subId the subscription ID that this action applies to. | 
 | 3338 |      * @param enabled control enable or disable radio. | 
 | 3339 |      * {@hide} | 
 | 3340 |      */ | 
 | 3341 |     @Override | 
 | 3342 |     public void carrierActionSetRadioEnabled(int subId, boolean enabled) { | 
 | 3343 |         enforceModifyPermission(); | 
 | 3344 |         final Phone phone = getPhone(subId); | 
 | 3345 |         if (phone == null) { | 
 | 3346 |             loge("carrierAction: SetRadioEnabled fails with invalid sibId: " + subId); | 
 | 3347 |             return; | 
 | 3348 |         } | 
 | 3349 |         try { | 
 | 3350 |             phone.carrierActionSetRadioEnabled(enabled); | 
 | 3351 |         } catch (Exception e) { | 
 | 3352 |             Log.e(LOG_TAG, "carrierAction: SetRadioEnabled fails. Exception ex=" + e); | 
 | 3353 |         } | 
 | 3354 |     } | 
 | 3355 |  | 
| Ta-wei Yen | c236d6b | 2016-06-21 13:33:12 -0700 | [diff] [blame] | 3356 |     /** | 
 | 3357 |      * Called when "adb shell dumpsys phone" is invoked. Dump is also automatically invoked when a | 
 | 3358 |      * bug report is being generated. | 
 | 3359 |      */ | 
 | 3360 |     @Override | 
| Ta-wei Yen | 99282e0 | 2016-06-21 18:19:35 -0700 | [diff] [blame] | 3361 |     protected void dump(FileDescriptor fd, PrintWriter writer, String[] args) { | 
| dcashman | 22b950d | 2016-06-27 11:39:02 -0700 | [diff] [blame] | 3362 |         if (mPhone.getContext().checkCallingOrSelfPermission(android.Manifest.permission.DUMP) | 
 | 3363 |                 != PackageManager.PERMISSION_GRANTED) { | 
 | 3364 |             writer.println("Permission Denial: can't dump Phone from pid=" | 
 | 3365 |                     + Binder.getCallingPid() | 
 | 3366 |                     + ", uid=" + Binder.getCallingUid() | 
 | 3367 |                     + "without permission " | 
 | 3368 |                     + android.Manifest.permission.DUMP); | 
 | 3369 |             return; | 
 | 3370 |         } | 
| Ta-wei Yen | 99282e0 | 2016-06-21 18:19:35 -0700 | [diff] [blame] | 3371 |         DumpsysHandler.dump(mPhone.getContext(), fd, writer, args); | 
| Ta-wei Yen | c236d6b | 2016-06-21 13:33:12 -0700 | [diff] [blame] | 3372 |     } | 
| Jack Yu | eb89b24 | 2016-06-22 13:27:47 -0700 | [diff] [blame] | 3373 |  | 
 | 3374 |     /** | 
 | 3375 |      * Get aggregated video call data usage from all subscriptions since boot. | 
 | 3376 |      * @return total data usage in bytes | 
 | 3377 |      * {@hide} | 
 | 3378 |      */ | 
 | 3379 |     @Override | 
 | 3380 |     public long getVtDataUsage() { | 
 | 3381 |         mApp.enforceCallingOrSelfPermission(android.Manifest.permission.READ_NETWORK_USAGE_HISTORY, | 
 | 3382 |                 null); | 
 | 3383 |  | 
 | 3384 |         // NetworkStatsService keeps tracking the active network interface and identity. It will | 
 | 3385 |         // record the delta with the corresponding network identity. What we need to do here is | 
 | 3386 |         // returning total video call data usage from all subscriptions since boot. | 
 | 3387 |  | 
 | 3388 |         // TODO: Add sub id support in the future. We'll need it when we support DSDA and | 
 | 3389 |         // simultaneous VT calls. | 
 | 3390 |         final Phone[] phones = PhoneFactory.getPhones(); | 
 | 3391 |         long total = 0; | 
 | 3392 |         for (Phone phone : phones) { | 
 | 3393 |             total += phone.getVtDataUsage(); | 
 | 3394 |         } | 
 | 3395 |         return total; | 
 | 3396 |     } | 
| Jack Yu | 75ab295 | 2016-07-08 14:29:33 -0700 | [diff] [blame] | 3397 |  | 
 | 3398 |     /** | 
 | 3399 |      * Policy control of data connection. Usually used when data limit is passed. | 
 | 3400 |      * @param enabled True if enabling the data, otherwise disabling. | 
 | 3401 |      * @param subId Subscription index | 
 | 3402 |      * {@hide} | 
 | 3403 |      */ | 
 | 3404 |     @Override | 
 | 3405 |     public void setPolicyDataEnabled(boolean enabled, int subId) { | 
 | 3406 |         enforceModifyPermission(); | 
 | 3407 |         Phone phone = getPhone(subId); | 
 | 3408 |         if (phone != null) { | 
 | 3409 |             phone.setPolicyDataEnabled(enabled); | 
 | 3410 |         } | 
 | 3411 |     } | 
| Santos Cordon | 7d4ddf6 | 2013-07-10 11:58:08 -0700 | [diff] [blame] | 3412 | } |