-
Notifications
You must be signed in to change notification settings - Fork 8
/
cm256.h
200 lines (169 loc) · 7.21 KB
/
cm256.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
/*
C++ version:
Copyright (c) 2016 Edouard M. Griffiths. All rights reserved.
Copyright (c) 2015 Christopher A. Taylor. All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
* Neither the name of CM256 nor the names of its contributors may be
used to endorse or promote products derived from this software without
specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef CM256_H
#define CM256_H
#include <assert.h>
#include "gf256.h"
#include "export.h"
class CM256CC_API CM256
{
public:
// Encoder parameters
typedef struct cm256_encoder_params_t {
// Original block count < 256
int OriginalCount;
// Recovery block count < 256
int RecoveryCount;
// Number of bytes per block (all blocks are the same size in bytes)
int BlockBytes;
} cm256_encoder_params;
// Descriptor for data block
typedef struct cm256_block_t {
// Pointer to data received.
void* Block;
// Block index.
// For original data, it will be in the range
// [0..(originalCount-1)] inclusive.
// For recovery data, the first one's Index must be originalCount,
// and it will be in the range
// [originalCount..(originalCount+recoveryCount-1)] inclusive.
unsigned char Index;
// Ignored during encoding, required during decoding.
} cm256_block;
CM256();
~CM256();
bool isInitialized() const { return m_initialized; };
/*
* Cauchy MDS GF(256) encode
*
* This produces a set of recovery blocks that should be transmitted after the
* original data blocks.
*
* It takes in 'originalCount' equal-sized blocks and produces 'recoveryCount'
* equally-sized recovery blocks.
*
* The input 'originals' array allows more natural usage of the library.
* The output recovery blocks are stored end-to-end in 'recoveryBlocks'.
* 'recoveryBlocks' should have recoveryCount * blockBytes bytes available.
*
* Precondition: originalCount + recoveryCount <= 256
*
* When transmitting the data, the block index of the data should be sent,
* and the recovery block index is also needed. The decoder should also
* be provided with the values of originalCount, recoveryCount and blockBytes.
*
* Example wire format:
* [originalCount(1 byte)] [recoveryCount(1 byte)]
* [blockIndex(1 byte)] [blockData(blockBytes bytes)]
*
* Be careful not to mix blocks from different encoders.
*
* It is possible to support variable-length data by including the original
* data length at the front of each message in 2 bytes, such that when it is
* recovered after a loss the data length is available in the block data and
* the remaining bytes of padding can be neglected.
*
* Returns 0 on success, and any other code indicates failure.
*/
int cm256_encode(
cm256_encoder_params params, // Encoder parameters
cm256_block* originals, // Array of pointers to original blocks
void* recoveryBlocks); // Output recovery blocks end-to-end
/*
* Cauchy MDS GF(256) decode
*
* This recovers the original data from the recovery data in the provided
* blocks. There should be 'originalCount' blocks in the provided array.
* Recovery will always be possible if that many blocks are received.
*
* Provide the same values for 'originalCount', 'recoveryCount', and
* 'blockBytes' used by the encoder.
*
* The block Index should be set to the block index of the original data,
* as described in the cm256_block struct comments above.
*
* Recovery blocks will be replaced with original data and the Index
* will be updated to indicate the original block that was recovered.
*
* Returns 0 on success, and any other code indicates failure.
*/
int cm256_decode(
cm256_encoder_params params, // Encoder parameters
cm256_block* blocks); // Array of 'originalCount' blocks as described above
/*
* Commodity functions
*/
// Compute the value to put in the Index member of cm256_block
static inline unsigned char cm256_get_recovery_block_index(cm256_encoder_params params, int recoveryBlockIndex)
{
assert(recoveryBlockIndex >= 0 && recoveryBlockIndex < params.RecoveryCount);
return (unsigned char)(params.OriginalCount + recoveryBlockIndex);
}
static inline unsigned char cm256_get_original_block_index(cm256_encoder_params params, int originalBlockIndex)
{
(void) params;
assert(originalBlockIndex >= 0 && originalBlockIndex < params.OriginalCount);
return (unsigned char)(originalBlockIndex);
}
private:
class CM256CC_API CM256Decoder
{
public:
CM256Decoder(gf256_ctx& gf256Ctx);
~CM256Decoder();
// Encode parameters
cm256_encoder_params Params;
// Recovery blocks
cm256_block* Recovery[256];
int RecoveryCount;
// Original blocks
cm256_block* Original[256];
int OriginalCount;
// Row indices that were erased
uint8_t ErasuresIndices[256];
// Initialize the decoder
bool Initialize(cm256_encoder_params& params, cm256_block* blocks);
// Decode m=1 case
void DecodeM1();
// Decode for m>1 case
void Decode();
// Generate the LU decomposition of the matrix
void GenerateLDUDecomposition(uint8_t* matrix_L, uint8_t* diag_D, uint8_t* matrix_U);
private:
gf256_ctx& m_gf256Ctx;
};
// Encode one block.
// Note: This function does not validate input, use with care.
void cm256_encode_block(
cm256_encoder_params params, // Encoder parameters
cm256_block* originals, // Array of pointers to original blocks
int recoveryBlockIndex, // Return value from cm256_get_recovery_block_index()
void* recoveryBlock); // Output recovery block
gf256_ctx m_gf256Ctx;
bool m_initialized;
};
#endif // CM256_H