1 /* Test of fputc() function.
2 Copyright (C) 2011-2023 Free Software Foundation, Inc.
3
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 3, or (at your option)
7 any later version.
8
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
13
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, see <https://www.gnu.org/licenses/>. */
16
17 #include <config.h>
18
19 #include <stdio.h>
20
21 #include "signature.h"
22 SIGNATURE_CHECK (fputc, int, (int, FILE *));
23
24 #include <errno.h>
25 #include <fcntl.h>
26 #include <unistd.h>
27
28 #if HAVE_MSVC_INVALID_PARAMETER_HANDLER
29 # include "msvc-inval.h"
30 #endif
31
32 #include "macros.h"
33
34 int
35 main (int argc, char **argv)
36 {
37 const char *filename = "test-fputc.txt";
38
39 /* We don't have an fputc() function that installs an invalid parameter
40 handler so far. So install that handler here, explicitly. */
41 #if HAVE_MSVC_INVALID_PARAMETER_HANDLER \
42 && MSVC_INVALID_PARAMETER_HANDLING == DEFAULT_HANDLING
43 gl_msvc_inval_ensure_handler ();
44 #endif
45
46 /* Test that fputc() on an unbuffered stream sets errno if someone else
47 closes the stream fd behind the back of stdio. */
48 #if !defined __ANDROID__ /* fdsan */
49 {
50 FILE *fp = fopen (filename, "w");
51 ASSERT (fp != NULL);
52 setvbuf (fp, NULL, _IONBF, 0);
53 ASSERT (close (fileno (fp)) == 0);
54 errno = 0;
55 ASSERT (fputc ('x', fp) == EOF);
56 ASSERT (errno == EBADF);
57 ASSERT (ferror (fp));
58 fclose (fp);
59 }
60 #endif
61
62 /* Test that fputc() on an unbuffered stream sets errno if the stream
63 was constructed with an invalid file descriptor. */
64 {
65 FILE *fp = fdopen (-1, "w");
66 if (fp != NULL)
67 {
68 setvbuf (fp, NULL, _IONBF, 0);
69 errno = 0;
70 ASSERT (fputc ('x', fp) == EOF);
71 ASSERT (errno == EBADF);
72 ASSERT (ferror (fp));
73 fclose (fp);
74 }
75 }
76 {
77 FILE *fp;
78 close (99);
79 fp = fdopen (99, "w");
80 if (fp != NULL)
81 {
82 setvbuf (fp, NULL, _IONBF, 0);
83 errno = 0;
84 ASSERT (fputc ('x', fp) == EOF);
85 ASSERT (errno == EBADF);
86 ASSERT (ferror (fp));
87 fclose (fp);
88 }
89 }
90
91 /* Clean up. */
92 unlink (filename);
93
94 return 0;
95 }