From: Bernd Schubert Date: Mon, 26 Feb 2024 18:09:00 +0000 (+0100) Subject: Fix tests/test_write_cache in write back mode (#892) X-Git-Tag: fuse-3.17.1-rc0~150 X-Git-Url: http://git.maquefel.me/?a=commitdiff_plain;h=31bf17c7448944a04d4c84df534013a4d6924acc;p=qemu-gpiodev%2Flibfuse.git Fix tests/test_write_cache in write back mode (#892) This test could fail whenever a something (kernel, userspace) decides to flush in between of two 2048B writes. These two writes are supposed to be merged into a single 4906 byte write by the kernel writeback cache, but _sometimes_ the test fails because 2048 byte writes get through. Fixes #882 Solution here is a modification how the test works - instead of requiring an exact aggregation of 2x2048B into 4096B, it now writes 64x2048B and requires in write-back modes the number of received writes requests is lower than 64 - we can expect that at least some writes get aggregated, but we do know how many. Co-authored-by: Bernd Schubert --- diff --git a/test/test_write_cache.c b/test/test_write_cache.c index e0bdeb9..92471f1 100644 --- a/test/test_write_cache.c +++ b/test/test_write_cache.c @@ -21,6 +21,7 @@ #include #include #include +#include #ifndef __linux__ #include @@ -42,6 +43,8 @@ struct options { .delay_ms = 0, }; +#define WRITE_SYSCALLS 64 + #define OPTION(t, p) \ { t, offsetof(struct options, p), 1 } static const struct fuse_opt option_spec[] = { @@ -51,6 +54,7 @@ static const struct fuse_opt option_spec[] = { FUSE_OPT_END }; static int got_write; +static atomic_int write_cnt; pthread_cond_t cond = PTHREAD_COND_INITIALIZER; pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER; @@ -142,11 +146,13 @@ static void tfs_write(fuse_req_t req, fuse_ino_t ino, const char *buf, if(options.writeback) expected *= 2; - if(size != expected) - fprintf(stderr, "ERROR: Expected %zd bytes, got %zd\n!", - expected, size); - else - got_write = 1; + write_cnt++; + + if(size != expected && !options.writeback) + fprintf(stderr, "ERROR: Expected %zd bytes, got %zd\n!", + expected, size); + else + got_write = 1; /* Simulate waiting for pending writes */ if (options.delay_ms) { @@ -202,9 +208,11 @@ static void* run_fs(void *data) { static void test_fs(char *mountpoint) { char fname[PATH_MAX]; char *buf; - size_t dsize = options.data_size; + const size_t iosize = options.data_size; + const size_t dsize = options.data_size * WRITE_SYSCALLS; int fd, rofd; pthread_t rofd_thread; + loff_t off = 0; buf = malloc(dsize); assert(buf != NULL); @@ -228,8 +236,11 @@ static void test_fs(char *mountpoint) { usleep(options.delay_ms * 1000); } - assert(write(fd, buf, dsize) == dsize); - assert(write(fd, buf, dsize) == dsize); + for (int cnt = 0; cnt < WRITE_SYSCALLS; cnt++) { + assert(pwrite(fd, buf + off, iosize, off) == iosize); + off += iosize; + assert(off <= dsize); + } free(buf); close(fd); @@ -270,6 +281,20 @@ int main(int argc, char *argv[]) { assert(pthread_join(fs_thread, NULL) == 0); assert(got_write == 1); + + /* + * when writeback cache is enabled, kernel side can merge requests, but + * memory pressure, system 'sync' might trigger data flushes before - flush + * might happen in between write syscalls - merging subpage writes into + * a single page and pages into large fuse requests might or might not work. + * Though we can expect that that at least some (but maybe all) write + * system calls can be merged. + */ + if (options.writeback) + assert(write_cnt < WRITE_SYSCALLS); + else + assert(write_cnt == WRITE_SYSCALLS); + fuse_remove_signal_handlers(se); fuse_session_destroy(se);