GRIB-488: GRIB2: step calculation for typeOfTimeIncrement=1 (Part 2)

This commit is contained in:
Shahram Najm 2014-02-20 22:25:22 +00:00
parent dc8be541c0
commit 9895229135
2 changed files with 370 additions and 180 deletions

View File

@ -36,7 +36,6 @@ alias totalNumberOfDataValuesMissingInStatisticalProcess=numberOfMissingInStatis
statisticalProcessesList list(numberOfTimeRange)
{
# Statistical process used to calculate the processed field from the field at each time increment during the time range
codetable[1] typeOfStatisticalProcessing ('4.10.table',masterDir,localDir) : edition_specific;
@ -59,7 +58,9 @@ unsigned[4] timeIncrement=0 ;
alias timeIncrementBetweenSuccessiveFields=timeIncrement;
}
if (numberOfTimeRange==1) {
# See GRIB-488. We only support maximum of 2 time ranges
if (numberOfTimeRange == 1 || numberOfTimeRange == 2) {
concept stepTypeInternal {
"instant" = {typeOfStatisticalProcessing=255;}
"avg" = {typeOfStatisticalProcessing=0;typeOfTimeIncrement=2;}
@ -95,7 +96,9 @@ if (numberOfTimeRange==1) {
secondOfEndOfOverallTimeInterval,
indicatorOfUnitForTimeRange,
lengthOfTimeRange
lengthOfTimeRange,
typeOfTimeIncrement,
numberOfTimeRange
) : dump,no_copy;
meta stepRange g2step_range(startStep,endStep): dump;

View File

@ -36,6 +36,8 @@
MEMBERS = const char* coded_unit
MEMBERS = const char* coded_time_range
MEMBERS = const char* typeOfTimeIncrement
MEMBERS = const char* numberOfTimeRange
END_CLASS_DEF
@ -78,6 +80,8 @@ typedef struct grib_accessor_g2end_step {
const char* second_of_end_of_interval;
const char* coded_unit;
const char* coded_time_range;
const char* typeOfTimeIncrement;
const char* numberOfTimeRange;
} grib_accessor_g2end_step;
extern grib_accessor_class* grib_accessor_class_long;
@ -182,6 +186,8 @@ static void init(grib_accessor* a,const long l, grib_arguments* c)
self->coded_unit = grib_arguments_get_name(a->parent->h,c,n++);
self->coded_time_range = grib_arguments_get_name(a->parent->h,c,n++);
self->typeOfTimeIncrement = grib_arguments_get_name(a->parent->h,c,n++);
self->numberOfTimeRange = grib_arguments_get_name(a->parent->h,c,n++);
}
static void dump(grib_accessor* a, grib_dumper* dumper)
@ -225,6 +231,170 @@ static int u2s[] = {
1800 /* (15) 30 minutes */
};
/* See GRIB-488 */
static int is_special_expver(grib_handle* h)
{
if (grib_is_defined(h, "experimentVersionNumber")) {
int ret = 0;
char strExpVer[50]={0,};
size_t slen=50;
ret = grib_get_string(h, "experimentVersionNumber", strExpVer, &slen);
if (ret == GRIB_SUCCESS && !strcmp(strExpVer, "1605")) {
return 1; /* Special case of expVer 1605! */
}
}
return 0;
}
static int convert_time_range(
grib_handle* h,
long stepUnits, /* unit */
long indicatorOfUnitForTimeRange, /* coded_unit */
long* lengthOfTimeRange /* coded_time_range */
)
{
Assert(lengthOfTimeRange != NULL);
if (indicatorOfUnitForTimeRange != stepUnits) {
long u2sf_step_unit;
long coded_time_range_sec = (*lengthOfTimeRange)*u2s2[indicatorOfUnitForTimeRange];
if (coded_time_range_sec < 0) {
long u2sf;
int factor = 60;
if (u2s2[indicatorOfUnitForTimeRange] % factor)
return GRIB_DECODING_ERROR;
if (u2s[stepUnits] % factor)
return GRIB_DECODING_ERROR;
u2sf = u2s2[indicatorOfUnitForTimeRange]/factor;
coded_time_range_sec = (*lengthOfTimeRange)*u2sf;
u2sf_step_unit = u2s[stepUnits]/factor;
} else {
u2sf_step_unit = u2s[stepUnits];
}
if (coded_time_range_sec % u2sf_step_unit != 0) {
grib_context_log(h->context,GRIB_LOG_ERROR,"unable to convert endStep in stepUnits");
return GRIB_WRONG_STEP_UNIT;
}
*lengthOfTimeRange = coded_time_range_sec / u2sf_step_unit;
}
return GRIB_SUCCESS;
}
static int unpack_one_time_range(grib_accessor* a, long* val, size_t *len)
{
grib_accessor_g2end_step* self = (grib_accessor_g2end_step*)a;
int err = 0;
long start_step;
long unit;
long coded_unit;
long coded_time_range, typeOfTimeIncrement;
int add_time_range = 1; /* whether we add lengthOfTimeRange */
grib_handle* h = a->parent->h;
if((err = grib_get_long_internal(h,self->start_step,&start_step))) return err;
if((err = grib_get_long_internal(h,self->unit,&unit))) return err;
if((err = grib_get_long_internal(h,self->coded_unit,&coded_unit))) return err;
if((err = grib_get_long_internal(h,self->coded_time_range, &coded_time_range))) return err;
if((err = grib_get_long_internal(h,self->typeOfTimeIncrement, &typeOfTimeIncrement))) return err;
err = convert_time_range(h, unit, coded_unit, &coded_time_range);
if (err != GRIB_SUCCESS) return err;
if (typeOfTimeIncrement == 1) {
/* See GRIB-488 */
/* Note: For this case, lengthOfTimeRange is not related to step and should not be used to calculate step */
add_time_range = 0;
if (is_special_expver(h)) {
add_time_range = 1;
}
}
if (add_time_range) {
*val = start_step + coded_time_range;
} else {
*val = start_step;
}
return GRIB_SUCCESS;
}
#define MAX_NUM_TIME_RANGES 16 /* maximum number of time range specifications */
static int unpack_multiple_time_ranges(grib_accessor* a, long* val, size_t *len)
{
grib_accessor_g2end_step* self = (grib_accessor_g2end_step*)a;
int i = 0, err = 0;
grib_handle* h = a->parent->h;
long numberOfTimeRange = 0, unit = 0, start_step = 0;
size_t count = 0;
long arr_typeOfTimeIncrement[MAX_NUM_TIME_RANGES] = {0,};
long arr_coded_unit[MAX_NUM_TIME_RANGES] = {0,};
long arr_coded_time_range[MAX_NUM_TIME_RANGES] = {0,};
if((err = grib_get_long_internal(h,self->start_step,&start_step))) return err;
if((err = grib_get_long_internal(h,self->unit,&unit))) return err;
if((err = grib_get_long_internal(h,self->numberOfTimeRange, &numberOfTimeRange))) return err;
if (numberOfTimeRange > MAX_NUM_TIME_RANGES) {
grib_context_log(h->context, GRIB_LOG_ERROR, "Too many time range specifications!");
return GRIB_DECODING_ERROR;
}
count = numberOfTimeRange;
/* Get the arrays for the N time ranges */
if ((err = grib_get_long_array(h, self->typeOfTimeIncrement, arr_typeOfTimeIncrement, &count))) return err;
if ((err = grib_get_long_array(h, self->coded_unit, arr_coded_unit, &count))) return err;
if ((err = grib_get_long_array(h, self->coded_time_range, arr_coded_time_range, &count))) return err;
/* Look in the array of typeOfTimeIncrements for first entry whose typeOfTimeIncrement == 2 */
for(i=0; i<count; i++) {
if (arr_typeOfTimeIncrement[i] == 2) {
/* Found the required time range. Get the other two keys from it */
long the_coded_unit = arr_coded_unit[i];
long the_coded_time_range = arr_coded_time_range[i];
err = convert_time_range(h, unit, the_coded_unit, &the_coded_time_range);
if (err != GRIB_SUCCESS) return err;
*val = start_step + the_coded_time_range;
return GRIB_SUCCESS;
}
}
grib_context_log(h->context, GRIB_LOG_ERROR,
"Cannot calculate endStep. No time range specification with typeOfTimeIncrement = 2");
return GRIB_DECODING_ERROR;
}
static int unpack_long(grib_accessor* a, long* val, size_t *len)
{
grib_accessor_g2end_step* self = (grib_accessor_g2end_step*)a;
int err = 0;
long start_step;
long numberOfTimeRange;
grib_handle* h=a->parent->h;
if((err = grib_get_long_internal(h,self->start_step,&start_step))) return err;
/* point in time */
if (self->year == NULL) {
*val=start_step;
return 0;
}
if((err = grib_get_long_internal(h,self->numberOfTimeRange, &numberOfTimeRange))) return err;
Assert(numberOfTimeRange == 1 || numberOfTimeRange == 2);
if (numberOfTimeRange == 1) {
return unpack_one_time_range(a,val,len);
} else {
return unpack_multiple_time_ranges(a,val,len);
}
return GRIB_SUCCESS;
}
#if 0
static int unpack_long(grib_accessor* a, long* val, size_t *len)
{
grib_accessor_g2end_step* self = (grib_accessor_g2end_step*)a;
@ -232,10 +402,11 @@ static int unpack_long(grib_accessor* a, long* val, size_t *len)
long start_step;
long unit;
long coded_unit;
long coded_time_range;
long coded_time_range, typeOfTimeIncrement, numberOfTimeRange;
long coded_time_range_sec=0;
int factor;
long u2sf,u2sf_step_unit;
int add_time_range = 1; /* whether we add lengthOfTimeRange */
grib_handle* h=a->parent->h;
@ -250,9 +421,16 @@ static int unpack_long(grib_accessor* a, long* val, size_t *len)
if((err = grib_get_long_internal(h,self->unit,&unit))) return err;
if((err = grib_get_long_internal(h,self->coded_unit,&coded_unit))) return err;
if((err = grib_get_long_internal(h,self->coded_time_range,
&coded_time_range))) return err;
if((err = grib_get_long_internal(h,self->coded_time_range, &coded_time_range))) return err;
if((err = grib_get_long_internal(h,self->typeOfTimeIncrement, &typeOfTimeIncrement))) return err;
if((err = grib_get_long_internal(h,self->numberOfTimeRange, &numberOfTimeRange))) return err;
Assert(numberOfTimeRange == 1 || numberOfTimeRange == 2);
err = convert_time_range(h, unit, coded_unit, &coded_time_range);
if (err != GRIB_SUCCESS) return err;
#if 0
if (coded_unit!=unit) {
coded_time_range_sec=coded_time_range*u2s2[coded_unit];
if (coded_time_range_sec<0) {
@ -271,11 +449,24 @@ static int unpack_long(grib_accessor* a, long* val, size_t *len)
}
coded_time_range = coded_time_range_sec / u2sf_step_unit;
}
#endif
if (typeOfTimeIncrement == 1) {
/* See GRIB-488 */
/* Note: For this case, lengthOfTimeRange is not related to step and should not be used to calculate step */
add_time_range = 0;
if (is_special_expver(h)) {
add_time_range = 1;
}
}
if (add_time_range) {
*val = start_step + coded_time_range;
} else {
*val = start_step;
}
return GRIB_SUCCESS;
}
#endif
static int pack_long(grib_accessor* a, const long* val, size_t *len)
{
@ -300,7 +491,7 @@ static int pack_long(grib_accessor* a, const long* val, size_t *len)
long minute_of_end_of_interval = 0;
long second_of_end_of_interval = 0;
long coded_time_range,time_range;
long coded_time_range,time_range, typeOfTimeIncrement;
double dend, dstep;
@ -319,8 +510,8 @@ static int pack_long(grib_accessor* a, const long* val, size_t *len)
if((err = grib_get_long_internal(h,self->minute,&minute))) return err;
if((err = grib_get_long_internal(h,self->second,&second))) return err;
if((err = grib_get_long_internal(h,self->start_step,&start_step)))
return err;
if((err = grib_get_long_internal(h,self->start_step,&start_step))) return err;
if((err = grib_get_long_internal(h,self->typeOfTimeIncrement, &typeOfTimeIncrement))) return err;
time_range = *val-start_step;
@ -341,29 +532,25 @@ static int pack_long(grib_accessor* a, const long* val, size_t *len)
&minute_of_end_of_interval,&second_of_end_of_interval);
if (err!=GRIB_SUCCESS) return err;
if((err = grib_set_long_internal(a->parent->h,self->year_of_end_of_interval,
year_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->month_of_end_of_interval,
month_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->day_of_end_of_interval,
day_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->hour_of_end_of_interval,
hour_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->minute_of_end_of_interval,
minute_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->second_of_end_of_interval,
second_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->year_of_end_of_interval, year_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->month_of_end_of_interval, month_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->day_of_end_of_interval, day_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->hour_of_end_of_interval, hour_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->minute_of_end_of_interval, minute_of_end_of_interval))) return err;
if((err = grib_set_long_internal(a->parent->h,self->second_of_end_of_interval, second_of_end_of_interval))) return err;
if (time_range*u2s[unit]%u2s2[coded_unit]) {
coded_unit=unit;
if((err = grib_set_long_internal(a->parent->h,self->coded_unit,
coded_unit))) return err;
if((err = grib_set_long_internal(a->parent->h,self->coded_unit, coded_unit))) return err;
coded_time_range=time_range;
} else
coded_time_range=(time_range*u2s[unit])/u2s2[coded_unit];
if((err = grib_set_long_internal(a->parent->h,self->coded_time_range,
coded_time_range))) return err;
if (typeOfTimeIncrement != 1) {
/* 1 means "Successive times processed have same forecast time, start time of forecast is incremented" */
/* Note: For this case, length of timeRange is not related to step and so should NOT be used to calculate step */
if((err = grib_set_long_internal(a->parent->h,self->coded_time_range, coded_time_range))) return err;
}
return GRIB_SUCCESS;
}