2015-07-21 13:24:02 +00:00
|
|
|
/*
|
2018-01-02 11:31:02 +00:00
|
|
|
* Copyright 2005-2018 ECMWF.
|
2015-07-21 13:24:02 +00:00
|
|
|
*
|
|
|
|
* This software is licensed under the terms of the Apache Licence Version 2.0
|
|
|
|
* which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
|
|
|
|
*
|
|
|
|
* In applying this licence, ECMWF does not waive the privileges and immunities granted to it by
|
|
|
|
* virtue of its status as an intergovernmental organisation nor does it submit to any jurisdiction.
|
|
|
|
*/
|
2016-10-10 18:14:34 +00:00
|
|
|
|
|
|
|
#include "grib_api_internal.h"
|
2015-07-21 13:24:02 +00:00
|
|
|
#include "eccodes.h"
|
|
|
|
#include <assert.h>
|
2018-06-06 14:50:47 +00:00
|
|
|
#include <float.h>
|
2015-07-21 13:24:02 +00:00
|
|
|
|
2015-09-23 14:30:30 +00:00
|
|
|
#define STR_EQUAL(s1, s2) (strcmp((s1), (s2)) == 0)
|
|
|
|
|
2017-09-01 16:36:16 +00:00
|
|
|
static int get_packing_type_code(const char* packingType)
|
2016-10-10 18:14:34 +00:00
|
|
|
{
|
|
|
|
int result = GRIB_UTIL_PACKING_TYPE_GRID_SIMPLE;
|
|
|
|
if (packingType==NULL)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
if (STR_EQUAL(packingType, "grid_jpeg"))
|
|
|
|
result = GRIB_UTIL_PACKING_TYPE_JPEG;
|
|
|
|
else if (STR_EQUAL(packingType, "grid_simple"))
|
|
|
|
result = GRIB_UTIL_PACKING_TYPE_GRID_SIMPLE;
|
|
|
|
else if (STR_EQUAL(packingType, "grid_second_order"))
|
|
|
|
result = GRIB_UTIL_PACKING_TYPE_GRID_SECOND_ORDER;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2017-09-01 16:36:16 +00:00
|
|
|
static void test_reduced_gg(int remove_local_def, int edition, const char* packingType,
|
2016-10-10 18:14:34 +00:00
|
|
|
const char* input_filename, const char* output_filename)
|
2015-09-23 14:30:30 +00:00
|
|
|
{
|
|
|
|
/* based on copy_spec_from_ksec */
|
|
|
|
int err = 0;
|
|
|
|
size_t slen = 32, inlen = 0, outlen = 0;
|
2016-08-31 16:38:31 +00:00
|
|
|
size_t i=0, size=0;
|
2015-09-23 14:30:30 +00:00
|
|
|
int set_spec_flags=0;
|
|
|
|
double* values = NULL;
|
|
|
|
FILE* in = NULL;
|
|
|
|
FILE* out = NULL;
|
|
|
|
const void* buffer = NULL;
|
|
|
|
char gridType[128] = {0,};
|
|
|
|
|
|
|
|
codes_handle *handle = 0;
|
|
|
|
codes_handle *finalh = 0;
|
|
|
|
grib_util_grid_spec spec={0,};
|
|
|
|
grib_util_packing_spec packing_spec={0,};
|
|
|
|
|
2018-12-10 17:23:06 +00:00
|
|
|
assert(input_filename);
|
2015-09-23 14:30:30 +00:00
|
|
|
in = fopen(input_filename,"r"); assert(in);
|
|
|
|
handle = grib_handle_new_from_file(0,in,&err); assert(handle);
|
|
|
|
|
|
|
|
CODES_CHECK(grib_get_string(handle, "gridType", gridType, &slen),0);
|
|
|
|
if (!STR_EQUAL(gridType, "reduced_gg")) {
|
|
|
|
grib_handle_delete(handle);
|
|
|
|
return;
|
|
|
|
}
|
2018-12-10 17:23:06 +00:00
|
|
|
assert(output_filename);
|
2015-09-23 14:30:30 +00:00
|
|
|
out = fopen(output_filename,"w"); assert(out);
|
|
|
|
|
|
|
|
CODES_CHECK(grib_get_size(handle,"values",&inlen), 0);
|
|
|
|
values = (double*)malloc(sizeof(double)*inlen);
|
|
|
|
CODES_CHECK(grib_get_double_array(handle, "values", values,&inlen), 0);
|
2016-08-31 16:38:31 +00:00
|
|
|
for(i=0; i<inlen; ++i) {
|
|
|
|
values[i] *= 1.10;
|
|
|
|
}
|
2015-09-23 14:30:30 +00:00
|
|
|
|
|
|
|
spec.grid_type = GRIB_UTIL_GRID_SPEC_REDUCED_GG;
|
2016-08-31 16:38:31 +00:00
|
|
|
spec.N = 32; /* hardcoded for now */
|
2015-09-23 14:30:30 +00:00
|
|
|
spec.Nj = 2 * spec.N;
|
|
|
|
outlen = inlen;
|
|
|
|
spec.iDirectionIncrementInDegrees = 1.5;
|
|
|
|
spec.jDirectionIncrementInDegrees = 1.5;
|
2018-02-05 16:05:31 +00:00
|
|
|
spec.latitudeOfFirstGridPointInDegrees = 87.8637991;
|
2015-09-23 14:30:30 +00:00
|
|
|
spec.longitudeOfFirstGridPointInDegrees = 0.0;
|
2016-08-31 16:38:31 +00:00
|
|
|
spec.latitudeOfLastGridPointInDegrees = -spec.latitudeOfFirstGridPointInDegrees;
|
|
|
|
spec.longitudeOfLastGridPointInDegrees = 357.187500;
|
2015-09-23 14:30:30 +00:00
|
|
|
spec.bitmapPresent = 0;
|
|
|
|
|
2016-10-10 18:14:34 +00:00
|
|
|
/*packing_spec.packing_type = GRIB_UTIL_PACKING_TYPE_GRID_SECOND_ORDER;*/
|
|
|
|
packing_spec.packing_type = get_packing_type_code(packingType);
|
2015-09-23 14:30:30 +00:00
|
|
|
packing_spec.bitsPerValue = 24;
|
|
|
|
packing_spec.accuracy=GRIB_UTIL_ACCURACY_USE_PROVIDED_BITS_PER_VALUES;
|
|
|
|
packing_spec.packing=GRIB_UTIL_PACKING_USE_PROVIDED;
|
2018-02-05 16:05:31 +00:00
|
|
|
|
|
|
|
/*Extra settings
|
|
|
|
packing_spec.extra_settings_count++;
|
|
|
|
packing_spec.extra_settings[0].type = GRIB_TYPE_LONG;
|
|
|
|
packing_spec.extra_settings[0].name = "expandBoundingBox";
|
|
|
|
packing_spec.extra_settings[0].long_value = 1;
|
|
|
|
*/
|
2015-09-23 14:30:30 +00:00
|
|
|
|
|
|
|
finalh = grib_util_set_spec(
|
|
|
|
handle,
|
|
|
|
&spec,
|
|
|
|
&packing_spec,
|
|
|
|
set_spec_flags,
|
|
|
|
values,
|
|
|
|
outlen,
|
|
|
|
&err);
|
|
|
|
assert(finalh);
|
|
|
|
assert(err == 0);
|
|
|
|
|
2018-06-06 14:50:47 +00:00
|
|
|
/* Try some invalid inputs and check it is handled */
|
|
|
|
{
|
|
|
|
codes_handle *h2 = 0;
|
|
|
|
packing_spec.accuracy=999;
|
|
|
|
h2 = grib_util_set_spec(handle, &spec, &packing_spec, set_spec_flags, values, outlen, &err);
|
|
|
|
assert(err == GRIB_INTERNAL_ERROR);
|
|
|
|
assert(!h2);
|
2018-11-15 14:25:35 +00:00
|
|
|
#ifdef INFINITY
|
2018-06-06 14:50:47 +00:00
|
|
|
packing_spec.accuracy=GRIB_UTIL_ACCURACY_USE_PROVIDED_BITS_PER_VALUES;
|
|
|
|
values[0] = INFINITY;
|
|
|
|
h2 = grib_util_set_spec(handle, &spec, &packing_spec, set_spec_flags, values, outlen, &err);
|
|
|
|
assert(err == GRIB_ENCODING_ERROR);
|
|
|
|
assert(!h2);
|
2018-11-15 14:25:35 +00:00
|
|
|
#endif
|
2018-06-06 14:50:47 +00:00
|
|
|
}
|
|
|
|
|
2015-09-23 14:30:30 +00:00
|
|
|
/* Write out the message to the output file */
|
|
|
|
CODES_CHECK(grib_get_message(finalh, &buffer, &size),0);
|
|
|
|
if(fwrite(buffer,1,size,out) != size) {
|
|
|
|
assert(0);
|
|
|
|
}
|
|
|
|
grib_handle_delete(handle);
|
|
|
|
grib_handle_delete(finalh);
|
|
|
|
fclose(in);
|
|
|
|
fclose(out);
|
2018-09-06 13:56:41 +00:00
|
|
|
free(values);
|
2018-02-05 16:05:31 +00:00
|
|
|
/*printf("ALL OK: %s\n", __func__);*/
|
2015-09-23 14:30:30 +00:00
|
|
|
}
|
|
|
|
|
2017-09-01 16:36:16 +00:00
|
|
|
static void test_regular_ll(int remove_local_def, int edition, const char* packingType,
|
2016-10-10 18:14:34 +00:00
|
|
|
const char* input_filename, const char* output_filename)
|
2015-07-21 13:24:02 +00:00
|
|
|
{
|
|
|
|
/* based on copy_spec_from_ksec */
|
|
|
|
int err = 0;
|
2015-09-23 14:30:30 +00:00
|
|
|
size_t slen = 32, inlen = 0, outlen = 0;
|
2015-07-21 13:24:02 +00:00
|
|
|
size_t size=0;
|
|
|
|
int set_spec_flags=0;
|
|
|
|
double* values = NULL;
|
|
|
|
FILE* in = NULL;
|
|
|
|
FILE* out = NULL;
|
|
|
|
const void* buffer = NULL;
|
2015-09-23 14:30:30 +00:00
|
|
|
char gridType[128] = {0,};
|
2018-04-20 15:43:16 +00:00
|
|
|
long input_edition = 0;
|
2015-07-21 13:24:02 +00:00
|
|
|
|
|
|
|
codes_handle *handle = 0;
|
|
|
|
codes_handle *finalh = 0;
|
|
|
|
grib_util_grid_spec spec={0,};
|
|
|
|
grib_util_packing_spec packing_spec={0,};
|
|
|
|
|
2018-12-10 17:23:06 +00:00
|
|
|
assert(input_filename);
|
2015-07-21 13:24:02 +00:00
|
|
|
in = fopen(input_filename,"r"); assert(in);
|
2015-09-23 14:30:30 +00:00
|
|
|
handle = codes_handle_new_from_file(0, in, PRODUCT_GRIB, &err); assert(handle);
|
2018-04-20 15:43:16 +00:00
|
|
|
|
|
|
|
CODES_CHECK(codes_get_long(handle, "edition", &input_edition), 0);
|
2015-09-23 14:30:30 +00:00
|
|
|
|
|
|
|
CODES_CHECK(grib_get_string(handle, "gridType", gridType, &slen),0);
|
|
|
|
if (!STR_EQUAL(gridType, "regular_ll")) {
|
|
|
|
grib_handle_delete(handle);
|
|
|
|
return;
|
|
|
|
}
|
2018-12-10 17:23:06 +00:00
|
|
|
assert(output_filename);
|
2015-07-21 13:24:02 +00:00
|
|
|
out = fopen(output_filename,"w"); assert(out);
|
|
|
|
|
|
|
|
CODES_CHECK(codes_get_size(handle,"values",&inlen), 0);
|
|
|
|
values = (double*)malloc(sizeof(double)*inlen);
|
2015-09-23 14:30:30 +00:00
|
|
|
CODES_CHECK(codes_get_double_array(handle, "values", values,&inlen), 0);
|
2015-07-21 13:24:02 +00:00
|
|
|
|
|
|
|
spec.grid_type = GRIB_UTIL_GRID_SPEC_REGULAR_LL;
|
|
|
|
spec.Nj = 14;
|
|
|
|
spec.Ni = 17;
|
|
|
|
outlen = spec.Nj * spec.Ni;
|
|
|
|
spec.iDirectionIncrementInDegrees = 1.5;
|
|
|
|
spec.jDirectionIncrementInDegrees = 1.5;
|
2018-04-20 15:43:16 +00:00
|
|
|
spec.latitudeOfFirstGridPointInDegrees = 60.0001;
|
2015-07-21 13:24:02 +00:00
|
|
|
spec.longitudeOfFirstGridPointInDegrees = -9.0;
|
|
|
|
spec.latitudeOfLastGridPointInDegrees = 40.5;
|
|
|
|
spec.longitudeOfLastGridPointInDegrees = 15.0;
|
|
|
|
spec.bitmapPresent = 0;
|
|
|
|
|
2016-10-10 18:14:34 +00:00
|
|
|
/*packing_spec.packing_type = GRIB_UTIL_PACKING_TYPE_GRID_SIMPLE;*/
|
|
|
|
packing_spec.packing_type = get_packing_type_code(packingType);
|
2015-07-21 13:24:02 +00:00
|
|
|
packing_spec.bitsPerValue = 24;
|
|
|
|
packing_spec.accuracy=GRIB_UTIL_ACCURACY_USE_PROVIDED_BITS_PER_VALUES;
|
|
|
|
packing_spec.packing=GRIB_UTIL_PACKING_USE_PROVIDED;
|
2016-10-10 18:14:34 +00:00
|
|
|
|
2018-04-20 15:43:16 +00:00
|
|
|
packing_spec.extra_settings_count++;
|
|
|
|
packing_spec.extra_settings[0].type = GRIB_TYPE_LONG;
|
|
|
|
packing_spec.extra_settings[0].name = "expandBoundingBox";
|
|
|
|
packing_spec.extra_settings[0].long_value = 1;
|
|
|
|
|
2016-10-10 18:14:34 +00:00
|
|
|
if (edition != 0) {
|
|
|
|
packing_spec.editionNumber = edition;
|
|
|
|
}
|
|
|
|
if (remove_local_def) {
|
|
|
|
packing_spec.deleteLocalDefinition = 1;
|
|
|
|
}
|
2015-07-21 13:24:02 +00:00
|
|
|
|
|
|
|
finalh = codes_grib_util_set_spec(
|
|
|
|
handle,
|
|
|
|
&spec,
|
|
|
|
&packing_spec,
|
|
|
|
set_spec_flags,
|
|
|
|
values,
|
|
|
|
outlen,
|
|
|
|
&err);
|
|
|
|
assert(finalh);
|
|
|
|
assert(err == 0);
|
|
|
|
|
2018-04-20 15:43:16 +00:00
|
|
|
/* Check expand_bounding_box worked.
|
|
|
|
* Specified latitudeOfFirstGridPointInDegrees cannot be encoded in GRIB1
|
|
|
|
* so it is rounded up to nearest millidegree */
|
|
|
|
if (input_edition == 1) {
|
|
|
|
const double expected_lat1 = 60.001;
|
|
|
|
double lat1 = 0;
|
|
|
|
CODES_CHECK(codes_get_double(finalh, "latitudeOfFirstGridPointInDegrees", &lat1), 0);
|
|
|
|
assert( fabs(lat1 - expected_lat1) < 1e-10 );
|
|
|
|
}
|
|
|
|
|
2015-07-21 13:24:02 +00:00
|
|
|
/* Write out the message to the output file */
|
|
|
|
CODES_CHECK(codes_get_message(finalh, &buffer, &size),0);
|
|
|
|
if(fwrite(buffer,1,size,out) != size) {
|
|
|
|
assert(0);
|
|
|
|
}
|
|
|
|
codes_handle_delete(handle);
|
|
|
|
codes_handle_delete(finalh);
|
2015-09-23 14:30:30 +00:00
|
|
|
fclose(in);
|
|
|
|
fclose(out);
|
2015-07-21 13:24:02 +00:00
|
|
|
}
|
|
|
|
|
2017-09-01 16:36:16 +00:00
|
|
|
#if 0
|
|
|
|
static void test_grid_complex_spatial_differencing(int remove_local_def, int edition, const char* packingType,
|
2017-08-02 14:18:38 +00:00
|
|
|
const char* input_filename, const char* output_filename)
|
|
|
|
{
|
|
|
|
/* based on copy_spec_from_ksec */
|
|
|
|
int err = 0;
|
|
|
|
size_t slen = 34, inlen = 0, outlen = 0;
|
|
|
|
size_t size=0;
|
|
|
|
int set_spec_flags=0;
|
|
|
|
double* values = NULL;
|
|
|
|
FILE* in = NULL;
|
|
|
|
FILE* out = NULL;
|
|
|
|
const void* buffer = NULL;
|
|
|
|
char gridType[128] = {0,};
|
|
|
|
double theMax,theMin,theAverage;
|
|
|
|
|
|
|
|
codes_handle *handle = 0;
|
|
|
|
codes_handle *finalh = 0;
|
|
|
|
grib_util_grid_spec spec={0,};
|
|
|
|
grib_util_packing_spec packing_spec={0,};
|
|
|
|
|
|
|
|
in = fopen(input_filename,"r"); assert(in);
|
|
|
|
handle = codes_handle_new_from_file(0, in, PRODUCT_GRIB, &err); assert(handle);
|
|
|
|
|
|
|
|
CODES_CHECK(grib_get_string(handle, "packingType", gridType, &slen),0);
|
|
|
|
if (!STR_EQUAL(gridType, "grid_complex_spatial_differencing")) {
|
|
|
|
grib_handle_delete(handle);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
out = fopen(output_filename,"w"); assert(out);
|
|
|
|
|
|
|
|
CODES_CHECK(codes_get_size(handle,"values",&inlen), 0);
|
|
|
|
values = (double*)malloc(sizeof(double)*inlen);
|
|
|
|
CODES_CHECK(codes_get_double_array(handle, "values", values,&inlen), 0);
|
|
|
|
|
|
|
|
|
|
|
|
CODES_CHECK(codes_get_double(handle, "max", &theMax),0);
|
|
|
|
CODES_CHECK(codes_get_double(handle, "min", &theMin),0);
|
|
|
|
CODES_CHECK(codes_get_double(handle, "average",&theAverage),0);
|
2017-09-01 16:36:16 +00:00
|
|
|
printf("inlen=%lu \t max=%g \t min=%g \t avg=%g\n", inlen, theMax, theMin, theAverage);
|
2017-08-02 14:18:38 +00:00
|
|
|
|
|
|
|
spec.grid_type = GRIB_UTIL_GRID_SPEC_REGULAR_LL;
|
|
|
|
spec.Nj = 721;
|
|
|
|
spec.Ni = 1440;
|
|
|
|
outlen = spec.Nj * spec.Ni;
|
|
|
|
spec.iDirectionIncrementInDegrees = 0.25;
|
|
|
|
spec.jDirectionIncrementInDegrees = 0.25;
|
|
|
|
spec.latitudeOfFirstGridPointInDegrees = 90.0;
|
|
|
|
spec.longitudeOfFirstGridPointInDegrees = 0.0;
|
|
|
|
spec.latitudeOfLastGridPointInDegrees = -90.0;
|
|
|
|
spec.longitudeOfLastGridPointInDegrees = 359.75;
|
|
|
|
spec.bitmapPresent = 1; /* there are missing values inside the data section! */
|
|
|
|
spec.missingValue = 9999;
|
|
|
|
|
2017-08-03 13:14:17 +00:00
|
|
|
packing_spec.packing_type = GRIB_UTIL_PACKING_TYPE_GRID_SIMPLE; /*Convert to Grid Simple Packing*/
|
2017-08-02 14:18:38 +00:00
|
|
|
packing_spec.bitsPerValue = 11;
|
|
|
|
packing_spec.accuracy=GRIB_UTIL_ACCURACY_USE_PROVIDED_BITS_PER_VALUES;
|
2017-08-03 13:14:17 +00:00
|
|
|
/*packing_spec.packing=GRIB_UTIL_PACKING_USE_PROVIDED;*/
|
2017-08-02 14:18:38 +00:00
|
|
|
|
|
|
|
if (edition != 0) {
|
|
|
|
packing_spec.editionNumber = edition;
|
|
|
|
}
|
|
|
|
if (remove_local_def) {
|
|
|
|
packing_spec.deleteLocalDefinition = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
finalh = codes_grib_util_set_spec(
|
|
|
|
handle,
|
|
|
|
&spec,
|
|
|
|
&packing_spec,
|
|
|
|
set_spec_flags,
|
|
|
|
values,
|
|
|
|
outlen,
|
|
|
|
&err);
|
|
|
|
assert(finalh);
|
|
|
|
assert(err == 0);
|
|
|
|
|
|
|
|
/* Write out the message to the output file */
|
|
|
|
CODES_CHECK(codes_get_message(finalh, &buffer, &size),0);
|
|
|
|
if(fwrite(buffer,1,size,out) != size) {
|
|
|
|
assert(0);
|
|
|
|
}
|
|
|
|
codes_handle_delete(handle);
|
|
|
|
codes_handle_delete(finalh);
|
|
|
|
fclose(in);
|
|
|
|
fclose(out);
|
|
|
|
}
|
2017-09-01 16:36:16 +00:00
|
|
|
#endif
|
2017-08-02 14:18:38 +00:00
|
|
|
|
2017-09-01 16:36:16 +00:00
|
|
|
static void usage(const char *prog)
|
2016-10-10 18:14:34 +00:00
|
|
|
{
|
|
|
|
fprintf(stderr, "%s: [-p packingType] [-r] [-e edition] in.grib out.grib\n", prog);
|
|
|
|
fprintf(stderr, "-p packingType: one of grid_jpeg, grid_second_order or grid_simple\n");
|
|
|
|
fprintf(stderr, "-r remove local definition\n");
|
|
|
|
fprintf(stderr, "-e edition: 1 or 2\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2015-07-21 13:24:02 +00:00
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2018-12-10 17:23:06 +00:00
|
|
|
int i = 0, remove_local_def = 0;
|
2016-10-10 18:14:34 +00:00
|
|
|
int edition = 0;
|
|
|
|
char* packingType = NULL;
|
|
|
|
const char* prog = argv[0];
|
|
|
|
char* infile_name = NULL;
|
|
|
|
char* outfile_name = NULL;
|
2018-12-13 22:21:27 +00:00
|
|
|
|
|
|
|
if (argc==1 || argc >8) usage(prog);
|
2018-12-10 17:23:06 +00:00
|
|
|
|
|
|
|
for (i = 1; i < argc; i++) {
|
|
|
|
if (strcmp(argv[i],"-p")==0) {
|
|
|
|
packingType = argv[i+1];
|
|
|
|
++i;
|
|
|
|
} else if (strcmp(argv[i],"-e")==0) {
|
|
|
|
edition = atoi( argv[i+1] );
|
|
|
|
++i;
|
|
|
|
} else if (strcmp(argv[i],"-r")==0) {
|
|
|
|
remove_local_def = 1;
|
|
|
|
} else {
|
|
|
|
/* Expect 2 filenames */
|
|
|
|
infile_name = argv[i];
|
|
|
|
outfile_name = argv[i+1];
|
|
|
|
break;
|
2016-10-10 18:14:34 +00:00
|
|
|
}
|
|
|
|
}
|
2018-12-10 17:23:06 +00:00
|
|
|
#if 0
|
|
|
|
printf("DEBUG remove_local_def = %d\n", remove_local_def);
|
|
|
|
printf("DEBUG edition = %d\n", edition);
|
|
|
|
printf("DEBUG packingType = %s\n", packingType);
|
|
|
|
printf("DEBUG infile_name = %s\n", infile_name);
|
|
|
|
printf("DEBUG outfile_name = %s\n", outfile_name);
|
|
|
|
#endif
|
2016-10-10 18:14:34 +00:00
|
|
|
test_regular_ll(remove_local_def, edition, packingType, infile_name, outfile_name);
|
|
|
|
test_reduced_gg(remove_local_def, edition, packingType, infile_name, outfile_name);
|
2017-08-03 13:14:17 +00:00
|
|
|
/*test_grid_complex_spatial_differencing(remove_local_def, edition, packingType, infile_name, outfile_name);*/
|
2015-07-21 13:24:02 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|