2007-01-23 00:17:31 +00:00
/**
* copyright 2007 Allmydata Inc .
* author : Zooko O ' Whielacronx
* based on fecmodule . c by the Mnet Project
*/
# include <Python.h>
# include <structmember.h>
# include "fec.h"
static PyObject * py_fec_error ;
static PyObject * py_raise_fec_error ( const char * format , . . . ) ;
static char fec__doc__ [ ] = " \
FEC - Forward Error Correction \ n \
" ;
static PyObject *
py_raise_fec_error ( const char * format , . . . ) {
char exceptionMsg [ 1024 ] ;
va_list ap ;
va_start ( ap , format ) ;
vsnprintf ( exceptionMsg , 1024 , format , ap ) ;
va_end ( ap ) ;
exceptionMsg [ 1023 ] = ' \0 ' ;
PyErr_SetString ( py_fec_error , exceptionMsg ) ;
return NULL ;
}
static char Encoder__doc__ [ ] = " \
Hold static encoder state ( an in - memory table for matrix multiplication ) , and k and m parameters , and provide { encode ( ) } method . \ n \
" ;
typedef struct {
PyObject_HEAD
/* expose these */
int kk ;
int mm ;
/* internal */
fec_t * fec_matrix ;
} Encoder ;
static PyObject *
Encoder_new ( PyTypeObject * type , PyObject * args , PyObject * kwds ) {
Encoder * self ;
self = ( Encoder * ) type - > tp_alloc ( type , 0 ) ;
if ( self ! = NULL ) {
self - > kk = 0 ;
self - > mm = 0 ;
self - > fec_matrix = NULL ;
}
return ( PyObject * ) self ;
}
static char Encoder_init__doc__ [ ] = " \
@ param k : the number of packets required for reconstruction \ n \
@ param m : the number of packets generated \ n \
" ;
static int
Encoder_init ( Encoder * self , PyObject * args , PyObject * kwdict ) {
static char * kwlist [ ] = {
" k " ,
" m " ,
NULL
} ;
if ( ! PyArg_ParseTupleAndKeywords ( args , kwdict , " ii " , kwlist , & self - > kk , & self - > mm ) )
return - 1 ;
self - > fec_matrix = fec_new ( self - > kk , self - > mm ) ;
if ( self - > fec_matrix = = NULL ) {
py_raise_fec_error ( fec_error ) ; /* xyz */
return - 1 ;
}
return 0 ;
}
static char Encoder_encode__doc__ [ ] = " \
Encode data into m packets . \
@ param inshares : a sequence of k buffers of data to encode - - these are the k primary shares , i . e . the input data split into k pieces ( for best performance , make it a tuple instead of a list ) \ n \
2007-01-24 22:34:02 +00:00
@ param desired_shares_ids optional sorted sequence of shareids indicating which shares to produce and return ; If None , all m shares will be returned ( in order ) . ( For best performance , make it a tuple instead of a list . ) \ n \
2007-01-23 00:17:31 +00:00
@ returns : a list of buffers containing the requested shares \ n \
" ;
static PyObject *
Encoder_encode ( Encoder * self , PyObject * args ) {
PyObject * inshares ;
PyObject * desired_shares_ids = NULL ; /* The shareids of the shares that should be returned. */
PyObject * result = NULL ;
if ( ! PyArg_ParseTuple ( args , " O!|O! " , & PyList_Type , & inshares , & PyList_Type , & desired_shares_ids ) )
return NULL ;
gf * check_shares_produced [ self - > mm - self - > kk ] ; /* This is an upper bound -- we will actually use only num_check_shares_produced of these elements (see below). */
PyObject * pystrs_produced [ self - > mm - self - > kk ] ; /* This is an upper bound -- we will actually use only num_check_shares_produced of these elements (see below). */
unsigned num_check_shares_produced = 0 ; /* The first num_check_shares_produced elements of the check_shares_produced array and of the pystrs_produced array will be used. */
const gf * incshares [ self - > kk ] ;
unsigned char num_desired_shares ;
PyObject * fast_desired_shares_ids = NULL ;
PyObject * * fast_desired_shares_ids_items ;
unsigned char c_desired_shares_ids [ self - > mm ] ;
unsigned i ;
unsigned prev_desired_id = 256 ; /* impossible value */
if ( desired_shares_ids ) {
fast_desired_shares_ids = PySequence_Fast ( desired_shares_ids , " Second argument (optional) was not a sequence. " ) ;
num_desired_shares = PySequence_Fast_GET_SIZE ( fast_desired_shares_ids ) ;
fast_desired_shares_ids_items = PySequence_Fast_ITEMS ( fast_desired_shares_ids ) ;
for ( i = 0 ; i < num_desired_shares ; i + + ) {
if ( ! PyInt_Check ( fast_desired_shares_ids_items [ i ] ) )
goto err ;
c_desired_shares_ids [ i ] = PyInt_AsLong ( fast_desired_shares_ids_items [ i ] ) ;
if ( prev_desired_id ! = 256 & & prev_desired_id > = c_desired_shares_ids [ i ] ) {
py_raise_fec_error ( " Precondition violation: first argument is required to be in order -- each requested shareid in the sequence must be a higher number than the previous one. current requested shareid: %u, previous requested shareid: %u \n " , c_desired_shares_ids [ i ] , prev_desired_id ) ;
goto err ;
}
prev_desired_id = c_desired_shares_ids [ i ] ;
if ( c_desired_shares_ids [ i ] > = self - > kk )
num_check_shares_produced + + ;
}
} else {
num_desired_shares = self - > mm ;
for ( i = 0 ; i < num_desired_shares ; i + + )
c_desired_shares_ids [ i ] = i ;
num_check_shares_produced = self - > mm - self - > kk ;
}
for ( i = 0 ; i < num_check_shares_produced ; i + + )
pystrs_produced [ i ] = NULL ;
PyObject * fastinshares = PySequence_Fast ( inshares , " First argument was not a sequence. " ) ;
if ( ! fastinshares )
goto err ;
if ( PySequence_Fast_GET_SIZE ( fastinshares ) ! = self - > kk ) {
py_raise_fec_error ( " Precondition violation: Wrong length -- first argument is required to contain exactly k shares. len(first): %d, k: %d " , PySequence_Fast_GET_SIZE ( fastinshares ) , self - > kk ) ;
goto err ;
}
/* Construct a C array of gf*'s of the input data. */
PyObject * * fastinsharesitems = PySequence_Fast_ITEMS ( fastinshares ) ;
if ( ! fastinsharesitems )
goto err ;
int sz , oldsz = 0 ;
for ( i = 0 ; i < self - > kk ; i + + ) {
if ( ! PyObject_CheckReadBuffer ( fastinsharesitems [ i ] ) ) {
py_raise_fec_error ( " Precondition violation: %u'th item is required to offer the single-segment read character buffer protocol, but it does not. \n " , i ) ;
goto err ;
}
if ( PyObject_AsCharBuffer ( fastinsharesitems [ i ] , & ( incshares [ i ] ) , & sz ) )
goto err ;
if ( oldsz ! = 0 & & oldsz ! = sz ) {
py_raise_fec_error ( " Precondition violation: Input shares are required to be all the same length. oldsz: %Zu, sz: %Zu \n " , oldsz , sz ) ;
goto err ;
}
oldsz = sz ;
}
/* Allocate space for all of the check shares. */
unsigned check_share_index = 0 ; /* index into the check_shares_produced and (parallel) pystrs_produced arrays */
for ( i = 0 ; i < num_desired_shares ; i + + ) {
if ( c_desired_shares_ids [ i ] > = self - > kk ) {
pystrs_produced [ check_share_index ] = PyString_FromStringAndSize ( NULL , sz ) ;
if ( pystrs_produced [ check_share_index ] = = NULL )
goto err ;
check_shares_produced [ check_share_index ] = PyString_AsString ( pystrs_produced [ check_share_index ] ) ;
if ( check_shares_produced [ check_share_index ] = = NULL )
goto err ;
check_share_index + + ;
}
}
assert ( check_share_index = = num_check_shares_produced ) ;
/* Encode any check shares that are needed. */
fec_encode_all ( self - > fec_matrix , incshares , check_shares_produced , c_desired_shares_ids , num_desired_shares , sz ) ;
/* Wrap all requested shares up into a Python list of Python strings. */
result = PyList_New ( num_desired_shares ) ;
if ( result = = NULL )
goto err ;
check_share_index = 0 ;
for ( i = 0 ; i < num_desired_shares ; i + + ) {
if ( c_desired_shares_ids [ i ] < self - > kk ) {
Py_INCREF ( fastinsharesitems [ c_desired_shares_ids [ i ] ] ) ;
if ( PyList_SetItem ( result , i , fastinsharesitems [ c_desired_shares_ids [ i ] ] ) = = - 1 ) {
Py_DECREF ( fastinsharesitems [ c_desired_shares_ids [ i ] ] ) ;
goto err ;
}
} else {
if ( PyList_SetItem ( result , i , pystrs_produced [ check_share_index ] ) = = - 1 )
goto err ;
pystrs_produced [ check_share_index ] = NULL ;
check_share_index + + ;
}
}
goto cleanup ;
err :
for ( i = 0 ; i < num_check_shares_produced ; i + + )
Py_XDECREF ( pystrs_produced [ i ] ) ;
Py_XDECREF ( result ) ; result = NULL ;
cleanup :
Py_XDECREF ( fastinshares ) ; fastinshares = NULL ;
Py_XDECREF ( fast_desired_shares_ids ) ; fast_desired_shares_ids = NULL ;
return result ;
}
static void
Encoder_dealloc ( Encoder * self ) {
fec_free ( self - > fec_matrix ) ;
self - > ob_type - > tp_free ( ( PyObject * ) self ) ;
}
static PyMethodDef Encoder_methods [ ] = {
{ " encode " , ( PyCFunction ) Encoder_encode , METH_VARARGS , Encoder_encode__doc__ } ,
{ NULL } ,
} ;
static PyMemberDef Encoder_members [ ] = {
{ " k " , T_INT , offsetof ( Encoder , kk ) , READONLY , " k " } ,
{ " m " , T_INT , offsetof ( Encoder , mm ) , READONLY , " m " } ,
{ NULL } /* Sentinel */
} ;
static PyTypeObject Encoder_type = {
PyObject_HEAD_INIT ( NULL )
0 , /*ob_size*/
" fec.Encoder " , /*tp_name*/
sizeof ( Encoder ) , /*tp_basicsize*/
0 , /*tp_itemsize*/
( destructor ) Encoder_dealloc , /*tp_dealloc*/
0 , /*tp_print*/
0 , /*tp_getattr*/
0 , /*tp_setattr*/
0 , /*tp_compare*/
0 , /*tp_repr*/
0 , /*tp_as_number*/
0 , /*tp_as_sequence*/
0 , /*tp_as_mapping*/
0 , /*tp_hash */
0 , /*tp_call*/
0 , /*tp_str*/
0 , /*tp_getattro*/
0 , /*tp_setattro*/
0 , /*tp_as_buffer*/
Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE , /*tp_flags*/
Encoder__doc__ , /* tp_doc */
0 , /* tp_traverse */
0 , /* tp_clear */
0 , /* tp_richcompare */
0 , /* tp_weaklistoffset */
0 , /* tp_iter */
0 , /* tp_iternext */
Encoder_methods , /* tp_methods */
Encoder_members , /* tp_members */
0 , /* tp_getset */
0 , /* tp_base */
0 , /* tp_dict */
0 , /* tp_descr_get */
0 , /* tp_descr_set */
0 , /* tp_dictoffset */
( initproc ) Encoder_init , /* tp_init */
0 , /* tp_alloc */
Encoder_new , /* tp_new */
} ;
static char Decoder__doc__ [ ] = " \
Hold static decoder state ( an in - memory table for matrix multiplication ) , and k and m parameters , and provide { decode ( ) } method . \ n \
" ;
typedef struct {
PyObject_HEAD
/* expose these */
int kk ;
int mm ;
/* internal */
fec_t * fec_matrix ;
} Decoder ;
static PyObject *
Decoder_new ( PyTypeObject * type , PyObject * args , PyObject * kwds ) {
Decoder * self ;
self = ( Decoder * ) type - > tp_alloc ( type , 0 ) ;
if ( self ! = NULL ) {
self - > kk = 0 ;
self - > mm = 0 ;
self - > fec_matrix = NULL ;
}
return ( PyObject * ) self ;
}
static char Decoder_init__doc__ [ ] = " \
@ param k : the number of packets required for reconstruction \ n \
@ param m : the number of packets generated \ n \
" ;
static int
Decoder_init ( Encoder * self , PyObject * args , PyObject * kwdict ) {
static char * kwlist [ ] = {
" k " ,
" m " ,
NULL
} ;
if ( ! PyArg_ParseTupleAndKeywords ( args , kwdict , " ii " , kwlist , & self - > kk , & self - > mm ) )
return - 1 ;
self - > fec_matrix = fec_new ( self - > kk , self - > mm ) ;
if ( self - > fec_matrix = = NULL ) {
py_raise_fec_error ( fec_error ) ; /* xyz */
return - 1 ;
}
return 0 ;
}
2007-01-24 22:20:53 +00:00
# define SWAP(a,b,t) {t tmp; tmp=a; a=b; b=tmp;}
2007-01-23 00:17:31 +00:00
static char Decoder_decode__doc__ [ ] = " \
Decode a list shares into a list of segments . \ n \
@ param shares a sequence of buffers containing share data ( for best performance , make it a tuple instead of a list ) \ n \
@ param sharenums a sequence of integers of the sharenumber for each share in shares ( for best performance , make it a tuple instead of a list ) \ n \
\ n \
@ return a list of strings containing the segment data ( i . e . ' ' . join ( retval ) yields a string containing the decoded data ) \ n \
" ;
static PyObject *
Decoder_decode ( Decoder * self , PyObject * args ) {
PyObject * restrict shares ;
PyObject * restrict sharenums ;
PyObject * result = NULL ;
if ( ! PyArg_ParseTuple ( args , " O!O! " , & PyList_Type , & shares , & PyList_Type , & sharenums ) )
return NULL ;
const gf * restrict cshares [ self - > kk ] ;
2007-01-24 22:17:51 +00:00
unsigned char csharenums [ self - > kk ] ;
2007-01-23 00:17:31 +00:00
gf * restrict recoveredcstrs [ self - > kk ] ; /* self->kk is actually an upper bound -- we probably won't need all of this space. */
PyObject * restrict recoveredpystrs [ self - > kk ] ; /* self->kk is actually an upper bound -- we probably won't need all of this space. */
unsigned i ;
for ( i = 0 ; i < self - > kk ; i + + )
recoveredpystrs [ i ] = NULL ;
PyObject * restrict fastshares = PySequence_Fast ( shares , " First argument was not a sequence. " ) ;
if ( ! fastshares )
goto err ;
PyObject * restrict fastsharenums = PySequence_Fast ( sharenums , " Second argument was not a sequence. " ) ;
if ( ! fastsharenums )
goto err ;
if ( PySequence_Fast_GET_SIZE ( fastshares ) ! = self - > kk ) {
py_raise_fec_error ( " Precondition violation: Wrong length -- first argument is required to contain exactly k shares. len(first): %d, k: %d " , PySequence_Fast_GET_SIZE ( fastshares ) , self - > kk ) ;
goto err ;
}
if ( PySequence_Fast_GET_SIZE ( fastsharenums ) ! = self - > kk ) {
py_raise_fec_error ( " Precondition violation: Wrong length -- sharenums is required to contain exactly k shares. len(sharenums): %d, k: %d " , PySequence_Fast_GET_SIZE ( fastsharenums ) , self - > kk ) ;
goto err ;
}
/* Construct a C array of gf*'s of the data and another of C ints of the sharenums. */
unsigned needtorecover = 0 ;
PyObject * * fastsharenumsitems = PySequence_Fast_ITEMS ( fastsharenums ) ;
if ( ! fastsharenumsitems )
goto err ;
PyObject * * fastsharesitems = PySequence_Fast_ITEMS ( fastshares ) ;
if ( ! fastsharesitems )
goto err ;
int sz , oldsz = 0 ;
for ( i = 0 ; i < self - > kk ; i + + ) {
if ( ! PyInt_Check ( fastsharenumsitems [ i ] ) )
goto err ;
2007-01-24 22:17:51 +00:00
long tmpl = PyInt_AsLong ( fastsharenumsitems [ i ] ) ;
if ( tmpl < 0 | | tmpl > = UCHAR_MAX ) {
py_raise_fec_error ( " Precondition violation: Share ids can't be less than zero or greater than 255. %ld \n " , tmpl ) ;
goto err ;
}
csharenums [ i ] = ( unsigned char ) tmpl ;
2007-01-23 00:17:31 +00:00
if ( csharenums [ i ] > = self - > kk )
needtorecover + = 1 ;
if ( ! PyObject_CheckReadBuffer ( fastsharesitems [ i ] ) )
goto err ;
if ( PyObject_AsCharBuffer ( fastsharesitems [ i ] , & ( cshares [ i ] ) , & sz ) )
goto err ;
if ( oldsz ! = 0 & & oldsz ! = sz ) {
py_raise_fec_error ( " Precondition violation: Input shares are required to be all the same length. oldsz: %Zu, sz: %Zu \n " , oldsz , sz ) ;
goto err ;
}
oldsz = sz ;
}
2007-01-24 22:20:53 +00:00
/* move src packets into position */
for ( i = 0 ; i < self - > kk ; ) {
if ( csharenums [ i ] > = self - > kk | | csharenums [ i ] = = i )
i + + ;
else {
/* put pkt in the right position. */
unsigned char c = csharenums [ i ] ;
SWAP ( csharenums [ i ] , csharenums [ c ] , int ) ;
SWAP ( cshares [ i ] , cshares [ c ] , gf * ) ;
SWAP ( fastsharesitems [ i ] , fastsharesitems [ c ] , PyObject * ) ;
}
}
2007-01-23 00:17:31 +00:00
/* Allocate space for all of the recovered shares. */
for ( i = 0 ; i < needtorecover ; i + + ) {
recoveredpystrs [ i ] = PyString_FromStringAndSize ( NULL , sz ) ;
if ( recoveredpystrs [ i ] = = NULL )
goto err ;
recoveredcstrs [ i ] = PyString_AsString ( recoveredpystrs [ i ] ) ;
if ( recoveredcstrs [ i ] = = NULL )
goto err ;
}
/* Decode any recovered shares that are needed. */
fec_decode_all ( self - > fec_matrix , cshares , recoveredcstrs , csharenums , sz ) ;
/* Wrap up both original primary shares and decoded shares into a Python list of Python strings. */
unsigned nextrecoveredix = 0 ;
result = PyList_New ( self - > kk ) ;
if ( result = = NULL )
goto err ;
for ( i = 0 ; i < self - > kk ; i + + ) {
if ( csharenums [ i ] = = i ) {
/* Original primary share. */
Py_INCREF ( fastsharesitems [ i ] ) ;
if ( PyList_SetItem ( result , i , fastsharesitems [ i ] ) = = - 1 ) {
Py_DECREF ( fastsharesitems [ i ] ) ;
goto err ;
}
} else {
/* Recovered share. */
if ( PyList_SetItem ( result , i , recoveredpystrs [ nextrecoveredix ] ) = = - 1 )
goto err ;
recoveredpystrs [ nextrecoveredix ] = NULL ;
nextrecoveredix + + ;
}
}
goto cleanup ;
err :
for ( i = 0 ; i < self - > kk ; i + + )
Py_XDECREF ( recoveredpystrs [ i ] ) ;
Py_XDECREF ( result ) ; result = NULL ;
cleanup :
Py_XDECREF ( fastshares ) ; fastshares = NULL ;
Py_XDECREF ( fastsharenums ) ; fastsharenums = NULL ;
return result ;
}
static void
Decoder_dealloc ( Decoder * self ) {
fec_free ( self - > fec_matrix ) ;
self - > ob_type - > tp_free ( ( PyObject * ) self ) ;
}
static PyMethodDef Decoder_methods [ ] = {
{ " decode " , ( PyCFunction ) Decoder_decode , METH_VARARGS , Decoder_decode__doc__ } ,
{ NULL } ,
} ;
static PyMemberDef Decoder_members [ ] = {
{ " k " , T_INT , offsetof ( Encoder , kk ) , READONLY , " k " } ,
{ " m " , T_INT , offsetof ( Encoder , mm ) , READONLY , " m " } ,
{ NULL } /* Sentinel */
} ;
static PyTypeObject Decoder_type = {
PyObject_HEAD_INIT ( NULL )
0 , /*ob_size*/
" fec.Decoder " , /*tp_name*/
sizeof ( Decoder ) , /*tp_basicsize*/
0 , /*tp_itemsize*/
( destructor ) Decoder_dealloc , /*tp_dealloc*/
0 , /*tp_print*/
0 , /*tp_getattr*/
0 , /*tp_setattr*/
0 , /*tp_compare*/
0 , /*tp_repr*/
0 , /*tp_as_number*/
0 , /*tp_as_sequence*/
0 , /*tp_as_mapping*/
0 , /*tp_hash */
0 , /*tp_call*/
0 , /*tp_str*/
0 , /*tp_getattro*/
0 , /*tp_setattro*/
0 , /*tp_as_buffer*/
Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE , /*tp_flags*/
Decoder__doc__ , /* tp_doc */
0 , /* tp_traverse */
0 , /* tp_clear */
0 , /* tp_richcompare */
0 , /* tp_weaklistoffset */
0 , /* tp_iter */
0 , /* tp_iternext */
Decoder_methods , /* tp_methods */
Decoder_members , /* tp_members */
0 , /* tp_getset */
0 , /* tp_base */
0 , /* tp_dict */
0 , /* tp_descr_get */
0 , /* tp_descr_set */
0 , /* tp_dictoffset */
( initproc ) Decoder_init , /* tp_init */
0 , /* tp_alloc */
Decoder_new , /* tp_new */
} ;
static PyMethodDef fec_methods [ ] = {
{ NULL }
} ;
# ifndef PyMODINIT_FUNC /* declarations for DLL import/export */
# define PyMODINIT_FUNC void
# endif
PyMODINIT_FUNC
initfec ( void ) {
PyObject * module ;
PyObject * module_dict ;
if ( PyType_Ready ( & Encoder_type ) < 0 )
return ;
if ( PyType_Ready ( & Decoder_type ) < 0 )
return ;
module = Py_InitModule3 ( " fec " , fec_methods , fec__doc__ ) ;
if ( module = = NULL )
return ;
Py_INCREF ( & Encoder_type ) ;
Py_INCREF ( & Decoder_type ) ;
PyModule_AddObject ( module , " Encoder " , ( PyObject * ) & Encoder_type ) ;
PyModule_AddObject ( module , " Decoder " , ( PyObject * ) & Decoder_type ) ;
module_dict = PyModule_GetDict ( module ) ;
py_fec_error = PyErr_NewException ( " fec.Error " , NULL , NULL ) ;
PyDict_SetItemString ( module_dict , " Error " , py_fec_error ) ;
}