root/src/dps8/doAppendCycleOperandRead.h

/* [previous][next][first][last][top][bottom][index][help] */

INCLUDED FROM


DEFINITIONS

This source file includes following definitions.
  1. doAppendCycleOperandRead

   1 /*
   2  * vim: filetype=c:tabstop=4:ai:expandtab
   3  * SPDX-License-Identifier: ICU
   4  * scspell-id: 3b4b8be2-171d-11ee-84fd-80ee73e9b8e7
   5  *
   6  * ---------------------------------------------------------------------------
   7  *
   8  * Copyright (c) 2022-2023 Charles Anthony
   9  * Copyright (c) 2022-2023 Jeffrey H. Johnson
  10  * Copyright (c) 2022-2025 The DPS8M Development Team
  11  *
  12  * This software is made available under the terms of the ICU License.
  13  * See the LICENSE.md file at the top-level directory of this distribution.
  14  *
  15  * ---------------------------------------------------------------------------
  16  */
  17 
  18 //
  19 //        A:   fetch sdw
  20 //        B:   check RB consistency
  21 //        B1:  Is CALL6? Yes---------------------------------------+
  22 //               No                                                |
  23 //             Is transfer? Yes---------------+                E: Check RB
  24 //               No                           |                    |
  25 //             Check Read RB             F: Check Execute RB       |
  26 //                |                      D: Check RALR             |
  27 //                |                           |                    |
  28 //                +<--------------------------+<-------------------+
  29 //                |
  30 //                V
  31 //        G:   Check bound
  32 //             Paged? No----------------------+
  33 //               Yes                          |
  34 //             Fetch PTW                      |
  35 //        I:   Compute final address     H:   Compute Final Address
  36 //                |                           |
  37 //                +<--------------------------+
  38 //                |
  39 //                V
  40 //        HI:  Read operand
  41 //             CALL6? Yes--------------------------------------+
  42 //               No                                            |
  43 //             Transfer? Yes -----------------+                |
  44 //               No                           |                |
  45 //                |                           |                |
  46 //                |                      L:  Handle TSPn   N: Handle CALL6
  47 //                |                      KL: Set IC           Set IC
  48 //                |                           |                |
  49 //                |                           +<---------------+
  50 //                |                           |
  51 //                |                      M: Set P
  52 //                |                           |
  53 //                +<--------------------------+
  54 //                |
  55 //                V
  56 //              Exit
  57 
  58 word24 doAppendCycleOperandRead (cpu_state_t * cpup, word36 * data, uint nWords) {
     /* [previous][next][first][last][top][bottom][index][help] */
  59 static int evcnt = 0;
  60   DCDstruct * i = & cpu.currentInstruction;
  61   (void)evcnt;
  62   DBGAPP ("doAppendCycleOperandRead(Entry) thisCycle=OPERAND_READ\r\n");
  63   DBGAPP ("doAppendCycleOperandRead(Entry) lastCycle=%s\r\n", str_pct (cpu.apu.lastCycle));
  64   DBGAPP ("doAppendCycleOperandRead(Entry) CA %06o\r\n", cpu.TPR.CA);
  65   DBGAPP ("doAppendCycleOperandRead(Entry) n=%2u\r\n", nWords);
  66   DBGAPP ("doAppendCycleOperandRead(Entry) PPR.PRR=%o PPR.PSR=%05o\r\n", cpu.PPR.PRR, cpu.PPR.PSR);
  67   DBGAPP ("doAppendCycleOperandRead(Entry) TPR.TRR=%o TPR.TSR=%05o\r\n", cpu.TPR.TRR, cpu.TPR.TSR);
  68 
  69   if (i->b29) {
  70     DBGAPP ("doAppendCycleOperandRead(Entry) isb29 PRNO %o\r\n", GET_PRN (IWB_IRODD));
  71   }
  72 
  73   uint this = UC_OPERAND_READ;
  74   if (i->info->flags & TRANSFER_INS)
  75     this = UC_OPERAND_READ_TRA;
  76   if (i->info->flags & CALL6_INS)
  77     this = UC_OPERAND_READ_CALL6;
  78 
  79   word24 finalAddress = 0;
  80   word24 pageAddress = 0;
  81   word3 RSDWH_R1 = 0;
  82   word14 bound = 0;
  83   word1 p = 0;
  84   bool paged;
  85 
  86 // Is this cycle a candidate for ucache?
  87 
  88 //#define TEST_UCACHE
  89 #if defined(TEST_UCACHE)
  90   bool cacheHit;
  91   cacheHit = false; // Assume skip...
  92 #endif /* if defined(TEST_UCACHE) */
  93 
  94 
  95   // Is OPCODE call6?
  96 
  97   // See E1; The TRR needs to be checked and set to R2; this will vary across different
  98   // CALL6 calls.
  99   if (i->info->flags & CALL6_INS) {
 100 # if defined(UCACHE_STATS)
 101     cpu.uCache.call6Skips ++;
 102 # endif /* if defined(UCACHE_STATS) */
 103     goto skip;
 104   }
 105 
 106 
 107 
 108 
 109 
 110 
 111 
 112 
 113 
 114 
 115   // Transfer?
 116   if (i->info->flags & TRANSFER_INS) {
 117     // check ring alarm to catch outbound transfers
 118     if (cpu.rRALR && (cpu.PPR.PRR >= cpu.rRALR)) {
 119 #if defined(UCACHE_STATS)
 120       cpu.uCache.ralrSkips ++;
 121 #endif /* if defined(UCACHE_STATS) */
 122       goto skip;
 123     }
 124   }
 125 
 126 // Yes; check the ucache
 127 
 128 #if defined(TEST_UCACHE)
 129   word24 cachedAddress;
 130   word3 cachedR1;
 131   word14 cachedBound;
 132   word1 cachedP;
 133   bool cachedPaged;
 134   cacheHit =
 135       ucCacheCheck (cpup, this, cpu.TPR.TSR, cpu.TPR.CA, & cachedBound, & cachedP, & cachedAddress, & cachedR1, & cachedPaged);
 136 # if defined(HDBG)
 137   hdbgNote ("doAppendCycleOperandRead.h", "test cache check %s %d %u %05o:%06o %05o %o %08o %o %o",
 138             cacheHit ? "hit" : "miss", evcnt, this, cpu.TPR.TSR, cpu.TPR.CA, cachedBound,
 139             cachedP, cachedAddress, cachedR1, cachedPaged);
 140 # endif /* if defined(HDBG) */
 141   goto miss;
 142 #else
 143   if (! ucCacheCheck (cpup, this, cpu.TPR.TSR, cpu.TPR.CA, & bound, & p, & pageAddress, & RSDWH_R1, & paged)) {
 144 # if defined(HDBG)
 145     hdbgNote ("doAppendCycleOperandRead.h", "miss %d %05o:%06o\r\n", evcnt, cpu.TPR.TSR, cpu.TPR.CA);
 146 # endif /* if defined(HDBG) */
 147     goto miss;
 148   }
 149 #endif /* if defined(TEST_UCACHE) */
 150 
 151   if (paged) {
 152     finalAddress = pageAddress + (cpu.TPR.CA & OS18MASK);
 153   } else {
 154     finalAddress = pageAddress + cpu.TPR.CA;
 155   }
 156   cpu.RSDWH_R1 = RSDWH_R1;
 157 
 158 // ucache hit; housekeeping...
 159   //sim_printf ("hit  %d %05o:%06o\r\n", evcnt, cpu.TPR.TSR, cpu.TPR.CA);
 160 #if defined(HDBG)
 161   hdbgNote ("doAppendCycleOperandRead.h", "hit  %d %05o:%06o\r\n", evcnt, cpu.TPR.TSR, cpu.TPR.CA);
 162 #endif /* if defined(HDBG) */
 163 
 164   cpu.apu.lastCycle = OPERAND_READ;
 165   goto HI;
 166 
 167 
 168 skip:;
 169   //sim_printf ("miss %d %05o:%06o\r\n", evcnt, cpu.TPR.TSR, cpu.TPR.CA);
 170 # if defined(HDBG)
 171   hdbgNote ("doAppendCycleOperandRead.h", "skip %d %05o:%06o\r\n", evcnt, cpu.TPR.TSR, cpu.TPR.CA);
 172 # endif /* if defined(HDBG) */
 173 # if defined(UCACHE_STATS)
 174   cpu.uCache.skips[this] ++;
 175 # endif /* if defined(UCACHE_STATS) */
 176 
 177 
 178 miss:;
 179 
 180   bool nomatch = true;
 181   if (cpu.tweaks.enable_wam) {
 182     // AL39: The associative memory is ignored (forced to "no match") during
 183     // address preparation.
 184     // lptp,lptr,lsdp,lsdr,sptp,sptr,ssdp,ssdr
 185     // Unfortunately, ISOLTS doesn't try to execute any of these in append mode.
 186     // XXX should this be only for OPERAND_READ and OPERAND_STORE?
 187     nomatch = ((i->opcode == 0232 || i->opcode == 0254 ||
 188                 i->opcode == 0154 || i->opcode == 0173) &&
 189                 i->opcodeX ) ||
 190                ((i->opcode == 0557 || i->opcode == 0257) &&
 191                 ! i->opcodeX);
 192   }
 193 
 194   processor_cycle_type lastCycle = cpu.apu.lastCycle;
 195   cpu.apu.lastCycle = OPERAND_READ;
 196 
 197   DBGAPP ("doAppendCycleOperandRead(Entry) XSF %o\r\n", cpu.cu.XSF);
 198 
 199   PNL (L68_ (cpu.apu.state = 0;))
 200 
 201   cpu.RSDWH_R1 = 0;
 202 
 203   cpu.acvFaults = 0;
 204 
 205 //#define FMSG(x) x
 206 #define FMSG(x)
 207   FMSG (char * acvFaultsMsg = "<unknown>";)
 208 
 209 ////////////////////////////////////////
 210 //
 211 // Sheet 1: "START APPEND"
 212 //
 213 ////////////////////////////////////////
 214 
 215 // START APPEND
 216   word3 n = 0; // PRn to be saved to TSN_PRNO
 217 
 218 ////////////////////////////////////////
 219 //
 220 // Sheet 2: "A"
 221 //
 222 ////////////////////////////////////////
 223 
 224 //
 225 //  A:
 226 //    Get SDW
 227 
 228   //PNL (cpu.APUMemAddr = address;)
 229   PNL (cpu.APUMemAddr = cpu.TPR.CA;)
 230 
 231   DBGAPP ("doAppendCycleOperandRead(A)\r\n");
 232 
 233   // is SDW for C(TPR.TSR) in SDWAM?
 234   if (nomatch || ! fetch_sdw_from_sdwam (cpup, cpu.TPR.TSR)) {
 235     // No
 236     DBGAPP ("doAppendCycleOperandRead(A):SDW for segment %05o not in SDWAM\r\n", cpu.TPR.TSR);
 237     DBGAPP ("doAppendCycleOperandRead(A):DSBR.U=%o\r\n", cpu.DSBR.U);
 238 
 239     if (cpu.DSBR.U == 0) {
 240       fetch_dsptw (cpup, cpu.TPR.TSR);
 241 
 242       if (! cpu.PTW0.DF)
 243         doFault (FAULT_DF0 + cpu.PTW0.FC, fst_zero, "doAppendCycleOperandRead(A): PTW0.F == 0");
 244 
 245       if (! cpu.PTW0.U)
 246         modify_dsptw (cpup, cpu.TPR.TSR);
 247 
 248       fetch_psdw (cpup, cpu.TPR.TSR);
 249     } else
 250       fetch_nsdw (cpup, cpu.TPR.TSR); // load SDW0 from descriptor segment table.
 251 
 252     if (cpu.SDW0.DF == 0) {
 253       DBGAPP ("doAppendCycleOperandRead(A): SDW0.F == 0! " "Initiating directed fault\r\n");
 254       // initiate a directed fault ...
 255       doFault (FAULT_DF0 + cpu.SDW0.FC, fst_zero, "SDW0.F == 0");
 256     }
 257     // load SDWAM .....
 258     load_sdwam (cpup, cpu.TPR.TSR, nomatch);
 259   }
 260   DBGAPP ("doAppendCycleOperandRead(A) R1 %o R2 %o R3 %o E %o\r\n", cpu.SDW->R1, cpu.SDW->R2, cpu.SDW->R3, cpu.SDW->E);
 261 
 262   // Yes...
 263   RSDWH_R1 = cpu.RSDWH_R1 = cpu.SDW->R1;
 264 
 265 ////////////////////////////////////////
 266 //
 267 // Sheet 3: "B"
 268 //
 269 ////////////////////////////////////////
 270 
 271 //
 272 // B: Check the ring
 273 //
 274 
 275   DBGAPP ("doAppendCycleOperandRead(B)\r\n");
 276 
 277   // check ring bracket consistency
 278 
 279   //C(SDW.R1) <= C(SDW.R2) <= C(SDW .R3)?
 280   if (! (cpu.SDW->R1 <= cpu.SDW->R2 && cpu.SDW->R2 <= cpu.SDW->R3)) {
 281     // Set fault ACV0 = IRO
 282     cpu.acvFaults |= ACV0;
 283     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 284     FMSG (acvFaultsMsg = "acvFaults(B) C(SDW.R1) <= C(SDW.R2) <= " "C(SDW .R3)";)
 285   }
 286 
 287   // lastCycle == RTCD_OPERAND_FETCH
 288   // if a fault happens between the RTCD_OPERAND_FETCH and the INSTRUCTION_FETCH
 289   // of the next instruction - this happens about 35 time for just booting  and
 290   // shutting down multics -- a stored lastCycle is useless.
 291   // the opcode is preserved across faults and only replaced as the
 292   // INSTRUCTION_FETCH succeeds.
 293   if (lastCycle == RTCD_OPERAND_FETCH)
 294     sim_warn ("%s: lastCycle == RTCD_OPERAND_FETCH opcode %0#o\r\n", __func__, i->opcode);
 295 
 296   //
 297   // B1: The operand is one of: an instruction, data to be read or data to be
 298   //     written
 299   //
 300 
 301   // Is OPCODE call6?
 302   if (i->info->flags & CALL6_INS)
 303     goto E;
 304 
 305   // Transfer
 306   if (i->info->flags & TRANSFER_INS)
 307     goto F;
 308 
 309   //
 310   // check read bracket for read access
 311   //
 312 
 313   DBGAPP ("doAppendCycleOperandRead(B):!STR-OP\r\n");
 314 
 315   // No
 316   // C(TPR.TRR) > C(SDW .R2)?
 317   if (cpu.TPR.TRR > cpu.SDW->R2) {
 318     DBGAPP ("ACV3\r\n");
 319     DBGAPP ("doAppendCycleOperandRead(B) ACV3\r\n");
 320     //Set fault ACV3 = ORB
 321     cpu.acvFaults |= ACV3;
 322     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 323     FMSG (acvFaultsMsg = "acvFaults(B) C(TPR.TRR) > C(SDW .R2)";)
 324   }
 325 
 326   if (cpu.SDW->R == 0) {
 327     // isolts 870
 328     cpu.TPR.TRR = cpu.PPR.PRR;
 329 
 330     //C(PPR.PSR) = C(TPR.TSR)?
 331     if (cpu.PPR.PSR != cpu.TPR.TSR) {
 332       DBGAPP ("ACV4\r\n");
 333       DBGAPP ("doAppendCycleOperandRead(B) ACV4\r\n");
 334       //Set fault ACV4 = R-OFF
 335       cpu.acvFaults |= ACV4;
 336       PNL (L68_ (cpu.apu.state |= apu_FLT;))
 337       FMSG (acvFaultsMsg = "acvFaults(B) C(PPR.PSR) = C(TPR.TSR)";)
 338     //} else {
 339       // sim_warn ("doAppendCycleOperandRead(B) SDW->R == 0 && cpu.PPR.PSR == cpu.TPR.TSR: %0#o\r\n", cpu.PPR.PSR);
 340     }
 341   }
 342 
 343   goto G;
 344 
 345 ////////////////////////////////////////
 346 //
 347 // Sheet 4: "C" "D"
 348 //
 349 ////////////////////////////////////////
 350 
 351 D:;
 352   DBGAPP ("doAppendCycleOperandRead(D)\r\n");
 353 
 354   // transfer or instruction fetch
 355 
 356   // check ring alarm to catch outbound transfers
 357 
 358   if (cpu.rRALR == 0)
 359     goto G;
 360 
 361   // C(PPR.PRR) < RALR?
 362   if (! (cpu.PPR.PRR < cpu.rRALR)) {
 363     DBGAPP ("ACV13\r\n");
 364     DBGAPP ("acvFaults(D) C(PPR.PRR) %o < RALR %o\r\n", cpu.PPR.PRR, cpu.rRALR);
 365     cpu.acvFaults |= ACV13;
 366     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 367     FMSG (acvFaultsMsg = "acvFaults(D) C(PPR.PRR) < RALR";)
 368   }
 369 
 370   goto G;
 371 
 372 ////////////////////////////////////////
 373 //
 374 // Sheet 5: "E"
 375 //
 376 ////////////////////////////////////////
 377 
 378 E:;
 379 
 380   //
 381   // check ring bracket for instruction fetch after call6 instruction
 382   //   (this is the call6 read operand)
 383   //
 384 
 385   DBGAPP ("doAppendCycleOperandRead(E): CALL6\r\n");
 386   DBGAPP ("doAppendCycleOperandRead(E): E %o G %o PSR %05o TSR %05o CA %06o " "EB %06o R %o%o%o TRR %o PRR %o\r\n",
 387           cpu.SDW->E,  cpu.SDW->G,  cpu.PPR.PSR, cpu.TPR.TSR, cpu.TPR.CA, cpu.SDW->EB,
 388           cpu.SDW->R1, cpu.SDW->R2, cpu.SDW->R3, cpu.TPR.TRR, cpu.PPR.PRR);
 389 
 390   //SDW.E set ON?
 391   if (! cpu.SDW->E) {
 392     DBGAPP ("ACV2 b\r\n");
 393     DBGAPP ("doAppendCycleOperandRead(E) ACV2\r\n");
 394     // Set fault ACV2 = E-OFF
 395     cpu.acvFaults |= ACV2;
 396     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 397     FMSG (acvFaultsMsg = "acvFaults(E) SDW .E set OFF";)
 398   }
 399 
 400   //SDW .G set ON?
 401   if (cpu.SDW->G)
 402     goto E1;
 403 
 404   // C(PPR.PSR) = C(TPR.TSR)?
 405   if (cpu.PPR.PSR == cpu.TPR.TSR && ! TST_I_ABS)
 406     goto E1;
 407 
 408   // XXX This doesn't seem right
 409   // EB is word 15; masking address makes no sense; rather 0-extend EB
 410   // Fixes ISOLTS 880-01
 411   if (cpu.TPR.CA >= (word18) cpu.SDW->EB) {
 412     DBGAPP ("ACV7\r\n");
 413     DBGAPP ("doAppendCycleOperandRead(E) ACV7\r\n");
 414     // Set fault ACV7 = NO GA
 415     cpu.acvFaults |= ACV7;
 416     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 417     FMSG (acvFaultsMsg = "acvFaults(E) TPR.CA4-17 >= SDW.CL";)
 418   }
 419 
 420 E1:
 421   DBGAPP ("doAppendCycleOperandRead(E1): CALL6 (cont'd)\r\n");
 422 
 423   // C(TPR.TRR) > SDW.R3?
 424   if (cpu.TPR.TRR > cpu.SDW->R3) {
 425     DBGAPP ("ACV8\r\n");
 426     DBGAPP ("doAppendCycleOperandRead(E) ACV8\r\n");
 427     //Set fault ACV8 = OCB
 428     cpu.acvFaults |= ACV8;
 429     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 430     FMSG (acvFaultsMsg = "acvFaults(E1) C(TPR.TRR) > SDW.R3";)
 431   }
 432 
 433   // C(TPR.TRR) < SDW.R1?
 434   if (cpu.TPR.TRR < cpu.SDW->R1) {
 435     DBGAPP ("ACV9\r\n");
 436     DBGAPP ("doAppendCycleOperandRead(E) ACV9\r\n");
 437     // Set fault ACV9 = OCALL
 438     cpu.acvFaults |= ACV9;
 439     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 440     FMSG (acvFaultsMsg = "acvFaults(E1) C(TPR.TRR) < SDW.R1";)
 441   }
 442 
 443   // C(TPR.TRR) > C(PPR.PRR)?
 444   if (cpu.TPR.TRR > cpu.PPR.PRR) {
 445     // C(PPR.PRR) < SDW.R2?
 446     if (cpu.PPR.PRR < cpu.SDW->R2) {
 447       DBGAPP ("ACV10\r\n");
 448       DBGAPP ("doAppendCycleOperandRead(E) ACV10\r\n");
 449       // Set fault ACV10 = BOC
 450       cpu.acvFaults |= ACV10;
 451       PNL (L68_ (cpu.apu.state |= apu_FLT;))
 452       FMSG (acvFaultsMsg = "acvFaults(E1) C(TPR.TRR) > C(PPR.PRR) && " "C(PPR.PRR) < SDW.R2";)
 453     }
 454   }
 455 
 456   DBGAPP ("doAppendCycleOperandRead(E1): CALL6 TPR.TRR %o SDW->R2 %o\r\n", cpu.TPR.TRR, cpu.SDW->R2);
 457 
 458   // C(TPR.TRR) > SDW.R2?
 459   if (cpu.TPR.TRR > cpu.SDW->R2) {
 460     // SDW.R2 -> C(TPR.TRR)
 461     cpu.TPR.TRR = cpu.SDW->R2;
 462   }
 463 
 464   DBGAPP ("doAppendCycleOperandRead(E1): CALL6 TPR.TRR %o\r\n", cpu.TPR.TRR);
 465 
 466   goto G;
 467 
 468 ////////////////////////////////////////
 469 //
 470 // Sheet 6: "F"
 471 //
 472 ////////////////////////////////////////
 473 
 474 F:;
 475   PNL (L68_ (cpu.apu.state |= apu_PIAU;))
 476   DBGAPP ("doAppendCycleOperandRead(F): transfer or instruction fetch\r\n");
 477 
 478   //
 479   // check ring bracket for instruction fetch
 480   //
 481 
 482   // C(TPR.TRR) < C(SDW .R1)?
 483   // C(TPR.TRR) > C(SDW .R2)?
 484   if (cpu.TPR.TRR < cpu.SDW->R1 || cpu.TPR.TRR > cpu.SDW->R2) {
 485     DBGAPP ("ACV1 a/b\r\n");
 486     DBGAPP ("acvFaults(F) ACV1 !( C(SDW .R1) %o <= C(TPR.TRR) %o <= C(SDW .R2) %o )\r\n", cpu.SDW->R1, cpu.TPR.TRR, cpu.SDW->R2);
 487     cpu.acvFaults |= ACV1;
 488     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 489     FMSG (acvFaultsMsg = "acvFaults(F) C(TPR.TRR) < C(SDW .R1)";)
 490   }
 491   // SDW .E set ON?
 492   if (! cpu.SDW->E) {
 493     DBGAPP ("ACV2 c\r\n");
 494     DBGAPP ("doAppendCycleOperandRead(F) ACV2\r\n");
 495     cpu.acvFaults |= ACV2;
 496     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 497     FMSG (acvFaultsMsg = "acvFaults(F) SDW .E set OFF";)
 498   }
 499 
 500   // C(PPR.PRR) = C(TPR.TRR)?
 501   if (cpu.PPR.PRR != cpu.TPR.TRR) {
 502     DBGAPP ("ACV12\r\n");
 503     DBGAPP ("doAppendCycleOperandRead(F) ACV12\r\n");
 504     //Set fault ACV12 = CRT
 505     cpu.acvFaults |= ACV12;
 506     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 507     FMSG (acvFaultsMsg = "acvFaults(F) C(PPR.PRR) != C(TPR.TRR)";)
 508   }
 509 
 510   goto D;
 511 
 512 ////////////////////////////////////////
 513 //
 514 // Sheet 7: "G"
 515 //
 516 ////////////////////////////////////////
 517 
 518 G:;
 519 
 520   DBGAPP ("doAppendCycleOperandRead(G)\r\n");
 521 
 522   //C(TPR.CA)0,13 > SDW.BOUND?
 523   if (((cpu.TPR.CA >> 4) & 037777) > cpu.SDW->BOUND) {
 524     DBGAPP ("ACV15\r\n");
 525     DBGAPP ("doAppendCycleOperandRead(G) ACV15\r\n");
 526     cpu.acvFaults |= ACV15;
 527     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 528     FMSG (acvFaultsMsg = "acvFaults(G) C(TPR.CA)0,13 > SDW.BOUND";)
 529     DBGAPP ("acvFaults(G) C(TPR.CA)0,13 > SDW.BOUND\r\n" "   CA %06o CA>>4 & 037777 %06o SDW->BOUND %06o",
 530             cpu.TPR.CA, ((cpu.TPR.CA >> 4) & 037777), cpu.SDW->BOUND);
 531   }
 532   bound = cpu.SDW->BOUND;
 533   p = cpu.SDW->P;
 534 
 535   if (cpu.acvFaults) {
 536     DBGAPP ("doAppendCycleOperandRead(G) acvFaults\r\n");
 537     PNL (L68_ (cpu.apu.state |= apu_FLT;))
 538     // Initiate an access violation fault
 539     doFault (FAULT_ACV, (_fault_subtype) {.fault_acv_subtype=cpu.acvFaults},
 540             "ACV fault");
 541   }
 542 
 543   // is segment C(TPR.TSR) paged?
 544   if (cpu.SDW->U)
 545     goto H; // Not paged
 546 
 547   // Yes. segment is paged ...
 548   // is PTW for C(TPR.CA) in PTWAM?
 549 
 550   DBGAPP ("doAppendCycleOperandRead(G) CA %06o\r\n", cpu.TPR.CA);
 551   if (nomatch ||
 552       ! fetch_ptw_from_ptwam (cpup, cpu.SDW->POINTER, cpu.TPR.CA)) {
 553     fetch_ptw (cpup, cpu.SDW, cpu.TPR.CA);
 554     if (! cpu.PTW0.DF) {
 555       // initiate a directed fault
 556       doFault (FAULT_DF0 + cpu.PTW0.FC, (_fault_subtype) {.bits=0},
 557               "PTW0.F == 0");
 558     }
 559     loadPTWAM (cpup, cpu.SDW->POINTER, cpu.TPR.CA, nomatch); // load PTW0 to PTWAM
 560   }
 561 
 562   // Prepage mode?
 563   // check for "uninterruptible" EIS instruction
 564   // ISOLTS-878 02: mvn,cmpn,mvne,ad3d; obviously also
 565   // ad2/3d,sb2/3d,mp2/3d,dv2/3d
 566   // DH03 p.8-13: probably also mve,btd,dtb
 567   if (i->opcodeX && ((i->opcode & 0770)== 0200|| (i->opcode & 0770) == 0220
 568       || (i->opcode & 0770)== 020|| (i->opcode & 0770) == 0300)) {
 569     do_ptw2 (cpup, cpu.SDW, cpu.TPR.CA);
 570   }
 571   goto I;
 572 
 573 ////////////////////////////////////////
 574 //
 575 // Sheet 8: "H", "I"
 576 //
 577 ////////////////////////////////////////
 578 
 579 H:;
 580   DBGAPP ("doAppendCycleOperandRead(H): FANP\r\n");
 581 
 582   paged = false;
 583 
 584   PNL (L68_ (cpu.apu.state |= apu_FANP;))
 585 
 586 
 587 
 588 
 589 
 590 
 591 
 592   set_apu_status (cpup, apuStatus_FANP);
 593 #if defined(HDBG)
 594   hdbgNote ("doAppendCycleOperandRead", "FANP");
 595 #endif /* if defined(HDBG) */
 596   DBGAPP ("doAppendCycleOperandRead(H): SDW->ADDR=%08o CA=%06o\r\n",
 597           cpu.SDW->ADDR, cpu.TPR.CA);
 598 
 599   pageAddress = (cpu.SDW->ADDR & 077777760);
 600   finalAddress = (cpu.SDW->ADDR & 077777760) + cpu.TPR.CA;
 601   finalAddress &= 0xffffff;
 602   PNL (cpu.APUMemAddr = finalAddress;)
 603 
 604   DBGAPP ("doAppendCycleOperandRead(H:FANP): (%05o:%06o) finalAddress=%08o\r\n",
 605           cpu.TPR.TSR, cpu.TPR.CA, finalAddress);
 606 
 607   goto HI;
 608 
 609 I:;
 610 
 611 // Set PTW.M
 612 
 613   DBGAPP ("doAppendCycleOperandRead(I): FAP\r\n");
 614 
 615   paged = true;
 616 
 617 #if defined(HDBG)
 618   hdbgNote ("doAppendCycleOperandRead", "FAP");
 619 #endif /* if defined(HDBG) */
 620   // final address paged
 621   set_apu_status (cpup, apuStatus_FAP);
 622   PNL (L68_ (cpu.apu.state |= apu_FAP;))
 623 
 624   word24 y2 = cpu.TPR.CA % 1024;
 625 
 626   pageAddress = (((word24)cpu.PTW->ADDR & 0777760) << 6);
 627   // AL39: The hardware ignores low order bits of the main memory page
 628   // address according to page size
 629   finalAddress = (((word24)cpu.PTW->ADDR & 0777760) << 6) + y2;
 630   finalAddress &= 0xffffff;
 631   PNL (cpu.APUMemAddr = finalAddress;)
 632 
 633   L68_ (
 634     if (cpu.MR_cache.emr && cpu.MR_cache.ihr)
 635       add_l68_APU_history (cpup, APUH_FAP);
 636   )
 637 
 638   DBGAPP ("doAppendCycleOperandRead(H:FAP): (%05o:%06o) finalAddress=%08o\r\n",
 639           cpu.TPR.TSR, cpu.TPR.CA, finalAddress);
 640 
 641   //goto HI;
 642 
 643 HI:
 644   DBGAPP ("doAppendCycleOperandRead(HI)\r\n");
 645 
 646 #if defined(TEST_UCACHE)
 647   if (cacheHit) {
 648     bool err = false;
 649     if (cachedAddress != pageAddress) {
 650      sim_printf ("cachedAddress %08o != pageAddress %08o\r\n",
 651              cachedAddress, pageAddress);
 652      err = true;
 653     }
 654     if (cachedR1 != RSDWH_R1) {
 655       sim_printf ("cachedR1 %01o != RSDWH_R1 %01o\r\n",
 656               cachedR1, RSDWH_R1);
 657       err = true;
 658     }
 659     if (cachedBound != bound) {
 660       sim_printf ("cachedBound %01o != bound %01o\r\n",
 661               cachedBound, bound);
 662       err = true;
 663     }
 664     if (cachedPaged != paged) {
 665       sim_printf ("cachedPaged %01o != paged %01o\r\n",
 666               cachedPaged, paged);
 667       err = true;
 668     }
 669     if (err) {
 670 # if defined(HDBG)
 671       HDBGPrint ();
 672 # endif /* if defined(HDBG) */
 673       sim_printf ("oprnd read err  %d %05o:%06o\r\n",
 674               evcnt, cpu.TPR.TSR, cpu.TPR.CA);
 675       exit (1);
 676     }
 677     //sim_printf ("hit  %d %05o:%06o\r\n", evcnt, cpu.TPR.TSR, cpu.TPR.CA);
 678 # if defined(HDBG)
 679     hdbgNote ("doAppendCycleOperandRead.h", "test hit %d %05o:%06o\r\n",
 680             evcnt, cpu.TPR.TSR, cpu.TPR.CA);
 681 # endif /* if defined(HDBG) */
 682   } else {
 683     //sim_printf ("miss %d %05o:%06o\r\n", evcnt, cpu.TPR.TSR, cpu.TPR.CA);
 684 # if defined(HDBG)
 685     hdbgNote ("doAppendCycleOperandRead.h", "test miss %d %05o:%06o\r\n",
 686             evcnt, cpu.TPR.TSR, cpu.TPR.CA);
 687 # endif /* if defined(HDBG) */
 688   }
 689 #endif
 690 
 691   ucCacheSave (cpup, this, cpu.TPR.TSR, cpu.TPR.CA, bound, p, pageAddress, RSDWH_R1, paged);
 692 #if defined(TEST_UCACHE)
 693 # if defined(HDBG)
 694   hdbgNote ("doAppendCycleOperandRead.h", "cache %d %u %05o:%06o %05o %o %08o %o %o",
 695           evcnt, this, cpu.TPR.TSR, cpu.TPR.CA, bound, p, pageAddress, RSDWH_R1, paged);
 696 # endif /* if defined(HDBG) */
 697 #endif /* if defined(TEST_UCACHE) */
 698 evcnt ++;
 699 
 700   // isolts 870
 701   cpu.cu.XSF = 1;
 702   sim_debug (DBG_TRACEEXT, & cpu_dev, "loading of cpu.TPR.TSR sets XSF to 1\r\n");
 703 
 704   core_readN (cpup, finalAddress, data, nWords, "OPERAND_READ");
 705 
 706   if (i->info->flags & CALL6_INS)
 707     goto N;
 708 
 709   // Transfer or instruction fetch?
 710   if (i->info->flags & TRANSFER_INS)
 711     goto L;
 712 
 713   // APU data movement?
 714   //  handled above
 715   goto Exit;
 716 
 717 ////////////////////////////////////////
 718 //
 719 // Sheet 10: "K", "L", "M", "N"
 720 //
 721 ////////////////////////////////////////
 722 
 723 L:; // Transfer or instruction fetch
 724 
 725   DBGAPP ("doAppendCycleOperandRead(L)\r\n");
 726 
 727   // Is OPCODE tspn?
 728   if (i->info->flags & TSPN_INS) {
 729     if (i->opcode <= 0273)
 730       n = (i->opcode & 3);
 731     else
 732       n = (i->opcode & 3) + 4;
 733 
 734     // C(PPR.PRR) -> C(PRn .RNR)
 735     // C(PPR.PSR) -> C(PRn .SNR)
 736     // C(PPR.IC) -> C(PRn .WORDNO)
 737     // 000000 -> C(PRn .BITNO)
 738     cpu.PR[n].RNR = cpu.PPR.PRR;
 739 // According the AL39, the PSR is 'undefined' in absolute mode.
 740 // ISOLTS thinks means don't change the operand
 741     if (get_addr_mode (cpup) == APPEND_mode)
 742       cpu.PR[n].SNR = cpu.PPR.PSR;
 743     cpu.PR[n].WORDNO = (cpu.PPR.IC + 1) & MASK18;
 744     SET_PR_BITNO (n, 0);
 745 #if defined(TESTING)
 746     HDBGRegPRW (n, "app tspn");
 747 #endif /* if defined(TESTING) */
 748   }
 749 
 750 // KL:
 751   DBGAPP ("doAppendCycleOperandRead(KL)\r\n");
 752 
 753   // C(TPR.TSR) -> C(PPR.PSR)
 754   cpu.PPR.PSR = cpu.TPR.TSR;
 755   // C(TPR.CA) -> C(PPR.IC)
 756   cpu.PPR.IC = cpu.TPR.CA;
 757 
 758   //goto M;
 759 
 760 M: // Set P
 761   DBGAPP ("doAppendCycleOperandRead(M)\r\n");
 762 
 763   // C(TPR.TRR) = 0?
 764   if (cpu.TPR.TRR == 0) {
 765     // C(SDW.P) -> C(PPR.P)
 766     cpu.PPR.P = p;
 767   } else {
 768     // 0 C(PPR.P)
 769     cpu.PPR.P = 0;
 770   }
 771 
 772   goto Exit;
 773 
 774 N: // CALL6
 775   DBGAPP ("doAppendCycleOperandRead(N)\r\n");
 776 
 777   // C(TPR.TRR) = C(PPR.PRR)?
 778   if (cpu.TPR.TRR == cpu.PPR.PRR) {
 779     // C(PR6.SNR) -> C(PR7.SNR)
 780     cpu.PR[7].SNR = cpu.PR[6].SNR;
 781     DBGAPP ("doAppendCycleOperandRead(N) PR7.SNR = PR6.SNR %05o\r\n", cpu.PR[7].SNR);
 782   } else {
 783     // C(DSBR.STACK) || C(TPR.TRR) -> C(PR7.SNR)
 784     cpu.PR[7].SNR = ((word15) (cpu.DSBR.STACK << 3)) | cpu.TPR.TRR;
 785     DBGAPP ("doAppendCycleOperandRead(N) STACK %05o TRR %o\r\n", cpu.DSBR.STACK, cpu.TPR.TRR);
 786     DBGAPP ("doAppendCycleOperandRead(N) PR7.SNR = STACK||TRR  %05o\r\n", cpu.PR[7].SNR);
 787   }
 788 
 789   // C(TPR.TRR) -> C(PR7.RNR)
 790   cpu.PR[7].RNR = cpu.TPR.TRR;
 791   // 00...0 -> C(PR7.WORDNO)
 792   cpu.PR[7].WORDNO = 0;
 793   // 000000 -> C(PR7.BITNO)
 794   SET_PR_BITNO (7, 0);
 795 #if defined(TESTING)
 796   HDBGRegPRW (7, "app call6");
 797 #endif /* if defined(TESTING) */
 798   // C(TPR.TRR) -> C(PPR.PRR)
 799   cpu.PPR.PRR = cpu.TPR.TRR;
 800   // C(TPR.TSR) -> C(PPR.PSR)
 801   cpu.PPR.PSR = cpu.TPR.TSR;
 802   // C(TPR.CA) -> C(PPR.IC)
 803   cpu.PPR.IC = cpu.TPR.CA;
 804 
 805   goto M;
 806 
 807 Exit:;
 808 
 809   PNL (cpu.APUDataBusOffset = cpu.TPR.CA;)
 810   PNL (cpu.APUDataBusAddr = finalAddress;)
 811 
 812   PNL (L68_ (cpu.apu.state |= apu_FA;))
 813 
 814   DBGAPP ("doAppendCycleOperandRead (Exit) PRR %o PSR %05o P %o IC %06o\r\n", cpu.PPR.PRR, cpu.PPR.PSR, cpu.PPR.P, cpu.PPR.IC);
 815   DBGAPP ("doAppendCycleOperandRead (Exit) TRR %o TSR %05o TBR %02o CA %06o\r\n", cpu.TPR.TRR, cpu.TPR.TSR, cpu.TPR.TBR, cpu.TPR.CA);
 816 
 817   return finalAddress;    // or 0 or -1???
 818 }
 819 #undef TEST_UCACHE

/* [previous][next][first][last][top][bottom][index][help] */