1#!/usr/bin/env bash 2# group: rw 3# 4# Test FUSE exports (in ways that are not captured by the generic 5# tests) 6# 7# Copyright (C) 2020 Red Hat, Inc. 8# 9# This program is free software; you can redistribute it and/or modify 10# it under the terms of the GNU General Public License as published by 11# the Free Software Foundation; either version 2 of the License, or 12# (at your option) any later version. 13# 14# This program is distributed in the hope that it will be useful, 15# but WITHOUT ANY WARRANTY; without even the implied warranty of 16# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 17# GNU General Public License for more details. 18# 19# You should have received a copy of the GNU General Public License 20# along with this program. If not, see <http://www.gnu.org/licenses/>. 21# 22 23seq=$(basename "$0") 24echo "QA output created by $seq" 25 26status=1 # failure is the default! 27 28_cleanup() 29{ 30 _cleanup_qemu 31 _cleanup_test_img 32 rmdir "$EXT_MP" 2>/dev/null 33 rm -f "$EXT_MP" 34 rm -f "$COPIED_IMG" 35} 36trap "_cleanup; exit \$status" 0 1 2 3 15 37 38# get standard environment, filters and checks 39. ./common.rc 40. ./common.filter 41. ./common.qemu 42 43# Generic format, but needs a plain filename 44_supported_fmt generic 45if [ "$IMGOPTSSYNTAX" = "true" ]; then 46 _unsupported_fmt $IMGFMT 47fi 48# We need the image to have exactly the specified size, and VPC does 49# not allow that by default 50_unsupported_fmt vpc 51 52_supported_proto file # We create the FUSE export manually 53_supported_os Linux # We need /dev/urandom 54 55# $1: Export ID 56# $2: Options (beyond the node-name and ID) 57# $3: Expected return value (defaults to 'return') 58# $4: Node to export (defaults to 'node-format') 59fuse_export_add() 60{ 61 _send_qemu_cmd $QEMU_HANDLE \ 62 "{'execute': 'block-export-add', 63 'arguments': { 64 'type': 'fuse', 65 'id': '$1', 66 'node-name': '${4:-node-format}', 67 $2 68 } }" \ 69 "${3:-return}" \ 70 | _filter_imgfmt 71} 72 73# $1: Export ID 74fuse_export_del() 75{ 76 _send_qemu_cmd $QEMU_HANDLE \ 77 "{'execute': 'block-export-del', 78 'arguments': { 79 'id': '$1' 80 } }" \ 81 'return' 82 83 _send_qemu_cmd $QEMU_HANDLE \ 84 '' \ 85 'BLOCK_EXPORT_DELETED' 86} 87 88# Return the length of the protocol file 89# $1: Protocol node export mount point 90# $2: Original file (to compare) 91get_proto_len() 92{ 93 len1=$(stat -c '%s' "$1") 94 len2=$(stat -c '%s' "$2") 95 96 if [ "$len1" != "$len2" ]; then 97 echo 'ERROR: Length of export and original differ:' >&2 98 echo "$len1 != $len2" >&2 99 else 100 echo '(OK: Lengths of export and original are the same)' >&2 101 fi 102 103 echo "$len1" 104} 105 106COPIED_IMG="$TEST_IMG.copy" 107EXT_MP="$TEST_IMG.fuse" 108 109echo '=== Set up ===' 110 111# Create image with random data 112_make_test_img 64M 113$QEMU_IO -c 'write -s /dev/urandom 0 64M' "$TEST_IMG" | _filter_qemu_io 114 115_launch_qemu 116_send_qemu_cmd $QEMU_HANDLE \ 117 "{'execute': 'qmp_capabilities'}" \ 118 'return' 119 120# Separate blockdev-add calls for format and protocol so we can remove 121# the format layer later on 122_send_qemu_cmd $QEMU_HANDLE \ 123 "{'execute': 'blockdev-add', 124 'arguments': { 125 'driver': 'file', 126 'node-name': 'node-protocol', 127 'filename': '$TEST_IMG' 128 } }" \ 129 'return' 130 131_send_qemu_cmd $QEMU_HANDLE \ 132 "{'execute': 'blockdev-add', 133 'arguments': { 134 'driver': '$IMGFMT', 135 'node-name': 'node-format', 136 'file': 'node-protocol' 137 } }" \ 138 'return' 139 140echo 141echo '=== Mountpoint not present ===' 142 143rmdir "$EXT_MP" 2>/dev/null 144rm -f "$EXT_MP" 145output=$(fuse_export_add 'export-err' "'mountpoint': '$EXT_MP'" error) 146 147if echo "$output" | grep -q "Invalid parameter 'fuse'"; then 148 _notrun 'No FUSE support' 149fi 150 151echo "$output" 152 153echo 154echo '=== Mountpoint is a directory ===' 155 156mkdir "$EXT_MP" 157fuse_export_add 'export-err' "'mountpoint': '$EXT_MP'" error 158rmdir "$EXT_MP" 159 160echo 161echo '=== Mountpoint is a regular file ===' 162 163touch "$EXT_MP" 164fuse_export_add 'export-mp' "'mountpoint': '$EXT_MP'" 165 166# Check that the export presents the same data as the original image 167$QEMU_IMG compare -f raw -F $IMGFMT -U "$EXT_MP" "$TEST_IMG" 168 169echo 170echo '=== Mount over existing file ===' 171 172# This is the coolest feature of FUSE exports: You can transparently 173# make images in any format appear as raw images 174fuse_export_add 'export-img' "'mountpoint': '$TEST_IMG'" 175 176# Accesses both exports at the same time, so we get a concurrency test 177$QEMU_IMG compare -f raw -F raw -U "$EXT_MP" "$TEST_IMG" 178 179# Just to be sure, we later want to compare the data offline. Also, 180# this allows us to see that cp works without complaining. 181# (This is not a given, because cp will expect a short read at EOF. 182# Internally, qemu does not allow short reads, so we have to check 183# whether the FUSE export driver lets them work.) 184cp "$TEST_IMG" "$COPIED_IMG" 185 186# $TEST_IMG will be in mode 0400 because it is read-only; we are going 187# to write to the copy, so make it writable 188chmod 0600 "$COPIED_IMG" 189 190echo 191echo '=== Double export ===' 192 193# We have already seen that exporting a node twice works fine, but you 194# cannot export anything twice on the same mount point. The reason is 195# that qemu has to stat the given mount point, and this would have to 196# be answered by the same qemu instance if it already has an export 197# there. However, it cannot answer the stat because it is itself 198# caught up in that same stat. 199fuse_export_add 'export-err' "'mountpoint': '$EXT_MP'" error 200 201echo 202echo '=== Remove export ===' 203 204# Double-check that $EXT_MP appears as a non-empty file (the raw image) 205$QEMU_IMG info -f raw "$EXT_MP" | grep 'virtual size' 206 207fuse_export_del 'export-mp' 208 209# See that the file appears empty again 210$QEMU_IMG info -f raw "$EXT_MP" | grep 'virtual size' 211 212echo 213echo '=== Writable export ===' 214 215fuse_export_add 'export-mp' "'mountpoint': '$EXT_MP', 'writable': true" 216 217# Check that writing to the read-only export fails 218$QEMU_IO -f raw -c 'write -P 42 1M 64k' "$TEST_IMG" 2>&1 \ 219 | _filter_qemu_io | _filter_testdir | _filter_imgfmt 220 221# But here it should work 222$QEMU_IO -f raw -c 'write -P 42 1M 64k' "$EXT_MP" | _filter_qemu_io 223 224# (Adjust the copy, too) 225$QEMU_IO -f raw -c 'write -P 42 1M 64k' "$COPIED_IMG" | _filter_qemu_io 226 227echo 228echo '=== Resizing exports ===' 229 230# Here, we need to export the protocol node -- the format layer may 231# not be growable, simply because the format does not support it. 232 233# Remove all exports and the format node first so permissions will not 234# get in the way 235fuse_export_del 'export-mp' 236fuse_export_del 'export-img' 237 238_send_qemu_cmd $QEMU_HANDLE \ 239 "{'execute': 'blockdev-del', 240 'arguments': { 241 'node-name': 'node-format' 242 } }" \ 243 'return' 244 245# Now export the protocol node 246fuse_export_add \ 247 'export-mp' \ 248 "'mountpoint': '$EXT_MP', 'writable': true" \ 249 'return' \ 250 'node-protocol' 251 252echo 253echo '--- Try growing non-growable export ---' 254 255# Get the current size so we can write beyond the EOF 256orig_len=$(get_proto_len "$EXT_MP" "$TEST_IMG") 257orig_disk_usage=$(stat -c '%b' "$TEST_IMG") 258 259# Should fail (exports are non-growable by default) 260# (Note that qemu-io can never write beyond the EOF, so we have to use 261# dd here) 262dd if=/dev/zero of="$EXT_MP" bs=1 count=64k seek=$orig_len 2>&1 \ 263 | _filter_testdir | _filter_imgfmt 264 265echo 266echo '--- Resize export ---' 267 268# But we can truncate it explicitly; even with fallocate 269fallocate -o "$orig_len" -l 64k "$EXT_MP" 270 271new_len=$(get_proto_len "$EXT_MP" "$TEST_IMG") 272if [ "$new_len" != "$((orig_len + 65536))" ]; then 273 echo 'ERROR: Unexpected post-truncate image size:' 274 echo "$new_len != $((orig_len + 65536))" 275else 276 echo 'OK: Post-truncate image size is as expected' 277fi 278 279new_disk_usage=$(stat -c '%b' "$TEST_IMG") 280if [ "$new_disk_usage" -gt "$orig_disk_usage" ]; then 281 echo 'OK: Disk usage grew with fallocate' 282else 283 echo 'ERROR: Disk usage did not grow despite fallocate:' 284 echo "$orig_disk_usage => $new_disk_usage" 285fi 286 287echo 288echo '--- Try growing growable export ---' 289 290# Now export as growable 291fuse_export_del 'export-mp' 292fuse_export_add \ 293 'export-mp' \ 294 "'mountpoint': '$EXT_MP', 'writable': true, 'growable': true" \ 295 'return' \ 296 'node-protocol' 297 298# Now we should be able to write beyond the EOF 299dd if=/dev/zero of="$EXT_MP" bs=1 count=64k seek=$new_len 2>&1 \ 300 | _filter_testdir | _filter_imgfmt 301 302new_len=$(get_proto_len "$EXT_MP" "$TEST_IMG") 303if [ "$new_len" != "$((orig_len + 131072))" ]; then 304 echo 'ERROR: Unexpected post-grow image size:' 305 echo "$new_len != $((orig_len + 131072))" 306else 307 echo 'OK: Post-grow image size is as expected' 308fi 309 310echo 311echo '--- Shrink export ---' 312 313# Now go back to the original size 314truncate -s "$orig_len" "$EXT_MP" 315 316new_len=$(get_proto_len "$EXT_MP" "$TEST_IMG") 317if [ "$new_len" != "$orig_len" ]; then 318 echo 'ERROR: Unexpected post-truncate image size:' 319 echo "$new_len != $orig_len" 320else 321 echo 'OK: Post-truncate image size is as expected' 322fi 323 324echo 325echo '=== Tear down ===' 326 327_send_qemu_cmd $QEMU_HANDLE \ 328 "{'execute': 'quit'}" \ 329 'return' 330 331wait=yes _cleanup_qemu 332 333echo 334echo '=== Compare copy with original ===' 335 336$QEMU_IMG compare -f raw -F $IMGFMT "$COPIED_IMG" "$TEST_IMG" 337 338# success, all done 339echo "*** done" 340rm -f $seq.full 341status=0 342