[PATCH 02/32] Initial support for HDP

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Creates the basic structure of the plugin, registering
adapter and device drivers.
---
 Makefile.am        |    8 +
 acinclude.m4       |    8 +-
 health/hdp.c       |  183 +++++++++++++++++++++
 health/hdp.h       |   27 +++
 health/hdp_types.h |   99 ++++++++++++
 health/hdp_util.c  |  447 ++++++++++++++++++++++++++++++++++++++++++++++++++++
 health/hdp_util.h  |   34 ++++
 health/main.c      |   60 +++++++
 health/manager.c   |  101 ++++++++++++
 health/manager.h   |   27 +++
 10 files changed, 993 insertions(+), 1 deletions(-)
 create mode 100644 health/hdp.c
 create mode 100644 health/hdp.h
 create mode 100644 health/hdp_types.h
 create mode 100644 health/hdp_util.c
 create mode 100644 health/hdp_util.h
 create mode 100644 health/main.c
 create mode 100644 health/manager.c
 create mode 100644 health/manager.h

diff --git a/Makefile.am b/Makefile.am
index 9ab5be2..d30850f 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -169,6 +169,14 @@ builtin_modules += service
 builtin_sources += plugins/service.c
 endif
 
+if HEALTHPLUGIN
+builtin_modules += health
+builtin_sources += health/main.c \
+			health/manager.h health/manager.c \
+			health/hdp.h health/hdp.c \
+			health/hdp_util.h health/hdp_util.c
+endif
+
 if MCAP
 mcap_sources += mcap/mcap_internal.h \
 		mcap/mcap_lib.h	mcap/sync.c \
diff --git a/acinclude.m4 b/acinclude.m4
index b512cfb..23c594a 100644
--- a/acinclude.m4
+++ b/acinclude.m4
@@ -167,6 +167,7 @@ AC_DEFUN([AC_ARG_BLUEZ], [
 	serial_enable=yes
 	network_enable=yes
 	service_enable=yes
+	health_enable=no
 	mcap_enable=no
 	pnat_enable=no
 	tracer_enable=no
@@ -216,6 +217,10 @@ AC_DEFUN([AC_ARG_BLUEZ], [
 		service_enable=${enableval}
 	])
 
+	AC_ARG_ENABLE(health, AC_HELP_STRING([--enable-health], [enable health plugin]), [
+			health_enable=${enableval}
+	])
+
 	AC_ARG_ENABLE(mcap, AC_HELP_STRING([--enable-mcap], [enable mcap support]), [
 			mcap_enable=${enableval}
 	])
@@ -330,7 +335,8 @@ AC_DEFUN([AC_ARG_BLUEZ], [
 	AM_CONDITIONAL(SERIALPLUGIN, test "${serial_enable}" = "yes")
 	AM_CONDITIONAL(NETWORKPLUGIN, test "${network_enable}" = "yes")
 	AM_CONDITIONAL(SERVICEPLUGIN, test "${service_enable}" = "yes")
-	AM_CONDITIONAL(MCAP, test "${mcap_enable}" = "yes")
+	AM_CONDITIONAL(HEALTHPLUGIN, test "${health_enable}" = "yes")
+	AM_CONDITIONAL(MCAP, test "${mcap_enable}" = "yes" || test "${health_enable}" = "yes")
 	AM_CONDITIONAL(ECHOPLUGIN, test "no" = "yes")
 	AM_CONDITIONAL(PNATPLUGIN, test "${pnat_enable}" = "yes")
 	AM_CONDITIONAL(TRACER, test "${tracer_enable}" = "yes")
diff --git a/health/hdp.c b/health/hdp.c
new file mode 100644
index 0000000..281be05
--- /dev/null
+++ b/health/hdp.c
@@ -0,0 +1,183 @@
+/*
+ *
+ *  BlueZ - Bluetooth protocol stack for Linux
+ *
+ *  Copyright (C) 2010 GSyC/LibreSoft, Universidad Rey Juan Carlos.
+ *  Authors:
+ *  Santiago Carot Nemesio <sancane at gmail.com>
+ *  Jose Antonio Santos-Cadenas <santoscadenas at gmail.com>
+ *
+ *  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.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, write to the Free Software
+ *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ *
+ */
+#include <gdbus.h>
+
+#include "logging.h"
+#include "error.h"
+
+#include "hdp_types.h"
+#include "hdp_util.h"
+
+#define HEALTH_MANAGER_INTERFACE 	"org.bluez.HealthAdapter"
+
+#include "../src/dbus-common.h"
+
+static GSList *adapters = NULL;
+
+static struct hdp_adapter *find_adapter(GSList *list,
+					struct btd_adapter *btd_adapter)
+{
+	GSList *l;
+	struct hdp_adapter *adapter;
+
+	for (l = list; l; l = l->next) {
+		adapter = l->data;
+		if (adapter->btd_adapter == btd_adapter)
+			return adapter;
+	}
+
+	return NULL;
+}
+
+static void hdp_set_instance_id(struct hdp_instance *hdpi)
+{
+	struct hdp_adapter *adapter = hdpi->adapter;
+
+	hdpi->id = adapter->ic++;
+}
+
+static DBusMessage *hdp_create_instance(DBusConnection *conn,
+					DBusMessage *msg, void *user_data)
+{
+	struct hdp_adapter *adapter = user_data;
+	const char *path, *name;
+	DBusMessageIter iter;
+	GError *err = NULL;
+	DBusMessage *reply;
+	struct hdp_instance *hdpi;
+	struct hdp_config *config;
+	int ctype;
+
+	dbus_message_iter_init(msg, &iter);
+	ctype = dbus_message_iter_get_arg_type(&iter);
+	if (ctype != DBUS_TYPE_OBJECT_PATH)
+		goto error;
+	dbus_message_iter_get_basic(&iter, &path);
+	dbus_message_iter_next(&iter);
+	config = hdp_get_config(&iter, &err);
+	if (err)
+		goto error;
+	name = dbus_message_get_sender(msg);
+	if (!name) {
+		g_set_error(&err, HDP_ERROR, HDP_UNSPECIFIED_ERROR,
+						"Can't get sender name");
+		goto error;
+	}
+
+	hdpi = g_new0(struct hdp_instance, 1);
+	hdpi->adapter = adapter;
+	hdpi->aname = g_strdup(name);
+	hdpi->apath = g_strdup(path);
+	hdpi->config = config;
+	if (!config->svc_dsc)
+		config->svc_dsc = g_strdup(HDP_SERVICE_DSC);
+	if (!config->svc_name)
+		config->svc_name = g_strdup(HDP_SERVICE_NAME);
+	if (!config->svc_prov)
+		config->svc_prov = g_strdup(HDP_SERVICE_PROVIDER);
+	hdp_set_instance_id(hdpi);
+
+	/* TODO: Create mcap instance */
+
+	/* TODO: Create SDP record if needed. */
+
+	return g_dbus_create_error(msg,
+					ERROR_INTERFACE ".HealthError",
+					"Incomplete call yet");
+error:
+	if (err) {
+		reply = g_dbus_create_error(msg,
+					ERROR_INTERFACE ".InvalidArguments",
+					"Invalid arguments: %s", err->message);
+		g_error_free(err);
+	} else
+		reply = g_dbus_create_error(msg,
+					ERROR_INTERFACE ".InvalidArguments",
+					"Invalid arguments in method call");
+	return reply;
+}
+
+static GDBusMethodTable hdp_methods[] = {
+	{ "CreateInstance", "oa{sv}", "u", hdp_create_instance },
+	{ NULL }
+};
+
+void hdp_delete_instance_iter(gpointer data, gpointer user_data)
+{
+	/* struct hdp_instance *hdpi = data; */
+
+	/* TODO: Create  a free function */
+}
+
+static void hdp_path_unregister(void *data)
+{
+	struct hdp_adapter *adapter = data;
+
+	g_slist_foreach(adapter->instances, hdp_delete_instance_iter, NULL);
+	g_slist_free(adapter->instances);
+	adapter->instances = NULL;
+	debug("All hdp instance for removed adapter were closed");
+}
+
+int hdp_adapter_register(DBusConnection *conn, struct btd_adapter *btd_adapter)
+{
+	const char *path = adapter_get_path(btd_adapter);
+
+	struct hdp_adapter *adapter;
+	adapter = g_new0(struct hdp_adapter, 1);
+
+	debug("HDP init");
+	if (!g_dbus_register_interface(conn, path, HEALTH_MANAGER_INTERFACE,
+					hdp_methods, NULL, NULL,
+					adapter, hdp_path_unregister)) {
+		error("Failed to register %s interface to %s",
+						HEALTH_MANAGER_INTERFACE, path);
+		g_free(adapter);
+		return -1;
+	}
+	adapter->conn = dbus_connection_ref(conn);
+	adapter->btd_adapter = btd_adapter_ref(btd_adapter);
+	adapters = g_slist_prepend(adapters, adapter);
+	return 0;
+}
+
+void hdp_adapter_unregister(struct btd_adapter *btd_adapter)
+{
+	struct hdp_adapter *adapter;
+
+	adapter = find_adapter(adapters, btd_adapter);
+	if (!adapter)
+		return;
+
+	g_dbus_unregister_interface(adapter->conn,
+			adapter_get_path(btd_adapter),
+			HEALTH_MANAGER_INTERFACE);
+	dbus_connection_unref(adapter->conn);
+	btd_adapter_unref(adapter->btd_adapter);
+	adapters = g_slist_remove(adapters, adapter);
+	g_free(adapter);
+
+	debug("HDP exit");
+}
diff --git a/health/hdp.h b/health/hdp.h
new file mode 100644
index 0000000..893f745
--- /dev/null
+++ b/health/hdp.h
@@ -0,0 +1,27 @@
+/*
+ *
+ *  BlueZ - Bluetooth protocol stack for Linux
+ *
+ *  Copyright (C) 2010 GSyC/LibreSoft, Universidad Rey Juan Carlos.
+ *  Authors:
+ *  Santiago Carot Nemesio <sancane at gmail.com>
+ *  Jose Antonio Santos-Cadenas <santoscadenas at gmail.com>
+ *
+ *  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.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, write to the Free Software
+ *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ *
+ */
+
+int hdp_adapter_register(DBusConnection *conn, struct btd_adapter *btd_adapter);
+void hdp_adapter_unregister(struct btd_adapter *btd_adapter);
diff --git a/health/hdp_types.h b/health/hdp_types.h
new file mode 100644
index 0000000..2db9adf
--- /dev/null
+++ b/health/hdp_types.h
@@ -0,0 +1,99 @@
+/*
+ *
+ *  BlueZ - Bluetooth protocol stack for Linux
+ *
+ *  Copyright (C) 2010 GSyC/LibreSoft, Universidad Rey Juan Carlos.
+ *  Authors:
+ *  Santiago Carot Nemesio <sancane at gmail.com>
+ *  Jose Antonio Santos-Cadenas <santoscadenas at gmail.com>
+ *
+ *  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.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, write to the Free Software
+ *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ *
+ */
+
+
+#ifndef __HDP_TYPES_H__
+#define __HDP_TYPES_H__
+
+#include <glib.h>
+#include "mcap_lib.h"
+
+#define HDP_SERVICE_NAME	"Bluez HDP"
+#define HDP_SERVICE_DSC		"A Bluez health device profile implementation"
+#define HDP_SERVICE_PROVIDER	"Bluez"
+
+#define HDP_VERSION		0x0100
+
+#define HDP_ERROR		g_quark_from_static_string("hdp-error-quark")
+
+typedef enum {
+	HDP_DIC_PARSE_ERROR,
+	HDP_DIC_ENTRY_PARSE_ERROR,
+	HDP_UNSPECIFIED_ERROR,
+	HDP_UNKNOWN_ERROR
+} HdpError;
+
+enum data_specs {
+	DATA_EXCHANGE_SPEC_11073 = 0x01
+};
+
+typedef enum {
+	HDP_SOURCE = 0x00,
+	HDP_SINK = 0x01
+} HdpRole;
+
+struct hdp_feature {
+	guint16		dtype;		/* Data type (see 5.2.9.2) */
+	gboolean	dtype_present;	/* Data type present in config */
+	char		*dscr;		/* Displayable TextName */
+};
+
+struct hdp_supp_fts {
+	guint8		mdepid;		/* (0x01-0x7F) Available for use */
+	HdpRole		role;		/* Role (see table 5.3) */
+	gboolean	role_present;	/* Role present in config */
+	GSList		*features;	/* Feature list */
+};
+
+struct hdp_config {
+	guint8		data_spec;	/* Data exchange specification */
+	GSList		*supp_fts;	/* Supported features list */
+	char		*svc_name;	/* Service name to register in SDP */
+	char		*svc_dsc;	/* Service description */
+	char		*svc_prov;	/* Service provider */
+	gboolean	ds_present;	/* Data spec has been assigned */
+};
+
+struct hdp_adapter {
+	struct btd_adapter	*btd_adapter;
+	DBusConnection		*conn;		/* DBus connection */
+	GSList			*instances;	/* HDP instances list */
+	uint16_t		ic;		/* Instances counter */
+};
+
+struct hdp_instance {
+	struct hdp_adapter	*adapter;	/* HDP adapter */
+	struct mcap_instance	*mi;		/* MCAP instance */
+	uint16_t		ccpsm;		/* Control channel psm */
+	uint16_t		dcpsm;		/* Data channel psm */
+	GSList			*hlink;		/* Health Links */
+	uint32_t		id;		/* HDP instance id */
+	char			*apath;		/* HDP agent path */
+	char			*aname;		/* HDP agent name */
+	struct hdp_config	*config;	/* Configuration */
+	uint32_t		sdp_handler;	/* SDP record handler */
+};
+
+#endif /* __HDP_TYPES_H__ */
diff --git a/health/hdp_util.c b/health/hdp_util.c
new file mode 100644
index 0000000..8f6befc
--- /dev/null
+++ b/health/hdp_util.c
@@ -0,0 +1,447 @@
+/*
+ *
+ *  BlueZ - Bluetooth protocol stack for Linux
+ *
+ *  Copyright (C) 2010 GSyC/LibreSoft, Universidad Rey Juan Carlos.
+ *  Authors:
+ *  Santiago Carot Nemesio <sancane at gmail.com>
+ *  Jose Antonio Santos-Cadenas <santoscadenas at gmail.com>
+ *
+ *  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.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, write to the Free Software
+ *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ *
+ */
+
+#include <gdbus.h>
+#include "logging.h"
+#include "hdp_types.h"
+
+typedef gboolean (*parse_item_f)(DBusMessageIter *iter, GError **err,
+							gpointer user_data);
+
+struct dict_entry_func {
+	char		*key;
+	parse_item_f	func;
+};
+
+static gboolean check_feature(struct hdp_feature *feature)
+{
+	return feature->dtype_present;
+}
+
+static gboolean check_feature_list(struct hdp_supp_fts *fts)
+{
+	return fts->role_present && (fts->features != NULL);
+}
+
+static gboolean check_config(struct hdp_config *config)
+{
+	if (config->ds_present)
+		return config->supp_fts != NULL;
+	return TRUE;
+}
+
+static gboolean hdp_check_data_spec(guint8 data_spec)
+{
+	/* Future versions may edit this function
+	 * If there are more supported data exchange specifications
+	 */
+	return data_spec == DATA_EXCHANGE_SPEC_11073;
+}
+
+static void free_feature(struct hdp_feature *feature)
+{
+	if (feature->dscr) {
+		g_free(feature->dscr);
+		feature->dscr = NULL;
+	}
+	g_free(feature);
+}
+
+static void free_feature_list(struct hdp_supp_fts *fts)
+{
+	GSList *l;
+
+	for (l = fts->features; l; l = l->next)
+		free_feature(l->data);
+	g_slist_free(fts->features);
+	fts->features = NULL;
+	g_free(fts);
+}
+
+static void free_config(struct hdp_config *config)
+{
+	GSList *l;
+
+	for (l = config->supp_fts; l; l = l->next)
+		free_feature_list(l->data);
+	g_slist_free(config->supp_fts);
+	config->supp_fts = NULL;
+	if (config->svc_dsc) {
+		g_free(config->svc_dsc);
+		config->svc_dsc = NULL;
+	}
+	if (config->svc_name) {
+		g_free(config->svc_name);
+		config->svc_name = NULL;
+	}
+	if (config->svc_prov) {
+		g_free(config->svc_prov);
+		config->svc_prov = NULL;
+	}
+	g_free(config);
+}
+
+static gboolean parse_dict_entry(struct dict_entry_func dict_context[],
+					DBusMessageIter *iter,
+					GError **err,
+					gpointer user_data)
+{
+	DBusMessageIter entry;
+	char *key;
+	int ctype, i;
+	struct dict_entry_func df;
+
+	dbus_message_iter_recurse(iter, &entry);
+	ctype = dbus_message_iter_get_arg_type(&entry);
+	if (ctype != DBUS_TYPE_STRING) {
+		g_set_error(err, HDP_ERROR, HDP_DIC_ENTRY_PARSE_ERROR,
+			"Dictionary entries should have a string as key");
+		return FALSE;
+	}
+	dbus_message_iter_get_basic(&entry, &key);
+	dbus_message_iter_next(&entry);
+	/* Find function and call it */
+	for (i = 0, df = dict_context[0]; df.key; i++, df = dict_context[i]) {
+		if (g_strcmp0(df.key, key) == 0) {
+			return df.func(&entry, err, user_data);
+		}
+	}
+	g_set_error(err, HDP_ERROR, HDP_DIC_ENTRY_PARSE_ERROR,
+			"No function found for parsing value for key %s", key);
+	return FALSE;
+}
+
+static gboolean parse_dict(struct dict_entry_func dict_context[],
+					DBusMessageIter *iter,
+					GError **err,
+					gpointer user_data)
+{
+	int ctype;
+	DBusMessageIter dict;
+
+	ctype = dbus_message_iter_get_arg_type(iter);
+	if (ctype != DBUS_TYPE_ARRAY) {
+		g_set_error(err, HDP_ERROR, HDP_DIC_PARSE_ERROR,
+					"Dictionary should be an array");
+		return FALSE;
+	}
+	dbus_message_iter_recurse(iter, &dict);
+	while ((ctype = dbus_message_iter_get_arg_type(&dict)) !=
+							DBUS_TYPE_INVALID) {
+		if (ctype != DBUS_TYPE_DICT_ENTRY) {
+			g_set_error(err, HDP_ERROR, HDP_DIC_PARSE_ERROR,
+					"Dictionary array should "
+				        "contain dict entries");
+			return FALSE;
+		}
+		/* Start parsing entry */
+		if (!parse_dict_entry(dict_context, &dict, err,
+							user_data))
+			return FALSE;
+		/* Finish entry parsing */
+		dbus_message_iter_next(&dict);
+	}
+	return TRUE;
+}
+
+static gboolean parse_description(DBusMessageIter *iter, GError **err,
+								gpointer data)
+{
+	struct hdp_feature *feat = data;
+	DBusMessageIter *string, variant;
+	int ctype;
+	const char *desc;
+
+	ctype = dbus_message_iter_get_arg_type(iter);
+	string = iter;
+	if (ctype == DBUS_TYPE_VARIANT) {
+		/* Get value inside the variable */
+		dbus_message_iter_recurse(iter, &variant);
+		ctype = dbus_message_iter_get_arg_type(&variant);
+		string = &variant;
+	}
+
+	if (ctype != DBUS_TYPE_STRING) {
+		g_set_error(err, HDP_ERROR, HDP_DIC_ENTRY_PARSE_ERROR,
+				"Value data spec should be variable or string");
+		return FALSE;
+	}
+
+	dbus_message_iter_get_basic(string, &desc);
+	feat->dscr = g_strdup(desc);
+	return TRUE;
+}
+
+static gboolean parse_data_type(DBusMessageIter *iter, GError **err,
+								gpointer data)
+{
+	struct hdp_feature *feat = data;
+	DBusMessageIter *value, variant;
+	int ctype;
+
+	ctype = dbus_message_iter_get_arg_type(iter);
+	value = iter;
+	if (ctype == DBUS_TYPE_VARIANT) {
+		/* Get value inside the variable */
+		dbus_message_iter_recurse(iter, &variant);
+		ctype = dbus_message_iter_get_arg_type(&variant);
+		value = &variant;
+	}
+
+	if (ctype != DBUS_TYPE_UINT16) {
+		g_set_error(err, HDP_ERROR, HDP_DIC_ENTRY_PARSE_ERROR,
+			"Final value for data type should be a uint16");
+		return FALSE;
+	}
+	dbus_message_iter_get_basic(value, &feat->dtype);
+
+	/*
+	 * This data should be check by the application layer because it
+	 * depends on the data_spec values and is specific for each value
+	 */
+	feat->dtype_present = TRUE;
+
+	return TRUE;
+}
+
+static struct dict_entry_func specs_context[] = {
+	{"data_type", parse_data_type},
+	{"description", parse_description},
+	{NULL, NULL}
+};
+
+static gboolean parse_specs(DBusMessageIter *iter, GError **err, gpointer data)
+{
+	struct hdp_supp_fts *fts = data;
+	struct hdp_feature *feature = NULL;
+	DBusMessageIter *array, value, dict;
+	int ctype;
+
+	ctype = dbus_message_iter_get_arg_type(iter);
+	array = iter;
+	if (ctype == DBUS_TYPE_VARIANT) {
+		/* Get value inside the variable */
+		dbus_message_iter_recurse(iter, &value);
+		ctype = dbus_message_iter_get_arg_type(&value);
+		array = &value;
+	}
+	if (ctype != DBUS_TYPE_ARRAY) {
+		g_set_error(err, HDP_ERROR, HDP_DIC_ENTRY_PARSE_ERROR,
+				"Value specs should be variable or array");
+		return FALSE;
+	}
+	dbus_message_iter_recurse(array, &dict);
+	while ((ctype = dbus_message_iter_get_arg_type(&dict)) !=
+							DBUS_TYPE_INVALID){
+		feature = g_new0(struct hdp_feature, 1);
+		if (!parse_dict(specs_context, &dict, err, feature))
+			goto error;
+		if (!check_feature(feature)) {
+			g_set_error(err, HDP_ERROR, HDP_DIC_ENTRY_PARSE_ERROR,
+					"Field \"data_type\" is mandatory");
+			goto error;
+		}
+		fts->features = g_slist_append(fts->features, feature);
+		dbus_message_iter_next(&dict);
+	}
+
+	return TRUE;
+error:
+	if (feature)
+		free_feature(feature);
+	return FALSE;
+}
+
+static gboolean parse_role(DBusMessageIter *iter, GError **err, gpointer data)
+{
+	struct hdp_supp_fts *fts = data;
+	DBusMessageIter value;
+	DBusMessageIter *string;
+	int ctype;
+	const char *role;
+
+	ctype = dbus_message_iter_get_arg_type(iter);
+	string = iter;
+	if (ctype == DBUS_TYPE_VARIANT) {
+		/* Get value inside the variable */
+		dbus_message_iter_recurse(iter, &value);
+		ctype = dbus_message_iter_get_arg_type(&value);
+		string = &value;
+	}
+
+	if (ctype != DBUS_TYPE_STRING) {
+		g_set_error(err, HDP_ERROR, HDP_UNSPECIFIED_ERROR,
+				"Value data spec should be variable or string");
+		return FALSE;
+	}
+
+	dbus_message_iter_get_basic(string, &role);
+	if (g_strcmp0(role, "sink") == 0)
+		fts->role = HDP_SINK;
+	else if (g_strcmp0(role, "source") == 0)
+		fts->role = HDP_SOURCE;
+	else {
+		g_set_error(err, HDP_ERROR, HDP_UNSPECIFIED_ERROR,
+			"Role value should be \"source\" or \"sink\"");
+		return FALSE;
+	}
+	fts->role_present = TRUE;
+	return TRUE;
+}
+
+static struct dict_entry_func end_point_context[] = {
+	{"role", parse_role},
+	{"specs", parse_specs},
+	{NULL, NULL}
+};
+
+static gboolean parse_end_points(DBusMessageIter *iter, GError **err,
+								gpointer data)
+{
+	struct hdp_config *config = data;
+	struct hdp_supp_fts *fts = NULL;
+	DBusMessageIter array, dict;
+	int ctype;
+
+	ctype = dbus_message_iter_get_arg_type(iter);
+	if (ctype != DBUS_TYPE_VARIANT) {
+		g_set_error(err, HDP_ERROR, HDP_UNSPECIFIED_ERROR,
+				"Value for end points should be variable");
+		return FALSE;
+	}
+	dbus_message_iter_recurse(iter, &array);
+	ctype = dbus_message_iter_get_arg_type(&array);
+	if (ctype != DBUS_TYPE_ARRAY) {
+		g_set_error(err, HDP_ERROR, HDP_UNSPECIFIED_ERROR,
+			"Value end_point should be array inside variable");
+		return FALSE;
+	}
+
+	dbus_message_iter_recurse(&array, &dict);
+	while ((ctype = dbus_message_iter_get_arg_type(&dict)) !=
+							DBUS_TYPE_INVALID){
+
+		if (ctype != DBUS_TYPE_ARRAY) {
+			g_set_error(err, HDP_ERROR, HDP_DIC_PARSE_ERROR,
+					"Dictionary should be an array");
+			return FALSE;
+		}
+		fts = g_new0(struct hdp_supp_fts, 1);
+		if (!parse_dict(end_point_context, &dict, err, fts))
+			goto error;
+		if (!check_feature_list(fts)) {
+			g_set_error(err, HDP_ERROR, HDP_DIC_PARSE_ERROR,
+					"Role field and specs are mandatory");
+			goto error;
+		}
+		config->supp_fts = g_slist_append(config->supp_fts, fts);
+		dbus_message_iter_next(&dict);
+	}
+	return TRUE;
+error:
+	if (fts)
+		free_feature_list(fts);
+	return FALSE;
+}
+
+static gboolean parse_data_spec(DBusMessageIter *iter, GError **err,
+								gpointer data)
+{
+	struct hdp_config *config = data;
+	DBusMessageIter value;
+	int ctype;
+
+	ctype = dbus_message_iter_get_arg_type(iter);
+	if (ctype != DBUS_TYPE_VARIANT) {
+		g_set_error(err, HDP_ERROR, HDP_UNSPECIFIED_ERROR,
+					"Value data spec should be variable");
+		return FALSE;
+	}
+	dbus_message_iter_recurse(iter, &value);
+	ctype = dbus_message_iter_get_arg_type(&value);
+	if (ctype != DBUS_TYPE_BYTE) {
+		g_set_error(err, HDP_ERROR, HDP_UNSPECIFIED_ERROR,
+				"Final value data spec should be byte");
+		return FALSE;
+	}
+
+	dbus_message_iter_get_basic(&value, &config->data_spec);
+
+	if (!hdp_check_data_spec(config->data_spec))
+		return FALSE;
+	config->ds_present = TRUE;
+	return TRUE;
+}
+
+static struct dict_entry_func main_context[] = {
+	{"data_spec", parse_data_spec},
+	{"end_points", parse_end_points},
+	{NULL, NULL}
+};
+
+static void print_feature(gpointer elem, gpointer data)
+{
+	struct hdp_feature *feat = elem;
+
+	debug("   Feature:");
+	debug("      description: %s", feat->dscr);
+	debug("      data type: %u", feat->dtype);
+}
+
+static void print_features(gpointer elem, gpointer data)
+{
+	struct hdp_supp_fts *fts = elem;
+
+	debug("Mdep:");
+	debug("   mdepid %u", fts->mdepid);
+	debug("   role %d", fts->role);
+	g_slist_foreach(fts->features, print_feature, NULL);
+}
+
+struct hdp_config *hdp_get_config(DBusMessageIter *iter, GError **err)
+{
+	struct hdp_config *config;
+
+	config = g_new0(struct hdp_config, 1);
+
+	if (!parse_dict(main_context, iter, err, config))
+		goto error;
+
+	/* TODO check config */
+	if (!check_config(config)) {
+		g_set_error(err, HDP_ERROR, HDP_DIC_ENTRY_PARSE_ERROR,
+			"\"data_spec\" and \"end_point\" should be set or not");
+		goto error;
+	}
+	if (!config->ds_present)
+		goto error;
+	debug("config->data_spec %d", config->data_spec);
+	g_slist_foreach(config->supp_fts, print_features, NULL);
+	return config;
+error:
+	if (config)
+		free_config(config);
+	return NULL;
+}
diff --git a/health/hdp_util.h b/health/hdp_util.h
new file mode 100644
index 0000000..f09e9a6
--- /dev/null
+++ b/health/hdp_util.h
@@ -0,0 +1,34 @@
+/*
+ *
+ *  BlueZ - Bluetooth protocol stack for Linux
+ *
+ *  Copyright (C) 2010 GSyC/LibreSoft, Universidad Rey Juan Carlos.
+ *  Authors:
+ *  Santiago Carot Nemesio <sancane at gmail.com>
+ *  Jose Antonio Santos-Cadenas <santoscadenas at gmail.com>
+ *
+ *  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.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, write to the Free Software
+ *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ *
+ */
+
+#ifndef __HDP_UTIL_H__
+#define __HDP_UTIL_H__
+
+#include <gdbus.h>
+#include "hdp_types.h"
+
+struct hdp_config *hdp_get_config(DBusMessageIter *iter, GError **err);
+
+#endif /* __HDP_UTIL_H__ */
diff --git a/health/main.c b/health/main.c
new file mode 100644
index 0000000..6ece69b
--- /dev/null
+++ b/health/main.c
@@ -0,0 +1,60 @@
+/*
+ *
+ *  BlueZ - Bluetooth protocol stack for Linux
+ *
+ *  Copyright (C) 2010 GSyC/LibreSoft, Universidad Rey Juan Carlos.
+ *  Authors:
+ *  Santiago Carot Nemesio <sancane at gmail.com>
+ *  Jose Antonio Santos-Cadenas <santoscadenas at gmail.com>
+ *
+ *  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.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, write to the Free Software
+ *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <errno.h>
+
+#include <gdbus.h>
+
+#include "plugin.h"
+#include "manager.h"
+
+static DBusConnection *connection;
+
+static int hdp_init(void)
+{
+	connection = dbus_bus_get(DBUS_BUS_SYSTEM, NULL);
+	if (connection == NULL)
+		return -EIO;
+
+	if (hdp_manager_init(connection) < 0) {
+		dbus_connection_unref(connection);
+		return -EIO;
+	}
+	return 0;
+}
+
+static void hdp_exit(void)
+{
+	hdp_manager_exit();
+
+	dbus_connection_unref(connection);
+}
+
+BLUETOOTH_PLUGIN_DEFINE(health, VERSION,
+			BLUETOOTH_PLUGIN_PRIORITY_DEFAULT, hdp_init, hdp_exit)
diff --git a/health/manager.c b/health/manager.c
new file mode 100644
index 0000000..b5dec5a
--- /dev/null
+++ b/health/manager.c
@@ -0,0 +1,101 @@
+/*
+ *
+ *  BlueZ - Bluetooth protocol stack for Linux
+ *
+ *  Copyright (C) 2010 GSyC/LibreSoft, Universidad Rey Juan Carlos.
+ *  Authors:
+ *  Santiago Carot Nemesio <sancane at gmail.com>
+ *  Jose Antonio Santos-Cadenas <santoscadenas at gmail.com>
+ *
+ *  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.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, write to the Free Software
+ *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <bluetooth/sdp.h>
+
+#include <glib.h>
+#include <gdbus.h>
+
+#include "adapter.h"
+#include "device.h"
+
+#include "logging.h"
+#include "manager.h"
+#include "hdp.h"
+
+#define HDP_UUID		"00001400-0000-1000-8000-00805F9B34FB"
+#define HDP_SOURCE_UUID		"00001401-0000-1000-8000-00805F9B34FB"
+#define HDP_SINK_UUID		"00001402-0000-1000-8000-00805F9B34FB"
+
+static DBusConnection *connection = NULL;
+
+static int hdp_adapter_probe(struct btd_adapter *adapter)
+{
+	return hdp_adapter_register(connection, adapter);
+}
+
+static void hdp_adapter_remove(struct btd_adapter *adapter)
+{
+	hdp_adapter_unregister(adapter);
+}
+
+static struct btd_adapter_driver hdp_adapter_driver = {
+	.name	= "hdp-adapter-driver",
+	.probe	= hdp_adapter_probe,
+	.remove	= hdp_adapter_remove,
+};
+
+static int hdp_driver_probe(struct btd_device *device, GSList *uuids)
+{
+	debug("hdp driver probe");
+	return 0;
+}
+
+static void hdp_driver_remove(struct btd_device *device)
+{
+	debug("hdp driver remove");
+}
+
+static struct btd_device_driver hdp_device_driver = {
+	.name	= "hdp_device-driver",
+	.uuids	= BTD_UUIDS(HDP_UUID, HDP_SOURCE_UUID, HDP_SINK_UUID),
+	.probe	= hdp_driver_probe,
+	.remove	= hdp_driver_remove,
+};
+
+int hdp_manager_init(DBusConnection *conn)
+{
+	connection = dbus_connection_ref(conn);
+
+	btd_register_adapter_driver(&hdp_adapter_driver);
+	btd_register_device_driver(&hdp_device_driver);
+
+	debug("hdp manager init");
+	return 0;
+}
+
+void hdp_manager_exit(void)
+{
+	btd_unregister_device_driver(&hdp_device_driver);
+	btd_unregister_adapter_driver(&hdp_adapter_driver);
+
+	dbus_connection_unref(connection);
+	connection = NULL;
+	debug("hdp manager exit");
+}
diff --git a/health/manager.h b/health/manager.h
new file mode 100644
index 0000000..b91ef75
--- /dev/null
+++ b/health/manager.h
@@ -0,0 +1,27 @@
+/*
+ *
+ *  BlueZ - Bluetooth protocol stack for Linux
+ *
+ *  Copyright (C) 2010 GSyC/LibreSoft, Universidad Rey Juan Carlos.
+ *  Authors:
+ *  Santiago Carot Nemesio <sancane at gmail.com>
+ *  Jose Antonio Santos-Cadenas <santoscadenas at gmail.com>
+ *
+ *  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.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, write to the Free Software
+ *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ *
+ */
+
+int hdp_manager_init(DBusConnection *conn);
+void hdp_manager_exit(void);
-- 
1.6.3.3

--
To unsubscribe from this list: send the line "unsubscribe linux-bluetooth" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [Bluez Devel]     [Linux Wireless Networking]     [Linux Wireless Personal Area Networking]     [Linux ATH6KL]     [Linux USB Devel]     [Linux Media Drivers]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Big List of Linux Books]

  Powered by Linux