1*cd62a92dSRobert Mustacchi /*
2*cd62a92dSRobert Mustacchi  * This file and its contents are supplied under the terms of the
3*cd62a92dSRobert Mustacchi  * Common Development and Distribution License ("CDDL"), version 1.0.
4*cd62a92dSRobert Mustacchi  * You may only use this file in accordance with the terms of version
5*cd62a92dSRobert Mustacchi  * 1.0 of the CDDL.
6*cd62a92dSRobert Mustacchi  *
7*cd62a92dSRobert Mustacchi  * A full copy of the text of the CDDL should have accompanied this
8*cd62a92dSRobert Mustacchi  * source.  A copy of the CDDL is also available via the Internet at
9*cd62a92dSRobert Mustacchi  * http://www.illumos.org/license/CDDL.
10*cd62a92dSRobert Mustacchi  */
11*cd62a92dSRobert Mustacchi 
12*cd62a92dSRobert Mustacchi /*
13*cd62a92dSRobert Mustacchi  * Copyright 2020 Robert Mustacchi
14*cd62a92dSRobert Mustacchi  */
15*cd62a92dSRobert Mustacchi 
16*cd62a92dSRobert Mustacchi /*
17*cd62a92dSRobert Mustacchi  * Tests to verify fileno(3C) behavior. This test explicitly leaks fds and FILE
18*cd62a92dSRobert Mustacchi  * structures to make it easier to verify the subsequent fd behavior works and
19*cd62a92dSRobert Mustacchi  * is apparent through the FILE *.
20*cd62a92dSRobert Mustacchi  */
21*cd62a92dSRobert Mustacchi 
22*cd62a92dSRobert Mustacchi #include <stdio.h>
23*cd62a92dSRobert Mustacchi #include <unistd.h>
24*cd62a92dSRobert Mustacchi #include <err.h>
25*cd62a92dSRobert Mustacchi #include <stdlib.h>
26*cd62a92dSRobert Mustacchi #include <sys/types.h>
27*cd62a92dSRobert Mustacchi #include <sys/stat.h>
28*cd62a92dSRobert Mustacchi #include <fcntl.h>
29*cd62a92dSRobert Mustacchi #include <wchar.h>
30*cd62a92dSRobert Mustacchi 
31*cd62a92dSRobert Mustacchi #define	FNO_DUPFD	150
32*cd62a92dSRobert Mustacchi 
33*cd62a92dSRobert Mustacchi static uint_t fno_nfails;
34*cd62a92dSRobert Mustacchi static uint_t fno_ntests;
35*cd62a92dSRobert Mustacchi static int fno_nextfd;
36*cd62a92dSRobert Mustacchi 
37*cd62a92dSRobert Mustacchi const char *
_umem_debug_init(void)38*cd62a92dSRobert Mustacchi _umem_debug_init(void)
39*cd62a92dSRobert Mustacchi {
40*cd62a92dSRobert Mustacchi 	return ("default,verbose");
41*cd62a92dSRobert Mustacchi }
42*cd62a92dSRobert Mustacchi 
43*cd62a92dSRobert Mustacchi const char *
_umem_logging_init(void)44*cd62a92dSRobert Mustacchi _umem_logging_init(void)
45*cd62a92dSRobert Mustacchi {
46*cd62a92dSRobert Mustacchi 	return ("fail,contents");
47*cd62a92dSRobert Mustacchi }
48*cd62a92dSRobert Mustacchi 
49*cd62a92dSRobert Mustacchi 
50*cd62a92dSRobert Mustacchi static void
check_file(FILE * fp,int fd,const char * msg)51*cd62a92dSRobert Mustacchi check_file(FILE *fp, int fd, const char *msg)
52*cd62a92dSRobert Mustacchi {
53*cd62a92dSRobert Mustacchi 	int act = fileno(fp);
54*cd62a92dSRobert Mustacchi 	if (act != fd) {
55*cd62a92dSRobert Mustacchi 		(void) printf("TEST FAILED: %s: expected fd %d, found %d\n",
56*cd62a92dSRobert Mustacchi 		    msg, fd, act);
57*cd62a92dSRobert Mustacchi 		fno_nfails++;
58*cd62a92dSRobert Mustacchi 	} else {
59*cd62a92dSRobert Mustacchi 		(void) printf("TEST PASSED: %s\n", msg);
60*cd62a92dSRobert Mustacchi 	}
61*cd62a92dSRobert Mustacchi 	fno_ntests++;
62*cd62a92dSRobert Mustacchi }
63*cd62a92dSRobert Mustacchi 
64*cd62a92dSRobert Mustacchi static void
check_open_n(int n)65*cd62a92dSRobert Mustacchi check_open_n(int n)
66*cd62a92dSRobert Mustacchi {
67*cd62a92dSRobert Mustacchi 	int fdbase;
68*cd62a92dSRobert Mustacchi 	uint_t i;
69*cd62a92dSRobert Mustacchi 
70*cd62a92dSRobert Mustacchi 	for (i = 0, fdbase = fno_nextfd; i < n; i++, fdbase++) {
71*cd62a92dSRobert Mustacchi 		FILE *f = fopen("/dev/null", "w+");
72*cd62a92dSRobert Mustacchi 		if (f == NULL) {
73*cd62a92dSRobert Mustacchi 			err(EXIT_FAILURE, "failed to open /dev/null");
74*cd62a92dSRobert Mustacchi 		}
75*cd62a92dSRobert Mustacchi 		check_file(f, fdbase, "Consecutive FDs");
76*cd62a92dSRobert Mustacchi 	}
77*cd62a92dSRobert Mustacchi }
78*cd62a92dSRobert Mustacchi 
79*cd62a92dSRobert Mustacchi static void
check_memstream(void)80*cd62a92dSRobert Mustacchi check_memstream(void)
81*cd62a92dSRobert Mustacchi {
82*cd62a92dSRobert Mustacchi 	FILE *fmem, *omem, *wmem;
83*cd62a92dSRobert Mustacchi 	char *buf;
84*cd62a92dSRobert Mustacchi 	wchar_t *wbuf;
85*cd62a92dSRobert Mustacchi 	size_t size;
86*cd62a92dSRobert Mustacchi 
87*cd62a92dSRobert Mustacchi 	fmem = fmemopen(NULL, 10, "w+");
88*cd62a92dSRobert Mustacchi 	if (fmem == NULL) {
89*cd62a92dSRobert Mustacchi 		err(EXIT_FAILURE, "failed to fmemopen()");
90*cd62a92dSRobert Mustacchi 	}
91*cd62a92dSRobert Mustacchi 
92*cd62a92dSRobert Mustacchi 	omem = open_memstream(&buf, &size);
93*cd62a92dSRobert Mustacchi 	if (omem == NULL) {
94*cd62a92dSRobert Mustacchi 		err(EXIT_FAILURE, "failed to open_memstream()");
95*cd62a92dSRobert Mustacchi 	}
96*cd62a92dSRobert Mustacchi 
97*cd62a92dSRobert Mustacchi 	wmem = open_wmemstream(&wbuf, &size);
98*cd62a92dSRobert Mustacchi 	if (wmem == NULL) {
99*cd62a92dSRobert Mustacchi 		err(EXIT_FAILURE, "failed to open_wmemstream()");
100*cd62a92dSRobert Mustacchi 	}
101*cd62a92dSRobert Mustacchi 
102*cd62a92dSRobert Mustacchi 	check_file(fmem, -1, "basic fmemopen()");
103*cd62a92dSRobert Mustacchi 	check_file(omem, -1, "basic open_memstream()");
104*cd62a92dSRobert Mustacchi 	check_file(wmem, -1, "basic open_wmemstream()");
105*cd62a92dSRobert Mustacchi }
106*cd62a92dSRobert Mustacchi 
107*cd62a92dSRobert Mustacchi static void
check_fdopen(void)108*cd62a92dSRobert Mustacchi check_fdopen(void)
109*cd62a92dSRobert Mustacchi {
110*cd62a92dSRobert Mustacchi 	int fd, dupfd;
111*cd62a92dSRobert Mustacchi 	FILE *f;
112*cd62a92dSRobert Mustacchi 
113*cd62a92dSRobert Mustacchi 	fd = open("/dev/null", O_RDWR);
114*cd62a92dSRobert Mustacchi 	if (fd < 0) {
115*cd62a92dSRobert Mustacchi 		err(EXIT_FAILURE, "failed to open /dev/null");
116*cd62a92dSRobert Mustacchi 	}
117*cd62a92dSRobert Mustacchi 	fno_nextfd = fd + 1;
118*cd62a92dSRobert Mustacchi 
119*cd62a92dSRobert Mustacchi 	f = fdopen(fd, "r+");
120*cd62a92dSRobert Mustacchi 	if (f == NULL) {
121*cd62a92dSRobert Mustacchi 		err(EXIT_FAILURE, "failed to fdopen /dev/null");
122*cd62a92dSRobert Mustacchi 	}
123*cd62a92dSRobert Mustacchi 	check_file(f, fd, "fdopen");
124*cd62a92dSRobert Mustacchi 
125*cd62a92dSRobert Mustacchi 	if ((dupfd = dup2(fd, FNO_DUPFD)) != FNO_DUPFD) {
126*cd62a92dSRobert Mustacchi 		err(EXIT_FAILURE, "failed to dup2 /dev/null");
127*cd62a92dSRobert Mustacchi 	}
128*cd62a92dSRobert Mustacchi 	f = fdopen(dupfd, "r+");
129*cd62a92dSRobert Mustacchi 	if (f == NULL) {
130*cd62a92dSRobert Mustacchi 		err(EXIT_FAILURE, "failed to fdopen dup2'd /dev/null");
131*cd62a92dSRobert Mustacchi 	}
132*cd62a92dSRobert Mustacchi 	check_file(f, dupfd, "fdopen of dup2'd file");
133*cd62a92dSRobert Mustacchi 
134*cd62a92dSRobert Mustacchi 	f = freopen("/dev/zero", "r+", f);
135*cd62a92dSRobert Mustacchi 	if (f == NULL) {
136*cd62a92dSRobert Mustacchi 		err(EXIT_FAILURE, "failed to freopen dup2'd FILE *");
137*cd62a92dSRobert Mustacchi 	}
138*cd62a92dSRobert Mustacchi 	check_file(f, fno_nextfd, "freopen dup2'd FILE *");
139*cd62a92dSRobert Mustacchi 	fno_nextfd++;
140*cd62a92dSRobert Mustacchi }
141*cd62a92dSRobert Mustacchi 
142*cd62a92dSRobert Mustacchi static void
check_alternate(void)143*cd62a92dSRobert Mustacchi check_alternate(void)
144*cd62a92dSRobert Mustacchi {
145*cd62a92dSRobert Mustacchi 	wchar_t *c;
146*cd62a92dSRobert Mustacchi 	size_t s, i;
147*cd62a92dSRobert Mustacchi 
148*cd62a92dSRobert Mustacchi 	for (i = 0; i < 10; i++) {
149*cd62a92dSRobert Mustacchi 		FILE *f, *save;
150*cd62a92dSRobert Mustacchi 		f = fmemopen(NULL, 10, "a+");
151*cd62a92dSRobert Mustacchi 		if (f == NULL) {
152*cd62a92dSRobert Mustacchi 			err(EXIT_FAILURE, "failed to create fmemopen stream");
153*cd62a92dSRobert Mustacchi 		}
154*cd62a92dSRobert Mustacchi 		check_file(f, -1, "alternating memstream, fopen (fmemopen)");
155*cd62a92dSRobert Mustacchi 
156*cd62a92dSRobert Mustacchi 		save = f;
157*cd62a92dSRobert Mustacchi 		f = fopen("/dev/zero", "r+");
158*cd62a92dSRobert Mustacchi 		if (f == NULL) {
159*cd62a92dSRobert Mustacchi 			err(EXIT_FAILURE, "failed to open /dev/zero");
160*cd62a92dSRobert Mustacchi 		}
161*cd62a92dSRobert Mustacchi 		check_file(f, fno_nextfd, "alternating memstream, fopen "
162*cd62a92dSRobert Mustacchi 		    "(file)");
163*cd62a92dSRobert Mustacchi 		fno_nextfd++;
164*cd62a92dSRobert Mustacchi 
165*cd62a92dSRobert Mustacchi 		f = open_wmemstream(&c, &s);
166*cd62a92dSRobert Mustacchi 		if (f == NULL) {
167*cd62a92dSRobert Mustacchi 			err(EXIT_FAILURE, "failed to create open_wmemstream() "
168*cd62a92dSRobert Mustacchi 			    "stream");
169*cd62a92dSRobert Mustacchi 		}
170*cd62a92dSRobert Mustacchi 		check_file(f, -1, "alternating memstream, fopen (wmemstream)");
171*cd62a92dSRobert Mustacchi 
172*cd62a92dSRobert Mustacchi 		f = freopen("/dev/null", "r+", save);
173*cd62a92dSRobert Mustacchi 		if (f == NULL) {
174*cd62a92dSRobert Mustacchi 			err(EXIT_FAILURE, "failed to freopen /dev/null from "
175*cd62a92dSRobert Mustacchi 			    "fmemopen()");
176*cd62a92dSRobert Mustacchi 		}
177*cd62a92dSRobert Mustacchi 		check_file(f, fno_nextfd, "alternating memstream, fopen "
178*cd62a92dSRobert Mustacchi 		    "(reopen)");
179*cd62a92dSRobert Mustacchi 
180*cd62a92dSRobert Mustacchi 		f = freopen("/dev/zero", "a+", f);
181*cd62a92dSRobert Mustacchi 		check_file(f, fno_nextfd, "alternating memstream, fopen "
182*cd62a92dSRobert Mustacchi 		    "(reopen file)");
183*cd62a92dSRobert Mustacchi 		fno_nextfd++;
184*cd62a92dSRobert Mustacchi 	}
185*cd62a92dSRobert Mustacchi }
186*cd62a92dSRobert Mustacchi 
187*cd62a92dSRobert Mustacchi int
main(void)188*cd62a92dSRobert Mustacchi main(void)
189*cd62a92dSRobert Mustacchi {
190*cd62a92dSRobert Mustacchi 	check_file(stdin, STDIN_FILENO, "default stdin fd is correct");
191*cd62a92dSRobert Mustacchi 	check_file(stdout, STDOUT_FILENO, "default stdout fd is correct");
192*cd62a92dSRobert Mustacchi 	check_file(stderr, STDERR_FILENO, "default stderr fd is correct");
193*cd62a92dSRobert Mustacchi 
194*cd62a92dSRobert Mustacchi 	/*
195*cd62a92dSRobert Mustacchi 	 * Establish our base fd. The test runner can open files on our behalf.
196*cd62a92dSRobert Mustacchi 	 */
197*cd62a92dSRobert Mustacchi 	fno_nextfd = open("/dev/null", O_RDONLY);
198*cd62a92dSRobert Mustacchi 	if (fno_nextfd < 0) {
199*cd62a92dSRobert Mustacchi 		err(EXIT_FAILURE, "failed to open /dev/null");
200*cd62a92dSRobert Mustacchi 	}
201*cd62a92dSRobert Mustacchi 	fno_nextfd++;
202*cd62a92dSRobert Mustacchi 	check_open_n(10);
203*cd62a92dSRobert Mustacchi 	fno_nextfd += 10;
204*cd62a92dSRobert Mustacchi 	check_memstream();
205*cd62a92dSRobert Mustacchi 	check_fdopen();
206*cd62a92dSRobert Mustacchi 	check_alternate();
207*cd62a92dSRobert Mustacchi 
208*cd62a92dSRobert Mustacchi 	printf("%d/%d tests passed\n", fno_ntests - fno_nfails, fno_ntests);
209*cd62a92dSRobert Mustacchi 	return (fno_nfails > 0 ? EXIT_FAILURE : EXIT_SUCCESS);
210*cd62a92dSRobert Mustacchi }
211