shithub: lwext4

Download patch

ref: d526904e82411b27f878e245b74a0411e3a8ea32
parent: 92053fed4a3c6e17c0c1c7596353c8c5465bdf1f
author: ngkaho1234 <[email protected]>
date: Sun Sep 27 09:09:11 EDT 2015

FIX: an extra block is freed when truncating an inode.

--- a/lwext4/ext4_fs.c
+++ b/lwext4/ext4_fs.c
@@ -908,15 +908,12 @@
 	}
 
 	/* Compute how many blocks will be released */
-	uint64_t size_diff = old_size - new_size;
 	uint32_t block_size = ext4_sb_get_block_size(sb);
-	uint32_t diff_blocks_count = size_diff / block_size;
-	if (size_diff % block_size != 0)
-		diff_blocks_count++;
-
-	uint32_t old_blocks_count = old_size / block_size;
-	if (old_size % block_size != 0)
-		old_blocks_count++;
+	uint32_t new_blocks_count = (new_size + block_size - 1) /
+				    block_size;
+	uint32_t old_blocks_count = (old_size + block_size - 1) /
+				    block_size;
+	uint32_t diff_blocks_count = old_blocks_count - new_blocks_count;
 #if CONFIG_EXTENT_ENABLE
 	if ((ext4_sb_has_feature_incompatible(sb,
 					      EXT4_FEATURE_INCOMPAT_EXTENTS)) &&
@@ -923,10 +920,14 @@
 	    (ext4_inode_has_flag(inode_ref->inode, EXT4_INODE_FLAG_EXTENTS))) {
 
 		/* Extents require special operation */
-		int rc = ext4_extent_release_blocks_from(
-		    inode_ref, old_blocks_count - diff_blocks_count);
-		if (rc != EOK)
-			return rc;
+		if (diff_blocks_count) {
+			int rc = ext4_extent_release_blocks_from(
+					inode_ref,
+					new_blocks_count);
+			if (rc != EOK)
+				return rc;
+
+		}
 	} else
 #endif
 	{
@@ -934,9 +935,9 @@
 
 		/* Starting from 1 because of logical blocks are numbered from 0
 		 */
-		for (i = 1; i <= diff_blocks_count; ++i) {
+		for (i = 0; i < diff_blocks_count; ++i) {
 			int rc = ext4_fs_release_inode_block(
-			    inode_ref, old_blocks_count - i);
+			    inode_ref, new_blocks_count + i);
 			if (rc != EOK)
 				return rc;
 		}