Отправьте дескриптор файла через сокет домена UNIX и выберите () - PullRequest
11 голосов
/ 20 декабря 2010

Я использую сокет домена UNIX для передачи дескриптора файла другому процессу.Это работает нормально, но когда я впервые пытаюсь определить, доступен ли для записи сокет, используя select (), вызов sendmsg () завершается неудачно с ошибкой Bad File Descriptor.

Функция sendmsg () прекрасно работает в сочетании с select (), если я не добавляю информацию о дескрипторе файла в структуру msghdr, так что конфликт, похоже, существует между 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 (), чтобы проверить, готов ли сокет для записи.

1 Ответ

10 голосов
/ 20 декабря 2010

Я попробовал код 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;
        }

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...