xref: /illumos-gate/usr/src/boot/i386/isoboot/cd9660read.c (revision 22028508fd28d36ff74dc02c5774a8ba1f0db045)
1*22028508SToomas Soome /*
2*22028508SToomas Soome  * Copyright (C) 1996 Wolfgang Solfrank.
3*22028508SToomas Soome  * Copyright (C) 1996 TooLs GmbH.
4*22028508SToomas Soome  * All rights reserved.
5*22028508SToomas Soome  *
6*22028508SToomas Soome  * Redistribution and use in source and binary forms, with or without
7*22028508SToomas Soome  * modification, are permitted provided that the following conditions
8*22028508SToomas Soome  * are met:
9*22028508SToomas Soome  * 1. Redistributions of source code must retain the above copyright
10*22028508SToomas Soome  *    notice, this list of conditions and the following disclaimer.
11*22028508SToomas Soome  * 2. Redistributions in binary form must reproduce the above copyright
12*22028508SToomas Soome  *    notice, this list of conditions and the following disclaimer in the
13*22028508SToomas Soome  *    documentation and/or other materials provided with the distribution.
14*22028508SToomas Soome  * 3. All advertising materials mentioning features or use of this software
15*22028508SToomas Soome  *    must display the following acknowledgement:
16*22028508SToomas Soome  *	This product includes software developed by TooLs GmbH.
17*22028508SToomas Soome  * 4. The name of TooLs GmbH may not be used to endorse or promote products
18*22028508SToomas Soome  *    derived from this software without specific prior written permission.
19*22028508SToomas Soome  *
20*22028508SToomas Soome  * THIS SOFTWARE IS PROVIDED BY TOOLS GMBH ``AS IS'' AND ANY EXPRESS OR
21*22028508SToomas Soome  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
22*22028508SToomas Soome  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
23*22028508SToomas Soome  * IN NO EVENT SHALL TOOLS GMBH BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24*22028508SToomas Soome  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
25*22028508SToomas Soome  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
26*22028508SToomas Soome  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
27*22028508SToomas Soome  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
28*22028508SToomas Soome  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
29*22028508SToomas Soome  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30*22028508SToomas Soome  */
31*22028508SToomas Soome 
32*22028508SToomas Soome /* Originally derived from libsa/cd9660.c: */
33*22028508SToomas Soome /*	$NetBSD: cd9660.c,v 1.5 1997/06/26 19:11:33 drochner Exp $	*/
34*22028508SToomas Soome 
35*22028508SToomas Soome #include <sys/cdefs.h>
36*22028508SToomas Soome #include <sys/param.h>
37*22028508SToomas Soome 
38*22028508SToomas Soome #include <fs/cd9660/iso.h>
39*22028508SToomas Soome #include <fs/cd9660/cd9660_rrip.h>
40*22028508SToomas Soome 
41*22028508SToomas Soome static uint64_t cd9660_lookup(const char *);
42*22028508SToomas Soome static ssize_t cd9660_fsread(uint64_t, void *, size_t);
43*22028508SToomas Soome 
44*22028508SToomas Soome #define	SUSP_CONTINUATION	"CE"
45*22028508SToomas Soome #define	SUSP_PRESENT		"SP"
46*22028508SToomas Soome #define	SUSP_STOP		"ST"
47*22028508SToomas Soome #define	SUSP_EXTREF		"ER"
48*22028508SToomas Soome #define	RRIP_NAME		"NM"
49*22028508SToomas Soome 
50*22028508SToomas Soome typedef struct {
51*22028508SToomas Soome 	ISO_SUSP_HEADER		h;
52*22028508SToomas Soome 	uint8_t signature	[ISODCL(5,    6)];
53*22028508SToomas Soome 	uint8_t len_skp		[ISODCL(7,    7)]; /* 711 */
54*22028508SToomas Soome } ISO_SUSP_PRESENT;
55*22028508SToomas Soome 
56*22028508SToomas Soome #define	cdb2devb(bno)	((bno) * ISO_DEFAULT_BLOCK_SIZE / DEV_BSIZE)
57*22028508SToomas Soome 
58*22028508SToomas Soome static int
read_iso_block(void * buffer,daddr_t blkno)59*22028508SToomas Soome read_iso_block(void *buffer, daddr_t blkno)
60*22028508SToomas Soome {
61*22028508SToomas Soome 
62*22028508SToomas Soome 	return (drvread(&dsk, buffer, cdb2devb(blkno),
63*22028508SToomas Soome 	    ISO_DEFAULT_BLOCK_SIZE / DEV_BSIZE));
64*22028508SToomas Soome }
65*22028508SToomas Soome 
66*22028508SToomas Soome static ISO_SUSP_HEADER *
susp_lookup_record(const char * identifier,struct iso_directory_record * dp,int lenskip)67*22028508SToomas Soome susp_lookup_record(const char *identifier, struct iso_directory_record *dp,
68*22028508SToomas Soome     int lenskip)
69*22028508SToomas Soome {
70*22028508SToomas Soome 	static char susp_buffer[ISO_DEFAULT_BLOCK_SIZE];
71*22028508SToomas Soome 	ISO_SUSP_HEADER *sh;
72*22028508SToomas Soome 	ISO_RRIP_CONT *shc;
73*22028508SToomas Soome 	char *p, *end;
74*22028508SToomas Soome 	int error;
75*22028508SToomas Soome 
76*22028508SToomas Soome 	p = dp->name + isonum_711(dp->name_len) + lenskip;
77*22028508SToomas Soome 	/* Names of even length have a padding byte after the name. */
78*22028508SToomas Soome 	if ((isonum_711(dp->name_len) & 1) == 0)
79*22028508SToomas Soome 		p++;
80*22028508SToomas Soome 	end = (char *)dp + isonum_711(dp->length);
81*22028508SToomas Soome 	while (p + 3 < end) {
82*22028508SToomas Soome 		sh = (ISO_SUSP_HEADER *)p;
83*22028508SToomas Soome 		if (bcmp(sh->type, identifier, 2) == 0)
84*22028508SToomas Soome 			return (sh);
85*22028508SToomas Soome 		if (bcmp(sh->type, SUSP_STOP, 2) == 0)
86*22028508SToomas Soome 			return (NULL);
87*22028508SToomas Soome 		if (bcmp(sh->type, SUSP_CONTINUATION, 2) == 0) {
88*22028508SToomas Soome 			shc = (ISO_RRIP_CONT *)sh;
89*22028508SToomas Soome 			error = read_iso_block(susp_buffer,
90*22028508SToomas Soome 			    isonum_733(shc->location));
91*22028508SToomas Soome 
92*22028508SToomas Soome 			/* Bail if it fails. */
93*22028508SToomas Soome 			if (error != 0)
94*22028508SToomas Soome 				return (NULL);
95*22028508SToomas Soome 			p = susp_buffer + isonum_733(shc->offset);
96*22028508SToomas Soome 			end = p + isonum_733(shc->length);
97*22028508SToomas Soome 		} else {
98*22028508SToomas Soome 			/* Ignore this record and skip to the next. */
99*22028508SToomas Soome 			p += isonum_711(sh->length);
100*22028508SToomas Soome 
101*22028508SToomas Soome 			/* Avoid infinite loops with corrupted file systems */
102*22028508SToomas Soome 			if (isonum_711(sh->length) == 0)
103*22028508SToomas Soome 				return (NULL);
104*22028508SToomas Soome 		}
105*22028508SToomas Soome 	}
106*22028508SToomas Soome 	return (NULL);
107*22028508SToomas Soome }
108*22028508SToomas Soome 
109*22028508SToomas Soome static const char *
rrip_lookup_name(struct iso_directory_record * dp,int lenskip,size_t * len)110*22028508SToomas Soome rrip_lookup_name(struct iso_directory_record *dp, int lenskip, size_t *len)
111*22028508SToomas Soome {
112*22028508SToomas Soome 	ISO_RRIP_ALTNAME *p;
113*22028508SToomas Soome 
114*22028508SToomas Soome 	if (len == NULL)
115*22028508SToomas Soome 		return (NULL);
116*22028508SToomas Soome 
117*22028508SToomas Soome 	p = (ISO_RRIP_ALTNAME *)susp_lookup_record(RRIP_NAME, dp, lenskip);
118*22028508SToomas Soome 	if (p == NULL)
119*22028508SToomas Soome 		return (NULL);
120*22028508SToomas Soome 	switch (*p->flags) {
121*22028508SToomas Soome 	case ISO_SUSP_CFLAG_CURRENT:
122*22028508SToomas Soome 		*len = 1;
123*22028508SToomas Soome 		return (".");
124*22028508SToomas Soome 	case ISO_SUSP_CFLAG_PARENT:
125*22028508SToomas Soome 		*len = 2;
126*22028508SToomas Soome 		return ("..");
127*22028508SToomas Soome 	case 0:
128*22028508SToomas Soome 		*len = isonum_711(p->h.length) - 5;
129*22028508SToomas Soome 		return ((char *)p + 5);
130*22028508SToomas Soome 	default:
131*22028508SToomas Soome 		/*
132*22028508SToomas Soome 		 * We don't handle hostnames or continued names as they are
133*22028508SToomas Soome 		 * too hard, so just bail and use the default name.
134*22028508SToomas Soome 		 */
135*22028508SToomas Soome 		return (NULL);
136*22028508SToomas Soome 	}
137*22028508SToomas Soome }
138*22028508SToomas Soome 
139*22028508SToomas Soome static int
rrip_check(struct iso_directory_record * dp,int * lenskip)140*22028508SToomas Soome rrip_check(struct iso_directory_record *dp, int *lenskip)
141*22028508SToomas Soome {
142*22028508SToomas Soome 	ISO_SUSP_PRESENT *sp;
143*22028508SToomas Soome 	ISO_RRIP_EXTREF *er;
144*22028508SToomas Soome 	char *p;
145*22028508SToomas Soome 
146*22028508SToomas Soome 	/* First, see if we can find a SP field. */
147*22028508SToomas Soome 	p = dp->name + isonum_711(dp->name_len);
148*22028508SToomas Soome 	if (p > (char *)dp + isonum_711(dp->length)) {
149*22028508SToomas Soome 		return (0);
150*22028508SToomas Soome 	}
151*22028508SToomas Soome 	sp = (ISO_SUSP_PRESENT *)p;
152*22028508SToomas Soome 	if (bcmp(sp->h.type, SUSP_PRESENT, 2) != 0) {
153*22028508SToomas Soome 		return (0);
154*22028508SToomas Soome 	}
155*22028508SToomas Soome 	if (isonum_711(sp->h.length) != sizeof (ISO_SUSP_PRESENT)) {
156*22028508SToomas Soome 		return (0);
157*22028508SToomas Soome 	}
158*22028508SToomas Soome 	if (sp->signature[0] != 0xbe || sp->signature[1] != 0xef) {
159*22028508SToomas Soome 		return (0);
160*22028508SToomas Soome 	}
161*22028508SToomas Soome 	*lenskip = isonum_711(sp->len_skp);
162*22028508SToomas Soome 
163*22028508SToomas Soome 	/*
164*22028508SToomas Soome 	 * Now look for an ER field.  If RRIP is present, then there must
165*22028508SToomas Soome 	 * be at least one of these.  It would be more pedantic to walk
166*22028508SToomas Soome 	 * through the list of fields looking for a Rock Ridge ER field.
167*22028508SToomas Soome 	 */
168*22028508SToomas Soome 	er = (ISO_RRIP_EXTREF *)susp_lookup_record(SUSP_EXTREF, dp, 0);
169*22028508SToomas Soome 	if (er == NULL) {
170*22028508SToomas Soome 		return (0);
171*22028508SToomas Soome 	}
172*22028508SToomas Soome 	return (1);
173*22028508SToomas Soome }
174*22028508SToomas Soome 
175*22028508SToomas Soome static int
dirmatch(const char * path,struct iso_directory_record * dp,int use_rrip,int lenskip)176*22028508SToomas Soome dirmatch(const char *path, struct iso_directory_record *dp, int use_rrip,
177*22028508SToomas Soome     int lenskip)
178*22028508SToomas Soome {
179*22028508SToomas Soome 	size_t len;
180*22028508SToomas Soome 	const char *cp = NULL;
181*22028508SToomas Soome 	int i, icase;
182*22028508SToomas Soome 
183*22028508SToomas Soome 	if (use_rrip)
184*22028508SToomas Soome 		cp = rrip_lookup_name(dp, lenskip, &len);
185*22028508SToomas Soome 	else
186*22028508SToomas Soome 		cp = NULL;
187*22028508SToomas Soome 	if (cp == NULL) {
188*22028508SToomas Soome 		len = isonum_711(dp->name_len);
189*22028508SToomas Soome 		cp = dp->name;
190*22028508SToomas Soome 		icase = 1;
191*22028508SToomas Soome 	} else
192*22028508SToomas Soome 		icase = 0;
193*22028508SToomas Soome 	for (i = len; --i >= 0; path++, cp++) {
194*22028508SToomas Soome 		if (!*path || *path == '/')
195*22028508SToomas Soome 			break;
196*22028508SToomas Soome 		if (*path == *cp)
197*22028508SToomas Soome 			continue;
198*22028508SToomas Soome 		if (!icase && toupper(*path) == *cp)
199*22028508SToomas Soome 			continue;
200*22028508SToomas Soome 		return (0);
201*22028508SToomas Soome 	}
202*22028508SToomas Soome 	if (*path && *path != '/') {
203*22028508SToomas Soome 		return (0);
204*22028508SToomas Soome 	}
205*22028508SToomas Soome 	/*
206*22028508SToomas Soome 	 * Allow stripping of trailing dots and the version number.
207*22028508SToomas Soome 	 * Note that this will find the first instead of the last version
208*22028508SToomas Soome 	 * of a file.
209*22028508SToomas Soome 	 */
210*22028508SToomas Soome 	if (i >= 0 && (*cp == ';' || *cp == '.')) {
211*22028508SToomas Soome 		/* This is to prevent matching of numeric extensions */
212*22028508SToomas Soome 		if (*cp == '.' && cp[1] != ';') {
213*22028508SToomas Soome 			return (0);
214*22028508SToomas Soome 		}
215*22028508SToomas Soome 		while (--i >= 0)
216*22028508SToomas Soome 			if (*++cp != ';' && (*cp < '0' || *cp > '9')) {
217*22028508SToomas Soome 				return (0);
218*22028508SToomas Soome 			}
219*22028508SToomas Soome 	}
220*22028508SToomas Soome 	return (1);
221*22028508SToomas Soome }
222*22028508SToomas Soome 
223*22028508SToomas Soome static uint64_t
cd9660_lookup(const char * path)224*22028508SToomas Soome cd9660_lookup(const char *path)
225*22028508SToomas Soome {
226*22028508SToomas Soome 	static char blkbuf[MAX(ISO_DEFAULT_BLOCK_SIZE,
227*22028508SToomas Soome 	    sizeof (struct iso_primary_descriptor))];
228*22028508SToomas Soome 	struct iso_primary_descriptor *vd;
229*22028508SToomas Soome 	struct iso_directory_record rec;
230*22028508SToomas Soome 	struct iso_directory_record *dp = NULL;
231*22028508SToomas Soome 	size_t dsize, off;
232*22028508SToomas Soome 	daddr_t bno, boff;
233*22028508SToomas Soome 	int rc, first, use_rrip, lenskip;
234*22028508SToomas Soome 	uint64_t cookie;
235*22028508SToomas Soome 
236*22028508SToomas Soome 	for (bno = 16; ; bno++) {
237*22028508SToomas Soome 		rc = read_iso_block(blkbuf, bno);
238*22028508SToomas Soome 		if (rc != 0)
239*22028508SToomas Soome 			return (0);
240*22028508SToomas Soome 		vd = (struct iso_primary_descriptor *)blkbuf;
241*22028508SToomas Soome 
242*22028508SToomas Soome 		if (bcmp(vd->id, ISO_STANDARD_ID, sizeof (vd->id)) != 0)
243*22028508SToomas Soome 			return (0);
244*22028508SToomas Soome 		if (isonum_711(vd->type) == ISO_VD_END)
245*22028508SToomas Soome 			return (0);
246*22028508SToomas Soome 		if (isonum_711(vd->type) == ISO_VD_PRIMARY)
247*22028508SToomas Soome 			break;
248*22028508SToomas Soome 	}
249*22028508SToomas Soome 
250*22028508SToomas Soome 	bcopy(vd->root_directory_record, &rec, sizeof (rec));
251*22028508SToomas Soome 	if (*path == '/') path++; /* eat leading '/' */
252*22028508SToomas Soome 
253*22028508SToomas Soome 	first = 1;
254*22028508SToomas Soome 	use_rrip = 0;
255*22028508SToomas Soome 	lenskip = 0;
256*22028508SToomas Soome 	while (*path) {
257*22028508SToomas Soome 		bno = isonum_733(rec.extent) + isonum_711(rec.ext_attr_length);
258*22028508SToomas Soome 		dsize = isonum_733(rec.size);
259*22028508SToomas Soome 		off = 0;
260*22028508SToomas Soome 		boff = 0;
261*22028508SToomas Soome 
262*22028508SToomas Soome 		while (off < dsize) {
263*22028508SToomas Soome 			if ((off % ISO_DEFAULT_BLOCK_SIZE) == 0) {
264*22028508SToomas Soome 				rc = read_iso_block(blkbuf, bno + boff);
265*22028508SToomas Soome 				if (rc) {
266*22028508SToomas Soome 					return (0);
267*22028508SToomas Soome 				}
268*22028508SToomas Soome 				boff++;
269*22028508SToomas Soome 				dp = (struct iso_directory_record *)blkbuf;
270*22028508SToomas Soome 			}
271*22028508SToomas Soome 			if (isonum_711(dp->length) == 0) {
272*22028508SToomas Soome 				/* skip to next block, if any */
273*22028508SToomas Soome 				off = boff * ISO_DEFAULT_BLOCK_SIZE;
274*22028508SToomas Soome 				continue;
275*22028508SToomas Soome 			}
276*22028508SToomas Soome 
277*22028508SToomas Soome 			/* See if RRIP is in use. */
278*22028508SToomas Soome 			if (first)
279*22028508SToomas Soome 				use_rrip = rrip_check(dp, &lenskip);
280*22028508SToomas Soome 
281*22028508SToomas Soome 			if (dirmatch(path, dp, use_rrip,
282*22028508SToomas Soome 			    first ? 0 : lenskip)) {
283*22028508SToomas Soome 				first = 0;
284*22028508SToomas Soome 				break;
285*22028508SToomas Soome 			} else
286*22028508SToomas Soome 				first = 0;
287*22028508SToomas Soome 
288*22028508SToomas Soome 			dp = (struct iso_directory_record *)
289*22028508SToomas Soome 			    ((char *)dp + isonum_711(dp->length));
290*22028508SToomas Soome 			/* If the new block has zero length, it is padding. */
291*22028508SToomas Soome 			if (isonum_711(dp->length) == 0) {
292*22028508SToomas Soome 				/* Skip to next block, if any. */
293*22028508SToomas Soome 				off = boff * ISO_DEFAULT_BLOCK_SIZE;
294*22028508SToomas Soome 				continue;
295*22028508SToomas Soome 			}
296*22028508SToomas Soome 			off += isonum_711(dp->length);
297*22028508SToomas Soome 		}
298*22028508SToomas Soome 		if (off >= dsize) {
299*22028508SToomas Soome 			return (0);
300*22028508SToomas Soome 		}
301*22028508SToomas Soome 
302*22028508SToomas Soome 		rec = *dp;
303*22028508SToomas Soome 		while (*path && *path != '/') /* look for next component */
304*22028508SToomas Soome 			path++;
305*22028508SToomas Soome 		if (*path) path++; /* skip '/' */
306*22028508SToomas Soome 	}
307*22028508SToomas Soome 
308*22028508SToomas Soome 	if ((isonum_711(rec.flags) & 2) != 0) {
309*22028508SToomas Soome 		return (0);
310*22028508SToomas Soome 	}
311*22028508SToomas Soome 
312*22028508SToomas Soome 	cookie = isonum_733(rec.extent) + isonum_711(rec.ext_attr_length);
313*22028508SToomas Soome 	cookie = (cookie << 32) | isonum_733(rec.size);
314*22028508SToomas Soome 
315*22028508SToomas Soome 	return (cookie);
316*22028508SToomas Soome }
317*22028508SToomas Soome 
318*22028508SToomas Soome static ssize_t
cd9660_fsread(uint64_t cookie,void * buf,size_t nbytes)319*22028508SToomas Soome cd9660_fsread(uint64_t cookie, void *buf, size_t nbytes)
320*22028508SToomas Soome {
321*22028508SToomas Soome 	static char blkbuf[ISO_DEFAULT_BLOCK_SIZE];
322*22028508SToomas Soome 	static daddr_t curstart = 0, curblk = 0;
323*22028508SToomas Soome 	daddr_t blk, blk_off;
324*22028508SToomas Soome 	off_t byte_off;
325*22028508SToomas Soome 	size_t size, remaining, n;
326*22028508SToomas Soome 	char *s;
327*22028508SToomas Soome 
328*22028508SToomas Soome 	size = cookie & 0xffffffff;
329*22028508SToomas Soome 	blk = (cookie >> 32) & 0xffffffff;
330*22028508SToomas Soome 
331*22028508SToomas Soome 	/* Make sure we're looking at the right file. */
332*22028508SToomas Soome 	if ((uint64_t)((blk << 32) | size) != cookie) {
333*22028508SToomas Soome 		return (-1);
334*22028508SToomas Soome 	}
335*22028508SToomas Soome 
336*22028508SToomas Soome 	if (blk != curstart) {
337*22028508SToomas Soome 		curstart = blk;
338*22028508SToomas Soome 		fs_off = 0;
339*22028508SToomas Soome 	}
340*22028508SToomas Soome 
341*22028508SToomas Soome 	size -= fs_off;
342*22028508SToomas Soome 	if (size < nbytes) {
343*22028508SToomas Soome 		nbytes = size;
344*22028508SToomas Soome 	}
345*22028508SToomas Soome 	remaining = nbytes;
346*22028508SToomas Soome 	s = buf;
347*22028508SToomas Soome 
348*22028508SToomas Soome 	while (remaining > 0) {
349*22028508SToomas Soome 		blk_off = fs_off >> ISO_DEFAULT_BLOCK_SHIFT;
350*22028508SToomas Soome 		byte_off = fs_off & (ISO_DEFAULT_BLOCK_SIZE - 1);
351*22028508SToomas Soome 
352*22028508SToomas Soome 		if (curblk != curstart + blk_off) {
353*22028508SToomas Soome 			curblk = curstart + blk_off;
354*22028508SToomas Soome 			read_iso_block(blkbuf, curblk);
355*22028508SToomas Soome 		}
356*22028508SToomas Soome 
357*22028508SToomas Soome 		if (remaining < ISO_DEFAULT_BLOCK_SIZE - byte_off) {
358*22028508SToomas Soome 			n = remaining;
359*22028508SToomas Soome 		} else {
360*22028508SToomas Soome 			n = ISO_DEFAULT_BLOCK_SIZE - byte_off;
361*22028508SToomas Soome 		}
362*22028508SToomas Soome 		memcpy(s, blkbuf + byte_off, n);
363*22028508SToomas Soome 		remaining -= n;
364*22028508SToomas Soome 		s += n;
365*22028508SToomas Soome 
366*22028508SToomas Soome 		fs_off += n;
367*22028508SToomas Soome 	}
368*22028508SToomas Soome 
369*22028508SToomas Soome 	return (nbytes);
370*22028508SToomas Soome }
371