DcTracker.java revision b237a11044ed842d2865ff8c8716befb06b6ca25
1c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville/*
2c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * Copyright (C) 2006 The Android Open Source Project
3c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville *
4c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * Licensed under the Apache License, Version 2.0 (the "License");
5c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * you may not use this file except in compliance with the License.
6c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * You may obtain a copy of the License at
7c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville *
8c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville *      http://www.apache.org/licenses/LICENSE-2.0
9c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville *
10c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * Unless required by applicable law or agreed to in writing, software
11c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * distributed under the License is distributed on an "AS IS" BASIS,
12c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * See the License for the specific language governing permissions and
14c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * limitations under the License.
15c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville */
16c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
174918296afe1c667e9523cdfc799f558f7ebc2bfbWink Savillepackage com.android.internal.telephony.dataconnection;
18c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
19cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kallaimport android.app.AlarmManager;
20c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.app.PendingIntent;
212b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensenimport android.app.ProgressDialog;
22c9b81a0c05128694c617fcdd67e73821895822feWink Savilleimport android.content.ActivityNotFoundException;
23b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensenimport android.content.BroadcastReceiver;
24c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.content.ContentResolver;
25cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.content.ContentValues;
26c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.content.Context;
27c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.content.Intent;
28c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.content.IntentFilter;
29cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.content.res.Resources;
30c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.database.ContentObserver;
31cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.database.Cursor;
32cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.net.ConnectivityManager;
33c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.net.LinkProperties;
3496cce86cf08e37e0f09ed5057b1196e26b302743Robert Greenwaltimport android.net.NetworkCapabilities;
35cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.net.NetworkConfig;
36cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.net.NetworkUtils;
379c180aedfc9f0d20525c0128487d3500e6c0a715Jason Monkimport android.net.ProxyInfo;
38cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.net.Uri;
39c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.AsyncResult;
403fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwaltimport android.os.Build;
41b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensenimport android.os.Bundle;
42a8467dd0c524787104b1ccdddc5e8af10ba729edWink Savilleimport android.os.Handler;
43071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwaltimport android.os.Looper;
44c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.Message;
451f2a2323572b2f3ab18234726cc1b048c670fbabRobert Greenwaltimport android.os.Messenger;
46a8467dd0c524787104b1ccdddc5e8af10ba729edWink Savilleimport android.os.RegistrantList;
47b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensenimport android.os.ServiceManager;
48c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.SystemClock;
49c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.SystemProperties;
50c9b81a0c05128694c617fcdd67e73821895822feWink Savilleimport android.os.UserHandle;
51c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.provider.Settings;
52cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.provider.Telephony;
53cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.telephony.CellLocation;
540e776303ca82b5bec5db19bb44e0f13b0c7c6400Etan Cohenimport android.telephony.ServiceState;
55c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.telephony.TelephonyManager;
56a8467dd0c524787104b1ccdddc5e8af10ba729edWink Savilleimport android.telephony.SubscriptionManager;
57cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.telephony.cdma.CdmaCellLocation;
58cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.telephony.gsm.GsmCellLocation;
59c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.text.TextUtils;
60cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kallaimport android.util.EventLog;
61bda761320929f714951c328bfec6a51a1978db97Wink Savilleimport android.util.SparseArray;
622b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensenimport android.view.WindowManager;
6399c2e1d6749cfad2a8ca94a47857d8c3bfc09454Wink Savilleimport android.telephony.Rlog;
64c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
65a8467dd0c524787104b1ccdddc5e8af10ba729edWink Savilleimport com.android.internal.telephony.cdma.CDMALTEPhone;
664918296afe1c667e9523cdfc799f558f7ebc2bfbWink Savilleimport com.android.internal.telephony.Phone;
674918296afe1c667e9523cdfc799f558f7ebc2bfbWink Savilleimport com.android.internal.telephony.PhoneBase;
68cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport com.android.internal.telephony.DctConstants;
69cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport com.android.internal.telephony.EventLogTags;
70b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensenimport com.android.internal.telephony.ITelephony;
71c9b81a0c05128694c617fcdd67e73821895822feWink Savilleimport com.android.internal.telephony.TelephonyIntents;
72cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport com.android.internal.telephony.gsm.GSMPhone;
734918296afe1c667e9523cdfc799f558f7ebc2bfbWink Savilleimport com.android.internal.telephony.PhoneConstants;
74cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport com.android.internal.telephony.RILConstants;
75d720945f2be5ea5fe0faf67e67d9ea0e184eba67Alex Yakavenkaimport com.android.internal.telephony.uicc.IccRecords;
76bb36adde615d3d85fa0fc23935197c6bc6a799edAlex Yakavenkaimport com.android.internal.telephony.uicc.UiccController;
77c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport com.android.internal.util.AsyncChannel;
7876f43316a5a6082d601bffd4b6898d0bd81e11fcramimport com.android.internal.util.ArrayUtils;
79c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
80c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport java.io.FileDescriptor;
81c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport java.io.PrintWriter;
82c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport java.util.ArrayList;
8329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwaltimport java.util.Arrays;
84187a39f896f88eb6c5e4306d9595546654825976Wink Savilleimport java.util.concurrent.atomic.AtomicBoolean;
8529c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwaltimport java.util.Objects;
86d86df358b8fe07160caa12147b6e4ad34d378ce6xinheimport java.lang.StringBuilder;
87c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
88a8467dd0c524787104b1ccdddc5e8af10ba729edWink Savilleimport android.provider.Settings;
89a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
90a8467dd0c524787104b1ccdddc5e8af10ba729edWink Savilleimport com.android.internal.telephony.ServiceStateTracker;
91c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville/**
92c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * {@hide}
93c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville */
94454b1dfd508844b42eb775e4ab2359be74d3672bWink Savillepublic final class DcTracker extends DcTrackerBase {
95cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected final String LOG_TAG = "DCT";
96c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
97b3a03455be44a2a0ffbeb757b193d860c1878599Naveen Kalla    /**
98a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville     * List of messages that are waiting to be posted, when data call disconnect
99a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville     * is complete
100a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville     */
101a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private ArrayList<Message> mDisconnectAllCompleteMsgList = new ArrayList<Message>();
102a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
103a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private RegistrantList mAllDataDisconnectedRegistrants = new RegistrantList();
104a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
105a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected int mDisconnectPendingCount = 0;
106a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
107a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    /**
108cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Handles changes to the APN db.
109b3a03455be44a2a0ffbeb757b193d860c1878599Naveen Kalla     */
110cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private class ApnChangeObserver extends ContentObserver {
111cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        public ApnChangeObserver () {
112cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            super(mDataConnectionTracker);
113cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
114c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
115cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        @Override
116cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        public void onChange(boolean selfChange) {
117cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            sendMessage(obtainMessage(DctConstants.EVENT_APN_CHANGED));
118cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
119cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
120c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
121cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    //***** Instance Variables
122c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
123cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private boolean mReregisterOnReconnectFailure = false;
124c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
125c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
126cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    //***** Constants
127c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
128ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    // Used by puppetmaster/*/radio_stress.py
129ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private static final String PUPPET_MASTER_RADIO_STRESS_TEST = "gsm.defaultpdpcontext.active";
130c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
131ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private static final int POLL_PDP_MILLIS = 5 * 1000;
132c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1332b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen    private static final int PROVISIONING_SPINNER_TIMEOUT_MILLIS = 120 * 1000;
1342b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen
1356bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville    static final Uri PREFERAPN_NO_UPDATE_URI_USING_SUBID =
1366bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville                        Uri.parse("content://telephony/carriers/preferapn_no_update/subId/");
137cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    static final String APN_ID = "apn_id";
138cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
139ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private boolean mCanSetPreferApn = false;
140c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
141187a39f896f88eb6c5e4306d9595546654825976Wink Saville    private AtomicBoolean mAttached = new AtomicBoolean(false);
142187a39f896f88eb6c5e4306d9595546654825976Wink Saville
143cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /** Watches for changes to the APN db. */
144cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private ApnChangeObserver mApnObserver;
145cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
146b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    private final String mProvisionActionName;
147b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    private BroadcastReceiver mProvisionBroadcastReceiver;
1482b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen    private ProgressDialog mProvisioningSpinner;
149b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
150a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean mImsRegistrationState = false;
151a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private ApnContext mWaitCleanUpApnContext = null;
152a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private boolean mDeregistrationAlarmState = false;
153a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private PendingIntent mImsDeregistrationDelayIntent = null;
154a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
155cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    //***** Constructor
156454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    public DcTracker(PhoneBase p) {
157cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        super(p);
158cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("GsmDCT.constructor");
159cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
160cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mDataConnectionTracker = this;
161a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        update();
162cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mApnObserver = new ApnChangeObserver();
163cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        p.getContext().getContentResolver().registerContentObserver(
164cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                Telephony.Carriers.CONTENT_URI, true, mApnObserver);
165cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
166d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        initApnContexts();
167d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt
168d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        for (ApnContext apnContext : mApnContexts.values()) {
169d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            // Register the reconnect and restart actions.
170d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            IntentFilter filter = new IntentFilter();
171d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            filter.addAction(INTENT_RECONNECT_ALARM + '.' + apnContext.getApnType());
172d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            filter.addAction(INTENT_RESTART_TRYSETUP_ALARM + '.' + apnContext.getApnType());
173d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            mPhone.getContext().registerReceiver(mIntentReceiver, filter, null, mPhone);
174d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        }
175d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt
17676f43316a5a6082d601bffd4b6898d0bd81e11fcram        // Add Emergency APN to APN setting list by default to support EPDN in sim absent cases
17776f43316a5a6082d601bffd4b6898d0bd81e11fcram        initEmergencyApnSetting();
17876f43316a5a6082d601bffd4b6898d0bd81e11fcram        addEmergencyApnSetting();
179b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
180b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        mProvisionActionName = "com.android.internal.telephony.PROVISION" + p.getPhoneId();
181cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
182c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
183a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void registerForAllEvents() {
184a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.mCi.registerForAvailable(this, DctConstants.EVENT_RADIO_AVAILABLE, null);
185a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.mCi.registerForOffOrNotAvailable(this,
186a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_RADIO_OFF_OR_NOT_AVAILABLE, null);
187a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.mCi.registerForDataNetworkStateChanged(this,
188a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_DATA_STATE_CHANGED, null);
189a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getCallTracker().registerForVoiceCallEnded (this,
190a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_VOICE_CALL_ENDED, null);
191a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getCallTracker().registerForVoiceCallStarted (this,
192a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_VOICE_CALL_STARTED, null);
193a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForDataConnectionAttached(this,
194a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_DATA_CONNECTION_ATTACHED, null);
195a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForDataConnectionDetached(this,
196a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_DATA_CONNECTION_DETACHED, null);
197a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForRoamingOn(this,
198a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_ROAMING_ON, null);
199a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForRoamingOff(this,
200a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_ROAMING_OFF, null);
201a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForPsRestrictedEnabled(this,
202a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                DctConstants.EVENT_PS_RESTRICT_ENABLED, null);
203a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForPsRestrictedDisabled(this,
204a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                DctConstants.EVENT_PS_RESTRICT_DISABLED, null);
205a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville     //   SubscriptionManager.registerForDdsSwitch(this,
206a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville     //          DctConstants.EVENT_CLEAN_UP_ALL_CONNECTIONS, null);
207a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForDataRegStateOrRatChanged(this,
208a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                DctConstants.EVENT_DATA_RAT_CHANGED, null);
209a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
210cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
211cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public void dispose() {
212bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (DBG) log("DcTracker.dispose");
2134dfda5470a2582c0fb543ead6c79ccf598c580e0Robert Greenwalt
214b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        if (mProvisionBroadcastReceiver != null) {
215b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            mPhone.getContext().unregisterReceiver(mProvisionBroadcastReceiver);
216b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            mProvisionBroadcastReceiver = null;
217b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
2182b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        if (mProvisioningSpinner != null) {
2192b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.dismiss();
2202b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner = null;
2212b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        }
222b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
223cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        cleanUpAllConnections(true, null);
224cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
225cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        super.dispose();
226cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
227a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getContext().getContentResolver().unregisterContentObserver(mApnObserver);
228a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mApnContexts.clear();
229a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPrioritySortedApnContexts.clear();
230a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
231a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        destroyDataConnections();
232a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
233a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void unregisterForAllEvents() {
234a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         //Unregister for all events
23522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mPhone.mCi.unregisterForAvailable(this);
23622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mPhone.mCi.unregisterForOffOrNotAvailable(this);
237cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
238a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (r != null) {
239a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            r.unregisterForRecordsLoaded(this);
240a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            mIccRecords.set(null);
241a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
24222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mPhone.mCi.unregisterForDataNetworkStateChanged(this);
243cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getCallTracker().unregisterForVoiceCallEnded(this);
244cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getCallTracker().unregisterForVoiceCallStarted(this);
245cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().unregisterForDataConnectionAttached(this);
246cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().unregisterForDataConnectionDetached(this);
247cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().unregisterForRoamingOn(this);
248cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().unregisterForRoamingOff(this);
249cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().unregisterForPsRestrictedEnabled(this);
250cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().unregisterForPsRestrictedDisabled(this);
251a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        //SubscriptionManager.unregisterForDdsSwitch(this);
252cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
253cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
254bda761320929f714951c328bfec6a51a1978db97Wink Saville    @Override
255bda761320929f714951c328bfec6a51a1978db97Wink Saville    public void incApnRefCount(String name) {
256bda761320929f714951c328bfec6a51a1978db97Wink Saville        ApnContext apnContext = mApnContexts.get(name);
257bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (apnContext != null) {
258bda761320929f714951c328bfec6a51a1978db97Wink Saville            apnContext.incRefCount();
259071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        }
260bda761320929f714951c328bfec6a51a1978db97Wink Saville    }
261071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt
262bda761320929f714951c328bfec6a51a1978db97Wink Saville    @Override
263bda761320929f714951c328bfec6a51a1978db97Wink Saville    public void decApnRefCount(String name) {
264bda761320929f714951c328bfec6a51a1978db97Wink Saville        ApnContext apnContext = mApnContexts.get(name);
265bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (apnContext != null) {
266bda761320929f714951c328bfec6a51a1978db97Wink Saville            apnContext.decRefCount();
267071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        }
268071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt    }
269071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt
270bda761320929f714951c328bfec6a51a1978db97Wink Saville    @Override
271bda761320929f714951c328bfec6a51a1978db97Wink Saville    public boolean isApnSupported(String name) {
272bda761320929f714951c328bfec6a51a1978db97Wink Saville        ApnContext apnContext = mApnContexts.get(name);
273bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (apnContext == null) {
274bda761320929f714951c328bfec6a51a1978db97Wink Saville            loge("Request for unsupported mobile name: " + name);
275bda761320929f714951c328bfec6a51a1978db97Wink Saville            return false;
276071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        }
277bda761320929f714951c328bfec6a51a1978db97Wink Saville        return true;
278bda761320929f714951c328bfec6a51a1978db97Wink Saville    }
279071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt
280bda761320929f714951c328bfec6a51a1978db97Wink Saville    @Override
281bda761320929f714951c328bfec6a51a1978db97Wink Saville    public int getApnPriority(String name) {
282071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        ApnContext apnContext = mApnContexts.get(name);
283071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        if (apnContext == null) {
284bda761320929f714951c328bfec6a51a1978db97Wink Saville            loge("Request for unsupported mobile name: " + name);
285071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        }
286bda761320929f714951c328bfec6a51a1978db97Wink Saville        return apnContext.priority;
287071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt    }
288071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt
289b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    // Turn telephony radio on or off.
290b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    private void setRadio(boolean on) {
291b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        final ITelephony phone = ITelephony.Stub.asInterface(ServiceManager.checkService("phone"));
292b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        try {
293b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            phone.setRadio(on);
294b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        } catch (Exception e) {
295b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            // Ignore.
296b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
297b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    }
298b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
299b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    // Class to handle Intent dispatched with user selects the "Sign-in to network"
300b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    // notification.
301b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    private class ProvisionNotificationBroadcastReceiver extends BroadcastReceiver {
3022b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        private final String mNetworkOperator;
303b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        // Mobile provisioning URL.  Valid while provisioning notification is up.
304b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        // Set prior to notification being posted as URL contains ICCID which
305b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        // disappears when radio is off (which is the case when notification is up).
306b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        private final String mProvisionUrl;
307b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
3082b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        public ProvisionNotificationBroadcastReceiver(String provisionUrl, String networkOperator) {
3092b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mNetworkOperator = networkOperator;
310b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            mProvisionUrl = provisionUrl;
311b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
312b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
313b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        private void setEnableFailFastMobileData(int enabled) {
3146395443719ec3ee0257085945e753d02f603886bRobert Greenwalt            sendMessage(obtainMessage(DctConstants.CMD_SET_ENABLE_FAIL_FAST_MOBILE_DATA, enabled, 0));
315b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
316b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
317b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        private void enableMobileProvisioning() {
318b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            final Message msg = obtainMessage(DctConstants.CMD_ENABLE_MOBILE_PROVISIONING);
319b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            msg.setData(Bundle.forPair(DctConstants.PROVISIONING_URL_KEY, mProvisionUrl));
320b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            sendMessage(msg);
321b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
322b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
323b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        @Override
324b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        public void onReceive(Context context, Intent intent) {
3252b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            // Turning back on the radio can take time on the order of a minute, so show user a
3262b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            // spinner so they know something is going on.
3272b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner = new ProgressDialog(context);
3282b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.setTitle(mNetworkOperator);
3292b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.setMessage(
3302b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    // TODO: Don't borrow "Connecting..." i18n string; give Telephony a version.
3312b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    context.getText(com.android.internal.R.string.media_route_status_connecting));
3322b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.setIndeterminate(true);
3332b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.setCancelable(true);
3342b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            // Allow non-Activity Service Context to create a View.
3352b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.getWindow().setType(
3362b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG);
3372b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.show();
3382b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            // After timeout, hide spinner so user can at least use their device.
3392b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            // TODO: Indicate to user that it is taking an unusually long time to connect?
3402b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            sendMessageDelayed(obtainMessage(DctConstants.CMD_CLEAR_PROVISIONING_SPINNER,
3412b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    mProvisioningSpinner), PROVISIONING_SPINNER_TIMEOUT_MILLIS);
342b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            // This code is almost identical to the old
343b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            // ConnectivityService.handleMobileProvisioningAction code.
344b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            setRadio(true);
345b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            setEnableFailFastMobileData(DctConstants.ENABLED);
346b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            enableMobileProvisioning();
347b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
348b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    }
349b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
350cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
351cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public boolean isApnTypeActive(String type) {
352cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(type);
353cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) return false;
354cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
355ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        return (apnContext.getDcAc() != null);
356cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
357cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
358cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
359cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public boolean isDataPossible(String apnType) {
360cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
361cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) {
362cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return false;
363cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
364cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean apnContextIsEnabled = apnContext.isEnabled();
365cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        DctConstants.State apnContextState = apnContext.getState();
366cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean apnTypePossible = !(apnContextIsEnabled &&
367cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                (apnContextState == DctConstants.State.FAILED));
368cf5205f70eb1eac497164124187a088ecb03fff5Ram        boolean isEmergencyApn = apnContext.getApnType().equals(PhoneConstants.APN_TYPE_EMERGENCY);
369cf5205f70eb1eac497164124187a088ecb03fff5Ram        // Set the emergency APN availability status as TRUE irrespective of conditions checked in
370cf5205f70eb1eac497164124187a088ecb03fff5Ram        // isDataAllowed() like IN_SERVICE, MOBILE DATA status etc.
371cf5205f70eb1eac497164124187a088ecb03fff5Ram        boolean dataAllowed = isEmergencyApn || isDataAllowed();
372cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean possible = dataAllowed && apnTypePossible;
373cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
374ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (VDBG) {
375cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log(String.format("isDataPossible(%s): possible=%b isDataAllowed=%b " +
376cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    "apnTypePossible=%b apnContextisEnabled=%b apnContextState()=%s",
377cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnType, possible, dataAllowed, apnTypePossible,
378cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContextIsEnabled, apnContextState));
379cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
380cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return possible;
381cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
382cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
383cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
384cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void finalize() {
385cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if(DBG) log("finalize");
386cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
387cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
388a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void supplyMessenger() {
389a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        ConnectivityManager cm = (ConnectivityManager)mPhone.getContext().getSystemService(
390a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                Context.CONNECTIVITY_SERVICE);
391a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        cm.supplyMessenger(ConnectivityManager.TYPE_MOBILE, new Messenger(this));
392a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        cm.supplyMessenger(ConnectivityManager.TYPE_MOBILE_MMS, new Messenger(this));
393a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        cm.supplyMessenger(ConnectivityManager.TYPE_MOBILE_SUPL, new Messenger(this));
394a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        cm.supplyMessenger(ConnectivityManager.TYPE_MOBILE_DUN, new Messenger(this));
395a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        cm.supplyMessenger(ConnectivityManager.TYPE_MOBILE_HIPRI, new Messenger(this));
396a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        cm.supplyMessenger(ConnectivityManager.TYPE_MOBILE_FOTA, new Messenger(this));
397a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        cm.supplyMessenger(ConnectivityManager.TYPE_MOBILE_IMS, new Messenger(this));
398a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        cm.supplyMessenger(ConnectivityManager.TYPE_MOBILE_CBS, new Messenger(this));
399cf5205f70eb1eac497164124187a088ecb03fff5Ram        cm.supplyMessenger(ConnectivityManager.TYPE_MOBILE_EMERGENCY, new Messenger(this));
400a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
401a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
4024a9b3afeb2ec4d573eca335a3706392ecf9f281eWink Saville    private ApnContext addApnContext(String type, NetworkConfig networkConfig) {
403071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        ApnContext apnContext = new ApnContext(mPhone.getContext(), type, LOG_TAG, networkConfig,
404071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt                this);
405cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mApnContexts.put(type, apnContext);
4063fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        mPrioritySortedApnContexts.add(apnContext);
407bce3d2575122929bb27ec8a37d56e96da39a3ca2Robert Greenwalt        return apnContext;
408cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
409c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
410d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt    protected void initApnContexts() {
411d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        log("initApnContexts: E");
412d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        // Load device network attributes from resources
413d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        String[] networkConfigStrings = mPhone.getContext().getResources().getStringArray(
414d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                com.android.internal.R.array.networkAttributes);
415d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        for (String networkConfigString : networkConfigStrings) {
416d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            NetworkConfig networkConfig = new NetworkConfig(networkConfigString);
417d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            ApnContext apnContext = null;
418d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt
419d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            switch (networkConfig.type) {
420d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE:
421d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_DEFAULT, networkConfig);
422d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
423d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_MMS:
424d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_MMS, networkConfig);
425d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
426d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_SUPL:
427d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_SUPL, networkConfig);
428d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
429d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_DUN:
430d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_DUN, networkConfig);
431d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
432d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_HIPRI:
433d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_HIPRI, networkConfig);
434d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
435d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_FOTA:
436d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_FOTA, networkConfig);
437d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
438d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_IMS:
439d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_IMS, networkConfig);
440d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
441d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_CBS:
442d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_CBS, networkConfig);
443d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
444d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_IA:
445d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_IA, networkConfig);
446d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
447cf5205f70eb1eac497164124187a088ecb03fff5Ram            case ConnectivityManager.TYPE_MOBILE_EMERGENCY:
448cf5205f70eb1eac497164124187a088ecb03fff5Ram                apnContext = addApnContext(PhoneConstants.APN_TYPE_EMERGENCY, networkConfig);
449cf5205f70eb1eac497164124187a088ecb03fff5Ram                break;
450d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            default:
451d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                log("initApnContexts: skipping unknown type=" + networkConfig.type);
452d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                continue;
453d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            }
454d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            log("initApnContexts: apnContext=" + apnContext);
455d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        }
456d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        log("initApnContexts: X mApnContexts=" + mApnContexts);
457d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt    }
458d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt
459cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
460cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public LinkProperties getLinkProperties(String apnType) {
461cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
462cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext != null) {
463454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville            DcAsyncChannel dcac = apnContext.getDcAc();
464cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (dcac != null) {
465cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("return link properites for " + apnType);
466cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return dcac.getLinkPropertiesSync();
467cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
468cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
469cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("return new LinkProperties");
470cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return new LinkProperties();
471cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
472cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
473608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt    @Override
474608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt    public NetworkCapabilities getNetworkCapabilities(String apnType) {
475608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt        ApnContext apnContext = mApnContexts.get(apnType);
476608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt        if (apnContext!=null) {
477608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt            DcAsyncChannel dataConnectionAc = apnContext.getDcAc();
478608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt            if (dataConnectionAc != null) {
479608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt                if (DBG) {
480608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt                    log("get active pdp is not null, return NetworkCapabilities for " + apnType);
481608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt                }
482608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt                return dataConnectionAc.getNetworkCapabilitiesSync();
483608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt            }
484608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt        }
485608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt        if (DBG) log("return new NetworkCapabilities");
486608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt        return new NetworkCapabilities();
487608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt    }
488cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
489cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
490cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // Return all active apn types
491cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public String[] getActiveApnTypes() {
492cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("get all active apn types");
493cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ArrayList<String> result = new ArrayList<String>();
494cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
495cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
496187a39f896f88eb6c5e4306d9595546654825976Wink Saville            if (mAttached.get() && apnContext.isReady()) {
497cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                result.add(apnContext.getApnType());
498cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla            }
499cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
500c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
501cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return result.toArray(new String[0]);
502cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
503cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
504cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
505cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // Return active apn of specific apn type
506cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public String getActiveApnString(String apnType) {
507ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (VDBG) log( "get active apn string for type:" + apnType);
508cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
509cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext != null) {
510cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            ApnSetting apnSetting = apnContext.getApnSetting();
511cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnSetting != null) {
512cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return apnSetting.apn;
513cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
514cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
515cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return null;
516cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
517cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
518cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
519cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public boolean isApnTypeEnabled(String apnType) {
520cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
521cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) {
522cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return false;
523cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
524cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return apnContext.isEnabled();
525cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
526cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
527cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
528cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void setState(DctConstants.State s) {
529cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("setState should not be used in GSM" + s);
530cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
531cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
532cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // Return state of specific apn type
533cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
534cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public DctConstants.State getState(String apnType) {
535cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
536cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext != null) {
537cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return apnContext.getState();
538c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
539cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return DctConstants.State.FAILED;
540cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
541c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
542c9b81a0c05128694c617fcdd67e73821895822feWink Saville    // Return if apn type is a provisioning apn.
543c9b81a0c05128694c617fcdd67e73821895822feWink Saville    @Override
544c9b81a0c05128694c617fcdd67e73821895822feWink Saville    protected boolean isProvisioningApn(String apnType) {
545c9b81a0c05128694c617fcdd67e73821895822feWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
546c9b81a0c05128694c617fcdd67e73821895822feWink Saville        if (apnContext != null) {
547c9b81a0c05128694c617fcdd67e73821895822feWink Saville            return apnContext.isProvisioningApn();
548c9b81a0c05128694c617fcdd67e73821895822feWink Saville        }
549c9b81a0c05128694c617fcdd67e73821895822feWink Saville        return false;
550c9b81a0c05128694c617fcdd67e73821895822feWink Saville    }
551c9b81a0c05128694c617fcdd67e73821895822feWink Saville
552cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // Return state of overall
553cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
554cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public DctConstants.State getOverallState() {
555cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean isConnecting = false;
556cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean isFailed = true; // All enabled Apns should be FAILED.
557cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean isAnyEnabled = false;
558cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
559cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
560cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnContext.isEnabled()) {
561cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                isAnyEnabled = true;
562cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                switch (apnContext.getState()) {
563cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                case CONNECTED:
564cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                case DISCONNECTING:
565cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (DBG) log("overall state is CONNECTED");
566cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return DctConstants.State.CONNECTED;
567ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                case RETRYING:
568cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                case CONNECTING:
569cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    isConnecting = true;
570cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    isFailed = false;
571cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    break;
572cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                case IDLE:
573cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                case SCANNING:
574cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    isFailed = false;
575cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    break;
576cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                default:
577cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    isAnyEnabled = true;
578cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    break;
579cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
580cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
581c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
582c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
583cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (!isAnyEnabled) { // Nothing enabled. return IDLE.
584cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log( "overall state is IDLE");
585cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return DctConstants.State.IDLE;
586c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
587c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
588cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (isConnecting) {
589cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log( "overall state is CONNECTING");
590cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return DctConstants.State.CONNECTING;
591cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else if (!isFailed) {
592cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log( "overall state is IDLE");
593cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return DctConstants.State.IDLE;
594cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
595cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log( "overall state is FAILED");
596cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return DctConstants.State.FAILED;
597c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
598c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
599c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
600cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
601cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected boolean isApnTypeAvailable(String type) {
602cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (type.equals(PhoneConstants.APN_TYPE_DUN) && fetchDunApn() != null) {
603cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return true;
604c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
605c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
606ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (mAllApnSettings != null) {
607ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            for (ApnSetting apn : mAllApnSettings) {
608cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (apn.canHandleType(type)) {
609cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return true;
610cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
611cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
612c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
613cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return false;
614c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
615c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
616cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
617cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Report on whether data connectivity is enabled for any APN.
618cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @return {@code false} if data connectivity has been explicitly disabled,
619cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * {@code true} otherwise.
620cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
621cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
622cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public boolean getAnyDataEnabled() {
623cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        synchronized (mDataEnabledLock) {
624cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!(mInternalDataEnabled && mUserDataEnabled && sPolicyDataEnabled)) return false;
625cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            for (ApnContext apnContext : mApnContexts.values()) {
626cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // Make sure we don't have a context that is going down
627cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // and is explicitly disabled.
628cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (isDataAllowed(apnContext)) {
629cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return true;
630cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
631cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
632cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return false;
633c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
634c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
635c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
636a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean getAnyDataEnabled(boolean checkUserDataEnabled) {
637a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        synchronized (mDataEnabledLock) {
638a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            if (!(mInternalDataEnabled && (!checkUserDataEnabled || mUserDataEnabled)
639a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                        && (!checkUserDataEnabled || sPolicyDataEnabled)))
640a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                return false;
641a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
642a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            for (ApnContext apnContext : mApnContexts.values()) {
643a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                // Make sure we dont have a context that going down
644a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                // and is explicitly disabled.
645a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                if (isDataAllowed(apnContext)) {
646a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    return true;
647a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                }
648a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            }
649a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            return false;
650a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
651a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
652a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
653cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private boolean isDataAllowed(ApnContext apnContext) {
654cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return apnContext.isReady() && isDataAllowed();
655c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
656c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
657cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    //****** Called from ServiceStateTracker
658c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
659cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Invoked when ServiceStateTracker observes a transition from GPRS
660cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * attach to detach.
661c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
662cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onDataConnectionDetached() {
663cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        /*
664cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * We presently believe it is unnecessary to tear down the PDP context
665cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * when GPRS detaches, but we should stop the network polling.
666cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         */
667cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log ("onDataConnectionDetached: stop polling and notify detached");
668cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        stopNetStatPoll();
669cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        stopDataStallAlarm();
670cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        notifyDataConnection(Phone.REASON_DATA_DETACHED);
671187a39f896f88eb6c5e4306d9595546654825976Wink Saville        mAttached.set(false);
672cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
673c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
674cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void onDataConnectionAttached() {
675cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onDataConnectionAttached");
6767ab10e4710bdb54c6d9a5ee01cd443a42a2689f5Sungmin Choi        mAttached.set(true);
677cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (getOverallState() == DctConstants.State.CONNECTED) {
678cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onDataConnectionAttached: start polling notify attached");
679cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            startNetStatPoll();
680cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            startDataStallAlarm(DATA_STALL_NOT_SUSPECTED);
681cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyDataConnection(Phone.REASON_DATA_ATTACHED);
682cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
683cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // update APN availability so that APN can be enabled.
684cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(Phone.REASON_DATA_ATTACHED);
685cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
68612fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao        if (mAutoAttachOnCreationConfig) {
68712fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao            mAutoAttachOnCreation = true;
68812fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao        }
689ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        setupDataOnConnectableApns(Phone.REASON_DATA_ATTACHED);
690cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
691c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
692cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
693cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected boolean isDataAllowed() {
694cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        final boolean internalDataEnabled;
695cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        synchronized (mDataEnabledLock) {
696cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            internalDataEnabled = mInternalDataEnabled;
697cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
698cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
6999894b3fb2f35e21d9cfd45f233ed093589e14c26sy.yun        boolean attachedState = mAttached.get();
700cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean desiredPowerState = mPhone.getServiceStateTracker().getDesiredPowerState();
701cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
702cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean recordsLoaded = (r != null) ? r.getRecordsLoaded() : false;
703cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
704a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        //FIXME always attach
705a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        boolean psRestricted = mIsPsRestricted;
706a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        int phoneNum = TelephonyManager.getDefault().getPhoneCount();
707a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (phoneNum > 1) {
708a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            attachedState = true;
709a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            psRestricted = false;
710a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
711a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
712cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean allowed =
7139894b3fb2f35e21d9cfd45f233ed093589e14c26sy.yun                    (attachedState || mAutoAttachOnCreation) &&
714cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    recordsLoaded &&
715cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    (mPhone.getState() == PhoneConstants.State.IDLE ||
716cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                     mPhone.getServiceStateTracker().isConcurrentVoiceAndDataAllowed()) &&
717cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    internalDataEnabled &&
718cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    (!mPhone.getServiceState().getRoaming() || getDataOnRoamingEnabled()) &&
719a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    //!mIsPsRestricted &&
720a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    !psRestricted &&
721cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    desiredPowerState;
722cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (!allowed && DBG) {
723cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            String reason = "";
7249894b3fb2f35e21d9cfd45f233ed093589e14c26sy.yun            if (!(attachedState || mAutoAttachOnCreation)) {
7259894b3fb2f35e21d9cfd45f233ed093589e14c26sy.yun                reason += " - Attached= " + attachedState;
726cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
727cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!recordsLoaded) reason += " - SIM not loaded";
728cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (mPhone.getState() != PhoneConstants.State.IDLE &&
729cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    !mPhone.getServiceStateTracker().isConcurrentVoiceAndDataAllowed()) {
730cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                reason += " - PhoneState= " + mPhone.getState();
731cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                reason += " - Concurrent voice and data not allowed";
732cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
733cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!internalDataEnabled) reason += " - mInternalDataEnabled= false";
734cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (mPhone.getServiceState().getRoaming() && !getDataOnRoamingEnabled()) {
735cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                reason += " - Roaming and data roaming not enabled";
736cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
737cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (mIsPsRestricted) reason += " - mIsPsRestricted= true";
738cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!desiredPowerState) reason += " - desiredPowerState= false";
739cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("isDataAllowed: not allowed due to" + reason);
740c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
741cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return allowed;
742cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
743c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
744ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private void setupDataOnConnectableApns(String reason) {
7453fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if (DBG) log("setupDataOnConnectableApns: " + reason);
7463fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
7473fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        for (ApnContext apnContext : mPrioritySortedApnContexts) {
7483fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (DBG) log("setupDataOnConnectableApns: apnContext " + apnContext);
749cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnContext.getState() == DctConstants.State.FAILED) {
750cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setState(DctConstants.State.IDLE);
751cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
752ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (apnContext.isConnectable()) {
753ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                log("setupDataOnConnectableApns: isConnectable() call trySetupData");
754ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                apnContext.setReason(reason);
755ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                trySetupData(apnContext);
756cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
757cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
758c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
759c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
760cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private boolean trySetupData(ApnContext apnContext) {
761cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) {
762cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("trySetupData for type:" + apnContext.getApnType() +
763cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    " due to " + apnContext.getReason() + " apnContext=" + apnContext);
764cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("trySetupData with mIsPsRestricted=" + mIsPsRestricted);
765cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
766cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
767cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (mPhone.getSimulatedRadioControl() != null) {
768cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // Assume data is connected on the simulator
769cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // FIXME  this can be improved
770cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setState(DctConstants.State.CONNECTED);
771cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
772cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
773cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("trySetupData: X We're on the simulator; assuming connected retValue=true");
774cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return true;
775cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
776cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
777cf5205f70eb1eac497164124187a088ecb03fff5Ram        // Allow SETUP_DATA request for E-APN to be completed during emergency call
778cf5205f70eb1eac497164124187a088ecb03fff5Ram        // and MOBILE DATA On/Off cases as well.
779cf5205f70eb1eac497164124187a088ecb03fff5Ram        boolean isEmergencyApn = apnContext.getApnType().equals(PhoneConstants.APN_TYPE_EMERGENCY);
780cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean desiredPowerState = mPhone.getServiceStateTracker().getDesiredPowerState();
781a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        boolean checkUserDataEnabled =
782a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    !(apnContext.getApnType().equals(PhoneConstants.APN_TYPE_IMS));
783cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
784cf5205f70eb1eac497164124187a088ecb03fff5Ram        if (apnContext.isConnectable() && (isEmergencyApn ||
785cf5205f70eb1eac497164124187a088ecb03fff5Ram                (isDataAllowed(apnContext) &&
786cf5205f70eb1eac497164124187a088ecb03fff5Ram                getAnyDataEnabled(checkUserDataEnabled) && !isEmergency()))) {
787ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (apnContext.getState() == DctConstants.State.FAILED) {
788ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (DBG) log("trySetupData: make a FAILED ApnContext IDLE so its reusable");
789ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                apnContext.setState(DctConstants.State.IDLE);
790ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            }
791203e588e3c42a81aa8a56f595119c181a63b12caWink Saville            int radioTech = mPhone.getServiceState().getRilDataRadioTechnology();
792cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnContext.getState() == DctConstants.State.IDLE) {
793203e588e3c42a81aa8a56f595119c181a63b12caWink Saville
794203e588e3c42a81aa8a56f595119c181a63b12caWink Saville                ArrayList<ApnSetting> waitingApns = buildWaitingApns(apnContext.getApnType(),
795203e588e3c42a81aa8a56f595119c181a63b12caWink Saville                        radioTech);
796cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (waitingApns.isEmpty()) {
797ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    notifyNoData(DcFailCause.MISSING_UNKNOWN_APN, apnContext);
798cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    notifyOffApnsOfAvailability(apnContext.getReason());
799cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (DBG) log("trySetupData: X No APN found retValue=false");
800cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return false;
801cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
802cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setWaitingApns(waitingApns);
803cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (DBG) {
804ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                        log ("trySetupData: Create from mAllApnSettings : "
805ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                    + apnListToString(mAllApnSettings));
806cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
807cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
808cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
809cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
810cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) {
811cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("trySetupData: call setupData, waitingApns : "
812cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        + apnListToString(apnContext.getWaitingApns()));
813cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
814203e588e3c42a81aa8a56f595119c181a63b12caWink Saville            boolean retValue = setupData(apnContext, radioTech);
815cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(apnContext.getReason());
816cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
817cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("trySetupData: X retValue=" + retValue);
818cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return retValue;
819cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
820cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!apnContext.getApnType().equals(PhoneConstants.APN_TYPE_DEFAULT)
821ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    && apnContext.isConnectable()) {
822cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPhone.notifyDataConnectionFailed(apnContext.getReason(), apnContext.getApnType());
823ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            }
824cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(apnContext.getReason());
825cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log ("trySetupData: X apnContext not 'ready' retValue=false");
826cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return false;
827cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
828c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
829c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
830cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
831cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // Disabled apn's still need avail/unavail notificiations - send them out
832cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void notifyOffApnsOfAvailability(String reason) {
833cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
834187a39f896f88eb6c5e4306d9595546654825976Wink Saville            if (!mAttached.get() || !apnContext.isReady()) {
835ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (VDBG) log("notifyOffApnOfAvailability type:" + apnContext.getApnType());
836cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPhone.notifyDataConnection(reason != null ? reason : apnContext.getReason(),
837cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                            apnContext.getApnType(),
838cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                            PhoneConstants.DataState.DISCONNECTED);
839cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
840ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (VDBG) {
841187a39f896f88eb6c5e4306d9595546654825976Wink Saville                    log("notifyOffApnsOfAvailability skipped apn due to attached && isReady " +
842cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            apnContext.toString());
843cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
844c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
845c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
846c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
847c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
848cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
849cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * If tearDown is true, this only tears down a CONNECTED session. Presently,
850cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * there is no mechanism for abandoning an CONNECTING session,
851cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * but would likely involve cancelling pending async requests or
852cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * setting a flag or new state to ignore them when they came in
853cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param tearDown true if the underlying DataConnection should be
854cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * disconnected.
855cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param reason reason for the clean up.
8563fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * @return boolean - true if we did cleanup any connections, false if they
8573fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     *                   were already all disconnected.
858cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
8593fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    protected boolean cleanUpAllConnections(boolean tearDown, String reason) {
860cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("cleanUpAllConnections: tearDown=" + tearDown + " reason=" + reason);
8613fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        boolean didDisconnect = false;
862a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        boolean specificdisable = false;
863a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
864a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (!TextUtils.isEmpty(reason)) {
865a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            specificdisable = reason.equals(Phone.REASON_DATA_SPECIFIC_DISABLED);
866a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
867cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
868cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
8693fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (apnContext.isDisconnected() == false) didDisconnect = true;
870a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            if (specificdisable) {
871a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                if (!apnContext.getApnType().equals(PhoneConstants.APN_TYPE_IMS)) {
872a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    if (DBG) log("ApnConextType: " + apnContext.getApnType());
873a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    apnContext.setReason(reason);
874a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    cleanUpConnection(tearDown, apnContext);
875a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                }
876a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            } else {
877a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                // TODO - only do cleanup if not disconnected
878a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                apnContext.setReason(reason);
879a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                cleanUpConnection(tearDown, apnContext);
880a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            }
881c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
882cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
883cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        stopNetStatPoll();
884cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        stopDataStallAlarm();
885cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
886cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // TODO: Do we need mRequestedApnType?
887cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mRequestedApnType = PhoneConstants.APN_TYPE_DEFAULT;
888a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
889a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("cleanUpConnection: mDisconnectPendingCount = " + mDisconnectPendingCount);
890a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (tearDown && mDisconnectPendingCount == 0) {
891a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            notifyDataDisconnectComplete();
892a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            notifyAllDataDisconnected();
893a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
894a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
8953fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        return didDisconnect;
896cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
897cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
898cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
899cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Cleanup all connections.
900cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *
901cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * TODO: Cleanup only a specified connection passed as a parameter.
902cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *       Also, make sure when you clean up a conn, if it is last apply
903cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *       logic as though it is cleanupAllConnections
904cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *
905cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param cause for the clean up.
906cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
907cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
908cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
909cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onCleanUpAllConnections(String cause) {
910cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        cleanUpAllConnections(true, cause);
911cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
912cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
913a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void cleanUpConnection(boolean tearDown, ApnContext apnContext) {
914cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
915cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) {
916cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("cleanUpConnection: apn context is null");
917cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
918cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
919cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
920454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville        DcAsyncChannel dcac = apnContext.getDcAc();
921cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) {
922cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("cleanUpConnection: E tearDown=" + tearDown + " reason=" + apnContext.getReason() +
923cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    " apnContext=" + apnContext);
924cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
925cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (tearDown) {
926cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnContext.isDisconnected()) {
927cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // The request is tearDown and but ApnContext is not connected.
928cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // If apnContext is not enabled anymore, break the linkage to the DCAC/DC.
929cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setState(DctConstants.State.IDLE);
930cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (!apnContext.isReady()) {
9314750c8c11836338b024e159f04f0cbd13c7444b9Wink Saville                    if (dcac != null) {
9324750c8c11836338b024e159f04f0cbd13c7444b9Wink Saville                        dcac.tearDown(apnContext, "", null);
9334750c8c11836338b024e159f04f0cbd13c7444b9Wink Saville                    }
934cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setDataConnectionAc(null);
935cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
936cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
937cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // Connection is still there. Try to clean up.
938cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (dcac != null) {
939cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (apnContext.getState() != DctConstants.State.DISCONNECTING) {
940cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        boolean disconnectAll = false;
941cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (PhoneConstants.APN_TYPE_DUN.equals(apnContext.getApnType())) {
942a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                            // CAF_MSIM is this below condition required.
943a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                            // if (PhoneConstants.APN_TYPE_DUN.equals(PhoneConstants.APN_TYPE_DEFAULT)) {
9441484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                            if (teardownForDun()) {
945cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                if (DBG) log("tearing down dedicated DUN connection");
946cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                // we need to tear it down - we brought it up just for dun and
947cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                // other people are camped on it and now dun is done.  We need
948cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                // to stop using it and let the normal apn list get used to find
949cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                // connections for the remaining desired connections
950cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                disconnectAll = true;
951cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            }
952cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
953cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (DBG) {
954cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            log("cleanUpConnection: tearing down" + (disconnectAll ? " all" :""));
955cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
956cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        Message msg = obtainMessage(DctConstants.EVENT_DISCONNECT_DONE, apnContext);
957cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (disconnectAll) {
958ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            apnContext.getDcAc().tearDownAll(apnContext.getReason(), msg);
959cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        } else {
960ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            apnContext.getDcAc()
961cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                .tearDown(apnContext, apnContext.getReason(), msg);
962cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
963cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        apnContext.setState(DctConstants.State.DISCONNECTING);
964a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                        mDisconnectPendingCount++;
965cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
966cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
967cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    // apn is connected but no reference to dcac.
968cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    // Should not be happen, but reset the state in case.
969cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setState(DctConstants.State.IDLE);
970cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    mPhone.notifyDataConnection(apnContext.getReason(),
971cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                                apnContext.getApnType());
972cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
973cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
974cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
975cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // force clean up the data connection.
976ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (dcac != null) dcac.reqReset();
977cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setState(DctConstants.State.IDLE);
978cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
979cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setDataConnectionAc(null);
980cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
981cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
982ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        // Make sure reconnection alarm is cleaned up if there is no ApnContext
983cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // associated to the connection.
984cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (dcac != null) {
985ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            cancelReconnectAlarm(apnContext);
986c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
987cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) {
988cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("cleanUpConnection: X tearDown=" + tearDown + " reason=" + apnContext.getReason() +
989ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    " apnContext=" + apnContext + " dcac=" + apnContext.getDcAc());
990cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
991cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
992c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
993cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
9941484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt     * Determine if DUN connection is special and we need to teardown on start/stop
9951484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt     */
9961484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt    private boolean teardownForDun() {
9971484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        // CDMA always needs to do this the profile id is correct
9981484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        final int rilRat = mPhone.getServiceState().getRilDataRadioTechnology();
9991484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        if (ServiceState.isCdma(rilRat)) return true;
10001484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt
10011484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        return (fetchDunApn() != null);
10021484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt    }
10031484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt
10041484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt    /**
1005ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * Cancels the alarm associated with apnContext.
1006cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *
1007ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * @param apnContext on which the alarm should be stopped.
1008cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
1009ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private void cancelReconnectAlarm(ApnContext apnContext) {
1010ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (apnContext == null) return;
1011cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1012ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        PendingIntent intent = apnContext.getReconnectIntent();
1013cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1014cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (intent != null) {
1015cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                AlarmManager am =
1016cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    (AlarmManager) mPhone.getContext().getSystemService(Context.ALARM_SERVICE);
1017cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                am.cancel(intent);
1018ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                apnContext.setReconnectIntent(null);
1019cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1020c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1021c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1022cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
1023cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param types comma delimited list of APN types
1024cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @return array of APN types
1025cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
1026cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private String[] parseTypes(String types) {
1027c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        String[] result;
1028cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // If unset, set to DEFAULT.
1029cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (types == null || types.equals("")) {
1030c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            result = new String[1];
1031cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            result[0] = PhoneConstants.APN_TYPE_ALL;
1032cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1033cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            result = types.split(",");
1034c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
1035c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return result;
1036c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1037c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
103861cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan    private boolean imsiMatches(String imsiDB, String imsiSIM) {
1039fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        // Note: imsiDB value has digit number or 'x' character for seperating USIM information
1040fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        // for MVNO operator. And then digit number is matched at same order and 'x' character
1041fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        // could replace by any digit number.
1042fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        // ex) if imsiDB inserted '310260x10xxxxxx' for GG Operator,
1043fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        //     that means first 6 digits, 8th and 9th digit
1044fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        //     should be set in USIM for GG Operator.
1045fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        int len = imsiDB.length();
1046fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        int idxCompare = 0;
1047fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi
1048fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        if (len <= 0) return false;
1049fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        if (len > imsiSIM.length()) return false;
1050fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi
1051fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        for (int idx=0; idx<len; idx++) {
1052fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi            char c = imsiDB.charAt(idx);
1053fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi            if ((c == 'x') || (c == 'X') || (c == imsiSIM.charAt(idx))) {
1054fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                continue;
1055fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi            } else {
1056fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                return false;
1057fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi            }
1058fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        }
1059fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        return true;
1060fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi    }
1061fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi
10623262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal    @Override
10633262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal    protected boolean mvnoMatches(IccRecords r, String mvnoType, String mvnoMatchData) {
10643262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal        if (mvnoType.equalsIgnoreCase("spn")) {
1065fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi            if ((r.getServiceProviderName() != null) &&
10663262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                    r.getServiceProviderName().equalsIgnoreCase(mvnoMatchData)) {
1067fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                return true;
1068fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi            }
10693262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal        } else if (mvnoType.equalsIgnoreCase("imsi")) {
1070fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi            String imsiSIM = r.getIMSI();
10713262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal            if ((imsiSIM != null) && imsiMatches(mvnoMatchData, imsiSIM)) {
1072fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                return true;
1073fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi            }
10743262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal        } else if (mvnoType.equalsIgnoreCase("gid")) {
1075fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi            String gid1 = r.getGid1();
10763262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal            int mvno_match_data_length = mvnoMatchData.length();
107759a71354e0169c7877ffd56d476ddd65ecf5a88dRobert Greenwalt            if ((gid1 != null) && (gid1.length() >= mvno_match_data_length) &&
10783262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                    gid1.substring(0, mvno_match_data_length).equalsIgnoreCase(mvnoMatchData)) {
1079fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                return true;
1080fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi            }
1081fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        }
1082fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        return false;
1083fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi    }
1084fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi
1085796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan    @Override
1086796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan    protected boolean isPermanentFail(DcFailCause dcFailCause) {
1087796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan        return (dcFailCause.isPermanentFail() &&
1088796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan                (mAttached.get() == false || dcFailCause != DcFailCause.SIGNAL_LOST));
1089796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan    }
1090796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan
1091fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi    private ApnSetting makeApnSetting(Cursor cursor) {
1092fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        String[] types = parseTypes(
1093fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.TYPE)));
1094fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        ApnSetting apn = new ApnSetting(
1095fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers._ID)),
1096fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.NUMERIC)),
1097fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.NAME)),
1098fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.APN)),
1099fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                NetworkUtils.trimV4AddrZeros(
1100fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getString(
1101fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getColumnIndexOrThrow(Telephony.Carriers.PROXY))),
1102fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.PORT)),
1103fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                NetworkUtils.trimV4AddrZeros(
1104fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getString(
1105fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getColumnIndexOrThrow(Telephony.Carriers.MMSC))),
1106fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                NetworkUtils.trimV4AddrZeros(
1107fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getString(
1108fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getColumnIndexOrThrow(Telephony.Carriers.MMSPROXY))),
1109fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.MMSPORT)),
1110fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.USER)),
1111fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.PASSWORD)),
1112fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.AUTH_TYPE)),
1113fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                types,
1114fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.PROTOCOL)),
1115fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(
1116fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        Telephony.Carriers.ROAMING_PROTOCOL)),
1117fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getInt(cursor.getColumnIndexOrThrow(
1118fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        Telephony.Carriers.CARRIER_ENABLED)) == 1,
11199d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.BEARER)),
11209d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.PROFILE_ID)),
11219d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(
11229d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                        Telephony.Carriers.MODEM_COGNITIVE)) == 1,
11239d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.MAX_CONNS)),
11249d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(
11259d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                        Telephony.Carriers.WAIT_TIME)),
1126e9701717e43cc5aacbcf624f77a53be92350662cw                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.MAX_CONNS_TIME)),
11273262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.MTU)),
11283262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.MVNO_TYPE)),
11293262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.MVNO_MATCH_DATA)));
1130fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        return apn;
1131fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi    }
1132fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi
1133cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private ArrayList<ApnSetting> createApnList(Cursor cursor) {
11343262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal        ArrayList<ApnSetting> mnoApns = new ArrayList<ApnSetting>();
11353262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal        ArrayList<ApnSetting> mvnoApns = new ArrayList<ApnSetting>();
1136fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        IccRecords r = mIccRecords.get();
1137fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi
1138cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (cursor.moveToFirst()) {
1139cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            do {
11403262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                ApnSetting apn = makeApnSetting(cursor);
11413262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                if (apn == null) {
11423262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                    continue;
11433262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                }
11443262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal
11453262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                if (apn.hasMvnoParams()) {
11463262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                    if (r != null && mvnoMatches(r, apn.mvnoType, apn.mvnoMatchData)) {
11473262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                        mvnoApns.add(apn);
1148fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                    }
1149fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                } else {
11503262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                    mnoApns.add(apn);
1151fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                }
1152cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } while (cursor.moveToNext());
1153cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
11543262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal
11553262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal        ArrayList<ApnSetting> result = mvnoApns.isEmpty() ? mnoApns : mvnoApns;
1156cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("createApnList: X result=" + result);
1157c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return result;
1158c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1159c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1160454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    private boolean dataConnectionNotInUse(DcAsyncChannel dcac) {
1161ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (DBG) log("dataConnectionNotInUse: check if dcac is inuse dcac=" + dcac);
1162cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
1163ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (apnContext.getDcAc() == dcac) {
1164cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("dataConnectionNotInUse: in use by apnContext=" + apnContext);
1165cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return false;
1166cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1167cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1168cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // TODO: Fix retry handling so free DataConnections have empty apnlists.
1169cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // Probably move retry handling into DataConnections and reduce complexity
1170cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // of DCT.
1171cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("dataConnectionNotInUse: tearDownAll");
1172ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        dcac.tearDownAll("No connection", null);
1173cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("dataConnectionNotInUse: not in use return true");
1174cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return true;
1175cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1176cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1177454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    private DcAsyncChannel findFreeDataConnection() {
1178454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville        for (DcAsyncChannel dcac : mDataConnectionAcHashMap.values()) {
1179cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (dcac.isInactiveSync() && dataConnectionNotInUse(dcac)) {
1180cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) {
1181cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("findFreeDataConnection: found free DataConnection=" +
1182ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                        " dcac=" + dcac);
1183cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1184ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                return dcac;
1185cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1186cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1187cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        log("findFreeDataConnection: NO free DataConnection");
1188cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return null;
1189cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1190cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1191203e588e3c42a81aa8a56f595119c181a63b12caWink Saville    private boolean setupData(ApnContext apnContext, int radioTech) {
1192cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("setupData: apnContext=" + apnContext);
1193ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        ApnSetting apnSetting;
11941484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        DcAsyncChannel dcac = null;
1195cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1196ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        apnSetting = apnContext.getNextWaitingApn();
1197ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (apnSetting == null) {
1198cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("setupData: return for no apn found!");
1199cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return false;
1200cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1201cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1202231c3c6271a27b5f03444f4d24b52c41c46566abEtan Cohen        int profileId = apnSetting.profileId;
1203231c3c6271a27b5f03444f4d24b52c41c46566abEtan Cohen        if (profileId == 0) {
1204231c3c6271a27b5f03444f4d24b52c41c46566abEtan Cohen            profileId = getApnProfileID(apnContext.getApnType());
1205231c3c6271a27b5f03444f4d24b52c41c46566abEtan Cohen        }
1206231c3c6271a27b5f03444f4d24b52c41c46566abEtan Cohen
12071484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        // On CDMA, if we're explicitly asking for DUN, we need have
12081484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        // a dun-profiled connection so we can't share an existing one
12091484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        // On GSM/LTE we can share existing apn connections provided they support
12101484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        // this type.
12111484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        if (apnContext.getApnType() != PhoneConstants.APN_TYPE_DUN ||
12121484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                teardownForDun() == false) {
12131484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt            dcac = checkForCompatibleConnectedApnContext(apnContext);
12141484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt            if (dcac != null) {
12151484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // Get the dcacApnSetting for the connection we want to share.
12161484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                ApnSetting dcacApnSetting = dcac.getApnSettingSync();
12171484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                if (dcacApnSetting != null) {
12181484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                    // Setting is good, so use it.
12191484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                    apnSetting = dcacApnSetting;
12201484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                }
1221ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            }
1222ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        }
1223ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (dcac == null) {
12243fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (isOnlySingleDcAllowed(radioTech)) {
12253fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                if (isHigherPriorityApnContextActive(apnContext)) {
12263fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    if (DBG) {
12273fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        log("setupData: Higher priority ApnContext active.  Ignoring call");
12283fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    }
12293fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    return false;
12303fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                }
12313fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
12323fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                // Only lower priority calls left.  Disconnect them all in this single PDP case
12333fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                // so that we can bring up the requested higher priority call (once we receive
12343fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                // repsonse for deactivate request for the calls we are about to disconnect
12353fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                if (cleanUpAllConnections(true, Phone.REASON_SINGLE_PDN_ARBITRATION)) {
12363fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    // If any call actually requested to be disconnected, means we can't
12373fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    // bring up this connection yet as we need to wait for those data calls
12383fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    // to be disconnected.
12393fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    if (DBG) log("setupData: Some calls are disconnecting first.  Wait and retry");
12403fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    return false;
12413fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                }
12423fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
12433fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                // No other calls are active, so proceed
12443fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                if (DBG) log("setupData: Single pdp. Continue setting up data call.");
12453fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            }
12463fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
1247ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            dcac = findFreeDataConnection();
1248cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1249ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (dcac == null) {
1250ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                dcac = createDataConnection();
1251cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1252cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1253ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (dcac == null) {
1254ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (DBG) log("setupData: No free DataConnection and couldn't create one, WEIRD");
1255cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return false;
1256cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1257cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1258ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (DBG) log("setupData: dcac=" + dcac + " apnSetting=" + apnSetting);
1259cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1260cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        apnContext.setDataConnectionAc(dcac);
1261ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        apnContext.setApnSetting(apnSetting);
1262cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        apnContext.setState(DctConstants.State.CONNECTING);
1263cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
1264cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1265cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        Message msg = obtainMessage();
1266cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        msg.what = DctConstants.EVENT_DATA_SETUP_COMPLETE;
1267cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        msg.obj = apnContext;
126812fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao        dcac.bringUp(apnContext, getInitialMaxRetry(), profileId, radioTech, mAutoAttachOnCreation,
126912fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao                msg);
1270cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1271cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("setupData: initing!");
1272cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return true;
1273cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1274cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1275c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
1276cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Handles changes to the APN database.
1277c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
1278cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void onApnChanged() {
1279cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        DctConstants.State overallState = getOverallState();
1280cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean isDisconnected = (overallState == DctConstants.State.IDLE ||
1281cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                overallState == DctConstants.State.FAILED);
1282cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1283cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (mPhone instanceof GSMPhone) {
1284cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // The "current" may no longer be valid.  MMS depends on this to send properly. TBD
1285cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            ((GSMPhone)mPhone).updateCurrentCarrierInProvider();
1286cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1287cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1288cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // TODO: It'd be nice to only do this if the changed entrie(s)
1289cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // match the current operator.
1290cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onApnChanged: createAllApnList and cleanUpAllConnections");
1291cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        createAllApnList();
12925d5eea6ed231163c225144316b0d1913d48678a4Sungmin Choi        setInitialAttachApn();
1293cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        cleanUpAllConnections(!isDisconnected, Phone.REASON_APN_CHANGED);
1294bda761320929f714951c328bfec6a51a1978db97Wink Saville
12958f6f52e4f7598e44cea1f9e5f4781291f9060d1dWink Saville        // FIXME: See bug 17426028 maybe no conditional is needed.
12968f6f52e4f7598e44cea1f9e5f4781291f9060d1dWink Saville        if (mPhone.getSubId() == SubscriptionManager.getDefaultDataSubId()) {
1297ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            setupDataOnConnectableApns(Phone.REASON_APN_CHANGED);
1298c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
1299c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1300c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1301c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
1302cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param cid Connection id provided from RIL.
1303cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @return DataConnectionAc associated with specified cid.
1304c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
1305454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    private DcAsyncChannel findDataConnectionAcByCid(int cid) {
1306454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville        for (DcAsyncChannel dcac : mDataConnectionAcHashMap.values()) {
1307cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (dcac.getCidSync() == cid) {
1308cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return dcac;
1309cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1310c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
1311cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return null;
1312c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1313c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1314cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // TODO: For multiple Active APNs not exactly sure how to do this.
1315c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    @Override
1316cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void gotoIdleAndNotifyDataConnection(String reason) {
1317cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("gotoIdleAndNotifyDataConnection: reason=" + reason);
1318cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        notifyDataConnection(reason);
1319cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mActiveApn = null;
1320cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1321cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
13223fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    /**
13233fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * "Active" here means ApnContext isEnabled() and not in FAILED state
13243fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * @param apnContext to compare with
13253fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * @return true if higher priority active apn found
13263fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     */
13273fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    private boolean isHigherPriorityApnContextActive(ApnContext apnContext) {
13283fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        for (ApnContext otherContext : mPrioritySortedApnContexts) {
13293fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (apnContext.getApnType().equalsIgnoreCase(otherContext.getApnType())) return false;
13303fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (otherContext.isEnabled() && otherContext.getState() != DctConstants.State.FAILED) {
13313fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                return true;
13323fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            }
13333fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        }
13343fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        return false;
13353fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    }
13363fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
13373fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    /**
13383fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * Reports if we support multiple connections or not.
13393fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * This is a combination of factors, based on carrier and RAT.
13403fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * @param rilRadioTech the RIL Radio Tech currently in use
13413fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * @return true if only single DataConnection is allowed
13423fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     */
13433fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    private boolean isOnlySingleDcAllowed(int rilRadioTech) {
13443fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        int[] singleDcRats = mPhone.getContext().getResources().getIntArray(
13453fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                com.android.internal.R.array.config_onlySingleDcAllowed);
13463fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        boolean onlySingleDcAllowed = false;
13473fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if (Build.IS_DEBUGGABLE &&
13483fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                SystemProperties.getBoolean("persist.telephony.test.singleDc", false)) {
13493fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            onlySingleDcAllowed = true;
13503fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        }
13513fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if (singleDcRats != null) {
13523fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            for (int i=0; i < singleDcRats.length && onlySingleDcAllowed == false; i++) {
13533fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                if (rilRadioTech == singleDcRats[i]) onlySingleDcAllowed = true;
13543fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            }
13553fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        }
13563fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
13573fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if (DBG) log("isOnlySingleDcAllowed(" + rilRadioTech + "): " + onlySingleDcAllowed);
13583fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        return onlySingleDcAllowed;
13593fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    }
13603fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
1361cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1362cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void restartRadio() {
1363cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("restartRadio: ************TURN OFF RADIO**************");
1364cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        cleanUpAllConnections(true, Phone.REASON_RADIO_TURNED_OFF);
1365cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().powerOffRadioSafely(this);
1366cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        /* Note: no need to call setRadioPower(true).  Assuming the desired
1367cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * radio power state is still ON (as tracked by ServiceStateTracker),
1368cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * ServiceStateTracker will call setRadioPower when it receives the
1369cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * RADIO_STATE_CHANGED notification for the power off.  And if the
1370cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * desired power state has changed in the interim, we don't want to
1371cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * override it with an unconditional power on.
1372cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         */
1373cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1374cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        int reset = Integer.parseInt(SystemProperties.get("net.ppp.reset-by-timeout", "0"));
1375cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        SystemProperties.set("net.ppp.reset-by-timeout", String.valueOf(reset+1));
1376cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1377cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1378cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
1379cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Return true if data connection need to be setup after disconnected due to
1380cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * reason.
1381cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *
1382cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param reason the reason why data is disconnected
1383cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @return true if try setup data connection is need for this reason
1384cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
13853fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    private boolean retryAfterDisconnected(ApnContext apnContext) {
1386cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean retry = true;
13873fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        String reason = apnContext.getReason();
1388cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
13893fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if ( Phone.REASON_RADIO_TURNED_OFF.equals(reason) ||
13903fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                (isOnlySingleDcAllowed(mPhone.getServiceState().getRilDataRadioTechnology())
13913fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                 && isHigherPriorityApnContextActive(apnContext))) {
1392cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            retry = false;
1393cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1394cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return retry;
1395cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1396cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1397cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void startAlarmForReconnect(int delay, ApnContext apnContext) {
1398cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        String apnType = apnContext.getApnType();
139974672e8ee972f12406b72551261b4cc7e0651933Wink Saville
140074672e8ee972f12406b72551261b4cc7e0651933Wink Saville        Intent intent = new Intent(INTENT_RECONNECT_ALARM + "." + apnType);
140174672e8ee972f12406b72551261b4cc7e0651933Wink Saville        intent.putExtra(INTENT_RECONNECT_ALARM_EXTRA_REASON, apnContext.getReason());
1402cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        intent.putExtra(INTENT_RECONNECT_ALARM_EXTRA_TYPE, apnType);
1403cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1404a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        // Get current sub id.
1405b237a11044ed842d2865ff8c8716befb06b6ca25Wink Saville        int subId = SubscriptionManager.getDefaultDataSubId();
1406a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        intent.putExtra(PhoneConstants.SUBSCRIPTION_KEY, subId);
1407a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
140874672e8ee972f12406b72551261b4cc7e0651933Wink Saville        if (DBG) {
140974672e8ee972f12406b72551261b4cc7e0651933Wink Saville            log("startAlarmForReconnect: delay=" + delay + " action=" + intent.getAction()
141074672e8ee972f12406b72551261b4cc7e0651933Wink Saville                    + " apn=" + apnContext);
141174672e8ee972f12406b72551261b4cc7e0651933Wink Saville        }
141274672e8ee972f12406b72551261b4cc7e0651933Wink Saville
1413ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        PendingIntent alarmIntent = PendingIntent.getBroadcast (mPhone.getContext(), 0,
1414ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                        intent, PendingIntent.FLAG_UPDATE_CURRENT);
1415ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        apnContext.setReconnectIntent(alarmIntent);
1416ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        mAlarmManager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,
1417ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                SystemClock.elapsedRealtime() + delay, alarmIntent);
1418ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    }
1419cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1420ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private void startAlarmForRestartTrySetup(int delay, ApnContext apnContext) {
1421ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        String apnType = apnContext.getApnType();
142274672e8ee972f12406b72551261b4cc7e0651933Wink Saville        Intent intent = new Intent(INTENT_RESTART_TRYSETUP_ALARM + "." + apnType);
1423ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        intent.putExtra(INTENT_RESTART_TRYSETUP_ALARM_EXTRA_TYPE, apnType);
1424cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
142574672e8ee972f12406b72551261b4cc7e0651933Wink Saville        if (DBG) {
142674672e8ee972f12406b72551261b4cc7e0651933Wink Saville            log("startAlarmForRestartTrySetup: delay=" + delay + " action=" + intent.getAction()
142774672e8ee972f12406b72551261b4cc7e0651933Wink Saville                    + " apn=" + apnContext);
142874672e8ee972f12406b72551261b4cc7e0651933Wink Saville        }
1429cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        PendingIntent alarmIntent = PendingIntent.getBroadcast (mPhone.getContext(), 0,
1430cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                        intent, PendingIntent.FLAG_UPDATE_CURRENT);
1431ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        apnContext.setReconnectIntent(alarmIntent);
1432ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        mAlarmManager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,
1433cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                SystemClock.elapsedRealtime() + delay, alarmIntent);
1434cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1435cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1436ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private void notifyNoData(DcFailCause lastFailCauseCode,
1437cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                              ApnContext apnContext) {
1438cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log( "notifyNoData: type=" + apnContext.getApnType());
1439796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan        if (isPermanentFail(lastFailCauseCode)
1440cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            && (!apnContext.getApnType().equals(PhoneConstants.APN_TYPE_DEFAULT))) {
1441cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mPhone.notifyDataConnectionFailed(apnContext.getReason(), apnContext.getApnType());
1442cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1443cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1444cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1445cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void onRecordsLoaded() {
1446cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onRecordsLoaded: createAllApnList");
144712fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao        mAutoAttachOnCreationConfig = mPhone.getContext().getResources()
144812fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao                .getBoolean(com.android.internal.R.bool.config_auto_attach_data_on_creation);
1449bda761320929f714951c328bfec6a51a1978db97Wink Saville
1450bda761320929f714951c328bfec6a51a1978db97Wink Saville        mAutoAttachOnCreationConfig = true;
1451cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        createAllApnList();
14525d5eea6ed231163c225144316b0d1913d48678a4Sungmin Choi        setInitialAttachApn();
145322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mPhone.mCi.getRadioState().isOn()) {
1454cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onRecordsLoaded: notifying data availability");
1455cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(Phone.REASON_SIM_LOADED);
1456cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1457bda761320929f714951c328bfec6a51a1978db97Wink Saville        setupDataOnConnectableApns(Phone.REASON_SIM_LOADED);
1458cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1459cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1460cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1461cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onSetDependencyMet(String apnType, boolean met) {
1462cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // don't allow users to tweak hipri to work around default dependency not met
1463cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (PhoneConstants.APN_TYPE_HIPRI.equals(apnType)) return;
1464cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1465cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
1466cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) {
1467cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            loge("onSetDependencyMet: ApnContext not found in onSetDependencyMet(" +
1468cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnType + ", " + met + ")");
1469cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
1470cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1471cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        applyNewState(apnContext, apnContext.isEnabled(), met);
1472cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (PhoneConstants.APN_TYPE_DEFAULT.equals(apnType)) {
1473cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // tie actions on default to similar actions on HIPRI regarding dependencyMet
1474cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext = mApnContexts.get(PhoneConstants.APN_TYPE_HIPRI);
1475cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnContext != null) applyNewState(apnContext, apnContext.isEnabled(), met);
1476cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1477cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1478c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1479cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void applyNewState(ApnContext apnContext, boolean enabled, boolean met) {
1480cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean cleanup = false;
1481cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean trySetup = false;
1482cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) {
1483cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("applyNewState(" + apnContext.getApnType() + ", " + enabled +
1484cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    "(" + apnContext.isEnabled() + "), " + met + "(" +
1485cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.getDependencyMet() +"))");
1486cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1487cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext.isReady()) {
1488305122cd621385652826cf7d8cd4e651dc6b5e9fRobert Greenwalt            cleanup = true;
14892428693913ae731d4ace3414429f5e91af24ea36Wink Saville            if (enabled && met) {
14902428693913ae731d4ace3414429f5e91af24ea36Wink Saville                DctConstants.State state = apnContext.getState();
14912428693913ae731d4ace3414429f5e91af24ea36Wink Saville                switch(state) {
14922428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case CONNECTING:
14932428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case SCANNING:
14942428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case CONNECTED:
14952428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case DISCONNECTING:
14962428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        // We're "READY" and active so just return
14972428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        if (DBG) log("applyNewState: 'ready' so return");
14982428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        return;
14992428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case IDLE:
15002428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        // fall through: this is unexpected but if it happens cleanup and try setup
15012428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case FAILED:
15022428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case RETRYING: {
15032428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        // We're "READY" but not active so disconnect (cleanup = true) and
15042428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        // connect (trySetup = true) to be sure we retry the connection.
15052428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        trySetup = true;
15062428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        apnContext.setReason(Phone.REASON_DATA_ENABLED);
15072428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        break;
15082428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    }
15092428693913ae731d4ace3414429f5e91af24ea36Wink Saville                }
1510305122cd621385652826cf7d8cd4e651dc6b5e9fRobert Greenwalt            } else if (met) {
1511cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setReason(Phone.REASON_DATA_DISABLED);
1512305122cd621385652826cf7d8cd4e651dc6b5e9fRobert Greenwalt                // If ConnectivityService has disabled this network, stop trying to bring
1513305122cd621385652826cf7d8cd4e651dc6b5e9fRobert Greenwalt                // it up, but do not tear it down - ConnectivityService will do that
1514305122cd621385652826cf7d8cd4e651dc6b5e9fRobert Greenwalt                // directly by talking with the DataConnection.
15151484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                //
15161484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // This doesn't apply to DUN, however.  Those connections have special
15171484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // requirements from carriers and we need stop using them when the dun
15181484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // request goes away.  This applies to both CDMA and GSM because they both
15191484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // can declare the DUN APN sharable by default traffic, thus still satisfying
15201484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // those requests and not torn down organically.
15211484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                if (apnContext.getApnType() == PhoneConstants.APN_TYPE_DUN && teardownForDun()) {
15221484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                    cleanup = true;
15231484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                } else {
15241484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                    cleanup = false;
15251484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                }
1526cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
1527cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setReason(Phone.REASON_DATA_DEPENDENCY_UNMET);
1528cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1529cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1530cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (enabled && met) {
1531cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (apnContext.isEnabled()) {
1532cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setReason(Phone.REASON_DATA_DEPENDENCY_MET);
1533cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
1534cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setReason(Phone.REASON_DATA_ENABLED);
1535c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
1536cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (apnContext.getState() == DctConstants.State.FAILED) {
1537cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setState(DctConstants.State.IDLE);
1538cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1539cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                trySetup = true;
1540cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1541cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1542cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        apnContext.setEnabled(enabled);
1543cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        apnContext.setDependencyMet(met);
1544cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (cleanup) cleanUpConnection(true, apnContext);
1545cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (trySetup) trySetupData(apnContext);
1546cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1547c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1548454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    private DcAsyncChannel checkForCompatibleConnectedApnContext(ApnContext apnContext) {
1549cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        String apnType = apnContext.getApnType();
1550cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnSetting dunSetting = null;
1551cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1552cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (PhoneConstants.APN_TYPE_DUN.equals(apnType)) {
1553cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            dunSetting = fetchDunApn();
1554cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1555ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (DBG) {
1556ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            log("checkForCompatibleConnectedApnContext: apnContext=" + apnContext );
1557ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        }
1558cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1559454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville        DcAsyncChannel potentialDcac = null;
1560ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        ApnContext potentialApnCtx = null;
1561ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        for (ApnContext curApnCtx : mApnContexts.values()) {
1562454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville            DcAsyncChannel curDcac = curApnCtx.getDcAc();
1563a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            log("curDcac: " + curDcac);
1564ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (curDcac != null) {
1565ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                ApnSetting apnSetting = curApnCtx.getApnSetting();
1566a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                log("apnSetting: " + apnSetting);
1567cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (dunSetting != null) {
1568cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (dunSetting.equals(apnSetting)) {
1569ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                        switch (curApnCtx.getState()) {
1570cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            case CONNECTED:
1571cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                if (DBG) {
1572ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                    log("checkForCompatibleConnectedApnContext:"
1573ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                            + " found dun conn=" + curDcac
1574ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                            + " curApnCtx=" + curApnCtx);
1575cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                }
1576ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                return curDcac;
1577ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            case RETRYING:
1578cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            case CONNECTING:
1579ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                potentialDcac = curDcac;
1580ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                potentialApnCtx = curApnCtx;
1581cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            default:
1582cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                // Not connected, potential unchanged
1583cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                break;
1584cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
1585cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
1586cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else if (apnSetting != null && apnSetting.canHandleType(apnType)) {
1587ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    switch (curApnCtx.getState()) {
1588cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        case CONNECTED:
1589cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            if (DBG) {
1590ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                log("checkForCompatibleConnectedApnContext:"
1591ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                        + " found canHandle conn=" + curDcac
1592ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                        + " curApnCtx=" + curApnCtx);
1593cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            }
1594ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            return curDcac;
1595ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                        case RETRYING:
1596cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        case CONNECTING:
1597ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            potentialDcac = curDcac;
1598ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            potentialApnCtx = curApnCtx;
1599cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        default:
1600cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            // Not connected, potential unchanged
1601cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            break;
1602cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
1603cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1604ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            } else {
1605ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (VDBG) {
1606ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    log("checkForCompatibleConnectedApnContext: not conn curApnCtx=" + curApnCtx);
1607ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                }
1608cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1609cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1610ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (potentialDcac != null) {
1611cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) {
1612ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                log("checkForCompatibleConnectedApnContext: found potential conn=" + potentialDcac
1613ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                        + " curApnCtx=" + potentialApnCtx);
1614cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1615ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            return potentialDcac;
1616cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1617c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1618ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (DBG) log("checkForCompatibleConnectedApnContext: NO conn apnContext=" + apnContext);
1619cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return null;
1620cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1621c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1622cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1623cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onEnableApn(int apnId, int enabled) {
1624cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnIdToType(apnId));
1625cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) {
1626cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            loge("onEnableApn(" + apnId + ", " + enabled + "): NO ApnContext");
1627cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
1628cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1629cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // TODO change our retry manager to use the appropriate numbers for the new APN
1630cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onEnableApn: apnContext=" + apnContext + " call applyNewState");
1631cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        applyNewState(apnContext, enabled == DctConstants.ENABLED, apnContext.getDependencyMet());
1632cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1633c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1634cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1635cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // TODO: We shouldnt need this.
1636cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected boolean onTrySetupData(String reason) {
1637cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onTrySetupData: reason=" + reason);
1638ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        setupDataOnConnectableApns(reason);
1639cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return true;
1640cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1641c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1642cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected boolean onTrySetupData(ApnContext apnContext) {
1643cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onTrySetupData: apnContext=" + apnContext);
1644cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return trySetupData(apnContext);
1645cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1646c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1647cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1648cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onRoamingOff() {
1649cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onRoamingOff");
1650c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
16516bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (!mUserDataEnabled) return;
1652c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1653bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (getDataOnRoamingEnabled() == false) {
1654cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(Phone.REASON_ROAMING_OFF);
1655bda761320929f714951c328bfec6a51a1978db97Wink Saville            setupDataOnConnectableApns(Phone.REASON_ROAMING_OFF);
1656cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1657cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyDataConnection(Phone.REASON_ROAMING_OFF);
1658cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1659cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1660cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1661cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1662cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onRoamingOn() {
16636bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (DBG) log("onRoamingOn");
1664bda761320929f714951c328bfec6a51a1978db97Wink Saville
16656bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (!mUserDataEnabled) return;
1666cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1667bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (getDataOnRoamingEnabled()) {
1668cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onRoamingOn: setup data on roaming");
1669ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            setupDataOnConnectableApns(Phone.REASON_ROAMING_ON);
1670cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyDataConnection(Phone.REASON_ROAMING_ON);
1671cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1672cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onRoamingOn: Tear down data connection on roaming.");
1673cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cleanUpAllConnections(true, Phone.REASON_ROAMING_ON);
1674cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(Phone.REASON_ROAMING_ON);
1675cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1676cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1677cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1678cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1679cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onRadioAvailable() {
1680cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onRadioAvailable");
1681cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (mPhone.getSimulatedRadioControl() != null) {
1682cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // Assume data is connected on the simulator
1683cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // FIXME  this can be improved
1684cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // setState(DctConstants.State.CONNECTED);
1685cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyDataConnection(null);
1686cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1687cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("onRadioAvailable: We're on the simulator; assuming data is connected");
1688cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1689cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1690cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
1691cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (r != null && r.getRecordsLoaded()) {
1692cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(null);
1693cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1694cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1695cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (getOverallState() != DctConstants.State.IDLE) {
1696cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cleanUpConnection(true, null);
1697cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1698cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1699cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1700cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1701cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onRadioOffOrNotAvailable() {
1702cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // Make sure our reconnect delay starts at the initial value
1703cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // next time the radio comes on
1704cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1705cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mReregisterOnReconnectFailure = false;
1706cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1707cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (mPhone.getSimulatedRadioControl() != null) {
1708cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // Assume data is connected on the simulator
1709cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // FIXME  this can be improved
1710cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("We're on the simulator; assuming radio off is meaningless");
1711cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1712cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onRadioOffOrNotAvailable: is off and clean up all connections");
1713cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cleanUpAllConnections(false, Phone.REASON_RADIO_TURNED_OFF);
1714cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1715cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        notifyOffApnsOfAvailability(null);
1716cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1717cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1718c9b81a0c05128694c617fcdd67e73821895822feWink Saville    @Override
1719c9b81a0c05128694c617fcdd67e73821895822feWink Saville    protected void completeConnection(ApnContext apnContext) {
1720c9b81a0c05128694c617fcdd67e73821895822feWink Saville        boolean isProvApn = apnContext.isProvisioningApn();
1721c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1722c9b81a0c05128694c617fcdd67e73821895822feWink Saville        if (DBG) log("completeConnection: successful, notify the world apnContext=" + apnContext);
1723c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1724c9b81a0c05128694c617fcdd67e73821895822feWink Saville        if (mIsProvisioning && !TextUtils.isEmpty(mProvisioningUrl)) {
1725c9b81a0c05128694c617fcdd67e73821895822feWink Saville            if (DBG) {
1726c9b81a0c05128694c617fcdd67e73821895822feWink Saville                log("completeConnection: MOBILE_PROVISIONING_ACTION url="
1727c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        + mProvisioningUrl);
1728c9b81a0c05128694c617fcdd67e73821895822feWink Saville            }
1729c8dc0c8244aac9f3985a53bc94b8ec2e295db430Robert Greenwalt            Intent newIntent = Intent.makeMainSelectorActivity(Intent.ACTION_MAIN,
1730c8dc0c8244aac9f3985a53bc94b8ec2e295db430Robert Greenwalt                    Intent.CATEGORY_APP_BROWSER);
1731c8dc0c8244aac9f3985a53bc94b8ec2e295db430Robert Greenwalt            newIntent.setData(Uri.parse(mProvisioningUrl));
1732c9b81a0c05128694c617fcdd67e73821895822feWink Saville            newIntent.setFlags(Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT |
1733c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    Intent.FLAG_ACTIVITY_NEW_TASK);
1734c9b81a0c05128694c617fcdd67e73821895822feWink Saville            try {
1735c9b81a0c05128694c617fcdd67e73821895822feWink Saville                mPhone.getContext().startActivity(newIntent);
1736c9b81a0c05128694c617fcdd67e73821895822feWink Saville            } catch (ActivityNotFoundException e) {
1737c9b81a0c05128694c617fcdd67e73821895822feWink Saville                loge("completeConnection: startActivityAsUser failed" + e);
1738c9b81a0c05128694c617fcdd67e73821895822feWink Saville            }
1739c9b81a0c05128694c617fcdd67e73821895822feWink Saville        }
1740c9b81a0c05128694c617fcdd67e73821895822feWink Saville        mIsProvisioning = false;
1741c9b81a0c05128694c617fcdd67e73821895822feWink Saville        mProvisioningUrl = null;
17422b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        if (mProvisioningSpinner != null) {
17432b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            sendMessage(obtainMessage(DctConstants.CMD_CLEAR_PROVISIONING_SPINNER,
17442b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    mProvisioningSpinner));
17452b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        }
1746c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1747c9b81a0c05128694c617fcdd67e73821895822feWink Saville        mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
1748c9b81a0c05128694c617fcdd67e73821895822feWink Saville        startNetStatPoll();
1749c9b81a0c05128694c617fcdd67e73821895822feWink Saville        startDataStallAlarm(DATA_STALL_NOT_SUSPECTED);
1750c9b81a0c05128694c617fcdd67e73821895822feWink Saville    }
1751c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1752ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    /**
1753ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * A SETUP (aka bringUp) has completed, possibly with an error. If
1754ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * there is an error this method will call {@link #onDataSetupCompleteError}.
1755ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     */
1756cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1757cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onDataSetupComplete(AsyncResult ar) {
1758608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt
1759ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        DcFailCause cause = DcFailCause.UNKNOWN;
1760cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean handleError = false;
1761cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = null;
1762cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1763cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if(ar.userObj instanceof ApnContext){
1764cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext = (ApnContext)ar.userObj;
1765cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1766cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            throw new RuntimeException("onDataSetupComplete: No apnContext");
1767cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1768cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1769cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (ar.exception == null) {
1770454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville            DcAsyncChannel dcac = apnContext.getDcAc();
1771cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1772cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (RADIO_TESTS) {
1773cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // Note: To change radio.test.onDSC.null.dcac from command line you need to
1774cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // adb root and adb remount and from the command line you can only change the
1775cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // value to 1 once. To change it a second time you can reboot or execute
1776cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // adb shell stop and then adb shell start. The command line to set the value is:
1777ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                // adb shell sqlite3 /data/data/com.android.providers.settings/databases/settings.db "insert into system (name,value) values ('radio.test.onDSC.null.dcac', '1');"
1778cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                ContentResolver cr = mPhone.getContext().getContentResolver();
1779cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                String radioTestProperty = "radio.test.onDSC.null.dcac";
1780cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (Settings.System.getInt(cr, radioTestProperty, 0) == 1) {
1781cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("onDataSetupComplete: " + radioTestProperty +
1782cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            " is true, set dcac to null and reset property to false");
1783cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    dcac = null;
1784cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    Settings.System.putInt(cr, radioTestProperty, 0);
1785cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("onDataSetupComplete: " + radioTestProperty + "=" +
1786cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            Settings.System.getInt(mPhone.getContext().getContentResolver(),
1787cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                    radioTestProperty, -1));
1788cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1789c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
1790cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (dcac == null) {
1791cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("onDataSetupComplete: no connection to DC, handle as error");
1792ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                cause = DcFailCause.CONNECTION_TO_DATACONNECTIONAC_BROKEN;
1793cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                handleError = true;
1794cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
1795cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                ApnSetting apn = apnContext.getApnSetting();
1796cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) {
1797cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("onDataSetupComplete: success apn=" + (apn == null ? "unknown" : apn.apn));
1798cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1799cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (apn != null && apn.proxy != null && apn.proxy.length() != 0) {
1800cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    try {
1801cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        String port = apn.port;
1802cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (TextUtils.isEmpty(port)) port = "8080";
18039c180aedfc9f0d20525c0128487d3500e6c0a715Jason Monk                        ProxyInfo proxy = new ProxyInfo(apn.proxy,
1804cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                Integer.parseInt(port), null);
1805cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        dcac.setLinkPropertiesHttpProxySync(proxy);
1806cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    } catch (NumberFormatException e) {
1807cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        loge("onDataSetupComplete: NumberFormatException making ProxyProperties (" +
1808cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                apn.port + "): " + e);
1809cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
1810cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1811cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1812cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // everything is setup
1813cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if(TextUtils.equals(apnContext.getApnType(),PhoneConstants.APN_TYPE_DEFAULT)) {
1814ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    SystemProperties.set(PUPPET_MASTER_RADIO_STRESS_TEST, "true");
181522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                    if (mCanSetPreferApn && mPreferredApn == null) {
1816cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (DBG) log("onDataSetupComplete: PREFERED APN is null");
1817cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        mPreferredApn = apn;
1818cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (mPreferredApn != null) {
1819cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            setPreferredApn(mPreferredApn.id);
1820cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
1821cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
1822cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
1823ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    SystemProperties.set(PUPPET_MASTER_RADIO_STRESS_TEST, "false");
1824cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1825c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1826c9b81a0c05128694c617fcdd67e73821895822feWink Saville                // A connection is setup
1827c9b81a0c05128694c617fcdd67e73821895822feWink Saville                apnContext.setState(DctConstants.State.CONNECTED);
1828c9b81a0c05128694c617fcdd67e73821895822feWink Saville                boolean isProvApn = apnContext.isProvisioningApn();
1829b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                final ConnectivityManager cm = ConnectivityManager.from(mPhone.getContext());
1830b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                if (mProvisionBroadcastReceiver != null) {
1831b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    mPhone.getContext().unregisterReceiver(mProvisionBroadcastReceiver);
1832b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    mProvisionBroadcastReceiver = null;
1833b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                }
1834c9b81a0c05128694c617fcdd67e73821895822feWink Saville                if ((!isProvApn) || mIsProvisioning) {
1835b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // Hide any provisioning notification.
1836b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    cm.setProvisioningNotificationVisible(false, ConnectivityManager.TYPE_MOBILE,
1837b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                            mProvisionActionName);
1838c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // Complete the connection normally notifying the world we're connected.
1839c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // We do this if this isn't a special provisioning apn or if we've been
1840c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // told its time to provision.
1841c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    completeConnection(apnContext);
1842c9b81a0c05128694c617fcdd67e73821895822feWink Saville                } else {
1843c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // This is a provisioning APN that we're reporting as connected. Later
1844c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // when the user desires to upgrade this to a "default" connection,
1845c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // mIsProvisioning == true, we'll go through the code path above.
1846c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // mIsProvisioning becomes true when CMD_ENABLE_MOBILE_PROVISIONING
1847c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // is sent to the DCT.
1848c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    if (DBG) {
1849c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        log("onDataSetupComplete: successful, BUT send connected to prov apn as"
1850c9b81a0c05128694c617fcdd67e73821895822feWink Saville                                + " mIsProvisioning:" + mIsProvisioning + " == false"
1851c9b81a0c05128694c617fcdd67e73821895822feWink Saville                                + " && (isProvisioningApn:" + isProvApn + " == true");
1852c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    }
1853c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1854b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // While radio is up, grab provisioning URL.  The URL contains ICCID which
1855b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // disappears when radio is off.
1856b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    mProvisionBroadcastReceiver = new ProvisionNotificationBroadcastReceiver(
18572b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                            cm.getMobileProvisioningUrl(),
18582b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                            TelephonyManager.getDefault().getNetworkOperatorName());
1859b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    mPhone.getContext().registerReceiver(mProvisionBroadcastReceiver,
1860b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                            new IntentFilter(mProvisionActionName));
1861b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // Put up user notification that sign-in is required.
1862b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    cm.setProvisioningNotificationVisible(true, ConnectivityManager.TYPE_MOBILE,
1863b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                            mProvisionActionName);
1864b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // Turn off radio to save battery and avoid wasting carrier resources.
1865b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // The network isn't usable and network validation will just fail anyhow.
1866b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    setRadio(false);
1867b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
1868c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    Intent intent = new Intent(
1869c9b81a0c05128694c617fcdd67e73821895822feWink Saville                            TelephonyIntents.ACTION_DATA_CONNECTION_CONNECTED_TO_PROVISIONING_APN);
1870c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    intent.putExtra(PhoneConstants.DATA_APN_KEY, apnContext.getApnSetting().apn);
1871c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    intent.putExtra(PhoneConstants.DATA_APN_TYPE_KEY, apnContext.getApnType());
1872c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1873c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    String apnType = apnContext.getApnType();
1874c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    LinkProperties linkProperties = getLinkProperties(apnType);
1875c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    if (linkProperties != null) {
1876c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        intent.putExtra(PhoneConstants.DATA_LINK_PROPERTIES_KEY, linkProperties);
1877c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        String iface = linkProperties.getInterfaceName();
1878c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        if (iface != null) {
1879c9b81a0c05128694c617fcdd67e73821895822feWink Saville                            intent.putExtra(PhoneConstants.DATA_IFACE_NAME_KEY, iface);
1880c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        }
1881c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    }
1882608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt                    NetworkCapabilities networkCapabilities = getNetworkCapabilities(apnType);
188396cce86cf08e37e0f09ed5057b1196e26b302743Robert Greenwalt                    if (networkCapabilities != null) {
188496cce86cf08e37e0f09ed5057b1196e26b302743Robert Greenwalt                        intent.putExtra(PhoneConstants.DATA_NETWORK_CAPABILITIES_KEY,
188596cce86cf08e37e0f09ed5057b1196e26b302743Robert Greenwalt                                networkCapabilities);
1886c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    }
1887c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1888c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    mPhone.getContext().sendBroadcastAsUser(intent, UserHandle.ALL);
1889c9b81a0c05128694c617fcdd67e73821895822feWink Saville                }
1890c9b81a0c05128694c617fcdd67e73821895822feWink Saville                if (DBG) {
1891c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    log("onDataSetupComplete: SETUP complete type=" + apnContext.getApnType()
1892c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        + ", reason:" + apnContext.getReason());
1893c9b81a0c05128694c617fcdd67e73821895822feWink Saville                }
1894c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
1895cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1896ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            cause = (DcFailCause) (ar.result);
1897cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) {
1898cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                ApnSetting apn = apnContext.getApnSetting();
1899cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log(String.format("onDataSetupComplete: error apn=%s cause=%s",
1900cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        (apn == null ? "unknown" : apn.apn), cause));
1901c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
1902cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (cause.isEventLoggable()) {
1903cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // Log this failure to the Event Logs.
1904cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                int cid = getCellLocationId();
1905cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                EventLog.writeEvent(EventLogTags.PDP_SETUP_FAIL,
1906cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        cause.ordinal(), cid, TelephonyManager.getDefault().getNetworkType());
1907c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
19080742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela            ApnSetting apn = apnContext.getApnSetting();
19090742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela            mPhone.notifyPreciseDataConnectionFailed(apnContext.getReason(),
19100742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela                    apnContext.getApnType(), apn != null ? apn.apn : "unknown", cause.toString());
1911cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1912cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // Count permanent failures and remove the APN we just tried
1913796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan            if (isPermanentFail(cause)) apnContext.decWaitingApnsPermFailCount();
1914cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1915cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.removeWaitingApn(apnContext.getApnSetting());
1916cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) {
1917cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log(String.format("onDataSetupComplete: WaitingApns.size=%d" +
1918cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        " WaitingApnsPermFailureCountDown=%d",
1919cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        apnContext.getWaitingApns().size(),
1920cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        apnContext.getWaitingApnsPermFailCount()));
1921c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
1922cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            handleError = true;
1923cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1924cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1925cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (handleError) {
1926ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            onDataSetupCompleteError(ar);
1927ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        }
1928a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
1929a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        /* If flag is set to false after SETUP_DATA_CALL is invoked, we need
1930a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         * to clean data connections.
1931a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         */
1932a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (!mInternalDataEnabled) {
1933a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            cleanUpAllConnections(null);
1934a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
1935a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
1936ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    }
1937cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1938ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    /**
1939ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville     * @return number of milli-seconds to delay between trying apns'
1940ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville     */
1941ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville    private int getApnDelay() {
1942ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville        if (mFailFast) {
1943ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville            return SystemProperties.getInt("persist.radio.apn_ff_delay",
1944ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville                    APN_FAIL_FAST_DELAY_DEFAULT_MILLIS);
1945ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville        } else {
1946ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville            return SystemProperties.getInt("persist.radio.apn_delay", APN_DELAY_DEFAULT_MILLIS);
1947ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville        }
1948ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville    }
1949ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville
1950ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville    /**
1951ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * Error has occurred during the SETUP {aka bringUP} request and the DCT
1952ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * should either try the next waiting APN or start over from the
1953ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * beginning if the list is empty. Between each SETUP request there will
1954ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville     * be a delay defined by {@link #getApnDelay()}.
1955ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     */
1956ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    @Override
1957ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    protected void onDataSetupCompleteError(AsyncResult ar) {
1958ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        String reason = "";
1959ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        ApnContext apnContext = null;
1960ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
1961ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if(ar.userObj instanceof ApnContext){
1962ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            apnContext = (ApnContext)ar.userObj;
1963ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        } else {
1964ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            throw new RuntimeException("onDataSetupCompleteError: No apnContext");
1965ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        }
1966ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
1967ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        // See if there are more APN's to try
1968ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (apnContext.getWaitingApns().isEmpty()) {
1969ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            apnContext.setState(DctConstants.State.FAILED);
1970ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            mPhone.notifyDataConnection(Phone.REASON_APN_FAILED, apnContext.getApnType());
1971ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
1972ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            apnContext.setDataConnectionAc(null);
1973ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
1974ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (apnContext.getWaitingApnsPermFailCount() == 0) {
1975ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (DBG) {
1976017166876a51eda9ae6b3254119023604e249bc5Wink Saville                    log("onDataSetupCompleteError: All APN's had permanent failures, stop retrying");
1977c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
1978cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
1979e056c8263473f67dc78630f5535c3664fabacd23Wink Saville                int delay = getApnDelay();
1980ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (DBG) {
1981da1dd2e1bccb6141109a65d4e40253f39c405537Wink Saville                    log("onDataSetupCompleteError: Not all APN's had permanent failures delay="
1982da1dd2e1bccb6141109a65d4e40253f39c405537Wink Saville                            + delay);
1983ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                }
1984017166876a51eda9ae6b3254119023604e249bc5Wink Saville                startAlarmForRestartTrySetup(delay, apnContext);
1985c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
1986ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        } else {
1987da1dd2e1bccb6141109a65d4e40253f39c405537Wink Saville            if (DBG) log("onDataSetupCompleteError: Try next APN");
1988ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            apnContext.setState(DctConstants.State.SCANNING);
1989ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            // Wait a bit before trying the next APN, so that
1990ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            // we're not tying up the RIL command channel
1991ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville            startAlarmForReconnect(getApnDelay(), apnContext);
1992c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
1993c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1994c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1995c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
1996cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Called when EVENT_DISCONNECT_DONE is received.
1997c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
1998cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1999cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onDisconnectDone(int connId, AsyncResult ar) {
2000cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = null;
2001cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2002cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (ar.userObj instanceof ApnContext) {
2003cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext = (ApnContext) ar.userObj;
2004cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
2005cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            loge("onDisconnectDone: Invalid ar in onDisconnectDone, ignore");
2006cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
2007c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2008c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2009cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if(DBG) log("onDisconnectDone: EVENT_DISCONNECT_DONE apnContext=" + apnContext);
2010cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        apnContext.setState(DctConstants.State.IDLE);
2011cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2012cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
2013cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2014cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // if all data connection are gone, check whether Airplane mode request was
2015cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // pending.
2016cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (isDisconnected()) {
2017cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (mPhone.getServiceStateTracker().processPendingRadioPowerOffAfterDataOff()) {
2018449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                if(DBG) log("onDisconnectDone: radio will be turned off, no retries");
2019cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // Radio will be turned off. No need to retry data setup
2020cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setApnSetting(null);
2021cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setDataConnectionAc(null);
2022a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2023a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                // Need to notify disconnect as well, in the case of switching Airplane mode.
2024a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                // Otherwise, it would cause 30s delayed to turn on Airplane mode.
2025a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                if (mDisconnectPendingCount > 0)
2026a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    mDisconnectPendingCount--;
2027a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2028a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                if (mDisconnectPendingCount == 0) {
2029a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    notifyDataDisconnectComplete();
2030a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    notifyAllDataDisconnected();
2031a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                }
2032cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return;
2033cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2034c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2035c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2036cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // If APN is still enabled, try to bring it back up automatically
20373fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if (mAttached.get() && apnContext.isReady() && retryAfterDisconnected(apnContext)) {
2038ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            SystemProperties.set(PUPPET_MASTER_RADIO_STRESS_TEST, "false");
2039cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // Wait a bit before trying the next APN, so that
2040cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // we're not tying up the RIL command channel.
2041cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // This also helps in any external dependency to turn off the context.
2042449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville            if(DBG) log("onDisconnectDone: attached, ready and retry after disconnect");
2043ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville            startAlarmForReconnect(getApnDelay(), apnContext);
2044c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
2045449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville            boolean restartRadioAfterProvisioning = mPhone.getContext().getResources().getBoolean(
2046449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                    com.android.internal.R.bool.config_restartRadioAfterProvisioning);
2047449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville
2048449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville            if (apnContext.isProvisioningApn() && restartRadioAfterProvisioning) {
2049449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                log("onDisconnectDone: restartRadio after provisioning");
2050449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                restartRadio();
2051449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville            }
2052cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setApnSetting(null);
2053cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setDataConnectionAc(null);
20543fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (isOnlySingleDcAllowed(mPhone.getServiceState().getRilDataRadioTechnology())) {
2055449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                if(DBG) log("onDisconnectDone: isOnlySigneDcAllowed true so setup single apn");
20563fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                setupDataOnConnectableApns(Phone.REASON_SINGLE_PDN_ARBITRATION);
2057449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville            } else {
2058449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                if(DBG) log("onDisconnectDone: not retrying");
20593fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            }
2060c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2061a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2062a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (mDisconnectPendingCount > 0)
2063a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            mDisconnectPendingCount--;
2064a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2065a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (mDisconnectPendingCount == 0) {
2066a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            notifyDataDisconnectComplete();
2067a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            notifyAllDataDisconnected();
2068a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2069a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2070c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2071c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2072ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    /**
2073ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * Called when EVENT_DISCONNECT_DC_RETRYING is received.
2074ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     */
2075ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    @Override
2076ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    protected void onDisconnectDcRetrying(int connId, AsyncResult ar) {
2077ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        // We could just do this in DC!!!
2078ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        ApnContext apnContext = null;
2079ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2080ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (ar.userObj instanceof ApnContext) {
2081ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            apnContext = (ApnContext) ar.userObj;
2082ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        } else {
2083ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            loge("onDisconnectDcRetrying: Invalid ar in onDisconnectDone, ignore");
2084ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            return;
2085ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        }
2086ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2087ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        apnContext.setState(DctConstants.State.RETRYING);
2088ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if(DBG) log("onDisconnectDcRetrying: apnContext=" + apnContext);
2089ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2090ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
2091ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    }
2092ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2093c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2094cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2095cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onVoiceCallStarted() {
2096cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onVoiceCallStarted");
2097ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville        mInVoiceCall = true;
2098cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (isConnected() && ! mPhone.getServiceStateTracker().isConcurrentVoiceAndDataAllowed()) {
2099cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onVoiceCallStarted stop polling");
2100cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            stopNetStatPoll();
2101cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            stopDataStallAlarm();
2102cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyDataConnection(Phone.REASON_VOICE_CALL_STARTED);
2103c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2104c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2105c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2106cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2107cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onVoiceCallEnded() {
2108cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onVoiceCallEnded");
2109ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville        mInVoiceCall = false;
2110cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (isConnected()) {
2111cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!mPhone.getServiceStateTracker().isConcurrentVoiceAndDataAllowed()) {
2112cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                startNetStatPoll();
2113cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                startDataStallAlarm(DATA_STALL_NOT_SUSPECTED);
2114cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                notifyDataConnection(Phone.REASON_VOICE_CALL_ENDED);
2115cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
2116cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // clean slate after call end.
2117cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                resetPollStats();
2118c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2119c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2120bda761320929f714951c328bfec6a51a1978db97Wink Saville        // reset reconnect timer
2121bda761320929f714951c328bfec6a51a1978db97Wink Saville        setupDataOnConnectableApns(Phone.REASON_VOICE_CALL_ENDED);
2122c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2123c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2124cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2125cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onCleanUpConnection(boolean tearDown, int apnId, String reason) {
2126cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onCleanUpConnection");
2127cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnIdToType(apnId));
2128cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext != null) {
2129cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setReason(reason);
2130cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cleanUpConnection(tearDown, apnContext);
2131c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2132c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2133c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2134cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2135cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected boolean isConnected() {
2136cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
2137ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (apnContext.getState() == DctConstants.State.CONNECTED) {
2138cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // At least one context is connected, return true
2139cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return true;
2140c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2141c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2142cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // There are not any contexts connected, return false
2143cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return false;
2144c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2145c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2146cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2147cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public boolean isDisconnected() {
2148cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
2149cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!apnContext.isDisconnected()) {
2150cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // At least one context was not disconnected return false
2151cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return false;
2152cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2153c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2154cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // All contexts were disconnected so return true
2155cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return true;
2156c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2157c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2158cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2159cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void notifyDataConnection(String reason) {
2160cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("notifyDataConnection: reason=" + reason);
2161cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
2162187a39f896f88eb6c5e4306d9595546654825976Wink Saville            if (mAttached.get() && apnContext.isReady()) {
2163187a39f896f88eb6c5e4306d9595546654825976Wink Saville                if (DBG) log("notifyDataConnection: type:" + apnContext.getApnType());
2164cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPhone.notifyDataConnection(reason != null ? reason : apnContext.getReason(),
2165cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        apnContext.getApnType());
2166cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2167c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2168cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        notifyOffApnsOfAvailability(reason);
2169c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2170c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2171c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
2172cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Based on the sim operator numeric, create a list for all possible
2173cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Data Connections and setup the preferredApn.
2174c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
2175cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void createAllApnList() {
2176ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        mAllApnSettings = new ArrayList<ApnSetting>();
2177cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
2178cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        String operator = (r != null) ? r.getOperatorNumeric() : "";
2179cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (operator != null) {
2180cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            String selection = "numeric = '" + operator + "'";
2181cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // query only enabled apn.
2182cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // carrier_enabled : 1 means enabled apn, 0 disabled apn.
21839d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan            // selection += " and carrier_enabled = 1";
2184cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("createAllApnList: selection=" + selection);
2185cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2186cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            Cursor cursor = mPhone.getContext().getContentResolver().query(
2187cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    Telephony.Carriers.CONTENT_URI, null, selection, null, null);
2188cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2189cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (cursor != null) {
2190cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (cursor.getCount() > 0) {
2191ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    mAllApnSettings = createApnList(cursor);
2192cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
2193cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                cursor.close();
2194cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2195c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2196c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
219776f43316a5a6082d601bffd4b6898d0bd81e11fcram        addEmergencyApnSetting();
219876f43316a5a6082d601bffd4b6898d0bd81e11fcram
219929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        dedupeApnSettings();
220029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
2201ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (mAllApnSettings.isEmpty()) {
2202cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("createAllApnList: No APN found for carrier: " + operator);
2203cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mPreferredApn = null;
2204ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            // TODO: What is the right behavior?
2205cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            //notifyNoData(DataConnection.FailCause.MISSING_UNKNOWN_APN);
2206c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
2207cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mPreferredApn = getPreferredApn();
2208cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (mPreferredApn != null && !mPreferredApn.numeric.equals(operator)) {
2209cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPreferredApn = null;
2210cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                setPreferredApn(-1);
2211cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2212cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("createAllApnList: mPreferredApn=" + mPreferredApn);
2213c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2214ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (DBG) log("createAllApnList: X mAllApnSettings=" + mAllApnSettings);
22159d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan
22169d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan        setDataProfilesAsNeeded();
2217c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2218c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
221929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    private void dedupeApnSettings() {
222029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        ArrayList<ApnSetting> resultApns = new ArrayList<ApnSetting>();
222129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
222229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        // coalesce APNs if they are similar enough to prevent
222329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        // us from bringing up two data calls with the same interface
222429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        int i = 0;
222529c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        while (i < mAllApnSettings.size() - 1) {
222629c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            ApnSetting first = mAllApnSettings.get(i);
222729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            ApnSetting second = null;
222829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            int j = i + 1;
222929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            while (j < mAllApnSettings.size()) {
223029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                second = mAllApnSettings.get(j);
223129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                if (apnsSimilar(first, second)) {
223229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                    ApnSetting newApn = mergeApns(first, second);
223329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                    mAllApnSettings.set(i, newApn);
223429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                    first = newApn;
223529c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                    mAllApnSettings.remove(j);
223629c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                } else {
223729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                    j++;
223829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                }
223929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            }
224029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            i++;
224129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        }
224229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    }
224329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
2244d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe    //check whether the types of two APN same (even only one type of each APN is same)
2245d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe    private boolean apnTypeSameAny(ApnSetting first, ApnSetting second) {
2246d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        if(VDBG) {
2247d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            StringBuilder apnType1 = new StringBuilder(first.apn + ": ");
2248d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            for(int index1 = 0; index1 < first.types.length; index1++) {
2249d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                apnType1.append(first.types[index1]);
2250d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                apnType1.append(",");
2251d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            }
2252d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe
2253d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            StringBuilder apnType2 = new StringBuilder(second.apn + ": ");
2254d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            for(int index1 = 0; index1 < second.types.length; index1++) {
2255d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                apnType2.append(second.types[index1]);
2256d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                apnType2.append(",");
2257d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            }
2258d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            log("APN1: is " + apnType1);
2259d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            log("APN2: is " + apnType2);
2260d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        }
2261d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe
2262d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        for(int index1 = 0; index1 < first.types.length; index1++) {
2263d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            for(int index2 = 0; index2 < second.types.length; index2++) {
2264d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                if(first.types[index1].equals(PhoneConstants.APN_TYPE_ALL) ||
2265d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                        second.types[index2].equals(PhoneConstants.APN_TYPE_ALL) ||
2266d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                        first.types[index1].equals(second.types[index2])) {
2267d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                    if(VDBG)log("apnTypeSameAny: return true");
2268d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                    return true;
2269d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                }
2270d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            }
2271d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        }
2272d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe
2273d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        if(VDBG)log("apnTypeSameAny: return false");
2274d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        return false;
2275d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe    }
2276d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe
227729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    // Check if neither mention DUN and are substantially similar
227829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    private boolean apnsSimilar(ApnSetting first, ApnSetting second) {
227929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        return (first.canHandleType(PhoneConstants.APN_TYPE_DUN) == false &&
228029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                second.canHandleType(PhoneConstants.APN_TYPE_DUN) == false &&
228129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                Objects.equals(first.apn, second.apn) &&
2282d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                !apnTypeSameAny(first, second) &&
228361cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                xorEquals(first.proxy, second.proxy) &&
228461cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                xorEquals(first.port, second.port) &&
228529c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                first.carrierEnabled == second.carrierEnabled &&
228629c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                first.bearer == second.bearer &&
228729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                first.profileId == second.profileId &&
228829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                Objects.equals(first.mvnoType, second.mvnoType) &&
228929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                Objects.equals(first.mvnoMatchData, second.mvnoMatchData) &&
229029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                xorEquals(first.mmsc, second.mmsc) &&
229129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                xorEquals(first.mmsProxy, second.mmsProxy) &&
229229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                xorEquals(first.mmsPort, second.mmsPort));
229329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    }
229429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
229529c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    // equal or one is not specified
229629c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    private boolean xorEquals(String first, String second) {
229729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        return (Objects.equals(first, second) ||
229829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                TextUtils.isEmpty(first) ||
229929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                TextUtils.isEmpty(second));
230029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    }
230129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
230229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    private ApnSetting mergeApns(ApnSetting dest, ApnSetting src) {
230329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        ArrayList<String> resultTypes = new ArrayList<String>();
230429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        resultTypes.addAll(Arrays.asList(dest.types));
230529c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        for (String srcType : src.types) {
230629c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            if (resultTypes.contains(srcType) == false) resultTypes.add(srcType);
230729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        }
230829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        String mmsc = (TextUtils.isEmpty(dest.mmsc) ? src.mmsc : dest.mmsc);
230929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        String mmsProxy = (TextUtils.isEmpty(dest.mmsProxy) ? src.mmsProxy : dest.mmsProxy);
231029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        String mmsPort = (TextUtils.isEmpty(dest.mmsPort) ? src.mmsPort : dest.mmsPort);
231161cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan        String proxy = (TextUtils.isEmpty(dest.proxy) ? src.proxy : dest.proxy);
231261cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan        String port = (TextUtils.isEmpty(dest.port) ? src.port : dest.port);
231361cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan        String protocol = src.protocol.equals("IPV4V6") ? src.protocol : dest.protocol;
231461cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan        String roamingProtocol = src.roamingProtocol.equals("IPV4V6") ? src.roamingProtocol :
231561cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                dest.roamingProtocol;
231629c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
231729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        return new ApnSetting(dest.id, dest.numeric, dest.carrier, dest.apn,
231861cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                proxy, port, mmsc, mmsProxy, mmsPort, dest.user, dest.password,
231961cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                dest.authType, resultTypes.toArray(new String[0]), protocol,
232061cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                roamingProtocol, dest.carrierEnabled, dest.bearer, dest.profileId,
232129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                (dest.modemCognitive || src.modemCognitive), dest.maxConns, dest.waitTime,
232229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                dest.maxConnsTime, dest.mtu, dest.mvnoType, dest.mvnoMatchData);
232329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    }
232429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
2325ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    /** Return the DC AsyncChannel for the new data connection */
2326454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    private DcAsyncChannel createDataConnection() {
2327cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("createDataConnection E");
2328cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2329cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        int id = mUniqueIdGenerator.getAndIncrement();
2330ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        DataConnection conn = DataConnection.makeDataConnection(mPhone, id,
2331ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                                this, mDcTesterFailBringUpAll, mDcc);
2332cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mDataConnections.put(id, conn);
2333454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville        DcAsyncChannel dcac = new DcAsyncChannel(conn, LOG_TAG);
2334cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        int status = dcac.fullyConnectSync(mPhone.getContext(), this, conn.getHandler());
2335cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (status == AsyncChannel.STATUS_SUCCESSFUL) {
2336ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            mDataConnectionAcHashMap.put(dcac.getDataConnectionIdSync(), dcac);
2337c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
2338ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            loge("createDataConnection: Could not connect to dcac=" + dcac + " status=" + status);
2339c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2340cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2341cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("createDataConnection() X id=" + id + " dc=" + conn);
2342ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        return dcac;
2343c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2344c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2345cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void destroyDataConnections() {
2346cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if(mDataConnections != null) {
2347cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("destroyDataConnections: clear mDataConnectionList");
2348cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mDataConnections.clear();
2349cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
2350cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("destroyDataConnections: mDataConnecitonList is empty, ignore");
2351c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2352c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2353c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2354c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
2355cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Build a list of APNs to be used to create PDP's.
2356c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *
2357cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param requestedApnType
2358cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @return waitingApns list to be used to create PDP
2359cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *          error when waitingApns.isEmpty()
2360c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
2361203e588e3c42a81aa8a56f595119c181a63b12caWink Saville    private ArrayList<ApnSetting> buildWaitingApns(String requestedApnType, int radioTech) {
2362cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("buildWaitingApns: E requestedApnType=" + requestedApnType);
2363cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ArrayList<ApnSetting> apnList = new ArrayList<ApnSetting>();
2364cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2365cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (requestedApnType.equals(PhoneConstants.APN_TYPE_DUN)) {
2366cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            ApnSetting dun = fetchDunApn();
2367cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (dun != null) {
2368cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnList.add(dun);
2369cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("buildWaitingApns: X added APN_TYPE_DUN apnList=" + apnList);
2370cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return apnList;
2371c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2372c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2373c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2374cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
2375cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        String operator = (r != null) ? r.getOperatorNumeric() : "";
2376c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2377cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // This is a workaround for a bug (7305641) where we don't failover to other
2378cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // suitable APNs if our preferred APN fails.  On prepaid ATT sims we need to
2379cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // failover to a provisioning APN, but once we've used their default data
2380cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // connection we are locked to it for life.  This change allows ATT devices
2381cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // to say they don't want to use preferred at all.
2382cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean usePreferred = true;
2383cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        try {
2384cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            usePreferred = ! mPhone.getContext().getResources().getBoolean(com.android.
2385cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    internal.R.bool.config_dontPreferApn);
2386cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } catch (Resources.NotFoundException e) {
2387cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("buildWaitingApns: usePreferred NotFoundException set to true");
2388cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            usePreferred = true;
2389cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2390cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) {
2391cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("buildWaitingApns: usePreferred=" + usePreferred
239222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                    + " canSetPreferApn=" + mCanSetPreferApn
2393cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    + " mPreferredApn=" + mPreferredApn
2394cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    + " operator=" + operator + " radioTech=" + radioTech
2395cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    + " IccRecords r=" + r);
2396cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2397c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
239822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (usePreferred && mCanSetPreferApn && mPreferredApn != null &&
2399cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPreferredApn.canHandleType(requestedApnType)) {
2400cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) {
2401cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("buildWaitingApns: Preferred APN:" + operator + ":"
2402cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        + mPreferredApn.numeric + ":" + mPreferredApn);
2403cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2404cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (mPreferredApn.numeric.equals(operator)) {
2405cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (mPreferredApn.bearer == 0 || mPreferredApn.bearer == radioTech) {
2406cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnList.add(mPreferredApn);
2407cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (DBG) log("buildWaitingApns: X added preferred apnList=" + apnList);
2408cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return apnList;
2409cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
2410cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (DBG) log("buildWaitingApns: no preferred APN");
2411cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    setPreferredApn(-1);
2412cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    mPreferredApn = null;
2413c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
2414cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
2415cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("buildWaitingApns: no preferred APN");
2416cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                setPreferredApn(-1);
2417cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPreferredApn = null;
2418c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2419c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2420ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (mAllApnSettings != null) {
2421ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (DBG) log("buildWaitingApns: mAllApnSettings=" + mAllApnSettings);
2422ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            for (ApnSetting apn : mAllApnSettings) {
2423cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("buildWaitingApns: apn=" + apn);
2424cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (apn.canHandleType(requestedApnType)) {
2425cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (apn.bearer == 0 || apn.bearer == radioTech) {
2426cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (DBG) log("buildWaitingApns: adding apn=" + apn.toString());
2427cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        apnList.add(apn);
2428c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    } else {
2429cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (DBG) {
2430cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            log("buildWaitingApns: bearer:" + apn.bearer + " != "
2431cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                    + "radioTech:" + radioTech);
2432cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
2433c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    }
2434cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
2435cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) {
2436cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("buildWaitingApns: couldn't handle requesedApnType="
2437cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            + requestedApnType);
2438c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
2439c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2440c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2441cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
2442ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            loge("mAllApnSettings is empty!");
2443c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2444cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("buildWaitingApns: X apnList=" + apnList);
2445cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return apnList;
2446c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2447c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2448cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private String apnListToString (ArrayList<ApnSetting> apns) {
2449cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        StringBuilder result = new StringBuilder();
2450cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (int i = 0, size = apns.size(); i < size; i++) {
2451cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            result.append('[')
2452cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                  .append(apns.get(i).toString())
2453cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                  .append(']');
2454c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2455cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return result.toString();
2456c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2457c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2458cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void setPreferredApn(int pos) {
245922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (!mCanSetPreferApn) {
2460cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("setPreferredApn: X !canSEtPreferApn");
2461cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
2462cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2463cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
24646bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        String subId = Long.toString(mPhone.getSubId());
24656bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        Uri uri = Uri.withAppendedPath(PREFERAPN_NO_UPDATE_URI_USING_SUBID, subId);
2466cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        log("setPreferredApn: delete");
2467cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ContentResolver resolver = mPhone.getContext().getContentResolver();
24686bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        resolver.delete(uri, null, null);
2469cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2470cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (pos >= 0) {
2471cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("setPreferredApn: insert");
2472cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            ContentValues values = new ContentValues();
2473cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            values.put(APN_ID, pos);
24746bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville            resolver.insert(uri, values);
2475cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2476cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
2477cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2478cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private ApnSetting getPreferredApn() {
2479ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (mAllApnSettings.isEmpty()) {
2480ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            log("getPreferredApn: X not found mAllApnSettings.isEmpty");
2481cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return null;
2482cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2483cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
24846bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        String subId = Long.toString(mPhone.getSubId());
24856bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        Uri uri = Uri.withAppendedPath(PREFERAPN_NO_UPDATE_URI_USING_SUBID, subId);
2486cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        Cursor cursor = mPhone.getContext().getContentResolver().query(
24876bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville                uri, new String[] { "_id", "name", "apn" },
2488cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                null, null, Telephony.Carriers.DEFAULT_SORT_ORDER);
2489cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2490cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (cursor != null) {
249122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mCanSetPreferApn = true;
2492cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
249322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mCanSetPreferApn = false;
2494cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2495cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        log("getPreferredApn: mRequestedApnType=" + mRequestedApnType + " cursor=" + cursor
2496cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                + " cursor.count=" + ((cursor != null) ? cursor.getCount() : 0));
2497cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
249822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mCanSetPreferApn && cursor.getCount() > 0) {
2499cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            int pos;
2500cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cursor.moveToFirst();
2501cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            pos = cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers._ID));
2502ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            for(ApnSetting p : mAllApnSettings) {
2503cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("getPreferredApn: apnSetting=" + p);
2504cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (p.id == pos && p.canHandleType(mRequestedApnType)) {
2505cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("getPreferredApn: X found apnSetting" + p);
2506cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    cursor.close();
2507cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return p;
2508cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
2509cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla            }
2510cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2511cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2512cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (cursor != null) {
2513cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cursor.close();
2514cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2515cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2516cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        log("getPreferredApn: X not found");
2517cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return null;
2518cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2519cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2520cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2521cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public void handleMessage (Message msg) {
2522cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("handleMessage msg=" + msg);
2523cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2524cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (!mPhone.mIsTheCurrentActivePhone || mIsDisposed) {
2525cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            loge("handleMessage: Ignore GSM msgs since GSM phone is inactive");
2526cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
2527cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2528cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2529cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        switch (msg.what) {
2530cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_RECORDS_LOADED:
2531bda761320929f714951c328bfec6a51a1978db97Wink Saville                onRecordsLoaded();
2532cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2533cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2534cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_DATA_CONNECTION_DETACHED:
2535cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                onDataConnectionDetached();
2536cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2537cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2538cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_DATA_CONNECTION_ATTACHED:
2539bda761320929f714951c328bfec6a51a1978db97Wink Saville                onDataConnectionAttached();
2540cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2541cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2542cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_DO_RECOVERY:
2543cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                doRecovery();
2544cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2545cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2546cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_APN_CHANGED:
2547cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                onApnChanged();
2548cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2549cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2550cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_PS_RESTRICT_ENABLED:
2551cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                /**
2552cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * We don't need to explicitly to tear down the PDP context
2553cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * when PS restricted is enabled. The base band will deactive
2554cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * PDP context and notify us with PDP_CONTEXT_CHANGED.
2555cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * But we should stop the network polling and prevent reset PDP.
2556cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 */
2557cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("EVENT_PS_RESTRICT_ENABLED " + mIsPsRestricted);
2558cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                stopNetStatPoll();
2559cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                stopDataStallAlarm();
2560cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mIsPsRestricted = true;
2561cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla                break;
2562cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2563cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_PS_RESTRICT_DISABLED:
2564cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                /**
2565cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * When PS restrict is removed, we need setup PDP connection if
2566cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * PDP connection is down.
2567cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 */
2568cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("EVENT_PS_RESTRICT_DISABLED " + mIsPsRestricted);
2569cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mIsPsRestricted  = false;
2570cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (isConnected()) {
2571cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    startNetStatPoll();
2572cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    startDataStallAlarm(DATA_STALL_NOT_SUSPECTED);
2573cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
2574cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    // TODO: Should all PDN states be checked to fail?
2575ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    if (mState == DctConstants.State.FAILED) {
2576cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        cleanUpAllConnections(false, Phone.REASON_PS_RESTRICT_ENABLED);
2577cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        mReregisterOnReconnectFailure = false;
2578cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
25793fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    ApnContext apnContext = mApnContexts.get(PhoneConstants.APN_TYPE_DEFAULT);
25803fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    if (apnContext != null) {
25813fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        apnContext.setReason(Phone.REASON_PS_RESTRICT_ENABLED);
25823fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        trySetupData(apnContext);
25833fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    } else {
25843fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        loge("**** Default ApnContext not found ****");
25853fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        if (Build.IS_DEBUGGABLE) {
25863fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                            throw new RuntimeException("Default ApnContext not found");
25873fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        }
25883fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    }
2589cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
2590cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla                break;
2591ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2592cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_TRY_SETUP_DATA:
2593cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (msg.obj instanceof ApnContext) {
2594cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    onTrySetupData((ApnContext)msg.obj);
2595cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else if (msg.obj instanceof String) {
2596cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    onTrySetupData((String)msg.obj);
2597cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
2598cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    loge("EVENT_TRY_SETUP request w/o apnContext or String");
2599cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
2600cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla                break;
2601cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2602cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_CLEAN_UP_CONNECTION:
2603cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                boolean tearDown = (msg.arg1 == 0) ? false : true;
2604cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("EVENT_CLEAN_UP_CONNECTION tearDown=" + tearDown);
2605cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (msg.obj instanceof ApnContext) {
2606cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    cleanUpConnection(tearDown, (ApnContext)msg.obj);
2607cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
2608ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    loge("EVENT_CLEAN_UP_CONNECTION request w/o apn context, call super");
2609ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    super.handleMessage(msg);
2610cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
2611cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla                break;
2612a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            case DctConstants.EVENT_SET_INTERNAL_DATA_ENABLE:
2613a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                boolean enabled = (msg.arg1 == DctConstants.ENABLED) ? true : false;
2614a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                onSetInternalDataEnabled(enabled, (Message) msg.obj);
2615a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                break;
2616a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2617a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            case DctConstants.EVENT_CLEAN_UP_ALL_CONNECTIONS:
2618a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                Message mCause = obtainMessage(DctConstants.EVENT_CLEAN_UP_ALL_CONNECTIONS, null);
2619a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                if ((msg.obj != null) && (msg.obj instanceof String)) {
2620a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    mCause.obj = msg.obj;
2621a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                }
2622a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                super.handleMessage(mCause);
2623a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                break;
2624ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2625220d43e0611edd0f3a2eb7a33bb38a008fff0868Wink Saville            case DctConstants.EVENT_DATA_RAT_CHANGED:
2626220d43e0611edd0f3a2eb7a33bb38a008fff0868Wink Saville                //May new Network allow setupData, so try it here
2627220d43e0611edd0f3a2eb7a33bb38a008fff0868Wink Saville                setupDataOnConnectableApns(Phone.REASON_NW_TYPE_CHANGED);
2628220d43e0611edd0f3a2eb7a33bb38a008fff0868Wink Saville                break;
2629220d43e0611edd0f3a2eb7a33bb38a008fff0868Wink Saville
26302b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            case DctConstants.CMD_CLEAR_PROVISIONING_SPINNER:
26312b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                // Check message sender intended to clear the current spinner.
26322b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                if (mProvisioningSpinner == msg.obj) {
26332b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    mProvisioningSpinner.dismiss();
26342b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    mProvisioningSpinner = null;
26352b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                }
26362b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                break;
26372b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen
2638cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla            default:
2639cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // handle the message in the super class DataConnectionTracker
2640cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                super.handleMessage(msg);
2641cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2642cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2643cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2644cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2645cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected int getApnProfileID(String apnType) {
2646cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_IMS)) {
2647cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return RILConstants.DATA_PROFILE_IMS;
2648cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_FOTA)) {
2649cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return RILConstants.DATA_PROFILE_FOTA;
2650cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_CBS)) {
2651cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return RILConstants.DATA_PROFILE_CBS;
26521b5fe200e47f40f82f0e28502a5f40bce64a82e6Wink Saville        } else if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_IA)) {
26531b5fe200e47f40f82f0e28502a5f40bce64a82e6Wink Saville            return RILConstants.DATA_PROFILE_DEFAULT; // DEFAULT for now
265445df26444864daad60afdd4d121ab4043da3834bSungmin Choi        } else if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_DUN)) {
265545df26444864daad60afdd4d121ab4043da3834bSungmin Choi            return RILConstants.DATA_PROFILE_TETHERED;
2656cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
2657cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return RILConstants.DATA_PROFILE_DEFAULT;
2658cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2659cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
2660cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2661cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private int getCellLocationId() {
2662cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        int cid = -1;
2663cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        CellLocation loc = mPhone.getCellLocation();
2664cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2665cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (loc != null) {
2666cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (loc instanceof GsmCellLocation) {
2667cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                cid = ((GsmCellLocation)loc).getCid();
2668cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else if (loc instanceof CdmaCellLocation) {
2669cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                cid = ((CdmaCellLocation)loc).getBaseStationId();
2670cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla            }
2671cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2672cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return cid;
2673cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2674cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2675a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private IccRecords getUiccRecords(int appFamily) {
2676a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return mUiccController.getIccRecords(mPhone.getPhoneId(), appFamily);
2677a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2678a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2679a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2680cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2681cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onUpdateIcc() {
2682cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (mUiccController == null ) {
2683cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla            return;
2684cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2685cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2686a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        IccRecords newIccRecords = getUiccRecords(UiccController.APP_FAM_3GPP);
2687cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2688cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
2689cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (r != newIccRecords) {
2690cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (r != null) {
2691cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("Removing stale icc objects.");
2692cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                r.unregisterForRecordsLoaded(this);
2693cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mIccRecords.set(null);
26949aaa228cbb37657d12a87b5058676e449866f2acJeevaka Badrappan            }
2695cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (newIccRecords != null) {
2696cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("New records found");
2697cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mIccRecords.set(newIccRecords);
2698cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                newIccRecords.registerForRecordsLoaded(
2699cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        this, DctConstants.EVENT_RECORDS_LOADED, null);
27009aaa228cbb37657d12a87b5058676e449866f2acJeevaka Badrappan            }
27019aaa228cbb37657d12a87b5058676e449866f2acJeevaka Badrappan        }
2702cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2703cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2704a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void update() {
2705a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("update sub = " + mPhone.getSubId());
2706bda761320929f714951c328bfec6a51a1978db97Wink Saville        log("update(): Active DDS, register for all events now!");
2707bda761320929f714951c328bfec6a51a1978db97Wink Saville        registerForAllEvents();
2708bda761320929f714951c328bfec6a51a1978db97Wink Saville        onUpdateIcc();
2709a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2710bda761320929f714951c328bfec6a51a1978db97Wink Saville        mUserDataEnabled = Settings.Global.getInt(mPhone.getContext().getContentResolver(),
2711bda761320929f714951c328bfec6a51a1978db97Wink Saville                Settings.Global.MOBILE_DATA, 1) == 1;
2712bda761320929f714951c328bfec6a51a1978db97Wink Saville
2713bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (mPhone instanceof CDMALTEPhone) {
2714bda761320929f714951c328bfec6a51a1978db97Wink Saville            ((CDMALTEPhone)mPhone).updateCurrentCarrierInProvider();
2715bda761320929f714951c328bfec6a51a1978db97Wink Saville            supplyMessenger();
2716bda761320929f714951c328bfec6a51a1978db97Wink Saville        } else if (mPhone instanceof GSMPhone) {
2717bda761320929f714951c328bfec6a51a1978db97Wink Saville            ((GSMPhone)mPhone).updateCurrentCarrierInProvider();
2718bda761320929f714951c328bfec6a51a1978db97Wink Saville            supplyMessenger();
2719a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        } else {
2720bda761320929f714951c328bfec6a51a1978db97Wink Saville            log("Phone object is not MultiSim. This should not hit!!!!");
2721a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2722a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2723a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2724a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
2725a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void cleanUpAllConnections(String cause) {
2726a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        cleanUpAllConnections(cause, null);
2727a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2728a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2729a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void updateRecords() {
2730bda761320929f714951c328bfec6a51a1978db97Wink Saville        onUpdateIcc();
2731a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2732a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2733a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void cleanUpAllConnections(String cause, Message disconnectAllCompleteMsg) {
2734a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("cleanUpAllConnections");
2735a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (disconnectAllCompleteMsg != null) {
2736a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            mDisconnectAllCompleteMsgList.add(disconnectAllCompleteMsg);
2737a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2738a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2739a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        Message msg = obtainMessage(DctConstants.EVENT_CLEAN_UP_ALL_CONNECTIONS);
2740a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        msg.obj = cause;
2741a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        sendMessage(msg);
2742a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2743a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2744a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void notifyDataDisconnectComplete() {
2745a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("notifyDataDisconnectComplete");
2746a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        for (Message m: mDisconnectAllCompleteMsgList) {
2747a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            m.sendToTarget();
2748a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2749a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mDisconnectAllCompleteMsgList.clear();
2750a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2751a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2752a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2753a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void notifyAllDataDisconnected() {
2754a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        sEnableFailFastRefCounter = 0;
2755a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mFailFast = false;
2756a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mAllDataDisconnectedRegistrants.notifyRegistrants();
2757a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2758a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2759a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void registerForAllDataDisconnected(Handler h, int what, Object obj) {
2760a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mAllDataDisconnectedRegistrants.addUnique(h, what, obj);
2761a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2762a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (isDisconnected()) {
2763a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            log("notify All Data Disconnected");
2764a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            notifyAllDataDisconnected();
2765a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2766a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2767a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2768a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void unregisterForAllDataDisconnected(Handler h) {
2769a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mAllDataDisconnectedRegistrants.remove(h);
2770a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2771a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2772a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2773a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
2774a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void onSetInternalDataEnabled(boolean enable) {
2775bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (DBG) log("onSetInternalDataEnabled: enabled=" + enable);
2776a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        onSetInternalDataEnabled(enable, null);
2777a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2778a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2779a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void onSetInternalDataEnabled(boolean enabled, Message onCompleteMsg) {
27806bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (DBG) log("onSetInternalDataEnabled: enabled=" + enabled);
2781a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        boolean sendOnComplete = true;
2782a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2783a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        synchronized (mDataEnabledLock) {
2784a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            mInternalDataEnabled = enabled;
2785a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            if (enabled) {
2786a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                log("onSetInternalDataEnabled: changed to enabled, try to setup data call");
2787a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                onTrySetupData(Phone.REASON_DATA_ENABLED);
2788a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            } else {
2789a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                sendOnComplete = false;
2790a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                log("onSetInternalDataEnabled: changed to disabled, cleanUpAllConnections");
2791a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                cleanUpAllConnections(null, onCompleteMsg);
2792a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            }
2793a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2794a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2795a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (sendOnComplete) {
2796a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            if (onCompleteMsg != null) {
2797a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                onCompleteMsg.sendToTarget();
2798a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            }
2799a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2800a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2801a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2802a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean setInternalDataEnabledFlag(boolean enable) {
28036bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (DBG) log("setInternalDataEnabledFlag(" + enable + ")");
2804a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2805a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (mInternalDataEnabled != enable) {
2806a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            mInternalDataEnabled = enable;
2807a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2808a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return true;
2809a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2810a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2811a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
2812a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean setInternalDataEnabled(boolean enable) {
2813a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return setInternalDataEnabled(enable, null);
2814a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2815a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2816a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean setInternalDataEnabled(boolean enable, Message onCompleteMsg) {
28176bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (DBG) log("setInternalDataEnabled(" + enable + ")");
2818a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2819a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        Message msg = obtainMessage(DctConstants.EVENT_SET_INTERNAL_DATA_ENABLE, onCompleteMsg);
2820a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        msg.arg1 = (enable ? DctConstants.ENABLED : DctConstants.DISABLED);
2821a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        sendMessage(msg);
2822a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return true;
2823a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2824a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2825a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void setDataAllowed(boolean enable, Message response) {
28266bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville         if (DBG) log("setDataAllowed: enable=" + enable);
2827a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         mIsCleanupRequired = !enable;
2828a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         mPhone.mCi.setDataAllowed(enable, response);
2829a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         mInternalDataEnabled = enable;
2830a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2831a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2832cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2833cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void log(String s) {
2834a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        Rlog.d(LOG_TAG, "[" + mPhone.getPhoneId() + "]" + s);
2835cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2836cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2837cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2838cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void loge(String s) {
2839a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        Rlog.e(LOG_TAG, "[" + mPhone.getPhoneId() + "]" + s);
2840cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2841cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2842cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2843c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
2844bda761320929f714951c328bfec6a51a1978db97Wink Saville        pw.println("DcTracker extends:");
2845cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        super.dump(fd, pw, args);
2846cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        pw.println(" mReregisterOnReconnectFailure=" + mReregisterOnReconnectFailure);
284722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        pw.println(" canSetPreferApn=" + mCanSetPreferApn);
2848cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        pw.println(" mApnObserver=" + mApnObserver);
2849cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        pw.println(" getOverallState=" + getOverallState());
2850ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        pw.println(" mDataConnectionAsyncChannels=%s\n" + mDataConnectionAcHashMap);
2851187a39f896f88eb6c5e4306d9595546654825976Wink Saville        pw.println(" mAttached=" + mAttached.get());
2852c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2853a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2854a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
2855bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram    public String[] getPcscfAddress(String apnType) {
2856a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("getPcscfAddress()");
2857bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        ApnContext apnContext = null;
2858bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram
2859bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        if(apnType == null){
2860bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            log("apnType is null, return null");
2861bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            return null;
2862bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        }
2863a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2864bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_EMERGENCY)) {
2865bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            apnContext = mApnContexts.get(PhoneConstants.APN_TYPE_EMERGENCY);
2866bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        } else if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_IMS)) {
2867bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            apnContext = mApnContexts.get(PhoneConstants.APN_TYPE_IMS);
2868bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        } else {
2869bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            log("apnType is invalid, return null");
2870bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            return null;
2871bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        }
2872a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2873a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (apnContext == null) {
2874a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            log("apnContext is null, return null");
2875a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            return null;
2876a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2877a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2878a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        DcAsyncChannel dcac = apnContext.getDcAc();
2879a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        String[] result = null;
2880a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2881a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (dcac != null) {
2882a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            result = dcac.getPcscfAddr();
2883a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2884a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            for (int i = 0; i < result.length; i++) {
2885a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                log("Pcscf[" + i + "]: " + result[i]);
2886a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            }
2887a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            return result;
2888a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2889a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return null;
2890a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2891a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2892a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
2893a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void setImsRegistrationState(boolean registered) {
2894a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("setImsRegistrationState - mImsRegistrationState(before): "+ mImsRegistrationState
2895a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                + ", registered(current) : " + registered);
2896a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2897a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (mPhone == null) return;
2898a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2899a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        ServiceStateTracker sst = mPhone.getServiceStateTracker();
2900a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (sst == null) return;
2901a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2902a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        sst.setImsRegistrationState(registered);
2903a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
290476f43316a5a6082d601bffd4b6898d0bd81e11fcram
290576f43316a5a6082d601bffd4b6898d0bd81e11fcram    /**
290676f43316a5a6082d601bffd4b6898d0bd81e11fcram     * Read APN configuration from Telephony.db for Emergency APN
290776f43316a5a6082d601bffd4b6898d0bd81e11fcram     * All opertors recognize the connection request for EPDN based on APN type
290876f43316a5a6082d601bffd4b6898d0bd81e11fcram     * PLMN name,APN name are not mandatory parameters
290976f43316a5a6082d601bffd4b6898d0bd81e11fcram     */
291076f43316a5a6082d601bffd4b6898d0bd81e11fcram    private void initEmergencyApnSetting() {
291176f43316a5a6082d601bffd4b6898d0bd81e11fcram        // Operator Numeric is not available when sim records are not loaded.
291276f43316a5a6082d601bffd4b6898d0bd81e11fcram        // Query Telephony.db with APN type as EPDN request does not
291376f43316a5a6082d601bffd4b6898d0bd81e11fcram        // require APN name, plmn and all operators support same APN config.
291476f43316a5a6082d601bffd4b6898d0bd81e11fcram        // DB will contain only one entry for Emergency APN
291576f43316a5a6082d601bffd4b6898d0bd81e11fcram        String selection = "type=\"emergency\"";
291676f43316a5a6082d601bffd4b6898d0bd81e11fcram        Cursor cursor = mPhone.getContext().getContentResolver().query(
291776f43316a5a6082d601bffd4b6898d0bd81e11fcram                Telephony.Carriers.CONTENT_URI, null, selection, null, null);
291876f43316a5a6082d601bffd4b6898d0bd81e11fcram
291976f43316a5a6082d601bffd4b6898d0bd81e11fcram        if (cursor != null) {
292076f43316a5a6082d601bffd4b6898d0bd81e11fcram            if (cursor.getCount() > 0) {
292176f43316a5a6082d601bffd4b6898d0bd81e11fcram                if (cursor.moveToFirst()) {
292276f43316a5a6082d601bffd4b6898d0bd81e11fcram                    mEmergencyApn = makeApnSetting(cursor);
292376f43316a5a6082d601bffd4b6898d0bd81e11fcram                }
292476f43316a5a6082d601bffd4b6898d0bd81e11fcram            }
292576f43316a5a6082d601bffd4b6898d0bd81e11fcram            cursor.close();
292676f43316a5a6082d601bffd4b6898d0bd81e11fcram        }
292776f43316a5a6082d601bffd4b6898d0bd81e11fcram    }
292876f43316a5a6082d601bffd4b6898d0bd81e11fcram
292976f43316a5a6082d601bffd4b6898d0bd81e11fcram    /**
293076f43316a5a6082d601bffd4b6898d0bd81e11fcram     * Add the Emergency APN settings to APN settings list
293176f43316a5a6082d601bffd4b6898d0bd81e11fcram     */
293276f43316a5a6082d601bffd4b6898d0bd81e11fcram    private void addEmergencyApnSetting() {
293376f43316a5a6082d601bffd4b6898d0bd81e11fcram        if(mEmergencyApn != null) {
293476f43316a5a6082d601bffd4b6898d0bd81e11fcram            if(mAllApnSettings == null) {
293576f43316a5a6082d601bffd4b6898d0bd81e11fcram                mAllApnSettings = new ArrayList<ApnSetting>();
293676f43316a5a6082d601bffd4b6898d0bd81e11fcram            } else {
293776f43316a5a6082d601bffd4b6898d0bd81e11fcram                boolean hasEmergencyApn = false;
293876f43316a5a6082d601bffd4b6898d0bd81e11fcram                for (ApnSetting apn : mAllApnSettings) {
293976f43316a5a6082d601bffd4b6898d0bd81e11fcram                    if (ArrayUtils.contains(apn.types, PhoneConstants.APN_TYPE_EMERGENCY)) {
294076f43316a5a6082d601bffd4b6898d0bd81e11fcram                        hasEmergencyApn = true;
294176f43316a5a6082d601bffd4b6898d0bd81e11fcram                        break;
294276f43316a5a6082d601bffd4b6898d0bd81e11fcram                    }
294376f43316a5a6082d601bffd4b6898d0bd81e11fcram                }
294476f43316a5a6082d601bffd4b6898d0bd81e11fcram
294576f43316a5a6082d601bffd4b6898d0bd81e11fcram                if(hasEmergencyApn == false) {
294676f43316a5a6082d601bffd4b6898d0bd81e11fcram                    mAllApnSettings.add(mEmergencyApn);
294776f43316a5a6082d601bffd4b6898d0bd81e11fcram                } else {
294876f43316a5a6082d601bffd4b6898d0bd81e11fcram                    log("addEmergencyApnSetting - E-APN setting is already present");
294976f43316a5a6082d601bffd4b6898d0bd81e11fcram                }
295076f43316a5a6082d601bffd4b6898d0bd81e11fcram            }
295176f43316a5a6082d601bffd4b6898d0bd81e11fcram        }
295276f43316a5a6082d601bffd4b6898d0bd81e11fcram    }
2953c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville}
2954