/* * Copyright (C) 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #undef NDEBUG #define _LARGEFILE64_SOURCE extern "C" { #include <fec.h> } #include <assert.h> #include <android-base/file.h> #include <errno.h> #include <fcntl.h> #include <getopt.h> #include <openssl/sha.h> #include <pthread.h> #include <stdbool.h> #include <stdlib.h> #include <string.h> #include <sys/ioctl.h> #include <sys/mman.h> #ifndef IMAGE_NO_SPARSE #include <sparse/sparse.h> #endif #include "image.h" #if defined(__linux__) #include <linux/fs.h> #elif defined(__APPLE__) #include <sys/disk.h> #define BLKGETSIZE64 DKIOCGETBLOCKCOUNT #define O_LARGEFILE 0 #endif void image_init(image *ctx) { memset(ctx, 0, sizeof(*ctx)); } static void mmap_image_free(image *ctx) { if (ctx->input) { munmap(ctx->input, (size_t)ctx->inp_size); close(ctx->inp_fd); } if (ctx->fec_mmap_addr) { munmap(ctx->fec_mmap_addr, FEC_BLOCKSIZE + ctx->fec_size); close(ctx->fec_fd); } if (!ctx->inplace && ctx->output) { delete[] ctx->output; } } static void file_image_free(image *ctx) { assert(ctx->input == ctx->output); if (ctx->input) { delete[] ctx->input; } if (ctx->fec) { delete[] ctx->fec; } } void image_free(image *ctx) { if (ctx->mmap) { mmap_image_free(ctx); } else { file_image_free(ctx); } image_init(ctx); } static uint64_t get_size(int fd) { struct stat st; if (fstat(fd, &st) == -1) { FATAL("failed to fstat: %s\n", strerror(errno)); } uint64_t size = 0; if (S_ISBLK(st.st_mode)) { if (ioctl(fd, BLKGETSIZE64, &size) == -1) { FATAL("failed to ioctl(BLKGETSIZE64): %s\n", strerror(errno)); } } else if (S_ISREG(st.st_mode)) { size = st.st_size; } else { FATAL("unknown file mode: %d\n", (int)st.st_mode); } return size; } static void calculate_rounds(uint64_t size, image *ctx) { if (!size) { FATAL("empty file?\n"); } else if (size % FEC_BLOCKSIZE) { FATAL("file size %" PRIu64 " is not a multiple of %u bytes\n", size, FEC_BLOCKSIZE); } ctx->inp_size = size; ctx->blocks = fec_div_round_up(ctx->inp_size, FEC_BLOCKSIZE); ctx->rounds = fec_div_round_up(ctx->blocks, ctx->rs_n); } static void mmap_image_load(const std::vector<int>& fds, image *ctx, bool output_needed) { if (fds.size() != 1) { FATAL("multiple input files not supported with mmap\n"); } int fd = fds.front(); calculate_rounds(get_size(fd), ctx); /* check that we can memory map the file; on 32-bit platforms we are limited to encoding at most 4 GiB files */ if (ctx->inp_size > SIZE_MAX) { FATAL("cannot mmap %" PRIu64 " bytes\n", ctx->inp_size); } if (ctx->verbose) { INFO("memory mapping '%s' (size %" PRIu64 ")\n", ctx->fec_filename, ctx->inp_size); } int flags = PROT_READ; if (ctx->inplace) { flags |= PROT_WRITE; } void *p = mmap(NULL, (size_t)ctx->inp_size, flags, MAP_SHARED, fd, 0); if (p == MAP_FAILED) { FATAL("failed to mmap '%s' (size %" PRIu64 "): %s\n", ctx->fec_filename, ctx->inp_size, strerror(errno)); } ctx->inp_fd = fd; ctx->input = (uint8_t *)p; if (ctx->inplace) { ctx->output = ctx->input; } else if (output_needed) { if (ctx->verbose) { INFO("allocating %" PRIu64 " bytes of memory\n", ctx->inp_size); } ctx->output = new uint8_t[ctx->inp_size]; if (!ctx->output) { FATAL("failed to allocate memory\n"); } memcpy(ctx->output, ctx->input, ctx->inp_size); } /* fd is closed in mmap_image_free */ } #ifndef IMAGE_NO_SPARSE static int process_chunk(void *priv, const void *data, int len) { image *ctx = (image *)priv; assert(len % FEC_BLOCKSIZE == 0); if (data) { memcpy(&ctx->input[ctx->pos], data, len); } ctx->pos += len; return 0; } #endif static void file_image_load(const std::vector<int>& fds, image *ctx) { uint64_t size = 0; #ifndef IMAGE_NO_SPARSE std::vector<struct sparse_file *> files; #endif for (auto fd : fds) { uint64_t len = 0; #ifdef IMAGE_NO_SPARSE if (ctx->sparse) { FATAL("sparse files not supported\n"); } len = get_size(fd); #else struct sparse_file *file; if (ctx->sparse) { file = sparse_file_import(fd, false, false); } else { file = sparse_file_import_auto(fd, false, ctx->verbose); } if (!file) { FATAL("failed to read file %s\n", ctx->fec_filename); } len = sparse_file_len(file, false, false); files.push_back(file); #endif /* IMAGE_NO_SPARSE */ size += len; } calculate_rounds(size, ctx); if (ctx->verbose) { INFO("allocating %" PRIu64 " bytes of memory\n", ctx->inp_size); } ctx->input = new uint8_t[ctx->inp_size]; if (!ctx->input) { FATAL("failed to allocate memory\n"); } memset(ctx->input, 0, ctx->inp_size); ctx->output = ctx->input; ctx->pos = 0; #ifdef IMAGE_NO_SPARSE for (auto fd : fds) { uint64_t len = get_size(fd); if (!android::base::ReadFully(fd, &ctx->input[ctx->pos], len)) { FATAL("failed to read: %s\n", strerror(errno)); } ctx->pos += len; close(fd); } #else for (auto file : files) { sparse_file_callback(file, false, false, process_chunk, ctx); sparse_file_destroy(file); } for (auto fd : fds) { close(fd); } #endif } bool image_load(const std::vector<std::string>& filenames, image *ctx, bool output_needed) { assert(ctx->roots > 0 && ctx->roots < FEC_RSM); ctx->rs_n = FEC_RSM - ctx->roots; int flags = O_RDONLY; if (ctx->inplace) { flags = O_RDWR; } std::vector<int> fds; for (auto fn : filenames) { int fd = TEMP_FAILURE_RETRY(open(fn.c_str(), flags | O_LARGEFILE)); if (fd < 0) { FATAL("failed to open file '%s': %s\n", fn.c_str(), strerror(errno)); } fds.push_back(fd); } if (ctx->mmap) { mmap_image_load(fds, ctx, output_needed); } else { file_image_load(fds, ctx); } return true; } bool image_save(const std::string& filename, image *ctx) { if (ctx->inplace && ctx->mmap) { return true; /* nothing to do */ } /* TODO: support saving as a sparse file */ int fd = TEMP_FAILURE_RETRY(open(filename.c_str(), O_WRONLY | O_CREAT | O_TRUNC, 0666)); if (fd < 0) { FATAL("failed to open file '%s: %s'\n", filename.c_str(), strerror(errno)); } if (!android::base::WriteFully(fd, ctx->output, ctx->inp_size)) { FATAL("failed to write to output: %s\n", strerror(errno)); } close(fd); return true; } static void mmap_image_ecc_new(image *ctx) { if (ctx->verbose) { INFO("mmaping '%s' (size %u)\n", ctx->fec_filename, ctx->fec_size); } int fd = TEMP_FAILURE_RETRY(open(ctx->fec_filename, O_RDWR | O_CREAT, 0666)); if (fd < 0) { FATAL("failed to open file '%s': %s\n", ctx->fec_filename, strerror(errno)); } assert(sizeof(fec_header) <= FEC_BLOCKSIZE); size_t fec_size = FEC_BLOCKSIZE + ctx->fec_size; if (ftruncate(fd, fec_size) == -1) { FATAL("failed to ftruncate file '%s': %s\n", ctx->fec_filename, strerror(errno)); } if (ctx->verbose) { INFO("memory mapping '%s' (size %zu)\n", ctx->fec_filename, fec_size); } void *p = mmap(NULL, fec_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0); if (p == MAP_FAILED) { FATAL("failed to mmap '%s' (size %zu): %s\n", ctx->fec_filename, fec_size, strerror(errno)); } ctx->fec_fd = fd; ctx->fec_mmap_addr = (uint8_t *)p; ctx->fec = ctx->fec_mmap_addr; } static void file_image_ecc_new(image *ctx) { if (ctx->verbose) { INFO("allocating %u bytes of memory\n", ctx->fec_size); } ctx->fec = new uint8_t[ctx->fec_size]; if (!ctx->fec) { FATAL("failed to allocate %u bytes\n", ctx->fec_size); } } bool image_ecc_new(const std::string& filename, image *ctx) { assert(ctx->rounds > 0); /* image_load should be called first */ ctx->fec_filename = filename.c_str(); ctx->fec_size = ctx->rounds * ctx->roots * FEC_BLOCKSIZE; if (ctx->mmap) { mmap_image_ecc_new(ctx); } else { file_image_ecc_new(ctx); } return true; } bool image_ecc_load(const std::string& filename, image *ctx) { int fd = TEMP_FAILURE_RETRY(open(filename.c_str(), O_RDONLY)); if (fd < 0) { FATAL("failed to open file '%s': %s\n", filename.c_str(), strerror(errno)); } if (lseek64(fd, -FEC_BLOCKSIZE, SEEK_END) < 0) { FATAL("failed to seek to header in '%s': %s\n", filename.c_str(), strerror(errno)); } assert(sizeof(fec_header) <= FEC_BLOCKSIZE); uint8_t header[FEC_BLOCKSIZE]; fec_header *p = (fec_header *)header; if (!android::base::ReadFully(fd, header, sizeof(header))) { FATAL("failed to read %zd bytes from '%s': %s\n", sizeof(header), filename.c_str(), strerror(errno)); } if (p->magic != FEC_MAGIC) { FATAL("invalid magic in '%s': %08x\n", filename.c_str(), p->magic); } if (p->version != FEC_VERSION) { FATAL("unsupported version in '%s': %u\n", filename.c_str(), p->version); } if (p->size != sizeof(fec_header)) { FATAL("unexpected header size in '%s': %u\n", filename.c_str(), p->size); } if (p->roots == 0 || p->roots >= FEC_RSM) { FATAL("invalid roots in '%s': %u\n", filename.c_str(), p->roots); } if (p->fec_size % p->roots || p->fec_size % FEC_BLOCKSIZE) { FATAL("invalid length in '%s': %u\n", filename.c_str(), p->fec_size); } ctx->roots = (int)p->roots; ctx->rs_n = FEC_RSM - ctx->roots; calculate_rounds(p->inp_size, ctx); if (!image_ecc_new(filename, ctx)) { FATAL("failed to allocate ecc\n"); } if (p->fec_size != ctx->fec_size) { FATAL("inconsistent header in '%s'\n", filename.c_str()); } if (lseek64(fd, 0, SEEK_SET) < 0) { FATAL("failed to rewind '%s': %s", filename.c_str(), strerror(errno)); } if (!ctx->mmap && !android::base::ReadFully(fd, ctx->fec, ctx->fec_size)) { FATAL("failed to read %u bytes from '%s': %s\n", ctx->fec_size, filename.c_str(), strerror(errno)); } close(fd); uint8_t hash[SHA256_DIGEST_LENGTH]; SHA256(ctx->fec, ctx->fec_size, hash); if (memcmp(hash, p->hash, SHA256_DIGEST_LENGTH) != 0) { FATAL("invalid ecc data\n"); } return true; } bool image_ecc_save(image *ctx) { assert(sizeof(fec_header) <= FEC_BLOCKSIZE); uint8_t header[FEC_BLOCKSIZE]; uint8_t *p = header; if (ctx->mmap) { p = (uint8_t *)&ctx->fec_mmap_addr[ctx->fec_size]; } memset(p, 0, FEC_BLOCKSIZE); fec_header *f = (fec_header *)p; f->magic = FEC_MAGIC; f->version = FEC_VERSION; f->size = sizeof(fec_header); f->roots = ctx->roots; f->fec_size = ctx->fec_size; f->inp_size = ctx->inp_size; SHA256(ctx->fec, ctx->fec_size, f->hash); /* store a copy of the fec_header at the end of the header block */ memcpy(&p[sizeof(header) - sizeof(fec_header)], p, sizeof(fec_header)); if (!ctx->mmap) { assert(ctx->fec_filename); int fd = TEMP_FAILURE_RETRY(open(ctx->fec_filename, O_WRONLY | O_CREAT | O_TRUNC, 0666)); if (fd < 0) { FATAL("failed to open file '%s': %s\n", ctx->fec_filename, strerror(errno)); } if (!android::base::WriteFully(fd, ctx->fec, ctx->fec_size) || !android::base::WriteFully(fd, header, sizeof(header))) { FATAL("failed to write to output: %s\n", strerror(errno)); } close(fd); } return true; } static void * process(void *cookie) { image_proc_ctx *ctx = (image_proc_ctx *)cookie; ctx->func(ctx); return NULL; } bool image_process(image_proc_func func, image *ctx) { int threads = ctx->threads; if (threads < IMAGE_MIN_THREADS) { threads = sysconf(_SC_NPROCESSORS_ONLN); if (threads < IMAGE_MIN_THREADS) { threads = IMAGE_MIN_THREADS; } } assert(ctx->rounds > 0); if ((uint64_t)threads > ctx->rounds) { threads = (int)ctx->rounds; } if (threads > IMAGE_MAX_THREADS) { threads = IMAGE_MAX_THREADS; } if (ctx->verbose) { INFO("starting %d threads to compute RS(255, %d)\n", threads, ctx->rs_n); } pthread_t pthreads[threads]; image_proc_ctx args[threads]; uint64_t current = 0; uint64_t end = ctx->rounds * ctx->rs_n * FEC_BLOCKSIZE; uint64_t rs_blocks_per_thread = fec_div_round_up(ctx->rounds * FEC_BLOCKSIZE, threads); if (ctx->verbose) { INFO("computing %" PRIu64 " codes per thread\n", rs_blocks_per_thread); } for (int i = 0; i < threads; ++i) { args[i].func = func; args[i].id = i; args[i].ctx = ctx; args[i].rv = 0; args[i].fec_pos = current * ctx->roots; args[i].start = current * ctx->rs_n; args[i].end = (current + rs_blocks_per_thread) * ctx->rs_n; args[i].rs = init_rs_char(FEC_PARAMS(ctx->roots)); if (!args[i].rs) { FATAL("failed to initialize encoder for thread %d\n", i); } if (args[i].end > end) { args[i].end = end; } else if (i == threads && args[i].end + rs_blocks_per_thread * ctx->rs_n > end) { args[i].end = end; } if (ctx->verbose) { INFO("thread %d: [%" PRIu64 ", %" PRIu64 ")\n", i, args[i].start, args[i].end); } assert(args[i].start < args[i].end); assert((args[i].end - args[i].start) % ctx->rs_n == 0); if (pthread_create(&pthreads[i], NULL, process, &args[i]) != 0) { FATAL("failed to create thread %d\n", i); } current += rs_blocks_per_thread; } ctx->rv = 0; for (int i = 0; i < threads; ++i) { if (pthread_join(pthreads[i], NULL) != 0) { FATAL("failed to join thread %d: %s\n", i, strerror(errno)); } ctx->rv += args[i].rv; if (args[i].rs) { free_rs_char(args[i].rs); args[i].rs = NULL; } } return true; }