/******************************************************************************
*
* Copyright (C) 2014 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.
*
******************************************************************************/
#define LOG_TAG "bt_hci"
#include "hci_layer.h"
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "btcore/include/module.h"
#include "btsnoop.h"
#include "buffer_allocator.h"
#include "hci_inject.h"
#include "hci_internals.h"
#include "hcidefs.h"
#include "hcimsgs.h"
#include "osi/include/alarm.h"
#include "osi/include/list.h"
#include "osi/include/log.h"
#include "osi/include/properties.h"
#include "osi/include/reactor.h"
#include "packet_fragmenter.h"
#define BT_HCI_TIMEOUT_TAG_NUM 1010000
extern void hci_initialize();
extern void hci_transmit(BT_HDR* packet);
extern void hci_close();
extern int hci_open_firmware_log_file();
extern void hci_close_firmware_log_file(int fd);
extern void hci_log_firmware_debug_packet(int fd, BT_HDR* packet);
static int hci_firmware_log_fd = INVALID_FD;
typedef struct {
uint16_t opcode;
future_t* complete_future;
command_complete_cb complete_callback;
command_status_cb status_callback;
void* context;
BT_HDR* command;
std::chrono::time_point timestamp;
} waiting_command_t;
// Using a define here, because it can be stringified for the property lookup
#define DEFAULT_STARTUP_TIMEOUT_MS 8000
#define STRING_VALUE_OF(x) #x
// RT priority for HCI thread
static const int BT_HCI_RT_PRIORITY = 1;
// Abort if there is no response to an HCI command.
static const uint32_t COMMAND_PENDING_TIMEOUT_MS = 2000;
static const uint32_t COMMAND_TIMEOUT_RESTART_US = 500000;
// Our interface
static bool interface_created;
static hci_t interface;
// Modules we import and callbacks we export
static const allocator_t* buffer_allocator;
static const btsnoop_t* btsnoop;
static const packet_fragmenter_t* packet_fragmenter;
static future_t* startup_future;
static thread_t* thread; // We own this
static std::mutex message_loop_mutex;
static base::MessageLoop* message_loop_ = nullptr;
static base::RunLoop* run_loop_ = nullptr;
static alarm_t* startup_timer;
// Outbound-related
static int command_credits = 1;
static std::mutex command_credits_mutex;
static std::queue command_queue;
// Inbound-related
static alarm_t* command_response_timer;
static list_t* commands_pending_response;
static std::recursive_mutex commands_pending_response_mutex;
// The hand-off point for data going to a higher layer, set by the higher layer
static fixed_queue_t* upwards_data_queue;
static bool filter_incoming_event(BT_HDR* packet);
static waiting_command_t* get_waiting_command(command_opcode_t opcode);
static int get_num_waiting_commands();
static void event_finish_startup(void* context);
static void startup_timer_expired(void* context);
static void enqueue_command(waiting_command_t* wait_entry);
static void event_command_ready(waiting_command_t* wait_entry);
static void enqueue_packet(void* packet);
static void event_packet_ready(void* packet);
static void command_timed_out(void* context);
static void update_command_response_timer(void);
static void transmit_fragment(BT_HDR* packet, bool send_transmit_finished);
static void dispatch_reassembled(BT_HDR* packet);
static void fragmenter_transmit_finished(BT_HDR* packet,
bool all_fragments_sent);
static const packet_fragmenter_callbacks_t packet_fragmenter_callbacks = {
transmit_fragment, dispatch_reassembled, fragmenter_transmit_finished};
void initialization_complete() {
std::lock_guard lock(message_loop_mutex);
message_loop_->task_runner()->PostTask(
FROM_HERE, base::Bind(&event_finish_startup, nullptr));
}
void hci_event_received(BT_HDR* packet) {
btsnoop->capture(packet, true);
if (!filter_incoming_event(packet)) {
data_dispatcher_dispatch(interface.event_dispatcher, packet->data[0],
packet);
}
}
void acl_event_received(BT_HDR* packet) {
btsnoop->capture(packet, true);
packet_fragmenter->reassemble_and_dispatch(packet);
}
void sco_data_received(BT_HDR* packet) {
btsnoop->capture(packet, true);
packet_fragmenter->reassemble_and_dispatch(packet);
}
// Module lifecycle functions
static future_t* hci_module_shut_down();
void message_loop_run(UNUSED_ATTR void* context) {
{
std::lock_guard lock(message_loop_mutex);
message_loop_ = new base::MessageLoop();
run_loop_ = new base::RunLoop();
}
message_loop_->task_runner()->PostTask(FROM_HERE,
base::Bind(&hci_initialize));
run_loop_->Run();
{
std::lock_guard lock(message_loop_mutex);
delete message_loop_;
message_loop_ = nullptr;
delete run_loop_;
run_loop_ = nullptr;
}
}
static future_t* hci_module_start_up(void) {
LOG_INFO(LOG_TAG, "%s", __func__);
// The host is only allowed to send at most one command initially,
// as per the Bluetooth spec, Volume 2, Part E, 4.4 (Command Flow Control)
// This value can change when you get a command complete or command status
// event.
command_credits = 1;
// For now, always use the default timeout on non-Android builds.
period_ms_t startup_timeout_ms = DEFAULT_STARTUP_TIMEOUT_MS;
// Grab the override startup timeout ms, if present.
char timeout_prop[PROPERTY_VALUE_MAX];
if (!osi_property_get("bluetooth.enable_timeout_ms", timeout_prop,
STRING_VALUE_OF(DEFAULT_STARTUP_TIMEOUT_MS)) ||
(startup_timeout_ms = atoi(timeout_prop)) < 100)
startup_timeout_ms = DEFAULT_STARTUP_TIMEOUT_MS;
startup_timer = alarm_new("hci.startup_timer");
if (!startup_timer) {
LOG_ERROR(LOG_TAG, "%s unable to create startup timer.", __func__);
goto error;
}
command_response_timer = alarm_new("hci.command_response_timer");
if (!command_response_timer) {
LOG_ERROR(LOG_TAG, "%s unable to create command response timer.", __func__);
goto error;
}
thread = thread_new("hci_thread");
if (!thread) {
LOG_ERROR(LOG_TAG, "%s unable to create thread.", __func__);
goto error;
}
if (!thread_set_rt_priority(thread, BT_HCI_RT_PRIORITY)) {
LOG_ERROR(LOG_TAG, "%s unable to make thread RT.", __func__);
}
commands_pending_response = list_new(NULL);
if (!commands_pending_response) {
LOG_ERROR(LOG_TAG,
"%s unable to create list for commands pending response.",
__func__);
goto error;
}
// Make sure we run in a bounded amount of time
future_t* local_startup_future;
local_startup_future = future_new();
startup_future = local_startup_future;
alarm_set(startup_timer, startup_timeout_ms, startup_timer_expired, NULL);
packet_fragmenter->init(&packet_fragmenter_callbacks);
thread_post(thread, message_loop_run, NULL);
LOG_DEBUG(LOG_TAG, "%s starting async portion", __func__);
return local_startup_future;
error:
hci_module_shut_down(); // returns NULL so no need to wait for it
return future_new_immediate(FUTURE_FAIL);
}
static future_t* hci_module_shut_down() {
LOG_INFO(LOG_TAG, "%s", __func__);
// Free the timers
{
std::lock_guard lock(commands_pending_response_mutex);
alarm_free(command_response_timer);
command_response_timer = NULL;
alarm_free(startup_timer);
startup_timer = NULL;
}
{
std::lock_guard lock(message_loop_mutex);
message_loop_->task_runner()->PostTask(FROM_HERE, run_loop_->QuitClosure());
}
// Stop the thread to prevent Send() calls.
if (thread) {
thread_stop(thread);
thread_join(thread);
}
// Close HCI to prevent callbacks.
hci_close();
{
std::lock_guard lock(commands_pending_response_mutex);
list_free(commands_pending_response);
commands_pending_response = NULL;
}
packet_fragmenter->cleanup();
thread_free(thread);
thread = NULL;
return NULL;
}
EXPORT_SYMBOL extern const module_t hci_module = {
.name = HCI_MODULE,
.init = NULL,
.start_up = hci_module_start_up,
.shut_down = hci_module_shut_down,
.clean_up = NULL,
.dependencies = {BTSNOOP_MODULE, NULL}};
// Interface functions
static void set_data_queue(fixed_queue_t* queue) { upwards_data_queue = queue; }
static void transmit_command(BT_HDR* command,
command_complete_cb complete_callback,
command_status_cb status_callback, void* context) {
waiting_command_t* wait_entry = reinterpret_cast(
osi_calloc(sizeof(waiting_command_t)));
uint8_t* stream = command->data + command->offset;
STREAM_TO_UINT16(wait_entry->opcode, stream);
wait_entry->complete_callback = complete_callback;
wait_entry->status_callback = status_callback;
wait_entry->command = command;
wait_entry->context = context;
// Store the command message type in the event field
// in case the upper layer didn't already
command->event = MSG_STACK_TO_HC_HCI_CMD;
enqueue_command(wait_entry);
}
static future_t* transmit_command_futured(BT_HDR* command) {
waiting_command_t* wait_entry = reinterpret_cast(
osi_calloc(sizeof(waiting_command_t)));
future_t* future = future_new();
uint8_t* stream = command->data + command->offset;
STREAM_TO_UINT16(wait_entry->opcode, stream);
wait_entry->complete_future = future;
wait_entry->command = command;
// Store the command message type in the event field
// in case the upper layer didn't already
command->event = MSG_STACK_TO_HC_HCI_CMD;
enqueue_command(wait_entry);
return future;
}
static void transmit_downward(data_dispatcher_type_t type, void* data) {
if (type == MSG_STACK_TO_HC_HCI_CMD) {
// TODO(zachoverflow): eliminate this call
transmit_command((BT_HDR*)data, NULL, NULL, NULL);
LOG_WARN(LOG_TAG,
"%s legacy transmit of command. Use transmit_command instead.",
__func__);
} else {
enqueue_packet(data);
}
}
// Start up functions
static void event_finish_startup(UNUSED_ATTR void* context) {
LOG_INFO(LOG_TAG, "%s", __func__);
std::lock_guard lock(commands_pending_response_mutex);
alarm_cancel(startup_timer);
future_ready(startup_future, FUTURE_SUCCESS);
startup_future = NULL;
}
static void startup_timer_expired(UNUSED_ATTR void* context) {
LOG_ERROR(LOG_TAG, "%s", __func__);
std::lock_guard lock(commands_pending_response_mutex);
future_ready(startup_future, FUTURE_FAIL);
startup_future = NULL;
}
// Command/packet transmitting functions
static void enqueue_command(waiting_command_t* wait_entry) {
base::Closure callback = base::Bind(&event_command_ready, wait_entry);
std::lock_guard command_credits_lock(command_credits_mutex);
if (command_credits > 0) {
std::lock_guard message_loop_lock(message_loop_mutex);
if (message_loop_ == nullptr) {
// HCI Layer was shut down
buffer_allocator->free(wait_entry->command);
osi_free(wait_entry);
return;
}
message_loop_->task_runner()->PostTask(FROM_HERE, std::move(callback));
command_credits--;
} else {
command_queue.push(std::move(callback));
}
}
static void event_command_ready(waiting_command_t* wait_entry) {
/// Move it to the list of commands awaiting response
std::lock_guard lock(commands_pending_response_mutex);
wait_entry->timestamp = std::chrono::steady_clock::now();
list_append(commands_pending_response, wait_entry);
// Send it off
packet_fragmenter->fragment_and_dispatch(wait_entry->command);
update_command_response_timer();
}
static void enqueue_packet(void* packet) {
std::lock_guard lock(message_loop_mutex);
if (message_loop_ == nullptr) {
// HCI Layer was shut down
buffer_allocator->free(packet);
return;
}
message_loop_->task_runner()->PostTask(
FROM_HERE, base::Bind(&event_packet_ready, packet));
}
static void event_packet_ready(void* pkt) {
// The queue may be the command queue or the packet queue, we don't care
BT_HDR* packet = (BT_HDR*)pkt;
packet_fragmenter->fragment_and_dispatch(packet);
}
// Callback for the fragmenter to send a fragment
static void transmit_fragment(BT_HDR* packet, bool send_transmit_finished) {
btsnoop->capture(packet, false);
hci_transmit(packet);
uint16_t event = packet->event & MSG_EVT_MASK;
if (event != MSG_STACK_TO_HC_HCI_CMD && send_transmit_finished)
buffer_allocator->free(packet);
}
static void fragmenter_transmit_finished(BT_HDR* packet,
bool all_fragments_sent) {
if (all_fragments_sent) {
buffer_allocator->free(packet);
} else {
// This is kind of a weird case, since we're dispatching a partially sent
// packet up to a higher layer.
// TODO(zachoverflow): rework upper layer so this isn't necessary.
data_dispatcher_dispatch(interface.event_dispatcher,
packet->event & MSG_EVT_MASK, packet);
}
}
// Print debugging information and quit. Don't dereference original_wait_entry.
static void command_timed_out(void* original_wait_entry) {
std::unique_lock lock(commands_pending_response_mutex);
LOG_ERROR(LOG_TAG, "%s: %d commands pending response", __func__,
get_num_waiting_commands());
for (const list_node_t* node = list_begin(commands_pending_response);
node != list_end(commands_pending_response); node = list_next(node)) {
waiting_command_t* wait_entry =
reinterpret_cast(list_node(node));
int wait_time_ms =
std::chrono::duration_cast(
std::chrono::steady_clock::now() - wait_entry->timestamp)
.count();
LOG_ERROR(LOG_TAG, "%s: Waited %d ms for a response to opcode: 0x%x %s",
__func__, wait_time_ms, wait_entry->opcode,
(wait_entry == original_wait_entry) ? "*matches timer*" : "");
// Dump the length field and the first byte of the payload, if present.
uint8_t* command = wait_entry->command->data + wait_entry->command->offset;
if (wait_entry->command->len > 3) {
LOG_ERROR(LOG_TAG, "%s: Size %d Hex %02x %02x %02x %02x", __func__,
wait_entry->command->len, command[0], command[1], command[2],
command[3]);
} else {
LOG_ERROR(LOG_TAG, "%s: Size %d Hex %02x %02x %02x", __func__,
wait_entry->command->len, command[0], command[1], command[2]);
}
LOG_EVENT_INT(BT_HCI_TIMEOUT_TAG_NUM, wait_entry->opcode);
}
lock.unlock();
LOG_ERROR(LOG_TAG, "%s: requesting a firmware dump.", __func__);
/* Allocate a buffer to hold the HCI command. */
BT_HDR* bt_hdr =
static_cast(osi_malloc(sizeof(BT_HDR) + HCIC_PREAMBLE_SIZE));
bt_hdr->len = HCIC_PREAMBLE_SIZE;
bt_hdr->event = MSG_STACK_TO_HC_HCI_CMD;
bt_hdr->offset = 0;
uint8_t* hci_packet = reinterpret_cast(bt_hdr + 1);
UINT16_TO_STREAM(hci_packet,
HCI_GRP_VENDOR_SPECIFIC | HCI_CONTROLLER_DEBUG_INFO_OCF);
UINT8_TO_STREAM(hci_packet, 0); // No parameters
hci_firmware_log_fd = hci_open_firmware_log_file();
transmit_fragment(bt_hdr, true);
osi_free(bt_hdr);
LOG_ERROR(LOG_TAG, "%s restarting the Bluetooth process.", __func__);
usleep(COMMAND_TIMEOUT_RESTART_US);
hci_close_firmware_log_file(hci_firmware_log_fd);
// We shouldn't try to recover the stack from this command timeout.
// If it's caused by a software bug, fix it. If it's a hardware bug, fix it.
abort();
}
// Event/packet receiving functions
void process_command_credits(int credits) {
std::lock_guard command_credits_lock(command_credits_mutex);
std::lock_guard message_loop_lock(message_loop_mutex);
if (message_loop_ == nullptr) {
// HCI Layer was shut down
return;
}
// Subtract commands in flight.
command_credits = credits - get_num_waiting_commands();
while (command_credits > 0 && command_queue.size() > 0) {
message_loop_->task_runner()->PostTask(FROM_HERE,
std::move(command_queue.front()));
command_queue.pop();
command_credits--;
}
}
// Returns true if the event was intercepted and should not proceed to
// higher layers. Also inspects an incoming event for interesting
// information, like how many commands are now able to be sent.
static bool filter_incoming_event(BT_HDR* packet) {
waiting_command_t* wait_entry = NULL;
uint8_t* stream = packet->data;
uint8_t event_code;
int credits = 0;
command_opcode_t opcode;
STREAM_TO_UINT8(event_code, stream);
STREAM_SKIP_UINT8(stream); // Skip the parameter total length field
if (event_code == HCI_COMMAND_COMPLETE_EVT) {
STREAM_TO_UINT8(credits, stream);
STREAM_TO_UINT16(opcode, stream);
wait_entry = get_waiting_command(opcode);
process_command_credits(credits);
if (!wait_entry) {
if (opcode != HCI_COMMAND_NONE) {
LOG_WARN(LOG_TAG,
"%s command complete event with no matching command (opcode: "
"0x%04x).",
__func__, opcode);
}
} else {
update_command_response_timer();
if (wait_entry->complete_callback) {
wait_entry->complete_callback(packet, wait_entry->context);
} else if (wait_entry->complete_future) {
future_ready(wait_entry->complete_future, packet);
}
}
goto intercepted;
} else if (event_code == HCI_COMMAND_STATUS_EVT) {
uint8_t status;
STREAM_TO_UINT8(status, stream);
STREAM_TO_UINT8(credits, stream);
STREAM_TO_UINT16(opcode, stream);
// If a command generates a command status event, it won't be getting a
// command complete event
wait_entry = get_waiting_command(opcode);
process_command_credits(credits);
if (!wait_entry) {
LOG_WARN(
LOG_TAG,
"%s command status event with no matching command. opcode: 0x%04x",
__func__, opcode);
} else {
update_command_response_timer();
if (wait_entry->status_callback)
wait_entry->status_callback(status, wait_entry->command,
wait_entry->context);
}
goto intercepted;
} else if (event_code == HCI_VSE_SUBCODE_DEBUG_INFO_SUB_EVT) {
if (hci_firmware_log_fd == INVALID_FD)
hci_firmware_log_fd = hci_open_firmware_log_file();
if (hci_firmware_log_fd != INVALID_FD)
hci_log_firmware_debug_packet(hci_firmware_log_fd, packet);
buffer_allocator->free(packet);
return true;
}
return false;
intercepted:
if (wait_entry) {
// If it has a callback, it's responsible for freeing the packet
if (event_code == HCI_COMMAND_STATUS_EVT ||
(!wait_entry->complete_callback && !wait_entry->complete_future))
buffer_allocator->free(packet);
// If it has a callback, it's responsible for freeing the command
if (event_code == HCI_COMMAND_COMPLETE_EVT || !wait_entry->status_callback)
buffer_allocator->free(wait_entry->command);
osi_free(wait_entry);
} else {
buffer_allocator->free(packet);
}
return true;
}
// Callback for the fragmenter to dispatch up a completely reassembled packet
static void dispatch_reassembled(BT_HDR* packet) {
// Events should already have been dispatched before this point
CHECK((packet->event & MSG_EVT_MASK) != MSG_HC_TO_STACK_HCI_EVT);
CHECK(upwards_data_queue != NULL);
fixed_queue_enqueue(upwards_data_queue, packet);
}
// Misc internal functions
static waiting_command_t* get_waiting_command(command_opcode_t opcode) {
std::lock_guard lock(commands_pending_response_mutex);
for (const list_node_t* node = list_begin(commands_pending_response);
node != list_end(commands_pending_response); node = list_next(node)) {
waiting_command_t* wait_entry =
reinterpret_cast(list_node(node));
if (!wait_entry || wait_entry->opcode != opcode) continue;
list_remove(commands_pending_response, wait_entry);
return wait_entry;
}
return NULL;
}
static int get_num_waiting_commands() {
std::lock_guard lock(commands_pending_response_mutex);
return list_length(commands_pending_response);
}
static void update_command_response_timer(void) {
std::lock_guard lock(commands_pending_response_mutex);
if (command_response_timer == NULL) return;
if (list_is_empty(commands_pending_response)) {
alarm_cancel(command_response_timer);
} else {
alarm_set(command_response_timer, COMMAND_PENDING_TIMEOUT_MS,
command_timed_out, list_front(commands_pending_response));
}
}
static void init_layer_interface() {
if (!interface_created) {
// It's probably ok for this to live forever. It's small and
// there's only one instance of the hci interface.
interface.event_dispatcher = data_dispatcher_new("hci_layer");
if (!interface.event_dispatcher) {
LOG_ERROR(LOG_TAG, "%s could not create upward dispatcher.", __func__);
return;
}
interface.set_data_queue = set_data_queue;
interface.transmit_command = transmit_command;
interface.transmit_command_futured = transmit_command_futured;
interface.transmit_downward = transmit_downward;
interface_created = true;
}
}
void hci_layer_cleanup_interface() {
if (interface_created) {
data_dispatcher_free(interface.event_dispatcher);
interface.event_dispatcher = NULL;
interface.set_data_queue = NULL;
interface.transmit_command = NULL;
interface.transmit_command_futured = NULL;
interface.transmit_downward = NULL;
interface_created = false;
}
}
const hci_t* hci_layer_get_interface() {
buffer_allocator = buffer_allocator_get_interface();
btsnoop = btsnoop_get_interface();
packet_fragmenter = packet_fragmenter_get_interface();
init_layer_interface();
return &interface;
}
const hci_t* hci_layer_get_test_interface(
const allocator_t* buffer_allocator_interface,
const btsnoop_t* btsnoop_interface,
const packet_fragmenter_t* packet_fragmenter_interface) {
buffer_allocator = buffer_allocator_interface;
btsnoop = btsnoop_interface;
packet_fragmenter = packet_fragmenter_interface;
init_layer_interface();
return &interface;
}