/* * Mini Object Storage, (C) 2014 Minio, Inc. * * 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. */ #include <stdio.h> #include <stdlib.h> #include <erasure-code.h> #include "matrix_decode.h" static int src_in_err (int r, int *src_err_list) { int i; for (i = 0; src_err_list[i] != -1; i++) { if (src_err_list[i] == r) { return 1; } } // false return 0; } /* Generate decode matrix during the decoding phase */ int gf_gen_decode_matrix (int *src_err_list, unsigned char *encode_matrix, unsigned char *decode_matrix, int k, int n, int errs, size_t matrix_size) { int i, j, r, s, l, z; unsigned char *input_matrix = NULL; unsigned char *inverse_matrix = NULL; input_matrix = malloc(k * n); if (!input_matrix) { return -1; } inverse_matrix = malloc(matrix_size); if (!inverse_matrix) { return -1; } for (i = 0, r = 0; i < k; i++, r++) { while (src_in_err(r, src_err_list)) r++; for (j = 0; j < k; j++) { input_matrix[k * i + j] = encode_matrix[k * r + j]; } } // Not all Vandermonde matrix can be inverted if (gf_invert_matrix(input_matrix, inverse_matrix, k) < 0) { return -1; } for (l = 0; l < errs; l++) { if (src_err_list[l] < k) { // decoding matrix elements for data chunks for (j = 0; j < k; j++) { decode_matrix[k * l + j] = inverse_matrix[k * src_err_list[l] + j]; } } else { int s = 0; // decoding matrix element for coding chunks for (i = 0; i < k; i++) { s = 0; for (j = 0; j < k; j++) { s ^= gf_mul(inverse_matrix[j * k + i], encode_matrix[k * src_err_list[l] + j]); } decode_matrix[k * l + i] = s; } } } free(input_matrix); free(inverse_matrix); return 0; }