777d0abb00
Use switch.h to reduce warnings from clang about implicit declarations of functions. I used `grep ... do_*.c | sed ... | sort`, and some manual editing, to make the big list of Do???() functions.
86 lines
1.5 KiB
C
86 lines
1.5 KiB
C
/** @file
|
|
* Sources of the "POINTER ARITHMETIC" group instructions
|
|
*/
|
|
|
|
/* $Id$ */
|
|
|
|
#include "em_abs.h"
|
|
#include "segcheck.h"
|
|
#include "global.h"
|
|
#include "segment.h"
|
|
#include "log.h"
|
|
#include "mem.h"
|
|
#include "trap.h"
|
|
#include "warn.h"
|
|
#include "text.h"
|
|
#include "fra.h"
|
|
#include "switch.h"
|
|
|
|
#define adp(p,w) ((p) + (w))
|
|
#define sbs(t,s) ((s) - (t))
|
|
|
|
#ifdef SEGCHECK
|
|
|
|
#define check_seg(s1,s2,w) if (s1 != s2) { warning(w); }
|
|
|
|
#else
|
|
|
|
#define check_seg(s1,s2,w)
|
|
|
|
#endif /* SEGCHECK */
|
|
|
|
/** ADP f: Add f to pointer on top of stack */
|
|
void DoADP(register long l)
|
|
{
|
|
|
|
register ptr p, t = st_lddp(SP);
|
|
|
|
LOG(("@R6 DoADP(%ld)", l));
|
|
spoilFRA();
|
|
if (t == 0) {
|
|
warning(WNULLPA);
|
|
}
|
|
l = arg_f(l);
|
|
p = adp(t, l);
|
|
check_seg(ptr2seg(t), ptr2seg(p), WSEGADP);
|
|
st_stdp(SP, p);
|
|
}
|
|
|
|
/** ADS w: Add w-byte value and pointer */
|
|
void DoADS(register size l)
|
|
{
|
|
register long t = spop(arg_wi(l));
|
|
register ptr p, s = st_lddp(SP);
|
|
|
|
LOG(("@R6 DoADS(%ld)", l));
|
|
spoilFRA();
|
|
t = arg_f(t);
|
|
if (s == 0) {
|
|
warning(WNULLPA);
|
|
}
|
|
p = adp(s, t);
|
|
check_seg(ptr2seg(s), ptr2seg(p), WSEGADP);
|
|
st_stdp(SP, p);
|
|
}
|
|
|
|
/** SBS w: Subtract pointers in same fragment and push diff as size w integer */
|
|
void DoSBS(register size l)
|
|
{
|
|
register ptr t = st_lddp(SP);
|
|
register ptr s = st_lddp(SP + psize);
|
|
register long w;
|
|
|
|
LOG(("@R6 DoSBS(%ld)", l));
|
|
spoilFRA();
|
|
l = arg_wi(l);
|
|
check_seg(ptr2seg(t), ptr2seg(s), WSEGSBS);
|
|
w = sbs(t, s);
|
|
if (must_test && !(IgnMask&BIT(EIOVFL))) {
|
|
if (l == 2 && (w < I_MINS2 || w > I_MAXS2))
|
|
trap(EIOVFL);
|
|
}
|
|
dppop();
|
|
dppop();
|
|
npush(w, l);
|
|
}
|