mirror of
https://git.proxmox.com/git/mirror_zfs.git
synced 2025-09-15 13:50:11 +03:00
ZTS: mmap_ftruncate test to confirm async writeback behaviour
Sponsored-by: Klara, Inc. Sponsored-by: Wasabi Technology, Inc. Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Reviewed-by: Alexander Motin <alexander.motin@TrueNAS.com> Signed-off-by: Rob Norris <rob.norris@klarasystems.com> Closes #17584
This commit is contained in:
parent
df5e02d253
commit
97fe86837c
@ -788,7 +788,7 @@ tags = ['functional', 'migration']
|
|||||||
|
|
||||||
[tests/functional/mmap]
|
[tests/functional/mmap]
|
||||||
tests = ['mmap_mixed', 'mmap_read_001_pos', 'mmap_seek_001_pos',
|
tests = ['mmap_mixed', 'mmap_read_001_pos', 'mmap_seek_001_pos',
|
||||||
'mmap_sync_001_pos', 'mmap_write_001_pos']
|
'mmap_sync_001_pos', 'mmap_write_001_pos', 'mmap_ftruncate']
|
||||||
tags = ['functional', 'mmap']
|
tags = ['functional', 'mmap']
|
||||||
|
|
||||||
[tests/functional/mount]
|
[tests/functional/mount]
|
||||||
|
1
tests/zfs-tests/cmd/.gitignore
vendored
1
tests/zfs-tests/cmd/.gitignore
vendored
@ -23,6 +23,7 @@
|
|||||||
/mkfiles
|
/mkfiles
|
||||||
/mktree
|
/mktree
|
||||||
/mmap_exec
|
/mmap_exec
|
||||||
|
/mmap_ftruncate
|
||||||
/mmap_libaio
|
/mmap_libaio
|
||||||
/mmap_seek
|
/mmap_seek
|
||||||
/mmap_sync
|
/mmap_sync
|
||||||
|
@ -72,7 +72,9 @@ scripts_zfs_tests_bin_PROGRAMS += %D%/mkbusy %D%/mkfile %D%/mkfiles %D%/mktree
|
|||||||
%C%_mkfile_LDADD = $(LTLIBINTL)
|
%C%_mkfile_LDADD = $(LTLIBINTL)
|
||||||
|
|
||||||
|
|
||||||
scripts_zfs_tests_bin_PROGRAMS += %D%/mmap_exec %D%/mmap_seek %D%/mmap_sync %D%/mmapwrite %D%/readmmap
|
scripts_zfs_tests_bin_PROGRAMS += \
|
||||||
|
%D%/mmap_exec %D%/mmap_ftruncate %D%/mmap_seek \
|
||||||
|
%D%/mmap_sync %D%/mmapwrite %D%/readmmap
|
||||||
%C%_mmapwrite_LDADD = -lpthread
|
%C%_mmapwrite_LDADD = -lpthread
|
||||||
|
|
||||||
if WANT_MMAP_LIBAIO
|
if WANT_MMAP_LIBAIO
|
||||||
|
85
tests/zfs-tests/cmd/mmap_ftruncate.c
Normal file
85
tests/zfs-tests/cmd/mmap_ftruncate.c
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
// SPDX-License-Identifier: CDDL-1.0
|
||||||
|
/*
|
||||||
|
* CDDL HEADER START
|
||||||
|
*
|
||||||
|
* The contents of this file are subject to the terms of the
|
||||||
|
* Common Development and Distribution License (the "License").
|
||||||
|
* You may not use this file except in compliance with the License.
|
||||||
|
*
|
||||||
|
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
|
||||||
|
* or http://opensource.org/licenses/CDDL-1.0.
|
||||||
|
* See the License for the specific language governing permissions
|
||||||
|
* and limitations under the License.
|
||||||
|
*
|
||||||
|
* When distributing Covered Code, include this CDDL HEADER in each
|
||||||
|
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
|
||||||
|
* If applicable, add the following below this CDDL HEADER, with the
|
||||||
|
* fields enclosed by brackets "[]" replaced with your own identifying
|
||||||
|
* information: Portions Copyright [yyyy] [name of copyright owner]
|
||||||
|
*
|
||||||
|
* CDDL HEADER END
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2025, Klara, Inc.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Tests async writeback behaviour. Creates a file, maps it into memory, and
|
||||||
|
* dirties every page within it. Then, calls ftruncate() to collapse the file
|
||||||
|
* back down to 0. This causes the kernel to begin writeback on the dirty
|
||||||
|
* pages so they can be freed, before it can complete the ftruncate() call.
|
||||||
|
* None of these are sync operations, so they should avoid the various "force
|
||||||
|
* flush" codepaths.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/mman.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#define _pdfail(f, l, s) \
|
||||||
|
do { perror("[" f "#" #l "] " s); exit(2); } while (0)
|
||||||
|
#define pdfail(str) _pdfail(__FILE__, __LINE__, str)
|
||||||
|
|
||||||
|
int
|
||||||
|
main(int argc, char **argv) {
|
||||||
|
if (argc != 3) {
|
||||||
|
printf("usage: mmap_ftruncate <file> <size>\n");
|
||||||
|
exit(2);
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *file = argv[1];
|
||||||
|
|
||||||
|
char *end;
|
||||||
|
off_t sz = strtoull(argv[2], &end, 0);
|
||||||
|
if (end == argv[2] || *end != '\0' || sz == 0) {
|
||||||
|
fprintf(stderr, "E: invalid size");
|
||||||
|
exit(2);
|
||||||
|
}
|
||||||
|
|
||||||
|
int fd = open(file, O_CREAT|O_TRUNC|O_RDWR, S_IRUSR|S_IWUSR);
|
||||||
|
if (fd < 0)
|
||||||
|
pdfail("open");
|
||||||
|
|
||||||
|
if (ftruncate(fd, sz) < 0)
|
||||||
|
pdfail("ftruncate");
|
||||||
|
|
||||||
|
char *p = mmap(NULL, sz, PROT_WRITE, MAP_SHARED, fd, 0);
|
||||||
|
if (p == MAP_FAILED)
|
||||||
|
pdfail("mmap");
|
||||||
|
|
||||||
|
for (off_t off = 0; off < sz; off += 4096)
|
||||||
|
p[off] = 1;
|
||||||
|
|
||||||
|
if (ftruncate(fd, 0) < 0)
|
||||||
|
pdfail("ftruncate");
|
||||||
|
|
||||||
|
if (munmap(p, sz) < 0)
|
||||||
|
pdfail("munmap");
|
||||||
|
|
||||||
|
close(fd);
|
||||||
|
return (0);
|
||||||
|
}
|
@ -205,6 +205,7 @@ export ZFSTEST_FILES='badsend
|
|||||||
mkfiles
|
mkfiles
|
||||||
mktree
|
mktree
|
||||||
mmap_exec
|
mmap_exec
|
||||||
|
mmap_ftruncate
|
||||||
mmap_libaio
|
mmap_libaio
|
||||||
mmap_seek
|
mmap_seek
|
||||||
mmap_sync
|
mmap_sync
|
||||||
|
@ -1660,6 +1660,7 @@ nobase_dist_datadir_zfs_tests_tests_SCRIPTS += \
|
|||||||
functional/mmap/mmap_seek_001_pos.ksh \
|
functional/mmap/mmap_seek_001_pos.ksh \
|
||||||
functional/mmap/mmap_sync_001_pos.ksh \
|
functional/mmap/mmap_sync_001_pos.ksh \
|
||||||
functional/mmap/mmap_write_001_pos.ksh \
|
functional/mmap/mmap_write_001_pos.ksh \
|
||||||
|
functional/mmap/mmap_ftruncate.ksh \
|
||||||
functional/mmap/setup.ksh \
|
functional/mmap/setup.ksh \
|
||||||
functional/mmp/cleanup.ksh \
|
functional/mmp/cleanup.ksh \
|
||||||
functional/mmp/mmp_active_import.ksh \
|
functional/mmp/mmp_active_import.ksh \
|
||||||
|
80
tests/zfs-tests/tests/functional/mmap/mmap_ftruncate.ksh
Executable file
80
tests/zfs-tests/tests/functional/mmap/mmap_ftruncate.ksh
Executable file
@ -0,0 +1,80 @@
|
|||||||
|
#!/bin/ksh -p
|
||||||
|
# SPDX-License-Identifier: CDDL-1.0
|
||||||
|
#
|
||||||
|
# CDDL HEADER START
|
||||||
|
#
|
||||||
|
# The contents of this file are subject to the terms of the
|
||||||
|
# Common Development and Distribution License (the "License").
|
||||||
|
# You may not use this file except in compliance with the License.
|
||||||
|
#
|
||||||
|
# You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
|
||||||
|
# or https://opensource.org/licenses/CDDL-1.0.
|
||||||
|
# See the License for the specific language governing permissions
|
||||||
|
# and limitations under the License.
|
||||||
|
#
|
||||||
|
# When distributing Covered Code, include this CDDL HEADER in each
|
||||||
|
# file and include the License file at usr/src/OPENSOLARIS.LICENSE.
|
||||||
|
# If applicable, add the following below this CDDL HEADER, with the
|
||||||
|
# fields enclosed by brackets "[]" replaced with your own identifying
|
||||||
|
# information: Portions Copyright [yyyy] [name of copyright owner]
|
||||||
|
#
|
||||||
|
# CDDL HEADER END
|
||||||
|
#
|
||||||
|
|
||||||
|
#
|
||||||
|
# Copyright (c) 2025, Klara, Inc.
|
||||||
|
#
|
||||||
|
|
||||||
|
. $STF_SUITE/include/libtest.shlib
|
||||||
|
|
||||||
|
#
|
||||||
|
# This verifies that async writeback of dirty mmap()'d pages completes quickly.
|
||||||
|
# ftruncate() is an operation that will trigger async writeback, but is not
|
||||||
|
# itself a syncing operation, making it a useful proxy for any way the kernel
|
||||||
|
# might trigger async writeback.
|
||||||
|
#
|
||||||
|
# The guts of this test is in the mmap_ftruncate program. This driver sets a
|
||||||
|
# larger zfs_txg_timeout. Test failure occurs ftruncate() blocks waiting for
|
||||||
|
# the writeback until the txg timeout is reached and the changes are forcibly
|
||||||
|
# written out. Success means the DMU has accepted the changes and cleared the
|
||||||
|
# page dirty flags.
|
||||||
|
#
|
||||||
|
|
||||||
|
TIMEOUT=180
|
||||||
|
TESTFILE=/$TESTPOOL/truncfile
|
||||||
|
TESTSIZE=$((2*1024*1024*1024)) # 2G
|
||||||
|
|
||||||
|
verify_runnable "global"
|
||||||
|
|
||||||
|
typeset claim="async writeback of dirty mmap()'d pages completes quickly"
|
||||||
|
|
||||||
|
log_assert $claim
|
||||||
|
|
||||||
|
log_must save_tunable TXG_TIMEOUT
|
||||||
|
|
||||||
|
function cleanup
|
||||||
|
{
|
||||||
|
log_must restore_tunable TXG_TIMEOUT
|
||||||
|
rm -f $TESTFILE
|
||||||
|
}
|
||||||
|
log_onexit cleanup
|
||||||
|
|
||||||
|
log_must set_tunable32 TXG_TIMEOUT $TIMEOUT
|
||||||
|
log_must zpool sync -f
|
||||||
|
|
||||||
|
# run mmap_ftruncate and record the run time
|
||||||
|
typeset -i start=$(date +%s)
|
||||||
|
log_must mmap_ftruncate $TESTFILE $TESTSIZE
|
||||||
|
typeset -i end=$(date +%s)
|
||||||
|
typeset -i delta=$((end - start))
|
||||||
|
|
||||||
|
# in practice, mmap_ftruncate needs a few seconds to dirty all the pages, and
|
||||||
|
# when this test passes, the ftruncate() call itself should be near-instant.
|
||||||
|
# when it fails, then its only the txg sync that allows ftruncate() to
|
||||||
|
# complete, in that case, the run time will be extremely close to the timeout,
|
||||||
|
# so to avoid any confusion at the edges, we require that it complets within
|
||||||
|
# half the transaction time. for any timeout higher than ~30s that should be a
|
||||||
|
# very bright line down the middle.
|
||||||
|
log_must test $delta -lt $((TIMEOUT / 2))
|
||||||
|
|
||||||
|
log_pass $claim
|
Loading…
Reference in New Issue
Block a user