xref: /plan9/sys/src/cmd/tapefs/v6fs.c (revision 9f2726c34299ea5a81cda1b22133dd5a4b421e04)
1bd389b36SDavid du Colombier /*
2bd389b36SDavid du Colombier  * old (V6 and before) PDP-11 Unix filesystem
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  * v6 disk inode
12bd389b36SDavid du Colombier  */
13bd389b36SDavid du Colombier #define	V6NADDR	8
14bd389b36SDavid du Colombier #define	V6FMT	0160000
15bd389b36SDavid du Colombier #define	V6IFREG	0100000
16bd389b36SDavid du Colombier #define	V6IFDIR	0140000
17bd389b36SDavid du Colombier #define	V6IFCHR	0120000
18bd389b36SDavid du Colombier #define	V6IFBLK	0160000
19bd389b36SDavid du Colombier #define	V6MODE	0777
20bd389b36SDavid du Colombier #define	V6LARGE	010000
21bd389b36SDavid du Colombier #define	V6SUPERB	1
22bd389b36SDavid du Colombier #define	V6ROOT		1	/* root inode */
23bd389b36SDavid du Colombier #define	V6NAMELEN	14
24bd389b36SDavid du Colombier #define	BLSIZE	512
25bd389b36SDavid du Colombier #define	LINOPB	(BLSIZE/sizeof(struct v6dinode))
26bd389b36SDavid du Colombier #define	LNINDIR	(BLSIZE/sizeof(unsigned short))
27bd389b36SDavid du Colombier 
28bd389b36SDavid du Colombier struct v6dinode {
29bd389b36SDavid du Colombier 	unsigned char flags[2];
30bd389b36SDavid du Colombier 	unsigned char nlinks;
31bd389b36SDavid du Colombier 	unsigned char uid;
32bd389b36SDavid du Colombier 	unsigned char gid;
33bd389b36SDavid du Colombier 	unsigned char hisize;
34bd389b36SDavid du Colombier 	unsigned char losize[2];
35bd389b36SDavid du Colombier 	unsigned char addr[V6NADDR][2];
36bd389b36SDavid du Colombier 	unsigned char atime[4];	/* pdp-11 order */
37bd389b36SDavid du Colombier 	unsigned char mtime[4];	/* pdp-11 order */
38bd389b36SDavid du Colombier };
39bd389b36SDavid du Colombier 
40bd389b36SDavid du Colombier struct	v6dir {
41bd389b36SDavid du Colombier 	uchar	ino[2];
42bd389b36SDavid du Colombier 	char	name[V6NAMELEN];
43bd389b36SDavid du Colombier };
44bd389b36SDavid du Colombier 
45bd389b36SDavid du Colombier int	tapefile;
46bd389b36SDavid du Colombier Fileinf	iget(int ino);
47bd389b36SDavid du Colombier long	bmap(Ram *r, long bno);
48bd389b36SDavid du Colombier void	getblk(Ram *r, long bno, char *buf);
49bd389b36SDavid du Colombier 
50bd389b36SDavid du Colombier void
populate(char * name)51bd389b36SDavid du Colombier populate(char *name)
52bd389b36SDavid du Colombier {
53bd389b36SDavid du Colombier 	Fileinf f;
54bd389b36SDavid du Colombier 
55bd389b36SDavid du Colombier 	replete = 0;
56bd389b36SDavid du Colombier 	tapefile = open(name, OREAD);
57bd389b36SDavid du Colombier 	if (tapefile<0)
58bd389b36SDavid du Colombier 		error("Can't open argument file");
59bd389b36SDavid du Colombier 	f = iget(V6ROOT);
60bd389b36SDavid du Colombier 	ram->perm = f.mode;
61bd389b36SDavid du Colombier 	ram->mtime = f.mdate;
6241fe996aSDavid du Colombier 	ram->addr = f.addr;
6341fe996aSDavid du Colombier 	ram->data = f.data;
64bd389b36SDavid du Colombier 	ram->ndata = f.size;
65bd389b36SDavid du Colombier }
66bd389b36SDavid du Colombier 
67bd389b36SDavid du Colombier void
popdir(Ram * r)68bd389b36SDavid du Colombier popdir(Ram *r)
69bd389b36SDavid du Colombier {
70bd389b36SDavid du Colombier 	int i, ino;
71bd389b36SDavid du Colombier 	char *cp;
72bd389b36SDavid du Colombier 	struct v6dir *dp;
73bd389b36SDavid du Colombier 	Fileinf f;
74bd389b36SDavid du Colombier 	char name[V6NAMELEN+1];
75bd389b36SDavid du Colombier 
767dd7cddfSDavid du Colombier 	cp = 0;
77bd389b36SDavid du Colombier 	for (i=0; i<r->ndata; i+=sizeof(struct v6dir)) {
78bd389b36SDavid du Colombier 		if (i%BLSIZE==0)
79bd389b36SDavid du Colombier 			cp = doread(r, i, BLSIZE);
80bd389b36SDavid du Colombier 		dp = (struct v6dir *)(cp+i%BLSIZE);
81bd389b36SDavid du Colombier 		ino = dp->ino[0] + (dp->ino[1]<<8);
82bd389b36SDavid du Colombier 		if (strcmp(dp->name, ".")==0 || strcmp(dp->name, "..")==0)
83bd389b36SDavid du Colombier 			continue;
84bd389b36SDavid du Colombier 		if (ino==0)
85bd389b36SDavid du Colombier 			continue;
86bd389b36SDavid du Colombier 		f = iget(ino);
87bd389b36SDavid du Colombier 		strncpy(name, dp->name, V6NAMELEN);
88*9f2726c3SDavid du Colombier 		name[V6NAMELEN] = '\0';
89bd389b36SDavid du Colombier 		f.name = name;
90bd389b36SDavid du Colombier 		popfile(r, f);
91bd389b36SDavid du Colombier 	}
92bd389b36SDavid du Colombier 	r->replete = 1;
93bd389b36SDavid du Colombier }
94bd389b36SDavid du Colombier 
95bd389b36SDavid du Colombier void
dotrunc(Ram * r)96bd389b36SDavid du Colombier dotrunc(Ram *r)
97bd389b36SDavid du Colombier {
98bd389b36SDavid du Colombier 	USED(r);
99bd389b36SDavid du Colombier }
100bd389b36SDavid du Colombier 
101bd389b36SDavid du Colombier void
docreate(Ram * r)102bd389b36SDavid du Colombier docreate(Ram *r)
103bd389b36SDavid du Colombier {
104bd389b36SDavid du Colombier 	USED(r);
105bd389b36SDavid du Colombier }
106bd389b36SDavid du Colombier 
107bd389b36SDavid du Colombier char *
doread(Ram * r,vlong off,long cnt)10841fe996aSDavid du Colombier doread(Ram *r, vlong off, long cnt)
109bd389b36SDavid du Colombier {
1109a747e4fSDavid du Colombier 	static char buf[Maxbuf+BLSIZE];
111bd389b36SDavid du Colombier 	int bno, i;
112bd389b36SDavid du Colombier 
113bd389b36SDavid du Colombier 	bno = off/BLSIZE;
114bd389b36SDavid du Colombier 	off -= bno*BLSIZE;
1159a747e4fSDavid du Colombier 	if (cnt>Maxbuf)
116bd389b36SDavid du Colombier 		error("count too large");
117bd389b36SDavid du Colombier 	if (off)
1187dd7cddfSDavid du Colombier 		cnt += off;
119bd389b36SDavid du Colombier 	i = 0;
120bd389b36SDavid du Colombier 	while (cnt>0) {
121bd389b36SDavid du Colombier 		getblk(r, bno, &buf[i*BLSIZE]);
122bd389b36SDavid du Colombier 		cnt -= BLSIZE;
123bd389b36SDavid du Colombier 		bno++;
124bd389b36SDavid du Colombier 		i++;
125bd389b36SDavid du Colombier 	}
126bd389b36SDavid du Colombier 	return buf;
127bd389b36SDavid du Colombier }
128bd389b36SDavid du Colombier 
129bd389b36SDavid du Colombier void
dowrite(Ram * r,char * buf,long off,long cnt)130bd389b36SDavid du Colombier dowrite(Ram *r, char *buf, long off, long cnt)
131bd389b36SDavid du Colombier {
132bd389b36SDavid du Colombier 	USED(r); USED(buf); USED(off); USED(cnt);
133bd389b36SDavid du Colombier }
134bd389b36SDavid du Colombier 
135bd389b36SDavid du Colombier int
dopermw(Ram * r)136bd389b36SDavid du Colombier dopermw(Ram *r)
137bd389b36SDavid du Colombier {
138bd389b36SDavid du Colombier 	USED(r);
139bd389b36SDavid du Colombier 	return 0;
140bd389b36SDavid du Colombier }
141bd389b36SDavid du Colombier 
142bd389b36SDavid du Colombier /*
143bd389b36SDavid du Colombier  * fetch an i-node
144bd389b36SDavid du Colombier  * -- no sanity check for now
145bd389b36SDavid du Colombier  * -- magic inode-to-disk-block stuff here
146bd389b36SDavid du Colombier  */
147bd389b36SDavid du Colombier 
148bd389b36SDavid du Colombier Fileinf
iget(int ino)149bd389b36SDavid du Colombier iget(int ino)
150bd389b36SDavid du Colombier {
151bd389b36SDavid du Colombier 	char buf[BLSIZE];
152bd389b36SDavid du Colombier 	struct v6dinode *dp;
153bd389b36SDavid du Colombier 	long flags, i;
154bd389b36SDavid du Colombier 	Fileinf f;
155bd389b36SDavid du Colombier 
156bd389b36SDavid du Colombier 	seek(tapefile, BLSIZE*((ino-1)/LINOPB + V6SUPERB + 1), 0);
157bd389b36SDavid du Colombier 	if (read(tapefile, buf, BLSIZE) != BLSIZE)
158bd389b36SDavid du Colombier 		error("Can't read inode");
159bd389b36SDavid du Colombier 	dp = ((struct v6dinode *)buf) + ((ino-1)%LINOPB);
160bd389b36SDavid du Colombier 	flags = (dp->flags[1]<<8) + dp->flags[0];
161bd389b36SDavid du Colombier 	f.size = (dp->hisize << 16) + (dp->losize[1]<<8) + dp->losize[0];
162bd389b36SDavid du Colombier 	if ((flags&V6FMT)==V6IFCHR || (flags&V6FMT)==V6IFBLK)
163bd389b36SDavid du Colombier 		f.size = 0;
16441fe996aSDavid du Colombier 	f.data = emalloc(V6NADDR*sizeof(ushort));
165bd389b36SDavid du Colombier 	for (i = 0; i < V6NADDR; i++)
16641fe996aSDavid du Colombier 		((ushort*)f.data)[i] = (dp->addr[i][1]<<8) + dp->addr[i][0];
167bd389b36SDavid du Colombier 	f.mode = flags & V6MODE;
168bd389b36SDavid du Colombier 	if ((flags&V6FMT)==V6IFDIR)
1699a747e4fSDavid du Colombier 		f.mode |= DMDIR;
170bd389b36SDavid du Colombier 	f.uid = dp->uid;
171bd389b36SDavid du Colombier 	f.gid = dp->gid;
172bd389b36SDavid du Colombier 	f.mdate = (dp->mtime[2]<<0) + (dp->mtime[3]<<8)
173bd389b36SDavid du Colombier 	     +(dp->mtime[0]<<16) + (dp->mtime[1]<<24);
174bd389b36SDavid du Colombier 	return f;
175bd389b36SDavid du Colombier }
176bd389b36SDavid du Colombier 
177bd389b36SDavid du Colombier void
getblk(Ram * r,long bno,char * buf)178bd389b36SDavid du Colombier getblk(Ram *r, long bno, char *buf)
179bd389b36SDavid du Colombier {
180bd389b36SDavid du Colombier 	long dbno;
181bd389b36SDavid du Colombier 
182bd389b36SDavid du Colombier 	if ((dbno = bmap(r, bno)) == 0) {
183bd389b36SDavid du Colombier 		memset(buf, 0, BLSIZE);
184bd389b36SDavid du Colombier 		return;
185bd389b36SDavid du Colombier 	}
186bd389b36SDavid du Colombier 	seek(tapefile, dbno*BLSIZE, 0);
187bd389b36SDavid du Colombier 	if (read(tapefile, buf, BLSIZE) != BLSIZE)
188bd389b36SDavid du Colombier 		error("bad read");
189bd389b36SDavid du Colombier }
190bd389b36SDavid du Colombier 
191bd389b36SDavid du Colombier /*
192bd389b36SDavid du Colombier  * logical to physical block
193bd389b36SDavid du Colombier  * only singly-indirect files for now
194bd389b36SDavid du Colombier  */
195bd389b36SDavid du Colombier 
196bd389b36SDavid du Colombier long
bmap(Ram * r,long bno)197bd389b36SDavid du Colombier bmap(Ram *r, long bno)
198bd389b36SDavid du Colombier {
199bd389b36SDavid du Colombier 	unsigned char indbuf[LNINDIR][2];
200bd389b36SDavid du Colombier 
20180ee5cbfSDavid du Colombier 	if (r->ndata <= V6NADDR*BLSIZE) {	/* assume size predicts largeness of file */
202bd389b36SDavid du Colombier 		if (bno < V6NADDR)
203bd389b36SDavid du Colombier 			return ((ushort*)r->data)[bno];
204bd389b36SDavid du Colombier 		return 0;
205bd389b36SDavid du Colombier 	}
206bd389b36SDavid du Colombier 	if (bno < V6NADDR*LNINDIR) {
207bd389b36SDavid du Colombier 		seek(tapefile, ((ushort *)r->data)[bno/LNINDIR]*BLSIZE, 0);
208bd389b36SDavid du Colombier 		if (read(tapefile, (char *)indbuf, BLSIZE) != BLSIZE)
209bd389b36SDavid du Colombier 			return 0;
210bd389b36SDavid du Colombier 		return ((indbuf[bno%LNINDIR][1]<<8) + indbuf[bno%LNINDIR][0]);
211bd389b36SDavid du Colombier 	}
212bd389b36SDavid du Colombier 	return 0;
213bd389b36SDavid du Colombier }
214