blob: 2bc725ba478bf6f774d6a073f2e859a1ee7e5df3 [file] [log] [blame] [raw]
/* i7094_io.c: IBM 7094 I/O subsystem (channels)
Copyright (c) 2003-2006, 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.
chana..chanh I/O channels
Notes on channels and CTSS.
- CTSS B-core is supported by the addition of a 16th bit to the current
address field of the channel command. Both the channel location counter
and the channel current address register are widened to 16b. Thus,
channel programs can run in B-core, and channel transfers can access B-core.
CTSS assumes that a channel command which starts a transfer in B-core
will not access A-core; the 16th bit does not increment.
- The channel start commands (RCHx and LCHx) incorporate the A-core/B-core
select as part of effective address generation. CTSS does not relocate
RCHx and LCHx target addresses; because the relocation indicator is
always zero, it's impossible to tell whether the protection indicator
affects address generation.
- The CTSS protection RPQ does not cover channel operations. Thus, CTSS
must inspect and vet all channel programs initiated by user mode programs,
notably the background processor FMS. CTSS inspects in-progress 7607
channel programs to make sure than either the nostore bit or the B-core
bit is set; thus, SCHx must store all 16b of the current address.
*/
#include "i7094_defs.h"
#define CHAMASK ((cpu_model & I_CT)? PAMASK: AMASK) /* chan addr mask */
#define CHAINC(x) (((x) & ~AMASK) | (((x) + 1) & AMASK))
typedef struct {
char *name;
uint32 flags;
} DEV_CHAR;
uint32 ch_sta[NUM_CHAN]; /* channel state */
uint32 ch_dso[NUM_CHAN]; /* data select op */
uint32 ch_dsu[NUM_CHAN]; /* data select unit */
uint32 ch_ndso[NUM_CHAN]; /* non-data select op */
uint32 ch_ndsu[NUM_CHAN]; /* non-data select unit */
uint32 ch_flags[NUM_CHAN]; /* flags */
uint32 ch_clc[NUM_CHAN]; /* chan loc ctr */
uint32 ch_op[NUM_CHAN]; /* channel op */
uint32 ch_wc[NUM_CHAN]; /* word count */
uint32 ch_ca[NUM_CHAN]; /* core address */
uint32 ch_lcc[NUM_CHAN]; /* control cntr (7909) */
uint32 ch_cnd[NUM_CHAN]; /* cond reg (7909) */
uint32 ch_sms[NUM_CHAN]; /* cond mask reg (7909) */
t_uint64 ch_ar[NUM_CHAN]; /* assembly register */
uint32 ch_idf[NUM_CHAN]; /* channel input data flags */
DEVICE *ch2dev[NUM_CHAN] = { NULL };
uint32 ch_tpoll = 5; /* channel poll */
extern t_uint64 *M;
extern uint32 cpu_model, data_base;
extern uint32 hst_ch;
extern uint32 ch_req;
extern uint32 chtr_inht, chtr_inhi, chtr_enab;
extern uint32 ind_ioc;
extern uint32 chtr_clk;
extern DEVICE cdr_dev, cdp_dev;
extern DEVICE lpt_dev;
extern DEVICE mt_dev[NUM_CHAN];
extern DEVICE drm_dev;
extern DEVICE dsk_dev;
extern DEVICE com_dev;
extern int32 sim_brk_summ;
t_stat ch_reset (DEVICE *dptr);
t_stat ch6_svc (UNIT *uptr);
t_stat ch_set_enable (UNIT *uptr, int32 val, char *cptr, void *desc);
t_stat ch_set_disable (UNIT *uptr, int32 val, char *cptr, void *desc);
t_stat ch_show_type (FILE *st, UNIT *uptr, int32 val, void *desc);
DEVICE *ch_find_dev (uint32 ch, uint32 unit);
t_stat ch6_sel (uint32 ch, uint32 sel, uint32 unit, uint32 sta);
t_bool ch6_rd_putw (uint32 ch);
t_stat ch6_wr_getw (uint32 ch, t_bool eorz);
t_stat ch6_new_cmd (uint32 ch, t_bool ch_ld);
t_stat ch6_ioxt (uint32 ch);
void ch6_iosp_cclr (uint32 ch);
t_stat ch9_new_cmd (uint32 ch);
t_stat ch9_exec_cmd (uint32 ch, t_uint64 ir);
t_stat ch9_sel (uint32 ch, uint32 sel);
t_stat ch9_wr (uint32 ch, t_uint64 dat, uint32 fl);
t_stat ch9_rd_putw (uint32 ch);
t_stat ch9_wr_getw (uint32 ch);
void ch9_eval_int (uint32 ch, uint32 iflags);
DEVICE *ch_map_flags (uint32 ch, int32 fl);
extern CTAB *sim_vm_cmd;
extern t_stat ch_bkpt (uint32 ch, uint32 clc);
const uint32 col_masks[12] = { /* row 9,8,..,0,11,12 */
00001, 00002, 00004,
00010, 00020, 00040,
00100, 00200, 00400,
01000, 02000, 04000
};
const t_uint64 bit_masks[36] = {
0000000000001, 0000000000002, 0000000000004,
0000000000010, 0000000000020, 0000000000040,
0000000000100, 0000000000200, 0000000000400,
0000000001000, 0000000002000, 0000000004000,
0000000010000, 0000000020000, 0000000040000,
0000000100000, 0000000200000, 0000000400000,
0000001000000, 0000002000000, 0000004000000,
0000010000000, 0000020000000, 0000040000000,
0000100000000, 0000200000000, 0000400000000,
0001000000000, 0002000000000, 0004000000000,
0010000000000, 0020000000000, 0040000000000,
0100000000000, 0200000000000, 0400000000000
};
const DEV_CHAR dev_table[] = {
{ "729", 0 },
{ "TAPE", 0 },
{ "7289", DEV_7289 },
{ "DRUM", DEV_7289 },
{ "7631", DEV_7909|DEV_7631 },
{ "FILE", DEV_7909|DEV_7631 },
{ "7750", DEV_7909|DEV_7750 },
{ "COMM", DEV_7909|DEV_7750 },
{ NULL },
};
const char *sel_name[] = {
"UNK", "RDS", "WRS", "SNS", "CTL", "FMT", "UNK", "UNK",
"WEF", "WBT", "BSR", "BSF", "REW", "RUN", "SDN", "UNK"
};
/* Channel data structures */
UNIT ch_unit[NUM_CHAN] = {
{ UDATA (&ch6_svc, 0, 0) },
{ UDATA (&ch6_svc, 0, 0) },
{ UDATA (&ch6_svc, 0, 0) },
{ UDATA (&ch6_svc, 0, 0) },
{ UDATA (&ch6_svc, 0, 0) },
{ UDATA (&ch6_svc, 0, 0) },
{ UDATA (&ch6_svc, 0, 0) },
{ UDATA (&ch6_svc, 0, 0) }
};
MTAB ch_mod[] = {
{ MTAB_XTD|MTAB_VDV, 0, "TYPE", NULL,
NULL, &ch_show_type, NULL },
{ MTAB_XTD|MTAB_VDV, 0, NULL, "ENABLED",
&ch_set_enable, NULL, NULL },
{ MTAB_XTD|MTAB_VDV, 0, NULL, "DISABLED",
&ch_set_disable, NULL, NULL },
{ 0 }
};
REG cha_reg[] = {
{ ORDATA (STA, ch_sta[CH_A], 8) },
{ ORDATA (DSC, ch_dso[CH_A], 4) },
{ ORDATA (DSU, ch_dsu[CH_A], 9) },
{ ORDATA (NDSC, ch_ndso[CH_A], 4) },
{ ORDATA (NDSU, ch_ndsu[CH_A], 9) },
{ ORDATA (FLAGS, ch_flags[CH_A], 30) },
{ ORDATA (IDF, ch_idf[CH_A], 2) },
{ ORDATA (OP, ch_op[CH_A], 5) },
{ ORDATA (CLC, ch_clc[CH_A], 16) },
{ ORDATA (WC, ch_wc[CH_A], 15) },
{ ORDATA (CA, ch_ca[CH_A], 16) },
{ ORDATA (AR, ch_ar[CH_A], 36) },
{ ORDATA (CND, ch_cnd[CH_A], 6), REG_HRO },
{ ORDATA (LCC, ch_lcc[CH_A], 6), REG_HRO },
{ ORDATA (SMS, ch_sms[CH_A], 7), REG_HRO },
{ 0 }
};
REG chb_reg[] = {
{ ORDATA (STATE, ch_sta[CH_B], 8) },
{ ORDATA (DSC, ch_dso[CH_B], 4) },
{ ORDATA (DSU, ch_dsu[CH_B], 9) },
{ ORDATA (NDSC, ch_ndso[CH_B], 4) },
{ ORDATA (NDSU, ch_ndsu[CH_B], 9) },
{ ORDATA (FLAGS, ch_flags[CH_B], 30) },
{ ORDATA (IDF, ch_idf[CH_B], 2) },
{ ORDATA (OP, ch_op[CH_B], 5) },
{ ORDATA (CLC, ch_clc[CH_B], 16) },
{ ORDATA (WC, ch_wc[CH_B], 15) },
{ ORDATA (CA, ch_ca[CH_B], 16) },
{ ORDATA (AR, ch_ar[CH_B], 36) },
{ ORDATA (CND, ch_cnd[CH_B], 6) },
{ ORDATA (LCC, ch_lcc[CH_B], 6) },
{ ORDATA (SMS, ch_sms[CH_B], 7) },
{ 0 }
};
REG chc_reg[] = {
{ ORDATA (STATE, ch_sta[CH_C], 8) },
{ ORDATA (DSC, ch_dso[CH_C], 4) },
{ ORDATA (DSU, ch_dsu[CH_C], 9) },
{ ORDATA (NDSC, ch_ndso[CH_C], 4) },
{ ORDATA (NDSU, ch_ndsu[CH_C], 9) },
{ ORDATA (FLAGS, ch_flags[CH_C], 30) },
{ ORDATA (IDF, ch_idf[CH_C], 2) },
{ ORDATA (OP, ch_op[CH_C], 5) },
{ ORDATA (CLC, ch_clc[CH_C], 16) },
{ ORDATA (WC, ch_wc[CH_C], 15) },
{ ORDATA (CA, ch_ca[CH_C], 16) },
{ ORDATA (AR, ch_ar[CH_C], 36) },
{ ORDATA (CND, ch_cnd[CH_C], 6) },
{ ORDATA (LCC, ch_lcc[CH_C], 6) },
{ ORDATA (SMS, ch_sms[CH_C], 7) },
{ 0 }
};
REG chd_reg[] = {
{ ORDATA (STATE, ch_sta[CH_D], 8) },
{ ORDATA (DSC, ch_dso[CH_D], 4) },
{ ORDATA (DSU, ch_dsu[CH_D], 9) },
{ ORDATA (NDSC, ch_ndso[CH_D], 4) },
{ ORDATA (NDSU, ch_ndsu[CH_D], 9) },
{ ORDATA (FLAGS, ch_flags[CH_D], 30) },
{ ORDATA (IDF, ch_idf[CH_D], 2) },
{ ORDATA (OP, ch_op[CH_D], 5) },
{ ORDATA (CLC, ch_clc[CH_D], 16) },
{ ORDATA (WC, ch_wc[CH_D], 15) },
{ ORDATA (CA, ch_ca[CH_D], 16) },
{ ORDATA (AR, ch_ar[CH_D], 36) },
{ ORDATA (CND, ch_cnd[CH_D], 6) },
{ ORDATA (LCC, ch_lcc[CH_D], 6) },
{ ORDATA (SMS, ch_sms[CH_D], 7) },
{ 0 }
};
REG che_reg[] = {
{ ORDATA (STATE, ch_sta[CH_E], 8) },
{ ORDATA (DSC, ch_dso[CH_E], 4) },
{ ORDATA (DSU, ch_dsu[CH_E], 9) },
{ ORDATA (NDSC, ch_ndso[CH_E], 4) },
{ ORDATA (NDSU, ch_ndsu[CH_E], 9) },
{ ORDATA (FLAGS, ch_flags[CH_E], 30) },
{ ORDATA (IDF, ch_idf[CH_E], 2) },
{ ORDATA (OP, ch_op[CH_E], 5) },
{ ORDATA (CLC, ch_clc[CH_E], 16) },
{ ORDATA (WC, ch_wc[CH_E], 15) },
{ ORDATA (CA, ch_ca[CH_E], 16) },
{ ORDATA (AR, ch_ar[CH_E], 36) },
{ ORDATA (CND, ch_cnd[CH_E], 6) },
{ ORDATA (LCC, ch_lcc[CH_E], 6) },
{ ORDATA (SMS, ch_sms[CH_E], 7) },
{ 0 }
};
REG chf_reg[] = {
{ ORDATA (STATE, ch_sta[CH_F], 8) },
{ ORDATA (DSC, ch_dso[CH_F], 4) },
{ ORDATA (DSU, ch_dsu[CH_F], 9) },
{ ORDATA (NDSC, ch_ndso[CH_F], 4) },
{ ORDATA (NDSU, ch_ndsu[CH_F], 9) },
{ ORDATA (FLAGS, ch_flags[CH_F], 30) },
{ ORDATA (IDF, ch_idf[CH_F], 2) },
{ ORDATA (OP, ch_op[CH_F], 5) },
{ ORDATA (CLC, ch_clc[CH_F], 16) },
{ ORDATA (WC, ch_wc[CH_F], 15) },
{ ORDATA (CA, ch_ca[CH_F], 16) },
{ ORDATA (AR, ch_ar[CH_F], 36) },
{ ORDATA (CND, ch_cnd[CH_F], 6) },
{ ORDATA (LCC, ch_lcc[CH_F], 6) },
{ ORDATA (SMS, ch_sms[CH_F], 7) },
{ 0 }
};
REG chg_reg[] = {
{ ORDATA (STATE, ch_sta[CH_G], 8) },
{ ORDATA (DSC, ch_dso[CH_G], 4) },
{ ORDATA (DSU, ch_dsu[CH_G], 9) },
{ ORDATA (NDSC, ch_ndso[CH_G], 4) },
{ ORDATA (NDSU, ch_ndsu[CH_G], 9) },
{ ORDATA (FLAGS, ch_flags[CH_G], 30) },
{ ORDATA (IDF, ch_idf[CH_G], 2) },
{ ORDATA (OP, ch_op[CH_G], 5) },
{ ORDATA (CLC, ch_clc[CH_G], 16) },
{ ORDATA (WC, ch_wc[CH_G], 15) },
{ ORDATA (CA, ch_ca[CH_G], 16) },
{ ORDATA (AR, ch_ar[CH_G], 36) },
{ ORDATA (CND, ch_cnd[CH_G], 6) },
{ ORDATA (LCC, ch_lcc[CH_G], 6) },
{ ORDATA (SMS, ch_sms[CH_G], 7) },
{ 0 }
};
REG chh_reg[] = {
{ ORDATA (STATE, ch_sta[CH_H], 8) },
{ ORDATA (DSC, ch_dso[CH_H], 4) },
{ ORDATA (DSU, ch_dsu[CH_H], 9) },
{ ORDATA (NDSC, ch_ndso[CH_H], 4) },
{ ORDATA (NDSU, ch_ndsu[CH_H],9) },
{ ORDATA (FLAGS, ch_flags[CH_H], 30) },
{ ORDATA (IDF, ch_idf[CH_H], 2) },
{ ORDATA (OP, ch_op[CH_H], 5) },
{ ORDATA (CLC, ch_clc[CH_H], 16) },
{ ORDATA (WC, ch_wc[CH_H], 15) },
{ ORDATA (CA, ch_ca[CH_H], 16) },
{ ORDATA (AR, ch_ar[CH_H], 36) },
{ ORDATA (CND, ch_cnd[CH_H], 6) },
{ ORDATA (LCC, ch_lcc[CH_H], 6) },
{ ORDATA (SMS, ch_sms[CH_H], 7) },
{ 0 }
};
DEVICE ch_dev[NUM_CHAN] = {
{
"CHANA", &ch_unit[CH_A], cha_reg, ch_mod,
1, 8, 8, 1, 8, 8,
NULL, NULL, &ch_reset,
NULL, NULL, NULL,
NULL, 0
},
{
"CHANB", &ch_unit[CH_B], chb_reg, ch_mod,
1, 8, 8, 1, 8, 8,
NULL, NULL, &ch_reset,
NULL, NULL, NULL,
NULL, DEV_DISABLE | DEV_DIS
},
{
"CHANC", &ch_unit[CH_C], chc_reg, ch_mod,
1, 8, 8, 1, 8, 8,
NULL, NULL, &ch_reset,
NULL, NULL, NULL,
NULL, DEV_DISABLE | DEV_DIS
},
{
"CHAND", &ch_unit[CH_D], chd_reg, ch_mod,
1, 8, 8, 1, 8, 8,
NULL, NULL, &ch_reset,
NULL, NULL, NULL,
NULL, DEV_DISABLE | DEV_DIS
},
{
"CHANE", &ch_unit[CH_E], che_reg, ch_mod,
1, 8, 8, 1, 8, 8,
NULL, NULL, &ch_reset,
NULL, NULL, NULL,
NULL, DEV_DISABLE | DEV_DIS
},
{
"CHANF", &ch_unit[CH_F], chf_reg, ch_mod,
1, 8, 8, 1, 8, 8,
NULL, NULL, &ch_reset,
NULL, NULL, NULL,
NULL, DEV_DISABLE | DEV_DIS
},
{
"CHANG", &ch_unit[CH_G], chg_reg, ch_mod,
1, 8, 8, 1, 8, 8,
NULL, NULL, &ch_reset,
NULL, NULL, NULL,
NULL, DEV_DISABLE | DEV_DIS
},
{
"CHANH", &ch_unit[CH_H], chh_reg, ch_mod,
1, 8, 8, 1, 8, 8,
NULL, NULL, &ch_reset,
NULL, NULL, NULL,
NULL, DEV_DISABLE | DEV_DIS
}
};
/* 7607 channel overview
Channel variables:
ch_sta channel state
ch_dso, ch_dsu operation and unit for current data select
ch_ndso, ch_ndsu operation and unit for current non-data select
ch_clc current location counter
ch_ca memory addres
ch_wc word count
ch_op channel opcode (bits <S,1:2,19>)
ch_flags channel flags
States of a channel
IDLE - channel is not in operation
RDS, WDS: -> DSW if device is idle, schedule device
device timeout drives next transition
-> stall if device is busy
repeat until device is idle
other I/O: -> NDS if device is idle, schedule device
device timeout drives next transition
-> stall if device is busy
repeat until device is idle
chan reset: -> IDLE
PDS (PNDS) - channel is polling device to start data (non-data) select
chan timeout: -> DSW (NDS) if device is idle
device timeout drives next transition
-> no change if device is busy, schedule channel
chan reset: -> IDLE
DSW - channel is waiting for channel start command
dev timeout: -> IDLE if no stacked non-data select
-> PNDS if stacked non-data select
channel timeout drives next transition
start chan: -> DSX if chan program transfers data
device timeout drives next transition
-> IDLE if channel disconnects, no stacked NDS
-> PNDS if channel disconnects, stacked NDS
channel timeout drives next transition
chan reset: -> IDLE
DSX - channel is executing data select
dev timeout: -> DSX if transfer not complete, reschedule device
device timeout drives next transition
-> DSW if channel command completes, CHF_LDW set
-> IDLE if transfer complete, no stacked NDS, or
if channel command completes, CHF_LDW clear
-> PNDS if channel disconnects, stacked NDS
channel timeout drives next transition
start chan: -> DSX with CHF_LDW, CPU stall
chan reset: -> IDLE
NDS - channel is executing non-data select
dev timeout: -> IDLE if transfer complete, no stacked DS
-> PDS if channel disconnects, stacked DS
channel timeout drives next transition
chan reset: -> IDLE
The channel has two interfaces to a device. The select routine:
dev_select (uint32 ch, uint32 sel, uint32 unit)
Returns can include device errors and ERR_STALL. If ERR_STALL, the
device is busy. For I/O instructions, ERR_STALL stalls execution of
the instruction until the device is not busy. For stacked command
polls, ERR_STALL causes the poll to be repeated after a delay.
The device write routine is used to place output data in the device
write buffer.
Channel transfers are driven by the channel. When a device needs to
read or write data, it sets a channel request in ch_req. The channel
process transfers the data and updates channel control parameters
accordingly. Note that the channel may disconnect; in this case, the
transfer completes 'correctly' from the point of view of the device.
The channel transfer commands (IOxT) require the channel to 'hold'
a new channel command in anticipation of the current transfer. If
the channel is currently executing (CH6S_DSX) and a channel start
is issued by the CPU, a 'start pending' flag is set and the CPU is
stalled. When the channel reaches the end of an IOxT command, it
checks the 'start pending' flag. If the flag is set, the channel
sets itself to waiting and then requeues itself for one cycle later.
The CPU tries the channel start, sees that the channel is waiting,
and issues the new channel command.
state op device channel
IDLE RDS,WDS start I/O ->DSW
DSW LCHx (timed wait) ->DSX
DSX -- timeout, req svc
(timed wait) transfer word
timeout, req svc
(timed wait)
LCHx, stalls :
timeout, EOR/EOC IOxT: ->DSW, resched
DSW LCHx (timed wait) ->DSX, etc
7909 channel overview
Channel variables:
ch_sta channel state
ch_clc current location counter
ch_ca memory addres
ch_wc word count
ch_op channel opcode (bits <S,1:3,19>)
ch_sms status mask
ch_cond interrupt conditions
ch_lcc control counter
ch_flags channel flags
States of a channel
IDLE - channel is not in operation
RDCx, SDCx, interrupt -> DSX
DSX - channel is executing data select
TWT, WTR -> IDLE
The 7909 is more capable than the 7607 but also simpler in some ways.
It has many more instructions, built in counters and status checking,
and interrupts. But it has only two states and no concept of records.
The 7909 read process is driven by the device:
channel CTLR/SNS: send select
device: schedule timeout
device timeout: device to AR, request channel
channel: AR to memory
device timeout: device to AR, request channel
channel: AR to memory
:
device timeout: set end, request channel
channel: disconnect on CPYD, send STOP
The 7909 write process is also driven by the device:
channel CTL/CTLW: send select
device: schedule timeout, request channel
channel: memory to output buffer
device timeout: output buffer to device, request channel
channel: memory to output buffer
device timeout: output buffer to device, request channel
:
channel: memory to output buffer
device timeout: output buffer to device, set end, request channel
channel: disconnect on CPYD, send STOP
For both reads and writes, devices must implement an 'interblock' or
'interrecord' state that is long enough for the channel to see the
end, disconnect, and send a stop signal.
*/
/* Data select - called by RDS or WDS instructions - 7607/7289 only
- Channel is from address and has been corrected
- Channel must be an enabled 7607
- If data select already in use, stall CPU
- If non-data select is a write end-of-file, stall CPU
- If channel is busy, stack command
- Otherwise, start IO, set channel to waiting */
t_stat ch_op_ds (uint32 ch, uint32 ds, uint32 unit)
{
t_stat r;
if (ch >= NUM_CHAN) return STOP_NXCHN; /* invalid arg? */
if (ch_dev[ch].flags & DEV_DIS) return STOP_NXCHN; /* disabled? stop */
if (ch_dev[ch].flags & DEV_7909) return STOP_7909; /* 7909? stop */
if (ch_dso[ch]) return ERR_STALL; /* DS in use? */
if (ch_ndso[ch] == CHSL_WEF) return ERR_STALL; /* NDS = WEF? */
if (ch_sta[ch] == CHXS_IDLE) { /* chan idle? */
r = ch6_sel (ch, ds, unit, CH6S_DSW); /* select device */
if (r != SCPE_OK) return r;
}
ch_dso[ch] = ds; /* set command, unit */
ch_dsu[ch] = unit;
ch_flags[ch] &= ~(CHF_LDW|CHF_EOR|CHF_CMD); /* clear flags */
ch_idf[ch] = 0;
return SCPE_OK;
}
/* Non-data select - called by BSR, BSF, WEF, REW, RUN, SDS instructions - 7607 only
- Channel is from address and has been corrected
- Channel must be an enabled 7607
- If non-data select already in use, stall CPU
- If data select is card or printer, stall CPU
- If channel is busy, stack command
- Otherwise, start IO, set channel to waiting */
t_stat ch_op_nds (uint32 ch, uint32 nds, uint32 unit)
{
DEVICE *dptr;
t_stat r;
if (ch >= NUM_CHAN) return STOP_NXCHN; /* invalid arg? */
if (ch_dev[ch].flags & DEV_DIS) return STOP_NXCHN; /* disabled? stop */
if (ch_dev[ch].flags & DEV_7909) return STOP_7909; /* 7909? stop */
if (ch_ndso[ch]) return ERR_STALL; /* NDS in use? */
if (ch_dso[ch] && (dptr = ch_find_dev (ch, ch_dsu[ch])) /* DS, cd or lpt? */
&& (dptr->flags & DEV_CDLP)) return ERR_STALL;
if (ch_sta[ch] == CHXS_IDLE) { /* chan idle? */
r = ch6_sel (ch, nds, unit, CH6S_NDS); /* select device */
if (r != SCPE_OK) return r;
}
ch_ndso[ch] = nds; /* set command, unit */
ch_ndsu[ch] = unit;
return SCPE_OK;
}
/* End of data select - called from channel - 7607/7289 only
- If executing, set command trap flag
- Set channel idle
- If stacked nds, set up immediate channel timeout */
t_stat ch6_end_ds (uint32 ch)
{
if (ch >= NUM_CHAN) return STOP_NXCHN; /* invalid arg? */
ch_dso[ch] = ch_dsu[ch] = 0; /* no data select */
if (ch_ndso[ch]) { /* stacked non-data sel? */
sim_activate (ch_dev[ch].units, 0); /* immediate poll */
ch_sta[ch] = CH6S_PNDS; /* state = polling */
}
else ch_sta[ch] = CHXS_IDLE; /* else state = idle */
return SCPE_OK;
}
/* End of non-data select - called from I/O device completion - 7607/7289 only
- Set channel idle
- If stacked ds, set up immediate channel timeout */
t_stat ch6_end_nds (uint32 ch)
{
if (ch >= NUM_CHAN) return STOP_NXCHN; /* invalid arg? */
ch_ndso[ch] = ch_ndsu[ch] = 0; /* no non-data select */
if (ch_dso[ch]) { /* stacked data sel? */
sim_activate (ch_dev[ch].units, 0); /* immediate poll */
ch_sta[ch] = CH6S_PDS; /* state = polling */
}
else ch_sta[ch] = CHXS_IDLE; /* else state = idle */
return SCPE_OK;
}
/* Send select to device - 7607/7289 only */
t_stat ch6_sel (uint32 ch, uint32 sel, uint32 unit, uint32 sta)
{
DEVICE *dptr;
DIB *dibp;
t_stat r;
if (ch >= NUM_CHAN) return STOP_NXCHN; /* invalid arg? */
dptr = ch_find_dev (ch, unit); /* find device */
if (dptr == NULL) return STOP_NXDEV; /* invalid device? */
dibp = (DIB *) dptr->ctxt;
r = dibp->chsel (ch, sel, unit); /* select device */
if (r == SCPE_OK) ch_sta[ch] = sta; /* set status */
return r;
}
/* Channel unit service - called to start stacked command - 7607 only */
t_stat ch6_svc (UNIT *uptr)
{
uint32 ch = uptr - &ch_unit[0]; /* get channel */
t_stat r;
if (ch >= NUM_CHAN) return SCPE_IERR; /* invalid chan? */
switch (ch_sta[ch]) { /* case on state */
case CH6S_PDS: /* polling for ds */
r = ch6_sel (ch, ch_dso[ch], ch_dsu[ch], CH6S_DSW);
break;
case CH6S_PNDS: /* polling for nds */
r = ch6_sel (ch, ch_ndso[ch], ch_ndsu[ch], CH6S_NDS);
break;
default:
return SCPE_OK;
}
if (r == ERR_STALL) { /* stalled? */
sim_activate (uptr, ch_tpoll); /* continue poll */
return SCPE_OK;
}
return r;
}
/* Map channel and unit number to device - all channels */
DEVICE *ch_find_dev (uint32 ch, uint32 unit)
{
if (ch >= NUM_CHAN) return NULL; /* invalid arg? */
if (ch_dev[ch].flags & (DEV_7909|DEV_7289)) return ch2dev[ch];
unit = unit & 0777;
if (((unit >= U_MTBCD) && (unit <= (U_MTBCD + MT_NUMDR))) ||
((unit >= U_MTBIN) && (unit <= (U_MTBIN + MT_NUMDR))))
return ch2dev[ch];
if (ch != 0) return NULL;
if (unit == U_CDR) return &cdr_dev;
if (unit == U_CDP) return &cdp_dev;
if ((unit == U_LPBCD) || (unit == U_LPBIN)) return &lpt_dev;
return NULL;
}
/* Start channel - channel is from opcode
7607: channel should have a data select operation pending (DSW state)
7909: channel should be idle (IDLE state) */
t_stat ch_op_start (uint32 ch, uint32 clc, t_bool reset)
{
t_uint64 ir;
t_stat r;
clc = clc | data_base; /* add A/B select */
if (ch >= NUM_CHAN) return STOP_NXCHN; /* invalid argument? */
if (ch_dev[ch].flags & DEV_DIS) return STOP_NXCHN; /* disabled? stop */
if (ch_dev[ch].flags & DEV_7909) { /* 7909? */
if (ch_sta[ch] != CHXS_IDLE) return ERR_STALL; /* must be idle */
if (reset) { /* RDCx? */
ch_cnd[ch] = 0; /* clear conditions */
ch_clc[ch] = clc; /* set clc */
}
else { /* SDCx */
if (BIT_TST (chtr_enab, CHTR_V_TWT + ch) && /* pending trap? */
(ch_flags[ch] & CHF_TWT)) return ERR_STALL;
ch_clc[ch] = ch_ca[ch] & CHAMASK; /* finish WTR, TWT */
}
ch_flags[ch] &= ~CHF_CLR_7909; /* clear flags, not IP */
ch_idf[ch] = 0;
ch_sta[ch] = CHXS_DSX; /* set state */
return ch9_new_cmd (ch); /* start executing */
}
/* 7607, 7289 */
if (reset) { /* reset? */
if (ch_sta[ch] == CHXS_DSX) ch_sta[ch] = CH6S_DSW;
ch_flags[ch] &= ~(CHF_LDW|CHF_EOR|CHF_TRC|CHF_CMD);
ch_idf[ch] = 0;
}
switch (ch_sta[ch]) { /* case on chan state */
case CHXS_IDLE: /* idle */
ind_ioc = 1; /* IO check */
ir = ReadP (clc); /* get chan word */
ch_clc[ch] = CHAINC (clc); /* incr chan pc */
ch_wc[ch] = GET_DEC (ir); /* get word cnt */
ch_ca[ch] = ((uint32) ir) & CHAMASK; /* get address */
ch_op[ch] = (GET_OPD (ir) << 1) | /* get opcode */
((((uint32) ir) & CH6I_NST)? 1: 0); /* plus 'no store' */
break;
case CH6S_PNDS: /* NDS polling */
case CH6S_PDS: /* DS polling */
case CH6S_NDS: /* NDS executing */
return ERR_STALL; /* wait it out */
case CH6S_DSW: /* expecting command */
ch_sta[ch] = CHXS_DSX; /* update state */
if (ch_dev[ch].flags & DEV_7289) { /* drum channel? */
ir = ReadP (clc); /* read addr */
ch_clc[ch] = CHAINC (clc); /* incr chan pc */
if (r = ch9_wr (ch, ir, 0)) return r; /* write to dev */
}
else ch_clc[ch] = clc; /* set clc */
return ch6_new_cmd (ch, TRUE); /* start channel */
case CHXS_DSX: /* executing */
ch_flags[ch] = ch_flags[ch] | CHF_LDW; /* flag pending LCH */
return ERR_STALL; /* stall */
}
return SCPE_OK;
}
/* Store channel
7607/7289 stores op,ca,nostore,clc
7909 stores clc,,ca */
t_stat ch_op_store (uint32 ch, t_uint64 *dat)
{
if ((ch >= NUM_CHAN) || (ch_dev[ch].flags & DEV_DIS)) return STOP_NXCHN;
if (ch_dev[ch].flags & DEV_7909)
*dat = (((t_uint64) ch_ca[ch] & CHAMASK) << INST_V_DEC) |
(((t_uint64) ch_clc[ch] & CHAMASK) << INST_V_ADDR);
else *dat = (((t_uint64) ch_clc[ch] & CHAMASK) << INST_V_DEC) |
(((t_uint64) ch_ca[ch] & CHAMASK) << INST_V_ADDR) |
(((t_uint64) (ch_op[ch] & 1)) << 16) |
(((t_uint64) (ch_op[ch] & 016)) << 32);
return SCPE_OK;
}
/* Store channel diagnostic
7607 is undefined
7289 stores IOC+???
7909 stores 7909 lcc+flags */
t_stat ch_op_store_diag (uint32 ch, t_uint64 *dat)
{
if ((ch >= NUM_CHAN) || (ch_dev[ch].flags & DEV_DIS)) return STOP_NXCHN;
if (ch_flags[ch] & DEV_7289) *dat = ind_ioc? SIGN: 0;
else if (ch_flags[ch] & DEV_7909) *dat =
(((t_uint64) (ch_lcc[ch] & CHF_M_LCC)) << CHF_V_LCC) |
(ch_flags[ch] & CHF_SDC_7909);
else *dat = 0;
return SCPE_OK;
}
/* Reset data channel
7607 responds to RDC
7909 responds to RIC */
t_stat ch_op_reset (uint32 ch, t_bool ch7909)
{
DEVICE *dptr;
if (ch >= NUM_CHAN) return STOP_NXCHN; /* invalid argument? */
if (ch_dev[ch].flags & DEV_DIS) return SCPE_OK; /* disabled? ok */
if (ch_dev[ch].flags & DEV_7909) { /* 7909? */
if (!ch7909) return SCPE_OK; /* wrong reset is NOP */
dptr = ch2dev[ch]; /* get device */
}
else { /* 7607, 7289 */
if (ch7909) return STOP_NT7909; /* wrong reset is err */
dptr = ch_find_dev (ch, ch_ndsu[ch]); /* find device */
}
ch_reset (&ch_dev[ch]); /* reset channel */
if (dptr && dptr->reset) dptr->reset (dptr); /* reset device */
return SCPE_OK;
}
/* Channel process - called from main CPU loop. If the channel is unable
to get a valid command, it will reschedule itself for the next cycle.
The read process is basically synchronous with the device timeout routine.
The device requests the channel and supplies the word to be stored in memory.
In the next time slot, the channel stores the word in memory. */
t_stat ch_proc (uint32 ch)
{
t_stat r;
if (ch >= NUM_CHAN) return SCPE_IERR; /* bad channel? */
ch_req &= ~REQ_CH (ch); /* clear request */
if (ch_dev[ch].flags & DEV_DIS) return SCPE_IERR; /* disabled? */
if (ch_dev[ch].flags & DEV_7909) { /* 7909 */
t_uint64 sr;
uint32 csel, sc, tval, mask, ta;
t_bool xfr;
if (ch_flags[ch] & CHF_IRQ) { /* interrupt? */
ta = CHINT_CHA_SAV + (ch << 1); /* save location */
if (ch_sta[ch] == CHXS_IDLE) /* waiting? */
sr = (((t_uint64) ch_ca[ch] & CHAMASK) << INST_V_DEC) |
((t_uint64) ch_clc[ch] & CHAMASK); /* save CLC */
else sr = (((t_uint64) ch_ca[ch] & CHAMASK) << INST_V_DEC) |
((t_uint64) CHAINC (ch_clc[ch])); /* no, save CLC+1 */
ch_sta[ch] = CHXS_DSX; /* set running */
ch_flags[ch] = (ch_flags[ch] | CHF_INT) & /* set intr state */
~(CHF_IRQ|CHF_PRD|CHF_PWR|CHF_RDS|CHF_WRS); /* clr flags */
WriteP (ta, sr); /* write ca,,clc */
sr = ReadP (ta + 1); /* get chan cmd */
return ch9_exec_cmd (ch, sr); /* exec cmd */
}
switch (ch_op[ch] & CH9_OPMASK) { /* switch on op */
case CH9_TWT: /* transfer of TWT */
case CH9_WTR: /* transfer of WTR */
case CH9_TCH: /* transfer */
ch_clc[ch] = ch_ca[ch] & CHAMASK; /* change CLC */
break;
case CH9_TDC: /* decr & transfer */
if (ch_lcc[ch] != 0) { /* counter != 0? */
ch_lcc[ch]--; /* decr counter */
ch_clc[ch] = ch_ca[ch] & CHAMASK; /* change CLC */
}
break;
case CH9_TCM: /* transfer on cond */
csel = CH9D_COND (ch_wc[ch]);
mask = CH9D_MASK (ch_wc[ch]);
if (csel == 7) xfr = (mask == 0); /* C = 7? mask mbz */
else { /* C = 0..6 */
if (csel == 0) tval = ch_cnd[ch]; /* C = 0? test cond */
else tval = (uint32) (ch_ar[ch] >> (6 * (6 - csel))) & 077;
if (ch_wc[ch] & CH9D_B11)
xfr = ((tval & mask) == mask);
else xfr = (tval == mask);
}
if (xfr) ch_clc[ch] = ch_ca[ch] & CHAMASK; /* change CLC */
break;
case CH9_LIP: /* leave interrupt */
ta = CHINT_CHA_SAV + (ch << 1); /* save location */
ch_flags[ch] &= ~(CHF_INT|CHF_IRQ); /* clear intr */
ch_cnd[ch] = 0; /* clear channel cond */
ch_clc[ch] = (uint32) ReadP (ta) & CHAMASK;
break;
case CH9_LIPT: /* leave intr, transfer */
ch_flags[ch] &= ~(CHF_INT|CHF_IRQ); /* clear intr */
ch_cnd[ch] = 0; /* clear channel cond */
ch_clc[ch] = ch_ca[ch] & CHAMASK; /* change CLC */
break;
case CH9_LAR: /* load assembly reg */
ch_ar[ch] = ReadP (ch_ca[ch]);
break;
case CH9_SAR: /* store assembly reg */
WriteP (ch_ca[ch], ch_ar[ch]);
break;
case CH9_SMS: /* load SMS reg */
ch_sms[ch] = CH9A_SMS (ch_ca[ch]); /* from eff addr */
if (!(ch_sms[ch] & CHSMS_IATN1) && /* atn inhbit off */
(ch_flags[ch] & CHF_ATN1)) /* and atn pending? */
ch9_eval_int (ch, 0); /* force int eval */
break;
case CH9_LCC: /* load control cntr */
ch_lcc[ch] = CH9A_LCC (ch_ca[ch]); /* from eff addr */
break;
case CH9_ICC: /* insert control cntr */
case CH9_ICCA:
csel = CH9D_COND (ch_wc[ch]); /* get C */
if (csel == 0) ch_ar[ch] = /* C = 0? read SMS */
(ch_ar[ch] & 0777777770000) | ((t_uint64) ch_sms[ch]);
else if (csel < 7) { /* else read cond cntr */
sc = 6 * (6 - csel);
ch_ar[ch] = (ch_ar[ch] & ~(((t_uint64) 077) << sc)) |
(((t_uint64) ch_lcc[ch]) << sc);
}
break;
case CH9_XMT: /* transmit */
if (ch_wc[ch] == 0) break;
sr = ReadP (ch_clc[ch]); /* next word */
WriteP (ch_ca[ch], sr);
ch_clc[ch] = CHAINC (ch_clc[ch]); /* incr pointers */
ch_ca[ch] = CHAINC (ch_ca[ch]);
ch_wc[ch] = ch_wc[ch] - 1; /* decr count */
ch_req |= REQ_CH (ch); /* go again */
return SCPE_OK;
case CH9_SNS: /* sense */
if (r = ch9_sel (ch, CHSL_SNS)) return r; /* send sense to dev */
ch_flags[ch] |= CHF_PRD; /* prepare to read */
break; /* next command */
case CH9_CTL:
case CH9_CTLR:
case CH9_CTLW: /* control */
if (((ch_wc[ch] & CH9D_NST) == 0) && /* N = 0 and */
!(ch_flags[ch] & CHF_EOR)) { /* end not set? */
sr = ReadP (ch_ca[ch]);
ch_ca[ch] = CHAINC (ch_ca[ch]); /* incr ca */
return ch9_wr (ch, sr, 0); /* write ctrl wd */
}
ch_flags[ch] &= ~CHF_EOR; /* clear end */
if (ch_op[ch] == CH9_CTLR) { /* CTLR? */
if (r = ch9_sel (ch, CHSL_RDS)) return r; /* send read sel */
ch_flags[ch] |= CHF_PRD; /* prep to read */
ch_idf[ch] = 0;
}
else if (ch_op[ch] == CH9_CTLW) { /* CTLW? */
if (r = ch9_sel (ch, CHSL_WRS)) return r; /* end write sel */
ch_flags[ch] |= CHF_PWR; /* prep to write */
}
break;
case CH9_CPYD: /* copy & disc */
if ((ch_wc[ch] == 0) || (ch_flags[ch] & CHF_EOR)) { /* wc == 0 or EOR? */
if (ch_flags[ch] & (CHF_PRD|CHF_PWR|CHF_RDS|CHF_WRS)) {
ch_flags[ch] &= ~(CHF_PRD|CHF_PWR|CHF_RDS|CHF_WRS);
if (r = ch9_wr (ch, 0, CH9DF_STOP)) return r; /* send stop */
}
if (ch_flags[ch] & CHF_EOR) { /* EOR? */
ch_flags[ch] &= ~CHF_EOR; /* clear flag */
break; /* new command */
}
return SCPE_OK; /* wait for end */
}
if (ch_flags[ch] & CHF_RDS) /* read? */
return ch9_rd_putw (ch);
return ch9_wr_getw (ch); /* no, write */
case CH9_CPYP: /* anything to do? */
if (ch_wc[ch] == 0) break; /* (new, wc = 0) next */
if (ch_flags[ch] & CHF_EOR) /* end? */
ch_flags[ch] &= ~CHF_EOR; /* ignore */
else if (ch_flags[ch] & CHF_RDS) /* read? */
ch9_rd_putw (ch);
else if (r = ch9_wr_getw (ch)) return r; /* no, write */
if (ch_wc[ch] == 0) break; /* done? get next */
return SCPE_OK; /* more to do */
default:
return STOP_ILLIOP;
}
return ch9_new_cmd (ch); /* next command */
}
else if (ch_flags[ch] & CHF_RDS) { /* 7607 read? */
if (ch_sta[ch] != CHXS_DSX) return ch6_end_ds (ch); /* chan exec? no, disc */
switch (ch_op[ch] & CH6_OPMASK) { /* switch on op */
case CH6_TCH: /* transfer */
ch_clc[ch] = ch_ca[ch] & CHAMASK; /* change clc */
return ch6_new_cmd (ch, FALSE); /* unpack new cmd */
case CH6_IOCD: /* IOCD */
if (ch_wc[ch]) { /* wc > 0? */
if (ch6_rd_putw (ch)) return SCPE_OK; /* store; more? cont */
}
return ch6_end_ds (ch); /* no, disconnect */
case CH6_IOCP: /* IOCP */
if (ch_wc[ch]) { /* wc > 0? */
if (ch6_rd_putw (ch)) return SCPE_OK; /* store; more? cont */
}
return ch6_new_cmd (ch, FALSE); /* unpack new cmd */
case CH6_IOCT: /* IOCT */
if (ch_wc[ch]) { /* wc > 0? */
if (ch6_rd_putw (ch)) return SCPE_OK; /* store; more? cont */
}
return ch6_ioxt (ch); /* unstall or disc */
case CH6_IOSP: /* IOSP */
if (ch_flags[ch] & CHF_EOR) { /* (new) EOR set? */
ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* clear flag */
return ch6_new_cmd (ch, FALSE); /* get next cmd */
}
if (ch_wc[ch]) { /* wc > 0? */
if (ch6_rd_putw (ch) && !(ch_flags[ch] & CHF_EOR))
return SCPE_OK; /* yes, store; more? */
ch6_iosp_cclr (ch); /* cond clear eor */
}
return ch6_new_cmd (ch, FALSE); /* next cmd */
case CH6_IOST: /* IOST */
if (ch_flags[ch] & CHF_EOR) { /* (new) EOR set? */
ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* clear flag */
return ch6_ioxt (ch); /* get next cmd */
}
if (ch_wc[ch]) { /* wc > 0? */
if (ch6_rd_putw (ch) && !(ch_flags[ch] & CHF_EOR))
return SCPE_OK; /* yes, store; more? */
ch6_iosp_cclr (ch); /* cond clear eor */
}
return ch6_ioxt (ch); /* unstall or disc */
case CH6_IORP: /* IORP */
if (ch_flags[ch] & CHF_EOR) { /* (new) EOR set? */
ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* clear flag */
return ch6_new_cmd (ch, FALSE); /* get next cmd */
}
ch6_rd_putw (ch); /* store wd; ignore wc */
if (ch_flags[ch] & CHF_EOR) { /* EOR? */
ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* clear flag */
return ch6_new_cmd (ch, FALSE); /* get next cmd */
}
return SCPE_OK; /* done */
case CH6_IORT: /* IORT */
if (ch_flags[ch] & CHF_EOR) { /* (new) EOR set? */
ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* clear flag */
return ch6_ioxt (ch); /* get next cmd */
}
ch6_rd_putw (ch); /* store wd; ignore wc */
if (ch_flags[ch] & CHF_EOR) { /* EOR? */
ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* clear flag */
return ch6_ioxt (ch); /* unstall or disc */
}
return SCPE_OK; /* done */
default:
return SCPE_IERR;
} /* end case */
} /* end if read */
else { /* 7607 write */
if (ch_sta[ch] != CHXS_DSX) return ch6_end_ds (ch); /* chan exec? no, disc */
switch (ch_op[ch] & CH6_OPMASK) { /* switch on op */
case CH6_TCH: /* transfer */
ch_clc[ch] = ch_ca[ch] & CHAMASK; /* change clc */
return ch6_new_cmd (ch, FALSE); /* unpack new cmd */
case CH6_IOCD: /* IOCD */
if (ch_wc[ch]) { /* wc > 0? */
if (r = ch6_wr_getw (ch, TRUE)) return r; /* send wd to dev; err? */
if (ch_wc[ch]) return SCPE_OK; /* more to do? */
}
return ch6_end_ds (ch); /* disconnect */
case CH6_IOCP: /* IOCP */
case CH6_IOSP: /* IOSP */
if (ch_wc[ch]) { /* wc > 0? */
if (r = ch6_wr_getw (ch, FALSE)) return r; /* send wd to dev; err? */
if (ch_wc[ch]) return SCPE_OK; /* more to do? */
}
return ch6_new_cmd (ch, FALSE); /* get next cmd */
case CH6_IOCT: /* IOCT */
case CH6_IOST: /* IOST */
if (ch_wc[ch]) { /* wc > 0? */
if (r = ch6_wr_getw (ch, FALSE)) return r; /* send wd to dev; err? */
if (ch_wc[ch]) return SCPE_OK; /* more to do? */
}
return ch6_ioxt (ch); /* get next cmd */
case CH6_IORP: /* IORP */
if (!(ch_flags[ch] & CHF_EOR) && ch_wc[ch]) { /* not EOR? (cdp, lpt) */
if (r = ch6_wr_getw (ch, TRUE)) return r; /* send wd to dev; err? */
if (ch_wc[ch]) return SCPE_OK; /* more to do? */
}
ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* clear EOR */
return ch6_new_cmd (ch, FALSE); /* get next cmd */
case CH6_IORT: /* IORT */
if (!(ch_flags[ch] & CHF_EOR) && ch_wc[ch]) { /* not EOR? (cdp, lpt) */
if (r = ch6_wr_getw (ch, TRUE)) return r; /* send wd to dev; err? */
if (ch_wc[ch]) return SCPE_OK; /* more to do? */
}
ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* clear EOR */
return ch6_ioxt (ch); /* unstall or disc */
default:
return SCPE_IERR;
} /* end switch */
} /* end else write */
}
/* 7607 channel support routines */
/* 7607 channel input routine - put one word to memory */
t_bool ch6_rd_putw (uint32 ch)
{
if (ch_idf[ch] & CH6DF_EOR) ch_flags[ch] |= CHF_EOR; /* eor from dev? */
else ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* set/clr chan eor */
ch_idf[ch] = 0; /* clear eor, valid */
if (ch_wc[ch]) { /* wc > 0? */
if ((ch_op[ch] & 1) == 0) { /* do store? */
WriteP (ch_ca[ch], ch_ar[ch]);
ch_ca[ch] = CHAINC (ch_ca[ch]); /* incr ca */
}
ch_wc[ch] = ch_wc[ch] - 1;
}
return (ch_wc[ch]? TRUE: FALSE);
}
/* 7607 channel output routine - get one word from memory */
t_stat ch6_wr_getw (uint32 ch, t_bool eorz)
{
DEVICE *dptr;
DIB *dibp;
uint32 eorfl;
ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* clr eor */
if (ch_wc[ch]) {
ch_ar[ch] = ReadP (ch_ca[ch]); /* get word */
ch_ca[ch] = CHAINC (ch_ca[ch]); /* incr ca */
ch_wc[ch] = ch_wc[ch] - 1;
}
else ch_ar[ch] = 0;
if (eorz && (ch_wc[ch] == 0)) eorfl = 1; /* eor on wc = 0? */
else eorfl = 0;
dptr = ch_find_dev (ch, ch_dsu[ch]); /* find device */
if (dptr && /* valid device? */
(dibp = (DIB *) dptr->ctxt) && /* with DIB? */
dibp->write) /* and write routine? */
return dibp->write (ch, ch_ar[ch], eorfl);
return SCPE_IERR; /* huh? */
}
/* 7607 channel new command - on channel load, check for disconnects
The protocol for new commands is as follows:
- If IOCD 0,,0, disconnect immediately
- If IOCT 0,,0 or IOST 0,,0 and loaded by RCHA, disconnect immediately
- If an effective NOP (TCH, IOCx 0,,0, IOSx 0,,0), force a channel
cycle to retire the channel comand as quickly as possible.
- If an IORx and EOR is set, force a channel cycle to retire the
channel command as quickly as possible.
*/
t_stat ch6_new_cmd (uint32 ch, t_bool ch_ld)
{
t_uint64 ir;
uint32 op, t;
ir = ReadP (t = ch_clc[ch]); /* read cmd */
ch_wc[ch] = GET_DEC (ir); /* get word cnt */
ch_ca[ch] = ((uint32) ir) & CHAMASK; /* get address */
op = GET_OPD (ir) << 1; /* get opcode */
ch_op[ch] = op | ((((uint32) ir) & CH6I_NST)? 1: 0); /* plus 'no store' */
if ((ir & CHI_IND) && (ch_wc[ch] || /* indirect? */
((op != CH6_IOCP) && (op != CH6_IOSP)))) { /* wc >0, or !IOxP? */
t_uint64 sr = ReadP (ch_ca[ch] & AMASK); /* read indirect */
ch_ca[ch] = ((uint32) sr) & ((cpu_model & I_CT)? PAMASK: AMASK);
}
if (hst_ch) cpu_ent_hist (ch_clc[ch] | ((ch + 1) << HIST_V_CH), ch_ca[ch], ir, 0);
ch_clc[ch] = (ch_clc[ch] + 1) & AMASK; /* incr chan pc */
switch (op) { /* case on opcode */
case CH6_IOCD: /* IOCD */
if (ch_wc[ch] == 0) ch6_end_ds (ch); /* wc 0? end now */
break;
case CH6_IOST: /* IOST */
if (ch_flags[ch] & CHF_EOR) /* EOR set? immed ch req */
ch_req |= REQ_CH (ch);
case CH6_IOCT: /* IOCT */
if (ch_wc[ch] == 0) { /* wc 0? */
if (ch_ld) ch6_end_ds (ch); /* load? end now */
else ch_req |= REQ_CH (ch); /* else immed ch req */
}
break;
case CH6_IOSP: /* IOSP */
if (ch_flags[ch] & CHF_EOR) /* EOR set? immed ch req */
ch_req |= REQ_CH (ch);
case CH6_IOCP: /* IOCP */
if (ch_wc[ch] == 0) ch_req |= REQ_CH (ch); /* wc 0? immed ch req */
break;
case CH6_IORT: /* IORT */
case CH6_IORP: /* IORP */
if (ch_flags[ch] & CHF_EOR) /* EOR set? immed ch req */
ch_req |= REQ_CH (ch);
break;
case CH6_TCH: /* TCH */
ch_req |= REQ_CH (ch); /* immed ch req */
break;
default: /* all others */
break;
} /* end case */
if (sim_brk_summ && sim_brk_test (t, SWMASK ('E')))
return ch_bkpt (ch, t);
return SCPE_OK;
}
/* 7607 channel IOxT: if LCH stall, set state back to DSW; else disconnect and trap */
t_stat ch6_ioxt (uint32 ch)
{
if (ch_flags[ch] & CHF_LDW) { /* LCH cmd pending? */
ch_flags[ch] &= ~CHF_LDW; /* clr pending flag */
ch_sta[ch] = CH6S_DSW; /* unstall pending LCH */
}
else {
ch_flags[ch] |= CHF_CMD; /* set cmd trap flag */
ch6_end_ds (ch); /* disconnect */
}
return SCPE_OK;
}
/* 7607 conditionally clear EOR on IOSx completion */
void ch6_iosp_cclr (uint32 ch)
{
uint32 i, op;
if (ch_wc[ch] == 0) { /* wc = 0? */
uint32 ccnt = 5; /* allow 5 for CPU */
for (i = 0; i < NUM_CHAN; i++) { /* test channels */
if (ch_sta[ch] != CHXS_DSX) continue; /* idle? skip */
op = ch_op[ch] & ~1; /* get op */
ccnt++; /* 1 per active ch */
if ((op == CH6_IOCP) || (op == CH6_IORP) || /* 1 per proceed */
(op == CH6_IOSP)) ccnt++;
}
if (ccnt <= 11) return; /* <= 11? ok */
}
ch_flags[ch] = ch_flags[ch] & ~CHF_EOR; /* clear eor */
return;
}
/* 7607 external interface routines */
/* Input - store word, request channel input service */
t_stat ch6_req_rd (uint32 ch, uint32 unit, t_uint64 val, uint32 fl)
{
if (ch6_qconn (ch, unit)) { /* ch conn to caller? */
if (ch_idf[ch] & CH6DF_VLD) ind_ioc = 1; /* overrun? */
ch_idf[ch] = CH6DF_VLD; /* set ar valid */
if (fl) ch_idf[ch] |= CH6DF_EOR; /* set eor if requested */
ch_req |= REQ_CH (ch); /* request chan */
ch_flags[ch] |= CHF_RDS;
ch_ar[ch] = val & DMASK; /* save data */
}
return SCPE_OK;
}
/* Disconnect on error */
t_stat ch6_err_disc (uint32 ch, uint32 unit, uint32 fl)
{
if (ch6_qconn (ch, unit)) { /* ch conn to caller? */
ch_flags[ch] |= fl; /* set flag */
return ch6_end_ds (ch); /* disconnect */
}
return SCPE_OK;
}
/* Output - request channel output service */
t_bool ch6_req_wr (uint32 ch, uint32 unit)
{
if (ch6_qconn (ch, unit)) { /* ch conn to caller? */
ch_req |= REQ_CH (ch);
ch_flags[ch] &= ~CHF_RDS;
}
return SCPE_OK;
}
/* Set/read channel flags */
uint32 ch6_set_flags (uint32 ch, uint32 unit, uint32 flags)
{
if (ch6_qconn (ch, unit)) { /* ch conn to caller? */
ch_flags[ch] = ch_flags[ch] | flags;
return ch_flags[ch];
}
return 0;
}
/* Channel connected to unit? */
t_bool ch6_qconn (uint32 ch, uint32 unit)
{
if ((ch < NUM_CHAN) && /* valid chan */
(ch_dsu[ch] == unit)) return TRUE; /* for right unit? */
return FALSE;
}
/* 7909 channel support routines */
/* 7909 channel input routine - put one word to memory */
t_stat ch9_rd_putw (uint32 ch)
{
ch_idf[ch] = 0; /* invalidate */
if (ch_wc[ch]) { /* wc > 0? */
WriteP (ch_ca[ch], ch_ar[ch]);
ch_ca[ch] = CHAINC (ch_ca[ch]);
ch_wc[ch] = ch_wc[ch] - 1;
}
return SCPE_OK;
}
/* 7909 channel output routine - get one word from memory */
t_stat ch9_wr_getw (uint32 ch)
{
if (ch_wc[ch]) {
ch_ar[ch] = ReadP (ch_ca[ch]); /* get word */
ch_ca[ch] = CHAINC (ch_ca[ch]);
ch_wc[ch] = ch_wc[ch] - 1;
}
else ch_ar[ch] = 0;
return ch9_wr (ch, ch_ar[ch], 0); /* write to device */
}
/* 7909 send select to device */
t_stat ch9_sel (uint32 ch, uint32 sel)
{
DEVICE *dptr = ch2dev[ch];
DIB *dibp;
if (dptr == NULL) return SCPE_IERR;
dibp = (DIB *) dptr->ctxt;
if (dibp && dibp->chsel) return dibp->chsel (ch, sel, 0);
return SCPE_IERR;
}
/* 7909 send word to device */
t_stat ch9_wr (uint32 ch, t_uint64 dat, uint32 fl)
{
DEVICE *dptr = ch2dev[ch];
DIB *dibp;
if (dptr == NULL) return SCPE_IERR;
dibp = (DIB *) dptr->ctxt;
if (dibp && dibp->write) return dibp->write (ch, dat, fl);
return SCPE_IERR;
}
/* 7909 channel new command */
t_stat ch9_new_cmd (uint32 ch)
{
t_uint64 ir;
uint32 t;
t_stat r;
ir = ReadP (t = ch_clc[ch]); /* read cmd */
r = ch9_exec_cmd (ch, ir); /* exec cmd */
if (ch_sta[ch] != CHXS_IDLE) /* chan running? */
ch_clc[ch] = CHAINC (ch_clc[ch]); /* incr chan pc */
if ((r == SCPE_OK) && sim_brk_summ && sim_brk_test (t, SWMASK ('E')))
return ch_bkpt (ch, t);
return r;
}
t_stat ch9_exec_cmd (uint32 ch, t_uint64 ir)
{
uint32 op;
ch_wc[ch] = GET_DEC (ir); /* get word cnt */
ch_ca[ch] = ((uint32) ir) & CHAMASK; /* get address */
op = (GET_OPD (ir) << 2); /* get opcode */
ch_op[ch] = op | ((((uint32) ir) & 0200000)? 1: 0) | /* plus bit<19> */
(((op & 010) && (ch_wc[ch] & 040000))? 2: 0); /* plus bit 3 if used */
if (ir & CHI_IND) { /* indirect? */
t_uint64 sr = ReadP (ch_ca[ch] & CHAMASK); /* read indirect */
ch_ca[ch] = ((uint32) sr) & CHAMASK; /* get address */
}
if (hst_ch)
cpu_ent_hist (ch_clc[ch] | ((ch + 1) << HIST_V_CH), ch_ca[ch], ir, 0);
switch (ch_op[ch]) { /* check initial cond */
case CH9_LAR: /* misc processing */
case CH9_SAR:
case CH9_ICC:
case CH9_ICCA:
case CH9_XMT:
case CH9_LCC:
case CH9_SMS:
if (ch_flags[ch] & (CHF_PRD|CHF_PWR|CHF_RDS|CHF_WRS))
ch9_eval_int (ch, CHINT_SEQC); /* not during data */
/* fall through */
case CH9_TCM: /* jumps */
case CH9_TCH:
case CH9_TDC:
case CH9_LIPT:
case CH9_LIP:
ch_req |= REQ_CH (ch); /* process in chan */
break;
case CH9_CTL: /* control */
case CH9_CTLR:
case CH9_CTLW:
if (ch_flags[ch] & (CHF_PRD|CHF_PWR|CHF_RDS|CHF_WRS))
ch9_eval_int (ch, CHINT_SEQC); /* not during data */
ch_flags[ch] &= ~CHF_EOR;
if (ch_wc[ch] & CH9D_NST) ch_req |= REQ_CH (ch); /* N set? proc in chan */
else return ch9_sel (ch, CHSL_CTL); /* sel, dev sets ch_req! */
break;
case CH9_SNS: /* sense */
if (ch_flags[ch] & (CHF_PRD|CHF_PWR|CHF_RDS|CHF_WRS))
ch9_eval_int (ch, CHINT_SEQC);
ch_flags[ch] &= ~CHF_EOR;
ch_req |= REQ_CH (ch); /* process in chan */
break;
case CH9_CPYD: /* data transfers */
case CH9_CPYP:
if ((ch_flags[ch] & (CHF_PRD|CHF_PWR|CHF_RDS|CHF_WRS)) == 0)
ch9_eval_int (ch, CHINT_SEQC); /* not unless data */
if (ch_flags[ch] & CHF_PRD) ch_flags[ch] |= CHF_RDS;
else if (ch_flags[ch] & CHF_PWR) ch_flags[ch] |= CHF_WRS;
ch_flags[ch] &= ~(CHF_EOR|CHF_PRD|CHF_PWR);
if ((ch_op[ch] == CH9_CPYP) && (ch_wc[ch] == 0))
ch_req |= REQ_CH (ch); /* CPYP x,,0? */
break; /* dev sets ch_req! */
case CH9_WTR: /* wait */
ch_sta[ch] = CHXS_IDLE; /* stop */
break;
case CH9_TWT: /* trap and wait */
ch_sta[ch] = CHXS_IDLE; /* stop */
ch_flags[ch] |= CHF_TWT; /* set trap */
break;
default:
return STOP_ILLIOP;
}
return SCPE_OK;
}
/* 7909 external interface routines */
/* Input - store word, request channel input service */
t_stat ch9_req_rd (uint32 ch, t_uint64 val)
{
if (ch < NUM_CHAN) { /* valid chan? */
if (ch_idf[ch] & CH9DF_VLD) ch9_set_ioc (ch); /* prev still valid? io chk */
ch_idf[ch] = CH9DF_VLD; /* set ar valid */
ch_req |= REQ_CH (ch); /* request chan */
ch_ar[ch] = val & DMASK; /* save data */
}
return SCPE_OK;
}
/* Set attention */
void ch9_set_atn (uint32 ch)
{
if (ch < NUM_CHAN) ch9_eval_int (ch, CHINT_ATN1);
return;
}
/* Set IO check - UEND will occur at end - not recognized in int mode */
void ch9_set_ioc (uint32 ch)
{
if ((ch < NUM_CHAN) && !(ch_flags[ch] & CHF_INT)) {
ind_ioc = 1; /* IO check */
ch_flags[ch] |= CHF_IOC; /* ch IOC for end */
}
return;
}
/* Set end */
void ch9_set_end (uint32 ch, uint32 iflags)
{
if (ch < NUM_CHAN) { /* valid chan? */
ch_flags[ch] |= CHF_EOR;
ch9_eval_int (ch, iflags);
}
return;
}
/* Test connected */
t_bool ch9_qconn (uint32 ch)
{
if ((ch < NUM_CHAN) && (ch_sta[ch] == CHXS_DSX)) return TRUE;
return FALSE;
}
/* Evaluate interrupts
- Interrupt requests set flags in the channel flags word
- If an interrupt is not in progress, interrupt requests are evaluated
- If an interrupt request is found, the interruptable flags are
transferred to the channel condition register and cleared in
the channel flags
This provides an effective stage of buffering for interrupt requests
that are not immediately serviced */
void ch9_eval_int (uint32 ch, uint32 iflags)
{
uint32 ireq;
ch_flags[ch] |= (iflags << CHF_V_COND); /* or into chan flags */
if ((ch_flags[ch] & CHF_INT) == 0) { /* int not in prog? */
ireq = ((ch_flags[ch] >> CHF_V_COND) & CHF_M_COND) &
~(((ch_sms[ch] & CHSMS_IUEND)? CHINT_UEND: 0) |
((ch_sms[ch] & CHSMS_IATN1)? CHINT_ATN1: 0) |
((ch_sms[ch] & CHSMS_IATN2)? CHINT_ATN2: 0) |
((ch_flags[ch] & (CHF_PRD|CHF_PWR|CHF_RDS|CHF_WRS))? CHINT_SEQC: 0));
if (ireq) { /* int pending? */
ch_cnd[ch] = ireq; /* set cond reg */
ch_flags[ch] &= ~(ireq << CHF_V_COND); /* clear chan flags */
ch_flags[ch] |= CHF_IRQ; /* set int req */
ch_req |= REQ_CH (ch); /* request channel */
}
}
return;
}
/* Test for all channels idle */
t_bool ch_qidle (void)
{
uint32 i;
for (i = 0; i < NUM_CHAN; i++) {
if (ch_sta[i] != CHXS_IDLE) return FALSE;
}
return TRUE;
}
/* Evaluate/execute channel traps */
uint32 chtr_eval (uint32 *decr)
{
uint32 i, cme;
if (!chtr_inht && !chtr_inhi && chtr_enab) {
if (BIT_TST (chtr_enab, CHTR_V_CLK) && chtr_clk) { /* clock trap? */
if (decr) { /* exec? */
chtr_clk = 0; /* clr flag */
*decr = 0;
}
return CHTR_CLK_SAV;
}
for (i = 0; i < NUM_CHAN; i++) { /* loop thru chan */
cme = BIT_TST (chtr_enab, CHTR_V_CME + i); /* cmd/eof enab? */
if (cme && (ch_flags[i] & CHF_CMD)) { /* cmd enab and set? */
if (decr) { /* exec? */
ch_flags[i] &= ~CHF_CMD; /* clr flag */
*decr = CHTR_F_CMD;
}
return (CHTR_CHA_SAV + (i << 1));
}
if (cme && (ch_flags[i] & CHF_EOF)) { /* eof enab and set? */
if (decr) { /* exec? */
ch_flags[i] &= ~CHF_EOF; /* clr flag */
*decr = CHTR_F_EOF;
}
return (CHTR_CHA_SAV + (i << 1));
}
if (BIT_TST (chtr_enab, CHTR_V_TRC + i) && /* trc enab? */
(ch_flags[i] & CHF_TRC)) { /* trc flag? */
if (decr) { /* exec? */
ch_flags[i] &= ~CHF_TRC; /* clr flag */
*decr = CHTR_F_TRC;
}
return (CHTR_CHA_SAV + (i << 1));
} /* end if BIT_TST */
} /* end for */
} /* end if !chtr_inht */
if (decr) *decr = 0;
return 0;
}
/* Channel reset */
t_stat ch_reset (DEVICE *dptr)
{
uint32 ch = dptr - &ch_dev[0]; /* get channel */
if (ch == CH_A) ch2dev[ch] = &mt_dev[0]; /* channel A fixed */
ch_sta[ch] = 0;
ch_flags[ch] = 0;
ch_idf[ch] = 0;
ch_dso[ch] = 0;
ch_dsu[ch] = 0;
ch_ndso[ch] = 0;
ch_ndsu[ch] = 0;
ch_op[ch] = 0;
ch_clc[ch] = 0;
ch_wc[ch] = 0;
ch_ca[ch] = 0;
ch_ar[ch] = 0;
ch_sms[ch] = 0;
ch_cnd[ch] = 0;
ch_lcc[ch] = 0;
sim_cancel (&ch_unit[ch]);
return SCPE_OK;
}
/* Show channel type */
t_stat ch_show_type (FILE *st, UNIT *uptr, int32 val, void *desc)
{
DEVICE *dptr;
dptr = find_dev_from_unit (uptr);
if (dptr == NULL) return SCPE_IERR;
if (dptr->flags & DEV_7909) fputs ("7909", st);
else if (dptr->flags & DEV_7289) fputs ("7289", st);
else fputs ("7607", st);
return SCPE_OK;
}
/* Enable channel, assign device */
t_stat ch_set_enable (UNIT *uptr, int32 val, char *cptr, void *desc)
{
DEVICE *dptr, *dptr1;
char gbuf[CBUFSIZE];
uint32 i, ch;
dptr = find_dev_from_unit (uptr);
if (dptr == NULL) return SCPE_IERR;
ch = dptr - &ch_dev[0];
if ((ch == 0) || !(dptr->flags & DEV_DIS)) return SCPE_ARG;
if (cptr == NULL) cptr = "TAPE";
get_glyph (cptr, gbuf, 0);
for (i = 0; dev_table[i].name; i++) {
if (strcmp (dev_table[i].name, gbuf) == 0) {
dptr1 = ch_map_flags (ch, dev_table[i].flags);
if (!dptr1 || !(dptr1->flags & DEV_DIS)) return SCPE_ARG;
dptr->flags &= ~(DEV_DIS|DEV_7909|DEV_7289|DEV_7750|DEV_7631);
dptr->flags |= dev_table[i].flags;
dptr1->flags &= ~DEV_DIS;
ch2dev[ch] = dptr1;
return reset_all (0);
}
}
return SCPE_ARG;
}
/* Map device flags to device pointer */
DEVICE *ch_map_flags (uint32 ch, int32 fl)
{
if (fl & DEV_7289) return &drm_dev;
if (!(fl & DEV_7909)) return &mt_dev[ch];
if (fl & DEV_7631) return &dsk_dev;
if (fl & DEV_7750) return &com_dev;
return NULL;
}
/* Set up channel map */
void ch_set_map (void)
{
uint32 i;
for (i = 0; i < NUM_CHAN; i++) {
if (ch_dev[i].flags & DEV_DIS) ch2dev[i] = NULL;
else ch2dev[i] = ch_map_flags (i, ch_dev[i].flags);
}
return;
}
/* Disable channel, deassign device */
t_stat ch_set_disable (UNIT *uptr, int32 val, char *cptr, void *desc)
{
DEVICE *dptr, *dptr1;
UNIT *uptr1;
uint32 i, ch;
dptr = find_dev_from_unit (uptr);
if (dptr == NULL) return SCPE_IERR;
ch = dptr - &ch_dev[0];
if ((ch == 0) || (dptr->flags & DEV_DIS) || (cptr != NULL)) return SCPE_ARG;
dptr1 = ch2dev[ch];
if (dptr1 == NULL) return SCPE_IERR;
if (dptr1->units) {
for (i = 0; i < dptr1->numunits; i++) {
uptr1 = dptr1->units + i;
if (dptr1->detach) dptr1->detach (uptr1);
else detach_unit (uptr1);
}
}
dptr->flags &= ~(DEV_7909|DEV_7289);
dptr->flags |= DEV_DIS;
dptr1->flags |= DEV_DIS;
return reset_all (0);
}
/* Show channel that device is on (tapes, 7289, 7909 only) */
t_stat ch_show_chan (FILE *st, UNIT *uptr, int32 val, void *desc)
{
DEVICE *dptr;
uint32 i;
dptr = find_dev_from_unit (uptr);
if (dptr) {
for (i = 0; i < NUM_CHAN; i++) {
if (ch2dev[i] == dptr) {
fprintf (st, "channel %c", 'A' + i);
return SCPE_OK;
}
}
}
fprintf (st, "not assigned to channel");
return SCPE_OK;
}