DcTracker.java revision bca51fc3a191d3ca30df627b75374db0941571c5
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;
43c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.Message;
44a8467dd0c524787104b1ccdddc5e8af10ba729edWink Savilleimport android.os.RegistrantList;
45b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensenimport android.os.ServiceManager;
46c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.SystemClock;
47c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.SystemProperties;
48c9b81a0c05128694c617fcdd67e73821895822feWink Savilleimport android.os.UserHandle;
49c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.provider.Settings;
50cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.provider.Telephony;
51cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.telephony.CellLocation;
520e776303ca82b5bec5db19bb44e0f13b0c7c6400Etan Cohenimport android.telephony.ServiceState;
53c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.telephony.TelephonyManager;
54a8467dd0c524787104b1ccdddc5e8af10ba729edWink Savilleimport android.telephony.SubscriptionManager;
55cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.telephony.cdma.CdmaCellLocation;
56cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport android.telephony.gsm.GsmCellLocation;
57c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.text.TextUtils;
58cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kallaimport android.util.EventLog;
592dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwaltimport android.util.LocalLog;
602b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensenimport android.view.WindowManager;
6199c2e1d6749cfad2a8ca94a47857d8c3bfc09454Wink Savilleimport android.telephony.Rlog;
62c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
63a8467dd0c524787104b1ccdddc5e8af10ba729edWink Savilleimport com.android.internal.telephony.cdma.CDMALTEPhone;
644918296afe1c667e9523cdfc799f558f7ebc2bfbWink Savilleimport com.android.internal.telephony.Phone;
654918296afe1c667e9523cdfc799f558f7ebc2bfbWink Savilleimport com.android.internal.telephony.PhoneBase;
66cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport com.android.internal.telephony.DctConstants;
67cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport com.android.internal.telephony.EventLogTags;
68b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensenimport com.android.internal.telephony.ITelephony;
69c9b81a0c05128694c617fcdd67e73821895822feWink Savilleimport com.android.internal.telephony.TelephonyIntents;
70cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport com.android.internal.telephony.gsm.GSMPhone;
714918296afe1c667e9523cdfc799f558f7ebc2bfbWink Savilleimport com.android.internal.telephony.PhoneConstants;
72f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwaltimport com.android.internal.telephony.PhoneFactory;
73cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Savilleimport com.android.internal.telephony.RILConstants;
74d720945f2be5ea5fe0faf67e67d9ea0e184eba67Alex Yakavenkaimport com.android.internal.telephony.uicc.IccRecords;
75bb36adde615d3d85fa0fc23935197c6bc6a799edAlex Yakavenkaimport com.android.internal.telephony.uicc.UiccController;
76c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport com.android.internal.util.AsyncChannel;
7776f43316a5a6082d601bffd4b6898d0bd81e11fcramimport com.android.internal.util.ArrayUtils;
78c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
79c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport java.io.FileDescriptor;
80c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport java.io.PrintWriter;
81c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport java.util.ArrayList;
8229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwaltimport java.util.Arrays;
83187a39f896f88eb6c5e4306d9595546654825976Wink Savilleimport java.util.concurrent.atomic.AtomicBoolean;
8429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwaltimport java.util.Objects;
85d86df358b8fe07160caa12147b6e4ad34d378ce6xinheimport java.lang.StringBuilder;
86c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
87a8467dd0c524787104b1ccdddc5e8af10ba729edWink Savilleimport com.android.internal.telephony.ServiceStateTracker;
88c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville/**
89c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * {@hide}
90c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville */
91454b1dfd508844b42eb775e4ab2359be74d3672bWink Savillepublic final class DcTracker extends DcTrackerBase {
92cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected final String LOG_TAG = "DCT";
93c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
94b3a03455be44a2a0ffbeb757b193d860c1878599Naveen Kalla    /**
95a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville     * List of messages that are waiting to be posted, when data call disconnect
96a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville     * is complete
97a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville     */
98a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private ArrayList<Message> mDisconnectAllCompleteMsgList = new ArrayList<Message>();
99a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
100a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private RegistrantList mAllDataDisconnectedRegistrants = new RegistrantList();
101a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
102a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected int mDisconnectPendingCount = 0;
103a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
104a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    /**
105cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Handles changes to the APN db.
106b3a03455be44a2a0ffbeb757b193d860c1878599Naveen Kalla     */
107cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private class ApnChangeObserver extends ContentObserver {
108cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        public ApnChangeObserver () {
109cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            super(mDataConnectionTracker);
110cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
111c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
112cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        @Override
113cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        public void onChange(boolean selfChange) {
114cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            sendMessage(obtainMessage(DctConstants.EVENT_APN_CHANGED));
115cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
116cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
117c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
118cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    //***** Instance Variables
119c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
120cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private boolean mReregisterOnReconnectFailure = false;
121c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
122c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
123cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    //***** Constants
124c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
125ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    // Used by puppetmaster/*/radio_stress.py
126ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private static final String PUPPET_MASTER_RADIO_STRESS_TEST = "gsm.defaultpdpcontext.active";
127c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
128ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private static final int POLL_PDP_MILLIS = 5 * 1000;
129c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1302b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen    private static final int PROVISIONING_SPINNER_TIMEOUT_MILLIS = 120 * 1000;
1312b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen
1326bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville    static final Uri PREFERAPN_NO_UPDATE_URI_USING_SUBID =
1336bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville                        Uri.parse("content://telephony/carriers/preferapn_no_update/subId/");
134cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    static final String APN_ID = "apn_id";
135cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
136ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private boolean mCanSetPreferApn = false;
137c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
138187a39f896f88eb6c5e4306d9595546654825976Wink Saville    private AtomicBoolean mAttached = new AtomicBoolean(false);
139187a39f896f88eb6c5e4306d9595546654825976Wink Saville
140cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /** Watches for changes to the APN db. */
141cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private ApnChangeObserver mApnObserver;
142cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
143b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    private final String mProvisionActionName;
144b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    private BroadcastReceiver mProvisionBroadcastReceiver;
1452b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen    private ProgressDialog mProvisioningSpinner;
146b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
147a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean mImsRegistrationState = false;
148a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private ApnContext mWaitCleanUpApnContext = null;
149a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private boolean mDeregistrationAlarmState = false;
150a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private PendingIntent mImsDeregistrationDelayIntent = null;
151a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
152cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    //***** Constructor
153454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    public DcTracker(PhoneBase p) {
154cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        super(p);
155cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("GsmDCT.constructor");
156cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
157cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mDataConnectionTracker = this;
158a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        update();
159cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mApnObserver = new ApnChangeObserver();
160cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        p.getContext().getContentResolver().registerContentObserver(
161cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                Telephony.Carriers.CONTENT_URI, true, mApnObserver);
162cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
163d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        initApnContexts();
164d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt
165d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        for (ApnContext apnContext : mApnContexts.values()) {
166d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            // Register the reconnect and restart actions.
167d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            IntentFilter filter = new IntentFilter();
168d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            filter.addAction(INTENT_RECONNECT_ALARM + '.' + apnContext.getApnType());
169d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            filter.addAction(INTENT_RESTART_TRYSETUP_ALARM + '.' + apnContext.getApnType());
170d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            mPhone.getContext().registerReceiver(mIntentReceiver, filter, null, mPhone);
171d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        }
172d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt
17376f43316a5a6082d601bffd4b6898d0bd81e11fcram        // Add Emergency APN to APN setting list by default to support EPDN in sim absent cases
17476f43316a5a6082d601bffd4b6898d0bd81e11fcram        initEmergencyApnSetting();
17576f43316a5a6082d601bffd4b6898d0bd81e11fcram        addEmergencyApnSetting();
176b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
177b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        mProvisionActionName = "com.android.internal.telephony.PROVISION" + p.getPhoneId();
178cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
179c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
180a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void registerForAllEvents() {
181a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.mCi.registerForAvailable(this, DctConstants.EVENT_RADIO_AVAILABLE, null);
182a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.mCi.registerForOffOrNotAvailable(this,
183a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_RADIO_OFF_OR_NOT_AVAILABLE, null);
184a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.mCi.registerForDataNetworkStateChanged(this,
185a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_DATA_STATE_CHANGED, null);
1860710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // Note, this is fragile - the Phone is now presenting a merged picture
1870710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // of PS (volte) & CS and by diving into its internals you're just seeing
1880710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // the CS data.  This works well for the purposes this is currently used for
1890710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // but that may not always be the case.  Should probably be redesigned to
1900710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // accurately reflect what we're really interested in (registerForCSVoiceCallEnded).
191a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getCallTracker().registerForVoiceCallEnded (this,
192a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_VOICE_CALL_ENDED, null);
193a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getCallTracker().registerForVoiceCallStarted (this,
194a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_VOICE_CALL_STARTED, null);
195a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForDataConnectionAttached(this,
196a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_DATA_CONNECTION_ATTACHED, null);
197a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForDataConnectionDetached(this,
198a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_DATA_CONNECTION_DETACHED, null);
199ffdf8ce51e37e5e45791c9ea11604aa00dffc88eJing Zhao        mPhone.getServiceStateTracker().registerForDataRoamingOn(this,
200a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_ROAMING_ON, null);
201ffdf8ce51e37e5e45791c9ea11604aa00dffc88eJing Zhao        mPhone.getServiceStateTracker().registerForDataRoamingOff(this,
202a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville               DctConstants.EVENT_ROAMING_OFF, null);
203a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForPsRestrictedEnabled(this,
204a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                DctConstants.EVENT_PS_RESTRICT_ENABLED, null);
205a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForPsRestrictedDisabled(this,
206a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                DctConstants.EVENT_PS_RESTRICT_DISABLED, null);
207a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville     //   SubscriptionManager.registerForDdsSwitch(this,
208a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville     //          DctConstants.EVENT_CLEAN_UP_ALL_CONNECTIONS, null);
209a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getServiceStateTracker().registerForDataRegStateOrRatChanged(this,
210a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                DctConstants.EVENT_DATA_RAT_CHANGED, null);
211a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
212cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
213cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public void dispose() {
214bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (DBG) log("DcTracker.dispose");
2154dfda5470a2582c0fb543ead6c79ccf598c580e0Robert Greenwalt
216b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        if (mProvisionBroadcastReceiver != null) {
217b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            mPhone.getContext().unregisterReceiver(mProvisionBroadcastReceiver);
218b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            mProvisionBroadcastReceiver = null;
219b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
2202b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        if (mProvisioningSpinner != null) {
2212b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.dismiss();
2222b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner = null;
2232b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        }
224b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
225cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        cleanUpAllConnections(true, null);
226cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
227cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        super.dispose();
228cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
229a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPhone.getContext().getContentResolver().unregisterContentObserver(mApnObserver);
230a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mApnContexts.clear();
231a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mPrioritySortedApnContexts.clear();
232a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
233a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        destroyDataConnections();
234a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
235a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void unregisterForAllEvents() {
236a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         //Unregister for all events
23722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mPhone.mCi.unregisterForAvailable(this);
23822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mPhone.mCi.unregisterForOffOrNotAvailable(this);
239cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
240a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (r != null) {
241a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            r.unregisterForRecordsLoaded(this);
242a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            mIccRecords.set(null);
243a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
24422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mPhone.mCi.unregisterForDataNetworkStateChanged(this);
245cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getCallTracker().unregisterForVoiceCallEnded(this);
246cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getCallTracker().unregisterForVoiceCallStarted(this);
247cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().unregisterForDataConnectionAttached(this);
248cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().unregisterForDataConnectionDetached(this);
249ffdf8ce51e37e5e45791c9ea11604aa00dffc88eJing Zhao        mPhone.getServiceStateTracker().unregisterForDataRoamingOn(this);
250ffdf8ce51e37e5e45791c9ea11604aa00dffc88eJing Zhao        mPhone.getServiceStateTracker().unregisterForDataRoamingOff(this);
251cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().unregisterForPsRestrictedEnabled(this);
252cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().unregisterForPsRestrictedDisabled(this);
253a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        //SubscriptionManager.unregisterForDdsSwitch(this);
254cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
255cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
256bda761320929f714951c328bfec6a51a1978db97Wink Saville    @Override
257f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt    public void incApnRefCount(String name, LocalLog log, int serialNum) {
258bda761320929f714951c328bfec6a51a1978db97Wink Saville        ApnContext apnContext = mApnContexts.get(name);
259f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        log.log("DcTracker.incApnRefCount on " + name + " found " + apnContext + " (" +
260f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt                serialNum + ")");
261bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (apnContext != null) {
262f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt            apnContext.incRefCount(log, serialNum);
263071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        }
264f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt
265bda761320929f714951c328bfec6a51a1978db97Wink Saville    }
266071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt
267bda761320929f714951c328bfec6a51a1978db97Wink Saville    @Override
268f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt    public void decApnRefCount(String name, LocalLog log, int serialNum) {
269bda761320929f714951c328bfec6a51a1978db97Wink Saville        ApnContext apnContext = mApnContexts.get(name);
270f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        log.log("DcTracker.decApnRefCount on " + name + " found " + apnContext + " (" +
271f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt                serialNum + ")");
272bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (apnContext != null) {
273f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt            apnContext.decRefCount(log, serialNum);
274f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        }
275f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt    }
276f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt
277f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt    @Override
278f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt    public int currentRequestCount() {
279f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        int total = 0;
280f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        for (ApnContext apnContext: mApnContexts.values()) {
281f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt            total += apnContext.getRefCount();
282f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        }
283f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        return total;
284f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt    }
285f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt
286f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt    @Override
287f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt    public void clearApnRefCounts() {
288f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        for (ApnContext apnContext : mApnContexts.values()) {
289f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt            apnContext.clearRefCount();
290f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        }
291f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt    }
292f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt
293f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt    public void snapshotContexts(String logName) {
294f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        LocalLog l = PhoneFactory.getLocalLog(logName);
295f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt        if (l != null) {
296f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt            for (ApnContext apnContext : mApnContexts.values()) {
297f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt                l.log(apnContext.toString());
298f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt                apnContext.copyLogTo(l);
299f65485c99cf0fb0e844ae9e39c2c6cd6bae0fd2cRobert Greenwalt            }
300071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        }
301071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt    }
302071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt
303bda761320929f714951c328bfec6a51a1978db97Wink Saville    @Override
304bda761320929f714951c328bfec6a51a1978db97Wink Saville    public boolean isApnSupported(String name) {
30591bce2abae052df918cb546b9c5d205706ede026Shishir Agrawal        if (name == null) {
30691bce2abae052df918cb546b9c5d205706ede026Shishir Agrawal            loge("isApnSupported: name=null");
30791bce2abae052df918cb546b9c5d205706ede026Shishir Agrawal            return false;
30891bce2abae052df918cb546b9c5d205706ede026Shishir Agrawal        }
309bda761320929f714951c328bfec6a51a1978db97Wink Saville        ApnContext apnContext = mApnContexts.get(name);
310bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (apnContext == null) {
311bda761320929f714951c328bfec6a51a1978db97Wink Saville            loge("Request for unsupported mobile name: " + name);
312bda761320929f714951c328bfec6a51a1978db97Wink Saville            return false;
313071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        }
314bda761320929f714951c328bfec6a51a1978db97Wink Saville        return true;
315bda761320929f714951c328bfec6a51a1978db97Wink Saville    }
316071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt
317bda761320929f714951c328bfec6a51a1978db97Wink Saville    @Override
318bda761320929f714951c328bfec6a51a1978db97Wink Saville    public int getApnPriority(String name) {
319071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        ApnContext apnContext = mApnContexts.get(name);
320071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        if (apnContext == null) {
321bda761320929f714951c328bfec6a51a1978db97Wink Saville            loge("Request for unsupported mobile name: " + name);
322071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        }
323bda761320929f714951c328bfec6a51a1978db97Wink Saville        return apnContext.priority;
324071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt    }
325071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt
326b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    // Turn telephony radio on or off.
327b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    private void setRadio(boolean on) {
328b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        final ITelephony phone = ITelephony.Stub.asInterface(ServiceManager.checkService("phone"));
329b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        try {
330b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            phone.setRadio(on);
331b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        } catch (Exception e) {
332b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            // Ignore.
333b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
334b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    }
335b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
336b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    // Class to handle Intent dispatched with user selects the "Sign-in to network"
337b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    // notification.
338b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    private class ProvisionNotificationBroadcastReceiver extends BroadcastReceiver {
3392b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        private final String mNetworkOperator;
340b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        // Mobile provisioning URL.  Valid while provisioning notification is up.
341b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        // Set prior to notification being posted as URL contains ICCID which
342b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        // disappears when radio is off (which is the case when notification is up).
343b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        private final String mProvisionUrl;
344b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
3452b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        public ProvisionNotificationBroadcastReceiver(String provisionUrl, String networkOperator) {
3462b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mNetworkOperator = networkOperator;
347b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            mProvisionUrl = provisionUrl;
348b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
349b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
350b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        private void setEnableFailFastMobileData(int enabled) {
3516395443719ec3ee0257085945e753d02f603886bRobert Greenwalt            sendMessage(obtainMessage(DctConstants.CMD_SET_ENABLE_FAIL_FAST_MOBILE_DATA, enabled, 0));
352b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
353b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
354b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        private void enableMobileProvisioning() {
355b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            final Message msg = obtainMessage(DctConstants.CMD_ENABLE_MOBILE_PROVISIONING);
356b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            msg.setData(Bundle.forPair(DctConstants.PROVISIONING_URL_KEY, mProvisionUrl));
357b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            sendMessage(msg);
358b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
359b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
360b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        @Override
361b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        public void onReceive(Context context, Intent intent) {
3622b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            // Turning back on the radio can take time on the order of a minute, so show user a
3632b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            // spinner so they know something is going on.
3642b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner = new ProgressDialog(context);
3652b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.setTitle(mNetworkOperator);
3662b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.setMessage(
3672b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    // TODO: Don't borrow "Connecting..." i18n string; give Telephony a version.
3682b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    context.getText(com.android.internal.R.string.media_route_status_connecting));
3692b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.setIndeterminate(true);
3702b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.setCancelable(true);
3712b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            // Allow non-Activity Service Context to create a View.
3722b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.getWindow().setType(
3732b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG);
3742b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            mProvisioningSpinner.show();
3752b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            // After timeout, hide spinner so user can at least use their device.
3762b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            // TODO: Indicate to user that it is taking an unusually long time to connect?
3772b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            sendMessageDelayed(obtainMessage(DctConstants.CMD_CLEAR_PROVISIONING_SPINNER,
3782b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    mProvisioningSpinner), PROVISIONING_SPINNER_TIMEOUT_MILLIS);
379b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            // This code is almost identical to the old
380b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            // ConnectivityService.handleMobileProvisioningAction code.
381b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            setRadio(true);
382b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            setEnableFailFastMobileData(DctConstants.ENABLED);
383b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen            enableMobileProvisioning();
384b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen        }
385b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen    }
386b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
387cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
388cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public boolean isApnTypeActive(String type) {
389cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(type);
390cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) return false;
391cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
392ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        return (apnContext.getDcAc() != null);
393cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
394cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
395cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
396cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public boolean isDataPossible(String apnType) {
397cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
398cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) {
399cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return false;
400cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
401cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean apnContextIsEnabled = apnContext.isEnabled();
402cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        DctConstants.State apnContextState = apnContext.getState();
403cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean apnTypePossible = !(apnContextIsEnabled &&
404cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                (apnContextState == DctConstants.State.FAILED));
405cf5205f70eb1eac497164124187a088ecb03fff5Ram        boolean isEmergencyApn = apnContext.getApnType().equals(PhoneConstants.APN_TYPE_EMERGENCY);
406cf5205f70eb1eac497164124187a088ecb03fff5Ram        // Set the emergency APN availability status as TRUE irrespective of conditions checked in
407cf5205f70eb1eac497164124187a088ecb03fff5Ram        // isDataAllowed() like IN_SERVICE, MOBILE DATA status etc.
408cf5205f70eb1eac497164124187a088ecb03fff5Ram        boolean dataAllowed = isEmergencyApn || isDataAllowed();
409cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean possible = dataAllowed && apnTypePossible;
410cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
4110e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh        if ((apnContext.getApnType().equals(PhoneConstants.APN_TYPE_DEFAULT)
4120e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh                    || apnContext.getApnType().equals(PhoneConstants.APN_TYPE_IA))
4130e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh                && (mPhone.getServiceState().getRilDataRadioTechnology()
4140e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh                == ServiceState.RIL_RADIO_TECHNOLOGY_IWLAN)) {
4150e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh            log("Default data call activation not possible in iwlan.");
4160e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh            possible = false;
4170e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh        }
4180e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh
419ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (VDBG) {
420cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log(String.format("isDataPossible(%s): possible=%b isDataAllowed=%b " +
421cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    "apnTypePossible=%b apnContextisEnabled=%b apnContextState()=%s",
422cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnType, possible, dataAllowed, apnTypePossible,
423cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContextIsEnabled, apnContextState));
424cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
425cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return possible;
426cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
427cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
428cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
429cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void finalize() {
430cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if(DBG) log("finalize");
431cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
432cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
4334a9b3afeb2ec4d573eca335a3706392ecf9f281eWink Saville    private ApnContext addApnContext(String type, NetworkConfig networkConfig) {
434071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt        ApnContext apnContext = new ApnContext(mPhone.getContext(), type, LOG_TAG, networkConfig,
435071b9f85ac559a35430ed37c03a66271977b9d17Robert Greenwalt                this);
436cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mApnContexts.put(type, apnContext);
4373fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        mPrioritySortedApnContexts.add(apnContext);
438bce3d2575122929bb27ec8a37d56e96da39a3ca2Robert Greenwalt        return apnContext;
439cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
440c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
441d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt    protected void initApnContexts() {
442d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        log("initApnContexts: E");
443d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        // Load device network attributes from resources
444d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        String[] networkConfigStrings = mPhone.getContext().getResources().getStringArray(
445d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                com.android.internal.R.array.networkAttributes);
446d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        for (String networkConfigString : networkConfigStrings) {
447d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            NetworkConfig networkConfig = new NetworkConfig(networkConfigString);
448d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            ApnContext apnContext = null;
449d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt
450d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            switch (networkConfig.type) {
451d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE:
452d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_DEFAULT, networkConfig);
453d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
454d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_MMS:
455d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_MMS, networkConfig);
456d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
457d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_SUPL:
458d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_SUPL, networkConfig);
459d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
460d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_DUN:
461d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_DUN, networkConfig);
462d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
463d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_HIPRI:
464d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_HIPRI, networkConfig);
465d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
466d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_FOTA:
467d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_FOTA, networkConfig);
468d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
469d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_IMS:
470d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_IMS, networkConfig);
471d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
472d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_CBS:
473d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_CBS, networkConfig);
474d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
475d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            case ConnectivityManager.TYPE_MOBILE_IA:
476d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                apnContext = addApnContext(PhoneConstants.APN_TYPE_IA, networkConfig);
477d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                break;
478cf5205f70eb1eac497164124187a088ecb03fff5Ram            case ConnectivityManager.TYPE_MOBILE_EMERGENCY:
479cf5205f70eb1eac497164124187a088ecb03fff5Ram                apnContext = addApnContext(PhoneConstants.APN_TYPE_EMERGENCY, networkConfig);
480cf5205f70eb1eac497164124187a088ecb03fff5Ram                break;
481d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            default:
482d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                log("initApnContexts: skipping unknown type=" + networkConfig.type);
483d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt                continue;
484d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            }
485d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt            log("initApnContexts: apnContext=" + apnContext);
486d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        }
487d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt        log("initApnContexts: X mApnContexts=" + mApnContexts);
488d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt    }
489d32b58c8e7cea693c98f49b2291455e917cd9301Robert Greenwalt
490cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
491cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public LinkProperties getLinkProperties(String apnType) {
492cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
493cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext != null) {
494454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville            DcAsyncChannel dcac = apnContext.getDcAc();
495cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (dcac != null) {
496cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("return link properites for " + apnType);
497cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return dcac.getLinkPropertiesSync();
498cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
499cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
500cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("return new LinkProperties");
501cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return new LinkProperties();
502cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
503cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
504608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt    @Override
505608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt    public NetworkCapabilities getNetworkCapabilities(String apnType) {
506608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt        ApnContext apnContext = mApnContexts.get(apnType);
507608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt        if (apnContext!=null) {
508608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt            DcAsyncChannel dataConnectionAc = apnContext.getDcAc();
509608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt            if (dataConnectionAc != null) {
510608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt                if (DBG) {
511608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt                    log("get active pdp is not null, return NetworkCapabilities for " + apnType);
512608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt                }
513608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt                return dataConnectionAc.getNetworkCapabilitiesSync();
514608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt            }
515608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt        }
516608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt        if (DBG) log("return new NetworkCapabilities");
517608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt        return new NetworkCapabilities();
518608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt    }
519cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
520cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
521cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // Return all active apn types
522cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public String[] getActiveApnTypes() {
523cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("get all active apn types");
524cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ArrayList<String> result = new ArrayList<String>();
525cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
526cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
527187a39f896f88eb6c5e4306d9595546654825976Wink Saville            if (mAttached.get() && apnContext.isReady()) {
528cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                result.add(apnContext.getApnType());
529cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla            }
530cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
531c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
532cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return result.toArray(new String[0]);
533cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
534cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
535cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
536cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // Return active apn of specific apn type
537cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public String getActiveApnString(String apnType) {
538ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (VDBG) log( "get active apn string for type:" + apnType);
539cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
540cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext != null) {
541cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            ApnSetting apnSetting = apnContext.getApnSetting();
542cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnSetting != null) {
543cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return apnSetting.apn;
544cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
545cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
546cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return null;
547cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
548cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
549cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
550cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public boolean isApnTypeEnabled(String apnType) {
551cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
552cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) {
553cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return false;
554cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
555cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return apnContext.isEnabled();
556cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
557cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
558cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
559cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void setState(DctConstants.State s) {
560cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("setState should not be used in GSM" + s);
561cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
562cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
563cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // Return state of specific apn type
564cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
565cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public DctConstants.State getState(String apnType) {
566cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
567cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext != null) {
568cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return apnContext.getState();
569c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
570cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return DctConstants.State.FAILED;
571cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
572c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
573c9b81a0c05128694c617fcdd67e73821895822feWink Saville    // Return if apn type is a provisioning apn.
574c9b81a0c05128694c617fcdd67e73821895822feWink Saville    @Override
575c9b81a0c05128694c617fcdd67e73821895822feWink Saville    protected boolean isProvisioningApn(String apnType) {
576c9b81a0c05128694c617fcdd67e73821895822feWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
577c9b81a0c05128694c617fcdd67e73821895822feWink Saville        if (apnContext != null) {
578c9b81a0c05128694c617fcdd67e73821895822feWink Saville            return apnContext.isProvisioningApn();
579c9b81a0c05128694c617fcdd67e73821895822feWink Saville        }
580c9b81a0c05128694c617fcdd67e73821895822feWink Saville        return false;
581c9b81a0c05128694c617fcdd67e73821895822feWink Saville    }
582c9b81a0c05128694c617fcdd67e73821895822feWink Saville
583cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // Return state of overall
584cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
585cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public DctConstants.State getOverallState() {
586cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean isConnecting = false;
587cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean isFailed = true; // All enabled Apns should be FAILED.
588cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean isAnyEnabled = false;
589cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
590cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
591cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnContext.isEnabled()) {
592cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                isAnyEnabled = true;
593cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                switch (apnContext.getState()) {
594cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                case CONNECTED:
595cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                case DISCONNECTING:
596cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (DBG) log("overall state is CONNECTED");
597cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return DctConstants.State.CONNECTED;
598ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                case RETRYING:
599cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                case CONNECTING:
600cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    isConnecting = true;
601cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    isFailed = false;
602cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    break;
603cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                case IDLE:
604cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                case SCANNING:
605cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    isFailed = false;
606cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    break;
607cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                default:
608cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    isAnyEnabled = true;
609cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    break;
610cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
611cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
612c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
613c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
614cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (!isAnyEnabled) { // Nothing enabled. return IDLE.
615cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log( "overall state is IDLE");
616cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return DctConstants.State.IDLE;
617c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
618c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
619cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (isConnecting) {
620cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log( "overall state is CONNECTING");
621cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return DctConstants.State.CONNECTING;
622cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else if (!isFailed) {
623cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log( "overall state is IDLE");
624cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return DctConstants.State.IDLE;
625cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
626cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log( "overall state is FAILED");
627cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return DctConstants.State.FAILED;
628c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
629c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
630c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
631cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
632cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected boolean isApnTypeAvailable(String type) {
633cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (type.equals(PhoneConstants.APN_TYPE_DUN) && fetchDunApn() != null) {
634cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return true;
635c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
636c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
637ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (mAllApnSettings != null) {
638ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            for (ApnSetting apn : mAllApnSettings) {
639cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (apn.canHandleType(type)) {
640cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return true;
641cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
642cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
643c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
644cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return false;
645c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
646c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
647cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
648cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Report on whether data connectivity is enabled for any APN.
649cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @return {@code false} if data connectivity has been explicitly disabled,
650cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * {@code true} otherwise.
651cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
652cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
653cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public boolean getAnyDataEnabled() {
654cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        synchronized (mDataEnabledLock) {
655cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!(mInternalDataEnabled && mUserDataEnabled && sPolicyDataEnabled)) return false;
656cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            for (ApnContext apnContext : mApnContexts.values()) {
657cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // Make sure we don't have a context that is going down
658cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // and is explicitly disabled.
659cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (isDataAllowed(apnContext)) {
660cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return true;
661cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
662cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
663cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return false;
664c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
665c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
666c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
667a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean getAnyDataEnabled(boolean checkUserDataEnabled) {
668a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        synchronized (mDataEnabledLock) {
669a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            if (!(mInternalDataEnabled && (!checkUserDataEnabled || mUserDataEnabled)
670a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                        && (!checkUserDataEnabled || sPolicyDataEnabled)))
671a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                return false;
672a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
673a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            for (ApnContext apnContext : mApnContexts.values()) {
674a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                // Make sure we dont have a context that going down
675a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                // and is explicitly disabled.
676a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                if (isDataAllowed(apnContext)) {
677a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    return true;
678a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                }
679a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            }
680a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            return false;
681a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
682a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
683a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
684cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private boolean isDataAllowed(ApnContext apnContext) {
6850e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh        //If RAT is iwlan then dont allow default/IA PDP at all.
6860e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh        //Rest of APN types can be evaluated for remaining conditions.
6870e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh        if ((apnContext.getApnType().equals(PhoneConstants.APN_TYPE_DEFAULT)
6880e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh                    || apnContext.getApnType().equals(PhoneConstants.APN_TYPE_IA))
6890e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh                && (mPhone.getServiceState().getRilDataRadioTechnology()
6900e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh                == ServiceState.RIL_RADIO_TECHNOLOGY_IWLAN)) {
6910e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh            log("Default data call activation not allowed in iwlan.");
6920e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh            return false;
6930e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh        } else {
6940e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh            return apnContext.isReady() && isDataAllowed();
6950e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh        }
696c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
697c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
698cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    //****** Called from ServiceStateTracker
699c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
700cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Invoked when ServiceStateTracker observes a transition from GPRS
701cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * attach to detach.
702c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
703cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onDataConnectionDetached() {
704cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        /*
705cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * We presently believe it is unnecessary to tear down the PDP context
706cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * when GPRS detaches, but we should stop the network polling.
707cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         */
708cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log ("onDataConnectionDetached: stop polling and notify detached");
709cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        stopNetStatPoll();
710cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        stopDataStallAlarm();
711cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        notifyDataConnection(Phone.REASON_DATA_DETACHED);
712187a39f896f88eb6c5e4306d9595546654825976Wink Saville        mAttached.set(false);
713cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
714c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
715cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void onDataConnectionAttached() {
716cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onDataConnectionAttached");
7177ab10e4710bdb54c6d9a5ee01cd443a42a2689f5Sungmin Choi        mAttached.set(true);
718cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (getOverallState() == DctConstants.State.CONNECTED) {
719cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onDataConnectionAttached: start polling notify attached");
720cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            startNetStatPoll();
721cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            startDataStallAlarm(DATA_STALL_NOT_SUSPECTED);
722cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyDataConnection(Phone.REASON_DATA_ATTACHED);
723cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
724cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // update APN availability so that APN can be enabled.
725cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(Phone.REASON_DATA_ATTACHED);
726cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
72712fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao        if (mAutoAttachOnCreationConfig) {
72812fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao            mAutoAttachOnCreation = true;
72912fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao        }
730ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        setupDataOnConnectableApns(Phone.REASON_DATA_ATTACHED);
731cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
732c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
733cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
734cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected boolean isDataAllowed() {
735cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        final boolean internalDataEnabled;
736cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        synchronized (mDataEnabledLock) {
737cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            internalDataEnabled = mInternalDataEnabled;
738cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
739cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
7409894b3fb2f35e21d9cfd45f233ed093589e14c26sy.yun        boolean attachedState = mAttached.get();
741cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean desiredPowerState = mPhone.getServiceStateTracker().getDesiredPowerState();
7420e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh        int radioTech = mPhone.getServiceState().getRilDataRadioTechnology();
7430e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh        if (radioTech == ServiceState.RIL_RADIO_TECHNOLOGY_IWLAN) {
7440e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh            desiredPowerState = true;
7450e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh        }
7460e42864afb21261d6bd2e9b4aa97f6d01d039a25Yashdev Singh
747cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
7482b0f0da4f9fe8449c578632b491e2f158c201bd5Stuart Scott        boolean recordsLoaded = false;
7492b0f0da4f9fe8449c578632b491e2f158c201bd5Stuart Scott        if (r != null) {
7502b0f0da4f9fe8449c578632b491e2f158c201bd5Stuart Scott            recordsLoaded = r.getRecordsLoaded();
7519232dafa7ea833fc0b3a6024d6c7e23fc8e961eaRobert Greenwalt            if (DBG && !recordsLoaded) log("isDataAllowed getRecordsLoaded=" + recordsLoaded);
7522b0f0da4f9fe8449c578632b491e2f158c201bd5Stuart Scott        }
753cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
754a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        //FIXME always attach
755a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        boolean psRestricted = mIsPsRestricted;
756a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        int phoneNum = TelephonyManager.getDefault().getPhoneCount();
757a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (phoneNum > 1) {
758a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            attachedState = true;
759a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            psRestricted = false;
760a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
7612b0f0da4f9fe8449c578632b491e2f158c201bd5Stuart Scott        int dataSub = SubscriptionManager.getDefaultDataSubId();
762434fa420329b093f68d83862a637c7ded93a4dafCraig Lafayette        boolean defaultDataSelected = SubscriptionManager.isValidSubscriptionId(dataSub);
763b79f845a0451895b0f0b8a926a8571511d476ce8Libin.Tang@motorola.com        PhoneConstants.State state = PhoneConstants.State.IDLE;
7640710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // Note this is explicitly not using mPhone.getState.  See b/19090488.
7650710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // mPhone.getState reports the merge of CS and PS (volte) voice call state
7660710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // but we only care about CS calls here for data/voice concurrency issues.
7670710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // Calling getCallTracker currently gives you just the CS side where the
7680710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // ImsCallTracker is held internally where applicable.
7690710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // This should be redesigned to ask explicitly what we want:
7700710ff7ae80f217782f08528c5406f6d26a29f75Robert Greenwalt        // voiceCallStateAllowDataCall, or dataCallAllowed or something similar.
771b79f845a0451895b0f0b8a926a8571511d476ce8Libin.Tang@motorola.com        if (mPhone.getCallTracker() != null) {
772b79f845a0451895b0f0b8a926a8571511d476ce8Libin.Tang@motorola.com            state = mPhone.getCallTracker().getState();
773b79f845a0451895b0f0b8a926a8571511d476ce8Libin.Tang@motorola.com        }
774cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean allowed =
7759894b3fb2f35e21d9cfd45f233ed093589e14c26sy.yun                    (attachedState || mAutoAttachOnCreation) &&
776cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    recordsLoaded &&
777b79f845a0451895b0f0b8a926a8571511d476ce8Libin.Tang@motorola.com                    (state == PhoneConstants.State.IDLE ||
778cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                     mPhone.getServiceStateTracker().isConcurrentVoiceAndDataAllowed()) &&
779cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    internalDataEnabled &&
7802b0f0da4f9fe8449c578632b491e2f158c201bd5Stuart Scott                    defaultDataSelected &&
781ffdf8ce51e37e5e45791c9ea11604aa00dffc88eJing Zhao                    (!mPhone.getServiceState().getDataRoaming() || getDataOnRoamingEnabled()) &&
782a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    //!mIsPsRestricted &&
783a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    !psRestricted &&
784cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    desiredPowerState;
785cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (!allowed && DBG) {
786cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            String reason = "";
7879894b3fb2f35e21d9cfd45f233ed093589e14c26sy.yun            if (!(attachedState || mAutoAttachOnCreation)) {
7889894b3fb2f35e21d9cfd45f233ed093589e14c26sy.yun                reason += " - Attached= " + attachedState;
789cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
790cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!recordsLoaded) reason += " - SIM not loaded";
791b79f845a0451895b0f0b8a926a8571511d476ce8Libin.Tang@motorola.com            if (state != PhoneConstants.State.IDLE &&
792cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    !mPhone.getServiceStateTracker().isConcurrentVoiceAndDataAllowed()) {
793b79f845a0451895b0f0b8a926a8571511d476ce8Libin.Tang@motorola.com                reason += " - PhoneState= " + state;
794cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                reason += " - Concurrent voice and data not allowed";
795cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
796cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!internalDataEnabled) reason += " - mInternalDataEnabled= false";
7972b0f0da4f9fe8449c578632b491e2f158c201bd5Stuart Scott            if (!defaultDataSelected) reason += " - defaultDataSelected= false";
798ffdf8ce51e37e5e45791c9ea11604aa00dffc88eJing Zhao            if (mPhone.getServiceState().getDataRoaming() && !getDataOnRoamingEnabled()) {
799cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                reason += " - Roaming and data roaming not enabled";
800cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
801cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (mIsPsRestricted) reason += " - mIsPsRestricted= true";
802cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!desiredPowerState) reason += " - desiredPowerState= false";
803cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("isDataAllowed: not allowed due to" + reason);
804c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
805cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return allowed;
806cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
807c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
808c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt    // arg for setupDataOnConnectableApns
809c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt    private enum RetryFailures {
810c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt        // retry failed networks always (the old default)
811c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt        ALWAYS,
812c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt        // retry only when a substantial change has occured.  Either:
813c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt        // 1) we were restricted by voice/data concurrency and aren't anymore
814c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt        // 2) our apn list has change
815c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt        ONLY_ON_CHANGE
816c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt    };
817c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt
818ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private void setupDataOnConnectableApns(String reason) {
819c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt        setupDataOnConnectableApns(reason, RetryFailures.ALWAYS);
820c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt    }
821c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt
822c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt    private void setupDataOnConnectableApns(String reason, RetryFailures retryFailures) {
8233fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if (DBG) log("setupDataOnConnectableApns: " + reason);
8243fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
8253fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        for (ApnContext apnContext : mPrioritySortedApnContexts) {
826735bc2f4524d68155765351912ffae11306c3bd5Chris Manton            ArrayList<ApnSetting> waitingApns = null;
827735bc2f4524d68155765351912ffae11306c3bd5Chris Manton
8283fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (DBG) log("setupDataOnConnectableApns: apnContext " + apnContext);
829735bc2f4524d68155765351912ffae11306c3bd5Chris Manton            if (apnContext.getState() == DctConstants.State.FAILED
830735bc2f4524d68155765351912ffae11306c3bd5Chris Manton                    || apnContext.getState() == DctConstants.State.RETRYING) {
831c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                if (retryFailures == RetryFailures.ALWAYS) {
832c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                    apnContext.setState(DctConstants.State.IDLE);
833c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                } else if (apnContext.isConcurrentVoiceAndDataAllowed() == false &&
834c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                         mPhone.getServiceStateTracker().isConcurrentVoiceAndDataAllowed()) {
835c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                    // RetryFailures.ONLY_ON_CHANGE - check if voice concurrency has changed
836c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                    apnContext.setState(DctConstants.State.IDLE);
837c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                } else {
838c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                    // RetryFailures.ONLY_ON_CHANGE - check if the apns have changed
839c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                    int radioTech = mPhone.getServiceState().getRilDataRadioTechnology();
840c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                    ArrayList<ApnSetting> originalApns = apnContext.getOriginalWaitingApns();
841c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                    if (originalApns != null && originalApns.isEmpty() == false) {
842c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                        waitingApns = buildWaitingApns(apnContext.getApnType(), radioTech);
843c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                        if (originalApns.size() != waitingApns.size() ||
844c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                                originalApns.containsAll(waitingApns) == false) {
845c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                            apnContext.setState(DctConstants.State.IDLE);
846c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                        }
847c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                    }
848c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                }
849cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
850ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (apnContext.isConnectable()) {
851ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                log("setupDataOnConnectableApns: isConnectable() call trySetupData");
852ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                apnContext.setReason(reason);
853c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                trySetupData(apnContext, waitingApns);
854cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
855cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
856c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
857c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
858cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private boolean trySetupData(ApnContext apnContext) {
859c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt        return trySetupData(apnContext, null);
860c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt    }
861c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt
862c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt    private boolean trySetupData(ApnContext apnContext, ArrayList<ApnSetting> waitingApns) {
863cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) {
864cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("trySetupData for type:" + apnContext.getApnType() +
865cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    " due to " + apnContext.getReason() + " apnContext=" + apnContext);
866cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("trySetupData with mIsPsRestricted=" + mIsPsRestricted);
867cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
8682dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        apnContext.requestLog("trySetupData due to " + apnContext.getReason());
869cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
870cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (mPhone.getSimulatedRadioControl() != null) {
871cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // Assume data is connected on the simulator
872cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // FIXME  this can be improved
873cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setState(DctConstants.State.CONNECTED);
874cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
875cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
876cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("trySetupData: X We're on the simulator; assuming connected retValue=true");
877cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return true;
878cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
879cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
880cf5205f70eb1eac497164124187a088ecb03fff5Ram        // Allow SETUP_DATA request for E-APN to be completed during emergency call
881cf5205f70eb1eac497164124187a088ecb03fff5Ram        // and MOBILE DATA On/Off cases as well.
882cf5205f70eb1eac497164124187a088ecb03fff5Ram        boolean isEmergencyApn = apnContext.getApnType().equals(PhoneConstants.APN_TYPE_EMERGENCY);
883c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt        final ServiceStateTracker sst = mPhone.getServiceStateTracker();
884c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt        boolean desiredPowerState = sst.getDesiredPowerState();
885a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        boolean checkUserDataEnabled =
886a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    !(apnContext.getApnType().equals(PhoneConstants.APN_TYPE_IMS));
887cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
888cf5205f70eb1eac497164124187a088ecb03fff5Ram        if (apnContext.isConnectable() && (isEmergencyApn ||
889cf5205f70eb1eac497164124187a088ecb03fff5Ram                (isDataAllowed(apnContext) &&
890cf5205f70eb1eac497164124187a088ecb03fff5Ram                getAnyDataEnabled(checkUserDataEnabled) && !isEmergency()))) {
891ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (apnContext.getState() == DctConstants.State.FAILED) {
8922dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                String str ="trySetupData: make a FAILED ApnContext IDLE so its reusable";
8932dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                if (DBG) log(str);
8942dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                apnContext.requestLog(str);
895ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                apnContext.setState(DctConstants.State.IDLE);
896ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            }
897203e588e3c42a81aa8a56f595119c181a63b12caWink Saville            int radioTech = mPhone.getServiceState().getRilDataRadioTechnology();
898c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt            apnContext.setConcurrentVoiceAndDataAllowed(sst.isConcurrentVoiceAndDataAllowed());
899cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnContext.getState() == DctConstants.State.IDLE) {
900c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                if (waitingApns == null) {
901c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                    waitingApns = buildWaitingApns(apnContext.getApnType(), radioTech);
902c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                }
903cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (waitingApns.isEmpty()) {
904ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    notifyNoData(DcFailCause.MISSING_UNKNOWN_APN, apnContext);
905cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    notifyOffApnsOfAvailability(apnContext.getReason());
9062dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                    String str = "trySetupData: X No APN found retValue=false";
9072dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                    if (DBG) log(str);
9082dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                    apnContext.requestLog(str);
909cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return false;
910cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
911cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setWaitingApns(waitingApns);
912cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (DBG) {
913ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                        log ("trySetupData: Create from mAllApnSettings : "
914ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                    + apnListToString(mAllApnSettings));
915cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
916cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
917cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
918cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
919cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) {
920cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("trySetupData: call setupData, waitingApns : "
921cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        + apnListToString(apnContext.getWaitingApns()));
922cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
923203e588e3c42a81aa8a56f595119c181a63b12caWink Saville            boolean retValue = setupData(apnContext, radioTech);
924cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(apnContext.getReason());
925cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
926cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("trySetupData: X retValue=" + retValue);
927cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return retValue;
928cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
929cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!apnContext.getApnType().equals(PhoneConstants.APN_TYPE_DEFAULT)
930ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    && apnContext.isConnectable()) {
931cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPhone.notifyDataConnectionFailed(apnContext.getReason(), apnContext.getApnType());
932ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            }
933cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(apnContext.getReason());
9342dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt            String str = "trySetupData: X apnContext not 'ready' retValue=false";
9352dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt            apnContext.requestLog(str);
9362dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt            if (DBG) {
9372dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                log(str);
9382dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                if (!apnContext.isConnectable()) log("apnContext.isConnectable = false");
9392dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                if (!isDataAllowed(apnContext)) log("isDataAllowed = false");
9402dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                if (!getAnyDataEnabled(checkUserDataEnabled)) {
9412dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                    log("getAnyDataEnabled(" + checkUserDataEnabled + ") = false");
9422dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                }
9432dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt            }
944cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return false;
945cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
946c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
947c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
948cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
949cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // Disabled apn's still need avail/unavail notificiations - send them out
950cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void notifyOffApnsOfAvailability(String reason) {
951cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
952187a39f896f88eb6c5e4306d9595546654825976Wink Saville            if (!mAttached.get() || !apnContext.isReady()) {
953ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (VDBG) log("notifyOffApnOfAvailability type:" + apnContext.getApnType());
954cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPhone.notifyDataConnection(reason != null ? reason : apnContext.getReason(),
955cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                            apnContext.getApnType(),
956cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                            PhoneConstants.DataState.DISCONNECTED);
957cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
958ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (VDBG) {
959187a39f896f88eb6c5e4306d9595546654825976Wink Saville                    log("notifyOffApnsOfAvailability skipped apn due to attached && isReady " +
960cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            apnContext.toString());
961cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
962c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
963c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
964c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
965c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
966cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
967cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * If tearDown is true, this only tears down a CONNECTED session. Presently,
968cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * there is no mechanism for abandoning an CONNECTING session,
969cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * but would likely involve cancelling pending async requests or
970cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * setting a flag or new state to ignore them when they came in
971cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param tearDown true if the underlying DataConnection should be
972cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * disconnected.
973cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param reason reason for the clean up.
9743fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * @return boolean - true if we did cleanup any connections, false if they
9753fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     *                   were already all disconnected.
976cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
9773fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    protected boolean cleanUpAllConnections(boolean tearDown, String reason) {
978cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("cleanUpAllConnections: tearDown=" + tearDown + " reason=" + reason);
9793fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        boolean didDisconnect = false;
980a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        boolean specificdisable = false;
981a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
982a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (!TextUtils.isEmpty(reason)) {
983a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            specificdisable = reason.equals(Phone.REASON_DATA_SPECIFIC_DISABLED);
984a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
985cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
986cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
9873fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (apnContext.isDisconnected() == false) didDisconnect = true;
988a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            if (specificdisable) {
989a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                if (!apnContext.getApnType().equals(PhoneConstants.APN_TYPE_IMS)) {
990a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    if (DBG) log("ApnConextType: " + apnContext.getApnType());
991a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    apnContext.setReason(reason);
992a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    cleanUpConnection(tearDown, apnContext);
993a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                }
994a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            } else {
995a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                // TODO - only do cleanup if not disconnected
996a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                apnContext.setReason(reason);
997a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                cleanUpConnection(tearDown, apnContext);
998a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            }
999c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
1000cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1001cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        stopNetStatPoll();
1002cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        stopDataStallAlarm();
1003cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1004cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // TODO: Do we need mRequestedApnType?
1005cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mRequestedApnType = PhoneConstants.APN_TYPE_DEFAULT;
1006a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
1007a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("cleanUpConnection: mDisconnectPendingCount = " + mDisconnectPendingCount);
1008a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (tearDown && mDisconnectPendingCount == 0) {
1009a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            notifyDataDisconnectComplete();
1010a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            notifyAllDataDisconnected();
1011a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
1012a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
10133fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        return didDisconnect;
1014cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1015cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1016cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
1017cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Cleanup all connections.
1018cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *
1019cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * TODO: Cleanup only a specified connection passed as a parameter.
1020cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *       Also, make sure when you clean up a conn, if it is last apply
1021cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *       logic as though it is cleanupAllConnections
1022cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *
1023cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param cause for the clean up.
1024cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
1025cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1026cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1027cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onCleanUpAllConnections(String cause) {
1028cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        cleanUpAllConnections(true, cause);
1029cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1030cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1031a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void cleanUpConnection(boolean tearDown, ApnContext apnContext) {
1032cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1033cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) {
1034cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("cleanUpConnection: apn context is null");
1035cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
1036cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1037cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1038454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville        DcAsyncChannel dcac = apnContext.getDcAc();
10392dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        String str = "cleanUpConnection: tearDown=" + tearDown + " reason=" +
10402dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                apnContext.getReason();
10412dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        if (DBG) log(str + " apnContext=" + apnContext);
10422dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        apnContext.requestLog(str);
1043cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (tearDown) {
1044cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnContext.isDisconnected()) {
1045cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // The request is tearDown and but ApnContext is not connected.
1046cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // If apnContext is not enabled anymore, break the linkage to the DCAC/DC.
1047cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setState(DctConstants.State.IDLE);
1048cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (!apnContext.isReady()) {
10494750c8c11836338b024e159f04f0cbd13c7444b9Wink Saville                    if (dcac != null) {
10502dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                        str = "cleanUpConnection: teardown, disconnectd, !ready";
10512dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                        if (DBG) log(str + " apnContext=" + apnContext);
10522dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                        apnContext.requestLog(str);
10534750c8c11836338b024e159f04f0cbd13c7444b9Wink Saville                        dcac.tearDown(apnContext, "", null);
10544750c8c11836338b024e159f04f0cbd13c7444b9Wink Saville                    }
1055cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setDataConnectionAc(null);
1056cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1057cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
1058cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // Connection is still there. Try to clean up.
1059cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (dcac != null) {
1060cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (apnContext.getState() != DctConstants.State.DISCONNECTING) {
1061cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        boolean disconnectAll = false;
1062cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (PhoneConstants.APN_TYPE_DUN.equals(apnContext.getApnType())) {
1063a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                            // CAF_MSIM is this below condition required.
1064a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                            // if (PhoneConstants.APN_TYPE_DUN.equals(PhoneConstants.APN_TYPE_DEFAULT)) {
10651484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                            if (teardownForDun()) {
106645eaa2335e64a8ff1ad8d5e8224c580ef996f370Wink Saville                                if (DBG) {
106745eaa2335e64a8ff1ad8d5e8224c580ef996f370Wink Saville                                    log("cleanUpConnection: disconnectAll DUN connection");
106845eaa2335e64a8ff1ad8d5e8224c580ef996f370Wink Saville                                }
1069cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                // we need to tear it down - we brought it up just for dun and
1070cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                // other people are camped on it and now dun is done.  We need
1071cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                // to stop using it and let the normal apn list get used to find
1072cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                // connections for the remaining desired connections
1073cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                disconnectAll = true;
1074cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            }
1075cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
10762dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                        str = "cleanUpConnection: tearing down" + (disconnectAll ? " all" : "");
10772dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                        if (DBG) log(str + "apnContext=" + apnContext);
10782dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                        apnContext.requestLog(str);
1079cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        Message msg = obtainMessage(DctConstants.EVENT_DISCONNECT_DONE, apnContext);
1080cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (disconnectAll) {
1081ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            apnContext.getDcAc().tearDownAll(apnContext.getReason(), msg);
1082cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        } else {
1083ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            apnContext.getDcAc()
1084cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                .tearDown(apnContext, apnContext.getReason(), msg);
1085cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
1086cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        apnContext.setState(DctConstants.State.DISCONNECTING);
1087a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                        mDisconnectPendingCount++;
1088cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
1089cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
1090cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    // apn is connected but no reference to dcac.
1091cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    // Should not be happen, but reset the state in case.
1092cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setState(DctConstants.State.IDLE);
10932dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                    apnContext.requestLog("cleanUpConnection: connected, bug no DCAC");
1094cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    mPhone.notifyDataConnection(apnContext.getReason(),
1095cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                                apnContext.getApnType());
1096cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1097cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1098cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1099cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // force clean up the data connection.
1100ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (dcac != null) dcac.reqReset();
1101cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setState(DctConstants.State.IDLE);
1102cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
1103cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setDataConnectionAc(null);
1104cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1105cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1106ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        // Make sure reconnection alarm is cleaned up if there is no ApnContext
1107cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // associated to the connection.
1108cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (dcac != null) {
1109ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            cancelReconnectAlarm(apnContext);
1110c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
11112dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        str = "cleanUpConnection: X tearDown=" + tearDown + " reason=" + apnContext.getReason();
11122dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        if (DBG) log(str + " apnContext=" + apnContext + " dcac=" + apnContext.getDcAc());
11132dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        apnContext.requestLog(str);
1114cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1115c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1116cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
11171484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt     * Determine if DUN connection is special and we need to teardown on start/stop
11181484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt     */
11191484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt    private boolean teardownForDun() {
11201484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        // CDMA always needs to do this the profile id is correct
11211484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        final int rilRat = mPhone.getServiceState().getRilDataRadioTechnology();
11221484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        if (ServiceState.isCdma(rilRat)) return true;
11231484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt
11241484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        return (fetchDunApn() != null);
11251484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt    }
11261484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt
11271484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt    /**
1128ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * Cancels the alarm associated with apnContext.
1129cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *
1130ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * @param apnContext on which the alarm should be stopped.
1131cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
1132ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private void cancelReconnectAlarm(ApnContext apnContext) {
1133ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (apnContext == null) return;
1134cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1135ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        PendingIntent intent = apnContext.getReconnectIntent();
1136cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1137cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (intent != null) {
1138cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                AlarmManager am =
1139cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    (AlarmManager) mPhone.getContext().getSystemService(Context.ALARM_SERVICE);
1140cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                am.cancel(intent);
1141ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                apnContext.setReconnectIntent(null);
1142cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1143c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1144c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1145cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
1146cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param types comma delimited list of APN types
1147cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @return array of APN types
1148cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
1149cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private String[] parseTypes(String types) {
1150c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        String[] result;
1151cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // If unset, set to DEFAULT.
1152cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (types == null || types.equals("")) {
1153c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            result = new String[1];
1154cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            result[0] = PhoneConstants.APN_TYPE_ALL;
1155cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1156cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            result = types.split(",");
1157c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
1158c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return result;
1159c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1160c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1161796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan    @Override
1162796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan    protected boolean isPermanentFail(DcFailCause dcFailCause) {
1163796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan        return (dcFailCause.isPermanentFail() &&
1164796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan                (mAttached.get() == false || dcFailCause != DcFailCause.SIGNAL_LOST));
1165796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan    }
1166796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan
1167fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi    private ApnSetting makeApnSetting(Cursor cursor) {
1168fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        String[] types = parseTypes(
1169fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.TYPE)));
1170fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        ApnSetting apn = new ApnSetting(
1171fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers._ID)),
1172fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.NUMERIC)),
1173fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.NAME)),
1174fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.APN)),
1175fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                NetworkUtils.trimV4AddrZeros(
1176fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getString(
1177fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getColumnIndexOrThrow(Telephony.Carriers.PROXY))),
1178fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.PORT)),
1179fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                NetworkUtils.trimV4AddrZeros(
1180fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getString(
1181fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getColumnIndexOrThrow(Telephony.Carriers.MMSC))),
1182fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                NetworkUtils.trimV4AddrZeros(
1183fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getString(
1184fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        cursor.getColumnIndexOrThrow(Telephony.Carriers.MMSPROXY))),
1185fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.MMSPORT)),
1186fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.USER)),
1187fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.PASSWORD)),
1188fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.AUTH_TYPE)),
1189fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                types,
1190fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.PROTOCOL)),
1191fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getString(cursor.getColumnIndexOrThrow(
1192fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        Telephony.Carriers.ROAMING_PROTOCOL)),
1193fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                cursor.getInt(cursor.getColumnIndexOrThrow(
1194fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                        Telephony.Carriers.CARRIER_ENABLED)) == 1,
11959d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.BEARER)),
1196aaf0e8d623b1e94004557573aef9235177ca19e3Amit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.BEARER_BITMASK)),
11979d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.PROFILE_ID)),
11989d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(
11999d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                        Telephony.Carriers.MODEM_COGNITIVE)) == 1,
12009d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.MAX_CONNS)),
12019d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                cursor.getInt(cursor.getColumnIndexOrThrow(
12029d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan                        Telephony.Carriers.WAIT_TIME)),
1203e9701717e43cc5aacbcf624f77a53be92350662cw                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.MAX_CONNS_TIME)),
12043262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers.MTU)),
12053262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.MVNO_TYPE)),
12063262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                cursor.getString(cursor.getColumnIndexOrThrow(Telephony.Carriers.MVNO_MATCH_DATA)));
1207fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        return apn;
1208fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi    }
1209fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi
1210cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private ArrayList<ApnSetting> createApnList(Cursor cursor) {
12113262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal        ArrayList<ApnSetting> mnoApns = new ArrayList<ApnSetting>();
12123262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal        ArrayList<ApnSetting> mvnoApns = new ArrayList<ApnSetting>();
1213fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi        IccRecords r = mIccRecords.get();
1214fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi
1215cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (cursor.moveToFirst()) {
1216cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            do {
12173262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                ApnSetting apn = makeApnSetting(cursor);
12183262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                if (apn == null) {
12193262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                    continue;
12203262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                }
12213262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal
12223262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                if (apn.hasMvnoParams()) {
122363913dc903872c45bab7d2483d633d845dd9c5d6Amit Mahajan                    if (r != null && ApnSetting.mvnoMatches(r, apn.mvnoType, apn.mvnoMatchData)) {
12243262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                        mvnoApns.add(apn);
1225fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                    }
1226fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                } else {
12273262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal                    mnoApns.add(apn);
1228fd5de4dae153c57e13bf8979d6f7a8aa86e3dedeSungmin Choi                }
1229cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } while (cursor.moveToNext());
1230cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
12313262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal
12323262c21feb1eede2917450cfdfd3faa86ee70cabShishir Agrawal        ArrayList<ApnSetting> result = mvnoApns.isEmpty() ? mnoApns : mvnoApns;
1233cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("createApnList: X result=" + result);
1234c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return result;
1235c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1236c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1237454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    private boolean dataConnectionNotInUse(DcAsyncChannel dcac) {
1238ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (DBG) log("dataConnectionNotInUse: check if dcac is inuse dcac=" + dcac);
1239cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
1240ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (apnContext.getDcAc() == dcac) {
1241cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("dataConnectionNotInUse: in use by apnContext=" + apnContext);
1242cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return false;
1243cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1244cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1245cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // TODO: Fix retry handling so free DataConnections have empty apnlists.
1246cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // Probably move retry handling into DataConnections and reduce complexity
1247cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // of DCT.
1248cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("dataConnectionNotInUse: tearDownAll");
1249ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        dcac.tearDownAll("No connection", null);
1250cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("dataConnectionNotInUse: not in use return true");
1251cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return true;
1252cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1253cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1254454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    private DcAsyncChannel findFreeDataConnection() {
1255454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville        for (DcAsyncChannel dcac : mDataConnectionAcHashMap.values()) {
1256cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (dcac.isInactiveSync() && dataConnectionNotInUse(dcac)) {
1257cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) {
1258cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("findFreeDataConnection: found free DataConnection=" +
1259ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                        " dcac=" + dcac);
1260cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1261ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                return dcac;
1262cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1263cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1264cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        log("findFreeDataConnection: NO free DataConnection");
1265cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return null;
1266cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1267cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1268203e588e3c42a81aa8a56f595119c181a63b12caWink Saville    private boolean setupData(ApnContext apnContext, int radioTech) {
1269cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("setupData: apnContext=" + apnContext);
12702dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        apnContext.requestLog("setupData");
1271ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        ApnSetting apnSetting;
12721484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        DcAsyncChannel dcac = null;
1273cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1274ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        apnSetting = apnContext.getNextWaitingApn();
1275ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (apnSetting == null) {
1276cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("setupData: return for no apn found!");
1277cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return false;
1278cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1279cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1280231c3c6271a27b5f03444f4d24b52c41c46566abEtan Cohen        int profileId = apnSetting.profileId;
1281231c3c6271a27b5f03444f4d24b52c41c46566abEtan Cohen        if (profileId == 0) {
1282231c3c6271a27b5f03444f4d24b52c41c46566abEtan Cohen            profileId = getApnProfileID(apnContext.getApnType());
1283231c3c6271a27b5f03444f4d24b52c41c46566abEtan Cohen        }
1284231c3c6271a27b5f03444f4d24b52c41c46566abEtan Cohen
12851484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        // On CDMA, if we're explicitly asking for DUN, we need have
12861484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        // a dun-profiled connection so we can't share an existing one
12871484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        // On GSM/LTE we can share existing apn connections provided they support
12881484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        // this type.
12891484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt        if (apnContext.getApnType() != PhoneConstants.APN_TYPE_DUN ||
12901484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                teardownForDun() == false) {
12911484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt            dcac = checkForCompatibleConnectedApnContext(apnContext);
12921484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt            if (dcac != null) {
12931484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // Get the dcacApnSetting for the connection we want to share.
12941484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                ApnSetting dcacApnSetting = dcac.getApnSettingSync();
12951484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                if (dcacApnSetting != null) {
12961484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                    // Setting is good, so use it.
12971484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                    apnSetting = dcacApnSetting;
12981484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                }
1299ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            }
1300ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        }
1301ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (dcac == null) {
13023fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (isOnlySingleDcAllowed(radioTech)) {
13033fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                if (isHigherPriorityApnContextActive(apnContext)) {
13043fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    if (DBG) {
13053fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        log("setupData: Higher priority ApnContext active.  Ignoring call");
13063fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    }
13073fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    return false;
13083fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                }
13093fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
13103fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                // Only lower priority calls left.  Disconnect them all in this single PDP case
13113fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                // so that we can bring up the requested higher priority call (once we receive
13123fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                // repsonse for deactivate request for the calls we are about to disconnect
13133fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                if (cleanUpAllConnections(true, Phone.REASON_SINGLE_PDN_ARBITRATION)) {
13143fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    // If any call actually requested to be disconnected, means we can't
13153fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    // bring up this connection yet as we need to wait for those data calls
13163fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    // to be disconnected.
13173fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    if (DBG) log("setupData: Some calls are disconnecting first.  Wait and retry");
13183fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    return false;
13193fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                }
13203fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
13213fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                // No other calls are active, so proceed
13223fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                if (DBG) log("setupData: Single pdp. Continue setting up data call.");
13233fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            }
13243fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
1325ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            dcac = findFreeDataConnection();
1326cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1327ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (dcac == null) {
1328ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                dcac = createDataConnection();
1329cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1330cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1331ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (dcac == null) {
1332ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (DBG) log("setupData: No free DataConnection and couldn't create one, WEIRD");
1333cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return false;
1334cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1335cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1336ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (DBG) log("setupData: dcac=" + dcac + " apnSetting=" + apnSetting);
1337cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1338cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        apnContext.setDataConnectionAc(dcac);
1339ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        apnContext.setApnSetting(apnSetting);
1340cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        apnContext.setState(DctConstants.State.CONNECTING);
1341cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
1342cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1343cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        Message msg = obtainMessage();
1344cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        msg.what = DctConstants.EVENT_DATA_SETUP_COMPLETE;
1345cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        msg.obj = apnContext;
134612fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao        dcac.bringUp(apnContext, getInitialMaxRetry(), profileId, radioTech, mAutoAttachOnCreation,
134712fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao                msg);
1348cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1349cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("setupData: initing!");
1350cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return true;
1351cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1352cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1353c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
1354cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Handles changes to the APN database.
1355c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
1356cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void onApnChanged() {
1357cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        DctConstants.State overallState = getOverallState();
1358cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean isDisconnected = (overallState == DctConstants.State.IDLE ||
1359cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                overallState == DctConstants.State.FAILED);
1360cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1361cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (mPhone instanceof GSMPhone) {
1362cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // The "current" may no longer be valid.  MMS depends on this to send properly. TBD
1363cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            ((GSMPhone)mPhone).updateCurrentCarrierInProvider();
1364cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1365cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1366cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // TODO: It'd be nice to only do this if the changed entrie(s)
1367cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // match the current operator.
1368cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onApnChanged: createAllApnList and cleanUpAllConnections");
1369cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        createAllApnList();
13705d5eea6ed231163c225144316b0d1913d48678a4Sungmin Choi        setInitialAttachApn();
13719a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        cleanUpConnectionsOnUpdatedApns(!isDisconnected);
1372bda761320929f714951c328bfec6a51a1978db97Wink Saville
13738f6f52e4f7598e44cea1f9e5f4781291f9060d1dWink Saville        // FIXME: See bug 17426028 maybe no conditional is needed.
13748f6f52e4f7598e44cea1f9e5f4781291f9060d1dWink Saville        if (mPhone.getSubId() == SubscriptionManager.getDefaultDataSubId()) {
1375ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            setupDataOnConnectableApns(Phone.REASON_APN_CHANGED);
1376c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
1377c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1378c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1379c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
1380cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param cid Connection id provided from RIL.
1381cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @return DataConnectionAc associated with specified cid.
1382c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
1383454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    private DcAsyncChannel findDataConnectionAcByCid(int cid) {
1384454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville        for (DcAsyncChannel dcac : mDataConnectionAcHashMap.values()) {
1385cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (dcac.getCidSync() == cid) {
1386cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return dcac;
1387cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1388c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
1389cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return null;
1390c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
1391c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1392cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // TODO: For multiple Active APNs not exactly sure how to do this.
1393c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    @Override
1394cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void gotoIdleAndNotifyDataConnection(String reason) {
1395cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("gotoIdleAndNotifyDataConnection: reason=" + reason);
1396cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        notifyDataConnection(reason);
1397cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mActiveApn = null;
1398cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1399cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
14003fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    /**
14013fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * "Active" here means ApnContext isEnabled() and not in FAILED state
14023fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * @param apnContext to compare with
14033fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * @return true if higher priority active apn found
14043fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     */
14053fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    private boolean isHigherPriorityApnContextActive(ApnContext apnContext) {
14063fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        for (ApnContext otherContext : mPrioritySortedApnContexts) {
14073fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (apnContext.getApnType().equalsIgnoreCase(otherContext.getApnType())) return false;
14083fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (otherContext.isEnabled() && otherContext.getState() != DctConstants.State.FAILED) {
14093fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                return true;
14103fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            }
14113fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        }
14123fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        return false;
14133fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    }
14143fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
14153fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    /**
14163fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * Reports if we support multiple connections or not.
14173fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * This is a combination of factors, based on carrier and RAT.
14183fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * @param rilRadioTech the RIL Radio Tech currently in use
14193fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     * @return true if only single DataConnection is allowed
14203fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt     */
14213fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    private boolean isOnlySingleDcAllowed(int rilRadioTech) {
14223fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        int[] singleDcRats = mPhone.getContext().getResources().getIntArray(
14233fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                com.android.internal.R.array.config_onlySingleDcAllowed);
14243fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        boolean onlySingleDcAllowed = false;
14253fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if (Build.IS_DEBUGGABLE &&
14263fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                SystemProperties.getBoolean("persist.telephony.test.singleDc", false)) {
14273fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            onlySingleDcAllowed = true;
14283fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        }
14293fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if (singleDcRats != null) {
14303fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            for (int i=0; i < singleDcRats.length && onlySingleDcAllowed == false; i++) {
14313fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                if (rilRadioTech == singleDcRats[i]) onlySingleDcAllowed = true;
14323fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            }
14333fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        }
14343fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
14353fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if (DBG) log("isOnlySingleDcAllowed(" + rilRadioTech + "): " + onlySingleDcAllowed);
14363fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        return onlySingleDcAllowed;
14373fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    }
14383fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt
1439cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1440cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void restartRadio() {
1441cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("restartRadio: ************TURN OFF RADIO**************");
1442cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        cleanUpAllConnections(true, Phone.REASON_RADIO_TURNED_OFF);
1443cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.getServiceStateTracker().powerOffRadioSafely(this);
1444cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        /* Note: no need to call setRadioPower(true).  Assuming the desired
1445cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * radio power state is still ON (as tracked by ServiceStateTracker),
1446cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * ServiceStateTracker will call setRadioPower when it receives the
1447cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * RADIO_STATE_CHANGED notification for the power off.  And if the
1448cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * desired power state has changed in the interim, we don't want to
1449cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         * override it with an unconditional power on.
1450cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville         */
1451cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1452cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        int reset = Integer.parseInt(SystemProperties.get("net.ppp.reset-by-timeout", "0"));
1453cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        SystemProperties.set("net.ppp.reset-by-timeout", String.valueOf(reset+1));
1454cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1455cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1456cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    /**
1457cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Return true if data connection need to be setup after disconnected due to
1458cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * reason.
1459cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *
1460cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param reason the reason why data is disconnected
1461cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @return true if try setup data connection is need for this reason
1462cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     */
14633fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt    private boolean retryAfterDisconnected(ApnContext apnContext) {
1464cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean retry = true;
14653fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        String reason = apnContext.getReason();
1466cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
14673fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if ( Phone.REASON_RADIO_TURNED_OFF.equals(reason) ||
14683fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                (isOnlySingleDcAllowed(mPhone.getServiceState().getRilDataRadioTechnology())
14693fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                 && isHigherPriorityApnContextActive(apnContext))) {
1470cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            retry = false;
1471cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1472cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return retry;
1473cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1474cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1475cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void startAlarmForReconnect(int delay, ApnContext apnContext) {
1476cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        String apnType = apnContext.getApnType();
147774672e8ee972f12406b72551261b4cc7e0651933Wink Saville
147874672e8ee972f12406b72551261b4cc7e0651933Wink Saville        Intent intent = new Intent(INTENT_RECONNECT_ALARM + "." + apnType);
147974672e8ee972f12406b72551261b4cc7e0651933Wink Saville        intent.putExtra(INTENT_RECONNECT_ALARM_EXTRA_REASON, apnContext.getReason());
1480cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        intent.putExtra(INTENT_RECONNECT_ALARM_EXTRA_TYPE, apnType);
1481cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1482a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        // Get current sub id.
1483b237a11044ed842d2865ff8c8716befb06b6ca25Wink Saville        int subId = SubscriptionManager.getDefaultDataSubId();
1484a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        intent.putExtra(PhoneConstants.SUBSCRIPTION_KEY, subId);
1485a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
148674672e8ee972f12406b72551261b4cc7e0651933Wink Saville        if (DBG) {
148774672e8ee972f12406b72551261b4cc7e0651933Wink Saville            log("startAlarmForReconnect: delay=" + delay + " action=" + intent.getAction()
148874672e8ee972f12406b72551261b4cc7e0651933Wink Saville                    + " apn=" + apnContext);
148974672e8ee972f12406b72551261b4cc7e0651933Wink Saville        }
149074672e8ee972f12406b72551261b4cc7e0651933Wink Saville
1491ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        PendingIntent alarmIntent = PendingIntent.getBroadcast (mPhone.getContext(), 0,
1492ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                        intent, PendingIntent.FLAG_UPDATE_CURRENT);
1493ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        apnContext.setReconnectIntent(alarmIntent);
1494ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        mAlarmManager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,
1495ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                SystemClock.elapsedRealtime() + delay, alarmIntent);
1496ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    }
1497cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1498ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private void startAlarmForRestartTrySetup(int delay, ApnContext apnContext) {
1499ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        String apnType = apnContext.getApnType();
150074672e8ee972f12406b72551261b4cc7e0651933Wink Saville        Intent intent = new Intent(INTENT_RESTART_TRYSETUP_ALARM + "." + apnType);
1501ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        intent.putExtra(INTENT_RESTART_TRYSETUP_ALARM_EXTRA_TYPE, apnType);
1502cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
150374672e8ee972f12406b72551261b4cc7e0651933Wink Saville        if (DBG) {
150474672e8ee972f12406b72551261b4cc7e0651933Wink Saville            log("startAlarmForRestartTrySetup: delay=" + delay + " action=" + intent.getAction()
150574672e8ee972f12406b72551261b4cc7e0651933Wink Saville                    + " apn=" + apnContext);
150674672e8ee972f12406b72551261b4cc7e0651933Wink Saville        }
1507cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        PendingIntent alarmIntent = PendingIntent.getBroadcast (mPhone.getContext(), 0,
1508cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                        intent, PendingIntent.FLAG_UPDATE_CURRENT);
1509ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        apnContext.setReconnectIntent(alarmIntent);
1510ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        mAlarmManager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,
1511cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                SystemClock.elapsedRealtime() + delay, alarmIntent);
1512cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1513cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1514ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    private void notifyNoData(DcFailCause lastFailCauseCode,
1515cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                              ApnContext apnContext) {
1516cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log( "notifyNoData: type=" + apnContext.getApnType());
1517796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan        if (isPermanentFail(lastFailCauseCode)
1518cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            && (!apnContext.getApnType().equals(PhoneConstants.APN_TYPE_DEFAULT))) {
1519cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mPhone.notifyDataConnectionFailed(apnContext.getReason(), apnContext.getApnType());
1520cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1521cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1522cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1523cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void onRecordsLoaded() {
1524cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onRecordsLoaded: createAllApnList");
152512fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao        mAutoAttachOnCreationConfig = mPhone.getContext().getResources()
152612fffcf0d8df6b8268806d9aa7cc7a662e73743bJing Zhao                .getBoolean(com.android.internal.R.bool.config_auto_attach_data_on_creation);
1527bda761320929f714951c328bfec6a51a1978db97Wink Saville
1528cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        createAllApnList();
15295d5eea6ed231163c225144316b0d1913d48678a4Sungmin Choi        setInitialAttachApn();
153022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mPhone.mCi.getRadioState().isOn()) {
1531cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onRecordsLoaded: notifying data availability");
1532cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(Phone.REASON_SIM_LOADED);
1533cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1534bda761320929f714951c328bfec6a51a1978db97Wink Saville        setupDataOnConnectableApns(Phone.REASON_SIM_LOADED);
1535cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1536cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
15370469925aace7a2254e0cef90bd7398fbd72622faShishir Agrawal    private void onSimNotReady() {
15380469925aace7a2254e0cef90bd7398fbd72622faShishir Agrawal        if (DBG) log("onSimNotReady");
15390469925aace7a2254e0cef90bd7398fbd72622faShishir Agrawal
15400469925aace7a2254e0cef90bd7398fbd72622faShishir Agrawal        cleanUpAllConnections(true, Phone.REASON_SIM_NOT_READY);
15410469925aace7a2254e0cef90bd7398fbd72622faShishir Agrawal        mAllApnSettings = null;
15420469925aace7a2254e0cef90bd7398fbd72622faShishir Agrawal        mAutoAttachOnCreationConfig = false;
15430469925aace7a2254e0cef90bd7398fbd72622faShishir Agrawal    }
15440469925aace7a2254e0cef90bd7398fbd72622faShishir Agrawal
1545cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1546cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onSetDependencyMet(String apnType, boolean met) {
1547cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // don't allow users to tweak hipri to work around default dependency not met
1548cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (PhoneConstants.APN_TYPE_HIPRI.equals(apnType)) return;
1549cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1550cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnType);
1551cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) {
1552cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            loge("onSetDependencyMet: ApnContext not found in onSetDependencyMet(" +
1553cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnType + ", " + met + ")");
1554cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
1555cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1556cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        applyNewState(apnContext, apnContext.isEnabled(), met);
1557cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (PhoneConstants.APN_TYPE_DEFAULT.equals(apnType)) {
1558cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // tie actions on default to similar actions on HIPRI regarding dependencyMet
1559cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext = mApnContexts.get(PhoneConstants.APN_TYPE_HIPRI);
1560cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (apnContext != null) applyNewState(apnContext, apnContext.isEnabled(), met);
1561cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1562cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1563c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1564cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void applyNewState(ApnContext apnContext, boolean enabled, boolean met) {
1565cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean cleanup = false;
1566cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean trySetup = false;
15672dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        String str ="applyNewState(" + apnContext.getApnType() + ", " + enabled +
15682dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                "(" + apnContext.isEnabled() + "), " + met + "(" +
15692dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                apnContext.getDependencyMet() +"))";
15702dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        if (DBG) log(str);
15712dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt        apnContext.requestLog(str);
15722dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt
1573cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext.isReady()) {
1574305122cd621385652826cf7d8cd4e651dc6b5e9fRobert Greenwalt            cleanup = true;
15752428693913ae731d4ace3414429f5e91af24ea36Wink Saville            if (enabled && met) {
15762428693913ae731d4ace3414429f5e91af24ea36Wink Saville                DctConstants.State state = apnContext.getState();
15772428693913ae731d4ace3414429f5e91af24ea36Wink Saville                switch(state) {
15782428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case CONNECTING:
15792428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case SCANNING:
15802428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case CONNECTED:
15812428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case DISCONNECTING:
15822428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        // We're "READY" and active so just return
15832428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        if (DBG) log("applyNewState: 'ready' so return");
15842dde8b1659b1eb231e667bc19307ef294b627bebRobert Greenwalt                        apnContext.requestLog("applyNewState state=" + state + ", so return");
15852428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        return;
15862428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case IDLE:
15872428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        // fall through: this is unexpected but if it happens cleanup and try setup
15882428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case FAILED:
15892428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    case RETRYING: {
15902428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        // We're "READY" but not active so disconnect (cleanup = true) and
15912428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        // connect (trySetup = true) to be sure we retry the connection.
15922428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        trySetup = true;
15932428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        apnContext.setReason(Phone.REASON_DATA_ENABLED);
15942428693913ae731d4ace3414429f5e91af24ea36Wink Saville                        break;
15952428693913ae731d4ace3414429f5e91af24ea36Wink Saville                    }
15962428693913ae731d4ace3414429f5e91af24ea36Wink Saville                }
1597305122cd621385652826cf7d8cd4e651dc6b5e9fRobert Greenwalt            } else if (met) {
1598cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setReason(Phone.REASON_DATA_DISABLED);
1599305122cd621385652826cf7d8cd4e651dc6b5e9fRobert Greenwalt                // If ConnectivityService has disabled this network, stop trying to bring
1600305122cd621385652826cf7d8cd4e651dc6b5e9fRobert Greenwalt                // it up, but do not tear it down - ConnectivityService will do that
1601305122cd621385652826cf7d8cd4e651dc6b5e9fRobert Greenwalt                // directly by talking with the DataConnection.
16021484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                //
16031484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // This doesn't apply to DUN, however.  Those connections have special
16041484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // requirements from carriers and we need stop using them when the dun
16051484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // request goes away.  This applies to both CDMA and GSM because they both
16061484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // can declare the DUN APN sharable by default traffic, thus still satisfying
16071484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                // those requests and not torn down organically.
16081484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                if (apnContext.getApnType() == PhoneConstants.APN_TYPE_DUN && teardownForDun()) {
16091484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                    cleanup = true;
16101484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                } else {
16111484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                    cleanup = false;
16121484bfe9c58cfd2ddf59a5b8dad7373a36de946dRobert Greenwalt                }
1613cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
1614cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setReason(Phone.REASON_DATA_DEPENDENCY_UNMET);
1615cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1616cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1617cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (enabled && met) {
1618cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (apnContext.isEnabled()) {
1619cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setReason(Phone.REASON_DATA_DEPENDENCY_MET);
1620cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
1621cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setReason(Phone.REASON_DATA_ENABLED);
1622c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
1623cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (apnContext.getState() == DctConstants.State.FAILED) {
1624cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnContext.setState(DctConstants.State.IDLE);
1625cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1626cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                trySetup = true;
1627cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1628cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1629cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        apnContext.setEnabled(enabled);
1630cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        apnContext.setDependencyMet(met);
1631cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (cleanup) cleanUpConnection(true, apnContext);
16320d2abb5518d6a86619d2c2db04867c338b2092d4Robert Greenwalt        if (trySetup) {
16330d2abb5518d6a86619d2c2db04867c338b2092d4Robert Greenwalt            apnContext.resetErrorCodeRetries();
16340d2abb5518d6a86619d2c2db04867c338b2092d4Robert Greenwalt            trySetupData(apnContext);
16350d2abb5518d6a86619d2c2db04867c338b2092d4Robert Greenwalt        }
1636cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1637c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1638454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    private DcAsyncChannel checkForCompatibleConnectedApnContext(ApnContext apnContext) {
1639cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        String apnType = apnContext.getApnType();
1640cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnSetting dunSetting = null;
1641cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1642cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (PhoneConstants.APN_TYPE_DUN.equals(apnType)) {
1643cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            dunSetting = fetchDunApn();
1644cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1645ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (DBG) {
1646ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            log("checkForCompatibleConnectedApnContext: apnContext=" + apnContext );
1647ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        }
1648cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1649454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville        DcAsyncChannel potentialDcac = null;
1650ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        ApnContext potentialApnCtx = null;
1651ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        for (ApnContext curApnCtx : mApnContexts.values()) {
1652454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville            DcAsyncChannel curDcac = curApnCtx.getDcAc();
1653ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (curDcac != null) {
1654ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                ApnSetting apnSetting = curApnCtx.getApnSetting();
1655a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                log("apnSetting: " + apnSetting);
1656cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (dunSetting != null) {
1657cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (dunSetting.equals(apnSetting)) {
1658ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                        switch (curApnCtx.getState()) {
1659cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            case CONNECTED:
1660cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                if (DBG) {
1661ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                    log("checkForCompatibleConnectedApnContext:"
1662ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                            + " found dun conn=" + curDcac
1663ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                            + " curApnCtx=" + curApnCtx);
1664cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                }
1665ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                return curDcac;
1666ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            case RETRYING:
1667cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            case CONNECTING:
1668ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                potentialDcac = curDcac;
1669ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                potentialApnCtx = curApnCtx;
1670cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            default:
1671cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                // Not connected, potential unchanged
1672cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                break;
1673cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
1674cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
1675cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else if (apnSetting != null && apnSetting.canHandleType(apnType)) {
1676ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    switch (curApnCtx.getState()) {
1677cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        case CONNECTED:
1678cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            if (DBG) {
1679ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                log("checkForCompatibleConnectedApnContext:"
1680ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                        + " found canHandle conn=" + curDcac
1681ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                        + " curApnCtx=" + curApnCtx);
1682cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            }
1683ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            return curDcac;
1684ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                        case RETRYING:
1685cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        case CONNECTING:
1686ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            potentialDcac = curDcac;
1687ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                            potentialApnCtx = curApnCtx;
1688cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        default:
1689cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            // Not connected, potential unchanged
1690cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            break;
1691cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
1692cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1693ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            } else {
1694ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (VDBG) {
1695ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    log("checkForCompatibleConnectedApnContext: not conn curApnCtx=" + curApnCtx);
1696ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                }
1697cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1698cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1699ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (potentialDcac != null) {
1700cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) {
1701ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                log("checkForCompatibleConnectedApnContext: found potential conn=" + potentialDcac
1702ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                        + " curApnCtx=" + potentialApnCtx);
1703cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
1704ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            return potentialDcac;
1705cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1706c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1707ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (DBG) log("checkForCompatibleConnectedApnContext: NO conn apnContext=" + apnContext);
1708cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return null;
1709cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1710c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1711cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1712cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onEnableApn(int apnId, int enabled) {
1713cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnIdToType(apnId));
1714cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext == null) {
1715cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            loge("onEnableApn(" + apnId + ", " + enabled + "): NO ApnContext");
1716cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
1717cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1718cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // TODO change our retry manager to use the appropriate numbers for the new APN
1719cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onEnableApn: apnContext=" + apnContext + " call applyNewState");
1720cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        applyNewState(apnContext, enabled == DctConstants.ENABLED, apnContext.getDependencyMet());
1721cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1722c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1723cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1724cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    // TODO: We shouldnt need this.
1725cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected boolean onTrySetupData(String reason) {
1726cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onTrySetupData: reason=" + reason);
1727ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        setupDataOnConnectableApns(reason);
1728cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return true;
1729cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1730c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1731cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected boolean onTrySetupData(ApnContext apnContext) {
1732cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onTrySetupData: apnContext=" + apnContext);
1733cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return trySetupData(apnContext);
1734cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1735c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1736cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1737cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onRoamingOff() {
1738cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onRoamingOff");
1739c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
17406bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (!mUserDataEnabled) return;
1741c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
1742bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (getDataOnRoamingEnabled() == false) {
1743cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(Phone.REASON_ROAMING_OFF);
1744bda761320929f714951c328bfec6a51a1978db97Wink Saville            setupDataOnConnectableApns(Phone.REASON_ROAMING_OFF);
1745cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1746cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyDataConnection(Phone.REASON_ROAMING_OFF);
1747cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1748cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1749cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1750cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1751cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onRoamingOn() {
17526bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (DBG) log("onRoamingOn");
1753bda761320929f714951c328bfec6a51a1978db97Wink Saville
17546bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (!mUserDataEnabled) return;
1755cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1756bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (getDataOnRoamingEnabled()) {
1757cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onRoamingOn: setup data on roaming");
1758ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            setupDataOnConnectableApns(Phone.REASON_ROAMING_ON);
1759cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyDataConnection(Phone.REASON_ROAMING_ON);
1760cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1761cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onRoamingOn: Tear down data connection on roaming.");
1762cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cleanUpAllConnections(true, Phone.REASON_ROAMING_ON);
1763cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(Phone.REASON_ROAMING_ON);
1764cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1765cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1766cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1767cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1768cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onRadioAvailable() {
1769cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onRadioAvailable");
1770cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (mPhone.getSimulatedRadioControl() != null) {
1771cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // Assume data is connected on the simulator
1772cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // FIXME  this can be improved
1773cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // setState(DctConstants.State.CONNECTED);
1774cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyDataConnection(null);
1775cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1776cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("onRadioAvailable: We're on the simulator; assuming data is connected");
1777cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1778cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1779cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
1780cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (r != null && r.getRecordsLoaded()) {
1781cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyOffApnsOfAvailability(null);
1782cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1783cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1784cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (getOverallState() != DctConstants.State.IDLE) {
1785cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cleanUpConnection(true, null);
1786cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1787cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1788cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1789cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1790cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onRadioOffOrNotAvailable() {
1791cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // Make sure our reconnect delay starts at the initial value
1792cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // next time the radio comes on
1793cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1794cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mReregisterOnReconnectFailure = false;
1795cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1796cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (mPhone.getSimulatedRadioControl() != null) {
1797cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // Assume data is connected on the simulator
1798cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // FIXME  this can be improved
1799cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("We're on the simulator; assuming radio off is meaningless");
1800cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1801cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onRadioOffOrNotAvailable: is off and clean up all connections");
1802cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cleanUpAllConnections(false, Phone.REASON_RADIO_TURNED_OFF);
1803cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1804cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        notifyOffApnsOfAvailability(null);
1805cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
1806cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1807c9b81a0c05128694c617fcdd67e73821895822feWink Saville    @Override
1808c9b81a0c05128694c617fcdd67e73821895822feWink Saville    protected void completeConnection(ApnContext apnContext) {
1809c9b81a0c05128694c617fcdd67e73821895822feWink Saville        boolean isProvApn = apnContext.isProvisioningApn();
1810c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1811c9b81a0c05128694c617fcdd67e73821895822feWink Saville        if (DBG) log("completeConnection: successful, notify the world apnContext=" + apnContext);
1812c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1813c9b81a0c05128694c617fcdd67e73821895822feWink Saville        if (mIsProvisioning && !TextUtils.isEmpty(mProvisioningUrl)) {
1814c9b81a0c05128694c617fcdd67e73821895822feWink Saville            if (DBG) {
1815c9b81a0c05128694c617fcdd67e73821895822feWink Saville                log("completeConnection: MOBILE_PROVISIONING_ACTION url="
1816c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        + mProvisioningUrl);
1817c9b81a0c05128694c617fcdd67e73821895822feWink Saville            }
1818c8dc0c8244aac9f3985a53bc94b8ec2e295db430Robert Greenwalt            Intent newIntent = Intent.makeMainSelectorActivity(Intent.ACTION_MAIN,
1819c8dc0c8244aac9f3985a53bc94b8ec2e295db430Robert Greenwalt                    Intent.CATEGORY_APP_BROWSER);
1820c8dc0c8244aac9f3985a53bc94b8ec2e295db430Robert Greenwalt            newIntent.setData(Uri.parse(mProvisioningUrl));
1821c9b81a0c05128694c617fcdd67e73821895822feWink Saville            newIntent.setFlags(Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT |
1822c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    Intent.FLAG_ACTIVITY_NEW_TASK);
1823c9b81a0c05128694c617fcdd67e73821895822feWink Saville            try {
1824c9b81a0c05128694c617fcdd67e73821895822feWink Saville                mPhone.getContext().startActivity(newIntent);
1825c9b81a0c05128694c617fcdd67e73821895822feWink Saville            } catch (ActivityNotFoundException e) {
1826c9b81a0c05128694c617fcdd67e73821895822feWink Saville                loge("completeConnection: startActivityAsUser failed" + e);
1827c9b81a0c05128694c617fcdd67e73821895822feWink Saville            }
1828c9b81a0c05128694c617fcdd67e73821895822feWink Saville        }
1829c9b81a0c05128694c617fcdd67e73821895822feWink Saville        mIsProvisioning = false;
1830c9b81a0c05128694c617fcdd67e73821895822feWink Saville        mProvisioningUrl = null;
18312b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        if (mProvisioningSpinner != null) {
18322b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            sendMessage(obtainMessage(DctConstants.CMD_CLEAR_PROVISIONING_SPINNER,
18332b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    mProvisioningSpinner));
18342b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen        }
1835c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1836c9b81a0c05128694c617fcdd67e73821895822feWink Saville        mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
1837c9b81a0c05128694c617fcdd67e73821895822feWink Saville        startNetStatPoll();
1838c9b81a0c05128694c617fcdd67e73821895822feWink Saville        startDataStallAlarm(DATA_STALL_NOT_SUSPECTED);
1839c9b81a0c05128694c617fcdd67e73821895822feWink Saville    }
1840c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1841ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    /**
1842ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * A SETUP (aka bringUp) has completed, possibly with an error. If
1843ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * there is an error this method will call {@link #onDataSetupCompleteError}.
1844ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     */
1845cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
1846cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onDataSetupComplete(AsyncResult ar) {
1847608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt
1848ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        DcFailCause cause = DcFailCause.UNKNOWN;
1849cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean handleError = false;
1850cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = null;
1851cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1852cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if(ar.userObj instanceof ApnContext){
1853cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext = (ApnContext)ar.userObj;
1854cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1855cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            throw new RuntimeException("onDataSetupComplete: No apnContext");
1856cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
1857cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1858cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (ar.exception == null) {
1859454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville            DcAsyncChannel dcac = apnContext.getDcAc();
1860cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1861cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (RADIO_TESTS) {
1862cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // Note: To change radio.test.onDSC.null.dcac from command line you need to
1863cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // adb root and adb remount and from the command line you can only change the
1864cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // value to 1 once. To change it a second time you can reboot or execute
1865cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // adb shell stop and then adb shell start. The command line to set the value is:
1866ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink 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');"
1867cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                ContentResolver cr = mPhone.getContext().getContentResolver();
1868cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                String radioTestProperty = "radio.test.onDSC.null.dcac";
1869cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (Settings.System.getInt(cr, radioTestProperty, 0) == 1) {
1870cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("onDataSetupComplete: " + radioTestProperty +
1871cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            " is true, set dcac to null and reset property to false");
1872cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    dcac = null;
1873cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    Settings.System.putInt(cr, radioTestProperty, 0);
1874cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("onDataSetupComplete: " + radioTestProperty + "=" +
1875cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            Settings.System.getInt(mPhone.getContext().getContentResolver(),
1876cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                    radioTestProperty, -1));
1877cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1878c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
1879cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (dcac == null) {
1880cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("onDataSetupComplete: no connection to DC, handle as error");
1881ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                cause = DcFailCause.CONNECTION_TO_DATACONNECTIONAC_BROKEN;
1882cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                handleError = true;
1883cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
1884cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                ApnSetting apn = apnContext.getApnSetting();
1885cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) {
1886cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("onDataSetupComplete: success apn=" + (apn == null ? "unknown" : apn.apn));
1887cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1888cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (apn != null && apn.proxy != null && apn.proxy.length() != 0) {
1889cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    try {
1890cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        String port = apn.port;
1891cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (TextUtils.isEmpty(port)) port = "8080";
18929c180aedfc9f0d20525c0128487d3500e6c0a715Jason Monk                        ProxyInfo proxy = new ProxyInfo(apn.proxy,
1893cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                Integer.parseInt(port), null);
1894cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        dcac.setLinkPropertiesHttpProxySync(proxy);
1895cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    } catch (NumberFormatException e) {
1896cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        loge("onDataSetupComplete: NumberFormatException making ProxyProperties (" +
1897cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                                apn.port + "): " + e);
1898cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
1899cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1900cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
1901cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // everything is setup
1902cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if(TextUtils.equals(apnContext.getApnType(),PhoneConstants.APN_TYPE_DEFAULT)) {
1903ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    SystemProperties.set(PUPPET_MASTER_RADIO_STRESS_TEST, "true");
190422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                    if (mCanSetPreferApn && mPreferredApn == null) {
1905cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (DBG) log("onDataSetupComplete: PREFERED APN is null");
1906cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        mPreferredApn = apn;
1907cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (mPreferredApn != null) {
1908cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            setPreferredApn(mPreferredApn.id);
1909cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
1910cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
1911cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
1912ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    SystemProperties.set(PUPPET_MASTER_RADIO_STRESS_TEST, "false");
1913cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
1914c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1915c9b81a0c05128694c617fcdd67e73821895822feWink Saville                // A connection is setup
1916c9b81a0c05128694c617fcdd67e73821895822feWink Saville                apnContext.setState(DctConstants.State.CONNECTED);
1917c9b81a0c05128694c617fcdd67e73821895822feWink Saville                boolean isProvApn = apnContext.isProvisioningApn();
1918b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                final ConnectivityManager cm = ConnectivityManager.from(mPhone.getContext());
1919b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                if (mProvisionBroadcastReceiver != null) {
1920b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    mPhone.getContext().unregisterReceiver(mProvisionBroadcastReceiver);
1921b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    mProvisionBroadcastReceiver = null;
1922b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                }
1923c9b81a0c05128694c617fcdd67e73821895822feWink Saville                if ((!isProvApn) || mIsProvisioning) {
1924b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // Hide any provisioning notification.
1925b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    cm.setProvisioningNotificationVisible(false, ConnectivityManager.TYPE_MOBILE,
1926b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                            mProvisionActionName);
1927c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // Complete the connection normally notifying the world we're connected.
1928c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // We do this if this isn't a special provisioning apn or if we've been
1929c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // told its time to provision.
1930c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    completeConnection(apnContext);
1931c9b81a0c05128694c617fcdd67e73821895822feWink Saville                } else {
1932c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // This is a provisioning APN that we're reporting as connected. Later
1933c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // when the user desires to upgrade this to a "default" connection,
1934c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // mIsProvisioning == true, we'll go through the code path above.
1935c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // mIsProvisioning becomes true when CMD_ENABLE_MOBILE_PROVISIONING
1936c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    // is sent to the DCT.
1937c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    if (DBG) {
1938c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        log("onDataSetupComplete: successful, BUT send connected to prov apn as"
1939c9b81a0c05128694c617fcdd67e73821895822feWink Saville                                + " mIsProvisioning:" + mIsProvisioning + " == false"
1940c9b81a0c05128694c617fcdd67e73821895822feWink Saville                                + " && (isProvisioningApn:" + isProvApn + " == true");
1941c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    }
1942c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1943b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // While radio is up, grab provisioning URL.  The URL contains ICCID which
1944b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // disappears when radio is off.
1945b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    mProvisionBroadcastReceiver = new ProvisionNotificationBroadcastReceiver(
19462b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                            cm.getMobileProvisioningUrl(),
19472b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                            TelephonyManager.getDefault().getNetworkOperatorName());
1948b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    mPhone.getContext().registerReceiver(mProvisionBroadcastReceiver,
1949b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                            new IntentFilter(mProvisionActionName));
1950b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // Put up user notification that sign-in is required.
1951b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    cm.setProvisioningNotificationVisible(true, ConnectivityManager.TYPE_MOBILE,
1952b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                            mProvisionActionName);
1953b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // Turn off radio to save battery and avoid wasting carrier resources.
1954b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    // The network isn't usable and network validation will just fail anyhow.
1955b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen                    setRadio(false);
1956b449dc75ef4b9cb996c34a11e758f7e6ca193240Paul Jensen
1957c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    Intent intent = new Intent(
1958c9b81a0c05128694c617fcdd67e73821895822feWink Saville                            TelephonyIntents.ACTION_DATA_CONNECTION_CONNECTED_TO_PROVISIONING_APN);
1959c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    intent.putExtra(PhoneConstants.DATA_APN_KEY, apnContext.getApnSetting().apn);
1960c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    intent.putExtra(PhoneConstants.DATA_APN_TYPE_KEY, apnContext.getApnType());
1961c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1962c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    String apnType = apnContext.getApnType();
1963c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    LinkProperties linkProperties = getLinkProperties(apnType);
1964c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    if (linkProperties != null) {
1965c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        intent.putExtra(PhoneConstants.DATA_LINK_PROPERTIES_KEY, linkProperties);
1966c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        String iface = linkProperties.getInterfaceName();
1967c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        if (iface != null) {
1968c9b81a0c05128694c617fcdd67e73821895822feWink Saville                            intent.putExtra(PhoneConstants.DATA_IFACE_NAME_KEY, iface);
1969c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        }
1970c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    }
1971608588e9af271f0f5640236809f27c626f9d98e9Robert Greenwalt                    NetworkCapabilities networkCapabilities = getNetworkCapabilities(apnType);
197296cce86cf08e37e0f09ed5057b1196e26b302743Robert Greenwalt                    if (networkCapabilities != null) {
197396cce86cf08e37e0f09ed5057b1196e26b302743Robert Greenwalt                        intent.putExtra(PhoneConstants.DATA_NETWORK_CAPABILITIES_KEY,
197496cce86cf08e37e0f09ed5057b1196e26b302743Robert Greenwalt                                networkCapabilities);
1975c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    }
1976c9b81a0c05128694c617fcdd67e73821895822feWink Saville
1977c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    mPhone.getContext().sendBroadcastAsUser(intent, UserHandle.ALL);
1978c9b81a0c05128694c617fcdd67e73821895822feWink Saville                }
1979c9b81a0c05128694c617fcdd67e73821895822feWink Saville                if (DBG) {
1980c9b81a0c05128694c617fcdd67e73821895822feWink Saville                    log("onDataSetupComplete: SETUP complete type=" + apnContext.getApnType()
1981c9b81a0c05128694c617fcdd67e73821895822feWink Saville                        + ", reason:" + apnContext.getReason());
1982c9b81a0c05128694c617fcdd67e73821895822feWink Saville                }
1983c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
1984cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
1985ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            cause = (DcFailCause) (ar.result);
1986cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) {
1987cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                ApnSetting apn = apnContext.getApnSetting();
1988cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log(String.format("onDataSetupComplete: error apn=%s cause=%s",
1989cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        (apn == null ? "unknown" : apn.apn), cause));
1990c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
1991cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (cause.isEventLoggable()) {
1992cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // Log this failure to the Event Logs.
1993cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                int cid = getCellLocationId();
1994cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                EventLog.writeEvent(EventLogTags.PDP_SETUP_FAIL,
1995cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        cause.ordinal(), cid, TelephonyManager.getDefault().getNetworkType());
1996c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
19970742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela            ApnSetting apn = apnContext.getApnSetting();
19980742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela            mPhone.notifyPreciseDataConnectionFailed(apnContext.getReason(),
19990742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela                    apnContext.getApnType(), apn != null ? apn.apn : "unknown", cause.toString());
2000cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2001cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // Count permanent failures and remove the APN we just tried
2002796d3c22f21041116110735c92d7e2c3a7c8f60dAmit Mahajan            if (isPermanentFail(cause)) apnContext.decWaitingApnsPermFailCount();
2003cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2004cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.removeWaitingApn(apnContext.getApnSetting());
2005cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) {
2006cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log(String.format("onDataSetupComplete: WaitingApns.size=%d" +
2007cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        " WaitingApnsPermFailureCountDown=%d",
2008cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        apnContext.getWaitingApns().size(),
2009cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        apnContext.getWaitingApnsPermFailCount()));
2010c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2011cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            handleError = true;
2012cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2013cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2014cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (handleError) {
2015ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            onDataSetupCompleteError(ar);
2016ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        }
2017a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2018a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        /* If flag is set to false after SETUP_DATA_CALL is invoked, we need
2019a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         * to clean data connections.
2020a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         */
2021a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (!mInternalDataEnabled) {
2022a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            cleanUpAllConnections(null);
2023a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2024a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2025ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    }
2026cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2027ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    /**
2028ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville     * @return number of milli-seconds to delay between trying apns'
2029ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville     */
2030ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville    private int getApnDelay() {
2031ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville        if (mFailFast) {
2032ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville            return SystemProperties.getInt("persist.radio.apn_ff_delay",
2033ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville                    APN_FAIL_FAST_DELAY_DEFAULT_MILLIS);
2034ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville        } else {
2035ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville            return SystemProperties.getInt("persist.radio.apn_delay", APN_DELAY_DEFAULT_MILLIS);
2036ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville        }
2037ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville    }
2038ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville
2039ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville    /**
2040ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * Error has occurred during the SETUP {aka bringUP} request and the DCT
2041ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * should either try the next waiting APN or start over from the
2042ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * beginning if the list is empty. Between each SETUP request there will
2043ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville     * be a delay defined by {@link #getApnDelay()}.
2044ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     */
2045ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    @Override
2046ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    protected void onDataSetupCompleteError(AsyncResult ar) {
2047ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        String reason = "";
2048ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        ApnContext apnContext = null;
2049ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2050ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if(ar.userObj instanceof ApnContext){
2051ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            apnContext = (ApnContext)ar.userObj;
2052ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        } else {
2053ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            throw new RuntimeException("onDataSetupCompleteError: No apnContext");
2054ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        }
2055ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2056ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        // See if there are more APN's to try
2057ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (apnContext.getWaitingApns().isEmpty()) {
2058ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            apnContext.setState(DctConstants.State.FAILED);
2059ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            mPhone.notifyDataConnection(Phone.REASON_APN_FAILED, apnContext.getApnType());
2060ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2061ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            apnContext.setDataConnectionAc(null);
2062ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2063ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (apnContext.getWaitingApnsPermFailCount() == 0) {
2064ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (DBG) {
2065017166876a51eda9ae6b3254119023604e249bc5Wink Saville                    log("onDataSetupCompleteError: All APN's had permanent failures, stop retrying");
2066c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
2067cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
2068e056c8263473f67dc78630f5535c3664fabacd23Wink Saville                int delay = getApnDelay();
2069ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                if (DBG) {
2070da1dd2e1bccb6141109a65d4e40253f39c405537Wink Saville                    log("onDataSetupCompleteError: Not all APN's had permanent failures delay="
2071da1dd2e1bccb6141109a65d4e40253f39c405537Wink Saville                            + delay);
2072ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                }
2073017166876a51eda9ae6b3254119023604e249bc5Wink Saville                startAlarmForRestartTrySetup(delay, apnContext);
2074c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2075ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        } else {
2076da1dd2e1bccb6141109a65d4e40253f39c405537Wink Saville            if (DBG) log("onDataSetupCompleteError: Try next APN");
2077ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            apnContext.setState(DctConstants.State.SCANNING);
2078ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            // Wait a bit before trying the next APN, so that
2079ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            // we're not tying up the RIL command channel
2080ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville            startAlarmForReconnect(getApnDelay(), apnContext);
2081c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2082c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2083c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2084c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
2085cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Called when EVENT_DISCONNECT_DONE is received.
2086c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
2087cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2088cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onDisconnectDone(int connId, AsyncResult ar) {
2089cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = null;
2090cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2091cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (ar.userObj instanceof ApnContext) {
2092cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext = (ApnContext) ar.userObj;
2093cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
2094cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            loge("onDisconnectDone: Invalid ar in onDisconnectDone, ignore");
2095cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
2096c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2097c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2098cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if(DBG) log("onDisconnectDone: EVENT_DISCONNECT_DONE apnContext=" + apnContext);
2099cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        apnContext.setState(DctConstants.State.IDLE);
2100cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2101cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
2102cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2103cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // if all data connection are gone, check whether Airplane mode request was
2104cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // pending.
2105cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (isDisconnected()) {
2106cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (mPhone.getServiceStateTracker().processPendingRadioPowerOffAfterDataOff()) {
2107449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                if(DBG) log("onDisconnectDone: radio will be turned off, no retries");
2108cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // Radio will be turned off. No need to retry data setup
2109cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setApnSetting(null);
2110cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnContext.setDataConnectionAc(null);
2111a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2112a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                // Need to notify disconnect as well, in the case of switching Airplane mode.
2113a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                // Otherwise, it would cause 30s delayed to turn on Airplane mode.
2114a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                if (mDisconnectPendingCount > 0)
2115a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    mDisconnectPendingCount--;
2116a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2117a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                if (mDisconnectPendingCount == 0) {
2118a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    notifyDataDisconnectComplete();
2119a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    notifyAllDataDisconnected();
2120a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                }
2121cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return;
2122cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2123c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2124c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2125cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // If APN is still enabled, try to bring it back up automatically
21263fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt        if (mAttached.get() && apnContext.isReady() && retryAfterDisconnected(apnContext)) {
2127ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            SystemProperties.set(PUPPET_MASTER_RADIO_STRESS_TEST, "false");
2128cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // Wait a bit before trying the next APN, so that
2129cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // we're not tying up the RIL command channel.
2130cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // This also helps in any external dependency to turn off the context.
2131449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville            if(DBG) log("onDisconnectDone: attached, ready and retry after disconnect");
2132ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville            startAlarmForReconnect(getApnDelay(), apnContext);
2133c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
2134449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville            boolean restartRadioAfterProvisioning = mPhone.getContext().getResources().getBoolean(
2135449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                    com.android.internal.R.bool.config_restartRadioAfterProvisioning);
2136449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville
2137449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville            if (apnContext.isProvisioningApn() && restartRadioAfterProvisioning) {
2138449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                log("onDisconnectDone: restartRadio after provisioning");
2139449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                restartRadio();
2140449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville            }
2141cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setApnSetting(null);
2142cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setDataConnectionAc(null);
21433fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            if (isOnlySingleDcAllowed(mPhone.getServiceState().getRilDataRadioTechnology())) {
2144449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                if(DBG) log("onDisconnectDone: isOnlySigneDcAllowed true so setup single apn");
21453fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                setupDataOnConnectableApns(Phone.REASON_SINGLE_PDN_ARBITRATION);
2146449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville            } else {
2147449cbf85380bd1d6716b7ab002da1efb9493bb02Wink Saville                if(DBG) log("onDisconnectDone: not retrying");
21483fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt            }
2149c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2150a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2151a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (mDisconnectPendingCount > 0)
2152a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            mDisconnectPendingCount--;
2153a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2154a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (mDisconnectPendingCount == 0) {
2155c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt            apnContext.setConcurrentVoiceAndDataAllowed(
2156c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                    mPhone.getServiceStateTracker().isConcurrentVoiceAndDataAllowed());
2157a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            notifyDataDisconnectComplete();
2158a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            notifyAllDataDisconnected();
2159a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2160a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2161c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2162c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2163ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    /**
2164ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     * Called when EVENT_DISCONNECT_DC_RETRYING is received.
2165ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville     */
2166ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    @Override
2167ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    protected void onDisconnectDcRetrying(int connId, AsyncResult ar) {
2168ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        // We could just do this in DC!!!
2169ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        ApnContext apnContext = null;
2170ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2171ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (ar.userObj instanceof ApnContext) {
2172ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            apnContext = (ApnContext) ar.userObj;
2173ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        } else {
2174ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            loge("onDisconnectDcRetrying: Invalid ar in onDisconnectDone, ignore");
2175ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            return;
2176ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        }
2177ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2178ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        apnContext.setState(DctConstants.State.RETRYING);
2179ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if(DBG) log("onDisconnectDcRetrying: apnContext=" + apnContext);
2180ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2181ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        mPhone.notifyDataConnection(apnContext.getReason(), apnContext.getApnType());
2182ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    }
2183ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2184c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2185cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2186cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onVoiceCallStarted() {
2187cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onVoiceCallStarted");
2188ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville        mInVoiceCall = true;
2189cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (isConnected() && ! mPhone.getServiceStateTracker().isConcurrentVoiceAndDataAllowed()) {
2190cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("onVoiceCallStarted stop polling");
2191cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            stopNetStatPoll();
2192cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            stopDataStallAlarm();
2193cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            notifyDataConnection(Phone.REASON_VOICE_CALL_STARTED);
2194c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2195c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2196c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2197cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2198cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onVoiceCallEnded() {
2199cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onVoiceCallEnded");
2200ec107cb8a36181fa26e61ab51f423ba862deb0faWink Saville        mInVoiceCall = false;
2201cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (isConnected()) {
2202cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!mPhone.getServiceStateTracker().isConcurrentVoiceAndDataAllowed()) {
2203cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                startNetStatPoll();
2204cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                startDataStallAlarm(DATA_STALL_NOT_SUSPECTED);
2205cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                notifyDataConnection(Phone.REASON_VOICE_CALL_ENDED);
2206cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
2207cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // clean slate after call end.
2208cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                resetPollStats();
2209c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2210c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2211bda761320929f714951c328bfec6a51a1978db97Wink Saville        // reset reconnect timer
2212bda761320929f714951c328bfec6a51a1978db97Wink Saville        setupDataOnConnectableApns(Phone.REASON_VOICE_CALL_ENDED);
2213c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2214c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2215cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2216cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onCleanUpConnection(boolean tearDown, int apnId, String reason) {
2217cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("onCleanUpConnection");
2218cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ApnContext apnContext = mApnContexts.get(apnIdToType(apnId));
2219cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (apnContext != null) {
2220cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            apnContext.setReason(reason);
2221cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cleanUpConnection(tearDown, apnContext);
2222c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2223c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2224c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2225cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2226cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected boolean isConnected() {
2227cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
2228ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (apnContext.getState() == DctConstants.State.CONNECTED) {
2229cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // At least one context is connected, return true
2230cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return true;
2231c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2232c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2233cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // There are not any contexts connected, return false
2234cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return false;
2235c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2236c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2237cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2238cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public boolean isDisconnected() {
2239cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
2240cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (!apnContext.isDisconnected()) {
2241cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // At least one context was not disconnected return false
2242cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return false;
2243cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2244c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2245cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // All contexts were disconnected so return true
2246cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return true;
2247c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2248c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2249cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2250cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void notifyDataConnection(String reason) {
2251cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("notifyDataConnection: reason=" + reason);
2252cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (ApnContext apnContext : mApnContexts.values()) {
2253187a39f896f88eb6c5e4306d9595546654825976Wink Saville            if (mAttached.get() && apnContext.isReady()) {
2254187a39f896f88eb6c5e4306d9595546654825976Wink Saville                if (DBG) log("notifyDataConnection: type:" + apnContext.getApnType());
2255cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPhone.notifyDataConnection(reason != null ? reason : apnContext.getReason(),
2256cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        apnContext.getApnType());
2257cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2258c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2259cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        notifyOffApnsOfAvailability(reason);
2260c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2261c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2262c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
2263cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Based on the sim operator numeric, create a list for all possible
2264cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Data Connections and setup the preferredApn.
2265c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
2266cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void createAllApnList() {
2267ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        mAllApnSettings = new ArrayList<ApnSetting>();
2268cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
2269cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        String operator = (r != null) ? r.getOperatorNumeric() : "";
2270cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (operator != null) {
2271cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            String selection = "numeric = '" + operator + "'";
227260bc489803a3557526e1f95e34c237e70f28bc50Sungmin Choi            String orderBy = "_id";
2273cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // query only enabled apn.
2274cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            // carrier_enabled : 1 means enabled apn, 0 disabled apn.
22759d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan            // selection += " and carrier_enabled = 1";
2276cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("createAllApnList: selection=" + selection);
2277cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2278cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            Cursor cursor = mPhone.getContext().getContentResolver().query(
227960bc489803a3557526e1f95e34c237e70f28bc50Sungmin Choi                    Telephony.Carriers.CONTENT_URI, null, selection, null, orderBy);
2280cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2281cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (cursor != null) {
2282cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (cursor.getCount() > 0) {
2283ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    mAllApnSettings = createApnList(cursor);
2284cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
2285cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                cursor.close();
2286cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2287c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2288c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
228976f43316a5a6082d601bffd4b6898d0bd81e11fcram        addEmergencyApnSetting();
229076f43316a5a6082d601bffd4b6898d0bd81e11fcram
229129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        dedupeApnSettings();
229229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
2293ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (mAllApnSettings.isEmpty()) {
2294cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("createAllApnList: No APN found for carrier: " + operator);
2295cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mPreferredApn = null;
2296ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            // TODO: What is the right behavior?
2297cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            //notifyNoData(DataConnection.FailCause.MISSING_UNKNOWN_APN);
2298c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
2299cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mPreferredApn = getPreferredApn();
2300cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (mPreferredApn != null && !mPreferredApn.numeric.equals(operator)) {
2301cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPreferredApn = null;
2302cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                setPreferredApn(-1);
2303cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2304cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("createAllApnList: mPreferredApn=" + mPreferredApn);
2305c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2306ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (DBG) log("createAllApnList: X mAllApnSettings=" + mAllApnSettings);
23079d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan
23089d4ec7d45061f1da05f16cd244eb0a798e7f36bbAmit Mahajan        setDataProfilesAsNeeded();
2309c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2310c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
231129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    private void dedupeApnSettings() {
231229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        ArrayList<ApnSetting> resultApns = new ArrayList<ApnSetting>();
231329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
231429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        // coalesce APNs if they are similar enough to prevent
231529c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        // us from bringing up two data calls with the same interface
231629c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        int i = 0;
231729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        while (i < mAllApnSettings.size() - 1) {
231829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            ApnSetting first = mAllApnSettings.get(i);
231929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            ApnSetting second = null;
232029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            int j = i + 1;
232129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            while (j < mAllApnSettings.size()) {
232229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                second = mAllApnSettings.get(j);
232329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                if (apnsSimilar(first, second)) {
232429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                    ApnSetting newApn = mergeApns(first, second);
232529c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                    mAllApnSettings.set(i, newApn);
232629c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                    first = newApn;
232729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                    mAllApnSettings.remove(j);
232829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                } else {
232929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                    j++;
233029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                }
233129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            }
233229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            i++;
233329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        }
233429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    }
233529c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
2336d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe    //check whether the types of two APN same (even only one type of each APN is same)
2337d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe    private boolean apnTypeSameAny(ApnSetting first, ApnSetting second) {
2338d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        if(VDBG) {
2339d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            StringBuilder apnType1 = new StringBuilder(first.apn + ": ");
2340d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            for(int index1 = 0; index1 < first.types.length; index1++) {
2341d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                apnType1.append(first.types[index1]);
2342d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                apnType1.append(",");
2343d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            }
2344d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe
2345d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            StringBuilder apnType2 = new StringBuilder(second.apn + ": ");
2346d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            for(int index1 = 0; index1 < second.types.length; index1++) {
2347d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                apnType2.append(second.types[index1]);
2348d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                apnType2.append(",");
2349d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            }
2350d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            log("APN1: is " + apnType1);
2351d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            log("APN2: is " + apnType2);
2352d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        }
2353d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe
2354d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        for(int index1 = 0; index1 < first.types.length; index1++) {
2355d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            for(int index2 = 0; index2 < second.types.length; index2++) {
2356d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                if(first.types[index1].equals(PhoneConstants.APN_TYPE_ALL) ||
2357d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                        second.types[index2].equals(PhoneConstants.APN_TYPE_ALL) ||
2358d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                        first.types[index1].equals(second.types[index2])) {
2359d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                    if(VDBG)log("apnTypeSameAny: return true");
2360d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                    return true;
2361d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                }
2362d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe            }
2363d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        }
2364d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe
2365d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        if(VDBG)log("apnTypeSameAny: return false");
2366d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe        return false;
2367d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe    }
2368d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe
236929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    // Check if neither mention DUN and are substantially similar
237029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    private boolean apnsSimilar(ApnSetting first, ApnSetting second) {
237129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        return (first.canHandleType(PhoneConstants.APN_TYPE_DUN) == false &&
237229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                second.canHandleType(PhoneConstants.APN_TYPE_DUN) == false &&
237329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                Objects.equals(first.apn, second.apn) &&
2374d86df358b8fe07160caa12147b6e4ad34d378ce6xinhe                !apnTypeSameAny(first, second) &&
237561cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                xorEquals(first.proxy, second.proxy) &&
237661cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                xorEquals(first.port, second.port) &&
237729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                first.carrierEnabled == second.carrierEnabled &&
2378aaf0e8d623b1e94004557573aef9235177ca19e3Amit Mahajan                first.bearerBitmask == second.bearerBitmask &&
237929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                first.profileId == second.profileId &&
238029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                Objects.equals(first.mvnoType, second.mvnoType) &&
238129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                Objects.equals(first.mvnoMatchData, second.mvnoMatchData) &&
238229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                xorEquals(first.mmsc, second.mmsc) &&
238329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                xorEquals(first.mmsProxy, second.mmsProxy) &&
238429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                xorEquals(first.mmsPort, second.mmsPort));
238529c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    }
238629c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
238729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    // equal or one is not specified
238829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    private boolean xorEquals(String first, String second) {
238929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        return (Objects.equals(first, second) ||
239029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                TextUtils.isEmpty(first) ||
239129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                TextUtils.isEmpty(second));
239229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    }
239329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
239429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    private ApnSetting mergeApns(ApnSetting dest, ApnSetting src) {
2395bca51fc3a191d3ca30df627b75374db0941571c5Sungmin Choi        int id = dest.id;
239629c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        ArrayList<String> resultTypes = new ArrayList<String>();
239729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        resultTypes.addAll(Arrays.asList(dest.types));
239829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        for (String srcType : src.types) {
239929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt            if (resultTypes.contains(srcType) == false) resultTypes.add(srcType);
2400bca51fc3a191d3ca30df627b75374db0941571c5Sungmin Choi            if (srcType.equals(PhoneConstants.APN_TYPE_DEFAULT)) id = src.id;
240129c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        }
240229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        String mmsc = (TextUtils.isEmpty(dest.mmsc) ? src.mmsc : dest.mmsc);
240329c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        String mmsProxy = (TextUtils.isEmpty(dest.mmsProxy) ? src.mmsProxy : dest.mmsProxy);
240429c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt        String mmsPort = (TextUtils.isEmpty(dest.mmsPort) ? src.mmsPort : dest.mmsPort);
240561cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan        String proxy = (TextUtils.isEmpty(dest.proxy) ? src.proxy : dest.proxy);
240661cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan        String port = (TextUtils.isEmpty(dest.port) ? src.port : dest.port);
240761cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan        String protocol = src.protocol.equals("IPV4V6") ? src.protocol : dest.protocol;
240861cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan        String roamingProtocol = src.roamingProtocol.equals("IPV4V6") ? src.roamingProtocol :
240961cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                dest.roamingProtocol;
2410aaf0e8d623b1e94004557573aef9235177ca19e3Amit Mahajan        int bearerBitmask = (dest.bearerBitmask == 0 || src.bearerBitmask == 0) ?
2411aaf0e8d623b1e94004557573aef9235177ca19e3Amit Mahajan                0 : (dest.bearerBitmask | src.bearerBitmask);
241229c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
2413bca51fc3a191d3ca30df627b75374db0941571c5Sungmin Choi        return new ApnSetting(id, dest.numeric, dest.carrier, dest.apn,
241461cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                proxy, port, mmsc, mmsProxy, mmsPort, dest.user, dest.password,
241561cdbcf2de392d043de0e4a533fbc615fc423000Amit Mahajan                dest.authType, resultTypes.toArray(new String[0]), protocol,
2416aaf0e8d623b1e94004557573aef9235177ca19e3Amit Mahajan                roamingProtocol, dest.carrierEnabled, 0, bearerBitmask, dest.profileId,
241729c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                (dest.modemCognitive || src.modemCognitive), dest.maxConns, dest.waitTime,
241829c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt                dest.maxConnsTime, dest.mtu, dest.mvnoType, dest.mvnoMatchData);
241929c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt    }
242029c6659c8767212d23d417f2b7f032b6c0d82119Robert Greenwalt
2421ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville    /** Return the DC AsyncChannel for the new data connection */
2422454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville    private DcAsyncChannel createDataConnection() {
2423cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("createDataConnection E");
2424cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2425cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        int id = mUniqueIdGenerator.getAndIncrement();
2426ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        DataConnection conn = DataConnection.makeDataConnection(mPhone, id,
2427ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                                                this, mDcTesterFailBringUpAll, mDcc);
2428cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        mDataConnections.put(id, conn);
2429454b1dfd508844b42eb775e4ab2359be74d3672bWink Saville        DcAsyncChannel dcac = new DcAsyncChannel(conn, LOG_TAG);
2430cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        int status = dcac.fullyConnectSync(mPhone.getContext(), this, conn.getHandler());
2431cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (status == AsyncChannel.STATUS_SUCCESSFUL) {
2432ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            mDataConnectionAcHashMap.put(dcac.getDataConnectionIdSync(), dcac);
2433c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
2434ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            loge("createDataConnection: Could not connect to dcac=" + dcac + " status=" + status);
2435c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2436cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2437cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("createDataConnection() X id=" + id + " dc=" + conn);
2438ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        return dcac;
2439c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2440c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2441cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void destroyDataConnections() {
2442cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if(mDataConnections != null) {
2443cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("destroyDataConnections: clear mDataConnectionList");
2444cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            mDataConnections.clear();
2445cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
2446cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("destroyDataConnections: mDataConnecitonList is empty, ignore");
2447c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2448c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2449c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2450c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
2451cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * Build a list of APNs to be used to create PDP's.
2452c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *
2453cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @param requestedApnType
2454cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     * @return waitingApns list to be used to create PDP
2455cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville     *          error when waitingApns.isEmpty()
2456c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
2457203e588e3c42a81aa8a56f595119c181a63b12caWink Saville    private ArrayList<ApnSetting> buildWaitingApns(String requestedApnType, int radioTech) {
2458cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("buildWaitingApns: E requestedApnType=" + requestedApnType);
2459cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ArrayList<ApnSetting> apnList = new ArrayList<ApnSetting>();
2460cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2461cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (requestedApnType.equals(PhoneConstants.APN_TYPE_DUN)) {
2462cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            ApnSetting dun = fetchDunApn();
2463cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (dun != null) {
2464cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                apnList.add(dun);
2465cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("buildWaitingApns: X added APN_TYPE_DUN apnList=" + apnList);
2466cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                return apnList;
2467c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2468c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2469c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2470cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
2471cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        String operator = (r != null) ? r.getOperatorNumeric() : "";
2472c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2473cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // This is a workaround for a bug (7305641) where we don't failover to other
2474cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // suitable APNs if our preferred APN fails.  On prepaid ATT sims we need to
2475cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // failover to a provisioning APN, but once we've used their default data
2476cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // connection we are locked to it for life.  This change allows ATT devices
2477cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // to say they don't want to use preferred at all.
2478cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        boolean usePreferred = true;
2479cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        try {
2480cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            usePreferred = ! mPhone.getContext().getResources().getBoolean(com.android.
2481cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    internal.R.bool.config_dontPreferApn);
2482cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } catch (Resources.NotFoundException e) {
2483cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) log("buildWaitingApns: usePreferred NotFoundException set to true");
2484cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            usePreferred = true;
2485cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2486bf660a4647db8151609cdfa0ecc4c96e1518947fSungmin Choi        if (usePreferred) {
2487bf660a4647db8151609cdfa0ecc4c96e1518947fSungmin Choi            mPreferredApn = getPreferredApn();
2488bf660a4647db8151609cdfa0ecc4c96e1518947fSungmin Choi        }
2489cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) {
2490cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("buildWaitingApns: usePreferred=" + usePreferred
249122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                    + " canSetPreferApn=" + mCanSetPreferApn
2492cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    + " mPreferredApn=" + mPreferredApn
2493cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    + " operator=" + operator + " radioTech=" + radioTech
2494cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    + " IccRecords r=" + r);
2495cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2496c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
249722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (usePreferred && mCanSetPreferApn && mPreferredApn != null &&
2498cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPreferredApn.canHandleType(requestedApnType)) {
2499cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (DBG) {
2500cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("buildWaitingApns: Preferred APN:" + operator + ":"
2501cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        + mPreferredApn.numeric + ":" + mPreferredApn);
2502cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            }
2503cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (mPreferredApn.numeric.equals(operator)) {
2504aaf0e8d623b1e94004557573aef9235177ca19e3Amit Mahajan                if (ServiceState.bitmaskHasTech(mPreferredApn.bearerBitmask, radioTech)) {
2505cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    apnList.add(mPreferredApn);
2506cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (DBG) log("buildWaitingApns: X added preferred apnList=" + apnList);
2507cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return apnList;
2508cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
2509cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    if (DBG) log("buildWaitingApns: no preferred APN");
2510cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    setPreferredApn(-1);
2511cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    mPreferredApn = null;
2512c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
2513cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else {
2514cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("buildWaitingApns: no preferred APN");
2515cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                setPreferredApn(-1);
2516cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mPreferredApn = null;
2517c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2518c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2519ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        if (mAllApnSettings != null) {
2520ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            if (DBG) log("buildWaitingApns: mAllApnSettings=" + mAllApnSettings);
2521ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            for (ApnSetting apn : mAllApnSettings) {
2522cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (apn.canHandleType(requestedApnType)) {
2523aaf0e8d623b1e94004557573aef9235177ca19e3Amit Mahajan                    if (ServiceState.bitmaskHasTech(apn.bearerBitmask, radioTech)) {
25249232dafa7ea833fc0b3a6024d6c7e23fc8e961eaRobert Greenwalt                        if (DBG) log("buildWaitingApns: adding apn=" + apn);
2525cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        apnList.add(apn);
2526c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    } else {
2527cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        if (DBG) {
2528aaf0e8d623b1e94004557573aef9235177ca19e3Amit Mahajan                            log("buildWaitingApns: bearerBitmask:" + apn.bearerBitmask + " does " +
2529aaf0e8d623b1e94004557573aef9235177ca19e3Amit Mahajan                                    "not include radioTech:" + radioTech);
2530cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        }
2531c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    }
25329232dafa7ea833fc0b3a6024d6c7e23fc8e961eaRobert Greenwalt                } else if (DBG) {
2533cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("buildWaitingApns: couldn't handle requesedApnType="
2534cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                            + requestedApnType);
2535c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
2536c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
2537cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
25384bd0ae43eb0cb9969dee4f30cddc18a71da68190Jack Yu            loge("mAllApnSettings is null!");
2539c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2540cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("buildWaitingApns: X apnList=" + apnList);
2541cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return apnList;
2542c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2543c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2544cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private String apnListToString (ArrayList<ApnSetting> apns) {
2545cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        StringBuilder result = new StringBuilder();
2546cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        for (int i = 0, size = apns.size(); i < size; i++) {
2547cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            result.append('[')
2548cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                  .append(apns.get(i).toString())
2549cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                  .append(']');
2550c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
2551cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return result.toString();
2552c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2553c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
2554cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private void setPreferredApn(int pos) {
255522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (!mCanSetPreferApn) {
2556cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("setPreferredApn: X !canSEtPreferApn");
2557cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
2558cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2559cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
25606bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        String subId = Long.toString(mPhone.getSubId());
25616bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        Uri uri = Uri.withAppendedPath(PREFERAPN_NO_UPDATE_URI_USING_SUBID, subId);
2562cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        log("setPreferredApn: delete");
2563cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        ContentResolver resolver = mPhone.getContext().getContentResolver();
25646bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        resolver.delete(uri, null, null);
2565cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2566cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (pos >= 0) {
2567cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("setPreferredApn: insert");
2568cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            ContentValues values = new ContentValues();
2569cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            values.put(APN_ID, pos);
25706bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville            resolver.insert(uri, values);
2571cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2572cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
2573cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2574cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private ApnSetting getPreferredApn() {
25754bd0ae43eb0cb9969dee4f30cddc18a71da68190Jack Yu        if (mAllApnSettings == null || mAllApnSettings.isEmpty()) {
25764bd0ae43eb0cb9969dee4f30cddc18a71da68190Jack Yu            log("getPreferredApn: mAllApnSettings is " + ((mAllApnSettings == null)?"null":"empty"));
2577cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return null;
2578cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2579cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
25806bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        String subId = Long.toString(mPhone.getSubId());
25816bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        Uri uri = Uri.withAppendedPath(PREFERAPN_NO_UPDATE_URI_USING_SUBID, subId);
2582cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        Cursor cursor = mPhone.getContext().getContentResolver().query(
25836bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville                uri, new String[] { "_id", "name", "apn" },
2584cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                null, null, Telephony.Carriers.DEFAULT_SORT_ORDER);
2585cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2586cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (cursor != null) {
258722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mCanSetPreferApn = true;
2588cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
258922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mCanSetPreferApn = false;
2590cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2591cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        log("getPreferredApn: mRequestedApnType=" + mRequestedApnType + " cursor=" + cursor
2592cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                + " cursor.count=" + ((cursor != null) ? cursor.getCount() : 0));
2593cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
259422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mCanSetPreferApn && cursor.getCount() > 0) {
2595cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            int pos;
2596cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cursor.moveToFirst();
2597cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            pos = cursor.getInt(cursor.getColumnIndexOrThrow(Telephony.Carriers._ID));
2598ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville            for(ApnSetting p : mAllApnSettings) {
2599cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("getPreferredApn: apnSetting=" + p);
2600cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (p.id == pos && p.canHandleType(mRequestedApnType)) {
2601cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    log("getPreferredApn: X found apnSetting" + p);
2602cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    cursor.close();
2603cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    return p;
2604cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
2605cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla            }
2606cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        }
2607cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2608cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (cursor != null) {
2609cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            cursor.close();
2610cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2611cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2612cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        log("getPreferredApn: X not found");
2613cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return null;
2614cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2615cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2616cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2617cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    public void handleMessage (Message msg) {
2618cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (DBG) log("handleMessage msg=" + msg);
2619cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2620cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (!mPhone.mIsTheCurrentActivePhone || mIsDisposed) {
2621cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            loge("handleMessage: Ignore GSM msgs since GSM phone is inactive");
2622cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return;
2623cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2624cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2625cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        switch (msg.what) {
2626cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_RECORDS_LOADED:
2627bda761320929f714951c328bfec6a51a1978db97Wink Saville                onRecordsLoaded();
2628cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2629cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2630cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_DATA_CONNECTION_DETACHED:
2631cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                onDataConnectionDetached();
2632cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2633cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2634cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_DATA_CONNECTION_ATTACHED:
2635bda761320929f714951c328bfec6a51a1978db97Wink Saville                onDataConnectionAttached();
2636cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2637cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2638cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_DO_RECOVERY:
2639cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                doRecovery();
2640cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2641cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2642cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_APN_CHANGED:
2643cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                onApnChanged();
2644cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2645cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2646cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_PS_RESTRICT_ENABLED:
2647cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                /**
2648cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * We don't need to explicitly to tear down the PDP context
2649cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * when PS restricted is enabled. The base band will deactive
2650cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * PDP context and notify us with PDP_CONTEXT_CHANGED.
2651cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * But we should stop the network polling and prevent reset PDP.
2652cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 */
2653cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("EVENT_PS_RESTRICT_ENABLED " + mIsPsRestricted);
2654cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                stopNetStatPoll();
2655cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                stopDataStallAlarm();
2656cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mIsPsRestricted = true;
2657cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla                break;
2658cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2659cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_PS_RESTRICT_DISABLED:
2660cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                /**
2661cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * When PS restrict is removed, we need setup PDP connection if
2662cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 * PDP connection is down.
2663cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                 */
2664cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("EVENT_PS_RESTRICT_DISABLED " + mIsPsRestricted);
2665cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mIsPsRestricted  = false;
2666cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (isConnected()) {
2667cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    startNetStatPoll();
2668cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    startDataStallAlarm(DATA_STALL_NOT_SUSPECTED);
2669cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
2670cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    // TODO: Should all PDN states be checked to fail?
2671ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    if (mState == DctConstants.State.FAILED) {
2672cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        cleanUpAllConnections(false, Phone.REASON_PS_RESTRICT_ENABLED);
2673cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        mReregisterOnReconnectFailure = false;
2674cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    }
26753fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    ApnContext apnContext = mApnContexts.get(PhoneConstants.APN_TYPE_DEFAULT);
26763fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    if (apnContext != null) {
26773fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        apnContext.setReason(Phone.REASON_PS_RESTRICT_ENABLED);
26783fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        trySetupData(apnContext);
26793fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    } else {
26803fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        loge("**** Default ApnContext not found ****");
26813fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        if (Build.IS_DEBUGGABLE) {
26823fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                            throw new RuntimeException("Default ApnContext not found");
26833fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                        }
26843fad7daba62dcb7aafc4adc7f8cc123726ed5a7cRobert Greenwalt                    }
2685cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
2686cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla                break;
2687ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2688cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_TRY_SETUP_DATA:
2689cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (msg.obj instanceof ApnContext) {
2690cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    onTrySetupData((ApnContext)msg.obj);
2691cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else if (msg.obj instanceof String) {
2692cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    onTrySetupData((String)msg.obj);
2693cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
2694cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    loge("EVENT_TRY_SETUP request w/o apnContext or String");
2695cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
2696cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla                break;
2697cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
2698cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            case DctConstants.EVENT_CLEAN_UP_CONNECTION:
2699cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                boolean tearDown = (msg.arg1 == 0) ? false : true;
2700cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (DBG) log("EVENT_CLEAN_UP_CONNECTION tearDown=" + tearDown);
2701cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                if (msg.obj instanceof ApnContext) {
2702cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                    cleanUpConnection(tearDown, (ApnContext)msg.obj);
2703cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                } else {
2704ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    loge("EVENT_CLEAN_UP_CONNECTION request w/o apn context, call super");
2705ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville                    super.handleMessage(msg);
2706cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                }
2707cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla                break;
2708a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            case DctConstants.EVENT_SET_INTERNAL_DATA_ENABLE:
2709a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                boolean enabled = (msg.arg1 == DctConstants.ENABLED) ? true : false;
2710a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                onSetInternalDataEnabled(enabled, (Message) msg.obj);
2711a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                break;
2712a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2713a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            case DctConstants.EVENT_CLEAN_UP_ALL_CONNECTIONS:
2714a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                Message mCause = obtainMessage(DctConstants.EVENT_CLEAN_UP_ALL_CONNECTIONS, null);
2715a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                if ((msg.obj != null) && (msg.obj instanceof String)) {
2716a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                    mCause.obj = msg.obj;
2717a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                }
2718a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                super.handleMessage(mCause);
2719a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                break;
2720ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville
2721220d43e0611edd0f3a2eb7a33bb38a008fff0868Wink Saville            case DctConstants.EVENT_DATA_RAT_CHANGED:
2722220d43e0611edd0f3a2eb7a33bb38a008fff0868Wink Saville                //May new Network allow setupData, so try it here
2723c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                setupDataOnConnectableApns(Phone.REASON_NW_TYPE_CHANGED,
2724c2d1d6b2725b4611360d2725624a0d8905d75694Robert Greenwalt                        RetryFailures.ONLY_ON_CHANGE);
2725220d43e0611edd0f3a2eb7a33bb38a008fff0868Wink Saville                break;
2726220d43e0611edd0f3a2eb7a33bb38a008fff0868Wink Saville
27272b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen            case DctConstants.CMD_CLEAR_PROVISIONING_SPINNER:
27282b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                // Check message sender intended to clear the current spinner.
27292b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                if (mProvisioningSpinner == msg.obj) {
27302b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    mProvisioningSpinner.dismiss();
27312b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                    mProvisioningSpinner = null;
27322b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                }
27332b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen                break;
27342b7b6016c7a5f2c3ce9c7e623ea10a9fe9239dc2Paul Jensen
2735cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla            default:
2736cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                // handle the message in the super class DataConnectionTracker
2737cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                super.handleMessage(msg);
2738cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                break;
2739cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2740cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2741cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2742cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected int getApnProfileID(String apnType) {
2743cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_IMS)) {
2744cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return RILConstants.DATA_PROFILE_IMS;
2745cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_FOTA)) {
2746cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return RILConstants.DATA_PROFILE_FOTA;
2747cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_CBS)) {
2748cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return RILConstants.DATA_PROFILE_CBS;
27491b5fe200e47f40f82f0e28502a5f40bce64a82e6Wink Saville        } else if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_IA)) {
27501b5fe200e47f40f82f0e28502a5f40bce64a82e6Wink Saville            return RILConstants.DATA_PROFILE_DEFAULT; // DEFAULT for now
275145df26444864daad60afdd4d121ab4043da3834bSungmin Choi        } else if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_DUN)) {
275245df26444864daad60afdd4d121ab4043da3834bSungmin Choi            return RILConstants.DATA_PROFILE_TETHERED;
2753cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        } else {
2754cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            return RILConstants.DATA_PROFILE_DEFAULT;
2755cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2756cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    }
2757cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2758cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private int getCellLocationId() {
2759cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        int cid = -1;
2760cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        CellLocation loc = mPhone.getCellLocation();
2761cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2762cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (loc != null) {
2763cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (loc instanceof GsmCellLocation) {
2764cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                cid = ((GsmCellLocation)loc).getCid();
2765cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            } else if (loc instanceof CdmaCellLocation) {
2766cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                cid = ((CdmaCellLocation)loc).getBaseStationId();
2767cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla            }
2768cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2769cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        return cid;
2770cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2771cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2772a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    private IccRecords getUiccRecords(int appFamily) {
2773a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return mUiccController.getIccRecords(mPhone.getPhoneId(), appFamily);
2774a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2775a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2776a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2777cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2778cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void onUpdateIcc() {
2779cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (mUiccController == null ) {
2780cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla            return;
2781cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla        }
2782cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2783a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        IccRecords newIccRecords = getUiccRecords(UiccController.APP_FAM_3GPP);
2784cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2785cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        IccRecords r = mIccRecords.get();
2786cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (r != newIccRecords) {
2787cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (r != null) {
2788cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("Removing stale icc objects.");
2789cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                r.unregisterForRecordsLoaded(this);
2790cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mIccRecords.set(null);
27919aaa228cbb37657d12a87b5058676e449866f2acJeevaka Badrappan            }
2792cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (newIccRecords != null) {
2793cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                log("New records found");
2794cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                mIccRecords.set(newIccRecords);
2795cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                newIccRecords.registerForRecordsLoaded(
2796cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville                        this, DctConstants.EVENT_RECORDS_LOADED, null);
27970469925aace7a2254e0cef90bd7398fbd72622faShishir Agrawal            } else {
27980469925aace7a2254e0cef90bd7398fbd72622faShishir Agrawal                onSimNotReady();
27999aaa228cbb37657d12a87b5058676e449866f2acJeevaka Badrappan            }
28009aaa228cbb37657d12a87b5058676e449866f2acJeevaka Badrappan        }
2801cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2802cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2803a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void update() {
2804a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("update sub = " + mPhone.getSubId());
2805bda761320929f714951c328bfec6a51a1978db97Wink Saville        log("update(): Active DDS, register for all events now!");
2806bda761320929f714951c328bfec6a51a1978db97Wink Saville        registerForAllEvents();
2807bda761320929f714951c328bfec6a51a1978db97Wink Saville        onUpdateIcc();
2808a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
28092bb2331aa5c5285f70a7404d61ee71ede4831056Shishir Agrawal        mUserDataEnabled = getDataEnabled();
2810bda761320929f714951c328bfec6a51a1978db97Wink Saville
2811bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (mPhone instanceof CDMALTEPhone) {
2812bda761320929f714951c328bfec6a51a1978db97Wink Saville            ((CDMALTEPhone)mPhone).updateCurrentCarrierInProvider();
2813bda761320929f714951c328bfec6a51a1978db97Wink Saville        } else if (mPhone instanceof GSMPhone) {
2814bda761320929f714951c328bfec6a51a1978db97Wink Saville            ((GSMPhone)mPhone).updateCurrentCarrierInProvider();
2815a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        } else {
2816bda761320929f714951c328bfec6a51a1978db97Wink Saville            log("Phone object is not MultiSim. This should not hit!!!!");
2817a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2818a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2819a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2820a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
2821a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void cleanUpAllConnections(String cause) {
2822a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        cleanUpAllConnections(cause, null);
2823a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2824a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2825a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void updateRecords() {
2826bda761320929f714951c328bfec6a51a1978db97Wink Saville        onUpdateIcc();
2827a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2828a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2829a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void cleanUpAllConnections(String cause, Message disconnectAllCompleteMsg) {
2830a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("cleanUpAllConnections");
2831a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (disconnectAllCompleteMsg != null) {
2832a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            mDisconnectAllCompleteMsgList.add(disconnectAllCompleteMsg);
2833a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2834a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2835a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        Message msg = obtainMessage(DctConstants.EVENT_CLEAN_UP_ALL_CONNECTIONS);
2836a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        msg.obj = cause;
2837a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        sendMessage(msg);
2838a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2839a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2840a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void notifyDataDisconnectComplete() {
2841a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("notifyDataDisconnectComplete");
2842a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        for (Message m: mDisconnectAllCompleteMsgList) {
2843a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            m.sendToTarget();
2844a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2845a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mDisconnectAllCompleteMsgList.clear();
2846a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2847a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2848a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2849a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void notifyAllDataDisconnected() {
2850a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        sEnableFailFastRefCounter = 0;
2851a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mFailFast = false;
2852a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mAllDataDisconnectedRegistrants.notifyRegistrants();
2853a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2854a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2855a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void registerForAllDataDisconnected(Handler h, int what, Object obj) {
2856a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mAllDataDisconnectedRegistrants.addUnique(h, what, obj);
2857a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2858a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (isDisconnected()) {
2859a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            log("notify All Data Disconnected");
2860a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            notifyAllDataDisconnected();
2861a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2862a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2863a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2864a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void unregisterForAllDataDisconnected(Handler h) {
2865a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        mAllDataDisconnectedRegistrants.remove(h);
2866a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2867a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2868a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2869a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
2870a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void onSetInternalDataEnabled(boolean enable) {
2871bda761320929f714951c328bfec6a51a1978db97Wink Saville        if (DBG) log("onSetInternalDataEnabled: enabled=" + enable);
2872a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        onSetInternalDataEnabled(enable, null);
2873a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2874a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2875a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    protected void onSetInternalDataEnabled(boolean enabled, Message onCompleteMsg) {
28766bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (DBG) log("onSetInternalDataEnabled: enabled=" + enabled);
2877a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        boolean sendOnComplete = true;
2878a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2879a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        synchronized (mDataEnabledLock) {
2880a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            mInternalDataEnabled = enabled;
2881a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            if (enabled) {
2882a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                log("onSetInternalDataEnabled: changed to enabled, try to setup data call");
2883a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                onTrySetupData(Phone.REASON_DATA_ENABLED);
2884a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            } else {
2885a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                sendOnComplete = false;
2886a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                log("onSetInternalDataEnabled: changed to disabled, cleanUpAllConnections");
2887a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                cleanUpAllConnections(null, onCompleteMsg);
2888a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            }
2889a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2890a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2891a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (sendOnComplete) {
2892a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            if (onCompleteMsg != null) {
2893a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                onCompleteMsg.sendToTarget();
2894a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            }
2895a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2896a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2897a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2898a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean setInternalDataEnabledFlag(boolean enable) {
28996bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (DBG) log("setInternalDataEnabledFlag(" + enable + ")");
2900a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2901a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (mInternalDataEnabled != enable) {
2902a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            mInternalDataEnabled = enable;
2903a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2904a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return true;
2905a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2906a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2907a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
2908a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean setInternalDataEnabled(boolean enable) {
2909a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return setInternalDataEnabled(enable, null);
2910a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2911a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2912a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean setInternalDataEnabled(boolean enable, Message onCompleteMsg) {
29136bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville        if (DBG) log("setInternalDataEnabled(" + enable + ")");
2914a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2915a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        Message msg = obtainMessage(DctConstants.EVENT_SET_INTERNAL_DATA_ENABLE, onCompleteMsg);
2916a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        msg.arg1 = (enable ? DctConstants.ENABLED : DctConstants.DISABLED);
2917a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        sendMessage(msg);
2918a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return true;
2919a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2920a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2921a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void setDataAllowed(boolean enable, Message response) {
29226bc4098827f3070a44b5e51508b455d7c7be9c07Wink Saville         if (DBG) log("setDataAllowed: enable=" + enable);
2923a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         mIsCleanupRequired = !enable;
2924a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         mPhone.mCi.setDataAllowed(enable, response);
2925a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville         mInternalDataEnabled = enable;
2926a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2927a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2928cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2929cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void log(String s) {
2930a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        Rlog.d(LOG_TAG, "[" + mPhone.getPhoneId() + "]" + s);
2931cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2932cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2933cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2934cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    protected void loge(String s) {
2935a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        Rlog.e(LOG_TAG, "[" + mPhone.getPhoneId() + "]" + s);
2936cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla    }
2937cebb2cc576c652dd642d7f419532ec04e0f59d7dNaveen Kalla
2938cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
2939c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
2940bda761320929f714951c328bfec6a51a1978db97Wink Saville        pw.println("DcTracker extends:");
2941cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        super.dump(fd, pw, args);
2942cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        pw.println(" mReregisterOnReconnectFailure=" + mReregisterOnReconnectFailure);
294322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        pw.println(" canSetPreferApn=" + mCanSetPreferApn);
2944cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        pw.println(" mApnObserver=" + mApnObserver);
2945cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        pw.println(" getOverallState=" + getOverallState());
2946ff4e317d24f0d23bdc0f306d53ddc51f2f1ecf6aWink Saville        pw.println(" mDataConnectionAsyncChannels=%s\n" + mDataConnectionAcHashMap);
2947187a39f896f88eb6c5e4306d9595546654825976Wink Saville        pw.println(" mAttached=" + mAttached.get());
2948c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
2949a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2950a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
2951bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram    public String[] getPcscfAddress(String apnType) {
2952a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("getPcscfAddress()");
2953bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        ApnContext apnContext = null;
2954bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram
2955bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        if(apnType == null){
2956bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            log("apnType is null, return null");
2957bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            return null;
2958bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        }
2959a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2960bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_EMERGENCY)) {
2961bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            apnContext = mApnContexts.get(PhoneConstants.APN_TYPE_EMERGENCY);
2962bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        } else if (TextUtils.equals(apnType, PhoneConstants.APN_TYPE_IMS)) {
2963bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            apnContext = mApnContexts.get(PhoneConstants.APN_TYPE_IMS);
2964bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        } else {
2965bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            log("apnType is invalid, return null");
2966bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram            return null;
2967bc78e2f9988f380a3b88d4cb4a9c0b80b8f44beeram        }
2968a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2969a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (apnContext == null) {
2970a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            log("apnContext is null, return null");
2971a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            return null;
2972a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2973a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2974a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        DcAsyncChannel dcac = apnContext.getDcAc();
2975a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        String[] result = null;
2976a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2977a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (dcac != null) {
2978a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            result = dcac.getPcscfAddr();
2979a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2980a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            for (int i = 0; i < result.length; i++) {
2981a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                log("Pcscf[" + i + "]: " + result[i]);
2982a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            }
2983a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville            return result;
2984a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        }
2985a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return null;
2986a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
2987a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2988a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
2989a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public void setImsRegistrationState(boolean registered) {
2990a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        log("setImsRegistrationState - mImsRegistrationState(before): "+ mImsRegistrationState
2991a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville                + ", registered(current) : " + registered);
2992a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2993a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (mPhone == null) return;
2994a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2995a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        ServiceStateTracker sst = mPhone.getServiceStateTracker();
2996a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        if (sst == null) return;
2997a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
2998a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        sst.setImsRegistrationState(registered);
2999a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
300076f43316a5a6082d601bffd4b6898d0bd81e11fcram
300176f43316a5a6082d601bffd4b6898d0bd81e11fcram    /**
300276f43316a5a6082d601bffd4b6898d0bd81e11fcram     * Read APN configuration from Telephony.db for Emergency APN
300376f43316a5a6082d601bffd4b6898d0bd81e11fcram     * All opertors recognize the connection request for EPDN based on APN type
300476f43316a5a6082d601bffd4b6898d0bd81e11fcram     * PLMN name,APN name are not mandatory parameters
300576f43316a5a6082d601bffd4b6898d0bd81e11fcram     */
300676f43316a5a6082d601bffd4b6898d0bd81e11fcram    private void initEmergencyApnSetting() {
300776f43316a5a6082d601bffd4b6898d0bd81e11fcram        // Operator Numeric is not available when sim records are not loaded.
300876f43316a5a6082d601bffd4b6898d0bd81e11fcram        // Query Telephony.db with APN type as EPDN request does not
300976f43316a5a6082d601bffd4b6898d0bd81e11fcram        // require APN name, plmn and all operators support same APN config.
301076f43316a5a6082d601bffd4b6898d0bd81e11fcram        // DB will contain only one entry for Emergency APN
301176f43316a5a6082d601bffd4b6898d0bd81e11fcram        String selection = "type=\"emergency\"";
301276f43316a5a6082d601bffd4b6898d0bd81e11fcram        Cursor cursor = mPhone.getContext().getContentResolver().query(
301376f43316a5a6082d601bffd4b6898d0bd81e11fcram                Telephony.Carriers.CONTENT_URI, null, selection, null, null);
301476f43316a5a6082d601bffd4b6898d0bd81e11fcram
301576f43316a5a6082d601bffd4b6898d0bd81e11fcram        if (cursor != null) {
301676f43316a5a6082d601bffd4b6898d0bd81e11fcram            if (cursor.getCount() > 0) {
301776f43316a5a6082d601bffd4b6898d0bd81e11fcram                if (cursor.moveToFirst()) {
301876f43316a5a6082d601bffd4b6898d0bd81e11fcram                    mEmergencyApn = makeApnSetting(cursor);
301976f43316a5a6082d601bffd4b6898d0bd81e11fcram                }
302076f43316a5a6082d601bffd4b6898d0bd81e11fcram            }
302176f43316a5a6082d601bffd4b6898d0bd81e11fcram            cursor.close();
302276f43316a5a6082d601bffd4b6898d0bd81e11fcram        }
302376f43316a5a6082d601bffd4b6898d0bd81e11fcram    }
302476f43316a5a6082d601bffd4b6898d0bd81e11fcram
302576f43316a5a6082d601bffd4b6898d0bd81e11fcram    /**
302676f43316a5a6082d601bffd4b6898d0bd81e11fcram     * Add the Emergency APN settings to APN settings list
302776f43316a5a6082d601bffd4b6898d0bd81e11fcram     */
302876f43316a5a6082d601bffd4b6898d0bd81e11fcram    private void addEmergencyApnSetting() {
302976f43316a5a6082d601bffd4b6898d0bd81e11fcram        if(mEmergencyApn != null) {
303076f43316a5a6082d601bffd4b6898d0bd81e11fcram            if(mAllApnSettings == null) {
303176f43316a5a6082d601bffd4b6898d0bd81e11fcram                mAllApnSettings = new ArrayList<ApnSetting>();
303276f43316a5a6082d601bffd4b6898d0bd81e11fcram            } else {
303376f43316a5a6082d601bffd4b6898d0bd81e11fcram                boolean hasEmergencyApn = false;
303476f43316a5a6082d601bffd4b6898d0bd81e11fcram                for (ApnSetting apn : mAllApnSettings) {
303576f43316a5a6082d601bffd4b6898d0bd81e11fcram                    if (ArrayUtils.contains(apn.types, PhoneConstants.APN_TYPE_EMERGENCY)) {
303676f43316a5a6082d601bffd4b6898d0bd81e11fcram                        hasEmergencyApn = true;
303776f43316a5a6082d601bffd4b6898d0bd81e11fcram                        break;
303876f43316a5a6082d601bffd4b6898d0bd81e11fcram                    }
303976f43316a5a6082d601bffd4b6898d0bd81e11fcram                }
304076f43316a5a6082d601bffd4b6898d0bd81e11fcram
304176f43316a5a6082d601bffd4b6898d0bd81e11fcram                if(hasEmergencyApn == false) {
304276f43316a5a6082d601bffd4b6898d0bd81e11fcram                    mAllApnSettings.add(mEmergencyApn);
304376f43316a5a6082d601bffd4b6898d0bd81e11fcram                } else {
304476f43316a5a6082d601bffd4b6898d0bd81e11fcram                    log("addEmergencyApnSetting - E-APN setting is already present");
304576f43316a5a6082d601bffd4b6898d0bd81e11fcram                }
304676f43316a5a6082d601bffd4b6898d0bd81e11fcram            }
304776f43316a5a6082d601bffd4b6898d0bd81e11fcram        }
304876f43316a5a6082d601bffd4b6898d0bd81e11fcram    }
30499a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang
30509a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang    private void cleanUpConnectionsOnUpdatedApns(boolean tearDown) {
30519a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        if (DBG) log("cleanUpConnectionsOnUpdatedApns: tearDown=" + tearDown);
30529a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        if (mAllApnSettings.isEmpty()) {
30539a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang            cleanUpAllConnections(tearDown, Phone.REASON_APN_CHANGED);
30549a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        } else {
30559a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang            for (ApnContext apnContext : mApnContexts.values()) {
30569a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                if (VDBG) log("cleanUpConnectionsOnUpdatedApns for "+ apnContext);
30579a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang
30589a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                boolean cleanUpApn = true;
30599a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                ArrayList<ApnSetting> currentWaitingApns = apnContext.getWaitingApns();
30609a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang
30619a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                if ((currentWaitingApns != null) && (!apnContext.isDisconnected())) {
30629a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                    int radioTech = mPhone.getServiceState().getRilDataRadioTechnology();
30639a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                    ArrayList<ApnSetting> waitingApns = buildWaitingApns(
30649a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                            apnContext.getApnType(), radioTech);
30659a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                    if (VDBG) log("new waitingApns:" + waitingApns);
30669a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                    if (waitingApns.size() == currentWaitingApns.size()) {
30679a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                        cleanUpApn = false;
30689a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                        for (int i = 0; i < waitingApns.size(); i++) {
30699a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                            if (!currentWaitingApns.get(i).equals(waitingApns.get(i))) {
30709a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                                if (VDBG) log("new waiting apn is different at " + i);
30719a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                                cleanUpApn = true;
30729a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                                apnContext.setWaitingApns(waitingApns);
30739a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                                break;
30749a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                            }
30759a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                        }
30769a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                    }
30779a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                }
30789a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang
30799a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                if (cleanUpApn) {
30809a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                    apnContext.setReason(Phone.REASON_APN_CHANGED);
30819a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                    cleanUpConnection(true, apnContext);
30829a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang                }
30839a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang            }
30849a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        }
30859a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang
30869a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        if (!isConnected()) {
30879a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang            stopNetStatPoll();
30889a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang            stopDataStallAlarm();
30899a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        }
30909a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang
30919a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        mRequestedApnType = PhoneConstants.APN_TYPE_DEFAULT;
30929a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang
30939a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        if (DBG) log("mDisconnectPendingCount = " + mDisconnectPendingCount);
30949a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        if (tearDown && mDisconnectPendingCount == 0) {
30959a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang            notifyDataDisconnectComplete();
30969a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang            notifyAllDataDisconnected();
30979a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang        }
30989a90f1dde8fa4bd7b1606742909c2f2f28a28a2fHui Wang    }
3099c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville}
3100