<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
"http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv=Content-Type content="text/html; charset=utf8">
<title>/usr/web/sources/contrib/cinap_lenrek/tftpd.c - Plan 9 from Bell Labs</title>
<!-- THIS FILE IS AUTOMATICALLY GENERATED. -->
<!-- EDIT sources.tr INSTEAD. -->
</meta>
</head>
<body>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<p style="line-height: 1.2em; margin-left: 1.00in; text-indent: 0.00in; margin-right: 1.00in; margin-top: 0; margin-bottom: 0; text-align: center;">
<span style="font-size: 10pt"><a href="/plan9/">Plan 9 from Bell Labs</a>&rsquo;s /usr/web/sources/contrib/cinap_lenrek/tftpd.c</span></p>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<center><font size=-1>
Copyright © 2009 Alcatel-Lucent.<br />
Distributed under the
<a href="/plan9/license.html">Lucent Public License version 1.02</a>.
<br />
<a href="/plan9/download.html">Download the Plan 9 distribution.</a>
</font>
</center>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<table width="100%" cellspacing=0 border=0><tr><td align="center">
<table cellspacing=0 cellpadding=5 bgcolor="#eeeeff"><tr><td align="left">
<pre>
<!-- END HEADER -->
#include &lt;u.h&gt;
#include &lt;libc.h&gt;
#include &lt;auth.h&gt;
#include &lt;bio.h&gt;
#include &lt;ip.h&gt;
#include &lt;ndb.h&gt;

enum
{
       Maxpath=        128,
       Maxerr=         256,
};

int     dbg;
int     restricted;
void    sendfile(int, char*, char*);
void    recvfile(int, char*, char*);
void    nak(int, int, char*);
void    oack(int, char*, char*);
void    ack(int, ushort);
void    clrcon(void);
void    setuser(void);
char*   sunkernel(char*);
void    remoteaddr(char*, char*, int);
void    doserve(int);

char    bigbuf[32768];
char    raddr[64];

char    *dir = "/lib/tftpd";
char    *dirsl;
int     dirsllen;
char    flog[] = "ipboot";
char    net[Maxpath];

enum
{
       Tftp_READ       = 1,
       Tftp_WRITE      = 2,
       Tftp_DATA       = 3,
       Tftp_ACK        = 4,
       Tftp_ERROR      = 5,
       Tftp_OACK       = 6,
       Segsize         = 512,
};

void
usage(void)
{
       fprint(2, "usage: %s [-dr] [-h homedir] [-s svc] [-x netmtpt]\n",
               argv0);
       exits("usage");
}


int
getlist(char *buf, int nbuf, char **list, int nlist)
{
       int s, c;
       char *p;

       p = nil;
       c = 0;
       s = 0;
       while(nbuf-- &gt; 0){
               switch(s){
               case 0:
                       p = buf;
                       s = 1;

               case 1:
                       if(*buf == '\0'){
                               s = 0;
                               if(--nlist &lt;= 0)
                                       goto outout;
                               *list = p;
                               list++;
                               c++;
                       }
                       break;
               }
               buf++;
       }
outout:
       *list = nil;
       return c;
}

void
main(int argc, char **argv)
{
       char buf[64];
       char adir[64], ldir[64];
       int cfd, lcfd, dfd;
       char *p, *svc = "69";

       setnetmtpt(net, sizeof(net), nil);
       ARGBEGIN{
       case 'd':
               dbg++;
               break;
       case 'h':
               dir = ARGF();
               break;
       case 'r':
               restricted = 1;
               break;
       case 's':
               svc = EARGF(usage());
               break;
       case 'x':
               p = ARGF();
               if(p == nil)
                       usage();
               setnetmtpt(net, sizeof(net), p);
               break;
       default:
               usage();
       }ARGEND

       snprint(buf, sizeof buf, "%s/", dir);
       dirsl = strdup(buf);
       dirsllen = strlen(dirsl);

       fmtinstall('E', eipfmt);
       fmtinstall('I', eipfmt);

       if(chdir(dir) &lt; 0)
               sysfatal("can't get to directory %s: %r", dir);

       if(!dbg)
               switch(rfork(RFNOTEG|RFPROC|RFFDG)) {
               case -1:
                       sysfatal("fork: %r");
               case 0:
                       break;
               default:
                       exits(0);
               }

       snprint(buf, sizeof buf, "%s/udp!*!%s", net, svc);
       cfd = announce(buf, adir);
       if (cfd &lt; 0)
               sysfatal("announcing on %s: %r", buf);
       syslog(dbg, flog, "tftpd started on %s dir %s", buf, adir);
       setuser();
       for(;;) {
               lcfd = listen(adir, ldir);
               if(lcfd &lt; 0)
                       sysfatal("listening on %s: %r", adir);

               switch(fork()) {
               case -1:
                       sysfatal("fork: %r");
               case 0:
                       dfd = accept(lcfd, ldir);
                       if(dfd &lt; 0)
                               exits(0);
                       remoteaddr(ldir, raddr, sizeof(raddr));
                       syslog(0, flog, "tftp connection from %s dir %s",
                               raddr, ldir);
                       doserve(dfd);
                       exits("done");
                       break;
               default:
                       close(lcfd);
                       continue;
               }
       }
}

void
doserve(int fd)
{
       int dlen;
       char *field[10];
       char tmp[128];
       int nfield;
       char *name, *mode;
       char **opt;
       short op;
       Dir *dir;

       dlen = read(fd, bigbuf, sizeof(bigbuf));
       if(dlen &lt; 0)
               sysfatal("listen read: %r");

       op = (bigbuf[0]&lt;&lt;8) | bigbuf[1];
       dlen -= 2;

       nfield = getlist(bigbuf+2, dlen, field, sizeof(field)/sizeof(field[0]));
       if(nfield &lt; 2){
               nak(fd, 0, "bad request");
               close(fd);
               syslog(dbg, flog, "bad request from %s", raddr);
               return;
       }
       name = field[0];
       mode = field[1];
       if(op != Tftp_READ &amp;&amp; op != Tftp_WRITE) {
               nak(fd, 4, "Illegal TFTP operation");
               close(fd);
               syslog(dbg, flog, "bad request %d %s", op, raddr);
               return;
       }

       /*
       if(restricted){
               if(*name == '#' ||
                 strncmp(name, "../", 3)==0 || strstr(name) ||
                 (*name == '/' &amp;&amp; strncmp(name, dirsl, dirsllen)!=0)){
                       nak(fd, 4, "Permission denied");
                       close(fd);
                       syslog(dbg, flog, "bad request %d from %s file %s", op, raddr, name);
                       return;
               }
       }*/

       name = sunkernel(name);
       if(name == 0){
               nak(fd, 0, "file not found");
               close(fd);
               syslog(dbg, flog, "file %s not found for %s", name, raddr);
               return;
       }

       if((nfield&gt;2) &amp;&amp; ((nfield%2)==0)){
               /* handle options */
               nfield -= 2;
               nfield /= 2;
               opt = &amp;field[2];
               while(nfield-- &gt; 0){
                       if(cistrcmp(opt[0], "tsize")==0){
                               if(op == Tftp_READ){
                                       dir=dirstat(name);
                                       if(dir != nil){
                                               snprint(tmp, sizeof(tmp), "%lld", dir-&gt;length);
                                               free(dir);
                                               oack(fd, opt[0], tmp);
                                       }
                               }
                       }
                       opt+=2;
               }
       }

       if(op == Tftp_READ)
               sendfile(fd, name, mode);
       else
               recvfile(fd, name, mode);
}

void
catcher(void *junk, char *msg)
{
       USED(junk);

       if(strncmp(msg, "exit", 4) == 0)
               noted(NDFLT);
       noted(NCONT);
}

void
sendfile(int fd, char *name, char *mode)
{
       int file;
       uchar buf[Segsize+4];
       uchar ack[1024];
       char errbuf[Maxerr];
       int ackblock, block, ret;
       int rexmit, n, al, txtry, rxl;
       short op;

       syslog(dbg, flog, "send file '%s' %s to %s", name, mode, raddr);
       if(name == 0){
               nak(fd, 0, "not in our database");
               return;
       }

       notify(catcher);

       file = open(name, OREAD);
       if(file &lt; 0) {
               errstr(errbuf, sizeof errbuf);
               nak(fd, 0, errbuf);
               return;
       }
       block = 0;
       rexmit = 0;
       n = 0;
       for(txtry = 0; txtry &lt; 5;) {
               if(rexmit == 0) {
                       block++;
                       buf[0] = 0;
                       buf[1] = Tftp_DATA;
                       buf[2] = block&gt;&gt;8;
                       buf[3] = block;
                       n = read(file, buf+4, Segsize);
                       if(n &lt; 0) {
                               errstr(errbuf, sizeof errbuf);
                               nak(fd, 0, errbuf);
                               return;
                       }
                       txtry = 0;
               }
               else {
                       syslog(dbg, flog, "rexmit %d %s:%d to %s", 4+n, name, block, raddr);
                       txtry++;
               }

               ret = write(fd, buf, 4+n);
               if(ret &lt; 0)
                       sysfatal("tftpd: network write error: %r");

               for(rxl = 0; rxl &lt; 10; rxl++) {
                       rexmit = 0;
                       alarm(500);
                       al = read(fd, ack, sizeof(ack));
                       alarm(0);
                       if(al &lt; 0) {
                               rexmit = 1;
                               break;
                       }
                       op = ack[0]&lt;&lt;8|ack[1];
                       if(op == Tftp_ERROR)
                               goto error;
                       ackblock = ack[2]&lt;&lt;8|ack[3];
                       if(ackblock == block)
                               break;
                       if(ackblock == 0xffff) {
                               rexmit = 1;
                               break;
                       }
               }
               if(ret != Segsize+4 &amp;&amp; rexmit == 0)
                       break;
       }
error:
       close(fd);
       close(file);
}

void
recvfile(int fd, char *name, char *mode)
{
       ushort op, block, inblock;
       uchar buf[Segsize+8];
       char errbuf[Maxerr];
       int n, ret, file;

       syslog(dbg, flog, "receive file '%s' %s from %s", name, mode, raddr);

       file = create(name, OWRITE, 0666);
       if(file &lt; 0) {
               errstr(errbuf, sizeof errbuf);
               nak(fd, 0, errbuf);
               return;
       }

       block = 0;
       ack(fd, block);
       block++;

       for(;;) {
               alarm(15000);
               n = read(fd, buf, sizeof(buf));
               alarm(0);
               if(n &lt; 0)
                       goto error;
               op = buf[0]&lt;&lt;8|buf[1];
               if(op == Tftp_ERROR)
                       goto error;

               n -= 4;
               inblock = buf[2]&lt;&lt;8|buf[3];
               if(op == Tftp_DATA) {
                       if(inblock == block) {
                               ret = write(file, buf, n);
                               if(ret &lt; 0) {
                                       errstr(errbuf, sizeof errbuf);
                                       nak(fd, 0, errbuf);
                                       goto error;
                               }
                               ack(fd, block);
                               block++;
                       }
                       ack(fd, 0xffff);
               }
       }
error:
       close(file);
}

void
ack(int fd, ushort block)
{
       uchar ack[4];
       int n;

       ack[0] = 0;
       ack[1] = Tftp_ACK;
       ack[2] = block&gt;&gt;8;
       ack[3] = block;

       n = write(fd, ack, 4);
       if(n &lt; 0)
               sysfatal("network write: %r");
}

void
oack(int fd, char *option, char *value)
{
       char buf[128];
       int l1, l2, n;

       buf[0] = 0;
       buf[1] = Tftp_OACK;

       l1=strlen(option);
       l2=strlen(value);
       n = 2 + l1+1 + l2+1;

       if(n &lt; sizeof(buf)){
               memcpy(buf+2, option, l1+1);
               memcpy(buf+2+l1+1, value, l2+1);
               n = write(fd, buf, n);
               if(n &lt; 0)
                       sysfatal("write oack: %r");
       }
}

void
nak(int fd, int code, char *msg)
{
       char buf[128];
       int n;

       buf[0] = 0;
       buf[1] = Tftp_ERROR;
       buf[2] = 0;
       buf[3] = code;
       strcpy(buf+4, msg);
       n = strlen(msg) + 4 + 1;
       n = write(fd, buf, n);
       if(n &lt; 0)
               sysfatal("write nak: %r");
}

void
setuser(void)
{
       int fd;

       fd = open("#c/user", OWRITE);
       if(fd &lt; 0 || write(fd, "none", strlen("none")) &lt; 0)
               sysfatal("can't become none: %r");
       close(fd);
       if(newns("none", nil) &lt; 0)
               sysfatal("can't build namespace: %r");
       bind(dir, "/", MAFTER);
}

char*
lookup(char *sattr, char *sval, char *tattr, char *tval, int len)
{
       static Ndb *db;
       char *attrs[1];
       Ndbtuple *t;

       if(db == nil)
               db = ndbopen(0);
       if(db == nil)
               return nil;

       if(sattr == nil)
               sattr = ipattr(sval);

       attrs[0] = tattr;
       t = ndbipinfo(db, sattr, sval, attrs, 1);
       if(t == nil)
               return nil;
       strncpy(tval, t-&gt;val, len);
       tval[len-1] = 0;
       ndbfree(t);
       return tval;
}

/*
*  for sun kernel boots, replace the requested file name with
*  a one from our database.  If the database doesn't specify a file,
*  don't answer.
*/
char*
sunkernel(char *name)
{
       ulong addr;
       uchar v4[IPv4addrlen];
       uchar v6[IPaddrlen];
       char buf[256];
       char ipbuf[128];
       char *suffix;

       addr = strtoul(name, &amp;suffix, 16);
       if(suffix-name != 8 || (strcmp(suffix, "") != 0 &amp;&amp; strcmp(suffix, ".SUN") != 0))
               return name;

       v4[0] = addr&gt;&gt;24;
       v4[1] = addr&gt;&gt;16;
       v4[2] = addr&gt;&gt;8;
       v4[3] = addr;
       v4tov6(v6, v4);
       sprint(ipbuf, "%I", v6);
       return lookup("ip", ipbuf, "bootf", buf, sizeof buf);
}

void
remoteaddr(char *dir, char *raddr, int len)
{
       char buf[64];
       int fd, n;

       snprint(buf, sizeof(buf), "%s/remote", dir);
       fd = open(buf, OREAD);
       if(fd &lt; 0){
               snprint(raddr, sizeof(raddr), "unknown");
               return;
       }
       n = read(fd, raddr, len-1);
       close(fd);
       if(n &lt;= 0){
               snprint(raddr, sizeof(raddr), "unknown");
               return;
       }
       if(n &gt; 0)
               n--;
       raddr[n] = 0;
}
<!-- BEGIN TAIL -->
</pre>
</td></tr></table>
</td></tr></table>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<p style="line-height: 1.2em; margin-left: 1.00in; text-indent: 0.00in; margin-right: 1.00in; margin-top: 0; margin-bottom: 0; text-align: center;">
<span style="font-size: 10pt"></span></p>
<p style="margin-top: 0; margin-bottom: 0.50in"></p>
<p style="margin-top: 0; margin-bottom: 0.33in"></p>
<center><table border="0"><tr>
<td valign="middle"><a href="http://www.alcatel-lucent.com/"><img border="0" src="/plan9/img/logo_ft.gif" alt="Bell Labs" />
</a></td>
<td valign="middle"><a href="http://www.opensource.org"><img border="0" alt="OSI certified" src="/plan9/img/osi-certified-60x50.gif" />
</a></td>
<td><img style="padding-right: 45px;" alt="Powered by Plan 9" src="/plan9/img/power36.gif" />
</td>
</tr></table></center>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<center>
<span style="font-size: 10pt">(<a href="/plan9/">Return to Plan 9 Home Page</a>)</span>
</center>
<p style="margin-top: 0; margin-bottom: 0.17in"></p>
<center><font size=-1>
<span style="font-size: 10pt"><a href="http://www.lucent.com/copyright.html">Copyright</a></span>
<span style="font-size: 10pt">© 2009 Alcatel-Lucent.</span>
<span style="font-size: 10pt">All Rights Reserved.</span>
<br />
<span style="font-size: 10pt">Comments to</span>
<span style="font-size: 10pt"><a href="mailto:[email protected]">[email protected]</a>.</span>
</font></center>
</body>
</html>