c - 通过 UNIX 域套接字发送文件描述符和 select()

标签 c sockets unix dns

我正在使用 UNIX 域套接字将文件描述符传输到另一个进程。这工作正常,但是当我第一次尝试使用 select() 来查看套接字是否可写时,sendmsg() 调用失败并出现 Bad File Descriptor 错误。

如果我不将文件描述符信息添加到 msghdr 结构,则 sendmsg() 函数与 select() 结合使用时效果很好,因此冲突似乎发生在 select() 和传输文件描述符之间。

我在 select()、recvmsg() 或任何其他的手册页中找不到任何关于此的信息。由于这需要成为一个将文件描述符分发给多个进程的服务器,我仍然希望能够使用 select()。

我能做些什么来完成这项工作,或者有人知道替代解决方案吗?

平台是 Ubuntu 10.4。

这是初始化结构的代码:



struct cmsghdr_fd : public cmsghdr
{
  int fd;
};

int sendfd(int sock, int fd)
{
  struct msghdr hdr;
  struct iovec data;
  struct cmsghdr_fd msgdata;

  char dummy = '*';
  data.iov_base = &dummy;
  data.iov_len = sizeof(dummy);

  hdr.msg_name = NULL;
  hdr.msg_namelen = 0;
  hdr.msg_iov = &data;
  hdr.msg_iovlen = 1;
  hdr.msg_flags = 0;

  hdr.msg_control = &msgdata;
  hdr.msg_controllen = sizeof(msgdata);

  struct cmsghdr* cmsg = CMSG_FIRSTHDR(&hdr);
  cmsg->cmsg_len   = hdr.msg_controllen;
  cmsg->cmsg_level = SOL_SOCKET;
  cmsg->cmsg_type  = SCM_RIGHTS;

  *(int*)CMSG_DATA(cmsg) = fd;

  int n = sendmsg(sock, &hdr, 0);

  if(n == -1)
    printf("sendmsg() failed: %s (socket fd = %d)\n", strerror(errno), sock);

  return n;
}

同样,只要我不首先调用 select() 来检查套接字是否准备好写入,这就可以工作。

最佳答案

我在 this page 尝试了 sendfd 代码,由 nos 友情提供,尽管它只是略有不同,但即使我将它与 select() 结合使用,它也能正常工作。这就是代码现在的样子:



    int sendfd(int sock, int fd)
    {
      struct msghdr hdr;
      struct iovec data;

      char cmsgbuf[CMSG_SPACE(sizeof(int))];

      char dummy = '*';
      data.iov_base = &dummy;
      data.iov_len = sizeof(dummy);

      memset(&hdr, 0, sizeof(hdr));
      hdr.msg_name = NULL;
      hdr.msg_namelen = 0;
      hdr.msg_iov = &data;
      hdr.msg_iovlen = 1;
      hdr.msg_flags = 0;

      hdr.msg_control = cmsgbuf;
      hdr.msg_controllen = CMSG_LEN(sizeof(int));

      struct cmsghdr* cmsg = CMSG_FIRSTHDR(&hdr);
      cmsg->cmsg_len   = CMSG_LEN(sizeof(int));
      cmsg->cmsg_level = SOL_SOCKET;
      cmsg->cmsg_type  = SCM_RIGHTS;

      *(int*)CMSG_DATA(cmsg) = fd;

      int n = sendmsg(sock, &hdr, 0);

      if(n == -1)
        printf("sendmsg() failed: %s (socket fd = %d)\n", strerror(errno), sock);

      return n;
        }

关于c - 通过 UNIX 域套接字发送文件描述符和 select(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4489433/

相关文章:

c - K&R 中边界有限的 qsort |排序过程不会在字符串中的下一个位置继续

c - 为什么这个缓冲区定义在一个循环内?

c++ - 使用 epoll 的套接字服务器在断开连接时给出未知字节

unix - 使用命令行从以上目录中选择Gradle构建文件

java - 如何使用Java在unix上输入文件路径

c - 什么时候指向数组的指针有用?

基本函数的 C 编译错误

c - 如何使用 C 中读取的函数显示文件内容?

python - 如何在不阻塞父进程的情况下在python子进程中运行?

mysql不想启动