forked from openzfs/zfs
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Livelist logic should handle dedup blkptrs
= Problem When the livelist logic was designed it didn't take into account that when dedup is enabled the sublivelists can have consecutive FREE entries for the same block without an ALLOC entry for it in-between them. This caused panics in systems that were deleting/condesing clones where dedup is enabled. = This patch Update the logic to handle the dedup-case of consecutive FREEs in the livelist code. The logic still ensured that all the FREE entries are matched up with a respective ALLOC by keeping a refcount for each FREE blkptr that we encounter and ensuring that this refcount gets to zero by the time we are done processing the livelist. = Testing After I reproduced the issue with a shell script, I added a variant of that shell script to ZTS. After ensuring that this new test panics the system the same way as the original reproducer I tried it against the updated logic in this patch and verified that the system no longer panics. = Side Fixes * zdb -y no longer panics when encountering double frees Signed-off-by: Serapheim Dimitropoulos <[email protected]> Closes openzfs#11480
- Loading branch information
Showing
4 changed files
with
210 additions
and
64 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
88 changes: 88 additions & 0 deletions
88
tests/zfs-tests/tests/functional/cli_root/zfs_destroy/zfs_clone_livelist_dedup.ksh
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,88 @@ | ||
#!/bin/ksh -p | ||
# | ||
# This file and its contents are supplied under the terms of the | ||
# Common Development and Distribution License ("CDDL"), version 1.0. | ||
# You may only use this file in accordance with the terms of version | ||
# 1.0 of the CDDL. | ||
# | ||
# A full copy of the text of the CDDL should have accompanied this | ||
# source. A copy of the CDDL is also available via the Internet at | ||
# http://www.illumos.org/license/CDDL. | ||
# | ||
|
||
# | ||
# Copyright (c) 2021 by Delphix. All rights reserved. | ||
# | ||
|
||
# DESCRIPTION | ||
# Verify zfs destroy test for clones with livelists that contain | ||
# dedup blocks. This test is a baseline regression test created | ||
# to ensure that past bugs that we've encountered between dedup | ||
# and the livelist logic don't resurface. | ||
|
||
# STRATEGY | ||
# 1. Create a clone from a test filesystem and enable dedup. | ||
# 2. Write some data and create a livelist. | ||
# 3. Copy the data within the clone to create dedup blocks. | ||
# 4. Remove some of the dedup data to create multiple free | ||
# entries for the same block pointers. | ||
# 5. Process all the livelist entries by destroying the clone. | ||
|
||
. $STF_SUITE/include/libtest.shlib | ||
. $STF_SUITE/tests/functional/cli_root/zfs_destroy/zfs_destroy_common.kshlib | ||
|
||
function cleanup | ||
{ | ||
log_must zfs destroy -Rf $TESTPOOL/$TESTFS1 | ||
# Reset the minimum percent shared to 75 | ||
set_tunable32 LIVELIST_MIN_PERCENT_SHARED $ORIGINAL_MIN_SHARED | ||
} | ||
|
||
function test_dedup | ||
{ | ||
# Set a small percent shared threshold so the livelist is not disabled | ||
set_tunable32 LIVELIST_MIN_PERCENT_SHARED 10 | ||
clone_dataset $TESTFS1 snap $TESTCLONE | ||
|
||
# Enable dedup | ||
log_must zfs set dedup=on $TESTPOOL/$TESTCLONE | ||
|
||
# Create some data to be deduped | ||
log_must dd if=/dev/urandom of="/$TESTPOOL/$TESTCLONE/data" bs=512 count=10k | ||
|
||
# Create dedup blocks | ||
# Note: We sync before and after so all dedup blocks belong to the | ||
# same TXG, otherwise they won't look identical to the livelist | ||
# iterator due to their logical birth TXG being different. | ||
log_must zpool sync $TESTPOOL | ||
log_must cp /$TESTPOOL/$TESTCLONE/data /$TESTPOOL/$TESTCLONE/data-dup-0 | ||
log_must cp /$TESTPOOL/$TESTCLONE/data /$TESTPOOL/$TESTCLONE/data-dup-1 | ||
log_must cp /$TESTPOOL/$TESTCLONE/data /$TESTPOOL/$TESTCLONE/data-dup-2 | ||
log_must cp /$TESTPOOL/$TESTCLONE/data /$TESTPOOL/$TESTCLONE/data-dup-3 | ||
log_must zpool sync $TESTPOOL | ||
check_livelist_exists $TESTCLONE | ||
|
||
# Introduce "double frees" | ||
# We want to introduce consecutive FREEs of the same block as this | ||
# was what triggered past panics. | ||
# Note: Similarly to the previouys step we sync before and after our | ||
# our deletions so all the entries end up in the same TXG. | ||
log_must zpool sync $TESTPOOL | ||
log_must rm /$TESTPOOL/$TESTCLONE/data-dup-2 | ||
log_must rm /$TESTPOOL/$TESTCLONE/data-dup-3 | ||
log_must zpool sync $TESTPOOL | ||
check_livelist_exists $TESTCLONE | ||
|
||
log_must zfs destroy $TESTPOOL/$TESTCLONE | ||
check_livelist_gone | ||
} | ||
|
||
ORIGINAL_MIN_SHARED=$(get_tunable LIVELIST_MIN_PERCENT_SHARED) | ||
|
||
log_onexit cleanup | ||
log_must zfs create $TESTPOOL/$TESTFS1 | ||
log_must mkfile 5m /$TESTPOOL/$TESTFS1/atestfile | ||
log_must zfs snapshot $TESTPOOL/$TESTFS1@snap | ||
test_dedup | ||
|
||
log_pass "Clone's livelist processes dedup blocks as expected." |