summaryrefslogtreecommitdiffstats
path: root/libblkid/src/superblocks/erofs.c
diff options
context:
space:
mode:
Diffstat (limited to 'libblkid/src/superblocks/erofs.c')
-rw-r--r--libblkid/src/superblocks/erofs.c109
1 files changed, 109 insertions, 0 deletions
diff --git a/libblkid/src/superblocks/erofs.c b/libblkid/src/superblocks/erofs.c
new file mode 100644
index 0000000..14d272e
--- /dev/null
+++ b/libblkid/src/superblocks/erofs.c
@@ -0,0 +1,109 @@
+/*
+ * Copyright (C) 2020 Gao Xiang
+ *
+ * This file may be redistributed under the terms of the
+ * GNU Lesser General Public License
+ *
+ * https://docs.kernel.org/filesystems/erofs.html
+ */
+#include <stddef.h>
+#include <string.h>
+
+#include "superblocks.h"
+#include "crc32c.h"
+
+#define EROFS_SUPER_OFFSET 1024
+#define EROFS_SB_KBOFF (EROFS_SUPER_OFFSET >> 10)
+#define EROFS_FEATURE_SB_CSUM (1 << 0)
+
+#define EROFS_SUPER_MAGIC_V1 "\xe2\xe1\xf5\xe0"
+#define EROFS_MAGIC_OFF 0
+
+/* All in little-endian */
+struct erofs_super_block {
+ uint32_t magic;
+ uint32_t checksum;
+ uint32_t feature_compat;
+ uint8_t blkszbits;
+ uint8_t reserved;
+
+ uint16_t root_nid;
+ uint64_t inos;
+
+ uint64_t build_time;
+ uint32_t build_time_nsec;
+ uint32_t blocks;
+ uint32_t meta_blkaddr;
+ uint32_t xattr_blkaddr;
+ uint8_t uuid[16];
+ uint8_t volume_name[16];
+ uint32_t feature_incompat;
+ uint8_t reserved2[44];
+} __attribute__((packed));
+
+static int erofs_verify_checksum(blkid_probe pr, const struct blkid_idmag *mag,
+ const struct erofs_super_block *sb)
+{
+ uint32_t expected, csum;
+ size_t csummed_size;
+ unsigned char *csummed;
+
+ if (!(le32_to_cpu(sb->feature_compat) & EROFS_FEATURE_SB_CSUM))
+ return 1;
+
+ expected = le32_to_cpu(sb->checksum);
+ csummed_size = (1U << sb->blkszbits) - EROFS_SUPER_OFFSET;
+
+ csummed = blkid_probe_get_sb_buffer(pr, mag, csummed_size);
+ if (!csummed)
+ return 0;
+
+ csum = ul_crc32c_exclude_offset(~0L, csummed, csummed_size,
+ offsetof(struct erofs_super_block, checksum),
+ sizeof_member(struct erofs_super_block, checksum));
+
+ return blkid_probe_verify_csum(pr, csum, expected);
+}
+
+static int probe_erofs(blkid_probe pr, const struct blkid_idmag *mag)
+{
+ struct erofs_super_block *sb;
+
+ sb = blkid_probe_get_sb(pr, mag, struct erofs_super_block);
+ if (!sb)
+ return errno ? -errno : BLKID_PROBE_NONE;
+
+ /* EROFS is restricted to 4KiB block size */
+ if (sb->blkszbits > 31 || (1U << sb->blkszbits) > 4096)
+ return BLKID_PROBE_NONE;
+
+ if (!erofs_verify_checksum(pr, mag, sb))
+ return BLKID_PROBE_NONE;
+
+ if (sb->volume_name[0])
+ blkid_probe_set_label(pr, (unsigned char *)sb->volume_name,
+ sizeof(sb->volume_name));
+
+ blkid_probe_set_uuid(pr, sb->uuid);
+ blkid_probe_set_fsblocksize(pr, 1U << sb->blkszbits);
+ blkid_probe_set_block_size(pr, 1U << sb->blkszbits);
+ blkid_probe_set_fssize(pr, (uint64_t) (1U << sb->blkszbits) * le32_to_cpu(sb->blocks));
+
+ return BLKID_PROBE_OK;
+}
+
+const struct blkid_idinfo erofs_idinfo =
+{
+ .name = "erofs",
+ .usage = BLKID_USAGE_FILESYSTEM,
+ .probefunc = probe_erofs,
+ .magics =
+ {
+ {
+ .magic = EROFS_SUPER_MAGIC_V1,
+ .len = 4,
+ .kboff = EROFS_SB_KBOFF,
+ .sboff = EROFS_MAGIC_OFF,
+ }, { NULL }
+ }
+};