Skip to content

Commit 45e4ab3

Browse files
jankaratytso
authored andcommitted
ext4: move setting of trimmed bit into ext4_try_to_trim_range()
Currently we set the group's trimmed bit in ext4_trim_all_free() based on return value of ext4_try_to_trim_range(). However when we will want to abort trimming because of suspend attempt, we want to return success from ext4_try_to_trim_range() but not set the trimmed bit. Instead implementing awkward propagation of this information, just move setting of trimmed bit into ext4_try_to_trim_range() when the whole group is trimmed. Cc: stable@kernel.org Signed-off-by: Jan Kara <jack@suse.cz> Link: https://lore.kernel.org/r/20230913150504.9054-1-jack@suse.cz Signed-off-by: Theodore Ts'o <tytso@mit.edu>
1 parent 1bb0763 commit 45e4ab3

File tree

1 file changed

+25
-21
lines changed

1 file changed

+25
-21
lines changed

fs/ext4/mballoc.c

Lines changed: 25 additions & 21 deletions
Original file line numberDiff line numberDiff line change
@@ -6906,16 +6906,29 @@ __acquires(bitlock)
69066906
return ret;
69076907
}
69086908

6909+
static ext4_grpblk_t ext4_last_grp_cluster(struct super_block *sb,
6910+
ext4_group_t grp)
6911+
{
6912+
if (grp < ext4_get_groups_count(sb))
6913+
return EXT4_CLUSTERS_PER_GROUP(sb) - 1;
6914+
return (ext4_blocks_count(EXT4_SB(sb)->s_es) -
6915+
ext4_group_first_block_no(sb, grp) - 1) >>
6916+
EXT4_CLUSTER_BITS(sb);
6917+
}
6918+
69096919
static int ext4_try_to_trim_range(struct super_block *sb,
69106920
struct ext4_buddy *e4b, ext4_grpblk_t start,
69116921
ext4_grpblk_t max, ext4_grpblk_t minblocks)
69126922
__acquires(ext4_group_lock_ptr(sb, e4b->bd_group))
69136923
__releases(ext4_group_lock_ptr(sb, e4b->bd_group))
69146924
{
69156925
ext4_grpblk_t next, count, free_count;
6926+
bool set_trimmed = false;
69166927
void *bitmap;
69176928

69186929
bitmap = e4b->bd_bitmap;
6930+
if (start == 0 && max >= ext4_last_grp_cluster(sb, e4b->bd_group))
6931+
set_trimmed = true;
69196932
start = max(e4b->bd_info->bb_first_free, start);
69206933
count = 0;
69216934
free_count = 0;
@@ -6930,16 +6943,14 @@ __releases(ext4_group_lock_ptr(sb, e4b->bd_group))
69306943
int ret = ext4_trim_extent(sb, start, next - start, e4b);
69316944

69326945
if (ret && ret != -EOPNOTSUPP)
6933-
break;
6946+
return count;
69346947
count += next - start;
69356948
}
69366949
free_count += next - start;
69376950
start = next + 1;
69386951

6939-
if (fatal_signal_pending(current)) {
6940-
count = -ERESTARTSYS;
6941-
break;
6942-
}
6952+
if (fatal_signal_pending(current))
6953+
return -ERESTARTSYS;
69436954

69446955
if (need_resched()) {
69456956
ext4_unlock_group(sb, e4b->bd_group);
@@ -6951,6 +6962,9 @@ __releases(ext4_group_lock_ptr(sb, e4b->bd_group))
69516962
break;
69526963
}
69536964

6965+
if (set_trimmed)
6966+
EXT4_MB_GRP_SET_TRIMMED(e4b->bd_info);
6967+
69546968
return count;
69556969
}
69566970

@@ -6961,7 +6975,6 @@ __releases(ext4_group_lock_ptr(sb, e4b->bd_group))
69616975
* @start: first group block to examine
69626976
* @max: last group block to examine
69636977
* @minblocks: minimum extent block count
6964-
* @set_trimmed: set the trimmed flag if at least one block is trimmed
69656978
*
69666979
* ext4_trim_all_free walks through group's block bitmap searching for free
69676980
* extents. When the free extent is found, mark it as used in group buddy
@@ -6971,7 +6984,7 @@ __releases(ext4_group_lock_ptr(sb, e4b->bd_group))
69716984
static ext4_grpblk_t
69726985
ext4_trim_all_free(struct super_block *sb, ext4_group_t group,
69736986
ext4_grpblk_t start, ext4_grpblk_t max,
6974-
ext4_grpblk_t minblocks, bool set_trimmed)
6987+
ext4_grpblk_t minblocks)
69756988
{
69766989
struct ext4_buddy e4b;
69776990
int ret;
@@ -6988,13 +7001,10 @@ ext4_trim_all_free(struct super_block *sb, ext4_group_t group,
69887001
ext4_lock_group(sb, group);
69897002

69907003
if (!EXT4_MB_GRP_WAS_TRIMMED(e4b.bd_info) ||
6991-
minblocks < EXT4_SB(sb)->s_last_trim_minblks) {
7004+
minblocks < EXT4_SB(sb)->s_last_trim_minblks)
69927005
ret = ext4_try_to_trim_range(sb, &e4b, start, max, minblocks);
6993-
if (ret >= 0 && set_trimmed)
6994-
EXT4_MB_GRP_SET_TRIMMED(e4b.bd_info);
6995-
} else {
7006+
else
69967007
ret = 0;
6997-
}
69987008

69997009
ext4_unlock_group(sb, group);
70007010
ext4_mb_unload_buddy(&e4b);
@@ -7027,7 +7037,6 @@ int ext4_trim_fs(struct super_block *sb, struct fstrim_range *range)
70277037
ext4_fsblk_t first_data_blk =
70287038
le32_to_cpu(EXT4_SB(sb)->s_es->s_first_data_block);
70297039
ext4_fsblk_t max_blks = ext4_blocks_count(EXT4_SB(sb)->s_es);
7030-
bool whole_group, eof = false;
70317040
int ret = 0;
70327041

70337042
start = range->start >> sb->s_blocksize_bits;
@@ -7046,10 +7055,8 @@ int ext4_trim_fs(struct super_block *sb, struct fstrim_range *range)
70467055
if (minlen > EXT4_CLUSTERS_PER_GROUP(sb))
70477056
goto out;
70487057
}
7049-
if (end >= max_blks - 1) {
7058+
if (end >= max_blks - 1)
70507059
end = max_blks - 1;
7051-
eof = true;
7052-
}
70537060
if (end <= first_data_blk)
70547061
goto out;
70557062
if (start < first_data_blk)
@@ -7063,7 +7070,6 @@ int ext4_trim_fs(struct super_block *sb, struct fstrim_range *range)
70637070

70647071
/* end now represents the last cluster to discard in this group */
70657072
end = EXT4_CLUSTERS_PER_GROUP(sb) - 1;
7066-
whole_group = true;
70677073

70687074
for (group = first_group; group <= last_group; group++) {
70697075
grp = ext4_get_group_info(sb, group);
@@ -7082,13 +7088,11 @@ int ext4_trim_fs(struct super_block *sb, struct fstrim_range *range)
70827088
* change it for the last group, note that last_cluster is
70837089
* already computed earlier by ext4_get_group_no_and_offset()
70847090
*/
7085-
if (group == last_group) {
7091+
if (group == last_group)
70867092
end = last_cluster;
7087-
whole_group = eof ? true : end == EXT4_CLUSTERS_PER_GROUP(sb) - 1;
7088-
}
70897093
if (grp->bb_free >= minlen) {
70907094
cnt = ext4_trim_all_free(sb, group, first_cluster,
7091-
end, minlen, whole_group);
7095+
end, minlen);
70927096
if (cnt < 0) {
70937097
ret = cnt;
70947098
break;

0 commit comments

Comments
 (0)