root/src/dps8/dps8_iom.h

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

INCLUDED FROM


   1 /*
   2  * vim: filetype=c:tabstop=4:ai:expandtab
   3  * SPDX-License-Identifier: ICU
   4  * scspell-id: 8e66ce24-f62e-11ec-8690-80ee73e9b8e7
   5  *
   6  * ---------------------------------------------------------------------------
   7  *
   8  * Copyright (c) 2007-2013 Michael Mondy
   9  * Copyright (c) 2012-2016 Harry Reed
  10  * Copyright (c) 2013-2022 Charles Anthony
  11  * Copyright (c) 2021-2025 The DPS8M Development Team
  12  *
  13  * This software is made available under the terms of the ICU License.
  14  * See the LICENSE.md file at the top-level directory of this distribution.
  15  *
  16  * ---------------------------------------------------------------------------
  17  */
  18 
  19 // IDCW Instruction                   18-20  111
  20 // TDCW Transfer                      18-20 !111,  22-23 10
  21 // IOTD IO Transfer and disconnect    18-20 !111,  22-23 00
  22 // IOTP IO Transfer and proceed       18-20 !111,  22-23 01
  23 // IONTP IO Non-Transfer and proceed  18-20 !111,  22-23 11
  24 
  25 #if defined(IO_THREADZ)
  26 extern __thread uint this_iom_idx;
  27 extern __thread uint this_chan_num;
  28 //extern __thread bool thisIOMHaveLock;
  29 #endif
  30 
  31 //typedef enum
  32   //{
  33     //cm_LPW_init_state,       // No TDCWs encountered; state is:
  34                              ////    PCW64 (pcw64_pge): on   PAGE CHAN
  35                              ////    PCW64 (pcw64_pge): off  EXT MODE CHAN
  36     //cm_real_LPW_real_DCW,
  37     //cm_ext_LPW_real_DCW,
  38     //cm_paged_LPW_seg_DCW
  39   //} chanMode_t;
  40 
  41 typedef enum chanStat
  42   {
  43     chanStatNormal          = 0,
  44     chanStatUnexpectedPCW   = 1,
  45     chanStatInvalidInstrPCW = 2,
  46     chanStatIncorrectDCW    = 3,
  47     chanStatIncomplete      = 4,
  48     chanStatUnassigned      = 5,
  49     chanStatParityErrPeriph = 6,
  50     chanStatParityErrBus    = 7
  51   } chanStat;
  52 
  53 // Due to lack of documentation, chan_cmd is largely ignored
  54 //
  55 // iom_chan_control_words.incl.pl1
  56 //
  57 //   SINGLE_RECORD       init ("00"b3),
  58 //   NONDATA             init ("02"b3),
  59 //   MULTIRECORD         init ("06"b3),
  60 //   SINGLE_CHARACTER    init ("10"b3)
  61 //
  62 // bound_tolts_/mtdsim_.pl1
  63 //
  64 //    idcw.chan_cmd = "40"b3;           /* otherwise set special cont. cmd */
  65 //
  66 // bound_io_tools/exercise_disk.pl1
  67 //
  68 //   idcw.chan_cmd = INHIB_AUTO_RETRY; /* inhibit mpc auto retries */
  69 //   dcl     INHIB_AUTO_RETRY       bit (6) int static init ("010001"b);  // 021
  70 //
  71 // poll_mpc.pl1:
  72 //  /* Build dcw list to get statistics from EURC MPC */
  73 //  idcw.chan_cmd = "41"b3;            /* Indicate special controller command */
  74 //  /* Build dcw list to get configuration and statistics from DAU MSP */
  75 //  idcw.chan_cmd = "30"b3;                           /* Want list in dev# order */
  76 //
  77 // tape_ioi_io.pl1:
  78 //   idcw.chan_cmd = "03"b3;          /* data security erase */
  79 //   dcw.chan_cmd = "30"b3;           /* use normal values, auto-retry */
  80 
  81 // iom_word_macros.incl.alm
  82 //
  83 // Channel control
  84 #define CHAN_CTRL_TERMINATE 0
  85 #define CHAN_CTRL_PROCEED   2
  86 #define CHAN_CTRL_MARKER    3
  87 
  88 // Channel command
  89 #define CHAN_CMD_RECORD      0
  90 #define CHAN_CMD_NONDATA     2
  91 #define CHAN_CMD_MULTIRECORD 6
  92 #define CHAN_CMD_CHARACTER   8
  93 
  94 #define IS_IDCW(p)     ((p)->DCW_18_20_CP == 07)
  95 #define IS_NOT_IDCW(p) ((p)->DCW_18_20_CP != 07)
  96 #define IS_TDCW(p)     ((p)->DCW_18_20_CP !=  7 && (p)->DDCW_22_23_TYPE == 2)
  97 #define IS_IOTD(p)     ((p)->DCW_18_20_CP !=  7 && (p)->DDCW_22_23_TYPE == 0)
  98 #define IS_IONTP(p)    ((p)->DCW_18_20_CP !=  7 && (p)->DDCW_22_23_TYPE == 3)
  99 
 100 // exercise_disk.pl1
 101 #define CHAN_CMD_INHIB_AUTO_RETRY    021
 102 // load_mpc.pl1
 103 #define CHAN_CMD_SPECIAL_CTLR        040
 104 // poll_mpc.pl1
 105 #define CHAN_CMD_DEV_ORDER           030
 106 #define CHAN_CMD_SPECIAL_CTLR2       041
 107 // tape_ioi_io.pl1
 108 #define CHAN_CMD_DATA_SECURITY_ERASE  03
 109 #define CHAN_CMD_NORM_AUTO_TRY       030
 110 
 111 typedef volatile struct
 112   {
 113 // scratch pad
 114 
 115     // packed LPW
 116     word36 LPW;
 117     // unpacked LPW
 118     word18 LPW_DCW_PTR;
 119     word1 LPW_18_RES;
 120     word1 LPW_19_REL;
 121     word1 LPW_20_AE;
 122     word1 LPW_21_NC;
 123     word1 LPW_22_TAL;
 124     word1 LPW_23_REL;
 125     word12 LPW_TALLY;
 126 
 127     // packed LPWX
 128     word36 LPWX;
 129     // unpacked LPWX
 130     word18 LPWX_BOUND; // MOD 2 (pg B16) 0-2^19; ie val = LPX_BOUND * 2
 131     word18 LPWX_SIZE;  // MOD 1 (pg B16) 0-2^18
 132 
 133 // PCW_63_PTP indicates paging mode; indicates that a page table
 134 // is available.
 135 // XXX pg B11: cleared by a terminate interrupt service with the
 136 // character size bit of the transaction command = 1. (bit 32)
 137 // what is the 'transaction command.?
 138 
 139     // packed PCW
 140     word36 PCW0, PCW1;
 141     // unpacked PCW
 142     word6 PCW_CHAN;
 143     word6 PCW_AE;
 144     // Pg B2: "Absolute location (MOD 64) of the channels Page Table"
 145     word18 PCW_PAGE_TABLE_PTR;
 146     word1 PCW_63_PTP;
 147     word1 PCW_64_PGE;
 148     word1 PCW_65_AUX; // XXX
 149     word1 PCW_21_MSK; // Sometimes called 'M' // see 3.2.2, pg 25
 150 
 151     // packed DCW
 152     word36 DCW;
 153     // unpacked DCW
 154     // TDCW only
 155     word18 TDCW_DATA_ADDRESS;
 156     word1  TDCW_34_RES;
 157     word1  TDCW_35_REL;
 158     // TDCW, PCW 64 = 0
 159     word1  TDCW_33_EC;
 160     // TDCW, PCW 64 = 1
 161     word1  TDCW_31_SEG;
 162     word1  TDCW_32_PDTA;
 163     word1  TDCW_33_PDCW;
 164     // IDCW only
 165     word6  IDCW_DEV_CMD;
 166     word6  IDCW_DEV_CODE;
 167     word6  IDCW_AE;
 168     word1  IDCW_EC;
 169     word2  IDCW_CHAN_CTRL; // 0 terminate, 2 process, 3 marker
 170     word6  IDCW_CHAN_CMD;
 171     // POLTS sets this to one if there are IOTxes.
 172     word6  IDCW_COUNT;
 173     // DDCW only
 174     /*word18*/ uint DDCW_ADDR; // Allow overflow detection
 175     word12 DDCW_TALLY;
 176     word2  DDCW_22_23_TYPE; // '2' indicates TDCW
 177     // xDCW
 178     word3  DCW_18_20_CP; // '7' indicates IDCW
 179     // XXX pg 30; the indirect data service needs to use this.
 180 
 181     word6 ADDR_EXT; // 3.2.2, 3.2.3.1
 182     word1 SEG;  // pg B21
 183 
 184     enum { /* PGE */ cm1, cm2, cm3a, cm3b, cm4, cm5,
 185            /* EXT */ cm1e, cm2e } chanMode;
 186 
 187 // XXX CP XXXX
 188 // "Specifies the positions of the first character with the first word
 189 // of the block. The byte size, defined by the channel, determines
 190 // what CP values are valid/
 191 //  6 bit: 0-5; 9 bit: 0-4; 18 bit: 0-1; 36 bit: 0-6
 192 //
 193 // For word channels, CP is sent to the channel during list service,
 194 // and is zeros when placed in the mailbox for subsequent data
 195 // services to the channel.
 196 //
 197 //  [CAC: I think that this can be elided. To implement correctly,
 198 //  iom_list_service and/or doPayloadChannel would have to know the
 199 //  word or sub-word functionality of the channel. But it would
 200 //  be simpler to let the device handler just access the CP data,
 201 //  and make it's own decisions about "zeros". After all, it is
 202 //  not clear what a non-zero CP means for a word channel.]
 203 //
 204 // For sub-word channels which depend on IOM Central for packing and
 205 // unpacking words, [IOM Central uses and updates the CP to access
 206 // sub-words].
 207 //
 208 //  [CAC: Again, I think that this can be elided. It is simpler to
 209 //  to have the device handler pack and unpack.]
 210 //
 211 
 212 // LPW addressing mode
 213 
 214     //enum { LPW_REAL, LPW_EXT, LPW_PAGED, LPW_SEG } lpwAddrMode;
 215 
 216     // pg B2: "Scratchpad area for two Page Table Words ... one
 217     // for the DCW List (PTW-LPW) and one for the data (PTW-DCW).
 218 
 219     // PTW format: (pg B8)
 220     //   4-17: Address of page table, mod 64
 221     //   31: WRC: Write control bit (1: page may be written)
 222     //   32: HSE: Housekeeping
 223     //   33: PGP: Page present
 224     // To read or write PGP must be 1
 225     // To write, WRC must be 1, HSE 0; system fault 15 on fail
 226 // pg b8: PTWs are used iff (indirect store and LPw 23 (segmented)), or
 227 // direct store.
 228 //
 229 //  ADDR  0-13 <- PTW 4-17
 230 //       14-23 <- LPW 8-17; DCW 8-17; direct channel address 14-23
 231 
 232     word36 PTW_DCW;  // pg B8.
 233     word36 PTW_LPW;  // pg B6.
 234 
 235 // pg b11 defines two PTW flags to indicate the validity of the
 236 // PTW_DCW and PTW_LPW; it is simpler to simply always fetch
 237 // the PTWs on demand.
 238 
 239 //  flag
 240     //chanMode_t chanMode;
 241 
 242     // true if the DCW is from a PCW
 243     bool isPCW;
 244 
 245     // Information accumulated for status service.
 246     word12 stati;
 247     uint dev_code;
 248     // Initialized to IDCW_COUNT; decremented when a IDCW that expects an IOTx is processed by the channel.
 249     // XXX POLTS console code drives this; if it turns out to be common across channels, the decrement should
 250     // be moved into the IOM.
 251     word6 recordResidue;
 252     word12 tallyResidue;
 253     word3 charPos;
 254     bool isRead;
 255     // isOdd can be ignored; see https://ringzero.wikidot.com/wiki:cac-2015-10-22
 256     // bool isOdd;
 257     bool initiate;
 258 
 259     chanStat chanStatus;
 260 
 261     bool lsFirst;
 262 
 263     bool wasTDCW;
 264 
 265     bool masked;
 266 
 267     bool in_use;
 268 
 269     bool start;
 270 
 271   } iom_chan_data_t;
 272 
 273 extern iom_chan_data_t iom_chan_data [N_IOM_UNITS_MAX] [MAX_CHANNELS];
 274 
 275 extern DEVICE iom_dev;
 276 
 277 // Indirect data service data type
 278 typedef enum
 279   {
 280     idsTypeW36  // Incoming data is array of word36
 281   } idsType;
 282 
 283 typedef enum
 284   {
 285     direct_load,
 286     direct_store,
 287     direct_read_clear,
 288   } iom_direct_data_service_op;
 289 
 290 
 291 
 292 
 293 
 294 
 295 
 296 
 297 
 298 
 299 
 300 
 301 
 302 
 303 
 304 
 305 
 306 
 307 
 308 
 309 
 310 
 311 
 312 
 313 
 314 
 315 
 316 
 317 
 318 
 319 
 320 
 321 
 322 
 323 
 324 
 325 
 326 #define IOM_MBX_LPW     0
 327 #define IOM_MBX_LPWX    1
 328 #define IOM_MBX_SCW     2
 329 #define IOM_MBX_DCW     3
 330 
 331 /* From AN70-1 May84
 332  *  ... The IOM determines an interrupt
 333  * number. (The interrupt number is a five bit value, from 0 to 31.
 334  * The high order bits are the interrupt level.
 335  *
 336  * 0 - system fault
 337  * 1 - terminate
 338  * 2 - marker
 339  * 3 - special
 340  *
 341  * The low order three bits determines the IOM and IOM channel
 342  * group.
 343  *
 344  * 0 - IOM 0 channels 32-63
 345  * 1 - IOM 1 channels 32-63
 346  * 2 - IOM 2 channels 32-63
 347  * 3 - IOM 3 channels 32-63
 348  * 4 - IOM 0 channels 0-31
 349  * 5 - IOM 1 channels 0-31
 350  * 6 - IOM 2 channels 0-31
 351  * 7 - IOM 3 channels 0-31
 352  *
 353  *   3  3     3   3   3
 354  *   1  2     3   4   5
 355  *  ---------------------
 356  *  | pic | group | iom |
 357  *  -----------------------------
 358  *       2       1     2
 359  */
 360 
 361 enum iomImwPics
 362   {
 363     imwSystemFaultPic = 0,
 364     imwTerminatePic   = 1,
 365     imwMarkerPic      = 2,
 366     imwSpecialPic     = 3
 367   };
 368 
 369 int send_general_interrupt (uint iom_unit_idx, uint chan, enum iomImwPics pic);
 370 
 371 int send_special_interrupt (uint iom_unit_idx, uint chanNum, uint devCode,
 372                             word8 status0, word8 status1);
 373 //
 374 // iom_cmd_t returns:
 375 //
 376 //  0: ok
 377 //  1; ignored cmd, drop connect.
 378 //  2: did command, don't do DCW list
 379 //  3; command pending, don't sent terminate interrupt
 380 // -1: error
 381 
 382 //#define IOM_CMD_OK       0
 383 //#define IOM_CMD_IGNORED  1
 384 //#define IOM_CMD_NO_DCW   2
 385 //#define IOM_CMD_PENDING  3
 386 //#define IOM_CMD_ERROR   -1
 387 
 388 typedef enum
 389   {
 390      IOM_CMD_ERROR   = -1,
 391      IOM_CMD_PROCEED =  0,
 392      IOM_CMD_RESIDUE,
 393      IOM_CMD_DISCONNECT,
 394      IOM_CMD_PENDING
 395   } iom_cmd_rc_t;
 396 
 397 typedef iom_cmd_rc_t iom_cmd_t (uint iom_unit_idx, uint chan);
 398 int iom_list_service (uint iom_unit_idx, uint chan,
 399                       bool * ptro, bool * sendp, bool * uffp);
 400 int send_terminate_interrupt (uint iom_unit_idx, uint chanNum);
 401 void iom_interrupt (uint scuUnitNum, uint iom_unit_idx);
 402 void iom_direct_data_service (uint iom_unit_idx, uint chan, word24 daddr, word36 * data,
 403                               iom_direct_data_service_op op);
 404 void iom_indirect_data_service (uint iom_unit_idx, uint chan, word36 * data,
 405                                 uint * cnt, bool write);
 406 void iom_init (void);
 407 int send_marker_interrupt (uint iom_unit_idx, int chan);
 408 #if defined(PANEL68)
 409 void do_boot (void);
 410 #endif
 411 #if defined(IO_THREADZ)
 412 void * iom_thread_main (void * arg);
 413 void * chan_thread_main (void * arg);
 414 #endif
 415 void iom_core_read (uint iom_unit_idx, word24 addr, word36 *data, UNUSED const char * ctx);
 416 void iom_core_read2 (uint iom_unit_idx, word24 addr, word36 *even, word36 *odd, UNUSED const char * ctx);
 417 void iom_core_write (uint iom_unit_idx, word24 addr, word36 data, UNUSED const char * ctx);
 418 void iom_core_write2 (uint iom_unit_idx, word24 addr, word36 even, word36 odd, UNUSED const char * ctx);
 419 void iom_core_read_lock (uint iom_unit_idx, word24 addr, word36 *data, UNUSED const char * ctx);
 420 void iom_core_write_unlock (uint iom_unit_idx, word24 addr, word36 data, UNUSED const char * ctx);
 421 t_stat iom_unit_reset_idx (uint iom_unit_idx);
 422 
 423 #if defined(IO_ASYNC_PAYLOAD_CHAN) || defined(IO_ASYNC_PAYLOAD_CHAN_THREAD)
 424 void iomProcess (void);
 425 #endif
 426 
 427 char iomChar (uint iomUnitIdx);
 428 #if defined(TESTING)
 429 void dumpDCW (word36 DCW, word1 LPW_23_REL);
 430 #endif

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