chiark / gitweb /
91a5bcb9b0b558e426302b8449097592197fffb7
[elogind.git] / extras / volume_id / libvolume_id / reiserfs.c
1 /*
2  * volume_id - reads filesystem label and uuid
3  *
4  * Copyright (C) 2004 Kay Sievers <kay.sievers@vrfy.org>
5  * Copyright (C) 2005 Tobias Klauser <tklauser@access.unizh.ch>
6  *
7  *      This program is free software; you can redistribute it and/or modify it
8  *      under the terms of the GNU General Public License as published by the
9  *      Free Software Foundation version 2 of the License.
10  */
11
12 #ifndef _GNU_SOURCE
13 #define _GNU_SOURCE 1
14 #endif
15
16 #ifdef HAVE_CONFIG_H
17 #  include <config.h>
18 #endif
19
20 #include <stdio.h>
21 #include <stdlib.h>
22 #include <unistd.h>
23 #include <string.h>
24 #include <errno.h>
25 #include <ctype.h>
26
27 #include "volume_id.h"
28 #include "logging.h"
29 #include "util.h"
30
31 struct reiserfs_super_block {
32         uint32_t        blocks_count;
33         uint32_t        free_blocks;
34         uint32_t        root_block;
35         uint32_t        journal_block;
36         uint32_t        journal_dev;
37         uint32_t        orig_journal_size;
38         uint32_t        dummy2[5];
39         uint16_t        blocksize;
40         uint16_t        dummy3[3];
41         uint8_t         magic[12];
42         uint32_t        dummy4[5];
43         uint8_t         uuid[16];
44         uint8_t         label[16];
45 } PACKED;
46
47 struct reiser4_super_block {
48         uint8_t         magic[16];
49         uint16_t        dummy[2];
50         uint8_t         uuid[16];
51         uint8_t         label[16];
52         uint64_t        dummy2;
53 } PACKED;
54
55 #define REISERFS1_SUPERBLOCK_OFFSET             0x2000
56 #define REISERFS_SUPERBLOCK_OFFSET              0x10000
57
58 int volume_id_probe_reiserfs(struct volume_id *id, uint64_t off)
59 {
60         struct reiserfs_super_block *rs;
61         struct reiser4_super_block *rs4;
62         uint8_t  *buf;
63
64         dbg("probing at offset 0x%llx", (unsigned long long) off);
65
66         buf = volume_id_get_buffer(id, off + REISERFS_SUPERBLOCK_OFFSET, 0x200);
67         if (buf == NULL)
68                 return -1;
69
70         rs = (struct reiserfs_super_block *) buf;;
71         if (memcmp(rs->magic, "ReIsErFs", 8) == 0) {
72                 strcpy(id->type_version, "3.5");
73                 id->type = "reiserfs";
74                 goto found;
75         }
76         if (memcmp(rs->magic, "ReIsEr2Fs", 9) == 0) {
77                 strcpy(id->type_version, "3.6");
78                 id->type = "reiserfs";
79                 goto found_label;
80         }
81         if (memcmp(rs->magic, "ReIsEr3Fs", 9) == 0) {
82                 strcpy(id->type_version, "JR");
83                 id->type = "reiserfs";
84                 goto found_label;
85         }
86
87         rs4 = (struct reiser4_super_block *) buf;
88         if (memcmp(rs4->magic, "ReIsEr4", 7) == 0) {
89                 strcpy(id->type_version, "4");
90                 volume_id_set_label_raw(id, rs4->label, 16);
91                 volume_id_set_label_string(id, rs4->label, 16);
92                 volume_id_set_uuid(id, rs4->uuid, UUID_DCE);
93                 id->type = "reiser4";
94                 goto found;
95         }
96
97         buf = volume_id_get_buffer(id, off + REISERFS1_SUPERBLOCK_OFFSET, 0x200);
98         if (buf == NULL)
99                 return -1;
100
101         rs = (struct reiserfs_super_block *) buf;
102         if (memcmp(rs->magic, "ReIsErFs", 8) == 0) {
103                 strcpy(id->type_version, "3.5");
104                 id->type = "reiserfs";
105                 goto found;
106         }
107
108         return -1;
109
110 found_label:
111         volume_id_set_label_raw(id, rs->label, 16);
112         volume_id_set_label_string(id, rs->label, 16);
113         volume_id_set_uuid(id, rs->uuid, UUID_DCE);
114
115 found:
116         volume_id_set_usage(id, VOLUME_ID_FILESYSTEM);
117
118         return 0;
119 }