ack/util/ego/share/stack_chg.c
George Koehler 9037d137f5 Add prototypes, void in util/ego/share
This uncovers a problem in il/il_aux.c: it passes 3 arguments to
getlines(), but the function expects 4 arguments.  I add FALSE as the
4th argument.  TRUE would fill in the list of mesregs.  IL uses
mesregs during phase 1, but this call to getlines() is in phase 2.
TRUE would leak memory unless I added a call to Ldeleteset(mesregs).
So I pass FALSE.

Functions passed to go() now have a `void *` parameter because
no_action() now takes a `void *`.
2017-11-15 17:19:56 -05:00

104 lines
1.8 KiB
C

/* $Id$ */
/*
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
* See the copyright notice in the ACK home directory, in the file "Copyright".
*/
/* S T A C K _ C H A N G E . C */
#include <stdio.h>
#include <em_spec.h>
#include <em_mnem.h>
#include "types.h"
#include "debug.h"
#include "global.h"
#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)
{
/* Interpret the string in the third column of the em_table file */
char *s;
bool argdef;
short arg = 0;
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;
}
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;
}
void line_change(line_p l, bool *ok_out, int *pop_out, int *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;
}