mirror of
https://github.com/SEPPDROID/Digital-Research-Source-Code.git
synced 2025-10-24 17:04:19 +00:00
633 lines
19 KiB
Plaintext
633 lines
19 KiB
Plaintext
|
|
/* C P / M - M P / M D I R E C T O R Y C O M M O N (SDIR) */
|
|
|
|
/* B E G I N N I N G O F C O M M O N M A I N M O D U L E */
|
|
|
|
|
|
/* This module is included in main80.plm or main86.plm. */
|
|
/* The differences between 8080 and 8086 versions are */
|
|
/* contained in the modules main80.plm, main86.plm and */
|
|
/* dpb80.plm, dpb86.plm and the submit files showing */
|
|
/* the different link and location addresses. */
|
|
|
|
|
|
$include (comlit.lit)
|
|
$include (mon.plm)
|
|
|
|
|
|
dcl patch (128) address;
|
|
|
|
/* Scanner Entry Points in scan.plm */
|
|
|
|
scan: procedure(pcb$adr) external;
|
|
declare pcb$adr address;
|
|
end scan;
|
|
|
|
scan$init: procedure(pcb$adr) external;
|
|
declare pcb$adr address;
|
|
end scan$init;
|
|
|
|
/* -------- Routines in other modules -------- */
|
|
|
|
search$init: procedure external; /* initialization of search.plm */
|
|
end search$init;
|
|
|
|
get$files: procedure external; /* entry to search.plm */
|
|
end get$files;
|
|
|
|
sort: procedure external; /* entry to sort.plm */
|
|
end sort;
|
|
|
|
mult23: procedure (num) address external; /* in sort.plm */
|
|
dcl num address;
|
|
end mult23;
|
|
|
|
display$files: procedure external; /* entry to disp.plm */
|
|
end display$files;
|
|
|
|
/* -------- Routines in util.plm -------- */
|
|
|
|
printb: procedure external;
|
|
end printb;
|
|
|
|
print$char: procedure(c) external;
|
|
dcl c byte;
|
|
end print$char;
|
|
|
|
print: procedure(string$adr) external;
|
|
dcl string$adr address;
|
|
end print;
|
|
|
|
crlf: procedure external;
|
|
end crlf;
|
|
|
|
p$decimal: procedure(value,fieldsize,zsup) external;
|
|
dcl value address,
|
|
fieldsize address,
|
|
zsup boolean;
|
|
end p$decimal;
|
|
|
|
|
|
/* ------------------------------------- */
|
|
|
|
dcl debug boolean public initial (false);
|
|
|
|
/* -------- version information -------- */
|
|
|
|
dcl (os,bdos) byte public;
|
|
$include (vers.lit)
|
|
|
|
$include (fcb.lit)
|
|
|
|
$include(search.lit)
|
|
|
|
dcl find find$structure public initial
|
|
(false,false,false,false, false,false,false,false);
|
|
|
|
dcl
|
|
num$search$files byte public initial(0),
|
|
no$page$mode byte public initial(0),
|
|
search (max$search$files) search$structure public;
|
|
|
|
dcl first$f$i$adr address external;
|
|
dcl get$all$dir$entries boolean public;
|
|
dcl first$pass boolean public;
|
|
|
|
dcl usr$vector address public initial(0), /* bits for user #s to scan */
|
|
active$usr$vector address public, /* active users on curdrv */
|
|
drv$vector address initial (0); /* bits for drives to scan */
|
|
|
|
$include (format.lit)
|
|
|
|
dcl format byte public initial (form$full),
|
|
page$len address public initial (0ffffh),
|
|
/* lines on a page before printing new headers, 0 forces initial hdrs */
|
|
message boolean public initial(false),/* show titles when no files found*/
|
|
formfeeds boolean public initial(false),/* use form feeds */
|
|
date$opt boolean public initial(false), /* dates display */
|
|
display$attributes boolean public initial(false); /* attributes display */
|
|
|
|
dcl file$displayed boolean external;
|
|
/* true if 1 or more files displayed by dsh.plm */
|
|
|
|
dcl sort$op boolean initial (true); /* default is to do sorting */
|
|
dcl sorted boolean external; /* if successful sort */
|
|
|
|
|
|
dcl cur$usr byte public, /* current user being searched */
|
|
cur$drv byte public; /* current drive " " */
|
|
|
|
/* -------- BDOS calls --------- */
|
|
|
|
get$version: procedure address; /* returns current version information */
|
|
return mon2(12,0);
|
|
end get$version;
|
|
|
|
select$drive: procedure(d);
|
|
declare d byte;
|
|
call mon1(14,d);
|
|
end select$drive;
|
|
|
|
search$first: procedure(d) byte external;
|
|
dcl d address;
|
|
end search$first;
|
|
|
|
search$next: procedure byte external;
|
|
end search$next;
|
|
|
|
get$cur$drv: procedure byte; /* return current drive number */
|
|
return mon2(25,0);
|
|
end get$cur$drv;
|
|
|
|
getlogin: procedure address; /* get the login vector */
|
|
return mon3(24,0);
|
|
end getlogin;
|
|
|
|
getusr: procedure byte; /* return current user number */
|
|
return mon2(32,0ffh);
|
|
end getusr;
|
|
|
|
getscbbyte: procedure (offset) byte public; /* [JCE] public so the timest */
|
|
declare offset byte; /* code can use it */
|
|
declare scbpb structure
|
|
(offset byte,
|
|
set byte,
|
|
value address);
|
|
scbpb.offset = offset;
|
|
scbpb.set = 0;
|
|
return mon2(49,.scbpb);
|
|
end getscbbyte;
|
|
|
|
set$console$mode: procedure;
|
|
/* set console mode to control-c only */
|
|
call mon1(109,1);
|
|
end set$console$mode;
|
|
|
|
terminate: procedure public;
|
|
call mon1 (0,0);
|
|
end terminate;
|
|
|
|
|
|
/* -------- Utility routines -------- */
|
|
|
|
number: procedure (char) boolean;
|
|
dcl char byte;
|
|
return(char >= '0' and char <= '9');
|
|
end number;
|
|
|
|
make$numeric: procedure(char$adr,len,val$adr) boolean;
|
|
dcl (char$adr, val$adr, place) address,
|
|
chars based char$adr (1) byte,
|
|
value based val$adr address,
|
|
(i,len) byte;
|
|
|
|
value = 0;
|
|
place = 1;
|
|
do i = 1 to len;
|
|
if not number(chars(len - i)) then
|
|
return(false);
|
|
value = value + (chars(len - i) - '0') * place;
|
|
place = place * 10;
|
|
end;
|
|
return(true);
|
|
end make$numeric;
|
|
|
|
set$vec: procedure(v$adr,num) public;
|
|
dcl v$adr address, /* set bit number given by num */
|
|
vector based v$adr address, /* 0 <= num <= 15 */
|
|
num byte;
|
|
if num = 0 then
|
|
vector = vector or 1;
|
|
else
|
|
vector = vector or shl(double(1),num);
|
|
end set$vec;
|
|
|
|
bit$loc: procedure(vector) byte;
|
|
/* return location of right most on bit vector */
|
|
dcl vector address, /* 0 - 15 */
|
|
i byte;
|
|
i = 0;
|
|
do while i < 16 and (vector and double(1)) = 0;
|
|
vector = shr(vector,1);
|
|
i = i + 1;
|
|
end;
|
|
return(i);
|
|
end bit$loc;
|
|
|
|
get$nxt: procedure(vector$adr) byte;
|
|
dcl i byte,
|
|
(vector$adr,mask) address,
|
|
vector based vector$adr address;
|
|
/*
|
|
if debug then
|
|
do; call print(.(cr,lf,'getnxt: vector = $'));
|
|
call pdecimal(vector,10000,false);
|
|
end;
|
|
*/
|
|
if (i := bit$loc(vector)) > 15 then
|
|
return(0ffh);
|
|
mask = 1;
|
|
if i > 0 then
|
|
mask = shl(mask,i);
|
|
vector = vector xor mask; /* turn off bit */
|
|
/*
|
|
if debug then
|
|
do; call print(.(cr,lf,'getnxt: vector, i, mask $'));
|
|
call pdecimal(vector,10000,false);
|
|
call printb;
|
|
call pdecimal(i,10000,false);
|
|
call printb;
|
|
call pdecimal(mask,10000,false);
|
|
end;
|
|
*/
|
|
return(i);
|
|
end get$nxt; /* too bad plm rotates only work on byte values */
|
|
|
|
/* help: procedure; COMMENTED OUT - HELP PROGRAM REPLACE DISPLAY
|
|
|
|
call print(.(cr,lf,
|
|
tab,tab,tab,'DIR EXAMPLES',cr,lf,lf,
|
|
'dir file.one',tab,tab,tab,
|
|
'(find a file on current user and default drive)',cr,lf,
|
|
'dir *.com d:*.pli',tab,tab,'(find matching files on default and d: drive)',
|
|
cr,lf,
|
|
'dir [rw]',tab,tab,tab,'(find files that are read/write)',cr,lf,
|
|
'dir [ro dir sys]',tab,tab,'(same for read/only, directory, system)',cr,lf,
|
|
'dir [xfcb]',tab,tab,tab,'(find files with XFCB''s)',cr,lf,
|
|
'dir [nonxfcb]',tab,tab,tab,'(find files without XFCB''s)',cr,lf,
|
|
'dir [exclude] *.com',tab,tab,'(find files that don''t end in ''com'')',cr,lf,
|
|
'dir [nosort]',tab,tab,tab,'(don''t sort the files)',cr,lf,
|
|
'dir [full]',tab,tab,tab,'(show all file information)',cr,lf,
|
|
'dir [size]',tab,tab,tab,'(show name and size in kilobytes)',cr,lf,
|
|
'dir [short]',tab,tab,tab,'(show just the file names)',cr,lf,
|
|
'dir [drive = all]',tab,tab,'(search all logged in drives)',cr,lf,
|
|
'dir [drive = (a,b,p)]',tab,tab,
|
|
'(search specified drives, ''disk'' is synonym)',cr,lf,
|
|
'dir [user = all]',tab,tab,'(find files with any user number)',cr,lf,
|
|
'dir [user = (0,1,15), G12]',tab,'(find files with specified user number)',
|
|
cr,lf,
|
|
'dir [length = n]',tab,tab,'(print headers every n lines)',cr,lf,
|
|
'dir [ff]',tab,tab,tab,'(print form feeds between headers)',cr,lf,
|
|
'dir [message user=all]',tab,tab,'(show user/drive areas with no files)',
|
|
cr,lf,
|
|
'dir [help]',tab,tab,tab,'(show this message)',cr,lf,
|
|
'dir [dir sys rw ro sort xfcb nonxfcb full] d:*.*',tab,'(defaults)$'));
|
|
|
|
call terminate;
|
|
end help; */
|
|
|
|
|
|
/* -------- Scanner Info -------- */
|
|
|
|
$include (scan.lit)
|
|
|
|
dcl pcb pcb$structure
|
|
initial (0,.buff(0),.fcb,0,0,0,0) ;
|
|
|
|
dcl token based pcb.token$adr (12) byte;
|
|
dcl got$options boolean;
|
|
|
|
get$options: procedure;
|
|
dcl temp byte;
|
|
|
|
do while pcb.scan$adr <> 0ffffh and ((pcb.tok$typ and t$op) <> 0);
|
|
|
|
if pcb.nxt$token <> t$mod then do;
|
|
/* options with no modifiers */
|
|
if token(1) = 'A' then
|
|
display$attributes = true;
|
|
|
|
else if token(1) = 'D' and token(2) = 'I' then
|
|
find.dir = true;
|
|
|
|
else if token(1) = 'D' and token(2) = 'A' then do;
|
|
format = form$full;
|
|
date$opt = true;
|
|
end;
|
|
/*
|
|
else if token(1) = 'D' and token(2) = 'E' then
|
|
debug = true;
|
|
*/
|
|
else if token(1) = 'E' then
|
|
find.exclude = true;
|
|
|
|
else if token(1) = 'F'then do;
|
|
if token(2) = 'F' then
|
|
formfeeds = true;
|
|
else if token(2) = 'U' then
|
|
format = form$full;
|
|
else goto op$err;
|
|
end;
|
|
|
|
else if token(1) = 'G' then
|
|
do;
|
|
if pcb.token$len < 3 then
|
|
temp = token(2) - '0';
|
|
else
|
|
temp = (token(2) - '0') * 10 + (token(3) - '0');
|
|
if temp >= 0 and temp <= 15 then
|
|
call set$vec(.usr$vector,temp);
|
|
else goto op$err;
|
|
end;
|
|
|
|
/* else if token(1) = 'H' then
|
|
call help; */
|
|
|
|
else if token(1) = 'M' then
|
|
message = true;
|
|
|
|
else if token(1) = 'N' then
|
|
do;
|
|
if token(4) = 'X' then
|
|
find.nonxfcb = true;
|
|
else if token(3) = 'P' then
|
|
no$page$mode = 0FFh;
|
|
else if token(3) = 'S' then
|
|
sort$op = false;
|
|
else goto op$err;
|
|
end;
|
|
|
|
/* else if token(1) = 'P' then
|
|
find.pass = true; */
|
|
|
|
else if token(1) = 'R' and token(2) = 'O' then
|
|
find.ro = true;
|
|
|
|
else if token(1) = 'R' and token(2) = 'W' then
|
|
find.rw = true;
|
|
|
|
else if token(1) = 'S' then do;
|
|
if token(2) = 'Y' then
|
|
find.sys = true;
|
|
else if token(2) = 'I' then
|
|
format = form$size;
|
|
else if token(2) = 'O' then
|
|
sort$op = true;
|
|
else goto op$err;
|
|
end;
|
|
|
|
else if token(1) = 'X' then
|
|
find.xfcb = true;
|
|
|
|
else goto op$err;
|
|
|
|
call scan(.pcb);
|
|
end;
|
|
|
|
else
|
|
do; /* options with modifiers */
|
|
if token(1) = 'L' then
|
|
do;
|
|
call scan(.pcb);
|
|
if (pcb.tok$typ and t$numeric) <> 0 then
|
|
if make$numeric(.token(1),pcb.token$len,.page$len) then
|
|
if page$len < 5 then
|
|
goto op$err;
|
|
else call scan(.pcb);
|
|
else goto op$err;
|
|
else goto op$err;
|
|
end;
|
|
|
|
else if token(1) = 'U' then
|
|
do;
|
|
/*
|
|
if debug then
|
|
call print(.(cr,lf,'In User option$'));
|
|
*/
|
|
call scan(.pcb);
|
|
if (((pcb.tok$typ and t$mod) = 0) or (bdos < bdos20)) then
|
|
goto op$err;
|
|
do while (pcb.tok$typ and t$mod) <> 0 and
|
|
pcb.scan$adr <> 0ffffh;
|
|
if token(1) = 'A' and token(2) = 'L' then
|
|
usr$vector = 0ffffh;
|
|
else if (pcb.tok$typ and t$numeric) <> 0 and pcb.token$len < 3 then
|
|
do;
|
|
if pcb.token$len = 1 then
|
|
temp = token(1) - '0';
|
|
else
|
|
temp = (token(1) - '0') * 10 + (token(2) - '0');
|
|
if temp >= 0 and temp <= 15 then
|
|
call set$vec(.usr$vector,temp);
|
|
else goto op$err;
|
|
end;
|
|
else goto op$err;
|
|
call scan(.pcb);
|
|
end;
|
|
end; /* User option */
|
|
|
|
else if token(1) = 'D' and (token(2) = 'R' or token(2) = 'I') then
|
|
do; /* allow DRIVE or DISK */
|
|
call scan(.pcb);
|
|
if (pcb.tok$typ and t$mod) = 0 then
|
|
goto op$err;
|
|
do while (pcb.tok$typ and t$mod ) <> 0 and
|
|
pcb.scan$adr <> 0ffffh;
|
|
if token(1) = 'A' and token(2) = 'L' then
|
|
do;
|
|
drv$vector = 0ffffh;
|
|
drv$vector = drv$vector and get$login;
|
|
end;
|
|
else if token(1) >= 'A' and token(1) <= 'P' then
|
|
call set$vec(.drv$vector,token(1) - 'A');
|
|
else goto op$err;
|
|
call scan(.pcb);
|
|
end;
|
|
end; /* drive option */
|
|
|
|
else goto op$err;
|
|
|
|
end; /* options with modifiers */
|
|
|
|
end; /* do while */
|
|
|
|
got$options = true;
|
|
return;
|
|
|
|
op$err:
|
|
call print(.('ERROR: Illegal Option or Modifier.',
|
|
cr,lf,'$'));
|
|
call terminate;
|
|
end get$options;
|
|
|
|
get$file$spec: procedure;
|
|
dcl i byte;
|
|
if num$search$files < max$search$files then
|
|
do;
|
|
call move(f$namelen + f$typelen,.token(1),
|
|
.search(num$search$files).name(0));
|
|
|
|
if search(num$search$files).name(f$name - 1) = ' ' and
|
|
search(num$search$files).name(f$type - 1) = ' ' then
|
|
search(num$search$files).anyfile = true; /* match on any file */
|
|
else search(num$search$files).anyfile = false;/* speedier compare */
|
|
|
|
if token(0) = 0 then
|
|
search(num$search$files).drv = 0ffh; /* no drive letter with */
|
|
else /* file spec */
|
|
search(num$search$files).drv = token(0) - 1;
|
|
/* 0ffh in drv field indicates to look on all drives that will be */
|
|
/* scanned as set by the "drive =" option, see "match:" proc in */
|
|
/* search.plm module */
|
|
|
|
num$search$files = num$search$files + 1;
|
|
end;
|
|
else
|
|
do; call print(.('File Spec Limit is $'));
|
|
call p$decimal(max$search$files,100,true);
|
|
call crlf;
|
|
end;
|
|
call scan(.pcb);
|
|
end get$file$spec;
|
|
|
|
set$defaults: procedure;
|
|
/* set defaults if not explicitly set by user */
|
|
if not (find.dir or find.sys) then
|
|
find.dir, find.sys = true;
|
|
if not(find.ro or find.rw) then
|
|
find.rw, find.ro = true;
|
|
|
|
if find.xfcb or find.nonxfcb then
|
|
do; if format = form$short then
|
|
format = form$full;
|
|
end;
|
|
else /* both xfcb and nonxfcb are off */
|
|
find.nonxfcb, find.xfcb = true;
|
|
|
|
if num$search$files = 0 then
|
|
do;
|
|
search(num$search$files).anyfile = true;
|
|
search(num$search$files).drv = 0ffh;
|
|
num$search$files = 1;
|
|
end;
|
|
|
|
if drv$vector = 0 then
|
|
do i = 0 to num$search$files - 1;
|
|
if search(i).drv = 0ffh then search(i).drv = cur$drv;
|
|
call set$vec(.drv$vector,search(i).drv);
|
|
end;
|
|
else /* a "[drive =" option was found */
|
|
do i = 0 to num$search$files - 1;
|
|
if search(i).drv <> 0ffh and search(i).drv <> cur$drv then
|
|
do; call print(.('ERROR: Illegal Global/Local ',
|
|
'Drive Spec Mixing.',cr,lf,'$'));
|
|
call terminate;
|
|
end;
|
|
end;
|
|
if usr$vector = 0 then
|
|
call set$vec(.usr$vector,get$usr);
|
|
|
|
/* set up default page size for display */
|
|
if bdos > bdos30 then do;
|
|
if not formfeeds then do;
|
|
if page$len = 0ffffh then do;
|
|
page$len = getscbbyte(page$len$offset);
|
|
if page$len < 5 then
|
|
page$len = 24;
|
|
end;
|
|
end;
|
|
end;
|
|
end set$defaults;
|
|
|
|
dcl (save$uvec,temp) address;
|
|
dcl i byte;
|
|
declare last$dseg$byte byte
|
|
initial (0);
|
|
|
|
plm:
|
|
do;
|
|
os = high(get$version);
|
|
bdos = low(get$version);
|
|
|
|
if bdos < bdos30 or os = mpm then do;
|
|
call print(.('Requires CP/M 3',cr,lf,'$'));
|
|
call terminate; /* check to make sure function call is valid */
|
|
end;
|
|
else
|
|
call set$console$mode;
|
|
|
|
/* note - initialized declarations set defaults */
|
|
cur$drv = get$cur$drv;
|
|
call scan$init(.pcb);
|
|
call scan(.pcb);
|
|
no$page$mode = getscbbyte(nopage$mode$offset);
|
|
got$options = false;
|
|
do while pcb.scan$adr <> 0ffffh;
|
|
if (pcb.tok$typ and t$op) <> 0 then
|
|
if got$options = false then
|
|
call get$options;
|
|
else
|
|
do;
|
|
call print(.('ERROR: Options not grouped together.',
|
|
cr,lf,'$'));
|
|
call terminate;
|
|
end;
|
|
else if (pcb.tok$typ and t$filespec) <> 0 then
|
|
call get$file$spec;
|
|
else
|
|
do;
|
|
call print(.('ERROR: Illegal command tail.',cr,lf,'$'));
|
|
call terminate;
|
|
end;
|
|
end;
|
|
|
|
call set$defaults;
|
|
|
|
/* main control loop */
|
|
|
|
call search$init; /* set up memory pointers for subsequent storage */
|
|
|
|
do while (cur$drv := get$nxt(.drv$vector)) <> 0ffh;
|
|
call select$drive(cur$drv);
|
|
save$uvec = usr$vector; /* user numbers to search on each drive */
|
|
active$usr$vector = 0; /* users active on cur$drv */
|
|
cur$usr = get$nxt(.usr$vector); /* get first user num and mask */
|
|
get$all$dir$entries = false; /* off it off */
|
|
if usr$vector <> 0 and format <> form$short then
|
|
/* find high water mark if */
|
|
do; /* more than one user requested */
|
|
fcb(f$drvusr) = '?';
|
|
i = search$first(.fcb); /* get first directory entry */
|
|
temp = 0;
|
|
do while i <> 255;
|
|
temp = temp + 1;
|
|
i = search$next;
|
|
end; /* is there enough space in the */
|
|
/* worst case ? */
|
|
if maxb > mult23(temp) + shl(temp,1) then
|
|
get$all$dir$entries = true; /* location of last possible */
|
|
end; /* file info record and add */
|
|
first$pass = true; /* room for sort indices */
|
|
active$usr$vector = 0ffffh;
|
|
do while cur$usr <> 0ffh;
|
|
/*
|
|
if debug then
|
|
call print(.(cr,lf,'in user loop $'));
|
|
*/
|
|
call set$vec(.temp,cur$usr);
|
|
if (temp and active$usr$vector) <> 0 then
|
|
do;
|
|
if format <> form$short and
|
|
(first$pass or not get$all$dir$entries) then
|
|
do;
|
|
call get$files; /* collect files in memory and */
|
|
first$pass = false; /* build the active usr vector */
|
|
sorted = false; /* sort module will set sorted */
|
|
if sort$op then /* to true, if successful sort */
|
|
call sort;
|
|
end;
|
|
call display$files;
|
|
end;
|
|
cur$usr = get$nxt(.usr$vector);
|
|
end;
|
|
usr$vector = save$uvec; /* restore user vector for nxt */
|
|
end; /* do while drv$usr drive scan */
|
|
|
|
|
|
if not file$displayed and not message then
|
|
call print(.('No File',cr,lf,'$'));
|
|
call terminate;
|
|
|
|
end;
|
|
end sdir;
|