⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 devsrv.c

📁 著名操作系统Plan 9的第三版的部分核心源代码。现在很难找到了。Plan 9是bell实验室开发的Unix后继者。
💻 C
字号:
#include	"u.h"#include	"../port/lib.h"#include	"mem.h"#include	"dat.h"#include	"fns.h"#include	"../port/error.h"typedef struct Srv Srv;struct Srv{	char	name[NAMELEN];	char	owner[NAMELEN];	ulong	perm;	Chan	*chan;	Srv	*link;	ulong	path;};static QLock	srvlk;static Srv	*srv;static int	qidpath;static intsrvgen(Chan *c, Dirtab*, int, int s, Dir *dp){	Srv *sp;	if(s == DEVDOTDOT){		devdir(c, c->qid, "#s", 0, eve, 0555, dp);		return 1;	}	qlock(&srvlk);	for(sp = srv; sp && s; sp = sp->link)		s--;	if(sp == 0) {		qunlock(&srvlk);		return -1;	}	devdir(c, (Qid){sp->path, 0}, sp->name, 0, sp->owner, sp->perm, dp);	qunlock(&srvlk);	return 1;}static voidsrvinit(void){	qidpath = 1;}static Chan*srvattach(char *spec){	return devattach('s', spec);}static intsrvwalk(Chan *c, char *name){	return devwalk(c, name, 0, 0, srvgen);}static voidsrvstat(Chan *c, char *db){	devstat(c, db, 0, 0, srvgen);}static Srv*srvlookup(char *name, ulong qidpath){	Srv *sp;	for(sp = srv; sp; sp = sp->link)		if(sp->path == qidpath || (name && strcmp(sp->name, name) == 0))			return sp;	return nil;}char*srvname(Chan *c){	Srv *sp;	char *s;	for(sp = srv; sp; sp = sp->link)		if(sp->chan == c){			s = smalloc(3+strlen(sp->name)+1);			sprint(s, "#s/%s", sp->name);			return s;		}	return nil;}static Chan*srvopen(Chan *c, int omode){	Srv *sp;	if(c->qid.path == CHDIR){		if(omode & ORCLOSE)			error(Eperm);		if(omode != OREAD)			error(Eisdir);		c->mode = omode;		c->flag |= COPEN;		c->offset = 0;		return c;	}	qlock(&srvlk);	if(waserror()){		qunlock(&srvlk);		nexterror();	}	sp = srvlookup(nil, c->qid.path);	if(sp == 0 || sp->chan == 0)		error(Eshutdown);	if(omode&OTRUNC)		error("srv file already exists");	if(openmode(omode)!=sp->chan->mode && sp->chan->mode!=ORDWR)		error(Eperm);	cclose(c);	incref(sp->chan);	qunlock(&srvlk);	poperror();	return sp->chan;}static voidsrvcreate(Chan *c, char *name, int omode, ulong perm){	Srv *sp;	if(openmode(omode) != OWRITE)		error(Eperm);	if(omode & OCEXEC)	/* can't happen */		panic("someone broke namec");	sp = malloc(sizeof(Srv));	if(sp == 0)		error(Enomem);	qlock(&srvlk);	if(waserror()){		qunlock(&srvlk);		nexterror();	}	if(srvlookup(name, -1))		error(Eexist);	sp->path = qidpath++;	sp->link = srv;	c->qid.path = sp->path;	srv = sp;	qunlock(&srvlk);	poperror();	strncpy(sp->name, name, NAMELEN);	strncpy(sp->owner, up->user, NAMELEN);	sp->perm = perm&0777;	c->flag |= COPEN;	c->mode = OWRITE;}static voidsrvremove(Chan *c){	Srv *sp, **l;	if(c->qid.path == CHDIR)		error(Eperm);	qlock(&srvlk);	if(waserror()){		qunlock(&srvlk);		nexterror();	}	l = &srv;	for(sp = *l; sp; sp = sp->link) {		if(sp->path == c->qid.path)			break;		l = &sp->link;	}	if(sp == 0)		error(Enonexist);	if(strcmp(sp->name, "boot") == 0)		error(Eperm);	*l = sp->link;	qunlock(&srvlk);	poperror();	if(sp->chan)		cclose(sp->chan);	free(sp);}static voidsrvwstat(Chan *c, char *dp){	Dir d;	Srv *sp;	if(c->qid.path & CHDIR)		error(Eperm);	qlock(&srvlk);	if(waserror()){		qunlock(&srvlk);		nexterror();	}	sp = srvlookup(nil, c->qid.path);	if(sp == 0)		error(Enonexist);	if(strcmp(sp->owner, up->user) && !iseve())		error(Eperm);	convM2D(dp, &d);	d.mode &= 0777;	sp->perm = d.mode;	qunlock(&srvlk);	poperror();}static voidsrvclose(Chan *c){	/*	 * errors from srvremove will be caught by cclose and ignored.	 * in theory we need to override any changes in removability	 * since open, but since all that's checked is the owner,	 * which is immutable, all is well.	 */	if(c->flag & CRCLOSE)		srvremove(c);}static longsrvread(Chan *c, void *va, long n, vlong){	isdir(c);	return devdirread(c, va, n, 0, 0, srvgen);}static longsrvwrite(Chan *c, void *va, long n, vlong){	Srv *sp;	Chan *c1;	int fd;	char buf[32];	if(n >= sizeof buf)		error(Egreg);	memmove(buf, va, n);	/* so we can NUL-terminate */	buf[n] = 0;	fd = strtoul(buf, 0, 0);	c1 = fdtochan(fd, -1, 0, 1);	/* error check and inc ref */	qlock(&srvlk);	if(waserror()) {		qunlock(&srvlk);		cclose(c1);		nexterror();	}	if(c1->flag & (CCEXEC|CRCLOSE))		error("posted fd has remove-on-close or close-on-exec");	sp = srvlookup(nil, c->qid.path);	if(sp == 0)		error(Enonexist);	if(sp->chan)		error(Ebadusefd);	sp->chan = c1;	qunlock(&srvlk);	poperror();	return n;}Dev srvdevtab = {	's',	"srv",	devreset,	srvinit,	srvattach,	devclone,	srvwalk,	srvstat,	srvopen,	srvcreate,	srvclose,	srvread,	devbread,	srvwrite,	devbwrite,	srvremove,	srvwstat,};

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -