quakeforge/qtv/source/qtv.c

420 lines
8.4 KiB
C
Raw Normal View History

/*
2005-06-09 03:07:13 +00:00
qtv.c
2005-06-09 03:07:13 +00:00
QTV main file
2005-06-09 03:07:13 +00:00
Copyright (C) 2004 Bill Currie <bill@taniwha.org>
2005-06-09 03:07:13 +00:00
Author: Bill Currie
Date: 2004/02/18
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:
Free Software Foundation, Inc.
59 Temple Place - Suite 330
Boston, MA 02111-1307, USA
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#ifdef HAVE_STRING_H
# include <string.h>
#endif
#ifdef HAVE_STRINGS_H
# include <strings.h>
#endif
2004-02-18 23:21:11 +00:00
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include "QF/cbuf.h"
#include "QF/cmd.h"
#include "QF/console.h"
#include "QF/cvar.h"
#include "QF/dstring.h"
#include "QF/idparse.h"
#include "QF/info.h"
#include "QF/qargs.h"
#include "QF/quakefs.h"
#include "QF/sys.h"
#include "QF/va.h"
#include "QF/zone.h"
#include "QF/plugin/console.h"
#include "compat.h"
#include "netchan.h"
#include "qw/protocol.h"
#include "qtv/include/client.h"
#include "qtv/include/connection.h"
#include "qtv/include/qtv.h"
#include "qtv/include/server.h"
#undef qtv_print
SERVER_PLUGIN_PROTOS
static plugin_list_t server_plugin_list[] = {
SERVER_PLUGIN_LIST
};
double realtime;
float sv_timeout;
static cvar_t sv_timeout_cvar = {
.name = "timeout",
.description =
"Sets the amount of time in seconds before a client is considered "
"disconnected if the server does not receive a packet",
.default_value = "60",
.flags = CVAR_NONE,
.value = { .type = &cexpr_float, .value = &sv_timeout },
};
cbuf_t *qtv_cbuf;
cbuf_args_t *qtv_args;
static char *qtv_console_plugin;
static cvar_t qtv_console_plugin_cvar = {
.name = "qtv_console_plugin",
.description =
"Plugin used for the console",
.default_value = "server",
.flags = CVAR_ROM,
.value = { .type = 0, .value = &qtv_console_plugin },
};
static int qtv_port;
static cvar_t qtv_port_cvar = {
.name = "qtv_port",
.description =
"udp port to use",
.default_value = 0,
.flags = CVAR_ROM,
.value = { .type = &cexpr_int, .value = &qtv_port },
};
static float qtv_mem_size;
static cvar_t qtv_mem_size_cvar = {
.name = "qtv_mem_size",
.description =
"Amount of memory (in MB) to allocate for the "
PACKAGE_NAME
" heap",
.default_value = "8",
.flags = CVAR_ROM,
.value = { .type = &cexpr_float, .value = &qtv_mem_size },
};
redirect_t qtv_redirected;
client_t *qtv_redirect_client;
dstring_t outputbuf = {&dstring_default_mem};
static __attribute__((format(PRINTF, 1, 0))) void
qtv_print (const char *fmt, va_list args)
{
static int pending;
static dstring_t *msg;
if (!msg)
msg = dstring_new ();
dvsprintf (msg, fmt, args);
if (qtv_redirected)
dstring_appendstr (&outputbuf, msg->str);
else {
time_t mytime;
struct tm *local;
char stamp[123];
if (pending) {
Con_Printf ("%s", msg->str);
} else {
mytime = time (NULL);
local = localtime (&mytime);
#ifdef _WIN32
strftime (stamp, sizeof (stamp), "[%b %d %X] ", local);
#else
strftime (stamp, sizeof (stamp), "[%b %e %X] ", local);
#endif
Con_Printf ("%s%s", stamp, msg->str);
}
if (msg->str[0] && msg->str[strlen (msg->str) - 1] != '\n') {
pending = 1;
} else {
pending = 0;
}
}
}
void
qtv_printf (const char *fmt, ...)
{
va_list argptr;
va_start (argptr, fmt);
qtv_print (fmt, argptr);
va_end (argptr);
}
static void
qtv_flush_redirect (void)
{
char send[8000 + 6];
size_t count;
int bytes;
const char *p;
if (!outputbuf.size)
return;
count = strlen (outputbuf.str);
if (qtv_redirected == RD_PACKET) {
send[0] = 0xff;
send[1] = 0xff;
send[2] = 0xff;
send[3] = 0xff;
send[4] = A2C_PRINT;
p = outputbuf.str;
while (count) {
bytes = min (count, sizeof (send) - 5);
memcpy (send + 5, p, bytes);
Netchan_SendPacket (bytes + 5, send, net_from);
p += bytes;
count -= bytes;
}
} else if (qtv_redirected == RD_CLIENT) {
client_t *cl = qtv_redirect_client;
p = outputbuf.str;
while (count) {
// +/- 3 for svc_print, PRINT_HIGH and nul byte
// min of 4 because we don't want to send an effectively empty
// message
bytes = MSG_ReliableCheckSize (&cl->backbuf, count + 3, 4) - 3;
// if writing another packet would overflow the client, just drop
// the rest of the data. getting rudely disconnected would be much
// more annoying than losing the tail end of the output
if (bytes <= 0)
break;
MSG_ReliableWrite_Begin (&cl->backbuf, svc_print, bytes + 3);
MSG_ReliableWrite_Byte (&cl->backbuf, PRINT_HIGH);
MSG_ReliableWrite_SZ (&cl->backbuf, p, bytes);
MSG_ReliableWrite_Byte (&cl->backbuf, 0);
p += bytes;
count -= bytes;
}
}
dstring_clear (&outputbuf);
}
void
qtv_begin_redirect (redirect_t rd, client_t *cl)
{
qtv_redirected = rd;
qtv_redirect_client = cl;
dstring_clear (&outputbuf);
}
void
qtv_end_redirect (void)
{
qtv_flush_redirect ();
qtv_redirected = RD_NONE;
qtv_redirect_client = 0;
}
static memhunk_t *
qtv_memory_init (void)
{
int mem_size;
void *mem_base;
Cvar_Register (&qtv_mem_size_cvar, 0, 0);
mem_size = (int) (qtv_mem_size * 1024 * 1024);
mem_base = Sys_Alloc (mem_size);
if (!mem_base)
Sys_Error ("Can't allocate %d", mem_size);
return Memory_Init (mem_base, mem_size);
}
static void
qtv_shutdown (void *data)
{
2005-05-05 22:50:09 +00:00
Cbuf_Delete (qtv_cbuf);
Cbuf_ArgsDelete (qtv_args);
}
static void
qtv_quit_f (void)
{
Sys_Printf ("Shutting down.\n");
Sys_Quit ();
}
static void
qtv_net_init (void)
{
qtv_port_cvar.default_value = nva ("%d", PORT_QTV);
Cvar_Register (&qtv_port_cvar, 0, 0);
Cvar_Register (&sv_timeout_cvar, 0, 0);
NET_Init (qtv_port);
Connection_Init ();
net_realtime = &realtime;
Netchan_Init ();
}
static void
qtv_init (void)
{
qtv_cbuf = Cbuf_New (&id_interp);
qtv_args = Cbuf_ArgsNew ();
Sys_RegisterShutdown (qtv_shutdown, 0);
Sys_Init ();
COM_ParseConfig (qtv_cbuf);
cmd_warncmd = 1;
memhunk_t *hunk = qtv_memory_init ();
QFS_Init (hunk, "qw");
PI_Init ();
Cvar_Register (&qtv_console_plugin_cvar, 0, 0);
PI_RegisterPlugins (server_plugin_list);
Con_Load (qtv_console_plugin);
if (con_module)
con_module->data->console->cbuf = qtv_cbuf;
Con_Init ();
Sys_SetStdPrintf (qtv_print);
qtv_sbar_init ();
Netchan_Init_Cvars ();
Cmd_StuffCmds (qtv_cbuf);
Cbuf_Execute_Sets (qtv_cbuf);
qtv_net_init ();
Server_Init ();
Client_Init ();
Cmd_AddCommand ("quit", qtv_quit_f, "Shut down qtv");
Cmd_StuffCmds (qtv_cbuf);
}
2004-02-18 23:21:11 +00:00
static void
qtv_ping (void)
{
char data = A2A_ACK;
Netchan_SendPacket (1, &data, net_from);
}
static void
qtv_status (void)
{
qtv_begin_redirect (RD_PACKET, 0);
Sys_Printf ("\\*version\\%s qtv %s", QW_VERSION, PACKAGE_VERSION);
qtv_end_redirect ();
}
static void
qtv_connectionless_packet (void)
{
const char *cmd, *str;
MSG_BeginReading (net_message);
MSG_ReadLong (net_message); // skip the -1 marker
str = MSG_ReadString (net_message);
COM_TokenizeString (str, qtv_args);
cmd_args = qtv_args;
cmd = qtv_args->argv[0]->str;
if (!strcmp (cmd, "ping")) {
qtv_ping ();
} else if (!strcmp (cmd, "status")) {
qtv_status ();
} else if (!strcmp (cmd, "getchallenge")) {
Client_NewConnection ();
} else if (cmd[0]) {
switch (cmd[0]) {
default:
goto bad_packet;
case A2C_PRINT:
Sys_Printf ("%s", str + 1);
break;
case A2A_PING:
qtv_ping ();
break;
}
} else {
bad_packet:
Sys_Printf ("bad connectionless packet from %s:\n%s\n",
NET_AdrToString (net_from), str);
}
}
static void
qtv_read_packets (void)
{
connection_t *con;
while (NET_GetPacket ()) {
if ((con = Connection_Find (&net_from))) {
con->handler (con, con->object);
} else if (*(int *) net_message->message->data == -1) {
qtv_connectionless_packet ();
}
}
}
2004-02-18 23:21:11 +00:00
int
main (int argc, const char *argv[])
2004-02-18 23:21:11 +00:00
{
int frame = 0;
COM_InitArgv (argc, argv);
qtv_init ();
Sys_Printf ("Ohayou gozaimasu\n");
while (1) {
realtime = Sys_DoubleTime () + 1;
Cbuf_Execute_Stack (qtv_cbuf);
Sys_CheckInput (1, net_socket);
qtv_read_packets ();
Con_ProcessInput ();
Server_Frame ();
Client_Frame ();
if (++frame == 100) {
frame = 0;
Con_DrawConsole ();
}
}
2004-02-18 23:21:11 +00:00
return 0;
}