1 /* 2 * Copyright (C) 2016 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 package com.android.internal.telephony; 17 18 import android.content.BroadcastReceiver; 19 import android.content.Context; 20 import android.content.Intent; 21 import android.content.IntentFilter; 22 import android.database.ContentObserver; 23 import android.os.AsyncResult; 24 import android.os.Handler; 25 import android.os.Message; 26 import android.os.Registrant; 27 import android.os.RegistrantList; 28 import android.provider.Settings; 29 import android.provider.Telephony; 30 import android.telephony.Rlog; 31 import android.telephony.TelephonyManager; 32 import android.util.LocalLog; 33 import android.util.Log; 34 35 import com.android.internal.annotations.VisibleForTesting; 36 import com.android.internal.util.IndentingPrintWriter; 37 38 import java.io.FileDescriptor; 39 import java.io.PrintWriter; 40 41 /** 42 * Carrier Action Agent(CAA) paired with 43 * {@link com.android.internal.telephony.CarrierSignalAgent CarrierSignalAgent}, 44 * serves as an agent to dispatch carrier actions from carrier apps to different telephony modules, 45 * {@link android.telephony.TelephonyManager#carrierActionSetRadioEnabled(int, boolean) 46 * carrierActionSetRadioEnabled} for example. 47 * 48 * CAA supports dynamic registration where different telephony modules could listen for a specific 49 * carrier action event and implement their own handler. CCA will dispatch the event to all 50 * interested parties and maintain the received action states internally for future inspection. 51 * Each CarrierActionAgent is associated with a phone object. 52 */ 53 public class CarrierActionAgent extends Handler { 54 private static final String LOG_TAG = "CarrierActionAgent"; 55 private static final boolean DBG = true; 56 private static final boolean VDBG = Rlog.isLoggable(LOG_TAG, Log.VERBOSE); 57 58 /** A list of carrier actions */ 59 public static final int CARRIER_ACTION_SET_METERED_APNS_ENABLED = 0; 60 public static final int CARRIER_ACTION_SET_RADIO_ENABLED = 1; 61 public static final int CARRIER_ACTION_RESET = 2; 62 public static final int CARRIER_ACTION_REPORT_DEFAULT_NETWORK_STATUS = 3; 63 public static final int EVENT_APM_SETTINGS_CHANGED = 4; 64 public static final int EVENT_MOBILE_DATA_SETTINGS_CHANGED = 5; 65 public static final int EVENT_DATA_ROAMING_OFF = 6; 66 public static final int EVENT_SIM_STATE_CHANGED = 7; 67 public static final int EVENT_APN_SETTINGS_CHANGED = 8; 68 69 /** Member variables */ 70 private final Phone mPhone; 71 /** registrant list per carrier action */ 72 private RegistrantList mMeteredApnEnableRegistrants = new RegistrantList(); 73 private RegistrantList mRadioEnableRegistrants = new RegistrantList(); 74 private RegistrantList mDefaultNetworkReportRegistrants = new RegistrantList(); 75 /** local log for carrier actions */ 76 private LocalLog mMeteredApnEnabledLog = new LocalLog(10); 77 private LocalLog mRadioEnabledLog = new LocalLog(10); 78 private LocalLog mReportDefaultNetworkStatusLog = new LocalLog(10); 79 /** carrier actions */ 80 private Boolean mCarrierActionOnMeteredApnEnabled = true; 81 private Boolean mCarrierActionOnRadioEnabled = true; 82 private Boolean mCarrierActionReportDefaultNetworkStatus = false; 83 /** content observer for APM change */ 84 private final SettingsObserver mSettingsObserver; 85 86 private final BroadcastReceiver mReceiver = new BroadcastReceiver() { 87 @Override 88 public void onReceive(Context context, Intent intent) { 89 final String action = intent.getAction(); 90 final String iccState = intent.getStringExtra(IccCardConstants.INTENT_KEY_ICC_STATE); 91 if (TelephonyIntents.ACTION_SIM_STATE_CHANGED.equals(action)){ 92 if (intent.getBooleanExtra(TelephonyIntents.EXTRA_REBROADCAST_ON_UNLOCK, false)) { 93 // ignore rebroadcast since carrier apps are direct boot aware. 94 return; 95 } 96 sendMessage(obtainMessage(EVENT_SIM_STATE_CHANGED, iccState)); 97 } 98 } 99 }; 100 101 /** Constructor */ CarrierActionAgent(Phone phone)102 public CarrierActionAgent(Phone phone) { 103 mPhone = phone; 104 mPhone.getContext().registerReceiver(mReceiver, 105 new IntentFilter(TelephonyIntents.ACTION_SIM_STATE_CHANGED)); 106 mSettingsObserver = new SettingsObserver(mPhone.getContext(), this); 107 if (DBG) log("Creating CarrierActionAgent"); 108 } 109 110 @Override handleMessage(Message msg)111 public void handleMessage(Message msg) { 112 // skip notification if the input carrier action is same as the current one. 113 Boolean enabled = getCarrierActionEnabled(msg.what); 114 if (enabled != null && enabled == (boolean) msg.obj) return; 115 switch (msg.what) { 116 case CARRIER_ACTION_SET_METERED_APNS_ENABLED: 117 mCarrierActionOnMeteredApnEnabled = (boolean) msg.obj; 118 log("SET_METERED_APNS_ENABLED: " + mCarrierActionOnMeteredApnEnabled); 119 mMeteredApnEnabledLog.log("SET_METERED_APNS_ENABLED: " 120 + mCarrierActionOnMeteredApnEnabled); 121 int otaspState = (mCarrierActionOnMeteredApnEnabled) 122 ? mPhone.getServiceStateTracker().getOtasp() 123 : TelephonyManager.OTASP_SIM_UNPROVISIONED; 124 mPhone.notifyOtaspChanged(otaspState); 125 mMeteredApnEnableRegistrants.notifyRegistrants( 126 new AsyncResult(null, mCarrierActionOnMeteredApnEnabled, null)); 127 break; 128 case CARRIER_ACTION_SET_RADIO_ENABLED: 129 mCarrierActionOnRadioEnabled = (boolean) msg.obj; 130 log("SET_RADIO_ENABLED: " + mCarrierActionOnRadioEnabled); 131 mRadioEnabledLog.log("SET_RADIO_ENABLED: " + mCarrierActionOnRadioEnabled); 132 mRadioEnableRegistrants.notifyRegistrants( 133 new AsyncResult(null, mCarrierActionOnRadioEnabled, null)); 134 break; 135 case CARRIER_ACTION_REPORT_DEFAULT_NETWORK_STATUS: 136 mCarrierActionReportDefaultNetworkStatus = (boolean) msg.obj; 137 log("CARRIER_ACTION_REPORT_AT_DEFAULT_NETWORK_STATUS: " 138 + mCarrierActionReportDefaultNetworkStatus); 139 mReportDefaultNetworkStatusLog.log("REGISTER_DEFAULT_NETWORK_STATUS: " 140 + mCarrierActionReportDefaultNetworkStatus); 141 mDefaultNetworkReportRegistrants.notifyRegistrants( 142 new AsyncResult(null, mCarrierActionReportDefaultNetworkStatus, null)); 143 break; 144 case CARRIER_ACTION_RESET: 145 log("CARRIER_ACTION_RESET"); 146 carrierActionReset(); 147 break; 148 case EVENT_APM_SETTINGS_CHANGED: 149 log("EVENT_APM_SETTINGS_CHANGED"); 150 if ((Settings.Global.getInt(mPhone.getContext().getContentResolver(), 151 Settings.Global.AIRPLANE_MODE_ON, 0) != 0)) { 152 carrierActionReset(); 153 } 154 break; 155 case EVENT_MOBILE_DATA_SETTINGS_CHANGED: 156 log("EVENT_MOBILE_DATA_SETTINGS_CHANGED"); 157 if (!mPhone.isUserDataEnabled()) carrierActionReset(); 158 break; 159 case EVENT_DATA_ROAMING_OFF: 160 log("EVENT_DATA_ROAMING_OFF"); 161 // reset carrier actions when exit roaming state. 162 carrierActionReset(); 163 break; 164 case EVENT_SIM_STATE_CHANGED: 165 String iccState = (String) msg.obj; 166 if (IccCardConstants.INTENT_VALUE_ICC_LOADED.equals(iccState)) { 167 log("EVENT_SIM_STATE_CHANGED status: " + iccState); 168 carrierActionReset(); 169 String mobileData = Settings.Global.MOBILE_DATA; 170 if (TelephonyManager.getDefault().getSimCount() != 1) { 171 mobileData = mobileData + mPhone.getSubId(); 172 } 173 mSettingsObserver.observe(Settings.Global.getUriFor(mobileData), 174 EVENT_MOBILE_DATA_SETTINGS_CHANGED); 175 mSettingsObserver.observe( 176 Settings.Global.getUriFor(Settings.Global.AIRPLANE_MODE_ON), 177 EVENT_APM_SETTINGS_CHANGED); 178 mSettingsObserver.observe( 179 Telephony.Carriers.CONTENT_URI, EVENT_APN_SETTINGS_CHANGED); 180 if (mPhone.getServiceStateTracker() != null) { 181 mPhone.getServiceStateTracker().registerForDataRoamingOff( 182 this, EVENT_DATA_ROAMING_OFF, null, false); 183 } 184 } else if (IccCardConstants.INTENT_VALUE_ICC_ABSENT.equals(iccState)) { 185 log("EVENT_SIM_STATE_CHANGED status: " + iccState); 186 carrierActionReset(); 187 mSettingsObserver.unobserve(); 188 if (mPhone.getServiceStateTracker() != null) { 189 mPhone.getServiceStateTracker().unregisterForDataRoamingOff(this); 190 } 191 } 192 break; 193 case EVENT_APN_SETTINGS_CHANGED: 194 log("EVENT_APN_SETTINGS_CHANGED"); 195 // Reset carrier actions when APN change. 196 carrierActionReset(); 197 break; 198 default: 199 loge("Unknown carrier action: " + msg.what); 200 } 201 } 202 203 /** 204 * Action set from carrier app to enable/disable radio 205 */ carrierActionSetRadioEnabled(boolean enabled)206 public void carrierActionSetRadioEnabled(boolean enabled) { 207 sendMessage(obtainMessage(CARRIER_ACTION_SET_RADIO_ENABLED, enabled)); 208 } 209 210 /** 211 * Action set from carrier app to enable/disable metered APNs 212 */ carrierActionSetMeteredApnsEnabled(boolean enabled)213 public void carrierActionSetMeteredApnsEnabled(boolean enabled) { 214 sendMessage(obtainMessage(CARRIER_ACTION_SET_METERED_APNS_ENABLED, enabled)); 215 } 216 217 /** 218 * Action set from carrier app to start/stop reporting default network status. 219 */ carrierActionReportDefaultNetworkStatus(boolean report)220 public void carrierActionReportDefaultNetworkStatus(boolean report) { 221 sendMessage(obtainMessage(CARRIER_ACTION_REPORT_DEFAULT_NETWORK_STATUS, report)); 222 } 223 carrierActionReset()224 public void carrierActionReset() { 225 carrierActionReportDefaultNetworkStatus(false); 226 carrierActionSetMeteredApnsEnabled(true); 227 carrierActionSetRadioEnabled(true); 228 // notify configured carrier apps for reset 229 mPhone.getCarrierSignalAgent().notifyCarrierSignalReceivers( 230 new Intent(TelephonyIntents.ACTION_CARRIER_SIGNAL_RESET)); 231 } 232 getRegistrantsFromAction(int action)233 private RegistrantList getRegistrantsFromAction(int action) { 234 switch (action) { 235 case CARRIER_ACTION_SET_METERED_APNS_ENABLED: 236 return mMeteredApnEnableRegistrants; 237 case CARRIER_ACTION_SET_RADIO_ENABLED: 238 return mRadioEnableRegistrants; 239 case CARRIER_ACTION_REPORT_DEFAULT_NETWORK_STATUS: 240 return mDefaultNetworkReportRegistrants; 241 default: 242 loge("Unsupported action: " + action); 243 return null; 244 } 245 } 246 getCarrierActionEnabled(int action)247 private Boolean getCarrierActionEnabled(int action) { 248 switch (action) { 249 case CARRIER_ACTION_SET_METERED_APNS_ENABLED: 250 return mCarrierActionOnMeteredApnEnabled; 251 case CARRIER_ACTION_SET_RADIO_ENABLED: 252 return mCarrierActionOnRadioEnabled; 253 case CARRIER_ACTION_REPORT_DEFAULT_NETWORK_STATUS: 254 return mCarrierActionReportDefaultNetworkStatus; 255 default: 256 loge("Unsupported action: " + action); 257 return null; 258 } 259 } 260 261 /** 262 * Register with CAA for a specific event. 263 * @param action which carrier action registrant is interested in 264 * @param notifyNow if carrier action has once set, notify registrant right after 265 * registering, so that registrants will get the latest carrier action. 266 */ registerForCarrierAction(int action, Handler h, int what, Object obj, boolean notifyNow)267 public void registerForCarrierAction(int action, Handler h, int what, Object obj, 268 boolean notifyNow) { 269 Boolean carrierAction = getCarrierActionEnabled(action); 270 if (carrierAction == null) { 271 throw new IllegalArgumentException("invalid carrier action: " + action); 272 } 273 RegistrantList list = getRegistrantsFromAction(action); 274 Registrant r = new Registrant(h, what, obj); 275 list.add(r); 276 if (notifyNow) { 277 r.notifyRegistrant(new AsyncResult(null, carrierAction, null)); 278 } 279 } 280 281 /** 282 * Unregister with CAA for a specific event. Callers will no longer be notified upon such event. 283 * @param action which carrier action caller is no longer interested in 284 */ unregisterForCarrierAction(Handler h, int action)285 public void unregisterForCarrierAction(Handler h, int action) { 286 RegistrantList list = getRegistrantsFromAction(action); 287 if (list == null) { 288 throw new IllegalArgumentException("invalid carrier action: " + action); 289 } 290 list.remove(h); 291 } 292 293 @VisibleForTesting getContentObserver()294 public ContentObserver getContentObserver() { 295 return mSettingsObserver; 296 } 297 log(String s)298 private void log(String s) { 299 Rlog.d(LOG_TAG, "[" + mPhone.getPhoneId() + "]" + s); 300 } 301 loge(String s)302 private void loge(String s) { 303 Rlog.e(LOG_TAG, "[" + mPhone.getPhoneId() + "]" + s); 304 } 305 logv(String s)306 private void logv(String s) { 307 Rlog.v(LOG_TAG, "[" + mPhone.getPhoneId() + "]" + s); 308 } 309 dump(FileDescriptor fd, PrintWriter pw, String[] args)310 public void dump(FileDescriptor fd, PrintWriter pw, String[] args) { 311 IndentingPrintWriter ipw = new IndentingPrintWriter(pw, " "); 312 pw.println(" mCarrierActionOnMeteredApnsEnabled Log:"); 313 ipw.increaseIndent(); 314 mMeteredApnEnabledLog.dump(fd, ipw, args); 315 ipw.decreaseIndent(); 316 317 pw.println(" mCarrierActionOnRadioEnabled Log:"); 318 ipw.increaseIndent(); 319 mRadioEnabledLog.dump(fd, ipw, args); 320 ipw.decreaseIndent(); 321 322 pw.println(" mCarrierActionReportDefaultNetworkStatus Log:"); 323 ipw.increaseIndent(); 324 mReportDefaultNetworkStatusLog.dump(fd, ipw, args); 325 ipw.decreaseIndent(); 326 } 327 } 328