一:进程间传递的不是文件描述的值
进程间传递打开的文件描述符,并不是传递文件描述符的值。
二:文件描述符
文件描述符的值与文件没有关系,只是文件在该进程中的一个标志。同一个文件在不同进程中的文件描述符的值可能不一样,且一样的文件描述符的值可能指向不同的文件。
三:文件数据结构
一个打开的文件在不同进程中的组织关系及三种数据结构
文件描述符表(项)
存在于进程中,不同的进程有各自的文件描述符表,每个表项存放者文件描述相关的结构,包括fd值即文件表指针。
文件表(项)
存在内核中,进程中每个打开的文件生成的文件描述符表项都在内核会关联一个文件表项,包括
当前文件偏移量,这样才能使每个进程都有它自己的对该文件的当前偏移量。
V结点指针:指向的是同一V结点
3. V结点表(项)
存在于内核中,每个打开的文件只有一个V结点,包含文件类型,对文件进行操作的函数指针,有的还包括 i 节点。
四:使用socket传递文件描述符
创建UNIX域很简单,用socketpair函数即可,难的是如何使用sendmsg 和 recvmsg函数进行发送和接收。两个函数的定义如下:
#include<sys/socket.h>
ssize_t recvmsg(int sockfd, struct msghdr* msg, int flags);
ssize_t sendmsg(int sockfd, struct msghdr* msg, int flags);
参数sockfd的含义不必多说,msg可理解为被发送/接收的数据,在recvmsg中他的行为类似于传出参数,flags参数与recv/send的同名参数含义一样。成功时返回实际发送/接收的字节数,失败返回-1。
该函数最难理解的就是msg的类型 msghdr ,文件描述符就是通过msghdr结构体的msg_control成员发送的,而msg_control又是 cmsghdr 类型
msghdr 结构体
struct msghdr
{
void* msg_name;
socklen_t msg_namelen;
struct iovec* msg_iov;
int msg_iovlen;
void* msg_control;
socklen_t msg_conntrollen;
int msg_flags; 最后这个参数不管
}
辅助(或附属)数据:msg_control 和 msg_controllen
msg_control指向辅助数据起始地址,msg_controllen指明辅助数据的长度。msg_control的类型是struct cmsghdr*,其定义如下:
struct cmsghdr
{
size_t cmsg_len; 辅助数据的总长度,由 CMSG_LEN 宏(马上讲)直接获取
int cmsg_level; 表示通道使用的的原始协议级别,与 setsockopt 函数的 level 参数相同
int cmsg_type; /* Protocol-specific type */控制信息类型,例如,SCM_RIGHTS,辅助数据是文件描述符;SCM_CREDENTIALS,辅助数据是一个包含证书信息的结构
/* followed by unsigned char cmsg_data[]; */被注释的 cmsg_data 指明了物理内存中真正辅助数据的位置,帮助理解
};
参考链接:https://blog.csdn.net/m0_51551385/article/details/124635900