Chinaunix首页 | 论坛 | 博客
  • 博客访问: 2039007
  • 博文数量: 369
  • 博客积分: 10093
  • 博客等级: 上将
  • 技术积分: 4271
  • 用 户 组: 普通用户
  • 注册时间: 2005-03-21 00:59
文章分类

全部博文(369)

文章存档

2013年(1)

2011年(2)

2010年(10)

2009年(16)

2008年(33)

2007年(146)

2006年(160)

2005年(1)

分类: LINUX

2010-04-10 12:02:27

Linux connect(2)还有一个不怎么为人所知的属性:如果地址的family是AF_UNSPEC,对于TCP连接将通过发送RST包关闭连接,对于UDP连接将去掉目的地址关联,共同的结果是留下一个未连接的socket,这个socket和刚通过socket(2)创建的socket一样。这使得我们可以通过connect(2)到family是AF_UNSPEC的地址来替代close(2)关闭一个连接,并把这个连接的socket放到一个pool里面,以备后用,因为这能省去一个socket(2)调用的开销。

测试代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>

#define xconnect(fd, addr, len) \
        do { \
                if (connect(fd, addr, len) < 0) { \
                        fprintf(stderr, "connect error with: %s\n", \
                                strerror(errno)); \
                        exit(EXIT_FAILURE); \
                } \
        } while (0)

int main(int argc, char *argv[])
{
        int retval;
        struct sockaddr_in addr;
        socklen_t len;

        retval = socket(AF_INET, SOCK_STREAM, 0);
        memset(&addr, 0, sizeof(&addr));
        addr.sin_family = AF_INET;
        addr.sin_addr.s_addr = inet_addr("192.168.235.10");
        addr.sin_port = htons(22);
        len = sizeof(addr);
        xconnect(retval, (struct sockaddr*)&addr, len);
        getsockname(retval, (struct sockaddr*)&addr, &len);
        printf("%d\n", ntohs(addr.sin_port));
        addr.sin_family = AF_UNSPEC;
        xconnect(retval, (struct sockaddr*)&addr, len);
        addr.sin_family = AF_INET;
        addr.sin_port = htons(22);
        xconnect(retval, (struct sockaddr*)&addr, len);
        getsockname(retval, (struct sockaddr*)&addr, &len);
        printf("%d\n", ntohs(addr.sin_port));

        return 0;
}


以下是抓包的输出:

 # tcpdump -i lo -n
tcpdump: verbose output suppressed, use -v or -vv for full protocol decode
listening on lo, link-type EN10MB (Ethernet), capture size 96 bytes
22:43:17.608723 IP 192.168.235.10.60438 > 192.168.235.10.22: S 3365801819:3365801819(0) win 32792
22:43:17.609661 IP 192.168.235.10.22 > 192.168.235.10.60438: S 3378619916:3378619916(0) ack 3365801820 win 32768
22:43:17.609690 IP 192.168.235.10.60438 > 192.168.235.10.22: . ack 1 win 1025
22:43:17.610715 IP 192.168.235.10.60438 > 192.168.235.10.22: R 1:1(0) ack 1 win 1025
22:43:17.613609 IP 192.168.235.10.60439 > 192.168.235.10.22: S 3365834590:3365834590(0) win 32792
22:43:17.614335 IP 192.168.235.10.22 > 192.168.235.10.60439: S 3372878782:3372878782(0) ack 3365834591 win 32768
22:43:17.614351 IP 192.168.235.10.60439 > 192.168.235.10.22: . ack 1 win 1025
22:43:17.614633 IP 192.168.235.10.60439 > 192.168.235.10.22: F 1:1(0) ack 1 win 1025
22:43:17.617225 IP 192.168.235.10.22 > 192.168.235.10.60439: . ack 2 win 1024
22:43:17.729908 IP 192.168.235.10.22 > 192.168.235.10.60439: P 1:22(21) ack 2 win 1024
22:43:17.729943 IP 192.168.235.10.60439 > 192.168.235.10.22: R 3365834592:3365834592(0) win 0

我们尽量在收到对方的FIN包,也就是read(2)等返回0的时候再调用connect(AF_UNSPEC)关闭连接,防止造成误会。

可能的应用:反向HTTP代理中,由代理服务器发往HTTP服务器的连接。
阅读(2641) | 评论(2) | 转发(0) |
0

上一篇:监听网卡的链路通断

下一篇:2^6

给主人留下些什么吧!~~

CUDev2010-04-11 11:53:33

找到问题了,是因为你的代码中。 getsockname(retval, (struct sockaddr*)&addr, &len); 覆盖了原先设置的addr,这样最后的xconnect连接的就是本机了。

CUDev2010-04-11 11:47:02

请问你的内核版本是多少,我在debian 2.6.30-1-686上测试没有成功,直接RST断掉了,没有第二次的三次握手