scummvm/engines/saga2/code.h
2021-07-01 01:36:44 +02:00

418 lines
13 KiB
C++

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* 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 the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*
* Based on the original sources
* Faery Tale II -- The Halls of the Dead
* (c) 1993-1996 The Wyrmkeep Entertainment Co.
*/
#ifndef SAGA2_CODE_H
#define SAGA2_CODE_H
namespace Saga2 {
// types of operations for code generation
// note that the macros listed below depend on the ordering of this
// table to determine things like unary-op, binary-op, etc.
enum op_types {
op_undefined = 0,
// internal operations
op_nextblock, // continue execution at next block
op_dup, // duplicate 16-bit value on stack
op_drop, // drop 16-bit value on stack
// primary values
op_zero, // push a zero on the stack
op_one, // push a one on the stack
op_constint, // constant integer
op_constid, // constant id reference
op_strlit, // string literal
op_sym, // symbol address
op_symref, // symbol contents
op_classref, // reference to "this"
op_deref, // dereference of an ID
// references within this module
op_getflag, // read from flag bit (mode)
op_getbyte, // read from byte field (mode)
op_getint, // read from integer field (mode)
op_getstr, // read from string field (mode)
op_getid, // read from id field (mode)
op_putflag, // put to flag bit (mode)
op_putbyte, // put to byte field (mode)
op_putint, // put to integer field (mode)
op_putstr, // put to string field (mode)
op_putid, // put to id field (mode)
op_pea, // push effective address onto stack
// 'void' versions consume their arguments
op_putflag_v, // put to flag bit (mode)
op_putbyte_v, // put to byte field (mode)
op_putint_v, // put to integer field (mode)
op_putstr_v, // put to string field (mode)
op_putid_v, // put to id field (mode)
// function call
op_call_near, // call function in same segment
op_call_far, // call function in other segment
op_ccall, // call C function
op_ccall_v, // call C function (void)
op_call_member, // call member function
op_call_member_v, // call member function (void)
op_enter, // enter a function
op_return, // return from function
op_return_v, // return nothing from function
// branches
op_jmp,
op_jmp_true_v, // test arg and consume
op_jmp_false_v, // test arg and consume
op_jmp_true, // test arg and don't consume
op_jmp_false, // test arg and don't consume
op_jmp_switch, // switch statement (integer)
op_jmp_strswitch, // switch statement (string)
op_jmp_random, // random jump
// unary operators
op_negate,
op_not,
op_compl,
op_inc_v, // increment, don't push
op_dec_v, // decrement, don't push
op_postinc,
op_postdec,
// arithmetic
op_add,
op_sub,
op_mul,
op_div,
op_mod,
// conditional
op_conditional,
op_comma,
// comparison
op_eq,
op_ne,
op_gt,
op_lt,
op_ge,
op_le,
// string comparison
op_str_eq,
op_str_ne,
op_str_gt,
op_str_lt,
op_str_ge,
op_str_le,
// shift
op_rsh,
op_lsh,
// bitwise
op_and,
op_or,
op_xor,
// logical
op_land,
op_lor,
op_lxor,
// string functions
op_strcat, // string concatenation
op_strformat, // string formatting
// assignment operators -- none of these are actually compiled into
// code (they become get/put type operations)
op_assign,
// none of these are even used currently...
op_asplus,
op_asminus,
op_astimes,
op_asdiv,
op_asmod,
op_asrshift,
op_aslshift,
op_asand,
op_asor,
// Special ops
op_speak,
op_dialog_begin,
op_dialog_end,
op_reply,
op_animate,
// New opcodes
op_jmp_seedrandom, // seeded random jump
op_symref_x, // get the export number of the symbol
#if 0
op_type,
#endif
op_last /* about 90 so far */
};
// addressing modes for get and put
enum addr_types {
// Offset reference to the thread structure
addr_thread = 0,
// Offset reference to the stack
addr_stack,
// Implicit reference to the currently executing segment
addr_near,
// Implicit reference to data segment
addr_data,
// This addressing mode references any external segment
// using a 16-bit segment number and a 16-bit segment offset
// which immediately follow the instruction.
addr_far,
// This addressing mode is used for segment-array addressing
// it's a 16-bit segment followed by a 16-bit object number,
// followed by a (byte or bit) offset within the object
addr_array,
// This addressing mode uses a 16-bit segment number
// and a 16-bit offset which have been put on the stack.
// addr_indirect, // use SEG:offset on stack
// addr_indirect_index, // use SEG:index:offset on stack
// This addressing mode is used for dereferencing objects.
// It consists of an _embedded_ address for retrieving the
// object number, followed by a 16-bit segment number for
// the dereferenced objects, followed by a 16-bit index
// into the dereferenced object.
// REM: We also need a "far deref" for computing the
// dereferenced object's segment number.
addr_deref,
// Addressing mode used for class member functions. It
// specified that the address is relative to whatever
// object the 1st argument is referrring to.
addr_this, // relative to arg 1
// addr_common=0, // offset from global variables
// addr_static, // offset from global variables
// addr_id, // offset from const id object
// addr_id_indirect, // offset from stack id object
// addr_index, // index from id
};
#define IS_CONST(x) ((x) >= op_constint && (x) <= op_conststr)
#define IS_ADDRESS(x) ((x) == op_sym)
#define IS_UNOP(x) ((x) >= op_negate && (x) <= op_postdec)
#define IS_BINOP(x) ((x) >= op_add && (x) <= op_strcat)
#define IS_ASOP(x) ((x) >= op_assign && (x) <= op_asor)
// #define IS_UNOP2(x) ((x) == op_getarray || (x) == op_putarray)
// #define CONST(op) ((op) >= op_constflag && (op) <= op_conststr)
#if 0
op_conststr, // constant string
/* we would need an array ref for each type of thing we get,
as well as a way to get the array base...
(I.e. pointer math)
*/
op_getarray, /* arrays not supported currently */
op_putarray,
operands we need to add:
op_select implements selection statement
op_bin2str binary to string
op_fork might have to be done in - language so debugger works ?
op_foreach implements "foreach"
op_t, fast true and false
op_f,
- - also need to change statement producer to "drop" top of stack
if necessary.
op_min, /* comparison opcodes */
op_max,
op_clamp,
op_abs,
op_dropn, /* drop n stack locations */
op_swap, /* swap s[0] and s[1] */
op_swap2, /* transpose s[0] and s[2] */
op_dupn, /* duplicate s[0]-s[n-1] */
op_acalc, /* array calculation */
op_field, /* field reference */
#endif
// Flags for special statements
#define SPEAKF_NOANIMATE (1<<0) // speaker should animate
#define SPEAKF_ASYNC (1<<1) // async speech.
#define REPLYF_ONCE (1<<0) // 'once' flag
#define REPLYF_SUMMARY (1<<1) // response is only a summary
#define REPLYF_CONDITION (1<<2) // response has a condition
#define ANIMATEF_REPEAT (1<<0)
// BasicBlock describes a block of generated code
#ifdef COMPILE_H
typedef struct _BasicBlock {
struct _BasicBlock *next; // pointer to next block
short label; // label for this block
struct _BasicBlock *from[2], // where we could have come from
*jumpto, // where to go if jump
*fallto; // where to go if fall through
struct _Statement *first_st, // statement list for this bblock
*last_st; // it's a circular list
ENode *test_expr; // test expression for jump
char in_flags, // flags for entering this block
jump_type; // where to go after this block
uint16 start_offset, // offset in module of this block
jump_offset; // offset of module of jump at end
int16 source_file, // source line of statement
source_line; // source file of statement
} BasicBlock;
// Various flags for the block
#define BBLOCK_CLABEL (1<<0) /* bblock begins with a 'C' label */
#define BBLOCK_INTLABEL (1<<1) /* bblock begins with internal label */
#define BBLOCK_FIRST (1<<2) /* first bblock in the function */
#define BBLOCK_MANY (1<<3) /* many guys jump to here */
#define BBLOCK_CASE (1<<4) /* case stmts might not have 'from' ptr */
// Jump types
enum jump_types {
jump_never = 0, /* never jump */
jump_false, /* jump on expression if false */
jump_true, /* jump on expression if true */
jump_always, /* jump always */
jump_cjump, /* 'c' goto stm, jumpto is a label # */
jump_return, /* block ends with a 'return' statement */
jump_switch, /* jumps to lots of places */
};
#endif
/*
// Module describes all the code associated with a particular
// compilation unit.
struct Module {
uint16 exportCount, // number of symbols exported
importCount; // number of symbols imported
uint16 exportOffset, // start of export table
importOffset; // start of import table
// Q: Should ModuleName be somewhere else, like in the main program?
uint16 staticSize; // size of static data
uint16 stringOffset; // unused in this version
// int16 moduleName; // offset to name of module
// Followed by list of exports
// Followed by list of imports
// Followed by symbol names, each NULL-terminated
// Followed by actual code blocks
};
// Exports: Each Export contains the offset to find the name of
// the object. After that is the offset to find the object itself.
// The first Export is always the module entry point
struct ModuleExport { // an exported symbol
uint16 symbolOffset, // offset in module of symbol name
objectOffset; // where to find object
};
// Imports: Each Import contains the offset to find the name of
// the object. After that is a blank pointer, which will be filled
// in with the real address of the object. When addressing the
// imported object, the code in this module will refer to this
// ModuleImport entry, and indirectly locate the referenced object.
struct ModuleImport { // a module xref
uint16 symbolOffset; // where, from here, to find name
void *objectPointer; // NULL pointer, filled in on load
};
*/
} // end of namespace Saga2
#endif