Avahi dns_sd 兼容层无法运行浏览回调

Posted

技术标签:

【中文标题】Avahi dns_sd 兼容层无法运行浏览回调【英文标题】:Avahi dns_sd compatibility layer fails to run Browse callback 【发布时间】:2011-11-15 12:08:31 【问题描述】:

背景

我正在为 Haskell 开发一个跨平台的 Zeroconf/Bonjour/DNS-SD 库,并认为我最好的选择是针对 dns_sd.h API。在 Linux 下,这个接口的实现由 Avahi 提供,它声称支持 Bonjour API 的一个子集。

问题

作为对我的库的健全性检查,我用 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

问题

    Avahi dns_sd 兼容层仍然是跨平台绑定的合适目标吗?或者,关于使用本机 Avahi API 的警告消息是否足够严重,我应该考虑重新定位? C 语言中跨平台 Zeroconf 的最新技术是什么?

【问题讨论】:

对不起,我对 Avahi 了解不多,所以帮不了你。只是一个问题:指令“DNSServiceRef sd = malloc(sizeof(DNSServiceRef));”对我来说似乎很奇怪。似乎 DNSServiceRef 是一个指向某物的指针(我想是一个结构),但它是用一个指向同一指针大小的内存块的指针初始化的……对吗? @Giuseppe Guerrini:是的;我之前只用指针试过,然后切换到malloc,以防它因为一些奇怪的原因而有所不同。两个版本的工作方式相同。 setenv("AVAHI_COMPAT_NOWARN", "1", 0); // 禁用烦人的警告! 【参考方案1】:

由于我不知道的原因,它仅适用于非阻塞调用。下面是改进后的代码。 Avahi 的 Socket 设置为非阻塞模式,然后使用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;

【讨论】:

以上是关于Avahi dns_sd 兼容层无法运行浏览回调的主要内容,如果未能解决你的问题,请参考以下文章

使用 python 浏览 avahi 服务会错过服务

如何从 docker 容器中“浏览”?

Tensorflow ValueError:层“顺序”的输入0与层不兼容:预期形状=(无,20,20,3),找到形状=(无,20,3)

360安全浏览器怎么切换兼容模式

Avahi dnsconfd 是如何工作的?

浏览器链接:调用返回值回调失败:TypeError:无法读取 null 的属性“文件”