eccodes/src/grib_parse_utils.c

631 lines
15 KiB
C

/*
* Copyright 2005-2012 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.
*/
/***************************************************************************
* Jean Baptiste Filippi - 01.11.2005 *
* Enrico Fucile
* *
***************************************************************************/
#include "grib_api_internal.h"
grib_action* grib_parser_all_actions = 0;
grib_context* grib_parser_context = 0;
grib_concept_value* grib_parser_concept = 0;
grib_rule* grib_parser_rules = 0;
extern FILE* grib_yyin;
extern int grib_yydebug;
static const char* parse_file = 0;
#if GRIB_PTHREADS
static pthread_once_t once = PTHREAD_ONCE_INIT;
static pthread_mutex_t mutex_file = PTHREAD_MUTEX_INITIALIZER;
static pthread_mutex_t mutex_rules = PTHREAD_MUTEX_INITIALIZER;
static pthread_mutex_t mutex_concept = PTHREAD_MUTEX_INITIALIZER;
static pthread_mutex_t mutex_stream = 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_file,&attr);
pthread_mutex_init(&mutex_rules,&attr);
pthread_mutex_init(&mutex_concept,&attr);
pthread_mutex_init(&mutex_stream,&attr);
pthread_mutexattr_destroy(&attr);
}
#endif
int grib_recompose_name(grib_handle* h, grib_accessor *observer, const char* uname, char* fname,int fail)
{
grib_accessor* a;
char loc[1024];
int i = 0;
int ret = 0;
int mode = -1;
char val[1024];
double dval=0;
long lval=0;
int type=GRIB_TYPE_STRING;
size_t replen = 0;
loc[0] = 0 ;
fname[0] = 0 ;
for(i=0;i<strlen(uname);i++)
{
if(mode > -1)
{
if(uname[i] == ':') {
type=grib_type_to_int(uname[i+1]);
i++;
} else if(uname[i] == ']')
{
loc[mode] = 0;
mode = -1;
a = grib_find_accessor(h,loc);
if(!a){
if (!fail) {
sprintf(val,"undef");
} else {
grib_context_log(h->context, GRIB_LOG_WARNING,"grib_recompose_name: Problem to recompose filename with : %s ( %s no accessor found)", uname,loc);
return GRIB_NOT_FOUND;
}
} else {
switch (type) {
case GRIB_TYPE_STRING:
replen = sizeof(val)/sizeof(*val);
ret = grib_unpack_string(a,val,&replen);
break;
case GRIB_TYPE_DOUBLE:
replen=1;
ret = grib_unpack_double(a,&dval,&replen);
sprintf(val,"%g",dval);
break;
case GRIB_TYPE_LONG:
replen=1;
ret = grib_unpack_long(a,&lval,&replen);
sprintf(val,"%d",(int)lval);
break;
default:
grib_context_log(h->context, GRIB_LOG_WARNING,"grib_recompose_name: Problem to recompose filename with : %s, invalid type %d", loc,type);
}
grib_dependency_add(observer,a);
if((ret != GRIB_SUCCESS))
{
grib_context_log(h->context, GRIB_LOG_ERROR,"grib_recompose_name: Could not recompose filename : %s", uname);
return ret;
}
}
{
char* pc=fname;
while (*pc != '\0') pc++;
strcpy(pc,val);
}
/* sprintf(fname,"%s%s",fname,val); */
loc[0] = 0 ;
}
else
loc[mode++]=uname[i];
}
else if(uname[i]=='[')
mode = 0;
else {
int llen=strlen(fname);
fname[llen]=uname[i];
fname[llen+1]='\0';
/* sprintf(fname,"%s%c",fname, uname[i]); */
type=GRIB_TYPE_STRING;
}
}
/*fprintf(stdout,"parsed > %s\n",fname);*/
return GRIB_SUCCESS;
}
int grib_recompose_print(grib_handle* h, grib_accessor *observer, const char* uname, char* fname,int fail,FILE* out)
{
grib_accessor* a;
char loc[1024];
int i = 0;
int ret=0;
int maxcolsd=8;
int maxcols;
int mode = -1;
char val[1024];
char* sval=NULL;
char* p=NULL;
char* pp=NULL;
size_t size=0;
double* dval=0;
long* lval=0;
char double_format[]="%g"; /* default format for printing double keys */
char long_format[]="%ld"; /* default format for printing integer keys */
char buff[10]={0,};
char sbuf[200]={0,};
char buff1[1024]={0,};
char* format=NULL;
int type=-1;
int newline=1;
size_t replen = 0;
int l;
char* separator=NULL;
char default_separator[]=" ";
maxcols=maxcolsd;
loc[0] = 0 ;
fname[0] = 0 ;
for(i=0;i<strlen(uname);i++)
{
if(mode > -1)
{
switch (uname[i]) {
case ':':
type=grib_type_to_int(uname[i+1]);
i++;
break;
case '\'':
pp=(char*)(uname+i+1);
while(*pp!='%' && *pp!='!' && *pp!=']' && *pp!=':' && *pp!='\'') pp++;
l=pp-uname-i;
if (*pp == '\'') separator=strncpy(buff1,uname+i+1,l-1);
i+=l;
break;
case '%':
pp=(char*)(uname+i+1);
while(*pp!='%' && *pp!='!' && *pp!=']' && *pp!=':' && *pp!='\'') pp++;
l=pp-uname-i;
format=strncpy(buff,uname+i,l);
i+=l-1;
break;
case '!':
pp=(char*)uname;
maxcols=strtol(uname+i+1,&pp,10);
if (maxcols==0) maxcols=maxcolsd;
while(pp && *pp!='%' && *pp!='!' && *pp!=']' && *pp!=':' && *pp!='\'' ) pp++;
i+=pp-uname-i-1;
break;
case ']':
loc[mode] = 0;
mode = -1;
a = grib_find_accessor(h,loc);
if(!a){
if (!fail) {
fprintf(out,"undef");
ret=GRIB_NOT_FOUND;
} else {
grib_context_log(h->context, GRIB_LOG_WARNING,"grib_recompose_print: Problem to recompose print with : %s, no accessor found", loc);
return GRIB_NOT_FOUND;
}
} else {
if (type==-1) type=grib_accessor_get_native_type(a);
switch (type) {
case GRIB_TYPE_STRING:
replen=200;
ret = grib_get_string_internal(a->parent->h,a->name,sbuf,&replen);
fprintf(out,"%s",sbuf);
break;
case GRIB_TYPE_DOUBLE:
if (!format) format=double_format;
if (!separator) separator=default_separator;
ret=_grib_get_size(h,a,&size);
dval=(double*)grib_context_malloc_clear(h->context,sizeof(double)*size);
replen=0;
ret=_grib_get_double_array_internal(h,a,dval,size,&replen);
if (replen==1) fprintf(out,format,dval[0]);
else {
int i=0;
int cols=0;
for (i=0;i<replen;i++) {
newline=1;
fprintf(out,format,dval[i]);
if (i<replen-1) fprintf(out, "%s", separator);
cols++;
if (cols>=maxcols) {
fprintf(out,"\n");
newline=1;
cols=0;
}
}
}
grib_context_free( h->context,dval);
break;
case GRIB_TYPE_LONG:
if (!format) format=long_format;
if (!separator) separator=default_separator;
ret=_grib_get_size(h,a,&size);
lval=(long*)grib_context_malloc_clear(h->context,sizeof(long)*size);
replen=0;
ret=_grib_get_long_array_internal(h,a,lval,size,&replen);
if (replen==1) fprintf(out, format, lval[0]);
else {
int i=0;
int cols=0;
for (i=0;i<replen;i++) {
newline=1;
fprintf(out, format, lval[i]);
if (i<replen-1) fprintf(out, "%s", separator);
cols++;
if (cols>=maxcols) {
fprintf(out,"\n");
newline=1;
cols=0;
}
}
}
grib_context_free( h->context,lval);
break;
case GRIB_TYPE_BYTES:
replen=a->length;
sval=(char*)grib_context_malloc( h->context,replen*sizeof(char));
ret = grib_unpack_string(a,sval,&replen);
p=sval;
while ((replen--) > 0) fprintf(out,"%c",*(p++));
grib_context_free(h->context,sval);
newline=0;
break;
default:
grib_context_log(h->context, GRIB_LOG_WARNING,"grib_recompose_print: Problem to recompose print with : %s, invalid type %d", loc,type);
}
separator=NULL;
format=NULL;
if((ret != GRIB_SUCCESS))
{
/*
grib_context_log(h->context, GRIB_LOG_ERROR,"grib_recompose_print: Could not recompose print : %s", uname);
*/
return ret;
}
}
{
char* pc=fname;
while (*pc != '\0') pc++;
strcpy(pc,val);
}
loc[0] = 0 ;
break;
default:
loc[mode++]=uname[i];
}
} else if(uname[i]=='[') {
mode = 0;
} else {
fprintf(out,"%c",uname[i]);
type=-1;
}
}
if (newline) fprintf(out,"\n");
return ret;
}
grib_action_file* grib_find_action_file(const char* fname , grib_action_file_list* afl)
{
grib_action_file* act = afl->first;
while(act)
{
if(strcmp(act->filename,fname)==0)
return act;
act = act->next;
}
return 0;
}
void grib_push_action_file(grib_action_file* af, grib_action_file_list* afl)
{
if (!afl->first)
afl->first = afl->last = af;
else
afl->last->next = af;
afl->last = af;
}
#define MAXINCLUDE 10
typedef struct {
char *name;
FILE *file;
char *io_buffer;
int line;
} context;
static context stack[MAXINCLUDE];
static int top = 0;
extern FILE *grib_yyin;
extern int grib_yylineno;
extern void grib_yyrestart(FILE*);
static int error = 0;
int grib_yywrap()
{
/* int i; */
top--;
/* for(i = 0; i < top ; i++) printf(" "); */
/* printf("CLOSE %s\n",parse_file); */
fclose(stack[top].file);
/* if (stack[top].io_buffer) free(stack[top].io_buffer); */
grib_yylineno = stack[top].line;
if(top)
{
parse_file = stack[top-1].name;
grib_yyin = stack[top-1].file;
Assert(parse_file);
Assert(grib_yyin);
/* grib_yyrestart(grib_yyin); */
/* for(i = 0; i < top ; i++) printf(" "); */
/* printf("BACK TO %s\n",parse_file); */
grib_context_free(grib_parser_context,stack[top].name);
return 0;
}
else {
grib_context_free(grib_parser_context,stack[top].name);
parse_file = 0;
grib_yyin = NULL;
return 1;
}
}
int grib_yyerror(const char* msg)
{
grib_context_log(grib_parser_context, GRIB_LOG_ERROR,
"grib_parser: %s at line %d of %s", msg, grib_yylineno + 1,parse_file);
error = 1;
return 1;
}
void grib_parser_include(const char* fname)
{
FILE *f ;
char path[1204];
char* io_buffer=0;
/* int i; */
Assert(top < MAXINCLUDE);
Assert(fname);
if(parse_file == 0)
{
parse_file = fname;
Assert(top == 0);
}
else
{
const char *p = parse_file;
const char *q = NULL;
while(*p) {
if(*p == '/') q = p;
p++;
}
Assert(q);
q++;
strncpy(path,parse_file,q-parse_file);
path[q-parse_file] = 0;
strcat(path,fname);
Assert(*fname != '/');
parse_file = path;
}
f = fopen(parse_file,"r");
/* for(i = 0; i < top ; i++) printf(" "); */
/* printf("PARSING %s\n",parse_file); */
if(f == NULL)
{
char buffer[1024];
grib_context_log(grib_parser_context, (GRIB_LOG_ERROR)|(GRIB_LOG_PERROR),"grib_parser_include: cannot open: '%s'", parse_file);
sprintf(buffer,"Cannot include file: '%s'",parse_file);
grib_yyerror(buffer);
}
else
{
/*
c=grib_context_get_default();
if (c->io_buffer_size) {
if (posix_memalign(&(io_buffer),sysconf(_SC_PAGESIZE),c->io_buffer_size) ) {
grib_context_log(c,GRIB_LOG_FATAL,"grib_parser_include: posix_memalign unable to allocate io_buffer\n");
}
setvbuf(f,io_buffer,_IOFBF,c->io_buffer_size);
}
*/
grib_yyin = f;
stack[top].file = f;
stack[top].io_buffer = io_buffer;
stack[top].name = grib_context_strdup(grib_parser_context,parse_file);
parse_file = stack[top].name;
stack[top].line = grib_yylineno;
grib_yylineno = 0;
top++;
/* grib_yyrestart(f); */
}
}
extern int grib_yyparse(void);
static int parse(grib_context* gc,const char* filename)
{
int err = 0;
gc = gc ? gc : grib_context_get_default();
#ifdef YYDEBUG
extern int grib_yydebug;
grib_yydebug = getenv("YYDEBUG") != NULL;
#endif
grib_yyin = NULL;
top = 0;
parse_file = 0;
grib_parser_include(filename);
err = grib_yyparse();
parse_file = 0;
if (err) grib_context_log(gc,GRIB_LOG_ERROR,"Parsing error %d > %s\n",err, filename);
return err;
}
static grib_action* grib_parse_stream(grib_context* gc, const char* filename)
{
GRIB_PTHREAD_ONCE(&once,&init);
GRIB_MUTEX_LOCK(&mutex_stream);
grib_parser_all_actions = 0;
if (parse(gc,filename) == 0) {
if (grib_parser_all_actions) {
GRIB_MUTEX_UNLOCK(&mutex_stream)
return grib_parser_all_actions;
} else {
grib_action* ret=grib_action_create_noop(gc,filename);
GRIB_MUTEX_UNLOCK(&mutex_stream)
return ret;
}
} else {
GRIB_MUTEX_UNLOCK(&mutex_stream);
return NULL;
}
}
grib_concept_value* grib_parse_concept_file( grib_context* gc,const char* filename)
{
GRIB_PTHREAD_ONCE(&once,&init);
GRIB_MUTEX_LOCK(&mutex_concept);
gc = gc ? gc : grib_context_get_default();
grib_parser_context = gc;
if(parse(gc,filename) == 0) {
GRIB_MUTEX_UNLOCK(&mutex_concept);
return grib_parser_concept;
} else {
GRIB_MUTEX_UNLOCK(&mutex_concept);
return NULL;
}
}
grib_rule* grib_parse_rules_file( grib_context* gc,const char* filename)
{
if (!gc) gc=grib_context_get_default();
GRIB_PTHREAD_ONCE(&once,&init);
GRIB_MUTEX_LOCK(&mutex_rules);
gc = gc ? gc : grib_context_get_default();
grib_parser_context = gc;
if(parse(gc,filename) == 0) {
GRIB_MUTEX_UNLOCK(&mutex_rules);
return grib_parser_rules;
} else {
GRIB_MUTEX_UNLOCK(&mutex_rules);
return NULL;
}
}
grib_action* grib_parse_file( grib_context* gc,const char* filename)
{
grib_action_file* af;
GRIB_PTHREAD_ONCE(&once,&init);
GRIB_MUTEX_LOCK(&mutex_file);
af =0;
gc = gc ? gc : grib_context_get_default();
grib_parser_context = gc;
if(!gc->grib_reader)
gc->grib_reader =(grib_action_file_list*)grib_context_malloc_clear_persistent(gc,sizeof(grib_action_file_list));
else {
af = grib_find_action_file(filename, gc->grib_reader);
}
if(!af)
{
grib_action *a;
grib_context_log(gc,GRIB_LOG_DEBUG,"Loading %s",filename);
a = grib_parse_stream(gc,filename);
if(error)
{
#if 1
grib_free_action(gc,a);
GRIB_MUTEX_UNLOCK(&mutex_file);
return NULL;
#endif
a = NULL;
}
af =(grib_action_file*)grib_context_malloc_clear_persistent(gc,sizeof(grib_action_file));
af->root = a;
af->filename=grib_context_strdup_persistent(gc,filename);
grib_push_action_file(af,gc->grib_reader);
}
else grib_context_log(gc,GRIB_LOG_DEBUG,"Using cached version of %s",filename);
GRIB_MUTEX_UNLOCK(&mutex_file);
return af->root;
}
int grib_type_to_int(char id) {
switch (id) {
case 'd':
return GRIB_TYPE_DOUBLE;
break;
case 'f':
return GRIB_TYPE_DOUBLE;
break;
case 'l':
return GRIB_TYPE_LONG;
break;
case 'i':
return GRIB_TYPE_LONG;
break;
case 's':
return GRIB_TYPE_STRING;
break;
}
return GRIB_TYPE_UNDEFINED;
}