Уровень совместимости Avahi dns_sd не может запустить обратный вызов Browse

Задний план

Я работаю над кросс-платформенной библиотекой Zeroconf/Bonjour/DNS-SD для Haskell и решил, что лучше всего будет выбрать dns_sd.h API. В Linux реализация этого интерфейса обеспечивается Avahi, которая утверждает, что поддерживает подмножество API Bonjour. .

Проблема

В качестве проверки работоспособности моей библиотеки я написал небольшую тестовую программу на C, которая просто использует костяк API. Он просматривает любой сервис в сети типа _http._tcp, печатает сообщение, как только его видит, а затем умирает:

#include <dns_sd.h>
#include <stdio.h>
#include <stdlib.h>

void cb(DNSServiceRef sdRef,
        DNSServiceFlags flags,
        uint32_t interfaceIndex,
        DNSServiceErrorType errorCode,
        const char *serviceName,
        const char *regtype,
        const char *replyDomain,
        void *context) {
  printf("called!\n");
}

int main() {
  DNSServiceRef sd = malloc(sizeof(DNSServiceRef));
  const char *regtype = "_http._tcp";
  DNSServiceErrorType err1 = DNSServiceBrowse(&sd, 0, 0, regtype, NULL, &cb, NULL);
  printf("err1=%d\n", err1);
  DNSServiceErrorType err2 = DNSServiceProcessResult(sd);
  printf("err2=%d\n", err2);
  return 0;
}

На моем Mac эта тестовая программа отлично работает как на C, так и на эквивалентном Haskell (она находит мой принтер; интересно!):

$ gcc test.c -o test
$ ./test
err1=0
called!
err2=0

Но на моей машине с Linux программа ругает меня перед выходом без вызова обратного вызова:

$ gcc test.c -o test -ldns_sd
$ ./test
*** WARNING *** The program 'test' uses the Apple Bonjour compatibility layer of Avahi.
*** WARNING *** Please fix your application to use the native API of Avahi!
*** WARNING *** For more information see <http://0pointer.de/avahi-compat?s=libdns_sd&e=test>
err1=0
err2=0

Вопросы

  1. Является ли уровень совместимости Avahi dns_sd подходящей целью для межплатформенной привязки? Или это предупреждающее сообщение достаточно серьезно относится к использованию собственного API Avahi, и мне следует подумать о перенацеливании?
  2. Каково современное состояние кроссплатформенного Zeroconf на C?

person acfoltzer    schedule 12.09.2011    source источник
comment
Извините, я мало знаю об Avahi, поэтому не могу вам помочь. Сразу вопрос: в инструкции DNSServiceRef sd = malloc(sizeof(DNSServiceRef)); мне кажется довольно странным. Кажется, что DNSServiceRef — это указатель на что-то (видимо, на структуру), но он инициализируется указателем на блок памяти размером с такой же указатель… правильно ли это?   -  person Giuseppe Guerrini    schedule 18.09.2011
comment
@ Джузеппе Геррини: да; Раньше я пробовал только с указателем и переключился на malloc на случай, если по какой-то странной причине это имело значение. Обе версии работали одинаково.   -  person acfoltzer    schedule 18.09.2011
comment
setenv(AVAHI_COMPAT_NOWARN, 1, 0); // отключаем надоедливое предупреждение!   -  person RushPL    schedule 05.10.2011


Ответы (1)


По неизвестной мне причине работает только с неблокирующими звонками. Ниже приведен улучшенный код. Сокет от Avahi устанавливается в неблокирующий режим, а затем используется select (3) для ожидания доступных данных. DNSServiceProcessResult(sd) нужно вызывать каждый раз, когда в сокете есть данные, поэтому, возможно, вам повезло, что ваш пример работал на других платформах.

#include <fcntl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <dns_sd.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>

static int set_nonblocking(int fd)
{
    int flags;
    /* If they have O_NONBLOCK, use the Posix way to do it */
#if defined(O_NONBLOCK)
    /* Fixme: O_NONBLOCK is defined but broken on SunOS 4.1.x and AIX 3.2.5. */
    if (-1 == (flags = fcntl(fd, F_GETFL, 0)))
        flags = 0;
    return fcntl(fd, F_SETFL, flags | O_NONBLOCK);
#else
    /* Otherwise, use the old way of doing it */
    flags = 1;
    return ioctl(fd, FIOBIO, &flags);
#endif
}     

void cb(DNSServiceRef sdRef,
        DNSServiceFlags flags,
        uint32_t interfaceIndex,
        DNSServiceErrorType errorCode,
        const char *serviceName,
        const char *regtype,
        const char *replyDomain,
        void *context) {
    printf("called %s %s!\n", serviceName, regtype);
}

int main() {
    DNSServiceRef sd = malloc(sizeof(DNSServiceRef));
    const char *regtype = "_http._tcp";
    DNSServiceErrorType err1 = DNSServiceBrowse(&sd, 0, 0, regtype, NULL, &cb, NULL);
    printf("err1=%d\n", err1);
    int socket = DNSServiceRefSockFD(sd);
    set_nonblocking(socket);

    fd_set read_fds;
    FD_ZERO(&read_fds);
    FD_SET(socket, &read_fds);

    while(1) {
        if(select(socket+1, &read_fds, NULL, NULL, NULL)  < 0) {
            perror("select");  
        }
        DNSServiceErrorType err2 = DNSServiceProcessResult(sd);
        printf("err2=%d\n", err2);
        if(err2 != 0)
            return 2;
    }
    return 0;
}
person RushPL    schedule 05.10.2011