From 37da2899f40661e3e9631e497da8dc59b971cbd0 Mon Sep 17 00:00:00 2001 From: "Charles.Forsyth" Date: Fri, 22 Dec 2006 17:07:39 +0000 Subject: 20060303a --- appl/cmd/puttar.b | 183 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 183 insertions(+) create mode 100644 appl/cmd/puttar.b (limited to 'appl/cmd/puttar.b') diff --git a/appl/cmd/puttar.b b/appl/cmd/puttar.b new file mode 100644 index 00000000..de67105f --- /dev/null +++ b/appl/cmd/puttar.b @@ -0,0 +1,183 @@ +# read list of pathnames on stdin, write POSIX.1 tar on stdout +# Copyright(c)1996 Lucent Technologies. All Rights Reserved. +# 22 Dec 1996 ehg@bell-labs.com + +implement puttar; +include "sys.m"; + sys: Sys; + print, sprint, fprint: import sys; + stdout, stderr: ref sys->FD; +include "draw.m"; +include "bufio.m"; + bufio: Bufio; + Iobuf: import bufio; + +puttar: module{ + init: fn(nil: ref Draw->Context, nil: list of string); +}; + +Warning(mess: string) +{ + fprint(stderr,"warning: puttar: %s: %r\n",mess); +} + +Error(mess: string){ + fprint(stderr,"puttar: %s: %r\n",mess); + exit; +} + +TBLOCK: con 512; # tar logical blocksize +NBLOCK: con 20; # blocking factor for efficient write +tarbuf := array[NBLOCK*TBLOCK] of byte; # for output +nblock := 0; # how many blocks of data are in tarbuf + +flushblocks(){ + if(nblock<=0) return; + if(nblockwrite(stdout,tarbuf,NBLOCK*TBLOCK); + if(i!=NBLOCK*TBLOCK) + Error("write error"); + nblock = 0; +} + +putblock(data:array of byte){ + # all writes are done through here, so we can guarantee + # 10kbyte blocks if writing to tape device + if(len data!=TBLOCK) + Error("putblock wants TBLOCK chunks"); + tarbuf[nblock*TBLOCK:] = data; + nblock++; + if(nblock>=NBLOCK) + flushblocks(); +} + +packname(hdr:array of byte, name:string){ + utf := array of byte name; + n := len utf; + if(n<=100){ + hdr[0:] = utf; + return; + } + for(i:=n-101; i 100 bytes",name)); + if(i>155) Error(sprint("%s too long\n",name)); + hdr[0:] = utf[i+1:n]; + hdr[345:] = utf[0:i]; # tar supplies implicit slash +} + +octal(width:int, val:int):array of byte{ + octal := array of byte "01234567"; + a := array[width] of byte; + for(i:=width-1; i>=0; i--){ + a[i] = octal[val&7]; + val >>= 3; + } + return a; +} + +chksum(hdr: array of byte):int{ + sum := 0; + for(i:=0; iFD; + + (rc,stat) := sys->stat(file); + if(rc<0){ Warning(sprint("cannot stat %s",file)); return; }; + ifile = sys->open(file,sys->OREAD); + if(ifile==nil) Error(sprint("cannot open %s",file)); + hdr[0:] = zeros; + packname(hdr,file); + hdr[100:] = octal(7,stat.mode&8r777); + hdr[108:] = octal(7,1); + hdr[116:] = octal(7,1); + hdr[124:] = octal(11,int stat.length); + hdr[136:] = octal(11,stat.mtime); + hdr[148:] = array of byte " "; # for chksum + hdr[156] = byte '0'; + if(stat.mode&Sys->DMDIR) hdr[156] = byte '5'; + hdr[257:] = array of byte "ustar"; + hdr[263:] = array of byte "00"; + hdr[265:] = array of byte stat.uid; # assumes len uid<=32 + hdr[297:] = array of byte stat.gid; + hdr[329:] = octal(8,stat.dev); + hdr[337:] = octal(8,int stat.qid.path); + hdr[148:] = octal(7,chksum(hdr)); + hdr[155] = byte 0; + putblock(hdr); + for(bytes := int stat.length; bytes>0;){ + n := len ibuf; if(n>bytes) n = bytes; # min + if(sys->read(ifile,ibuf,n)!=n) + Error(sprint("read error on %s",file)); + nb := (n+TBLOCK-1)/TBLOCK; + fill := nb*TBLOCK; + for(i:=n; istat(file); + if (ok < 0){ + Warning(sprint("cannot stat %s", file)); + return; + } + if (dir.mode & Sys->DMDIR) { + fd := sys->open(file, sys->OREAD); + if (fd == nil) + Error(sprint("cannot open %s", file)); + for (;;) { + (n, d) := sys->dirread(fd); + if (n <= 0) + break; + for (i := 0; i < n; i++) { + if (file[len file - 1] == '/') + rtar(file + d[i].name); + else + rtar(file + "/" + d[i].name); + } + } + } +} + +init(nil: ref Draw->Context, args: list of string){ + sys = load Sys Sys->PATH; + bufio = load Bufio Bufio->PATH; + stdout = sys->fildes(1); + stderr = sys->fildes(2); + + hdr = array[TBLOCK] of byte; + zeros = array[TBLOCK] of {* => byte 0}; + ibuf = array[len tarbuf] of byte; + + if (tl args == nil) { + stdin := bufio->fopen(sys->fildes(0),bufio->OREAD); + if(stdin==nil) Error("can't fopen stdin"); + while((file := stdin.gets('\n'))!=nil){ + if(file[len file-1]=='\n') file = file[0:len file-1]; + tar(file); + } + } + else { + for (args = tl args; args != nil; args = tl args) + rtar(hd args); + } + putblock(zeros); +# putblock(zeros); # XXX is this necessary? + flushblocks(); +} -- cgit v1.2.3