2017-07-20 22:26:42 +00:00
|
|
|
#!/usr/bin/env bash
|
2017-05-30 14:17:21 +00:00
|
|
|
|
|
|
|
# - fallocate -z deallocates because BLKDEV_ZERO_NOUNMAP hint is ignored by
|
|
|
|
# krbd
|
|
|
|
#
|
2019-03-25 19:58:00 +00:00
|
|
|
# - big unaligned blkdiscard and fallocate -z/-p leave the objects in place
|
2017-05-30 14:17:21 +00:00
|
|
|
|
|
|
|
set -ex
|
|
|
|
|
|
|
|
# no blkdiscard(8) in trusty
|
|
|
|
function py_blkdiscard() {
|
|
|
|
local offset=$1
|
|
|
|
|
2019-11-26 11:06:58 +00:00
|
|
|
python3 <<EOF
|
2017-05-30 14:17:21 +00:00
|
|
|
import fcntl, struct
|
|
|
|
BLKDISCARD = 0x1277
|
|
|
|
with open('$DEV', 'w') as dev:
|
|
|
|
fcntl.ioctl(dev, BLKDISCARD, struct.pack('QQ', $offset, $IMAGE_SIZE - $offset))
|
|
|
|
EOF
|
|
|
|
}
|
|
|
|
|
|
|
|
# fallocate(1) in trusty doesn't support -z/-p
|
|
|
|
function py_fallocate() {
|
|
|
|
local mode=$1
|
|
|
|
local offset=$2
|
|
|
|
|
2019-11-26 11:06:58 +00:00
|
|
|
python3 <<EOF
|
2017-05-30 14:17:21 +00:00
|
|
|
import os, ctypes, ctypes.util
|
|
|
|
FALLOC_FL_KEEP_SIZE = 0x01
|
|
|
|
FALLOC_FL_PUNCH_HOLE = 0x02
|
|
|
|
FALLOC_FL_ZERO_RANGE = 0x10
|
|
|
|
libc = ctypes.CDLL(ctypes.util.find_library('c'), use_errno=True)
|
|
|
|
with open('$DEV', 'w') as dev:
|
|
|
|
if libc.fallocate(dev.fileno(), ctypes.c_int($mode), ctypes.c_long($offset), ctypes.c_long($IMAGE_SIZE - $offset)):
|
|
|
|
err = ctypes.get_errno()
|
|
|
|
raise OSError(err, os.strerror(err))
|
|
|
|
EOF
|
|
|
|
}
|
|
|
|
|
|
|
|
function allocate() {
|
|
|
|
xfs_io -c "pwrite -b $OBJECT_SIZE -W 0 $IMAGE_SIZE" $DEV
|
2018-04-12 09:19:09 +00:00
|
|
|
assert_allocated
|
|
|
|
}
|
|
|
|
|
|
|
|
function assert_allocated() {
|
2017-05-30 14:17:21 +00:00
|
|
|
cmp <(od -xAx $DEV) - <<EOF
|
|
|
|
000000 cdcd cdcd cdcd cdcd cdcd cdcd cdcd cdcd
|
|
|
|
*
|
|
|
|
$(printf %x $IMAGE_SIZE)
|
|
|
|
EOF
|
|
|
|
[[ $(rados -p rbd ls | grep -c rbd_data.$IMAGE_ID) -eq $NUM_OBJECTS ]]
|
|
|
|
}
|
|
|
|
|
2018-04-12 09:19:09 +00:00
|
|
|
function assert_zeroes() {
|
2018-04-12 09:19:09 +00:00
|
|
|
local num_objects_expected=$1
|
|
|
|
|
2017-05-30 14:17:21 +00:00
|
|
|
cmp <(od -xAx $DEV) - <<EOF
|
|
|
|
000000 0000 0000 0000 0000 0000 0000 0000 0000
|
|
|
|
*
|
|
|
|
$(printf %x $IMAGE_SIZE)
|
|
|
|
EOF
|
2018-04-12 09:19:09 +00:00
|
|
|
[[ $(rados -p rbd ls | grep -c rbd_data.$IMAGE_ID) -eq $num_objects_expected ]]
|
2017-05-30 14:17:21 +00:00
|
|
|
}
|
|
|
|
|
2018-04-12 09:19:09 +00:00
|
|
|
function assert_zeroes_unaligned() {
|
2017-05-30 14:17:21 +00:00
|
|
|
local num_objects_expected=$1
|
|
|
|
|
|
|
|
cmp <(od -xAx $DEV) - <<EOF
|
|
|
|
000000 cdcd cdcd cdcd cdcd cdcd cdcd cdcd cdcd
|
|
|
|
*
|
|
|
|
$(printf %x $((OBJECT_SIZE / 2))) 0000 0000 0000 0000 0000 0000 0000 0000
|
|
|
|
*
|
|
|
|
$(printf %x $IMAGE_SIZE)
|
|
|
|
EOF
|
|
|
|
[[ $(rados -p rbd ls | grep -c rbd_data.$IMAGE_ID) -eq $num_objects_expected ]]
|
|
|
|
for ((i = 0; i < $num_objects_expected; i++)); do
|
qa: krbd_fallocate.sh: zero can be munged to truncate
The test case is issuing discards that span two objects: the tail of
the first object is truncated, the head of the second object is zeroed.
These discards aren't serial, so there is a race:
discard i ~ i + 1: truncate i, zero i + 1
discard i + 1 ~ i + 2: truncate i + 1, zero i + 2
can be executed as
truncate i + 1, zero i + 2, truncate i, zero i + 1
For object i + 1, the sequence ends up being truncate tail, then zero
head. This zero op is munged to truncate on the OSD, resulting in size
0 instead of OBJECT_SIZE / 2.
Signed-off-by: Ilya Dryomov <idryomov@gmail.com>
2019-03-25 19:59:29 +00:00
|
|
|
rados -p rbd stat rbd_data.$IMAGE_ID.$(printf %016x $i) | egrep "(size $((OBJECT_SIZE / 2)))|(size 0)"
|
2017-05-30 14:17:21 +00:00
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
IMAGE_NAME="fallocate-test"
|
|
|
|
|
|
|
|
rbd create --size 200 $IMAGE_NAME
|
|
|
|
|
2019-11-26 11:06:58 +00:00
|
|
|
IMAGE_SIZE=$(rbd info --format=json $IMAGE_NAME | python3 -c 'import sys, json; print(json.load(sys.stdin)["size"])')
|
|
|
|
OBJECT_SIZE=$(rbd info --format=json $IMAGE_NAME | python3 -c 'import sys, json; print(json.load(sys.stdin)["object_size"])')
|
2017-05-30 14:17:21 +00:00
|
|
|
NUM_OBJECTS=$((IMAGE_SIZE / OBJECT_SIZE))
|
|
|
|
[[ $((IMAGE_SIZE % OBJECT_SIZE)) -eq 0 ]]
|
|
|
|
|
|
|
|
IMAGE_ID="$(rbd info --format=json $IMAGE_NAME |
|
2019-11-26 11:06:58 +00:00
|
|
|
python3 -c "import sys, json; print(json.load(sys.stdin)['block_name_prefix'].split('.')[1])")"
|
2017-05-30 14:17:21 +00:00
|
|
|
|
|
|
|
DEV=$(sudo rbd map $IMAGE_NAME)
|
|
|
|
|
2018-02-19 19:02:57 +00:00
|
|
|
# make sure -ENOENT is hidden
|
2018-04-12 09:19:09 +00:00
|
|
|
assert_zeroes 0
|
2018-02-19 19:02:57 +00:00
|
|
|
py_blkdiscard 0
|
2018-04-12 09:19:09 +00:00
|
|
|
assert_zeroes 0
|
2018-02-19 19:02:57 +00:00
|
|
|
|
2017-05-30 14:17:21 +00:00
|
|
|
# blkdev_issue_discard
|
|
|
|
allocate
|
|
|
|
py_blkdiscard 0
|
2018-04-12 09:19:09 +00:00
|
|
|
assert_zeroes 0
|
2017-05-30 14:17:21 +00:00
|
|
|
|
|
|
|
# blkdev_issue_zeroout w/ BLKDEV_ZERO_NOUNMAP
|
|
|
|
allocate
|
|
|
|
py_fallocate FALLOC_FL_ZERO_RANGE\|FALLOC_FL_KEEP_SIZE 0
|
2018-04-12 09:19:09 +00:00
|
|
|
assert_zeroes 0
|
2017-05-30 14:17:21 +00:00
|
|
|
|
|
|
|
# blkdev_issue_zeroout w/ BLKDEV_ZERO_NOFALLBACK
|
|
|
|
allocate
|
|
|
|
py_fallocate FALLOC_FL_PUNCH_HOLE\|FALLOC_FL_KEEP_SIZE 0
|
2018-04-12 09:19:09 +00:00
|
|
|
assert_zeroes 0
|
2017-05-30 14:17:21 +00:00
|
|
|
|
|
|
|
# unaligned blkdev_issue_discard
|
|
|
|
allocate
|
|
|
|
py_blkdiscard $((OBJECT_SIZE / 2))
|
2019-03-25 19:58:00 +00:00
|
|
|
assert_zeroes_unaligned $NUM_OBJECTS
|
2017-05-30 14:17:21 +00:00
|
|
|
|
|
|
|
# unaligned blkdev_issue_zeroout w/ BLKDEV_ZERO_NOUNMAP
|
|
|
|
allocate
|
|
|
|
py_fallocate FALLOC_FL_ZERO_RANGE\|FALLOC_FL_KEEP_SIZE $((OBJECT_SIZE / 2))
|
2018-04-12 09:19:09 +00:00
|
|
|
assert_zeroes_unaligned $NUM_OBJECTS
|
2017-05-30 14:17:21 +00:00
|
|
|
|
|
|
|
# unaligned blkdev_issue_zeroout w/ BLKDEV_ZERO_NOFALLBACK
|
|
|
|
allocate
|
|
|
|
py_fallocate FALLOC_FL_PUNCH_HOLE\|FALLOC_FL_KEEP_SIZE $((OBJECT_SIZE / 2))
|
2018-04-12 09:19:09 +00:00
|
|
|
assert_zeroes_unaligned $NUM_OBJECTS
|
2017-05-30 14:17:21 +00:00
|
|
|
|
|
|
|
sudo rbd unmap $DEV
|
|
|
|
|
2018-04-12 09:19:09 +00:00
|
|
|
DEV=$(sudo rbd map -o notrim $IMAGE_NAME)
|
|
|
|
|
|
|
|
# blkdev_issue_discard
|
|
|
|
allocate
|
|
|
|
py_blkdiscard 0 |& grep 'Operation not supported'
|
|
|
|
assert_allocated
|
|
|
|
|
|
|
|
# blkdev_issue_zeroout w/ BLKDEV_ZERO_NOUNMAP
|
|
|
|
allocate
|
|
|
|
py_fallocate FALLOC_FL_ZERO_RANGE\|FALLOC_FL_KEEP_SIZE 0
|
|
|
|
assert_zeroes $NUM_OBJECTS
|
|
|
|
|
|
|
|
# blkdev_issue_zeroout w/ BLKDEV_ZERO_NOFALLBACK
|
|
|
|
allocate
|
|
|
|
py_fallocate FALLOC_FL_PUNCH_HOLE\|FALLOC_FL_KEEP_SIZE 0 |& grep 'Operation not supported'
|
|
|
|
assert_allocated
|
|
|
|
|
|
|
|
sudo rbd unmap $DEV
|
|
|
|
|
2017-05-30 14:17:21 +00:00
|
|
|
echo OK
|