884 lines
20 KiB
C
884 lines
20 KiB
C
/*
|
|
*
|
|
* BlueZ - Bluetooth protocol stack for Linux
|
|
*
|
|
* Copyright (C) 2003-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 <stdio.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <assert.h>
|
|
#include <signal.h>
|
|
#include <sys/socket.h>
|
|
#include <glib.h>
|
|
|
|
#include <bluetooth/bluetooth.h>
|
|
#include <bluetooth/sdp.h>
|
|
#include <bluetooth/sdp_lib.h>
|
|
|
|
#include <gdbus/gdbus.h>
|
|
|
|
#include "cups.h"
|
|
|
|
struct cups_device {
|
|
char *bdaddr;
|
|
char *name;
|
|
char *id;
|
|
};
|
|
|
|
static GSList *device_list = NULL;
|
|
static GMainLoop *loop = NULL;
|
|
static DBusConnection *conn = NULL;
|
|
static gboolean doing_disco = FALSE;
|
|
|
|
#define ATTRID_1284ID 0x0300
|
|
|
|
struct context_data {
|
|
gboolean found;
|
|
char *id;
|
|
};
|
|
|
|
static void element_start(GMarkupParseContext *context,
|
|
const char *element_name,
|
|
const char **attribute_names,
|
|
const char **attribute_values,
|
|
gpointer user_data, GError **err)
|
|
{
|
|
struct context_data *ctx_data = user_data;
|
|
|
|
if (!strcmp(element_name, "record"))
|
|
return;
|
|
|
|
if (!strcmp(element_name, "attribute")) {
|
|
int i;
|
|
for (i = 0; attribute_names[i]; i++) {
|
|
if (strcmp(attribute_names[i], "id") != 0)
|
|
continue;
|
|
if (strtol(attribute_values[i], 0, 0) == ATTRID_1284ID)
|
|
ctx_data->found = TRUE;
|
|
break;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (ctx_data->found && !strcmp(element_name, "text")) {
|
|
int i;
|
|
for (i = 0; attribute_names[i]; i++) {
|
|
if (!strcmp(attribute_names[i], "value")) {
|
|
ctx_data->id = g_strdup(attribute_values[i] + 2);
|
|
ctx_data->found = FALSE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static GMarkupParser parser = {
|
|
element_start, NULL, NULL, NULL, NULL
|
|
};
|
|
|
|
static char *sdp_xml_parse_record(const char *data)
|
|
{
|
|
GMarkupParseContext *ctx;
|
|
struct context_data ctx_data;
|
|
int size;
|
|
|
|
size = strlen(data);
|
|
ctx_data.found = FALSE;
|
|
ctx_data.id = NULL;
|
|
ctx = g_markup_parse_context_new(&parser, 0, &ctx_data, NULL);
|
|
|
|
if (g_markup_parse_context_parse(ctx, data, size, NULL) == FALSE) {
|
|
g_markup_parse_context_free(ctx);
|
|
g_free(ctx_data.id);
|
|
return NULL;
|
|
}
|
|
|
|
g_markup_parse_context_free(ctx);
|
|
|
|
return ctx_data.id;
|
|
}
|
|
|
|
static char *device_get_ieee1284_id(const char *adapter, const char *device)
|
|
{
|
|
DBusMessage *message, *reply;
|
|
DBusMessageIter iter, reply_iter;
|
|
DBusMessageIter reply_iter_entry;
|
|
const char *hcr_print = "00001126-0000-1000-8000-00805f9b34fb";
|
|
const char *xml;
|
|
char *id = NULL;
|
|
|
|
/* Look for the service handle of the HCRP service */
|
|
message = dbus_message_new_method_call("org.bluez", device,
|
|
"org.bluez.Device1",
|
|
"DiscoverServices");
|
|
dbus_message_iter_init_append(message, &iter);
|
|
dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &hcr_print);
|
|
|
|
reply = dbus_connection_send_with_reply_and_block(conn,
|
|
message, -1, NULL);
|
|
|
|
dbus_message_unref(message);
|
|
|
|
if (!reply)
|
|
return NULL;
|
|
|
|
dbus_message_iter_init(reply, &reply_iter);
|
|
|
|
if (dbus_message_iter_get_arg_type(&reply_iter) != DBUS_TYPE_ARRAY) {
|
|
dbus_message_unref(reply);
|
|
return NULL;
|
|
}
|
|
|
|
dbus_message_iter_recurse(&reply_iter, &reply_iter_entry);
|
|
|
|
/* Hopefully we only get one handle, or take a punt */
|
|
while (dbus_message_iter_get_arg_type(&reply_iter_entry) ==
|
|
DBUS_TYPE_DICT_ENTRY) {
|
|
guint32 key;
|
|
DBusMessageIter dict_entry;
|
|
|
|
dbus_message_iter_recurse(&reply_iter_entry, &dict_entry);
|
|
|
|
/* Key ? */
|
|
dbus_message_iter_get_basic(&dict_entry, &key);
|
|
if (!key) {
|
|
dbus_message_iter_next(&reply_iter_entry);
|
|
continue;
|
|
}
|
|
|
|
/* Try to get the value */
|
|
if (!dbus_message_iter_next(&dict_entry)) {
|
|
dbus_message_iter_next(&reply_iter_entry);
|
|
continue;
|
|
}
|
|
|
|
dbus_message_iter_get_basic(&dict_entry, &xml);
|
|
|
|
id = sdp_xml_parse_record(xml);
|
|
if (id != NULL)
|
|
break;
|
|
dbus_message_iter_next(&reply_iter_entry);
|
|
}
|
|
|
|
dbus_message_unref(reply);
|
|
|
|
return id;
|
|
}
|
|
|
|
static void print_printer_details(const char *name, const char *bdaddr,
|
|
const char *id)
|
|
{
|
|
char *uri, *escaped;
|
|
|
|
escaped = g_strdelimit(g_strdup(name), "\"", '\'');
|
|
uri = g_strdup_printf("bluetooth://%c%c%c%c%c%c%c%c%c%c%c%c",
|
|
bdaddr[0], bdaddr[1],
|
|
bdaddr[3], bdaddr[4],
|
|
bdaddr[6], bdaddr[7],
|
|
bdaddr[9], bdaddr[10],
|
|
bdaddr[12], bdaddr[13],
|
|
bdaddr[15], bdaddr[16]);
|
|
printf("direct %s \"%s\" \"%s (Bluetooth)\"", uri, escaped, escaped);
|
|
if (id != NULL)
|
|
printf(" \"%s\"\n", id);
|
|
else
|
|
printf("\n");
|
|
g_free(escaped);
|
|
g_free(uri);
|
|
}
|
|
|
|
static void add_device_to_list(const char *name, const char *bdaddr,
|
|
const char *id)
|
|
{
|
|
struct cups_device *device;
|
|
GSList *l;
|
|
|
|
/* Look for the device in the list */
|
|
for (l = device_list; l != NULL; l = l->next) {
|
|
device = (struct cups_device *) l->data;
|
|
|
|
if (strcmp(device->bdaddr, bdaddr) == 0) {
|
|
if (device->name != name) {
|
|
g_free(device->name);
|
|
device->name = g_strdup(name);
|
|
}
|
|
g_free(device->id);
|
|
device->id = g_strdup(id);
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* Or add it to the list if it's not there */
|
|
device = g_new0(struct cups_device, 1);
|
|
device->bdaddr = g_strdup(bdaddr);
|
|
device->name = g_strdup(name);
|
|
device->id = g_strdup(id);
|
|
|
|
device_list = g_slist_prepend(device_list, device);
|
|
print_printer_details(device->name, device->bdaddr, device->id);
|
|
}
|
|
|
|
static gboolean parse_device_properties(DBusMessageIter *reply_iter,
|
|
char **name, char **bdaddr)
|
|
{
|
|
guint32 class = 0;
|
|
DBusMessageIter reply_iter_entry;
|
|
|
|
if (dbus_message_iter_get_arg_type(reply_iter) != DBUS_TYPE_ARRAY)
|
|
return FALSE;
|
|
|
|
dbus_message_iter_recurse(reply_iter, &reply_iter_entry);
|
|
|
|
while (dbus_message_iter_get_arg_type(&reply_iter_entry) ==
|
|
DBUS_TYPE_DICT_ENTRY) {
|
|
const char *key;
|
|
DBusMessageIter dict_entry, iter_dict_val;
|
|
|
|
dbus_message_iter_recurse(&reply_iter_entry, &dict_entry);
|
|
|
|
/* Key == Class ? */
|
|
dbus_message_iter_get_basic(&dict_entry, &key);
|
|
if (!key) {
|
|
dbus_message_iter_next(&reply_iter_entry);
|
|
continue;
|
|
}
|
|
|
|
if (strcmp(key, "Class") != 0 &&
|
|
strcmp(key, "Alias") != 0 &&
|
|
strcmp(key, "Address") != 0) {
|
|
dbus_message_iter_next(&reply_iter_entry);
|
|
continue;
|
|
}
|
|
|
|
/* Try to get the value */
|
|
if (!dbus_message_iter_next(&dict_entry)) {
|
|
dbus_message_iter_next(&reply_iter_entry);
|
|
continue;
|
|
}
|
|
dbus_message_iter_recurse(&dict_entry, &iter_dict_val);
|
|
if (strcmp(key, "Class") == 0) {
|
|
dbus_message_iter_get_basic(&iter_dict_val, &class);
|
|
} else {
|
|
const char *value;
|
|
dbus_message_iter_get_basic(&iter_dict_val, &value);
|
|
if (strcmp(key, "Alias") == 0) {
|
|
*name = g_strdup(value);
|
|
} else if (bdaddr) {
|
|
*bdaddr = g_strdup(value);
|
|
}
|
|
}
|
|
dbus_message_iter_next(&reply_iter_entry);
|
|
}
|
|
|
|
if (class == 0)
|
|
return FALSE;
|
|
if (((class & 0x1f00) >> 8) == 0x06 && (class & 0x80))
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean device_is_printer(const char *adapter, const char *device_path, char **name, char **bdaddr)
|
|
{
|
|
DBusMessage *message, *reply;
|
|
DBusMessageIter reply_iter;
|
|
gboolean retval;
|
|
|
|
message = dbus_message_new_method_call("org.bluez", device_path,
|
|
"org.bluez.Device1",
|
|
"GetProperties");
|
|
|
|
reply = dbus_connection_send_with_reply_and_block(conn,
|
|
message, -1, NULL);
|
|
|
|
dbus_message_unref(message);
|
|
|
|
if (!reply)
|
|
return FALSE;
|
|
|
|
dbus_message_iter_init(reply, &reply_iter);
|
|
|
|
retval = parse_device_properties(&reply_iter, name, bdaddr);
|
|
|
|
dbus_message_unref(reply);
|
|
|
|
return retval;
|
|
}
|
|
|
|
static void remote_device_found(const char *adapter, const char *bdaddr,
|
|
const char *name)
|
|
{
|
|
DBusMessage *message, *reply, *adapter_reply;
|
|
DBusMessageIter iter;
|
|
char *object_path = NULL;
|
|
char *id;
|
|
|
|
adapter_reply = NULL;
|
|
|
|
assert(adapter != NULL);
|
|
|
|
message = dbus_message_new_method_call("org.bluez", adapter,
|
|
"org.bluez.Adapter1",
|
|
"FindDevice");
|
|
dbus_message_iter_init_append(message, &iter);
|
|
dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &bdaddr);
|
|
|
|
if (adapter_reply != NULL)
|
|
dbus_message_unref(adapter_reply);
|
|
|
|
reply = dbus_connection_send_with_reply_and_block(conn,
|
|
message, -1, NULL);
|
|
|
|
dbus_message_unref(message);
|
|
|
|
if (!reply) {
|
|
message = dbus_message_new_method_call("org.bluez", adapter,
|
|
"org.bluez.Adapter1",
|
|
"CreateDevice");
|
|
dbus_message_iter_init_append(message, &iter);
|
|
dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &bdaddr);
|
|
|
|
reply = dbus_connection_send_with_reply_and_block(conn,
|
|
message, -1, NULL);
|
|
|
|
dbus_message_unref(message);
|
|
|
|
if (!reply)
|
|
return;
|
|
}
|
|
|
|
if (dbus_message_get_args(reply, NULL,
|
|
DBUS_TYPE_OBJECT_PATH, &object_path,
|
|
DBUS_TYPE_INVALID) == FALSE) {
|
|
dbus_message_unref(reply);
|
|
return;
|
|
}
|
|
|
|
id = device_get_ieee1284_id(adapter, object_path);
|
|
add_device_to_list(name, bdaddr, id);
|
|
g_free(id);
|
|
|
|
dbus_message_unref(reply);
|
|
}
|
|
|
|
static void discovery_completed(void)
|
|
{
|
|
g_slist_free(device_list);
|
|
device_list = NULL;
|
|
|
|
g_main_loop_quit(loop);
|
|
}
|
|
|
|
static void remote_device_disappeared(const char *bdaddr)
|
|
{
|
|
GSList *l;
|
|
|
|
for (l = device_list; l != NULL; l = l->next) {
|
|
struct cups_device *device = l->data;
|
|
|
|
if (strcmp(device->bdaddr, bdaddr) == 0) {
|
|
g_free(device->name);
|
|
g_free(device->bdaddr);
|
|
g_free(device);
|
|
device_list = g_slist_delete_link(device_list, l);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
static gboolean list_known_printers(const char *adapter)
|
|
{
|
|
DBusMessageIter reply_iter, iter_array;
|
|
DBusError error;
|
|
DBusMessage *message, *reply;
|
|
|
|
message = dbus_message_new_method_call("org.bluez", adapter,
|
|
"org.bluez.Adapter1",
|
|
"ListDevices");
|
|
if (message == NULL)
|
|
return FALSE;
|
|
|
|
dbus_error_init(&error);
|
|
reply = dbus_connection_send_with_reply_and_block(conn, message,
|
|
-1, &error);
|
|
|
|
dbus_message_unref(message);
|
|
|
|
if (dbus_error_is_set(&error)) {
|
|
dbus_error_free(&error);
|
|
return FALSE;
|
|
}
|
|
|
|
dbus_message_iter_init(reply, &reply_iter);
|
|
if (dbus_message_iter_get_arg_type(&reply_iter) != DBUS_TYPE_ARRAY) {
|
|
dbus_message_unref(reply);
|
|
return FALSE;
|
|
}
|
|
|
|
dbus_message_iter_recurse(&reply_iter, &iter_array);
|
|
while (dbus_message_iter_get_arg_type(&iter_array) ==
|
|
DBUS_TYPE_OBJECT_PATH) {
|
|
const char *object_path;
|
|
char *name = NULL;
|
|
char *bdaddr = NULL;
|
|
|
|
dbus_message_iter_get_basic(&iter_array, &object_path);
|
|
if (device_is_printer(adapter, object_path, &name, &bdaddr)) {
|
|
char *id;
|
|
|
|
id = device_get_ieee1284_id(adapter, object_path);
|
|
add_device_to_list(name, bdaddr, id);
|
|
g_free(id);
|
|
}
|
|
g_free(name);
|
|
g_free(bdaddr);
|
|
dbus_message_iter_next(&iter_array);
|
|
}
|
|
|
|
dbus_message_unref(reply);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static DBusHandlerResult filter_func(DBusConnection *connection,
|
|
DBusMessage *message, void *user_data)
|
|
{
|
|
if (dbus_message_is_signal(message, "org.bluez.Adapter1",
|
|
"DeviceFound")) {
|
|
const char *adapter, *bdaddr;
|
|
char *name;
|
|
DBusMessageIter iter;
|
|
|
|
dbus_message_iter_init(message, &iter);
|
|
dbus_message_iter_get_basic(&iter, &bdaddr);
|
|
dbus_message_iter_next(&iter);
|
|
|
|
adapter = dbus_message_get_path(message);
|
|
if (parse_device_properties(&iter, &name, NULL))
|
|
remote_device_found(adapter, bdaddr, name);
|
|
g_free (name);
|
|
} else if (dbus_message_is_signal(message, "org.bluez.Adapter1",
|
|
"DeviceDisappeared")) {
|
|
const char *bdaddr;
|
|
|
|
dbus_message_get_args(message, NULL,
|
|
DBUS_TYPE_STRING, &bdaddr,
|
|
DBUS_TYPE_INVALID);
|
|
remote_device_disappeared(bdaddr);
|
|
} else if (dbus_message_is_signal(message, "org.bluez.Adapter1",
|
|
"PropertyChanged")) {
|
|
DBusMessageIter iter, value_iter;
|
|
const char *name;
|
|
gboolean discovering;
|
|
|
|
dbus_message_iter_init(message, &iter);
|
|
dbus_message_iter_get_basic(&iter, &name);
|
|
if (name == NULL || strcmp(name, "Discovering") != 0)
|
|
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
|
|
dbus_message_iter_next(&iter);
|
|
dbus_message_iter_recurse(&iter, &value_iter);
|
|
dbus_message_iter_get_basic(&value_iter, &discovering);
|
|
|
|
if (discovering == FALSE && doing_disco) {
|
|
doing_disco = FALSE;
|
|
discovery_completed();
|
|
}
|
|
}
|
|
|
|
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
|
|
}
|
|
|
|
static gboolean list_printers(void)
|
|
{
|
|
/* 1. Connect to the bus
|
|
* 2. Get the manager
|
|
* 3. Get the default adapter
|
|
* 4. Get a list of devices
|
|
* 5. Get the class of each device
|
|
* 6. Print the details from each printer device
|
|
*/
|
|
DBusError error;
|
|
dbus_bool_t hcid_exists;
|
|
DBusMessage *reply, *message;
|
|
DBusMessageIter reply_iter;
|
|
char *adapter, *match;
|
|
|
|
conn = g_dbus_setup_bus(DBUS_BUS_SYSTEM, NULL, NULL);
|
|
if (conn == NULL)
|
|
return TRUE;
|
|
|
|
dbus_error_init(&error);
|
|
hcid_exists = dbus_bus_name_has_owner(conn, "org.bluez", &error);
|
|
if (dbus_error_is_set(&error)) {
|
|
dbus_error_free(&error);
|
|
return TRUE;
|
|
}
|
|
|
|
if (!hcid_exists)
|
|
return TRUE;
|
|
|
|
/* Get the default adapter */
|
|
message = dbus_message_new_method_call("org.bluez", "/",
|
|
"org.bluez.Manager",
|
|
"DefaultAdapter");
|
|
if (message == NULL) {
|
|
dbus_connection_unref(conn);
|
|
return FALSE;
|
|
}
|
|
|
|
reply = dbus_connection_send_with_reply_and_block(conn,
|
|
message, -1, &error);
|
|
|
|
dbus_message_unref(message);
|
|
|
|
if (dbus_error_is_set(&error)) {
|
|
dbus_error_free(&error);
|
|
dbus_connection_unref(conn);
|
|
/* No adapter */
|
|
return TRUE;
|
|
}
|
|
|
|
dbus_message_iter_init(reply, &reply_iter);
|
|
if (dbus_message_iter_get_arg_type(&reply_iter) !=
|
|
DBUS_TYPE_OBJECT_PATH) {
|
|
dbus_message_unref(reply);
|
|
dbus_connection_unref(conn);
|
|
return FALSE;
|
|
}
|
|
|
|
dbus_message_iter_get_basic(&reply_iter, &adapter);
|
|
adapter = g_strdup(adapter);
|
|
dbus_message_unref(reply);
|
|
|
|
if (!dbus_connection_add_filter(conn, filter_func, adapter, g_free)) {
|
|
g_free(adapter);
|
|
dbus_connection_unref(conn);
|
|
return FALSE;
|
|
}
|
|
|
|
#define MATCH_FORMAT \
|
|
"type='signal'," \
|
|
"interface='org.bluez.Adapter1'," \
|
|
"sender='org.bluez'," \
|
|
"path='%s'"
|
|
|
|
match = g_strdup_printf(MATCH_FORMAT, adapter);
|
|
dbus_bus_add_match(conn, match, &error);
|
|
g_free(match);
|
|
|
|
/* Add the the recent devices */
|
|
list_known_printers(adapter);
|
|
|
|
doing_disco = TRUE;
|
|
message = dbus_message_new_method_call("org.bluez", adapter,
|
|
"org.bluez.Adapter1",
|
|
"StartDiscovery");
|
|
|
|
if (!dbus_connection_send_with_reply(conn, message, NULL, -1)) {
|
|
dbus_message_unref(message);
|
|
dbus_connection_unref(conn);
|
|
g_free(adapter);
|
|
return FALSE;
|
|
}
|
|
dbus_message_unref(message);
|
|
|
|
loop = g_main_loop_new(NULL, TRUE);
|
|
g_main_loop_run(loop);
|
|
|
|
g_free(adapter);
|
|
dbus_connection_unref(conn);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean print_ieee1284(const char *bdaddr)
|
|
{
|
|
DBusMessage *message, *reply, *adapter_reply;
|
|
DBusMessageIter iter;
|
|
char *object_path = NULL;
|
|
char *adapter;
|
|
char *id;
|
|
|
|
adapter_reply = NULL;
|
|
|
|
conn = g_dbus_setup_bus(DBUS_BUS_SYSTEM, NULL, NULL);
|
|
if (conn == NULL)
|
|
return FALSE;
|
|
|
|
message = dbus_message_new_method_call("org.bluez", "/",
|
|
"org.bluez.Manager",
|
|
"DefaultAdapter");
|
|
|
|
adapter_reply = dbus_connection_send_with_reply_and_block(conn,
|
|
message, -1, NULL);
|
|
|
|
dbus_message_unref(message);
|
|
|
|
if (!adapter_reply)
|
|
return FALSE;
|
|
|
|
if (dbus_message_get_args(adapter_reply, NULL,
|
|
DBUS_TYPE_OBJECT_PATH, &adapter,
|
|
DBUS_TYPE_INVALID) == FALSE) {
|
|
dbus_message_unref(adapter_reply);
|
|
return FALSE;
|
|
}
|
|
|
|
message = dbus_message_new_method_call("org.bluez", adapter,
|
|
"org.bluez.Adapter1",
|
|
"FindDevice");
|
|
dbus_message_iter_init_append(message, &iter);
|
|
dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &bdaddr);
|
|
|
|
if (adapter_reply != NULL)
|
|
dbus_message_unref(adapter_reply);
|
|
|
|
reply = dbus_connection_send_with_reply_and_block(conn,
|
|
message, -1, NULL);
|
|
|
|
dbus_message_unref(message);
|
|
|
|
if (!reply) {
|
|
message = dbus_message_new_method_call("org.bluez", adapter,
|
|
"org.bluez.Adapter1",
|
|
"CreateDevice");
|
|
dbus_message_iter_init_append(message, &iter);
|
|
dbus_message_iter_append_basic(&iter,
|
|
DBUS_TYPE_STRING, &bdaddr);
|
|
|
|
reply = dbus_connection_send_with_reply_and_block(conn,
|
|
message, -1, NULL);
|
|
|
|
dbus_message_unref(message);
|
|
|
|
if (!reply)
|
|
return FALSE;
|
|
}
|
|
|
|
if (dbus_message_get_args(reply, NULL,
|
|
DBUS_TYPE_OBJECT_PATH, &object_path,
|
|
DBUS_TYPE_INVALID) == FALSE) {
|
|
dbus_message_unref(reply);
|
|
return FALSE;
|
|
}
|
|
|
|
id = device_get_ieee1284_id(adapter, object_path);
|
|
if (id == NULL) {
|
|
dbus_message_unref(reply);
|
|
return FALSE;
|
|
}
|
|
printf("%s", id);
|
|
g_free(id);
|
|
|
|
dbus_message_unref(reply);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* Usage: printer-uri job-id user title copies options [file]
|
|
*
|
|
*/
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
sdp_session_t *sdp;
|
|
bdaddr_t bdaddr;
|
|
unsigned short ctrl_psm, data_psm;
|
|
uint8_t channel, b[6];
|
|
char *ptr, str[3], device[18], service[12];
|
|
const char *uri, *cups_class;
|
|
int i, err, fd, copies, proto;
|
|
|
|
/* Make sure status messages are not buffered */
|
|
setbuf(stderr, NULL);
|
|
|
|
/* Make sure output is not buffered */
|
|
setbuf(stdout, NULL);
|
|
|
|
/* Ignore SIGPIPE signals */
|
|
#ifdef HAVE_SIGSET
|
|
sigset(SIGPIPE, SIG_IGN);
|
|
#elif defined(HAVE_SIGACTION)
|
|
memset(&action, 0, sizeof(action));
|
|
action.sa_handler = SIG_IGN;
|
|
sigaction(SIGPIPE, &action, NULL);
|
|
#else
|
|
signal(SIGPIPE, SIG_IGN);
|
|
#endif /* HAVE_SIGSET */
|
|
|
|
if (argc == 1) {
|
|
if (list_printers() == TRUE)
|
|
return CUPS_BACKEND_OK;
|
|
else
|
|
return CUPS_BACKEND_FAILED;
|
|
} else if (argc == 3 && strcmp(argv[1], "--get-deviceid") == 0) {
|
|
if (bachk(argv[2]) < 0) {
|
|
fprintf(stderr, "Invalid Bluetooth address '%s'\n",
|
|
argv[2]);
|
|
return CUPS_BACKEND_FAILED;
|
|
}
|
|
if (print_ieee1284(argv[2]) == FALSE)
|
|
return CUPS_BACKEND_FAILED;
|
|
return CUPS_BACKEND_OK;
|
|
}
|
|
|
|
if (argc < 6 || argc > 7) {
|
|
fprintf(stderr, "Usage: bluetooth job-id user title copies"
|
|
" options [file]\n");
|
|
fprintf(stderr, " bluetooth --get-deviceid [bdaddr]\n");
|
|
return CUPS_BACKEND_FAILED;
|
|
}
|
|
|
|
if (argc == 6) {
|
|
fd = 0;
|
|
copies = 1;
|
|
} else {
|
|
if ((fd = open(argv[6], O_RDONLY)) < 0) {
|
|
perror("ERROR: Unable to open print file");
|
|
return CUPS_BACKEND_FAILED;
|
|
}
|
|
copies = atoi(argv[4]);
|
|
}
|
|
|
|
uri = getenv("DEVICE_URI");
|
|
if (!uri)
|
|
uri = argv[0];
|
|
|
|
if (strncasecmp(uri, "bluetooth://", 12)) {
|
|
fprintf(stderr, "ERROR: No device URI found\n");
|
|
return CUPS_BACKEND_FAILED;
|
|
}
|
|
|
|
ptr = argv[0] + 12;
|
|
for (i = 0; i < 6; i++) {
|
|
strncpy(str, ptr, 2);
|
|
b[i] = (uint8_t) strtol(str, NULL, 16);
|
|
ptr += 2;
|
|
}
|
|
sprintf(device, "%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X",
|
|
b[0], b[1], b[2], b[3], b[4], b[5]);
|
|
|
|
str2ba(device, &bdaddr);
|
|
|
|
ptr = strchr(ptr, '/');
|
|
if (ptr) {
|
|
strncpy(service, ptr + 1, 12);
|
|
|
|
if (!strncasecmp(ptr + 1, "spp", 3))
|
|
proto = 1;
|
|
else if (!strncasecmp(ptr + 1, "hcrp", 4))
|
|
proto = 2;
|
|
else
|
|
proto = 0;
|
|
} else {
|
|
strcpy(service, "auto");
|
|
proto = 0;
|
|
}
|
|
|
|
cups_class = getenv("CLASS");
|
|
|
|
fprintf(stderr,
|
|
"DEBUG: %s device %s service %s fd %d copies %d class %s\n",
|
|
argv[0], device, service, fd, copies,
|
|
cups_class ? cups_class : "(none)");
|
|
|
|
fputs("STATE: +connecting-to-device\n", stderr);
|
|
|
|
service_search:
|
|
sdp = sdp_connect(BDADDR_ANY, &bdaddr, SDP_RETRY_IF_BUSY);
|
|
if (!sdp) {
|
|
fprintf(stderr, "ERROR: Can't open Bluetooth connection\n");
|
|
return CUPS_BACKEND_FAILED;
|
|
}
|
|
|
|
switch (proto) {
|
|
case 1:
|
|
err = sdp_search_spp(sdp, &channel);
|
|
break;
|
|
case 2:
|
|
err = sdp_search_hcrp(sdp, &ctrl_psm, &data_psm);
|
|
break;
|
|
default:
|
|
proto = 2;
|
|
err = sdp_search_hcrp(sdp, &ctrl_psm, &data_psm);
|
|
if (err) {
|
|
proto = 1;
|
|
err = sdp_search_spp(sdp, &channel);
|
|
}
|
|
break;
|
|
}
|
|
|
|
sdp_close(sdp);
|
|
|
|
if (err) {
|
|
if (cups_class) {
|
|
fputs("INFO: Unable to contact printer, queuing on "
|
|
"next printer in class...\n", stderr);
|
|
sleep(5);
|
|
return CUPS_BACKEND_FAILED;
|
|
}
|
|
sleep(20);
|
|
fprintf(stderr, "ERROR: Can't get service information\n");
|
|
goto service_search;
|
|
}
|
|
|
|
connect:
|
|
switch (proto) {
|
|
case 1:
|
|
err = spp_print(BDADDR_ANY, &bdaddr, channel,
|
|
fd, copies, cups_class);
|
|
break;
|
|
case 2:
|
|
err = hcrp_print(BDADDR_ANY, &bdaddr, ctrl_psm, data_psm,
|
|
fd, copies, cups_class);
|
|
break;
|
|
default:
|
|
err = CUPS_BACKEND_FAILED;
|
|
fprintf(stderr, "ERROR: Unsupported protocol\n");
|
|
break;
|
|
}
|
|
|
|
if (err == CUPS_BACKEND_FAILED && cups_class) {
|
|
fputs("INFO: Unable to contact printer, queuing on "
|
|
"next printer in class...\n", stderr);
|
|
sleep(5);
|
|
return CUPS_BACKEND_FAILED;
|
|
} else if (err == CUPS_BACKEND_RETRY) {
|
|
sleep(20);
|
|
goto connect;
|
|
}
|
|
|
|
if (fd != 0)
|
|
close(fd);
|
|
|
|
if (!err)
|
|
fprintf(stderr, "INFO: Ready to print\n");
|
|
|
|
return err;
|
|
}
|