|  | ##/bin/bash | 
|  | # SPDX-License-Identifier: GPL-2.0 | 
|  | # Copyright (c) 2013 Fusion IO, Inc.  All Rights Reserved. | 
|  | # | 
|  | # common functions for setting up and tearing down a dmflakey device | 
|  |  | 
|  | FLAKEY_ALLOW_WRITES=0 | 
|  | FLAKEY_DROP_WRITES=1 | 
|  | FLAKEY_ERROR_WRITES=2 | 
|  |  | 
|  | _init_flakey() | 
|  | { | 
|  | # Scratch device | 
|  | local BLK_DEV_SIZE=`blockdev --getsz $SCRATCH_DEV` | 
|  | FLAKEY_DEV=/dev/mapper/flakey-test | 
|  | FLAKEY_TABLE="0 $BLK_DEV_SIZE flakey $SCRATCH_DEV 0 180 0" | 
|  | FLAKEY_TABLE_DROP="0 $BLK_DEV_SIZE flakey $SCRATCH_DEV 0 0 180 1 drop_writes" | 
|  | FLAKEY_TABLE_ERROR="0 $BLK_DEV_SIZE flakey $SCRATCH_DEV 0 0 180 1 error_writes" | 
|  | _dmsetup_create flakey-test --table "$FLAKEY_TABLE" || \ | 
|  | _fatal "failed to create flakey device" | 
|  |  | 
|  | # Realtime device | 
|  | if [ -n "$SCRATCH_RTDEV" ]; then | 
|  | if [ -z "$NON_FLAKEY_RTDEV" ]; then | 
|  | # Set up the device switch | 
|  | local backing_dev="$SCRATCH_RTDEV" | 
|  | export NON_FLAKEY_RTDEV="$SCRATCH_RTDEV" | 
|  | SCRATCH_RTDEV=/dev/mapper/flakey-rttest | 
|  | else | 
|  | # Already set up; recreate tables | 
|  | local backing_dev="$NON_FLAKEY_RTDEV" | 
|  | fi | 
|  | local BLK_DEV_SIZE=`blockdev --getsz $backing_dev` | 
|  | FLAKEY_RTTABLE="0 $BLK_DEV_SIZE flakey $backing_dev 0 180 0" | 
|  | FLAKEY_RTTABLE_DROP="0 $BLK_DEV_SIZE flakey $backing_dev 0 0 180 1 drop_writes" | 
|  | FLAKEY_RTTABLE_ERROR="0 $BLK_DEV_SIZE flakey $backing_dev 0 0 180 1 error_writes" | 
|  | _dmsetup_create flakey-rttest --table "$FLAKEY_RTTABLE" || \ | 
|  | _fatal "failed to create flakey rt device" | 
|  | fi | 
|  |  | 
|  | # External log device | 
|  | if [ -n "$SCRATCH_LOGDEV" ]; then | 
|  | if [ -z "$NON_FLAKEY_LOGDEV" ]; then | 
|  | # Set up the device switch | 
|  | local backing_dev="$SCRATCH_LOGDEV" | 
|  | export NON_FLAKEY_LOGDEV="$SCRATCH_LOGDEV" | 
|  | SCRATCH_LOGDEV=/dev/mapper/flakey-logtest | 
|  | else | 
|  | # Already set up; recreate tables | 
|  | local backing_dev="$NON_FLAKEY_LOGDEV" | 
|  | fi | 
|  | local BLK_DEV_SIZE=`blockdev --getsz $backing_dev` | 
|  | FLAKEY_LOGTABLE="0 $BLK_DEV_SIZE flakey $backing_dev 0 180 0" | 
|  | FLAKEY_LOGTABLE_DROP="0 $BLK_DEV_SIZE flakey $backing_dev 0 0 180 1 drop_writes" | 
|  | FLAKEY_LOGTABLE_ERROR="0 $BLK_DEV_SIZE flakey $backing_dev 0 0 180 1 error_writes" | 
|  | _dmsetup_create flakey-logtest --table "$FLAKEY_LOGTABLE" || \ | 
|  | _fatal "failed to create flakey log device" | 
|  | fi | 
|  | } | 
|  |  | 
|  | _mount_flakey() | 
|  | { | 
|  | _scratch_options mount | 
|  |  | 
|  | mount -t $FSTYP $SCRATCH_OPTIONS $MOUNT_OPTIONS $FLAKEY_DEV $SCRATCH_MNT | 
|  | } | 
|  |  | 
|  | _unmount_flakey() | 
|  | { | 
|  | $UMOUNT_PROG $SCRATCH_MNT | 
|  | } | 
|  |  | 
|  | _cleanup_flakey() | 
|  | { | 
|  | # If dmsetup load fails then we need to make sure to do resume here | 
|  | # otherwise the umount will hang | 
|  | test -n "$NON_FLAKEY_LOGDEV" && $DMSETUP_PROG resume flakey-logtest &> /dev/null | 
|  | test -n "$NON_FLAKEY_RTDEV" && $DMSETUP_PROG resume flakey-rttest &> /dev/null | 
|  | $DMSETUP_PROG resume flakey-test > /dev/null 2>&1 | 
|  |  | 
|  | $UMOUNT_PROG $SCRATCH_MNT > /dev/null 2>&1 | 
|  |  | 
|  | _dmsetup_remove flakey-test | 
|  | test -n "$NON_FLAKEY_LOGDEV" && _dmsetup_remove flakey-logtest | 
|  | test -n "$NON_FLAKEY_RTDEV" && _dmsetup_remove flakey-rttest | 
|  |  | 
|  | if [ -n "$NON_FLAKEY_LOGDEV" ]; then | 
|  | SCRATCH_LOGDEV="$NON_FLAKEY_LOGDEV" | 
|  | unset NON_FLAKEY_LOGDEV | 
|  | fi | 
|  |  | 
|  | if [ -n "$NON_FLAKEY_RTDEV" ]; then | 
|  | SCRATCH_RTDEV="$NON_FLAKEY_RTDEV" | 
|  | unset NON_FLAKEY_RTDEV | 
|  | fi | 
|  | } | 
|  |  | 
|  | # _load_flakey_table <table> [lockfs] | 
|  | # | 
|  | # This defaults to --nolockfs, which doesn't freeze_fs() before loading the new | 
|  | # table, so it simulates power failure. | 
|  | _load_flakey_table() | 
|  | { | 
|  | case "$1" in | 
|  | "$FLAKEY_DROP_WRITES") | 
|  | table="$FLAKEY_TABLE_DROP" | 
|  | logtable="$FLAKEY_LOGTABLE_DROP" | 
|  | rttable="$FLAKEY_RTTABLE_DROP" | 
|  | ;; | 
|  | "$FLAKEY_ERROR_WRITES") | 
|  | table="$FLAKEY_TABLE_ERROR" | 
|  | logtable="$FLAKEY_LOGTABLE_ERROR" | 
|  | rttable="$FLAKEY_RTTABLE_ERROR" | 
|  | ;; | 
|  | *) | 
|  | table="$FLAKEY_TABLE" | 
|  | logtable="$FLAKEY_LOGTABLE" | 
|  | rttable="$FLAKEY_RTTABLE" | 
|  | ;; | 
|  | esac | 
|  |  | 
|  | suspend_opt="--nolockfs" | 
|  | [ $# -gt 1 ] && [ $2 -eq 1 ] && suspend_opt="" | 
|  |  | 
|  | # Suspend the scratch device before the log and realtime devices so | 
|  | # that the kernel can freeze and flush the filesystem if the caller | 
|  | # wanted a freeze. | 
|  | $DMSETUP_PROG suspend $suspend_opt flakey-test | 
|  | [ $? -ne 0 ] && _fatal "failed to suspend flakey-test" | 
|  |  | 
|  | if [ -n "$NON_FLAKEY_RTDEV" ]; then | 
|  | $DMSETUP_PROG suspend $suspend_opt flakey-rttest | 
|  | [ $? -ne 0 ] && _fatal "failed to suspend flakey-rttest" | 
|  | fi | 
|  |  | 
|  | if [ -n "$NON_FLAKEY_LOGDEV" ]; then | 
|  | $DMSETUP_PROG suspend $suspend_opt flakey-logtest | 
|  | [ $? -ne 0 ] && _fatal "failed to suspend flakey-logtest" | 
|  | fi | 
|  |  | 
|  | # There may be multiple dm targets in the table, and these dm targets | 
|  | # will be joined by the newline ("\n"). Option --table can not cope with | 
|  | # the multiple-targets case, so get them by reading from standard input. | 
|  | echo -e "$table" | $DMSETUP_PROG load flakey-test | 
|  | [ $? -ne 0 ] && _fatal "failed to load table into flakey-test" | 
|  |  | 
|  | if [ -n "$NON_FLAKEY_RTDEV" ]; then | 
|  | echo -e "$rttable" | $DMSETUP_PROG load flakey-rttest | 
|  | [ $? -ne 0 ] && _fatal "failed to load table into flakey-rttest" | 
|  | fi | 
|  |  | 
|  | if [ -n "$NON_FLAKEY_LOGDEV" ]; then | 
|  | echo -e "$logtable" | $DMSETUP_PROG load flakey-logtest | 
|  | [ $? -ne 0 ] && _fatal "failed to load table into flakey-logtest" | 
|  | fi | 
|  |  | 
|  | # Resume devices in the opposite order that we suspended them. | 
|  | if [ -n "$NON_FLAKEY_LOGDEV" ]; then | 
|  | $DMSETUP_PROG resume flakey-logtest | 
|  | [ $? -ne 0 ] && _fatal  "failed to resume flakey-logtest" | 
|  | fi | 
|  |  | 
|  | if [ -n "$NON_FLAKEY_RTDEV" ]; then | 
|  | $DMSETUP_PROG resume flakey-rttest | 
|  | [ $? -ne 0 ] && _fatal  "failed to resume flakey-rttest" | 
|  | fi | 
|  |  | 
|  | $DMSETUP_PROG resume flakey-test | 
|  | [ $? -ne 0 ] && _fatal  "failed to resume flakey-test" | 
|  | } | 
|  |  | 
|  | # Silently drop all writes and unmount/remount to simulate a crash/power | 
|  | # failure. | 
|  | _flakey_drop_and_remount() | 
|  | { | 
|  | _load_flakey_table $FLAKEY_DROP_WRITES | 
|  | _unmount_flakey | 
|  |  | 
|  | if [ "x$1" = "xyes" ]; then | 
|  | _check_scratch_fs $FLAKEY_DEV | 
|  | fi | 
|  |  | 
|  | _load_flakey_table $FLAKEY_ALLOW_WRITES | 
|  | _mount_flakey | 
|  | } | 
|  |  | 
|  | _require_flakey_with_error_writes() | 
|  | { | 
|  | local SIZE | 
|  | local TABLE | 
|  | local NAME=flakey-test | 
|  |  | 
|  | _require_dm_target flakey | 
|  |  | 
|  | SIZE=`blockdev --getsz $SCRATCH_DEV` | 
|  | TABLE="0 $SIZE flakey $SCRATCH_DEV 0 0 180 1 error_writes" | 
|  |  | 
|  | _dmsetup_create $NAME --table "$TABLE" || \ | 
|  | _notrun "This test requires error_writes feature in dm-flakey" | 
|  |  | 
|  | _cleanup_flakey | 
|  | } |