ack/util/ego/share/stack_chg.c

104 lines
1.8 KiB
C
Raw Permalink Normal View History

1994-06-24 11:31:16 +00:00
/* $Id$ */
1987-03-09 19:15:41 +00:00
/*
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
* See the copyright notice in the ACK home directory, in the file "Copyright".
*/
1984-11-26 15:04:22 +00:00
/* S T A C K _ C H A N G E . C */
#include <stdio.h>
1991-03-05 12:44:05 +00:00
#include <em_spec.h>
#include <em_mnem.h>
#include "types.h"
#include "debug.h"
#include "global.h"
1984-11-26 15:04:22 +00:00
#include "pop_push.h"
#define IS_LOC(l) (l!=(line_p) 0 && INSTR(l)==op_loc && TYPE(l)==OPSHORT)
STATIC int stack_change(line_p l, char sign)
1984-11-26 15:04:22 +00:00
{
/* Interpret the string in the third column of the em_table file */
char *s;
bool argdef;
1993-11-16 11:35:23 +00:00
short arg = 0;
1984-11-26 15:04:22 +00:00
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;
1993-11-16 11:35:23 +00:00
}
if (TYPE(l) == OPSHORT) {
arg = SHORT(l);
if (arg < ws) {
/* E.g. a LOI 1 loads word-size bytes,
* not 1 byte!
*/
arg = ws;
1984-11-26 15:04:22 +00:00
}
1993-11-16 11:35:23 +00:00
argdef = TRUE;
} else {
argdef = FALSE;
1984-11-26 15:04:22 +00:00
}
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;
}
void line_change(line_p l, bool *ok_out, int *pop_out, int *push_out)
1984-11-26 15:04:22 +00:00
{
short pop,push;
pop = stack_change(l,'-');
push = stack_change(l,'+');
*ok_out = (pop != -1 && push != -1);
*pop_out = pop;
*push_out = push;
}