Как передать структуру в методbrowse_callback, а затем перейти к resolve_callback в avahi?

avatar
Divyasheel
9 августа 2021 в 01:59
159
1
1

Я хочу заполнять виджет всякий раз, когда служба успешно разрешается в avahi. Поскольку большая часть информации, необходимой для заполнения, доступна внутри resolve_callback(), мне нужно, чтобы этот виджет присутствовал внутри него. Но resolve_callback(), являющийся обратным вызовом, не принимает никаких аргументов, которые я могу изменить. Он автоматически вызывается avahi_service_resolver_new().

static void resolve_callback(
    AvahiServiceResolver *r,
    AVAHI_GCC_UNUSED AvahiIfIndex interface,
    AVAHI_GCC_UNUSED AvahiProtocol protocol,
    AvahiResolverEvent event,
    const char *name,
    const char *type,
    const char *domain,
    const char *host_name,
    const AvahiAddress *address,
    uint16_t port,
    AvahiStringList *txt,
    AvahiLookupResultFlags flags,
    PpDnsWindow *self)
{
    assert(r);
    /* Called whenever a service has been resolved successfully or timed out */
    switch (event)
    {
    case AVAHI_RESOLVER_FAILURE:
         fprintf(stderr, "(Resolver) Failed to resolve service '%s' of type '%s' in domain '%s': %s\n", name, type, domain, avahi_strerror(avahi_client_errno(avahi_service_resolver_get_client(r))));
         break;
    case AVAHI_RESOLVER_FOUND:
    {
        // here I'll be using the self struct

        char a[AVAHI_ADDRESS_STR_MAX], *t;
        fprintf(stderr, "Service '%s' of type '%s' in domain '%s':\n", name, type, domain);
        avahi_address_snprint(a, sizeof(a), address);
        t = avahi_string_list_to_string(txt);
        fprintf(stderr,
                "\t%s:%u (%s)\n"
                "\tTXT=%s\n"
                "\tcookie is %u\n"
                "\tis_local: %i\n"
                "\tour_own: %i\n"
                "\twide_area: %i\n"
                "\tmulticast: %i\n"
                "\tcached: %i\n",
                host_name, port, a,
                t,
                avahi_string_list_get_service_cookie(txt),
                !!(flags & AVAHI_LOOKUP_RESULT_LOCAL),
                !!(flags & AVAHI_LOOKUP_RESULT_OUR_OWN),
                !!(flags & AVAHI_LOOKUP_RESULT_WIDE_AREA),
                !!(flags & AVAHI_LOOKUP_RESULT_MULTICAST),
                !!(flags & AVAHI_LOOKUP_RESULT_CACHED));
        avahi_free(t);
    }
    }
    avahi_service_resolver_free(r);
}

static void browse_callback(
    AvahiServiceBrowser *b,
    AvahiIfIndex interface,
    AvahiProtocol protocol,
    AvahiBrowserEvent event,
    const char *name,
    const char *type,
    const char *domain,
    AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
    PpDnsWindow *self)
{
    AvahiClient *c = self->client;
    assert(b);
    /* Called whenever a new services becomes available on the LAN or is removed from the LAN */
    switch (event)
    {
    case AVAHI_BROWSER_FAILURE:
        fprintf(stderr, "(Browser) %s\n", avahi_strerror(avahi_client_errno(avahi_service_browser_get_client(b))));
        return;
    case AVAHI_BROWSER_NEW:
        fprintf(stderr, "(Browser) NEW: service '%s' of type '%s' in domain '%s'\n", name, type, domain);
        /* We ignore the returned resolver object. In the callback
               function we free it. If the server is terminated before
               the callback function is called the server will free
               the resolver for us. */
        if (!(avahi_service_resolver_new(c, interface, protocol, name, type, domain, AVAHI_PROTO_UNSPEC, 0, resolve_callback, self)))
            fprintf(stderr, "Failed to resolve service '%s': %s\n", name, avahi_strerror(avahi_client_errno(c)));
        break;
    case AVAHI_BROWSER_REMOVE:
        fprintf(stderr, "(Browser) REMOVE: service '%s' of type '%s' in domain '%s'\n", name, type, domain);
        break;
    case AVAHI_BROWSER_ALL_FOR_NOW:
    case AVAHI_BROWSER_CACHE_EXHAUSTED:
        fprintf(stderr, "(Browser) %s\n", event == AVAHI_BROWSER_CACHE_EXHAUSTED ? "CACHE_EXHAUSTED" : "ALL_FOR_NOW");
        break;
    }
}

Я попытался изменить AVAHI_GCC_UNUSED void* userdata на PpDnsWindow* self в этом определении функции, а затем передал PpDnsWindow вниз по этим функциям, например так-

/* Create the service browser */
  if (!(sb = avahi_service_browser_new(self->client, AVAHI_IF_UNSPEC, AVAHI_PROTO_UNSPEC, "_ipp._tcp", NULL, 0, browse_callback, self)))
  {
      fprintf(stderr, "Failed to create service browser: %s\n", avahi_strerror(avahi_client_errno(self->client)));
      destroy(self);
  }

Но это приводит к разрешению сбоя с помощью AVAHI_RESOLVER_FAILURE.

Я не хочу делать виджет глобальным, так как глобальные обычно не рекомендуются. Можно ли как-то добиться этого?

Источник
kaylum
9 августа 2021 в 02:03
0

Обратные вызовы обычно предоставляют контекстный указатель на обратный вызов. В данном случае это последний параметр для avahi_service_resolver_new. Это помогает?

Divyasheel
9 августа 2021 в 02:21
0

@kaylum Я добавил код, в котором пытаюсь использовать свое небольшое понимание того, как работают указатели контекста. Я не знаю, должен ли я изменить сигнатуру функции, чтобы приспособиться к вводу.

Ответы (1)

avatar
Divyasheel
9 августа 2021 в 03:08
0

Я также должен был добавить AVAHI_GCC_UNUSED перед PpDnsWindow *self при изменении сигнатуры функции для resolve_callback() следующим образом. Но это не требуется для browser_callback().

static void resolve_callback(
    AvahiServiceResolver *r,
    AVAHI_GCC_UNUSED AvahiIfIndex interface,
    AVAHI_GCC_UNUSED AvahiProtocol protocol,
    AvahiResolverEvent event,
    const char *name,
    const char *type,
    const char *domain,
    const char *host_name,
    const AvahiAddress *address,
    uint16_t port,
    AvahiStringList *txt,
    AvahiLookupResultFlags flags,
    AVAHI_GCC_UNUSED PpDnsWindow *self)

Я искал AVAHI_GCC_UNUSED в Интернете, но не смог найти ничего существенного. Буду признателен, если кто-нибудь объяснит, что это значит.