xref: /plan9-contrib/sys/src/ape/lib/ap/plan9/dirread.c (revision 9a747e4fd48b9f4522c70c07e8f882a15030f964)
1*9a747e4fSDavid du Colombier #include "lib.h"
2*9a747e4fSDavid du Colombier #include <string.h>
3*9a747e4fSDavid du Colombier #include <stdlib.h>
4*9a747e4fSDavid du Colombier #include "sys9.h"
5*9a747e4fSDavid du Colombier #include "dir.h"
6*9a747e4fSDavid du Colombier 
7*9a747e4fSDavid du Colombier static int
statcheck(uchar * buf,uint nbuf)8*9a747e4fSDavid du Colombier statcheck(uchar *buf, uint nbuf)
9*9a747e4fSDavid du Colombier {
10*9a747e4fSDavid du Colombier 	uchar *ebuf;
11*9a747e4fSDavid du Colombier 	int i;
12*9a747e4fSDavid du Colombier 
13*9a747e4fSDavid du Colombier 	ebuf = buf + nbuf;
14*9a747e4fSDavid du Colombier 
15*9a747e4fSDavid du Colombier 	buf += STATFIXLEN - 4 * BIT16SZ;
16*9a747e4fSDavid du Colombier 
17*9a747e4fSDavid du Colombier 	for(i = 0; i < 4; i++){
18*9a747e4fSDavid du Colombier 		if(buf + BIT16SZ > ebuf)
19*9a747e4fSDavid du Colombier 			return -1;
20*9a747e4fSDavid du Colombier 		buf += BIT16SZ + GBIT16(buf);
21*9a747e4fSDavid du Colombier 	}
22*9a747e4fSDavid du Colombier 
23*9a747e4fSDavid du Colombier 	if(buf != ebuf)
24*9a747e4fSDavid du Colombier 		return -1;
25*9a747e4fSDavid du Colombier 
26*9a747e4fSDavid du Colombier 	return 0;
27*9a747e4fSDavid du Colombier }
28*9a747e4fSDavid du Colombier 
29*9a747e4fSDavid du Colombier static
30*9a747e4fSDavid du Colombier long
dirpackage(uchar * buf,long ts,Dir ** d)31*9a747e4fSDavid du Colombier dirpackage(uchar *buf, long ts, Dir **d)
32*9a747e4fSDavid du Colombier {
33*9a747e4fSDavid du Colombier 	char *s;
34*9a747e4fSDavid du Colombier 	long ss, i, n, nn, m;
35*9a747e4fSDavid du Colombier 
36*9a747e4fSDavid du Colombier 	if(ts == 0){
37*9a747e4fSDavid du Colombier 		*d = nil;
38*9a747e4fSDavid du Colombier 		return 0;
39*9a747e4fSDavid du Colombier 	}
40*9a747e4fSDavid du Colombier 
41*9a747e4fSDavid du Colombier 	/*
42*9a747e4fSDavid du Colombier 	 * first find number of all stats, check they look like stats, & size all associated strings
43*9a747e4fSDavid du Colombier 	 */
44*9a747e4fSDavid du Colombier 	ss = 0;
45*9a747e4fSDavid du Colombier 	n = 0;
46*9a747e4fSDavid du Colombier 	for(i = 0; i < ts; i += m){
47*9a747e4fSDavid du Colombier 		m = BIT16SZ + GBIT16(&buf[i]);
48*9a747e4fSDavid du Colombier 		if(statcheck(&buf[i], m) < 0)
49*9a747e4fSDavid du Colombier 			break;
50*9a747e4fSDavid du Colombier 		ss += m;
51*9a747e4fSDavid du Colombier 		n++;
52*9a747e4fSDavid du Colombier 	}
53*9a747e4fSDavid du Colombier 
54*9a747e4fSDavid du Colombier 	if(i != ts)
55*9a747e4fSDavid du Colombier 		return -1;
56*9a747e4fSDavid du Colombier 
57*9a747e4fSDavid du Colombier 	*d = malloc(n * sizeof(Dir) + ss);
58*9a747e4fSDavid du Colombier 	if(*d == nil)
59*9a747e4fSDavid du Colombier 		return -1;
60*9a747e4fSDavid du Colombier 
61*9a747e4fSDavid du Colombier 	/*
62*9a747e4fSDavid du Colombier 	 * then convert all buffers
63*9a747e4fSDavid du Colombier 	 */
64*9a747e4fSDavid du Colombier 	s = (char*)*d + n * sizeof(Dir);
65*9a747e4fSDavid du Colombier 	nn = 0;
66*9a747e4fSDavid du Colombier 	for(i = 0; i < ts; i += m){
67*9a747e4fSDavid du Colombier 		m = BIT16SZ + GBIT16((uchar*)&buf[i]);
68*9a747e4fSDavid du Colombier 		if(nn >= n || _convM2D(&buf[i], m, *d + nn, s) != m){
69*9a747e4fSDavid du Colombier 			free(*d);
70*9a747e4fSDavid du Colombier 			return -1;
71*9a747e4fSDavid du Colombier 		}
72*9a747e4fSDavid du Colombier 		nn++;
73*9a747e4fSDavid du Colombier 		s += m;
74*9a747e4fSDavid du Colombier 	}
75*9a747e4fSDavid du Colombier 
76*9a747e4fSDavid du Colombier 	return nn;
77*9a747e4fSDavid du Colombier }
78*9a747e4fSDavid du Colombier 
79*9a747e4fSDavid du Colombier long
_dirread(int fd,Dir ** d)80*9a747e4fSDavid du Colombier _dirread(int fd, Dir **d)
81*9a747e4fSDavid du Colombier {
82*9a747e4fSDavid du Colombier 	uchar *buf;
83*9a747e4fSDavid du Colombier 	long ts;
84*9a747e4fSDavid du Colombier 
85*9a747e4fSDavid du Colombier 	buf = malloc(DIRMAX);
86*9a747e4fSDavid du Colombier 	if(buf == nil)
87*9a747e4fSDavid du Colombier 		return -1;
88*9a747e4fSDavid du Colombier 	ts = _READ(fd, buf, DIRMAX);
89*9a747e4fSDavid du Colombier 	if(ts >= 0)
90*9a747e4fSDavid du Colombier 		ts = dirpackage(buf, ts, d);
91*9a747e4fSDavid du Colombier 	free(buf);
92*9a747e4fSDavid du Colombier 	return ts;
93*9a747e4fSDavid du Colombier }
94*9a747e4fSDavid du Colombier 
95*9a747e4fSDavid du Colombier long
_dirreadall(int fd,Dir ** d)96*9a747e4fSDavid du Colombier _dirreadall(int fd, Dir **d)
97*9a747e4fSDavid du Colombier {
98*9a747e4fSDavid du Colombier 	uchar *buf, *nbuf;
99*9a747e4fSDavid du Colombier 	long n, ts;
100*9a747e4fSDavid du Colombier 
101*9a747e4fSDavid du Colombier 	buf = nil;
102*9a747e4fSDavid du Colombier 	ts = 0;
103*9a747e4fSDavid du Colombier 	for(;;){
104*9a747e4fSDavid du Colombier 		nbuf = realloc(buf, ts+DIRMAX);
105*9a747e4fSDavid du Colombier 		if(nbuf == nil){
106*9a747e4fSDavid du Colombier 			free(buf);
107*9a747e4fSDavid du Colombier 			return -1;
108*9a747e4fSDavid du Colombier 		}
109*9a747e4fSDavid du Colombier 		buf = nbuf;
110*9a747e4fSDavid du Colombier 		n = _READ(fd, buf+ts, DIRMAX);
111*9a747e4fSDavid du Colombier 		if(n <= 0)
112*9a747e4fSDavid du Colombier 			break;
113*9a747e4fSDavid du Colombier 		ts += n;
114*9a747e4fSDavid du Colombier 	}
115*9a747e4fSDavid du Colombier 	if(ts >= 0)
116*9a747e4fSDavid du Colombier 		ts = dirpackage(buf, ts, d);
117*9a747e4fSDavid du Colombier 	free(buf);
118*9a747e4fSDavid du Colombier 	return ts;
119*9a747e4fSDavid du Colombier }
120