/* SPDX-License-Identifier: MIT */ /* * Description: basic madvise test */ #include #include #include #include #include #include #include #include #include #include "helpers.h" #include "liburing.h" #define FILE_SIZE (128 * 1024) #define LOOPS 100 #define MIN_LOOPS 10 static unsigned long long utime_since(const struct timeval *s, const struct timeval *e) { long long sec, usec; sec = e->tv_sec - s->tv_sec; usec = (e->tv_usec - s->tv_usec); if (sec > 0 && usec < 0) { sec--; usec += 1000000; } sec *= 1000000; return sec + usec; } static unsigned long long utime_since_now(struct timeval *tv) { struct timeval end; gettimeofday(&end, NULL); return utime_since(tv, &end); } static int do_madvise(struct io_uring *ring, void *addr, off_t len, int advice) { struct io_uring_sqe *sqe; struct io_uring_cqe *cqe; int ret; sqe = io_uring_get_sqe(ring); if (!sqe) { fprintf(stderr, "failed to get sqe\n"); return 1; } io_uring_prep_madvise(sqe, addr, len, advice); sqe->user_data = advice; ret = io_uring_submit_and_wait(ring, 1); if (ret != 1) { fprintf(stderr, "submit: %d\n", ret); return 1; } ret = io_uring_wait_cqe(ring, &cqe); if (ret) { fprintf(stderr, "wait: %d\n", ret); return 1; } ret = cqe->res; if (ret == -EINVAL || ret == -EBADF) { fprintf(stdout, "Madvise not supported, skipping\n"); unlink(".madvise.tmp"); exit(0); } else if (ret) { fprintf(stderr, "cqe->res=%d\n", cqe->res); } io_uring_cqe_seen(ring, cqe); return ret; } static long do_copy(int fd, char *buf, void *ptr) { struct timeval tv; gettimeofday(&tv, NULL); memcpy(buf, ptr, FILE_SIZE); return utime_since_now(&tv); } static int test_madvise(struct io_uring *ring, const char *filename) { unsigned long cached_read, uncached_read, cached_read2; int fd, ret; char *buf; void *ptr; fd = open(filename, O_RDONLY); if (fd < 0) { perror("open"); return 1; } buf = t_malloc(FILE_SIZE); ptr = mmap(NULL, FILE_SIZE, PROT_READ, MAP_PRIVATE, fd, 0); if (ptr == MAP_FAILED) { perror("mmap"); return 1; } cached_read = do_copy(fd, buf, ptr); if (cached_read == -1) return 1; cached_read = do_copy(fd, buf, ptr); if (cached_read == -1) return 1; ret = do_madvise(ring, ptr, FILE_SIZE, MADV_DONTNEED); if (ret) return 1; uncached_read = do_copy(fd, buf, ptr); if (uncached_read == -1) return 1; ret = do_madvise(ring, ptr, FILE_SIZE, MADV_DONTNEED); if (ret) return 1; ret = do_madvise(ring, ptr, FILE_SIZE, MADV_WILLNEED); if (ret) return 1; msync(ptr, FILE_SIZE, MS_SYNC); cached_read2 = do_copy(fd, buf, ptr); if (cached_read2 == -1) return 1; if (cached_read < uncached_read && cached_read2 < uncached_read) return 0; return 2; } int main(int argc, char *argv[]) { struct io_uring ring; int ret, i, good, bad; char *fname; if (argc > 1) { fname = argv[1]; } else { fname = ".madvise.tmp"; t_create_file(fname, FILE_SIZE); } if (io_uring_queue_init(8, &ring, 0)) { fprintf(stderr, "ring creation failed\n"); goto err; } good = bad = 0; for (i = 0; i < LOOPS; i++) { ret = test_madvise(&ring, fname); if (ret == 1) { fprintf(stderr, "test_madvise failed\n"); goto err; } else if (!ret) good++; else if (ret == 2) bad++; if (i >= MIN_LOOPS && !bad) break; } /* too hard to reliably test, just ignore */ if ((0) && bad > good) fprintf(stderr, "Suspicious timings (%u > %u)\n", bad, good); if (fname != argv[1]) unlink(fname); io_uring_queue_exit(&ring); return T_EXIT_PASS; err: if (fname != argv[1]) unlink(fname); return T_EXIT_FAIL; }