Blame src/unix/fs.c

Packit b5b901
/* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
Packit b5b901
 *
Packit b5b901
 * Permission is hereby granted, free of charge, to any person obtaining a copy
Packit b5b901
 * of this software and associated documentation files (the "Software"), to
Packit b5b901
 * deal in the Software without restriction, including without limitation the
Packit b5b901
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
Packit b5b901
 * sell copies of the Software, and to permit persons to whom the Software is
Packit b5b901
 * furnished to do so, subject to the following conditions:
Packit b5b901
 *
Packit b5b901
 * The above copyright notice and this permission notice shall be included in
Packit b5b901
 * all copies or substantial portions of the Software.
Packit b5b901
 *
Packit b5b901
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit b5b901
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit b5b901
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
Packit b5b901
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
Packit b5b901
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
Packit b5b901
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
Packit b5b901
 * IN THE SOFTWARE.
Packit b5b901
 */
Packit b5b901
Packit b5b901
/* Caveat emptor: this file deviates from the libuv convention of returning
Packit b5b901
 * negated errno codes. Most uv_fs_*() functions map directly to the system
Packit b5b901
 * call of the same name. For more complex wrappers, it's easier to just
Packit b5b901
 * return -1 with errno set. The dispatcher in uv__fs_work() takes care of
Packit b5b901
 * getting the errno to the right place (req->result or as the return value.)
Packit b5b901
 */
Packit b5b901
Packit b5b901
#include "uv.h"
Packit b5b901
#include "internal.h"
Packit b5b901
Packit b5b901
#include <errno.h>
Packit Service e08953
#include <dlfcn.h>
Packit b5b901
#include <stdio.h>
Packit b5b901
#include <stdlib.h>
Packit b5b901
#include <string.h>
Packit b5b901
#include <limits.h> /* PATH_MAX */
Packit b5b901
Packit b5b901
#include <sys/types.h>
Packit b5b901
#include <sys/socket.h>
Packit b5b901
#include <sys/stat.h>
Packit b5b901
#include <sys/time.h>
Packit b5b901
#include <sys/uio.h>
Packit b5b901
#include <pthread.h>
Packit b5b901
#include <unistd.h>
Packit b5b901
#include <fcntl.h>
Packit b5b901
#include <poll.h>
Packit b5b901
Packit b5b901
#if defined(__DragonFly__)        ||                                      \
Packit b5b901
    defined(__FreeBSD__)          ||                                      \
Packit Service e08953
    defined(__FreeBSD_kernel__)   ||                                      \
Packit b5b901
    defined(__OpenBSD__)          ||                                      \
Packit b5b901
    defined(__NetBSD__)
Packit b5b901
# define HAVE_PREADV 1
Packit b5b901
#else
Packit b5b901
# define HAVE_PREADV 0
Packit b5b901
#endif
Packit b5b901
Packit b5b901
#if defined(__linux__) || defined(__sun)
Packit b5b901
# include <sys/sendfile.h>
Packit b5b901
#endif
Packit b5b901
Packit b5b901
#if defined(__APPLE__)
Packit Service e08953
# include <sys/sysctl.h>
Packit b5b901
#elif defined(__linux__) && !defined(FICLONE)
Packit b5b901
# include <sys/ioctl.h>
Packit b5b901
# define FICLONE _IOW(0x94, 9, int)
Packit b5b901
#endif
Packit b5b901
Packit b5b901
#if defined(_AIX) && !defined(_AIX71)
Packit b5b901
# include <utime.h>
Packit b5b901
#endif
Packit b5b901
Packit Service e08953
#if defined(__APPLE__)            ||                                      \
Packit Service e08953
    defined(__DragonFly__)        ||                                      \
Packit Service e08953
    defined(__FreeBSD__)          ||                                      \
Packit Service e08953
    defined(__FreeBSD_kernel__)   ||                                      \
Packit Service e08953
    defined(__OpenBSD__)          ||                                      \
Packit Service e08953
    defined(__NetBSD__)
Packit Service e08953
# include <sys/param.h>
Packit Service e08953
# include <sys/mount.h>
Packit Service e08953
#elif defined(__sun) || defined(__MVS__) || defined(__NetBSD__) || defined(__HAIKU__)
Packit Service e08953
# include <sys/statvfs.h>
Packit Service e08953
#else
Packit Service e08953
# include <sys/statfs.h>
Packit Service e08953
#endif
Packit Service e08953
Packit Service e08953
#if defined(_AIX) && _XOPEN_SOURCE <= 600
Packit Service e08953
extern char *mkdtemp(char *template); /* See issue #740 on AIX < 7 */
Packit Service e08953
#endif
Packit Service e08953
Packit b5b901
#define INIT(subtype)                                                         \
Packit b5b901
  do {                                                                        \
Packit b5b901
    if (req == NULL)                                                          \
Packit b5b901
      return UV_EINVAL;                                                       \
Packit b5b901
    UV_REQ_INIT(req, UV_FS);                                                  \
Packit b5b901
    req->fs_type = UV_FS_ ## subtype;                                         \
Packit b5b901
    req->result = 0;                                                          \
Packit b5b901
    req->ptr = NULL;                                                          \
Packit b5b901
    req->loop = loop;                                                         \
Packit b5b901
    req->path = NULL;                                                         \
Packit b5b901
    req->new_path = NULL;                                                     \
Packit b5b901
    req->bufs = NULL;                                                         \
Packit b5b901
    req->cb = cb;                                                             \
Packit b5b901
  }                                                                           \
Packit b5b901
  while (0)
Packit b5b901
Packit b5b901
#define PATH                                                                  \
Packit b5b901
  do {                                                                        \
Packit b5b901
    assert(path != NULL);                                                     \
Packit b5b901
    if (cb == NULL) {                                                         \
Packit b5b901
      req->path = path;                                                       \
Packit b5b901
    } else {                                                                  \
Packit b5b901
      req->path = uv__strdup(path);                                           \
Packit b5b901
      if (req->path == NULL)                                                  \
Packit b5b901
        return UV_ENOMEM;                                                     \
Packit b5b901
    }                                                                         \
Packit b5b901
  }                                                                           \
Packit b5b901
  while (0)
Packit b5b901
Packit b5b901
#define PATH2                                                                 \
Packit b5b901
  do {                                                                        \
Packit b5b901
    if (cb == NULL) {                                                         \
Packit b5b901
      req->path = path;                                                       \
Packit b5b901
      req->new_path = new_path;                                               \
Packit b5b901
    } else {                                                                  \
Packit b5b901
      size_t path_len;                                                        \
Packit b5b901
      size_t new_path_len;                                                    \
Packit b5b901
      path_len = strlen(path) + 1;                                            \
Packit b5b901
      new_path_len = strlen(new_path) + 1;                                    \
Packit b5b901
      req->path = uv__malloc(path_len + new_path_len);                        \
Packit b5b901
      if (req->path == NULL)                                                  \
Packit b5b901
        return UV_ENOMEM;                                                     \
Packit b5b901
      req->new_path = req->path + path_len;                                   \
Packit b5b901
      memcpy((void*) req->path, path, path_len);                              \
Packit b5b901
      memcpy((void*) req->new_path, new_path, new_path_len);                  \
Packit b5b901
    }                                                                         \
Packit b5b901
  }                                                                           \
Packit b5b901
  while (0)
Packit b5b901
Packit b5b901
#define POST                                                                  \
Packit b5b901
  do {                                                                        \
Packit b5b901
    if (cb != NULL) {                                                         \
Packit b5b901
      uv__req_register(loop, req);                                            \
Packit b5b901
      uv__work_submit(loop,                                                   \
Packit b5b901
                      &req->work_req,                                         \
Packit b5b901
                      UV__WORK_FAST_IO,                                       \
Packit b5b901
                      uv__fs_work,                                            \
Packit b5b901
                      uv__fs_done);                                           \
Packit b5b901
      return 0;                                                               \
Packit b5b901
    }                                                                         \
Packit b5b901
    else {                                                                    \
Packit b5b901
      uv__fs_work(&req->work_req);                                            \
Packit b5b901
      return req->result;                                                     \
Packit b5b901
    }                                                                         \
Packit b5b901
  }                                                                           \
Packit b5b901
  while (0)
Packit b5b901
Packit b5b901
Packit Service e08953
static int uv__fs_close(int fd) {
Packit Service e08953
  int rc;
Packit Service e08953
Packit Service e08953
  rc = uv__close_nocancel(fd);
Packit Service e08953
  if (rc == -1)
Packit Service e08953
    if (errno == EINTR || errno == EINPROGRESS)
Packit Service e08953
      rc = 0;  /* The close is in progress, not an error. */
Packit Service e08953
Packit Service e08953
  return rc;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit b5b901
static ssize_t uv__fs_fsync(uv_fs_t* req) {
Packit b5b901
#if defined(__APPLE__)
Packit b5b901
  /* Apple's fdatasync and fsync explicitly do NOT flush the drive write cache
Packit b5b901
   * to the drive platters. This is in contrast to Linux's fdatasync and fsync
Packit b5b901
   * which do, according to recent man pages. F_FULLFSYNC is Apple's equivalent
Packit b5b901
   * for flushing buffered data to permanent storage. If F_FULLFSYNC is not
Packit Service e08953
   * supported by the file system we fall back to F_BARRIERFSYNC or fsync().
Packit Service e08953
   * This is the same approach taken by sqlite, except sqlite does not issue
Packit Service e08953
   * an F_BARRIERFSYNC call.
Packit b5b901
   */
Packit b5b901
  int r;
Packit b5b901
Packit b5b901
  r = fcntl(req->file, F_FULLFSYNC);
Packit Service e08953
  if (r != 0)
Packit Service e08953
    r = fcntl(req->file, 85 /* F_BARRIERFSYNC */);  /* fsync + barrier */
Packit Service e08953
  if (r != 0)
Packit b5b901
    r = fsync(req->file);
Packit b5b901
  return r;
Packit b5b901
#else
Packit b5b901
  return fsync(req->file);
Packit b5b901
#endif
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static ssize_t uv__fs_fdatasync(uv_fs_t* req) {
Packit b5b901
#if defined(__linux__) || defined(__sun) || defined(__NetBSD__)
Packit b5b901
  return fdatasync(req->file);
Packit b5b901
#elif defined(__APPLE__)
Packit b5b901
  /* See the comment in uv__fs_fsync. */
Packit b5b901
  return uv__fs_fsync(req);
Packit b5b901
#else
Packit b5b901
  return fsync(req->file);
Packit b5b901
#endif
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit Service e08953
UV_UNUSED(static struct timespec uv__fs_to_timespec(double time)) {
Packit Service e08953
  struct timespec ts;
Packit Service e08953
  ts.tv_sec  = time;
Packit Service e08953
  ts.tv_nsec = (uint64_t)(time * 1000000) % 1000000 * 1000;
Packit Service e08953
  return ts;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
UV_UNUSED(static struct timeval uv__fs_to_timeval(double time)) {
Packit Service e08953
  struct timeval tv;
Packit Service e08953
  tv.tv_sec  = time;
Packit Service e08953
  tv.tv_usec = (uint64_t)(time * 1000000) % 1000000;
Packit Service e08953
  return tv;
Packit Service e08953
}
Packit Service e08953
Packit b5b901
static ssize_t uv__fs_futime(uv_fs_t* req) {
Packit b5b901
#if defined(__linux__)                                                        \
Packit Service e08953
    || defined(_AIX71)                                                        \
Packit Service e08953
    || defined(__HAIKU__)
Packit b5b901
  /* utimesat() has nanosecond resolution but we stick to microseconds
Packit b5b901
   * for the sake of consistency with other platforms.
Packit b5b901
   */
Packit b5b901
  struct timespec ts[2];
Packit Service e08953
  ts[0] = uv__fs_to_timespec(req->atime);
Packit Service e08953
  ts[1] = uv__fs_to_timespec(req->mtime);
Packit Service e08953
#if defined(__ANDROID_API__) && __ANDROID_API__ < 21
Packit Service e08953
  return utimensat(req->file, NULL, ts, 0);
Packit Service e08953
#else
Packit b5b901
  return futimens(req->file, ts);
Packit Service e08953
#endif
Packit b5b901
#elif defined(__APPLE__)                                                      \
Packit b5b901
    || defined(__DragonFly__)                                                 \
Packit b5b901
    || defined(__FreeBSD__)                                                   \
Packit b5b901
    || defined(__FreeBSD_kernel__)                                            \
Packit b5b901
    || defined(__NetBSD__)                                                    \
Packit b5b901
    || defined(__OpenBSD__)                                                   \
Packit b5b901
    || defined(__sun)
Packit b5b901
  struct timeval tv[2];
Packit Service e08953
  tv[0] = uv__fs_to_timeval(req->atime);
Packit Service e08953
  tv[1] = uv__fs_to_timeval(req->mtime);
Packit b5b901
# if defined(__sun)
Packit b5b901
  return futimesat(req->file, NULL, tv);
Packit b5b901
# else
Packit b5b901
  return futimes(req->file, tv);
Packit b5b901
# endif
Packit b5b901
#elif defined(__MVS__)
Packit b5b901
  attrib_t atr;
Packit b5b901
  memset(&atr, 0, sizeof(atr));
Packit b5b901
  atr.att_mtimechg = 1;
Packit b5b901
  atr.att_atimechg = 1;
Packit b5b901
  atr.att_mtime = req->mtime;
Packit b5b901
  atr.att_atime = req->atime;
Packit b5b901
  return __fchattr(req->file, &atr, sizeof(atr));
Packit b5b901
#else
Packit b5b901
  errno = ENOSYS;
Packit b5b901
  return -1;
Packit b5b901
#endif
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static ssize_t uv__fs_mkdtemp(uv_fs_t* req) {
Packit b5b901
  return mkdtemp((char*) req->path) ? 0 : -1;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit Service e08953
static int (*uv__mkostemp)(char*, int);
Packit Service e08953
Packit Service e08953
Packit Service e08953
static void uv__mkostemp_initonce(void) {
Packit Service e08953
  /* z/os doesn't have RTLD_DEFAULT but that's okay
Packit Service e08953
   * because it doesn't have mkostemp(O_CLOEXEC) either.
Packit Service e08953
   */
Packit Service e08953
#ifdef RTLD_DEFAULT
Packit Service e08953
  uv__mkostemp = (int (*)(char*, int)) dlsym(RTLD_DEFAULT, "mkostemp");
Packit Service e08953
Packit Service e08953
  /* We don't care about errors, but we do want to clean them up.
Packit Service e08953
   * If there has been no error, then dlerror() will just return
Packit Service e08953
   * NULL.
Packit Service e08953
   */
Packit Service e08953
  dlerror();
Packit Service e08953
#endif  /* RTLD_DEFAULT */
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
static int uv__fs_mkstemp(uv_fs_t* req) {
Packit Service e08953
  static uv_once_t once = UV_ONCE_INIT;
Packit b5b901
  int r;
Packit Service e08953
#ifdef O_CLOEXEC
Packit Service e08953
  static int no_cloexec_support;
Packit Service e08953
#endif
Packit Service e08953
  static const char pattern[] = "XXXXXX";
Packit Service e08953
  static const size_t pattern_size = sizeof(pattern) - 1;
Packit Service e08953
  char* path;
Packit Service e08953
  size_t path_length;
Packit Service e08953
Packit Service e08953
  path = (char*) req->path;
Packit Service e08953
  path_length = strlen(path);
Packit Service e08953
Packit Service e08953
  /* EINVAL can be returned for 2 reasons:
Packit Service e08953
      1. The template's last 6 characters were not XXXXXX
Packit Service e08953
      2. open() didn't support O_CLOEXEC
Packit Service e08953
     We want to avoid going to the fallback path in case
Packit Service e08953
     of 1, so it's manually checked before. */
Packit Service e08953
  if (path_length < pattern_size ||
Packit Service e08953
      strcmp(path + path_length - pattern_size, pattern)) {
Packit Service e08953
    errno = EINVAL;
Packit Service e08953
    return -1;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  uv_once(&once, uv__mkostemp_initonce);
Packit b5b901
Packit b5b901
#ifdef O_CLOEXEC
Packit Service e08953
  if (no_cloexec_support == 0 && uv__mkostemp != NULL) {
Packit Service e08953
    r = uv__mkostemp(path, O_CLOEXEC);
Packit Service e08953
Packit b5b901
    if (r >= 0)
Packit b5b901
      return r;
Packit Service e08953
Packit Service e08953
    /* If mkostemp() returns EINVAL, it means the kernel doesn't
Packit Service e08953
       support O_CLOEXEC, so we just fallback to mkstemp() below. */
Packit b5b901
    if (errno != EINVAL)
Packit b5b901
      return r;
Packit Service e08953
Packit Service e08953
    /* We set the static variable so that next calls don't even
Packit Service e08953
       try to use mkostemp. */
Packit b5b901
    no_cloexec_support = 1;
Packit Service e08953
  }
Packit b5b901
#endif  /* O_CLOEXEC */
Packit Service e08953
Packit Service e08953
  if (req->cb != NULL)
Packit Service e08953
    uv_rwlock_rdlock(&req->loop->cloexec_lock);
Packit Service e08953
Packit Service e08953
  r = mkstemp(path);
Packit Service e08953
Packit Service e08953
  /* In case of failure `uv__cloexec` will leave error in `errno`,
Packit Service e08953
   * so it is enough to just set `r` to `-1`.
Packit Service e08953
   */
Packit Service e08953
  if (r >= 0 && uv__cloexec(r, 1) != 0) {
Packit Service e08953
    r = uv__close(r);
Packit Service e08953
    if (r != 0)
Packit Service e08953
      abort();
Packit Service e08953
    r = -1;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (req->cb != NULL)
Packit Service e08953
    uv_rwlock_rdunlock(&req->loop->cloexec_lock);
Packit Service e08953
Packit Service e08953
  return r;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
static ssize_t uv__fs_open(uv_fs_t* req) {
Packit Service e08953
#ifdef O_CLOEXEC
Packit Service e08953
  return open(req->path, req->flags | O_CLOEXEC, req->mode);
Packit Service e08953
#else  /* O_CLOEXEC */
Packit Service e08953
  int r;
Packit Service e08953
Packit Service e08953
  if (req->cb != NULL)
Packit b5b901
    uv_rwlock_rdlock(&req->loop->cloexec_lock);
Packit b5b901
Packit b5b901
  r = open(req->path, req->flags, req->mode);
Packit b5b901
Packit b5b901
  /* In case of failure `uv__cloexec` will leave error in `errno`,
Packit b5b901
   * so it is enough to just set `r` to `-1`.
Packit b5b901
   */
Packit b5b901
  if (r >= 0 && uv__cloexec(r, 1) != 0) {
Packit b5b901
    r = uv__close(r);
Packit b5b901
    if (r != 0)
Packit b5b901
      abort();
Packit b5b901
    r = -1;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (req->cb != NULL)
Packit b5b901
    uv_rwlock_rdunlock(&req->loop->cloexec_lock);
Packit b5b901
Packit b5b901
  return r;
Packit Service e08953
#endif  /* O_CLOEXEC */
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
#if !HAVE_PREADV
Packit Service e08953
static ssize_t uv__fs_preadv(uv_file fd,
Packit Service e08953
                             uv_buf_t* bufs,
Packit Service e08953
                             unsigned int nbufs,
Packit Service e08953
                             off_t off) {
Packit Service e08953
  uv_buf_t* buf;
Packit Service e08953
  uv_buf_t* end;
Packit Service e08953
  ssize_t result;
Packit Service e08953
  ssize_t rc;
Packit Service e08953
  size_t pos;
Packit Service e08953
Packit Service e08953
  assert(nbufs > 0);
Packit Service e08953
Packit Service e08953
  result = 0;
Packit Service e08953
  pos = 0;
Packit Service e08953
  buf = bufs + 0;
Packit Service e08953
  end = bufs + nbufs;
Packit Service e08953
Packit Service e08953
  for (;;) {
Packit Service e08953
    do
Packit Service e08953
      rc = pread(fd, buf->base + pos, buf->len - pos, off + result);
Packit Service e08953
    while (rc == -1 && errno == EINTR);
Packit Service e08953
Packit Service e08953
    if (rc == 0)
Packit Service e08953
      break;
Packit Service e08953
Packit Service e08953
    if (rc == -1 && result == 0)
Packit Service e08953
      return UV__ERR(errno);
Packit Service e08953
Packit Service e08953
    if (rc == -1)
Packit Service e08953
      break;  /* We read some data so return that, ignore the error. */
Packit Service e08953
Packit Service e08953
    pos += rc;
Packit Service e08953
    result += rc;
Packit Service e08953
Packit Service e08953
    if (pos < buf->len)
Packit Service e08953
      continue;
Packit Service e08953
Packit Service e08953
    pos = 0;
Packit Service e08953
    buf += 1;
Packit Service e08953
Packit Service e08953
    if (buf == end)
Packit Service e08953
      break;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  return result;
Packit b5b901
}
Packit Service e08953
#endif
Packit b5b901
Packit b5b901
Packit b5b901
static ssize_t uv__fs_read(uv_fs_t* req) {
Packit b5b901
#if defined(__linux__)
Packit b5b901
  static int no_preadv;
Packit b5b901
#endif
Packit Service e08953
  unsigned int iovmax;
Packit b5b901
  ssize_t result;
Packit b5b901
Packit Service e08953
  iovmax = uv__getiovmax();
Packit Service e08953
  if (req->nbufs > iovmax)
Packit Service e08953
    req->nbufs = iovmax;
Packit Service e08953
Packit b5b901
  if (req->off < 0) {
Packit b5b901
    if (req->nbufs == 1)
Packit b5b901
      result = read(req->file, req->bufs[0].base, req->bufs[0].len);
Packit b5b901
    else
Packit b5b901
      result = readv(req->file, (struct iovec*) req->bufs, req->nbufs);
Packit b5b901
  } else {
Packit b5b901
    if (req->nbufs == 1) {
Packit b5b901
      result = pread(req->file, req->bufs[0].base, req->bufs[0].len, req->off);
Packit b5b901
      goto done;
Packit b5b901
    }
Packit b5b901
Packit b5b901
#if HAVE_PREADV
Packit b5b901
    result = preadv(req->file, (struct iovec*) req->bufs, req->nbufs, req->off);
Packit b5b901
#else
Packit b5b901
# if defined(__linux__)
Packit b5b901
    if (no_preadv) retry:
Packit b5b901
# endif
Packit b5b901
    {
Packit Service e08953
      result = uv__fs_preadv(req->file, req->bufs, req->nbufs, req->off);
Packit b5b901
    }
Packit b5b901
# if defined(__linux__)
Packit b5b901
    else {
Packit b5b901
      result = uv__preadv(req->file,
Packit b5b901
                          (struct iovec*)req->bufs,
Packit b5b901
                          req->nbufs,
Packit b5b901
                          req->off);
Packit b5b901
      if (result == -1 && errno == ENOSYS) {
Packit b5b901
        no_preadv = 1;
Packit b5b901
        goto retry;
Packit b5b901
      }
Packit b5b901
    }
Packit b5b901
# endif
Packit b5b901
#endif
Packit b5b901
  }
Packit b5b901
Packit b5b901
done:
Packit Service e08953
  /* Early cleanup of bufs allocation, since we're done with it. */
Packit Service e08953
  if (req->bufs != req->bufsml)
Packit Service e08953
    uv__free(req->bufs);
Packit Service e08953
Packit Service e08953
  req->bufs = NULL;
Packit Service e08953
  req->nbufs = 0;
Packit Service e08953
Packit Service e08953
#ifdef __PASE__
Packit Service e08953
  /* PASE returns EOPNOTSUPP when reading a directory, convert to EISDIR */
Packit Service e08953
  if (result == -1 && errno == EOPNOTSUPP) {
Packit Service e08953
    struct stat buf;
Packit Service e08953
    ssize_t rc;
Packit Service e08953
    rc = fstat(req->file, &buf;;
Packit Service e08953
    if (rc == 0 && S_ISDIR(buf.st_mode)) {
Packit Service e08953
      errno = EISDIR;
Packit Service e08953
    }
Packit Service e08953
  }
Packit Service e08953
#endif
Packit Service e08953
Packit b5b901
  return result;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
#if defined(__APPLE__) && !defined(MAC_OS_X_VERSION_10_8)
Packit b5b901
#define UV_CONST_DIRENT uv__dirent_t
Packit b5b901
#else
Packit b5b901
#define UV_CONST_DIRENT const uv__dirent_t
Packit b5b901
#endif
Packit b5b901
Packit b5b901
Packit b5b901
static int uv__fs_scandir_filter(UV_CONST_DIRENT* dent) {
Packit b5b901
  return strcmp(dent->d_name, ".") != 0 && strcmp(dent->d_name, "..") != 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static int uv__fs_scandir_sort(UV_CONST_DIRENT** a, UV_CONST_DIRENT** b) {
Packit b5b901
  return strcmp((*a)->d_name, (*b)->d_name);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static ssize_t uv__fs_scandir(uv_fs_t* req) {
Packit Service e08953
  uv__dirent_t** dents;
Packit b5b901
  int n;
Packit b5b901
Packit b5b901
  dents = NULL;
Packit b5b901
  n = scandir(req->path, &dents, uv__fs_scandir_filter, uv__fs_scandir_sort);
Packit b5b901
Packit b5b901
  /* NOTE: We will use nbufs as an index field */
Packit b5b901
  req->nbufs = 0;
Packit b5b901
Packit b5b901
  if (n == 0) {
Packit b5b901
    /* OS X still needs to deallocate some memory.
Packit b5b901
     * Memory was allocated using the system allocator, so use free() here.
Packit b5b901
     */
Packit b5b901
    free(dents);
Packit b5b901
    dents = NULL;
Packit b5b901
  } else if (n == -1) {
Packit b5b901
    return n;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  req->ptr = dents;
Packit b5b901
Packit b5b901
  return n;
Packit b5b901
}
Packit b5b901
Packit Service e08953
static int uv__fs_opendir(uv_fs_t* req) {
Packit Service e08953
  uv_dir_t* dir;
Packit Service e08953
Packit Service e08953
  dir = uv__malloc(sizeof(*dir));
Packit Service e08953
  if (dir == NULL)
Packit Service e08953
    goto error;
Packit Service e08953
Packit Service e08953
  dir->dir = opendir(req->path);
Packit Service e08953
  if (dir->dir == NULL)
Packit Service e08953
    goto error;
Packit Service e08953
Packit Service e08953
  req->ptr = dir;
Packit Service e08953
  return 0;
Packit Service e08953
Packit Service e08953
error:
Packit Service e08953
  uv__free(dir);
Packit Service e08953
  req->ptr = NULL;
Packit Service e08953
  return -1;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
static int uv__fs_readdir(uv_fs_t* req) {
Packit Service e08953
  uv_dir_t* dir;
Packit Service e08953
  uv_dirent_t* dirent;
Packit Service e08953
  struct dirent* res;
Packit Service e08953
  unsigned int dirent_idx;
Packit Service e08953
  unsigned int i;
Packit Service e08953
Packit Service e08953
  dir = req->ptr;
Packit Service e08953
  dirent_idx = 0;
Packit Service e08953
Packit Service e08953
  while (dirent_idx < dir->nentries) {
Packit Service e08953
    /* readdir() returns NULL on end of directory, as well as on error. errno
Packit Service e08953
       is used to differentiate between the two conditions. */
Packit Service e08953
    errno = 0;
Packit Service e08953
    res = readdir(dir->dir);
Packit Service e08953
Packit Service e08953
    if (res == NULL) {
Packit Service e08953
      if (errno != 0)
Packit Service e08953
        goto error;
Packit Service e08953
      break;
Packit Service e08953
    }
Packit Service e08953
Packit Service e08953
    if (strcmp(res->d_name, ".") == 0 || strcmp(res->d_name, "..") == 0)
Packit Service e08953
      continue;
Packit Service e08953
Packit Service e08953
    dirent = &dir->dirents[dirent_idx];
Packit Service e08953
    dirent->name = uv__strdup(res->d_name);
Packit Service e08953
Packit Service e08953
    if (dirent->name == NULL)
Packit Service e08953
      goto error;
Packit Service e08953
Packit Service e08953
    dirent->type = uv__fs_get_dirent_type(res);
Packit Service e08953
    ++dirent_idx;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  return dirent_idx;
Packit Service e08953
Packit Service e08953
error:
Packit Service e08953
  for (i = 0; i < dirent_idx; ++i) {
Packit Service e08953
    uv__free((char*) dir->dirents[i].name);
Packit Service e08953
    dir->dirents[i].name = NULL;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  return -1;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
static int uv__fs_closedir(uv_fs_t* req) {
Packit Service e08953
  uv_dir_t* dir;
Packit Service e08953
Packit Service e08953
  dir = req->ptr;
Packit Service e08953
Packit Service e08953
  if (dir->dir != NULL) {
Packit Service e08953
    closedir(dir->dir);
Packit Service e08953
    dir->dir = NULL;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  uv__free(req->ptr);
Packit Service e08953
  req->ptr = NULL;
Packit Service e08953
  return 0;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
static int uv__fs_statfs(uv_fs_t* req) {
Packit Service e08953
  uv_statfs_t* stat_fs;
Packit Service e08953
#if defined(__sun) || defined(__MVS__) || defined(__NetBSD__) || defined(__HAIKU__)
Packit Service e08953
  struct statvfs buf;
Packit Service e08953
Packit Service e08953
  if (0 != statvfs(req->path, &buf))
Packit Service e08953
#else
Packit Service e08953
  struct statfs buf;
Packit Service e08953
Packit Service e08953
  if (0 != statfs(req->path, &buf))
Packit Service e08953
#endif /* defined(__sun) */
Packit Service e08953
    return -1;
Packit Service e08953
Packit Service e08953
  stat_fs = uv__malloc(sizeof(*stat_fs));
Packit Service e08953
  if (stat_fs == NULL) {
Packit Service e08953
    errno = ENOMEM;
Packit Service e08953
    return -1;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
#if defined(__sun) || defined(__MVS__) || defined(__OpenBSD__) || defined(__NetBSD__) || defined(__HAIKU__)
Packit Service e08953
  stat_fs->f_type = 0;  /* f_type is not supported. */
Packit Service e08953
#else
Packit Service e08953
  stat_fs->f_type = buf.f_type;
Packit Service e08953
#endif
Packit Service e08953
  stat_fs->f_bsize = buf.f_bsize;
Packit Service e08953
  stat_fs->f_blocks = buf.f_blocks;
Packit Service e08953
  stat_fs->f_bfree = buf.f_bfree;
Packit Service e08953
  stat_fs->f_bavail = buf.f_bavail;
Packit Service e08953
  stat_fs->f_files = buf.f_files;
Packit Service e08953
  stat_fs->f_ffree = buf.f_ffree;
Packit Service e08953
  req->ptr = stat_fs;
Packit Service e08953
  return 0;
Packit Service e08953
}
Packit b5b901
Packit b5b901
static ssize_t uv__fs_pathmax_size(const char* path) {
Packit b5b901
  ssize_t pathmax;
Packit b5b901
Packit b5b901
  pathmax = pathconf(path, _PC_PATH_MAX);
Packit b5b901
Packit Service e08953
  if (pathmax == -1)
Packit Service e08953
    pathmax = UV__PATH_MAX;
Packit b5b901
Packit b5b901
  return pathmax;
Packit b5b901
}
Packit b5b901
Packit b5b901
static ssize_t uv__fs_readlink(uv_fs_t* req) {
Packit Service e08953
  ssize_t maxlen;
Packit b5b901
  ssize_t len;
Packit b5b901
  char* buf;
Packit b5b901
Packit Service e08953
#if defined(_POSIX_PATH_MAX) || defined(PATH_MAX)
Packit Service e08953
  maxlen = uv__fs_pathmax_size(req->path);
Packit Service e08953
#else
Packit Service e08953
  /* We may not have a real PATH_MAX.  Read size of link.  */
Packit Service e08953
  struct stat st;
Packit Service e08953
  int ret;
Packit Service e08953
  ret = lstat(req->path, &st);
Packit Service e08953
  if (ret != 0)
Packit Service e08953
    return -1;
Packit Service e08953
  if (!S_ISLNK(st.st_mode)) {
Packit Service e08953
    errno = EINVAL;
Packit Service e08953
    return -1;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  maxlen = st.st_size;
Packit Service e08953
Packit Service e08953
  /* According to readlink(2) lstat can report st_size == 0
Packit Service e08953
     for some symlinks, such as those in /proc or /sys.  */
Packit Service e08953
  if (maxlen == 0)
Packit Service e08953
    maxlen = uv__fs_pathmax_size(req->path);
Packit Service e08953
#endif
Packit Service e08953
Packit Service e08953
  buf = uv__malloc(maxlen);
Packit b5b901
Packit b5b901
  if (buf == NULL) {
Packit b5b901
    errno = ENOMEM;
Packit b5b901
    return -1;
Packit b5b901
  }
Packit b5b901
Packit b5b901
#if defined(__MVS__)
Packit Service e08953
  len = os390_readlink(req->path, buf, maxlen);
Packit b5b901
#else
Packit Service e08953
  len = readlink(req->path, buf, maxlen);
Packit b5b901
#endif
Packit b5b901
Packit b5b901
  if (len == -1) {
Packit b5b901
    uv__free(buf);
Packit b5b901
    return -1;
Packit b5b901
  }
Packit b5b901
Packit Service e08953
  /* Uncommon case: resize to make room for the trailing nul byte. */
Packit Service e08953
  if (len == maxlen) {
Packit Service e08953
    buf = uv__reallocf(buf, len + 1);
Packit Service e08953
Packit Service e08953
    if (buf == NULL)
Packit Service e08953
      return -1;
Packit Service e08953
  }
Packit Service e08953
Packit b5b901
  buf[len] = '\0';
Packit b5b901
  req->ptr = buf;
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
static ssize_t uv__fs_realpath(uv_fs_t* req) {
Packit b5b901
  char* buf;
Packit b5b901
Packit Service e08953
#if defined(_POSIX_VERSION) && _POSIX_VERSION >= 200809L
Packit Service e08953
  buf = realpath(req->path, NULL);
Packit Service e08953
  if (buf == NULL)
Packit Service e08953
    return -1;
Packit Service e08953
#else
Packit Service e08953
  ssize_t len;
Packit Service e08953
Packit b5b901
  len = uv__fs_pathmax_size(req->path);
Packit b5b901
  buf = uv__malloc(len + 1);
Packit b5b901
Packit b5b901
  if (buf == NULL) {
Packit b5b901
    errno = ENOMEM;
Packit b5b901
    return -1;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (realpath(req->path, buf) == NULL) {
Packit b5b901
    uv__free(buf);
Packit b5b901
    return -1;
Packit b5b901
  }
Packit Service e08953
#endif
Packit b5b901
Packit b5b901
  req->ptr = buf;
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
static ssize_t uv__fs_sendfile_emul(uv_fs_t* req) {
Packit b5b901
  struct pollfd pfd;
Packit b5b901
  int use_pread;
Packit b5b901
  off_t offset;
Packit b5b901
  ssize_t nsent;
Packit b5b901
  ssize_t nread;
Packit b5b901
  ssize_t nwritten;
Packit b5b901
  size_t buflen;
Packit b5b901
  size_t len;
Packit b5b901
  ssize_t n;
Packit b5b901
  int in_fd;
Packit b5b901
  int out_fd;
Packit b5b901
  char buf[8192];
Packit b5b901
Packit b5b901
  len = req->bufsml[0].len;
Packit b5b901
  in_fd = req->flags;
Packit b5b901
  out_fd = req->file;
Packit b5b901
  offset = req->off;
Packit b5b901
  use_pread = 1;
Packit b5b901
Packit b5b901
  /* Here are the rules regarding errors:
Packit b5b901
   *
Packit b5b901
   * 1. Read errors are reported only if nsent==0, otherwise we return nsent.
Packit b5b901
   *    The user needs to know that some data has already been sent, to stop
Packit b5b901
   *    them from sending it twice.
Packit b5b901
   *
Packit b5b901
   * 2. Write errors are always reported. Write errors are bad because they
Packit b5b901
   *    mean data loss: we've read data but now we can't write it out.
Packit b5b901
   *
Packit b5b901
   * We try to use pread() and fall back to regular read() if the source fd
Packit b5b901
   * doesn't support positional reads, for example when it's a pipe fd.
Packit b5b901
   *
Packit b5b901
   * If we get EAGAIN when writing to the target fd, we poll() on it until
Packit b5b901
   * it becomes writable again.
Packit b5b901
   *
Packit b5b901
   * FIXME: If we get a write error when use_pread==1, it should be safe to
Packit b5b901
   *        return the number of sent bytes instead of an error because pread()
Packit b5b901
   *        is, in theory, idempotent. However, special files in /dev or /proc
Packit b5b901
   *        may support pread() but not necessarily return the same data on
Packit b5b901
   *        successive reads.
Packit b5b901
   *
Packit b5b901
   * FIXME: There is no way now to signal that we managed to send *some* data
Packit b5b901
   *        before a write error.
Packit b5b901
   */
Packit b5b901
  for (nsent = 0; (size_t) nsent < len; ) {
Packit b5b901
    buflen = len - nsent;
Packit b5b901
Packit b5b901
    if (buflen > sizeof(buf))
Packit b5b901
      buflen = sizeof(buf);
Packit b5b901
Packit b5b901
    do
Packit b5b901
      if (use_pread)
Packit b5b901
        nread = pread(in_fd, buf, buflen, offset);
Packit b5b901
      else
Packit b5b901
        nread = read(in_fd, buf, buflen);
Packit b5b901
    while (nread == -1 && errno == EINTR);
Packit b5b901
Packit b5b901
    if (nread == 0)
Packit b5b901
      goto out;
Packit b5b901
Packit b5b901
    if (nread == -1) {
Packit b5b901
      if (use_pread && nsent == 0 && (errno == EIO || errno == ESPIPE)) {
Packit b5b901
        use_pread = 0;
Packit b5b901
        continue;
Packit b5b901
      }
Packit b5b901
Packit b5b901
      if (nsent == 0)
Packit b5b901
        nsent = -1;
Packit b5b901
Packit b5b901
      goto out;
Packit b5b901
    }
Packit b5b901
Packit b5b901
    for (nwritten = 0; nwritten < nread; ) {
Packit b5b901
      do
Packit b5b901
        n = write(out_fd, buf + nwritten, nread - nwritten);
Packit b5b901
      while (n == -1 && errno == EINTR);
Packit b5b901
Packit b5b901
      if (n != -1) {
Packit b5b901
        nwritten += n;
Packit b5b901
        continue;
Packit b5b901
      }
Packit b5b901
Packit b5b901
      if (errno != EAGAIN && errno != EWOULDBLOCK) {
Packit b5b901
        nsent = -1;
Packit b5b901
        goto out;
Packit b5b901
      }
Packit b5b901
Packit b5b901
      pfd.fd = out_fd;
Packit b5b901
      pfd.events = POLLOUT;
Packit b5b901
      pfd.revents = 0;
Packit b5b901
Packit b5b901
      do
Packit b5b901
        n = poll(&pfd, 1, -1);
Packit b5b901
      while (n == -1 && errno == EINTR);
Packit b5b901
Packit b5b901
      if (n == -1 || (pfd.revents & ~POLLOUT) != 0) {
Packit b5b901
        errno = EIO;
Packit b5b901
        nsent = -1;
Packit b5b901
        goto out;
Packit b5b901
      }
Packit b5b901
    }
Packit b5b901
Packit b5b901
    offset += nread;
Packit b5b901
    nsent += nread;
Packit b5b901
  }
Packit b5b901
Packit b5b901
out:
Packit b5b901
  if (nsent != -1)
Packit b5b901
    req->off = offset;
Packit b5b901
Packit b5b901
  return nsent;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static ssize_t uv__fs_sendfile(uv_fs_t* req) {
Packit b5b901
  int in_fd;
Packit b5b901
  int out_fd;
Packit b5b901
Packit b5b901
  in_fd = req->flags;
Packit b5b901
  out_fd = req->file;
Packit b5b901
Packit b5b901
#if defined(__linux__) || defined(__sun)
Packit b5b901
  {
Packit b5b901
    off_t off;
Packit b5b901
    ssize_t r;
Packit b5b901
Packit b5b901
    off = req->off;
Packit b5b901
    r = sendfile(out_fd, in_fd, &off, req->bufsml[0].len);
Packit b5b901
Packit b5b901
    /* sendfile() on SunOS returns EINVAL if the target fd is not a socket but
Packit b5b901
     * it still writes out data. Fortunately, we can detect it by checking if
Packit b5b901
     * the offset has been updated.
Packit b5b901
     */
Packit b5b901
    if (r != -1 || off > req->off) {
Packit b5b901
      r = off - req->off;
Packit b5b901
      req->off = off;
Packit b5b901
      return r;
Packit b5b901
    }
Packit b5b901
Packit b5b901
    if (errno == EINVAL ||
Packit b5b901
        errno == EIO ||
Packit b5b901
        errno == ENOTSOCK ||
Packit b5b901
        errno == EXDEV) {
Packit b5b901
      errno = 0;
Packit b5b901
      return uv__fs_sendfile_emul(req);
Packit b5b901
    }
Packit b5b901
Packit b5b901
    return -1;
Packit b5b901
  }
Packit b5b901
#elif defined(__APPLE__)           || \
Packit b5b901
      defined(__DragonFly__)       || \
Packit b5b901
      defined(__FreeBSD__)         || \
Packit b5b901
      defined(__FreeBSD_kernel__)
Packit b5b901
  {
Packit b5b901
    off_t len;
Packit b5b901
    ssize_t r;
Packit b5b901
Packit b5b901
    /* sendfile() on FreeBSD and Darwin returns EAGAIN if the target fd is in
Packit b5b901
     * non-blocking mode and not all data could be written. If a non-zero
Packit b5b901
     * number of bytes have been sent, we don't consider it an error.
Packit b5b901
     */
Packit b5b901
Packit b5b901
#if defined(__FreeBSD__) || defined(__DragonFly__)
Packit b5b901
    len = 0;
Packit b5b901
    r = sendfile(in_fd, out_fd, req->off, req->bufsml[0].len, NULL, &len, 0);
Packit b5b901
#elif defined(__FreeBSD_kernel__)
Packit b5b901
    len = 0;
Packit b5b901
    r = bsd_sendfile(in_fd,
Packit b5b901
                     out_fd,
Packit b5b901
                     req->off,
Packit b5b901
                     req->bufsml[0].len,
Packit b5b901
                     NULL,
Packit b5b901
                     &len,
Packit b5b901
                     0);
Packit b5b901
#else
Packit b5b901
    /* The darwin sendfile takes len as an input for the length to send,
Packit b5b901
     * so make sure to initialize it with the caller's value. */
Packit b5b901
    len = req->bufsml[0].len;
Packit b5b901
    r = sendfile(in_fd, out_fd, req->off, &len, NULL, 0);
Packit b5b901
#endif
Packit b5b901
Packit b5b901
     /*
Packit b5b901
     * The man page for sendfile(2) on DragonFly states that `len` contains
Packit b5b901
     * a meaningful value ONLY in case of EAGAIN and EINTR.
Packit b5b901
     * Nothing is said about it's value in case of other errors, so better
Packit b5b901
     * not depend on the potential wrong assumption that is was not modified
Packit b5b901
     * by the syscall.
Packit b5b901
     */
Packit b5b901
    if (r == 0 || ((errno == EAGAIN || errno == EINTR) && len != 0)) {
Packit b5b901
      req->off += len;
Packit b5b901
      return (ssize_t) len;
Packit b5b901
    }
Packit b5b901
Packit b5b901
    if (errno == EINVAL ||
Packit b5b901
        errno == EIO ||
Packit b5b901
        errno == ENOTSOCK ||
Packit b5b901
        errno == EXDEV) {
Packit b5b901
      errno = 0;
Packit b5b901
      return uv__fs_sendfile_emul(req);
Packit b5b901
    }
Packit b5b901
Packit b5b901
    return -1;
Packit b5b901
  }
Packit b5b901
#else
Packit b5b901
  /* Squelch compiler warnings. */
Packit b5b901
  (void) &in_fd;
Packit b5b901
  (void) &out_fd;
Packit b5b901
Packit b5b901
  return uv__fs_sendfile_emul(req);
Packit b5b901
#endif
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static ssize_t uv__fs_utime(uv_fs_t* req) {
Packit b5b901
#if defined(__linux__)                                                         \
Packit b5b901
    || defined(_AIX71)                                                         \
Packit Service e08953
    || defined(__sun)                                                          \
Packit Service e08953
    || defined(__HAIKU__)
Packit b5b901
  /* utimesat() has nanosecond resolution but we stick to microseconds
Packit b5b901
   * for the sake of consistency with other platforms.
Packit b5b901
   */
Packit b5b901
  struct timespec ts[2];
Packit Service e08953
  ts[0] = uv__fs_to_timespec(req->atime);
Packit Service e08953
  ts[1] = uv__fs_to_timespec(req->mtime);
Packit b5b901
  return utimensat(AT_FDCWD, req->path, ts, 0);
Packit b5b901
#elif defined(__APPLE__)                                                      \
Packit b5b901
    || defined(__DragonFly__)                                                 \
Packit b5b901
    || defined(__FreeBSD__)                                                   \
Packit b5b901
    || defined(__FreeBSD_kernel__)                                            \
Packit b5b901
    || defined(__NetBSD__)                                                    \
Packit b5b901
    || defined(__OpenBSD__)
Packit b5b901
  struct timeval tv[2];
Packit Service e08953
  tv[0] = uv__fs_to_timeval(req->atime);
Packit Service e08953
  tv[1] = uv__fs_to_timeval(req->mtime);
Packit b5b901
  return utimes(req->path, tv);
Packit b5b901
#elif defined(_AIX)                                                           \
Packit b5b901
    && !defined(_AIX71)
Packit b5b901
  struct utimbuf buf;
Packit b5b901
  buf.actime = req->atime;
Packit b5b901
  buf.modtime = req->mtime;
Packit b5b901
  return utime(req->path, &buf;;
Packit b5b901
#elif defined(__MVS__)
Packit b5b901
  attrib_t atr;
Packit b5b901
  memset(&atr, 0, sizeof(atr));
Packit b5b901
  atr.att_mtimechg = 1;
Packit b5b901
  atr.att_atimechg = 1;
Packit b5b901
  atr.att_mtime = req->mtime;
Packit b5b901
  atr.att_atime = req->atime;
Packit Service e08953
  return __lchattr((char*) req->path, &atr, sizeof(atr));
Packit Service e08953
#else
Packit Service e08953
  errno = ENOSYS;
Packit Service e08953
  return -1;
Packit Service e08953
#endif
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
static ssize_t uv__fs_lutime(uv_fs_t* req) {
Packit Service e08953
#if defined(__linux__)            ||                                           \
Packit Service e08953
    defined(_AIX71)               ||                                           \
Packit Service e08953
    defined(__sun)                ||                                           \
Packit Service e08953
    defined(__HAIKU__)
Packit Service e08953
  struct timespec ts[2];
Packit Service e08953
  ts[0] = uv__fs_to_timespec(req->atime);
Packit Service e08953
  ts[1] = uv__fs_to_timespec(req->mtime);
Packit Service e08953
  return utimensat(AT_FDCWD, req->path, ts, AT_SYMLINK_NOFOLLOW);
Packit Service e08953
#elif defined(__APPLE__)          ||                                          \
Packit Service e08953
      defined(__DragonFly__)      ||                                          \
Packit Service e08953
      defined(__FreeBSD__)        ||                                          \
Packit Service e08953
      defined(__FreeBSD_kernel__) ||                                          \
Packit Service e08953
      defined(__NetBSD__)
Packit Service e08953
  struct timeval tv[2];
Packit Service e08953
  tv[0] = uv__fs_to_timeval(req->atime);
Packit Service e08953
  tv[1] = uv__fs_to_timeval(req->mtime);
Packit Service e08953
  return lutimes(req->path, tv);
Packit b5b901
#else
Packit b5b901
  errno = ENOSYS;
Packit b5b901
  return -1;
Packit b5b901
#endif
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static ssize_t uv__fs_write(uv_fs_t* req) {
Packit b5b901
#if defined(__linux__)
Packit b5b901
  static int no_pwritev;
Packit b5b901
#endif
Packit b5b901
  ssize_t r;
Packit b5b901
Packit b5b901
  /* Serialize writes on OS X, concurrent write() and pwrite() calls result in
Packit b5b901
   * data loss. We can't use a per-file descriptor lock, the descriptor may be
Packit b5b901
   * a dup().
Packit b5b901
   */
Packit b5b901
#if defined(__APPLE__)
Packit b5b901
  static pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;
Packit b5b901
Packit b5b901
  if (pthread_mutex_lock(&lock))
Packit b5b901
    abort();
Packit b5b901
#endif
Packit b5b901
Packit b5b901
  if (req->off < 0) {
Packit b5b901
    if (req->nbufs == 1)
Packit b5b901
      r = write(req->file, req->bufs[0].base, req->bufs[0].len);
Packit b5b901
    else
Packit b5b901
      r = writev(req->file, (struct iovec*) req->bufs, req->nbufs);
Packit b5b901
  } else {
Packit b5b901
    if (req->nbufs == 1) {
Packit b5b901
      r = pwrite(req->file, req->bufs[0].base, req->bufs[0].len, req->off);
Packit b5b901
      goto done;
Packit b5b901
    }
Packit b5b901
#if HAVE_PREADV
Packit b5b901
    r = pwritev(req->file, (struct iovec*) req->bufs, req->nbufs, req->off);
Packit b5b901
#else
Packit b5b901
# if defined(__linux__)
Packit b5b901
    if (no_pwritev) retry:
Packit b5b901
# endif
Packit b5b901
    {
Packit Service e08953
      r = pwrite(req->file, req->bufs[0].base, req->bufs[0].len, req->off);
Packit b5b901
    }
Packit b5b901
# if defined(__linux__)
Packit b5b901
    else {
Packit b5b901
      r = uv__pwritev(req->file,
Packit b5b901
                      (struct iovec*) req->bufs,
Packit b5b901
                      req->nbufs,
Packit b5b901
                      req->off);
Packit b5b901
      if (r == -1 && errno == ENOSYS) {
Packit b5b901
        no_pwritev = 1;
Packit b5b901
        goto retry;
Packit b5b901
      }
Packit b5b901
    }
Packit b5b901
# endif
Packit b5b901
#endif
Packit b5b901
  }
Packit b5b901
Packit b5b901
done:
Packit b5b901
#if defined(__APPLE__)
Packit b5b901
  if (pthread_mutex_unlock(&lock))
Packit b5b901
    abort();
Packit b5b901
#endif
Packit b5b901
Packit b5b901
  return r;
Packit b5b901
}
Packit b5b901
Packit b5b901
static ssize_t uv__fs_copyfile(uv_fs_t* req) {
Packit b5b901
  uv_fs_t fs_req;
Packit b5b901
  uv_file srcfd;
Packit b5b901
  uv_file dstfd;
Packit Service e08953
  struct stat src_statsbuf;
Packit Service e08953
  struct stat dst_statsbuf;
Packit b5b901
  int dst_flags;
Packit b5b901
  int result;
Packit b5b901
  int err;
Packit Service e08953
  off_t bytes_to_send;
Packit Service e08953
  off_t in_offset;
Packit Service e08953
  off_t bytes_written;
Packit Service e08953
  size_t bytes_chunk;
Packit b5b901
Packit b5b901
  dstfd = -1;
Packit b5b901
  err = 0;
Packit b5b901
Packit b5b901
  /* Open the source file. */
Packit b5b901
  srcfd = uv_fs_open(NULL, &fs_req, req->path, O_RDONLY, 0, NULL);
Packit b5b901
  uv_fs_req_cleanup(&fs_req);
Packit b5b901
Packit b5b901
  if (srcfd < 0)
Packit b5b901
    return srcfd;
Packit b5b901
Packit b5b901
  /* Get the source file's mode. */
Packit Service e08953
  if (fstat(srcfd, &src_statsbuf)) {
Packit b5b901
    err = UV__ERR(errno);
Packit b5b901
    goto out;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  dst_flags = O_WRONLY | O_CREAT | O_TRUNC;
Packit b5b901
Packit b5b901
  if (req->flags & UV_FS_COPYFILE_EXCL)
Packit b5b901
    dst_flags |= O_EXCL;
Packit b5b901
Packit b5b901
  /* Open the destination file. */
Packit b5b901
  dstfd = uv_fs_open(NULL,
Packit b5b901
                     &fs_req,
Packit b5b901
                     req->new_path,
Packit b5b901
                     dst_flags,
Packit Service e08953
                     src_statsbuf.st_mode,
Packit b5b901
                     NULL);
Packit b5b901
  uv_fs_req_cleanup(&fs_req);
Packit b5b901
Packit b5b901
  if (dstfd < 0) {
Packit b5b901
    err = dstfd;
Packit b5b901
    goto out;
Packit b5b901
  }
Packit b5b901
Packit Service e08953
  /* Get the destination file's mode. */
Packit Service e08953
  if (fstat(dstfd, &dst_statsbuf)) {
Packit b5b901
    err = UV__ERR(errno);
Packit b5b901
    goto out;
Packit b5b901
  }
Packit b5b901
Packit Service e08953
  /* Check if srcfd and dstfd refer to the same file */
Packit Service e08953
  if (src_statsbuf.st_dev == dst_statsbuf.st_dev &&
Packit Service e08953
      src_statsbuf.st_ino == dst_statsbuf.st_ino) {
Packit Service e08953
    goto out;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  if (fchmod(dstfd, src_statsbuf.st_mode) == -1) {
Packit Service e08953
    err = UV__ERR(errno);
Packit Service e08953
#ifdef __linux__
Packit Service e08953
    if (err != UV_EPERM)
Packit Service e08953
      goto out;
Packit Service e08953
Packit Service e08953
    {
Packit Service e08953
      struct statfs s;
Packit Service e08953
Packit Service e08953
      /* fchmod() on CIFS shares always fails with EPERM unless the share is
Packit Service e08953
       * mounted with "noperm". As fchmod() is a meaningless operation on such
Packit Service e08953
       * shares anyway, detect that condition and squelch the error.
Packit Service e08953
       */
Packit Service e08953
      if (fstatfs(dstfd, &s) == -1)
Packit Service e08953
        goto out;
Packit Service e08953
Packit Service e08953
      if (s.f_type != /* CIFS */ 0xFF534D42u)
Packit Service e08953
        goto out;
Packit Service e08953
    }
Packit Service e08953
Packit Service e08953
    err = 0;
Packit Service e08953
#else  /* !__linux__ */
Packit Service e08953
    goto out;
Packit Service e08953
#endif  /* !__linux__ */
Packit Service e08953
  }
Packit Service e08953
Packit b5b901
#ifdef FICLONE
Packit b5b901
  if (req->flags & UV_FS_COPYFILE_FICLONE ||
Packit b5b901
      req->flags & UV_FS_COPYFILE_FICLONE_FORCE) {
Packit Service e08953
    if (ioctl(dstfd, FICLONE, srcfd) == 0) {
Packit Service e08953
      /* ioctl() with FICLONE succeeded. */
Packit Service e08953
      goto out;
Packit Service e08953
    }
Packit Service e08953
    /* If an error occurred and force was set, return the error to the caller;
Packit Service e08953
     * fall back to sendfile() when force was not set. */
Packit Service e08953
    if (req->flags & UV_FS_COPYFILE_FICLONE_FORCE) {
Packit Service e08953
      err = UV__ERR(errno);
Packit b5b901
      goto out;
Packit b5b901
    }
Packit b5b901
  }
Packit b5b901
#else
Packit b5b901
  if (req->flags & UV_FS_COPYFILE_FICLONE_FORCE) {
Packit b5b901
    err = UV_ENOSYS;
Packit b5b901
    goto out;
Packit b5b901
  }
Packit b5b901
#endif
Packit b5b901
Packit Service e08953
  bytes_to_send = src_statsbuf.st_size;
Packit b5b901
  in_offset = 0;
Packit b5b901
  while (bytes_to_send != 0) {
Packit Service e08953
    bytes_chunk = SSIZE_MAX;
Packit Service e08953
    if (bytes_to_send < (off_t) bytes_chunk)
Packit Service e08953
      bytes_chunk = bytes_to_send;
Packit Service e08953
    uv_fs_sendfile(NULL, &fs_req, dstfd, srcfd, in_offset, bytes_chunk, NULL);
Packit Service e08953
    bytes_written = fs_req.result;
Packit b5b901
    uv_fs_req_cleanup(&fs_req);
Packit Service e08953
Packit Service e08953
    if (bytes_written < 0) {
Packit Service e08953
      err = bytes_written;
Packit b5b901
      break;
Packit Service e08953
    }
Packit Service e08953
Packit Service e08953
    bytes_to_send -= bytes_written;
Packit Service e08953
    in_offset += bytes_written;
Packit b5b901
  }
Packit b5b901
Packit b5b901
out:
Packit b5b901
  if (err < 0)
Packit b5b901
    result = err;
Packit b5b901
  else
Packit b5b901
    result = 0;
Packit b5b901
Packit b5b901
  /* Close the source file. */
Packit b5b901
  err = uv__close_nocheckstdio(srcfd);
Packit b5b901
Packit b5b901
  /* Don't overwrite any existing errors. */
Packit b5b901
  if (err != 0 && result == 0)
Packit b5b901
    result = err;
Packit b5b901
Packit b5b901
  /* Close the destination file if it is open. */
Packit b5b901
  if (dstfd >= 0) {
Packit b5b901
    err = uv__close_nocheckstdio(dstfd);
Packit b5b901
Packit b5b901
    /* Don't overwrite any existing errors. */
Packit b5b901
    if (err != 0 && result == 0)
Packit b5b901
      result = err;
Packit b5b901
Packit b5b901
    /* Remove the destination file if something went wrong. */
Packit b5b901
    if (result != 0) {
Packit b5b901
      uv_fs_unlink(NULL, &fs_req, req->new_path, NULL);
Packit b5b901
      /* Ignore the unlink return value, as an error already happened. */
Packit b5b901
      uv_fs_req_cleanup(&fs_req);
Packit b5b901
    }
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (result == 0)
Packit b5b901
    return 0;
Packit b5b901
Packit b5b901
  errno = UV__ERR(result);
Packit b5b901
  return -1;
Packit b5b901
}
Packit b5b901
Packit b5b901
static void uv__to_stat(struct stat* src, uv_stat_t* dst) {
Packit b5b901
  dst->st_dev = src->st_dev;
Packit b5b901
  dst->st_mode = src->st_mode;
Packit b5b901
  dst->st_nlink = src->st_nlink;
Packit b5b901
  dst->st_uid = src->st_uid;
Packit b5b901
  dst->st_gid = src->st_gid;
Packit b5b901
  dst->st_rdev = src->st_rdev;
Packit b5b901
  dst->st_ino = src->st_ino;
Packit b5b901
  dst->st_size = src->st_size;
Packit b5b901
  dst->st_blksize = src->st_blksize;
Packit b5b901
  dst->st_blocks = src->st_blocks;
Packit b5b901
Packit b5b901
#if defined(__APPLE__)
Packit b5b901
  dst->st_atim.tv_sec = src->st_atimespec.tv_sec;
Packit b5b901
  dst->st_atim.tv_nsec = src->st_atimespec.tv_nsec;
Packit b5b901
  dst->st_mtim.tv_sec = src->st_mtimespec.tv_sec;
Packit b5b901
  dst->st_mtim.tv_nsec = src->st_mtimespec.tv_nsec;
Packit b5b901
  dst->st_ctim.tv_sec = src->st_ctimespec.tv_sec;
Packit b5b901
  dst->st_ctim.tv_nsec = src->st_ctimespec.tv_nsec;
Packit b5b901
  dst->st_birthtim.tv_sec = src->st_birthtimespec.tv_sec;
Packit b5b901
  dst->st_birthtim.tv_nsec = src->st_birthtimespec.tv_nsec;
Packit b5b901
  dst->st_flags = src->st_flags;
Packit b5b901
  dst->st_gen = src->st_gen;
Packit b5b901
#elif defined(__ANDROID__)
Packit b5b901
  dst->st_atim.tv_sec = src->st_atime;
Packit b5b901
  dst->st_atim.tv_nsec = src->st_atimensec;
Packit b5b901
  dst->st_mtim.tv_sec = src->st_mtime;
Packit b5b901
  dst->st_mtim.tv_nsec = src->st_mtimensec;
Packit b5b901
  dst->st_ctim.tv_sec = src->st_ctime;
Packit b5b901
  dst->st_ctim.tv_nsec = src->st_ctimensec;
Packit b5b901
  dst->st_birthtim.tv_sec = src->st_ctime;
Packit b5b901
  dst->st_birthtim.tv_nsec = src->st_ctimensec;
Packit b5b901
  dst->st_flags = 0;
Packit b5b901
  dst->st_gen = 0;
Packit b5b901
#elif !defined(_AIX) && (       \
Packit b5b901
    defined(__DragonFly__)   || \
Packit b5b901
    defined(__FreeBSD__)     || \
Packit b5b901
    defined(__OpenBSD__)     || \
Packit b5b901
    defined(__NetBSD__)      || \
Packit b5b901
    defined(_GNU_SOURCE)     || \
Packit b5b901
    defined(_BSD_SOURCE)     || \
Packit b5b901
    defined(_SVID_SOURCE)    || \
Packit b5b901
    defined(_XOPEN_SOURCE)   || \
Packit b5b901
    defined(_DEFAULT_SOURCE))
Packit b5b901
  dst->st_atim.tv_sec = src->st_atim.tv_sec;
Packit b5b901
  dst->st_atim.tv_nsec = src->st_atim.tv_nsec;
Packit b5b901
  dst->st_mtim.tv_sec = src->st_mtim.tv_sec;
Packit b5b901
  dst->st_mtim.tv_nsec = src->st_mtim.tv_nsec;
Packit b5b901
  dst->st_ctim.tv_sec = src->st_ctim.tv_sec;
Packit b5b901
  dst->st_ctim.tv_nsec = src->st_ctim.tv_nsec;
Packit b5b901
# if defined(__FreeBSD__)    || \
Packit b5b901
     defined(__NetBSD__)
Packit b5b901
  dst->st_birthtim.tv_sec = src->st_birthtim.tv_sec;
Packit b5b901
  dst->st_birthtim.tv_nsec = src->st_birthtim.tv_nsec;
Packit b5b901
  dst->st_flags = src->st_flags;
Packit b5b901
  dst->st_gen = src->st_gen;
Packit b5b901
# else
Packit b5b901
  dst->st_birthtim.tv_sec = src->st_ctim.tv_sec;
Packit b5b901
  dst->st_birthtim.tv_nsec = src->st_ctim.tv_nsec;
Packit b5b901
  dst->st_flags = 0;
Packit b5b901
  dst->st_gen = 0;
Packit b5b901
# endif
Packit b5b901
#else
Packit b5b901
  dst->st_atim.tv_sec = src->st_atime;
Packit b5b901
  dst->st_atim.tv_nsec = 0;
Packit b5b901
  dst->st_mtim.tv_sec = src->st_mtime;
Packit b5b901
  dst->st_mtim.tv_nsec = 0;
Packit b5b901
  dst->st_ctim.tv_sec = src->st_ctime;
Packit b5b901
  dst->st_ctim.tv_nsec = 0;
Packit b5b901
  dst->st_birthtim.tv_sec = src->st_ctime;
Packit b5b901
  dst->st_birthtim.tv_nsec = 0;
Packit b5b901
  dst->st_flags = 0;
Packit b5b901
  dst->st_gen = 0;
Packit b5b901
#endif
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit Service e08953
static int uv__fs_statx(int fd,
Packit Service e08953
                        const char* path,
Packit Service e08953
                        int is_fstat,
Packit Service e08953
                        int is_lstat,
Packit Service e08953
                        uv_stat_t* buf) {
Packit Service e08953
  STATIC_ASSERT(UV_ENOSYS != -1);
Packit Service e08953
#ifdef __linux__
Packit Service e08953
  static int no_statx;
Packit Service e08953
  struct uv__statx statxbuf;
Packit Service e08953
  int dirfd;
Packit Service e08953
  int flags;
Packit Service e08953
  int mode;
Packit Service e08953
  int rc;
Packit Service e08953
Packit Service e08953
  if (no_statx)
Packit Service e08953
    return UV_ENOSYS;
Packit Service e08953
Packit Service e08953
  dirfd = AT_FDCWD;
Packit Service e08953
  flags = 0; /* AT_STATX_SYNC_AS_STAT */
Packit Service e08953
  mode = 0xFFF; /* STATX_BASIC_STATS + STATX_BTIME */
Packit Service e08953
Packit Service e08953
  if (is_fstat) {
Packit Service e08953
    dirfd = fd;
Packit Service e08953
    flags |= 0x1000; /* AT_EMPTY_PATH */
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  if (is_lstat)
Packit Service e08953
    flags |= AT_SYMLINK_NOFOLLOW;
Packit Service e08953
Packit Service e08953
  rc = uv__statx(dirfd, path, flags, mode, &statxbuf);
Packit Service e08953
Packit Service e08953
  switch (rc) {
Packit Service e08953
  case 0:
Packit Service e08953
    break;
Packit Service e08953
  case -1:
Packit Service e08953
    /* EPERM happens when a seccomp filter rejects the system call.
Packit Service e08953
     * Has been observed with libseccomp < 2.3.3 and docker < 18.04.
Packit Service e08953
     */
Packit Service e08953
    if (errno != EINVAL && errno != EPERM && errno != ENOSYS)
Packit Service e08953
      return -1;
Packit Service e08953
    /* Fall through. */
Packit Service e08953
  default:
Packit Service e08953
    /* Normally on success, zero is returned and On error, -1 is returned.
Packit Service e08953
     * Observed on S390 RHEL running in a docker container with statx not
Packit Service e08953
     * implemented, rc might return 1 with 0 set as the error code in which
Packit Service e08953
     * case we return ENOSYS.
Packit Service e08953
     */
Packit Service e08953
    no_statx = 1;
Packit Service e08953
    return UV_ENOSYS;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  buf->st_dev = 256 * statxbuf.stx_dev_major + statxbuf.stx_dev_minor;
Packit Service e08953
  buf->st_mode = statxbuf.stx_mode;
Packit Service e08953
  buf->st_nlink = statxbuf.stx_nlink;
Packit Service e08953
  buf->st_uid = statxbuf.stx_uid;
Packit Service e08953
  buf->st_gid = statxbuf.stx_gid;
Packit Service e08953
  buf->st_rdev = statxbuf.stx_rdev_major;
Packit Service e08953
  buf->st_ino = statxbuf.stx_ino;
Packit Service e08953
  buf->st_size = statxbuf.stx_size;
Packit Service e08953
  buf->st_blksize = statxbuf.stx_blksize;
Packit Service e08953
  buf->st_blocks = statxbuf.stx_blocks;
Packit Service e08953
  buf->st_atim.tv_sec = statxbuf.stx_atime.tv_sec;
Packit Service e08953
  buf->st_atim.tv_nsec = statxbuf.stx_atime.tv_nsec;
Packit Service e08953
  buf->st_mtim.tv_sec = statxbuf.stx_mtime.tv_sec;
Packit Service e08953
  buf->st_mtim.tv_nsec = statxbuf.stx_mtime.tv_nsec;
Packit Service e08953
  buf->st_ctim.tv_sec = statxbuf.stx_ctime.tv_sec;
Packit Service e08953
  buf->st_ctim.tv_nsec = statxbuf.stx_ctime.tv_nsec;
Packit Service e08953
  buf->st_birthtim.tv_sec = statxbuf.stx_btime.tv_sec;
Packit Service e08953
  buf->st_birthtim.tv_nsec = statxbuf.stx_btime.tv_nsec;
Packit Service e08953
  buf->st_flags = 0;
Packit Service e08953
  buf->st_gen = 0;
Packit Service e08953
Packit Service e08953
  return 0;
Packit Service e08953
#else
Packit Service e08953
  return UV_ENOSYS;
Packit Service e08953
#endif /* __linux__ */
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit b5b901
static int uv__fs_stat(const char *path, uv_stat_t *buf) {
Packit b5b901
  struct stat pbuf;
Packit b5b901
  int ret;
Packit b5b901
Packit Service e08953
  ret = uv__fs_statx(-1, path, /* is_fstat */ 0, /* is_lstat */ 0, buf);
Packit Service e08953
  if (ret != UV_ENOSYS)
Packit Service e08953
    return ret;
Packit Service e08953
Packit b5b901
  ret = stat(path, &pbuf);
Packit b5b901
  if (ret == 0)
Packit b5b901
    uv__to_stat(&pbuf, buf);
Packit b5b901
Packit b5b901
  return ret;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static int uv__fs_lstat(const char *path, uv_stat_t *buf) {
Packit b5b901
  struct stat pbuf;
Packit b5b901
  int ret;
Packit b5b901
Packit Service e08953
  ret = uv__fs_statx(-1, path, /* is_fstat */ 0, /* is_lstat */ 1, buf);
Packit Service e08953
  if (ret != UV_ENOSYS)
Packit Service e08953
    return ret;
Packit Service e08953
Packit b5b901
  ret = lstat(path, &pbuf);
Packit b5b901
  if (ret == 0)
Packit b5b901
    uv__to_stat(&pbuf, buf);
Packit b5b901
Packit b5b901
  return ret;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static int uv__fs_fstat(int fd, uv_stat_t *buf) {
Packit b5b901
  struct stat pbuf;
Packit b5b901
  int ret;
Packit b5b901
Packit Service e08953
  ret = uv__fs_statx(fd, "", /* is_fstat */ 1, /* is_lstat */ 0, buf);
Packit Service e08953
  if (ret != UV_ENOSYS)
Packit Service e08953
    return ret;
Packit Service e08953
Packit b5b901
  ret = fstat(fd, &pbuf);
Packit b5b901
  if (ret == 0)
Packit b5b901
    uv__to_stat(&pbuf, buf);
Packit b5b901
Packit b5b901
  return ret;
Packit b5b901
}
Packit b5b901
Packit Service e08953
static size_t uv__fs_buf_offset(uv_buf_t* bufs, size_t size) {
Packit Service e08953
  size_t offset;
Packit Service e08953
  /* Figure out which bufs are done */
Packit Service e08953
  for (offset = 0; size > 0 && bufs[offset].len <= size; ++offset)
Packit Service e08953
    size -= bufs[offset].len;
Packit Service e08953
Packit Service e08953
  /* Fix a partial read/write */
Packit Service e08953
  if (size > 0) {
Packit Service e08953
    bufs[offset].base += size;
Packit Service e08953
    bufs[offset].len -= size;
Packit Service e08953
  }
Packit Service e08953
  return offset;
Packit Service e08953
}
Packit b5b901
Packit Service e08953
static ssize_t uv__fs_write_all(uv_fs_t* req) {
Packit b5b901
  unsigned int iovmax;
Packit b5b901
  unsigned int nbufs;
Packit b5b901
  uv_buf_t* bufs;
Packit b5b901
  ssize_t total;
Packit b5b901
  ssize_t result;
Packit b5b901
Packit b5b901
  iovmax = uv__getiovmax();
Packit b5b901
  nbufs = req->nbufs;
Packit b5b901
  bufs = req->bufs;
Packit b5b901
  total = 0;
Packit b5b901
Packit b5b901
  while (nbufs > 0) {
Packit b5b901
    req->nbufs = nbufs;
Packit b5b901
    if (req->nbufs > iovmax)
Packit b5b901
      req->nbufs = iovmax;
Packit b5b901
Packit Service e08953
    do
Packit Service e08953
      result = uv__fs_write(req);
Packit Service e08953
    while (result < 0 && errno == EINTR);
Packit Service e08953
Packit b5b901
    if (result <= 0) {
Packit b5b901
      if (total == 0)
Packit b5b901
        total = result;
Packit b5b901
      break;
Packit b5b901
    }
Packit b5b901
Packit b5b901
    if (req->off >= 0)
Packit b5b901
      req->off += result;
Packit b5b901
Packit Service e08953
    req->nbufs = uv__fs_buf_offset(req->bufs, result);
Packit b5b901
    req->bufs += req->nbufs;
Packit b5b901
    nbufs -= req->nbufs;
Packit b5b901
    total += result;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (bufs != req->bufsml)
Packit b5b901
    uv__free(bufs);
Packit b5b901
Packit b5b901
  req->bufs = NULL;
Packit b5b901
  req->nbufs = 0;
Packit b5b901
Packit b5b901
  return total;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static void uv__fs_work(struct uv__work* w) {
Packit b5b901
  int retry_on_eintr;
Packit b5b901
  uv_fs_t* req;
Packit b5b901
  ssize_t r;
Packit b5b901
Packit b5b901
  req = container_of(w, uv_fs_t, work_req);
Packit Service e08953
  retry_on_eintr = !(req->fs_type == UV_FS_CLOSE ||
Packit Service e08953
                     req->fs_type == UV_FS_READ);
Packit b5b901
Packit b5b901
  do {
Packit b5b901
    errno = 0;
Packit b5b901
Packit b5b901
#define X(type, action)                                                       \
Packit b5b901
  case UV_FS_ ## type:                                                        \
Packit b5b901
    r = action;                                                               \
Packit b5b901
    break;
Packit b5b901
Packit b5b901
    switch (req->fs_type) {
Packit b5b901
    X(ACCESS, access(req->path, req->flags));
Packit b5b901
    X(CHMOD, chmod(req->path, req->mode));
Packit b5b901
    X(CHOWN, chown(req->path, req->uid, req->gid));
Packit Service e08953
    X(CLOSE, uv__fs_close(req->file));
Packit b5b901
    X(COPYFILE, uv__fs_copyfile(req));
Packit b5b901
    X(FCHMOD, fchmod(req->file, req->mode));
Packit b5b901
    X(FCHOWN, fchown(req->file, req->uid, req->gid));
Packit b5b901
    X(LCHOWN, lchown(req->path, req->uid, req->gid));
Packit b5b901
    X(FDATASYNC, uv__fs_fdatasync(req));
Packit b5b901
    X(FSTAT, uv__fs_fstat(req->file, &req->statbuf));
Packit b5b901
    X(FSYNC, uv__fs_fsync(req));
Packit b5b901
    X(FTRUNCATE, ftruncate(req->file, req->off));
Packit b5b901
    X(FUTIME, uv__fs_futime(req));
Packit Service e08953
    X(LUTIME, uv__fs_lutime(req));
Packit b5b901
    X(LSTAT, uv__fs_lstat(req->path, &req->statbuf));
Packit b5b901
    X(LINK, link(req->path, req->new_path));
Packit b5b901
    X(MKDIR, mkdir(req->path, req->mode));
Packit b5b901
    X(MKDTEMP, uv__fs_mkdtemp(req));
Packit Service e08953
    X(MKSTEMP, uv__fs_mkstemp(req));
Packit b5b901
    X(OPEN, uv__fs_open(req));
Packit Service e08953
    X(READ, uv__fs_read(req));
Packit b5b901
    X(SCANDIR, uv__fs_scandir(req));
Packit Service e08953
    X(OPENDIR, uv__fs_opendir(req));
Packit Service e08953
    X(READDIR, uv__fs_readdir(req));
Packit Service e08953
    X(CLOSEDIR, uv__fs_closedir(req));
Packit b5b901
    X(READLINK, uv__fs_readlink(req));
Packit b5b901
    X(REALPATH, uv__fs_realpath(req));
Packit b5b901
    X(RENAME, rename(req->path, req->new_path));
Packit b5b901
    X(RMDIR, rmdir(req->path));
Packit b5b901
    X(SENDFILE, uv__fs_sendfile(req));
Packit b5b901
    X(STAT, uv__fs_stat(req->path, &req->statbuf));
Packit Service e08953
    X(STATFS, uv__fs_statfs(req));
Packit b5b901
    X(SYMLINK, symlink(req->path, req->new_path));
Packit b5b901
    X(UNLINK, unlink(req->path));
Packit b5b901
    X(UTIME, uv__fs_utime(req));
Packit Service e08953
    X(WRITE, uv__fs_write_all(req));
Packit b5b901
    default: abort();
Packit b5b901
    }
Packit b5b901
#undef X
Packit b5b901
  } while (r == -1 && errno == EINTR && retry_on_eintr);
Packit b5b901
Packit b5b901
  if (r == -1)
Packit b5b901
    req->result = UV__ERR(errno);
Packit b5b901
  else
Packit b5b901
    req->result = r;
Packit b5b901
Packit b5b901
  if (r == 0 && (req->fs_type == UV_FS_STAT ||
Packit b5b901
                 req->fs_type == UV_FS_FSTAT ||
Packit b5b901
                 req->fs_type == UV_FS_LSTAT)) {
Packit b5b901
    req->ptr = &req->statbuf;
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static void uv__fs_done(struct uv__work* w, int status) {
Packit b5b901
  uv_fs_t* req;
Packit b5b901
Packit b5b901
  req = container_of(w, uv_fs_t, work_req);
Packit b5b901
  uv__req_unregister(req->loop, req);
Packit b5b901
Packit b5b901
  if (status == UV_ECANCELED) {
Packit b5b901
    assert(req->result == 0);
Packit b5b901
    req->result = UV_ECANCELED;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  req->cb(req);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_access(uv_loop_t* loop,
Packit b5b901
                 uv_fs_t* req,
Packit b5b901
                 const char* path,
Packit b5b901
                 int flags,
Packit b5b901
                 uv_fs_cb cb) {
Packit b5b901
  INIT(ACCESS);
Packit b5b901
  PATH;
Packit b5b901
  req->flags = flags;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_chmod(uv_loop_t* loop,
Packit b5b901
                uv_fs_t* req,
Packit b5b901
                const char* path,
Packit b5b901
                int mode,
Packit b5b901
                uv_fs_cb cb) {
Packit b5b901
  INIT(CHMOD);
Packit b5b901
  PATH;
Packit b5b901
  req->mode = mode;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_chown(uv_loop_t* loop,
Packit b5b901
                uv_fs_t* req,
Packit b5b901
                const char* path,
Packit b5b901
                uv_uid_t uid,
Packit b5b901
                uv_gid_t gid,
Packit b5b901
                uv_fs_cb cb) {
Packit b5b901
  INIT(CHOWN);
Packit b5b901
  PATH;
Packit b5b901
  req->uid = uid;
Packit b5b901
  req->gid = gid;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_close(uv_loop_t* loop, uv_fs_t* req, uv_file file, uv_fs_cb cb) {
Packit b5b901
  INIT(CLOSE);
Packit b5b901
  req->file = file;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_fchmod(uv_loop_t* loop,
Packit b5b901
                 uv_fs_t* req,
Packit b5b901
                 uv_file file,
Packit b5b901
                 int mode,
Packit b5b901
                 uv_fs_cb cb) {
Packit b5b901
  INIT(FCHMOD);
Packit b5b901
  req->file = file;
Packit b5b901
  req->mode = mode;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_fchown(uv_loop_t* loop,
Packit b5b901
                 uv_fs_t* req,
Packit b5b901
                 uv_file file,
Packit b5b901
                 uv_uid_t uid,
Packit b5b901
                 uv_gid_t gid,
Packit b5b901
                 uv_fs_cb cb) {
Packit b5b901
  INIT(FCHOWN);
Packit b5b901
  req->file = file;
Packit b5b901
  req->uid = uid;
Packit b5b901
  req->gid = gid;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_lchown(uv_loop_t* loop,
Packit b5b901
                 uv_fs_t* req,
Packit b5b901
                 const char* path,
Packit b5b901
                 uv_uid_t uid,
Packit b5b901
                 uv_gid_t gid,
Packit b5b901
                 uv_fs_cb cb) {
Packit b5b901
  INIT(LCHOWN);
Packit b5b901
  PATH;
Packit b5b901
  req->uid = uid;
Packit b5b901
  req->gid = gid;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_fdatasync(uv_loop_t* loop, uv_fs_t* req, uv_file file, uv_fs_cb cb) {
Packit b5b901
  INIT(FDATASYNC);
Packit b5b901
  req->file = file;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_fstat(uv_loop_t* loop, uv_fs_t* req, uv_file file, uv_fs_cb cb) {
Packit b5b901
  INIT(FSTAT);
Packit b5b901
  req->file = file;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_fsync(uv_loop_t* loop, uv_fs_t* req, uv_file file, uv_fs_cb cb) {
Packit b5b901
  INIT(FSYNC);
Packit b5b901
  req->file = file;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_ftruncate(uv_loop_t* loop,
Packit b5b901
                    uv_fs_t* req,
Packit b5b901
                    uv_file file,
Packit b5b901
                    int64_t off,
Packit b5b901
                    uv_fs_cb cb) {
Packit b5b901
  INIT(FTRUNCATE);
Packit b5b901
  req->file = file;
Packit b5b901
  req->off = off;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_futime(uv_loop_t* loop,
Packit b5b901
                 uv_fs_t* req,
Packit b5b901
                 uv_file file,
Packit b5b901
                 double atime,
Packit b5b901
                 double mtime,
Packit b5b901
                 uv_fs_cb cb) {
Packit b5b901
  INIT(FUTIME);
Packit b5b901
  req->file = file;
Packit b5b901
  req->atime = atime;
Packit b5b901
  req->mtime = mtime;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit Service e08953
int uv_fs_lutime(uv_loop_t* loop,
Packit Service e08953
                 uv_fs_t* req,
Packit Service e08953
                 const char* path,
Packit Service e08953
                 double atime,
Packit Service e08953
                 double mtime,
Packit Service e08953
                 uv_fs_cb cb) {
Packit Service e08953
  INIT(LUTIME);
Packit Service e08953
  PATH;
Packit Service e08953
  req->atime = atime;
Packit Service e08953
  req->mtime = mtime;
Packit Service e08953
  POST;
Packit Service e08953
}
Packit Service e08953
Packit b5b901
Packit b5b901
int uv_fs_lstat(uv_loop_t* loop, uv_fs_t* req, const char* path, uv_fs_cb cb) {
Packit b5b901
  INIT(LSTAT);
Packit b5b901
  PATH;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_link(uv_loop_t* loop,
Packit b5b901
               uv_fs_t* req,
Packit b5b901
               const char* path,
Packit b5b901
               const char* new_path,
Packit b5b901
               uv_fs_cb cb) {
Packit b5b901
  INIT(LINK);
Packit b5b901
  PATH2;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_mkdir(uv_loop_t* loop,
Packit b5b901
                uv_fs_t* req,
Packit b5b901
                const char* path,
Packit b5b901
                int mode,
Packit b5b901
                uv_fs_cb cb) {
Packit b5b901
  INIT(MKDIR);
Packit b5b901
  PATH;
Packit b5b901
  req->mode = mode;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_mkdtemp(uv_loop_t* loop,
Packit b5b901
                  uv_fs_t* req,
Packit b5b901
                  const char* tpl,
Packit b5b901
                  uv_fs_cb cb) {
Packit b5b901
  INIT(MKDTEMP);
Packit b5b901
  req->path = uv__strdup(tpl);
Packit b5b901
  if (req->path == NULL)
Packit b5b901
    return UV_ENOMEM;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit Service e08953
int uv_fs_mkstemp(uv_loop_t* loop,
Packit Service e08953
                  uv_fs_t* req,
Packit Service e08953
                  const char* tpl,
Packit Service e08953
                  uv_fs_cb cb) {
Packit Service e08953
  INIT(MKSTEMP);
Packit Service e08953
  req->path = uv__strdup(tpl);
Packit Service e08953
  if (req->path == NULL)
Packit Service e08953
    return UV_ENOMEM;
Packit Service e08953
  POST;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit b5b901
int uv_fs_open(uv_loop_t* loop,
Packit b5b901
               uv_fs_t* req,
Packit b5b901
               const char* path,
Packit b5b901
               int flags,
Packit b5b901
               int mode,
Packit b5b901
               uv_fs_cb cb) {
Packit b5b901
  INIT(OPEN);
Packit b5b901
  PATH;
Packit b5b901
  req->flags = flags;
Packit b5b901
  req->mode = mode;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_read(uv_loop_t* loop, uv_fs_t* req,
Packit b5b901
               uv_file file,
Packit b5b901
               const uv_buf_t bufs[],
Packit b5b901
               unsigned int nbufs,
Packit b5b901
               int64_t off,
Packit b5b901
               uv_fs_cb cb) {
Packit b5b901
  INIT(READ);
Packit b5b901
Packit b5b901
  if (bufs == NULL || nbufs == 0)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  req->file = file;
Packit b5b901
Packit b5b901
  req->nbufs = nbufs;
Packit b5b901
  req->bufs = req->bufsml;
Packit b5b901
  if (nbufs > ARRAY_SIZE(req->bufsml))
Packit b5b901
    req->bufs = uv__malloc(nbufs * sizeof(*bufs));
Packit b5b901
Packit b5b901
  if (req->bufs == NULL)
Packit b5b901
    return UV_ENOMEM;
Packit b5b901
Packit b5b901
  memcpy(req->bufs, bufs, nbufs * sizeof(*bufs));
Packit b5b901
Packit b5b901
  req->off = off;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_scandir(uv_loop_t* loop,
Packit b5b901
                  uv_fs_t* req,
Packit b5b901
                  const char* path,
Packit b5b901
                  int flags,
Packit b5b901
                  uv_fs_cb cb) {
Packit b5b901
  INIT(SCANDIR);
Packit b5b901
  PATH;
Packit b5b901
  req->flags = flags;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit Service e08953
int uv_fs_opendir(uv_loop_t* loop,
Packit Service e08953
                  uv_fs_t* req,
Packit Service e08953
                  const char* path,
Packit Service e08953
                  uv_fs_cb cb) {
Packit Service e08953
  INIT(OPENDIR);
Packit Service e08953
  PATH;
Packit Service e08953
  POST;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
int uv_fs_readdir(uv_loop_t* loop,
Packit Service e08953
                  uv_fs_t* req,
Packit Service e08953
                  uv_dir_t* dir,
Packit Service e08953
                  uv_fs_cb cb) {
Packit Service e08953
  INIT(READDIR);
Packit Service e08953
Packit Service e08953
  if (dir == NULL || dir->dir == NULL || dir->dirents == NULL)
Packit Service e08953
    return UV_EINVAL;
Packit Service e08953
Packit Service e08953
  req->ptr = dir;
Packit Service e08953
  POST;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
int uv_fs_closedir(uv_loop_t* loop,
Packit Service e08953
                   uv_fs_t* req,
Packit Service e08953
                   uv_dir_t* dir,
Packit Service e08953
                   uv_fs_cb cb) {
Packit Service e08953
  INIT(CLOSEDIR);
Packit Service e08953
Packit Service e08953
  if (dir == NULL)
Packit Service e08953
    return UV_EINVAL;
Packit Service e08953
Packit Service e08953
  req->ptr = dir;
Packit Service e08953
  POST;
Packit Service e08953
}
Packit b5b901
Packit b5b901
int uv_fs_readlink(uv_loop_t* loop,
Packit b5b901
                   uv_fs_t* req,
Packit b5b901
                   const char* path,
Packit b5b901
                   uv_fs_cb cb) {
Packit b5b901
  INIT(READLINK);
Packit b5b901
  PATH;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_realpath(uv_loop_t* loop,
Packit b5b901
                  uv_fs_t* req,
Packit b5b901
                  const char * path,
Packit b5b901
                  uv_fs_cb cb) {
Packit b5b901
  INIT(REALPATH);
Packit b5b901
  PATH;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_rename(uv_loop_t* loop,
Packit b5b901
                 uv_fs_t* req,
Packit b5b901
                 const char* path,
Packit b5b901
                 const char* new_path,
Packit b5b901
                 uv_fs_cb cb) {
Packit b5b901
  INIT(RENAME);
Packit b5b901
  PATH2;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_rmdir(uv_loop_t* loop, uv_fs_t* req, const char* path, uv_fs_cb cb) {
Packit b5b901
  INIT(RMDIR);
Packit b5b901
  PATH;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_sendfile(uv_loop_t* loop,
Packit b5b901
                   uv_fs_t* req,
Packit b5b901
                   uv_file out_fd,
Packit b5b901
                   uv_file in_fd,
Packit b5b901
                   int64_t off,
Packit b5b901
                   size_t len,
Packit b5b901
                   uv_fs_cb cb) {
Packit b5b901
  INIT(SENDFILE);
Packit b5b901
  req->flags = in_fd; /* hack */
Packit b5b901
  req->file = out_fd;
Packit b5b901
  req->off = off;
Packit b5b901
  req->bufsml[0].len = len;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_stat(uv_loop_t* loop, uv_fs_t* req, const char* path, uv_fs_cb cb) {
Packit b5b901
  INIT(STAT);
Packit b5b901
  PATH;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_symlink(uv_loop_t* loop,
Packit b5b901
                  uv_fs_t* req,
Packit b5b901
                  const char* path,
Packit b5b901
                  const char* new_path,
Packit b5b901
                  int flags,
Packit b5b901
                  uv_fs_cb cb) {
Packit b5b901
  INIT(SYMLINK);
Packit b5b901
  PATH2;
Packit b5b901
  req->flags = flags;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_unlink(uv_loop_t* loop, uv_fs_t* req, const char* path, uv_fs_cb cb) {
Packit b5b901
  INIT(UNLINK);
Packit b5b901
  PATH;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_utime(uv_loop_t* loop,
Packit b5b901
                uv_fs_t* req,
Packit b5b901
                const char* path,
Packit b5b901
                double atime,
Packit b5b901
                double mtime,
Packit b5b901
                uv_fs_cb cb) {
Packit b5b901
  INIT(UTIME);
Packit b5b901
  PATH;
Packit b5b901
  req->atime = atime;
Packit b5b901
  req->mtime = mtime;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_write(uv_loop_t* loop,
Packit b5b901
                uv_fs_t* req,
Packit b5b901
                uv_file file,
Packit b5b901
                const uv_buf_t bufs[],
Packit b5b901
                unsigned int nbufs,
Packit b5b901
                int64_t off,
Packit b5b901
                uv_fs_cb cb) {
Packit b5b901
  INIT(WRITE);
Packit b5b901
Packit b5b901
  if (bufs == NULL || nbufs == 0)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  req->file = file;
Packit b5b901
Packit b5b901
  req->nbufs = nbufs;
Packit b5b901
  req->bufs = req->bufsml;
Packit b5b901
  if (nbufs > ARRAY_SIZE(req->bufsml))
Packit b5b901
    req->bufs = uv__malloc(nbufs * sizeof(*bufs));
Packit b5b901
Packit b5b901
  if (req->bufs == NULL)
Packit b5b901
    return UV_ENOMEM;
Packit b5b901
Packit b5b901
  memcpy(req->bufs, bufs, nbufs * sizeof(*bufs));
Packit b5b901
Packit b5b901
  req->off = off;
Packit b5b901
  POST;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_fs_req_cleanup(uv_fs_t* req) {
Packit b5b901
  if (req == NULL)
Packit b5b901
    return;
Packit b5b901
Packit b5b901
  /* Only necessary for asychronous requests, i.e., requests with a callback.
Packit b5b901
   * Synchronous ones don't copy their arguments and have req->path and
Packit Service e08953
   * req->new_path pointing to user-owned memory.  UV_FS_MKDTEMP and 
Packit Service e08953
   * UV_FS_MKSTEMP are the exception to the rule, they always allocate memory.
Packit b5b901
   */
Packit Service e08953
  if (req->path != NULL &&
Packit Service e08953
      (req->cb != NULL ||
Packit Service e08953
        req->fs_type == UV_FS_MKDTEMP || req->fs_type == UV_FS_MKSTEMP))
Packit b5b901
    uv__free((void*) req->path);  /* Memory is shared with req->new_path. */
Packit b5b901
Packit b5b901
  req->path = NULL;
Packit b5b901
  req->new_path = NULL;
Packit b5b901
Packit Service e08953
  if (req->fs_type == UV_FS_READDIR && req->ptr != NULL)
Packit Service e08953
    uv__fs_readdir_cleanup(req);
Packit Service e08953
Packit b5b901
  if (req->fs_type == UV_FS_SCANDIR && req->ptr != NULL)
Packit b5b901
    uv__fs_scandir_cleanup(req);
Packit b5b901
Packit b5b901
  if (req->bufs != req->bufsml)
Packit b5b901
    uv__free(req->bufs);
Packit b5b901
  req->bufs = NULL;
Packit b5b901
Packit Service e08953
  if (req->fs_type != UV_FS_OPENDIR && req->ptr != &req->statbuf)
Packit b5b901
    uv__free(req->ptr);
Packit b5b901
  req->ptr = NULL;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_copyfile(uv_loop_t* loop,
Packit b5b901
                   uv_fs_t* req,
Packit b5b901
                   const char* path,
Packit b5b901
                   const char* new_path,
Packit b5b901
                   int flags,
Packit b5b901
                   uv_fs_cb cb) {
Packit b5b901
  INIT(COPYFILE);
Packit b5b901
Packit b5b901
  if (flags & ~(UV_FS_COPYFILE_EXCL |
Packit b5b901
                UV_FS_COPYFILE_FICLONE |
Packit b5b901
                UV_FS_COPYFILE_FICLONE_FORCE)) {
Packit b5b901
    return UV_EINVAL;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  PATH2;
Packit b5b901
  req->flags = flags;
Packit b5b901
  POST;
Packit b5b901
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
int uv_fs_statfs(uv_loop_t* loop,
Packit Service e08953
                 uv_fs_t* req,
Packit Service e08953
                 const char* path,
Packit Service e08953
                 uv_fs_cb cb) {
Packit Service e08953
  INIT(STATFS);
Packit Service e08953
  PATH;
Packit Service e08953
  POST;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
int uv_fs_get_system_error(const uv_fs_t* req) {
Packit Service e08953
  return -req->result;
Packit Service e08953
}