2014-08-04 01:02:04 +00:00
|
|
|
// Emacs style mode select -*- C++ -*-
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// Copyright (C) 1993-1996 by id Software, Inc.
|
|
|
|
// Portions Copyright (C) 1998-2000 by DooM Legacy Team.
|
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
/// \file
|
|
|
|
/// \brief SDL network interface
|
|
|
|
|
|
|
|
#include "../doomdef.h"
|
|
|
|
|
|
|
|
#include "../i_system.h"
|
|
|
|
#include "../d_event.h"
|
|
|
|
#include "../d_net.h"
|
|
|
|
#include "../m_argv.h"
|
|
|
|
|
|
|
|
#include "../doomstat.h"
|
|
|
|
|
|
|
|
#include "../i_net.h"
|
|
|
|
|
|
|
|
#include "../z_zone.h"
|
|
|
|
|
|
|
|
#include "../i_tcp.h"
|
|
|
|
|
2014-08-04 03:49:33 +00:00
|
|
|
#ifdef HAVE_SDL
|
2014-08-04 01:02:04 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_SDLNET
|
|
|
|
|
|
|
|
#include "SDL_net.h"
|
|
|
|
|
|
|
|
#define MAXBANS 20
|
|
|
|
|
|
|
|
static IPaddress clientaddress[MAXNETNODES+1];
|
|
|
|
static IPaddress banned[MAXBANS];
|
|
|
|
|
|
|
|
static UDPpacket mypacket;
|
|
|
|
static UDPsocket mysocket = NULL;
|
|
|
|
static SDLNet_SocketSet myset = NULL;
|
|
|
|
|
|
|
|
static size_t numbans = 0;
|
|
|
|
static boolean NET_bannednode[MAXNETNODES+1]; /// \note do we really need the +1?
|
|
|
|
static boolean init_SDLNet_driver = false;
|
|
|
|
|
|
|
|
static const char *NET_AddrToStr(IPaddress* sk)
|
|
|
|
{
|
|
|
|
static char s[22]; // 255.255.255.255:65535
|
|
|
|
strcpy(s, SDLNet_ResolveIP(sk));
|
|
|
|
if (sk->port != 0) strcat(s, va(":%d", sk->port));
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *NET_GetNodeAddress(INT32 node)
|
|
|
|
{
|
|
|
|
if (!nodeconnected[node])
|
|
|
|
return NULL;
|
|
|
|
return NET_AddrToStr(&clientaddress[node]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *NET_GetBanAddress(size_t ban)
|
|
|
|
{
|
|
|
|
if (ban > numbans)
|
|
|
|
return NULL;
|
|
|
|
return NET_AddrToStr(&banned[ban]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static boolean NET_cmpaddr(IPaddress* a, IPaddress* b)
|
|
|
|
{
|
|
|
|
return (a->host == b->host && (b->port == 0 || a->port == b->port));
|
|
|
|
}
|
|
|
|
|
|
|
|
static boolean NET_CanGet(void)
|
|
|
|
{
|
|
|
|
return myset?(SDLNet_CheckSockets(myset,0) == 1):false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void NET_Get(void)
|
|
|
|
{
|
|
|
|
INT32 mystatus;
|
|
|
|
INT32 newnode;
|
|
|
|
mypacket.len = MAXPACKETLENGTH;
|
|
|
|
if (!NET_CanGet())
|
|
|
|
{
|
|
|
|
doomcom->remotenode = -1; // no packet
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
mystatus = SDLNet_UDP_Recv(mysocket,&mypacket);
|
|
|
|
if (mystatus != -1)
|
|
|
|
{
|
|
|
|
if (mypacket.channel != -1)
|
|
|
|
{
|
|
|
|
doomcom->remotenode = mypacket.channel+1; // good packet from a game player
|
|
|
|
doomcom->datalength = mypacket.len;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
newnode = SDLNet_UDP_Bind(mysocket,-1,&mypacket.address);
|
|
|
|
if (newnode != -1)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
newnode++;
|
|
|
|
M_Memcpy(&clientaddress[newnode], &mypacket.address, sizeof (IPaddress));
|
|
|
|
DEBFILE(va("New node detected: node:%d address:%s\n", newnode,
|
|
|
|
NET_GetNodeAddress(newnode)));
|
|
|
|
doomcom->remotenode = newnode; // good packet from a game player
|
|
|
|
doomcom->datalength = mypacket.len;
|
|
|
|
for (i = 0; i < numbans; i++)
|
|
|
|
{
|
|
|
|
if (NET_cmpaddr(&mypacket.address, &banned[i]))
|
|
|
|
{
|
|
|
|
DEBFILE("This dude has been banned\n");
|
|
|
|
NET_bannednode[newnode] = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (i == numbans)
|
|
|
|
NET_bannednode[newnode] = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
I_OutputMsg("SDL_Net: %s",SDLNet_GetError());
|
|
|
|
}
|
|
|
|
else if (mystatus == -1)
|
|
|
|
{
|
|
|
|
I_OutputMsg("SDL_Net: %s",SDLNet_GetError());
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBFILE("New node detected: No more free slots\n");
|
|
|
|
doomcom->remotenode = -1; // no packet
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
static boolean NET_CanSend(void)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static void NET_Send(void)
|
|
|
|
{
|
|
|
|
if (!doomcom->remotenode)
|
|
|
|
return;
|
|
|
|
mypacket.len = doomcom->datalength;
|
|
|
|
if (SDLNet_UDP_Send(mysocket,doomcom->remotenode-1,&mypacket) == 0)
|
|
|
|
{
|
|
|
|
I_OutputMsg("SDL_Net: %s",SDLNet_GetError());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void NET_FreeNodenum(INT32 numnode)
|
|
|
|
{
|
|
|
|
// can't disconnect from self :)
|
|
|
|
if (!numnode)
|
|
|
|
return;
|
|
|
|
|
|
|
|
DEBFILE(va("Free node %d (%s)\n", numnode, NET_GetNodeAddress(numnode)));
|
|
|
|
|
|
|
|
SDLNet_UDP_Unbind(mysocket,numnode-1);
|
|
|
|
|
|
|
|
memset(&clientaddress[numnode], 0, sizeof (IPaddress));
|
|
|
|
}
|
|
|
|
|
|
|
|
static UDPsocket NET_Socket(void)
|
|
|
|
{
|
|
|
|
UDPsocket temp = NULL;
|
|
|
|
Uint16 portnum = 0;
|
|
|
|
IPaddress tempip = {INADDR_BROADCAST,0};
|
|
|
|
//Hurdler: I'd like to put a server and a client on the same computer
|
|
|
|
//Logan: Me too
|
|
|
|
//BP: in fact for client we can use any free port we want i have read
|
|
|
|
// in some doc that connect in udp can do it for us...
|
|
|
|
//Alam: where?
|
|
|
|
if (M_CheckParm("-clientport"))
|
|
|
|
{
|
|
|
|
if (!M_IsNextParm())
|
|
|
|
I_Error("syntax: -clientport <portnum>");
|
|
|
|
portnum = atoi(M_GetNextParm());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
portnum = sock_port;
|
|
|
|
temp = SDLNet_UDP_Open(portnum);
|
|
|
|
if (!temp)
|
|
|
|
{
|
|
|
|
I_OutputMsg("SDL_Net: %s",SDLNet_GetError());
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (SDLNet_UDP_Bind(temp,BROADCASTADDR-1,&tempip) == -1)
|
|
|
|
{
|
|
|
|
I_OutputMsg("SDL_Net: %s",SDLNet_GetError());
|
|
|
|
SDLNet_UDP_Close(temp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
clientaddress[BROADCASTADDR].port = sock_port;
|
|
|
|
clientaddress[BROADCASTADDR].host = INADDR_BROADCAST;
|
|
|
|
|
|
|
|
doomcom->extratics = 1; // internet is very high ping
|
|
|
|
|
|
|
|
return temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void I_ShutdownSDLNetDriver(void)
|
|
|
|
{
|
|
|
|
if (myset) SDLNet_FreeSocketSet(myset);
|
|
|
|
myset = NULL;
|
|
|
|
SDLNet_Quit();
|
|
|
|
init_SDLNet_driver = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void I_InitSDLNetDriver(void)
|
|
|
|
{
|
|
|
|
if (init_SDLNet_driver)
|
|
|
|
I_ShutdownSDLNetDriver();
|
|
|
|
if (SDLNet_Init() == -1)
|
|
|
|
{
|
|
|
|
I_OutputMsg("SDL_Net: %s",SDLNet_GetError());
|
|
|
|
return; // No good!
|
|
|
|
}
|
|
|
|
D_SetDoomcom();
|
|
|
|
mypacket.data = doomcom->data;
|
|
|
|
init_SDLNet_driver = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void NET_CloseSocket(void)
|
|
|
|
{
|
|
|
|
if (mysocket)
|
|
|
|
SDLNet_UDP_Close(mysocket);
|
|
|
|
mysocket = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static SINT8 NET_NetMakeNodewPort(const char *hostname, const char *port)
|
|
|
|
{
|
|
|
|
INT32 newnode;
|
|
|
|
UINT16 portnum = sock_port;
|
|
|
|
IPaddress hostnameIP;
|
|
|
|
|
|
|
|
// retrieve portnum from address!
|
|
|
|
if (port && !port[0])
|
|
|
|
portnum = atoi(port);
|
|
|
|
|
|
|
|
if (SDLNet_ResolveHost(&hostnameIP,hostname,portnum) == -1)
|
|
|
|
{
|
|
|
|
I_OutputMsg("SDL_Net: %s",SDLNet_GetError());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
newnode = SDLNet_UDP_Bind(mysocket,-1,&hostnameIP);
|
|
|
|
if (newnode == -1)
|
|
|
|
{
|
|
|
|
I_OutputMsg("SDL_Net: %s",SDLNet_GetError());
|
|
|
|
return newnode;
|
|
|
|
}
|
|
|
|
newnode++;
|
|
|
|
M_Memcpy(&clientaddress[newnode],&hostnameIP,sizeof (IPaddress));
|
|
|
|
return (SINT8)newnode;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static boolean NET_OpenSocket(void)
|
|
|
|
{
|
|
|
|
memset(clientaddress, 0, sizeof (clientaddress));
|
|
|
|
|
|
|
|
//I_OutputMsg("SDL_Net Code starting up\n");
|
|
|
|
|
|
|
|
I_NetSend = NET_Send;
|
|
|
|
I_NetGet = NET_Get;
|
|
|
|
I_NetCloseSocket = NET_CloseSocket;
|
|
|
|
I_NetFreeNodenum = NET_FreeNodenum;
|
|
|
|
I_NetMakeNodewPort = NET_NetMakeNodewPort;
|
|
|
|
|
|
|
|
//I_NetCanSend = NET_CanSend;
|
|
|
|
|
|
|
|
// build the socket but close it first
|
|
|
|
NET_CloseSocket();
|
|
|
|
mysocket = NET_Socket();
|
|
|
|
|
|
|
|
if (!mysocket)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// for select
|
|
|
|
myset = SDLNet_AllocSocketSet(1);
|
|
|
|
if (!myset)
|
|
|
|
{
|
|
|
|
I_OutputMsg("SDL_Net: %s",SDLNet_GetError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (SDLNet_UDP_AddSocket(myset,mysocket) == -1)
|
|
|
|
{
|
|
|
|
I_OutputMsg("SDL_Net: %s",SDLNet_GetError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static boolean NET_Ban(INT32 node)
|
|
|
|
{
|
|
|
|
if (numbans == MAXBANS)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
M_Memcpy(&banned[numbans], &clientaddress[node], sizeof (IPaddress));
|
|
|
|
banned[numbans].port = 0;
|
|
|
|
numbans++;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static boolean NET_SetBanAddress(const char *address, const char *mask)
|
|
|
|
{
|
|
|
|
(void)mask;
|
|
|
|
if (bans == MAXBANS)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (SDLNet_ResolveHost(&banned[numbans], address, 0) == -1)
|
|
|
|
return false;
|
|
|
|
numbans++;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void NET_ClearBans(void)
|
|
|
|
{
|
|
|
|
numbans = 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//
|
|
|
|
// I_InitNetwork
|
|
|
|
// Only required for DOS, so this is more a dummy
|
|
|
|
//
|
|
|
|
boolean I_InitNetwork(void)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_SDLNET
|
|
|
|
char serverhostname[255];
|
|
|
|
boolean ret = false;
|
|
|
|
SDL_version SDLcompiled;
|
|
|
|
const SDL_version *SDLlinked = SDLNet_Linked_Version();
|
|
|
|
SDL_NET_VERSION(&SDLcompiled)
|
|
|
|
I_OutputMsg("Compiled for SDL_Net version: %d.%d.%d\n",
|
|
|
|
SDLcompiled.major, SDLcompiled.minor, SDLcompiled.patch);
|
|
|
|
I_OutputMsg("Linked with SDL_Net version: %d.%d.%d\n",
|
|
|
|
SDLlinked->major, SDLlinked->minor, SDLlinked->patch);
|
|
|
|
//if (!M_CheckParm ("-sdlnet"))
|
|
|
|
// return false;
|
|
|
|
// initilize the driver
|
|
|
|
I_InitSDLNetDriver();
|
|
|
|
I_AddExitFunc(I_ShutdownSDLNetDriver);
|
|
|
|
if (!init_SDLNet_driver)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (M_CheckParm("-udpport"))
|
|
|
|
{
|
|
|
|
if (M_IsNextParm())
|
|
|
|
sock_port = (UINT16)atoi(M_GetNextParm());
|
|
|
|
else
|
|
|
|
sock_port = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// parse network game options,
|
|
|
|
if (M_CheckParm("-server") || dedicated)
|
|
|
|
{
|
|
|
|
server = true;
|
|
|
|
|
|
|
|
// If a number of clients (i.e. nodes) is specified, the server will wait for the clients
|
|
|
|
// to connect before starting.
|
|
|
|
// If no number is specified here, the server starts with 1 client, and others can join
|
|
|
|
// in-game.
|
|
|
|
// Since Boris has implemented join in-game, there is no actual need for specifying a
|
|
|
|
// particular number here.
|
|
|
|
// FIXME: for dedicated server, numnodes needs to be set to 0 upon start
|
|
|
|
/* if (M_IsNextParm())
|
|
|
|
doomcom->numnodes = (INT16)atoi(M_GetNextParm());
|
|
|
|
else */if (dedicated)
|
|
|
|
doomcom->numnodes = 0;
|
|
|
|
else
|
|
|
|
doomcom->numnodes = 1;
|
|
|
|
|
|
|
|
if (doomcom->numnodes < 0)
|
|
|
|
doomcom->numnodes = 0;
|
|
|
|
if (doomcom->numnodes > MAXNETNODES)
|
|
|
|
doomcom->numnodes = MAXNETNODES;
|
|
|
|
|
|
|
|
// server
|
|
|
|
servernode = 0;
|
|
|
|
// FIXME:
|
|
|
|
// ??? and now ?
|
|
|
|
// server on a big modem ??? 4*isdn
|
|
|
|
net_bandwidth = 16000;
|
|
|
|
hardware_MAXPACKETLENGTH = INETPACKETLENGTH;
|
|
|
|
|
|
|
|
ret = true;
|
|
|
|
}
|
|
|
|
else if (M_CheckParm("-connect"))
|
|
|
|
{
|
|
|
|
if (M_IsNextParm())
|
|
|
|
strcpy(serverhostname, M_GetNextParm());
|
|
|
|
else
|
|
|
|
serverhostname[0] = 0; // assuming server in the LAN, use broadcast to detect it
|
|
|
|
|
|
|
|
// server address only in ip
|
|
|
|
if (serverhostname[0])
|
|
|
|
{
|
|
|
|
COM_BufAddText("connect \"");
|
|
|
|
COM_BufAddText(serverhostname);
|
|
|
|
COM_BufAddText("\"\n");
|
|
|
|
|
|
|
|
// probably modem
|
|
|
|
hardware_MAXPACKETLENGTH = INETPACKETLENGTH;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// so we're on a LAN
|
|
|
|
COM_BufAddText("connect any\n");
|
|
|
|
|
|
|
|
net_bandwidth = 800000;
|
|
|
|
hardware_MAXPACKETLENGTH = MAXPACKETLENGTH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mypacket.maxlen = hardware_MAXPACKETLENGTH;
|
|
|
|
I_NetOpenSocket = NET_OpenSocket;
|
|
|
|
I_Ban = NET_Ban;
|
|
|
|
I_ClearBans = NET_ClearBans;
|
|
|
|
I_GetNodeAddress = NET_GetNodeAddress;
|
|
|
|
I_GetBenAddress = NET_GetBenAddress;
|
|
|
|
I_SetBanAddress = NET_SetBanAddress;
|
|
|
|
bannednode = NET_bannednode;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
#else
|
|
|
|
if ( M_CheckParm ("-net") )
|
|
|
|
{
|
|
|
|
I_Error("-net not supported, use -server and -connect\n"
|
|
|
|
"see docs for more\n");
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|