1 /*
2  * Copyright (C) 2021 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.Manifest;
20 import android.annotation.NonNull;
21 import android.annotation.Nullable;
22 import android.annotation.RequiresPermission;
23 import android.annotation.UserIdInt;
24 import android.content.ComponentName;
25 import android.content.Context;
26 import android.content.Intent;
27 import android.content.ServiceConnection;
28 import android.content.pm.PackageManager;
29 import android.content.pm.ResolveInfo;
30 import android.os.Bundle;
31 import android.os.IBinder;
32 import android.os.RemoteException;
33 import android.os.UserHandle;
34 import android.telecom.BluetoothCallQualityReport;
35 import android.telecom.CallAudioState;
36 import android.telecom.CallDiagnosticService;
37 import android.telecom.ConnectionService;
38 import android.telecom.CallDiagnostics;
39 import android.telecom.DisconnectCause;
40 import android.telecom.InCallService;
41 import android.telecom.Log;
42 import android.telecom.ParcelableCall;
43 import android.telephony.CallQuality;
44 import android.telephony.ims.ImsReasonInfo;
45 import android.text.TextUtils;
46 
47 import com.android.internal.telecom.ICallDiagnosticService;
48 import com.android.internal.util.IndentingPrintWriter;
49 
50 import java.util.List;
51 
52 /**
53  * Responsible for maintaining binding to the {@link CallDiagnosticService} defined by the
54  * {@code call_diagnostic_service_package_name} key in the
55  * {@code packages/services/Telecomm/res/values/config.xml} file.
56  */
57 public class CallDiagnosticServiceController extends CallsManagerListenerBase {
58     /**
59      * Context dependencies for the {@link CallDiagnosticServiceController}.
60      */
61     public interface ContextProxy {
62         List<ResolveInfo> queryIntentServicesAsUser(@NonNull Intent intent,
63                 @PackageManager.ResolveInfoFlags int flags, @UserIdInt int userId);
64         boolean bindServiceAsUser(@NonNull @RequiresPermission Intent service,
65                 @NonNull ServiceConnection conn, int flags, @NonNull UserHandle user);
66         void unbindService(@NonNull ServiceConnection conn);
67         UserHandle getCurrentUserHandle();
68     }
69 
70     /**
71      * Listener for {@link Call} events; used to propagate these changes to the
72      * {@link CallDiagnosticService}.
73      */
74     private final Call.Listener mCallListener = new Call.ListenerBase() {
75         @Override
76         public void onConnectionCapabilitiesChanged(Call call) {
77             updateCall(call);
78         }
79 
80         @Override
81         public void onConnectionPropertiesChanged(Call call, boolean didRttChange) {
82             updateCall(call);
83         }
84 
85         /**
86          * Listens for changes to extras reported by a Telecom {@link Call}.
87          *
88          * Extras changes can originate from a {@link ConnectionService} or an {@link InCallService}
89          * so we will only trigger an update of the call information if the source of the extras
90          * change was a {@link ConnectionService}.
91          *
92          * @param call The call.
93          * @param source The source of the extras change ({@link Call#SOURCE_CONNECTION_SERVICE} or
94          *               {@link Call#SOURCE_INCALL_SERVICE}).
95          * @param extras The extras.
96          */
97         @Override
98         public void onExtrasChanged(Call call, int source, Bundle extras) {
99             // Do not inform InCallServices of changes which originated there.
100             if (source == Call.SOURCE_INCALL_SERVICE) {
101                 return;
102             }
103             updateCall(call);
104         }
105 
106         /**
107          * Listens for changes to extras reported by a Telecom {@link Call}.
108          *
109          * Extras changes can originate from a {@link ConnectionService} or an {@link InCallService}
110          * so we will only trigger an update of the call information if the source of the extras
111          * change was a {@link ConnectionService}.
112          *  @param call The call.
113          * @param source The source of the extras change ({@link Call#SOURCE_CONNECTION_SERVICE} or
114          *               {@link Call#SOURCE_INCALL_SERVICE}).
115          * @param keys The extra key removed
116          */
117         @Override
118         public void onExtrasRemoved(Call call, int source, List<String> keys) {
119             // Do not inform InCallServices of changes which originated there.
120             if (source == Call.SOURCE_INCALL_SERVICE) {
121                 return;
122             }
123             updateCall(call);
124         }
125 
126         /**
127          * Handles changes to the video state of a call.
128          * @param call
129          * @param previousVideoState
130          * @param newVideoState
131          */
132         @Override
133         public void onVideoStateChanged(Call call, int previousVideoState, int newVideoState) {
134             updateCall(call);
135         }
136 
137         /**
138          * Relays a bluetooth call quality report received from the Bluetooth stack to the
139          * CallDiagnosticService.
140          * @param call The call.
141          * @param report The received report.
142          */
143         @Override
144         public void onBluetoothCallQualityReport(Call call, BluetoothCallQualityReport report) {
145             handleBluetoothCallQualityReport(call, report);
146         }
147 
148         /**
149          * Relays a device to device message received from Telephony to the CallDiagnosticService.
150          * @param call
151          * @param messageType
152          * @param messageValue
153          */
154         @Override
155         public void onReceivedDeviceToDeviceMessage(Call call, int messageType, int messageValue) {
156             handleReceivedDeviceToDeviceMessage(call, messageType, messageValue);
157         }
158 
159         /**
160          * Handles an incoming {@link CallQuality} report from a {@link android.telecom.Connection}.
161          * @param call The call.
162          * @param callQualityReport The call quality report.
163          */
164         @Override
165         public void onReceivedCallQualityReport(Call call, CallQuality callQualityReport) {
166             handleCallQualityReport(call, callQualityReport);
167         }
168     };
169 
170     /**
171      * {@link ServiceConnection} handling changes to binding of the {@link CallDiagnosticService}.
172      */
173     private class CallDiagnosticServiceConnection implements ServiceConnection {
174         @Override
175         public void onServiceConnected(ComponentName name, IBinder service) {
176             Log.startSession("CDSC.oSC", Log.getPackageAbbreviation(name));
177             try {
178                 synchronized (mLock) {
179                     mCallDiagnosticService = ICallDiagnosticService.Stub.asInterface(service);
180 
181                     handleConnectionComplete(mCallDiagnosticService);
182                 }
183                 Log.i(CallDiagnosticServiceController.this, "onServiceConnected: cmp=%s", name);
184             } finally {
185                 Log.endSession();
186             }
187         }
188 
189         @Override
190         public void onServiceDisconnected(ComponentName name) {
191             Log.startSession("CDSC.oSD", Log.getPackageAbbreviation(name));
192             try {
193                 synchronized (mLock) {
194                     mCallDiagnosticService = null;
195                     mConnection = null;
196                 }
197                 Log.i(CallDiagnosticServiceController.this, "onServiceDisconnected: cmp=%s", name);
198             } finally {
199                 Log.endSession();
200             }
201         }
202 
203         @Override
204         public void onBindingDied(ComponentName name) {
205             Log.startSession("CDSC.oBD", Log.getPackageAbbreviation(name));
206             try {
207                 synchronized (mLock) {
208                     mCallDiagnosticService = null;
209                     mConnection = null;
210                 }
211                 Log.w(CallDiagnosticServiceController.this, "onBindingDied: cmp=%s", name);
212             } finally {
213                 Log.endSession();
214             }
215         }
216 
217         @Override
218         public void onNullBinding(ComponentName name) {
219             Log.startSession("CDSC.oNB", Log.getPackageAbbreviation(name));
220             try {
221                 synchronized (mLock) {
222                     maybeUnbindCallScreeningService();
223                 }
224             } finally {
225                 Log.endSession();
226             }
227         }
228     }
229 
230     private final String mPackageName;
231     private final ContextProxy mContextProxy;
232     private InCallTonePlayer.Factory mPlayerFactory;
233     private String mTestPackageName;
234     private CallDiagnosticServiceConnection mConnection;
235     private CallDiagnosticServiceAdapter mAdapter;
236     private final TelecomSystem.SyncRoot mLock;
237     private ICallDiagnosticService mCallDiagnosticService;
238     private final CallIdMapper mCallIdMapper = new CallIdMapper(Call::getId);
239 
240     public CallDiagnosticServiceController(@NonNull ContextProxy contextProxy,
241             @Nullable String packageName, @NonNull TelecomSystem.SyncRoot lock) {
242         mContextProxy = contextProxy;
243         mPackageName = packageName;
244         mLock = lock;
245     }
246 
247     /**
248      * Sets the current {@link InCallTonePlayer.Factory} for this instance.
249      * @param factory the factory.
250      */
251     public void setInCallTonePlayerFactory(InCallTonePlayer.Factory factory) {
252         mPlayerFactory = factory;
253     }
254 
255     /**
256      * Handles Telecom adding new calls.  Will bind to the call diagnostic service if needed and
257      * send the calls, or send to an already bound service.
258      * @param call The call to add.
259      */
260     @Override
261     public void onCallAdded(@NonNull Call call) {
262         if (!call.isSimCall() || call.isExternalCall()) {
263             Log.i(this, "onCallAdded: skipping call %s as non-sim or external.", call.getId());
264             return;
265         }
266         if (mCallIdMapper.getCallId(call) == null) {
267             mCallIdMapper.addCall(call);
268             call.addListener(mCallListener);
269         }
270         if (isConnected()) {
271             sendCallToBoundService(call, mCallDiagnosticService);
272         } else {
273             maybeBindCallDiagnosticService();
274         }
275     }
276 
277     /**
278      * Handles a newly disconnected call signalled from {@link CallsManager}.
279      * @param call The call
280      * @param disconnectCause The disconnect cause
281      * @return {@code true} if the {@link CallDiagnosticService} was sent the call, {@code false}
282      * if the call was not applicable to the CDS or if there was an issue sending it.
283      */
284     public boolean onCallDisconnected(@NonNull Call call,
285             @NonNull DisconnectCause disconnectCause) {
286         if (!call.isSimCall() || call.isExternalCall()) {
287             Log.i(this, "onCallDisconnected: skipping call %s as non-sim or external.",
288                     call.getId());
289             return false;
290         }
291         String callId = mCallIdMapper.getCallId(call);
292         try {
293             if (isConnected()) {
294                 mCallDiagnosticService.notifyCallDisconnected(callId, disconnectCause);
295                 return true;
296             }
297         } catch (RemoteException e) {
298             Log.w(this, "onCallDisconnected: callId=%s, exception=%s", call.getId(), e);
299         }
300         return false;
301     }
302 
303     /**
304      * Handles Telecom removal of calls; will remove the call from the bound service and if the
305      * number of tracked calls falls to zero, unbind from the service.
306      * @param call The call to remove from the bound CDS.
307      */
308     @Override
309     public void onCallRemoved(@NonNull Call call) {
310         if (!call.isSimCall() || call.isExternalCall()) {
311             Log.i(this, "onCallRemoved: skipping call %s as non-sim or external.", call.getId());
312             return;
313         }
314         mCallIdMapper.removeCall(call);
315         call.removeListener(mCallListener);
316         removeCallFromBoundService(call, mCallDiagnosticService);
317 
318         if (mCallIdMapper.getCalls().size() == 0) {
319             maybeUnbindCallScreeningService();
320         }
321     }
322 
323     @Override
324     public void onCallStateChanged(Call call, int oldState, int newState) {
325         updateCall(call);
326     }
327 
328     @Override
329     public void onCallAudioStateChanged(CallAudioState oldCallAudioState,
330             CallAudioState newCallAudioState) {
331         if (mCallDiagnosticService != null) {
332             try {
333                 mCallDiagnosticService.updateCallAudioState(newCallAudioState);
334             } catch (RemoteException e) {
335                 Log.w(this, "onCallAudioStateChanged: failed %s", e);
336             }
337         }
338     }
339 
340     /**
341      * Sets the test call diagnostic service; used by the telecom command line command to override
342      * the {@link CallDiagnosticService} to bind to for CTS test purposes.
343      * @param packageName The package name to set to.
344      */
345     public void setTestCallDiagnosticService(@Nullable String packageName) {
346         if (TextUtils.isEmpty(packageName)) {
347             mTestPackageName = null;
348         } else {
349             mTestPackageName = packageName;
350         }
351 
352         Log.i(this, "setTestCallDiagnosticService: packageName=%s", packageName);
353     }
354 
355     /**
356      * Determines the active call diagnostic service, taking into account the test override.
357      * @return The package name of the active call diagnostic service.
358      */
359     private @Nullable String getActiveCallDiagnosticService() {
360         if (mTestPackageName != null) {
361             return mTestPackageName;
362         }
363 
364         return mPackageName;
365     }
366 
367     /**
368      * If we are not already bound to the {@link CallDiagnosticService}, attempts to initiate a
369      * binding tho that service.
370      * @return {@code true} if we bound, {@code false} otherwise.
371      */
372     private boolean maybeBindCallDiagnosticService() {
373         if (mConnection != null) {
374             return false;
375         }
376 
377         mConnection = new CallDiagnosticServiceConnection();
378         boolean bound = bindCallDiagnosticService(mContextProxy.getCurrentUserHandle(),
379                 getActiveCallDiagnosticService(), mConnection);
380         if (!bound) {
381             mConnection = null;
382         }
383         return bound;
384     }
385 
386     /**
387      * Performs binding to the {@link CallDiagnosticService}.
388      * @param userHandle user name to bind via.
389      * @param packageName package name of the CDS.
390      * @param serviceConnection The service connection to be notified of bind events.
391      * @return
392      */
393     private boolean bindCallDiagnosticService(UserHandle userHandle,
394             String packageName, CallDiagnosticServiceConnection serviceConnection) {
395 
396         if (TextUtils.isEmpty(packageName)) {
397             Log.i(this, "bindCallDiagnosticService: no package; skip binding.");
398             return false;
399         }
400 
401         Intent intent = new Intent(CallDiagnosticService.SERVICE_INTERFACE)
402                 .setPackage(packageName);
403         Log.i(this, "bindCallDiagnosticService: user %d.", userHandle.getIdentifier());
404         List<ResolveInfo> entries = mContextProxy.queryIntentServicesAsUser(intent, 0,
405                 userHandle.getIdentifier());
406         if (entries.isEmpty()) {
407             Log.i(this, "bindCallDiagnosticService: %s has no service.", packageName);
408             return false;
409         }
410 
411         ResolveInfo entry = entries.get(0);
412         if (entry.serviceInfo == null) {
413             Log.i(this, "bindCallDiagnosticService: %s has no service info.", packageName);
414             return false;
415         }
416 
417         if (entry.serviceInfo.permission == null || !entry.serviceInfo.permission.equals(
418                 Manifest.permission.BIND_CALL_DIAGNOSTIC_SERVICE)) {
419             Log.i(this, "bindCallDiagnosticService: %s doesn't require "
420                     + "BIND_CALL_DIAGNOSTIC_SERVICE.", packageName);
421             return false;
422         }
423 
424         ComponentName componentName =
425                 new ComponentName(entry.serviceInfo.packageName, entry.serviceInfo.name);
426         intent.setComponent(componentName);
427         if (mContextProxy.bindServiceAsUser(
428                 intent,
429                 serviceConnection,
430                 Context.BIND_AUTO_CREATE | Context.BIND_FOREGROUND_SERVICE,
431                 UserHandle.CURRENT)) {
432             Log.d(this, "bindCallDiagnosticService, found service, waiting for it to connect");
433             return true;
434         }
435         return false;
436     }
437 
438     /**
439      * If we are bound to a {@link CallDiagnosticService}, unbind from it.
440      */
441     public void maybeUnbindCallScreeningService() {
442         if (mConnection != null) {
443             Log.i(this, "maybeUnbindCallScreeningService - unbinding from %s",
444                     getActiveCallDiagnosticService());
445             try {
446                 mContextProxy.unbindService(mConnection);
447                 mCallDiagnosticService = null;
448                 mConnection = null;
449             } catch (IllegalArgumentException e) {
450                 Log.i(this, "maybeUnbindCallScreeningService: Exception when unbind %s : %s",
451                         getActiveCallDiagnosticService(), e.getMessage());
452             }
453         } else {
454             Log.w(this, "maybeUnbindCallScreeningService - already unbound");
455         }
456     }
457 
458     /**
459      * Implements the abstracted Telecom functionality the {@link CallDiagnosticServiceAdapter}
460      * depends on.
461      */
462     private CallDiagnosticServiceAdapter.TelecomAdapter mTelecomAdapter =
463             new CallDiagnosticServiceAdapter.TelecomAdapter() {
464 
465         @Override
466         public void displayDiagnosticMessage(String callId, int messageId, CharSequence message) {
467             handleDisplayDiagnosticMessage(callId, messageId, message);
468         }
469 
470         @Override
471         public void clearDiagnosticMessage(String callId, int messageId) {
472             handleClearDiagnosticMessage(callId, messageId);
473         }
474 
475         @Override
476         public void sendDeviceToDeviceMessage(String callId,
477                         @CallDiagnostics.MessageType int message, int value) {
478             handleSendD2DMessage(callId, message, value);
479         }
480 
481         @Override
482         public void overrideDisconnectMessage(String callId, CharSequence message) {
483             handleOverrideDisconnectMessage(callId, message);
484         }
485     };
486 
487     /**
488      * Sends all calls to the specified {@link CallDiagnosticService}.
489      * @param callDiagnosticService the CDS to send calls to.
490      */
491     private void handleConnectionComplete(@NonNull ICallDiagnosticService callDiagnosticService) {
492         mAdapter = new CallDiagnosticServiceAdapter(mTelecomAdapter,
493                 getActiveCallDiagnosticService(), mLock);
494         try {
495             // Add adapter for communication back from the call diagnostic service to Telecom.
496             callDiagnosticService.setAdapter(mAdapter);
497 
498             // Loop through all the calls we've got ready to send since binding.
499             for (Call call : mCallIdMapper.getCalls()) {
500                 sendCallToBoundService(call, callDiagnosticService);
501             }
502         } catch (RemoteException e) {
503             Log.w(this, "handleConnectionComplete: error=%s", e);
504         }
505     }
506 
507     /**
508      * Handles a request from a {@link CallDiagnosticService} to display a diagnostic message.
509      * @param callId the ID of the call to display the message for.
510      * @param message the message.
511      */
512     private void handleDisplayDiagnosticMessage(@NonNull String callId, int messageId,
513             @Nullable CharSequence message) {
514         Call call = mCallIdMapper.getCall(callId);
515         if (call == null) {
516             Log.w(this, "handleDisplayDiagnosticMessage: callId=%s; msg=%d/%s; invalid call",
517                     callId, messageId, message);
518             return;
519         }
520         Log.i(this, "handleDisplayDiagnosticMessage: callId=%s; msg=%d/%s",
521                 callId, messageId, message);
522         if (mPlayerFactory != null) {
523             // Play that tone!
524             mPlayerFactory.createPlayer(InCallTonePlayer.TONE_IN_CALL_QUALITY_NOTIFICATION)
525                     .startTone();
526         }
527         call.displayDiagnosticMessage(messageId, message);
528     }
529 
530     /**
531      * Handles a request from a {@link CallDiagnosticService} to clear a previously displayed
532      * diagnostic message.
533      * @param callId the ID of the call to display the message for.
534      * @param messageId the message ID which was previous posted.
535      */
536     private void handleClearDiagnosticMessage(@NonNull String callId, int messageId) {
537         Call call = mCallIdMapper.getCall(callId);
538         if (call == null) {
539             Log.w(this, "handleClearDiagnosticMessage: callId=%s; msg=%d; invalid call",
540                     callId, messageId);
541             return;
542         }
543         Log.i(this, "handleClearDiagnosticMessage: callId=%s; msg=%d; invalid call",
544                 callId, messageId);
545         call.clearDiagnosticMessage(messageId);
546     }
547 
548     /**
549      * Handles a request from a {@link CallDiagnosticService} to send a device to device message.
550      * @param callId The ID of the call to send the D2D message for.
551      * @param message The message type.
552      * @param value The message value.
553      */
554     private void handleSendD2DMessage(@NonNull String callId,
555             @CallDiagnostics.MessageType int message, int value) {
556         Call call = mCallIdMapper.getCall(callId);
557         if (call == null) {
558             Log.w(this, "handleSendD2DMessage: callId=%s; msg=%d/%d; invalid call", callId,
559                     message, value);
560             return;
561         }
562         Log.i(this, "handleSendD2DMessage: callId=%s; msg=%d/%d", callId, message, value);
563         call.sendDeviceToDeviceMessage(message, value);
564     }
565 
566     /**
567      * Handles a request from a {@link CallDiagnosticService} to override the disconnect message
568      * for a call.  This is the response path from a previous call into the
569      * {@link CallDiagnosticService} via {@link CallDiagnostics#onCallDisconnected(ImsReasonInfo)}.
570      * @param callId The telecom call ID the disconnect override is pending for.
571      * @param message The new disconnect message, or {@code null} if no override.
572      */
573     private void handleOverrideDisconnectMessage(@NonNull String callId,
574             @Nullable CharSequence message) {
575         Call call = mCallIdMapper.getCall(callId);
576         if (call == null) {
577             Log.w(this, "handleOverrideDisconnectMessage: callId=%s; msg=%s; invalid call", callId,
578                     message);
579             return;
580         }
581         Log.i(this, "handleOverrideDisconnectMessage: callId=%s; msg=%s", callId, message);
582         call.handleOverrideDisconnectMessage(message);
583     }
584 
585     /**
586      * Sends a single call to the bound {@link CallDiagnosticService}.
587      * @param call The call to send.
588      * @param callDiagnosticService The CDS to send it to.
589      */
590     private void sendCallToBoundService(@NonNull Call call,
591             @NonNull ICallDiagnosticService callDiagnosticService) {
592         try {
593             if (isConnected()) {
594                 Log.w(this, "sendCallToBoundService: initializing %s", call.getId());
595                 callDiagnosticService.initializeDiagnosticCall(getParceledCall(call));
596             } else {
597                 Log.w(this, "sendCallToBoundService: not bound, skipping %s", call.getId());
598             }
599         } catch (RemoteException e) {
600             Log.w(this, "sendCallToBoundService: callId=%s, exception=%s", call.getId(), e);
601         }
602     }
603 
604     /**
605      * Removes a call from a bound {@link CallDiagnosticService}.
606      * @param call The call to remove.
607      * @param callDiagnosticService The CDS to remove it from.
608      */
609     private void removeCallFromBoundService(@NonNull Call call,
610             @NonNull ICallDiagnosticService callDiagnosticService) {
611         try {
612             if (isConnected()) {
613                 callDiagnosticService.removeDiagnosticCall(call.getId());
614             }
615         } catch (RemoteException e) {
616             Log.w(this, "removeCallFromBoundService: callId=%s, exception=%s", call.getId(), e);
617         }
618     }
619 
620     /**
621      * @return {@code true} if the call diagnostic service is bound/connected.
622      */
623     public boolean isConnected() {
624         return mCallDiagnosticService != null;
625     }
626 
627     /**
628      * Updates the Call diagnostic service with changes to a call.
629      * @param call The updated call.
630      */
631     private void updateCall(@NonNull Call call) {
632         try {
633             if (isConnected()) {
634                 mCallDiagnosticService.updateCall(getParceledCall(call));
635             }
636         } catch (RemoteException e) {
637             Log.w(this, "updateCall: callId=%s, exception=%s", call.getId(), e);
638         }
639     }
640 
641     /**
642      * Updates the call diagnostic service with a received bluetooth quality report.
643      * @param call The call.
644      * @param report The bluetooth call quality report.
645      */
646     private void handleBluetoothCallQualityReport(@NonNull Call call,
647             @NonNull BluetoothCallQualityReport report) {
648         try {
649             if (isConnected()) {
650                 mCallDiagnosticService.receiveBluetoothCallQualityReport(report);
651             }
652         } catch (RemoteException e) {
653             Log.w(this, "handleBluetoothCallQualityReport: callId=%s, exception=%s", call.getId(),
654                     e);
655         }
656     }
657 
658     /**
659      * Informs a CallDiagnosticService of an incoming device to device message which was received
660      * via the carrier network.
661      * @param call the call the message was received via.
662      * @param messageType The message type.
663      * @param messageValue The message value.
664      */
665     private void handleReceivedDeviceToDeviceMessage(@NonNull Call call, int messageType,
666             int messageValue) {
667         try {
668             if (isConnected()) {
669                 mCallDiagnosticService.receiveDeviceToDeviceMessage(call.getId(), messageType,
670                         messageValue);
671             }
672         } catch (RemoteException e) {
673             Log.w(this, "handleReceivedDeviceToDeviceMessage: callId=%s, exception=%s",
674                     call.getId(), e);
675         }
676     }
677 
678     /**
679      * Handles a reported {@link CallQuality} report from a {@link android.telecom.Connection}.
680      * @param call The call the report originated from.
681      * @param callQualityReport The {@link CallQuality} report.
682      */
683     private void handleCallQualityReport(@NonNull Call call,
684             @NonNull CallQuality callQualityReport) {
685         try {
686             if (isConnected()) {
687                 mCallDiagnosticService.callQualityChanged(call.getId(), callQualityReport);
688             }
689         } catch (RemoteException e) {
690             Log.w(this, "handleCallQualityReport: callId=%s, exception=%s",
691                     call.getId(), e);
692         }
693     }
694 
695     /**
696      * Get a parcelled representation of a call for transport to the service.
697      * @param call The call.
698      * @return The parcelled call.
699      */
700     private @NonNull ParcelableCall getParceledCall(@NonNull Call call) {
701         return ParcelableCallUtils.toParcelableCall(
702                 call,
703                 false /* includeVideoProvider */,
704                 null /* phoneAcctRegistrar */,
705                 false /* supportsExternalCalls */,
706                 false /* includeRttCall */,
707                 false /* isForSystemDialer */
708         );
709     }
710 
711     /**
712      * Dumps the state of the {@link CallDiagnosticServiceController}.
713      *
714      * @param pw The {@code IndentingPrintWriter} to write the state to.
715      */
716     public void dump(IndentingPrintWriter pw) {
717         pw.print("activeCallDiagnosticService: ");
718         pw.println(getActiveCallDiagnosticService());
719         pw.print("isConnected: ");
720         pw.println(isConnected());
721     }
722 }
723