eccodes/perl/GRIB-API/API.xs

988 lines
16 KiB
Plaintext
Raw Normal View History

2013-03-25 12:04:10 +00:00
#include "EXTERN.h"
#include "perl.h"
#include "XSUB.h"
#include "ppport.h"
#include <grib_api.h>
#include <stdlib.h>
/*
http://world.std.com/~swmcd/steven/perl/pm/xs/modules/Set/Bit/
*/
static int _fail_on_error = 0;
static int error(int e)
{
if(e && _fail_on_error) croak(grib_get_error_message(e));
return e;
}
static int fail(int e)
{
if(e) croak(grib_get_error_message(e));
return e;
}
static SV* get_long(grib_handle* h,const char* what)
{
long value;
if(error(grib_get_long(h,what,&value)) == 0)
return sv_2mortal(newSViv(value));
return &PL_sv_undef;
}
static SV* get_double(grib_handle* h,const char* what)
{
double value;
if(error(grib_get_double(h,what,&value)) == 0)
return sv_2mortal(newSVnv(value));
return &PL_sv_undef;
}
static SV* get_string(grib_handle* h,const char* what)
{
char buf[1024];
size_t size = sizeof(buf);
if(error(grib_get_string(h,what,buf,&size)) == 0)
{
while(size>0 && buf[size-1] == 0) size--;
return sv_2mortal(newSVpv(buf,size));
}
return &PL_sv_undef;
}
static SV* get_bytes(grib_handle* h,const char* what)
{
size_t size;
SV* sv;
char *buffer = 0;
if(error(grib_get_size(h,what,&size)) != 0)
return &PL_sv_undef;
sv = newSVpv("",0);
buffer = SvGROW(sv,size);
if(error(grib_get_bytes(h,what,buffer,&size)) == 0)
{
SvCUR_set(sv, size);
SvPOK_only(sv);
return sv_2mortal(sv);
}
else
{
/* Memory leek ... */
return &PL_sv_undef;
}
}
static SV* get_double_array(grib_handle* h,const char* what)
{
size_t size = 0;
double *values = 0;
SV* result = &PL_sv_undef;
if(error(grib_get_size(h,what,&size)) != 0)
return &PL_sv_undef;
Newz(0,values,size,double);
if(error(grib_get_double_array(h,what,values,&size)) == 0)
{
AV* av = newAV();
int i;
for(i = 0; i < size; i++)
av_push(av,newSVnv(values[i]));
result = sv_2mortal(newRV_noinc((SV*)av));
/* sv_bless(result,gv_stashpv("GRIB::API::Debug",1)); */
}
Safefree(values);
return result;
}
static SV* get_long_array(grib_handle* h,const char* what)
{
size_t size = 0;
long *values = 0;
SV* result = &PL_sv_undef;
if(error(grib_get_size(h,what,&size)) != 0)
return &PL_sv_undef;
Newz(0,values,size,long);
if(error(grib_get_long_array(h,what,values,&size)) == 0)
{
AV* av = newAV();
int i;
for(i = 0; i < size; i++)
av_push(av,newSVnv(values[i]));
result = sv_2mortal(newRV_noinc((SV*)av));
}
Safefree(values);
return result;
}
static int compar(const void *a, const void *b)
{
double da = *(double*)a;
double db = *(double*)b;
if(da == db)
return 0;
if(da < db)
return -1;
return 1;
}
typedef grib_handle *GRIB__API;
typedef grib_iterator *GRIB__API__Iterator;
MODULE = GRIB::API PACKAGE = GRIB::API
PROTOTYPES: ENABLE
void debug()
CODE:
grib_context_set_debug(0,1);
GRIB::API
Read(file)
SV* file
PREINIT:
FILE* f = PerlIO_findFILE(IoIFP(sv_2io(file)));
int err;
CODE:
RETVAL = grib_handle_new_from_file(0,f,&err);
fail(err);
OUTPUT:
RETVAL
GRIB::API
create(bufsv)
SV* bufsv
PREINIT:
size_t size = 0;
char *buffer;
CODE:
RETVAL = 0;
if(SvOK(bufsv))
{
size = SvCUR(bufsv);
buffer = SvPV(bufsv,PL_na);
RETVAL = grib_handle_new_from_message(0,buffer,size);
}
OUTPUT:
RETVAL
GRIB::API
template(name)
char* name
PREINIT:
CODE:
RETVAL = grib_handle_new_from_template(0,name);
OUTPUT:
RETVAL
GRIB::API
clone(h)
GRIB::API h
CODE:
RETVAL = grib_handle_clone(h);
OUTPUT:
RETVAL
void
DESTROY(h)
GRIB::API h
CODE:
grib_handle_delete(h);
void
Write(h,file)
GRIB::API h
SV* file
PREINIT:
FILE* f = PerlIO_findFILE(IoIFP(sv_2io(file)));
size_t size = 0;
const void *message = 0;
PPCODE:
if( fail(grib_get_message(h,&message,&size) ) == 0)
{
/* printf("write %ld\n",size); */
if(fwrite(message,1,size,f) == size)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
}
else
XPUSHs(&PL_sv_undef);
void fail_on_error(f = 1)
long f
PPCODE:
_fail_on_error = f;
XPUSHs(&PL_sv_yes);
void
Dump(h,file,mode = "debug",flags = ~0,format = NULL)
GRIB::API h
SV* file
char *mode
long flags
char *format
PREINIT:
FILE* f = PerlIO_findFILE(IoIFP(sv_2io(file)));
PPCODE:
grib_dump_content(h,f,mode,flags,format);
XPUSHs(&PL_sv_yes);
void
get_long(h,what)
GRIB::API h
char *what
PPCODE:
XPUSHs(get_long(h,what));
void
get_double(h,what)
GRIB::API h
char *what
PPCODE:
XPUSHs(get_double(h,what));
void
get_string(h,what)
GRIB::API h
char *what
PPCODE:
XPUSHs(get_string(h,what));
void
get_bytes(h,what)
GRIB::API h
char *what
PPCODE:
XPUSHs(get_bytes(h,what));
void
set_bytes(h,what,bufsv)
GRIB::API h
char *what
SV* bufsv
PREINIT:
size_t size = 0;
char *buffer;
PPCODE:
if(!SvOK(bufsv))
XPUSHs(&PL_sv_undef);
else {
size = SvCUR(bufsv);
buffer = SvPV(bufsv,PL_na);
if(error(grib_set_bytes(h,what,buffer,&size)) == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
}
void
set_values(h,ref)
GRIB::API h
SV* ref
PREINIT:
size_t size = 0;
grib_values vals[1024];
HV* hash;
SV *value;
I32 ret;
char* key;
int i;
PPCODE:
{
if(SvTYPE(SvRV(ref)) != SVt_PVHV)
{
croak("not an HASH reference!");
XPUSHs(&PL_sv_undef);
}
else
{
hash = (HV*)SvRV(ref);
hv_iterinit(hash);
size = 0;
while((value = hv_iternextsv(hash, &key, &ret)) != 0)
{
/* printf("key=%s\n",key); */
vals[size].name = strdup(key);
if(SvIOK(value))
{
vals[size].type = GRIB_TYPE_LONG;
vals[size].long_value = SvIV(value);
/* printf("-- %d %d %ld\n",size,vals[size].type,vals[size].long_value); */
}
else if(SvNOK(value))
{
vals[size].type = GRIB_TYPE_DOUBLE;
vals[size].double_value = SvNV(value);
/* printf("-- %d %d %g\n",size,vals[size].type,vals[size].double_value); */
}
else if(SvPOK(value))
{
size_t len;
char *buffer = SvPV(value,len);
/* e = grib_set_string(h,what,buffer,&size); */
vals[size].type = GRIB_TYPE_STRING;
vals[size].string_value = buffer;
/* printf("-- %d %d %s\n",size,vals[size].type,vals[size].string_value); */
}
else if(!SvOK(value) || (value == &PL_sv_undef))
{
/* TODO: support other missing */
vals[size].type = GRIB_TYPE_LONG;
vals[size].long_value = GRIB_MISSING_LONG;
}
else
{
char buf[1024];
sprintf(buf,"Invalid type %s",key);
croak(buf);
XPUSHs(&PL_sv_undef);
break;
}
/* printf("%d %d %d\n",SvIOK(value),SvNOK(value),SvPOK(value)); */
size++;
if(size > 1024)
{
croak("Too many values");
XPUSHs(&PL_sv_undef);
break;
}
}
/* printf("%s %d\n",vals[0].name,vals[0].type); */
/* printf("SIZE %d\n",size); */
if(error(grib_set_values(h,vals,size)) == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
for(i = 0; i < size; i++)
free((char*)vals[i].name);
XPUSHs(&PL_sv_yes);
}
}
void
get_double_array(h,what)
GRIB::API h
char *what
PREINIT:
size_t size = 0;
PPCODE:
if(error(grib_get_size(h,what,&size)) != 0)
XPUSHs(&PL_sv_undef);
else {
double *values = 0;
Newz(0,values,size,double);
if(values)
{
if(error(grib_get_double_array(h,what,values,&size)) == 0)
{
int i;
if(GIMME_V == G_ARRAY)
{
EXTEND(SP,size);
for(i = 0; i < size; i++)
XPUSHs(sv_2mortal(newSVnv(values[i])));
}
else
{
AV* av = newAV();
for(i = 0; i < size; i++)
av_push(av,newSVnv(values[i]));
/* XPUSHs(bufsv); */
/* XPUSHs(sv_2mortal(newSViv(size)));*/
XPUSHs(newRV_noinc((SV*)av));
}
}
else
XPUSHs(&PL_sv_undef);
}
else
XPUSHs(&PL_sv_undef);
Safefree(values);
}
void
set_double_array(h,what,list)
GRIB::API h
char *what
SV* list
PREINIT:
size_t size = 0;
double *values = 0;
int i;
AV* av;
if(!SvROK(list))
croak("Argument 2 is not an ARRAY reference");
if(SvTYPE(SvRV(list))!=SVt_PVAV)
croak("Argument 2 is not an ARRAY reference");
av = (AV*)SvRV(list);
if(av_len(av) < 0)
croak("list has negative size ");
size = av_len(av) + 1;
/* printf("set_double_array: %d\n",size); */
PPCODE:
Newz(0,values,size,double);
if(!values)
croak("cannot allocate values");
for(i = 0; i < size; i++)
{
SV* sv = *av_fetch(av,i,1);
if(SvNOK(sv))
values[i] = SvNV(sv);
else if(SvIOK(sv))
values[i] = SvIV(sv);
else if(SvPOK(sv))
{
char *buffer = SvPV(sv,PL_na);
values[i] = atof(buffer);
}
else values[i] = 0;
}
/* printf("list %d\n",size); */
if(error(grib_set_double_array(h,what,values,size)) == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
Safefree(values);
void
set_long_array(h,what,list)
GRIB::API h
char *what
SV* list
PREINIT:
size_t size = 0;
if(!SvROK(list))
croak("Argument 2 is not an ARRAY reference");
if(SvTYPE(SvRV(list))!=SVt_PVAV)
croak("Argument 2 is not an ARRAY reference");
AV* av = (AV*)SvRV(list);
if(av_len(av) < 0)
croak("list has negative size ");
size = av_len(av);
PPCODE:
if(error(grib_get_size(h,what,&size)) != 0)
XPUSHs(&PL_sv_undef);
else {
long *values = 0;
Newz(0,values,size,long);
if(values)
{
int i;
AV* av = (AV*)SvRV(list);
size = av_len(av);
for(i = 0; i < size; i++)
{
SV* sv = *av_fetch(av,i,1);
if(SvNOK(sv))
values[i] = SvNV(sv);
else if(SvIOK(sv))
values[i] = SvIV(sv);
else if(SvPOK(sv))
{
char *buffer = SvPV(sv,PL_na);
values[i] = atof(buffer);
}
else
values[i] = 0;
}
/* printf("list %d\n",size); */
if(error(grib_set_long_array(h,what,values,size)) == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
}
else
XPUSHs(&PL_sv_undef);
Safefree(values);
}
void
get_long_array(h,what)
GRIB::API h
char *what
PREINIT:
size_t size = 0;
PPCODE:
if(error(grib_get_size(h,what,&size)) != 0)
XPUSHs(&PL_sv_undef);
else {
long *values = 0;
Newz(0,values,size,long);
if(values)
{
if(error(grib_get_long_array(h,what,values,&size)) == 0)
{
int i;
if(GIMME_V == G_ARRAY)
{
EXTEND(SP,size);
for(i = 0; i < size; i++)
XPUSHs(sv_2mortal(newSViv(values[i])));
}
else
{
AV* av = newAV();
for(i = 0; i < size; i++)
av_push(av,newSViv(values[i]));
/* XPUSHs(bufsv); */
/*XPUSHs(sv_2mortal(newSViv(size)));*/
XPUSHs(sv_2mortal(newRV_noinc((SV*)av)));
}
}
else
XPUSHs(&PL_sv_undef);
}
else
XPUSHs(&PL_sv_undef);
Safefree(values);
}
char*
get_accessor_class(h,what)
GRIB::API h
char *what
PREINIT:
char *p;
PPCODE:
p = (char*)grib_get_accessor_class_name(h,what);
if(p)
XPUSHs(sv_2mortal(newSVpv(p,0)));
else
XPUSHs(&PL_sv_undef);
void
get_size(h,what)
GRIB::API h
char *what
PREINIT:
size_t size = 0;
PPCODE:
if( error(grib_get_size(h,what,&size)) != 0)
XPUSHs(&PL_sv_undef);
else
XPUSHs(sv_2mortal(newSViv(size)));
void
get_type(h,what)
GRIB::API h
char *what
PREINIT:
int type = 0;
PPCODE:
if( error(grib_get_native_type(h,what,&type)) != 0)
XPUSHs(&PL_sv_undef);
else
XPUSHs(sv_2mortal(newSViv(type)));
void
set(h,what,value)
GRIB::API h
char *what
SV* value;
PREINIT:
int e = 1;
PPCODE:
if(SvIOK(value))
e = error(grib_set_long(h,what,SvIV(value)));
else if(SvNOK(value))
e = error(grib_set_double(h,what,SvNV(value)));
else if(SvPOK(value))
{
size_t size;
char *buffer = SvPV(value,size);
e = error(grib_set_string(h,what,buffer,&size));
}
if(e == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
SV*
get(h,what)
GRIB::API h
char *what
PREINIT:
int e1,e2;
int type;
size_t size;
long long_value;
double double_value;
SV* result = &PL_sv_undef;
char buf[1024];
PPCODE:
e1 = error(grib_get_native_type(h,what,&type));
e2 = error(grib_get_size(h,what,&size));
if(e1 == 0 && e2 == 0)
{
/* printf("what=%s type=%d size=%d\n",what,type,size); */
switch(type)
{
case GRIB_TYPE_LONG:
if(size > 1)
result = get_long_array(h,what);
else
result = get_long(h,what);
break;
case GRIB_TYPE_DOUBLE:
if(size > 1)
result = get_double_array(h,what);
else
result = get_double(h,what);
break;
case GRIB_TYPE_STRING:
result = get_string(h,what);
break;
case GRIB_TYPE_BYTES:
result = get_bytes(h,what);
break;
default:
/* result = get_bytes(h,what); */
break;
}
}
XPUSHs(result);
SV*
get_gaussian_latitudes(n)
int n
PREINIT:
double *values = 0;
SV* result = &PL_sv_undef;
PPCODE:
Newz(0,values,n*2,double);
if( error(grib_get_gaussian_latitudes(n,values)) == 0)
{
AV* av = newAV();
int i;
for(i = 0; i < n*2; i++)
av_push(av,newSVnv(values[i]));
result = sv_2mortal(newRV_noinc((SV*)av));
/* sv_bless(result,gv_stashpv("GRIB::API::Debug",1)); */
}
Safefree(values);
XPUSHs(result);
void
set_long(h,what,value)
GRIB::API h
char *what
long value;
PREINIT:
PPCODE:
if(error(grib_set_long(h,what,value)) == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
void
set_long_internal(h,what,value)
GRIB::API h
char *what
long value;
PREINIT:
PPCODE:
if(error(grib_set_long_internal(h,what,value)) == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
void
set_double(h,what,value)
GRIB::API h
char *what
double value;
PREINIT:
PPCODE:
if(error(grib_set_double(h,what,value)) == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
void
set_double_internal(h,what,value)
GRIB::API h
char *what
double value;
PREINIT:
PPCODE:
if(error(grib_set_double_internal(h,what,value)) == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
void
set_string(h,what,value)
GRIB::API h
char *what
char *value;
PREINIT:
size_t size = strlen(value);
PPCODE:
if(error(grib_set_string(h,what,value,&size)) == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
void
set_string_internal(h,what,value)
GRIB::API h
char *what
char *value;
PREINIT:
size_t size = strlen(value);
PPCODE:
if(error(grib_set_string_internal(h,what,value,&size)) == 0)
XPUSHs(&PL_sv_yes);
else
XPUSHs(&PL_sv_undef);
void
update_sections_lengths(h)
GRIB::API h
PPCODE:
grib_update_sections_lengths(h);
void
obsfucate(h,what = "values")
GRIB::API h
char *what
PREINIT:
int e;
size_t size = 0;
double *values = 0;
PPCODE:
if(error(grib_get_size(h,what,&size)) != 0)
{
croak(grib_get_error_message(e));
return;
}
Newz(0,values,size,double);
if(error(grib_get_double_array(h,what,values,&size)) != 0)
{
Safefree(values);
croak(grib_get_error_message(e));
return;
}
qsort(values,size,sizeof(double),&compar);
if(error(grib_set_double_array(h,what,values,size)) != 0)
{
Safefree(values);
croak(grib_get_error_message(e));
return;
}
Safefree(values);
void
zero(h,what = "values")
GRIB::API h
char *what
PREINIT:
PPCODE:
fail(grib_set_double_array(h,what,NULL,0));
void
get_keys(h,name_space = NULL)
GRIB::API h
char *name_space
PREINIT:
grib_keys_iterator* ks;
PPCODE:
ks = grib_keys_iterator_new(h,GRIB_KEYS_ITERATOR_ALL_KEYS,name_space);
if(GIMME_V == G_ARRAY)
{
EXTEND(SP,800);
while(grib_keys_iterator_next(ks))
{
const char* name = grib_keys_iterator_get_name(ks);
XPUSHs(sv_2mortal(newSVpvn(name,strlen(name))));
}
}
else
{
AV* av = newAV();
while(grib_keys_iterator_next(ks))
{
const char* name = grib_keys_iterator_get_name(ks);
av_push(av,newSVpvn(name,strlen(name)));
}
XPUSHs(newRV_noinc((SV*)av));
}
grib_keys_iterator_delete(ks);
GRIB::API::Iterator
iterator(h)
GRIB::API h
CODE:
/* FIXME: RETVAL = grib_iterator_new(h); */
abort();
/* printf("ITERATOR %p\n",RETVAL); */
OUTPUT:
RETVAL
MODULE = GRIB::API PACKAGE = GRIB::API::Iterator
void
next(i)
GRIB::API::Iterator i
PREINIT:
double lat,lon,value;
PPCODE:
if(grib_iterator_next(i,&lat,&lon,&value))
{
XPUSHs(sv_2mortal(newSVnv(lat)));
XPUSHs(sv_2mortal(newSVnv(lon)));
XPUSHs(sv_2mortal(newSVnv(value)));
}
/* else */
/* XPUSHs(&PL_sv_undef); */