implemented uncompressed decoding

This commit is contained in:
Enrico Fucile 2014-08-10 20:20:57 +01:00
parent 527f05f2c7
commit 3d94d6dd8d
7 changed files with 338 additions and 189 deletions

View File

@ -17,12 +17,15 @@ meta selectGroupNumber bufr_group_number();
transient subsetNumber=0;
if (compressedData) {
meta numericValues bufr_data_array(offsetSection4,offsetBeforeData,offsetEndSection4,section4Length,
meta numericValues bufr_data_array(offsetSection4,offsetBeforeData,offsetEndSection4,section4Length,
numberOfDataSubsets,subsetNumber,
expandedDescriptors,type,reference,scale,width,
expandedDescriptors,type,unit,reference,scale,width,
stringValues,elementsFXY,compressedData) ;
} else {
assert(0);
meta numericValues bufr_data_array(offsetSection4,offsetBeforeData,offsetEndSection4,section4Length,
numberOfDataSubsets,subsetNumber,
expandedDescriptors,type,unit,reference,scale,width,
stringValues,elementsFXY,compressedData) ;
}
meta unpack unpack_bufr_values(numericValues);

View File

@ -692,7 +692,7 @@ static int decode_elements(grib_accessor* a) {
if (width) {Assert(0);}
if(bitmapIndex[self->bitmapNumber[i]]==-1)
bitmapIndex[self->bitmapNumber[i]]=grib_iarray_get_used_size(bitmap);
bitmapIndex[self->bitmapNumber[i]]=grib_iarray_used_size(bitmap);
grib_iarray_push(index,i);
grib_iarray_push(groupNumberIndex,groupNumber);
groupNumberSection[groupNumber]=section;

View File

@ -33,6 +33,7 @@
MEMBERS= const char* subsetNumberName
MEMBERS= const char* expandedDescriptorsName
MEMBERS= const char* typeName
MEMBERS= const char* unitsName
MEMBERS= const char* referenceName
MEMBERS= const char* scaleName
MEMBERS= const char* widthName
@ -40,7 +41,7 @@
MEMBERS= const char* elementsFXYName
MEMBERS= const char* compressedDataName
MEMBERS= long* expandedDescriptors
MEMBERS= char** type
MEMBERS= int* type
MEMBERS= long* reference
MEMBERS= double* factor
MEMBERS= long* width
@ -94,6 +95,7 @@ typedef struct grib_accessor_bufr_data_array {
const char* subsetNumberName;
const char* expandedDescriptorsName;
const char* typeName;
const char* unitsName;
const char* referenceName;
const char* scaleName;
const char* widthName;
@ -101,7 +103,7 @@ typedef struct grib_accessor_bufr_data_array {
const char* elementsFXYName;
const char* compressedDataName;
long* expandedDescriptors;
char** type;
int* type;
long* reference;
double* factor;
long* width;
@ -216,6 +218,7 @@ static void init(grib_accessor* a,const long v, grib_arguments* params)
self->subsetNumberName = grib_arguments_get_name(a->parent->h,params,n++);
self->expandedDescriptorsName = grib_arguments_get_name(a->parent->h,params,n++);
self->typeName = grib_arguments_get_name(a->parent->h,params,n++);
self->unitsName = grib_arguments_get_name(a->parent->h,params,n++);
self->referenceName = grib_arguments_get_name(a->parent->h,params,n++);
self->scaleName = grib_arguments_get_name(a->parent->h,params,n++);
self->widthName = grib_arguments_get_name(a->parent->h,params,n++);
@ -234,10 +237,6 @@ static void init(grib_accessor* a,const long v, grib_arguments* params)
}
static void self_clear(grib_context* c,grib_accessor_bufr_data_array* self) {
int i=0;
for (i=0;i<self->numberOfDescriptors;i++) {
grib_context_free(c,self->type[i]);
}
grib_context_free(c,self->expandedDescriptors);
grib_context_free(c,self->type);
grib_context_free(c,self->reference);
@ -295,6 +294,8 @@ static int get_descriptors(grib_accessor* a) {
long* scale=0;
double* factor=0;
int i=0;
char** ctype=0;
char** cunits=0;
grib_accessor_bufr_data_array *self =(grib_accessor_bufr_data_array*)a;
grib_accessor* expandedDescriptors=0;
grib_handle* h=a->parent->h;
@ -323,9 +324,31 @@ static int get_descriptors(grib_accessor* a) {
!= GRIB_SUCCESS)
return err;
self->type=grib_context_malloc_clear(c,self->numberOfDescriptors*sizeof(int));
size=self->numberOfDescriptors;
self->type=(char**)grib_context_malloc_clear(c,size*sizeof(char*));
err=grib_get_string_array(h,self->typeName,self->type,&size);
ctype=(char**)grib_context_malloc_clear(c,size*sizeof(char*));
err=grib_get_string_array(h,self->typeName,ctype,&size);
size=self->numberOfDescriptors;
cunits=(char**)grib_context_malloc_clear(c,size*sizeof(char*));
err=grib_get_string_array(h,self->unitsName,cunits,&size);
for (i=0;i<self->numberOfDescriptors;i++) {
if (*(ctype[i]) =='s') {
self->type[i]=BUFR_TYPE_STRING;
} else {
if (strstr(cunits[i],"TABLE") ) {
if (strstr(cunits[i],"FLAG")) {
self->type[i]=BUFR_TYPE_FLAGTABLE;
} else {
self->type[i]=BUFR_TYPE_CODETABLE;
}
} else {
self->type[i]=BUFR_TYPE_DOUBLE;
}
}
}
grib_context_free(c,ctype);
grib_context_free(c,cunits);
size=self->numberOfDescriptors;
self->reference=grib_context_malloc_clear(c,size*sizeof(long));
@ -391,10 +414,16 @@ static grib_darray* decode_double_array(grib_context* c,unsigned char* data,long
int extraWidth,int extraScale,double referenceFactor,int associatedFieldWidth,int localDescriptorWidth) {
grib_darray* ret=NULL;
int j;
long lval;
unsigned long lval;
long localReference;
int width,modifiedWidth,modifiedReference;
double modifiedFactor,dval;
if (self->type[i]==BUFR_TYPE_CODETABLE || self->type[i]==BUFR_TYPE_FLAGTABLE) {
modifiedReference= self->reference[i];
modifiedFactor= self->factor[i];
modifiedWidth= self->width[i];
} else {
modifiedReference= self->reference[i]*referenceFactor;
modifiedFactor= extraScale ? self->factor[i]*grib_power(-extraScale,10) : self->factor[i];
@ -403,9 +432,10 @@ static grib_darray* decode_double_array(grib_context* c,unsigned char* data,long
} else {
modifiedWidth= self->width[i]+extraWidth;
}
}
lval=grib_decode_unsigned_long(data,pos,modifiedWidth);
localReference=lval+modifiedReference;
localReference=(long)lval+modifiedReference;
width=grib_decode_unsigned_long(data,pos,6);
ret=grib_darray_new(c,100,100);
if (width) {
@ -414,7 +444,7 @@ static grib_darray* decode_double_array(grib_context* c,unsigned char* data,long
if (grib_is_all_bits_one(lval,width) && canBeMissing) {
dval=GRIB_MISSING_DOUBLE;
} else {
dval=(lval+localReference)*modifiedFactor;
dval=((long)lval+localReference)*modifiedFactor;
}
grib_darray_push(c,ret,dval);
}
@ -426,41 +456,57 @@ static grib_darray* decode_double_array(grib_context* c,unsigned char* data,long
}
grib_darray_push(c,ret,dval);
}
/*
switch (width) {
case 0:
if (!grib_is_all_bits_one(lval,modifiedWidth) || self->expandedDescriptors[i]==31031) {
dval=localReference*modifiedFactor;
} else dval=GRIB_MISSING_DOUBLE;
grib_darray_push(c,ret,dval);
break;
case 1:
for (j=0;j<self->numberOfDataSubsets;j++) {
lval=grib_decode_unsigned_long(data,pos,width);
if (self->expandedDescriptors[i]!=31031 && lval==1) {
dval=GRIB_MISSING_DOUBLE;
} else {
dval=localReference*modifiedFactor;
}
grib_darray_push(c,ret,dval);
}
break;
default :
for (j=0;j<self->numberOfDataSubsets;j++) {
lval=grib_decode_unsigned_long(data,pos,width);
if (!grib_is_all_bits_one(lval,modifiedWidth)) {
dval=(lval+localReference)*modifiedFactor;
} else {
dval=GRIB_MISSING_DOUBLE;
}
grib_darray_push(c,ret,dval);
}
}
*/
return ret;
}
static char* decode_string_value(grib_context* c,unsigned char* data,long* pos, int i,
grib_accessor_bufr_data_array* self,
int extraWidth,int extraScale,double referenceFactor,int associatedFieldWidth,int localDescriptorWidth) {
char* sval=0;
int modifiedWidth,modifiedReference;
double modifiedFactor;
modifiedWidth= extraWidth ? self->width[i]+extraWidth : self->width[i];
modifiedReference= referenceFactor ? self->reference[i]*referenceFactor : self->reference[i];
modifiedFactor= extraScale ? self->factor[i]*grib_power(-extraScale,10) : self->factor[i];
sval=grib_context_malloc_clear(c,modifiedWidth/8+1);
grib_decode_string(data,pos,modifiedWidth/8,sval);
return sval;
}
static double decode_double_value(grib_context* c,unsigned char* data,long* pos,int i,
grib_accessor_bufr_data_array* self,int canBeMissing,
int extraWidth,int extraScale,double referenceFactor,int associatedFieldWidth,int localDescriptorWidth) {
unsigned long lval;
int modifiedWidth,modifiedReference;
double modifiedFactor,dval;
if (self->type[i]==BUFR_TYPE_CODETABLE || self->type[i]==BUFR_TYPE_FLAGTABLE) {
modifiedReference= self->reference[i];
modifiedFactor= self->factor[i];
modifiedWidth= self->width[i];
} else {
modifiedReference= self->reference[i]*referenceFactor;
modifiedFactor= extraScale ? self->factor[i]*grib_power(-extraScale,10) : self->factor[i];
if (localDescriptorWidth) {
modifiedWidth=localDescriptorWidth;
} else {
modifiedWidth= self->width[i]+extraWidth;
}
}
lval=grib_decode_unsigned_long(data,pos,modifiedWidth);
if (grib_is_all_bits_one(lval,modifiedWidth) && canBeMissing) {
dval=GRIB_MISSING_DOUBLE;
} else {
dval=((long)lval+modifiedReference)*modifiedFactor;
}
return dval;
}
static int can_be_missing(int F,int X,int Y) {
int ret=1;
if (F==0 && X==31 && Y==31) ret=0;
@ -480,8 +526,12 @@ static int decode_elements(grib_accessor* a) {
grib_sarray* sval=0;
grib_darray* dval;
grib_iarray* elementsFXY=0;
long localReference=0,width,pos=0;
long localReference=0,width=0;
long pos=0;
int index,sval_size,ii;
int iss,end,elementIndex;
double cdval;
char* csval;
grib_vdarray* dvalues = NULL;
grib_vsarray* svalues = NULL;
@ -517,15 +567,25 @@ static int decode_elements(grib_accessor* a) {
if (self->elementsFXY) grib_viarray_delete(c,self->elementsFXY);
self->elementsFXY=grib_viarray_new(c,100,100);
end= self->compressedData ? 1 : self->numberOfDataSubsets;
for (iss=0;iss<end;iss++) {
elementsFXY=grib_iarray_new(c,100,100);
if (!self->compressedData) {
dval=grib_darray_new(c,100,100);
sval=grib_sarray_new(c,10,10);
}
for (i=0;i<self->numberOfDescriptors;i++) {
canBeMissing=can_be_missing(F[i],X[i],Y[i]);
elementIndex=grib_iarray_used_size(elementsFXY);
switch(F[i]) {
case 0:
/* Table B element */
grib_iarray_push(elementsFXY,self->expandedDescriptors[i]);
if (*(self->type[i])=='s') {
if (self->type[i]==BUFR_TYPE_STRING) {
/* string */
if (self->compressedData) {
sval=decode_string_array(c,data,&pos,i,self,extraWidth,extraScale,referenceFactor,associatedFieldWidth,localDescriptorWidth);
grib_vsarray_push(c,svalues,sval);
index=grib_vsarray_used_size(svalues)-1;
@ -536,11 +596,23 @@ static int decode_elements(grib_accessor* a) {
grib_darray_push(c,dval,x);
}
grib_vdarray_push(c,dvalues,dval);
} else {
/* numeric */
csval=decode_string_value(c,data,&pos,i,self,extraWidth,extraScale,referenceFactor,associatedFieldWidth,localDescriptorWidth);
grib_sarray_push(c,sval,csval);
index=grib_sarray_used_size(sval);
sval_size=grib_sarray_used_size(sval);
cdval=index*1000+strlen(csval);
grib_darray_push(c,dval,cdval);
}
} else {
/* numeric or codetable or flagtable */
if (self->compressedData) {
dval=decode_double_array(c,data,&pos,i,self,canBeMissing,extraWidth,extraScale,referenceFactor,associatedFieldWidth,localDescriptorWidth);
grib_vdarray_push(c,dvalues,dval);
} else {
cdval=decode_double_value(c,data,&pos,i,self,canBeMissing,extraWidth,extraScale,referenceFactor,associatedFieldWidth,localDescriptorWidth);
grib_darray_push(c,dval,cdval);
}
}
break;
case 1:
@ -548,6 +620,7 @@ static int decode_elements(grib_accessor* a) {
numberOfElementsToRepeat=X[i];
nn=numberOfElementsToRepeat;
i++;
if (self->compressedData) {
localReference=grib_decode_unsigned_long(data,&pos,self->width[i])+self->reference[i];
width=grib_decode_unsigned_long(data,&pos,6);
if (width) {
@ -557,6 +630,18 @@ static int decode_elements(grib_accessor* a) {
numberOfRepetitions=localReference*self->factor[i];
startRepetition=i;
}
} else {
numberOfRepetitions=grib_decode_unsigned_long(data,&pos,self->width[i])+self->reference[i]*self->factor[i];
startRepetition=i;
}
grib_iarray_push(elementsFXY,self->expandedDescriptors[i]);
if (self->compressedData) {
dval=grib_darray_new(c,1,100);
grib_darray_push(c,dval,(double)numberOfRepetitions);
grib_vdarray_push(c,dvalues,dval);
} else {
grib_darray_push(c,dval,(double)numberOfRepetitions);
}
continue;
case 2:
/* Operator */
@ -609,9 +694,13 @@ static int decode_elements(grib_accessor* a) {
case 41:
case 42:
grib_iarray_push(elementsFXY,self->expandedDescriptors[i]);
if (self->compressedData) {
dval=grib_darray_new(c,1,100);
grib_darray_push(c,dval,0);
grib_vdarray_push(c,dvalues,dval);
} else {
grib_darray_push(c,dval,0);
}
break;
default :
grib_context_log(c,GRIB_LOG_ERROR,"unsupported operator %d\n",X[i]);
@ -636,12 +725,17 @@ static int decode_elements(grib_accessor* a) {
}
grib_viarray_push(c,self->elementsFXY,elementsFXY);
if (!self->compressedData) {
grib_vdarray_push(c,dvalues,dval);
grib_vsarray_push(c,svalues,sval);
}
}
if (self->numericValues) {
grib_vdarray_delete_content(c,dvalues);
grib_vdarray_delete(c,dvalues);
grib_vsarray_delete_content(c,svalues);
grib_vsarray_delete(c,svalues);
grib_vdarray_delete_content(c,self->numericValues);
grib_vdarray_delete(c,self->numericValues);
grib_vsarray_delete_content(c,self->stringValues);
grib_vsarray_delete(c,self->stringValues);
}
self->numericValues=dvalues;
self->stringValues=svalues;
@ -660,22 +754,36 @@ static void dump(grib_accessor* a, grib_dumper* dumper)
static int value_count(grib_accessor* a,long* count)
{
int err=0,l;
long numberOfSubsets=0,subsetNumber=0;
long i,subsetNumber=0;
grib_accessor_bufr_data_array *self =(grib_accessor_bufr_data_array*)a;
grib_context* c=a->parent->h->context;
err=decode_elements(a);
if (err) return err;
err=grib_get_long(a->parent->h,self->subsetNumberName,&subsetNumber);
if (err) return err;
if (subsetNumber>self->numberOfDataSubsets) {
err=GRIB_INVALID_KEY_VALUE;
grib_context_log(c,GRIB_LOG_ERROR,"%s=%ld is too big, %s=%ld",self->subsetNumberName,self->numberOfDataSubsetsName);
return err;
}
if (self->compressedData) {
l=grib_vdarray_used_size(self->numericValues);
*count=l;
if (subsetNumber<=0) {
err=grib_get_long(a->parent->h,self->numberOfDataSubsetsName,&numberOfSubsets);
if (err) return err;
*count *= numberOfSubsets;
*count *= self->numberOfDataSubsets;
}
} else {
if (subsetNumber>0) {
*count=grib_iarray_used_size(self->elementsFXY->v[subsetNumber-1]);
} else {
*count=0;
for (i=0;i<self->numberOfDataSubsets;i++)
*count+=grib_iarray_used_size(self->elementsFXY->v[i]);
}
}
return err;
@ -684,9 +792,10 @@ static int value_count(grib_accessor* a,long* count)
static int unpack_double(grib_accessor* a, double* val, size_t *len) {
int err=0,i,k,ii;
long n=0;
size_t l=0;
size_t l=0,elementsInSubset;
long numberOfSubsets=0;
grib_accessor_bufr_data_array *self =(grib_accessor_bufr_data_array*)a;
grib_context* c=a->parent->h->context;
err=decode_elements(a);
if (err) return err;
@ -695,15 +804,20 @@ static int unpack_double(grib_accessor* a, double* val, size_t *len) {
l=grib_vdarray_used_size(self->numericValues);
err=grib_get_long(a->parent->h,self->subsetNumberName,&n);
if (err) return err;
err=grib_get_long(a->parent->h,self->numberOfDataSubsetsName,&numberOfSubsets);
if (err) return err;
if (n>numberOfSubsets) {
err=GRIB_INVALID_KEY_VALUE;
grib_context_log(c,GRIB_LOG_ERROR,"%s=%ld is too big, %s=%ld",self->subsetNumberName,self->numberOfDataSubsetsName);
return err;
}
if (self->compressedData) {
if (n>0) {
for (i=0;i<l;i++) {
val[i]=self->numericValues->v[i]->n > 1 ? self->numericValues->v[i]->v[n-1] : self->numericValues->v[i]->v[0];
}
} else {
err=grib_get_long(a->parent->h,self->numberOfDataSubsetsName,&numberOfSubsets);
if (err) return err;
ii=0;
for (k=0;k<numberOfSubsets;k++) {
for (i=0;i<l;i++) {
@ -711,6 +825,20 @@ static int unpack_double(grib_accessor* a, double* val, size_t *len) {
}
}
}
} else {
if (n>0) {
elementsInSubset=grib_iarray_used_size(self->elementsFXY->v[n]);
for (i=0;i<elementsInSubset;i++) val[i]=self->numericValues->v[n-1]->v[i];
} else {
ii=0;
for (k=0;k<numberOfSubsets;k++) {
elementsInSubset=grib_iarray_used_size(self->elementsFXY->v[k]);
for (i=0;i<elementsInSubset;i++) {
val[ii++]=self->numericValues->v[k]->v[i];
}
}
}
}
return GRIB_SUCCESS;
}

View File

@ -174,7 +174,7 @@ size_t __expand(grib_accessor* a,grib_iarray* unexpanded,grib_iarray* expanded,
grib_context* c=a->parent->h->context;
long us;
if (grib_iarray_get_used_size(unexpanded)==0) return 0;
if (grib_iarray_used_size(unexpanded)==0) return 0;
us=grib_iarray_get(unexpanded,0);

View File

@ -200,6 +200,12 @@ double rint(double x);
#define MAX_SMART_TABLE_COLUMNS 20
#define MAX_CODETABLE_ENTRIES 65536
/* types of BUFR table B elements */
#define BUFR_TYPE_STRING 1
#define BUFR_TYPE_DOUBLE 2
#define BUFR_TYPE_CODETABLE 3
#define BUFR_TYPE_FLAGTABLE 4
/* ACCESSOR COMPARE FLAGS */
#define GRIB_COMPARE_NAMES (1<<0)
#define GRIB_COMPARE_TYPES (1<<1)

View File

@ -177,7 +177,7 @@ void grib_iarray_set(grib_iarray *a, size_t i, long v);
void grib_iarray_delete(grib_iarray *v);
void grib_iarray_delete_array(grib_iarray *v);
long *grib_iarray_get_array(grib_iarray *v);
size_t grib_iarray_get_used_size(grib_iarray *v);
size_t grib_iarray_used_size(grib_iarray *v);
/* grib_vdarray.c */
grib_vdarray *grib_vdarray_new(grib_context *c, size_t size, size_t incsize);
@ -620,8 +620,8 @@ int grib_index_select_double(grib_index *index, const char *skey, double value);
int grib_index_select_string(grib_index *index, const char *skey, char *value);
grib_handle *grib_index_get_handle(grib_field *field, int *err);
grib_handle *codes_index_get_handle(grib_field *field, int message_type, int *err);
int grib_index_dump_file(FILE *fout, const char *filename);
void grib_index_dump(FILE *fout, grib_index *index);
int grib_index_dump_file(FILE* fout, const char* filename);
char *grib_get_field_file(grib_index *index, off_t *offset);
grib_handle *grib_handle_new_from_index(grib_index *index, int *err);
grib_handle *codes_new_from_index(grib_index *index, int message_type, int *err);
@ -697,6 +697,14 @@ void grib_timer_print(grib_timer *t);
void grib_timer_partial_rate(grib_timer *t, double start, long total);
void grib_print_all_timers(void);
void grib_reset_all_timers(void);
grib_timer *grib_get_timer(grib_context *c, const char *name, const char *statname, int elapsed);
int grib_timer_start(grib_timer *t);
int grib_timer_stop(grib_timer *t, long total);
double grib_timer_value(grib_timer *t);
void grib_timer_print(grib_timer *t);
void grib_timer_partial_rate(grib_timer *t, double start, long total);
void grib_print_all_timers(void);
void grib_reset_all_timers(void);
/* grib_ibmfloat.c */
unsigned long grib_ibm_to_long(double x);
@ -715,6 +723,10 @@ double grib_ieeefloat_error(double x);
double grib_long_to_ieee(unsigned long x);
unsigned long grib_ieee_nearest_smaller_to_long(double x);
int grib_nearest_smaller_ieee_float(double a, double *ret);
double grib_ieeefloat_error(double x);
double grib_long_to_ieee(unsigned long x);
int grib_nearest_smaller_ieee_float(double a, double *x);
unsigned long grib_ieee_to_long(double x);
unsigned long grib_ieee64_to_long(double x);
double grib_long_to_ieee64(unsigned long x);
int grib_ieee_decode_array(grib_context *c, unsigned char *buf, size_t nvals, int bytes, double *val);

View File

@ -199,5 +199,5 @@ long* grib_iarray_get_array(grib_iarray* v) {
return vv;
}
size_t grib_iarray_get_used_size(grib_iarray* v) {return v->n;}
size_t grib_iarray_used_size(grib_iarray* v) {return v->n;}