blob: 3109c02e086a12006ebabd4b0c91de0910da27f9 [file] [log] [blame] [raw]
/* pdp8_fpp.c: PDP-8 floating point processor (FPP8A)
Copyright (c) 2007-2011, Robert M Supnik
Permission is hereby granted, free of charge, to any person obtaining a
copy of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the
Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
ROBERT M SUPNIK BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Except as contained in this notice, the name of Robert M Supnik shall not be
used in advertising or otherwise to promote the sale, use or other dealings
in this Software without prior written authorization from Robert M Supnik.
fpp FPP8A floating point processor
03-Jan-10 RMS Initialized variables statically, for VMS compiler
19-Apr-09 RHM FPICL does not clear all command and status reg bits
modify fpp_reset to conform with FPP
27-Mar-09 RHM Fixed handling of Underflow fix (zero FAC on underflow)
Implemented FPP division and multiplication algorithms
FPP behavior on traps - FEXIT does not update APT
Follow FPP settings for OPADD
Correct detection of DP add/sub overflow
Detect and handle add/sub overshift
Single-step mode made consistent with FPP
Write calculation results prior to traps
24-Mar-09 RMS Many fixes from Rick Murphy:
Fix calculation of ATX shift amount
Added missing () to read, write XR macros
Fixed indirect address calculation
Fixed == written as = in normalization
Fixed off-by-one count bug in multiplication
Removed extraneous ; in divide
Fixed direction of compare in divide
Fixed count direction bug in alignment
Floating point formats:
00 01 02 03 04 05 06 07 08 09 10 11
+--+--+--+--+--+--+--+--+--+--+--+--+
| S| hi integer | : double precision
+--+--+--+--+--+--+--+--+--+--+--+--+
| lo integer |
+--+--+--+--+--+--+--+--+--+--+--+--+
00 01 02 03 04 05 06 07 08 09 10 11
+--+--+--+--+--+--+--+--+--+--+--+--+
| S| exponent | : floating point
+--+--+--+--+--+--+--+--+--+--+--+--+
| S| hi fraction |
+--+--+--+--+--+--+--+--+--+--+--+--+
| lo fraction |
+--+--+--+--+--+--+--+--+--+--+--+--+
00 01 02 03 04 05 06 07 08 09 10 11
+--+--+--+--+--+--+--+--+--+--+--+--+
| S| exponent | : extended precision
+--+--+--+--+--+--+--+--+--+--+--+--+
| S| hi fraction |
+--+--+--+--+--+--+--+--+--+--+--+--+
| next fraction |
+--+--+--+--+--+--+--+--+--+--+--+--+
| next fraction |
+--+--+--+--+--+--+--+--+--+--+--+--+
| next fraction |
+--+--+--+--+--+--+--+--+--+--+--+--+
| lo fraction |
+--+--+--+--+--+--+--+--+--+--+--+--+
Exponents are 2's complement, as are fractions. Normalized numbers have
the form:
0.0...0
0.<non-zero>
1.<non-zero>
1.1...0
Note that 1.0...0 is normalized but considered illegal, since it cannot
be represented as a positive number. When a result is normalized, 1.0...0
is converted to 1.1...0 with exp+1.
*/
#include "pdp8_defs.h"
extern int32 int_req;
extern uint16 M[];
extern int32 stop_inst;
extern UNIT cpu_unit;
#define SEXT12(x) (((x) & 04000)? (x) | ~07777: (x) & 03777)
/* Index registers are in memory */
#define fpp_read_xr(xr) fpp_read (fpp_xra + (xr))
#define fpp_write_xr(xr,d) fpp_write (fpp_xra + (xr), d)
/* Command register */
#define FPC_DP 04000 /* integer double */
#define FPC_UNFX 02000 /* exit on fl undf */
#define FPC_FIXF 01000 /* lock mem field */
#define FPC_IE 00400 /* int enable */
#define FPC_V_FAST 4 /* startup bits */
#define FPC_M_FAST 017
#define FPC_LOCK 00010 /* lockout */
#define FPC_V_APTF 0
#define FPC_M_APTF 07 /* apta field */
#define FPC_STA (FPC_DP|FPC_LOCK)
#define FPC_GETFAST(x) (((x) >> FPC_V_FAST) & FPC_M_FAST)
#define FPC_GETAPTF(x) (((x) >> FPC_V_APTF) & FPC_M_APTF)
/* Status register */
#define FPS_DP (FPC_DP) /* integer double */
#define FPS_TRPX 02000 /* trap exit */
#define FPS_HLTX 01000 /* halt exit */
#define FPS_DVZX 00400 /* div zero exit */
#define FPS_IOVX 00200 /* int ovf exit */
#define FPS_FOVX 00100 /* flt ovf exit */
#define FPS_UNF 00040 /* underflow */
#define FPS_XXXM 00020 /* FADDM/FMULM */
#define FPS_LOCK (FPC_LOCK) /* lockout */
#define FPS_EP 00004 /* ext prec */
#define FPS_PAUSE 00002 /* paused */
#define FPS_RUN 00001 /* running */
/* Floating point number: 3-6 words */
#define FPN_FRSIGN 04000
#define FPN_NFR_FP 2 /* std precision */
#define FPN_NFR_EP 5 /* ext precision */
#define FPN_NFR_MDS 6 /* mul/div precision */
#define EXACT (uint32)((fpp_sta & FPS_EP)? FPN_NFR_EP: FPN_NFR_FP)
#define EXTEND ((uint32) FPN_NFR_EP)
typedef struct {
int32 exp;
uint32 fr[FPN_NFR_MDS+1];
} FPN;
uint32 fpp_apta = 0; /* APT pointer */
uint32 fpp_aptsvf = 0; /* APT saved field */
uint32 fpp_opa = 0; /* operand pointer */
uint32 fpp_fpc = 0; /* FP PC */
uint32 fpp_bra = 0; /* base reg pointer */
uint32 fpp_xra = 0; /* indx reg pointer */
uint32 fpp_cmd = 0; /* command */
uint32 fpp_sta = 0; /* status */
uint32 fpp_flag = 0; /* flag */
FPN fpp_ac; /* FAC */
uint32 fpp_ssf = 0; /* single-step flag */
uint32 fpp_last_lockbit = 0; /* last lockbit */
static FPN fpp_zero = { 0, { 0, 0, 0, 0, 0 } };
static FPN fpp_one = { 1, { 02000, 0, 0, 0, 0 } };
DEVICE fpp_dev;
int32 fpp55 (int32 IR, int32 AC);
int32 fpp56 (int32 IR, int32 AC);
void fpp_load_apt (uint32 apta);
void fpp_dump_apt (uint32 apta, uint32 sta);
uint32 fpp_1wd_dir (uint32 ir);
uint32 fpp_2wd_dir (uint32 ir);
uint32 fpp_indir (uint32 ir);
uint32 fpp_ad15 (uint32 hi);
uint32 fpp_adxr (uint32 ir, uint32 base_ad);
void fpp_add (FPN *a, FPN *b, uint32 sub);
void fpp_mul (FPN *a, FPN *b);
void fpp_div (FPN *a, FPN *b);
t_bool fpp_imul (FPN *a, FPN *b);
uint32 fpp_fr_add (uint32 *c, uint32 *a, uint32 *b, uint32 cnt);
void fpp_fr_sub (uint32 *c, uint32 *a, uint32 *b, uint32 cnt);
void fpp_fr_mul (uint32 *c, uint32 *a, uint32 *b, t_bool fix);
t_bool fpp_fr_div (uint32 *c, uint32 *a, uint32 *b);
uint32 fpp_fr_neg (uint32 *a, uint32 cnt);
int32 fpp_fr_cmp (uint32 *a, uint32 *b, uint32 cnt);
int32 fpp_fr_test (uint32 *a, uint32 v0, uint32 cnt);
uint32 fpp_fr_abs (uint32 *a, uint32 *b, uint32 cnt);
void fpp_fr_fill (uint32 *a, uint32 v, uint32 cnt);
void fpp_fr_lshn (uint32 *a, uint32 sc, uint32 cnt);
void fpp_fr_lsh12 (uint32 *a, uint32 cnt);
void fpp_fr_lsh1 (uint32 *a, uint32 cnt);
void fpp_fr_rsh1 (uint32 *a, uint32 sign, uint32 cnt);
void fpp_fr_algn (uint32 *a, uint32 sc, uint32 cnt);
t_bool fpp_cond_met (uint32 cond);
t_bool fpp_norm (FPN *a, uint32 cnt);
void fpp_round (FPN *a);
t_bool fpp_test_xp (FPN *a);
void fpp_copy (FPN *a, FPN *b);
void fpp_zcopy (FPN *a, FPN *b);
void fpp_read_op (uint32 ea, FPN *a);
void fpp_write_op (uint32 ea, FPN *a);
uint32 fpp_read (uint32 ea);
void fpp_write (uint32 ea, uint32 val);
uint32 apt_read (uint32 ea);
void apt_write (uint32 ea, uint32 val);
t_stat fpp_svc (UNIT *uptr);
t_stat fpp_reset (DEVICE *dptr);
/* FPP data structures
fpp_dev FPP device descriptor
fpp_unit FPP unit descriptor
fpp_reg FPP register list
*/
DIB fpp_dib = { DEV_FPP, 2, { &fpp55, &fpp56 } };
UNIT fpp_unit = { UDATA (&fpp_svc, 0, 0) };
REG fpp_reg[] = {
{ ORDATAD (FPACE, fpp_ac.exp, 12, "floating accumulator") },
{ ORDATAD (FPAC0, fpp_ac.fr[0], 12, "first mantissa") },
{ ORDATAD (FPAC1, fpp_ac.fr[1], 12, "second mantissa") },
{ ORDATAD (FPAC2, fpp_ac.fr[2], 12, "third mantissa") },
{ ORDATAD (FPAC3, fpp_ac.fr[3], 12, "fourth mantissa") },
{ ORDATAD (FPAC4, fpp_ac.fr[4], 12, "fifth mantissa") },
{ ORDATAD (CMD, fpp_cmd, 12, "FPP command register") },
{ ORDATAD (STA, fpp_sta, 12, "status register") },
{ ORDATAD (APTA, fpp_apta, 15, "active parameter table (APT) pointer") },
{ GRDATAD (APTSVF, fpp_aptsvf, 8, 3, 12, "APT field") },
{ ORDATAD (FPC, fpp_fpc, 15, "floating program counter") },
{ ORDATAD (BRA, fpp_bra, 15, "base register") },
{ ORDATAD (XRA, fpp_xra, 15, "pointer to index register 0") },
{ ORDATAD (OPA, fpp_opa, 15, "operand address register") },
{ ORDATAD (SSF, fpp_ssf, 12, "single step flag") },
{ ORDATAD (LASTLOCK, fpp_last_lockbit, 12, "lockout from FPCOM") },
{ FLDATAD (FLAG, fpp_flag, 0, "done flag") },
{ NULL }
};
DEVICE fpp_dev = {
"FPP", &fpp_unit, fpp_reg, NULL,
1, 10, 31, 1, 8, 8,
NULL, NULL, &fpp_reset,
NULL, NULL, NULL,
&fpp_dib, DEV_DISABLE | DEV_DIS
};
/* IOT routines */
int32 fpp55 (int32 IR, int32 AC)
{
switch (IR & 07) { /* decode IR<9:11> */
case 1: /* FPINT */
return (fpp_flag? IOT_SKP | AC: AC); /* skip on flag */
case 2: /* FPICL */
fpp_reset (&fpp_dev); /* reset device */
break;
case 3: /* FPCOM */
if (!fpp_flag && !(fpp_sta & FPS_RUN)) { /* flag clr, !run? */
fpp_cmd = AC; /* load cmd */
fpp_last_lockbit = fpp_cmd & FPS_LOCK; /* remember lock state */
fpp_sta = (fpp_sta & ~FPC_STA) | /* copy flags */
(fpp_cmd & FPC_STA); /* to status */
}
break;
case 4: /* FPHLT */
if (fpp_sta & FPS_RUN) { /* running? */
if (fpp_sta & FPS_PAUSE) /* paused? */
fpp_fpc = (fpp_fpc - 1) & ADDRMASK; /* decr FPC */
fpp_sta &= ~FPS_PAUSE; /* no longer paused */
sim_cancel (&fpp_unit); /* stop execution */
fpp_dump_apt (fpp_apta, FPS_HLTX); /* dump APT */
fpp_ssf = 1; /* assume sstep */
}
else if (!fpp_flag)
fpp_ssf = 1; /* FPST sing steps */
if (fpp_sta & FPS_DVZX) /* fix diag timing */
fpp_sta |= FPS_HLTX;
break;
case 5: /* FPST */
if (!fpp_flag && !(fpp_sta & FPS_RUN)) { /* flag clr, !run? */
if (fpp_ssf)
fpp_sta |= fpp_last_lockbit;
fpp_sta &= ~FPS_HLTX; /* Clear halted */
fpp_apta = (FPC_GETAPTF (fpp_cmd) << 12) | AC;
fpp_load_apt (fpp_apta); /* load APT */
fpp_opa = fpp_fpc;
sim_activate (&fpp_unit, 0); /* start unit */
return IOT_SKP | AC;
}
if ((fpp_sta & (FPS_RUN|FPS_PAUSE)) == (FPS_RUN|FPS_PAUSE)) {
fpp_sta &= ~FPS_PAUSE; /* continue */
sim_activate (&fpp_unit, 0); /* start unit */
return (IOT_SKP | AC);
}
break;
case 6: /* FPRST */
return fpp_sta;
case 7: /* FPIST */
if (fpp_flag) { /* if flag set */
uint32 old_sta = fpp_sta;
fpp_flag = 0; /* clr flag, status */
fpp_sta &= ~(FPS_DP|FPS_EP|FPS_TRPX|FPS_DVZX|FPS_IOVX|FPS_FOVX|FPS_UNF);
int_req &= ~INT_FPP; /* clr int req */
return IOT_SKP | old_sta; /* ret old status */
}
break;
default:
return (stop_inst << IOT_V_REASON) | AC;
} /* end switch */
return AC;
}
int32 fpp56 (int32 IR, int32 AC)
{
switch (IR & 07) { /* decode IR<9:11> */
case 7: /* FPEP */
if ((AC & 04000) && !(fpp_sta & FPS_RUN)) { /* if AC0, not run, */
fpp_sta = (fpp_sta | FPS_EP) & ~FPS_DP; /* set ep */
AC = 0;
}
break;
default:
return (stop_inst << IOT_V_REASON) | AC;
} /* end switch */
return AC;
}
/* Service routine */
t_stat fpp_svc (UNIT *uptr)
{
FPN x;
uint32 ir, op, op2, op3, ad, ea, wd;
uint32 i;
int32 sc;
fpp_ac.exp = SEXT12 (fpp_ac.exp); /* sext AC exp */
do { /* repeat */
ir = fpp_read (fpp_fpc); /* get instr */
fpp_fpc = (fpp_fpc + 1) & ADDRMASK; /* incr FP PC */
op = (ir >> 7) & 037; /* get op+mode */
op2 = (ir >> 3) & 017; /* get subop */
op3 = ir & 07; /* get field/xr */
fpp_sta &= ~FPS_XXXM; /* not mem op */
switch (op) { /* case on op+mode */
case 000: /* operates */
switch (op2) { /* case on subop */
case 000: /* no-operands */
switch (op3) { /* case on subsubop */
case 0: /* FEXIT */
/* if already trapped, don't update APT, just update status */
if (fpp_sta & (FPS_DVZX|FPS_IOVX|FPS_FOVX|FPS_UNF))
fpp_sta |= FPS_HLTX;
else
fpp_dump_apt (fpp_apta, 0);
break;
case 1: /* FPAUSE */
fpp_sta |= FPS_PAUSE;
break;
case 2: /* FCLA */
fpp_copy (&fpp_ac, &fpp_zero); /* clear FAC */
break;
case 3: /* FNEG */
fpp_fr_neg (fpp_ac.fr, EXACT); /* do exact length */
break;
case 4: /* FNORM */
if (!(fpp_sta & FPS_DP)) { /* fp or ep only */
fpp_copy (&x, &fpp_ac); /* copy AC */
fpp_norm (&x, EXACT); /* do exact length */
fpp_copy (&fpp_ac, &x); /* copy back */
}
break;
case 5: /* STARTF */
if (fpp_sta & FPS_EP) { /* if ep, */
fpp_copy (&x, &fpp_ac); /* copy AC */
fpp_round (&x); /* round */
fpp_copy (&fpp_ac, &x); /* copy back */
}
fpp_sta &= ~(FPS_DP|FPS_EP);
break;
case 6: /* STARTD */
fpp_sta = (fpp_sta | FPS_DP) & ~FPS_EP;
break;
case 7: /* JAC */
fpp_fpc = ((fpp_ac.fr[0] & 07) << 12) | fpp_ac.fr[1];
break;
}
break;
case 001: /* ALN */
if (op3 != 0) { /* if xr, */
wd = fpp_read_xr (op3); /* use val */
fpp_opa = fpp_xra + op3;
}
else wd = 027; /* else 23 */
if (!(fpp_sta & FPS_DP)) { /* fp or ep? */
sc = (SEXT12(wd) - fpp_ac.exp) & 07777; /* alignment */
sc = SEXT12 (sc);
fpp_ac.exp = SEXT12(wd); /* new exp */
}
else sc = SEXT12 (wd); /* dp - simple cnt */
if (sc < 0) /* left? */
fpp_fr_lshn (fpp_ac.fr, -sc, EXACT);
else fpp_fr_algn (fpp_ac.fr, sc, EXACT);
if (fpp_fr_test (fpp_ac.fr, 0, EXACT) == 0) /* zero? */
fpp_ac.exp = 0; /* clean exp */
break;
case 002: /* ATX */
if (fpp_sta & FPS_DP) /* dp? */
fpp_write_xr (op3, fpp_ac.fr[1]); /* xr<-FAC<12:23> */
else {
fpp_copy (&x, &fpp_ac); /* copy AC */
sc = 027 - x.exp; /* shift amt */
if (sc < 0) /* left? */
fpp_fr_lshn (x.fr, -sc, EXACT);
else fpp_fr_algn (x.fr, sc, EXACT);
fpp_write_xr (op3, x.fr[1]); /* xr<-val<12:23> */
}
break;
case 003: /* XTA */
for (i = FPN_NFR_FP; i < FPN_NFR_EP; i++)
x.fr[i] = 0; /* clear FOP2-4 */
x.fr[1] = fpp_read_xr (op3); /* get XR value */
x.fr[0] = (x.fr[1] & 04000)? 07777: 0;
x.exp = 027; /* standard exp */
if (!(fpp_sta & FPS_DP)) { /* fp or ep? */
fpp_norm (&x, EXACT); /* normalize */
}
fpp_copy (&fpp_ac, &x); /* result to AC */
if (fpp_sta & FPS_DP) /* dp skips exp */
fpp_ac.exp = x.exp; /* so force copy */
fpp_opa = fpp_xra + op3;
break;
case 004: /* NOP */
break;
case 005: /* STARTE */
if (!(fpp_sta & FPS_EP)) {
fpp_sta = (fpp_sta | FPS_EP) & ~FPS_DP;
for (i = FPN_NFR_FP; i < FPN_NFR_EP; i++)
fpp_ac.fr[i] = 0; /* clear FAC2-4 */
}
break;
case 010: /* LDX */
wd = fpp_ad15 (0); /* load XR immed */
fpp_write_xr (op3, wd);
fpp_opa = fpp_xra + op3;
break;
case 011: /* ADDX */
wd = fpp_ad15 (0);
wd = wd + fpp_read_xr (op3); /* add to XR immed */
fpp_write_xr (op3, wd); /* trims to 12b */
fpp_opa = fpp_xra + op3;
break;
default:
return stop_inst;
} /* end case subop */
break;
case 001: /* FLDA */
ea = fpp_1wd_dir (ir);
fpp_read_op (ea, &fpp_ac);
break;
case 002:
ea = fpp_2wd_dir (ir);
fpp_read_op (ea, &fpp_ac);
if (fpp_sta & FPS_DP)
fpp_opa = ea + 1;
else fpp_opa = ea + 2;
break;
case 003:
ea = fpp_indir (ir);
fpp_read_op (ea, &fpp_ac);
break;
case 004: /* jumps and sets */
ad = fpp_ad15 (op3); /* get 15b address */
switch (op2) { /* case on subop */
case 000: case 001: case 002: case 003: /* cond jump */
case 004: case 005: case 006: case 007:
if (fpp_cond_met (op2)) /* br if cond */
fpp_fpc = ad;
break;
case 010: /* SETX */
fpp_xra = ad;
break;
case 011: /* SETB */
fpp_bra = ad;
break;
case 012: /* JSA */
fpp_write (ad, 01030 + (fpp_fpc >> 12)); /* save return */
fpp_write (ad + 1, fpp_fpc); /* trims to 12b */
fpp_fpc = (ad + 2) & ADDRMASK;
fpp_opa = fpp_fpc - 1;
break;
case 013: /* JSR */
fpp_write (fpp_bra + 1, 01030 + (fpp_fpc >> 12));
fpp_write (fpp_bra + 2, fpp_fpc); /* trims to 12b */
fpp_opa = fpp_fpc = ad;
break;
default:
return stop_inst;
} /* end case subop */
break;
case 005: /* FADD */
ea = fpp_1wd_dir (ir);
fpp_read_op (ea, &x);
fpp_add (&fpp_ac, &x, 0);
break;
case 006:
ea = fpp_2wd_dir (ir);
fpp_read_op (ea, &x);
fpp_add (&fpp_ac, &x, 0);
break;
case 007:
ea = fpp_indir (ir);
fpp_read_op (ea, &x);
fpp_add (&fpp_ac, &x, 0);
break;
case 010: { /* JNX */
uint32 xrn = op2 & 07;
ad = fpp_ad15 (op3); /* get 15b addr */
wd = fpp_read_xr (xrn); /* read xr */
if (op2 & 010) { /* inc? */
wd = (wd + 1) & 07777;
fpp_write_xr (xrn, wd); /* ++xr */
}
if (wd != 0) /* xr != 0? */
fpp_fpc = ad; /* jump */
break;
}
case 011: /* FSUB */
ea = fpp_1wd_dir (ir);
fpp_read_op (ea, &x);
fpp_add (&fpp_ac, &x, 1);
break;
case 012:
ea = fpp_2wd_dir (ir);
fpp_read_op (ea, &x);
fpp_add (&fpp_ac, &x, 1);
break;
case 013:
ea = fpp_indir (ir);
fpp_read_op (ea, &x);
fpp_add (&fpp_ac, &x, 1);
break;
case 014: /* TRAP3 */
case 020: /* TRAP4 */
fpp_opa = fpp_ad15 (op3);
fpp_dump_apt (fpp_apta, FPS_TRPX);
break;
case 015: /* FDIV */
ea = fpp_1wd_dir (ir);
fpp_read_op (ea, &x);
fpp_div (&fpp_ac, &x);
break;
case 016:
ea = fpp_2wd_dir (ir);
fpp_read_op (ea, &x);
fpp_div (&fpp_ac, &x);
break;
case 017:
ea = fpp_indir (ir);
fpp_read_op (ea, &x);
fpp_div (&fpp_ac, &x);
break;
case 021: /* FMUL */
ea = fpp_1wd_dir (ir);
fpp_read_op (ea, &x);
fpp_mul (&fpp_ac, &x);
break;
case 022:
ea = fpp_2wd_dir (ir);
fpp_read_op (ea, &x);
fpp_mul (&fpp_ac, &x);
break;
case 023:
ea = fpp_indir (ir);
fpp_read_op (ea, &x);
fpp_mul (&fpp_ac, &x);
break;
case 024: /* LTR */
fpp_copy (&fpp_ac, (fpp_cond_met (op2 & 07)? &fpp_one: &fpp_zero));
break;
case 025: /* FADDM */
fpp_sta |= FPS_XXXM;
ea = fpp_1wd_dir (ir);
fpp_read_op (ea, &x);
fpp_add (&x, &fpp_ac, 0);
fpp_write_op (ea, &x); /* store result */
break;
case 026:
fpp_sta |= FPS_XXXM;
ea = fpp_2wd_dir (ir);
fpp_read_op (ea, &x);
fpp_add (&x, &fpp_ac, 0);
fpp_write_op (ea, &x); /* store result */
break;
case 027:
fpp_sta |= FPS_XXXM;
ea = fpp_indir (ir);
fpp_read_op (ea, &x);
fpp_add (&x, &fpp_ac, 0);
fpp_write_op (ea, &x); /* store result */
break;
case 030: /* IMUL/LEA */
ea = fpp_2wd_dir (ir); /* 2-word direct */
if (fpp_sta & FPS_DP) { /* dp? */
fpp_read_op (ea, &x); /* IMUL */
fpp_imul (&fpp_ac, &x);
}
else { /* LEA */
fpp_sta = (fpp_sta | FPS_DP) & ~FPS_EP; /* set dp */
fpp_ac.fr[0] = (ea >> 12) & 07;
fpp_ac.fr[1] = ea & 07777;
}
break;
case 031: /* FSTA */
ea = fpp_1wd_dir (ir);
fpp_write_op (ea, &fpp_ac);
break;
case 032:
ea = fpp_2wd_dir (ir);
fpp_write_op (ea, &fpp_ac);
break;
case 033:
ea = fpp_indir (ir);
fpp_write_op (ea, &fpp_ac);
break;
case 034: /* IMULI/LEAI */
ea = fpp_indir (ir); /* 1-word indir */
if (fpp_sta & FPS_DP) { /* dp? */
fpp_read_op (ea, &x); /* IMUL */
fpp_imul (&fpp_ac, &x);
}
else { /* LEA */
fpp_sta = (fpp_sta | FPS_DP) & ~FPS_EP; /* set dp */
fpp_ac.fr[0] = (ea >> 12) & 07;
fpp_ac.fr[1] = ea & 07777;
fpp_opa = ea;
}
break;
case 035: /* FMULM */
fpp_sta |= FPS_XXXM;
ea = fpp_1wd_dir (ir);
fpp_read_op (ea, &x);
fpp_mul (&x, &fpp_ac);
fpp_write_op (ea, &x); /* store result */
break;
case 036:
fpp_sta |= FPS_XXXM;
ea = fpp_2wd_dir (ir);
fpp_read_op (ea, &x);
fpp_mul (&x, &fpp_ac);
fpp_write_op (ea, &x); /* store result */
break;
case 037:
fpp_sta |= FPS_XXXM;
ea = fpp_indir (ir);
fpp_read_op (ea, &x);
fpp_mul (&x, &fpp_ac);
fpp_write_op (ea, &x); /* store result */
break;
} /* end sw op+mode */
if (fpp_ssf) {
fpp_dump_apt (fpp_apta, FPS_HLTX); /* dump APT */
fpp_ssf = 0;
}
if (sim_interval)
sim_interval = sim_interval - 1;
} while ((sim_interval > 0) &&
((fpp_sta & (FPS_RUN|FPS_PAUSE|FPS_LOCK)) == (FPS_RUN|FPS_LOCK)));
if ((fpp_sta & (FPS_RUN|FPS_PAUSE)) == FPS_RUN)
sim_activate (uptr, 1);
fpp_ac.exp &= 07777; /* mask AC exp */
return SCPE_OK;
}
/* Address decoding routines */
uint32 fpp_1wd_dir (uint32 ir)
{
uint32 ad;
ad = fpp_bra + ((ir & 0177) * 3); /* base + 3*7b off */
if (fpp_sta & FPS_DP) /* dp? skip exp */
ad = ad + 1;
ad = ad & ADDRMASK;
if (fpp_sta & FPS_DP)
fpp_opa = ad + 1;
else fpp_opa = ad + 2;
return ad;
}
uint32 fpp_2wd_dir (uint32 ir)
{
uint32 ad;
ad = fpp_ad15 (ir); /* get 15b addr */
return fpp_adxr (ir, ad); /* do indexing */
}
uint32 fpp_indir (uint32 ir)
{
uint32 ad, wd1, wd2;
ad = fpp_bra + ((ir & 07) * 3); /* base + 3*3b off */
wd1 = fpp_read (ad + 1); /* bp+off points to */
wd2 = fpp_read (ad + 2);
ad = ((wd1 & 07) << 12) | wd2; /* indirect ptr */
ad = fpp_adxr (ir, ad); /* do indexing */
if (fpp_sta & FPS_DP)
fpp_opa = ad + 1;
else fpp_opa = ad + 2;
return ad;
}
uint32 fpp_ad15 (uint32 hi)
{
uint32 ad;
ad = ((hi & 07) << 12) | fpp_read (fpp_fpc); /* 15b addr */
fpp_fpc = (fpp_fpc + 1) & ADDRMASK; /* incr FPC */
return ad; /* return addr */
}
uint32 fpp_adxr (uint32 ir, uint32 base_ad)
{
uint32 xr, wd;
xr = (ir >> 3) & 07;
wd = fpp_read_xr (xr); /* get xr */
if (ir & 0100) { /* increment? */
wd = (wd + 1) & 07777; /* inc, rewrite */
fpp_write_xr (xr, wd);
}
if (xr != 0) { /* indexed? */
if (fpp_sta & FPS_EP)
wd = wd * 6; /* scale by len */
else if (fpp_sta & FPS_DP)
wd = wd * 2;
else wd = wd * 3;
return (base_ad + wd) & ADDRMASK; /* return index */
}
else return base_ad & ADDRMASK; /* return addr */
}
/* Computation routines */
/* Fraction/floating add */
void fpp_add (FPN *a, FPN *b, uint32 sub)
{
FPN x, y, z;
uint32 c, ediff;
fpp_zcopy (&x, a); /* copy opnds */
fpp_zcopy (&y, b);
if (sub) /* subtract? */
fpp_fr_neg (y.fr, EXACT); /* neg B, exact */
if (fpp_sta & FPS_DP) { /* dp? */
uint32 cout = fpp_fr_add (z.fr, x.fr, y.fr, EXTEND);/* z = a + b */
uint32 zsign = z.fr[0] & FPN_FRSIGN;
cout = (cout? 04000: 0); /* make sign bit */
/* overflow is indicated when signs are equal and overflow does not
match the result sign bit */
fpp_copy (a, &z); /* result is z */
if (!((x.fr[0] ^ y.fr[0]) & FPN_FRSIGN) && (cout != zsign)) {
fpp_copy (a, &z); /* copy out result */
fpp_dump_apt (fpp_apta, FPS_IOVX); /* int ovf? */
return;
}
}
else { /* fp or ep */
if (fpp_fr_test (b->fr, 0, EXACT) == 0) /* B == 0? */
z = x; /* result is A */
else if (fpp_fr_test (a->fr, 0, EXACT) == 0) /* A == 0? */
z = y; /* result is B */
else { /* fp or ep */
if (x.exp < y.exp) { /* |a| < |b|? */
z = x; /* exchange ops */
x = y;
y = z;
}
ediff = x.exp - y.exp; /* exp diff */
if (ediff <= (uint32) ((fpp_sta & FPS_EP)? 59: 24)) { /* any add? */
z.exp = x.exp; /* result exp */
if (ediff != 0) /* any align? */
fpp_fr_algn (y.fr, ediff, EXTEND); /* align, 60b */
c = fpp_fr_add (z.fr, x.fr, y.fr, EXTEND); /* add fractions */
if ((((x.fr[0] ^ y.fr[0]) & FPN_FRSIGN) == 0) && /* same signs? */
(c || /* carry out? */
((~x.fr[0] & z.fr[0] & FPN_FRSIGN)))) { /* + to - change? */
fpp_fr_rsh1 (z.fr, c << 11, EXTEND); /* rsh, insert cout */
z.exp = z.exp + 1; /* incr exp */
} /* end same signs */
} /* end in range */
else z = x; /* ovrshift */
} /* end ops != 0 */
if (fpp_norm (&z, EXTEND)) /* norm, !exact? */
fpp_round (&z); /* round */
fpp_copy (a, &z); /* copy out */
fpp_test_xp (&z); /* ovf, unf? */
} /* end else */
return;
}
/* Fraction/floating multiply */
void fpp_mul (FPN *a, FPN *b)
{
FPN x, y, z;
fpp_zcopy (&x, a); /* copy opnds */
fpp_zcopy (&y, b);
if ((fpp_fr_test(y.fr, 0, EXACT-1) == 0) && (y.fr[EXACT-1] < 2)) {
y.exp = 0;
y.fr[EXACT-1] = 0;
}
if (fpp_sta & FPS_DP) /* dp? */
fpp_fr_mul (z.fr, x.fr, y.fr, TRUE); /* mult frac */
else { /* fp or ep */
fpp_norm (&x, EXACT);
fpp_norm (&y, EXACT);
z.exp = x.exp + y.exp; /* add exp */
fpp_fr_mul (z.fr, x.fr, y.fr, TRUE); /* mult frac */
if (fpp_norm (&z, EXTEND)) /* norm, !exact? */
fpp_round (&z); /* round */
fpp_copy (a, &z);
if (z.exp > 2047)
fpp_dump_apt (fpp_apta, FPS_FOVX); /* trap */
return;
}
fpp_copy (a, &z); /* result is z */
return;
}
/* Fraction/floating divide */
void fpp_div (FPN *a, FPN *b)
{
FPN x, y, z;
if (fpp_fr_test (b->fr, 0, EXACT) == 0) { /* divisor 0? */
fpp_dump_apt (fpp_apta, FPS_DVZX); /* error */
return;
}
if (fpp_fr_test (a->fr, 0, EXACT) == 0) /* dividend 0? */
return; /* quotient is 0 */
fpp_zcopy (&x, a); /* copy opnds */
fpp_zcopy (&y, b);
if (fpp_sta & FPS_DP) { /* dp? */
if (fpp_fr_div (z.fr, x.fr, y.fr)) { /* fr div, ovflo? */
fpp_dump_apt (fpp_apta, FPS_IOVX); /* error */
return;
}
fpp_copy (a, &z); /* result is z */
}
else { /* fp or ep */
fpp_norm (&y, EXACT); /* norm divisor */
if (fpp_fr_test (x.fr, 04000, EXACT) == 0) { /* divd 1.000...? */
x.fr[0] = 06000; /* fix */
x.exp = x.exp + 1;
}
z.exp = x.exp - y.exp; /* calc exp */
if (fpp_fr_div (z.fr, x.fr, y.fr)) { /* fr div, ovflo? */
uint32 cin = (a->fr[0] ^ b->fr[0]) & FPN_FRSIGN;
fpp_fr_rsh1 (z.fr, cin, EXTEND); /* rsh, insert sign */
z.exp = z.exp + 1; /* incr exp */
}
if (fpp_norm (&z, EXTEND)) /* norm, !exact? */
fpp_round (&z); /* round */
fpp_copy (a, &z);
if (z.exp > 2048) { /* underflow? */
if (fpp_cmd & FPC_UNFX) { /* trap? */
fpp_dump_apt (fpp_apta, FPS_UNF);
return;
}
}
}
return;
}
/* Integer multiply - returns true if overflow */
t_bool fpp_imul (FPN *a, FPN *b)
{
uint32 sext;
FPN x, y, z;
fpp_zcopy (&x, a); /* copy args */
fpp_zcopy (&y, b);
fpp_fr_mul (z.fr, x.fr, y.fr, FALSE); /* mult fracs */
a->fr[0] = z.fr[1]; /* low 24b */
a->fr[1] = z.fr[2];
if ((a->fr[0] == 0) && (a->fr[1] == 0)) /* fpp zeroes exp */
a->exp = 0; /* even in dp mode */
sext = (z.fr[2] & FPN_FRSIGN)? 07777: 0;
if (((z.fr[0] | z.fr[1] | sext) != 0) && /* hi 25b == 0 */
((z.fr[0] & z.fr[1] & sext) != 07777)) { /* or 777777774? */
fpp_dump_apt (fpp_apta, FPS_IOVX);
return TRUE;
}
return FALSE;
}
/* Auxiliary floating point routines */
t_bool fpp_cond_met (uint32 cond)
{
switch (cond) {
case 0:
return (fpp_fr_test (fpp_ac.fr, 0, EXACT) == 0);
case 1:
return (fpp_fr_test (fpp_ac.fr, 0, EXACT) >= 0);
case 2:
return (fpp_fr_test (fpp_ac.fr, 0, EXACT) <= 0);
case 3:
return 1;
case 4:
return (fpp_fr_test (fpp_ac.fr, 0, EXACT) != 0);
case 5:
return (fpp_fr_test (fpp_ac.fr, 0, EXACT) < 0);
case 6:
return (fpp_fr_test (fpp_ac.fr, 0, EXACT) > 0);
case 7:
return (fpp_ac.exp > 027);
}
return 0;
}
/* Normalization - returns TRUE if rounding possible, FALSE if exact */
t_bool fpp_norm (FPN *a, uint32 cnt)
{
if (fpp_fr_test (a->fr, 0, cnt) == 0) { /* zero? */
a->exp = 0; /* clean exp */
return FALSE; /* don't round */
}
while (((a->fr[0] == 0) && !(a->fr[1] & 04000)) || /* lead 13b same? */
((a->fr[0] == 07777) && (a->fr[1] & 04000))) {
fpp_fr_lsh12 (a->fr, cnt); /* move word */
a->exp = a->exp - 12;
}
while (((a->fr[0] ^ (a->fr[0] << 1)) & FPN_FRSIGN) == 0) { /* until norm */
fpp_fr_lsh1 (a->fr, cnt); /* shift 1b */
a->exp = a->exp - 1;
}
if (fpp_fr_test (a->fr, 04000, EXACT) == 0) { /* 4000...0000? */
a->fr[0] = 06000; /* chg to 6000... */
a->exp = a->exp + 1; /* with exp+1 */
return FALSE; /* don't round */
}
return TRUE;
}
/* Exact fp number copy */
void fpp_copy (FPN *a, FPN *b)
{
uint32 i;
if (!(fpp_sta & FPS_DP))
a->exp = b->exp;
for (i = 0; i < EXACT; i++)
a->fr[i] = b->fr[i];
return;
}
/* Zero extended fp number copy (60b) */
void fpp_zcopy (FPN *a, FPN *b)
{
uint32 i;
a->exp = b->exp;
for (i = 0; i < FPN_NFR_EP; i++) {
if ((i < FPN_NFR_FP) || (fpp_sta & FPS_EP))
a->fr[i] = b->fr[i];
else a->fr[i] = 0;
}
a->fr[i++] = 0;
a->fr[i] = 0;
return;
}
/* Test exp for overflow or underflow, returns TRUE on trap */
t_bool fpp_test_xp (FPN *a)
{
if (a->exp > 2047) { /* overflow? */
fpp_dump_apt (fpp_apta, FPS_FOVX); /* trap */
return TRUE;
}
if (a->exp < -2048) { /* underflow? */
if (fpp_cmd & FPC_UNFX) { /* trap? */
fpp_dump_apt (fpp_apta, FPS_UNF);
return TRUE;
}
fpp_copy (a, &fpp_zero); /* flush to 0 */
}
return FALSE;
}
/* Round dp/fp value */
void fpp_round (FPN *a)
{
int32 i;
uint32 cin, afr0_sign;
if (fpp_sta & FPS_EP) /* ep? */
return; /* don't round */
afr0_sign = a->fr[0] & FPN_FRSIGN; /* save input sign */
cin = afr0_sign? 03777: 04000;
for (i = FPN_NFR_FP; i >= 0; i--) { /* 3 words */
a->fr[i] = a->fr[i] + cin; /* add in carry */
cin = (a->fr[i] >> 12) & 1;
a->fr[i] = a->fr[i] & 07777;
}
if (!(fpp_sta & FPS_DP) && /* fp? */
(afr0_sign ^ (a->fr[0] & FPN_FRSIGN))) { /* sign change? */
fpp_fr_rsh1 (a->fr, afr0_sign, EXACT); /* rsh, insert sign */
a->exp = a->exp + 1;
}
return;
}
/* N-precision integer routines */
/* Fraction add/sub */
uint32 fpp_fr_add (uint32 *c, uint32 *a, uint32 *b, uint32 cnt)
{
uint32 i, cin;
for (i = cnt, cin = 0; i > 0; i--) {
c[i - 1] = a[i - 1] + b[i - 1] + cin;
cin = (c[i - 1] >> 12) & 1;
c[i - 1] = c[i - 1] & 07777;
}
return cin;
}
void fpp_fr_sub (uint32 *c, uint32 *a, uint32 *b, uint32 cnt)
{
uint32 i, cin;
for (i = cnt, cin = 0; i > 0; i--) {
c[i - 1] = a[i - 1] - b[i - 1] - cin;
cin = (c[i - 1] >> 12) & 1;
c[i - 1] = c[i - 1] & 07777;
}
return;
}
/* Fraction multiply - always develop 60b, multiply is
either 24b*24b or 60b*60b
This is a signed multiply. The shift in for signed multiply is
technically ALU_N XOR ALU_V. This can be simplified as follows:
a-sign c-sign result-sign cout overflow N XOR V = shift in
0 0 0 0 0 0
0 0 1 0 1 0
0 1 0 1 0 0
0 1 1 0 0 1
1 0 0 1 0 0
1 0 1 0 0 1
1 1 0 1 1 1
1 1 1 1 0 1
If a-sign == c-sign, shift-in = a-sign
If a-sign != c-sign, shift-in = result-sign
*/
void fpp_fr_mul (uint32 *c, uint32 *a, uint32 *b, t_bool fix)
{
uint32 i, cnt, lo, wc, fill, b_sign;
b_sign = b[0] & FPN_FRSIGN; /* remember b's sign */
fpp_fr_fill (c, 0, FPN_NFR_MDS); /* clr answer */
if (fpp_sta & FPS_EP) /* ep? */
lo = FPN_NFR_EP; /* low order mpyr word */
else
lo = FPN_NFR_FP; /* low order mpyr word */
if (fix)
fpp_fr_algn (a, 12, FPN_NFR_MDS + 1); /* fill left with sign */
wc = 2; /* 3 words at start */
fill = 0;
cnt = lo * 12; /* total steps */
for (i = 0; i < cnt; i++) {
if ((i % 12) == 0) {
wc++; /* do another word */
lo--; /* and next mpyr word */
fpp_fr_algn (c, 24, wc + 1);
c[wc] = 0;
c[0] = c[1] = fill; /* propagate sign */
}
if (b[lo] & FPN_FRSIGN) /* mpyr bit set? */
fpp_fr_add(c, a, c, wc);
fill = ((c[0] & FPN_FRSIGN) ? 07777 : 0); /* remember sign */
fpp_fr_lsh1 (c, wc); /* shift the result */
fpp_fr_lsh1 (b + lo, 1); /* shift mpcd */
}
if (!fix) /* imul shifts result */
fpp_fr_rsh1 (c, c[0] & FPN_FRSIGN, EXACT + 1); /* result is 1 wd right */
if (b_sign) { /* if mpyr was negative */
if (fix)
fpp_fr_lsh12 (a, FPN_NFR_MDS+1); /* restore a */
fpp_fr_sub (c, c, a, EXACT); /* adjust result */
fpp_fr_sub (c, c, a, EXACT);
}
return;
}
/* Fraction divide */
t_bool fpp_fr_div (uint32 *c, uint32 *a, uint32 *b)
{
uint32 i, old_c, lo, cnt, sign, b_sign, addsub, limit;
/* Number of words processed by each divide step */
static uint32 limits[7] = {6, 6, 5, 4, 3, 3, 2};
fpp_fr_fill (c, 0, FPN_NFR_MDS); /* clr answer */
sign = (a[0] ^ b[0]) & FPN_FRSIGN; /* sign of result */
b_sign = (b[0] & FPN_FRSIGN);
if (a[0] & FPN_FRSIGN) /* |a| */
fpp_fr_neg (a, EXACT);
if (fpp_sta & FPS_EP) /* ep? 6 words */
lo = FPN_NFR_EP-1;
else lo = FPN_NFR_FP-1; /* fp, dp? 3 words */
cnt = (lo + 1) * 12;
addsub = 04000; /* setup first op */
for (i = 0; i < cnt; i++) { /* loop */
limit = limits[i / 12]; /* how many wds this time */
fpp_fr_lsh1 (c, FPN_NFR_MDS); /* shift quotient */
if (addsub ^ b_sign) /* diff signs, subtr */
fpp_fr_sub (a, a, b, limit); /* divd - divr */
else
fpp_fr_add (a, a, b, limit); /* restore */
if (!(a[0] & FPN_FRSIGN)) {
c[lo] |= 1; /* set quo bit */
addsub = 04000; /* sign for nxt loop */
}
else addsub = 0;
fpp_fr_lsh1 (a, limit); /* shift dividend */
}
old_c = c[0]; /* save ho quo */
if (sign) /* expect neg ans? */
fpp_fr_neg (c, EXTEND); /* -quo */
if (old_c & FPN_FRSIGN) /* sign set before */
return TRUE; /* neg? */
return FALSE;
}
/* Negate - 24b or 60b */
uint32 fpp_fr_neg (uint32 *a, uint32 cnt)
{
uint32 i, cin;
for (i = cnt, cin = 1; i > 0; i--) {
a[i - 1] = (~a[i - 1] + cin) & 07777;
cin = (cin != 0 && a[i - 1] == 0);
}
return cin;
}
/* Test (compare to x'0...0) - 24b or 60b */
int32 fpp_fr_test (uint32 *a, uint32 v0, uint32 cnt)
{
uint32 i;
if (a[0] != v0)
return (a[0] & FPN_FRSIGN)? -1: +1;
for (i = 1; i < cnt; i++) {
if (a[i] != 0)
return (a[0] & FPN_FRSIGN)? -1: +1;
}
return 0;
}
/* Fraction compare - 24b or 60b */
int32 fpp_fr_cmp (uint32 *a, uint32 *b, uint32 cnt)
{
uint32 i;
if ((a[0] ^ b[0]) & FPN_FRSIGN)
return (b[0] & FPN_FRSIGN)? +1: -1;
for (i = 0; i < cnt; i++) {
if (a[i] > b[i])
return (b[0] & FPN_FRSIGN)? +1: -1;
if (a[i] < b[i])
return (b[0] & FPN_FRSIGN)? -1: +1;
}
return 0;
}
/* Fraction fill */
void fpp_fr_fill (uint32 *a, uint32 v, uint32 cnt)
{
uint32 i;
for (i = 0; i < cnt; i++)
a[i] = v;
return;
}
/* Left shift n (unsigned) */
void fpp_fr_lshn (uint32 *a, uint32 sc, uint32 cnt)
{
uint32 i;
if (sc >= (cnt * 12)) { /* out of range? */
fpp_fr_fill (a, 0, cnt);
return;
}
while (sc >= 12) { /* word shift? */
fpp_fr_lsh12 (a, cnt);
sc = sc - 12;
}
if (sc == 0) /* any more? */
return;
for (i = 1; i < cnt; i++) /* bit shift */
a[i - 1] = ((a[i - 1] << sc) | (a[i] >> (12 - sc))) & 07777;
a[cnt - 1] = (a[cnt - 1] << sc) & 07777;
return;
}
/* Left shift 12b (unsigned) */
void fpp_fr_lsh12 (uint32 *a, uint32 cnt)
{
uint32 i;
for (i = 1; i < cnt; i++)
a[i - 1] = a[i];
a[cnt - 1] = 0;
return;
}
/* Left shift 1b (unsigned) */
void fpp_fr_lsh1 (uint32 *a, uint32 cnt)
{
uint32 i;
for (i = 1; i < cnt; i++)
a[i - 1] = ((a[i - 1] << 1) | (a[i] >> 11)) & 07777;
a[cnt - 1] = (a[cnt - 1] << 1) & 07777;
return;
}
/* Right shift 1b, with shift in */
void fpp_fr_rsh1 (uint32 *a, uint32 sign, uint32 cnt)
{
uint32 i;
for (i = cnt - 1; i > 0; i--)
a[i] = ((a[i] >> 1) | (a[i - 1] << 11)) & 07777;
a[0] = (a[0] >> 1) | sign;
return;
}
/* Right shift n (signed) */
void fpp_fr_algn (uint32 *a, uint32 sc, uint32 cnt)
{
uint32 i, sign;
sign = (a[0] & FPN_FRSIGN)? 07777: 0;
if (sc >= (cnt * 12)) { /* out of range? */
fpp_fr_fill (a, sign, cnt);
return;
}
while (sc >= 12) {
for (i = cnt - 1; i > 0; i--)
a[i] = a[i - 1];
a[0] = sign;
sc = sc - 12;
}
if (sc == 0)
return;
for (i = cnt - 1; i > 0; i--)
a[i] = ((a[i] >> sc) | (a[i - 1] << (12 - sc))) & 07777;
a[0] = ((a[0] >> sc) | (sign << (12 - sc))) & 07777;
return;
}
/* Read/write routines */
void fpp_read_op (uint32 ea, FPN *a)
{
uint32 i;
if (!(fpp_sta & FPS_DP)) {
a->exp = fpp_read (ea++);
a->exp = SEXT12 (a->exp);
}
for (i = 0; i < EXACT; i++)
a->fr[i] = fpp_read (ea + i);
return;
}
void fpp_write_op (uint32 ea, FPN *a)
{
uint32 i;
fpp_opa = ea + 2;
if (!(fpp_sta & FPS_DP))
fpp_write (ea++, a->exp);
for (i = 0; i < EXACT; i++)
fpp_write (ea + i, a->fr[i]);
return;
}
uint32 fpp_read (uint32 ea)
{
ea = ea & ADDRMASK;
if (fpp_cmd & FPC_FIXF)
ea = fpp_aptsvf | (ea & 07777);
return M[ea];
}
void fpp_write (uint32 ea, uint32 val)
{
ea = ea & ADDRMASK;
if (fpp_cmd & FPC_FIXF)
ea = fpp_aptsvf | (ea & 07777);
if (MEM_ADDR_OK (ea))
M[ea] = val & 07777;
return;
}
uint32 apt_read (uint32 ea)
{
ea = ea & ADDRMASK;
return M[ea];
}
void apt_write (uint32 ea, uint32 val)
{
ea = ea & ADDRMASK;
if (MEM_ADDR_OK (ea))
M[ea] = val & 07777;
return;
}
/* Utility routines */
void fpp_load_apt (uint32 ad)
{
uint32 wd0, i;
wd0 = apt_read (ad++);
fpp_fpc = ((wd0 & 07) << 12) | apt_read (ad++);
if (FPC_GETFAST (fpp_cmd) != 017) {
fpp_xra = ((wd0 & 00070) << 9) | apt_read (ad++);
fpp_bra = ((wd0 & 00700) << 6) | apt_read (ad++);
fpp_opa = ((wd0 & 07000) << 3) | apt_read (ad++);
fpp_ac.exp = apt_read (ad++);
for (i = 0; i < EXACT; i++)
fpp_ac.fr[i] = apt_read (ad++);
}
fpp_aptsvf = (ad - 1) & 070000;
fpp_sta |= FPS_RUN;
return;
}
void fpp_dump_apt (uint32 ad, uint32 sta)
{
uint32 wd0, i;
wd0 = (fpp_fpc >> 12) & 07;
if (FPC_GETFAST (fpp_cmd) != 017)
wd0 = wd0 |
((fpp_opa >> 3) & 07000) |
((fpp_bra >> 6) & 00700) |
((fpp_xra >> 9) & 00070);
apt_write (ad++, wd0);
apt_write (ad++, fpp_fpc);
if (FPC_GETFAST (fpp_cmd) != 017) {
apt_write (ad++, fpp_xra);
apt_write (ad++, fpp_bra);
apt_write (ad++, fpp_opa);
apt_write (ad++, fpp_ac.exp);
for (i = 0; i < EXACT; i++)
apt_write (ad++, fpp_ac.fr[i]);
}
fpp_sta = (fpp_sta | sta) & ~FPS_RUN;
fpp_flag = 1;
if (fpp_cmd & FPC_IE)
int_req |= INT_FPP;
return;
}
/* Reset routine */
t_stat fpp_reset (DEVICE *dptr)
{
sim_cancel (&fpp_unit);
fpp_flag = 0;
fpp_last_lockbit = 0;
int_req &= ~INT_FPP;
if (sim_switches & SWMASK ('P')) {
fpp_apta = 0;
fpp_aptsvf = 0;
fpp_fpc = 0;
fpp_bra = 0;
fpp_xra = 0;
fpp_opa = 0;
fpp_ac = fpp_zero;
fpp_ssf = 0;
fpp_sta = 0;
fpp_cmd = 0;
}
else {
fpp_sta &= ~(FPS_DP|FPS_EP|FPS_TRPX|FPS_DVZX|FPS_IOVX|FPS_FOVX|FPS_UNF);
fpp_cmd &= (FPC_DP|FPC_UNFX|FPC_IE);
}
return SCPE_OK;
}