64664a
From 8646f60d1b421b48b43405fa286548a0bf3f5da4 Mon Sep 17 00:00:00 2001
64664a
From: Michal Humpula <michal.humpula@hudrydum.cz>
64664a
Date: Sat, 28 Feb 2015 21:19:42 +0100
64664a
Subject: [PATCH 109/116] zfs: make less syscalls
64664a
64664a
Upstream: https://github.com/karelzak/util-linux/commit/e44a4c7ac9522c03b76d8b62ce88b443771fdb0b
64664a
Addresses: https://bugzilla.redhat.com/show_bug.cgi?id=1392661
64664a
Signed-off-by: Karel Zak <kzak@redhat.com>
64664a
---
64664a
 libblkid/src/superblocks/zfs.c | 95 ++++++++++++++++++++++++++----------------
64664a
 1 file changed, 58 insertions(+), 37 deletions(-)
64664a
64664a
diff --git a/libblkid/src/superblocks/zfs.c b/libblkid/src/superblocks/zfs.c
64664a
index 4a64a03..9b5601e 100644
64664a
--- a/libblkid/src/superblocks/zfs.c
64664a
+++ b/libblkid/src/superblocks/zfs.c
64664a
@@ -20,6 +20,7 @@
64664a
 #define VDEV_LABEL_NVPAIR	( 16 * 1024ULL)
64664a
 #define VDEV_LABEL_SIZE		(256 * 1024ULL)
64664a
 #define UBERBLOCK_SIZE		1024ULL
64664a
+#define UBERBLOCKS_COUNT   128
64664a
 
64664a
 /* #include <sys/uberblock_impl.h> */
64664a
 #define UBERBLOCK_MAGIC         0x00bab10c              /* oo-ba-bloc!  */
64664a
@@ -32,7 +33,6 @@ struct zfs_uberblock {
64664a
 	char		ub_rootbp;	/* MOS objset_phys_t		*/
64664a
 } __attribute__((packed));
64664a
 
64664a
-#define ZFS_TRIES	512
64664a
 #define ZFS_WANT	 4
64664a
 
64664a
 #define DATA_TYPE_UINT64 8
64664a
@@ -163,61 +163,82 @@ static void zfs_extract_guid_name(blkid_probe pr, loff_t offset)
64664a
 #define zdebug(fmt, ...)	do {} while(0)
64664a
 /*#define zdebug(fmt, a...)	fprintf(stderr, fmt, ##a)*/
64664a
 
64664a
-/* ZFS has 128x1kB host-endian root blocks, stored in 2 areas (labels)
64664a
- * at the start of the disk, and 2 areas at the end of the disk.
64664a
- */
64664a
+static int find_uberblocks(const void *label, loff_t *ub_offset, int *swap_endian)
64664a
+{
64664a
+	uint64_t swab_magic = swab64(UBERBLOCK_MAGIC);
64664a
+	struct zfs_uberblock *ub;
64664a
+	int i, found = 0;
64664a
+	loff_t offset = VDEV_LABEL_UBERBLOCK;
64664a
+
64664a
+	for (i = 0; i < UBERBLOCKS_COUNT; i++, offset += UBERBLOCK_SIZE) {
64664a
+		ub = (struct zfs_uberblock *)(label + offset);
64664a
+
64664a
+		if (ub->ub_magic == UBERBLOCK_MAGIC) {
64664a
+			*ub_offset = offset;
64664a
+			*swap_endian = 0;
64664a
+			found++;
64664a
+			zdebug("probe_zfs: found little-endian uberblock at %llu\n", offset >> 10);
64664a
+		}
64664a
+
64664a
+		if (ub->ub_magic == swab_magic) {
64664a
+			*ub_offset = offset;
64664a
+			*swap_endian = 1;
64664a
+			found++;
64664a
+			zdebug("probe_zfs: found big-endian uberblock at %llu\n", offset >> 10);
64664a
+		}
64664a
+  }
64664a
+
64664a
+  return found;
64664a
+}
64664a
+
64664a
+/* ZFS has 128x1kB host-endian root blocks, stored in 2 areas at the start
64664a
+ * of the disk, and 2 areas at the end of the disk.  Check only some of them...
64664a
+ * #4 (@ 132kB) is the first one written on a new filesystem. */
64664a
 static int probe_zfs(blkid_probe pr, const struct blkid_idmag *mag)
64664a
 {
64664a
 	uint64_t swab_magic = swab64(UBERBLOCK_MAGIC);
64664a
+	int swab_endian = 0;
64664a
 	struct zfs_uberblock *ub;
64664a
-	int swab_endian;
64664a
 	loff_t offset, ub_offset = 0;
64664a
-	int tried;
64664a
-	int found;
64664a
+	int label_no, found = 0, found_in_label;
64664a
+	void *label;
64664a
 	loff_t blk_align = (pr->size % (256 * 1024ULL));
64664a
 
64664a
 	zdebug("probe_zfs\n");
64664a
-	/* Look for at least 4 uberblocks to ensure a positive match.
64664a
-	   Begin with Label 0 (L0) at the start of the block device. */
64664a
-	for (tried = found = 0, offset = VDEV_LABEL_UBERBLOCK;
64664a
-	     found < ZFS_WANT && tried < ZFS_TRIES;
64664a
-	     tried++, offset += UBERBLOCK_SIZE)
64664a
-	{
64664a
-		/* Leave L0 to try other labels */
64664a
-		switch(tried) {
64664a
-		case 128: // jump to L1, just after L0
64664a
-			offset = VDEV_LABEL_SIZE + VDEV_LABEL_UBERBLOCK;
64664a
+	/* Look for at least 4 uberblocks to ensure a positive match */
64664a
+	for (label_no = 0; label_no < 4; label_no++) {
64664a
+		switch(label_no) {
64664a
+		case 0: // jump to L0
64664a
+			offset = 0;
64664a
 			break;
64664a
-		case 256: // jump to L2 near the far end of the block device
64664a
-			offset = pr->size - 2 * VDEV_LABEL_SIZE + VDEV_LABEL_UBERBLOCK - blk_align;
64664a
-			zdebug("probe_zfs: l2 offset %llu\n", offset >> 10);
64664a
+		case 1: // jump to L1
64664a
+			offset = VDEV_LABEL_SIZE;
64664a
 			break;
64664a
-		case 384: // jump to L3 at the furthest end of the block device
64664a
-			offset = pr->size - VDEV_LABEL_SIZE + VDEV_LABEL_UBERBLOCK - blk_align;
64664a
-			zdebug("probe_zfs: l3 offset %llu\n", offset >> 10);
64664a
+		case 2: // jump to L2
64664a
+			offset = pr->size - 2 * VDEV_LABEL_SIZE - blk_align;
64664a
+			break;
64664a
+		case 3: // jump to L3
64664a
+			offset = pr->size - VDEV_LABEL_SIZE - blk_align;
64664a
 			break;
64664a
 		}
64664a
 
64664a
-		ub = (struct zfs_uberblock *)
64664a
-			blkid_probe_get_buffer(pr, offset,
64664a
-					       sizeof(struct zfs_uberblock));
64664a
-		if (ub == NULL)
64664a
+		label = blkid_probe_get_buffer(pr, offset, VDEV_LABEL_SIZE);
64664a
+		if (label == NULL)
64664a
 			return errno ? -errno : 1;
64664a
 
64664a
-		if (ub->ub_magic == UBERBLOCK_MAGIC) {
64664a
-			ub_offset = offset;
64664a
-			found++;
64664a
-			zdebug("probe_zfs: found little-endian uberblock at %llu\n", offset >> 10);
64664a
-		}
64664a
+		found_in_label = find_uberblocks(label, &ub_offset, &swab_endian);
64664a
 
64664a
-		if ((swab_endian = (ub->ub_magic == swab_magic))) {
64664a
-			ub_offset = offset;
64664a
-			found++;
64664a
-			zdebug("probe_zfs: found big-endian uberblock at %llu\n", offset >> 10);
64664a
+		if (found_in_label > 0) {
64664a
+			found+= found_in_label;
64664a
+			ub = (struct zfs_uberblock *)(label + ub_offset);
64664a
+			ub_offset += offset;
64664a
+
64664a
+			if (found >= ZFS_WANT)
64664a
+				break;
64664a
 		}
64664a
 	}
64664a
 
64664a
-	if (found < 4)
64664a
+	if (found < ZFS_WANT)
64664a
 		return 1;
64664a
 
64664a
 	/* If we found the 4th uberblock, then we will have exited from the
64664a
-- 
64664a
2.9.3
64664a