7f28c94eda
The interpreter more or less passes all the tests in the z-machine test suite. It should build for every target except Archos (for which it is disabled). git-svn-id: svn://svn.rockbox.org/rockbox/trunk@24267 a1c6a512-1295-4272-9138-f99709370657
304 lines
5.2 KiB
C
304 lines
5.2 KiB
C
/* variable.c - Variable and stack related opcodes
|
|
* Copyright (c) 1995-1997 Stefan Jokisch
|
|
*
|
|
* This file is part of Frotz.
|
|
*
|
|
* Frotz 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.
|
|
*
|
|
* Frotz 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 the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
|
*/
|
|
|
|
#include "frotz.h"
|
|
|
|
/*
|
|
* z_dec, decrement a variable.
|
|
*
|
|
* zargs[0] = variable to decrement
|
|
*
|
|
*/
|
|
|
|
void z_dec (void)
|
|
{
|
|
zword value;
|
|
|
|
if (zargs[0] == 0)
|
|
(*sp)--;
|
|
else if (zargs[0] < 16)
|
|
(*(fp - zargs[0]))--;
|
|
else {
|
|
zword addr = h_globals + 2 * (zargs[0] - 16);
|
|
LOW_WORD (addr, value)
|
|
value--;
|
|
SET_WORD (addr, value)
|
|
}
|
|
|
|
}/* z_dec */
|
|
|
|
/*
|
|
* z_dec_chk, decrement a variable and branch if now less than value.
|
|
*
|
|
* zargs[0] = variable to decrement
|
|
* zargs[1] = value to check variable against
|
|
*
|
|
*/
|
|
|
|
void z_dec_chk (void)
|
|
{
|
|
zword value;
|
|
|
|
if (zargs[0] == 0)
|
|
value = --(*sp);
|
|
else if (zargs[0] < 16)
|
|
value = --(*(fp - zargs[0]));
|
|
else {
|
|
zword addr = h_globals + 2 * (zargs[0] - 16);
|
|
LOW_WORD (addr, value)
|
|
value--;
|
|
SET_WORD (addr, value)
|
|
}
|
|
|
|
branch ((short) value < (short) zargs[1]);
|
|
|
|
}/* z_dec_chk */
|
|
|
|
/*
|
|
* z_inc, increment a variable.
|
|
*
|
|
* zargs[0] = variable to increment
|
|
*
|
|
*/
|
|
|
|
void z_inc (void)
|
|
{
|
|
zword value;
|
|
|
|
if (zargs[0] == 0)
|
|
(*sp)++;
|
|
else if (zargs[0] < 16)
|
|
(*(fp - zargs[0]))++;
|
|
else {
|
|
zword addr = h_globals + 2 * (zargs[0] - 16);
|
|
LOW_WORD (addr, value)
|
|
value++;
|
|
SET_WORD (addr, value)
|
|
}
|
|
|
|
}/* z_inc */
|
|
|
|
/*
|
|
* z_inc_chk, increment a variable and branch if now greater than value.
|
|
*
|
|
* zargs[0] = variable to increment
|
|
* zargs[1] = value to check variable against
|
|
*
|
|
*/
|
|
|
|
void z_inc_chk (void)
|
|
{
|
|
zword value;
|
|
|
|
if (zargs[0] == 0)
|
|
value = ++(*sp);
|
|
else if (zargs[0] < 16)
|
|
value = ++(*(fp - zargs[0]));
|
|
else {
|
|
zword addr = h_globals + 2 * (zargs[0] - 16);
|
|
LOW_WORD (addr, value)
|
|
value++;
|
|
SET_WORD (addr, value)
|
|
}
|
|
|
|
branch ((short) value > (short) zargs[1]);
|
|
|
|
}/* z_inc_chk */
|
|
|
|
/*
|
|
* z_load, store the value of a variable.
|
|
*
|
|
* zargs[0] = variable to store
|
|
*
|
|
*/
|
|
|
|
void z_load (void)
|
|
{
|
|
zword value;
|
|
|
|
if (zargs[0] == 0)
|
|
value = *sp;
|
|
else if (zargs[0] < 16)
|
|
value = *(fp - zargs[0]);
|
|
else {
|
|
zword addr = h_globals + 2 * (zargs[0] - 16);
|
|
LOW_WORD (addr, value)
|
|
}
|
|
|
|
store (value);
|
|
|
|
}/* z_load */
|
|
|
|
/*
|
|
* z_pop, pop a value off the game stack and discard it.
|
|
*
|
|
* no zargs used
|
|
*
|
|
*/
|
|
|
|
void z_pop (void)
|
|
{
|
|
|
|
sp++;
|
|
|
|
}/* z_pop */
|
|
|
|
/*
|
|
* z_pop_stack, pop n values off the game or user stack and discard them.
|
|
*
|
|
* zargs[0] = number of values to discard
|
|
* zargs[1] = address of user stack (optional)
|
|
*
|
|
*/
|
|
|
|
void z_pop_stack (void)
|
|
{
|
|
|
|
if (zargc == 2) { /* it's a user stack */
|
|
|
|
zword size;
|
|
zword addr = zargs[1];
|
|
|
|
LOW_WORD (addr, size)
|
|
|
|
size += zargs[0];
|
|
storew (addr, size);
|
|
|
|
} else sp += zargs[0]; /* it's the game stack */
|
|
|
|
}/* z_pop_stack */
|
|
|
|
/*
|
|
* z_pull, pop a value off...
|
|
*
|
|
* a) ...the game or a user stack and store it (V6)
|
|
*
|
|
* zargs[0] = address of user stack (optional)
|
|
*
|
|
* b) ...the game stack and write it to a variable (other than V6)
|
|
*
|
|
* zargs[0] = variable to write value to
|
|
*
|
|
*/
|
|
|
|
void z_pull (void)
|
|
{
|
|
zword value;
|
|
|
|
if (h_version != V6) { /* not a V6 game, pop stack and write */
|
|
|
|
value = *sp++;
|
|
|
|
if (zargs[0] == 0)
|
|
*sp = value;
|
|
else if (zargs[0] < 16)
|
|
*(fp - zargs[0]) = value;
|
|
else {
|
|
zword addr = h_globals + 2 * (zargs[0] - 16);
|
|
SET_WORD (addr, value)
|
|
}
|
|
|
|
} else { /* it's V6, but is there a user stack? */
|
|
|
|
if (zargc == 1) { /* it's a user stack */
|
|
|
|
zword size;
|
|
zword addr = zargs[0];
|
|
|
|
LOW_WORD (addr, size)
|
|
|
|
size++;
|
|
storew (addr, size);
|
|
|
|
addr += 2 * size;
|
|
LOW_WORD (addr, value)
|
|
|
|
} else value = *sp++; /* it's the game stack */
|
|
|
|
store (value);
|
|
|
|
}
|
|
|
|
}/* z_pull */
|
|
|
|
/*
|
|
* z_push, push a value onto the game stack.
|
|
*
|
|
* zargs[0] = value to push onto the stack
|
|
*
|
|
*/
|
|
|
|
void z_push (void)
|
|
{
|
|
|
|
*--sp = zargs[0];
|
|
|
|
}/* z_push */
|
|
|
|
/*
|
|
* z_push_stack, push a value onto a user stack then branch if successful.
|
|
*
|
|
* zargs[0] = value to push onto the stack
|
|
* zargs[1] = address of user stack
|
|
*
|
|
*/
|
|
|
|
void z_push_stack (void)
|
|
{
|
|
zword size;
|
|
zword addr = zargs[1];
|
|
|
|
LOW_WORD (addr, size)
|
|
|
|
if (size != 0) {
|
|
|
|
storew ((zword) (addr + 2 * size), zargs[0]);
|
|
|
|
size--;
|
|
storew (addr, size);
|
|
|
|
}
|
|
|
|
branch (size);
|
|
|
|
}/* z_push_stack */
|
|
|
|
/*
|
|
* z_store, write a value to a variable.
|
|
*
|
|
* zargs[0] = variable to be written to
|
|
* zargs[1] = value to write
|
|
*
|
|
*/
|
|
|
|
void z_store (void)
|
|
{
|
|
zword value = zargs[1];
|
|
|
|
if (zargs[0] == 0)
|
|
*sp = value;
|
|
else if (zargs[0] < 16)
|
|
*(fp - zargs[0]) = value;
|
|
else {
|
|
zword addr = h_globals + 2 * (zargs[0] - 16);
|
|
SET_WORD (addr, value)
|
|
}
|
|
|
|
}/* z_store */
|