opnsense-src/tests/sys/geom/class/mirror/sync_error.sh
Alan Somers cf551b8a98 Unbreak geli/gmirror testcases if their geom classes cannot be loaded
The problem with the logic prior to this commit was twofold:

1. The wrong set of idioms (TAP-compatible) were being applied to the ATF
   testcases when run, resulting in confusing ATF failure results on setup.
2. The cleanup subroutines were broken when the geom classes could not be
   loaded as they exited with 0 unexpectedly.

This commit changes the test code to source the class-specific configuration
(conf.sh) once globally, instead of sourcing it per testcase and per cleanup
subroutine, and to call the ATF-specific setup subroutine(s) inline in
the testcases.

The refactoring done is effectively a no-op for the TAP testcases, modulo
any refactoring done to create common code between the ATF and TAP
testcases.

This unbreaks the geli testcases converted to ATF in r327662 and r327683,
and the gmirror testcases added in r327780, respectively, when the geom
class could not be loaded.

tests/sys/geom/class/mirror/...
    While here, ignore errors when turning debug failpoint sysctl off, which
    could occur if the gmirror class was not loaded.

Submitted by:	ngie
MFC after:	2 weeks
Pull Request:	https://github.com/freebsd/freebsd/pull/241
2018-12-02 05:06:37 +00:00

109 lines
2.9 KiB
Bash

# $FreeBSD$
ATF_TEST=true
. $(atf_get_srcdir)/conf.sh
REG_READ_FP=debug.fail_point.g_mirror_regular_request_read
atf_test_case sync_read_error_2_disks cleanup
sync_read_error_2_disks_head()
{
atf_set "descr" \
"Ensure that we properly handle read errors during synchronization."
atf_set "require.user" "root"
}
sync_read_error_2_disks_body()
{
geom_atf_test_setup
f1=$(mktemp ${base}.XXXXXX)
f2=$(mktemp ${base}.XXXXXX)
atf_check dd if=/dev/zero bs=1M count=32 of=$f1 status=none
atf_check truncate -s 32M $f2
md1=$(attach_md -t vnode -f ${f1})
md2=$(attach_md -t vnode -f ${f2})
atf_check gmirror label $name $md1
devwait
atf_check -s ignore -e empty -o not-empty sysctl ${REG_READ_FP}='1*return(5)'
# If a read error occurs while synchronizing and the mirror contains
# a single active disk, gmirror has no choice but to fail the
# synchronization and kick the new disk out of the mirror.
atf_check gmirror insert $name $md2
sleep 0.1
syncwait
atf_check [ $(gmirror status -s $name | wc -l) -eq 1 ]
atf_check -s exit:0 -o match:"DEGRADED $md1 \(ACTIVE\)" \
gmirror status -s $name
}
sync_read_error_2_disks_cleanup()
{
atf_check -s ignore -e ignore -o ignore sysctl ${REG_READ_FP}='off'
gmirror_test_cleanup
}
atf_test_case sync_read_error_3_disks cleanup
sync_read_error_3_disks_head()
{
atf_set "descr" \
"Ensure that we properly handle read errors during synchronization."
atf_set "require.user" "root"
}
sync_read_error_3_disks_body()
{
geom_atf_test_setup
f1=$(mktemp ${base}.XXXXXX)
f2=$(mktemp ${base}.XXXXXX)
f3=$(mktemp ${base}.XXXXXX)
atf_check dd if=/dev/random bs=1M count=32 of=$f1 status=none
atf_check truncate -s 32M $f2
atf_check truncate -s 32M $f3
md1=$(attach_md -t vnode -f ${f1})
md2=$(attach_md -t vnode -f ${f2})
md3=$(attach_md -t vnode -f ${f3})
atf_check gmirror label $name $md1
devwait
atf_check gmirror insert $name $md2
syncwait
atf_check -s exit:0 -e empty -o not-empty sysctl ${REG_READ_FP}='1*return(5)'
# If a read error occurs while synchronizing a new disk, and we have
# multiple active disks, we retry the read after an error. The disk
# which returned the read error is kicked out of the mirror.
atf_check gmirror insert $name $md3
syncwait
atf_check [ $(gmirror status -s $name | wc -l) -eq 2 ]
atf_check -s exit:0 -o match:"DEGRADED $md3 \(ACTIVE\)" \
gmirror status -s $name
# Make sure that the two active disks are identical. Destroy the
# mirror first so that the metadata sectors are wiped.
if $(gmirror status -s $name | grep -q $md1); then
active=$md1
else
active=$md2
fi
atf_check gmirror destroy $name
atf_check cmp /dev/$active /dev/$md3
}
sync_read_error_3_disks_cleanup()
{
atf_check -s ignore -e ignore -o ignore sysctl ${REG_READ_FP}='off'
gmirror_test_cleanup
}
atf_init_test_cases()
{
atf_add_test_case sync_read_error_2_disks
atf_add_test_case sync_read_error_3_disks
}