我正在使用UNIX域套接字将文件描述符传输到另一个进程。这个方法是有效的,但当我首次尝试使用select()检测套接字是否可写时,sendmsg()调用会失败,并出现Bad File Descriptor错误。
如果我不将文件描述符信息添加到msghdr结构中,那么sendmsg()函数与select()一起正常工作,因此冲突似乎出现在select()和传输文件描述符之间。
我在select()、recvmsg()或其他任何文档中都没有找到有关此问题的信息。由于需要将其变成一个向多个进程分发文件描述符的服务器,我仍然想能够使用select()。
有没有办法使这个方法能够正常工作,或者是否有其他替代方案?
平台为Ubuntu 10.4。
这是初始化结构的代码:
如果我不将文件描述符信息添加到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()检查套接字是否准备好写入,这段代码就能正常工作。
msghdr
和cmsghdr
的代码? - Nikolai FetissovCMSG_LEN
宏来设置cmsg_len
,但我不认为这是实际的问题所在... - Hasturkunselect()
的使用方法。 - Nikolai Fetissov