1bd389b36SDavid du Colombier /*
2bd389b36SDavid du Colombier * Vax 32V Unix filesystem (same as pre-FFS Berkeley)
3bd389b36SDavid du Colombier */
4bd389b36SDavid du Colombier #include <u.h>
5bd389b36SDavid du Colombier #include <libc.h>
6219b2ee8SDavid du Colombier #include <auth.h>
7bd389b36SDavid du Colombier #include <fcall.h>
8bd389b36SDavid du Colombier #include "tapefs.h"
9bd389b36SDavid du Colombier
10bd389b36SDavid du Colombier /*
11bd389b36SDavid du Colombier * v32 disk inode
12bd389b36SDavid du Colombier */
13bd389b36SDavid du Colombier #define VNADDR 13
14bd389b36SDavid du Colombier #define VFMT 0160000
15bd389b36SDavid du Colombier #define VIFREG 0100000
16bd389b36SDavid du Colombier #define VIFDIR 0040000
17bd389b36SDavid du Colombier #define VIFCHR 0120000
18bd389b36SDavid du Colombier #define VIFBLK 0160000
19bd389b36SDavid du Colombier #define VMODE 0777
20bd389b36SDavid du Colombier #define VSUPERB 1
21bd389b36SDavid du Colombier #define VROOT 2 /* root inode */
22bd389b36SDavid du Colombier #define VNAMELEN 14
23bbd061d4SDavid du Colombier #define MAXBLSIZE 1024
24bbd061d4SDavid du Colombier int BLSIZE;
25bd389b36SDavid du Colombier #define LINOPB (BLSIZE/sizeof(struct v32dinode))
26bbd061d4SDavid du Colombier #define LNINDIR (BLSIZE/4)
27bbd061d4SDavid du Colombier #define MAXLNINDIR (MAXBLSIZE/4)
28bd389b36SDavid du Colombier
29bd389b36SDavid du Colombier struct v32dinode {
30bd389b36SDavid du Colombier unsigned char flags[2];
31bd389b36SDavid du Colombier unsigned char nlinks[2];
32bd389b36SDavid du Colombier unsigned char uid[2];
33bd389b36SDavid du Colombier unsigned char gid[2];
34bd389b36SDavid du Colombier unsigned char size[4];
35bd389b36SDavid du Colombier unsigned char addr[40];
36bd389b36SDavid du Colombier unsigned char atime[4];
37bd389b36SDavid du Colombier unsigned char mtime[4];
38bd389b36SDavid du Colombier unsigned char ctime[4];
39bd389b36SDavid du Colombier };
40bd389b36SDavid du Colombier
41bd389b36SDavid du Colombier struct v32dir {
42bd389b36SDavid du Colombier uchar ino[2];
43bd389b36SDavid du Colombier char name[VNAMELEN];
44bd389b36SDavid du Colombier };
45bd389b36SDavid du Colombier
46bd389b36SDavid du Colombier int tapefile;
47bd389b36SDavid du Colombier Fileinf iget(int ino);
48bd389b36SDavid du Colombier long bmap(Ram *r, long bno);
49bd389b36SDavid du Colombier void getblk(Ram *r, long bno, char *buf);
50bd389b36SDavid du Colombier
51bd389b36SDavid du Colombier void
populate(char * name)52bd389b36SDavid du Colombier populate(char *name)
53bd389b36SDavid du Colombier {
54bd389b36SDavid du Colombier Fileinf f;
55bd389b36SDavid du Colombier
56bbd061d4SDavid du Colombier BLSIZE = 512; /* 32v */
57bbd061d4SDavid du Colombier if(blocksize){
58bbd061d4SDavid du Colombier /* 1024 for 4.1BSD */
59bbd061d4SDavid du Colombier if(blocksize != 512 && blocksize != 1024)
60bbd061d4SDavid du Colombier error("bad block size");
61bbd061d4SDavid du Colombier BLSIZE = blocksize;
62bbd061d4SDavid du Colombier }
63bd389b36SDavid du Colombier replete = 0;
64bd389b36SDavid du Colombier tapefile = open(name, OREAD);
65bd389b36SDavid du Colombier if (tapefile<0)
66bd389b36SDavid du Colombier error("Can't open argument file");
67bd389b36SDavid du Colombier f = iget(VROOT);
68bd389b36SDavid du Colombier ram->perm = f.mode;
69bd389b36SDavid du Colombier ram->mtime = f.mdate;
7041fe996aSDavid du Colombier ram->addr = f.addr;
7141fe996aSDavid du Colombier ram->data = f.data;
72bd389b36SDavid du Colombier ram->ndata = f.size;
73bd389b36SDavid du Colombier }
74bd389b36SDavid du Colombier
75bd389b36SDavid du Colombier void
popdir(Ram * r)76bd389b36SDavid du Colombier popdir(Ram *r)
77bd389b36SDavid du Colombier {
78bd389b36SDavid du Colombier int i, ino;
79bd389b36SDavid du Colombier char *cp;
80bd389b36SDavid du Colombier struct v32dir *dp;
81bd389b36SDavid du Colombier Fileinf f;
82bd389b36SDavid du Colombier char name[VNAMELEN+1];
83bd389b36SDavid du Colombier
847dd7cddfSDavid du Colombier cp = 0;
85bd389b36SDavid du Colombier for (i=0; i<r->ndata; i+=sizeof(struct v32dir)) {
86bd389b36SDavid du Colombier if (i%BLSIZE==0)
87bd389b36SDavid du Colombier cp = doread(r, i, BLSIZE);
88bd389b36SDavid du Colombier dp = (struct v32dir *)(cp+i%BLSIZE);
89bd389b36SDavid du Colombier ino = g2byte(dp->ino);
90bd389b36SDavid du Colombier if (strcmp(dp->name, ".")==0 || strcmp(dp->name, "..")==0)
91bd389b36SDavid du Colombier continue;
92bd389b36SDavid du Colombier if (ino==0)
93bd389b36SDavid du Colombier continue;
94bd389b36SDavid du Colombier f = iget(ino);
95bd389b36SDavid du Colombier strncpy(name, dp->name, VNAMELEN);
96*9f2726c3SDavid du Colombier name[VNAMELEN] = '\0';
97bd389b36SDavid du Colombier f.name = name;
98bd389b36SDavid du Colombier popfile(r, f);
99bd389b36SDavid du Colombier }
100bd389b36SDavid du Colombier r->replete = 1;
101bd389b36SDavid du Colombier }
102bd389b36SDavid du Colombier
103bd389b36SDavid du Colombier void
dotrunc(Ram * r)104bd389b36SDavid du Colombier dotrunc(Ram *r)
105bd389b36SDavid du Colombier {
106bd389b36SDavid du Colombier USED(r);
107bd389b36SDavid du Colombier }
108bd389b36SDavid du Colombier
109bd389b36SDavid du Colombier void
docreate(Ram * r)110bd389b36SDavid du Colombier docreate(Ram *r)
111bd389b36SDavid du Colombier {
112bd389b36SDavid du Colombier USED(r);
113bd389b36SDavid du Colombier }
114bd389b36SDavid du Colombier
115bd389b36SDavid du Colombier char *
doread(Ram * r,vlong off,long cnt)11641fe996aSDavid du Colombier doread(Ram *r, vlong off, long cnt)
117bd389b36SDavid du Colombier {
118bbd061d4SDavid du Colombier static char buf[Maxbuf+MAXBLSIZE];
119bd389b36SDavid du Colombier int bno, i;
120bd389b36SDavid du Colombier
121bd389b36SDavid du Colombier bno = off/BLSIZE;
122bd389b36SDavid du Colombier off -= bno*BLSIZE;
1239a747e4fSDavid du Colombier if (cnt>Maxbuf)
124bd389b36SDavid du Colombier error("count too large");
125bd389b36SDavid du Colombier if (off)
1267dd7cddfSDavid du Colombier cnt += off;
127bd389b36SDavid du Colombier i = 0;
128bd389b36SDavid du Colombier while (cnt>0) {
129bd389b36SDavid du Colombier getblk(r, bno, &buf[i*BLSIZE]);
130bd389b36SDavid du Colombier cnt -= BLSIZE;
131bd389b36SDavid du Colombier bno++;
132bd389b36SDavid du Colombier i++;
133bd389b36SDavid du Colombier }
134bd389b36SDavid du Colombier return buf;
135bd389b36SDavid du Colombier }
136bd389b36SDavid du Colombier
137bd389b36SDavid du Colombier void
dowrite(Ram * r,char * buf,long off,long cnt)138bd389b36SDavid du Colombier dowrite(Ram *r, char *buf, long off, long cnt)
139bd389b36SDavid du Colombier {
140bd389b36SDavid du Colombier USED(r); USED(buf); USED(off); USED(cnt);
141bd389b36SDavid du Colombier }
142bd389b36SDavid du Colombier
143bd389b36SDavid du Colombier int
dopermw(Ram * r)144bd389b36SDavid du Colombier dopermw(Ram *r)
145bd389b36SDavid du Colombier {
146bd389b36SDavid du Colombier USED(r);
147bd389b36SDavid du Colombier return 0;
148bd389b36SDavid du Colombier }
149bd389b36SDavid du Colombier
150bd389b36SDavid du Colombier /*
151bd389b36SDavid du Colombier * fetch an i-node
152bd389b36SDavid du Colombier * -- no sanity check for now
153bd389b36SDavid du Colombier * -- magic inode-to-disk-block stuff here
154bd389b36SDavid du Colombier */
155bd389b36SDavid du Colombier
156bd389b36SDavid du Colombier Fileinf
iget(int ino)157bd389b36SDavid du Colombier iget(int ino)
158bd389b36SDavid du Colombier {
159bbd061d4SDavid du Colombier char buf[MAXBLSIZE];
160bd389b36SDavid du Colombier struct v32dinode *dp;
161bd389b36SDavid du Colombier long flags, i;
162bd389b36SDavid du Colombier Fileinf f;
163bd389b36SDavid du Colombier
164bd389b36SDavid du Colombier seek(tapefile, BLSIZE*((ino-1)/LINOPB + VSUPERB + 1), 0);
165bd389b36SDavid du Colombier if (read(tapefile, buf, BLSIZE) != BLSIZE)
166bd389b36SDavid du Colombier error("Can't read inode");
167bd389b36SDavid du Colombier dp = ((struct v32dinode *)buf) + ((ino-1)%LINOPB);
168bd389b36SDavid du Colombier flags = g2byte(dp->flags);
169bd389b36SDavid du Colombier f.size = g4byte(dp->size);
170bd389b36SDavid du Colombier if ((flags&VFMT)==VIFCHR || (flags&VFMT)==VIFBLK)
171bd389b36SDavid du Colombier f.size = 0;
17241fe996aSDavid du Colombier f.data = emalloc(VNADDR*sizeof(long));
173bd389b36SDavid du Colombier for (i = 0; i < VNADDR; i++)
17441fe996aSDavid du Colombier ((long*)f.data)[i] = g3byte(dp->addr+3*i);
175bd389b36SDavid du Colombier f.mode = flags & VMODE;
176bd389b36SDavid du Colombier if ((flags&VFMT)==VIFDIR)
1779a747e4fSDavid du Colombier f.mode |= DMDIR;
178bd389b36SDavid du Colombier f.uid = g2byte(dp->uid);
179bd389b36SDavid du Colombier f.gid = g2byte(dp->gid);
180bd389b36SDavid du Colombier f.mdate = g4byte(dp->mtime);
181bd389b36SDavid du Colombier return f;
182bd389b36SDavid du Colombier }
183bd389b36SDavid du Colombier
184bd389b36SDavid du Colombier void
getblk(Ram * r,long bno,char * buf)185bd389b36SDavid du Colombier getblk(Ram *r, long bno, char *buf)
186bd389b36SDavid du Colombier {
187bd389b36SDavid du Colombier long dbno;
188bd389b36SDavid du Colombier
189bd389b36SDavid du Colombier if ((dbno = bmap(r, bno)) == 0) {
190bd389b36SDavid du Colombier memset(buf, 0, BLSIZE);
191bd389b36SDavid du Colombier return;
192bd389b36SDavid du Colombier }
193bd389b36SDavid du Colombier seek(tapefile, dbno*BLSIZE, 0);
194bd389b36SDavid du Colombier if (read(tapefile, buf, BLSIZE) != BLSIZE)
195bd389b36SDavid du Colombier error("bad read");
196bd389b36SDavid du Colombier }
197bd389b36SDavid du Colombier
198bd389b36SDavid du Colombier /*
199bd389b36SDavid du Colombier * logical to physical block
200bd389b36SDavid du Colombier * only singly-indirect files for now
201bd389b36SDavid du Colombier */
202bd389b36SDavid du Colombier
203bd389b36SDavid du Colombier long
bmap(Ram * r,long bno)204bd389b36SDavid du Colombier bmap(Ram *r, long bno)
205bd389b36SDavid du Colombier {
206bbd061d4SDavid du Colombier unsigned char indbuf[MAXLNINDIR][4];
207bd389b36SDavid du Colombier
208bd389b36SDavid du Colombier if (bno < VNADDR-3)
209bd389b36SDavid du Colombier return ((long*)r->data)[bno];
210bd389b36SDavid du Colombier if (bno < VNADDR*LNINDIR) {
211bd389b36SDavid du Colombier seek(tapefile, ((long *)r->data)[(bno-(VNADDR-3))/LNINDIR]*BLSIZE, 0);
212bd389b36SDavid du Colombier if (read(tapefile, (char *)indbuf, BLSIZE) != BLSIZE)
213bd389b36SDavid du Colombier return 0;
214bd389b36SDavid du Colombier return ((indbuf[bno%LNINDIR][1]<<8) + indbuf[bno%LNINDIR][0]);
215bd389b36SDavid du Colombier }
216bd389b36SDavid du Colombier return 0;
217bd389b36SDavid du Colombier }
218