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