⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 fdtunnel.cpp

📁 一个UNIX/LINUX下的基于内容的过滤服务器源代码
💻 CPP
字号:
//Please refer to http://dansguardian.org/?page=copyright//for the license for this code.//Written by Daniel Barron (daniel@//jadeb/.com).//For support go to http://groups.yahoo.com/group/dansguardian//  This program 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 2 of the License, or//  (at your option) any later version.////  This program 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 this program; if not, write to the Free Software//  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA// This class is a generic multiplexing tunnel// that uses blocking select() to be as efficient as possible.  It tunnels// between the two supplied FDs.// Linux Application Development by Michael K. Johnson and Erik W. Troan// was used extensivly to produce this class#include "autoconf/platform.h"#include "FDTunnel.hpp"#include <sys/time.h>#include <unistd.h>#include <cerrno>#include <sys/types.h>#include <sys/socket.h>#ifdef __BSD	#include "autoconf/select.h"	#include <string.h>#else	#include <sys/select.h>#endif#include <algorithm>FDTunnel::FDTunnel():throughput(0){}void FDTunnel::tunnel(int fdfrom, int fdto) {    int maxfd, rc;    maxfd = fdfrom > fdto ? fdfrom : fdto;  // find the maximum file                        // descriptor.  As Linux normally allows each process                        // to have up to 1024 file descriptors, maxfd                        // prevents the kernel having to look through all                        // 1024 fds each fdSet could contain    char buff[32768];  // buffer for the input    timeval timeout;  // timeval struct    timeout.tv_sec = 120;  // modify the struct so its a 120 sec timeout    timeout.tv_usec = 0;    fd_set fdSet;  // file descriptor set    FD_ZERO(&fdSet);  // clear the set    FD_SET(fdto, &fdSet);  // add fdto to the set    FD_SET(fdfrom, &fdSet);  // add fdfrom to the set    timeval t; // we need a 2nd copy used later    fd_set inset; // we need a 2nd copy used later    fd_set outset; // we need a 3rd copy used later    bool done = false;  // so we get past the first while    while (!done) {        done = true;  // if we don't make a sucessful read and write this                      // flag will stay true and so the while() will exit        inset = fdSet;  // as select() can modify the sets we need to take        t = timeout; // a copy each time round and use that        if (FDTunnel::selectEINTR(maxfd + 1, &inset, NULL, NULL, &t) < 1) {            break;  // an error occured or it timed out so end while()        }        if (FD_ISSET(fdfrom, &inset)) {  // fdfrom is ready to be read from            rc = FDTunnel::read(fdfrom, buff, sizeof(buff) - 1);                                          // read as much as is available            if (rc < 0) {                break;  // an error occured so end the while()            }            if (!rc) {                done = true;  // none received so pipe is closed so flag it            }            if (rc > 0) {  // some data read                throughput += rc;  // increment our counter used to log                outset = fdSet;  // take a copy to work with                FD_CLR(fdfrom, &outset);  // remove fdfrom from the set                           // as we are only interested in writing to fdto                t = timeout;  // take a copy to work with                if (FDTunnel::selectEINTR(fdto + 1, NULL, &outset, NULL, &t) < 1) {                    break; // an error occured or timed out so end while()                }                if (FD_ISSET(fdto, &outset)) {  // fdto ready to write to                    if (!FDTunnel::write(fdto, buff, rc)) {  // write data                        break; // was an error writing                    }                    done = false; // flag to say data still to be handled                } else {                    break;  // should never get here                }            }        }        if (FD_ISSET(fdto, &inset)) {  // fdto is ready to be read from            rc = FDTunnel::read(fdto, buff, sizeof(buff) - 1);                                   // read as much as is available            if (rc < 0) {                break;  // an error occured so end the while()            }            if (!rc) {                done = true;  // none received so pipe is closed so flag it            }            if (rc > 0) {  // some data read                outset = fdSet;  // take a copy to work with                FD_CLR(fdto, &outset);  // remove fdto from the set                           // as we are only interested in writing to fdfrom                t = timeout;  // take a copy to work with                if (FDTunnel::selectEINTR(fdfrom + 1, NULL, &outset, NULL, &t) < 1) {                    break;  // an error occured or timed out so end while()                }                if (FD_ISSET(fdfrom, &outset)) { // fdfrom ready to write to                    if (!FDTunnel::write(fdfrom, buff, rc)) {  // write data                        break; // was an error writing                    }                    done = false;  // flag to say data still to be handled                } else {                    break;  // should never get here                }            }        }    }}int FDTunnel::read(int fd, char* buff, int len) {    int rc;    while (true) {  // using the while as a restart point with continue        rc = recv(fd, buff, len, 0);  // read whats available        if (rc < 0) {  // error occured            if (errno == EINTR) {                continue;  // was just a signal, so restart            }        }        break;  // end the while    }    return rc;  // return how much received or status}bool FDTunnel::write(int fd, char* buff, int len) {    int actuallysent = 0;    int sent;    while (actuallysent < len) {  // until we're finished        // maybe need an fd ready-for-output check with timout here        sent = send(fd, buff + actuallysent, len - actuallysent, 0);        if (sent < 0) {            if (errno == EINTR) {                continue;            }            return false; //error occurred        }        if (sent == 0) {            return false; // other end is closed        }        actuallysent += sent;    }    return true;  // sucess!}// a wrapper for select to handle EINTR neaterint FDTunnel::selectEINTR(int numfds, fd_set * readfds, fd_set * writefds, fd_set * exceptfds, struct timeval * timeout) {    int rc;    while (true) {  // using the while as a restart point with continue        rc = select(numfds, readfds, writefds, exceptfds, timeout);        if (rc < 0) {            if (errno == EINTR) {                continue;  // was interupted by a signal so restart            }        }        break;  // end the while    }    return rc;  // return status}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -