1 /* 2 * Copyright (C) 2015 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.content.Context; 20 import android.os.SystemProperties; 21 22 import android.telecom.Connection; 23 import android.telecom.DisconnectCause; 24 import android.telecom.Logging.EventManager; 25 import android.telecom.ParcelableCallAnalytics; 26 import android.telecom.TelecomAnalytics; 27 import android.telecom.TelecomManager; 28 import android.telephony.SubscriptionInfo; 29 import android.telephony.SubscriptionManager; 30 import android.util.Base64; 31 import android.telecom.Log; 32 33 import com.android.internal.annotations.VisibleForTesting; 34 import com.android.internal.util.IndentingPrintWriter; 35 import com.android.server.telecom.nano.TelecomLogClass; 36 37 import java.io.PrintWriter; 38 import java.time.Instant; 39 import java.time.ZoneOffset; 40 import java.util.ArrayList; 41 import java.util.Arrays; 42 import java.util.Collections; 43 import java.util.Comparator; 44 import java.util.HashMap; 45 import java.util.LinkedList; 46 import java.util.List; 47 import java.util.Map; 48 import java.util.concurrent.LinkedBlockingDeque; 49 import java.util.stream.Collectors; 50 51 import static android.telecom.ParcelableCallAnalytics.AnalyticsEvent; 52 import static android.telecom.TelecomAnalytics.SessionTiming; 53 54 /** 55 * A class that collects and stores data on how calls are being made, in order to 56 * aggregate these into useful statistics. 57 */ 58 public class Analytics { 59 public static final String ANALYTICS_DUMPSYS_ARG = "analytics"; 60 private static final String CLEAR_ANALYTICS_ARG = "clear"; 61 62 public static final Map<String, Integer> sLogEventToAnalyticsEvent = 63 new HashMap<String, Integer>() {{ 64 put(LogUtils.Events.SET_SELECT_PHONE_ACCOUNT, 65 AnalyticsEvent.SET_SELECT_PHONE_ACCOUNT); 66 put(LogUtils.Events.REQUEST_HOLD, AnalyticsEvent.REQUEST_HOLD); 67 put(LogUtils.Events.REQUEST_UNHOLD, AnalyticsEvent.REQUEST_UNHOLD); 68 put(LogUtils.Events.SWAP, AnalyticsEvent.SWAP); 69 put(LogUtils.Events.SKIP_RINGING, AnalyticsEvent.SKIP_RINGING); 70 put(LogUtils.Events.CONFERENCE_WITH, AnalyticsEvent.CONFERENCE_WITH); 71 put(LogUtils.Events.SPLIT_FROM_CONFERENCE, AnalyticsEvent.SPLIT_CONFERENCE); 72 put(LogUtils.Events.SET_PARENT, AnalyticsEvent.SET_PARENT); 73 put(LogUtils.Events.MUTE, AnalyticsEvent.MUTE); 74 put(LogUtils.Events.UNMUTE, AnalyticsEvent.UNMUTE); 75 put(LogUtils.Events.AUDIO_ROUTE_BT, AnalyticsEvent.AUDIO_ROUTE_BT); 76 put(LogUtils.Events.AUDIO_ROUTE_EARPIECE, AnalyticsEvent.AUDIO_ROUTE_EARPIECE); 77 put(LogUtils.Events.AUDIO_ROUTE_HEADSET, AnalyticsEvent.AUDIO_ROUTE_HEADSET); 78 put(LogUtils.Events.AUDIO_ROUTE_SPEAKER, AnalyticsEvent.AUDIO_ROUTE_SPEAKER); 79 put(LogUtils.Events.SILENCE, AnalyticsEvent.SILENCE); 80 put(LogUtils.Events.SCREENING_COMPLETED, AnalyticsEvent.SCREENING_COMPLETED); 81 put(LogUtils.Events.BLOCK_CHECK_FINISHED, AnalyticsEvent.BLOCK_CHECK_FINISHED); 82 put(LogUtils.Events.DIRECT_TO_VM_FINISHED, AnalyticsEvent.DIRECT_TO_VM_FINISHED); 83 put(LogUtils.Events.REMOTELY_HELD, AnalyticsEvent.REMOTELY_HELD); 84 put(LogUtils.Events.REMOTELY_UNHELD, AnalyticsEvent.REMOTELY_UNHELD); 85 put(LogUtils.Events.REQUEST_PULL, AnalyticsEvent.REQUEST_PULL); 86 put(LogUtils.Events.REQUEST_ACCEPT, AnalyticsEvent.REQUEST_ACCEPT); 87 put(LogUtils.Events.REQUEST_REJECT, AnalyticsEvent.REQUEST_REJECT); 88 put(LogUtils.Events.SET_ACTIVE, AnalyticsEvent.SET_ACTIVE); 89 put(LogUtils.Events.SET_DISCONNECTED, AnalyticsEvent.SET_DISCONNECTED); 90 put(LogUtils.Events.SET_HOLD, AnalyticsEvent.SET_HOLD); 91 put(LogUtils.Events.SET_DIALING, AnalyticsEvent.SET_DIALING); 92 put(LogUtils.Events.START_CONNECTION, AnalyticsEvent.START_CONNECTION); 93 put(LogUtils.Events.BIND_CS, AnalyticsEvent.BIND_CS); 94 put(LogUtils.Events.CS_BOUND, AnalyticsEvent.CS_BOUND); 95 put(LogUtils.Events.SCREENING_SENT, AnalyticsEvent.SCREENING_SENT); 96 put(LogUtils.Events.DIRECT_TO_VM_INITIATED, AnalyticsEvent.DIRECT_TO_VM_INITIATED); 97 put(LogUtils.Events.BLOCK_CHECK_INITIATED, AnalyticsEvent.BLOCK_CHECK_INITIATED); 98 put(LogUtils.Events.FILTERING_INITIATED, AnalyticsEvent.FILTERING_INITIATED); 99 put(LogUtils.Events.FILTERING_COMPLETED, AnalyticsEvent.FILTERING_COMPLETED); 100 put(LogUtils.Events.FILTERING_TIMED_OUT, AnalyticsEvent.FILTERING_TIMED_OUT); 101 }}; 102 103 public static final Map<String, Integer> sLogSessionToSessionId = 104 new HashMap<String, Integer> () {{ 105 put(LogUtils.Sessions.ICA_ANSWER_CALL, SessionTiming.ICA_ANSWER_CALL); 106 put(LogUtils.Sessions.ICA_REJECT_CALL, SessionTiming.ICA_REJECT_CALL); 107 put(LogUtils.Sessions.ICA_DISCONNECT_CALL, SessionTiming.ICA_DISCONNECT_CALL); 108 put(LogUtils.Sessions.ICA_HOLD_CALL, SessionTiming.ICA_HOLD_CALL); 109 put(LogUtils.Sessions.ICA_UNHOLD_CALL, SessionTiming.ICA_UNHOLD_CALL); 110 put(LogUtils.Sessions.ICA_MUTE, SessionTiming.ICA_MUTE); 111 put(LogUtils.Sessions.ICA_SET_AUDIO_ROUTE, SessionTiming.ICA_SET_AUDIO_ROUTE); 112 put(LogUtils.Sessions.ICA_CONFERENCE, SessionTiming.ICA_CONFERENCE); 113 put(LogUtils.Sessions.CSW_HANDLE_CREATE_CONNECTION_COMPLETE, 114 SessionTiming.CSW_HANDLE_CREATE_CONNECTION_COMPLETE); 115 put(LogUtils.Sessions.CSW_SET_ACTIVE, SessionTiming.CSW_SET_ACTIVE); 116 put(LogUtils.Sessions.CSW_SET_RINGING, SessionTiming.CSW_SET_RINGING); 117 put(LogUtils.Sessions.CSW_SET_DIALING, SessionTiming.CSW_SET_DIALING); 118 put(LogUtils.Sessions.CSW_SET_DISCONNECTED, SessionTiming.CSW_SET_DISCONNECTED); 119 put(LogUtils.Sessions.CSW_SET_ON_HOLD, SessionTiming.CSW_SET_ON_HOLD); 120 put(LogUtils.Sessions.CSW_REMOVE_CALL, SessionTiming.CSW_REMOVE_CALL); 121 put(LogUtils.Sessions.CSW_SET_IS_CONFERENCED, SessionTiming.CSW_SET_IS_CONFERENCED); 122 put(LogUtils.Sessions.CSW_ADD_CONFERENCE_CALL, 123 SessionTiming.CSW_ADD_CONFERENCE_CALL); 124 125 }}; 126 127 public static final Map<String, Integer> sLogEventTimingToAnalyticsEventTiming = 128 new HashMap<String, Integer>() {{ 129 put(LogUtils.Events.Timings.ACCEPT_TIMING, 130 ParcelableCallAnalytics.EventTiming.ACCEPT_TIMING); 131 put(LogUtils.Events.Timings.REJECT_TIMING, 132 ParcelableCallAnalytics.EventTiming.REJECT_TIMING); 133 put(LogUtils.Events.Timings.DISCONNECT_TIMING, 134 ParcelableCallAnalytics.EventTiming.DISCONNECT_TIMING); 135 put(LogUtils.Events.Timings.HOLD_TIMING, 136 ParcelableCallAnalytics.EventTiming.HOLD_TIMING); 137 put(LogUtils.Events.Timings.UNHOLD_TIMING, 138 ParcelableCallAnalytics.EventTiming.UNHOLD_TIMING); 139 put(LogUtils.Events.Timings.OUTGOING_TIME_TO_DIALING_TIMING, 140 ParcelableCallAnalytics.EventTiming.OUTGOING_TIME_TO_DIALING_TIMING); 141 put(LogUtils.Events.Timings.BIND_CS_TIMING, 142 ParcelableCallAnalytics.EventTiming.BIND_CS_TIMING); 143 put(LogUtils.Events.Timings.SCREENING_COMPLETED_TIMING, 144 ParcelableCallAnalytics.EventTiming.SCREENING_COMPLETED_TIMING); 145 put(LogUtils.Events.Timings.DIRECT_TO_VM_FINISHED_TIMING, 146 ParcelableCallAnalytics.EventTiming.DIRECT_TO_VM_FINISHED_TIMING); 147 put(LogUtils.Events.Timings.BLOCK_CHECK_FINISHED_TIMING, 148 ParcelableCallAnalytics.EventTiming.BLOCK_CHECK_FINISHED_TIMING); 149 put(LogUtils.Events.Timings.FILTERING_COMPLETED_TIMING, 150 ParcelableCallAnalytics.EventTiming.FILTERING_COMPLETED_TIMING); 151 put(LogUtils.Events.Timings.FILTERING_TIMED_OUT_TIMING, 152 ParcelableCallAnalytics.EventTiming.FILTERING_TIMED_OUT_TIMING); 153 put(LogUtils.Events.Timings.START_CONNECTION_TO_REQUEST_DISCONNECT_TIMING, 154 ParcelableCallAnalytics.EventTiming. 155 START_CONNECTION_TO_REQUEST_DISCONNECT_TIMING); 156 }}; 157 158 public static final Map<Integer, String> sSessionIdToLogSession = new HashMap<>(); 159 static { 160 for (Map.Entry<String, Integer> e : sLogSessionToSessionId.entrySet()) { 161 sSessionIdToLogSession.put(e.getValue(), e.getKey()); 162 } 163 } 164 165 public static class CallInfo { 166 public void setCallStartTime(long startTime) { 167 } 168 169 public void setCallEndTime(long endTime) { 170 } 171 172 public void setCallIsAdditional(boolean isAdditional) { 173 } 174 175 public void setCallIsEmergency(boolean isEmergency) { 176 } 177 178 public void setCallIsInterrupted(boolean isInterrupted) { 179 } 180 181 public void setCallDisconnectCause(DisconnectCause disconnectCause) { 182 } 183 184 public void addCallTechnology(int callTechnology) { 185 } 186 187 public void setCreatedFromExistingConnection(boolean createdFromExistingConnection) { 188 } 189 190 public void setCallConnectionService(String connectionServiceName) { 191 } 192 193 public void setCallEvents(EventManager.EventRecord records) { 194 } 195 196 public void setCallIsVideo(boolean isVideo) { 197 } 198 199 public void addVideoEvent(int eventId, int videoState) { 200 } 201 202 public void addInCallService(String serviceName, int type, long boundDuration, 203 boolean isNullBinding) { 204 } 205 206 public void addCallProperties(int properties) { 207 } 208 209 public void setCallSource(int callSource) { 210 } 211 212 public void setMissedReason(long missedReason) { 213 } 214 } 215 216 /** 217 * A class that holds data associated with a call. 218 */ 219 @VisibleForTesting 220 public static class CallInfoImpl extends CallInfo { 221 public String callId; 222 public long startTime; // start time in milliseconds since the epoch. 0 if not yet set. 223 public long endTime; // end time in milliseconds since the epoch. 0 if not yet set. 224 public int callDirection; // one of UNKNOWN_DIRECTION, INCOMING_DIRECTION, 225 // or OUTGOING_DIRECTION. 226 public boolean isAdditionalCall = false; // true if the call came in while another call was 227 // in progress or if the user dialed this call 228 // while in the middle of another call. 229 public boolean isInterrupted = false; // true if the call was interrupted by an incoming 230 // or outgoing call. 231 public int callTechnologies; // bitmask denoting which technologies a call used. 232 233 // true if the Telecom Call object was created from an existing connection via 234 // CallsManager#createCallForExistingConnection, for example, by ImsConference. 235 public boolean createdFromExistingConnection = false; 236 237 public DisconnectCause callTerminationReason; 238 public String connectionService; 239 public boolean isEmergency = false; 240 241 public EventManager.EventRecord callEvents; 242 243 public boolean isVideo = false; 244 public List<TelecomLogClass.VideoEvent> videoEvents; 245 public List<TelecomLogClass.InCallServiceInfo> inCallServiceInfos; 246 public int callProperties = 0; 247 public int callSource = CALL_SOURCE_UNSPECIFIED; 248 public long missedReason; 249 250 private long mTimeOfLastVideoEvent = -1; 251 252 CallInfoImpl(String callId, int callDirection) { 253 this.callId = callId; 254 startTime = 0; 255 endTime = 0; 256 this.callDirection = callDirection; 257 callTechnologies = 0; 258 connectionService = ""; 259 videoEvents = new LinkedList<>(); 260 inCallServiceInfos = new LinkedList<>(); 261 missedReason = 0; 262 } 263 264 CallInfoImpl(CallInfoImpl other) { 265 this.callId = other.callId; 266 this.startTime = other.startTime; 267 this.endTime = other.endTime; 268 this.callDirection = other.callDirection; 269 this.isAdditionalCall = other.isAdditionalCall; 270 this.isInterrupted = other.isInterrupted; 271 this.callTechnologies = other.callTechnologies; 272 this.createdFromExistingConnection = other.createdFromExistingConnection; 273 this.connectionService = other.connectionService; 274 this.isEmergency = other.isEmergency; 275 this.callEvents = other.callEvents; 276 this.isVideo = other.isVideo; 277 this.videoEvents = other.videoEvents; 278 this.callProperties = other.callProperties; 279 this.callSource = other.callSource; 280 this.missedReason = other.missedReason; 281 282 if (other.callTerminationReason != null) { 283 this.callTerminationReason = new DisconnectCause( 284 other.callTerminationReason.getCode(), 285 other.callTerminationReason.getLabel(), 286 other.callTerminationReason.getDescription(), 287 other.callTerminationReason.getReason(), 288 other.callTerminationReason.getTone()); 289 } else { 290 this.callTerminationReason = null; 291 } 292 } 293 294 @Override 295 public void setCallStartTime(long startTime) { 296 Log.d(TAG, "setting startTime for call " + callId + " to " + startTime); 297 this.startTime = startTime; 298 } 299 300 @Override 301 public void setCallEndTime(long endTime) { 302 Log.d(TAG, "setting endTime for call " + callId + " to " + endTime); 303 this.endTime = endTime; 304 } 305 306 @Override 307 public void setCallIsAdditional(boolean isAdditional) { 308 Log.d(TAG, "setting isAdditional for call " + callId + " to " + isAdditional); 309 this.isAdditionalCall = isAdditional; 310 } 311 312 @Override 313 public void setCallIsInterrupted(boolean isInterrupted) { 314 Log.d(TAG, "setting isInterrupted for call " + callId + " to " + isInterrupted); 315 this.isInterrupted = isInterrupted; 316 } 317 318 @Override 319 public void addCallTechnology(int callTechnology) { 320 Log.d(TAG, "adding callTechnology for call " + callId + ": " + callTechnology); 321 this.callTechnologies |= callTechnology; 322 } 323 324 @Override 325 public void setCallIsEmergency(boolean isEmergency) { 326 Log.d(TAG, "setting call as emergency: " + isEmergency); 327 this.isEmergency = isEmergency; 328 } 329 330 @Override 331 public void setCallDisconnectCause(DisconnectCause disconnectCause) { 332 Log.d(TAG, "setting disconnectCause for call " + callId + " to " + disconnectCause); 333 this.callTerminationReason = disconnectCause; 334 } 335 336 @Override 337 public void setCreatedFromExistingConnection(boolean createdFromExistingConnection) { 338 Log.d(TAG, "setting createdFromExistingConnection for call " + callId + " to " 339 + createdFromExistingConnection); 340 this.createdFromExistingConnection = createdFromExistingConnection; 341 } 342 343 @Override 344 public void setCallConnectionService(String connectionServiceName) { 345 Log.d(TAG, "setting connection service for call " + callId + ": " 346 + connectionServiceName); 347 this.connectionService = connectionServiceName; 348 } 349 350 @Override 351 public void setMissedReason(long missedReason) { 352 Log.d(TAG, "setting missedReason for call " + callId + ": " 353 + missedReason); 354 this.missedReason = missedReason; 355 } 356 357 @Override 358 public void setCallEvents(EventManager.EventRecord records) { 359 this.callEvents = records; 360 } 361 362 @Override 363 public void setCallIsVideo(boolean isVideo) { 364 this.isVideo = isVideo; 365 } 366 367 @Override 368 public void addVideoEvent(int eventId, int videoState) { 369 long timeSinceLastEvent; 370 long currentTime = System.currentTimeMillis(); 371 if (mTimeOfLastVideoEvent < 0) { 372 timeSinceLastEvent = -1; 373 } else { 374 timeSinceLastEvent = roundToOneSigFig(currentTime - mTimeOfLastVideoEvent); 375 } 376 mTimeOfLastVideoEvent = currentTime; 377 378 videoEvents.add(new TelecomLogClass.VideoEvent() 379 .setEventName(eventId) 380 .setTimeSinceLastEventMillis(timeSinceLastEvent) 381 .setVideoState(videoState)); 382 } 383 384 @Override 385 public void addInCallService(String serviceName, int type, long boundDuration, 386 boolean isNullBinding) { 387 inCallServiceInfos.add(new TelecomLogClass.InCallServiceInfo() 388 .setInCallServiceName(serviceName) 389 .setInCallServiceType(type) 390 .setBoundDurationMillis(boundDuration) 391 .setIsNullBinding(isNullBinding)); 392 } 393 394 @Override 395 public void addCallProperties(int properties) { 396 this.callProperties |= properties; 397 } 398 399 @Override 400 public void setCallSource(int callSource) { 401 this.callSource = callSource; 402 } 403 404 @Override 405 public String toString() { 406 return "{\n" 407 + " startTime: " + startTime + '\n' 408 + " endTime: " + endTime + '\n' 409 + " direction: " + getCallDirectionString() + '\n' 410 + " isAdditionalCall: " + isAdditionalCall + '\n' 411 + " isInterrupted: " + isInterrupted + '\n' 412 + " isEmergency: " + isEmergency + '\n' 413 + " callTechnologies: " + getCallTechnologiesAsString() + '\n' 414 + " callTerminationReason: " + getCallDisconnectReasonString() + '\n' 415 + " missedReason: " + getMissedReasonString() + '\n' 416 + " connectionService: " + connectionService + '\n' 417 + " isVideoCall: " + isVideo + '\n' 418 + " inCallServices: " + getInCallServicesString() + '\n' 419 + " callProperties: " + Connection.propertiesToStringShort(callProperties) 420 + '\n' 421 + " callSource: " + getCallSourceString() + '\n' 422 + "}\n"; 423 } 424 425 public ParcelableCallAnalytics toParcelableAnalytics() { 426 TelecomLogClass.CallLog analyticsProto = toProto(); 427 List<ParcelableCallAnalytics.AnalyticsEvent> events = 428 Arrays.stream(analyticsProto.callEvents) 429 .map(callEventProto -> new ParcelableCallAnalytics.AnalyticsEvent( 430 callEventProto.getEventName(), 431 callEventProto.getTimeSinceLastEventMillis()) 432 ).collect(Collectors.toList()); 433 434 List<ParcelableCallAnalytics.EventTiming> timings = 435 Arrays.stream(analyticsProto.callTimings) 436 .map(callTimingProto -> new ParcelableCallAnalytics.EventTiming( 437 callTimingProto.getTimingName(), 438 callTimingProto.getTimeMillis()) 439 ).collect(Collectors.toList()); 440 441 ParcelableCallAnalytics result = new ParcelableCallAnalytics( 442 // rounds down to nearest 5 minute mark 443 analyticsProto.getStartTime5Min(), 444 analyticsProto.getCallDurationMillis(), 445 analyticsProto.getType(), 446 analyticsProto.getIsAdditionalCall(), 447 analyticsProto.getIsInterrupted(), 448 analyticsProto.getCallTechnologies(), 449 analyticsProto.getCallTerminationCode(), 450 analyticsProto.getIsEmergencyCall(), 451 analyticsProto.connectionService[0], 452 analyticsProto.getIsCreatedFromExistingConnection(), 453 events, 454 timings); 455 456 result.setIsVideoCall(analyticsProto.getIsVideoCall()); 457 result.setVideoEvents(Arrays.stream(analyticsProto.videoEvents) 458 .map(videoEventProto -> new ParcelableCallAnalytics.VideoEvent( 459 videoEventProto.getEventName(), 460 videoEventProto.getTimeSinceLastEventMillis(), 461 videoEventProto.getVideoState()) 462 ).collect(Collectors.toList())); 463 464 result.setCallSource(analyticsProto.getCallSource()); 465 466 return result; 467 } 468 469 public TelecomLogClass.CallLog toProto() { 470 TelecomLogClass.CallLog result = new TelecomLogClass.CallLog(); 471 result.setStartTime5Min( 472 startTime - startTime % ParcelableCallAnalytics.MILLIS_IN_5_MINUTES); 473 474 // Rounds up to the nearest second. 475 long callDuration = (endTime == 0 || startTime == 0) ? 0 : endTime - startTime; 476 callDuration += (callDuration % MILLIS_IN_1_SECOND == 0) ? 477 0 : (MILLIS_IN_1_SECOND - callDuration % MILLIS_IN_1_SECOND); 478 result.setCallDurationMillis(callDuration); 479 480 result.setType(callDirection) 481 .setIsAdditionalCall(isAdditionalCall) 482 .setIsInterrupted(isInterrupted) 483 .setCallTechnologies(callTechnologies) 484 .setCallTerminationCode( 485 callTerminationReason == null ? 486 ParcelableCallAnalytics.STILL_CONNECTED : 487 callTerminationReason.getCode()) 488 .setIsEmergencyCall(isEmergency) 489 .setIsCreatedFromExistingConnection(createdFromExistingConnection) 490 .setIsEmergencyCall(isEmergency) 491 .setIsVideoCall(isVideo) 492 .setConnectionProperties(callProperties) 493 .setCallSource(callSource); 494 495 result.connectionService = new String[] {connectionService}; 496 if (callEvents != null) { 497 result.callEvents = convertLogEventsToProtoEvents(callEvents.getEvents()); 498 result.callTimings = callEvents.extractEventTimings().stream() 499 .map(Analytics::logEventTimingToProtoEventTiming) 500 .toArray(TelecomLogClass.EventTimingEntry[]::new); 501 } 502 result.videoEvents = 503 videoEvents.toArray(new TelecomLogClass.VideoEvent[videoEvents.size()]); 504 result.inCallServices = inCallServiceInfos.toArray( 505 new TelecomLogClass.InCallServiceInfo[inCallServiceInfos.size()]); 506 507 return result; 508 } 509 510 private String getCallDirectionString() { 511 switch (callDirection) { 512 case UNKNOWN_DIRECTION: 513 return "UNKNOWN"; 514 case INCOMING_DIRECTION: 515 return "INCOMING"; 516 case OUTGOING_DIRECTION: 517 return "OUTGOING"; 518 default: 519 return "UNKNOWN"; 520 } 521 } 522 523 private String getCallTechnologiesAsString() { 524 StringBuilder s = new StringBuilder(); 525 s.append('['); 526 if ((callTechnologies & CDMA_PHONE) != 0) s.append("CDMA "); 527 if ((callTechnologies & GSM_PHONE) != 0) s.append("GSM "); 528 if ((callTechnologies & SIP_PHONE) != 0) s.append("SIP "); 529 if ((callTechnologies & IMS_PHONE) != 0) s.append("IMS "); 530 if ((callTechnologies & THIRD_PARTY_PHONE) != 0) s.append("THIRD_PARTY "); 531 s.append(']'); 532 return s.toString(); 533 } 534 535 private String getCallDisconnectReasonString() { 536 if (callTerminationReason != null) { 537 return callTerminationReason.toString(); 538 } else { 539 return "NOT SET"; 540 } 541 } 542 543 private String getMissedReasonString() { 544 //TODO: Implement this 545 return null; 546 } 547 548 private String getInCallServicesString() { 549 StringBuilder s = new StringBuilder(); 550 s.append("[\n"); 551 if (inCallServiceInfos != null) { 552 for (TelecomLogClass.InCallServiceInfo service : inCallServiceInfos) { 553 s.append(" "); 554 s.append("name: "); 555 s.append(service.getInCallServiceName()); 556 s.append(" type: "); 557 s.append(service.getInCallServiceType()); 558 s.append(" is crashed: "); 559 s.append(service.getIsNullBinding()); 560 s.append(" service last time in ms: "); 561 s.append(service.getBoundDurationMillis()); 562 s.append("\n"); 563 } 564 } 565 s.append("]"); 566 return s.toString(); 567 } 568 569 private String getCallSourceString() { 570 switch (callSource) { 571 case CALL_SOURCE_UNSPECIFIED: 572 return "UNSPECIFIED"; 573 case CALL_SOURCE_EMERGENCY_DIALPAD: 574 return "EMERGENCY_DIALPAD"; 575 case CALL_SOURCE_EMERGENCY_SHORTCUT: 576 return "EMERGENCY_SHORTCUT"; 577 default: 578 return "UNSPECIFIED"; 579 } 580 } 581 } 582 public static final String TAG = "TelecomAnalytics"; 583 584 // Constants for call direction 585 public static final int UNKNOWN_DIRECTION = ParcelableCallAnalytics.CALLTYPE_UNKNOWN; 586 public static final int INCOMING_DIRECTION = ParcelableCallAnalytics.CALLTYPE_INCOMING; 587 public static final int OUTGOING_DIRECTION = ParcelableCallAnalytics.CALLTYPE_OUTGOING; 588 589 // Constants for call technology 590 public static final int CDMA_PHONE = ParcelableCallAnalytics.CDMA_PHONE; 591 public static final int GSM_PHONE = ParcelableCallAnalytics.GSM_PHONE; 592 public static final int IMS_PHONE = ParcelableCallAnalytics.IMS_PHONE; 593 public static final int SIP_PHONE = ParcelableCallAnalytics.SIP_PHONE; 594 public static final int THIRD_PARTY_PHONE = ParcelableCallAnalytics.THIRD_PARTY_PHONE; 595 596 // Constants for call source 597 public static final int CALL_SOURCE_UNSPECIFIED = 598 TelecomManager.CALL_SOURCE_UNSPECIFIED; 599 public static final int CALL_SOURCE_EMERGENCY_DIALPAD = 600 TelecomManager.CALL_SOURCE_EMERGENCY_DIALPAD; 601 public static final int CALL_SOURCE_EMERGENCY_SHORTCUT = 602 TelecomManager.CALL_SOURCE_EMERGENCY_SHORTCUT; 603 604 // Constants for video events 605 public static final int SEND_LOCAL_SESSION_MODIFY_REQUEST = 606 ParcelableCallAnalytics.VideoEvent.SEND_LOCAL_SESSION_MODIFY_REQUEST; 607 public static final int SEND_LOCAL_SESSION_MODIFY_RESPONSE = 608 ParcelableCallAnalytics.VideoEvent.SEND_LOCAL_SESSION_MODIFY_RESPONSE; 609 public static final int RECEIVE_REMOTE_SESSION_MODIFY_REQUEST = 610 ParcelableCallAnalytics.VideoEvent.RECEIVE_REMOTE_SESSION_MODIFY_REQUEST; 611 public static final int RECEIVE_REMOTE_SESSION_MODIFY_RESPONSE = 612 ParcelableCallAnalytics.VideoEvent.RECEIVE_REMOTE_SESSION_MODIFY_RESPONSE; 613 614 public static final long MILLIS_IN_1_SECOND = ParcelableCallAnalytics.MILLIS_IN_1_SECOND; 615 616 public static final int MAX_NUM_CALLS_TO_STORE = 100; 617 public static final int MAX_NUM_DUMP_TIMES_TO_STORE = 100; 618 619 private static final Object sLock = new Object(); // Coarse lock for all of analytics 620 private static final LinkedBlockingDeque<Long> sDumpTimes = 621 new LinkedBlockingDeque<>(MAX_NUM_DUMP_TIMES_TO_STORE); 622 private static final Map<String, CallInfoImpl> sCallIdToInfo = new HashMap<>(); 623 private static final LinkedList<String> sActiveCallIds = new LinkedList<>(); 624 private static final List<SessionTiming> sSessionTimings = new LinkedList<>(); 625 626 public static void addSessionTiming(String sessionName, long time) { 627 if (sLogSessionToSessionId.containsKey(sessionName)) { 628 synchronized (sLock) { 629 sSessionTimings.add(new SessionTiming(sLogSessionToSessionId.get(sessionName), 630 time)); 631 } 632 } 633 } 634 635 public static CallInfo initiateCallAnalytics(String callId, int direction) { 636 Log.i(TAG, "Starting analytics for call " + callId); 637 CallInfoImpl callInfo = new CallInfoImpl(callId, direction); 638 synchronized (sLock) { 639 while (sActiveCallIds.size() >= MAX_NUM_CALLS_TO_STORE) { 640 String callToRemove = sActiveCallIds.remove(); 641 sCallIdToInfo.remove(callToRemove); 642 } 643 sCallIdToInfo.put(callId, callInfo); 644 sActiveCallIds.add(callId); 645 } 646 return callInfo; 647 } 648 649 public static TelecomAnalytics dumpToParcelableAnalytics() { 650 List<ParcelableCallAnalytics> calls = new LinkedList<>(); 651 List<SessionTiming> sessionTimings = new LinkedList<>(); 652 synchronized (sLock) { 653 calls.addAll(sCallIdToInfo.values().stream() 654 .map(CallInfoImpl::toParcelableAnalytics) 655 .collect(Collectors.toList())); 656 sessionTimings.addAll(sSessionTimings); 657 sCallIdToInfo.clear(); 658 sSessionTimings.clear(); 659 } 660 return new TelecomAnalytics(sessionTimings, calls); 661 } 662 663 public static void dumpToEncodedProto(Context context, PrintWriter pw, String[] args) { 664 TelecomLogClass.TelecomLog result = new TelecomLogClass.TelecomLog(); 665 666 synchronized (sLock) { 667 noteDumpTime(); 668 result.callLogs = sCallIdToInfo.values().stream() 669 .map(CallInfoImpl::toProto) 670 .toArray(TelecomLogClass.CallLog[]::new); 671 result.sessionTimings = sSessionTimings.stream() 672 .map(timing -> new TelecomLogClass.LogSessionTiming() 673 .setSessionEntryPoint(timing.getKey()) 674 .setTimeMillis(timing.getTime())) 675 .toArray(TelecomLogClass.LogSessionTiming[]::new); 676 result.setHardwareRevision(SystemProperties.get("ro.boot.revision", "")); 677 result.setCarrierId(getCarrierId(context)); 678 if (args.length > 1 && CLEAR_ANALYTICS_ARG.equals(args[1])) { 679 sCallIdToInfo.clear(); 680 sSessionTimings.clear(); 681 } 682 } 683 String encodedProto = Base64.encodeToString( 684 TelecomLogClass.TelecomLog.toByteArray(result), Base64.DEFAULT); 685 pw.write(encodedProto); 686 } 687 688 private static int getCarrierId(Context context) { 689 SubscriptionManager subscriptionManager = 690 context.getSystemService(SubscriptionManager.class); 691 List<SubscriptionInfo> subInfos = subscriptionManager.getActiveSubscriptionInfoList(); 692 if (subInfos == null) { 693 return -1; 694 } 695 return subInfos.stream() 696 .max(Comparator.comparing(Analytics::scoreSubscriptionInfo)) 697 .map(SubscriptionInfo::getCarrierId).orElse(-1); 698 } 699 700 // Copied over from Telephony's server-side logic for consistency 701 private static int scoreSubscriptionInfo(SubscriptionInfo subInfo) { 702 final int scoreCarrierId = 0b100; 703 final int scoreNotOpportunistic = 0b010; 704 final int scoreSlot0 = 0b001; 705 706 return ((subInfo.getCarrierId() >= 0) ? scoreCarrierId : 0) 707 + (subInfo.isOpportunistic() ? 0 : scoreNotOpportunistic) 708 + ((subInfo.getSimSlotIndex() == 0) ? scoreSlot0 : 0); 709 } 710 711 public static void dump(IndentingPrintWriter writer) { 712 synchronized (sLock) { 713 int prefixLength = CallsManager.TELECOM_CALL_ID_PREFIX.length(); 714 List<String> callIds = new ArrayList<>(sCallIdToInfo.keySet()); 715 // Sort the analytics in increasing order of call IDs 716 try { 717 Collections.sort(callIds, (id1, id2) -> { 718 int i1, i2; 719 try { 720 i1 = Integer.valueOf(id1.substring(prefixLength)); 721 } catch (NumberFormatException e) { 722 i1 = Integer.MAX_VALUE; 723 } 724 725 try { 726 i2 = Integer.valueOf(id2.substring(prefixLength)); 727 } catch (NumberFormatException e) { 728 i2 = Integer.MAX_VALUE; 729 } 730 return i1 - i2; 731 }); 732 } catch (IllegalArgumentException e) { 733 // do nothing, leave the list in a partially sorted state. 734 } 735 736 for (String callId : callIds) { 737 writer.printf("Call %s: ", callId); 738 writer.println(sCallIdToInfo.get(callId).toString()); 739 } 740 741 Map<Integer, Double> averageTimings = SessionTiming.averageTimings(sSessionTimings); 742 averageTimings.entrySet().stream() 743 .filter(e -> sSessionIdToLogSession.containsKey(e.getKey())) 744 .forEach(e -> writer.printf("%s: %.2f\n", 745 sSessionIdToLogSession.get(e.getKey()), e.getValue())); 746 writer.println("Hardware Version: " + SystemProperties.get("ro.boot.revision", "")); 747 writer.println("Past analytics dumps: "); 748 writer.increaseIndent(); 749 for (long time : sDumpTimes) { 750 writer.println(Instant.ofEpochMilli(time).atZone(ZoneOffset.UTC)); 751 } 752 writer.decreaseIndent(); 753 } 754 } 755 756 public static void reset() { 757 synchronized (sLock) { 758 sCallIdToInfo.clear(); 759 } 760 } 761 762 public static void noteDumpTime() { 763 if (sDumpTimes.remainingCapacity() == 0) { 764 sDumpTimes.removeLast(); 765 } 766 try { 767 sDumpTimes.addFirst(System.currentTimeMillis()); 768 } catch (IllegalStateException e) { 769 Log.w(TAG, "Failed to note dump time -- full"); 770 } 771 } 772 773 /** 774 * Returns a copy of callIdToInfo. Use only for testing. 775 */ 776 @VisibleForTesting 777 public static Map<String, CallInfoImpl> cloneData() { 778 synchronized (sLock) { 779 Map<String, CallInfoImpl> result = new HashMap<>(sCallIdToInfo.size()); 780 for (Map.Entry<String, CallInfoImpl> entry : sCallIdToInfo.entrySet()) { 781 result.put(entry.getKey(), new CallInfoImpl(entry.getValue())); 782 } 783 return result; 784 } 785 } 786 787 private static TelecomLogClass.Event[] convertLogEventsToProtoEvents( 788 List<EventManager.Event> logEvents) { 789 long timeOfLastEvent = -1; 790 ArrayList<TelecomLogClass.Event> events = new ArrayList<>(logEvents.size()); 791 for (EventManager.Event logEvent : logEvents) { 792 if (sLogEventToAnalyticsEvent.containsKey(logEvent.eventId)) { 793 TelecomLogClass.Event event = new TelecomLogClass.Event(); 794 event.setEventName(sLogEventToAnalyticsEvent.get(logEvent.eventId)); 795 event.setTimeSinceLastEventMillis(roundToOneSigFig( 796 timeOfLastEvent < 0 ? -1 : logEvent.time - timeOfLastEvent)); 797 events.add(event); 798 timeOfLastEvent = logEvent.time; 799 } 800 } 801 return events.toArray(new TelecomLogClass.Event[events.size()]); 802 } 803 804 private static TelecomLogClass.EventTimingEntry logEventTimingToProtoEventTiming( 805 EventManager.EventRecord.EventTiming logEventTiming) { 806 int analyticsEventTimingName = 807 sLogEventTimingToAnalyticsEventTiming.containsKey(logEventTiming.name) ? 808 sLogEventTimingToAnalyticsEventTiming.get(logEventTiming.name) : 809 ParcelableCallAnalytics.EventTiming.INVALID; 810 return new TelecomLogClass.EventTimingEntry() 811 .setTimingName(analyticsEventTimingName) 812 .setTimeMillis(logEventTiming.time); 813 } 814 815 @VisibleForTesting 816 public static long roundToOneSigFig(long val) { 817 if (val == 0) { 818 return val; 819 } 820 int logVal = (int) Math.floor(Math.log10(val < 0 ? -val : val)); 821 double s = Math.pow(10, logVal); 822 double dec = val / s; 823 return (long) (Math.round(dec) * s); 824 } 825 } 826