summaryrefslogtreecommitdiff
path: root/util-linux/volume_id/erofs.c
diff options
context:
space:
mode:
Diffstat (limited to 'util-linux/volume_id/erofs.c')
-rw-r--r--util-linux/volume_id/erofs.c67
1 files changed, 67 insertions, 0 deletions
diff --git a/util-linux/volume_id/erofs.c b/util-linux/volume_id/erofs.c
new file mode 100644
index 0000000..a163a35
--- /dev/null
+++ b/util-linux/volume_id/erofs.c
@@ -0,0 +1,67 @@
+/*
+ * volume_id - reads filesystem label and uuid
+ *
+ * Copyright (C) 2020 Norbert Lange <nolange79@gmail.com>
+ *
+ * Licensed under GPLv2, see file LICENSE in this source tree.
+ */
+//config:config FEATURE_VOLUMEID_EROFS
+//config: bool "erofs filesystem"
+//config: default y
+//config: depends on VOLUMEID
+//config: help
+//config: Erofs is a compressed readonly filesystem for Linux.
+
+//kbuild:lib-$(CONFIG_FEATURE_VOLUMEID_EROFS) += erofs.o
+
+#include "volume_id_internal.h"
+
+#define EROFS_SUPER_MAGIC_V1 0xE0F5E1E2
+#define EROFS_SUPER_OFFSET 1024
+#define EROFS_FEATURE_COMPAT_SB_CHKSUM 0x00000001
+
+/* 128-byte erofs on-disk super block */
+struct erofs_super_block {
+ uint32_t magic; /* file system magic number */
+ uint32_t checksum; /* crc32c(super_block) */
+ uint32_t feature_compat;
+ uint8_t blkszbits; /* support block_size == PAGE_SIZE only */
+ uint8_t reserved;
+
+ uint16_t root_nid; /* nid of root directory */
+ uint64_t inos; /* total valid ino # (== f_files - f_favail) */
+
+ uint64_t build_time; /* inode v1 time derivation */
+ uint32_t build_time_nsec; /* inode v1 time derivation in nano scale */
+ uint32_t blocks; /* used for statfs */
+ uint32_t meta_blkaddr; /* start block address of metadata area */
+ uint32_t xattr_blkaddr; /* start block address of shared xattr area */
+ uint8_t uuid[16]; /* 128-bit uuid for volume */
+ uint8_t volume_name[16]; /* volume name */
+ uint32_t feature_incompat;
+ uint8_t reserved2[44];
+} PACKED;
+
+int FAST_FUNC volume_id_probe_erofs(struct volume_id *id /*,uint64_t off*/)
+{
+ struct erofs_super_block *sb;
+
+ BUILD_BUG_ON(sizeof(struct erofs_super_block) != 128);
+
+ dbg("erofs: probing at offset 0x%llx", EROFS_SUPER_OFFSET);
+ sb = volume_id_get_buffer(id, EROFS_SUPER_OFFSET, sizeof(*sb));
+ if (!sb)
+ return -1;
+
+ if (sb->magic != cpu_to_le32(EROFS_SUPER_MAGIC_V1))
+ return -1;
+
+ IF_FEATURE_BLKID_TYPE(id->type = "erofs");
+
+ volume_id_set_label_string(id, sb->volume_name,
+ MIN(sizeof(sb->volume_name), VOLUME_ID_LABEL_SIZE));
+
+ volume_id_set_uuid(id, sb->uuid, UUID_DCE);
+
+ return 0;
+}