xref: /plan9/sys/src/cmd/unix/drawterm/kern/dev.c (revision 8ccd4a6360d974db7bd7bbd4f37e7018419ea908)
1*8ccd4a63SDavid du Colombier #include	"u.h"
2*8ccd4a63SDavid du Colombier #include	"lib.h"
3*8ccd4a63SDavid du Colombier #include	"dat.h"
4*8ccd4a63SDavid du Colombier #include	"fns.h"
5*8ccd4a63SDavid du Colombier #include	"error.h"
6*8ccd4a63SDavid du Colombier 
7*8ccd4a63SDavid du Colombier extern ulong	kerndate;
8*8ccd4a63SDavid du Colombier 
9*8ccd4a63SDavid du Colombier void
mkqid(Qid * q,vlong path,ulong vers,int type)10*8ccd4a63SDavid du Colombier mkqid(Qid *q, vlong path, ulong vers, int type)
11*8ccd4a63SDavid du Colombier {
12*8ccd4a63SDavid du Colombier 	q->type = type;
13*8ccd4a63SDavid du Colombier 	q->vers = vers;
14*8ccd4a63SDavid du Colombier 	q->path = path;
15*8ccd4a63SDavid du Colombier }
16*8ccd4a63SDavid du Colombier 
17*8ccd4a63SDavid du Colombier int
devno(int c,int user)18*8ccd4a63SDavid du Colombier devno(int c, int user)
19*8ccd4a63SDavid du Colombier {
20*8ccd4a63SDavid du Colombier 	int i;
21*8ccd4a63SDavid du Colombier 
22*8ccd4a63SDavid du Colombier 	for(i = 0; devtab[i] != nil; i++) {
23*8ccd4a63SDavid du Colombier 		if(devtab[i]->dc == c)
24*8ccd4a63SDavid du Colombier 			return i;
25*8ccd4a63SDavid du Colombier 	}
26*8ccd4a63SDavid du Colombier 	if(user == 0)
27*8ccd4a63SDavid du Colombier 		panic("devno %C 0x%ux", c, c);
28*8ccd4a63SDavid du Colombier 
29*8ccd4a63SDavid du Colombier 	return -1;
30*8ccd4a63SDavid du Colombier }
31*8ccd4a63SDavid du Colombier 
32*8ccd4a63SDavid du Colombier void
devdir(Chan * c,Qid qid,char * n,vlong length,char * user,long perm,Dir * db)33*8ccd4a63SDavid du Colombier devdir(Chan *c, Qid qid, char *n, vlong length, char *user, long perm, Dir *db)
34*8ccd4a63SDavid du Colombier {
35*8ccd4a63SDavid du Colombier 	db->name = n;
36*8ccd4a63SDavid du Colombier 	if(c->flag&CMSG)
37*8ccd4a63SDavid du Colombier 		qid.type |= QTMOUNT;
38*8ccd4a63SDavid du Colombier 	db->qid = qid;
39*8ccd4a63SDavid du Colombier 	db->type = devtab[c->type]->dc;
40*8ccd4a63SDavid du Colombier 	db->dev = c->dev;
41*8ccd4a63SDavid du Colombier 	db->mode = perm;
42*8ccd4a63SDavid du Colombier 	db->mode |= qid.type << 24;
43*8ccd4a63SDavid du Colombier 	db->atime = seconds();
44*8ccd4a63SDavid du Colombier 	db->mtime = kerndate;
45*8ccd4a63SDavid du Colombier 	db->length = length;
46*8ccd4a63SDavid du Colombier 	db->uid = user;
47*8ccd4a63SDavid du Colombier 	db->gid = eve;
48*8ccd4a63SDavid du Colombier 	db->muid = user;
49*8ccd4a63SDavid du Colombier }
50*8ccd4a63SDavid du Colombier 
51*8ccd4a63SDavid du Colombier /*
52*8ccd4a63SDavid du Colombier  * (here, Devgen is the prototype; devgen is the function in dev.c.)
53*8ccd4a63SDavid du Colombier  *
54*8ccd4a63SDavid du Colombier  * a Devgen is expected to return the directory entry for ".."
55*8ccd4a63SDavid du Colombier  * if you pass it s==DEVDOTDOT (-1).  otherwise...
56*8ccd4a63SDavid du Colombier  *
57*8ccd4a63SDavid du Colombier  * there are two contradictory rules.
58*8ccd4a63SDavid du Colombier  *
59*8ccd4a63SDavid du Colombier  * (i) if c is a directory, a Devgen is expected to list its children
60*8ccd4a63SDavid du Colombier  * as you iterate s.
61*8ccd4a63SDavid du Colombier  *
62*8ccd4a63SDavid du Colombier  * (ii) whether or not c is a directory, a Devgen is expected to list
63*8ccd4a63SDavid du Colombier  * its siblings as you iterate s.
64*8ccd4a63SDavid du Colombier  *
65*8ccd4a63SDavid du Colombier  * devgen always returns the list of children in the root
66*8ccd4a63SDavid du Colombier  * directory.  thus it follows (i) when c is the root and (ii) otherwise.
67*8ccd4a63SDavid du Colombier  * many other Devgens follow (i) when c is a directory and (ii) otherwise.
68*8ccd4a63SDavid du Colombier  *
69*8ccd4a63SDavid du Colombier  * devwalk assumes (i).  it knows that devgen breaks (i)
70*8ccd4a63SDavid du Colombier  * for children that are themselves directories, and explicitly catches them.
71*8ccd4a63SDavid du Colombier  *
72*8ccd4a63SDavid du Colombier  * devstat assumes (ii).  if the Devgen in question follows (i)
73*8ccd4a63SDavid du Colombier  * for this particular c, devstat will not find the necessary info.
74*8ccd4a63SDavid du Colombier  * with our particular Devgen functions, this happens only for
75*8ccd4a63SDavid du Colombier  * directories, so devstat makes something up, assuming
76*8ccd4a63SDavid du Colombier  * c->name, c->qid, eve, DMDIR|0555.
77*8ccd4a63SDavid du Colombier  *
78*8ccd4a63SDavid du Colombier  * devdirread assumes (i).  the callers have to make sure
79*8ccd4a63SDavid du Colombier  * that the Devgen satisfies (i) for the chan being read.
80*8ccd4a63SDavid du Colombier  */
81*8ccd4a63SDavid du Colombier /*
82*8ccd4a63SDavid du Colombier  * the zeroth element of the table MUST be the directory itself for ..
83*8ccd4a63SDavid du Colombier */
84*8ccd4a63SDavid du Colombier int
devgen(Chan * c,char * name,Dirtab * tab,int ntab,int i,Dir * dp)85*8ccd4a63SDavid du Colombier devgen(Chan *c, char *name, Dirtab *tab, int ntab, int i, Dir *dp)
86*8ccd4a63SDavid du Colombier {
87*8ccd4a63SDavid du Colombier 	if(tab == 0)
88*8ccd4a63SDavid du Colombier 		return -1;
89*8ccd4a63SDavid du Colombier 	if(i == DEVDOTDOT){
90*8ccd4a63SDavid du Colombier 		/* nothing */
91*8ccd4a63SDavid du Colombier 	}else if(name){
92*8ccd4a63SDavid du Colombier 		for(i=1; i<ntab; i++)
93*8ccd4a63SDavid du Colombier 			if(strcmp(tab[i].name, name) == 0)
94*8ccd4a63SDavid du Colombier 				break;
95*8ccd4a63SDavid du Colombier 		if(i==ntab)
96*8ccd4a63SDavid du Colombier 			return -1;
97*8ccd4a63SDavid du Colombier 		tab += i;
98*8ccd4a63SDavid du Colombier 	}else{
99*8ccd4a63SDavid du Colombier 		/* skip over the first element, that for . itself */
100*8ccd4a63SDavid du Colombier 		i++;
101*8ccd4a63SDavid du Colombier 		if(i >= ntab)
102*8ccd4a63SDavid du Colombier 			return -1;
103*8ccd4a63SDavid du Colombier 		tab += i;
104*8ccd4a63SDavid du Colombier 	}
105*8ccd4a63SDavid du Colombier 	devdir(c, tab->qid, tab->name, tab->length, eve, tab->perm, dp);
106*8ccd4a63SDavid du Colombier 	return 1;
107*8ccd4a63SDavid du Colombier }
108*8ccd4a63SDavid du Colombier 
109*8ccd4a63SDavid du Colombier void
devreset(void)110*8ccd4a63SDavid du Colombier devreset(void)
111*8ccd4a63SDavid du Colombier {
112*8ccd4a63SDavid du Colombier }
113*8ccd4a63SDavid du Colombier 
114*8ccd4a63SDavid du Colombier void
devinit(void)115*8ccd4a63SDavid du Colombier devinit(void)
116*8ccd4a63SDavid du Colombier {
117*8ccd4a63SDavid du Colombier }
118*8ccd4a63SDavid du Colombier 
119*8ccd4a63SDavid du Colombier void
devshutdown(void)120*8ccd4a63SDavid du Colombier devshutdown(void)
121*8ccd4a63SDavid du Colombier {
122*8ccd4a63SDavid du Colombier }
123*8ccd4a63SDavid du Colombier 
124*8ccd4a63SDavid du Colombier Chan*
devattach(int tc,char * spec)125*8ccd4a63SDavid du Colombier devattach(int tc, char *spec)
126*8ccd4a63SDavid du Colombier {
127*8ccd4a63SDavid du Colombier 	Chan *c;
128*8ccd4a63SDavid du Colombier 	char *buf;
129*8ccd4a63SDavid du Colombier 
130*8ccd4a63SDavid du Colombier 	c = newchan();
131*8ccd4a63SDavid du Colombier 	mkqid(&c->qid, 0, 0, QTDIR);
132*8ccd4a63SDavid du Colombier 	c->type = devno(tc, 0);
133*8ccd4a63SDavid du Colombier 	if(spec == nil)
134*8ccd4a63SDavid du Colombier 		spec = "";
135*8ccd4a63SDavid du Colombier 	buf = smalloc(4+strlen(spec)+1);
136*8ccd4a63SDavid du Colombier 	sprint(buf, "#%C%s", tc, spec);
137*8ccd4a63SDavid du Colombier 	c->name = newcname(buf);
138*8ccd4a63SDavid du Colombier 	free(buf);
139*8ccd4a63SDavid du Colombier 	return c;
140*8ccd4a63SDavid du Colombier }
141*8ccd4a63SDavid du Colombier 
142*8ccd4a63SDavid du Colombier 
143*8ccd4a63SDavid du Colombier Chan*
devclone(Chan * c)144*8ccd4a63SDavid du Colombier devclone(Chan *c)
145*8ccd4a63SDavid du Colombier {
146*8ccd4a63SDavid du Colombier 	Chan *nc;
147*8ccd4a63SDavid du Colombier 
148*8ccd4a63SDavid du Colombier 	if(c->flag & COPEN)
149*8ccd4a63SDavid du Colombier 		panic("clone of open file type %C\n", devtab[c->type]->dc);
150*8ccd4a63SDavid du Colombier 
151*8ccd4a63SDavid du Colombier 	nc = newchan();
152*8ccd4a63SDavid du Colombier 
153*8ccd4a63SDavid du Colombier 	nc->type = c->type;
154*8ccd4a63SDavid du Colombier 	nc->dev = c->dev;
155*8ccd4a63SDavid du Colombier 	nc->mode = c->mode;
156*8ccd4a63SDavid du Colombier 	nc->qid = c->qid;
157*8ccd4a63SDavid du Colombier 	nc->offset = c->offset;
158*8ccd4a63SDavid du Colombier 	nc->umh = nil;
159*8ccd4a63SDavid du Colombier 	nc->mountid = c->mountid;
160*8ccd4a63SDavid du Colombier 	nc->aux = c->aux;
161*8ccd4a63SDavid du Colombier 	nc->pgrpid = c->pgrpid;
162*8ccd4a63SDavid du Colombier 	nc->mid = c->mid;
163*8ccd4a63SDavid du Colombier 	nc->mqid = c->mqid;
164*8ccd4a63SDavid du Colombier 	nc->mcp = c->mcp;
165*8ccd4a63SDavid du Colombier 	return nc;
166*8ccd4a63SDavid du Colombier }
167*8ccd4a63SDavid du Colombier 
168*8ccd4a63SDavid du Colombier Walkqid*
devwalk(Chan * c,Chan * nc,char ** name,int nname,Dirtab * tab,int ntab,Devgen * gen)169*8ccd4a63SDavid du Colombier devwalk(Chan *c, Chan *nc, char **name, int nname, Dirtab *tab, int ntab, Devgen *gen)
170*8ccd4a63SDavid du Colombier {
171*8ccd4a63SDavid du Colombier 	int i, j, alloc;
172*8ccd4a63SDavid du Colombier 	Walkqid *wq;
173*8ccd4a63SDavid du Colombier 	char *n;
174*8ccd4a63SDavid du Colombier 	Dir dir;
175*8ccd4a63SDavid du Colombier 
176*8ccd4a63SDavid du Colombier 	if(nname > 0)
177*8ccd4a63SDavid du Colombier 		isdir(c);
178*8ccd4a63SDavid du Colombier 
179*8ccd4a63SDavid du Colombier 	alloc = 0;
180*8ccd4a63SDavid du Colombier 	wq = smalloc(sizeof(Walkqid)+(nname-1)*sizeof(Qid));
181*8ccd4a63SDavid du Colombier 	if(waserror()){
182*8ccd4a63SDavid du Colombier 		if(alloc && wq->clone!=nil)
183*8ccd4a63SDavid du Colombier 			cclose(wq->clone);
184*8ccd4a63SDavid du Colombier 		free(wq);
185*8ccd4a63SDavid du Colombier 		return nil;
186*8ccd4a63SDavid du Colombier 	}
187*8ccd4a63SDavid du Colombier 	if(nc == nil){
188*8ccd4a63SDavid du Colombier 		nc = devclone(c);
189*8ccd4a63SDavid du Colombier 		nc->type = 0;	/* device doesn't know about this channel yet */
190*8ccd4a63SDavid du Colombier 		alloc = 1;
191*8ccd4a63SDavid du Colombier 	}
192*8ccd4a63SDavid du Colombier 	wq->clone = nc;
193*8ccd4a63SDavid du Colombier 
194*8ccd4a63SDavid du Colombier 	for(j=0; j<nname; j++){
195*8ccd4a63SDavid du Colombier 		if(!(nc->qid.type&QTDIR)){
196*8ccd4a63SDavid du Colombier 			if(j==0)
197*8ccd4a63SDavid du Colombier 				error(Enotdir);
198*8ccd4a63SDavid du Colombier 			goto Done;
199*8ccd4a63SDavid du Colombier 		}
200*8ccd4a63SDavid du Colombier 		n = name[j];
201*8ccd4a63SDavid du Colombier 		if(strcmp(n, ".") == 0){
202*8ccd4a63SDavid du Colombier     Accept:
203*8ccd4a63SDavid du Colombier 			wq->qid[wq->nqid++] = nc->qid;
204*8ccd4a63SDavid du Colombier 			continue;
205*8ccd4a63SDavid du Colombier 		}
206*8ccd4a63SDavid du Colombier 		if(strcmp(n, "..") == 0){
207*8ccd4a63SDavid du Colombier 			if((*gen)(nc, nil, tab, ntab, DEVDOTDOT, &dir) != 1){
208*8ccd4a63SDavid du Colombier 				print("devgen walk .. in dev%s %llux broken\n",
209*8ccd4a63SDavid du Colombier 					devtab[nc->type]->name, nc->qid.path);
210*8ccd4a63SDavid du Colombier 				error("broken devgen");
211*8ccd4a63SDavid du Colombier 			}
212*8ccd4a63SDavid du Colombier 			nc->qid = dir.qid;
213*8ccd4a63SDavid du Colombier 			goto Accept;
214*8ccd4a63SDavid du Colombier 		}
215*8ccd4a63SDavid du Colombier 		/*
216*8ccd4a63SDavid du Colombier 		 * Ugly problem: If we're using devgen, make sure we're
217*8ccd4a63SDavid du Colombier 		 * walking the directory itself, represented by the first
218*8ccd4a63SDavid du Colombier 		 * entry in the table, and not trying to step into a sub-
219*8ccd4a63SDavid du Colombier 		 * directory of the table, e.g. /net/net. Devgen itself
220*8ccd4a63SDavid du Colombier 		 * should take care of the problem, but it doesn't have
221*8ccd4a63SDavid du Colombier 		 * the necessary information (that we're doing a walk).
222*8ccd4a63SDavid du Colombier 		 */
223*8ccd4a63SDavid du Colombier 		if(gen==devgen && nc->qid.path!=tab[0].qid.path)
224*8ccd4a63SDavid du Colombier 			goto Notfound;
225*8ccd4a63SDavid du Colombier 		for(i=0;; i++) {
226*8ccd4a63SDavid du Colombier 			switch((*gen)(nc, n, tab, ntab, i, &dir)){
227*8ccd4a63SDavid du Colombier 			case -1:
228*8ccd4a63SDavid du Colombier 			Notfound:
229*8ccd4a63SDavid du Colombier 				if(j == 0)
230*8ccd4a63SDavid du Colombier 					error(Enonexist);
231*8ccd4a63SDavid du Colombier 				kstrcpy(up->errstr, Enonexist, ERRMAX);
232*8ccd4a63SDavid du Colombier 				goto Done;
233*8ccd4a63SDavid du Colombier 			case 0:
234*8ccd4a63SDavid du Colombier 				continue;
235*8ccd4a63SDavid du Colombier 			case 1:
236*8ccd4a63SDavid du Colombier 				if(strcmp(n, dir.name) == 0){
237*8ccd4a63SDavid du Colombier 					nc->qid = dir.qid;
238*8ccd4a63SDavid du Colombier 					goto Accept;
239*8ccd4a63SDavid du Colombier 				}
240*8ccd4a63SDavid du Colombier 				continue;
241*8ccd4a63SDavid du Colombier 			}
242*8ccd4a63SDavid du Colombier 		}
243*8ccd4a63SDavid du Colombier 	}
244*8ccd4a63SDavid du Colombier 	/*
245*8ccd4a63SDavid du Colombier 	 * We processed at least one name, so will return some data.
246*8ccd4a63SDavid du Colombier 	 * If we didn't process all nname entries succesfully, we drop
247*8ccd4a63SDavid du Colombier 	 * the cloned channel and return just the Qids of the walks.
248*8ccd4a63SDavid du Colombier 	 */
249*8ccd4a63SDavid du Colombier Done:
250*8ccd4a63SDavid du Colombier 	poperror();
251*8ccd4a63SDavid du Colombier 	if(wq->nqid < nname){
252*8ccd4a63SDavid du Colombier 		if(alloc)
253*8ccd4a63SDavid du Colombier 			cclose(wq->clone);
254*8ccd4a63SDavid du Colombier 		wq->clone = nil;
255*8ccd4a63SDavid du Colombier 	}else if(wq->clone){
256*8ccd4a63SDavid du Colombier 		/* attach cloned channel to same device */
257*8ccd4a63SDavid du Colombier 		wq->clone->type = c->type;
258*8ccd4a63SDavid du Colombier 	}
259*8ccd4a63SDavid du Colombier 	return wq;
260*8ccd4a63SDavid du Colombier }
261*8ccd4a63SDavid du Colombier 
262*8ccd4a63SDavid du Colombier int
devstat(Chan * c,uchar * db,int n,Dirtab * tab,int ntab,Devgen * gen)263*8ccd4a63SDavid du Colombier devstat(Chan *c, uchar *db, int n, Dirtab *tab, int ntab, Devgen *gen)
264*8ccd4a63SDavid du Colombier {
265*8ccd4a63SDavid du Colombier 	int i;
266*8ccd4a63SDavid du Colombier 	Dir dir;
267*8ccd4a63SDavid du Colombier 	char *p, *elem;
268*8ccd4a63SDavid du Colombier 
269*8ccd4a63SDavid du Colombier 	for(i=0;; i++)
270*8ccd4a63SDavid du Colombier 		switch((*gen)(c, nil, tab, ntab, i, &dir)){
271*8ccd4a63SDavid du Colombier 		case -1:
272*8ccd4a63SDavid du Colombier 			if(c->qid.type & QTDIR){
273*8ccd4a63SDavid du Colombier 				if(c->name == nil)
274*8ccd4a63SDavid du Colombier 					elem = "???";
275*8ccd4a63SDavid du Colombier 				else if(strcmp(c->name->s, "/") == 0)
276*8ccd4a63SDavid du Colombier 					elem = "/";
277*8ccd4a63SDavid du Colombier 				else
278*8ccd4a63SDavid du Colombier 					for(elem=p=c->name->s; *p; p++)
279*8ccd4a63SDavid du Colombier 						if(*p == '/')
280*8ccd4a63SDavid du Colombier 							elem = p+1;
281*8ccd4a63SDavid du Colombier 				devdir(c, c->qid, elem, 0, eve, DMDIR|0555, &dir);
282*8ccd4a63SDavid du Colombier 				n = convD2M(&dir, db, n);
283*8ccd4a63SDavid du Colombier 				if(n == 0)
284*8ccd4a63SDavid du Colombier 					error(Ebadarg);
285*8ccd4a63SDavid du Colombier 				return n;
286*8ccd4a63SDavid du Colombier 			}
287*8ccd4a63SDavid du Colombier 			print("devstat %C %llux\n", devtab[c->type]->dc, c->qid.path);
288*8ccd4a63SDavid du Colombier 
289*8ccd4a63SDavid du Colombier 			error(Enonexist);
290*8ccd4a63SDavid du Colombier 		case 0:
291*8ccd4a63SDavid du Colombier 			break;
292*8ccd4a63SDavid du Colombier 		case 1:
293*8ccd4a63SDavid du Colombier 			if(c->qid.path == dir.qid.path) {
294*8ccd4a63SDavid du Colombier 				if(c->flag&CMSG)
295*8ccd4a63SDavid du Colombier 					dir.mode |= DMMOUNT;
296*8ccd4a63SDavid du Colombier 				n = convD2M(&dir, db, n);
297*8ccd4a63SDavid du Colombier 				if(n == 0)
298*8ccd4a63SDavid du Colombier 					error(Ebadarg);
299*8ccd4a63SDavid du Colombier 				return n;
300*8ccd4a63SDavid du Colombier 			}
301*8ccd4a63SDavid du Colombier 			break;
302*8ccd4a63SDavid du Colombier 		}
303*8ccd4a63SDavid du Colombier 	error(Egreg);	/* not reached? */
304*8ccd4a63SDavid du Colombier 	return -1;
305*8ccd4a63SDavid du Colombier }
306*8ccd4a63SDavid du Colombier 
307*8ccd4a63SDavid du Colombier long
devdirread(Chan * c,char * d,long n,Dirtab * tab,int ntab,Devgen * gen)308*8ccd4a63SDavid du Colombier devdirread(Chan *c, char *d, long n, Dirtab *tab, int ntab, Devgen *gen)
309*8ccd4a63SDavid du Colombier {
310*8ccd4a63SDavid du Colombier 	long m, dsz;
311*8ccd4a63SDavid du Colombier 	struct{
312*8ccd4a63SDavid du Colombier 		Dir d;
313*8ccd4a63SDavid du Colombier 		char slop[100];
314*8ccd4a63SDavid du Colombier 	}dir;
315*8ccd4a63SDavid du Colombier 
316*8ccd4a63SDavid du Colombier 	for(m=0; m<n; c->dri++) {
317*8ccd4a63SDavid du Colombier 		switch((*gen)(c, nil, tab, ntab, c->dri, &dir.d)){
318*8ccd4a63SDavid du Colombier 		case -1:
319*8ccd4a63SDavid du Colombier 			return m;
320*8ccd4a63SDavid du Colombier 
321*8ccd4a63SDavid du Colombier 		case 0:
322*8ccd4a63SDavid du Colombier 			break;
323*8ccd4a63SDavid du Colombier 
324*8ccd4a63SDavid du Colombier 		case 1:
325*8ccd4a63SDavid du Colombier 			dsz = convD2M(&dir.d, (uchar*)d, n-m);
326*8ccd4a63SDavid du Colombier 			if(dsz <= BIT16SZ){	/* <= not < because this isn't stat; read is stuck */
327*8ccd4a63SDavid du Colombier 				if(m == 0)
328*8ccd4a63SDavid du Colombier 					error(Eshort);
329*8ccd4a63SDavid du Colombier 				return m;
330*8ccd4a63SDavid du Colombier 			}
331*8ccd4a63SDavid du Colombier 			m += dsz;
332*8ccd4a63SDavid du Colombier 			d += dsz;
333*8ccd4a63SDavid du Colombier 			break;
334*8ccd4a63SDavid du Colombier 		}
335*8ccd4a63SDavid du Colombier 	}
336*8ccd4a63SDavid du Colombier 
337*8ccd4a63SDavid du Colombier 	return m;
338*8ccd4a63SDavid du Colombier }
339*8ccd4a63SDavid du Colombier 
340*8ccd4a63SDavid du Colombier /*
341*8ccd4a63SDavid du Colombier  * error(Eperm) if open permission not granted for up->user.
342*8ccd4a63SDavid du Colombier  */
343*8ccd4a63SDavid du Colombier void
devpermcheck(char * fileuid,ulong perm,int omode)344*8ccd4a63SDavid du Colombier devpermcheck(char *fileuid, ulong perm, int omode)
345*8ccd4a63SDavid du Colombier {
346*8ccd4a63SDavid du Colombier 	ulong t;
347*8ccd4a63SDavid du Colombier 	static int access[] = { 0400, 0200, 0600, 0100 };
348*8ccd4a63SDavid du Colombier 
349*8ccd4a63SDavid du Colombier 	if(strcmp(up->user, fileuid) == 0)
350*8ccd4a63SDavid du Colombier 		perm <<= 0;
351*8ccd4a63SDavid du Colombier 	else
352*8ccd4a63SDavid du Colombier 	if(strcmp(up->user, eve) == 0)
353*8ccd4a63SDavid du Colombier 		perm <<= 3;
354*8ccd4a63SDavid du Colombier 	else
355*8ccd4a63SDavid du Colombier 		perm <<= 6;
356*8ccd4a63SDavid du Colombier 
357*8ccd4a63SDavid du Colombier 	t = access[omode&3];
358*8ccd4a63SDavid du Colombier 	if((t&perm) != t)
359*8ccd4a63SDavid du Colombier 		error(Eperm);
360*8ccd4a63SDavid du Colombier }
361*8ccd4a63SDavid du Colombier 
362*8ccd4a63SDavid du Colombier Chan*
devopen(Chan * c,int omode,Dirtab * tab,int ntab,Devgen * gen)363*8ccd4a63SDavid du Colombier devopen(Chan *c, int omode, Dirtab *tab, int ntab, Devgen *gen)
364*8ccd4a63SDavid du Colombier {
365*8ccd4a63SDavid du Colombier 	int i;
366*8ccd4a63SDavid du Colombier 	Dir dir;
367*8ccd4a63SDavid du Colombier 
368*8ccd4a63SDavid du Colombier 	for(i=0;; i++) {
369*8ccd4a63SDavid du Colombier 		switch((*gen)(c, nil, tab, ntab, i, &dir)){
370*8ccd4a63SDavid du Colombier 		case -1:
371*8ccd4a63SDavid du Colombier 			goto Return;
372*8ccd4a63SDavid du Colombier 		case 0:
373*8ccd4a63SDavid du Colombier 			break;
374*8ccd4a63SDavid du Colombier 		case 1:
375*8ccd4a63SDavid du Colombier 			if(c->qid.path == dir.qid.path) {
376*8ccd4a63SDavid du Colombier 				devpermcheck(dir.uid, dir.mode, omode);
377*8ccd4a63SDavid du Colombier 				goto Return;
378*8ccd4a63SDavid du Colombier 			}
379*8ccd4a63SDavid du Colombier 			break;
380*8ccd4a63SDavid du Colombier 		}
381*8ccd4a63SDavid du Colombier 	}
382*8ccd4a63SDavid du Colombier Return:
383*8ccd4a63SDavid du Colombier 	c->offset = 0;
384*8ccd4a63SDavid du Colombier 	if((c->qid.type&QTDIR) && omode!=OREAD)
385*8ccd4a63SDavid du Colombier 		error(Eperm);
386*8ccd4a63SDavid du Colombier 	c->mode = openmode(omode);
387*8ccd4a63SDavid du Colombier 	c->flag |= COPEN;
388*8ccd4a63SDavid du Colombier 	return c;
389*8ccd4a63SDavid du Colombier }
390*8ccd4a63SDavid du Colombier 
391*8ccd4a63SDavid du Colombier void
devcreate(Chan * c,char * name,int mode,ulong perm)392*8ccd4a63SDavid du Colombier devcreate(Chan *c, char *name, int mode, ulong perm)
393*8ccd4a63SDavid du Colombier {
394*8ccd4a63SDavid du Colombier 	USED(c);
395*8ccd4a63SDavid du Colombier 	USED(name);
396*8ccd4a63SDavid du Colombier 	USED(mode);
397*8ccd4a63SDavid du Colombier 	USED(perm);
398*8ccd4a63SDavid du Colombier 
399*8ccd4a63SDavid du Colombier 	error(Eperm);
400*8ccd4a63SDavid du Colombier }
401*8ccd4a63SDavid du Colombier 
402*8ccd4a63SDavid du Colombier Block*
devbread(Chan * c,long n,ulong offset)403*8ccd4a63SDavid du Colombier devbread(Chan *c, long n, ulong offset)
404*8ccd4a63SDavid du Colombier {
405*8ccd4a63SDavid du Colombier 	Block *bp;
406*8ccd4a63SDavid du Colombier 
407*8ccd4a63SDavid du Colombier 	bp = allocb(n);
408*8ccd4a63SDavid du Colombier 	if(bp == 0)
409*8ccd4a63SDavid du Colombier 		error(Enomem);
410*8ccd4a63SDavid du Colombier 	if(waserror()) {
411*8ccd4a63SDavid du Colombier 		freeb(bp);
412*8ccd4a63SDavid du Colombier 		nexterror();
413*8ccd4a63SDavid du Colombier 	}
414*8ccd4a63SDavid du Colombier 	bp->wp += devtab[c->type]->read(c, bp->wp, n, offset);
415*8ccd4a63SDavid du Colombier 	poperror();
416*8ccd4a63SDavid du Colombier 	return bp;
417*8ccd4a63SDavid du Colombier }
418*8ccd4a63SDavid du Colombier 
419*8ccd4a63SDavid du Colombier long
devbwrite(Chan * c,Block * bp,ulong offset)420*8ccd4a63SDavid du Colombier devbwrite(Chan *c, Block *bp, ulong offset)
421*8ccd4a63SDavid du Colombier {
422*8ccd4a63SDavid du Colombier 	long n;
423*8ccd4a63SDavid du Colombier 
424*8ccd4a63SDavid du Colombier 	if(waserror()) {
425*8ccd4a63SDavid du Colombier 		freeb(bp);
426*8ccd4a63SDavid du Colombier 		nexterror();
427*8ccd4a63SDavid du Colombier 	}
428*8ccd4a63SDavid du Colombier 	n = devtab[c->type]->write(c, bp->rp, BLEN(bp), offset);
429*8ccd4a63SDavid du Colombier 	poperror();
430*8ccd4a63SDavid du Colombier 	freeb(bp);
431*8ccd4a63SDavid du Colombier 
432*8ccd4a63SDavid du Colombier 	return n;
433*8ccd4a63SDavid du Colombier }
434*8ccd4a63SDavid du Colombier 
435*8ccd4a63SDavid du Colombier void
devremove(Chan * c)436*8ccd4a63SDavid du Colombier devremove(Chan *c)
437*8ccd4a63SDavid du Colombier {
438*8ccd4a63SDavid du Colombier 	USED(c);
439*8ccd4a63SDavid du Colombier 	error(Eperm);
440*8ccd4a63SDavid du Colombier }
441*8ccd4a63SDavid du Colombier 
442*8ccd4a63SDavid du Colombier int
devwstat(Chan * c,uchar * a,int n)443*8ccd4a63SDavid du Colombier devwstat(Chan *c, uchar *a, int n)
444*8ccd4a63SDavid du Colombier {
445*8ccd4a63SDavid du Colombier 	USED(c);
446*8ccd4a63SDavid du Colombier 	USED(a);
447*8ccd4a63SDavid du Colombier 	USED(n);
448*8ccd4a63SDavid du Colombier 
449*8ccd4a63SDavid du Colombier 	error(Eperm);
450*8ccd4a63SDavid du Colombier 	return 0;
451*8ccd4a63SDavid du Colombier }
452*8ccd4a63SDavid du Colombier 
453*8ccd4a63SDavid du Colombier void
devpower(int a)454*8ccd4a63SDavid du Colombier devpower(int a)
455*8ccd4a63SDavid du Colombier {
456*8ccd4a63SDavid du Colombier 	USED(a);
457*8ccd4a63SDavid du Colombier 	error(Eperm);
458*8ccd4a63SDavid du Colombier }
459*8ccd4a63SDavid du Colombier 
460*8ccd4a63SDavid du Colombier int
devconfig(int a,char * b,DevConf * c)461*8ccd4a63SDavid du Colombier devconfig(int a, char *b, DevConf *c)
462*8ccd4a63SDavid du Colombier {
463*8ccd4a63SDavid du Colombier 	USED(a);
464*8ccd4a63SDavid du Colombier 	USED(b);
465*8ccd4a63SDavid du Colombier 	USED(c);
466*8ccd4a63SDavid du Colombier 	error(Eperm);
467*8ccd4a63SDavid du Colombier 	return 0;
468*8ccd4a63SDavid du Colombier }
469