/******************************************************************************
*
* Copyright (C) 2016 Google, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
******************************************************************************/
#pragma once
#include <stdint.h>
#include <memory>
#include <string>
namespace system_bt_osi {
// Typedefs to hide protobuf definition to the rest of stack
typedef enum {
DEVICE_TYPE_UNKNOWN,
DEVICE_TYPE_BREDR,
DEVICE_TYPE_LE,
DEVICE_TYPE_DUMO,
} device_type_t;
typedef enum {
WAKE_EVENT_UNKNOWN,
WAKE_EVENT_ACQUIRED,
WAKE_EVENT_RELEASED,
} wake_event_type_t;
typedef enum {
SCAN_TYPE_UNKNOWN,
SCAN_TECH_TYPE_LE,
SCAN_TECH_TYPE_BREDR,
SCAN_TECH_TYPE_BOTH,
} scan_tech_t;
typedef enum {
CONNECTION_TECHNOLOGY_TYPE_UNKNOWN,
CONNECTION_TECHNOLOGY_TYPE_LE,
CONNECTION_TECHNOLOGY_TYPE_BREDR,
} connection_tech_t;
typedef enum {
DISCONNECT_REASON_UNKNOWN,
DISCONNECT_REASON_METRICS_DUMP,
DISCONNECT_REASON_NEXT_START_WITHOUT_END_PREVIOUS,
} disconnect_reason_t;
/* Values of A2DP metrics that we care about
*
* audio_duration_ms : sum of audio duration (in milliseconds).
* device_class: device class of the paired device.
* media_timer_min_ms : minimum scheduled time (in milliseconds)
* of the media timer.
* media_timer_max_ms: maximum scheduled time (in milliseconds)
* of the media timer.
* media_timer_avg_ms: average scheduled time (in milliseconds)
* of the media timer.
* buffer_overruns_max_count: TODO - not clear what this is.
* buffer_overruns_total : number of times the media buffer with
* audio data has overrun
* buffer_underruns_average: TODO - not clear what this is.
* buffer_underruns_count: number of times there was no enough
* audio data to add to the media buffer.
* NOTE: Negative values are invalid
*/
class A2dpSessionMetrics {
public:
A2dpSessionMetrics() {}
/*
* Update the metrics value in the current metrics object using the metrics
* objects supplied
*/
void Update(const A2dpSessionMetrics& metrics);
/*
* Compare whether two metrics objects are equal
*/
bool operator==(const A2dpSessionMetrics& rhs) const;
/*
* Initialize all values to -1 which is invalid in order to make a distinction
* between 0 and invalid values
*/
int64_t audio_duration_ms = -1;
int32_t media_timer_min_ms = -1;
int32_t media_timer_max_ms = -1;
int32_t media_timer_avg_ms = -1;
int64_t total_scheduling_count = -1;
int32_t buffer_overruns_max_count = -1;
int32_t buffer_overruns_total = -1;
float buffer_underruns_average = -1;
int32_t buffer_underruns_count = -1;
};
class BluetoothMetricsLogger {
public:
static BluetoothMetricsLogger* GetInstance() {
static BluetoothMetricsLogger* instance = new BluetoothMetricsLogger();
return instance;
}
/*
* Record a pairing event
*
* Parameters:
* timestamp_ms: Unix epoch time in milliseconds
* device_class: class of remote device
* device_type: type of remote device
* disconnect_reason: HCI reason for pairing disconnection.
* See: stack/include/hcidefs.h
*/
void LogPairEvent(uint32_t disconnect_reason, uint64_t timestamp_ms,
uint32_t device_class, device_type_t device_type);
/*
* Record a wake event
*
* Parameters:
* timestamp_ms: Unix epoch time in milliseconds
* type: whether it was acquired or released
* requestor: if provided is the service requesting the wake lock
* name: the name of the wake lock held
*/
void LogWakeEvent(wake_event_type_t type, const std::string& requestor,
const std::string& name, uint64_t timestamp_ms);
/*
* Record a scan event
*
* Parameters
* timestamp_ms : Unix epoch time in milliseconds
* start : true if this is the beginning of the scan
* initiator: a unique ID identifying the app starting the scan
* type: whether the scan reports BR/EDR, LE, or both.
* results: number of results to be reported.
*/
void LogScanEvent(bool start, const std::string& initator, scan_tech_t type,
uint32_t results, uint64_t timestamp_ms);
/*
* Start logging a Bluetooth session
*
* A Bluetooth session is defined a a connection between this device and
* another remote device which may include multiple profiles and protocols
*
* Only one Bluetooth session can exist at one time. Calling this method twice
* without LogBluetoothSessionEnd will result in logging a premature end of
* current Bluetooth session
*
* Parameters:
* connection_tech_type : type of connection technology
* timestamp_ms : the timestamp for session start, 0 means now
*
*/
void LogBluetoothSessionStart(connection_tech_t connection_tech_type,
uint64_t timestamp_ms);
/*
* Stop logging a Bluetooth session and pushes it to the log queue
*
* If no Bluetooth session exist, this method exits immediately
*
* Parameters:
* disconnect_reason : A string representation of disconnect reason
* timestamp_ms : the timestamp of session end, 0 means now
*
*/
void LogBluetoothSessionEnd(disconnect_reason_t disconnect_reason,
uint64_t timestamp_ms);
/*
* Log information about remote device in a current Bluetooth session
*
* If a Bluetooth session does not exist, create one with default parameter
* and timestamp now
*
* Parameters:
* device_class : device_class defined in btm_api_types.h
* device_type : type of remote device
*/
void LogBluetoothSessionDeviceInfo(uint32_t device_class,
device_type_t device_type);
/*
* Log A2DP Audio Session Information
*
* - Repeated calls to this method will override previous metrics if in the
* same Bluetooth connection
* - If a Bluetooth session does not exist, create one with default parameter
* and timestamp now
*
* Parameters:
* a2dp_session_metrics - pointer to struct holding a2dp stats
*
*/
void LogA2dpSession(const A2dpSessionMetrics& a2dp_session_metrics);
/*
* Writes the metrics, in base64 protobuf format, into the descriptor FD
* If CLEAR is true, metrics events are cleared afterwards.
*/
void WriteBase64(int fd, bool clear);
void WriteBase64String(std::string* serialized, bool clear);
void WriteString(std::string* serialized, bool clear);
/*
* Reset the metrics logger by cleaning up its staging queues and existing
* protobuf objects.
*/
void Reset();
/*
* Maximum number of log entries for each session or event
*/
static const size_t kMaxNumBluetoothSession = 50;
static const size_t kMaxNumPairEvent = 50;
static const size_t kMaxNumWakeEvent = 1000;
static const size_t kMaxNumScanEvent = 50;
private:
BluetoothMetricsLogger();
/*
* When a Bluetooth session is on and the user initiates a metrics dump, we
* need to be able to upload whatever we have first. This method breaks the
* ongoing Bluetooth session into two sessions with the previous one labeled
* as "METRICS_DUMP" for the disconnect reason.
*/
void CutoffSession();
/*
* Build the internal metrics object using information gathered
*/
void Build();
/*
* Reset objects related to current Bluetooth session
*/
void ResetSession();
/*
* Reset the underlining BluetoothLog object
*/
void ResetLog();
/*
* PIMPL style implementation to hide internal dependencies
*/
struct impl;
std::unique_ptr<impl> const pimpl_;
};
}