| // SPDX-License-Identifier: GPL-2.0 |
| /* |
| * Copyright (c) 2022 SUSE Linux Products GmbH. All Rights Reserved. |
| */ |
| |
| /* |
| * Create a file with 4 extents, each with a size matching the page size. |
| * Then allocate a buffer to read all extents with io_uring, using O_DIRECT and |
| * IOCB_NOWAIT. Before doing the read with io_uring, access the first page of |
| * the read buffer to fault it in, so that during the read we only trigger page |
| * faults when accessing the other pages (mapping to 2nd, 3rd and 4th extents). |
| */ |
| |
| /* Get the O_DIRECT definition. */ |
| #ifndef _GNU_SOURCE |
| #define _GNU_SOURCE |
| #endif |
| |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <unistd.h> |
| #include <fcntl.h> |
| #include <errno.h> |
| #include <string.h> |
| #include <liburing.h> |
| |
| int main(int argc, char *argv[]) |
| { |
| struct io_uring ring; |
| struct io_uring_sqe *sqe; |
| struct io_uring_cqe *cqe; |
| struct iovec iovec; |
| int fd; |
| long pagesize; |
| void *write_buf; |
| void *read_buf; |
| ssize_t ret; |
| int i; |
| |
| if (argc != 2) { |
| fprintf(stderr, "Use: %s <file path>\n", argv[0]); |
| return 1; |
| } |
| |
| fd = open(argv[1], O_CREAT | O_TRUNC | O_WRONLY | O_DIRECT, 0666); |
| if (fd == -1) { |
| fprintf(stderr, "Failed to create file %s: %s (errno %d)\n", |
| argv[1], strerror(errno), errno); |
| return 1; |
| } |
| |
| pagesize = sysconf(_SC_PAGE_SIZE); |
| if (pagesize == -1) { |
| fprintf(stderr, "Failed to get page size: %s (errno %d)\n", |
| strerror(errno), errno); |
| return 1; |
| } |
| |
| ret = posix_memalign(&write_buf, pagesize, 4 * pagesize); |
| if (ret) { |
| fprintf(stderr, "Failed to allocate write buffer\n"); |
| return 1; |
| } |
| |
| memset(write_buf, 0xab, pagesize); |
| memset(write_buf + pagesize, 0xcd, pagesize); |
| memset(write_buf + 2 * pagesize, 0xef, pagesize); |
| memset(write_buf + 3 * pagesize, 0x73, pagesize); |
| |
| /* Create the 4 extents, each with a size matching page size. */ |
| for (i = 0; i < 4; i++) { |
| ret = pwrite(fd, write_buf + i * pagesize, pagesize, |
| i * pagesize); |
| if (ret != pagesize) { |
| fprintf(stderr, |
| "Write failure, ret = %ld errno %d (%s)\n", |
| ret, errno, strerror(errno)); |
| return 1; |
| } |
| ret = fsync(fd); |
| if (ret != 0) { |
| fprintf(stderr, "Fsync failure: %s (errno %d)\n", |
| strerror(errno), errno); |
| return 1; |
| } |
| } |
| |
| close(fd); |
| fd = open(argv[1], O_RDONLY | O_DIRECT); |
| if (fd == -1) { |
| fprintf(stderr, |
| "Failed to open file %s: %s (errno %d)", |
| argv[1], strerror(errno), errno); |
| return 1; |
| } |
| |
| ret = posix_memalign(&read_buf, pagesize, 4 * pagesize); |
| if (ret) { |
| fprintf(stderr, "Failed to allocate read buffer\n"); |
| return 1; |
| } |
| |
| /* |
| * Fault in only the first page of the read buffer. |
| * We want to trigger a page fault for the 2nd page of the read buffer |
| * during the read operation with io_uring (O_DIRECT and IOCB_NOWAIT). |
| */ |
| memset(read_buf, 0, 1); |
| |
| ret = io_uring_queue_init(1, &ring, 0); |
| if (ret != 0) { |
| fprintf(stderr, "Failed to creating io_uring queue\n"); |
| return 1; |
| } |
| |
| sqe = io_uring_get_sqe(&ring); |
| if (!sqe) { |
| fprintf(stderr, "Failed to get io_uring sqe\n"); |
| return 1; |
| } |
| |
| iovec.iov_base = read_buf; |
| iovec.iov_len = 4 * pagesize; |
| io_uring_prep_readv(sqe, fd, &iovec, 1, 0); |
| |
| ret = io_uring_submit_and_wait(&ring, 1); |
| if (ret != 1) { |
| fprintf(stderr, "Failed at io_uring_submit_and_wait()\n"); |
| return 1; |
| } |
| |
| ret = io_uring_wait_cqe(&ring, &cqe); |
| if (ret < 0) { |
| fprintf(stderr, "Failed at io_uring_wait_cqe()\n"); |
| return 1; |
| } |
| |
| if (cqe->res != (4 * pagesize)) |
| fprintf(stderr, "Unexpected cqe->res, got %d expected %ld\n", |
| cqe->res, 4 * pagesize); |
| |
| if (memcmp(read_buf, write_buf, 4 * pagesize) != 0) |
| fprintf(stderr, |
| "Unexpected mismatch between read and write buffers\n"); |
| |
| io_uring_cqe_seen(&ring, cqe); |
| io_uring_queue_exit(&ring); |
| |
| return 0; |
| } |