1 /****************************************************************************** 2 * 3 * Copyright 2016 Google, Inc. 4 * 5 * Licensed under the Apache License, Version 2.0 (the "License"); 6 * you may not use this file except in compliance with the License. 7 * You may obtain a copy of the License at: 8 * 9 * http://www.apache.org/licenses/LICENSE-2.0 10 * 11 * Unless required by applicable law or agreed to in writing, software 12 * distributed under the License is distributed on an "AS IS" BASIS, 13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14 * See the License for the specific language governing permissions and 15 * limitations under the License. 16 * 17 ******************************************************************************/ 18 19 #pragma once 20 21 #include <bta/include/bta_api.h> 22 #include <frameworks/proto_logging/stats/enums/bluetooth/enums.pb.h> 23 #include <frameworks/proto_logging/stats/enums/bluetooth/hci/enums.pb.h> 24 #include <frameworks/proto_logging/stats/enums/bluetooth/le/enums.pb.h> 25 #include <stdint.h> 26 27 #include <memory> 28 #include <string> 29 #include <vector> 30 31 #include "hci/address.h" 32 #include "os/metrics.h" 33 #include "types/raw_address.h" 34 35 namespace bluetooth { 36 37 namespace common { 38 39 // Typedefs to hide protobuf definition to the rest of stack 40 41 typedef enum { 42 DEVICE_TYPE_UNKNOWN, 43 DEVICE_TYPE_BREDR, 44 DEVICE_TYPE_LE, 45 DEVICE_TYPE_DUMO, 46 } device_type_t; 47 48 typedef enum { 49 WAKE_EVENT_UNKNOWN, 50 WAKE_EVENT_ACQUIRED, 51 WAKE_EVENT_RELEASED, 52 } wake_event_type_t; 53 54 typedef enum { 55 SCAN_TYPE_UNKNOWN, 56 SCAN_TECH_TYPE_LE, 57 SCAN_TECH_TYPE_BREDR, 58 SCAN_TECH_TYPE_BOTH, 59 } scan_tech_t; 60 61 typedef enum { 62 CONNECTION_TECHNOLOGY_TYPE_UNKNOWN, 63 CONNECTION_TECHNOLOGY_TYPE_LE, 64 CONNECTION_TECHNOLOGY_TYPE_BREDR, 65 } connection_tech_t; 66 67 typedef enum { 68 DISCONNECT_REASON_UNKNOWN, 69 DISCONNECT_REASON_METRICS_DUMP, 70 DISCONNECT_REASON_NEXT_START_WITHOUT_END_PREVIOUS, 71 } disconnect_reason_t; 72 73 /* Values of A2DP metrics that we care about 74 * 75 * audio_duration_ms : sum of audio duration (in milliseconds). 76 * device_class: device class of the paired device. 77 * media_timer_min_ms : minimum scheduled time (in milliseconds) 78 * of the media timer. 79 * media_timer_max_ms: maximum scheduled time (in milliseconds) 80 * of the media timer. 81 * media_timer_avg_ms: average scheduled time (in milliseconds) 82 * of the media timer. 83 * buffer_overruns_max_count: TODO - not clear what this is. 84 * buffer_overruns_total : number of times the media buffer with 85 * audio data has overrun 86 * buffer_underruns_average: TODO - not clear what this is. 87 * buffer_underruns_count: number of times there was no enough 88 * audio data to add to the media buffer. 89 * NOTE: Negative values are invalid 90 */ 91 class A2dpSessionMetrics { 92 public: A2dpSessionMetrics()93 A2dpSessionMetrics() {} 94 95 /* 96 * Update the metrics value in the current metrics object using the metrics 97 * objects supplied 98 */ 99 void Update(const A2dpSessionMetrics& metrics); 100 101 /* 102 * Compare whether two metrics objects are equal 103 */ 104 bool operator==(const A2dpSessionMetrics& rhs) const; 105 106 /* 107 * Initialize all values to -1 which is invalid in order to make a distinction 108 * between 0 and invalid values 109 */ 110 int64_t audio_duration_ms = -1; 111 int32_t media_timer_min_ms = -1; 112 int32_t media_timer_max_ms = -1; 113 int32_t media_timer_avg_ms = -1; 114 int64_t total_scheduling_count = -1; 115 int32_t buffer_overruns_max_count = -1; 116 int32_t buffer_overruns_total = -1; 117 float buffer_underruns_average = -1; 118 int32_t buffer_underruns_count = -1; 119 int64_t codec_index = -1; 120 bool is_a2dp_offload = false; 121 }; 122 123 class BluetoothMetricsLogger { 124 public: GetInstance()125 static BluetoothMetricsLogger* GetInstance() { 126 static BluetoothMetricsLogger* instance = new BluetoothMetricsLogger(); 127 return instance; 128 } 129 130 /* 131 * Record a pairing event 132 * 133 * Parameters: 134 * timestamp_ms: Unix epoch time in milliseconds 135 * device_class: class of remote device 136 * device_type: type of remote device 137 * disconnect_reason: HCI reason for pairing disconnection. 138 * See: stack/include/hcidefs.h 139 */ 140 void LogPairEvent(uint32_t disconnect_reason, uint64_t timestamp_ms, 141 uint32_t device_class, device_type_t device_type); 142 143 /* 144 * Record a wake event 145 * 146 * Parameters: 147 * timestamp_ms: Unix epoch time in milliseconds 148 * type: whether it was acquired or released 149 * requestor: if provided is the service requesting the wake lock 150 * name: the name of the wake lock held 151 */ 152 void LogWakeEvent(wake_event_type_t type, const std::string& requestor, 153 const std::string& name, uint64_t timestamp_ms); 154 155 /* 156 * Record a scan event 157 * 158 * Parameters 159 * timestamp_ms : Unix epoch time in milliseconds 160 * start : true if this is the beginning of the scan 161 * initiator: a unique ID identifying the app starting the scan 162 * type: whether the scan reports BR/EDR, LE, or both. 163 * results: number of results to be reported. 164 */ 165 void LogScanEvent(bool start, const std::string& initator, scan_tech_t type, 166 uint32_t results, uint64_t timestamp_ms); 167 168 /* 169 * Start logging a Bluetooth session 170 * 171 * A Bluetooth session is defined a a connection between this device and 172 * another remote device which may include multiple profiles and protocols 173 * 174 * Only one Bluetooth session can exist at one time. Calling this method twice 175 * without LogBluetoothSessionEnd will result in logging a premature end of 176 * current Bluetooth session 177 * 178 * Parameters: 179 * connection_tech_type : type of connection technology 180 * timestamp_ms : the timestamp for session start, 0 means now 181 * 182 */ 183 void LogBluetoothSessionStart(connection_tech_t connection_tech_type, 184 uint64_t timestamp_ms); 185 186 /* 187 * Stop logging a Bluetooth session and pushes it to the log queue 188 * 189 * If no Bluetooth session exist, this method exits immediately 190 * 191 * Parameters: 192 * disconnect_reason : A string representation of disconnect reason 193 * timestamp_ms : the timestamp of session end, 0 means now 194 * 195 */ 196 void LogBluetoothSessionEnd(disconnect_reason_t disconnect_reason, 197 uint64_t timestamp_ms); 198 199 /* 200 * Log information about remote device in a current Bluetooth session 201 * 202 * If a Bluetooth session does not exist, create one with default parameter 203 * and timestamp now 204 * 205 * Parameters: 206 * device_class : device_class defined in btm_api_types.h 207 * device_type : type of remote device 208 */ 209 void LogBluetoothSessionDeviceInfo(uint32_t device_class, 210 device_type_t device_type); 211 212 /* 213 * Log A2DP Audio Session Information 214 * 215 * - Repeated calls to this method will override previous metrics if in the 216 * same Bluetooth connection 217 * - If a Bluetooth session does not exist, create one with default parameter 218 * and timestamp now 219 * 220 * Parameters: 221 * a2dp_session_metrics - pointer to struct holding a2dp stats 222 * 223 */ 224 void LogA2dpSession(const A2dpSessionMetrics& a2dp_session_metrics); 225 226 /** 227 * Log Headset profile RFCOMM connection event 228 * 229 * @param service_id the BTA service ID for this headset connection 230 */ 231 void LogHeadsetProfileRfcConnection(tBTA_SERVICE_ID service_id); 232 233 /* 234 * Writes the metrics, in base64 protobuf format, into the descriptor FD, 235 * metrics events are always cleared after dump 236 */ 237 void WriteBase64(int fd); 238 void WriteBase64String(std::string* serialized); 239 void WriteString(std::string* serialized); 240 241 /* 242 * Reset the metrics logger by cleaning up its staging queues and existing 243 * protobuf objects. 244 */ 245 void Reset(); 246 247 /* 248 * Maximum number of log entries for each session or event 249 */ 250 static const size_t kMaxNumBluetoothSession = 50; 251 static const size_t kMaxNumPairEvent = 50; 252 static const size_t kMaxNumWakeEvent = 1000; 253 static const size_t kMaxNumScanEvent = 50; 254 255 private: 256 BluetoothMetricsLogger(); 257 258 /* 259 * When a Bluetooth session is on and the user initiates a metrics dump, we 260 * need to be able to upload whatever we have first. This method breaks the 261 * ongoing Bluetooth session into two sessions with the previous one labeled 262 * as "METRICS_DUMP" for the disconnect reason. 263 */ 264 void CutoffSession(); 265 266 /* 267 * Build the internal metrics object using information gathered 268 */ 269 void Build(); 270 271 /* 272 * Reset objects related to current Bluetooth session 273 */ 274 void ResetSession(); 275 276 /* 277 * Reset the underlining BluetoothLog object 278 */ 279 void ResetLog(); 280 281 /* 282 * PIMPL style implementation to hide internal dependencies 283 */ 284 struct impl; 285 std::unique_ptr<impl> const pimpl_; 286 }; 287 288 /** 289 * Unknown connection handle for metrics purpose 290 */ 291 static const uint32_t kUnknownConnectionHandle = 0xFFFF; 292 293 /** 294 * Log link layer connection event 295 * 296 * @param address Stack wide consistent Bluetooth address of this event, 297 * nullptr if unknown 298 * @param connection_handle connection handle of this event, 299 * {@link kUnknownConnectionHandle} if unknown 300 * @param direction direction of this connection 301 * @param link_type type of the link 302 * @param hci_cmd HCI command opecode associated with this event, if any 303 * @param hci_event HCI event code associated with this event, if any 304 * @param hci_ble_event HCI BLE event code associated with this event, if any 305 * @param cmd_status Command status associated with this event, if any 306 * @param reason_code Reason code associated with this event, if any 307 */ 308 void LogLinkLayerConnectionEvent(const RawAddress* address, 309 uint32_t connection_handle, 310 android::bluetooth::DirectionEnum direction, 311 uint16_t link_type, uint32_t hci_cmd, 312 uint16_t hci_event, uint16_t hci_ble_event, 313 uint16_t cmd_status, uint16_t reason_code); 314 315 /** 316 * Logs when Bluetooth controller failed to reply with command status within 317 * a timeout period after receiving an HCI command from the host 318 * 319 * @param hci_cmd opcode of HCI command that caused this timeout 320 */ 321 void LogHciTimeoutEvent(uint32_t hci_cmd); 322 323 /** 324 * Logs when we receive Bluetooth Read Remote Version Information Complete 325 * Event from the remote device, as documented by the Bluetooth Core HCI 326 * specification 327 * 328 * Reference: 5.0 Core Specification, Vol 2, Part E, Page 1118 329 * 330 * @param handle handle of associated ACL connection 331 * @param status HCI command status of this event 332 * @param version version code from read remote version complete event 333 * @param manufacturer_name manufacturer code from read remote version complete 334 * event 335 * @param subversion subversion code from read remote version complete event 336 */ 337 void LogRemoteVersionInfo(uint16_t handle, uint8_t status, uint8_t version, 338 uint16_t manufacturer_name, uint16_t subversion); 339 340 /** 341 * Log A2DP audio buffer underrun event 342 * 343 * @param address A2DP device associated with this event 344 * @param encoding_interval_millis encoding interval in milliseconds 345 * @param num_missing_pcm_bytes number of PCM bytes that cannot be read from 346 * the source 347 */ 348 void LogA2dpAudioUnderrunEvent(const RawAddress& address, 349 uint64_t encoding_interval_millis, 350 int num_missing_pcm_bytes); 351 352 /** 353 * Log A2DP audio buffer overrun event 354 * 355 * @param address A2DP device associated with this event 356 * @param encoding_interval_millis encoding interval in milliseconds 357 * @param num_dropped_buffers number of encoded buffers dropped from Tx queue 358 * @param num_dropped_encoded_frames number of encoded frames dropped from Tx 359 * queue 360 * @param num_dropped_encoded_bytes number of encoded bytes dropped from Tx 361 * queue 362 */ 363 void LogA2dpAudioOverrunEvent(const RawAddress& address, 364 uint64_t encoding_interval_millis, 365 int num_dropped_buffers, 366 int num_dropped_encoded_frames, 367 int num_dropped_encoded_bytes); 368 369 /** 370 * Log A2DP playback state changed event 371 * 372 * @param address A2DP device associated with this event 373 * @param playback_state audio playback state 374 * @param audio_coding_mode audio codec encoding mode 375 */ 376 void LogA2dpPlaybackEvent(const RawAddress& address, int playback_state, 377 int audio_coding_mode); 378 379 /** 380 * Log read RSSI result 381 * 382 * @param address device associated with this event 383 * @param handle connection handle of this event, 384 * {@link kUnknownConnectionHandle} if unknown 385 * @param cmd_status command status from read RSSI command 386 * @param rssi rssi value in dBm 387 */ 388 void LogReadRssiResult(const RawAddress& address, uint16_t handle, 389 uint32_t cmd_status, int8_t rssi); 390 391 /** 392 * Log failed contact counter report 393 * 394 * @param address device associated with this event 395 * @param handle connection handle of this event, 396 * {@link kUnknownConnectionHandle} if unknown 397 * @param cmd_status command status from read failed contact counter command 398 * @param failed_contact_counter Number of consecutive failed contacts for a 399 * connection corresponding to the Handle 400 */ 401 void LogReadFailedContactCounterResult(const RawAddress& address, 402 uint16_t handle, uint32_t cmd_status, 403 int32_t failed_contact_counter); 404 405 /** 406 * Log transmit power level for a particular device after read 407 * 408 * @param address device associated with this event 409 * @param handle connection handle of this event, 410 * {@link kUnknownConnectionHandle} if unknown 411 * @param cmd_status command status from read failed contact counter command 412 * @param transmit_power_level transmit power level for connection to this 413 * device 414 */ 415 void LogReadTxPowerLevelResult(const RawAddress& address, uint16_t handle, 416 uint32_t cmd_status, 417 int32_t transmit_power_level); 418 419 /** 420 * Logs when there is an event related to Bluetooth Security Manager Protocol 421 * 422 * @param address address of associated device 423 * @param smp_cmd SMP command code associated with this event 424 * @param direction direction of this SMP command 425 * @param smp_fail_reason SMP pairing failure reason code from SMP spec 426 */ 427 void LogSmpPairingEvent(const RawAddress& address, uint8_t smp_cmd, 428 android::bluetooth::DirectionEnum direction, 429 uint8_t smp_fail_reason); 430 431 /** 432 * Logs there is an event related Bluetooth classic pairing 433 * 434 * @param address address of associated device 435 * @param handle connection handle of this event, 436 * {@link kUnknownConnectionHandle} if unknown 437 * @param hci_cmd HCI command associated with this event 438 * @param hci_event HCI event associated with this event 439 * @param cmd_status Command status associated with this event 440 * @param reason_code Reason code associated with this event 441 * @param event_value A status value related to this specific event 442 */ 443 void LogClassicPairingEvent(const RawAddress& address, uint16_t handle, uint32_t hci_cmd, uint16_t hci_event, 444 uint16_t cmd_status, uint16_t reason_code, int64_t event_value); 445 446 /** 447 * Logs when certain Bluetooth SDP attributes are discovered 448 * 449 * @param address address of associated device 450 * @param protocol_uuid 16 bit protocol UUID from Bluetooth Assigned Numbers 451 * @param attribute_id 16 bit attribute ID from Bluetooth Assigned Numbers 452 * @param attribute_size size of this attribute 453 * @param attribute_value pointer to the attribute data, must be larger than 454 * attribute_size 455 */ 456 void LogSdpAttribute(const RawAddress& address, uint16_t protocol_uuid, 457 uint16_t attribute_id, size_t attribute_size, 458 const char* attribute_value); 459 460 /** 461 * Logs when there is a change in Bluetooth socket connection state 462 * 463 * @param address address of associated device, empty if this is a server port 464 * @param port port of this socket connection 465 * @param type type of socket 466 * @param connection_state socket connection state 467 * @param tx_bytes number of bytes transmitted 468 * @param rx_bytes number of bytes received 469 * @param server_port server port of this socket, if any. When both 470 * |server_port| and |port| fields are populated, |port| must be spawned 471 * by |server_port| 472 * @param socket_role role of this socket, server or connection 473 * @param uid socket owner's uid 474 */ 475 void LogSocketConnectionState( 476 const RawAddress& address, int port, int type, 477 android::bluetooth::SocketConnectionstateEnum connection_state, 478 int64_t tx_bytes, int64_t rx_bytes, int uid, int server_port, 479 android::bluetooth::SocketRoleEnum socket_role); 480 481 /** 482 * Logs when a Bluetooth device's manufacturer information is learnt 483 * 484 * @param address address of associated device 485 * @param source_type where is this device info obtained from 486 * @param source_name name of the data source, internal or external 487 * @param manufacturer name of the manufacturer of this device 488 * @param model model of this device 489 * @param hardware_version hardware version of this device 490 * @param software_version software version of this device 491 */ 492 void LogManufacturerInfo(const RawAddress& address, 493 android::bluetooth::AddressTypeEnum address_type, 494 android::bluetooth::DeviceInfoSrcEnum source_type, 495 const std::string& source_name, 496 const std::string& manufacturer, 497 const std::string& model, 498 const std::string& hardware_version, 499 const std::string& software_version); 500 501 /** 502 * Logs when received Bluetooth HAL crash reason report. 503 * 504 * @param address current connected address. 505 * @param error_code the crash reason from bluetooth hal 506 * @param vendor_error_code the vendor crash reason from bluetooth Firmware 507 */ 508 void LogBluetoothHalCrashReason(const RawAddress& address, uint32_t error_code, 509 uint32_t vendor_error_code); 510 511 void LogLeAudioConnectionSessionReported( 512 int32_t group_size, int32_t group_metric_id, 513 int64_t connection_duration_nanos, 514 std::vector<int64_t>& device_connecting_offset_nanos, 515 std::vector<int64_t>& device_connected_offset_nanos, 516 std::vector<int64_t>& device_connection_duration_nanos, 517 std::vector<int32_t>& device_connection_status, 518 std::vector<int32_t>& device_disconnection_status, 519 std::vector<RawAddress>& device_address, 520 std::vector<int64_t>& streaming_offset_nanos, 521 std::vector<int64_t>& streaming_duration_nanos, 522 std::vector<int32_t>& streaming_context_type); 523 524 void LogLeAudioBroadcastSessionReported(int64_t duration_nanos); 525 526 void LogLeBluetoothConnectionMetricEventReported( 527 const RawAddress& raw_address, 528 android::bluetooth::le::LeConnectionOriginType origin_type, 529 android::bluetooth::le::LeConnectionType connection_type, 530 android::bluetooth::le::LeConnectionState transaction_state, 531 std::vector<std::pair<os::ArgumentType, int>> 532 argument_list); 533 534 } // namespace common 535 536 } // namespace bluetooth 537