ECC-133 able to decode and re-encode synop. Added test to grib_filter

This commit is contained in:
Enrico Fucile 2015-09-04 11:24:09 +01:00
parent a5b8dc423e
commit 6b5411c974
3 changed files with 441 additions and 342 deletions

View File

@ -203,8 +203,8 @@ static void init_class(grib_accessor_class* c)
static int process_elements(grib_accessor* a,int flag);
typedef int (*codec_element_proc) (grib_context* c,grib_accessor_bufr_data_array* self,int subsetIndex, grib_buffer* b,unsigned char* data,long *pos,int i,grib_darray* dval,grib_sarray* sval);
typedef int (*codec_replication_proc) (grib_context* c,grib_accessor_bufr_data_array* self,grib_buffer* buff,unsigned char* data,long *pos,int i,grib_darray* dval,long* numberOfRepetitions);
typedef int (*codec_element_proc) (grib_context* c,grib_accessor_bufr_data_array* self,int subsetIndex, grib_buffer* b,unsigned char* data,long *pos,int i,long elementIndex,grib_darray* dval,grib_sarray* sval);
typedef int (*codec_replication_proc) (grib_context* c,grib_accessor_bufr_data_array* self,int subsetIndex,grib_buffer* buff,unsigned char* data,long *pos,int i,long elementIndex,grib_darray* dval,long* numberOfRepetitions);
static int can_be_missing(int descriptor)
{
@ -347,7 +347,7 @@ static int pack_double(grib_accessor* a, const double* val, size_t *len)
{
grib_accessor_bufr_data_array *self =(grib_accessor_bufr_data_array*)a;
self->do_decode=1;
return GRIB_NOT_IMPLEMENTED;
return process_elements(a,PROCESS_ENCODE);
}
grib_sarray* accessor_bufr_data_array_get_stringValues(grib_accessor* a)
@ -477,12 +477,18 @@ static grib_darray* decode_double_array(grib_context* c,unsigned char* data,long
}
static int encode_string_array(grib_context* c,grib_buffer* buff,long* pos, int i,
grib_accessor_bufr_data_array* self,grib_sarray* stringValues)
grib_accessor_bufr_data_array* self,grib_sarray* stringValues,int start,int n)
{
char* sval=0;
int err=0;
int err=0,end;
int j,modifiedWidth,modifiedReference,width;
double modifiedFactor;
if (n==start && start==0) {
n=stringValues->n;
end=n+1;
} else {
end=start+n+1;
}
modifiedWidth= self->expanded->v[i]->width;
modifiedReference= self->expanded->v[i]->reference;
@ -490,14 +496,14 @@ static int encode_string_array(grib_context* c,grib_buffer* buff,long* pos, int
grib_buffer_set_ulength_bits(c,buff,buff->ulength_bits+modifiedWidth);
grib_encode_string(buff->data,pos,modifiedWidth/8,stringValues->v[0]);
width= stringValues->n > 1 ? modifiedWidth : 0;
width= n > 1 ? modifiedWidth : 0;
grib_buffer_set_ulength_bits(c,buff,buff->ulength_bits+6);
grib_encode_unsigned_longb(buff->data,width/8,pos,6);
if (width) {
grib_buffer_set_ulength_bits(c,buff,buff->ulength_bits+6);
grib_buffer_set_ulength_bits(c,buff,buff->ulength_bits+width*stringValues->n);
for (j=0;j<stringValues->n;j++) {
grib_buffer_set_ulength_bits(c,buff,buff->ulength_bits+width*n);
for (j=start;j<end;j++) {
grib_encode_string(buff->data,pos,width,stringValues->v[j]);
}
}
@ -545,7 +551,7 @@ static int encode_double_array(grib_context* c,grib_buffer* buff,long* pos,int i
int reference=min/modifiedFactor;
localReference=reference-modifiedReference;
localRange = (max/modifiedFactor-localReference-modifiedReference);
localWidth=(long)ceil(log((double)localRange)/log(2.0));
localWidth=ceil(log((double)localRange)/log(2.0));
} else {
localWidth=0;
}
@ -565,7 +571,7 @@ static int encode_double_array(grib_context* c,grib_buffer* buff,long* pos,int i
if (dvalues->v[j]==GRIB_MISSING_DOUBLE) {
grib_set_bits_on(buff->data,pos,localWidth);
} else {
lval=(long)((dvalues->v[j]-min)/modifiedFactor);
lval=round((dvalues->v[j]-min)/modifiedFactor);
grib_encode_unsigned_longb(buff->data,lval,pos,localWidth);
}
}
@ -594,7 +600,7 @@ static int encode_double_value(grib_context* c,grib_buffer* buff,long* pos,int i
} else {
maxAllowed=(grib_power(modifiedWidth,2)+modifiedReference)*modifiedFactor;
if (value>maxAllowed) return GRIB_OUT_OF_RANGE;
lval=(long)(value/modifiedFactor-modifiedReference);
lval=round(value/modifiedFactor)-modifiedReference;
grib_encode_unsigned_longb(buff->data,lval,pos,modifiedWidth);
}
@ -661,7 +667,7 @@ static double decode_double_value(grib_context* c,unsigned char* data,long* pos,
}
static int decode_element(grib_context* c,grib_accessor_bufr_data_array* self,int subsetIndex,
grib_buffer* b,unsigned char* data,long *pos,int i,grib_darray* dval,grib_sarray* sval)
grib_buffer* b,unsigned char* data,long *pos,int i,long elementIndex,grib_darray* dval,grib_sarray* sval)
{
grib_accessor* a=(grib_accessor*)self;
grib_darray* dar=0;
@ -709,52 +715,8 @@ static int decode_element(grib_context* c,grib_accessor_bufr_data_array* self,in
return err;
}
static int encode_new_element(grib_context* c,grib_accessor_bufr_data_array* self,int subsetIndex,
grib_buffer* buff,unsigned char* data,long *pos,int i,grib_darray* dval,grib_sarray* sval)
{
int index=0,ii;
char* csval=0;
unsigned char missingChar=0xFF;
double cdval=GRIB_MISSING_DOUBLE;
int err=0;
size_t slen;
grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data encoding: \tcode=%6.6ld width=%ld pos=%ld ulength=%ld ulength_bits=%ld",
self->expanded->v[i]->code,self->expanded->v[i]->width,(long)*pos,buff->ulength,buff->ulength_bits);
if (self->expanded->v[i]->type==BUFR_DESCRIPTOR_TYPE_STRING) {
/* string */
slen=self->expanded->v[i]->width/8;
csval=grib_context_malloc_clear(c,slen);
for (ii=0;ii<slen;ii++) csval[ii]=missingChar;
grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data encoding: \t %s = %s",
self->expanded->v[i]->shortName,csval);
if (self->compressedData) {
grib_sarray* stringValues=grib_sarray_new(c,1,1);
grib_sarray_push(c,stringValues,csval);
err=encode_string_array(c,buff,pos,i,self,stringValues);
grib_sarray_delete_content(c,stringValues);
grib_sarray_delete(c,stringValues);
} else {
err=encode_string_value(c,buff,pos,i,self,csval);
}
} else {
/* numeric or codetable or flagtable */
grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data encoding: \t %s = %g",
self->expanded->v[i]->shortName,cdval);
if (self->expanded->v[i]->code==31031) cdval=0;
if (self->compressedData) {
grib_darray* doubleValues=grib_darray_new(c,1,1);
grib_darray_push(c,doubleValues,cdval);
err=encode_double_array(c,buff,pos,i,self,doubleValues);
grib_darray_delete(c,doubleValues);
} else {
err=encode_double_value(c,buff,pos,i,self,cdval);
}
}
return err;
}
static int decode_replication(grib_context* c,grib_accessor_bufr_data_array* self,grib_buffer* buff,unsigned char* data,long *pos,int i,grib_darray* dval,long* numberOfRepetitions)
static int decode_replication(grib_context* c,grib_accessor_bufr_data_array* self,int subsetIndex,
grib_buffer* buff,unsigned char* data,long *pos,int i,long elementIndex,grib_darray* dval,long* numberOfRepetitions)
{
int err=0;
int localReference,width;
@ -790,7 +752,53 @@ static int decode_replication(grib_context* c,grib_accessor_bufr_data_array* sel
return err;
}
static int encode_new_replication(grib_context* c,grib_accessor_bufr_data_array* self,grib_buffer* buff,unsigned char* data,long *pos,int i,grib_darray* dval,long* numberOfRepetitions)
static int encode_new_element(grib_context* c,grib_accessor_bufr_data_array* self,int subsetIndex,
grib_buffer* buff,unsigned char* data,long *pos,int i,long elementIndex,grib_darray* dval,grib_sarray* sval)
{
int index=0,ii;
char* csval=0;
unsigned char missingChar=0xFF;
double cdval=GRIB_MISSING_DOUBLE;
int err=0;
size_t slen;
grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data encoding: \tcode=%6.6ld width=%ld pos=%ld ulength=%ld ulength_bits=%ld",
self->expanded->v[i]->code,self->expanded->v[i]->width,(long)*pos,buff->ulength,buff->ulength_bits);
if (self->expanded->v[i]->type==BUFR_DESCRIPTOR_TYPE_STRING) {
/* string */
slen=self->expanded->v[i]->width/8;
csval=grib_context_malloc_clear(c,slen);
for (ii=0;ii<slen;ii++) csval[ii]=missingChar;
grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data encoding: \t %s = %s",
self->expanded->v[i]->shortName,csval);
if (self->compressedData) {
grib_sarray* stringValues=grib_sarray_new(c,1,1);
grib_sarray_push(c,stringValues,csval);
err=encode_string_array(c,buff,pos,i,self,stringValues,0,0);
grib_sarray_delete_content(c,stringValues);
grib_sarray_delete(c,stringValues);
} else {
err=encode_string_value(c,buff,pos,i,self,csval);
}
} else {
/* numeric or codetable or flagtable */
grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data encoding: \t %s = %g",
self->expanded->v[i]->shortName,cdval);
if (self->expanded->v[i]->code==31031) cdval=0;
if (self->compressedData) {
grib_darray* doubleValues=grib_darray_new(c,1,1);
grib_darray_push(c,doubleValues,cdval);
err=encode_double_array(c,buff,pos,i,self,doubleValues);
grib_darray_delete(c,doubleValues);
} else {
err=encode_double_value(c,buff,pos,i,self,cdval);
}
}
return err;
}
static int encode_new_replication(grib_context* c,grib_accessor_bufr_data_array* self,int subsetIndex,
grib_buffer* buff,unsigned char* data,long *pos,int i,long elementIndex,grib_darray* dval,long* numberOfRepetitions)
{
int err=0;
/*new data -> only 1 repetition*/
@ -812,6 +820,54 @@ static int encode_new_replication(grib_context* c,grib_accessor_bufr_data_array*
return err;
}
static int encode_element(grib_context* c,grib_accessor_bufr_data_array* self,int subsetIndex,
grib_buffer* buff,unsigned char* data,long *pos,int i,long elementIndex,grib_darray* dval,grib_sarray* sval)
{
int index=0,ii,idx;
char* csval=0;
unsigned char missingChar=0xFF;
double cdval=GRIB_MISSING_DOUBLE;
int err=0;
size_t slen;
grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data encoding: \tcode=%6.6ld width=%ld pos=%ld ulength=%ld ulength_bits=%ld",
self->expanded->v[i]->code,self->expanded->v[i]->width,(long)*pos,buff->ulength,buff->ulength_bits);
if (self->expanded->v[i]->type==BUFR_DESCRIPTOR_TYPE_STRING) {
/* string */
/* grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data encoding: \t %s = %s",
self->expanded->v[i]->shortName,csval); */
if (self->compressedData) {
idx=(int)self->numericValues->v[elementIndex]->v[0]/1000-1;
err=encode_string_array(c,buff,pos,i,self,self->stringValues,idx,self->numberOfSubsets);
} else {
idx=(int)self->numericValues->v[self->subsetNumber]->v[elementIndex]/1000-1;
err=encode_string_value(c,buff,pos,i,self,self->stringValues->v[idx]);
}
} else {
/* numeric or codetable or flagtable */
/* grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data encoding: \t %s = %g",
self->expanded->v[i]->shortName,cdval); */
if (self->compressedData) {
err=encode_double_array(c,buff,pos,i,self,self->numericValues->v[elementIndex]);
} else {
err=encode_double_value(c,buff,pos,i,self,self->numericValues->v[subsetIndex]->v[elementIndex]);
}
}
return err;
}
static int encode_replication(grib_context* c,grib_accessor_bufr_data_array* self,int subsetIndex,
grib_buffer* buff,unsigned char* data,long *pos,int i,long elementIndex,grib_darray* dval,long* numberOfRepetitions)
{
bufr_descriptor** descriptors=0;
if (self->compressedData)
Assert(grib_darray_used_size(self->numericValues->v[elementIndex])==1);
return encode_element(c,self,subsetIndex,buff,data,pos,i,elementIndex,dval,0);
}
static int build_bitmap(grib_accessor_bufr_data_array *self,unsigned char* data,long* pos,grib_iarray* elementsDescriptorsIndex,int iBitmapOperator)
{
int bitmapSize=0,iDelayedReplication=0;
@ -1232,6 +1288,8 @@ static int create_keys(grib_accessor* a)
sectionUp=self->dataKeys;
accessor_constant_set_type(gaGroup,GRIB_TYPE_LONG);
accessor_constant_set_dval(gaGroup,groupNumber);
self->dataKeys->block->first=0;
self->dataKeys->block->last=0;
grib_push_accessor(gaGroup,self->dataKeys->block);
indexOfGroupNumber=0;
@ -1416,7 +1474,6 @@ static int process_elements(grib_accessor* a,int flag)
case PROCESS_NEW_DATA:
buffer=grib_create_growable_buffer(c);
decoding=0;
self->do_decode=1;
do_clean=1;
self->do_decode=1;
pos=0;
@ -1424,7 +1481,13 @@ static int process_elements(grib_accessor* a,int flag)
codec_replication=&encode_new_replication;
break;
case PROCESS_ENCODE:
return GRIB_NOT_IMPLEMENTED;
buffer=grib_create_growable_buffer(c);
decoding=0;
do_clean=0;
self->do_decode=0;
pos=0;
codec_element=&encode_element;
codec_replication=&encode_replication;
break;
default :
return GRIB_NOT_IMPLEMENTED;
@ -1442,35 +1505,45 @@ static int process_elements(grib_accessor* a,int flag)
grib_sarray_delete_content(c,self->stringValues);
grib_sarray_delete(c,self->stringValues);
}
if (flag!=PROCESS_ENCODE) {
self->numericValues=grib_vdarray_new(c,100,100);
self->stringValues=grib_sarray_new(c,10,10);
if (self->elementsDescriptorsIndex) grib_viarray_delete(c,self->elementsDescriptorsIndex);
self->elementsDescriptorsIndex=grib_viarray_new(c,100,100);
}
end= self->compressedData ? 1 : self->numberOfSubsets;
numberOfDescriptors=grib_bufr_descriptors_array_used_size(self->expanded);
for (iss=0;iss<end;iss++) {
elementsDescriptorsIndex=grib_iarray_new(c,100,100);
icount=1;
grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data processing: subsetNumber=%ld", iss+1);
if (decoding==1 && !self->compressedData) {
if (flag!=PROCESS_ENCODE) {
elementsDescriptorsIndex=grib_iarray_new(c,100,100);
if (!self->compressedData) {
dval=grib_darray_new(c,100,100);
/* sval=grib_sarray_new(c,10,10); */
}
} else {
elementsDescriptorsIndex=self->elementsDescriptorsIndex->v[iss];
}
elementIndex=0;
for (i=0;i<numberOfDescriptors;i++) {
grib_context_log(c, GRIB_LOG_DEBUG,"BUFR data processing: elementNumber=%ld code=%6.6ld", icount++,descriptors[i]->code);
elementIndex=grib_iarray_used_size(elementsDescriptorsIndex);
switch(descriptors[i]->F) {
case 0:
/* Table B element */
grib_iarray_push(elementsDescriptorsIndex,i);
if (flag!=PROCESS_ENCODE) grib_iarray_push(elementsDescriptorsIndex,i);
if (descriptors[i]->code==31031 && !is_bitmap_start_defined(self))
self->bitmapStart=grib_iarray_used_size(elementsDescriptorsIndex)-1;
err=codec_element(c,self,iss,buffer,data,&pos,i,dval,sval);
err=codec_element(c,self,iss,buffer,data,&pos,i,elementIndex,dval,sval);
if (err) return err;
elementIndex++;
break;
case 1:
/* Delayed replication */
@ -1481,12 +1554,13 @@ static int process_elements(grib_accessor* a,int flag)
n[inr]=numberOfElementsToRepeat[inr];
i++;
err=codec_replication(c,self,buffer,data,&pos,i,dval,&(numberOfRepetitions[inr]));
err=codec_replication(c,self,iss,buffer,data,&pos,i,elementIndex,dval,&(numberOfRepetitions[inr]));
if (err) return err;
startRepetition[inr]=i;
nn[inr]=numberOfRepetitions[inr];
grib_iarray_push(elementsDescriptorsIndex,i);
if (flag!=PROCESS_ENCODE) grib_iarray_push(elementsDescriptorsIndex,i);
elementIndex++;
if (numberOfRepetitions[inr]==0) {
i+=numberOfElementsToRepeat[inr];
if (inr>0) n[inr-1]-=numberOfElementsToRepeat[inr]+2;
@ -1501,9 +1575,10 @@ static int process_elements(grib_accessor* a,int flag)
case 5:
descriptors[i]->width=descriptors[i]->Y*8;
descriptors[i]->type=BUFR_DESCRIPTOR_TYPE_STRING;
grib_iarray_push(elementsDescriptorsIndex,i);
err=codec_element(c,self,iss,buffer,data,&pos,i,dval,sval);
err=codec_element(c,self,iss,buffer,data,&pos,i,elementIndex,dval,sval);
if (err) return err;
if (flag!=PROCESS_ENCODE) grib_iarray_push(elementsDescriptorsIndex,i);
elementIndex++;
break;
case 22:
case 26:
@ -1518,8 +1593,9 @@ static int process_elements(grib_accessor* a,int flag)
case 40:
case 41:
case 42:
grib_iarray_push(elementsDescriptorsIndex,i);
if (flag!=PROCESS_ENCODE) grib_iarray_push(elementsDescriptorsIndex,i);
if (decoding) push_zero_element(self,dval);
elementIndex++;
break;
case 23:
/* substituted values marker operator */
@ -1529,13 +1605,15 @@ static int process_elements(grib_accessor* a,int flag)
/*replaced/retained values marker operator*/
if (descriptors[i]->Y==255) {
index=get_next_bitmap_descriptor_index(self,elementsDescriptorsIndex,dval);
err=codec_element(c,self,iss,buffer,data,&pos,index,dval,sval);
err=codec_element(c,self,iss,buffer,data,&pos,index,elementIndex,dval,sval);
if (err) return err;
/* self->expanded->v[index] */
grib_iarray_push(elementsDescriptorsIndex,i);
if (flag!=PROCESS_ENCODE) grib_iarray_push(elementsDescriptorsIndex,i);
elementIndex++;
} else {
grib_iarray_push(elementsDescriptorsIndex,i);
if (flag!=PROCESS_ENCODE) grib_iarray_push(elementsDescriptorsIndex,i);
if (decoding) push_zero_element(self,dval);
elementIndex++;
}
break;
case 25:
@ -1543,9 +1621,10 @@ static int process_elements(grib_accessor* a,int flag)
break;
case 35:
/* cancel bitmap */
grib_iarray_push(elementsDescriptorsIndex,i);
if (flag!=PROCESS_ENCODE) grib_iarray_push(elementsDescriptorsIndex,i);
if (decoding) push_zero_element(self,dval);
if (descriptors[i]->Y==0) cancel_bitmap(self);
elementIndex++;
break;
case 36:
/* bitmap */
@ -1555,11 +1634,12 @@ static int process_elements(grib_accessor* a,int flag)
break;
case 37:
/* reuse defined bitmap */
grib_iarray_push(elementsDescriptorsIndex,i);
if (flag!=PROCESS_ENCODE) grib_iarray_push(elementsDescriptorsIndex,i);
if (decoding) push_zero_element(self,dval);
if (descriptors[i]->Y==0) restart_bitmap(self);
/* cancel reuse */
else cancel_bitmap(self);
elementIndex++;
break;
default :
grib_context_log(c,GRIB_LOG_ERROR,"unsupported operator %d\n",descriptors[i]->X);
@ -1569,9 +1649,10 @@ static int process_elements(grib_accessor* a,int flag)
case 9:
/* associated field */
if (descriptors[i]->X==99 && descriptors[i]->Y==999) {
grib_iarray_push(elementsDescriptorsIndex,i);
err=codec_element(c,self,iss,buffer,data,&pos,i,dval,sval);
err=codec_element(c,self,iss,buffer,data,&pos,i,elementIndex,dval,sval);
if (err) return err;
if (flag!=PROCESS_ENCODE) grib_iarray_push(elementsDescriptorsIndex,i);
elementIndex++;
} else {
return GRIB_INTERNAL_ERROR;
}
@ -1613,7 +1694,7 @@ static int process_elements(grib_accessor* a,int flag)
}
}
grib_viarray_push(c,self->elementsDescriptorsIndex,elementsDescriptorsIndex);
if (flag!=PROCESS_ENCODE) grib_viarray_push(c,self->elementsDescriptorsIndex,elementsDescriptorsIndex);
if (decoding && !self->compressedData) {
grib_vdarray_push(c,self->numericValues,dval);
}

View File

@ -713,8 +713,26 @@ ${tools_dir}/bufr_filter $fRulesReady $f 2>> $fLog 1>> $testScript
./$testScript
#rm -f new_*bufr
#rm -f $testScript
rm -f new_*bufr
rm -f $testScript
#-----------------------------------------------------------
# Test: packing
#-----------------------------------------------------------
cat > $fRules <<EOF
set unpack=1;
set pack=1;
write;
EOF
f="syno_1.bufr"
echo "Test: packing " >> $fLog
echo "file: $f" >> $fLog
${tools_dir}/bufr_filter -o ${f}.out $fRules $f 2>> $fLog 1>> $fLog
${tools_dir}/bufr_compare ${f}.out $f
rm -f ${f}.out
#-----------------------------------------------------------
# Test: get string

View File

@ -1001,7 +1001,7 @@ static int compare_all_dump_keys(grib_handle* h1,grib_handle* h2,grib_runtime_op
{
grib_accessor* xa=grib_keys_iterator_get_accessor(iter);
name=grib_keys_iterator_get_name(iter);
printf("----- comparing %s\n",name);
/* printf("----- comparing %s\n",name); */
if (blacklisted(name)) continue;
if (xa==NULL || ( xa->flags & GRIB_ACCESSOR_FLAG_DUMP )==0 ) continue;