/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│ │ vi: set et ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi │ ╞══════════════════════════════════════════════════════════════════════════════╡ │ Copyright 2022 Justine Alexandra Roberts Tunney │ │ │ │ Permission to use, copy, modify, and/or distribute this software for │ │ any purpose with or without fee is hereby granted, provided that the │ │ above copyright notice and this permission notice appear in all copies. │ │ │ │ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │ │ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │ │ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │ │ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │ │ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │ │ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │ │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │ │ PERFORMANCE OF THIS SOFTWARE. │ ╚─────────────────────────────────────────────────────────────────────────────*/ #include "libc/calls/calls.h" #include "libc/calls/struct/stat.h" #include "libc/errno.h" #include "libc/fmt/conv.h" #include "libc/fmt/libgen.h" #include "libc/fmt/magnumstrs.internal.h" #include "libc/limits.h" #include "libc/mem/gc.h" #include "libc/runtime/runtime.h" #include "libc/stdio/ftw.h" #include "libc/stdio/stdio.h" #include "libc/str/str.h" #include "libc/sysv/consts/at.h" #include "libc/sysv/consts/ex.h" #include "libc/sysv/consts/exit.h" #include "libc/sysv/consts/o.h" #include "libc/sysv/consts/ok.h" #include "libc/sysv/consts/s.h" #include "libc/x/x.h" #include "third_party/getopt/getopt.internal.h" #define USAGE \ " SRC... DST\n\ \n\ SYNOPSIS\n\ \n\ Copies Files\n\ \n\ FLAGS\n\ \n\ -?\n\ -h help\n\ -f force\n\ -r recursive\n\ -n no clobber\n\ -a preserve all\n\ -p preserve owner and timestamps\n\ \n" int flags; bool force; int striplen; bool recursive; const char *prog; char mkbuf[PATH_MAX]; char srcdir[PATH_MAX]; char dstdir[PATH_MAX]; char srcfile[PATH_MAX]; char dstfile[PATH_MAX]; char linkbuf[PATH_MAX]; void Cp(char *, char *); bool IsDirectory(const char *path) { int e; bool res; struct stat st; e = errno; res = stat(path, &st) != -1 && S_ISDIR(st.st_mode); errno = e; return res; } bool IsSymlink(const char *path) { int e; bool res; struct stat st; e = errno; res = fstatat(AT_FDCWD, path, &st, AT_SYMLINK_NOFOLLOW) != -1 && S_ISLNK(st.st_mode); errno = e; return res; } wontreturn void PrintUsage(int rc, int fd) { tinyprint(fd, "USAGE\n\n ", prog, USAGE, NULL); exit(rc); } void GetOpts(int argc, char *argv[]) { int opt; while ((opt = getopt(argc, argv, "hfnaprR")) != -1) { switch (opt) { case 'f': force = true; break; case 'r': case 'R': recursive = true; break; case 'n': // TODO(jart): restore functionality // flags |= COPYFILE_NOCLOBBER; break; case 'a': case 'p': // TODO(jart): restore functionality // flags |= COPYFILE_PRESERVE_OWNER; // flags |= COPYFILE_PRESERVE_TIMESTAMPS; break; case 'h': PrintUsage(0, 1); default: PrintUsage(1, 2); } } } int Visit(const char *fpath, const struct stat *sb, int tflag, struct FTW *ftwbuf) { char *src; strcpy(srcfile, fpath); src = srcfile + striplen; strcpy(dstfile, dstdir); if (!endswith(dstfile, "/")) { strcat(dstfile, "/"); } strcat(dstfile, src); strcpy(srcfile, fpath); switch (tflag) { case FTW_D: return 0; case FTW_F: case FTW_SL: case FTW_SLN: Cp(srcfile, dstfile); return 0; default: tinyprint(2, fpath, ": bad file type\n", NULL); exit(1); } } char *Join(const char *a, const char *b) { size_t n, m; n = strlen(a); m = strlen(b); if (n + 1 + m + 1 > sizeof(dstfile)) { tinyprint(2, prog, ": path too long\n", NULL); exit(1); } stpcpy(stpcpy(stpcpy(dstfile, a), "/"), b); return dstfile; } bool MovePreservingDestinationInode(const char *from, const char *to) { bool res; struct stat st; int fdin, fdout; if ((fdin = open(from, O_RDONLY)) == -1) { return false; } fstat(fdin, &st); if ((fdout = creat(to, st.st_mode)) == -1) { close(fdin); return false; } res = copyfd(fdin, fdout, -1) != -1; close(fdin); close(fdout); return res; } void Cp(char *src, char *dst) { ssize_t rc; const char *s; if (strlen(src) + 1 > PATH_MAX) _Exit(2); if (strlen(dst) + 1 > PATH_MAX) _Exit(2); basename(src); basename(dst); if (IsDirectory(src)) { if (!recursive) { tinyprint(2, prog, ": won't copy directory without -r flag\n", NULL); exit(1); } strcpy(dstdir, dst); if (IsDirectory(dst)) { strcpy(srcdir, src); basename(srcdir); striplen = 0; strcpy(srcdir, basename(src)); } else { strcpy(srcdir, src); basename(srcdir); striplen = strlen(srcdir); strcpy(srcdir, ""); } if (nftw(src, Visit, 20, 0) == -1) { perror(src); exit(1); } return; } if (IsDirectory(dst)) { dst = Join(dst, basename(src)); } if (!force && access(dst, W_OK) == -1 && errno != ENOENT) { perror(dst); exit(1); } strcpy(mkbuf, dst); if (makedirs((s = dirname(mkbuf)), 0755) == -1) { perror(s); exit(1); } if (IsSymlink(src)) { if ((rc = readlink(src, linkbuf, sizeof(linkbuf) - 1)) == -1) { perror(src); exit(1); } linkbuf[rc] = 0; if (symlink(linkbuf, dst) == -1) { perror(dst); exit(1); } } else { if (!MovePreservingDestinationInode(src, dst)) { perror(src); exit(1); } } } int main(int argc, char *argv[]) { int i; prog = argv[0]; if (!prog) prog = "cp"; GetOpts(argc, argv); if (argc - optind < 2) { tinyprint(2, prog, ": missing operand\n", NULL); exit(1); } for (i = optind; i < argc - 1; ++i) { Cp(argv[i], argv[argc - 1]); } return 0; }