|  | /* | 
|  | * | 
|  | *  OBEX Server | 
|  | * | 
|  | *  Copyright (C) 2007-2010  Nokia Corporation | 
|  | *  Copyright (C) 2007-2010  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 <errno.h> | 
|  | #include <string.h> | 
|  | #include <unistd.h> | 
|  | #include <inttypes.h> | 
|  | #include <fcntl.h> | 
|  | #include <sys/socket.h> | 
|  |  | 
|  | #include <glib.h> | 
|  |  | 
|  | #include "lib/bluetooth.h" | 
|  | #include "lib/uuid.h" | 
|  |  | 
|  | #include "gdbus/gdbus.h" | 
|  |  | 
|  | #include "btio/btio.h" | 
|  | #include "obexd/src/obexd.h" | 
|  | #include "obexd/src/plugin.h" | 
|  | #include "obexd/src/server.h" | 
|  | #include "obexd/src/obex.h" | 
|  | #include "obexd/src/transport.h" | 
|  | #include "obexd/src/service.h" | 
|  | #include "obexd/src/log.h" | 
|  |  | 
|  | #define BT_RX_MTU 32767 | 
|  | #define BT_TX_MTU 32767 | 
|  |  | 
|  | struct bluetooth_profile { | 
|  | struct obex_server *server; | 
|  | struct obex_service_driver *driver; | 
|  | char *uuid; | 
|  | char *path; | 
|  | }; | 
|  |  | 
|  | static GSList *profiles = NULL; | 
|  |  | 
|  | static DBusConnection *connection = NULL; | 
|  |  | 
|  | static DBusMessage *profile_release(DBusConnection *conn, DBusMessage *msg, | 
|  | void *data) | 
|  | { | 
|  | return g_dbus_create_reply(msg, DBUS_TYPE_INVALID); | 
|  | } | 
|  |  | 
|  | static void connect_event(GIOChannel *io, GError *err, void *user_data) | 
|  | { | 
|  | int sk = g_io_channel_unix_get_fd(io); | 
|  | struct bluetooth_profile *profile = user_data; | 
|  | struct obex_server *server = profile->server; | 
|  | int type; | 
|  | int omtu = BT_TX_MTU; | 
|  | int imtu = BT_RX_MTU; | 
|  | gboolean stream = TRUE; | 
|  | socklen_t len = sizeof(int); | 
|  |  | 
|  | if (err) | 
|  | goto drop; | 
|  |  | 
|  | if (getsockopt(sk, SOL_SOCKET, SO_TYPE, &type, &len) < 0) | 
|  | goto done; | 
|  |  | 
|  | if (type != SOCK_SEQPACKET) | 
|  | goto done; | 
|  |  | 
|  | stream = FALSE; | 
|  |  | 
|  | /* Read MTU if io is an L2CAP socket */ | 
|  | bt_io_get(io, NULL, BT_IO_OPT_OMTU, &omtu, BT_IO_OPT_IMTU, &imtu, | 
|  | BT_IO_OPT_INVALID); | 
|  |  | 
|  | done: | 
|  | if (obex_server_new_connection(server, io, omtu, imtu, stream) < 0) | 
|  | g_io_channel_shutdown(io, TRUE, NULL); | 
|  |  | 
|  | return; | 
|  |  | 
|  | drop: | 
|  | error("%s", err->message); | 
|  | g_io_channel_shutdown(io, TRUE, NULL); | 
|  | return; | 
|  | } | 
|  |  | 
|  | static DBusMessage *invalid_args(DBusMessage *msg) | 
|  | { | 
|  | return g_dbus_create_error(msg, "org.bluez.Error.InvalidArguments", | 
|  | "Invalid arguments in method call"); | 
|  | } | 
|  |  | 
|  | static DBusMessage *profile_new_connection(DBusConnection *conn, | 
|  | DBusMessage *msg, void *data) | 
|  | { | 
|  | DBusMessageIter args; | 
|  | const char *device; | 
|  | int fd; | 
|  | GIOChannel *io; | 
|  |  | 
|  | dbus_message_iter_init(msg, &args); | 
|  |  | 
|  | if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_OBJECT_PATH) | 
|  | return invalid_args(msg); | 
|  |  | 
|  | dbus_message_iter_get_basic(&args, &device); | 
|  |  | 
|  | dbus_message_iter_next(&args); | 
|  |  | 
|  | if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_UNIX_FD) | 
|  | return invalid_args(msg); | 
|  |  | 
|  | dbus_message_iter_get_basic(&args, &fd); | 
|  |  | 
|  | if (fd < 0) { | 
|  | error("bluetooth: NewConnection invalid fd"); | 
|  | return invalid_args(msg); | 
|  | } | 
|  |  | 
|  | /* Read fd flags to make sure it can be used */ | 
|  | if (fcntl(fd, F_GETFD) < 0) { | 
|  | error("bluetooth: fcntl(%d, F_GETFD): %s (%d)", fd, | 
|  | strerror(errno), errno); | 
|  | return invalid_args(msg); | 
|  | } | 
|  |  | 
|  | io = g_io_channel_unix_new(fd); | 
|  | if (io == NULL) | 
|  | return invalid_args(msg); | 
|  |  | 
|  | DBG("device %s", device); | 
|  |  | 
|  | connect_event(io, NULL, data); | 
|  | g_io_channel_unref(io); | 
|  |  | 
|  | return g_dbus_create_reply(msg, DBUS_TYPE_INVALID); | 
|  | } | 
|  |  | 
|  | static DBusMessage *profile_request_disconnection(DBusConnection *conn, | 
|  | DBusMessage *msg, void *data) | 
|  | { | 
|  | return g_dbus_create_reply(msg, DBUS_TYPE_INVALID); | 
|  | } | 
|  |  | 
|  | static DBusMessage *profile_cancel(DBusConnection *conn, | 
|  | DBusMessage *msg, void *data) | 
|  | { | 
|  | return g_dbus_create_reply(msg, DBUS_TYPE_INVALID); | 
|  | } | 
|  |  | 
|  | static const GDBusMethodTable profile_methods[] = { | 
|  | { GDBUS_METHOD("Release", | 
|  | NULL, NULL, | 
|  | profile_release) }, | 
|  | { GDBUS_METHOD("NewConnection", | 
|  | GDBUS_ARGS({ "device", "o" }, { "fd", "h" }, | 
|  | { "options", "a{sv}" }), NULL, | 
|  | profile_new_connection) }, | 
|  | { GDBUS_METHOD("RequestDisconnection", | 
|  | GDBUS_ARGS({ "device", "o" }), NULL, | 
|  | profile_request_disconnection) }, | 
|  | { GDBUS_METHOD("Cancel", | 
|  | NULL, NULL, | 
|  | profile_cancel) }, | 
|  | { } | 
|  | }; | 
|  |  | 
|  | static void unregister_profile(struct bluetooth_profile *profile) | 
|  | { | 
|  | g_dbus_unregister_interface(connection, profile->path, | 
|  | "org.bluez.Profile1"); | 
|  | g_free(profile->path); | 
|  | profile->path = NULL; | 
|  | } | 
|  |  | 
|  | static void register_profile_reply(DBusPendingCall *call, void *user_data) | 
|  | { | 
|  | struct bluetooth_profile *profile = user_data; | 
|  | DBusMessage *reply = dbus_pending_call_steal_reply(call); | 
|  | DBusError derr; | 
|  |  | 
|  | dbus_error_init(&derr); | 
|  | if (!dbus_set_error_from_message(&derr, reply)) { | 
|  | DBG("Profile %s registered", profile->path); | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | unregister_profile(profile); | 
|  |  | 
|  | error("bluetooth: RequestProfile error: %s, %s", derr.name, | 
|  | derr.message); | 
|  | dbus_error_free(&derr); | 
|  | done: | 
|  | dbus_message_unref(reply); | 
|  | } | 
|  |  | 
|  | static void profile_free(void *data) | 
|  | { | 
|  | struct bluetooth_profile *profile = data; | 
|  |  | 
|  | if (profile->path != NULL) | 
|  | unregister_profile(profile); | 
|  |  | 
|  | g_free(profile->uuid); | 
|  | g_free(profile); | 
|  | } | 
|  |  | 
|  | static void append_variant(DBusMessageIter *iter, int type, void *val) | 
|  | { | 
|  | DBusMessageIter value; | 
|  | char sig[2] = { type, '\0' }; | 
|  |  | 
|  | dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT, sig, &value); | 
|  |  | 
|  | dbus_message_iter_append_basic(&value, type, val); | 
|  |  | 
|  | dbus_message_iter_close_container(iter, &value); | 
|  | } | 
|  |  | 
|  |  | 
|  | static void dict_append_entry(DBusMessageIter *dict, | 
|  | const char *key, int type, void *val) | 
|  | { | 
|  | DBusMessageIter entry; | 
|  |  | 
|  | if (type == DBUS_TYPE_STRING) { | 
|  | const char *str = *((const char **) val); | 
|  | if (str == NULL) | 
|  | return; | 
|  | } | 
|  |  | 
|  | dbus_message_iter_open_container(dict, DBUS_TYPE_DICT_ENTRY, | 
|  | NULL, &entry); | 
|  |  | 
|  | dbus_message_iter_append_basic(&entry, DBUS_TYPE_STRING, &key); | 
|  |  | 
|  | append_variant(&entry, type, val); | 
|  |  | 
|  | dbus_message_iter_close_container(dict, &entry); | 
|  | } | 
|  |  | 
|  | static int register_profile(struct bluetooth_profile *profile) | 
|  | { | 
|  | DBusMessage *msg; | 
|  | DBusMessageIter iter, opt; | 
|  | DBusPendingCall *call; | 
|  | dbus_bool_t auto_connect = FALSE; | 
|  | char *xml; | 
|  | int ret = 0; | 
|  |  | 
|  | profile->path = g_strconcat("/org/bluez/obex/", profile->uuid, NULL); | 
|  | g_strdelimit(profile->path, "-", '_'); | 
|  |  | 
|  | if (!g_dbus_register_interface(connection, profile->path, | 
|  | "org.bluez.Profile1", profile_methods, | 
|  | NULL, NULL, | 
|  | profile, NULL)) { | 
|  | error("D-Bus failed to register %s", profile->path); | 
|  | g_free(profile->path); | 
|  | profile->path = NULL; | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | msg = dbus_message_new_method_call("org.bluez", "/org/bluez", | 
|  | "org.bluez.ProfileManager1", | 
|  | "RegisterProfile"); | 
|  |  | 
|  | dbus_message_iter_init_append(msg, &iter); | 
|  |  | 
|  | dbus_message_iter_append_basic(&iter, DBUS_TYPE_OBJECT_PATH, | 
|  | &profile->path); | 
|  | dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, | 
|  | &profile->uuid); | 
|  | 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, | 
|  | &opt); | 
|  | dict_append_entry(&opt, "AutoConnect", DBUS_TYPE_BOOLEAN, | 
|  | &auto_connect); | 
|  | if (profile->driver->record) { | 
|  | if (profile->driver->port != 0) | 
|  | xml = g_markup_printf_escaped(profile->driver->record, | 
|  | profile->driver->channel, | 
|  | profile->driver->name, | 
|  | profile->driver->port); | 
|  | else | 
|  | xml = g_markup_printf_escaped(profile->driver->record, | 
|  | profile->driver->channel, | 
|  | profile->driver->name); | 
|  | dict_append_entry(&opt, "ServiceRecord", DBUS_TYPE_STRING, | 
|  | &xml); | 
|  | g_free(xml); | 
|  | } | 
|  | dbus_message_iter_close_container(&iter, &opt); | 
|  |  | 
|  | if (!g_dbus_send_message_with_reply(connection, msg, &call, -1)) { | 
|  | ret = -1; | 
|  | unregister_profile(profile); | 
|  | goto failed; | 
|  | } | 
|  |  | 
|  | dbus_pending_call_set_notify(call, register_profile_reply, profile, | 
|  | NULL); | 
|  | dbus_pending_call_unref(call); | 
|  |  | 
|  | failed: | 
|  | dbus_message_unref(msg); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static const char *service2uuid(uint16_t service) | 
|  | { | 
|  | switch (service) { | 
|  | case OBEX_OPP: | 
|  | return OBEX_OPP_UUID; | 
|  | case OBEX_FTP: | 
|  | return OBEX_FTP_UUID; | 
|  | case OBEX_PBAP: | 
|  | return OBEX_PSE_UUID; | 
|  | case OBEX_IRMC: | 
|  | return OBEX_SYNC_UUID; | 
|  | case OBEX_PCSUITE: | 
|  | return "00005005-0000-1000-8000-0002ee000001"; | 
|  | case OBEX_SYNCEVOLUTION: | 
|  | return "00000002-0000-1000-8000-0002ee000002"; | 
|  | case OBEX_MAS: | 
|  | return OBEX_MAS_UUID; | 
|  | case OBEX_MNS: | 
|  | return OBEX_MNS_UUID; | 
|  | } | 
|  |  | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | static void name_acquired(DBusConnection *conn, void *user_data) | 
|  | { | 
|  | GSList *l; | 
|  |  | 
|  | DBG("org.bluez appeared"); | 
|  |  | 
|  | for (l = profiles; l; l = l->next) { | 
|  | struct bluetooth_profile *profile = l->data; | 
|  |  | 
|  | if (profile->path != NULL) | 
|  | continue; | 
|  |  | 
|  | if (register_profile(profile) < 0) { | 
|  | error("bluetooth: Failed to register profile %s", | 
|  | profile->path); | 
|  | g_free(profile->path); | 
|  | profile->path = NULL; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | static void name_released(DBusConnection *conn, void *user_data) | 
|  | { | 
|  | GSList *l; | 
|  |  | 
|  | DBG("org.bluez disappered"); | 
|  |  | 
|  | for (l = profiles; l; l = l->next) { | 
|  | struct bluetooth_profile *profile = l->data; | 
|  |  | 
|  | if (profile->path == NULL) | 
|  | continue; | 
|  |  | 
|  | unregister_profile(profile); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void *bluetooth_start(struct obex_server *server, int *err) | 
|  | { | 
|  | const GSList *l; | 
|  |  | 
|  | for (l = server->drivers; l; l = l->next) { | 
|  | struct obex_service_driver *driver = l->data; | 
|  | struct bluetooth_profile *profile; | 
|  | const char *uuid; | 
|  |  | 
|  | uuid = service2uuid(driver->service); | 
|  | if (uuid == NULL) | 
|  | continue; | 
|  |  | 
|  | profile = g_new0(struct bluetooth_profile, 1); | 
|  | profile->driver = driver; | 
|  | profile->server = server; | 
|  | profile->uuid = g_strdup(uuid); | 
|  |  | 
|  | profiles = g_slist_prepend(profiles, profile); | 
|  | } | 
|  |  | 
|  | return profiles; | 
|  | } | 
|  |  | 
|  | static void bluetooth_stop(void *data) | 
|  | { | 
|  | g_slist_free_full(profiles, profile_free); | 
|  | profiles = NULL; | 
|  | } | 
|  |  | 
|  | static int bluetooth_getpeername(GIOChannel *io, char **name) | 
|  | { | 
|  | GError *gerr = NULL; | 
|  | char address[18]; | 
|  |  | 
|  | bt_io_get(io, &gerr, BT_IO_OPT_DEST, address, BT_IO_OPT_INVALID); | 
|  |  | 
|  | if (gerr) { | 
|  | error("%s", gerr->message); | 
|  | g_error_free(gerr); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | *name = g_strdup(address); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int bluetooth_getsockname(GIOChannel *io, char **name) | 
|  | { | 
|  | GError *gerr = NULL; | 
|  | char address[18]; | 
|  |  | 
|  | bt_io_get(io, &gerr, BT_IO_OPT_SOURCE, address, BT_IO_OPT_INVALID); | 
|  |  | 
|  | if (gerr) { | 
|  | error("%s", gerr->message); | 
|  | g_error_free(gerr); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | *name = g_strdup(address); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static struct obex_transport_driver driver = { | 
|  | .name = "bluetooth", | 
|  | .start = bluetooth_start, | 
|  | .getpeername = bluetooth_getpeername, | 
|  | .getsockname = bluetooth_getsockname, | 
|  | .stop = bluetooth_stop | 
|  | }; | 
|  |  | 
|  | static unsigned int listener_id = 0; | 
|  |  | 
|  | static int bluetooth_init(void) | 
|  | { | 
|  | connection = g_dbus_setup_private(DBUS_BUS_SYSTEM, NULL, NULL); | 
|  | if (connection == NULL) | 
|  | return -EPERM; | 
|  |  | 
|  | listener_id = g_dbus_add_service_watch(connection, "org.bluez", | 
|  | name_acquired, name_released, NULL, NULL); | 
|  |  | 
|  | return obex_transport_driver_register(&driver); | 
|  | } | 
|  |  | 
|  | static void bluetooth_exit(void) | 
|  | { | 
|  | g_dbus_remove_watch(connection, listener_id); | 
|  |  | 
|  | g_slist_free_full(profiles, profile_free); | 
|  |  | 
|  | if (connection) | 
|  | dbus_connection_unref(connection); | 
|  |  | 
|  | obex_transport_driver_unregister(&driver); | 
|  | } | 
|  |  | 
|  | OBEX_PLUGIN_DEFINE(bluetooth, bluetooth_init, bluetooth_exit) |