105 lines
1.7 KiB
C
105 lines
1.7 KiB
C
/* S T A C K _ C H A N G E . C */
|
|
|
|
|
|
#include <stdio.h>
|
|
#include "../share/types.h"
|
|
#include "../share/debug.h"
|
|
#include "../share/global.h"
|
|
#include "../../../h/em_spec.h"
|
|
#include "../../../h/em_mnem.h"
|
|
|
|
#include "pop_push.h"
|
|
|
|
#define IS_LOC(l) (l!=(line_p) 0 && INSTR(l)==op_loc && TYPE(l)==OPSHORT)
|
|
|
|
int stack_change(l,sign)
|
|
line_p l;
|
|
char sign;
|
|
{
|
|
/* Interpret the string in the third column of the em_table file */
|
|
|
|
char *s;
|
|
bool argdef;
|
|
short arg;
|
|
int sum = 0;
|
|
line_p p = PREV(l);
|
|
line_p pp = (p == (line_p) 0 ? (line_p) 0 : PREV(p));
|
|
short i = INSTR(l);
|
|
|
|
if (i < sp_fmnem || i > sp_lmnem) {
|
|
return 0;
|
|
} else {
|
|
if (TYPE(l) == OPSHORT) {
|
|
arg = SHORT(l);
|
|
if (arg < ws) {
|
|
/* E.g. a LOI 1 loads word-size bytes,
|
|
* not 1 byte!
|
|
*/
|
|
arg = ws;
|
|
}
|
|
argdef = TRUE;
|
|
} else {
|
|
argdef = FALSE;
|
|
}
|
|
}
|
|
s = pop_push[i];
|
|
if (*s == '0') return 0;
|
|
while (*s != '\0') {
|
|
if (*s++ == sign) {
|
|
switch(*s) {
|
|
case 'w':
|
|
sum += ws;
|
|
break;
|
|
case 'd':
|
|
sum += 2 * ws;
|
|
break;
|
|
case 'p':
|
|
sum += ps;
|
|
break;
|
|
case 'a':
|
|
if (!argdef) return -1;
|
|
sum += arg;
|
|
break;
|
|
case 'x':
|
|
if (IS_LOC(p)) {
|
|
sum += SHORT(p);
|
|
break;
|
|
} else {
|
|
return -1;
|
|
}
|
|
case 'y':
|
|
if (IS_LOC(pp)) {
|
|
sum += SHORT(pp);
|
|
break;
|
|
} else {
|
|
return -1;
|
|
}
|
|
case '?':
|
|
return -1;
|
|
default:
|
|
assert(FALSE);
|
|
}
|
|
}
|
|
s++;
|
|
}
|
|
return sum;
|
|
}
|
|
|
|
|
|
|
|
line_change(l,ok_out,pop_out,push_out)
|
|
line_p l;
|
|
bool *ok_out;
|
|
int *pop_out,*push_out;
|
|
{
|
|
short pop,push;
|
|
|
|
pop = stack_change(l,'-');
|
|
push = stack_change(l,'+');
|
|
*ok_out = (pop != -1 && push != -1);
|
|
*pop_out = pop;
|
|
*push_out = push;
|
|
}
|
|
|
|
|