chiark / gitweb /
[PATCH] udev_volume_id: volume_id version 034
[elogind.git] / extras / volume_id / volume_id / iso9660 / iso9660.c
1 /*
2  * volume_id - reads filesystem label and uuid
3  *
4  * Copyright (C) 2004 Kay Sievers <kay.sievers@vrfy.org>
5  *
6  *      This library is free software; you can redistribute it and/or
7  *      modify it under the terms of the GNU Lesser General Public
8  *      License as published by the Free Software Foundation; either
9  *      version 2.1 of the License, or (at your option) any later version.
10  *
11  *      This library is distributed in the hope that it will be useful,
12  *      but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *      MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14  *      Lesser General Public License for more details.
15  *
16  *      You should have received a copy of the GNU Lesser General Public
17  *      License along with this library; if not, write to the Free Software
18  *      Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19  */
20
21 #ifndef _GNU_SOURCE
22 #define _GNU_SOURCE 1
23 #endif
24
25 #ifdef HAVE_CONFIG_H
26 #  include <config.h>
27 #endif
28
29 #include <stdio.h>
30 #include <stdlib.h>
31 #include <unistd.h>
32 #include <string.h>
33 #include <errno.h>
34 #include <ctype.h>
35 #include <asm/types.h>
36
37 #include "../volume_id.h"
38 #include "../logging.h"
39 #include "../util.h"
40 #include "iso9660.h"
41
42 #define ISO_SUPERBLOCK_OFFSET           0x8000
43 #define ISO_SECTOR_SIZE                 0x800
44 #define ISO_VD_OFFSET                   (ISO_SUPERBLOCK_OFFSET + ISO_SECTOR_SIZE)
45 #define ISO_VD_PRIMARY                  0x1
46 #define ISO_VD_SUPPLEMENTARY            0x2
47 #define ISO_VD_END                      0xff
48 #define ISO_VD_MAX                      16
49
50 union iso_super_block {
51         struct iso_header {
52                 __u8    type;
53                 __u8    id[5];
54                 __u8    version;
55                 __u8    unused1;
56                 __u8            system_id[32];
57                 __u8            volume_id[32];
58         } __attribute__((__packed__)) iso;
59         struct hs_header {
60                 __u8    foo[8];
61                 __u8    type;
62                 __u8    id[4];
63                 __u8    version;
64         } __attribute__((__packed__)) hs;
65 } __attribute__((__packed__));
66
67 int volume_id_probe_iso9660(struct volume_id *id, __u64 off)
68 {
69         union iso_super_block *is;
70
71         dbg("probing at offset %llu", off);
72
73         is = (union iso_super_block *) volume_id_get_buffer(id, off + ISO_SUPERBLOCK_OFFSET, 0x200);
74         if (is == NULL)
75                 return -1;
76
77         if (memcmp(is->iso.id, "CD001", 5) == 0) {
78                 char root_label[VOLUME_ID_LABEL_SIZE+1];
79                 int vd_offset;
80                 int i;
81                 int found_svd;
82
83                 memset(root_label, 0, sizeof(root_label));
84                 strncpy(root_label, is->iso.volume_id, sizeof(root_label)-1);
85
86                 found_svd = 0;
87                 vd_offset = ISO_VD_OFFSET;
88                 for (i = 0; i < ISO_VD_MAX; i++) {
89                         is = (union iso_super_block *) volume_id_get_buffer(id, off + vd_offset, 0x200);
90                         if (is == NULL || is->iso.type == ISO_VD_END)
91                                 break;
92                         if (is->iso.type == ISO_VD_SUPPLEMENTARY) {
93                                 dbg("found ISO supplementary VD at offset 0x%llx", off + vd_offset);
94                                 volume_id_set_label_raw(id, is->iso.volume_id, 32);
95                                 volume_id_set_label_unicode16(id, is->iso.volume_id, BE, 32);
96                                 found_svd = 1;
97                                 break;
98                         }
99                         vd_offset += ISO_SECTOR_SIZE;
100                 }
101
102                 if (!found_svd ||
103                     (found_svd && !memcmp(root_label, id->label, 16)))
104                 {
105                         volume_id_set_label_raw(id, root_label, 32);
106                         volume_id_set_label_string(id, root_label, 32);
107                 }
108                 goto found;
109         }
110         if (memcmp(is->hs.id, "CDROM", 5) == 0)
111                 goto found;
112         return -1;
113
114 found:
115         volume_id_set_usage(id, VOLUME_ID_FILESYSTEM);
116         id->type = "iso9660";
117
118         return 0;
119 }