/** @file Data access routines */ /* $Id$ */ #include "em_abs.h" #include "logging.h" #include "nofloat.h" #include "global.h" #include "log.h" #include "trap.h" #include "warn.h" #include "alloc.h" #include "memdirect.h" #include "mem.h" #include "shadow.h" /** Initial heap size in bytes. */ #define HEAPSIZE 1000L extern size maxheap; /* from main.c */ #ifdef LOGGING char *data_sh; /* shadowbytes */ PRIVATE void warn_dtbits(ptr, size); PRIVATE void dt_clear_area(ptr, ptr ); #endif /* LOGGING */ /** Initialize the heap with "hb" address. */ void init_data(ptr hb) { HB = hb; /* set Heap Base */ HP = HB; /* initialize Heap Pointer */ HL = HB + HEAPSIZE; /* initialize Heap Limit */ data = Malloc((size)p2i(HL), "data space"); #ifdef LOGGING data_sh = Malloc((size)p2i(HL), "shadowspace for data"); dt_clear_area(i2p(0), HL); #endif /* LOGGING */ } /******************************************************** * EM-register division. * ******************************************************** * * * newHP(p) - check and adjust HeapPointer. * * * ********************************************************/ /** Grows the heap space with the new heap pointer. */ void newHP(ptr ap) { register ptr p = ap; if (in_gda(p)) { wtrap(WHPGDA, EHEAP); } if (in_stack(p)) { wtrap(WHPSTACK, EHEAP); } if (!is_wordaligned(p)) { wtrap(WHPODD, EHEAP); } if (maxheap) { /* more than allowed on command line */ if (p - HB > maxheap) { warning(WEHEAP); trap(EHEAP); } } if (p > HL) { /* extend heap space */ HL = i2p(allocfrac(p2i(p)) - 1); data = Realloc(data, (size)(p2i(HL) + 1), "heap space"); #ifdef LOGGING data_sh = Realloc(data_sh, (size)(p2i(HL) + 1), "shadowspace for heap"); #endif /* LOGGING */ } #ifdef LOGGING if (p > HP) { dt_clear_area(HP, p); } #endif /* LOGGING */ HP = p; } /************************************************************************ * Data store division. * ************************************************************************ * * * dt_stdp(addr, p) - STore Data Pointer. * * dt_stn(addr, l, n) - STore N byte integer. * * dt_stw(addr, l) - STore wsize byte integer. * * dt_stf(addr, f, n) - STore n byte Floating point number. * * * ************************************************************************/ /** Store data pointer "ap" at address "addr". */ void dt_stdp(register ptr addr, ptr ap) { register int i; register long p = (long) ap; LOG(("@g6 dt_stdp(%lu, %lu)", addr, p)); ch_in_data(addr, psize); ch_wordaligned(addr); for (i = (int) psize; i > 0; i--, addr++) { ch_dt_prot(addr); data_loc(addr) = (char) (p); dt_dp(addr); p = p>>8; } } void dt_stip(register ptr addr, ptr ap) { register int i; register long p = (long) ap; LOG(("@g6 dt_stip(%lu, %lu)", addr, p)); ch_in_data(addr, psize); ch_wordaligned(addr); for (i = (int) psize; i > 0; i--, addr++) { ch_dt_prot(addr); data_loc(addr) = (char) (p); dt_ip(addr); p = p>>8; } } /** Store "n" byte integer "al" at address "addr". */ void dt_stn(register ptr addr, long al, size n) { register int i; register long l = al; #ifdef LOGGING /* a psize zero is ambiguous */ int sh_flags = (l == 0 && n == psize) ? (SH_INT|SH_DATAP) : SH_INT; #endif /* LOGGING */ LOG(("@g6 dt_stn(%lu, %lu, %lu)", addr, l, n)); ch_in_data(addr, n); ch_aligned(addr, n); for (i = (int) n; i > 0; i--, addr++) { ch_dt_prot(addr); data_loc(addr) = (char) l; #ifdef LOGGING dt_sh(addr) = sh_flags; #endif /* LOGGING */ l = l>>8; } } /** Store word sized integer "al" at address. */ void dt_stw(register ptr addr, long al) { register int i; register long l = al; #ifdef LOGGING /* a psize zero is ambiguous */ int sh_flags = (l == 0 && wsize == psize) ? (SH_INT|SH_DATAP) : SH_INT; #endif /* LOGGING */ LOG(("@g6 dt_stw(%lu, %lu)", addr, l)); ch_in_data(addr, wsize); ch_wordaligned(addr); for (i = (int) wsize; i > 0; i--, addr++) { ch_dt_prot(addr); data_loc(addr) = (char) l; #ifdef LOGGING dt_sh(addr) = sh_flags; #endif /* LOGGING */ l = l>>8; } } #ifndef NOFLOAT /** Store a real value "f" or size "n" bytes at address "addr". */ void dt_stf(register ptr addr, double f, register size n) { register char *cp = (char *) &f; register int i; float fl; LOG(("@g6 dt_stf(%lu, %g, %lu)", addr, f, n)); ch_in_data(addr, n); ch_wordaligned(addr); if ((int) n == 4) { fl = f; cp = (char *) &fl; } for (i = (int) n; i > 0; i--, addr++) { ch_dt_prot(addr); data_loc(addr) = *cp++; dt_fl(addr); } } #endif /* NOFLOAT */ /************************************************************************ * Data load division. * ************************************************************************ * * * dt_lddp(addr) - LoaD Data Pointer from data. * * dt_ldip(addr) - LoaD Instruction Pointer from data. * * dt_ldu(addr, n) - LoaD n Unsigned bytes from data. * * dt_lduw(addr) - LoaD wsize Unsigned bytes from data. * * dt_lds(addr, n) - LoaD n Signed bytes from data. * * dt_ldsw(addr) - LoaD wsize Signed bytes from data. * * * ************************************************************************/ /** Load a data segment pointer located at address "addr". */ ptr dt_lddp(register ptr addr) { register ptr p; LOG(("@g6 dt_lddp(%lu)", addr)); ch_in_data(addr, psize); ch_wordaligned(addr); #ifdef LOGGING if (!is_dt_set(addr, psize, SH_DATAP)) { warning(WGDPEXP); warn_dtbits(addr, psize); } #endif /* LOGGING */ p = p_in_data(addr); LOG(("@g6 dt_lddp() returns %lu", p)); return (p); } ptr dt_ldip(register ptr addr) { register ptr p; LOG(("@g6 dt_ldip(%lu)", addr)); ch_in_data(addr, psize); ch_wordaligned(addr); #ifdef LOGGING if (!is_dt_set(addr, psize, SH_INSP)) { warning(WGIPEXP); warn_dtbits(addr, psize); } #endif /* LOGGING */ p = p_in_data(addr); LOG(("@g6 dt_ldip() returns %lu", p)); return (p); } /** Load an unsigned integer of "n" bytes from address "addr". */ unsigned long dt_ldu(register ptr addr, size n) { register int i; register unsigned long u = 0; LOG(("@g6 dt_ldu(%lu, %lu)", addr, n)); ch_in_data(addr, n); ch_aligned(addr, n); #ifdef LOGGING if (!is_dt_set(addr, n, SH_INT)) { warning(n == 1 ? WGCEXP : WGIEXP); warn_dtbits(addr, n); } #endif /* LOGGING */ addr += n-1; for (i = (int) n-1; i >= 0; i--, addr--) { u = (u<<8) | btou(data_loc(addr)); } LOG(("@g6 dt_ldu() returns %lu", u)); return (u); } /** Load an unsigned integer of word size from address "addr". */ unsigned long dt_lduw(register ptr addr) { register int i; register unsigned long u = 0; LOG(("@g6 dt_lduw(%lu)", addr)); ch_in_data(addr, wsize); ch_wordaligned(addr); #ifdef LOGGING if (!is_dt_set(addr, wsize, SH_INT)) { warning(WGIEXP); warn_dtbits(addr, wsize); } #endif /* LOGGING */ addr += wsize-1; for (i = (int) wsize-1; i >= 0; i--, addr--) { u = (u<<8) | btou(data_loc(addr)); } LOG(("@g6 dt_lduw() returns %lu", u)); return (u); } /** Load an integer of size "n" bytes from address "addr". */ long dt_lds(register ptr addr, size n) { register int i; register long l; LOG(("@g6 dt_lds(%lu, %lu)", addr, n)); ch_in_data(addr, n); ch_aligned(addr, n); #ifdef LOGGING if (!is_dt_set(addr, n, SH_INT)) { warning(n == 1 ? WGCEXP : WGIEXP); warn_dtbits(addr, n); } #endif /* LOGGING */ addr += n-2; l = btos(data_loc(addr + 1)); for (i = n - 2; i >= 0; i--, addr--) { l = (l<<8) | btol(data_loc(addr)); } LOG(("@g6 dt_lds() returns %lu", l)); return (l); } /** Load a word size integer from address "addr". */ long dt_ldsw(register ptr addr) { register int i; register long l; LOG(("@g6 dt_ldsw(%lu)", addr)); ch_in_data(addr, wsize); ch_wordaligned(addr); #ifdef LOGGING if (!is_dt_set(addr, wsize, SH_INT)) { warning(WGIEXP); warn_dtbits(addr, wsize); } #endif /* LOGGING */ addr += wsize-2; l = btos(data_loc(addr + 1)); for (i = wsize - 2; i >= 0; i--, addr--) { l = (l<<8) | btol(data_loc(addr)); } LOG(("@g6 dt_ldsw() returns %lu", l)); return (l); } /************************************************************************ * Data move division * ************************************************************************ * * * dt_mvd(d2, d1, n) - Move n bytes in data from d1 to d2. * * dt_mvs(d, s, n) - Move n bytes from s in stack to d in data. * * * * See st_mvs() in stack.c for a description. * * * ************************************************************************/ /** Move "n" bytes from "d1" to "d2". */ void dt_mvd(ptr d2, ptr d1, size n) { register int i; ch_in_data(d1, n); ch_wordaligned(d1); ch_in_data(d2, n); ch_wordaligned(d2); for (i = (int) n; i > 0; i--, d1++, d2++) { ch_dt_prot(d2); data_loc(d2) = data_loc(d1); #ifdef LOGGING dt_sh(d2) = dt_sh(d1) & ~SH_PROT; #endif /* LOGGING */ } } /** Move "n" bytes from stack address "s" to data address "d". */ void dt_mvs(ptr d, ptr s, size n) /* s -> d */ { register int i; ch_in_stack(s, n); ch_wordaligned(s); ch_in_data(d, n); ch_wordaligned(d); for (i = (int) n; i > 0; i--, d++, s++) { ch_dt_prot(d); ch_st_prot(s); data_loc(d) = stack_loc(s); #ifdef LOGGING dt_sh(d) = st_sh(s) & ~SH_PROT; #endif /* LOGGING */ } } #ifdef LOGGING PRIVATE void warn_dtbits(ptr addr, size n) { register int or_bits = 0; register int and_bits = 0xff; while (n--) { or_bits |= dt_sh(addr); and_bits &= dt_sh(addr); addr++; } if (or_bits != and_bits) { /* no use trying to diagnose */ warningcont(WWASMISC); return; } if (or_bits == 0) warningcont(WWASUND); if (or_bits & SH_INT) warningcont(WWASINT); if (or_bits & SH_FLOAT) warningcont(WWASFLOAT); if (or_bits & SH_DATAP) warningcont(WWASDATAP); if (or_bits & SH_INSP) warningcont(WWASINSP); } void dt_clear_area(ptr from, ptr to) { /* includes *from but excludes *to */ register ptr a; for (a = from; a < to; a++) { dt_undef(a); } } #endif /* LOGGING */