#include #include #include #include #include #include #include #include #include char *ifaceName; struct sockaddr_in addrLoc; struct sockaddr_in addrRem; int portLoc; int portRem; int commSock; int dumpPipe[4]; int dumpChld; int playPipe[4]; int playChld; pthread_t threadUdp; pthread_t threadRaw; pthread_t threadStat; long byteRx; long packRx; long byteTx; long packTx; void err(char*buf) { printf("%s\n", buf); _exit(1); } #define packHead 16 void doRawLoop() { unsigned char bufD[16384]; int bufS; for (;;) { bufS = sizeof (bufD); bufS = read(dumpPipe[2], bufD, bufS); if (bufS < 0) break; packRx++; byteRx += bufS; send(commSock, bufD + packHead, bufS - packHead, 0); } kill(dumpChld, SIGKILL); kill(playChld, SIGKILL); err("raw thread exited"); } void doUdpLoop() { unsigned char bufD[16384]; int* bufI = (int*)&bufD; int bufS; for (;;) { bufS = sizeof (bufD) - packHead; bufS = recv(commSock, bufD + packHead, bufS, 0); if (bufS < 0) break; packTx++; byteTx += bufS; bufI[0] = 0; bufI[1] = 0; bufI[2] = bufS; bufI[3] = bufS; if (write(playPipe[1], bufD, bufS + packHead) < 0) break; } kill(dumpChld, SIGKILL); kill(playChld, SIGKILL); err("udp thread exited"); } void doMainLoop() { unsigned char buf[1024]; doer: printf("> "); buf[0] = 0; int i = scanf("%1023s", buf); if (i < 1) { sleep(1); goto doer; } switch (buf[0]) { case 0: goto doer; break; case 'H': case 'h': case '?': printf("commands:\n"); printf("h - this help\n"); printf("q - exit process\n"); printf("d - display counters\n"); printf("c - clear counters\n"); break; case 'Q': case 'q': kill(dumpChld, SIGKILL); kill(playChld, SIGKILL); err("exiting"); break; case 'D': case 'd': printf("iface counters:\n"); printf(" packets bytes\n"); printf("received %20li %20li\n", packRx, byteRx); printf("sent %20li %20li\n", packTx, byteTx); break; case 'C': case 'c': printf("counters cleared.\n"); byteRx = 0; packRx = 0; byteTx = 0; packTx = 0; break; default: printf("unknown command '%s', try ?\n", buf); break; } printf("\n"); goto doer; } int main(int argc, char **argv) { if (argc < 5) { if (argc <= 1) goto help; char*curr = argv[1]; if ((curr[0] == '-') || (curr[0] == '/')) curr++; switch (curr[0]) { case 'V': case 'v': err("compatible interface driver v1.0\n"); break; case '?': case 'h': case 'H': help : curr = argv[0]; printf("using: %s [laddr]\n", curr); printf(" or: %s \n", curr); printf("commands: v=version\n"); printf(" h=this help\n"); _exit(1); break; default: err("unknown command, try -h"); break; } _exit(1); } portLoc = atoi(argv[2]); portRem = atoi(argv[4]); memset(&addrLoc, 0, sizeof (addrLoc)); memset(&addrRem, 0, sizeof (addrRem)); if (inet_aton(argv[3], &addrRem.sin_addr) == 0) err("bad raddr address"); if (argc > 5) { if (inet_aton(argv[5], &addrLoc.sin_addr) == 0) err("bad laddr address"); } else { addrLoc.sin_addr.s_addr = htonl(INADDR_ANY); } addrLoc.sin_family = AF_INET; addrLoc.sin_port = htons(portLoc); addrRem.sin_family = AF_INET; addrRem.sin_port = htons(portRem); if ((commSock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) < 0) err("unable to udp open socket"); if (bind(commSock, (struct sockaddr *) &addrLoc, sizeof (addrLoc)) < 0) err("failed to bind socket"); printf("binded to local port %s %i.\n", inet_ntoa(addrLoc.sin_addr), portLoc); if (connect(commSock, (struct sockaddr *) &addrRem, sizeof (addrRem)) < 0) err("failed to connect socket"); printf("will send to %s %i.\n", inet_ntoa(addrRem.sin_addr), portRem); ifaceName = malloc(strlen(argv[1]) + 1); if (ifaceName == NULL) err("error allocating memory"); strcpy(ifaceName, argv[1]); printf("opening interface %s.\n", ifaceName); if (pipe(&playPipe[0]) == -1) err("error creating pipe"); if (pipe(&playPipe[2]) == -1) err("error creating pipe"); playChld = fork(); if (playChld == 0) { dup2(playPipe[0], STDIN_FILENO); dup2(playPipe[3], STDOUT_FILENO); if (execlp("tcpreplay", "tcpreplay", "-q", "-t", "-i", ifaceName, "/dev/stdin", (char*) NULL) == -1) err("error executing process"); return 0; } pipe(&dumpPipe[0]); pipe(&dumpPipe[2]); dumpChld = fork(); if (dumpChld == 0) { dup2(dumpPipe[0], STDIN_FILENO); dup2(dumpPipe[3], STDOUT_FILENO); if (execlp("tcpdump", "tcpdump", "--immediate-mode", "-O", "-U", "-Q", "in", "-i", ifaceName, "-w", "/dev/stdout", (char*) NULL) == -1) err("error executing process"); return 0; } char buf[1024]; int i = read(dumpPipe[2], buf, sizeof(buf)); write(playPipe[1], buf, i); printf("tcpreplay is %i, tcpdump is %i, header is %i, record is %i\n", playChld, dumpChld, i, packHead); if (pthread_create(&threadRaw, NULL, (void*) & doRawLoop, NULL)) err("error creating raw thread"); if (pthread_create(&threadUdp, NULL, (void*) & doUdpLoop, NULL)) err("error creating udp thread"); doMainLoop(); }