-
getsockopt和setsockopt函数:
#include <sys/socket.h>
int getsockopt(int fd, int level, int optname, void *optval, socklen_t *optlen);
int setsockopt(int fd,int level, int optname,const void *optval, socklen_t option);
-
SO_RECVBUF/SO_SNDBUF:
-
每一个套接字都有一个接收缓冲区和一个发送缓冲区(udp的发送缓冲区实际并不存在,它只是一个大小限制而已)。
-
接收缓冲区被tcp、udp、sctp用来保存接收到的数据,直到应用进程来读取。对于tcp来说,套接字接收缓冲区中可用空间的大小限定了tcp的窗口大小。tcp套接字缓冲区不可能溢出,因为不允许对端发出超过本端窗口限制大小的数据。然而对于udp来说,当接收到的数据包装不进套接字接收缓冲区时,就会被丢弃,udp没有流量控制。
-
当设置接收缓冲区大小时,函数调用顺序很重要。因为tcp窗口规模是在建立连接的syn报文中交互的,所以对于客户端来说,应该在connect之前设置SO_RCVBUF,对于服务器来说,应该在listen之前设置该值。
-
SO_RCVLOWAT/SO_SNDLOWAT:
-
每个套接字都有这两个低水位标志,它们由select函数使用。
SO_RCVLOWAT指定接收缓冲区中至少有多少数据时,才可读。
SO_SNDLOWAT指定发送缓冲区有多少可用空间时,才可写。
因为udp发送缓冲区的可用字节数从不改变(udp并不为已发送数据包里副本),所以只要udp发送缓冲区大小大于该标志为,那么该套接字就总是可写的。
-
fcntl函数:
-
fcntl代表file control,可控制各种描述符。
下面是 fcntl与ioctl、路由套接字对比:
操作 |
fcntl |
ioctl |
路由套接字 |
posix |
设置套接字阻塞式IO模型 |
F_SETFL,O_NOBLOCK |
FIONBIO |
|
fcntl |
设置套接字为信号驱动IO模型 |
F_SETFL,O_ASYNC |
FIOASYNC |
|
fcntl |
设置套接字属主 |
F_SETOWN |
SIOCSPGRP/FIOSETOWN |
|
fcntl |
获取到接字属主 |
F_GETOWN |
SIOCGPGRP/FIOGETOWN |
|
fcntl |
获取套接字接收缓冲区中的字节数 |
|
FIONREAD |
|
|
测试套接字是否处于带外标志 |
|
SIOCATMARK |
|
sockatmark |
获取结构列表 |
|
SIOCGIFCONF |
sysctl |
|
接口操作 |
|
SIOC[GS] IFxxx |
|
|
ARP高速缓存操作 |
|
SIOCxARP |
RTM_xxx |
|
路由表操作 |
|
SIOCxxxRT |
RTM_xxx |
|
//fcntl
int fcntl(int fd, int cmd,.../*arg*/);
//成功取决于cmd,出错返回-1
//设置非阻塞
int flags;
if((flags = fcntl(fd,F_GETFL,0) < 0){
err_sys("F_GETFL error");
}
flags |= O_NONBLOCK;
if(fcntl(fd,F_SETFL,flags) < 0){
err_sys("F_SETFL error");
}
//关闭非阻塞
int flags
if((flags = fcntl(fd,F_GETFL,0) < 0){
err_sys("F_GETFL error");
}
flags &= ~O_NONBLOCK;
if(fcntl(fd,F_SETFL,flags) < 0){
err_sys("F_SETFL error");
}