C语言 如何确定 Linux 控制消息中多个辅助消息的大小



我尝试一次通过 unix 套接字发送多个文件描述符。这对于单个插槽来说没有问题。尽管当我尝试用CMSG_NXTHDR附加另一个时,我得到一个空指针,指示我的缓冲区太短。对于一个文件描述符,我用CMSG_SPACE计算缓冲区大小,并假设我只需要乘以这个。虽然这似乎还不够。 我写了一个简短的测试程序来检查这一点:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <sys/types.h>
int main(int argc, char *argv[])
{
struct msghdr msg;
if(argc != 2){
return 1;
}
union {
struct cmsghdr cmsghdr;
char control[CMSG_SPACE(sizeof(int)) * atoi(argv[1])];
} cmsgu;
struct cmsghdr *cmsg;
msg.msg_name = 0;
msg.msg_namelen = 0;
msg.msg_iov = 0;
msg.msg_iovlen = 0;
msg.msg_control = cmsgu.control;
msg.msg_controllen = sizeof(cmsgu.control);
cmsg = CMSG_FIRSTHDR(&msg);
cmsg->cmsg_len = CMSG_LEN(sizeof(int));
cmsg->cmsg_level = SOL_SOCKET;
cmsg->cmsg_type = SCM_RIGHTS;
*((int *)CMSG_DATA(cmsg)) = -1;
cmsg = CMSG_NXTHDR(&msg, cmsg);
fprintf(stderr, "%pn", cmsg);
return 0;
}

当我用 1 调用它时,它会输出一个预期的空指针。虽然我的期望是,如果用 2 调用它,CMSG_NXTHDR将返回一个有效的指针。第一个工作值是 5(因此额外的 120 字节(。我以为CMSG_SPACE会处理好这件事。有没有办法计算所需的空间?或者有没有更直接的方法在一条消息中发送多个文件描述符?

请参阅此问题和答案。如果你 memset(( cmsgu 到 0s,你的代码应该可以工作。

但是(为了回答您的第二个问题(,如果要传递多个文件描述符,则可以使用包含整数数组的单个cmsghdr。修改您的示例,

int n = atoi(argv[1]);
int myfds[n]; // use this later
union {
struct cmsghdr cmsghdr; // for alignment
char control[CMSG_SPACE(sizeof(int) * n)]; // space for one message
} cmsgu;
struct cmsghdr *cmsg;
memset(&cmsgu, 0, sizeof(cmsgu)); // as noted
msg.msg_name = 0;
msg.msg_namelen = 0;
msg.msg_iov = 0;
msg.msg_iovlen = 0;
msg.msg_control = cmsgu.control;
msg.msg_controllen = sizeof(cmsgu.control);
cmsg = CMSG_FIRSTHDR(&msg);
cmsg->cmsg_len = CMSG_LEN(sizeof(int) * n);
cmsg->cmsg_level = SOL_SOCKET;
cmsg->cmsg_type = SCM_RIGHTS;
int *fdptr = (int *) CMSG_DATA(cmsg);
memcpy(fdptr, myfds, sizeof(int) * n);

这与 cmsg(3( 手册页中的示例基本相同。

最新更新