xref: /netbsd-src/tests/lib/libc/stdio/t_fopen.c (revision 510d9ff8d5962e998e5e11f2cb5b2b5e2eb67ddc)
1*510d9ff8Skamil /*	$NetBSD: t_fopen.c,v 1.8 2020/02/21 22:14:59 kamil Exp $ */
24c55f62cSjruoho 
34c55f62cSjruoho /*-
44c55f62cSjruoho  * Copyright (c) 2011 The NetBSD Foundation, Inc.
54c55f62cSjruoho  * All rights reserved.
64c55f62cSjruoho  *
74c55f62cSjruoho  * This code is derived from software contributed to The NetBSD Foundation
84c55f62cSjruoho  * by Jukka Ruohonen.
94c55f62cSjruoho  *
104c55f62cSjruoho  * Redistribution and use in source and binary forms, with or without
114c55f62cSjruoho  * modification, are permitted provided that the following conditions
124c55f62cSjruoho  * are met:
134c55f62cSjruoho  * 1. Redistributions of source code must retain the above copyright
144c55f62cSjruoho  *    notice, this list of conditions and the following disclaimer.
154c55f62cSjruoho  * 2. Redistributions in binary form must reproduce the above copyright
164c55f62cSjruoho  *    notice, this list of conditions and the following disclaimer in the
174c55f62cSjruoho  *    documentation and/or other materials provided with the distribution.
184c55f62cSjruoho  *
194c55f62cSjruoho  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
204c55f62cSjruoho  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
214c55f62cSjruoho  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
224c55f62cSjruoho  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
234c55f62cSjruoho  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
244c55f62cSjruoho  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
254c55f62cSjruoho  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
264c55f62cSjruoho  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
274c55f62cSjruoho  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
284c55f62cSjruoho  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
294c55f62cSjruoho  * POSSIBILITY OF SUCH DAMAGE.
304c55f62cSjruoho  */
314c55f62cSjruoho #include <sys/cdefs.h>
32*510d9ff8Skamil __RCSID("$NetBSD: t_fopen.c,v 1.8 2020/02/21 22:14:59 kamil Exp $");
334c55f62cSjruoho 
34fdaf7bbfSkamil #include <sys/param.h>
35fdaf7bbfSkamil #include <sys/types.h>
36fdaf7bbfSkamil #include <sys/module.h>
374c55f62cSjruoho #include <atf-c.h>
384c55f62cSjruoho #include <errno.h>
394c55f62cSjruoho #include <fcntl.h>
404c55f62cSjruoho #include <limits.h>
414c55f62cSjruoho #include <paths.h>
424c55f62cSjruoho #include <stdio.h>
43fdaf7bbfSkamil #include <stdlib.h>
444c55f62cSjruoho #include <string.h>
454c55f62cSjruoho #include <unistd.h>
464c55f62cSjruoho 
474c55f62cSjruoho static const char *path = "fopen";
484c55f62cSjruoho 
493ace9e82Sjruoho ATF_TC_WITH_CLEANUP(fdopen_close);
ATF_TC_HEAD(fdopen_close,tc)503ace9e82Sjruoho ATF_TC_HEAD(fdopen_close, tc)
513ace9e82Sjruoho {
523ace9e82Sjruoho 	atf_tc_set_md_var(tc, "descr", "See that descriptors are closed");
533ace9e82Sjruoho }
543ace9e82Sjruoho 
ATF_TC_BODY(fdopen_close,tc)553ace9e82Sjruoho ATF_TC_BODY(fdopen_close, tc)
563ace9e82Sjruoho {
573ace9e82Sjruoho 	FILE *f;
583ace9e82Sjruoho 	int fd;
593ace9e82Sjruoho 
603ace9e82Sjruoho 	/*
613ace9e82Sjruoho 	 * Check that the file descriptor
623ace9e82Sjruoho 	 * used to fdopen(3) a stream is
633ace9e82Sjruoho 	 * closed once the stream is closed.
643ace9e82Sjruoho 	 */
65653f037eSmartin 	fd = open(path, O_RDWR | O_CREAT, 0600);
663ace9e82Sjruoho 
673ace9e82Sjruoho 	ATF_REQUIRE(fd >= 0);
683ace9e82Sjruoho 
693ace9e82Sjruoho 	f = fdopen(fd, "w+");
703ace9e82Sjruoho 
713ace9e82Sjruoho 	ATF_REQUIRE(f != NULL);
723ace9e82Sjruoho 	ATF_REQUIRE(fclose(f) == 0);
733ace9e82Sjruoho 	ATF_REQUIRE(close(fd) == -1);
743ace9e82Sjruoho 	ATF_REQUIRE(unlink(path) == 0);
753ace9e82Sjruoho }
763ace9e82Sjruoho 
ATF_TC_CLEANUP(fdopen_close,tc)773ace9e82Sjruoho ATF_TC_CLEANUP(fdopen_close, tc)
783ace9e82Sjruoho {
793ace9e82Sjruoho 	(void)unlink(path);
803ace9e82Sjruoho }
813ace9e82Sjruoho 
824c55f62cSjruoho ATF_TC_WITH_CLEANUP(fdopen_err);
ATF_TC_HEAD(fdopen_err,tc)834c55f62cSjruoho ATF_TC_HEAD(fdopen_err, tc)
844c55f62cSjruoho {
854c55f62cSjruoho 	atf_tc_set_md_var(tc, "descr", "Test errors from fdopen(3)");
864c55f62cSjruoho }
874c55f62cSjruoho 
ATF_TC_BODY(fdopen_err,tc)884c55f62cSjruoho ATF_TC_BODY(fdopen_err, tc)
894c55f62cSjruoho {
904c55f62cSjruoho 	int fd;
914c55f62cSjruoho 
92653f037eSmartin 	fd = open(path, O_RDONLY | O_CREAT, 0600);
934c55f62cSjruoho 	ATF_REQUIRE(fd >= 0);
944c55f62cSjruoho 
954c55f62cSjruoho 	errno = 0;
964c55f62cSjruoho 	ATF_REQUIRE_ERRNO(EINVAL, fdopen(fd, "w") == NULL);
974c55f62cSjruoho 
984c55f62cSjruoho 	errno = 0;
994c55f62cSjruoho 	ATF_REQUIRE_ERRNO(EINVAL, fdopen(fd, "a") == NULL);
1004c55f62cSjruoho 
1014c55f62cSjruoho 	ATF_REQUIRE(close(fd) == 0);
1024c55f62cSjruoho 
1034c55f62cSjruoho 	errno = 0;
1044c55f62cSjruoho 	ATF_REQUIRE_ERRNO(EBADF, fdopen(fd, "r") == NULL);
1054c55f62cSjruoho 
1064c55f62cSjruoho 	errno = 0;
1074c55f62cSjruoho 	ATF_REQUIRE_ERRNO(EBADF, fdopen(-1, "w+") == NULL);
1084c55f62cSjruoho 
1094c55f62cSjruoho 	(void)unlink(path);
1104c55f62cSjruoho }
1114c55f62cSjruoho 
ATF_TC_CLEANUP(fdopen_err,tc)1124c55f62cSjruoho ATF_TC_CLEANUP(fdopen_err, tc)
1134c55f62cSjruoho {
1144c55f62cSjruoho 	(void)unlink(path);
1154c55f62cSjruoho }
1164c55f62cSjruoho 
1174c55f62cSjruoho ATF_TC_WITH_CLEANUP(fdopen_seek);
ATF_TC_HEAD(fdopen_seek,tc)1184c55f62cSjruoho ATF_TC_HEAD(fdopen_seek, tc)
1194c55f62cSjruoho {
1204c55f62cSjruoho 	atf_tc_set_md_var(tc, "descr", "Test stream position with fdopen(3)");
1214c55f62cSjruoho }
1224c55f62cSjruoho 
ATF_TC_BODY(fdopen_seek,tc)1234c55f62cSjruoho ATF_TC_BODY(fdopen_seek, tc)
1244c55f62cSjruoho {
1254c55f62cSjruoho 	FILE *f;
1264c55f62cSjruoho 	int fd;
1274c55f62cSjruoho 
1284c55f62cSjruoho 	/*
1294c55f62cSjruoho 	 * Verify that the file position associated
1304c55f62cSjruoho 	 * with the stream corresponds with the offset
1314c55f62cSjruoho 	 * set earlier for the file descriptor.
1324c55f62cSjruoho 	 */
133653f037eSmartin 	fd = open(path, O_RDWR | O_CREAT, 0600);
1344c55f62cSjruoho 
1354c55f62cSjruoho 	ATF_REQUIRE(fd >= 0);
1364c55f62cSjruoho 	ATF_REQUIRE(write(fd, "garbage", 7) == 7);
1374c55f62cSjruoho 	ATF_REQUIRE(lseek(fd, 3, SEEK_SET) == 3);
1384c55f62cSjruoho 
1394c55f62cSjruoho 	f = fdopen(fd, "r+");
1404c55f62cSjruoho 
1414c55f62cSjruoho 	ATF_REQUIRE(f != NULL);
1424c55f62cSjruoho 	ATF_REQUIRE(ftell(f) == 3);
1434c55f62cSjruoho 	ATF_REQUIRE(fclose(f) == 0);
1443ace9e82Sjruoho 	ATF_REQUIRE(unlink(path) == 0);
1454c55f62cSjruoho }
1464c55f62cSjruoho 
ATF_TC_CLEANUP(fdopen_seek,tc)1474c55f62cSjruoho ATF_TC_CLEANUP(fdopen_seek, tc)
1484c55f62cSjruoho {
1494c55f62cSjruoho 	(void)unlink(path);
1504c55f62cSjruoho }
1514c55f62cSjruoho 
1524c55f62cSjruoho ATF_TC_WITH_CLEANUP(fopen_err);
ATF_TC_HEAD(fopen_err,tc)1534c55f62cSjruoho ATF_TC_HEAD(fopen_err, tc)
1544c55f62cSjruoho {
1554c55f62cSjruoho 	atf_tc_set_md_var(tc, "descr", "Test errors from fopen(3)");
1564c55f62cSjruoho }
1574c55f62cSjruoho 
ATF_TC_BODY(fopen_err,tc)1584c55f62cSjruoho ATF_TC_BODY(fopen_err, tc)
1594c55f62cSjruoho {
1604c55f62cSjruoho 	static const char *mode[] = {
1614c55f62cSjruoho 		"x", "xr", "xr", "+r+", "R", "W+", " aXX", "Xr", " r+", "" };
1624c55f62cSjruoho 
1634c55f62cSjruoho 	char buf[PATH_MAX + 1];
1644c55f62cSjruoho 	size_t i;
1654c55f62cSjruoho 	FILE *f;
1664c55f62cSjruoho 
1674c55f62cSjruoho 	f = fopen(path, "w+");
1684c55f62cSjruoho 
1694c55f62cSjruoho 	ATF_REQUIRE(f != NULL);
1704c55f62cSjruoho 	ATF_REQUIRE(fclose(f) == 0);
1714c55f62cSjruoho 
1724c55f62cSjruoho 	/*
1734c55f62cSjruoho 	 * Note that also "invalid" characters
1744c55f62cSjruoho 	 * may follow the mode-string whenever
1754c55f62cSjruoho 	 * the first character is valid.
1764c55f62cSjruoho 	 */
1774c55f62cSjruoho 	for (i = 0; i < __arraycount(mode); i++) {
1784c55f62cSjruoho 
1794c55f62cSjruoho 		errno = 0;
1804c55f62cSjruoho 		f = fopen(path, mode[i]);
1814c55f62cSjruoho 
1824c55f62cSjruoho 		if (f == NULL && errno == EINVAL)
1834c55f62cSjruoho 			continue;
1844c55f62cSjruoho 
1854c55f62cSjruoho 		if (f != NULL)
1864c55f62cSjruoho 			(void)fclose(f);
1874c55f62cSjruoho 
1884c55f62cSjruoho 		atf_tc_fail_nonfatal("opened file as '%s'",  mode[i]);
1894c55f62cSjruoho 	}
1904c55f62cSjruoho 
1914c55f62cSjruoho 	(void)unlink(path);
1924c55f62cSjruoho 	(void)memset(buf, 'x', sizeof(buf));
1934c55f62cSjruoho 
1944c55f62cSjruoho 	errno = 0;
1954c55f62cSjruoho 	ATF_REQUIRE_ERRNO(EISDIR, fopen("/usr/bin", "w") == NULL);
1964c55f62cSjruoho 
1974c55f62cSjruoho 	errno = 0;
1984c55f62cSjruoho 	ATF_REQUIRE_ERRNO(ENOENT, fopen("/a/b/c/d/e/f", "r") == NULL);
1994c55f62cSjruoho 
2004c55f62cSjruoho 	errno = 0;
2014c55f62cSjruoho 	ATF_REQUIRE_ERRNO(ENAMETOOLONG, fopen(buf, "r+") == NULL);
2024c55f62cSjruoho }
2034c55f62cSjruoho 
ATF_TC_CLEANUP(fopen_err,tc)2044c55f62cSjruoho ATF_TC_CLEANUP(fopen_err, tc)
2054c55f62cSjruoho {
2064c55f62cSjruoho 	(void)unlink(path);
2074c55f62cSjruoho }
2084c55f62cSjruoho 
2094c55f62cSjruoho ATF_TC_WITH_CLEANUP(fopen_append);
ATF_TC_HEAD(fopen_append,tc)2104c55f62cSjruoho ATF_TC_HEAD(fopen_append, tc)
2114c55f62cSjruoho {
2124c55f62cSjruoho 	atf_tc_set_md_var(tc, "descr", "Test that append-mode works");
2134c55f62cSjruoho }
2144c55f62cSjruoho 
ATF_TC_BODY(fopen_append,tc)2154c55f62cSjruoho ATF_TC_BODY(fopen_append, tc)
2164c55f62cSjruoho {
2174c55f62cSjruoho 	char buf[15];
2184c55f62cSjruoho 	FILE *f;
2194c55f62cSjruoho 
2204c55f62cSjruoho 	(void)memset(buf, 'x', sizeof(buf));
2214c55f62cSjruoho 
2224c55f62cSjruoho 	f = fopen(path, "w+");
2234c55f62cSjruoho 
2244c55f62cSjruoho 	ATF_REQUIRE(f != NULL);
2254c55f62cSjruoho 	ATF_REQUIRE(fwrite("garbage", 1, 7, f) == 7);
2264c55f62cSjruoho 	ATF_REQUIRE(fclose(f) == 0);
2274c55f62cSjruoho 
2284c55f62cSjruoho 	f = fopen(path, "a");
2293ace9e82Sjruoho 
2304c55f62cSjruoho 	ATF_REQUIRE(fwrite("garbage", 1, 7, f) == 7);
2314c55f62cSjruoho 	ATF_REQUIRE(fclose(f) == 0);
2324c55f62cSjruoho 
2334c55f62cSjruoho 	f = fopen(path, "r");
2343ace9e82Sjruoho 
2354c55f62cSjruoho 	ATF_REQUIRE(fread(buf, 1, sizeof(buf), f) == 14);
2364c55f62cSjruoho 	ATF_REQUIRE(strncmp(buf, "garbagegarbage", 14) == 0);
2374c55f62cSjruoho 
2384c55f62cSjruoho 	ATF_REQUIRE(fclose(f) == 0);
2394c55f62cSjruoho 	ATF_REQUIRE(unlink(path) == 0);
2404c55f62cSjruoho }
2414c55f62cSjruoho 
ATF_TC_CLEANUP(fopen_append,tc)2424c55f62cSjruoho ATF_TC_CLEANUP(fopen_append, tc)
2434c55f62cSjruoho {
2444c55f62cSjruoho 	(void)unlink(path);
2454c55f62cSjruoho }
2464c55f62cSjruoho 
2474c55f62cSjruoho ATF_TC_WITH_CLEANUP(fopen_mode);
ATF_TC_HEAD(fopen_mode,tc)2484c55f62cSjruoho ATF_TC_HEAD(fopen_mode, tc)
2494c55f62cSjruoho {
2504c55f62cSjruoho 	atf_tc_set_md_var(tc, "descr", "Test fopen(3) modes");
2514c55f62cSjruoho }
2524c55f62cSjruoho 
ATF_TC_BODY(fopen_mode,tc)2534c55f62cSjruoho ATF_TC_BODY(fopen_mode, tc)
2544c55f62cSjruoho {
2554c55f62cSjruoho 	size_t i;
2564c55f62cSjruoho 	FILE *f;
2574c55f62cSjruoho 
2584c55f62cSjruoho 	static const char *mode[] = {
2594c55f62cSjruoho 		"r", "r+", "w", "w+", "a", "a+",
26045595902Skre 		"rb", "r+b", "wb", "w+b", "ab", "a+b",
26145595902Skre 		"re", "r+e", "we", "w+e", "ae", "a+e",
26245595902Skre 		"rf", "r+f", "wf", "w+f", "af", "a+f",
26345595902Skre 		"rl", "r+l", "wl", "w+l", "al", "a+l"
2644c55f62cSjruoho 	};
2654c55f62cSjruoho 
2664c55f62cSjruoho 	f = fopen(path, "w+");
2674c55f62cSjruoho 
2684c55f62cSjruoho 	ATF_REQUIRE(f != NULL);
2694c55f62cSjruoho 	ATF_REQUIRE(fclose(f) == 0);
2704c55f62cSjruoho 
2714c55f62cSjruoho 	/*
2724c55f62cSjruoho 	 * Verify that various modes work.
2734c55f62cSjruoho 	 */
2744c55f62cSjruoho 	for (i = 0; i < __arraycount(mode); i++) {
2754c55f62cSjruoho 
2764c55f62cSjruoho 		f = fopen(path, mode[i]);
2774c55f62cSjruoho 
2784c55f62cSjruoho 		if (f != NULL) {
2794c55f62cSjruoho 			ATF_REQUIRE(fclose(f) == 0);
2804c55f62cSjruoho 			continue;
2814c55f62cSjruoho 		}
2824c55f62cSjruoho 
2834c55f62cSjruoho 		atf_tc_fail_nonfatal("failed to open file as %s",  mode[i]);
2844c55f62cSjruoho 	}
2854c55f62cSjruoho 
2864c55f62cSjruoho 	(void)unlink(path);
2874c55f62cSjruoho }
2884c55f62cSjruoho 
ATF_TC_CLEANUP(fopen_mode,tc)2894c55f62cSjruoho ATF_TC_CLEANUP(fopen_mode, tc)
2904c55f62cSjruoho {
2914c55f62cSjruoho 	(void)unlink(path);
2924c55f62cSjruoho }
2934c55f62cSjruoho 
294fdaf7bbfSkamil static void
check_kernel_modular(void)295fdaf7bbfSkamil check_kernel_modular(void)
296fdaf7bbfSkamil {
297fdaf7bbfSkamil 	int err;
298fdaf7bbfSkamil 
299fdaf7bbfSkamil 	err = modctl(MODCTL_EXISTS, 0);
300fdaf7bbfSkamil 	if (err == 0) return;
301fdaf7bbfSkamil 	if (errno == ENOSYS)
302fdaf7bbfSkamil 		atf_tc_skip("Kernel does not have 'options MODULAR'.");
303fdaf7bbfSkamil 	if (errno == EPERM)
304fdaf7bbfSkamil 		return; /* Module loading can be administratively forbidden */
305fdaf7bbfSkamil 	ATF_REQUIRE_EQ_MSG(errno, 0, "unexpected error %d from "
306fdaf7bbfSkamil 	    "modctl(MODCTL_EXISTS, 0)", errno);
307fdaf7bbfSkamil }
308fdaf7bbfSkamil 
309fdaf7bbfSkamil static bool
is_module_present(const char * name)310fdaf7bbfSkamil is_module_present(const char *name)
311fdaf7bbfSkamil {
312fdaf7bbfSkamil 	bool found;
313fdaf7bbfSkamil 	size_t len;
314fdaf7bbfSkamil 	int count;
315fdaf7bbfSkamil 	struct iovec iov;
316fdaf7bbfSkamil 	modstat_t *ms;
317*510d9ff8Skamil 	modstat_t m;
318fdaf7bbfSkamil 
319fdaf7bbfSkamil 	for (len = 8192; ;) {
320fdaf7bbfSkamil 		iov.iov_base = malloc(len);
321fdaf7bbfSkamil 		iov.iov_len = len;
322fdaf7bbfSkamil 
323fdaf7bbfSkamil 		errno = 0;
324fdaf7bbfSkamil 
325fdaf7bbfSkamil 		if (modctl(MODCTL_STAT, &iov) != 0) {
326fdaf7bbfSkamil 			fprintf(stderr, "modctl(MODCTL_STAT) failed: %s\n",
327fdaf7bbfSkamil 			    strerror(errno));
328fdaf7bbfSkamil 			atf_tc_fail("Failed to query module status");
329fdaf7bbfSkamil 		}
330fdaf7bbfSkamil 		if (len >= iov.iov_len)
331fdaf7bbfSkamil 			break;
332fdaf7bbfSkamil 		free(iov.iov_base);
333fdaf7bbfSkamil 		len = iov.iov_len;
334fdaf7bbfSkamil 	}
335fdaf7bbfSkamil 
336fdaf7bbfSkamil 	found = false;
337fdaf7bbfSkamil 	count = *(int *)iov.iov_base;
338fdaf7bbfSkamil 	ms = (modstat_t *)((char *)iov.iov_base + sizeof(int));
339fdaf7bbfSkamil 	while (count > 0) {
340*510d9ff8Skamil 		memcpy(&m, ms, sizeof(m));
341*510d9ff8Skamil 		if (strcmp(m.ms_name, name) == 0) {
342fdaf7bbfSkamil 			found = true;
343fdaf7bbfSkamil 			break;
344fdaf7bbfSkamil 		}
345fdaf7bbfSkamil 		ms++;
346fdaf7bbfSkamil 		count--;
347fdaf7bbfSkamil 	}
348fdaf7bbfSkamil 
349fdaf7bbfSkamil 	free(iov.iov_base);
350fdaf7bbfSkamil 
351fdaf7bbfSkamil 	return found;
352fdaf7bbfSkamil }
353fdaf7bbfSkamil 
354fdaf7bbfSkamil #define COMPAT10_MODNAME "compat_10"
355fdaf7bbfSkamil 
356fdaf7bbfSkamil ATF_TC(fopen_nullptr);
ATF_TC_HEAD(fopen_nullptr,tc)357fdaf7bbfSkamil ATF_TC_HEAD(fopen_nullptr, tc)
358fdaf7bbfSkamil {
359fdaf7bbfSkamil 	atf_tc_set_md_var(tc, "descr", "Test fopen(3) with NULL path (without "
360fdaf7bbfSkamil 	    COMPAT10_MODNAME ")");
361fdaf7bbfSkamil }
362fdaf7bbfSkamil 
ATF_TC_BODY(fopen_nullptr,tc)363fdaf7bbfSkamil ATF_TC_BODY(fopen_nullptr, tc)
364fdaf7bbfSkamil {
365fdaf7bbfSkamil 	bool compat10;
366fdaf7bbfSkamil 
367fdaf7bbfSkamil 	check_kernel_modular();
368fdaf7bbfSkamil 	compat10 = is_module_present(COMPAT10_MODNAME);
369fdaf7bbfSkamil 
370fdaf7bbfSkamil 	if (compat10)
371fdaf7bbfSkamil 		atf_tc_skip("Kernel does have the " COMPAT10_MODNAME
372fdaf7bbfSkamil 		    " module loaded into the kernel");
373fdaf7bbfSkamil 
374fdaf7bbfSkamil 	/* NULL shall trigger error */
375fdaf7bbfSkamil 	ATF_REQUIRE_ERRNO(EFAULT, fopen(NULL, "r") == NULL);
376fdaf7bbfSkamil }
377fdaf7bbfSkamil 
378fdaf7bbfSkamil ATF_TC(fopen_nullptr_compat10);
ATF_TC_HEAD(fopen_nullptr_compat10,tc)379fdaf7bbfSkamil ATF_TC_HEAD(fopen_nullptr_compat10, tc)
380fdaf7bbfSkamil {
381fdaf7bbfSkamil 	atf_tc_set_md_var(tc, "descr", "Test fopen(3) with NULL path (with "
382fdaf7bbfSkamil 	    COMPAT10_MODNAME ")");
383fdaf7bbfSkamil }
384fdaf7bbfSkamil 
ATF_TC_BODY(fopen_nullptr_compat10,tc)385fdaf7bbfSkamil ATF_TC_BODY(fopen_nullptr_compat10, tc)
386fdaf7bbfSkamil {
387fdaf7bbfSkamil 	FILE *fp;
388fdaf7bbfSkamil 	bool compat10;
389fdaf7bbfSkamil 
390fdaf7bbfSkamil 	check_kernel_modular();
391fdaf7bbfSkamil 	compat10 = is_module_present(COMPAT10_MODNAME);
392fdaf7bbfSkamil 
393fdaf7bbfSkamil 	if (!compat10)
394fdaf7bbfSkamil 		atf_tc_skip("Kernel does not have the " COMPAT10_MODNAME
395fdaf7bbfSkamil 		    " module loaded into the kernel");
396fdaf7bbfSkamil 
397fdaf7bbfSkamil 	/* NULL is translated to "." and shall success */
398fdaf7bbfSkamil 	fp = fopen(NULL, "r");
399fdaf7bbfSkamil 
400fdaf7bbfSkamil 	ATF_REQUIRE(fp != NULL);
401fdaf7bbfSkamil 	ATF_REQUIRE(fclose(fp) == 0);
402fdaf7bbfSkamil }
403fdaf7bbfSkamil 
4044c55f62cSjruoho ATF_TC(fopen_perm);
ATF_TC_HEAD(fopen_perm,tc)4054c55f62cSjruoho ATF_TC_HEAD(fopen_perm, tc)
4064c55f62cSjruoho {
4074c55f62cSjruoho 	atf_tc_set_md_var(tc, "descr", "Test permissions with fopen(3)");
4084c55f62cSjruoho 	atf_tc_set_md_var(tc, "require.user", "unprivileged");
4094c55f62cSjruoho }
4104c55f62cSjruoho 
ATF_TC_BODY(fopen_perm,tc)4114c55f62cSjruoho ATF_TC_BODY(fopen_perm, tc)
4124c55f62cSjruoho {
4134c55f62cSjruoho 
4144c55f62cSjruoho 	errno = 0;
4154c55f62cSjruoho 	ATF_REQUIRE_ERRNO(EACCES, fopen("/bin/ls", "a+") == NULL);
4164c55f62cSjruoho 
4174c55f62cSjruoho 	errno = 0;
4184c55f62cSjruoho 	ATF_REQUIRE_ERRNO(EACCES, fopen("/bin/ls", "w+") == NULL);
4194c55f62cSjruoho }
4204c55f62cSjruoho 
4214c55f62cSjruoho ATF_TC(fopen_regular);
ATF_TC_HEAD(fopen_regular,tc)4224c55f62cSjruoho ATF_TC_HEAD(fopen_regular, tc)
4234c55f62cSjruoho {
4244c55f62cSjruoho 	atf_tc_set_md_var(tc, "descr", "Test fopen(3) with 'f' mode");
4254c55f62cSjruoho }
4264c55f62cSjruoho 
ATF_TC_BODY(fopen_regular,tc)4274c55f62cSjruoho ATF_TC_BODY(fopen_regular, tc)
4284c55f62cSjruoho {
4294c55f62cSjruoho 	static const char *mode[] = { "rf", "r+f", "wf", "w+f", "af", "a+f" };
4302f1110cfSmartin 	static const char *devs[] = { _PATH_DEVNULL };
4314c55f62cSjruoho 
4324c55f62cSjruoho 	size_t i, j;
4334c55f62cSjruoho 	FILE *f;
4344c55f62cSjruoho 
4354c55f62cSjruoho 	for (i = 0; i < __arraycount(devs); i++) {
4364c55f62cSjruoho 
4374c55f62cSjruoho 		for (j = 0; j < __arraycount(mode); j++) {
4384c55f62cSjruoho 
4394c55f62cSjruoho 			errno = 0;
4404c55f62cSjruoho 			f = fopen(devs[i], mode[j]);
4414c55f62cSjruoho 
4424c55f62cSjruoho 			if (f == NULL && errno == EFTYPE)
4434c55f62cSjruoho 				continue;
4444c55f62cSjruoho 
44545595902Skre 			if (f != NULL) {
4464c55f62cSjruoho 				(void)fclose(f);
4474c55f62cSjruoho 
4484c55f62cSjruoho 				atf_tc_fail_nonfatal("opened %s as %s",
4494c55f62cSjruoho 				    devs[i], mode[j]);
45045595902Skre 			} else {
45145595902Skre 				atf_tc_fail_nonfatal(
45245595902Skre 				    "err %d (%s) from open of %s as %s", errno,
45345595902Skre 				    strerror(errno), devs[i], mode[j]);
45445595902Skre 			}
4554c55f62cSjruoho 		}
4564c55f62cSjruoho 	}
4574c55f62cSjruoho }
4584c55f62cSjruoho 
459f3898d4dSchristos static char linkpath[] = "symlink";
460f3898d4dSchristos 
461f3898d4dSchristos ATF_TC_WITH_CLEANUP(fopen_symlink);
ATF_TC_HEAD(fopen_symlink,tc)462f3898d4dSchristos ATF_TC_HEAD(fopen_symlink, tc)
463f3898d4dSchristos {
464f3898d4dSchristos 	atf_tc_set_md_var(tc, "descr", "Test fopen(3) with 'l' mode");
465f3898d4dSchristos }
466f3898d4dSchristos 
ATF_TC_BODY(fopen_symlink,tc)467f3898d4dSchristos ATF_TC_BODY(fopen_symlink, tc)
468f3898d4dSchristos {
469f3898d4dSchristos 	static const char *mode[] = { "rl", "r+l", "wl", "w+l", "al", "a+l" };
470f3898d4dSchristos 	size_t j;
471f3898d4dSchristos 	FILE *f;
472f3898d4dSchristos 
473f3898d4dSchristos 	ATF_CHECK(symlink("/dev/null", linkpath) != -1);
474f3898d4dSchristos 
475f3898d4dSchristos 	for (j = 0; j < __arraycount(mode); j++) {
476f3898d4dSchristos 
477f3898d4dSchristos 		errno = 0;
478f3898d4dSchristos 		f = fopen(linkpath, mode[j]);
479f3898d4dSchristos 
480f3898d4dSchristos 		if (f == NULL && errno == EFTYPE)
481f3898d4dSchristos 			continue;
482f3898d4dSchristos 
48345595902Skre 		if (f != NULL) {
484f3898d4dSchristos 			(void)fclose(f);
485f3898d4dSchristos 
48645595902Skre 			atf_tc_fail_nonfatal("opened %s as %s", linkpath,
48745595902Skre 			    mode[j]);
48845595902Skre 		} else {
48945595902Skre 			atf_tc_fail_nonfatal(
49045595902Skre 			    "err %d (%s) from open of %s as %s", errno,
49145595902Skre 			    strerror(errno), linkpath, mode[j]);
49245595902Skre 		}
493f3898d4dSchristos 	}
494f3898d4dSchristos 	ATF_REQUIRE(unlink(linkpath) == 0);
495f3898d4dSchristos }
496f3898d4dSchristos 
ATF_TC_CLEANUP(fopen_symlink,tc)497f3898d4dSchristos ATF_TC_CLEANUP(fopen_symlink, tc)
498f3898d4dSchristos {
499f3898d4dSchristos 	(void)unlink(linkpath);
500f3898d4dSchristos }
501f3898d4dSchristos 
5024c55f62cSjruoho ATF_TC_WITH_CLEANUP(fopen_seek);
ATF_TC_HEAD(fopen_seek,tc)5034c55f62cSjruoho ATF_TC_HEAD(fopen_seek, tc)
5044c55f62cSjruoho {
5054c55f62cSjruoho 	atf_tc_set_md_var(tc, "descr", "Test initial stream position");
5064c55f62cSjruoho }
5074c55f62cSjruoho 
ATF_TC_BODY(fopen_seek,tc)5084c55f62cSjruoho ATF_TC_BODY(fopen_seek, tc)
5094c55f62cSjruoho {
5104c55f62cSjruoho 	FILE *f;
5114c55f62cSjruoho 
5124c55f62cSjruoho 	f = fopen(path, "w+");
5134c55f62cSjruoho 
5144c55f62cSjruoho 	ATF_REQUIRE(f != NULL);
5154c55f62cSjruoho 	ATF_REQUIRE(fwrite("garbage", 1, 7, f) == 7);
5164c55f62cSjruoho 	ATF_REQUIRE(fclose(f) == 0);
5174c55f62cSjruoho 
5184c55f62cSjruoho 	/*
5194c55f62cSjruoho 	 * The position of the stream should be
5204c55f62cSjruoho 	 * at the start, except for append-mode.
5214c55f62cSjruoho 	 */
5224c55f62cSjruoho 	f = fopen(path, "r");
5234c55f62cSjruoho 
5244c55f62cSjruoho 	ATF_REQUIRE(f != NULL);
5254c55f62cSjruoho 	ATF_REQUIRE(ftello(f) == 0);
5264c55f62cSjruoho 	ATF_REQUIRE(fclose(f) == 0);
5274c55f62cSjruoho 
5284c55f62cSjruoho 	f = fopen(path, "a");
5294c55f62cSjruoho 
5304c55f62cSjruoho 	ATF_REQUIRE(f != NULL);
5314c55f62cSjruoho 	ATF_REQUIRE(ftello(f) == 7);
5324c55f62cSjruoho 	ATF_REQUIRE(fclose(f) == 0);
5334c55f62cSjruoho 	ATF_REQUIRE(unlink(path) == 0);
5344c55f62cSjruoho }
5354c55f62cSjruoho 
ATF_TC_CLEANUP(fopen_seek,tc)5364c55f62cSjruoho ATF_TC_CLEANUP(fopen_seek, tc)
5374c55f62cSjruoho {
5384c55f62cSjruoho 	(void)unlink(path);
5394c55f62cSjruoho }
5404c55f62cSjruoho 
5414c55f62cSjruoho ATF_TC_WITH_CLEANUP(freopen_std);
ATF_TC_HEAD(freopen_std,tc)5424c55f62cSjruoho ATF_TC_HEAD(freopen_std, tc)
5434c55f62cSjruoho {
5444c55f62cSjruoho 	atf_tc_set_md_var(tc, "descr", "A basic test of freopen(3)");
5454c55f62cSjruoho }
5464c55f62cSjruoho 
ATF_TC_BODY(freopen_std,tc)5474c55f62cSjruoho ATF_TC_BODY(freopen_std, tc)
5484c55f62cSjruoho {
5493ace9e82Sjruoho 	FILE *std[2] = { stdin, stdout };
5504c55f62cSjruoho 	char buf[15];
5514c55f62cSjruoho 	size_t i;
5524c55f62cSjruoho 	FILE *f;
5534c55f62cSjruoho 
5544c55f62cSjruoho 	/*
5554c55f62cSjruoho 	 * Associate a standard stream with a custom stream.
5564c55f62cSjruoho 	 * Then write to the standard stream and verify that
5574c55f62cSjruoho 	 * the result now appears in the custom stream.
5584c55f62cSjruoho 	 */
5594c55f62cSjruoho 	for (i = 0; i < __arraycount(std); i++) {
5604c55f62cSjruoho 
5614c55f62cSjruoho 		(void)memset(buf, 'x', sizeof(buf));
5624c55f62cSjruoho 
5634c55f62cSjruoho 		f = fopen(path, "w+");
5644c55f62cSjruoho 		ATF_REQUIRE(f != NULL);
5654c55f62cSjruoho 
5664c55f62cSjruoho 		f = freopen(path, "w+", std[i]);
5674c55f62cSjruoho 		ATF_REQUIRE(f != NULL);
5684c55f62cSjruoho 
5694c55f62cSjruoho 		ATF_REQUIRE(fwrite("garbage", 1, 7, f) == 7);
5704c55f62cSjruoho 		ATF_REQUIRE(fprintf(std[i], "garbage") == 7);
5714c55f62cSjruoho 		ATF_REQUIRE(fclose(f) == 0);
5724c55f62cSjruoho 
5734c55f62cSjruoho 		f = fopen(path, "r");
5744c55f62cSjruoho 
5754c55f62cSjruoho 		ATF_REQUIRE(f != NULL);
5764c55f62cSjruoho 		ATF_REQUIRE(fread(buf, 1, sizeof(buf), f) == 14);
5774c55f62cSjruoho 		ATF_REQUIRE(strncmp(buf, "garbagegarbage", 14) == 0);
5784c55f62cSjruoho 
5794c55f62cSjruoho 		ATF_REQUIRE(fclose(f) == 0);
5804c55f62cSjruoho 	}
5814c55f62cSjruoho 
5824c55f62cSjruoho 	ATF_REQUIRE(unlink(path) == 0);
5834c55f62cSjruoho }
5844c55f62cSjruoho 
ATF_TC_CLEANUP(freopen_std,tc)5854c55f62cSjruoho ATF_TC_CLEANUP(freopen_std, tc)
5864c55f62cSjruoho {
5874c55f62cSjruoho 	(void)unlink(path);
5884c55f62cSjruoho }
5894c55f62cSjruoho 
ATF_TP_ADD_TCS(tp)5904c55f62cSjruoho ATF_TP_ADD_TCS(tp)
5914c55f62cSjruoho {
5924c55f62cSjruoho 
5933ace9e82Sjruoho 	ATF_TP_ADD_TC(tp, fdopen_close);
5944c55f62cSjruoho 	ATF_TP_ADD_TC(tp, fdopen_err);
5954c55f62cSjruoho 	ATF_TP_ADD_TC(tp, fdopen_seek);
5964c55f62cSjruoho 	ATF_TP_ADD_TC(tp, fopen_append);
5974c55f62cSjruoho 	ATF_TP_ADD_TC(tp, fopen_err);
5984c55f62cSjruoho 	ATF_TP_ADD_TC(tp, fopen_mode);
599fdaf7bbfSkamil 	ATF_TP_ADD_TC(tp, fopen_nullptr);
600fdaf7bbfSkamil 	ATF_TP_ADD_TC(tp, fopen_nullptr_compat10);
6014c55f62cSjruoho 	ATF_TP_ADD_TC(tp, fopen_perm);
6024c55f62cSjruoho 	ATF_TP_ADD_TC(tp, fopen_regular);
603f3898d4dSchristos 	ATF_TP_ADD_TC(tp, fopen_symlink);
6044c55f62cSjruoho 	ATF_TP_ADD_TC(tp, fopen_seek);
6054c55f62cSjruoho 	ATF_TP_ADD_TC(tp, freopen_std);
6064c55f62cSjruoho 
6074c55f62cSjruoho 	return atf_no_error();
6084c55f62cSjruoho }
609