summaryrefslogtreecommitdiffstats
path: root/avahi-gobject/ga-service-resolver.c
diff options
context:
space:
mode:
Diffstat (limited to 'avahi-gobject/ga-service-resolver.c')
-rw-r--r--avahi-gobject/ga-service-resolver.c403
1 files changed, 403 insertions, 0 deletions
diff --git a/avahi-gobject/ga-service-resolver.c b/avahi-gobject/ga-service-resolver.c
new file mode 100644
index 0000000..380e0a5
--- /dev/null
+++ b/avahi-gobject/ga-service-resolver.c
@@ -0,0 +1,403 @@
+/*
+ * ga-service-resolver.c - Source for GaServiceResolver
+ * Copyright (C) 2005 Collabora Ltd.
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; 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 <stdlib.h>
+
+#include "ga-service-resolver.h"
+#include "signals-marshal.h"
+
+#include "ga-errors.h"
+
+#include "ga-enums.h"
+#include "ga-enums-enumtypes.h"
+
+#include <avahi-client/lookup.h>
+
+G_DEFINE_TYPE(GaServiceResolver, ga_service_resolver, G_TYPE_OBJECT)
+
+/* signal enum */
+enum {
+ FOUND,
+ FAILURE,
+ LAST_SIGNAL
+};
+
+static guint signals[LAST_SIGNAL] = { 0 };
+
+/* properties */
+enum {
+ PROP_PROTOCOL = 1,
+ PROP_IFINDEX,
+ PROP_NAME,
+ PROP_TYPE,
+ PROP_DOMAIN,
+ PROP_FLAGS,
+ PROP_APROTOCOL
+};
+
+/* private structure */
+typedef struct _GaServiceResolverPrivate GaServiceResolverPrivate;
+
+struct _GaServiceResolverPrivate {
+ GaClient *client;
+ AvahiServiceResolver *resolver;
+ AvahiIfIndex interface;
+ AvahiProtocol protocol;
+ AvahiAddress address;
+ uint16_t port;
+ char *name;
+ char *type;
+ char *domain;
+ AvahiProtocol aprotocol;
+ AvahiLookupFlags flags;
+ gboolean dispose_has_run;
+};
+
+#define GA_SERVICE_RESOLVER_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), GA_TYPE_SERVICE_RESOLVER, GaServiceResolverPrivate))
+
+static void ga_service_resolver_init(GaServiceResolver * obj) {
+ GaServiceResolverPrivate *priv = GA_SERVICE_RESOLVER_GET_PRIVATE(obj);
+
+ /* allocate any data required by the object here */
+ priv->client = NULL;
+ priv->resolver = NULL;
+ priv->name = NULL;
+ priv->type = NULL;
+ priv->domain = NULL;
+ priv->port = 0;
+}
+
+static void ga_service_resolver_dispose(GObject * object);
+static void ga_service_resolver_finalize(GObject * object);
+
+static void ga_service_resolver_set_property(GObject * object,
+ guint property_id,
+ const GValue * value, GParamSpec * pspec) {
+ GaServiceResolver *resolver = GA_SERVICE_RESOLVER(object);
+ GaServiceResolverPrivate *priv =
+ GA_SERVICE_RESOLVER_GET_PRIVATE(resolver);
+
+ g_assert(priv->resolver == NULL);
+ switch (property_id) {
+ case PROP_PROTOCOL:
+ priv->protocol = g_value_get_enum(value);
+ break;
+ case PROP_APROTOCOL:
+ priv->aprotocol = g_value_get_enum(value);
+ break;
+ case PROP_IFINDEX:
+ priv->interface = g_value_get_int(value);
+ break;
+ case PROP_NAME:
+ priv->name = g_strdup(g_value_get_string(value));
+ break;
+ case PROP_TYPE:
+ priv->type = g_strdup(g_value_get_string(value));
+ break;
+ case PROP_DOMAIN:
+ priv->domain = g_strdup(g_value_get_string(value));
+ break;
+ case PROP_FLAGS:
+ priv->flags = g_value_get_enum(value);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, property_id, pspec);
+ break;
+ }
+}
+
+static void ga_service_resolver_get_property(GObject * object,
+ guint property_id,
+ GValue * value, GParamSpec * pspec) {
+ GaServiceResolver *resolver = GA_SERVICE_RESOLVER(object);
+ GaServiceResolverPrivate *priv =
+ GA_SERVICE_RESOLVER_GET_PRIVATE(resolver);
+
+ switch (property_id) {
+ case PROP_APROTOCOL:
+ g_value_set_enum(value, priv->aprotocol);
+ break;
+ case PROP_PROTOCOL:
+ g_value_set_enum(value, priv->protocol);
+ break;
+ case PROP_IFINDEX:
+ g_value_set_int(value, priv->interface);
+ break;
+ case PROP_NAME:
+ g_value_set_string(value, priv->name);
+ break;
+ case PROP_TYPE:
+ g_value_set_string(value, priv->type);
+ break;
+ case PROP_DOMAIN:
+ g_value_set_string(value, priv->domain);
+ break;
+ case PROP_FLAGS:
+ g_value_set_enum(value, priv->flags);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, property_id, pspec);
+ break;
+ }
+}
+
+
+static void ga_service_resolver_class_init(GaServiceResolverClass *
+ ga_service_resolver_class) {
+ GObjectClass *object_class = G_OBJECT_CLASS(ga_service_resolver_class);
+ GParamSpec *param_spec;
+
+ g_type_class_add_private(ga_service_resolver_class,
+ sizeof (GaServiceResolverPrivate));
+
+ object_class->set_property = ga_service_resolver_set_property;
+ object_class->get_property = ga_service_resolver_get_property;
+
+ object_class->dispose = ga_service_resolver_dispose;
+ object_class->finalize = ga_service_resolver_finalize;
+
+ signals[FOUND] =
+ g_signal_new("found",
+ G_OBJECT_CLASS_TYPE(ga_service_resolver_class),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL, NULL,
+ ga_signals_marshal_VOID__INT_ENUM_STRING_STRING_STRING_STRING_POINTER_INT_POINTER_INT,
+ G_TYPE_NONE, 10,
+ G_TYPE_INT,
+ GA_TYPE_PROTOCOL,
+ G_TYPE_STRING,
+ G_TYPE_STRING,
+ G_TYPE_STRING,
+ G_TYPE_STRING,
+ G_TYPE_POINTER,
+ G_TYPE_INT,
+ G_TYPE_POINTER, GA_TYPE_LOOKUP_RESULT_FLAGS);
+
+ signals[FAILURE] =
+ g_signal_new("failure",
+ G_OBJECT_CLASS_TYPE(ga_service_resolver_class),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL, NULL,
+ g_cclosure_marshal_VOID__POINTER,
+ G_TYPE_NONE, 1, G_TYPE_POINTER);
+
+ param_spec = g_param_spec_enum("protocol", "Avahi protocol to resolve on",
+ "Avahi protocol to resolve on",
+ GA_TYPE_PROTOCOL,
+ GA_PROTOCOL_UNSPEC,
+ G_PARAM_READWRITE |
+ G_PARAM_STATIC_NAME |
+ G_PARAM_STATIC_BLURB);
+ g_object_class_install_property(object_class, PROP_PROTOCOL, param_spec);
+
+ param_spec = g_param_spec_enum("aprotocol", "Address protocol",
+ "Avahi protocol of the address to be resolved",
+ GA_TYPE_PROTOCOL,
+ GA_PROTOCOL_UNSPEC,
+ G_PARAM_READWRITE |
+ G_PARAM_STATIC_NAME |
+ G_PARAM_STATIC_BLURB);
+ g_object_class_install_property(object_class, PROP_APROTOCOL, param_spec);
+
+ param_spec = g_param_spec_int("interface", "interface index",
+ "Interface use for resolver",
+ AVAHI_IF_UNSPEC,
+ G_MAXINT,
+ AVAHI_IF_UNSPEC,
+ G_PARAM_READWRITE |
+ G_PARAM_STATIC_NAME | G_PARAM_STATIC_BLURB);
+ g_object_class_install_property(object_class, PROP_IFINDEX, param_spec);
+
+ param_spec = g_param_spec_string("name", "service name",
+ "name to resolve",
+ NULL,
+ G_PARAM_READWRITE |
+ G_PARAM_STATIC_NAME |
+ G_PARAM_STATIC_BLURB);
+ g_object_class_install_property(object_class, PROP_NAME, param_spec);
+
+ param_spec = g_param_spec_string("type", "service type",
+ "Service type to browse for",
+ NULL,
+ G_PARAM_READWRITE |
+ G_PARAM_STATIC_NAME |
+ G_PARAM_STATIC_BLURB);
+ g_object_class_install_property(object_class, PROP_TYPE, param_spec);
+
+ param_spec = g_param_spec_string("domain", "service domain",
+ "Domain to browse in",
+ NULL,
+ G_PARAM_READWRITE |
+ G_PARAM_STATIC_NAME |
+ G_PARAM_STATIC_BLURB);
+ g_object_class_install_property(object_class, PROP_DOMAIN, param_spec);
+
+ param_spec = g_param_spec_enum("flags", "Lookup flags for the resolver",
+ "Resolver lookup flags",
+ GA_TYPE_PROTOCOL,
+ GA_LOOKUP_NO_FLAGS,
+ G_PARAM_READWRITE |
+ G_PARAM_STATIC_NAME |
+ G_PARAM_STATIC_BLURB);
+ g_object_class_install_property(object_class, PROP_FLAGS, param_spec);
+}
+
+void ga_service_resolver_dispose(GObject * object) {
+ GaServiceResolver *self = GA_SERVICE_RESOLVER(object);
+ GaServiceResolverPrivate *priv = GA_SERVICE_RESOLVER_GET_PRIVATE(self);
+
+ if (priv->dispose_has_run)
+ return;
+
+ priv->dispose_has_run = TRUE;
+
+ if (priv->client)
+ g_object_unref(priv->client);
+ priv->client = NULL;
+
+ if (priv->resolver)
+ avahi_service_resolver_free(priv->resolver);
+ priv->resolver = NULL;
+
+ /* release any references held by the object here */
+
+ if (G_OBJECT_CLASS(ga_service_resolver_parent_class)->dispose)
+ G_OBJECT_CLASS(ga_service_resolver_parent_class)->dispose(object);
+}
+
+void ga_service_resolver_finalize(GObject * object) {
+ GaServiceResolver *self = GA_SERVICE_RESOLVER(object);
+ GaServiceResolverPrivate *priv = GA_SERVICE_RESOLVER_GET_PRIVATE(self);
+
+ /* free any data held directly by the object here */
+ g_free(priv->name);
+ priv->name = NULL;
+
+ g_free(priv->type);
+ priv->type = NULL;
+
+ g_free(priv->domain);
+ priv->domain = NULL;
+
+ G_OBJECT_CLASS(ga_service_resolver_parent_class)->finalize(object);
+}
+
+static void _avahi_service_resolver_cb(AVAHI_GCC_UNUSED AvahiServiceResolver * resolver,
+ AvahiIfIndex interface,
+ AvahiProtocol protocol,
+ AvahiResolverEvent event,
+ const char *name, const char *type,
+ const char *domain, const char *host_name,
+ const AvahiAddress * a,
+ uint16_t port,
+ AvahiStringList * txt,
+ AvahiLookupResultFlags flags, void *userdata) {
+ GaServiceResolver *self = GA_SERVICE_RESOLVER(userdata);
+ GaServiceResolverPrivate *priv = GA_SERVICE_RESOLVER_GET_PRIVATE(self);
+
+ switch (event) {
+ case AVAHI_RESOLVER_FOUND:{
+ /* FIXME: Double check if this address is always the same */
+ priv->address = *a;
+ priv->port = port;
+ g_signal_emit(self, signals[FOUND], 0,
+ interface, protocol,
+ name, type,
+ domain, host_name, a, port, txt, flags);
+ break;
+ }
+ case AVAHI_RESOLVER_FAILURE:{
+ GError *error;
+ int aerrno = avahi_client_errno(priv->client->avahi_client);
+ error = g_error_new(GA_ERRORS, aerrno,
+ "Resolving failed: %s",
+ avahi_strerror(aerrno));
+ g_signal_emit(self, signals[FAILURE], 0, error);
+ g_error_free(error);
+ break;
+ }
+ }
+}
+
+
+GaServiceResolver *ga_service_resolver_new(AvahiIfIndex interface,
+ AvahiProtocol protocol,
+ const gchar * name,
+ const gchar * type,
+ const gchar * domain,
+ AvahiProtocol address_protocol,
+ GaLookupFlags flags) {
+ return g_object_new(GA_TYPE_SERVICE_RESOLVER, "interface", interface,
+ "protocol", protocol, "name", name, "type", type,
+ "domain", domain, "aprotocol", address_protocol,
+ "flags", flags, NULL);
+}
+
+gboolean ga_service_resolver_attach(GaServiceResolver * resolver,
+ GaClient * client, GError ** error) {
+ GaServiceResolverPrivate *priv =
+ GA_SERVICE_RESOLVER_GET_PRIVATE(resolver);
+
+ g_assert(client != NULL);
+ g_object_ref(client);
+
+ priv->client = client;
+
+ priv->resolver = avahi_service_resolver_new(client->avahi_client,
+ priv->interface,
+ priv->protocol,
+ priv->name,
+ priv->type, priv->domain,
+ priv->aprotocol,
+ priv->flags,
+ _avahi_service_resolver_cb,
+ resolver);
+ if (priv->resolver == NULL) {
+ if (error != NULL) {
+ int aerrno = avahi_client_errno(client->avahi_client);
+ *error = g_error_new(GA_ERRORS, aerrno,
+ "Attaching group failed: %s",
+ avahi_strerror(aerrno));
+ }
+/* printf("Failed to add resolver\n"); */
+ return FALSE;
+ }
+ return TRUE;
+}
+
+gboolean ga_service_resolver_get_address(GaServiceResolver * resolver,
+ AvahiAddress * address, uint16_t * port) {
+ GaServiceResolverPrivate *priv =
+ GA_SERVICE_RESOLVER_GET_PRIVATE(resolver);
+ if (priv->port == 0) {
+/* printf("PORT == 0\n"); */
+ return FALSE;
+ }
+
+ *address = priv->address;
+ *port = priv->port;
+ return TRUE;
+}