/* * 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. */ #ifndef __FEC_H__ #define __FEC_H__ #include #include #include #include #include #define IMAGE_MIN_THREADS 1 #define IMAGE_MAX_THREADS 128 #define INFO(x...) \ fprintf(stderr, x); #define FATAL(x...) { \ fprintf(stderr, x); \ exit(1); \ } #define unlikely(x) __builtin_expect(!!(x), 0) struct image { /* if true, decode file in place instead of creating a new output file */ bool inplace; /* if true, use memory mapping instead of copying all input into memory */ bool mmap; /* if true, assume input is a sparse file */ bool sparse; /* if true, print more verbose information to stderr */ bool verbose; const char *fec_filename; int fec_fd; int inp_fd; /* the number of Reed-Solomon generator polynomial roots, also the number of parity bytes generated for each N bytes in RS(M, N) */ int roots; /* for RS(M, N), N = M - roots */ int rs_n; int threads; uint32_t fec_size; uint64_t blocks; uint64_t inp_size; uint64_t pos; uint64_t rounds; uint64_t rv; uint8_t *fec; uint8_t *fec_mmap_addr; uint8_t *input; uint8_t *output; }; struct image_proc_ctx; typedef void (*image_proc_func)(image_proc_ctx *); struct image_proc_ctx { image_proc_func func; int id; image *ctx; uint64_t rv; uint64_t fec_pos; uint64_t start; uint64_t end; void *rs; }; extern bool image_load(const std::vector& filename, image *ctx, bool output_needed); extern bool image_save(const std::string& filename, image *ctx); extern bool image_ecc_new(const std::string& filename, image *ctx); extern bool image_ecc_load(const std::string& filename, image *ctx); extern bool image_ecc_save(image *ctx); extern bool image_process(image_proc_func f, image *ctx); extern void image_init(image *ctx); extern void image_free(image *ctx); inline uint8_t image_get_interleaved_byte(uint64_t i, image *ctx) { uint64_t offset = fec_ecc_interleave(i, ctx->rs_n, ctx->rounds); if (unlikely(offset >= ctx->inp_size)) { return 0; } return ctx->input[offset]; } inline void image_set_interleaved_byte(uint64_t i, image *ctx, uint8_t value) { uint64_t offset = fec_ecc_interleave(i, ctx->rs_n, ctx->rounds); if (unlikely(offset >= ctx->inp_size)) { assert(value == 0); } else if (ctx->output && ctx->output[offset] != value) { ctx->output[offset] = value; } } #endif // __FEC_H__