diff options
Diffstat (limited to 'roms/skiboot/libflash/ecc.c')
-rw-r--r-- | roms/skiboot/libflash/ecc.c | 449 |
1 files changed, 449 insertions, 0 deletions
diff --git a/roms/skiboot/libflash/ecc.c b/roms/skiboot/libflash/ecc.c new file mode 100644 index 000000000..25f518e66 --- /dev/null +++ b/roms/skiboot/libflash/ecc.c @@ -0,0 +1,449 @@ +// SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later +/* + * This is based on the hostboot ecc code + * + * Copyright 2013-2019 IBM Corp. + */ + +#include <stdint.h> +#include <inttypes.h> +#include <string.h> + +#include <ccan/endian/endian.h> + +#include "libflash.h" +#include "ecc.h" + +/* Bit field identifiers for syndrome calculations. */ +enum eccbitfields +{ + GD = 0xff, //< Good, ECC matches. + UE = 0xfe, //< Uncorrectable. + E0 = 71, //< Error in ECC bit 0 + E1 = 70, //< Error in ECC bit 1 + E2 = 69, //< Error in ECC bit 2 + E3 = 68, //< Error in ECC bit 3 + E4 = 67, //< Error in ECC bit 4 + E5 = 66, //< Error in ECC bit 5 + E6 = 65, //< Error in ECC bit 6 + E7 = 64 //< Error in ECC bit 7 + /* 0-63 Correctable bit in byte */ +}; + +/* + * Matrix used for ECC calculation. + * + * Each row of this is the set of data word bits that are used for + * the calculation of the corresponding ECC bit. The parity of the + * bitset is the value of the ECC bit. + * + * ie. ECC[n] = eccMatrix[n] & data + * + * Note: To make the math easier (and less shifts in resulting code), + * row0 = ECC7. HW numbering is MSB, order here is LSB. + * + * These values come from the HW design of the ECC algorithm. + */ +static uint64_t eccmatrix[] = { + 0x0000e8423c0f99ffull, + 0x00e8423c0f99ff00ull, + 0xe8423c0f99ff0000ull, + 0x423c0f99ff0000e8ull, + 0x3c0f99ff0000e842ull, + 0x0f99ff0000e8423cull, + 0x99ff0000e8423c0full, + 0xff0000e8423c0f99ull +}; + +/** + * Syndrome calculation matrix. + * + * Maps syndrome to flipped bit. + * + * To perform ECC correction, this matrix is a look-up of the bit + * that is bad based on the binary difference of the good and bad + * ECC. This difference is called the "syndrome". + * + * When a particular bit is on in the data, it cause a column from + * eccMatrix being XOR'd into the ECC field. This column is the + * "effect" of each bit. If a bit is flipped in the data then its + * "effect" is missing from the ECC. You can calculate ECC on unknown + * quality data and compare the ECC field between the calculated + * value and the stored value. If the difference is zero, then the + * data is clean. If the difference is non-zero, you look up the + * difference in the syndrome table to identify the "effect" that + * is missing, which is the bit that is flipped. + * + * Notice that ECC bit flips are recorded by a single "effect" + * bit (ie. 0x1, 0x2, 0x4, 0x8 ...) and double bit flips are identified + * by the UE status in the table. + * + * Bits are in MSB order. + */ +static enum eccbitfields syndromematrix[] = { + GD, E7, E6, UE, E5, UE, UE, 47, E4, UE, UE, 37, UE, 35, 39, UE, + E3, UE, UE, 48, UE, 30, 29, UE, UE, 57, 27, UE, 31, UE, UE, UE, + E2, UE, UE, 17, UE, 18, 40, UE, UE, 58, 22, UE, 21, UE, UE, UE, + UE, 16, 49, UE, 19, UE, UE, UE, 23, UE, UE, UE, UE, 20, UE, UE, + E1, UE, UE, 51, UE, 46, 9, UE, UE, 34, 10, UE, 32, UE, UE, 36, + UE, 62, 50, UE, 14, UE, UE, UE, 13, UE, UE, UE, UE, UE, UE, UE, + UE, 61, 8, UE, 41, UE, UE, UE, 11, UE, UE, UE, UE, UE, UE, UE, + 15, UE, UE, UE, UE, UE, UE, UE, UE, UE, 12, UE, UE, UE, UE, UE, + E0, UE, UE, 55, UE, 45, 43, UE, UE, 56, 38, UE, 1, UE, UE, UE, + UE, 25, 26, UE, 2, UE, UE, UE, 24, UE, UE, UE, UE, UE, 28, UE, + UE, 59, 54, UE, 42, UE, UE, 44, 6, UE, UE, UE, UE, UE, UE, UE, + 5, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, + UE, 63, 53, UE, 0, UE, UE, UE, 33, UE, UE, UE, UE, UE, UE, UE, + 3, UE, UE, 52, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, + 7, UE, UE, UE, UE, UE, UE, UE, UE, 60, UE, UE, UE, UE, UE, UE, + UE, UE, UE, UE, 4, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, UE, +}; + +/** + * Create the ECC field corresponding to a 8-byte data field + * + * @data: The 8 byte data to generate ECC for. + * @return: The 1 byte ECC corresponding to the data. + */ +static uint8_t eccgenerate(uint64_t data) +{ + int i; + uint8_t result = 0; + + for (i = 0; i < 8; i++) + result |= __builtin_parityll(eccmatrix[i] & data) << i; + + return result; +} + +/** + * Verify the data and ECC match or indicate how they are wrong. + * + * @data: The data to check ECC on. + * @ecc: The [supposed] ECC for the data. + * + * @return: eccBitfield or 0-64. + * + * @retval GD - Indicates the data is good (matches ECC). + * @retval UE - Indicates the data is uncorrectable. + * @retval all others - Indication of which bit is incorrect. + */ +static enum eccbitfields eccverify(uint64_t data, uint8_t ecc) +{ + return syndromematrix[eccgenerate(data) ^ ecc]; +} + +/* IBM bit ordering */ +static inline uint64_t eccflipbit(uint64_t data, uint8_t bit) +{ + if (bit > 63) + return data; + + return data ^ (1ul << (63 - bit)); +} + +static int eccbyte(beint64_t *dst, struct ecc64 *src) +{ + uint8_t ecc, badbit; + uint64_t data; + + data = be64_to_cpu(src->data); + ecc = src->ecc; + + badbit = eccverify(data, ecc); + if (badbit == UE) { + FL_ERR("ECC: uncorrectable error: %016llx %02x\n", (unsigned long long int)data, ecc); + return badbit; + } + if (badbit <= UE) + FL_INF("ECC: correctable error: %i\n", badbit); + if (badbit < 64) + *dst = cpu_to_be64(eccflipbit(data, badbit)); + else + *dst = cpu_to_be64(data); + + return 0; +} + +static beint64_t *inc_beint64_by(const void *p, uint64_t i) +{ + return (beint64_t *)(((char *)p) + i); +} + +static uint64_t *inc_uint64_by(const void *p, uint64_t i) +{ + return (uint64_t *)(((char *)p) + i); +} + +static struct ecc64 *inc_ecc64_by(struct ecc64 *p, uint64_t i) +{ + return (struct ecc64 *)(((char *)p) + i); +} + +static uint64_t whole_ecc_bytes(uint64_t i) +{ + return i & ~(BYTES_PER_ECC - 1); +} + +static uint64_t whole_ecc_structs(uint64_t i) +{ + return whole_ecc_bytes(i) >> 3; +} + +/** + * Copy data from an input buffer with ECC to an output buffer without ECC. + * Correct it along the way and check for errors. + * + * @dst: destination buffer without ECC + * @src: source buffer with ECC + * @len: number of bytes of data to copy (without ecc). + * Must be 8 byte aligned. + * + * @return: Success or error + * + * @retval: 0 - success + * @retfal: other - fail + */ +int memcpy_from_ecc(beint64_t *dst, struct ecc64 *src, uint64_t len) +{ + uint32_t i; + + if (len & 0x7) { + /* TODO: we could probably handle this */ + FL_ERR("ECC data length must be 8 byte aligned length:%" PRIx64 "\n", + len); + return -1; + } + + /* Handle in chunks of 8 bytes, so adjust the length */ + len >>= 3; + + for (i = 0; i < len; i++) { + int rc; + rc = eccbyte(dst, src + i); + if (rc) + return rc; + dst++; + } + return 0; +} + +/** + * Copy data from an input buffer with ECC to an output buffer without ECC. + * Correct it along the way and check for errors. + * + * Unlike memcmp_from_ecc() which requires that the first byte into + * dst be the first byte in src (which must also be aligned to a + * struct ecc64 struct boundary) this function can cope with the first + * byte in dst not being the first byte in src. + * + * Note: src MUST still be aligned to a struct ecc64 otherwise ECC + * calculations are impossible. + * + * The alignment parameter species the number of bytes present at the + * start of src that should be skipped and not written to dst. Once + * again, these bytes must be in src otherwise the ECC cannot be + * checked. + * + * len also doesn't have any value limitation for this function. Of + * course src must contain an exact multiple of struct ecc64 otherwise + * ECC calculation cannot be performed but this function won't copy + * the entirety of the last src data word if len is not mutiple of 8 + * + * @dst: destination buffer without ECC + * @src: source buffer with ECC + * @len: number of bytes of data to copy (without ecc). + * @alignment: number of leading bytes in src which shouldn't be + * copied to dst + * @return: Success or error + * + * @retval: 0 - success + * @retfal: other - fail + */ +int memcpy_from_ecc_unaligned(beint64_t *dst, struct ecc64 *src, + uint64_t len, uint8_t alignment) +{ + char data[BYTES_PER_ECC]; + uint8_t bytes_wanted; + int rc; + + if (alignment > 8) + return -1; + + bytes_wanted = BYTES_PER_ECC - alignment; + + /* + * Only actually do the first calculation if an alignment is + * required - otherwise jump straight to memcpy_from_ecc() + */ + if (alignment) { + rc = eccbyte((beint64_t *)data, src); + if (rc) + return rc; + + memcpy(dst, &data[alignment], bytes_wanted); + + src = inc_ecc64_by(src, sizeof(struct ecc64)); + dst = inc_beint64_by(dst, bytes_wanted); + len -= bytes_wanted; + } + + if (len >= BYTES_PER_ECC) { + rc = memcpy_from_ecc(dst, src, whole_ecc_bytes(len)); + if (rc) + return rc; + + /* + * It helps to let the compiler to the pointer arithmetic + * here, (dst and src are different types) + */ + dst += whole_ecc_structs(len); + src += whole_ecc_structs(len); + len -= whole_ecc_bytes(len); + } + + if (len) { + rc = eccbyte((beint64_t *)data, src); + if (rc) + return rc; + + memcpy(dst, data, len); + } + + return 0; +} + +/** + * Copy data from an input buffer without ECC to an output buffer with ECC. + * + * @dst: destination buffer with ECC + * @src: source buffer without ECC + * @len: number of bytes of data to copy (without ecc, length of src). + * Note: dst must be big enough to hold ecc bytes as well. + * Must be 8 byte aligned. + * + * @return: success or failure + * + * @retval: 0 - success + * @retfal: other - fail + */ +int memcpy_to_ecc(struct ecc64 *dst, const beint64_t *src, uint64_t len) +{ + struct ecc64 ecc_word; + uint64_t i; + + if (len & 0x7) { + /* TODO: we could probably handle this */ + FL_ERR("Data to add ECC bytes to must be 8 byte aligned length: %" + PRIx64 "\n", len); + return -1; + } + + /* Handle in chunks of 8 bytes, so adjust the length */ + len >>= 3; + + for (i = 0; i < len; i++) { + ecc_word.ecc = eccgenerate(be64_to_cpu(*(src + i))); + ecc_word.data = *(src + i); + + *(dst + i) = ecc_word; + } + + return 0; +} + +/** + * Copy data from an input buffer without ECC to an output buffer with ECC. + * + * Unlike memcmp_to_ecc() which requires that the first byte in src be + * the first byte of a struct ecc64 structure this function does not + * have this requirement. + * + * Like memcpy_to_ecc_unaligned() the alignment parameter specfies the + * number of bytes in the first src word that are missing and would be + * required to form a struct ecc64 structure. + * + * It must be noted here that extra work IN THE CALLER must be done + * if your data is unaligned. In order to peform ECC calculations + * whatever portions of the ecc words are missing in src must be in + * dst. + * + * For example, if there is an alignment value of 1 then this means + * there is 1 byte (of the total of 8 bytes) missing in src which is + * needed to calculate the first ECC byte. Therefore the first byte of + * dst MUST CONTAIN IT! + * + * The same problem exists for the end of the buffer where src may not + * end exactly aligned, if this is the case dst must contain the + * required bytes to calculate the last ECC byte - they should be in + * dst where they would normally be found if src did contain those + * bytes. + * + * @dst: destination buffer with ECC + * @src: source buffer without ECC + * @len: number of bytes of data to copy (without ecc, length of src). + * @alignment: The number of bytes 'missing' from the start of src to + * be struct ecc64 aligned + * + * Note: dst must be big enough to hold ecc bytes as well. + * Must be 8 byte aligned. + * + * @return: success or failure + * + * @retval: 0 - success + * @retfal: other - fail + */ + +int memcpy_to_ecc_unaligned(struct ecc64 *dst, const beint64_t *src, + uint64_t len, uint8_t alignment) +{ + struct ecc64 ecc_word; + uint8_t bytes_wanted; + int rc; + + bytes_wanted = BYTES_PER_ECC - alignment; + + /* + * Only actually do the first calculation if an alignment is + * required - otherwise jump straight to memcpy_to_ecc() + */ + if (alignment) { + ecc_word.data = dst->data; + memcpy(inc_uint64_by(&ecc_word.data, alignment), src, bytes_wanted); + + ecc_word.ecc = eccgenerate(be64_to_cpu(ecc_word.data)); + memcpy(dst, inc_ecc64_by(&ecc_word, alignment), + sizeof(struct ecc64) - alignment); + + dst = inc_ecc64_by(dst, sizeof(struct ecc64) - alignment); + src = inc_beint64_by(src, bytes_wanted); + len -= bytes_wanted; + } + + if (len >= BYTES_PER_ECC) { + rc = memcpy_to_ecc(dst, src, whole_ecc_bytes(len)); + if (rc) + return rc; + + /* + * It helps to let the compiler to the pointer arithmetic + * here, (dst and src are different types) + */ + dst += whole_ecc_structs(len); + src += whole_ecc_structs(len); + len -= whole_ecc_bytes(len); + } + + if (len) { + bytes_wanted = BYTES_PER_ECC - len; + + ecc_word.data = *src; + memcpy(inc_uint64_by(&ecc_word.data, len), inc_ecc64_by(dst, len), + bytes_wanted); + ecc_word.ecc = eccgenerate(be64_to_cpu(ecc_word.data)); + + *dst = ecc_word; + } + + return 0; +} |