分类: C/C++
2015-06-19 11:15:59
转自(http://blog.sina.com.cn/s/blog_976bcdc201018su5.html)
参考(http://www.cnblogs.com/kissazi2/p/3158603.html)
这两个函数是最通用的I/O函数。实际上我们可以把所有read、readv、recv和recvfrom调用替换成recvmsg调用。类似地,各种输出函数调用也可以替换成sendmsg调用。
#include
ssize_t recvmsg(int sockfd,struct msghdr *msg,int flags);
ssize_t sendmsg(int sockfd,struct msghdr *msg,int flags);
返回:若成功则为读入或写出的字节数,若出错则为-1.
这两个函数把大部分参数封装到一个msghdr结构中:
struct msghdr {
void *msg_name;
socklen_t msg_namelen;
struct iovec *msg_iov;
int msg_iovlen;
void *msg_control;
socklen_t msg_controllen;
int msg_flags;
};
其中msg_name和msg_namelen这两个成员用于套接字未连接的场合(譬如未连接的UDP套接字)。它们类似recvfrom和sendto的第五个和第六个参数:msg_name指向一个套接字地址结构,用户存放接收者(对于sendmsg)或发送者(对于recvmsg)的协议地址。如果无需指明协议地址(例如TCP套接字或已连接UDP套接字),msg_name应置为空指针。msg_namelen对于sendmsg是一个值参数,对于recvmsg却是一个值-结果参数。
msg_iov和msg_iovlen这两个成员指定输入或输出缓冲区数组(即iovec结构数组),类似readv或writev的第二个和第三个参数。msg_control和msg_controllen这两个成员指定可选的辅助数据的位置和大小。msg_controllen对于recvmsg是一个值-结果参数。
对于recvmsg和sendmsg,我们必须区别它们的两个标志变量,一个是传递值flags参数,另一个是所传递msghdr结构的msg_flags成员,它传递的是引用,因为传递给函数的是该结构的地址。
新的概念——辅助数据
辅助数据可以通过调用sendmsg和recvmsg这两个函数,使用msghdr结构中的msg_control和msg_controllen这两个成员发送和接收。
辅助数据由一个或多个辅助数据对象构成,每个对象以一个定义在头文件中的cmsghdr结构开头。
struct cmsghdr {
socklen_t cmsg_len;
int cmsg_level;
int cmsg_type;
};
下图展示了在一个控制缓冲区中出现2个辅助数据对象的例子。
msg_control指向第一个辅助数据对象,辅助数据的长度则由msg_controllen指定。每个对象开头都是一个描述该对象的cmsghdr结构。在cmsg_type成员和实际数据之间可以有填充字节,从数据结尾处到下一个辅助数据之前也可以有填充字节。
既然由recvmsg返回的辅助数据可含有任意数目的辅助数据对象,为了对应用程序屏蔽可能出现的填充字节,头文件中定义了以下5个宏,以简化对辅助数据的处理。