/* * Copyright 2005-2014 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 pthread_mutex_t mutex_parse = 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_mutex_init(&mutex_parse,&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 -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 -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=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=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; GRIB_PTHREAD_ONCE(&once,&init); GRIB_MUTEX_LOCK(&mutex_parse); #ifdef YYDEBUG { extern int grib_yydebug; grib_yydebug = getenv("YYDEBUG") != NULL; } #endif gc = gc ? gc : grib_context_get_default(); grib_yyin = NULL; top = 0; parse_file = 0; grib_parser_include(filename); if (!grib_yyin) { /* Could not read from file */ parse_file = 0; GRIB_MUTEX_UNLOCK(&mutex_parse); return GRIB_FILE_NOT_FOUND; } err = grib_yyparse(); parse_file = 0; if (err) grib_context_log(gc,GRIB_LOG_ERROR,"Parsing error %d > %s\n",err, filename); GRIB_MUTEX_UNLOCK(&mutex_parse); 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 if (a) 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; }