chiark / gitweb /
switch tools and volume_id from LGPL to GPLv2
[elogind.git] / extras / volume_id / volume_id / linux_raid.c
1 /*
2  * volume_id - reads filesystem label and uuid
3  *
4  * Copyright (C) 2004 Kay Sievers <kay.sievers@vrfy.org>
5  *
6  *      This program is free software; you can redistribute it and/or modify it
7  *      under the terms of the GNU General Public License as published by the
8  *      Free Software Foundation version 2 of the License.
9  */
10
11 #ifndef _GNU_SOURCE
12 #define _GNU_SOURCE 1
13 #endif
14
15 #ifdef HAVE_CONFIG_H
16 #  include <config.h>
17 #endif
18
19 #include <stdio.h>
20 #include <stdlib.h>
21 #include <unistd.h>
22 #include <string.h>
23 #include <errno.h>
24 #include <ctype.h>
25
26 #include "volume_id.h"
27 #include "logging.h"
28 #include "util.h"
29 #include "linux_raid.h"
30
31 struct mdp_super_block {
32         uint32_t        md_magic;
33         uint32_t        major_version;
34         uint32_t        minor_version;
35         uint32_t        patch_version;
36         uint32_t        gvalid_words;
37         uint32_t        set_uuid0;
38         uint32_t        ctime;
39         uint32_t        level;
40         uint32_t        size;
41         uint32_t        nr_disks;
42         uint32_t        raid_disks;
43         uint32_t        md_minor;
44         uint32_t        not_persistent;
45         uint32_t        set_uuid1;
46         uint32_t        set_uuid2;
47         uint32_t        set_uuid3;
48 } __attribute__((packed)) *mdp;
49
50 #define MD_RESERVED_BYTES               0x10000
51 #define MD_MAGIC                        0xa92b4efc
52
53 int volume_id_probe_linux_raid(struct volume_id *id, uint64_t off, uint64_t size)
54 {
55         const uint8_t *buf;
56         uint64_t sboff;
57         uint8_t uuid[16];
58
59         dbg("probing at offset 0x%llx, size 0x%llx",
60             (unsigned long long) off, (unsigned long long) size);
61
62         if (size < 0x10000)
63                 return -1;
64
65         sboff = (size & ~(MD_RESERVED_BYTES - 1)) - MD_RESERVED_BYTES;
66         buf = volume_id_get_buffer(id, off + sboff, 0x800);
67         if (buf == NULL)
68                 return -1;
69
70         mdp = (struct mdp_super_block *) buf;
71
72         if (le32_to_cpu(mdp->md_magic) != MD_MAGIC)
73                 return -1;
74
75         memcpy(uuid, &mdp->set_uuid0, 4);
76         memcpy(&uuid[4], &mdp->set_uuid1, 12);
77         volume_id_set_uuid(id, uuid, UUID_DCE);
78
79         snprintf(id->type_version, sizeof(id->type_version)-1, "%u.%u.%u",
80                  le32_to_cpu(mdp->major_version),
81                  le32_to_cpu(mdp->minor_version),
82                  le32_to_cpu(mdp->patch_version));
83
84         dbg("found raid signature");
85         volume_id_set_usage(id, VOLUME_ID_RAID);
86         id->type = "linux_raid_member";
87
88         return 0;
89 }