mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2025-01-23 09:20:40 +00:00
c9f1d770e0
That was a mess
302 lines
6.7 KiB
C
302 lines
6.7 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 <string.h>
|
|
#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
|
|
#define MAX_STACK 64*1024 // 64kB 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 test_edicts[MAX_EDICTS];
|
|
|
|
static edict_t *edicts;
|
|
static int num_edicts;
|
|
static int reserved_edicts;
|
|
static progs_t test_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 (0, "%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 int
|
|
init_edicts (progs_t *pr)
|
|
{
|
|
memset (test_edicts, 0, sizeof (test_edicts));
|
|
|
|
// init the data field of the edicts
|
|
for (int i = 0; i < MAX_EDICTS; i++) {
|
|
edict_t *ent = EDICT_NUM (&test_pr, i);
|
|
ent->pr = &test_pr;
|
|
ent->entnum = i;
|
|
ent->edict = EDICT_TO_PROG (&test_pr, ent);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
init_qf (void)
|
|
{
|
|
Sys_Init ();
|
|
Cvar_Get ("developer", va (0, "%d", options.developer), 0, 0, 0);
|
|
|
|
Memory_Init (Sys_Alloc (1024 * 1024), 1024 * 1024);
|
|
|
|
cvar_t *debug = Cvar_Get ("pr_debug", "2", 0, 0, 0);
|
|
Cvar_Get ("pr_boundscheck", "2", 0, 0, 0);
|
|
Cvar_Get ("pr_deadbeef_locals", "1", 0, 0, 0);
|
|
|
|
if (options.trace > 1) {
|
|
Cvar_SetValue (debug, 4);
|
|
}
|
|
|
|
test_pr.pr_edicts = &edicts;
|
|
test_pr.num_edicts = &num_edicts;
|
|
test_pr.reserved_edicts = &reserved_edicts;
|
|
test_pr.load_file = load_file;
|
|
test_pr.allocate_progs_mem = allocate_progs_mem;
|
|
test_pr.free_progs_mem = free_progs_mem;
|
|
test_pr.no_exec_limit = 0; // absolutely want a limit!
|
|
|
|
PR_Init_Cvars ();
|
|
|
|
PR_AddLoadFunc (&test_pr, init_edicts);
|
|
PR_Init (&test_pr);
|
|
|
|
RUA_Init (&test_pr, 0);
|
|
PR_Cmds_Init(&test_pr);
|
|
BI_Init (&test_pr);
|
|
}
|
|
|
|
static int
|
|
load_progs (const char *name)
|
|
{
|
|
QFile *file;
|
|
int size;
|
|
|
|
file = open_file (name, &size);
|
|
if (!file) {
|
|
return 0;
|
|
}
|
|
test_pr.progs_name = name;
|
|
test_pr.max_edicts = MAX_EDICTS;
|
|
test_pr.zone_size = MAX_HEAP;
|
|
test_pr.stack_size = MAX_STACK;
|
|
edicts = test_edicts;
|
|
|
|
PR_LoadProgsFile (&test_pr, file, size);
|
|
Qclose (file);
|
|
if (!PR_RunLoadFuncs (&test_pr))
|
|
PR_Error (&test_pr, "unable to load %s", test_pr.progs_name);
|
|
if (!PR_RunPostLoadFuncs (&test_pr))
|
|
PR_Error (&test_pr, "unable to load %s", test_pr.progs_name);
|
|
test_pr.pr_trace_depth = -1;
|
|
test_pr.pr_trace = options.trace;
|
|
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 (&test_pr);
|
|
if (argc > 2)
|
|
pr_argc = argc;
|
|
pr_argv = PR_Zone_Malloc (&test_pr, (pr_argc + 1) * 4);
|
|
pr_argv[0] = PR_SetTempString (&test_pr, name);
|
|
for (i = 1; i < pr_argc; i++)
|
|
pr_argv[i] = PR_SetTempString (&test_pr, argv[i]);
|
|
pr_argv[i] = 0;
|
|
|
|
if ((dfunc = PR_FindFunction (&test_pr, ".main"))
|
|
|| (dfunc = PR_FindFunction (&test_pr, "main")))
|
|
main_func = dfunc - test_pr.pr_functions;
|
|
else
|
|
PR_Undefined (&test_pr, "function", "main");
|
|
PR_RESET_PARAMS (&test_pr);
|
|
P_INT (&test_pr, 0) = pr_argc;
|
|
P_POINTER (&test_pr, 1) = PR_SetPointer (&test_pr, pr_argv);
|
|
test_pr.pr_argc = 2;
|
|
PR_ExecuteProgram (&test_pr, main_func);
|
|
PR_PopFrame (&test_pr);
|
|
if (options.flote)
|
|
return R_FLOAT (&test_pr);
|
|
return R_INT (&test_pr);
|
|
}
|