📄 dev.c
字号:
#include "u.h"#include "../port/lib.h"#include "mem.h"#include "dat.h"#include "fns.h"#include "../port/error.h"extern ulong kerndate;intdevno(int c, int user){ int i; for(i = 0; devtab[i] != nil; i++) { if(devtab[i]->dc == c) return i; } if(user == 0) panic("devno %C 0x%ux", c, c); return -1;}voiddevdir(Chan *c, Qid qid, char *n, vlong length, char *user, long perm, Dir *db){ strcpy(db->name, n); db->qid = qid; db->type = devtab[c->type]->dc; db->dev = c->dev; if(qid.path & CHDIR) db->mode = CHDIR|perm; else db->mode = perm; if(c->flag&CMSG) db->mode |= CHMOUNT; db->atime = seconds(); db->mtime = kerndate; db->length = length; memmove(db->uid, user, NAMELEN); memmove(db->gid, eve, NAMELEN);}//// the zeroth element of the table MUST be the directory itself for ..//intdevgen(Chan *c, Dirtab *tab, int ntab, int i, Dir *dp){ if(tab==0) return -1; if(i!=DEVDOTDOT){ if(i>=ntab) return -1; tab += i; } devdir(c, tab->qid, tab->name, tab->length, eve, tab->perm, dp); return 1;}voiddevreset(void){}voiddevinit(void){}Chan*devattach(int tc, char *spec){ Chan *c; char buf[NAMELEN+4]; c = newchan(); c->qid = (Qid){CHDIR, 0}; c->type = devno(tc, 0); sprint(buf, "#%C%s", tc, spec==nil? "" : spec); c->name = newcname(buf); return c;}Chan*devclone(Chan *c, Chan *nc){ if(c->flag & COPEN) panic("clone of open file type %C\n", devtab[c->type]->dc); if(nc == 0) nc = newchan(); nc->type = c->type; nc->dev = c->dev; nc->mode = c->mode; nc->qid = c->qid; nc->offset = c->offset; nc->flag = c->flag; nc->mh = c->mh; if(c->mh != nil) incref(c->mh); nc->mountid = c->mountid; nc->aux = c->aux; nc->mchan = c->mchan; nc->mqid = c->mqid; nc->mcp = c->mcp; return nc;}intdevwalk(Chan *c, char *name, Dirtab *tab, int ntab, Devgen *gen){ long i; Dir dir; isdir(c); if(name[0]=='.' && name[1]==0) return 1; if(name[0]=='.' && name[1]=='.' && name[2]==0){ (*gen)(c, tab, ntab, DEVDOTDOT, &dir); c->qid = dir.qid; return 1; } for(i=0;; i++) { switch((*gen)(c, tab, ntab, i, &dir)){ case -1: strncpy(up->error, Enonexist, NAMELEN); return 0; case 0: continue; case 1: if(strcmp(name, dir.name) == 0){ c->qid = dir.qid; return 1; } continue; } } return 0;}voiddevstat(Chan *c, char *db, Dirtab *tab, int ntab, Devgen *gen){ int i; Dir dir; char *p, *elem; for(i=0;; i++) switch((*gen)(c, tab, ntab, i, &dir)){ case -1: if(c->qid.path & CHDIR){ if(c->name == nil) elem = "???"; else if(strcmp(c->name->s, "/") == 0) elem = "/"; else for(elem=p=c->name->s; *p; p++) if(*p == '/') elem = p+1; devdir(c, c->qid, elem, i*DIRLEN, eve, CHDIR|0555, &dir); convD2M(&dir, db); return; } print("%s %s: devstat %C %lux\n", up->text, up->user, devtab[c->type]->dc, c->qid.path); error(Enonexist); case 0: break; case 1: if(c->qid.path == dir.qid.path) { if(c->flag&CMSG) dir.mode |= CHMOUNT; convD2M(&dir, db); return; } break; }}longdevdirread(Chan *c, char *d, long n, Dirtab *tab, int ntab, Devgen *gen){ long k, m; Dir dir; k = c->offset/DIRLEN; for(m=0; m<n; k++) { switch((*gen)(c, tab, ntab, k, &dir)){ case -1: return m; case 0: c->offset += DIRLEN; break; case 1: convD2M(&dir, d); m += DIRLEN; d += DIRLEN; break; } } return m;}Chan*devopen(Chan *c, int omode, Dirtab *tab, int ntab, Devgen *gen){ int i; Dir dir; ulong t, mode; static int access[] = { 0400, 0200, 0600, 0100 }; for(i=0;; i++) { switch((*gen)(c, tab, ntab, i, &dir)){ case -1: goto Return; case 0: break; case 1: if(c->qid.path == dir.qid.path) { if(strcmp(up->user, dir.uid) == 0) mode = dir.mode; else if(strcmp(up->user, eve) == 0) mode = dir.mode<<3; else mode = dir.mode<<6; t = access[omode&3]; if((t & mode) == t) goto Return; error(Eperm); } break; } }Return: c->offset = 0; if((c->qid.path&CHDIR) && omode!=OREAD) error(Eperm); c->mode = openmode(omode); c->flag |= COPEN; return c;}voiddevcreate(Chan*, char*, int, ulong){ error(Eperm);}Block*devbread(Chan *c, long n, ulong offset){ Block *bp; bp = allocb(n); if(bp == 0) error(Enomem); if(waserror()) { freeb(bp); nexterror(); } bp->wp += devtab[c->type]->read(c, bp->wp, n, offset); poperror(); return bp;}longdevbwrite(Chan *c, Block *bp, ulong offset){ long n; n = devtab[c->type]->write(c, bp->rp, BLEN(bp), offset); freeb(bp); return n;}voiddevremove(Chan*){ error(Eperm);}voiddevwstat(Chan*, char*){ error(Eperm);}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -