diff options
| author | Lennart Poettering <lennart@poettering.net> | 2017-02-21 17:13:36 +0100 | 
|---|---|---|
| committer | Lennart Poettering <lennart@poettering.net> | 2017-02-21 21:55:43 +0100 | 
| commit | 175d308cad2075658b925e5abdbdfe7fa5fda466 (patch) | |
| tree | 9e67252e8719929f80c3ad5cfd5689d2cdf1db12 /src/boot/bootctl.c | |
| parent | ce21ed5c6120e33dd3502b25237c721dbbea891a (diff) | |
bootctl: rework file copy routines to reuse copy_bytes() from copy.c
Also, make sure to reuse temporary file handling used elsewhere.
Diffstat (limited to 'src/boot/bootctl.c')
| -rw-r--r-- | src/boot/bootctl.c | 121 | 
1 files changed, 42 insertions, 79 deletions
| diff --git a/src/boot/bootctl.c b/src/boot/bootctl.c index b747a95133..37f52c430a 100644 --- a/src/boot/bootctl.c +++ b/src/boot/bootctl.c @@ -38,20 +38,22 @@  #include "alloc-util.h"  #include "blkid-util.h" +#include "copy.h"  #include "dirent-util.h"  #include "efivars.h"  #include "fd-util.h"  #include "fileio.h" +#include "fs-util.h"  #include "locale-util.h"  #include "parse-util.h"  #include "rm-rf.h" +#include "stat-util.h"  #include "string-util.h"  #include "strv.h"  #include "umask-util.h"  #include "util.h"  #include "verbs.h"  #include "virt.h" -#include "stat-util.h"  static char *arg_path = NULL;  static bool arg_touch_variables = true; @@ -476,16 +478,16 @@ static int compare_version(const char *a, const char *b) {          return strverscmp(a, b);  } -static int version_check(int fd, const char *from, const char *to) { +static int version_check(int fd_from, const char *from, int fd_to, const char *to) {          _cleanup_free_ char *a = NULL, *b = NULL; -        _cleanup_close_ int fd2 = -1;          int r; -        assert(fd >= 0); +        assert(fd_from >= 0);          assert(from); +        assert(fd_to >= 0);          assert(to); -        r = get_file_version(fd, &a); +        r = get_file_version(fd_from, &a);          if (r < 0)                  return r;          if (r == 0) { @@ -493,15 +495,7 @@ static int version_check(int fd, const char *from, const char *to) {                  return -EINVAL;          } -        fd2 = open(to, O_RDONLY|O_CLOEXEC); -        if (fd2 < 0) { -                if (errno == ENOENT) -                        return 0; - -                return log_error_errno(errno, "Failed to open \"%s\" for reading: %m", to); -        } - -        r = get_file_version(fd2, &b); +        r = get_file_version(fd_to, &b);          if (r < 0)                  return r;          if (r == 0 || compare_product(a, b) != 0) { @@ -517,90 +511,59 @@ static int version_check(int fd, const char *from, const char *to) {          return 0;  } -static int copy_file(const char *from, const char *to, bool force) { -        _cleanup_fclose_ FILE *f = NULL, *g = NULL; -        char *p; +static int copy_file_with_version_check(const char *from, const char *to, bool force) { +        _cleanup_close_ int fd_from = -1, fd_to = -1; +        _cleanup_free_ char *t = NULL;          int r; -        struct timespec t[2]; -        struct stat st; - -        assert(from); -        assert(to); -        f = fopen(from, "re"); -        if (!f) +        fd_from = open(from, O_RDONLY|O_CLOEXEC|O_NOCTTY); +        if (fd_from < 0)                  return log_error_errno(errno, "Failed to open \"%s\" for reading: %m", from);          if (!force) { -                /* If this is an update, then let's compare versions first */ -                r = version_check(fileno(f), from, to); -                if (r < 0) -                        return r; -        } - -        p = strjoina(to, "~"); -        g = fopen(p, "wxe"); -        if (!g) { -                /* Directory doesn't exist yet? Then let's skip this... */ -                if (!force && errno == ENOENT) -                        return 0; - -                return log_error_errno(errno, "Failed to open \"%s\" for writing: %m", to); -        } +                fd_to = open(to, O_RDONLY|O_CLOEXEC|O_NOCTTY); +                if (fd_to < 0) { +                        if (errno != -ENOENT) +                                return log_error_errno(errno, "Failed to open \"%s\" for reading: %m", to); +                } else { +                        r = version_check(fd_from, from, fd_to, to); +                        if (r < 0) +                                return r; -        rewind(f); -        do { -                size_t k; -                uint8_t buf[32*1024]; +                        if (lseek(fd_from, 0, SEEK_SET) == (off_t) -1) +                                return log_error_errno(errno, "Failed to seek in \%s\": %m", from); -                k = fread(buf, 1, sizeof(buf), f); -                if (ferror(f)) { -                        r = log_error_errno(EIO, "Failed to read \"%s\": %m", from); -                        goto error; +                        fd_to = safe_close(fd_to);                  } +        } -                if (k == 0) -                        break; - -                fwrite(buf, 1, k, g); -                if (ferror(g)) { -                        r = log_error_errno(EIO, "Failed to write \"%s\": %m", to); -                        goto error; -                } -        } while (!feof(f)); +        r = tempfn_random(to, NULL, &t); +        if (r < 0) +                return log_oom(); -        r = fflush_and_check(g); -        if (r < 0) { -                log_error_errno(r, "Failed to write \"%s\": %m", to); -                goto error; +        RUN_WITH_UMASK(0000) { +                fd_to = open(t, O_WRONLY|O_CREAT|O_CLOEXEC|O_EXCL|O_NOFOLLOW, 0644); +                if (fd_to < 0) +                        return log_error_errno(errno, "Failed to open \"%s\" for writing: %m", t);          } -        r = fstat(fileno(f), &st); +        r = copy_bytes(fd_from, fd_to, (uint64_t) -1, COPY_REFLINK);          if (r < 0) { -                r = log_error_errno(errno, "Failed to get file timestamps of \"%s\": %m", from); -                goto error; +                unlink(t); +                return log_error_errno(errno, "Failed to copy data from \"%s\" to \"%s\": %m", from, t);          } -        t[0] = st.st_atim; -        t[1] = st.st_mtim; +        (void) copy_times(fd_from, fd_to); -        r = futimens(fileno(g), t); +        r = renameat(AT_FDCWD, t, AT_FDCWD, to);          if (r < 0) { -                r = log_error_errno(errno, "Failed to set file timestamps on \"%s\": %m", p); -                goto error; -        } - -        if (rename(p, to) < 0) { -                r = log_error_errno(errno, "Failed to rename \"%s\" to \"%s\": %m", p, to); -                goto error; +                (void) unlink_noerrno(t); +                return log_error_errno(errno, "Failed to rename \"%s\" to \"%s\": %m", t, to);          }          log_info("Copied \"%s\" to \"%s\".", from, to); -        return 0; -error: -        (void) unlink(p); -        return r; +        return 0;  }  static int mkdir_one(const char *prefix, const char *suffix) { @@ -644,7 +607,7 @@ static int copy_one_file(const char *esp_path, const char *name, bool force) {          p = strjoina(BOOTLIBDIR "/", name);          q = strjoina(esp_path, "/EFI/systemd/", name); -        r = copy_file(p, q, force); +        r = copy_file_with_version_check(p, q, force);          if (startswith(name, "systemd-boot")) {                  int k; @@ -654,7 +617,7 @@ static int copy_one_file(const char *esp_path, const char *name, bool force) {                  v = strjoina(esp_path, "/EFI/BOOT/BOOT", name + strlen("systemd-boot"));                  ascii_strupper(strrchr(v, '/') + 1); -                k = copy_file(p, v, force); +                k = copy_file_with_version_check(p, v, force);                  if (k < 0 && r == 0)                          r = k;          } | 
