xref: /plan9/sys/src/cmd/ratfs/main.c (revision 9a747e4fd48b9f4522c70c07e8f882a15030f964)
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