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

📄 bnproxy.c

📁 打魔兽战网的都知道他是什么
💻 C
📖 第 1 页 / 共 3 页
字号:
/* * Copyright (C) 1999,2000  Ross Combs (rocombs@cs.nmsu.edu) * * 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. */#include "common/setup_before.h"#include <stdio.h>#ifdef HAVE_STDDEF_H# include <stddef.h>#else# ifndef NULL#  define NULL ((void *)0)# endif#endif#ifdef STDC_HEADERS# include <stdlib.h>#endif#include "compat/exitstatus.h"#ifdef HAVE_UNISTD_H# include <unistd.h>#endif#include "compat/stdfileno.h"#ifdef HAVE_FCNTL_H# include <fcntl.h>#else# ifdef HAVE_SYS_FILE_H#  include <sys/file.h># endif#endif#ifdef HAVE_STRING_H# include <string.h>#else# ifdef HAVE_STRINGS_H#  include <strings.h># endif#endif#ifdef HAVE_MEMORY_H# include <memory.h>#endif#include "compat/memset.h"#include "compat/memcpy.h"#include <ctype.h>#include <errno.h>#include "compat/strerror.h"#ifdef TIME_WITH_SYS_TIME# include <sys/time.h># include <time.h>#else# ifdef HAVE_SYS_TIME_H#  include <sys/time.h># else#  include <time.h># endif#endif#ifdef HAVE_SYS_TYPES_H# include <sys/types.h>#endif#ifdef HAVE_SYS_SELECT_H# include <sys/select.h>#endif#ifdef HAVE_SYS_SOCKET_H# include <sys/socket.h>#endif#include "compat/socket.h"#include "compat/recv.h"#ifdef HAVE_SYS_PARAM_H# include <sys/param.h>#endif#ifdef HAVE_NETINET_IN_H# include <netinet/in.h>#endif#include "compat/netinet_in.h"#ifdef HAVE_ARPA_INET_H# include <arpa/inet.h>#endif#include "compat/inet_ntoa.h"#ifdef HAVE_NETDB_H# include <netdb.h>#endif#include "compat/psock.h"#include "common/packet.h"#include "common/init_protocol.h"#include "common/hexdump.h"#include "common/eventlog.h"#include "common/bn_type.h"#include "common/queue.h"#include "common/network.h"#include "common/list.h"#include "common/util.h"#include "virtconn.h"#include "common/version.h"#include "common/setup_after.h"FILE * hexstrm=NULL;/* FIXME: This code is horribly unreadable. The UDP stuff is a hack for now. */#define PROXY_FLAG_UDP 1static int init_virtconn(t_virtconn * vc, struct sockaddr_in servaddr);static int proxy_process(unsigned short server_listen_port, struct sockaddr_in servaddr);static void usage(char const * progname);static int init_virtconn(t_virtconn * vc, struct sockaddr_in servaddr){    int  addlen;    char connect_type;        /* determine connection type by first character sent by client */    addlen = psock_recv(virtconn_get_client_socket(vc),&connect_type,sizeof(char),0);        if (addlen<0 && (psock_errno()==PSOCK_EINTR || psock_errno()==PSOCK_EAGAIN || psock_errno()==PSOCK_EWOULDBLOCK))	return 0;        /* error occurred or connection lost */    if (addlen<1)    {	eventlog(eventlog_level_error,__FUNCTION__,"[%d] could not get virtconn class (closing connection) (psock_recv: %s)",virtconn_get_client_socket(vc),pstrerror(psock_errno()));	return -1;    }        switch (connect_type)    {    case CLIENT_INITCONN_CLASS_BNET:	eventlog(eventlog_level_info,__FUNCTION__,"[%d] client initiated normal connection",virtconn_get_client_socket(vc));	virtconn_set_class(vc,virtconn_class_bnet);		break;	    case CLIENT_INITCONN_CLASS_FILE:	eventlog(eventlog_level_info,__FUNCTION__,"[%d] client initiated file download connection",virtconn_get_client_socket(vc));	virtconn_set_class(vc,virtconn_class_file);		break;	    case CLIENT_INITCONN_CLASS_BOT:	eventlog(eventlog_level_info,__FUNCTION__,"[%d] client initiated chat bot connection",virtconn_get_client_socket(vc));	virtconn_set_class(vc,virtconn_class_bot);		break;	    default:	eventlog(eventlog_level_error,__FUNCTION__,"[%d] client initiated unknown connection type 0x%02hx (length %d) (closing connection)",virtconn_get_client_socket(vc),(unsigned short)connect_type,addlen);	return -1;    }        /* now connect to the real server */    if (psock_connect(virtconn_get_server_socket(vc),(struct sockaddr *)&servaddr,(psock_t_socklen)sizeof(servaddr))<0)    {	if (psock_errno()!=PSOCK_EINPROGRESS)	{	    eventlog(eventlog_level_error,__FUNCTION__,"[%d] could not connect to server (psock_connect: %s)\n",virtconn_get_client_socket(vc),pstrerror(psock_errno()));	    return -1;	}	virtconn_set_state(vc,virtconn_state_connecting);    }    else	virtconn_set_state(vc,virtconn_state_connected);        {	t_packet * packet;		if (!(packet = packet_create(packet_class_raw)))	{	    eventlog(eventlog_level_error,__FUNCTION__,"[%d] could not create packet",virtconn_get_client_socket(vc));	    return -1;	}	packet_append_data(packet,&connect_type,1);	queue_push_packet(virtconn_get_serverout_queue(vc),packet);	packet_del_ref(packet);    }        return 0;}static int proxy_process(unsigned short server_listen_port, struct sockaddr_in servaddr){    int                lsock;    struct sockaddr_in laddr;    t_psock_fd_set     rfds, wfds;    int                highest_fd;    int                udpsock;    t_virtconn *       vc;    t_elem const *     curr;    int                csocket;    int                ssocket;        if ((udpsock = psock_socket(PSOCK_PF_INET,PSOCK_SOCK_DGRAM,PSOCK_IPPROTO_UDP))<0)    {	eventlog(eventlog_level_error,__FUNCTION__,"could not create UDP socket (psock_socket: %s)",pstrerror(psock_errno()));	return -1;    }    if (psock_ctl(udpsock,PSOCK_NONBLOCK)<0)	eventlog(eventlog_level_error,__FUNCTION__,"could not set UDP listen socket to non-blocking mode (psock_ctl: %s)",pstrerror(psock_errno()));        if ((lsock = psock_socket(PSOCK_PF_INET,PSOCK_SOCK_STREAM,PSOCK_IPPROTO_TCP))<0)    {	eventlog(eventlog_level_error,__FUNCTION__,"could not create listening socket (psock_socket: %s)",pstrerror(psock_errno()));        psock_close(udpsock);	return -1;    }        {	int val=1;		if (psock_setsockopt(lsock,PSOCK_SOL_SOCKET,PSOCK_SO_REUSEADDR,&val,(psock_t_socklen)sizeof(int))<0)	    eventlog(eventlog_level_error,__FUNCTION__,"[%d] could not set socket option SO_REUSEADDR (psock_setsockopt: %s)",lsock,pstrerror(psock_errno()));	/* not a fatal error... */    }        memset(&laddr,0,sizeof(laddr));    laddr.sin_family = PSOCK_AF_INET;    laddr.sin_port = htons(server_listen_port);    laddr.sin_addr.s_addr = htonl(INADDR_ANY);    if (psock_bind(lsock,(struct sockaddr *)&laddr,(psock_t_socklen)sizeof(laddr))<0)    {	eventlog(eventlog_level_error,__FUNCTION__,"could not bind socket to address 0.0.0.0:%hu TCP (psock_bind: %s)",server_listen_port,pstrerror(psock_errno()));        psock_close(udpsock);	psock_close(lsock);	return -1;    }        memset(&laddr,0,sizeof(laddr));    laddr.sin_family = PSOCK_AF_INET;    laddr.sin_port = htons(server_listen_port);    laddr.sin_addr.s_addr = htonl(INADDR_ANY);    if (psock_bind(udpsock,(struct sockaddr *)&laddr,(psock_t_socklen)sizeof(laddr))<0)    {	eventlog(eventlog_level_error,__FUNCTION__,"could not bind socket to address 0.0.0.0:%hu UDP (psock_bind: %s)",server_listen_port,pstrerror(psock_errno()));	psock_close(udpsock);	psock_close(lsock);	return -1;    }    eventlog(eventlog_level_info,__FUNCTION__,"bound to UDP port %hu",server_listen_port);        /* tell socket to listen for connections */    if (psock_listen(lsock,LISTEN_QUEUE)<0)    {	eventlog(eventlog_level_error,__FUNCTION__,"could not listen (psock_listen: %s)",pstrerror(psock_errno()));        psock_close(udpsock);	psock_close(lsock);	return -1;    }    if (psock_ctl(lsock,PSOCK_NONBLOCK)<0)	eventlog(eventlog_level_error,__FUNCTION__,"could not set TCP listen socket to non-blocking mode (psock_ctl: %s)",pstrerror(psock_errno()));        eventlog(eventlog_level_info,__FUNCTION__,"listening on TCP port %hu",server_listen_port);        for (;;)    {	/* loop over all connections to create the sets for select() */	PSOCK_FD_ZERO(&rfds);	PSOCK_FD_ZERO(&wfds);	highest_fd = lsock;	PSOCK_FD_SET(lsock,&rfds);	if (udpsock>highest_fd)	    highest_fd = udpsock;	PSOCK_FD_SET(udpsock,&rfds);		LIST_TRAVERSE_CONST(virtconnlist(),curr)	{	    vc = elem_get_data(curr);            csocket = virtconn_get_client_socket(vc);	    if (queue_get_length((t_queue const * const *)virtconn_get_clientout_queue(vc))>0)	        PSOCK_FD_SET(csocket,&wfds); /* pending output, also check for writeability */	    PSOCK_FD_SET(csocket,&rfds);            	    if (csocket>highest_fd)                highest_fd = csocket;	    	    switch (virtconn_get_state(vc))	    {	    case virtconn_state_connecting:		eventlog(eventlog_level_debug,__FUNCTION__,"waiting for %d to finish connecting",ssocket);		ssocket = virtconn_get_server_socket(vc);		PSOCK_FD_SET(ssocket,&wfds); /* wait for connect to complete */				if (ssocket>highest_fd)		    highest_fd = ssocket;		break;	    case virtconn_state_connected:		eventlog(eventlog_level_debug,__FUNCTION__,"checking for reading on connected socket %d",ssocket);		ssocket = virtconn_get_server_socket(vc);		if (queue_get_length((t_queue const * const *)virtconn_get_serverout_queue(vc))>0)		    PSOCK_FD_SET(ssocket,&wfds); /* pending output, also check for writeability */		PSOCK_FD_SET(ssocket,&rfds);				if (ssocket>highest_fd)		    highest_fd = ssocket;		break;	    default: /* avoid warning */		break;	    }	}		/* find which sockets need servicing */	if (psock_select(highest_fd+1,&rfds,&wfds,NULL,NULL)<0)	{	    if (errno!=PSOCK_EINTR)	        eventlog(eventlog_level_error,__FUNCTION__,"select failed (select: %s)",pstrerror(errno));	    continue;	}		/* check for incoming connection */	if (PSOCK_FD_ISSET(lsock,&rfds))	{            int                asock;	    struct sockaddr_in caddr;	    psock_t_socklen    caddr_len;            	    /* accept the connection */	    caddr_len = sizeof(caddr);	    if ((asock = psock_accept(lsock,(struct sockaddr *)&caddr,&caddr_len))<0)	    {		if (psock_errno()==PSOCK_EWOULDBLOCK || psock_errno()==PSOCK_ECONNABORTED) /* BSD, POSIX error for aborted connections, SYSV often uses EAGAIN */		    eventlog(eventlog_level_error,__FUNCTION__,"client aborted connection (psock_accept: %s)",pstrerror(psock_errno()));		else /* EAGAIN can mean out of resources _or_ connection aborted */		    if (psock_errno()!=PSOCK_EINTR)			eventlog(eventlog_level_error,__FUNCTION__,"could not accept new connection (psock_accept: %s)",pstrerror(psock_errno()));	    }	    else	    {		int ssd;		int val=1;				eventlog(eventlog_level_info,__FUNCTION__,"[%d] accepted connection from %s:%hu",asock,inet_ntoa(caddr.sin_addr),ntohs(caddr.sin_port));				if (psock_setsockopt(asock,PSOCK_SOL_SOCKET,PSOCK_SO_KEEPALIVE,&val,(psock_t_socklen)sizeof(val))<0)		    eventlog(eventlog_level_error,__FUNCTION__,"[%d] could not set socket option SO_KEEPALIVE (psock_setsockopt: %s)",asock,pstrerror(psock_errno()));		    /* not a fatal error */				if (psock_ctl(asock,PSOCK_NONBLOCK)<0)		{		    eventlog(eventlog_level_error,__FUNCTION__,"[%d] could not set TCP socket to non-blocking mode (closing connection) (psock_ctl: %s)",asock,pstrerror(psock_errno()));		    psock_close(asock);

⌨️ 快捷键说明

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