| /* |
| * |
| * BlueZ - Bluetooth protocol stack for Linux |
| * |
| * Copyright (C) 2015 Google Inc. |
| * |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| #include <config.h> |
| #endif |
| |
| #include <stdint.h> |
| #include <stdlib.h> |
| #include <errno.h> |
| |
| #include "lib/bluetooth.h" |
| #include "lib/sdp.h" |
| #include "lib/sdp_lib.h" |
| #include "lib/uuid.h" |
| #include "btio/btio.h" |
| #include "gdbus/gdbus.h" |
| #include "src/shared/util.h" |
| #include "src/shared/queue.h" |
| #include "src/shared/att.h" |
| #include "src/shared/gatt-db.h" |
| #include "src/shared/gatt-server.h" |
| #include "log.h" |
| #include "error.h" |
| #include "adapter.h" |
| #include "device.h" |
| #include "gatt-database.h" |
| #include "dbus-common.h" |
| #include "profile.h" |
| |
| #ifndef ATT_CID |
| #define ATT_CID 4 |
| #endif |
| |
| #ifndef ATT_PSM |
| #define ATT_PSM 31 |
| #endif |
| |
| #define GATT_MANAGER_IFACE "org.bluez.GattManager1" |
| #define GATT_SERVICE_IFACE "org.bluez.GattService1" |
| #define GATT_CHRC_IFACE "org.bluez.GattCharacteristic1" |
| #define GATT_DESC_IFACE "org.bluez.GattDescriptor1" |
| |
| #define UUID_GAP 0x1800 |
| #define UUID_GATT 0x1801 |
| |
| #ifndef MIN |
| #define MIN(a, b) ((a) < (b) ? (a) : (b)) |
| #endif |
| |
| struct btd_gatt_database { |
| struct btd_adapter *adapter; |
| struct gatt_db *db; |
| unsigned int db_id; |
| GIOChannel *le_io; |
| GIOChannel *l2cap_io; |
| uint32_t gap_handle; |
| uint32_t gatt_handle; |
| struct queue *device_states; |
| struct queue *ccc_callbacks; |
| struct gatt_db_attribute *svc_chngd; |
| struct gatt_db_attribute *svc_chngd_ccc; |
| struct queue *services; |
| struct queue *profiles; |
| }; |
| |
| struct external_service { |
| struct btd_gatt_database *database; |
| bool failed; |
| char *owner; |
| char *path; /* Path to GattService1 */ |
| DBusMessage *reg; |
| GDBusClient *client; |
| GDBusProxy *proxy; |
| struct gatt_db_attribute *attrib; |
| uint16_t attr_cnt; |
| struct queue *chrcs; |
| struct queue *descs; |
| }; |
| |
| struct external_profile { |
| struct btd_gatt_database *database; |
| char *owner; |
| char *path; /* Path to GattProfile1 */ |
| unsigned int id; |
| struct queue *profiles; /* btd_profile list */ |
| }; |
| |
| struct external_chrc { |
| struct external_service *service; |
| char *path; |
| GDBusProxy *proxy; |
| uint8_t props; |
| uint8_t ext_props; |
| struct gatt_db_attribute *attrib; |
| struct gatt_db_attribute *ccc; |
| struct queue *pending_reads; |
| struct queue *pending_writes; |
| unsigned int ntfy_cnt; |
| }; |
| |
| struct external_desc { |
| struct external_service *service; |
| char *chrc_path; |
| GDBusProxy *proxy; |
| uint32_t perm; |
| struct gatt_db_attribute *attrib; |
| bool handled; |
| struct queue *pending_reads; |
| struct queue *pending_writes; |
| }; |
| |
| struct pending_op { |
| unsigned int id; |
| struct gatt_db_attribute *attrib; |
| struct queue *owner_queue; |
| struct iovec data; |
| }; |
| |
| struct device_state { |
| bdaddr_t bdaddr; |
| uint8_t bdaddr_type; |
| struct queue *ccc_states; |
| }; |
| |
| struct ccc_state { |
| uint16_t handle; |
| uint8_t value[2]; |
| }; |
| |
| struct ccc_cb_data { |
| uint16_t handle; |
| btd_gatt_database_ccc_write_t callback; |
| btd_gatt_database_destroy_t destroy; |
| void *user_data; |
| }; |
| |
| struct device_info { |
| bdaddr_t bdaddr; |
| uint8_t bdaddr_type; |
| }; |
| |
| static void ccc_cb_free(void *data) |
| { |
| struct ccc_cb_data *ccc_cb = data; |
| |
| if (ccc_cb->destroy) |
| ccc_cb->destroy(ccc_cb->user_data); |
| |
| free(ccc_cb); |
| } |
| |
| static bool ccc_cb_match_service(const void *data, const void *match_data) |
| { |
| const struct ccc_cb_data *ccc_cb = data; |
| const struct gatt_db_attribute *attrib = match_data; |
| uint16_t start, end; |
| |
| if (!gatt_db_attribute_get_service_handles(attrib, &start, &end)) |
| return false; |
| |
| return ccc_cb->handle >= start && ccc_cb->handle <= end; |
| } |
| |
| static bool ccc_cb_match_handle(const void *data, const void *match_data) |
| { |
| const struct ccc_cb_data *ccc_cb = data; |
| uint16_t handle = PTR_TO_UINT(match_data); |
| |
| return ccc_cb->handle == handle; |
| } |
| |
| static bool dev_state_match(const void *a, const void *b) |
| { |
| const struct device_state *dev_state = a; |
| const struct device_info *dev_info = b; |
| |
| return bacmp(&dev_state->bdaddr, &dev_info->bdaddr) == 0 && |
| dev_state->bdaddr_type == dev_info->bdaddr_type; |
| } |
| |
| static struct device_state * |
| find_device_state(struct btd_gatt_database *database, bdaddr_t *bdaddr, |
| uint8_t bdaddr_type) |
| { |
| struct device_info dev_info; |
| |
| memset(&dev_info, 0, sizeof(dev_info)); |
| |
| bacpy(&dev_info.bdaddr, bdaddr); |
| dev_info.bdaddr_type = bdaddr_type; |
| |
| return queue_find(database->device_states, dev_state_match, &dev_info); |
| } |
| |
| static bool ccc_state_match(const void *a, const void *b) |
| { |
| const struct ccc_state *ccc = a; |
| uint16_t handle = PTR_TO_UINT(b); |
| |
| return ccc->handle == handle; |
| } |
| |
| static struct ccc_state *find_ccc_state(struct device_state *dev_state, |
| uint16_t handle) |
| { |
| return queue_find(dev_state->ccc_states, ccc_state_match, |
| UINT_TO_PTR(handle)); |
| } |
| |
| static struct device_state *device_state_create(bdaddr_t *bdaddr, |
| uint8_t bdaddr_type) |
| { |
| struct device_state *dev_state; |
| |
| dev_state = new0(struct device_state, 1); |
| if (!dev_state) |
| return NULL; |
| |
| dev_state->ccc_states = queue_new(); |
| if (!dev_state->ccc_states) { |
| free(dev_state); |
| return NULL; |
| } |
| |
| bacpy(&dev_state->bdaddr, bdaddr); |
| dev_state->bdaddr_type = bdaddr_type; |
| |
| return dev_state; |
| } |
| |
| static struct device_state *get_device_state(struct btd_gatt_database *database, |
| bdaddr_t *bdaddr, |
| uint8_t bdaddr_type) |
| { |
| struct device_state *dev_state; |
| |
| /* |
| * Find and return a device state. If a matching state doesn't exist, |
| * then create a new one. |
| */ |
| dev_state = find_device_state(database, bdaddr, bdaddr_type); |
| if (dev_state) |
| return dev_state; |
| |
| dev_state = device_state_create(bdaddr, bdaddr_type); |
| if (!dev_state) |
| return NULL; |
| |
| queue_push_tail(database->device_states, dev_state); |
| |
| return dev_state; |
| } |
| |
| static struct ccc_state *get_ccc_state(struct btd_gatt_database *database, |
| bdaddr_t *bdaddr, |
| uint8_t bdaddr_type, |
| uint16_t handle) |
| { |
| struct device_state *dev_state; |
| struct ccc_state *ccc; |
| |
| dev_state = get_device_state(database, bdaddr, bdaddr_type); |
| if (!dev_state) |
| return NULL; |
| |
| ccc = find_ccc_state(dev_state, handle); |
| if (ccc) |
| return ccc; |
| |
| ccc = new0(struct ccc_state, 1); |
| if (!ccc) |
| return NULL; |
| |
| ccc->handle = handle; |
| queue_push_tail(dev_state->ccc_states, ccc); |
| |
| return ccc; |
| } |
| |
| static void device_state_free(void *data) |
| { |
| struct device_state *state = data; |
| |
| queue_destroy(state->ccc_states, free); |
| free(state); |
| } |
| |
| static void cancel_pending_read(void *data) |
| { |
| struct pending_op *op = data; |
| |
| gatt_db_attribute_read_result(op->attrib, op->id, |
| BT_ATT_ERROR_REQUEST_NOT_SUPPORTED, |
| NULL, 0); |
| op->owner_queue = NULL; |
| } |
| |
| static void cancel_pending_write(void *data) |
| { |
| struct pending_op *op = data; |
| |
| gatt_db_attribute_write_result(op->attrib, op->id, |
| BT_ATT_ERROR_REQUEST_NOT_SUPPORTED); |
| op->owner_queue = NULL; |
| } |
| |
| static void chrc_free(void *data) |
| { |
| struct external_chrc *chrc = data; |
| |
| queue_destroy(chrc->pending_reads, cancel_pending_read); |
| queue_destroy(chrc->pending_writes, cancel_pending_write); |
| |
| g_free(chrc->path); |
| |
| g_dbus_proxy_set_property_watch(chrc->proxy, NULL, NULL); |
| g_dbus_proxy_unref(chrc->proxy); |
| |
| free(chrc); |
| } |
| |
| static void desc_free(void *data) |
| { |
| struct external_desc *desc = data; |
| |
| queue_destroy(desc->pending_reads, cancel_pending_read); |
| queue_destroy(desc->pending_writes, cancel_pending_write); |
| |
| g_dbus_proxy_unref(desc->proxy); |
| g_free(desc->chrc_path); |
| |
| free(desc); |
| } |
| |
| static void service_free(void *data) |
| { |
| struct external_service *service = data; |
| |
| queue_destroy(service->chrcs, chrc_free); |
| queue_destroy(service->descs, desc_free); |
| |
| gatt_db_remove_service(service->database->db, service->attrib); |
| |
| if (service->client) { |
| g_dbus_client_set_disconnect_watch(service->client, NULL, NULL); |
| g_dbus_client_set_proxy_handlers(service->client, NULL, NULL, |
| NULL, NULL); |
| g_dbus_client_set_ready_watch(service->client, NULL, NULL); |
| g_dbus_proxy_unref(service->proxy); |
| g_dbus_client_unref(service->client); |
| } |
| |
| if (service->reg) |
| dbus_message_unref(service->reg); |
| |
| g_free(service->owner); |
| g_free(service->path); |
| |
| free(service); |
| } |
| |
| static void profile_remove(void *data) |
| { |
| struct btd_profile *p = data; |
| |
| DBG("Removed \"%s\"", p->name); |
| |
| adapter_foreach(adapter_remove_profile, p); |
| |
| g_free((void *) p->name); |
| g_free((void *) p->remote_uuid); |
| |
| free(p); |
| } |
| |
| static void profile_release(struct external_profile *profile) |
| { |
| DBusMessage *msg; |
| |
| if (!profile->id) |
| return; |
| |
| DBG("Releasing \"%s\"", profile->owner); |
| |
| g_dbus_remove_watch(btd_get_dbus_connection(), profile->id); |
| |
| msg = dbus_message_new_method_call(profile->owner, profile->path, |
| "org.bluez.GattProfile1", |
| "Release"); |
| if (msg) |
| g_dbus_send_message(btd_get_dbus_connection(), msg); |
| } |
| |
| static void profile_free(void *data) |
| { |
| struct external_profile *profile = data; |
| |
| queue_destroy(profile->profiles, profile_remove); |
| |
| profile_release(profile); |
| |
| g_free(profile->owner); |
| g_free(profile->path); |
| |
| free(profile); |
| } |
| |
| static void gatt_database_free(void *data) |
| { |
| struct btd_gatt_database *database = data; |
| |
| if (database->le_io) { |
| g_io_channel_shutdown(database->le_io, FALSE, NULL); |
| g_io_channel_unref(database->le_io); |
| } |
| |
| if (database->l2cap_io) { |
| g_io_channel_shutdown(database->l2cap_io, FALSE, NULL); |
| g_io_channel_unref(database->l2cap_io); |
| } |
| |
| if (database->gatt_handle) |
| adapter_service_remove(database->adapter, |
| database->gatt_handle); |
| |
| if (database->gap_handle) |
| adapter_service_remove(database->adapter, database->gap_handle); |
| |
| /* TODO: Persistently store CCC states before freeing them */ |
| gatt_db_unregister(database->db, database->db_id); |
| |
| queue_destroy(database->device_states, device_state_free); |
| queue_destroy(database->services, service_free); |
| queue_destroy(database->profiles, profile_free); |
| queue_destroy(database->ccc_callbacks, ccc_cb_free); |
| database->device_states = NULL; |
| database->ccc_callbacks = NULL; |
| |
| gatt_db_unref(database->db); |
| |
| btd_adapter_unref(database->adapter); |
| free(database); |
| } |
| |
| static void connect_cb(GIOChannel *io, GError *gerr, gpointer user_data) |
| { |
| struct btd_adapter *adapter; |
| struct btd_device *device; |
| uint8_t dst_type; |
| bdaddr_t src, dst; |
| |
| DBG("New incoming LE ATT connection"); |
| |
| if (gerr) { |
| error("%s", gerr->message); |
| return; |
| } |
| |
| bt_io_get(io, &gerr, BT_IO_OPT_SOURCE_BDADDR, &src, |
| BT_IO_OPT_DEST_BDADDR, &dst, |
| BT_IO_OPT_DEST_TYPE, &dst_type, |
| BT_IO_OPT_INVALID); |
| if (gerr) { |
| error("bt_io_get: %s", gerr->message); |
| g_error_free(gerr); |
| return; |
| } |
| |
| adapter = adapter_find(&src); |
| if (!adapter) |
| return; |
| |
| device = btd_adapter_get_device(adapter, &dst, dst_type); |
| if (!device) |
| return; |
| |
| device_attach_att(device, io); |
| } |
| |
| static void gap_device_name_read_cb(struct gatt_db_attribute *attrib, |
| unsigned int id, uint16_t offset, |
| uint8_t opcode, struct bt_att *att, |
| void *user_data) |
| { |
| struct btd_gatt_database *database = user_data; |
| uint8_t error = 0; |
| size_t len = 0; |
| const uint8_t *value = NULL; |
| const char *device_name; |
| |
| DBG("GAP Device Name read request\n"); |
| |
| device_name = btd_adapter_get_name(database->adapter); |
| len = strlen(device_name); |
| |
| if (offset > len) { |
| error = BT_ATT_ERROR_INVALID_OFFSET; |
| goto done; |
| } |
| |
| len -= offset; |
| value = len ? (const uint8_t *) &device_name[offset] : NULL; |
| |
| done: |
| gatt_db_attribute_read_result(attrib, id, error, value, len); |
| } |
| |
| static void gap_appearance_read_cb(struct gatt_db_attribute *attrib, |
| unsigned int id, uint16_t offset, |
| uint8_t opcode, struct bt_att *att, |
| void *user_data) |
| { |
| struct btd_gatt_database *database = user_data; |
| uint8_t error = 0; |
| size_t len = 2; |
| const uint8_t *value = NULL; |
| uint8_t appearance[2]; |
| uint32_t dev_class; |
| |
| DBG("GAP Appearance read request\n"); |
| |
| dev_class = btd_adapter_get_class(database->adapter); |
| |
| if (offset > 2) { |
| error = BT_ATT_ERROR_INVALID_OFFSET; |
| goto done; |
| } |
| |
| appearance[0] = dev_class & 0x00ff; |
| appearance[1] = (dev_class >> 8) & 0x001f; |
| |
| len -= offset; |
| value = len ? &appearance[offset] : NULL; |
| |
| done: |
| gatt_db_attribute_read_result(attrib, id, error, value, len); |
| } |
| |
| static sdp_record_t *record_new(uuid_t *uuid, uint16_t start, uint16_t end) |
| { |
| sdp_list_t *svclass_id, *apseq, *proto[2], *root, *aproto; |
| uuid_t root_uuid, proto_uuid, l2cap; |
| sdp_record_t *record; |
| sdp_data_t *psm, *sh, *eh; |
| uint16_t lp = ATT_PSM; |
| |
| if (uuid == NULL) |
| return NULL; |
| |
| if (start > end) |
| return NULL; |
| |
| record = sdp_record_alloc(); |
| if (record == NULL) |
| return NULL; |
| |
| sdp_uuid16_create(&root_uuid, PUBLIC_BROWSE_GROUP); |
| root = sdp_list_append(NULL, &root_uuid); |
| sdp_set_browse_groups(record, root); |
| sdp_list_free(root, NULL); |
| |
| svclass_id = sdp_list_append(NULL, uuid); |
| sdp_set_service_classes(record, svclass_id); |
| sdp_list_free(svclass_id, NULL); |
| |
| sdp_uuid16_create(&l2cap, L2CAP_UUID); |
| proto[0] = sdp_list_append(NULL, &l2cap); |
| psm = sdp_data_alloc(SDP_UINT16, &lp); |
| proto[0] = sdp_list_append(proto[0], psm); |
| apseq = sdp_list_append(NULL, proto[0]); |
| |
| sdp_uuid16_create(&proto_uuid, ATT_UUID); |
| proto[1] = sdp_list_append(NULL, &proto_uuid); |
| sh = sdp_data_alloc(SDP_UINT16, &start); |
| proto[1] = sdp_list_append(proto[1], sh); |
| eh = sdp_data_alloc(SDP_UINT16, &end); |
| proto[1] = sdp_list_append(proto[1], eh); |
| apseq = sdp_list_append(apseq, proto[1]); |
| |
| aproto = sdp_list_append(NULL, apseq); |
| sdp_set_access_protos(record, aproto); |
| |
| sdp_data_free(psm); |
| sdp_data_free(sh); |
| sdp_data_free(eh); |
| sdp_list_free(proto[0], NULL); |
| sdp_list_free(proto[1], NULL); |
| sdp_list_free(apseq, NULL); |
| sdp_list_free(aproto, NULL); |
| |
| return record; |
| } |
| |
| static uint32_t database_add_record(struct btd_gatt_database *database, |
| uint16_t uuid, |
| struct gatt_db_attribute *attr, |
| const char *name) |
| { |
| sdp_record_t *record; |
| uint16_t start, end; |
| uuid_t svc, gap_uuid; |
| |
| sdp_uuid16_create(&svc, uuid); |
| gatt_db_attribute_get_service_handles(attr, &start, &end); |
| |
| record = record_new(&svc, start, end); |
| if (!record) |
| return 0; |
| |
| if (name != NULL) |
| sdp_set_info_attr(record, name, "BlueZ", NULL); |
| |
| sdp_uuid16_create(&gap_uuid, UUID_GAP); |
| if (sdp_uuid_cmp(&svc, &gap_uuid) == 0) { |
| sdp_set_url_attr(record, "http://www.bluez.org/", |
| "http://www.bluez.org/", |
| "http://www.bluez.org/"); |
| } |
| |
| if (adapter_service_add(database->adapter, record) == 0) |
| return record->handle; |
| |
| sdp_record_free(record); |
| return 0; |
| } |
| |
| static void populate_gap_service(struct btd_gatt_database *database) |
| { |
| bt_uuid_t uuid; |
| struct gatt_db_attribute *service; |
| |
| /* Add the GAP service */ |
| bt_uuid16_create(&uuid, UUID_GAP); |
| service = gatt_db_add_service(database->db, &uuid, true, 5); |
| database->gap_handle = database_add_record(database, UUID_GAP, service, |
| "Generic Access Profile"); |
| |
| /* |
| * Device Name characteristic. |
| */ |
| bt_uuid16_create(&uuid, GATT_CHARAC_DEVICE_NAME); |
| gatt_db_service_add_characteristic(service, &uuid, BT_ATT_PERM_READ, |
| BT_GATT_CHRC_PROP_READ, |
| gap_device_name_read_cb, |
| NULL, database); |
| |
| /* |
| * Device Appearance characteristic. |
| */ |
| bt_uuid16_create(&uuid, GATT_CHARAC_APPEARANCE); |
| gatt_db_service_add_characteristic(service, &uuid, BT_ATT_PERM_READ, |
| BT_GATT_CHRC_PROP_READ, |
| gap_appearance_read_cb, |
| NULL, database); |
| |
| gatt_db_service_set_active(service, true); |
| } |
| |
| static bool get_dst_info(struct bt_att *att, bdaddr_t *dst, uint8_t *dst_type) |
| { |
| GIOChannel *io = NULL; |
| GError *gerr = NULL; |
| |
| io = g_io_channel_unix_new(bt_att_get_fd(att)); |
| if (!io) |
| return false; |
| |
| bt_io_get(io, &gerr, BT_IO_OPT_DEST_BDADDR, dst, |
| BT_IO_OPT_DEST_TYPE, dst_type, |
| BT_IO_OPT_INVALID); |
| if (gerr) { |
| error("gatt: bt_io_get: %s", gerr->message); |
| g_error_free(gerr); |
| g_io_channel_unref(io); |
| return false; |
| } |
| |
| g_io_channel_unref(io); |
| return true; |
| } |
| |
| static void gatt_ccc_read_cb(struct gatt_db_attribute *attrib, |
| unsigned int id, uint16_t offset, |
| uint8_t opcode, struct bt_att *att, |
| void *user_data) |
| { |
| struct btd_gatt_database *database = user_data; |
| struct ccc_state *ccc; |
| uint16_t handle; |
| uint8_t ecode = 0; |
| const uint8_t *value = NULL; |
| size_t len = 0; |
| bdaddr_t bdaddr; |
| uint8_t bdaddr_type; |
| |
| handle = gatt_db_attribute_get_handle(attrib); |
| |
| DBG("CCC read called for handle: 0x%04x", handle); |
| |
| if (offset > 2) { |
| ecode = BT_ATT_ERROR_INVALID_OFFSET; |
| goto done; |
| } |
| |
| if (!get_dst_info(att, &bdaddr, &bdaddr_type)) { |
| ecode = BT_ATT_ERROR_UNLIKELY; |
| goto done; |
| } |
| |
| ccc = get_ccc_state(database, &bdaddr, bdaddr_type, handle); |
| if (!ccc) { |
| ecode = BT_ATT_ERROR_UNLIKELY; |
| goto done; |
| } |
| |
| len = 2 - offset; |
| value = len ? &ccc->value[offset] : NULL; |
| |
| done: |
| gatt_db_attribute_read_result(attrib, id, ecode, value, len); |
| } |
| |
| static void gatt_ccc_write_cb(struct gatt_db_attribute *attrib, |
| unsigned int id, uint16_t offset, |
| const uint8_t *value, size_t len, |
| uint8_t opcode, struct bt_att *att, |
| void *user_data) |
| { |
| struct btd_gatt_database *database = user_data; |
| struct ccc_state *ccc; |
| struct ccc_cb_data *ccc_cb; |
| uint16_t handle; |
| uint8_t ecode = 0; |
| bdaddr_t bdaddr; |
| uint8_t bdaddr_type; |
| |
| handle = gatt_db_attribute_get_handle(attrib); |
| |
| DBG("CCC write called for handle: 0x%04x", handle); |
| |
| if (!value || len != 2) { |
| ecode = BT_ATT_ERROR_INVALID_ATTRIBUTE_VALUE_LEN; |
| goto done; |
| } |
| |
| if (offset > 2) { |
| ecode = BT_ATT_ERROR_INVALID_OFFSET; |
| goto done; |
| } |
| |
| if (!get_dst_info(att, &bdaddr, &bdaddr_type)) { |
| ecode = BT_ATT_ERROR_UNLIKELY; |
| goto done; |
| } |
| |
| ccc = get_ccc_state(database, &bdaddr, bdaddr_type, handle); |
| if (!ccc) { |
| ecode = BT_ATT_ERROR_UNLIKELY; |
| goto done; |
| } |
| |
| ccc_cb = queue_find(database->ccc_callbacks, ccc_cb_match_handle, |
| UINT_TO_PTR(gatt_db_attribute_get_handle(attrib))); |
| if (!ccc_cb) { |
| ecode = BT_ATT_ERROR_UNLIKELY; |
| goto done; |
| } |
| |
| /* If value is identical, then just succeed */ |
| if (ccc->value[0] == value[0] && ccc->value[1] == value[1]) |
| goto done; |
| |
| if (ccc_cb->callback) |
| ecode = ccc_cb->callback(get_le16(value), ccc_cb->user_data); |
| |
| if (!ecode) { |
| ccc->value[0] = value[0]; |
| ccc->value[1] = value[1]; |
| } |
| |
| done: |
| gatt_db_attribute_write_result(attrib, id, ecode); |
| } |
| |
| static struct gatt_db_attribute * |
| service_add_ccc(struct gatt_db_attribute *service, |
| struct btd_gatt_database *database, |
| btd_gatt_database_ccc_write_t write_callback, |
| void *user_data, |
| btd_gatt_database_destroy_t destroy) |
| { |
| struct gatt_db_attribute *ccc; |
| struct ccc_cb_data *ccc_cb; |
| bt_uuid_t uuid; |
| |
| ccc_cb = new0(struct ccc_cb_data, 1); |
| if (!ccc_cb) { |
| error("Could not allocate memory for callback data"); |
| return NULL; |
| } |
| |
| bt_uuid16_create(&uuid, GATT_CLIENT_CHARAC_CFG_UUID); |
| ccc = gatt_db_service_add_descriptor(service, &uuid, |
| BT_ATT_PERM_READ | BT_ATT_PERM_WRITE, |
| gatt_ccc_read_cb, gatt_ccc_write_cb, database); |
| if (!ccc) { |
| error("Failed to create CCC entry in database"); |
| free(ccc_cb); |
| return NULL; |
| } |
| |
| ccc_cb->handle = gatt_db_attribute_get_handle(ccc); |
| ccc_cb->callback = write_callback; |
| ccc_cb->destroy = destroy; |
| ccc_cb->user_data = user_data; |
| |
| queue_push_tail(database->ccc_callbacks, ccc_cb); |
| |
| return ccc; |
| } |
| |
| struct gatt_db_attribute * |
| btd_gatt_database_add_ccc(struct btd_gatt_database *database, |
| uint16_t service_handle, |
| btd_gatt_database_ccc_write_t write_callback, |
| void *user_data, |
| btd_gatt_database_destroy_t destroy) |
| { |
| struct gatt_db_attribute *service; |
| |
| if (!database || !service_handle) |
| return NULL; |
| |
| service = gatt_db_get_attribute(database->db, service_handle); |
| if (!service) { |
| error("No service exists with handle: 0x%04x", service_handle); |
| return NULL; |
| } |
| |
| return service_add_ccc(service, database, write_callback, user_data, |
| destroy); |
| } |
| |
| static void populate_gatt_service(struct btd_gatt_database *database) |
| { |
| bt_uuid_t uuid; |
| struct gatt_db_attribute *service; |
| |
| /* Add the GATT service */ |
| bt_uuid16_create(&uuid, UUID_GATT); |
| service = gatt_db_add_service(database->db, &uuid, true, 4); |
| database->gatt_handle = database_add_record(database, UUID_GATT, |
| service, |
| "Generic Attribute Profile"); |
| |
| bt_uuid16_create(&uuid, GATT_CHARAC_SERVICE_CHANGED); |
| database->svc_chngd = gatt_db_service_add_characteristic(service, &uuid, |
| BT_ATT_PERM_READ, BT_GATT_CHRC_PROP_INDICATE, |
| NULL, NULL, database); |
| |
| database->svc_chngd_ccc = service_add_ccc(service, database, NULL, NULL, |
| NULL); |
| |
| gatt_db_service_set_active(service, true); |
| } |
| |
| static void register_core_services(struct btd_gatt_database *database) |
| { |
| populate_gap_service(database); |
| populate_gatt_service(database); |
| } |
| |
| struct notify { |
| struct btd_gatt_database *database; |
| uint16_t handle, ccc_handle; |
| const uint8_t *value; |
| uint16_t len; |
| bool indicate; |
| }; |
| |
| static void conf_cb(void *user_data) |
| { |
| DBG("GATT server received confirmation"); |
| } |
| |
| static void send_notification_to_device(void *data, void *user_data) |
| { |
| struct device_state *device_state = data; |
| struct notify *notify = user_data; |
| struct ccc_state *ccc; |
| struct btd_device *device; |
| |
| ccc = find_ccc_state(device_state, notify->ccc_handle); |
| if (!ccc) |
| return; |
| |
| if (!ccc->value[0] || (notify->indicate && !(ccc->value[0] & 0x02))) |
| return; |
| |
| device = btd_adapter_get_device(notify->database->adapter, |
| &device_state->bdaddr, |
| device_state->bdaddr_type); |
| if (!device) |
| return; |
| |
| /* |
| * TODO: If the device is not connected but bonded, send the |
| * notification/indication when it becomes connected. |
| */ |
| if (!notify->indicate) { |
| DBG("GATT server sending notification"); |
| bt_gatt_server_send_notification( |
| btd_device_get_gatt_server(device), |
| notify->handle, notify->value, |
| notify->len); |
| return; |
| } |
| |
| DBG("GATT server sending indication"); |
| bt_gatt_server_send_indication(btd_device_get_gatt_server(device), |
| notify->handle, |
| notify->value, |
| notify->len, conf_cb, |
| NULL, NULL); |
| } |
| |
| static void send_notification_to_devices(struct btd_gatt_database *database, |
| uint16_t handle, const uint8_t *value, |
| uint16_t len, uint16_t ccc_handle, |
| bool indicate) |
| { |
| struct notify notify; |
| |
| memset(¬ify, 0, sizeof(notify)); |
| |
| notify.database = database; |
| notify.handle = handle; |
| notify.ccc_handle = ccc_handle; |
| notify.value = value; |
| notify.len = len; |
| notify.indicate = indicate; |
| |
| queue_foreach(database->device_states, send_notification_to_device, |
| ¬ify); |
| } |
| |
| static void send_service_changed(struct btd_gatt_database *database, |
| struct gatt_db_attribute *attrib) |
| { |
| uint16_t start, end; |
| uint8_t value[4]; |
| uint16_t handle, ccc_handle; |
| |
| if (!gatt_db_attribute_get_service_handles(attrib, &start, &end)) { |
| error("Failed to obtain changed service handles"); |
| return; |
| } |
| |
| handle = gatt_db_attribute_get_handle(database->svc_chngd); |
| ccc_handle = gatt_db_attribute_get_handle(database->svc_chngd_ccc); |
| |
| if (!handle || !ccc_handle) { |
| error("Failed to obtain handles for \"Service Changed\"" |
| " characteristic"); |
| return; |
| } |
| |
| put_le16(start, value); |
| put_le16(end, value + 2); |
| |
| send_notification_to_devices(database, handle, value, sizeof(value), |
| ccc_handle, true); |
| } |
| |
| static void gatt_db_service_added(struct gatt_db_attribute *attrib, |
| void *user_data) |
| { |
| struct btd_gatt_database *database = user_data; |
| |
| DBG("GATT Service added to local database"); |
| |
| send_service_changed(database, attrib); |
| } |
| |
| static bool ccc_match_service(const void *data, const void *match_data) |
| { |
| const struct ccc_state *ccc = data; |
| const struct gatt_db_attribute *attrib = match_data; |
| uint16_t start, end; |
| |
| if (!gatt_db_attribute_get_service_handles(attrib, &start, &end)) |
| return false; |
| |
| return ccc->handle >= start && ccc->handle <= end; |
| } |
| |
| static void remove_device_ccc(void *data, void *user_data) |
| { |
| struct device_state *state = data; |
| |
| queue_remove_all(state->ccc_states, ccc_match_service, user_data, free); |
| } |
| |
| static void gatt_db_service_removed(struct gatt_db_attribute *attrib, |
| void *user_data) |
| { |
| struct btd_gatt_database *database = user_data; |
| |
| DBG("Local GATT service removed"); |
| |
| send_service_changed(database, attrib); |
| |
| queue_foreach(database->device_states, remove_device_ccc, attrib); |
| queue_remove_all(database->ccc_callbacks, ccc_cb_match_service, attrib, |
| ccc_cb_free); |
| } |
| |
| struct svc_match_data { |
| const char *path; |
| const char *sender; |
| }; |
| |
| static bool match_service(const void *a, const void *b) |
| { |
| const struct external_service *service = a; |
| const struct svc_match_data *data = b; |
| |
| return g_strcmp0(service->path, data->path) == 0 && |
| g_strcmp0(service->owner, data->sender) == 0; |
| } |
| |
| static gboolean service_free_idle_cb(void *data) |
| { |
| service_free(data); |
| |
| return FALSE; |
| } |
| |
| static void service_remove_helper(void *data) |
| { |
| struct external_service *service = data; |
| |
| queue_remove(service->database->services, service); |
| |
| /* |
| * Do not run in the same loop, this may be a disconnect |
| * watch call and GDBusClient should not be destroyed. |
| */ |
| g_idle_add(service_free_idle_cb, service); |
| } |
| |
| static void client_disconnect_cb(DBusConnection *conn, void *user_data) |
| { |
| DBG("Client disconnected"); |
| |
| service_remove_helper(user_data); |
| } |
| |
| static void service_remove(void *data) |
| { |
| struct external_service *service = data; |
| |
| /* |
| * Set callback to NULL to avoid potential race condition |
| * when calling remove_service and GDBusClient unref. |
| */ |
| g_dbus_client_set_disconnect_watch(service->client, NULL, NULL); |
| |
| /* |
| * Set proxy handlers to NULL, so that this gets called only once when |
| * the first proxy that belongs to this service gets removed. |
| */ |
| g_dbus_client_set_proxy_handlers(service->client, NULL, NULL, |
| NULL, NULL); |
| |
| service_remove_helper(service); |
| } |
| |
| static struct external_chrc *chrc_create(struct external_service *service, |
| GDBusProxy *proxy, |
| const char *path) |
| { |
| struct external_chrc *chrc; |
| |
| chrc = new0(struct external_chrc, 1); |
| if (!chrc) |
| return NULL; |
| |
| chrc->pending_reads = queue_new(); |
| if (!chrc->pending_reads) { |
| free(chrc); |
| return NULL; |
| } |
| |
| chrc->pending_writes = queue_new(); |
| if (!chrc->pending_writes) { |
| queue_destroy(chrc->pending_reads, NULL); |
| free(chrc); |
| return NULL; |
| } |
| |
| chrc->path = g_strdup(path); |
| if (!chrc->path) { |
| queue_destroy(chrc->pending_reads, NULL); |
| queue_destroy(chrc->pending_writes, NULL); |
| free(chrc); |
| return NULL; |
| } |
| |
| chrc->service = service; |
| chrc->proxy = g_dbus_proxy_ref(proxy); |
| |
| return chrc; |
| } |
| |
| static struct external_desc *desc_create(struct external_service *service, |
| GDBusProxy *proxy, |
| const char *chrc_path) |
| { |
| struct external_desc *desc; |
| |
| desc = new0(struct external_desc, 1); |
| if (!desc) |
| return NULL; |
| |
| desc->pending_reads = queue_new(); |
| if (!desc->pending_reads) { |
| free(desc); |
| return NULL; |
| } |
| |
| desc->pending_writes = queue_new(); |
| if (!desc->pending_writes) { |
| queue_destroy(desc->pending_reads, NULL); |
| free(desc); |
| return NULL; |
| } |
| |
| desc->chrc_path = g_strdup(chrc_path); |
| if (!desc->chrc_path) { |
| queue_destroy(desc->pending_reads, NULL); |
| queue_destroy(desc->pending_writes, NULL); |
| free(desc); |
| return NULL; |
| } |
| |
| desc->service = service; |
| desc->proxy = g_dbus_proxy_ref(proxy); |
| |
| return desc; |
| } |
| |
| static bool incr_attr_count(struct external_service *service, uint16_t incr) |
| { |
| if (service->attr_cnt > UINT16_MAX - incr) |
| return false; |
| |
| service->attr_cnt += incr; |
| |
| return true; |
| } |
| |
| static bool parse_path(GDBusProxy *proxy, const char *name, const char **path) |
| { |
| DBusMessageIter iter; |
| |
| if (!g_dbus_proxy_get_property(proxy, name, &iter)) |
| return false; |
| |
| if (dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_OBJECT_PATH) |
| return false; |
| |
| dbus_message_iter_get_basic(&iter, path); |
| |
| return true; |
| } |
| |
| static bool check_service_path(GDBusProxy *proxy, |
| struct external_service *service) |
| { |
| const char *service_path; |
| |
| if (!parse_path(proxy, "Service", &service_path)) |
| return false; |
| |
| return g_strcmp0(service_path, service->path) == 0; |
| } |
| |
| static bool parse_chrc_flags(DBusMessageIter *array, uint8_t *props, |
| uint8_t *ext_props) |
| { |
| const char *flag; |
| |
| *props = *ext_props = 0; |
| |
| do { |
| if (dbus_message_iter_get_arg_type(array) != DBUS_TYPE_STRING) |
| return false; |
| |
| dbus_message_iter_get_basic(array, &flag); |
| |
| if (!strcmp("broadcast", flag)) |
| *props |= BT_GATT_CHRC_PROP_BROADCAST; |
| else if (!strcmp("read", flag)) |
| *props |= BT_GATT_CHRC_PROP_READ; |
| else if (!strcmp("write-without-response", flag)) |
| *props |= BT_GATT_CHRC_PROP_WRITE_WITHOUT_RESP; |
| else if (!strcmp("write", flag)) |
| *props |= BT_GATT_CHRC_PROP_WRITE; |
| else if (!strcmp("notify", flag)) |
| *props |= BT_GATT_CHRC_PROP_NOTIFY; |
| else if (!strcmp("indicate", flag)) |
| *props |= BT_GATT_CHRC_PROP_INDICATE; |
| else if (!strcmp("authenticated-signed-writes", flag)) |
| *props |= BT_GATT_CHRC_PROP_AUTH; |
| else if (!strcmp("reliable-write", flag)) |
| *ext_props |= BT_GATT_CHRC_EXT_PROP_RELIABLE_WRITE; |
| else if (!strcmp("writable-auxiliaries", flag)) |
| *ext_props |= BT_GATT_CHRC_EXT_PROP_WRITABLE_AUX; |
| else if (!strcmp("encrypt-read", flag)) { |
| *props |= BT_GATT_CHRC_PROP_READ; |
| *ext_props |= BT_GATT_CHRC_EXT_PROP_ENC_READ; |
| } else if (!strcmp("encrypt-write", flag)) { |
| *props |= BT_GATT_CHRC_PROP_WRITE; |
| *ext_props |= BT_GATT_CHRC_EXT_PROP_ENC_WRITE; |
| } else if (!strcmp("encrypt-authenticated-read", flag)) { |
| *props |= BT_GATT_CHRC_PROP_READ; |
| *ext_props |= BT_GATT_CHRC_EXT_PROP_AUTH_READ; |
| } else if (!strcmp("encrypt-authenticated-write", flag)) { |
| *props |= BT_GATT_CHRC_PROP_WRITE; |
| *ext_props |= BT_GATT_CHRC_EXT_PROP_AUTH_WRITE; |
| } else { |
| error("Invalid characteristic flag: %s", flag); |
| return false; |
| } |
| } while (dbus_message_iter_next(array)); |
| |
| if (*ext_props) |
| *props |= BT_GATT_CHRC_PROP_EXT_PROP; |
| |
| return true; |
| } |
| |
| static bool parse_desc_flags(DBusMessageIter *array, uint32_t *perm) |
| { |
| const char *flag; |
| |
| *perm = 0; |
| |
| do { |
| if (dbus_message_iter_get_arg_type(array) != DBUS_TYPE_STRING) |
| return false; |
| |
| dbus_message_iter_get_basic(array, &flag); |
| |
| if (!strcmp("read", flag)) |
| *perm |= BT_ATT_PERM_READ; |
| else if (!strcmp("write", flag)) |
| *perm |= BT_ATT_PERM_WRITE; |
| else if (!strcmp("encrypt-read", flag)) |
| *perm |= BT_ATT_PERM_READ | BT_ATT_PERM_READ_ENCRYPT; |
| else if (!strcmp("encrypt-write", flag)) |
| *perm |= BT_ATT_PERM_WRITE | BT_ATT_PERM_WRITE_ENCRYPT; |
| else if (!strcmp("encrypt-authenticated-read", flag)) |
| *perm |= BT_ATT_PERM_READ | BT_ATT_PERM_READ_AUTHEN; |
| else if (!strcmp("encrypt-authenticated-write", flag)) |
| *perm |= BT_ATT_PERM_WRITE | BT_ATT_PERM_WRITE_AUTHEN; |
| else { |
| error("Invalid descriptor flag: %s", flag); |
| return false; |
| } |
| } while (dbus_message_iter_next(array)); |
| |
| return true; |
| } |
| |
| static bool parse_flags(GDBusProxy *proxy, uint8_t *props, uint8_t *ext_props, |
| uint32_t *perm) |
| { |
| DBusMessageIter iter, array; |
| |
| if (!g_dbus_proxy_get_property(proxy, "Flags", &iter)) |
| return false; |
| |
| if (dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_ARRAY) |
| return false; |
| |
| dbus_message_iter_recurse(&iter, &array); |
| |
| if (perm) |
| return parse_desc_flags(&array, perm); |
| |
| return parse_chrc_flags(&array, props, ext_props); |
| } |
| |
| static void proxy_added_cb(GDBusProxy *proxy, void *user_data) |
| { |
| struct external_service *service = user_data; |
| const char *iface, *path; |
| |
| if (service->failed || service->attrib) |
| return; |
| |
| iface = g_dbus_proxy_get_interface(proxy); |
| path = g_dbus_proxy_get_path(proxy); |
| |
| if (!g_str_has_prefix(path, service->path)) |
| return; |
| |
| if (g_strcmp0(iface, GATT_SERVICE_IFACE) == 0) { |
| if (service->proxy) |
| return; |
| |
| /* |
| * TODO: We may want to support adding included services in a |
| * single hierarchy. |
| */ |
| if (g_strcmp0(path, service->path) != 0) { |
| error("Multiple services added within hierarchy"); |
| service->failed = true; |
| return; |
| } |
| |
| /* Add 1 for the service declaration */ |
| if (!incr_attr_count(service, 1)) { |
| error("Failed to increment attribute count"); |
| service->failed = true; |
| return; |
| } |
| |
| service->proxy = g_dbus_proxy_ref(proxy); |
| } else if (g_strcmp0(iface, GATT_CHRC_IFACE) == 0) { |
| struct external_chrc *chrc; |
| |
| if (g_strcmp0(path, service->path) == 0) { |
| error("Characteristic path same as service path"); |
| service->failed = true; |
| return; |
| } |
| |
| chrc = chrc_create(service, proxy, path); |
| if (!chrc) { |
| service->failed = true; |
| return; |
| } |
| |
| /* |
| * Add 2 for the characteristic declaration and the value |
| * attribute. |
| */ |
| if (!incr_attr_count(service, 2)) { |
| error("Failed to increment attribute count"); |
| service->failed = true; |
| return; |
| } |
| |
| /* |
| * Parse characteristic flags (i.e. properties) here since they |
| * are used to determine if any special descriptors should be |
| * created. |
| */ |
| if (!parse_flags(proxy, &chrc->props, &chrc->ext_props, NULL)) { |
| error("Failed to parse characteristic properties"); |
| service->failed = true; |
| return; |
| } |
| |
| if ((chrc->props & BT_GATT_CHRC_PROP_NOTIFY || |
| chrc->props & BT_GATT_CHRC_PROP_INDICATE) && |
| !incr_attr_count(service, 1)) { |
| error("Failed to increment attribute count for CCC"); |
| service->failed = true; |
| return; |
| } |
| |
| if (chrc->ext_props && !incr_attr_count(service, 1)) { |
| error("Failed to increment attribute count for CEP"); |
| service->failed = true; |
| return; |
| } |
| |
| queue_push_tail(service->chrcs, chrc); |
| } else if (g_strcmp0(iface, GATT_DESC_IFACE) == 0) { |
| struct external_desc *desc; |
| const char *chrc_path; |
| |
| if (!parse_path(proxy, "Characteristic", &chrc_path)) { |
| error("Failed to obtain characteristic path for " |
| "descriptor"); |
| service->failed = true; |
| return; |
| } |
| |
| desc = desc_create(service, proxy, chrc_path); |
| if (!desc) { |
| service->failed = true; |
| return; |
| } |
| |
| /* Add 1 for the descriptor attribute */ |
| if (!incr_attr_count(service, 1)) { |
| error("Failed to increment attribute count"); |
| service->failed = true; |
| return; |
| } |
| |
| /* |
| * Parse descriptors flags here since they are used to |
| * determine the permission the descriptor should have |
| */ |
| if (!parse_flags(proxy, NULL, NULL, &desc->perm)) { |
| error("Failed to parse characteristic properties"); |
| service->failed = true; |
| return; |
| } |
| |
| queue_push_tail(service->descs, desc); |
| } else { |
| DBG("Ignoring unrelated interface: %s", iface); |
| return; |
| } |
| |
| DBG("Object added to service - path: %s, iface: %s", path, iface); |
| } |
| |
| static void proxy_removed_cb(GDBusProxy *proxy, void *user_data) |
| { |
| struct external_service *service = user_data; |
| const char *path; |
| |
| path = g_dbus_proxy_get_path(proxy); |
| |
| if (!g_str_has_prefix(path, service->path)) |
| return; |
| |
| DBG("Proxy removed - removing service: %s", service->path); |
| |
| service_remove(service); |
| } |
| |
| static bool parse_uuid(GDBusProxy *proxy, bt_uuid_t *uuid) |
| { |
| DBusMessageIter iter; |
| bt_uuid_t tmp; |
| const char *uuidstr; |
| |
| if (!g_dbus_proxy_get_property(proxy, "UUID", &iter)) |
| return false; |
| |
| if (dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_STRING) |
| return false; |
| |
| dbus_message_iter_get_basic(&iter, &uuidstr); |
| |
| if (bt_string_to_uuid(uuid, uuidstr) < 0) |
| return false; |
| |
| /* GAP & GATT services are created and managed by BlueZ */ |
| bt_uuid16_create(&tmp, UUID_GAP); |
| if (!bt_uuid_cmp(&tmp, uuid)) { |
| error("GAP service must be handled by BlueZ"); |
| return false; |
| } |
| |
| bt_uuid16_create(&tmp, UUID_GATT); |
| if (!bt_uuid_cmp(&tmp, uuid)) { |
| error("GATT service must be handled by BlueZ"); |
| return false; |
| } |
| |
| return true; |
| } |
| |
| static bool parse_primary(GDBusProxy *proxy, bool *primary) |
| { |
| DBusMessageIter iter; |
| dbus_bool_t val; |
| |
| if (!g_dbus_proxy_get_property(proxy, "Primary", &iter)) |
| return false; |
| |
| if (dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_BOOLEAN) |
| return false; |
| |
| dbus_message_iter_get_basic(&iter, &val); |
| |
| *primary = val; |
| |
| return true; |
| } |
| |
| static uint8_t dbus_error_to_att_ecode(const char *error_name) |
| { |
| |
| if (strcmp(error_name, "org.bluez.Error.Failed") == 0) |
| return 0x80; /* For now return this "application error" */ |
| |
| if (strcmp(error_name, "org.bluez.Error.NotSupported") == 0) |
| return BT_ATT_ERROR_REQUEST_NOT_SUPPORTED; |
| |
| if (strcmp(error_name, "org.bluez.Error.NotAuthorized") == 0) |
| return BT_ATT_ERROR_AUTHORIZATION; |
| |
| if (strcmp(error_name, "org.bluez.Error.InvalidValueLength") == 0) |
| return BT_ATT_ERROR_INVALID_ATTRIBUTE_VALUE_LEN; |
| |
| if (strcmp(error_name, "org.bluez.Error.InProgress") == 0) |
| return BT_ERROR_ALREADY_IN_PROGRESS; |
| |
| return 0; |
| } |
| |
| static void read_reply_cb(DBusMessage *message, void *user_data) |
| { |
| struct pending_op *op = user_data; |
| DBusError err; |
| DBusMessageIter iter, array; |
| uint8_t ecode = 0; |
| uint8_t *value = NULL; |
| int len = 0; |
| |
| if (!op->owner_queue) { |
| DBG("Pending read was canceled when object got removed"); |
| return; |
| } |
| |
| dbus_error_init(&err); |
| |
| if (dbus_set_error_from_message(&err, message) == TRUE) { |
| DBG("Failed to read value: %s: %s", err.name, err.message); |
| ecode = dbus_error_to_att_ecode(err.name); |
| ecode = ecode ? ecode : BT_ATT_ERROR_READ_NOT_PERMITTED; |
| dbus_error_free(&err); |
| goto done; |
| } |
| |
| dbus_message_iter_init(message, &iter); |
| |
| if (dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_ARRAY) { |
| /* |
| * Return not supported for this, as the external app basically |
| * doesn't properly support reading from this characteristic. |
| */ |
| ecode = BT_ATT_ERROR_REQUEST_NOT_SUPPORTED; |
| error("Invalid return value received for \"ReadValue\""); |
| goto done; |
| } |
| |
| dbus_message_iter_recurse(&iter, &array); |
| dbus_message_iter_get_fixed_array(&array, &value, &len); |
| |
| if (len < 0) { |
| ecode = BT_ATT_ERROR_REQUEST_NOT_SUPPORTED; |
| value = NULL; |
| len = 0; |
| goto done; |
| } |
| |
| /* Truncate the value if it's too large */ |
| len = MIN(BT_ATT_MAX_VALUE_LEN, len); |
| value = len ? value : NULL; |
| |
| done: |
| gatt_db_attribute_read_result(op->attrib, op->id, ecode, value, len); |
| } |
| |
| static void pending_op_free(void *data) |
| { |
| struct pending_op *op = data; |
| |
| if (op->owner_queue) |
| queue_remove(op->owner_queue, op); |
| |
| free(op); |
| } |
| |
| static struct pending_op *pending_read_new(struct queue *owner_queue, |
| struct gatt_db_attribute *attrib, |
| unsigned int id) |
| { |
| struct pending_op *op; |
| |
| op = new0(struct pending_op, 1); |
| if (!op) |
| return NULL; |
| |
| op->owner_queue = owner_queue; |
| op->attrib = attrib; |
| op->id = id; |
| queue_push_tail(owner_queue, op); |
| |
| return op; |
| } |
| |
| static void send_read(struct gatt_db_attribute *attrib, GDBusProxy *proxy, |
| struct queue *owner_queue, |
| unsigned int id) |
| { |
| struct pending_op *op; |
| uint8_t ecode = BT_ATT_ERROR_UNLIKELY; |
| |
| op = pending_read_new(owner_queue, attrib, id); |
| if (!op) { |
| error("Failed to allocate memory for pending read call"); |
| ecode = BT_ATT_ERROR_INSUFFICIENT_RESOURCES; |
| goto error; |
| } |
| |
| if (g_dbus_proxy_method_call(proxy, "ReadValue", NULL, read_reply_cb, |
| op, pending_op_free) == TRUE) |
| return; |
| |
| pending_op_free(op); |
| |
| error: |
| gatt_db_attribute_read_result(attrib, id, ecode, NULL, 0); |
| } |
| |
| static void write_setup_cb(DBusMessageIter *iter, void *user_data) |
| { |
| struct pending_op *op = user_data; |
| DBusMessageIter array; |
| |
| dbus_message_iter_open_container(iter, DBUS_TYPE_ARRAY, "y", &array); |
| dbus_message_iter_append_fixed_array(&array, DBUS_TYPE_BYTE, |
| &op->data.iov_base, op->data.iov_len); |
| dbus_message_iter_close_container(iter, &array); |
| } |
| |
| static void write_reply_cb(DBusMessage *message, void *user_data) |
| { |
| struct pending_op *op = user_data; |
| DBusError err; |
| DBusMessageIter iter; |
| uint8_t ecode = 0; |
| |
| if (!op->owner_queue) { |
| DBG("Pending write was canceled when object got removed"); |
| return; |
| } |
| |
| dbus_error_init(&err); |
| |
| if (dbus_set_error_from_message(&err, message) == TRUE) { |
| DBG("Failed to write value: %s: %s", err.name, err.message); |
| ecode = dbus_error_to_att_ecode(err.name); |
| ecode = ecode ? ecode : BT_ATT_ERROR_WRITE_NOT_PERMITTED; |
| dbus_error_free(&err); |
| goto done; |
| } |
| |
| dbus_message_iter_init(message, &iter); |
| if (dbus_message_iter_has_next(&iter)) { |
| /* |
| * Return not supported for this, as the external app basically |
| * doesn't properly support the "WriteValue" API. |
| */ |
| ecode = BT_ATT_ERROR_REQUEST_NOT_SUPPORTED; |
| error("Invalid return value received for \"WriteValue\""); |
| } |
| |
| done: |
| gatt_db_attribute_write_result(op->attrib, op->id, ecode); |
| } |
| |
| static struct pending_op *pending_write_new(struct queue *owner_queue, |
| struct gatt_db_attribute *attrib, |
| unsigned int id, |
| const uint8_t *value, |
| size_t len) |
| { |
| struct pending_op *op; |
| |
| op = new0(struct pending_op, 1); |
| if (!op) |
| return NULL; |
| |
| op->data.iov_base = (uint8_t *) value; |
| op->data.iov_len = len; |
| |
| op->owner_queue = owner_queue; |
| op->attrib = attrib; |
| op->id = id; |
| queue_push_tail(owner_queue, op); |
| |
| return op; |
| } |
| |
| static void send_write(struct gatt_db_attribute *attrib, GDBusProxy *proxy, |
| struct queue *owner_queue, |
| unsigned int id, |
| const uint8_t *value, size_t len) |
| { |
| struct pending_op *op; |
| uint8_t ecode = BT_ATT_ERROR_UNLIKELY; |
| |
| op = pending_write_new(owner_queue, attrib, id, value, len); |
| if (!op) { |
| error("Failed to allocate memory for pending read call"); |
| ecode = BT_ATT_ERROR_INSUFFICIENT_RESOURCES; |
| goto error; |
| } |
| |
| if (g_dbus_proxy_method_call(proxy, "WriteValue", write_setup_cb, |
| write_reply_cb, op, |
| pending_op_free) == TRUE) |
| return; |
| |
| pending_op_free(op); |
| |
| error: |
| gatt_db_attribute_write_result(attrib, id, ecode); |
| } |
| |
| static uint32_t permissions_from_props(uint8_t props, uint8_t ext_props) |
| { |
| uint32_t perm = 0; |
| |
| if (props & BT_GATT_CHRC_PROP_WRITE || |
| props & BT_GATT_CHRC_PROP_WRITE_WITHOUT_RESP || |
| ext_props & BT_GATT_CHRC_EXT_PROP_RELIABLE_WRITE || |
| ext_props & BT_GATT_CHRC_EXT_PROP_ENC_WRITE || |
| ext_props & BT_GATT_CHRC_EXT_PROP_AUTH_WRITE) |
| perm |= BT_ATT_PERM_WRITE; |
| |
| if (props & BT_GATT_CHRC_PROP_READ || |
| ext_props & BT_GATT_CHRC_EXT_PROP_ENC_READ || |
| ext_props & BT_GATT_CHRC_EXT_PROP_AUTH_READ) |
| perm |= BT_ATT_PERM_READ; |
| |
| if (ext_props & BT_GATT_CHRC_EXT_PROP_ENC_READ) |
| perm |= BT_ATT_PERM_READ_ENCRYPT; |
| |
| if (ext_props & BT_GATT_CHRC_EXT_PROP_ENC_WRITE) |
| perm |= BT_ATT_PERM_WRITE_ENCRYPT; |
| |
| if (ext_props & BT_GATT_CHRC_EXT_PROP_AUTH_READ) |
| perm |= BT_ATT_PERM_READ_AUTHEN; |
| |
| if (ext_props & BT_GATT_CHRC_EXT_PROP_AUTH_WRITE) |
| perm |= BT_ATT_PERM_WRITE_AUTHEN; |
| |
| return perm; |
| } |
| |
| static uint8_t ccc_write_cb(uint16_t value, void *user_data) |
| { |
| struct external_chrc *chrc = user_data; |
| |
| DBG("External CCC write received with value: 0x%04x", value); |
| |
| /* Notifications/indications disabled */ |
| if (!value) { |
| if (!chrc->ntfy_cnt) |
| return 0; |
| |
| if (__sync_sub_and_fetch(&chrc->ntfy_cnt, 1)) |
| return 0; |
| |
| /* |
| * Send request to stop notifying. This is best-effort |
| * operation, so simply ignore the return the value. |
| */ |
| g_dbus_proxy_method_call(chrc->proxy, "StopNotify", NULL, |
| NULL, NULL, NULL); |
| return 0; |
| } |
| |
| /* |
| * TODO: All of the errors below should fall into the so called |
| * "Application Error" range. Since there is no well defined error for |
| * these, we return a generic ATT protocol error for now. |
| */ |
| |
| if (chrc->ntfy_cnt == UINT_MAX) { |
| /* Maximum number of per-device CCC descriptors configured */ |
| return BT_ATT_ERROR_REQUEST_NOT_SUPPORTED; |
| } |
| |
| /* Don't support undefined CCC values yet */ |
| if (value > 2 || |
| (value == 1 && !(chrc->props & BT_GATT_CHRC_PROP_NOTIFY)) || |
| (value == 2 && !(chrc->props & BT_GATT_CHRC_PROP_INDICATE))) |
| return BT_ATT_ERROR_REQUEST_NOT_SUPPORTED; |
| |
| /* |
| * Always call StartNotify for an incoming enable and ignore the return |
| * value for now. |
| */ |
| if (g_dbus_proxy_method_call(chrc->proxy, |
| "StartNotify", NULL, NULL, |
| NULL, NULL) == FALSE) |
| return BT_ATT_ERROR_REQUEST_NOT_SUPPORTED; |
| |
| __sync_fetch_and_add(&chrc->ntfy_cnt, 1); |
| |
| return 0; |
| } |
| |
| static void property_changed_cb(GDBusProxy *proxy, const char *name, |
| DBusMessageIter *iter, void *user_data) |
| { |
| struct external_chrc *chrc = user_data; |
| DBusMessageIter array; |
| uint8_t *value = NULL; |
| int len = 0; |
| |
| if (strcmp(name, "Value")) |
| return; |
| |
| if (dbus_message_iter_get_arg_type(iter) != DBUS_TYPE_ARRAY) { |
| DBG("Malformed \"Value\" property received"); |
| return; |
| } |
| |
| dbus_message_iter_recurse(iter, &array); |
| dbus_message_iter_get_fixed_array(&array, &value, &len); |
| |
| if (len < 0) { |
| DBG("Malformed \"Value\" property received"); |
| return; |
| } |
| |
| /* Truncate the value if it's too large */ |
| len = MIN(BT_ATT_MAX_VALUE_LEN, len); |
| value = len ? value : NULL; |
| |
| send_notification_to_devices(chrc->service->database, |
| gatt_db_attribute_get_handle(chrc->attrib), |
| value, len, |
| gatt_db_attribute_get_handle(chrc->ccc), |
| chrc->props & BT_GATT_CHRC_PROP_INDICATE); |
| } |
| |
| static bool database_add_ccc(struct external_service *service, |
| struct external_chrc *chrc) |
| { |
| if (!(chrc->props & BT_GATT_CHRC_PROP_NOTIFY) && |
| !(chrc->props & BT_GATT_CHRC_PROP_INDICATE)) |
| return true; |
| |
| chrc->ccc = service_add_ccc(service->attrib, service->database, |
| ccc_write_cb, chrc, NULL); |
| if (!chrc->ccc) { |
| error("Failed to create CCC entry for characteristic"); |
| return false; |
| } |
| |
| if (g_dbus_proxy_set_property_watch(chrc->proxy, property_changed_cb, |
| chrc) == FALSE) { |
| error("Failed to set up property watch for characteristic"); |
| return false; |
| } |
| |
| DBG("Created CCC entry for characteristic"); |
| |
| return true; |
| } |
| |
| static void cep_write_cb(struct gatt_db_attribute *attrib, int err, |
| void *user_data) |
| { |
| if (err) |
| DBG("Failed to store CEP value in the database"); |
| else |
| DBG("Stored CEP value in the database"); |
| } |
| |
| static bool database_add_cep(struct external_service *service, |
| struct external_chrc *chrc) |
| { |
| struct gatt_db_attribute *cep; |
| bt_uuid_t uuid; |
| uint8_t value[2]; |
| |
| if (!chrc->ext_props) |
| return true; |
| |
| bt_uuid16_create(&uuid, GATT_CHARAC_EXT_PROPER_UUID); |
| cep = gatt_db_service_add_descriptor(service->attrib, &uuid, |
| BT_ATT_PERM_READ, |
| NULL, NULL, NULL); |
| if (!cep) { |
| error("Failed to create CEP entry for characteristic"); |
| return false; |
| } |
| |
| memset(value, 0, sizeof(value)); |
| value[0] = chrc->ext_props; |
| |
| if (!gatt_db_attribute_write(cep, 0, value, sizeof(value), 0, NULL, |
| cep_write_cb, NULL)) { |
| DBG("Failed to store CEP value in the database"); |
| return false; |
| } |
| |
| DBG("Created CEP entry for characteristic"); |
| |
| return true; |
| } |
| |
| static void desc_read_cb(struct gatt_db_attribute *attrib, |
| unsigned int id, uint16_t offset, |
| uint8_t opcode, struct bt_att *att, |
| void *user_data) |
| { |
| struct external_desc *desc = user_data; |
| |
| if (desc->attrib != attrib) { |
| error("Read callback called with incorrect attribute"); |
| return; |
| } |
| |
| send_read(attrib, desc->proxy, desc->pending_reads, id); |
| } |
| |
| static void desc_write_cb(struct gatt_db_attribute *attrib, |
| unsigned int id, uint16_t offset, |
| const uint8_t *value, size_t len, |
| uint8_t opcode, struct bt_att *att, |
| void *user_data) |
| { |
| struct external_desc *desc = user_data; |
| |
| if (desc->attrib != attrib) { |
| error("Read callback called with incorrect attribute"); |
| return; |
| } |
| |
| send_write(attrib, desc->proxy, desc->pending_writes, id, value, len); |
| } |
| |
| static bool database_add_desc(struct external_service *service, |
| struct external_desc *desc) |
| { |
| bt_uuid_t uuid; |
| |
| if (!parse_uuid(desc->proxy, &uuid)) { |
| error("Failed to read \"UUID\" property of descriptor"); |
| return false; |
| } |
| |
| desc->attrib = gatt_db_service_add_descriptor(service->attrib, &uuid, |
| desc->perm, |
| desc_read_cb, |
| desc_write_cb, desc); |
| if (!desc->attrib) { |
| error("Failed to create descriptor entry in database"); |
| return false; |
| } |
| |
| desc->handled = true; |
| |
| return true; |
| } |
| |
| static void chrc_read_cb(struct gatt_db_attribute *attrib, |
| unsigned int id, uint16_t offset, |
| uint8_t opcode, struct bt_att *att, |
| void *user_data) |
| { |
| struct external_chrc *chrc = user_data; |
| |
| if (chrc->attrib != attrib) { |
| error("Read callback called with incorrect attribute"); |
| return; |
| } |
| |
| send_read(attrib, chrc->proxy, chrc->pending_reads, id); |
| } |
| |
| static void chrc_write_cb(struct gatt_db_attribute *attrib, |
| unsigned int id, uint16_t offset, |
| const uint8_t *value, size_t len, |
| uint8_t opcode, struct bt_att *att, |
| void *user_data) |
| { |
| struct external_chrc *chrc = user_data; |
| |
| if (chrc->attrib != attrib) { |
| error("Write callback called with incorrect attribute"); |
| return; |
| } |
| |
| send_write(attrib, chrc->proxy, chrc->pending_writes, id, value, len); |
| } |
| |
| static bool database_add_chrc(struct external_service *service, |
| struct external_chrc *chrc) |
| { |
| bt_uuid_t uuid; |
| uint32_t perm; |
| const struct queue_entry *entry; |
| |
| if (!parse_uuid(chrc->proxy, &uuid)) { |
| error("Failed to read \"UUID\" property of characteristic"); |
| return false; |
| } |
| |
| if (!check_service_path(chrc->proxy, service)) { |
| error("Invalid service path for characteristic"); |
| return false; |
| } |
| |
| /* |
| * TODO: Once shared/gatt-server properly supports permission checks, |
| * set the permissions based on a D-Bus property of the external |
| * characteristic. |
| */ |
| perm = permissions_from_props(chrc->props, chrc->ext_props); |
| chrc->attrib = gatt_db_service_add_characteristic(service->attrib, |
| &uuid, perm, |
| chrc->props, chrc_read_cb, |
| chrc_write_cb, chrc); |
| if (!chrc->attrib) { |
| error("Failed to create characteristic entry in database"); |
| return false; |
| } |
| |
| if (!database_add_ccc(service, chrc)) |
| return false; |
| |
| if (!database_add_cep(service, chrc)) |
| return false; |
| |
| /* Handle the descriptors that belong to this characteristic. */ |
| for (entry = queue_get_entries(service->descs); entry; |
| entry = entry->next) { |
| struct external_desc *desc = entry->data; |
| |
| if (desc->handled || g_strcmp0(desc->chrc_path, chrc->path)) |
| continue; |
| |
| if (!database_add_desc(service, desc)) { |
| chrc->attrib = NULL; |
| error("Failed to create descriptor entry"); |
| return false; |
| } |
| } |
| |
| return true; |
| } |
| |
| static bool match_desc_unhandled(const void *a, const void *b) |
| { |
| const struct external_desc *desc = a; |
| |
| return !desc->handled; |
| } |
| |
| static bool create_service_entry(struct external_service *service) |
| { |
| bt_uuid_t uuid; |
| bool primary; |
| const struct queue_entry *entry; |
| |
| if (!parse_uuid(service->proxy, &uuid)) { |
| error("Failed to read \"UUID\" property of service"); |
| return false; |
| } |
| |
| if (!parse_primary(service->proxy, &primary)) { |
| error("Failed to read \"Primary\" property of service"); |
| return false; |
| } |
| |
| service->attrib = gatt_db_add_service(service->database->db, &uuid, |
| primary, service->attr_cnt); |
| if (!service->attrib) |
| return false; |
| |
| entry = queue_get_entries(service->chrcs); |
| while (entry) { |
| struct external_chrc *chrc = entry->data; |
| |
| if (!database_add_chrc(service, chrc)) { |
| error("Failed to add characteristic"); |
| goto fail; |
| } |
| |
| entry = entry->next; |
| } |
| |
| /* If there are any unhandled descriptors, return an error */ |
| if (queue_find(service->descs, match_desc_unhandled, NULL)) { |
| error("Found descriptor with no matching characteristic!"); |
| goto fail; |
| } |
| |
| gatt_db_service_set_active(service->attrib, true); |
| |
| return true; |
| |
| fail: |
| gatt_db_remove_service(service->database->db, service->attrib); |
| service->attrib = NULL; |
| |
| return false; |
| } |
| |
| static void client_ready_cb(GDBusClient *client, void *user_data) |
| { |
| struct external_service *service = user_data; |
| DBusMessage *reply; |
| bool fail = false; |
| |
| if (!service->proxy || service->failed) { |
| error("No valid external GATT objects found"); |
| fail = true; |
| reply = btd_error_failed(service->reg, |
| "No valid service object found"); |
| goto reply; |
| } |
| |
| if (!create_service_entry(service)) { |
| error("Failed to create GATT service entry in local database"); |
| fail = true; |
| reply = btd_error_failed(service->reg, |
| "Failed to create entry in database"); |
| goto reply; |
| } |
| |
| DBG("GATT service registered: %s", service->path); |
| |
| reply = dbus_message_new_method_return(service->reg); |
| |
| reply: |
| g_dbus_send_message(btd_get_dbus_connection(), reply); |
| dbus_message_unref(service->reg); |
| service->reg = NULL; |
| |
| if (fail) |
| service_remove(service); |
| } |
| |
| static struct external_service *service_create(DBusConnection *conn, |
| DBusMessage *msg, const char *path) |
| { |
| struct external_service *service; |
| const char *sender = dbus_message_get_sender(msg); |
| |
| if (!path || !g_str_has_prefix(path, "/")) |
| return NULL; |
| |
| service = new0(struct external_service, 1); |
| if (!service) |
| return NULL; |
| |
| service->client = g_dbus_client_new_full(conn, sender, path, path); |
| if (!service->client) |
| goto fail; |
| |
| service->owner = g_strdup(sender); |
| if (!service->owner) |
| goto fail; |
| |
| service->path = g_strdup(path); |
| if (!service->path) |
| goto fail; |
| |
| service->chrcs = queue_new(); |
| if (!service->chrcs) |
| goto fail; |
| |
| service->descs = queue_new(); |
| if (!service->descs) |
| goto fail; |
| |
| service->reg = dbus_message_ref(msg); |
| |
| g_dbus_client_set_disconnect_watch(service->client, |
| client_disconnect_cb, service); |
| g_dbus_client_set_proxy_handlers(service->client, proxy_added_cb, |
| proxy_removed_cb, NULL, |
| service); |
| g_dbus_client_set_ready_watch(service->client, client_ready_cb, |
| service); |
| |
| return service; |
| |
| fail: |
| service_free(service); |
| return NULL; |
| } |
| |
| static DBusMessage *manager_register_service(DBusConnection *conn, |
| DBusMessage *msg, void *user_data) |
| { |
| struct btd_gatt_database *database = user_data; |
| const char *sender = dbus_message_get_sender(msg); |
| DBusMessageIter args; |
| const char *path; |
| struct external_service *service; |
| struct svc_match_data match_data; |
| |
| if (!dbus_message_iter_init(msg, &args)) |
| return btd_error_invalid_args(msg); |
| |
| if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_OBJECT_PATH) |
| return btd_error_invalid_args(msg); |
| |
| dbus_message_iter_get_basic(&args, &path); |
| |
| match_data.path = path; |
| match_data.sender = sender; |
| |
| if (queue_find(database->services, match_service, &match_data)) |
| return btd_error_already_exists(msg); |
| |
| dbus_message_iter_next(&args); |
| if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_ARRAY) |
| return btd_error_invalid_args(msg); |
| |
| service = service_create(conn, msg, path); |
| if (!service) |
| return btd_error_failed(msg, "Failed to register service"); |
| |
| DBG("Registering service - path: %s", path); |
| |
| service->database = database; |
| queue_push_tail(database->services, service); |
| |
| return NULL; |
| } |
| |
| static DBusMessage *manager_unregister_service(DBusConnection *conn, |
| DBusMessage *msg, void *user_data) |
| { |
| struct btd_gatt_database *database = user_data; |
| const char *sender = dbus_message_get_sender(msg); |
| const char *path; |
| DBusMessageIter args; |
| struct external_service *service; |
| struct svc_match_data match_data; |
| |
| if (!dbus_message_iter_init(msg, &args)) |
| return btd_error_invalid_args(msg); |
| |
| if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_OBJECT_PATH) |
| return btd_error_invalid_args(msg); |
| |
| dbus_message_iter_get_basic(&args, &path); |
| |
| match_data.path = path; |
| match_data.sender = sender; |
| |
| service = queue_remove_if(database->services, match_service, |
| &match_data); |
| if (!service) |
| return btd_error_does_not_exist(msg); |
| |
| service_free(service); |
| |
| return dbus_message_new_method_return(msg); |
| } |
| |
| static void profile_exited(DBusConnection *conn, void *user_data) |
| { |
| struct external_profile *profile = user_data; |
| |
| DBG("\"%s\" exited", profile->owner); |
| |
| profile->id = 0; |
| |
| queue_remove(profile->database->profiles, profile); |
| |
| profile_free(profile); |
| } |
| |
| static int profile_add(struct external_profile *profile, const char *uuid) |
| { |
| struct btd_profile *p; |
| |
| p = new0(struct btd_profile, 1); |
| if (!p) |
| return -ENOMEM; |
| |
| /* Assign directly to avoid having extra fields */ |
| p->name = (const void *) g_strdup_printf("%s%s/%s", profile->owner, |
| profile->path, uuid); |
| if (!p->name) |
| return -ENOMEM; |
| |
| p->remote_uuid = (const void *) g_strdup(uuid); |
| if (!p->remote_uuid) |
| return -ENOMEM; |
| |
| p->auto_connect = true; |
| |
| queue_push_tail(profile->profiles, p); |
| |
| DBG("Added \"%s\"", p->name); |
| |
| return 0; |
| } |
| |
| static void add_profile(void *data, void *user_data) |
| { |
| struct btd_adapter *adapter = user_data; |
| |
| adapter_add_profile(adapter, data); |
| } |
| |
| static int profile_create(DBusConnection *conn, |
| struct btd_gatt_database *database, |
| const char *sender, const char *path, |
| DBusMessageIter *iter) |
| { |
| struct external_profile *profile; |
| DBusMessageIter uuids; |
| |
| if (!path || !g_str_has_prefix(path, "/")) |
| return -EINVAL; |
| |
| profile = new0(struct external_profile, 1); |
| if (!profile) |
| return -ENOMEM; |
| |
| profile->owner = g_strdup(sender); |
| if (!profile->owner) |
| goto fail; |
| |
| profile->path = g_strdup(path); |
| if (!profile->path) |
| goto fail; |
| |
| profile->profiles = queue_new(); |
| if (!profile->profiles) |
| goto fail; |
| |
| profile->database = database; |
| profile->id = g_dbus_add_disconnect_watch(conn, sender, profile_exited, |
| profile, NULL); |
| |
| dbus_message_iter_recurse(iter, &uuids); |
| |
| while (dbus_message_iter_get_arg_type(&uuids) == DBUS_TYPE_STRING) { |
| const char *uuid; |
| |
| dbus_message_iter_get_basic(&uuids, &uuid); |
| |
| if (profile_add(profile, uuid) < 0) |
| goto fail; |
| |
| dbus_message_iter_next(&uuids); |
| } |
| |
| if (queue_isempty(profile->profiles)) |
| goto fail; |
| |
| queue_foreach(profile->profiles, add_profile, database->adapter); |
| queue_push_tail(database->profiles, profile); |
| |
| return 0; |
| |
| fail: |
| profile_free(profile); |
| return -EINVAL; |
| } |
| |
| static bool match_profile(const void *a, const void *b) |
| { |
| const struct external_profile *profile = a; |
| const struct svc_match_data *data = b; |
| |
| return g_strcmp0(profile->path, data->path) == 0 && |
| g_strcmp0(profile->owner, data->sender) == 0; |
| } |
| |
| static DBusMessage *manager_register_profile(DBusConnection *conn, |
| DBusMessage *msg, void *user_data) |
| { |
| struct btd_gatt_database *database = user_data; |
| const char *sender = dbus_message_get_sender(msg); |
| DBusMessageIter args; |
| const char *path; |
| struct svc_match_data match_data; |
| |
| DBG("sender %s", sender); |
| |
| if (!dbus_message_iter_init(msg, &args)) |
| return btd_error_invalid_args(msg); |
| |
| if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_OBJECT_PATH) |
| return btd_error_invalid_args(msg); |
| |
| dbus_message_iter_get_basic(&args, &path); |
| |
| match_data.path = path; |
| match_data.sender = sender; |
| |
| if (queue_find(database->profiles, match_profile, &match_data)) |
| return btd_error_already_exists(msg); |
| |
| dbus_message_iter_next(&args); |
| if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_ARRAY) |
| return btd_error_invalid_args(msg); |
| |
| if (profile_create(conn, database, sender, path, &args) < 0) |
| return btd_error_failed(msg, "Failed to register profile"); |
| |
| return dbus_message_new_method_return(msg); |
| } |
| |
| static DBusMessage *manager_unregister_profile(DBusConnection *conn, |
| DBusMessage *msg, void *user_data) |
| { |
| struct btd_gatt_database *database = user_data; |
| const char *sender = dbus_message_get_sender(msg); |
| const char *path; |
| DBusMessageIter args; |
| struct external_profile *profile; |
| struct svc_match_data match_data; |
| |
| if (!dbus_message_iter_init(msg, &args)) |
| return btd_error_invalid_args(msg); |
| |
| if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_OBJECT_PATH) |
| return btd_error_invalid_args(msg); |
| |
| dbus_message_iter_get_basic(&args, &path); |
| |
| match_data.path = path; |
| match_data.sender = sender; |
| |
| profile = queue_remove_if(database->profiles, match_profile, |
| &match_data); |
| if (!profile) |
| return btd_error_does_not_exist(msg); |
| |
| profile_free(profile); |
| |
| return dbus_message_new_method_return(msg); |
| } |
| |
| static const GDBusMethodTable manager_methods[] = { |
| { GDBUS_EXPERIMENTAL_ASYNC_METHOD("RegisterService", |
| GDBUS_ARGS({ "service", "o" }, { "options", "a{sv}" }), |
| NULL, manager_register_service) }, |
| { GDBUS_EXPERIMENTAL_ASYNC_METHOD("UnregisterService", |
| GDBUS_ARGS({ "service", "o" }), |
| NULL, manager_unregister_service) }, |
| { GDBUS_EXPERIMENTAL_ASYNC_METHOD("RegisterProfile", |
| GDBUS_ARGS({ "profile", "o" }, { "UUIDs", "as" }, |
| { "options", "a{sv}" }), NULL, |
| manager_register_profile) }, |
| { GDBUS_EXPERIMENTAL_ASYNC_METHOD("UnregisterProfile", |
| GDBUS_ARGS({ "profile", "o" }), |
| NULL, manager_unregister_profile) }, |
| { } |
| }; |
| |
| struct btd_gatt_database *btd_gatt_database_new(struct btd_adapter *adapter) |
| { |
| struct btd_gatt_database *database; |
| GError *gerr = NULL; |
| const bdaddr_t *addr; |
| |
| if (!adapter) |
| return NULL; |
| |
| database = new0(struct btd_gatt_database, 1); |
| if (!database) |
| return NULL; |
| |
| database->adapter = btd_adapter_ref(adapter); |
| database->db = gatt_db_new(); |
| if (!database->db) |
| goto fail; |
| |
| database->device_states = queue_new(); |
| if (!database->device_states) |
| goto fail; |
| |
| database->services = queue_new(); |
| if (!database->services) |
| goto fail; |
| |
| database->profiles = queue_new(); |
| if (!database->profiles) |
| goto fail; |
| |
| database->ccc_callbacks = queue_new(); |
| if (!database->ccc_callbacks) |
| goto fail; |
| |
| database->db_id = gatt_db_register(database->db, gatt_db_service_added, |
| gatt_db_service_removed, |
| database, NULL); |
| if (!database->db_id) |
| goto fail; |
| |
| addr = btd_adapter_get_address(adapter); |
| database->le_io = bt_io_listen(connect_cb, NULL, NULL, NULL, &gerr, |
| BT_IO_OPT_SOURCE_BDADDR, addr, |
| BT_IO_OPT_SOURCE_TYPE, BDADDR_LE_PUBLIC, |
| BT_IO_OPT_CID, ATT_CID, |
| BT_IO_OPT_SEC_LEVEL, BT_IO_SEC_LOW, |
| BT_IO_OPT_INVALID); |
| if (!database->le_io) { |
| error("Failed to start listening: %s", gerr->message); |
| g_error_free(gerr); |
| goto fail; |
| } |
| |
| /* BR/EDR socket */ |
| database->l2cap_io = bt_io_listen(connect_cb, NULL, NULL, NULL, &gerr, |
| BT_IO_OPT_SOURCE_BDADDR, addr, |
| BT_IO_OPT_PSM, ATT_PSM, |
| BT_IO_OPT_SEC_LEVEL, BT_IO_SEC_LOW, |
| BT_IO_OPT_INVALID); |
| if (database->l2cap_io == NULL) { |
| error("Failed to start listening: %s", gerr->message); |
| g_error_free(gerr); |
| goto fail; |
| } |
| |
| if (g_dbus_register_interface(btd_get_dbus_connection(), |
| adapter_get_path(adapter), |
| GATT_MANAGER_IFACE, |
| manager_methods, NULL, NULL, |
| database, NULL)) |
| DBG("GATT Manager registered for adapter: %s", |
| adapter_get_path(adapter)); |
| |
| register_core_services(database); |
| |
| return database; |
| |
| fail: |
| gatt_database_free(database); |
| |
| return NULL; |
| } |
| |
| void btd_gatt_database_destroy(struct btd_gatt_database *database) |
| { |
| if (!database) |
| return; |
| |
| g_dbus_unregister_interface(btd_get_dbus_connection(), |
| adapter_get_path(database->adapter), |
| GATT_MANAGER_IFACE); |
| |
| gatt_database_free(database); |
| } |
| |
| struct gatt_db *btd_gatt_database_get_db(struct btd_gatt_database *database) |
| { |
| if (!database) |
| return NULL; |
| |
| return database->db; |
| } |