ref: babf901b4a508c3ec5d1f89655f10377bbdf9637
dir: /appl/cmd/src.b/
implement Src;
include "sys.m";
sys: Sys;
include "draw.m";
include "dis.m";
dis: Dis;
Src: module
{
init: fn(nil: ref Draw->Context, argv: list of string);
};
init(nil: ref Draw->Context, argv: list of string)
{
sys = load Sys Sys->PATH;
dis = load Dis Dis->PATH;
if(dis != nil){
dis->init();
for(argv = tl argv; argv != nil; argv = tl argv){
s := src(hd argv);
if(s == nil)
s = "?";
sys->print("%s: %s\n", hd argv, s);
}
}
}
src(progname: string): string
{
disfile := 0;
if (len progname >= 4 && progname[len progname-4:] == ".dis")
disfile = 1;
pathlist: list of string;
if (absolute(progname))
pathlist = list of {""};
else
pathlist = list of {"/dis", "."};
err := "";
do {
path: string;
if (hd pathlist != "")
path = hd pathlist + "/" + progname;
else
path = progname;
npath := path;
if (!disfile)
npath += ".dis";
src := dis->src(npath);
if(src != nil)
return src;
err = sys->sprint("%r");
if (nonexistent(err)) {
# try and find it as a shell script
if (!disfile) {
(ok, info) := sys->stat(path);
if (ok == 0 && (info.mode & Sys->DMDIR) == 0
&& (info.mode & 8r111) != 0)
return path;
else
err = sys->sprint("%r");
}
}
pathlist = tl pathlist;
} while (pathlist != nil && nonexistent(err));
return nil;
}
absolute(p: string): int
{
if (len p < 2)
return 0;
if (p[0] == '/' || p[0] == '#')
return 1;
if (len p < 3 || p[0] != '.')
return 0;
if (p[1] == '/')
return 1;
if (p[1] == '.' && p[2] == '/')
return 1;
return 0;
}
nonexistent(e: string): int
{
errs := array[] of {"does not exist", "directory entry not found"};
for (i := 0; i < len errs; i++){
j := len errs[i];
if (j <= len e && e[len e-j:] == errs[i])
return 1;
}
return 0;
}