2013-03-25 12:04:10 +00:00
|
|
|
/*
|
2019-04-15 13:44:45 +00:00
|
|
|
* Copyright 2005-2019 ECMWF.
|
2013-03-25 12:04:10 +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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "grib_api_internal.h"
|
|
|
|
#include <errno.h>
|
2019-09-12 10:46:13 +00:00
|
|
|
#include <stdarg.h>
|
2013-03-25 12:04:10 +00:00
|
|
|
#include <stdlib.h>
|
2015-02-18 17:42:02 +00:00
|
|
|
#ifndef ECCODES_ON_WINDOWS
|
2013-03-25 12:04:10 +00:00
|
|
|
#include <unistd.h>
|
2013-03-25 14:23:07 +00:00
|
|
|
#else
|
|
|
|
#include <fcntl.h> /* Windows: for _O_BINARY */
|
|
|
|
#endif
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2015-07-07 08:45:39 +00:00
|
|
|
#ifdef ENABLE_FLOATING_POINT_EXCEPTIONS
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#include <fenv.h>
|
|
|
|
int feenableexcept(int excepts);
|
|
|
|
#endif
|
|
|
|
|
2013-03-25 12:04:10 +00:00
|
|
|
grib_string_list grib_file_not_found;
|
|
|
|
|
2020-01-10 13:10:30 +00:00
|
|
|
/* Windows always has a colon in pathnames e.g. C:\temp\file. So instead we use semi-colons as delimiter */
|
|
|
|
/* in order to have multiple definitions directories */
|
|
|
|
#ifdef ECCODES_ON_WINDOWS
|
|
|
|
# define DEFS_PATH_DELIMITER_CHAR ';'
|
|
|
|
# define DEFS_PATH_DELIMITER_STR ";"
|
|
|
|
#else
|
|
|
|
# define DEFS_PATH_DELIMITER_CHAR ':'
|
|
|
|
# define DEFS_PATH_DELIMITER_STR ":"
|
|
|
|
#endif
|
|
|
|
|
2013-03-25 12:04:10 +00:00
|
|
|
#if GRIB_PTHREADS
|
|
|
|
static pthread_once_t once = PTHREAD_ONCE_INIT;
|
|
|
|
|
|
|
|
static pthread_mutex_t mutex_mem = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
static pthread_mutex_t mutex_c = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static void init()
|
|
|
|
{
|
|
|
|
pthread_mutexattr_t attr;
|
|
|
|
pthread_mutexattr_init(&attr);
|
|
|
|
pthread_mutexattr_settype(&attr,PTHREAD_MUTEX_RECURSIVE);
|
|
|
|
pthread_mutex_init(&mutex_c,&attr);
|
|
|
|
pthread_mutex_init(&mutex_mem,&attr);
|
|
|
|
pthread_mutexattr_destroy(&attr);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
2015-12-30 14:39:02 +00:00
|
|
|
#elif GRIB_OMP_THREADS
|
|
|
|
static int once = 0;
|
|
|
|
static omp_nest_lock_t mutex_mem;
|
|
|
|
static omp_nest_lock_t mutex_c;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2015-12-30 14:39:02 +00:00
|
|
|
static void init()
|
|
|
|
{
|
|
|
|
GRIB_OMP_CRITICAL(lock_grib_context_c)
|
|
|
|
{
|
|
|
|
if (once == 0)
|
|
|
|
{
|
|
|
|
omp_init_nest_lock(&mutex_mem);
|
|
|
|
omp_init_nest_lock(&mutex_c);
|
|
|
|
once = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if MANAGE_MEM
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static void default_long_lasting_free(const grib_context* c, void* p)
|
|
|
|
{
|
|
|
|
free(p);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static void* default_long_lasting_malloc(const grib_context* c, size_t size)
|
|
|
|
{
|
|
|
|
void* ret;
|
|
|
|
ret=malloc(size);
|
2017-05-19 12:39:51 +00:00
|
|
|
if (!ret) {
|
2017-06-16 13:49:28 +00:00
|
|
|
grib_context_log(c,GRIB_LOG_FATAL,"default_long_lasting_malloc: error allocating %lu bytes",(unsigned long)size);
|
2017-05-19 12:39:51 +00:00
|
|
|
Assert(0);
|
|
|
|
}
|
2013-12-13 13:00:09 +00:00
|
|
|
return ret;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static void default_buffer_free(const grib_context* c, void* p)
|
|
|
|
{
|
|
|
|
free(p);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static void* default_buffer_malloc(const grib_context* c, size_t size)
|
|
|
|
{
|
|
|
|
void* ret;
|
|
|
|
ret=malloc(size);
|
2017-05-19 12:39:51 +00:00
|
|
|
if (!ret) {
|
|
|
|
grib_context_log(c,GRIB_LOG_FATAL,"default_buffer_malloc: error allocating %lu bytes",(unsigned long)size);
|
|
|
|
Assert(0);
|
|
|
|
}
|
2013-12-13 13:00:09 +00:00
|
|
|
return ret;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static void* default_buffer_realloc(const grib_context* c, void* p, size_t size)
|
|
|
|
{
|
|
|
|
void* ret;
|
|
|
|
ret=realloc(p,size);
|
2017-05-19 12:39:51 +00:00
|
|
|
if (!ret) {
|
|
|
|
grib_context_log(c,GRIB_LOG_FATAL,"default_buffer_realloc: error allocating %lu bytes",(unsigned long)size);
|
|
|
|
Assert(0);
|
|
|
|
}
|
2013-12-13 13:00:09 +00:00
|
|
|
return ret;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
2013-12-13 13:00:09 +00:00
|
|
|
|
|
|
|
static void default_free(const grib_context* c, void* p)
|
|
|
|
{
|
|
|
|
free(p);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static void* default_malloc(const grib_context* c, size_t size)
|
|
|
|
{
|
|
|
|
void* ret;
|
|
|
|
ret=malloc(size);
|
2017-05-19 12:39:51 +00:00
|
|
|
if (!ret) {
|
|
|
|
grib_context_log(c,GRIB_LOG_FATAL,"default_malloc: error allocating %lu bytes",(unsigned long)size);
|
|
|
|
Assert(0);
|
|
|
|
}
|
2013-12-13 13:00:09 +00:00
|
|
|
return ret;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static void* default_realloc(const grib_context* c, void* p, size_t size)
|
|
|
|
{
|
|
|
|
void* ret;
|
|
|
|
ret=realloc(p,size);
|
2017-05-19 12:39:51 +00:00
|
|
|
if (!ret) {
|
|
|
|
grib_context_log(c,GRIB_LOG_FATAL,"default_realloc: error allocating %lu bytes",(unsigned long)size);
|
|
|
|
Assert(0);
|
|
|
|
}
|
2013-12-13 13:00:09 +00:00
|
|
|
return ret;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
2013-12-13 13:00:09 +00:00
|
|
|
#endif
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static size_t default_read(const grib_context* c, void *ptr, size_t size, void *stream)
|
|
|
|
{
|
2014-10-02 15:50:49 +00:00
|
|
|
return fread(ptr, 1, size, (FILE*)stream);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static off_t default_tell(const grib_context* c, void *stream)
|
|
|
|
{
|
2014-10-02 15:50:49 +00:00
|
|
|
return ftello((FILE*)stream);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
2013-12-13 13:00:09 +00:00
|
|
|
|
|
|
|
static off_t default_seek(const grib_context* c, off_t offset,int whence, void *stream)
|
|
|
|
{
|
2014-10-02 15:50:49 +00:00
|
|
|
return fseeko((FILE*)stream,offset,whence);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static int default_feof(const grib_context* c, void *stream)
|
|
|
|
{
|
|
|
|
return feof((FILE*)stream);
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static size_t default_write(const grib_context* c,const void *ptr, size_t size, void *stream)
|
|
|
|
{
|
2014-10-02 15:50:49 +00:00
|
|
|
return fwrite(ptr, 1, size, (FILE*)stream);
|
2013-12-13 13:00:09 +00:00
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
size_t grib_context_read(const grib_context* c, void *ptr, size_t size, void *stream)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
return c->read(c,ptr, size, stream);
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
off_t grib_context_tell(const grib_context* c, void *stream)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
return c->tell(c,stream);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
int grib_context_seek(const grib_context* c, off_t offset ,int whence , void *stream)
|
2013-03-25 12:04:10 +00:00
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
return c->seek(c,offset,whence,stream);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
int grib_context_eof(const grib_context* c, void *stream)
|
2013-03-25 12:04:10 +00:00
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
return c->eof(c,stream);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
size_t grib_context_write(const grib_context* c,const void *ptr, size_t size, void *stream)
|
2013-03-25 12:04:10 +00:00
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
return c->write(c,ptr, size, stream);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static void default_log(const grib_context* c, int level, const char* mess)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
if(level == GRIB_LOG_ERROR) {
|
2015-10-15 10:52:22 +00:00
|
|
|
fprintf(c->log_stream, "ECCODES ERROR : %s\n", mess);
|
2013-12-13 13:00:09 +00:00
|
|
|
/*Assert(1==0);*/
|
|
|
|
}
|
2015-10-15 10:52:22 +00:00
|
|
|
if(level == GRIB_LOG_FATAL) fprintf(c->log_stream, "ECCODES ERROR : %s\n", mess);
|
|
|
|
if(level == GRIB_LOG_DEBUG && c->debug>0) fprintf(c->log_stream, "ECCODES DEBUG : %s\n", mess);
|
|
|
|
if(level == GRIB_LOG_WARNING) fprintf(c->log_stream, "ECCODES WARNING : %s\n", mess);
|
|
|
|
if(level == GRIB_LOG_INFO) fprintf(c->log_stream, "ECCODES INFO : %s\n", mess);
|
2013-12-13 13:00:09 +00:00
|
|
|
|
|
|
|
if(level == GRIB_LOG_FATAL) { Assert(0);}
|
|
|
|
|
2015-01-23 22:47:51 +00:00
|
|
|
if(getenv("ECCODES_FAIL_IF_LOG_MESSAGE"))
|
2013-12-13 13:00:09 +00:00
|
|
|
{
|
2015-01-23 22:47:51 +00:00
|
|
|
long n = atol(getenv("ECCODES_FAIL_IF_LOG_MESSAGE"));
|
2013-12-13 13:00:09 +00:00
|
|
|
if(n >= 1 && level == GRIB_LOG_ERROR) Assert(0);
|
|
|
|
if(n >= 2 && level == GRIB_LOG_WARNING) Assert(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void default_print(const grib_context* c, void* descriptor, const char* mess)
|
2013-03-25 12:04:10 +00:00
|
|
|
{
|
2014-10-02 15:50:49 +00:00
|
|
|
fprintf((FILE*)descriptor, "%s", mess);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void grib_context_set_print_proc(grib_context* c, grib_print_proc p)
|
|
|
|
{
|
|
|
|
c = c ? c : grib_context_get_default();
|
|
|
|
c->print = p;
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void grib_context_set_debug(grib_context* c, int mode)
|
|
|
|
{
|
|
|
|
c = c ? c : grib_context_get_default();
|
|
|
|
c->debug = mode;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void grib_context_set_logging_proc(grib_context* c, grib_log_proc p)
|
|
|
|
{
|
|
|
|
c = c ? c : grib_context_get_default();
|
|
|
|
c->output_log = p;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
long grib_get_api_version()
|
|
|
|
{
|
2015-02-25 17:42:04 +00:00
|
|
|
return ECCODES_VERSION;
|
2013-12-13 13:00:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void grib_print_api_version(FILE* out)
|
|
|
|
{
|
|
|
|
fprintf(out,"%d.%d.%d",
|
2015-02-25 17:42:04 +00:00
|
|
|
ECCODES_MAJOR_VERSION,
|
|
|
|
ECCODES_MINOR_VERSION,
|
|
|
|
ECCODES_REVISION_VERSION);
|
2019-03-27 12:33:27 +00:00
|
|
|
|
2019-03-26 12:02:57 +00:00
|
|
|
if (ECCODES_MAJOR_VERSION < 1) {
|
|
|
|
fprintf(out, "%s", " PRE-RELEASE");
|
|
|
|
}
|
2013-12-13 13:00:09 +00:00
|
|
|
}
|
|
|
|
|
2016-11-11 16:41:38 +00:00
|
|
|
const char* grib_get_package_name()
|
|
|
|
{
|
|
|
|
return "ecCodes";
|
|
|
|
}
|
|
|
|
|
2017-06-22 14:06:57 +00:00
|
|
|
#define DEFAULT_FILE_POOL_MAX_OPENED_FILES 200
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static grib_context default_grib_context = {
|
|
|
|
0, /* inited */
|
|
|
|
0, /* debug */
|
|
|
|
0, /* write_on_fail */
|
|
|
|
0, /* no_abort */
|
|
|
|
0, /* io_buffer_size */
|
|
|
|
0, /* no_big_group_split */
|
|
|
|
0, /* no_spd */
|
|
|
|
0, /* keep_matrix */
|
|
|
|
0, /* grib_definition_files_path */
|
|
|
|
0, /* grib_samples_path */
|
|
|
|
0, /* grib_concept_path */
|
|
|
|
0, /* grib_reader */
|
|
|
|
0, /* user data */
|
|
|
|
GRIB_REAL_MODE8, /* real mode for fortran */
|
2013-03-25 12:04:10 +00:00
|
|
|
|
|
|
|
#if MANAGE_MEM
|
2013-12-13 13:00:09 +00:00
|
|
|
&grib_transient_free, /* free_mem */
|
|
|
|
&grib_transient_malloc, /* alloc_mem */
|
|
|
|
&grib_transient_realloc, /* realloc_mem */
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
&grib_permanent_free, /* free_persistant_mem */
|
|
|
|
&grib_permanent_malloc, /* alloc_persistant_mem */
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
&grib_buffer_free, /* buffer_free_mem */
|
|
|
|
&grib_buffer_malloc, /* buffer_alloc_mem */
|
|
|
|
&grib_buffer_realloc, /* buffer_realloc_mem */
|
2013-03-25 12:04:10 +00:00
|
|
|
|
|
|
|
#else
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
&default_free, /* free_mem */
|
|
|
|
&default_malloc, /* alloc_mem */
|
|
|
|
&default_realloc, /* realloc_mem */
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
&default_long_lasting_free, /* free_persistant_mem */
|
|
|
|
&default_long_lasting_malloc, /* alloc_persistant_mem */
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
&default_buffer_free, /* free_buffer_mem */
|
|
|
|
&default_buffer_malloc, /* alloc_buffer_mem */
|
|
|
|
&default_buffer_realloc, /* realloc_buffer_mem */
|
2013-03-25 12:04:10 +00:00
|
|
|
#endif
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
&default_read, /* file read procedure */
|
|
|
|
&default_write, /* file write procedure */
|
|
|
|
&default_tell, /* lfile tell procedure */
|
|
|
|
&default_seek, /* lfile seek procedure */
|
|
|
|
&default_feof, /* file feof procedure */
|
|
|
|
|
2019-10-10 17:21:05 +00:00
|
|
|
&default_log, /* output_log */
|
|
|
|
&default_print, /* print */
|
|
|
|
0, /* codetable */
|
|
|
|
0, /* smart_table */
|
|
|
|
0, /* outfilename */
|
|
|
|
0, /* multi_support_on */
|
|
|
|
0, /* multi_support */
|
2013-12-13 13:00:09 +00:00
|
|
|
0, /* grib_definition_files_dir */
|
|
|
|
0, /* handle_file_count */
|
|
|
|
0, /* handle_total_count */
|
|
|
|
0, /* message_file_offset */
|
|
|
|
0, /* no_fail_on_wrong_length */
|
|
|
|
0, /* gts_header_on */
|
|
|
|
0, /* gribex_mode_on */
|
|
|
|
0, /* large_constant_fields */
|
2019-10-10 17:21:05 +00:00
|
|
|
0, /* keys */
|
2013-12-13 13:00:09 +00:00
|
|
|
0, /* keys_count */
|
2019-10-10 17:21:05 +00:00
|
|
|
0, /* concepts_index */
|
2013-12-13 13:00:09 +00:00
|
|
|
0, /* concepts_count */
|
|
|
|
{0,}, /* concepts */
|
2016-09-30 14:50:57 +00:00
|
|
|
0, /* hash_array_index */
|
|
|
|
0, /* hash_array_count */
|
|
|
|
{0,}, /* hash_array */
|
2013-12-13 13:00:09 +00:00
|
|
|
0, /* def_files */
|
|
|
|
0, /* blacklist */
|
2016-09-30 14:50:57 +00:00
|
|
|
0, /* ieee_packing */
|
2016-09-30 17:28:40 +00:00
|
|
|
0, /* bufrdc_mode */
|
2016-11-28 16:17:27 +00:00
|
|
|
0, /* bufr_set_to_missing_if_out_of_range */
|
2019-07-31 17:52:53 +00:00
|
|
|
0, /* bufr_multi_element_constant_arrays */
|
2019-10-10 17:21:05 +00:00
|
|
|
0, /* grib_data_quality_checks */
|
2013-12-13 13:00:09 +00:00
|
|
|
0, /* log_stream */
|
2016-09-30 14:50:57 +00:00
|
|
|
0, /* classes */
|
2017-05-31 16:21:18 +00:00
|
|
|
0, /* lists */
|
2017-06-22 14:06:57 +00:00
|
|
|
0, /* expanded_descriptors */
|
|
|
|
DEFAULT_FILE_POOL_MAX_OPENED_FILES /* file_pool_max_opened_files */
|
2013-03-25 12:04:10 +00:00
|
|
|
#if GRIB_PTHREADS
|
2016-09-30 14:50:57 +00:00
|
|
|
,PTHREAD_MUTEX_INITIALIZER /* mutex */
|
2013-03-25 12:04:10 +00:00
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2015-06-08 17:05:29 +00:00
|
|
|
/* Hopefully big enough. Note: GRIB_DEFINITION_PATH can contain SEVERAL colon-separated sub-paths */
|
|
|
|
#define DEF_PATH_MAXLEN 8192
|
|
|
|
|
2013-06-19 12:13:45 +00:00
|
|
|
grib_context* grib_context_get_default()
|
|
|
|
{
|
2015-12-30 11:55:03 +00:00
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
2015-07-07 08:45:39 +00:00
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
|
2013-06-19 12:13:45 +00:00
|
|
|
if(!default_grib_context.inited)
|
|
|
|
{
|
2016-09-30 17:28:40 +00:00
|
|
|
const char* write_on_fail = NULL;
|
|
|
|
const char* large_constant_fields = NULL;
|
|
|
|
const char* no_abort = NULL;
|
|
|
|
const char* debug = NULL;
|
|
|
|
const char* gribex = NULL;
|
|
|
|
const char* ieee_packing = NULL;
|
|
|
|
const char* io_buffer_size = NULL;
|
|
|
|
const char* log_stream = NULL;
|
|
|
|
const char* no_big_group_split = NULL;
|
|
|
|
const char* no_spd = NULL;
|
|
|
|
const char* keep_matrix = NULL;
|
|
|
|
const char* bufrdc_mode = NULL;
|
2016-11-28 16:17:27 +00:00
|
|
|
const char* bufr_set_to_missing_if_out_of_range = NULL;
|
2019-07-31 17:52:53 +00:00
|
|
|
const char* bufr_multi_element_constant_arrays = NULL;
|
2019-10-10 17:21:05 +00:00
|
|
|
const char* grib_data_quality_checks = NULL;
|
2017-06-22 14:06:57 +00:00
|
|
|
const char* file_pool_max_opened_files = NULL;
|
2013-06-19 12:13:45 +00:00
|
|
|
|
2018-11-13 10:03:30 +00:00
|
|
|
#ifdef ENABLE_FLOATING_POINT_EXCEPTIONS
|
|
|
|
feenableexcept(FE_ALL_EXCEPT & ~FE_INEXACT);
|
|
|
|
#endif
|
|
|
|
|
2016-01-11 16:36:46 +00:00
|
|
|
write_on_fail = codes_getenv("ECCODES_GRIB_WRITE_ON_FAIL");
|
2019-07-31 17:52:53 +00:00
|
|
|
bufrdc_mode = getenv("ECCODES_BUFRDC_MODE_ON");
|
|
|
|
bufr_set_to_missing_if_out_of_range = getenv("ECCODES_BUFR_SET_TO_MISSING_IF_OUT_OF_RANGE");
|
|
|
|
bufr_multi_element_constant_arrays = getenv("ECCODES_BUFR_MULTI_ELEMENT_CONSTANT_ARRAYS");
|
2019-10-10 17:21:05 +00:00
|
|
|
grib_data_quality_checks = getenv("ECCODES_GRIB_DATA_QUALITY_CHECKS");
|
2016-01-11 16:36:46 +00:00
|
|
|
large_constant_fields = codes_getenv("ECCODES_GRIB_LARGE_CONSTANT_FIELDS");
|
|
|
|
no_abort = codes_getenv("ECCODES_NO_ABORT");
|
|
|
|
debug = codes_getenv("ECCODES_DEBUG");
|
|
|
|
gribex = codes_getenv("ECCODES_GRIBEX_MODE_ON");
|
|
|
|
ieee_packing = codes_getenv("ECCODES_GRIB_IEEE_PACKING");
|
|
|
|
io_buffer_size = codes_getenv("ECCODES_IO_BUFFER_SIZE");
|
|
|
|
log_stream = codes_getenv("ECCODES_LOG_STREAM");
|
|
|
|
no_big_group_split = codes_getenv("ECCODES_GRIB_NO_BIG_GROUP_SPLIT");
|
|
|
|
no_spd = codes_getenv("ECCODES_GRIB_NO_SPD");
|
|
|
|
keep_matrix = codes_getenv("ECCODES_GRIB_KEEP_MATRIX");
|
2019-07-31 17:52:53 +00:00
|
|
|
file_pool_max_opened_files = getenv("ECCODES_FILE_POOL_MAX_OPENED_FILES");
|
2013-06-19 12:13:45 +00:00
|
|
|
|
|
|
|
/* On UNIX, when we read from a file we get exactly what is in the file on disk.
|
|
|
|
* But on Windows a file can be opened in binary or text mode. In binary mode the system behaves exactly as in UNIX.
|
|
|
|
*/
|
2015-02-18 17:42:02 +00:00
|
|
|
#ifdef ECCODES_ON_WINDOWS
|
2013-06-19 12:13:45 +00:00
|
|
|
_set_fmode(_O_BINARY);
|
2013-03-25 14:23:07 +00:00
|
|
|
#endif
|
|
|
|
|
2013-06-19 12:13:45 +00:00
|
|
|
default_grib_context.inited = 1;
|
|
|
|
default_grib_context.io_buffer_size = io_buffer_size ? atoi(io_buffer_size) : 0;
|
|
|
|
default_grib_context.no_big_group_split = no_big_group_split ? atoi(no_big_group_split) : 0;
|
|
|
|
default_grib_context.no_spd = no_spd ? atoi(no_spd) : 0;
|
|
|
|
default_grib_context.keep_matrix = keep_matrix ? atoi(keep_matrix) : 1;
|
|
|
|
default_grib_context.write_on_fail = write_on_fail ? atoi(write_on_fail) : 0;
|
|
|
|
default_grib_context.no_abort = no_abort ? atoi(no_abort) : 0;
|
|
|
|
default_grib_context.debug = debug ? atoi(debug) : 0;
|
|
|
|
default_grib_context.gribex_mode_on=gribex ? atoi(gribex) : 0;
|
|
|
|
default_grib_context.large_constant_fields = large_constant_fields ? atoi(large_constant_fields) : 0;
|
|
|
|
default_grib_context.ieee_packing=ieee_packing ? atoi(ieee_packing) : 0;
|
2016-01-11 16:36:46 +00:00
|
|
|
default_grib_context.grib_samples_path = codes_getenv("ECCODES_SAMPLES_PATH");
|
2013-06-19 12:13:45 +00:00
|
|
|
default_grib_context.log_stream=stderr;
|
|
|
|
if (!log_stream) {
|
|
|
|
default_grib_context.log_stream=stderr;
|
|
|
|
} else if (!strcmp(log_stream,"stderr") ) {
|
|
|
|
default_grib_context.log_stream=stderr;
|
|
|
|
} else if (!strcmp(log_stream,"stdout") ) {
|
|
|
|
default_grib_context.log_stream=stdout;
|
|
|
|
}
|
|
|
|
|
2015-01-24 22:19:32 +00:00
|
|
|
#ifdef ECCODES_SAMPLES_PATH
|
2013-06-19 12:13:45 +00:00
|
|
|
if(!default_grib_context.grib_samples_path)
|
2015-01-24 22:19:32 +00:00
|
|
|
default_grib_context.grib_samples_path = ECCODES_SAMPLES_PATH ;
|
2013-03-25 12:04:10 +00:00
|
|
|
#endif
|
2013-06-19 12:13:45 +00:00
|
|
|
|
2016-01-11 16:36:46 +00:00
|
|
|
default_grib_context.grib_definition_files_path = codes_getenv("ECCODES_DEFINITION_PATH");
|
2015-01-24 22:19:32 +00:00
|
|
|
#ifdef ECCODES_DEFINITION_PATH
|
2014-07-10 10:34:39 +00:00
|
|
|
if(!default_grib_context.grib_definition_files_path) {
|
2015-01-24 22:19:32 +00:00
|
|
|
default_grib_context.grib_definition_files_path = ECCODES_DEFINITION_PATH ;
|
2014-07-10 10:34:39 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Temp bug fix when putenv() is called from program that moves getenv() stuff around */
|
|
|
|
default_grib_context.grib_definition_files_path = strdup(default_grib_context.grib_definition_files_path);
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
#endif
|
|
|
|
|
2015-06-08 17:05:29 +00:00
|
|
|
/* GRIB-779: Special case for ECMWF testing. Not for external use! */
|
|
|
|
/* Append the new path to our existing path */
|
|
|
|
{
|
2016-01-11 16:36:46 +00:00
|
|
|
const char* test_defs = codes_getenv("_ECCODES_ECMWF_TEST_DEFINITION_PATH");
|
|
|
|
const char* test_samp = codes_getenv("_ECCODES_ECMWF_TEST_SAMPLES_PATH");
|
2015-06-08 17:05:29 +00:00
|
|
|
if (test_defs) {
|
|
|
|
char buffer[DEF_PATH_MAXLEN];
|
|
|
|
strcpy(buffer, default_grib_context.grib_definition_files_path);
|
|
|
|
strcat(buffer, ":");
|
|
|
|
strcat(buffer, strdup(test_defs));
|
|
|
|
default_grib_context.grib_definition_files_path = strdup(buffer);
|
|
|
|
}
|
|
|
|
if (test_samp) {
|
|
|
|
char buffer[DEF_PATH_MAXLEN];
|
|
|
|
strcpy(buffer, default_grib_context.grib_samples_path);
|
|
|
|
strcat(buffer, ":");
|
|
|
|
strcat(buffer, strdup(test_samp));
|
|
|
|
default_grib_context.grib_samples_path = strdup(buffer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-29 14:45:55 +00:00
|
|
|
/* Definitions path extra: Added at the head of (i.e. before) existing path */
|
2019-11-28 15:53:46 +00:00
|
|
|
{
|
2019-11-29 14:45:55 +00:00
|
|
|
const char* defs_extra = getenv("ECCODES_EXTRA_DEFINITION_PATH");
|
|
|
|
if (defs_extra) {
|
2019-11-28 15:53:46 +00:00
|
|
|
char buffer[DEF_PATH_MAXLEN];
|
2020-01-10 13:10:30 +00:00
|
|
|
ecc_snprintf(buffer, DEF_PATH_MAXLEN, "%s%c%s", defs_extra, DEFS_PATH_DELIMITER_CHAR, default_grib_context.grib_definition_files_path);
|
2019-11-28 15:53:46 +00:00
|
|
|
default_grib_context.grib_definition_files_path = strdup(buffer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-19 12:13:45 +00:00
|
|
|
grib_context_log(&default_grib_context, GRIB_LOG_DEBUG, "Definitions path: %s",
|
|
|
|
default_grib_context.grib_definition_files_path);
|
|
|
|
grib_context_log(&default_grib_context, GRIB_LOG_DEBUG, "Samples path: %s",
|
|
|
|
default_grib_context.grib_samples_path);
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-06-19 12:13:45 +00:00
|
|
|
default_grib_context.keys_count=0;
|
|
|
|
default_grib_context.keys=grib_hash_keys_new(&(default_grib_context),
|
|
|
|
&(default_grib_context.keys_count));
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-06-19 12:13:45 +00:00
|
|
|
default_grib_context.concepts_index=grib_itrie_new(&(default_grib_context),
|
|
|
|
&(default_grib_context.concepts_count));
|
2014-07-10 10:34:39 +00:00
|
|
|
default_grib_context.hash_array_index=grib_itrie_new(&(default_grib_context),
|
|
|
|
&(default_grib_context.hash_array_count));
|
2013-06-19 12:13:45 +00:00
|
|
|
default_grib_context.def_files=grib_trie_new(&(default_grib_context));
|
2014-07-10 10:34:39 +00:00
|
|
|
default_grib_context.lists=grib_trie_new(&(default_grib_context));
|
2013-06-19 12:13:45 +00:00
|
|
|
default_grib_context.classes=grib_trie_new(&(default_grib_context));
|
2016-09-30 17:28:40 +00:00
|
|
|
default_grib_context.bufrdc_mode = bufrdc_mode ? atoi(bufrdc_mode) : 0;
|
2017-06-22 14:06:57 +00:00
|
|
|
default_grib_context.bufr_set_to_missing_if_out_of_range = bufr_set_to_missing_if_out_of_range ?
|
|
|
|
atoi(bufr_set_to_missing_if_out_of_range) : 0;
|
2019-07-31 17:52:53 +00:00
|
|
|
default_grib_context.bufr_multi_element_constant_arrays = bufr_multi_element_constant_arrays ?
|
|
|
|
atoi(bufr_multi_element_constant_arrays) : 0;
|
2019-10-10 17:21:05 +00:00
|
|
|
default_grib_context.grib_data_quality_checks = grib_data_quality_checks ?
|
|
|
|
atoi(grib_data_quality_checks) : 0;
|
2017-06-22 14:06:57 +00:00
|
|
|
default_grib_context.file_pool_max_opened_files = file_pool_max_opened_files ?
|
|
|
|
atoi(file_pool_max_opened_files) : DEFAULT_FILE_POOL_MAX_OPENED_FILES;
|
2013-06-19 12:13:45 +00:00
|
|
|
}
|
|
|
|
|
2015-07-07 08:45:39 +00:00
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
2013-06-19 12:13:45 +00:00
|
|
|
return &default_grib_context;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2015-12-31 13:23:48 +00:00
|
|
|
#if 0 /* function removed */
|
2013-03-25 12:04:10 +00:00
|
|
|
grib_context* grib_context_new(grib_context* parent)
|
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
grib_context* c;
|
2013-03-25 12:04:10 +00:00
|
|
|
#if GRIB_PTHREADS
|
2013-12-13 13:00:09 +00:00
|
|
|
pthread_mutexattr_t attr;
|
2013-03-25 12:04:10 +00:00
|
|
|
#endif
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
if (!parent) parent=grib_context_get_default();
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2015-12-30 11:55:03 +00:00
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
2013-12-13 13:00:09 +00:00
|
|
|
GRIB_MUTEX_LOCK(&(parent->mutex));
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
c = (grib_context*)grib_context_malloc_clear_persistent(&default_grib_context,sizeof(grib_context));
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
c->inited = default_grib_context.inited;
|
|
|
|
c->debug = default_grib_context.debug;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
c->real_mode = default_grib_context.real_mode;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
c->free_mem = default_grib_context.free_mem;
|
|
|
|
c->alloc_mem = default_grib_context.alloc_mem;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
c->free_persistent_mem = default_grib_context.free_persistent_mem;
|
|
|
|
c->alloc_persistent_mem= default_grib_context.alloc_persistent_mem;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
c->read = default_grib_context.read;
|
|
|
|
c->write = default_grib_context.write;
|
|
|
|
c->tell = default_grib_context.tell;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
c->output_log = default_grib_context.output_log;
|
|
|
|
c->print = default_grib_context.print ;
|
|
|
|
c->user_data = default_grib_context.user_data;
|
|
|
|
c->def_files = default_grib_context.def_files;
|
2014-07-10 10:34:39 +00:00
|
|
|
c->lists = default_grib_context.lists;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
|
|
|
#if GRIB_PTHREADS
|
2013-12-13 13:00:09 +00:00
|
|
|
pthread_mutexattr_settype(&attr,PTHREAD_MUTEX_RECURSIVE);
|
|
|
|
pthread_mutex_init(&mutex_c,&attr);
|
|
|
|
pthread_mutexattr_destroy(&attr);
|
2013-03-25 12:04:10 +00:00
|
|
|
#endif
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
GRIB_MUTEX_UNLOCK(&(parent->mutex));
|
|
|
|
return c;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
2015-12-31 13:23:48 +00:00
|
|
|
#endif /* function removed */
|
2013-03-25 12:04:10 +00:00
|
|
|
|
|
|
|
/* GRIB-235: Resolve path to expand symbolic links etc */
|
|
|
|
static char* resolve_path(grib_context* c, char* path)
|
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
char* result = NULL;
|
2015-02-18 17:42:02 +00:00
|
|
|
#ifdef ECCODES_ON_WINDOWS
|
2013-12-13 13:00:09 +00:00
|
|
|
result = grib_context_strdup(c, path);
|
2013-03-25 14:23:07 +00:00
|
|
|
#else
|
2013-12-13 13:00:09 +00:00
|
|
|
char resolved[DEF_PATH_MAXLEN+1];
|
|
|
|
if (!realpath(path, resolved)) {
|
|
|
|
result = grib_context_strdup(c, path); /* Failed to resolve. Use original path */
|
|
|
|
} else {
|
|
|
|
result = grib_context_strdup(c, resolved);
|
|
|
|
}
|
2013-03-25 14:23:07 +00:00
|
|
|
#endif
|
2013-12-13 13:00:09 +00:00
|
|
|
return result;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
static int init_definition_files_dir(grib_context* c)
|
|
|
|
{
|
|
|
|
int err=0;
|
|
|
|
char path[DEF_PATH_MAXLEN];
|
|
|
|
char* p=NULL;
|
|
|
|
grib_string_list* next=NULL;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
if (!c) c=grib_context_get_default();
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
if (c->grib_definition_files_dir) return 0;
|
|
|
|
if (!c->grib_definition_files_path) return GRIB_NO_DEFINITIONS;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
/* Note: strtok modifies its first argument so we copy */
|
|
|
|
strncpy(path, c->grib_definition_files_path, DEF_PATH_MAXLEN);
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2015-12-30 11:55:03 +00:00
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
2013-12-13 13:00:09 +00:00
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
|
|
|
|
p=path;
|
|
|
|
|
|
|
|
while(*p!=DEFS_PATH_DELIMITER_CHAR && *p!='\0') p++;
|
|
|
|
|
|
|
|
if (*p != DEFS_PATH_DELIMITER_CHAR) {
|
|
|
|
/* No delimiter found so this is a single directory */
|
|
|
|
c->grib_definition_files_dir=(grib_string_list*)grib_context_malloc_clear_persistent(c,sizeof(grib_string_list));
|
|
|
|
c->grib_definition_files_dir->value = resolve_path(c, path);
|
|
|
|
} else {
|
|
|
|
/* Definitions path contains multiple directories */
|
2015-07-07 08:45:39 +00:00
|
|
|
char* dir=NULL;
|
2013-12-13 13:00:09 +00:00
|
|
|
dir=strtok(path, DEFS_PATH_DELIMITER_STR);
|
|
|
|
|
|
|
|
while (dir != NULL) {
|
|
|
|
if (next) {
|
|
|
|
next->next=(grib_string_list*)grib_context_malloc_clear_persistent(c,sizeof(grib_string_list));
|
|
|
|
next=next->next;
|
|
|
|
} else {
|
|
|
|
c->grib_definition_files_dir=(grib_string_list*)grib_context_malloc_clear_persistent(c,sizeof(grib_string_list));
|
|
|
|
next=c->grib_definition_files_dir;
|
|
|
|
}
|
|
|
|
next->value = resolve_path(c, dir);
|
|
|
|
dir=strtok(NULL, DEFS_PATH_DELIMITER_STR);
|
|
|
|
}
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
return err;
|
|
|
|
}
|
2013-03-25 14:23:07 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
char *grib_context_full_defs_path(grib_context* c,const char* basename)
|
|
|
|
{
|
|
|
|
int err=0;
|
|
|
|
char full[1024]={0,};
|
|
|
|
grib_string_list* dir=NULL;
|
|
|
|
grib_string_list* fullpath=0;
|
|
|
|
if (!c) c=grib_context_get_default();
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2015-12-30 11:55:03 +00:00
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
if(*basename == '/' || *basename == '.') {
|
|
|
|
return (char*)basename;
|
|
|
|
} else {
|
2018-05-03 16:52:37 +00:00
|
|
|
GRIB_MUTEX_LOCK(&mutex_c); /* See ECC-604 */
|
2014-10-02 15:50:49 +00:00
|
|
|
fullpath=(grib_string_list*)grib_trie_get(c->def_files,basename);
|
2018-05-03 16:52:37 +00:00
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
2013-12-13 13:00:09 +00:00
|
|
|
if (fullpath!=NULL) {
|
|
|
|
return fullpath->value;
|
|
|
|
}
|
|
|
|
if (!c->grib_definition_files_dir) {
|
|
|
|
err=init_definition_files_dir(c);
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
if (err != GRIB_SUCCESS) {
|
|
|
|
grib_context_log(c,GRIB_LOG_ERROR,
|
|
|
|
"Unable to find definition files directory");
|
|
|
|
return NULL;
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
dir=c->grib_definition_files_dir;
|
|
|
|
|
|
|
|
while (dir) {
|
|
|
|
sprintf(full,"%s/%s",dir->value,basename);
|
2016-06-19 08:22:15 +00:00
|
|
|
if (!codes_access(full,F_OK)) {
|
2014-10-02 15:50:49 +00:00
|
|
|
fullpath=(grib_string_list*)grib_context_malloc_clear_persistent(c,sizeof(grib_string_list));
|
2013-12-13 13:00:09 +00:00
|
|
|
Assert(fullpath);
|
|
|
|
fullpath->value=grib_context_strdup(c,full);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
grib_trie_insert(c->def_files,basename,fullpath);
|
|
|
|
grib_context_log(c,GRIB_LOG_DEBUG,"Found def file %s",full);
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
return fullpath->value;
|
|
|
|
}
|
|
|
|
dir=dir->next;
|
|
|
|
}
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
/* Store missing files so we don't check for them again and again */
|
|
|
|
grib_trie_insert(c->def_files,basename,&grib_file_not_found);
|
|
|
|
/*grib_context_log(c,GRIB_LOG_ERROR,"Def file \"%s\" not found",basename);*/
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
full[0]=0;
|
|
|
|
return NULL;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2014-06-18 16:14:01 +00:00
|
|
|
char* grib_samples_path(const grib_context *c)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
return c->grib_samples_path;
|
|
|
|
}
|
2016-03-16 18:10:40 +00:00
|
|
|
char* grib_definition_path(const grib_context *c)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
return c->grib_definition_files_path;
|
|
|
|
}
|
2014-06-18 16:14:01 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void grib_context_free(const grib_context* c, void* p)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
if(p) c->free_mem(c,p);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void grib_context_free_persistent(const grib_context* c, void* p)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
if(p) c->free_persistent_mem(c,p);
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void grib_context_reset(grib_context* c)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
if(c->grib_reader)
|
|
|
|
{
|
|
|
|
grib_action_file *fr = c->grib_reader->first;
|
|
|
|
grib_action_file *fn = fr;
|
|
|
|
grib_action* a;
|
|
|
|
|
|
|
|
while(fn){
|
|
|
|
fr = fn;
|
|
|
|
fn = fn->next;
|
|
|
|
|
|
|
|
a = fr->root;
|
|
|
|
while(a)
|
|
|
|
{
|
|
|
|
grib_action *na = a->next;
|
2015-02-11 11:33:13 +00:00
|
|
|
grib_action_delete(c, a);
|
2013-12-13 13:00:09 +00:00
|
|
|
a = na;
|
|
|
|
}
|
|
|
|
grib_context_free_persistent(c, fr->filename);
|
|
|
|
grib_context_free_persistent(c, fr);
|
|
|
|
}
|
|
|
|
grib_context_free_persistent(c, c->grib_reader);
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
c->grib_reader = NULL;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
if(c->codetable) grib_codetable_delete(c);
|
|
|
|
c->codetable = NULL;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2014-07-10 10:34:39 +00:00
|
|
|
if(c->smart_table) grib_smart_table_delete(c);
|
|
|
|
c->smart_table = NULL;
|
2014-06-20 17:18:57 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
if(c->grib_definition_files_dir)
|
|
|
|
grib_context_free(c,c->grib_definition_files_dir);
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
if(c->multi_support_on)
|
|
|
|
grib_multi_support_reset(c);
|
2013-03-25 12:04:10 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void grib_context_delete( grib_context* c)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
grib_hash_keys_delete( c->keys);
|
|
|
|
grib_trie_delete(c->def_files);
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
grib_context_reset( c );
|
|
|
|
if(c != &default_grib_context)
|
|
|
|
grib_context_free_persistent(&default_grib_context,c);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2019-08-09 12:06:58 +00:00
|
|
|
void codes_bufr_multi_element_constant_arrays_on(grib_context* c)
|
|
|
|
{
|
|
|
|
if ( !c ) c=grib_context_get_default();
|
|
|
|
c->bufr_multi_element_constant_arrays = 1;
|
|
|
|
}
|
|
|
|
void codes_bufr_multi_element_constant_arrays_off(grib_context* c)
|
|
|
|
{
|
|
|
|
if ( !c ) c=grib_context_get_default();
|
|
|
|
c->bufr_multi_element_constant_arrays = 0;
|
|
|
|
}
|
|
|
|
/*int codes_get_bufr_multi_element_constant_arrays(grib_context* c);*/
|
|
|
|
|
|
|
|
|
2016-06-01 17:01:47 +00:00
|
|
|
void grib_context_set_definitions_path(grib_context* c, const char* path)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
|
|
|
|
c->grib_definition_files_path = strdup(path);
|
|
|
|
grib_context_log(c, GRIB_LOG_DEBUG, "Definitions path changed to: %s", c->grib_definition_files_path);
|
|
|
|
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
}
|
|
|
|
void grib_context_set_samples_path(grib_context* c, const char* path)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
|
|
|
|
c->grib_samples_path = strdup(path);
|
|
|
|
grib_context_log(c, GRIB_LOG_DEBUG, "Samples path changed to: %s", c->grib_samples_path);
|
|
|
|
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void* grib_context_malloc_persistent(const grib_context* c, size_t size)
|
|
|
|
{
|
|
|
|
void* p = c->alloc_persistent_mem(c,size);
|
|
|
|
if(!p) {
|
2017-05-19 12:39:51 +00:00
|
|
|
grib_context_log(c,GRIB_LOG_FATAL,
|
|
|
|
"grib_context_malloc_persistent: error allocating %lu bytes",(unsigned long)size);
|
2017-02-02 18:26:58 +00:00
|
|
|
Assert(0);
|
2013-12-13 13:00:09 +00:00
|
|
|
}
|
|
|
|
return p;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
char* grib_context_strdup_persistent(const grib_context* c,const char* s)
|
|
|
|
{
|
|
|
|
char *dup = (char*)grib_context_malloc_persistent(c,(strlen(s)*sizeof(char))+1);
|
|
|
|
if(dup) strcpy(dup,s);
|
|
|
|
return dup;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void* grib_context_malloc_clear_persistent(const grib_context* c, size_t size)
|
|
|
|
{
|
|
|
|
void *p = grib_context_malloc_persistent(c,size);
|
|
|
|
if(p) memset(p,0,size);
|
|
|
|
return p;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void* grib_context_malloc(const grib_context* c, size_t size)
|
|
|
|
{
|
|
|
|
void* p = NULL;
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
if(size == 0) return p;
|
|
|
|
else p=c->alloc_mem(c,size);
|
|
|
|
if(!p) {
|
|
|
|
grib_context_log(c,GRIB_LOG_FATAL,"grib_context_malloc: error allocating %lu bytes",(unsigned long)size);
|
2017-02-02 18:26:58 +00:00
|
|
|
Assert(0);
|
2013-12-13 13:00:09 +00:00
|
|
|
}
|
|
|
|
return p;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void* grib_context_realloc(const grib_context* c, void *p,size_t size)
|
|
|
|
{
|
|
|
|
void* q;
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
q=c->realloc_mem(c,p,size);
|
|
|
|
if(!q) {
|
|
|
|
grib_context_log(c,GRIB_LOG_FATAL,"grib_context_realloc: error allocating %lu bytes",(unsigned long)size);
|
2016-06-22 17:41:15 +00:00
|
|
|
return NULL;
|
2013-12-13 13:00:09 +00:00
|
|
|
}
|
|
|
|
return q;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
char* grib_context_strdup(const grib_context* c,const char* s)
|
|
|
|
{
|
2014-11-04 13:59:20 +00:00
|
|
|
char *dup=0;
|
|
|
|
if (s) {
|
|
|
|
dup = (char*)grib_context_malloc(c,(strlen(s)*sizeof(char))+1);
|
|
|
|
if(dup) strcpy(dup,s);
|
|
|
|
}
|
2013-12-13 13:00:09 +00:00
|
|
|
return dup;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void* grib_context_malloc_clear(const grib_context* c, size_t size)
|
|
|
|
{
|
|
|
|
void *p = grib_context_malloc(c,size);
|
|
|
|
if(p) memset(p,0,size);
|
|
|
|
return p;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void* grib_context_buffer_malloc(const grib_context* c, size_t size)
|
|
|
|
{
|
|
|
|
void* p = NULL;
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
if(size == 0) return p;
|
|
|
|
else p=c->alloc_buffer_mem(c,size);
|
|
|
|
if(!p) {
|
|
|
|
grib_context_log(c,GRIB_LOG_FATAL,"grib_context_buffer_malloc: error allocating %lu bytes",(unsigned long)size);
|
2016-06-22 17:41:15 +00:00
|
|
|
return NULL;
|
2013-12-13 13:00:09 +00:00
|
|
|
}
|
|
|
|
return p;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void grib_context_buffer_free(const grib_context* c, void* p)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
if(p) c->free_buffer_mem(c,p);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void* grib_context_buffer_realloc(const grib_context* c, void *p,size_t size)
|
|
|
|
{
|
|
|
|
void* q=c->realloc_buffer_mem(c,p,size);
|
|
|
|
if(!q) {
|
|
|
|
grib_context_log(c,GRIB_LOG_FATAL,"grib_context_buffer_realloc: error allocating %lu bytes",(unsigned long)size);
|
2016-06-22 17:41:15 +00:00
|
|
|
return NULL;
|
2013-12-13 13:00:09 +00:00
|
|
|
}
|
|
|
|
return q;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 13:00:09 +00:00
|
|
|
void* grib_context_buffer_malloc_clear(const grib_context* c, size_t size)
|
|
|
|
{
|
|
|
|
void *p = grib_context_buffer_malloc(c,size);
|
|
|
|
if(p) memset(p,0,size);
|
|
|
|
return p;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void grib_context_set_memory_proc(grib_context* c, grib_malloc_proc m, grib_free_proc f,grib_realloc_proc r)
|
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
c->free_mem = f;
|
|
|
|
c->alloc_mem = m;
|
|
|
|
c->realloc_mem = r;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void grib_context_set_persistent_memory_proc(grib_context* c, grib_malloc_proc m, grib_free_proc f)
|
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
c->free_persistent_mem = f;
|
|
|
|
c->alloc_persistent_mem = m;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void grib_context_set_buffer_memory_proc(grib_context* c, grib_malloc_proc m, grib_free_proc f,grib_realloc_proc r)
|
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
c->free_buffer_mem = f;
|
|
|
|
c->alloc_buffer_mem = m;
|
|
|
|
c->realloc_buffer_mem = r;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void grib_context_set_data_accessing_proc(grib_context* c, grib_data_read_proc read, grib_data_write_proc write, grib_data_tell_proc tell)
|
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
c->read = read;
|
|
|
|
c->write = write;
|
|
|
|
c->tell = tell;
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2017-06-14 10:23:08 +00:00
|
|
|
/* Logging procedure */
|
2013-03-25 12:04:10 +00:00
|
|
|
void grib_context_log(const grib_context *c,int level, const char* fmt, ...)
|
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
/* Save some CPU */
|
|
|
|
if( (level == GRIB_LOG_DEBUG && c->debug<1) ||
|
|
|
|
(level == GRIB_LOG_WARNING && c->debug<2) )
|
2017-06-14 10:23:08 +00:00
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
return;
|
2017-06-14 10:23:08 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
char msg[1024];
|
|
|
|
va_list list;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2017-06-14 10:23:08 +00:00
|
|
|
va_start(list,fmt);
|
|
|
|
vsprintf(msg, fmt, list);
|
|
|
|
va_end(list);
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2017-06-14 10:23:08 +00:00
|
|
|
if(level & GRIB_LOG_PERROR)
|
|
|
|
{
|
|
|
|
level = level & ~GRIB_LOG_PERROR;
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2017-06-14 10:23:08 +00:00
|
|
|
/* #if HAS_STRERROR */
|
2013-03-25 12:04:10 +00:00
|
|
|
#if 1
|
2013-12-13 13:00:09 +00:00
|
|
|
strcat(msg," (");
|
2017-06-14 10:23:08 +00:00
|
|
|
strcat(msg,strerror(errno));
|
|
|
|
strcat(msg,")");
|
|
|
|
#else
|
|
|
|
if(errno > 0 && errno < sys_nerr)
|
|
|
|
{
|
|
|
|
strcat(msg," (");
|
|
|
|
strcat(msg,sys_errlist[errno]);
|
|
|
|
strcat(msg," )");
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
#endif
|
2017-06-14 10:23:08 +00:00
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
|
2017-06-14 10:23:08 +00:00
|
|
|
if(c->output_log)
|
|
|
|
c->output_log(c,level,msg);
|
|
|
|
}
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
|
|
|
|
2017-06-14 10:23:08 +00:00
|
|
|
/* Logging procedure */
|
2013-03-25 12:04:10 +00:00
|
|
|
void grib_context_print(const grib_context *c, void* descriptor,const char* fmt, ...)
|
|
|
|
{
|
2013-12-13 13:00:09 +00:00
|
|
|
char msg[1024];
|
|
|
|
va_list list;
|
|
|
|
va_start(list,fmt);
|
|
|
|
vsprintf(msg, fmt, list);
|
|
|
|
va_end(list);
|
|
|
|
c->print(c,descriptor,msg);
|
2013-03-25 12:04:10 +00:00
|
|
|
}
|
2016-04-14 11:44:53 +00:00
|
|
|
|
2018-05-30 17:13:06 +00:00
|
|
|
int grib_context_get_handle_file_count(grib_context *c)
|
|
|
|
{
|
|
|
|
int r = 0;
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
r = c->handle_file_count;
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
int grib_context_get_handle_total_count(grib_context *c)
|
|
|
|
{
|
|
|
|
int r = 0;
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
r=c->handle_total_count;
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2016-04-14 11:44:53 +00:00
|
|
|
void grib_context_set_handle_file_count(grib_context *c, int new_count)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
c->handle_file_count = new_count;
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
}
|
|
|
|
void grib_context_set_handle_total_count(grib_context *c, int new_count)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
c->handle_total_count = new_count;
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
}
|
|
|
|
|
|
|
|
void grib_context_increment_handle_file_count(grib_context *c)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
c->handle_file_count++;
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
}
|
|
|
|
void grib_context_increment_handle_total_count(grib_context *c)
|
|
|
|
{
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
c->handle_total_count++;
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
}
|
|
|
|
|
2017-11-20 11:48:41 +00:00
|
|
|
bufr_descriptors_array* grib_context_expanded_descriptors_list_get(grib_context* c,const char* key,long* u,size_t size)
|
|
|
|
{
|
|
|
|
bufr_descriptors_map_list* expandedUnexpandedMapList;
|
|
|
|
size_t i=0;
|
|
|
|
int found=0;
|
|
|
|
bufr_descriptors_array* result = NULL;
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
|
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
|
|
|
|
if (!c->expanded_descriptors) {
|
|
|
|
c->expanded_descriptors=(grib_trie*)grib_trie_new(c);
|
|
|
|
result = NULL;
|
|
|
|
goto the_end;
|
|
|
|
}
|
|
|
|
expandedUnexpandedMapList=(bufr_descriptors_map_list*)grib_trie_get(c->expanded_descriptors,key);
|
|
|
|
found=0;
|
|
|
|
while (expandedUnexpandedMapList) {
|
|
|
|
if (expandedUnexpandedMapList->unexpanded->n==size) {
|
|
|
|
found=1;
|
|
|
|
for (i=0;i<size;i++) {
|
|
|
|
if (expandedUnexpandedMapList->unexpanded->v[i]->code!=u[i]) {
|
|
|
|
found=0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (found) {
|
|
|
|
result = expandedUnexpandedMapList->expanded;
|
|
|
|
goto the_end;
|
|
|
|
}
|
|
|
|
expandedUnexpandedMapList=expandedUnexpandedMapList->next;
|
|
|
|
}
|
|
|
|
the_end:
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void grib_context_expanded_descriptors_list_push(grib_context* c,
|
|
|
|
const char* key,bufr_descriptors_array* expanded,bufr_descriptors_array* unexpanded)
|
|
|
|
{
|
|
|
|
bufr_descriptors_map_list* descriptorsList=NULL;
|
|
|
|
bufr_descriptors_map_list* next=NULL;
|
|
|
|
bufr_descriptors_map_list* newdescriptorsList=NULL;
|
|
|
|
if (!c) c=grib_context_get_default();
|
|
|
|
|
|
|
|
GRIB_MUTEX_INIT_ONCE(&once,&init);
|
|
|
|
GRIB_MUTEX_LOCK(&mutex_c);
|
|
|
|
|
|
|
|
newdescriptorsList=(bufr_descriptors_map_list*)grib_context_malloc_clear(c,sizeof(bufr_descriptors_map_list));
|
|
|
|
newdescriptorsList->expanded=expanded;
|
|
|
|
newdescriptorsList->unexpanded=unexpanded;
|
|
|
|
|
|
|
|
descriptorsList=(bufr_descriptors_map_list*)grib_trie_get(c->expanded_descriptors,key);
|
|
|
|
if (descriptorsList) {
|
|
|
|
next=descriptorsList;
|
|
|
|
while(next->next) {
|
|
|
|
next=next->next;
|
|
|
|
}
|
|
|
|
next->next=newdescriptorsList;
|
|
|
|
} else {
|
|
|
|
grib_trie_insert(c->expanded_descriptors,key,newdescriptorsList);
|
|
|
|
}
|
|
|
|
GRIB_MUTEX_UNLOCK(&mutex_c);
|
|
|
|
}
|
|
|
|
|
2017-03-15 19:04:21 +00:00
|
|
|
static codes_assertion_failed_proc assertion = NULL;
|
|
|
|
|
2017-03-30 15:19:50 +00:00
|
|
|
void codes_set_codes_assertion_failed_proc(codes_assertion_failed_proc proc)
|
|
|
|
{
|
2017-03-15 19:04:21 +00:00
|
|
|
assertion = proc;
|
|
|
|
}
|
|
|
|
|
2017-03-30 15:19:50 +00:00
|
|
|
void codes_assertion_failed(const char* message, const char* file, int line)
|
|
|
|
{
|
|
|
|
/* Default behaviour is to abort
|
|
|
|
* unless user has supplied his own assertion routine */
|
|
|
|
if (assertion == NULL) {
|
2017-10-09 15:18:50 +00:00
|
|
|
grib_context *c = grib_context_get_default();
|
2017-03-30 15:19:50 +00:00
|
|
|
fprintf(stderr, "ecCodes assertion failed: `%s' in %s:%d\n", message, file, line);
|
2017-10-09 15:18:50 +00:00
|
|
|
if (!c->no_abort) {
|
|
|
|
abort();
|
|
|
|
}
|
2017-03-15 19:04:21 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
char buffer[10240];
|
2017-03-30 15:19:50 +00:00
|
|
|
sprintf(buffer, "ecCodes assertion failed: `%s' in %s:%d", message, file, line);
|
2017-03-15 19:04:21 +00:00
|
|
|
assertion(buffer);
|
|
|
|
}
|
|
|
|
}
|
2019-08-09 16:08:59 +00:00
|
|
|
|
|
|
|
int grib_get_gribex_mode(grib_context* c)
|
|
|
|
{
|
|
|
|
if ( !c ) c=grib_context_get_default();
|
|
|
|
return c->gribex_mode_on;
|
|
|
|
}
|
|
|
|
void grib_gribex_mode_on(grib_context* c)
|
|
|
|
{
|
|
|
|
if ( !c ) c=grib_context_get_default();
|
|
|
|
c->gribex_mode_on=1;
|
|
|
|
}
|
|
|
|
void grib_gribex_mode_off(grib_context* c)
|
|
|
|
{
|
|
|
|
if ( !c ) c=grib_context_get_default();
|
|
|
|
c->gribex_mode_on=0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void grib_gts_header_on(grib_context* c)
|
|
|
|
{
|
|
|
|
if ( !c ) c=grib_context_get_default();
|
|
|
|
c->gts_header_on=1;
|
|
|
|
}
|
|
|
|
void grib_gts_header_off(grib_context* c)
|
|
|
|
{
|
|
|
|
if ( !c ) c=grib_context_get_default();
|
|
|
|
c->gts_header_on=0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void grib_multi_support_on(grib_context* c)
|
|
|
|
{
|
|
|
|
if ( !c ) c=grib_context_get_default();
|
|
|
|
c->multi_support_on=1;
|
|
|
|
}
|
|
|
|
void grib_multi_support_off(grib_context* c)
|
|
|
|
{
|
|
|
|
if ( !c ) c=grib_context_get_default();
|
|
|
|
c->multi_support_on=0;
|
|
|
|
}
|