/* * This file is part of the UCB release of Plan 9. It is subject to the license * terms in the LICENSE file found in the top-level directory of this * distribution and at http://akaros.cs.berkeley.edu/files/Plan9License. No * part of the UCB release of Plan 9, including this file, may be copied, * modified, propagated, or distributed except according to the terms contained * in the LICENSE file. */ /* Portions of this file are Copyright (C) 2015-2018 Giacomo Tesio * See /doc/license/gpl-2.0.txt for details about the licensing. */ /* Portions of this file are Copyright (C) 9front's team. * See /doc/license/9front-mit for details about the licensing. * See http://git.9front.org/plan9front/plan9front/HEAD/info.html for a list of authors. */ #include #include #include #include #include #include enum{ LEN = 4096, /* * types of destination file sytems */ Fs = 0, Archive, }; void protowarn(char *msg, void *); void protoenum(char *new, char *old, Dir *d, void *); void arch(Dir*); void copy(Dir*); void error(char *, ...); void mkdir(Dir*); int uptodate(Dir*, char*); void usage(void); void warn(char *, ...); Biobufhdr bout; /* stdout when writing archive */ uint8_t boutbuf[2*LEN]; char newfile[LEN]; char oldfile[LEN]; char *proto; char *cputype; char *oldroot; char *newroot; char *prog = "mkfs"; char *buf; char *zbuf; int buflen = 1024-8; int verb; int modes; int ream; int debug; int xflag; int oflag; int sfd; int fskind; /* Fs, Archive */ int setuid; /* on Fs: set uid and gid? */ char *user; void main(int argc, char **argv) { int i, errs; quotefmtinstall(); user = getuser(); oldroot = ""; newroot = "/n/newfs"; fskind = Fs; ARGBEGIN{ case 'a': fskind = Archive; newroot = ""; Binits(&bout, 1, OWRITE, boutbuf, sizeof boutbuf); break; case 'd': if(fskind != Fs) { fprint(2, "cannot use -d with -a\n"); usage(); } fskind = Fs; newroot = EARGF(usage()); break; case 'D': debug = 1; break; case 'p': modes = 1; break; case 'r': ream = 1; break; case 's': oldroot = EARGF(usage()); break; case 'U': setuid = 1; break; case 'v': verb = 1; break; case 'o': oflag = 1; break; case 'x': xflag = 1; break; case 'z': buflen = atoi(EARGF(usage()))-8; break; default: usage(); }ARGEND if(!argc) usage(); if((xflag || oflag) && fskind != Archive){ fprint(2, "cannot use -x and -o without -a\n"); usage(); } buf = malloc(buflen); zbuf = malloc(buflen); memset(zbuf, 0, buflen); cputype = getenv(ENV_CPUTYPE); if(cputype == 0) cputype = "386"; errs = 0; for(i = 0; i < argc; i++){ proto = argv[i]; fprint(2, "processing %q\n", proto); if(rdproto(proto, oldroot, protoenum, protowarn, nil) < 0){ fprint(2, "%q: can't open %q: skipping\n", prog, proto); errs++; continue; } } fprint(2, "file system made\n"); if(errs) exits("skipped protos"); if(fskind == Archive){ if(!xflag && !oflag) Bprint(&bout, "end of archive\n"); Bterm(&bout); } exits(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; int64_t tot, len; Dir nd; f = sys_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 = ocreate(cptmp, OWRITE, 0666); if(t < 0){ warn("can't create %q: %r", newfile); sys_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 = jehanne_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(sys_seek(t, n, 1) < 0) error("can't write zeros to %q: %r", newfile); needwrite = 1; }else{ if(jehanne_write(t, buf, n) < n) error("can't write %q: %r", newfile); needwrite = 0; } } sys_close(f); if(needwrite){ if(sys_seek(t, -1, 1) < 0 || jehanne_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; sys_remove(newfile); nulldir(&nd); nd.mode = d->mode; 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); sys_close(t); } void mkdir(Dir *d) { Dir *d1; Dir nd; int fd; if(fskind == Archive){ arch(d); return; } fd = ocreate(newfile, OREAD, d->mode); nulldir(&nd); nd.mode = d->mode; nd.gid = d->gid; nd.mtime = d->mtime; if(fd < 0){ if((d1 = dirstat(newfile)) == nil || !(d1->mode & DMDIR)){ free(d1); error("can't create %q", 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); sys_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); } void protowarn(char *msg, void * _) { warn("%s", msg); } void protoenum(char *new, char *old, Dir *d, void * _) { Dir nd; sprint(newfile, "%s%s", newroot, new); sprint(oldfile, "%s", old); if(oflag){ if(!(d->mode & DMDIR)) Bprint(&bout, "%q\n", cleanname(oldfile)); return; } if(xflag){ Bprint(&bout, "%q\t%ld\t%lld\n", new, d->mtime, d->length); return; } if(verb && (fskind == Archive || ream)) fprint(2, "%q\n", new); if(fskind == Fs && !setuid){ d->uid = ""; d->gid = ""; } if(!uptodate(d, newfile)){ if(verb && (fskind != Archive && ream == 0)) fprint(2, "%q\n", new); if(d->mode & DMDIR) mkdir(d); else copy(d); }else if(modes){ nulldir(&nd); nd.mode = d->mode; nd.gid = d->gid; nd.mtime = d->mtime; if(verb && (fskind != Archive && ream == 0)) fprint(2, "%q\n", new); if(dirwstat(newfile, &nd) < 0) warn("can't set modes for %q: %r", new); nulldir(&nd); nd.uid = d->uid; dirwstat(newfile, &nd); } } void error(char *fmt, ...) { char buf[1024]; va_list arg; sprint(buf, "%q: %q: ", prog, proto); va_start(arg, fmt); vseprint(buf+strlen(buf), buf+sizeof(buf), fmt, arg); va_end(arg); fprint(2, "%s\n", buf); exits(0); } void warn(char *fmt, ...) { char buf[1024]; va_list arg; sprint(buf, "%q: %q: ", prog, proto); va_start(arg, fmt); vseprint(buf+strlen(buf), buf+sizeof(buf), fmt, arg); va_end(arg); fprint(2, "%s\n", buf); } void usage(void) { fprint(2, "usage: %q [-adprvoxUD] [-d root] [-s source] [-z n] proto ...\n", prog); exits("usage"); }