[PATCH RESEND] imsm: fix assemble with ppl during rebuild

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



When assembling an array undergoing rebuild the kernel will switch to
resync if there are no ppl entries to recover. Prevent that by adding an
empty entry when validating the ppl header.

Signed-off-by: Artur Paszkiewicz <artur.paszkiewicz@xxxxxxxxx>
---
 super-intel.c | 99 ++++++++++++++++++++++++++++++++++++++++-------------------
 1 file changed, 68 insertions(+), 31 deletions(-)

diff --git a/super-intel.c b/super-intel.c
index a429940d..fb1b6936 100644
--- a/super-intel.c
+++ b/super-intel.c
@@ -6076,6 +6076,30 @@ static int mgmt_disk(struct supertype *st)
 
 __u32 crc32c_le(__u32 crc, unsigned char const *p, size_t len);
 
+static int write_ppl_header(unsigned long long ppl_sector, int fd, void *buf)
+{
+	struct ppl_header *ppl_hdr = buf;
+	int ret;
+
+	ppl_hdr->checksum = __cpu_to_le32(~crc32c_le(~0, buf, PPL_HEADER_SIZE));
+
+	if (lseek64(fd, ppl_sector * 512, SEEK_SET) < 0) {
+		ret = -errno;
+		perror("Failed to seek to PPL header location");
+		return ret;
+	}
+
+	if (write(fd, buf, PPL_HEADER_SIZE) != PPL_HEADER_SIZE) {
+		ret = -errno;
+		perror("Write PPL header failed");
+		return ret;
+	}
+
+	fsync(fd);
+
+	return 0;
+}
+
 static int write_init_ppl_imsm(struct supertype *st, struct mdinfo *info, int fd)
 {
 	struct intel_super *super = st->sb;
@@ -6091,7 +6115,7 @@ static int write_init_ppl_imsm(struct supertype *st, struct mdinfo *info, int fd
 	ret = posix_memalign(&buf, MAX_SECTOR_SIZE, PPL_HEADER_SIZE);
 	if (ret) {
 		pr_err("Failed to allocate PPL header buffer\n");
-		return ret;
+		return -ret;
 	}
 
 	memset(buf, 0, PPL_HEADER_SIZE);
@@ -6108,33 +6132,22 @@ static int write_init_ppl_imsm(struct supertype *st, struct mdinfo *info, int fd
 		ppl_hdr->entries[0].checksum = ~0;
 	}
 
-	ppl_hdr->checksum = __cpu_to_le32(~crc32c_le(~0, buf, PPL_HEADER_SIZE));
-
-	if (lseek64(fd, info->ppl_sector * 512, SEEK_SET) < 0) {
-		ret = errno;
-		perror("Failed to seek to PPL header location");
-	}
-
-	if (!ret && write(fd, buf, PPL_HEADER_SIZE) != PPL_HEADER_SIZE) {
-		ret = errno;
-		perror("Write PPL header failed");
-	}
-
-	if (!ret)
-		fsync(fd);
+	ret = write_ppl_header(info->ppl_sector, fd, buf);
 
 	free(buf);
 	return ret;
 }
 
+static int is_rebuilding(struct imsm_dev *dev);
+
 static int validate_ppl_imsm(struct supertype *st, struct mdinfo *info,
 			     struct mdinfo *disk)
 {
 	struct intel_super *super = st->sb;
 	struct dl *d;
-	void *buf;
+	void *buf_orig, *buf, *buf_prev = NULL;
 	int ret = 0;
-	struct ppl_header *ppl_hdr;
+	struct ppl_header *ppl_hdr = NULL;
 	__u32 crc;
 	struct imsm_dev *dev;
 	__u32 idx;
@@ -6145,33 +6158,36 @@ static int validate_ppl_imsm(struct supertype *st, struct mdinfo *info,
 	if (disk->disk.raid_disk < 0)
 		return 0;
 
-	if (posix_memalign(&buf, MAX_SECTOR_SIZE, PPL_HEADER_SIZE)) {
-		pr_err("Failed to allocate PPL header buffer\n");
-		return -1;
-	}
-
 	dev = get_imsm_dev(super, info->container_member);
 	idx = get_imsm_disk_idx(dev, disk->disk.raid_disk, MAP_0);
 	d = get_imsm_dl_disk(super, idx);
 
 	if (!d || d->index < 0 || is_failed(&d->disk))
-		goto out;
+		return 0;
+
+	if (posix_memalign(&buf_orig, MAX_SECTOR_SIZE, PPL_HEADER_SIZE * 2)) {
+		pr_err("Failed to allocate PPL header buffer\n");
+		return -1;
+	}
+	buf = buf_orig;
 
 	ret = 1;
 	while (ppl_offset < MULTIPLE_PPL_AREA_SIZE_IMSM) {
+		void *tmp;
+
 		dprintf("Checking potential PPL at offset: %llu\n", ppl_offset);
 
 		if (lseek64(d->fd, info->ppl_sector * 512 + ppl_offset,
 			    SEEK_SET) < 0) {
 			perror("Failed to seek to PPL header location");
 			ret = -1;
-			goto out;
+			break;
 		}
 
 		if (read(d->fd, buf, PPL_HEADER_SIZE) != PPL_HEADER_SIZE) {
 			perror("Read PPL header failed");
 			ret = -1;
-			goto out;
+			break;
 		}
 
 		ppl_hdr = buf;
@@ -6182,12 +6198,12 @@ static int validate_ppl_imsm(struct supertype *st, struct mdinfo *info,
 		if (crc != ~crc32c_le(~0, buf, PPL_HEADER_SIZE)) {
 			dprintf("Wrong PPL header checksum on %s\n",
 				d->devname);
-			goto out;
+			break;
 		}
 
 		if (prev_gen_num > __le64_to_cpu(ppl_hdr->generation)) {
 			/* previous was newest, it was already checked */
-			goto out;
+			break;
 		}
 
 		if ((__le32_to_cpu(ppl_hdr->signature) !=
@@ -6195,7 +6211,7 @@ static int validate_ppl_imsm(struct supertype *st, struct mdinfo *info,
 			dprintf("Wrong PPL header signature on %s\n",
 				d->devname);
 			ret = 1;
-			goto out;
+			break;
 		}
 
 		ret = 0;
@@ -6205,10 +6221,18 @@ static int validate_ppl_imsm(struct supertype *st, struct mdinfo *info,
 		for (i = 0; i < __le32_to_cpu(ppl_hdr->entries_count); i++)
 			ppl_offset +=
 				   __le32_to_cpu(ppl_hdr->entries[i].pp_size);
+
+		if (!buf_prev)
+			buf_prev = buf + PPL_HEADER_SIZE;
+		tmp = buf_prev;
+		buf_prev = buf;
+		buf = tmp;
 	}
 
-out:
-	free(buf);
+	if (buf_prev) {
+		buf = buf_prev;
+		ppl_hdr = buf_prev;
+	}
 
 	/*
 	 * Update metadata to use mutliple PPLs area (1MB).
@@ -6246,14 +6270,27 @@ out:
 		if (map->map_state == IMSM_T_STATE_UNINITIALIZED ||
 		   (map->map_state == IMSM_T_STATE_NORMAL &&
 		   !(dev->vol.dirty & RAIDVOL_DIRTY)) ||
-		   (dev->vol.migr_state == MIGR_REBUILD &&
+		   (is_rebuilding(dev) &&
 		    dev->vol.curr_migr_unit == 0 &&
 		    get_imsm_disk_idx(dev, disk->disk.raid_disk, MAP_1) != idx))
 			ret = st->ss->write_init_ppl(st, info, d->fd);
 		else
 			info->mismatch_cnt++;
+	} else if (ret == 0 &&
+		   ppl_hdr->entries_count == 0 &&
+		   is_rebuilding(dev) &&
+		   info->resync_start == 0) {
+		/*
+		 * The header has no entries - add a single empty entry and
+		 * rewrite the header to prevent the kernel from going into
+		 * resync after an interrupted rebuild.
+		 */
+		ppl_hdr->entries_count = __cpu_to_le32(1);
+		ret = write_ppl_header(info->ppl_sector, d->fd, buf);
 	}
 
+	free(buf_orig);
+
 	return ret;
 }
 
-- 
2.15.1

--
To unsubscribe from this list: send the line "unsubscribe linux-raid" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [Linux RAID Wiki]     [ATA RAID]     [Linux SCSI Target Infrastructure]     [Linux Block]     [Linux IDE]     [Linux SCSI]     [Linux Hams]     [Device Mapper]     [Device Mapper Cryptographics]     [Kernel]     [Linux Admin]     [Linux Net]     [GFS]     [RPM]     [git]     [Yosemite Forum]


  Powered by Linux