#include #include #include #include enum{ LEN = 8*1024, HUNKS = 128, /* * types of destination file sytems */ Kfs = 0, Fs, Archive, }; typedef struct File File; struct File{ char *new; char *elem; char *old; char *uid; char *gid; ulong mode; }; void arch(Dir*); void copy(Dir*); int copyfile(File*, Dir*, int); void* emalloc(ulong); void error(char *, ...); void freefile(File*); File* getfile(File*); char* getmode(char*, ulong*); char* getname(char*, char**); char* getpath(char*); void kfscmd(char *); void mkdir(Dir*); int mkfile(File*); void mkfs(File*, int); char* mkpath(char*, char*); void mktree(File*, int); void mountkfs(char*); void printfile(File*); void setnames(File*); void setusers(void); void skipdir(void); char* strdup(char*); int uptodate(Dir*, char*); void usage(void); void warn(char *, ...); Biobuf *b; Biobufhdr bout; /* stdout when writing archive */ uchar boutbuf[2*LEN]; char newfile[LEN]; char oldfile[LEN]; char *proto; char *users; char *oldroot; char *newroot; char *prog = "mkfs"; int lineno; char *buf; char *zbuf; int buflen = 1024-8; int indent; int verb; int modes; int ream; int debug; int xflag; int sfd; int fskind; /* Kfs, Fs, Archive */ int setuid; /* on Fs: set uid and gid? */ int badchar; char *user; ulong modemask = ~0ul; void main(int argc, char **argv) { File file; char *name; int i, errs; quotefmtinstall(); user = getuser(); name = ""; memset(&file, 0, sizeof file); file.new = ""; file.old = 0; oldroot = ""; newroot = "/n/kfs"; users = 0; fskind = Kfs; ARGBEGIN{ case 'a': if(fskind != Kfs) { fprint(2, "cannot use -a with -d\n"); usage(); } fskind = Archive; newroot = ""; Binits(&bout, 1, OWRITE, boutbuf, sizeof boutbuf); break; case 'b': badchar = 1; break; case 'd': if(fskind != Kfs) { fprint(2, "cannot use -d with -a\n"); usage(); } fskind = Fs; newroot = EARGF(usage()); break; case 'D': debug = 1; break; case 'k': modemask = DMDIR|DMAPPEND|DMEXCL|0777; break; case 'n': name = EARGF(usage()); break; case 'p': modes = 1; break; case 'r': ream = 1; break; case 's': oldroot = EARGF(usage()); break; case 'u': users = EARGF(usage()); break; case 'U': setuid = 1; break; case 'v': verb = 1; break; case 'x': xflag = 1; break; case 'z': buflen = atoi(EARGF(usage()))-8; break; default: usage(); }ARGEND if(!argc) usage(); buf = emalloc(buflen); zbuf = emalloc(buflen); memset(zbuf, 0, buflen); mountkfs(name); kfscmd("allow"); proto = "users"; setusers(); errs = 0; for(i = 0; i < argc; i++){ proto = argv[i]; fprint(2, "processing %q\n", proto); b = Bopen(proto, OREAD); if(!b){ fprint(2, "%q: can't open %q: skipping\n", prog, proto); errs++; continue; } lineno = 0; indent = 0; mkfs(&file, -1); Bterm(b); } fprint(2, "file system made\n"); kfscmd("disallow"); kfscmd("sync"); if(errs) exits("skipped protos"); if(fskind == Archive){ Bprint(&bout, "end of archive\n"); Bterm(&bout); } exits(0); } void mkfs(File *me, int level) { File *child; int rec; child = getfile(me); if(!child) return; if((child->elem[0] == '+' || child->elem[0] == '*') && child->elem[1] == '\0'){ rec = child->elem[0] == '+'; free(child->new); child->new = strdup(me->new); setnames(child); mktree(child, rec); freefile(child); child = getfile(me); } while(child && indent > level){ if(mkfile(child)) mkfs(child, indent); freefile(child); child = getfile(me); } if(child){ freefile(child); Bseek(b, -Blinelen(b), 1); lineno--; } } void mktree(File *me, int rec) { File child; Dir *d; int i, n, fd; fd = open(oldfile, OREAD); if(fd < 0){ warn("can't open %q: %r", oldfile); return; } child = *me; while((n = dirread(fd, &d)) > 0){ for(i = 0; i < n; i++){ child.new = mkpath(me->new, d[i].name); if(me->old) child.old = mkpath(me->old, d[i].name); child.elem = d[i].name; setnames(&child); if(copyfile(&child, &d[i], 1) && rec) mktree(&child, rec); free(child.new); if(child.old) free(child.old); } } close(fd); } int mkfile(File *f) { Dir *dir; if((dir = dirstat(oldfile)) == nil){ warn("can't stat file %q: %r", oldfile); skipdir(); return 0; } return copyfile(f, dir, 0); } int copyfile(File *f, Dir *d, int permonly) { ulong mode; Dir nd; if(xflag){ Bprint(&bout, "%q\t%ld\t%lld\n", f->new, d->mtime, d->length); return (d->mode & DMDIR) != 0; } if(verb && (fskind == Archive || ream)) fprint(2, "%q\n", f->new); d->name = f->elem; if(d->type != 'M'){ d->uid = "sys"; d->gid = "sys"; mode = (d->mode >> 6) & 7; d->mode |= mode | (mode << 3); } if(strcmp(f->uid, "-") != 0) d->uid = f->uid; if(strcmp(f->gid, "-") != 0) d->gid = f->gid; if(fskind == Fs && !setuid){ d->uid = ""; d->gid = ""; } if(f->mode != ~0){ if(permonly) d->mode = (d->mode & ~0666) | (f->mode & 0666); else if((d->mode&DMDIR) != (f->mode&DMDIR)) warn("inconsistent mode for %q", f->new); else d->mode = f->mode; } if(!uptodate(d, newfile)){ if(verb && (fskind != Archive && ream == 0)) fprint(2, "%q\n", f->new); if(d->mode & DMDIR) mkdir(d); else copy(d); }else if(modes){ nulldir(&nd); nd.mode = d->mode & modemask; nd.gid = d->gid; nd.mtime = d->mtime; if(verb && (fskind != Archive && ream == 0)) fprint(2, "%q\n", f->new); if(dirwstat(newfile, &nd) < 0) warn("can't set modes for %q: %r", f->new); nulldir(&nd); nd.uid = d->uid; dirwstat(newfile, &nd); } return (d->mode & DMDIR) != 0; } /* * check if file to is up to date with * respect to the file represented by df */ int uptodate(Dir *df, char *to) { int ret; Dir *dt; if(fskind == Archive || ream || (dt = dirstat(to)) == nil) return 0; ret = dt->mtime >= df->mtime; free(dt); return ret; } void copy(Dir *d) { char cptmp[LEN], *p; int f, t, n, needwrite, nowarnyet = 1; vlong tot, len; Dir nd; if(badchar && strpbrk(oldfile, "\t \n\r")){ warn("bogus filename %q", oldfile); return; } f = open(oldfile, OREAD); if(f < 0){ warn("can't open %q: %r", oldfile); return; } t = -1; if(fskind == Archive) arch(d); else{ strcpy(cptmp, newfile); p = utfrrune(cptmp, L'/'); if(!p) error("internal temporary file error"); strcpy(p+1, "__mkfstmp"); t = create(cptmp, OWRITE, 0666); if(t < 0){ warn("can't create %q: %r", newfile); close(f); return; } } needwrite = 0; for(tot = 0; tot < d->length; tot += n){ len = d->length - tot; /* don't read beyond d->length */ if (len > buflen) len = buflen; n = read(f, buf, len); if(n <= 0) { if(n < 0 && nowarnyet) { warn("can't read %q: %r", oldfile); nowarnyet = 0; } /* * don't quit: pad to d->length (in pieces) to agree * with the length in the header, already emitted. */ memset(buf, 0, len); n = len; } if(fskind == Archive){ if(Bwrite(&bout, buf, n) != n) error("write error: %r"); }else if(memcmp(buf, zbuf, n) == 0){ if(seek(t, n, 1) < 0) error("can't write zeros to %q: %r", newfile); needwrite = 1; }else{ if(write(t, buf, n) < n) error("can't write %q: %r", newfile); needwrite = 0; } } close(f); if(needwrite){ if(seek(t, -1, 1) < 0 || write(t, zbuf, 1) != 1) error("can't write zero at end of %q: %r", newfile); } if(tot != d->length){ /* this should no longer happen */ warn("wrong number of bytes written to %q (was %lld should be %lld)\n", newfile, tot, d->length); if(fskind == Archive){ warn("seeking to proper position\n"); /* does no good if stdout is a pipe */ Bseek(&bout, d->length - tot, 1); } } if(fskind == Archive) return; remove(newfile); nulldir(&nd); nd.mode = d->mode & modemask; nd.gid = d->gid; nd.mtime = d->mtime; nd.name = d->name; if(dirfwstat(t, &nd) < 0) error("can't move tmp file to %q: %r", newfile); nulldir(&nd); nd.uid = d->uid; dirfwstat(t, &nd); close(t); } void mkdir(Dir *d) { Dir *d1; Dir nd; int fd; if(fskind == Archive){ arch(d); return; } fd = create(newfile, OREAD, d->mode & modemask); nulldir(&nd); nd.mode = d->mode & modemask; nd.gid = d->gid; nd.mtime = d->mtime; if(fd < 0){ if((d1 = dirstat(newfile)) == nil || !(d1->mode & DMDIR)){ free(d1); warn("can't create %q: %r", newfile); } free(d1); if(dirwstat(newfile, &nd) < 0) warn("can't set modes for %q: %r", newfile); nulldir(&nd); nd.uid = d->uid; dirwstat(newfile, &nd); return; } if(dirfwstat(fd, &nd) < 0) warn("can't set modes for %q: %r", newfile); nulldir(&nd); nd.uid = d->uid; dirfwstat(fd, &nd); close(fd); } void arch(Dir *d) { Bprint(&bout, "%q %luo %q %q %lud %lld\n", newfile, d->mode, d->uid, d->gid, d->mtime, d->length); } char * mkpath(char *prefix, char *elem) { char *p; int n; n = strlen(prefix) + strlen(elem) + 2; p = emalloc(n); sprint(p, "%s/%s", prefix, elem); return p; } char * strdup(char *s) { char *t; t = emalloc(strlen(s) + 1); return strcpy(t, s); } void setnames(File *f) { sprint(newfile, "%s%s", newroot, f->new); if(f->old){ if(f->old[0] == '/') sprint(oldfile, "%s%s", oldroot, f->old); else strcpy(oldfile, f->old); }else sprint(oldfile, "%s%s", oldroot, f->new); if(strlen(newfile) >= sizeof newfile || strlen(oldfile) >= sizeof oldfile) error("name overfile"); } void freefile(File *f) { if(f->old) free(f->old); if(f->new) free(f->new); free(f); } /* * skip all files in the proto that * could be in the current dir */ void skipdir(void) { char *p, c; int level; if(indent < 0 || b == nil) /* b is nil when copying adm/users */ return; level = indent; for(;;){ indent = 0; p = Brdline(b, '\n'); lineno++; if(!p){ indent = -1; return; } while((c = *p++) != '\n') if(c == ' ') indent++; else if(c == '\t') indent += 8; else break; if(indent <= level){ Bseek(b, -Blinelen(b), 1); lineno--; return; } } } File* getfile(File *old) { File *f; char *elem; char *p; int c; if(indent < 0) return 0; loop: indent = 0; p = Brdline(b, '\n'); lineno++; if(!p){ indent = -1; return 0; } while((c = *p++) != '\n') if(c == ' ') indent++; else if(c == '\t') indent += 8; else break; if(c == '\n' || c == '#') goto loop; p--; f = emalloc(sizeof *f); p = getname(p, &elem); if(debug) fprint(2, "getfile: %q root %q\n", elem, old->new); f->new = mkpath(old->new, elem); f->elem = utfrrune(f->new, L'/') + 1; p = getmode(p, &f->mode); p = getname(p, &f->uid); if(!*f->uid) f->uid = "-"; p = getname(p, &f->gid); if(!*f->gid) f->gid = "-"; f->old = nil; getname(p, &f->old); if(f->old != nil && (f->old[0] == 0 || strcmp(f->old, "-") == 0)){ free(f->old); f->old = nil; } setnames(f); if(debug) printfile(f); return f; } char* getpath(char *p) { char *q, *new; int c, n; while((c = *p) == ' ' || c == '\t') p++; q = p; while((c = *q) != '\n' && c != ' ' && c != '\t') q++; if(q == p) return 0; n = q - p; new = emalloc(n + 1); memcpy(new, p, n); new[n] = 0; return new; } char* getenv1(char *s) { char *p, *env, *r; p = strchr(s, '/'); if(p != nil) *p++ = 0; env = getenv(s); if(env == nil) return nil; if(p){ r = smprint("%s/%s", env, p); free(env); }else r = env; return r; } char* getname(char *p, char **buf) { char *s, *start; int c; while((c = *p) == ' ' || c == '\t') p++; start = p; while((c = *p) != '\n' && c != ' ' && c != '\t' && c != '\0') p++; *buf = emalloc(p+1-start); memmove(*buf, start, p-start); (*buf)[p-start] = '\0'; if(**buf == '$'){ s = getenv1(*buf+1); if(s == 0){ warn("can't read environment variable %q", *buf+1); skipdir(); free(*buf); *buf = nil; return nil; } free(*buf); *buf = s; } return p; } char* getmode(char *p, ulong *xmode) { char *buf, *s; ulong m; *xmode = ~0; p = getname(p, &buf); if(p == nil) return nil; s = buf; if(!*s || strcmp(s, "-") == 0) return p; m = 0; if(*s == 'd'){ m |= DMDIR; s++; } if(*s == 'a'){ m |= DMAPPEND; s++; } if(*s == 'l'){ m |= DMEXCL; s++; } if(s[0] < '0' || s[0] > '7' || s[1] < '0' || s[1] > '7' || s[2] < '0' || s[2] > '7' || s[3]){ warn("bad mode specification %q", buf); free(buf); return p; } *xmode = m | strtoul(s, 0, 8); free(buf); return p; } void setusers(void) { File file; int m; if(fskind != Kfs) return; m = modes; modes = 1; file.uid = "adm"; file.gid = "adm"; file.mode = DMDIR|0775; file.new = "/adm"; file.elem = "adm"; file.old = 0; setnames(&file); strcpy(oldfile, file.new); /* Don't use root for /adm */ mkfile(&file); file.new = "/adm/users"; file.old = users; file.elem = "users"; file.mode = 0664; setnames(&file); if (file.old) strcpy(oldfile, file.old); /* Don't use root for /adm/users */ mkfile(&file); kfscmd("user"); mkfile(&file); file.mode = DMDIR|0775; file.new = "/adm"; file.old = "/adm"; file.elem = "adm"; setnames(&file); strcpy(oldfile, file.old); /* Don't use root for /adm */ mkfile(&file); modes = m; } void mountkfs(char *name) { char kname[64]; if(fskind != Kfs) return; if(name[0]) snprint(kname, sizeof kname, "/srv/kfs.%s", name); else strcpy(kname, "/srv/kfs"); sfd = open(kname, ORDWR); if(sfd < 0){ fprint(2, "can't open %q\n", kname); exits("open /srv/kfs"); } if(mount(sfd, -1, "/n/kfs", MREPL|MCREATE, "") < 0){ fprint(2, "can't mount kfs on /n/kfs\n"); exits("mount kfs"); } close(sfd); strcat(kname, ".cmd"); sfd = open(kname, ORDWR); if(sfd < 0){ fprint(2, "can't open %q\n", kname); exits("open /srv/kfs"); } } void kfscmd(char *cmd) { char buf[4*1024]; int n; if(fskind != Kfs) return; if(write(sfd, cmd, strlen(cmd)) != strlen(cmd)){ fprint(2, "%q: error writing %q: %r", prog, cmd); return; } for(;;){ n = read(sfd, buf, sizeof buf - 1); if(n <= 0) return; buf[n] = '\0'; if(strcmp(buf, "done") == 0 || strcmp(buf, "success") == 0) return; if(strcmp(buf, "unknown command") == 0){ fprint(2, "%q: command %q not recognized\n", prog, cmd); return; } } } void * emalloc(ulong n) { void *p; if((p = malloc(n)) == 0) error("out of memory"); return p; } void error(char *fmt, ...) { char buf[1024]; va_list arg; sprint(buf, "%q: %q:%d: ", prog, proto, lineno); va_start(arg, fmt); vseprint(buf+strlen(buf), buf+sizeof(buf), fmt, arg); va_end(arg); fprint(2, "%s\n", buf); kfscmd("disallow"); kfscmd("sync"); exits(0); } void warn(char *fmt, ...) { char buf[1024]; va_list arg; sprint(buf, "%q: %q:%d: ", prog, proto, lineno); va_start(arg, fmt); vseprint(buf+strlen(buf), buf+sizeof(buf), fmt, arg); va_end(arg); fprint(2, "%s\n", buf); } void printfile(File *f) { if(f->old) fprint(2, "%q from %q %q %q %lo\n", f->new, f->old, f->uid, f->gid, f->mode); else fprint(2, "%q %q %q %lo\n", f->new, f->uid, f->gid, f->mode); } void usage(void) { fprint(2, "usage: %q [-abkprvx] [-d root] [-n name] [-s source] [-u users] [-z n] proto ...\n", prog); exits("usage"); }