| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306 |
- /*
- * Licensed to the Apache Software Foundation (ASF) under one
- * or more contributor license agreements. See the NOTICE file
- * distributed with this work for additional information
- * regarding copyright ownership. The ASF licenses this file
- * to you 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 "esp_log.h"
- #include "nvs_flash.h"
- #include "freertos/FreeRTOSConfig.h"
- /* BLE */
- #include "nimble/nimble_port.h"
- #include "nimble/nimble_port_freertos.h"
- #include "host/ble_hs.h"
- #include "host/util/util.h"
- #include "console/console.h"
- #include "services/gap/ble_svc_gap.h"
- #include "blehr_sens.h"
- static const char *tag = "NimBLE_BLE_HeartRate";
- static TimerHandle_t blehr_tx_timer;
- static bool notify_state;
- static uint16_t conn_handle;
- static const char *device_name = "blehr_sensor_1.0";
- static int blehr_gap_event(struct ble_gap_event *event, void *arg);
- static uint8_t blehr_addr_type;
- /* Variable to simulate heart beats */
- static uint8_t heartrate = 90;
- /**
- * Utility function to log an array of bytes.
- */
- void
- print_bytes(const uint8_t *bytes, int len)
- {
- int i;
- for (i = 0; i < len; i++) {
- MODLOG_DFLT(INFO, "%s0x%02x", i != 0 ? ":" : "", bytes[i]);
- }
- }
- void
- print_addr(const void *addr)
- {
- const uint8_t *u8p;
- u8p = addr;
- MODLOG_DFLT(INFO, "%02x:%02x:%02x:%02x:%02x:%02x",
- u8p[5], u8p[4], u8p[3], u8p[2], u8p[1], u8p[0]);
- }
- /*
- * Enables advertising with parameters:
- * o General discoverable mode
- * o Undirected connectable mode
- */
- static void
- blehr_advertise(void)
- {
- struct ble_gap_adv_params adv_params;
- struct ble_hs_adv_fields fields;
- int rc;
- /*
- * Set the advertisement data included in our advertisements:
- * o Flags (indicates advertisement type and other general info)
- * o Advertising tx power
- * o Device name
- */
- memset(&fields, 0, sizeof(fields));
- /*
- * Advertise two flags:
- * o Discoverability in forthcoming advertisement (general)
- * o BLE-only (BR/EDR unsupported)
- */
- fields.flags = BLE_HS_ADV_F_DISC_GEN |
- BLE_HS_ADV_F_BREDR_UNSUP;
- /*
- * Indicate that the TX power level field should be included; have the
- * stack fill this value automatically. This is done by assigning the
- * special value BLE_HS_ADV_TX_PWR_LVL_AUTO.
- */
- fields.tx_pwr_lvl_is_present = 1;
- fields.tx_pwr_lvl = BLE_HS_ADV_TX_PWR_LVL_AUTO;
- fields.name = (uint8_t *)device_name;
- fields.name_len = strlen(device_name);
- fields.name_is_complete = 1;
- rc = ble_gap_adv_set_fields(&fields);
- if (rc != 0) {
- MODLOG_DFLT(ERROR, "error setting advertisement data; rc=%d\n", rc);
- return;
- }
- /* Begin advertising */
- memset(&adv_params, 0, sizeof(adv_params));
- adv_params.conn_mode = BLE_GAP_CONN_MODE_UND;
- adv_params.disc_mode = BLE_GAP_DISC_MODE_GEN;
- rc = ble_gap_adv_start(blehr_addr_type, NULL, BLE_HS_FOREVER,
- &adv_params, blehr_gap_event, NULL);
- if (rc != 0) {
- MODLOG_DFLT(ERROR, "error enabling advertisement; rc=%d\n", rc);
- return;
- }
- }
- static void
- blehr_tx_hrate_stop(void)
- {
- xTimerStop( blehr_tx_timer, 1000 / portTICK_PERIOD_MS );
- }
- /* Reset heart rate measurement */
- static void
- blehr_tx_hrate_reset(void)
- {
- int rc;
- if (xTimerReset(blehr_tx_timer, 1000 / portTICK_PERIOD_MS ) == pdPASS) {
- rc = 0;
- } else {
- rc = 1;
- }
- assert(rc == 0);
- }
- /* This function simulates heart beat and notifies it to the client */
- static void
- blehr_tx_hrate(TimerHandle_t ev)
- {
- static uint8_t hrm[2];
- int rc;
- struct os_mbuf *om;
- if (!notify_state) {
- blehr_tx_hrate_stop();
- heartrate = 90;
- return;
- }
- hrm[0] = 0x06; /* contact of a sensor */
- hrm[1] = heartrate; /* storing dummy data */
- /* Simulation of heart beats */
- heartrate++;
- if (heartrate == 160) {
- heartrate = 90;
- }
- om = ble_hs_mbuf_from_flat(hrm, sizeof(hrm));
- rc = ble_gatts_notify_custom(conn_handle, hrs_hrm_handle, om);
- assert(rc == 0);
- blehr_tx_hrate_reset();
- }
- static int
- blehr_gap_event(struct ble_gap_event *event, void *arg)
- {
- switch (event->type) {
- case BLE_GAP_EVENT_CONNECT:
- /* A new connection was established or a connection attempt failed */
- MODLOG_DFLT(INFO, "connection %s; status=%d\n",
- event->connect.status == 0 ? "established" : "failed",
- event->connect.status);
- if (event->connect.status != 0) {
- /* Connection failed; resume advertising */
- blehr_advertise();
- }
- conn_handle = event->connect.conn_handle;
- break;
- case BLE_GAP_EVENT_DISCONNECT:
- MODLOG_DFLT(INFO, "disconnect; reason=%d\n", event->disconnect.reason);
- /* Connection terminated; resume advertising */
- blehr_advertise();
- break;
- case BLE_GAP_EVENT_ADV_COMPLETE:
- MODLOG_DFLT(INFO, "adv complete\n");
- blehr_advertise();
- break;
- case BLE_GAP_EVENT_SUBSCRIBE:
- MODLOG_DFLT(INFO, "subscribe event; cur_notify=%d\n value handle; "
- "val_handle=%d\n",
- event->subscribe.cur_notify, hrs_hrm_handle);
- if (event->subscribe.attr_handle == hrs_hrm_handle) {
- notify_state = event->subscribe.cur_notify;
- blehr_tx_hrate_reset();
- } else if (event->subscribe.attr_handle != hrs_hrm_handle) {
- notify_state = event->subscribe.cur_notify;
- blehr_tx_hrate_stop();
- }
- ESP_LOGI("BLE_GAP_SUBSCRIBE_EVENT", "conn_handle from subscribe=%d", conn_handle);
- break;
- case BLE_GAP_EVENT_MTU:
- MODLOG_DFLT(INFO, "mtu update event; conn_handle=%d mtu=%d\n",
- event->mtu.conn_handle,
- event->mtu.value);
- break;
- }
- return 0;
- }
- static void
- blehr_on_sync(void)
- {
- int rc;
- rc = ble_hs_id_infer_auto(0, &blehr_addr_type);
- assert(rc == 0);
- uint8_t addr_val[6] = {0};
- rc = ble_hs_id_copy_addr(blehr_addr_type, addr_val, NULL);
- MODLOG_DFLT(INFO, "Device Address: ");
- print_addr(addr_val);
- MODLOG_DFLT(INFO, "\n");
- /* Begin advertising */
- blehr_advertise();
- }
- static void
- blehr_on_reset(int reason)
- {
- MODLOG_DFLT(ERROR, "Resetting state; reason=%d\n", reason);
- }
- void blehr_host_task(void *param)
- {
- ESP_LOGI(tag, "BLE Host Task Started");
- /* This function will return only when nimble_port_stop() is executed */
- nimble_port_run();
- nimble_port_freertos_deinit();
- }
- void app_main(void)
- {
- int rc;
- /* Initialize NVS — it is used to store PHY calibration data */
- esp_err_t ret = nvs_flash_init();
- if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND) {
- ESP_ERROR_CHECK(nvs_flash_erase());
- ret = nvs_flash_init();
- }
- ESP_ERROR_CHECK(ret);
- ret = nimble_port_init();
- if (ret != ESP_OK) {
- MODLOG_DFLT(ERROR, "Failed to init nimble %d \n", ret);
- return;
- }
- /* Initialize the NimBLE host configuration */
- ble_hs_cfg.sync_cb = blehr_on_sync;
- ble_hs_cfg.reset_cb = blehr_on_reset;
- /* name, period/time, auto reload, timer ID, callback */
- blehr_tx_timer = xTimerCreate("blehr_tx_timer", pdMS_TO_TICKS(1000), pdTRUE, (void *)0, blehr_tx_hrate);
- rc = gatt_svr_init();
- assert(rc == 0);
- /* Set the default device name */
- rc = ble_svc_gap_device_name_set(device_name);
- assert(rc == 0);
- /* Start the task */
- nimble_port_freertos_init(blehr_host_task);
- }
|