summaryrefslogtreecommitdiff
path: root/Userland/Utilities/cp.cpp
diff options
context:
space:
mode:
authorBui Quang Minh <minhquangbui99@gmail.com>2021-02-18 21:34:43 +0700
committerAndreas Kling <kling@serenityos.org>2021-02-20 09:30:11 +0100
commitff67340d819f3ccafb079cd853385d3d6d82fc74 (patch)
treee53bff7fd60e54fa6a67b1bd08f6eaa0d8af869a /Userland/Utilities/cp.cpp
parent14058b6858ff2b0b3b03c72a65a8f31b9b59024f (diff)
downloadserenity-ff67340d819f3ccafb079cd853385d3d6d82fc74.zip
Userland: Support moving files between different mounted filesystems
In case we cannot use rename() because of cross-device error, copy file to the destination then unlink the old source file.
Diffstat (limited to 'Userland/Utilities/cp.cpp')
-rw-r--r--Userland/Utilities/cp.cpp158
1 files changed, 1 insertions, 157 deletions
diff --git a/Userland/Utilities/cp.cpp b/Userland/Utilities/cp.cpp
index ee640f6f3d..b52c06eeeb 100644
--- a/Userland/Utilities/cp.cpp
+++ b/Userland/Utilities/cp.cpp
@@ -24,24 +24,11 @@
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
-#include <AK/LexicalPath.h>
-#include <AK/String.h>
-#include <AK/StringBuilder.h>
+#include "cp.h"
#include <LibCore/ArgsParser.h>
-#include <LibCore/DirIterator.h>
-#include <LibCore/File.h>
-#include <assert.h>
-#include <fcntl.h>
#include <stdio.h>
-#include <sys/stat.h>
#include <unistd.h>
-bool copy_file_or_directory(String, String, bool, bool);
-bool copy_file(String, String, const struct stat&, int);
-bool copy_directory(String, String, bool);
-
-static mode_t my_umask = 0;
-
int main(int argc, char** argv)
{
if (pledge("stdio rpath wpath cpath fattr", nullptr) < 0) {
@@ -76,146 +63,3 @@ int main(int argc, char** argv)
}
return 0;
}
-
-/**
- * Copy a file or directory to a new location. Returns true if successful, false
- * otherwise. If there is an error, its description is output to stderr.
- *
- * Directories should only be copied if recursion_allowed is set.
- */
-bool copy_file_or_directory(String src_path, String dst_path, bool recursion_allowed, bool link)
-{
- int src_fd = open(src_path.characters(), O_RDONLY);
- if (src_fd < 0) {
- perror("open src");
- return false;
- }
-
- struct stat src_stat;
- int rc = fstat(src_fd, &src_stat);
- if (rc < 0) {
- perror("stat src");
- return false;
- }
-
- if (S_ISDIR(src_stat.st_mode)) {
- if (!recursion_allowed) {
- fprintf(stderr, "cp: -R not specified; omitting directory '%s'\n", src_path.characters());
- return false;
- }
- return copy_directory(src_path, dst_path, link);
- }
- if (link) {
- if (::link(src_path.characters(), dst_path.characters()) < 0) {
- perror("link");
- return false;
- }
- return true;
- }
-
- return copy_file(src_path, dst_path, src_stat, src_fd);
-}
-
-/**
- * Copy a source file to a destination file. Returns true if successful, false
- * otherwise. If there is an error, its description is output to stderr.
- *
- * To avoid repeated work, the source file's stat and file descriptor are required.
- */
-bool copy_file(String src_path, String dst_path, const struct stat& src_stat, int src_fd)
-{
- // NOTE: We don't copy the set-uid and set-gid bits.
- mode_t mode = (src_stat.st_mode & ~my_umask) & ~06000;
-
- int dst_fd = creat(dst_path.characters(), mode);
- if (dst_fd < 0) {
- if (errno != EISDIR) {
- perror("open dst");
- return false;
- }
- StringBuilder builder;
- builder.append(dst_path);
- builder.append('/');
- builder.append(LexicalPath(src_path).basename());
- dst_path = builder.to_string();
- dst_fd = creat(dst_path.characters(), 0666);
- if (dst_fd < 0) {
- perror("open dst");
- return false;
- }
- }
-
- if (src_stat.st_size > 0) {
- if (ftruncate(dst_fd, src_stat.st_size) < 0) {
- perror("cp: ftruncate");
- return false;
- }
- }
-
- for (;;) {
- char buffer[32768];
- ssize_t nread = read(src_fd, buffer, sizeof(buffer));
- if (nread < 0) {
- perror("read src");
- return false;
- }
- if (nread == 0)
- break;
- ssize_t remaining_to_write = nread;
- char* bufptr = buffer;
- while (remaining_to_write) {
- ssize_t nwritten = write(dst_fd, bufptr, remaining_to_write);
- if (nwritten < 0) {
- perror("write dst");
- return false;
- }
- assert(nwritten > 0);
- remaining_to_write -= nwritten;
- bufptr += nwritten;
- }
- }
-
- close(src_fd);
- close(dst_fd);
- return true;
-}
-
-/**
- * Copy the contents of a source directory into a destination directory.
- */
-bool copy_directory(String src_path, String dst_path, bool link)
-{
- int rc = mkdir(dst_path.characters(), 0755);
- if (rc < 0) {
- perror("cp: mkdir");
- return false;
- }
-
- String src_rp = Core::File::real_path_for(src_path);
- src_rp = String::format("%s/", src_rp.characters());
- String dst_rp = Core::File::real_path_for(dst_path);
- dst_rp = String::format("%s/", dst_rp.characters());
-
- if (!dst_rp.is_empty() && dst_rp.starts_with(src_rp)) {
- fprintf(stderr, "cp: Cannot copy %s into itself (%s)\n",
- src_path.characters(), dst_path.characters());
- return false;
- }
-
- Core::DirIterator di(src_path, Core::DirIterator::SkipDots);
- if (di.has_error()) {
- fprintf(stderr, "cp: DirIterator: %s\n", di.error_string());
- return false;
- }
- while (di.has_next()) {
- String filename = di.next_path();
- bool is_copied = copy_file_or_directory(
- String::format("%s/%s", src_path.characters(), filename.characters()),
- String::format("%s/%s", dst_path.characters(), filename.characters()),
- true, link);
- if (!is_copied) {
- return false;
- }
- }
- return true;
-}