mirror of
https://git.kernel.org/pub/scm/bluetooth/bluez.git
synced 2024-11-15 08:14:28 +08:00
gatt: Implement AcquireNotify for server
This enables IO via file descriptors using AcquireWrite if server implements it.
This commit is contained in:
parent
29a8563ce1
commit
88dfe179a5
@ -24,6 +24,7 @@
|
||||
#include <stdint.h>
|
||||
#include <stdlib.h>
|
||||
#include <errno.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include "lib/bluetooth.h"
|
||||
#include "lib/sdp.h"
|
||||
@ -118,7 +119,9 @@ struct external_chrc {
|
||||
uint8_t props;
|
||||
uint8_t ext_props;
|
||||
uint32_t perm;
|
||||
uint16_t mtu;
|
||||
struct io *write_io;
|
||||
struct io *notify_io;
|
||||
struct gatt_db_attribute *attrib;
|
||||
struct gatt_db_attribute *ccc;
|
||||
struct queue *pending_reads;
|
||||
@ -152,7 +155,8 @@ struct device_state {
|
||||
struct queue *ccc_states;
|
||||
};
|
||||
|
||||
typedef uint8_t (*btd_gatt_database_ccc_write_t) (uint16_t value,
|
||||
typedef uint8_t (*btd_gatt_database_ccc_write_t) (struct bt_att *att,
|
||||
uint16_t value,
|
||||
void *user_data);
|
||||
typedef void (*btd_gatt_database_destroy_t) (void *data);
|
||||
|
||||
@ -328,6 +332,7 @@ static void chrc_free(void *data)
|
||||
struct external_chrc *chrc = data;
|
||||
|
||||
io_destroy(chrc->write_io);
|
||||
io_destroy(chrc->notify_io);
|
||||
|
||||
queue_destroy(chrc->pending_reads, cancel_pending_read);
|
||||
queue_destroy(chrc->pending_writes, cancel_pending_write);
|
||||
@ -792,7 +797,8 @@ static void gatt_ccc_write_cb(struct gatt_db_attribute *attrib,
|
||||
goto done;
|
||||
|
||||
if (ccc_cb->callback)
|
||||
ecode = ccc_cb->callback(get_le16(value), ccc_cb->user_data);
|
||||
ecode = ccc_cb->callback(att, get_le16(value),
|
||||
ccc_cb->user_data);
|
||||
|
||||
if (!ecode) {
|
||||
ccc->value[0] = value[0];
|
||||
@ -1801,12 +1807,34 @@ static bool pipe_hup(struct io *io, void *user_data)
|
||||
|
||||
if (io == chrc->write_io)
|
||||
chrc->write_io = NULL;
|
||||
else
|
||||
chrc->notify_io = NULL;
|
||||
|
||||
io_destroy(io);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool pipe_io_read(struct io *io, void *user_data)
|
||||
{
|
||||
struct external_chrc *chrc = user_data;
|
||||
uint8_t buf[512];
|
||||
int fd = io_get_fd(io);
|
||||
ssize_t bytes_read;
|
||||
|
||||
bytes_read = read(fd, buf, sizeof(buf));
|
||||
if (bytes_read < 0)
|
||||
return false;
|
||||
|
||||
send_notification_to_devices(chrc->service->app->database,
|
||||
gatt_db_attribute_get_handle(chrc->attrib),
|
||||
buf, bytes_read,
|
||||
gatt_db_attribute_get_handle(chrc->ccc),
|
||||
false, NULL);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static struct io *pipe_io_new(int fd, void *user_data)
|
||||
{
|
||||
struct io *io;
|
||||
@ -1815,6 +1843,8 @@ static struct io *pipe_io_new(int fd, void *user_data)
|
||||
|
||||
io_set_close_on_destroy(io, true);
|
||||
|
||||
io_set_read_handler(io, pipe_io_read, user_data, NULL);
|
||||
|
||||
io_set_disconnect_handler(io, pipe_hup, user_data, NULL);
|
||||
|
||||
return io;
|
||||
@ -1915,9 +1945,64 @@ static struct pending_op *acquire_write(struct external_chrc *chrc,
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static uint8_t ccc_write_cb(uint16_t value, void *user_data)
|
||||
static void acquire_notify_reply(DBusMessage *message, void *user_data)
|
||||
{
|
||||
struct external_chrc *chrc = user_data;
|
||||
DBusError err;
|
||||
int fd;
|
||||
uint16_t mtu;
|
||||
|
||||
dbus_error_init(&err);
|
||||
|
||||
if (dbus_set_error_from_message(&err, message) == TRUE) {
|
||||
error("Failed to acquire notify: %s\n", err.name);
|
||||
dbus_error_free(&err);
|
||||
goto retry;
|
||||
}
|
||||
|
||||
if ((dbus_message_get_args(message, NULL, DBUS_TYPE_UNIX_FD, &fd,
|
||||
DBUS_TYPE_UINT16, &mtu,
|
||||
DBUS_TYPE_INVALID) == false)) {
|
||||
error("Invalid AcquirNotify response\n");
|
||||
goto retry;
|
||||
}
|
||||
|
||||
DBG("AcquireNotify success: fd %d MTU %u\n", fd, mtu);
|
||||
|
||||
chrc->notify_io = pipe_io_new(fd, chrc);
|
||||
|
||||
__sync_fetch_and_add(&chrc->ntfy_cnt, 1);
|
||||
|
||||
return;
|
||||
|
||||
retry:
|
||||
g_dbus_proxy_method_call(chrc->proxy, "StartNotify", NULL, NULL,
|
||||
NULL, NULL);
|
||||
|
||||
__sync_fetch_and_add(&chrc->ntfy_cnt, 1);
|
||||
}
|
||||
|
||||
static void acquire_notify_setup(DBusMessageIter *iter, void *user_data)
|
||||
{
|
||||
DBusMessageIter dict;
|
||||
struct external_chrc *chrc = user_data;
|
||||
|
||||
dbus_message_iter_open_container(iter, DBUS_TYPE_ARRAY,
|
||||
DBUS_DICT_ENTRY_BEGIN_CHAR_AS_STRING
|
||||
DBUS_TYPE_STRING_AS_STRING
|
||||
DBUS_TYPE_VARIANT_AS_STRING
|
||||
DBUS_DICT_ENTRY_END_CHAR_AS_STRING,
|
||||
&dict);
|
||||
|
||||
dict_append_entry(&dict, "MTU", DBUS_TYPE_UINT16, &chrc->mtu);
|
||||
|
||||
dbus_message_iter_close_container(iter, &dict);
|
||||
}
|
||||
|
||||
static uint8_t ccc_write_cb(struct bt_att *att, uint16_t value, void *user_data)
|
||||
{
|
||||
struct external_chrc *chrc = user_data;
|
||||
DBusMessageIter iter;
|
||||
|
||||
DBG("External CCC write received with value: 0x%04x", value);
|
||||
|
||||
@ -1929,6 +2014,12 @@ static uint8_t ccc_write_cb(uint16_t value, void *user_data)
|
||||
if (__sync_sub_and_fetch(&chrc->ntfy_cnt, 1))
|
||||
return 0;
|
||||
|
||||
if (chrc->notify_io) {
|
||||
io_destroy(chrc->notify_io);
|
||||
chrc->notify_io = NULL;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Send request to stop notifying. This is best-effort
|
||||
* operation, so simply ignore the return the value.
|
||||
@ -1949,12 +2040,27 @@ static uint8_t ccc_write_cb(uint16_t value, void *user_data)
|
||||
(value == 2 && !(chrc->props & BT_GATT_CHRC_PROP_INDICATE)))
|
||||
return BT_ERROR_CCC_IMPROPERLY_CONFIGURED;
|
||||
|
||||
if (chrc->notify_io) {
|
||||
__sync_fetch_and_add(&chrc->ntfy_cnt, 1);
|
||||
return 0;
|
||||
}
|
||||
|
||||
chrc->mtu = bt_att_get_mtu(att);
|
||||
|
||||
/* Make use of AcquireNotify if supported */
|
||||
if (g_dbus_proxy_get_property(chrc->proxy, "NotifyAcquired", &iter)) {
|
||||
if (g_dbus_proxy_method_call(chrc->proxy, "AcquireNotify",
|
||||
acquire_notify_setup,
|
||||
acquire_notify_reply,
|
||||
chrc, NULL))
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* 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,
|
||||
if (g_dbus_proxy_method_call(chrc->proxy, "StartNotify", NULL, NULL,
|
||||
NULL, NULL) == FALSE)
|
||||
return BT_ATT_ERROR_UNLIKELY;
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user