From de6fc99ef5a1767fae07fbea9bc384f512639e57 Mon Sep 17 00:00:00 2001 From: Lennart Poettering Date: Sun, 28 Oct 2007 00:17:16 +0000 Subject: split off libavahi-gobject into its own directory git-svn-id: file:///home/lennart/svn/public/avahi/trunk@1564 941a03a8-eaeb-0310-b9a0-b1bbd8fe43fe --- avahi-gobject/ga-service-resolver.c | 403 ++++++++++++++++++++++++++++++++++++ 1 file changed, 403 insertions(+) create mode 100644 avahi-gobject/ga-service-resolver.c (limited to 'avahi-gobject/ga-service-resolver.c') 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 +#endif + +#include +#include + +#include "ga-service-resolver.h" +#include "signals-marshal.h" + +#include "ga-errors.h" + +#include "ga-enums.h" +#include "ga-enums-enumtypes.h" + +#include + +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; +} -- cgit