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