blob: 33702793994adb1abab1dfe5d302b6591f361bca [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2007 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package com.android.server;
18
19import android.content.Context;
20import android.content.Intent;
21import android.content.pm.PackageManager;
Robert Greenwalt8afddad2010-08-20 10:01:55 -070022import android.net.NetworkUtils;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080023import android.os.Binder;
24import android.os.Bundle;
25import android.os.IBinder;
26import android.os.RemoteException;
27import android.telephony.CellLocation;
28import android.telephony.PhoneStateListener;
29import android.telephony.ServiceState;
Wink Savillee9b06d72009-05-18 21:47:50 -070030import android.telephony.SignalStrength;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080031import android.telephony.TelephonyManager;
32import android.text.TextUtils;
Joe Onorato8a9b2202010-02-26 18:56:32 -080033import android.util.Slog;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080034
35import java.util.ArrayList;
36import java.io.FileDescriptor;
37import java.io.PrintWriter;
38
39import com.android.internal.app.IBatteryStats;
40import com.android.internal.telephony.ITelephonyRegistry;
41import com.android.internal.telephony.IPhoneStateListener;
42import com.android.internal.telephony.DefaultPhoneNotifier;
43import com.android.internal.telephony.Phone;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080044import com.android.internal.telephony.TelephonyIntents;
45import com.android.server.am.BatteryStatsService;
46
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080047/**
Wink Savillee9b06d72009-05-18 21:47:50 -070048 * Since phone process can be restarted, this class provides a centralized place
49 * that applications can register and be called back from.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080050 */
51class TelephonyRegistry extends ITelephonyRegistry.Stub {
52 private static final String TAG = "TelephonyRegistry";
53
54 private static class Record {
55 String pkgForDebug;
Wink Savillee9b06d72009-05-18 21:47:50 -070056
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080057 IBinder binder;
Wink Savillee9b06d72009-05-18 21:47:50 -070058
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080059 IPhoneStateListener callback;
Wink Savillee9b06d72009-05-18 21:47:50 -070060
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080061 int events;
62 }
63
64 private final Context mContext;
Wink Savillee9b06d72009-05-18 21:47:50 -070065
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080066 private final ArrayList<Record> mRecords = new ArrayList();
Wink Savillee9b06d72009-05-18 21:47:50 -070067
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080068 private final IBatteryStats mBatteryStats;
69
70 private int mCallState = TelephonyManager.CALL_STATE_IDLE;
Wink Savillee9b06d72009-05-18 21:47:50 -070071
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080072 private String mCallIncomingNumber = "";
Wink Savillee9b06d72009-05-18 21:47:50 -070073
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080074 private ServiceState mServiceState = new ServiceState();
Wink Savillee9b06d72009-05-18 21:47:50 -070075
76 private SignalStrength mSignalStrength = new SignalStrength();
77
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080078 private boolean mMessageWaiting = false;
Wink Savillee9b06d72009-05-18 21:47:50 -070079
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080080 private boolean mCallForwarding = false;
Wink Savillee9b06d72009-05-18 21:47:50 -070081
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080082 private int mDataActivity = TelephonyManager.DATA_ACTIVITY_NONE;
Wink Savillee9b06d72009-05-18 21:47:50 -070083
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080084 private int mDataConnectionState = TelephonyManager.DATA_CONNECTED;
Wink Savillee9b06d72009-05-18 21:47:50 -070085
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080086 private boolean mDataConnectionPossible = false;
Wink Savillee9b06d72009-05-18 21:47:50 -070087
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080088 private String mDataConnectionReason = "";
Wink Savillee9b06d72009-05-18 21:47:50 -070089
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080090 private String mDataConnectionApn = "";
Wink Savillee9b06d72009-05-18 21:47:50 -070091
Robert Greenwalt42acef32009-08-12 16:08:25 -070092 private String[] mDataConnectionApnTypes = null;
93
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080094 private String mDataConnectionInterfaceName = "";
Wink Savillee9b06d72009-05-18 21:47:50 -070095
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080096 private Bundle mCellLocation = new Bundle();
97
Robert Greenwalt98e0b142009-10-08 21:15:52 -070098 private int mDataConnectionNetworkType;
99
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700100 static final int PHONE_STATE_PERMISSION_MASK =
101 PhoneStateListener.LISTEN_CALL_FORWARDING_INDICATOR |
102 PhoneStateListener.LISTEN_CALL_STATE |
103 PhoneStateListener.LISTEN_DATA_ACTIVITY |
104 PhoneStateListener.LISTEN_DATA_CONNECTION_STATE |
105 PhoneStateListener.LISTEN_MESSAGE_WAITING_INDICATOR;
106
Wink Savillee9b06d72009-05-18 21:47:50 -0700107 // we keep a copy of all of the state so we can send it out when folks
108 // register for it
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800109 //
Wink Savillee9b06d72009-05-18 21:47:50 -0700110 // In these calls we call with the lock held. This is safe becasuse remote
111 // calls go through a oneway interface and local calls going through a
112 // handler before they get to app code.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800113
114 TelephonyRegistry(Context context) {
David 'Digit' Turner4ef8ec32009-09-25 11:33:24 -0700115 CellLocation location = CellLocation.getEmpty();
116
117 // Note that location can be null for non-phone builds like
118 // like the generic one.
119 if (location != null) {
120 location.fillInNotifierBundle(mCellLocation);
121 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800122 mContext = context;
123 mBatteryStats = BatteryStatsService.getService();
124 }
125
126 public void listen(String pkgForDebug, IPhoneStateListener callback, int events,
127 boolean notifyNow) {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800128 // Slog.d(TAG, "listen pkg=" + pkgForDebug + " events=0x" +
Wink Savillee9b06d72009-05-18 21:47:50 -0700129 // Integer.toHexString(events));
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800130 if (events != 0) {
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700131 /* Checks permission and throws Security exception */
132 checkListenerPermission(events);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800133
134 synchronized (mRecords) {
135 // register
136 Record r = null;
137 find_and_add: {
138 IBinder b = callback.asBinder();
139 final int N = mRecords.size();
Wink Savillee9b06d72009-05-18 21:47:50 -0700140 for (int i = 0; i < N; i++) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800141 r = mRecords.get(i);
142 if (b == r.binder) {
143 break find_and_add;
144 }
145 }
146 r = new Record();
147 r.binder = b;
148 r.callback = callback;
149 r.pkgForDebug = pkgForDebug;
150 mRecords.add(r);
151 }
152 int send = events & (events ^ r.events);
153 r.events = events;
154 if (notifyNow) {
155 if ((events & PhoneStateListener.LISTEN_SERVICE_STATE) != 0) {
156 sendServiceState(r, mServiceState);
157 }
158 if ((events & PhoneStateListener.LISTEN_SIGNAL_STRENGTH) != 0) {
159 try {
Wink Savillee9b06d72009-05-18 21:47:50 -0700160 int gsmSignalStrength = mSignalStrength.getGsmSignalStrength();
161 r.callback.onSignalStrengthChanged((gsmSignalStrength == 99 ? -1
162 : gsmSignalStrength));
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800163 } catch (RemoteException ex) {
164 remove(r.binder);
165 }
166 }
167 if ((events & PhoneStateListener.LISTEN_MESSAGE_WAITING_INDICATOR) != 0) {
168 try {
169 r.callback.onMessageWaitingIndicatorChanged(mMessageWaiting);
170 } catch (RemoteException ex) {
171 remove(r.binder);
172 }
173 }
174 if ((events & PhoneStateListener.LISTEN_CALL_FORWARDING_INDICATOR) != 0) {
175 try {
176 r.callback.onCallForwardingIndicatorChanged(mCallForwarding);
177 } catch (RemoteException ex) {
178 remove(r.binder);
179 }
180 }
181 if ((events & PhoneStateListener.LISTEN_CELL_LOCATION) != 0) {
182 sendCellLocation(r, mCellLocation);
183 }
184 if ((events & PhoneStateListener.LISTEN_CALL_STATE) != 0) {
185 try {
186 r.callback.onCallStateChanged(mCallState, mCallIncomingNumber);
187 } catch (RemoteException ex) {
188 remove(r.binder);
189 }
190 }
191 if ((events & PhoneStateListener.LISTEN_DATA_CONNECTION_STATE) != 0) {
192 try {
Robert Greenwalt98e0b142009-10-08 21:15:52 -0700193 r.callback.onDataConnectionStateChanged(mDataConnectionState,
194 mDataConnectionNetworkType);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800195 } catch (RemoteException ex) {
196 remove(r.binder);
197 }
198 }
199 if ((events & PhoneStateListener.LISTEN_DATA_ACTIVITY) != 0) {
200 try {
201 r.callback.onDataActivity(mDataActivity);
202 } catch (RemoteException ex) {
203 remove(r.binder);
204 }
205 }
Wink Savillee9b06d72009-05-18 21:47:50 -0700206 if ((events & PhoneStateListener.LISTEN_SIGNAL_STRENGTHS) != 0) {
207 try {
208 r.callback.onSignalStrengthsChanged(mSignalStrength);
209 } catch (RemoteException ex) {
210 remove(r.binder);
211 }
212 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800213 }
214 }
215 } else {
216 remove(callback.asBinder());
217 }
218 }
219
220 private void remove(IBinder binder) {
221 synchronized (mRecords) {
Wink Savillee9b06d72009-05-18 21:47:50 -0700222 final int recordCount = mRecords.size();
223 for (int i = 0; i < recordCount; i++) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800224 if (mRecords.get(i).binder == binder) {
225 mRecords.remove(i);
226 return;
227 }
228 }
229 }
230 }
231
232 public void notifyCallState(int state, String incomingNumber) {
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700233 if (!checkNotifyPermission("notifyCallState()")) {
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700234 return;
235 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800236 synchronized (mRecords) {
237 mCallState = state;
238 mCallIncomingNumber = incomingNumber;
Wink Savillee9b06d72009-05-18 21:47:50 -0700239 for (int i = mRecords.size() - 1; i >= 0; i--) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800240 Record r = mRecords.get(i);
241 if ((r.events & PhoneStateListener.LISTEN_CALL_STATE) != 0) {
242 try {
243 r.callback.onCallStateChanged(state, incomingNumber);
244 } catch (RemoteException ex) {
245 remove(r.binder);
246 }
247 }
248 }
249 }
250 broadcastCallStateChanged(state, incomingNumber);
251 }
252
253 public void notifyServiceState(ServiceState state) {
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700254 if (!checkNotifyPermission("notifyServiceState()")){
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700255 return;
Wink Savillee9b06d72009-05-18 21:47:50 -0700256 }
Joe Onorato431bb222010-10-18 19:13:23 -0400257 Slog.i(TAG, "notifyServiceState: " + state);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800258 synchronized (mRecords) {
259 mServiceState = state;
Wink Savillee9b06d72009-05-18 21:47:50 -0700260 for (int i = mRecords.size() - 1; i >= 0; i--) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800261 Record r = mRecords.get(i);
262 if ((r.events & PhoneStateListener.LISTEN_SERVICE_STATE) != 0) {
263 sendServiceState(r, state);
264 }
265 }
266 }
267 broadcastServiceStateChanged(state);
268 }
269
Wink Savillee9b06d72009-05-18 21:47:50 -0700270 public void notifySignalStrength(SignalStrength signalStrength) {
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700271 if (!checkNotifyPermission("notifySignalStrength()")) {
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700272 return;
Wink Savillee9b06d72009-05-18 21:47:50 -0700273 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800274 synchronized (mRecords) {
Wink Savillee9b06d72009-05-18 21:47:50 -0700275 mSignalStrength = signalStrength;
276 for (int i = mRecords.size() - 1; i >= 0; i--) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800277 Record r = mRecords.get(i);
Wink Savillee9b06d72009-05-18 21:47:50 -0700278 if ((r.events & PhoneStateListener.LISTEN_SIGNAL_STRENGTHS) != 0) {
279 sendSignalStrength(r, signalStrength);
280 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800281 if ((r.events & PhoneStateListener.LISTEN_SIGNAL_STRENGTH) != 0) {
282 try {
Wink Savillee9b06d72009-05-18 21:47:50 -0700283 int gsmSignalStrength = signalStrength.getGsmSignalStrength();
284 r.callback.onSignalStrengthChanged((gsmSignalStrength == 99 ? -1
285 : gsmSignalStrength));
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800286 } catch (RemoteException ex) {
287 remove(r.binder);
288 }
289 }
290 }
291 }
Wink Savillee9b06d72009-05-18 21:47:50 -0700292 broadcastSignalStrengthChanged(signalStrength);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800293 }
294
295 public void notifyMessageWaitingChanged(boolean mwi) {
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700296 if (!checkNotifyPermission("notifyMessageWaitingChanged()")) {
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700297 return;
Wink Savillee9b06d72009-05-18 21:47:50 -0700298 }
Joe Onorato431bb222010-10-18 19:13:23 -0400299 Slog.i(TAG, "notifyMessageWaitingChanged: " + mwi);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800300 synchronized (mRecords) {
301 mMessageWaiting = mwi;
Wink Savillee9b06d72009-05-18 21:47:50 -0700302 for (int i = mRecords.size() - 1; i >= 0; i--) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800303 Record r = mRecords.get(i);
304 if ((r.events & PhoneStateListener.LISTEN_MESSAGE_WAITING_INDICATOR) != 0) {
305 try {
306 r.callback.onMessageWaitingIndicatorChanged(mwi);
307 } catch (RemoteException ex) {
308 remove(r.binder);
309 }
310 }
311 }
312 }
313 }
314
315 public void notifyCallForwardingChanged(boolean cfi) {
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700316 if (!checkNotifyPermission("notifyCallForwardingChanged()")) {
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700317 return;
Wink Savillee9b06d72009-05-18 21:47:50 -0700318 }
Joe Onorato431bb222010-10-18 19:13:23 -0400319 Slog.i(TAG, "notifyCallForwardingChanged: " + cfi);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800320 synchronized (mRecords) {
321 mCallForwarding = cfi;
Wink Savillee9b06d72009-05-18 21:47:50 -0700322 for (int i = mRecords.size() - 1; i >= 0; i--) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800323 Record r = mRecords.get(i);
324 if ((r.events & PhoneStateListener.LISTEN_CALL_FORWARDING_INDICATOR) != 0) {
325 try {
326 r.callback.onCallForwardingIndicatorChanged(cfi);
327 } catch (RemoteException ex) {
328 remove(r.binder);
329 }
330 }
331 }
332 }
333 }
334
335 public void notifyDataActivity(int state) {
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700336 if (!checkNotifyPermission("notifyDataActivity()" )) {
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700337 return;
Wink Savillee9b06d72009-05-18 21:47:50 -0700338 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800339 synchronized (mRecords) {
340 mDataActivity = state;
Wink Savillee9b06d72009-05-18 21:47:50 -0700341 for (int i = mRecords.size() - 1; i >= 0; i--) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800342 Record r = mRecords.get(i);
343 if ((r.events & PhoneStateListener.LISTEN_DATA_ACTIVITY) != 0) {
344 try {
345 r.callback.onDataActivity(state);
346 } catch (RemoteException ex) {
347 remove(r.binder);
348 }
349 }
350 }
351 }
352 }
353
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700354 public void notifyDataConnection(int state, boolean isDataConnectivityPossible,
Robert Greenwalt8afddad2010-08-20 10:01:55 -0700355 String reason, String apn, String[] apnTypes, String interfaceName, int networkType,
356 String gateway) {
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700357 if (!checkNotifyPermission("notifyDataConnection()" )) {
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700358 return;
Wink Savillee9b06d72009-05-18 21:47:50 -0700359 }
Joe Onorato431bb222010-10-18 19:13:23 -0400360 Slog.i(TAG, "notifyDataConnection: state=" + state + " isDataConnectivityPossible="
361 + isDataConnectivityPossible + " reason=" + reason
362 + " interfaceName=" + interfaceName + " networkType=" + networkType);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800363 synchronized (mRecords) {
364 mDataConnectionState = state;
Wink Savillee9b06d72009-05-18 21:47:50 -0700365 mDataConnectionPossible = isDataConnectivityPossible;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800366 mDataConnectionReason = reason;
367 mDataConnectionApn = apn;
Robert Greenwalt42acef32009-08-12 16:08:25 -0700368 mDataConnectionApnTypes = apnTypes;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800369 mDataConnectionInterfaceName = interfaceName;
Robert Greenwalt98e0b142009-10-08 21:15:52 -0700370 mDataConnectionNetworkType = networkType;
Wink Savillee9b06d72009-05-18 21:47:50 -0700371 for (int i = mRecords.size() - 1; i >= 0; i--) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800372 Record r = mRecords.get(i);
373 if ((r.events & PhoneStateListener.LISTEN_DATA_CONNECTION_STATE) != 0) {
374 try {
Robert Greenwalt98e0b142009-10-08 21:15:52 -0700375 r.callback.onDataConnectionStateChanged(state, networkType);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800376 } catch (RemoteException ex) {
377 remove(r.binder);
378 }
379 }
380 }
381 }
Wink Savillee9b06d72009-05-18 21:47:50 -0700382 broadcastDataConnectionStateChanged(state, isDataConnectivityPossible, reason, apn,
Robert Greenwalt8afddad2010-08-20 10:01:55 -0700383 apnTypes, interfaceName, gateway);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800384 }
385
386 public void notifyDataConnectionFailed(String reason) {
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700387 if (!checkNotifyPermission("notifyDataConnectionFailed()")) {
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700388 return;
Wink Savillee9b06d72009-05-18 21:47:50 -0700389 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800390 /*
391 * This is commented out because there is on onDataConnectionFailed callback
Wink Savillee9b06d72009-05-18 21:47:50 -0700392 * on PhoneStateListener. There should be
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800393 synchronized (mRecords) {
394 mDataConnectionFailedReason = reason;
395 final int N = mRecords.size();
396 for (int i=N-1; i>=0; i--) {
397 Record r = mRecords.get(i);
398 if ((r.events & PhoneStateListener.LISTEN_DATA_CONNECTION_FAILED) != 0) {
399 // XXX
400 }
401 }
402 }
403 */
404 broadcastDataConnectionFailed(reason);
405 }
406
407 public void notifyCellLocation(Bundle cellLocation) {
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700408 if (!checkNotifyPermission("notifyCellLocation()")) {
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700409 return;
Wink Savillee9b06d72009-05-18 21:47:50 -0700410 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800411 synchronized (mRecords) {
412 mCellLocation = cellLocation;
Wink Savillee9b06d72009-05-18 21:47:50 -0700413 for (int i = mRecords.size() - 1; i >= 0; i--) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800414 Record r = mRecords.get(i);
415 if ((r.events & PhoneStateListener.LISTEN_CELL_LOCATION) != 0) {
416 sendCellLocation(r, cellLocation);
417 }
418 }
419 }
420 }
421
Wink Savillee9b06d72009-05-18 21:47:50 -0700422 /**
423 * Copy the service state object so they can't mess it up in the local calls
424 */
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700425 public void sendServiceState(Record r, ServiceState state) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800426 try {
427 r.callback.onServiceStateChanged(new ServiceState(state));
428 } catch (RemoteException ex) {
429 remove(r.binder);
430 }
431 }
432
Wink Savillee9b06d72009-05-18 21:47:50 -0700433 private void sendCellLocation(Record r, Bundle cellLocation) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800434 try {
435 r.callback.onCellLocationChanged(new Bundle(cellLocation));
436 } catch (RemoteException ex) {
437 remove(r.binder);
438 }
439 }
440
Wink Savillee9b06d72009-05-18 21:47:50 -0700441 private void sendSignalStrength(Record r, SignalStrength signalStrength) {
442 try {
443 r.callback.onSignalStrengthsChanged(new SignalStrength(signalStrength));
444 } catch (RemoteException ex) {
445 remove(r.binder);
446 }
447 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800448
449 @Override
450 public void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
451 if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.DUMP)
452 != PackageManager.PERMISSION_GRANTED) {
453 pw.println("Permission Denial: can't dump telephony.registry from from pid="
Wink Savillee9b06d72009-05-18 21:47:50 -0700454 + Binder.getCallingPid() + ", uid=" + Binder.getCallingUid());
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800455 return;
456 }
457 synchronized (mRecords) {
Wink Savillee9b06d72009-05-18 21:47:50 -0700458 final int recordCount = mRecords.size();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800459 pw.println("last known state:");
460 pw.println(" mCallState=" + mCallState);
461 pw.println(" mCallIncomingNumber=" + mCallIncomingNumber);
462 pw.println(" mServiceState=" + mServiceState);
463 pw.println(" mSignalStrength=" + mSignalStrength);
464 pw.println(" mMessageWaiting=" + mMessageWaiting);
465 pw.println(" mCallForwarding=" + mCallForwarding);
466 pw.println(" mDataActivity=" + mDataActivity);
467 pw.println(" mDataConnectionState=" + mDataConnectionState);
468 pw.println(" mDataConnectionPossible=" + mDataConnectionPossible);
469 pw.println(" mDataConnectionReason=" + mDataConnectionReason);
470 pw.println(" mDataConnectionApn=" + mDataConnectionApn);
471 pw.println(" mDataConnectionInterfaceName=" + mDataConnectionInterfaceName);
472 pw.println(" mCellLocation=" + mCellLocation);
Wink Savillee9b06d72009-05-18 21:47:50 -0700473 pw.println("registrations: count=" + recordCount);
474 for (int i = 0; i < recordCount; i++) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800475 Record r = mRecords.get(i);
476 pw.println(" " + r.pkgForDebug + " 0x" + Integer.toHexString(r.events));
477 }
478 }
479 }
480
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800481 //
482 // the legacy intent broadcasting
483 //
484
485 private void broadcastServiceStateChanged(ServiceState state) {
Amith Yamasani32dbefd2009-06-19 09:21:17 -0700486 long ident = Binder.clearCallingIdentity();
487 try {
Amith Yamasanif37447b2009-10-08 18:28:01 -0700488 mBatteryStats.notePhoneState(state.getState());
Amith Yamasani32dbefd2009-06-19 09:21:17 -0700489 } catch (RemoteException re) {
490 // Can't do much
491 } finally {
492 Binder.restoreCallingIdentity(ident);
493 }
494
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800495 Intent intent = new Intent(TelephonyIntents.ACTION_SERVICE_STATE_CHANGED);
Dianne Hackborn1c633fc2009-12-08 19:45:14 -0800496 intent.addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800497 Bundle data = new Bundle();
498 state.fillInNotifierBundle(data);
499 intent.putExtras(data);
500 mContext.sendStickyBroadcast(intent);
501 }
502
Wink Savillee9b06d72009-05-18 21:47:50 -0700503 private void broadcastSignalStrengthChanged(SignalStrength signalStrength) {
Dianne Hackborn627bba72009-03-24 22:32:56 -0700504 long ident = Binder.clearCallingIdentity();
505 try {
Wink Savillee9b06d72009-05-18 21:47:50 -0700506 mBatteryStats.notePhoneSignalStrength(signalStrength);
Dianne Hackborn627bba72009-03-24 22:32:56 -0700507 } catch (RemoteException e) {
Wink Savillee9b06d72009-05-18 21:47:50 -0700508 /* The remote entity disappeared, we can safely ignore the exception. */
Dianne Hackborn627bba72009-03-24 22:32:56 -0700509 } finally {
510 Binder.restoreCallingIdentity(ident);
511 }
Wink Savillee9b06d72009-05-18 21:47:50 -0700512
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800513 Intent intent = new Intent(TelephonyIntents.ACTION_SIGNAL_STRENGTH_CHANGED);
Dianne Hackborn1c633fc2009-12-08 19:45:14 -0800514 intent.addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING);
Wink Savillee9b06d72009-05-18 21:47:50 -0700515 Bundle data = new Bundle();
516 signalStrength.fillInNotifierBundle(data);
517 intent.putExtras(data);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800518 mContext.sendStickyBroadcast(intent);
519 }
520
521 private void broadcastCallStateChanged(int state, String incomingNumber) {
522 long ident = Binder.clearCallingIdentity();
523 try {
524 if (state == TelephonyManager.CALL_STATE_IDLE) {
525 mBatteryStats.notePhoneOff();
526 } else {
527 mBatteryStats.notePhoneOn();
528 }
529 } catch (RemoteException e) {
Wink Savillee9b06d72009-05-18 21:47:50 -0700530 /* The remote entity disappeared, we can safely ignore the exception. */
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800531 } finally {
532 Binder.restoreCallingIdentity(ident);
533 }
Wink Savillee9b06d72009-05-18 21:47:50 -0700534
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800535 Intent intent = new Intent(TelephonyManager.ACTION_PHONE_STATE_CHANGED);
Dianne Hackborn1c633fc2009-12-08 19:45:14 -0800536 intent.addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING);
Wink Savillee9b06d72009-05-18 21:47:50 -0700537 intent.putExtra(Phone.STATE_KEY, DefaultPhoneNotifier.convertCallState(state).toString());
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800538 if (!TextUtils.isEmpty(incomingNumber)) {
539 intent.putExtra(TelephonyManager.EXTRA_INCOMING_NUMBER, incomingNumber);
540 }
541 mContext.sendBroadcast(intent, android.Manifest.permission.READ_PHONE_STATE);
542 }
543
Robert Greenwalt42acef32009-08-12 16:08:25 -0700544 private void broadcastDataConnectionStateChanged(int state,
545 boolean isDataConnectivityPossible,
Robert Greenwalt8afddad2010-08-20 10:01:55 -0700546 String reason, String apn, String[] apnTypes, String interfaceName, String gateway) {
Dianne Hackborn627bba72009-03-24 22:32:56 -0700547 // Note: not reporting to the battery stats service here, because the
548 // status bar takes care of that after taking into account all of the
549 // required info.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800550 Intent intent = new Intent(TelephonyIntents.ACTION_ANY_DATA_CONNECTION_STATE_CHANGED);
Dianne Hackborn1c633fc2009-12-08 19:45:14 -0800551 intent.addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800552 intent.putExtra(Phone.STATE_KEY, DefaultPhoneNotifier.convertDataState(state).toString());
553 if (!isDataConnectivityPossible) {
554 intent.putExtra(Phone.NETWORK_UNAVAILABLE_KEY, true);
555 }
556 if (reason != null) {
557 intent.putExtra(Phone.STATE_CHANGE_REASON_KEY, reason);
558 }
559 intent.putExtra(Phone.DATA_APN_KEY, apn);
Robert Greenwalt75e1d312009-08-19 11:18:53 -0700560 String types = new String("");
561 if (apnTypes.length > 0) {
562 types = apnTypes[0];
563 for (int i = 1; i < apnTypes.length; i++) {
564 types = types+","+apnTypes[i];
565 }
Robert Greenwalt42acef32009-08-12 16:08:25 -0700566 }
567 intent.putExtra(Phone.DATA_APN_TYPES_KEY, types);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800568 intent.putExtra(Phone.DATA_IFACE_NAME_KEY, interfaceName);
Robert Greenwalt8afddad2010-08-20 10:01:55 -0700569 int gatewayAddr = 0;
570 if (gateway != null) {
571 gatewayAddr = NetworkUtils.v4StringToInt(gateway);
572 }
573 intent.putExtra(Phone.DATA_GATEWAY_KEY, gatewayAddr);
574
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800575 mContext.sendStickyBroadcast(intent);
576 }
577
578 private void broadcastDataConnectionFailed(String reason) {
579 Intent intent = new Intent(TelephonyIntents.ACTION_DATA_CONNECTION_FAILED);
Dianne Hackborn1c633fc2009-12-08 19:45:14 -0800580 intent.addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800581 intent.putExtra(Phone.FAILURE_REASON_KEY, reason);
582 mContext.sendStickyBroadcast(intent);
583 }
Wink Savillee9b06d72009-05-18 21:47:50 -0700584
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700585 private boolean checkNotifyPermission(String method) {
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700586 if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.MODIFY_PHONE_STATE)
587 == PackageManager.PERMISSION_GRANTED) {
588 return true;
589 }
590 String msg = "Modify Phone State Permission Denial: " + method + " from pid="
Wink Savillee9b06d72009-05-18 21:47:50 -0700591 + Binder.getCallingPid() + ", uid=" + Binder.getCallingUid();
Joe Onorato8a9b2202010-02-26 18:56:32 -0800592 Slog.w(TAG, msg);
The Android Open Source Projectba87e3e2009-03-13 13:04:22 -0700593 return false;
594 }
Jaikumar Ganesh45515652009-04-23 15:20:21 -0700595
596 private void checkListenerPermission(int events) {
597 if ((events & PhoneStateListener.LISTEN_CELL_LOCATION) != 0) {
598 mContext.enforceCallingOrSelfPermission(
599 android.Manifest.permission.ACCESS_COARSE_LOCATION, null);
600
601 }
602
603 if ((events & PHONE_STATE_PERMISSION_MASK) != 0) {
604 mContext.enforceCallingOrSelfPermission(
605 android.Manifest.permission.READ_PHONE_STATE, null);
606 }
607 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800608}