/* * tcpproxy * * tcpproxy is a simple tcp connection proxy which combines the * features of rinetd and 6tunnel. tcpproxy supports IPv4 and * IPv6 and also supports connections from IPv6 to IPv4 * endpoints and vice versa. * * * Copyright (C) 2010-2015 Christian Pointner * * This file is part of tcpproxy. * * tcpproxy is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * tcpproxy is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with tcpproxy. If not, see . */ #include #include #include #include #include #include #include #include #include #include #include int main(int argc, char* argv[]) { if(argc < 4) { fprintf(stderr, "Usage: %s (A|B)\n", argv[0]); return -1; } int c = socket(AF_INET, SOCK_STREAM, 0); if(!c) { perror("socket()"); return -1; } struct sockaddr_in addr; addr.sin_family = AF_INET; addr.sin_port = htons(atoi(argv[2])); int ret = inet_pton(AF_INET, argv[1], &(addr.sin_addr)); if (ret <= 0) { if (ret == 0) fprintf(stderr, "inet_pton(): Not in presentation format\n"); else perror("inet_pton()"); return -1; } ret = connect(c, (struct sockaddr*)(&addr), sizeof(struct sockaddr_in)); if(ret) { perror("connect()"); return -1; } switch(toupper(argv[3][0])) { /*************** MODE A ***************/ case 'A': { printf("MODE: A\n"); char buf[1234567]; unsigned int i; for(i = 0; i= sizeof(buf)) { break; } } shutdown(c, SHUT_WR); int rtot = 0; for(;;) { int nbread = recv(c, &(buf[rtot]), sizeof(buf) - rtot, 0); if(nbread <= 0) { if(!nbread) { fprintf(stderr, "connection closed\n"); } else if (errno != EAGAIN && errno != EWOULDBLOCK && errno != EINTR) { perror("recv()"); } else continue; return -1; } rtot += nbread; printf("%d bytes received, total = %d, left = %d", nbread, rtot, rtot - wtot); if(rtot == wtot) { printf(" .. finished\n"); for(i = 0; i wtot) { fprintf(stderr, "got too many bytes back???\n"); return -1; } printf("\n"); } while(recv(c, buf, 1, 0) > 0); break; } /*************** MODE A ***************/ case 'B': { printf("MODE: B\n"); char request[] = "hello world\n"; int nbwritten = send(c, request, sizeof(request), 0); if(nbwritten <= 0) { if(nbwritten < 0) perror("send()"); else fprintf(stderr, "nothing sent... aborting\n"); return -1; } printf("%d bytes sent\n", nbwritten); if(nbwritten != sizeof(request)) { fprintf(stderr, "to few bytes sent ... aborting!"); return -1; } char buf[1000]; int rtot = 0; for(;;) { int nbread = recv(c, buf, sizeof(buf), 0); if(nbread <= 0) { if(!nbread) { fprintf(stderr, "connection closed ... finished\n"); return 0; } else if (errno != EAGAIN && errno != EWOULDBLOCK && errno != EINTR) { perror("recv()"); return -1; } continue; } rtot += nbread; printf("%d bytes received, total = %d\n", nbread, rtot); } break; } /**************************************/ } return 0; }