Blame gnulib-tests/test-strerror_r.c

Packit 33f14e
/* Test of strerror_r() function.
Packit 33f14e
   Copyright (C) 2007-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 <string.h>
Packit 33f14e
Packit 33f14e
#include "signature.h"
Packit 33f14e
SIGNATURE_CHECK (strerror_r, int, (int, char *, size_t));
Packit 33f14e
Packit 33f14e
#include <errno.h>
Packit 33f14e
Packit 33f14e
#include "macros.h"
Packit 33f14e
Packit 33f14e
int
Packit 33f14e
main (void)
Packit 33f14e
{
Packit 33f14e
  char buf[100];
Packit 33f14e
  int ret;
Packit 33f14e
Packit 33f14e
  /* Test results with valid errnum and enough room.  */
Packit 33f14e
Packit 33f14e
  errno = 0;
Packit 33f14e
  buf[0] = '\0';
Packit 33f14e
  ASSERT (strerror_r (EACCES, buf, sizeof buf) == 0);
Packit 33f14e
  ASSERT (buf[0] != '\0');
Packit 33f14e
  ASSERT (errno == 0);
Packit 33f14e
  ASSERT (strlen (buf) < sizeof buf);
Packit 33f14e
Packit 33f14e
  errno = 0;
Packit 33f14e
  buf[0] = '\0';
Packit 33f14e
  ASSERT (strerror_r (ETIMEDOUT, buf, sizeof buf) == 0);
Packit 33f14e
  ASSERT (buf[0] != '\0');
Packit 33f14e
  ASSERT (errno == 0);
Packit 33f14e
  ASSERT (strlen (buf) < sizeof buf);
Packit 33f14e
Packit 33f14e
  errno = 0;
Packit 33f14e
  buf[0] = '\0';
Packit 33f14e
  ASSERT (strerror_r (EOVERFLOW, buf, sizeof buf) == 0);
Packit 33f14e
  ASSERT (buf[0] != '\0');
Packit 33f14e
  ASSERT (errno == 0);
Packit 33f14e
  ASSERT (strlen (buf) < sizeof buf);
Packit 33f14e
Packit 33f14e
  /* POSIX requires strerror (0) to succeed.  Reject use of "Unknown
Packit 33f14e
     error", but allow "Success", "No error", or even Solaris' "Error
Packit 33f14e
     0" which are distinct patterns from true out-of-range strings.
Packit 33f14e
     http://austingroupbugs.net/view.php?id=382  */
Packit 33f14e
  errno = 0;
Packit 33f14e
  buf[0] = '\0';
Packit 33f14e
  ret = strerror_r (0, buf, sizeof buf);
Packit 33f14e
  ASSERT (ret == 0);
Packit 33f14e
  ASSERT (buf[0]);
Packit 33f14e
  ASSERT (errno == 0);
Packit 33f14e
  ASSERT (strstr (buf, "nknown") == NULL);
Packit 33f14e
  ASSERT (strstr (buf, "ndefined") == NULL);
Packit 33f14e
Packit 33f14e
  /* Test results with out-of-range errnum and enough room.  POSIX
Packit 33f14e
     allows an empty string on success, and allows an unchanged buf on
Packit 33f14e
     error, but these are not useful, so we guarantee contents.  */
Packit 33f14e
  errno = 0;
Packit 33f14e
  buf[0] = '^';
Packit 33f14e
  ret = strerror_r (-3, buf, sizeof buf);
Packit 33f14e
  ASSERT (ret == 0 || ret == EINVAL);
Packit 33f14e
  ASSERT (buf[0] != '^');
Packit 33f14e
  ASSERT (*buf);
Packit 33f14e
  ASSERT (errno == 0);
Packit 33f14e
  ASSERT (strlen (buf) < sizeof buf);
Packit 33f14e
Packit 33f14e
  /* Test results with a too small buffer.  POSIX requires an error;
Packit 33f14e
     only ERANGE for 0 and valid errors, and a choice of ERANGE or
Packit 33f14e
     EINVAL for out-of-range values.  On error, POSIX permits buf to
Packit 33f14e
     be empty, unchanged, or unterminated, but these are not useful,
Packit 33f14e
     so we guarantee NUL-terminated truncated contents for all but
Packit 33f14e
     size 0.  http://austingroupbugs.net/view.php?id=398.  Also ensure
Packit 33f14e
     that no out-of-bounds writes occur.  */
Packit 33f14e
  {
Packit 33f14e
    int errs[] = { EACCES, 0, -3, };
Packit 33f14e
    int j;
Packit 33f14e
Packit 33f14e
    buf[sizeof buf - 1] = '\0';
Packit 33f14e
    for (j = 0; j < SIZEOF (errs); j++)
Packit 33f14e
      {
Packit 33f14e
        int err = errs[j];
Packit 33f14e
        char buf2[sizeof buf] = "";
Packit 33f14e
        size_t len;
Packit 33f14e
        size_t i;
Packit 33f14e
Packit 33f14e
        strerror_r (err, buf2, sizeof buf2);
Packit 33f14e
        len = strlen (buf2);
Packit 33f14e
        ASSERT (len < sizeof buf);
Packit 33f14e
Packit 33f14e
        for (i = 0; i <= len; i++)
Packit 33f14e
          {
Packit 33f14e
            memset (buf, '^', sizeof buf - 1);
Packit 33f14e
            errno = 0;
Packit 33f14e
            ret = strerror_r (err, buf, i);
Packit 33f14e
            ASSERT (errno == 0);
Packit 33f14e
            if (err < 0)
Packit 33f14e
              ASSERT (ret == ERANGE || ret == EINVAL);
Packit 33f14e
            else
Packit 33f14e
              ASSERT (ret == ERANGE);
Packit 33f14e
            if (i)
Packit 33f14e
              {
Packit 33f14e
                ASSERT (strncmp (buf, buf2, i - 1) == 0);
Packit 33f14e
                ASSERT (buf[i - 1] == '\0');
Packit 33f14e
              }
Packit 33f14e
            ASSERT (strspn (buf + i, "^") == sizeof buf - 1 - i);
Packit 33f14e
          }
Packit 33f14e
Packit 33f14e
        strcpy (buf, "BADFACE");
Packit 33f14e
        errno = 0;
Packit 33f14e
        ret = strerror_r (err, buf, len + 1);
Packit 33f14e
        ASSERT (ret != ERANGE);
Packit 33f14e
        ASSERT (errno == 0);
Packit 33f14e
        ASSERT (strcmp (buf, buf2) == 0);
Packit 33f14e
      }
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
#if GNULIB_STRERROR
Packit 33f14e
  /* Test that strerror_r does not clobber strerror buffer.  On some
Packit 33f14e
     platforms, this test can only succeed if gnulib also replaces
Packit 33f14e
     strerror.  */
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
    strerror_r (EACCES, buf, sizeof buf);
Packit 33f14e
    strerror_r (-5, buf, sizeof buf);
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
#endif
Packit 33f14e
Packit 33f14e
  return 0;
Packit 33f14e
}