初めに
↓みたいな感じになります。
参考動画
使ったもの
・AE-TYBLE16
秋月電子で購入
・電池ケース
秋月電子で購入
・タクトスイッチ
秋月電子で購入
・ケーブル
モノタロウで購入
回路図
スイッチと電池つなぐだけです。
ソースコード
nRF5_SDK_12.3のble_app_hrs_cをベースプロジェクトとして修正しました。
修正したファイルは以下の3つ。(クロックの設定などは前回の記事で変更していること前提)
・main.c
・sdk_config.h(ble_app_hrs_c\pca10028\s130\config内)
・pca10028.h(components\boards内)
修正してビルドして、作成したhexファイルをSoftDeviceS130のHexと一緒に書き込んでおしまいです。
sdk_config.h
以下の箇所を変更
706行目
#define GPIOTE_CONFIG_IRQ_PRIORITY 3
=> #define GPIOTE_CONFIG_IRQ_PRIORITY 1
2708行目
#define UART_ENABLED 1
=> #define UART_ENABLED 0
2782行目
#define UART0_ENABLED 1
=> #define UART0_ENABLED 0
3670行目
#define NRF_LOG_BACKEND_SERIAL_USES_UART 1
=>#define NRF_LOG_BACKEND_SERIAL_USES_UART 0
3716行目
#define NRF_LOG_BACKEND_SERIAL_USES_RTT 0
=> #define NRF_LOG_BACKEND_SERIAL_USES_RTT 1
pca10028.h
73,74,75行目
#define BUTTON_1 17
#define BUTTON_2 18
#define BUTTON_3 19
=>
#define BUTTON_1 18
#define BUTTON_2 17
#define BUTTON_3 5
にする。
main.c
変更箇所が多いのでコードをそのまま載せます。
/**
* Copyright (c) 2014 - 2017, Nordic Semiconductor ASA
*
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form, except as embedded into a Nordic
* Semiconductor ASA integrated circuit in a product or a software update for
* such product, must reproduce the above copyright notice, this list of
* conditions and the following disclaimer in the documentation and/or other
* materials provided with the distribution.
*
* 3. Neither the name of Nordic Semiconductor ASA nor the names of its
* contributors may be used to endorse or promote products derived from this
* software without specific prior written permission.
*
* 4. This software, with or without modification, must only be used with a
* Nordic Semiconductor ASA integrated circuit.
*
* 5. Any software provided in binary form under this license must not be reverse
* engineered, decompiled, modified and/or disassembled.
*
* THIS SOFTWARE IS PROVIDED BY NORDIC SEMICONDUCTOR ASA "AS IS" AND ANY EXPRESS
* OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY, NONINFRINGEMENT, AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL NORDIC SEMICONDUCTOR ASA OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
* GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
/**
* @brief BLE Heart Rate Collector application main file.
*
* This file contains the source code for a sample heart rate collector.
*/
#include <stdint.h>
#include <stdio.h>
#include <string.h>
#include "nordic_common.h"
#include "nrf_sdm.h"
#include "ble.h"
#include "ble_hci.h"
#include "ble_db_discovery.h"
#include "softdevice_handler.h"
#include "app_util.h"
#include "app_error.h"
#include "boards.h"
#include "nrf_gpio.h"
#include "peer_manager.h"
#include "ble_hrs_c.h"
#include "ble_bas_c.h"
#include "app_util.h"
#include "app_timer.h"
#include "bsp.h"
#include "bsp_btn_ble.h"
#include "fds.h"
#include "fstorage.h"
#include "ble_conn_state.h"
#include "nrf_ble_gatt.h"
#define NRF_LOG_MODULE_NAME "APP"
#include "nrf_log.h"
#include "nrf_log_ctrl.h"
#define CENTRAL_LINK_COUNT 1 /**< Number of central links used by the application. When changing this number remember to adjust the RAM settings*/
#define PERIPHERAL_LINK_COUNT 0 /**< Number of peripheral links used by the application. When changing this number remember to adjust the RAM settings*/
#define APP_TIMER_PRESCALER 2 /**< Value of the RTC1 PRESCALER register. */
#define APP_TIMER_OP_QUEUE_SIZE 2 /**< Size of timer operation queues. */
#define SEC_PARAM_BOND 1 /**< Perform bonding. */
#define SEC_PARAM_MITM 0 /**< Man In The Middle protection not required. */
#define SEC_PARAM_LESC 0 /**< LE Secure Connections not enabled. */
#define SEC_PARAM_KEYPRESS 0 /**< Keypress notifications not enabled. */
#define SEC_PARAM_IO_CAPABILITIES BLE_GAP_IO_CAPS_NONE /**< No I/O capabilities. */
#define SEC_PARAM_OOB 0 /**< Out Of Band data not available. */
#define SEC_PARAM_MIN_KEY_SIZE 7 /**< Minimum encryption key size in octets. */
#define SEC_PARAM_MAX_KEY_SIZE 16 /**< Maximum encryption key size in octets. */
#define SCAN_INTERVAL 0x00A0 /**< Determines scan interval in units of 0.625 millisecond. */
#define SCAN_WINDOW 0x0050 /**< Determines scan window in units of 0.625 millisecond. */
#define MIN_CONNECTION_INTERVAL MSEC_TO_UNITS(7.5, UNIT_1_25_MS) /**< Determines minimum connection interval in millisecond. */
#define MAX_CONNECTION_INTERVAL MSEC_TO_UNITS(30, UNIT_1_25_MS) /**< Determines maximum connection interval in millisecond. */
#define SLAVE_LATENCY 0 /**< Determines slave latency in counts of connection events. */
#define SUPERVISION_TIMEOUT MSEC_TO_UNITS(4000, UNIT_10_MS) /**< Determines supervision time-out in units of 10 millisecond. */
#define BLE_UUID_ELINK_SHIFT_CHARA 0x1524U /**< Target device name that application is looking for. */
#define BLE_UUID_ELINK_SERVICE 0x1523U
#define INIT_WRITE_MESSAGE_LENGTH 18U
#define SHIFT_WRITE_MESSAGE_LENGTH 4U
#define SYSTEM_OFF_TIME_SCAN APP_TIMER_TICKS(60000, APP_TIMER_PRESCALER) /*10s*/
#define SYSTEM_OFF_TIME_CONN APP_TIMER_TICKS(600000, APP_TIMER_PRESCALER) /*600s*/
#define BUTTON_PUSH_CHECK_TIME APP_TIMER_TICKS(300, APP_TIMER_PRESCALER) /*ms*/
/**
* @defgroup hrs_c_enums Enumerations
* @{
*/
/**@breif Macro to unpack 16bit unsigned UUID from octet stream. */
#define UUID16_EXTRACT(DST, SRC) \
do \
{ \
(*(DST)) = (SRC); \
(*(DST)) <<= 8; \
(*(DST)) |= (SRC)[0]; \
} while (0)
/**@brief Variable length data encapsulation in terms of length and pointer to data */
typedef struct
{
uint8_t * p_data; /**< Pointer to data. */
uint16_t data_len; /**< Length of data. */
} data_t;
typedef enum
{
READ_REQ, /**< Type identifying that this tx_message is a read request. */
WRITE_REQ /**< Type identifying that this tx_message is a write request. */
} tx_request_t;
typedef struct
{
uint8_t gattc_value[INIT_WRITE_MESSAGE_LENGTH]; /**< The message to write. */
ble_gattc_write_params_t gattc_params; /**< GATTC parameters for this message. */
} write_params_t;
/**@brief Structure for holding data to be transmitted to the connected central.
*/
typedef struct
{
uint16_t conn_handle; /**< Connection handle to be used when transmitting this message. */
tx_request_t type; /**< Type of this message, i.e. read or write message. */
union
{
uint16_t read_handle; /**< Read request message. */
write_params_t write_req; /**< Write request message. */
} req;
} tx_message_t;
static ble_db_discovery_t m_ble_db_discovery; /**< Structure used to identify the DB Discovery module. */
static ble_gap_scan_params_t m_scan_param; /**< Scan parameters requested for scanning and connection. */
static uint16_t m_conn_handle; /**< Current connection handle. */
static bool m_whitelist_disabled; /**< True if whitelist has been temporarily disabled. */
static bool m_memory_access_in_progress; /**< Flag to keep track of ongoing operations on persistent memory. */
static nrf_ble_gatt_t m_gatt; /**< Structure for gatt module*/
static bool m_retry_db_disc; /**< Flag to keep track of whether the DB discovery should be retried. */
static uint16_t m_pending_db_disc_conn = BLE_CONN_HANDLE_INVALID; /**< Connection handle for which the DB discovery is retried. */
static tx_message_t m_tx_buffer; /**< Transmit buffer for messages to be transmitted to the central. */
APP_TIMER_DEF(m_sec_req_timer_id);
APP_TIMER_DEF(m_button1_push_timer_id);
APP_TIMER_DEF(m_button2_push_timer_id);
static bool m_push_switch= false;
static bool m_button_timer = false;
/**
* @brief Connection parameters requested for connection.
*/
static const ble_gap_conn_params_t m_connection_param =
{
(uint16_t)MIN_CONNECTION_INTERVAL, // Minimum connection
(uint16_t)MAX_CONNECTION_INTERVAL, // Maximum connection
(uint16_t)SLAVE_LATENCY, // Slave latency.
(uint16_t)SUPERVISION_TIMEOUT // Supervision time-out
};
/**@brief Names which the central applications will scan for, and which will be advertised by the peripherals.
* if these are set to empty strings, the UUIDs defined below will be used
*/
static const char m_target_periph_name[] = "Elink"; /**< If you want to connect to a peripheral using a given advertising name, type its name here. */
static bool is_connect_per_addr = true; /**< If you want to connect to a peripheral with a given address, set this to true and put the correct address in the variable below. */
static const ble_gap_addr_t m_target_periph_addr =
{
/* Possible values for addr_type:
BLE_GAP_ADDR_TYPE_PUBLIC,
BLE_GAP_ADDR_TYPE_RANDOM_STATIC,
BLE_GAP_ADDR_TYPE_RANDOM_PRIVATE_RESOLVABLE,
BLE_GAP_ADDR_TYPE_RANDOM_PRIVATE_NON_RESOLVABLE. */
.addr_type = BLE_GAP_ADDR_TYPE_RANDOM_STATIC,
.addr = {0xEF, 0xE3, 0xC8, 0x09, 0x96, 0xEB} /*Elink MacAddr*/
};
static void scan_start(void);
static uint32_t Elin_Init(uint16_t conn_handle);
static void Elin_ShiftUp(uint16_t conn_handle);
static void Elin_ShiftDown(uint16_t conn_handle);
static void ble_elink_on_db_disc_evt(const ble_db_discovery_evt_t * p_evt);
static void elin_c_init(void);
static uint32_t ble_elin_c_init(void);
static void button1_event_handler(void);
static void button2_event_handler(void);
static void sec_req_timeout_handler(void * p_context);
static void sleep_timer_reset(uint32_t timeout);
static void button1_timer_start(void);
static void button2_timer_start(void);
static void button1_timer_event_handler(void * p_context);
static void button2_timer_event_handler(void * p_context);
static uint32_t Elin_Init(uint16_t conn_handle)
{
uint32_t err_code;
m_tx_buffer.req.write_req.gattc_params.handle = 0x0010U;
m_tx_buffer.req.write_req.gattc_params.len = INIT_WRITE_MESSAGE_LENGTH;
m_tx_buffer.req.write_req.gattc_params.p_value = m_tx_buffer.req.write_req.gattc_value;
m_tx_buffer.req.write_req.gattc_params.offset = 0;
m_tx_buffer.req.write_req.gattc_params.write_op = BLE_GATT_OP_WRITE_REQ;
m_tx_buffer.req.write_req.gattc_value[0] = 0xAA;
m_tx_buffer.req.write_req.gattc_value = 0xCF;
m_tx_buffer.req.write_req.gattc_value = 0x00;
m_tx_buffer.req.write_req.gattc_value = 0x00;
m_tx_buffer.req.write_req.gattc_value = 0x5D;
m_tx_buffer.req.write_req.gattc_value[5] = 0x4A;
m_tx_buffer.req.write_req.gattc_value[6] = 0x81;
m_tx_buffer.req.write_req.gattc_value[7] = 0x89;
m_tx_buffer.req.write_req.gattc_value[8] = 0x32;
m_tx_buffer.req.write_req.gattc_value[9] = 0xF7;
m_tx_buffer.req.write_req.gattc_value[10] = 0x52;
m_tx_buffer.req.write_req.gattc_value[11] = 0x12;
m_tx_buffer.req.write_req.gattc_value[12] = 0x36;
m_tx_buffer.req.write_req.gattc_value[13] = 0x00;
m_tx_buffer.req.write_req.gattc_value[14] = 0x00;
m_tx_buffer.req.write_req.gattc_value[15] = 0x00;
m_tx_buffer.req.write_req.gattc_value[16] = 0x00;
m_tx_buffer.req.write_req.gattc_value[17] = 0x00;
m_tx_buffer.conn_handle = conn_handle;
m_tx_buffer.type = WRITE_REQ;
err_code = sd_ble_gattc_write(m_tx_buffer.conn_handle,
&m_tx_buffer.req.write_req.gattc_params);
if (err_code == NRF_SUCCESS)
{
NRF_LOG_INFO("WriteOK\r\n");
}
else
{
NRF_LOG_INFO("WriteError\r\n");
}
return err_code;
}
static void Elin_ShiftDown(uint16_t conn_handle)
{
uint32_t err_code;
m_tx_buffer.req.write_req.gattc_params.len = SHIFT_WRITE_MESSAGE_LENGTH;
m_tx_buffer.req.write_req.gattc_value[0] = 0x21U;
m_tx_buffer.req.write_req.gattc_value = 0x03U;
m_tx_buffer.req.write_req.gattc_value = 0x00U;
m_tx_buffer.req.write_req.gattc_value = 0x00U;
err_code = sd_ble_gattc_write(m_tx_buffer.conn_handle,
&m_tx_buffer.req.write_req.gattc_params);
if (err_code == NRF_SUCCESS)
{
NRF_LOG_INFO("WriteOK\r\n");
}
else
{
NRF_LOG_INFO("WriteError\r\n");
}
}
static void Elin_ShiftUp(uint16_t conn_handle)
{
uint32_t err_code;
m_tx_buffer.req.write_req.gattc_params.handle = 0x0010U;
m_tx_buffer.req.write_req.gattc_params.len = SHIFT_WRITE_MESSAGE_LENGTH;
m_tx_buffer.req.write_req.gattc_params.p_value = m_tx_buffer.req.write_req.gattc_value;
m_tx_buffer.req.write_req.gattc_params.offset = 0;
m_tx_buffer.req.write_req.gattc_params.write_op = BLE_GATT_OP_WRITE_REQ;
m_tx_buffer.req.write_req.gattc_value[0] = 0x21U;
m_tx_buffer.req.write_req.gattc_value = 0x02U;
m_tx_buffer.req.write_req.gattc_value = 0x00U;
m_tx_buffer.req.write_req.gattc_value = 0x00U;
m_tx_buffer.conn_handle = conn_handle;
m_tx_buffer.type = WRITE_REQ;
err_code = sd_ble_gattc_write(m_tx_buffer.conn_handle,
&m_tx_buffer.req.write_req.gattc_params);
if (err_code == NRF_SUCCESS)
{
NRF_LOG_INFO("WriteOK\r\n");
}
else
{
NRF_LOG_INFO("WriteError\r\n");
}
}
/**@brief Function for asserts in the SoftDevice.
*
* @details This function will be called in case of an assert in the SoftDevice.
*
* @warning This handler is an example only and does not fit a final product. You need to analyze
* how your product is supposed to react in case of Assert.
* @warning On assert from the SoftDevice, the system can only recover on reset.
*
* @param[in] line_num Line number of the failing ASSERT call.
* @param[in] p_file_name File name of the failing ASSERT call.
*/
void assert_nrf_callback(uint16_t line_num, const uint8_t * p_file_name)
{
app_error_handler(0xDEADBEEF, line_num, p_file_name);
}
/**@brief Function for handling database discovery events.
*
* @details This function is callback function to handle events from the database discovery module.
* Depending on the UUIDs that are discovered, this function should forward the events
* to their respective services.
*
* @param[in] p_event Pointer to the database discovery event.
*/
static void db_disc_handler(ble_db_discovery_evt_t * p_evt)
{
ble_elink_on_db_disc_evt( p_evt);
}
static void ble_elink_on_db_disc_evt( const ble_db_discovery_evt_t * p_evt)
{
uint32_t err_code;
// Check if the Heart Rate Service was discovered.
if (p_evt->evt_type == BLE_DB_DISCOVERY_COMPLETE &&
p_evt->params.discovered_db.srv_uuid.uuid == BLE_UUID_ELINK_SERVICE &&
p_evt->params.discovered_db.srv_uuid.type == BLE_UUID_TYPE_BLE)
{
NRF_LOG_INFO("Comp Discover.\r\n");
err_code = Elin_Init(p_evt->conn_handle);
if (err_code == NRF_SUCCESS)
{
m_conn_handle = p_evt->conn_handle;
sleep_timer_reset(SYSTEM_OFF_TIME_CONN);
}else{
err_code = sd_ble_gap_disconnect(p_evt->conn_handle,
BLE_HCI_REMOTE_USER_TERMINATED_CONNECTION);
APP_ERROR_CHECK(err_code);
}
}
}
static void timers_init(void){
uint32_t err_code;
err_code = app_timer_create(&m_sec_req_timer_id,
APP_TIMER_MODE_SINGLE_SHOT,
sec_req_timeout_handler);
APP_ERROR_CHECK(err_code);
}
static void sleep_timer_reset(uint32_t timeout){
uint32_t err_code;
err_code = app_timer_stop(m_sec_req_timer_id);
APP_ERROR_CHECK(err_code);
err_code = app_timer_create(&m_sec_req_timer_id,
APP_TIMER_MODE_SINGLE_SHOT,
sec_req_timeout_handler);
APP_ERROR_CHECK(err_code);
err_code = app_timer_start(m_sec_req_timer_id,
timeout,
NULL);
APP_ERROR_CHECK(err_code);
}
/**@brief Function for handling Peer Manager events.
*
* @param[in] p_evt Peer Manager event.
*/
static void pm_evt_handler(pm_evt_t const * p_evt)
{
ret_code_t err_code;
switch (p_evt->evt_id)
{
case PM_EVT_BONDED_PEER_CONNECTED:
{
NRF_LOG_INFO("Connected to a previously bonded device.\r\n");
} break;
case PM_EVT_CONN_SEC_SUCCEEDED:
{
NRF_LOG_INFO("Connection secured. Role: %d. conn_handle: %d, Procedure: %d\r\n",
ble_conn_state_role(p_evt->conn_handle),
p_evt->conn_handle,
p_evt->params.conn_sec_succeeded.procedure);
} break;
case PM_EVT_CONN_SEC_FAILED:
{
/* Often, when securing fails, it shouldn't be restarted, for security reasons.
* Other times, it can be restarted directly.
* Sometimes it can be restarted, but only after changing some Security Parameters.
* Sometimes, it cannot be restarted until the link is disconnected and reconnected.
* Sometimes it is impossible, to secure the link, or the peer device does not support it.
* How to handle this error is highly application dependent. */
} break;
case PM_EVT_CONN_SEC_CONFIG_REQ:
{
// Reject pairing request from an already bonded peer.
pm_conn_sec_config_t conn_sec_config = {.allow_repairing = false};
pm_conn_sec_config_reply(p_evt->conn_handle, &conn_sec_config);
} break;
case PM_EVT_STORAGE_FULL:
{
// Run garbage collection on the flash.
err_code = fds_gc();
if (err_code == FDS_ERR_BUSY || err_code == FDS_ERR_NO_SPACE_IN_QUEUES)
{
// Retry.
}
else
{
APP_ERROR_CHECK(err_code);
}
} break;
case PM_EVT_PEERS_DELETE_SUCCEEDED:
{
scan_start();
} break;
case PM_EVT_LOCAL_DB_CACHE_APPLY_FAILED:
{
// The local database has likely changed, send service changed indications.
pm_local_database_has_changed();
} break;
case PM_EVT_PEER_DATA_UPDATE_FAILED:
{
// Assert.
APP_ERROR_CHECK(p_evt->params.peer_data_update_failed.error);
} break;
case PM_EVT_PEER_DELETE_FAILED:
{
// Assert.
APP_ERROR_CHECK(p_evt->params.peer_delete_failed.error);
} break;
case PM_EVT_PEERS_DELETE_FAILED:
{
// Assert.
APP_ERROR_CHECK(p_evt->params.peers_delete_failed_evt.error);
} break;
case PM_EVT_ERROR_UNEXPECTED:
{
// Assert.
APP_ERROR_CHECK(p_evt->params.error_unexpected.error);
} break;
case PM_EVT_CONN_SEC_START:
case PM_EVT_PEER_DATA_UPDATE_SUCCEEDED:
case PM_EVT_PEER_DELETE_SUCCEEDED:
case PM_EVT_LOCAL_DB_CACHE_APPLIED:
case PM_EVT_SERVICE_CHANGED_IND_SENT:
case PM_EVT_SERVICE_CHANGED_IND_CONFIRMED:
default:
break;
}
}
/**
* @brief Parses advertisement data, providing length and location of the field in case
* matching data is found.
*
* @param[in] Type of data to be looked for in advertisement data.
* @param[in] Advertisement report length and pointer to report.
* @param[out] If data type requested is found in the data report, type data length and
* pointer to data will be populated here.
*
* @retval NRF_SUCCESS if the data type is found in the report.
* @retval NRF_ERROR_NOT_FOUND if the data type could not be found.
*/
static uint32_t adv_report_parse(uint8_t type, data_t * p_advdata, data_t * p_typedata)
{
uint32_t index = 0;
uint8_t * p_data;
p_data = p_advdata->p_data;
while (index < p_advdata->data_len)
{
uint8_t field_length = p_data[index];
uint8_t field_type = p_data[index + 1];
if (field_type == type)
{
p_typedata->p_data = &p_data[index + 2];
p_typedata->data_len = field_length - 1;
return NRF_SUCCESS;
}
index += field_length + 1;
}
return NRF_ERROR_NOT_FOUND;
}
/**@brief Function for putting the chip into sleep mode.
*
* @note This function will not return.
*/
static void sleep_mode_enter(void)
{ uint32_t err_code;
// Go to system-off mode (this function will not return; wakeup will cause a reset).
err_code = sd_power_system_off();
APP_ERROR_CHECK(err_code);
}
/**@brief Function for searching a given name in the advertisement packets.
*
* @details Use this function to parse received advertising data and to find a given
* name in them either as 'complete_local_name' or as 'short_local_name'.
*
* @param[in] p_adv_report advertising data to parse.
* @param[in] name_to_find name to search.
* @return true if the given name was found, false otherwise.
*/
static bool find_adv_name(const ble_gap_evt_adv_report_t *p_adv_report, const char * name_to_find)
{
uint32_t err_code;
data_t adv_data;
data_t dev_name;
// Initialize advertisement report for parsing
adv_data.p_data = (uint8_t *)p_adv_report->data;
adv_data.data_len = p_adv_report->dlen;
//search for advertising names
err_code = adv_report_parse(BLE_GAP_AD_TYPE_COMPLETE_LOCAL_NAME,
&adv_data,
&dev_name);
if (err_code == NRF_SUCCESS)
{
if (memcmp(name_to_find, dev_name.p_data, dev_name.data_len )== 0)
{
return true;
}
}
else
{
// Look for the short local name if it was not found as complete
err_code = adv_report_parse(BLE_GAP_AD_TYPE_SHORT_LOCAL_NAME,
&adv_data,
&dev_name);
if (err_code != NRF_SUCCESS)
{
return false;
}
if (memcmp(m_target_periph_name, dev_name.p_data, dev_name.data_len )== 0)
{
return true;
}
}
return false;
}
/**@brief Function for searching a given addr in the advertisement packets.
*
* @details Use this function to parse received advertising data and to find a given
* addr in them.
*
* @param[in] p_adv_report advertising data to parse.
* @param[in] p_addr name to search.
* @return true if the given name was found, false otherwise.
*/
static bool find_peer_addr(const ble_gap_evt_adv_report_t *p_adv_report, const ble_gap_addr_t * p_addr)
{
if (p_addr->addr_type == p_adv_report->peer_addr.addr_type)
{
if (memcmp(p_addr->addr, p_adv_report->peer_addr.addr, sizeof(p_adv_report->peer_addr.addr)) == 0)
{
return true;
}
}
return false;
}
/**@brief Function for searching a UUID in the advertisement packets.
*
* @details Use this function to parse received advertising data and to find a given
* UUID in them.
*
* @param[in] p_adv_report advertising data to parse.
* @param[in] uuid_to_find UUIID to search.
* @return true if the given UUID was found, false otherwise.
*/
static bool find_adv_uuid(const ble_gap_evt_adv_report_t *p_adv_report, const uint16_t uuid_to_find)
{
uint32_t err_code;
data_t adv_data;
data_t type_data;
// Initialize advertisement report for parsing.
adv_data.p_data = (uint8_t *)p_adv_report->data;
adv_data.data_len = p_adv_report->dlen;
err_code = adv_report_parse(BLE_GAP_AD_TYPE_16BIT_SERVICE_UUID_MORE_AVAILABLE,
&adv_data,
&type_data);
if (err_code != NRF_SUCCESS)
{
// Look for the services in 'complete' if it was not found in 'more available'.
err_code = adv_report_parse(BLE_GAP_AD_TYPE_16BIT_SERVICE_UUID_COMPLETE,
&adv_data,
&type_data);
if (err_code != NRF_SUCCESS)
{
// If we can't parse the data, then exit.
return false;
}
}
// Verify if any UUID match the given UUID.
for (uint32_t u_index = 0; u_index < (type_data.data_len / sizeof(uint16_t)); u_index++)
{
uint16_t extracted_uuid;
UUID16_EXTRACT(&extracted_uuid, &type_data.p_data[u_index * sizeof(uint16_t)]);
if (extracted_uuid == uuid_to_find)
{
return true;
}
}
return false;
}
/**@brief Function for handling the Application's BLE Stack events.
*
* @param[in] p_ble_evt Bluetooth stack event.
*/
static void on_ble_evt(ble_evt_t * p_ble_evt)
{
uint32_t err_code;
const ble_gap_evt_t * p_gap_evt = &p_ble_evt->evt.gap_evt;
switch (p_ble_evt->header.evt_id)
{
case BLE_GAP_EVT_CONNECTED:
{
NRF_LOG_INFO("Connected.\r\n");
m_pending_db_disc_conn = p_ble_evt->evt.gap_evt.conn_handle;
m_retry_db_disc = false;
// Discover peer's services.
err_code = ble_db_discovery_start(&m_ble_db_discovery, m_pending_db_disc_conn);
if (err_code == NRF_ERROR_BUSY)
{
NRF_LOG_INFO("ble_db_discovery_start() returned busy, will retry later.\r\n");
m_retry_db_disc = true;
}
else
{
APP_ERROR_CHECK(err_code);
}
if (ble_conn_state_n_centrals() < NRF_BLE_CENTRAL_LINK_COUNT)
{
scan_start();
}
} break;
case BLE_GAP_EVT_ADV_REPORT:
{
bool do_connect = false;
if (is_connect_per_addr)
{
if (find_peer_addr(&p_gap_evt->params.adv_report, &m_target_periph_addr))
{
NRF_LOG_INFO("Address match send connect_request.\r\n");
do_connect = true;
}
}
if(!do_connect){
if (strlen(m_target_periph_name) != 0)
{
if (find_adv_name(&p_gap_evt->params.adv_report, m_target_periph_name))
{
do_connect = true;
NRF_LOG_INFO("Name match send connect_request.\r\n");
}
}
else
{
if (find_adv_uuid(&p_gap_evt->params.adv_report, BLE_UUID_ELINK_SERVICE))
{
do_connect = true;
NRF_LOG_INFO("UUID match send connect_request.\r\n");
}
}
}
if (do_connect)
{
// Stop scanning.
(void) sd_ble_gap_scan_stop();
#if (NRF_SD_BLE_API_VERSION == 2)
m_scan_param.selective = 0;
#endif
#if (NRF_SD_BLE_API_VERSION == 3)
m_scan_param.use_whitelist = 0;
#endif
// Initiate connection.
err_code = sd_ble_gap_connect(&p_gap_evt->params.adv_report.peer_addr,
&m_scan_param,
&m_connection_param);
m_whitelist_disabled = false;
if (err_code != NRF_SUCCESS)
{
NRF_LOG_ERROR("Connection Request Failed, reason %d.\r\n", err_code);
}
}
} break; // BLE_GAP_EVT_ADV_REPORT
case BLE_GAP_EVT_DISCONNECTED:
{
NRF_LOG_INFO("Disconnected, reason 0x%x.\r\n",
p_ble_evt->evt.gap_evt.params.disconnected.reason);
m_conn_handle = BLE_CONN_HANDLE_INVALID;
// Reset DB discovery structure.
memset(&m_ble_db_discovery, 0 , sizeof (m_ble_db_discovery));
if (ble_conn_state_n_centrals() < NRF_BLE_CENTRAL_LINK_COUNT)
{
scan_start();
}
} break;
case BLE_GAP_EVT_TIMEOUT:
{
if (p_gap_evt->params.timeout.src == BLE_GAP_TIMEOUT_SRC_SCAN)
{
NRF_LOG_DEBUG("Scan timed out.\r\n");
scan_start();
}
else if (p_gap_evt->params.timeout.src == BLE_GAP_TIMEOUT_SRC_CONN)
{
NRF_LOG_INFO("Connection Request timed out.\r\n");
}
} break;
case BLE_GAP_EVT_CONN_PARAM_UPDATE_REQUEST:
// Accepting parameters requested by peer.
err_code = sd_ble_gap_conn_param_update(p_gap_evt->conn_handle,
&p_gap_evt->params.conn_param_update_request.conn_params);
APP_ERROR_CHECK(err_code);
break;
case BLE_GATTC_EVT_TIMEOUT:
// Disconnect on GATT Client timeout event.
NRF_LOG_DEBUG("GATT Client Timeout.\r\n");
err_code = sd_ble_gap_disconnect(p_ble_evt->evt.gattc_evt.conn_handle,
BLE_HCI_REMOTE_USER_TERMINATED_CONNECTION);
APP_ERROR_CHECK(err_code);
break;
case BLE_GATTS_EVT_TIMEOUT:
// Disconnect on GATT Server timeout event.
NRF_LOG_DEBUG("GATT Server Timeout.\r\n");
err_code = sd_ble_gap_disconnect(p_ble_evt->evt.gatts_evt.conn_handle,
BLE_HCI_REMOTE_USER_TERMINATED_CONNECTION);
APP_ERROR_CHECK(err_code);
break;
default:
break;
}
}
/**@brief Function for handling the Application's system events.
*
* @param[in] sys_evt system event.
*/
static void on_sys_evt(uint32_t sys_evt)
{
switch (sys_evt)
{
case NRF_EVT_FLASH_OPERATION_SUCCESS:
/* fall through */
case NRF_EVT_FLASH_OPERATION_ERROR:
if (m_memory_access_in_progress)
{
m_memory_access_in_progress = false;
scan_start();
}
break;
default:
// No implementation needed.
break;
}
}
/**@brief Function for dispatching a BLE stack event to all modules with a BLE stack event handler.
*
* @details This function is called from the scheduler in the main loop after a BLE stack event has
* been received.
*
* @param[in] p_ble_evt Bluetooth stack event.
*/
static void ble_evt_dispatch(ble_evt_t * p_ble_evt)
{
// Modules which depend on ble_conn_state, like Peer Manager,
// should have their callbacks invoked after ble_conn_state's.
ble_conn_state_on_ble_evt(p_ble_evt);
pm_on_ble_evt(p_ble_evt);
ble_db_discovery_on_ble_evt(&m_ble_db_discovery, p_ble_evt);
bsp_btn_ble_on_ble_evt(p_ble_evt);
nrf_ble_gatt_on_ble_evt(&m_gatt, p_ble_evt);
on_ble_evt(p_ble_evt);
}
/**@brief Function for dispatching a system event to interested modules.
*
* @details This function is called from the System event interrupt handler after a system
* event has been received.
*
* @param[in] sys_evt System stack event.
*/
static void sys_evt_dispatch(uint32_t sys_evt)
{
fs_sys_event_handler(sys_evt);
on_sys_evt(sys_evt);
}
/**@brief Function for initializing the BLE stack.
*
* @details Initializes the SoftDevice and the BLE event interrupt.
*/
static void ble_stack_init(void)
{
uint32_t err_code;
nrf_clock_lf_cfg_t clock_lf_cfg;
clock_lf_cfg.rc_ctiv = 16;
clock_lf_cfg.rc_temp_ctiv = 2;
clock_lf_cfg.source = NRF_CLOCK_LF_SRC_RC;
clock_lf_cfg.xtal_accuracy = NRF_CLOCK_LF_XTAL_ACCURACY_250_PPM;
// Initialize the SoftDevice handler module.
SOFTDEVICE_HANDLER_INIT(&clock_lf_cfg, NULL);
ble_enable_params_t ble_enable_params;
err_code = softdevice_enable_get_default_config(NRF_BLE_CENTRAL_LINK_COUNT,
NRF_BLE_PERIPHERAL_LINK_COUNT,
&ble_enable_params);
APP_ERROR_CHECK(err_code);
//Check the ram settings against the used number of links
CHECK_RAM_START_ADDR(NRF_BLE_CENTRAL_LINK_COUNT, NRF_BLE_PERIPHERAL_LINK_COUNT);
// Enable BLE stack.
#if (NRF_SD_BLE_API_VERSION == 3)
ble_enable_params.gatt_enable_params.att_mtu = NRF_BLE_GATT_MAX_MTU_SIZE;
#endif
err_code = softdevice_enable(&ble_enable_params);
APP_ERROR_CHECK(err_code);
// Register with the SoftDevice handler module for BLE events.
err_code = softdevice_ble_evt_handler_set(ble_evt_dispatch);
APP_ERROR_CHECK(err_code);
// Register with the SoftDevice handler module for System events.
err_code = softdevice_sys_evt_handler_set(sys_evt_dispatch);
APP_ERROR_CHECK(err_code);
}
/**@brief Function for the Peer Manager initialization.
*
* @param[in] erase_bonds Indicates whether bonding information should be cleared from
* persistent storage during initialization of the Peer Manager.
*/
static void peer_manager_init(bool erase_bonds)
{
ble_gap_sec_params_t sec_param;
ret_code_t err_code;
err_code = pm_init();
APP_ERROR_CHECK(err_code);
if (erase_bonds)
{
err_code = pm_peers_delete();
APP_ERROR_CHECK(err_code);
}
memset(&sec_param, 0, sizeof(ble_gap_sec_params_t));
// Security parameters to be used for all security procedures.
sec_param.bond = SEC_PARAM_BOND;
sec_param.mitm = SEC_PARAM_MITM;
sec_param.lesc = SEC_PARAM_LESC;
sec_param.keypress = SEC_PARAM_KEYPRESS;
sec_param.io_caps = SEC_PARAM_IO_CAPABILITIES;
sec_param.oob = SEC_PARAM_OOB;
sec_param.min_key_size = SEC_PARAM_MIN_KEY_SIZE;
sec_param.max_key_size = SEC_PARAM_MAX_KEY_SIZE;
sec_param.kdist_own.enc = 1;
sec_param.kdist_own.id = 1;
sec_param.kdist_peer.enc = 1;
sec_param.kdist_peer.id = 1;
err_code = pm_sec_params_set(&sec_param);
APP_ERROR_CHECK(err_code);
err_code = pm_register(pm_evt_handler);
APP_ERROR_CHECK(err_code);
}
/**@brief Function for handling events from the BSP module.
*
* @param[in] event Event generated by button press.
*/
void bsp_event_handler(bsp_event_t event)
{
switch (event)
{
case BSP_EVENT_KEY_1: /*P017 Push*/
button1_event_handler();
button1_timer_start();
/*timer start*/
break;
case BSP_EVENT_KEY_2: /*P005 Push*/
button2_event_handler();
button2_timer_start();
break;
default:
break;
}
}
static void button1_timer_start(void){
uint32_t err_code;
if(!m_button_timer){
m_button_timer = true;
err_code = app_timer_create(&m_button1_push_timer_id,
APP_TIMER_MODE_SINGLE_SHOT,
button1_timer_event_handler);
APP_ERROR_CHECK(err_code);
err_code = app_timer_start(m_button1_push_timer_id,
BUTTON_PUSH_CHECK_TIME,
NULL);
APP_ERROR_CHECK(err_code);
}
}
static void button1_timer_event_handler(void * context){
uint32_t err_code;
if(bsp_button_is_pressed(1)){/*push */
button1_event_handler();
err_code = app_timer_start(m_button1_push_timer_id,
BUTTON_PUSH_CHECK_TIME,
NULL);
APP_ERROR_CHECK(err_code);
}else{
m_button_timer = false;
}
}
static void button2_timer_start(void){
uint32_t err_code;
if(!m_button_timer){
m_button_timer = true;
err_code = app_timer_create(&m_button2_push_timer_id,
APP_TIMER_MODE_SINGLE_SHOT,
button2_timer_event_handler);
APP_ERROR_CHECK(err_code);
err_code = app_timer_start(m_button2_push_timer_id,
BUTTON_PUSH_CHECK_TIME,
NULL);
APP_ERROR_CHECK(err_code);
}
}
static void button2_timer_event_handler(void * context){
uint32_t err_code;
if(bsp_button_is_pressed(2)){/*push */
button2_event_handler();
err_code = app_timer_start(m_button2_push_timer_id,
BUTTON_PUSH_CHECK_TIME,
NULL);
APP_ERROR_CHECK(err_code);
}else{
m_button_timer = false;
}
}
/**@brief Function for handling the security request timer time-out.
*
* @details This function is called each time the security request timer expires.
*
* @param[in] p_context Pointer used for passing context information from the
* app_start_timer() call to the time-out handler.
*/
static void sec_req_timeout_handler(void * p_context)
{
sleep_mode_enter();
}
/**
* @brief Heart rate collector initialization.
*/
static void elin_c_init(void)
{
uint32_t err_code = ble_elin_c_init();
APP_ERROR_CHECK(err_code);
}
static uint32_t ble_elin_c_init(void)
{
ble_uuid_t elin_uuid;
elin_uuid.type = BLE_UUID_TYPE_BLE;
elin_uuid.uuid = BLE_UUID_ELINK_SERVICE;
m_conn_handle = BLE_CONN_HANDLE_INVALID;
return ble_db_discovery_evt_register(&elin_uuid);
}
/**
* @brief Database discovery collector initialization.
*/
static void db_discovery_init(void)
{
uint32_t err_code = ble_db_discovery_init(db_disc_handler);
APP_ERROR_CHECK(err_code);
}
/**@brief Retrieve a list of peer manager peer IDs.
*
* @param[inout] p_peers The buffer where to store the list of peer IDs.
* @param[inout] p_size In: The size of the @p p_peers buffer.
* Out: The number of peers copied in the buffer.
*/
static void peer_list_get(pm_peer_id_t * p_peers, uint32_t * p_size)
{
pm_peer_id_t peer_id;
uint32_t peers_to_copy;
peers_to_copy = (*p_size < BLE_GAP_WHITELIST_ADDR_MAX_COUNT) ?
*p_size : BLE_GAP_WHITELIST_ADDR_MAX_COUNT;
peer_id = pm_next_peer_id_get(PM_PEER_ID_INVALID);
*p_size = 0;
while ((peer_id != PM_PEER_ID_INVALID) && (peers_to_copy--))
{
p_peers[(*p_size)++] = peer_id;
peer_id = pm_next_peer_id_get(peer_id);
}
}
static void whitelist_load()
{
ret_code_t ret;
pm_peer_id_t peers[8];
uint32_t peer_cnt;
memset(peers, PM_PEER_ID_INVALID, sizeof(peers));
peer_cnt = (sizeof(peers) / sizeof(pm_peer_id_t));
// Load all peers from flash and whitelist them.
peer_list_get(peers, &peer_cnt);
ret = pm_whitelist_set(peers, peer_cnt);
APP_ERROR_CHECK(ret);
// Setup the device identies list.
// Some SoftDevices do not support this feature.
ret = pm_device_identities_list_set(peers, peer_cnt);
if (ret != NRF_ERROR_NOT_SUPPORTED)
{
APP_ERROR_CHECK(ret);
}
}
/**@brief Function to start scanning.
*/
static void scan_start(void)
{
uint32_t flash_busy;
// If there is any pending write to flash, defer scanning until it completes.
(void) fs_queued_op_count_get(&flash_busy);
if (flash_busy != 0)
{
m_memory_access_in_progress = true;
return;
}
// Whitelist buffers.
ble_gap_addr_t whitelist_addrs[8];
ble_gap_irk_t whitelist_irks[8];
memset(whitelist_addrs, 0x00, sizeof(whitelist_addrs));
memset(whitelist_irks, 0x00, sizeof(whitelist_irks));
uint32_t addr_cnt = (sizeof(whitelist_addrs) / sizeof(ble_gap_addr_t));
uint32_t irk_cnt = (sizeof(whitelist_irks) / sizeof(ble_gap_irk_t));
#if (NRF_SD_BLE_API_VERSION == 2)
ble_gap_addr_t * p_whitelist_addrs[8];
ble_gap_irk_t * p_whitelist_irks[8];
for (uint32_t i = 0; i < 8; i++)
{
p_whitelist_addrs[i] = &whitelist_addrs[i];
p_whitelist_irks[i] = &whitelist_irks[i];
}
ble_gap_whitelist_t whitelist =
{
.pp_addrs = p_whitelist_addrs,
.pp_irks = p_whitelist_irks,
};
#endif
// Reload the whitelist and whitelist all peers.
whitelist_load();
ret_code_t ret;
// Get the whitelist previously set using pm_whitelist_set().
ret = pm_whitelist_get(whitelist_addrs, &addr_cnt,
whitelist_irks, &irk_cnt);
m_scan_param.active = 0;
m_scan_param.interval = SCAN_INTERVAL;
m_scan_param.window = SCAN_WINDOW;
if (((addr_cnt == 0) && (irk_cnt == 0)) ||
(m_whitelist_disabled))
{
// Don't use whitelist.
#if (NRF_SD_BLE_API_VERSION == 2)
m_scan_param.selective = 0;
m_scan_param.p_whitelist = NULL;
#endif
#if (NRF_SD_BLE_API_VERSION == 3)
m_scan_param.use_whitelist = 0;
m_scan_param.adv_dir_report = 0;
#endif
m_scan_param.timeout = 0x0000; // No timeout.
}
else
{
// Use whitelist.
#if (NRF_SD_BLE_API_VERSION == 2)
whitelist.addr_count = addr_cnt;
whitelist.irk_count = irk_cnt;
m_scan_param.selective = 1;
m_scan_param.p_whitelist = &whitelist;
#endif
#if (NRF_SD_BLE_API_VERSION == 3)
m_scan_param.use_whitelist = 1;
m_scan_param.adv_dir_report = 0;
#endif
m_scan_param.timeout = 0x001E; // 30 seconds.
}
NRF_LOG_INFO("Starting scan.\r\n");
ret = sd_ble_gap_scan_start(&m_scan_param);
APP_ERROR_CHECK(ret);
sleep_timer_reset(SYSTEM_OFF_TIME_SCAN);
}
/**@brief Function for initializing buttons and leds.
*
* @param[out] p_erase_bonds Will be true if the clear bonding button was pressed to wake the application up.
*/
static void buttons_leds_init(bool * p_erase_bonds)
{
bsp_event_t startup_event;
uint32_t err_code = bsp_init(BSP_INIT_BUTTONS,
APP_TIMER_TICKS(100, APP_TIMER_PRESCALER),
bsp_event_handler);
APP_ERROR_CHECK(err_code);
err_code = bsp_btn_ble_init(NULL, &startup_event);
APP_ERROR_CHECK(err_code);
*p_erase_bonds = (startup_event == BSP_EVENT_CLEAR_BONDING_DATA);
}
/**@brief Function for initializing the nrf log module.
*/
static void log_init(void)
{
ret_code_t err_code = NRF_LOG_INIT(NULL);
APP_ERROR_CHECK(err_code);
}
/** @brief Function for the Power manager.
*/
static void power_manage(void)
{
uint32_t err_code = sd_app_evt_wait();
APP_ERROR_CHECK(err_code);
}
/* GATT generic Event handler. */
void gatt_evt_handler(nrf_ble_gatt_t * p_gatt, nrf_ble_gatt_evt_t * p_evt)
{
NRF_LOG_INFO("GATT MTU on link %d changed to %d\r\n",
p_evt->conn_handle,
p_evt->att_mtu_effective);
if (m_retry_db_disc)
{
ret_code_t err_code;
NRF_LOG_DEBUG("Retrying DB discovery.\r\n");
m_retry_db_disc = false;
// Discover peer's services.
err_code = ble_db_discovery_start(&m_ble_db_discovery, m_pending_db_disc_conn);
if (err_code == NRF_ERROR_BUSY)
{
NRF_LOG_DEBUG("ble_db_discovery_start() returned busy, will retry later.\r\n");
m_retry_db_disc = true;
}
else
{
APP_ERROR_CHECK(err_code);
}
}
}
/* GATT Module init. */
void gatt_init(void)
{
ret_code_t err_code = nrf_ble_gatt_init(&m_gatt, gatt_evt_handler);
APP_ERROR_CHECK(err_code);
}
int main(void)
{
bool erase_bonds;
// Initialize.
APP_TIMER_INIT(APP_TIMER_PRESCALER, APP_TIMER_OP_QUEUE_SIZE, NULL);
timers_init();
buttons_leds_init(&erase_bonds);
log_init();
ble_stack_init();
peer_manager_init(false);
gatt_init();
db_discovery_init();
elin_c_init();
scan_start();
for (;;)
{
if (NRF_LOG_PROCESS() == false)
{
power_manage();
if(m_push_switch){
sleep_timer_reset(SYSTEM_OFF_TIME_CONN);
m_push_switch = false;
}
}
}
}
static void button1_event_handler(void){
if(m_conn_handle != BLE_CONN_HANDLE_INVALID){
m_push_switch = true;
Elin_ShiftUp(m_conn_handle);
}
}
static void button2_event_handler(void){
if(m_conn_handle != BLE_CONN_HANDLE_INVALID){
m_push_switch = true;
Elin_ShiftDown(m_conn_handle);
}
}