/* * Stripe a flash image across multiple files. * * Copyright (C) 2019 Xilinx, Inc. All rights reserved. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #include #include #include #include #include #include #include #include #include /* N way (num) in place bit striper. Lay out row wise bits column wise * (from element 0 to N-1). num is the length of x, and dir reverses the * direction of the transform. be determines the bit endianess scheme. * false to lay out bits LSB to MSB (little endian) and true for big endian. * * Best illustrated by examples: * Each digit in the below array is a single bit (num == 3, be == false): * * {{ 76543210, } ----- stripe (dir == false) -----> {{ FCheb630, } * { hgfedcba, } { GDAfc741, } * { HGFEDCBA, }} <---- upstripe (dir == true) ----- { HEBgda52, }} * * Same but with be == true: * * {{ 76543210, } ----- stripe (dir == false) -----> {{ 741gdaFC, } * { hgfedcba, } { 630fcHEB, } * { HGFEDCBA, }} <---- upstripe (dir == true) ----- { 52hebGDA, }} */ static inline void stripe8(uint8_t *x, int num, bool dir, bool be) { uint8_t r[num]; memset(r, 0, sizeof(uint8_t) * num); int idx[2] = {0, 0}; int bit[2] = {0, be ? 7 : 0}; int d = dir; for (idx[0] = 0; idx[0] < num; ++idx[0]) { for (bit[0] = be ? 7 : 0; bit[0] != (be ? -1 : 8); bit[0] += be ? -1 : 1) { r[idx[!d]] |= x[idx[d]] & 1 << bit[d] ? 1 << bit[!d] : 0; idx[1] = (idx[1] + 1) % num; if (!idx[1]) { bit[1] += be ? -1 : 1; } } } memcpy(x, r, sizeof(uint8_t) * num); } int main (int argc, char *argv []) { #ifdef UNSTRIPE bool unstripe = true; #else bool unstripe = false; #endif #ifdef FLASH_STRIPE_BE bool be = true; #else bool be = false; #endif int i; const char *exe_name = argv[0]; argc--; argv++; if (argc < 2) { fprintf(stderr, "ERROR: %s requires at least two args\n", exe_name); return 1; } const char *single_f = argv[0]; int single; if (unstripe) { single = creat(single_f, 0644); } else { single = open(single_f, 0); } if (single == -1) { perror(argv[0]); return 1; } argv++; argc--; int multiple[argc]; for (i = 0; i < argc; ++i) { if (unstripe) { multiple[i] = open(argv[i], 0); } else { multiple[i] = creat(argv[i], 0644); } if (multiple[i] == -1) { perror(argv[i]); return 1; } } while (true) { uint8_t buf[argc]; for (i = 0; i < argc; ++i) { switch (read(!unstripe ? single : multiple[ #if defined(FLASH_STRIPE_BW) && defined (FLASH_STRIPE_BE) argc - 1 - #endif i], &buf[i], 1)) { case 0: if (i == 0) { goto done; } else if (!unstripe) { fprintf(stderr, "WARNING:input file %s is not multiple of " "%d bytes, padding with garbage byte\n", single_f, argc); } break; case -1: perror(unstripe ? argv[i] : single_f); return 1; } } #ifndef FLASH_STRIPE_BW stripe8(buf, argc, unstripe, be); #endif for (i = 0; i < argc; ++i) { switch (write(unstripe ? single : multiple[ #if defined(FLASH_STRIPE_BW) && defined (FLASH_STRIPE_BE) argc - 1 - #endif i], &buf[i], 1)) { case -1: perror(unstripe ? single_f : argv[i]); return 1; case 0: i--; /* try again */ } } } done: close(single); for (i = 0; i < argc; ++i) { close(multiple[argc]); } return 0; }