继续学习socket编程,今天的内容会有些难以理解,一步步来分解,也就不难了,正入正题:

linux网络编程之socket编程(十六)_数据

 linux网络编程之socket编程(十六)_数据_02

实际上sockpair有点像之前linux系统编程中学习的pipe匿名管道匿名管道是半双工的,只能用于亲缘关系的进程间进行通信,也就是说父子进程或兄弟进程间进行通讯,因为它是没有名称的,父子进程可以通过共享描述符的方式来进行通信,子进程继承了父进程的文件描述符,从而达到了通信的目的。而今天学习的sockpair是一个全双工的流管道,其它也一样,也只能用于父子进程或亲缘关系之间进行通讯,所以其中sv套接字对就很容易理解了,但是跟pipe有些区别,先来回顾下pipe:

linux网络编程之socket编程(十六)_#include_03

其中的fd也是套接字对,一端表示读,一端表示写,而sockpair中的sv[0]、sv[1]两端分别既可以表示读端,也可以表示写端。

认识了sockpair函数原形,下面用程序来说明它的用法:

首先创业一个套接字对:

linux网络编程之socket编程(十六)_数据_04

由于它也是只能用于父子进程或亲缘关系之间进行通讯,所以需要fork进程出来:

linux网络编程之socket编程(十六)_#include_05

下面就来实现父子进程进行通讯:

linux网络编程之socket编程(十六)_父子进程_06

而对于子进程而言,代码基本类似:

linux网络编程之socket编程(十六)_父子进程_07

编译运行看结果:

linux网络编程之socket编程(十六)_数据_08

从结果运行来看,通过sockpair就完成了全双工的通讯。

linux网络编程之socket编程(十六)_#include_09

学习这两个函数的目的,是为了通过UNIX域协议如何传递文件描述字,关于这个函数的使用会比较复杂,需慢慢理解。

首先来查看一下man帮助:

linux网络编程之socket编程(十六)_#include_10

其中第二个参数是msghdr结构体,所以有必要来研究一下这个结构体:

linux网络编程之socket编程(十六)_#include_11

哇,这个结构体貌似挺复杂的,下面一一来熟悉其字段含义:

linux网络编程之socket编程(十六)_数据_12

linux网络编程之socket编程(十六)_数据_13

这时,需要来看另外一个函数了,该结构体在其中有介绍到:

linux网络编程之socket编程(十六)_父子进程_14

那怎么理解该参数呢?这个需要从send函数来分析:

linux网络编程之socket编程(十六)_父子进程_15

所以iovec结构体的字段就可以从send的这两个参数来理解:

linux网络编程之socket编程(十六)_父子进程_16

并且,可以发现:

linux网络编程之socket编程(十六)_#include_17

下面来看一个示意图:

linux网络编程之socket编程(十六)_#include_18

linux网络编程之socket编程(十六)_#include_19

从上面示意图中可以发现,如果用sendmsg函数,就可以发送多个缓冲区的数据了,而如果用send只能发送一个缓冲区,所以从这也可以看出sendmsg的强大。

如果说要传递文件描述字,还需要发送一些辅助的数据,这些辅助数据是一些控制信息,也就是下面这些参数:

linux网络编程之socket编程(十六)_数据_20

linux网络编程之socket编程(十六)_父子进程_21

而其中msg_control是指向一个结构体,那它长啥样呢?需要从另外一个函数的帮助文档中得知:

linux网络编程之socket编程(十六)_数据_22

那具体属性的含议是啥呢?

linux网络编程之socket编程(十六)_父子进程_23

实际上,在填充这些数据的时候,并没有这么简单,它还会按照一定的方式来进行对齐,接下来再来看另外一个示意图---辅助数据的示意图:

linux网络编程之socket编程(十六)_#include_24

linux网络编程之socket编程(十六)_父子进程_25

linux网络编程之socket编程(十六)_#include_26

其中可以看到定义了一些宏,这是由于:

linux网络编程之socket编程(十六)_数据_27

所以,下面来认识一下这些宏定义:

linux网络编程之socket编程(十六)_数据_28

其中"size_t CMSG_SPACE(size_t length)",结合图来说明就是:

linux网络编程之socket编程(十六)_数据_29

大致了解了以上这些数据结构,下面则可以开始编写代码来传递描述字了,但是代码会比较复杂,可以一步步来理解,下面开始。

linux网络编程之socket编程(十六)_父子进程_30

实际上,就是能过以下两个函数来封装发送和接收文件描述字的功能,如下:

linux网络编程之socket编程(十六)_父子进程_31

首先封装发送文件描述字的方法:

linux网络编程之socket编程(十六)_#include_32

下面一步步来实现该函数,首先准备第二个参数:

linux网络编程之socket编程(十六)_数据_33

所以,先声明一个该结构体:

linux网络编程之socket编程(十六)_数据_34

接下来填充里面的各个字段,还是看图说话:

linux网络编程之socket编程(十六)_数据_35

所以:

linux网络编程之socket编程(十六)_数据_36

接下来指定缓冲区:

linux网络编程之socket编程(十六)_父子进程_37

linux网络编程之socket编程(十六)_父子进程_38

最后则要准备辅助数据了,因为我们是发送文件描述字,这也是最关键的:

linux网络编程之socket编程(十六)_#include_39

所以msg_control需要指向一个辅助数据的缓冲区,其大小根据发送的文件描述符来获得,如下:

linux网络编程之socket编程(十六)_#include_40

接下来,则需要准备缓冲区中cmsghdr中的数据,也就是发送文件描述字主要是靠它:

linux网络编程之socket编程(十六)_#include_41

linux网络编程之socket编程(十六)_父子进程_42

另外关于数据的填充我们不需要关心,因为都是用系统提供的宏来操作数据的,当所有的数据都准备好之后,下面则可以开始发送了:

linux网络编程之socket编程(十六)_父子进程_43

接下来,则需要封装一个接收文件描述字的函数了,由于怎么发送文件描述字已经很明白了,所以接收也就很简单了,基本类似,这里面就不一一进行说明了:

linux网络编程之socket编程(十六)_父子进程_44

以上发送和接收文件描述字的函数都已经封装好了,接下来利用这两个函数来实现文件描述字的真正传递实验,实验的思路是这样:如果父进程打开了一个文件描述字,再fork()时,子进程是能共享父进程的文件描述字的,也就是只要在fork()之前,打开文件描述字,子进程就能共享它;但是当fork()进程之后,如果一个子进程打开一个文件描述字,父进程是无法共享获取的,所以,这里就可以利用这个原理,来将文件描述字从子进程传递给父进程,还是用sockpair函数,具体如下:

linux网络编程之socket编程(十六)_父子进程_45

下面编译运行看一下效果:

linux网络编程之socket编程(十六)_数据_46

另外,文件描述字的传递,只能通过UNIX域协议的套接字,当前是利用了sockpair函数来实现了父子进程文件描述字的传递,而如果要实现不相关的两个进程之间传递,就不能用socketpair了,就得用上一节中介绍的UNIX域套接字来进行传递,而普通的TCP套接字是不能传递文件描述字的,这个是需要明白了。

最后贴出代码:

send_fd.c:


#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <fcntl.h>

#define ERR_EXIT(m) \
do \
{ \
perror(m); \
exit(EXIT_FAILURE); \
} while(0)


void send_fd(int sock_fd, int send_fd)
{
struct msghdr msg;
struct iovec vec;
struct cmsghdr *p_cmsg;

char sendchar = 0;
vec.iov_base = &sendchar;
vec.iov_len = sizeof(sendchar);

msg.msg_name = NULL;
msg.msg_namelen = 0;
msg.msg_iov = &vec;
msg.msg_iovlen = 1;

char cmsgbuf[CMSG_SPACE(sizeof(send_fd))];

msg.msg_control = cmsgbuf;
msg.msg_controllen = sizeof(cmsgbuf);

p_cmsg = CMSG_FIRSTHDR(&msg);
p_cmsg->cmsg_level = SOL_SOCKET;
p_cmsg->cmsg_type = SCM_RIGHTS;
p_cmsg->cmsg_len = CMSG_LEN(sizeof(send_fd));
int *p_fds;
p_fds = (int*)CMSG_DATA(p_cmsg);
*p_fds = send_fd;

int ret;
ret = sendmsg(sock_fd, &msg, 0);
if (ret != 1)
ERR_EXIT("sendmsg");
}

int recv_fd(const int sock_fd)
{
int ret;
struct msghdr msg;
char recvchar;
struct iovec vec;
int recv_fd;
char cmsgbuf[CMSG_SPACE(sizeof(recv_fd))];
struct cmsghdr *p_cmsg;
int *p_fd;
vec.iov_base = &recvchar;
vec.iov_len = sizeof(recvchar);
msg.msg_name = NULL;
msg.msg_namelen = 0;
msg.msg_iov = &vec;
msg.msg_iovlen = 1;
msg.msg_control = cmsgbuf;
msg.msg_controllen = sizeof(cmsgbuf);
msg.msg_flags = 0;

p_fd = (int*)CMSG_DATA(CMSG_FIRSTHDR(&msg));
*p_fd = -1;
ret = recvmsg(sock_fd, &msg, 0);
if (ret != 1)
ERR_EXIT("recvmsg");

p_cmsg = CMSG_FIRSTHDR(&msg);
if (p_cmsg == NULL)
ERR_EXIT("no passed fd");


p_fd = (int*)CMSG_DATA(p_cmsg);
recv_fd = *p_fd;
if (recv_fd == -1)
ERR_EXIT("no passed fd");

return recv_fd;
}

int main(void)
{
int sockfds[2];

if (socketpair(PF_UNIX, SOCK_STREAM, 0, sockfds) < 0)
ERR_EXIT("socketpair");

pid_t pid;
pid = fork();
if (pid == -1)
ERR_EXIT("fork");

if (pid > 0)
{
close(sockfds[1]);
int fd = recv_fd(sockfds[0]);
char buf[1024] = {0};
read(fd, buf, sizeof(buf));
printf("buf=%s\n", buf);
}
else if (pid == 0)
{
close(sockfds[0]);
int fd;
fd = open("test.txt", O_RDONLY);
if (fd == -1);
send_fd(sockfds[1], fd);
}
return 0;
}


今天学的东西有点复杂,主要是得搞清楚其结构体的填充,对照着示意图来其实也不难,需要好好消化,下节再见~~