• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
2  *
3  * Permission is hereby granted, free of charge, to any person obtaining a copy
4  * of this software and associated documentation files (the "Software"), to
5  * deal in the Software without restriction, including without limitation the
6  * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
7  * sell copies of the Software, and to permit persons to whom the Software is
8  * furnished to do so, subject to the following conditions:
9  *
10  * The above copyright notice and this permission notice shall be included in
11  * all copies or substantial portions of the Software.
12  *
13  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
18  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
19  * IN THE SOFTWARE.
20  */
21 
22 /* Caveat emptor: this file deviates from the libuv convention of returning
23  * negated errno codes. Most uv_fs_*() functions map directly to the system
24  * call of the same name. For more complex wrappers, it's easier to just
25  * return -1 with errno set. The dispatcher in uv__fs_work() takes care of
26  * getting the errno to the right place (req->result or as the return value.)
27  */
28 
29 #include "uv.h"
30 #include "internal.h"
31 
32 #include <errno.h>
33 #include <dlfcn.h>
34 #include <stdio.h>
35 #include <stdlib.h>
36 #include <string.h>
37 #include <limits.h> /* PATH_MAX */
38 
39 #include <sys/types.h>
40 #include <sys/socket.h>
41 #include <sys/stat.h>
42 #include <sys/time.h>
43 #include <sys/uio.h>
44 #include <pthread.h>
45 #include <unistd.h>
46 #include <fcntl.h>
47 #include <poll.h>
48 
49 #if defined(__DragonFly__)        ||                                      \
50     defined(__FreeBSD__)          ||                                      \
51     defined(__FreeBSD_kernel__)   ||                                      \
52     defined(__OpenBSD__)          ||                                      \
53     defined(__NetBSD__)
54 # define HAVE_PREADV 1
55 #else
56 # define HAVE_PREADV 0
57 #endif
58 
59 #if defined(__linux__)
60 # include "sys/utsname.h"
61 #endif
62 
63 #if defined(__linux__) || defined(__sun)
64 # include <sys/sendfile.h>
65 # include <sys/sysmacros.h>
66 #endif
67 
68 #if defined(__APPLE__)
69 # include <sys/sysctl.h>
70 #elif defined(__linux__) && !defined(FICLONE)
71 # include <sys/ioctl.h>
72 # define FICLONE _IOW(0x94, 9, int)
73 #endif
74 
75 #if defined(_AIX) && !defined(_AIX71)
76 # include <utime.h>
77 #endif
78 
79 #if defined(__APPLE__)            ||                                      \
80     defined(__DragonFly__)        ||                                      \
81     defined(__FreeBSD__)          ||                                      \
82     defined(__FreeBSD_kernel__)   ||                                      \
83     defined(__OpenBSD__)          ||                                      \
84     defined(__NetBSD__)
85 # include <sys/param.h>
86 # include <sys/mount.h>
87 #elif defined(__sun)      || \
88       defined(__MVS__)    || \
89       defined(__NetBSD__) || \
90       defined(__HAIKU__)  || \
91       defined(__QNX__)
92 # include <sys/statvfs.h>
93 #else
94 # include <sys/statfs.h>
95 #endif
96 
97 #if defined(_AIX) && _XOPEN_SOURCE <= 600
98 extern char *mkdtemp(char *template); /* See issue #740 on AIX < 7 */
99 #endif
100 
101 #define INIT(subtype)                                                         \
102   do {                                                                        \
103     if (req == NULL)                                                          \
104       return UV_EINVAL;                                                       \
105     UV_REQ_INIT(req, UV_FS);                                                  \
106     req->fs_type = UV_FS_ ## subtype;                                         \
107     req->result = 0;                                                          \
108     req->ptr = NULL;                                                          \
109     req->loop = loop;                                                         \
110     req->path = NULL;                                                         \
111     req->new_path = NULL;                                                     \
112     req->bufs = NULL;                                                         \
113     req->cb = cb;                                                             \
114   }                                                                           \
115   while (0)
116 
117 #define PATH                                                                  \
118   do {                                                                        \
119     assert(path != NULL);                                                     \
120     if (cb == NULL) {                                                         \
121       req->path = path;                                                       \
122     } else {                                                                  \
123       req->path = uv__strdup(path);                                           \
124       if (req->path == NULL)                                                  \
125         return UV_ENOMEM;                                                     \
126     }                                                                         \
127   }                                                                           \
128   while (0)
129 
130 #define PATH2                                                                 \
131   do {                                                                        \
132     if (cb == NULL) {                                                         \
133       req->path = path;                                                       \
134       req->new_path = new_path;                                               \
135     } else {                                                                  \
136       size_t path_len;                                                        \
137       size_t new_path_len;                                                    \
138       path_len = strlen(path) + 1;                                            \
139       new_path_len = strlen(new_path) + 1;                                    \
140       req->path = uv__malloc(path_len + new_path_len);                        \
141       if (req->path == NULL)                                                  \
142         return UV_ENOMEM;                                                     \
143       req->new_path = req->path + path_len;                                   \
144       memcpy((void*) req->path, path, path_len);                              \
145       memcpy((void*) req->new_path, new_path, new_path_len);                  \
146     }                                                                         \
147   }                                                                           \
148   while (0)
149 
150 #define POST                                                                  \
151   do {                                                                        \
152     if (cb != NULL) {                                                         \
153       uv__req_register(loop, req);                                            \
154       uv__work_submit(loop,                                                   \
155                       &req->work_req,                                         \
156                       UV__WORK_FAST_IO,                                       \
157                       uv__fs_work,                                            \
158                       uv__fs_done);                                           \
159       return 0;                                                               \
160     }                                                                         \
161     else {                                                                    \
162       uv__fs_work(&req->work_req);                                            \
163       return req->result;                                                     \
164     }                                                                         \
165   }                                                                           \
166   while (0)
167 
168 
uv__fs_close(int fd)169 static int uv__fs_close(int fd) {
170   int rc;
171 
172   rc = uv__close_nocancel(fd);
173   if (rc == -1)
174     if (errno == EINTR || errno == EINPROGRESS)
175       rc = 0;  /* The close is in progress, not an error. */
176 
177   return rc;
178 }
179 
180 
uv__fs_fsync(uv_fs_t * req)181 static ssize_t uv__fs_fsync(uv_fs_t* req) {
182 #if defined(__APPLE__)
183   /* Apple's fdatasync and fsync explicitly do NOT flush the drive write cache
184    * to the drive platters. This is in contrast to Linux's fdatasync and fsync
185    * which do, according to recent man pages. F_FULLFSYNC is Apple's equivalent
186    * for flushing buffered data to permanent storage. If F_FULLFSYNC is not
187    * supported by the file system we fall back to F_BARRIERFSYNC or fsync().
188    * This is the same approach taken by sqlite, except sqlite does not issue
189    * an F_BARRIERFSYNC call.
190    */
191   int r;
192 
193   r = fcntl(req->file, F_FULLFSYNC);
194   if (r != 0)
195     r = fcntl(req->file, 85 /* F_BARRIERFSYNC */);  /* fsync + barrier */
196   if (r != 0)
197     r = fsync(req->file);
198   return r;
199 #else
200   return fsync(req->file);
201 #endif
202 }
203 
204 
uv__fs_fdatasync(uv_fs_t * req)205 static ssize_t uv__fs_fdatasync(uv_fs_t* req) {
206 #if defined(__linux__) || defined(__sun) || defined(__NetBSD__)
207   return fdatasync(req->file);
208 #elif defined(__APPLE__)
209   /* See the comment in uv__fs_fsync. */
210   return uv__fs_fsync(req);
211 #else
212   return fsync(req->file);
213 #endif
214 }
215 
216 
UV_UNUSED(static struct timespec uv__fs_to_timespec (double time))217 UV_UNUSED(static struct timespec uv__fs_to_timespec(double time)) {
218   struct timespec ts;
219   ts.tv_sec  = time;
220   ts.tv_nsec = (time - ts.tv_sec) * 1e9;
221 
222  /* TODO(bnoordhuis) Remove this. utimesat() has nanosecond resolution but we
223   * stick to microsecond resolution for the sake of consistency with other
224   * platforms. I'm the original author of this compatibility hack but I'm
225   * less convinced it's useful nowadays.
226   */
227   ts.tv_nsec -= ts.tv_nsec % 1000;
228 
229   if (ts.tv_nsec < 0) {
230     ts.tv_nsec += 1e9;
231     ts.tv_sec -= 1;
232   }
233   return ts;
234 }
235 
UV_UNUSED(static struct timeval uv__fs_to_timeval (double time))236 UV_UNUSED(static struct timeval uv__fs_to_timeval(double time)) {
237   struct timeval tv;
238   tv.tv_sec  = time;
239   tv.tv_usec = (time - tv.tv_sec) * 1e6;
240   if (tv.tv_usec < 0) {
241     tv.tv_usec += 1e6;
242     tv.tv_sec -= 1;
243   }
244   return tv;
245 }
246 
uv__fs_futime(uv_fs_t * req)247 static ssize_t uv__fs_futime(uv_fs_t* req) {
248 #if defined(__linux__)                                                        \
249     || defined(_AIX71)                                                        \
250     || defined(__HAIKU__)                                                     \
251     || defined(__GNU__)
252   struct timespec ts[2];
253   ts[0] = uv__fs_to_timespec(req->atime);
254   ts[1] = uv__fs_to_timespec(req->mtime);
255   return futimens(req->file, ts);
256 #elif defined(__APPLE__)                                                      \
257     || defined(__DragonFly__)                                                 \
258     || defined(__FreeBSD__)                                                   \
259     || defined(__FreeBSD_kernel__)                                            \
260     || defined(__NetBSD__)                                                    \
261     || defined(__OpenBSD__)                                                   \
262     || defined(__sun)
263   struct timeval tv[2];
264   tv[0] = uv__fs_to_timeval(req->atime);
265   tv[1] = uv__fs_to_timeval(req->mtime);
266 # if defined(__sun)
267   return futimesat(req->file, NULL, tv);
268 # else
269   return futimes(req->file, tv);
270 # endif
271 #elif defined(__MVS__)
272   attrib_t atr;
273   memset(&atr, 0, sizeof(atr));
274   atr.att_mtimechg = 1;
275   atr.att_atimechg = 1;
276   atr.att_mtime = req->mtime;
277   atr.att_atime = req->atime;
278   return __fchattr(req->file, &atr, sizeof(atr));
279 #else
280   errno = ENOSYS;
281   return -1;
282 #endif
283 }
284 
285 
uv__fs_mkdtemp(uv_fs_t * req)286 static ssize_t uv__fs_mkdtemp(uv_fs_t* req) {
287   return mkdtemp((char*) req->path) ? 0 : -1;
288 }
289 
290 
291 static int (*uv__mkostemp)(char*, int);
292 
293 
uv__mkostemp_initonce(void)294 static void uv__mkostemp_initonce(void) {
295   /* z/os doesn't have RTLD_DEFAULT but that's okay
296    * because it doesn't have mkostemp(O_CLOEXEC) either.
297    */
298 #ifdef RTLD_DEFAULT
299   uv__mkostemp = (int (*)(char*, int)) dlsym(RTLD_DEFAULT, "mkostemp");
300 
301   /* We don't care about errors, but we do want to clean them up.
302    * If there has been no error, then dlerror() will just return
303    * NULL.
304    */
305   dlerror();
306 #endif  /* RTLD_DEFAULT */
307 }
308 
309 
uv__fs_mkstemp(uv_fs_t * req)310 static int uv__fs_mkstemp(uv_fs_t* req) {
311   static uv_once_t once = UV_ONCE_INIT;
312   int r;
313 #ifdef O_CLOEXEC
314   static int no_cloexec_support;
315 #endif
316   static const char pattern[] = "XXXXXX";
317   static const size_t pattern_size = sizeof(pattern) - 1;
318   char* path;
319   size_t path_length;
320 
321   path = (char*) req->path;
322   path_length = strlen(path);
323 
324   /* EINVAL can be returned for 2 reasons:
325       1. The template's last 6 characters were not XXXXXX
326       2. open() didn't support O_CLOEXEC
327      We want to avoid going to the fallback path in case
328      of 1, so it's manually checked before. */
329   if (path_length < pattern_size ||
330       strcmp(path + path_length - pattern_size, pattern)) {
331     errno = EINVAL;
332     r = -1;
333     goto clobber;
334   }
335 
336   uv_once(&once, uv__mkostemp_initonce);
337 
338 #ifdef O_CLOEXEC
339   if (uv__load_relaxed(&no_cloexec_support) == 0 && uv__mkostemp != NULL) {
340     r = uv__mkostemp(path, O_CLOEXEC);
341 
342     if (r >= 0)
343       return r;
344 
345     /* If mkostemp() returns EINVAL, it means the kernel doesn't
346        support O_CLOEXEC, so we just fallback to mkstemp() below. */
347     if (errno != EINVAL)
348       goto clobber;
349 
350     /* We set the static variable so that next calls don't even
351        try to use mkostemp. */
352     uv__store_relaxed(&no_cloexec_support, 1);
353   }
354 #endif  /* O_CLOEXEC */
355 
356   if (req->cb != NULL)
357     uv_rwlock_rdlock(&req->loop->cloexec_lock);
358 
359   r = mkstemp(path);
360 
361   /* In case of failure `uv__cloexec` will leave error in `errno`,
362    * so it is enough to just set `r` to `-1`.
363    */
364   if (r >= 0 && uv__cloexec(r, 1) != 0) {
365     r = uv__close(r);
366     if (r != 0)
367       abort();
368     r = -1;
369   }
370 
371   if (req->cb != NULL)
372     uv_rwlock_rdunlock(&req->loop->cloexec_lock);
373 
374 clobber:
375   if (r < 0)
376     path[0] = '\0';
377   return r;
378 }
379 
380 
uv__fs_open(uv_fs_t * req)381 static ssize_t uv__fs_open(uv_fs_t* req) {
382 #ifdef O_CLOEXEC
383   return open(req->path, req->flags | O_CLOEXEC, req->mode);
384 #else  /* O_CLOEXEC */
385   int r;
386 
387   if (req->cb != NULL)
388     uv_rwlock_rdlock(&req->loop->cloexec_lock);
389 
390   r = open(req->path, req->flags, req->mode);
391 
392   /* In case of failure `uv__cloexec` will leave error in `errno`,
393    * so it is enough to just set `r` to `-1`.
394    */
395   if (r >= 0 && uv__cloexec(r, 1) != 0) {
396     r = uv__close(r);
397     if (r != 0)
398       abort();
399     r = -1;
400   }
401 
402   if (req->cb != NULL)
403     uv_rwlock_rdunlock(&req->loop->cloexec_lock);
404 
405   return r;
406 #endif  /* O_CLOEXEC */
407 }
408 
409 
410 #if !HAVE_PREADV
uv__fs_preadv(uv_file fd,uv_buf_t * bufs,unsigned int nbufs,off_t off)411 static ssize_t uv__fs_preadv(uv_file fd,
412                              uv_buf_t* bufs,
413                              unsigned int nbufs,
414                              off_t off) {
415   uv_buf_t* buf;
416   uv_buf_t* end;
417   ssize_t result;
418   ssize_t rc;
419   size_t pos;
420 
421   assert(nbufs > 0);
422 
423   result = 0;
424   pos = 0;
425   buf = bufs + 0;
426   end = bufs + nbufs;
427 
428   for (;;) {
429     do
430       rc = pread(fd, buf->base + pos, buf->len - pos, off + result);
431     while (rc == -1 && errno == EINTR);
432 
433     if (rc == 0)
434       break;
435 
436     if (rc == -1 && result == 0)
437       return UV__ERR(errno);
438 
439     if (rc == -1)
440       break;  /* We read some data so return that, ignore the error. */
441 
442     pos += rc;
443     result += rc;
444 
445     if (pos < buf->len)
446       continue;
447 
448     pos = 0;
449     buf += 1;
450 
451     if (buf == end)
452       break;
453   }
454 
455   return result;
456 }
457 #endif
458 
459 
uv__fs_read(uv_fs_t * req)460 static ssize_t uv__fs_read(uv_fs_t* req) {
461 #if defined(__linux__)
462   static int no_preadv;
463 #endif
464   unsigned int iovmax;
465   ssize_t result;
466 
467   iovmax = uv__getiovmax();
468   if (req->nbufs > iovmax)
469     req->nbufs = iovmax;
470 
471   if (req->off < 0) {
472     if (req->nbufs == 1)
473       result = read(req->file, req->bufs[0].base, req->bufs[0].len);
474     else
475       result = readv(req->file, (struct iovec*) req->bufs, req->nbufs);
476   } else {
477     if (req->nbufs == 1) {
478       result = pread(req->file, req->bufs[0].base, req->bufs[0].len, req->off);
479       goto done;
480     }
481 
482 #if HAVE_PREADV
483     result = preadv(req->file, (struct iovec*) req->bufs, req->nbufs, req->off);
484 #else
485 # if defined(__linux__)
486     if (uv__load_relaxed(&no_preadv)) retry:
487 # endif
488     {
489       result = uv__fs_preadv(req->file, req->bufs, req->nbufs, req->off);
490     }
491 # if defined(__linux__)
492     else {
493       result = uv__preadv(req->file,
494                           (struct iovec*)req->bufs,
495                           req->nbufs,
496                           req->off);
497       if (result == -1 && errno == ENOSYS) {
498         uv__store_relaxed(&no_preadv, 1);
499         goto retry;
500       }
501     }
502 # endif
503 #endif
504   }
505 
506 done:
507   /* Early cleanup of bufs allocation, since we're done with it. */
508   if (req->bufs != req->bufsml)
509     uv__free(req->bufs);
510 
511   req->bufs = NULL;
512   req->nbufs = 0;
513 
514 #ifdef __PASE__
515   /* PASE returns EOPNOTSUPP when reading a directory, convert to EISDIR */
516   if (result == -1 && errno == EOPNOTSUPP) {
517     struct stat buf;
518     ssize_t rc;
519     rc = fstat(req->file, &buf);
520     if (rc == 0 && S_ISDIR(buf.st_mode)) {
521       errno = EISDIR;
522     }
523   }
524 #endif
525 
526   return result;
527 }
528 
529 
530 #if defined(__APPLE__) && !defined(MAC_OS_X_VERSION_10_8)
531 #define UV_CONST_DIRENT uv__dirent_t
532 #else
533 #define UV_CONST_DIRENT const uv__dirent_t
534 #endif
535 
536 
uv__fs_scandir_filter(UV_CONST_DIRENT * dent)537 static int uv__fs_scandir_filter(UV_CONST_DIRENT* dent) {
538   return strcmp(dent->d_name, ".") != 0 && strcmp(dent->d_name, "..") != 0;
539 }
540 
541 
uv__fs_scandir_sort(UV_CONST_DIRENT ** a,UV_CONST_DIRENT ** b)542 static int uv__fs_scandir_sort(UV_CONST_DIRENT** a, UV_CONST_DIRENT** b) {
543   return strcmp((*a)->d_name, (*b)->d_name);
544 }
545 
546 
uv__fs_scandir(uv_fs_t * req)547 static ssize_t uv__fs_scandir(uv_fs_t* req) {
548   uv__dirent_t** dents;
549   int n;
550 
551   dents = NULL;
552   n = scandir(req->path, &dents, uv__fs_scandir_filter, uv__fs_scandir_sort);
553 
554   /* NOTE: We will use nbufs as an index field */
555   req->nbufs = 0;
556 
557   if (n == 0) {
558     /* OS X still needs to deallocate some memory.
559      * Memory was allocated using the system allocator, so use free() here.
560      */
561     free(dents);
562     dents = NULL;
563   } else if (n == -1) {
564     return n;
565   }
566 
567   req->ptr = dents;
568 
569   return n;
570 }
571 
uv__fs_opendir(uv_fs_t * req)572 static int uv__fs_opendir(uv_fs_t* req) {
573   uv_dir_t* dir;
574 
575   dir = uv__malloc(sizeof(*dir));
576   if (dir == NULL)
577     goto error;
578 
579   dir->dir = opendir(req->path);
580   if (dir->dir == NULL)
581     goto error;
582 
583   req->ptr = dir;
584   return 0;
585 
586 error:
587   uv__free(dir);
588   req->ptr = NULL;
589   return -1;
590 }
591 
uv__fs_readdir(uv_fs_t * req)592 static int uv__fs_readdir(uv_fs_t* req) {
593   uv_dir_t* dir;
594   uv_dirent_t* dirent;
595   struct dirent* res;
596   unsigned int dirent_idx;
597   unsigned int i;
598 
599   dir = req->ptr;
600   dirent_idx = 0;
601 
602   while (dirent_idx < dir->nentries) {
603     /* readdir() returns NULL on end of directory, as well as on error. errno
604        is used to differentiate between the two conditions. */
605     errno = 0;
606     res = readdir(dir->dir);
607 
608     if (res == NULL) {
609       if (errno != 0)
610         goto error;
611       break;
612     }
613 
614     if (strcmp(res->d_name, ".") == 0 || strcmp(res->d_name, "..") == 0)
615       continue;
616 
617     dirent = &dir->dirents[dirent_idx];
618     dirent->name = uv__strdup(res->d_name);
619 
620     if (dirent->name == NULL)
621       goto error;
622 
623     dirent->type = uv__fs_get_dirent_type(res);
624     ++dirent_idx;
625   }
626 
627   return dirent_idx;
628 
629 error:
630   for (i = 0; i < dirent_idx; ++i) {
631     uv__free((char*) dir->dirents[i].name);
632     dir->dirents[i].name = NULL;
633   }
634 
635   return -1;
636 }
637 
uv__fs_closedir(uv_fs_t * req)638 static int uv__fs_closedir(uv_fs_t* req) {
639   uv_dir_t* dir;
640 
641   dir = req->ptr;
642 
643   if (dir->dir != NULL) {
644     closedir(dir->dir);
645     dir->dir = NULL;
646   }
647 
648   uv__free(req->ptr);
649   req->ptr = NULL;
650   return 0;
651 }
652 
uv__fs_statfs(uv_fs_t * req)653 static int uv__fs_statfs(uv_fs_t* req) {
654   uv_statfs_t* stat_fs;
655 #if defined(__sun)      || \
656     defined(__MVS__)    || \
657     defined(__NetBSD__) || \
658     defined(__HAIKU__)  || \
659     defined(__QNX__)
660   struct statvfs buf;
661 
662   if (0 != statvfs(req->path, &buf))
663 #else
664   struct statfs buf;
665 
666   if (0 != statfs(req->path, &buf))
667 #endif /* defined(__sun) */
668     return -1;
669 
670   stat_fs = uv__malloc(sizeof(*stat_fs));
671   if (stat_fs == NULL) {
672     errno = ENOMEM;
673     return -1;
674   }
675 
676 #if defined(__sun)        || \
677     defined(__MVS__)      || \
678     defined(__OpenBSD__)  || \
679     defined(__NetBSD__)   || \
680     defined(__HAIKU__)    || \
681     defined(__QNX__)
682   stat_fs->f_type = 0;  /* f_type is not supported. */
683 #else
684   stat_fs->f_type = buf.f_type;
685 #endif
686   stat_fs->f_bsize = buf.f_bsize;
687   stat_fs->f_blocks = buf.f_blocks;
688   stat_fs->f_bfree = buf.f_bfree;
689   stat_fs->f_bavail = buf.f_bavail;
690   stat_fs->f_files = buf.f_files;
691   stat_fs->f_ffree = buf.f_ffree;
692   req->ptr = stat_fs;
693   return 0;
694 }
695 
uv__fs_pathmax_size(const char * path)696 static ssize_t uv__fs_pathmax_size(const char* path) {
697   ssize_t pathmax;
698 
699   pathmax = pathconf(path, _PC_PATH_MAX);
700 
701   if (pathmax == -1)
702     pathmax = UV__PATH_MAX;
703 
704   return pathmax;
705 }
706 
uv__fs_readlink(uv_fs_t * req)707 static ssize_t uv__fs_readlink(uv_fs_t* req) {
708   ssize_t maxlen;
709   ssize_t len;
710   char* buf;
711 
712 #if defined(_POSIX_PATH_MAX) || defined(PATH_MAX)
713   maxlen = uv__fs_pathmax_size(req->path);
714 #else
715   /* We may not have a real PATH_MAX.  Read size of link.  */
716   struct stat st;
717   int ret;
718   ret = lstat(req->path, &st);
719   if (ret != 0)
720     return -1;
721   if (!S_ISLNK(st.st_mode)) {
722     errno = EINVAL;
723     return -1;
724   }
725 
726   maxlen = st.st_size;
727 
728   /* According to readlink(2) lstat can report st_size == 0
729      for some symlinks, such as those in /proc or /sys.  */
730   if (maxlen == 0)
731     maxlen = uv__fs_pathmax_size(req->path);
732 #endif
733 
734   buf = uv__malloc(maxlen);
735 
736   if (buf == NULL) {
737     errno = ENOMEM;
738     return -1;
739   }
740 
741 #if defined(__MVS__)
742   len = os390_readlink(req->path, buf, maxlen);
743 #else
744   len = readlink(req->path, buf, maxlen);
745 #endif
746 
747   if (len == -1) {
748     uv__free(buf);
749     return -1;
750   }
751 
752   /* Uncommon case: resize to make room for the trailing nul byte. */
753   if (len == maxlen) {
754     buf = uv__reallocf(buf, len + 1);
755 
756     if (buf == NULL)
757       return -1;
758   }
759 
760   buf[len] = '\0';
761   req->ptr = buf;
762 
763   return 0;
764 }
765 
uv__fs_realpath(uv_fs_t * req)766 static ssize_t uv__fs_realpath(uv_fs_t* req) {
767   char* buf;
768 
769 #if defined(_POSIX_VERSION) && _POSIX_VERSION >= 200809L
770   buf = realpath(req->path, NULL);
771   if (buf == NULL)
772     return -1;
773 #else
774   ssize_t len;
775 
776   len = uv__fs_pathmax_size(req->path);
777   buf = uv__malloc(len + 1);
778 
779   if (buf == NULL) {
780     errno = ENOMEM;
781     return -1;
782   }
783 
784   if (realpath(req->path, buf) == NULL) {
785     uv__free(buf);
786     return -1;
787   }
788 #endif
789 
790   req->ptr = buf;
791 
792   return 0;
793 }
794 
uv__fs_sendfile_emul(uv_fs_t * req)795 static ssize_t uv__fs_sendfile_emul(uv_fs_t* req) {
796   struct pollfd pfd;
797   int use_pread;
798   off_t offset;
799   ssize_t nsent;
800   ssize_t nread;
801   ssize_t nwritten;
802   size_t buflen;
803   size_t len;
804   ssize_t n;
805   int in_fd;
806   int out_fd;
807   char buf[8192];
808 
809   len = req->bufsml[0].len;
810   in_fd = req->flags;
811   out_fd = req->file;
812   offset = req->off;
813   use_pread = 1;
814 
815   /* Here are the rules regarding errors:
816    *
817    * 1. Read errors are reported only if nsent==0, otherwise we return nsent.
818    *    The user needs to know that some data has already been sent, to stop
819    *    them from sending it twice.
820    *
821    * 2. Write errors are always reported. Write errors are bad because they
822    *    mean data loss: we've read data but now we can't write it out.
823    *
824    * We try to use pread() and fall back to regular read() if the source fd
825    * doesn't support positional reads, for example when it's a pipe fd.
826    *
827    * If we get EAGAIN when writing to the target fd, we poll() on it until
828    * it becomes writable again.
829    *
830    * FIXME: If we get a write error when use_pread==1, it should be safe to
831    *        return the number of sent bytes instead of an error because pread()
832    *        is, in theory, idempotent. However, special files in /dev or /proc
833    *        may support pread() but not necessarily return the same data on
834    *        successive reads.
835    *
836    * FIXME: There is no way now to signal that we managed to send *some* data
837    *        before a write error.
838    */
839   for (nsent = 0; (size_t) nsent < len; ) {
840     buflen = len - nsent;
841 
842     if (buflen > sizeof(buf))
843       buflen = sizeof(buf);
844 
845     do
846       if (use_pread)
847         nread = pread(in_fd, buf, buflen, offset);
848       else
849         nread = read(in_fd, buf, buflen);
850     while (nread == -1 && errno == EINTR);
851 
852     if (nread == 0)
853       goto out;
854 
855     if (nread == -1) {
856       if (use_pread && nsent == 0 && (errno == EIO || errno == ESPIPE)) {
857         use_pread = 0;
858         continue;
859       }
860 
861       if (nsent == 0)
862         nsent = -1;
863 
864       goto out;
865     }
866 
867     for (nwritten = 0; nwritten < nread; ) {
868       do
869         n = write(out_fd, buf + nwritten, nread - nwritten);
870       while (n == -1 && errno == EINTR);
871 
872       if (n != -1) {
873         nwritten += n;
874         continue;
875       }
876 
877       if (errno != EAGAIN && errno != EWOULDBLOCK) {
878         nsent = -1;
879         goto out;
880       }
881 
882       pfd.fd = out_fd;
883       pfd.events = POLLOUT;
884       pfd.revents = 0;
885 
886       do
887         n = poll(&pfd, 1, -1);
888       while (n == -1 && errno == EINTR);
889 
890       if (n == -1 || (pfd.revents & ~POLLOUT) != 0) {
891         errno = EIO;
892         nsent = -1;
893         goto out;
894       }
895     }
896 
897     offset += nread;
898     nsent += nread;
899   }
900 
901 out:
902   if (nsent != -1)
903     req->off = offset;
904 
905   return nsent;
906 }
907 
908 
909 #ifdef __linux__
uv__kernel_version(void)910 static unsigned uv__kernel_version(void) {
911   static unsigned cached_version;
912   struct utsname u;
913   unsigned version;
914   unsigned major;
915   unsigned minor;
916   unsigned patch;
917 
918   version = uv__load_relaxed(&cached_version);
919   if (version != 0)
920     return version;
921 
922   if (-1 == uname(&u))
923     return 0;
924 
925   if (3 != sscanf(u.release, "%u.%u.%u", &major, &minor, &patch))
926     return 0;
927 
928   version = major * 65536 + minor * 256 + patch;
929   uv__store_relaxed(&cached_version, version);
930 
931   return version;
932 }
933 
934 
935 /* Pre-4.20 kernels have a bug where CephFS uses the RADOS copy-from command
936  * in copy_file_range() when it shouldn't. There is no workaround except to
937  * fall back to a regular copy.
938  */
uv__is_buggy_cephfs(int fd)939 static int uv__is_buggy_cephfs(int fd) {
940   struct statfs s;
941 
942   if (-1 == fstatfs(fd, &s))
943     return 0;
944 
945   if (s.f_type != /* CephFS */ 0xC36400)
946     return 0;
947 
948   return uv__kernel_version() < /* 4.20.0 */ 0x041400;
949 }
950 
951 
uv__is_cifs_or_smb(int fd)952 static int uv__is_cifs_or_smb(int fd) {
953   struct statfs s;
954 
955   if (-1 == fstatfs(fd, &s))
956     return 0;
957 
958   switch ((unsigned) s.f_type) {
959   case 0x0000517Bu:  /* SMB */
960   case 0xFE534D42u:  /* SMB2 */
961   case 0xFF534D42u:  /* CIFS */
962     return 1;
963   }
964 
965   return 0;
966 }
967 
968 
uv__fs_try_copy_file_range(int in_fd,off_t * off,int out_fd,size_t len)969 static ssize_t uv__fs_try_copy_file_range(int in_fd, off_t* off,
970                                           int out_fd, size_t len) {
971   static int no_copy_file_range_support;
972   ssize_t r;
973 
974   if (uv__load_relaxed(&no_copy_file_range_support)) {
975     errno = ENOSYS;
976     return -1;
977   }
978 
979   r = uv__fs_copy_file_range(in_fd, off, out_fd, NULL, len, 0);
980 
981   if (r != -1)
982     return r;
983 
984   switch (errno) {
985   case EACCES:
986     /* Pre-4.20 kernels have a bug where CephFS uses the RADOS
987      * copy-from command when it shouldn't.
988      */
989     if (uv__is_buggy_cephfs(in_fd))
990       errno = ENOSYS;  /* Use fallback. */
991     break;
992   case ENOSYS:
993     uv__store_relaxed(&no_copy_file_range_support, 1);
994     break;
995   case EPERM:
996     /* It's been reported that CIFS spuriously fails.
997      * Consider it a transient error.
998      */
999     if (uv__is_cifs_or_smb(out_fd))
1000       errno = ENOSYS;  /* Use fallback. */
1001     break;
1002   case ENOTSUP:
1003   case EXDEV:
1004     /* ENOTSUP - it could work on another file system type.
1005      * EXDEV - it will not work when in_fd and out_fd are not on the same
1006      *         mounted filesystem (pre Linux 5.3)
1007      */
1008     errno = ENOSYS;  /* Use fallback. */
1009     break;
1010   }
1011 
1012   return -1;
1013 }
1014 
1015 #endif  /* __linux__ */
1016 
1017 
uv__fs_sendfile(uv_fs_t * req)1018 static ssize_t uv__fs_sendfile(uv_fs_t* req) {
1019   int in_fd;
1020   int out_fd;
1021 
1022   in_fd = req->flags;
1023   out_fd = req->file;
1024 
1025 #if defined(__linux__) || defined(__sun)
1026   {
1027     off_t off;
1028     ssize_t r;
1029     size_t len;
1030     int try_sendfile;
1031 
1032     off = req->off;
1033     len = req->bufsml[0].len;
1034     try_sendfile = 1;
1035 
1036 #ifdef __linux__
1037     r = uv__fs_try_copy_file_range(in_fd, &off, out_fd, len);
1038     try_sendfile = (r == -1 && errno == ENOSYS);
1039 #endif
1040 
1041     if (try_sendfile)
1042       r = sendfile(out_fd, in_fd, &off, len);
1043 
1044     /* sendfile() on SunOS returns EINVAL if the target fd is not a socket but
1045      * it still writes out data. Fortunately, we can detect it by checking if
1046      * the offset has been updated.
1047      */
1048     if (r != -1 || off > req->off) {
1049       r = off - req->off;
1050       req->off = off;
1051       return r;
1052     }
1053 
1054     if (errno == EINVAL ||
1055         errno == EIO ||
1056         errno == ENOTSOCK ||
1057         errno == EXDEV) {
1058       errno = 0;
1059       return uv__fs_sendfile_emul(req);
1060     }
1061 
1062     return -1;
1063   }
1064 #elif defined(__APPLE__)           || \
1065       defined(__DragonFly__)       || \
1066       defined(__FreeBSD__)         || \
1067       defined(__FreeBSD_kernel__)
1068   {
1069     off_t len;
1070     ssize_t r;
1071 
1072     /* sendfile() on FreeBSD and Darwin returns EAGAIN if the target fd is in
1073      * non-blocking mode and not all data could be written. If a non-zero
1074      * number of bytes have been sent, we don't consider it an error.
1075      */
1076 
1077 #if defined(__FreeBSD__) || defined(__DragonFly__)
1078 #if defined(__FreeBSD__)
1079     off_t off;
1080 
1081     off = req->off;
1082     r = uv__fs_copy_file_range(in_fd, &off, out_fd, NULL, req->bufsml[0].len, 0);
1083     if (r >= 0) {
1084         r = off - req->off;
1085         req->off = off;
1086         return r;
1087     }
1088 #endif
1089     len = 0;
1090     r = sendfile(in_fd, out_fd, req->off, req->bufsml[0].len, NULL, &len, 0);
1091 #elif defined(__FreeBSD_kernel__)
1092     len = 0;
1093     r = bsd_sendfile(in_fd,
1094                      out_fd,
1095                      req->off,
1096                      req->bufsml[0].len,
1097                      NULL,
1098                      &len,
1099                      0);
1100 #else
1101     /* The darwin sendfile takes len as an input for the length to send,
1102      * so make sure to initialize it with the caller's value. */
1103     len = req->bufsml[0].len;
1104     r = sendfile(in_fd, out_fd, req->off, &len, NULL, 0);
1105 #endif
1106 
1107      /*
1108      * The man page for sendfile(2) on DragonFly states that `len` contains
1109      * a meaningful value ONLY in case of EAGAIN and EINTR.
1110      * Nothing is said about it's value in case of other errors, so better
1111      * not depend on the potential wrong assumption that is was not modified
1112      * by the syscall.
1113      */
1114     if (r == 0 || ((errno == EAGAIN || errno == EINTR) && len != 0)) {
1115       req->off += len;
1116       return (ssize_t) len;
1117     }
1118 
1119     if (errno == EINVAL ||
1120         errno == EIO ||
1121         errno == ENOTSOCK ||
1122         errno == EXDEV) {
1123       errno = 0;
1124       return uv__fs_sendfile_emul(req);
1125     }
1126 
1127     return -1;
1128   }
1129 #else
1130   /* Squelch compiler warnings. */
1131   (void) &in_fd;
1132   (void) &out_fd;
1133 
1134   return uv__fs_sendfile_emul(req);
1135 #endif
1136 }
1137 
1138 
uv__fs_utime(uv_fs_t * req)1139 static ssize_t uv__fs_utime(uv_fs_t* req) {
1140 #if defined(__linux__)                                                         \
1141     || defined(_AIX71)                                                         \
1142     || defined(__sun)                                                          \
1143     || defined(__HAIKU__)
1144   struct timespec ts[2];
1145   ts[0] = uv__fs_to_timespec(req->atime);
1146   ts[1] = uv__fs_to_timespec(req->mtime);
1147   return utimensat(AT_FDCWD, req->path, ts, 0);
1148 #elif defined(__APPLE__)                                                      \
1149     || defined(__DragonFly__)                                                 \
1150     || defined(__FreeBSD__)                                                   \
1151     || defined(__FreeBSD_kernel__)                                            \
1152     || defined(__NetBSD__)                                                    \
1153     || defined(__OpenBSD__)
1154   struct timeval tv[2];
1155   tv[0] = uv__fs_to_timeval(req->atime);
1156   tv[1] = uv__fs_to_timeval(req->mtime);
1157   return utimes(req->path, tv);
1158 #elif defined(_AIX)                                                           \
1159     && !defined(_AIX71)
1160   struct utimbuf buf;
1161   buf.actime = req->atime;
1162   buf.modtime = req->mtime;
1163   return utime(req->path, &buf);
1164 #elif defined(__MVS__)
1165   attrib_t atr;
1166   memset(&atr, 0, sizeof(atr));
1167   atr.att_mtimechg = 1;
1168   atr.att_atimechg = 1;
1169   atr.att_mtime = req->mtime;
1170   atr.att_atime = req->atime;
1171   return __lchattr((char*) req->path, &atr, sizeof(atr));
1172 #else
1173   errno = ENOSYS;
1174   return -1;
1175 #endif
1176 }
1177 
1178 
uv__fs_lutime(uv_fs_t * req)1179 static ssize_t uv__fs_lutime(uv_fs_t* req) {
1180 #if defined(__linux__)            ||                                           \
1181     defined(_AIX71)               ||                                           \
1182     defined(__sun)                ||                                           \
1183     defined(__HAIKU__)            ||                                           \
1184     defined(__GNU__)              ||                                           \
1185     defined(__OpenBSD__)
1186   struct timespec ts[2];
1187   ts[0] = uv__fs_to_timespec(req->atime);
1188   ts[1] = uv__fs_to_timespec(req->mtime);
1189   return utimensat(AT_FDCWD, req->path, ts, AT_SYMLINK_NOFOLLOW);
1190 #elif defined(__APPLE__)          ||                                          \
1191       defined(__DragonFly__)      ||                                          \
1192       defined(__FreeBSD__)        ||                                          \
1193       defined(__FreeBSD_kernel__) ||                                          \
1194       defined(__NetBSD__)
1195   struct timeval tv[2];
1196   tv[0] = uv__fs_to_timeval(req->atime);
1197   tv[1] = uv__fs_to_timeval(req->mtime);
1198   return lutimes(req->path, tv);
1199 #else
1200   errno = ENOSYS;
1201   return -1;
1202 #endif
1203 }
1204 
1205 
uv__fs_write(uv_fs_t * req)1206 static ssize_t uv__fs_write(uv_fs_t* req) {
1207 #if defined(__linux__)
1208   static int no_pwritev;
1209 #endif
1210   ssize_t r;
1211 
1212   /* Serialize writes on OS X, concurrent write() and pwrite() calls result in
1213    * data loss. We can't use a per-file descriptor lock, the descriptor may be
1214    * a dup().
1215    */
1216 #if defined(__APPLE__)
1217   static pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;
1218 
1219   if (pthread_mutex_lock(&lock))
1220     abort();
1221 #endif
1222 
1223   if (req->off < 0) {
1224     if (req->nbufs == 1)
1225       r = write(req->file, req->bufs[0].base, req->bufs[0].len);
1226     else
1227       r = writev(req->file, (struct iovec*) req->bufs, req->nbufs);
1228   } else {
1229     if (req->nbufs == 1) {
1230       r = pwrite(req->file, req->bufs[0].base, req->bufs[0].len, req->off);
1231       goto done;
1232     }
1233 #if HAVE_PREADV
1234     r = pwritev(req->file, (struct iovec*) req->bufs, req->nbufs, req->off);
1235 #else
1236 # if defined(__linux__)
1237     if (no_pwritev) retry:
1238 # endif
1239     {
1240       r = pwrite(req->file, req->bufs[0].base, req->bufs[0].len, req->off);
1241     }
1242 # if defined(__linux__)
1243     else {
1244       r = uv__pwritev(req->file,
1245                       (struct iovec*) req->bufs,
1246                       req->nbufs,
1247                       req->off);
1248       if (r == -1 && errno == ENOSYS) {
1249         no_pwritev = 1;
1250         goto retry;
1251       }
1252     }
1253 # endif
1254 #endif
1255   }
1256 
1257 done:
1258 #if defined(__APPLE__)
1259   if (pthread_mutex_unlock(&lock))
1260     abort();
1261 #endif
1262 
1263   return r;
1264 }
1265 
uv__fs_copyfile(uv_fs_t * req)1266 static ssize_t uv__fs_copyfile(uv_fs_t* req) {
1267   uv_fs_t fs_req;
1268   uv_file srcfd;
1269   uv_file dstfd;
1270   struct stat src_statsbuf;
1271   struct stat dst_statsbuf;
1272   int dst_flags;
1273   int result;
1274   int err;
1275   off_t bytes_to_send;
1276   off_t in_offset;
1277   off_t bytes_written;
1278   size_t bytes_chunk;
1279 
1280   dstfd = -1;
1281   err = 0;
1282 
1283   /* Open the source file. */
1284   srcfd = uv_fs_open(NULL, &fs_req, req->path, O_RDONLY, 0, NULL);
1285   uv_fs_req_cleanup(&fs_req);
1286 
1287   if (srcfd < 0)
1288     return srcfd;
1289 
1290   /* Get the source file's mode. */
1291   if (fstat(srcfd, &src_statsbuf)) {
1292     err = UV__ERR(errno);
1293     goto out;
1294   }
1295 
1296   dst_flags = O_WRONLY | O_CREAT;
1297 
1298   if (req->flags & UV_FS_COPYFILE_EXCL)
1299     dst_flags |= O_EXCL;
1300 
1301   /* Open the destination file. */
1302   dstfd = uv_fs_open(NULL,
1303                      &fs_req,
1304                      req->new_path,
1305                      dst_flags,
1306                      src_statsbuf.st_mode,
1307                      NULL);
1308   uv_fs_req_cleanup(&fs_req);
1309 
1310   if (dstfd < 0) {
1311     err = dstfd;
1312     goto out;
1313   }
1314 
1315   /* If the file is not being opened exclusively, verify that the source and
1316      destination are not the same file. If they are the same, bail out early. */
1317   if ((req->flags & UV_FS_COPYFILE_EXCL) == 0) {
1318     /* Get the destination file's mode. */
1319     if (fstat(dstfd, &dst_statsbuf)) {
1320       err = UV__ERR(errno);
1321       goto out;
1322     }
1323 
1324     /* Check if srcfd and dstfd refer to the same file */
1325     if (src_statsbuf.st_dev == dst_statsbuf.st_dev &&
1326         src_statsbuf.st_ino == dst_statsbuf.st_ino) {
1327       goto out;
1328     }
1329 
1330     /* Truncate the file in case the destination already existed. */
1331     if (ftruncate(dstfd, 0) != 0) {
1332       err = UV__ERR(errno);
1333       goto out;
1334     }
1335   }
1336 
1337   if (fchmod(dstfd, src_statsbuf.st_mode) == -1) {
1338     err = UV__ERR(errno);
1339 #ifdef __linux__
1340     /* fchmod() on CIFS shares always fails with EPERM unless the share is
1341      * mounted with "noperm". As fchmod() is a meaningless operation on such
1342      * shares anyway, detect that condition and squelch the error.
1343      */
1344     if (err != UV_EPERM)
1345       goto out;
1346 
1347     if (!uv__is_cifs_or_smb(dstfd))
1348       goto out;
1349 
1350     err = 0;
1351 #else  /* !__linux__ */
1352     goto out;
1353 #endif  /* !__linux__ */
1354   }
1355 
1356 #ifdef FICLONE
1357   if (req->flags & UV_FS_COPYFILE_FICLONE ||
1358       req->flags & UV_FS_COPYFILE_FICLONE_FORCE) {
1359     if (ioctl(dstfd, FICLONE, srcfd) == 0) {
1360       /* ioctl() with FICLONE succeeded. */
1361       goto out;
1362     }
1363     /* If an error occurred and force was set, return the error to the caller;
1364      * fall back to sendfile() when force was not set. */
1365     if (req->flags & UV_FS_COPYFILE_FICLONE_FORCE) {
1366       err = UV__ERR(errno);
1367       goto out;
1368     }
1369   }
1370 #else
1371   if (req->flags & UV_FS_COPYFILE_FICLONE_FORCE) {
1372     err = UV_ENOSYS;
1373     goto out;
1374   }
1375 #endif
1376 
1377   bytes_to_send = src_statsbuf.st_size;
1378   in_offset = 0;
1379   while (bytes_to_send != 0) {
1380     bytes_chunk = SSIZE_MAX;
1381     if (bytes_to_send < (off_t) bytes_chunk)
1382       bytes_chunk = bytes_to_send;
1383     uv_fs_sendfile(NULL, &fs_req, dstfd, srcfd, in_offset, bytes_chunk, NULL);
1384     bytes_written = fs_req.result;
1385     uv_fs_req_cleanup(&fs_req);
1386 
1387     if (bytes_written < 0) {
1388       err = bytes_written;
1389       break;
1390     }
1391 
1392     bytes_to_send -= bytes_written;
1393     in_offset += bytes_written;
1394   }
1395 
1396 out:
1397   if (err < 0)
1398     result = err;
1399   else
1400     result = 0;
1401 
1402   /* Close the source file. */
1403   err = uv__close_nocheckstdio(srcfd);
1404 
1405   /* Don't overwrite any existing errors. */
1406   if (err != 0 && result == 0)
1407     result = err;
1408 
1409   /* Close the destination file if it is open. */
1410   if (dstfd >= 0) {
1411     err = uv__close_nocheckstdio(dstfd);
1412 
1413     /* Don't overwrite any existing errors. */
1414     if (err != 0 && result == 0)
1415       result = err;
1416 
1417     /* Remove the destination file if something went wrong. */
1418     if (result != 0) {
1419       uv_fs_unlink(NULL, &fs_req, req->new_path, NULL);
1420       /* Ignore the unlink return value, as an error already happened. */
1421       uv_fs_req_cleanup(&fs_req);
1422     }
1423   }
1424 
1425   if (result == 0)
1426     return 0;
1427 
1428   errno = UV__ERR(result);
1429   return -1;
1430 }
1431 
uv__to_stat(struct stat * src,uv_stat_t * dst)1432 static void uv__to_stat(struct stat* src, uv_stat_t* dst) {
1433   dst->st_dev = src->st_dev;
1434   dst->st_mode = src->st_mode;
1435   dst->st_nlink = src->st_nlink;
1436   dst->st_uid = src->st_uid;
1437   dst->st_gid = src->st_gid;
1438   dst->st_rdev = src->st_rdev;
1439   dst->st_ino = src->st_ino;
1440   dst->st_size = src->st_size;
1441   dst->st_blksize = src->st_blksize;
1442   dst->st_blocks = src->st_blocks;
1443 
1444 #if defined(__APPLE__)
1445   dst->st_atim.tv_sec = src->st_atimespec.tv_sec;
1446   dst->st_atim.tv_nsec = src->st_atimespec.tv_nsec;
1447   dst->st_mtim.tv_sec = src->st_mtimespec.tv_sec;
1448   dst->st_mtim.tv_nsec = src->st_mtimespec.tv_nsec;
1449   dst->st_ctim.tv_sec = src->st_ctimespec.tv_sec;
1450   dst->st_ctim.tv_nsec = src->st_ctimespec.tv_nsec;
1451   dst->st_birthtim.tv_sec = src->st_birthtimespec.tv_sec;
1452   dst->st_birthtim.tv_nsec = src->st_birthtimespec.tv_nsec;
1453   dst->st_flags = src->st_flags;
1454   dst->st_gen = src->st_gen;
1455 #elif defined(__ANDROID__)
1456   dst->st_atim.tv_sec = src->st_atime;
1457   dst->st_atim.tv_nsec = src->st_atimensec;
1458   dst->st_mtim.tv_sec = src->st_mtime;
1459   dst->st_mtim.tv_nsec = src->st_mtimensec;
1460   dst->st_ctim.tv_sec = src->st_ctime;
1461   dst->st_ctim.tv_nsec = src->st_ctimensec;
1462   dst->st_birthtim.tv_sec = src->st_ctime;
1463   dst->st_birthtim.tv_nsec = src->st_ctimensec;
1464   dst->st_flags = 0;
1465   dst->st_gen = 0;
1466 #elif !defined(_AIX) &&         \
1467     !defined(__MVS__) && (      \
1468     defined(__DragonFly__)   || \
1469     defined(__FreeBSD__)     || \
1470     defined(__OpenBSD__)     || \
1471     defined(__NetBSD__)      || \
1472     defined(_GNU_SOURCE)     || \
1473     defined(_BSD_SOURCE)     || \
1474     defined(_SVID_SOURCE)    || \
1475     defined(_XOPEN_SOURCE)   || \
1476     defined(_DEFAULT_SOURCE))
1477   dst->st_atim.tv_sec = src->st_atim.tv_sec;
1478   dst->st_atim.tv_nsec = src->st_atim.tv_nsec;
1479   dst->st_mtim.tv_sec = src->st_mtim.tv_sec;
1480   dst->st_mtim.tv_nsec = src->st_mtim.tv_nsec;
1481   dst->st_ctim.tv_sec = src->st_ctim.tv_sec;
1482   dst->st_ctim.tv_nsec = src->st_ctim.tv_nsec;
1483 # if defined(__FreeBSD__)    || \
1484      defined(__NetBSD__)
1485   dst->st_birthtim.tv_sec = src->st_birthtim.tv_sec;
1486   dst->st_birthtim.tv_nsec = src->st_birthtim.tv_nsec;
1487   dst->st_flags = src->st_flags;
1488   dst->st_gen = src->st_gen;
1489 # else
1490   dst->st_birthtim.tv_sec = src->st_ctim.tv_sec;
1491   dst->st_birthtim.tv_nsec = src->st_ctim.tv_nsec;
1492   dst->st_flags = 0;
1493   dst->st_gen = 0;
1494 # endif
1495 #else
1496   dst->st_atim.tv_sec = src->st_atime;
1497   dst->st_atim.tv_nsec = 0;
1498   dst->st_mtim.tv_sec = src->st_mtime;
1499   dst->st_mtim.tv_nsec = 0;
1500   dst->st_ctim.tv_sec = src->st_ctime;
1501   dst->st_ctim.tv_nsec = 0;
1502   dst->st_birthtim.tv_sec = src->st_ctime;
1503   dst->st_birthtim.tv_nsec = 0;
1504   dst->st_flags = 0;
1505   dst->st_gen = 0;
1506 #endif
1507 }
1508 
1509 
uv__fs_statx(int fd,const char * path,int is_fstat,int is_lstat,uv_stat_t * buf)1510 static int uv__fs_statx(int fd,
1511                         const char* path,
1512                         int is_fstat,
1513                         int is_lstat,
1514                         uv_stat_t* buf) {
1515   STATIC_ASSERT(UV_ENOSYS != -1);
1516 #ifdef __linux__
1517   static int no_statx;
1518   struct uv__statx statxbuf;
1519   int dirfd;
1520   int flags;
1521   int mode;
1522   int rc;
1523 
1524   if (uv__load_relaxed(&no_statx))
1525     return UV_ENOSYS;
1526 
1527   dirfd = AT_FDCWD;
1528   flags = 0; /* AT_STATX_SYNC_AS_STAT */
1529   mode = 0xFFF; /* STATX_BASIC_STATS + STATX_BTIME */
1530 
1531   if (is_fstat) {
1532     dirfd = fd;
1533     flags |= 0x1000; /* AT_EMPTY_PATH */
1534   }
1535 
1536   if (is_lstat)
1537     flags |= AT_SYMLINK_NOFOLLOW;
1538 
1539   rc = uv__statx(dirfd, path, flags, mode, &statxbuf);
1540 
1541   switch (rc) {
1542   case 0:
1543     break;
1544   case -1:
1545     /* EPERM happens when a seccomp filter rejects the system call.
1546      * Has been observed with libseccomp < 2.3.3 and docker < 18.04.
1547      * EOPNOTSUPP is used on DVS exported filesystems
1548      */
1549     if (errno != EINVAL && errno != EPERM && errno != ENOSYS && errno != EOPNOTSUPP)
1550       return -1;
1551     /* Fall through. */
1552   default:
1553     /* Normally on success, zero is returned and On error, -1 is returned.
1554      * Observed on S390 RHEL running in a docker container with statx not
1555      * implemented, rc might return 1 with 0 set as the error code in which
1556      * case we return ENOSYS.
1557      */
1558     uv__store_relaxed(&no_statx, 1);
1559     return UV_ENOSYS;
1560   }
1561 
1562   buf->st_dev = makedev(statxbuf.stx_dev_major, statxbuf.stx_dev_minor);
1563   buf->st_mode = statxbuf.stx_mode;
1564   buf->st_nlink = statxbuf.stx_nlink;
1565   buf->st_uid = statxbuf.stx_uid;
1566   buf->st_gid = statxbuf.stx_gid;
1567   buf->st_rdev = makedev(statxbuf.stx_rdev_major, statxbuf.stx_rdev_minor);
1568   buf->st_ino = statxbuf.stx_ino;
1569   buf->st_size = statxbuf.stx_size;
1570   buf->st_blksize = statxbuf.stx_blksize;
1571   buf->st_blocks = statxbuf.stx_blocks;
1572   buf->st_atim.tv_sec = statxbuf.stx_atime.tv_sec;
1573   buf->st_atim.tv_nsec = statxbuf.stx_atime.tv_nsec;
1574   buf->st_mtim.tv_sec = statxbuf.stx_mtime.tv_sec;
1575   buf->st_mtim.tv_nsec = statxbuf.stx_mtime.tv_nsec;
1576   buf->st_ctim.tv_sec = statxbuf.stx_ctime.tv_sec;
1577   buf->st_ctim.tv_nsec = statxbuf.stx_ctime.tv_nsec;
1578   buf->st_birthtim.tv_sec = statxbuf.stx_btime.tv_sec;
1579   buf->st_birthtim.tv_nsec = statxbuf.stx_btime.tv_nsec;
1580   buf->st_flags = 0;
1581   buf->st_gen = 0;
1582 
1583   return 0;
1584 #else
1585   return UV_ENOSYS;
1586 #endif /* __linux__ */
1587 }
1588 
1589 
uv__fs_stat(const char * path,uv_stat_t * buf)1590 static int uv__fs_stat(const char *path, uv_stat_t *buf) {
1591   struct stat pbuf;
1592   int ret;
1593 
1594   ret = uv__fs_statx(-1, path, /* is_fstat */ 0, /* is_lstat */ 0, buf);
1595   if (ret != UV_ENOSYS)
1596     return ret;
1597 
1598   ret = stat(path, &pbuf);
1599   if (ret == 0)
1600     uv__to_stat(&pbuf, buf);
1601 
1602   return ret;
1603 }
1604 
1605 
uv__fs_lstat(const char * path,uv_stat_t * buf)1606 static int uv__fs_lstat(const char *path, uv_stat_t *buf) {
1607   struct stat pbuf;
1608   int ret;
1609 
1610   ret = uv__fs_statx(-1, path, /* is_fstat */ 0, /* is_lstat */ 1, buf);
1611   if (ret != UV_ENOSYS)
1612     return ret;
1613 
1614   ret = lstat(path, &pbuf);
1615   if (ret == 0)
1616     uv__to_stat(&pbuf, buf);
1617 
1618   return ret;
1619 }
1620 
1621 
uv__fs_fstat(int fd,uv_stat_t * buf)1622 static int uv__fs_fstat(int fd, uv_stat_t *buf) {
1623   struct stat pbuf;
1624   int ret;
1625 
1626   ret = uv__fs_statx(fd, "", /* is_fstat */ 1, /* is_lstat */ 0, buf);
1627   if (ret != UV_ENOSYS)
1628     return ret;
1629 
1630   ret = fstat(fd, &pbuf);
1631   if (ret == 0)
1632     uv__to_stat(&pbuf, buf);
1633 
1634   return ret;
1635 }
1636 
uv__fs_buf_offset(uv_buf_t * bufs,size_t size)1637 static size_t uv__fs_buf_offset(uv_buf_t* bufs, size_t size) {
1638   size_t offset;
1639   /* Figure out which bufs are done */
1640   for (offset = 0; size > 0 && bufs[offset].len <= size; ++offset)
1641     size -= bufs[offset].len;
1642 
1643   /* Fix a partial read/write */
1644   if (size > 0) {
1645     bufs[offset].base += size;
1646     bufs[offset].len -= size;
1647   }
1648   return offset;
1649 }
1650 
uv__fs_write_all(uv_fs_t * req)1651 static ssize_t uv__fs_write_all(uv_fs_t* req) {
1652   unsigned int iovmax;
1653   unsigned int nbufs;
1654   uv_buf_t* bufs;
1655   ssize_t total;
1656   ssize_t result;
1657 
1658   iovmax = uv__getiovmax();
1659   nbufs = req->nbufs;
1660   bufs = req->bufs;
1661   total = 0;
1662 
1663   while (nbufs > 0) {
1664     req->nbufs = nbufs;
1665     if (req->nbufs > iovmax)
1666       req->nbufs = iovmax;
1667 
1668     do
1669       result = uv__fs_write(req);
1670     while (result < 0 && errno == EINTR);
1671 
1672     if (result <= 0) {
1673       if (total == 0)
1674         total = result;
1675       break;
1676     }
1677 
1678     if (req->off >= 0)
1679       req->off += result;
1680 
1681     req->nbufs = uv__fs_buf_offset(req->bufs, result);
1682     req->bufs += req->nbufs;
1683     nbufs -= req->nbufs;
1684     total += result;
1685   }
1686 
1687   if (bufs != req->bufsml)
1688     uv__free(bufs);
1689 
1690   req->bufs = NULL;
1691   req->nbufs = 0;
1692 
1693   return total;
1694 }
1695 
1696 
uv__fs_work(struct uv__work * w)1697 static void uv__fs_work(struct uv__work* w) {
1698   int retry_on_eintr;
1699   uv_fs_t* req;
1700   ssize_t r;
1701 
1702   req = container_of(w, uv_fs_t, work_req);
1703   retry_on_eintr = !(req->fs_type == UV_FS_CLOSE ||
1704                      req->fs_type == UV_FS_READ);
1705 
1706   do {
1707     errno = 0;
1708 
1709 #define X(type, action)                                                       \
1710   case UV_FS_ ## type:                                                        \
1711     r = action;                                                               \
1712     break;
1713 
1714     switch (req->fs_type) {
1715     X(ACCESS, access(req->path, req->flags));
1716     X(CHMOD, chmod(req->path, req->mode));
1717     X(CHOWN, chown(req->path, req->uid, req->gid));
1718     X(CLOSE, uv__fs_close(req->file));
1719     X(COPYFILE, uv__fs_copyfile(req));
1720     X(FCHMOD, fchmod(req->file, req->mode));
1721     X(FCHOWN, fchown(req->file, req->uid, req->gid));
1722     X(LCHOWN, lchown(req->path, req->uid, req->gid));
1723     X(FDATASYNC, uv__fs_fdatasync(req));
1724     X(FSTAT, uv__fs_fstat(req->file, &req->statbuf));
1725     X(FSYNC, uv__fs_fsync(req));
1726     X(FTRUNCATE, ftruncate(req->file, req->off));
1727     X(FUTIME, uv__fs_futime(req));
1728     X(LUTIME, uv__fs_lutime(req));
1729     X(LSTAT, uv__fs_lstat(req->path, &req->statbuf));
1730     X(LINK, link(req->path, req->new_path));
1731     X(MKDIR, mkdir(req->path, req->mode));
1732     X(MKDTEMP, uv__fs_mkdtemp(req));
1733     X(MKSTEMP, uv__fs_mkstemp(req));
1734     X(OPEN, uv__fs_open(req));
1735     X(READ, uv__fs_read(req));
1736     X(SCANDIR, uv__fs_scandir(req));
1737     X(OPENDIR, uv__fs_opendir(req));
1738     X(READDIR, uv__fs_readdir(req));
1739     X(CLOSEDIR, uv__fs_closedir(req));
1740     X(READLINK, uv__fs_readlink(req));
1741     X(REALPATH, uv__fs_realpath(req));
1742     X(RENAME, rename(req->path, req->new_path));
1743     X(RMDIR, rmdir(req->path));
1744     X(SENDFILE, uv__fs_sendfile(req));
1745     X(STAT, uv__fs_stat(req->path, &req->statbuf));
1746     X(STATFS, uv__fs_statfs(req));
1747     X(SYMLINK, symlink(req->path, req->new_path));
1748     X(UNLINK, unlink(req->path));
1749     X(UTIME, uv__fs_utime(req));
1750     X(WRITE, uv__fs_write_all(req));
1751     default: abort();
1752     }
1753 #undef X
1754   } while (r == -1 && errno == EINTR && retry_on_eintr);
1755 
1756   if (r == -1)
1757     req->result = UV__ERR(errno);
1758   else
1759     req->result = r;
1760 
1761   if (r == 0 && (req->fs_type == UV_FS_STAT ||
1762                  req->fs_type == UV_FS_FSTAT ||
1763                  req->fs_type == UV_FS_LSTAT)) {
1764     req->ptr = &req->statbuf;
1765   }
1766 }
1767 
1768 
uv__fs_done(struct uv__work * w,int status)1769 static void uv__fs_done(struct uv__work* w, int status) {
1770   uv_fs_t* req;
1771 
1772   req = container_of(w, uv_fs_t, work_req);
1773   uv__req_unregister(req->loop, req);
1774 
1775   if (status == UV_ECANCELED) {
1776     assert(req->result == 0);
1777     req->result = UV_ECANCELED;
1778   }
1779 
1780   req->cb(req);
1781 }
1782 
1783 
uv_fs_access(uv_loop_t * loop,uv_fs_t * req,const char * path,int flags,uv_fs_cb cb)1784 int uv_fs_access(uv_loop_t* loop,
1785                  uv_fs_t* req,
1786                  const char* path,
1787                  int flags,
1788                  uv_fs_cb cb) {
1789   INIT(ACCESS);
1790   PATH;
1791   req->flags = flags;
1792   POST;
1793 }
1794 
1795 
uv_fs_chmod(uv_loop_t * loop,uv_fs_t * req,const char * path,int mode,uv_fs_cb cb)1796 int uv_fs_chmod(uv_loop_t* loop,
1797                 uv_fs_t* req,
1798                 const char* path,
1799                 int mode,
1800                 uv_fs_cb cb) {
1801   INIT(CHMOD);
1802   PATH;
1803   req->mode = mode;
1804   POST;
1805 }
1806 
1807 
uv_fs_chown(uv_loop_t * loop,uv_fs_t * req,const char * path,uv_uid_t uid,uv_gid_t gid,uv_fs_cb cb)1808 int uv_fs_chown(uv_loop_t* loop,
1809                 uv_fs_t* req,
1810                 const char* path,
1811                 uv_uid_t uid,
1812                 uv_gid_t gid,
1813                 uv_fs_cb cb) {
1814   INIT(CHOWN);
1815   PATH;
1816   req->uid = uid;
1817   req->gid = gid;
1818   POST;
1819 }
1820 
1821 
uv_fs_close(uv_loop_t * loop,uv_fs_t * req,uv_file file,uv_fs_cb cb)1822 int uv_fs_close(uv_loop_t* loop, uv_fs_t* req, uv_file file, uv_fs_cb cb) {
1823   INIT(CLOSE);
1824   req->file = file;
1825   POST;
1826 }
1827 
1828 
uv_fs_fchmod(uv_loop_t * loop,uv_fs_t * req,uv_file file,int mode,uv_fs_cb cb)1829 int uv_fs_fchmod(uv_loop_t* loop,
1830                  uv_fs_t* req,
1831                  uv_file file,
1832                  int mode,
1833                  uv_fs_cb cb) {
1834   INIT(FCHMOD);
1835   req->file = file;
1836   req->mode = mode;
1837   POST;
1838 }
1839 
1840 
uv_fs_fchown(uv_loop_t * loop,uv_fs_t * req,uv_file file,uv_uid_t uid,uv_gid_t gid,uv_fs_cb cb)1841 int uv_fs_fchown(uv_loop_t* loop,
1842                  uv_fs_t* req,
1843                  uv_file file,
1844                  uv_uid_t uid,
1845                  uv_gid_t gid,
1846                  uv_fs_cb cb) {
1847   INIT(FCHOWN);
1848   req->file = file;
1849   req->uid = uid;
1850   req->gid = gid;
1851   POST;
1852 }
1853 
1854 
uv_fs_lchown(uv_loop_t * loop,uv_fs_t * req,const char * path,uv_uid_t uid,uv_gid_t gid,uv_fs_cb cb)1855 int uv_fs_lchown(uv_loop_t* loop,
1856                  uv_fs_t* req,
1857                  const char* path,
1858                  uv_uid_t uid,
1859                  uv_gid_t gid,
1860                  uv_fs_cb cb) {
1861   INIT(LCHOWN);
1862   PATH;
1863   req->uid = uid;
1864   req->gid = gid;
1865   POST;
1866 }
1867 
1868 
uv_fs_fdatasync(uv_loop_t * loop,uv_fs_t * req,uv_file file,uv_fs_cb cb)1869 int uv_fs_fdatasync(uv_loop_t* loop, uv_fs_t* req, uv_file file, uv_fs_cb cb) {
1870   INIT(FDATASYNC);
1871   req->file = file;
1872   POST;
1873 }
1874 
1875 
uv_fs_fstat(uv_loop_t * loop,uv_fs_t * req,uv_file file,uv_fs_cb cb)1876 int uv_fs_fstat(uv_loop_t* loop, uv_fs_t* req, uv_file file, uv_fs_cb cb) {
1877   INIT(FSTAT);
1878   req->file = file;
1879   POST;
1880 }
1881 
1882 
uv_fs_fsync(uv_loop_t * loop,uv_fs_t * req,uv_file file,uv_fs_cb cb)1883 int uv_fs_fsync(uv_loop_t* loop, uv_fs_t* req, uv_file file, uv_fs_cb cb) {
1884   INIT(FSYNC);
1885   req->file = file;
1886   POST;
1887 }
1888 
1889 
uv_fs_ftruncate(uv_loop_t * loop,uv_fs_t * req,uv_file file,int64_t off,uv_fs_cb cb)1890 int uv_fs_ftruncate(uv_loop_t* loop,
1891                     uv_fs_t* req,
1892                     uv_file file,
1893                     int64_t off,
1894                     uv_fs_cb cb) {
1895   INIT(FTRUNCATE);
1896   req->file = file;
1897   req->off = off;
1898   POST;
1899 }
1900 
1901 
uv_fs_futime(uv_loop_t * loop,uv_fs_t * req,uv_file file,double atime,double mtime,uv_fs_cb cb)1902 int uv_fs_futime(uv_loop_t* loop,
1903                  uv_fs_t* req,
1904                  uv_file file,
1905                  double atime,
1906                  double mtime,
1907                  uv_fs_cb cb) {
1908   INIT(FUTIME);
1909   req->file = file;
1910   req->atime = atime;
1911   req->mtime = mtime;
1912   POST;
1913 }
1914 
uv_fs_lutime(uv_loop_t * loop,uv_fs_t * req,const char * path,double atime,double mtime,uv_fs_cb cb)1915 int uv_fs_lutime(uv_loop_t* loop,
1916                  uv_fs_t* req,
1917                  const char* path,
1918                  double atime,
1919                  double mtime,
1920                  uv_fs_cb cb) {
1921   INIT(LUTIME);
1922   PATH;
1923   req->atime = atime;
1924   req->mtime = mtime;
1925   POST;
1926 }
1927 
1928 
uv_fs_lstat(uv_loop_t * loop,uv_fs_t * req,const char * path,uv_fs_cb cb)1929 int uv_fs_lstat(uv_loop_t* loop, uv_fs_t* req, const char* path, uv_fs_cb cb) {
1930   INIT(LSTAT);
1931   PATH;
1932   POST;
1933 }
1934 
1935 
uv_fs_link(uv_loop_t * loop,uv_fs_t * req,const char * path,const char * new_path,uv_fs_cb cb)1936 int uv_fs_link(uv_loop_t* loop,
1937                uv_fs_t* req,
1938                const char* path,
1939                const char* new_path,
1940                uv_fs_cb cb) {
1941   INIT(LINK);
1942   PATH2;
1943   POST;
1944 }
1945 
1946 
uv_fs_mkdir(uv_loop_t * loop,uv_fs_t * req,const char * path,int mode,uv_fs_cb cb)1947 int uv_fs_mkdir(uv_loop_t* loop,
1948                 uv_fs_t* req,
1949                 const char* path,
1950                 int mode,
1951                 uv_fs_cb cb) {
1952   INIT(MKDIR);
1953   PATH;
1954   req->mode = mode;
1955   POST;
1956 }
1957 
1958 
uv_fs_mkdtemp(uv_loop_t * loop,uv_fs_t * req,const char * tpl,uv_fs_cb cb)1959 int uv_fs_mkdtemp(uv_loop_t* loop,
1960                   uv_fs_t* req,
1961                   const char* tpl,
1962                   uv_fs_cb cb) {
1963   INIT(MKDTEMP);
1964   req->path = uv__strdup(tpl);
1965   if (req->path == NULL)
1966     return UV_ENOMEM;
1967   POST;
1968 }
1969 
1970 
uv_fs_mkstemp(uv_loop_t * loop,uv_fs_t * req,const char * tpl,uv_fs_cb cb)1971 int uv_fs_mkstemp(uv_loop_t* loop,
1972                   uv_fs_t* req,
1973                   const char* tpl,
1974                   uv_fs_cb cb) {
1975   INIT(MKSTEMP);
1976   req->path = uv__strdup(tpl);
1977   if (req->path == NULL)
1978     return UV_ENOMEM;
1979   POST;
1980 }
1981 
1982 
uv_fs_open(uv_loop_t * loop,uv_fs_t * req,const char * path,int flags,int mode,uv_fs_cb cb)1983 int uv_fs_open(uv_loop_t* loop,
1984                uv_fs_t* req,
1985                const char* path,
1986                int flags,
1987                int mode,
1988                uv_fs_cb cb) {
1989   INIT(OPEN);
1990   PATH;
1991   req->flags = flags;
1992   req->mode = mode;
1993   POST;
1994 }
1995 
1996 
uv_fs_read(uv_loop_t * loop,uv_fs_t * req,uv_file file,const uv_buf_t bufs[],unsigned int nbufs,int64_t off,uv_fs_cb cb)1997 int uv_fs_read(uv_loop_t* loop, uv_fs_t* req,
1998                uv_file file,
1999                const uv_buf_t bufs[],
2000                unsigned int nbufs,
2001                int64_t off,
2002                uv_fs_cb cb) {
2003   INIT(READ);
2004 
2005   if (bufs == NULL || nbufs == 0)
2006     return UV_EINVAL;
2007 
2008   req->file = file;
2009 
2010   req->nbufs = nbufs;
2011   req->bufs = req->bufsml;
2012   if (nbufs > ARRAY_SIZE(req->bufsml))
2013     req->bufs = uv__malloc(nbufs * sizeof(*bufs));
2014 
2015   if (req->bufs == NULL)
2016     return UV_ENOMEM;
2017 
2018   memcpy(req->bufs, bufs, nbufs * sizeof(*bufs));
2019 
2020   req->off = off;
2021   POST;
2022 }
2023 
2024 
uv_fs_scandir(uv_loop_t * loop,uv_fs_t * req,const char * path,int flags,uv_fs_cb cb)2025 int uv_fs_scandir(uv_loop_t* loop,
2026                   uv_fs_t* req,
2027                   const char* path,
2028                   int flags,
2029                   uv_fs_cb cb) {
2030   INIT(SCANDIR);
2031   PATH;
2032   req->flags = flags;
2033   POST;
2034 }
2035 
uv_fs_opendir(uv_loop_t * loop,uv_fs_t * req,const char * path,uv_fs_cb cb)2036 int uv_fs_opendir(uv_loop_t* loop,
2037                   uv_fs_t* req,
2038                   const char* path,
2039                   uv_fs_cb cb) {
2040   INIT(OPENDIR);
2041   PATH;
2042   POST;
2043 }
2044 
uv_fs_readdir(uv_loop_t * loop,uv_fs_t * req,uv_dir_t * dir,uv_fs_cb cb)2045 int uv_fs_readdir(uv_loop_t* loop,
2046                   uv_fs_t* req,
2047                   uv_dir_t* dir,
2048                   uv_fs_cb cb) {
2049   INIT(READDIR);
2050 
2051   if (dir == NULL || dir->dir == NULL || dir->dirents == NULL)
2052     return UV_EINVAL;
2053 
2054   req->ptr = dir;
2055   POST;
2056 }
2057 
uv_fs_closedir(uv_loop_t * loop,uv_fs_t * req,uv_dir_t * dir,uv_fs_cb cb)2058 int uv_fs_closedir(uv_loop_t* loop,
2059                    uv_fs_t* req,
2060                    uv_dir_t* dir,
2061                    uv_fs_cb cb) {
2062   INIT(CLOSEDIR);
2063 
2064   if (dir == NULL)
2065     return UV_EINVAL;
2066 
2067   req->ptr = dir;
2068   POST;
2069 }
2070 
uv_fs_readlink(uv_loop_t * loop,uv_fs_t * req,const char * path,uv_fs_cb cb)2071 int uv_fs_readlink(uv_loop_t* loop,
2072                    uv_fs_t* req,
2073                    const char* path,
2074                    uv_fs_cb cb) {
2075   INIT(READLINK);
2076   PATH;
2077   POST;
2078 }
2079 
2080 
uv_fs_realpath(uv_loop_t * loop,uv_fs_t * req,const char * path,uv_fs_cb cb)2081 int uv_fs_realpath(uv_loop_t* loop,
2082                   uv_fs_t* req,
2083                   const char * path,
2084                   uv_fs_cb cb) {
2085   INIT(REALPATH);
2086   PATH;
2087   POST;
2088 }
2089 
2090 
uv_fs_rename(uv_loop_t * loop,uv_fs_t * req,const char * path,const char * new_path,uv_fs_cb cb)2091 int uv_fs_rename(uv_loop_t* loop,
2092                  uv_fs_t* req,
2093                  const char* path,
2094                  const char* new_path,
2095                  uv_fs_cb cb) {
2096   INIT(RENAME);
2097   PATH2;
2098   POST;
2099 }
2100 
2101 
uv_fs_rmdir(uv_loop_t * loop,uv_fs_t * req,const char * path,uv_fs_cb cb)2102 int uv_fs_rmdir(uv_loop_t* loop, uv_fs_t* req, const char* path, uv_fs_cb cb) {
2103   INIT(RMDIR);
2104   PATH;
2105   POST;
2106 }
2107 
2108 
uv_fs_sendfile(uv_loop_t * loop,uv_fs_t * req,uv_file out_fd,uv_file in_fd,int64_t off,size_t len,uv_fs_cb cb)2109 int uv_fs_sendfile(uv_loop_t* loop,
2110                    uv_fs_t* req,
2111                    uv_file out_fd,
2112                    uv_file in_fd,
2113                    int64_t off,
2114                    size_t len,
2115                    uv_fs_cb cb) {
2116   INIT(SENDFILE);
2117   req->flags = in_fd; /* hack */
2118   req->file = out_fd;
2119   req->off = off;
2120   req->bufsml[0].len = len;
2121   POST;
2122 }
2123 
2124 
uv_fs_stat(uv_loop_t * loop,uv_fs_t * req,const char * path,uv_fs_cb cb)2125 int uv_fs_stat(uv_loop_t* loop, uv_fs_t* req, const char* path, uv_fs_cb cb) {
2126   INIT(STAT);
2127   PATH;
2128   POST;
2129 }
2130 
2131 
uv_fs_symlink(uv_loop_t * loop,uv_fs_t * req,const char * path,const char * new_path,int flags,uv_fs_cb cb)2132 int uv_fs_symlink(uv_loop_t* loop,
2133                   uv_fs_t* req,
2134                   const char* path,
2135                   const char* new_path,
2136                   int flags,
2137                   uv_fs_cb cb) {
2138   INIT(SYMLINK);
2139   PATH2;
2140   req->flags = flags;
2141   POST;
2142 }
2143 
2144 
uv_fs_unlink(uv_loop_t * loop,uv_fs_t * req,const char * path,uv_fs_cb cb)2145 int uv_fs_unlink(uv_loop_t* loop, uv_fs_t* req, const char* path, uv_fs_cb cb) {
2146   INIT(UNLINK);
2147   PATH;
2148   POST;
2149 }
2150 
2151 
uv_fs_utime(uv_loop_t * loop,uv_fs_t * req,const char * path,double atime,double mtime,uv_fs_cb cb)2152 int uv_fs_utime(uv_loop_t* loop,
2153                 uv_fs_t* req,
2154                 const char* path,
2155                 double atime,
2156                 double mtime,
2157                 uv_fs_cb cb) {
2158   INIT(UTIME);
2159   PATH;
2160   req->atime = atime;
2161   req->mtime = mtime;
2162   POST;
2163 }
2164 
2165 
uv_fs_write(uv_loop_t * loop,uv_fs_t * req,uv_file file,const uv_buf_t bufs[],unsigned int nbufs,int64_t off,uv_fs_cb cb)2166 int uv_fs_write(uv_loop_t* loop,
2167                 uv_fs_t* req,
2168                 uv_file file,
2169                 const uv_buf_t bufs[],
2170                 unsigned int nbufs,
2171                 int64_t off,
2172                 uv_fs_cb cb) {
2173   INIT(WRITE);
2174 
2175   if (bufs == NULL || nbufs == 0)
2176     return UV_EINVAL;
2177 
2178   req->file = file;
2179 
2180   req->nbufs = nbufs;
2181   req->bufs = req->bufsml;
2182   if (nbufs > ARRAY_SIZE(req->bufsml))
2183     req->bufs = uv__malloc(nbufs * sizeof(*bufs));
2184 
2185   if (req->bufs == NULL)
2186     return UV_ENOMEM;
2187 
2188   memcpy(req->bufs, bufs, nbufs * sizeof(*bufs));
2189 
2190   req->off = off;
2191   POST;
2192 }
2193 
2194 
uv_fs_req_cleanup(uv_fs_t * req)2195 void uv_fs_req_cleanup(uv_fs_t* req) {
2196   if (req == NULL)
2197     return;
2198 
2199   /* Only necessary for asychronous requests, i.e., requests with a callback.
2200    * Synchronous ones don't copy their arguments and have req->path and
2201    * req->new_path pointing to user-owned memory.  UV_FS_MKDTEMP and
2202    * UV_FS_MKSTEMP are the exception to the rule, they always allocate memory.
2203    */
2204   if (req->path != NULL &&
2205       (req->cb != NULL ||
2206         req->fs_type == UV_FS_MKDTEMP || req->fs_type == UV_FS_MKSTEMP))
2207     uv__free((void*) req->path);  /* Memory is shared with req->new_path. */
2208 
2209   req->path = NULL;
2210   req->new_path = NULL;
2211 
2212   if (req->fs_type == UV_FS_READDIR && req->ptr != NULL)
2213     uv__fs_readdir_cleanup(req);
2214 
2215   if (req->fs_type == UV_FS_SCANDIR && req->ptr != NULL)
2216     uv__fs_scandir_cleanup(req);
2217 
2218   if (req->bufs != req->bufsml)
2219     uv__free(req->bufs);
2220   req->bufs = NULL;
2221 
2222   if (req->fs_type != UV_FS_OPENDIR && req->ptr != &req->statbuf)
2223     uv__free(req->ptr);
2224   req->ptr = NULL;
2225 }
2226 
2227 
uv_fs_copyfile(uv_loop_t * loop,uv_fs_t * req,const char * path,const char * new_path,int flags,uv_fs_cb cb)2228 int uv_fs_copyfile(uv_loop_t* loop,
2229                    uv_fs_t* req,
2230                    const char* path,
2231                    const char* new_path,
2232                    int flags,
2233                    uv_fs_cb cb) {
2234   INIT(COPYFILE);
2235 
2236   if (flags & ~(UV_FS_COPYFILE_EXCL |
2237                 UV_FS_COPYFILE_FICLONE |
2238                 UV_FS_COPYFILE_FICLONE_FORCE)) {
2239     return UV_EINVAL;
2240   }
2241 
2242   PATH2;
2243   req->flags = flags;
2244   POST;
2245 }
2246 
2247 
uv_fs_statfs(uv_loop_t * loop,uv_fs_t * req,const char * path,uv_fs_cb cb)2248 int uv_fs_statfs(uv_loop_t* loop,
2249                  uv_fs_t* req,
2250                  const char* path,
2251                  uv_fs_cb cb) {
2252   INIT(STATFS);
2253   PATH;
2254   POST;
2255 }
2256 
uv_fs_get_system_error(const uv_fs_t * req)2257 int uv_fs_get_system_error(const uv_fs_t* req) {
2258   return -req->result;
2259 }
2260