GsmCdmaConnection.java revision 33cfb500a7cf192adfc5ca06792fe3847073b6c1
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
17c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savillepackage com.android.internal.telephony.cdma;
18c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
19c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport com.android.internal.telephony.*;
20c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.content.Context;
21c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.AsyncResult;
22c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.Handler;
23c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.Looper;
24c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.Message;
25c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.PowerManager;
26c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.Registrant;
27c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.SystemClock;
28b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensenimport android.telephony.DisconnectCause;
29ded9c0af7fa49504c047275ed34c2d3b22bf0c3aWink Savilleimport android.telephony.Rlog;
30c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.text.TextUtils;
31c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
32c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.telephony.PhoneNumberUtils;
33c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.telephony.ServiceState;
34e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka
35a914a2429cb352f1f294e15dc236b3721e81a823Alex Yakavenkaimport com.android.internal.telephony.uicc.UiccCardApplication;
36e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenkaimport com.android.internal.telephony.uicc.UiccController;
37d720945f2be5ea5fe0faf67e67d9ea0e184eba67Alex Yakavenkaimport com.android.internal.telephony.uicc.IccCardApplicationStatus.AppState;
38c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
39c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville/**
40c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * {@hide}
41c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville */
42c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savillepublic class CdmaConnection extends Connection {
43cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    static final String LOG_TAG = "CdmaConnection";
44cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    private static final boolean VDBG = false;
45c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
46c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    //***** Instance Variables
47c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
4822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville    CdmaCallTracker mOwner;
4922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville    CdmaCall mParent;
50c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
51c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
5222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville    String mPostDialString;      // outgoing calls only
5322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville    boolean mDisconnected;
5422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville    int mIndex;          // index in CdmaCallTracker.connections[], -1 if unassigned
55c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
56c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*
57c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * These time/timespan values are based on System.currentTimeMillis(),
58c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * i.e., "wall clock" time.
59c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
6022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville    long mDisconnectTime;
61c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
6222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville    int mNextPostDialChar;       // index into postDialString
63c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
64b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen    int mCause = DisconnectCause.NOT_DISCONNECTED;
6522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville    PostDialState mPostDialState = PostDialState.NOT_STARTED;
660742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela    int mPreciseCause = 0;
6733cfb500a7cf192adfc5ca06792fe3847073b6c1Chao Liu    String mVendorCause;
68c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
6922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville    Handler mHandler;
70c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
71c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private PowerManager.WakeLock mPartialWakeLock;
72c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
73c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    //***** Event Constants
74c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int EVENT_DTMF_DONE = 1;
75c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int EVENT_PAUSE_DONE = 2;
76c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int EVENT_NEXT_POST_DIAL = 3;
77c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int EVENT_WAKE_LOCK_TIMEOUT = 4;
78c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
79c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    //***** Constants
80c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int WAKE_LOCK_TIMEOUT_MILLIS = 60*1000;
81c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int PAUSE_DELAY_MILLIS = 2 * 1000;
82c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
83c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    //***** Inner Classes
84c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
85c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    class MyHandler extends Handler {
86c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        MyHandler(Looper l) {super(l);}
87c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
88cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        @Override
89c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        public void
90c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        handleMessage(Message msg) {
91c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
92c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            switch (msg.what) {
93c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                case EVENT_NEXT_POST_DIAL:
94c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                case EVENT_DTMF_DONE:
95c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                case EVENT_PAUSE_DONE:
96c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    processNextPostDialChar();
97c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    break;
98c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                case EVENT_WAKE_LOCK_TIMEOUT:
99c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    releaseWakeLock();
100c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    break;
101c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
102c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
103c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
104c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
105c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    //***** Constructors
106c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
107c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** This is probably an MT call that we first saw in a CLCC response */
108c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/
109c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    CdmaConnection (Context context, DriverCall dc, CdmaCallTracker ct, int index) {
110c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        createWakeLock(context);
111c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        acquireWakeLock();
112c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
11322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mOwner = ct;
11422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mHandler = new MyHandler(mOwner.getLooper());
115c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
11622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mAddress = dc.number;
117c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
11822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mIsIncoming = dc.isMT;
11922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCreateTime = System.currentTimeMillis();
12022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCnapName = dc.name;
12122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCnapNamePresentation = dc.namePresentation;
12222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mNumberPresentation = dc.numberPresentation;
123c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
12422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mIndex = index;
125c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
12622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mParent = parentFromDCState (dc.state);
12722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mParent.attach(this, dc);
128c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
129c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
130c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** This is an MO call/three way call, created when dialing */
131c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/
132c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    CdmaConnection(Context context, String dialString, CdmaCallTracker ct, CdmaCall parent) {
133c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        createWakeLock(context);
134c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        acquireWakeLock();
135c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
13622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mOwner = ct;
13722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mHandler = new MyHandler(mOwner.getLooper());
138c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
13922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mDialString = dialString;
140936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen        Rlog.d(LOG_TAG, "[CDMAConn] CdmaConnection: dialString=" + maskDialString(dialString));
141c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        dialString = formatDialString(dialString);
142936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen        Rlog.d(LOG_TAG,
143936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen                "[CDMAConn] CdmaConnection:formated dialString=" + maskDialString(dialString));
144c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
14522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mAddress = PhoneNumberUtils.extractNetworkPortionAlt(dialString);
14622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mPostDialString = PhoneNumberUtils.extractPostDialPortion(dialString);
147c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
14822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mIndex = -1;
149c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
15022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mIsIncoming = false;
15122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCnapName = null;
15222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCnapNamePresentation = PhoneConstants.PRESENTATION_ALLOWED;
15322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mNumberPresentation = PhoneConstants.PRESENTATION_ALLOWED;
15422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCreateTime = System.currentTimeMillis();
155c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
156c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (parent != null) {
15722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mParent = parent;
158c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
159c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            //for the three way call case, not change parent state
16022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            if (parent.mState == CdmaCall.State.ACTIVE) {
161c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                parent.attachFake(this, CdmaCall.State.ACTIVE);
162c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            } else {
163c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                parent.attachFake(this, CdmaCall.State.DIALING);
164c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
165c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
166c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
167c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
168c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** This is a Call waiting call*/
169c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    CdmaConnection(Context context, CdmaCallWaitingNotification cw, CdmaCallTracker ct,
170c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            CdmaCall parent) {
171c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        createWakeLock(context);
172c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        acquireWakeLock();
173c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
17422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mOwner = ct;
17522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mHandler = new MyHandler(mOwner.getLooper());
17622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mAddress = cw.number;
17722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mNumberPresentation = cw.numberPresentation;
17822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCnapName = cw.name;
17922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCnapNamePresentation = cw.namePresentation;
18022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mIndex = -1;
18122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mIsIncoming = true;
18222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCreateTime = System.currentTimeMillis();
18322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mConnectTime = 0;
18422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mParent = parent;
185c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        parent.attachFake(this, CdmaCall.State.WAITING);
186c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
187c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
188c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void dispose() {
189c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
190c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
191c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static boolean
192c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    equalsHandlesNulls (Object a, Object b) {
193c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return (a == null) ? (b == null) : a.equals (b);
194c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
195c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
196c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ boolean
197c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    compareTo(DriverCall c) {
198c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // On mobile originated (MO) calls, the phone number may have changed
199c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // due to a SIM Toolkit call control modification.
200c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        //
201c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // We assume we know when MO calls are created (since we created them)
202c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // and therefore don't need to compare the phone number anyway.
20322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (! (mIsIncoming || c.isMT)) return true;
204c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
205c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // ... but we can compare phone numbers on MT calls, and we have
206c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // no control over when they begin, so we might as well
207c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
208c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        String cAddress = PhoneNumberUtils.stringFromStringAndTOA(c.number, c.TOA);
20922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        return mIsIncoming == c.isMT && equalsHandlesNulls(mAddress, cAddress);
210c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
211c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
212c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
213cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
214c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public String getOrigDialString(){
21522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        return mDialString;
216c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
217c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
218cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
219c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public CdmaCall getCall() {
22022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        return mParent;
221c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
222c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
223cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
224c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public long getDisconnectTime() {
22522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        return mDisconnectTime;
226c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
227c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
228cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
229c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public long getHoldDurationMillis() {
230c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (getState() != CdmaCall.State.HOLDING) {
231c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // If not holding, return 0
232c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return 0;
233c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
23422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            return SystemClock.elapsedRealtime() - mHoldingStartTime;
235c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
236c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
237c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
238cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
239b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen    public int getDisconnectCause() {
24022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        return mCause;
241c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
242c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
243cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
244c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public CdmaCall.State getState() {
24522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mDisconnected) {
246c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return CdmaCall.State.DISCONNECTED;
247c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
248c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return super.getState();
249c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
250c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
251c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
252cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
253c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void hangup() throws CallStateException {
25422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (!mDisconnected) {
25522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mOwner.hangup(this);
256c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
257c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            throw new CallStateException ("disconnected");
258c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
259c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
260c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
261cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
262c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void separate() throws CallStateException {
26322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (!mDisconnected) {
26422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mOwner.separate(this);
265c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
266c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            throw new CallStateException ("disconnected");
267c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
268c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
269c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
270cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
271c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public PostDialState getPostDialState() {
27222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        return mPostDialState;
273c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
274c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
275cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
276c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void proceedAfterWaitChar() {
27722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mPostDialState != PostDialState.WAIT) {
278ded9c0af7fa49504c047275ed34c2d3b22bf0c3aWink Saville            Rlog.w(LOG_TAG, "CdmaConnection.proceedAfterWaitChar(): Expected "
27922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                + "getPostDialState() to be WAIT but was " + mPostDialState);
280c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return;
281c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
282c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
283c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        setPostDialState(PostDialState.STARTED);
284c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
285c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        processNextPostDialChar();
286c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
287c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
288cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
289c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void proceedAfterWildChar(String str) {
29022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mPostDialState != PostDialState.WILD) {
291ded9c0af7fa49504c047275ed34c2d3b22bf0c3aWink Saville            Rlog.w(LOG_TAG, "CdmaConnection.proceedAfterWaitChar(): Expected "
29222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                + "getPostDialState() to be WILD but was " + mPostDialState);
293c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return;
294c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
295c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
296c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        setPostDialState(PostDialState.STARTED);
297c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
298cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // make a new postDialString, with the wild char replacement string
299cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        // at the beginning, followed by the remaining postDialString.
300c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
301cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        StringBuilder buf = new StringBuilder(str);
30222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        buf.append(mPostDialString.substring(mNextPostDialChar));
30322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mPostDialString = buf.toString();
30422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mNextPostDialChar = 0;
305cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        if (Phone.DEBUG_PHONE) {
306cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            log("proceedAfterWildChar: new postDialString is " +
30722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                    mPostDialString);
308c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
309cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville
310cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville        processNextPostDialChar();
311c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
312c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
313cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
314c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void cancelPostDial() {
315c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        setPostDialState(PostDialState.CANCELLED);
316c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
317c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
318c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
319c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * Called when this Connection is being hung up locally (eg, user pressed "end")
320c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * Note that at this point, the hangup request has been dispatched to the radio
321c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * but no response has yet been received so update() has not yet been called
322c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
323c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    void
324c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    onHangupLocal() {
32522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCause = DisconnectCause.LOCAL;
3260742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela        mPreciseCause = 0;
32733cfb500a7cf192adfc5ca06792fe3847073b6c1Chao Liu        mVendorCause = null;
328c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
329c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
330b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen    /**
331b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen     * Maps RIL call disconnect code to {@link DisconnectCause}.
332b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen     * @param causeCode RIL disconnect code
333b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen     * @return the corresponding value from {@link DisconnectCause}
334b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen     */
335b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen    int disconnectCauseFromCode(int causeCode) {
336c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        /**
337c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * See 22.001 Annex F.4 for mapping of cause codes
338c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * to local tones
339c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         */
340c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
341c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        switch (causeCode) {
342c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.USER_BUSY:
343c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.BUSY;
344c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.NO_CIRCUIT_AVAIL:
345c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CONGESTION;
346c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.ACM_LIMIT_EXCEEDED:
347c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.LIMIT_EXCEEDED;
348c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CALL_BARRED:
349c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CALL_BARRED;
350c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.FDN_BLOCKED:
351c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.FDN_BLOCKED;
35209b65da3b46e18c8acc146d722047ba2fb161bfePreeti Ahuja            case CallFailCause.DIAL_MODIFIED_TO_USSD:
35309b65da3b46e18c8acc146d722047ba2fb161bfePreeti Ahuja                return DisconnectCause.DIAL_MODIFIED_TO_USSD;
35409b65da3b46e18c8acc146d722047ba2fb161bfePreeti Ahuja            case CallFailCause.DIAL_MODIFIED_TO_SS:
35509b65da3b46e18c8acc146d722047ba2fb161bfePreeti Ahuja                return DisconnectCause.DIAL_MODIFIED_TO_SS;
35609b65da3b46e18c8acc146d722047ba2fb161bfePreeti Ahuja            case CallFailCause.DIAL_MODIFIED_TO_DIAL:
35709b65da3b46e18c8acc146d722047ba2fb161bfePreeti Ahuja                return DisconnectCause.DIAL_MODIFIED_TO_DIAL;
358c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_LOCKED_UNTIL_POWER_CYCLE:
359c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_LOCKED_UNTIL_POWER_CYCLE;
360c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_DROP:
361c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_DROP;
362c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_INTERCEPT:
363c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_INTERCEPT;
364c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_REORDER:
365c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_REORDER;
366c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_SO_REJECT:
367c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_SO_REJECT;
368c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_RETRY_ORDER:
369c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_RETRY_ORDER;
370c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_ACCESS_FAILURE:
371c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_ACCESS_FAILURE;
372c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_PREEMPTED:
373c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_PREEMPTED;
374c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_NOT_EMERGENCY:
375c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_NOT_EMERGENCY;
376c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_ACCESS_BLOCKED:
377c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_ACCESS_BLOCKED;
378c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.ERROR_UNSPECIFIED:
379c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.NORMAL_CLEARING:
380c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            default:
38122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                CDMAPhone phone = mOwner.mPhone;
382c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                int serviceState = phone.getServiceState().getState();
383a914a2429cb352f1f294e15dc236b3721e81a823Alex Yakavenka                UiccCardApplication app = UiccController
384e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka                        .getInstance()
385062a2a3838c8d8adf16f4d9fbde8d52450da0336Shishir Agrawal                        .getUiccCardApplication(phone.getPhoneId(), UiccController.APP_FAM_3GPP2);
386a914a2429cb352f1f294e15dc236b3721e81a823Alex Yakavenka                AppState uiccAppState = (app != null) ? app.getState() : AppState.APPSTATE_UNKNOWN;
387c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                if (serviceState == ServiceState.STATE_POWER_OFF) {
388c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    return DisconnectCause.POWER_OFF;
389c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                } else if (serviceState == ServiceState.STATE_OUT_OF_SERVICE
390c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        || serviceState == ServiceState.STATE_EMERGENCY_ONLY) {
391c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    return DisconnectCause.OUT_OF_SERVICE;
392c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                } else if (phone.mCdmaSubscriptionSource ==
393e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka                        CdmaSubscriptionSourceManager.SUBSCRIPTION_FROM_RUIM
394e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka                        && uiccAppState != AppState.APPSTATE_READY) {
395c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    return DisconnectCause.ICC_ERROR;
396c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                } else if (causeCode==CallFailCause.NORMAL_CLEARING) {
397c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    return DisconnectCause.NORMAL;
398c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                } else {
399c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    return DisconnectCause.ERROR_UNSPECIFIED;
400c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
401c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
402c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
403c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
404c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ void
40533cfb500a7cf192adfc5ca06792fe3847073b6c1Chao Liu    onRemoteDisconnect(int causeCode, String vendorCause) {
4060742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela        this.mPreciseCause = causeCode;
40733cfb500a7cf192adfc5ca06792fe3847073b6c1Chao Liu        this.mVendorCause = vendorCause;
408c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        onDisconnect(disconnectCauseFromCode(causeCode));
409c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
410c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
411b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen    /**
412b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen     * Called when the radio indicates the connection has been disconnected.
413b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen     * @param cause call disconnect cause; values are defined in {@link DisconnectCause}
414b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen     */
4151220a4e283def0598468376cf112d3b904026fb8Danny Baumann    /*package*/ boolean
416b7b7a62112b79571adf74372c5f5366fd62d0031Anders Kristensen    onDisconnect(int cause) {
4171220a4e283def0598468376cf112d3b904026fb8Danny Baumann        boolean changed = false;
4181220a4e283def0598468376cf112d3b904026fb8Danny Baumann
41922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCause = cause;
420c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
42122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (!mDisconnected) {
422c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            doDisconnect();
423cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (VDBG) Rlog.d(LOG_TAG, "onDisconnect: cause=" + cause);
424c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
42522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mOwner.mPhone.notifyDisconnect(this);
426c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
42722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            if (mParent != null) {
4281220a4e283def0598468376cf112d3b904026fb8Danny Baumann                changed = mParent.connectionDisconnected(this);
429c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
430c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
431c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        releaseWakeLock();
4321220a4e283def0598468376cf112d3b904026fb8Danny Baumann        return changed;
433c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
434c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
435c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** Called when the call waiting connection has been hung up */
436c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ void
437c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    onLocalDisconnect() {
43822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (!mDisconnected) {
439c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            doDisconnect();
440cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville            if (VDBG) Rlog.d(LOG_TAG, "onLoalDisconnect" );
441c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
44222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            if (mParent != null) {
44322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                mParent.detach(this);
444c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
445c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
446c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        releaseWakeLock();
447c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
448c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
449c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // Returns true if state has changed, false if nothing changed
450c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ boolean
451c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    update (DriverCall dc) {
452c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        CdmaCall newParent;
453c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        boolean changed = false;
454c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        boolean wasConnectingInOrOut = isConnectingInOrOut();
455c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        boolean wasHolding = (getState() == CdmaCall.State.HOLDING);
456c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
457c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        newParent = parentFromDCState(dc.state);
458c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
45922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (Phone.DEBUG_PHONE) log("parent= " +mParent +", newParent= " + newParent);
460c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
4612d265969ca301ba3720458ea9da2ec8d6a18bb8dXia Ying        log(" mNumberConverted " + mNumberConverted);
4622d265969ca301ba3720458ea9da2ec8d6a18bb8dXia Ying        if (!equalsHandlesNulls(mAddress, dc.number) && (!mNumberConverted
4632d265969ca301ba3720458ea9da2ec8d6a18bb8dXia Ying                || !equalsHandlesNulls(mConvertedNumber, dc.number)))  {
464c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (Phone.DEBUG_PHONE) log("update: phone # changed!");
46522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mAddress = dc.number;
466c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            changed = true;
467c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
468c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
469c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // A null cnapName should be the same as ""
470c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (TextUtils.isEmpty(dc.name)) {
47122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            if (!TextUtils.isEmpty(mCnapName)) {
472c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                changed = true;
47322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                mCnapName = "";
474c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
47522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        } else if (!dc.name.equals(mCnapName)) {
476c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            changed = true;
47722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mCnapName = dc.name;
478c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
479c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
48022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (Phone.DEBUG_PHONE) log("--dssds----"+mCnapName);
48122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mCnapNamePresentation = dc.namePresentation;
48222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mNumberPresentation = dc.numberPresentation;
483c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
48422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (newParent != mParent) {
48522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            if (mParent != null) {
48622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                mParent.detach(this);
487c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
488c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            newParent.attach(this, dc);
48922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mParent = newParent;
490c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            changed = true;
491c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
492c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            boolean parentStateChange;
49322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            parentStateChange = mParent.update (this, dc);
494c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            changed = changed || parentStateChange;
495c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
496c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
497c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        /** Some state-transition events */
498c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
499c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (Phone.DEBUG_PHONE) log(
500c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                "Update, wasConnectingInOrOut=" + wasConnectingInOrOut +
501c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                ", wasHolding=" + wasHolding +
502c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                ", isConnectingInOrOut=" + isConnectingInOrOut() +
503c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                ", changed=" + changed);
504c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
505c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
506c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (wasConnectingInOrOut && !isConnectingInOrOut()) {
507c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            onConnectedInOrOut();
508c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
509c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
510c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (changed && !wasHolding && (getState() == CdmaCall.State.HOLDING)) {
511c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // We've transitioned into HOLDING
512c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            onStartedHolding();
513c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
514c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
515c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return changed;
516c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
517c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
518c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
519c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * Called when this Connection is in the foregroundCall
520c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * when a dial is initiated.
521c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * We know we're ACTIVE, and we know we're going to end up
522c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * HOLDING in the backgroundCall
523c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
524c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    void
525c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    fakeHoldBeforeDial() {
52622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mParent != null) {
52722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mParent.detach(this);
528c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
529c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
53022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mParent = mOwner.mBackgroundCall;
53122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mParent.attachFake(this, CdmaCall.State.HOLDING);
532c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
533c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        onStartedHolding();
534c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
535c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
536c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ int
537c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    getCDMAIndex() throws CallStateException {
53822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mIndex >= 0) {
53922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            return mIndex + 1;
540c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
541c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            throw new CallStateException ("CDMA connection index not assigned");
542c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
543c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
544c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
545c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
546c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * An incoming or outgoing call has connected
547c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
548c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    void
549c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    onConnectedInOrOut() {
55022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mConnectTime = System.currentTimeMillis();
55122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mConnectTimeReal = SystemClock.elapsedRealtime();
55222d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mDuration = 0;
553c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
554c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // bug #678474: incoming call interpreted as missed call, even though
555c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // it sounds like the user has picked up the call.
556c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (Phone.DEBUG_PHONE) {
55722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            log("onConnectedInOrOut: connectTime=" + mConnectTime);
558c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
559c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
56022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (!mIsIncoming) {
561c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // outgoing calls only
562c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            processNextPostDialChar();
563c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
564c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // Only release wake lock for incoming calls, for outgoing calls the wake lock
565c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // will be released after any pause-dial is completed
566c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            releaseWakeLock();
567c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
568c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
569c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
570c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void
571c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    doDisconnect() {
5720b9246d6254bed6f625fa9c551f7f9dcc33d4e38Evan Charlton        mIndex = -1;
5730b9246d6254bed6f625fa9c551f7f9dcc33d4e38Evan Charlton        mDisconnectTime = System.currentTimeMillis();
5740b9246d6254bed6f625fa9c551f7f9dcc33d4e38Evan Charlton        mDuration = SystemClock.elapsedRealtime() - mConnectTimeReal;
5750b9246d6254bed6f625fa9c551f7f9dcc33d4e38Evan Charlton        mDisconnected = true;
5760b9246d6254bed6f625fa9c551f7f9dcc33d4e38Evan Charlton        clearPostDialListeners();
577c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
578c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
5799b41acc443e068fa3c3e547e820f710c6e2297baUma Maheswari Ramalingam    /*package*/ void
580c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    onStartedHolding() {
58122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mHoldingStartTime = SystemClock.elapsedRealtime();
582c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
583c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
584c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * Performs the appropriate action for a post-dial char, but does not
585c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * notify application. returns false if the character is invalid and
586c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * should be ignored
587c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
588c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private boolean
589c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    processPostDialChar(char c) {
590c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (PhoneNumberUtils.is12Key(c)) {
59122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mOwner.mCi.sendDtmf(c, mHandler.obtainMessage(EVENT_DTMF_DONE));
592c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else if (c == PhoneNumberUtils.PAUSE) {
593c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            setPostDialState(PostDialState.PAUSE);
594c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
595c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // Upon occurrences of the separator, the UE shall
596c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // pause again for 2 seconds before sending any
597c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // further DTMF digits.
59822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mHandler.sendMessageDelayed(mHandler.obtainMessage(EVENT_PAUSE_DONE),
599c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                                            PAUSE_DELAY_MILLIS);
600c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else if (c == PhoneNumberUtils.WAIT) {
601c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            setPostDialState(PostDialState.WAIT);
602c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else if (c == PhoneNumberUtils.WILD) {
603c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            setPostDialState(PostDialState.WILD);
604c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
605c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return false;
606c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
607c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
608c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return true;
609c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
610c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
611cbaa45bbf2cab852b6c9c3a887e9f803d4e857eaWink Saville    @Override
612c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public String getRemainingPostDialString() {
61322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mPostDialState == PostDialState.CANCELLED
61422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                || mPostDialState == PostDialState.COMPLETE
61522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                || mPostDialString == null
61622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                || mPostDialString.length() <= mNextPostDialChar) {
617c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return "";
618c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
619c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
62022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        String subStr = mPostDialString.substring(mNextPostDialChar);
621c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (subStr != null) {
622c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            int wIndex = subStr.indexOf(PhoneNumberUtils.WAIT);
623c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            int pIndex = subStr.indexOf(PhoneNumberUtils.PAUSE);
624c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
625c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (wIndex > 0 && (wIndex < pIndex || pIndex <= 0)) {
626c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                subStr = subStr.substring(0, wIndex);
627c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            } else if (pIndex > 0) {
628c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                subStr = subStr.substring(0, pIndex);
629c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
630c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
631c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return subStr;
632c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
633c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
634c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void updateParent(CdmaCall oldParent, CdmaCall newParent){
635c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (newParent != oldParent) {
636c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (oldParent != null) {
637c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                oldParent.detach(this);
638c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
639c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            newParent.attachFake(this, CdmaCall.State.ACTIVE);
64022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mParent = newParent;
641c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
642c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
643c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
644c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    @Override
645c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    protected void finalize()
646c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    {
647c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        /**
648c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * It is understood that This finializer is not guaranteed
649c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * to be called and the release lock call is here just in
650c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * case there is some path that doesn't call onDisconnect
651c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * and or onConnectedInOrOut.
652c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         */
653c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (mPartialWakeLock.isHeld()) {
654ded9c0af7fa49504c047275ed34c2d3b22bf0c3aWink Saville            Rlog.e(LOG_TAG, "[CdmaConn] UNEXPECTED; mPartialWakeLock is held when finalizing.");
655c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
656c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        releaseWakeLock();
657c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
658c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
659c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    void processNextPostDialChar() {
660c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        char c = 0;
661c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        Registrant postDialHandler;
662c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
66322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mPostDialState == PostDialState.CANCELLED) {
664c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            releaseWakeLock();
665ded9c0af7fa49504c047275ed34c2d3b22bf0c3aWink Saville            //Rlog.v("CDMA", "##### processNextPostDialChar: postDialState == CANCELLED, bail");
666c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return;
667c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
668c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
66922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        if (mPostDialString == null ||
67022d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                mPostDialString.length() <= mNextPostDialChar) {
671c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            setPostDialState(PostDialState.COMPLETE);
672c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
673c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // We were holding a wake lock until pause-dial was complete, so give it up now
674c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            releaseWakeLock();
675c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
676c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // notifyMessage.arg1 is 0 on complete
677c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            c = 0;
678c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
679c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            boolean isValid;
680c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
681c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            setPostDialState(PostDialState.STARTED);
682c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
68322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            c = mPostDialString.charAt(mNextPostDialChar++);
684c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
685c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            isValid = processPostDialChar(c);
686c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
687c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (!isValid) {
688c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                // Will call processNextPostDialChar
68922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                mHandler.obtainMessage(EVENT_NEXT_POST_DIAL).sendToTarget();
690c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                // Don't notify application
691ded9c0af7fa49504c047275ed34c2d3b22bf0c3aWink Saville                Rlog.e("CDMA", "processNextPostDialChar: c=" + c + " isn't valid!");
692c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return;
693c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
694c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
695c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
6964567a0789e9966929c71af9a2c3866582c85c9e0Nancy Chen        notifyPostDialListenersNextChar(c);
6974567a0789e9966929c71af9a2c3866582c85c9e0Nancy Chen
6984567a0789e9966929c71af9a2c3866582c85c9e0Nancy Chen        // TODO: remove the following code since the handler no longer executes anything.
69922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        postDialHandler = mOwner.mPhone.mPostDialHandler;
700c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
701c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        Message notifyMessage;
702c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
703c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (postDialHandler != null &&
704c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                (notifyMessage = postDialHandler.messageForRegistrant()) != null) {
705c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // The AsyncResult.result is the Connection object
70622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            PostDialState state = mPostDialState;
707c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            AsyncResult ar = AsyncResult.forMessage(notifyMessage);
708c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            ar.result = this;
709c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            ar.userObj = state;
710c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
711c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // arg1 is the character that was/is being processed
712c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            notifyMessage.arg1 = c;
713c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
714c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            notifyMessage.sendToTarget();
715c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
716c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
717c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
718c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
719c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** "connecting" means "has never been ACTIVE" for both incoming
720c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *  and outgoing calls
721c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
722c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private boolean
723c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    isConnectingInOrOut() {
72422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        return mParent == null || mParent == mOwner.mRingingCall
72522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            || mParent.mState == CdmaCall.State.DIALING
72622d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            || mParent.mState == CdmaCall.State.ALERTING;
727c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
728c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
729c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private CdmaCall
730c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    parentFromDCState (DriverCall.State state) {
731c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        switch (state) {
732c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case ACTIVE:
733c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case DIALING:
734c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case ALERTING:
73522d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                return mOwner.mForegroundCall;
736c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            //break;
737c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
738c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case HOLDING:
73922d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                return mOwner.mBackgroundCall;
740c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            //break;
741c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
742c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case INCOMING:
743c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case WAITING:
74422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                return mOwner.mRingingCall;
745c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            //break;
746c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
747c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            default:
748c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                throw new RuntimeException("illegal call state: " + state);
749c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
750c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
751c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
752c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
753c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * Set post dial state and acquire wake lock while switching to "started" or "wait"
754c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * state, the wake lock will be released if state switches out of "started" or "wait"
755c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * state or after WAKE_LOCK_TIMEOUT_MILLIS.
756c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * @param s new PostDialState
757c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
758c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void setPostDialState(PostDialState s) {
759c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (s == PostDialState.STARTED ||
760c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                s == PostDialState.PAUSE) {
761c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            synchronized (mPartialWakeLock) {
762c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                if (mPartialWakeLock.isHeld()) {
76322d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                    mHandler.removeMessages(EVENT_WAKE_LOCK_TIMEOUT);
764c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                } else {
765c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    acquireWakeLock();
766c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
76722d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                Message msg = mHandler.obtainMessage(EVENT_WAKE_LOCK_TIMEOUT);
76822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville                mHandler.sendMessageDelayed(msg, WAKE_LOCK_TIMEOUT_MILLIS);
769c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
770c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
77122d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville            mHandler.removeMessages(EVENT_WAKE_LOCK_TIMEOUT);
772c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            releaseWakeLock();
773c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
77422d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        mPostDialState = s;
7750b9246d6254bed6f625fa9c551f7f9dcc33d4e38Evan Charlton        notifyPostDialListeners();
776c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
777c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
778c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void createWakeLock(Context context) {
779c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        PowerManager pm = (PowerManager)context.getSystemService(Context.POWER_SERVICE);
780c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        mPartialWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, LOG_TAG);
781c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
782c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
783c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void acquireWakeLock() {
784c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        log("acquireWakeLock");
785c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        mPartialWakeLock.acquire();
786c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
787c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
788c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void releaseWakeLock() {
789c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        synchronized (mPartialWakeLock) {
790c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (mPartialWakeLock.isHeld()) {
791c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                log("releaseWakeLock");
792c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                mPartialWakeLock.release();
793c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
794c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
795c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
796c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
797c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private static boolean isPause(char c) {
798c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return c == PhoneNumberUtils.PAUSE;
799c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
800c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
801c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private static boolean isWait(char c) {
802c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return c == PhoneNumberUtils.WAIT;
803c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
804c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
805c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // This function is to find the next PAUSE character index if
806c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // multiple pauses in a row. Otherwise it finds the next non PAUSE or
807c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // non WAIT character index.
808c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private static int
809c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    findNextPCharOrNonPOrNonWCharIndex(String phoneNumber, int currIndex) {
810c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        boolean wMatched = isWait(phoneNumber.charAt(currIndex));
811c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        int index = currIndex + 1;
812c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        int length = phoneNumber.length();
813c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        while (index < length) {
814c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            char cNext = phoneNumber.charAt(index);
815c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // if there is any W inside P/W sequence,mark it
816c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (isWait(cNext)) {
817c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                wMatched = true;
818c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
819c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // if any characters other than P/W chars after P/W sequence
820c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // we break out the loop and append the correct
821c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (!isWait(cNext) && !isPause(cNext)) {
822c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                break;
823c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
824c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            index++;
825c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
826c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
827c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // It means the PAUSE character(s) is in the middle of dial string
828c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // and it needs to be handled one by one.
829c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if ((index < length) && (index > (currIndex + 1))  &&
830c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            ((wMatched == false) && isPause(phoneNumber.charAt(currIndex)))) {
831c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return (currIndex + 1);
832c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
833c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return index;
834c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
835c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
836c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // This function returns either PAUSE or WAIT character to append.
837c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // It is based on the next non PAUSE/WAIT character in the phoneNumber and the
838c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // index for the current PAUSE/WAIT character
839c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private static char
840c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    findPOrWCharToAppend(String phoneNumber, int currPwIndex, int nextNonPwCharIndex) {
841c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        char c = phoneNumber.charAt(currPwIndex);
842c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        char ret;
843c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
844c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // Append the PW char
845c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        ret = (isPause(c)) ? PhoneNumberUtils.PAUSE : PhoneNumberUtils.WAIT;
846c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
847c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // If the nextNonPwCharIndex is greater than currPwIndex + 1,
848c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // it means the PW sequence contains not only P characters.
849c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // Since for the sequence that only contains P character,
850c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // the P character is handled one by one, the nextNonPwCharIndex
851c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // equals to currPwIndex + 1.
852c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // In this case, skip P, append W.
853c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (nextNonPwCharIndex > (currPwIndex + 1)) {
854c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            ret = PhoneNumberUtils.WAIT;
855c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
856c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return ret;
857c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
858c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
859c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
860c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * format original dial string
861c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * 1) convert international dialing prefix "+" to
862c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *    string specified per region
863c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *
864c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * 2) handle corner cases for PAUSE/WAIT dialing:
865c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *
866c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *    If PAUSE/WAIT sequence at the end, ignore them.
867c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *
868c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *    If consecutive PAUSE/WAIT sequence in the middle of the string,
869c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *    and if there is any WAIT in PAUSE/WAIT sequence, treat them like WAIT.
870c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
871c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public static String formatDialString(String phoneNumber) {
872c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        /**
873c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * TODO(cleanup): This function should move to PhoneNumberUtils, and
874c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * tests should be added.
875c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         */
876c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
877c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (phoneNumber == null) {
878c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return null;
879c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
880c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        int length = phoneNumber.length();
881c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        StringBuilder ret = new StringBuilder();
882c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        char c;
883c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        int currIndex = 0;
884c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
885c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        while (currIndex < length) {
886c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            c = phoneNumber.charAt(currIndex);
887c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (isPause(c) || isWait(c)) {
888c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                if (currIndex < length - 1) {
889c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    // if PW not at the end
890c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    int nextIndex = findNextPCharOrNonPOrNonWCharIndex(phoneNumber, currIndex);
891c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    // If there is non PW char following PW sequence
892c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    if (nextIndex < length) {
893c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        char pC = findPOrWCharToAppend(phoneNumber, currIndex, nextIndex);
894c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        ret.append(pC);
895c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        // If PW char sequence has more than 2 PW characters,
896c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        // skip to the last PW character since the sequence already be
897c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        // converted to WAIT character
898c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        if (nextIndex > (currIndex + 1)) {
899c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                            currIndex = nextIndex - 1;
900c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        }
901c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    } else if (nextIndex == length) {
902c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        // It means PW characters at the end, ignore
903c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        currIndex = length - 1;
904c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    }
905c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
906c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            } else {
907c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                ret.append(c);
908c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
909c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            currIndex++;
910c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
911c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return PhoneNumberUtils.cdmaCheckAndProcessPlusCode(ret.toString());
912c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
913c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
914c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void log(String msg) {
915ded9c0af7fa49504c047275ed34c2d3b22bf0c3aWink Saville        Rlog.d(LOG_TAG, "[CDMAConn] " + msg);
916c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
917c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
918936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen    private String maskDialString(String dialString) {
919936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen        if (VDBG) {
920936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen            return dialString;
921936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen        }
922936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen
923936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen        return "<MASKED>";
924936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen    }
925936cf96aeb696feeec28638c08acbb53d5cb725bEtan Cohen
926c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    @Override
927c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public int getNumberPresentation() {
92822d85a8e3a575a6d01d2c788587971657dfe20c6Wink Saville        return mNumberPresentation;
929c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
930c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
931c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    @Override
932c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public UUSInfo getUUSInfo() {
933c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // UUS information not supported in CDMA
934c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return null;
935c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
9360742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela
9370742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela    public int getPreciseDisconnectCause() {
9380742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela        return mPreciseCause;
9390742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela    }
9400742246233c3f03f864d23e52d89e1845d994701Antonio Marín Cerezuela
941a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
94233cfb500a7cf192adfc5ca06792fe3847073b6c1Chao Liu    public String getVendorDisconnectCause() {
94333cfb500a7cf192adfc5ca06792fe3847073b6c1Chao Liu        return mVendorCause;
94433cfb500a7cf192adfc5ca06792fe3847073b6c1Chao Liu    }
94533cfb500a7cf192adfc5ca06792fe3847073b6c1Chao Liu
94633cfb500a7cf192adfc5ca06792fe3847073b6c1Chao Liu    @Override
947a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public Connection getOrigConnection() {
948a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return null;
949a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
950a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville
951a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    @Override
952a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    public boolean isMultiparty() {
953a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville        return false;
954a8467dd0c524787104b1ccdddc5e8af10ba729edWink Saville    }
955c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville}
956