2000-05-10 20:33:16 +00:00
|
|
|
/*
|
2000-05-11 16:03:29 +00:00
|
|
|
cl_pred.c
|
2000-05-10 20:33:16 +00:00
|
|
|
|
2000-05-11 16:03:29 +00:00
|
|
|
(description)
|
2000-05-10 20:33:16 +00:00
|
|
|
|
2000-05-11 16:03:29 +00:00
|
|
|
Copyright (C) 1996-1997 Id Software, Inc.
|
2000-05-10 20:33:16 +00:00
|
|
|
|
2000-05-11 16:03:29 +00:00
|
|
|
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.
|
2000-05-10 20:33:16 +00:00
|
|
|
|
2000-05-11 16:03:29 +00:00
|
|
|
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.
|
2000-05-10 20:33:16 +00:00
|
|
|
|
2000-05-11 16:03:29 +00:00
|
|
|
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
|
|
|
|
|
|
|
|
$Id$
|
2000-05-10 20:33:16 +00:00
|
|
|
*/
|
2000-05-11 16:03:29 +00:00
|
|
|
|
2000-05-17 10:03:19 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
2000-05-22 07:03:41 +00:00
|
|
|
# include "config.h"
|
2000-05-17 10:03:19 +00:00
|
|
|
#endif
|
2000-12-30 02:16:36 +00:00
|
|
|
|
2001-01-24 17:41:34 +00:00
|
|
|
#include <math.h>
|
|
|
|
|
2000-05-21 08:24:45 +00:00
|
|
|
#include "bothdefs.h"
|
2000-10-29 15:35:24 +00:00
|
|
|
#include "cl_ents.h"
|
2000-12-30 02:16:36 +00:00
|
|
|
#include "client.h"
|
|
|
|
#include "commdef.h"
|
|
|
|
#include "console.h"
|
|
|
|
#include "cvar.h"
|
2001-05-15 05:35:42 +00:00
|
|
|
#include "keys.h"
|
2001-04-22 05:18:03 +00:00
|
|
|
#include "mathlib.h"
|
2000-12-30 02:16:36 +00:00
|
|
|
#include "pmove.h"
|
2000-05-21 08:24:45 +00:00
|
|
|
|
2000-12-08 06:51:37 +00:00
|
|
|
cvar_t *cl_nopred;
|
|
|
|
cvar_t *cl_pushlatency;
|
2001-02-19 15:57:36 +00:00
|
|
|
cvar_t *cl_nostatpred;
|
2000-05-10 11:29:38 +00:00
|
|
|
|
2000-12-08 06:51:37 +00:00
|
|
|
extern frame_t *view_frame;
|
2000-05-10 11:29:38 +00:00
|
|
|
|
|
|
|
/*
|
2001-02-09 02:53:09 +00:00
|
|
|
CL_PredictUsercmd
|
2000-05-10 11:29:38 +00:00
|
|
|
*/
|
2000-12-08 06:51:37 +00:00
|
|
|
void
|
|
|
|
CL_PredictUsercmd (player_state_t * from, player_state_t * to, usercmd_t *u,
|
|
|
|
qboolean spectator)
|
2000-05-10 11:29:38 +00:00
|
|
|
{
|
2001-02-19 15:57:36 +00:00
|
|
|
|
|
|
|
// Dabb: if there is no movement to start with, don't predict...
|
2001-03-16 12:27:11 +00:00
|
|
|
if(cl_nostatpred->int_val && VectorIsNull(from->velocity)) {
|
2001-03-16 12:10:59 +00:00
|
|
|
VectorCopy (from->origin, to->origin);
|
|
|
|
VectorCopy (u->angles, to->viewangles);
|
|
|
|
VectorCopy (from->velocity, to->velocity);
|
2001-02-19 15:57:36 +00:00
|
|
|
return;
|
2001-03-16 12:10:59 +00:00
|
|
|
}
|
2001-02-19 15:57:36 +00:00
|
|
|
|
2000-05-10 11:29:38 +00:00
|
|
|
// split up very long moves
|
2000-12-08 06:51:37 +00:00
|
|
|
if (u->msec > 50) {
|
|
|
|
player_state_t temp;
|
|
|
|
usercmd_t split;
|
2000-05-10 11:29:38 +00:00
|
|
|
|
|
|
|
split = *u;
|
|
|
|
split.msec /= 2;
|
|
|
|
|
|
|
|
CL_PredictUsercmd (from, &temp, &split, spectator);
|
|
|
|
CL_PredictUsercmd (&temp, to, &split, spectator);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
VectorCopy (from->origin, pmove.origin);
|
2000-12-08 06:51:37 +00:00
|
|
|
// VectorCopy (from->viewangles, pmove.angles);
|
2000-05-10 11:29:38 +00:00
|
|
|
VectorCopy (u->angles, pmove.angles);
|
|
|
|
VectorCopy (from->velocity, pmove.velocity);
|
|
|
|
|
|
|
|
pmove.oldbuttons = from->oldbuttons;
|
|
|
|
pmove.waterjumptime = from->waterjumptime;
|
|
|
|
pmove.dead = cl.stats[STAT_HEALTH] <= 0;
|
|
|
|
pmove.spectator = spectator;
|
2000-05-13 22:51:05 +00:00
|
|
|
pmove.flying = cl.stats[STAT_FLYMODE];
|
2000-05-10 11:29:38 +00:00
|
|
|
|
|
|
|
pmove.cmd = *u;
|
|
|
|
|
|
|
|
PlayerMove ();
|
|
|
|
//for (i=0 ; i<3 ; i++)
|
|
|
|
//pmove.origin[i] = ((int)(pmove.origin[i]*8))*0.125;
|
|
|
|
to->waterjumptime = pmove.waterjumptime;
|
2000-05-22 07:03:41 +00:00
|
|
|
to->oldbuttons = pmove.oldbuttons; // Tonik
|
2000-12-08 06:51:37 +00:00
|
|
|
// to->oldbuttons = pmove.cmd.buttons;
|
2000-05-10 11:29:38 +00:00
|
|
|
VectorCopy (pmove.origin, to->origin);
|
|
|
|
VectorCopy (pmove.angles, to->viewangles);
|
|
|
|
VectorCopy (pmove.velocity, to->velocity);
|
|
|
|
to->onground = onground;
|
|
|
|
|
|
|
|
to->weaponframe = from->weaponframe;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2001-02-09 02:53:09 +00:00
|
|
|
CL_PredictMove
|
2000-05-10 11:29:38 +00:00
|
|
|
*/
|
2000-12-08 06:51:37 +00:00
|
|
|
void
|
|
|
|
CL_PredictMove (void)
|
2000-05-10 11:29:38 +00:00
|
|
|
{
|
2000-12-08 06:51:37 +00:00
|
|
|
int i;
|
|
|
|
float f;
|
|
|
|
frame_t *from, *to = NULL;
|
|
|
|
int oldphysent;
|
2000-05-10 20:33:16 +00:00
|
|
|
|
2000-05-16 04:47:41 +00:00
|
|
|
if (cl_pushlatency->value > 0)
|
2000-05-17 23:16:23 +00:00
|
|
|
Cvar_Set (cl_pushlatency, "0");
|
2000-05-10 11:29:38 +00:00
|
|
|
|
2000-05-10 20:33:16 +00:00
|
|
|
if (cl.paused)
|
|
|
|
return;
|
|
|
|
|
2001-03-16 12:27:11 +00:00
|
|
|
// assume on ground unless prediction says different
|
|
|
|
cl.onground = 0;
|
2000-12-15 12:18:54 +00:00
|
|
|
|
2000-12-08 06:51:37 +00:00
|
|
|
cl.time = realtime - cls.latency - cl_pushlatency->value * 0.001;
|
2000-05-10 11:29:38 +00:00
|
|
|
if (cl.time > realtime)
|
|
|
|
cl.time = realtime;
|
|
|
|
|
|
|
|
if (cl.intermission)
|
2000-05-10 20:33:16 +00:00
|
|
|
return;
|
|
|
|
|
2000-05-10 11:29:38 +00:00
|
|
|
if (!cl.validsequence)
|
|
|
|
return;
|
|
|
|
|
2000-12-08 06:51:37 +00:00
|
|
|
if (cls.netchan.outgoing_sequence - cls.netchan.incoming_sequence >=
|
|
|
|
UPDATE_BACKUP - 1)
|
2000-05-10 11:29:38 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
VectorCopy (cl.viewangles, cl.simangles);
|
|
|
|
|
|
|
|
// this is the last frame received from the server
|
|
|
|
from = &cl.frames[cls.netchan.incoming_sequence & UPDATE_MASK];
|
|
|
|
|
2000-05-10 20:33:16 +00:00
|
|
|
// we can now render a frame
|
2001-03-16 12:27:11 +00:00
|
|
|
if (cls.state == ca_onserver) {
|
|
|
|
// first update is the final signon stage
|
2000-12-08 06:51:37 +00:00
|
|
|
VID_SetCaption (cls.servername);
|
2000-10-04 05:45:48 +00:00
|
|
|
cls.state = ca_active;
|
2001-05-15 05:35:42 +00:00
|
|
|
key_dest = key_game;
|
2000-05-10 20:33:16 +00:00
|
|
|
}
|
|
|
|
|
2000-12-08 06:51:37 +00:00
|
|
|
if (cl_nopred->int_val) {
|
2000-05-10 11:29:38 +00:00
|
|
|
VectorCopy (from->playerstate[cl.playernum].velocity, cl.simvel);
|
|
|
|
VectorCopy (from->playerstate[cl.playernum].origin, cl.simorg);
|
|
|
|
return;
|
|
|
|
}
|
2001-02-19 15:57:36 +00:00
|
|
|
|
|
|
|
// Dabb: if there is no movement to start with, don't predict...
|
|
|
|
|
2001-03-16 12:27:11 +00:00
|
|
|
if(cl_nostatpred->int_val
|
|
|
|
&& VectorIsNull(from->playerstate[cl.playernum].velocity)) {
|
2001-02-19 15:57:36 +00:00
|
|
|
VectorCopy (from->playerstate[cl.playernum].velocity, cl.simvel);
|
|
|
|
VectorCopy (from->playerstate[cl.playernum].origin, cl.simorg);
|
|
|
|
return;
|
2001-03-16 12:10:59 +00:00
|
|
|
}
|
2001-02-19 15:57:36 +00:00
|
|
|
|
2001-03-16 12:10:59 +00:00
|
|
|
// predict forward until cl.time <= to->senttime
|
2000-05-10 20:33:16 +00:00
|
|
|
oldphysent = pmove.numphysent;
|
|
|
|
CL_SetSolidPlayers (cl.playernum);
|
|
|
|
|
2000-12-08 06:51:37 +00:00
|
|
|
// to = &cl.frames[cls.netchan.incoming_sequence & UPDATE_MASK];
|
2000-05-10 20:33:16 +00:00
|
|
|
|
2000-12-08 06:51:37 +00:00
|
|
|
for (i = 1; i < UPDATE_BACKUP - 1 && cls.netchan.incoming_sequence + i <
|
|
|
|
cls.netchan.outgoing_sequence; i++) {
|
|
|
|
to = &cl.frames[(cls.netchan.incoming_sequence + i) & UPDATE_MASK];
|
2001-03-16 12:27:11 +00:00
|
|
|
CL_PredictUsercmd (&from->playerstate[cl.playernum],
|
|
|
|
&to->playerstate[cl.playernum], &to->cmd,
|
2000-12-08 06:51:37 +00:00
|
|
|
cl.spectator);
|
2000-12-15 12:18:54 +00:00
|
|
|
cl.onground = onground;
|
2000-05-10 11:29:38 +00:00
|
|
|
if (to->senttime >= cl.time)
|
|
|
|
break;
|
|
|
|
from = to;
|
|
|
|
}
|
2000-05-10 20:33:16 +00:00
|
|
|
|
|
|
|
pmove.numphysent = oldphysent;
|
|
|
|
|
2000-12-08 06:51:37 +00:00
|
|
|
if (i == UPDATE_BACKUP - 1 || !to)
|
|
|
|
return; // net hasn't deliver packets in a
|
|
|
|
// long time...
|
2000-05-10 11:29:38 +00:00
|
|
|
|
|
|
|
// now interpolate some fraction of the final frame
|
|
|
|
if (to->senttime == from->senttime)
|
|
|
|
f = 0;
|
2001-03-16 12:27:11 +00:00
|
|
|
else
|
|
|
|
f = bound(0, (cl.time - from->senttime) / (to->senttime - from->senttime), 1);
|
2000-05-10 11:29:38 +00:00
|
|
|
|
2000-12-08 06:51:37 +00:00
|
|
|
for (i = 0; i < 3; i++)
|
2001-03-16 12:27:11 +00:00
|
|
|
if (fabs(from->playerstate[cl.playernum].origin[i] -
|
|
|
|
to->playerstate[cl.playernum].origin[i]) > 128) {
|
|
|
|
// teleported, so don't lerp
|
2000-05-10 11:29:38 +00:00
|
|
|
VectorCopy (to->playerstate[cl.playernum].velocity, cl.simvel);
|
|
|
|
VectorCopy (to->playerstate[cl.playernum].origin, cl.simorg);
|
|
|
|
return;
|
|
|
|
}
|
2000-12-08 06:51:37 +00:00
|
|
|
|
|
|
|
for (i = 0; i < 3; i++) {
|
|
|
|
cl.simorg[i] = from->playerstate[cl.playernum].origin[i]
|
|
|
|
+ f * (to->playerstate[cl.playernum].origin[i] -
|
|
|
|
from->playerstate[cl.playernum].origin[i]);
|
|
|
|
cl.simvel[i] = from->playerstate[cl.playernum].velocity[i]
|
|
|
|
+ f * (to->playerstate[cl.playernum].velocity[i] -
|
|
|
|
from->playerstate[cl.playernum].velocity[i]);
|
|
|
|
}
|
2000-05-10 11:29:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2001-02-09 02:53:09 +00:00
|
|
|
CL_Prediction_Init_Cvars
|
2000-05-10 11:29:38 +00:00
|
|
|
*/
|
2000-12-08 06:51:37 +00:00
|
|
|
void
|
|
|
|
CL_Prediction_Init_Cvars (void)
|
2000-05-10 11:29:38 +00:00
|
|
|
{
|
2000-12-31 04:46:04 +00:00
|
|
|
/* I'm not totally sure what cl_pushlatency is for. Or if it is SUPPOSED TO BE SETTABLE. */
|
2001-05-13 05:10:02 +00:00
|
|
|
cl_pushlatency = Cvar_Get ("pushlatency", "-999", CVAR_NONE, NULL, "How much prediction should the client make");
|
|
|
|
cl_nopred = Cvar_Get ("cl_nopred", "0", CVAR_NONE, NULL, "Set to turn off client prediction");
|
|
|
|
cl_nostatpred = Cvar_Get ("cl_nostatpred", "0", CVAR_NONE, NULL, "Set to turn off static player prediction");
|
2000-05-10 11:29:38 +00:00
|
|
|
}
|