2020-03-19 20:09:55 +00:00
// SONIC ROBO BLAST 2
//-----------------------------------------------------------------------------
// Copyright (C) 1993-1996 by id Software, Inc.
// Copyright (C) 1998-2000 by DooM Legacy Team.
2024-09-22 00:35:01 +00:00
// Copyright (C) 1999-2024 by Sonic Team Junior.
2020-03-19 20:09:55 +00:00
//
// This program is free software distributed under the
// terms of the GNU General Public License, version 2.
// See the 'LICENSE' file for more details.
//-----------------------------------------------------------------------------
/// \file g_demo.c
/// \brief Demo recording and playback
# include "doomdef.h"
# include "console.h"
# include "d_main.h"
# include "d_player.h"
2022-12-31 13:10:19 +00:00
# include "netcode/d_clisrv.h"
2020-03-19 20:09:55 +00:00
# include "p_setup.h"
2022-04-30 21:33:23 +00:00
# include "i_time.h"
2020-03-19 20:09:55 +00:00
# include "i_system.h"
# include "m_random.h"
# include "p_local.h"
# include "r_draw.h"
# include "r_main.h"
# include "g_game.h"
# include "g_demo.h"
# include "m_misc.h"
# include "m_menu.h"
# include "m_argv.h"
# include "hu_stuff.h"
# include "z_zone.h"
# include "i_video.h"
# include "byteptr.h"
# include "i_joy.h"
# include "r_local.h"
# include "r_skins.h"
# include "y_inter.h"
# include "v_video.h"
# include "lua_hook.h"
# include "md5.h" // demo checksums
2023-08-01 16:24:07 +00:00
# include "netcode/d_netfil.h" // G_CheckDemoExtraFiles
2020-03-19 20:09:55 +00:00
boolean timingdemo ; // if true, exit with report on completion
boolean nodrawers ; // for comparative timing purposes
boolean noblit ; // for comparative timing purposes
tic_t demostarttime ; // for comparative timing purposes
static char demoname [ 64 ] ;
boolean demorecording ;
boolean demoplayback ;
boolean titledemo ; // Title Screen demo can be cancelled by any key
2023-07-25 12:40:36 +00:00
demo_file_override_e demofileoverride ;
2020-03-19 20:09:55 +00:00
static UINT8 * demobuffer = NULL ;
static UINT8 * demo_p , * demotime_p ;
static UINT8 * demoend ;
static UINT8 demoflags ;
2023-12-29 15:04:03 +00:00
UINT16 demoversion ;
2020-03-19 20:09:55 +00:00
boolean singledemo ; // quit after playing a demo from cmdline
boolean demo_start ; // don't start playing demo right away
2023-07-26 11:50:47 +00:00
boolean demo_forwardmove_rng ; // old demo backwards compatibility
2020-03-19 20:09:55 +00:00
boolean demosynced = true ; // console warning message
boolean metalrecording ; // recording as metal sonic
mobj_t * metalplayback ;
static UINT8 * metalbuffer = NULL ;
static UINT8 * metal_p ;
static UINT16 metalversion ;
2024-02-21 21:22:00 +00:00
consvar_t cv_resyncdemo = CVAR_INIT ( " resyncdemo " , " On " , 0 , CV_OnOff , NULL ) ;
2020-03-19 20:09:55 +00:00
// extra data stuff (events registered this frame while recording)
static struct {
UINT8 flags ; // EZT flags
// EZT_COLOR
2020-05-24 00:29:07 +00:00
UINT16 color , lastcolor ;
2020-03-19 20:09:55 +00:00
// EZT_SCALE
fixed_t scale , lastscale ;
// EZT_HIT
UINT16 hits ;
mobj_t * * hitlist ;
} ghostext ;
// Your naming conventions are stupid and useless.
// There is no conflict here.
typedef struct demoghost {
UINT8 checksum [ 16 ] ;
2020-05-24 00:29:07 +00:00
UINT8 * buffer , * p , fadein ;
UINT16 color ;
2020-03-19 20:09:55 +00:00
UINT16 version ;
mobj_t oldmo , * mo ;
struct demoghost * next ;
} demoghost ;
demoghost * ghosts = NULL ;
//
// DEMO RECORDING
//
2024-02-24 14:41:36 +00:00
# define DEMOVERSION 0x0012
2020-03-19 20:09:55 +00:00
# define DEMOHEADER "\xF0" "SRB2Replay" "\x0F"
# define DF_GHOST 0x01 // This demo contains ghost data too!
# define DF_RECORDATTACK 0x02 // This demo is from record attack and contains its final completion time, score, and rings!
# define DF_NIGHTSATTACK 0x04 // This demo is from NiGHTS attack and contains its time left, score, and mares!
# define DF_ATTACKMASK 0x06 // This demo is from ??? attack and contains ???
# define DF_ATTACKSHIFT 1
// For demos
# define ZT_FWD 0x01
# define ZT_SIDE 0x02
# define ZT_ANGLE 0x04
# define ZT_BUTTONS 0x08
# define ZT_AIMING 0x10
2021-04-21 23:58:14 +00:00
# define ZT_LATENCY 0x20
2020-03-19 20:09:55 +00:00
# define DEMOMARKER 0x80 // demoend
# define METALDEATH 0x44
# define METALSNICE 0x69
static ticcmd_t oldcmd ;
// For Metal Sonic and time attack ghosts
# define GZT_XYZ 0x01
# define GZT_MOMXY 0x02
# define GZT_MOMZ 0x04
# define GZT_ANGLE 0x08
# define GZT_FRAME 0x10 // Animation frame
# define GZT_SPR2 0x20 // Player animations
# define GZT_EXTRA 0x40
# define GZT_FOLLOW 0x80 // Followmobj
// GZT_EXTRA flags
# define EZT_THOK 0x01 // Spawned a thok object
# define EZT_SPIN 0x02 // Because one type of thok object apparently wasn't enough
# define EZT_REV 0x03 // And two types wasn't enough either yet
# define EZT_THOKMASK 0x03
# define EZT_COLOR 0x04 // Changed color (Super transformation, Mario fireflowers/invulnerability, etc.)
# define EZT_FLIP 0x08 // Reversed gravity
# define EZT_SCALE 0x10 // Changed size
# define EZT_HIT 0x20 // Damaged a mobj
# define EZT_SPRITE 0x40 // Changed sprite set completely out of PLAY (NiGHTS, SOCs, whatever)
# define EZT_HEIGHT 0x80 // Changed height
// GZT_FOLLOW flags
# define FZT_SPAWNED 0x01 // just been spawned
# define FZT_SKIN 0x02 // has skin
# define FZT_LINKDRAW 0x04 // has linkdraw (combine with spawned only)
# define FZT_COLORIZED 0x08 // colorized (ditto)
# define FZT_SCALE 0x10 // different scale to object
// spare FZT slots 0x20 to 0x80
static mobj_t oldmetal , oldghost ;
void G_SaveMetal ( UINT8 * * buffer )
{
I_Assert ( buffer ! = NULL & & * buffer ! = NULL ) ;
WRITEUINT32 ( * buffer , metal_p - metalbuffer ) ;
}
void G_LoadMetal ( UINT8 * * buffer )
{
I_Assert ( buffer ! = NULL & & * buffer ! = NULL ) ;
G_DoPlayMetal ( ) ;
metal_p = metalbuffer + READUINT32 ( * buffer ) ;
}
void G_ReadDemoTiccmd ( ticcmd_t * cmd , INT32 playernum )
{
UINT8 ziptic ;
if ( ! demo_p | | ! demo_start )
return ;
ziptic = READUINT8 ( demo_p ) ;
if ( ziptic & ZT_FWD )
oldcmd . forwardmove = READSINT8 ( demo_p ) ;
if ( ziptic & ZT_SIDE )
oldcmd . sidemove = READSINT8 ( demo_p ) ;
if ( ziptic & ZT_ANGLE )
oldcmd . angleturn = READINT16 ( demo_p ) ;
if ( ziptic & ZT_BUTTONS )
2024-02-24 14:41:36 +00:00
{
2020-03-19 20:09:55 +00:00
oldcmd . buttons = ( oldcmd . buttons & ( BT_CAMLEFT | BT_CAMRIGHT ) ) | ( READUINT16 ( demo_p ) & ~ ( BT_CAMLEFT | BT_CAMRIGHT ) ) ;
2024-02-24 14:41:36 +00:00
if ( demoversion < 0x0012 & & oldcmd . buttons & BT_SPIN )
oldcmd . buttons | = BT_SHIELD ; // Copy BT_SPIN to BT_SHIELD for pre-Shield-button demos
}
2020-03-19 20:09:55 +00:00
if ( ziptic & ZT_AIMING )
oldcmd . aiming = READINT16 ( demo_p ) ;
2021-04-21 23:58:14 +00:00
if ( ziptic & ZT_LATENCY )
oldcmd . latency = READUINT8 ( demo_p ) ;
2020-03-19 20:09:55 +00:00
G_CopyTiccmd ( cmd , & oldcmd , 1 ) ;
2020-05-28 09:03:35 +00:00
players [ playernum ] . angleturn = cmd - > angleturn ;
2020-03-19 20:09:55 +00:00
if ( ! ( demoflags & DF_GHOST ) & & * demo_p = = DEMOMARKER )
{
// end of demo data stream
G_CheckDemoStatus ( ) ;
return ;
}
}
void G_WriteDemoTiccmd ( ticcmd_t * cmd , INT32 playernum )
{
char ziptic = 0 ;
UINT8 * ziptic_p ;
( void ) playernum ;
if ( ! demo_p )
return ;
ziptic_p = demo_p + + ; // the ziptic, written at the end of this function
if ( cmd - > forwardmove ! = oldcmd . forwardmove )
{
WRITEUINT8 ( demo_p , cmd - > forwardmove ) ;
oldcmd . forwardmove = cmd - > forwardmove ;
ziptic | = ZT_FWD ;
}
if ( cmd - > sidemove ! = oldcmd . sidemove )
{
WRITEUINT8 ( demo_p , cmd - > sidemove ) ;
oldcmd . sidemove = cmd - > sidemove ;
ziptic | = ZT_SIDE ;
}
if ( cmd - > angleturn ! = oldcmd . angleturn )
{
WRITEINT16 ( demo_p , cmd - > angleturn ) ;
oldcmd . angleturn = cmd - > angleturn ;
ziptic | = ZT_ANGLE ;
}
if ( cmd - > buttons ! = oldcmd . buttons )
{
WRITEUINT16 ( demo_p , cmd - > buttons ) ;
oldcmd . buttons = cmd - > buttons ;
ziptic | = ZT_BUTTONS ;
}
if ( cmd - > aiming ! = oldcmd . aiming )
{
WRITEINT16 ( demo_p , cmd - > aiming ) ;
oldcmd . aiming = cmd - > aiming ;
ziptic | = ZT_AIMING ;
}
2021-04-21 23:58:14 +00:00
if ( cmd - > latency ! = oldcmd . latency )
{
WRITEUINT8 ( demo_p , cmd - > latency ) ;
oldcmd . latency = cmd - > latency ;
ziptic | = ZT_LATENCY ;
}
2020-03-19 20:09:55 +00:00
* ziptic_p = ziptic ;
// attention here for the ticcmd size!
// latest demos with mouse aiming byte in ticcmd
if ( ! ( demoflags & DF_GHOST ) & & ziptic_p > demoend - 9 )
{
G_CheckDemoStatus ( ) ; // no more space
return ;
}
}
void G_GhostAddThok ( void )
{
if ( ! metalrecording & & ( ! demorecording | | ! ( demoflags & DF_GHOST ) ) )
return ;
ghostext . flags = ( ghostext . flags & ~ EZT_THOKMASK ) | EZT_THOK ;
}
void G_GhostAddSpin ( void )
{
if ( ! metalrecording & & ( ! demorecording | | ! ( demoflags & DF_GHOST ) ) )
return ;
ghostext . flags = ( ghostext . flags & ~ EZT_THOKMASK ) | EZT_SPIN ;
}
void G_GhostAddRev ( void )
{
if ( ! metalrecording & & ( ! demorecording | | ! ( demoflags & DF_GHOST ) ) )
return ;
ghostext . flags = ( ghostext . flags & ~ EZT_THOKMASK ) | EZT_REV ;
}
void G_GhostAddFlip ( void )
{
if ( ! metalrecording & & ( ! demorecording | | ! ( demoflags & DF_GHOST ) ) )
return ;
ghostext . flags | = EZT_FLIP ;
}
void G_GhostAddColor ( ghostcolor_t color )
{
if ( ! demorecording | | ! ( demoflags & DF_GHOST ) )
return ;
2020-05-24 00:29:07 +00:00
if ( ghostext . lastcolor = = ( UINT16 ) color )
2020-03-19 20:09:55 +00:00
{
ghostext . flags & = ~ EZT_COLOR ;
return ;
}
ghostext . flags | = EZT_COLOR ;
2020-05-24 00:29:07 +00:00
ghostext . color = ( UINT16 ) color ;
2020-03-19 20:09:55 +00:00
}
void G_GhostAddScale ( fixed_t scale )
{
if ( ! metalrecording & & ( ! demorecording | | ! ( demoflags & DF_GHOST ) ) )
return ;
if ( ghostext . lastscale = = scale )
{
ghostext . flags & = ~ EZT_SCALE ;
return ;
}
ghostext . flags | = EZT_SCALE ;
ghostext . scale = scale ;
}
void G_GhostAddHit ( mobj_t * victim )
{
if ( ! demorecording | | ! ( demoflags & DF_GHOST ) )
return ;
ghostext . flags | = EZT_HIT ;
ghostext . hits + + ;
ghostext . hitlist = Z_Realloc ( ghostext . hitlist , ghostext . hits * sizeof ( mobj_t * ) , PU_LEVEL , NULL ) ;
ghostext . hitlist [ ghostext . hits - 1 ] = victim ;
}
void G_WriteGhostTic ( mobj_t * ghost )
{
char ziptic = 0 ;
UINT8 * ziptic_p ;
UINT32 i ;
fixed_t height ;
if ( ! demo_p )
return ;
if ( ! ( demoflags & DF_GHOST ) )
return ; // No ghost data to write.
ziptic_p = demo_p + + ; // the ziptic, written at the end of this function
# define MAXMOM (0xFFFF<<8)
// GZT_XYZ is only useful if you've moved 256 FRACUNITS or more in a single tic.
if ( abs ( ghost - > x - oldghost . x ) > MAXMOM
| | abs ( ghost - > y - oldghost . y ) > MAXMOM
| | abs ( ghost - > z - oldghost . z ) > MAXMOM )
{
oldghost . x = ghost - > x ;
oldghost . y = ghost - > y ;
oldghost . z = ghost - > z ;
ziptic | = GZT_XYZ ;
WRITEFIXED ( demo_p , oldghost . x ) ;
WRITEFIXED ( demo_p , oldghost . y ) ;
WRITEFIXED ( demo_p , oldghost . z ) ;
}
else
{
// For moving normally:
2020-09-23 18:26:51 +00:00
fixed_t momx = ghost - > x - oldghost . x ;
fixed_t momy = ghost - > y - oldghost . y ;
2020-03-19 20:09:55 +00:00
if ( momx ! = oldghost . momx
| | momy ! = oldghost . momy )
{
oldghost . momx = momx ;
oldghost . momy = momy ;
ziptic | = GZT_MOMXY ;
2020-09-23 18:26:51 +00:00
WRITEFIXED ( demo_p , momx ) ;
WRITEFIXED ( demo_p , momy ) ;
2020-03-19 20:09:55 +00:00
}
2020-09-23 18:26:51 +00:00
momx = ghost - > z - oldghost . z ;
2020-03-19 20:09:55 +00:00
if ( momx ! = oldghost . momz )
{
oldghost . momz = momx ;
ziptic | = GZT_MOMZ ;
2020-09-23 18:26:51 +00:00
WRITEFIXED ( demo_p , momx ) ;
2020-03-19 20:09:55 +00:00
}
// This SHOULD set oldghost.x/y/z to match ghost->x/y/z
2020-09-23 18:26:51 +00:00
oldghost . x + = oldghost . momx ;
oldghost . y + = oldghost . momy ;
oldghost . z + = oldghost . momz ;
2020-03-19 20:09:55 +00:00
}
# undef MAXMOM
// Only store the 8 most relevant bits of angle
// because exact values aren't too easy to discern to begin with when only 8 angles have different sprites
// and it does not affect this mode of movement at all anyway.
if ( ghost - > player & & ghost - > player - > drawangle > > 24 ! = oldghost . angle )
{
oldghost . angle = ghost - > player - > drawangle > > 24 ;
ziptic | = GZT_ANGLE ;
WRITEUINT8 ( demo_p , oldghost . angle ) ;
}
// Store the sprite frame.
if ( ( ghost - > frame & FF_FRAMEMASK ) ! = oldghost . frame )
{
oldghost . frame = ( ghost - > frame & FF_FRAMEMASK ) ;
ziptic | = GZT_FRAME ;
WRITEUINT8 ( demo_p , oldghost . frame ) ;
}
if ( ghost - > sprite = = SPR_PLAY
& & ghost - > sprite2 ! = oldghost . sprite2 )
{
oldghost . sprite2 = ghost - > sprite2 ;
ziptic | = GZT_SPR2 ;
2023-11-13 14:46:55 +00:00
WRITEUINT16 ( demo_p , oldghost . sprite2 ) ;
2020-03-19 20:09:55 +00:00
}
// Check for sprite set changes
if ( ghost - > sprite ! = oldghost . sprite )
{
oldghost . sprite = ghost - > sprite ;
ghostext . flags | = EZT_SPRITE ;
}
if ( ( height = FixedDiv ( ghost - > height , ghost - > scale ) ) ! = oldghost . height )
{
oldghost . height = height ;
ghostext . flags | = EZT_HEIGHT ;
}
if ( ghostext . flags )
{
ziptic | = GZT_EXTRA ;
if ( ghostext . color = = ghostext . lastcolor )
ghostext . flags & = ~ EZT_COLOR ;
if ( ghostext . scale = = ghostext . lastscale )
ghostext . flags & = ~ EZT_SCALE ;
WRITEUINT8 ( demo_p , ghostext . flags ) ;
if ( ghostext . flags & EZT_COLOR )
{
2020-05-24 00:29:07 +00:00
WRITEUINT16 ( demo_p , ghostext . color ) ;
2020-03-19 20:09:55 +00:00
ghostext . lastcolor = ghostext . color ;
}
if ( ghostext . flags & EZT_SCALE )
{
WRITEFIXED ( demo_p , ghostext . scale ) ;
ghostext . lastscale = ghostext . scale ;
}
if ( ghostext . flags & EZT_HIT )
{
WRITEUINT16 ( demo_p , ghostext . hits ) ;
for ( i = 0 ; i < ghostext . hits ; i + + )
{
mobj_t * mo = ghostext . hitlist [ i ] ;
//WRITEUINT32(demo_p,UINT32_MAX); // reserved for some method of determining exactly which mobj this is. (mobjnum doesn't work here.)
WRITEUINT32 ( demo_p , mo - > type ) ;
WRITEUINT16 ( demo_p , ( UINT16 ) mo - > health ) ;
WRITEFIXED ( demo_p , mo - > x ) ;
WRITEFIXED ( demo_p , mo - > y ) ;
WRITEFIXED ( demo_p , mo - > z ) ;
WRITEANGLE ( demo_p , mo - > angle ) ;
}
Z_Free ( ghostext . hitlist ) ;
ghostext . hits = 0 ;
ghostext . hitlist = NULL ;
}
if ( ghostext . flags & EZT_SPRITE )
WRITEUINT16 ( demo_p , oldghost . sprite ) ;
if ( ghostext . flags & EZT_HEIGHT )
{
2020-11-09 15:01:20 +00:00
WRITEFIXED ( demo_p , height ) ;
2020-03-19 20:09:55 +00:00
}
ghostext . flags = 0 ;
}
if ( ghost - > player & & ghost - > player - > followmobj & & ! ( ghost - > player - > followmobj - > sprite = = SPR_NULL | | ( ghost - > player - > followmobj - > flags2 & MF2_DONTDRAW ) ) ) // bloats tails runs but what can ya do
{
2020-09-23 18:26:51 +00:00
fixed_t temp ;
2020-03-19 20:09:55 +00:00
UINT8 * followtic_p = demo_p + + ;
UINT8 followtic = 0 ;
ziptic | = GZT_FOLLOW ;
if ( ghost - > player - > followmobj - > skin )
followtic | = FZT_SKIN ;
if ( ! ( oldghost . flags2 & MF2_AMBUSH ) )
{
followtic | = FZT_SPAWNED ;
WRITEINT16 ( demo_p , ghost - > player - > followmobj - > info - > height > > FRACBITS ) ;
if ( ghost - > player - > followmobj - > flags2 & MF2_LINKDRAW )
followtic | = FZT_LINKDRAW ;
if ( ghost - > player - > followmobj - > colorized )
followtic | = FZT_COLORIZED ;
if ( followtic & FZT_SKIN )
2021-08-09 18:57:07 +00:00
WRITEUINT8 ( demo_p , ( UINT8 ) ( ( ( skin_t * ) ghost - > player - > followmobj - > skin ) - > skinnum ) ) ;
2020-03-19 20:09:55 +00:00
oldghost . flags2 | = MF2_AMBUSH ;
}
if ( ghost - > player - > followmobj - > scale ! = ghost - > scale )
{
followtic | = FZT_SCALE ;
WRITEFIXED ( demo_p , ghost - > player - > followmobj - > scale ) ;
}
2020-09-23 18:26:51 +00:00
temp = ghost - > player - > followmobj - > x - ghost - > x ;
WRITEFIXED ( demo_p , temp ) ;
temp = ghost - > player - > followmobj - > y - ghost - > y ;
WRITEFIXED ( demo_p , temp ) ;
temp = ghost - > player - > followmobj - > z - ghost - > z ;
WRITEFIXED ( demo_p , temp ) ;
2020-03-19 20:09:55 +00:00
if ( followtic & FZT_SKIN )
2023-11-13 14:46:55 +00:00
WRITEUINT16 ( demo_p , ghost - > player - > followmobj - > sprite2 ) ;
2020-03-19 20:09:55 +00:00
WRITEUINT16 ( demo_p , ghost - > player - > followmobj - > sprite ) ;
WRITEUINT8 ( demo_p , ( ghost - > player - > followmobj - > frame & FF_FRAMEMASK ) ) ;
2020-05-24 00:29:07 +00:00
WRITEUINT16 ( demo_p , ghost - > player - > followmobj - > color ) ;
2020-03-19 20:09:55 +00:00
* followtic_p = followtic ;
}
else
oldghost . flags2 & = ~ MF2_AMBUSH ;
* ziptic_p = ziptic ;
// attention here for the ticcmd size!
// latest demos with mouse aiming byte in ticcmd
if ( demo_p > = demoend - ( 13 + 9 + 9 ) )
{
G_CheckDemoStatus ( ) ; // no more space
return ;
}
}
// Uses ghost data to do consistency checks on your position.
// This fixes desynchronising demos when fighting eggman.
void G_ConsGhostTic ( void )
{
UINT8 ziptic ;
UINT16 px , py , pz , gx , gy , gz ;
mobj_t * testmo ;
if ( ! demo_p | | ! demo_start )
return ;
if ( ! ( demoflags & DF_GHOST ) )
return ; // No ghost data to use.
testmo = players [ 0 ] . mo ;
2024-06-03 14:47:27 +00:00
if ( P_MobjWasRemoved ( testmo ) )
return ; // No valid mobj exists, probably because of unexpected quit
2020-03-19 20:09:55 +00:00
// Grab ghost data.
ziptic = READUINT8 ( demo_p ) ;
if ( ziptic & GZT_XYZ )
{
oldghost . x = READFIXED ( demo_p ) ;
oldghost . y = READFIXED ( demo_p ) ;
oldghost . z = READFIXED ( demo_p ) ;
}
else
{
if ( ziptic & GZT_MOMXY )
{
2020-09-23 18:26:51 +00:00
oldghost . momx = ( demoversion < 0x000e ) ? READINT16 ( demo_p ) < < 8 : READFIXED ( demo_p ) ;
oldghost . momy = ( demoversion < 0x000e ) ? READINT16 ( demo_p ) < < 8 : READFIXED ( demo_p ) ;
2020-03-19 20:09:55 +00:00
}
if ( ziptic & GZT_MOMZ )
2020-09-23 18:26:51 +00:00
oldghost . momz = ( demoversion < 0x000e ) ? READINT16 ( demo_p ) < < 8 : READFIXED ( demo_p ) ;
2020-03-19 20:09:55 +00:00
oldghost . x + = oldghost . momx ;
oldghost . y + = oldghost . momy ;
oldghost . z + = oldghost . momz ;
}
if ( ziptic & GZT_ANGLE )
demo_p + + ;
if ( ziptic & GZT_FRAME )
demo_p + + ;
if ( ziptic & GZT_SPR2 )
2023-11-13 16:09:00 +00:00
demo_p + = ( demoversion < 0x0011 ) ? sizeof ( UINT8 ) : sizeof ( UINT16 ) ;
2020-03-19 20:09:55 +00:00
if ( ziptic & GZT_EXTRA )
{ // But wait, there's more!
UINT8 xziptic = READUINT8 ( demo_p ) ;
if ( xziptic & EZT_COLOR )
2020-05-24 00:29:07 +00:00
demo_p + = ( demoversion = = 0x000c ) ? 1 : sizeof ( UINT16 ) ;
2020-03-19 20:09:55 +00:00
if ( xziptic & EZT_SCALE )
demo_p + = sizeof ( fixed_t ) ;
if ( xziptic & EZT_HIT )
{ // Resync mob damage.
UINT16 i , count = READUINT16 ( demo_p ) ;
thinker_t * th ;
mobj_t * mobj ;
UINT32 type ;
UINT16 health ;
fixed_t x ;
fixed_t y ;
fixed_t z ;
for ( i = 0 ; i < count ; i + + )
{
//demo_p += 4; // reserved.
type = READUINT32 ( demo_p ) ;
health = READUINT16 ( demo_p ) ;
x = READFIXED ( demo_p ) ;
y = READFIXED ( demo_p ) ;
z = READFIXED ( demo_p ) ;
demo_p + = sizeof ( angle_t ) ; // angle, unnecessary for cons.
mobj = NULL ;
for ( th = thlist [ THINK_MOBJ ] . next ; th ! = & thlist [ THINK_MOBJ ] ; th = th - > next )
{
if ( th - > function . acp1 = = ( actionf_p1 ) P_RemoveThinkerDelayed )
continue ;
mobj = ( mobj_t * ) th ;
if ( mobj - > type = = ( mobjtype_t ) type & & mobj - > x = = x & & mobj - > y = = y & & mobj - > z = = z )
break ;
}
if ( th ! = & thlist [ THINK_MOBJ ] & & mobj - > health ! = health ) // Wasn't damaged?! This is desync! Fix it!
{
if ( demosynced )
CONS_Alert ( CONS_WARNING , M_GetText ( " Demo playback has desynced! \n " ) ) ;
demosynced = false ;
P_DamageMobj ( mobj , players [ 0 ] . mo , players [ 0 ] . mo , 1 , 0 ) ;
}
}
}
if ( xziptic & EZT_SPRITE )
demo_p + = sizeof ( UINT16 ) ;
if ( xziptic & EZT_HEIGHT )
2020-11-09 15:01:20 +00:00
demo_p + = ( demoversion < 0x000e ) ? sizeof ( INT16 ) : sizeof ( fixed_t ) ;
2020-03-19 20:09:55 +00:00
}
if ( ziptic & GZT_FOLLOW )
{ // Even more...
UINT8 followtic = READUINT8 ( demo_p ) ;
if ( followtic & FZT_SPAWNED )
{
demo_p + = sizeof ( INT16 ) ;
if ( followtic & FZT_SKIN )
demo_p + + ;
}
if ( followtic & FZT_SCALE )
demo_p + = sizeof ( fixed_t ) ;
2020-09-23 18:26:51 +00:00
// momx, momy and momz
2020-10-20 00:20:08 +00:00
demo_p + = ( demoversion < 0x000e ) ? sizeof ( INT16 ) * 3 : sizeof ( fixed_t ) * 3 ;
2020-03-19 20:09:55 +00:00
if ( followtic & FZT_SKIN )
2023-11-13 16:09:00 +00:00
demo_p + = ( demoversion < 0x0011 ) ? sizeof ( UINT8 ) : sizeof ( UINT16 ) ;
2020-03-19 20:09:55 +00:00
demo_p + = sizeof ( UINT16 ) ;
demo_p + + ;
2020-05-24 00:29:07 +00:00
demo_p + = ( demoversion = = 0x000c ) ? 1 : sizeof ( UINT16 ) ;
2020-03-19 20:09:55 +00:00
}
// Re-synchronise
px = testmo - > x > > FRACBITS ;
py = testmo - > y > > FRACBITS ;
pz = testmo - > z > > FRACBITS ;
gx = oldghost . x > > FRACBITS ;
gy = oldghost . y > > FRACBITS ;
gz = oldghost . z > > FRACBITS ;
if ( px ! = gx | | py ! = gy | | pz ! = gz )
{
if ( demosynced )
CONS_Alert ( CONS_WARNING , M_GetText ( " Demo playback has desynced! \n " ) ) ;
demosynced = false ;
2024-02-21 21:22:00 +00:00
if ( cv_resyncdemo . value )
2024-02-21 19:38:11 +00:00
{
P_UnsetThingPosition ( testmo ) ;
testmo - > x = oldghost . x ;
testmo - > y = oldghost . y ;
P_SetThingPosition ( testmo ) ;
testmo - > z = oldghost . z ;
}
2020-03-19 20:09:55 +00:00
}
if ( * demo_p = = DEMOMARKER )
{
// end of demo data stream
G_CheckDemoStatus ( ) ;
return ;
}
}
void G_GhostTicker ( void )
{
demoghost * g , * p ;
for ( g = ghosts , p = NULL ; g ; g = g - > next )
{
// Skip normal demo data.
UINT8 ziptic = READUINT8 ( g - > p ) ;
UINT8 xziptic = 0 ;
if ( ziptic & ZT_FWD )
g - > p + + ;
if ( ziptic & ZT_SIDE )
g - > p + + ;
if ( ziptic & ZT_ANGLE )
g - > p + = 2 ;
if ( ziptic & ZT_BUTTONS )
g - > p + = 2 ;
if ( ziptic & ZT_AIMING )
g - > p + = 2 ;
2021-04-21 23:58:14 +00:00
if ( ziptic & ZT_LATENCY )
g - > p + + ;
2020-03-19 20:09:55 +00:00
// Grab ghost data.
ziptic = READUINT8 ( g - > p ) ;
if ( ziptic & GZT_XYZ )
{
g - > oldmo . x = READFIXED ( g - > p ) ;
g - > oldmo . y = READFIXED ( g - > p ) ;
g - > oldmo . z = READFIXED ( g - > p ) ;
}
else
{
if ( ziptic & GZT_MOMXY )
{
2020-09-23 18:26:51 +00:00
g - > oldmo . momx = ( g - > version < 0x000e ) ? READINT16 ( g - > p ) < < 8 : READFIXED ( g - > p ) ;
g - > oldmo . momy = ( g - > version < 0x000e ) ? READINT16 ( g - > p ) < < 8 : READFIXED ( g - > p ) ;
2020-03-19 20:09:55 +00:00
}
if ( ziptic & GZT_MOMZ )
2020-09-23 18:26:51 +00:00
g - > oldmo . momz = ( g - > version < 0x000e ) ? READINT16 ( g - > p ) < < 8 : READFIXED ( g - > p ) ;
2020-03-19 20:09:55 +00:00
g - > oldmo . x + = g - > oldmo . momx ;
g - > oldmo . y + = g - > oldmo . momy ;
g - > oldmo . z + = g - > oldmo . momz ;
}
if ( ziptic & GZT_ANGLE )
g - > mo - > angle = READUINT8 ( g - > p ) < < 24 ;
if ( ziptic & GZT_FRAME )
g - > oldmo . frame = READUINT8 ( g - > p ) ;
if ( ziptic & GZT_SPR2 )
2023-11-13 14:46:55 +00:00
g - > oldmo . sprite2 = ( g - > version < 0x0011 ) ? READUINT8 ( g - > p ) : READUINT16 ( g - > p ) ;
2020-03-19 20:09:55 +00:00
// Update ghost
P_UnsetThingPosition ( g - > mo ) ;
g - > mo - > x = g - > oldmo . x ;
g - > mo - > y = g - > oldmo . y ;
g - > mo - > z = g - > oldmo . z ;
P_SetThingPosition ( g - > mo ) ;
g - > mo - > frame = g - > oldmo . frame | tr_trans30 < < FF_TRANSSHIFT ;
if ( g - > fadein )
{
g - > mo - > frame + = ( ( ( - - g - > fadein ) / 6 ) < < FF_TRANSSHIFT ) ; // this calc never exceeds 9 unless g->fadein is bad, and it's only set once, so...
g - > mo - > flags2 & = ~ MF2_DONTDRAW ;
}
g - > mo - > sprite2 = g - > oldmo . sprite2 ;
if ( ziptic & GZT_EXTRA )
{ // But wait, there's more!
xziptic = READUINT8 ( g - > p ) ;
if ( xziptic & EZT_COLOR )
{
2020-05-24 00:29:07 +00:00
g - > color = ( g - > version = = 0x000c ) ? READUINT8 ( g - > p ) : READUINT16 ( g - > p ) ;
2020-03-19 20:09:55 +00:00
switch ( g - > color )
{
default :
case GHC_RETURNSKIN :
g - > mo - > skin = g - > oldmo . skin ;
/* FALLTHRU */
case GHC_NORMAL : // Go back to skin color
g - > mo - > color = g - > oldmo . color ;
break ;
// Handled below
case GHC_SUPER :
case GHC_INVINCIBLE :
break ;
case GHC_FIREFLOWER : // Fireflower
g - > mo - > color = SKINCOLOR_WHITE ;
break ;
case GHC_NIGHTSSKIN : // not actually a colour
2021-08-09 18:57:07 +00:00
g - > mo - > skin = skins [ DEFAULTNIGHTSSKIN ] ;
2020-03-19 20:09:55 +00:00
break ;
}
}
if ( xziptic & EZT_FLIP )
g - > mo - > eflags ^ = MFE_VERTICALFLIP ;
if ( xziptic & EZT_SCALE )
{
g - > mo - > destscale = READFIXED ( g - > p ) ;
if ( g - > mo - > destscale ! = g - > mo - > scale )
2024-01-02 16:51:43 +00:00
P_SetScale ( g - > mo , g - > mo - > destscale , false ) ;
2020-03-19 20:09:55 +00:00
}
if ( xziptic & EZT_THOKMASK )
{ // Let's only spawn ONE of these per frame, thanks.
mobj_t * mobj ;
2020-05-28 15:39:31 +00:00
UINT32 type = MT_NULL ;
2020-03-19 20:09:55 +00:00
if ( g - > mo - > skin )
{
skin_t * skin = ( skin_t * ) g - > mo - > skin ;
switch ( xziptic & EZT_THOKMASK )
{
case EZT_THOK :
type = skin - > thokitem < 0 ? ( UINT32 ) mobjinfo [ MT_PLAYER ] . painchance : ( UINT32 ) skin - > thokitem ;
break ;
case EZT_SPIN :
type = skin - > spinitem < 0 ? ( UINT32 ) mobjinfo [ MT_PLAYER ] . damage : ( UINT32 ) skin - > spinitem ;
break ;
case EZT_REV :
type = skin - > revitem < 0 ? ( UINT32 ) mobjinfo [ MT_PLAYER ] . raisestate : ( UINT32 ) skin - > revitem ;
break ;
}
}
if ( type ! = MT_NULL )
{
if ( type = = MT_GHOST )
{
mobj = P_SpawnGhostMobj ( g - > mo ) ; // does a large portion of the work for us
2023-04-30 10:43:31 +00:00
if ( ! P_MobjWasRemoved ( mobj ) )
mobj - > frame = ( mobj - > frame & ~ FF_FRAMEMASK ) | tr_trans60 < < FF_TRANSSHIFT ; // P_SpawnGhostMobj sets trans50, we want trans60
2020-03-19 20:09:55 +00:00
}
else
{
mobj = P_SpawnMobjFromMobj ( g - > mo , 0 , 0 , - FixedDiv ( FixedMul ( g - > mo - > info - > height , g - > mo - > scale ) - g - > mo - > height , 3 * FRACUNIT ) , MT_THOK ) ;
2023-04-30 10:43:31 +00:00
if ( ! P_MobjWasRemoved ( mobj ) )
2020-03-19 20:09:55 +00:00
{
2023-04-30 10:43:31 +00:00
mobj - > sprite = states [ mobjinfo [ type ] . spawnstate ] . sprite ;
mobj - > frame = ( states [ mobjinfo [ type ] . spawnstate ] . frame & FF_FRAMEMASK ) | tr_trans60 < < FF_TRANSSHIFT ;
mobj - > color = g - > mo - > color ;
mobj - > skin = g - > mo - > skin ;
2024-01-02 16:51:43 +00:00
P_SetScale ( mobj , g - > mo - > scale , true ) ;
2023-04-30 10:43:31 +00:00
if ( type = = MT_THOK ) // spintrail-specific modification for MT_THOK
{
mobj - > frame = FF_TRANS80 ;
mobj - > fuse = mobj - > tics ;
}
mobj - > tics = - 1 ; // nope.
2020-03-19 20:09:55 +00:00
}
}
2023-04-30 10:43:31 +00:00
if ( ! P_MobjWasRemoved ( mobj ) )
{
mobj - > floorz = mobj - > z ;
mobj - > ceilingz = mobj - > z + mobj - > height ;
P_UnsetThingPosition ( mobj ) ;
mobj - > flags = MF_NOBLOCKMAP | MF_NOCLIP | MF_NOCLIPHEIGHT | MF_NOGRAVITY ; // make an ATTEMPT to curb crazy SOCs fucking stuff up...
P_SetThingPosition ( mobj ) ;
if ( ! mobj - > fuse )
mobj - > fuse = 8 ;
P_SetTarget ( & mobj - > target , g - > mo ) ;
}
2020-03-19 20:09:55 +00:00
}
}
if ( xziptic & EZT_HIT )
{ // Spawn hit poofs for killing things!
UINT16 i , count = READUINT16 ( g - > p ) , health ;
UINT32 type ;
fixed_t x , y , z ;
angle_t angle ;
mobj_t * poof ;
for ( i = 0 ; i < count ; i + + )
{
//g->p += 4; // reserved
type = READUINT32 ( g - > p ) ;
health = READUINT16 ( g - > p ) ;
x = READFIXED ( g - > p ) ;
y = READFIXED ( g - > p ) ;
z = READFIXED ( g - > p ) ;
angle = READANGLE ( g - > p ) ;
if ( ! ( mobjinfo [ type ] . flags & MF_SHOOTABLE )
| | ! ( mobjinfo [ type ] . flags & ( MF_ENEMY | MF_MONITOR ) )
| | health ! = 0 | | i > = 4 ) // only spawn for the first 4 hits per frame, to prevent ghosts from splode-spamming too bad.
continue ;
poof = P_SpawnMobj ( x , y , z , MT_GHOST ) ;
2023-04-30 10:43:31 +00:00
if ( P_MobjWasRemoved ( poof ) )
continue ;
2020-03-19 20:09:55 +00:00
poof - > angle = angle ;
poof - > flags = MF_NOBLOCKMAP | MF_NOCLIP | MF_NOCLIPHEIGHT | MF_NOGRAVITY ; // make an ATTEMPT to curb crazy SOCs fucking stuff up...
poof - > health = 0 ;
P_SetMobjStateNF ( poof , S_XPLD1 ) ;
}
}
if ( xziptic & EZT_SPRITE )
g - > mo - > sprite = READUINT16 ( g - > p ) ;
if ( xziptic & EZT_HEIGHT )
{
2020-11-09 15:01:20 +00:00
fixed_t temp = ( g - > version < 0x000e ) ? READINT16 ( g - > p ) < < FRACBITS : READFIXED ( g - > p ) ;
2020-03-19 20:09:55 +00:00
g - > mo - > height = FixedMul ( temp , g - > mo - > scale ) ;
}
}
// Tick ghost colors (Super and Mario Invincibility flashing)
switch ( g - > color )
{
case GHC_SUPER : // Super (P_DoSuperStuff)
if ( g - > mo - > skin )
{
skin_t * skin = ( skin_t * ) g - > mo - > skin ;
g - > mo - > color = skin - > supercolor ;
}
else
g - > mo - > color = SKINCOLOR_SUPERGOLD1 ;
g - > mo - > color + = abs ( ( ( signed ) ( ( unsigned ) leveltime > > 1 ) % 9 ) - 4 ) ;
break ;
case GHC_INVINCIBLE : // Mario invincibility (P_CheckInvincibilityTimer)
2020-05-24 00:29:07 +00:00
g - > mo - > color = ( UINT16 ) ( SKINCOLOR_RUBY + ( leveltime % ( FIRSTSUPERCOLOR - SKINCOLOR_RUBY ) ) ) ; // Passes through all saturated colours
2020-03-19 20:09:55 +00:00
break ;
default :
break ;
}
# define follow g->mo->tracer
if ( ziptic & GZT_FOLLOW )
{ // Even more...
UINT8 followtic = READUINT8 ( g - > p ) ;
fixed_t temp ;
if ( followtic & FZT_SPAWNED )
{
if ( follow )
P_RemoveMobj ( follow ) ;
P_SetTarget ( & follow , P_SpawnMobjFromMobj ( g - > mo , 0 , 0 , 0 , MT_GHOST ) ) ;
2023-04-30 10:43:31 +00:00
if ( ! P_MobjWasRemoved ( follow ) )
{
P_SetTarget ( & follow - > tracer , g - > mo ) ;
follow - > tics = - 1 ;
temp = READINT16 ( g - > p ) < < FRACBITS ;
follow - > height = FixedMul ( follow - > scale , temp ) ;
2020-03-19 20:09:55 +00:00
2023-04-30 10:43:31 +00:00
if ( followtic & FZT_LINKDRAW )
follow - > flags2 | = MF2_LINKDRAW ;
2020-03-19 20:09:55 +00:00
2023-04-30 10:43:31 +00:00
if ( followtic & FZT_COLORIZED )
follow - > colorized = true ;
2020-03-19 20:09:55 +00:00
2023-04-30 10:43:31 +00:00
if ( followtic & FZT_SKIN )
2024-01-02 23:58:55 +00:00
follow - > skin = skins [ READUINT8 ( g - > p ) ] ;
2023-04-30 10:43:31 +00:00
}
2020-03-19 20:09:55 +00:00
}
if ( follow )
{
if ( followtic & FZT_SCALE )
follow - > destscale = READFIXED ( g - > p ) ;
else
follow - > destscale = g - > mo - > destscale ;
if ( follow - > destscale ! = follow - > scale )
2024-01-02 16:51:43 +00:00
P_SetScale ( follow , follow - > destscale , false ) ;
2020-03-19 20:09:55 +00:00
P_UnsetThingPosition ( follow ) ;
2020-09-23 18:26:51 +00:00
temp = ( g - > version < 0x000e ) ? READINT16 ( g - > p ) < < 8 : READFIXED ( g - > p ) ;
2020-03-19 20:09:55 +00:00
follow - > x = g - > mo - > x + temp ;
2020-09-23 18:26:51 +00:00
temp = ( g - > version < 0x000e ) ? READINT16 ( g - > p ) < < 8 : READFIXED ( g - > p ) ;
2020-03-19 20:09:55 +00:00
follow - > y = g - > mo - > y + temp ;
2020-09-23 18:26:51 +00:00
temp = ( g - > version < 0x000e ) ? READINT16 ( g - > p ) < < 8 : READFIXED ( g - > p ) ;
2020-03-19 20:09:55 +00:00
follow - > z = g - > mo - > z + temp ;
P_SetThingPosition ( follow ) ;
if ( followtic & FZT_SKIN )
2023-11-13 14:46:55 +00:00
follow - > sprite2 = ( g - > version < 0x0011 ) ? READUINT8 ( g - > p ) : READUINT16 ( g - > p ) ;
2020-03-19 20:09:55 +00:00
else
follow - > sprite2 = 0 ;
follow - > sprite = READUINT16 ( g - > p ) ;
follow - > frame = ( READUINT8 ( g - > p ) ) | ( g - > mo - > frame & FF_TRANSMASK ) ;
follow - > angle = g - > mo - > angle ;
2020-05-24 00:29:07 +00:00
follow - > color = ( g - > version = = 0x000c ) ? READUINT8 ( g - > p ) : READUINT16 ( g - > p ) ;
2020-03-19 20:09:55 +00:00
if ( ! ( followtic & FZT_SPAWNED ) )
{
if ( xziptic & EZT_FLIP )
{
follow - > flags2 ^ = MF2_OBJECTFLIP ;
follow - > eflags ^ = MFE_VERTICALFLIP ;
}
}
}
}
else if ( follow )
{
P_RemoveMobj ( follow ) ;
P_SetTarget ( & follow , NULL ) ;
}
// Demo ends after ghost data.
if ( * g - > p = = DEMOMARKER )
{
g - > mo - > momx = g - > mo - > momy = g - > mo - > momz = 0 ;
# if 1 // freeze frame (maybe more useful for time attackers)
g - > mo - > colorized = true ;
if ( follow )
follow - > colorized = true ;
# else // dissapearing act
g - > mo - > fuse = TICRATE ;
if ( follow )
follow - > fuse = TICRATE ;
# endif
if ( p )
p - > next = g - > next ;
else
ghosts = g - > next ;
Z_Free ( g ) ;
continue ;
}
p = g ;
# undef follow
}
}
void G_ReadMetalTic ( mobj_t * metal )
{
UINT8 ziptic ;
UINT8 xziptic = 0 ;
if ( ! metal_p )
return ;
if ( ! metal - > health )
{
G_StopMetalDemo ( ) ;
return ;
}
switch ( * metal_p )
{
case METALSNICE :
break ;
case METALDEATH :
if ( metal - > tracer )
P_RemoveMobj ( metal - > tracer ) ;
P_KillMobj ( metal , NULL , NULL , 0 ) ;
/* FALLTHRU */
case DEMOMARKER :
default :
// end of demo data stream
G_StopMetalDemo ( ) ;
return ;
}
metal_p + + ;
ziptic = READUINT8 ( metal_p ) ;
// Read changes from the tic
if ( ziptic & GZT_XYZ )
{
2020-05-28 15:49:12 +00:00
// make sure the values are read in the right order
oldmetal . x = READFIXED ( metal_p ) ;
oldmetal . y = READFIXED ( metal_p ) ;
oldmetal . z = READFIXED ( metal_p ) ;
2021-11-29 13:20:27 +00:00
P_MoveOrigin ( metal , oldmetal . x , oldmetal . y , oldmetal . z ) ;
2020-03-19 20:09:55 +00:00
oldmetal . x = metal - > x ;
oldmetal . y = metal - > y ;
oldmetal . z = metal - > z ;
}
else
{
if ( ziptic & GZT_MOMXY )
{
2020-09-23 18:26:51 +00:00
oldmetal . momx = ( metalversion < 0x000e ) ? READINT16 ( metal_p ) < < 8 : READFIXED ( metal_p ) ;
oldmetal . momy = ( metalversion < 0x000e ) ? READINT16 ( metal_p ) < < 8 : READFIXED ( metal_p ) ;
2020-03-19 20:09:55 +00:00
}
if ( ziptic & GZT_MOMZ )
2020-09-23 18:26:51 +00:00
oldmetal . momz = ( metalversion < 0x000e ) ? READINT16 ( metal_p ) < < 8 : READFIXED ( metal_p ) ;
2020-03-19 20:09:55 +00:00
oldmetal . x + = oldmetal . momx ;
oldmetal . y + = oldmetal . momy ;
oldmetal . z + = oldmetal . momz ;
}
if ( ziptic & GZT_ANGLE )
metal - > angle = READUINT8 ( metal_p ) < < 24 ;
if ( ziptic & GZT_FRAME )
2022-02-14 13:57:00 +00:00
{
2020-03-19 20:09:55 +00:00
oldmetal . frame = READUINT32 ( metal_p ) ;
2022-02-14 13:57:00 +00:00
if ( metalversion < 0x000f )
oldmetal . frame = G_ConvertOldFrameFlags ( oldmetal . frame ) ;
}
2020-03-19 20:09:55 +00:00
if ( ziptic & GZT_SPR2 )
2023-11-13 14:52:12 +00:00
oldmetal . sprite2 = ( metalversion < 0x0011 ) ? READUINT8 ( metal_p ) : READUINT16 ( metal_p ) ;
2020-03-19 20:09:55 +00:00
// Set movement, position, and angle
// oldmetal contains where you're supposed to be.
metal - > momx = oldmetal . momx ;
metal - > momy = oldmetal . momy ;
metal - > momz = oldmetal . momz ;
P_UnsetThingPosition ( metal ) ;
metal - > x = oldmetal . x ;
metal - > y = oldmetal . y ;
metal - > z = oldmetal . z ;
P_SetThingPosition ( metal ) ;
metal - > frame = oldmetal . frame ;
metal - > sprite2 = oldmetal . sprite2 ;
if ( ziptic & GZT_EXTRA )
{ // But wait, there's more!
xziptic = READUINT8 ( metal_p ) ;
if ( xziptic & EZT_FLIP )
{
metal - > eflags ^ = MFE_VERTICALFLIP ;
metal - > flags2 ^ = MF2_OBJECTFLIP ;
}
if ( xziptic & EZT_SCALE )
{
metal - > destscale = READFIXED ( metal_p ) ;
if ( metal - > destscale ! = metal - > scale )
2024-01-02 16:51:43 +00:00
P_SetScale ( metal , metal - > destscale , false ) ;
2020-03-19 20:09:55 +00:00
}
if ( xziptic & EZT_THOKMASK )
{ // Let's only spawn ONE of these per frame, thanks.
mobj_t * mobj ;
2020-05-28 15:39:31 +00:00
UINT32 type = MT_NULL ;
2020-03-19 20:09:55 +00:00
if ( metal - > skin )
{
skin_t * skin = ( skin_t * ) metal - > skin ;
switch ( xziptic & EZT_THOKMASK )
{
case EZT_THOK :
type = skin - > thokitem < 0 ? ( UINT32 ) mobjinfo [ MT_PLAYER ] . painchance : ( UINT32 ) skin - > thokitem ;
break ;
case EZT_SPIN :
type = skin - > spinitem < 0 ? ( UINT32 ) mobjinfo [ MT_PLAYER ] . damage : ( UINT32 ) skin - > spinitem ;
break ;
case EZT_REV :
type = skin - > revitem < 0 ? ( UINT32 ) mobjinfo [ MT_PLAYER ] . raisestate : ( UINT32 ) skin - > revitem ;
break ;
}
}
if ( type ! = MT_NULL )
{
if ( type = = MT_GHOST )
{
mobj = P_SpawnGhostMobj ( metal ) ; // does a large portion of the work for us
}
else
{
mobj = P_SpawnMobjFromMobj ( metal , 0 , 0 , - FixedDiv ( FixedMul ( metal - > info - > height , metal - > scale ) - metal - > height , 3 * FRACUNIT ) , MT_THOK ) ;
2023-04-30 10:43:31 +00:00
if ( ! P_MobjWasRemoved ( mobj ) )
2020-03-19 20:09:55 +00:00
{
2023-04-30 10:43:31 +00:00
mobj - > sprite = states [ mobjinfo [ type ] . spawnstate ] . sprite ;
mobj - > frame = states [ mobjinfo [ type ] . spawnstate ] . frame ;
mobj - > angle = metal - > angle ;
mobj - > color = metal - > color ;
mobj - > skin = metal - > skin ;
2024-01-02 16:51:43 +00:00
P_SetScale ( mobj , metal - > scale , true ) ;
2023-04-30 10:43:31 +00:00
if ( type = = MT_THOK ) // spintrail-specific modification for MT_THOK
{
mobj - > frame = FF_TRANS70 ;
mobj - > fuse = mobj - > tics ;
}
mobj - > tics = - 1 ; // nope.
2020-03-19 20:09:55 +00:00
}
}
2023-04-30 10:43:31 +00:00
if ( ! P_MobjWasRemoved ( mobj ) )
{
mobj - > floorz = mobj - > z ;
mobj - > ceilingz = mobj - > z + mobj - > height ;
P_UnsetThingPosition ( mobj ) ;
mobj - > flags = MF_NOBLOCKMAP | MF_NOCLIP | MF_NOCLIPHEIGHT | MF_NOGRAVITY ; // make an ATTEMPT to curb crazy SOCs fucking stuff up...
P_SetThingPosition ( mobj ) ;
if ( ! mobj - > fuse )
mobj - > fuse = 8 ;
P_SetTarget ( & mobj - > target , metal ) ;
}
2020-03-19 20:09:55 +00:00
}
}
if ( xziptic & EZT_SPRITE )
metal - > sprite = READUINT16 ( metal_p ) ;
if ( xziptic & EZT_HEIGHT )
{
2020-11-09 15:01:20 +00:00
fixed_t temp = ( metalversion < 0x000e ) ? READINT16 ( metal_p ) < < FRACBITS : READFIXED ( metal_p ) ;
2020-03-19 20:09:55 +00:00
metal - > height = FixedMul ( temp , metal - > scale ) ;
}
}
# define follow metal->tracer
if ( ziptic & GZT_FOLLOW )
{ // Even more...
UINT8 followtic = READUINT8 ( metal_p ) ;
fixed_t temp ;
if ( followtic & FZT_SPAWNED )
{
if ( follow )
P_RemoveMobj ( follow ) ;
P_SetTarget ( & follow , P_SpawnMobjFromMobj ( metal , 0 , 0 , 0 , MT_GHOST ) ) ;
2023-04-30 10:43:31 +00:00
if ( ! P_MobjWasRemoved ( follow ) )
{
P_SetTarget ( & follow - > tracer , metal ) ;
follow - > tics = - 1 ;
temp = READINT16 ( metal_p ) < < FRACBITS ;
follow - > height = FixedMul ( follow - > scale , temp ) ;
2020-03-19 20:09:55 +00:00
2023-04-30 10:43:31 +00:00
if ( followtic & FZT_LINKDRAW )
follow - > flags2 | = MF2_LINKDRAW ;
2020-03-19 20:09:55 +00:00
2023-04-30 10:43:31 +00:00
if ( followtic & FZT_COLORIZED )
follow - > colorized = true ;
2020-03-19 20:09:55 +00:00
2023-04-30 10:43:31 +00:00
if ( followtic & FZT_SKIN )
2024-01-02 23:58:55 +00:00
follow - > skin = skins [ READUINT8 ( metal_p ) ] ;
2023-04-30 10:43:31 +00:00
}
2020-03-19 20:09:55 +00:00
}
if ( follow )
{
if ( followtic & FZT_SCALE )
follow - > destscale = READFIXED ( metal_p ) ;
else
follow - > destscale = metal - > destscale ;
if ( follow - > destscale ! = follow - > scale )
2024-01-02 16:51:43 +00:00
P_SetScale ( follow , follow - > destscale , false ) ;
2020-03-19 20:09:55 +00:00
P_UnsetThingPosition ( follow ) ;
2020-09-23 18:26:51 +00:00
temp = ( metalversion < 0x000e ) ? READINT16 ( metal_p ) < < 8 : READFIXED ( metal_p ) ;
2020-03-19 20:09:55 +00:00
follow - > x = metal - > x + temp ;
2020-09-23 18:26:51 +00:00
temp = ( metalversion < 0x000e ) ? READINT16 ( metal_p ) < < 8 : READFIXED ( metal_p ) ;
2020-03-19 20:09:55 +00:00
follow - > y = metal - > y + temp ;
2020-09-23 18:26:51 +00:00
temp = ( metalversion < 0x000e ) ? READINT16 ( metal_p ) < < 8 : READFIXED ( metal_p ) ;
2020-03-19 20:09:55 +00:00
follow - > z = metal - > z + temp ;
P_SetThingPosition ( follow ) ;
if ( followtic & FZT_SKIN )
2023-11-13 14:52:12 +00:00
follow - > sprite2 = ( metalversion < 0x0011 ) ? READUINT8 ( metal_p ) : READUINT16 ( metal_p ) ;
2020-03-19 20:09:55 +00:00
else
follow - > sprite2 = 0 ;
follow - > sprite = READUINT16 ( metal_p ) ;
follow - > frame = READUINT32 ( metal_p ) ; // NOT & FF_FRAMEMASK here, so 32 bits
2022-02-14 13:57:00 +00:00
if ( metalversion < 0x000f )
follow - > frame = G_ConvertOldFrameFlags ( follow - > frame ) ;
2020-03-19 20:09:55 +00:00
follow - > angle = metal - > angle ;
2023-11-13 14:46:55 +00:00
follow - > color = ( metalversion = = 0x000c ) ? READUINT8 ( metal_p ) : READUINT16 ( metal_p ) ;
2020-03-19 20:09:55 +00:00
if ( ! ( followtic & FZT_SPAWNED ) )
{
if ( xziptic & EZT_FLIP )
{
follow - > flags2 ^ = MF2_OBJECTFLIP ;
follow - > eflags ^ = MFE_VERTICALFLIP ;
}
}
}
}
else if ( follow )
{
P_RemoveMobj ( follow ) ;
P_SetTarget ( & follow , NULL ) ;
}
# undef follow
}
void G_WriteMetalTic ( mobj_t * metal )
{
UINT8 ziptic = 0 ;
UINT8 * ziptic_p ;
fixed_t height ;
if ( ! demo_p ) // demo_p will be NULL until the race start linedef executor is activated!
return ;
WRITEUINT8 ( demo_p , METALSNICE ) ;
ziptic_p = demo_p + + ; // the ziptic, written at the end of this function
# define MAXMOM (0xFFFF<<8)
// GZT_XYZ is only useful if you've moved 256 FRACUNITS or more in a single tic.
if ( abs ( metal - > x - oldmetal . x ) > MAXMOM
| | abs ( metal - > y - oldmetal . y ) > MAXMOM
| | abs ( metal - > z - oldmetal . z ) > MAXMOM )
{
oldmetal . x = metal - > x ;
oldmetal . y = metal - > y ;
oldmetal . z = metal - > z ;
ziptic | = GZT_XYZ ;
WRITEFIXED ( demo_p , oldmetal . x ) ;
WRITEFIXED ( demo_p , oldmetal . y ) ;
WRITEFIXED ( demo_p , oldmetal . z ) ;
}
else
{
// For moving normally:
2020-09-23 18:26:51 +00:00
// Store movement as a fixed value
fixed_t momx = metal - > x - oldmetal . x ;
fixed_t momy = metal - > y - oldmetal . y ;
2020-03-19 20:09:55 +00:00
if ( momx ! = oldmetal . momx
| | momy ! = oldmetal . momy )
{
oldmetal . momx = momx ;
oldmetal . momy = momy ;
ziptic | = GZT_MOMXY ;
2020-09-23 18:26:51 +00:00
WRITEFIXED ( demo_p , momx ) ;
WRITEFIXED ( demo_p , momy ) ;
2020-03-19 20:09:55 +00:00
}
2020-09-23 18:26:51 +00:00
momx = metal - > z - oldmetal . z ;
2020-03-19 20:09:55 +00:00
if ( momx ! = oldmetal . momz )
{
oldmetal . momz = momx ;
ziptic | = GZT_MOMZ ;
2020-09-23 18:26:51 +00:00
WRITEFIXED ( demo_p , momx ) ;
2020-03-19 20:09:55 +00:00
}
// This SHOULD set oldmetal.x/y/z to match metal->x/y/z
2020-09-23 18:26:51 +00:00
oldmetal . x + = oldmetal . momx ;
oldmetal . y + = oldmetal . momy ;
oldmetal . z + = oldmetal . momz ;
2020-03-19 20:09:55 +00:00
}
# undef MAXMOM
// Only store the 8 most relevant bits of angle
// because exact values aren't too easy to discern to begin with when only 8 angles have different sprites
// and it does not affect movement at all anyway.
if ( metal - > player & & metal - > player - > drawangle > > 24 ! = oldmetal . angle )
{
oldmetal . angle = metal - > player - > drawangle > > 24 ;
ziptic | = GZT_ANGLE ;
WRITEUINT8 ( demo_p , oldmetal . angle ) ;
}
// Store the sprite frame.
if ( ( metal - > frame & FF_FRAMEMASK ) ! = oldmetal . frame )
{
oldmetal . frame = metal - > frame ; // NOT & FF_FRAMEMASK here, so 32 bits
ziptic | = GZT_FRAME ;
WRITEUINT32 ( demo_p , oldmetal . frame ) ;
}
if ( metal - > sprite = = SPR_PLAY
& & metal - > sprite2 ! = oldmetal . sprite2 )
{
oldmetal . sprite2 = metal - > sprite2 ;
ziptic | = GZT_SPR2 ;
2023-11-13 14:46:55 +00:00
WRITEUINT16 ( demo_p , oldmetal . sprite2 ) ;
2020-03-19 20:09:55 +00:00
}
// Check for sprite set changes
if ( metal - > sprite ! = oldmetal . sprite )
{
oldmetal . sprite = metal - > sprite ;
ghostext . flags | = EZT_SPRITE ;
}
if ( ( height = FixedDiv ( metal - > height , metal - > scale ) ) ! = oldmetal . height )
{
oldmetal . height = height ;
ghostext . flags | = EZT_HEIGHT ;
}
if ( ghostext . flags & ~ ( EZT_COLOR | EZT_HIT ) ) // these two aren't handled by metal ever
{
ziptic | = GZT_EXTRA ;
if ( ghostext . scale = = ghostext . lastscale )
ghostext . flags & = ~ EZT_SCALE ;
WRITEUINT8 ( demo_p , ghostext . flags ) ;
if ( ghostext . flags & EZT_SCALE )
{
WRITEFIXED ( demo_p , ghostext . scale ) ;
ghostext . lastscale = ghostext . scale ;
}
if ( ghostext . flags & EZT_SPRITE )
WRITEUINT16 ( demo_p , oldmetal . sprite ) ;
if ( ghostext . flags & EZT_HEIGHT )
{
2020-11-09 15:01:20 +00:00
WRITEFIXED ( demo_p , height ) ;
2020-03-19 20:09:55 +00:00
}
ghostext . flags = 0 ;
}
if ( metal - > player & & metal - > player - > followmobj & & ! ( metal - > player - > followmobj - > sprite = = SPR_NULL | | ( metal - > player - > followmobj - > flags2 & MF2_DONTDRAW ) ) )
{
2020-09-23 18:26:51 +00:00
fixed_t temp ;
2020-03-19 20:09:55 +00:00
UINT8 * followtic_p = demo_p + + ;
UINT8 followtic = 0 ;
ziptic | = GZT_FOLLOW ;
if ( metal - > player - > followmobj - > skin )
followtic | = FZT_SKIN ;
if ( ! ( oldmetal . flags2 & MF2_AMBUSH ) )
{
followtic | = FZT_SPAWNED ;
WRITEINT16 ( demo_p , metal - > player - > followmobj - > info - > height > > FRACBITS ) ;
if ( metal - > player - > followmobj - > flags2 & MF2_LINKDRAW )
followtic | = FZT_LINKDRAW ;
if ( metal - > player - > followmobj - > colorized )
followtic | = FZT_COLORIZED ;
if ( followtic & FZT_SKIN )
2021-08-09 18:57:07 +00:00
WRITEUINT8 ( demo_p , ( UINT8 ) ( ( ( skin_t * ) metal - > player - > followmobj - > skin ) - > skinnum ) ) ;
2020-03-19 20:09:55 +00:00
oldmetal . flags2 | = MF2_AMBUSH ;
}
if ( metal - > player - > followmobj - > scale ! = metal - > scale )
{
followtic | = FZT_SCALE ;
WRITEFIXED ( demo_p , metal - > player - > followmobj - > scale ) ;
}
2020-09-23 18:26:51 +00:00
temp = metal - > player - > followmobj - > x - metal - > x ;
WRITEFIXED ( demo_p , temp ) ;
temp = metal - > player - > followmobj - > y - metal - > y ;
WRITEFIXED ( demo_p , temp ) ;
temp = metal - > player - > followmobj - > z - metal - > z ;
WRITEFIXED ( demo_p , temp ) ;
2020-03-19 20:09:55 +00:00
if ( followtic & FZT_SKIN )
2023-11-13 14:46:55 +00:00
WRITEUINT16 ( demo_p , metal - > player - > followmobj - > sprite2 ) ;
2020-03-19 20:09:55 +00:00
WRITEUINT16 ( demo_p , metal - > player - > followmobj - > sprite ) ;
WRITEUINT32 ( demo_p , metal - > player - > followmobj - > frame ) ; // NOT & FF_FRAMEMASK here, so 32 bits
2020-05-24 00:29:07 +00:00
WRITEUINT16 ( demo_p , metal - > player - > followmobj - > color ) ;
2020-03-19 20:09:55 +00:00
* followtic_p = followtic ;
}
else
oldmetal . flags2 & = ~ MF2_AMBUSH ;
* ziptic_p = ziptic ;
// attention here for the ticcmd size!
// latest demos with mouse aiming byte in ticcmd
if ( demo_p > = demoend - 32 )
{
G_StopMetalRecording ( false ) ; // no more space
return ;
}
}
//
// G_RecordDemo
//
void G_RecordDemo ( const char * name )
{
INT32 maxsize ;
strcpy ( demoname , name ) ;
strcat ( demoname , " .lmp " ) ;
maxsize = 1024 * 1024 ;
if ( M_CheckParm ( " -maxdemo " ) & & M_IsNextParm ( ) )
maxsize = atoi ( M_GetNextParm ( ) ) * 1024 ;
// if (demobuffer)
// free(demobuffer);
demo_p = NULL ;
demobuffer = malloc ( maxsize ) ;
demoend = demobuffer + maxsize ;
demorecording = true ;
}
void G_RecordMetal ( void )
{
INT32 maxsize ;
maxsize = 1024 * 1024 ;
if ( M_CheckParm ( " -maxdemo " ) & & M_IsNextParm ( ) )
maxsize = atoi ( M_GetNextParm ( ) ) * 1024 ;
demo_p = NULL ;
demobuffer = malloc ( maxsize ) ;
demoend = demobuffer + maxsize ;
metalrecording = true ;
}
void G_BeginRecording ( void )
{
UINT8 i ;
2020-05-24 00:29:07 +00:00
char name [ MAXCOLORNAME + 1 ] ;
2020-03-19 20:09:55 +00:00
player_t * player = & players [ consoleplayer ] ;
2023-07-25 12:40:36 +00:00
char * filename ;
2023-07-25 13:29:09 +00:00
UINT16 totalfiles ;
2023-07-25 12:40:36 +00:00
UINT8 * m ;
2024-08-25 10:24:03 +00:00
save_t savebuffer ;
2023-07-25 12:40:36 +00:00
2020-03-19 20:09:55 +00:00
if ( demo_p )
return ;
memset ( name , 0 , sizeof ( name ) ) ;
demo_p = demobuffer ;
demoflags = DF_GHOST | ( modeattacking < < DF_ATTACKSHIFT ) ;
// Setup header.
M_Memcpy ( demo_p , DEMOHEADER , 12 ) ; demo_p + = 12 ;
WRITEUINT8 ( demo_p , VERSION ) ;
WRITEUINT8 ( demo_p , SUBVERSION ) ;
WRITEUINT16 ( demo_p , DEMOVERSION ) ;
// demo checksum
demo_p + = 16 ;
// game data
M_Memcpy ( demo_p , " PLAY " , 4 ) ; demo_p + = 4 ;
WRITEINT16 ( demo_p , gamemap ) ;
M_Memcpy ( demo_p , mapmd5 , 16 ) ; demo_p + = 16 ;
WRITEUINT8 ( demo_p , demoflags ) ;
2023-07-25 12:40:36 +00:00
// file list
m = demo_p ; /* file count */
2023-07-25 13:34:55 +00:00
demo_p + = 2 ;
2023-07-25 12:40:36 +00:00
totalfiles = 0 ;
for ( i = mainwads ; + + i < numwadfiles ; )
{
if ( wadfiles [ i ] - > important )
{
nameonly ( ( filename = va ( " %s " , wadfiles [ i ] - > filename ) ) ) ;
WRITESTRINGL ( demo_p , filename , MAX_WADPATH ) ;
WRITEMEM ( demo_p , wadfiles [ i ] - > md5sum , 16 ) ;
totalfiles + + ;
}
}
2023-07-25 13:29:09 +00:00
WRITEUINT16 ( m , totalfiles ) ;
2023-07-25 12:40:36 +00:00
2020-03-19 20:09:55 +00:00
switch ( ( demoflags & DF_ATTACKMASK ) > > DF_ATTACKSHIFT )
{
2023-07-25 12:40:36 +00:00
case ATTACKING_NONE : // 0
break ;
case ATTACKING_RECORD : // 1
demotime_p = demo_p ;
WRITEUINT32 ( demo_p , UINT32_MAX ) ; // time
WRITEUINT32 ( demo_p , 0 ) ; // score
WRITEUINT16 ( demo_p , 0 ) ; // rings
break ;
case ATTACKING_NIGHTS : // 2
demotime_p = demo_p ;
WRITEUINT32 ( demo_p , UINT32_MAX ) ; // time
WRITEUINT32 ( demo_p , 0 ) ; // score
break ;
default : // 3
break ;
2020-03-19 20:09:55 +00:00
}
WRITEUINT32 ( demo_p , P_GetInitSeed ( ) ) ;
// Name
for ( i = 0 ; i < 16 & & cv_playername . string [ i ] ; i + + )
name [ i ] = cv_playername . string [ i ] ;
for ( ; i < 16 ; i + + )
name [ i ] = ' \0 ' ;
M_Memcpy ( demo_p , name , 16 ) ;
demo_p + = 16 ;
// Skin
2023-11-23 16:48:18 +00:00
const char * skinname = skins [ players [ 0 ] . skin ] - > name ;
2023-10-31 07:04:01 +00:00
for ( i = 0 ; i < 16 & & skinname [ i ] ; i + + )
name [ i ] = skinname [ i ] ;
2020-03-19 20:09:55 +00:00
for ( ; i < 16 ; i + + )
name [ i ] = ' \0 ' ;
M_Memcpy ( demo_p , name , 16 ) ;
demo_p + = 16 ;
// Color
2023-08-15 07:29:50 +00:00
UINT16 skincolor = players [ 0 ] . skincolor ;
if ( skincolor > = numskincolors )
skincolor = SKINCOLOR_NONE ;
const char * skincolor_name = skincolors [ skincolor ] . name ;
for ( i = 0 ; i < MAXCOLORNAME & & skincolor_name [ i ] ; i + + )
name [ i ] = skincolor_name [ i ] ;
2020-05-24 00:29:07 +00:00
for ( ; i < MAXCOLORNAME ; i + + )
2020-03-19 20:09:55 +00:00
name [ i ] = ' \0 ' ;
2020-05-24 00:29:07 +00:00
M_Memcpy ( demo_p , name , MAXCOLORNAME ) ;
demo_p + = MAXCOLORNAME ;
2020-03-19 20:09:55 +00:00
// Stats
WRITEUINT8 ( demo_p , player - > charability ) ;
WRITEUINT8 ( demo_p , player - > charability2 ) ;
2023-07-08 21:14:37 +00:00
WRITEFIXED ( demo_p , player - > actionspd ) ;
WRITEFIXED ( demo_p , player - > mindash ) ;
WRITEFIXED ( demo_p , player - > maxdash ) ;
WRITEFIXED ( demo_p , player - > normalspeed ) ;
WRITEFIXED ( demo_p , player - > runspeed ) ;
2020-03-19 20:09:55 +00:00
WRITEUINT8 ( demo_p , player - > thrustfactor ) ;
WRITEUINT8 ( demo_p , player - > accelstart ) ;
WRITEUINT8 ( demo_p , player - > acceleration ) ;
2020-11-09 15:01:20 +00:00
WRITEFIXED ( demo_p , player - > height ) ;
WRITEFIXED ( demo_p , player - > spinheight ) ;
2023-07-08 20:17:53 +00:00
WRITEFIXED ( demo_p , player - > camerascale ) ;
WRITEFIXED ( demo_p , player - > shieldscale ) ;
2020-03-19 20:09:55 +00:00
// Trying to convert it back to % causes demo desync due to precision loss.
// Don't do it.
WRITEFIXED ( demo_p , player - > jumpfactor ) ;
// And mobjtype_t is best with UINT32 too...
WRITEUINT32 ( demo_p , player - > followitem ) ;
// Save pflag data - see SendWeaponPref()
{
UINT8 buf = 0 ;
pflags_t pflags = 0 ;
if ( cv_flipcam . value )
{
buf | = 0x01 ;
pflags | = PF_FLIPCAM ;
}
if ( cv_analog [ 0 ] . value )
{
buf | = 0x02 ;
pflags | = PF_ANALOGMODE ;
}
if ( cv_directionchar [ 0 ] . value )
{
buf | = 0x04 ;
pflags | = PF_DIRECTIONCHAR ;
}
if ( cv_autobrake . value )
{
buf | = 0x08 ;
pflags | = PF_AUTOBRAKE ;
}
if ( cv_usejoystick . value )
buf | = 0x10 ;
CV_SetValue ( & cv_showinputjoy , ! ! ( cv_usejoystick . value ) ) ;
WRITEUINT8 ( demo_p , buf ) ;
player - > pflags = pflags ;
}
// Save netvar data
2024-08-25 10:24:03 +00:00
savebuffer . buf = demo_p ;
savebuffer . size = demoend - demo_p ;
savebuffer . pos = 0 ;
CV_SaveDemoVars ( & savebuffer ) ;
demo_p = & savebuffer . buf [ savebuffer . pos ] ;
2020-03-19 20:09:55 +00:00
memset ( & oldcmd , 0 , sizeof ( oldcmd ) ) ;
memset ( & oldghost , 0 , sizeof ( oldghost ) ) ;
memset ( & ghostext , 0 , sizeof ( ghostext ) ) ;
ghostext . lastcolor = ghostext . color = GHC_NORMAL ;
ghostext . lastscale = ghostext . scale = FRACUNIT ;
if ( player - > mo )
{
oldghost . x = player - > mo - > x ;
oldghost . y = player - > mo - > y ;
oldghost . z = player - > mo - > z ;
oldghost . angle = player - > mo - > angle > > 24 ;
// preticker started us gravity flipped
if ( player - > mo - > eflags & MFE_VERTICALFLIP )
ghostext . flags | = EZT_FLIP ;
}
}
void G_BeginMetal ( void )
{
mobj_t * mo = players [ consoleplayer ] . mo ;
#if 0
if ( demo_p )
return ;
# endif
demo_p = demobuffer ;
// Write header.
M_Memcpy ( demo_p , DEMOHEADER , 12 ) ; demo_p + = 12 ;
WRITEUINT8 ( demo_p , VERSION ) ;
WRITEUINT8 ( demo_p , SUBVERSION ) ;
WRITEUINT16 ( demo_p , DEMOVERSION ) ;
// demo checksum
demo_p + = 16 ;
M_Memcpy ( demo_p , " METL " , 4 ) ; demo_p + = 4 ;
memset ( & ghostext , 0 , sizeof ( ghostext ) ) ;
ghostext . lastscale = ghostext . scale = FRACUNIT ;
// Set up our memory.
memset ( & oldmetal , 0 , sizeof ( oldmetal ) ) ;
oldmetal . x = mo - > x ;
oldmetal . y = mo - > y ;
oldmetal . z = mo - > z ;
oldmetal . angle = mo - > angle > > 24 ;
}
2023-08-04 18:36:56 +00:00
static void G_LoadDemoExtraFiles ( UINT8 * * pp , UINT16 this_demo_version )
2023-07-25 12:40:36 +00:00
{
2023-07-25 13:29:09 +00:00
UINT16 totalfiles ;
2023-07-25 12:40:36 +00:00
char filename [ MAX_WADPATH ] ;
UINT8 md5sum [ 16 ] ;
2024-03-01 19:35:04 +00:00
filestatus_t ncs = FS_NOTFOUND ;
2023-07-25 12:40:36 +00:00
boolean toomany = false ;
boolean alreadyloaded ;
2023-07-25 13:29:09 +00:00
UINT16 i , j ;
2023-07-25 12:40:36 +00:00
2023-08-04 18:36:56 +00:00
if ( this_demo_version < 0x0010 )
{
// demo has no file list
return ;
}
2023-07-25 13:29:09 +00:00
totalfiles = READUINT16 ( ( * pp ) ) ;
2023-07-25 12:40:36 +00:00
for ( i = 0 ; i < totalfiles ; + + i )
{
if ( toomany )
SKIPSTRING ( ( * pp ) ) ;
else
{
strlcpy ( filename , ( char * ) ( * pp ) , sizeof filename ) ;
SKIPSTRING ( ( * pp ) ) ;
}
READMEM ( ( * pp ) , md5sum , 16 ) ;
if ( ! toomany )
{
alreadyloaded = false ;
for ( j = 0 ; j < numwadfiles ; + + j )
{
if ( memcmp ( md5sum , wadfiles [ j ] - > md5sum , 16 ) = = 0 )
{
alreadyloaded = true ;
break ;
}
}
if ( alreadyloaded )
continue ;
if ( numwadfiles > = MAX_WADFILES )
toomany = true ;
else
ncs = findfile ( filename , md5sum , false ) ;
if ( toomany )
{
CONS_Alert ( CONS_WARNING , M_GetText ( " Too many files loaded to add anymore for demo playback \n " ) ) ;
if ( ! CON_Ready ( ) )
M_StartMessage ( M_GetText ( " There are too many files loaded to add this demo's addons. \n \n Demo playback may desync. \n \n Press ESC \n " ) , NULL , MM_NOTHING ) ;
}
else if ( ncs ! = FS_FOUND )
{
if ( ncs = = FS_NOTFOUND )
CONS_Alert ( CONS_NOTICE , M_GetText ( " You do not have a copy of %s \n " ) , filename ) ;
else if ( ncs = = FS_MD5SUMBAD )
CONS_Alert ( CONS_NOTICE , M_GetText ( " Checksum mismatch on %s \n " ) , filename ) ;
else
CONS_Alert ( CONS_NOTICE , M_GetText ( " Unknown error finding file %s \n " ) , filename ) ;
if ( ! CON_Ready ( ) )
M_StartMessage ( M_GetText ( " There were errors trying to add this demo's addons. Check the console for more information. \n \n Demo playback may desync. \n \n Press ESC \n " ) , NULL , MM_NOTHING ) ;
}
else
{
P_AddWadFile ( filename ) ;
}
}
}
}
2023-08-04 18:36:56 +00:00
static void G_SkipDemoExtraFiles ( UINT8 * * pp , UINT16 this_demo_version )
2023-07-25 12:40:36 +00:00
{
2023-07-25 13:29:09 +00:00
UINT16 totalfiles ;
UINT16 i ;
2023-07-25 12:40:36 +00:00
2023-08-04 18:36:56 +00:00
if ( this_demo_version < 0x0010 )
2023-07-25 12:40:36 +00:00
{
// demo has no file list
return ;
}
2023-07-25 13:29:09 +00:00
totalfiles = READUINT16 ( ( * pp ) ) ;
2023-07-25 12:40:36 +00:00
for ( i = 0 ; i < totalfiles ; + + i )
{
SKIPSTRING ( ( * pp ) ) ; // file name
( * pp ) + = 16 ; // md5
}
}
// G_CheckDemoExtraFiles: checks if our loaded WAD list matches the demo's.
// Enabling quick prevents filesystem checks to see if needed files are available to load.
2023-08-04 18:36:56 +00:00
static UINT8 G_CheckDemoExtraFiles ( UINT8 * * pp , boolean quick , UINT16 this_demo_version )
2023-07-25 12:40:36 +00:00
{
2023-07-25 13:29:09 +00:00
UINT16 totalfiles , filesloaded , nmusfilecount ;
2023-07-25 12:40:36 +00:00
char filename [ MAX_WADPATH ] ;
UINT8 md5sum [ 16 ] ;
boolean toomany = false ;
boolean alreadyloaded ;
2023-07-25 13:29:09 +00:00
UINT16 i , j ;
2023-07-25 12:40:36 +00:00
UINT8 error = DFILE_ERROR_NONE ;
2023-08-04 18:36:56 +00:00
if ( this_demo_version < 0x0010 )
2023-07-25 12:40:36 +00:00
{
// demo has no file list
return DFILE_ERROR_NONE ;
}
2023-07-25 13:29:09 +00:00
totalfiles = READUINT16 ( ( * pp ) ) ;
2023-07-25 12:40:36 +00:00
filesloaded = 0 ;
for ( i = 0 ; i < totalfiles ; + + i )
{
if ( toomany )
SKIPSTRING ( ( * pp ) ) ;
else
{
strlcpy ( filename , ( char * ) ( * pp ) , sizeof filename ) ;
SKIPSTRING ( ( * pp ) ) ;
}
READMEM ( ( * pp ) , md5sum , 16 ) ;
if ( ! toomany )
{
alreadyloaded = false ;
nmusfilecount = 0 ;
for ( j = 0 ; j < numwadfiles ; + + j )
{
if ( wadfiles [ j ] - > important & & j > mainwads )
nmusfilecount + + ;
else
continue ;
if ( memcmp ( md5sum , wadfiles [ j ] - > md5sum , 16 ) = = 0 )
{
alreadyloaded = true ;
if ( i ! = nmusfilecount - 1 & & error < DFILE_ERROR_OUTOFORDER )
error | = DFILE_ERROR_OUTOFORDER ;
break ;
}
}
if ( alreadyloaded )
{
filesloaded + + ;
continue ;
}
if ( numwadfiles > = MAX_WADFILES )
error = DFILE_ERROR_CANNOTLOAD ;
else if ( ! quick & & findfile ( filename , md5sum , false ) ! = FS_FOUND )
error = DFILE_ERROR_CANNOTLOAD ;
else if ( error < DFILE_ERROR_INCOMPLETEOUTOFORDER )
error | = DFILE_ERROR_NOTLOADED ;
} else
error = DFILE_ERROR_CANNOTLOAD ;
}
// Get final file count
nmusfilecount = 0 ;
for ( j = 0 ; j < numwadfiles ; + + j )
if ( wadfiles [ j ] - > important & & j > mainwads )
nmusfilecount + + ;
if ( ! error & & filesloaded < nmusfilecount )
error = DFILE_ERROR_EXTRAFILES ;
return error ;
}
2020-03-19 20:09:55 +00:00
void G_SetDemoTime ( UINT32 ptime , UINT32 pscore , UINT16 prings )
{
if ( ! demorecording | | ! demotime_p )
return ;
if ( demoflags & DF_RECORDATTACK )
{
WRITEUINT32 ( demotime_p , ptime ) ;
WRITEUINT32 ( demotime_p , pscore ) ;
WRITEUINT16 ( demotime_p , prings ) ;
demotime_p = NULL ;
}
else if ( demoflags & DF_NIGHTSATTACK )
{
WRITEUINT32 ( demotime_p , ptime ) ;
WRITEUINT32 ( demotime_p , pscore ) ;
demotime_p = NULL ;
}
}
// Returns bitfield:
// 1 == new demo has lower time
// 2 == new demo has higher score
// 4 == new demo has higher rings
UINT8 G_CmpDemoTime ( char * oldname , char * newname )
{
UINT8 * buffer , * p ;
UINT8 flags ;
UINT32 oldtime , newtime , oldscore , newscore ;
2023-08-04 18:36:56 +00:00
UINT16 oldrings , newrings , oldversion , newversion ;
2020-03-19 20:09:55 +00:00
size_t bufsize ATTRUNUSED ;
UINT8 c ;
UINT8 aflags = 0 ;
// load the new file
FIL_DefaultExtension ( newname , " .lmp " ) ;
bufsize = FIL_ReadFile ( newname , & buffer ) ;
I_Assert ( bufsize ! = 0 ) ;
p = buffer ;
// read demo header
I_Assert ( ! memcmp ( p , DEMOHEADER , 12 ) ) ;
p + = 12 ; // DEMOHEADER
c = READUINT8 ( p ) ; // VERSION
I_Assert ( c = = VERSION ) ;
c = READUINT8 ( p ) ; // SUBVERSION
I_Assert ( c = = SUBVERSION ) ;
2023-08-04 18:36:56 +00:00
newversion = READUINT16 ( p ) ;
I_Assert ( newversion = = DEMOVERSION ) ;
2020-03-19 20:09:55 +00:00
p + = 16 ; // demo checksum
I_Assert ( ! memcmp ( p , " PLAY " , 4 ) ) ;
p + = 4 ; // PLAY
p + = 2 ; // gamemap
p + = 16 ; // map md5
flags = READUINT8 ( p ) ; // demoflags
2023-08-04 18:36:56 +00:00
G_SkipDemoExtraFiles ( & p , newversion ) ;
2020-03-19 20:09:55 +00:00
aflags = flags & ( DF_RECORDATTACK | DF_NIGHTSATTACK ) ;
I_Assert ( aflags ) ;
if ( flags & DF_RECORDATTACK )
{
newtime = READUINT32 ( p ) ;
newscore = READUINT32 ( p ) ;
newrings = READUINT16 ( p ) ;
}
else if ( flags & DF_NIGHTSATTACK )
{
newtime = READUINT32 ( p ) ;
newscore = READUINT32 ( p ) ;
newrings = 0 ;
}
else // appease compiler
return 0 ;
Z_Free ( buffer ) ;
// load old file
FIL_DefaultExtension ( oldname , " .lmp " ) ;
if ( ! FIL_ReadFile ( oldname , & buffer ) )
{
CONS_Alert ( CONS_ERROR , M_GetText ( " Failed to read file '%s'. \n " ) , oldname ) ;
return UINT8_MAX ;
}
p = buffer ;
// read demo header
if ( memcmp ( p , DEMOHEADER , 12 ) )
{
CONS_Alert ( CONS_NOTICE , M_GetText ( " File '%s' invalid format. It will be overwritten. \n " ) , oldname ) ;
Z_Free ( buffer ) ;
return UINT8_MAX ;
} p + = 12 ; // DEMOHEADER
p + + ; // VERSION
p + + ; // SUBVERSION
oldversion = READUINT16 ( p ) ;
2023-09-22 16:16:54 +00:00
if ( oldversion < 0x000c | | oldversion > DEMOVERSION )
2020-03-19 20:09:55 +00:00
{
2023-09-22 16:16:54 +00:00
// too old (or new), cannot support
2020-03-19 20:09:55 +00:00
CONS_Alert ( CONS_NOTICE , M_GetText ( " File '%s' invalid format. It will be overwritten. \n " ) , oldname ) ;
Z_Free ( buffer ) ;
return UINT8_MAX ;
}
p + = 16 ; // demo checksum
if ( memcmp ( p , " PLAY " , 4 ) )
{
CONS_Alert ( CONS_NOTICE , M_GetText ( " File '%s' invalid format. It will be overwritten. \n " ) , oldname ) ;
Z_Free ( buffer ) ;
return UINT8_MAX ;
} p + = 4 ; // "PLAY"
if ( oldversion < = 0x0008 )
p + + ; // gamemap
else
p + = 2 ; // gamemap
p + = 16 ; // mapmd5
flags = READUINT8 ( p ) ;
2023-08-04 18:36:56 +00:00
G_SkipDemoExtraFiles ( & p , oldversion ) ;
2020-03-19 20:09:55 +00:00
if ( ! ( flags & aflags ) )
{
CONS_Alert ( CONS_NOTICE , M_GetText ( " File '%s' not from same game mode. It will be overwritten. \n " ) , oldname ) ;
Z_Free ( buffer ) ;
return UINT8_MAX ;
}
if ( flags & DF_RECORDATTACK )
{
oldtime = READUINT32 ( p ) ;
oldscore = READUINT32 ( p ) ;
oldrings = READUINT16 ( p ) ;
}
else if ( flags & DF_NIGHTSATTACK )
{
oldtime = READUINT32 ( p ) ;
oldscore = READUINT32 ( p ) ;
oldrings = 0 ;
}
else // appease compiler
return UINT8_MAX ;
Z_Free ( buffer ) ;
c = 0 ;
if ( newtime < oldtime
| | ( newtime = = oldtime & & ( newscore > oldscore | | newrings > oldrings ) ) )
c | = 1 ; // Better time
if ( newscore > oldscore
| | ( newscore = = oldscore & & newtime < oldtime ) )
c | = 1 < < 1 ; // Better score
if ( newrings > oldrings
| | ( newrings = = oldrings & & newtime < oldtime ) )
c | = 1 < < 2 ; // Better rings
return c ;
}
//
// G_PlayDemo
//
void G_DeferedPlayDemo ( const char * name )
{
COM_BufAddText ( " playdemo \" " ) ;
COM_BufAddText ( name ) ;
COM_BufAddText ( " \" \n " ) ;
}
//
// Start a demo from a .LMP file or from a wad resource
//
void G_DoPlayDemo ( char * defdemoname )
{
UINT8 i ;
lumpnum_t l ;
2020-05-24 00:29:07 +00:00
char skin [ 17 ] , color [ MAXCOLORNAME + 1 ] , * n , * pdemoname ;
2023-09-22 16:16:54 +00:00
UINT8 version , subversion , charability , charability2 , thrustfactor , accelstart , acceleration ;
2020-03-19 20:09:55 +00:00
pflags_t pflags ;
UINT32 randseed , followitem ;
fixed_t camerascale , shieldscale , actionspd , mindash , maxdash , normalspeed , runspeed , jumpfactor , height , spinheight ;
char msg [ 1024 ] ;
skin [ 16 ] = ' \0 ' ;
2020-05-24 00:29:07 +00:00
color [ MAXCOLORNAME ] = ' \0 ' ;
2020-03-19 20:09:55 +00:00
n = defdemoname + strlen ( defdemoname ) ;
while ( * n ! = ' / ' & & * n ! = ' \\ ' & & n ! = defdemoname )
n - - ;
if ( n ! = defdemoname )
n + + ;
pdemoname = ZZ_Alloc ( strlen ( n ) + 1 ) ;
strcpy ( pdemoname , n ) ;
// Internal if no extension, external if one exists
if ( FIL_CheckExtension ( defdemoname ) )
{
//FIL_DefaultExtension(defdemoname, ".lmp");
if ( ! FIL_ReadFile ( defdemoname , & demobuffer ) )
{
snprintf ( msg , 1024 , M_GetText ( " Failed to read file '%s'. \n " ) , defdemoname ) ;
CONS_Alert ( CONS_ERROR , " %s " , msg ) ;
gameaction = ga_nothing ;
M_StartMessage ( msg , NULL , MM_NOTHING ) ;
return ;
}
demo_p = demobuffer ;
}
// load demo resource from WAD
else if ( ( l = W_CheckNumForName ( defdemoname ) ) = = LUMPERROR )
{
snprintf ( msg , 1024 , M_GetText ( " Failed to read lump '%s'. \n " ) , defdemoname ) ;
CONS_Alert ( CONS_ERROR , " %s " , msg ) ;
gameaction = ga_nothing ;
M_StartMessage ( msg , NULL , MM_NOTHING ) ;
return ;
}
else // it's an internal demo
demobuffer = demo_p = W_CacheLumpNum ( l , PU_STATIC ) ;
// read demo header
gameaction = ga_nothing ;
demoplayback = true ;
if ( memcmp ( demo_p , DEMOHEADER , 12 ) )
{
snprintf ( msg , 1024 , M_GetText ( " %s is not a SRB2 replay file. \n " ) , pdemoname ) ;
CONS_Alert ( CONS_ERROR , " %s " , msg ) ;
M_StartMessage ( msg , NULL , MM_NOTHING ) ;
Z_Free ( pdemoname ) ;
Z_Free ( demobuffer ) ;
demoplayback = false ;
titledemo = false ;
return ;
}
demo_p + = 12 ; // DEMOHEADER
version = READUINT8 ( demo_p ) ;
subversion = READUINT8 ( demo_p ) ;
demoversion = READUINT16 ( demo_p ) ;
2023-07-26 11:50:47 +00:00
demo_forwardmove_rng = ( demoversion < 0x0010 ) ;
2020-06-20 01:04:20 +00:00
# ifdef OLD22DEMOCOMPAT
2023-09-22 16:16:54 +00:00
if ( demoversion < 0x000c | | demoversion > DEMOVERSION )
# else
if ( demoversion < 0x000d | | demoversion > DEMOVERSION )
2020-06-20 01:04:20 +00:00
# endif
2023-09-22 16:16:54 +00:00
{
// too old (or new), cannot support
2020-03-19 20:09:55 +00:00
snprintf ( msg , 1024 , M_GetText ( " %s is an incompatible replay format and cannot be played. \n " ) , pdemoname ) ;
CONS_Alert ( CONS_ERROR , " %s " , msg ) ;
M_StartMessage ( msg , NULL , MM_NOTHING ) ;
Z_Free ( pdemoname ) ;
Z_Free ( demobuffer ) ;
demoplayback = false ;
titledemo = false ;
return ;
}
demo_p + = 16 ; // demo checksum
if ( memcmp ( demo_p , " PLAY " , 4 ) )
{
snprintf ( msg , 1024 , M_GetText ( " %s is the wrong type of recording and cannot be played. \n " ) , pdemoname ) ;
CONS_Alert ( CONS_ERROR , " %s " , msg ) ;
M_StartMessage ( msg , NULL , MM_NOTHING ) ;
Z_Free ( pdemoname ) ;
Z_Free ( demobuffer ) ;
demoplayback = false ;
titledemo = false ;
return ;
}
demo_p + = 4 ; // "PLAY"
gamemap = READINT16 ( demo_p ) ;
demo_p + = 16 ; // mapmd5
demoflags = READUINT8 ( demo_p ) ;
2023-07-25 12:40:36 +00:00
2023-08-04 18:36:56 +00:00
if ( titledemo )
2023-07-25 12:40:36 +00:00
{
// Titledemos should always play and ought to always be compatible with whatever wadlist is running.
2023-08-04 18:36:56 +00:00
G_SkipDemoExtraFiles ( & demo_p , demoversion ) ;
2023-07-25 12:40:36 +00:00
}
else if ( demofileoverride = = DFILE_OVERRIDE_LOAD )
{
2023-08-04 18:36:56 +00:00
G_LoadDemoExtraFiles ( & demo_p , demoversion ) ;
2023-07-25 12:40:36 +00:00
}
else if ( demofileoverride = = DFILE_OVERRIDE_SKIP )
{
2023-08-04 18:36:56 +00:00
G_SkipDemoExtraFiles ( & demo_p , demoversion ) ;
2023-07-25 12:40:36 +00:00
}
else
{
2023-08-04 18:36:56 +00:00
UINT8 error = G_CheckDemoExtraFiles ( & demo_p , false , demoversion ) ;
2023-07-25 12:40:36 +00:00
if ( error )
{
switch ( error )
{
case DFILE_ERROR_NOTLOADED :
snprintf ( msg , 1024 ,
M_GetText ( " Required files for this demo are not loaded. \n \n Use \n \" playdemo %s -addfiles \" \n to load them and play the demo. \n " ) ,
pdemoname ) ;
break ;
case DFILE_ERROR_OUTOFORDER :
snprintf ( msg , 1024 ,
M_GetText ( " Required files for this demo are loaded out of order. \n \n Use \n \" playdemo %s -force \" \n to play the demo anyway. \n " ) ,
pdemoname ) ;
break ;
case DFILE_ERROR_INCOMPLETEOUTOFORDER :
snprintf ( msg , 1024 ,
M_GetText ( " Required files for this demo are not loaded, and some are out of order. \n \n Use \n \" playdemo %s -addfiles \" \n to load needed files and play the demo. \n " ) ,
pdemoname ) ;
break ;
case DFILE_ERROR_CANNOTLOAD :
snprintf ( msg , 1024 ,
M_GetText ( " Required files for this demo cannot be loaded. \n \n Use \n \" playdemo %s -force \" \n to play the demo anyway. \n " ) ,
pdemoname ) ;
break ;
case DFILE_ERROR_EXTRAFILES :
snprintf ( msg , 1024 ,
M_GetText ( " You have additional files loaded beyond the demo's file list. \n \n Use \n \" playdemo %s -force \" \n to play the demo anyway. \n " ) ,
pdemoname ) ;
break ;
}
CONS_Alert ( CONS_ERROR , " %s " , msg ) ;
M_StartMessage ( msg , NULL , MM_NOTHING ) ;
Z_Free ( pdemoname ) ;
Z_Free ( demobuffer ) ;
demoplayback = false ;
titledemo = false ;
return ;
}
}
2020-03-19 20:09:55 +00:00
modeattacking = ( demoflags & DF_ATTACKMASK ) > > DF_ATTACKSHIFT ;
CON_ToggleOff ( ) ;
hu_demoscore = 0 ;
hu_demotime = UINT32_MAX ;
hu_demorings = 0 ;
switch ( modeattacking )
{
case ATTACKING_NONE : // 0
break ;
case ATTACKING_RECORD : // 1
hu_demotime = READUINT32 ( demo_p ) ;
hu_demoscore = READUINT32 ( demo_p ) ;
hu_demorings = READUINT16 ( demo_p ) ;
break ;
case ATTACKING_NIGHTS : // 2
hu_demotime = READUINT32 ( demo_p ) ;
hu_demoscore = READUINT32 ( demo_p ) ;
break ;
default : // 3
modeattacking = ATTACKING_NONE ;
break ;
}
// Random seed
randseed = READUINT32 ( demo_p ) ;
// Player name
M_Memcpy ( player_names [ 0 ] , demo_p , 16 ) ;
demo_p + = 16 ;
// Skin
M_Memcpy ( skin , demo_p , 16 ) ;
demo_p + = 16 ;
// Color
2023-09-22 16:16:54 +00:00
M_Memcpy ( color , demo_p , ( demoversion < 0x000d ) ? 16 : MAXCOLORNAME ) ;
demo_p + = ( demoversion < 0x000d ) ? 16 : MAXCOLORNAME ;
2020-03-19 20:09:55 +00:00
charability = READUINT8 ( demo_p ) ;
charability2 = READUINT8 ( demo_p ) ;
2023-07-08 21:14:37 +00:00
actionspd = ( demoversion < 0x0010 ) ? ( fixed_t ) READUINT8 ( demo_p ) < < FRACBITS : READFIXED ( demo_p ) ;
mindash = ( demoversion < 0x0010 ) ? ( fixed_t ) READUINT8 ( demo_p ) < < FRACBITS : READFIXED ( demo_p ) ;
maxdash = ( demoversion < 0x0010 ) ? ( fixed_t ) READUINT8 ( demo_p ) < < FRACBITS : READFIXED ( demo_p ) ;
normalspeed = ( demoversion < 0x0010 ) ? ( fixed_t ) READUINT8 ( demo_p ) < < FRACBITS : READFIXED ( demo_p ) ;
runspeed = ( demoversion < 0x0010 ) ? ( fixed_t ) READUINT8 ( demo_p ) < < FRACBITS : READFIXED ( demo_p ) ;
2020-03-19 20:09:55 +00:00
thrustfactor = READUINT8 ( demo_p ) ;
accelstart = READUINT8 ( demo_p ) ;
acceleration = READUINT8 ( demo_p ) ;
2020-11-09 15:01:20 +00:00
height = ( demoversion < 0x000e ) ? ( fixed_t ) READUINT8 ( demo_p ) < < FRACBITS : READFIXED ( demo_p ) ;
spinheight = ( demoversion < 0x000e ) ? ( fixed_t ) READUINT8 ( demo_p ) < < FRACBITS : READFIXED ( demo_p ) ;
2023-07-08 21:14:37 +00:00
camerascale = ( demoversion < 0x0010 ) ? ( fixed_t ) READUINT8 ( demo_p ) < < FRACBITS : READFIXED ( demo_p ) ;
shieldscale = ( demoversion < 0x0010 ) ? ( fixed_t ) READUINT8 ( demo_p ) < < FRACBITS : READFIXED ( demo_p ) ;
2020-03-19 20:09:55 +00:00
jumpfactor = READFIXED ( demo_p ) ;
followitem = READUINT32 ( demo_p ) ;
// pflag data
{
UINT8 buf = READUINT8 ( demo_p ) ;
pflags = 0 ;
if ( buf & 0x01 )
pflags | = PF_FLIPCAM ;
if ( buf & 0x02 )
pflags | = PF_ANALOGMODE ;
if ( buf & 0x04 )
pflags | = PF_DIRECTIONCHAR ;
if ( buf & 0x08 )
pflags | = PF_AUTOBRAKE ;
CV_SetValue ( & cv_showinputjoy , ! ! ( buf & 0x10 ) ) ;
}
// net var data
2020-06-20 01:04:20 +00:00
# ifdef OLD22DEMOCOMPAT
2023-09-22 16:16:54 +00:00
if ( demoversion < 0x000d )
2024-08-25 10:24:03 +00:00
{
save_t savebuffer ;
savebuffer . buf = demo_p ;
savebuffer . size = demoend - demo_p ;
savebuffer . pos = 0 ;
CV_LoadOldDemoVars ( & savebuffer ) ;
demo_p = & savebuffer . buf [ savebuffer . pos ] ;
}
2020-06-20 01:04:20 +00:00
else
# endif
2024-08-25 10:24:03 +00:00
{
save_t savebuffer ;
savebuffer . buf = demo_p ;
savebuffer . size = demoend - demo_p ;
savebuffer . pos = 0 ;
CV_LoadDemoVars ( & savebuffer ) ;
demo_p = & savebuffer . buf [ savebuffer . pos ] ;
}
2020-03-19 20:09:55 +00:00
// Sigh ... it's an empty demo.
if ( * demo_p = = DEMOMARKER )
{
snprintf ( msg , 1024 , M_GetText ( " %s contains no data to be played. \n " ) , pdemoname ) ;
CONS_Alert ( CONS_ERROR , " %s " , msg ) ;
M_StartMessage ( msg , NULL , MM_NOTHING ) ;
Z_Free ( pdemoname ) ;
Z_Free ( demobuffer ) ;
demoplayback = false ;
titledemo = false ;
return ;
}
Z_Free ( pdemoname ) ;
memset ( & oldcmd , 0 , sizeof ( oldcmd ) ) ;
memset ( & oldghost , 0 , sizeof ( oldghost ) ) ;
if ( VERSION ! = version | | SUBVERSION ! = subversion )
CONS_Alert ( CONS_WARNING , M_GetText ( " Demo version does not match game version. Desyncs may occur. \n " ) ) ;
// didn't start recording right away.
demo_start = false ;
// Set skin
SetPlayerSkin ( 0 , skin ) ;
2020-12-12 11:06:57 +00:00
LUA_HookInt ( gamemap , HOOK ( MapChange ) ) ;
2020-03-19 20:09:55 +00:00
displayplayer = consoleplayer = 0 ;
memset ( playeringame , 0 , sizeof ( playeringame ) ) ;
playeringame [ 0 ] = true ;
P_SetRandSeed ( randseed ) ;
G_InitNew ( false , G_BuildMapName ( gamemap ) , true , true , false ) ;
// Set color
2021-08-09 18:57:07 +00:00
players [ 0 ] . skincolor = skins [ players [ 0 ] . skin ] - > prefcolor ;
2020-05-24 00:29:07 +00:00
for ( i = 0 ; i < numskincolors ; i + + )
if ( ! stricmp ( skincolors [ i ] . name , color ) )
2020-03-19 20:09:55 +00:00
{
players [ 0 ] . skincolor = i ;
break ;
}
if ( players [ 0 ] . mo )
{
players [ 0 ] . mo - > color = players [ 0 ] . skincolor ;
oldghost . x = players [ 0 ] . mo - > x ;
oldghost . y = players [ 0 ] . mo - > y ;
oldghost . z = players [ 0 ] . mo - > z ;
}
// Set saved attribute values
// No cheat checking here, because even if they ARE wrong...
// it would only break the replay if we clipped them.
players [ 0 ] . camerascale = camerascale ;
players [ 0 ] . shieldscale = shieldscale ;
players [ 0 ] . charability = charability ;
players [ 0 ] . charability2 = charability2 ;
players [ 0 ] . actionspd = actionspd ;
players [ 0 ] . mindash = mindash ;
players [ 0 ] . maxdash = maxdash ;
players [ 0 ] . normalspeed = normalspeed ;
players [ 0 ] . runspeed = runspeed ;
players [ 0 ] . thrustfactor = thrustfactor ;
players [ 0 ] . accelstart = accelstart ;
players [ 0 ] . acceleration = acceleration ;
players [ 0 ] . height = height ;
players [ 0 ] . spinheight = spinheight ;
players [ 0 ] . jumpfactor = jumpfactor ;
players [ 0 ] . followitem = followitem ;
players [ 0 ] . pflags = pflags ;
demo_start = true ;
}
2023-07-25 12:40:36 +00:00
//
// Check if a replay can be loaded from the menu
//
UINT8 G_CheckDemoForError ( char * defdemoname )
{
lumpnum_t l ;
char * n , * pdemoname ;
2023-08-04 18:36:56 +00:00
UINT16 our_demo_version ;
if ( titledemo )
{
// Don't do anything with files for these.
return DFILE_ERROR_NONE ;
}
2023-07-25 12:40:36 +00:00
n = defdemoname + strlen ( defdemoname ) ;
while ( * n ! = ' / ' & & * n ! = ' \\ ' & & n ! = defdemoname )
n - - ;
if ( n ! = defdemoname )
n + + ;
pdemoname = ZZ_Alloc ( strlen ( n ) + 1 ) ;
strcpy ( pdemoname , n ) ;
// Internal if no extension, external if one exists
if ( FIL_CheckExtension ( defdemoname ) )
{
//FIL_DefaultExtension(defdemoname, ".lmp");
if ( ! FIL_ReadFile ( defdemoname , & demobuffer ) )
{
return DFILE_ERROR_NOTDEMO ;
}
demo_p = demobuffer ;
}
// load demo resource from WAD
else if ( ( l = W_CheckNumForName ( defdemoname ) ) = = LUMPERROR )
{
return DFILE_ERROR_NOTDEMO ;
}
else // it's an internal demo
{
demobuffer = demo_p = W_CacheLumpNum ( l , PU_STATIC ) ;
}
// read demo header
if ( memcmp ( demo_p , DEMOHEADER , 12 ) )
{
return DFILE_ERROR_NOTDEMO ;
}
demo_p + = 12 ; // DEMOHEADER
demo_p + + ; // version
demo_p + + ; // subversion
2023-08-04 18:36:56 +00:00
our_demo_version = READUINT16 ( demo_p ) ;
2023-07-25 12:40:36 +00:00
# ifdef OLD22DEMOCOMPAT
2023-09-22 16:16:54 +00:00
if ( our_demo_version < 0x000c | | our_demo_version > DEMOVERSION )
# else
if ( our_demo_version < 0x000d | | our_demo_version > DEMOVERSION )
2023-07-25 12:40:36 +00:00
# endif
2023-09-22 16:16:54 +00:00
{
// too old (or new), cannot support
2023-07-25 12:40:36 +00:00
return DFILE_ERROR_NOTDEMO ;
}
demo_p + = 16 ; // demo checksum
if ( memcmp ( demo_p , " PLAY " , 4 ) )
{
return DFILE_ERROR_NOTDEMO ;
}
demo_p + = 4 ; // "PLAY"
demo_p + = 2 ; // gamemap
demo_p + = 16 ; // mapmd5
demo_p + + ; // demoflags
2023-08-04 19:59:47 +00:00
return G_CheckDemoExtraFiles ( & demo_p , true , our_demo_version ) ;
2023-07-25 12:40:36 +00:00
}
2020-03-19 20:09:55 +00:00
void G_AddGhost ( char * defdemoname )
{
INT32 i ;
lumpnum_t l ;
2020-05-24 00:29:07 +00:00
char name [ 17 ] , skin [ 17 ] , color [ MAXCOLORNAME + 1 ] , * n , * pdemoname , md5 [ 16 ] ;
2020-03-19 20:09:55 +00:00
demoghost * gh ;
2021-06-10 18:18:45 +00:00
UINT8 flags , subversion ;
2020-03-19 20:09:55 +00:00
UINT8 * buffer , * p ;
mapthing_t * mthing ;
UINT16 count , ghostversion ;
name [ 16 ] = ' \0 ' ;
skin [ 16 ] = ' \0 ' ;
color [ 16 ] = ' \0 ' ;
n = defdemoname + strlen ( defdemoname ) ;
while ( * n ! = ' / ' & & * n ! = ' \\ ' & & n ! = defdemoname )
n - - ;
if ( n ! = defdemoname )
n + + ;
pdemoname = ZZ_Alloc ( strlen ( n ) + 1 ) ;
strcpy ( pdemoname , n ) ;
// Internal if no extension, external if one exists
if ( FIL_CheckExtension ( defdemoname ) )
{
//FIL_DefaultExtension(defdemoname, ".lmp");
if ( ! FIL_ReadFileTag ( defdemoname , & buffer , PU_LEVEL ) )
{
CONS_Alert ( CONS_ERROR , M_GetText ( " Failed to read file '%s'. \n " ) , defdemoname ) ;
Z_Free ( pdemoname ) ;
return ;
}
p = buffer ;
}
// load demo resource from WAD
else if ( ( l = W_CheckNumForName ( defdemoname ) ) = = LUMPERROR )
{
CONS_Alert ( CONS_ERROR , M_GetText ( " Failed to read lump '%s'. \n " ) , defdemoname ) ;
Z_Free ( pdemoname ) ;
return ;
}
else // it's an internal demo
buffer = p = W_CacheLumpNum ( l , PU_LEVEL ) ;
// read demo header
if ( memcmp ( p , DEMOHEADER , 12 ) )
{
CONS_Alert ( CONS_NOTICE , M_GetText ( " Ghost %s: Not a SRB2 replay. \n " ) , pdemoname ) ;
Z_Free ( pdemoname ) ;
Z_Free ( buffer ) ;
return ;
} p + = 12 ; // DEMOHEADER
p + + ; // VERSION
2021-06-10 18:18:45 +00:00
subversion = READUINT8 ( p ) ; // SUBVERSION
2020-03-19 20:09:55 +00:00
ghostversion = READUINT16 ( p ) ;
2023-09-22 16:16:54 +00:00
if ( ghostversion < 0x000c | | ghostversion > DEMOVERSION )
2020-03-19 20:09:55 +00:00
{
2023-09-22 16:16:54 +00:00
// too old (or new), cannot support
2020-03-19 20:09:55 +00:00
CONS_Alert ( CONS_NOTICE , M_GetText ( " Ghost %s: Demo version incompatible. \n " ) , pdemoname ) ;
Z_Free ( pdemoname ) ;
Z_Free ( buffer ) ;
return ;
}
M_Memcpy ( md5 , p , 16 ) ; p + = 16 ; // demo checksum
for ( gh = ghosts ; gh ; gh = gh - > next )
if ( ! memcmp ( md5 , gh - > checksum , 16 ) ) // another ghost in the game already has this checksum?
{ // Don't add another one, then!
CONS_Debug ( DBG_SETUP , " Rejecting duplicate ghost %s (MD5 was matched) \n " , pdemoname ) ;
Z_Free ( pdemoname ) ;
Z_Free ( buffer ) ;
return ;
}
if ( memcmp ( p , " PLAY " , 4 ) )
{
CONS_Alert ( CONS_NOTICE , M_GetText ( " Ghost %s: Demo format unacceptable. \n " ) , pdemoname ) ;
Z_Free ( pdemoname ) ;
Z_Free ( buffer ) ;
return ;
} p + = 4 ; // "PLAY"
if ( ghostversion < = 0x0008 )
p + + ; // gamemap
else
p + = 2 ; // gamemap
p + = 16 ; // mapmd5 (possibly check for consistency?)
flags = READUINT8 ( p ) ;
if ( ! ( flags & DF_GHOST ) )
{
CONS_Alert ( CONS_NOTICE , M_GetText ( " Ghost %s: No ghost data in this demo. \n " ) , pdemoname ) ;
Z_Free ( pdemoname ) ;
Z_Free ( buffer ) ;
return ;
}
2023-07-25 12:40:36 +00:00
2023-08-04 18:36:56 +00:00
G_SkipDemoExtraFiles ( & p , ghostversion ) ; // Don't wanna modify the file list for ghosts.
2023-07-25 12:40:36 +00:00
2020-03-19 20:09:55 +00:00
switch ( ( flags & DF_ATTACKMASK ) > > DF_ATTACKSHIFT )
{
case ATTACKING_NONE : // 0
break ;
case ATTACKING_RECORD : // 1
p + = 10 ; // demo time, score, and rings
break ;
case ATTACKING_NIGHTS : // 2
p + = 8 ; // demo time left, score
break ;
default : // 3
break ;
}
p + = 4 ; // random seed
// Player name (TODO: Display this somehow if it doesn't match cv_playername!)
M_Memcpy ( name , p , 16 ) ;
p + = 16 ;
// Skin
M_Memcpy ( skin , p , 16 ) ;
p + = 16 ;
// Color
2023-09-22 16:16:54 +00:00
M_Memcpy ( color , p , ( ghostversion < 0x000d ) ? 16 : MAXCOLORNAME ) ;
p + = ( ghostversion < 0x000d ) ? 16 : MAXCOLORNAME ;
2020-03-19 20:09:55 +00:00
// Ghosts do not have a player structure to put this in.
p + + ; // charability
p + + ; // charability2
2023-07-08 21:14:37 +00:00
p + = ( ghostversion < 0x0010 ) ? 5 : 5 * sizeof ( fixed_t ) ; // actionspd, mindash, maxdash, normalspeed, and runspeed
2020-03-19 20:09:55 +00:00
p + + ; // thrustfactor
p + + ; // accelstart
p + + ; // acceleration
2020-11-09 15:01:20 +00:00
p + = ( ghostversion < 0x000e ) ? 2 : 2 * sizeof ( fixed_t ) ; // height and spinheight
2023-07-08 21:14:37 +00:00
p + = ( ghostversion < 0x0010 ) ? 2 : 2 * sizeof ( fixed_t ) ; // camerascale and shieldscale
2020-03-19 20:09:55 +00:00
p + = 4 ; // jumpfactor
p + = 4 ; // followitem
p + + ; // pflag data
// net var data
count = READUINT16 ( p ) ;
while ( count - - )
{
2021-06-10 18:18:45 +00:00
// In 2.2.7 netvar saving was updated
if ( subversion < 7 )
{
p + = 2 ;
SKIPSTRING ( p ) ;
p + + ;
}
else
{
SKIPSTRING ( p ) ;
SKIPSTRING ( p ) ;
p + + ;
}
2020-03-19 20:09:55 +00:00
}
if ( * p = = DEMOMARKER )
{
CONS_Alert ( CONS_NOTICE , M_GetText ( " Failed to add ghost %s: Replay is empty. \n " ) , pdemoname ) ;
Z_Free ( pdemoname ) ;
Z_Free ( buffer ) ;
return ;
}
gh = Z_Calloc ( sizeof ( demoghost ) , PU_LEVEL , NULL ) ;
gh - > next = ghosts ;
gh - > buffer = buffer ;
M_Memcpy ( gh - > checksum , md5 , 16 ) ;
gh - > p = p ;
ghosts = gh ;
gh - > version = ghostversion ;
mthing = playerstarts [ 0 ] ;
I_Assert ( mthing ) ;
{ // A bit more complex than P_SpawnPlayer because ghosts aren't solid and won't just push themselves out of the ceiling.
fixed_t z , f , c ;
fixed_t offset = mthing - > z < < FRACBITS ;
2023-04-30 10:43:31 +00:00
P_SetTarget ( & gh - > mo , P_SpawnMobj ( mthing - > x < < FRACBITS , mthing - > y < < FRACBITS , 0 , MT_GHOST ) ) ;
if ( P_MobjWasRemoved ( gh - > mo ) )
return ;
2020-03-19 20:09:55 +00:00
gh - > mo - > angle = FixedAngle ( mthing - > angle < < FRACBITS ) ;
f = gh - > mo - > floorz ;
c = gh - > mo - > ceilingz - mobjinfo [ MT_PLAYER ] . height ;
2021-12-26 20:11:24 +00:00
if ( ! ! ( mthing - > args [ 0 ] ) ^ ! ! ( mthing - > options & MTF_OBJECTFLIP ) )
2020-03-19 20:09:55 +00:00
{
z = c - offset ;
if ( z < f )
z = f ;
}
else
{
z = f + offset ;
if ( z > c )
z = c ;
}
gh - > mo - > z = z ;
}
gh - > oldmo . x = gh - > mo - > x ;
gh - > oldmo . y = gh - > mo - > y ;
gh - > oldmo . z = gh - > mo - > z ;
// Set skin
2021-08-09 18:57:07 +00:00
gh - > mo - > skin = skins [ 0 ] ;
2020-03-19 20:09:55 +00:00
for ( i = 0 ; i < numskins ; i + + )
2021-08-09 18:57:07 +00:00
if ( ! stricmp ( skins [ i ] - > name , skin ) )
2020-03-19 20:09:55 +00:00
{
2021-08-09 18:57:07 +00:00
gh - > mo - > skin = skins [ i ] ;
2020-03-19 20:09:55 +00:00
break ;
}
gh - > oldmo . skin = gh - > mo - > skin ;
// Set color
gh - > mo - > color = ( ( skin_t * ) gh - > mo - > skin ) - > prefcolor ;
2020-05-24 00:29:07 +00:00
for ( i = 0 ; i < numskincolors ; i + + )
if ( ! stricmp ( skincolors [ i ] . name , color ) )
2020-03-19 20:09:55 +00:00
{
2020-05-24 00:29:07 +00:00
gh - > mo - > color = ( UINT16 ) i ;
2020-03-19 20:09:55 +00:00
break ;
}
gh - > oldmo . color = gh - > mo - > color ;
2023-11-13 00:46:49 +00:00
gh - > mo - > state = & states [ S_PLAY_STND ] ;
2020-03-19 20:09:55 +00:00
gh - > mo - > sprite = gh - > mo - > state - > sprite ;
2023-11-13 00:46:49 +00:00
gh - > mo - > sprite2 = P_GetStateSprite2 ( gh - > mo - > state ) ;
gh - > mo - > frame = ( gh - > mo - > state - > frame & ~ FF_FRAMEMASK ) | P_GetSprite2StateFrame ( gh - > mo - > state ) ;
2020-03-19 20:09:55 +00:00
gh - > mo - > flags2 | = MF2_DONTDRAW ;
gh - > fadein = ( 9 - 3 ) * 6 ; // fade from invisible to trans30 over as close to 35 tics as possible
gh - > mo - > tics = - 1 ;
CONS_Printf ( M_GetText ( " Added ghost %s from %s \n " ) , name , pdemoname ) ;
Z_Free ( pdemoname ) ;
}
2020-03-19 20:42:51 +00:00
// Clean up all ghosts
void G_FreeGhosts ( void )
{
while ( ghosts )
{
demoghost * next = ghosts - > next ;
Z_Free ( ghosts ) ;
ghosts = next ;
}
ghosts = NULL ;
}
2020-03-19 20:09:55 +00:00
//
// G_TimeDemo
// NOTE: name is a full filename for external demos
//
static INT32 restorecv_vidwait ;
void G_TimeDemo ( const char * name )
{
nodrawers = M_CheckParm ( " -nodraw " ) ;
noblit = M_CheckParm ( " -noblit " ) ;
restorecv_vidwait = cv_vidwait . value ;
if ( cv_vidwait . value )
CV_Set ( & cv_vidwait , " 0 " ) ;
timingdemo = true ;
singletics = true ;
framecount = 0 ;
demostarttime = I_GetTime ( ) ;
G_DeferedPlayDemo ( name ) ;
}
void G_DoPlayMetal ( void )
{
lumpnum_t l ;
mobj_t * mo = NULL ;
thinker_t * th ;
// it's an internal demo
if ( ( l = W_CheckNumForName ( va ( " %sMS " , G_BuildMapName ( gamemap ) ) ) ) = = LUMPERROR )
{
CONS_Alert ( CONS_WARNING , M_GetText ( " No bot recording for this map. \n " ) ) ;
return ;
}
else
metalbuffer = metal_p = W_CacheLumpNum ( l , PU_STATIC ) ;
// find metal sonic
for ( th = thlist [ THINK_MOBJ ] . next ; th ! = & thlist [ THINK_MOBJ ] ; th = th - > next )
{
if ( th - > function . acp1 = = ( actionf_p1 ) P_RemoveThinkerDelayed )
continue ;
mo = ( mobj_t * ) th ;
if ( mo - > type ! = MT_METALSONIC_RACE )
continue ;
break ;
}
if ( th = = & thlist [ THINK_MOBJ ] )
{
CONS_Alert ( CONS_ERROR , M_GetText ( " Failed to find bot entity. \n " ) ) ;
Z_Free ( metalbuffer ) ;
return ;
}
// read demo header
metal_p + = 12 ; // DEMOHEADER
metal_p + + ; // VERSION
metal_p + + ; // SUBVERSION
metalversion = READUINT16 ( metal_p ) ;
2023-09-22 16:16:54 +00:00
if ( metalversion < 0x000c | | metalversion > DEMOVERSION )
2020-03-19 20:09:55 +00:00
{
2023-09-22 16:16:54 +00:00
// too old (or new), cannot support
2020-03-19 20:09:55 +00:00
CONS_Alert ( CONS_WARNING , M_GetText ( " Failed to load bot recording for this map, format version incompatible. \n " ) ) ;
Z_Free ( metalbuffer ) ;
return ;
}
metal_p + = 16 ; // demo checksum
if ( memcmp ( metal_p , " METL " , 4 ) )
{
CONS_Alert ( CONS_WARNING , M_GetText ( " Failed to load bot recording for this map, wasn't recorded in Metal format. \n " ) ) ;
Z_Free ( metalbuffer ) ;
return ;
} metal_p + = 4 ; // "METL"
// read initial tic
memset ( & oldmetal , 0 , sizeof ( oldmetal ) ) ;
oldmetal . x = mo - > x ;
oldmetal . y = mo - > y ;
oldmetal . z = mo - > z ;
metalplayback = mo ;
}
void G_DoneLevelLoad ( void )
{
CONS_Printf ( M_GetText ( " Loaded level in %f sec \n " ) , ( double ) ( I_GetTime ( ) - demostarttime ) / TICRATE ) ;
framecount = 0 ;
demostarttime = I_GetTime ( ) ;
}
/*
= = = = = = = = = = = = = = = = = = =
=
= G_CheckDemoStatus
=
= Called after a death or level completion to allow demos to be cleaned up
= Returns true if a new demo loop action will take place
= = = = = = = = = = = = = = = = = = =
*/
2020-05-14 19:57:21 +00:00
// Writes the demo's checksum, or just random garbage if you can't do that for some reason.
static void WriteDemoChecksum ( void )
{
UINT8 * p = demobuffer + 16 ; // checksum position
# ifdef NOMD5
UINT8 i ;
for ( i = 0 ; i < 16 ; i + + , p + + )
* p = P_RandomByte ( ) ; // This MD5 was chosen by fair dice roll and most likely < 50% correct.
# else
md5_buffer ( ( char * ) p + 16 , demo_p - ( p + 16 ) , p ) ; // make a checksum of everything after the checksum in the file.
# endif
}
// Stops recording a demo.
static void G_StopDemoRecording ( void )
{
boolean saved = false ;
2020-07-21 02:19:44 +00:00
if ( demo_p )
{
WRITEUINT8 ( demo_p , DEMOMARKER ) ; // add the demo end marker
WriteDemoChecksum ( ) ;
saved = FIL_WriteFile ( va ( pandf , srb2home , demoname ) , demobuffer , demo_p - demobuffer ) ; // finally output the file.
}
2020-05-14 19:57:21 +00:00
free ( demobuffer ) ;
demorecording = false ;
if ( modeattacking ! = ATTACKING_RECORD )
{
if ( saved )
CONS_Printf ( M_GetText ( " Demo %s recorded \n " ) , demoname ) ;
else
CONS_Alert ( CONS_WARNING , M_GetText ( " Demo %s not saved \n " ) , demoname ) ;
}
}
2020-03-19 20:09:55 +00:00
// Stops metal sonic's demo. Separate from other functions because metal + replays can coexist
void G_StopMetalDemo ( void )
{
// Metal Sonic finishing doesn't end the game, dammit.
Z_Free ( metalbuffer ) ;
metalbuffer = NULL ;
metalplayback = NULL ;
metal_p = NULL ;
}
// Stops metal sonic recording.
ATTRNORETURN void FUNCNORETURN G_StopMetalRecording ( boolean kill )
{
boolean saved = false ;
if ( demo_p )
{
2020-05-14 19:57:21 +00:00
WRITEUINT8 ( demo_p , ( kill ) ? METALDEATH : DEMOMARKER ) ; // add the demo end (or metal death) marker
WriteDemoChecksum ( ) ;
2021-08-22 20:13:40 +00:00
sprintf ( demoname , " %sMS.LMP " , G_BuildMapName ( gamemap ) ) ;
saved = FIL_WriteFile ( va ( pandf , srb2home , demoname ) , demobuffer , demo_p - demobuffer ) ; // finally output the file.
2020-03-19 20:09:55 +00:00
}
free ( demobuffer ) ;
metalrecording = false ;
if ( saved )
2021-08-22 20:13:40 +00:00
I_Error ( " Saved to %s " , demoname ) ;
2020-03-19 20:09:55 +00:00
I_Error ( " Failed to save demo! " ) ;
}
2020-05-14 19:57:21 +00:00
// Stops timing a demo.
static void G_StopTimingDemo ( void )
{
INT32 demotime ;
double f1 , f2 ;
demotime = I_GetTime ( ) - demostarttime ;
if ( ! demotime )
return ;
G_StopDemo ( ) ;
timingdemo = false ;
f1 = ( double ) demotime ;
f2 = ( double ) framecount * TICRATE ;
CONS_Printf ( M_GetText ( " timed %u gametics in %d realtics - %u frames \n %f seconds, %f avg fps \n " ) ,
leveltime , demotime , ( UINT32 ) framecount , f1 / TICRATE , f2 / f1 ) ;
// CSV-readable timedemo results, for external parsing
if ( timedemo_csv )
{
FILE * f ;
const char * csvpath = va ( " %s " PATHSEP " %s " , srb2home , " timedemo.csv " ) ;
const char * header = " id,demoname,seconds,avgfps,leveltime,demotime,framecount,ticrate,rendermode,vidmode,vidwidth,vidheight,procbits \n " ;
const char * rowformat = " \" %s \" , \" %s \" ,%f,%f,%u,%d,%u,%u,%u,%u,%u,%u,%u \n " ;
boolean headerrow = ! FIL_FileExists ( csvpath ) ;
UINT8 procbits = 0 ;
// Bitness
if ( sizeof ( void * ) = = 4 )
procbits = 32 ;
else if ( sizeof ( void * ) = = 8 )
procbits = 64 ;
f = fopen ( csvpath , " a+ " ) ;
if ( f )
{
if ( headerrow )
fputs ( header , f ) ;
fprintf ( f , rowformat ,
timedemo_csv_id , timedemo_name , f1 / TICRATE , f2 / f1 , leveltime , demotime , ( UINT32 ) framecount , TICRATE , rendermode , vid . modenum , vid . width , vid . height , procbits ) ;
fclose ( f ) ;
CONS_Printf ( " Timedemo results saved to '%s' \n " , csvpath ) ;
}
else
{
// Just print the CSV output to console
CON_LogMessage ( header ) ;
CONS_Printf ( rowformat ,
timedemo_csv_id , timedemo_name , f1 / TICRATE , f2 / f1 , leveltime , demotime , ( UINT32 ) framecount , TICRATE , rendermode , vid . modenum , vid . width , vid . height , procbits ) ;
}
}
if ( restorecv_vidwait ! = cv_vidwait . value )
CV_SetValue ( & cv_vidwait , restorecv_vidwait ) ;
D_AdvanceDemo ( ) ;
}
2020-03-19 20:09:55 +00:00
// reset engine variable set for the demos
// called from stopdemo command, map command, and g_checkdemoStatus.
void G_StopDemo ( void )
{
Z_Free ( demobuffer ) ;
demobuffer = NULL ;
demoplayback = false ;
titledemo = false ;
timingdemo = false ;
singletics = false ;
if ( gamestate = = GS_INTERMISSION )
Y_EndIntermission ( ) ; // cleanup
G_SetGamestate ( GS_NULL ) ;
wipegamestate = GS_NULL ;
SV_StopServer ( ) ;
SV_ResetServer ( ) ;
}
boolean G_CheckDemoStatus ( void )
{
2020-03-19 20:42:51 +00:00
G_FreeGhosts ( ) ;
2020-03-19 20:09:55 +00:00
// DO NOT end metal sonic demos here
if ( timingdemo )
{
2020-05-14 19:57:21 +00:00
G_StopTimingDemo ( ) ;
2020-03-19 20:09:55 +00:00
return true ;
}
if ( demoplayback )
{
if ( singledemo )
I_Quit ( ) ;
G_StopDemo ( ) ;
if ( modeattacking )
M_EndModeAttackRun ( ) ;
else
D_AdvanceDemo ( ) ;
return true ;
}
if ( demorecording )
{
2020-05-14 19:57:21 +00:00
G_StopDemoRecording ( ) ;
2020-03-19 20:09:55 +00:00
return true ;
}
return false ;
}
2022-02-14 13:57:00 +00:00
// 2.2.10 shifted some frame flags around, this function converts frame flags from older versions to their 2.2.10 equivalents.
INT32 G_ConvertOldFrameFlags ( INT32 frame )
{
if ( frame & 0x01000000 ) // was FF_ANIMATE, is now FF_VERTICALFLIP
{
frame & = ~ 0x01000000 ;
frame | = FF_ANIMATE ;
}
if ( frame & 0x02000000 ) // was FF_RANDOMANIM, is now FF_HORIZONTALFLIP
{
frame & = ~ 0x02000000 ;
frame | = FF_RANDOMANIM ;
}
if ( frame & 0x04000000 ) // was FF_GLOBALANIM, is now empty
{
frame & = ~ 0x04000000 ;
frame | = FF_GLOBALANIM ;
}
if ( frame & 0x00200000 ) // was FF_VERTICALFLIP, is now FF_FULLDARK
{
frame & = ~ 0x00200000 ;
frame | = FF_VERTICALFLIP ;
}
if ( frame & 0x00400000 ) // was FF_HORIZONTALFLIP, is now FF_PAPERSPRITE
{
frame & = ~ 0x00400000 ;
frame | = FF_HORIZONTALFLIP ;
}
if ( frame & 0x00800000 ) // was FF_PAPERSPRITE, is now FF_FLOORSPRITE
{
frame & = ~ 0x00800000 ;
frame | = FF_PAPERSPRITE ;
}
return frame ;
}