From: Bastien Nocera <hadess@hadess.net>
To: BlueZ development <bluez-devel@lists.sourceforge.net>
Subject: Re: [Bluez-devel] CUPS plugin discovery bits
Date: Thu, 24 May 2007 14:39:52 +0100 [thread overview]
Message-ID: <1180013992.3151.6.camel@cookie.hadess.net> (raw)
In-Reply-To: <59278C28-851A-44FF-AE01-25D8F074CE1E@nokia.com>
[-- Attachment #1: Type: text/plain, Size: 461 bytes --]
On Wed, 2007-05-23 at 18:56 +0200, Johan Hedberg wrote:
> Hi Bastien,
>
> On May 23, 2007, at 18:26, Bastien Nocera wrote:
> > Updated patch attached.
>
> A little about optimizing the device discovery.
Fixed in this version, and also added recent devices (through
ListRemoteDevices and ListBondings).
Marcel told me this would be mergeable if I removed the dbus-glib
dependency, which I'm going to do now.
Cheers
--
Bastien Nocera <hadess@hadess.net>
[-- Attachment #2: bluez-utils-cups-printer-listing-4.patch --]
[-- Type: text/x-patch, Size: 10629 bytes --]
Index: acinclude.m4
===================================================================
RCS file: /cvsroot/bluez/utils/acinclude.m4,v
retrieving revision 1.107
diff -u -p -r1.107 acinclude.m4
--- acinclude.m4 8 Apr 2007 19:56:52 -0000 1.107
+++ acinclude.m4 24 May 2007 13:37:36 -0000
@@ -78,6 +78,8 @@ AC_DEFUN([AC_PATH_GLIB], [
PKG_CHECK_MODULES(GLIB, glib-2.0, glib_found=yes, glib_found=no)
AC_SUBST(GLIB_CFLAGS)
AC_SUBST(GLIB_LIBS)
+ GLIB_GENMARSHAL=`$PKG_CONFIG --variable=glib_genmarshal glib-2.0`
+ AC_SUBST(GLIB_GENMARSHAL)
])
AC_DEFUN([AC_PATH_DBUS], [
Index: cups/Makefile.am
===================================================================
RCS file: /cvsroot/bluez/utils/cups/Makefile.am,v
retrieving revision 1.9
diff -u -p -r1.9 Makefile.am
--- cups/Makefile.am 20 Aug 2006 02:21:03 -0000 1.9
+++ cups/Makefile.am 24 May 2007 13:37:36 -0000
@@ -7,9 +7,19 @@ else
noinst_PROGRAMS = bluetooth
endif
-bluetooth_SOURCES = main.c sdp.c spp.c hcrp.c
-bluetooth_LDADD = @BLUEZ_LIBS@
+MARSHAL_FILES = bluez-cups-marshal.h bluez-cups-marshal.c
-AM_CFLAGS = @BLUEZ_CFLAGS@
+bluetooth_SOURCES = main.c sdp.c spp.c hcrp.c $(MARSHAL_FILES)
+bluetooth_LDADD = @DBUS_GLIB_LIBS@ @BLUEZ_LIBS@
-MAINTAINERCLEANFILES = Makefile.in
+bluez-cups-marshal.h: bluez-cups-marshal.list
+ $(GLIB_GENMARSHAL) --prefix=bluez_cups_marshal $< --header > $@
+bluez-cups-marshal.c: bluez-cups-marshal.list
+ $(GLIB_GENMARSHAL) --prefix=bluez_cups_marshal $< --body > $@
+main.c: $(MARSHAL_FILES)
+
+AM_CFLAGS = @BLUEZ_CFLAGS@ @DBUS_GLIB_CFLAGS@
+
+MAINTAINERCLEANFILES = Makefile.in $(MARSHAL_FILES)
+CLEANFILES = $(MARSHAL_FILES)
+EXTRA_DIST = bluez-cups-marshal.list
Index: cups/main.c
===================================================================
RCS file: /cvsroot/bluez/utils/cups/main.c,v
retrieving revision 1.6
diff -u -p -r1.6 main.c
--- cups/main.c 3 Jan 2006 13:28:57 -0000 1.6
+++ cups/main.c 24 May 2007 13:37:36 -0000
@@ -38,12 +38,317 @@
#include <bluetooth/sdp.h>
#include <bluetooth/sdp_lib.h>
+#include <dbus/dbus-glib.h>
+#include <glib/gi18n.h>
+#include "bluez-cups-marshal.h"
+
extern int sdp_search_spp(sdp_session_t *sdp, uint8_t *channel);
extern int sdp_search_hcrp(sdp_session_t *sdp, unsigned short *ctrl_psm, unsigned short *data_psm);
extern int spp_print(bdaddr_t *src, bdaddr_t *dst, uint8_t channel, int fd, int copies);
extern int hcrp_print(bdaddr_t *src, bdaddr_t *dst, unsigned short ctrl_psm, unsigned short data_psm, int fd, int copies);
+#define PRINTER_SERVICE_CLASS_NAME "printer"
+
+typedef struct {
+ char *bdaddr;
+ char *name;
+} BluezCupsDevice;
+
+static GList *device_list = NULL;
+static GMainLoop *loop = NULL;
+
+static void
+add_device_to_list(const char *name, const char *bdaddr)
+{
+ BluezCupsDevice *device;
+ GList *l;
+
+ /* Look for the device in the list */
+ for (l = device_list; l != NULL; l = l->next) {
+ device = (BluezCupsDevice *) l->data;
+
+ if (strcmp(device->bdaddr, bdaddr) == 0) {
+ g_free(device->name);
+ device->name = g_strdup(name);
+ return;
+ }
+ }
+
+ /* Or add it to the list if it's not there */
+ device = g_new0(BluezCupsDevice, 1);
+ device->bdaddr = g_strdup(bdaddr);
+ device->name = g_strdup(name);
+
+ device_list = g_list_prepend(device_list, device);
+}
+
+static void
+print_printer_details(const char *name, const char *bdaddr)
+{
+ 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]);
+ g_print("network %s \"Unknown\" \"%s (Bluetooth)\"\n", uri, escaped);
+ g_free(escaped);
+}
+
+static gboolean
+device_is_printer(DBusGProxy *object, const char *bdaddr)
+{
+ GError *e = NULL;
+ char *class;
+
+ dbus_g_proxy_call(object, "GetRemoteMinorClass", &e,
+ G_TYPE_STRING, bdaddr, G_TYPE_INVALID,
+ G_TYPE_STRING, &class, G_TYPE_INVALID);
+
+ if (e == NULL && g_str_equal(class, PRINTER_SERVICE_CLASS_NAME)) {
+ return TRUE;
+ } else if(e != NULL) {
+ g_error_free(e);
+ }
+
+ return FALSE;
+}
+
+static void
+add_devices_to_list(DBusGProxy *object, const char **array)
+{
+ GError *e = NULL;
+ while (*array) {
+ const char *name;
+
+ if (!device_is_printer(object, *array)) {
+ array++;
+ continue;
+ }
+ dbus_g_proxy_call (object, "GetRemoteName", &e,
+ G_TYPE_STRING, *array, G_TYPE_INVALID,
+ G_TYPE_STRING, &name, G_TYPE_INVALID);
+ if (e == NULL) {
+ add_device_to_list (name, *array);
+ } else {
+ g_error_free (e);
+ e = NULL;
+ }
+ array++;
+ }
+}
+
+static void
+remote_device_found(DBusGProxy *object,
+ const char *bdaddr, guint class, int rssi,
+ gpointer user_data)
+{
+ GError *e = NULL;
+ const char *name;
+
+ /* Don't add non-printer devices */
+ if (!device_is_printer(object, bdaddr))
+ return;
+
+ dbus_g_proxy_call(object, "GetRemoteName", &e,
+ G_TYPE_STRING, bdaddr, G_TYPE_INVALID,
+ G_TYPE_STRING, &name, G_TYPE_INVALID);
+
+ if(e != NULL) {
+ const char *name;
+
+ name = dbus_g_error_get_name(e);
+ if (g_str_equal(name, "org.bluez.Error.RequestDeferred") != FALSE) {
+ add_device_to_list(NULL, bdaddr);
+ }
+ g_error_free(e);
+ } else {
+ add_device_to_list(name, bdaddr);
+ }
+}
+
+static void
+remote_name_updated(DBusGProxy *object,
+ const char *bdaddr, const char *name,
+ gpointer user_data)
+{
+ add_device_to_list(name, bdaddr);
+}
+
+static void
+discovery_completed(DBusGProxy *object, gpointer user_data)
+{
+ GList *l;
+ guint num_unnamed = 1;
+
+ for (l = device_list; l != NULL; l = l->next) {
+ BluezCupsDevice *device = (BluezCupsDevice *) l->data;
+
+ if (device->name == NULL) {
+ device->name = g_strdup_printf ("Unnamed printer #%d",
+ num_unnamed);
+ num_unnamed++;
+ }
+ print_printer_details(device->name, device->bdaddr);
+ g_free(device->name);
+ g_free(device->bdaddr);
+ g_free(device);
+ }
+
+ g_list_free(device_list);
+ device_list = NULL;
+
+ g_main_loop_quit(loop);
+}
+
+static void
+remote_device_disappeared(DBusGProxy *object,
+ const char *bdaddr,
+ gpointer user_data)
+{
+ GList *l;
+
+ for (l = device_list; l != NULL; l = l->next) {
+ BluezCupsDevice *device = (BluezCupsDevice *) l->data;
+
+ if (strcmp(device->bdaddr, bdaddr) == 0) {
+ g_free(device->name);
+ g_free(device->bdaddr);
+ g_free(device);
+ device_list = g_list_delete_link(device_list, l);
+ return;
+ }
+ }
+}
+
+static gboolean
+list_known_printers(DBusGProxy *object)
+{
+ GError *e = NULL;
+ const char **array;
+
+ dbus_g_proxy_call (object, "ListRemoteDevices", &e,
+ G_TYPE_INVALID, G_TYPE_STRV, &array, G_TYPE_INVALID);
+ if (e == NULL) {
+ add_devices_to_list (object, array);
+ } else {
+ /* Most likely bluez-utils < 3.8, so no ListRemoteDevices */
+ const char *name;
+
+ name = dbus_g_error_get_name (e);
+ if (g_str_equal (name, "org.bluez.Error.UnknownMethod") != FALSE) {
+ g_error_free (e);
+ e = NULL;
+ dbus_g_proxy_call (object, "ListBondings", &e,
+ G_TYPE_INVALID, G_TYPE_STRV, &array, G_TYPE_INVALID);
+ if (e == NULL) {
+ add_devices_to_list (object, array);
+ } else {
+ g_error_free (e);
+ }
+ } else {
+ g_error_free (e);
+ }
+ }
+
+ return FALSE;
+}
+
+static void
+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
+ */
+ GError *e = NULL;
+ DBusGConnection *conn;
+ DBusGProxy *object;
+ const char *adapter;
+
+ g_type_init();
+
+ conn = dbus_g_bus_get(DBUS_BUS_SYSTEM, &e);
+ if (e != NULL) {
+ g_warning("Couldn't connect to bus: %s", e->message);
+ g_error_free(e);
+ return;
+ }
+
+ object = dbus_g_proxy_new_for_name(conn, "org.bluez",
+ "/org/bluez", "org.bluez.Manager");
+ dbus_g_proxy_call(object, "DefaultAdapter", &e,
+ G_TYPE_INVALID, G_TYPE_STRING, &adapter, G_TYPE_INVALID);
+ if (e != NULL) {
+ const char *name;
+
+ name = dbus_g_error_get_name(e);
+
+ /* No adapter */
+ if (g_str_equal(name, "org.bluez.Error.NoSuchAdapter") != FALSE) {
+ g_error_free(e);
+ return;
+ }
+
+ g_warning("Couldn't get default bluetooth adapter: %s",
+ e->message);
+ g_error_free(e);
+ return;
+ }
+
+ object = dbus_g_proxy_new_for_name(conn, "org.bluez",
+ adapter, "org.bluez.Adapter");
+
+ dbus_g_object_register_marshaller(bluez_cups_marshal_VOID__STRING_UINT_INT,
+ G_TYPE_NONE, G_TYPE_STRING, G_TYPE_UINT,
+ G_TYPE_INT, G_TYPE_INVALID);
+ dbus_g_proxy_add_signal(object, "RemoteDeviceFound",
+ G_TYPE_STRING, G_TYPE_UINT, G_TYPE_INT, G_TYPE_INVALID);
+ dbus_g_proxy_connect_signal(object, "RemoteDeviceFound",
+ G_CALLBACK(remote_device_found), NULL, NULL);
+
+ dbus_g_object_register_marshaller(bluez_cups_marshal_VOID__STRING_STRING,
+ G_TYPE_NONE, G_TYPE_STRING,
+ G_TYPE_STRING, G_TYPE_INVALID);
+ dbus_g_proxy_add_signal(object, "RemoteNameUpdated",
+ G_TYPE_STRING, G_TYPE_STRING, G_TYPE_INVALID);
+ dbus_g_proxy_connect_signal(object, "RemoteNameUpdated",
+ G_CALLBACK(remote_name_updated), NULL, NULL);
+
+ dbus_g_proxy_add_signal(object, "RemoteDeviceDisappeared",
+ G_TYPE_STRING, G_TYPE_INVALID);
+ dbus_g_proxy_connect_signal(object, "RemoteDeviceDisappeared",
+ G_CALLBACK(remote_device_disappeared), NULL, NULL);
+
+ dbus_g_proxy_add_signal(object, "DiscoveryCompleted", G_TYPE_INVALID);
+ dbus_g_proxy_connect_signal(object, "DiscoveryCompleted",
+ G_CALLBACK(discovery_completed), NULL, NULL);
+
+ dbus_g_proxy_call(object, "DiscoverDevicesWithoutNameResolving", &e,
+ G_TYPE_INVALID, G_TYPE_INVALID);
+ if (e != NULL) {
+ g_warning("Couldn't start discovery: %s: %s",
+ dbus_g_error_get_name(e), e->message);
+ g_error_free(e);
+ return;
+ }
+
+ /* Also add the the recent devices */
+ g_idle_add ((GSourceFunc) list_known_printers, object);
+
+ loop = g_main_loop_new(NULL, TRUE);
+ g_main_loop_run(loop);
+}
+
/*
* Usage: printer-uri job-id user title copies options [file]
*
@@ -73,7 +378,7 @@ int main(int argc, char *argv[])
#endif /* HAVE_SIGSET */
if (argc == 1) {
- puts("network bluetooth \"Unknown\" \"Bluetooth printer\"");
+ list_printers();
return 0;
}
--- /dev/null 2007-05-24 12:42:54.675807888 +0100
+++ cups/bluez-cups-marshal.list 2007-05-23 11:51:07.000000000 +0100
@@ -0,0 +1,2 @@
+VOID:STRING,UINT,INT
+VOID:STRING,STRING
[-- Attachment #3: Type: text/plain, Size: 286 bytes --]
-------------------------------------------------------------------------
This SF.net email is sponsored by DB2 Express
Download DB2 Express C - the FREE version of DB2 express and take
control of your XML. No limits. Just data. Click to get it now.
http://sourceforge.net/powerbar/db2/
[-- Attachment #4: Type: text/plain, Size: 164 bytes --]
_______________________________________________
Bluez-devel mailing list
Bluez-devel@lists.sourceforge.net
https://lists.sourceforge.net/lists/listinfo/bluez-devel
next prev parent reply other threads:[~2007-05-24 13:39 UTC|newest]
Thread overview: 21+ messages / expand[flat|nested] mbox.gz Atom feed top
2007-05-23 15:24 [Bluez-devel] CUPS plugin discovery bits Bastien Nocera
2007-05-23 15:51 ` Marcel Holtmann
2007-05-23 15:57 ` Tim Waugh
2007-05-23 16:15 ` Bastien Nocera
2007-05-23 16:26 ` Bastien Nocera
2007-05-23 16:56 ` Johan Hedberg
2007-05-23 17:07 ` Bastien Nocera
2007-05-24 13:39 ` Bastien Nocera [this message]
2007-05-24 15:31 ` Johan Hedberg
2007-05-24 18:49 ` Bastien Nocera
2007-05-25 12:30 ` Bastien Nocera
2007-05-25 12:58 ` Bastien Nocera
2007-05-25 15:08 ` Bastien Nocera
2007-05-29 3:58 ` Luiz Augusto von Dentz
2007-05-29 10:26 ` Bastien Nocera
2007-05-29 13:18 ` Luiz Augusto von Dentz
2007-05-29 13:37 ` Marcel Holtmann
2007-06-07 10:21 ` Marcel Holtmann
2007-06-07 12:59 ` Bastien Nocera
2007-05-24 15:03 ` Bastien Nocera
2007-05-24 15:20 ` Marcel Holtmann
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=1180013992.3151.6.camel@cookie.hadess.net \
--to=hadess@hadess.net \
--cc=bluez-devel@lists.sourceforge.net \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox