quakeforge/tools/qfcc/test/test-harness.c
Bill Currie 4b7ecdf74a Make PR_Init take an instance to initialize
This allows internal sub-systems to do per-instance initializations
without other engine systems having to know about them.
2020-02-23 18:32:16 +09:00

274 lines
5.8 KiB
C

/*
test-harness.c
Program for testing qfcc generated code.
Copyright (C) 2012 Bill Currie <bill@taniwha.org>
Author: Bill Currie <bill@taniwha.org>
Date: 2012/11/22
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to:
Free Software Foundation, Inc.
59 Temple Place - Suite 330
Boston, MA 02111-1307, USA
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include <getopt.h>
#include <stdlib.h>
#include <QF/cmd.h>
#include <QF/cvar.h>
#include <QF/progs.h>
#include <QF/quakefs.h>
#include "QF/ruamoko.h"
#include <QF/sys.h>
#include "QF/va.h"
#include <QF/zone.h>
#include "test-bi.h"
#define MAX_EDICTS 64 // 64 edicts should be enough for testing
#define MAX_HEAP 1024*1024 // 1MB should be enough for testing
// keep me sane when adding long options :P
enum {
start_opts = 255, // not used, starts the enum.
OPT_DEVELOPER,
OPT_FLOAT,
};
static struct option const long_options[] = {
{"developer", required_argument, 0, OPT_DEVELOPER},
{"float", no_argument, 0, OPT_FLOAT},
{"trace", no_argument, 0, 't'},
{"help", no_argument, 0, 'h'},
{"version", no_argument, 0, 'V'},
};
static const char *short_options =
"+" // magic option parsing mode doohicky (must come first)
"h" // help
"t" // tracing
"V" // version
;
static edict_t *edicts;
static int num_edicts;
static int reserved_edicts;
static progs_t pr;
static const char *this_program;
static struct {
int developer;
int trace;
int flote;
} options;
static QFile *
open_file (const char *path, int *len)
{
QFile *file = Qopen (path, "rbz");
if (!file) {
perror (path);
return 0;
}
*len = Qfilesize (file);
return file;
}
static void *
load_file (progs_t *pr, const char *name, off_t *_size)
{
QFile *file;
int size;
char *sym;
file = open_file (name, &size);
if (!file) {
file = open_file (va ("%s.gz", name), &size);
if (!file) {
return 0;
}
}
sym = malloc (size + 1);
sym[size] = 0;
Qread (file, sym, size);
*_size = size;
return sym;
}
static void *
allocate_progs_mem (progs_t *pr, int size)
{
return malloc (size);
}
static void
free_progs_mem (progs_t *pr, void *mem)
{
free (mem);
}
static void
init_qf (void)
{
Sys_Init ();
Cvar_Get ("developer", va ("%d", options.developer), 0, 0, 0);
Memory_Init (malloc (1024 * 1024), 1024 * 1024);
cvar_t *debug = Cvar_Get ("pr_debug", "2", 0, 0, 0);
Cvar_Get ("pr_boundscheck", "2", 0, 0, 0);
if (options.trace > 1) {
Cvar_SetValue (debug, 4);
}
pr.edicts = &edicts;
pr.num_edicts = &num_edicts;
pr.reserved_edicts = &reserved_edicts;
pr.load_file = load_file;
pr.allocate_progs_mem = allocate_progs_mem;
pr.free_progs_mem = free_progs_mem;
pr.no_exec_limit = 0; // absolutely want a limit!
pr.pr_trace_depth = -1;
pr.pr_trace = options.trace;
PR_Init_Cvars ();
PR_Init (&pr);
RUA_Init (&pr, 0);
PR_Cmds_Init(&pr);
BI_Init (&pr);
}
static int
load_progs (const char *name)
{
QFile *file;
int size;
file = open_file (name, &size);
if (!file) {
return 0;
}
pr.progs_name = name;
pr.max_edicts = 16;
pr.zone_size = 1024 * 1024;
pr.stack_size = 64 * 1024;
PR_LoadProgsFile (&pr, file, size);
Qclose (file);
if (!PR_RunLoadFuncs (&pr))
PR_Error (&pr, "unable to load %s", pr.progs_name);
return 1;
}
static void
usage (int exitval)
{
printf ("%s - QuakeForge VM test harness\n", this_program);
printf ("Usage: %s [options] [progs.dat [progs options]]", this_program);
printf (
"Options:\n"
" --developer FLAGS Set the developer cvar to FLAGS.\n"
" --float main () returns float instead of int.\n"
" -h, --help Display this help and exit\n"
" -t, --trace Set the trace flag in the VM.\n"
" -V, --version Output version information and exit\n"
);
exit (exitval);
}
static int
parse_options (int argc, char **argv)
{
int c;
this_program = argv[0];
while ((c = getopt_long (argc, argv, short_options, long_options, 0))
!= -1) {
switch (c) {
case OPT_DEVELOPER:
options.developer = atoi (optarg);
break;
case OPT_FLOAT:
options.flote = 1;
break;
case 't':
options.trace++;
break;
case 'h':
usage (0);
break;
case 'V':
printf ("%s version %s\n", PACKAGE_NAME, PACKAGE_VERSION);
exit (0);
default:
usage (1);
}
}
return optind;
}
int
main (int argc, char **argv)
{
dfunction_t *dfunc;
func_t main_func = 0;
const char *name = "progs.dat";
string_t *pr_argv;
int pr_argc = 1, i;
i = parse_options (argc, argv);
argc -= i;
argv += i;
init_qf ();
if (argc > 0)
name = argv[0];
if (!load_progs (name))
Sys_Error ("couldn't load %s", name);
PR_PushFrame (&pr);
if (argc > 2)
pr_argc = argc;
pr_argv = PR_Zone_Malloc (&pr, (pr_argc + 1) * 4);
pr_argv[0] = PR_SetTempString (&pr, name);
for (i = 1; i < pr_argc; i++)
pr_argv[i] = PR_SetTempString (&pr, argv[i]);
pr_argv[i] = 0;
if ((dfunc = PR_FindFunction (&pr, ".main"))
|| (dfunc = PR_FindFunction (&pr, "main")))
main_func = dfunc - pr.pr_functions;
else
PR_Undefined (&pr, "function", "main");
PR_RESET_PARAMS (&pr);
P_INT (&pr, 0) = pr_argc;
P_POINTER (&pr, 1) = PR_SetPointer (&pr, pr_argv);
PR_ExecuteProgram (&pr, main_func);
PR_PopFrame (&pr);
if (options.flote)
return R_FLOAT (&pr);
return R_INT (&pr);
}