1 /*
2 * Copyright (C) Ricardo Salveti de Araujo, 2007
3 *
4 * This program is free software; you can redistribute it and/or modify it
5 * under the terms of version 2 of the GNU General Public License as
6 * published by the Free Software Foundation.
7 *
8 * This program is distributed in the hope that it would be useful, but
9 * WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
11 *
12 * Further, this software is distributed without any warranty that it is
13 * free of the rightful claim of any third person regarding infringement
14 * or the like. Any license provided herein, whether implied or
15 * otherwise, applies only to this software file. Patent licenses, if
16 * any, provided herein do not apply to combinations of this program with
17 * other software, or any other product whatsoever.
18 *
19 * You should have received a copy of the GNU General Public License along
20 * with this program; if not, write the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
22 */
23
24 /*
25 * NAME
26 * remap_file_pages02
27 *
28 * DESCRIPTION
29 * The remap_file_pages() system call is used to create a non-linear
30 * mapping, that is, a mapping in which the pages of the file are mapped
31 * into a non-sequential order in memory. The advantage of using
32 * remap_file_pages() over using repeated calls to mmap(2) is that
33 * the former approach does not require the kernel to create
34 * additional VMA (Virtual Memory Area) data structures.
35 *
36 * Runs remap_file_pages with wrong values and see if got the expected error
37 *
38 * Setup:
39 * 1. Global:
40 * 2. Create a file, do a normal mmap with MAP_SHARED flag
41 *
42 * Test:
43 * 1. Test with a valid mmap but without MAP_SHARED flag
44 * 2. Test with a invalid start argument
45 * 3. Test with a invalid size argument
46 * 4. Test with a invalid prot argument
47 *
48 * Cleanup:
49 * Remove the file and erase the tmp directory
50 *
51 * Usage: <for command-line>
52 * remap_file_pages02 [-c n] [-f] [-i n] [-I x] [-P x] [-t]
53 * where, -c n : Run n copies concurrently.
54 * -f : Turn off functionality Testing.
55 * -i n : Execute test n times.
56 * -I x : Execute test for x seconds.
57 * -P x : Pause for x seconds between iterations.
58 * -t : Turn on syscall timing.
59 *
60 * HISTORY
61 *
62 * 02/11/2008 - Removed the pgoff test case, as the latest kernels doesn't
63 * verify the page offset (http://lkml.org/lkml/2007/11/29/325) - Ricardo
64 * Salveti de Araujo, <rsalvetidev@gmail.com>
65 *
66 * 19/10/2007 - Created by Ricardo Salveti de Araujo, <rsalvetidev@gmail.com>
67 */
68
69 #define _GNU_SOURCE
70 #include <sys/mman.h>
71 #include <sys/types.h>
72 #include <sys/stat.h>
73 #include <fcntl.h>
74 #include <stdio.h>
75 #include <unistd.h>
76 #include <errno.h>
77 #include <syscall.h>
78 #include <linux/unistd.h>
79
80 #include "test.h" /*LTP Specific Include File */
81
82 /* Test case defines */
83 #define WINDOW_START 0x48000000
84
85 static int page_sz;
86 size_t page_words;
87 size_t cache_pages;
88 size_t cache_sz;
89 size_t window_pages;
90 size_t window_sz;
91
92 static void setup();
93 static int setup01(int test);
94 static int setup02(int test);
95 static int setup03(int test);
96 static int setup04(int test);
97 static void cleanup();
98
99 char *TCID = "remap_file_pages02";
100 int TST_TOTAL = 4;
101
102 static char *cache_contents;
103 int fd; /* File descriptor used at the test */
104 char *data = NULL;
105 char *data01 = NULL;
106
107 static struct test_case_t {
108 char *err_desc; /* Error description */
109 int exp_errno; /* Expected error number */
110 char *exp_errval; /* Expected error value string */
111 int (*setupfunc) (int); /* Test setup function */
112 int (*cleanfunc) (int); /* Test clean function */
113 void *start; /* Start argument */
114 size_t size; /* Size argument */
115 int prot; /* Prot argument */
116 ssize_t pgoff; /* Pgoff argument */
117 int flags; /* Flags argument */
118 } testcase[] = {
119 {
120 "start does not refer to a valid mapping created with the "
121 "MAP_SHARED flag", EINVAL, "EINVAL", setup01, NULL,
122 NULL, 0, 0, 2, 0}, {
123 "start is invalid", EINVAL, "EINVAL", setup02, NULL, NULL, 0, 0, 2, 0},
124 {
125 "size is invalid", EINVAL, "EINVAL", setup03, NULL, NULL, 0, 0, 0, 0},
126 {
127 "prot is invalid", EINVAL, "EINVAL", setup04, NULL, NULL, 0, 0,
128 2, 0}
129 };
130
main(int ac,char ** av)131 int main(int ac, char **av)
132 {
133 int lc, i;
134
135 #if defined (__s390__) || (__s390x__) || (__ia64__)
136 /* Disables the test in case the kernel version is lower than 2.6.12 and arch is s390 */
137 if ((tst_kvercmp(2, 6, 12)) < 0) {
138 tst_resm(TWARN,
139 "This test can only run on kernels that are 2.6.12 and higher");
140 exit(0);
141 }
142 #endif
143
144 tst_parse_opts(ac, av, NULL, NULL);
145
146 setup();
147
148 for (lc = 0; TEST_LOOPING(lc); lc++) {
149
150 tst_count = 0;
151
152 for (i = 0; i < TST_TOTAL; i++) {
153 /* do the setup if the test have one */
154 if (testcase[i].setupfunc
155 && testcase[i].setupfunc(i) == -1) {
156 tst_resm(TWARN,
157 "Failed to setup test %d"
158 " Skipping test", i);
159 continue;
160 }
161
162 /* run the test */
163 TEST(remap_file_pages
164 (testcase[i].start, testcase[i].size,
165 testcase[i].prot, testcase[i].pgoff,
166 testcase[i].flags));
167
168 /* do the cleanup if the test have one */
169 if (testcase[i].cleanfunc
170 && testcase[i].cleanfunc(i) == -1) {
171 tst_brkm(TBROK, cleanup,
172 "Failed to cleanup test %d,"
173 " quitting the test", i);
174 }
175
176 /* verify the return code */
177 if ((TEST_RETURN == -1)
178 && (TEST_ERRNO == testcase[i].exp_errno)) {
179 tst_resm(TPASS,
180 "remap_file_pages(2) expected failure;"
181 " Got errno - %s : %s",
182 testcase[i].exp_errval,
183 testcase[i].err_desc);
184 } else {
185 tst_resm(TFAIL,
186 "remap_file_pages(2) failed to produce"
187 " expected error: %d, errno: %s."
188 " because got error %d",
189 testcase[i].exp_errno,
190 testcase[i].exp_errval, TEST_ERRNO);
191 }
192 } /* end of test loops */
193 } /* end of test looping */
194
195 /* clean up and exit */
196 cleanup();
197
198 tst_exit();
199 }
200
201 /*
202 * setup01() - create a mmap area without MAP_SHARED flag
203 * - it uses the fd created at the main setup function
204 */
setup01(int test)205 int setup01(int test)
206 {
207 data01 = mmap(NULL, cache_sz, PROT_READ | PROT_WRITE,
208 MAP_PRIVATE, fd, 0);
209
210 if (data01 == MAP_FAILED) {
211 tst_resm(TWARN, "mmap Error, errno=%d : %s", errno,
212 strerror(errno));
213 return -1;
214 }
215
216 /* set up the test case struct for this test */
217 testcase[test].start = data01;
218 testcase[test].size = page_sz;
219
220 return 0;
221 }
222
223 /*
224 * setup02() - start is invalid
225 */
setup02(int test)226 int setup02(int test)
227 {
228 /* set up the test case struct for this test */
229 testcase[test].start = data + cache_sz;
230 testcase[test].size = page_sz;
231
232 return 0;
233 }
234
235 /*
236 * setup03() - size is invalid
237 */
setup03(int test)238 int setup03(int test)
239 {
240 /* set up the test case struct for this test */
241 testcase[test].start = data;
242 testcase[test].size = cache_sz + page_sz;
243
244 return 0;
245 }
246
247 /*
248 * setup04() - prot is invalid
249 */
setup04(int test)250 int setup04(int test)
251 {
252 /* set up the test case struct for this test */
253 testcase[test].start = data;
254 testcase[test].size = page_sz;
255 testcase[test].prot = -1;
256
257 return 0;
258 }
259
260 /*
261 * setup() - performs all ONE TIME setup for this test
262 * - creates a defaul mmaped area to be able to run remap_file_pages
263 */
setup(void)264 void setup(void)
265 {
266 int i, j;
267
268 tst_sig(FORK, DEF_HANDLER, cleanup);
269
270 TEST_PAUSE;
271
272 tst_tmpdir();
273
274 /* Get page size */
275 if ((page_sz = getpagesize()) < 0) {
276 tst_brkm(TFAIL, cleanup,
277 "getpagesize() fails to get system page size");
278 }
279
280 page_words = (page_sz / sizeof(char));
281
282 /* Set the cache size */
283 cache_pages = 32;
284 cache_sz = cache_pages * page_sz;
285 cache_contents = malloc(cache_sz * sizeof(char));
286
287 for (i = 0; i < cache_pages; i++) {
288 char *page = cache_contents + i * page_sz;
289
290 for (j = 0; j < page_words; j++)
291 page[j] = i;
292 }
293
294 if ((fd = open("cache", O_RDWR | O_CREAT | O_TRUNC, S_IRWXU)) < 0) {
295 tst_brkm(TBROK, cleanup,
296 "open(%s, O_RDWR|O_CREAT|O_TRUNC,S_IRWXU) Failed, errno=%d : %s",
297 "cache", errno, strerror(errno));
298 }
299
300 if (write(fd, cache_contents, cache_sz) != cache_sz) {
301 tst_resm(TFAIL,
302 "Write Error for \"cache_contents\" to \"cache_sz\" of %zu (errno=%d : %s)",
303 cache_sz, errno, strerror(errno));
304 cleanup();
305 }
306
307 data = mmap((void *)WINDOW_START,
308 cache_sz, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
309
310 if (data == MAP_FAILED) {
311 tst_resm(TFAIL, "mmap Error, errno=%d : %s", errno,
312 strerror(errno));
313 cleanup();
314 }
315
316 }
317
318 /*
319 * cleanup() - Performs one time cleanup for this test at
320 * completion or premature exit
321 */
cleanup(void)322 void cleanup(void)
323 {
324 /* Close the file descriptor */
325 close(fd);
326
327 if (data)
328 munmap(data, cache_sz);
329 if (data01)
330 munmap(data01, cache_sz);
331
332 tst_rmdir();
333
334 }
335