Files
Digital-Research-Source-Code/CPM OPERATING SYSTEMS/CPM 3.X/CPM 3.0/3.0 SOURCE/GENCPM.PLM
Sepp J Morris 31738079c4 Upload
Digital Research
2020-11-06 18:50:37 +01:00

1479 lines
40 KiB
Plaintext
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

$title('CP/M 3 System Generation')
gencpm:
do;
/*
Copyright (C) 1982
Digital Research
P.O. Box 579
Pacific Grove, CA 93950
*/
/*
Revised:
02 Dec 82 by Bruce Skidmore
*/
declare true literally '0FFH';
declare false literally '0';
declare forever literally 'while true';
declare boolean literally 'byte';
declare cr literally '0dh';
declare lf literally '0ah';
mon1:
procedure (func,info) external;
declare func byte;
declare info address;
end mon1;
mon2:
procedure (func,info) byte external;
declare func byte;
declare info address;
end mon2;
relfix:
procedure byte external;
end relfix;
setbuf:
procedure external;
end setbuf;
getdef:
procedure external;
end getdef;
crtdef:
procedure external;
end crtdef;
declare reset label external;
declare fcb (1) byte external;
declare fcb16 (1) byte external;
declare tbuff (1) byte external;
declare maxb address external;
declare bitmap (128) byte external;
declare FCBin address public;
declare bios$fcb (36) byte initial (
0,'BNKBIOS3','SPR',0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0);
declare res$fcb (36) byte initial (
0,'RESBDOS3','SPR',0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0);
declare bnk$fcb (36) byte initial (
0,'BNKBDOS3','SPR',0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0);
declare FCBout (36) byte initial (
0,'CPM3 ','SYS',0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0);
declare data$fcb (36) byte public initial (
0,'GENCPM ','DAT',0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0);
declare offset byte public;
declare prgsiz address public;
declare bufsiz address public;
declare codsiz address public;
declare bios$pg byte public;
declare scb$pg byte public;
declare res$pg byte public;
declare bnk$pg byte public;
declare bnk$off byte public;
declare res$len byte public;
declare non$bnk byte public;
declare dma address public;
declare hexASCII (16) byte public data (
'0123456789ABCDEF');
declare lnbfr (14) byte public initial (12);
declare sctbfr (1) structure (
record (128) byte) public at (.memory);
declare fcb$msg (13) byte initial (' . $');
declare query boolean public;
/*
B D O S P r o c e d u r e & F u n c t i o n C a l l s
*/
system$reset:
procedure public;
call mon1 (0,0);
end system$reset;
write$console:
procedure (char) public;
declare char byte;
if display then
call mon1 (2,char);
end write$console;
print$console$buffer:
procedure (buffer$address) public;
declare buffer$address address;
if display then
call mon1 (9,buffer$address);
end print$console$buffer;
read$console$buffer:
procedure (buffer$address) public;
declare buffer$address address;
declare buf based buffer$address (1) byte;
buf(1) = 0;
if automatic then
do;
if not query then
return;
end;
call mon1 (10,buffer$address);
buf(buf(1)+2) = 0;
end read$console$buffer;
crlf:
procedure public;
call write$console (cr);
call write$console (lf);
end crlf;
error:
procedure(term$code,err$type,err$msg$adr) public;
declare (term$code,err$type) byte;
declare err$msg$adr address;
display = true;
call print$console$buffer (.(cr,lf,
'ERROR: $'));
call print$console$buffer (err$msg$adr);
if err$type = 1 then
call print$console$buffer(.fcb$msg);
call crlf;
if term$code then
call system$reset;
if automatic and not query then
do;
fcb(1),
fcb16(1) = ' ';
goto reset;
end;
end error;
open$file:
procedure (fcb$address) byte public;
declare fcb$address address;
declare fcb based fcb$address (1) byte;
fcb(12), /* ex = 0 */
fcb(32) = 0; /* cr = 0 */
return mon2 (15,fcb$address);
end open$file;
close$file:
procedure (fcb$address) public;
declare fcb$address address;
call mon1 (16,fcb$address);
end close$file;
delete$file:
procedure (fcb$address) public;
declare fcb$address address;
call mon1 (19,fcb$address);
end delete$file;
read$record:
procedure (fcb$address) public;
declare fcb$address address;
if mon2 (20,fcb$address) <> 0 then
do;
call error(true,1,.(
'Reading file: $'));
end;
end read$record;
write$record:
procedure (fcb$address) public;
declare fcb$address address;
if mon2 (21,fcb$address) <> 0 then
do;
call error(true,1,.(
'Writing file: ','$'));
end;
end write$record;
create$file:
procedure (fcb$address) public;
declare fcb$address address;
declare fcb based fcb$address (1) byte;
if mon2 (22,fcb$address) = 255 then
do;
call error(true,0,.(
'Directory full','$'));
end;
fcb(32) = 0; /* set cr = 0 */
end create$file;
set$DMA$address:
procedure (DMA$address) public;
declare DMA$address address;
call mon1 (26,DMA$address);
end set$DMA$address;
read$random$record:
procedure (fcb$address) public;
declare fcb$address address;
if mon2 (33,fcb$address) <> 0 then
do;
call error(true,1,.(
'Reading file: ','$'));
end;
end read$random$record;
write$random$record:
procedure (fcb$address) public;
declare fcb$address address;
if mon2 (34,fcb$address) <> 0 then
do;
call error(true,1,.(
'Writing file: ','$'));
end;
end write$random$record;
set$random$record:
procedure (fcb$address) public;
declare fcb$address address;
call mon1 (36,fcb$address);
end set$random$record;
/*
D a t a S t r u c t u r e s
*/
declare automatic boolean;
declare display boolean public;
declare nmb$sect address;
declare link address at (.memory);
declare bios$atts(3) address public;
declare res$atts(3) address public;
declare bnk$atts(3) address public;
declare res$bios$len byte public;
declare res$base byte public;
declare pg$dif byte public;
declare xmove$implemented boolean public;
declare os$top address;
declare system$data (256) byte;
declare common$len byte public at (.system$data(1));
declare banked$len byte public at (.system$data(3));
declare sys$entry address public at (.system$data(4));
declare prt$msg$ptr byte;
declare dont$hash boolean;
declare wordadr address;
declare word based wordadr address;
declare len byte;
declare off address;
declare res$flg byte;
declare save$mem$top byte;
declare drvtbl$adr address public;
declare drvtbl based drvtbl$adr (16) address;
declare dph$adr address public;
declare dph based dph$adr structure (
xlt address,
scratch1(4) address,
scratch2 byte,
mf byte,
dpb address,
csv address,
alv address,
dirbcb address,
dtabcb address,
hash address,
hbank byte);
declare dpb$adr address public;
declare dpb based dpb$adr structure (
spt address,
bsh byte,
blm byte,
exm byte,
dsm address,
drm address,
al0 byte,
al1 byte,
cks address,
off address,
psh byte,
phm byte);
declare bnk$swt boolean external;
declare dbl$alv boolean external;
declare mem$top byte external;
declare bnk$top byte external;
declare lerror boolean external;
declare bdrive byte external;
declare con$wid byte external;
declare con$pag byte external;
declare bck$spc boolean external;
declare rubout boolean external;
declare prt$msg boolean external;
declare hash(16) boolean external;
declare num$seg byte external;
declare crdatf boolean external;
declare mem$tbl (17) structure(
base byte,
len byte,
bank byte,
attr address) external;
declare record(16) structure(
size address,
attr byte,
altbnks byte,
no$dirrecs byte,
no$dtarecs byte,
ovlydir$dr byte,
ovlydta$dr byte,
dir$resp byte,
dta$resp byte) external;
declare quest(157) boolean external;
declare hash$data(16) address public;
declare hash$space address public;
declare alloc(16) address public;
declare alloc$space address public;
declare chk(16) address public;
declare chk$space address public;
/*
L o c a l P r o c e d u r e s
*/
movef:
procedure (count,source$adr,dest$adr) public;
declare count byte;
declare (source$adr,dest$adr) address;
if count = 0
then return;
else call move (count,source$adr,dest$adr);
end movef;
shift$left:
procedure (pattern, count) address public;
declare count byte;
declare pattern address;
if count = 0
then return pattern;
else return shl(pattern,count);
end shift$left;
upper:
procedure(b) byte public;
declare b byte;
if b < ' ' then return cr; /* all non-graphics */
/* translate alpha to upper case */
if b >= 'a' and b <= 'z' then
b = b and 101$1111b; /* upper case */
return b;
end upper;
valid$drive:
procedure(drv) boolean public;
declare drv byte;
if (drv >= 0) and (drv <= 15) then
return true;
call error(false,0,.('Invalid drive.$'));
return false;
end valid$drive;
get$response:
procedure (val$adr) public;
declare val$adr address;
declare val based val$adr byte;
call write$console ('(');
if val = 0ffh
then call write$console ('Y');
else call write$console ('N');
call print$console$buffer (.(') ? ','$'));
call read$console$buffer (.lnbfr);
if lnbfr(1) = 0
then return; /* accept default */
val = (upper(lnbfr(2)) = 'Y');
end get$response;
dsply$hex:
procedure (val) public;
declare val byte;
call write$console (hexASCII(shr (val,4)));
call write$console (hexASCII(val and 0fh));
end dsply$hex;
dsply$hex$adr:
procedure (val) public;
declare val address;
call write$console (' ');
call dsply$hex (high (val));
call dsply$hex (low (val));
call write$console ('H');
end dsply$hex$adr;
dsply$hex$high$adr:
procedure (val) public;
declare val byte;
call dsply$hex$adr (double (val)*256);
end dsply$hex$high$adr;
dsply$dec$adr:
procedure (val) public;
declare val address;
declare big address;
declare (digit,i) byte;
declare pdigit boolean;
pdigit = false;
digit = '0';
big = 10000;
if val = 0 then
call write$console(digit);
else
do;
do i = 0 to 4;
do while val >= big;
pdigit = true;
digit = digit + 1;
val = val - big;
end;
if pdigit then
do;
call write$console(digit);
digit = '0';
end;
big = big / 10;
end;
end;
end dsply$dec$adr;
dsply$param:
procedure (val,base) public;
declare (val,base) byte;
call write$console ('(');
if base = 10 then
do;
call write$console ('#');
call dsply$dec$adr(double(val));
end;
else
do;
call dsply$hex (val);
end;
call print$console$buffer (.(') ? ','$'));
end dsply$param;
get$param:
procedure (string$adr,val$adr,pbase) public;
declare (string$adr,val$adr) address;
declare pbase byte;
declare base byte;
declare val based val$adr byte;
declare string based string$adr (1) byte;
declare char byte;
declare lbindx byte;
prompt$read:
procedure;
call print$console$buffer (string$adr);
if string(0) = ' ' then
do;
call write$console ('(');
call dsply$hex (val);
do lbindx = 1 to 2;
val$adr = val$adr + 1;
if (lbindx=2) and (not bnk$swt) then
do;
val = 0;
end;
else
do;
call write$console (',');
call dsply$hex (val);
end;
end;
val$adr = val$adr - 2;
call print$console$buffer (.(') ? ','$'));
end;
else
do;
call dsply$param (val,pbase);
end;
base = 16;
lbindx = 1;
call read$console$buffer (.lnbfr);
end prompt$read;
call prompt$read;
if lnbfr(1) = 0 then
do;
/* accept default value */
call crlf;
return;
end;
val = 0;
do while (char := upper(lnbfr(lbindx:=lbindx+1))) <> cr;
if char = ',' then
do;
val$adr = val$adr + 1;
val = 0;
base = 16;
end;
else
do;
if char = '#' then
do;
base = 10;
end;
else
do;
char = char - '0';
if (base = 16) and (char > 9) then
do;
if char > 16
then char = char - 7;
else char = 255;
end;
if char < base then
do;
val = val*base + char;
end;
else
do;
char,
val = 0;
call error (false,0,.(
'Bad character, re-enter $'));
call prompt$read;
val = 0;
end;
end;
end;
end;
call crlf;
end get$param;
get$seg:
procedure(type,record$size) byte public;
declare (type,k,seg$no) byte;
declare (record$size,max$attr) address;
if not bnk$swt then
return 0;
seg$no = 0ffh;
max$attr = 0ffffh;
do k = 1 to num$seg;
if mem$tbl(k).attr >= record$size then
if type = 1 then
do;
if (mem$tbl(k).bank = 0) and
(mem$tbl(k).attr < max$attr) then
do;
seg$no = k;
max$attr = mem$tbl(k).attr;
end;
end;
else
do;
if (mem$tbl(k).bank <> 0) and
(mem$tbl(k).attr < max$attr) then
do;
seg$no = k;
max$attr = mem$tbl(k).attr;
end;
end;
end;
if (seg$no = 0ffh) and (type = 2) then
do k = 1 to num$seg;
if (mem$tbl(k).attr >= record$size) and
(mem$tbl(k).bank = 0) and
(mem$tbl(k).attr < max$attr) then
do;
seg$no = k;
max$attr = mem$tbl(k).attr;
end;
end;
return seg$no;
end get$seg;
plm:
procedure public;
st$ascii$hex:
procedure(string$adr,val);
declare string$adr address;
declare string based string$adr (6) byte;
declare val address;
declare i byte;
string(0) = ' ';
string(1) = ' ';
string(2) = hexASCII(shr(high(val),4));
string(3) = hexASCII(high(val) and 0fh);
string(4) = hexASCII(shr(low(val),4));
string(5) = hexASCII(low(val) and 0fh);
end st$ascii$hex;
setup$scb:
procedure;
declare scb$adr address;
declare scb$dat based scb$adr (100) byte;
scb$adr = .memory + shl(double(scb$pg-res$pg),8) + 09ch;
scb$dat(13h) = bdrive;
scb$dat(1ah) = con$wid;
scb$dat(1ch) = con$pag;
scb$dat(2eh) = bck$spc;
scb$dat(2fh) = rubout;
call movef(5,.(012h,07h,0,0,0),.scb$dat(58h)); /* December 15, 1982 */
if not lerror then
scb$dat(57h) = scb$dat(57h) and 7fh;
if not dbl$alv and not bnk$swt then
scb$dat(57h) = scb$dat(57h) or 0100$0000B;
else
scb$dat(57h) = scb$dat(57h) and 1011$1111B;
scb$dat(5eh) = bnk$top;
end setup$scb;
get$drvtbl$adr:
procedure address;
declare temp$adr address;
declare temp2 based temp$adr address;
declare temp3 address;
temp$adr = .memory(43h);
temp3 = temp2 + 1 + .memory;
temp$adr = temp3;
if temp2 = 0fffeh
then res$flg = 2;
else res$flg = 0;
if temp2 < 0fffeh
then return temp2 + .memory;
else return 0ffffh;
end get$drvtbl$adr;
page$chop:
procedure;
declare i byte;
drvtbl$adr = get$drvtbl$adr;
dont$hash = true;
if (drvtbl$adr <> 0ffffh) then
do;
do i = 0 to 15;
if drvtbl(i) <> 0 then
do;
dph$adr = drvtbl(i) + .memory;
if dph.hash <> 0ffffh then
dont$hash = false;
end;
end;
if dont$hash and not bnk$swt then
res$flg = 2;
else
res$flg = 0;
end;
end page$chop;
get$xmove:
procedure boolean;
declare xmove$adr address;
declare xmove$val based xmove$adr byte;
call movef(2,.memory(58h),.xmove$adr);
xmove$adr = xmove$adr + .memory;
if xmove$val = 0c9h /* ret instr. */ then
return false;
else
return true;
end get$xmove;
display$layout:
procedure(string$adr,base,length);
declare string$adr address;
declare base address;
declare length byte;
call print$console$buffer (.(cr,lf,' ','$'));
call print$console$buffer (string$adr);
call write$console(' ');
call dsply$hex$adr (base);
call write$console(' ');
call dsply$hex$high$adr (length);
if prt$msg then
do;
call movef(12,string$adr,.system$data(prt$msg$ptr));
prt$msg$ptr = prt$msg$ptr + 12;
call st$ascii$hex(.system$data(prt$msg$ptr),base);
prt$msg$ptr = prt$msg$ptr + 6;
call st$ascii$hex(.system$data(prt$msg$ptr),
double(length)*256);
prt$msg$ptr = prt$msg$ptr + 6;
call movef(3,.(cr,lf,' '),.system$data(prt$msg$ptr));
prt$msg$ptr = prt$msg$ptr + 3;
end;
end display$layout;
reloc$module:
procedure (fcb$adr);
declare fcb$adr address;
FCBin = fcb$adr;
if relfix <> 0 then
do;
call error(true,1,.('Disk read error: $'));
end;
call close$file(fcb$adr);
end reloc$module;
load:
procedure (fcb$adr,atts$adr);
declare fcb$adr address;
declare atts$adr address;
declare atts based atts$adr (3) address;
declare (i,rdcnt) byte;
prgsiz = atts(0);
bufsiz = atts(1);
codsiz = atts(2);
call movef(8,fcb$adr+1,.fcb$msg);
call movef(3,fcb$adr+9,.fcb$msg+9);
if shr(prgsiz+255,7) > nmb$sect then
do;
call error(true,1,.('File cannot fit into GENCPM buffer: ','$'));
end;
rdcnt = low(shr(prgsiz-1,7)) + 1;
i = 0;
do while (i < rdcnt);
call set$dma$address(dma:=.sctbfr(i));
call read$record(fcb$adr);
i = i + 1;
end;
call movef(128,dma,.bitmap); /* copy the last sector read, into */
/* the bitmap buffer, relocation */
/* info might be that last sector */
dma = prgsiz + .memory;
end load;
wrtbuf:
procedure (wrtlen,wrtoff$adr);
declare (i,wrtlen,wrtcnt) byte;
declare wrtoff$adr address;
declare wrtoff based wrtoff$adr address;
if wrtlen <> 0 then
do;
call movef(8,.FCBout+1,.fcb$msg);
call movef(3,.FCBout+9,.fcb$msg+9);
FCBout(33) = low(wrtoff);
FCBout(34) = high(wrtoff);
call write$random$record(.FCBout);
dma = dma + low(256 - low(dma - .memory));
wrtcnt = wrtlen * 2 - 1;
do i = 0 to wrtcnt;
call set$dma$address(dma:=dma-80h);
call write$record(.FCBout);
end;
call set$random$record(.FCBout);
call movef(2,.FCBout(33),wrtoff$adr);
end;
end wrtbuf;
get$file$info:
procedure;
declare fcb$adr address;
declare atts$adr address;
declare file$atts based atts$adr(3) address;
declare header$record structure (
fill1 byte,
psize address,
fill2 byte,
dsize address,
fill3 (4) byte,
csize address,
fill4 (116) byte) at (.memory);
get$atts:
procedure;
call movef(8,fcb$adr+1,.fcb$msg);
call movef(3,fcb$adr+9,.fcb$msg+9);
if open$file(fcb$adr) = 0ffh
then call error(true,1,.('Unable to open: $'));
call set$dma$address(.header$record);
call read$record(fcb$adr);
file$atts(0) = header$record.psize;
file$atts(1) = header$record.dsize;
file$atts(2) = header$record.csize;
call read$record(fcb$adr);
end get$atts;
if not bnk$swt then
do;
call movef(8,.('BDOS3 '),.res$fcb+1);
call movef(8,.('BIOS3 '),.bios$fcb+1);
end;
else
do;
fcb$adr = .bnk$fcb;
atts$adr = .bnk$atts;
call get$atts;
end;
fcb$adr = .bios$fcb;
atts$adr = .bios$atts;
call get$atts;
fcb$adr = .res$fcb;
atts$adr = .res$atts;
call get$atts;
end get$file$info;
need$tbl:
procedure byte;
declare (all$some,i) byte;
all$some = false;
if drvtbl$adr = 0ffffh
then return false;
else
do i = 0 to 15;
if drvtbl(i) <> 0 then
do;
dph$adr = drvtbl(i) + .memory;
/* zero the reserved bytes in the DPH */
call movef(9,.(0,0,0,0,0,0,0,0,0),dph$adr+2);
if (dph.dirbcb = 0fffeh) or (dph.dtabcb = 0fffeh) or
(dph.hash = 0fffeh) or (dph.alv = 0fffeh) or
(dph.csv = 0fffeh)
then all$some = true;
end;
end;
return all$some;
end need$tbl;
setup$hash:
procedure;
declare (i,j,printed,seg$no,seg0$no,h$bank,hohash) byte;
declare (size,h$attr,max$attr,max0$attr) address;
declare nohash boolean;
printed = false;
nohash = true;
do i = 0 to 15;
dph$adr = drvtbl(i) + .memory;
if drvtbl(i) <> 0 then
do;
if dph.hash < 0fffeh then
nohash = false;
if dph.hash = 0fffeh then
do;
if not printed then
do;
printed = true;
call print$console$buffer(.
(lf,cr,'Setting up directory hash tables:',
lf,cr,'$'));
end;
query = quest(27 + i);
dpb$adr = dph.dpb + .memory;
size = shl(dpb.drm+1,2);
call print$console$buffer(.
(' Enable hashing for drive $'));
call write$console('A'+i);
call print$console$buffer(.(': $'));
call get$response(.hash(i));
call crlf;
if not hash(i) then
do;
dph.hash = 0ffffh;
end;
else
if not bnk$swt then
do;
nohash = false;
hash$data(i) = size;
hash$space = hash$space + size;
end;
else
do;
if (seg$no := get$seg(2,size)) = 0ffh then
call error(false,0,.(
'Unable to allocate space for hash table.$'));
else
do;
dph.hbank = mem$tbl(seg$no).bank;
dph.hash = shl(double(mem$tbl(seg$no).base),8) +
(shl(double(mem$tbl(seg$no).len),8) -
mem$tbl(seg$no).attr);
mem$tbl(seg$no).attr = mem$tbl(seg$no).attr - size;
end;
end;
end;
end;
end;
if (not bnk$swt) and (nohash) then
do;
res$flg = 2;
scb$pg = scb$pg + 2;
res$pg = res$pg + 2;
end;
end setup$hash;
get$alloc$chk:
procedure;
declare (i,dbl$alloc) byte;
declare printed boolean;
do i = 0 to 15;
alloc(i) = 0;
chk(i) = 0;
end;
if not dbl$alv and not bnk$swt then
dbl$alloc = 1;
else
dbl$alloc = 2;
alloc$space = 0;
chk$space = 0;
printed = false;
do i = 0 to 15;
if drvtbl(i) <> 0 then
do;
dph$adr = drvtbl(i) + .memory;
dpb$adr = dph.dpb + .memory;
if dph.alv = 0fffeh then
do;
call print$console$buffer(.(cr,lf,
'Setting up Allocation vector for drive $'));
call write$console('A'+i);
call write$console(':');
printed = true;
alloc(i) = (dpb.dsm/8 + 1) * dbl$alloc;
alloc$space = alloc$space + alloc(i);
end;
if dph.csv = 0fffeh then
do;
call print$console$buffer(.(cr,lf,
'Setting up Checksum vector for drive $'));
call write$console('A'+i);
call write$console(':');
printed = true;
chk(i) = (dpb.drm + 4)/4;
chk$space = chk$space + chk(i);
dpb.cks = (dpb.cks and 8000h) or chk(i);
end;
end;
end;
if printed then
call crlf;
end get$alloc$chk;
setup$mem$seg$tbl:
procedure;
declare (i,j,ok,accept,mlow,mhigh,tlow,thigh) byte;
declare mem$temp address;
/* Create first memory segment table entry */
mem$tbl(0).base = bnk$pg;
mem$tbl(0).len = bnk$top - bnk$pg;
mem$tbl(0).attr = 0;
mem$tbl(0).bank = 0;
accept = false;
call print$console$buffer(
.(lf,cr,
'*** Bank 1 and Common are not included ***',
lf,cr,
'*** in the memory segment table. ***',
lf,cr,lf,cr,'$'));
query = quest(10);
call get$param (.('Number of memory segments $'),
.num$seg,10);
call print$console$buffer(.(cr,lf,
'CP/M 3 Base,size,bank ($'));
call dsply$hex(mem$tbl(0).base);
call write$console(',');
call dsply$hex(mem$tbl(0).len);
call write$console(',');
call dsply$hex(mem$tbl(0).bank);
call print$console$buffer(.(')',lf,cr,'$'));
do while not accept;
/* Bank switched memory segment table input */
call print$console$buffer (.(cr,lf,
'Enter memory segment table:',lf,cr,'$'));
do j = 1 to num$seg;
ok = false;
do while not ok;
query = quest(11 + j - 1);
call get$param (.(' Base,size,bank ','$'),
.mem$tbl(j),16);
mem$tbl(j).attr = shl(double(mem$tbl(j).len),8);
if mem$tbl(j).len = 0 then
do;
call error(false,0,.(
'Zero length segment not allowed.$'));
end;
else
if mem$tbl(j).bank = 1 then
do;
call error(false,0,.(
'Bank one not allowed.$'));
end;
else
do;
tlow = mem$tbl(j).base;
mem$temp = double(tlow) + double(mem$tbl(j).len);
if (high(mem$temp) <> 0) or (low(mem$temp) > bnk$top) then
do;
call print$console$buffer(.(cr,lf,'ERROR: ',
'Memory conflict - segment trimmed.',
cr,lf,'$'));
mem$tbl(j).len = bnk$top - tlow;
mem$tbl(j).attr = shl(double(bnk$top - tlow),8);
end;
else
do;
thigh = low(mem$temp);
i = 0;
ok = true;
do while ((i < j) and ok);
mlow = mem$tbl(i).base;
mhigh = mlow + mem$tbl(i).len;
if mem$tbl(i).bank = mem$tbl(j).bank then
do;
if (mhigh >= thigh) and (tlow >= mlow) then
do;
call error(false,0,.(
'Memory conflict - cannot trim segment.$'));
ok = false;
end;
else
if ((thigh > mhigh) and (mhigh > tlow)) then
do;
call print$console$buffer(.(cr,lf,'ERROR: ',
'Memory conflict - segment trimmed.',
cr,lf,'$'));
mem$tbl(j).base = mhigh;
ok = false;
end;
else
if ((thigh > mlow) and (mlow > tlow)) then
do;
call print$console$buffer(.(cr,lf,'ERROR: ',
'Memory conflict - segment trimmed.',
cr,lf,'$'));
mem$tbl(j).len = mlow - tlow;
mem$tbl(j).attr = shl(double(mlow-tlow),8);
ok = false;
end;
end;
i = i + 1;
end;
end;
end;
end;
end;
call crlf;
do j = 0 to num$seg;
if j = 0 then
call print$console$buffer (.(' CP/M 3 Sys ','$'));
else
do;
call print$console$buffer (.(' Memseg No. ','$'));
call dsply$hex(j-1);
end;
call dsply$hex$high$adr (mem$tbl(j).base);
call dsply$hex$high$adr (mem$tbl(j).len);
if bnk$swt then
do;
call print$console$buffer (.(' Bank ','$'));
call dsply$hex (mem$tbl(j).bank);
end;
call crlf;
end;
query = false;
accept = true;
call print$console$buffer (.(cr,lf,
'Accept new memory segment table entries ','$'));
call get$response (.accept);
end; /* do while not accept */
call crlf;
end setup$mem$seg$tbl;
get$default$file:
procedure;
declare ret byte;
call print$console$buffer(.(
'Default entries are shown in (parens).',cr,lf,
'Default base is Hex, precede entry with # for decimal',
cr,lf,'$'));
if (ret:=open$file(.data$fcb)) <> 255 then
do;
call movef(8,.data$fcb+1,.fcb$msg);
call movef(3,.data$fcb+9,.fcb$msg+9);
call print$console$buffer(.(
cr,lf,'Use GENCPM.DAT for defaults $'));
ret = 0ffh;
call get$response(.ret);
call crlf;
if ret then
call getdef;
call close$file(.data$fcb);
end;
else
do;
display = true;
automatic = false;
end;
end get$default$file;
setup$system$dat:
procedure;
declare (i,j,ok,temp) byte;
ok = false;
call get$default$file;
do while not ok;
query = quest(155);
call crlf;
call print$console$buffer(.('Create a new GENCPM.DAT file $'));
call get$response(.crdatf);
query = quest(0);
call crlf;
call crlf;
call print$console$buffer(.('Display Load Map at Cold Boot $'));
call get$response(.prt$msg);
call crlf;
call crlf;
query = quest(1);
con$wid = con$wid + 1;
call get$param (.('Number of console columns $'),
.con$wid,10);
con$wid = con$wid - 1;
query = quest(2);
con$pag = con$pag + 1;
call get$param (.('Number of lines in console page $'),
.con$pag,10);
con$pag = con$pag - 1;
query = quest(3);
call print$console$buffer(.
('Backspace echoes erased character $'));
call get$response (.bck$spc);
call crlf;
query = quest(4);
call print$console$buffer(.
('Rubout echoes erased character $'));
call get$response (.rubout);
call crlf;
call crlf;
query = quest(5);
err1:
call print$console$buffer(.('Initial default drive ($'));
call write$console('A'+bdrive);
call print$console$buffer(.(':) ? $'));
call read$console$buffer(.lnbfr);
if lnbfr(1) <> 0 then
do;
temp = upper(lnbfr(2))-'A';
if not valid$drive(temp) then
goto err1;
bdrive = temp;
end;
call crlf;
call crlf;
query = quest(6);
call get$param (.('Top page of memory $'),
.mem$top,16);
os$top = shl(double(mem$top),8) + 100h;
query = quest(7);
call print$console$buffer(.('Bank switched memory $'));
call get$response (.bnk$swt);
call crlf;
non$bnk = not bnk$swt;
if bnk$swt then
do;
query = quest(8);
call get$param (.('Common memory base page $'),
.bnk$top,16);
call crlf;
query = quest(9);
call print$console$buffer(.('Long error messages $'));
call get$response(.lerror);
call crlf;
end;
else
do;
query = quest(156);
call crlf;
call print$console$buffer(.('Double allocation vectors $'));
call get$response(.dbl$alv);
call crlf;
bnk$top = 0;
end;
query = false;
ok = true;
call crlf;
call print$console$buffer(.('Accept new system definition $'));
call get$response(.ok);
call crlf;
end;
save$mem$top = mem$top;
mem$top = mem$top + 1;
rubout = not rubout;
end setup$system$dat;
setup$CPM80$sys:
procedure;
declare i byte;
call print$console$buffer (.( cr,lf,lf,
'CP/M 3.0 System Generation',cr,lf,
'Copyright (C) 1982, Digital Research',
cr,lf,cr,lf,'$'));
call delete$file (.fcbout);
call create$file (.fcbout);
FCBout(32) = 0;
do i = 0 to 127;
system$data(i) = 0;
end;
do i = 128 to 255;
system$data(i) = '$';
end;
prt$msg$ptr = 128;
call movef(3,.(cr,lf,' '),.system$data(prt$msg$ptr));
prt$msg$ptr = 131;
call movef(8,.FCBout+1,.fcb$msg);
call movef(3,.FCBout+9,.fcb$msg+9);
call set$DMA$address (.sctbfr);
call write$record (.FCBout);
call write$record (.FCBout);
end setup$CPM80$sys;
initialization:
procedure;
declare i byte;
nmb$sect = shr ((maxb-.sctbfr+1),7);
do i = 0 to 15;
hash$data(i) = 0;
end;
hash$space = 0;
if fcb(1) = 'A' then
do;
automatic = true;
display = false;
do i = 0 to 154;
quest(i) = false;
end;
end;
else
do;
automatic = false;
display = true;
end;
if fcb16(1) = 'D' then
do;
display = true;
end;
query = false;
end initialization;
/*
G E N C P M M a i n P r o g r a m
*/
res$flg = 0;
display = true;
call setup$CPM80$sys;
call initialization;
call setup$system$dat;
call get$file$info;
if bios$atts(2) <> 0
then res$bios$len = high(bios$atts(2) + 255);
else res$bios$len = high(bios$atts(0) + 255);
bios$pg = mem$top - res$bios$len;
bnk$off = bnk$top - (high(bios$atts(0) + 255) - res$bios$len);
bnk$pg = bnk$off - high(bnk$atts(0) + 255);
call load(.bios$fcb,.bios$atts);
call page$chop;
if not bnk$swt then
do;
scb$pg = bios$pg - (3 - res$flg);
res$pg = bios$pg - high(res$atts(0) + 255) + res$flg;
end;
else
do;
scb$pg = bios$pg - 1;
res$pg = bios$pg - high(res$atts(0) + 255);
end;
if need$tbl then
do;
call get$alloc$chk;
if bnk$swt then
do;
bnk$off = bnk$top - (high(bios$atts(0) + alloc$space +
chk$space + 255) - res$bios$len);
bnk$pg = bnk$off - high(bnk$atts(0) + 255);
xmove$implemented = get$xmove;
call setup$mem$seg$tbl;
if (not xmove$implemented) then
do len = 0 to 15;
record(len).altbnks = false;
end;
end;
else
xmove$implemented = false;
if not dont$hash then
call setup$hash;
call setbuf;
end;
res$len = res$bios$len;
offset = bios$pg;
call reloc$module(.bios$fcb);
if bnk$swt
then call display$layout(.('BNKBIOS3 SPR$'),
double(bios$pg)*256,res$bios$len);
else call display$layout(.('BIOS3 SPR$'),
double(bios$pg)*256,res$bios$len);
if not bnk$swt then
do;
len = res$bios$len;
off = 2;
call wrtbuf(len,.off);
common$len = len;
banked$len = 0;
end;
else
do;
len = high(bios$atts(0) + 255) - res$bios$len;
off = (high(res$atts(0) + 255) + res$bios$len) * 2 + 2;
call display$layout(.('BNKBIOS3 SPR$'),double(bnk$off)*256,len);
call wrtbuf(len,.off);
banked$len = len;
len = res$bios$len;
off = 2;
dma = dma - 80h;
call wrtbuf(len,.off);
common$len = len;
end;
res$len = high(res$atts(0) + 255) - res$flg;
offset = res$pg;
call load(.res$fcb,.res$atts);
call reloc$module(.res$fcb);
call setup$scb;
dma = dma - (res$flg * 256);
len = high(res$atts(0) + 255) - res$flg;
if not bnk$swt
then call display$layout(.('BDOS3 SPR$'),double(res$pg)*256,len);
else call display$layout(.('RESBDOS3 SPR$'),double(res$pg)*256,len);
call wrtbuf(len,.off);
common$len = common$len + len;
if bnk$swt then
do;
res$len = 0ffh;
offset = bnk$pg;
call load(.bnk$fcb,.bnk$atts);
call reloc$module(.bnk$fcb);
len = high(bnk$atts(0) + 255);
off = off + (high(bios$atts(0) + 255) - res$bios$len) * 2;
call display$layout(.('BNKBDOS3 SPR$'),double(bnk$pg)*256,len);
call wrtbuf(len,.off);
banked$len = banked$len + len;
end;
if not prt$msg then prt$msg$ptr = prt$msg$ptr - 3;
call movef(12,.(lf,cr,' 64K TPA',lf,cr),.system$data(prt$msg$ptr));
res$pg = shr(res$pg,2);
system$data(prt$msg$ptr+3) = res$pg/10 + '0';
system$data(prt$msg$ptr+4) = res$pg mod 10 + '0';
prt$msg$ptr = prt$msg$ptr + 12;
sys$entry = bios$pg * 256;
call movef(36,.('Copyright (C) 1982, Digital Research'),.system$data(10h));
call movef(6,.memory,.system$data(35h)); /* Copy Serial No. into header */
FCBout(33) = 0; FCBout(34) = 0; FCBout(35) = 0;
system$data(0) = mem$top;
system$data(2) = bnk$top;
call movef(8,.FCBout+1,.fcb$msg);
call movef(3,.FCBout+9,.fcb$msg+9);
call set$DMA$address(.system$data);
call write$random$record(.FCBout);
FCBout(33) = 1;
call set$DMA$address(.system$data(128));
call write$random$record(.FCBout);
call close$file(.fcbout);
if crdatf then
do; /* create a new data file for GENCPM */
crdatf = false;
mem$top = save$mem$top;
rubout = not rubout;
call movef(8,.data$fcb+1,.fcb$msg);
call movef(3,.data$fcb+9,.fcb$msg+9);
call crtdef;
end;
display = true;
call print$console$buffer (.(cr,lf,lf,
'*** CP/M 3.0 SYSTEM GENERATION DONE ***','$'));
return;
end plm;
end gencpm;