Blame gnulib-tests/test-strerror_r.c

Packit Service fdd496
/* Test of strerror_r() function.
Packit Service fdd496
   Copyright (C) 2007-2017 Free Software Foundation, Inc.
Packit Service fdd496
Packit Service fdd496
   This program is free software; you can redistribute it and/or modify
Packit Service fdd496
   it under the terms of the GNU General Public License as published by
Packit Service fdd496
   the Free Software Foundation; either version 3, or (at your option)
Packit Service fdd496
   any later version.
Packit Service fdd496
Packit Service fdd496
   This program is distributed in the hope that it will be useful,
Packit Service fdd496
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service fdd496
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service fdd496
   GNU General Public License for more details.
Packit Service fdd496
Packit Service fdd496
   You should have received a copy of the GNU General Public License
Packit Service fdd496
   along with this program; if not, see <http://www.gnu.org/licenses/>.  */
Packit Service fdd496
Packit Service fdd496
#include <config.h>
Packit Service fdd496
Packit Service fdd496
#include <string.h>
Packit Service fdd496
Packit Service fdd496
#include "signature.h"
Packit Service fdd496
SIGNATURE_CHECK (strerror_r, int, (int, char *, size_t));
Packit Service fdd496
Packit Service fdd496
#include <errno.h>
Packit Service fdd496
Packit Service fdd496
#include "macros.h"
Packit Service fdd496
Packit Service fdd496
int
Packit Service fdd496
main (void)
Packit Service fdd496
{
Packit Service fdd496
  char buf[100];
Packit Service fdd496
  int ret;
Packit Service fdd496
Packit Service fdd496
  /* Test results with valid errnum and enough room.  */
Packit Service fdd496
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  buf[0] = '\0';
Packit Service fdd496
  ASSERT (strerror_r (EACCES, buf, sizeof buf) == 0);
Packit Service fdd496
  ASSERT (buf[0] != '\0');
Packit Service fdd496
  ASSERT (errno == 0);
Packit Service fdd496
  ASSERT (strlen (buf) < sizeof buf);
Packit Service fdd496
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  buf[0] = '\0';
Packit Service fdd496
  ASSERT (strerror_r (ETIMEDOUT, buf, sizeof buf) == 0);
Packit Service fdd496
  ASSERT (buf[0] != '\0');
Packit Service fdd496
  ASSERT (errno == 0);
Packit Service fdd496
  ASSERT (strlen (buf) < sizeof buf);
Packit Service fdd496
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  buf[0] = '\0';
Packit Service fdd496
  ASSERT (strerror_r (EOVERFLOW, buf, sizeof buf) == 0);
Packit Service fdd496
  ASSERT (buf[0] != '\0');
Packit Service fdd496
  ASSERT (errno == 0);
Packit Service fdd496
  ASSERT (strlen (buf) < sizeof buf);
Packit Service fdd496
Packit Service fdd496
  /* POSIX requires strerror (0) to succeed.  Reject use of "Unknown
Packit Service fdd496
     error", but allow "Success", "No error", or even Solaris' "Error
Packit Service fdd496
     0" which are distinct patterns from true out-of-range strings.
Packit Service fdd496
     http://austingroupbugs.net/view.php?id=382  */
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  buf[0] = '\0';
Packit Service fdd496
  ret = strerror_r (0, buf, sizeof buf);
Packit Service fdd496
  ASSERT (ret == 0);
Packit Service fdd496
  ASSERT (buf[0]);
Packit Service fdd496
  ASSERT (errno == 0);
Packit Service fdd496
  ASSERT (strstr (buf, "nknown") == NULL);
Packit Service fdd496
  ASSERT (strstr (buf, "ndefined") == NULL);
Packit Service fdd496
Packit Service fdd496
  /* Test results with out-of-range errnum and enough room.  POSIX
Packit Service fdd496
     allows an empty string on success, and allows an unchanged buf on
Packit Service fdd496
     error, but these are not useful, so we guarantee contents.  */
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  buf[0] = '^';
Packit Service fdd496
  ret = strerror_r (-3, buf, sizeof buf);
Packit Service fdd496
  ASSERT (ret == 0 || ret == EINVAL);
Packit Service fdd496
  ASSERT (buf[0] != '^');
Packit Service fdd496
  ASSERT (*buf);
Packit Service fdd496
  ASSERT (errno == 0);
Packit Service fdd496
  ASSERT (strlen (buf) < sizeof buf);
Packit Service fdd496
Packit Service fdd496
  /* Test results with a too small buffer.  POSIX requires an error;
Packit Service fdd496
     only ERANGE for 0 and valid errors, and a choice of ERANGE or
Packit Service fdd496
     EINVAL for out-of-range values.  On error, POSIX permits buf to
Packit Service fdd496
     be empty, unchanged, or unterminated, but these are not useful,
Packit Service fdd496
     so we guarantee NUL-terminated truncated contents for all but
Packit Service fdd496
     size 0.  http://austingroupbugs.net/view.php?id=398.  Also ensure
Packit Service fdd496
     that no out-of-bounds writes occur.  */
Packit Service fdd496
  {
Packit Service fdd496
    int errs[] = { EACCES, 0, -3, };
Packit Service fdd496
    int j;
Packit Service fdd496
Packit Service fdd496
    buf[sizeof buf - 1] = '\0';
Packit Service fdd496
    for (j = 0; j < SIZEOF (errs); j++)
Packit Service fdd496
      {
Packit Service fdd496
        int err = errs[j];
Packit Service fdd496
        char buf2[sizeof buf] = "";
Packit Service fdd496
        size_t len;
Packit Service fdd496
        size_t i;
Packit Service fdd496
Packit Service fdd496
        strerror_r (err, buf2, sizeof buf2);
Packit Service fdd496
        len = strlen (buf2);
Packit Service fdd496
        ASSERT (len < sizeof buf);
Packit Service fdd496
Packit Service fdd496
        for (i = 0; i <= len; i++)
Packit Service fdd496
          {
Packit Service fdd496
            memset (buf, '^', sizeof buf - 1);
Packit Service fdd496
            errno = 0;
Packit Service fdd496
            ret = strerror_r (err, buf, i);
Packit Service fdd496
            ASSERT (errno == 0);
Packit Service fdd496
            if (err < 0)
Packit Service fdd496
              ASSERT (ret == ERANGE || ret == EINVAL);
Packit Service fdd496
            else
Packit Service fdd496
              ASSERT (ret == ERANGE);
Packit Service fdd496
            if (i)
Packit Service fdd496
              {
Packit Service fdd496
                ASSERT (strncmp (buf, buf2, i - 1) == 0);
Packit Service fdd496
                ASSERT (buf[i - 1] == '\0');
Packit Service fdd496
              }
Packit Service fdd496
            ASSERT (strspn (buf + i, "^") == sizeof buf - 1 - i);
Packit Service fdd496
          }
Packit Service fdd496
Packit Service fdd496
        strcpy (buf, "BADFACE");
Packit Service fdd496
        errno = 0;
Packit Service fdd496
        ret = strerror_r (err, buf, len + 1);
Packit Service fdd496
        ASSERT (ret != ERANGE);
Packit Service fdd496
        ASSERT (errno == 0);
Packit Service fdd496
        ASSERT (strcmp (buf, buf2) == 0);
Packit Service fdd496
      }
Packit Service fdd496
  }
Packit Service fdd496
Packit Service fdd496
#if GNULIB_STRERROR
Packit Service fdd496
  /* Test that strerror_r does not clobber strerror buffer.  On some
Packit Service fdd496
     platforms, this test can only succeed if gnulib also replaces
Packit Service fdd496
     strerror.  */
Packit Service fdd496
  {
Packit Service fdd496
    const char *msg1;
Packit Service fdd496
    const char *msg2;
Packit Service fdd496
    const char *msg3;
Packit Service fdd496
    const char *msg4;
Packit Service fdd496
    char *str1;
Packit Service fdd496
    char *str2;
Packit Service fdd496
    char *str3;
Packit Service fdd496
    char *str4;
Packit Service fdd496
Packit Service fdd496
    msg1 = strerror (ENOENT);
Packit Service fdd496
    ASSERT (msg1);
Packit Service fdd496
    str1 = strdup (msg1);
Packit Service fdd496
    ASSERT (str1);
Packit Service fdd496
Packit Service fdd496
    msg2 = strerror (ERANGE);
Packit Service fdd496
    ASSERT (msg2);
Packit Service fdd496
    str2 = strdup (msg2);
Packit Service fdd496
    ASSERT (str2);
Packit Service fdd496
Packit Service fdd496
    msg3 = strerror (-4);
Packit Service fdd496
    ASSERT (msg3);
Packit Service fdd496
    str3 = strdup (msg3);
Packit Service fdd496
    ASSERT (str3);
Packit Service fdd496
Packit Service fdd496
    msg4 = strerror (1729576);
Packit Service fdd496
    ASSERT (msg4);
Packit Service fdd496
    str4 = strdup (msg4);
Packit Service fdd496
    ASSERT (str4);
Packit Service fdd496
Packit Service fdd496
    strerror_r (EACCES, buf, sizeof buf);
Packit Service fdd496
    strerror_r (-5, buf, sizeof buf);
Packit Service fdd496
    ASSERT (msg1 == msg2 || msg1 == msg4 || STREQ (msg1, str1));
Packit Service fdd496
    ASSERT (msg2 == msg4 || STREQ (msg2, str2));
Packit Service fdd496
    ASSERT (msg3 == msg4 || STREQ (msg3, str3));
Packit Service fdd496
    ASSERT (STREQ (msg4, str4));
Packit Service fdd496
Packit Service fdd496
    free (str1);
Packit Service fdd496
    free (str2);
Packit Service fdd496
    free (str3);
Packit Service fdd496
    free (str4);
Packit Service fdd496
  }
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
  return 0;
Packit Service fdd496
}