mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2025-01-18 23:11:38 +00:00
Add a stack and push and pop instructions.
The stack is for data, not return addresses.
This commit is contained in:
parent
9501d30a8f
commit
364c2d2c2b
6 changed files with 785 additions and 21 deletions
|
@ -295,6 +295,66 @@ typedef enum {
|
||||||
OP_RCALL8,
|
OP_RCALL8,
|
||||||
|
|
||||||
OP_RETURN_V,
|
OP_RETURN_V,
|
||||||
|
|
||||||
|
OP_PUSH_S,
|
||||||
|
OP_PUSH_F,
|
||||||
|
OP_PUSH_V,
|
||||||
|
OP_PUSH_ENT,
|
||||||
|
OP_PUSH_FLD,
|
||||||
|
OP_PUSH_FN,
|
||||||
|
OP_PUSH_P,
|
||||||
|
OP_PUSH_Q,
|
||||||
|
OP_PUSH_I,
|
||||||
|
|
||||||
|
OP_PUSHB_S,
|
||||||
|
OP_PUSHB_F,
|
||||||
|
OP_PUSHB_V,
|
||||||
|
OP_PUSHB_ENT,
|
||||||
|
OP_PUSHB_FLD,
|
||||||
|
OP_PUSHB_FN,
|
||||||
|
OP_PUSHB_P,
|
||||||
|
OP_PUSHB_Q,
|
||||||
|
OP_PUSHB_I,
|
||||||
|
|
||||||
|
OP_PUSHBI_S,
|
||||||
|
OP_PUSHBI_F,
|
||||||
|
OP_PUSHBI_V,
|
||||||
|
OP_PUSHBI_ENT,
|
||||||
|
OP_PUSHBI_FLD,
|
||||||
|
OP_PUSHBI_FN,
|
||||||
|
OP_PUSHBI_P,
|
||||||
|
OP_PUSHBI_Q,
|
||||||
|
OP_PUSHBI_I,
|
||||||
|
|
||||||
|
OP_POP_S,
|
||||||
|
OP_POP_F,
|
||||||
|
OP_POP_V,
|
||||||
|
OP_POP_ENT,
|
||||||
|
OP_POP_FLD,
|
||||||
|
OP_POP_FN,
|
||||||
|
OP_POP_P,
|
||||||
|
OP_POP_Q,
|
||||||
|
OP_POP_I,
|
||||||
|
|
||||||
|
OP_POPB_S,
|
||||||
|
OP_POPB_F,
|
||||||
|
OP_POPB_V,
|
||||||
|
OP_POPB_ENT,
|
||||||
|
OP_POPB_FLD,
|
||||||
|
OP_POPB_FN,
|
||||||
|
OP_POPB_P,
|
||||||
|
OP_POPB_Q,
|
||||||
|
OP_POPB_I,
|
||||||
|
|
||||||
|
OP_POPBI_S,
|
||||||
|
OP_POPBI_F,
|
||||||
|
OP_POPBI_V,
|
||||||
|
OP_POPBI_ENT,
|
||||||
|
OP_POPBI_FLD,
|
||||||
|
OP_POPBI_FN,
|
||||||
|
OP_POPBI_P,
|
||||||
|
OP_POPBI_Q,
|
||||||
|
OP_POPBI_I,
|
||||||
} pr_opcode_e;
|
} pr_opcode_e;
|
||||||
|
|
||||||
typedef struct opcode_s {
|
typedef struct opcode_s {
|
||||||
|
|
|
@ -1612,6 +1612,19 @@ struct progs_s {
|
||||||
prstack_t pr_stack[MAX_STACK_DEPTH];
|
prstack_t pr_stack[MAX_STACK_DEPTH];
|
||||||
int pr_depth;
|
int pr_depth;
|
||||||
|
|
||||||
|
/// \name progs visible stack
|
||||||
|
/// Usable by the progs for any purpose. Will not be accessible unless
|
||||||
|
/// a .stack global is found. Space is allocated from the top of the stack
|
||||||
|
/// (as is common for hardware). The push and pop instructions will not
|
||||||
|
/// be considered valid if there is no .stack global.
|
||||||
|
/// \note The return address and saved locals will not ever be on this
|
||||||
|
/// stack.
|
||||||
|
//@{
|
||||||
|
pr_type_t *stack;
|
||||||
|
pointer_t stack_bottom;
|
||||||
|
int stack_size; ///< set by user
|
||||||
|
//@}
|
||||||
|
|
||||||
int localstack[LOCALSTACK_SIZE];
|
int localstack[LOCALSTACK_SIZE];
|
||||||
int localstack_used;
|
int localstack_used;
|
||||||
//@}
|
//@}
|
||||||
|
@ -1656,6 +1669,7 @@ struct progs_s {
|
||||||
struct {
|
struct {
|
||||||
float *time; ///< required for OP_STATE
|
float *time; ///< required for OP_STATE
|
||||||
pr_int_t *self; ///< required for OP_STATE
|
pr_int_t *self; ///< required for OP_STATE
|
||||||
|
pointer_t *stack; ///< required for OP_(PUSH|POP)*
|
||||||
} globals;
|
} globals;
|
||||||
struct {
|
struct {
|
||||||
pr_int_t nextthink; ///< required for OP_STATE
|
pr_int_t nextthink; ///< required for OP_STATE
|
||||||
|
|
|
@ -358,6 +358,17 @@ PR_CallFunction (progs_t *pr, func_t fnum)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
check_stack_pointer (progs_t *pr, pointer_t stack, int size)
|
||||||
|
{
|
||||||
|
if (stack < pr->stack_bottom) {
|
||||||
|
PR_RunError (pr, "Progs stack overflow");
|
||||||
|
}
|
||||||
|
if (stack > pr->globals_size - size) {
|
||||||
|
PR_RunError (pr, "Progs stack underflow");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
PR_ExecuteProgram
|
PR_ExecuteProgram
|
||||||
|
|
||||||
|
@ -859,6 +870,318 @@ PR_ExecuteProgram (progs_t * pr, func_t fnum)
|
||||||
QuatCopy (OPA.quat_var, ptr->quat_var);
|
QuatCopy (OPA.quat_var, ptr->quat_var);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case OP_PUSH_F:
|
||||||
|
case OP_PUSH_FLD:
|
||||||
|
case OP_PUSH_ENT:
|
||||||
|
case OP_PUSH_S:
|
||||||
|
case OP_PUSH_FN:
|
||||||
|
case OP_PUSH_I:
|
||||||
|
case OP_PUSH_P:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack - 1;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 1);
|
||||||
|
}
|
||||||
|
stk->integer_var = OPA.integer_var;
|
||||||
|
*pr->globals.stack = stack;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_PUSH_V:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack - 3;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 3);
|
||||||
|
}
|
||||||
|
memcpy (stk, &OPA, 3 * sizeof (OPC));
|
||||||
|
*pr->globals.stack = stack;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_PUSH_Q:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack - 4;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 4);
|
||||||
|
}
|
||||||
|
memcpy (stk, &OPA, 4 * sizeof (OPC));
|
||||||
|
*pr->globals.stack = stack;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case OP_PUSHB_F:
|
||||||
|
case OP_PUSHB_S:
|
||||||
|
case OP_PUSHB_ENT:
|
||||||
|
case OP_PUSHB_FLD:
|
||||||
|
case OP_PUSHB_FN:
|
||||||
|
case OP_PUSHB_I:
|
||||||
|
case OP_PUSHB_P:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack - 1;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + OPB.integer_var;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 1);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_integer);
|
||||||
|
}
|
||||||
|
|
||||||
|
stk->integer_var = ptr->integer_var;
|
||||||
|
*pr->globals.stack = stack;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_PUSHB_V:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack - 3;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + OPB.integer_var;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 3);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_integer);
|
||||||
|
}
|
||||||
|
|
||||||
|
VectorCopy (ptr->vector_var, stk->vector_var);
|
||||||
|
*pr->globals.stack = stack;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_PUSHB_Q:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack - 4;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + OPB.integer_var;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 4);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_quat);
|
||||||
|
}
|
||||||
|
|
||||||
|
QuatCopy (ptr->quat_var, stk->quat_var);
|
||||||
|
*pr->globals.stack = stack;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case OP_PUSHBI_F:
|
||||||
|
case OP_PUSHBI_S:
|
||||||
|
case OP_PUSHBI_ENT:
|
||||||
|
case OP_PUSHBI_FLD:
|
||||||
|
case OP_PUSHBI_FN:
|
||||||
|
case OP_PUSHBI_I:
|
||||||
|
case OP_PUSHBI_P:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack - 1;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + st->b;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 1);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_integer);
|
||||||
|
}
|
||||||
|
|
||||||
|
stk->integer_var = ptr->integer_var;
|
||||||
|
*pr->globals.stack = stack;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_PUSHBI_V:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack - 3;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + st->b;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 3);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_integer);
|
||||||
|
}
|
||||||
|
|
||||||
|
VectorCopy (ptr->vector_var, stk->vector_var);
|
||||||
|
*pr->globals.stack = stack;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_PUSHBI_Q:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack - 4;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + st->b;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 4);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_quat);
|
||||||
|
}
|
||||||
|
|
||||||
|
QuatCopy (ptr->quat_var, stk->quat_var);
|
||||||
|
*pr->globals.stack = stack;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case OP_POP_F:
|
||||||
|
case OP_POP_FLD:
|
||||||
|
case OP_POP_ENT:
|
||||||
|
case OP_POP_S:
|
||||||
|
case OP_POP_FN:
|
||||||
|
case OP_POP_I:
|
||||||
|
case OP_POP_P:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 1);
|
||||||
|
}
|
||||||
|
stk->integer_var = OPA.integer_var;
|
||||||
|
*pr->globals.stack = stack + 1;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_POP_V:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 3);
|
||||||
|
}
|
||||||
|
memcpy (stk, &OPA, 3 * sizeof (OPC));
|
||||||
|
*pr->globals.stack = stack + 3;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_POP_Q:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 4);
|
||||||
|
}
|
||||||
|
memcpy (stk, &OPA, 4 * sizeof (OPC));
|
||||||
|
*pr->globals.stack = stack + 4;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case OP_POPB_F:
|
||||||
|
case OP_POPB_S:
|
||||||
|
case OP_POPB_ENT:
|
||||||
|
case OP_POPB_FLD:
|
||||||
|
case OP_POPB_FN:
|
||||||
|
case OP_POPB_I:
|
||||||
|
case OP_POPB_P:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + OPB.integer_var;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 1);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_integer);
|
||||||
|
}
|
||||||
|
|
||||||
|
stk->integer_var = ptr->integer_var;
|
||||||
|
*pr->globals.stack = stack + 1;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_POPB_V:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + OPB.integer_var;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 3);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_integer);
|
||||||
|
}
|
||||||
|
|
||||||
|
VectorCopy (ptr->vector_var, stk->vector_var);
|
||||||
|
*pr->globals.stack = stack + 3;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_POPB_Q:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + OPB.integer_var;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 4);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_quat);
|
||||||
|
}
|
||||||
|
|
||||||
|
QuatCopy (ptr->quat_var, stk->quat_var);
|
||||||
|
*pr->globals.stack = stack + 4;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case OP_POPBI_F:
|
||||||
|
case OP_POPBI_S:
|
||||||
|
case OP_POPBI_ENT:
|
||||||
|
case OP_POPBI_FLD:
|
||||||
|
case OP_POPBI_FN:
|
||||||
|
case OP_POPBI_I:
|
||||||
|
case OP_POPBI_P:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + st->b;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 1);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_integer);
|
||||||
|
}
|
||||||
|
|
||||||
|
stk->integer_var = ptr->integer_var;
|
||||||
|
*pr->globals.stack = stack + 1;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_POPBI_V:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + st->b;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 3);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_integer);
|
||||||
|
}
|
||||||
|
|
||||||
|
VectorCopy (ptr->vector_var, stk->vector_var);
|
||||||
|
*pr->globals.stack = stack + 3;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case OP_POPBI_Q:
|
||||||
|
{
|
||||||
|
pointer_t stack = *pr->globals.stack;
|
||||||
|
pr_type_t *stk = pr->pr_globals + stack;
|
||||||
|
|
||||||
|
pointer = OPA.integer_var + st->b;
|
||||||
|
ptr = pr->pr_globals + pointer;
|
||||||
|
|
||||||
|
if (pr_boundscheck->int_val) {
|
||||||
|
check_stack_pointer (pr, stack, 4);
|
||||||
|
PR_BoundsCheck (pr, pointer, ev_quat);
|
||||||
|
}
|
||||||
|
|
||||||
|
QuatCopy (ptr->quat_var, stk->quat_var);
|
||||||
|
*pr->globals.stack = stack + 4;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
// ==================
|
// ==================
|
||||||
case OP_IFNOT:
|
case OP_IFNOT:
|
||||||
if (!OPA.integer_var) {
|
if (!OPA.integer_var) {
|
||||||
|
|
|
@ -97,6 +97,17 @@ free_progs_mem (progs_t *pr, void *mem)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
align_size (int size)
|
||||||
|
{
|
||||||
|
// round off to next highest whole word address (esp for Alpha)
|
||||||
|
// this ensures that pointers in the engine data area are always
|
||||||
|
// properly aligned
|
||||||
|
size += sizeof (void*) - 1;
|
||||||
|
size &= ~(sizeof (void*) - 1);
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
VISIBLE void
|
VISIBLE void
|
||||||
PR_LoadProgsFile (progs_t *pr, QFile *file, int size)
|
PR_LoadProgsFile (progs_t *pr, QFile *file, int size)
|
||||||
{
|
{
|
||||||
|
@ -160,33 +171,27 @@ PR_LoadProgsFile (progs_t *pr, QFile *file, int size)
|
||||||
// size of progs themselves
|
// size of progs themselves
|
||||||
pr->progs_size = size + offset_tweak;
|
pr->progs_size = size + offset_tweak;
|
||||||
Sys_MaskPrintf (SYS_DEV, "Programs occupy %iK.\n", size / 1024);
|
Sys_MaskPrintf (SYS_DEV, "Programs occupy %iK.\n", size / 1024);
|
||||||
// round off to next highest whole word address (esp for Alpha)
|
|
||||||
// this ensures that pointers in the engine data area are always
|
|
||||||
// properly aligned
|
|
||||||
pr->progs_size += sizeof (void*) - 1;
|
|
||||||
pr->progs_size &= ~(sizeof (void*) - 1);
|
|
||||||
|
|
||||||
// round off to next highest whole word address (esp for Alpha)
|
pr->progs_size = align_size (pr->progs_size);
|
||||||
// this ensures that pointers in the engine data area are always
|
pr->zone_size = align_size (pr->zone_size);
|
||||||
// properly aligned
|
pr->stack_size = align_size (pr->stack_size);
|
||||||
pr->zone_size += sizeof (void*) - 1;
|
|
||||||
pr->zone_size &= ~(sizeof (void*) - 1);
|
|
||||||
|
|
||||||
// size of edict asked for by progs
|
// size of edict asked for by progs
|
||||||
pr->pr_edict_size = max (1, progs.entityfields) * 4;
|
pr->pr_edict_size = max (1, progs.entityfields) * 4;
|
||||||
// size of engine data
|
// size of engine data
|
||||||
pr->pr_edict_size += sizeof (edict_t);
|
pr->pr_edict_size += sizeof (edict_t);
|
||||||
// round off to next highest whole word address (esp for Alpha)
|
pr->pr_edict_size = align_size (pr->progs_size);
|
||||||
// this ensures that pointers in the engine data area are always
|
|
||||||
// properly aligned
|
|
||||||
pr->pr_edict_size += sizeof (void*) - 1;
|
|
||||||
pr->pr_edict_size &= ~(sizeof (void*) - 1);
|
|
||||||
pr->pr_edictareasize = pr->max_edicts * pr->pr_edict_size;
|
pr->pr_edictareasize = pr->max_edicts * pr->pr_edict_size;
|
||||||
|
|
||||||
mem_size = pr->progs_size + pr->zone_size + pr->pr_edictareasize;
|
mem_size = pr->progs_size + pr->zone_size + pr->pr_edictareasize
|
||||||
|
+ pr->stack_size;
|
||||||
|
// +1 for a nul terminator
|
||||||
pr->progs = pr->allocate_progs_mem (pr, mem_size + 1);
|
pr->progs = pr->allocate_progs_mem (pr, mem_size + 1);
|
||||||
if (!pr->progs)
|
if (!pr->progs)
|
||||||
return;
|
return;
|
||||||
|
// Place a nul at the end of progs memory to ensure any unterminated
|
||||||
|
// strings within progs memory don't run off the end.
|
||||||
((byte *) pr->progs)[mem_size] = 0;
|
((byte *) pr->progs)[mem_size] = 0;
|
||||||
|
|
||||||
memcpy (pr->progs, &progs, sizeof (progs));
|
memcpy (pr->progs, &progs, sizeof (progs));
|
||||||
|
@ -195,8 +200,10 @@ PR_LoadProgsFile (progs_t *pr, QFile *file, int size)
|
||||||
CRC_ProcessBlock (base, &pr->crc, size - sizeof (progs));
|
CRC_ProcessBlock (base, &pr->crc, size - sizeof (progs));
|
||||||
base -= sizeof (progs); // offsets are from file start
|
base -= sizeof (progs); // offsets are from file start
|
||||||
|
|
||||||
if (pr->edicts)
|
if (pr->edicts) {
|
||||||
*pr->edicts = (edict_t *)((byte *) pr->progs + pr->progs_size);
|
*pr->edicts = (edict_t *)((byte *) pr->progs + pr->progs_size);
|
||||||
|
}
|
||||||
|
|
||||||
if (pr->zone_size) {
|
if (pr->zone_size) {
|
||||||
//FIXME zone_size needs to be at least as big as memzone_t, but
|
//FIXME zone_size needs to be at least as big as memzone_t, but
|
||||||
//memzone_t is opaque so its size is unknown
|
//memzone_t is opaque so its size is unknown
|
||||||
|
@ -213,9 +220,11 @@ PR_LoadProgsFile (progs_t *pr, QFile *file, int size)
|
||||||
pr->pr_statements = (dstatement_t *) (base + pr->progs->ofs_statements);
|
pr->pr_statements = (dstatement_t *) (base + pr->progs->ofs_statements);
|
||||||
|
|
||||||
pr->pr_globals = (pr_type_t *) (base + pr->progs->ofs_globals);
|
pr->pr_globals = (pr_type_t *) (base + pr->progs->ofs_globals);
|
||||||
|
pr->stack = (pr_type_t *) ((byte *) pr->zone + pr->zone_size);
|
||||||
pr->globals_size = (pr_type_t*)((byte *) pr->zone + pr->zone_size)
|
pr->stack_bottom = pr->stack - pr->pr_globals;
|
||||||
|
pr->globals_size = (pr_type_t *) ((byte *) pr->stack + pr->stack_size)
|
||||||
- pr->pr_globals;
|
- pr->pr_globals;
|
||||||
|
|
||||||
if (pr->zone) {
|
if (pr->zone) {
|
||||||
PR_Zone_Init (pr);
|
PR_Zone_Init (pr);
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,6 +44,8 @@
|
||||||
#include "QF/progs.h"
|
#include "QF/progs.h"
|
||||||
#include "QF/sys.h"
|
#include "QF/sys.h"
|
||||||
|
|
||||||
|
#include "compat.h"
|
||||||
|
|
||||||
hashtab_t *opcode_table;
|
hashtab_t *opcode_table;
|
||||||
|
|
||||||
VISIBLE int pr_type_size[ev_type_count] = {
|
VISIBLE int pr_type_size[ev_type_count] = {
|
||||||
|
@ -1041,6 +1043,282 @@ VISIBLE opcode_t pr_opcodes[] = {
|
||||||
"%Ga, %Gb, %Gc",
|
"%Ga, %Gb, %Gc",
|
||||||
},
|
},
|
||||||
|
|
||||||
|
{"<PUSH>", "push.s", OP_PUSH_S, false,
|
||||||
|
ev_string, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%Ga",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "push.f", OP_PUSH_F, false,
|
||||||
|
ev_float, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%Ga",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "push.v", OP_PUSH_V, false,
|
||||||
|
ev_vector, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%Ga",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "push.ent", OP_PUSH_ENT, false,
|
||||||
|
ev_entity, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%Ga",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "push.fld", OP_PUSH_FLD, false,
|
||||||
|
ev_field, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%Ga",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "push.fn", OP_PUSH_FN, false,
|
||||||
|
ev_func, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%Ga",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "push.p", OP_PUSH_P, false,
|
||||||
|
ev_pointer, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%Ga",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "push.q", OP_PUSH_Q, false,
|
||||||
|
ev_quat, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%Ga",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "push.i", OP_PUSH_I, false,
|
||||||
|
ev_integer, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%Ga",
|
||||||
|
},
|
||||||
|
|
||||||
|
{"<PUSH>", "pushb.s", OP_PUSHB_S, false,
|
||||||
|
ev_pointer, ev_integer, ev_string,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushb.f", OP_PUSHB_F, false,
|
||||||
|
ev_pointer, ev_integer, ev_float,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushb.v", OP_PUSHB_V, false,
|
||||||
|
ev_pointer, ev_integer, ev_vector,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushb.ent", OP_PUSHB_ENT, false,
|
||||||
|
ev_pointer, ev_integer, ev_entity,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushb.fld", OP_PUSHB_FLD, false,
|
||||||
|
ev_pointer, ev_integer, ev_field,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushb.fn", OP_PUSHB_FN, false,
|
||||||
|
ev_pointer, ev_integer, ev_func,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushb.p", OP_PUSHB_P, false,
|
||||||
|
ev_pointer, ev_integer, ev_pointer,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushb.q", OP_PUSHB_Q, false,
|
||||||
|
ev_pointer, ev_integer, ev_quat,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushb.i", OP_PUSHB_I, false,
|
||||||
|
ev_pointer, ev_integer, ev_integer,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
|
||||||
|
{"<PUSH>", "pushbi.s", OP_PUSHBI_S, false,
|
||||||
|
ev_pointer, ev_short, ev_string,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushbi.f", OP_PUSHBI_F, false,
|
||||||
|
ev_pointer, ev_short, ev_float,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushbi.v", OP_PUSHBI_V, false,
|
||||||
|
ev_pointer, ev_short, ev_vector,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushbi.ent", OP_PUSHBI_ENT, false,
|
||||||
|
ev_pointer, ev_short, ev_entity,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushbi.fld", OP_PUSHBI_FLD, false,
|
||||||
|
ev_pointer, ev_short, ev_field,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushbi.fn", OP_PUSHBI_FN, false,
|
||||||
|
ev_pointer, ev_short, ev_func,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushbi.p", OP_PUSHBI_P, false,
|
||||||
|
ev_pointer, ev_short, ev_pointer,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushbi.q", OP_PUSHBI_Q, false,
|
||||||
|
ev_pointer, ev_short, ev_quat,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<PUSH>", "pushbi.i", OP_PUSHBI_I, false,
|
||||||
|
ev_pointer, ev_short, ev_integer,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
|
||||||
|
{"<POP>", "pop.s", OP_POP_S, false,
|
||||||
|
ev_string, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%ga",
|
||||||
|
},
|
||||||
|
{"<POP>", "pop.f", OP_POP_F, false,
|
||||||
|
ev_float, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%ga",
|
||||||
|
},
|
||||||
|
{"<POP>", "pop.v", OP_POP_V, false,
|
||||||
|
ev_vector, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%ga",
|
||||||
|
},
|
||||||
|
{"<POP>", "pop.ent", OP_POP_ENT, false,
|
||||||
|
ev_entity, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%ga",
|
||||||
|
},
|
||||||
|
{"<POP>", "pop.fld", OP_POP_FLD, false,
|
||||||
|
ev_field, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%ga",
|
||||||
|
},
|
||||||
|
{"<POP>", "pop.fn", OP_POP_FN, false,
|
||||||
|
ev_func, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%ga",
|
||||||
|
},
|
||||||
|
{"<POP>", "pop.p", OP_POP_P, false,
|
||||||
|
ev_pointer, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%ga",
|
||||||
|
},
|
||||||
|
{"<POP>", "pop.q", OP_POP_Q, false,
|
||||||
|
ev_quat, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%ga",
|
||||||
|
},
|
||||||
|
{"<POP>", "pop.i", OP_POP_I, false,
|
||||||
|
ev_integer, ev_invalid, ev_invalid,
|
||||||
|
PROG_VERSION,
|
||||||
|
"%ga",
|
||||||
|
},
|
||||||
|
|
||||||
|
{"<POP>", "popb.s", OP_POPB_S, false,
|
||||||
|
ev_pointer, ev_integer, ev_string,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popb.f", OP_POPB_F, false,
|
||||||
|
ev_pointer, ev_integer, ev_float,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popb.v", OP_POPB_V, false,
|
||||||
|
ev_pointer, ev_integer, ev_vector,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popb.ent", OP_POPB_ENT, false,
|
||||||
|
ev_pointer, ev_integer, ev_entity,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popb.fld", OP_POPB_FLD, false,
|
||||||
|
ev_pointer, ev_integer, ev_field,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popb.fn", OP_POPB_FN, false,
|
||||||
|
ev_pointer, ev_integer, ev_func,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popb.p", OP_POPB_P, false,
|
||||||
|
ev_pointer, ev_integer, ev_pointer,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popb.q", OP_POPB_Q, false,
|
||||||
|
ev_pointer, ev_integer, ev_quat,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popb.i", OP_POPB_I, false,
|
||||||
|
ev_pointer, ev_integer, ev_integer,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %Gb)",
|
||||||
|
},
|
||||||
|
|
||||||
|
{"<POP>", "popbi.s", OP_POPBI_S, false,
|
||||||
|
ev_pointer, ev_short, ev_string,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popbi.f", OP_POPBI_F, false,
|
||||||
|
ev_pointer, ev_short, ev_float,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popbi.v", OP_POPBI_V, false,
|
||||||
|
ev_pointer, ev_short, ev_vector,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popbi.ent", OP_POPBI_ENT, false,
|
||||||
|
ev_pointer, ev_short, ev_entity,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popbi.fld", OP_POPBI_FLD, false,
|
||||||
|
ev_pointer, ev_short, ev_field,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popbi.fn", OP_POPBI_FN, false,
|
||||||
|
ev_pointer, ev_short, ev_func,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popbi.p", OP_POPBI_P, false,
|
||||||
|
ev_pointer, ev_short, ev_pointer,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popbi.q", OP_POPBI_Q, false,
|
||||||
|
ev_pointer, ev_short, ev_quat,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
{"<POP>", "popbi.i", OP_POPBI_I, false,
|
||||||
|
ev_pointer, ev_short, ev_integer,
|
||||||
|
PROG_VERSION,
|
||||||
|
"*(%Ga + %sb)",
|
||||||
|
},
|
||||||
|
|
||||||
// end of table
|
// end of table
|
||||||
{0},
|
{0},
|
||||||
};
|
};
|
||||||
|
@ -1192,11 +1470,16 @@ PR_Check_Opcodes (progs_t *pr)
|
||||||
opcode_t *op;
|
opcode_t *op;
|
||||||
dstatement_t *st;
|
dstatement_t *st;
|
||||||
int state_ok = 0;
|
int state_ok = 0;
|
||||||
|
int pushpop_ok = 0;
|
||||||
pr_uint_t i;
|
pr_uint_t i;
|
||||||
|
|
||||||
if (pr->globals.time && pr->globals.self && pr->fields.nextthink != -1
|
if (pr->globals.time && pr->globals.self && pr->fields.nextthink != -1
|
||||||
&& pr->fields.think != -1 && pr->fields.frame != -1)
|
&& pr->fields.think != -1 && pr->fields.frame != -1) {
|
||||||
state_ok = 1;
|
state_ok = 1;
|
||||||
|
}
|
||||||
|
if (pr->globals.stack) {
|
||||||
|
pushpop_ok = 1;
|
||||||
|
}
|
||||||
|
|
||||||
//FIXME need to decide if I really want to always do static bounds checking
|
//FIXME need to decide if I really want to always do static bounds checking
|
||||||
// the only problem is that it slows progs load a little, but it's the only
|
// the only problem is that it slows progs load a little, but it's the only
|
||||||
|
@ -1214,6 +1497,11 @@ PR_Check_Opcodes (progs_t *pr)
|
||||||
PR_Error (pr, "PR_Check_Opcodes: %s used with missing fields "
|
PR_Error (pr, "PR_Check_Opcodes: %s used with missing fields "
|
||||||
"or globals", op->opname);
|
"or globals", op->opname);
|
||||||
}
|
}
|
||||||
|
if ((strequal(op->name, "<PUSH>") || strequal(op->name, "<POP>"))
|
||||||
|
&& !pushpop_ok) {
|
||||||
|
PR_Error (pr, "PR_Check_Opcodes: %s used with missing .stack "
|
||||||
|
"globals", op->opname);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
for (i = 0, st = pr->pr_statements; i < pr->progs->numstatements;
|
for (i = 0, st = pr->pr_statements; i < pr->progs->numstatements;
|
||||||
|
@ -1267,6 +1555,69 @@ PR_Check_Opcodes (progs_t *pr)
|
||||||
check_global_size (pr, st, op, st->b, st->a);
|
check_global_size (pr, st, op, st->b, st->a);
|
||||||
check_global_size (pr, st, op, st->b, st->c);
|
check_global_size (pr, st, op, st->b, st->c);
|
||||||
break;
|
break;
|
||||||
|
case OP_PUSHB_F:
|
||||||
|
case OP_PUSHB_S:
|
||||||
|
case OP_PUSHB_ENT:
|
||||||
|
case OP_PUSHB_FLD:
|
||||||
|
case OP_PUSHB_FN:
|
||||||
|
case OP_PUSHB_I:
|
||||||
|
case OP_PUSHB_P:
|
||||||
|
case OP_PUSHB_V:
|
||||||
|
case OP_PUSHB_Q:
|
||||||
|
case OP_PUSHBI_F:
|
||||||
|
case OP_PUSHBI_S:
|
||||||
|
case OP_PUSHBI_ENT:
|
||||||
|
case OP_PUSHBI_FLD:
|
||||||
|
case OP_PUSHBI_FN:
|
||||||
|
case OP_PUSHBI_I:
|
||||||
|
case OP_PUSHBI_P:
|
||||||
|
case OP_PUSHBI_V:
|
||||||
|
case OP_PUSHBI_Q:
|
||||||
|
// op->type_c is used for selecting the operator during
|
||||||
|
// compilation, but is invalid when running
|
||||||
|
check_global (pr, st, op, op->type_a, st->a, 1);
|
||||||
|
check_global (pr, st, op, op->type_b, st->b, 1);
|
||||||
|
check_global (pr, st, op, ev_invalid, st->c, 1);
|
||||||
|
break;
|
||||||
|
case OP_POP_F:
|
||||||
|
case OP_POP_FLD:
|
||||||
|
case OP_POP_ENT:
|
||||||
|
case OP_POP_S:
|
||||||
|
case OP_POP_FN:
|
||||||
|
case OP_POP_I:
|
||||||
|
case OP_POP_P:
|
||||||
|
case OP_POP_V:
|
||||||
|
case OP_POP_Q:
|
||||||
|
// don't want to check for denormal floats, otherwise
|
||||||
|
// OP_POP_* could use the defualt rule
|
||||||
|
check_global (pr, st, op, op->type_a, st->a, 0);
|
||||||
|
check_global (pr, st, op, ev_invalid, st->b, 1);
|
||||||
|
check_global (pr, st, op, ev_invalid, st->c, 1);
|
||||||
|
break;
|
||||||
|
case OP_POPB_F:
|
||||||
|
case OP_POPB_S:
|
||||||
|
case OP_POPB_ENT:
|
||||||
|
case OP_POPB_FLD:
|
||||||
|
case OP_POPB_FN:
|
||||||
|
case OP_POPB_I:
|
||||||
|
case OP_POPB_P:
|
||||||
|
case OP_POPB_V:
|
||||||
|
case OP_POPB_Q:
|
||||||
|
case OP_POPBI_F:
|
||||||
|
case OP_POPBI_S:
|
||||||
|
case OP_POPBI_ENT:
|
||||||
|
case OP_POPBI_FLD:
|
||||||
|
case OP_POPBI_FN:
|
||||||
|
case OP_POPBI_I:
|
||||||
|
case OP_POPBI_P:
|
||||||
|
case OP_POPBI_V:
|
||||||
|
case OP_POPBI_Q:
|
||||||
|
// op->type_c is used for selecting the operator during
|
||||||
|
// compilation, but is invalid when running
|
||||||
|
check_global (pr, st, op, op->type_a, st->a, 1);
|
||||||
|
check_global (pr, st, op, op->type_b, st->b, 1);
|
||||||
|
check_global (pr, st, op, ev_invalid, st->c, 1);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
check_global (pr, st, op, op->type_a, st->a, 1);
|
check_global (pr, st, op, op->type_a, st->a, 1);
|
||||||
check_global (pr, st, op, op->type_b, st->b,
|
check_global (pr, st, op, op->type_b, st->b,
|
||||||
|
|
|
@ -149,6 +149,13 @@ PR_ResolveGlobals (progs_t *pr)
|
||||||
|| (def = PR_FindGlobal (pr, "self")))
|
|| (def = PR_FindGlobal (pr, "self")))
|
||||||
pr->globals.self = &G_INT (pr, def->ofs);
|
pr->globals.self = &G_INT (pr, def->ofs);
|
||||||
}
|
}
|
||||||
|
if (!pr->globals.stack) {
|
||||||
|
if ((def = PR_FindGlobal (pr, ".stack"))
|
||||||
|
|| (def = PR_FindGlobal (pr, "stack")))
|
||||||
|
pr->globals.stack = &G_POINTER (pr, def->ofs);
|
||||||
|
// the stack is at the very end of the progs memory map
|
||||||
|
*pr->globals.stack = pr->globals_size;
|
||||||
|
}
|
||||||
if (pr->fields.nextthink == -1)
|
if (pr->fields.nextthink == -1)
|
||||||
if ((def = PR_FindField (pr, "nextthink")))
|
if ((def = PR_FindField (pr, "nextthink")))
|
||||||
pr->fields.nextthink = def->ofs;
|
pr->fields.nextthink = def->ofs;
|
||||||
|
|
Loading…
Reference in a new issue