江湖上一直都有这位哥哥的传说,也有很多人说自己就他的真身!

但是。。。

今天分享一下TCP连接的P2P demo,江湖的规矩也要与时俱进。。。

————————————————————————————————————

原理之步骤(原理说了也不是那个理,做NAT的不说,谁知道呢?还不如过程呢)

一.A,B连接打洞服务器S

二.S记录下A的公网IP和端口,记录下B的公网

IP和端口

三.S将A的公网IP和端口以及B的公网端口发给B

四.S将B的公网IP和端口发给A

五. S命令B向A的公网IP和端口发起connect(打洞)

六. S命令B在B的公网端口上监听A的连接(connect)

七.S命令A向B的公网IP和端口发出连接命令(connect

// server.c

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

#include <unistd.h>

#include <signal.h>

#include <sys/socket.h>

#include <arpa/inet.h>

#include <fcntl.h>

#include <errno.h>

typedef struct IP_PORT

{

char ip[64];

int port;

}IP_PORT;

void sigint(int signum)

{

if (SIGINT==signum)

{

exit(1);

}

}

int main()

{

int listenfd;

int value = 1;

socklen_t clilen;

struct sockaddr_in cliaddr, seraddr;

int sock_a, sock_b;

char tmpaddr[128], tmpaddr_port[128], tmpaddr_port_port[128];

signal(SIGINT, sigint);

listenfd = socket(AF_INET, SOCK_STREAM, 0);

if (-1==listenfd)

{

printf("socket -server error: %s\n", strerror(errno));

return 1;

}

// reuse socket's address

setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR|SO_REUSEPORT, &value, sizeof(value));

// set bind server

memset(&seraddr, 0, sizeof(seraddr));

seraddr.sin_family = AF_INET;

seraddr.sin_addr.s_addr = htonl(INADDR_ANY);

seraddr.sin_port = htons(8099);

if (-1==bind(listenfd, (struct sockaddr*)&seraddr, sizeof(seraddr)))

{

printf("bind -server error: %s\n", strerror(errno));

close(listenfd);

return 1;

}

if (-1==listen(listenfd, 5))

{

printf("listen -server error: %s\n", strerror(errno));

close(listenfd);

return 1;

}

sock_a = accept(listenfd, (struct sockaddr*)&cliaddr, &clilen);

if (-1==sock_a)

{

printf("accept -server error: %s\n", strerror(errno));

close(listenfd);

return 1;

}

inet_ntop(AF_INET, (void*)&cliaddr.sin_addr, tmpaddr, sizeof(tmpaddr));

sprintf(tmpaddr_port, "%s %d", tmpaddr, ntohs(cliaddr.sin_port));

sock_b = accept(listenfd, (struct sockaddr*)&cliaddr, &clilen);

if (-1==sock_b)

{

printf("accept -server error: %s\n", strerror(errno));

close(sock_a);

close(listenfd);

return 1;

}

sprintf(tmpaddr_port_port, "%s %d", tmpaddr_port, ntohs(cliaddr.sin_port));

write(sock_b, tmpaddr_port_port, strlen(tmpaddr_port_port)+1);

close(sock_b);

printf("cli_a IP PORT cli_b PORT: %s\n", tmpaddr_port_port);

inet_ntop(AF_INET, (void*)&cliaddr.sin_addr, tmpaddr, sizeof(tmpaddr));

sprintf(tmpaddr_port, "%s %d", tmpaddr, ntohs(cliaddr.sin_port));

write(sock_a, tmpaddr_port, strlen(tmpaddr_port)+1);

close(sock_a);

printf("cli_b IP PORT: %s\n", tmpaddr_port);

close(listenfd);

printf("exit -server\n");

return 0;

}

//cli_a.c

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

#include <unistd.h>

#include <signal.h>

#include <sys/socket.h>

#include <arpa/inet.h>

#include <fcntl.h>

#include <errno.h>

typedef struct IP_PORT

{

char ip[64];

int port;

}IP_PORT;

void sigint(int signum)

{

if (SIGINT==signum)

{

exit(1);

}

}

int main(int argc, char **argv)

{

struct sockaddr_in seraddr;

int sockfd;

char tmpaddr_port[128];

IP_PORT peer;

int len;

char tmpbuf[128];

signal(SIGINT, sigint);

if (2!=argc)

{

printf("CmdLine should be: cli_a <IP>\n");

return 1;

}

sockfd = socket(AF_INET, SOCK_STREAM, 0);

if (-1==sockfd)

{

printf("socket -cli_a error: %s\n", strerror(errno));

return 1;

}

memset(&seraddr, 0, sizeof(seraddr));

seraddr.sin_family = AF_INET;

inet_pton(AF_INET, argv[1], &seraddr.sin_addr);

seraddr.sin_port = htons(8099);

if (0>connect(sockfd, (struct sockaddr*)&seraddr, sizeof(seraddr)))

{

printf("connect main -cli_a error: %s\n", strerror(errno));

return 1;

}

if (0>read(sockfd, tmpaddr_port, 128))

{

printf("read main -cli_a error: %s\n", strerror(errno));

return 1;

}

sscanf(tmpaddr_port, "%s %d", peer.ip, &peer.port);

// close main socket

close(sockfd);

// create new p2p client socket

sockfd = socket(AF_INET, SOCK_STREAM, 0);

if (-1==sockfd)

{

printf("socket -cli_a error: %s\n", strerror(errno));

return 1;

}

inet_pton(AF_INET, peer.ip, &seraddr.sin_addr);

seraddr.sin_port = htons(peer.port);

printf("Peer cli_b IP PORT: %s %d\n", peer.ip, peer.port);

// wait for cli_b to punch hole to cli_a self and cli_b is in accepting

sleep(5);

if (0>connect(sockfd, (struct sockaddr*)&seraddr, sizeof(seraddr)))

{

printf("connect p2p -cli_a error: %s\n", strerror(errno));

close(sockfd);

return 1;

}

len = write(sockfd, "I am a\n", strlen("I am a\n")+1);

if (0>=len)

{

printf("write p2p -cli_a error: %s\n", strerror(errno));

close(sockfd);

return 1;

}

len = read(sockfd, tmpbuf, 128);

if (0<len)

{

printf("%s", tmpbuf);

printf("\n");

}

close(sockfd);

printf("exit -cli_a\n");

return 0;

}

//cli_b.c

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

#include <unistd.h>

#include <signal.h>

#include <sys/socket.h>

#include <arpa/inet.h>

#include <fcntl.h>

#include <errno.h>

typedef struct IP_PORT

{

char ip[64];

int port;

}IP_PORT;

void sigint(int signum)

{

if (SIGINT==signum)

{

exit(1);

}

}

int main(int argc, char **argv)

{

struct sockaddr_in seraddr, cliaddr;

socklen_t clilen;

int sockfd, listenfd;

unsigned int value = 1;

char tmpaddr_port_port[128];

IP_PORT peer;

int len;

char tmpbuf[128];

int port;

signal(SIGINT, sigint);

if (2!=argc)

{

printf("CmdLine should be: cli_b <IP>\n");

return 1;

}

sockfd = socket(AF_INET, SOCK_STREAM, 0);

if (-1==sockfd)

{

printf("socket -cli_b error: %s\n", strerror(errno));

return 1;

}

memset(&seraddr, 0, sizeof(seraddr));

seraddr.sin_family = AF_INET;

inet_pton(AF_INET, argv[1], &seraddr.sin_addr);

seraddr.sin_port = htons(8099);

if (0>connect(sockfd, (struct sockaddr*)&seraddr, sizeof(seraddr)))

{

printf("connect main -cli_b error: %s\n", strerror(errno));

return 1;

}

if (0>read(sockfd, tmpaddr_port_port, 128))

{

printf("read main -cli_b error: %s\n", strerror(errno));

return 1;

}

sscanf(tmpaddr_port_port, "%s %d %d", peer.ip, &peer.port, &port);

// close main socket

close(sockfd);

// create new p2p connect/listen socket

listenfd = socket(AF_INET, SOCK_STREAM, 0);

if (-1==listenfd)

{

printf("socket p2p connect -cli_b error: %s\n", strerror(errno));

return 1;

}

inet_pton(AF_INET, peer.ip, &seraddr.sin_addr);

seraddr.sin_port = htons(peer.port);

printf("Peer cli_a IP PORT: %s %d\n", peer.ip, peer.port);

// punch hole to cli_a

connect(listenfd, (struct sockaddr*)&seraddr, sizeof(seraddr));

printf("Punching hole is finished!\n");

// reuse socket's address

setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR|SO_REUSEPORT, &value, sizeof(value));

seraddr.sin_addr.s_addr = htonl(INADDR_ANY);

seraddr.sin_port = htons(port);

if (0>bind(listenfd, (struct sockaddr*)&seraddr, sizeof(seraddr)))

{

printf("bind p2p -cli_b error: %s\n", strerror(errno));

close(listenfd);

return 1;

}

if (0>listen(listenfd, 5))

{

printf("listen p2p -cli_b error: %s\n", strerror(errno));

close(listenfd);

return 1;

}

sockfd = accept(listenfd, (struct sockaddr*)&cliaddr, &clilen);

if (-1==sockfd)

{

printf("accept p2p -cli_b error: %s\n", strerror(errno));

close(listenfd);

return 1;

}

len = read(sockfd, tmpbuf, 128);

if (0<len)

{

printf("%s", tmpbuf);

printf("\n");

}

len = write(sockfd, "I am b\n", strlen("I am b\n")+1);

if (0>=len)

{

printf("write p2p -cli_b error: %s\n", strerror(errno));

close(sockfd);

close(listenfd);

return 1;

}

close(sockfd);

close(listenfd);

printf("exit -cli_b\n");

return 0;

}

//result

Finally:

这下,有需要的朋友去p2p吧!!!

linux c tcp p2p的更多相关文章

  1. linux下TCP/IP及内核参数优化调优(转)

    Linux下TCP/IP及内核参数优化有多种方式,参数配置得当可以大大提高系统的性能,也可以根据特定场景进行专门的优化,如TIME_WAIT过高,DDOS攻击等等. 如下配置是写在sysctl.con ...

  2. Linux下TCP网络编程与基于Windows下C#socket编程间通信

    一.linux下TCP网络编程基础,需要了解相关函数 Socket():用于套接字初始化. Bind():将 socket 与本机上的一个端口绑定,就可以在该端口监听服务请求. Listen():使s ...

  3. Linux内核 TCP/IP、Socket参数调优

    Linux内核 TCP/IP.Socket参数调优 2014-06-06  Harrison....   阅 9611  转 165 转藏到我的图书馆   微信分享:   Doc1: /proc/sy ...

  4. linux上TCP connection timeout的原因查找

    linux上TCP connection timeout的原因查找 好久没有写文章了, 今天解决了一个网络连接超时的问题, 记录以备查看. 最近在线上nginx经常出现输出connection tim ...

  5. Linux中TCP wrapper的使用

    Linux中TCP wrapper的使用 tcpwrapper的目的是对那些访问控制功能较弱的服务提供访问控制功能要想了解访问控制就必须先知道服务监听的概念: 服务监听的两种方式: listen    ...

  6. Level-IP(Linux userspace TCP/IP stack)

    转自:github.com/saminiir/level-ip Level-IP is a Linux userspace TCP/IP stack, implemented with TUN/TAP ...

  7. 修改linux系统TCP连接数

    修改linux系统TCP连接数 centOS 6.x (1)vi /etc/sysctl.conf (2)添加参数 net.nf_conntrack_max = 655360 (3)sysctl -p ...

  8. Linux上TCP的几个内核参数调优

    Linux作为一个强大的操作系统,提供了一系列内核参数供我们进行调优.光TCP的调优参数就有50多个.在和线上问题斗智斗勇的过程中,笔者积累了一些在内网环境应该进行调优的参数.在此分享出来,希望对大家 ...

  9. 基于Linux的TCP网络聊天室

    1.实验项目名称:基于Linux的TCP网络聊天室 2.实验目的:通过TCP完成多用户群聊和私聊功能. 3.实验过程: 通过socket建立用户连接并传送用户输入的信息,分别来写客户端和服务器端,利用 ...

随机推荐

  1. (一)Knockout 计算属性

    1 Computed 首先,创建一个view model如下: <body> <p>The fullname is: <span data-bind="text ...

  2. Linux安装Tomcat-Nginx-FastDFS-Redis-Solr-集群——【第九集-补充-之安装iptables】

    1,安装完了jdk,tomcat,启动tomcat的bin/startup.sh后,发现在浏览器输入公网ip地址和tomcat的默认(server.xml)中的端口port:8080,无法访问,这主要 ...

  3. Codeforces 730 J.Bottles (01背包)

    <题目链接> 题目大意: 有n个瓶子,各有水量和容量.现在要将这写瓶子里的水存入最少的瓶子里.问你最少需要的瓶子数?在保证瓶子数最少的情况下,要求转移的水量最少. 解题分析:首先,最少的瓶 ...

  4. POJ 1985 Cow Marathon (模板题)(树的直径)

    <题目链接> 题目大意: 给定一颗树,求出树的直径. 解题分析:树的直径模板题,以下程序分别用树形DP和两次BFS来求解. 树形DP: #include <cstdio> #i ...

  5. F#周报2019年第11期

    新闻 Bolero:WebAssembly中的F# 尝试WebAssembly里的F# JetBrains的fsharp-support 2019.1 ML.NET 0.11发布 Outreachy内 ...

  6. BZOJ 5104

    这个模数比较有趣 可以求出 $\sqrt{5}$ 然后就可以做了 $f_n=\dfrac{\sqrt{5}}{5}[(\dfrac{\sqrt{5}+1}{2})^n-(\dfrac{1-\sqrt{ ...

  7. Build Tools

    构建工具能够帮助你创建一个可重复的.可靠的.携带的且不需要手动干预的构建.构建工具是一个可编程的工具,它能够让你以可执行和有序的任务来表达自动化需求.假设你想要编译源代码,将生成的class文件拷贝到 ...

  8. shell中,我们可以通过简单的一个判断来判断命令是否存在

    shell中,我们可以通过简单的一个判断来判断命令是否存在 which "Command" > /dev/null if [ $? -eq 0 ] then echo com ...

  9. SQL2005EXPress自动备份

    STEP1:在数据库服务器的master表中创建存储过程sp_BackupDatabase 代码如下 USE [master] GO /****** 对象: StoredProcedure [dbo] ...

  10. iOS—使用picker View

    iOS—使用picker View 一.实现效果 说明:点击随机按钮,能够自动选取,下方数据自动刷新. 二.实现思路 1.picker view的有默认高度为162,不可修改. 2.显示数据,需要设置 ...