/******************************************************************************
*
* Copyright 2015 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.
*
******************************************************************************/
#include <stdio.h>
#include <string.h>
#include <time.h>
#include "btif/include/btif_debug_conn.h"
#include "common/time_util.h"
#define NUM_CONNECTION_EVENTS 16
#define TEMP_BUFFER_SIZE 30
typedef struct conn_event_t {
uint64_t ts;
btif_debug_conn_state_t state;
RawAddress bda;
tGATT_DISCONN_REASON disconnect_reason;
} conn_event_t;
static conn_event_t connection_events[NUM_CONNECTION_EVENTS];
static uint8_t current_event = 0;
static char* format_ts(const uint64_t ts, char* buffer, int len) {
const uint64_t ms = ts / 1000;
const time_t secs = ms / 1000;
struct tm* ptm = localtime(&secs);
char tempbuff[20];
strftime(tempbuff, sizeof(tempbuff), "%m-%d %H:%M:%S", ptm);
snprintf(buffer, len, "%s.%03u", tempbuff, (uint16_t)(ms % 1000));
return buffer;
}
static const char* format_state(const btif_debug_conn_state_t state) {
switch (state) {
case BTIF_DEBUG_CONNECTED:
return "CONNECTED ";
case BTIF_DEBUG_DISCONNECTED:
return "DISCONNECTED";
}
return "UNKNOWN";
}
static void next_event() {
++current_event;
if (current_event == NUM_CONNECTION_EVENTS) current_event = 0;
}
void btif_debug_conn_state(const RawAddress& bda,
const btif_debug_conn_state_t state,
const tGATT_DISCONN_REASON disconnect_reason) {
next_event();
conn_event_t* evt = &connection_events[current_event];
evt->ts = bluetooth::common::time_gettimeofday_us();
evt->state = state;
evt->disconnect_reason = disconnect_reason;
evt->bda = bda;
}
void btif_debug_conn_dump(int fd) {
const uint8_t current_event_local =
current_event; // Cache to avoid threading issues
uint8_t dump_event = current_event_local;
char ts_buffer[TEMP_BUFFER_SIZE] = {0};
dprintf(fd, "\nConnection Events:\n");
if (connection_events[dump_event].ts == 0) dprintf(fd, " None\n");
while (connection_events[dump_event].ts) {
conn_event_t* evt = &connection_events[dump_event];
dprintf(fd, " %s %s %s", format_ts(evt->ts, ts_buffer, sizeof(ts_buffer)),
format_state(evt->state), evt->bda.ToString().c_str());
if (evt->state == BTIF_DEBUG_DISCONNECTED)
dprintf(fd, " reason=%d", evt->disconnect_reason);
dprintf(fd, "\n");
// Go to previous event; wrap if needed
if (dump_event > 0)
--dump_event;
else
dump_event = NUM_CONNECTION_EVENTS - 1;
// Check if we dumped all events
if (dump_event == current_event_local) break;
}
}