summaryrefslogtreecommitdiff
path: root/sys/src/cmd/pkg/unpkg.c
diff options
context:
space:
mode:
authorcinap_lenrek <cinap_lenrek@centraldogma>2011-05-09 08:32:55 +0000
committercinap_lenrek <cinap_lenrek@centraldogma>2011-05-09 08:32:55 +0000
commit22ec6165bc7ffe68c82e1b14d6d77045c8c3605b (patch)
tree0a4e4baa2c5d445722ba723f37566ceec2018ac1 /sys/src/cmd/pkg/unpkg.c
parent26c2845917cb4661a6e04e3b41d52fdc1338a2c4 (diff)
parent28859a83f41b4c5231865cc556665f2060c70010 (diff)
downloadplan9front-22ec6165bc7ffe68c82e1b14d6d77045c8c3605b.tar.xz
merge
Diffstat (limited to 'sys/src/cmd/pkg/unpkg.c')
-rw-r--r--sys/src/cmd/pkg/unpkg.c100
1 files changed, 100 insertions, 0 deletions
diff --git a/sys/src/cmd/pkg/unpkg.c b/sys/src/cmd/pkg/unpkg.c
new file mode 100644
index 000000000..2c749e3b3
--- /dev/null
+++ b/sys/src/cmd/pkg/unpkg.c
@@ -0,0 +1,100 @@
+#include <u.h>
+#include <libc.h>
+#include <mp.h>
+#include <libsec.h>
+
+struct th {
+ char *name;
+ ulong perm;
+ ulong size;
+ char type;
+ char *user, *group;
+};
+
+static char *sndup(char* s, ulong n) {
+ char *d, *p;
+ p = memchr(s, 0, n);
+ if(p)
+ n = p-s;
+ d = malloc(n+1);
+ memcpy(d,s,n);
+ d[n] = 0;
+ return d;
+}
+
+
+int readheader(int fd, struct th* th) {
+ int i;
+ char b[512];
+ if(readn(fd, b, 512) != 512) return -1;
+
+ // Check for end of archive
+ for(i=0; i<512; i++) {
+ if(b[i]!=0) goto rhok;
+ }
+ if(readn(fd, b, 512) != 512) return -1;
+ for(i=0; i<512; i++) {
+ if(b[i]!=0) return -1;
+ }
+ return 0;
+
+ rhok:
+ th->name = sndup(b, 100);
+ th->perm = strtoul(b+100, nil, 8);
+ th->size = strtoul(b+124, nil, 8);
+ th->type = b[156];
+ th->user = sndup(b+265, 32);
+ th->group= sndup(b+297, 32);
+ return 1;
+}
+
+int main(void) {
+ while(1) {
+ struct th th;
+ ulong off;
+ uchar b[512];
+ DigestState *s;
+ int wfd;
+ int r = readheader(0, &th);
+ if(r <= 0) return r;
+
+ switch(th.type) {
+ case '5':
+ create(th.name, OREAD, DMDIR|th.perm);
+ break;
+ case '0': case 0:
+ print("A %s\n", th.name);
+ r = access(th.name, 0);
+ if(r == 0) {
+ print("File already exists: %s\n", th.name);
+ return -1;
+ }
+ if((wfd = create(th.name, OWRITE, th.perm)) < 0) {
+ print("Create failed: %s\n", th.name);
+ return -1;
+ }
+ s = nil;
+ for(off=0; off<th.size; off+=512) {
+ int n = th.size-off;
+ n = n<512 ? n : 512;
+ if(readn(0, b, 512) != 512) return -1;
+ if(write(wfd, b, n) != n) return -1;
+ s = sha1(b, n, nil, s);
+ }
+
+ uchar digest[20], hdigest[41];
+ sha1(nil, 0, digest, s);
+ enc16((char*)hdigest, 41, digest, 20);
+ fprint(2, "%s\t%s\n", th.name, hdigest);
+ close(wfd);
+ break;
+ default:
+ print("Unknown file type '%c'\n", th.type);
+ return -1;
+ }
+
+ free(th.name);
+ free(th.user);
+ free(th.group);
+ }
+}