/* * ga-service-resolver.c - Source for GaServiceResolver * Copyright (C) 2006-2007 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-error.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_LOOKUP_FLAGS, 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_ERROR, 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_ERROR, 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; }