Blame SOURCES/e2fsprogs-1.45.6-tune2fs-update-dir-checksums-when-clearing-dir_index.patch

f239de
From fb4a6ed596f6a9f6b1c6d3d9307ef8259d988c04 Mon Sep 17 00:00:00 2001
f239de
From: Jan Kara <jack@suse.cz>
f239de
Date: Thu, 13 Feb 2020 11:16:02 +0100
f239de
Subject: [PATCH 06/46] tune2fs: update dir checksums when clearing dir_index
f239de
 feature
f239de
Content-Type: text/plain
f239de
f239de
When clearing dir_index feature while metadata_csum is enabled, we have
f239de
to rewrite checksums of all indexed directories to update checksums of
f239de
internal tree nodes.
f239de
f239de
Signed-off-by: Jan Kara <jack@suse.cz>
f239de
Signed-off-by: Theodore Ts'o <tytso@mit.edu>
f239de
Signed-off-by: Lukas Czerner <lczerner@redhat.com>
f239de
---
f239de
 misc/tune2fs.c | 143 ++++++++++++++++++++++++++++++++-----------------
f239de
 1 file changed, 95 insertions(+), 48 deletions(-)
f239de
f239de
diff --git a/misc/tune2fs.c b/misc/tune2fs.c
f239de
index 39cf8587..a7a779b8 100644
f239de
--- a/misc/tune2fs.c
f239de
+++ b/misc/tune2fs.c
f239de
@@ -504,7 +504,8 @@ struct rewrite_dir_context {
f239de
 	char *buf;
f239de
 	errcode_t errcode;
f239de
 	ext2_ino_t dir;
f239de
-	int is_htree;
f239de
+	int is_htree:1;
f239de
+	int clear_htree:1;
f239de
 };
f239de
 
f239de
 static int rewrite_dir_block(ext2_filsys fs,
f239de
@@ -523,8 +524,13 @@ static int rewrite_dir_block(ext2_filsys fs,
f239de
 	if (ctx->errcode)
f239de
 		return BLOCK_ABORT;
f239de
 
f239de
-	/* if htree node... */
f239de
-	if (ctx->is_htree)
f239de
+	/*
f239de
+	 * if htree node... Note that if we are clearing htree structures from
f239de
+	 * the directory, we treat the htree internal block as an ordinary leaf.
f239de
+	 * The code below will do the right thing and make space for checksum
f239de
+	 * there.
f239de
+	 */
f239de
+	if (ctx->is_htree && !ctx->clear_htree)
f239de
 		ext2fs_get_dx_countlimit(fs, (struct ext2_dir_entry *)ctx->buf,
f239de
 					 &dcl, &dcl_offset);
f239de
 	if (dcl) {
f239de
@@ -653,7 +659,8 @@ static errcode_t rewrite_directory(ext2_filsys fs, ext2_ino_t dir,
f239de
 	if (retval)
f239de
 		return retval;
f239de
 
f239de
-	ctx.is_htree = (inode->i_flags & EXT2_INDEX_FL);
f239de
+	ctx.is_htree = !!(inode->i_flags & EXT2_INDEX_FL);
f239de
+	ctx.clear_htree = !ext2fs_has_feature_dir_index(fs->super);
f239de
 	ctx.dir = dir;
f239de
 	ctx.errcode = 0;
f239de
 	retval = ext2fs_block_iterate3(fs, dir, BLOCK_FLAG_READ_ONLY |
f239de
@@ -664,6 +671,13 @@ static errcode_t rewrite_directory(ext2_filsys fs, ext2_ino_t dir,
f239de
 	if (retval)
f239de
 		return retval;
f239de
 
f239de
+	if (ctx.is_htree && ctx.clear_htree) {
f239de
+		inode->i_flags &= ~EXT2_INDEX_FL;
f239de
+		retval = ext2fs_write_inode(fs, dir, inode);
f239de
+		if (retval)
f239de
+			return retval;
f239de
+	}
f239de
+
f239de
 	return ctx.errcode;
f239de
 }
f239de
 
f239de
@@ -818,28 +832,67 @@ static void rewrite_one_inode(struct rewrite_context *ctx, ext2_ino_t ino,
f239de
 		fatal_err(retval, "while rewriting extended attribute");
f239de
 }
f239de
 
f239de
-/*
f239de
- * Forcibly set checksums in all inodes.
f239de
- */
f239de
-static void rewrite_inodes(ext2_filsys fs)
f239de
+#define REWRITE_EA_FL		0x01	/* Rewrite EA inodes */
f239de
+#define REWRITE_DIR_FL		0x02	/* Rewrite directories */
f239de
+#define REWRITE_NONDIR_FL	0x04	/* Rewrite other inodes */
f239de
+#define REWRITE_ALL (REWRITE_EA_FL | REWRITE_DIR_FL | REWRITE_NONDIR_FL)
f239de
+
f239de
+static void rewrite_inodes_pass(struct rewrite_context *ctx, unsigned int flags)
f239de
 {
f239de
 	ext2_inode_scan	scan;
f239de
 	errcode_t	retval;
f239de
 	ext2_ino_t	ino;
f239de
 	struct ext2_inode *inode;
f239de
-	int pass;
f239de
+	int rewrite;
f239de
+
f239de
+	retval = ext2fs_get_mem(ctx->inode_size, &inode;;
f239de
+	if (retval)
f239de
+		fatal_err(retval, "while allocating memory");
f239de
+
f239de
+	retval = ext2fs_open_inode_scan(ctx->fs, 0, &scan;;
f239de
+	if (retval)
f239de
+		fatal_err(retval, "while opening inode scan");
f239de
+
f239de
+	do {
f239de
+		retval = ext2fs_get_next_inode_full(scan, &ino, inode,
f239de
+						    ctx->inode_size);
f239de
+		if (retval)
f239de
+			fatal_err(retval, "while getting next inode");
f239de
+		if (!ino)
f239de
+			break;
f239de
+
f239de
+		rewrite = 0;
f239de
+		if (inode->i_flags & EXT4_EA_INODE_FL) {
f239de
+			if (flags & REWRITE_EA_FL)
f239de
+				rewrite = 1;
f239de
+		} else if (LINUX_S_ISDIR(inode->i_mode)) {
f239de
+			if (flags & REWRITE_DIR_FL)
f239de
+				rewrite = 1;
f239de
+		} else {
f239de
+			if (flags & REWRITE_NONDIR_FL)
f239de
+				rewrite = 1;
f239de
+		}
f239de
+		if (rewrite)
f239de
+			rewrite_one_inode(ctx, ino, inode);
f239de
+	} while (ino);
f239de
+	ext2fs_close_inode_scan(scan);
f239de
+	ext2fs_free_mem(&inode;;
f239de
+}
f239de
+
f239de
+/*
f239de
+ * Forcibly rewrite checksums in inodes specified by 'flags'
f239de
+ */
f239de
+static void rewrite_inodes(ext2_filsys fs, unsigned int flags)
f239de
+{
f239de
 	struct rewrite_context ctx = {
f239de
 		.fs = fs,
f239de
 		.inode_size = EXT2_INODE_SIZE(fs->super),
f239de
 	};
f239de
+	errcode_t retval;
f239de
 
f239de
 	if (fs->super->s_creator_os == EXT2_OS_HURD)
f239de
 		return;
f239de
 
f239de
-	retval = ext2fs_get_mem(ctx.inode_size, &inode;;
f239de
-	if (retval)
f239de
-		fatal_err(retval, "while allocating memory");
f239de
-
f239de
 	retval = ext2fs_get_memzero(ctx.inode_size, &ctx.zero_inode);
f239de
 	if (retval)
f239de
 		fatal_err(retval, "while allocating memory");
f239de
@@ -858,39 +911,16 @@ static void rewrite_inodes(ext2_filsys fs)
f239de
 	 *
f239de
 	 * pass 2: go over other inodes to update their checksums.
f239de
 	 */
f239de
-	if (ext2fs_has_feature_ea_inode(fs->super))
f239de
-		pass = 1;
f239de
-	else
f239de
-		pass = 2;
f239de
-	for (;pass <= 2; pass++) {
f239de
-		retval = ext2fs_open_inode_scan(fs, 0, &scan;;
f239de
-		if (retval)
f239de
-			fatal_err(retval, "while opening inode scan");
f239de
-
f239de
-		do {
f239de
-			retval = ext2fs_get_next_inode_full(scan, &ino, inode,
f239de
-							    ctx.inode_size);
f239de
-			if (retval)
f239de
-				fatal_err(retval, "while getting next inode");
f239de
-			if (!ino)
f239de
-				break;
f239de
-
f239de
-			if (((pass == 1) &&
f239de
-			     (inode->i_flags & EXT4_EA_INODE_FL)) ||
f239de
-			    ((pass == 2) &&
f239de
-			     !(inode->i_flags & EXT4_EA_INODE_FL)))
f239de
-				rewrite_one_inode(&ctx, ino, inode);
f239de
-		} while (ino);
f239de
-
f239de
-		ext2fs_close_inode_scan(scan);
f239de
-	}
f239de
+	if (ext2fs_has_feature_ea_inode(fs->super) && (flags & REWRITE_EA_FL))
f239de
+		rewrite_inodes_pass(&ctx, REWRITE_EA_FL);
f239de
+	flags &= ~REWRITE_EA_FL;
f239de
+	rewrite_inodes_pass(&ctx, flags);
f239de
 
f239de
 	ext2fs_free_mem(&ctx.zero_inode);
f239de
 	ext2fs_free_mem(&ctx.ea_buf);
f239de
-	ext2fs_free_mem(&inode;;
f239de
 }
f239de
 
f239de
-static void rewrite_metadata_checksums(ext2_filsys fs)
f239de
+static void rewrite_metadata_checksums(ext2_filsys fs, unsigned int flags)
f239de
 {
f239de
 	errcode_t retval;
f239de
 	dgrp_t i;
f239de
@@ -902,7 +932,7 @@ static void rewrite_metadata_checksums(ext2_filsys fs)
f239de
 	retval = ext2fs_read_bitmaps(fs);
f239de
 	if (retval)
f239de
 		fatal_err(retval, "while reading bitmaps");
f239de
-	rewrite_inodes(fs);
f239de
+	rewrite_inodes(fs, flags);
f239de
 	ext2fs_mark_ib_dirty(fs);
f239de
 	ext2fs_mark_bb_dirty(fs);
f239de
 	ext2fs_mmp_update2(fs, 1);
f239de
@@ -1201,6 +1231,23 @@ mmp_error:
f239de
 			uuid_generate((unsigned char *) sb->s_hash_seed);
f239de
 	}
f239de
 
f239de
+	if (FEATURE_OFF(E2P_FEATURE_COMPAT, EXT2_FEATURE_COMPAT_DIR_INDEX) &&
f239de
+	    ext2fs_has_feature_metadata_csum(sb)) {
f239de
+		check_fsck_needed(fs,
f239de
+			_("Disabling directory index on filesystem with "
f239de
+			  "checksums could take some time."));
f239de
+		if (mount_flags & EXT2_MF_MOUNTED) {
f239de
+			fputs(_("Cannot disable dir_index on a mounted "
f239de
+				"filesystem!\n"), stderr);
f239de
+			exit(1);
f239de
+		}
f239de
+		/*
f239de
+		 * Clearing dir_index on checksummed filesystem requires
f239de
+		 * rewriting all directories to update checksums.
f239de
+		 */
f239de
+		rewrite_checksums |= REWRITE_DIR_FL;
f239de
+	}
f239de
+
f239de
 	if (FEATURE_OFF(E2P_FEATURE_INCOMPAT, EXT4_FEATURE_INCOMPAT_FLEX_BG)) {
f239de
 		if (ext2fs_check_desc(fs)) {
f239de
 			fputs(_("Clearing the flex_bg flag would "
f239de
@@ -1244,7 +1291,7 @@ mmp_error:
f239de
 				 "The larger fields afforded by this feature "
f239de
 				 "enable full-strength checksumming.  "
f239de
 				 "Run resize2fs -b to rectify.\n"));
f239de
-		rewrite_checksums = 1;
f239de
+		rewrite_checksums = REWRITE_ALL;
f239de
 		/* metadata_csum supersedes uninit_bg */
f239de
 		ext2fs_clear_feature_gdt_csum(fs->super);
f239de
 
f239de
@@ -1272,7 +1319,7 @@ mmp_error:
f239de
 				"filesystem!\n"), stderr);
f239de
 			exit(1);
f239de
 		}
f239de
-		rewrite_checksums = 1;
f239de
+		rewrite_checksums = REWRITE_ALL;
f239de
 
f239de
 		/* Enable uninit_bg unless the user expressly turned it off */
f239de
 		memcpy(test_features, old_features, sizeof(test_features));
f239de
@@ -1454,7 +1501,7 @@ mmp_error:
f239de
 			}
f239de
 			check_fsck_needed(fs, _("Recalculating checksums "
f239de
 						"could take some time."));
f239de
-			rewrite_checksums = 1;
f239de
+			rewrite_checksums = REWRITE_ALL;
f239de
 		}
f239de
 	}
f239de
 
f239de
@@ -3191,7 +3238,7 @@ _("Warning: The journal is dirty. You may wish to replay the journal like:\n\n"
f239de
 			check_fsck_needed(fs,
f239de
 				_("Setting the UUID on this "
f239de
 				  "filesystem could take some time."));
f239de
-			rewrite_checksums = 1;
f239de
+			rewrite_checksums = REWRITE_ALL;
f239de
 		}
f239de
 
f239de
 		if (ext2fs_has_group_desc_csum(fs)) {
f239de
@@ -3302,7 +3349,7 @@ _("Warning: The journal is dirty. You may wish to replay the journal like:\n\n"
f239de
 		if (retval == 0) {
f239de
 			printf(_("Setting inode size %lu\n"),
f239de
 							new_inode_size);
f239de
-			rewrite_checksums = 1;
f239de
+			rewrite_checksums = REWRITE_ALL;
f239de
 		} else {
f239de
 			printf("%s", _("Failed to change inode size\n"));
f239de
 			rc = 1;
f239de
@@ -3311,7 +3358,7 @@ _("Warning: The journal is dirty. You may wish to replay the journal like:\n\n"
f239de
 	}
f239de
 
f239de
 	if (rewrite_checksums)
f239de
-		rewrite_metadata_checksums(fs);
f239de
+		rewrite_metadata_checksums(fs, rewrite_checksums);
f239de
 
f239de
 	if (l_flag)
f239de
 		list_super(sb);
f239de
-- 
f239de
2.35.1
f239de