我正在使用UNIX域套接字将文件描述符传输到另一个进程。这工作正常,但是当我第一次尝试使用select()查看套接字是否可写时,sendmsg()调用失败并显示错误的文件描述符错误。
如果我没有将文件描述符信息添加到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()来检查套接字是否已准备好写入。
答案 0 :(得分:10)
我在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;
}