2018-12-28 22:55:29 -05:00
|
|
|
/*
|
|
|
|
* This file is part of the MicroPython project, http://micropython.org/
|
|
|
|
*
|
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
2018-12-30 22:31:51 -05:00
|
|
|
* Copyright (c) 2018 Dan Halbert for Adafruit Industries
|
2019-06-19 10:42:36 -04:00
|
|
|
* Copyright (c) 2018 Artur Pacholec
|
2018-12-28 22:55:29 -05:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "ble.h"
|
|
|
|
#include "ble_drv.h"
|
|
|
|
#include "ble_hci.h"
|
|
|
|
#include "nrf_soc.h"
|
2019-06-13 21:55:07 -04:00
|
|
|
#include "py/gc.h"
|
2019-06-18 23:46:20 -04:00
|
|
|
#include "py/objlist.h"
|
2018-12-28 22:55:29 -05:00
|
|
|
#include "py/objstr.h"
|
|
|
|
#include "py/runtime.h"
|
2019-08-16 15:18:53 -04:00
|
|
|
#include "shared-bindings/bleio/__init__.h"
|
2018-12-28 22:55:29 -05:00
|
|
|
#include "shared-bindings/bleio/Adapter.h"
|
|
|
|
#include "shared-bindings/bleio/Characteristic.h"
|
2018-12-30 22:31:51 -05:00
|
|
|
#include "shared-bindings/bleio/Peripheral.h"
|
2018-12-28 22:55:29 -05:00
|
|
|
#include "shared-bindings/bleio/Service.h"
|
|
|
|
#include "shared-bindings/bleio/UUID.h"
|
|
|
|
|
|
|
|
#define BLE_MIN_CONN_INTERVAL MSEC_TO_UNITS(15, UNIT_0_625_MS)
|
|
|
|
#define BLE_MAX_CONN_INTERVAL MSEC_TO_UNITS(300, UNIT_0_625_MS)
|
|
|
|
#define BLE_SLAVE_LATENCY 0
|
|
|
|
#define BLE_CONN_SUP_TIMEOUT MSEC_TO_UNITS(4000, UNIT_10_MS)
|
|
|
|
|
|
|
|
#define BLE_ADV_LENGTH_FIELD_SIZE 1
|
|
|
|
#define BLE_ADV_AD_TYPE_FIELD_SIZE 1
|
|
|
|
#define BLE_AD_TYPE_FLAGS_DATA_SIZE 1
|
|
|
|
|
2019-07-30 14:26:26 -04:00
|
|
|
static const ble_gap_sec_params_t pairing_sec_params = {
|
|
|
|
.bond = 0, // TODO: add bonding
|
|
|
|
.mitm = 0,
|
|
|
|
.lesc = 0,
|
|
|
|
.keypress = 0,
|
|
|
|
.oob = 0,
|
|
|
|
.io_caps = BLE_GAP_IO_CAPS_NONE,
|
|
|
|
.min_key_size = 7,
|
|
|
|
.max_key_size = 16,
|
|
|
|
.kdist_own = { .enc = 1, .id = 1},
|
|
|
|
.kdist_peer = { .enc = 1, .id = 1},
|
|
|
|
};
|
|
|
|
|
2019-06-02 23:21:30 -04:00
|
|
|
STATIC void check_data_fit(size_t data_len) {
|
|
|
|
if (data_len > BLE_GAP_ADV_SET_DATA_SIZE_MAX) {
|
2018-12-28 22:55:29 -05:00
|
|
|
mp_raise_ValueError(translate("Data too large for advertisement packet"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-03 21:42:42 -05:00
|
|
|
STATIC void peripheral_on_ble_evt(ble_evt_t *ble_evt, void *self_in) {
|
2018-12-30 22:31:51 -05:00
|
|
|
bleio_peripheral_obj_t *self = (bleio_peripheral_obj_t*)self_in;
|
2018-12-28 22:55:29 -05:00
|
|
|
|
2019-07-30 14:26:26 -04:00
|
|
|
// For debugging.
|
|
|
|
// mp_printf(&mp_plat_print, "Peripheral event: 0x%04x\n", ble_evt->header.evt_id);
|
|
|
|
|
2018-12-28 22:55:29 -05:00
|
|
|
switch (ble_evt->header.evt_id) {
|
2019-07-02 22:34:54 -04:00
|
|
|
case BLE_GAP_EVT_CONNECTED: {
|
|
|
|
// Central has connected.
|
|
|
|
ble_gap_conn_params_t conn_params;
|
|
|
|
self->conn_handle = ble_evt->evt.gap_evt.conn_handle;
|
|
|
|
sd_ble_gap_ppcp_get(&conn_params);
|
|
|
|
sd_ble_gap_conn_param_update(ble_evt->evt.gap_evt.conn_handle, &conn_params);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case BLE_GAP_EVT_DISCONNECTED:
|
|
|
|
// Central has disconnected.
|
|
|
|
self->conn_handle = BLE_CONN_HANDLE_INVALID;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BLE_GAP_EVT_PHY_UPDATE_REQUEST: {
|
|
|
|
ble_gap_phys_t const phys = {
|
|
|
|
.rx_phys = BLE_GAP_PHY_AUTO,
|
|
|
|
.tx_phys = BLE_GAP_PHY_AUTO,
|
|
|
|
};
|
|
|
|
sd_ble_gap_phy_update(ble_evt->evt.gap_evt.conn_handle, &phys);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case BLE_GAP_EVT_ADV_SET_TERMINATED:
|
|
|
|
// Someday may handle timeouts or limit reached.
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BLE_GAP_EVT_CONN_PARAM_UPDATE_REQUEST: {
|
|
|
|
ble_gap_evt_conn_param_update_request_t *request =
|
|
|
|
&ble_evt->evt.gap_evt.params.conn_param_update_request;
|
|
|
|
sd_ble_gap_conn_param_update(self->conn_handle, &request->conn_params);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case BLE_GAP_EVT_DATA_LENGTH_UPDATE_REQUEST:
|
|
|
|
sd_ble_gap_data_length_update(self->conn_handle, NULL, NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BLE_GATTS_EVT_EXCHANGE_MTU_REQUEST: {
|
|
|
|
sd_ble_gatts_exchange_mtu_reply(self->conn_handle, BLE_GATT_ATT_MTU_DEFAULT);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case BLE_GATTS_EVT_SYS_ATTR_MISSING:
|
|
|
|
sd_ble_gatts_sys_attr_set(self->conn_handle, NULL, 0, 0);
|
|
|
|
break;
|
|
|
|
|
2019-07-30 14:26:26 -04:00
|
|
|
case BLE_GAP_EVT_SEC_PARAMS_REQUEST:
|
|
|
|
sd_ble_gap_sec_params_reply(self->conn_handle, BLE_GAP_SEC_STATUS_SUCCESS, &pairing_sec_params, NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BLE_GAP_EVT_LESC_DHKEY_REQUEST:
|
|
|
|
// TODO for LESC pairing:
|
|
|
|
// sd_ble_gap_lesc_dhkey_reply(...);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BLE_GAP_EVT_AUTH_STATUS: {
|
|
|
|
// Pairing process completed
|
|
|
|
ble_gap_evt_auth_status_t* status = &ble_evt->evt.gap_evt.params.auth_status;
|
|
|
|
if (BLE_GAP_SEC_STATUS_SUCCESS == status->auth_status) {
|
2019-08-02 17:56:44 -04:00
|
|
|
// mp_printf(&mp_plat_print, "Pairing succeeded, status: 0x%04x\n", status->auth_status);
|
2019-07-30 14:26:26 -04:00
|
|
|
self->pair_status = PAIR_PAIRED;
|
|
|
|
} else {
|
2019-08-02 17:56:44 -04:00
|
|
|
// mp_printf(&mp_plat_print, "Pairing failed, status: 0x%04x\n", status->auth_status);
|
2019-07-30 14:26:26 -04:00
|
|
|
self->pair_status = PAIR_NOT_PAIRED;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case BLE_GAP_EVT_CONN_SEC_UPDATE: {
|
|
|
|
ble_gap_conn_sec_t* conn_sec = &ble_evt->evt.gap_evt.params.conn_sec_update.conn_sec;
|
|
|
|
if (conn_sec->sec_mode.sm <= 1 && conn_sec->sec_mode.lv <= 1) {
|
|
|
|
// Security setup did not succeed:
|
|
|
|
// mode 0, level 0 means no access
|
|
|
|
// mode 1, level 1 means open link
|
|
|
|
// mode >=1 and/or level >=1 means encryption is set up
|
|
|
|
self->pair_status = PAIR_NOT_PAIRED;
|
|
|
|
} else {
|
|
|
|
// TODO: see Bluefruit lib
|
|
|
|
// if ( !bond_load_cccd(_role, _conn_hdl, _ediv) ) {
|
|
|
|
// sd_ble_gatts_sys_attr_set(_conn_hdl, NULL, 0, 0);
|
|
|
|
// }
|
|
|
|
self->pair_status = PAIR_PAIRED;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-02 22:34:54 -04:00
|
|
|
default:
|
|
|
|
// For debugging.
|
|
|
|
// mp_printf(&mp_plat_print, "Unhandled peripheral event: 0x%04x\n", ble_evt->header.evt_id);
|
|
|
|
break;
|
2018-12-28 22:55:29 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-27 13:20:59 -04:00
|
|
|
void common_hal_bleio_peripheral_construct(bleio_peripheral_obj_t *self, mp_obj_list_t *services_list, mp_obj_t name) {
|
2019-05-23 22:05:16 -04:00
|
|
|
common_hal_bleio_adapter_set_enabled(true);
|
2018-12-28 22:55:29 -05:00
|
|
|
|
2019-07-27 13:20:59 -04:00
|
|
|
self->services_list = services_list;
|
|
|
|
// Used only for discovery when acting as a client.
|
|
|
|
self->remote_services_list = mp_obj_new_list(0, NULL);
|
2019-06-18 23:46:20 -04:00
|
|
|
self->name = name;
|
|
|
|
|
2018-12-28 22:55:29 -05:00
|
|
|
self->conn_handle = BLE_CONN_HANDLE_INVALID;
|
2019-06-03 20:40:05 -04:00
|
|
|
self->adv_handle = BLE_GAP_ADV_SET_HANDLE_NOT_SET;
|
2019-07-30 14:26:26 -04:00
|
|
|
self->pair_status = PAIR_NOT_PAIRED;
|
2018-12-28 22:55:29 -05:00
|
|
|
|
|
|
|
// Add all the services.
|
|
|
|
|
2019-07-27 13:20:59 -04:00
|
|
|
for (size_t service_idx = 0; service_idx < services_list->len; ++service_idx) {
|
|
|
|
bleio_service_obj_t *service = MP_OBJ_TO_PTR(services_list->items[service_idx]);
|
2018-12-28 22:55:29 -05:00
|
|
|
|
2019-07-09 00:21:46 -04:00
|
|
|
service->device = MP_OBJ_FROM_PTR(self);
|
2019-06-18 23:46:20 -04:00
|
|
|
|
2018-12-28 22:55:29 -05:00
|
|
|
ble_uuid_t uuid;
|
|
|
|
bleio_uuid_convert_to_nrf_ble_uuid(service->uuid, &uuid);
|
|
|
|
|
|
|
|
uint8_t service_type = BLE_GATTS_SRVC_TYPE_PRIMARY;
|
2019-07-09 00:21:46 -04:00
|
|
|
if (common_hal_bleio_service_get_is_secondary(service)) {
|
2018-12-28 22:55:29 -05:00
|
|
|
service_type = BLE_GATTS_SRVC_TYPE_SECONDARY;
|
|
|
|
}
|
|
|
|
|
|
|
|
const uint32_t err_code = sd_ble_gatts_service_add(service_type, &uuid, &service->handle);
|
|
|
|
if (err_code != NRF_SUCCESS) {
|
|
|
|
mp_raise_OSError_msg_varg(translate("Failed to add service, err 0x%04x"), err_code);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Once the service has been registered, its characteristics can be added.
|
|
|
|
common_hal_bleio_service_add_all_characteristics(service);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-30 14:26:26 -04:00
|
|
|
mp_obj_list_t *common_hal_bleio_peripheral_get_services(bleio_peripheral_obj_t *self) {
|
2019-07-27 13:20:59 -04:00
|
|
|
return self->services_list;
|
2019-06-18 23:46:20 -04:00
|
|
|
}
|
|
|
|
|
2018-12-30 22:31:51 -05:00
|
|
|
bool common_hal_bleio_peripheral_get_connected(bleio_peripheral_obj_t *self) {
|
2018-12-28 22:55:29 -05:00
|
|
|
return self->conn_handle != BLE_CONN_HANDLE_INVALID;
|
|
|
|
}
|
|
|
|
|
2019-06-18 23:46:20 -04:00
|
|
|
mp_obj_t common_hal_bleio_peripheral_get_name(bleio_peripheral_obj_t *self) {
|
|
|
|
return self->name;
|
|
|
|
}
|
|
|
|
|
2019-06-03 20:40:05 -04:00
|
|
|
void common_hal_bleio_peripheral_start_advertising(bleio_peripheral_obj_t *self, bool connectable, mp_float_t interval, mp_buffer_info_t *advertising_data_bufinfo, mp_buffer_info_t *scan_response_data_bufinfo) {
|
|
|
|
|
|
|
|
// interval value has already been validated.
|
|
|
|
|
2018-12-28 22:55:29 -05:00
|
|
|
if (connectable) {
|
2019-01-03 21:42:42 -05:00
|
|
|
ble_drv_add_event_handler(peripheral_on_ble_evt, self);
|
2018-12-28 22:55:29 -05:00
|
|
|
}
|
|
|
|
|
2019-06-03 20:40:05 -04:00
|
|
|
common_hal_bleio_adapter_set_enabled(true);
|
|
|
|
|
|
|
|
uint32_t err_code;
|
|
|
|
|
|
|
|
GET_STR_DATA_LEN(self->name, name_data, name_len);
|
|
|
|
if (name_len > 0) {
|
|
|
|
ble_gap_conn_sec_mode_t sec_mode;
|
|
|
|
BLE_GAP_CONN_SEC_MODE_SET_OPEN(&sec_mode);
|
|
|
|
|
|
|
|
err_code = sd_ble_gap_device_name_set(&sec_mode, name_data, name_len);
|
|
|
|
if (err_code != NRF_SUCCESS) {
|
|
|
|
mp_raise_OSError_msg_varg(translate("Failed to set device name, err 0x%04x"), err_code);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
check_data_fit(advertising_data_bufinfo->len);
|
2019-06-05 20:08:53 -04:00
|
|
|
// The advertising data buffers must not move, because the SoftDevice depends on them.
|
|
|
|
// So make them long-lived.
|
|
|
|
self->advertising_data = (uint8_t *) gc_alloc(BLE_GAP_ADV_SET_DATA_SIZE_MAX * sizeof(uint8_t), false, true);
|
2019-06-03 20:40:05 -04:00
|
|
|
memcpy(self->advertising_data, advertising_data_bufinfo->buf, advertising_data_bufinfo->len);
|
|
|
|
|
|
|
|
check_data_fit(scan_response_data_bufinfo->len);
|
2019-06-05 20:08:53 -04:00
|
|
|
self->scan_response_data = (uint8_t *) gc_alloc(BLE_GAP_ADV_SET_DATA_SIZE_MAX * sizeof(uint8_t), false, true);
|
2019-06-03 20:40:05 -04:00
|
|
|
memcpy(self->scan_response_data, scan_response_data_bufinfo->buf, scan_response_data_bufinfo->len);
|
|
|
|
|
|
|
|
|
|
|
|
ble_gap_adv_params_t adv_params = {
|
|
|
|
.interval = SEC_TO_UNITS(interval, UNIT_0_625_MS),
|
|
|
|
.properties.type = connectable ? BLE_GAP_ADV_TYPE_CONNECTABLE_SCANNABLE_UNDIRECTED
|
|
|
|
: BLE_GAP_ADV_TYPE_NONCONNECTABLE_NONSCANNABLE_UNDIRECTED,
|
|
|
|
.duration = BLE_GAP_ADV_TIMEOUT_GENERAL_UNLIMITED,
|
|
|
|
.filter_policy = BLE_GAP_ADV_FP_ANY,
|
|
|
|
.primary_phy = BLE_GAP_PHY_1MBPS,
|
|
|
|
};
|
|
|
|
|
|
|
|
common_hal_bleio_peripheral_stop_advertising(self);
|
|
|
|
|
|
|
|
const ble_gap_adv_data_t ble_gap_adv_data = {
|
|
|
|
.adv_data.p_data = self->advertising_data,
|
|
|
|
.adv_data.len = advertising_data_bufinfo->len,
|
|
|
|
.scan_rsp_data.p_data = scan_response_data_bufinfo-> len > 0 ? self->scan_response_data : NULL,
|
|
|
|
.scan_rsp_data.len = scan_response_data_bufinfo->len,
|
|
|
|
};
|
|
|
|
|
|
|
|
err_code = sd_ble_gap_adv_set_configure(&self->adv_handle, &ble_gap_adv_data, &adv_params);
|
|
|
|
if (err_code != NRF_SUCCESS) {
|
|
|
|
mp_raise_OSError_msg_varg(translate("Failed to configure advertising, err 0x%04x"), err_code);
|
|
|
|
}
|
|
|
|
|
|
|
|
err_code = sd_ble_gap_adv_start(self->adv_handle, BLE_CONN_CFG_TAG_CUSTOM);
|
|
|
|
|
2018-12-28 22:55:29 -05:00
|
|
|
if (err_code != NRF_SUCCESS) {
|
|
|
|
mp_raise_OSError_msg_varg(translate("Failed to start advertising, err 0x%04x"), err_code);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-30 22:31:51 -05:00
|
|
|
void common_hal_bleio_peripheral_stop_advertising(bleio_peripheral_obj_t *self) {
|
2019-06-03 20:40:05 -04:00
|
|
|
if (self->adv_handle == BLE_GAP_ADV_SET_HANDLE_NOT_SET)
|
2018-12-28 22:55:29 -05:00
|
|
|
return;
|
|
|
|
|
2019-06-03 20:40:05 -04:00
|
|
|
const uint32_t err_code = sd_ble_gap_adv_stop(self->adv_handle);
|
2018-12-28 22:55:29 -05:00
|
|
|
|
|
|
|
if ((err_code != NRF_SUCCESS) && (err_code != NRF_ERROR_INVALID_STATE)) {
|
|
|
|
mp_raise_OSError_msg_varg(translate("Failed to stop advertising, err 0x%04x"), err_code);
|
|
|
|
}
|
|
|
|
}
|
2019-07-07 00:07:47 -04:00
|
|
|
|
|
|
|
void common_hal_bleio_peripheral_disconnect(bleio_peripheral_obj_t *self) {
|
|
|
|
sd_ble_gap_disconnect(self->conn_handle, BLE_HCI_REMOTE_USER_TERMINATED_CONNECTION);
|
|
|
|
}
|
2019-07-30 14:26:26 -04:00
|
|
|
|
2019-08-16 15:18:53 -04:00
|
|
|
mp_obj_tuple_t *common_hal_bleio_peripheral_discover_remote_services(bleio_peripheral_obj_t *self, mp_obj_t service_uuids_whitelist) {
|
|
|
|
common_hal_bleio_device_discover_remote_services(MP_OBJ_FROM_PTR(self), service_uuids_whitelist);
|
|
|
|
// Convert to a tuple and then clear the list so the callee will take ownership.
|
|
|
|
mp_obj_tuple_t *services_tuple = mp_obj_new_tuple(self->remote_services_list->len,
|
|
|
|
self->remote_services_list->items);
|
|
|
|
mp_obj_list_clear(self->remote_services_list);
|
|
|
|
return services_tuple;
|
2019-07-30 14:26:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void common_hal_bleio_peripheral_pair(bleio_peripheral_obj_t *self) {
|
|
|
|
self->pair_status = PAIR_WAITING;
|
|
|
|
|
|
|
|
uint32_t err_code = sd_ble_gap_authenticate(self->conn_handle, &pairing_sec_params);
|
|
|
|
|
|
|
|
if (err_code != NRF_SUCCESS) {
|
|
|
|
mp_raise_OSError_msg_varg(translate("Failed to start pairing, error 0x%04x"), err_code);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (self->pair_status == PAIR_WAITING) {
|
|
|
|
MICROPY_VM_HOOK_LOOP;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self->pair_status == PAIR_NOT_PAIRED) {
|
|
|
|
mp_raise_OSError_msg(translate("Failed to pair"));
|
|
|
|
}
|
|
|
|
}
|