home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
World of A1200
/
World_Of_A1200.iso
/
programs
/
compress
/
filearchivers
/
zip
/
zip-1.00.lha
/
interpre.c
< prev
next >
Wrap
C/C++ Source or Header
|
1992-10-09
|
7KB
|
179 lines
/*
* interpre.c
*
* Main interpreter loop
*
* Mark Howell 28-Jul-1992 V1.0
*
*/
#include "ztypes.h"
/*
* interpret
*
* Interpret Z code
*
*/
#ifdef __STDC__
void interpret (void)
#else
void interpret ()
#endif
{
zbyte_t opcode;
zword_t specifier, operand[8];
int maxoperands, count, i;
/* Loop until HALT instruction executed */
for ( ; ; ) {
/* Load opcode and set operand count */
opcode = read_code_byte ();
count = 0;
/* Multiple operand instructions */
if (opcode < 0x80 || opcode > 0xc0) {
/* Two operand class, load both operands */
if (opcode < 0x80) {
operand[count++] = load_operand ((opcode & 0x40) ? 2 : 1);
operand[count++] = load_operand ((opcode & 0x20) ? 2 : 1);
opcode &= 0x1f;
} else {
/* Variable operand class, load operand specifier */
if (opcode == 0xec) { /* Extended CALL instruction */
specifier = read_code_word ();
maxoperands = 8;
} else {
specifier = read_code_byte ();
maxoperands = 4;
}
/* Load operands */
for (i = (maxoperands - 1) * 2; i >= 0; i -= 2)
if (((specifier >> i) & 0x03) != 3)
operand[count++] = load_operand ((specifier >> i) & 0x03);
else
i = 0;
opcode &= 0x3f;
}
switch ((char) opcode) {
/* Two or multiple operand instructions */
case 0x01: compare_je (count, operand); break;
case 0x02: compare_jl (operand[0], operand[1]); break;
case 0x03: compare_jg (operand[0], operand[1]); break;
case 0x04: decrement_check (operand[0], operand[1]); break;
case 0x05: increment_check (operand[0], operand[1]); break;
case 0x06: compare_parent_object (operand[0], operand[1]); break;
case 0x07: test (operand[0], operand[1]); break;
case 0x08: or (operand[0], operand[1]); break;
case 0x09: and (operand[0], operand[1]); break;
case 0x0a: test_attr (operand[0], operand[1]); break;
case 0x0b: set_attr (operand[0], operand[1]); break;
case 0x0c: clear_attr (operand[0], operand[1]); break;
case 0x0d: store_variable (operand[0], operand[1]); break;
case 0x0e: insert_object (operand[0], operand[1]); break;
case 0x0f: load_word (operand[0], operand[1]); break;
case 0x10: load_byte (operand[0], operand[1]); break;
case 0x11: load_property (operand[0], operand[1]); break;
case 0x12: load_property_address (operand[0], operand[1]); break;
case 0x13: load_next_property (operand[0], operand[1]); break;
case 0x14: add (operand[0], operand[1]); break;
case 0x15: subtract (operand[0], operand[1]); break;
case 0x16: multiply (operand[0], operand[1]); break;
case 0x17: divide (operand[0], operand[1]); break;
case 0x18: remainder (operand[0], operand[1]); break;
case 0x19: call (count, operand); break;
/* Multiple operand instructions */
case 0x20: call (count, operand); break;
case 0x21: store_word (operand[0], operand[1], operand[2]); break;
case 0x22: store_byte (operand[0], operand[1], operand[2]); break;
case 0x23: store_property (operand[0], operand[1], operand[2]); break;
case 0x24: read_line (count, operand); break;
case 0x25: print_character (operand[0]); break;
case 0x26: print_number (operand[0]); break;
case 0x27: random (operand[0]); break;
case 0x28: push_var (operand[0]); break;
case 0x29: pop_var (operand[0]); break;
case 0x2a: set_status_size (operand[0]); break;
case 0x2b: select_window (operand[0]); break;
case 0x2c: call (count, operand); break;
case 0x2d: erase_window (operand[0]); break;
case 0x2e: erase_line (operand[0]); break;
case 0x2f: set_cursor_position (operand[0], operand[1]); break;
case 0x30: break;
case 0x31: set_video_attribute (operand[0]); break;
case 0x32: set_format_mode (operand[0]); break;
case 0x33: set_print_modes (operand[0], operand[1]); break;
case 0x34: break;
case 0x35: sound (count, operand); break;
case 0x36: read_character (count, operand); break;
case 0x37: scan_word (operand[0], operand[1], operand[2]); break;
default: fatal ("Illegal operation");
}
} else {
/* Single operand class, load operand and execute instruction */
if (opcode < 0xb0) {
operand[0] = load_operand ((opcode >> 4) & 0x03);
switch ((char) opcode & 0x0f) {
case 0x00: compare_zero (operand[0]); break;
case 0x01: load_next_object (operand[0]); break;
case 0x02: load_child_object (operand[0]); break;
case 0x03: load_parent_object (operand[0]); break;
case 0x04: load_property_length (operand[0]); break;
case 0x05: increment (operand[0]); break;
case 0x06: decrement (operand[0]); break;
case 0x07: print_offset (operand[0]); break;
case 0x08: call (1, operand); break;
case 0x09: remove_object (operand[0]); break;
case 0x0a: print_object (operand[0]); break;
case 0x0b: ret (operand[0]); break;
case 0x0c: jump (operand[0]); break;
case 0x0d: print_address (operand[0]); break;
case 0x0e: load (operand[0]); break;
case 0x0f: not (operand[0]); break;
}
} else {
/* Zero operand class, execute instruction */
switch ((char) opcode & 0x0f) {
case 0x00: ret (TRUE); break;
case 0x01: ret (FALSE); break;
case 0x02: print_literal (); break;
case 0x03: println_return (); break;
case 0x04: break;
case 0x05: save (); break;
case 0x06: restore (); break;
case 0x07: restart (); break;
case 0x08: pop_ret (); break;
case 0x09: pop (); break;
case 0x0a: return;
case 0x0b: new_line (); break;
case 0x0c: display_status_line (); break;
case 0x0d: verify (); break;
default: fatal ("Illegal operation");
}
}
}
}
}/* interpret */