2017-10-12 12:58:41 +00:00
|
|
|
/* mbed Microcontroller Library
|
|
|
|
* Copyright (c) 2017-2017 ARM Limited
|
|
|
|
*
|
|
|
|
* 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 <algorithm>
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "ble/BLEProtocol.h"
|
|
|
|
#include "ble/Gap.h"
|
|
|
|
#include "ble/pal/PalGap.h"
|
|
|
|
#include "ble/pal/GapEvents.h"
|
|
|
|
#include "ble/pal/GapTypes.h"
|
|
|
|
#include "ble/pal/GenericAccessService.h"
|
|
|
|
#include "ble/generic/GenericGap.h"
|
|
|
|
|
|
|
|
#include "drivers/Timeout.h"
|
|
|
|
|
|
|
|
namespace ble {
|
|
|
|
namespace generic {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Constants
|
|
|
|
static const uint16_t scan_interval_min = 0x0004;
|
|
|
|
static const uint16_t scan_interval_max = 0x4000;
|
|
|
|
static const uint16_t connection_interval_min = 0x0006;
|
|
|
|
static const uint16_t connection_interval_max = 0x0C80;
|
|
|
|
static const uint16_t slave_latency_min = 0x0000;
|
|
|
|
static const uint16_t slave_latency_max = 0x01F3;
|
|
|
|
static const uint16_t advertising_interval_min = 0x0020;
|
|
|
|
static const uint16_t advertising_interval_max = 0x4000;
|
|
|
|
static const uint16_t supervision_timeout_min = 0x000A;
|
|
|
|
static const uint16_t supervision_timeout_max = 0x0C80;
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true if value is included in the range [lower_bound : higher_bound]
|
|
|
|
*/
|
|
|
|
template<typename T>
|
|
|
|
static bool is_in_range(T value, T lower_bound, T higher_bound) {
|
|
|
|
if (value < lower_bound || value > higher_bound) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true if the scan parameters are valid or false otherwise.
|
|
|
|
*/
|
|
|
|
static bool is_scan_params_valid(const GapScanningParams* params)
|
|
|
|
{
|
|
|
|
if (params == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_in_range(params->getInterval(), scan_interval_min, scan_interval_max) == false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_in_range(params->getWindow(), scan_interval_min, params->getInterval()) == false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true if the connection parameters are valid or false otherwise.
|
|
|
|
*/
|
|
|
|
static bool is_connection_params_valid(const Gap::ConnectionParams_t* params)
|
|
|
|
{
|
|
|
|
if (params == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_in_range(params->slaveLatency, slave_latency_min, slave_latency_max) == false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_in_range(params->maxConnectionInterval, connection_interval_min, connection_interval_max) == false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_in_range(params->minConnectionInterval, connection_interval_min, params->maxConnectionInterval) == false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_in_range(params->connectionSupervisionTimeout, supervision_timeout_min, supervision_timeout_max) == false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t max_connection_interval_ms =
|
|
|
|
((uint32_t)params->maxConnectionInterval * 125) / 100;
|
|
|
|
uint16_t min_connection_supervision_timeout =
|
|
|
|
((1 + params->slaveLatency) * max_connection_interval_ms * 2) / 10;
|
|
|
|
|
|
|
|
if (params->connectionSupervisionTimeout < min_connection_supervision_timeout) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true of the connection parameters are acceptable as preferred connection
|
|
|
|
* parameters.
|
|
|
|
*
|
|
|
|
* Prefered connection parameters unlike actual connection parameters allow the
|
|
|
|
* max connection interval, min connection interval and connection supervision
|
|
|
|
* timeout to be equal to 0xFFFF. When it is the case that value can be
|
|
|
|
* interpreted as "non specific".
|
|
|
|
*/
|
|
|
|
static bool is_preferred_connection_params_valid(const Gap::ConnectionParams_t* params)
|
|
|
|
{
|
|
|
|
if (params == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_in_range(params->slaveLatency, slave_latency_min, slave_latency_max) == false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((is_in_range(params->maxConnectionInterval, connection_interval_min, connection_interval_max) == false) &&
|
|
|
|
(params->maxConnectionInterval != 0xFFFF)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((is_in_range(params->minConnectionInterval, connection_interval_min, params->maxConnectionInterval) == false) &&
|
|
|
|
(params->minConnectionInterval != 0xFFFF)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params->connectionSupervisionTimeout == 0xFFFF) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((is_in_range(params->connectionSupervisionTimeout, supervision_timeout_min, supervision_timeout_max) == false)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params->maxConnectionInterval == 0xFFFF) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t max_connection_interval_ms =
|
|
|
|
((uint32_t)params->maxConnectionInterval * 125) / 100;
|
|
|
|
uint16_t min_connection_supervision_timeout =
|
|
|
|
((1 + params->slaveLatency) * max_connection_interval_ms * 2) / 10;
|
|
|
|
|
|
|
|
if (params->connectionSupervisionTimeout < min_connection_supervision_timeout) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if random bytes of an address are valid.
|
|
|
|
*/
|
|
|
|
static bool is_prand_valid(const uint8_t* bytes, size_t len)
|
|
|
|
{
|
|
|
|
// at least one bit of the random part of the static address shall be
|
|
|
|
// equal to 0 and at least one bit of the random part of the static
|
|
|
|
// address shall be equal to 1
|
|
|
|
for (size_t i = 0; i < (len - 1); ++i) {
|
|
|
|
if ((bytes[i] != 0x00) && (bytes[i] != 0xFF)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((i > 0) && (bytes[i] != bytes[i - 1])) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (((bytes[len - 1] & 0x3F) == 0x3F) && (bytes[len - 2] == 0xFF)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (((bytes[len - 1] & 0x3F) == 0x00) && (bytes[len - 2] == 0x00)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check if the random part of a random address with 48 random bytes are valid
|
|
|
|
* or not.
|
|
|
|
* Return true if it is the case and false otherwise.
|
|
|
|
*/
|
|
|
|
static bool is_prand_48_bits_valid(const BLEProtocol::AddressBytes_t address)
|
|
|
|
{
|
|
|
|
return is_prand_valid(address, 6);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check if the random part of a random address with 24 random bytes are valid
|
|
|
|
* or not.
|
|
|
|
* Return true if it is the case and false otherwise.
|
|
|
|
*/
|
|
|
|
static bool is_prand_24_bits_valid(const BLEProtocol::AddressBytes_t address)
|
|
|
|
{
|
|
|
|
return is_prand_valid(address + 3, 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true if address is a random static address.
|
|
|
|
*/
|
|
|
|
static bool is_random_static_address(const BLEProtocol::AddressBytes_t address)
|
|
|
|
{
|
|
|
|
// top two msb bits shall be equal to 1.
|
|
|
|
if ((address[5] >> 6) != 0x03) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return is_prand_48_bits_valid(address);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true if address is a random private non resolvable address.
|
|
|
|
*/
|
|
|
|
static bool is_random_private_non_resolvable_address(
|
|
|
|
const BLEProtocol::AddressBytes_t address
|
|
|
|
) {
|
|
|
|
// top two msb bits shall be equal to 0.
|
|
|
|
if ((address[5] >> 6) != 0x00) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return is_prand_48_bits_valid(address);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true if address is a random private resolvable address.
|
|
|
|
*/
|
|
|
|
static bool is_random_private_resolvable_address(
|
|
|
|
const BLEProtocol::AddressBytes_t address
|
|
|
|
) {
|
|
|
|
// top two msb bits shall be equal to 01.
|
|
|
|
if ((address[5] >> 6) != 0x01) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return is_prand_24_bits_valid(address);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true if the address is a random address.
|
|
|
|
*/
|
|
|
|
static bool is_random_address(const BLEProtocol::AddressBytes_t address)
|
|
|
|
{
|
|
|
|
return is_random_private_resolvable_address(address) ||
|
|
|
|
is_random_private_non_resolvable_address(address) ||
|
|
|
|
is_random_static_address(address);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check disconnection reason validity.
|
|
|
|
*/
|
|
|
|
static bool is_disconnection_reason_valid(Gap::DisconnectionReason_t reason)
|
|
|
|
{
|
|
|
|
switch (reason) {
|
|
|
|
/**
|
|
|
|
* Note: accepted reasons are:
|
|
|
|
typedef pal::disconnection_reason_t reason_t;
|
|
|
|
case reason_t::AUTHENTICATION_FAILLURE:
|
|
|
|
case reason_t::REMOTE_USER_TERMINATED_CONNECTION:
|
|
|
|
case reason_t::REMOTE_DEVICE_TERMINATED_CONNECTION_DUE_TO_LOW_RESOURCES:
|
|
|
|
case reason_t::REMOTE_DEVICE_TERMINATED_CONNECTION_DUE_TO_POWER_OFF:
|
|
|
|
case reason_t::UNSUPPORTED_REMOTE_FEATURE:
|
|
|
|
case reason_t::PAIRING_WITH_UNIT_KEY_NOT_SUPPORTED:
|
|
|
|
case reason_t::UNACCEPTABLE_CONNECTION_PARAMETERS:
|
|
|
|
*/
|
|
|
|
|
|
|
|
// TODO Fix Disconnectionreason_t which expose invalid value
|
|
|
|
case Gap::REMOTE_USER_TERMINATED_CONNECTION:
|
|
|
|
case Gap::REMOTE_DEV_TERMINATION_DUE_TO_LOW_RESOURCES:
|
|
|
|
case Gap::REMOTE_DEV_TERMINATION_DUE_TO_POWER_OFF:
|
|
|
|
case Gap::CONN_INTERVAL_UNACCEPTABLE:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true if the whitelist in input is valid or false otherwise.
|
|
|
|
*/
|
|
|
|
static bool is_whitelist_valid(const Gap::Whitelist_t& whitelist)
|
|
|
|
{
|
|
|
|
if (whitelist.size > whitelist.capacity) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < whitelist.size; ++i) {
|
|
|
|
const BLEProtocol::Address_t& address = whitelist.addresses[i];
|
|
|
|
if (address.type > BLEProtocol::AddressType::RANDOM_PRIVATE_NON_RESOLVABLE) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (address.type != BLEProtocol::AddressType::PUBLIC) {
|
|
|
|
if (is_random_address(address.address) == false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true if device is present in the whitelist.
|
|
|
|
*/
|
|
|
|
static bool is_in_whitelist(
|
|
|
|
const BLEProtocol::Address_t& device, const Gap::Whitelist_t& whitelist
|
|
|
|
) {
|
|
|
|
for (size_t i = 0; i < whitelist.size; ++i) {
|
|
|
|
const BLEProtocol::Address_t& potential_device = whitelist.addresses[i];
|
|
|
|
|
|
|
|
if (potential_device.type != device.type) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memcmp(potential_device.address, device.address, sizeof(device.address)) == 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert a BLEProtocol::AddressType_t into a pal::whitelist_address_type_t.
|
|
|
|
*/
|
|
|
|
static pal::whitelist_address_type_t to_device_address_type(
|
|
|
|
BLEProtocol::AddressType_t address_type
|
|
|
|
) {
|
|
|
|
return (address_type == BLEProtocol::AddressType::PUBLIC) ?
|
|
|
|
pal::whitelist_address_type_t::PUBLIC_DEVICE_ADDRESS :
|
|
|
|
pal::whitelist_address_type_t::RANDOM_DEVICE_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return true if the advertising parameters are valid.
|
|
|
|
*/
|
|
|
|
static bool is_advertising_params_valid(const GapAdvertisingParams& params)
|
|
|
|
{
|
|
|
|
if (is_in_range(params.getIntervalInADVUnits(), advertising_interval_min, advertising_interval_max) == false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.getAdvertisingType() > GapAdvertisingParams::ADV_NON_CONNECTABLE_UNDIRECTED) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // end of anonymous namespace
|
|
|
|
|
|
|
|
GenericGap::GenericGap(
|
|
|
|
pal::EventQueue& event_queue,
|
|
|
|
pal::Gap& pal_gap,
|
|
|
|
pal::GenericAccessService& generic_access_service
|
|
|
|
) : _event_queue(event_queue),
|
|
|
|
_pal_gap(pal_gap),
|
|
|
|
_gap_service(generic_access_service),
|
|
|
|
_address_type(BLEProtocol::AddressType::PUBLIC),
|
|
|
|
_initiator_policy_mode(pal::initiator_policy_t::NO_FILTER),
|
|
|
|
_scanning_filter_policy(pal::scanning_filter_policy_t::NO_FILTER),
|
|
|
|
_advertising_filter_policy(pal::advertising_filter_policy_t::NO_FILTER),
|
|
|
|
_whitelist(),
|
|
|
|
_advertising_timeout(),
|
2018-02-19 15:32:07 +00:00
|
|
|
_scan_timeout(),
|
|
|
|
_connection_event_handler(NULL)
|
2017-10-12 12:58:41 +00:00
|
|
|
{
|
|
|
|
_pal_gap.when_gap_event_received(
|
|
|
|
mbed::callback(this, &GenericGap::on_gap_event_received)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
GenericGap::~GenericGap()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::setAddress(
|
|
|
|
BLEProtocol::AddressType_t type,
|
|
|
|
const BLEProtocol::AddressBytes_t address
|
|
|
|
) {
|
|
|
|
switch (type) {
|
|
|
|
case BLEProtocol::AddressType::PUBLIC:
|
|
|
|
// The public address cannot be set, just set the type to public
|
|
|
|
_address_type = type;
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
|
|
|
|
case BLEProtocol::AddressType::RANDOM_STATIC: {
|
|
|
|
if (is_random_static_address(address) == false) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t err = _pal_gap.set_random_address(
|
2018-02-20 18:01:21 +00:00
|
|
|
ble::address_t(address)
|
2017-10-12 12:58:41 +00:00
|
|
|
);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
_address_type = type;
|
2018-02-20 18:01:21 +00:00
|
|
|
_address = ble::address_t(address);
|
2017-10-12 12:58:41 +00:00
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
case BLEProtocol::AddressType::RANDOM_PRIVATE_RESOLVABLE:
|
|
|
|
// TODO: Fix with the privacy/security rework
|
|
|
|
return BLE_ERROR_NOT_IMPLEMENTED;
|
|
|
|
|
|
|
|
case BLEProtocol::AddressType::RANDOM_PRIVATE_NON_RESOLVABLE:
|
|
|
|
// TODO: add process to set the random private non resolvable
|
|
|
|
// address (privacy/security work)
|
|
|
|
return BLE_ERROR_NOT_IMPLEMENTED;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return BLE_ERROR_PARAM_OUT_OF_RANGE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::getAddress(
|
|
|
|
BLEProtocol::AddressType_t *type,
|
|
|
|
BLEProtocol::AddressBytes_t address
|
|
|
|
) {
|
|
|
|
*type = _address_type;
|
2018-02-19 17:26:24 +00:00
|
|
|
ble::address_t address_value;
|
2017-12-21 17:39:25 +00:00
|
|
|
if (_address_type == BLEProtocol::AddressType::PUBLIC) {
|
|
|
|
address_value = _pal_gap.get_device_address();
|
|
|
|
} else {
|
|
|
|
address_value = _pal_gap.get_random_address();
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(address, address_value.data(), address_value.size());
|
2017-10-12 12:58:41 +00:00
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t GenericGap::getMinAdvertisingInterval() const
|
|
|
|
{
|
|
|
|
return GapAdvertisingParams::GAP_ADV_PARAMS_INTERVAL_MIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t GenericGap::getMinNonConnectableAdvertisingInterval() const
|
|
|
|
{
|
|
|
|
return GapAdvertisingParams::GAP_ADV_PARAMS_INTERVAL_MIN_NONCON;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t GenericGap::getMaxAdvertisingInterval() const
|
|
|
|
{
|
|
|
|
return GapAdvertisingParams::GAP_ADV_PARAMS_INTERVAL_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::stopAdvertising()
|
|
|
|
{
|
|
|
|
ble_error_t err = _pal_gap.advertising_enable(false);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
_advertising_timeout.detach();
|
|
|
|
state.advertising = false;
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::stopScan()
|
|
|
|
{
|
|
|
|
ble_error_t err = _pal_gap.scan_enable(false, false);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
_scan_timeout.detach();
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::connect(
|
|
|
|
const BLEProtocol::AddressBytes_t peerAddr,
|
|
|
|
BLEProtocol::AddressType_t peerAddrType,
|
|
|
|
const ConnectionParams_t* connectionParams,
|
|
|
|
const GapScanningParams* scanParams
|
|
|
|
) {
|
|
|
|
if (is_scan_params_valid(scanParams) == false) {
|
|
|
|
return BLE_ERROR_PARAM_OUT_OF_RANGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_connection_params_valid(connectionParams) == false) {
|
|
|
|
return BLE_ERROR_PARAM_OUT_OF_RANGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO fix upper layer API, address type factorization is incorrect.
|
|
|
|
|
|
|
|
return _pal_gap.create_connection(
|
|
|
|
scanParams->getInterval(),
|
|
|
|
scanParams->getWindow(),
|
|
|
|
_initiator_policy_mode,
|
|
|
|
(pal::connection_peer_address_type_t::type) peerAddrType,
|
2018-02-20 18:01:21 +00:00
|
|
|
ble::address_t(peerAddr),
|
2017-10-12 12:58:41 +00:00
|
|
|
(pal::own_address_type_t::type) _address_type,
|
|
|
|
connectionParams->minConnectionInterval,
|
|
|
|
connectionParams->maxConnectionInterval,
|
|
|
|
connectionParams->slaveLatency,
|
|
|
|
connectionParams->connectionSupervisionTimeout,
|
|
|
|
/* minimum_connection_event_length */ 0,
|
|
|
|
/* maximum_connection_event_length */ 0
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::disconnect(Handle_t connectionHandle, DisconnectionReason_t reason)
|
|
|
|
{
|
|
|
|
if (is_disconnection_reason_valid(reason) == false) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
return _pal_gap.disconnect(
|
|
|
|
connectionHandle,
|
|
|
|
(pal::disconnection_reason_t::type) reason
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::updateConnectionParams(Handle_t handle, const ConnectionParams_t *params)
|
|
|
|
{
|
|
|
|
if (is_connection_params_valid(params) == false) {
|
|
|
|
return BLE_ERROR_PARAM_OUT_OF_RANGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return _pal_gap.connection_parameters_update(
|
|
|
|
handle,
|
|
|
|
params->minConnectionInterval,
|
|
|
|
params->maxConnectionInterval,
|
|
|
|
params->slaveLatency,
|
|
|
|
params->connectionSupervisionTimeout,
|
|
|
|
/* minimum_connection_event_length */ 0,
|
|
|
|
/* maximum_connection_event_length */ 0
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::getPreferredConnectionParams(ConnectionParams_t *params)
|
|
|
|
{
|
|
|
|
if (params == NULL) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
return _gap_service.get_peripheral_prefered_connection_parameters(
|
|
|
|
*params
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::setPreferredConnectionParams(const ConnectionParams_t *params)
|
|
|
|
{
|
|
|
|
if(is_preferred_connection_params_valid(params) == false) {
|
|
|
|
return BLE_ERROR_PARAM_OUT_OF_RANGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return _gap_service.set_peripheral_prefered_connection_parameters(
|
|
|
|
*params
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::setDeviceName(const uint8_t *deviceName)
|
|
|
|
{
|
|
|
|
return _gap_service.set_device_name(deviceName);
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::getDeviceName(uint8_t *deviceName, unsigned *lengthP)
|
|
|
|
{
|
|
|
|
if (lengthP == NULL) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t length = 0;
|
|
|
|
ble_error_t err = _gap_service.get_device_name_length(length);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (deviceName != NULL) {
|
|
|
|
if (*lengthP < length) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
ArrayView<uint8_t> name(deviceName, *lengthP);
|
|
|
|
err = _gap_service.get_device_name(name);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*lengthP = length;
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::setAppearance(GapAdvertisingData::Appearance appearance)
|
|
|
|
{
|
|
|
|
return _gap_service.set_appearance(appearance);
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::getAppearance(GapAdvertisingData::Appearance *appearanceP)
|
|
|
|
{
|
|
|
|
if (appearanceP == NULL) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
return _gap_service.get_appearance(*appearanceP);
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::setTxPower(int8_t txPower)
|
|
|
|
{
|
|
|
|
// TODO: This is not standard, expose it as an extension API and document it
|
|
|
|
// as such
|
|
|
|
return BLE_ERROR_NOT_IMPLEMENTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::getPermittedTxPowerValues(const int8_t **valueArrayPP, size_t *countP)
|
|
|
|
{
|
|
|
|
*countP = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t GenericGap::getMaxWhitelistSize(void) const
|
|
|
|
{
|
|
|
|
return _pal_gap.read_white_list_capacity();
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::getWhitelist(Whitelist_t &whitelist) const
|
|
|
|
{
|
|
|
|
if(initialize_whitelist() == false) {
|
|
|
|
return BLE_ERROR_INVALID_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (whitelist.capacity < _whitelist.capacity) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < _whitelist.size; ++i) {
|
|
|
|
whitelist.addresses[i] = _whitelist.addresses[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
whitelist.capacity = _whitelist.capacity;
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::setWhitelist(const Whitelist_t &whitelist)
|
|
|
|
{
|
|
|
|
if (is_whitelist_valid(whitelist) == false) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(initialize_whitelist() == false) {
|
|
|
|
return BLE_ERROR_INVALID_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (whitelist.capacity > _whitelist.capacity) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
// first evict devices not in the existing whitelist
|
|
|
|
for (size_t i = 0; i < _whitelist.size; ++i) {
|
|
|
|
const BLEProtocol::Address_t& device = _whitelist.addresses[i];
|
|
|
|
|
|
|
|
if (is_in_whitelist(device, whitelist) == false) {
|
|
|
|
ble_error_t err = _pal_gap.remove_device_from_whitelist(
|
|
|
|
to_device_address_type(device.type),
|
|
|
|
device.address
|
|
|
|
);
|
|
|
|
|
|
|
|
// try to restore the whitelist to its initial state
|
|
|
|
if (err) {
|
|
|
|
for (size_t j = 0; j < i; ++j) {
|
|
|
|
const BLEProtocol::Address_t& device = _whitelist.addresses[j];
|
|
|
|
|
|
|
|
if (is_in_whitelist(device, whitelist) == false) {
|
|
|
|
_pal_gap.add_device_to_whitelist(
|
|
|
|
to_device_address_type(device.type),
|
|
|
|
device.address
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// second add devices which were not in the initial whitelist
|
|
|
|
for (size_t i = 0; i < whitelist.size; ++i) {
|
|
|
|
const BLEProtocol::Address_t& device = whitelist.addresses[i];
|
|
|
|
|
|
|
|
if (is_in_whitelist(device, _whitelist) == false) {
|
|
|
|
ble_error_t err = _pal_gap.add_device_to_whitelist(
|
|
|
|
to_device_address_type(device.type),
|
|
|
|
device.address
|
|
|
|
);
|
|
|
|
|
|
|
|
// try to restore the whitelist to its initial state
|
|
|
|
if (err) {
|
|
|
|
// first remove the devices added
|
|
|
|
for (size_t j = 0; j < i; ++j) {
|
|
|
|
const BLEProtocol::Address_t& device = whitelist.addresses[j];
|
|
|
|
|
|
|
|
if (is_in_whitelist(device, _whitelist) == false) {
|
|
|
|
_pal_gap.remove_device_from_whitelist(
|
|
|
|
to_device_address_type(device.type),
|
|
|
|
device.address
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// second add the devices of the initial list evicted
|
|
|
|
for (size_t i = 0; i < _whitelist.size; ++i) {
|
|
|
|
const BLEProtocol::Address_t& device = _whitelist.addresses[i];
|
|
|
|
|
|
|
|
if (is_in_whitelist(device, whitelist) == false) {
|
|
|
|
_pal_gap.add_device_to_whitelist(
|
|
|
|
to_device_address_type(device.type),
|
|
|
|
device.address
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// commit the devices into the whitelist
|
|
|
|
for (size_t i = 0; i < whitelist.size; ++i) {
|
|
|
|
_whitelist.addresses[i] = whitelist.addresses[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
_whitelist.size = whitelist.size;
|
|
|
|
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::setAdvertisingPolicyMode(AdvertisingPolicyMode_t mode)
|
|
|
|
{
|
|
|
|
if (mode > Gap::ADV_POLICY_FILTER_ALL_REQS) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
_advertising_filter_policy = (pal::advertising_filter_policy_t::type) mode;
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::setScanningPolicyMode(ScanningPolicyMode_t mode)
|
|
|
|
{
|
|
|
|
if (mode > Gap::SCAN_POLICY_FILTER_ALL_ADV) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
_scanning_filter_policy = (pal::scanning_filter_policy_t::type) mode;
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::setInitiatorPolicyMode(InitiatorPolicyMode_t mode)
|
|
|
|
{
|
|
|
|
if (mode > Gap::INIT_POLICY_FILTER_ALL_ADV) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
_initiator_policy_mode = (pal::initiator_policy_t::type) mode;
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
Gap::AdvertisingPolicyMode_t GenericGap::getAdvertisingPolicyMode(void) const
|
|
|
|
{
|
|
|
|
return (AdvertisingPolicyMode_t) _advertising_filter_policy.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
Gap::ScanningPolicyMode_t GenericGap::getScanningPolicyMode(void) const
|
|
|
|
{
|
|
|
|
return (ScanningPolicyMode_t) _scanning_filter_policy.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
Gap::InitiatorPolicyMode_t GenericGap::getInitiatorPolicyMode(void) const
|
|
|
|
{
|
|
|
|
return (InitiatorPolicyMode_t) _initiator_policy_mode.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::startRadioScan(const GapScanningParams &scanningParams)
|
|
|
|
{
|
|
|
|
if (is_scan_params_valid(&scanningParams) == false) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_scanning_filter_policy == pal::scanning_filter_policy_t::FILTER_ADVERTISING &&
|
|
|
|
_whitelist.size == 0) {
|
|
|
|
return BLE_ERROR_INVALID_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t err = _pal_gap.set_scan_parameters(
|
|
|
|
scanningParams.getActiveScanning(),
|
|
|
|
scanningParams.getInterval(),
|
|
|
|
scanningParams.getWindow(),
|
|
|
|
get_own_address_type(),
|
|
|
|
_scanning_filter_policy
|
|
|
|
);
|
|
|
|
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = _pal_gap.scan_enable(true, false);
|
|
|
|
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
_scan_timeout.detach();
|
|
|
|
uint16_t timeout = scanningParams.getTimeout();
|
|
|
|
if (timeout) {
|
|
|
|
_scan_timeout.attach_us(
|
|
|
|
mbed::callback(this, &GenericGap::on_scan_timeout),
|
|
|
|
scanningParams.getTimeout() * 1000000U
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::initRadioNotification(void)
|
|
|
|
{
|
|
|
|
return BLE_ERROR_NOT_IMPLEMENTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::setAdvertisingData(const GapAdvertisingData &advData, const GapAdvertisingData &scanResponse)
|
|
|
|
{
|
|
|
|
ble_error_t err = _pal_gap.set_advertising_data(
|
|
|
|
advData.getPayloadLen(),
|
|
|
|
pal::advertising_data_t(advData.getPayload(), advData.getPayloadLen())
|
|
|
|
);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return _pal_gap.set_scan_response_data(
|
|
|
|
scanResponse.getPayloadLen(),
|
|
|
|
pal::advertising_data_t(scanResponse.getPayload(), scanResponse.getPayloadLen())
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::startAdvertising(const GapAdvertisingParams& params)
|
|
|
|
{
|
|
|
|
if (is_advertising_params_valid(params) == false) {
|
|
|
|
return BLE_ERROR_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: fix the high level API to have a min/max range
|
|
|
|
// Going against recommendations (The Advertising_Interval_Min and
|
|
|
|
// Advertising_Interval_Max should not be the same value to enable the
|
|
|
|
// Controller to determine the best advertising interval given other activities.)
|
|
|
|
// for now but not against specification: "The Advertising_Interval_Min
|
|
|
|
// shall be less than or equal to the Advertising_Interval_Max"
|
|
|
|
ble_error_t err = _pal_gap.set_advertising_parameters(
|
|
|
|
/* advertising_interval_min */ params.getIntervalInADVUnits(),
|
|
|
|
/* advertising_interval_max */ params.getIntervalInADVUnits(),
|
|
|
|
(pal::advertising_type_t::type) params.getAdvertisingType(),
|
|
|
|
get_own_address_type(),
|
|
|
|
pal::advertising_peer_address_type_t::PUBLIC_ADDRESS,
|
2018-02-06 22:59:21 +00:00
|
|
|
ble::address_t(),
|
2017-10-12 12:58:41 +00:00
|
|
|
pal::advertising_channel_map_t::ALL_ADVERTISING_CHANNELS,
|
|
|
|
_advertising_filter_policy
|
|
|
|
);
|
|
|
|
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = _pal_gap.advertising_enable(true);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
state.advertising = true;
|
|
|
|
|
|
|
|
_advertising_timeout.detach();
|
|
|
|
uint16_t timeout = params.getTimeout();
|
|
|
|
if (timeout) {
|
|
|
|
_advertising_timeout.attach_us(
|
|
|
|
mbed::callback(this, &GenericGap::on_advertising_timeout),
|
|
|
|
params.getTimeout() * 1000000U
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ble_error_t GenericGap::reset(void)
|
|
|
|
{
|
|
|
|
Gap::reset();
|
|
|
|
_advertising_timeout.detach();
|
|
|
|
_scan_timeout.detach();
|
|
|
|
|
|
|
|
return BLE_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
2018-02-19 15:32:07 +00:00
|
|
|
void GenericGap::processConnectionEvent(
|
|
|
|
Handle_t handle,
|
|
|
|
Role_t role,
|
|
|
|
BLEProtocol::AddressType_t peerAddrType,
|
|
|
|
const BLEProtocol::AddressBytes_t peerAddr,
|
|
|
|
BLEProtocol::AddressType_t ownAddrType,
|
|
|
|
const BLEProtocol::AddressBytes_t ownAddr,
|
|
|
|
const ConnectionParams_t *connectionParams
|
|
|
|
) {
|
|
|
|
if (_connection_event_handler) {
|
|
|
|
_connection_event_handler->on_connected(
|
|
|
|
handle,
|
|
|
|
role,
|
|
|
|
peerAddrType,
|
|
|
|
peerAddr,
|
|
|
|
ownAddrType,
|
|
|
|
ownAddr,
|
|
|
|
connectionParams
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
::Gap::processConnectionEvent(
|
|
|
|
handle,
|
|
|
|
role,
|
|
|
|
peerAddrType,
|
|
|
|
peerAddr,
|
|
|
|
ownAddrType,
|
|
|
|
ownAddr,
|
|
|
|
connectionParams
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::processDisconnectionEvent(
|
|
|
|
Handle_t handle,
|
|
|
|
DisconnectionReason_t reason
|
|
|
|
) {
|
|
|
|
if (_connection_event_handler) {
|
|
|
|
_connection_event_handler->on_disconnected(
|
|
|
|
handle,
|
|
|
|
reason
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
::Gap::processDisconnectionEvent(
|
|
|
|
handle,
|
|
|
|
reason
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-10-12 12:58:41 +00:00
|
|
|
void GenericGap::on_scan_timeout()
|
|
|
|
{
|
|
|
|
_event_queue.post(mbed::callback(this, &GenericGap::process_scan_timeout));
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::process_scan_timeout()
|
|
|
|
{
|
|
|
|
ble_error_t err = _pal_gap.scan_enable(false, false);
|
|
|
|
if (err) {
|
|
|
|
// TODO: define the mechanism signaling the error
|
|
|
|
}
|
|
|
|
processTimeoutEvent(Gap::TIMEOUT_SRC_SCAN);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::on_advertising_timeout()
|
|
|
|
{
|
|
|
|
_event_queue.post(mbed::callback(this, &GenericGap::process_advertising_timeout));
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::process_advertising_timeout()
|
|
|
|
{
|
|
|
|
ble_error_t err = _pal_gap.advertising_enable(false);
|
|
|
|
if (err) {
|
|
|
|
// TODO: define the mechanism signaling the error
|
|
|
|
}
|
|
|
|
processTimeoutEvent(Gap::TIMEOUT_SRC_ADVERTISING);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::on_gap_event_received(const pal::GapEvent& e)
|
|
|
|
{
|
|
|
|
switch (e.type.value()) {
|
|
|
|
case pal::GapEventType::ADVERTISING_REPORT:
|
|
|
|
on_advertising_report(static_cast<const pal::GapAdvertisingReportEvent&>(e));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case pal::GapEventType::CONNECTION_COMPLETE:
|
|
|
|
on_connection_complete(static_cast<const pal::GapConnectionCompleteEvent&>(e));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case pal::GapEventType::CONNECTION_UPDATE:
|
|
|
|
on_connection_update(static_cast<const pal::GapConnectionUpdateEvent&>(e));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case pal::GapEventType::DISCONNECTION_COMPLETE:
|
|
|
|
on_disconnection_complete(static_cast<const pal::GapDisconnectionCompleteEvent&>(e));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case pal::GapEventType::REMOTE_CONNECTION_PARAMETER_REQUEST:
|
|
|
|
on_connection_parameter_request(static_cast<const pal::GapRemoteConnectionParameterRequestEvent&>(e));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case pal::GapEventType::UNEXPECTED_ERROR:
|
|
|
|
on_unexpected_error(static_cast<const pal::GapUnexpectedErrorEvent&>(e));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::on_advertising_report(const pal::GapAdvertisingReportEvent& e)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < e.size(); ++i) {
|
|
|
|
pal::GapAdvertisingReportEvent::advertising_t advertising = e[i];
|
|
|
|
|
|
|
|
processAdvertisementReport(
|
|
|
|
advertising.address.data(),
|
|
|
|
advertising.rssi,
|
|
|
|
advertising.type == pal::received_advertising_type_t::SCAN_RESPONSE,
|
|
|
|
(GapAdvertisingParams::AdvertisingType_t) advertising.type.value(),
|
|
|
|
advertising.data.size(),
|
|
|
|
advertising.data.data()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::on_connection_complete(const pal::GapConnectionCompleteEvent& e)
|
|
|
|
{
|
|
|
|
// TODO: deprecate ownAddrType and ownAddr, those are not specified
|
|
|
|
// from the Bluetooth perspective
|
|
|
|
if (e.status == pal::hci_error_code_t::SUCCESS) {
|
|
|
|
if (e.role.value() == e.role.SLAVE) {
|
|
|
|
_advertising_timeout.detach();
|
|
|
|
_pal_gap.advertising_enable(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// using these parameters if stupid, there is no range for the
|
|
|
|
// connection interval when the connection is established
|
|
|
|
ConnectionParams_t connection_params = {
|
|
|
|
/* minConnectionInterval */ e.connection_interval,
|
|
|
|
/* maxConnectionInterval */ e.connection_interval,
|
|
|
|
e.connection_latency,
|
|
|
|
e.supervision_timeout
|
|
|
|
};
|
2018-02-19 17:26:24 +00:00
|
|
|
ble::address_t address;
|
2017-12-21 17:39:25 +00:00
|
|
|
if (_address_type == BLEProtocol::AddressType::PUBLIC) {
|
|
|
|
address = _pal_gap.get_device_address();
|
|
|
|
} else {
|
|
|
|
address = _pal_gap.get_random_address();
|
|
|
|
}
|
2017-10-12 12:58:41 +00:00
|
|
|
|
|
|
|
processConnectionEvent(
|
|
|
|
e.connection_handle,
|
|
|
|
e.role.value() == e.role.MASTER ? ::Gap::CENTRAL : ::Gap::PERIPHERAL,
|
|
|
|
(BLEProtocol::AddressType_t) e.peer_address_type.value(),
|
|
|
|
e.peer_address.data(),
|
|
|
|
_address_type,
|
2017-12-21 17:39:25 +00:00
|
|
|
address.data(),
|
2017-10-12 12:58:41 +00:00
|
|
|
&connection_params
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
// for now notify user that the connection failled by issuing a timeout
|
|
|
|
// event
|
|
|
|
|
|
|
|
// TODO: Define events in case of connection faillure
|
|
|
|
processTimeoutEvent(Gap::TIMEOUT_SRC_CONN);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::on_disconnection_complete(const pal::GapDisconnectionCompleteEvent& e)
|
|
|
|
{
|
|
|
|
if (e.status == pal::hci_error_code_t::SUCCESS) {
|
|
|
|
processDisconnectionEvent(
|
|
|
|
e.connection_handle,
|
|
|
|
(Gap::DisconnectionReason_t) e.reason
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
// TODO: define what to do in case of faillure
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::on_connection_parameter_request(const pal::GapRemoteConnectionParameterRequestEvent& e)
|
|
|
|
{
|
|
|
|
// intern behavior, accept all new parameter requests
|
|
|
|
// TODO: expose an API so user code can accept or reject such request
|
|
|
|
_pal_gap.accept_connection_parameter_request(
|
|
|
|
e.connection_handle,
|
|
|
|
e.min_connection_interval,
|
|
|
|
e.max_connection_interval,
|
|
|
|
e.connection_latency,
|
|
|
|
e.supervision_timeout,
|
|
|
|
/* minimum_connection_event_length */ 0,
|
|
|
|
/* maximum_connection_event_length */ 0
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::on_connection_update(const pal::GapConnectionUpdateEvent& e)
|
|
|
|
{
|
|
|
|
// TODO: add feature in interface to notify the user that the connection
|
|
|
|
// has been updated.
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenericGap::on_unexpected_error(const pal::GapUnexpectedErrorEvent& e)
|
|
|
|
{
|
|
|
|
// TODO: add feature in interface to notify the user that the connection
|
|
|
|
// has been updated.
|
|
|
|
}
|
|
|
|
|
|
|
|
pal::own_address_type_t GenericGap::get_own_address_type()
|
|
|
|
{
|
|
|
|
switch (_address_type) {
|
|
|
|
case BLEProtocol::AddressType::PUBLIC:
|
|
|
|
return pal::own_address_type_t::PUBLIC_ADDRESS;
|
|
|
|
case BLEProtocol::AddressType::RANDOM_STATIC:
|
|
|
|
case BLEProtocol::AddressType::RANDOM_PRIVATE_NON_RESOLVABLE:
|
|
|
|
return pal::own_address_type_t::RANDOM_ADDRESS;
|
|
|
|
case BLEProtocol::AddressType::RANDOM_PRIVATE_RESOLVABLE:
|
|
|
|
return pal::own_address_type_t::RESOLVABLE_PRIVATE_ADDRESS_PUBLIC_FALLBACK;
|
|
|
|
default:
|
|
|
|
// not reachable
|
|
|
|
return pal::own_address_type_t::PUBLIC_ADDRESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GenericGap::initialize_whitelist() const
|
|
|
|
{
|
|
|
|
if (_whitelist.addresses != NULL) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t whitelist_capacity = _pal_gap.read_white_list_capacity();
|
|
|
|
|
|
|
|
if (whitelist_capacity == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_whitelist.addresses = new (std::nothrow) BLEProtocol::Address_t[whitelist_capacity] ;
|
|
|
|
if (_whitelist.addresses == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_whitelist.size = 0;
|
|
|
|
_whitelist.capacity = whitelist_capacity;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-02-19 17:41:18 +00:00
|
|
|
void GenericGap::set_connection_event_handler(pal::ConnectionEventHandler *connection_event_handler)
|
2018-02-19 15:32:07 +00:00
|
|
|
{
|
|
|
|
_connection_event_handler = connection_event_handler;
|
|
|
|
}
|
|
|
|
|
2017-10-12 12:58:41 +00:00
|
|
|
} // namespace generic
|
|
|
|
} // namespace ble
|