1 /*
2  * Copyright 2017 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 com.android.server.telecom;
18 
19 import android.annotation.Nullable;
20 import android.content.ComponentName;
21 import android.os.Handler;
22 import android.os.HandlerThread;
23 import android.os.Looper;
24 import android.os.Message;
25 import android.telecom.Log;
26 import android.telecom.Logging.Session;
27 import android.text.TextUtils;
28 
29 import com.android.internal.annotations.VisibleForTesting;
30 
31 import java.util.ArrayList;
32 import java.util.List;
33 import java.util.Objects;
34 import java.util.Optional;
35 import java.util.concurrent.BlockingQueue;
36 import java.util.concurrent.LinkedBlockingQueue;
37 import java.util.concurrent.TimeUnit;
38 import java.util.stream.Collectors;
39 
40 public class ConnectionServiceFocusManager {
41     private static final String TAG = "ConnectionSvrFocusMgr";
42     private static final int GET_CURRENT_FOCUS_TIMEOUT_MILLIS = 1000;
43 
44     /** Factory interface used to create the {@link ConnectionServiceFocusManager} instance. */
45     public interface ConnectionServiceFocusManagerFactory {
46         ConnectionServiceFocusManager create(CallsManagerRequester requester);
47     }
48 
49     /**
50      * Interface used by ConnectionServiceFocusManager to communicate with
51      * {@link ConnectionServiceWrapper}.
52      */
53     public interface ConnectionServiceFocus {
54         /**
55          * Notifies the {@link android.telecom.ConnectionService} that it has lose the connection
56          * service focus. It should release all call resource i.e camera, audio once it lost the
57          * focus.
58          */
59         void connectionServiceFocusLost();
60 
61         /**
62          * Notifies the {@link android.telecom.ConnectionService} that it has gain the connection
63          * service focus. It can request the call resource i.e camera, audio as they expected to be
64          * free at the moment.
65          */
66         void connectionServiceFocusGained();
67 
68         /**
69          * Sets the ConnectionServiceFocusListener.
70          *
71          * @see {@link ConnectionServiceFocusListener}.
72          */
73         void setConnectionServiceFocusListener(ConnectionServiceFocusListener listener);
74 
75         /**
76          * Get the {@link ComponentName} of the ConnectionService for logging purposes.
77          * @return the {@link ComponentName}.
78          */
79         ComponentName getComponentName();
80     }
81 
82     /**
83      * Interface used to receive the changed of {@link android.telecom.ConnectionService} that
84      * ConnectionServiceFocusManager cares about.
85      */
86     public interface ConnectionServiceFocusListener {
87         /**
88          * Calls when {@link android.telecom.ConnectionService} has released the call resource. This
89          * usually happen after the {@link android.telecom.ConnectionService} lost the focus.
90          *
91          * @param connectionServiceFocus the {@link android.telecom.ConnectionService} that released
92          * the call resources.
93          */
94         void onConnectionServiceReleased(ConnectionServiceFocus connectionServiceFocus);
95 
96         /**
97          * Calls when {@link android.telecom.ConnectionService} is disconnected.
98          *
99          * @param connectionServiceFocus the {@link android.telecom.ConnectionService} which is
100          * disconnected.
101          */
102         void onConnectionServiceDeath(ConnectionServiceFocus connectionServiceFocus);
103     }
104 
105     /**
106      * Interface define to expose few information of {@link Call} that ConnectionServiceFocusManager
107      * cares about.
108      */
109     public interface CallFocus {
110         /**
111          * Returns the ConnectionService associated with the call.
112          */
113         ConnectionServiceFocus getConnectionServiceWrapper();
114 
115         /**
116          * Returns the state of the call.
117          *
118          * @see {@link CallState}
119          */
120         int getState();
121 
122         /**
123          * @return {@code True} if this call can receive focus, {@code false} otherwise.
124          */
125         boolean isFocusable();
126     }
127 
128     /** Interface define a call back for focus request event. */
129     public interface RequestFocusCallback {
130         /**
131          * Invokes after the focus request is done.
132          *
133          * @param call the call associated with the focus request.
134          */
135         void onRequestFocusDone(CallFocus call);
136     }
137 
138     /**
139      * Interface define to allow the ConnectionServiceFocusManager to communicate with
140      * {@link CallsManager}.
141      */
142     public interface CallsManagerRequester {
143         /**
144          * Requests {@link CallsManager} to disconnect a {@link ConnectionServiceFocus}. This
145          * usually happen when the connection service doesn't respond to focus lost event.
146          */
147         void releaseConnectionService(ConnectionServiceFocus connectionService);
148 
149         /**
150          * Sets the {@link com.android.server.telecom.CallsManager.CallsManagerListener} to listen
151          * the call event that ConnectionServiceFocusManager cares about.
152          */
153         void setCallsManagerListener(CallsManager.CallsManagerListener listener);
154     }
155 
156     private static final int[] PRIORITY_FOCUS_CALL_STATE = new int[] {
157             CallState.ACTIVE, CallState.CONNECTING, CallState.DIALING, CallState.AUDIO_PROCESSING
158     };
159 
160     private static final int MSG_REQUEST_FOCUS = 1;
161     private static final int MSG_RELEASE_CONNECTION_FOCUS = 2;
162     private static final int MSG_RELEASE_FOCUS_TIMEOUT = 3;
163     private static final int MSG_CONNECTION_SERVICE_DEATH = 4;
164     private static final int MSG_ADD_CALL = 5;
165     private static final int MSG_REMOVE_CALL = 6;
166     private static final int MSG_CALL_STATE_CHANGED = 7;
167 
168     @VisibleForTesting
169     public static final int RELEASE_FOCUS_TIMEOUT_MS = 5000;
170 
171     private final List<CallFocus> mCalls;
172 
173     private final CallsManagerListenerBase mCallsManagerListener =
174             new CallsManagerListenerBase() {
175                 @Override
176                 public void onCallAdded(Call call) {
177                     if (callShouldBeIgnored(call)) {
178                         return;
179                     }
180 
181                     mEventHandler
182                             .obtainMessage(MSG_ADD_CALL,
183                                     new MessageArgs(
184                                             Log.createSubsession(),
185                                             "CSFM.oCA",
186                                             call))
187                             .sendToTarget();
188                 }
189 
190                 @Override
191                 public void onCallRemoved(Call call) {
192                     if (callShouldBeIgnored(call)) {
193                         return;
194                     }
195 
196                     mEventHandler
197                             .obtainMessage(MSG_REMOVE_CALL,
198                                     new MessageArgs(
199                                             Log.createSubsession(),
200                                             "CSFM.oCR",
201                                             call))
202                             .sendToTarget();
203                 }
204 
205                 @Override
206                 public void onCallStateChanged(Call call, int oldState, int newState) {
207                     if (callShouldBeIgnored(call)) {
208                         return;
209                     }
210 
211                     mEventHandler
212                             .obtainMessage(MSG_CALL_STATE_CHANGED, oldState, newState,
213                                     new MessageArgs(
214                                             Log.createSubsession(),
215                                             "CSFM.oCSS",
216                                             call))
217                             .sendToTarget();
218                 }
219 
220                 @Override
221                 public void onExternalCallChanged(Call call, boolean isExternalCall) {
222                     if (isExternalCall) {
223                         mEventHandler
224                                 .obtainMessage(MSG_REMOVE_CALL,
225                                         new MessageArgs(
226                                                 Log.createSubsession(),
227                                                 "CSFM.oECC",
228                                                 call))
229                                 .sendToTarget();
230                     } else {
231                         mEventHandler
232                                 .obtainMessage(MSG_ADD_CALL,
233                                         new MessageArgs(
234                                                 Log.createSubsession(),
235                                                 "CSFM.oECC",
236                                                 call))
237                                 .sendToTarget();
238                     }
239                 }
240 
241                 boolean callShouldBeIgnored(Call call) {
242                     return call.isExternalCall();
243                 }
244             };
245 
246     private final ConnectionServiceFocusListener mConnectionServiceFocusListener =
247             new ConnectionServiceFocusListener() {
248                 @Override
249                 public void onConnectionServiceReleased(
250                         ConnectionServiceFocus connectionServiceFocus) {
251                     mEventHandler
252                             .obtainMessage(MSG_RELEASE_CONNECTION_FOCUS,
253                                     new MessageArgs(
254                                             Log.createSubsession(),
255                                             "CSFM.oCSR",
256                                             connectionServiceFocus))
257                             .sendToTarget();
258                 }
259 
260                 @Override
261                 public void onConnectionServiceDeath(
262                         ConnectionServiceFocus connectionServiceFocus) {
263                     mEventHandler
264                             .obtainMessage(MSG_CONNECTION_SERVICE_DEATH,
265                                     new MessageArgs(
266                                             Log.createSubsession(),
267                                             "CSFM.oCSD",
268                                             connectionServiceFocus))
269                             .sendToTarget();
270                 }
271             };
272 
273     private ConnectionServiceFocus mCurrentFocus;
274     private CallFocus mCurrentFocusCall;
275     private CallsManagerRequester mCallsManagerRequester;
276     private FocusRequest mCurrentFocusRequest;
277     private FocusManagerHandler mEventHandler;
278 
279     public ConnectionServiceFocusManager(
280             CallsManagerRequester callsManagerRequester) {
281         mCallsManagerRequester = callsManagerRequester;
282         mCallsManagerRequester.setCallsManagerListener(mCallsManagerListener);
283         HandlerThread handlerThread = new HandlerThread(TAG);
284         handlerThread.start();
285         mEventHandler = new FocusManagerHandler(handlerThread.getLooper());
286         mCalls = new ArrayList<>();
287     }
288 
289     /**
290      * Requests the call focus for the given call. The {@code callback} will be invoked once
291      * the request is done.
292      * @param focus the call need to be focus.
293      * @param callback the callback associated with this request.
294      */
295     public void requestFocus(CallFocus focus, RequestFocusCallback callback) {
296         mEventHandler.obtainMessage(MSG_REQUEST_FOCUS,
297                 new MessageArgs(
298                         Log.createSubsession(),
299                         "CSFM.rF",
300                         new FocusRequest(focus, callback)))
301                 .sendToTarget();
302     }
303 
304     /**
305      * Returns the current focus call. The {@link android.telecom.ConnectionService} of the focus
306      * call is the current connection service focus. Also the state of the focus call must be one
307      * of {@link #PRIORITY_FOCUS_CALL_STATE}.
308      */
309     public @Nullable CallFocus getCurrentFocusCall() {
310         if (mEventHandler.getLooper().isCurrentThread()) {
311             // return synchronously if we're on the same thread.
312             return mCurrentFocusCall;
313         }
314         final BlockingQueue<Optional<CallFocus>> currentFocusedCallQueue =
315                 new LinkedBlockingQueue<>(1);
316         mEventHandler.post(() -> {
317             currentFocusedCallQueue.offer(
318                     mCurrentFocusCall == null ? Optional.empty() : Optional.of(mCurrentFocusCall));
319         });
320         try {
321             Optional<CallFocus> syncCallFocus = currentFocusedCallQueue.poll(
322                     GET_CURRENT_FOCUS_TIMEOUT_MILLIS, TimeUnit.MILLISECONDS);
323             if (syncCallFocus != null) {
324                 return syncCallFocus.orElse(null);
325             } else {
326                 Log.w(TAG, "Timed out waiting for synchronous current focus. Returning possibly"
327                         + " inaccurate result");
328                 return mCurrentFocusCall;
329             }
330         } catch (InterruptedException e) {
331             Log.w(TAG, "Interrupted when waiting for synchronous current focus."
332                     + " Returning possibly inaccurate result.");
333             return mCurrentFocusCall;
334         }
335     }
336 
337     /** Returns the current connection service focus. */
338     public ConnectionServiceFocus getCurrentFocusConnectionService() {
339         return mCurrentFocus;
340     }
341 
342     @VisibleForTesting
343     public Handler getHandler() {
344         return mEventHandler;
345     }
346 
347     @VisibleForTesting
348     public List<CallFocus> getAllCall() { return mCalls; }
349 
350     private void updateConnectionServiceFocus(ConnectionServiceFocus connSvrFocus) {
351         if (!Objects.equals(mCurrentFocus, connSvrFocus)) {
352             if (connSvrFocus != null) {
353                 connSvrFocus.setConnectionServiceFocusListener(mConnectionServiceFocusListener);
354                 connSvrFocus.connectionServiceFocusGained();
355             }
356             mCurrentFocus = connSvrFocus;
357             Log.d(this, "updateConnectionServiceFocus connSvr = %s", connSvrFocus);
358         }
359     }
360 
361     private void updateCurrentFocusCall() {
362         mCurrentFocusCall = null;
363 
364         if (mCurrentFocus == null) {
365             return;
366         }
367 
368         List<CallFocus> calls = mCalls
369                 .stream()
370                 .filter(call -> mCurrentFocus.equals(call.getConnectionServiceWrapper())
371                         && call.isFocusable())
372                 .collect(Collectors.toList());
373 
374         for (int i = 0; i < PRIORITY_FOCUS_CALL_STATE.length; i++) {
375             for (CallFocus call : calls) {
376                 if (call.getState() == PRIORITY_FOCUS_CALL_STATE[i]) {
377                     mCurrentFocusCall = call;
378                     Log.d(this, "updateCurrentFocusCall %s", mCurrentFocusCall);
379                     return;
380                 }
381             }
382         }
383 
384         Log.d(this, "updateCurrentFocusCall = null");
385     }
386 
387     private void onRequestFocusDone(FocusRequest focusRequest) {
388         if (focusRequest.callback != null) {
389             focusRequest.callback.onRequestFocusDone(focusRequest.call);
390         }
391     }
392 
393     private void handleRequestFocus(FocusRequest focusRequest) {
394         Log.i(this, "handleRequestFocus req = %s", focusRequest);
395         if (mCurrentFocus == null
396                 || mCurrentFocus.equals(focusRequest.call.getConnectionServiceWrapper())) {
397             updateConnectionServiceFocus(focusRequest.call.getConnectionServiceWrapper());
398             updateCurrentFocusCall();
399             onRequestFocusDone(focusRequest);
400         } else {
401             mCurrentFocus.connectionServiceFocusLost();
402             mCurrentFocusRequest = focusRequest;
403             Message msg = mEventHandler.obtainMessage(
404                     MSG_RELEASE_FOCUS_TIMEOUT,
405                     new MessageArgs(
406                             Log.createSubsession(),
407                             "CSFM.hRF",
408                             focusRequest));
409             mEventHandler.sendMessageDelayed(msg, RELEASE_FOCUS_TIMEOUT_MS);
410         }
411     }
412 
413     private void handleReleasedFocus(ConnectionServiceFocus connectionServiceFocus) {
414         Log.d(this, "handleReleasedFocus connSvr = %s", connectionServiceFocus);
415         // The ConnectionService can call onConnectionServiceFocusReleased even if it's not the
416         // current focus connection service, nothing will be changed in this case.
417         if (Objects.equals(mCurrentFocus, connectionServiceFocus)) {
418             mEventHandler.removeMessages(MSG_RELEASE_FOCUS_TIMEOUT);
419             ConnectionServiceFocus newCSF = null;
420             if (mCurrentFocusRequest != null) {
421                 newCSF = mCurrentFocusRequest.call.getConnectionServiceWrapper();
422             }
423             updateConnectionServiceFocus(newCSF);
424             updateCurrentFocusCall();
425             if (mCurrentFocusRequest != null) {
426                 onRequestFocusDone(mCurrentFocusRequest);
427                 mCurrentFocusRequest = null;
428             }
429         }
430     }
431 
432     private void handleReleasedFocusTimeout(FocusRequest focusRequest) {
433         Log.d(this, "handleReleasedFocusTimeout req = %s", focusRequest);
434         mCallsManagerRequester.releaseConnectionService(mCurrentFocus);
435         updateConnectionServiceFocus(focusRequest.call.getConnectionServiceWrapper());
436         updateCurrentFocusCall();
437         onRequestFocusDone(focusRequest);
438         mCurrentFocusRequest = null;
439     }
440 
441     private void handleConnectionServiceDeath(ConnectionServiceFocus connectionServiceFocus) {
442         Log.d(this, "handleConnectionServiceDeath %s", connectionServiceFocus);
443         if (Objects.equals(connectionServiceFocus, mCurrentFocus)) {
444             updateConnectionServiceFocus(null);
445             updateCurrentFocusCall();
446         }
447     }
448 
449     private void handleAddedCall(CallFocus call) {
450         Log.d(this, "handleAddedCall %s", call);
451         if (!mCalls.contains(call)) {
452             mCalls.add(call);
453         }
454         if (Objects.equals(mCurrentFocus, call.getConnectionServiceWrapper())) {
455             updateCurrentFocusCall();
456         }
457     }
458 
459     private void handleRemovedCall(CallFocus call) {
460         Log.d(this, "handleRemovedCall %s", call);
461         mCalls.remove(call);
462         if (call.equals(mCurrentFocusCall)) {
463             updateCurrentFocusCall();
464         }
465     }
466 
467     private void handleCallStateChanged(CallFocus call, int oldState, int newState) {
468         Log.d(this,
469                 "handleCallStateChanged %s, oldState = %d, newState = %d",
470                 call,
471                 oldState,
472                 newState);
473         if (mCalls.contains(call)
474                 && Objects.equals(mCurrentFocus, call.getConnectionServiceWrapper())) {
475             updateCurrentFocusCall();
476         }
477     }
478 
479     private final class FocusManagerHandler extends Handler {
480         FocusManagerHandler(Looper looper) {
481             super(looper);
482         }
483 
484         @Override
485         public void handleMessage(Message msg) {
486             Session session = ((MessageArgs) msg.obj).logSession;
487             String shortName = ((MessageArgs) msg.obj).shortName;
488             if (TextUtils.isEmpty(shortName)) {
489                 shortName = "hM";
490             }
491             Log.continueSession(session, shortName);
492             Object msgObj = ((MessageArgs) msg.obj).obj;
493 
494             try {
495                 switch (msg.what) {
496                     case MSG_REQUEST_FOCUS:
497                         handleRequestFocus((FocusRequest) msgObj);
498                         break;
499                     case MSG_RELEASE_CONNECTION_FOCUS:
500                         handleReleasedFocus((ConnectionServiceFocus) msgObj);
501                         break;
502                     case MSG_RELEASE_FOCUS_TIMEOUT:
503                         handleReleasedFocusTimeout((FocusRequest) msgObj);
504                         break;
505                     case MSG_CONNECTION_SERVICE_DEATH:
506                         handleConnectionServiceDeath((ConnectionServiceFocus) msgObj);
507                         break;
508                     case MSG_ADD_CALL:
509                         handleAddedCall((CallFocus) msgObj);
510                         break;
511                     case MSG_REMOVE_CALL:
512                         handleRemovedCall((CallFocus) msgObj);
513                         break;
514                     case MSG_CALL_STATE_CHANGED:
515                         handleCallStateChanged((CallFocus) msgObj, msg.arg1, msg.arg2);
516                         break;
517                 }
518             } finally {
519                 Log.endSession();
520             }
521         }
522     }
523 
524     private static final class FocusRequest {
525         CallFocus call;
526         @Nullable RequestFocusCallback callback;
527 
528         FocusRequest(CallFocus call, RequestFocusCallback callback) {
529             this.call = call;
530             this.callback = callback;
531         }
532     }
533 
534     private static final class MessageArgs {
535         Session logSession;
536         String shortName;
537         Object obj;
538 
539         MessageArgs(Session logSession, String shortName, Object obj) {
540             this.logSession = logSession;
541             this.shortName = shortName;
542             this.obj = obj;
543         }
544     }
545 }
546