#include <u.h> #include <libc.h> #include <regexp.h> #include <thread.h> #include <fcall.h> int debug; int dfd; int srvfd; int netfd; int srv_to_net[2]; int net_to_srv[2]; char *srv; char *addr; char *ns; void shuffle(void *arg); int post(char *srv); void remoteside(char *ns, char *srv); int call(char *rsys, char *ns, char *srv); void* emalloc(int size); void runproc(void *arg); char *REXEXEC = "ssh"; char *prog = "import"; enum { Stack= 32*1024, }; void usage(void) { fprint(2, "usage: %s [-df] [-s service] [-n remote-ns] [-p remote-prog] remote-system\n", argv0); threadexitsall("usage"); } void fatal(char *fmt, ...) { char buf[256]; va_list arg; va_start(arg, fmt); vseprint(buf, buf+sizeof buf, fmt, arg); va_end(arg); fprint(2, "%s: %s\n", argv0 ? argv0 : "<prog>", buf); threadexitsall("fatal"); } void threadmain(int argc, char *argv[]) { int dofork; int rem; dofork = 1; rem = 0; ns = nil; srv = "plumb"; ARGBEGIN{ case 'd': debug = 1; break; case 'f': dofork = 0; break; case 'n': // name of remote namespace ns = EARGF(usage()); break; case 'p': prog = EARGF(usage()); break; case 's': // name of service srv = EARGF(usage()); break; case 'R': rem = 1; break; }ARGEND if(debug){ char *dbgfile; if(rem) dbgfile = smprint("/tmp/%s.export.debug", getuser()); else dbgfile = smprint("/tmp/%s.import.debug", getuser()); dfd = create(dbgfile, OWRITE, 0664); free(dbgfile); fmtinstall('F', fcallfmt); } // is this the remote side? if(rem){ if(srv == nil) fatal("-R requires -s"); remoteside(ns, srv); threadexitsall(0); } if(argc != 1) usage(); addr = argv[0]; if(dofork) proccreate(runproc, nil, Stack); else runproc(nil); } void runproc(void *arg) { USED(arg); // start a loal service and connect to remote service srvfd = post(srv); netfd = call(addr, ns, srv); // threads to shuffle messages each way srv_to_net[0] = srvfd; srv_to_net[1] = netfd; proccreate(shuffle, srv_to_net, Stack); net_to_srv[0] = netfd; net_to_srv[1] = srvfd; shuffle(net_to_srv); } /* post a local service */ int post(char *srv) { int p[2]; if(pipe(p) < 0) fatal("can't create pipe: %r"); /* 0 will be server end, 1 will be client end */ if(post9pservice(p[1], "plumb") < 0) fatal("post9pservice plumb: %r"); close(p[1]); return p[0]; } /* start a stub on the remote server */ int call(char *rsys, char *ns, char *srv) { int p[2]; int ac; char *av[12]; char buf[2]; if(pipe(p) < 0) fatal("can't create pipe: %r"); ac = 0; av[ac++] = REXEXEC; av[ac++] = rsys; av[ac++] = prog; if(debug) av[ac++] = "-d"; av[ac++] = "-R"; if(ns != nil){ av[ac++] = "-n"; av[ac++] = ns; } av[ac++] = "-s"; av[ac++] = srv; av[ac] = 0; if(debug){ fprint(dfd, "execing "); for(ac = 0; av[ac]; ac++) fprint(dfd, " %s", av[ac]); fprint(dfd, "\n"); } switch(fork()){ case -1: fatal("%r"); case 0: dup(p[1], 0); dup(p[1], 1); close(p[0]); close(p[1]); execvp(REXEXEC, av); fatal("can't exec %s", REXEXEC); default: break; } close(p[1]); // ignore crap that might come out of the .profile // keep reading till we have an "OK" if(read(p[0], &buf[0], 1) != 1) fatal("EOF"); for(;;){ if(read(p[0], &buf[1], 1) != 1) fatal("EOF"); if(strncmp(buf, "OK", 2) == 0) break; buf[0] = buf[1]; } if(debug) fprint(dfd, "got OK\n"); return p[0]; } enum { BLEN=16*1024 }; void shuffle(void *arg) { int *fd; char *buf, *tbuf; int n; Fcall *t; fd = (int*)arg; buf = emalloc(BLEN+1); t = nil; tbuf = nil; for(;;){ n = read9pmsg(fd[0], buf, BLEN); if(n <= 0){ if(debug) fprint(dfd, "%d->%d read returns %d: %r\n", fd[0], fd[1], n); break; } if(debug){ if(t == nil) t = emalloc(sizeof(Fcall)); if(tbuf == nil) tbuf = emalloc(BLEN+1); memmove(tbuf, buf, n); // because convM2S is destructive if(convM2S((uchar*)tbuf, n, t) != n) fprint(dfd, "%d->%d convert error in convM2S", fd[0], fd[1]); else fprint(dfd, "%d->%d %F\n", fd[0], fd[1], t); } if(write(fd[1], buf, n) != n) break; } } void remoteside(char *ns, char *srv) { int srv_to_net[2]; int net_to_srv[2]; char *addr; int srvfd; if(ns == nil) ns = getns(); addr = smprint("unix!%s/%s", ns, srv); if(addr == nil) fatal("%r"); if(debug) fprint(dfd, "remoteside starting %s\n", addr); srvfd = dial(addr, 0, 0, 0); if(srvfd < 0) fatal("dial %s: %r", addr); if(debug) fprint(dfd, "remoteside dial %s succeeded\n", addr); fcntl(srvfd, F_SETFL, FD_CLOEXEC); write(1, "OK", 2); /* threads to shuffle messages each way */ srv_to_net[0] = srvfd; srv_to_net[1] = 1; proccreate(shuffle, srv_to_net, Stack); net_to_srv[0] = 0; net_to_srv[1] = srvfd; shuffle(net_to_srv); threadexitsall(0); } void* emalloc(int size) { void *x; x = malloc(size); if(x == nil) fatal("allocation fails: %r"); return x; }