Blame gnulib-tests/test-perror2.c

Packit 33f14e
/* Test of perror() function.
Packit 33f14e
   Copyright (C) 2011-2017 Free Software Foundation, Inc.
Packit 33f14e
Packit 33f14e
   This program is free software; you can redistribute it and/or modify
Packit 33f14e
   it under the terms of the GNU General Public License as published by
Packit 33f14e
   the Free Software Foundation; either version 3, or (at your option)
Packit 33f14e
   any later version.
Packit 33f14e
Packit 33f14e
   This program is distributed in the hope that it will be useful,
Packit 33f14e
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 33f14e
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 33f14e
   GNU General Public License for more details.
Packit 33f14e
Packit 33f14e
   You should have received a copy of the GNU General Public License
Packit 33f14e
   along with this program; if not, see <http://www.gnu.org/licenses/>.  */
Packit 33f14e
Packit 33f14e
#include <config.h>
Packit 33f14e
Packit 33f14e
#include <stdio.h>
Packit 33f14e
Packit 33f14e
#include <errno.h>
Packit 33f14e
#include <string.h>
Packit 33f14e
#include <unistd.h>
Packit 33f14e
Packit 33f14e
/* This test intentionally parses stderr.  So, we arrange to have fd 10
Packit 33f14e
   (outside the range of interesting fd's during the test) set up to
Packit 33f14e
   duplicate the original stderr.  */
Packit 33f14e
#define BACKUP_STDERR_FILENO 10
Packit 33f14e
#define ASSERT_STREAM myerr
Packit 33f14e
#include "macros.h"
Packit 33f14e
Packit 33f14e
static FILE *myerr;
Packit 33f14e
Packit 33f14e
#define BASE "test-perror2"
Packit 33f14e
Packit 33f14e
int
Packit 33f14e
main (void)
Packit 33f14e
{
Packit 33f14e
  /* We change fd 2 later, so save it in fd 10.  */
Packit 33f14e
  if (dup2 (STDERR_FILENO, BACKUP_STDERR_FILENO) != BACKUP_STDERR_FILENO
Packit 33f14e
      || (myerr = fdopen (BACKUP_STDERR_FILENO, "w")) == NULL)
Packit 33f14e
    return 2;
Packit 33f14e
Packit 33f14e
  ASSERT (freopen (BASE ".tmp", "w+", stderr) == stderr);
Packit 33f14e
Packit 33f14e
  /* Test that perror does not clobber strerror buffer.  */
Packit 33f14e
  {
Packit 33f14e
    const char *msg1;
Packit 33f14e
    const char *msg2;
Packit 33f14e
    const char *msg3;
Packit 33f14e
    const char *msg4;
Packit 33f14e
    char *str1;
Packit 33f14e
    char *str2;
Packit 33f14e
    char *str3;
Packit 33f14e
    char *str4;
Packit 33f14e
Packit 33f14e
    msg1 = strerror (ENOENT);
Packit 33f14e
    ASSERT (msg1);
Packit 33f14e
    str1 = strdup (msg1);
Packit 33f14e
    ASSERT (str1);
Packit 33f14e
Packit 33f14e
    msg2 = strerror (ERANGE);
Packit 33f14e
    ASSERT (msg2);
Packit 33f14e
    str2 = strdup (msg2);
Packit 33f14e
    ASSERT (str2);
Packit 33f14e
Packit 33f14e
    msg3 = strerror (-4);
Packit 33f14e
    ASSERT (msg3);
Packit 33f14e
    str3 = strdup (msg3);
Packit 33f14e
    ASSERT (str3);
Packit 33f14e
Packit 33f14e
    msg4 = strerror (1729576);
Packit 33f14e
    ASSERT (msg4);
Packit 33f14e
    str4 = strdup (msg4);
Packit 33f14e
    ASSERT (str4);
Packit 33f14e
Packit 33f14e
    errno = EACCES;
Packit 33f14e
    perror ("");
Packit 33f14e
    errno = -5;
Packit 33f14e
    perror ("");
Packit 33f14e
    ASSERT (!ferror (stderr));
Packit 33f14e
    ASSERT (msg1 == msg2 || msg1 == msg4 || STREQ (msg1, str1));
Packit 33f14e
    ASSERT (msg2 == msg4 || STREQ (msg2, str2));
Packit 33f14e
    ASSERT (msg3 == msg4 || STREQ (msg3, str3));
Packit 33f14e
    ASSERT (STREQ (msg4, str4));
Packit 33f14e
Packit 33f14e
    free (str1);
Packit 33f14e
    free (str2);
Packit 33f14e
    free (str3);
Packit 33f14e
    free (str4);
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  /* Test that perror uses the same message as strerror.  */
Packit 33f14e
  {
Packit 33f14e
    int errs[] = { EACCES, 0, -3, };
Packit 33f14e
    int i;
Packit 33f14e
    for (i = 0; i < SIZEOF (errs); i++)
Packit 33f14e
      {
Packit 33f14e
        char buf[256];
Packit 33f14e
        char *err = strerror (errs[i]);
Packit 33f14e
Packit 33f14e
        ASSERT (err);
Packit 33f14e
        ASSERT (strlen (err) < sizeof buf);
Packit 33f14e
        rewind (stderr);
Packit 33f14e
        ASSERT (ftruncate (fileno (stderr), 0) == 0);
Packit 33f14e
        errno = errs[i];
Packit 33f14e
        perror (NULL);
Packit 33f14e
        ASSERT (!ferror (stderr));
Packit 33f14e
        rewind (stderr);
Packit 33f14e
        ASSERT (fgets (buf, sizeof buf, stderr) == buf);
Packit 33f14e
        ASSERT (strstr (buf, err));
Packit 33f14e
      }
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  /* Test that perror reports write failure.  */
Packit 33f14e
  {
Packit 33f14e
    ASSERT (freopen (BASE ".tmp", "r", stderr) == stderr);
Packit 33f14e
    ASSERT (setvbuf (stderr, NULL, _IONBF, BUFSIZ) == 0);
Packit 33f14e
    errno = -1;
Packit 33f14e
    ASSERT (!ferror (stderr));
Packit 33f14e
    perror (NULL);
Packit 33f14e
#if 0
Packit 33f14e
    /* Commented out until cygwin behaves:
Packit 33f14e
       http://sourceware.org/ml/newlib/2011/msg00228.html */
Packit 33f14e
    ASSERT (errno > 0);
Packit 33f14e
    /* Commented out until glibc behaves:
Packit 33f14e
       http://sourceware.org/bugzilla/show_bug.cgi?id=12792 */
Packit 33f14e
    ASSERT (ferror (stderr));
Packit 33f14e
#endif
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  ASSERT (fclose (stderr) == 0);
Packit 33f14e
  ASSERT (remove (BASE ".tmp") == 0);
Packit 33f14e
Packit 33f14e
  return 0;
Packit 33f14e
}