Arkshine 17114347d1 Compiler: Fix runtime error in variadic functions that return strings.
Imported from SM-AM: 1d1244c2f0.

"This fixes a bug where returning a string from a variadic function caused
an invalid memory access error during runtime. It seems like they forgot
to update existing string return code for variadic functions."
2014-08-26 09:04:56 +02:00

1383 lines
35 KiB
C
Executable File

/* Pawn compiler - code generation (unoptimized "assembler" code)
*
* Copyright (c) ITB CompuPhase, 1997-2005
*
* This software is provided "as-is", without any express or implied warranty.
* In no event will the authors be held liable for any damages arising from
* the use of this software.
*
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
*
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software in
* a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
#include <assert.h>
#include <ctype.h>
#include <stdio.h>
#include <stdlib.h> /* for _MAX_PATH */
#include <string.h>
#if defined FORTIFY
#include "fortify.h"
#endif
#include "sc.h"
/* When a subroutine returns to address 0, the AMX must halt. In earlier
* releases, the RET and RETN opcodes checked for the special case 0 address.
* Today, the compiler simply generates a HALT instruction at address 0. So
* a subroutine can savely return to 0, and then encounter a HALT.
*/
SC_FUNC void writeleader(symbol *root)
{
int lbl_nostate,lbl_table;
int statecount;
symbol *sym;
constvalue *fsa, *state, *stlist;
int fsa_id,listid;
char lbl_default[sNAMEMAX+1];
assert(code_idx==0);
begcseg();
stgwrite(";program exit point\n");
stgwrite("\thalt 0\n");
code_idx+=opcodes(1)+opargs(1); /* calculate code length */
/* check whether there are any functions that have states */
for (sym=root->next; sym!=NULL; sym=sym->next)
if (sym->ident==iFUNCTN && (sym->usage & uREAD)!=0 && sym->states!=NULL)
break;
if (sym==NULL)
return; /* no function has states, nothing to do next */
/* generate an error function that is called for an undefined state */
stgwrite("\n;exit point for functions called from the wrong state\n");
lbl_nostate=getlabel();
setlabel(lbl_nostate);
stgwrite("\thalt ");
outval(AMX_ERR_INVSTATE,TRUE);
code_idx+=opcodes(1)+opargs(1); /* calculate code length */
/* write the "state-selectors" table with all automatons (update the
* automatons structure too, as we are now assigning the address to
* each automaton state-selector variable)
*/
assert(glb_declared==0);
begdseg();
for (fsa=sc_automaton_tab.next; fsa!=NULL; fsa=fsa->next) {
defstorage();
stgwrite("0\t; automaton ");
if (strlen(fsa->name)==0)
stgwrite("(anonymous)");
else
stgwrite(fsa->name);
stgwrite("\n");
fsa->value=glb_declared*sizeof(cell);
glb_declared++;
} /* for */
/* write stubs and jump tables for all state functions */
begcseg();
for (sym=root->next; sym!=NULL; sym=sym->next) {
if (sym->ident==iFUNCTN && (sym->usage & uREAD)!=0 && sym->states!=NULL) {
stlist=sym->states->next;
assert(stlist!=NULL); /* there should be at least one state item */
listid=stlist->index;
assert(listid==-1 || listid>0);
if (listid==-1 && stlist->next!=NULL) {
/* first index is the "fallback", take the next one (if available) */
stlist=stlist->next;
listid=stlist->index;
} /* if */
if (listid==-1) {
/* first index is the fallback, there is no second... */
strcpy(stlist->name,"0"); /* insert dummy label number */
/* this is an error, but we postpone adding the error message until the
* function definition
*/
continue;
} /* if */
/* generate label numbers for all statelist ids */
for (stlist=sym->states->next; stlist!=NULL; stlist=stlist->next) {
assert(strlen(stlist->name)==0);
strcpy(stlist->name,itoh(getlabel()));
} /* for */
if (strcmp(sym->name,uENTRYFUNC)==0)
continue; /* do not generate stubs for this special function */
sym->addr=code_idx; /* fix the function address now */
/* get automaton id for this function */
assert(listid>0);
fsa_id=state_getfsa(listid);
assert(fsa_id>=0); /* automaton 0 exists */
fsa=automaton_findid(fsa_id);
/* count the number of states actually used; at the sane time, check
* whether there is a default state function
*/
statecount=0;
strcpy(lbl_default,itoh(lbl_nostate));
for (stlist=sym->states->next; stlist!=NULL; stlist=stlist->next) {
if (stlist->index==-1) {
assert(strlen(stlist->name)<sizeof lbl_default);
strcpy(lbl_default,stlist->name);
} else {
statecount+=state_count(stlist->index);
} /* if */
} /* for */
/* generate a stub entry for the functions */
stgwrite("\tload.pri ");
outval(fsa->value,FALSE);
stgwrite("\t; ");
stgwrite(sym->name);
stgwrite("\n");
code_idx+=opcodes(1)+opargs(1); /* calculate code length */
lbl_table=getlabel();
ffswitch(lbl_table);
/* generate the jump table */
setlabel(lbl_table);
ffcase(statecount,lbl_default,TRUE);
for (state=sc_state_tab.next; state!=NULL; state=state->next) {
if (state->index==fsa_id) {
/* find the label for this list id */
for (stlist=sym->states->next; stlist!=NULL; stlist=stlist->next) {
if (stlist->index!=-1 && state_inlist(stlist->index,(int)state->value)) {
ffcase(state->value,stlist->name,FALSE);
break;
} /* if */
} /* for */
if (stlist==NULL && strtol(lbl_default,NULL,16)==lbl_nostate)
error(230,state->name,sym->name); /* unimplemented state, no fallback */
} /* if (state belongs to automaton of function) */
} /* for (state) */
stgwrite("\n");
} /* if (is function, used & having states) */
} /* for (sym) */
}
/* writetrailer
* Not much left of this once important function.
*
* Global references: sc_stksize (referred to only)
* sc_dataalign (referred to only)
* code_idx (altered)
* glb_declared (altered)
*/
SC_FUNC void writetrailer(void)
{
assert(sc_dataalign % opcodes(1) == 0); /* alignment must be a multiple of
* the opcode size */
assert(sc_dataalign!=0);
/* pad code to align data segment */
if ((code_idx % sc_dataalign)!=0) {
begcseg();
while ((code_idx % sc_dataalign)!=0)
nooperation();
} /* if */
/* pad data segment to align the stack and the heap */
assert(litidx==0); /* literal queue should have been emptied */
assert(sc_dataalign % sizeof(cell) == 0);
if (((glb_declared*sizeof(cell)) % sc_dataalign)!=0) {
begdseg();
defstorage();
while (((glb_declared*sizeof(cell)) % sc_dataalign)!=0) {
stgwrite("0 ");
glb_declared++;
} /* while */
} /* if */
stgwrite("\nSTKSIZE "); /* write stack size (align stack top) */
outval(sc_stksize - (sc_stksize % sc_dataalign), TRUE);
}
/*
* Start (or restart) the CODE segment.
*
* In fact, the code and data segment specifiers are purely informational;
* the "DUMP" instruction itself already specifies that the following values
* should go to the data segment. All otherinstructions go to the code
* segment.
*
* Global references: curseg
*/
SC_FUNC void begcseg(void)
{
if (curseg!=sIN_CSEG) {
stgwrite("\n");
stgwrite("CODE\t; ");
outval(code_idx,TRUE);
curseg=sIN_CSEG;
} /* endif */
}
/*
* Start (or restart) the DATA segment.
*
* Global references: curseg
*/
SC_FUNC void begdseg(void)
{
if (curseg!=sIN_DSEG) {
stgwrite("\n");
stgwrite("DATA\t; ");
outval(glb_declared-litidx,TRUE);
curseg=sIN_DSEG;
} /* if */
}
SC_FUNC void setline(int chkbounds)
{
if ((sc_debug & sSYMBOLIC)!=0 || (chkbounds && (sc_debug & sCHKBOUNDS)!=0)) {
/* generate a "break" (start statement) opcode rather than a "line" opcode
* because earlier versions of Small/Pawn have an incompatible version of the
* line opcode
*/
stgwrite("\tbreak\t; ");
outval(code_idx,TRUE);
code_idx+=opcodes(1);
} /* if */
}
SC_FUNC void setfiledirect(char *name)
{
if (sc_status==statFIRST && sc_listing) {
assert(name!=NULL);
pc_writeasm(outf,"#file ");
pc_writeasm(outf,name);
pc_writeasm(outf,"\n");
} /* if */
}
SC_FUNC void setlinedirect(int line)
{
if (sc_status==statFIRST && sc_listing) {
char string[40];
sprintf(string,"#line %d\n",line);
pc_writeasm(outf,string);
} /* if */
}
/* setlabel
*
* Post a code label (specified as a number), on a new line.
*/
SC_FUNC void setlabel(int number)
{
assert(number>=0);
stgwrite("l.");
stgwrite((char *)itoh(number));
/* To assist verification of the assembled code, put the address of the
* label as a comment. However, labels that occur inside an expression
* may move (through optimization or through re-ordering). So write the
* address only if it is known to accurate.
*/
if (!staging) {
stgwrite("\t\t; ");
outval(code_idx,FALSE);
} /* if */
stgwrite("\n");
}
/* Write a token that signifies the start or end of an expression or special
* statement. This allows several simple optimizations by the peephole
* optimizer.
*/
SC_FUNC void markexpr(optmark type,const char *name,cell offset)
{
switch (type) {
case sEXPR:
stgwrite("\t;$exp\n");
break;
case sPARM:
stgwrite("\t;$par\n");
break;
case sLDECL:
assert(name!=NULL);
stgwrite("\t;$lcl ");
stgwrite(name);
stgwrite(" ");
outval(offset,TRUE);
break;
default:
assert(0);
} /* switch */
}
/* startfunc - declare a CODE entry point (function start)
*
* Global references: funcstatus (referred to only)
*/
SC_FUNC void startfunc(char *fname)
{
stgwrite("\tproc");
if (sc_asmfile) {
char symname[2*sNAMEMAX+16];
funcdisplayname(symname,fname);
stgwrite("\t; ");
stgwrite(symname);
} /* if */
stgwrite("\n");
code_idx+=opcodes(1);
}
/* endfunc
*
* Declare a CODE ending point (function end)
*/
SC_FUNC void endfunc(void)
{
stgwrite("\n"); /* skip a line */
}
/* alignframe
*
* Aligns the frame (and the stack) of the current function to a multiple
* of the specified byte count. Two caveats: the alignment ("numbytes") should
* be a power of 2, and this alignment must be done right after the frame
* is set up (before the first variable is declared)
*/
SC_FUNC void alignframe(int numbytes)
{
#if !defined NDEBUG
/* "numbytes" should be a power of 2 for this code to work */
int i,count=0;
for (i=0; i<sizeof numbytes*8; i++)
if (numbytes & (1 << i))
count++;
assert(count==1);
#endif
stgwrite("\tlctrl 4\n"); /* get STK in PRI */
stgwrite("\tconst.alt "); /* get ~(numbytes-1) in ALT */
outval(~(numbytes-1),TRUE);
stgwrite("\tand\n"); /* PRI = STK "and" ~(numbytes-1) */
stgwrite("\tsctrl 4\n"); /* set the new value of STK ... */
stgwrite("\tsctrl 5\n"); /* ... and FRM */
code_idx+=opcodes(5)+opargs(4);
}
SC_FUNC void load_i()
{
stgwrite("\tload.i\n");
code_idx+=opcodes(1);
}
/* rvalue
*
* Generate code to get the value of a symbol into "primary".
*/
SC_FUNC void rvalue(value *lval)
{
symbol *sym;
sym=lval->sym;
if (lval->ident==iARRAYCELL) {
/* indirect fetch, address already in PRI */
load_i();
} else if (lval->ident==iARRAYCHAR) {
/* indirect fetch of a character from a pack, address already in PRI */
stgwrite("\tlodb.i ");
outval(sCHARBITS/8,TRUE); /* read one or two bytes */
code_idx+=opcodes(1)+opargs(1);
} else if (lval->ident==iREFERENCE) {
/* indirect fetch, but address not yet in PRI */
assert(sym!=NULL);
assert(sym->vclass==sLOCAL);/* global references don't exist in Pawn */
if (sym->vclass==sLOCAL)
stgwrite("\tlref.s.pri ");
else
stgwrite("\tlref.pri ");
outval(sym->addr,TRUE);
markusage(sym,uREAD);
code_idx+=opcodes(1)+opargs(1);
} else {
/* direct or stack relative fetch */
assert(sym!=NULL);
if (sym->vclass==sLOCAL)
stgwrite("\tload.s.pri ");
else
stgwrite("\tload.pri ");
outval(sym->addr,TRUE);
markusage(sym,uREAD);
code_idx+=opcodes(1)+opargs(1);
} /* if */
}
/* dereference
*
* Get a cell from a memory address stored in the primary register
*/
SC_FUNC void dereference(void)
{
stgwrite("\tload.i\n");
code_idx+=opcodes(1);
}
/* Get the address of a symbol into the primary or alternate register (used
* for arrays, and for passing arguments by reference).
*/
SC_FUNC void address(symbol *sym,regid reg)
{
assert(sym!=NULL);
assert(reg==sPRI || reg==sALT);
/* the symbol can be a local array, a global array, or an array
* that is passed by reference.
*/
if (sym->ident==iREFARRAY || sym->ident==iREFERENCE) {
/* reference to a variable or to an array; currently this is
* always a local variable */
switch (reg) {
case sPRI:
stgwrite("\tload.s.pri ");
break;
case sALT:
stgwrite("\tload.s.alt ");
break;
} /* switch */
} else {
/* a local array or local variable */
switch (reg) {
case sPRI:
if (sym->vclass==sLOCAL)
stgwrite("\taddr.pri ");
else
stgwrite("\tconst.pri ");
break;
case sALT:
if (sym->vclass==sLOCAL)
stgwrite("\taddr.alt ");
else
stgwrite("\tconst.alt ");
break;
} /* switch */
} /* if */
outval(sym->addr,TRUE);
markusage(sym,uREAD);
code_idx+=opcodes(1)+opargs(1);
}
static void addr_reg(int val, regid reg)
{
if (reg == sPRI)
stgwrite("\taddr.pri ");
else
stgwrite("\taddr.alt ");
outval(val, TRUE);
code_idx += opcodes(1) + opargs(1);
}
// Load the number of arguments into PRI. Frame layout:
// base + 0*sizeof(cell) == previous "base"
// base + 1*sizeof(cell) == function return address
// base + 2*sizeof(cell) == number of arguments
// base + 3*sizeof(cell) == first argument of the function
static void load_argcount(regid reg)
{
if (reg == sPRI)
stgwrite("\tload.s.pri ");
else
stgwrite("\tload.s.alt ");
outval(2 * sizeof(cell), TRUE);
code_idx += opcodes(1) + opargs(1);
}
// Load the hidden array argument into ALT.
SC_FUNC void load_hidden_arg()
{
pushreg(sPRI);
// Compute an address to the first argument, then add the argument count
// to find the address after the final argument:
// addr.alt 0xc ; Compute &first_arg
// load.s.alt 0x8 ; Load arg count in bytes
// add ; Compute (&first_arg) + argcount
// load.i ; Load *(&first_arg + argcount)
// move.alt ; Move result into ALT.
addr_reg(0xc, sALT);
load_argcount(sPRI);
ob_add();
load_i();
move_alt();
popreg(sPRI);
}
/* store
*
* Saves the contents of "primary" into a memory cell, either directly
* or indirectly (at the address given in the alternate register).
*/
SC_FUNC void store(value *lval)
{
symbol *sym;
sym=lval->sym;
if (lval->ident==iARRAYCELL) {
/* store at address in ALT */
stgwrite("\tstor.i\n");
code_idx+=opcodes(1);
} else if (lval->ident==iARRAYCHAR) {
/* store at address in ALT */
stgwrite("\tstrb.i ");
outval(sCHARBITS/8,TRUE); /* write one or two bytes */
code_idx+=opcodes(1)+opargs(1);
} else if (lval->ident==iREFERENCE) {
assert(sym!=NULL);
if (sym->vclass==sLOCAL)
stgwrite("\tsref.s.pri ");
else
stgwrite("\tsref.pri ");
outval(sym->addr,TRUE);
code_idx+=opcodes(1)+opargs(1);
} else {
assert(sym!=NULL);
markusage(sym,uWRITTEN);
if (sym->vclass==sLOCAL)
stgwrite("\tstor.s.pri ");
else
stgwrite("\tstor.pri ");
outval(sym->addr,TRUE);
code_idx+=opcodes(1)+opargs(1);
} /* if */
}
SC_FUNC void storereg(cell address,regid reg)
{
assert(reg==sPRI || reg==sALT);
if (reg==sPRI)
stgwrite("\tstor.pri ");
else
stgwrite("\tstor.alt ");
outval(address,TRUE);
code_idx+=opcodes(1)+opargs(1);
}
/* source must in PRI, destination address in ALT. The "size"
* parameter is in bytes, not cells.
*/
SC_FUNC void memcopy(cell size)
{
stgwrite("\tmovs ");
outval(size,TRUE);
code_idx+=opcodes(1)+opargs(1);
}
/* Address of the source must already have been loaded in PRI
* "size" is the size in bytes (not cells).
*/
SC_FUNC void copyarray(symbol *sym,cell size)
{
assert(sym!=NULL);
/* the symbol can be a local array, a global array, or an array
* that is passed by reference.
*/
if (sym->ident==iREFARRAY) {
/* reference to an array; currently this is always a local variable */
assert(sym->vclass==sLOCAL); /* symbol must be stack relative */
stgwrite("\tload.s.alt ");
} else {
/* a local or global array */
if (sym->vclass==sLOCAL)
stgwrite("\taddr.alt ");
else
stgwrite("\tconst.alt ");
} /* if */
outval(sym->addr,TRUE);
markusage(sym,uWRITTEN);
code_idx+=opcodes(1)+opargs(1);
memcopy(size);
}
SC_FUNC void fillarray(symbol *sym,cell size,cell value)
{
ldconst(value,sPRI); /* load value in PRI */
assert(sym!=NULL);
/* the symbol can be a local array, a global array, or an array
* that is passed by reference.
*/
if (sym->ident==iREFARRAY) {
/* reference to an array; currently this is always a local variable */
assert(sym->vclass==sLOCAL); /* symbol must be stack relative */
stgwrite("\tload.s.alt ");
} else {
/* a local or global array */
if (sym->vclass==sLOCAL)
stgwrite("\taddr.alt ");
else
stgwrite("\tconst.alt ");
} /* if */
outval(sym->addr,TRUE);
markusage(sym,uWRITTEN);
assert(size>0);
stgwrite("\tfill ");
outval(size,TRUE);
code_idx+=opcodes(2)+opargs(2);
}
/* Instruction to get an immediate value into the primary or the alternate
* register
*/
SC_FUNC void ldconst(cell val,regid reg)
{
assert(reg==sPRI || reg==sALT);
switch (reg) {
case sPRI:
if (val==0) {
stgwrite("\tzero.pri\n");
code_idx+=opcodes(1);
} else {
stgwrite("\tconst.pri ");
outval(val, TRUE);
code_idx+=opcodes(1)+opargs(1);
} /* if */
break;
case sALT:
if (val==0) {
stgwrite("\tzero.alt\n");
code_idx+=opcodes(1);
} else {
stgwrite("\tconst.alt ");
outval(val, TRUE);
code_idx+=opcodes(1)+opargs(1);
} /* if */
break;
} /* switch */
}
/* Copy value in alternate register to the primary register */
SC_FUNC void moveto1(void)
{
stgwrite("\tmove.pri\n");
code_idx+=opcodes(1)+opargs(0);
}
SC_FUNC void move_alt(void)
{
stgwrite("\tmove.alt\n");
code_idx+=opcodes(1)+opargs(0);
}
/* Push primary or the alternate register onto the stack
*/
SC_FUNC void pushreg(regid reg)
{
assert(reg==sPRI || reg==sALT);
switch (reg) {
case sPRI:
stgwrite("\tpush.pri\n");
break;
case sALT:
stgwrite("\tpush.alt\n");
break;
} /* switch */
code_idx+=opcodes(1);
}
/*
* Push a constant value onto the stack
*/
SC_FUNC void pushval(cell val)
{
stgwrite("\tpush.c ");
outval(val, TRUE);
code_idx+=opcodes(1)+opargs(1);
}
/* Pop stack into the primary or the alternate register
*/
SC_FUNC void popreg(regid reg)
{
assert(reg==sPRI || reg==sALT);
switch (reg) {
case sPRI:
stgwrite("\tpop.pri\n");
break;
case sALT:
stgwrite("\tpop.alt\n");
break;
} /* switch */
code_idx+=opcodes(1);
}
/*
* swap the top-of-stack with the value in primary register
*/
SC_FUNC void swap1(void)
{
stgwrite("\tswap.pri\n");
code_idx+=opcodes(1);
}
/* Switch statements
* The "switch" statement generates a "case" table using the "CASE" opcode.
* The case table contains a list of records, each record holds a comparison
* value and a label to branch to on a match. The very first record is an
* exception: it holds the size of the table (excluding the first record) and
* the label to branch to when none of the values in the case table match.
* The case table is sorted on the comparison value. This allows more advanced
* abstract machines to sift the case table with a binary search.
*/
SC_FUNC void ffswitch(int label)
{
stgwrite("\tswitch ");
outval(label,TRUE); /* the label is the address of the case table */
code_idx+=opcodes(1)+opargs(1);
}
SC_FUNC void ffcase(cell value,char *labelname,int newtable)
{
if (newtable) {
stgwrite("\tcasetbl\n");
code_idx+=opcodes(1);
} /* if */
stgwrite("\tcase ");
outval(value,FALSE);
stgwrite(" ");
stgwrite(labelname);
stgwrite("\n");
code_idx+=opcodes(0)+opargs(2);
}
/*
* Call specified function
*/
SC_FUNC void ffcall(symbol *sym,const char *label,int numargs)
{
char symname[2*sNAMEMAX+16];
assert(sym!=NULL);
assert(sym->ident==iFUNCTN);
if (sc_asmfile)
funcdisplayname(symname,sym->name);
if ((sym->usage & uNATIVE)!=0) {
/* reserve a SYSREQ id if called for the first time */
assert(label==NULL);
if (sc_status==statWRITE && (sym->usage & uREAD)==0 && sym->addr>=0)
sym->addr=ntv_funcid++;
stgwrite("\tsysreq.c ");
outval(sym->addr,FALSE);
if (sc_asmfile) {
stgwrite("\t; ");
stgwrite(symname);
} /* if */
stgwrite("\n\tstack ");
outval((numargs+1)*sizeof(cell), TRUE);
code_idx+=opcodes(2)+opargs(2);
} else {
/* normal function */
stgwrite("\tcall ");
if (label!=NULL) {
stgwrite("l.");
stgwrite(label);
} else {
stgwrite(sym->name);
} /* if */
if (sc_asmfile
&& (label!=NULL || (!isalpha(sym->name[0]) && sym->name[0]!='_' && sym->name[0]!=sc_ctrlchar)))
{
stgwrite("\t; ");
stgwrite(symname);
} /* if */
stgwrite("\n");
code_idx+=opcodes(1)+opargs(1);
} /* if */
}
/* Return from function
*
* Global references: funcstatus (referred to only)
*/
SC_FUNC void ffret(void)
{
stgwrite("\tretn\n");
code_idx+=opcodes(1);
}
SC_FUNC void ffabort(int reason)
{
stgwrite("\thalt ");
outval(reason,TRUE);
code_idx+=opcodes(1)+opargs(1);
}
SC_FUNC void ffbounds(cell size)
{
if ((sc_debug & sCHKBOUNDS)!=0) {
stgwrite("\tbounds ");
outval(size,TRUE);
code_idx+=opcodes(1)+opargs(1);
} /* if */
}
/*
* Jump to local label number (the number is converted to a name)
*/
SC_FUNC void jumplabel(int number)
{
stgwrite("\tjump ");
outval(number,TRUE);
code_idx+=opcodes(1)+opargs(1);
}
/*
* Define storage (global and static variables)
*/
SC_FUNC void defstorage(void)
{
stgwrite("dump ");
}
/*
* Copies frame address to primary register
*/
SC_FUNC void getfrm(void)
{
stgwrite("\tlctrl 5\n");
code_idx+=opcodes(1)+opargs(1);
}
/*
/*
* Inclrement/decrement stack pointer. Note that this routine does
* nothing if the delta is zero.
*/
SC_FUNC void modstk(int delta)
{
if (delta) {
stgwrite("\tstack ");
outval(delta, TRUE);
code_idx+=opcodes(1)+opargs(1);
} /* if */
}
/* set the stack to a hard offset from the frame */
SC_FUNC void setstk(cell value)
{
stgwrite("\tlctrl 5\n"); /* get FRM in PRI */
assert(value<=0); /* STK should always become <= FRM */
if (value<0) {
stgwrite("\tadd.c ");
outval(value, TRUE); /* add (negative) offset */
code_idx+=opcodes(1)+opargs(1);
// ??? write zeros in the space between STK and the value in PRI (the new stk)
// get value of STK in ALT
// zero PRI
// need new FILL opcode that takes a variable size
} /* if */
stgwrite("\tsctrl 4\n"); /* store in STK */
code_idx+=opcodes(2)+opargs(2);
}
SC_FUNC void modheap(int delta)
{
if (delta) {
stgwrite("\theap ");
outval(delta, TRUE);
code_idx+=opcodes(1)+opargs(1);
} /* if */
}
SC_FUNC void setheap_pri(void)
{
stgwrite("\theap "); /* ALT = HEA++ */
outval(sizeof(cell), TRUE);
stgwrite("\tstor.i\n"); /* store PRI (default value) at address ALT */
stgwrite("\tmove.pri\n"); /* move ALT to PRI: PRI contains the address */
code_idx+=opcodes(3)+opargs(1);
}
SC_FUNC void setheap(cell value)
{
stgwrite("\tconst.pri "); /* load default value in PRI */
outval(value, TRUE);
code_idx+=opcodes(1)+opargs(1);
setheap_pri();
}
/*
* Convert a cell number to a "byte" address; i.e. double or quadruple
* the primary register.
*/
SC_FUNC void cell2addr(void)
{
#if PAWN_CELL_SIZE==16
stgwrite("\tshl.c.pri 1\n");
#elif PAWN_CELL_SIZE==32
stgwrite("\tshl.c.pri 2\n");
#elif PAWN_CELL_SIZE==64
stgwrite("\tshl.c.pri 3\n");
#else
#error Unsupported cell size
#endif
code_idx+=opcodes(1)+opargs(1);
}
/*
* Double or quadruple the alternate register.
*/
SC_FUNC void cell2addr_alt(void)
{
#if PAWN_CELL_SIZE==16
stgwrite("\tshl.c.alt 1\n");
#elif PAWN_CELL_SIZE==32
stgwrite("\tshl.c.alt 2\n");
#elif PAWN_CELL_SIZE==64
stgwrite("\tshl.c.alt 3\n");
#else
#error Unsupported cell size
#endif
code_idx+=opcodes(1)+opargs(1);
}
/*
* Convert "distance of addresses" to "number of cells" in between.
* Or convert a number of packed characters to the number of cells (with
* truncation).
*/
SC_FUNC void addr2cell(void)
{
#if PAWN_CELL_SIZE==16
stgwrite("\tshr.c.pri 1\n");
#elif PAWN_CELL_SIZE==32
stgwrite("\tshr.c.pri 2\n");
#elif PAWN_CELL_SIZE==64
stgwrite("\tshr.c.pri 3\n");
#else
#error Unsupported cell size
#endif
code_idx+=opcodes(1)+opargs(1);
}
/* Convert from character index to byte address. This routine does
* nothing if a character has the size of a byte.
*/
SC_FUNC void char2addr(void)
{
#if sCHARBITS==16
stgwrite("\tshl.c.pri 1\n");
code_idx+=opcodes(1)+opargs(1);
#endif
}
/* Align PRI (which should hold a character index) to an address.
* The first character in a "pack" occupies the highest bits of
* the cell. This is at the lower memory address on Big Endian
* computers and on the higher address on Little Endian computers.
* The ALIGN.pri/alt instructions must solve this machine dependence;
* that is, on Big Endian computers, ALIGN.pri/alt shuold do nothing
* and on Little Endian computers they should toggle the address.
*/
SC_FUNC void charalign(void)
{
stgwrite("\talign.pri ");
outval(sCHARBITS/8,TRUE);
code_idx+=opcodes(1)+opargs(1);
}
/*
* Add a constant to the primary register.
*/
SC_FUNC void addconst(cell value)
{
if (value!=0) {
stgwrite("\tadd.c ");
outval(value,TRUE);
code_idx+=opcodes(1)+opargs(1);
} /* if */
}
/*
* signed multiply of primary and secundairy registers (result in primary)
*/
SC_FUNC void os_mult(void)
{
stgwrite("\tsmul\n");
code_idx+=opcodes(1);
}
/*
* signed divide of alternate register by primary register (quotient in
* primary; remainder in alternate)
*/
SC_FUNC void os_div(void)
{
stgwrite("\tsdiv.alt\n");
code_idx+=opcodes(1);
}
/*
* modulus of (alternate % primary), result in primary (signed)
*/
SC_FUNC void os_mod(void)
{
stgwrite("\tsdiv.alt\n");
stgwrite("\tmove.pri\n"); /* move ALT to PRI */
code_idx+=opcodes(2);
}
/*
* Add primary and alternate registers (result in primary).
*/
SC_FUNC void ob_add(void)
{
stgwrite("\tadd\n");
code_idx+=opcodes(1);
}
/*
* subtract primary register from alternate register (result in primary)
*/
SC_FUNC void ob_sub(void)
{
stgwrite("\tsub.alt\n");
code_idx+=opcodes(1);
}
/*
* arithmic shift left alternate register the number of bits
* given in the primary register (result in primary).
* There is no need for a "logical shift left" routine, since
* logical shift left is identical to arithmic shift left.
*/
SC_FUNC void ob_sal(void)
{
stgwrite("\txchg\n");
stgwrite("\tshl\n");
code_idx+=opcodes(2);
}
/*
* arithmic shift right alternate register the number of bits
* given in the primary register (result in primary).
*/
SC_FUNC void os_sar(void)
{
stgwrite("\txchg\n");
stgwrite("\tsshr\n");
code_idx+=opcodes(2);
}
/*
* logical (unsigned) shift right of the alternate register by the
* number of bits given in the primary register (result in primary).
*/
SC_FUNC void ou_sar(void)
{
stgwrite("\txchg\n");
stgwrite("\tshr\n");
code_idx+=opcodes(2);
}
/*
* inclusive "or" of primary and alternate registers (result in primary)
*/
SC_FUNC void ob_or(void)
{
stgwrite("\tor\n");
code_idx+=opcodes(1);
}
/*
* "exclusive or" of primary and alternate registers (result in primary)
*/
SC_FUNC void ob_xor(void)
{
stgwrite("\txor\n");
code_idx+=opcodes(1);
}
/*
* "and" of primary and secundairy registers (result in primary)
*/
SC_FUNC void ob_and(void)
{
stgwrite("\tand\n");
code_idx+=opcodes(1);
}
/*
* test ALT==PRI; result in primary register (1 or 0).
*/
SC_FUNC void ob_eq(void)
{
stgwrite("\teq\n");
code_idx+=opcodes(1);
}
/*
* test ALT!=PRI
*/
SC_FUNC void ob_ne(void)
{
stgwrite("\tneq\n");
code_idx+=opcodes(1);
}
/* The abstract machine defines the relational instructions so that PRI is
* on the left side and ALT on the right side of the operator. For example,
* SLESS sets PRI to either 1 or 0 depending on whether the expression
* "PRI < ALT" is true.
*
* The compiler generates comparisons with ALT on the left side of the
* relational operator and PRI on the right side. The XCHG instruction
* prefixing the relational operators resets this. We leave it to the
* peephole optimizer to choose more compact instructions where possible.
*/
/* Relational operator prefix for chained relational expressions. The
* "suffix" code restores the stack.
* For chained relational operators, the goal is to keep the comparison
* result "so far" in PRI and the value of the most recent operand in
* ALT, ready for a next comparison.
* The "prefix" instruction pushed the comparison result (PRI) onto the
* stack and moves the value of ALT into PRI. If there is a next comparison,
* PRI can now serve as the "left" operand of the relational operator.
*/
SC_FUNC void relop_prefix(void)
{
stgwrite("\tpush.pri\n");
stgwrite("\tmove.pri\n");
code_idx+=opcodes(2);
}
SC_FUNC void relop_suffix(void)
{
stgwrite("\tswap.alt\n");
stgwrite("\tand\n");
stgwrite("\tpop.alt\n");
code_idx+=opcodes(3);
}
/*
* test ALT<PRI (signed)
*/
SC_FUNC void os_lt(void)
{
stgwrite("\txchg\n");
stgwrite("\tsless\n");
code_idx+=opcodes(2);
}
/*
* test ALT<=PRI (signed)
*/
SC_FUNC void os_le(void)
{
stgwrite("\txchg\n");
stgwrite("\tsleq\n");
code_idx+=opcodes(2);
}
/*
* test ALT>PRI (signed)
*/
SC_FUNC void os_gt(void)
{
stgwrite("\txchg\n");
stgwrite("\tsgrtr\n");
code_idx+=opcodes(2);
}
/*
* test ALT>=PRI (signed)
*/
SC_FUNC void os_ge(void)
{
stgwrite("\txchg\n");
stgwrite("\tsgeq\n");
code_idx+=opcodes(2);
}
/*
* logical negation of primary register
*/
SC_FUNC void lneg(void)
{
stgwrite("\tnot\n");
code_idx+=opcodes(1);
}
/*
* two's complement primary register
*/
SC_FUNC void neg(void)
{
stgwrite("\tneg\n");
code_idx+=opcodes(1);
}
/*
* one's complement of primary register
*/
SC_FUNC void invert(void)
{
stgwrite("\tinvert\n");
code_idx+=opcodes(1);
}
/*
* nop
*/
SC_FUNC void nooperation(void)
{
stgwrite("\tnop\n");
code_idx+=opcodes(1);
}
/* increment symbol
*/
SC_FUNC void inc(value *lval)
{
symbol *sym;
sym=lval->sym;
if (lval->ident==iARRAYCELL) {
/* indirect increment, address already in PRI */
stgwrite("\tinc.i\n");
code_idx+=opcodes(1);
} else if (lval->ident==iARRAYCHAR) {
/* indirect increment of single character, address already in PRI */
stgwrite("\tpush.pri\n");
stgwrite("\tpush.alt\n");
stgwrite("\tmove.alt\n"); /* copy address */
stgwrite("\tlodb.i "); /* read from PRI into PRI */
outval(sCHARBITS/8,TRUE); /* read one or two bytes */
stgwrite("\tinc.pri\n");
stgwrite("\tstrb.i "); /* write PRI to ALT */
outval(sCHARBITS/8,TRUE); /* write one or two bytes */
stgwrite("\tpop.alt\n");
stgwrite("\tpop.pri\n");
code_idx+=opcodes(8)+opargs(2);
} else if (lval->ident==iREFERENCE) {
assert(sym!=NULL);
stgwrite("\tpush.pri\n");
/* load dereferenced value */
assert(sym->vclass==sLOCAL); /* global references don't exist in Pawn */
if (sym->vclass==sLOCAL)
stgwrite("\tlref.s.pri ");
else
stgwrite("\tlref.pri ");
outval(sym->addr,TRUE);
/* increment */
stgwrite("\tinc.pri\n");
/* store dereferenced value */
if (sym->vclass==sLOCAL)
stgwrite("\tsref.s.pri ");
else
stgwrite("\tsref.pri ");
outval(sym->addr,TRUE);
stgwrite("\tpop.pri\n");
code_idx+=opcodes(5)+opargs(2);
} else {
/* local or global variable */
assert(sym!=NULL);
if (sym->vclass==sLOCAL)
stgwrite("\tinc.s ");
else
stgwrite("\tinc ");
outval(sym->addr,TRUE);
code_idx+=opcodes(1)+opargs(1);
} /* if */
}
/* decrement symbol
*
* in case of an integer pointer, the symbol must be incremented by 2.
*/
SC_FUNC void dec(value *lval)
{
symbol *sym;
sym=lval->sym;
if (lval->ident==iARRAYCELL) {
/* indirect decrement, address already in PRI */
stgwrite("\tdec.i\n");
code_idx+=opcodes(1);
} else if (lval->ident==iARRAYCHAR) {
/* indirect decrement of single character, address already in PRI */
stgwrite("\tpush.pri\n");
stgwrite("\tpush.alt\n");
stgwrite("\tmove.alt\n"); /* copy address */
stgwrite("\tlodb.i "); /* read from PRI into PRI */
outval(sCHARBITS/8,TRUE); /* read one or two bytes */
stgwrite("\tdec.pri\n");
stgwrite("\tstrb.i "); /* write PRI to ALT */
outval(sCHARBITS/8,TRUE); /* write one or two bytes */
stgwrite("\tpop.alt\n");
stgwrite("\tpop.pri\n");
code_idx+=opcodes(8)+opargs(2);
} else if (lval->ident==iREFERENCE) {
assert(sym!=NULL);
stgwrite("\tpush.pri\n");
/* load dereferenced value */
assert(sym->vclass==sLOCAL); /* global references don't exist in Pawn */
if (sym->vclass==sLOCAL)
stgwrite("\tlref.s.pri ");
else
stgwrite("\tlref.pri ");
outval(sym->addr,TRUE);
/* decrement */
stgwrite("\tdec.pri\n");
/* store dereferenced value */
if (sym->vclass==sLOCAL)
stgwrite("\tsref.s.pri ");
else
stgwrite("\tsref.pri ");
outval(sym->addr,TRUE);
stgwrite("\tpop.pri\n");
code_idx+=opcodes(5)+opargs(2);
} else {
/* local or global variable */
assert(sym!=NULL);
if (sym->vclass==sLOCAL)
stgwrite("\tdec.s ");
else
stgwrite("\tdec ");
outval(sym->addr,TRUE);
code_idx+=opcodes(1)+opargs(1);
} /* if */
}
/*
* Jumps to "label" if PRI != 0
*/
SC_FUNC void jmp_ne0(int number)
{
stgwrite("\tjnz ");
outval(number,TRUE);
code_idx+=opcodes(1)+opargs(1);
}
/*
* Jumps to "label" if PRI == 0
*/
SC_FUNC void jmp_eq0(int number)
{
stgwrite("\tjzer ");
outval(number,TRUE);
code_idx+=opcodes(1)+opargs(1);
}
/* write a value in hexadecimal; optionally adds a newline */
SC_FUNC void outval(cell val,int newline)
{
stgwrite(itoh(val));
if (newline)
stgwrite("\n");
}