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