mirror of
https://git.proxmox.com/git/mirror_zfs.git
synced 2025-01-12 19:20:28 +03:00
Fix taskq creation failure in vdev_open_children()
When creating and destroying pools in tight loop it's possible to exhaust the number of allowed threads on a system. This results in taskq_create() failling and a NULL dereference. Resolve the issue by falling back to opening the vdevs all synchronously. Reviewed-by: Denys Rtveliashvili <denys@rtveliashvili.name> Reviewed-by: Håkan Johansson <f96hajo@chalmers.se> Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov> Closes zfsonlinux/spl#521 Closes #4637
This commit is contained in:
parent
1bbd877049
commit
13d9a004fe
@ -89,6 +89,7 @@ AC_DEFUN([ZFS_AC_CONFIG_USER_COMMANDS_COMMON], [
|
|||||||
AC_PATH_TOOL(USERADD, useradd, "/usr/sbin/useradd")
|
AC_PATH_TOOL(USERADD, useradd, "/usr/sbin/useradd")
|
||||||
AC_PATH_TOOL(USERDEL, userdel, "/usr/sbin/userdel")
|
AC_PATH_TOOL(USERDEL, userdel, "/usr/sbin/userdel")
|
||||||
AC_PATH_TOOL(USERMOD, usermod, "/usr/sbin/usermod")
|
AC_PATH_TOOL(USERMOD, usermod, "/usr/sbin/usermod")
|
||||||
|
AC_PATH_TOOL(UUIDGEN, uuidgen, "")
|
||||||
AC_PATH_TOOL(WAIT, wait, "wait") dnl # Builtin in bash
|
AC_PATH_TOOL(WAIT, wait, "wait") dnl # Builtin in bash
|
||||||
AC_PATH_TOOL(WC, wc, "")
|
AC_PATH_TOOL(WC, wc, "")
|
||||||
])
|
])
|
||||||
|
@ -1179,12 +1179,15 @@ vdev_open_children(vdev_t *vd)
|
|||||||
* spa_namespace_lock
|
* spa_namespace_lock
|
||||||
*/
|
*/
|
||||||
if (vdev_uses_zvols(vd)) {
|
if (vdev_uses_zvols(vd)) {
|
||||||
|
retry_sync:
|
||||||
for (c = 0; c < children; c++)
|
for (c = 0; c < children; c++)
|
||||||
vd->vdev_child[c]->vdev_open_error =
|
vd->vdev_child[c]->vdev_open_error =
|
||||||
vdev_open(vd->vdev_child[c]);
|
vdev_open(vd->vdev_child[c]);
|
||||||
} else {
|
} else {
|
||||||
tq = taskq_create("vdev_open", children, minclsyspri,
|
tq = taskq_create("vdev_open", children, minclsyspri,
|
||||||
children, children, TASKQ_PREPOPULATE);
|
children, children, TASKQ_PREPOPULATE);
|
||||||
|
if (tq == NULL)
|
||||||
|
goto retry_sync;
|
||||||
|
|
||||||
for (c = 0; c < children; c++)
|
for (c = 0; c < children; c++)
|
||||||
VERIFY(taskq_dispatch(tq, vdev_open_child,
|
VERIFY(taskq_dispatch(tq, vdev_open_child,
|
||||||
|
@ -248,6 +248,7 @@ tests = [
|
|||||||
'zpool_create_009_neg', 'zpool_create_010_neg', 'zpool_create_017_neg',
|
'zpool_create_009_neg', 'zpool_create_010_neg', 'zpool_create_017_neg',
|
||||||
'zpool_create_018_pos', 'zpool_create_019_pos',
|
'zpool_create_018_pos', 'zpool_create_019_pos',
|
||||||
'zpool_create_021_pos', 'zpool_create_022_pos', 'zpool_create_023_neg',
|
'zpool_create_021_pos', 'zpool_create_022_pos', 'zpool_create_023_neg',
|
||||||
|
'zpool_create_024_pos',
|
||||||
'zpool_create_features_001_pos', 'zpool_create_features_002_pos',
|
'zpool_create_features_001_pos', 'zpool_create_features_002_pos',
|
||||||
'zpool_create_features_003_pos', 'zpool_create_features_004_neg']
|
'zpool_create_features_003_pos', 'zpool_create_features_004_neg']
|
||||||
|
|
||||||
|
@ -121,6 +121,7 @@ export UNSHARE="@UNSHARE@"
|
|||||||
export USERADD="@USERADD@"
|
export USERADD="@USERADD@"
|
||||||
export USERDEL="@USERDEL@"
|
export USERDEL="@USERDEL@"
|
||||||
export USERMOD="@USERMOD@"
|
export USERMOD="@USERMOD@"
|
||||||
|
export UUIDGEN="@UUIDGEN@"
|
||||||
export VMSTAT="@VMSTAT@"
|
export VMSTAT="@VMSTAT@"
|
||||||
export WAIT="@WAIT@"
|
export WAIT="@WAIT@"
|
||||||
export WC="@WC@"
|
export WC="@WC@"
|
||||||
|
@ -26,6 +26,7 @@ dist_pkgdata_SCRIPTS = \
|
|||||||
zpool_create_021_pos.ksh \
|
zpool_create_021_pos.ksh \
|
||||||
zpool_create_022_pos.ksh \
|
zpool_create_022_pos.ksh \
|
||||||
zpool_create_023_neg.ksh \
|
zpool_create_023_neg.ksh \
|
||||||
|
zpool_create_024_pos.ksh \
|
||||||
zpool_create_features_001_pos.ksh \
|
zpool_create_features_001_pos.ksh \
|
||||||
zpool_create_features_002_pos.ksh \
|
zpool_create_features_002_pos.ksh \
|
||||||
zpool_create_features_003_pos.ksh \
|
zpool_create_features_003_pos.ksh \
|
||||||
|
148
tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_024_pos.ksh
Executable file
148
tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_024_pos.ksh
Executable file
@ -0,0 +1,148 @@
|
|||||||
|
#!/bin/ksh -p
|
||||||
|
#
|
||||||
|
# CDDL HEADER START
|
||||||
|
#
|
||||||
|
# The contents of this file are subject to the terms of the
|
||||||
|
# Common Development and Distribution License (the "License").
|
||||||
|
# You may not use this file except in compliance with the License.
|
||||||
|
#
|
||||||
|
# You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
|
||||||
|
# or http://www.opensolaris.org/os/licensing.
|
||||||
|
# See the License for the specific language governing permissions
|
||||||
|
# and limitations under the License.
|
||||||
|
#
|
||||||
|
# When distributing Covered Code, include this CDDL HEADER in each
|
||||||
|
# file and include the License file at usr/src/OPENSOLARIS.LICENSE.
|
||||||
|
# If applicable, add the following below this CDDL HEADER, with the
|
||||||
|
# fields enclosed by brackets "[]" replaced with your own identifying
|
||||||
|
# information: Portions Copyright [yyyy] [name of copyright owner]
|
||||||
|
#
|
||||||
|
# CDDL HEADER END
|
||||||
|
#
|
||||||
|
|
||||||
|
#
|
||||||
|
# Copyright (c) 2016, Lawrence Livermore National Security, LLC.
|
||||||
|
#
|
||||||
|
|
||||||
|
. $STF_SUITE/include/libtest.shlib
|
||||||
|
. $STF_SUITE/tests/functional/cli_root/zpool_create/zpool_create.cfg
|
||||||
|
|
||||||
|
#
|
||||||
|
# DESCRIPTION:
|
||||||
|
# Many 'zpool create' and 'zpool destroy' must succeed concurrently.
|
||||||
|
#
|
||||||
|
# STRATEGY:
|
||||||
|
# 1. Create N process each of which create/destroy a pool M times.
|
||||||
|
# 2. Allow all process to run to completion.
|
||||||
|
# 3. Verify all pools and their vdevs were destroyed.
|
||||||
|
#
|
||||||
|
|
||||||
|
verify_runnable "global"
|
||||||
|
|
||||||
|
function cleanup
|
||||||
|
{
|
||||||
|
if [[ -n "$child_pids" ]]; then
|
||||||
|
for wait_pid in $child_pids; do
|
||||||
|
$KILL $wait_pid 2>/dev/null
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ -n "$child_pools" ]]; then
|
||||||
|
for pool in $child_pools; do
|
||||||
|
typeset vdev0="$TEST_BASE_DIR/$pool-vdev0.img"
|
||||||
|
typeset vdev1="$TEST_BASE_DIR/$pool-vdev1.img"
|
||||||
|
|
||||||
|
if poolexists $pool; then
|
||||||
|
destroy_pool $pool
|
||||||
|
fi
|
||||||
|
|
||||||
|
$RM -f $vdev0 $vdev1
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
log_onexit cleanup
|
||||||
|
|
||||||
|
log_assert "Many 'zpool create' and 'zpool destroy' must succeed concurrently."
|
||||||
|
|
||||||
|
child_pids=""
|
||||||
|
child_pools=""
|
||||||
|
|
||||||
|
function zpool_stress
|
||||||
|
{
|
||||||
|
typeset pool=$1
|
||||||
|
typeset vdev0="$TEST_BASE_DIR/$pool-vdev0.img"
|
||||||
|
typeset vdev1="$TEST_BASE_DIR/$pool-vdev1.img"
|
||||||
|
typeset -i iters=$2
|
||||||
|
typeset retry=10
|
||||||
|
typeset j=0
|
||||||
|
|
||||||
|
$TRUNCATE -s $FILESIZE $vdev0
|
||||||
|
$TRUNCATE -s $FILESIZE $vdev1
|
||||||
|
|
||||||
|
while [[ $j -lt $iters ]]; do
|
||||||
|
((j = j + 1))
|
||||||
|
$SLEEP 1
|
||||||
|
|
||||||
|
$ZPOOL create $pool $vdev0 $vdev1
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
return 1;
|
||||||
|
fi
|
||||||
|
|
||||||
|
# The 'zfs destroy' command is retried because it can
|
||||||
|
# transiently return EBUSY when blkid is concurrently
|
||||||
|
# probing new volumes and therefore has them open.
|
||||||
|
typeset k=0;
|
||||||
|
while [[ $k -lt $retry ]]; do
|
||||||
|
((k = k + 1))
|
||||||
|
|
||||||
|
$ZPOOL destroy $pool
|
||||||
|
if [ $? -eq 0 ]; then
|
||||||
|
break;
|
||||||
|
elif [ $k -eq $retry ]; then
|
||||||
|
return 1;
|
||||||
|
fi
|
||||||
|
|
||||||
|
$SLEEP 3
|
||||||
|
done
|
||||||
|
done
|
||||||
|
|
||||||
|
$RM -f $vdev0 $vdev1
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
# 1. Create 128 process each of which create/destroy a pool 5 times.
|
||||||
|
typeset i=0
|
||||||
|
while [[ $i -lt 128 ]]; do
|
||||||
|
typeset uuid=$($UUIDGEN | $CUT -c1-13)
|
||||||
|
|
||||||
|
zpool_stress $TESTPOOL-$uuid 5 &
|
||||||
|
typeset pid=$!
|
||||||
|
|
||||||
|
child_pids="$child_pids $pid"
|
||||||
|
child_pools="$child_pools $TESTPOOL-$uuid"
|
||||||
|
((i = i + 1))
|
||||||
|
done
|
||||||
|
|
||||||
|
# 2. Allow all process to run to completion.
|
||||||
|
wait
|
||||||
|
|
||||||
|
# 3. Verify all pools and their vdevs were destroyed.
|
||||||
|
for pool in $child_pools; do
|
||||||
|
typeset vdev0="$TEST_BASE_DIR/$pool-vdev0.img"
|
||||||
|
typeset vdev1="$TEST_BASE_DIR/$pool-vdev1.img"
|
||||||
|
|
||||||
|
if poolexists $pool; then
|
||||||
|
log_fail "pool $pool exists"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -e $vdev0 ]; then
|
||||||
|
log_fail "pool vdev $vdev0 exists"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -e $vdev1 ]; then
|
||||||
|
log_fail "pool vdev $vdev1 exists"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
log_pass "Many 'zpool create' and 'zpool destroy' must succeed concurrently."
|
Loading…
Reference in New Issue
Block a user