mirror of https://github.com/ecmwf/eccodes.git
528 lines
12 KiB
C
528 lines
12 KiB
C
/*
|
|
* Copyright 2005-2018 ECMWF.
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
/***************************************************************************
|
|
* Enrico Fucile - 06.01.2009 *
|
|
* *
|
|
***************************************************************************/
|
|
#include "grib_api_internal.h"
|
|
|
|
#if GRIB_PTHREADS
|
|
static pthread_once_t once = PTHREAD_ONCE_INIT;
|
|
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
static void init() {
|
|
pthread_mutexattr_t attr;
|
|
pthread_mutexattr_init(&attr);
|
|
pthread_mutexattr_settype(&attr,PTHREAD_MUTEX_RECURSIVE);
|
|
pthread_mutex_init(&mutex,&attr);
|
|
pthread_mutexattr_destroy(&attr);
|
|
}
|
|
#elif GRIB_OMP_THREADS
|
|
static int once = 0;
|
|
static omp_nest_lock_t mutex;
|
|
|
|
static void init()
|
|
{
|
|
GRIB_OMP_CRITICAL(lock_grib_ieeefloat_c)
|
|
{
|
|
if (once == 0)
|
|
{
|
|
omp_init_nest_lock(&mutex);
|
|
once = 1;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#if 1
|
|
|
|
typedef struct ieee_table_t ieee_table_t;
|
|
|
|
struct ieee_table_t {
|
|
int inited;
|
|
double e[255];
|
|
double v[255];
|
|
double vmin;
|
|
double vmax;
|
|
};
|
|
|
|
static ieee_table_t ieee_table={ 0,{0,},{0,}, 0, 0 };
|
|
|
|
static void init_ieee_table()
|
|
{
|
|
if (!ieee_table.inited) {
|
|
unsigned long i;
|
|
unsigned long mmin=0x800000;
|
|
unsigned long mmax=0xffffff;
|
|
double e=1;
|
|
for (i=1; i<=104;i++) {
|
|
e*=2;
|
|
ieee_table.e[i+150]=e;
|
|
ieee_table.v[i+150]=e*mmin;
|
|
}
|
|
ieee_table.e[150]=1;
|
|
ieee_table.v[150]=mmin;
|
|
e=1;
|
|
for (i=1; i<150;i++) {
|
|
e/=2;
|
|
ieee_table.e[150-i]=e;
|
|
ieee_table.v[150-i]=e*mmin;
|
|
}
|
|
ieee_table.vmin=ieee_table.v[1];
|
|
ieee_table.vmax=ieee_table.e[254]*mmax;
|
|
ieee_table.inited=1;
|
|
/*for (i=0;i<128;i++) printf("++++ ieee_table.v[%d]=%g\n",i,ieee_table.v[i]);*/
|
|
}
|
|
}
|
|
|
|
static void init_table_if_needed()
|
|
{
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init)
|
|
GRIB_MUTEX_LOCK(&mutex)
|
|
|
|
if (!ieee_table.inited) init_ieee_table();
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex)
|
|
}
|
|
|
|
static void binary_search(double xx[], const unsigned long n, double x, unsigned long *j)
|
|
{
|
|
/*These routine works only on ascending ordered arrays*/
|
|
unsigned long ju,jm,jl;
|
|
jl=0;
|
|
ju=n;
|
|
while (ju-jl > 1) {
|
|
jm=(ju+jl) >> 1;
|
|
/* printf("jl=%lu jm=%lu ju=%lu\n",jl,jm,ju); */
|
|
/* printf("xx[jl]=%.10e xx[jm]=%.10e xx[ju]=%.10e\n",xx[jl],xx[jm],xx[ju]); */
|
|
if (x >= xx[jm]) jl=jm;
|
|
else ju=jm;
|
|
}
|
|
*j=jl;
|
|
}
|
|
|
|
double grib_ieee_table_e(unsigned long e)
|
|
{
|
|
init_table_if_needed();
|
|
return ieee_table.e[e];
|
|
}
|
|
|
|
double grib_ieee_table_v(unsigned long e)
|
|
{
|
|
init_table_if_needed();
|
|
return ieee_table.v[e];
|
|
}
|
|
|
|
unsigned long grib_ieee_to_long(double x)
|
|
{
|
|
unsigned long s = 0;
|
|
unsigned long mmax = 0xffffff;
|
|
unsigned long mmin = 0x800000;
|
|
unsigned long m = mmax;
|
|
unsigned long e=0;
|
|
double rmmax=mmax+0.5;
|
|
|
|
init_table_if_needed();
|
|
|
|
/* printf("\ngrib_ieee_to_long: x=%.20e\n",x); */
|
|
if (x < 0) { s = 1; x = -x; }
|
|
|
|
/* Underflow */
|
|
if (x < ieee_table.vmin) {
|
|
/*printf("grib_ieee_to_long: (x < ieee_table.vmin) x=%.20e vmin=%.20e v=0x%lX\n",x,ieee_table.vmin,(s<<31));*/
|
|
return (s << 31);
|
|
}
|
|
|
|
/* Overflow */
|
|
if (x > ieee_table.vmax) {
|
|
fprintf(stderr, "grib_ieee_to_long: Number is too large: x=%.20e > xmax=%.20e\n", x, ieee_table.vmax);
|
|
Assert(0);
|
|
return 0;
|
|
}
|
|
|
|
binary_search(ieee_table.v, 254, x, &e);
|
|
|
|
/* printf("grib_ieee_to_long: e=%ld\n",e); */
|
|
|
|
x/=ieee_table.e[e];
|
|
|
|
/* printf("grib_ieee_to_long: x=%.20e\n",x); */
|
|
|
|
while(x < mmin ) { x *= 2; e--;
|
|
/* printf("grib_ieee_to_long (e--): x=%.20e e=%ld \n",x,e); */
|
|
}
|
|
|
|
while(x > rmmax ) { x /= 2; e++;
|
|
/* printf("grib_ieee_to_long (e++): x=%.20e e=%ld \n",x,e); */
|
|
}
|
|
|
|
m=x+0.5;
|
|
/* printf("grib_ieee_to_long: m=0x%lX (%lu) x=%.10e \n",m,m,x ); */
|
|
if ( m > mmax ) { e++; m=0x800000;
|
|
/* printf("grib_ieee_to_long: ( m > mmax ) m=0x%lX (%lu) x=%.10e \n",m,m,x ); */
|
|
}
|
|
|
|
/* printf("grib_ieee_to_long: s=%lu c=%lu (0x%lX) m=%lu (0x%lX)\n",s,e,e,m,m ); */
|
|
|
|
return (s << 31) | ( e << 23 ) | ( m & 0x7fffff );
|
|
}
|
|
|
|
double grib_ieeefloat_error(double x)
|
|
{
|
|
unsigned long e=0;
|
|
|
|
init_table_if_needed();
|
|
|
|
if (x < 0) x = -x;
|
|
|
|
/* Underflow */
|
|
if (x < ieee_table.vmin) return ieee_table.vmin;
|
|
|
|
/* Overflow */
|
|
if (x > ieee_table.vmax) {
|
|
fprintf(stderr, "grib_ieeefloat_error: Number is too large: x=%.20e > xmax=%.20e\n", x, ieee_table.vmax);
|
|
Assert(0);
|
|
return 0;
|
|
}
|
|
|
|
binary_search(ieee_table.v, 254, x, &e);
|
|
|
|
return ieee_table.e[e] ;
|
|
}
|
|
|
|
double grib_long_to_ieee(unsigned long x)
|
|
{
|
|
unsigned long s = x & 0x80000000;
|
|
unsigned long c = (x & 0x7f800000) >> 23;
|
|
unsigned long m = (x & 0x007fffff);
|
|
|
|
double val;
|
|
|
|
#ifdef DEBUG
|
|
if ( x > 0 && x < 0x800000 ) {
|
|
fprintf(stderr, "grib_long_to_ieee: Invalid input %lu\n", x);
|
|
Assert(0);
|
|
}
|
|
#endif
|
|
init_table_if_needed();
|
|
|
|
if (c == 0 && m==0) return 0;
|
|
|
|
if (c == 0) {
|
|
m |= 0x800000;
|
|
c=1;
|
|
} else m |= 0x800000;
|
|
|
|
val=m*ieee_table.e[c];
|
|
if(s) val = -val;
|
|
|
|
return val;
|
|
}
|
|
|
|
unsigned long grib_ieee_nearest_smaller_to_long(double x)
|
|
{
|
|
unsigned long l;
|
|
unsigned long e;
|
|
unsigned long m ;
|
|
unsigned long s;
|
|
unsigned long mmin = 0x800000;
|
|
double y,eps;
|
|
|
|
if(x == 0) return 0;
|
|
|
|
init_table_if_needed();
|
|
|
|
l=grib_ieee_to_long(x);
|
|
y=grib_long_to_ieee(l);
|
|
|
|
if ( x < y ) {
|
|
if ( x < 0 && -x < ieee_table.vmin ) {
|
|
l=0x80800000;
|
|
} else {
|
|
e = (l & 0x7f800000) >> 23;
|
|
m = (l & 0x007fffff) | 0x800000;
|
|
s = l & 0x80000000;
|
|
|
|
if ( m == mmin ) {
|
|
/* printf("grib_ieee_nearest_smaller_to_long: m == mmin (0x%lX) e=%lu\n",m,e); */
|
|
e = s ? e : e-1;
|
|
if (e<1) e=1;
|
|
if (e>254) e=254;
|
|
/* printf("grib_ieee_nearest_smaller_to_long: e=%lu \n",e); */
|
|
}
|
|
|
|
eps=ieee_table.e[e];
|
|
|
|
/* printf("grib_ieee_nearest_smaller_to_long: x<y\n"); */
|
|
l=grib_ieee_to_long(y-eps);
|
|
/* printf("grib_ieee_nearest_smaller_to_long: grib_ieee_to_long(y-eps)=0x%lX y=%.10e eps=%.10e x=%.10e\n",l,y,eps,x); */
|
|
}
|
|
} else return l;
|
|
|
|
if (x<grib_long_to_ieee(l)) {
|
|
printf("grib_ieee_nearest_smaller_to_long: x=%.20e grib_long_to_ieee(0x%lX)=%.20e\n",x,l,grib_long_to_ieee(l));
|
|
Assert(x >= grib_long_to_ieee(l));
|
|
}
|
|
|
|
return l;
|
|
}
|
|
|
|
int grib_nearest_smaller_ieee_float(double a,double* ret)
|
|
{
|
|
unsigned long l=0;
|
|
|
|
init_table_if_needed();
|
|
|
|
if (a>ieee_table.vmax) return GRIB_INTERNAL_ERROR;
|
|
|
|
l=grib_ieee_nearest_smaller_to_long(a);
|
|
*ret=grib_long_to_ieee(l);
|
|
return GRIB_SUCCESS;
|
|
}
|
|
|
|
#else
|
|
/* old code to be deleted */
|
|
|
|
double grib_ieeefloat_error(double x) {return 0;}
|
|
|
|
double grib_long_to_ieee(unsigned long x)
|
|
{
|
|
unsigned long s = x & 0x80000000;
|
|
unsigned long c = (x & 0x7f800000) >> 23;
|
|
unsigned long m;
|
|
double val;
|
|
long e;
|
|
|
|
if(x == 0) return 0;
|
|
Assert(c != 255);
|
|
|
|
if(c == 0) { m = x & 0x007fffff; e = -126 - 23; }
|
|
else { m = (x & 0x007fffff) | (1<<23); e = c - 127 - 23; }
|
|
|
|
val = m;
|
|
|
|
while(e < 0) { val /= 2.0; e++; }
|
|
while(e > 0) { val *= 2.0; e--; }
|
|
|
|
if(s) val = -val;
|
|
|
|
return val;
|
|
}
|
|
|
|
int grib_nearest_smaller_ieee_float(double a,double* x)
|
|
{
|
|
double e = grib_long_to_ieee(grib_ieee_to_long(a));
|
|
double b = a;
|
|
|
|
/* printf("----> a=%g e=%g e-a=%g\n",a,e,e-a); */
|
|
|
|
if( e > b )
|
|
{
|
|
unsigned long ub = grib_ieee_to_long(b);
|
|
unsigned long ue;
|
|
while(e>b)
|
|
{
|
|
/* printf("a=%g e=%g e-a=%g\n",a,e,e-a); */
|
|
a -= (e-a);
|
|
e = grib_long_to_ieee(grib_ieee_to_long(a));
|
|
}
|
|
ue = grib_ieee_to_long(e);
|
|
Assert((ue-ub) == 1);
|
|
}
|
|
|
|
Assert(b >= e);
|
|
*x=e;
|
|
return GRIB_SUCCESS;
|
|
}
|
|
|
|
unsigned long grib_ieee_to_long(double x)
|
|
{
|
|
/* double y = x; */
|
|
unsigned long s = 0;
|
|
unsigned long m;
|
|
long p = 0;
|
|
unsigned long e = 0;
|
|
|
|
if(x == 0) return 0;
|
|
|
|
if(x < 0) { s = 1; x = -x; }
|
|
while(x < 2) { x *= 2; p--; }
|
|
|
|
while(x >= 2) { x /= 2; p++; }
|
|
|
|
if(p > 127 ) {
|
|
/* Overflow */
|
|
e = 255;
|
|
m = 0;
|
|
}
|
|
else if(p < -126) {
|
|
/* int i; */
|
|
e = 0;
|
|
/* printf("p=%ld x=%g %ld\n",p,x,p+126+23); */
|
|
m = x * grib_power(p+126+23,2);
|
|
}
|
|
else {
|
|
e = p + 127;
|
|
m = x * ( 1 << 23);
|
|
m &= 0x007fffff;
|
|
}
|
|
|
|
m = (s << 31) | ( e << 23 ) | m;
|
|
|
|
return m;
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef IEEE
|
|
|
|
|
|
/*
|
|
* To make these two routines consistent to grib_ieee_to_long and grib_long_to_ieee,
|
|
* we should not do any byte swapping but rather perform a raw copy.
|
|
* Byte swapping is actually implemented in grib_decode_unsigned_long and
|
|
* grib_encode_unsigned_long.
|
|
*/
|
|
|
|
unsigned long grib_ieee64_to_long(double x)
|
|
{
|
|
unsigned long lval;
|
|
memcpy(&lval,&x,8);
|
|
return lval;
|
|
}
|
|
|
|
double grib_long_to_ieee64 (unsigned long x)
|
|
{
|
|
double dval;
|
|
memcpy(&dval,&x,8);
|
|
return dval;
|
|
}
|
|
|
|
|
|
int grib_ieee_decode_array(grib_context* c,unsigned char* buf,size_t nvals,int bytes,double* val)
|
|
{
|
|
int err=0,i=0,j=0;
|
|
unsigned char s[8]={0,};
|
|
float fval;
|
|
double* pval=val;
|
|
|
|
switch (bytes) {
|
|
case 4:
|
|
for (i=0;i<nvals;i++) {
|
|
#if IEEE_LE
|
|
for (j=3;j>=0;j--)
|
|
s[j]=*(buf++);
|
|
memcpy(&fval,s,4);
|
|
val[i]=(double)fval;
|
|
#elif IEEE_BE
|
|
memcpy(&fval,buf,4);
|
|
val[i]=(double)fval;
|
|
buf+=4;
|
|
#endif
|
|
}
|
|
break;
|
|
case 8:
|
|
for (i=0;i<nvals;i++) {
|
|
#if IEEE_LE
|
|
for (j=7;j>=0;j--)
|
|
s[j]=*(buf++);
|
|
memcpy(pval++,s,8);
|
|
#elif IEEE_BE
|
|
memcpy(pval++,buf,8);
|
|
buf+=8;
|
|
#endif
|
|
}
|
|
break;
|
|
default:
|
|
grib_context_log(c,GRIB_LOG_ERROR,
|
|
"grib_ieee_decode_array: %d bits not implemented",bytes*8);
|
|
return GRIB_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
#else
|
|
|
|
int grib_ieee_decode_array(grib_context* c,unsigned char* buf,size_t nvals,int bytes,double* val)
|
|
{
|
|
int err=0,i=0;
|
|
long bitr=0;
|
|
|
|
for(i=0;i< nvals;i++)
|
|
val[i] = grib_long_to_ieee(grib_decode_unsigned_long(buf,&bitr,bytes*8));
|
|
|
|
return err;
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef IEEE
|
|
|
|
int grib_ieee_encode_array(grib_context* c,double* val,size_t nvals,int bytes,
|
|
unsigned char* buf) {
|
|
int err=0,i=0,j=0;
|
|
unsigned char s4[4];
|
|
unsigned char s8[8];
|
|
float fval=0;
|
|
double *pval=val;
|
|
|
|
switch (bytes) {
|
|
case 4:
|
|
for (i=0;i<nvals;i++) {
|
|
fval=(float)val[i];
|
|
|
|
#if IEEE_LE
|
|
memcpy(s4,&(fval),4);
|
|
for (j=3;j>=0;j--)
|
|
*(buf++)=s4[j];
|
|
#elif IEEE_BE
|
|
memcpy(buf,&(fval),4);
|
|
buf+=4;
|
|
#endif
|
|
}
|
|
break;
|
|
case 8:
|
|
for (i=0;i<nvals;i++) {
|
|
#if IEEE_LE
|
|
memcpy(s8,pval++,8);
|
|
for (j=7;j>=0;j--)
|
|
*(buf++)=s8[j];
|
|
#elif IEEE_BE
|
|
memcpy(buf,pval++,8);
|
|
buf+=8;
|
|
#endif
|
|
}
|
|
break;
|
|
default:
|
|
grib_context_log(c,GRIB_LOG_ERROR,
|
|
"grib_ieee_encode_array: %d bits not implemented",bytes*8);
|
|
return GRIB_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
#else
|
|
|
|
int grib_ieee_encode_array(grib_context* c, double* val, size_t nvals, int bytes, unsigned char* buf)
|
|
{
|
|
int err=0,i=0;
|
|
long bitr=0;
|
|
|
|
for(i=0;i< nvals;i++)
|
|
grib_encode_unsigned_long(buf, grib_ieee_to_long(val[i]), &bitr, bytes*8);
|
|
|
|
return err;
|
|
}
|
|
|
|
#endif
|