implement Cp; include "sys.m"; sys: Sys; include "draw.m"; include "arg.m"; include "readdir.m"; readdir: Readdir; Cp: module { init: fn(nil: ref Draw->Context, argv: list of string); }; stderr: ref Sys->FD; errors := 0; usage() { sys->fprint(stderr, "usage:\tcp src target\n"); sys->fprint(stderr, "\tcp [-r] src ... directory\n"); sys->raise("fail:usage"); } init(nil: ref Draw->Context, argv: list of string) { sys = load Sys Sys->PATH; stderr = sys->fildes(2); arg := load Arg Arg->PATH; if (arg == nil) { sys->fprint(stderr, "cp: cannot load %s: %r\n", Arg->PATH); sys->raise("fail:bad module"); } recursive := 0; arg->init(argv); while ((opt := arg->opt()) != 0) { if (opt != 'r') { sys->fprint(stderr, "cp: invalid option -%c\n", opt); usage(); } recursive = 1; } argv = arg->argv(); argc := len argv; if (argc < 2) usage(); dst: string; for (t := argv; t != nil; t = tl t) dst = hd t; (ok, dir) := sys->stat(dst); todir := (ok != -1 && (dir.mode & sys->CHDIR)); if (argc > 2 && !todir) { sys->fprint(stderr, "cp: %s not a directory\n", dst); sys->raise("fail:error"); } if (recursive) cpdir(argv, dst); else { for (; tl argv != nil; argv = tl argv) { if (todir) cp(hd argv, dst, basename(hd argv)); else cp(hd argv, dst, nil); } } if (errors) sys->raise("fail:error"); } basename(s: string): string { for ((nil, ls) := sys->tokenize(s, "/"); ls != nil; ls = tl ls) s = hd ls; return s; } cp(src, dst: string, newname: string) { ok: int; ds, dd: Sys->Dir; if (newname != nil) dst += "/" + newname; (ok, ds) = sys->stat(src); if (ok < 0) { warning(sys->sprint("%s: %r", src)); return; } if (ds.mode & sys->CHDIR) { warning(src + " is a directory"); return; } (ok, dd) = sys->stat(dst); if (ok != -1 && ds.qid.path == dd.qid.path && ds.dev == dd.dev && ds.dtype == dd.dtype) { warning(src + " and " + dst + " are the same file"); return; } sfd := sys->open(src, sys->OREAD); if (sfd == nil) { warning(sys->sprint("cannot open %s: %r", src)); return; } dfd := sys->create(dst, sys->OWRITE, ds.mode); if (dfd == nil) { warning(sys->sprint("cannot create %s: %r", dst)); return; } copy(sfd, dfd, src, dst); } mkdir(d: string, mode: int): int { dfd := sys->create(d, sys->OREAD, sys->CHDIR | mode); if (dfd == nil) { warning(sys->sprint("cannot make directory %s: %r", d)); return -1; } return 0; } copy(sfd, dfd: ref Sys->FD, src, dst: string): int { buf := array[Sys->ATOMICIO] of byte; for (;;) { r := sys->read(sfd, buf, Sys->ATOMICIO); if (r < 0) { warning(sys->sprint("error reading %s: %r", src)); return -1; } if (r == 0) return 0; if (sys->write(dfd, buf, r) != r) { warning(sys->sprint("error writing %s: %r", dst)); return -1; } } } cpdir(argv: list of string, dst: string) { readdir = load Readdir Readdir->PATH; if (readdir == nil) { sys->fprint(stderr, "cp: cannot load %s: %r\n", Readdir->PATH); sys->raise("fail:bad module"); } cache = array[NCACHE] of list of ref Sys->Dir; dexists := 0; (ok, dd) := sys->stat(dst); # destination file exists if (ok != -1) { if ((dd.mode & sys->CHDIR) == 0) { warning(dst + ": destination not a directory"); return; } dexists = 1; } err := 0; for (; tl argv != nil; argv = tl argv) { ds: Sys->Dir; src := hd argv; (ok, ds) = sys->stat(src); if (ok < 0) { warning(sys->sprint("can't stat %s: %r", src)); continue; } if ((ds.mode & Sys->CHDIR) == 0) { cp(hd argv, dst, basename(hd argv)); } else if (dexists) { if (ds.qid.path==dd.qid.path && ds.dev==dd.dev && ds.dtype==dd.dtype) { warning("cannot copy " + src + " into itself"); continue; } copydir(src, dst + "/" + basename(src), ds.mode); } else { copydir(src, dst, ds.mode); } } } copydir(src, dst: string, srcmode: int) { (ok, d) := sys->stat(dst); if (ok != -1) { warning("cannot copy " + src + " onto another directory"); return; } tmode := srcmode | 8r300; if (mkdir(dst, tmode) == -1) return; (entries, n) := readdir->init(src, Readdir->COMPACT); for (i := 0; i < n; i++) { e := entries[i]; path := src + "/" + e.name; if ((e.mode & Sys->CHDIR) == 0) cp(path, dst, e.name); else if (seen(e)) warning(path + ": directory loop found"); else copydir(path, dst + "/" + e.name, e.mode); } chmod(dst, srcmode); } # Avoid loops in tangled namespaces. (from du.b) NCACHE: con 64; # must be power of two cache: array of list of ref sys->Dir; seen(dir: ref sys->Dir): int { savlist := cache[dir.qid.path&(NCACHE-1)]; for(c := savlist; c!=nil; c = tl c){ sav := hd c; if(dir.qid.path==sav.qid.path && dir.dtype==sav.dtype && dir.dev==sav.dev) return 1; } cache[dir.qid.path&(NCACHE-1)] = dir :: savlist; return 0; } warning(e: string) { sys->fprint(stderr, "cp: %s\n", e); errors++; } chmod(s: string, mode: int): int { (ok, d) := sys->stat(s); if (ok < 0) return -1; d.mode = mode; if (sys->wstat(s, d) < 0) { warning(sys->sprint("cannot wstat %s: %r", s)); return -1; } return 0; }