xref: /freebsd/lib/libc/tests/stdio/fmemopen2_test.c (revision d9f0ce31900a48d1a2bfc1c8c86f79d1e831451a)
1 /*-
2 Copyright (C) 2013 Pietro Cerutti <gahr@FreeBSD.org>
3 
4 Redistribution and use in source and binary forms, with or without
5 modification, are permitted provided that the following conditions
6 are met:
7 1. Redistributions of source code must retain the above copyright
8    notice, this list of conditions and the following disclaimer.
9 2. Redistributions in binary form must reproduce the above copyright
10    notice, this list of conditions and the following disclaimer in the
11    documentation and/or other materials provided with the distribution.
12 
13 THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14 ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16 ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
17 FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18 DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19 OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20 HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21 LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22 OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23 SUCH DAMAGE.
24 */
25 
26 /*
27  * Test basic FILE * functions (fread, fwrite, fseek, fclose) against
28  * a FILE * retrieved using fmemopen()
29  */
30 
31 #include <sys/cdefs.h>
32 __FBSDID("$FreeBSD$");
33 
34 #include <errno.h>
35 #include <stdio.h>
36 #include <string.h>
37 #include <strings.h>
38 
39 #include <atf-c.h>
40 
41 ATF_TC_WITHOUT_HEAD(test_preexisting);
42 ATF_TC_BODY(test_preexisting, tc)
43 {
44 	/* Use a pre-existing buffer. */
45 	char buf[512];
46 	char buf2[512];
47 	char str[]  = "Test writing some stuff";
48 	char str2[] = "AAAAAAAAA";
49 	char str3[] = "AAAA writing some stuff";
50 	FILE *fp;
51 	size_t nofw, nofr;
52 	int rc;
53 
54 	/* Open a FILE * using fmemopen. */
55 	fp = fmemopen(buf, sizeof(buf), "w");
56 	ATF_REQUIRE(fp != NULL);
57 
58 	/* Write to the buffer. */
59 	nofw = fwrite(str, 1, sizeof(str), fp);
60 	ATF_REQUIRE(nofw == sizeof(str));
61 
62 	/* Close the FILE *. */
63 	rc = fclose(fp);
64 	ATF_REQUIRE(rc == 0);
65 
66 	/* Re-open the FILE * to read back the data. */
67 	fp = fmemopen(buf, sizeof(buf), "r");
68 	ATF_REQUIRE(fp != NULL);
69 
70 	/* Read from the buffer. */
71 	bzero(buf2, sizeof(buf2));
72 	nofr = fread(buf2, 1, sizeof(buf2), fp);
73 	ATF_REQUIRE(nofr == sizeof(buf2));
74 
75 	/*
76 	 * Since a write on a FILE * retrieved by fmemopen
77 	 * will add a '\0' (if there's space), we can check
78 	 * the strings for equality.
79 	 */
80 	ATF_REQUIRE(strcmp(str, buf2) == 0);
81 
82 	/* Close the FILE *. */
83 	rc = fclose(fp);
84 	ATF_REQUIRE(rc == 0);
85 
86 	/* Now open a FILE * on the first 4 bytes of the string. */
87 	fp = fmemopen(str, 4, "w");
88 	ATF_REQUIRE(fp != NULL);
89 
90 	/*
91 	 * Try to write more bytes than we shoud, we'll get a short count (4).
92 	 */
93 	nofw = fwrite(str2, 1, sizeof(str2), fp);
94 	ATF_REQUIRE(nofw == 4);
95 
96 	/* Close the FILE *. */
97 	rc = fclose(fp);
98 	ATF_REQUIRE(rc == 0);
99 
100 	/* Check that the string was not modified after the first 4 bytes. */
101 	ATF_REQUIRE(strcmp(str, str3) == 0);
102 }
103 
104 ATF_TC_WITHOUT_HEAD(test_autoalloc);
105 ATF_TC_BODY(test_autoalloc, tc)
106 {
107 	/* Let fmemopen allocate the buffer. */
108 	FILE *fp;
109 	long pos;
110 	size_t nofw, i;
111 	int rc;
112 
113 	/* Open a FILE * using fmemopen. */
114 	fp = fmemopen(NULL, 512, "w+");
115 	ATF_REQUIRE(fp != NULL);
116 
117 	/* fill the buffer */
118 	for (i = 0; i < 512; i++) {
119 		nofw = fwrite("a", 1, 1, fp);
120 		ATF_REQUIRE(nofw == 1);
121 	}
122 
123 	/* Get the current position into the stream. */
124 	pos = ftell(fp);
125 	ATF_REQUIRE(pos == 512);
126 
127 	/* Try to write past the end, we should get a short object count (0) */
128 	nofw = fwrite("a", 1, 1, fp);
129 	ATF_REQUIRE(nofw == 0);
130 
131 	/* Close the FILE *. */
132 	rc = fclose(fp);
133 	ATF_REQUIRE(rc == 0);
134 
135 	/* Open a FILE * using a wrong mode */
136 	fp = fmemopen(NULL, 512, "r");
137 	ATF_REQUIRE(fp == NULL);
138 
139 	fp = fmemopen(NULL, 512, "w");
140 	ATF_REQUIRE(fp == NULL);
141 }
142 
143 ATF_TC_WITHOUT_HEAD(test_data_length);
144 ATF_TC_BODY(test_data_length, tc)
145 {
146 	/*
147 	 * Here we test that a read operation doesn't go past the end of the
148 	 * data actually written, and that a SEEK_END seeks from the end of the
149 	 * data, not of the whole buffer.
150 	 */
151 	FILE *fp;
152 	char buf[512] = {'\0'};
153 	char str[]  = "Test data length. ";
154 	char str2[] = "Do we have two sentences?";
155 	char str3[sizeof(str) + sizeof(str2) -1];
156 	long pos;
157 	size_t nofw, nofr;
158 	int rc;
159 
160 	/* Open a FILE * for updating our buffer. */
161 	fp = fmemopen(buf, sizeof(buf), "w+");
162 	ATF_REQUIRE(fp != NULL);
163 
164 	/* Write our string into the buffer. */
165 	nofw = fwrite(str, 1, sizeof(str), fp);
166 	ATF_REQUIRE(nofw == sizeof(str));
167 
168 	/* Now seek to the end and check that ftell gives us sizeof(str). */
169 	rc = fseek(fp, 0, SEEK_END);
170 	ATF_REQUIRE(rc == 0);
171 	pos = ftell(fp);
172 	ATF_REQUIRE(pos == sizeof(str));
173 
174 	/* Close the FILE *. */
175 	rc = fclose(fp);
176 	ATF_REQUIRE(rc == 0);
177 
178 	/* Reopen the buffer for appending. */
179 	fp = fmemopen(buf, sizeof(buf), "a+");
180 	ATF_REQUIRE(fp != NULL);
181 
182 	/* We should now be writing after the first string. */
183 	nofw = fwrite(str2, 1, sizeof(str2), fp);
184 	ATF_REQUIRE(nofw == sizeof(str2));
185 
186 	/* Rewind the FILE *. */
187 	rc = fseek(fp, 0, SEEK_SET);
188 	ATF_REQUIRE(rc == 0);
189 
190 	/* Make sure we're at the beginning. */
191 	pos = ftell(fp);
192 	ATF_REQUIRE(pos == 0);
193 
194 	/* Read the whole buffer. */
195 	nofr = fread(str3, 1, sizeof(buf), fp);
196 	ATF_REQUIRE(nofr == sizeof(str3));
197 
198 	/* Make sure the two strings are there. */
199 	ATF_REQUIRE(strncmp(str3, str, sizeof(str) - 1) == 0);
200 	ATF_REQUIRE(strncmp(str3 + sizeof(str) - 1, str2, sizeof(str2)) == 0);
201 
202 	/* Close the FILE *. */
203 	rc = fclose(fp);
204 	ATF_REQUIRE(rc == 0);
205 }
206 
207 ATF_TC_WITHOUT_HEAD(test_binary);
208 ATF_TC_BODY(test_binary, tc)
209 {
210 	/*
211 	 * Make sure that NULL bytes are never appended when opening a buffer
212 	 * in binary mode.
213 	 */
214 
215 	FILE *fp;
216 	char buf[20];
217 	char str[] = "Test";
218 	size_t nofw;
219 	int rc, i;
220 
221 	/* Pre-fill the buffer. */
222 	memset(buf, 'A', sizeof(buf));
223 
224 	/* Open a FILE * in binary mode. */
225 	fp = fmemopen(buf, sizeof(buf), "w+b");
226 	ATF_REQUIRE(fp != NULL);
227 
228 	/* Write some data into it. */
229 	nofw = fwrite(str, 1, strlen(str), fp);
230 	ATF_REQUIRE(nofw == strlen(str));
231 
232 	/* Make sure that the buffer doesn't contain any NULL bytes. */
233 	for (i = 0; i < sizeof(buf); i++)
234 		ATF_REQUIRE(buf[i] != '\0');
235 
236 	/* Close the FILE *. */
237 	rc = fclose(fp);
238 	ATF_REQUIRE(rc == 0);
239 }
240 
241 ATF_TC_WITHOUT_HEAD(test_append_binary_pos);
242 ATF_TC_BODY(test_append_binary_pos, tc)
243 {
244 	/*
245 	 * For compatibility with other implementations (glibc), we set the
246 	 * position to 0 when opening an automatically allocated binary stream
247 	 * for appending.
248 	 */
249 
250 	FILE *fp;
251 
252 	fp = fmemopen(NULL, 16, "ab+");
253 	ATF_REQUIRE(ftell(fp) == 0L);
254 	fclose(fp);
255 
256 	/* Make sure that a pre-allocated buffer behaves correctly. */
257 	char buf[] = "Hello";
258 	fp = fmemopen(buf, sizeof(buf), "ab+");
259 	ATF_REQUIRE(ftell(fp) == strlen(buf));
260 	fclose(fp);
261 }
262 
263 ATF_TC_WITHOUT_HEAD(test_size_0);
264 ATF_TC_BODY(test_size_0, tc)
265 {
266 	/* POSIX mandates that we return EINVAL if size is 0. */
267 
268 	FILE *fp;
269 
270 	fp = fmemopen(NULL, 0, "r+");
271 	ATF_REQUIRE(fp == NULL);
272 	ATF_REQUIRE(errno == EINVAL);
273 }
274 
275 ATF_TP_ADD_TCS(tp)
276 {
277 
278 	ATF_TP_ADD_TC(tp, test_autoalloc);
279 	ATF_TP_ADD_TC(tp, test_preexisting);
280 	ATF_TP_ADD_TC(tp, test_data_length);
281 	ATF_TP_ADD_TC(tp, test_binary);
282 	ATF_TP_ADD_TC(tp, test_append_binary_pos);
283 	ATF_TP_ADD_TC(tp, test_size_0);
284 
285 	return (atf_no_error());
286 }
287