blob: 72167626218717bcf55365f8bbe3f39a2c3785b2 [file] [log] [blame] [raw]
#include "ibm1130_defs.h"
/* ibm1130_cr.c: IBM 1130 1442 Card Reader simulator
Based on the SIMH package written by Robert M Supnik
* (C) Copyright 2002, Brian Knittel.
* You may freely use this program, but: it offered strictly on an AS-IS, AT YOUR OWN
* RISK basis, there is no warranty of fitness for any purpose, and the rest of the
* usual yada-yada. Please keep this notice and the copyright in any distributions
* or modifications.
*
* This is not a supported product, but I welcome bug reports and fixes.
* Mail to sim@ibm1130.org
NOTE - there is a problem with this code. The Device Status Word (DSW) is
computed from current conditions when requested by an XIO load status
command; the value of DSW available to the simulator's examine & save
commands may NOT be accurate. This should probably be fixed.
* Update 2002-02-29: Added deck-list option. If you issue an attach
command and specify the filename as "@filename", the named file is interpreted
as a list of filenames to be read in sequence; the effect is that the reader
sees the concatenation of all of the files named. "reset" rewinds the deck
list. Filenames can be followed by whitespace and the letter "a" or "b",
which indicates "ascii to 029" or "binary", respectively. Example:
attach cr @decklist
where file "decklist" contains:
file01 a
file02 b
file03 b
file04 b
If "a" or "b" is not specified, the device mode setting is used.
('a' means 029, so, if you need 026 coding, specify the
device default as the correct 026 code and omit the 'a' on the text files lines).
* note: I'm not sure but I think we'll need a way to simulate the 'start'
button. What may end up being necessary is to fake this in the 'attach'
command. In a GUI build we may want to wait until they press a button.
Have to research: does DMS issue a read request which is only
satisfied when START is pressed, or does START cause an interrupt that
then asks DMS to issue a read request. I think it's the former but need
to check. After all the status register says "empty" and "not ready"
when the hopper is empty. So what gives? On the 360 I think the start
button causes issues some sort of attention request.
* Card image format.
Card files can be ascii text or binary. There are several ASCII modes:
CODE_029, CODE_26F, etc, corresponding to different code sets.
Punch and reader modes can be set independently.
The 1442 card read/punch has several cycles:
feed cycle: moves card from hopper to read station
card from read station to punch station
card from punch station to stacker
read or punch: operates on card at read or punch station (but not both).
The simulator requires input cards to be read from the file attached
to the card reader unit. A feed cycle reads one line (text mode) or
160 bytes (binary mode) from the input file to the read station buffer,
copies the read station buffer to the punch station buffer, and if
the punch unit is attached to a file, writes the punch station buffer to
the output file.
The read and punch cycles operate on the appropriate card buffer.
Detaching the card punch flushes the punch station buffer if necessary.
As does the 1442, a read or punch cycle w/o a feed cycle causes a
feed cycle first.
A feed cycle on an empty deck (reader unattaced or at EOF) clears
the appropriate buffer, so you can punch w/o attaching a deck to
the card reader.
// -- this may need changing depending on how things work in hardware. TBD.
|| A read cycle on an empty deck causes an error.
|| Hmmm -- what takes the place of the Start button on
\\ the card reader?
Binary format is stored using fxwrite of short ints, in this format:
1 1
2 2 0 1 2 3 4 5 6 7 8 9
* * * * * * * * * * * * 0 0 0 0
MSB LSB
byte 0 [ 6] [ 7] [ 8] [ 9] 0 0 0 0
byte 1 [12] [11] [ 0] [ 1] [ 2] [ 3] [ 4] [ 5]
This means we can read words (little endian) and get this in memory:
12 11 0 1 2 3 4 5 6 7 8 9 - - - -
which is what the 1130 sees.
ASCII can be read in blocks of 80 characters but can be terminated by newline prematurely.
Booting: card reader IPL loads 80 columns (1 card) into memory starting
at location 0 in a split fashion:
________________ _ _ _
/
12 |
11 |
0 |
1 |
2 |
3 | Punched card
4 |
5 |
6 |
7 |
8 |
9 |
+------------------ - - -
12 11 0 1 2 3 4 5 6 7 8 9
| | | | | 0 0 0 / \ | | | | | |
+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
| 0| 1| 2| 3| 4| 5| 6| 7| 8| 9|10|11|12|13|14|15|
+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
| OPCODE | F| Tag | DISPLACEMENT |
+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+
The zeros mean that all IPL instructions are short form,
nonindexed. The 3 column is repeated in bits 8 and 9 so
it's a sign bit.
Boot command on a binary deck does this. Boot on an unattached
reader loads the standard boot2 card image. Boot with an ASCII
deck will not be very helpful.
*/
#define READ_DELAY 35 // see how small a number we can get away with
#define PUNCH_DELAY 35
#define FEED_DELAY 25
// #define IS_ONLINE(u) (((u)->flags & (UNIT_ATT|UNIT_DIS)) == UNIT_ATT)
extern int32 sim_switches;
static t_stat cr_svc (UNIT *uptr);
static t_stat cr_reset (DEVICE *dptr);
static t_stat cr_set_code (UNIT *uptr, int32 match, char *cptr, void *desc);
static t_stat cr_attach (UNIT *uptr, char *cptr);
static t_stat cr_detach (UNIT *uptr);
static t_stat cp_reset (DEVICE *dptr);
static t_stat cp_set_code (UNIT *uptr, int32 match, char *cptr, void *desc);
static t_stat cp_detach (UNIT *uptr);
static int16 cr_dsw = 0; /* device status word */
static int32 cr_wait = READ_DELAY; /* read per-column wait */
static int32 cf_wait = PUNCH_DELAY; /* punch per-column wait */
static int32 cp_wait = FEED_DELAY; /* feed op wait */
#define UNIT_V_OPERATION (UNIT_V_UF + 0) /* operation in progress */
#define UNIT_V_CODE (UNIT_V_UF + 2)
#define UNIT_V_EMPTY (UNIT_V_UF + 4)
#define UNIT_V_SCRATCH (UNIT_V_UF + 5)
#define UNIT_V_QUIET (UNIT_V_UF + 6)
#define UNIT_V_DEBUG (UNIT_V_UF + 7)
#define UNIT_V_LASTPUNCH (UNIT_V_UF + 0) /* bit in unit_cp flags */
#define UNIT_OP (3u << UNIT_V_OPERATION) /* two bits */
#define UNIT_CODE (3u << UNIT_V_CODE) /* two bits */
#define UNIT_EMPTY (1u << UNIT_V_EMPTY)
#define UNIT_SCRATCH (1u << UNIT_V_SCRATCH) /* temp file */
#define UNIT_QUIET (1u << UNIT_V_QUIET)
#define UNIT_DEBUG (1u << UNIT_V_DEBUG)
#define UNIT_LASTPUNCH (1u << UNIT_V_LASTPUNCH)
#define OP_IDLE (0u << UNIT_V_OPERATION)
#define OP_READING (1u << UNIT_V_OPERATION)
#define OP_PUNCHING (2u << UNIT_V_OPERATION)
#define OP_FEEDING (3u << UNIT_V_OPERATION)
#define SET_OP(op) {cr_unit.flags &= ~UNIT_OP; cr_unit.flags |= op;}
#define CURRENT_OP (cr_unit.flags & UNIT_OP)
#define CODE_029 (0u << UNIT_V_CODE)
#define CODE_026F (1u << UNIT_V_CODE)
#define CODE_026C (2u << UNIT_V_CODE)
#define CODE_BINARY (3u << UNIT_V_CODE)
#define SET_CODE(un,cd) {un.flags &= ~UNIT_CODE; un.flags |= cd;}
#define COLUMN u4 /* column field in unit record */
UNIT cr_unit = { UDATA (&cr_svc, UNIT_ATTABLE|UNIT_ROABLE, 0) };
UNIT cp_unit = { UDATA (NULL, UNIT_ATTABLE, 0) };
MTAB cr_mod[] = {
{ UNIT_CODE, CODE_029, "029", "029", &cr_set_code},
{ UNIT_CODE, CODE_026F, "026F", "026F", &cr_set_code},
{ UNIT_CODE, CODE_026C, "026C", "026C", &cr_set_code},
{ UNIT_CODE, CODE_BINARY, "BINARY", "BINARY", &cr_set_code},
{ 0 } };
MTAB cp_mod[] = {
{ UNIT_CODE, CODE_029, "029", "029", &cp_set_code},
{ UNIT_CODE, CODE_026F, "026F", "026F", &cp_set_code},
{ UNIT_CODE, CODE_026C, "026C", "026C", &cp_set_code},
{ UNIT_CODE, CODE_BINARY, "BINARY", "BINARY", &cp_set_code},
{ 0 } };
REG cr_reg[] = {
{ HRDATA (CRDSW, cr_dsw, 16) }, /* device status word */
{ DRDATA (CRTIME, cr_wait, 24), PV_LEFT }, /* operation wait */
{ DRDATA (CFTIME, cf_wait, 24), PV_LEFT }, /* operation wait */
{ NULL } };
REG cp_reg[] = {
{ DRDATA (CPTIME, cp_wait, 24), PV_LEFT }, /* operation wait */
{ NULL } };
DEVICE cr_dev = {
"CR", &cr_unit, cr_reg, cr_mod,
1, 16, 16, 1, 16, 16,
NULL, NULL, cr_reset,
cr_boot, cr_attach, cr_detach};
DEVICE cp_dev = {
"CP", &cp_unit, cp_reg, cp_mod,
1, 16, 16, 1, 16, 16,
NULL, NULL, cp_reset,
NULL, NULL, cp_detach};
#define CR_DSW_READ_RESPONSE 0x8000 /* device status word bits */
#define CR_DSW_PUNCH_RESPONSE 0x4000
#define CR_DSW_ERROR_CHECK 0x2000
#define CR_DSW_LAST_CARD 0x1000
#define CR_DSW_OP_COMPLETE 0x0800
#define CR_DSW_FEED_CHECK 0x0100
#define CR_DSW_BUSY 0x0002
#define CR_DSW_NOT_READY 0x0001
typedef struct {
int hollerith;
char ascii;
} CPCODE;
static CPCODE cardcode_029[] =
{
0x0000, ' ',
0x8000, '&', // + in 026 Fortran
0x4000, '-',
0x2000, '0',
0x1000, '1',
0x0800, '2',
0x0400, '3',
0x0200, '4',
0x0100, '5',
0x0080, '6',
0x0040, '7',
0x0020, '8',
0x0010, '9',
0x9000, 'A',
0x8800, 'B',
0x8400, 'C',
0x8200, 'D',
0x8100, 'E',
0x8080, 'F',
0x8040, 'G',
0x8020, 'H',
0x8010, 'I',
0x5000, 'J',
0x4800, 'K',
0x4400, 'L',
0x4200, 'M',
0x4100, 'N',
0x4080, 'O',
0x4040, 'P',
0x4020, 'Q',
0x4010, 'R',
0x3000, '/',
0x2800, 'S',
0x2400, 'T',
0x2200, 'U',
0x2100, 'V',
0x2080, 'W',
0x2040, 'X',
0x2020, 'Y',
0x2010, 'Z',
0x0820, ':',
0x0420, '#', // = in 026 Fortran
0x0220, '@', // ' in 026 Fortran
0x0120, '\'',
0x00A0, '=',
0x0060, '"',
0x8820, 'c', // cent
0x8420, '.',
0x8220, '<', // ) in 026 Fortran
0x8120, '(',
0x80A0, '+',
0x8060, '|',
0x4820, '!',
0x4420, '$',
0x4220, '*',
0x4120, ')',
0x40A0, ';',
0x4060, 'n', // not
0x2820, 'x', // what?
0x2420, ',',
0x2220, '%', // ( in 026 Fortran
0x2120, '_',
0x20A0, '>',
0x2060, '>',
};
static CPCODE cardcode_026F[] = // 026 fortran
{
0x0000, ' ',
0x8000, '+',
0x4000, '-',
0x2000, '0',
0x1000, '1',
0x0800, '2',
0x0400, '3',
0x0200, '4',
0x0100, '5',
0x0080, '6',
0x0040, '7',
0x0020, '8',
0x0010, '9',
0x9000, 'A',
0x8800, 'B',
0x8400, 'C',
0x8200, 'D',
0x8100, 'E',
0x8080, 'F',
0x8040, 'G',
0x8020, 'H',
0x8010, 'I',
0x5000, 'J',
0x4800, 'K',
0x4400, 'L',
0x4200, 'M',
0x4100, 'N',
0x4080, 'O',
0x4040, 'P',
0x4020, 'Q',
0x4010, 'R',
0x3000, '/',
0x2800, 'S',
0x2400, 'T',
0x2200, 'U',
0x2100, 'V',
0x2080, 'W',
0x2040, 'X',
0x2020, 'Y',
0x2010, 'Z',
0x0420, '=',
0x0220, '\'', // ' in 026 Fortran
0x8420, '.',
0x8220, ')',
0x4420, '$',
0x4220, '*',
0x2420, ',',
0x2220, '(',
};
static CPCODE cardcode_026C[] = // 026 commercial
{
0x0000, ' ',
0x8000, '+',
0x4000, '-',
0x2000, '0',
0x1000, '1',
0x0800, '2',
0x0400, '3',
0x0200, '4',
0x0100, '5',
0x0080, '6',
0x0040, '7',
0x0020, '8',
0x0010, '9',
0x9000, 'A',
0x8800, 'B',
0x8400, 'C',
0x8200, 'D',
0x8100, 'E',
0x8080, 'F',
0x8040, 'G',
0x8020, 'H',
0x8010, 'I',
0x5000, 'J',
0x4800, 'K',
0x4400, 'L',
0x4200, 'M',
0x4100, 'N',
0x4080, 'O',
0x4040, 'P',
0x4020, 'Q',
0x4010, 'R',
0x3000, '/',
0x2800, 'S',
0x2400, 'T',
0x2200, 'U',
0x2100, 'V',
0x2080, 'W',
0x2040, 'X',
0x2020, 'Y',
0x2010, 'Z',
0x0420, '=',
0x0220, '\'', // ' in 026 Fortran
0x8420, '.',
0x8220, ')',
0x4420, '$',
0x4220, '*',
0x2420, ',',
0x2220, '(',
};
extern int cgi;
extern void sub_args (char *instr, char *tmpbuf, int32 maxstr, int32 nargs, char *arg[]);
static int16 ascii_to_card[256];
static CPCODE *cardcode;
static int ncardcode;
static int32 active_cr_code; /* the code most recently specified */
static FILE *deckfile = NULL;
static char tempfile[128];
static int cardnum;
static int any_punched = 0;
#define MAXARG 80 /* saved arguments to attach command */
static char list_save[MAXARG][10], *list_arg[MAXARG];
static int list_nargs = 0;
static int16 punchstation[80];
static int16 readstation[80];
static enum {STATION_EMPTY, STATION_LOADED, STATION_READ, STATION_PUNCHED} punchstate = STATION_EMPTY, readstate = STATION_EMPTY;
static t_bool nextdeck (void);
static void checkdeck (void);
/* lookup_codetable - use code flag setting to get code table pointer and length */
static t_bool lookup_codetable (int32 match, CPCODE **pcode, int *pncode)
{
switch (match) {
case CODE_029:
*pcode = cardcode_029;
*pncode = sizeof(cardcode_029) / sizeof(CPCODE);
break;
case CODE_026F:
*pcode = cardcode_026F;
*pncode = sizeof(cardcode_026F) / sizeof(CPCODE);
break;
case CODE_026C:
*pcode = cardcode_026C;
*pncode = sizeof(cardcode_026C) / sizeof(CPCODE);
break;
case CODE_BINARY:
*pcode = NULL;
*pncode = 0;
break;
default:
printf("Eek! Undefined code table index");
return FALSE;
}
return TRUE;
}
t_stat set_active_cr_code (int match)
{
CPCODE *code;
int i, ncode;
active_cr_code = match;
if (! lookup_codetable(match, &code, &ncode))
return SCPE_ARG;
memset(ascii_to_card, 0, sizeof(ascii_to_card));
for (i = 0; i < ncode; i++) // set ascii to card code table
ascii_to_card[code[i].ascii] = (int16) code[i].hollerith;
return SCPE_OK;
}
t_stat cr_set_code (UNIT *uptr, int32 match, char *cptr, void *desc)
{
return set_active_cr_code(match);
}
t_stat cp_set_code (UNIT *uptr, int32 match, char *cptr, void *desc)
{
CPCODE *code;
int ncode;
if (! lookup_codetable(match, &code, &ncode))
return SCPE_ARG;
cardcode = code; // save code table for punch output
ncardcode = ncode;
return SCPE_OK;
}
t_stat load_cr_boot (int drvno)
{
/* this is from the "boot2" cold start card. Columns have been */
/* expanded already from 12 to 16 bits. */
static unsigned short boot2_data[] = {
0xc80a, 0x18c2, 0xd008, 0xc019, 0x8007, 0xd017, 0xc033, 0x100a,
0xd031, 0x7015, 0x000c, 0xe800, 0x0020, 0x08f8, 0x4828, 0x7035,
0x70fa, 0x4814, 0xf026, 0x2000, 0x8800, 0x9000, 0x9800, 0xa000,
0xb000, 0xb800, 0xb810, 0xb820, 0xb830, 0xb820, 0x3000, 0x08ea,
0xc0eb, 0x4828, 0x70fb, 0x9027, 0x4830, 0x70f8, 0x8001, 0xd000,
0xc0f4, 0xd0d9, 0xc01d, 0x1804, 0xe8d6, 0xd0d9, 0xc8e3, 0x18d3,
0xd017, 0x18c4, 0xd0d8, 0x9016, 0xd815, 0x90db, 0xe8cc, 0xd0ef,
0xc016, 0x1807, 0x0035, 0x00d0, 0xc008, 0x1803, 0xe8c4, 0xd00f,
0x080d, 0x08c4, 0x1003, 0x4810, 0x70d9, 0x3000, 0x08df, 0x3000,
0x7010, 0x00d1, 0x0028, 0x000a, 0x70f3, 0x0000, 0x00d0, 0xa0c0
};
int i;
if (drvno >= 0) /* if specified, set toggle switches to disk drive no */
CES = drvno; /* so BOOT DSK1 will work correctly */
IAR = 0; /* clear IAR */
for (i = 0; i < 80; i++) /* copy memory */
WriteW(i, boot2_data[i]);
#ifdef GUI_SUPPORT
if (! cgi)
remark_cmd("Loaded BOOT2 cold start card\n");
#endif
return SCPE_OK;
}
t_stat cr_boot (int unitno, DEVICE *dptr)
{
t_stat rval;
short buf[80];
int i;
if ((rval = reset_all(0)) != SCPE_OK)
return rval;
if (! (cr_unit.flags & UNIT_ATT)) // no deck; load standard boot anyway
return load_cr_boot(-1);
if ((active_cr_code & UNIT_CODE) != CODE_BINARY) {
printf("Can only boot from card reader when set to BINARY mode");
return SCPE_IOERR;
}
if (fxread(buf, sizeof(short), 80, cr_unit.fileref) != 80)
return SCPE_IOERR;
IAR = 0; /* Program Load sets IAR = 0 */
for (i = 0; i < 80; i++) /* shift 12 bits into 16 */
WriteW(i, (buf[i] & 0xF800) | ((buf[i] & 0x0400) ? 0x00C0 : 0x0000) | ((buf[i] & 0x03F0) >> 4));
return SCPE_OK;
}
char card_to_ascii (int16 hol)
{
int i;
for (i = 0; i < ncardcode; i++)
if (cardcode[i].hollerith == hol)
return cardcode[i].ascii;
return ' ';
}
// hollerith_to_ascii - provide a generic conversion for simulator debugging
char hollerith_to_ascii (int16 hol)
{
int i;
for (i = 0; i < ncardcode; i++)
if (cardcode_029[i].hollerith == hol)
return cardcode[i].ascii;
return ' ';
}
/* feedcycle - move cards to next station */
static void feedcycle (t_bool load, t_bool punching)
{
char buf[84], *x;
int i, nread, nwrite, ch;
/* write punched card if punch is attached to a file */
if (cp_unit.flags & UNIT_ATT) {
if (any_punched && punchstate != STATION_EMPTY) {
if ((cp_unit.flags & UNIT_CODE) == CODE_BINARY) {
fxwrite(punchstation, sizeof(short), 80, cp_unit.fileref);
}
else {
for (i = 80; --i >= 0; ) { /* find last nonblank column */
if (buf[i] != 0)
break;
}
/* i is now index of last character to output or -1 if all blank */
for (nwrite = 0; nwrite <= i; nwrite++) { /* convert characters */
buf[nwrite] = card_to_ascii(punchstation[nwrite]);
}
/* nwrite is now number of characters to output */
buf[nwrite++] = '\n'; /* append newline */
fxwrite(buf, sizeof(char), nwrite, cp_unit.fileref);
}
}
}
if (! load) // all we wanted to do was flush the punch
return;
/* slide cards from reader to punch. If we know we're punching,
* generate a blank card in any case. Otherwise, it should take two feed
* cycles to get a read card from the hopper to punch station */
if (readstate == STATION_EMPTY) {
if (punching) {
memset(punchstation, 0, sizeof(punchstation));
punchstate = STATION_LOADED;
}
else
punchstate = STATION_EMPTY;
}
else {
memcpy(punchstation, readstation, sizeof(punchstation));
punchstate = STATION_LOADED;
}
/* load card into read station */
again: /* jump here if we've loaded a new deck after emptying the previous one */
if (cr_unit.flags & UNIT_ATT) {
memset(readstation, 0, sizeof(readstation)); /* blank out the card image */
if (cr_unit.fileref == NULL)
nread = 0;
else if ((active_cr_code & UNIT_CODE) == CODE_BINARY) /* binary read is straightforward */
nread = fxread(readstation, sizeof(short), 80, cr_unit.fileref);
else if (fgets(buf, sizeof(buf), cr_unit.fileref) == NULL) /* read up to 80 chars */
nread = 0; /* hmm, end of file */
else { /* check for newline */
if ((x = strchr(buf, '\r')) == NULL)
x = strchr(buf, '\n');
if (x == NULL) { /* there were no delimiters, burn rest of line */
while ((ch = getc(cr_unit.fileref)) != EOF) { /* get character */
if (ch == '\n') /* newline, done */
break;
if (ch == '\r') { /* CR, try to take newline too */
ch = getc(cr_unit.fileref);
if (ch != EOF && ch != '\n') /* hmm, put it back */
ungetc(ch, cr_unit.fileref);
break;
}
}
nread = 80; /* take just the first 80 characters */
}
else
nread = x-buf; /* reduce length of string */
upcase(buf); /* force uppercase */
for (i = 0; i < nread; i++) /* convert ascii to punch code */
readstation[i] = ascii_to_card[buf[i]];
nread = 80; /* even if line was blank consider it present */
}
if (nread <= 0) { /* set hopper flag accordingly */
if (deckfile != NULL && nextdeck())
goto again;
SETBIT(cr_unit.flags, UNIT_EMPTY);
readstate = STATION_EMPTY;
cardnum = -1; /* nix the card counter */
}
else {
CLRBIT(cr_unit.flags, UNIT_EMPTY);
readstate = STATION_LOADED;
cardnum++; /* advance card counter */
}
}
else
readstate = STATION_EMPTY;
cr_unit.COLUMN = -1; /* neither device is currently cycling */
cp_unit.COLUMN = -1;
}
#ifdef NO_USE_FOR_THIS_CURRENTLY
/* this routine should probably be hooked up to the GUI somehow */
/* NPRO - nonprocess runout, flushes out the reader/punch */
static void npro (void)
{
if (cr_unit.flags & UNIT_ATT)
fseek(cr_unit.fileref, 0, SEEK_END); /* push reader to EOF */
if (deckfile != NULL)
fseek(deckfile, 0, SEEK_END); /* skip to end of deck list */
cardnum = -1; /* nix the card counter */
if (punchstate == STATION_PUNCHED)
feedcycle(FALSE, FALSE); /* flush out card just punched */
readstate = punchstate = STATION_EMPTY;
cr_unit.COLUMN = -1; /* neither device is currently cycling */
cp_unit.COLUMN = -1;
SETBIT(cr_unit.flags, UNIT_EMPTY); /* set hopper empty */
}
#endif
/* skipbl - skip leading whitespace in a string */
static char * skipbl (char *str)
{
while (*str && *str <= ' ')
str++;
return str;
}
/* alltrim - remove all leading and trailing whitespace from a string */
static char * alltrim (char *str)
{
char *s, *lastnb;
if ((s = skipbl(str)) != str) /* slide down over leading whitespace */
strcpy(str, s);
for (lastnb = str-1, s = str; *s; s++) /* point to last nonblank characteter in string */
if (*s > ' ')
lastnb = s;
lastnb[1] = '\0'; /* clip just after it */
return str;
}
/* checkdeck - set hopper empty status based on condition of current reader file */
static void checkdeck (void)
{
t_bool empty;
if (cr_unit.fileref == NULL) { /* there is no open file */
empty = TRUE;
}
else {
fseek(cr_unit.fileref, 0, SEEK_END);
empty = ftell(cr_unit.fileref) <= 0; /* see if file has anything) */
fseek(cr_unit.fileref, 0, SEEK_SET); /* rewind deck */
cardnum = 0; /* reset card counter */
}
if (empty) {
SETBIT(cr_unit.flags, UNIT_EMPTY);
if (cr_unit.fileref != NULL) /* real file but it's empty, hmmm, try another */
nextdeck();
}
else
CLRBIT(cr_unit.flags, UNIT_EMPTY);
}
/* nextdeck - attempt to load a new file from the deck list into the hopper */
static t_bool nextdeck (void)
{
char buf[200], tmpbuf[200], *fname, *mode, *tn;
int code;
long fpos;
static char white[] = " \t\r\n";
cardnum = 0; /* reset card counter */
if (deckfile == NULL) /* we can't help */
return FALSE;
code = cr_unit.flags & UNIT_CODE; /* default code */
if (cr_unit.fileref != NULL) { /* this pulls the rug out from under scp */
fclose(cr_unit.fileref); /* since the attach flag is still set. be careful! */
cr_unit.fileref = NULL;
if (cr_unit.flags & UNIT_SCRATCH) {
unlink(tempfile);
CLRBIT(cr_unit.flags, UNIT_SCRATCH);
}
}
for (;;) { /* get a filename */
if (fgets(buf, sizeof(buf), deckfile) == NULL)
break; /* oops, no more names */
alltrim(buf);
if (! *buf)
continue; /* empty line */
if (strnicmp(buf, "!BREAK", 6) == 0) { /* stop the simulation */
break_simulation(STOP_DECK_BREAK);
continue;
}
if (buf[0] == '!') { /* literal text line, make a temporary file */
if (*tempfile == '\0') {
if ((tn = tempnam(".", "1130")) == NULL) {
printf("Cannot create temporary card file name\n");
break_simulation(STOP_DECK_BREAK);
return 0;
}
strcpy(tempfile, tn);
strcat(tempfile, ".tmp");
}
if ((cr_unit.fileref = fopen(tempfile, "wb+")) == NULL) {
printf("Cannot create temporary file %s\n", tempfile);
break_simulation(STOP_DECK_BREAK);
return 0;
}
SETBIT(cr_unit.flags, UNIT_SCRATCH);
for (;;) { /* store literal cards into temporary file */
upcase(buf+1);
fputs(buf+1, cr_unit.fileref);
putc('\n', cr_unit.fileref);
trace_io("(Literal card %s\n)", buf+1);
if (! (cr_unit.flags & UNIT_QUIET))
printf("(Literal card %s)\n", buf+1);
fpos = ftell(deckfile);
if (fgets(buf, sizeof(buf), deckfile) == NULL)
break; /* oops, end of file */
if (buf[0] != '!' || strnicmp(buf, "!BREAK", 6) == 0)
break;
alltrim(buf);
}
fseek(deckfile, fpos, SEEK_SET); /* restore deck file to just before non-literal card */
fseek(cr_unit.fileref, 0, SEEK_SET); /* rewind scratch file for reading */
code = CODE_029; /* assume keycode 029 */
break;
}
sub_args(buf, tmpbuf, sizeof(buf), list_nargs, list_arg); /* substitute in stuff from the attach command line */
if ((fname = strtok(buf, white)) == NULL)
continue;
if (*fname == '#' || *fname == '*' || *fname == ';')
continue; /* comment */
if ((mode = strtok(NULL, white)) != NULL) {
if (*mode == 'b' || *mode == 'B')
code = CODE_BINARY;
else if (*mode == 'a' || *mode == 'A')
code = CODE_029;
}
if ((cr_unit.fileref = fopen(fname, "rb")) == NULL)
printf("File '%s' specified in deck file '%s' cannot be opened\n", fname, cr_unit.filename+1);
else {
trace_io("(Opened %s deck %s)\n", (code == CODE_BINARY) ? "binary" : "text", fname);
if (! (cr_unit.flags & UNIT_QUIET))
printf("(Opened %s deck %s)\n", (code == CODE_BINARY) ? "binary" : "text", fname);
break;
}
}
checkdeck();
set_active_cr_code(code); /* set specified code */
return (cr_unit.flags & UNIT_EMPTY) == 0; /* return TRUE if a deck has been loaded */
}
static t_stat cr_reset (DEVICE *dptr)
{
cr_set_code(&cr_unit, active_cr_code & UNIT_CODE, NULL, NULL); /* reset to specified code table */
readstate = STATION_EMPTY;
cr_dsw = 0;
sim_cancel(&cr_unit); /* cancel any pending ops */
calc_ints();
SET_OP(OP_IDLE);
SETBIT(cr_unit.flags, UNIT_EMPTY); /* assume hopper empty */
if (cr_unit.flags & UNIT_ATT) {
// if (deckfile != NULL) {
// fseek(deckfile, 0, SEEK_SET);
// nextdeck();
// }
// else
// checkdeck();
if (cr_unit.fileref != NULL)
feedcycle(FALSE, FALSE);
}
cr_unit.COLUMN = -1; /* neither device is currently cycling */
return SCPE_OK;
}
static t_stat cp_reset (DEVICE *dptr)
{
cp_set_code(&cp_unit, cp_unit.flags & UNIT_CODE, NULL, NULL);
punchstate = STATION_EMPTY;
cp_unit.COLUMN = -1;
return SCPE_OK;
}
static t_stat cr_attach (UNIT *uptr, char *cptr)
{
t_stat rval;
t_bool use_decklist;
char *c, *arg, quote;
// no - don't cancel pending read?
// sim_cancel(uptr); /* cancel pending operations */
CLRBIT(uptr->flags, UNIT_QUIET|UNIT_DEBUG); /* set debug/quiet flags */
if (sim_switches & SWMASK('D')) SETBIT(uptr->flags, UNIT_DEBUG);
else if (sim_switches & SWMASK('Q')) SETBIT(uptr->flags, UNIT_QUIET);
cr_detach(uptr); /* detach file and possibly deckfile */
CLRBIT(uptr->flags, UNIT_SCRATCH);
c = cptr;
for (list_nargs = 0; list_nargs < 10; ) { /* extract arguments */
while (*c && (*c <= ' ')) /* skip blanks */
c++;
if (! *c)
break; /* all done */
arg = c; /* save start */
while (*c && (*c > ' ')) {
if (*c == '\'' || *c == '"') { /* quoted string */
for (quote = *c++; *c;)
if (*c++ == quote)
break;
}
else c++;
}
if (*c)
*c++ = 0; /* term arg at space */
list_arg[list_nargs] = list_save[list_nargs]; /* set pointer to permanent storage location */
strncpy(list_arg[list_nargs++], arg, MAXARG); /* store copy */
}
if (list_nargs <= 0) /* need at least 1 */
return SCPE_2FARG;
cptr = list_arg[0]; /* filename is first argument */
use_decklist = (*cptr == '@'); /* filename starts with @: it's a deck list */
if (use_decklist)
cptr++;
if (strcmp(cptr, "-") == 0 && ! use_decklist) { /* standard input */
if (uptr -> flags & UNIT_DIS) return SCPE_UDIS; /* disabled? */
uptr->filename = calloc(CBUFSIZE, sizeof(char));
strcpy(uptr->filename, "(stdin)");
uptr->fileref = stdin;
SETBIT(uptr->flags, UNIT_ATT);
uptr->pos = 0;
}
else if ((rval = attach_unit(uptr, cptr)) != SCPE_OK)
return rval;
if (use_decklist) { /* if we skipped the '@', store the actually-specified name */
strncpy(uptr->filename, cptr-1, CBUFSIZE);
deckfile = cr_unit.fileref; /* save the deck file stream in our local variable */
cr_unit.fileref = NULL;
nextdeck();
}
else
checkdeck();
// there is a read pending. Pull the card in to make it go
if (CURRENT_OP == OP_READING || CURRENT_OP == OP_PUNCHING || CURRENT_OP == OP_FEEDING)
feedcycle(TRUE, CURRENT_OP == OP_PUNCHING);
// no - don't reset the reader
// cr_reset(&cr_dev); /* reset the whole thing */
// cp_reset(&cp_dev);
cardnum = 0; /* reset card counter */
return SCPE_OK;
}
static t_stat cr_detach (UNIT *uptr)
{
t_stat rval;
if (cr_unit.flags & UNIT_ATT && deckfile != NULL) {
if (cr_unit.fileref != NULL) /* close the active card deck */
fclose(cr_unit.fileref);
if (cr_unit.flags & UNIT_SCRATCH) {
unlink(tempfile);
CLRBIT(cr_unit.flags, UNIT_SCRATCH);
}
cr_unit.fileref = deckfile; /* give scp a file to close */
}
if (uptr->fileref == stdout) {
CLRBIT(uptr->flags, UNIT_ATT);
free(uptr->filename);
uptr->filename = NULL;
rval = SCPE_OK;
}
else
rval = detach_unit(uptr);
return rval;
}
static t_stat cp_detach (UNIT *uptr)
{
if (cp_unit.flags & UNIT_ATT)
if (punchstate == STATION_PUNCHED)
feedcycle(FALSE, FALSE); /* flush out card just punched */
any_punched = 0; /* reset punch detected */
return detach_unit(uptr);
}
static void op_done (void)
{
if (cr_unit.flags & UNIT_DEBUG)
DEBUG_PRINT("!CR Op Complete, card %d", cardnum);
SET_OP(OP_IDLE);
SETBIT(cr_dsw, CR_DSW_OP_COMPLETE);
SETBIT(ILSW[4], ILSW_4_1442_CARD);
calc_ints();
}
static t_stat cr_svc (UNIT *uptr)
{
switch (CURRENT_OP) {
case OP_IDLE:
break;
case OP_FEEDING:
op_done();
break;
case OP_READING:
if (readstate == STATION_EMPTY) { /* read active but no cards? hang */
sim_activate(&cr_unit, cf_wait);
break;
}
if (++cr_unit.COLUMN < 80) {
SETBIT(cr_dsw, CR_DSW_READ_RESPONSE);
SETBIT(ILSW[0], ILSW_0_1442_CARD);
calc_ints();
sim_activate(&cr_unit, cr_wait);
if (cr_unit.flags & UNIT_DEBUG)
DEBUG_PRINT("!CR Read Response %d : %d", cardnum, cr_unit.COLUMN+1);
}
else {
readstate = STATION_READ;
op_done();
}
break;
case OP_PUNCHING:
if (punchstate == STATION_EMPTY) { /* punch active but no cards? hang */
sim_activate(&cr_unit, cf_wait);
break;
}
if (cp_unit.flags & UNIT_LASTPUNCH) {
punchstate = STATION_PUNCHED;
op_done();
}
else {
SETBIT(cr_dsw, CR_DSW_PUNCH_RESPONSE);
SETBIT(ILSW[0], ILSW_0_1442_CARD);
calc_ints();
sim_activate(&cr_unit, cp_wait);
if (cr_unit.flags & UNIT_DEBUG)
DEBUG_PRINT("!CR Punch Response");
}
break;
}
return SCPE_OK;
}
void xio_1142_card (int32 addr, int32 func, int32 modify)
{
char msg[80];
int ch;
int16 wd;
t_bool lastcard;
switch (func) {
case XIO_SENSE_DEV:
if (cp_unit.flags & UNIT_ATT)
lastcard = FALSE; /* if punch file is open, assume infinite blank cards in reader */
else if ((cr_unit.flags & UNIT_ATT) == 0)
lastcard = TRUE; /* if nothing to read, hopper's empty */
else if (readstate == STATION_LOADED)
lastcard = FALSE;
else if (cr_unit.fileref == NULL)
lastcard = TRUE;
else if ((ch = getc(cr_unit.fileref)) != EOF) {
ungetc(ch, cr_unit.fileref); /* put character back; hopper's not empty */
lastcard = FALSE;
}
else if (deckfile != NULL && nextdeck())
lastcard = FALSE;
else
lastcard = TRUE; /* there is nothing left to read for a next card */
CLRBIT(cr_dsw, CR_DSW_LAST_CARD|CR_DSW_BUSY|CR_DSW_NOT_READY);
if (lastcard)
SETBIT(cr_dsw, CR_DSW_LAST_CARD);
if (CURRENT_OP != OP_IDLE)
SETBIT(cr_dsw, CR_DSW_BUSY|CR_DSW_NOT_READY);
else if (readstate == STATION_EMPTY && punchstate == STATION_EMPTY && lastcard)
SETBIT(cr_dsw, CR_DSW_NOT_READY);
if (modify & 0x01) { /* reset interrupts */
CLRBIT(cr_dsw, CR_DSW_READ_RESPONSE|CR_DSW_PUNCH_RESPONSE);
CLRBIT(ILSW[0], ILSW_0_1442_CARD);
}
if (modify & 0x02) {
CLRBIT(cr_dsw, CR_DSW_OP_COMPLETE);
CLRBIT(ILSW[4], ILSW_4_1442_CARD);
}
ACC = cr_dsw; /* return the DSW */
if (cr_unit.flags & UNIT_DEBUG)
DEBUG_PRINT("#CR Sense %04x%s%s", cr_dsw, (modify & 1) ? " RESET0" : "", (modify & 2) ? " RESET4" : "");
break;
case XIO_READ: /* get card data into word pointed to in IOCC packet */
if (cr_unit.flags & OP_READING) {
if (cr_unit.COLUMN < 0) {
xio_error("1442: Premature read!");
}
else if (cr_unit.COLUMN < 80) {
WriteW(addr, readstation[cr_unit.COLUMN]);
if (cr_unit.flags & UNIT_DEBUG)
DEBUG_PRINT("#CR Read %03x", (readstation[cr_unit.COLUMN] >> 4));
}
else if (cr_unit.COLUMN == 80) {
xio_error("1442: Read past column 80!");
cr_unit.COLUMN++; // don't report it again
}
}
else {
xio_error("1442: Read when not in a read cycle!");
}
break;
case XIO_WRITE:
if (cr_unit.flags & OP_PUNCHING) {
if (cp_unit.COLUMN < 0) {
xio_error("1442: Premature write!");
}
else if (cp_unit.flags & UNIT_LASTPUNCH) {
xio_error("1442: Punch past last-punch column!");
cp_unit.COLUMN = 81;
}
else if (cp_unit.COLUMN < 80) {
wd = ReadW(addr); /* store one word to punch buffer */
punchstation[cp_unit.COLUMN] = wd & 0xFFF0;
if (wd & 0x0008) /* mark this as last column to be punched */
SETBIT(cp_unit.flags, UNIT_LASTPUNCH);
if (cr_unit.flags & UNIT_DEBUG)
DEBUG_PRINT("#CR Punch %03x%s", (wd >> 4) & 0xFFF, (wd & 8) ? " LAST" : "");
}
else if (cp_unit.COLUMN == 80) {
xio_error("1442: Punch past column 80!");
cp_unit.COLUMN++; // don't report it again
}
}
else {
xio_error("1442: Write when not in a punch cycle!");
}
break;
case XIO_CONTROL:
switch (modify & 7) {
case 1: /* start punch */
if (cr_unit.flags & UNIT_DEBUG)
DEBUG_PRINT("#CR Start Punch");
if (punchstate != STATION_LOADED)
feedcycle(TRUE, TRUE);
SET_OP(OP_PUNCHING);
cp_unit.COLUMN = -1;
CLRBIT(cp_unit.flags, UNIT_LASTPUNCH);
any_punched = 1; /* we've started punching, so enable writing to output deck file */
sim_cancel(&cr_unit);
sim_activate(&cr_unit, cp_wait);
break;
case 2: /* feed cycle */
if (cr_unit.flags & UNIT_DEBUG)
DEBUG_PRINT("#CR Feed");
feedcycle(TRUE, FALSE);
SET_OP(OP_FEEDING);
sim_cancel(&cr_unit);
sim_activate(&cr_unit, cf_wait);
break;
case 4: /* start read */
if (cr_unit.flags & UNIT_DEBUG)
DEBUG_PRINT("#CR Start read");
if (readstate != STATION_LOADED)
feedcycle(TRUE, FALSE);
SET_OP(OP_READING);
cr_unit.COLUMN = -1;
sim_cancel(&cr_unit);
sim_activate(&cr_unit, cr_wait);
break;
case 0:
if (cr_unit.flags & UNIT_DEBUG)
DEBUG_PRINT("#CR NOP");
break;
default:
sprintf(msg, "1442: Multiple operations in XIO_CONTROL: %x", modify);
xio_error(msg);
return;
}
break;
default:
sprintf(msg, "Invalid 1442 XIO function %x", func);
xio_error(msg);
break;
}
}