The following changes since commit 625b155dcd3d56595ced60806e091126446c1e08: examples: cmdprio_bssplit: add CDL example (2024-01-27 11:18:57 -0500) are available in the Git repository at: git://git.kernel.dk/fio.git master for you to fetch changes up to 12067650d11d4777dee0cd64a136923c2fd2d073: t/zbd: add -s option to test-zbd-support script (2024-02-07 08:43:13 -0500) ---------------------------------------------------------------- Dmitry Fomichev (5): zbd: avoid assertions during sequential read I/O oslib: log BLKREPORTZONE error code zbd: use a helper to calculate zone index t/zbd: check device for unrestricted read support t/zbd: add -s option to test-zbd-support script oslib/linux-blkzoned.c | 2 ++ t/zbd/functions | 22 ++++++++++++++++++++++ t/zbd/test-zbd-support | 20 ++++++++++++++++++-- zbd.c | 21 +++++++++++++++------ 4 files changed, 57 insertions(+), 8 deletions(-) --- Diff of recent changes: diff --git a/oslib/linux-blkzoned.c b/oslib/linux-blkzoned.c index 2c3ecf33..1cc8d288 100644 --- a/oslib/linux-blkzoned.c +++ b/oslib/linux-blkzoned.c @@ -242,6 +242,8 @@ int blkzoned_report_zones(struct thread_data *td, struct fio_file *f, hdr->sector = offset >> 9; ret = ioctl(fd, BLKREPORTZONE, hdr); if (ret) { + log_err("%s: BLKREPORTZONE ioctl failed, ret=%d, err=%d.\n", + f->file_name, ret, -errno); ret = -errno; goto out; } diff --git a/t/zbd/functions b/t/zbd/functions index 028df404..7734371e 100644 --- a/t/zbd/functions +++ b/t/zbd/functions @@ -290,6 +290,28 @@ min_seq_write_size() { fi } +urswrz() { + local dev=$1 + + if [ -n "${sg_inq}" ] && [ ! -n "${use_libzbc}" ]; then + if ! ${sg_inq} -e --page=0xB6 --len=10 --hex "$dev" \ + > /dev/null 2>&1; then + # Couldn't get URSWRZ bit. Assume the reads are unrestricted + # because this configuration is more common. + echo 1 + else + ${sg_inq} -e --page=0xB6 --len=10 --hex "$dev" | tail -1 | + { + read -r offset b0 b1 b2 b3 b4 trailer && \ + echo $(( $b4 & 0x01 )) || echo 0 + } + fi + else + ${zbc_info} "$dev" | + sed -n 's/^[[:blank:]].*Read commands are \(un\)restricted*/\1/p' | grep -q ^ && echo 1 || echo 0 + fi +} + is_zbc() { local dev=$1 diff --git a/t/zbd/test-zbd-support b/t/zbd/test-zbd-support index 532860eb..c27d2ad6 100755 --- a/t/zbd/test-zbd-support +++ b/t/zbd/test-zbd-support @@ -15,6 +15,7 @@ usage() { echo -e "\t-w Reset all zones before executing each write test case" echo -e "\t-o <max_open_zones> Run fio with max_open_zones limit" echo -e "\t-t <test #> Run only a single test case with specified number" + echo -e "\t-s <test #> Start testing from the case with the specified number" echo -e "\t-q Quit the test run after any failed test" echo -e "\t-z Run fio with debug=zbd option" echo -e "\t-u Use io_uring ioengine in place of libaio" @@ -412,8 +413,16 @@ test4() { opts+=("--size=$size" "--thread=1" "--read_beyond_wp=1") opts+=("$(ioengine "psync")" "--rw=read" "--direct=1" "--disable_lat=1") opts+=("--zonemode=zbd" "--zonesize=${zone_size}") - run_fio "${opts[@]}" >> "${logfile}.${test_number}" 2>&1 || return $? - check_read $size || return $? + run_fio "${opts[@]}" >> "${logfile}.${test_number}" 2>&1 + fio_rc=$? + if [[ $unrestricted_reads != 0 ]]; then + if [[ $fio_rc != 0 ]]; then + return "$fio_rc" + fi + check_read $size || return $? + else + [ $fio_rc == 0 ] && return 1 || return 0 + fi } # Sequential write to sequential zones. @@ -1594,6 +1603,7 @@ zbd_debug= max_open_zones_opt= quit_on_err= force_io_uring= +start_test=1 while [ "${1#-}" != "$1" ]; do case "$1" in @@ -1607,6 +1617,7 @@ while [ "${1#-}" != "$1" ]; do -w) reset_before_write=1; shift;; -t) tests+=("$2"); shift; shift;; -o) max_open_zones_opt="${2}"; shift; shift;; + -s) start_test=$2; shift; shift;; -v) dynamic_analyzer=(valgrind "--read-var-info=yes"); shift;; -q) quit_on_err=1; shift;; @@ -1664,6 +1675,7 @@ if [[ -b "$realdev" ]]; then first_sequential_zone_sector=${result[0]} sectors_per_zone=${result[1]} zone_size=$((sectors_per_zone * 512)) + unrestricted_reads=$(urswrz "$dev") if ! max_open_zones=$(max_open_zones "$dev"); then echo "Failed to determine maximum number of open zones" exit 1 @@ -1681,9 +1693,11 @@ if [[ -b "$realdev" ]]; then sectors_per_zone=$((zone_size / 512)) max_open_zones=128 max_active_zones=0 + unrestricted_reads=1 set_io_scheduler "$basename" none || exit $? ;; esac + elif [[ -c "$realdev" ]]; then # For an SG node, we must have libzbc option specified if [[ ! -n "$use_libzbc" ]]; then @@ -1712,6 +1726,7 @@ elif [[ -c "$realdev" ]]; then first_sequential_zone_sector=${result[0]} sectors_per_zone=${result[1]} zone_size=$((sectors_per_zone * 512)) + unrestricted_reads=$(urswrz "$dev") if ! max_open_zones=$(max_open_zones "$dev"); then echo "Failed to determine maximum number of open zones" exit 1 @@ -1761,6 +1776,7 @@ trap 'intr=1' SIGINT ret=0 for test_number in "${tests[@]}"; do + [ "${test_number}" -lt "${start_test}" ] && continue rm -f "${logfile}.${test_number}" unset SKIP_REASON echo -n "Running test $(printf "%02d" $test_number) ... " diff --git a/zbd.c b/zbd.c index 61b5b688..37417660 100644 --- a/zbd.c +++ b/zbd.c @@ -104,8 +104,7 @@ static void zone_lock(struct thread_data *td, const struct fio_file *f, struct fio_zone_info *z) { #ifndef NDEBUG - struct zoned_block_device_info *zbd = f->zbd_info; - uint32_t const nz = z - zbd->zone_info; + unsigned int const nz = zbd_zone_idx(f, z); /* A thread should never lock zones outside its working area. */ assert(f->min_zone <= nz && nz < f->max_zone); assert(z->has_wp); @@ -674,9 +673,20 @@ static bool zbd_zone_align_file_sizes(struct thread_data *td, return false; } + if (td->o.td_ddir == TD_DDIR_READ) { + z = zbd_offset_to_zone(f, f->file_offset + f->io_size); + new_end = z->start; + if (f->file_offset + f->io_size > new_end) { + log_info("%s: rounded io_size from %"PRIu64" to %"PRIu64"\n", + f->file_name, f->io_size, + new_end - f->file_offset); + f->io_size = new_end - f->file_offset; + } + return true; + } + z = zbd_offset_to_zone(f, f->file_offset); - if ((f->file_offset != z->start) && - (td->o.td_ddir != TD_DDIR_READ)) { + if (f->file_offset != z->start) { new_offset = zbd_zone_end(z); if (new_offset >= f->file_offset + f->io_size) { log_info("%s: io_size must be at least one zone\n", @@ -692,8 +702,7 @@ static bool zbd_zone_align_file_sizes(struct thread_data *td, z = zbd_offset_to_zone(f, f->file_offset + f->io_size); new_end = z->start; - if ((td->o.td_ddir != TD_DDIR_READ) && - (f->file_offset + f->io_size != new_end)) { + if (f->file_offset + f->io_size != new_end) { if (new_end <= f->file_offset) { log_info("%s: io_size must be at least one zone\n", f->file_name);