bt_gatt_service_register not working

I tested out bt_gatt_service_register using the samples/bluetooth/peripheral sample by adding

CONFIG_BT_GATT_DYNAMIC_DB=y

to the proj.conf and the following to the start of main:
static struct bt_uuid_128 uuid;
uuid = (struct bt_uuid_128)BT_UUID_INIT_128(0x53, 0xF5, 0x3C, 0x12, 0xF0, 0xD6, 0x25, 0xAD, 0xD6, 0x4E, 0xA6, 0x48, 0x01, 0x00, 0x54, 0xE7);
static struct bt_gatt_attr attr;
attr = (struct bt_gatt_attr)BT_GATT_PRIMARY_SERVICE(&uuid);
static struct bt_gatt_service service;
service = (struct bt_gatt_service) { .attrs = &attr, .attr_count = 1 };
int result = bt_gatt_service_register(&service);
if (result != 0) {
printk("Bluetooth dynamic service registration failed (result %d)\n", result);
return 0;
}
This works and I can see the service on an iOS device.
However, if I do the same with my own products application then the service can NOT be seen on an iOS device.  The bt_gatt_service_register does return success.  I can see changes to statically defined services.
I have copied all my proj.conf options into the samples/bluetooth/peripheral sample to see if maybe one the settings was causing the issue, but it still works.
Anyone else run into this?  Or have any ideas for how to debug and figure out why the bt_gatt_service_register service does not seem to be working in my application?
Parents
  • Hi,


    I would recommend that you compare the two .config files, this way you will be able to see what is different in the projects and assess what could be the problem.  
    Also might be of value to compare any log files also. 

    Regards,
    Jonathan 

  • I've already compared config and copied my config to the sample as indicated in my original message.  I've looked at the log and did not see any relevant BLE errors logged.  The bt_gatt_service_register function returns success.  However, that gatt service is not found when connecting via BLE.

    Any tips on where to set break points in the Zephyr BLE code to see why service discovery is not including the dynamic services?

  • Did you compare the output .config file? So the files that are generated when building the project. 

    Regards,
    Jonathan

  • I have the same problem.

    Did you find a solution?

    Br Mads

  • Pleas create a new ticket and link to this ticket if you are not able to solve the issue. We will do our best to help solve this. 

    Regards,
    Jonathan

  • I have created ticket "Define a service dynamically" with Case ID: 321172.

    Could you help me solving the issue?

    An examples of how to define a GATT service dynamically could help me.

  • This is an example of what is working for me:

    typedef struct {
        struct bt_gatt_attr attrs[6];
        struct bt_gatt_service service;
    } pt_ble_nus_t;
    
    pt_ble_nus_t pt_ble_nus;
    
    void pt_ble_nus_tx_ccc_cfg_changed(const struct bt_gatt_attr *attr fd_unused, uint16_t value) {   
        bool notify = (value == BT_GATT_CCC_NOTIFY);
        ...
    }
    
    ssize_t pt_ble_nus_rx_write(
        struct bt_conn *conn fd_unused, const struct bt_gatt_attr *attr fd_unused, const void *data, uint16_t length, uint16_t offset fd_unused, uint8_t flags fd_unused
    ) {
        ...
        return length;
    }
    
    void pt_ble_nus_attr_tx(void) {
        static struct bt_uuid_128 uuid = {
            .uuid = { BT_UUID_TYPE_128 },
            .val = { PT_BLE_NUS_UUID_TX_VAL },
        };
        static struct bt_uuid_16 gatt_chrc_uuid = { .uuid = { BT_UUID_TYPE_16 }, .val = BT_UUID_GATT_CHRC_VAL };
        static struct bt_gatt_chrc chrc_user_data = {
            .uuid = (struct bt_uuid *)&uuid,
            .value_handle = 0U,
            .properties = BT_GATT_CHRC_NOTIFY,
        };
        pt_ble_nus.attrs[1] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&gatt_chrc_uuid,
            .read = bt_gatt_attr_read_chrc,
            .write = (void *)0,
            .user_data = (void *)&chrc_user_data,
            .handle = 0,
            .perm = BT_GATT_PERM_READ,
        };
    
        pt_ble_nus.attrs[2] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&uuid,
            .read = (void *)0,
            .write = (void *)0,
            .user_data = (void *)0,
            .handle = 0,
            .perm = BT_GATT_PERM_READ,
        };
    
        static struct bt_uuid_16 gatt_ccc_uuid = { .uuid = { BT_UUID_TYPE_16 }, .val = BT_UUID_GATT_CCC_VAL };
        static struct _bt_gatt_ccc ccc = {
            .cfg = {},
            .cfg_changed = pt_ble_nus_tx_ccc_cfg_changed,
            .cfg_write = (void *)0,
            .cfg_match = (void *)0,
        };
        pt_ble_nus.attrs[3] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&gatt_ccc_uuid,
            .read = bt_gatt_attr_read_ccc,
            .write = bt_gatt_attr_write_ccc,
            .user_data = (void *)&ccc,
            .handle = 0,
            .perm = BT_GATT_PERM_READ | BT_GATT_PERM_WRITE,
        };
    }
    
    void pt_ble_nus_attr_rx(void) {
        static struct bt_uuid_128 uuid = {
            .uuid = { BT_UUID_TYPE_128 },
            .val = { PT_BLE_NUS_UUID_RX_VAL },
        };
        static struct bt_uuid_16 gatt_chrc_uuid = { .uuid = { BT_UUID_TYPE_16 }, .val = BT_UUID_GATT_CHRC_VAL };
        static struct bt_gatt_chrc chrc_user_data = {
            .uuid = (struct bt_uuid *)&uuid,
            .value_handle = 0U,
            .properties = BT_GATT_CHRC_WRITE | BT_GATT_CHRC_WRITE_WITHOUT_RESP,
        };
        pt_ble_nus.attrs[4] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&gatt_chrc_uuid,
            .read = bt_gatt_attr_read_chrc,
            .write = (void *)0,
            .user_data = (void *)&chrc_user_data,
            .handle = 0,
            .perm = BT_GATT_PERM_READ,
        };
        pt_ble_nus.attrs[5] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&uuid,
            .read = (void *)0,
            .write = pt_ble_nus_rx_write,
            .user_data = (void *)0,
            .handle = 0,
            .perm = BT_GATT_PERM_READ | BT_GATT_PERM_WRITE,
        };
    }
    
    void pt_ble_nus_register(void) {
        static struct bt_uuid_16 gatt_primary_uuid = { .uuid = { BT_UUID_TYPE_16 }, .val = BT_UUID_GATT_PRIMARY_VAL };
        static struct bt_uuid_128 service_uuid = { .uuid = { BT_UUID_TYPE_128 }, .val = { PT_BLE_NUS_UUID_SERVICE_VAL } };
        pt_ble_nus.attrs[0] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&gatt_primary_uuid,
            .read = bt_gatt_attr_read_service,
            .write = (void *)0,
            .user_data = (void *)&service_uuid,
            .handle = 0,
            .perm = BT_GATT_PERM_READ,
        };
    
        pt_ble_nus_attr_tx();
        pt_ble_nus_attr_rx();
    
        pt_ble_nus.service = (struct bt_gatt_service) { .attrs = pt_ble_nus.attrs, .attr_count = ARRAY_SIZE(pt_ble_nus.attrs) };
    	int result = bt_gatt_service_register(&pt_ble_nus.service);
        fd_assert(result >= 0);
    }
    

Reply
  • This is an example of what is working for me:

    typedef struct {
        struct bt_gatt_attr attrs[6];
        struct bt_gatt_service service;
    } pt_ble_nus_t;
    
    pt_ble_nus_t pt_ble_nus;
    
    void pt_ble_nus_tx_ccc_cfg_changed(const struct bt_gatt_attr *attr fd_unused, uint16_t value) {   
        bool notify = (value == BT_GATT_CCC_NOTIFY);
        ...
    }
    
    ssize_t pt_ble_nus_rx_write(
        struct bt_conn *conn fd_unused, const struct bt_gatt_attr *attr fd_unused, const void *data, uint16_t length, uint16_t offset fd_unused, uint8_t flags fd_unused
    ) {
        ...
        return length;
    }
    
    void pt_ble_nus_attr_tx(void) {
        static struct bt_uuid_128 uuid = {
            .uuid = { BT_UUID_TYPE_128 },
            .val = { PT_BLE_NUS_UUID_TX_VAL },
        };
        static struct bt_uuid_16 gatt_chrc_uuid = { .uuid = { BT_UUID_TYPE_16 }, .val = BT_UUID_GATT_CHRC_VAL };
        static struct bt_gatt_chrc chrc_user_data = {
            .uuid = (struct bt_uuid *)&uuid,
            .value_handle = 0U,
            .properties = BT_GATT_CHRC_NOTIFY,
        };
        pt_ble_nus.attrs[1] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&gatt_chrc_uuid,
            .read = bt_gatt_attr_read_chrc,
            .write = (void *)0,
            .user_data = (void *)&chrc_user_data,
            .handle = 0,
            .perm = BT_GATT_PERM_READ,
        };
    
        pt_ble_nus.attrs[2] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&uuid,
            .read = (void *)0,
            .write = (void *)0,
            .user_data = (void *)0,
            .handle = 0,
            .perm = BT_GATT_PERM_READ,
        };
    
        static struct bt_uuid_16 gatt_ccc_uuid = { .uuid = { BT_UUID_TYPE_16 }, .val = BT_UUID_GATT_CCC_VAL };
        static struct _bt_gatt_ccc ccc = {
            .cfg = {},
            .cfg_changed = pt_ble_nus_tx_ccc_cfg_changed,
            .cfg_write = (void *)0,
            .cfg_match = (void *)0,
        };
        pt_ble_nus.attrs[3] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&gatt_ccc_uuid,
            .read = bt_gatt_attr_read_ccc,
            .write = bt_gatt_attr_write_ccc,
            .user_data = (void *)&ccc,
            .handle = 0,
            .perm = BT_GATT_PERM_READ | BT_GATT_PERM_WRITE,
        };
    }
    
    void pt_ble_nus_attr_rx(void) {
        static struct bt_uuid_128 uuid = {
            .uuid = { BT_UUID_TYPE_128 },
            .val = { PT_BLE_NUS_UUID_RX_VAL },
        };
        static struct bt_uuid_16 gatt_chrc_uuid = { .uuid = { BT_UUID_TYPE_16 }, .val = BT_UUID_GATT_CHRC_VAL };
        static struct bt_gatt_chrc chrc_user_data = {
            .uuid = (struct bt_uuid *)&uuid,
            .value_handle = 0U,
            .properties = BT_GATT_CHRC_WRITE | BT_GATT_CHRC_WRITE_WITHOUT_RESP,
        };
        pt_ble_nus.attrs[4] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&gatt_chrc_uuid,
            .read = bt_gatt_attr_read_chrc,
            .write = (void *)0,
            .user_data = (void *)&chrc_user_data,
            .handle = 0,
            .perm = BT_GATT_PERM_READ,
        };
        pt_ble_nus.attrs[5] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&uuid,
            .read = (void *)0,
            .write = pt_ble_nus_rx_write,
            .user_data = (void *)0,
            .handle = 0,
            .perm = BT_GATT_PERM_READ | BT_GATT_PERM_WRITE,
        };
    }
    
    void pt_ble_nus_register(void) {
        static struct bt_uuid_16 gatt_primary_uuid = { .uuid = { BT_UUID_TYPE_16 }, .val = BT_UUID_GATT_PRIMARY_VAL };
        static struct bt_uuid_128 service_uuid = { .uuid = { BT_UUID_TYPE_128 }, .val = { PT_BLE_NUS_UUID_SERVICE_VAL } };
        pt_ble_nus.attrs[0] = (struct bt_gatt_attr) {
            .uuid = (struct bt_uuid *)&gatt_primary_uuid,
            .read = bt_gatt_attr_read_service,
            .write = (void *)0,
            .user_data = (void *)&service_uuid,
            .handle = 0,
            .perm = BT_GATT_PERM_READ,
        };
    
        pt_ble_nus_attr_tx();
        pt_ble_nus_attr_rx();
    
        pt_ble_nus.service = (struct bt_gatt_service) { .attrs = pt_ble_nus.attrs, .attr_count = ARRAY_SIZE(pt_ble_nus.attrs) };
    	int result = bt_gatt_service_register(&pt_ble_nus.service);
        fd_assert(result >= 0);
    }
    

Children
Related