diff options
| author | Marcel Holtmann <marcel@holtmann.org> | 2006-02-12 05:49:17 +0000 | 
|---|---|---|
| committer | Marcel Holtmann <marcel@holtmann.org> | 2006-02-12 05:49:17 +0000 | 
| commit | 47981636cf44293edb866e8565dcf812d2a3e8fb (patch) | |
| tree | 6e341fff4e8b2e9f39f869d196e4af3808a2ab96 /hcid/dbus-manager.c | |
| parent | 96449478703013c0028864afc1ec544b6a098f65 (diff) | |
Split D-Bus services and errors into separate files
Diffstat (limited to 'hcid/dbus-manager.c')
| -rw-r--r-- | hcid/dbus-manager.c | 183 | 
1 files changed, 183 insertions, 0 deletions
diff --git a/hcid/dbus-manager.c b/hcid/dbus-manager.c new file mode 100644 index 00000000..dfb00022 --- /dev/null +++ b/hcid/dbus-manager.c @@ -0,0 +1,183 @@ +/* + * + *  BlueZ - Bluetooth protocol stack for Linux + * + *  Copyright (C) 2004-2006  Marcel Holtmann <marcel@holtmann.org> + * + * + *  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 <stdio.h> +#include <errno.h> +#include <unistd.h> +#include <sys/socket.h> + +#include <bluetooth/bluetooth.h> +#include <bluetooth/hci.h> +#include <bluetooth/hci_lib.h> + +#include <dbus/dbus.h> + +#include "hcid.h" +#include "dbus.h" + +static DBusMessage* handle_mgr_device_list_req(DBusMessage *msg, void *data) +{ +	DBusMessageIter iter; +	DBusMessageIter array_iter; +	DBusMessage *reply; +	struct hci_dev_list_req *dl; +	struct hci_dev_req *dr; +	int i, sk = -1; + +	sk = socket(AF_BLUETOOTH, SOCK_RAW, BTPROTO_HCI); +	if (sk < 0) { +		syslog(LOG_ERR, "Can't open HCI socket: %s (%d)", strerror(errno), errno); +		return bluez_new_failure_msg(msg, BLUEZ_ESYSTEM_OFFSET + errno); +	} + +	dl = malloc(HCI_MAX_DEV * sizeof(*dr) + sizeof(*dl)); +	if (!dl) { +		syslog(LOG_ERR, "Can't allocate memory"); +		close(sk); +		return bluez_new_failure_msg(msg, BLUEZ_EDBUS_NO_MEM); +	} + +	dl->dev_num = HCI_MAX_DEV; +	dr = dl->dev_req; + +	if (ioctl(sk, HCIGETDEVLIST, dl) < 0) { +		reply = bluez_new_failure_msg(msg, BLUEZ_ESYSTEM_OFFSET + errno); +		goto failed; +	} + +	dr = dl->dev_req; + +	reply = dbus_message_new_method_return(msg); +	if (reply == NULL) { +		syslog(LOG_ERR, "Out of memory while calling dbus_message_new_method_return"); +		goto failed; +	} + +	dbus_message_iter_init_append(reply, &iter); + +	dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, +				DBUS_TYPE_STRING_AS_STRING, &array_iter); + +	for (i = 0; i < dl->dev_num; i++, dr++) { +		char path[MAX_PATH_LENGTH], *path_ptr = path; +		struct hci_dev_info di; + +		memset(&di, 0 , sizeof(struct hci_dev_info)); +		di.dev_id = dr->dev_id; + +		if (ioctl(sk, HCIGETDEVINFO, &di) < 0) +			continue; + +		snprintf(path, sizeof(path), "%s/%s", DEVICE_PATH, di.name); + +		dbus_message_iter_append_basic(&array_iter, +						DBUS_TYPE_STRING, &path_ptr); +	} + +	dbus_message_iter_close_container(&iter, &array_iter); + +failed: +	free(dl); + +	close(sk); + +	return reply; +} + +static DBusMessage* handle_mgr_default_device_req(DBusMessage *msg, void *data) +{ +	DBusMessage *reply; +	char path[MAX_PATH_LENGTH], *path_ptr = path; +	int default_dev = get_default_dev_id(); + +	if (default_dev < 0) +		return bluez_new_failure_msg(msg, BLUEZ_ESYSTEM_ENODEV); + +	reply = dbus_message_new_method_return(msg); +	if (!reply) { +		syslog(LOG_ERR, "Out of memory while calling dbus_message_new_method_return"); +		return reply; +	} + +	snprintf(path, sizeof(path), "%s/hci%d", DEVICE_PATH, default_dev); + +	dbus_message_append_args(reply, DBUS_TYPE_STRING, &path_ptr, +					DBUS_TYPE_INVALID); + +	return reply; +} + +static const struct service_data mgr_services[] = { +	{ MGR_DEVICE_LIST,	handle_mgr_device_list_req,		MGR_DEVICE_LIST_SIGNATURE	}, +	{ MGR_DEFAULT_DEVICE,	handle_mgr_default_device_req,		MGR_DEFAULT_DEVICE_SIGNATURE	}, +	{ NULL, NULL, NULL } +}; + +DBusHandlerResult msg_func_manager(DBusConnection *conn, DBusMessage *msg, void *data) +{ +	const struct service_data *handlers; +	DBusMessage *reply = NULL; +	const char *iface; +	const char *method; +	const char *signature; +	uint32_t error = BLUEZ_EDBUS_UNKNOWN_METHOD; +	DBusHandlerResult ret = DBUS_HANDLER_RESULT_NOT_YET_HANDLED; + +	iface = dbus_message_get_interface(msg); +	method = dbus_message_get_member(msg); +	signature = dbus_message_get_signature(msg); + +	syslog(LOG_INFO, "[%s,%d] path:%s, method:%s", __PRETTY_FUNCTION__, __LINE__, dbus_message_get_path(msg), method); + +	if (strcmp(iface, MANAGER_INTERFACE) != 0) +		return ret; + +	for (handlers = mgr_services; handlers->name != NULL; handlers++) { +		if (strcmp(handlers->name, method)) +			continue; + +		if (strcmp(handlers->signature, signature) != 0) +			error = BLUEZ_EDBUS_WRONG_SIGNATURE; +		else { +			reply = handlers->handler_func(msg, data); +			error = 0; +		} + +		ret = DBUS_HANDLER_RESULT_HANDLED; +	} + +	if (error) +		reply = bluez_new_failure_msg(msg, error); + +	if (reply) { +		if (!dbus_connection_send (conn, reply, NULL)) +			syslog(LOG_ERR, "Can't send reply message!"); +		dbus_message_unref(reply); +	} + +	return ret; +}  | 
