mirror of
https://github.com/tahoe-lafs/tahoe-lafs.git
synced 2025-01-23 21:08:22 +00:00
185 lines
8.3 KiB
Plaintext
185 lines
8.3 KiB
Plaintext
* Intro and Licence
|
|
|
|
This package implements an "erasure code", or "forward error correction code".
|
|
It is offered under the GNU General Public License v2 or (at your option) any
|
|
later version. This package also comes with the added permission that, in the
|
|
case that you are obligated to release a derived work under this licence (as
|
|
per section 2.b of the GPL), you may delay the fulfillment of this obligation
|
|
for up to 12 months.
|
|
|
|
The most widely known example of an erasure code is the RAID-5 algorithm which
|
|
makes it so that in the event of the loss of any one hard drive, the stored
|
|
data can be completely recovered. The algorithm in the pyfec package has a
|
|
similar effect, but instead of recovering from the loss of only a single
|
|
element, it can be parameterized to choose in advance the number of elements
|
|
whose loss it can tolerate.
|
|
|
|
This package is largely based on the old "fec" library by Luigi Rizzo et al.,
|
|
which is a mature and optimized implementation of erasure coding. The pyfec
|
|
package makes several changes from the original "fec" package, including
|
|
addition of the Python API, refactoring of the C API to be faster (for the way
|
|
that I use it, at least), and a few clean-ups and micro-optimizations of the
|
|
core code itself.
|
|
|
|
|
|
* Community
|
|
|
|
The source is currently available via darcs on the web with the command:
|
|
|
|
darcs get http://www.allmydata.com/source/pyfec
|
|
|
|
More information on darcs is available at http://darcs.net
|
|
|
|
Please join the pyfec mailing list and submit patches:
|
|
|
|
<https://postman.allmydata.com/cgi-bin/mailman/listinfo/pyfec>
|
|
|
|
|
|
* Overview
|
|
|
|
This package performs two operations, encoding and decoding. Encoding takes
|
|
some input data and expands its size by producing extra "check blocks", also
|
|
called "secondary blocks". Decoding takes some data -- any combination of
|
|
blocks of the original data (called "primary blocks") and "secondary blocks",
|
|
and produces the original data.
|
|
|
|
The encoding is parameterized by two integers, k and m. m is the total number
|
|
of blocks produced, and k is how many of those blocks are necessary to
|
|
reconstruct the original data. m is required to be at least 1 and at most 256,
|
|
and k is required to be at least 1 and at most m.
|
|
|
|
(Note that when k == m then there is no point in doing erasure coding -- it
|
|
degenerates to the equivalent of the Unix "split" utility which simply splits
|
|
the input into successive segments. Similarly, when k == 1 it degenerates to
|
|
the equivalent of the unix "cp" utility -- each block is a complete copy of the
|
|
input data.)
|
|
|
|
Note that each "primary block" is a segment of the original data, so its size
|
|
is 1/k'th of the size of original data, and each "secondary block" is of the
|
|
same size, so the total space used by all the blocks is m/k times the size of
|
|
the original data (plus some padding to fill out the last primary block to be
|
|
the same size as all the others).
|
|
|
|
The decoding step requires as input k of the blocks which were produced by the
|
|
encoding step. The decoding step produces as output the data that was earlier
|
|
input to the encoding step.
|
|
|
|
|
|
* API
|
|
|
|
Each block is associated with "blocknum". The blocknum of each primary block is
|
|
its index (starting from zero), so the 0'th block is the first primary block,
|
|
which is the first few bytes of the file, the 1'st block is the next primary
|
|
block, which is the next few bytes of the file, and so on. The last primary
|
|
block has blocknum k-1. The blocknum of each secondary block is an arbitrary
|
|
integer between k and 256 inclusive. (When using the Python API, if you don't
|
|
specify which blocknums you want for your secondary blocks when invoking
|
|
encode(), then it will by default provide the blocks with ids from k to m-1
|
|
inclusive.)
|
|
|
|
** C API
|
|
|
|
fec_encode() takes as input an array of k pointers, where each pointer points
|
|
to a memory buffer containing the input data (i.e., the i'th buffer contains
|
|
the i'th primary block). There is also a second parameter which is an array of
|
|
the blocknums of the secondary blocks which are to be produced. (Each element
|
|
in that array is required to be the blocknum of a secondary block, i.e. it is
|
|
required to be >= k and < m.)
|
|
|
|
The output from fec_encode() is the requested set of secondary blocks which are
|
|
written into output buffers provided by the caller.
|
|
|
|
fec_decode() takes as input an array of k pointers, where each pointer points
|
|
to a buffer containing a block. There is also a separate input parameter which
|
|
is an array of blocknums, indicating the blocknum of each of the blocks which is
|
|
being passed in.
|
|
|
|
The output from fec_decode() is the set of primary blocks which were missing
|
|
from the input and had to be reconstructed. These reconstructed blocks are
|
|
written into putput buffers provided by the caller.
|
|
|
|
** Python API
|
|
|
|
encode() and decode() take as input a sequence of k buffers, where a "sequence"
|
|
is any object that implements the Python sequence protocol (such as a list or
|
|
tuple) and a "buffer" is any object that implements the Python buffer protocol
|
|
(such as a string or array). The contents that are required to be present in
|
|
these buffers are the same as for the C API.
|
|
|
|
encode() also takes a list of desired blocknums. Unlike the C API, the Python
|
|
API accepts blocknums of primary blocks as well as secondary blocks in its list
|
|
of desired blocknums. encode() returns a list of buffer objects which contain
|
|
the blocks requested. For each requested block which is a primary block, the
|
|
resulting list contains a reference to the apppropriate primary block from the
|
|
input list. For each requested block which is a secondary block, the list
|
|
contains a newly created string object containing that block.
|
|
|
|
decode() also takes a list of integers indicating the blocknums of the blocks
|
|
being passed int. decode() returns a list of buffer objects which contain all
|
|
of the primary blocks of the original data (in order). For each primary block
|
|
which was present in the input list, then the result list simply contains a
|
|
reference to the object that was passed in the input list. For each primary
|
|
block which was not present in the input, the result list contains a newly
|
|
created string object containing that primary block.
|
|
|
|
Beware of a "gotcha" that can result from the combination of mutable data and
|
|
the fact that the Python API returns references to inputs when possible.
|
|
|
|
Returning references to its inputs is efficient since it avoids making an
|
|
unnecessary copy of the data, but if the object which was passed as input is
|
|
mutable and if that object is mutated after the call to pyfec returns, then the
|
|
result from pyfec -- which is just a reference to that same object -- will also
|
|
be mutated. This subtlety is the price you pay for avoiding data copying. If
|
|
you don't want to have to worry about this then you can simply use immutable
|
|
objects (e.g. Python strings) to hold the data that you pass to pyfec.
|
|
|
|
|
|
* Utilities
|
|
|
|
See also the filefec.py module which has a utility function for efficiently
|
|
reading a file and encoding it piece by piece.
|
|
|
|
|
|
* Dependencies
|
|
|
|
A C compiler is required. For the Python API, we have tested it with Python
|
|
v2.4 and v2.5.
|
|
|
|
|
|
* Performance Measurements
|
|
|
|
On Peter's fancy Intel Mac laptop (2.16 GHz Core Duo), it encoded from a file
|
|
at about 6.2 million bytes per second.
|
|
|
|
On my even fancier Intel Mac laptop (2.33 GHz Core Duo), it encoded from a file
|
|
at about 6.8 million bytes per second.
|
|
|
|
On my old PowerPC G4 867 MHz Mac laptop, it encoded from a file at about 1.3
|
|
million bytes per second.
|
|
|
|
On my Athlon 64 2.4 GHz workstation (running Linux), it encoded from a file at
|
|
about 4.9 million bytes per second and decoded at about 5.8 million bytes per
|
|
second.
|
|
|
|
|
|
* Acknowledgements
|
|
|
|
Thanks to the author of the original fec lib, Luigi Rizzo, and the folks that
|
|
contributed to it: Phil Karn, Robert Morelos-Zaragoza, Hari Thirumoorthy, and
|
|
Dan Rubenstein. Thanks to the Mnet hackers who wrote an earlier Python
|
|
wrapper, especially Myers Carpenter and Hauke Johannknecht. Thanks to Brian
|
|
Warner for help with the API, documentation, debugging, and unit tests.
|
|
Thanks to the creators of GCC (starting with Richard M. Stallman) and
|
|
Valgrind (starting with Julian Seward) for a pair of excellent tools. Thanks
|
|
to my coworkers at Allmydata -- http://allmydata.com -- Fabrice Grinda, Peter
|
|
Secor, Rob Kinninmont, Brian Warner, Zandr Milewski, Justin Boreta, Mark
|
|
Meras for sponsoring this work and releasing it under a Free Software
|
|
licence.
|
|
|
|
|
|
Enjoy!
|
|
|
|
Zooko Wilcox-O'Hearn
|
|
2007-08-01
|
|
Boulder, Colorado
|