/* * * BlueZ - Bluetooth protocol stack for Linux * * Copyright (C) 2004-2007 Marcel Holtmann * * * 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 #endif #include #include #include #include #include #include #include #include #include #include #include #include "dbus.h" #include "dbus-helper.h" #include "logging.h" #include "textfile.h" #define NETWORK_PATH "/org/bluez/network" #define NETWORK_MANAGER_INTERFACE "org.bluez.network.Manager" #include "error.h" #include "bridge.h" #include "manager.h" #include "server.h" #include "connection.h" #include "common.h" #define MAX_PATH_LENGTH 64 /* D-Bus path */ struct pending_reply { DBusConnection *conn; DBusMessage *msg; bdaddr_t src; /* Source address */ bdaddr_t dst; /* Destination address */ char *addr; /* Destination address */ char *path; /* D-Bus object path */ char *adapter_path; /* Default adapter path */ uint16_t id; /* Role */ }; static GSList *server_paths = NULL; /* Network registered servers paths */ static GSList *connection_paths = NULL; /* Network registered connections paths */ static DBusConnection *connection = NULL; static int net_uid = 0; /* Network objects identifier */ static void pending_reply_free(struct pending_reply *pr) { if (pr->addr) g_free(pr->addr); if (pr->path) g_free(pr->path); if (pr->adapter_path) g_free(pr->adapter_path); if (pr->msg) dbus_message_unref(pr->msg); if (pr->conn) dbus_connection_unref(pr->conn); } static DBusHandlerResult create_path(DBusConnection *conn, DBusMessage *msg, const char *path, const char *sname) { DBusMessage *reply; /* emit signal when it is a new path */ if (sname) { dbus_connection_emit_signal(conn, NETWORK_PATH, NETWORK_MANAGER_INTERFACE, sname, DBUS_TYPE_STRING, &path, DBUS_TYPE_INVALID); } reply = dbus_message_new_method_return(msg); if (!reply) return DBUS_HANDLER_RESULT_NEED_MEMORY; dbus_message_append_args(reply, DBUS_TYPE_STRING, &path, DBUS_TYPE_INVALID); return send_message_and_unref(conn, reply); } static DBusHandlerResult list_paths(DBusConnection *conn, DBusMessage *msg, GSList *list) { DBusMessage *reply; DBusMessageIter iter; DBusMessageIter array_iter; reply = dbus_message_new_method_return(msg); if (!reply) return DBUS_HANDLER_RESULT_NEED_MEMORY; dbus_message_iter_init_append(reply, &iter); dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, DBUS_TYPE_STRING_AS_STRING, &array_iter); for (; list; list = list->next) { dbus_message_iter_append_basic(&array_iter, DBUS_TYPE_STRING, &list->data); } dbus_message_iter_close_container(&iter, &array_iter); return send_message_and_unref(conn, reply); } static DBusHandlerResult remove_path(DBusConnection *conn, DBusMessage *msg, GSList **list, const char *sname) { const char *path; DBusMessage *reply; DBusError derr; GSList *l; dbus_error_init(&derr); if (!dbus_message_get_args(msg, &derr, DBUS_TYPE_STRING, &path, DBUS_TYPE_INVALID)) { err_invalid_args(conn, msg, derr.message); dbus_error_free(&derr); return DBUS_HANDLER_RESULT_HANDLED; } l = g_slist_find_custom(*list, path, (GCompareFunc) strcmp); if (!l) return err_does_not_exist(conn, msg, "Path doesn't exist"); if (*list == connection_paths && connection_has_pending (conn, path)) return err_failed(conn, msg, "Connection is Busy"); g_free(l->data); *list = g_slist_remove(*list, l->data); reply = dbus_message_new_method_return(msg); if (!reply) return DBUS_HANDLER_RESULT_NEED_MEMORY; /* Remove the nap or gn file from the file system */ server_remove_stored(conn, path); if (!dbus_connection_destroy_object_path(conn, path)) error("Network path unregister failed"); dbus_connection_emit_signal(conn, NETWORK_PATH, NETWORK_MANAGER_INTERFACE, sname, DBUS_TYPE_STRING, &path, DBUS_TYPE_INVALID); return send_message_and_unref(conn, reply); } static void pan_record_reply(DBusPendingCall *call, void *data) { struct pending_reply *pr = data; DBusMessage *reply = dbus_pending_call_steal_reply(call); DBusError derr; int len, scanned; uint8_t *rec_bin; sdp_data_t *d; sdp_record_t *rec = NULL; char *name = NULL, *desc = NULL; dbus_error_init(&derr); if (dbus_set_error_from_message(&derr, reply)) { if (dbus_error_has_name(&derr, "org.bluez.Error.ConnectionAttemptFailed")) err_connection_failed(pr->conn, pr->msg, derr.message); else err_not_supported(pr->conn, pr->msg); error("GetRemoteServiceRecord failed: %s(%s)", derr.name, derr.message); goto fail; } if (!dbus_message_get_args(reply, &derr, DBUS_TYPE_ARRAY, DBUS_TYPE_BYTE, &rec_bin, &len, DBUS_TYPE_INVALID)) { err_not_supported(pr->conn, pr->msg); error("%s: %s", derr.name, derr.message); goto fail; } if (len == 0) { err_not_supported(pr->conn, pr->msg); error("Invalid PAN service record length"); goto fail; } rec = sdp_extract_pdu(rec_bin, &scanned); /* Extract service name from record */ d = sdp_data_get(rec, SDP_ATTR_SVCNAME_PRIMARY); if (d) { name = g_new0(char, d->unitSize); snprintf(name, d->unitSize, "%.*s", d->unitSize, d->val.str); } /* Extract service description from record */ d = sdp_data_get(rec, SDP_ATTR_SVCDESC_PRIMARY); if (d) { desc = g_new0(char, d->unitSize); snprintf(desc, d->unitSize, "%.*s", d->unitSize, d->val.str); } if (connection_register(pr->conn, pr->path, &pr->src, &pr->dst, pr->id, name, desc) < 0) { err_failed(pr->conn, pr->msg, "D-Bus path registration failed"); goto fail; } connection_store(pr->conn, pr->path); connection_paths = g_slist_append(connection_paths, g_strdup(pr->path)); create_path(pr->conn, pr->msg, pr->path, "ConnectionCreated"); fail: if (name) g_free(name); if (desc) g_free(desc); sdp_record_free(rec); dbus_error_free(&derr); pending_reply_free(pr); dbus_message_unref(reply); } static int get_record(struct pending_reply *pr, uint32_t handle, DBusPendingCallNotifyFunction cb) { DBusMessage *msg; DBusPendingCall *pending; msg = dbus_message_new_method_call("org.bluez", pr->adapter_path, "org.bluez.Adapter", "GetRemoteServiceRecord"); if (!msg) return -1; dbus_message_append_args(msg, DBUS_TYPE_STRING, &pr->addr, DBUS_TYPE_UINT32, &handle, DBUS_TYPE_INVALID); if (dbus_connection_send_with_reply(pr->conn, msg, &pending, -1) == FALSE) { error("Can't send D-Bus message."); return -1; } dbus_pending_call_set_notify(pending, cb, pr, NULL); dbus_message_unref(msg); dbus_pending_call_unref(pending); return 0; } static void pan_handle_reply(DBusPendingCall *call, void *data) { struct pending_reply *pr = data; DBusMessage *reply = dbus_pending_call_steal_reply(call); DBusError derr; uint32_t *phandle; int len; dbus_error_init(&derr); if (dbus_set_error_from_message(&derr, reply)) { if (dbus_error_has_name(&derr, "org.bluez.Error.ConnectionAttemptFailed")) err_connection_failed(pr->conn, pr->msg, derr.message); else err_not_supported(pr->conn, pr->msg); error("GetRemoteServiceHandles: %s(%s)", derr.name, derr.message); goto fail; } if (!dbus_message_get_args(reply, &derr, DBUS_TYPE_ARRAY, DBUS_TYPE_UINT32, &phandle, &len, DBUS_TYPE_INVALID)) { err_not_supported(pr->conn, pr->msg); error("%s: %s", derr.name, derr.message); goto fail; } if (!len) { err_not_supported(pr->conn, pr->msg); goto fail; } if (get_record(pr, *phandle, pan_record_reply) < 0) { err_not_supported(pr->conn, pr->msg); goto fail; } dbus_message_unref(reply); return; fail: dbus_error_free(&derr); pending_reply_free(pr); } static int get_handles(struct pending_reply *pr, DBusPendingCallNotifyFunction cb) { DBusMessage *msg; DBusPendingCall *pending; const char *uuid; msg = dbus_message_new_method_call("org.bluez", pr->adapter_path, "org.bluez.Adapter", "GetRemoteServiceHandles"); if (!msg) return -1; uuid = bnep_uuid(pr->id); dbus_message_append_args(msg, DBUS_TYPE_STRING, &pr->addr, DBUS_TYPE_STRING, &uuid, DBUS_TYPE_INVALID); if (dbus_connection_send_with_reply(pr->conn, msg, &pending, -1) == FALSE) { error("Can't send D-Bus message."); return -1; } dbus_pending_call_set_notify(pending, cb, pr, NULL); dbus_message_unref(msg); dbus_pending_call_unref(pending); return 0; } static DBusHandlerResult list_servers(DBusConnection *conn, DBusMessage *msg, void *data) { return list_paths(conn, msg, server_paths); } static DBusHandlerResult create_server(DBusConnection *conn, DBusMessage *msg, void *data) { char path[MAX_PATH_LENGTH]; DBusError derr; const char *str; bdaddr_t src; uint16_t id; int dev_id; dbus_error_init(&derr); if (!dbus_message_get_args(msg, &derr, DBUS_TYPE_STRING, &str, DBUS_TYPE_INVALID)) { err_invalid_args(conn, msg, derr.message); dbus_error_free(&derr); return DBUS_HANDLER_RESULT_HANDLED; } id = bnep_service_id(str); if ((id != BNEP_SVC_GN) && (id != BNEP_SVC_NAP)) return err_invalid_args(conn, msg, "Not supported"); snprintf(path, MAX_PATH_LENGTH, NETWORK_PATH"/server/%s%d", bnep_name(id), net_uid++); if (g_slist_find_custom(server_paths, path, (GCompareFunc) strcmp)) { err_already_exists(conn, msg, "Server Already exists"); return DBUS_HANDLER_RESULT_HANDLED; } bacpy(&src, BDADDR_ANY); dev_id = hci_get_route(NULL); if (dev_id >= 0) hci_devba(dev_id, &src); if (server_register(conn, path, &src, id) < 0) return err_failed(conn, msg, "D-Bus path registration failed"); if (bacmp(&src, BDADDR_ANY) != 0) server_store(conn, path); server_paths = g_slist_append(server_paths, g_strdup(path)); return create_path(conn, msg, path, "ServerCreated"); } static DBusHandlerResult find_server(DBusConnection *conn, DBusMessage *msg, void *data) { DBusError derr; const char *pattern; const char *path; GSList *list; DBusMessage *reply; dbus_error_init(&derr); if (!dbus_message_get_args(msg, &derr, DBUS_TYPE_STRING, &pattern, DBUS_TYPE_INVALID)) { err_invalid_args(conn, msg, derr.message); dbus_error_free(&derr); return DBUS_HANDLER_RESULT_HANDLED; } for (list = server_paths; list; list = list->next) { path = (const char *) list->data; if (server_find_data(conn, path, pattern) == 0) break; } if (list == NULL) { err_failed(conn, msg, "No such server"); return DBUS_HANDLER_RESULT_HANDLED; } reply = dbus_message_new_method_return(msg); if (!reply) return DBUS_HANDLER_RESULT_NEED_MEMORY; dbus_message_append_args(reply, DBUS_TYPE_STRING, &path, DBUS_TYPE_INVALID); return send_message_and_unref(conn, reply); } static DBusHandlerResult remove_server(DBusConnection *conn, DBusMessage *msg, void *data) { return remove_path(conn, msg, &server_paths, "ServerRemoved"); } static DBusHandlerResult list_connections(DBusConnection *conn, DBusMessage *msg, void *data) { return list_paths(conn, msg, connection_paths); } static DBusHandlerResult find_connection(DBusConnection *conn, DBusMessage *msg, void *data) { DBusError derr; const char *pattern; const char *path; GSList *list; DBusMessage *reply; dbus_error_init(&derr); if (!dbus_message_get_args(msg, &derr, DBUS_TYPE_STRING, &pattern, DBUS_TYPE_INVALID)) { err_invalid_args(conn, msg, derr.message); dbus_error_free(&derr); return DBUS_HANDLER_RESULT_HANDLED; } for (list = connection_paths; list; list = list->next) { path = (const char *) list->data; if (connection_find_data(conn, path, pattern) == 0) break; } if (list == NULL) { err_failed(conn, msg, "No such connection"); return DBUS_HANDLER_RESULT_HANDLED; } reply = dbus_message_new_method_return(msg); if (!reply) return DBUS_HANDLER_RESULT_NEED_MEMORY; dbus_message_append_args(reply, DBUS_TYPE_STRING, &path, DBUS_TYPE_INVALID); return send_message_and_unref(conn, reply); } static DBusHandlerResult create_connection(DBusConnection *conn, DBusMessage *msg, void *data) { struct pending_reply *pr; DBusError derr; const char *addr; const char *str; bdaddr_t src; uint16_t id; int dev_id; dbus_error_init(&derr); if (!dbus_message_get_args(msg, &derr, DBUS_TYPE_STRING, &addr, DBUS_TYPE_STRING, &str, DBUS_TYPE_INVALID)) { err_invalid_args(conn, msg, derr.message); dbus_error_free(&derr); return DBUS_HANDLER_RESULT_HANDLED; } id = bnep_service_id(str); if ((id != BNEP_SVC_GN) && (id != BNEP_SVC_NAP)) return err_invalid_args(conn, msg, "Not supported"); bacpy(&src, BDADDR_ANY); dev_id = hci_get_route(NULL); if ((dev_id < 0) || (hci_devba(dev_id, &src) < 0)) return err_failed(conn, msg, "Adapter not available"); pr = g_new0(struct pending_reply, 1); pr->conn = dbus_connection_ref(conn); pr->msg = dbus_message_ref(msg); bacpy(&pr->src, &src); str2ba(addr, &pr->dst); pr->addr = g_strdup(addr); pr->id = id; pr->path = g_new0(char, MAX_PATH_LENGTH); snprintf(pr->path, MAX_PATH_LENGTH, NETWORK_PATH"/connection%d", net_uid++); pr->adapter_path = g_malloc0(16); snprintf(pr->adapter_path, 16, "/org/bluez/hci%d", dev_id); if (get_handles(pr, pan_handle_reply) < 0) return err_not_supported(conn, msg); return DBUS_HANDLER_RESULT_HANDLED; } static DBusHandlerResult remove_connection(DBusConnection *conn, DBusMessage *msg, void *data) { return remove_path(conn, msg, &connection_paths, "ConnectionRemoved"); } static void manager_unregister(DBusConnection *conn, void *data) { info("Unregistered manager path"); if (server_paths) { g_slist_foreach(server_paths, (GFunc)g_free, NULL); g_slist_free(server_paths); server_paths = NULL; } if (connection_paths) { g_slist_foreach(connection_paths, (GFunc)g_free, NULL); g_slist_free(connection_paths); connection_paths = NULL; } bnep_kill_all_connections(); } static void parse_stored_connection(char *key, char *value, void *data) { bdaddr_t dst, *src = data; char path[MAX_PATH_LENGTH]; char addr[18]; const char *ptr; char *name; int len, id; /* Format: XX:XX:XX:XX:XX:XX#{NAP, GN} name:description */ /* Parsing the key: address#role */ ptr = strchr(key, '#'); /* Empty address or invalid len */ if (!ptr || ((ptr - key) != 17)) return; memset(addr, 0, 18); strncpy(addr, key, 17); str2ba(addr, &dst); /* Empty role */ if (++ptr == NULL) return; if (strcasecmp("nap", ptr) == 0) id = BNEP_SVC_NAP; else if (strcasecmp("gn", ptr) == 0) id = BNEP_SVC_GN; else return; snprintf(path, MAX_PATH_LENGTH, NETWORK_PATH"/connection%d", net_uid++); /* Parsing the value: name and description */ ptr = strchr(value, ':'); /* Empty name */ if (!ptr) return; len = ptr-value; name = g_malloc0(len + 1); strncpy(name, value, len); /* Empty description */ if (++ptr == NULL) { g_free(name); return; } if (connection_register(connection, path, src, &dst, id, name, ptr) == 0) { connection_paths = g_slist_append(connection_paths, g_strdup(path)); } g_free(name); } static void register_stored(void) { char dirname[PATH_MAX + 1]; char filename[PATH_MAX + 1]; char path[MAX_PATH_LENGTH]; struct dirent *de; DIR *dir; struct stat s; bdaddr_t src; snprintf(dirname, PATH_MAX, "%s", STORAGEDIR); dir = opendir(dirname); if (!dir) return; while ((de = readdir(dir)) != NULL) { if (!isdigit(de->d_name[0])) continue; /* Connection objects */ create_name(filename, PATH_MAX, STORAGEDIR, de->d_name, "network"); str2ba(de->d_name, &src); if (stat (filename, &s) == 0 && (s.st_mode & __S_IFREG)) textfile_foreach(filename, parse_stored_connection, &src); /* NAP objects */ create_name(filename, PATH_MAX, STORAGEDIR, de->d_name, "nap"); if (stat (filename, &s) == 0 && (s.st_mode & __S_IFREG)) { snprintf(path, MAX_PATH_LENGTH, NETWORK_PATH"/server/nap%d", net_uid++); if (server_register_from_file(connection, path, &src, BNEP_SVC_NAP, filename) == 0) server_paths = g_slist_append(server_paths, g_strdup(path)); } /* GN objects */ create_name(filename, PATH_MAX, STORAGEDIR, de->d_name, "gn"); if (stat (filename, &s) == 0 && (s.st_mode & __S_IFREG)) { snprintf(path, MAX_PATH_LENGTH, NETWORK_PATH"/server/gn%d", net_uid++); if (server_register_from_file(connection, path, &src, BNEP_SVC_GN, filename) == 0) server_paths = g_slist_append(server_paths, g_strdup(path)); } } closedir(dir); } static DBusMethodVTable manager_methods[] = { { "ListServers", list_servers, "", "as" }, { "CreateServer", create_server, "s", "s" }, { "FindServer", find_server, "s", "s" }, { "RemoveServer", remove_server, "s", "" }, { "ListConnections", list_connections, "", "as" }, { "FindConnection", find_connection, "s", "s" }, { "CreateConnection", create_connection, "ss", "s" }, { "RemoveConnection", remove_connection, "s", "" }, { NULL, NULL, NULL, NULL } }; static DBusSignalVTable manager_signals[] = { { "ServerCreated", "s" }, { "ServerRemoved", "s" }, { "ConnectionCreated", "s" }, { "ConnectionRemoved", "s" }, { NULL, NULL } }; int network_init(DBusConnection *conn) { if (bridge_init() < 0) { error("Can't init bridge module"); return -1; } if (bridge_create("pan0") < 0) error("Can't create bridge"); if (bnep_init()) { error("Can't init bnep module"); return -1; } if (!dbus_connection_create_object_path(conn, NETWORK_PATH, NULL, manager_unregister)) { error("D-Bus failed to create %s path", NETWORK_PATH); return -1; } if (!dbus_connection_register_interface(conn, NETWORK_PATH, NETWORK_MANAGER_INTERFACE, manager_methods, manager_signals, NULL)) { error("Failed to register %s interface to %s", NETWORK_MANAGER_INTERFACE, NETWORK_PATH); dbus_connection_destroy_object_path(connection, NETWORK_PATH); return -1; } connection = dbus_connection_ref(conn); info("Registered manager path:%s", NETWORK_PATH); register_stored(); return 0; } void network_exit(void) { dbus_connection_destroy_object_path(connection, NETWORK_PATH); dbus_connection_unref(connection); connection = NULL; if (bridge_remove("pan0") < 0) error("Can't remove bridge"); bnep_cleanup(); bridge_cleanup(); } static inline int create_filename(char *buf, size_t size, bdaddr_t *bdaddr, const char *name) { char addr[18]; ba2str(bdaddr, addr); return create_name(buf, size, STORAGEDIR, addr, name); }