[PATCH v14 14/14] audio: Add fast connectable to telephony interface

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

 



---
 audio/telephony.c |  146 ++++++++++++++++++++++++++++++++++++++++++++++++-----
 doc/audio-api.txt |   26 ++++++++++
 2 files changed, 160 insertions(+), 12 deletions(-)

diff --git a/audio/telephony.c b/audio/telephony.c
index eb9faeb..605b09a 100644
--- a/audio/telephony.c
+++ b/audio/telephony.c
@@ -64,6 +64,11 @@
 #define DEFAULT_HF_AG_CHANNEL 13
 #define DEFAULT_SAP_GW_CHANNEL 8
 
+struct tel_adapter {
+	struct btd_adapter *btd_adapter;
+	gboolean fast_connectable;
+};
+
 struct tel_device {
 	void			*device;
 	struct btd_device	*btd_dev;
@@ -91,7 +96,7 @@ struct default_agent {
 };
 
 struct tel_agent {
-	struct btd_adapter	*adapter;
+	struct tel_adapter	*adapter;
 	char			*name;		/* agent DBus bus id */
 	char			*path;		/* agent object path */
 	guint			watch;		/* agent disconnect watcher */
@@ -123,7 +128,7 @@ static struct tel_agent *find_agent(struct btd_adapter *adapter,
 	for (l = agents; l; l = l->next) {
 		struct tel_agent *agent = l->data;
 
-		if (agent->adapter != adapter)
+		if (agent->adapter->btd_adapter != adapter)
 			continue;
 
 		if (sender && g_strcmp0(agent->name, sender) != 0)
@@ -143,8 +148,19 @@ static struct tel_agent *find_agent(struct btd_adapter *adapter,
 
 static void free_agent(struct tel_agent *agent)
 {
+	struct btd_adapter *adapter = agent->adapter->btd_adapter;
 	DBusMessage *msg;
 
+	/* if there is no more agent for this adapter and it has been set to
+	 * fast connectable mode, reset it to FALSE
+	 */
+	if (find_agent(adapter, NULL, NULL, NULL) == NULL &&
+			agent->adapter->fast_connectable) {
+		if (btd_adapter_set_fast_connectable(adapter, FALSE))
+			error("Changing fast connectable for hci%d failed",
+				adapter_get_dev_id(adapter));
+	}
+
 	if (agent->record_id)
 		remove_record_from_server(agent->record_id);
 
@@ -163,7 +179,6 @@ static void free_agent(struct tel_agent *agent)
 		agent->watch = 0;
 	}
 
-	btd_adapter_unref(agent->adapter);
 	g_free(agent->name);
 	g_free(agent->path);
 	g_free(agent);
@@ -1232,7 +1247,7 @@ static void agent_disconnect_cb(DBusConnection *conn, void *user_data)
 	free_agent(agent);
 }
 
-static struct tel_agent *agent_new(struct btd_adapter *adapter,
+static struct tel_agent *agent_new(struct tel_adapter *adapter,
 					const char *sender, const char *path,
 					const char *uuid, uint16_t version,
 					uint16_t features)
@@ -1244,7 +1259,7 @@ static struct tel_agent *agent_new(struct btd_adapter *adapter,
 				sizeof(struct default_agent) ; i++) {
 		if (strcasecmp(uuid, default_properties[i].uuid) == 0) {
 			agent = g_new0(struct tel_agent, 1);
-			agent->adapter = btd_adapter_ref(adapter);
+			agent->adapter = adapter;
 			agent->properties = &default_properties[i];
 			agent->name = g_strdup(sender);
 			agent->path = g_strdup(path);
@@ -1260,7 +1275,8 @@ static struct tel_agent *agent_new(struct btd_adapter *adapter,
 static DBusMessage *register_agent(DBusConnection *conn,
 					DBusMessage *msg, void *data)
 {
-	struct btd_adapter *adapter = data;
+	struct tel_adapter *tel_adapter = data;
+	struct btd_adapter *adapter = tel_adapter->btd_adapter;
 	DBusMessageIter args, props;
 	const char *sender, *path, *uuid;
 	uint16_t version = 0;
@@ -1292,7 +1308,7 @@ static DBusMessage *register_agent(DBusConnection *conn,
 		return btd_error_already_exists(msg);
 
 	/* initialize agent properties */
-	agent = agent_new(adapter, sender, path, uuid, version, features);
+	agent = agent_new(tel_adapter, sender, path, uuid, version, features);
 	if (agent == NULL)
 		return btd_error_invalid_args(msg);
 
@@ -1346,7 +1362,8 @@ static DBusMessage *register_agent(DBusConnection *conn,
 static DBusMessage *unregister_agent(DBusConnection *conn,
 				DBusMessage *msg, void *data)
 {
-	struct btd_adapter *adapter = data;
+	struct tel_adapter *tel_adapter = data;
+	struct btd_adapter *adapter = tel_adapter->btd_adapter;
 	const char *sender, *path;
 	struct tel_agent *agent;
 
@@ -1370,35 +1387,140 @@ static DBusMessage *unregister_agent(DBusConnection *conn,
 	return g_dbus_create_reply(msg, DBUS_TYPE_INVALID);
 }
 
+static DBusMessage *telephony_get_properties(DBusConnection *conn,
+					DBusMessage *msg, void *data)
+{
+	struct tel_adapter *tel_adapter = data;
+	DBusMessage *reply;
+	DBusMessageIter iter;
+	DBusMessageIter dict;
+
+	reply = dbus_message_new_method_return(msg);
+	if (!reply)
+		return NULL;
+
+	dbus_message_iter_init_append(reply, &iter);
+
+	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);
+
+	/* FastConnectable */
+	dict_append_entry(&dict, "FastConnectable", DBUS_TYPE_BOOLEAN,
+			  &tel_adapter->fast_connectable);
+
+	dbus_message_iter_close_container(&iter, &dict);
+
+	return reply;
+}
+
+static DBusMessage *telephony_set_property(DBusConnection *conn,
+					DBusMessage *msg, void *data)
+{
+	struct tel_adapter *tel_adapter = data;
+	struct btd_adapter *adapter = tel_adapter->btd_adapter;
+	const char *property;
+	DBusMessageIter iter;
+	DBusMessageIter sub;
+	gboolean enable;
+	DBusMessage *reply;
+
+	if (!dbus_message_iter_init(msg, &iter))
+		return btd_error_invalid_args(msg);
+
+	if (dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_STRING)
+		return btd_error_invalid_args(msg);
+
+	dbus_message_iter_get_basic(&iter, &property);
+	dbus_message_iter_next(&iter);
+
+	if (dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_VARIANT)
+		return btd_error_invalid_args(msg);
+	dbus_message_iter_recurse(&iter, &sub);
+
+	if (g_str_equal("FastConnectable", property)) {
+		if (dbus_message_iter_get_arg_type(&sub) !=
+				DBUS_TYPE_BOOLEAN)
+			return btd_error_invalid_args(msg);
+
+		dbus_message_iter_get_basic(&sub, &enable);
+
+		reply = dbus_message_new_method_return(msg);
+		if (!reply)
+			return NULL;
+
+		if (btd_adapter_set_fast_connectable(adapter, enable)) {
+			error("Changing fast connectable for hci%d failed",
+				adapter_get_dev_id(adapter));
+			dbus_message_unref(reply);
+			return btd_error_failed(msg,
+					"Changing to fast connectable failed");
+		}
+
+		tel_adapter->fast_connectable = enable;
+		emit_property_changed(conn, adapter_get_path(adapter),
+					AUDIO_TELEPHONY_INTERFACE,
+					"FastConnectable",
+					DBUS_TYPE_BOOLEAN, &enable);
+		return reply;
+	}
+
+	return btd_error_invalid_args(msg);
+}
+
 static const GDBusMethodTable telsrv_methods[] = {
 	{ GDBUS_METHOD("RegisterAgent",
 			GDBUS_ARGS({ "agent", "o" }, { "properties", "a{sv}" }),
 			NULL, register_agent) },
 	{ GDBUS_METHOD("UnregisterAgent",
 			GDBUS_ARGS({ "agent", "o" }), NULL, unregister_agent) },
+	{ GDBUS_METHOD("GetProperties",
+			NULL, GDBUS_ARGS({ "properties", "a{sv}" }),
+			telephony_get_properties) },
+	{ GDBUS_METHOD("SetProperty",
+			GDBUS_ARGS({ "name", "s" }, { "value", "v" }), NULL,
+			telephony_set_property) },
+	{ }
+};
+
+static const GDBusSignalTable telsrv_signals[] = {
+	{ GDBUS_SIGNAL("PropertyChanged",
+			GDBUS_ARGS({ "name", "s" }, { "value", "v" })) },
 	{ }
 };
 
 static void path_unregister(void *data)
 {
+	struct tel_adapter *tel_adapter = data;
+
+	btd_adapter_unref(tel_adapter->btd_adapter);
+	g_free(tel_adapter);
+
 	DBG("Unregistered interface %s", AUDIO_TELEPHONY_INTERFACE);
 }
 
 static int register_interface(struct btd_adapter *adapter)
 {
 	const char *path;
+	struct tel_adapter *tel_adapter;
 
 	if (DBUS_TYPE_UNIX_FD < 0)
 		return -1;
 
 	path = adapter_get_path(adapter);
 
+	tel_adapter = g_new0(struct tel_adapter, 1);
+	tel_adapter->btd_adapter = btd_adapter_ref(adapter);
+
 	if (!g_dbus_register_interface(connection, path,
 					AUDIO_TELEPHONY_INTERFACE,
-					telsrv_methods, NULL,
-					NULL, adapter, path_unregister)) {
+					telsrv_methods, telsrv_signals,
+					NULL, tel_adapter, path_unregister)) {
 		error("D-Bus failed to register %s interface",
 				AUDIO_TELEPHONY_INTERFACE);
+		btd_adapter_unref(tel_adapter->btd_adapter);
+		g_free(tel_adapter);
 		return -1;
 	}
 
@@ -1426,12 +1548,12 @@ void telephony_adapter_exit(struct btd_adapter *adapter)
 
 	DBG("adapter: %p", adapter);
 
-	unregister_interface(adapter);
-
 	while ((agent = find_agent(adapter, NULL, NULL, NULL)) != NULL) {
 		agents = g_slist_remove(agents, agent);
 		free_agent(agent);
 	}
+
+	unregister_interface(adapter);
 }
 
 int telephony_init(void)
diff --git a/doc/audio-api.txt b/doc/audio-api.txt
index 8f4dcb2..abbd680 100644
--- a/doc/audio-api.txt
+++ b/doc/audio-api.txt
@@ -334,6 +334,32 @@ Methods		void RegisterAgent(object path, dict properties)
 
 			Unregister sender agent.
 
+		dict GetProperties()
+
+			Returns all properties for the interface. See the
+			properties section for available properties.
+
+			Possible Errors: org.bluez.Error.InvalidArguments
+
+		void SetProperty(string name, variant value)
+
+			Changes the value of the specified property. Only
+			properties that are listed as read-write are changeable.
+			On success this will emit a PropertyChanged signal.
+
+			Possible Errors: org.bluez.Error.DoesNotExist
+					 org.bluez.Error.InvalidArguments
+
+Signals		void PropertyChanged(string name, variant value)
+
+			This signal indicates a changed value of the given
+			property.
+
+Properties	boolean FastConnectable  [readwrite]
+
+			Indicates if there is adapter is in fast connectable
+			mode.
+
 TelephonyAgent hierarchy
 ========================
 
-- 
1.7.9.5

--
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