Blame gnulib-tests/test-openat.c

Packit 709fb3
/* Test that openat works.
Packit 709fb3
   Copyright (C) 2009-2017 Free Software Foundation, Inc.
Packit 709fb3
Packit 709fb3
   This program is free software: you can redistribute it and/or modify
Packit 709fb3
   it under the terms of the GNU General Public License as published by
Packit 709fb3
   the Free Software Foundation; either version 3 of the License, or
Packit 709fb3
   (at your option) any later version.
Packit 709fb3
Packit 709fb3
   This program is distributed in the hope that it will be useful,
Packit 709fb3
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 709fb3
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 709fb3
   GNU General Public License for more details.
Packit 709fb3
Packit 709fb3
   You should have received a copy of the GNU General Public License
Packit 709fb3
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 709fb3
Packit 709fb3
/* Written by Eric Blake <ebb9@byu.net>, 2009.  */
Packit 709fb3
Packit 709fb3
#include <config.h>
Packit 709fb3
Packit 709fb3
#include <fcntl.h>
Packit 709fb3
Packit 709fb3
#include "signature.h"
Packit 709fb3
SIGNATURE_CHECK (openat, int, (int, char const *, int, ...));
Packit 709fb3
Packit 709fb3
#include <errno.h>
Packit 709fb3
#include <stdarg.h>
Packit 709fb3
#include <stdbool.h>
Packit 709fb3
#include <stdio.h>
Packit 709fb3
#include <unistd.h>
Packit 709fb3
Packit 709fb3
#include "macros.h"
Packit 709fb3
Packit 709fb3
#define BASE "test-openat.t"
Packit 709fb3
Packit 709fb3
#include "test-open.h"
Packit 709fb3
Packit 709fb3
static int dfd = AT_FDCWD;
Packit 709fb3
Packit 709fb3
/* Wrapper around openat to test open behavior.  */
Packit 709fb3
static int
Packit 709fb3
do_open (char const *name, int flags, ...)
Packit 709fb3
{
Packit 709fb3
  if (flags & O_CREAT)
Packit 709fb3
    {
Packit 709fb3
      mode_t mode = 0;
Packit 709fb3
      va_list arg;
Packit 709fb3
      va_start (arg, flags);
Packit 709fb3
Packit 709fb3
      /* We have to use PROMOTED_MODE_T instead of mode_t, otherwise GCC 4
Packit 709fb3
         creates crashing code when 'mode_t' is smaller than 'int'.  */
Packit 709fb3
      mode = va_arg (arg, PROMOTED_MODE_T);
Packit 709fb3
Packit 709fb3
      va_end (arg);
Packit 709fb3
      return openat (dfd, name, flags, mode);
Packit 709fb3
    }
Packit 709fb3
  return openat (dfd, name, flags);
Packit 709fb3
}
Packit 709fb3
Packit 709fb3
int
Packit 709fb3
main (int argc _GL_UNUSED, char *argv[])
Packit 709fb3
{
Packit 709fb3
  int result;
Packit 709fb3
Packit 709fb3
  /* Test behaviour for invalid file descriptors.  */
Packit 709fb3
  {
Packit 709fb3
    errno = 0;
Packit 709fb3
    ASSERT (openat (-1, "foo", O_RDONLY) == -1);
Packit 709fb3
    ASSERT (errno == EBADF);
Packit 709fb3
  }
Packit 709fb3
  {
Packit 709fb3
    close (99);
Packit 709fb3
    errno = 0;
Packit 709fb3
    ASSERT (openat (99, "foo", O_RDONLY) == -1);
Packit 709fb3
    ASSERT (errno == EBADF);
Packit 709fb3
  }
Packit 709fb3
Packit 709fb3
  /* Basic checks.  */
Packit 709fb3
  result = test_open (do_open, false);
Packit 709fb3
  dfd = open (".", O_RDONLY);
Packit 709fb3
  ASSERT (0 <= dfd);
Packit 709fb3
  ASSERT (test_open (do_open, false) == result);
Packit 709fb3
  ASSERT (close (dfd) == 0);
Packit 709fb3
Packit 709fb3
  /* Check that even when *-safer modules are in use, plain openat can
Packit 709fb3
     land in fd 0.  Do this test last, since it is destructive to
Packit 709fb3
     stdin.  */
Packit 709fb3
  ASSERT (close (STDIN_FILENO) == 0);
Packit 709fb3
  ASSERT (openat (AT_FDCWD, ".", O_RDONLY) == STDIN_FILENO);
Packit 709fb3
  {
Packit 709fb3
    dfd = open (".", O_RDONLY);
Packit 709fb3
    ASSERT (STDIN_FILENO < dfd);
Packit 709fb3
    ASSERT (chdir ("..") == 0);
Packit 709fb3
    ASSERT (close (STDIN_FILENO) == 0);
Packit 709fb3
    ASSERT (openat (dfd, ".", O_RDONLY) == STDIN_FILENO);
Packit 709fb3
    ASSERT (close (dfd) == 0);
Packit 709fb3
  }
Packit 709fb3
  return result;
Packit 709fb3
}