/* * 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 #include #include #include "host/ble_hs.h" #include "host/ble_uuid.h" #include "blehr_sens.h" #include "services/dis/ble_svc_dis.h" #include "services/bas/ble_svc_bas.h" /* UUID = 1bce38b3-d137-48ff-a13e-033e14c7a335 */ static const ble_uuid128_t gatt_svr_svc_rw_demo_uuid = BLE_UUID128_INIT(0x35, 0xa3, 0xc7, 0x14, 0x3e, 0x03, 0x3e, 0xa1, 0xff, 0x48, 0x37, 0xd1, 0xb3, 0x38, 0xce, 0x1b); /* UUID = 35f28386-3070-4f3b-ba38-27507e991762 */ static const ble_uuid128_t gatt_svr_chr_rw_demo_write_uuid = BLE_UUID128_INIT(0x62, 0x17, 0x99, 0x7e, 0x50, 0x27, 0x38, 0xba, 0x3b, 0x4f, 0x70, 0x30, 0x86, 0x83, 0xf2, 0x35); /* UUID = ccdd113f-40d5-4d68-86ac-a728dd82f4aa */ static const ble_uuid128_t gatt_svr_chr_rw_demo_readonly_uuid = BLE_UUID128_INIT(0xaa, 0xf4, 0x82, 0xdd, 0x28, 0xa7, 0xac, 0x86, 0x68, 0x4d, 0xd5, 0x40, 0x3f, 0x11, 0xdd, 0xcc); #define REPLY_BUFFER_SIZE 100 static char reply[REPLY_BUFFER_SIZE]; static char rm_demo_write_data[64] = "This characteristic is readable and writeable!"; static int gatt_svr_chr_access_device_info_manufacturer( uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt *ctxt, void *arg); static int gatt_svr_chr_access_device_info_model( uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt *ctxt, void *arg); static int gatt_svr_chr_access_rw_demo( uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt *ctxt, void *arg); /* define several bluetooth services for our device */ static const struct ble_gatt_svc_def gatt_svr_svcs[] = { /* * access_cb defines a callback for read and write access events on * given characteristics */ { /* Service: Device Information */ .type = BLE_GATT_SVC_TYPE_PRIMARY, .uuid = BLE_UUID16_DECLARE(GATT_DEVICE_INFO_UUID), .characteristics = (struct ble_gatt_chr_def[]) { { /* Characteristic: * Manufacturer name */ .uuid = BLE_UUID16_DECLARE(GATT_MANUFACTURER_NAME_UUID), .access_cb = gatt_svr_chr_access_device_info_manufacturer, .flags = BLE_GATT_CHR_F_READ, }, { /* Characteristic: Model number string */ .uuid = BLE_UUID16_DECLARE(GATT_MODEL_NUMBER_UUID), .access_cb = gatt_svr_chr_access_device_info_model, .flags = BLE_GATT_CHR_F_READ, }, { 0, /* No more characteristics in this service */ }, } }, { /* Service: Read/Write Demo */ .type = BLE_GATT_SVC_TYPE_PRIMARY, .uuid = (ble_uuid_t*) &gatt_svr_svc_rw_demo_uuid.u, .characteristics = (struct ble_gatt_chr_def[]) { { /* Characteristic: Read/Write Demo write */ .uuid = (ble_uuid_t*) &gatt_svr_chr_rw_demo_write_uuid.u, .access_cb = gatt_svr_chr_access_rw_demo, .flags = BLE_GATT_CHR_F_READ | BLE_GATT_CHR_F_WRITE, }, { /* Characteristic: Read/Write Demo read only */ .uuid = (ble_uuid_t*) &gatt_svr_chr_rw_demo_readonly_uuid.u, .access_cb = gatt_svr_chr_access_rw_demo, .flags = BLE_GATT_CHR_F_READ, }, { 0, /* No more characteristics in this service */ }, } }, { 0, /* No more services */ }, }; static int gatt_svr_chr_access_device_info_manufacturer( uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt *ctxt, void *arg) { printf("service 'device info: manufacturer' callback triggered\n"); snprintf(reply, REPLY_BUFFER_SIZE, "This is TencentOS tiny\n"); printf("reply: %s\n", reply); int rc = os_mbuf_append(ctxt->om, reply, strlen(reply)); printf("\n"); return rc; } static int gatt_svr_chr_access_device_info_model( uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt *ctxt, void *arg) { printf("service 'device info: model' callback triggered\n"); snprintf(reply, REPLY_BUFFER_SIZE, "running TencentOS tiny on nordic ble board"); printf("reply: %s\n", reply); int rc = os_mbuf_append(ctxt->om, reply, strlen(reply)); printf("\n"); return rc; } static int gatt_svr_chr_access_rw_demo( uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt *ctxt, void *arg) { printf("service 'rw demo' callback triggered\n"); int rc = 0; ble_uuid_t* write_uuid = (ble_uuid_t*) &gatt_svr_chr_rw_demo_write_uuid.u; ble_uuid_t* readonly_uuid = (ble_uuid_t*) &gatt_svr_chr_rw_demo_readonly_uuid.u; if (ble_uuid_cmp(ctxt->chr->uuid, write_uuid) == 0) { printf("access to characteristic 'rw demo (write)'\n"); switch (ctxt->op) { case BLE_GATT_ACCESS_OP_READ_CHR: printf("read from characteristic\n"); printf("current value of rm_demo_write_data: '%s'\n", rm_demo_write_data); /* send given data to the client */ rc = os_mbuf_append(ctxt->om, &rm_demo_write_data, strlen(rm_demo_write_data)); break; case BLE_GATT_ACCESS_OP_WRITE_CHR: printf("write to characteristic\n"); printf("old value of rm_demo_write_data: '%s'\n", rm_demo_write_data); uint16_t om_len; om_len = OS_MBUF_PKTLEN(ctxt->om); /* read sent data */ rc = ble_hs_mbuf_to_flat(ctxt->om, &rm_demo_write_data, sizeof rm_demo_write_data, &om_len); /* we need to null-terminate the received string */ rm_demo_write_data[om_len] = '\0'; printf("new value of rm_demo_write_data: '%s'\n", rm_demo_write_data); break; case BLE_GATT_ACCESS_OP_READ_DSC: printf("read from descriptor\n"); break; case BLE_GATT_ACCESS_OP_WRITE_DSC: printf("write to descriptor\n"); break; default: printf("unhandled operation!\n"); rc = 1; break; } printf("\n"); return rc; } else if (ble_uuid_cmp(ctxt->chr->uuid, readonly_uuid) == 0) { printf("access to characteristic 'rw demo (read-only)'\n"); if (ctxt->op == BLE_GATT_ACCESS_OP_READ_CHR) { char random_digit; /* get random char between '0' and '9' */ random_digit = 48 + (rand() % 10); snprintf(reply, REPLY_BUFFER_SIZE, "new random number: %c", random_digit); printf("reply: %s\n", reply); rc = os_mbuf_append(ctxt->om, &reply, strlen(reply)); printf("\n"); return rc; } return 0; } printf("unhandled uuid!\n"); return 1; } void gatt_svr_register_cb(struct ble_gatt_register_ctxt *ctxt, void *arg) { char buf[BLE_UUID_STR_LEN]; switch (ctxt->op) { case BLE_GATT_REGISTER_OP_SVC: MODLOG_DFLT(DEBUG, "registered service %s with handle=%d\n", ble_uuid_to_str(ctxt->svc.svc_def->uuid, buf), ctxt->svc.handle); break; case BLE_GATT_REGISTER_OP_CHR: MODLOG_DFLT(DEBUG, "registering characteristic %s with " "def_handle=%d val_handle=%d\n", ble_uuid_to_str(ctxt->chr.chr_def->uuid, buf), ctxt->chr.def_handle, ctxt->chr.val_handle); break; case BLE_GATT_REGISTER_OP_DSC: MODLOG_DFLT(DEBUG, "registering descriptor %s with handle=%d\n", ble_uuid_to_str(ctxt->dsc.dsc_def->uuid, buf), ctxt->dsc.handle); break; default: assert(0); break; } } int gatt_svr_init(void) { int rc; rc = ble_gatts_count_cfg(gatt_svr_svcs); if (rc != 0) { return rc; } rc = ble_gatts_add_svcs(gatt_svr_svcs); if (rc != 0) { return rc; } return 0; }