2016-09-26 13:37:39 +00:00
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
|
2018-04-08 04:10:50 +00:00
|
|
|
#include "common/bt_defs.h"
|
|
|
|
|
|
|
|
#include "osi/allocator.h"
|
|
|
|
#include "stack/bt_types.h"
|
|
|
|
#include "stack/hcidefs.h"
|
|
|
|
#include "stack/hcimsgs.h"
|
|
|
|
#include "hci/hci_internals.h"
|
|
|
|
#include "hci/hci_layer.h"
|
|
|
|
#include "hci/hci_packet_factory.h"
|
2016-09-26 13:37:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
static BT_HDR *make_packet(size_t data_size);
|
|
|
|
static BT_HDR *make_command_no_params(uint16_t opcode);
|
|
|
|
static BT_HDR *make_command(uint16_t opcode, size_t parameter_size, uint8_t **stream_out);
|
|
|
|
|
|
|
|
// Interface functions
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_reset(void)
|
|
|
|
{
|
|
|
|
return make_command_no_params(HCI_RESET);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_read_buffer_size(void)
|
|
|
|
{
|
|
|
|
return make_command_no_params(HCI_READ_BUFFER_SIZE);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2018-06-21 03:41:42 +00:00
|
|
|
static BT_HDR *make_set_c2h_flow_control(uint8_t enable)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
const uint8_t parameter_size = 1;
|
|
|
|
BT_HDR *packet = make_command(HCI_SET_HC_TO_HOST_FLOW_CTRL, parameter_size, &stream);
|
|
|
|
|
|
|
|
UINT8_TO_STREAM(stream, enable);
|
|
|
|
return packet;
|
|
|
|
}
|
|
|
|
|
2019-03-01 13:59:55 +00:00
|
|
|
static BT_HDR *make_set_adv_report_flow_control(uint8_t enable, uint16_t num, uint16_t lost_threshold)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
const uint8_t parameter_size = 1 + 2 + 2;
|
|
|
|
BT_HDR *packet = make_command(HCI_VENDOR_BLE_SET_ADV_FLOW_CONTROL, parameter_size, &stream);
|
|
|
|
|
|
|
|
UINT8_TO_STREAM(stream, enable);
|
|
|
|
UINT16_TO_STREAM(stream, num);
|
|
|
|
UINT16_TO_STREAM(stream, lost_threshold);
|
|
|
|
return packet;
|
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_host_buffer_size(uint16_t acl_size, uint8_t sco_size, uint16_t acl_count, uint16_t sco_count)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
const uint8_t parameter_size = 2 + 1 + 2 + 2; // from each of the parameters
|
|
|
|
BT_HDR *packet = make_command(HCI_HOST_BUFFER_SIZE, parameter_size, &stream);
|
2016-09-26 13:37:39 +00:00
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
UINT16_TO_STREAM(stream, acl_size);
|
|
|
|
UINT8_TO_STREAM(stream, sco_size);
|
|
|
|
UINT16_TO_STREAM(stream, acl_count);
|
|
|
|
UINT16_TO_STREAM(stream, sco_count);
|
|
|
|
return packet;
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_read_local_version_info(void)
|
|
|
|
{
|
|
|
|
return make_command_no_params(HCI_READ_LOCAL_VERSION_INFO);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_read_bd_addr(void)
|
|
|
|
{
|
|
|
|
return make_command_no_params(HCI_READ_BD_ADDR);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_read_local_supported_commands(void)
|
|
|
|
{
|
|
|
|
return make_command_no_params(HCI_READ_LOCAL_SUPPORTED_CMDS);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_read_local_extended_features(uint8_t page_number)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
const uint8_t parameter_size = 1;
|
|
|
|
BT_HDR *packet = make_command(HCI_READ_LOCAL_EXT_FEATURES, parameter_size, &stream);
|
2016-09-26 13:37:39 +00:00
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
UINT8_TO_STREAM(stream, page_number);
|
|
|
|
return packet;
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_write_simple_pairing_mode(uint8_t mode)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
const uint8_t parameter_size = 1;
|
|
|
|
BT_HDR *packet = make_command(HCI_WRITE_SIMPLE_PAIRING_MODE, parameter_size, &stream);
|
2016-09-26 13:37:39 +00:00
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
UINT8_TO_STREAM(stream, mode);
|
|
|
|
return packet;
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_write_secure_connections_host_support(uint8_t mode)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
const uint8_t parameter_size = 1;
|
|
|
|
BT_HDR *packet = make_command(HCI_WRITE_SECURE_CONNS_SUPPORT, parameter_size, &stream);
|
2016-09-26 13:37:39 +00:00
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
UINT8_TO_STREAM(stream, mode);
|
|
|
|
return packet;
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_set_event_mask(const bt_event_mask_t *event_mask)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
uint8_t parameter_size = sizeof(bt_event_mask_t);
|
|
|
|
BT_HDR *packet = make_command(HCI_SET_EVENT_MASK, parameter_size, &stream);
|
2016-09-26 13:37:39 +00:00
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
ARRAY8_TO_STREAM(stream, event_mask->as_array);
|
|
|
|
return packet;
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_ble_write_host_support(uint8_t supported_host, uint8_t simultaneous_host)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
const uint8_t parameter_size = 1 + 1;
|
|
|
|
BT_HDR *packet = make_command(HCI_WRITE_LE_HOST_SUPPORT, parameter_size, &stream);
|
2016-09-26 13:37:39 +00:00
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
UINT8_TO_STREAM(stream, supported_host);
|
|
|
|
UINT8_TO_STREAM(stream, simultaneous_host);
|
|
|
|
return packet;
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_ble_read_white_list_size(void)
|
|
|
|
{
|
|
|
|
return make_command_no_params(HCI_BLE_READ_WHITE_LIST_SIZE);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_ble_read_buffer_size(void)
|
|
|
|
{
|
|
|
|
return make_command_no_params(HCI_BLE_READ_BUFFER_SIZE);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_ble_read_supported_states(void)
|
|
|
|
{
|
|
|
|
return make_command_no_params(HCI_BLE_READ_SUPPORTED_STATES);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_ble_read_local_supported_features(void)
|
|
|
|
{
|
|
|
|
return make_command_no_params(HCI_BLE_READ_LOCAL_SPT_FEAT);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_ble_read_resolving_list_size(void)
|
|
|
|
{
|
|
|
|
return make_command_no_params(HCI_BLE_READ_RESOLVING_LIST_SIZE);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_ble_read_suggested_default_data_length(void)
|
|
|
|
{
|
2016-09-26 13:37:39 +00:00
|
|
|
return make_command_no_params(HCI_BLE_READ_DEFAULT_DATA_LENGTH);
|
|
|
|
}
|
|
|
|
|
2017-02-16 06:10:44 +00:00
|
|
|
static BT_HDR *make_ble_write_suggested_default_data_length(uint16_t SuggestedMaxTxOctets, uint16_t SuggestedMaxTxTime)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
uint8_t parameter_size = sizeof(uint16_t) + sizeof(uint16_t);
|
|
|
|
BT_HDR *packet = make_command(HCI_BLE_WRITE_DEFAULT_DATA_LENGTH, parameter_size, &stream);
|
|
|
|
|
|
|
|
UINT16_TO_STREAM(stream, SuggestedMaxTxOctets);
|
|
|
|
UINT16_TO_STREAM(stream, SuggestedMaxTxTime);
|
|
|
|
return packet;
|
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_ble_set_event_mask(const bt_event_mask_t *event_mask)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
uint8_t parameter_size = sizeof(bt_event_mask_t);
|
|
|
|
BT_HDR *packet = make_command(HCI_BLE_SET_EVENT_MASK, parameter_size, &stream);
|
2016-09-26 13:37:39 +00:00
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
ARRAY8_TO_STREAM(stream, event_mask->as_array);
|
|
|
|
return packet;
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2018-05-21 03:33:30 +00:00
|
|
|
static BT_HDR *make_write_sync_flow_control_enable(uint8_t enable)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
const uint8_t parameter_size = 1;
|
|
|
|
BT_HDR *packet = make_command(HCI_WRITE_SCO_FLOW_CTRL_ENABLE, parameter_size, &stream);
|
|
|
|
|
|
|
|
UINT8_TO_STREAM(stream, enable);
|
|
|
|
return packet;
|
|
|
|
}
|
2018-07-05 07:22:44 +00:00
|
|
|
|
|
|
|
static BT_HDR *make_write_default_erroneous_data_report(uint8_t enable)
|
|
|
|
{
|
|
|
|
uint8_t *stream;
|
|
|
|
const uint8_t parameter_size = 1;
|
|
|
|
BT_HDR *packet = make_command(HCI_WRITE_ERRONEOUS_DATA_RPT, parameter_size, &stream);
|
|
|
|
|
|
|
|
UINT8_TO_STREAM(stream, enable);
|
|
|
|
return packet;
|
|
|
|
}
|
2016-09-26 13:37:39 +00:00
|
|
|
// Internal functions
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_command_no_params(uint16_t opcode)
|
|
|
|
{
|
|
|
|
return make_command(opcode, 0, NULL);
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_command(uint16_t opcode, size_t parameter_size, uint8_t **stream_out)
|
|
|
|
{
|
|
|
|
BT_HDR *packet = make_packet(HCI_COMMAND_PREAMBLE_SIZE + parameter_size);
|
2016-09-26 13:37:39 +00:00
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
uint8_t *stream = packet->data;
|
|
|
|
UINT16_TO_STREAM(stream, opcode);
|
|
|
|
UINT8_TO_STREAM(stream, parameter_size);
|
2016-09-26 13:37:39 +00:00
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
if (stream_out != NULL) {
|
|
|
|
*stream_out = stream;
|
|
|
|
}
|
2016-09-26 13:37:39 +00:00
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
return packet;
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
static BT_HDR *make_packet(size_t data_size)
|
|
|
|
{
|
2018-08-15 02:40:51 +00:00
|
|
|
BT_HDR *ret = (BT_HDR *)osi_calloc(sizeof(BT_HDR) + data_size);
|
2016-11-24 18:10:15 +00:00
|
|
|
assert(ret);
|
|
|
|
ret->event = 0;
|
|
|
|
ret->offset = 0;
|
|
|
|
ret->layer_specific = 0;
|
|
|
|
ret->len = data_size;
|
|
|
|
return ret;
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static const hci_packet_factory_t interface = {
|
2016-11-24 18:10:15 +00:00
|
|
|
make_reset,
|
|
|
|
make_read_buffer_size,
|
2018-06-21 03:41:42 +00:00
|
|
|
make_set_c2h_flow_control,
|
2019-03-01 13:59:55 +00:00
|
|
|
make_set_adv_report_flow_control,
|
2016-11-24 18:10:15 +00:00
|
|
|
make_host_buffer_size,
|
|
|
|
make_read_local_version_info,
|
|
|
|
make_read_bd_addr,
|
|
|
|
make_read_local_supported_commands,
|
|
|
|
make_read_local_extended_features,
|
|
|
|
make_write_simple_pairing_mode,
|
|
|
|
make_write_secure_connections_host_support,
|
|
|
|
make_set_event_mask,
|
|
|
|
make_ble_write_host_support,
|
|
|
|
make_ble_read_white_list_size,
|
|
|
|
make_ble_read_buffer_size,
|
|
|
|
make_ble_read_supported_states,
|
|
|
|
make_ble_read_local_supported_features,
|
|
|
|
make_ble_read_resolving_list_size,
|
|
|
|
make_ble_read_suggested_default_data_length,
|
2017-02-16 06:10:44 +00:00
|
|
|
make_ble_write_suggested_default_data_length,
|
2018-05-21 03:33:30 +00:00
|
|
|
make_ble_set_event_mask,
|
2018-07-05 07:22:44 +00:00
|
|
|
make_write_sync_flow_control_enable,
|
|
|
|
make_write_default_erroneous_data_report,
|
2016-09-26 13:37:39 +00:00
|
|
|
};
|
|
|
|
|
2016-11-24 18:10:15 +00:00
|
|
|
const hci_packet_factory_t *hci_packet_factory_get_interface()
|
|
|
|
{
|
|
|
|
return &interface;
|
2016-09-26 13:37:39 +00:00
|
|
|
}
|