iohelper.c 6.9 KB
Newer Older
1 2 3
/*
 * iohelper.c: Helper program to perform I/O operations on files
 *
4
 * Copyright (C) 2011-2012 Red Hat, Inc.
5 6 7 8 9 10 11 12 13 14 15 16
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
17
 * License along with this library.  If not, see
O
Osier Yang 已提交
18
 * <http://www.gnu.org/licenses/>.
19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
 *
 * Author: Daniel P. Berrange <berrange@redhat.com>
 *
 * Current support
 *   - Read existing file
 *   - Write existing file
 *   - Create & write new file
 */

#include <config.h>

#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>

35
#include "virutil.h"
36
#include "virthread.h"
E
Eric Blake 已提交
37
#include "virfile.h"
38
#include "viralloc.h"
39
#include "virerror.h"
40
#include "virrandom.h"
41
#include "virstring.h"
42
#include "virgettext.h"
43 44 45

#define VIR_FROM_THIS VIR_FROM_STORAGE

46
static int
47
runIO(const char *path, int fd, int oflags)
48
{
49 50
    void *base = NULL; /* Location to be freed */
    char *buf = NULL; /* Aligned location within base */
51
    size_t buflen = 1024*1024;
52
    intptr_t alignMask = 64*1024 - 1;
53 54 55 56
    int ret = -1;
    int fdin, fdout;
    const char *fdinname, *fdoutname;
    unsigned long long total = 0;
57 58
    bool direct = O_DIRECT && ((oflags & O_DIRECT) != 0);
    off_t end = 0;
59

60 61
#if HAVE_POSIX_MEMALIGN
    if (posix_memalign(&base, alignMask + 1, buflen)) {
62 63 64
        virReportOOMError();
        goto cleanup;
    }
65 66
    buf = base;
#else
67
    if (VIR_ALLOC_N(buf, buflen + alignMask) < 0)
68 69
        goto cleanup;
    base = buf;
P
Paolo Bonzini 已提交
70
    buf = (char *) (((intptr_t) base + alignMask) & ~alignMask);
71
#endif
72

73
    switch (oflags & O_ACCMODE) {
74 75 76 77 78
    case O_RDONLY:
        fdin = fd;
        fdinname = path;
        fdout = STDOUT_FILENO;
        fdoutname = "stdout";
79 80
        /* To make the implementation simpler, we give up on any
         * attempt to use O_DIRECT in a non-trivial manner.  */
81
        if (direct && ((end = lseek(fd, 0, SEEK_CUR)) != 0)) {
82 83 84 85
            virReportSystemError(end < 0 ? errno : EINVAL, "%s",
                                 _("O_DIRECT read needs entire seekable file"));
            goto cleanup;
        }
86 87 88 89 90 91
        break;
    case O_WRONLY:
        fdin = STDIN_FILENO;
        fdinname = "stdin";
        fdout = fd;
        fdoutname = path;
92 93 94 95 96 97 98
        /* To make the implementation simpler, we give up on any
         * attempt to use O_DIRECT in a non-trivial manner.  */
        if (direct && (end = lseek(fd, 0, SEEK_END)) != 0) {
            virReportSystemError(end < 0 ? errno : EINVAL, "%s",
                                 _("O_DIRECT write needs empty seekable file"));
            goto cleanup;
        }
99 100 101 102 103 104
        break;

    case O_RDWR:
    default:
        virReportSystemError(EINVAL,
                             _("Unable to process file with flags %d"),
105
                             (oflags & O_ACCMODE));
106 107 108 109 110 111
        goto cleanup;
    }

    while (1) {
        ssize_t got;

112 113 114 115 116 117 118 119 120
        /* If we read with O_DIRECT from file we can't use saferead as
         * it can lead to unaligned read after reading last bytes.
         * If we write with O_DIRECT use should use saferead so that
         * writes will be aligned.
         * In other cases using saferead reduces number of syscalls.
         */
        if (fdin == fd && direct) {
            if ((got = read(fdin, buf, buflen)) < 0 &&
                errno == EINTR)
121
                continue;
122 123 124 125 126
        } else {
            got = saferead(fdin, buf, buflen);
        }

        if (got < 0) {
127 128 129 130
            virReportSystemError(errno, _("Unable to read %s"), fdinname);
            goto cleanup;
        }
        if (got == 0)
131
            break;
132 133

        total += got;
134 135 136

        /* handle last write size align in direct case */
        if (got < buflen && direct && fdout == fd) {
137
            ssize_t aligned_got = (got + alignMask) & ~alignMask;
138

139 140 141
            memset(buf + got, 0, aligned_got - got);

            if (safewrite(fdout, buf, aligned_got) < 0) {
142 143 144 145 146 147 148 149 150 151
                virReportSystemError(errno, _("Unable to write %s"), fdoutname);
                goto cleanup;
            }

            if (ftruncate(fd, total) < 0) {
                virReportSystemError(errno, _("Unable to truncate %s"), fdoutname);
                goto cleanup;
            }

            break;
152
        }
153

154 155 156 157 158 159
        if (safewrite(fdout, buf, got) < 0) {
            virReportSystemError(errno, _("Unable to write %s"), fdoutname);
            goto cleanup;
        }
    }

160 161
    /* Ensure all data is written */
    if (fdatasync(fdout) < 0) {
162 163 164 165 166
        if (errno != EINVAL && errno != EROFS) {
            /* fdatasync() may fail on some special FDs, e.g. pipes */
            virReportSystemError(errno, _("unable to fsync %s"), fdoutname);
            goto cleanup;
        }
167 168
    }

169 170
    ret = 0;

171
 cleanup:
172 173 174 175 176 177
    if (VIR_CLOSE(fd) < 0 &&
        ret == 0) {
        virReportSystemError(errno, _("Unable to close %s"), path);
        ret = -1;
    }

178
    VIR_FREE(base);
179 180 181
    return ret;
}

182 183 184 185 186 187 188 189
static const char *program_name;

ATTRIBUTE_NORETURN static void
usage(int status)
{
    if (status) {
        fprintf(stderr, _("%s: try --help for more details"), program_name);
    } else {
190
        printf(_("Usage: %s FILENAME FD\n"), program_name);
191 192 193 194 195 196
    }
    exit(status);
}

int
main(int argc, char **argv)
197 198
{
    const char *path;
199 200 201 202
    int oflags = -1;
    int fd = -1;

    program_name = argv[0];
203

204 205
    if (virGettextInitialize() < 0 ||
        virThreadInitialize() < 0 ||
206
        virErrorInitialize() < 0) {
207
        fprintf(stderr, _("%s: initialization failed\n"), program_name);
208 209 210 211 212
        exit(EXIT_FAILURE);
    }

    path = argv[1];

213 214
    if (argc > 1 && STREQ(argv[1], "--help"))
        usage(EXIT_SUCCESS);
215 216
    if (argc == 3) { /* FILENAME FD */
        if (virStrToLong_i(argv[2], NULL, 10, &fd) < 0) {
217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236
            fprintf(stderr, _("%s: malformed fd %s"),
                    program_name, argv[3]);
            exit(EXIT_FAILURE);
        }
#ifdef F_GETFL
        oflags = fcntl(fd, F_GETFL);
#else
        /* Stupid mingw.  */
        if (fd == STDIN_FILENO)
            oflags = O_RDONLY;
        else if (fd == STDOUT_FILENO)
            oflags = O_WRONLY;
#endif
        if (oflags < 0) {
            fprintf(stderr, _("%s: unable to determine access mode of fd %d"),
                    program_name, fd);
            exit(EXIT_FAILURE);
        }
    } else { /* unknown argc pattern */
        usage(EXIT_FAILURE);
237 238
    }

239
    if (fd < 0 || runIO(path, fd, oflags) < 0)
240 241 242 243
        goto error;

    return 0;

244
 error:
245 246
    fprintf(stderr, _("%s: failure with %s\n: %s"),
            program_name, path, virGetLastErrorMessage());
247 248
    exit(EXIT_FAILURE);
}