t_io.c revision 313498
1/*	$NetBSD: t_io.c,v 1.16 2015/04/04 12:34:44 riastradh Exp $	*/
2
3/*-
4 * Copyright (c) 2010 The NetBSD Foundation, Inc.
5 * All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 *    notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 *    notice, this list of conditions and the following disclaimer in the
14 *    documentation and/or other materials provided with the distribution.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
17 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
18 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19 * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
20 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26 * POSSIBILITY OF SUCH DAMAGE.
27 */
28
29#include <sys/stat.h>
30#include <sys/statvfs.h>
31
32#include <atf-c.h>
33#include <fcntl.h>
34#include <libgen.h>
35#include <stdlib.h>
36#include <unistd.h>
37
38#include <rump/rump_syscalls.h>
39#include <rump/rump.h>
40
41#include "../common/h_fsmacros.h"
42#include "../../h_macros.h"
43
44#define TESTSTR "this is a string.  collect enough and you'll have Em"
45#define TESTSZ sizeof(TESTSTR)
46
47static void
48holywrite(const atf_tc_t *tc, const char *mp)
49{
50	char buf[1024];
51	char *b2, *b3;
52	size_t therange = getpagesize()+1;
53	int fd;
54
55	FSTEST_ENTER();
56
57	RL(fd = rump_sys_open("file", O_RDWR|O_CREAT|O_TRUNC, 0666));
58
59	memset(buf, 'A', sizeof(buf));
60	RL(rump_sys_pwrite(fd, buf, 1, getpagesize()));
61
62	memset(buf, 'B', sizeof(buf));
63	RL(rump_sys_pwrite(fd, buf, 2, getpagesize()-1));
64
65	REQUIRE_LIBC(b2 = malloc(2 * getpagesize()), NULL);
66	REQUIRE_LIBC(b3 = malloc(2 * getpagesize()), NULL);
67
68	RL(rump_sys_pread(fd, b2, therange, 0));
69
70	memset(b3, 0, therange);
71	memset(b3 + getpagesize() - 1, 'B', 2);
72
73	ATF_REQUIRE_EQ(memcmp(b2, b3, therange), 0);
74
75	rump_sys_close(fd);
76	FSTEST_EXIT();
77}
78
79static void
80extendbody(const atf_tc_t *tc, off_t seekcnt)
81{
82	char buf[TESTSZ+1];
83	struct stat sb;
84	int fd;
85
86	FSTEST_ENTER();
87	RL(fd = rump_sys_open("testfile",
88	    O_CREAT | O_RDWR | (seekcnt ? O_APPEND : 0)));
89	RL(rump_sys_ftruncate(fd, seekcnt));
90	RL(rump_sys_fstat(fd, &sb));
91	ATF_REQUIRE_EQ(sb.st_size, seekcnt);
92
93	ATF_REQUIRE_EQ(rump_sys_write(fd, TESTSTR, TESTSZ), TESTSZ);
94	ATF_REQUIRE_EQ(rump_sys_pread(fd, buf, TESTSZ, seekcnt), TESTSZ);
95	ATF_REQUIRE_STREQ(buf, TESTSTR);
96
97	RL(rump_sys_fstat(fd, &sb));
98	ATF_REQUIRE_EQ(sb.st_size, (off_t)TESTSZ + seekcnt);
99	RL(rump_sys_close(fd));
100	FSTEST_EXIT();
101}
102
103static void
104extendfile(const atf_tc_t *tc, const char *mp)
105{
106
107	extendbody(tc, 0);
108}
109
110static void
111extendfile_append(const atf_tc_t *tc, const char *mp)
112{
113
114	extendbody(tc, 37);
115}
116
117static void
118overwritebody(const atf_tc_t *tc, off_t count, bool dotrunc)
119{
120	char *buf;
121	int fd;
122
123	REQUIRE_LIBC(buf = malloc(count), NULL);
124	FSTEST_ENTER();
125	RL(fd = rump_sys_open("testi", O_CREAT | O_RDWR, 0666));
126	ATF_REQUIRE_EQ(rump_sys_write(fd, buf, count), count);
127	RL(rump_sys_close(fd));
128
129	RL(fd = rump_sys_open("testi", O_RDWR));
130	if (dotrunc)
131		RL(rump_sys_ftruncate(fd, 0));
132	ATF_REQUIRE_EQ(rump_sys_write(fd, buf, count), count);
133	RL(rump_sys_close(fd));
134	FSTEST_EXIT();
135}
136
137static void
138overwrite512(const atf_tc_t *tc, const char *mp)
139{
140
141	overwritebody(tc, 512, false);
142}
143
144static void
145overwrite64k(const atf_tc_t *tc, const char *mp)
146{
147
148	overwritebody(tc, 1<<16, false);
149}
150
151static void
152overwrite_trunc(const atf_tc_t *tc, const char *mp)
153{
154
155	overwritebody(tc, 1<<16, true);
156}
157
158static void
159shrinkfile(const atf_tc_t *tc, const char *mp)
160{
161	int fd;
162
163	FSTEST_ENTER();
164	RL(fd = rump_sys_open("file", O_RDWR|O_CREAT|O_TRUNC, 0666));
165	RL(rump_sys_ftruncate(fd, 2));
166	RL(rump_sys_ftruncate(fd, 1));
167	rump_sys_close(fd);
168	FSTEST_EXIT();
169}
170
171#define TBSIZE 9000
172static void
173read_after_unlink(const atf_tc_t *tc, const char *mp)
174{
175	char buf[TBSIZE], buf2[TBSIZE];
176	int fd;
177
178	FSTEST_ENTER();
179
180	/* create file and put some content into it */
181	RL(fd = rump_sys_open("file", O_RDWR|O_CREAT, 0666));
182	memset(buf, 'D', TBSIZE);
183	ATF_REQUIRE_EQ(rump_sys_write(fd, buf, TBSIZE), TBSIZE);
184	rump_sys_close(fd);
185
186	/* flush buffers from UBC to file system */
187	ATF_REQUIRE_ERRNO(EBUSY, rump_sys_unmount(mp, 0) == -1);
188
189	RL(fd = rump_sys_open("file", O_RDWR));
190	RL(rump_sys_unlink("file"));
191
192	ATF_REQUIRE_EQ(rump_sys_read(fd, buf2, TBSIZE), TBSIZE);
193	ATF_REQUIRE_EQ(memcmp(buf, buf2, TBSIZE), 0);
194	rump_sys_close(fd);
195
196	FSTEST_EXIT();
197}
198
199static void
200wrrd_after_unlink(const atf_tc_t *tc, const char *mp)
201{
202	int value = 0x11;
203	int v2;
204	int fd;
205
206	FSTEST_ENTER();
207
208	RL(fd = rump_sys_open("file", O_RDWR|O_CREAT, 0666));
209	RL(rump_sys_unlink("file"));
210
211	RL(rump_sys_pwrite(fd, &value, sizeof(value), 654321));
212
213	/*
214	 * We can't easily invalidate the buffer since we hold a
215	 * reference, but try to get them to flush anyway.
216	 */
217	RL(rump_sys_fsync(fd));
218	RL(rump_sys_pread(fd, &v2, sizeof(v2), 654321));
219	rump_sys_close(fd);
220
221	ATF_REQUIRE_EQ(value, v2);
222	FSTEST_EXIT();
223}
224
225static void
226read_fault(const atf_tc_t *tc, const char *mp)
227{
228	char ch = 123;
229	int fd;
230
231	FSTEST_ENTER();
232	RL(fd = rump_sys_open("file", O_CREAT | O_RDWR, 0777));
233	ATF_REQUIRE_EQ(rump_sys_write(fd, &ch, 1), 1);
234	RL(rump_sys_close(fd));
235	RL(fd = rump_sys_open("file", O_RDONLY | O_SYNC | O_RSYNC));
236	ATF_REQUIRE_ERRNO(EFAULT, rump_sys_read(fd, NULL, 1) == -1);
237	RL(rump_sys_close(fd));
238	FSTEST_EXIT();
239}
240
241ATF_TC_FSAPPLY(holywrite, "create a sparse file and fill hole");
242ATF_TC_FSAPPLY(extendfile, "check that extending a file works");
243ATF_TC_FSAPPLY(extendfile_append, "check that extending a file works "
244				  "with a append-only fd (PR kern/44307)");
245ATF_TC_FSAPPLY(overwrite512, "write a 512 byte file twice");
246ATF_TC_FSAPPLY(overwrite64k, "write a 64k byte file twice");
247ATF_TC_FSAPPLY(overwrite_trunc, "write 64k + truncate + rewrite");
248ATF_TC_FSAPPLY(shrinkfile, "shrink file");
249ATF_TC_FSAPPLY(read_after_unlink, "contents can be read off disk after unlink");
250ATF_TC_FSAPPLY(wrrd_after_unlink, "file can be written and read after unlink");
251ATF_TC_FSAPPLY(read_fault, "read at bad address must return EFAULT");
252
253ATF_TP_ADD_TCS(tp)
254{
255
256	ATF_TP_FSAPPLY(holywrite);
257	ATF_TP_FSAPPLY(extendfile);
258	ATF_TP_FSAPPLY(extendfile_append);
259	ATF_TP_FSAPPLY(overwrite512);
260	ATF_TP_FSAPPLY(overwrite64k);
261	ATF_TP_FSAPPLY(overwrite_trunc);
262	ATF_TP_FSAPPLY(shrinkfile);
263	ATF_TP_FSAPPLY(read_after_unlink);
264	ATF_TP_FSAPPLY(wrrd_after_unlink);
265	ATF_TP_FSAPPLY(read_fault);
266
267	return atf_no_error();
268}
269