1 /* 2 * Copyright (C) 2014 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 17 package android.telecom; 18 19 import android.net.Uri; 20 import android.os.IBinder.DeathRecipient; 21 import android.os.RemoteException; 22 23 import com.android.internal.telecom.IConnectionServiceAdapter; 24 import com.android.internal.telecom.RemoteServiceCallback; 25 26 import java.util.Collections; 27 import java.util.Iterator; 28 import java.util.List; 29 import java.util.Set; 30 import java.util.concurrent.ConcurrentHashMap; 31 32 /** 33 * Provides methods for IConnectionService implementations to interact with the system phone app. 34 * 35 * @hide 36 */ 37 final class ConnectionServiceAdapter implements DeathRecipient { 38 /** 39 * ConcurrentHashMap constructor params: 8 is initial table size, 0.9f is 40 * load factor before resizing, 1 means we only expect a single thread to 41 * access the map so make only a single shard 42 */ 43 private final Set<IConnectionServiceAdapter> mAdapters = Collections.newSetFromMap( 44 new ConcurrentHashMap<IConnectionServiceAdapter, Boolean>(8, 0.9f, 1)); 45 ConnectionServiceAdapter()46 ConnectionServiceAdapter() { 47 } 48 addAdapter(IConnectionServiceAdapter adapter)49 void addAdapter(IConnectionServiceAdapter adapter) { 50 if (mAdapters.add(adapter)) { 51 try { 52 adapter.asBinder().linkToDeath(this, 0); 53 } catch (RemoteException e) { 54 mAdapters.remove(adapter); 55 } 56 } 57 } 58 removeAdapter(IConnectionServiceAdapter adapter)59 void removeAdapter(IConnectionServiceAdapter adapter) { 60 if (adapter != null && mAdapters.remove(adapter)) { 61 adapter.asBinder().unlinkToDeath(this, 0); 62 } 63 } 64 65 /** ${inheritDoc} */ 66 @Override binderDied()67 public void binderDied() { 68 Iterator<IConnectionServiceAdapter> it = mAdapters.iterator(); 69 while (it.hasNext()) { 70 IConnectionServiceAdapter adapter = it.next(); 71 if (!adapter.asBinder().isBinderAlive()) { 72 it.remove(); 73 adapter.asBinder().unlinkToDeath(this, 0); 74 } 75 } 76 } 77 handleCreateConnectionComplete( String id, ConnectionRequest request, ParcelableConnection connection)78 void handleCreateConnectionComplete( 79 String id, 80 ConnectionRequest request, 81 ParcelableConnection connection) { 82 for (IConnectionServiceAdapter adapter : mAdapters) { 83 try { 84 adapter.handleCreateConnectionComplete(id, request, connection); 85 } catch (RemoteException e) { 86 } 87 } 88 } 89 90 /** 91 * Sets a call's state to active (e.g., an ongoing call where two parties can actively 92 * communicate). 93 * 94 * @param callId The unique ID of the call whose state is changing to active. 95 */ setActive(String callId)96 void setActive(String callId) { 97 for (IConnectionServiceAdapter adapter : mAdapters) { 98 try { 99 adapter.setActive(callId); 100 } catch (RemoteException e) { 101 } 102 } 103 } 104 105 /** 106 * Sets a call's state to ringing (e.g., an inbound ringing call). 107 * 108 * @param callId The unique ID of the call whose state is changing to ringing. 109 */ setRinging(String callId)110 void setRinging(String callId) { 111 for (IConnectionServiceAdapter adapter : mAdapters) { 112 try { 113 adapter.setRinging(callId); 114 } catch (RemoteException e) { 115 } 116 } 117 } 118 119 /** 120 * Sets a call's state to dialing (e.g., dialing an outbound call). 121 * 122 * @param callId The unique ID of the call whose state is changing to dialing. 123 */ setDialing(String callId)124 void setDialing(String callId) { 125 for (IConnectionServiceAdapter adapter : mAdapters) { 126 try { 127 adapter.setDialing(callId); 128 } catch (RemoteException e) { 129 } 130 } 131 } 132 133 /** 134 * Sets a call's state to disconnected. 135 * 136 * @param callId The unique ID of the call whose state is changing to disconnected. 137 * @param disconnectCause The reason for the disconnection, as described by 138 * {@link android.telecomm.DisconnectCause}. 139 */ setDisconnected(String callId, DisconnectCause disconnectCause)140 void setDisconnected(String callId, DisconnectCause disconnectCause) { 141 for (IConnectionServiceAdapter adapter : mAdapters) { 142 try { 143 adapter.setDisconnected(callId, disconnectCause); 144 } catch (RemoteException e) { 145 } 146 } 147 } 148 149 /** 150 * Sets a call's state to be on hold. 151 * 152 * @param callId - The unique ID of the call whose state is changing to be on hold. 153 */ setOnHold(String callId)154 void setOnHold(String callId) { 155 for (IConnectionServiceAdapter adapter : mAdapters) { 156 try { 157 adapter.setOnHold(callId); 158 } catch (RemoteException e) { 159 } 160 } 161 } 162 163 /** 164 * Asks Telecom to start or stop a ringback tone for a call. 165 * 166 * @param callId The unique ID of the call whose ringback is being changed. 167 * @param ringback Whether Telecom should start playing a ringback tone. 168 */ setRingbackRequested(String callId, boolean ringback)169 void setRingbackRequested(String callId, boolean ringback) { 170 for (IConnectionServiceAdapter adapter : mAdapters) { 171 try { 172 adapter.setRingbackRequested(callId, ringback); 173 } catch (RemoteException e) { 174 } 175 } 176 } 177 setCallCapabilities(String callId, int capabilities)178 void setCallCapabilities(String callId, int capabilities) { 179 for (IConnectionServiceAdapter adapter : mAdapters) { 180 try { 181 adapter.setCallCapabilities(callId, capabilities); 182 } catch (RemoteException ignored) { 183 } 184 } 185 } 186 187 /** 188 * Indicates whether or not the specified call is currently conferenced into the specified 189 * conference call. 190 * 191 * @param callId The unique ID of the call being conferenced. 192 * @param conferenceCallId The unique ID of the conference call. Null if call is not 193 * conferenced. 194 */ setIsConferenced(String callId, String conferenceCallId)195 void setIsConferenced(String callId, String conferenceCallId) { 196 for (IConnectionServiceAdapter adapter : mAdapters) { 197 try { 198 Log.d(this, "sending connection %s with conference %s", callId, conferenceCallId); 199 adapter.setIsConferenced(callId, conferenceCallId); 200 } catch (RemoteException ignored) { 201 } 202 } 203 } 204 205 /** 206 * Indicates that the call no longer exists. Can be used with either a call or a conference 207 * call. 208 * 209 * @param callId The unique ID of the call. 210 */ removeCall(String callId)211 void removeCall(String callId) { 212 for (IConnectionServiceAdapter adapter : mAdapters) { 213 try { 214 adapter.removeCall(callId); 215 } catch (RemoteException ignored) { 216 } 217 } 218 } 219 onPostDialWait(String callId, String remaining)220 void onPostDialWait(String callId, String remaining) { 221 for (IConnectionServiceAdapter adapter : mAdapters) { 222 try { 223 adapter.onPostDialWait(callId, remaining); 224 } catch (RemoteException ignored) { 225 } 226 } 227 } 228 229 /** 230 * Indicates that a new conference call has been created. 231 * 232 * @param callId The unique ID of the conference call. 233 */ addConferenceCall(String callId, ParcelableConference parcelableConference)234 void addConferenceCall(String callId, ParcelableConference parcelableConference) { 235 for (IConnectionServiceAdapter adapter : mAdapters) { 236 try { 237 adapter.addConferenceCall(callId, parcelableConference); 238 } catch (RemoteException ignored) { 239 } 240 } 241 } 242 243 /** 244 * Retrieves a list of remote connection services usable to place calls. 245 */ queryRemoteConnectionServices(RemoteServiceCallback callback)246 void queryRemoteConnectionServices(RemoteServiceCallback callback) { 247 // Only supported when there is only one adapter. 248 if (mAdapters.size() == 1) { 249 try { 250 mAdapters.iterator().next().queryRemoteConnectionServices(callback); 251 } catch (RemoteException e) { 252 Log.e(this, e, "Exception trying to query for remote CSs"); 253 } 254 } 255 } 256 257 /** 258 * Sets the call video provider for a call. 259 * 260 * @param callId The unique ID of the call to set with the given call video provider. 261 * @param videoProvider The call video provider instance to set on the call. 262 */ setVideoProvider( String callId, Connection.VideoProvider videoProvider)263 void setVideoProvider( 264 String callId, Connection.VideoProvider videoProvider) { 265 for (IConnectionServiceAdapter adapter : mAdapters) { 266 try { 267 adapter.setVideoProvider( 268 callId, 269 videoProvider == null ? null : videoProvider.getInterface()); 270 } catch (RemoteException e) { 271 } 272 } 273 } 274 275 /** 276 * Requests that the framework use VOIP audio mode for this connection. 277 * 278 * @param callId The unique ID of the call to set with the given call video provider. 279 * @param isVoip True if the audio mode is VOIP. 280 */ setIsVoipAudioMode(String callId, boolean isVoip)281 void setIsVoipAudioMode(String callId, boolean isVoip) { 282 for (IConnectionServiceAdapter adapter : mAdapters) { 283 try { 284 adapter.setIsVoipAudioMode(callId, isVoip); 285 } catch (RemoteException e) { 286 } 287 } 288 } 289 setStatusHints(String callId, StatusHints statusHints)290 void setStatusHints(String callId, StatusHints statusHints) { 291 for (IConnectionServiceAdapter adapter : mAdapters) { 292 try { 293 adapter.setStatusHints(callId, statusHints); 294 } catch (RemoteException e) { 295 } 296 } 297 } 298 setAddress(String callId, Uri address, int presentation)299 void setAddress(String callId, Uri address, int presentation) { 300 for (IConnectionServiceAdapter adapter : mAdapters) { 301 try { 302 adapter.setAddress(callId, address, presentation); 303 } catch (RemoteException e) { 304 } 305 } 306 } 307 setCallerDisplayName(String callId, String callerDisplayName, int presentation)308 void setCallerDisplayName(String callId, String callerDisplayName, int presentation) { 309 for (IConnectionServiceAdapter adapter : mAdapters) { 310 try { 311 adapter.setCallerDisplayName(callId, callerDisplayName, presentation); 312 } catch (RemoteException e) { 313 } 314 } 315 } 316 317 /** 318 * Sets the video state associated with a call. 319 * 320 * Valid values: {@link VideoProfile.VideoState#AUDIO_ONLY}, 321 * {@link VideoProfile.VideoState#BIDIRECTIONAL}, 322 * {@link VideoProfile.VideoState#TX_ENABLED}, 323 * {@link VideoProfile.VideoState#RX_ENABLED}. 324 * 325 * @param callId The unique ID of the call to set the video state for. 326 * @param videoState The video state. 327 */ setVideoState(String callId, int videoState)328 void setVideoState(String callId, int videoState) { 329 Log.v(this, "setVideoState: %d", videoState); 330 for (IConnectionServiceAdapter adapter : mAdapters) { 331 try { 332 adapter.setVideoState(callId, videoState); 333 } catch (RemoteException ignored) { 334 } 335 } 336 } 337 setConferenceableConnections(String callId, List<String> conferenceableCallIds)338 void setConferenceableConnections(String callId, List<String> conferenceableCallIds) { 339 Log.v(this, "setConferenceableConnections: %s, %s", callId, conferenceableCallIds); 340 for (IConnectionServiceAdapter adapter : mAdapters) { 341 try { 342 adapter.setConferenceableConnections(callId, conferenceableCallIds); 343 } catch (RemoteException ignored) { 344 } 345 } 346 } 347 } 348