(root)/
gettext-0.22.4/
gettext-tools/
gnulib-tests/
test-fputc.c
       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 ()
      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  }