#include <u.h>
#include <libc.h>
#include <bio.h>
#include <ndb.h>
#include <fcall.h>
enum
{
Maxpath= 128,
};
typedef struct Endpoints Endpoints;
struct Endpoints
{
char *net;
char *lsys;
char *lserv;
char *rsys;
char *rserv;
};
void xfer(int, int);
void xfer9p(int, int);
Endpoints* getendpoints(char*);
void freeendpoints(Endpoints*);
char* iptomac(char*, char*);
int macok(char*);
char* keyfile;
void
usage(void)
{
fprint(2, "usage: vernam [-9] [-a addr] [-m netdir] keyfile addr\n");
exits("usage");
}
void
main(int argc, char **argv)
{
char *altaddr, *checkmac, *mac;
int fd, fd0, fd1;
void (*x)(int, int);
Endpoints *ep;
checkmac = nil;
altaddr = nil;
x = xfer;
ARGBEGIN{
case '9':
x = xfer9p;
break;
case 'a':
altaddr = EARGF(usage());
break;
case 'm':
checkmac = EARGF(usage());
break;
default:
usage();
}ARGEND;
if(argc != 2)
usage();
if(checkmac){
ep = getendpoints(checkmac);
mac = iptomac(ep->rsys, ep->net);
if(!macok(mac)){
syslog(0, "vernam", "badmac %s from %s!%s for %s!%s on %s",
mac, ep->rsys, ep->rserv, ep->lsys, ep->lserv, ep->net);
exits("bad mac");
}
}
keyfile = argv[0];
fd0 = 0;
fd1 = 1;
if(altaddr){
fd0 = dial(altaddr, 0, 0, 0);
if(fd0 < 0)
sysfatal("dial %s: %r", altaddr);
fd1 = fd0;
}
fd = dial(argv[1], 0, 0, 0);
if(fd < 0)
sysfatal("dial %s: %r", argv[1]);
rfork(RFNOTEG);
switch(fork()){
case -1:
fprint(2, "%s: fork: %r\n", argv0);
exits("dial");
case 0:
(*x)(fd0, fd);
break;
default:
(*x)(fd, fd1);
break;
}
postnote(PNGROUP, getpid(), "die yankee pig dog");
exits(0);
}
int
readkey(int keyfd, char* buf, int sz)
{
int i;
if((i = readn(keyfd, buf, sz)) < sz) {
/* We've run out of secrets. It's "one time", so bail out. */
syslog(0, "vernam", "%s exhausted: wanted %d, got %d; aborting",
keyfile, sz, i);
return -1;
/* VernamTunnel starts over, instead; I never got this working reliably.
* syslog(0, "vernam", "keyfile %s exhausted; reset", keyfile);
* seek(keyfd, i, 0);
* readkey(keyfd, buf+i, sz-i);
*/
}
return i;
}
void
xfer(int from, int to)
{
char buf[12*1024], xbuf[12*1024];
int n, i, keyfd;
keyfd = open(keyfile, OREAD);
if(keyfd < 0)
sysfatal("couldn't open key file %s: %r", keyfile);
while((n = read(from, buf, sizeof buf)) > 0) {
if(readkey(keyfd, xbuf, n) < 0)
break;
for(i=0; i<n; i++)
buf[i] ^= xbuf[i];
if(write(to, buf, n) < 0)
break;
}
}
void
xfer9p(int from, int to)
{
uchar *buf;
char xbuf[12*1024];
uint nbuf;
int i, n, keyfd;
nbuf = 256;
buf = malloc(nbuf);
if(buf == nil)
sysfatal("xfer: buf malloc %ud: %r", nbuf);
keyfd = open(keyfile, OREAD);
if(keyfd < 0)
sysfatal("couldn't open key file %s: %r", keyfile);
for(;;){
if(readn(from, buf, 4) != 4)
break;
n = GBIT32(buf);
if(n > nbuf){
nbuf = n+8192;
buf = realloc(buf, nbuf);
if(buf == nil)
sysfatal("xfer: realloc %ud: %r", nbuf);
}
if(readn(from, buf+4, n-4) != n-4)
break;
if(readkey(keyfd, xbuf+4, n-4) < 0)
break;
for(i=4; i<n; i++)
buf[i] ^= xbuf[i];
if(write(to, buf, n) != n){
sysfatal("oops: %r");
break;
}
}
}
void
getendpoint(char *dir, char *file, char **sysp, char **servp)
{
int fd, n;
char buf[Maxpath];
char *sys, *serv;
sys = serv = 0;
snprint(buf, sizeof buf, "%s/%s", dir, file);
fd = open(buf, OREAD);
if(fd >= 0){
n = read(fd, buf, sizeof(buf)-1);
if(n>0){
buf[n-1] = 0;
serv = strchr(buf, '!');
if(serv){
*serv++ = 0;
serv = strdup(serv);
}
sys = strdup(buf);
}
close(fd);
}
if(serv == 0)
serv = strdup("unknown");
if(sys == 0)
sys = strdup("unknown");
*servp = serv;
*sysp = sys;
}
Endpoints *
getendpoints(char *dir)
{
Endpoints *ep;
char *p;
ep = malloc(sizeof(*ep));
ep->net = strdup(dir);
p = strchr(ep->net+1, '/');
if(p == nil){
free(ep->net);
ep->net = "/net";
} else
*p = 0;
getendpoint(dir, "local", &ep->lsys, &ep->lserv);
getendpoint(dir, "remote", &ep->rsys, &ep->rserv);
return ep;
}
void
freeendpoints(Endpoints *ep)
{
free(ep->lsys);
free(ep->rsys);
free(ep->lserv);
free(ep->rserv);
free(ep);
}
char*
iptomac(char *ip, char *net)
{
char file[Maxpath];
Biobuf *b;
char *p;
char *f[5];
snprint(file, sizeof(file), "%s/arp", net);
b = Bopen(file, OREAD);
if(b == nil)
return nil;
while((p = Brdline(b, '\n')) != nil){
p[Blinelen(b)-1] = 0;
if(tokenize(p, f, nelem(f)) < 4)
continue;
if(strcmp(f[1], "OK") == 0
&& strcmp(f[2], ip) == 0){
p = strdup(f[3]);
Bterm(b);
return p;
}
}
Bterm(b);
return nil;
}
int
macok(char *mac)
{
char *p;
if(mac == nil)
return 0;
#ifdef PLAN9PORT
free(p = ndbgetvalue(nil, nil, "ether", mac, "trampok", nil));
#else
free(p = csgetvalue("/net", "ether", mac, "trampok", nil));
#endif
return !(p == nil);
}
|