mirror of
https://github.com/SEPPDROID/Digital-Research-Source-Code.git
synced 2025-10-24 08:54:17 +00:00
1914 lines
53 KiB
Plaintext
1914 lines
53 KiB
Plaintext
$ TITLE('CP/M 3.0 --- SHOW 3.1')
|
|
/*
|
|
Revised:
|
|
18 Sep 1998 by John Elliott (YMD format dates)
|
|
17 May 1998 by John Elliott (year 2000 fix, CP/M Patch 16)
|
|
Oct 82 by Phillip Balma
|
|
14 Sept 81 by Doug Huskey
|
|
*/
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * SHOW * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
show:
|
|
do;
|
|
declare
|
|
mpm literally '30h';
|
|
|
|
declare plm label public;
|
|
|
|
declare copyright(*) byte data
|
|
(' Copyright (c) 1982, 1998 Caldera, Inc. ');
|
|
|
|
declare verdate(*) byte data('18Sep98 '),
|
|
version(*) byte data('Show 3.1');
|
|
|
|
|
|
/*
|
|
copyright(c) 1975, 1976, 1977, 1978, 1979, 1980, 1981,1982
|
|
digital research
|
|
box 579
|
|
pacific grove, ca
|
|
93950
|
|
|
|
*/
|
|
|
|
/* modified 10/30/78 to fix the space computation */
|
|
/* modified 01/28/79 to remove despool dependencies */
|
|
/* modified 07/26/79 to operate under cp/m 2.0 */
|
|
/* modified 01/20/80 by Thomas Rolander */
|
|
/* show created 05/19/81 */
|
|
/* modified 7/82 to add new options parser, # dir FCB's left, new DISK option,
|
|
# of files by Phillip Balma */
|
|
/* added paging, # SFCB's Phillip Balma*/
|
|
/* Modified 17 May 1998 for Year 2000 fix (John Elliott) */
|
|
/* Modified 18 Sep 1998 for YMD format dates (John Elliott) */
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * DISK INTERFACE * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
declare dcnt byte,
|
|
anything byte,
|
|
dirbuf(128) byte;
|
|
|
|
declare
|
|
line$page byte,
|
|
line$out byte,
|
|
drives(16) byte,
|
|
drive byte,
|
|
all byte initial(0),
|
|
once$only byte initial(0),
|
|
done$drive(16) byte initial(0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0),
|
|
|
|
PAGE byte initial(0),
|
|
NONBANK byte initial(0),
|
|
|
|
user(16) byte, /* any files in user i? */
|
|
used(16) address, /* # files in user i */
|
|
free$dir address, /* # free directories */
|
|
nSFCB address, /* # SFCB's */
|
|
|
|
SCBPB structure(
|
|
where byte,
|
|
set byte,
|
|
value address) initial(0,0,0),
|
|
|
|
ERRORM(*) byte data('ERROR: ',0),
|
|
input(*) byte data('INPUT: ',0),
|
|
eoption(*) byte data('OPTION: ',0),
|
|
dirdrive(*) byte data('DRIVE: ',0),
|
|
|
|
err$unrecopt(*) byte data('Unrecognized Option.',0),
|
|
err$unrecd(*) byte data('Unrecognized drive.',0),
|
|
err$version(*) byte data('Requires CP/M 3 or higher.',0),
|
|
err$nolabel(*) byte
|
|
data('No directory label exists on drive ',0),
|
|
err$input(*) byte data('Unrecognized input.',0),
|
|
|
|
opt$dir byte data(1),
|
|
opt$drive byte data(2),
|
|
opt$label byte data(3),
|
|
opt$space byte data(0),
|
|
opt$user byte data(4),
|
|
opt$page byte data(6), /*rel to 1 */
|
|
opt$nopage byte data(7);
|
|
|
|
declare
|
|
|
|
dirs(*) byte data
|
|
('A:0B:0C:0D:0E:0F:0G:0H:0I:0J:0K:0L:0M:0N:0',
|
|
'O:0P:',0ffh),
|
|
options(*) byte data('SPACE0DIRECTORY0DRIVES0LABEL0USERS0',
|
|
'PAGE0NOPAGE',0ffh),
|
|
|
|
off$dirs(*) byte data(0,3,6,9,12,15,18,21,24,27,30,33,36,39,42,
|
|
45,47),
|
|
off$opt(*) byte data(0,6,16,23,29,35,40,46),
|
|
|
|
end$list byte data (0ffh),
|
|
end$of$string byte data (0),
|
|
|
|
delimiters(*) byte data (0,'[]=, :;<>%\|"()/#!@&+-*?',0,0ffh),
|
|
SPACE byte data(5), /* index into delim to space */
|
|
EOS byte data(25),
|
|
COMMA byte data(4),
|
|
COLON byte data(6),
|
|
LBRACKET byte data(1),
|
|
RBRACKET byte data(2),
|
|
|
|
opt$map(21) structure ( option(5) byte),
|
|
|
|
j byte initial(0),
|
|
buf$ptr address,
|
|
opt$index byte,
|
|
endbuf byte,
|
|
delimiter byte;
|
|
$ eject
|
|
|
|
declare
|
|
maxb address external, /* addr field of jmp BDOS */
|
|
fcb(33) byte external, /* default fcb */
|
|
buff(128) byte external, /* default buffer */
|
|
fcba literally '.fcb', /* default fcb */
|
|
dolla literally '.fcb(6dh-5ch)', /* $ position */
|
|
rreca literally '.fcb(7dh-5ch)', /* ran rcd 7d,7e,7f */
|
|
rreco literally '.fcb(7fh-5ch)', /* ran overflow */
|
|
sectorlen literally '128', /* sector length */
|
|
rrec address at(rreca), /* random record address */
|
|
rovf byte at(rreco), /* overflow on getfile */
|
|
doll byte at(dolla), /* dollar parameter */
|
|
user$code byte, /* current user code */
|
|
cversion address, /* BDOS version # */
|
|
cdisk byte, /* current disk */
|
|
|
|
/* function call 32 returns the address of the disk parameter
|
|
block for the currently selected disk, which consists of:
|
|
scptrk (2 by) number of sectors per track
|
|
blkshf (1 by) log2 of blocksize (2**blkshf=blksize)
|
|
blkmsk (1 by) 2**blkshf-1
|
|
extmsk (1 by) logical/physical extents
|
|
maxall (2 by) max alloc number
|
|
dirmax (2 by) size of directory-1
|
|
alloc (2 by) reservation bits for directory
|
|
chksiz (2 by) size of checksum vector
|
|
offset (2 by) offset for operating system
|
|
psh (1 by) log2 of physical record size(2**psh * 128 = size)
|
|
psm (1 by) 2**psh - 1
|
|
*/
|
|
|
|
dpba address, /* disk parameter block address */
|
|
dpb based dpba structure(
|
|
spt address,
|
|
bls byte,
|
|
bms byte,
|
|
exm byte,
|
|
mxa address,
|
|
dmx address,
|
|
dbl address,
|
|
cks address,
|
|
ofs address,
|
|
psh byte,
|
|
psm byte),
|
|
|
|
scptrk literally 'dpb.spt',
|
|
blkshf literally 'dpb.bls',
|
|
blkmsk literally 'dpb.bms',
|
|
extmsk literally 'dpb.exm',
|
|
maxall literally 'dpb.mxa',
|
|
dirmax literally 'dpb.dmx',
|
|
dirblk literally 'dpb.dbl',
|
|
chksiz literally 'dpb.cks',
|
|
offset literally 'dpb.ofs',
|
|
physhf literally 'dpb.psh',
|
|
phymsk literally 'dpb.psm';
|
|
|
|
|
|
boot: procedure external;
|
|
/* reboot */
|
|
end boot;
|
|
|
|
mon1: procedure(f,a) external;
|
|
declare f byte, a address;
|
|
end mon1;
|
|
|
|
mon2: procedure(f,a) byte external;
|
|
declare f byte, a address;
|
|
end mon2;
|
|
|
|
declare mon3 literally 'mon2a';
|
|
|
|
mon3: procedure(f,a) address external;
|
|
declare f byte, a address;
|
|
end mon3;
|
|
|
|
declare alloca address,
|
|
/* alloca is the address of the disk allocation vector */
|
|
alloc based alloca (1024) byte; /* allocation vector */
|
|
|
|
declare
|
|
true literally '1',
|
|
false literally '0',
|
|
forever literally 'while true',
|
|
lit literally 'literally',
|
|
proc literally 'procedure',
|
|
dcl literally 'declare',
|
|
addr literally 'address',
|
|
ctlc literally '3',
|
|
cr literally '13',
|
|
lf literally '10';
|
|
|
|
|
|
printchar: procedure(char);
|
|
declare char byte;
|
|
call mon1(2,char);
|
|
end printchar;
|
|
|
|
printb: procedure;
|
|
/* print blank character */
|
|
call printchar(' ');
|
|
end printb;
|
|
|
|
printx: procedure(a);
|
|
declare a address;
|
|
declare s based a byte;
|
|
|
|
do while s <> 0;
|
|
call printchar(s);
|
|
a = a + 1;
|
|
end;
|
|
|
|
end printx;
|
|
|
|
break: procedure byte;
|
|
return mon2(11,0); /* console ready */
|
|
end break;
|
|
|
|
|
|
crlf2: procedure;
|
|
|
|
call printchar(cr);
|
|
call printchar(lf);
|
|
|
|
end crlf2;
|
|
|
|
|
|
terminate: procedure;
|
|
call crlf2;
|
|
call mon1 (0,0); /* system reset */
|
|
end terminate;
|
|
|
|
|
|
|
|
crlf: procedure;
|
|
|
|
if PAGE then do;
|
|
line$out = line$out + 1;
|
|
if line$out + 2 > line$page then do;
|
|
|
|
call crlf2;
|
|
call crlf2;
|
|
|
|
call printx(.('Press RETURN to continue.',0));
|
|
|
|
do while not break; /* wait until a console break*/
|
|
end;
|
|
if mon2(1,0) = ctlc then call terminate;
|
|
line$out = 1;
|
|
call crlf2;
|
|
end;
|
|
end;
|
|
|
|
call crlf2;
|
|
|
|
end crlf;
|
|
|
|
|
|
print: procedure(a);
|
|
declare a address;
|
|
/* print the string starting at address a until the
|
|
next 0 is encountered */
|
|
call crlf;
|
|
call printx(a);
|
|
|
|
end print;
|
|
|
|
|
|
get$version: procedure byte;
|
|
/* returns current cp/m version # */
|
|
return mon3(12,0);
|
|
end get$version;
|
|
|
|
select: procedure(d);
|
|
declare d byte;
|
|
|
|
call mon1(14,d);
|
|
end select;
|
|
|
|
check$user: procedure;
|
|
do forever;
|
|
if anything then return;
|
|
if dcnt = 0ffh then return;
|
|
if dirbuf(ror (dcnt,3) and 110$0000b) = user$code then return;
|
|
|
|
dcnt = mon2(18,0);
|
|
|
|
end;
|
|
end check$user;
|
|
|
|
search: procedure(fcb);
|
|
declare fcb address;
|
|
declare fcb0 based fcb byte;
|
|
|
|
anything = (fcb0 = '?');
|
|
dcnt = mon2(17,fcb);
|
|
call check$user;
|
|
end search;
|
|
|
|
searchn: procedure;
|
|
dcnt = mon2(18,0);
|
|
call check$user;
|
|
end searchn;
|
|
|
|
cselect: procedure byte;
|
|
/* return current disk number */
|
|
return mon2(25,0);
|
|
end cselect;
|
|
|
|
setdma: procedure(dma);
|
|
declare dma address;
|
|
|
|
call mon1(26,dma);
|
|
end setdma;
|
|
|
|
getalloca: procedure address;
|
|
/* get base address of alloc vector */
|
|
return mon3(27,0);
|
|
end getalloca;
|
|
|
|
getlogin: procedure address;
|
|
/* get the login vector */
|
|
return mon3(24,0);
|
|
end getlogin;
|
|
|
|
getukdate: procedure byte; /* [JCE] Date in UK format? */
|
|
|
|
SCBPB.where = 0ch;
|
|
return (mon2(49,.SCBPB) and 3);
|
|
|
|
end getukdate;
|
|
|
|
|
|
getpage: procedure byte; /* get the conole page length */
|
|
|
|
SCBPB.where = 01ch;
|
|
return mon2(49,.SCBPB);
|
|
|
|
end getpage;
|
|
|
|
|
|
getpagemode: procedure byte;
|
|
|
|
SCBPB.where = 02ch;
|
|
return mon2(49,.SCBPB);
|
|
|
|
end getpagemode;
|
|
|
|
getNB: procedure byte;
|
|
SCBPB.where = 05dh;
|
|
return high(mon3(49,.SCBPB));
|
|
end getNB;
|
|
|
|
getrodisk: procedure address;
|
|
/* get the read-only disk vector */
|
|
return mon3(29,0);
|
|
end getrodisk;
|
|
|
|
/*setind: procedure;
|
|
call mon1(30,fcba);
|
|
end setind;
|
|
*/
|
|
|
|
set$dpb: procedure;
|
|
/* set disk parameter block values */
|
|
dpba = mon3(31,0); /* base of dpb */
|
|
end set$dpb;
|
|
|
|
getuser: procedure byte;
|
|
/* return current user number */
|
|
return mon2(32,0ffh);
|
|
end getuser;
|
|
|
|
/*setuser: procedure(user);
|
|
declare user byte;
|
|
|
|
call mon1(32,user);
|
|
end setuser;
|
|
*/
|
|
|
|
getfreesp: procedure(d);
|
|
declare d byte;
|
|
|
|
call mon1(46,d);
|
|
end getfreesp;
|
|
|
|
getlbl: procedure(d) byte;
|
|
declare d byte;
|
|
|
|
return mon2(101,d);
|
|
end getlbl;
|
|
|
|
e$print: procedure(msg);
|
|
declare msg address;
|
|
|
|
call print(.ERRORM);
|
|
call printx(msg);
|
|
|
|
end e$print;
|
|
|
|
|
|
/*****************************************************
|
|
|
|
Time & Date ASCII Conversion Code
|
|
|
|
*****************************************************/
|
|
|
|
declare tod$adr address;
|
|
declare tod based tod$adr structure (
|
|
opcode byte,
|
|
date address,
|
|
hrs byte,
|
|
min byte,
|
|
sec byte,
|
|
ASCII (21) byte );
|
|
|
|
declare string$adr address;
|
|
declare string based string$adr (1) byte;
|
|
declare index byte;
|
|
|
|
emitchar: procedure(c);
|
|
declare c byte;
|
|
string(index := index + 1) = c;
|
|
end emitchar;
|
|
|
|
emitn: procedure(a);
|
|
declare a address;
|
|
declare c based a byte;
|
|
do while c <> '$';
|
|
string(index := index + 1) = c;
|
|
a = a + 1;
|
|
end;
|
|
end emitn;
|
|
|
|
|
|
emit$bcd: procedure(b);
|
|
declare b byte;
|
|
call emitchar('0'+b);
|
|
end emit$bcd;
|
|
|
|
emit$bcd$pair: procedure(b);
|
|
declare b byte;
|
|
call emit$bcd(shr(b,4));
|
|
call emit$bcd(b and 0fh);
|
|
end emit$bcd$pair;
|
|
|
|
emit$colon: procedure(b);
|
|
declare b byte;
|
|
call emit$bcd$pair(b);
|
|
call emitchar(':');
|
|
end emit$colon;
|
|
|
|
emit$bin$pair: procedure(b);
|
|
declare b byte;
|
|
b = b mod 100; /* [JCE] Year 2000 fix */
|
|
call emit$bcd(b/10); /* makes garbage if not < 10 */
|
|
call emit$bcd(b mod 10);
|
|
end emit$bin$pair;
|
|
|
|
emit$slant: procedure(b);
|
|
declare b byte;
|
|
call emit$bin$pair(b);
|
|
call emitchar('/');
|
|
end emit$slant;
|
|
|
|
emit$dash: procedure(b); /* [JCE] for YMD format dates */
|
|
declare b byte;
|
|
call emit$bin$pair(b);
|
|
call emitchar('-');
|
|
end emit$dash;
|
|
|
|
declare chr byte;
|
|
|
|
gnc: procedure;
|
|
/* get next command byte */
|
|
if chr = 0 then return;
|
|
if index = 20 then
|
|
do;
|
|
chr = 0;
|
|
return;
|
|
end;
|
|
chr = string(index := index + 1);
|
|
end gnc;
|
|
|
|
deblank: procedure;
|
|
do while chr = ' ';
|
|
call gnc;
|
|
end;
|
|
end deblank;
|
|
|
|
numeric: procedure byte;
|
|
/* test for numeric */
|
|
return (chr - '0') < 10;
|
|
end numeric;
|
|
|
|
scan$numeric: procedure(lb,ub) byte;
|
|
declare (lb,ub) byte;
|
|
declare b byte;
|
|
b = 0;
|
|
call deblank;
|
|
if not numeric then call terminate;
|
|
do while numeric;
|
|
if (b and 1110$0000b) <> 0 then call terminate;
|
|
b = shl(b,3) + shl(b,1); /* b = b * 10 */
|
|
if carry then call terminate;
|
|
b = b + (chr - '0');
|
|
if carry then call terminate;
|
|
call gnc;
|
|
end;
|
|
if (b < lb) or (b > ub) then call terminate;
|
|
return b;
|
|
end scan$numeric;
|
|
|
|
scan$delimiter: procedure(d,lb,ub) byte;
|
|
declare (d,lb,ub) byte;
|
|
call deblank;
|
|
if chr <> d then call terminate;
|
|
call gnc;
|
|
return scan$numeric(lb,ub);
|
|
end scan$delimiter;
|
|
|
|
declare
|
|
base$year lit '78', /* base year for computations */
|
|
base$day lit '0', /* starting day for base$year 0..6 */
|
|
month$size (*) byte data
|
|
/* jan feb mar apr may jun jul aug sep oct nov dec */
|
|
( 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31),
|
|
month$days (*) address data
|
|
/* jan feb mar apr may jun jul aug sep oct nov dec */
|
|
( 000,031,059,090,120,151,181,212,243,273,304,334);
|
|
|
|
leap$days: procedure(y,m) byte;
|
|
declare (y,m) byte;
|
|
/* compute days accumulated by leap years */
|
|
declare yp byte;
|
|
yp = shr(y,2); /* yp = y/4 */
|
|
if (y and 11b) = 0 and month$days(m) < 59 then
|
|
/* y not 00, y mod 4 = 0, before march, so not leap yr */
|
|
return yp - 1;
|
|
/* otherwise, yp is the number of accumulated leap days */
|
|
return yp;
|
|
end leap$days;
|
|
|
|
declare word$value address;
|
|
|
|
bcd:
|
|
procedure (val) byte;
|
|
declare val byte;
|
|
return shl((val/10),4) + val mod 10;
|
|
end bcd;
|
|
|
|
declare (month, day, year, hrs, min, sec) byte;
|
|
|
|
set$date$time: procedure;
|
|
declare
|
|
(i, leap$flag) byte; /* temporaries */
|
|
month = scan$numeric(1,12) - 1;
|
|
/* may be feb 29 */
|
|
if (leap$flag := month = 1) then i = 29;
|
|
else i = month$size(month);
|
|
day = scan$delimiter('/',1,i);
|
|
year = scan$delimiter('/',base$year,99);
|
|
/* ensure that feb 29 is in a leap year */
|
|
if leap$flag and day = 29 and (year and 11b) <> 0 then
|
|
/* feb 29 of non-leap year */ call terminate;
|
|
/* compute total days */
|
|
tod.date = month$days(month)
|
|
+ 365 * (year - base$year)
|
|
+ day
|
|
- leap$days(base$year,0)
|
|
+ leap$days(year,month);
|
|
|
|
tod.hrs = bcd (scan$numeric(0,23));
|
|
tod.min = bcd (scan$delimiter(':',0,59));
|
|
if tod.opcode = 2 then
|
|
/* date, hours and minutes only */
|
|
do;
|
|
if chr = ':'
|
|
then i = scan$delimiter (':',0,59);
|
|
tod.sec = 0;
|
|
end;
|
|
/* include seconds */
|
|
else tod.sec = bcd (scan$delimiter(':',0,59));
|
|
|
|
end set$date$time;
|
|
|
|
bcd$pair: procedure(a,b) byte;
|
|
declare (a,b) byte;
|
|
return shl(a,4) or b;
|
|
end bcd$pair;
|
|
|
|
|
|
compute$year: procedure;
|
|
/* compute year from number of days in word$value */
|
|
declare year$length address;
|
|
year = base$year;
|
|
do forever;
|
|
year$length = 365;
|
|
if (year and 11b) = 0 then /* leap year */
|
|
year$length = 366;
|
|
if word$value <= year$length then
|
|
return;
|
|
word$value = word$value - year$length;
|
|
year = year + 1;
|
|
end;
|
|
end compute$year;
|
|
|
|
declare
|
|
week$day byte, /* day of week 0 ... 6 */
|
|
day$list (*) byte data
|
|
('Sun$Mon$Tue$Wed$Thu$Fri$Sat$'),
|
|
leap$bias byte; /* bias for feb 29 */
|
|
|
|
compute$month: procedure;
|
|
month = 12;
|
|
do while month > 0;
|
|
if (month := month - 1) < 2 then /* jan or feb */
|
|
leapbias = 0;
|
|
if month$days(month) + leap$bias < word$value then return;
|
|
end;
|
|
end compute$month;
|
|
|
|
get$date$time: procedure;
|
|
/* get date and time */
|
|
hrs = tod.hrs;
|
|
min = tod.min;
|
|
sec = tod.sec;
|
|
word$value = tod.date;
|
|
/* word$value contains total number of days */
|
|
week$day = (word$value + base$day - 1) mod 7;
|
|
call compute$year;
|
|
/* year has been set, word$value is remainder */
|
|
leap$bias = 0;
|
|
if (year and 11b) = 0 and word$value > 59 then
|
|
/* after feb 29 on leap year */ leap$bias = 1;
|
|
call compute$month;
|
|
day = word$value - (month$days(month) + leap$bias);
|
|
month = month + 1;
|
|
end get$date$time;
|
|
|
|
emit$date$time: procedure;
|
|
|
|
if tod.opcode = 0 then
|
|
do;
|
|
call emitn(.day$list(shl(week$day,2)));
|
|
call emitchar(' ');
|
|
end;
|
|
if getukdate = 0 then /* [JCE] Vary the date format */
|
|
do;
|
|
call emit$slant(month);
|
|
call emit$slant(day);
|
|
call emit$bin$pair(year);
|
|
end;
|
|
else if getukdate = 1 then
|
|
do;
|
|
call emit$slant(day);
|
|
call emit$slant(month);
|
|
call emit$bin$pair(year);
|
|
end;
|
|
else
|
|
do;
|
|
call emit$dash(year);
|
|
call emit$dash(month);
|
|
call emit$bin$pair(day); /* [JCE] ends */
|
|
end;
|
|
call emitchar(' ');
|
|
call emit$colon(hrs);
|
|
call emit$colon(min);
|
|
if tod.opcode = 0 then
|
|
call emit$bcd$pair(sec);
|
|
end emit$date$time;
|
|
|
|
tod$ASCII:
|
|
procedure (parameter);
|
|
declare parameter address;
|
|
declare ret address;
|
|
|
|
ret = 0;
|
|
tod$adr = parameter;
|
|
string$adr = .tod.ASCII;
|
|
if (tod.opcode = 0) or
|
|
(tod.opcode = 3) then
|
|
do;
|
|
call get$date$time;
|
|
index = -1;
|
|
call emit$date$time;
|
|
end;
|
|
else
|
|
do;
|
|
if (tod.opcode = 1) or
|
|
(tod.opcode = 2) then
|
|
do;
|
|
chr = string(index:=0);
|
|
call set$date$time;
|
|
ret = .string(index);
|
|
end;
|
|
else
|
|
do;
|
|
call terminate;
|
|
end;
|
|
end;
|
|
end tod$ASCII;
|
|
|
|
/********************************************************
|
|
|
|
|
|
TOD INTERFACE TO SHOW
|
|
|
|
|
|
********************************************************/
|
|
|
|
|
|
declare lcltod structure (
|
|
opcode byte,
|
|
date address,
|
|
hrs byte,
|
|
min byte,
|
|
sec byte,
|
|
ASCII (21) byte );
|
|
|
|
/* declare extrnl$todadr address;
|
|
declare extrnl$tod based extrnl$todadr structure (
|
|
date address,
|
|
hrs byte,
|
|
min byte,
|
|
sec byte );
|
|
*/
|
|
|
|
declare ret address;
|
|
|
|
/* display$tod:
|
|
procedure;
|
|
lcltod.opcode = 0;
|
|
call move (5,.extrnl$tod.date,.lcltod.date);
|
|
call tod$ASCII (.lcltod);
|
|
call write$console (0dh);
|
|
do i = 0 to 20;
|
|
call write$console (lcltod.ASCII(i));
|
|
end;
|
|
end display$tod; */
|
|
|
|
display$ts:
|
|
procedure (tsadr);
|
|
dcl i byte;
|
|
dcl tsadr address;
|
|
lcltod.opcode = 3; /* display time and date stamp, no seconds */
|
|
call move (4,tsadr,.lcltod.date); /* don't copy seconds */
|
|
call tod$ASCII (.lcltod);
|
|
do i = 0 to 13;
|
|
call printchar (lcltod.ASCII(i));
|
|
end;
|
|
end display$ts;
|
|
|
|
/******** End TOD Code ********/
|
|
|
|
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * BASIC ROUTINES * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
declare
|
|
fcbmax literally '512'; /* max fcb count */
|
|
|
|
declare bpb address; /* bytes per block */
|
|
|
|
|
|
set$bpb: procedure;
|
|
|
|
call set$dpb; /* disk parameters set */
|
|
bpb = shl(double(1),blkshf) * sectorlen;
|
|
|
|
end set$bpb;
|
|
|
|
|
|
select$disk: procedure(d);
|
|
declare d byte;
|
|
/* select disk and set bpb */
|
|
call select(cdisk:=d);
|
|
call set$bpb; /* bytes per block */
|
|
|
|
end select$disk;
|
|
|
|
|
|
getalloc: procedure(i) byte; /* return the ith bit of the alloc vector */
|
|
declare i address;
|
|
|
|
return
|
|
rol(alloc(shr(i,3)), (i and 111b) + 1);
|
|
end getalloc;
|
|
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
|
|
/* fill string @ s for c bytes with f */
|
|
fill: proc(s,f,c);
|
|
dcl s addr,
|
|
(f,c) byte,
|
|
a based s byte;
|
|
|
|
do while (c:=c-1)<>255;
|
|
a = f;
|
|
s = s+1;
|
|
end;
|
|
end fill;
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * PRINT A NUMBER * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
declare
|
|
val (7) byte initial(0,0,0,0,0,0,0), /* BCD digits */
|
|
fac (7) byte initial(0,0,0,0,0,0,0), /* hibyte factor */
|
|
f0 (7) byte initial(6,3,5,5,6,0,0), /* 65,536 */
|
|
f1 (7) byte initial(2,7,0,1,3,1,0), /* 131,072 */
|
|
f2 (7) byte initial(4,4,1,2,6,2,0), /* 262,144 */
|
|
f3 (7) byte initial(8,8,2,4,2,5,0), /* 524,288 */
|
|
f4 (7) byte initial(6,7,5,8,4,0,1), /* 1,048,576 */
|
|
f5 (7) byte initial(2,5,1,7,9,0,2), /* 2,097,152 */
|
|
f6 (7) byte initial(4,0,3,4,9,1,4), /* 4,194,304 */
|
|
ptr (7) address initial(.f0,.f1,.f2,.f3,.f4,.f5,.f6);
|
|
|
|
|
|
|
|
/* print decimal value of address v */
|
|
pdecimal: procedure(v,prec,zerosup);
|
|
/* print value v with precision prec (1,10,100,1000,10000)
|
|
with leading zero suppression if zerosup = true */
|
|
declare
|
|
v address, /* value to print */
|
|
prec address, /* precision */
|
|
zerosup byte, /* zero suppression flag */
|
|
d byte; /* current decimal digit */
|
|
|
|
do while prec <> 0;
|
|
d = v / prec; /* get next digit */
|
|
v = v mod prec; /* get remainder back to v */
|
|
prec = prec/10; /* ready for next digit */
|
|
|
|
if prec = 0 then go to pd0;
|
|
if d <> 0 then go to pd0;
|
|
if zerosup then do;
|
|
call printb;
|
|
go to pd1;
|
|
end;
|
|
pd0: zerosup = false;
|
|
call printchar('0'+d);
|
|
pd1: end;
|
|
|
|
end pdecimal;
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
|
|
/* BCD - convert 16 bit binary to
|
|
7 one byte BCD digits */
|
|
getbcd: procedure(value);
|
|
declare
|
|
(value,prec) address,
|
|
i byte;
|
|
|
|
prec = 10000;
|
|
i = 5; /* digits: 4,3,2,1,0 */
|
|
do while prec <> 0;
|
|
val(i:=i-1) = value / prec; /* get next digit */
|
|
value = value mod prec; /* remainder in value */
|
|
prec = prec / 10;
|
|
end;
|
|
end getbcd;
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
|
|
/* print BCD number in val array */
|
|
printbcd: procedure;
|
|
declare
|
|
(zerosup, i) byte;
|
|
|
|
pchar: procedure(c);
|
|
declare c byte;
|
|
if val(i) = 0 then
|
|
if zerosup then
|
|
if i <> 0 then do;
|
|
call printb;
|
|
return;
|
|
end;
|
|
/* else */
|
|
call printchar(c);
|
|
zerosup = false;
|
|
end pchar;
|
|
|
|
zerosup = true;
|
|
i = 7;
|
|
do while (i:=i-1) <> -1;
|
|
call pchar('0'+val(i));
|
|
if i = 6 or i = 3 then
|
|
call pchar(',');
|
|
end;
|
|
end printbcd;
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
|
|
/* add two BCD numbers result in second */
|
|
add: procedure(ap,bp);
|
|
declare
|
|
(ap,bp) address,
|
|
a based ap (7) byte,
|
|
b based bp (7) byte,
|
|
(c,i) byte;
|
|
|
|
c = 0; /* carry */
|
|
do i = 0 to 6; /* 0 = LSB */
|
|
b(i) = a(i) + b(i) + c;
|
|
c = b(i) / 10;
|
|
b(i) = b(i) mod 10;
|
|
end;
|
|
end add;
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
|
|
/* print 3 byte value based at byte3adr */
|
|
p3byte: procedure(byte3adr);
|
|
declare
|
|
i byte,
|
|
high$byte byte,
|
|
byte3adr address,
|
|
b3 based byte3adr structure (
|
|
lword address,
|
|
hbyte byte);
|
|
|
|
call fill(.val,0,7);
|
|
call fill(.fac,0,7);
|
|
call getbcd(b3.lword); /* put 16 bit value in val */
|
|
high$byte = b3.hbyte;
|
|
do i = 0 to 6; /* factor for bit i */
|
|
if high$byte then /* LSB is 1 */
|
|
call add(ptr(i),.fac); /* add in factor */
|
|
high$byte = shr(high$byte,1); /* get next bit */
|
|
end;
|
|
call add(.fac,.val); /* add factor to value */
|
|
call printbcd; /* print value */
|
|
end p3byte;
|
|
|
|
|
|
/* divide 3 byte value by 8 */
|
|
shr3byte: procedure(byte3adr);
|
|
dcl byte3adr address,
|
|
b3 based byte3adr structure (
|
|
lword address,
|
|
hbyte byte),
|
|
temp1 based byte3adr (2) byte,
|
|
temp2 byte;
|
|
|
|
temp2 = ror(b3.hbyte,3) and 11100000b; /* get 3 bits */
|
|
b3.hbyte = shr(b3.hbyte,3);
|
|
b3.lword = shr(b3.lword,3);
|
|
temp1(1) = temp1(1) or temp2; /* or in 3 bits from hbyte */
|
|
end shr3byte;
|
|
|
|
|
|
/* multiply 3 byte value by #records per block */
|
|
shl3byte: procedure(byte3adr);
|
|
dcl byte3adr address,
|
|
b3 based byte3adr structure (
|
|
lword address,
|
|
hbyte byte),
|
|
temp1 based byte3adr (2) byte;
|
|
|
|
b3.hbyte = (rol(temp1(1),blkshf) and blkmsk) or shl(b3.hbyte,blkshf);
|
|
b3.lword = shl(b3.lword,blkshf);
|
|
end shl3byte;
|
|
|
|
|
|
show$drive: procedure;
|
|
|
|
call printchar(cdisk+'A');
|
|
call printx(.(': ',0));
|
|
|
|
end show$drive;
|
|
|
|
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * CALCULATE SIZE * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
add$block: procedure(ak,ab);
|
|
declare (ak, ab) address;
|
|
/* add one block to the kilobyte accumulator */
|
|
declare kaccum based ak address; /* kilobyte accum */
|
|
declare baccum based ab address; /* byte accum */
|
|
baccum = baccum + bpb;
|
|
do while baccum >= 1024;
|
|
baccum = baccum - 1024;
|
|
kaccum = kaccum + 1;
|
|
end;
|
|
end add$block;
|
|
|
|
count: procedure(mode) address;
|
|
declare mode byte; /* true if counting 0's */
|
|
/* count kb remaining, kaccum set upon exit */
|
|
declare
|
|
ka address, /* kb accumulator */
|
|
ba address, /* byte accumulator */
|
|
i address, /* local index */
|
|
bit byte; /* always 1 if mode = false */
|
|
ka, ba = 0;
|
|
bit = 0;
|
|
do i = 0 to maxall;
|
|
if mode then bit = getalloc(i);
|
|
if not bit then call add$block(.ka,.ba);
|
|
end;
|
|
return ka;
|
|
end count;
|
|
|
|
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * STATUS ROUTINES * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
|
|
/* characteristics of current drive */
|
|
drivestatus: procedure;
|
|
dcl b3a address,
|
|
b3 based b3a structure (
|
|
lword address,
|
|
hbyte byte),
|
|
|
|
psize address;
|
|
|
|
|
|
/* print 3 byte value */
|
|
pv3: procedure;
|
|
call crlf;
|
|
call p3byte(.dirbuf);
|
|
call printchar(':');
|
|
call printb;
|
|
end pv3;
|
|
|
|
/* print address value v */
|
|
pv: procedure(v);
|
|
dcl v address;
|
|
b3.hbyte = 0;
|
|
b3.lword = v;
|
|
call pv3;
|
|
end pv;
|
|
|
|
/* print the characteristics of the currently selected drive */
|
|
|
|
b3a = .dirbuf;
|
|
call print(.(' ',0));
|
|
call show$drive;
|
|
call printx(.('Drive Characteristics',0));
|
|
b3.hbyte = 0;
|
|
b3.lword = maxall + 1; /* = # blocks */
|
|
call shl3byte(.dirbuf); /* # blocks * records/block */
|
|
call pv3;
|
|
call printx(.('128 Byte Record Capacity',0));
|
|
call shr3byte(.dirbuf); /* divide by 8 */
|
|
call pv3;
|
|
call printx(.('Kilobyte Drive Capacity',0));
|
|
call pv(dirmax+1);
|
|
call printx(.('32 Byte Directory Entries',0));
|
|
call pv(shl(chksiz,2));
|
|
call printx(.('Checked Directory Entries',0));
|
|
call pv((extmsk+1) * 128);
|
|
call printx(.('Records / Directory Entry',0));
|
|
call pv(shl(double(1),blkshf));
|
|
call printx(.('Records / Block',0));
|
|
call pv(scptrk);
|
|
call printx(.('Records / Track',0)); /* [JCE] Saying "Sectors" is */
|
|
call pv(offset); /* misleading if sector size */
|
|
call printx(.('Reserved Tracks',0)); /* is >128 bytes */
|
|
|
|
psize = 128; /* 2**psh * 128 */
|
|
if physhf > 0 then psize = shl(psize,physhf);
|
|
|
|
call pv(psize);
|
|
call printx(.('Bytes / Physical Record',0));
|
|
call crlf;
|
|
|
|
end drivestatus;
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * DISK STATUS * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
pvalue: procedure(v);
|
|
declare (d,zero) byte,
|
|
(k,v) address;
|
|
k = 10000;
|
|
zero = false;
|
|
do while k <> 0;
|
|
d = low(v/k); v = v mod k;
|
|
k = k / 10;
|
|
if zero or k = 0 or d <> 0 then
|
|
do; zero = true; call printchar('0'+d);
|
|
end;
|
|
end;
|
|
end pvalue;
|
|
|
|
|
|
prcount: procedure;
|
|
|
|
/* print the actual byte count */
|
|
if cversion < mpm then do;
|
|
alloca = getalloca;
|
|
call pvalue(count(true));
|
|
end;
|
|
else do;
|
|
call setdma(.dirbuf);
|
|
call getfreesp(cdisk);
|
|
call shr3byte(.dirbuf);
|
|
call p3byte(.dirbuf);
|
|
end;
|
|
call printchar('k');
|
|
end prcount;
|
|
|
|
stat: procedure(rodisk);
|
|
declare rodisk address;
|
|
|
|
call crlf;
|
|
call show$drive;
|
|
call printchar('R');
|
|
if low(rodisk) then
|
|
call printchar('O'); else
|
|
call printchar('W');
|
|
call printx(.(', Space: ',0));
|
|
call prcount;
|
|
end stat;
|
|
|
|
prstatus: procedure; /* print the status of the disk system */
|
|
declare (login, rodisk) address;
|
|
declare (d,save) byte;
|
|
|
|
if once$only then return; /* only execute this once if
|
|
all was specified > 1 */
|
|
|
|
save = cdisk;
|
|
login = getlogin; /* login vector set */
|
|
rodisk = getrodisk; /* read only disk vector set */
|
|
|
|
d = 0;
|
|
do while login <> 0;
|
|
if low(login) then do;
|
|
if not all then do; /* do specified disk */
|
|
if d = save then call stat(rodisk);
|
|
end;
|
|
|
|
else do;
|
|
call select$disk(d); /* do all disks */
|
|
call stat(rodisk);
|
|
end;
|
|
end;
|
|
|
|
login = shr(login,1); rodisk = shr(rodisk,1);
|
|
d = d + 1;
|
|
end;
|
|
|
|
if all then once$only = true;
|
|
call crlf;
|
|
|
|
end prstatus;
|
|
|
|
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * USER STATUS * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
prdir: procedure;
|
|
|
|
call crlf;
|
|
call crlf;
|
|
call show$drive;
|
|
|
|
if nSFCB > 0 then do;
|
|
call printx(.('Number of time/date directory entries: ',0));
|
|
call pdecimal(nSFCB,1000,true);
|
|
call crlf;
|
|
call show$drive;
|
|
end;
|
|
|
|
call printx(.('Number of free directory entries: ',0));
|
|
call pdecimal(free$dir,1000,true);
|
|
call crlf;
|
|
|
|
end prdir;
|
|
|
|
|
|
get$usr$files: procedure;
|
|
declare ufcb(*) byte data ('????????????',0,0,0),
|
|
(i,j) byte,
|
|
nfcbs address,
|
|
extptr address,
|
|
modptr address,
|
|
fmod based modptr byte,
|
|
fext based extptr byte;
|
|
|
|
do i = 0 to 15;
|
|
user(i),used(i) = 0;
|
|
end;
|
|
nSFCB = 0;
|
|
|
|
call setdma(.dirbuf);
|
|
call search(.ufcb);
|
|
nfcbs = 0;
|
|
|
|
do while dcnt <> 255;
|
|
j = shl(dcnt,5); /* which fcb in dirbuf */
|
|
|
|
ge0: if (i := dirbuf(j)) <> 0e5h then do;
|
|
if i <> 33 then do; /* SFCB ? */
|
|
extptr = .dirbuf(j + 12);
|
|
modptr = extptr + 2;
|
|
nfcbs = nfcbs + 1;
|
|
j = i; /* save for xfcb test */
|
|
user(i := i and 0fh) = true;
|
|
|
|
if j > 15 then go to ge2;
|
|
if fext > extmsk then go to ge2;
|
|
if fmod = 0 then used(i) = used(i) + 1;
|
|
end;
|
|
else nSFCB = nSFCB + 1;
|
|
end;
|
|
|
|
ge2: call searchn;
|
|
end;
|
|
|
|
done$drive(cdisk) = true;
|
|
if nSFCB > 0 then nSFCB = shr(dirmax+1,2); /* because search ends
|
|
at high water mark*/
|
|
free$dir = ((dirmax + 1) - nSFCB) - nfcbs;
|
|
|
|
end get$usr$files;
|
|
|
|
|
|
userstatus: procedure; /* display active user numbers */
|
|
declare i byte;
|
|
|
|
call crlf;
|
|
call show$drive;
|
|
call printx(.('Active User :',0,0)); /* [JCE] Patch 16 */
|
|
call pdecimal(getuser,1000,true);
|
|
call crlf;
|
|
call show$drive;
|
|
call printx(.('Active Files:',0,0)); /* [JCE] Patch 16 */
|
|
|
|
if not done$drive(cdisk) then call get$usr$files;
|
|
|
|
do i = 0 to last(user);
|
|
if user(i) then call pdecimal(i,1000,true);
|
|
end;
|
|
|
|
call crlf;
|
|
call show$drive;
|
|
call printx(.('# of files :',0,0)); /* [JCE] Patch 16 */
|
|
do i = 0 to last(user);
|
|
if user(i) then call pdecimal(used(i),1000,true);
|
|
end;
|
|
|
|
call prdir;
|
|
|
|
end userstatus;
|
|
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * DISK & FILE STATUS * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
|
|
directory: procedure;
|
|
|
|
if not done$drive(cdisk) then call get$usr$files;
|
|
call prdir;
|
|
|
|
end directory;
|
|
|
|
/*******************************************************
|
|
|
|
L A B E L S T A T U S
|
|
|
|
********************************************************/
|
|
|
|
readlbl: proc(relog);
|
|
declare relog byte,
|
|
d byte data('?');
|
|
|
|
call setdma(.dirbuf);
|
|
call search(.d);
|
|
if relog > 0 then return;
|
|
|
|
do while dcnt <> 0ffH;
|
|
if dirbuf(ror(dcnt,3) and 110$0000b)=20H then return;
|
|
call searchn;
|
|
end;
|
|
|
|
end readlbl;
|
|
|
|
/* HEADER */
|
|
|
|
dcl label1 (*) byte data (
|
|
'Directory Passwds Stamp Stamp',0);
|
|
dcl label2 (*) byte data (
|
|
'Label Reqd ',0);
|
|
dcl label3 (*) byte data (
|
|
' Update Label Created Label Updated',0)
|
|
|
|
;
|
|
dcl label4 (*) byte data (
|
|
'------------ ------- ------ ------ -------------- --------------',0
|
|
|
|
);
|
|
|
|
|
|
labelstatus: procedure;
|
|
dcl lbl byte;
|
|
dcl fnam lit '11';
|
|
dcl ftyp lit '9';
|
|
dcl fcbp address;
|
|
dcl fcbv based fcbp (32) byte; /* template over dirbuf */
|
|
|
|
printfn: proc; /* print file name */
|
|
declare k byte;
|
|
|
|
do k = 1 to fnam;
|
|
if k = ftyp then
|
|
call printchar('.');
|
|
call printchar(fcbv(k) and 7fh);
|
|
end;
|
|
end printfn;
|
|
|
|
|
|
lbl = getlbl(cdisk);
|
|
if lbl > 0 then do;
|
|
call readlbl(0);
|
|
fcbp = shl(dcnt,5) + .dirbuf;
|
|
|
|
call print(.('Label for drive ',0)); /* print heading */
|
|
call show$drive;
|
|
call crlf;
|
|
call print(.label1);
|
|
call print(.label2);
|
|
if (lbl and 40h) = 40h then
|
|
call printx(.('Access',0));
|
|
else
|
|
call printx(.('Create',0));
|
|
call printx(.label3);
|
|
call print(.label4);
|
|
call crlf;
|
|
call printfn;
|
|
if not NONBANK and ((lbl and 80h) = 80h) then
|
|
call printx(.(' on ',0));
|
|
else
|
|
call printx(.(' off ',0));
|
|
|
|
if (lbl and 40h) = 40h then
|
|
call printx(.(' on ',0));
|
|
else if(lbl and 10h) = 10h then
|
|
call printx(.(' on ',0));
|
|
else call printx(.(' off ',0));
|
|
|
|
if (lbl and 20h) = 20h then
|
|
call printx(.(' on ',0));
|
|
else
|
|
call printx(.(' off',0));
|
|
|
|
call printx(.(' ',0));
|
|
call display$ts(.fcbv(24));
|
|
call printx(.(' ',0));
|
|
call display$ts(.fcbv(28));
|
|
end;
|
|
else do;
|
|
call e$print(.err$nolabel);
|
|
call printchar(cdisk+'A');
|
|
end;
|
|
|
|
call crlf;
|
|
|
|
end labelstatus;
|
|
|
|
|
|
$eject
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * Option scanner * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
separator: procedure(character) byte;
|
|
|
|
/* determines if character is a
|
|
delimiter and which one */
|
|
declare k byte,
|
|
character byte;
|
|
|
|
k = 1;
|
|
loop: if delimiters(k) = end$list then return(0);
|
|
if delimiters(k) = character then return(k); /* null = 25 */
|
|
k = k + 1;
|
|
go to loop;
|
|
|
|
end separator;
|
|
|
|
opt$scanner: procedure(list$ptr,off$ptr) byte;
|
|
/* scans the list pointed at by idxptr
|
|
for any strings that are in the
|
|
list pointed at by list$ptr.
|
|
Offptr points at an array that
|
|
contains the indices for the known
|
|
list. Idxptr points at the index
|
|
into the list. If the input string
|
|
is unrecognizable then the index is
|
|
0, otherwise > 0.
|
|
|
|
First, find the string in the known
|
|
list that starts with the same first
|
|
character. Compare up until the next
|
|
delimiter on the input. if every input
|
|
character matches then check for
|
|
uniqueness. Otherwise try to find
|
|
another known string that has its first
|
|
character match, and repeat. If none
|
|
can be found then return invalid.
|
|
|
|
To test for uniqueness, start at the
|
|
next string in the knwon list and try
|
|
to get another match with the input.
|
|
If there is a match then return invalid.
|
|
|
|
else move pointer past delimiter and
|
|
return.
|
|
|
|
P.Balma */
|
|
|
|
declare
|
|
buff based buf$ptr (1) byte,
|
|
off$ptr address,
|
|
list$ptr address;
|
|
|
|
declare
|
|
i byte,
|
|
j byte,
|
|
list based list$ptr (1) byte,
|
|
offsets based off$ptr (1) byte,
|
|
wrd$pos byte,
|
|
character byte,
|
|
letter$in$word byte,
|
|
found$first byte,
|
|
start byte,
|
|
index byte,
|
|
save$index byte,
|
|
(len$new,len$found) byte,
|
|
valid byte;
|
|
|
|
/*****************************************************************************/
|
|
/* internal subroutines */
|
|
/*****************************************************************************/
|
|
|
|
check$in$list: procedure;
|
|
/* find known string that has a match with
|
|
input on the first character. Set index
|
|
= invalid if none found. */
|
|
|
|
declare i byte;
|
|
|
|
i = start;
|
|
wrd$pos = offsets(i);
|
|
do while list(wrd$pos) <> end$list;
|
|
i = i + 1;
|
|
index = i;
|
|
if list(wrd$pos) = character then return;
|
|
wrd$pos = offsets(i);
|
|
end;
|
|
/* could not find character */
|
|
index = 0;
|
|
return;
|
|
end check$in$list;
|
|
|
|
setup: procedure;
|
|
character = buff(0);
|
|
call check$in$list;
|
|
letter$in$word = wrd$pos;
|
|
/* even though no match may have occurred, position
|
|
to next input character. */
|
|
i = 1;
|
|
character = buff(1);
|
|
end setup;
|
|
|
|
test$letter: procedure;
|
|
/* test each letter in input and known string */
|
|
|
|
letter$in$word = letter$in$word + 1;
|
|
|
|
/* too many chars input? 0 means
|
|
past end of known string */
|
|
if list(letter$in$word) = end$of$string then valid = false;
|
|
else
|
|
if list(letter$in$word) <> character then valid = false;
|
|
|
|
i = i + 1;
|
|
character = buff(i);
|
|
|
|
end test$letter;
|
|
|
|
skip: procedure;
|
|
/* scan past the offending string;
|
|
position buf$ptr to next string...
|
|
skip entire offending string;
|
|
ie., falseopt=mod, [note: comma or
|
|
space is considered to be group
|
|
delimiter] */
|
|
character = buff(i);
|
|
delimiter = separator(character);
|
|
do while ((delimiter <> 2) and (delimiter <> 4) and (delimiter <> 5)
|
|
and (delimiter <> 25));
|
|
i = i + 1;
|
|
character = buff(i);
|
|
delimiter = separator(character);
|
|
end;
|
|
endbuf = i;
|
|
buf$ptr = buf$ptr + endbuf + 1;
|
|
return;
|
|
end skip;
|
|
|
|
eat$blanks: procedure;
|
|
|
|
declare charac based buf$ptr byte;
|
|
|
|
|
|
do while(delimiter := separator(charac)) = SPACE;
|
|
bufptr = buf$ptr + 1;
|
|
end;
|
|
|
|
end eat$blanks;
|
|
|
|
/*****************************************************************************/
|
|
/* end of internals */
|
|
/*****************************************************************************/
|
|
|
|
|
|
/* start of procedure */
|
|
call eat$blanks;
|
|
start = 0;
|
|
call setup;
|
|
|
|
/* match each character with the option
|
|
for as many chars as input
|
|
Please note that due to the array
|
|
indices being relative to 0 and the
|
|
use of index both as a validity flag
|
|
and as a index into the option/mods
|
|
list, index is forced to be +1 as an
|
|
index into array and 0 as a flag*/
|
|
|
|
do while index <> 0;
|
|
start = index;
|
|
delimiter = separator(character);
|
|
|
|
/* check up to input delimiter */
|
|
|
|
valid = true; /* test$letter resets this */
|
|
do while delimiter = 0;
|
|
call test$letter;
|
|
if not valid then go to exit1;
|
|
delimiter = separator(character);
|
|
end;
|
|
|
|
go to good;
|
|
|
|
/* input ~= this known string;
|
|
get next known string that
|
|
matches */
|
|
exit1: call setup;
|
|
end;
|
|
/* fell through from above, did
|
|
not find a good match*/
|
|
endbuf = i; /* skip over string & return*/
|
|
call skip;
|
|
return(index);
|
|
|
|
/* is it a unique match in options
|
|
list? */
|
|
good: endbuf = i;
|
|
len$found = endbuf;
|
|
save$index = index;
|
|
valid = false;
|
|
next$opt:
|
|
start = index;
|
|
call setup;
|
|
if index = 0 then go to finished;
|
|
|
|
/* look at other options and check
|
|
uniqueness */
|
|
|
|
len$new = offsets(index + 1) - offsets(index) - 1;
|
|
if len$new = len$found then do;
|
|
valid = true;
|
|
do j = 1 to len$found;
|
|
call test$letter;
|
|
if not valid then go to next$opt;
|
|
end;
|
|
end;
|
|
else go to nextopt;
|
|
/* fell through...found another valid
|
|
match --> ambiguous reference */
|
|
index = 0;
|
|
call skip; /* skip input field to next delimiter*/
|
|
return(0);
|
|
|
|
finished: /* unambiguous reference */
|
|
index = save$index;
|
|
buf$ptr = buf$ptr + endbuf;
|
|
call eat$blanks;
|
|
if delimiter <> 0 then buf$ptr = buf$ptr + 1;
|
|
else delimiter = SPACE;
|
|
return(index);
|
|
|
|
end opt$scanner;
|
|
|
|
error$prt: procedure;
|
|
declare i byte,
|
|
t address,
|
|
char based t byte;
|
|
|
|
t = buf$ptr - endbuf - 1;
|
|
do i = 1 to endbuf;
|
|
call printchar(char);
|
|
t = t + 1;
|
|
end;
|
|
|
|
end error$prt;
|
|
|
|
$eject
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * EXECUTE * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
do$option: procedure(i);
|
|
declare i byte;
|
|
|
|
|
|
if opt$map(i).option(opt$space) <> 0 then call prstatus;
|
|
if opt$map(i).option(opt$label) <> 0 then call labelstatus;
|
|
if opt$map(i).option(opt$drive) <> 0 then call drivestatus;
|
|
if opt$map(i).option(opt$user) <> 0 then call userstatus;
|
|
if opt$map(i).option(opt$dir) <> 0 then call directory;
|
|
|
|
end do$option;
|
|
|
|
$eject
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
|
|
* * * PARSING * * *
|
|
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
declare character based buf$ptr byte;
|
|
|
|
setdef$drive: procedure;
|
|
|
|
if drive = 0ffh then do;
|
|
drive = cdisk;
|
|
drives(drive) = drive;
|
|
end;
|
|
|
|
return;
|
|
|
|
end setdef$drive;
|
|
|
|
|
|
parseoptions: procedure byte;
|
|
/* find all options within [...] */
|
|
|
|
buf$ptr = buf$ptr + 1;
|
|
delimiter = separator(character);
|
|
call setdef$drive;
|
|
|
|
if delimiter = 0 then go to preloop;
|
|
if delimiter <> RBRACKET then
|
|
if delimiter <> EOS then go to preloop;
|
|
|
|
/* [], turn on space */
|
|
opt$map(drive).option(opt$space) = 1;
|
|
buf$ptr = buf$ptr + 1;
|
|
return(2);
|
|
|
|
preloop:
|
|
if opt$map(drive).option(opt$space) = 0ffh then /* reset forced space*/
|
|
opt$map(drive).option(opt$space) = 0;
|
|
|
|
loop: if (opt$index := optscanner(.options,.off$opt)) = 0 then go to error;
|
|
|
|
if opt$index = opt$page then PAGE = true;
|
|
else if opt$index = opt$nopage then PAGE = false;
|
|
else opt$map(drive).option(opt$index - 1) = 1;
|
|
|
|
go to looptest;
|
|
|
|
error: call e$print(.err$unrecopt);
|
|
call print(.eoption);
|
|
call error$prt;
|
|
|
|
looptest:
|
|
if delimiter = EOS then return(25);
|
|
if delimiter = RBRACKET then return(2);
|
|
|
|
go to loop;
|
|
|
|
end parseoptions;
|
|
|
|
parsedir: procedure;
|
|
|
|
declare dirindex byte;
|
|
|
|
if (dir$index := optscanner(.dirs,.off$dirs)) = 0 then go to error1;
|
|
|
|
drive = dir$index - 1;
|
|
drives(drive) = drive;
|
|
opt$map(drive).option(opt$space) = 0ffh;/* only drive:,reset
|
|
if other options and
|
|
not space picked */
|
|
if delimiter <> COLON then buf$ptr = buf$ptr - 1;
|
|
|
|
return;
|
|
|
|
error1: call e$print(.err$unrecd);
|
|
dprint: call print(.dirdrive);
|
|
call error$prt;
|
|
call terminate;
|
|
|
|
end parsedir;
|
|
|
|
|
|
parser: procedure;
|
|
|
|
drive = 0ffh;
|
|
|
|
if (delimiter := separator(character)) = EOS then do;
|
|
call setdef$drive;
|
|
opt$map(drive).option(opt$space) = 1; /* default*/
|
|
all = true;
|
|
return;
|
|
end;
|
|
|
|
loop: if delimiter = LBRACKET then delimiter = parseoptions;
|
|
else if delimiter = 0 then call parsedir;
|
|
|
|
else do;
|
|
if delimiter <> COMMA then
|
|
if delimiter <> SPACE then go to error;
|
|
|
|
drive = 0ffh;
|
|
buf$ptr = buf$ptr + 1;
|
|
end;
|
|
|
|
|
|
looptest:
|
|
if delimiter <> EOS then
|
|
if (delimiter := separator(character)) <> EOS then go to loop;
|
|
|
|
return;
|
|
|
|
error: call e$print(.err$input);
|
|
call print(.input);
|
|
call error$prt;
|
|
call terminate;
|
|
|
|
end parser;
|
|
|
|
$eject
|
|
/*************************************************************************
|
|
|
|
|
|
*** MAIN PROGRAM ***
|
|
|
|
|
|
**************************************************************************/
|
|
|
|
declare
|
|
i byte initial(1);
|
|
|
|
plm:
|
|
cversion = get$version;
|
|
if cversion < mpm then call e$print(.err$version);
|
|
else do;
|
|
|
|
do while buff(i) = ' ';
|
|
i = i + 1;
|
|
end;
|
|
buf$ptr = .buff(i);
|
|
|
|
cdisk = cselect;
|
|
user$code = getuser;
|
|
|
|
do i = 0 to 15;
|
|
drives(i) = 0ffh;
|
|
end;
|
|
|
|
if getpagemode = 0 then PAGE = true;
|
|
line$page = getpage;
|
|
line$out = 0;
|
|
if getNB = 0 then NONBANK = true;
|
|
|
|
call parser;
|
|
|
|
do i = 0 to 15;
|
|
if (drive := drives(i)) <> 0ffh then do;
|
|
call select$disk(drives(i));
|
|
call readlbl(1); /* force login
|
|
by wild card drive
|
|
search. */
|
|
call do$option(i);
|
|
end;
|
|
end;
|
|
|
|
end;
|
|
call terminate;
|
|
|
|
end;
|