1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
|
/*
* Copyright (C) 1999 by Andries Brouwer
* Copyright (C) 1999, 2000, 2003 by Theodore Ts'o
* Copyright (C) 2001 by Andreas Dilger
* Copyright (C) 2004 Kay Sievers <kay.sievers@vrfy.org>
* Copyright (C) 2008 Karel Zak <kzak@redhat.com>
*
* This file may be redistributed under the terms of the
* GNU Lesser General Public License.
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <stdint.h>
#include "superblocks.h"
#include "crc32.h"
struct cramfs_super
{
uint8_t magic[4];
uint32_t size;
uint32_t flags;
uint32_t future;
uint8_t signature[16];
struct cramfs_info
{
uint32_t crc;
uint32_t edition;
uint32_t blocks;
uint32_t files;
} __attribute__((packed)) info;
uint8_t name[16];
} __attribute__((packed));
#define CRAMFS_FLAG_FSID_VERSION_2 0x00000001 /* fsid version #2 */
static int cramfs_is_little_endian(const struct blkid_idmag *mag)
{
assert(mag->len == 4);
return memcmp(mag->magic, "\x45\x3d\xcd\x28", 4) == 0;
}
static uint32_t cfs32_to_cpu(int le, uint32_t value)
{
if (le)
return le32_to_cpu(value);
else
return be32_to_cpu(value);
}
static int cramfs_verify_csum(blkid_probe pr, const struct blkid_idmag *mag,
struct cramfs_super *cs, int le)
{
uint32_t crc, expected, csummed_size;
unsigned char *csummed;
expected = cfs32_to_cpu(le, cs->info.crc);
csummed_size = cfs32_to_cpu(le, cs->size);
if (csummed_size > (1 << 16)
|| csummed_size < sizeof(struct cramfs_super))
return 0;
csummed = blkid_probe_get_sb_buffer(pr, mag, csummed_size);
if (!csummed)
return 0;
memset(csummed + offsetof(struct cramfs_super, info.crc), 0, sizeof(uint32_t));
crc = ~ul_crc32(~0LL, csummed, csummed_size);
return blkid_probe_verify_csum(pr, crc, expected);
}
static int probe_cramfs(blkid_probe pr, const struct blkid_idmag *mag)
{
struct cramfs_super *cs;
cs = blkid_probe_get_sb(pr, mag, struct cramfs_super);
if (!cs)
return errno ? -errno : 1;
int le = cramfs_is_little_endian(mag);
int v2 = cfs32_to_cpu(le, cs->flags) & CRAMFS_FLAG_FSID_VERSION_2;
if (v2 && !cramfs_verify_csum(pr, mag, cs, le))
return 1;
blkid_probe_set_label(pr, cs->name, sizeof(cs->name));
blkid_probe_set_fssize(pr, cfs32_to_cpu(le, cs->size));
blkid_probe_sprintf_version(pr, "%d", v2 ? 2 : 1);
blkid_probe_set_fsendianness(pr,
le ? BLKID_ENDIANNESS_LITTLE : BLKID_ENDIANNESS_BIG);
return 0;
}
const struct blkid_idinfo cramfs_idinfo =
{
.name = "cramfs",
.usage = BLKID_USAGE_FILESYSTEM,
.probefunc = probe_cramfs,
.magics =
{
{ .magic = "\x45\x3d\xcd\x28", .len = 4 },
{ .magic = "\x28\xcd\x3d\x45", .len = 4 },
{ NULL }
}
};
|