1 /* Basic sync_file_range (not specific flag is checked).
2 Copyright (C) 2016-2022 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
9
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
14
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, see
17 <https://www.gnu.org/licenses/>. */
18
19 /* sync_file_range is only define for LFS. */
20 #define _FILE_OFFSET_BITS 64
21 #include <errno.h>
22 #include <fcntl.h>
23 #include <limits.h>
24 #include <stdint.h>
25 #include <stdlib.h>
26 #include <sys/stat.h>
27
28 #include <support/temp_file.h>
29 #include <support/check.h>
30
31 #define XSTR(s) STR(S)
32 #define STR(s) #s
33
34 static char *temp_filename;
35 static int temp_fd;
36
37 static char fifoname[] = "/tmp/tst-posix_fadvise-fifo-XXXXXX";
38 static int fifofd;
39
40 void
do_prepare(int argc,char ** argv)41 do_prepare (int argc, char **argv)
42 {
43 temp_fd = create_temp_file ("tst-file_sync_range.", &temp_filename);
44 if (temp_fd == -1)
45 FAIL_EXIT1 ("cannot create temporary file: %m");
46
47 if (mktemp (fifoname) == NULL)
48 FAIL_EXIT1 ("cannot generate temp file name: %m");
49 add_temp_file (fifoname);
50
51 if (mkfifo (fifoname, S_IWUSR | S_IRUSR) != 0)
52 FAIL_EXIT1 ("cannot create fifo: %m");
53
54 fifofd = open (fifoname, O_RDONLY | O_NONBLOCK);
55 if (fifofd == -1)
56 FAIL_EXIT1 ("cannot open fifo: %m");
57 }
58 #define PREPARE do_prepare
59
60 static int
do_test(void)61 do_test (void)
62 {
63 int ret;
64
65 /* This tests first check for some invalid usage and then check for
66 a simple usage. It does not cover for all possible issue since for
67 EIO/ENOMEM/ENOSPC would require to create very specific scenarios that
68 are outside the current test coverage (basically correct kernel argument
69 passing. */
70
71 /* Check for invalid file descriptor. */
72 if ((ret = sync_file_range (-1, 0, 0, 0)) != -1)
73 FAIL_EXIT1 ("sync_file_range did not fail on an invalid descriptor "
74 "(returned %d, expected -1)", ret);
75 if (errno != EBADF)
76 FAIL_EXIT1 ("sync_file_range on an invalid descriptor did not set errno to "
77 "EBADF (%d)", errno);
78
79 if ((ret = sync_file_range (fifofd, 0, 0, 0)) != -1)
80 FAIL_EXIT1 ("sync_file_range did not fail on an invalid descriptor "
81 "(returned %d, expected -1)", ret);
82 if (errno != ESPIPE)
83 FAIL_EXIT1 ("sync_file_range on an invalid descriptor did not set errno to "
84 "EBADF (%d)", errno);
85
86 /* Check for invalid flags (it must be
87 SYNC_FILE_RANGE_{WAIT_BEFORE,WRITE,WAIT_AFTER) or a 'or' combination of
88 them. */
89 if ((ret = sync_file_range (temp_fd, 0, 0, -1)) != -1)
90 FAIL_EXIT1 ("sync_file_range did not failed with invalid flags "
91 "(returned %d, " "expected -1)", ret);
92 if (errno != EINVAL)
93 FAIL_EXIT1 ("sync_file_range with invalid flag did not set errno to "
94 "EINVAL (%d)", errno);
95
96 /* Check for negative offset. */
97 if ((ret = sync_file_range (temp_fd, -1, 1, 0)) != -1)
98 FAIL_EXIT1 ("sync_file_range did not failed with invalid offset "
99 "(returned %d, expected -1)", ret);
100 if (errno != EINVAL)
101 FAIL_EXIT1 ("sync_file_range with invalid offset did not set errno to "
102 "EINVAL (%d)", errno);
103
104 /* offset + nbytes must be a positive value. */
105 if ((ret = sync_file_range (temp_fd, 1024, -2048, 0)) != -1)
106 FAIL_EXIT1 ("sync_file_range did not failed with invalid nbytes (returned %d, "
107 "expected -1)", ret);
108 if (errno != EINVAL)
109 FAIL_EXIT1 ("sync_file_range with invalid offset did not set errno to "
110 "EINVAL (%d)", errno);
111
112 /* offset + nbytes must be larger or equal than offset */
113 if ((ret = sync_file_range (temp_fd, -1024, 1024, 0)) != -1)
114 FAIL_EXIT1 ("sync_file_range did not failed with invalid offset "
115 "(returned %d, expected -1)", ret);
116 if (errno != EINVAL)
117 FAIL_EXIT1 ("sync_file_range with invalid offset did not set errno to "
118 "EINVAL (%d)", errno);
119
120 /* Check simple successful case. */
121 if ((ret = sync_file_range (temp_fd, 0, 1024, 0)) == -1)
122 FAIL_EXIT1 ("sync_file_range failed (errno = %d)", errno);
123
124 /* Finally check also a successful case with a 64-bit offset. */
125 off_t large_offset = UINT32_MAX + 2048LL;
126 if ((ret = sync_file_range (temp_fd, large_offset, 1024, 0)) == -1)
127 FAIL_EXIT1 ("sync_file_range failed (errno = %d)", errno);
128
129 return 0;
130 }
131
132 #include <support/test-driver.c>
133