dxx-rebirth/similar/misc/args.cpp
Kp 07869d9dba Add automatic demo recording
Various users want automatic demo recording.  Add new command line options:
	-auto-record-demo	start recording on level entry
	-record-demo-format	set demo name automatically

For -record-demo-format, specify a template containing any mix of strftime
insertions, variable insertions, and literal text.  As a special case, specify
"." to use the builtin template "%Y%m%d.%H%M%S-$p-$m".  Most users should use
".".  Supported variables:
	$p	name of pilot recording the demo
	$m	name of msn/mn2 file played

Suggested by jeffersoncarpenter in https://github.com/dxx-rebirth/dxx-rebirth/pull/33, but implemented by vLKp.
2015-02-28 19:36:02 +00:00

354 lines
9.5 KiB
C++

/*
* This file is part of the DXX-Rebirth project <http://www.dxx-rebirth.com/>.
* It is copyright by its individual contributors, as recorded in the
* project's Git history. See COPYING.txt at the top level for license
* terms and a link to the Git history.
*/
/*
*
* Functions for accessing arguments.
*
*/
#include <string>
#include <vector>
#include <stdlib.h>
#include <string.h>
#include <SDL_stdinc.h>
#include "physfsx.h"
#include "args.h"
#include "u_mem.h"
#include "strutil.h"
#include "digi.h"
#include "game.h"
#include "gauges.h"
#include "console.h"
#ifdef USE_UDP
#include "net_udp.h"
#endif
#define MAX_ARGS 1000
#if defined(DXX_BUILD_DESCENT_I)
#define INI_FILENAME "d1x.ini"
#elif defined(DXX_BUILD_DESCENT_II)
#define INI_FILENAME "d2x.ini"
#endif
typedef std::vector<std::string> Arglist;
static Arglist Args;
class argument_exception
{
public:
const char *arg;
argument_exception(const char *a) : arg(a) {}
};
class missing_parameter : public argument_exception
{
public:
missing_parameter(const char *a) : argument_exception(a) {}
};
class unhandled_argument : public argument_exception
{
public:
unhandled_argument(const char *a) : argument_exception(a) {}
};
class conversion_failure : public argument_exception
{
public:
const char *value;
conversion_failure(const char *a, const char *v) : argument_exception(a), value(v) {}
};
struct Arg GameArg;
static void AppendIniArgs(void)
{
if (auto f = PHYSFSX_openReadBuffered(INI_FILENAME))
{
PHYSFSX_gets_line_t<1024> line;
while(!PHYSFS_eof(f) && Args.size() < MAX_ARGS && PHYSFSX_fgets(line, f))
{
static const char separator[] = " ";
for(char *token = strtok(line, separator); token != NULL; token = strtok(NULL, separator))
{
if (*token == ';')
break;
Args.push_back(token);
}
}
}
}
static const char *arg_string(Arglist::const_iterator &pp, Arglist::const_iterator end)
{
Arglist::const_iterator arg = pp;
if (++pp == end)
throw missing_parameter(arg->c_str());
return pp->c_str();
}
static long arg_integer(Arglist::const_iterator &pp, Arglist::const_iterator end)
{
Arglist::const_iterator arg = pp;
const char *value = arg_string(pp, end);
char *p;
long i = strtol(value, &p, 10);
if (*p)
throw conversion_failure(arg->c_str(), value);
return i;
}
static void arg_port_number(Arglist::const_iterator &pp, Arglist::const_iterator end, uint16_t &out, bool allow_privileged)
{
auto port = arg_integer(pp, end);
if (static_cast<uint16_t>(port) == port && (allow_privileged || port >= 1024))
out = port;
}
static void ReadCmdArgs(void)
{
GameArg.SysMaxFPS = MAXIMUM_FPS;
#if defined(DXX_BUILD_DESCENT_II)
GameArg.SndDigiSampleRate = SAMPLE_RATE_22K;
#endif
#ifdef USE_UDP
GameArg.MplUdpHostAddr = UDP_MANUAL_ADDR_DEFAULT;
#ifdef USE_TRACKER
GameArg.MplTrackerAddr = TRACKER_ADDR_DEFAULT;
GameArg.MplTrackerPort = TRACKER_PORT_DEFAULT;
#endif
#endif
GameArg.DbgVerbose = CON_NORMAL;
GameArg.DbgBpp = 32;
#ifdef OGL
GameArg.DbgGlIntensity4Ok = 1;
GameArg.DbgGlLuminance4Alpha4Ok = 1;
GameArg.DbgGlRGBA2Ok = 1;
GameArg.DbgGlReadPixelsOk = 1;
GameArg.DbgGlGetTexLevelParamOk = 1;
#endif
for (Arglist::const_iterator pp = Args.begin(), end = Args.end(); pp != end; ++pp)
{
const char *p = pp->c_str();
// System Options
if (!d_stricmp(p, "-help") || !d_stricmp(p, "-h") || !d_stricmp(p, "-?") || !d_stricmp(p, "?"))
GameArg.SysShowCmdHelp = 1;
else if (!d_stricmp(p, "-nonicefps"))
GameArg.SysNoNiceFPS = 1;
else if (!d_stricmp(p, "-maxfps"))
GameArg.SysMaxFPS = arg_integer(pp, end);
else if (!d_stricmp(p, "-hogdir"))
GameArg.SysHogDir = arg_string(pp, end);
else if (!d_stricmp(p, "-nohogdir"))
GameArg.SysNoHogDir = 1;
else if (!d_stricmp(p, "-use_players_dir"))
GameArg.SysUsePlayersDir = 1;
else if (!d_stricmp(p, "-lowmem"))
GameArg.SysLowMem = 1;
else if (!d_stricmp(p, "-pilot"))
GameArg.SysPilot = arg_string(pp, end);
else if (!d_stricmp(p, "-record-demo-format"))
GameArg.SysRecordDemoNameTemplate = arg_string(pp, end);
else if (!d_stricmp(p, "-auto-record-demo"))
GameArg.SysAutoRecordDemo = 1;
else if (!d_stricmp(p, "-window"))
GameArg.SysWindow = 1;
else if (!d_stricmp(p, "-noborders"))
GameArg.SysNoBorders = 1;
#if defined(DXX_BUILD_DESCENT_I)
else if (!d_stricmp(p, "-notitles"))
GameArg.SysNoTitles = 1;
#elif defined(DXX_BUILD_DESCENT_II)
else if (!d_stricmp(p, "-nomovies"))
GameArg.SysNoMovies = 1;
#endif
else if (!d_stricmp(p, "-autodemo"))
GameArg.SysAutoDemo = 1;
// Control Options
else if (!d_stricmp(p, "-nocursor"))
GameArg.CtlNoCursor = 1;
else if (!d_stricmp(p, "-nomouse"))
GameArg.CtlNoMouse = 1;
else if (!d_stricmp(p, "-nojoystick"))
GameArg.CtlNoJoystick = 1;
else if (!d_stricmp(p, "-nostickykeys"))
GameArg.CtlNoStickyKeys = 1;
// Sound Options
else if (!d_stricmp(p, "-nosound"))
GameArg.SndNoSound = 1;
else if (!d_stricmp(p, "-nomusic"))
GameArg.SndNoMusic = 1;
#if defined(DXX_BUILD_DESCENT_II)
else if (!d_stricmp(p, "-sound11k"))
GameArg.SndDigiSampleRate = SAMPLE_RATE_11K;
#endif
else if (!d_stricmp(p, "-nosdlmixer"))
{
#ifdef USE_SDLMIXER
GameArg.SndDisableSdlMixer = true;
#endif
}
// Graphics Options
else if (!d_stricmp(p, "-lowresfont"))
GameArg.GfxSkipHiresFNT = 1;
#if defined(DXX_BUILD_DESCENT_II)
else if (!d_stricmp(p, "-lowresgraphics"))
GameArg.GfxSkipHiresGFX = 1;
else if (!d_stricmp(p, "-lowresmovies"))
GameArg.GfxSkipHiresMovie = 1;
#endif
#ifdef OGL
// OpenGL Options
else if (!d_stricmp(p, "-gl_fixedfont"))
GameArg.OglFixedFont = 1;
#endif
// Multiplayer Options
#ifdef USE_UDP
else if (!d_stricmp(p, "-udp_hostaddr"))
GameArg.MplUdpHostAddr = arg_string(pp, end);
else if (!d_stricmp(p, "-udp_hostport"))
/* Peers use -udp_myport to change, so peer cannot set a
* privileged port.
*/
arg_port_number(pp, end, GameArg.MplUdpHostPort, false);
else if (!d_stricmp(p, "-udp_myport"))
{
arg_port_number(pp, end, GameArg.MplUdpMyPort, false);
}
else if (!d_stricmp(p, "-no-tracker"))
{
/* Always recognized. No-op if tracker support compiled
* out. */
#ifdef USE_TRACKER
GameArg.MplTrackerAddr = nullptr;
#endif
}
#ifdef USE_TRACKER
else if (!d_stricmp(p, "-tracker_hostaddr"))
{
GameArg.MplTrackerAddr = arg_string(pp, end);
if (!*GameArg.MplTrackerAddr)
GameArg.MplTrackerAddr = nullptr;
}
else if (!d_stricmp(p, "-tracker_hostport"))
arg_port_number(pp, end, GameArg.MplTrackerPort, true);
#endif
#endif
#if defined(DXX_BUILD_DESCENT_I)
else if (!d_stricmp(p, "-nobm"))
GameArg.EdiNoBm = 1;
#elif defined(DXX_BUILD_DESCENT_II)
#ifdef EDITOR
// Editor Options
else if (!d_stricmp(p, "-autoload"))
GameArg.EdiAutoLoad = arg_string(pp, end);
else if (!d_stricmp(p, "-macdata"))
GameArg.EdiMacData = 1;
else if (!d_stricmp(p, "-hoarddata"))
GameArg.EdiSaveHoardData = 1;
#endif
#endif
// Debug Options
else if (!d_stricmp(p, "-debug"))
GameArg.DbgVerbose = CON_DEBUG;
else if (!d_stricmp(p, "-verbose"))
GameArg.DbgVerbose = CON_VERBOSE;
else if (!d_stricmp(p, "-no-grab"))
GameArg.DbgForbidConsoleGrab = 1;
else if (!d_stricmp(p, "-safelog"))
GameArg.DbgSafelog = 1;
else if (!d_stricmp(p, "-norun"))
GameArg.DbgNoRun = 1;
else if (!d_stricmp(p, "-renderstats"))
GameArg.DbgRenderStats = 1;
else if (!d_stricmp(p, "-text"))
GameArg.DbgAltTex = arg_string(pp, end);
else if (!d_stricmp(p, "-tmap"))
GameArg.DbgTexMap = arg_string(pp, end);
else if (!d_stricmp(p, "-showmeminfo"))
GameArg.DbgShowMemInfo = 1;
else if (!d_stricmp(p, "-nodoublebuffer"))
GameArg.DbgNoDoubleBuffer = 1;
else if (!d_stricmp(p, "-bigpig"))
GameArg.DbgNoCompressPigBitmap = 1;
else if (!d_stricmp(p, "-16bpp"))
GameArg.DbgBpp = 16;
#ifdef OGL
else if (!d_stricmp(p, "-gl_oldtexmerge"))
GameArg.DbgUseOldTextureMerge = 1;
else if (!d_stricmp(p, "-gl_intensity4_ok"))
GameArg.DbgGlIntensity4Ok = arg_integer(pp, end);
else if (!d_stricmp(p, "-gl_luminance4_alpha4_ok"))
GameArg.DbgGlLuminance4Alpha4Ok = arg_integer(pp, end);
else if (!d_stricmp(p, "-gl_rgba2_ok"))
GameArg.DbgGlRGBA2Ok = arg_integer(pp, end);
else if (!d_stricmp(p, "-gl_readpixels_ok"))
GameArg.DbgGlReadPixelsOk = arg_integer(pp, end);
else if (!d_stricmp(p, "-gl_gettexlevelparam_ok"))
GameArg.DbgGlGetTexLevelParamOk = arg_integer(pp, end);
#else
else if (!d_stricmp(p, "-hwsurface"))
GameArg.DbgSdlHWSurface = 1;
else if (!d_stricmp(p, "-asyncblit"))
GameArg.DbgSdlASyncBlit = 1;
#endif
else
throw unhandled_argument(p);
}
if (GameArg.SysMaxFPS < MINIMUM_FPS)
GameArg.SysMaxFPS = MINIMUM_FPS;
else if (GameArg.SysMaxFPS > MAXIMUM_FPS)
GameArg.SysMaxFPS = MAXIMUM_FPS;
static char sdl_disable_lock_keys[] = "SDL_DISABLE_LOCK_KEYS=0";
if (GameArg.CtlNoStickyKeys) // Must happen before SDL_Init!
sdl_disable_lock_keys[sizeof(sdl_disable_lock_keys) - 1] = '1';
SDL_putenv(sdl_disable_lock_keys);
}
void args_exit(void)
{
Args.clear();
}
bool InitArgs( int argc,char **argv )
{
for (int i=1; i < argc; i++ )
Args.push_back(argv[i]);
AppendIniArgs();
try {
ReadCmdArgs();
return true;
} catch(const missing_parameter& e) {
Warning("Missing parameter for argument \"%s\"", e.arg);
} catch(const unhandled_argument& e) {
Warning("Unhandled argument \"%s\"", e.arg);
} catch(const conversion_failure& e) {
Warning("Failed to convert parameter \"%s\" for argument \"%s\"", e.value, e.arg);
}
return false;
}