1*9a747e4fSDavid du Colombier #include "ratfs.h"
2*9a747e4fSDavid du Colombier
3*9a747e4fSDavid du Colombier #define SRVFILE "/srv/ratify"
4*9a747e4fSDavid du Colombier #define MOUNTPOINT "/mail/ratify"
5*9a747e4fSDavid du Colombier #define CTLFILE "/mail/lib/blocked"
6*9a747e4fSDavid du Colombier #define CONFFILE "/mail/lib/smtpd.conf.ext"
7*9a747e4fSDavid du Colombier
8*9a747e4fSDavid du Colombier typedef struct Filetree Filetree;
9*9a747e4fSDavid du Colombier
10*9a747e4fSDavid du Colombier /* prototype file tree */
11*9a747e4fSDavid du Colombier struct Filetree
12*9a747e4fSDavid du Colombier {
13*9a747e4fSDavid du Colombier int level;
14*9a747e4fSDavid du Colombier char *name;
15*9a747e4fSDavid du Colombier ushort type;
16*9a747e4fSDavid du Colombier int mode;
17*9a747e4fSDavid du Colombier ulong qid;
18*9a747e4fSDavid du Colombier };
19*9a747e4fSDavid du Colombier
20*9a747e4fSDavid du Colombier /* names of first-level directories - must be in order of level*/
21*9a747e4fSDavid du Colombier Filetree filetree[] =
22*9a747e4fSDavid du Colombier {
23*9a747e4fSDavid du Colombier 0, "/", Directory, 0555|DMDIR, Qroot,
24*9a747e4fSDavid du Colombier 1, "allow", Addrdir, 0555|DMDIR, Qallow,
25*9a747e4fSDavid du Colombier 1, "delay", Addrdir, 0555|DMDIR, Qdelay,
26*9a747e4fSDavid du Colombier 1, "block", Addrdir, 0555|DMDIR, Qblock,
27*9a747e4fSDavid du Colombier 1, "dial", Addrdir, 0555|DMDIR, Qdial,
28*9a747e4fSDavid du Colombier 1, "deny", Addrdir, 0555|DMDIR, Qdeny,
29*9a747e4fSDavid du Colombier 1, "trusted", Trusted, 0777|DMDIR, Qtrusted, /* creation allowed */
30*9a747e4fSDavid du Colombier 1, "ctl", Ctlfile, 0222, Qctl,
31*9a747e4fSDavid du Colombier 2, "ip", IPaddr, 0555|DMDIR, Qaddr,
32*9a747e4fSDavid du Colombier 2, "account", Acctaddr, 0555|DMDIR, Qaddr,
33*9a747e4fSDavid du Colombier 0, 0, 0, 0, 0,
34*9a747e4fSDavid du Colombier
35*9a747e4fSDavid du Colombier };
36*9a747e4fSDavid du Colombier
37*9a747e4fSDavid du Colombier int debugfd = -1;
38*9a747e4fSDavid du Colombier int trustedqid = Qtrustedfile;
39*9a747e4fSDavid du Colombier char *ctlfile = CTLFILE;
40*9a747e4fSDavid du Colombier char *conffile = CONFFILE;
41*9a747e4fSDavid du Colombier
42*9a747e4fSDavid du Colombier #pragma varargck type "I" Cidraddr*
43*9a747e4fSDavid du Colombier
44*9a747e4fSDavid du Colombier static int ipconv(Fmt*);
45*9a747e4fSDavid du Colombier static void post(int, char*);
46*9a747e4fSDavid du Colombier static void setroot(void);
47*9a747e4fSDavid du Colombier
48*9a747e4fSDavid du Colombier void
usage(void)49*9a747e4fSDavid du Colombier usage(void)
50*9a747e4fSDavid du Colombier {
51*9a747e4fSDavid du Colombier fprint(2, "ratfs [-d] [-c conffile] [-f ctlfile] [-m mountpoint]\n");
52*9a747e4fSDavid du Colombier exits("usage");
53*9a747e4fSDavid du Colombier }
54*9a747e4fSDavid du Colombier
55*9a747e4fSDavid du Colombier void
main(int argc,char * argv[])56*9a747e4fSDavid du Colombier main(int argc, char *argv[])
57*9a747e4fSDavid du Colombier {
58*9a747e4fSDavid du Colombier char *mountpoint = MOUNTPOINT;
59*9a747e4fSDavid du Colombier int p[2];
60*9a747e4fSDavid du Colombier
61*9a747e4fSDavid du Colombier ARGBEGIN {
62*9a747e4fSDavid du Colombier case 'c':
63*9a747e4fSDavid du Colombier conffile = ARGF();
64*9a747e4fSDavid du Colombier break;
65*9a747e4fSDavid du Colombier case 'd':
66*9a747e4fSDavid du Colombier debugfd = 2; /* stderr*/
67*9a747e4fSDavid du Colombier break;
68*9a747e4fSDavid du Colombier case 'f':
69*9a747e4fSDavid du Colombier ctlfile = ARGF();
70*9a747e4fSDavid du Colombier break;
71*9a747e4fSDavid du Colombier case 'm':
72*9a747e4fSDavid du Colombier mountpoint = ARGF();
73*9a747e4fSDavid du Colombier break;
74*9a747e4fSDavid du Colombier } ARGEND
75*9a747e4fSDavid du Colombier if(argc != 0)
76*9a747e4fSDavid du Colombier usage();
77*9a747e4fSDavid du Colombier
78*9a747e4fSDavid du Colombier fmtinstall('I', ipconv);
79*9a747e4fSDavid du Colombier setroot();
80*9a747e4fSDavid du Colombier getconf();
81*9a747e4fSDavid du Colombier reload();
82*9a747e4fSDavid du Colombier
83*9a747e4fSDavid du Colombier /* get a pipe and mount it in /srv */
84*9a747e4fSDavid du Colombier if(pipe(p) < 0)
85*9a747e4fSDavid du Colombier fatal("pipe failed: %r");
86*9a747e4fSDavid du Colombier srvfd = p[0];
87*9a747e4fSDavid du Colombier post(p[1], mountpoint);
88*9a747e4fSDavid du Colombier
89*9a747e4fSDavid du Colombier /* start the 9fs protocol */
90*9a747e4fSDavid du Colombier switch(rfork(RFPROC|RFNAMEG|RFENVG|RFFDG|RFNOTEG|RFREND)){
91*9a747e4fSDavid du Colombier case -1:
92*9a747e4fSDavid du Colombier fatal("fork: %r");
93*9a747e4fSDavid du Colombier case 0:
94*9a747e4fSDavid du Colombier /* seal off standard input/output */
95*9a747e4fSDavid du Colombier close(0);
96*9a747e4fSDavid du Colombier open("/dev/null", OREAD);
97*9a747e4fSDavid du Colombier close(1);
98*9a747e4fSDavid du Colombier open("/dev/null", OWRITE);
99*9a747e4fSDavid du Colombier
100*9a747e4fSDavid du Colombier close(p[1]);
101*9a747e4fSDavid du Colombier fmtinstall('F', fcallfmt); /* debugging */
102*9a747e4fSDavid du Colombier io();
103*9a747e4fSDavid du Colombier fprint(2, "ratfs dying\n");
104*9a747e4fSDavid du Colombier break;
105*9a747e4fSDavid du Colombier default:
106*9a747e4fSDavid du Colombier close(p[0]);
107*9a747e4fSDavid du Colombier if(mount(p[1], -1, mountpoint, MREPL|MCREATE, "") < 0)
108*9a747e4fSDavid du Colombier fatal("mount failed: %r");
109*9a747e4fSDavid du Colombier }
110*9a747e4fSDavid du Colombier exits(0);
111*9a747e4fSDavid du Colombier }
112*9a747e4fSDavid du Colombier
113*9a747e4fSDavid du Colombier static void
setroot(void)114*9a747e4fSDavid du Colombier setroot(void)
115*9a747e4fSDavid du Colombier {
116*9a747e4fSDavid du Colombier Filetree *fp;
117*9a747e4fSDavid du Colombier Node *np;
118*9a747e4fSDavid du Colombier int qid;
119*9a747e4fSDavid du Colombier
120*9a747e4fSDavid du Colombier root = 0;
121*9a747e4fSDavid du Colombier qid = Qaddr;
122*9a747e4fSDavid du Colombier for(fp = filetree; fp->name; fp++) {
123*9a747e4fSDavid du Colombier switch(fp->level) {
124*9a747e4fSDavid du Colombier case 0: /* root */
125*9a747e4fSDavid du Colombier case 1: /* second level directory */
126*9a747e4fSDavid du Colombier newnode(root, fp->name, fp->type, fp->mode, fp->qid);
127*9a747e4fSDavid du Colombier break;
128*9a747e4fSDavid du Colombier case 2: /* lay down the Ipaddr and Acctaddr subdirectories */
129*9a747e4fSDavid du Colombier for (np = root->children; np; np = np->sibs){
130*9a747e4fSDavid du Colombier if(np->d.type == Addrdir)
131*9a747e4fSDavid du Colombier newnode(np, fp->name, fp->type, fp->mode, qid++);
132*9a747e4fSDavid du Colombier }
133*9a747e4fSDavid du Colombier break;
134*9a747e4fSDavid du Colombier default:
135*9a747e4fSDavid du Colombier fatal("bad filetree");
136*9a747e4fSDavid du Colombier }
137*9a747e4fSDavid du Colombier }
138*9a747e4fSDavid du Colombier dummy.d.type = Dummynode;
139*9a747e4fSDavid du Colombier dummy.d.mode = 0444;
140*9a747e4fSDavid du Colombier dummy.d.uid = "upas";
141*9a747e4fSDavid du Colombier dummy.d.gid = "upas";
142*9a747e4fSDavid du Colombier dummy.d.atime = dummy.d.mtime = time(0);
143*9a747e4fSDavid du Colombier dummy.d.qid.path = Qdummy; /* for now */
144*9a747e4fSDavid du Colombier }
145*9a747e4fSDavid du Colombier
146*9a747e4fSDavid du Colombier static void
post(int fd,char * mountpoint)147*9a747e4fSDavid du Colombier post(int fd, char *mountpoint)
148*9a747e4fSDavid du Colombier {
149*9a747e4fSDavid du Colombier
150*9a747e4fSDavid du Colombier int f;
151*9a747e4fSDavid du Colombier char buf[128];
152*9a747e4fSDavid du Colombier
153*9a747e4fSDavid du Colombier if(access(SRVFILE,0) >= 0){
154*9a747e4fSDavid du Colombier /*
155*9a747e4fSDavid du Colombier * If we can open and mount the /srv node,
156*9a747e4fSDavid du Colombier * another server is already running, so just exit.
157*9a747e4fSDavid du Colombier */
158*9a747e4fSDavid du Colombier f = open(SRVFILE, ORDWR);
159*9a747e4fSDavid du Colombier if(f >= 0 && mount(f, -1, mountpoint, MREPL|MCREATE, "") >= 0){
160*9a747e4fSDavid du Colombier unmount(0, mountpoint);
161*9a747e4fSDavid du Colombier close(f);
162*9a747e4fSDavid du Colombier exits(0);
163*9a747e4fSDavid du Colombier }
164*9a747e4fSDavid du Colombier remove(SRVFILE);
165*9a747e4fSDavid du Colombier }
166*9a747e4fSDavid du Colombier
167*9a747e4fSDavid du Colombier /*
168*9a747e4fSDavid du Colombier * create the server node and post our pipe to it
169*9a747e4fSDavid du Colombier */
170*9a747e4fSDavid du Colombier f = create(SRVFILE, OWRITE, 0666);
171*9a747e4fSDavid du Colombier if(f < 0)
172*9a747e4fSDavid du Colombier fatal("can't create %s", SRVFILE);
173*9a747e4fSDavid du Colombier
174*9a747e4fSDavid du Colombier sprint(buf, "%d", fd);
175*9a747e4fSDavid du Colombier if(write(f, buf, strlen(buf)) != strlen(buf))
176*9a747e4fSDavid du Colombier fatal("can't write %s", SRVFILE);
177*9a747e4fSDavid du Colombier
178*9a747e4fSDavid du Colombier close(f);
179*9a747e4fSDavid du Colombier }
180*9a747e4fSDavid du Colombier
181*9a747e4fSDavid du Colombier /*
182*9a747e4fSDavid du Colombier * print message and die
183*9a747e4fSDavid du Colombier */
184*9a747e4fSDavid du Colombier void
fatal(char * fmt,...)185*9a747e4fSDavid du Colombier fatal(char *fmt, ...)
186*9a747e4fSDavid du Colombier {
187*9a747e4fSDavid du Colombier va_list arg;
188*9a747e4fSDavid du Colombier char buf[8*1024];
189*9a747e4fSDavid du Colombier
190*9a747e4fSDavid du Colombier va_start(arg, fmt);
191*9a747e4fSDavid du Colombier vseprint(buf, buf + (sizeof(buf)-1) / sizeof(*buf), fmt, arg);
192*9a747e4fSDavid du Colombier va_end(arg);
193*9a747e4fSDavid du Colombier
194*9a747e4fSDavid du Colombier fprint(2, "%s: %s\n", argv0, buf);
195*9a747e4fSDavid du Colombier exits(buf);
196*9a747e4fSDavid du Colombier }
197*9a747e4fSDavid du Colombier
198*9a747e4fSDavid du Colombier /*
199*9a747e4fSDavid du Colombier * create a new directory node
200*9a747e4fSDavid du Colombier */
201*9a747e4fSDavid du Colombier Node*
newnode(Node * parent,char * name,ushort type,int mode,ulong qid)202*9a747e4fSDavid du Colombier newnode(Node *parent, char *name, ushort type, int mode, ulong qid)
203*9a747e4fSDavid du Colombier {
204*9a747e4fSDavid du Colombier Node *np;
205*9a747e4fSDavid du Colombier
206*9a747e4fSDavid du Colombier np = mallocz(sizeof(Node), 1);
207*9a747e4fSDavid du Colombier if(np == 0)
208*9a747e4fSDavid du Colombier fatal("out of memory");
209*9a747e4fSDavid du Colombier np->d.name = atom(name);
210*9a747e4fSDavid du Colombier np->d.type = type;
211*9a747e4fSDavid du Colombier np->d.mode = mode;
212*9a747e4fSDavid du Colombier np->d.mtime = np->d.atime = time(0);
213*9a747e4fSDavid du Colombier np->d.uid = atom("upas");
214*9a747e4fSDavid du Colombier np->d.gid = atom("upas");
215*9a747e4fSDavid du Colombier np->d.muid = atom("upas");
216*9a747e4fSDavid du Colombier if(np->d.mode&DMDIR)
217*9a747e4fSDavid du Colombier np->d.qid.type = QTDIR;
218*9a747e4fSDavid du Colombier np->d.qid.path = qid;
219*9a747e4fSDavid du Colombier np->d.qid.vers = 0;
220*9a747e4fSDavid du Colombier if(parent){
221*9a747e4fSDavid du Colombier np->parent = parent;
222*9a747e4fSDavid du Colombier np->sibs = parent->children;
223*9a747e4fSDavid du Colombier parent->children = np;
224*9a747e4fSDavid du Colombier parent->count++;
225*9a747e4fSDavid du Colombier } else {
226*9a747e4fSDavid du Colombier /* the root node */
227*9a747e4fSDavid du Colombier root = np;
228*9a747e4fSDavid du Colombier np->parent = np;
229*9a747e4fSDavid du Colombier np->children = 0;
230*9a747e4fSDavid du Colombier np->sibs = 0;
231*9a747e4fSDavid du Colombier }
232*9a747e4fSDavid du Colombier return np;
233*9a747e4fSDavid du Colombier }
234*9a747e4fSDavid du Colombier
235*9a747e4fSDavid du Colombier void
printnode(Node * np)236*9a747e4fSDavid du Colombier printnode(Node *np)
237*9a747e4fSDavid du Colombier {
238*9a747e4fSDavid du Colombier fprint(debugfd, "Node at %p: %s (%s %s)", np, np->d.name, np->d.uid, np->d.gid);
239*9a747e4fSDavid du Colombier if(np->d.qid.type&QTDIR)
240*9a747e4fSDavid du Colombier fprint(debugfd, " QTDIR");
241*9a747e4fSDavid du Colombier fprint(debugfd, "\n");
242*9a747e4fSDavid du Colombier fprint(debugfd,"\tQID: %llud.%lud Mode: %lo Type: %d\n", np->d.qid.path,
243*9a747e4fSDavid du Colombier np->d.qid.vers, np->d.mode, np->d.type);
244*9a747e4fSDavid du Colombier fprint(debugfd, "\tMod: %.15s Acc: %.15s Count: %d\n", ctime(np->d.mtime)+4,
245*9a747e4fSDavid du Colombier ctime(np->d.atime)+4, np->count);
246*9a747e4fSDavid du Colombier switch(np->d.type)
247*9a747e4fSDavid du Colombier {
248*9a747e4fSDavid du Colombier case Directory:
249*9a747e4fSDavid du Colombier fprint(debugfd, "\tDirectory Child: %p", np->children);
250*9a747e4fSDavid du Colombier break;
251*9a747e4fSDavid du Colombier case Addrdir:
252*9a747e4fSDavid du Colombier fprint(debugfd, "\tAddrdir Child: %p", np->children);
253*9a747e4fSDavid du Colombier break;
254*9a747e4fSDavid du Colombier case IPaddr:
255*9a747e4fSDavid du Colombier fprint(debugfd, "\tIPaddr Base: %p Alloc: %d BaseQid %lud", np->addrs,
256*9a747e4fSDavid du Colombier np->allocated, np->baseqid);
257*9a747e4fSDavid du Colombier break;
258*9a747e4fSDavid du Colombier case Acctaddr:
259*9a747e4fSDavid du Colombier fprint(debugfd, "\tAcctaddr Base: %p Alloc: %d BaseQid %lud", np->addrs,
260*9a747e4fSDavid du Colombier np->allocated, np->baseqid);
261*9a747e4fSDavid du Colombier break;
262*9a747e4fSDavid du Colombier case Trusted:
263*9a747e4fSDavid du Colombier fprint(debugfd, "\tTrusted Child: %p", np->children);
264*9a747e4fSDavid du Colombier break;
265*9a747e4fSDavid du Colombier case Trustedperm:
266*9a747e4fSDavid du Colombier fprint(debugfd, "\tPerm Trustedfile: %I", &np->ip);
267*9a747e4fSDavid du Colombier break;
268*9a747e4fSDavid du Colombier case Trustedtemp:
269*9a747e4fSDavid du Colombier fprint(debugfd, "\tTemp Trustedfile: %I", &np->ip);
270*9a747e4fSDavid du Colombier break;
271*9a747e4fSDavid du Colombier case Ctlfile:
272*9a747e4fSDavid du Colombier fprint(debugfd, "\tCtlfile");
273*9a747e4fSDavid du Colombier break;
274*9a747e4fSDavid du Colombier case Dummynode:
275*9a747e4fSDavid du Colombier fprint(debugfd, "\tDummynode");
276*9a747e4fSDavid du Colombier break;
277*9a747e4fSDavid du Colombier default:
278*9a747e4fSDavid du Colombier fprint(debugfd, "\tUnknown Node Type\n\n");
279*9a747e4fSDavid du Colombier return;
280*9a747e4fSDavid du Colombier }
281*9a747e4fSDavid du Colombier fprint(debugfd, " Parent %p Sib: %p\n\n", np->parent, np->sibs);
282*9a747e4fSDavid du Colombier }
283*9a747e4fSDavid du Colombier
284*9a747e4fSDavid du Colombier void
printfid(Fid * fp)285*9a747e4fSDavid du Colombier printfid(Fid *fp)
286*9a747e4fSDavid du Colombier {
287*9a747e4fSDavid du Colombier fprint(debugfd, "FID: %d (%s %s) Busy: %d Open: %d\n", fp->fid, fp->name,
288*9a747e4fSDavid du Colombier fp->uid, fp->busy, fp->open);
289*9a747e4fSDavid du Colombier printnode(fp->node);
290*9a747e4fSDavid du Colombier }
291*9a747e4fSDavid du Colombier
292*9a747e4fSDavid du Colombier void
printtree(Node * np)293*9a747e4fSDavid du Colombier printtree(Node *np)
294*9a747e4fSDavid du Colombier {
295*9a747e4fSDavid du Colombier printnode(np);
296*9a747e4fSDavid du Colombier if(np->d.type == IPaddr
297*9a747e4fSDavid du Colombier || np->d.type == Acctaddr
298*9a747e4fSDavid du Colombier || np->d.type == Trustedperm
299*9a747e4fSDavid du Colombier || np->d.type == Trustedtemp)
300*9a747e4fSDavid du Colombier return;
301*9a747e4fSDavid du Colombier for (np = np->children; np; np = np->sibs)
302*9a747e4fSDavid du Colombier printtree(np);
303*9a747e4fSDavid du Colombier }
304*9a747e4fSDavid du Colombier
305*9a747e4fSDavid du Colombier static int
ipconv(Fmt * f)306*9a747e4fSDavid du Colombier ipconv(Fmt *f)
307*9a747e4fSDavid du Colombier {
308*9a747e4fSDavid du Colombier Cidraddr *ip;
309*9a747e4fSDavid du Colombier int i, j;
310*9a747e4fSDavid du Colombier char *p;
311*9a747e4fSDavid du Colombier
312*9a747e4fSDavid du Colombier ip = va_arg(f->args, Cidraddr*);
313*9a747e4fSDavid du Colombier p = (char*)&ip->ipaddr;
314*9a747e4fSDavid du Colombier i = 0;
315*9a747e4fSDavid du Colombier for (j = ip->mask; j; j <<= 1)
316*9a747e4fSDavid du Colombier i++;
317*9a747e4fSDavid du Colombier return fmtprint(f, "%d.%d.%d.%d/%d", p[3]&0xff, p[2]&0xff, p[1]&0xff, p[0]&0xff, i);
318*9a747e4fSDavid du Colombier }
319