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;
28c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.os.SystemProperties;
29c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.util.Log;
30c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.text.TextUtils;
31c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
32c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.telephony.PhoneNumberUtils;
33c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport android.telephony.ServiceState;
34e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka
35e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenkaimport com.android.internal.telephony.IccCardApplicationStatus.AppState;
36c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport com.android.internal.telephony.TelephonyProperties;
37c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savilleimport com.android.internal.telephony.RILConstants;
38e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenkaimport com.android.internal.telephony.uicc.UiccController;
39c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
40c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville/**
41c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville * {@hide}
42c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville */
43c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Savillepublic class CdmaConnection extends Connection {
44c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final String LOG_TAG = "CDMA";
45c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
46c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    //***** Instance Variables
47c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
48c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    CdmaCallTracker owner;
49c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    CdmaCall parent;
50c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
51c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
52c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    String address;             // MAY BE NULL!!!
53c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    String dialString;          // outgoing calls only
54c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    String postDialString;      // outgoing calls only
55c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    boolean isIncoming;
56c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    boolean disconnected;
57c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    int index;          // index in CdmaCallTracker.connections[], -1 if unassigned
58c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
59c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*
60c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * These time/timespan values are based on System.currentTimeMillis(),
61c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * i.e., "wall clock" time.
62c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
63c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    long createTime;
64c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    long connectTime;
65c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    long disconnectTime;
66c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
67c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*
68c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * These time/timespan values are based on SystemClock.elapsedRealTime(),
69c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * i.e., time since boot.  They are appropriate for comparison and
70c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * calculating deltas.
71c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
72c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    long connectTimeReal;
73c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    long duration;
74c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    long holdingStartTime;  // The time when the Connection last transitioned
75c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                            // into HOLDING
76c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
77c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    int nextPostDialChar;       // index into postDialString
78c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
79c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    DisconnectCause cause = DisconnectCause.NOT_DISCONNECTED;
80c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    PostDialState postDialState = PostDialState.NOT_STARTED;
81c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    int numberPresentation = PhoneConstants.PRESENTATION_ALLOWED;
82c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
83c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
84c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    Handler h;
85c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
86c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private PowerManager.WakeLock mPartialWakeLock;
87c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
88c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    //***** Event Constants
89c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int EVENT_DTMF_DONE = 1;
90c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int EVENT_PAUSE_DONE = 2;
91c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int EVENT_NEXT_POST_DIAL = 3;
92c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int EVENT_WAKE_LOCK_TIMEOUT = 4;
93c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
94c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    //***** Constants
95c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int WAKE_LOCK_TIMEOUT_MILLIS = 60*1000;
96c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static final int PAUSE_DELAY_MILLIS = 2 * 1000;
97c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
98c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    //***** Inner Classes
99c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
100c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    class MyHandler extends Handler {
101c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        MyHandler(Looper l) {super(l);}
102c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
103c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        public void
104c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        handleMessage(Message msg) {
105c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
106c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            switch (msg.what) {
107c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                case EVENT_NEXT_POST_DIAL:
108c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                case EVENT_DTMF_DONE:
109c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                case EVENT_PAUSE_DONE:
110c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    processNextPostDialChar();
111c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    break;
112c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                case EVENT_WAKE_LOCK_TIMEOUT:
113c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    releaseWakeLock();
114c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    break;
115c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
116c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
117c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
118c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
119c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    //***** Constructors
120c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
121c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** This is probably an MT call that we first saw in a CLCC response */
122c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/
123c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    CdmaConnection (Context context, DriverCall dc, CdmaCallTracker ct, int index) {
124c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        createWakeLock(context);
125c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        acquireWakeLock();
126c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
127c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        owner = ct;
128c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        h = new MyHandler(owner.getLooper());
129c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
130c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        address = dc.number;
131c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
132c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        isIncoming = dc.isMT;
133c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        createTime = System.currentTimeMillis();
134c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        cnapName = dc.name;
135c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        cnapNamePresentation = dc.namePresentation;
136c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        numberPresentation = dc.numberPresentation;
137c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
138c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        this.index = index;
139c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
140c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        parent = parentFromDCState (dc.state);
141c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        parent.attach(this, dc);
142c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
143c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
144c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** This is an MO call/three way call, created when dialing */
145c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/
146c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    CdmaConnection(Context context, String dialString, CdmaCallTracker ct, CdmaCall parent) {
147c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        createWakeLock(context);
148c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        acquireWakeLock();
149c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
150c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        owner = ct;
151c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        h = new MyHandler(owner.getLooper());
152c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
153c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        this.dialString = dialString;
154c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        Log.d(LOG_TAG, "[CDMAConn] CdmaConnection: dialString=" + dialString);
155c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        dialString = formatDialString(dialString);
156c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        Log.d(LOG_TAG, "[CDMAConn] CdmaConnection:formated dialString=" + dialString);
157c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
158c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        this.address = PhoneNumberUtils.extractNetworkPortionAlt(dialString);
159c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        this.postDialString = PhoneNumberUtils.extractPostDialPortion(dialString);
160c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
161c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        index = -1;
162c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
163c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        isIncoming = false;
164c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        cnapName = null;
165c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        cnapNamePresentation = PhoneConstants.PRESENTATION_ALLOWED;
166c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        numberPresentation = PhoneConstants.PRESENTATION_ALLOWED;
167c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        createTime = System.currentTimeMillis();
168c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
169c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (parent != null) {
170c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            this.parent = parent;
171c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
172c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            //for the three way call case, not change parent state
173c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (parent.state == CdmaCall.State.ACTIVE) {
174c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                parent.attachFake(this, CdmaCall.State.ACTIVE);
175c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            } else {
176c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                parent.attachFake(this, CdmaCall.State.DIALING);
177c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
178c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
179c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
180c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
181c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** This is a Call waiting call*/
182c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    CdmaConnection(Context context, CdmaCallWaitingNotification cw, CdmaCallTracker ct,
183c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            CdmaCall parent) {
184c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        createWakeLock(context);
185c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        acquireWakeLock();
186c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
187c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        owner = ct;
188c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        h = new MyHandler(owner.getLooper());
189c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        address = cw.number;
190c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        numberPresentation = cw.numberPresentation;
191c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        cnapName = cw.name;
192c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        cnapNamePresentation = cw.namePresentation;
193c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        index = -1;
194c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        isIncoming = true;
195c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        createTime = System.currentTimeMillis();
196c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        connectTime = 0;
197c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        this.parent = parent;
198c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        parent.attachFake(this, CdmaCall.State.WAITING);
199c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
200c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
201c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void dispose() {
202c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
203c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
204c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    static boolean
205c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    equalsHandlesNulls (Object a, Object b) {
206c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return (a == null) ? (b == null) : a.equals (b);
207c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
208c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
209c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ boolean
210c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    compareTo(DriverCall c) {
211c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // On mobile originated (MO) calls, the phone number may have changed
212c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // due to a SIM Toolkit call control modification.
213c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        //
214c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // We assume we know when MO calls are created (since we created them)
215c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // and therefore don't need to compare the phone number anyway.
216c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (! (isIncoming || c.isMT)) return true;
217c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
218c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // ... but we can compare phone numbers on MT calls, and we have
219c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // no control over when they begin, so we might as well
220c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
221c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        String cAddress = PhoneNumberUtils.stringFromStringAndTOA(c.number, c.TOA);
222c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return isIncoming == c.isMT && equalsHandlesNulls(address, cAddress);
223c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
224c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
225c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
226c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public String getOrigDialString(){
227c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return dialString;
228c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
229c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
230c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public String getAddress() {
231c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return address;
232c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
233c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
234c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public CdmaCall getCall() {
235c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return parent;
236c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
237c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
238c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public long getCreateTime() {
239c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return createTime;
240c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
241c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
242c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public long getConnectTime() {
243c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return connectTime;
244c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
245c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
246c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public long getDisconnectTime() {
247c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return disconnectTime;
248c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
249c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
250c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public long getDurationMillis() {
251c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (connectTimeReal == 0) {
252c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return 0;
253c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else if (duration == 0) {
254c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return SystemClock.elapsedRealtime() - connectTimeReal;
255c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
256c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return duration;
257c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
258c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
259c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
260c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public long getHoldDurationMillis() {
261c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (getState() != CdmaCall.State.HOLDING) {
262c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // If not holding, return 0
263c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return 0;
264c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
265c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return SystemClock.elapsedRealtime() - holdingStartTime;
266c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
267c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
268c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
269c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public DisconnectCause getDisconnectCause() {
270c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return cause;
271c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
272c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
273c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public boolean isIncoming() {
274c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return isIncoming;
275c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
276c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
277c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public CdmaCall.State getState() {
278c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (disconnected) {
279c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return CdmaCall.State.DISCONNECTED;
280c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
281c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return super.getState();
282c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
283c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
284c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
285c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void hangup() throws CallStateException {
286c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (!disconnected) {
287c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            owner.hangup(this);
288c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
289c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            throw new CallStateException ("disconnected");
290c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
291c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
292c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
293c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void separate() throws CallStateException {
294c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (!disconnected) {
295c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            owner.separate(this);
296c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
297c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            throw new CallStateException ("disconnected");
298c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
299c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
300c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
301c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public PostDialState getPostDialState() {
302c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return postDialState;
303c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
304c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
305c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void proceedAfterWaitChar() {
306c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (postDialState != PostDialState.WAIT) {
307c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            Log.w(LOG_TAG, "CdmaConnection.proceedAfterWaitChar(): Expected "
308c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                + "getPostDialState() to be WAIT but was " + postDialState);
309c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return;
310c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
311c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
312c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        setPostDialState(PostDialState.STARTED);
313c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
314c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        processNextPostDialChar();
315c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
316c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
317c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void proceedAfterWildChar(String str) {
318c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (postDialState != PostDialState.WILD) {
319c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            Log.w(LOG_TAG, "CdmaConnection.proceedAfterWaitChar(): Expected "
320c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                + "getPostDialState() to be WILD but was " + postDialState);
321c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return;
322c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
323c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
324c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        setPostDialState(PostDialState.STARTED);
325c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
326c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (false) {
327c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            boolean playedTone = false;
328c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            int len = (str != null ? str.length() : 0);
329c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
330c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            for (int i=0; i<len; i++) {
331c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                char c = str.charAt(i);
332c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                Message msg = null;
333c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
334c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                if (i == len-1) {
335c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    msg = h.obtainMessage(EVENT_DTMF_DONE);
336c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
337c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
338c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                if (PhoneNumberUtils.is12Key(c)) {
339c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    owner.cm.sendDtmf(c, msg);
340c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    playedTone = true;
341c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
342c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
343c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
344c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (!playedTone) {
345c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                processNextPostDialChar();
346c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
347c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
348c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // make a new postDialString, with the wild char replacement string
349c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // at the beginning, followed by the remaining postDialString.
350c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
351c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            StringBuilder buf = new StringBuilder(str);
352c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            buf.append(postDialString.substring(nextPostDialChar));
353c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            postDialString = buf.toString();
354c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            nextPostDialChar = 0;
355c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (Phone.DEBUG_PHONE) {
356c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                log("proceedAfterWildChar: new postDialString is " +
357c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        postDialString);
358c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
359c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
360c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            processNextPostDialChar();
361c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
362c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
363c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
364c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void cancelPostDial() {
365c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        setPostDialState(PostDialState.CANCELLED);
366c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
367c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
368c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
369c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * Called when this Connection is being hung up locally (eg, user pressed "end")
370c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * Note that at this point, the hangup request has been dispatched to the radio
371c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * but no response has yet been received so update() has not yet been called
372c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
373c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    void
374c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    onHangupLocal() {
375c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        cause = DisconnectCause.LOCAL;
376c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
377c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
378c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    DisconnectCause
379c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    disconnectCauseFromCode(int causeCode) {
380c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        /**
381c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * See 22.001 Annex F.4 for mapping of cause codes
382c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * to local tones
383c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         */
384c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
385c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        switch (causeCode) {
386c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.USER_BUSY:
387c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.BUSY;
388c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.NO_CIRCUIT_AVAIL:
389c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CONGESTION;
390c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.ACM_LIMIT_EXCEEDED:
391c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.LIMIT_EXCEEDED;
392c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CALL_BARRED:
393c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CALL_BARRED;
394c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.FDN_BLOCKED:
395c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.FDN_BLOCKED;
396c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_LOCKED_UNTIL_POWER_CYCLE:
397c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_LOCKED_UNTIL_POWER_CYCLE;
398c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_DROP:
399c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_DROP;
400c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_INTERCEPT:
401c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_INTERCEPT;
402c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_REORDER:
403c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_REORDER;
404c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_SO_REJECT:
405c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_SO_REJECT;
406c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_RETRY_ORDER:
407c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_RETRY_ORDER;
408c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_ACCESS_FAILURE:
409c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_ACCESS_FAILURE;
410c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_PREEMPTED:
411c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_PREEMPTED;
412c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_NOT_EMERGENCY:
413c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_NOT_EMERGENCY;
414c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.CDMA_ACCESS_BLOCKED:
415c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return DisconnectCause.CDMA_ACCESS_BLOCKED;
416c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.ERROR_UNSPECIFIED:
417c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case CallFailCause.NORMAL_CLEARING:
418c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            default:
419c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                CDMAPhone phone = owner.phone;
420c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                int serviceState = phone.getServiceState().getState();
421e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka                AppState uiccAppState = UiccController
422e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka                        .getInstance()
423e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka                        .getUiccCardApplication(UiccController.APP_FAM_3GPP2)
424e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka                        .getState();
425c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                if (serviceState == ServiceState.STATE_POWER_OFF) {
426c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    return DisconnectCause.POWER_OFF;
427c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                } else if (serviceState == ServiceState.STATE_OUT_OF_SERVICE
428c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        || serviceState == ServiceState.STATE_EMERGENCY_ONLY) {
429c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    return DisconnectCause.OUT_OF_SERVICE;
430c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                } else if (phone.mCdmaSubscriptionSource ==
431e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka                        CdmaSubscriptionSourceManager.SUBSCRIPTION_FROM_RUIM
432e287feac673ff68565b766e0e463d105fa9cef9dAlex Yakavenka                        && uiccAppState != AppState.APPSTATE_READY) {
433c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    return DisconnectCause.ICC_ERROR;
434c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                } else if (causeCode==CallFailCause.NORMAL_CLEARING) {
435c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    return DisconnectCause.NORMAL;
436c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                } else {
437c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    return DisconnectCause.ERROR_UNSPECIFIED;
438c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
439c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
440c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
441c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
442c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ void
443c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    onRemoteDisconnect(int causeCode) {
444c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        onDisconnect(disconnectCauseFromCode(causeCode));
445c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
446c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
447c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** Called when the radio indicates the connection has been disconnected */
448c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ void
449c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    onDisconnect(DisconnectCause cause) {
450c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        this.cause = cause;
451c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
452c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (!disconnected) {
453c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            doDisconnect();
454c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (false) Log.d(LOG_TAG,
455c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    "[CDMAConn] onDisconnect: cause=" + cause);
456c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
457c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            owner.phone.notifyDisconnect(this);
458c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
459c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (parent != null) {
460c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                parent.connectionDisconnected(this);
461c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
462c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
463c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        releaseWakeLock();
464c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
465c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
466c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** Called when the call waiting connection has been hung up */
467c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ void
468c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    onLocalDisconnect() {
469c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (!disconnected) {
470c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            doDisconnect();
471c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (false) Log.d(LOG_TAG,
472c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    "[CDMAConn] onLoalDisconnect" );
473c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
474c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (parent != null) {
475c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                parent.detach(this);
476c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
477c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
478c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        releaseWakeLock();
479c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
480c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
481c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // Returns true if state has changed, false if nothing changed
482c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ boolean
483c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    update (DriverCall dc) {
484c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        CdmaCall newParent;
485c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        boolean changed = false;
486c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        boolean wasConnectingInOrOut = isConnectingInOrOut();
487c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        boolean wasHolding = (getState() == CdmaCall.State.HOLDING);
488c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
489c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        newParent = parentFromDCState(dc.state);
490c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
491c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (Phone.DEBUG_PHONE) log("parent= " +parent +", newParent= " + newParent);
492c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
493c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (!equalsHandlesNulls(address, dc.number)) {
494c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (Phone.DEBUG_PHONE) log("update: phone # changed!");
495c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            address = dc.number;
496c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            changed = true;
497c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
498c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
499c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // A null cnapName should be the same as ""
500c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (TextUtils.isEmpty(dc.name)) {
501c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (!TextUtils.isEmpty(cnapName)) {
502c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                changed = true;
503c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                cnapName = "";
504c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
505c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else if (!dc.name.equals(cnapName)) {
506c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            changed = true;
507c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            cnapName = dc.name;
508c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
509c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
510c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (Phone.DEBUG_PHONE) log("--dssds----"+cnapName);
511c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        cnapNamePresentation = dc.namePresentation;
512c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        numberPresentation = dc.numberPresentation;
513c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
514c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (newParent != parent) {
515c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (parent != null) {
516c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                parent.detach(this);
517c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
518c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            newParent.attach(this, dc);
519c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            parent = newParent;
520c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            changed = true;
521c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
522c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            boolean parentStateChange;
523c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            parentStateChange = parent.update (this, dc);
524c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            changed = changed || parentStateChange;
525c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
526c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
527c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        /** Some state-transition events */
528c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
529c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (Phone.DEBUG_PHONE) log(
530c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                "Update, wasConnectingInOrOut=" + wasConnectingInOrOut +
531c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                ", wasHolding=" + wasHolding +
532c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                ", isConnectingInOrOut=" + isConnectingInOrOut() +
533c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                ", changed=" + changed);
534c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
535c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
536c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (wasConnectingInOrOut && !isConnectingInOrOut()) {
537c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            onConnectedInOrOut();
538c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
539c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
540c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (changed && !wasHolding && (getState() == CdmaCall.State.HOLDING)) {
541c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // We've transitioned into HOLDING
542c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            onStartedHolding();
543c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
544c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
545c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return changed;
546c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
547c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
548c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
549c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * Called when this Connection is in the foregroundCall
550c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * when a dial is initiated.
551c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * We know we're ACTIVE, and we know we're going to end up
552c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * HOLDING in the backgroundCall
553c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
554c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    void
555c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    fakeHoldBeforeDial() {
556c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (parent != null) {
557c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            parent.detach(this);
558c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
559c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
560c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        parent = owner.backgroundCall;
561c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        parent.attachFake(this, CdmaCall.State.HOLDING);
562c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
563c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        onStartedHolding();
564c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
565c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
566c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /*package*/ int
567c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    getCDMAIndex() throws CallStateException {
568c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (index >= 0) {
569c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return index + 1;
570c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
571c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            throw new CallStateException ("CDMA connection index not assigned");
572c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
573c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
574c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
575c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
576c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * An incoming or outgoing call has connected
577c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
578c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    void
579c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    onConnectedInOrOut() {
580c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        connectTime = System.currentTimeMillis();
581c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        connectTimeReal = SystemClock.elapsedRealtime();
582c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        duration = 0;
583c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
584c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // bug #678474: incoming call interpreted as missed call, even though
585c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // it sounds like the user has picked up the call.
586c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (Phone.DEBUG_PHONE) {
587c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            log("onConnectedInOrOut: connectTime=" + connectTime);
588c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
589c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
590c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (!isIncoming) {
591c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // outgoing calls only
592c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            processNextPostDialChar();
593c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
594c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // Only release wake lock for incoming calls, for outgoing calls the wake lock
595c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // will be released after any pause-dial is completed
596c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            releaseWakeLock();
597c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
598c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
599c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
600c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void
601c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    doDisconnect() {
602c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville       index = -1;
603c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville       disconnectTime = System.currentTimeMillis();
604c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville       duration = SystemClock.elapsedRealtime() - connectTimeReal;
605c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville       disconnected = true;
606c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
607c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
608c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void
609c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    onStartedHolding() {
610c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        holdingStartTime = SystemClock.elapsedRealtime();
611c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
612c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
613c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * Performs the appropriate action for a post-dial char, but does not
614c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * notify application. returns false if the character is invalid and
615c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * should be ignored
616c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
617c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private boolean
618c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    processPostDialChar(char c) {
619c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (PhoneNumberUtils.is12Key(c)) {
620c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            owner.cm.sendDtmf(c, h.obtainMessage(EVENT_DTMF_DONE));
621c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else if (c == PhoneNumberUtils.PAUSE) {
622c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            setPostDialState(PostDialState.PAUSE);
623c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
624c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // Upon occurrences of the separator, the UE shall
625c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // pause again for 2 seconds before sending any
626c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // further DTMF digits.
627c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            h.sendMessageDelayed(h.obtainMessage(EVENT_PAUSE_DONE),
628c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                                            PAUSE_DELAY_MILLIS);
629c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else if (c == PhoneNumberUtils.WAIT) {
630c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            setPostDialState(PostDialState.WAIT);
631c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else if (c == PhoneNumberUtils.WILD) {
632c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            setPostDialState(PostDialState.WILD);
633c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
634c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return false;
635c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
636c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
637c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return true;
638c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
639c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
640c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public String getRemainingPostDialString() {
641c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (postDialState == PostDialState.CANCELLED
642c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                || postDialState == PostDialState.COMPLETE
643c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                || postDialString == null
644c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                || postDialString.length() <= nextPostDialChar) {
645c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return "";
646c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
647c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
648c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        String subStr = postDialString.substring(nextPostDialChar);
649c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (subStr != null) {
650c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            int wIndex = subStr.indexOf(PhoneNumberUtils.WAIT);
651c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            int pIndex = subStr.indexOf(PhoneNumberUtils.PAUSE);
652c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
653c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (wIndex > 0 && (wIndex < pIndex || pIndex <= 0)) {
654c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                subStr = subStr.substring(0, wIndex);
655c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            } else if (pIndex > 0) {
656c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                subStr = subStr.substring(0, pIndex);
657c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
658c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
659c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return subStr;
660c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
661c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
662c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public void updateParent(CdmaCall oldParent, CdmaCall newParent){
663c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (newParent != oldParent) {
664c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (oldParent != null) {
665c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                oldParent.detach(this);
666c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
667c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            newParent.attachFake(this, CdmaCall.State.ACTIVE);
668c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            parent = newParent;
669c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
670c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
671c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
672c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    @Override
673c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    protected void finalize()
674c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    {
675c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        /**
676c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * It is understood that This finializer is not guaranteed
677c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * to be called and the release lock call is here just in
678c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * case there is some path that doesn't call onDisconnect
679c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * and or onConnectedInOrOut.
680c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         */
681c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (mPartialWakeLock.isHeld()) {
682c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            Log.e(LOG_TAG, "[CdmaConn] UNEXPECTED; mPartialWakeLock is held when finalizing.");
683c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
684c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        releaseWakeLock();
685c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
686c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
687c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    void processNextPostDialChar() {
688c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        char c = 0;
689c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        Registrant postDialHandler;
690c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
691c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (postDialState == PostDialState.CANCELLED) {
692c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            releaseWakeLock();
693c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            //Log.v("CDMA", "##### processNextPostDialChar: postDialState == CANCELLED, bail");
694c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return;
695c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
696c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
697c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (postDialString == null ||
698c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                postDialString.length() <= nextPostDialChar) {
699c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            setPostDialState(PostDialState.COMPLETE);
700c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
701c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // We were holding a wake lock until pause-dial was complete, so give it up now
702c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            releaseWakeLock();
703c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
704c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // notifyMessage.arg1 is 0 on complete
705c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            c = 0;
706c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
707c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            boolean isValid;
708c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
709c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            setPostDialState(PostDialState.STARTED);
710c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
711c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            c = postDialString.charAt(nextPostDialChar++);
712c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
713c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            isValid = processPostDialChar(c);
714c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
715c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (!isValid) {
716c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                // Will call processNextPostDialChar
717c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                h.obtainMessage(EVENT_NEXT_POST_DIAL).sendToTarget();
718c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                // Don't notify application
719c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                Log.e("CDMA", "processNextPostDialChar: c=" + c + " isn't valid!");
720c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return;
721c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
722c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
723c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
724c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        postDialHandler = owner.phone.mPostDialHandler;
725c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
726c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        Message notifyMessage;
727c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
728c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (postDialHandler != null &&
729c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                (notifyMessage = postDialHandler.messageForRegistrant()) != null) {
730c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // The AsyncResult.result is the Connection object
731c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            PostDialState state = postDialState;
732c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            AsyncResult ar = AsyncResult.forMessage(notifyMessage);
733c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            ar.result = this;
734c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            ar.userObj = state;
735c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
736c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // arg1 is the character that was/is being processed
737c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            notifyMessage.arg1 = c;
738c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
739c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            notifyMessage.sendToTarget();
740c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
741c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
742c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
743c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
744c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /** "connecting" means "has never been ACTIVE" for both incoming
745c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *  and outgoing calls
746c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
747c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private boolean
748c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    isConnectingInOrOut() {
749c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return parent == null || parent == owner.ringingCall
750c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            || parent.state == CdmaCall.State.DIALING
751c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            || parent.state == CdmaCall.State.ALERTING;
752c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
753c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
754c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private CdmaCall
755c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    parentFromDCState (DriverCall.State state) {
756c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        switch (state) {
757c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case ACTIVE:
758c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case DIALING:
759c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case ALERTING:
760c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return owner.foregroundCall;
761c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            //break;
762c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
763c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case HOLDING:
764c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return owner.backgroundCall;
765c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            //break;
766c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
767c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case INCOMING:
768c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            case WAITING:
769c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                return owner.ringingCall;
770c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            //break;
771c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
772c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            default:
773c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                throw new RuntimeException("illegal call state: " + state);
774c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
775c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
776c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
777c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
778c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * Set post dial state and acquire wake lock while switching to "started" or "wait"
779c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * state, the wake lock will be released if state switches out of "started" or "wait"
780c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * state or after WAKE_LOCK_TIMEOUT_MILLIS.
781c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * @param s new PostDialState
782c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
783c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void setPostDialState(PostDialState s) {
784c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (s == PostDialState.STARTED ||
785c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                s == PostDialState.PAUSE) {
786c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            synchronized (mPartialWakeLock) {
787c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                if (mPartialWakeLock.isHeld()) {
788c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    h.removeMessages(EVENT_WAKE_LOCK_TIMEOUT);
789c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                } else {
790c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    acquireWakeLock();
791c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
792c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                Message msg = h.obtainMessage(EVENT_WAKE_LOCK_TIMEOUT);
793c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                h.sendMessageDelayed(msg, WAKE_LOCK_TIMEOUT_MILLIS);
794c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
795c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        } else {
796c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            h.removeMessages(EVENT_WAKE_LOCK_TIMEOUT);
797c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            releaseWakeLock();
798c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
799c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        postDialState = s;
800c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
801c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
802c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void createWakeLock(Context context) {
803c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        PowerManager pm = (PowerManager)context.getSystemService(Context.POWER_SERVICE);
804c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        mPartialWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, LOG_TAG);
805c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
806c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
807c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void acquireWakeLock() {
808c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        log("acquireWakeLock");
809c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        mPartialWakeLock.acquire();
810c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
811c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
812c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void releaseWakeLock() {
813c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        synchronized (mPartialWakeLock) {
814c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (mPartialWakeLock.isHeld()) {
815c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                log("releaseWakeLock");
816c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                mPartialWakeLock.release();
817c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
818c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
819c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
820c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
821c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private static boolean isPause(char c) {
822c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return c == PhoneNumberUtils.PAUSE;
823c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
824c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
825c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private static boolean isWait(char c) {
826c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return c == PhoneNumberUtils.WAIT;
827c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
828c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
829c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // This function is to find the next PAUSE character index if
830c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // multiple pauses in a row. Otherwise it finds the next non PAUSE or
831c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // non WAIT character index.
832c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private static int
833c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    findNextPCharOrNonPOrNonWCharIndex(String phoneNumber, int currIndex) {
834c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        boolean wMatched = isWait(phoneNumber.charAt(currIndex));
835c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        int index = currIndex + 1;
836c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        int length = phoneNumber.length();
837c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        while (index < length) {
838c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            char cNext = phoneNumber.charAt(index);
839c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // if there is any W inside P/W sequence,mark it
840c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (isWait(cNext)) {
841c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                wMatched = true;
842c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
843c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // if any characters other than P/W chars after P/W sequence
844c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            // we break out the loop and append the correct
845c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (!isWait(cNext) && !isPause(cNext)) {
846c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                break;
847c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
848c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            index++;
849c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
850c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
851c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // It means the PAUSE character(s) is in the middle of dial string
852c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // and it needs to be handled one by one.
853c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if ((index < length) && (index > (currIndex + 1))  &&
854c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            ((wMatched == false) && isPause(phoneNumber.charAt(currIndex)))) {
855c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return (currIndex + 1);
856c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
857c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return index;
858c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
859c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
860c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // This function returns either PAUSE or WAIT character to append.
861c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // It is based on the next non PAUSE/WAIT character in the phoneNumber and the
862c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    // index for the current PAUSE/WAIT character
863c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private static char
864c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    findPOrWCharToAppend(String phoneNumber, int currPwIndex, int nextNonPwCharIndex) {
865c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        char c = phoneNumber.charAt(currPwIndex);
866c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        char ret;
867c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
868c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // Append the PW char
869c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        ret = (isPause(c)) ? PhoneNumberUtils.PAUSE : PhoneNumberUtils.WAIT;
870c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
871c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // If the nextNonPwCharIndex is greater than currPwIndex + 1,
872c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // it means the PW sequence contains not only P characters.
873c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // Since for the sequence that only contains P character,
874c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // the P character is handled one by one, the nextNonPwCharIndex
875c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // equals to currPwIndex + 1.
876c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // In this case, skip P, append W.
877c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (nextNonPwCharIndex > (currPwIndex + 1)) {
878c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            ret = PhoneNumberUtils.WAIT;
879c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
880c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return ret;
881c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
882c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
883c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    /**
884c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * format original dial string
885c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * 1) convert international dialing prefix "+" to
886c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *    string specified per region
887c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *
888c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     * 2) handle corner cases for PAUSE/WAIT dialing:
889c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *
890c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *    If PAUSE/WAIT sequence at the end, ignore them.
891c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *
892c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *    If consecutive PAUSE/WAIT sequence in the middle of the string,
893c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     *    and if there is any WAIT in PAUSE/WAIT sequence, treat them like WAIT.
894c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville     */
895c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public static String formatDialString(String phoneNumber) {
896c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        /**
897c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * TODO(cleanup): This function should move to PhoneNumberUtils, and
898c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         * tests should be added.
899c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville         */
900c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
901c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        if (phoneNumber == null) {
902c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            return null;
903c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
904c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        int length = phoneNumber.length();
905c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        StringBuilder ret = new StringBuilder();
906c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        char c;
907c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        int currIndex = 0;
908c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
909c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        while (currIndex < length) {
910c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            c = phoneNumber.charAt(currIndex);
911c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            if (isPause(c) || isWait(c)) {
912c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                if (currIndex < length - 1) {
913c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    // if PW not at the end
914c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    int nextIndex = findNextPCharOrNonPOrNonWCharIndex(phoneNumber, currIndex);
915c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    // If there is non PW char following PW sequence
916c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    if (nextIndex < length) {
917c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        char pC = findPOrWCharToAppend(phoneNumber, currIndex, nextIndex);
918c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        ret.append(pC);
919c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        // If PW char sequence has more than 2 PW characters,
920c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        // skip to the last PW character since the sequence already be
921c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        // converted to WAIT character
922c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        if (nextIndex > (currIndex + 1)) {
923c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                            currIndex = nextIndex - 1;
924c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        }
925c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    } else if (nextIndex == length) {
926c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        // It means PW characters at the end, ignore
927c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                        currIndex = length - 1;
928c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                    }
929c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                }
930c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            } else {
931c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville                ret.append(c);
932c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            }
933c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville            currIndex++;
934c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        }
935c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return PhoneNumberUtils.cdmaCheckAndProcessPlusCode(ret.toString());
936c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
937c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
938c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    private void log(String msg) {
939c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        Log.d(LOG_TAG, "[CDMAConn] " + msg);
940c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
941c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
942c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    @Override
943c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public int getNumberPresentation() {
944c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return numberPresentation;
945c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
946c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville
947c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    @Override
948c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    public UUSInfo getUUSInfo() {
949c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        // UUS information not supported in CDMA
950c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville        return null;
951c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville    }
952c38bb60d867c5d61d90b7179a9ed2b2d1848124fWink Saville}
953