/* #FILENAME# #DESCRIPTION# Copyright (C) 2004 Bill Currie Author: Bill Currie Date: 2004/3/4 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 #endif #ifdef HAVE_STRINGS_H # include #endif #include "QF/cmd.h" #include "QF/cbuf.h" #include "QF/dstring.h" #include "QF/hash.h" #include "QF/idparse.h" #include "QF/info.h" #include "QF/sys.h" #include "QF/va.h" #include "compat.h" #include "qw/include/server.h" #include "qw/include/sv_qtv.h" #include "qw/include/sv_recorder.h" typedef struct { netchan_t netchan; backbuf_t backbuf; info_t *info; info_key_t *name_key; qboolean packet; recorder_t *recorder; sizebuf_t datagram; byte datagram_buf[MAX_DATAGRAM]; int begun; } sv_qtv_t; #define MAX_PROXIES 2 static sv_qtv_t proxies[MAX_PROXIES]; static sv_qtv_t * alloc_proxy (void) { int i; for (i = 0; i < MAX_PROXIES; i++) { if (!proxies[i].info) return proxies; } return 0; } static void drop_proxy (sv_qtv_t *proxy) { SV_Printf ("dropped %s\n", proxy->name_key->value); if (proxy->recorder) { SVR_RemoveUser (proxy->recorder); proxy->recorder = 0; } MSG_WriteByte (&proxy->netchan.message, qtv_disconnect); Netchan_Transmit (&proxy->netchan, 0, 0); proxy->packet = false; Info_Destroy (proxy->info); proxy->info = 0; } static void qtv_new_f (sv_qtv_t *proxy) { const char *gamedir; int pos, len; byte *buf; gamedir = Info_ValueForKey (svs.info, "*gamedir"); if (!gamedir[0]) gamedir = "qw"; MSG_WriteByte (&proxy->netchan.message, qtv_packet); pos = proxy->netchan.message.cursize; MSG_WriteShort (&proxy->netchan.message, qtv_p_signon); MSG_WriteByte (&proxy->netchan.message, svc_serverdata); MSG_WriteLong (&proxy->netchan.message, PROTOCOL_VERSION); MSG_WriteLong (&proxy->netchan.message, svs.spawncount); MSG_WriteString (&proxy->netchan.message, gamedir); SV_WriteWorldVars (&proxy->netchan); len = proxy->netchan.message.cursize - pos - 2; buf = proxy->netchan.message.data + pos; buf[0] = len & 0xff; buf[1] |= (len >> 8) & 0x0f; proxy->begun = 0; } static void qtv_soundlist_f (sv_qtv_t *proxy) { int n; int pos, len; byte *buf; if (atoi (Cmd_Argv (1)) != svs.spawncount) { qtv_new_f (proxy); return; } n = atoi (Cmd_Argv (2)); if (n >= MAX_SOUNDS) { qtv_new_f (proxy); return; } MSG_WriteByte (&proxy->netchan.message, qtv_packet); pos = proxy->netchan.message.cursize; MSG_WriteShort (&proxy->netchan.message, qtv_p_signon); SV_WriteSoundlist (&proxy->netchan, n); len = proxy->netchan.message.cursize - pos - 2; buf = proxy->netchan.message.data + pos; buf[0] = len & 0xff; buf[1] |= (len >> 8) & 0x0f; } static void qtv_modellist_f (sv_qtv_t *proxy) { int n; int pos, len; byte *buf; if (atoi (Cmd_Argv (1)) != svs.spawncount) { qtv_new_f (proxy); return; } n = atoi (Cmd_Argv (2)); if (n >= MAX_MODELS) { qtv_new_f (proxy); return; } MSG_WriteByte (&proxy->netchan.message, qtv_packet); pos = proxy->netchan.message.cursize; MSG_WriteShort (&proxy->netchan.message, qtv_p_signon); SV_WriteModellist (&proxy->netchan, n); len = proxy->netchan.message.cursize - pos - 2; buf = proxy->netchan.message.data + pos; buf[0] = len & 0xff; buf[1] |= (len >> 8) & 0x0f; } static void qtv_prespawn_f (sv_qtv_t *proxy) { int buf; int size; const char *command; sizebuf_t *msg; if (atoi (Cmd_Argv (1)) != svs.spawncount) { qtv_new_f (proxy); return; } buf = atoi (Cmd_Argv (2)); if (buf >= sv.num_signon_buffers) buf = 0; if (buf == sv.num_signon_buffers - 1) command = va (0, "cmd spawn %i 0\n", svs.spawncount); else command = va (0, "cmd prespawn %i %i\n", svs.spawncount, buf + 1); size = (3 + sv.signon_buffer_size[buf]) + (1 + strlen (command) + 1); msg = MSG_ReliableCheckBlock (&proxy->backbuf, size); MSG_WriteByte (msg, qtv_packet); MSG_WriteShort (msg, qtv_p_signon | sv.signon_buffer_size[buf]); SZ_Write (msg, sv.signon_buffers[buf], sv.signon_buffer_size[buf]); MSG_WriteByte (msg, qtv_stringcmd); MSG_WriteString (msg, command); } static void qtv_spawn_f (sv_qtv_t *proxy) { int pos = -1, len = 0; byte *buf = 0; if (atoi (Cmd_Argv (1)) != svs.spawncount) { qtv_new_f (proxy); return; } if (!proxy->backbuf.num_backbuf) { MSG_WriteByte (&proxy->netchan.message, qtv_packet); pos = proxy->netchan.message.cursize; MSG_WriteShort (&proxy->netchan.message, qtv_p_signon); } SV_WriteSpawn1 (&proxy->backbuf, 0); SV_WriteSpawn2 (&proxy->backbuf); MSG_ReliableWrite_Begin (&proxy->backbuf, svc_stufftext, 8); MSG_ReliableWrite_String (&proxy->backbuf, "skins\n"); if (pos != -1) { len = proxy->netchan.message.cursize - pos - 2; buf = proxy->netchan.message.data + pos; buf[0] = len & 0xff; buf[1] |= (len >> 8) & 0x0f; } } static void qtv_write (void *r, sizebuf_t *msg, int reliable) { sv_qtv_t *proxy = (sv_qtv_t *) r; sizebuf_t *buf; int type; if (!msg->cursize) return; if (reliable) { buf = MSG_ReliableCheckBlock (&proxy->backbuf, msg->cursize + 3); type = qtv_p_reliable; } else { buf = &proxy->datagram; type = qtv_p_unreliable; } MSG_WriteByte (buf, qtv_packet); MSG_WriteShort (buf, type | msg->cursize); SZ_Write (buf, msg->data, msg->cursize); } static int qtv_frame (void *r) { sv_qtv_t *proxy = (sv_qtv_t *) r; return proxy->packet; } static void qtv_finish (void *r, sizebuf_t *msg) { } static void qtv_begin_f (sv_qtv_t *proxy) { if (atoi (Cmd_Argv (1)) != svs.spawncount) { qtv_new_f (proxy); return; } if (!proxy->recorder) proxy->recorder = SVR_AddUser (qtv_write, qtv_frame, 0, qtv_finish, 0, proxy); else SVR_Continue (proxy->recorder); proxy->begun = 1; } typedef struct { const char *name; void (*func) (sv_qtv_t *proxy); unsigned no_redirect:1; } qcmd_t; qcmd_t qcmds[] = { {"drop", drop_proxy}, {"new", qtv_new_f}, {"soundlist", qtv_soundlist_f}, {"modellist", qtv_modellist_f}, {"prespawn", qtv_prespawn_f}, {"spawn", qtv_spawn_f}, {"begin", qtv_begin_f}, {0, 0}, }; static void qtv_command (sv_qtv_t *proxy, const char *s) { qcmd_t *c; const char *name; COM_TokenizeString (s, sv_args); cmd_args = sv_args; name = Cmd_Argv (0); for (c = qcmds; c->name; c++) if (strcmp (c->name, name) == 0) break; if (!c->name) { SV_Printf ("Bad QTV command: %s\n", sv_args->argv[0]->str); return; } c->func (proxy); } static void qtv_parse (sv_qtv_t *proxy) { int c, delta; // make sure the reply sequence number matches the incoming // sequence number if (proxy->netchan.incoming_sequence >= proxy->netchan.outgoing_sequence) { proxy->netchan.outgoing_sequence = proxy->netchan.incoming_sequence; proxy->packet = true; } else { SV_Printf ("qtv_parse: slipped sequence: %d %d\n", proxy->netchan.incoming_sequence, proxy->netchan.outgoing_sequence); proxy->packet = false; // don't reply, sequences have slipped return; // FIXME right thing? } if (proxy->recorder) SVR_SetDelta (proxy->recorder, -1, proxy->netchan.incoming_sequence); while (1) { if (net_message->badread) { SV_Printf ("qtv_parse: badread\n"); drop_proxy (proxy); return; } c = MSG_ReadByte (net_message); if (c == -1) return; switch (c) { default: SV_Printf ("qtv_parse: unknown command char\n"); drop_proxy (proxy); return; case qtv_nop: break; case qtv_stringcmd: qtv_command (proxy, MSG_ReadString (net_message)); break; case qtv_delta: delta = MSG_ReadByte (net_message); if (proxy->recorder) SVR_SetDelta (proxy->recorder, delta, proxy->netchan.incoming_sequence); break; } } } static void qtv_reliable_send (sv_qtv_t *proxy) { int pos = 0, len = 0; byte *buf = 0; SV_Printf ("proxy->begun: %d\n", proxy->begun); if (!proxy->begun) { MSG_WriteByte (&proxy->netchan.message, qtv_packet); pos = proxy->netchan.message.cursize; MSG_WriteShort (&proxy->netchan.message, qtv_p_signon); } MSG_Reliable_Send (&proxy->backbuf); if (!proxy->begun) { len = proxy->netchan.message.cursize - pos - 2; buf = proxy->netchan.message.data + pos; buf[0] = len & 0xff; buf[1] |= (len >> 8) & 0x0f; } } void SV_qtvConnect (int qport, info_t *info) { sv_qtv_t *proxy; SV_Printf ("QTV proxy connection: %s\n", NET_AdrToString (net_from)); if (!(proxy = alloc_proxy ())) { SV_Printf ("%s:full proxy connect\n", NET_AdrToString (net_from)); Netchan_OutOfBandPrint (net_from, "%c\nserver is full\n\n", A2C_PRINT); return; } proxy->info = info; while (!(proxy->name_key = Info_Key (proxy->info, "name"))) Info_SetValueForKey (proxy->info, "name", "\"unnamed proxy\"", 1); Netchan_Setup (&proxy->netchan, net_from, qport, NC_QPORT_READ); proxy->backbuf.netchan = &proxy->netchan; proxy->backbuf.name = proxy->name_key->value; proxy->datagram.data = proxy->datagram_buf; proxy->datagram.maxsize = sizeof (proxy->datagram_buf); proxy->begun = 0; Netchan_OutOfBandPrint (net_from, "%c", S2C_CONNECTION); } int SV_qtvPacket (int qport) { int i; for (i = 0; i < MAX_PROXIES; i++) { if (!proxies[i].info) continue; if (!NET_CompareBaseAdr (net_from, proxies[i].netchan.remote_address)) continue; if (proxies[i].netchan.qport != qport) continue; if (proxies[i].netchan.remote_address.port != net_from.port) { Sys_MaskPrintf (SYS_dev, "SV_ReadPackets: fixing up a translated port\n"); proxies[i].netchan.remote_address.port = net_from.port; } if (Netchan_Process (&proxies[i].netchan)) { // this is a valid, sequenced packet, so process it svs.stats.packets++; qtv_parse (&proxies[i]); } return 1; } return 0; } void SV_qtvCheckTimeouts (void) { int i; float droptime; sv_qtv_t *proxy; droptime = realtime - sv_timeout->value; for (i = 0; i < MAX_PROXIES; i++) { proxy = proxies + i; if (proxy->info) { if (proxy->netchan.last_received < droptime) { SV_Printf ("%s timed out\n", proxy->name_key->value); drop_proxy (proxy); } } } } void SV_qtvSendMessages (void) { int i; sv_qtv_t *proxy; for (i = 0; i < MAX_PROXIES; i++) { proxy = proxies + i; if (!proxy->info) continue; if (!proxy->packet) continue; if (proxy->backbuf.num_backbuf) qtv_reliable_send (proxy); if (proxy->netchan.message.overflowed) { SZ_Clear (&proxy->netchan.message); drop_proxy (proxy); continue; } if (proxy->datagram.cursize) { Netchan_Transmit (&proxy->netchan, proxy->datagram.cursize, proxy->datagram.data); SZ_Clear (&proxy->datagram); } else { Netchan_Transmit (&proxy->netchan, 0, 0); } proxy->packet = 0; } } void SV_qtvFinalMessage (const char *message) { sv_qtv_t *proxy; int i; int pos, len; byte *buf; SZ_Clear (net_message->message); MSG_WriteByte (net_message->message, qtv_packet); pos = net_message->message->cursize; MSG_WriteShort (net_message->message, qtv_p_print); MSG_WriteByte (net_message->message, PRINT_HIGH); MSG_WriteString (net_message->message, message); len = net_message->message->cursize - pos - 2; buf = net_message->message->data + pos; buf[0] = len & 0xff; buf[1] |= (len >> 8) & 0x0f; MSG_WriteByte (net_message->message, qtv_disconnect); for (i = 0; i < MAX_PROXIES; i++) { proxy = proxies + i; if (!proxy->info) continue; Netchan_Transmit (&proxy->netchan, net_message->message->cursize, net_message->message->data); } } void SV_qtvChanging (void) { sv_qtv_t *proxy; int i, len; const char *msg; msg = va (0, "%cchanging", qtv_stringcmd); len = strlen (msg) + 1; for (i = 0; i < MAX_PROXIES; i++) { proxy = proxies + i; if (!proxy->info) continue; SVR_Pause (proxy->recorder); MSG_ReliableCheckBlock (&proxy->backbuf, len); MSG_ReliableWrite_SZ (&proxy->backbuf, msg, len); } SV_qtvSendMessages (); } void SV_qtvReconnect (void) { sv_qtv_t *proxy; int i, len; const char *msg; msg = va (0, "%creconnect", qtv_stringcmd); len = strlen (msg) + 1; for (i = 0; i < MAX_PROXIES; i++) { proxy = proxies + i; if (!proxy->info) continue; MSG_ReliableCheckBlock (&proxy->backbuf, len); MSG_ReliableWrite_SZ (&proxy->backbuf, msg, len); } }